Merge branch 'rainwashed-main'
This commit is contained in:
commit
0e27a8b54c
10 changed files with 265 additions and 39 deletions
|
@ -79,6 +79,12 @@ Once the server is up and running, you can interact with it via the frontend:
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Adding Proxies and User-Agents
|
||||||
|
|
||||||
|
Access to the ``data/proxies.txt`` and ``data/uas.txt`` can now be done fully in the frontend. Click the text button to the right of the beam button to open up the editor.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
## Worker-Based Attack Handling 🔧💡
|
## Worker-Based Attack Handling 🔧💡
|
||||||
|
|
||||||
Each attack type is handled in a separate worker thread, ensuring that the main server remains responsive. The attack workers are dynamically loaded based on the selected attack method (HTTP, etc...).
|
Each attack type is handled in a separate worker thread, ensuring that the main server remains responsive. The attack workers are dynamically loaded based on the selected attack method (HTTP, etc...).
|
||||||
|
|
BIN
bun.lockb
BIN
bun.lockb
Binary file not shown.
|
@ -1,5 +1,5 @@
|
||||||
socks4://199.58.184.97:4145
|
socks4://199.58.184.97:4145
|
||||||
socks4://192.111.139.163:19404
|
socks4://192.111.139.163:19404
|
||||||
socks4://199.187.210.54:4145
|
socks4://199.187.210.54:4145
|
||||||
socks4://46.105.127.74:45108
|
socks4://46.105.127.74:45108
|
||||||
socks4://98.188.47.150:4145
|
socks4://98.188.47.150:4145
|
BIN
docs/annotated-button.png
Normal file
BIN
docs/annotated-button.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 246 KiB |
Binary file not shown.
Before Width: | Height: | Size: 284 KiB After Width: | Height: | Size: 370 KiB |
|
@ -13,6 +13,7 @@
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"axios": "^1.7.9",
|
"axios": "^1.7.9",
|
||||||
|
"body-parser": "^1.20.3",
|
||||||
"concurrently": "^9.1.2",
|
"concurrently": "^9.1.2",
|
||||||
"express": "^4.21.2",
|
"express": "^4.21.2",
|
||||||
"lucide-react": "^0.344.0",
|
"lucide-react": "^0.344.0",
|
||||||
|
|
BIN
public/loading.gif
Normal file
BIN
public/loading.gif
Normal file
Binary file not shown.
After Width: | Height: | Size: 334 KiB |
|
@ -3,7 +3,7 @@ import { join } from "path";
|
||||||
|
|
||||||
import { Proxy } from "./lib";
|
import { Proxy } from "./lib";
|
||||||
|
|
||||||
const currentPath = () => {
|
export const currentPath = () => {
|
||||||
const path = process.cwd();
|
const path = process.cwd();
|
||||||
return path === "/" ? "." : path;
|
return path === "/" ? "." : path;
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,13 +1,15 @@
|
||||||
import express from "express";
|
import express from "express";
|
||||||
import { createServer } from "http";
|
import { createServer } from "http";
|
||||||
import { dirname, join } from "path";
|
import { dirname, join } from "path";
|
||||||
|
import { readFileSync, writeFileSync } from "fs";
|
||||||
import { Server } from "socket.io";
|
import { Server } from "socket.io";
|
||||||
import { fileURLToPath } from "url";
|
import { fileURLToPath } from "url";
|
||||||
import { Worker } from "worker_threads";
|
import { Worker } from "worker_threads";
|
||||||
|
|
||||||
import { loadProxies, loadUserAgents } from "./fileLoader";
|
import { currentPath, loadProxies, loadUserAgents } from "./fileLoader";
|
||||||
import { AttackMethod } from "./lib";
|
import { AttackMethod } from "./lib";
|
||||||
import { filterProxies } from "./proxyUtils";
|
import { filterProxies } from "./proxyUtils";
|
||||||
|
import bodyParser from "body-parser";
|
||||||
|
|
||||||
// Define the workers based on attack type
|
// Define the workers based on attack type
|
||||||
const attackWorkers: { [key in AttackMethod]: string } = {
|
const attackWorkers: { [key in AttackMethod]: string } = {
|
||||||
|
@ -26,6 +28,7 @@ const io = new Server(httpServer, {
|
||||||
cors: {
|
cors: {
|
||||||
origin: "http://localhost:5173",
|
origin: "http://localhost:5173",
|
||||||
methods: ["GET", "POST"],
|
methods: ["GET", "POST"],
|
||||||
|
allowedHeaders: ["Content-Type"]
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -105,6 +108,47 @@ io.on("connection", (socket) => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
app.get("/configuration", (req, res) => {
|
||||||
|
res.setHeader("Access-Control-Allow-Origin", "http://localhost:5173")
|
||||||
|
res.setHeader("Content-Type", "application/json");
|
||||||
|
let proxiesText = readFileSync(join(currentPath(), "data", "proxies.txt"), "utf-8");
|
||||||
|
let uasText = readFileSync(join(currentPath(), "data", "uas.txt"), "utf-8");
|
||||||
|
|
||||||
|
res.send({
|
||||||
|
proxies: btoa(proxiesText),
|
||||||
|
uas: btoa(uasText),
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
app.options('/configuration', (req, res) => {
|
||||||
|
res.setHeader('Access-Control-Allow-Origin', 'http://localhost:5173');
|
||||||
|
res.setHeader('Access-Control-Allow-Methods', 'POST, OPTIONS');
|
||||||
|
res.setHeader('Access-Control-Allow-Headers', 'Content-Type');
|
||||||
|
res.send();
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
app.post("/configuration", bodyParser.json(), (req, res) => {
|
||||||
|
res.setHeader("Access-Control-Allow-Methods", "POST");
|
||||||
|
res.setHeader("Access-Control-Allow-Headers", "Content-Type")
|
||||||
|
res.setHeader("Access-Control-Allow-Origin", "http://localhost:5173")
|
||||||
|
res.setHeader("Content-Type", "application/text");
|
||||||
|
|
||||||
|
// console.log(req.body)
|
||||||
|
|
||||||
|
// atob and btoa are used to avoid the problems in sending data with // characters, etc.
|
||||||
|
let proxies = atob(req.body["proxies"]);
|
||||||
|
let uas = atob(req.body["uas"]);
|
||||||
|
writeFileSync(join(currentPath(), "data", "proxies.txt"), proxies, {
|
||||||
|
encoding: "utf-8"
|
||||||
|
});
|
||||||
|
writeFileSync(join(currentPath(), "data", "uas.txt"), uas, {
|
||||||
|
encoding: "utf-8"
|
||||||
|
});
|
||||||
|
|
||||||
|
res.send("OK")
|
||||||
|
})
|
||||||
|
|
||||||
const PORT = 3000;
|
const PORT = 3000;
|
||||||
httpServer.listen(PORT, () => {
|
httpServer.listen(PORT, () => {
|
||||||
console.log(`Server running on port ${PORT}`);
|
console.log(`Server running on port ${PORT}`);
|
||||||
|
|
241
src/App.tsx
241
src/App.tsx
|
@ -1,9 +1,97 @@
|
||||||
import { Bot, Wand2, Wifi, Zap } from "lucide-react";
|
import { Bot, ScrollText, Wand2, Wifi, Zap } from "lucide-react";
|
||||||
import { useEffect, useRef, useState } from "react";
|
import { useEffect, useRef, useState } from "react";
|
||||||
import { io } from "socket.io-client";
|
import { io } from "socket.io-client";
|
||||||
|
|
||||||
const socket = io("http://localhost:3000");
|
const socket = io("http://localhost:3000");
|
||||||
|
|
||||||
|
function ConfigureProxiesAndAgentsView() {
|
||||||
|
const [loadingConfiguration, setLoadingConfiguration] = useState(false);
|
||||||
|
const [configuration, setConfiguration] = useState<string[]>([]);
|
||||||
|
|
||||||
|
async function retrieveConfiguration(): Promise<string[]> {
|
||||||
|
const response = await fetch(`http://localhost:3000/configuration`);
|
||||||
|
const information = (await response.json()) as {
|
||||||
|
proxies: string;
|
||||||
|
uas: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const proxies = atob(information.proxies);
|
||||||
|
const uas = atob(information.uas);
|
||||||
|
|
||||||
|
return [proxies, uas];
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!loadingConfiguration) {
|
||||||
|
setLoadingConfiguration(true);
|
||||||
|
retrieveConfiguration().then((config) => {
|
||||||
|
setLoadingConfiguration(false);
|
||||||
|
setConfiguration(config);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
function saveConfiguration() {
|
||||||
|
const obj = {
|
||||||
|
proxies: btoa(configuration[0]),
|
||||||
|
uas: btoa(configuration[1]),
|
||||||
|
};
|
||||||
|
|
||||||
|
// console.log(obj)
|
||||||
|
|
||||||
|
const response = fetch(`http://localhost:3000/configuration`, {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify(obj),
|
||||||
|
});
|
||||||
|
|
||||||
|
response.then(() => {
|
||||||
|
alert("Saved");
|
||||||
|
window.location.reload();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="fixed grid p-8 mx-auto -translate-x-1/2 -translate-y-1/2 bg-white rounded-md shadow-lg max-w-7xl place-items-center left-1/2 top-1/2">
|
||||||
|
{loadingConfiguration ? (
|
||||||
|
<div className="flex flex-col items-center justify-center space-y-2">
|
||||||
|
<img src="/loading.gif" className="rounded-sm shadow-sm" />
|
||||||
|
<p>Loading proxies.txt and uas.txt...</p>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="w-[56rem] flex flex-col">
|
||||||
|
<p className="pl-1 mb-1 italic">proxies.txt</p>
|
||||||
|
<textarea
|
||||||
|
value={configuration[0]}
|
||||||
|
className="w-full h-40 p-2 border-black/10 border-[1px] rounded-sm resize-none"
|
||||||
|
onChange={(e) =>
|
||||||
|
setConfiguration([e.target.value, configuration[1]])
|
||||||
|
}
|
||||||
|
placeholder="socks5://0.0.0.0"
|
||||||
|
></textarea>
|
||||||
|
<p className="pl-1 mt-2 mb-1 italic">uas.txt</p>
|
||||||
|
<textarea
|
||||||
|
value={configuration[1]}
|
||||||
|
className="w-full h-40 p-2 border-black/10 border-[1px] rounded-sm resize-none"
|
||||||
|
onChange={(e) =>
|
||||||
|
setConfiguration([configuration[0], e.target.value])
|
||||||
|
}
|
||||||
|
placeholder="Mozilla/5.0 (Linux; Android 10; K)..."
|
||||||
|
></textarea>
|
||||||
|
<button
|
||||||
|
onClick={saveConfiguration}
|
||||||
|
className="p-4 mt-4 text-white bg-gray-800 rounded-md hover:bg-gray-900"
|
||||||
|
>
|
||||||
|
Write Changes
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
const [isAttacking, setIsAttacking] = useState(false);
|
const [isAttacking, setIsAttacking] = useState(false);
|
||||||
const [actuallyAttacking, setActuallyAttacking] = useState(false);
|
const [actuallyAttacking, setActuallyAttacking] = useState(false);
|
||||||
|
@ -24,6 +112,8 @@ function App() {
|
||||||
const [lastTotalPackets, setLastTotalPackets] = useState(0);
|
const [lastTotalPackets, setLastTotalPackets] = useState(0);
|
||||||
const [currentTask, setCurrentTask] = useState<NodeJS.Timeout | null>(null);
|
const [currentTask, setCurrentTask] = useState<NodeJS.Timeout | null>(null);
|
||||||
const [audioVol, setAudioVol] = useState(100);
|
const [audioVol, setAudioVol] = useState(100);
|
||||||
|
const [openedConfig, setOpenedConfig] = useState(false);
|
||||||
|
|
||||||
const audioRef = useRef<HTMLAudioElement>(null);
|
const audioRef = useRef<HTMLAudioElement>(null);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -32,7 +122,11 @@ function App() {
|
||||||
const handler = () => {
|
const handler = () => {
|
||||||
if (audio.paused) return;
|
if (audio.paused) return;
|
||||||
|
|
||||||
if (animState !== 2 && audio.currentTime > 5.24 && audio.currentTime < 9.4) {
|
if (
|
||||||
|
animState !== 2 &&
|
||||||
|
audio.currentTime > 5.24 &&
|
||||||
|
audio.currentTime < 9.4
|
||||||
|
) {
|
||||||
setAnimState(2);
|
setAnimState(2);
|
||||||
}
|
}
|
||||||
if (audio.currentTime > 17.53) {
|
if (audio.currentTime > 17.53) {
|
||||||
|
@ -98,13 +192,11 @@ function App() {
|
||||||
};
|
};
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (audioRef.current) {
|
if (audioRef.current) {
|
||||||
audioRef.current.volume = audioVol / 100;
|
audioRef.current.volume = audioVol / 100;
|
||||||
}
|
}
|
||||||
}, [audioVol])
|
}, [audioVol]);
|
||||||
|
|
||||||
|
|
||||||
const addLog = (message: string) => {
|
const addLog = (message: string) => {
|
||||||
setLogs((prev) => [message, ...prev].slice(0, 12));
|
setLogs((prev) => [message, ...prev].slice(0, 12));
|
||||||
|
@ -158,9 +250,13 @@ function App() {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={`w-screen h-screen bg-gradient-to-br ${animState === 0 || animState === 3 ? "from-pink-100 to-blue-100" : animState === 2 ? "background-pulse" : "bg-gray-950"} p-8 overflow-y-auto ${
|
className={`w-screen h-screen bg-gradient-to-br ${
|
||||||
actuallyAttacking ? "shake" : ""
|
animState === 0 || animState === 3
|
||||||
}`}
|
? "from-pink-100 to-blue-100"
|
||||||
|
: animState === 2
|
||||||
|
? "background-pulse"
|
||||||
|
: "bg-gray-950"
|
||||||
|
} p-8 overflow-y-auto ${actuallyAttacking ? "shake" : ""}`}
|
||||||
>
|
>
|
||||||
<audio ref={audioRef} src="/audio.mp3" />
|
<audio ref={audioRef} src="/audio.mp3" />
|
||||||
|
|
||||||
|
@ -169,13 +265,23 @@ function App() {
|
||||||
<h1 className="mb-2 text-4xl font-bold text-pink-500">
|
<h1 className="mb-2 text-4xl font-bold text-pink-500">
|
||||||
Miku Miku Beam
|
Miku Miku Beam
|
||||||
</h1>
|
</h1>
|
||||||
<p className={`${animState === 0 || animState === 3 ? "text-gray-600" : "text-white"}`}>
|
<p
|
||||||
|
className={`${
|
||||||
|
animState === 0 || animState === 3
|
||||||
|
? "text-gray-600"
|
||||||
|
: "text-white"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
Because DDoS attacks are also cute and even more so when Miku does
|
Because DDoS attacks are also cute and even more so when Miku does
|
||||||
them.
|
them.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className={`relative p-6 overflow-hidden rounded-lg shadow-xl ${animState === 0 || animState === 3 ? "bg-white" : "bg-gray-950"}`}>
|
<div
|
||||||
|
className={`relative p-6 overflow-hidden rounded-lg shadow-xl ${
|
||||||
|
animState === 0 || animState === 3 ? "bg-white" : "bg-gray-950"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
{/* Miku GIF */}
|
{/* Miku GIF */}
|
||||||
<div
|
<div
|
||||||
className="flex justify-center w-full h-48 mb-6"
|
className="flex justify-center w-full h-48 mb-6"
|
||||||
|
@ -197,7 +303,9 @@ function App() {
|
||||||
value={target}
|
value={target}
|
||||||
onChange={(e) => setTarget(e.target.value)}
|
onChange={(e) => setTarget(e.target.value)}
|
||||||
placeholder="Enter target URL or IP"
|
placeholder="Enter target URL or IP"
|
||||||
className={`${animState === 0 || animState === 3 ? "" : "text-white"} px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
className={`${
|
||||||
|
animState === 0 || animState === 3 ? "" : "text-white"
|
||||||
|
} px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
||||||
disabled={isAttacking}
|
disabled={isAttacking}
|
||||||
/>
|
/>
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
|
@ -232,18 +340,32 @@ function App() {
|
||||||
>
|
>
|
||||||
<Zap className="w-5 h-5" />
|
<Zap className="w-5 h-5" />
|
||||||
</button>
|
</button>
|
||||||
|
<button
|
||||||
|
className={`px-2 py-2 rounded-lg font-semibold text-white transition-all flex items-center justify-center gap-2 bg-slate-800 hover:bg-slate-900`}
|
||||||
|
onClick={() => setOpenedConfig(true)}
|
||||||
|
>
|
||||||
|
<ScrollText className="w-5 h-5" />
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="grid grid-cols-4 gap-4">
|
<div className="grid grid-cols-4 gap-4">
|
||||||
<div>
|
<div>
|
||||||
<label className={`block mb-1 text-sm font-medium ${animState === 0 || animState === 3 ? "text-gray-700" : "text-white"}`}>
|
<label
|
||||||
|
className={`block mb-1 text-sm font-medium ${
|
||||||
|
animState === 0 || animState === 3
|
||||||
|
? "text-gray-700"
|
||||||
|
: "text-white"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
Attack Method
|
Attack Method
|
||||||
</label>
|
</label>
|
||||||
<select
|
<select
|
||||||
value={attackMethod}
|
value={attackMethod}
|
||||||
onChange={(e) => setAttackMethod(e.target.value)}
|
onChange={(e) => setAttackMethod(e.target.value)}
|
||||||
className={`${animState === 0 || animState === 3 ? "" : "text-gray-900"} w-full px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
className={`${
|
||||||
|
animState === 0 || animState === 3 ? "" : "text-gray-900"
|
||||||
|
} w-full px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
||||||
disabled={isAttacking}
|
disabled={isAttacking}
|
||||||
>
|
>
|
||||||
<option value="http_flood">HTTP/Flood</option>
|
<option value="http_flood">HTTP/Flood</option>
|
||||||
|
@ -253,42 +375,66 @@ function App() {
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<label className={`block mb-1 text-sm font-medium ${animState === 0 || animState === 3 ? "text-gray-700" : "text-white"}`}>
|
<label
|
||||||
|
className={`block mb-1 text-sm font-medium ${
|
||||||
|
animState === 0 || animState === 3
|
||||||
|
? "text-gray-700"
|
||||||
|
: "text-white"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
Packet Size (kb)
|
Packet Size (kb)
|
||||||
</label>
|
</label>
|
||||||
<input
|
<input
|
||||||
type="number"
|
type="number"
|
||||||
value={packetSize}
|
value={packetSize}
|
||||||
onChange={(e) => setPacketSize(Number(e.target.value))}
|
onChange={(e) => setPacketSize(Number(e.target.value))}
|
||||||
className={`${animState === 0 || animState === 3 ? "" : "text-white"} w-full px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
className={`${
|
||||||
|
animState === 0 || animState === 3 ? "" : "text-white"
|
||||||
|
} w-full px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
||||||
disabled={isAttacking}
|
disabled={isAttacking}
|
||||||
min="1"
|
min="1"
|
||||||
max="1500"
|
max="1500"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<label className={`block mb-1 text-sm font-medium ${animState === 0 || animState === 3 ? "text-gray-700" : "text-white"}`}>
|
<label
|
||||||
|
className={`block mb-1 text-sm font-medium ${
|
||||||
|
animState === 0 || animState === 3
|
||||||
|
? "text-gray-700"
|
||||||
|
: "text-white"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
Duration (seconds)
|
Duration (seconds)
|
||||||
</label>
|
</label>
|
||||||
<input
|
<input
|
||||||
type="number"
|
type="number"
|
||||||
value={duration}
|
value={duration}
|
||||||
onChange={(e) => setDuration(Number(e.target.value))}
|
onChange={(e) => setDuration(Number(e.target.value))}
|
||||||
className={`${animState === 0 || animState === 3 ? "" : "text-white"} w-full px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
className={`${
|
||||||
|
animState === 0 || animState === 3 ? "" : "text-white"
|
||||||
|
} w-full px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
||||||
disabled={isAttacking}
|
disabled={isAttacking}
|
||||||
min="1"
|
min="1"
|
||||||
max="300"
|
max="300"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<label className={`block mb-1 text-sm font-medium ${animState === 0 || animState === 3 ? "text-gray-700" : "text-white"}`}>
|
<label
|
||||||
|
className={`block mb-1 text-sm font-medium ${
|
||||||
|
animState === 0 || animState === 3
|
||||||
|
? "text-gray-700"
|
||||||
|
: "text-white"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
Packet Delay (ms)
|
Packet Delay (ms)
|
||||||
</label>
|
</label>
|
||||||
<input
|
<input
|
||||||
type="number"
|
type="number"
|
||||||
value={packetDelay}
|
value={packetDelay}
|
||||||
onChange={(e) => setPacketDelay(Number(e.target.value))}
|
onChange={(e) => setPacketDelay(Number(e.target.value))}
|
||||||
className={`${animState === 0 || animState === 3 ? "" : "text-white"} w-full px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
className={`${
|
||||||
|
animState === 0 || animState === 3 ? "" : "text-white"
|
||||||
|
} w-full px-4 py-2 border border-pink-200 rounded-lg outline-none focus:border-pink-500 focus:ring-2 focus:ring-pink-200`}
|
||||||
disabled={isAttacking}
|
disabled={isAttacking}
|
||||||
min="1"
|
min="1"
|
||||||
max="1000"
|
max="1000"
|
||||||
|
@ -304,7 +450,13 @@ function App() {
|
||||||
<Zap className="w-4 h-4" />
|
<Zap className="w-4 h-4" />
|
||||||
<span className="font-semibold">Packets/sec</span>
|
<span className="font-semibold">Packets/sec</span>
|
||||||
</div>
|
</div>
|
||||||
<div className={`text-2xl font-bold ${animState === 0 || animState === 3 ? "text-gray-800" : "text-white"}`}>
|
<div
|
||||||
|
className={`text-2xl font-bold ${
|
||||||
|
animState === 0 || animState === 3
|
||||||
|
? "text-gray-800"
|
||||||
|
: "text-white"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
{stats.pps.toLocaleString()}
|
{stats.pps.toLocaleString()}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -313,7 +465,13 @@ function App() {
|
||||||
<Bot className="w-4 h-4" />
|
<Bot className="w-4 h-4" />
|
||||||
<span className="font-semibold">Active Bots</span>
|
<span className="font-semibold">Active Bots</span>
|
||||||
</div>
|
</div>
|
||||||
<div className={`text-2xl font-bold ${animState === 0 || animState === 3 ? "text-gray-800" : "text-white"}`}>
|
<div
|
||||||
|
className={`text-2xl font-bold ${
|
||||||
|
animState === 0 || animState === 3
|
||||||
|
? "text-gray-800"
|
||||||
|
: "text-white"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
{stats.bots.toLocaleString()}
|
{stats.bots.toLocaleString()}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -322,7 +480,13 @@ function App() {
|
||||||
<Wifi className="w-4 h-4" />
|
<Wifi className="w-4 h-4" />
|
||||||
<span className="font-semibold">Total Packets</span>
|
<span className="font-semibold">Total Packets</span>
|
||||||
</div>
|
</div>
|
||||||
<div className={`text-2xl font-bold ${animState === 0 || animState === 3 ? "text-gray-800" : "text-white"}`}>
|
<div
|
||||||
|
className={`text-2xl font-bold ${
|
||||||
|
animState === 0 || animState === 3
|
||||||
|
? "text-gray-800"
|
||||||
|
: "text-white"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
{stats.totalPackets.toLocaleString()}
|
{stats.totalPackets.toLocaleString()}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -363,20 +527,31 @@ function App() {
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{openedConfig ? <ConfigureProxiesAndAgentsView /> : undefined}
|
||||||
|
|
||||||
<div className="flex flex-col items-center">
|
<div className="flex flex-col items-center">
|
||||||
<span className="text-sm text-center text-gray-500">
|
<span className="text-sm text-center text-gray-500">
|
||||||
🎵 v1.0 made by{" "}
|
🎵 v1.0 made by{" "}
|
||||||
<a
|
<a
|
||||||
href="https://github.com/sammwyy/mikumikubeam"
|
href="https://github.com/sammwyy/mikumikubeam"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noreferrer"
|
rel="noreferrer"
|
||||||
>
|
>
|
||||||
@Sammwy
|
@Sammwy
|
||||||
</a>{" "}
|
</a>{" "}
|
||||||
🎵
|
🎵
|
||||||
</span>
|
</span>
|
||||||
<span>
|
<span>
|
||||||
<input className="shadow-sm volume_bar focus:border-pink-500" type="range" min="0" max="100" step="5" draggable="false" value={audioVol} onChange={(e) => setAudioVol(parseInt(e.target?.value))} />
|
<input
|
||||||
|
className="shadow-sm volume_bar focus:border-pink-500"
|
||||||
|
type="range"
|
||||||
|
min="0"
|
||||||
|
max="100"
|
||||||
|
step="5"
|
||||||
|
draggable="false"
|
||||||
|
value={audioVol}
|
||||||
|
onChange={(e) => setAudioVol(parseInt(e.target?.value))}
|
||||||
|
/>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Add table
Reference in a new issue