tt-help-cli-ycl 1.3.12 → 1.3.14

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.
Files changed (55) hide show
  1. package/README.md +17 -17
  2. package/cli.js +9 -9
  3. package/package.json +47 -45
  4. package/scripts/run-explore.bat +68 -68
  5. package/scripts/run-explore.ps1 +81 -81
  6. package/scripts/run-explore.sh +73 -73
  7. package/scripts/test-captcha-lib.mjs +68 -0
  8. package/scripts/test-captcha.mjs +81 -0
  9. package/scripts/test-incognito-lib.mjs +36 -0
  10. package/scripts/test-login-state.mjs +128 -0
  11. package/scripts/test-safe-click.mjs +45 -0
  12. package/src/cli/attach.js +160 -0
  13. package/src/cli/auto.js +186 -157
  14. package/src/cli/config.js +39 -3
  15. package/src/cli/explore.js +234 -193
  16. package/src/cli/info.js +88 -0
  17. package/src/cli/progress.js +111 -111
  18. package/src/cli/refresh.js +216 -0
  19. package/src/cli/scrape.js +47 -47
  20. package/src/cli/utils.js +18 -18
  21. package/src/cli/videos.js +41 -41
  22. package/src/cli/watch.js +31 -31
  23. package/src/lib/args.js +517 -402
  24. package/src/lib/browser/anti-detect.js +23 -23
  25. package/src/lib/browser/cdp.js +52 -10
  26. package/src/lib/browser/launch.js +43 -43
  27. package/src/lib/browser/page.js +146 -87
  28. package/src/lib/constants.js +199 -115
  29. package/src/lib/delay.js +54 -54
  30. package/src/lib/explore-fetch.js +118 -118
  31. package/src/lib/fetcher.js +45 -45
  32. package/src/lib/filter.js +66 -66
  33. package/src/lib/io.js +54 -54
  34. package/src/lib/output.js +80 -80
  35. package/src/lib/parse-ssr.mjs +69 -0
  36. package/src/lib/parser.js +47 -47
  37. package/src/lib/retry.js +45 -45
  38. package/src/lib/scrape.js +89 -40
  39. package/src/lib/tiktok-scraper.mjs +176 -0
  40. package/src/lib/url.js +52 -52
  41. package/src/main.js +12 -16
  42. package/src/results/user-videos-bar.lar.lar.moeta.json +37 -0
  43. package/src/scraper/auto-core.js +203 -194
  44. package/src/scraper/core.js +211 -190
  45. package/src/scraper/explore-core.js +162 -171
  46. package/src/scraper/modules/captcha-handler.js +114 -114
  47. package/src/scraper/modules/comment-extractor.js +74 -69
  48. package/src/scraper/modules/follow-extractor.js +121 -121
  49. package/src/scraper/modules/guess-extractor.js +51 -51
  50. package/src/scraper/modules/page-helpers.js +48 -48
  51. package/src/scraper/refresh-core.js +179 -0
  52. package/src/videos/core.js +126 -126
  53. package/src/watch/data-store.js +536 -302
  54. package/src/watch/public/index.html +721 -701
  55. package/src/watch/server.js +527 -359
@@ -1,171 +1,162 @@
1
- import {
2
- delay,
3
- ensureBrowserReady,
4
- setDelayConfig,
5
- closeCommentPanel,
6
- retryWithBackoff,
7
- detectPageError,
8
- isLoggedIn,
9
- assertPageUrl,
10
- } from './modules/page-helpers.js';
11
- import { detectCaptcha } from './modules/captcha-handler.js';
12
- export { ensureBrowserReady };
13
- import {
14
- getUserInfo,
15
- collectVideos,
16
- } from '../videos/core.js';
17
- import { scrapeSingleVideo } from './core.js';
18
- import { extractFollowAndFollowers } from './modules/follow-extractor.js';
19
- import { extractCommentAuthors } from './modules/comment-extractor.js';
20
- import { extractGuessVideos } from './modules/guess-extractor.js';
21
-
22
- async function processExplore(page, username, options, log) {
23
- const {
24
- maxComments = 0,
25
- maxGuess = 0,
26
- enableFollow = true,
27
- maxFollowing = 5,
28
- maxFollowers = 5,
29
- location = 'ES',
30
- } = options;
31
-
32
- const result = {
33
- userInfo: null,
34
- discoveredVideoAuthors: [],
35
- discoveredCommentAuthors: [],
36
- discoveredGuessAuthors: [],
37
- discoveredFollowing: [],
38
- discoveredFollowers: [],
39
- collectedVideos: 0,
40
- processed: false,
41
- hasFollowData: false,
42
- keepFollow: false,
43
- locationCreated: null,
44
- noVideo: false,
45
- error: null,
46
- };
47
-
48
- try {
49
- log(` 访问 @${username} 主页...`);
50
- const homeUrl = `https://www.tiktok.com/@${username}`;
51
- await retryWithBackoff(async () => {
52
- await page.goto(homeUrl, { waitUntil: 'domcontentloaded', timeout: 30000 });
53
- assertPageUrl(page, `@${username}`);
54
- }, { log });
55
- await page.waitForSelector('[class*="DivVideoList"]', { timeout: 10000 }).catch(() => {});
56
- await delay(1000, 2000);
57
-
58
- log(' 获取用户信息...');
59
- const info = await getUserInfo(page);
60
- if (info) {
61
- result.userInfo = info;
62
- log(` 用户: ${info.nickname || username} | 粉丝: ${info.followerCount || '-'} | 视频: ${info.videoCount || '-'}`);
63
- }
64
-
65
- const captcha = await detectCaptcha(page);
66
- if (captcha && captcha.visible) {
67
- log(`[验证码] @${username} 页面出现验证码`);
68
- result.captchaDetected = true;
69
- }
70
-
71
- const videoList = await collectVideos(page, username, 1, log);
72
- const videoArray = videoList ? [...videoList.values()] : [];
73
- result.collectedVideos = videoArray.length;
74
-
75
- if (videoArray.length <= 0) {
76
- result.processed = true;
77
- result.noVideo = true;
78
- const pageError = await detectPageError(page);
79
- if (pageError) {
80
- result.restricted = true;
81
- log(` @${username} 页面受限(${pageError}),标记跳过`);
82
- } else {
83
- log(` @${username} 没有视频,标记已处理`);
84
- }
85
- return result;
86
- }
87
-
88
- if (enableFollow) {
89
- const loggedIn = await isLoggedIn(page);
90
- if (!loggedIn) {
91
- log(' [跳过] 获取关注/粉丝:未登录,请先登录 TikTok');
92
- result.hasFollowData = false;
93
- result.discoveredFollowing = [];
94
- result.discoveredFollowers = [];
95
- } else {
96
- try {
97
- log(' 获取关注/粉丝...');
98
- const { following, followers } = await extractFollowAndFollowers(
99
- page, { maxFollowing, maxFollowers, log }
100
- );
101
- result.discoveredFollowing = following || [];
102
- result.discoveredFollowers = followers || [];
103
- result.hasFollowData = true;
104
- log(` 关注: ${result.discoveredFollowing.length}, 粉丝: ${result.discoveredFollowers.length}`);
105
- } catch (e) {
106
- log(` 关注/粉丝提取失败: ${e.message}`);
107
- result.hasFollowData = false;
108
- result.discoveredFollowing = [];
109
- result.discoveredFollowers = [];
110
- }
111
- }
112
- }
113
-
114
- const firstVideo = videoArray[0];
115
- const videoUrl = firstVideo.href.startsWith('http')
116
- ? firstVideo.href
117
- : `https://www.tiktok.com${firstVideo.href}`;
118
-
119
- log(` 进入第一个视频: ${videoUrl}`);
120
- await retryWithBackoff(() => page.goto(videoUrl, { waitUntil: 'domcontentloaded', timeout: 30000 }), { log });
121
- assertPageUrl(page, videoUrl.split('/video/')[0]);
122
- await delay(1500, 2500);
123
-
124
- const videoData = await scrapeSingleVideo(page, 0, 0, log, 'NEVER_MATCH');
125
- result.locationCreated = videoData.locationCreated || null;
126
- log(` 视频作者: ${videoData.videoAuthor} | 国家: ${result.locationCreated || '未知'}`);
127
-
128
- const isTargetLocation = result.locationCreated === location;
129
-
130
- if (isTargetLocation) {
131
- result.keepFollow = true;
132
- log(` 国家匹配 (${location}),获取评论和猜你喜欢...`);
133
-
134
- if (maxComments > 0) {
135
- const commentResult = await extractCommentAuthors(page, maxComments);
136
- result.discoveredCommentAuthors = commentResult || [];
137
- await closeCommentPanel(page);
138
- await delay(500, 1000);
139
- log(` 评论用户: ${result.discoveredCommentAuthors.length}`);
140
- }
141
-
142
- if (maxGuess > 0) {
143
- const guessResult = await extractGuessVideos(page, maxGuess);
144
- result.discoveredGuessAuthors = (guessResult || []).map(v => v.author).filter(Boolean);
145
- await closeCommentPanel(page);
146
- await delay(500, 1000);
147
- log(` 猜你喜欢作者: ${result.discoveredGuessAuthors.length}`);
148
- }
149
-
150
- result.discoveredVideoAuthors = [{
151
- uniqueId: videoData.uniqueId,
152
- nickname: videoData.nickname,
153
- locationCreated: videoData.locationCreated,
154
- }];
155
- } else {
156
- result.keepFollow = false;
157
- log(` 国家不匹配 (${result.locationCreated} !== ${location}),跳过评论/猜你喜欢,丢弃关注/粉丝`);
158
- result.discoveredFollowing = [];
159
- result.discoveredFollowers = [];
160
- }
161
-
162
- result.processed = true;
163
- } catch (e) {
164
- result.error = e.message;
165
- log(` [错误] ${e.message}`);
166
- }
167
-
168
- return result;
169
- }
170
-
171
- export { processExplore };
1
+ import {
2
+ delay,
3
+ ensureBrowserReady,
4
+ retryWithBackoff,
5
+ detectPageError,
6
+ isLoggedIn,
7
+ assertPageUrl,
8
+ } from './modules/page-helpers.js';
9
+ import { detectCaptcha } from './modules/captcha-handler.js';
10
+ export { ensureBrowserReady };
11
+ import { getUserInfo, collectVideos } from '../videos/core.js';
12
+ import { extractFollowAndFollowers } from './modules/follow-extractor.js';
13
+ import { extractVideoLocation } from '../lib/scrape.js';
14
+ import { maxFollowing as globalMaxFollowing, maxFollowers as globalMaxFollowers, maxVideos as globalMaxVideos } from '../lib/constants.js';
15
+
16
+ async function processExplore(page, username, options, log) {
17
+ const {
18
+ maxVideos = 1,
19
+ enableFollow = true,
20
+ maxFollowing = 5,
21
+ maxFollowers = 5,
22
+ location = 'PL,NL,BE,DE,FR,IT,ES,IE',
23
+ } = options;
24
+
25
+ const result = {
26
+ userInfo: null,
27
+ discoveredVideoAuthors: [],
28
+ discoveredCommentAuthors: [],
29
+ discoveredGuessAuthors: [],
30
+ discoveredFollowing: [],
31
+ discoveredFollowers: [],
32
+ collectedVideos: 0,
33
+ processed: false,
34
+ hasFollowData: false,
35
+ keepFollow: false,
36
+ locationCreated: null,
37
+ noVideo: false,
38
+ error: null,
39
+ };
40
+
41
+ try {
42
+ log(` 访问 @${username} 主页...`);
43
+ const homeUrl = `https://www.tiktok.com/@${username}`;
44
+ await retryWithBackoff(async () => {
45
+ await page.goto(homeUrl, { waitUntil: 'domcontentloaded', timeout: 30000 });
46
+ assertPageUrl(page, `@${username}`);
47
+ }, { log });
48
+ await page.waitForSelector('[class*="DivVideoList"]', { timeout: 10000 }).catch(() => {});
49
+ await delay(1000, 2000);
50
+
51
+ log(' 获取用户信息...');
52
+ const info = await getUserInfo(page);
53
+ if (info) {
54
+ result.userInfo = info;
55
+ log(` 用户: ${info.nickname || username} | 粉丝: ${info.followerCount || '-'} | 视频: ${info.videoCount || '-'}`);
56
+ }
57
+
58
+ const captcha = await detectCaptcha(page);
59
+ if (captcha && captcha.visible) {
60
+ log(`[验证码] @${username} 页面出现验证码`);
61
+ result.captchaDetected = true;
62
+ result.captchaStage = result.captchaStage || 'video-page';
63
+ result.captchaMessage = result.captchaMessage || '视频页出现验证码';
64
+ }
65
+
66
+ const isSeller = result.userInfo?.ttSeller === true;
67
+ const effectiveMaxVideos = isSeller ? globalMaxVideos : maxVideos;
68
+ if (isSeller) log(` 商家用户,视频采集数: ${effectiveMaxVideos}`);
69
+ const videoList = await collectVideos(page, username, effectiveMaxVideos, log);
70
+ const videoArray = videoList ? [...videoList.values()] : [];
71
+ result.collectedVideos = videoArray.length;
72
+
73
+ if (videoArray.length <= 0) {
74
+ result.processed = true;
75
+ result.noVideo = true;
76
+ const pageError = await detectPageError(page);
77
+ if (pageError) {
78
+ result.restricted = true;
79
+ log(` @${username} 页面受限(${pageError}),标记跳过`);
80
+ } else {
81
+ log(` @${username} 没有视频,标记已处理`);
82
+ }
83
+ return result;
84
+ }
85
+
86
+ // 从第一个视频获取 locationCreated
87
+ let locationCreated = null;
88
+ if (videoArray.length > 0) {
89
+ const firstVideo = videoArray[0];
90
+ const firstVideoUrl = firstVideo.href.startsWith('http')
91
+ ? firstVideo.href
92
+ : `https://www.tiktok.com${firstVideo.href}`;
93
+
94
+ try {
95
+ locationCreated = await extractVideoLocation(firstVideoUrl);
96
+ } catch (e) {
97
+ log(` 获取视频国家失败: ${e.message}`);
98
+ }
99
+ }
100
+
101
+ result.locationCreated = locationCreated || null;
102
+ log(` 国家: ${result.locationCreated || '未知'}`);
103
+
104
+ // 国家筛选
105
+ const locationList = (location || 'ES').split(',').map(s => s.trim().toUpperCase());
106
+ const isTargetLocation = locationList.includes(result.locationCreated?.toUpperCase?.() || result.locationCreated);
107
+
108
+ if (isTargetLocation) {
109
+ result.keepFollow = true;
110
+ log(` 国家匹配,获取关注/粉丝...`);
111
+
112
+ // 提取关注/粉丝
113
+ if (enableFollow) {
114
+ const loggedIn = await isLoggedIn(page);
115
+ if (!loggedIn) {
116
+ log(' [跳过] 获取关注/粉丝:未登录,请先登录 TikTok');
117
+ result.hasFollowData = false;
118
+ result.discoveredFollowing = [];
119
+ result.discoveredFollowers = [];
120
+ } else {
121
+ try {
122
+ const effectiveMaxFollowing = isSeller ? globalMaxFollowing : maxFollowing;
123
+ const effectiveMaxFollowers = isSeller ? globalMaxFollowers : maxFollowers;
124
+ if (isSeller) log(` 商家用户,关注采集: ${effectiveMaxFollowing}, 粉丝采集: ${effectiveMaxFollowers}`);
125
+ const { following, followers } = await extractFollowAndFollowers(
126
+ page, { maxFollowing: effectiveMaxFollowing, maxFollowers: effectiveMaxFollowers, log }
127
+ );
128
+ result.discoveredFollowing = following || [];
129
+ result.discoveredFollowers = followers || [];
130
+ result.hasFollowData = true;
131
+ log(` 关注: ${result.discoveredFollowing.length}, 粉丝: ${result.discoveredFollowers.length}`);
132
+ } catch (e) {
133
+ log(` 关注/粉丝提取失败: ${e.message}`);
134
+ result.hasFollowData = false;
135
+ result.discoveredFollowing = [];
136
+ result.discoveredFollowers = [];
137
+ }
138
+ }
139
+ }
140
+
141
+ // 携带视频列表供登记
142
+ result.videoList = videoArray;
143
+ } else {
144
+ // 国家不匹配
145
+ result.keepFollow = false;
146
+ result.discoveredFollowing = [];
147
+ result.discoveredFollowers = [];
148
+ result.hasFollowData = false;
149
+ log(` 国家不匹配,跳过`);
150
+ }
151
+
152
+ result.processed = true;
153
+ } catch (e) {
154
+ result.error = e.message;
155
+ result.errorStack = e.stack || '';
156
+ log(` [错误] ${e.message}`);
157
+ }
158
+
159
+ return result;
160
+ }
161
+
162
+ 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
+ }