tt-help-cli-ycl 1.3.34 → 1.3.35
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +17 -17
- package/cli.js +9 -9
- package/package.json +47 -47
- package/scripts/run-explore copy.bat +101 -101
- package/scripts/run-explore.bat +132 -132
- package/scripts/run-explore.ps1 +157 -157
- package/scripts/run-explore.sh +119 -119
- package/scripts/test-captcha-lib.mjs +68 -0
- package/scripts/test-captcha.mjs +81 -0
- package/scripts/test-incognito-lib.mjs +36 -0
- package/scripts/test-login-state.mjs +128 -0
- package/scripts/test-safe-click.mjs +45 -0
- package/src/cli/attach.js +180 -180
- package/src/cli/auto.js +240 -240
- package/src/cli/config.js +152 -152
- package/src/cli/explore.js +488 -488
- package/src/cli/info.js +88 -88
- package/src/cli/open.js +111 -111
- package/src/cli/progress.js +111 -111
- package/src/cli/refresh.js +216 -216
- package/src/cli/scrape.js +47 -47
- package/src/cli/utils.js +18 -18
- package/src/cli/videos.js +41 -41
- package/src/cli/watch.js +31 -31
- package/src/lib/args.js +722 -722
- package/src/lib/browser/anti-detect.js +23 -23
- package/src/lib/browser/cdp.js +261 -261
- package/src/lib/browser/health-checker.js +114 -114
- package/src/lib/browser/launch.js +43 -43
- package/src/lib/browser/page.js +183 -183
- package/src/lib/constants.js +216 -216
- package/src/lib/delay.js +54 -54
- package/src/lib/explore-fetch.js +118 -118
- package/src/lib/fetcher.js +45 -45
- package/src/lib/filter.js +66 -66
- package/src/lib/io.js +54 -54
- package/src/lib/output.js +80 -80
- package/src/lib/page-error-detector.js +105 -105
- package/src/lib/parse-ssr.mjs +69 -69
- package/src/lib/parser.js +47 -47
- package/src/lib/retry.js +45 -45
- package/src/lib/scrape.js +89 -89
- package/src/lib/tiktok-scraper.mjs +194 -194
- package/src/lib/url.js +52 -52
- package/src/main.js +48 -48
- package/src/results/user-videos-bar.lar.lar.moeta.json +37 -0
- package/src/scraper/auto-core.js +203 -203
- package/src/scraper/core.js +211 -211
- package/src/scraper/explore-core.js +177 -167
- package/src/scraper/modules/captcha-handler.js +114 -114
- package/src/scraper/modules/follow-extractor.js +194 -194
- package/src/scraper/modules/guess-extractor.js +51 -51
- package/src/scraper/modules/page-helpers.js +48 -48
- package/src/scraper/refresh-core.js +179 -179
- package/src/videos/core.js +125 -125
- package/src/watch/data-store.js +1040 -1030
- package/src/watch/public/index.html +1458 -753
- package/src/watch/server.js +939 -933
|
@@ -1,167 +1,177 @@
|
|
|
1
|
-
import { ensureBrowserReady, delay } from "./modules/page-helpers.js";
|
|
2
|
-
import { detectCaptcha } from "./modules/captcha-handler.js";
|
|
3
|
-
export { ensureBrowserReady };
|
|
4
|
-
import { getUserInfo, collectVideos } from "../videos/core.js";
|
|
5
|
-
import { extractFollowAndFollowers } from "./modules/follow-extractor.js";
|
|
6
|
-
import { extractVideoLocation } from "../lib/scrape.js";
|
|
7
|
-
import {
|
|
8
|
-
maxFollowing as globalMaxFollowing,
|
|
9
|
-
maxFollowers as globalMaxFollowers,
|
|
10
|
-
} from "../lib/constants.js";
|
|
11
|
-
|
|
12
|
-
async function processExplore(page, username, options, log) {
|
|
13
|
-
const {
|
|
14
|
-
maxVideos = 16,
|
|
15
|
-
enableFollow = true,
|
|
16
|
-
loggedIn = false, // 由外部传入登录状态,避免每次调用 isLoggedIn(page)
|
|
17
|
-
maxFollowing = 50,
|
|
18
|
-
maxFollowers = 50,
|
|
19
|
-
location = "PL,NL,BE,DE,FR,IT,ES,IE",
|
|
20
|
-
} = options;
|
|
21
|
-
|
|
22
|
-
const result = {
|
|
23
|
-
userInfo: null,
|
|
24
|
-
discoveredVideoAuthors: [],
|
|
25
|
-
discoveredCommentAuthors: [],
|
|
26
|
-
discoveredGuessAuthors: [],
|
|
27
|
-
discoveredFollowing: [],
|
|
28
|
-
discoveredFollowers: [],
|
|
29
|
-
collectedVideos: 0,
|
|
30
|
-
processed: false,
|
|
31
|
-
hasFollowData: false,
|
|
32
|
-
keepFollow: false,
|
|
33
|
-
locationCreated: null,
|
|
34
|
-
noVideo: false,
|
|
35
|
-
error: null,
|
|
36
|
-
};
|
|
37
|
-
|
|
38
|
-
try {
|
|
39
|
-
log(` 访问 @${username} 主页...`);
|
|
40
|
-
const videoList = await collectVideos(page, username, maxVideos, log);
|
|
41
|
-
|
|
42
|
-
log(" 获取用户信息...");
|
|
43
|
-
const info = await getUserInfo(page);
|
|
44
|
-
if (info) {
|
|
45
|
-
result.userInfo = info;
|
|
46
|
-
log(
|
|
47
|
-
` 用户: ${info.nickname || username} | 粉丝: ${info.followerCount || "-"} | 视频: ${info.videoCount || "-"}`,
|
|
48
|
-
);
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
const captcha = await detectCaptcha(page);
|
|
52
|
-
if (captcha && captcha.visible) {
|
|
53
|
-
log(`[验证码] @${username} 页面出现验证码`);
|
|
54
|
-
result.captchaDetected = true;
|
|
55
|
-
result.captchaStage = result.captchaStage || "video-page";
|
|
56
|
-
result.captchaMessage = result.captchaMessage || "视频页出现验证码";
|
|
57
|
-
}
|
|
58
|
-
const videoArray = videoList ? [...videoList.values()] : [];
|
|
59
|
-
result.collectedVideos = videoArray.length;
|
|
60
|
-
|
|
61
|
-
if (videoArray.length <= 0) {
|
|
62
|
-
// 视频为空:可能是页面受限或用户真的没有视频
|
|
63
|
-
result.processed = true;
|
|
64
|
-
result.noVideo = true;
|
|
65
|
-
log(` @${username} 没有视频,标记已处理`);
|
|
66
|
-
return result;
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
//
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
: `https://www.tiktok.com${
|
|
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
|
-
const
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
result.
|
|
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
|
-
|
|
1
|
+
import { ensureBrowserReady, delay } from "./modules/page-helpers.js";
|
|
2
|
+
import { detectCaptcha } from "./modules/captcha-handler.js";
|
|
3
|
+
export { ensureBrowserReady };
|
|
4
|
+
import { getUserInfo, collectVideos } from "../videos/core.js";
|
|
5
|
+
import { extractFollowAndFollowers } from "./modules/follow-extractor.js";
|
|
6
|
+
import { extractVideoLocation } from "../lib/scrape.js";
|
|
7
|
+
import {
|
|
8
|
+
maxFollowing as globalMaxFollowing,
|
|
9
|
+
maxFollowers as globalMaxFollowers,
|
|
10
|
+
} from "../lib/constants.js";
|
|
11
|
+
|
|
12
|
+
async function processExplore(page, username, options, log) {
|
|
13
|
+
const {
|
|
14
|
+
maxVideos = 16,
|
|
15
|
+
enableFollow = true,
|
|
16
|
+
loggedIn = false, // 由外部传入登录状态,避免每次调用 isLoggedIn(page)
|
|
17
|
+
maxFollowing = 50,
|
|
18
|
+
maxFollowers = 50,
|
|
19
|
+
location = "PL,NL,BE,DE,FR,IT,ES,IE",
|
|
20
|
+
} = options;
|
|
21
|
+
|
|
22
|
+
const result = {
|
|
23
|
+
userInfo: null,
|
|
24
|
+
discoveredVideoAuthors: [],
|
|
25
|
+
discoveredCommentAuthors: [],
|
|
26
|
+
discoveredGuessAuthors: [],
|
|
27
|
+
discoveredFollowing: [],
|
|
28
|
+
discoveredFollowers: [],
|
|
29
|
+
collectedVideos: 0,
|
|
30
|
+
processed: false,
|
|
31
|
+
hasFollowData: false,
|
|
32
|
+
keepFollow: false,
|
|
33
|
+
locationCreated: null,
|
|
34
|
+
noVideo: false,
|
|
35
|
+
error: null,
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
try {
|
|
39
|
+
log(` 访问 @${username} 主页...`);
|
|
40
|
+
const videoList = await collectVideos(page, username, maxVideos, log);
|
|
41
|
+
|
|
42
|
+
log(" 获取用户信息...");
|
|
43
|
+
const info = await getUserInfo(page);
|
|
44
|
+
if (info) {
|
|
45
|
+
result.userInfo = info;
|
|
46
|
+
log(
|
|
47
|
+
` 用户: ${info.nickname || username} | 粉丝: ${info.followerCount || "-"} | 视频: ${info.videoCount || "-"}`,
|
|
48
|
+
);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
const captcha = await detectCaptcha(page);
|
|
52
|
+
if (captcha && captcha.visible) {
|
|
53
|
+
log(`[验证码] @${username} 页面出现验证码`);
|
|
54
|
+
result.captchaDetected = true;
|
|
55
|
+
result.captchaStage = result.captchaStage || "video-page";
|
|
56
|
+
result.captchaMessage = result.captchaMessage || "视频页出现验证码";
|
|
57
|
+
}
|
|
58
|
+
const videoArray = videoList ? [...videoList.values()] : [];
|
|
59
|
+
result.collectedVideos = videoArray.length;
|
|
60
|
+
|
|
61
|
+
if (videoArray.length <= 0) {
|
|
62
|
+
// 视频为空:可能是页面受限或用户真的没有视频
|
|
63
|
+
result.processed = true;
|
|
64
|
+
result.noVideo = true;
|
|
65
|
+
log(` @${username} 没有视频,标记已处理`);
|
|
66
|
+
return result;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// 从最多 5 个视频并发获取 locationCreated,取众数
|
|
70
|
+
const SAMPLE_SIZE = 5;
|
|
71
|
+
let locationCreated = null;
|
|
72
|
+
const sampleVideos = videoArray.slice(0, SAMPLE_SIZE);
|
|
73
|
+
if (sampleVideos.length > 0) {
|
|
74
|
+
const sampleUrls = sampleVideos.map(v =>
|
|
75
|
+
v.href.startsWith("http") ? v.href : `https://www.tiktok.com${v.href}`
|
|
76
|
+
);
|
|
77
|
+
const locations = await Promise.all(sampleUrls.map(url => extractVideoLocation(url)));
|
|
78
|
+
log(` 国家采样(${locations.length}个): [${locations.filter(Boolean).join(", ") || "无数据"}]`);
|
|
79
|
+
const freq = {};
|
|
80
|
+
for (const loc of locations) {
|
|
81
|
+
if (loc) {
|
|
82
|
+
const key = loc.toUpperCase();
|
|
83
|
+
freq[key] = (freq[key] || 0) + 1;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
const entries = Object.entries(freq).sort((a, b) => b[1] - a[1]);
|
|
87
|
+
locationCreated = entries.length > 0 ? entries[0][0] : null;
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
result.locationCreated = locationCreated || null;
|
|
91
|
+
log(` 国家: ${result.locationCreated || "未知"} (众数)`);
|
|
92
|
+
|
|
93
|
+
// 国家筛选
|
|
94
|
+
const locationList = (location || "ES")
|
|
95
|
+
.split(",")
|
|
96
|
+
.map((s) => s.trim().toUpperCase());
|
|
97
|
+
const isTargetLocation = locationList.includes(
|
|
98
|
+
result.locationCreated?.toUpperCase?.() || result.locationCreated,
|
|
99
|
+
);
|
|
100
|
+
|
|
101
|
+
if (isTargetLocation) {
|
|
102
|
+
result.keepFollow = true;
|
|
103
|
+
log(` 国家匹配,获取关注/粉丝...`);
|
|
104
|
+
|
|
105
|
+
// 提取关注/粉丝
|
|
106
|
+
if (enableFollow) {
|
|
107
|
+
await delay(100, 1000);
|
|
108
|
+
if (!loggedIn) {
|
|
109
|
+
log(" [跳过] 获取关注/粉丝:未登录,请先登录 TikTok");
|
|
110
|
+
result.hasFollowData = false;
|
|
111
|
+
result.discoveredFollowing = [];
|
|
112
|
+
result.discoveredFollowers = [];
|
|
113
|
+
} else {
|
|
114
|
+
try {
|
|
115
|
+
const isSeller = result.userInfo?.ttSeller === true;
|
|
116
|
+
const effectiveMaxFollowing = isSeller
|
|
117
|
+
? globalMaxFollowing
|
|
118
|
+
: maxFollowing;
|
|
119
|
+
const effectiveMaxFollowers = isSeller
|
|
120
|
+
? globalMaxFollowers
|
|
121
|
+
: maxFollowers;
|
|
122
|
+
if (isSeller)
|
|
123
|
+
log(
|
|
124
|
+
` 商家用户,关注采集: ${effectiveMaxFollowing}, 粉丝采集: ${effectiveMaxFollowers}`,
|
|
125
|
+
);
|
|
126
|
+
const { following, followers } = await extractFollowAndFollowers(
|
|
127
|
+
page,
|
|
128
|
+
{
|
|
129
|
+
maxFollowing: effectiveMaxFollowing,
|
|
130
|
+
maxFollowers: effectiveMaxFollowers,
|
|
131
|
+
log,
|
|
132
|
+
},
|
|
133
|
+
);
|
|
134
|
+
result.discoveredFollowing = following || [];
|
|
135
|
+
result.discoveredFollowers = followers || [];
|
|
136
|
+
result.hasFollowData = true;
|
|
137
|
+
log(
|
|
138
|
+
` 关注: ${result.discoveredFollowing.length}, 粉丝: ${result.discoveredFollowers.length}`,
|
|
139
|
+
);
|
|
140
|
+
} catch (e) {
|
|
141
|
+
log(` 关注/粉丝提取失败: ${e.message}`);
|
|
142
|
+
result.hasFollowData = false;
|
|
143
|
+
result.discoveredFollowing = [];
|
|
144
|
+
result.discoveredFollowers = [];
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// 携带视频列表供登记
|
|
150
|
+
result.videoList = videoArray;
|
|
151
|
+
} else {
|
|
152
|
+
// 国家不匹配
|
|
153
|
+
result.keepFollow = false;
|
|
154
|
+
result.discoveredFollowing = [];
|
|
155
|
+
result.discoveredFollowers = [];
|
|
156
|
+
result.hasFollowData = false;
|
|
157
|
+
log(` 国家不匹配,跳过`);
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
result.processed = true;
|
|
161
|
+
} catch (e) {
|
|
162
|
+
result.error = e.message;
|
|
163
|
+
result.errorStack = e.stack || "";
|
|
164
|
+
log(` [错误] ${e.message}`);
|
|
165
|
+
|
|
166
|
+
// 被封会抛出 "被封: username" 异常
|
|
167
|
+
if (e.message.startsWith("被封:")) {
|
|
168
|
+
result.processed = false;
|
|
169
|
+
result.noVideo = false;
|
|
170
|
+
log(` @${username} 认定为被封`);
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
return result;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
export { processExplore };
|
|
@@ -1,114 +1,114 @@
|
|
|
1
|
-
export async function detectCaptcha(page) {
|
|
2
|
-
return page.evaluate(() => {
|
|
3
|
-
const container = document.querySelector('.captcha-verify-container');
|
|
4
|
-
if (!container) return null;
|
|
5
|
-
|
|
6
|
-
const r = container.getBoundingClientRect();
|
|
7
|
-
return {
|
|
8
|
-
exists: true,
|
|
9
|
-
visible: container.offsetParent !== null,
|
|
10
|
-
rect: {
|
|
11
|
-
x: Math.round(r.x),
|
|
12
|
-
y: Math.round(r.y),
|
|
13
|
-
w: Math.round(r.width),
|
|
14
|
-
h: Math.round(r.height),
|
|
15
|
-
},
|
|
16
|
-
};
|
|
17
|
-
});
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
export async function closeCaptcha(page) {
|
|
21
|
-
return page.evaluate(() => {
|
|
22
|
-
const closeBtn = document.getElementById('captcha_close_button');
|
|
23
|
-
if (!closeBtn) return { success: false, reason: 'close button not found' };
|
|
24
|
-
|
|
25
|
-
closeBtn.dispatchEvent(new MouseEvent('mousedown', { bubbles: true }));
|
|
26
|
-
closeBtn.dispatchEvent(new MouseEvent('mouseup', { bubbles: true }));
|
|
27
|
-
closeBtn.dispatchEvent(new MouseEvent('click', { bubbles: true }));
|
|
28
|
-
|
|
29
|
-
return { success: true };
|
|
30
|
-
});
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
export async function handleCaptcha(page, options = {}) {
|
|
34
|
-
const { waitMs = 2000 } = options;
|
|
35
|
-
|
|
36
|
-
const captcha = await detectCaptcha(page);
|
|
37
|
-
if (!captcha) return { detected: false, closed: false };
|
|
38
|
-
|
|
39
|
-
await new Promise(r => setTimeout(r, waitMs));
|
|
40
|
-
|
|
41
|
-
const result = await closeCaptcha(page);
|
|
42
|
-
if (!result.success) return { detected: true, closed: false, reason: result.reason };
|
|
43
|
-
|
|
44
|
-
await new Promise(r => setTimeout(r, 1000));
|
|
45
|
-
|
|
46
|
-
const stillThere = await detectCaptcha(page);
|
|
47
|
-
return { detected: true, closed: !stillThere };
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
export async function getIncognitoPage(browser, url, options = {}) {
|
|
51
|
-
const { waitMs = 3000 } = options;
|
|
52
|
-
const context = await browser.newContext();
|
|
53
|
-
const page = await context.newPage();
|
|
54
|
-
await page.goto(url, { waitUntil: 'domcontentloaded', timeout: 30000 });
|
|
55
|
-
await new Promise(r => setTimeout(r, waitMs));
|
|
56
|
-
return { page, context };
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
export async function waitAndGetCaptcha(page, options = {}) {
|
|
60
|
-
const { waitMs = 180000, pollInterval = 5000, log } = options;
|
|
61
|
-
|
|
62
|
-
const captcha = await detectCaptcha(page);
|
|
63
|
-
if (!captcha) return { detected: false, resolved: false, waited: 0 };
|
|
64
|
-
|
|
65
|
-
if (log) log(' 检测到验证码,等待用户手动输入...');
|
|
66
|
-
|
|
67
|
-
const startTime = Date.now();
|
|
68
|
-
const deadline = startTime + waitMs;
|
|
69
|
-
|
|
70
|
-
while (Date.now() < deadline) {
|
|
71
|
-
await new Promise(r => setTimeout(r, pollInterval));
|
|
72
|
-
const remaining = await detectCaptcha(page);
|
|
73
|
-
if (!remaining) {
|
|
74
|
-
const waited = Math.round((Date.now() - startTime) / 1000);
|
|
75
|
-
if (log) log(` 验证码已解决(等待 ${waited}s)`);
|
|
76
|
-
return { detected: true, resolved: true, waited };
|
|
77
|
-
}
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
const waited = Math.round(waitMs / 1000);
|
|
81
|
-
if (log) log(` 验证码等待超时(${waited}s),继续执行`);
|
|
82
|
-
|
|
83
|
-
// 超时后尝试关闭验证码弹窗
|
|
84
|
-
await closeCaptcha(page);
|
|
85
|
-
await new Promise(r => setTimeout(r, 1000));
|
|
86
|
-
|
|
87
|
-
return { detected: true, resolved: false, waited };
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
export async function safeClickComment(page, options = {}) {
|
|
91
|
-
const { waitMs = 3000 } = options;
|
|
92
|
-
|
|
93
|
-
// 点击评论
|
|
94
|
-
await page.evaluate(() => {
|
|
95
|
-
const all = document.querySelectorAll('button');
|
|
96
|
-
for (const el of all) {
|
|
97
|
-
if (/^评论$/.test(el.textContent?.trim()) && el.offsetParent !== null && el.getBoundingClientRect().width > 0) {
|
|
98
|
-
el.click();
|
|
99
|
-
break;
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
});
|
|
103
|
-
|
|
104
|
-
await new Promise(r => setTimeout(r, waitMs));
|
|
105
|
-
|
|
106
|
-
// 检测并关闭验证码
|
|
107
|
-
const captcha = await detectCaptcha(page);
|
|
108
|
-
if (captcha) {
|
|
109
|
-
const result = await handleCaptcha(page);
|
|
110
|
-
return { clicked: true, captchaDetected: true, captchaClosed: result.closed };
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
return { clicked: true, captchaDetected: false, captchaClosed: false };
|
|
114
|
-
}
|
|
1
|
+
export async function detectCaptcha(page) {
|
|
2
|
+
return page.evaluate(() => {
|
|
3
|
+
const container = document.querySelector('.captcha-verify-container');
|
|
4
|
+
if (!container) return null;
|
|
5
|
+
|
|
6
|
+
const r = container.getBoundingClientRect();
|
|
7
|
+
return {
|
|
8
|
+
exists: true,
|
|
9
|
+
visible: container.offsetParent !== null,
|
|
10
|
+
rect: {
|
|
11
|
+
x: Math.round(r.x),
|
|
12
|
+
y: Math.round(r.y),
|
|
13
|
+
w: Math.round(r.width),
|
|
14
|
+
h: Math.round(r.height),
|
|
15
|
+
},
|
|
16
|
+
};
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export async function closeCaptcha(page) {
|
|
21
|
+
return page.evaluate(() => {
|
|
22
|
+
const closeBtn = document.getElementById('captcha_close_button');
|
|
23
|
+
if (!closeBtn) return { success: false, reason: 'close button not found' };
|
|
24
|
+
|
|
25
|
+
closeBtn.dispatchEvent(new MouseEvent('mousedown', { bubbles: true }));
|
|
26
|
+
closeBtn.dispatchEvent(new MouseEvent('mouseup', { bubbles: true }));
|
|
27
|
+
closeBtn.dispatchEvent(new MouseEvent('click', { bubbles: true }));
|
|
28
|
+
|
|
29
|
+
return { success: true };
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export async function handleCaptcha(page, options = {}) {
|
|
34
|
+
const { waitMs = 2000 } = options;
|
|
35
|
+
|
|
36
|
+
const captcha = await detectCaptcha(page);
|
|
37
|
+
if (!captcha) return { detected: false, closed: false };
|
|
38
|
+
|
|
39
|
+
await new Promise(r => setTimeout(r, waitMs));
|
|
40
|
+
|
|
41
|
+
const result = await closeCaptcha(page);
|
|
42
|
+
if (!result.success) return { detected: true, closed: false, reason: result.reason };
|
|
43
|
+
|
|
44
|
+
await new Promise(r => setTimeout(r, 1000));
|
|
45
|
+
|
|
46
|
+
const stillThere = await detectCaptcha(page);
|
|
47
|
+
return { detected: true, closed: !stillThere };
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export async function getIncognitoPage(browser, url, options = {}) {
|
|
51
|
+
const { waitMs = 3000 } = options;
|
|
52
|
+
const context = await browser.newContext();
|
|
53
|
+
const page = await context.newPage();
|
|
54
|
+
await page.goto(url, { waitUntil: 'domcontentloaded', timeout: 30000 });
|
|
55
|
+
await new Promise(r => setTimeout(r, waitMs));
|
|
56
|
+
return { page, context };
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
export async function waitAndGetCaptcha(page, options = {}) {
|
|
60
|
+
const { waitMs = 180000, pollInterval = 5000, log } = options;
|
|
61
|
+
|
|
62
|
+
const captcha = await detectCaptcha(page);
|
|
63
|
+
if (!captcha) return { detected: false, resolved: false, waited: 0 };
|
|
64
|
+
|
|
65
|
+
if (log) log(' 检测到验证码,等待用户手动输入...');
|
|
66
|
+
|
|
67
|
+
const startTime = Date.now();
|
|
68
|
+
const deadline = startTime + waitMs;
|
|
69
|
+
|
|
70
|
+
while (Date.now() < deadline) {
|
|
71
|
+
await new Promise(r => setTimeout(r, pollInterval));
|
|
72
|
+
const remaining = await detectCaptcha(page);
|
|
73
|
+
if (!remaining) {
|
|
74
|
+
const waited = Math.round((Date.now() - startTime) / 1000);
|
|
75
|
+
if (log) log(` 验证码已解决(等待 ${waited}s)`);
|
|
76
|
+
return { detected: true, resolved: true, waited };
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
const waited = Math.round(waitMs / 1000);
|
|
81
|
+
if (log) log(` 验证码等待超时(${waited}s),继续执行`);
|
|
82
|
+
|
|
83
|
+
// 超时后尝试关闭验证码弹窗
|
|
84
|
+
await closeCaptcha(page);
|
|
85
|
+
await new Promise(r => setTimeout(r, 1000));
|
|
86
|
+
|
|
87
|
+
return { detected: true, resolved: false, waited };
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
export async function safeClickComment(page, options = {}) {
|
|
91
|
+
const { waitMs = 3000 } = options;
|
|
92
|
+
|
|
93
|
+
// 点击评论
|
|
94
|
+
await page.evaluate(() => {
|
|
95
|
+
const all = document.querySelectorAll('button');
|
|
96
|
+
for (const el of all) {
|
|
97
|
+
if (/^评论$/.test(el.textContent?.trim()) && el.offsetParent !== null && el.getBoundingClientRect().width > 0) {
|
|
98
|
+
el.click();
|
|
99
|
+
break;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
await new Promise(r => setTimeout(r, waitMs));
|
|
105
|
+
|
|
106
|
+
// 检测并关闭验证码
|
|
107
|
+
const captcha = await detectCaptcha(page);
|
|
108
|
+
if (captcha) {
|
|
109
|
+
const result = await handleCaptcha(page);
|
|
110
|
+
return { clicked: true, captchaDetected: true, captchaClosed: result.closed };
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
return { clicked: true, captchaDetected: false, captchaClosed: false };
|
|
114
|
+
}
|