diff --git a/README.md b/README.md index 68127f9..fc5be56 100644 --- a/README.md +++ b/README.md @@ -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. + +![AnnotatedImage](docs/annotated-button.png) + ## 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...). diff --git a/bun.lockb b/bun.lockb index 1faf35e..228d6ec 100644 Binary files a/bun.lockb and b/bun.lockb differ diff --git a/data/proxies.txt b/data/proxies.txt index 94e52ca..603bef3 100644 --- a/data/proxies.txt +++ b/data/proxies.txt @@ -1,5 +1,5 @@ -socks4://199.58.184.97:4145 -socks4://192.111.139.163:19404 -socks4://199.187.210.54:4145 -socks4://46.105.127.74:45108 +socks4://199.58.184.97:4145 +socks4://192.111.139.163:19404 +socks4://199.187.210.54:4145 +socks4://46.105.127.74:45108 socks4://98.188.47.150:4145 \ No newline at end of file diff --git a/docs/annotated-button.png b/docs/annotated-button.png new file mode 100644 index 0000000..5abb16a Binary files /dev/null and b/docs/annotated-button.png differ diff --git a/docs/screenshot.png b/docs/screenshot.png index 70dd61b..787f8ce 100644 Binary files a/docs/screenshot.png and b/docs/screenshot.png differ diff --git a/package.json b/package.json index 448129d..87d6237 100644 --- a/package.json +++ b/package.json @@ -13,6 +13,7 @@ }, "dependencies": { "axios": "^1.7.9", + "body-parser": "^1.20.3", "concurrently": "^9.1.2", "express": "^4.21.2", "lucide-react": "^0.344.0", diff --git a/public/loading.gif b/public/loading.gif new file mode 100644 index 0000000..5ddd893 Binary files /dev/null and b/public/loading.gif differ diff --git a/server/fileLoader.ts b/server/fileLoader.ts index 1a039cd..c232d95 100644 --- a/server/fileLoader.ts +++ b/server/fileLoader.ts @@ -3,7 +3,7 @@ import { join } from "path"; import { Proxy } from "./lib"; -const currentPath = () => { +export const currentPath = () => { const path = process.cwd(); return path === "/" ? "." : path; }; diff --git a/server/index.ts b/server/index.ts index 5c87996..c8691b4 100644 --- a/server/index.ts +++ b/server/index.ts @@ -1,13 +1,15 @@ import express from "express"; import { createServer } from "http"; import { dirname, join } from "path"; +import { readFileSync, writeFileSync } from "fs"; import { Server } from "socket.io"; import { fileURLToPath } from "url"; import { Worker } from "worker_threads"; -import { loadProxies, loadUserAgents } from "./fileLoader"; +import { currentPath, loadProxies, loadUserAgents } from "./fileLoader"; import { AttackMethod } from "./lib"; import { filterProxies } from "./proxyUtils"; +import bodyParser from "body-parser"; // Define the workers based on attack type const attackWorkers: { [key in AttackMethod]: string } = { @@ -26,6 +28,7 @@ const io = new Server(httpServer, { cors: { origin: "http://localhost:5173", 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; httpServer.listen(PORT, () => { console.log(`Server running on port ${PORT}`); diff --git a/src/App.tsx b/src/App.tsx index f4bfb58..ac6134d 100644 --- a/src/App.tsx +++ b/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 { io } from "socket.io-client"; const socket = io("http://localhost:3000"); +function ConfigureProxiesAndAgentsView() { + const [loadingConfiguration, setLoadingConfiguration] = useState(false); + const [configuration, setConfiguration] = useState([]); + + async function retrieveConfiguration(): Promise { + 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 ( +
+ {loadingConfiguration ? ( +
+ +

Loading proxies.txt and uas.txt...

+
+ ) : ( +
+

proxies.txt

+ +

uas.txt

+ + +
+ )} +
+ ); +} + function App() { const [isAttacking, setIsAttacking] = useState(false); const [actuallyAttacking, setActuallyAttacking] = useState(false); @@ -24,6 +112,8 @@ function App() { const [lastTotalPackets, setLastTotalPackets] = useState(0); const [currentTask, setCurrentTask] = useState(null); const [audioVol, setAudioVol] = useState(100); + const [openedConfig, setOpenedConfig] = useState(false); + const audioRef = useRef(null); useEffect(() => { @@ -32,7 +122,11 @@ function App() { const handler = () => { 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); } if (audio.currentTime > 17.53) { @@ -98,13 +192,11 @@ function App() { }; }, []); - useEffect(() => { if (audioRef.current) { audioRef.current.volume = audioVol / 100; } - }, [audioVol]) - + }, [audioVol]); const addLog = (message: string) => { setLogs((prev) => [message, ...prev].slice(0, 12)); @@ -158,9 +250,13 @@ function App() { return (
-
+
{/* Miku GIF */}
setTarget(e.target.value)} 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} />
@@ -232,18 +340,32 @@ function App() { > +
-
-
-
-
-
+
{stats.pps.toLocaleString()}
@@ -313,7 +465,13 @@ function App() { Active Bots
-
+
{stats.bots.toLocaleString()}
@@ -322,7 +480,13 @@ function App() { Total Packets
-
+
{stats.totalPackets.toLocaleString()}
@@ -363,20 +527,31 @@ function App() { )}
+ {openedConfig ? : undefined} +
- - 🎵 v1.0 made by{" "} - - @Sammwy - {" "} - 🎵 + + 🎵 v1.0 made by{" "} + + @Sammwy + {" "} + 🎵 - setAudioVol(parseInt(e.target?.value))} /> + setAudioVol(parseInt(e.target?.value))} + />