initial commit
This commit is contained in:
9
proxy/Dockerfile
Normal file
9
proxy/Dockerfile
Normal file
@@ -0,0 +1,9 @@
|
||||
FROM python:3.12-slim
|
||||
|
||||
RUN pip install --no-cache-dir setuptools selenium websockets
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
COPY main.py .
|
||||
|
||||
CMD ["python", "main.py"]
|
||||
182
proxy/main.py
Normal file
182
proxy/main.py
Normal file
@@ -0,0 +1,182 @@
|
||||
import os
|
||||
import time
|
||||
import asyncio
|
||||
import websockets
|
||||
from datetime import datetime
|
||||
from selenium import webdriver
|
||||
from selenium.webdriver.chrome.options import Options
|
||||
from selenium.webdriver.chrome.service import Service
|
||||
from selenium.webdriver.common.by import By
|
||||
from selenium.webdriver.support.ui import WebDriverWait
|
||||
from selenium.webdriver.support import expected_conditions as EC
|
||||
|
||||
TARGET_URL = os.getenv("TARGET_URL", "https://omegleweb.io/")
|
||||
TARGET_WS_URL = os.getenv("TARGET_WS_URL", "wss://omegleweb.io:8443/socket.io/?EIO=4&transport=websocket")
|
||||
LOCAL_HOST = os.getenv("LOCAL_HOST", "0.0.0.0")
|
||||
LOCAL_PORT = int(os.getenv("LOCAL_PORT", "8765"))
|
||||
HEADLESS = os.getenv("HEADLESS", "false").lower() in ("true", "1", "yes")
|
||||
CF_WAIT_TIME = int(os.getenv("CF_WAIT_TIME", "60"))
|
||||
SELENIUM_URL = os.getenv("SELENIUM_URL", "http://localhost:4444")
|
||||
|
||||
credentials = {
|
||||
"user_agent": None,
|
||||
"cookies": None
|
||||
}
|
||||
|
||||
def stealth_js(driver):
|
||||
driver.execute_script("""
|
||||
Object.defineProperty(navigator, 'webdriver', {get: () => undefined});
|
||||
Object.defineProperty(navigator, 'plugins', {get: () => [1, 2, 3, 4, 5]});
|
||||
Object.defineProperty(navigator, 'languages', {get: () => ['en-US', 'en']});
|
||||
window.chrome = {runtime: {}};
|
||||
const originalQuery = window.navigator.permissions.query;
|
||||
window.navigator.permissions.query = (parameters) => (
|
||||
parameters.name === 'notifications' ?
|
||||
Promise.resolve({state: Notification.permission}) :
|
||||
originalQuery(parameters)
|
||||
);
|
||||
Object.defineProperty(navigator, 'permissions', {get: () => window.navigator.permissions});
|
||||
""")
|
||||
|
||||
def extract_credentials():
|
||||
print("\n" + "="*50)
|
||||
print("[*] PHASE 1: SELENIUM EXTRACTION")
|
||||
print("="*50)
|
||||
print("[*] Launching Chrome browser...")
|
||||
|
||||
options = Options()
|
||||
if HEADLESS:
|
||||
options.add_argument("--headless=new")
|
||||
options.add_argument("--no-sandbox")
|
||||
options.add_argument("--disable-dev-shm-usage")
|
||||
options.add_argument("--disable-gpu")
|
||||
options.add_argument("--disable-software-rasterizer")
|
||||
options.add_argument("--disable-extensions")
|
||||
options.add_argument("--disable-blink-features=AutomationControlled")
|
||||
options.add_argument("--disable-setuid-sandbox")
|
||||
options.add_argument("--disable-background-networking")
|
||||
options.add_argument("--disable-default-apps")
|
||||
options.add_argument("--disable-sync")
|
||||
options.add_argument("--disable-translate")
|
||||
options.add_argument("--metrics-recording-only")
|
||||
options.add_argument("--mute-audio")
|
||||
options.add_argument("--no-first-run")
|
||||
options.add_argument("--safebrowsing-disable-auto-update")
|
||||
options.add_argument("--ignore-certificate-errors")
|
||||
options.add_argument("--ignore-ssl-errors")
|
||||
options.add_argument("--user-data-dir=/tmp/chrome-data")
|
||||
options.add_argument("--disable-features=IsolateOrigins,site-per-process")
|
||||
|
||||
driver = webdriver.Remote(
|
||||
command_executor=SELENIUM_URL + "/wd/hub",
|
||||
options=options
|
||||
)
|
||||
|
||||
stealth_js(driver)
|
||||
|
||||
try:
|
||||
print("[*] Navigating to OmegleWeb homepage...")
|
||||
driver.get(TARGET_URL)
|
||||
|
||||
wait = WebDriverWait(driver, CF_WAIT_TIME)
|
||||
wait.until(EC.presence_of_element_located((By.ID, "logo")))
|
||||
|
||||
print("[*] Time's up! Extracting the goods...")
|
||||
user_agent = driver.execute_script("return navigator.userAgent;")
|
||||
selenium_cookies = driver.get_cookies()
|
||||
cookie_string = "; ".join([f"{c['name']}={c['value']}" for c in selenium_cookies])
|
||||
|
||||
credentials["user_agent"] = user_agent
|
||||
credentials["cookies"] = cookie_string
|
||||
|
||||
print("[+] Extraction successful!")
|
||||
return True
|
||||
except Exception as e:
|
||||
print(f"[!] Extraction failed: {e}")
|
||||
return False
|
||||
finally:
|
||||
print("[*] Closing browser...")
|
||||
driver.quit()
|
||||
|
||||
async def start_bridge():
|
||||
async def bridge_handler(local_client):
|
||||
print(f"\n[{datetime.now().strftime('%H:%M:%S')}] [+] Local client connected to the bridge!")
|
||||
|
||||
while True:
|
||||
headers = {
|
||||
"User-Agent": credentials["user_agent"],
|
||||
"Cookie": credentials["cookies"]
|
||||
}
|
||||
|
||||
try:
|
||||
print(f"[*] Attempting tunnel to OmegleWeb...")
|
||||
async with websockets.connect(TARGET_WS_URL, additional_headers=headers) as target_server:
|
||||
print("[+] Tunnel established! Relaying messages...")
|
||||
|
||||
async def forward_local_to_target():
|
||||
async for message in local_client:
|
||||
print(f"[{datetime.now().strftime('%H:%M:%S.%f')[:-3]}] [Local -> Omegle]: {message}")
|
||||
await target_server.send(message)
|
||||
|
||||
async def forward_target_to_local():
|
||||
async for message in target_server:
|
||||
print(f"[{datetime.now().strftime('%H:%M:%S.%f')[:-3]}] [Omegle -> Local]: {message}")
|
||||
await local_client.send(message)
|
||||
|
||||
t1 = asyncio.create_task(forward_local_to_target())
|
||||
t2 = asyncio.create_task(forward_target_to_local())
|
||||
|
||||
done, pending = await asyncio.wait(
|
||||
[t1, t2],
|
||||
return_when=asyncio.FIRST_COMPLETED
|
||||
)
|
||||
|
||||
for task in pending:
|
||||
task.cancel()
|
||||
|
||||
if t1 in done and t1.exception() is None:
|
||||
print("[-] Local client disconnected.")
|
||||
break
|
||||
|
||||
print("[-] OmegleWeb connection closed.")
|
||||
break
|
||||
|
||||
except websockets.exceptions.InvalidStatusCode as e:
|
||||
if e.status_code == 403:
|
||||
print(f"[!] HTTP 403 Forbidden: Cookies likely expired. Refreshing...")
|
||||
loop = asyncio.get_running_loop()
|
||||
success = await loop.run_in_executor(None, extract_credentials)
|
||||
if not success:
|
||||
print("[!] Failed to refresh cookies. Retrying in 5s...")
|
||||
await asyncio.sleep(5)
|
||||
continue
|
||||
else:
|
||||
print(f"[!] Bridge error (Status {e.status_code}): {e}")
|
||||
break
|
||||
except websockets.exceptions.ConnectionClosed:
|
||||
print("[-] OmegleWeb disconnected.")
|
||||
break
|
||||
except Exception as e:
|
||||
print(f"[!] Bridge error: {type(e).__name__}: {e}")
|
||||
break
|
||||
|
||||
print(f"[*] Starting local WebSocket proxy on ws://{LOCAL_HOST}:{LOCAL_PORT}")
|
||||
async with websockets.serve(bridge_handler, LOCAL_HOST, LOCAL_PORT):
|
||||
await asyncio.Future()
|
||||
|
||||
def main():
|
||||
if not extract_credentials():
|
||||
print("[!] Initial extraction failed. Exiting.")
|
||||
return
|
||||
|
||||
print("\n" + "="*50)
|
||||
print("[*] PHASE 2: LAUNCH THE WEBSOCKET PROXY")
|
||||
print("="*50)
|
||||
|
||||
try:
|
||||
asyncio.run(start_bridge())
|
||||
except KeyboardInterrupt:
|
||||
print("\n[*] Shutting down...")
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
||||
3
proxy/start.sh
Executable file
3
proxy/start.sh
Executable file
@@ -0,0 +1,3 @@
|
||||
#!/bin/bash
|
||||
|
||||
env/bin/python3 main.py
|
||||
Reference in New Issue
Block a user