feat: 优先使用系统 Chrome 过 Vercel challenge,fallback 到 Playwright Chromium

系统 Chrome 指纹更真实,不易被 bot 检测拦截。
支持 macOS / Linux / Windows 自动检测 Chrome 路径。
Docker 环境无系统 Chrome 时自动回退到 Playwright Chromium。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
BaskDuan
2026-04-03 00:05:37 +08:00
parent 941f8e234a
commit 49de433965
2 changed files with 68 additions and 23 deletions

View File

@@ -29,6 +29,28 @@ const pendingRequests = new Map();
// ==================== 浏览器管理 ====================
const fs = require('fs');
// 自动检测系统 Chrome 路径(优先使用,指纹更真实)
function findSystemChrome() {
const paths = [
// macOS
'/Applications/Google Chrome.app/Contents/MacOS/Google Chrome',
// Linux
'/usr/bin/google-chrome',
'/usr/bin/google-chrome-stable',
'/usr/bin/chromium',
'/usr/bin/chromium-browser',
// Windows
'C:\\Program Files\\Google\\Chrome\\Application\\chrome.exe',
'C:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe',
];
for (const p of paths) {
if (fs.existsSync(p)) return p;
}
return null;
}
async function loadStealth() {
const { chromium } = require('playwright-extra');
const stealth = require('puppeteer-extra-plugin-stealth');
@@ -38,9 +60,9 @@ async function loadStealth() {
async function initBrowser() {
const chromium = await loadStealth();
const chromePath = findSystemChrome();
console.log('[Stealth] Launching browser...');
browser = await chromium.launch({
const launchOptions = {
headless: true,
args: [
'--no-sandbox',
@@ -48,7 +70,17 @@ async function initBrowser() {
'--disable-dev-shm-usage',
'--disable-gpu',
],
});
};
if (chromePath) {
launchOptions.executablePath = chromePath;
console.log(`[Stealth] Using system Chrome: ${chromePath}`);
} else {
console.log('[Stealth] System Chrome not found, using Playwright Chromium');
}
console.log('[Stealth] Launching browser...');
browser = await chromium.launch(launchOptions);
context = await browser.newContext({
userAgent:

View File

@@ -1,23 +1,29 @@
/**
* 测试 Vercel Challenge 耗时
* 用法: npm run test:stealth
* 优先使用系统 Chrome找不到则 fallback 到 Playwright Chromium
*/
const fs = require('fs');
const CHALLENGE_URL = process.env.CHALLENGE_URL || 'https://cursor.com/cn/docs';
async function simulateHuman(page) {
await new Promise(r => setTimeout(r, 500 + Math.random() * 1000));
const points = Array.from({ length: 5 }, () => ({
x: 100 + Math.random() * 600,
y: 100 + Math.random() * 400,
}));
for (const p of points) {
await page.mouse.move(p.x, p.y, { steps: 5 + Math.floor(Math.random() * 10) });
await new Promise(r => setTimeout(r, 100 + Math.random() * 300));
// 自动检测系统 Chrome 路径
function findSystemChrome() {
const paths = [
// macOS
'/Applications/Google Chrome.app/Contents/MacOS/Google Chrome',
// Linux
'/usr/bin/google-chrome',
'/usr/bin/google-chrome-stable',
'/usr/bin/chromium',
'/usr/bin/chromium-browser',
// Windows
'C:\\Program Files\\Google\\Chrome\\Application\\chrome.exe',
'C:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe',
];
for (const p of paths) {
if (fs.existsSync(p)) return p;
}
await page.mouse.wheel(0, 100 + Math.random() * 200);
await new Promise(r => setTimeout(r, 300 + Math.random() * 500));
await page.mouse.click(300 + Math.random() * 400, 300 + Math.random() * 200);
console.log(`[Test] Human behavior simulation done`);
return null;
}
(async () => {
@@ -28,11 +34,21 @@ async function simulateHuman(page) {
const start = Date.now();
const elapsed = () => ((Date.now() - start) / 1000).toFixed(1) + 's';
console.log(`[Test] Launching browser...`);
const browser = await chromium.launch({
const chromePath = findSystemChrome();
if (chromePath) {
console.log(`[Test] Using system Chrome: ${chromePath}`);
} else {
console.log(`[Test] System Chrome not found, using Playwright Chromium`);
}
const launchOptions = {
headless: true,
args: ['--no-sandbox', '--disable-setuid-sandbox', '--disable-dev-shm-usage'],
});
};
if (chromePath) launchOptions.executablePath = chromePath;
console.log(`[Test] Launching browser...`);
const browser = await chromium.launch(launchOptions);
const ctx = await browser.newContext({
userAgent: 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/146.0.0.0 Safari/537.36',
@@ -43,16 +59,13 @@ async function simulateHuman(page) {
try {
await page.goto(CHALLENGE_URL, { waitUntil: 'domcontentloaded', timeout: 120000 });
console.log(`[Test] [${elapsed()}] Page loaded, simulating human behavior...`);
console.log(`[Test] [${elapsed()}] Page loaded, waiting for _vcrcs cookie...`);
} catch (e) {
console.error(`[Test] [${elapsed()}] Page load failed: ${e.message}`);
await browser.close();
process.exit(1);
}
await simulateHuman(page);
console.log(`[Test] [${elapsed()}] Waiting for _vcrcs cookie...`);
for (let i = 0; i < 60; i++) {
const cookies = await ctx.cookies();
const vcrcs = cookies.find(c => c.name === '_vcrcs');