File size: 11,217 Bytes
6a7089a | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 | <!DOCTYPE html>
<html>
<head>
<title>Bot Detection Tests</title>
<style>
body { font-family: monospace; padding: 20px; background: #1a1a1a; color: #eee; }
.pass { color: #4caf50; }
.fail { color: #f44336; }
.warn { color: #ff9800; }
table { border-collapse: collapse; margin: 20px 0; width: 100%; }
td, th { border: 1px solid #444; padding: 8px; text-align: left; }
th { background: #333; }
h2 { color: #888; margin-top: 30px; }
</style>
</head>
<body>
<h1>π€ Bot Detection Tests</h1>
<p>Comprehensive checks based on bot.sannysoft.com and other detection methods</p>
<h2>Critical Tests (must pass)</h2>
<table id="critical">
<tr><th>Test</th><th>Result</th><th>Value</th></tr>
</table>
<h2>Warning Tests (should pass)</h2>
<table id="warnings">
<tr><th>Test</th><th>Result</th><th>Value</th></tr>
</table>
<h2>Info</h2>
<table id="info">
<tr><th>Property</th><th>Value</th></tr>
</table>
<script>
const results = {};
const criticalTable = document.getElementById('critical');
const warnTable = document.getElementById('warnings');
const infoTable = document.getElementById('info');
function addResult(name, passed, value, severity = 'critical') {
results[name] = { passed, value, severity };
const table = severity === 'critical' ? criticalTable : warnTable;
const row = table.insertRow();
row.innerHTML = `
<td>${name}</td>
<td class="${passed ? 'pass' : (severity === 'warn' ? 'warn' : 'fail')}">${passed ? 'β PASS' : (severity === 'warn' ? 'β WARN' : 'β FAIL')}</td>
<td>${String(value).substring(0, 100)}</td>
`;
}
function addInfo(name, value) {
const row = infoTable.insertRow();
row.innerHTML = `<td>${name}</td><td>${String(value).substring(0, 150)}</td>`;
}
// βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
// CRITICAL TESTS - These MUST pass to avoid detection
// βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
// Test: navigator.webdriver value (should be undefined/false, not true)
const webdriverValue = navigator.webdriver;
addResult('webdriver_value', webdriverValue === undefined || webdriverValue === false,
`navigator.webdriver = ${webdriverValue}`);
// Test: "webdriver" in navigator (property should not exist)
const webdriverExists = 'webdriver' in navigator;
addResult('webdriver_in_navigator', !webdriverExists,
`"webdriver" in navigator = ${webdriverExists}`);
// Test: Plugins instanceof PluginArray
const pluginsValid = navigator.plugins instanceof PluginArray;
addResult('plugins_instanceof', pluginsValid,
`navigator.plugins instanceof PluginArray = ${pluginsValid}`);
// Test: Plugins has items
const hasPlugins = navigator.plugins && navigator.plugins.length > 0;
addResult('plugins_present', hasPlugins,
`navigator.plugins.length = ${navigator.plugins?.length || 0}`);
// Test: chrome.runtime exists (Chrome extension API)
const chromeRuntime = !!(window.chrome && window.chrome.runtime);
addResult('chrome_runtime', chromeRuntime,
`window.chrome.runtime = ${chromeRuntime}`);
// Test: User agent not headless
const ua = navigator.userAgent;
const isHeadlessUA = ua.includes('HeadlessChrome') || ua.includes('Headless');
addResult('ua_not_headless', !isHeadlessUA,
isHeadlessUA ? 'Contains HeadlessChrome' : 'Normal Chrome UA');
// Test: Languages defined
const hasLanguages = navigator.languages && navigator.languages.length > 0;
addResult('languages', hasLanguages,
`navigator.languages = ${navigator.languages?.join(', ') || 'none'}`);
// Test: Permissions API exists and works
let permissionsOk = false;
try {
permissionsOk = navigator.permissions && typeof navigator.permissions.query === 'function';
} catch (e) {}
addResult('permissions_api', permissionsOk,
permissionsOk ? 'Available' : 'Broken/Missing');
// Test: CDP Runtime domain detection (automation leaves traces)
const hasCDPTraces = !!(window.cdc_adoQpoasnfa76pfcZLmcfl_Array ||
window.cdc_adoQpoasnfa76pfcZLmcfl_Promise ||
window.__webdriver_script_fn);
addResult('no_cdp_traces', !hasCDPTraces,
hasCDPTraces ? 'CDP traces found!' : 'Clean');
// βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
// WARNING TESTS - Nice to pass, but not always critical
// βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
// Test: Platform matches UA
const platform = navigator.platform;
let platformMatch = true;
if (ua.includes('Linux') && !platform.includes('Linux')) platformMatch = false;
if (ua.includes('Macintosh') && platform !== 'MacIntel') platformMatch = false;
if (ua.includes('Windows') && !platform.includes('Win')) platformMatch = false;
addResult('platform_ua_match', platformMatch,
`UA: ${ua.includes('Linux') ? 'Linux' : ua.includes('Mac') ? 'Mac' : 'Win'}, platform: ${platform}`, 'warn');
// Test: WebGL not SwiftShader (CPU renderer = headless)
let webglRenderer = 'unknown';
try {
const canvas = document.createElement('canvas');
const gl = canvas.getContext('webgl') || canvas.getContext('experimental-webgl');
if (gl) {
const dbg = gl.getExtension('WEBGL_debug_renderer_info');
if (dbg) webglRenderer = gl.getParameter(dbg.UNMASKED_RENDERER_WEBGL);
}
} catch (e) {}
const isSwiftShader = webglRenderer.toLowerCase().includes('swiftshader');
addResult('webgl_not_swiftshader', !isSwiftShader, webglRenderer, 'warn');
// Test: Outer dimensions exist (headless may have 0)
const hasOuterDims = window.outerWidth > 0 && window.outerHeight > 0;
addResult('outer_dimensions', hasOuterDims,
`${window.outerWidth}x${window.outerHeight}`, 'warn');
// Test: Hardware concurrency reasonable
const cores = navigator.hardwareConcurrency || 0;
const coresOk = cores >= 2 && cores <= 64;
addResult('hardware_concurrency', coresOk, `${cores} cores`, 'warn');
// Test: Device memory exists
const deviceMem = navigator.deviceMemory;
const hasDeviceMem = deviceMem && deviceMem > 0;
addResult('device_memory', hasDeviceMem, `${deviceMem || 0} GB`, 'warn');
// Test: Connection RTT exists
const rtt = navigator.connection?.rtt;
const hasRtt = rtt !== undefined && rtt > 0;
addResult('connection_rtt', hasRtt, `${rtt || 0} ms`, 'warn');
// Test: Screen dimensions reasonable
const screenOk = screen.width > 0 && screen.height > 0 && screen.colorDepth >= 24;
addResult('screen_dimensions', screenOk,
`${screen.width}x${screen.height} @ ${screen.colorDepth}bit`, 'warn');
// Test: Notification permission queryable (not denied by default)
let notifOk = true;
try {
// In automation, this might throw or return unexpected values
notifOk = Notification.permission !== undefined;
} catch (e) { notifOk = false; }
addResult('notification_api', notifOk,
`Notification.permission = ${Notification.permission}`, 'warn');
// Test: Function.toString not modified (automation tools sometimes wrap functions)
let toStringOk = true;
try {
const fnStr = Function.prototype.toString.call(navigator.permissions.query);
toStringOk = fnStr.includes('[native code]');
} catch (e) { toStringOk = false; }
addResult('function_tostring', toStringOk,
toStringOk ? 'Native' : 'Modified', 'warn');
// βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
// INFO - Just for debugging
// βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
addInfo('User Agent', navigator.userAgent);
addInfo('Platform', navigator.platform);
addInfo('Vendor', navigator.vendor);
addInfo('Languages', navigator.languages?.join(', '));
addInfo('Timezone', Intl.DateTimeFormat().resolvedOptions().timeZone);
addInfo('Screen', `${screen.width}x${screen.height} @ ${screen.colorDepth}bit`);
addInfo('Window Inner', `${window.innerWidth}x${window.innerHeight}`);
addInfo('Window Outer', `${window.outerWidth}x${window.outerHeight}`);
addInfo('Device Pixel Ratio', window.devicePixelRatio);
addInfo('Cookies Enabled', navigator.cookieEnabled);
addInfo('Do Not Track', navigator.doNotTrack);
addInfo('WebGL Renderer', webglRenderer);
// βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
// SCORING
// βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
window.__botDetectResults = results;
const criticalTests = Object.values(results).filter(r => r.severity === 'critical');
const warnTests = Object.values(results).filter(r => r.severity === 'warn');
const criticalPassed = criticalTests.filter(r => r.passed).length;
const warnPassed = warnTests.filter(r => r.passed).length;
window.__botDetectScore = {
critical: criticalPassed,
criticalTotal: criticalTests.length,
warnings: warnPassed,
warningsTotal: warnTests.length,
total: criticalPassed + warnPassed,
totalTests: criticalTests.length + warnTests.length,
passed: criticalPassed === criticalTests.length
};
// Summary
const summaryDiv = document.createElement('div');
summaryDiv.style.cssText = 'margin: 20px 0; padding: 20px; border-radius: 8px; font-size: 18px;';
summaryDiv.style.background = window.__botDetectScore.passed ? '#1b5e20' : '#b71c1c';
summaryDiv.innerHTML = `
<strong>${window.__botDetectScore.passed ? 'β PASSED' : 'β FAILED'}</strong><br>
Critical: ${criticalPassed}/${criticalTests.length} |
Warnings: ${warnPassed}/${warnTests.length}
`;
document.body.insertBefore(summaryDiv, document.querySelector('h2'));
</script>
</body>
</html>
|