SpyScan

发现网站上的跟踪脚本、指纹识别和监控技术。

// ==UserScript==
// @name         SpyScan
// @namespace    https://gf.qytechs.cn/fr/users/1451802
// @version      1.0
// @description  Uncover tracking scripts, fingerprinting, and surveillance tactics lurking on the websites you visit
// @description:de Untersuche Websites auf Tracking-Skripte, Fingerprinting und Überwachungsmethoden.
// @description:es Descubre scripts de seguimiento, técnicas de huellas digitales y tácticas de vigilancia en las páginas web que visitas.
// @description:fr Détecte les scripts de suivi, le fingerprinting et les techniques de surveillance cachées sur les sites que vous visitez.
// @description:it Scopri script di tracciamento, tecniche di fingerprinting e metodi di sorveglianza sui siti web che visiti.
// @description:ru Раскрывает трекинговые скрипты, отпечатки браузера и методы слежки на посещаемых сайтах.
// @description:zh-CN 发现网站上的跟踪脚本、指纹识别和监控技术。
// @description:zh-TW 發現網站上的追蹤腳本、指紋辨識和監控技術。
// @description:ja 訪問したサイトに潜むトラッキングスクリプト、フィンガープリント、監視技術を検出。
// @description:ko 방문한 웹사이트에서 추적 스크립트, 브라우저 지문, 감시 기술을 찾아냅니다.
// @author       NormalRandomPeople (https://github.com/NormalRandomPeople)
// @match        *://*/*
// @grant        GM_addStyle
// @license      MIT
// @icon         https://www.svgrepo.com/show/360090/analyse.svg
// @compatible      chrome
// @compatible      firefox
// @compatible      opera
// @compatible      edge
// @compatible      brave
// @run-at document-end
// @noframes
// ==/UserScript==

/* jshint esversion: 8 */

(function() {
    'use strict';

    // Global arrays to hold detected network responses
    let detectedETags = [];
    let detectedIPGeolocationRequests = [];
    let detectedWebRTCLeaks = [];

    // List of known IP Geolocation service domains
    const ipGeoServices = [
        "ipinfo.io",
        "ip-api.com",
        "ipgeolocation.io",
        "geoip-db.com",
        "freegeoip.app",
        "ip2location.com",
        "extreme-ip-lookup.com",
        "ip-geolocation.whoisxmlapi.com",
        "ipligence.com",
        "bigdatacloud.com",
        "maxmind.com",
        "db-ip.com",
        "ipinfodb.com",
        "ipdata.co",
        "abstractapi.com",
        "ipapi.com",
        "ipstack.com",
        "geo.ipify.org",
        "ipwhois.io",
        "ipregistry.co",
        "telize.com",
        "geoplugin.com"
    ];

    // Patch fetch to capture responses with ETag headers and IP Geolocation requests
    const originalFetch = window.fetch;
    window.fetch = async function(...args) {
        let reqUrl = "";
        if (typeof args[0] === "string") {
            reqUrl = args[0];
        } else if (args[0] instanceof Request) {
            reqUrl = args[0].url;
        }
        if (ipGeoServices.some(domain => reqUrl.includes(domain))) {
            detectedIPGeolocationRequests.push({ url: reqUrl });
        }
        const response = await originalFetch.apply(this, args);
        const responseClone = response.clone();
        try {
            const etag = responseClone.headers.get("ETag");
            if (etag) {
                detectedETags.push({
                    url: responseClone.url,
                    etag: etag
                });
            }
        } catch (err) {
            console.warn("ETag header could not be read:", err);
        }
        return response;
    };

    // Patch XMLHttpRequest to capture responses with ETag headers and IP Geolocation requests
    const originalXHROpen = XMLHttpRequest.prototype.open;
    XMLHttpRequest.prototype.open = function(...args) {
        let reqUrl = args[1];
        if (ipGeoServices.some(domain => reqUrl.includes(domain))) {
            detectedIPGeolocationRequests.push({ url: reqUrl });
        }
        this.addEventListener("readystatechange", function() {
            if (this.readyState === 4) {
                try {
                    const etag = this.getResponseHeader("ETag");
                    if (etag) {
                        detectedETags.push({
                            url: this.responseURL,
                            etag: etag
                        });
                    }
                } catch (err) {
                    console.warn("ETag header could not be read from XHR:", err);
                }
            }
        });
        return originalXHROpen.apply(this, args);
    };

    let scanButton = document.createElement("button");
    scanButton.id = "aptScanButton";
    scanButton.textContent = "";
    let svgImg = document.createElement("img");
    svgImg.src = "https://www.svgrepo.com/show/360090/analyse.svg";
    svgImg.style.width = "32px";
    svgImg.style.height = "32px";
    svgImg.style.display = "block";
    svgImg.style.margin = "0 auto";
    scanButton.appendChild(svgImg);
    scanButton.style.position = "fixed";
    scanButton.style.bottom = "10px";
    scanButton.style.left = "10px";
    scanButton.style.padding = "15px 20px";
    scanButton.style.border = "none";
    scanButton.style.backgroundColor = "black";
    scanButton.style.color = "#fff";
    scanButton.style.borderRadius = "10px";
    scanButton.style.cursor = "pointer";
    scanButton.style.zIndex = "9999999999";
    scanButton.style.boxShadow = "0 4px 8px rgba(0, 0, 0, 0.3)";
    scanButton.style.transition = "background-color 0.3s, transform 0.3s";
    scanButton.addEventListener("mouseover", function() {
        scanButton.style.backgroundColor = "#333";
        scanButton.style.transform = "scale(1.05)";
    });
    scanButton.addEventListener("mouseout", function() {
        scanButton.style.backgroundColor = "black";
        scanButton.style.transform = "scale(1)";
    });

    document.body.appendChild(scanButton);

    let auditWindow = document.createElement("div");
    auditWindow.id = "aptAuditWindow";
    let windowContent = document.createElement("div");
    windowContent.className = "aptWindowContent";
    auditWindow.appendChild(windowContent);
    document.body.appendChild(auditWindow);

    auditWindow.addEventListener("click", function(event) {
        if (event.target === auditWindow) {
            auditWindow.style.display = "none";
        }
    });

    GM_addStyle(`
        #aptScanButton {
            font-family: Arial, sans-serif;
            background-color: black;
            color: #fff;
            border: none;
            padding: 15px 20px;
            font-size: 18px;
            border-radius: 10px;
            cursor: pointer;
            transition: background-color 0.3s, transform 0.3s;
        }
        #aptScanButton:hover {
            background-color: #333;
        }
        #aptAuditWindow {
            display: none;
            position: fixed;
            top: 0;
            left: 0;
            width: 100%;
            height: 100%;
            background-color: rgba(0, 0, 0, 0.7);
            color: #fff;
            font-family: Arial, sans-serif;
            overflow: auto;
            padding: 20px;
            z-index: 99999999999;
            box-sizing: border-box;
        }
        .aptWindowContent {
            max-width: 800px;
            margin: 50px auto;
            background-color: #333;
            border-radius: 8px;
            padding: 20px;
            box-shadow: 0 0 20px rgba(0, 0, 0, 0.5);
            overflow-y: auto;
            max-height: 80%;
        }
        .aptWindowContent h2 {
            text-align: center;
            margin-bottom: 10px;
            font-size: 1.8em;
        }
        .aptWindowContent p {
            font-size: 1em;
            line-height: 1.5;
        }
        .aptWindowContent ul {
            list-style-type: none;
            padding: 0;
        }
        .aptWindowContent li {
            background-color: #444;
            padding: 10px;
            margin: 5px 0;
            border-radius: 5px;
            word-wrap: break-word;
            position: relative;
        }
        .aptTitle {
            font-weight: bold;
            font-family: Arial;
            color: grey;
        }
        .aptSectionTitle {
            font-size: 1.3em;
            font-weight: bold;
            margin-bottom: 10px;
            padding-bottom: 5px;
            border-bottom: 2px solid #666;
            margin-top: 20px;
        }
        .aptDangerLevel {
            font-weight: bold;
            font-size: 1.1em;
        }
        .aptDangerLevelLow {
            color: #28A745;
        }
        .aptDangerLevelMedium {
            color: #FFA500;
        }
        .aptDangerLevelHigh {
            color: #FF4C4C;
        }
        .aptloading-spinner {
            border: 4px solid rgba(255, 255, 255, 0.3);
            border-top: 4px solid #fff;
            border-radius: 50%;
            width: 40px;
            height: 40px;
            animation: spin 1s linear infinite;
            margin: 20px auto;
        }
        @keyframes spin {
            0% { transform: rotate(0deg); }
            100% { transform: rotate(360deg); }
        }
    `);

    function getCookies() {
        return document.cookie.split(';').map(cookie => cookie.trim()).filter(cookie => cookie);
    }

    async function detectWebRTCLeak() {
        return new Promise(resolve => {
            const rtcPeerConnection = new RTCPeerConnection({
                iceServers: [{ urls: "stun:stun.l.google.com:19302" }]
            });

            rtcPeerConnection.createDataChannel("");
            rtcPeerConnection.createOffer().then(offer => rtcPeerConnection.setLocalDescription(offer));

            rtcPeerConnection.onicecandidate = event => {
                if (event.candidate) {
                    const ipRegex = /([0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3})/;
                    const match = ipRegex.exec(event.candidate.candidate);
                    if (match && !match[1].startsWith("192.168") && !match[1].startsWith("10.") && !match[1].startsWith("172.")) {
                        detectedWebRTCLeaks.push({
                            name: "WebRTC Leak",
                            danger: "high",
                            description: `Your real IP is exposed via WebRTC: ${match[1]}`
                        });
                    }
                }
            };

            setTimeout(() => {
                rtcPeerConnection.close();
                resolve(detectedWebRTCLeaks);
            }, 5000);
        });
    }

    function detectWebBeacons() {
        let beacons = [];
        let images = document.getElementsByTagName("img");
        for (let img of images) {
            let width = img.getAttribute("width") || img.width;
            let height = img.getAttribute("height") || img.height;
            let computedStyle = window.getComputedStyle(img);
            if ((parseInt(width) === 1 && parseInt(height) === 1) ||
                (img.naturalWidth === 1 && img.naturalHeight === 1) ||
                (computedStyle.width === "1px" && computedStyle.height === "1px")) {
                beacons.push({
                    name: "Web Beacon",
                    src: img.src,
                    danger: "medium",
                    description: "Detected a 1x1 pixel image that could be used as a web beacon."
                });
            }
        }
        return beacons;
    }

    function detectEtagTracking() {
        let etagTrackers = [];
        detectedETags.forEach(item => {
            etagTrackers.push({
                name: "Etag Tracking",
                danger: "medium",
                description: `ETag detected from ${item.url} with value ${item.etag}`
            });
        });
        return etagTrackers;
    }

    function detectIPGeolocation() {
        let ipGeoTrackers = [];
        detectedIPGeolocationRequests.forEach(item => {
            ipGeoTrackers.push({
                name: "IP Geolocation",
                danger: "high",
                description: `IP Geolocation request detected to ${item.url}`
            });
        });
        return ipGeoTrackers;
    }

    function detectTrackersSync() {
        const trackers = [];
        const knownTrackers = [
            { name: 'Google Analytics', regex: /analytics\.js/, danger: 'medium', description: 'Tracks user behavior for analytics and advertising purposes.' },
            { name: 'Facebook Pixel', regex: /facebook\.com\/tr\.js/, danger: 'medium', description: 'Tracks user activity for targeted ads on Facebook.' },
            { name: 'Hotjar', regex: /hotjar\.com/, danger: 'medium', description: 'Records user behavior such as clicks and scrolling for website optimization.' },
            { name: 'AdSense', regex: /pagead2\.googlesyndication\.com/, danger: 'medium', description: 'Google\'s ad network, tracks user activity for ads.' },
            { name: 'Google Tag Manager', regex: /googletagmanager\.com/, danger: 'medium', description: 'Manages JavaScript and HTML tags for tracking purposes.' },
            { name: 'Amazon Tracking', regex: /amazon\.com\/at\/tag/, danger: 'low', description: 'Tracks activity for Amazon ads and recommendations.' },
            { name: 'Twitter', regex: /twitter\.com\/widgets\.js/, danger: 'low', description: 'Tracks activity for Twitter widgets and ads.' },
            { name: 'Local Storage', regex: /localStorage/, danger: 'low', description: 'Stores data in the browser that can be used for persistent tracking.' },
            { name: 'Session Storage', regex: /sessionStorage/, danger: 'low', description: 'Stores data temporarily in the browser during a session.' },
        ];

        knownTrackers.forEach(tracker => {
            if (document.body.innerHTML.match(tracker.regex)) {
                trackers.push({ name: tracker.name, danger: tracker.danger, description: tracker.description });
            }
        });

        let webBeacons = detectWebBeacons();
        webBeacons.forEach(beacon => trackers.push(beacon));

        let etagTrackers = detectEtagTracking();
        etagTrackers.forEach(etag => trackers.push(etag));

        let ipGeoTrackers = detectIPGeolocation();
        ipGeoTrackers.forEach(ipgeo => trackers.push(ipgeo));

        return trackers;
    }

    function detectZombieCookies() {
        return new Promise(resolve => {
            const testName = "aptZombieTest";
            document.cookie = `${testName}=test; path=/;`;
            document.cookie = `${testName}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;`;
            setTimeout(() => {
                if (document.cookie.includes(testName + "=")) {
                    resolve([{
                        name: "Zombie Cookies",
                        danger: "high",
                        description: "Test cookie was recreated, indicating persistent zombie cookie behavior."
                    }]);
                } else {
                    resolve([]);
                }
            }, 1000);
        });
    }

    async function detectAllTrackers() {
        const trackersSync = detectTrackersSync();
        const zombieTrackers = await detectZombieCookies();
        const webrtcLeaks = await detectWebRTCLeak();
        return trackersSync.concat(zombieTrackers, webrtcLeaks);
    }

    async function detectFingerprinting() {
        let fingerprintingMethods = [];
        try {
            // Canvas Fingerprinting
            const canvas = document.createElement('canvas');
            const ctx = canvas.getContext('2d');
            ctx.textBaseline = "top";
            ctx.font = "14px 'Arial'";
            ctx.fillText('test', 2, 2);
            const data = canvas.toDataURL();
            if (data !== '') {
                fingerprintingMethods.push({ name: 'Canvas Fingerprinting', danger: 'high', description: 'Uses HTML5 canvas to uniquely identify users based on their rendering properties.' });
            }
        } catch (e) {}

        try {
            // WebGL Fingerprinting
            const glCanvas = document.createElement('canvas');
            const gl = glCanvas.getContext('webgl');
            if (gl) {
                const fingerprint = gl.getParameter(gl.VERSION);
                if (fingerprint) {
                    fingerprintingMethods.push({ name: 'WebGL Fingerprinting', danger: 'high', description: 'Uses WebGL rendering data to track users.' });
                }
            }
        } catch (e) {}

        try {
            // Font Fingerprinting
            const fontFingerprint = document.createElement('div');
            fontFingerprint.style.fontFamily = "'Arial', 'sans-serif'";
            fontFingerprint.innerText = "test";
            document.body.appendChild(fontFingerprint);
            const fontFingerprintData = window.getComputedStyle(fontFingerprint).fontFamily;
            document.body.removeChild(fontFingerprint);
            if (fontFingerprintData.includes('Arial')) {
                fingerprintingMethods.push({ name: 'Font Fingerprinting', danger: 'medium', description: 'Uses unique system fonts to track users across sessions.' });
            }
        } catch (e) {}

        try {
            // AudioContext Fingerprinting using AudioWorkletNode if available
            const audioContext = new (window.AudioContext || window.webkitAudioContext)();
            let audioFingerprintValue = 0;
            if (audioContext.audioWorklet) {
                // Define an inline AudioWorkletProcessor as a string
                const workletCode = `
                class FingerprintProcessor extends AudioWorkletProcessor {
                  process(inputs, outputs, parameters) {
                    let sum = 0;
                    if (inputs[0] && inputs[0][0]) {
                      const input = inputs[0][0];
                      for (let i = 0; i < input.length; i++) {
                        sum += input[i];
                      }
                    }
                    this.port.postMessage(sum);
                    return false;
                  }
                }
                registerProcessor('fingerprint-processor', FingerprintProcessor);
                `;
                const blob = new Blob([workletCode], { type: 'application/javascript' });
                const moduleUrl = URL.createObjectURL(blob);
                await audioContext.audioWorklet.addModule(moduleUrl);
                const workletNode = new AudioWorkletNode(audioContext, 'fingerprint-processor');
                audioFingerprintValue = await new Promise(resolve => {
                    workletNode.port.onmessage = (event) => {
                        resolve(event.data);
                    };
                    workletNode.connect(audioContext.destination);
                });
                workletNode.disconnect();
            } else {
                // Fallback to ScriptProcessorNode if AudioWorklet is not available
                const buffer = audioContext.createBuffer(1, 1, 22050);
                const processor = audioContext.createScriptProcessor(0, 1, 1);
                processor.connect(audioContext.destination);
                audioFingerprintValue = buffer.getChannelData(0)[0];
                processor.disconnect();
            }
            if (audioFingerprintValue !== 0) {
                fingerprintingMethods.push({ name: 'AudioContext Fingerprinting', danger: 'high', description: 'Uses audio hardware properties to uniquely identify users. Value: ' + audioFingerprintValue });
            }
        } catch (e) {}
        return fingerprintingMethods;
    }

    async function showAuditResults() {
        windowContent.innerHTML = '<div class="aptloading-spinner"></div><p style="text-align: center;">Scanning...</p>';
        auditWindow.style.display = "block";
        const cookies = getCookies();
        const trackers = await detectAllTrackers();
        const fingerprinting = await detectFingerprinting();
        windowContent.innerHTML = `
            <h2 class="aptTitle">Privacy Audit Results</h2>
            <div class="aptSectionTitle">Trackers & Fingerprinting</div>
            <ul>
                ${trackers.length > 0 ? trackers.map(tracker => `
                    <li>${tracker.name} <span class="aptDangerLevel aptDangerLevel${capitalizeFirstLetter(tracker.danger)}">${capitalizeFirstLetter(tracker.danger)}</span> - ${tracker.description}</li>`).join('') : '<li>No trackers found.</li>'}
            </ul>
            <div class="aptSectionTitle">Cookies</div>
            <ul>
                ${!cookies.length ? '<li>No cookies found.</li>' : cookies.map(cookie => `<li>${cookie}</li>`).join('') }
            </ul>
        `;
        auditWindow.style.display = "block";
    }

    function capitalizeFirstLetter(string) {
        return string.charAt(0).toUpperCase() + string.slice(1);
    }

    scanButton.addEventListener("click", async function() {
        await showAuditResults();
    });

})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址