您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Analyse passive d’un site web : robots.txt, métadonnées, IP / DNS, commentaires HTML et outils OSINT externes.
// ==UserScript== // @name Wrench // @namespace http://tampermonkey.net/ // @version 2.9 // @description Analyse passive d’un site web : robots.txt, métadonnées, IP / DNS, commentaires HTML et outils OSINT externes. // @author Th3rd // @match *://*/* // @grant GM_xmlhttpRequest // @connect ipwhois.app // @connect dns.google // @connect www.google.com // @connect urlscan.io // @connect shodan.io // @connect hunter.io // @connect who.is // @connect web.archive.org // @grant unsafeWindow // @connect * // @run-at document-end // @license GPL-3.0 // @icon https://github.com/Th3rdMan/wrench-userscript/blob/main/wrench.png?raw=true // @namespace https://github.com/Th3rdMan/wrench-userscript // ==/UserScript== (function () { 'use strict'; const ICON_WRENCH = ''; const ICON_CLOSE = ''; const baseUrl = location.origin; const robotsUrl = `${baseUrl}/robots.txt`; // flagcdn bloqué par CSP const FLAG_EMOJIS = { "AD": "🇦🇩", "AE": "🇦🇪", "AF": "🇦🇫", "AG": "🇦🇬", "AI": "🇦🇮", "AL": "🇦🇱", "AM": "🇦🇲", "AO": "🇦🇴", "AR": "🇦🇷", "AS": "🇦🇸", "AT": "🇦🇹", "AU": "🇦🇺", "AW": "🇦🇼", "AX": "🇦🇽", "AZ": "🇦🇿", "BA": "🇧🇦", "BB": "🇧🇧", "BD": "🇧🇩", "BE": "🇧🇪", "BF": "🇧🇫", "BG": "🇧🇬", "BH": "🇧🇭", "BI": "🇧🇮", "BJ": "🇧🇯", "BL": "🇧🇱", "BM": "🇧🇲", "BN": "🇧🇳", "BO": "🇧🇴", "BQ": "🇧🇶", "BR": "🇧🇷", "BS": "🇧🇸", "BT": "🇧🇹", "BV": "🇧🇻", "BW": "🇧🇼", "BY": "🇧🇾", "BZ": "🇧🇿", "CA": "🇨🇦", "CC": "🇨🇨", "CD": "🇨🇩", "CF": "🇨🇫", "CG": "🇨🇬", "CH": "🇨🇭", "CI": "🇨🇮", "CK": "🇨🇰", "CL": "🇨🇱", "CM": "🇨🇲", "CN": "🇨🇳", "CO": "🇨🇴", "CR": "🇨🇷", "CU": "🇨🇺", "CV": "🇨🇻", "CW": "🇨🇼", "CX": "🇨🇽", "CY": "🇨🇾", "CZ": "🇨🇿", "DE": "🇩🇪", "DJ": "🇩🇯", "DK": "🇩🇰", "DM": "🇩🇲", "DO": "🇩🇴", "DZ": "🇩🇿", "EC": "🇪🇨", "EE": "🇪🇪", "EG": "🇪🇬", "EH": "🇪🇭", "ER": "🇪🇷", "ES": "🇪🇸", "ET": "🇪🇹", "FI": "🇫🇮", "FJ": "🇫🇯", "FM": "🇫🇲", "FO": "🇫🇴", "FR": "🇫🇷", "GA": "🇬🇦", "GB": "🇬🇧", "GD": "🇬🇩", "GE": "🇬🇪", "GF": "🇬🇫", "GG": "🇬🇬", "GH": "🇬🇭", "GI": "🇬🇮", "GL": "🇬🇱", "GM": "🇬🇲", "GN": "🇬🇳", "GP": "🇬🇵", "GQ": "🇬🇶", "GR": "🇬🇷", "GT": "🇬🇹", "GU": "🇬🇺", "GW": "🇬🇼", "GY": "🇬🇾", "HK": "🇭🇰", "HM": "🇭🇲", "HN": "🇭🇳", "HR": "🇭🇷", "HT": "🇭🇹", "HU": "🇭🇺", "ID": "🇮🇩", "IE": "🇮🇪", "IL": "🇮🇱", "IM": "🇮🇲", "IN": "🇮🇳", "IO": "🇮🇴", "IQ": "🇮🇶", "IR": "🇮🇷", "IS": "🇮🇸", "IT": "🇮🇹", "JE": "🇯🇪", "JM": "🇯🇲", "JO": "🇯🇴", "JP": "🇯🇵", "KE": "🇰🇪", "KG": "🇰🇬", "KH": "🇰🇭", "KI": "🇰🇮", "KM": "🇰🇲", "KN": "🇰🇳", "KP": "🇰🇵", "KR": "🇰🇷", "KW": "🇰🇼", "KY": "🇰🇾", "KZ": "🇰🇿", "LA": "🇱🇦", "LB": "🇱🇧", "LC": "🇱🇨", "LI": "🇱🇮", "LK": "🇱🇰", "LR": "🇱🇷", "LS": "🇱🇸", "LT": "🇱🇹", "LU": "🇱🇺", "LV": "🇱🇻", "LY": "🇱🇾", "MA": "🇲🇦", "MC": "🇲🇨", "MD": "🇲🇩", "ME": "🇲🇪", "MF": "🇲🇫", "MG": "🇲🇬", "MH": "🇲🇭", "MK": "🇲🇰", "ML": "🇲🇱", "MM": "🇲🇲", "MN": "🇲🇳", "MO": "🇲🇴", "MP": "🇲🇵", "MQ": "🇲🇶", "MR": "🇲🇷", "MS": "🇲🇸", "MT": "🇲🇹", "MU": "🇲🇺", "MV": "🇲🇻", "MW": "🇲🇼", "MX": "🇲🇽", "MY": "🇲🇾", "MZ": "🇲🇿", "NA": "🇳🇦", "NC": "🇳🇨", "NE": "🇳🇪", "NF": "🇳🇫", "NG": "🇳🇬", "NI": "🇳🇮", "NL": "🇳🇱", "NO": "🇳🇴", "NP": "🇳🇵", "NR": "🇳🇷", "NU": "🇳🇺", "NZ": "🇳🇿", "OM": "🇴🇲", "PA": "🇵🇦", "PE": "🇵🇪", "PF": "🇵🇫", "PG": "🇵🇬", "PH": "🇵🇭", "PK": "🇵🇰", "PL": "🇵🇱", "PM": "🇵🇲", "PN": "🇵🇳", "PR": "🇵🇷", "PT": "🇵🇹", "PW": "🇵🇼", "PY": "🇵🇾", "QA": "🇶🇦", "RE": "🇷🇪", "RO": "🇷🇴", "RS": "🇷🇸", "RU": "🇷🇺", "RW": "🇷🇼", "SA": "🇸🇦", "SB": "🇸🇧", "SC": "🇸🇨", "SD": "🇸🇩", "SE": "🇸🇪", "SG": "🇸🇬", "SH": "🇸🇭", "SI": "🇸🇮", "SJ": "🇸🇯", "SK": "🇸🇰", "SL": "🇸🇱", "SM": "🇸🇲", "SN": "🇸🇳", "SO": "🇸🇴", "SR": "🇸🇷", "SS": "🇸🇸", "ST": "🇸🇹", "SV": "🇸🇻", "SX": "🇸🇽", "SY": "🇸🇾", "SZ": "🇸🇿", "TC": "🇹🇨", "TD": "🇹🇩", "TF": "🇹🇫", "TG": "🇹🇬", "TH": "🇹🇭", "TJ": "🇹🇯", "TK": "🇹🇰", "TL": "🇹🇱", "TM": "🇹🇲", "TN": "🇹🇳", "TO": "🇹🇴", "TR": "🇹🇷", "TT": "🇹🇹", "TV": "🇹🇻", "TZ": "🇹🇿", "UA": "🇺🇦", "UG": "🇺🇬", "UM": "🇺🇲", "US": "🇺🇸", "UY": "🇺🇾", "UZ": "🇺🇿", "VA": "🇻🇦", "VC": "🇻🇨", "VE": "🇻🇪", "VG": "🇻🇬", "VI": "🇻🇮", "VN": "🇻🇳", "VU": "🇻🇺", "WF": "🇼🇫", "WS": "🇼🇸", "YE": "🇾🇪", "YT": "🇾🇹", "ZA": "🇿🇦", "ZM": "🇿🇲", "ZW": "🇿🇼" }; function getFlagEmoji(countryCode) { return FLAG_EMOJIS[countryCode?.toUpperCase()] || ''; } let bannerVisible = false; const toggleIcon = document.createElement('img'); toggleIcon.src = ICON_WRENCH; toggleIcon.style.cssText = 'position:fixed;top:60px;right:10px;width:36px;height:36px;cursor:pointer;z-index:100000;border-radius:50%;box-shadow:0 2px 6px rgba(0,0,0,0.4);transition:transform 0.2s;'; toggleIcon.addEventListener('mouseenter', () => { toggleIcon.style.transform = 'scale(1.1)'; }); toggleIcon.addEventListener('mouseleave', () => { toggleIcon.style.transform = 'scale(1)'; }); toggleIcon.addEventListener('click', toggleBanner); document.body.appendChild(toggleIcon); const banner = document.createElement('div'); banner.id = 'osinter-banner'; banner.style.cssText = 'display:none;position:fixed;top:0;left:0;width:100%;max-height:300px;overflow:auto;background:#111;color:#0f0;font-family:monospace;font-size:13px;white-space:pre-wrap;padding:10px 16px;z-index:99999;border-bottom:2px solid #444;box-shadow:0 2px 4px rgba(0,0,0,0.3);'; document.body.prepend(banner); const menu = document.createElement('div'); menu.style.cssText = 'display:flex;flex-wrap:wrap;gap:8px;margin-bottom:8px;'; banner.appendChild(menu); const content = document.createElement('div'); banner.appendChild(content); function addButton(label, action) { const btn = document.createElement('button'); btn.textContent = label; btn.style.cssText = 'background:#222;color:#0f0;border:1px solid #444;padding:4px 8px;cursor:pointer;font-family:monospace;'; btn.addEventListener('click', action); menu.appendChild(btn); } function toggleBanner() { bannerVisible = !bannerVisible; banner.style.display = bannerVisible ? 'block' : 'none'; toggleIcon.src = bannerVisible ? ICON_CLOSE : ICON_WRENCH; } function loadRobotsTxt() { content.innerHTML = 'Chargement robots.txt...'; GM_xmlhttpRequest({ method: 'GET', url: robotsUrl, onload: res => { if (res.status === 404) { content.innerHTML = "Aucun fichier robots.txt trouvé (404)."; return; } if (res.status >= 400) { content.innerHTML = `Erreur lors du chargement du robots.txt (HTTP ${res.status})`; return; } const lines = res.responseText.trim().split('\n'); const sitemaps = [], others = []; for (let line of lines) { if (/^Sitemap:/i.test(line)) { const url = line.replace(/^Sitemap:\s*/i, '').trim(); sitemaps.push(`<strong><u>Sitemap:</u></strong> <a href='${url}' target='_blank' style='color:#6cf'>${url}</a>`); } else if (/^User-agent:/i.test(line)) others.push(`<span style='color:#ff0;'>${line}</span>`); else if (/^Disallow:/i.test(line)) others.push(`<span style='color:#f55;'>${line}</span>`); else if (/^Allow:/i.test(line)) others.push(`<span style='color:#5f5;'>${line}</span>`); else others.push(line); } content.innerHTML = [...sitemaps, ...others].join('\n'); }, onerror: () => { content.innerHTML = 'Erreur lors du chargement.'; } }); } function loadMeta() { const meta = document.getElementsByTagName('meta'); let info = `<strong>Titre</strong> : ${document.title}`; for (let m of meta) { if (m.name === 'description') info += `<br><strong>Description</strong> : ${m.content}`; if (m.name === 'author') info += `<br><strong>Auteur</strong> : ${m.content}`; } const c = document.querySelector("link[rel='canonical']"); if (c) info += `<br><strong>Canonical</strong> : ${c.href}`; content.innerHTML = info; } function loadIPDNS() { const d = location.hostname; content.innerHTML = 'Résolution DNS...'; GM_xmlhttpRequest({ method: 'GET', url: `https://dns.google/resolve?name=${d}&type=A`, onload: res => { const data = JSON.parse(res.responseText); if (!data.Answer) { content.innerHTML = 'Aucune IP trouvée.'; return; } const aRecords = data.Answer.filter(a => a.type === 1); if (aRecords.length === 0) { content.innerHTML = 'Aucune IP trouvée.'; return; } content.innerHTML = 'Chargement des infos IP...'; Promise.all( aRecords.map(a => new Promise(resolve => { const ip = a.data; GM_xmlhttpRequest({ method: 'GET', url: `https://ipwhois.app/json/${ip}`, onload: r => { const g = JSON.parse(r.responseText); const f = getFlagEmoji(g.country_code); resolve(`IP : ${ip}<br>Pays : ${g.country} ${f} (${g.country_code})<br>ASN : ${g.org}`); }, onerror: () => resolve(`IP : ${ip}<br>Localisation indisponible.`) }); })) ).then(results => { content.innerHTML = results.join('<br><br>'); }); }, onerror: function() { content.innerHTML = 'Erreur DNS.'; } }); } function showTools() { const d = location.hostname; const tools = [ { name: 'URLScan', url: `https://urlscan.io/domain/${d}` }, { name: 'Shodan', url: `https://www.shodan.io/search?query=hostname:${d}` }, { name: 'Hunter.io', url: `https://hunter.io/search/${d}` }, { name: 'WHOIS', url: `https://who.is/whois/${d}` }, { name: 'Wayback Machine', url: `https://web.archive.org/web/*/${d}` } ]; const emojiMap = { "URLScan": "🔎", "Shodan": "🛰️", "Hunter.io": "🦊", "WHOIS": "🕵️", "Wayback Machine": "⏳" }; content.innerHTML = tools.map(t => `${emojiMap[t.name] || '🔗'} <a href="${t.url}" target="_blank" style="color:#6cf;text-decoration:none;">${t.name}</a>` ).join('<br>'); } function escapeHTML(str) { return str.replace(/[&<>'"]/g, c => ({ '&':'&', '<':'<', '>':'>', "'":''', '"':'"' }[c])); } function extractCommentsFromDOM(node, arr = []) { for (let child of node.childNodes) { if (child.nodeType === Node.COMMENT_NODE) arr.push(child.nodeValue.trim()); else extractCommentsFromDOM(child, arr); } return arr; } function showComments() { content.innerHTML = 'Chargement et analyse du code source...'; GM_xmlhttpRequest({ method: 'GET', url: document.location.href, onload: res => { const matches = [...res.responseText.matchAll(/<!--([\s\S]*?)-->/g)]; const comments = matches.map(m => m[1].trim()).filter(Boolean); const emails = [...res.responseText.matchAll(/[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/g)] .map(m => m[0]); const uniqueEmails = Array.from(new Set(emails)); let html = ''; html += `<strong><u>Commentaires HTML trouvés :</u></strong><br>`; html += comments.length ? comments.map(c => `<pre style="white-space:pre-wrap;background:#222;color:#6cf;padding:4px;"><!-- ${escapeHTML(c)} --></pre>`).join('') : "<i>Aucun commentaire HTML détecté dans le code source.</i>"; html += `<hr style="margin:10px 0;border:0;border-top:1px solid #333;">`; html += `<strong><u>Adresses e-mail détectées :</u></strong><br>`; html += uniqueEmails.length ? uniqueEmails.map(email => `<span style="color:#ffd700">${escapeHTML(email)}</span>`).join('<br>') : "<i>Aucune adresse e-mail détectée dans le code source.</i>"; content.innerHTML = html; }, onerror: function() { content.innerHTML = 'Erreur lors du chargement du code source.'; } }); } [ ['Robots.txt', loadRobotsTxt], ['Métadonnées', loadMeta], ['IP / DNS', loadIPDNS], ['Code Source', showComments], ['Outils externes', showTools] ].forEach(([label, action]) => addButton(label, action)); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址