@jackwener/opencli 1.0.1 → 1.0.4

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 (253) hide show
  1. package/.github/workflows/build-extension.yml +80 -0
  2. package/.github/workflows/ci.yml +6 -6
  3. package/.github/workflows/docs.yml +52 -0
  4. package/.github/workflows/e2e-headed.yml +2 -2
  5. package/.github/workflows/pkg-pr-new.yml +2 -2
  6. package/.github/workflows/release.yml +2 -5
  7. package/.github/workflows/security.yml +2 -2
  8. package/CDP.md +1 -1
  9. package/CDP.zh-CN.md +1 -1
  10. package/README.md +42 -34
  11. package/README.zh-CN.md +42 -34
  12. package/SKILL.md +3 -5
  13. package/dist/browser/cdp.d.ts +42 -0
  14. package/dist/browser/cdp.js +339 -0
  15. package/dist/browser/daemon-client.d.ts +3 -1
  16. package/dist/browser/daemon-client.js +4 -0
  17. package/dist/browser/dom-helpers.d.ts +20 -0
  18. package/dist/browser/dom-helpers.js +109 -0
  19. package/dist/browser/index.d.ts +3 -0
  20. package/dist/browser/index.js +4 -0
  21. package/dist/browser/mcp.d.ts +1 -0
  22. package/dist/browser/mcp.js +10 -5
  23. package/dist/browser/page.d.ts +7 -0
  24. package/dist/browser/page.js +39 -123
  25. package/dist/browser/utils.d.ts +10 -0
  26. package/dist/browser/utils.js +27 -0
  27. package/dist/browser.test.js +49 -1
  28. package/dist/build-manifest.js +3 -1
  29. package/dist/build-manifest.test.js +34 -0
  30. package/dist/capabilityRouting.d.ts +2 -0
  31. package/dist/capabilityRouting.js +30 -0
  32. package/dist/capabilityRouting.test.d.ts +1 -0
  33. package/dist/capabilityRouting.test.js +42 -0
  34. package/dist/chaoxing.d.ts +58 -0
  35. package/dist/chaoxing.js +225 -0
  36. package/dist/chaoxing.test.d.ts +1 -0
  37. package/dist/chaoxing.test.js +45 -0
  38. package/dist/cli-manifest.json +885 -48
  39. package/dist/cli.d.ts +1 -0
  40. package/dist/cli.js +234 -0
  41. package/dist/clis/antigravity/serve.d.ts +14 -0
  42. package/dist/clis/antigravity/serve.js +263 -0
  43. package/dist/clis/bilibili/download.js +4 -14
  44. package/dist/clis/boss/chatlist.d.ts +1 -0
  45. package/dist/clis/boss/chatlist.js +50 -0
  46. package/dist/clis/boss/chatmsg.d.ts +1 -0
  47. package/dist/clis/boss/chatmsg.js +73 -0
  48. package/dist/clis/boss/resume.d.ts +1 -0
  49. package/dist/clis/boss/resume.js +249 -0
  50. package/dist/clis/boss/send.d.ts +1 -0
  51. package/dist/clis/boss/send.js +176 -0
  52. package/dist/clis/chaoxing/assignments.d.ts +1 -0
  53. package/dist/clis/chaoxing/assignments.js +74 -0
  54. package/dist/clis/chaoxing/exams.d.ts +1 -0
  55. package/dist/clis/chaoxing/exams.js +74 -0
  56. package/dist/clis/chatgpt/ask.js +15 -14
  57. package/dist/clis/chatgpt/ax.d.ts +1 -0
  58. package/dist/clis/chatgpt/ax.js +78 -0
  59. package/dist/clis/chatgpt/read.js +5 -6
  60. package/dist/clis/hf/top.d.ts +1 -0
  61. package/dist/clis/hf/top.js +119 -0
  62. package/dist/clis/jike/comment.d.ts +1 -0
  63. package/dist/clis/jike/comment.js +107 -0
  64. package/dist/clis/jike/create.d.ts +1 -0
  65. package/dist/clis/jike/create.js +106 -0
  66. package/dist/clis/jike/feed.d.ts +1 -0
  67. package/dist/clis/jike/feed.js +67 -0
  68. package/dist/clis/jike/like.d.ts +1 -0
  69. package/dist/clis/jike/like.js +61 -0
  70. package/dist/clis/jike/notifications.d.ts +1 -0
  71. package/dist/clis/jike/notifications.js +169 -0
  72. package/dist/clis/jike/post.yaml +58 -0
  73. package/dist/clis/jike/repost.d.ts +1 -0
  74. package/dist/clis/jike/repost.js +103 -0
  75. package/dist/clis/jike/search.d.ts +1 -0
  76. package/dist/clis/jike/search.js +67 -0
  77. package/dist/clis/jike/shared.d.ts +19 -0
  78. package/dist/clis/jike/shared.js +25 -0
  79. package/dist/clis/jike/topic.yaml +52 -0
  80. package/dist/clis/jike/user.yaml +51 -0
  81. package/dist/clis/smzdm/search.js +28 -39
  82. package/dist/clis/stackoverflow/bounties.yaml +29 -0
  83. package/dist/clis/stackoverflow/hot.yaml +28 -0
  84. package/dist/clis/stackoverflow/search.yaml +32 -0
  85. package/dist/clis/stackoverflow/unanswered.yaml +28 -0
  86. package/dist/clis/twitter/download.js +6 -16
  87. package/dist/clis/twitter/post.js +9 -2
  88. package/dist/clis/twitter/search.js +14 -33
  89. package/dist/clis/xiaohongshu/download.d.ts +1 -1
  90. package/dist/clis/xiaohongshu/download.js +4 -4
  91. package/dist/clis/zhihu/download.js +3 -3
  92. package/dist/doctor.d.ts +7 -0
  93. package/dist/doctor.js +16 -0
  94. package/dist/download/index.d.ts +12 -8
  95. package/dist/download/index.js +11 -3
  96. package/dist/download/index.test.d.ts +1 -0
  97. package/dist/download/index.test.js +14 -0
  98. package/dist/engine.js +25 -14
  99. package/dist/explore.d.ts +1 -0
  100. package/dist/explore.js +48 -103
  101. package/dist/generate.js +1 -0
  102. package/dist/interceptor.js +3 -2
  103. package/dist/main.js +4 -193
  104. package/dist/output.d.ts +2 -1
  105. package/dist/output.js +3 -1
  106. package/dist/pipeline/executor.test.js +1 -0
  107. package/dist/pipeline/steps/download.js +14 -18
  108. package/dist/registry.d.ts +4 -3
  109. package/dist/registry.js +5 -2
  110. package/dist/runtime.d.ts +4 -1
  111. package/dist/runtime.js +2 -2
  112. package/dist/scripts/framework.d.ts +4 -0
  113. package/dist/scripts/framework.js +21 -0
  114. package/dist/scripts/interact.d.ts +4 -0
  115. package/dist/scripts/interact.js +20 -0
  116. package/dist/scripts/store.d.ts +9 -0
  117. package/dist/scripts/store.js +44 -0
  118. package/dist/synthesize.js +1 -1
  119. package/dist/types.d.ts +12 -0
  120. package/dist/verify.d.ts +6 -1
  121. package/dist/verify.js +54 -2
  122. package/docs/.vitepress/config.mts +193 -0
  123. package/docs/adapters/browser/apple-podcasts.md +28 -0
  124. package/docs/adapters/browser/bbc.md +26 -0
  125. package/docs/adapters/browser/bilibili.md +38 -0
  126. package/docs/adapters/browser/boss.md +28 -0
  127. package/docs/adapters/browser/coupang.md +28 -0
  128. package/docs/adapters/browser/ctrip.md +27 -0
  129. package/docs/adapters/browser/github.md +26 -0
  130. package/docs/adapters/browser/hackernews.md +26 -0
  131. package/docs/adapters/browser/linkedin.md +27 -0
  132. package/docs/adapters/browser/reddit.md +41 -0
  133. package/docs/adapters/browser/reuters.md +27 -0
  134. package/docs/adapters/browser/smzdm.md +27 -0
  135. package/docs/adapters/browser/twitter.md +47 -0
  136. package/docs/adapters/browser/v2ex.md +32 -0
  137. package/docs/adapters/browser/weibo.md +27 -0
  138. package/docs/adapters/browser/xiaohongshu.md +32 -0
  139. package/docs/adapters/browser/xiaoyuzhou.md +28 -0
  140. package/docs/adapters/browser/xueqiu.md +32 -0
  141. package/docs/adapters/browser/yahoo-finance.md +26 -0
  142. package/docs/adapters/browser/youtube.md +29 -0
  143. package/docs/adapters/browser/zhihu.md +30 -0
  144. package/docs/adapters/desktop/antigravity.md +46 -0
  145. package/docs/adapters/desktop/chatgpt.md +43 -0
  146. package/docs/adapters/desktop/chatwise.md +38 -0
  147. package/docs/adapters/desktop/codex.md +32 -0
  148. package/docs/adapters/desktop/cursor.md +33 -0
  149. package/docs/adapters/desktop/discord.md +28 -0
  150. package/docs/adapters/desktop/feishu.md +20 -0
  151. package/docs/adapters/desktop/neteasemusic.md +31 -0
  152. package/docs/adapters/desktop/notion.md +29 -0
  153. package/docs/adapters/desktop/wechat.md +28 -0
  154. package/docs/adapters/index.md +49 -0
  155. package/docs/advanced/cdp.md +103 -0
  156. package/docs/advanced/download.md +63 -0
  157. package/docs/advanced/electron.md +125 -0
  158. package/docs/advanced/remote-chrome.md +72 -0
  159. package/docs/developer/ai-workflow.md +66 -0
  160. package/docs/developer/architecture.md +90 -0
  161. package/docs/developer/contributing.md +136 -0
  162. package/docs/developer/testing.md +237 -0
  163. package/docs/developer/ts-adapter.md +87 -0
  164. package/docs/developer/yaml-adapter.md +108 -0
  165. package/docs/guide/browser-bridge.md +38 -0
  166. package/docs/guide/getting-started.md +56 -0
  167. package/docs/guide/installation.md +37 -0
  168. package/docs/guide/troubleshooting.md +56 -0
  169. package/docs/index.md +35 -0
  170. package/docs/zh/adapters/index.md +5 -0
  171. package/docs/zh/advanced/cdp.md +3 -0
  172. package/docs/zh/developer/contributing.md +24 -0
  173. package/docs/zh/guide/browser-bridge.md +25 -0
  174. package/docs/zh/guide/getting-started.md +40 -0
  175. package/docs/zh/guide/installation.md +37 -0
  176. package/docs/zh/index.md +29 -0
  177. package/extension/dist/background.js +386 -438
  178. package/extension/manifest.json +2 -2
  179. package/extension/package-lock.json +1156 -0
  180. package/extension/src/background.test.ts +151 -0
  181. package/extension/src/background.ts +124 -53
  182. package/extension/src/protocol.ts +3 -1
  183. package/package.json +7 -3
  184. package/src/browser/cdp.ts +367 -0
  185. package/src/browser/daemon-client.ts +7 -1
  186. package/src/browser/dom-helpers.ts +116 -0
  187. package/src/browser/index.ts +4 -0
  188. package/src/browser/mcp.ts +14 -6
  189. package/src/browser/page.ts +47 -124
  190. package/src/browser/utils.ts +27 -0
  191. package/src/browser.test.ts +56 -0
  192. package/src/build-manifest.test.ts +36 -0
  193. package/src/build-manifest.ts +2 -1
  194. package/src/capabilityRouting.test.ts +47 -0
  195. package/src/capabilityRouting.ts +28 -0
  196. package/src/chaoxing.test.ts +53 -0
  197. package/src/chaoxing.ts +268 -0
  198. package/src/cli.ts +205 -0
  199. package/src/clis/antigravity/SKILL.md +5 -0
  200. package/src/clis/antigravity/serve.ts +329 -0
  201. package/src/clis/bilibili/download.ts +4 -15
  202. package/src/clis/boss/chatlist.ts +50 -0
  203. package/src/clis/boss/chatmsg.ts +70 -0
  204. package/src/clis/boss/resume.ts +262 -0
  205. package/src/clis/boss/send.ts +193 -0
  206. package/src/clis/chaoxing/README.md +36 -0
  207. package/src/clis/chaoxing/README.zh-CN.md +35 -0
  208. package/src/clis/chaoxing/assignments.ts +88 -0
  209. package/src/clis/chaoxing/exams.ts +88 -0
  210. package/src/clis/chatgpt/ask.ts +14 -15
  211. package/src/clis/chatgpt/ax.ts +81 -0
  212. package/src/clis/chatgpt/read.ts +5 -7
  213. package/src/clis/hf/top.ts +141 -0
  214. package/src/clis/jike/comment.ts +113 -0
  215. package/src/clis/jike/create.ts +113 -0
  216. package/src/clis/jike/feed.ts +74 -0
  217. package/src/clis/jike/like.ts +65 -0
  218. package/src/clis/jike/notifications.ts +185 -0
  219. package/src/clis/jike/post.yaml +58 -0
  220. package/src/clis/jike/repost.ts +114 -0
  221. package/src/clis/jike/search.ts +74 -0
  222. package/src/clis/jike/shared.ts +36 -0
  223. package/src/clis/jike/topic.yaml +52 -0
  224. package/src/clis/jike/user.yaml +51 -0
  225. package/src/clis/smzdm/search.ts +30 -39
  226. package/src/clis/stackoverflow/bounties.yaml +29 -0
  227. package/src/clis/stackoverflow/hot.yaml +28 -0
  228. package/src/clis/stackoverflow/search.yaml +32 -0
  229. package/src/clis/stackoverflow/unanswered.yaml +28 -0
  230. package/src/clis/twitter/download.ts +6 -17
  231. package/src/clis/twitter/post.ts +9 -2
  232. package/src/clis/twitter/search.ts +15 -33
  233. package/src/clis/xiaohongshu/download.ts +4 -4
  234. package/src/clis/zhihu/download.ts +3 -3
  235. package/src/doctor.ts +18 -2
  236. package/src/download/index.test.ts +16 -0
  237. package/src/download/index.ts +22 -4
  238. package/src/engine.ts +20 -13
  239. package/src/explore.ts +54 -103
  240. package/src/generate.ts +1 -0
  241. package/src/interceptor.ts +3 -2
  242. package/src/main.ts +4 -180
  243. package/src/output.ts +15 -13
  244. package/src/pipeline/executor.test.ts +1 -0
  245. package/src/pipeline/steps/download.ts +14 -17
  246. package/src/registry.ts +9 -5
  247. package/src/runtime.ts +3 -2
  248. package/src/scripts/framework.ts +20 -0
  249. package/src/scripts/interact.ts +22 -0
  250. package/src/scripts/store.ts +40 -0
  251. package/src/synthesize.ts +1 -1
  252. package/src/types.ts +9 -0
  253. package/src/verify.ts +64 -3
@@ -0,0 +1,268 @@
1
+ /**
2
+ * Chaoxing (学习通) shared helpers.
3
+ *
4
+ * Flow: initSession → getCourses → enterCourse → getTabIframeUrl → navigate → parse DOM
5
+ * Chaoxing has no flat "list all assignments" API; data is behind session-gated
6
+ * course pages loaded as iframes.
7
+ */
8
+
9
+ import type { IPage } from './types.js';
10
+
11
+ // ── Utilities ────────────────────────────────────────────────────────
12
+
13
+ /** Sleep for given milliseconds (anti-scraping delay). */
14
+ export function sleep(ms: number): Promise<void> {
15
+ return new Promise(resolve => setTimeout(resolve, ms));
16
+ }
17
+
18
+ /** Execute a credentialed fetch in the browser context, returning JSON or text. */
19
+ export async function fetchChaoxing(page: IPage, url: string): Promise<any> {
20
+ const urlJs = JSON.stringify(url);
21
+ return page.evaluate(`
22
+ async () => {
23
+ const res = await fetch(${urlJs}, { credentials: "include" });
24
+ const text = await res.text();
25
+ try { return JSON.parse(text); } catch {}
26
+ return text;
27
+ }
28
+ `);
29
+ }
30
+
31
+ /** Format a timestamp (seconds or milliseconds or date string) to YYYY-MM-DD HH:mm. */
32
+ export function formatTimestamp(ts: unknown): string {
33
+ if (ts == null || ts === '' || ts === 0) return '';
34
+ if (typeof ts === 'string' && !/^\d+$/.test(ts.trim())) return ts.trim();
35
+ const num = Number(ts);
36
+ if (Number.isNaN(num) || num <= 0) return String(ts);
37
+ const millis = num > 1e12 ? num : num * 1000;
38
+ const d = new Date(millis);
39
+ if (Number.isNaN(d.getTime())) return String(ts);
40
+ const yyyy = d.getFullYear();
41
+ const mm = String(d.getMonth() + 1).padStart(2, '0');
42
+ const dd = String(d.getDate()).padStart(2, '0');
43
+ const hh = String(d.getHours()).padStart(2, '0');
44
+ const mi = String(d.getMinutes()).padStart(2, '0');
45
+ return `${yyyy}-${mm}-${dd} ${hh}:${mi}`;
46
+ }
47
+
48
+ /** Map numeric work status to Chinese label. */
49
+ export function workStatusLabel(status: unknown): string {
50
+ if (status == null || status === '') return '未知';
51
+ const s = Number(status);
52
+ if (s === 0) return '未交';
53
+ if (s === 1) return '已交';
54
+ if (s === 2) return '已批阅';
55
+ const str = String(status).trim();
56
+ return str || '未知';
57
+ }
58
+
59
+ // ── Course list ──────────────────────────────────────────────────────
60
+
61
+ export interface ChaoxingCourse {
62
+ courseId: string;
63
+ classId: string;
64
+ cpi: string;
65
+ title: string;
66
+ }
67
+
68
+ /** Fetch enrolled course list via backclazzdata JSON API. */
69
+ export async function getCourses(page: IPage): Promise<ChaoxingCourse[]> {
70
+ const resp = await fetchChaoxing(
71
+ page,
72
+ 'https://mooc1-api.chaoxing.com/mycourse/backclazzdata?view=json&rss=1',
73
+ );
74
+ if (!resp || typeof resp !== 'object') return [];
75
+ const channelList: any[] = resp.channelList ?? [];
76
+ const courses: ChaoxingCourse[] = [];
77
+ for (const channel of channelList) {
78
+ const content = channel?.content;
79
+ if (!content) continue;
80
+ const courseData = content.course?.data;
81
+ if (!Array.isArray(courseData)) continue;
82
+ for (const c of courseData) {
83
+ courses.push({
84
+ courseId: String(c.id ?? ''),
85
+ classId: String(content.id ?? ''),
86
+ cpi: String(channel.cpi ?? ''),
87
+ title: String(c.name ?? ''),
88
+ });
89
+ }
90
+ }
91
+ return courses;
92
+ }
93
+
94
+ // ── Session & course entry ───────────────────────────────────────────
95
+
96
+ /** Navigate to the interaction page to establish a Chaoxing session. */
97
+ export async function initSession(page: IPage): Promise<void> {
98
+ await page.goto('https://mooc2-ans.chaoxing.com/mooc2-ans/visit/interaction');
99
+ await page.wait(3);
100
+ }
101
+
102
+ /**
103
+ * Enter a course via stucoursemiddle redirect (establishes course session + enc).
104
+ * After this call the browser is on the course page.
105
+ */
106
+ export async function enterCourse(page: IPage, course: ChaoxingCourse): Promise<void> {
107
+ const url =
108
+ `https://mooc1.chaoxing.com/visit/stucoursemiddle` +
109
+ `?courseid=${course.courseId}&clazzid=${course.classId}&cpi=${course.cpi}&ismooc2=1&v=2`;
110
+ await page.goto(url);
111
+ await page.wait(3);
112
+ }
113
+
114
+ /**
115
+ * On the course page, click a tab (作业 / 考试) and return the iframe src
116
+ * that gets loaded. Returns empty string if the tab is not found.
117
+ */
118
+ export async function getTabIframeUrl(page: IPage, tabName: string): Promise<string> {
119
+ const nameJs = JSON.stringify(tabName);
120
+ const result: any = await page.evaluate(`
121
+ async () => {
122
+ const tabs = document.querySelectorAll('a[data-url]');
123
+ let target = null;
124
+ for (const tab of tabs) {
125
+ if ((tab.innerText || '').trim() === ${nameJs}) { target = tab; break; }
126
+ }
127
+ if (!target) return '';
128
+ target.click();
129
+ await new Promise(r => setTimeout(r, 2000));
130
+ const iframe = document.getElementById('frame_content-hd') || document.querySelector('iframe');
131
+ return iframe?.src || '';
132
+ }
133
+ `);
134
+ return typeof result === 'string' ? result : '';
135
+ }
136
+
137
+ // ── Assignment parsing ───────────────────────────────────────────────
138
+
139
+ export interface AssignmentRow {
140
+ course: string;
141
+ title: string;
142
+ deadline: string;
143
+ status: string;
144
+ score: string;
145
+ }
146
+
147
+ /**
148
+ * Parse assignments from the current page DOM (the 作业列表 page).
149
+ * The page uses `.ulDiv li` items with status/deadline/score info.
150
+ */
151
+ export async function parseAssignmentsFromDom(page: IPage, courseName: string): Promise<AssignmentRow[]> {
152
+ const raw: any[] = await page.evaluate(`
153
+ (() => {
154
+ const items = [];
155
+ // Each assignment is a li or div block; try multiple selectors
156
+ const blocks = document.querySelectorAll('.ulDiv li, .work-list-item, .listContent > div, ul > li');
157
+ for (const block of blocks) {
158
+ const text = (block.innerText || '').trim();
159
+ if (!text || text.length < 3) continue;
160
+ // Skip filter buttons and headers
161
+ if (/^(全部|已完成|未完成|筛选)$/.test(text)) continue;
162
+ items.push(text);
163
+ }
164
+ // Fallback: split body text by common patterns
165
+ if (items.length === 0) {
166
+ const body = (document.body?.innerText || '').trim();
167
+ return [body];
168
+ }
169
+ return items;
170
+ })()
171
+ `) ?? [];
172
+
173
+ const rows: AssignmentRow[] = [];
174
+ for (const text of raw) {
175
+ if (typeof text !== 'string' || text.length < 3) continue;
176
+ // Skip noise
177
+ if (/^(全部|已完成|未完成|筛选|暂无|提交的作业将经过)/.test(text)) continue;
178
+
179
+ const lines = text.split('\n').map((l: string) => l.trim()).filter(Boolean);
180
+ if (!lines.length) continue;
181
+
182
+ // First meaningful line is the title
183
+ const title = lines[0].replace(/\s+/g, ' ').trim();
184
+ if (!title || /^(全部|已完成|未完成|筛选)$/.test(title)) continue;
185
+
186
+ // Extract status: 未交 / 待批阅 / 已完成 / 已批阅
187
+ const statusMatch = text.match(/(未交|待批阅|已完成|已批阅)/);
188
+ const status = statusMatch?.[1] ?? '';
189
+
190
+ // Extract deadline: "剩余XXX" or date pattern
191
+ const remainMatch = text.match(/(剩余[\d天小时分钟秒]+)/);
192
+ const dateMatch = text.match(/(\d{4}[-/.]\d{1,2}[-/.]\d{1,2}(?:\s+\d{1,2}:\d{2})?)/);
193
+ const deadline = remainMatch?.[1] ?? dateMatch?.[1] ?? '';
194
+
195
+ // Extract score (exclude "分钟")
196
+ const scoreMatch = text.match(/(\d+(?:\.\d+)?)\s*分(?!钟)/);
197
+ const score = scoreMatch?.[1] ?? '';
198
+
199
+ rows.push({ course: courseName, title, deadline, status, score });
200
+ }
201
+ return rows;
202
+ }
203
+
204
+ // ── Exam parsing ─────────────────────────────────────────────────────
205
+
206
+ export interface ExamRow {
207
+ course: string;
208
+ title: string;
209
+ start: string;
210
+ end: string;
211
+ status: string;
212
+ score: string;
213
+ }
214
+
215
+ /** Parse exams from the current page DOM (the 考试列表 page). */
216
+ export async function parseExamsFromDom(page: IPage, courseName: string): Promise<ExamRow[]> {
217
+ const raw: any[] = await page.evaluate(`
218
+ (() => {
219
+ const items = [];
220
+ const blocks = document.querySelectorAll('.ulDiv li, .exam-list-item, .listContent > div, ul > li');
221
+ for (const block of blocks) {
222
+ const text = (block.innerText || '').trim();
223
+ if (!text || text.length < 3) continue;
224
+ if (/^(全部|已完成|未完成|筛选|暂无)$/.test(text)) continue;
225
+ items.push(text);
226
+ }
227
+ if (items.length === 0) {
228
+ const body = (document.body?.innerText || '').trim();
229
+ return [body];
230
+ }
231
+ return items;
232
+ })()
233
+ `) ?? [];
234
+
235
+ // Check for "暂无考试"
236
+ if (raw.length === 1 && typeof raw[0] === 'string' && raw[0].includes('暂无考试')) {
237
+ return [];
238
+ }
239
+
240
+ const rows: ExamRow[] = [];
241
+ for (const text of raw) {
242
+ if (typeof text !== 'string' || text.length < 3) continue;
243
+ if (/^(全部|已完成|未完成|筛选|暂无)/.test(text)) continue;
244
+
245
+ const lines = text.split('\n').map((l: string) => l.trim()).filter(Boolean);
246
+ if (!lines.length) continue;
247
+
248
+ const title = lines[0].replace(/\s+/g, ' ').trim();
249
+ if (!title || /^(全部|已完成|未完成|筛选)$/.test(title)) continue;
250
+
251
+ // Extract dates
252
+ const dates = text.match(/\d{4}[-/.]\d{1,2}[-/.]\d{1,2}\s+\d{1,2}:\d{2}/g) ?? [];
253
+ const start = dates[0] ?? '';
254
+ const end = dates[1] ?? '';
255
+
256
+ // Status
257
+ const statusMatch = text.match(/(未开始|进行中|已结束|已完成|未交|待批阅)/);
258
+ let status = statusMatch?.[1] ?? '';
259
+ if (!status && text.includes('剩余')) status = '进行中';
260
+
261
+ // Score (exclude "分钟")
262
+ const scoreMatch = text.match(/(\d+(?:\.\d+)?)\s*分(?!钟)/);
263
+ const score = scoreMatch?.[1] ?? '';
264
+
265
+ rows.push({ course: courseName, title, start, end, status, score });
266
+ }
267
+ return rows;
268
+ }
package/src/cli.ts ADDED
@@ -0,0 +1,205 @@
1
+ import { Command } from 'commander';
2
+ import chalk from 'chalk';
3
+ import { executeCommand } from './engine.js';
4
+ import { Strategy, type CliCommand, fullName, getRegistry, strategyLabel } from './registry.js';
5
+ import { render as renderOutput } from './output.js';
6
+ import { BrowserBridge, CDPBridge } from './browser/index.js';
7
+ import { browserSession, DEFAULT_BROWSER_COMMAND_TIMEOUT, runWithTimeout } from './runtime.js';
8
+ import { PKG_VERSION } from './version.js';
9
+ import { printCompletionScript } from './completion.js';
10
+ import { CliError } from './errors.js';
11
+ import { shouldUseBrowserSession } from './capabilityRouting.js';
12
+
13
+ export function runCli(BUILTIN_CLIS: string, USER_CLIS: string): void {
14
+ const program = new Command();
15
+ program.name('opencli').description('Make any website your CLI. Zero setup. AI-powered.').version(PKG_VERSION);
16
+
17
+ // ── Built-in commands ──────────────────────────────────────────────────────
18
+
19
+ program.command('list').description('List all available CLI commands').option('-f, --format <fmt>', 'Output format: table, json, yaml, md, csv', 'table').option('--json', 'JSON output (deprecated)')
20
+ .action((opts) => {
21
+ const registry = getRegistry();
22
+ const commands = [...registry.values()].sort((a, b) => fullName(a).localeCompare(fullName(b)));
23
+ const rows = commands.map(c => ({
24
+ command: fullName(c),
25
+ site: c.site,
26
+ name: c.name,
27
+ description: c.description,
28
+ strategy: strategyLabel(c),
29
+ browser: c.browser,
30
+ args: c.args.map(a => a.name).join(', '),
31
+ }));
32
+ const fmt = opts.json && opts.format === 'table' ? 'json' : opts.format;
33
+ if (fmt !== 'table') {
34
+ renderOutput(rows, {
35
+ fmt,
36
+ columns: ['command', 'site', 'name', 'description', 'strategy', 'browser', 'args'],
37
+ title: 'opencli/list',
38
+ source: 'opencli list',
39
+ });
40
+ return;
41
+ }
42
+ const sites = new Map<string, CliCommand[]>();
43
+ for (const cmd of commands) { const g = sites.get(cmd.site) ?? []; g.push(cmd); sites.set(cmd.site, g); }
44
+ console.log(); console.log(chalk.bold(' opencli') + chalk.dim(' — available commands')); console.log();
45
+ for (const [site, cmds] of sites) {
46
+ console.log(chalk.bold.cyan(` ${site}`));
47
+ for (const cmd of cmds) { const tag = strategyLabel(cmd) === 'public' ? chalk.green('[public]') : chalk.yellow(`[${strategyLabel(cmd)}]`); console.log(` ${cmd.name} ${tag}${cmd.description ? chalk.dim(` — ${cmd.description}`) : ''}`); }
48
+ console.log();
49
+ }
50
+ console.log(chalk.dim(` ${commands.length} commands across ${sites.size} sites`)); console.log();
51
+ });
52
+
53
+ program.command('validate').description('Validate CLI definitions').argument('[target]', 'site or site/name')
54
+ .action(async (target) => {
55
+ const { validateClisWithTarget, renderValidationReport } = await import('./validate.js');
56
+ console.log(renderValidationReport(validateClisWithTarget([BUILTIN_CLIS, USER_CLIS], target)));
57
+ });
58
+
59
+ program.command('verify').description('Validate + smoke test').argument('[target]').option('--smoke', 'Run smoke tests', false)
60
+ .action(async (target, opts) => {
61
+ const { verifyClis, renderVerifyReport } = await import('./verify.js');
62
+ const r = await verifyClis({ builtinClis: BUILTIN_CLIS, userClis: USER_CLIS, target, smoke: opts.smoke });
63
+ console.log(renderVerifyReport(r));
64
+ process.exitCode = r.ok ? 0 : 1;
65
+ });
66
+
67
+ program.command('explore').alias('probe').description('Explore a website: discover APIs, stores, and recommend strategies').argument('<url>').option('--site <name>').option('--goal <text>').option('--wait <s>', '', '3').option('--auto', 'Enable interactive fuzzing (simulate clicks to trigger lazy APIs)').option('--click <labels>', 'Comma-separated labels to click before fuzzing (e.g. "字幕,CC,评论")')
68
+ .action(async (url, opts) => { const { exploreUrl, renderExploreSummary } = await import('./explore.js'); const clickLabels = opts.click ? opts.click.split(',').map((s: string) => s.trim()) : undefined; const BrowserFactory = process.env.OPENCLI_CDP_ENDPOINT ? CDPBridge : BrowserBridge; const workspace = `explore:${opts.site ?? (() => { try { return new URL(url).host; } catch { return 'default'; } })()}`; console.log(renderExploreSummary(await exploreUrl(url, { BrowserFactory: BrowserFactory as any, site: opts.site, goal: opts.goal, waitSeconds: parseFloat(opts.wait), auto: opts.auto, clickLabels, workspace }))); });
69
+
70
+ program.command('synthesize').description('Synthesize CLIs from explore').argument('<target>').option('--top <n>', '', '3')
71
+ .action(async (target, opts) => { const { synthesizeFromExplore, renderSynthesizeSummary } = await import('./synthesize.js'); console.log(renderSynthesizeSummary(synthesizeFromExplore(target, { top: parseInt(opts.top) }))); });
72
+
73
+ program.command('generate').description('One-shot: explore → synthesize → register').argument('<url>').option('--goal <text>').option('--site <name>')
74
+ .action(async (url, opts) => { const { generateCliFromUrl, renderGenerateSummary } = await import('./generate.js'); const BrowserFactory = process.env.OPENCLI_CDP_ENDPOINT ? CDPBridge : BrowserBridge; const workspace = `generate:${opts.site ?? (() => { try { return new URL(url).host; } catch { return 'default'; } })()}`; const r = await generateCliFromUrl({ url, BrowserFactory: BrowserFactory as any, builtinClis: BUILTIN_CLIS, userClis: USER_CLIS, goal: opts.goal, site: opts.site, workspace }); console.log(renderGenerateSummary(r)); process.exitCode = r.ok ? 0 : 1; });
75
+
76
+ program.command('cascade').description('Strategy cascade: find simplest working strategy').argument('<url>').option('--site <name>')
77
+ .action(async (url, opts) => {
78
+ const { cascadeProbe, renderCascadeResult } = await import('./cascade.js');
79
+ const BrowserFactory = process.env.OPENCLI_CDP_ENDPOINT ? CDPBridge : BrowserBridge;
80
+ const result = await browserSession(BrowserFactory as any, async (page) => {
81
+ // Navigate to the site first for cookie context
82
+ try { const siteUrl = new URL(url); await page.goto(`${siteUrl.protocol}//${siteUrl.host}`); await page.wait(2); } catch {}
83
+ return cascadeProbe(page, url);
84
+ }, { workspace: `cascade:${opts.site ?? (() => { try { return new URL(url).host; } catch { return 'default'; } })()}` });
85
+ console.log(renderCascadeResult(result));
86
+ });
87
+
88
+ program.command('doctor')
89
+ .description('Diagnose opencli browser bridge connectivity')
90
+ .option('--live', 'Test browser connectivity (requires Chrome running)', false)
91
+ .option('--sessions', 'Show active automation sessions', false)
92
+ .action(async (opts) => {
93
+ const { runBrowserDoctor, renderBrowserDoctorReport } = await import('./doctor.js');
94
+ const report = await runBrowserDoctor({ live: opts.live, sessions: opts.sessions, cliVersion: PKG_VERSION });
95
+ console.log(renderBrowserDoctorReport(report));
96
+ });
97
+
98
+ program.command('setup')
99
+ .description('Interactive setup: verify browser bridge connectivity')
100
+ .action(async () => {
101
+ const { runSetup } = await import('./setup.js');
102
+ await runSetup({ cliVersion: PKG_VERSION });
103
+ });
104
+
105
+ program.command('completion')
106
+ .description('Output shell completion script')
107
+ .argument('<shell>', 'Shell type: bash, zsh, or fish')
108
+ .action((shell) => {
109
+ printCompletionScript(shell);
110
+ });
111
+
112
+ // ── Antigravity serve (built-in, long-running) ──────────────────────────────
113
+
114
+ const antigravityCmd = program.command('antigravity').description('antigravity commands');
115
+ antigravityCmd.command('serve')
116
+ .description('Start Anthropic-compatible API proxy for Antigravity')
117
+ .option('--port <port>', 'Server port (default: 8082)', '8082')
118
+ .action(async (opts) => {
119
+ const { startServe } = await import('./clis/antigravity/serve.js');
120
+ await startServe({ port: parseInt(opts.port) });
121
+ });
122
+
123
+ // ── Dynamic site commands ──────────────────────────────────────────────────
124
+
125
+ const registry = getRegistry();
126
+ const siteGroups = new Map<string, Command>();
127
+ // Pre-seed with the antigravity command registered above to avoid duplicates
128
+ siteGroups.set('antigravity', antigravityCmd);
129
+
130
+ for (const [, cmd] of registry) {
131
+ let siteCmd = siteGroups.get(cmd.site);
132
+ if (!siteCmd) { siteCmd = program.command(cmd.site).description(`${cmd.site} commands`); siteGroups.set(cmd.site, siteCmd); }
133
+ const subCmd = siteCmd.command(cmd.name).description(cmd.description);
134
+
135
+ // Register positional args first, then named options
136
+ const positionalArgs: typeof cmd.args = [];
137
+ for (const arg of cmd.args) {
138
+ if (arg.positional) {
139
+ const bracket = arg.required ? `<${arg.name}>` : `[${arg.name}]`;
140
+ subCmd.argument(bracket, arg.help ?? '');
141
+ positionalArgs.push(arg);
142
+ } else {
143
+ const flag = arg.required ? `--${arg.name} <value>` : `--${arg.name} [value]`;
144
+ if (arg.required) subCmd.requiredOption(flag, arg.help ?? '');
145
+ else if (arg.default != null) subCmd.option(flag, arg.help ?? '', String(arg.default));
146
+ else subCmd.option(flag, arg.help ?? '');
147
+ }
148
+ }
149
+ subCmd.option('-f, --format <fmt>', 'Output format: table, json, yaml, md, csv', 'table').option('-v, --verbose', 'Debug output', false);
150
+
151
+ subCmd.action(async (...actionArgs: any[]) => {
152
+ // Commander passes positional args first, then options object, then the Command
153
+ const actionOpts = actionArgs[positionalArgs.length] ?? {};
154
+ const startTime = Date.now();
155
+ const kwargs: Record<string, any> = {};
156
+
157
+ // Collect positional args
158
+ for (let i = 0; i < positionalArgs.length; i++) {
159
+ const arg = positionalArgs[i];
160
+ const v = actionArgs[i];
161
+ if (v !== undefined) kwargs[arg.name] = v;
162
+ }
163
+
164
+ // Collect named options
165
+ for (const arg of cmd.args) {
166
+ if (arg.positional) continue;
167
+ const camelName = arg.name.replace(/-([a-z])/g, (_m, ch: string) => ch.toUpperCase());
168
+ const v = actionOpts[arg.name] ?? actionOpts[camelName];
169
+ if (v !== undefined) kwargs[arg.name] = v;
170
+ }
171
+
172
+ try {
173
+ if (actionOpts.verbose) process.env.OPENCLI_VERBOSE = '1';
174
+ let result: any;
175
+ if (shouldUseBrowserSession(cmd)) {
176
+ const BrowserFactory = process.env.OPENCLI_CDP_ENDPOINT ? CDPBridge : BrowserBridge;
177
+ result = await browserSession(BrowserFactory as any, async (page) => {
178
+ // Cookie/header strategies require same-origin context for credentialed fetch.
179
+ if ((cmd.strategy === Strategy.COOKIE || cmd.strategy === Strategy.HEADER) && cmd.domain) {
180
+ try { await page.goto(`https://${cmd.domain}`); await page.wait(2); } catch {}
181
+ }
182
+ return runWithTimeout(executeCommand(cmd, page, kwargs, actionOpts.verbose), { timeout: cmd.timeoutSeconds ?? DEFAULT_BROWSER_COMMAND_TIMEOUT, label: fullName(cmd) });
183
+ }, { workspace: `site:${cmd.site}` });
184
+ } else { result = await executeCommand(cmd, null, kwargs, actionOpts.verbose); }
185
+ if (actionOpts.verbose && (!result || (Array.isArray(result) && result.length === 0))) {
186
+ console.error(chalk.yellow(`[Verbose] Warning: Command returned an empty result. If the website structural API changed or requires authentication, check the network or update the adapter.`));
187
+ }
188
+ const resolved = getRegistry().get(fullName(cmd)) ?? cmd;
189
+ renderOutput(result, { fmt: actionOpts.format, columns: resolved.columns, title: `${resolved.site}/${resolved.name}`, elapsed: (Date.now() - startTime) / 1000, source: fullName(resolved), footerExtra: resolved.footerExtra?.(kwargs) });
190
+ } catch (err: any) {
191
+ if (err instanceof CliError) {
192
+ console.error(chalk.red(`Error [${err.code}]: ${err.message}`));
193
+ if (err.hint) console.error(chalk.yellow(`Hint: ${err.hint}`));
194
+ } else if (actionOpts.verbose && err.stack) {
195
+ console.error(chalk.red(err.stack));
196
+ } else {
197
+ console.error(chalk.red(`Error: ${err.message ?? err}`));
198
+ }
199
+ process.exitCode = 1;
200
+ }
201
+ });
202
+ }
203
+
204
+ program.parse();
205
+ }
@@ -17,6 +17,11 @@ The agent must configure the endpoint environment variable locally before invoki
17
17
  export OPENCLI_CDP_ENDPOINT="http://127.0.0.1:9224"
18
18
  \`\`\`
19
19
 
20
+ If the endpoint exposes multiple inspectable targets, also set:
21
+ \`\`\`bash
22
+ export OPENCLI_CDP_TARGET="antigravity"
23
+ \`\`\`
24
+
20
25
  ## High-Level Capabilities
21
26
  1. **Send Messages (`opencli antigravity send <message>`)**: Type and send a message directly into the chat UI.
22
27
  2. **Read History (`opencli antigravity read`)**: Scrape the raw chat transcript from the main UI container.