tt-help-cli-ycl 1.3.45 → 1.3.47

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 (67) hide show
  1. package/README.md +33 -33
  2. package/cli.js +9 -9
  3. package/package.json +52 -52
  4. package/scripts/run-explore copy.bat +101 -101
  5. package/scripts/run-explore.bat +134 -134
  6. package/scripts/run-explore.ps1 +159 -159
  7. package/scripts/run-explore.sh +121 -121
  8. package/src/cli/attach.js +331 -313
  9. package/src/cli/auto.js +265 -265
  10. package/src/cli/comments.js +620 -620
  11. package/src/cli/config.js +170 -170
  12. package/src/cli/db-import.js +51 -51
  13. package/src/cli/explore.js +555 -555
  14. package/src/cli/info.js +10 -16
  15. package/src/cli/open.js +111 -111
  16. package/src/cli/progress.js +111 -111
  17. package/src/cli/refresh.js +288 -288
  18. package/src/cli/scrape.js +47 -47
  19. package/src/cli/utils.js +18 -18
  20. package/src/cli/videos.js +41 -41
  21. package/src/cli/videostats.js +196 -196
  22. package/src/cli/watch.js +30 -30
  23. package/src/cli/webserver.js +19 -0
  24. package/src/lib/api-interceptor.js +161 -161
  25. package/src/lib/args.js +809 -778
  26. package/src/lib/browser/anti-detect.js +23 -23
  27. package/src/lib/browser/cdp.js +261 -261
  28. package/src/lib/browser/health-checker.js +114 -114
  29. package/src/lib/browser/launch.js +43 -43
  30. package/src/lib/browser/page.js +184 -184
  31. package/src/lib/constants.js +297 -287
  32. package/src/lib/delay.js +54 -54
  33. package/src/lib/explore-fetch.js +118 -118
  34. package/src/lib/fetcher.js +45 -45
  35. package/src/lib/filter.js +66 -66
  36. package/src/lib/io.js +54 -54
  37. package/src/lib/output.js +80 -80
  38. package/src/lib/page-error-detector.js +109 -109
  39. package/src/lib/parse-ssr.mjs +69 -69
  40. package/src/lib/parser.js +47 -47
  41. package/src/lib/retry.js +45 -45
  42. package/src/lib/scrape.js +90 -89
  43. package/src/lib/target-locations.js +61 -61
  44. package/src/lib/tiktok-scraper.mjs +173 -105
  45. package/src/lib/url.js +52 -52
  46. package/src/main.js +73 -70
  47. package/src/npm-main.js +70 -69
  48. package/src/scraper/auto-core.js +203 -203
  49. package/src/scraper/core.js +255 -255
  50. package/src/scraper/explore-core.js +208 -208
  51. package/src/scraper/modules/captcha-handler.js +114 -114
  52. package/src/scraper/modules/follow-extractor.js +250 -250
  53. package/src/scraper/modules/guess-extractor.js +51 -51
  54. package/src/scraper/modules/page-helpers.js +48 -48
  55. package/src/scraper/refresh-core.js +213 -213
  56. package/src/videos/core.js +143 -143
  57. package/src/watch/data-store.js +2980 -2978
  58. package/src/watch/public/index.html +2355 -2345
  59. package/src/watch/server.js +727 -727
  60. package/src/webserver/server.mjs +174 -0
  61. package/scripts/test-captcha-lib.mjs +0 -68
  62. package/scripts/test-captcha.mjs +0 -81
  63. package/scripts/test-incognito-lib.mjs +0 -36
  64. package/scripts/test-login-state.mjs +0 -128
  65. package/scripts/test-safe-click.mjs +0 -45
  66. package/scripts/test-watch-db-smoke.mjs +0 -246
  67. package/src/results/user-videos-bar.lar.lar.moeta.json +0 -37
@@ -0,0 +1,174 @@
1
+ import http from "http";
2
+ import os from "os";
3
+ import { TikTokScraper } from "../lib/tiktok-scraper.mjs";
4
+
5
+ function getLocalIP() {
6
+ const ifaces = os.networkInterfaces();
7
+ for (const name of Object.keys(ifaces)) {
8
+ for (const iface of ifaces[name]) {
9
+ if (iface.family === "IPv4" && !iface.internal) {
10
+ return iface.address;
11
+ }
12
+ }
13
+ }
14
+ return "127.0.0.1";
15
+ }
16
+
17
+ function sendJSON(res, status, data) {
18
+ const body = JSON.stringify(data, null, 2);
19
+ res.writeHead(status, {
20
+ "Content-Type": "application/json; charset=utf-8",
21
+ "Access-Control-Allow-Origin": "*",
22
+ "Access-Control-Allow-Methods": "POST, GET, OPTIONS",
23
+ "Access-Control-Allow-Headers": "Content-Type",
24
+ });
25
+ res.end(body);
26
+ }
27
+
28
+ function readBody(req) {
29
+ return new Promise((resolve, reject) => {
30
+ let body = "";
31
+ req.on("data", (chunk) => {
32
+ body += chunk.toString();
33
+ if (body.length > 10 * 1024 * 1024) {
34
+ req.destroy();
35
+ reject(new Error("Request body too large"));
36
+ }
37
+ });
38
+ req.on("end", () => {
39
+ try {
40
+ resolve(body ? JSON.parse(body) : {});
41
+ } catch (e) {
42
+ reject(new Error("Invalid JSON"));
43
+ }
44
+ });
45
+ req.on("error", reject);
46
+ });
47
+ }
48
+
49
+ let scraperInstance = null;
50
+ let scraperClosing = false;
51
+
52
+ async function getScraper() {
53
+ if (scraperInstance && !scraperClosing) {
54
+ return scraperInstance;
55
+ }
56
+ if (scraperClosing) {
57
+ throw new Error("Server is shutting down");
58
+ }
59
+ scraperInstance = new TikTokScraper({
60
+ poolSize: 3,
61
+ wafTTL: 120000,
62
+ maxRequestsPerPage: 50,
63
+ });
64
+ await scraperInstance.init();
65
+ return scraperInstance;
66
+ }
67
+
68
+ async function handleTiktokUser(req, res) {
69
+ try {
70
+ const body = await readBody(req);
71
+ if (!body.uniqueId) {
72
+ return sendJSON(res, 400, { error: "missing uniqueId" });
73
+ }
74
+ const scraper = await getScraper();
75
+ const result = await scraper.getUserInfo(body.uniqueId);
76
+ if (!result) {
77
+ return sendJSON(res, 404, { error: "User not found or failed to parse data" });
78
+ }
79
+ return sendJSON(res, 200, result);
80
+ } catch (e) {
81
+ return sendJSON(res, 500, { error: e.message });
82
+ }
83
+ }
84
+
85
+ async function handleTiktokVideo(req, res) {
86
+ try {
87
+ const body = await readBody(req);
88
+ if (!body.videoUrl) {
89
+ return sendJSON(res, 400, { error: "missing videoUrl" });
90
+ }
91
+ const scraper = await getScraper();
92
+ const result = await scraper.getVideoInfo(body.videoUrl);
93
+ if (!result) {
94
+ return sendJSON(res, 404, { error: "Video not found or failed to parse data" });
95
+ }
96
+ return sendJSON(res, 200, result);
97
+ } catch (e) {
98
+ return sendJSON(res, 500, { error: e.message });
99
+ }
100
+ }
101
+
102
+ async function handleTiktokLookup(req, res) {
103
+ try {
104
+ const body = await readBody(req);
105
+ if (!body.videoUrl) {
106
+ return sendJSON(res, 400, { error: "missing videoUrl" });
107
+ }
108
+ const scraper = await getScraper();
109
+ const result = await scraper.getUserAndVideo(body.videoUrl);
110
+ if (!result) {
111
+ return sendJSON(res, 404, { error: "Video not found or failed to parse data" });
112
+ }
113
+ return sendJSON(res, 200, result);
114
+ } catch (e) {
115
+ return sendJSON(res, 500, { error: e.message });
116
+ }
117
+ }
118
+
119
+ export async function startWebserver(port = 3000) {
120
+ const server = http.createServer(async (req, res) => {
121
+ const url = new URL(req.url, `http://127.0.0.1:${port}`);
122
+
123
+ if (req.method === "OPTIONS") {
124
+ res.writeHead(204, {
125
+ "Access-Control-Allow-Origin": "*",
126
+ "Access-Control-Allow-Methods": "POST, GET, OPTIONS",
127
+ "Access-Control-Allow-Headers": "Content-Type",
128
+ });
129
+ res.end();
130
+ return;
131
+ }
132
+
133
+ try {
134
+ if (url.pathname === "/api/tiktok/user" && req.method === "POST") {
135
+ return await handleTiktokUser(req, res);
136
+ }
137
+ if (url.pathname === "/api/tiktok/video" && req.method === "POST") {
138
+ return await handleTiktokVideo(req, res);
139
+ }
140
+ if (url.pathname === "/api/tiktok/lookup" && req.method === "POST") {
141
+ return await handleTiktokLookup(req, res);
142
+ }
143
+
144
+ sendJSON(res, 404, { error: "Not found", available: ["/api/tiktok/user", "/api/tiktok/video", "/api/tiktok/lookup"] });
145
+ } catch (e) {
146
+ sendJSON(res, 500, { error: e.message });
147
+ }
148
+ });
149
+
150
+ return new Promise((resolve, reject) => {
151
+ server.listen(port, () => {
152
+ const localIP = getLocalIP();
153
+ console.error(`\nTikTok Scraper WebService started`);
154
+ console.error(` Local: http://127.0.0.1:${port}`);
155
+ console.error(` Network: http://${localIP}:${port}`);
156
+ console.error(` APIs: POST /api/tiktok/user, /api/tiktok/video, /api/tiktok/lookup`);
157
+ console.error(` Press Ctrl+C to stop.\n`);
158
+ resolve({ server, port });
159
+ });
160
+ server.on("error", (err) => {
161
+ reject(err);
162
+ });
163
+ });
164
+ }
165
+
166
+ export async function closeScraper() {
167
+ if (scraperInstance) {
168
+ scraperClosing = true;
169
+ await scraperInstance.close();
170
+ scraperInstance = null;
171
+ }
172
+ }
173
+
174
+ export { getLocalIP };
@@ -1,68 +0,0 @@
1
- import { chromium } from 'playwright';
2
-
3
- const URL = 'https://www.tiktok.com/@mariaelenasanchez607/video/7630110959650000150';
4
-
5
- async function main() {
6
- const browser = await chromium.connectOverCDP('http://127.0.0.1:9222');
7
- const page = browser.contexts()[0].pages()[0];
8
-
9
- // 测试 detectCaptcha
10
- console.error('=== 测试 detectCaptcha (无验证码) ===');
11
- await page.goto(URL, { waitUntil: 'domcontentloaded', timeout: 30000 });
12
- await page.waitForTimeout(3000);
13
-
14
- const { detectCaptcha, closeCaptcha, handleCaptcha } = await import('../src/scraper/modules/captcha-handler.mjs');
15
-
16
- const r1 = await detectCaptcha(page);
17
- console.error('未点击评论:', JSON.stringify(r1));
18
-
19
- // 点击评论触发验证码
20
- await page.evaluate(() => {
21
- const all = document.querySelectorAll('button');
22
- for (const el of all) {
23
- if (/^评论$/.test(el.textContent?.trim()) && el.offsetParent !== null && el.getBoundingClientRect().width > 0) {
24
- el.click();
25
- break;
26
- }
27
- }
28
- });
29
- await page.waitForTimeout(3000);
30
-
31
- console.error('\n=== 测试 detectCaptcha (有验证码) ===');
32
- const r2 = await detectCaptcha(page);
33
- console.error('点击评论后:', JSON.stringify(r2));
34
-
35
- console.error('\n=== 测试 closeCaptcha ===');
36
- const r3 = await closeCaptcha(page);
37
- await page.waitForTimeout(1000);
38
- console.error('关闭结果:', JSON.stringify(r3));
39
-
40
- const r4 = await detectCaptcha(page);
41
- console.error('关闭后检测:', JSON.stringify(r4));
42
-
43
- console.error('\n=== 测试 handleCaptcha (完整流程) ===');
44
- // 重新触发
45
- await page.goto(URL, { waitUntil: 'domcontentloaded', timeout: 30000 });
46
- await page.waitForTimeout(3000);
47
- await page.evaluate(() => {
48
- const all = document.querySelectorAll('button');
49
- for (const el of all) {
50
- if (/^评论$/.test(el.textContent?.trim()) && el.offsetParent !== null && el.getBoundingClientRect().width > 0) {
51
- el.click();
52
- break;
53
- }
54
- }
55
- });
56
- await page.waitForTimeout(3000);
57
-
58
- const r5 = await handleCaptcha(page);
59
- console.error('handleCaptcha 结果:', JSON.stringify(r5));
60
-
61
- await page.screenshot({ path: '/tmp/lib-test-final.png' });
62
- console.error('\n最终截图: /tmp/lib-test-final.png');
63
- }
64
-
65
- main().catch(err => {
66
- console.error('错误:', err);
67
- process.exit(1);
68
- });
@@ -1,81 +0,0 @@
1
- import { ensureBrowserReady } from '../src/lib/browser/cdp.js';
2
-
3
- const url = 'https://www.tiktok.com/@mariaelenasanchez607/video/7630110959650000150';
4
-
5
- async function main() {
6
- const browser = await ensureBrowserReady();
7
- const defaultContext = browser.contexts()[0];
8
- const pages = defaultContext.pages();
9
- const page = pages[0] || await defaultContext.newPage();
10
-
11
- await page.goto(url, { waitUntil: 'domcontentloaded', timeout: 30000 });
12
- await page.waitForTimeout(5000);
13
-
14
- // 用 force: true 点击评论按钮
15
- const clicked = await page.evaluate(() => {
16
- const btn = document.querySelector('[data-e2e="comments"]');
17
- if (btn && btn.getBoundingClientRect().width > 0) {
18
- btn.click();
19
- return { success: true, rect: btn.getBoundingClientRect() };
20
- }
21
- return { success: false };
22
- });
23
-
24
- console.error('点击结果:', JSON.stringify(clicked));
25
-
26
- // 等待可能的验证码
27
- await page.waitForTimeout(5000);
28
-
29
- // 截图
30
- await page.screenshot({ path: '/tmp/tiktok-comment-clicked.png' });
31
- console.error('截图: /tmp/tiktok-comment-clicked.png');
32
-
33
- // 全面检测验证码
34
- const captcha = await page.evaluate(() => {
35
- const result = {};
36
-
37
- // 大尺寸 Verify 元素
38
- const verifyEls = Array.from(document.querySelectorAll('[class*="Verify"], [class*="verify"]'));
39
- result.verifyElements = verifyEls.filter(el => {
40
- const r = el.getBoundingClientRect();
41
- return r.width > 100 && r.height > 100 && el.offsetParent !== null;
42
- }).map(el => ({
43
- class: el.className.substring(0, 200),
44
- text: el.textContent?.substring(0, 300),
45
- rect: { w: Math.round(el.getBoundingClientRect().width), h: Math.round(el.getBoundingClientRect().height), x: Math.round(el.getBoundingClientRect().x), y: Math.round(el.getBoundingClientRect().y) }
46
- }));
47
-
48
- // 全屏遮罩
49
- result.fullScreenOverlays = Array.from(document.querySelectorAll('div')).filter(d => {
50
- const r = d.getBoundingClientRect();
51
- const style = window.getComputedStyle(d);
52
- return r.width > 500 && r.height > 500 && parseInt(style.zIndex) > 900 && d.offsetParent !== null;
53
- }).map(d => ({
54
- class: d.className.substring(0, 100),
55
- zIndex: window.getComputedStyle(d).zIndex,
56
- rect: { w: Math.round(d.getBoundingClientRect().width), h: Math.round(d.getBoundingClientRect().height) }
57
- }));
58
-
59
- result.iframes = Array.from(document.querySelectorAll('iframe')).map(f => ({
60
- src: (f.src || f.getAttribute('src') || '').substring(0, 300)
61
- }));
62
-
63
- return result;
64
- });
65
-
66
- console.error('\n=== 验证码检测 ===');
67
- console.error(JSON.stringify(captcha, null, 2));
68
-
69
- if (captcha.verifyElements.length > 0 || captcha.fullScreenOverlays.length > 0 || captcha.iframes.length > 0) {
70
- console.error('\n⚠️ 检测到验证码或遮罩层!');
71
- } else {
72
- console.error('\n✅ 未检测到验证码');
73
- }
74
-
75
- await browser.close();
76
- }
77
-
78
- main().catch(err => {
79
- console.error('错误:', err);
80
- process.exit(1);
81
- });
@@ -1,36 +0,0 @@
1
- import { chromium } from 'playwright';
2
- import { detectCaptcha, closeCaptcha, handleCaptcha, getIncognitoPage } from '../src/scraper/modules/captcha-handler.mjs';
3
-
4
- async function main() {
5
- const browser = await chromium.connectOverCDP('http://127.0.0.1:9222');
6
- const url = 'https://www.tiktok.com/@mariaelenasanchez607/video/7630110959650000150';
7
-
8
- // 测试1: 无痕模式打开 + 点击评论
9
- console.error('=== 测试: 无痕模式 ===');
10
- const { page, context } = await getIncognitoPage(browser, url);
11
- console.error('URL:', page.url());
12
-
13
- await page.evaluate(() => {
14
- const all = document.querySelectorAll('button');
15
- for (const el of all) {
16
- if (/^评论$/.test(el.textContent?.trim()) && el.offsetParent !== null && el.getBoundingClientRect().width > 0) {
17
- el.click();
18
- break;
19
- }
20
- }
21
- });
22
- await new Promise(r => setTimeout(r, 3000));
23
-
24
- const captcha = await detectCaptcha(page);
25
- console.error('验证码:', captcha);
26
-
27
- await page.screenshot({ path: '/tmp/incognito-lib-test.png' });
28
- console.error('截图: /tmp/incognito-lib-test.png');
29
-
30
- await context.close();
31
- }
32
-
33
- main().catch(err => {
34
- console.error('错误:', err);
35
- process.exit(1);
36
- });
@@ -1,128 +0,0 @@
1
- import { ensureBrowserReady } from '../src/lib/browser/cdp.js';
2
-
3
- async function main() {
4
- const browser = await ensureBrowserReady();
5
- const defaultContext = browser.contexts()[0];
6
- const page = defaultContext.pages()[0] || await defaultContext.newPage();
7
-
8
- // 确保在 tiktok 页面
9
- if (!page.url().includes('tiktok.com')) {
10
- console.error('当前不在 TikTok 页面:', page.url());
11
- await page.goto('https://www.tiktok.com', { waitUntil: 'domcontentloaded', timeout: 30000 });
12
- await new Promise(r => setTimeout(r, 3000));
13
- }
14
-
15
- console.error('当前 URL:', page.url());
16
-
17
- // 1. 检测现有 isLoggedIn 的逻辑
18
- const currentUserMenu = await page.evaluate(() => {
19
- const selectors = [
20
- '[class*="UserMenu"]',
21
- '[class*="user-menu"]',
22
- '[class*="CurrentUserInfo"]',
23
- ];
24
- const results = {};
25
- for (const sel of selectors) {
26
- const el = document.querySelector(sel);
27
- results[sel] = !!el;
28
- if (el) {
29
- results[sel + '_class'] = el.className;
30
- }
31
- }
32
- return results;
33
- });
34
-
35
- console.error('\n=== 当前选择器检测结果 ===');
36
- console.error(JSON.stringify(currentUserMenu, null, 2));
37
-
38
- const hasLoginButton = await page.evaluate(() => {
39
- const buttons = Array.from(document.querySelectorAll('button, [role="button"]'));
40
- const loginButtons = buttons.filter(el =>
41
- /^(登录|Log in|Sign in)$/i.test(el.textContent.trim())
42
- );
43
- return {
44
- total: buttons.length,
45
- loginCount: loginButtons.length,
46
- samples: loginButtons.slice(0, 5).map(el => ({
47
- text: el.textContent.trim(),
48
- class: el.className,
49
- })),
50
- };
51
- });
52
-
53
- console.error('\n=== 登录按钮检测 ===');
54
- console.error(JSON.stringify(hasLoginButton, null, 2));
55
-
56
- // 2. 用更宽泛的方式找用户相关元素
57
- const broadSearch = await page.evaluate(() => {
58
- const allClasses = [];
59
-
60
- // 搜索可能的用户头像/菜单元素
61
- const candidates = [];
62
-
63
- // 顶部导航区域
64
- const navArea = document.querySelector('[class*="nav"], [class*="Header"], [class*="header"]');
65
- if (navArea) {
66
- const avatars = navArea.querySelectorAll('[class*="avatar"], [class*="Avatar"], [class*="photo"], [class*="Photo"], [class*="image"], [class*="Image"]');
67
- avatars.forEach(el => {
68
- candidates.push({
69
- tag: el.tagName,
70
- class: el.className.substring(0, 200),
71
- text: el.textContent?.substring(0, 50),
72
- parent: el.parentElement?.className?.substring(0, 100),
73
- });
74
- });
75
- }
76
-
77
- // 搜索包含用户信息的链接
78
- const profileLinks = Array.from(document.querySelectorAll('a[href*="/@"]'));
79
- const profileSamples = profileLinks.slice(0, 10).map(el => ({
80
- href: el.href,
81
- class: el.className?.substring(0, 100),
82
- text: el.textContent?.substring(0, 50),
83
- }));
84
-
85
- // 搜索所有包含 User 关键字的类名
86
- const userElements = Array.from(document.querySelectorAll('*')).filter(el =>
87
- el.className && typeof el.className === 'string' &&
88
- /User|user|Profile|profile/.test(el.className) &&
89
- el.tagName !== 'STYLE' && el.tagName !== 'SCRIPT'
90
- ).slice(0, 30).map(el => ({
91
- tag: el.tagName,
92
- class: el.className.substring(0, 150),
93
- text: el.textContent?.substring(0, 30),
94
- }));
95
-
96
- return {
97
- navAreaFound: !!navArea,
98
- avatarCandidates: candidates,
99
- profileLinks: profileSamples,
100
- userElements,
101
- };
102
- });
103
-
104
- console.error('\n=== 宽泛搜索 ===');
105
- console.error(JSON.stringify(broadSearch, null, 2));
106
-
107
- // 3. 截图
108
- await page.screenshot({ path: '/tmp/login-debug.png' });
109
- console.error('\n截图已保存到 /tmp/login-debug.png');
110
-
111
- // 4. 汇总判断
112
- const isLoggedIn = currentUserMenu['[class*="UserMenu"]'] ||
113
- currentUserMenu['[class*="user-menu"]'] ||
114
- currentUserMenu['[class*="CurrentUserInfo"]'] ||
115
- broadSearch.userElements.length > 0;
116
-
117
- console.error('\n=== 结论 ===');
118
- console.error('isLoggedIn 函数返回:', !!(currentUserMenu['[class*="UserMenu"]'] || currentUserMenu['[class*="user-menu"]'] || currentUserMenu['[class*="CurrentUserInfo"]']) && !hasLoginButton.loginCount);
119
- console.error('宽泛搜索是否找到用户元素:', broadSearch.userElements.length > 0);
120
- console.error('找到用户相关元素数量:', broadSearch.userElements.length);
121
- console.error('头像候选数量:', broadSearch.avatarCandidates.length);
122
- console.error('Profile 链接数量:', broadSearch.profileLinks.length);
123
- }
124
-
125
- main().catch(err => {
126
- console.error('错误:', err);
127
- process.exit(1);
128
- });
@@ -1,45 +0,0 @@
1
- import { chromium } from 'playwright';
2
- import { safeClickComment, detectCaptcha } from '../src/scraper/modules/captcha-handler.mjs';
3
- import { ensureBrowserReady } from '../src/lib/browser/cdp.js';
4
-
5
- const URL = 'https://www.tiktok.com/@mariaelenasanchez607/video/7630110959650000150';
6
-
7
- async function main() {
8
- const browser = await ensureBrowserReady();
9
- const defaultContext = browser.contexts()[0];
10
- const page = defaultContext.pages()[0] || await defaultContext.newPage();
11
-
12
- for (let i = 1; i <= 3; i++) {
13
- console.error(`\n===== 第 ${i} 轮 =====`);
14
- await page.goto(URL, { waitUntil: 'domcontentloaded', timeout: 30000 });
15
- await new Promise(r => setTimeout(r, 5000));
16
-
17
- const result = await safeClickComment(page);
18
- console.error('结果:', JSON.stringify(result));
19
-
20
- const stillThere = await detectCaptcha(page);
21
- console.error('验证码残留:', !!stillThere);
22
-
23
- await page.screenshot({ path: `/tmp/safe-click-run-${i}.png` });
24
-
25
- // 关闭评论面板
26
- await page.evaluate(() => {
27
- const rightPanel = document.querySelector('[class*="RightPanelContainer"]');
28
- if (rightPanel) {
29
- const tabContainer = rightPanel.querySelector('[class*="TabContainer"]');
30
- if (tabContainer) {
31
- const closeOverlay = tabContainer.querySelector('div:last-child');
32
- if (closeOverlay) closeOverlay.click();
33
- }
34
- }
35
- });
36
- await new Promise(r => setTimeout(r, 2000));
37
- }
38
-
39
- console.error('\n完成');
40
- }
41
-
42
- main().catch(err => {
43
- console.error('错误:', err);
44
- process.exit(1);
45
- });