@jackwener/opencli 1.0.3 → 1.0.5

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 (190) hide show
  1. package/.github/workflows/build-extension.yml +21 -3
  2. package/.github/workflows/docs.yml +52 -0
  3. package/README.md +28 -28
  4. package/README.zh-CN.md +28 -28
  5. package/dist/browser/cdp.d.ts +16 -1
  6. package/dist/browser/cdp.js +124 -80
  7. package/dist/browser/daemon-client.d.ts +3 -1
  8. package/dist/browser/daemon-client.js +4 -0
  9. package/dist/browser/dom-helpers.d.ts +20 -0
  10. package/dist/browser/dom-helpers.js +109 -0
  11. package/dist/browser/mcp.d.ts +1 -0
  12. package/dist/browser/mcp.js +10 -5
  13. package/dist/browser/page.d.ts +7 -0
  14. package/dist/browser/page.js +37 -100
  15. package/dist/browser.test.js +7 -0
  16. package/dist/build-manifest.js +3 -1
  17. package/dist/build-manifest.test.js +34 -0
  18. package/dist/capabilityRouting.d.ts +2 -0
  19. package/dist/capabilityRouting.js +30 -0
  20. package/dist/capabilityRouting.test.d.ts +1 -0
  21. package/dist/capabilityRouting.test.js +42 -0
  22. package/dist/chaoxing.test.js +11 -4
  23. package/dist/cli-manifest.json +635 -1
  24. package/dist/cli.js +48 -8
  25. package/dist/clis/antigravity/serve.d.ts +14 -0
  26. package/dist/clis/antigravity/serve.js +263 -0
  27. package/dist/clis/bilibili/download.js +4 -14
  28. package/dist/clis/boss/resume.d.ts +1 -0
  29. package/dist/clis/boss/resume.js +249 -0
  30. package/dist/clis/hf/top.d.ts +1 -0
  31. package/dist/clis/hf/top.js +119 -0
  32. package/dist/clis/jike/comment.d.ts +1 -0
  33. package/dist/clis/jike/comment.js +107 -0
  34. package/dist/clis/jike/create.d.ts +1 -0
  35. package/dist/clis/jike/create.js +106 -0
  36. package/dist/clis/jike/feed.d.ts +1 -0
  37. package/dist/clis/jike/feed.js +67 -0
  38. package/dist/clis/jike/like.d.ts +1 -0
  39. package/dist/clis/jike/like.js +61 -0
  40. package/dist/clis/jike/notifications.d.ts +1 -0
  41. package/dist/clis/jike/notifications.js +169 -0
  42. package/dist/clis/jike/post.yaml +58 -0
  43. package/dist/clis/jike/repost.d.ts +1 -0
  44. package/dist/clis/jike/repost.js +103 -0
  45. package/dist/clis/jike/search.d.ts +1 -0
  46. package/dist/clis/jike/search.js +67 -0
  47. package/dist/clis/jike/shared.d.ts +19 -0
  48. package/dist/clis/jike/shared.js +25 -0
  49. package/dist/clis/jike/topic.yaml +52 -0
  50. package/dist/clis/jike/user.yaml +51 -0
  51. package/dist/clis/smzdm/search.js +28 -39
  52. package/dist/clis/stackoverflow/bounties.yaml +29 -0
  53. package/dist/clis/stackoverflow/hot.yaml +28 -0
  54. package/dist/clis/stackoverflow/search.yaml +32 -0
  55. package/dist/clis/stackoverflow/unanswered.yaml +28 -0
  56. package/dist/clis/twitter/download.js +6 -16
  57. package/dist/clis/xiaohongshu/download.js +3 -3
  58. package/dist/clis/zhihu/download.js +3 -3
  59. package/dist/doctor.d.ts +7 -0
  60. package/dist/doctor.js +16 -0
  61. package/dist/download/index.d.ts +12 -8
  62. package/dist/download/index.js +11 -3
  63. package/dist/download/index.test.d.ts +1 -0
  64. package/dist/download/index.test.js +14 -0
  65. package/dist/engine.js +5 -5
  66. package/dist/explore.d.ts +1 -0
  67. package/dist/explore.js +3 -3
  68. package/dist/generate.js +1 -0
  69. package/dist/interceptor.js +3 -2
  70. package/dist/output.d.ts +1 -0
  71. package/dist/output.js +3 -1
  72. package/dist/pipeline/executor.test.js +1 -0
  73. package/dist/pipeline/steps/download.js +14 -18
  74. package/dist/registry.d.ts +1 -0
  75. package/dist/registry.js +5 -2
  76. package/dist/runtime.d.ts +4 -1
  77. package/dist/runtime.js +2 -2
  78. package/dist/types.d.ts +12 -0
  79. package/dist/verify.d.ts +6 -1
  80. package/dist/verify.js +54 -2
  81. package/docs/.vitepress/config.mts +193 -0
  82. package/docs/adapters/browser/apple-podcasts.md +28 -0
  83. package/docs/adapters/browser/bbc.md +26 -0
  84. package/docs/adapters/browser/bilibili.md +38 -0
  85. package/docs/adapters/browser/boss.md +28 -0
  86. package/docs/adapters/browser/coupang.md +28 -0
  87. package/docs/adapters/browser/ctrip.md +27 -0
  88. package/docs/adapters/browser/github.md +26 -0
  89. package/docs/adapters/browser/hackernews.md +26 -0
  90. package/docs/adapters/browser/linkedin.md +27 -0
  91. package/docs/adapters/browser/reddit.md +41 -0
  92. package/docs/adapters/browser/reuters.md +27 -0
  93. package/docs/adapters/browser/smzdm.md +27 -0
  94. package/docs/adapters/browser/twitter.md +47 -0
  95. package/docs/adapters/browser/v2ex.md +32 -0
  96. package/docs/adapters/browser/weibo.md +27 -0
  97. package/docs/adapters/browser/xiaohongshu.md +32 -0
  98. package/docs/adapters/browser/xiaoyuzhou.md +28 -0
  99. package/docs/adapters/browser/xueqiu.md +32 -0
  100. package/docs/adapters/browser/yahoo-finance.md +26 -0
  101. package/docs/adapters/browser/youtube.md +29 -0
  102. package/docs/adapters/browser/zhihu.md +30 -0
  103. package/docs/adapters/desktop/antigravity.md +46 -0
  104. package/docs/adapters/desktop/chatgpt.md +43 -0
  105. package/docs/adapters/desktop/chatwise.md +38 -0
  106. package/docs/adapters/desktop/codex.md +32 -0
  107. package/docs/adapters/desktop/cursor.md +33 -0
  108. package/docs/adapters/desktop/discord.md +28 -0
  109. package/docs/adapters/desktop/feishu.md +20 -0
  110. package/docs/adapters/desktop/neteasemusic.md +31 -0
  111. package/docs/adapters/desktop/notion.md +29 -0
  112. package/docs/adapters/desktop/wechat.md +28 -0
  113. package/docs/adapters/index.md +49 -0
  114. package/docs/advanced/cdp.md +103 -0
  115. package/docs/advanced/download.md +63 -0
  116. package/docs/advanced/electron.md +125 -0
  117. package/docs/advanced/remote-chrome.md +72 -0
  118. package/docs/developer/ai-workflow.md +66 -0
  119. package/docs/developer/architecture.md +90 -0
  120. package/docs/developer/contributing.md +136 -0
  121. package/docs/developer/testing.md +237 -0
  122. package/docs/developer/ts-adapter.md +87 -0
  123. package/docs/developer/yaml-adapter.md +108 -0
  124. package/docs/guide/browser-bridge.md +38 -0
  125. package/docs/guide/getting-started.md +56 -0
  126. package/docs/guide/installation.md +37 -0
  127. package/docs/guide/troubleshooting.md +56 -0
  128. package/docs/index.md +35 -0
  129. package/docs/zh/adapters/index.md +5 -0
  130. package/docs/zh/advanced/cdp.md +3 -0
  131. package/docs/zh/developer/contributing.md +24 -0
  132. package/docs/zh/guide/browser-bridge.md +25 -0
  133. package/docs/zh/guide/getting-started.md +40 -0
  134. package/docs/zh/guide/installation.md +37 -0
  135. package/docs/zh/index.md +29 -0
  136. package/extension/dist/background.js +92 -52
  137. package/extension/package-lock.json +1156 -0
  138. package/extension/src/background.test.ts +151 -0
  139. package/extension/src/background.ts +122 -51
  140. package/extension/src/protocol.ts +3 -1
  141. package/package.json +7 -3
  142. package/src/browser/cdp.ts +154 -82
  143. package/src/browser/daemon-client.ts +7 -1
  144. package/src/browser/dom-helpers.ts +116 -0
  145. package/src/browser/mcp.ts +14 -6
  146. package/src/browser/page.ts +45 -100
  147. package/src/browser.test.ts +10 -0
  148. package/src/build-manifest.test.ts +36 -0
  149. package/src/build-manifest.ts +2 -1
  150. package/src/capabilityRouting.test.ts +47 -0
  151. package/src/capabilityRouting.ts +28 -0
  152. package/src/chaoxing.test.ts +12 -4
  153. package/src/cli.ts +30 -8
  154. package/src/clis/antigravity/serve.ts +329 -0
  155. package/src/clis/bilibili/download.ts +4 -15
  156. package/src/clis/boss/resume.ts +262 -0
  157. package/src/clis/hf/top.ts +141 -0
  158. package/src/clis/jike/comment.ts +113 -0
  159. package/src/clis/jike/create.ts +113 -0
  160. package/src/clis/jike/feed.ts +74 -0
  161. package/src/clis/jike/like.ts +65 -0
  162. package/src/clis/jike/notifications.ts +185 -0
  163. package/src/clis/jike/post.yaml +58 -0
  164. package/src/clis/jike/repost.ts +114 -0
  165. package/src/clis/jike/search.ts +74 -0
  166. package/src/clis/jike/shared.ts +36 -0
  167. package/src/clis/jike/topic.yaml +52 -0
  168. package/src/clis/jike/user.yaml +51 -0
  169. package/src/clis/smzdm/search.ts +30 -39
  170. package/src/clis/stackoverflow/bounties.yaml +29 -0
  171. package/src/clis/stackoverflow/hot.yaml +28 -0
  172. package/src/clis/stackoverflow/search.yaml +32 -0
  173. package/src/clis/stackoverflow/unanswered.yaml +28 -0
  174. package/src/clis/twitter/download.ts +6 -17
  175. package/src/clis/xiaohongshu/download.ts +3 -3
  176. package/src/clis/zhihu/download.ts +3 -3
  177. package/src/doctor.ts +18 -2
  178. package/src/download/index.test.ts +16 -0
  179. package/src/download/index.ts +22 -4
  180. package/src/engine.ts +4 -4
  181. package/src/explore.ts +4 -4
  182. package/src/generate.ts +1 -0
  183. package/src/interceptor.ts +3 -2
  184. package/src/output.ts +3 -1
  185. package/src/pipeline/executor.test.ts +1 -0
  186. package/src/pipeline/steps/download.ts +14 -17
  187. package/src/registry.ts +6 -2
  188. package/src/runtime.ts +3 -2
  189. package/src/types.ts +9 -0
  190. package/src/verify.ts +64 -3
@@ -0,0 +1,61 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ /**
3
+ * 点赞即刻帖子
4
+ *
5
+ * 即刻帖子详情页的操作栏是 div 元素(非 button),
6
+ * 点赞按钮可通过 class 前缀 _likeButton_ 定位。
7
+ */
8
+ cli({
9
+ site: 'jike',
10
+ name: 'like',
11
+ description: '点赞即刻帖子',
12
+ domain: 'web.okjike.com',
13
+ strategy: Strategy.UI,
14
+ browser: true,
15
+ args: [
16
+ { name: 'id', type: 'string', required: true, help: '帖子 ID' },
17
+ ],
18
+ columns: ['status', 'message'],
19
+ func: async (page, kwargs) => {
20
+ // 1. 导航到帖子详情页
21
+ await page.goto(`https://web.okjike.com/originalPost/${kwargs.id}`);
22
+ await page.wait(5);
23
+ // 2. 找到点赞按钮并点击
24
+ const result = await page.evaluate(`(async () => {
25
+ try {
26
+ // 点赞按钮:class 包含 _likeButton_,在 _actions_ 容器内
27
+ const likeBtn = document.querySelector('[class*="_likeButton_"]');
28
+ if (!likeBtn) {
29
+ return { ok: false, message: '未找到点赞按钮' };
30
+ }
31
+
32
+ // 检查是否已点赞(已赞按钮带有 _liked_ 类)
33
+ const cls = likeBtn.className || '';
34
+ if (cls.includes('_liked')) {
35
+ return { ok: true, message: '该帖子已赞过' };
36
+ }
37
+
38
+ // 记录点击前的类名
39
+ const beforeCls = likeBtn.className;
40
+
41
+ likeBtn.click();
42
+ await new Promise(r => setTimeout(r, 1500));
43
+
44
+ // 验证:类名变化表示点赞成功
45
+ const afterCls = likeBtn.className;
46
+ if (afterCls !== beforeCls) {
47
+ return { ok: true, message: '点赞成功' };
48
+ }
49
+
50
+ // 类名未变化,无法确认点赞是否成功
51
+ return { ok: false, message: '点赞状态未确认,请手动检查' };
52
+ } catch (e) {
53
+ return { ok: false, message: e.toString() };
54
+ }
55
+ })()`);
56
+ return [{
57
+ status: result.ok ? 'success' : 'failed',
58
+ message: result.message,
59
+ }];
60
+ },
61
+ });
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,169 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ // 将通知类型代码映射为中文标签
3
+ function resolveActionLabel(type) {
4
+ if (!type)
5
+ return '通知';
6
+ const upper = type.toUpperCase();
7
+ if (upper.includes('LIKE'))
8
+ return '赞了你';
9
+ if (upper.includes('COMMENT'))
10
+ return '评论了你';
11
+ if (upper.includes('FOLLOW'))
12
+ return '关注了你';
13
+ if (upper.includes('REPOST'))
14
+ return '转发了你';
15
+ if (upper.includes('MENTION'))
16
+ return '提到了你';
17
+ if (upper.includes('REPLY'))
18
+ return '回复了你';
19
+ return type;
20
+ }
21
+ cli({
22
+ site: 'jike',
23
+ name: 'notifications',
24
+ description: '即刻通知',
25
+ domain: 'web.okjike.com',
26
+ strategy: Strategy.COOKIE,
27
+ browser: true,
28
+ args: [
29
+ { name: 'limit', type: 'int', default: 20 },
30
+ ],
31
+ columns: ['type', 'user', 'content', 'time'],
32
+ func: async (page, kwargs) => {
33
+ const limit = kwargs.limit || 20;
34
+ // 1. 直接导航到通知页
35
+ await page.goto('https://web.okjike.com/notification');
36
+ await page.wait(5);
37
+ // 3. 优先用 React fiber 提取通知数据
38
+ // 通知 fiber 数据结构与帖子不同,需查找含 type + user 字段的 props
39
+ const fiberResults = (await page.evaluate(`(() => {
40
+ // 从 React fiber 树中提取通知数据,向上最多走 15 层
41
+ function getNotificationData(element) {
42
+ for (const key of Object.keys(element)) {
43
+ if (key.startsWith('__reactFiber$') || key.startsWith('__reactInternalInstance$')) {
44
+ let fiber = element[key];
45
+ for (let i = 0; i < 15 && fiber; i++) {
46
+ const props = fiber.memoizedProps || fiber.pendingProps;
47
+ if (props && props.data) {
48
+ const d = props.data;
49
+ // 通知条目特征:含 type/actionType 字段,以及来源用户字段
50
+ if (d.type || d.actionType || d.notificationType) return d;
51
+ }
52
+ fiber = fiber.return;
53
+ }
54
+ }
55
+ }
56
+ return null;
57
+ }
58
+
59
+ const results = [];
60
+ const seen = new Set();
61
+
62
+ // 通知页使用 _item_ 类前缀作为条目容器
63
+ const elements = Array.from(document.querySelectorAll('[class*="_item_"]'));
64
+
65
+ for (const el of elements) {
66
+ const data = getNotificationData(el);
67
+ if (!data) continue;
68
+
69
+ const actionType = data.actionType || data.type || data.notificationType || '';
70
+ const fromUser =
71
+ (data.sourceUser && data.sourceUser.screenName) ||
72
+ (data.user && data.user.screenName) ||
73
+ (data.actionUser && data.actionUser.screenName) ||
74
+ (data.actor && data.actor.screenName) ||
75
+ '';
76
+ const targetContent =
77
+ (data.targetPost && data.targetPost.content) ||
78
+ (data.post && data.post.content) ||
79
+ '';
80
+ const commentContent =
81
+ (data.comment && data.comment.content) ||
82
+ data.commentContent ||
83
+ '';
84
+ const content = commentContent || targetContent;
85
+ const time = data.createdAt || data.updatedAt || '';
86
+
87
+ // 用 user+time+type 去重,避免同一用户同一时间不同类型通知被合并
88
+ const key = fromUser + '\x00' + time + '\x00' + (data.type || data.actionType || '');
89
+ if (seen.has(key)) continue;
90
+ seen.add(key);
91
+
92
+ if (!fromUser && !content) continue;
93
+
94
+ results.push({ actionType, fromUser, content, time });
95
+ }
96
+
97
+ return results;
98
+ })()`));
99
+ // 4. fiber 提取成功,映射类型标签后返回
100
+ if (fiberResults.length > 0) {
101
+ const notifications = fiberResults.map((r) => ({
102
+ type: resolveActionLabel(r.actionType),
103
+ user: r.fromUser,
104
+ content: r.content.replace(/\n/g, ' ').slice(0, 100),
105
+ time: r.time,
106
+ }));
107
+ return notifications.slice(0, limit);
108
+ }
109
+ // 5. 回退:解析通知条目的 innerText(格式: 用户名\n操作描述\n日期)
110
+ await page.autoScroll({ times: Math.ceil(limit / 10), delayMs: 2000 });
111
+ const domResults = (await page.evaluate(`(() => {
112
+ const results = [];
113
+ const items = document.querySelectorAll('[class*="_item_"]');
114
+
115
+ // 动作关键词,用于定位通知类型行
116
+ // 长模式优先匹配,避免"赞了你"截断"赞了你的动态"
117
+ const actionPatterns = [
118
+ '赞了你的动态', '赞了你的评论', '赞了你的转发',
119
+ '评论了你的动态', '评论了你的转发',
120
+ '回复了你的评论', '回复了你',
121
+ '转发了你的动态',
122
+ '提到了你', '关注了你',
123
+ '赞了你', '评论了你', '转发了你',
124
+ ];
125
+
126
+ for (const item of items) {
127
+ const text = item.innerText || '';
128
+ const lines = text.split('\\n').map(l => l.trim()).filter(Boolean);
129
+ if (lines.length < 2) continue;
130
+
131
+ // 策略:在全文中搜索动作关键词来定位类型
132
+ let type = '';
133
+ let user = '';
134
+ let time = '';
135
+ let content = '';
136
+
137
+ const fullText = lines.join(' ');
138
+ for (const pattern of actionPatterns) {
139
+ const idx = fullText.indexOf(pattern);
140
+ if (idx >= 0) {
141
+ // 关键词前面是用户名(可能多人用、分隔)
142
+ user = fullText.slice(0, idx).replace(/[、,]/g, ' ').trim();
143
+ type = pattern;
144
+ // 关键词后面可能是时间和原帖内容
145
+ const rest = fullText.slice(idx + pattern.length).trim();
146
+ // 时间通常以数字或"刚刚"/"分钟前"等开头
147
+ const timeMatch = rest.match(/^[\\S]*?(?:刚刚|\\d+.*?前|\\d{4}\\/\\d{2}\\/\\d{2})/);
148
+ time = timeMatch ? timeMatch[0] : '';
149
+ content = rest.slice(time.length).trim().slice(0, 100);
150
+ break;
151
+ }
152
+ }
153
+
154
+ // 没匹配到动作关键词,回退到简单行序
155
+ if (!type) {
156
+ user = lines[0] || '';
157
+ type = lines[1] || '';
158
+ time = lines[2] || '';
159
+ }
160
+
161
+ if (!user && !type) continue;
162
+ results.push({ type, user, content, time });
163
+ }
164
+
165
+ return results;
166
+ })()`));
167
+ return domResults.slice(0, limit);
168
+ },
169
+ });
@@ -0,0 +1,58 @@
1
+ site: jike
2
+ name: post
3
+ description: 即刻帖子详情及评论
4
+ domain: m.okjike.com
5
+ browser: true
6
+
7
+ args:
8
+ id:
9
+ type: string
10
+ required: true
11
+ description: Post ID (from post URL)
12
+
13
+ pipeline:
14
+ - navigate: https://m.okjike.com/originalPosts/${{ args.id }}
15
+
16
+ # 从 Next.js SSR 内嵌 JSON 中提取帖子和评论
17
+ - evaluate: |
18
+ (() => {
19
+ try {
20
+ const el = document.querySelector('script[type="application/json"]');
21
+ if (!el) return [];
22
+ const data = JSON.parse(el.textContent);
23
+ const pageProps = data?.props?.pageProps || {};
24
+ const post = pageProps.post || {};
25
+ const comments = pageProps.comments || [];
26
+
27
+ const result = [{
28
+ type: 'post',
29
+ author: post.user?.screenName || '',
30
+ content: post.content || '',
31
+ likes: post.likeCount || 0,
32
+ time: post.createdAt || '',
33
+ }];
34
+
35
+ for (const c of comments) {
36
+ result.push({
37
+ type: 'comment',
38
+ author: c.user?.screenName || '',
39
+ content: (c.content || '').replace(/\n/g, ' '),
40
+ likes: c.likeCount || 0,
41
+ time: c.createdAt || '',
42
+ });
43
+ }
44
+
45
+ return result;
46
+ } catch (e) {
47
+ return [];
48
+ }
49
+ })()
50
+
51
+ - map:
52
+ type: ${{ item.type }}
53
+ author: ${{ item.author }}
54
+ content: ${{ item.content }}
55
+ likes: ${{ item.likes }}
56
+ time: ${{ item.time }}
57
+
58
+ columns: [type, author, content, likes, time]
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,103 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ /**
3
+ * 转发即刻帖子
4
+ *
5
+ * 操作栏转发按钮点击后弹出 Popover 菜单,
6
+ * 选择"转发动态"后弹出编辑器弹窗(可添加附言),
7
+ * 再点击"发布"确认转发。
8
+ */
9
+ cli({
10
+ site: 'jike',
11
+ name: 'repost',
12
+ description: '转发即刻帖子',
13
+ domain: 'web.okjike.com',
14
+ strategy: Strategy.UI,
15
+ browser: true,
16
+ args: [
17
+ { name: 'id', type: 'string', required: true, help: '帖子 ID' },
18
+ { name: 'text', type: 'string', required: false, help: '转发附言(可选)' },
19
+ ],
20
+ columns: ['status', 'message'],
21
+ func: async (page, kwargs) => {
22
+ await page.goto(`https://web.okjike.com/originalPost/${kwargs.id}`);
23
+ await page.wait(5);
24
+ // 1. 点击操作栏中的转发按钮(第三个子元素)
25
+ const clickResult = await page.evaluate(`(async () => {
26
+ try {
27
+ const actions = document.querySelector('[class*="_actions_"]');
28
+ if (!actions) return { ok: false, message: '未找到操作栏' };
29
+ const children = Array.from(actions.children).filter(c => c.offsetHeight > 0);
30
+ if (!children[2]) return { ok: false, message: '未找到转发按钮' };
31
+ // 注意:按位置定位,即刻操作栏顺序变化时需调整
32
+ children[2].click();
33
+ return { ok: true };
34
+ } catch (e) {
35
+ return { ok: false, message: e.toString() };
36
+ }
37
+ })()`);
38
+ if (!clickResult.ok) {
39
+ return [{ status: 'failed', message: clickResult.message }];
40
+ }
41
+ await page.wait(1);
42
+ // 2. 在弹出菜单中点击"转发动态"
43
+ const menuResult = await page.evaluate(`(async () => {
44
+ try {
45
+ const btn = Array.from(document.querySelectorAll('button')).find(
46
+ b => b.textContent?.trim() === '转发动态'
47
+ );
48
+ if (!btn) return { ok: false, message: '未找到"转发动态"菜单项' };
49
+ btn.click();
50
+ return { ok: true };
51
+ } catch (e) {
52
+ return { ok: false, message: e.toString() };
53
+ }
54
+ })()`);
55
+ if (!menuResult.ok) {
56
+ return [{ status: 'failed', message: menuResult.message }];
57
+ }
58
+ await page.wait(2);
59
+ // 3. 若有附言,在弹窗编辑器中填入
60
+ if (kwargs.text) {
61
+ const textResult = await page.evaluate(`(async () => {
62
+ try {
63
+ const textToInsert = ${JSON.stringify(kwargs.text)};
64
+ const editor = document.querySelector('[contenteditable="true"]');
65
+ if (!editor) return { ok: false, message: '未找到附言输入框' };
66
+ editor.focus();
67
+ const dt = new DataTransfer();
68
+ dt.setData('text/plain', textToInsert);
69
+ editor.dispatchEvent(new ClipboardEvent('paste', {
70
+ clipboardData: dt, bubbles: true, cancelable: true,
71
+ }));
72
+ await new Promise(r => setTimeout(r, 500));
73
+ return { ok: true };
74
+ } catch(e) { return { ok: false, message: '附言写入失败: ' + e.toString() }; }
75
+ })()`);
76
+ if (!textResult.ok) {
77
+ return [{ status: 'failed', message: textResult.message }];
78
+ }
79
+ }
80
+ // 4. 点击"发送"按钮确认转发
81
+ const confirmResult = await page.evaluate(`(async () => {
82
+ try {
83
+ await new Promise(r => setTimeout(r, 500));
84
+ const btn = Array.from(document.querySelectorAll('button')).find(b => {
85
+ const text = b.textContent?.trim() || '';
86
+ // 不匹配"转发动态",避免重复触发 Popover 菜单项
87
+ return (text === '发送' || text === '发布') && !b.disabled;
88
+ });
89
+ if (!btn) return { ok: false, message: '未找到发送按钮' };
90
+ btn.click();
91
+ return { ok: true, message: '转发成功' };
92
+ } catch (e) {
93
+ return { ok: false, message: e.toString() };
94
+ }
95
+ })()`);
96
+ if (confirmResult.ok)
97
+ await page.wait(3);
98
+ return [{
99
+ status: confirmResult.ok ? 'success' : 'failed',
100
+ message: confirmResult.message,
101
+ }];
102
+ },
103
+ });
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,67 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { getPostDataJs } from './shared.js';
3
+ /**
4
+ * 即刻搜索适配器
5
+ *
6
+ * 策略:直接导航到 web.okjike.com 搜索页,
7
+ * 通过 React fiber 树提取帖子数据。
8
+ */
9
+ cli({
10
+ site: 'jike',
11
+ name: 'search',
12
+ description: '搜索即刻帖子',
13
+ domain: 'web.okjike.com',
14
+ strategy: Strategy.COOKIE,
15
+ browser: true,
16
+ args: [
17
+ { name: 'keyword', type: 'string', required: true },
18
+ { name: 'limit', type: 'int', default: 20 },
19
+ ],
20
+ columns: ['author', 'content', 'likes', 'comments', 'time', 'url'],
21
+ func: async (page, kwargs) => {
22
+ const keyword = kwargs.keyword;
23
+ const limit = kwargs.limit || 20;
24
+ // 1. 直接导航到搜索页
25
+ const encodedKeyword = encodeURIComponent(keyword);
26
+ await page.goto(`https://web.okjike.com/search?q=${encodedKeyword}`);
27
+ await page.wait(5);
28
+ // 2. 通过 React fiber 提取帖子数据
29
+ const extract = async () => {
30
+ return (await page.evaluate(`(() => {
31
+ ${getPostDataJs}
32
+
33
+ const results = [];
34
+ const seen = new Set();
35
+ const elements = document.querySelectorAll('[class*="_post_"], [class*="_postItem_"]');
36
+
37
+ for (const el of elements) {
38
+ const data = getPostData(el);
39
+ if (!data || !data.id || seen.has(data.id)) continue;
40
+ seen.add(data.id);
41
+
42
+ const author = data.user?.screenName || data.target?.user?.screenName || '';
43
+ const content = data.content || data.target?.content || '';
44
+ if (!author && !content) continue;
45
+
46
+ results.push({
47
+ author,
48
+ content: content.replace(/\\n/g, ' ').slice(0, 120),
49
+ likes: data.likeCount || 0,
50
+ comments: data.commentCount || 0,
51
+ time: data.actionTime || data.createdAt || '',
52
+ url: 'https://web.okjike.com/originalPost/' + data.id,
53
+ });
54
+ }
55
+
56
+ return results;
57
+ })()`));
58
+ };
59
+ let posts = await extract();
60
+ // 3. 数量不足时自动滚动加载更多
61
+ if (posts.length < limit) {
62
+ await page.autoScroll({ times: Math.ceil(limit / 10), delayMs: 2000 });
63
+ posts = await extract();
64
+ }
65
+ return posts.slice(0, limit);
66
+ },
67
+ });
@@ -0,0 +1,19 @@
1
+ /**
2
+ * 即刻适配器公共定义
3
+ *
4
+ * JikePost 接口和 getPostData 函数在 feed.ts / search.ts 中复用,
5
+ * 统一维护于此文件避免重复。
6
+ */
7
+ export interface JikePost {
8
+ author: string;
9
+ content: string;
10
+ likes: number;
11
+ comments: number;
12
+ time: string;
13
+ url: string;
14
+ }
15
+ /**
16
+ * 注入浏览器 evaluate 的 JS 函数字符串。
17
+ * 从 React fiber 树中向上最多走 10 层,找到含 id 字段的 props.data。
18
+ */
19
+ export declare const getPostDataJs: string;
@@ -0,0 +1,25 @@
1
+ /**
2
+ * 即刻适配器公共定义
3
+ *
4
+ * JikePost 接口和 getPostData 函数在 feed.ts / search.ts 中复用,
5
+ * 统一维护于此文件避免重复。
6
+ */
7
+ /**
8
+ * 注入浏览器 evaluate 的 JS 函数字符串。
9
+ * 从 React fiber 树中向上最多走 10 层,找到含 id 字段的 props.data。
10
+ */
11
+ export const getPostDataJs = `
12
+ function getPostData(element) {
13
+ for (const key of Object.keys(element)) {
14
+ if (key.startsWith('__reactFiber$') || key.startsWith('__reactInternalInstance$')) {
15
+ let fiber = element[key];
16
+ for (let i = 0; i < 10 && fiber; i++) {
17
+ const props = fiber.memoizedProps || fiber.pendingProps;
18
+ if (props && props.data && props.data.id) return props.data;
19
+ fiber = fiber.return;
20
+ }
21
+ }
22
+ }
23
+ return null;
24
+ }
25
+ `.trim();
@@ -0,0 +1,52 @@
1
+ site: jike
2
+ name: topic
3
+ description: 即刻话题/圈子帖子
4
+ domain: m.okjike.com
5
+ browser: true
6
+
7
+ args:
8
+ id:
9
+ type: string
10
+ required: true
11
+ description: Topic ID (from topic URL, e.g. 553870e8e4b0cafb0a1bef68)
12
+ limit:
13
+ type: int
14
+ default: 20
15
+ description: Number of posts
16
+
17
+ pipeline:
18
+ - navigate: https://m.okjike.com/topics/${{ args.id }}
19
+
20
+ # 从 Next.js SSR 内嵌 JSON 中提取话题帖子
21
+ - evaluate: |
22
+ (() => {
23
+ try {
24
+ const el = document.querySelector('script[type="application/json"]');
25
+ if (!el) return [];
26
+ const data = JSON.parse(el.textContent);
27
+ const pageProps = data?.props?.pageProps || {};
28
+ const posts = pageProps.posts || [];
29
+ return posts.map(p => ({
30
+ content: (p.content || '').replace(/\n/g, ' ').slice(0, 80),
31
+ author: p.user?.screenName || '',
32
+ likes: p.likeCount || 0,
33
+ comments: p.commentCount || 0,
34
+ time: p.actionTime || p.createdAt || '',
35
+ id: p.id || '',
36
+ }));
37
+ } catch (e) {
38
+ return [];
39
+ }
40
+ })()
41
+
42
+ - map:
43
+ content: ${{ item.content }}
44
+ author: ${{ item.author }}
45
+ likes: ${{ item.likes }}
46
+ comments: ${{ item.comments }}
47
+ time: ${{ item.time }}
48
+ url: https://web.okjike.com/originalPost/${{ item.id }}
49
+
50
+ - limit: ${{ args.limit }}
51
+
52
+ columns: [content, author, likes, comments, time, url]
@@ -0,0 +1,51 @@
1
+ site: jike
2
+ name: user
3
+ description: 即刻用户动态
4
+ domain: m.okjike.com
5
+ browser: true
6
+
7
+ args:
8
+ username:
9
+ type: string
10
+ required: true
11
+ description: Username from profile URL (e.g. wenhao1996)
12
+ limit:
13
+ type: int
14
+ default: 20
15
+ description: Number of posts
16
+
17
+ pipeline:
18
+ - navigate: https://m.okjike.com/users/${{ args.username }}
19
+
20
+ # 从 Next.js SSR 内嵌 JSON 中提取用户动态
21
+ - evaluate: |
22
+ (() => {
23
+ try {
24
+ const el = document.querySelector('script[type="application/json"]');
25
+ if (!el) return [];
26
+ const data = JSON.parse(el.textContent);
27
+ const posts = data?.props?.pageProps?.posts || [];
28
+ return posts.map(p => ({
29
+ content: (p.content || '').replace(/\n/g, ' ').slice(0, 80),
30
+ type: p.type === 'ORIGINAL_POST' ? 'post' : p.type === 'REPOST' ? 'repost' : p.type || '',
31
+ likes: p.likeCount || 0,
32
+ comments: p.commentCount || 0,
33
+ time: p.actionTime || p.createdAt || '',
34
+ id: p.id || '',
35
+ }));
36
+ } catch (e) {
37
+ return [];
38
+ }
39
+ })()
40
+
41
+ - map:
42
+ content: ${{ item.content }}
43
+ type: ${{ item.type }}
44
+ likes: ${{ item.likes }}
45
+ comments: ${{ item.comments }}
46
+ time: ${{ item.time }}
47
+ url: https://web.okjike.com/originalPost/${{ item.id }}
48
+
49
+ - limit: ${{ args.limit }}
50
+
51
+ columns: [content, type, likes, comments, time, url]