@jackwener/opencli 1.5.5 → 1.5.7

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 (540) hide show
  1. package/CHANGELOG.md +26 -0
  2. package/README.md +31 -4
  3. package/README.zh-CN.md +40 -5
  4. package/SKILL.md +1 -1
  5. package/dist/browser/cdp.d.ts +1 -0
  6. package/dist/browser/cdp.js +30 -27
  7. package/dist/browser/daemon-client.d.ts +11 -1
  8. package/dist/browser/daemon-client.js +3 -0
  9. package/dist/browser/dom-helpers.js +1 -0
  10. package/dist/browser/dom-helpers.test.js +14 -1
  11. package/dist/browser/mcp.js +18 -13
  12. package/dist/browser/page.d.ts +6 -0
  13. package/dist/browser/page.js +37 -2
  14. package/dist/browser/page.test.d.ts +1 -0
  15. package/dist/browser/page.test.js +44 -0
  16. package/dist/browser/stealth.js +198 -0
  17. package/dist/browser/stealth.test.d.ts +1 -0
  18. package/dist/browser/stealth.test.js +134 -0
  19. package/dist/browser.test.js +1 -1
  20. package/dist/build-manifest.d.ts +1 -0
  21. package/dist/build-manifest.js +5 -1
  22. package/dist/build-manifest.test.js +2 -0
  23. package/dist/cli-manifest.json +1821 -252
  24. package/dist/cli.js +20 -3
  25. package/dist/clis/antigravity/serve.d.ts +1 -1
  26. package/dist/clis/antigravity/serve.js +5 -8
  27. package/dist/clis/band/bands.d.ts +1 -0
  28. package/dist/clis/band/bands.js +72 -0
  29. package/dist/clis/band/mentions.d.ts +1 -0
  30. package/dist/clis/band/mentions.js +127 -0
  31. package/dist/clis/band/post.d.ts +1 -0
  32. package/dist/clis/band/post.js +175 -0
  33. package/dist/clis/band/posts.d.ts +1 -0
  34. package/dist/clis/band/posts.js +94 -0
  35. package/dist/clis/bilibili/subtitle.js +4 -0
  36. package/dist/clis/bilibili/subtitle.test.d.ts +1 -0
  37. package/dist/clis/bilibili/subtitle.test.js +48 -0
  38. package/dist/clis/chatwise/ask.js +0 -2
  39. package/dist/clis/chatwise/export.js +0 -2
  40. package/dist/clis/chatwise/history.js +0 -2
  41. package/dist/clis/chatwise/model.js +0 -2
  42. package/dist/clis/chatwise/new.js +1 -2
  43. package/dist/clis/chatwise/read.js +0 -2
  44. package/dist/clis/chatwise/screenshot.js +1 -2
  45. package/dist/clis/chatwise/send.js +0 -2
  46. package/dist/clis/chatwise/status.js +1 -2
  47. package/dist/clis/ctrip/search.d.ts +13 -0
  48. package/dist/clis/ctrip/search.js +73 -48
  49. package/dist/clis/ctrip/search.test.d.ts +1 -0
  50. package/dist/clis/ctrip/search.test.js +64 -0
  51. package/dist/clis/doubao/detail.d.ts +1 -0
  52. package/dist/clis/doubao/detail.js +33 -0
  53. package/dist/clis/doubao/detail.test.d.ts +1 -0
  54. package/dist/clis/doubao/detail.test.js +42 -0
  55. package/dist/clis/doubao/history.d.ts +1 -0
  56. package/dist/clis/doubao/history.js +28 -0
  57. package/dist/clis/doubao/history.test.d.ts +1 -0
  58. package/dist/clis/doubao/history.test.js +37 -0
  59. package/dist/clis/doubao/meeting-summary.d.ts +1 -0
  60. package/dist/clis/doubao/meeting-summary.js +39 -0
  61. package/dist/clis/doubao/meeting-transcript.d.ts +1 -0
  62. package/dist/clis/doubao/meeting-transcript.js +36 -0
  63. package/dist/clis/doubao/utils.d.ts +27 -0
  64. package/dist/clis/doubao/utils.js +317 -0
  65. package/dist/clis/doubao/utils.test.d.ts +1 -0
  66. package/dist/clis/doubao/utils.test.js +24 -0
  67. package/dist/clis/douyin/_shared/public-api.d.ts +33 -0
  68. package/dist/clis/douyin/_shared/public-api.js +29 -0
  69. package/dist/clis/douyin/_shared/sts2.js +8 -2
  70. package/dist/clis/douyin/_shared/sts2.test.d.ts +1 -0
  71. package/dist/clis/douyin/_shared/sts2.test.js +27 -0
  72. package/dist/clis/douyin/activities.js +4 -2
  73. package/dist/clis/douyin/activities.test.js +34 -1
  74. package/dist/clis/douyin/collections.js +1 -1
  75. package/dist/clis/douyin/collections.test.js +24 -2
  76. package/dist/clis/douyin/draft.d.ts +8 -11
  77. package/dist/clis/douyin/draft.js +302 -185
  78. package/dist/clis/douyin/draft.test.d.ts +1 -1
  79. package/dist/clis/douyin/draft.test.js +357 -2
  80. package/dist/clis/douyin/hashtag.js +9 -2
  81. package/dist/clis/douyin/hashtag.test.js +35 -2
  82. package/dist/clis/douyin/profile.js +1 -1
  83. package/dist/clis/douyin/profile.test.js +36 -1
  84. package/dist/clis/douyin/user-videos.d.ts +5 -0
  85. package/dist/clis/douyin/user-videos.js +74 -0
  86. package/dist/clis/douyin/user-videos.test.d.ts +1 -0
  87. package/dist/clis/douyin/user-videos.test.js +108 -0
  88. package/dist/clis/douyin/videos.js +22 -5
  89. package/dist/clis/douyin/videos.test.js +45 -2
  90. package/dist/clis/facebook/search.test.d.ts +5 -0
  91. package/dist/clis/facebook/search.test.js +60 -0
  92. package/dist/clis/facebook/search.yaml +4 -3
  93. package/dist/clis/instagram/download.d.ts +16 -0
  94. package/dist/clis/instagram/download.js +225 -0
  95. package/dist/clis/instagram/download.test.d.ts +1 -0
  96. package/dist/clis/instagram/download.test.js +118 -0
  97. package/dist/clis/notebooklm/bind-current.d.ts +1 -0
  98. package/dist/clis/notebooklm/bind-current.js +29 -0
  99. package/dist/clis/notebooklm/bind-current.test.d.ts +1 -0
  100. package/dist/clis/notebooklm/bind-current.test.js +35 -0
  101. package/dist/clis/notebooklm/binding.test.d.ts +1 -0
  102. package/dist/clis/notebooklm/binding.test.js +44 -0
  103. package/dist/clis/notebooklm/compat.test.d.ts +3 -0
  104. package/dist/clis/notebooklm/compat.test.js +16 -0
  105. package/dist/clis/notebooklm/current.d.ts +1 -0
  106. package/dist/clis/notebooklm/current.js +28 -0
  107. package/dist/clis/notebooklm/get.d.ts +1 -0
  108. package/dist/clis/notebooklm/get.js +37 -0
  109. package/dist/clis/notebooklm/history.d.ts +1 -0
  110. package/dist/clis/notebooklm/history.js +25 -0
  111. package/dist/clis/notebooklm/history.test.d.ts +1 -0
  112. package/dist/clis/notebooklm/history.test.js +58 -0
  113. package/dist/clis/notebooklm/list.d.ts +1 -0
  114. package/dist/clis/notebooklm/list.js +35 -0
  115. package/dist/clis/notebooklm/note-list.d.ts +1 -0
  116. package/dist/clis/notebooklm/note-list.js +28 -0
  117. package/dist/clis/notebooklm/note-list.test.d.ts +1 -0
  118. package/dist/clis/notebooklm/note-list.test.js +56 -0
  119. package/dist/clis/notebooklm/notes-get.d.ts +1 -0
  120. package/dist/clis/notebooklm/notes-get.js +47 -0
  121. package/dist/clis/notebooklm/notes-get.test.d.ts +1 -0
  122. package/dist/clis/notebooklm/notes-get.test.js +72 -0
  123. package/dist/clis/notebooklm/rpc.d.ts +36 -0
  124. package/dist/clis/notebooklm/rpc.js +189 -0
  125. package/dist/clis/notebooklm/rpc.test.d.ts +1 -0
  126. package/dist/clis/notebooklm/rpc.test.js +105 -0
  127. package/dist/clis/notebooklm/shared.d.ts +87 -0
  128. package/dist/clis/notebooklm/shared.js +3 -0
  129. package/dist/clis/notebooklm/source-fulltext.d.ts +1 -0
  130. package/dist/clis/notebooklm/source-fulltext.js +44 -0
  131. package/dist/clis/notebooklm/source-fulltext.test.d.ts +1 -0
  132. package/dist/clis/notebooklm/source-fulltext.test.js +106 -0
  133. package/dist/clis/notebooklm/source-get.d.ts +1 -0
  134. package/dist/clis/notebooklm/source-get.js +40 -0
  135. package/dist/clis/notebooklm/source-get.test.d.ts +1 -0
  136. package/dist/clis/notebooklm/source-get.test.js +84 -0
  137. package/dist/clis/notebooklm/source-guide.d.ts +1 -0
  138. package/dist/clis/notebooklm/source-guide.js +44 -0
  139. package/dist/clis/notebooklm/source-guide.test.d.ts +1 -0
  140. package/dist/clis/notebooklm/source-guide.test.js +104 -0
  141. package/dist/clis/notebooklm/source-list.d.ts +1 -0
  142. package/dist/clis/notebooklm/source-list.js +30 -0
  143. package/dist/clis/notebooklm/status.d.ts +1 -0
  144. package/dist/clis/notebooklm/status.js +31 -0
  145. package/dist/clis/notebooklm/summary.d.ts +1 -0
  146. package/dist/clis/notebooklm/summary.js +30 -0
  147. package/dist/clis/notebooklm/summary.test.d.ts +1 -0
  148. package/dist/clis/notebooklm/summary.test.js +78 -0
  149. package/dist/clis/notebooklm/utils.d.ts +37 -0
  150. package/dist/clis/notebooklm/utils.js +739 -0
  151. package/dist/clis/notebooklm/utils.test.d.ts +1 -0
  152. package/dist/clis/notebooklm/utils.test.js +390 -0
  153. package/dist/clis/ones/common.d.ts +32 -0
  154. package/dist/clis/ones/common.js +144 -0
  155. package/dist/clis/ones/enrich-tasks.d.ts +5 -0
  156. package/dist/clis/ones/enrich-tasks.js +37 -0
  157. package/dist/clis/ones/login.d.ts +1 -0
  158. package/dist/clis/ones/login.js +80 -0
  159. package/dist/clis/ones/logout.d.ts +1 -0
  160. package/dist/clis/ones/logout.js +17 -0
  161. package/dist/clis/ones/me.d.ts +1 -0
  162. package/dist/clis/ones/me.js +30 -0
  163. package/dist/clis/ones/my-tasks.d.ts +1 -0
  164. package/dist/clis/ones/my-tasks.js +120 -0
  165. package/dist/clis/ones/resolve-labels.d.ts +10 -0
  166. package/dist/clis/ones/resolve-labels.js +64 -0
  167. package/dist/clis/ones/task-helpers.d.ts +29 -0
  168. package/dist/clis/ones/task-helpers.js +212 -0
  169. package/dist/clis/ones/task-helpers.test.d.ts +1 -0
  170. package/dist/clis/ones/task-helpers.test.js +12 -0
  171. package/dist/clis/ones/task.d.ts +1 -0
  172. package/dist/clis/ones/task.js +66 -0
  173. package/dist/clis/ones/tasks.d.ts +1 -0
  174. package/dist/clis/ones/tasks.js +79 -0
  175. package/dist/clis/ones/token-info.d.ts +1 -0
  176. package/dist/clis/ones/token-info.js +42 -0
  177. package/dist/clis/ones/worklog.d.ts +11 -0
  178. package/dist/clis/ones/worklog.js +267 -0
  179. package/dist/clis/ones/worklog.test.d.ts +1 -0
  180. package/dist/clis/ones/worklog.test.js +20 -0
  181. package/dist/clis/spotify/spotify.d.ts +1 -0
  182. package/dist/clis/spotify/spotify.js +316 -0
  183. package/dist/clis/spotify/utils.d.ts +21 -0
  184. package/dist/clis/spotify/utils.js +66 -0
  185. package/dist/clis/spotify/utils.test.d.ts +1 -0
  186. package/dist/clis/spotify/utils.test.js +67 -0
  187. package/dist/clis/substack/utils.d.ts +4 -0
  188. package/dist/clis/substack/utils.js +8 -2
  189. package/dist/clis/substack/utils.test.d.ts +1 -0
  190. package/dist/clis/substack/utils.test.js +46 -0
  191. package/dist/clis/tieba/commands.test.d.ts +4 -0
  192. package/dist/clis/tieba/commands.test.js +79 -0
  193. package/dist/clis/tieba/hot.d.ts +1 -0
  194. package/dist/clis/tieba/hot.js +48 -0
  195. package/dist/clis/tieba/posts.d.ts +1 -0
  196. package/dist/clis/tieba/posts.js +85 -0
  197. package/dist/clis/tieba/read.d.ts +1 -0
  198. package/dist/clis/tieba/read.js +140 -0
  199. package/dist/clis/tieba/search.d.ts +1 -0
  200. package/dist/clis/tieba/search.js +108 -0
  201. package/dist/clis/tieba/utils.d.ts +101 -0
  202. package/dist/clis/tieba/utils.js +240 -0
  203. package/dist/clis/tieba/utils.test.d.ts +1 -0
  204. package/dist/clis/tieba/utils.test.js +290 -0
  205. package/dist/clis/v2ex/hot.yaml +4 -1
  206. package/dist/clis/v2ex/latest.yaml +4 -1
  207. package/dist/clis/v2ex/topic.yaml +6 -1
  208. package/dist/clis/weixin/download.d.ts +9 -0
  209. package/dist/clis/weixin/download.js +76 -6
  210. package/dist/clis/weread/book.js +206 -13
  211. package/dist/clis/weread/commands.test.js +331 -0
  212. package/dist/clis/weread/private-api-regression.test.d.ts +1 -0
  213. package/dist/{weread-private-api-regression.test.js → clis/weread/private-api-regression.test.js} +92 -30
  214. package/dist/clis/weread/search-regression.test.d.ts +1 -0
  215. package/dist/clis/weread/search-regression.test.js +407 -0
  216. package/dist/clis/weread/search.js +143 -7
  217. package/dist/clis/weread/shelf.js +13 -95
  218. package/dist/clis/weread/utils.d.ts +56 -0
  219. package/dist/clis/weread/utils.js +234 -7
  220. package/dist/clis/weread/utils.test.js +71 -1
  221. package/dist/clis/xiaohongshu/comments.d.ts +3 -0
  222. package/dist/clis/xiaohongshu/comments.js +76 -17
  223. package/dist/clis/xiaohongshu/comments.test.js +70 -9
  224. package/dist/clis/xiaohongshu/download.d.ts +4 -1
  225. package/dist/clis/xiaohongshu/download.js +83 -22
  226. package/dist/clis/xiaohongshu/download.test.d.ts +1 -0
  227. package/dist/clis/xiaohongshu/download.test.js +75 -0
  228. package/dist/clis/xiaohongshu/note-helpers.d.ts +12 -0
  229. package/dist/clis/xiaohongshu/note-helpers.js +23 -0
  230. package/dist/clis/xiaohongshu/note.d.ts +7 -0
  231. package/dist/clis/xiaohongshu/note.js +76 -0
  232. package/dist/clis/xiaohongshu/note.test.d.ts +1 -0
  233. package/dist/clis/xiaohongshu/note.test.js +136 -0
  234. package/dist/clis/xiaohongshu/publish.d.ts +1 -1
  235. package/dist/clis/xiaohongshu/publish.js +78 -31
  236. package/dist/clis/xiaohongshu/publish.test.js +66 -1
  237. package/dist/clis/xiaohongshu/search.js +9 -0
  238. package/dist/clis/xiaohongshu/search.test.js +10 -4
  239. package/dist/clis/xiaohongshu/user-helpers.d.ts +1 -0
  240. package/dist/clis/xiaohongshu/user-helpers.js +2 -0
  241. package/dist/clis/xiaohongshu/user-helpers.test.js +18 -0
  242. package/dist/clis/xueqiu/comments.d.ts +118 -0
  243. package/dist/clis/xueqiu/comments.js +354 -0
  244. package/dist/clis/xueqiu/comments.test.d.ts +1 -0
  245. package/dist/clis/xueqiu/comments.test.js +696 -0
  246. package/dist/clis/youtube/search.js +57 -17
  247. package/dist/clis/youtube/transcript.js +2 -4
  248. package/dist/clis/youtube/utils.d.ts +9 -0
  249. package/dist/clis/youtube/utils.js +67 -3
  250. package/dist/clis/youtube/utils.test.d.ts +1 -0
  251. package/dist/clis/youtube/utils.test.js +37 -0
  252. package/dist/clis/youtube/video.js +16 -15
  253. package/dist/clis/zhihu/question.js +19 -17
  254. package/dist/clis/zhihu/question.test.d.ts +1 -0
  255. package/dist/clis/zhihu/question.test.js +54 -0
  256. package/dist/clis/zsxq/dynamics.d.ts +1 -0
  257. package/dist/clis/zsxq/dynamics.js +47 -0
  258. package/dist/clis/zsxq/groups.d.ts +1 -0
  259. package/dist/clis/zsxq/groups.js +32 -0
  260. package/dist/clis/zsxq/search.d.ts +1 -0
  261. package/dist/clis/zsxq/search.js +43 -0
  262. package/dist/clis/zsxq/search.test.d.ts +1 -0
  263. package/dist/clis/zsxq/search.test.js +24 -0
  264. package/dist/clis/zsxq/topic.d.ts +1 -0
  265. package/dist/clis/zsxq/topic.js +47 -0
  266. package/dist/clis/zsxq/topic.test.d.ts +1 -0
  267. package/dist/clis/zsxq/topic.test.js +29 -0
  268. package/dist/clis/zsxq/topics.d.ts +1 -0
  269. package/dist/clis/zsxq/topics.js +25 -0
  270. package/dist/clis/zsxq/topics.test.d.ts +1 -0
  271. package/dist/clis/zsxq/topics.test.js +24 -0
  272. package/dist/clis/zsxq/utils.d.ts +97 -0
  273. package/dist/clis/zsxq/utils.js +230 -0
  274. package/dist/commanderAdapter.js +10 -1
  275. package/dist/commanderAdapter.test.js +64 -0
  276. package/dist/commands/daemon.d.ts +9 -0
  277. package/dist/commands/daemon.js +124 -0
  278. package/dist/commands/daemon.test.d.ts +1 -0
  279. package/dist/commands/daemon.test.js +185 -0
  280. package/dist/completion.js +3 -1
  281. package/dist/constants.d.ts +2 -0
  282. package/dist/constants.js +2 -0
  283. package/dist/daemon.d.ts +1 -1
  284. package/dist/daemon.js +25 -14
  285. package/dist/daemon.test.d.ts +1 -0
  286. package/dist/daemon.test.js +65 -0
  287. package/dist/discovery.d.ts +9 -0
  288. package/dist/discovery.js +47 -2
  289. package/dist/electron-apps.d.ts +29 -0
  290. package/dist/electron-apps.js +65 -0
  291. package/dist/electron-apps.test.d.ts +1 -0
  292. package/dist/electron-apps.test.js +43 -0
  293. package/dist/engine.test.js +41 -9
  294. package/dist/execution.js +20 -16
  295. package/dist/external-clis.yaml +17 -0
  296. package/dist/idle-manager.d.ts +19 -0
  297. package/dist/idle-manager.js +54 -0
  298. package/dist/launcher.d.ts +36 -0
  299. package/dist/launcher.js +152 -0
  300. package/dist/launcher.test.d.ts +1 -0
  301. package/dist/launcher.test.js +57 -0
  302. package/dist/main.js +3 -3
  303. package/dist/registry.d.ts +1 -0
  304. package/dist/registry.js +31 -3
  305. package/dist/registry.test.js +13 -0
  306. package/dist/runtime.d.ts +5 -3
  307. package/dist/runtime.js +12 -5
  308. package/dist/serialization.d.ts +1 -0
  309. package/dist/serialization.js +3 -0
  310. package/dist/serialization.test.js +17 -1
  311. package/dist/tui.d.ts +7 -0
  312. package/dist/tui.js +52 -0
  313. package/dist/tui.test.d.ts +1 -0
  314. package/dist/tui.test.js +19 -0
  315. package/dist/types.d.ts +5 -0
  316. package/dist/weixin-download.test.js +14 -0
  317. package/docs/.vitepress/config.mts +4 -0
  318. package/docs/adapters/browser/band.md +63 -0
  319. package/docs/adapters/browser/notebooklm.md +69 -0
  320. package/docs/adapters/browser/ones.md +59 -0
  321. package/docs/adapters/browser/spotify.md +62 -0
  322. package/docs/adapters/browser/tieba.md +45 -0
  323. package/docs/adapters/browser/xiaohongshu.md +19 -10
  324. package/docs/adapters/browser/xueqiu.md +5 -0
  325. package/docs/adapters/browser/zsxq.md +49 -0
  326. package/docs/adapters/index.md +67 -63
  327. package/docs/adapters-doc/ones.md +32 -0
  328. package/docs/guide/browser-bridge.md +12 -0
  329. package/docs/guide/troubleshooting.md +9 -4
  330. package/docs/superpowers/plans/2026-03-31-daemon-lifecycle-redesign.md +857 -0
  331. package/docs/superpowers/specs/2026-03-31-daemon-lifecycle-redesign.md +208 -0
  332. package/docs/zh/guide/browser-bridge.md +12 -0
  333. package/extension/dist/background.js +794 -513
  334. package/extension/src/background.test.ts +202 -2
  335. package/extension/src/background.ts +189 -10
  336. package/extension/src/cdp.ts +54 -0
  337. package/extension/src/protocol.ts +11 -5
  338. package/package.json +1 -1
  339. package/scripts/postinstall.js +16 -0
  340. package/src/browser/cdp.ts +24 -17
  341. package/src/browser/daemon-client.ts +11 -1
  342. package/src/browser/dom-helpers.test.ts +15 -1
  343. package/src/browser/dom-helpers.ts +1 -0
  344. package/src/browser/mcp.ts +18 -13
  345. package/src/browser/page.test.ts +58 -0
  346. package/src/browser/page.ts +34 -2
  347. package/src/browser/stealth.test.ts +153 -0
  348. package/src/browser/stealth.ts +198 -0
  349. package/src/browser.test.ts +1 -1
  350. package/src/build-manifest.test.ts +2 -0
  351. package/src/build-manifest.ts +6 -1
  352. package/src/cli.ts +21 -3
  353. package/src/clis/antigravity/SKILL.md +3 -12
  354. package/src/clis/antigravity/serve.ts +5 -10
  355. package/src/clis/band/bands.ts +76 -0
  356. package/src/clis/band/mentions.ts +134 -0
  357. package/src/clis/band/post.ts +187 -0
  358. package/src/clis/band/posts.ts +106 -0
  359. package/src/clis/bilibili/subtitle.test.ts +60 -0
  360. package/src/clis/bilibili/subtitle.ts +4 -0
  361. package/src/clis/chatwise/ask.ts +0 -2
  362. package/src/clis/chatwise/export.ts +0 -2
  363. package/src/clis/chatwise/history.ts +0 -2
  364. package/src/clis/chatwise/model.ts +0 -2
  365. package/src/clis/chatwise/new.ts +1 -2
  366. package/src/clis/chatwise/read.ts +0 -2
  367. package/src/clis/chatwise/screenshot.ts +1 -2
  368. package/src/clis/chatwise/send.ts +0 -2
  369. package/src/clis/chatwise/status.ts +1 -2
  370. package/src/clis/ctrip/search.test.ts +73 -0
  371. package/src/clis/ctrip/search.ts +97 -47
  372. package/src/clis/doubao/detail.test.ts +53 -0
  373. package/src/clis/doubao/detail.ts +41 -0
  374. package/src/clis/doubao/history.test.ts +45 -0
  375. package/src/clis/doubao/history.ts +32 -0
  376. package/src/clis/doubao/meeting-summary.ts +53 -0
  377. package/src/clis/doubao/meeting-transcript.ts +48 -0
  378. package/src/clis/doubao/utils.test.ts +45 -0
  379. package/src/clis/doubao/utils.ts +371 -0
  380. package/src/clis/douyin/_shared/public-api.ts +84 -0
  381. package/src/clis/douyin/_shared/sts2.test.ts +31 -0
  382. package/src/clis/douyin/_shared/sts2.ts +11 -3
  383. package/src/clis/douyin/activities.test.ts +41 -1
  384. package/src/clis/douyin/activities.ts +12 -3
  385. package/src/clis/douyin/collections.test.ts +35 -2
  386. package/src/clis/douyin/collections.ts +1 -1
  387. package/src/clis/douyin/draft.test.ts +444 -2
  388. package/src/clis/douyin/draft.ts +382 -218
  389. package/src/clis/douyin/hashtag.test.ts +42 -2
  390. package/src/clis/douyin/hashtag.ts +11 -3
  391. package/src/clis/douyin/profile.test.ts +43 -1
  392. package/src/clis/douyin/profile.ts +9 -2
  393. package/src/clis/douyin/user-videos.test.ts +122 -0
  394. package/src/clis/douyin/user-videos.ts +101 -0
  395. package/src/clis/douyin/videos.test.ts +52 -2
  396. package/src/clis/douyin/videos.ts +49 -15
  397. package/src/clis/facebook/search.test.ts +70 -0
  398. package/src/clis/facebook/search.yaml +4 -3
  399. package/src/clis/instagram/download.test.ts +159 -0
  400. package/src/clis/instagram/download.ts +286 -0
  401. package/src/clis/notebooklm/bind-current.test.ts +43 -0
  402. package/src/clis/notebooklm/bind-current.ts +36 -0
  403. package/src/clis/notebooklm/binding.test.ts +53 -0
  404. package/src/clis/notebooklm/compat.test.ts +19 -0
  405. package/src/clis/notebooklm/current.ts +38 -0
  406. package/src/clis/notebooklm/get.ts +53 -0
  407. package/src/clis/notebooklm/history.test.ts +70 -0
  408. package/src/clis/notebooklm/history.ts +36 -0
  409. package/src/clis/notebooklm/list.ts +40 -0
  410. package/src/clis/notebooklm/note-list.test.ts +64 -0
  411. package/src/clis/notebooklm/note-list.ts +42 -0
  412. package/src/clis/notebooklm/notes-get.test.ts +88 -0
  413. package/src/clis/notebooklm/notes-get.ts +67 -0
  414. package/src/clis/notebooklm/rpc.test.ts +126 -0
  415. package/src/clis/notebooklm/rpc.ts +286 -0
  416. package/src/clis/notebooklm/shared.ts +98 -0
  417. package/src/clis/notebooklm/source-fulltext.test.ts +123 -0
  418. package/src/clis/notebooklm/source-fulltext.ts +69 -0
  419. package/src/clis/notebooklm/source-get.test.ts +100 -0
  420. package/src/clis/notebooklm/source-get.ts +60 -0
  421. package/src/clis/notebooklm/source-guide.test.ts +121 -0
  422. package/src/clis/notebooklm/source-guide.ts +69 -0
  423. package/src/clis/notebooklm/source-list.ts +45 -0
  424. package/src/clis/notebooklm/status.ts +34 -0
  425. package/src/clis/notebooklm/summary.test.ts +94 -0
  426. package/src/clis/notebooklm/summary.ts +45 -0
  427. package/src/clis/notebooklm/utils.test.ts +446 -0
  428. package/src/clis/notebooklm/utils.ts +893 -0
  429. package/src/clis/ones/common.ts +187 -0
  430. package/src/clis/ones/enrich-tasks.ts +47 -0
  431. package/src/clis/ones/login.ts +103 -0
  432. package/src/clis/ones/logout.ts +19 -0
  433. package/src/clis/ones/me.ts +34 -0
  434. package/src/clis/ones/my-tasks.ts +148 -0
  435. package/src/clis/ones/resolve-labels.ts +80 -0
  436. package/src/clis/ones/task-helpers.test.ts +14 -0
  437. package/src/clis/ones/task-helpers.ts +214 -0
  438. package/src/clis/ones/task.ts +79 -0
  439. package/src/clis/ones/tasks.ts +92 -0
  440. package/src/clis/ones/token-info.ts +46 -0
  441. package/src/clis/ones/worklog.test.ts +24 -0
  442. package/src/clis/ones/worklog.ts +306 -0
  443. package/src/clis/spotify/spotify.ts +328 -0
  444. package/src/clis/spotify/utils.test.ts +87 -0
  445. package/src/clis/spotify/utils.ts +92 -0
  446. package/src/clis/substack/utils.test.ts +54 -0
  447. package/src/clis/substack/utils.ts +10 -2
  448. package/src/clis/tieba/commands.test.ts +86 -0
  449. package/src/clis/tieba/hot.ts +52 -0
  450. package/src/clis/tieba/posts.ts +108 -0
  451. package/src/clis/tieba/read.ts +158 -0
  452. package/src/clis/tieba/search.ts +119 -0
  453. package/src/clis/tieba/utils.test.ts +322 -0
  454. package/src/clis/tieba/utils.ts +348 -0
  455. package/src/clis/v2ex/hot.yaml +4 -1
  456. package/src/clis/v2ex/latest.yaml +4 -1
  457. package/src/clis/v2ex/topic.yaml +6 -1
  458. package/src/clis/weixin/download.ts +95 -6
  459. package/src/clis/weread/book.ts +256 -13
  460. package/src/clis/weread/commands.test.ts +409 -0
  461. package/src/{weread-private-api-regression.test.ts → clis/weread/private-api-regression.test.ts} +108 -30
  462. package/src/clis/weread/search-regression.test.ts +440 -0
  463. package/src/clis/weread/search.ts +189 -9
  464. package/src/clis/weread/shelf.ts +20 -122
  465. package/src/clis/weread/utils.test.ts +81 -1
  466. package/src/clis/weread/utils.ts +293 -7
  467. package/src/clis/xiaohongshu/comments.test.ts +85 -9
  468. package/src/clis/xiaohongshu/comments.ts +76 -17
  469. package/src/clis/xiaohongshu/download.test.ts +96 -0
  470. package/src/clis/xiaohongshu/download.ts +83 -22
  471. package/src/clis/xiaohongshu/note-helpers.ts +25 -0
  472. package/src/clis/xiaohongshu/note.test.ts +164 -0
  473. package/src/clis/xiaohongshu/note.ts +86 -0
  474. package/src/clis/xiaohongshu/publish.test.ts +79 -1
  475. package/src/clis/xiaohongshu/publish.ts +84 -30
  476. package/src/clis/xiaohongshu/search.test.ts +11 -4
  477. package/src/clis/xiaohongshu/search.ts +13 -0
  478. package/src/clis/xiaohongshu/user-helpers.test.ts +23 -0
  479. package/src/clis/xiaohongshu/user-helpers.ts +4 -0
  480. package/src/clis/xueqiu/comments.test.ts +823 -0
  481. package/src/clis/xueqiu/comments.ts +461 -0
  482. package/src/clis/youtube/search.ts +57 -17
  483. package/src/clis/youtube/transcript.ts +2 -4
  484. package/src/clis/youtube/utils.test.ts +43 -0
  485. package/src/clis/youtube/utils.ts +69 -0
  486. package/src/clis/youtube/video.ts +16 -15
  487. package/src/clis/zhihu/question.test.ts +71 -0
  488. package/src/clis/zhihu/question.ts +27 -15
  489. package/src/clis/zsxq/dynamics.ts +60 -0
  490. package/src/clis/zsxq/groups.ts +41 -0
  491. package/src/clis/zsxq/search.test.ts +29 -0
  492. package/src/clis/zsxq/search.ts +54 -0
  493. package/src/clis/zsxq/topic.test.ts +34 -0
  494. package/src/clis/zsxq/topic.ts +68 -0
  495. package/src/clis/zsxq/topics.test.ts +29 -0
  496. package/src/clis/zsxq/topics.ts +36 -0
  497. package/src/clis/zsxq/utils.ts +351 -0
  498. package/src/commanderAdapter.test.ts +77 -0
  499. package/src/commanderAdapter.ts +8 -1
  500. package/src/commands/daemon.test.ts +238 -0
  501. package/src/commands/daemon.ts +135 -0
  502. package/src/completion.ts +2 -1
  503. package/src/constants.ts +3 -0
  504. package/src/daemon.test.ts +88 -0
  505. package/src/daemon.ts +26 -14
  506. package/src/discovery.ts +52 -2
  507. package/src/electron-apps.test.ts +50 -0
  508. package/src/electron-apps.ts +89 -0
  509. package/src/engine.test.ts +45 -9
  510. package/src/execution.ts +24 -19
  511. package/src/external-clis.yaml +17 -0
  512. package/src/idle-manager.ts +60 -0
  513. package/src/launcher.test.ts +67 -0
  514. package/src/launcher.ts +185 -0
  515. package/src/main.ts +3 -2
  516. package/src/registry.test.ts +15 -0
  517. package/src/registry.ts +32 -3
  518. package/src/runtime.ts +13 -7
  519. package/src/serialization.test.ts +19 -1
  520. package/src/serialization.ts +2 -0
  521. package/src/tui.test.ts +23 -0
  522. package/src/tui.ts +65 -0
  523. package/src/types.ts +5 -0
  524. package/src/weixin-download.test.ts +27 -0
  525. package/tests/e2e/band-auth.test.ts +20 -0
  526. package/tests/e2e/browser-auth-helpers.ts +18 -0
  527. package/tests/e2e/browser-auth.test.ts +35 -47
  528. package/tests/e2e/browser-public-extended.test.ts +6 -2
  529. package/tests/e2e/browser-public.test.ts +288 -0
  530. package/tests/e2e/management.test.ts +1 -1
  531. package/tests/e2e/plugin-management.test.ts +1 -1
  532. package/vitest.config.ts +1 -0
  533. package/chatwise-opencli.ps1 +0 -82
  534. package/dist/clis/chatwise/shared.d.ts +0 -2
  535. package/dist/clis/chatwise/shared.js +0 -6
  536. package/dist/weread-private-api-regression.test.d.ts +0 -1
  537. package/dist/weread-search-regression.test.d.ts +0 -1
  538. package/dist/weread-search-regression.test.js +0 -39
  539. package/src/clis/chatwise/shared.ts +0 -8
  540. package/src/weread-search-regression.test.ts +0 -44
@@ -0,0 +1,316 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { CliError } from '../../errors.js';
3
+ import { readFileSync, writeFileSync, mkdirSync, existsSync } from 'fs';
4
+ import { createServer } from 'http';
5
+ import { homedir } from 'os';
6
+ import { join } from 'path';
7
+ import { exec } from 'child_process';
8
+ import { assertSpotifyCredentialsConfigured, getFirstSpotifyTrack, mapSpotifyTrackResults, parseDotEnv, resolveSpotifyCredentials, } from './utils.js';
9
+ // ── Credentials ───────────────────────────────────────────────────────────────
10
+ // Set SPOTIFY_CLIENT_ID and SPOTIFY_CLIENT_SECRET as environment variables,
11
+ // or place them in ~/.opencli/spotify.env:
12
+ // SPOTIFY_CLIENT_ID=your_id
13
+ // SPOTIFY_CLIENT_SECRET=your_secret
14
+ const ENV_FILE = join(homedir(), '.opencli', 'spotify.env');
15
+ function loadEnv() {
16
+ if (!existsSync(ENV_FILE))
17
+ return {};
18
+ return parseDotEnv(readFileSync(ENV_FILE, 'utf-8'));
19
+ }
20
+ const env = loadEnv();
21
+ const credentials = resolveSpotifyCredentials(env);
22
+ const CLIENT_ID = credentials.clientId;
23
+ const CLIENT_SECRET = credentials.clientSecret;
24
+ const REDIRECT_URI = 'http://127.0.0.1:8888/callback';
25
+ const SCOPES = [
26
+ 'user-read-playback-state',
27
+ 'user-modify-playback-state',
28
+ 'user-read-currently-playing',
29
+ 'playlist-read-private',
30
+ ].join(' ');
31
+ // ── Token storage ─────────────────────────────────────────────────────────────
32
+ const TOKEN_FILE = join(homedir(), '.opencli', 'spotify-tokens.json');
33
+ function loadTokens() {
34
+ try {
35
+ return JSON.parse(readFileSync(TOKEN_FILE, 'utf-8'));
36
+ }
37
+ catch {
38
+ return null;
39
+ }
40
+ }
41
+ function saveTokens(tokens) {
42
+ mkdirSync(join(homedir(), '.opencli'), { recursive: true });
43
+ writeFileSync(TOKEN_FILE, JSON.stringify(tokens, null, 2));
44
+ }
45
+ async function refreshAccessToken(refreshToken) {
46
+ const res = await fetch('https://accounts.spotify.com/api/token', {
47
+ method: 'POST',
48
+ headers: {
49
+ 'Content-Type': 'application/x-www-form-urlencoded',
50
+ Authorization: 'Basic ' + Buffer.from(`${CLIENT_ID}:${CLIENT_SECRET}`).toString('base64'),
51
+ },
52
+ body: new URLSearchParams({ grant_type: 'refresh_token', refresh_token: refreshToken }),
53
+ });
54
+ if (!res.ok) {
55
+ const err = await res.json().catch(() => ({}));
56
+ throw new CliError('REFRESH_FAILED', err?.error_description || `Token refresh failed (${res.status})`);
57
+ }
58
+ const data = await res.json();
59
+ const tokens = {
60
+ access_token: data.access_token,
61
+ refresh_token: data.refresh_token || refreshToken,
62
+ expires_at: Date.now() + data.expires_in * 1000,
63
+ };
64
+ saveTokens(tokens);
65
+ return tokens.access_token;
66
+ }
67
+ async function getToken() {
68
+ const tokens = loadTokens();
69
+ if (!tokens)
70
+ throw new CliError('AUTH_REQUIRED', 'Not authenticated. Run: opencli spotify auth');
71
+ if (!tokens.access_token || !tokens.refresh_token || !(tokens.expires_at > 0)) {
72
+ throw new CliError('AUTH_CORRUPTED', 'Token file is corrupted. Run: opencli spotify auth');
73
+ }
74
+ if (Date.now() > tokens.expires_at - 60_000)
75
+ return refreshAccessToken(tokens.refresh_token);
76
+ return tokens.access_token;
77
+ }
78
+ // ── Spotify API helper ────────────────────────────────────────────────────────
79
+ async function api(method, path, body) {
80
+ const token = await getToken();
81
+ const res = await fetch(`https://api.spotify.com/v1${path}`, {
82
+ method,
83
+ headers: { Authorization: `Bearer ${token}`, 'Content-Type': 'application/json' },
84
+ body: body ? JSON.stringify(body) : undefined,
85
+ });
86
+ if (res.status === 204 || res.status === 202)
87
+ return null;
88
+ if (!res.ok) {
89
+ const err = await res.json().catch(() => ({}));
90
+ throw new CliError('API_ERROR', err?.error?.message || `Spotify API error ${res.status}`);
91
+ }
92
+ return res.json();
93
+ }
94
+ async function findTrackUri(query) {
95
+ const data = await api('GET', `/search?q=${encodeURIComponent(query)}&type=track&limit=1`);
96
+ const track = getFirstSpotifyTrack(data);
97
+ if (!track)
98
+ throw new CliError('EMPTY_RESULT', `No track found for: ${query}`);
99
+ return track;
100
+ }
101
+ function openBrowser(url) {
102
+ const cmd = process.platform === 'win32' ? `start "" "${url}"` : process.platform === 'darwin' ? `open "${url}"` : `xdg-open "${url}"`;
103
+ exec(cmd);
104
+ }
105
+ // ── Commands ──────────────────────────────────────────────────────────────────
106
+ cli({
107
+ site: 'spotify',
108
+ name: 'auth',
109
+ description: 'Authenticate with Spotify (OAuth — run once)',
110
+ strategy: Strategy.PUBLIC,
111
+ browser: false,
112
+ args: [],
113
+ columns: ['status'],
114
+ func: async () => {
115
+ assertSpotifyCredentialsConfigured(credentials, ENV_FILE);
116
+ return new Promise((resolve, reject) => {
117
+ const server = createServer(async (req, res) => {
118
+ try {
119
+ const url = new URL(req.url, 'http://localhost:8888');
120
+ if (url.pathname !== '/callback') {
121
+ res.end();
122
+ return;
123
+ }
124
+ const code = url.searchParams.get('code');
125
+ if (!code) {
126
+ res.end('Missing code');
127
+ return;
128
+ }
129
+ const tokenRes = await fetch('https://accounts.spotify.com/api/token', {
130
+ method: 'POST',
131
+ headers: {
132
+ 'Content-Type': 'application/x-www-form-urlencoded',
133
+ Authorization: 'Basic ' + Buffer.from(`${CLIENT_ID}:${CLIENT_SECRET}`).toString('base64'),
134
+ },
135
+ body: new URLSearchParams({ grant_type: 'authorization_code', code, redirect_uri: REDIRECT_URI }),
136
+ });
137
+ if (!tokenRes.ok) {
138
+ const err = await tokenRes.json().catch(() => ({}));
139
+ server.close();
140
+ reject(new CliError('AUTH_FAILED', err?.error_description || `Token exchange failed (${tokenRes.status})`));
141
+ return;
142
+ }
143
+ const data = await tokenRes.json();
144
+ saveTokens({ access_token: data.access_token, refresh_token: data.refresh_token, expires_at: Date.now() + data.expires_in * 1000 });
145
+ res.writeHead(200, { 'Content-Type': 'text/html' });
146
+ res.end('<h2>Spotify authenticated! You can close this tab.</h2>');
147
+ server.close();
148
+ resolve([{ status: 'Authenticated successfully' }]);
149
+ }
150
+ catch (e) {
151
+ server.close();
152
+ reject(e);
153
+ }
154
+ });
155
+ server.on('error', (e) => {
156
+ if (e.code === 'EADDRINUSE')
157
+ reject(new CliError('PORT_IN_USE', 'Port 8888 is already in use. Stop the other process and retry.'));
158
+ else
159
+ reject(e);
160
+ });
161
+ const timeout = setTimeout(() => { server.close(); reject(new CliError('AUTH_TIMEOUT', 'Authentication timed out after 5 minutes')); }, 5 * 60 * 1000);
162
+ server.listen(8888, () => {
163
+ const authUrl = `https://accounts.spotify.com/authorize?${new URLSearchParams({ client_id: CLIENT_ID, response_type: 'code', redirect_uri: REDIRECT_URI, scope: SCOPES })}`;
164
+ console.log('Opening browser for Spotify login...');
165
+ console.log('If it does not open, visit:', authUrl);
166
+ openBrowser(authUrl);
167
+ });
168
+ server.on('close', () => clearTimeout(timeout));
169
+ });
170
+ },
171
+ });
172
+ cli({
173
+ site: 'spotify',
174
+ name: 'status',
175
+ description: 'Show current playback status',
176
+ strategy: Strategy.PUBLIC,
177
+ browser: false,
178
+ args: [],
179
+ columns: ['track', 'artist', 'album', 'status', 'progress'],
180
+ func: async () => {
181
+ const data = await api('GET', '/me/player');
182
+ if (!data || !data.item)
183
+ return [{ track: 'Nothing playing', artist: '', album: '', status: '', progress: '' }];
184
+ const t = data.item;
185
+ if (t.type !== 'track')
186
+ return [{ track: t.name, artist: '', album: t.show?.name ?? '', status: data.is_playing ? 'playing' : 'paused', progress: '' }];
187
+ const prog = (data.progress_ms ?? 0) / 1000 | 0;
188
+ const dur = t.duration_ms / 1000 | 0;
189
+ const fmt = (s) => `${s / 60 | 0}:${String(s % 60).padStart(2, '0')}`;
190
+ return [{ track: t.name, artist: t.artists.map((a) => a.name).join(', '), album: t.album.name, status: data.is_playing ? 'playing' : 'paused', progress: `${fmt(prog)} / ${fmt(dur)}` }];
191
+ },
192
+ });
193
+ cli({
194
+ site: 'spotify',
195
+ name: 'play',
196
+ description: 'Resume playback or search and play a track/artist',
197
+ strategy: Strategy.PUBLIC,
198
+ browser: false,
199
+ args: [{ name: 'query', type: 'str', default: '', positional: true, help: 'Track or artist to play (optional)' }],
200
+ columns: ['track', 'artist', 'status'],
201
+ func: async (_page, kwargs) => {
202
+ if (kwargs.query) {
203
+ const { uri, name, artist } = await findTrackUri(kwargs.query);
204
+ await api('PUT', '/me/player/play', { uris: [uri] });
205
+ return [{ track: name, artist, status: 'playing' }];
206
+ }
207
+ await api('PUT', '/me/player/play');
208
+ return [{ track: '', artist: '', status: 'resumed' }];
209
+ },
210
+ });
211
+ cli({
212
+ site: 'spotify',
213
+ name: 'pause',
214
+ description: 'Pause playback',
215
+ strategy: Strategy.PUBLIC,
216
+ browser: false,
217
+ args: [],
218
+ columns: ['status'],
219
+ func: async () => { await api('PUT', '/me/player/pause'); return [{ status: 'paused' }]; },
220
+ });
221
+ cli({
222
+ site: 'spotify',
223
+ name: 'next',
224
+ description: 'Skip to next track',
225
+ strategy: Strategy.PUBLIC,
226
+ browser: false,
227
+ args: [],
228
+ columns: ['status'],
229
+ func: async () => { await api('POST', '/me/player/next'); return [{ status: 'skipped to next' }]; },
230
+ });
231
+ cli({
232
+ site: 'spotify',
233
+ name: 'prev',
234
+ description: 'Skip to previous track',
235
+ strategy: Strategy.PUBLIC,
236
+ browser: false,
237
+ args: [],
238
+ columns: ['status'],
239
+ func: async () => { await api('POST', '/me/player/previous'); return [{ status: 'skipped to previous' }]; },
240
+ });
241
+ cli({
242
+ site: 'spotify',
243
+ name: 'volume',
244
+ description: 'Set playback volume (0-100)',
245
+ strategy: Strategy.PUBLIC,
246
+ browser: false,
247
+ args: [{ name: 'level', type: 'int', default: 50, positional: true, required: true, help: 'Volume 0–100' }],
248
+ columns: ['volume'],
249
+ func: async (_page, kwargs) => {
250
+ const level = Math.round(kwargs.level);
251
+ if (level < 0 || level > 100)
252
+ throw new CliError('INVALID_ARGS', 'Volume must be between 0 and 100');
253
+ await api('PUT', `/me/player/volume?volume_percent=${level}`);
254
+ return [{ volume: `${level}%` }];
255
+ },
256
+ });
257
+ cli({
258
+ site: 'spotify',
259
+ name: 'search',
260
+ description: 'Search for tracks',
261
+ strategy: Strategy.PUBLIC,
262
+ browser: false,
263
+ args: [
264
+ { name: 'query', type: 'str', required: true, positional: true, help: 'Search query' },
265
+ { name: 'limit', type: 'int', default: 10, help: 'Number of results (default: 10)' },
266
+ ],
267
+ columns: ['track', 'artist', 'album', 'uri'],
268
+ func: async (_page, kwargs) => {
269
+ const limit = Math.min(50, Math.max(1, Math.round(kwargs.limit)));
270
+ const data = await api('GET', `/search?q=${encodeURIComponent(kwargs.query)}&type=track&limit=${limit}`);
271
+ const results = mapSpotifyTrackResults(data);
272
+ if (!results.length)
273
+ throw new CliError('EMPTY_RESULT', `No results found for: ${kwargs.query}`);
274
+ return results;
275
+ },
276
+ });
277
+ cli({
278
+ site: 'spotify',
279
+ name: 'queue',
280
+ description: 'Add a track to the playback queue',
281
+ strategy: Strategy.PUBLIC,
282
+ browser: false,
283
+ args: [{ name: 'query', type: 'str', required: true, positional: true, help: 'Track to add to queue' }],
284
+ columns: ['track', 'artist', 'status'],
285
+ func: async (_page, kwargs) => {
286
+ const { uri, name, artist } = await findTrackUri(kwargs.query);
287
+ await api('POST', `/me/player/queue?uri=${encodeURIComponent(uri)}`);
288
+ return [{ track: name, artist, status: 'added to queue' }];
289
+ },
290
+ });
291
+ cli({
292
+ site: 'spotify',
293
+ name: 'shuffle',
294
+ description: 'Toggle shuffle on/off',
295
+ strategy: Strategy.PUBLIC,
296
+ browser: false,
297
+ args: [{ name: 'state', type: 'str', default: 'on', positional: true, choices: ['on', 'off'], help: 'on or off' }],
298
+ columns: ['shuffle'],
299
+ func: async (_page, kwargs) => {
300
+ await api('PUT', `/me/player/shuffle?state=${kwargs.state === 'on'}`);
301
+ return [{ shuffle: kwargs.state }];
302
+ },
303
+ });
304
+ cli({
305
+ site: 'spotify',
306
+ name: 'repeat',
307
+ description: 'Set repeat mode (off / track / context)',
308
+ strategy: Strategy.PUBLIC,
309
+ browser: false,
310
+ args: [{ name: 'mode', type: 'str', default: 'context', positional: true, choices: ['off', 'track', 'context'], help: 'off / track / context' }],
311
+ columns: ['repeat'],
312
+ func: async (_page, kwargs) => {
313
+ await api('PUT', `/me/player/repeat?state=${kwargs.mode}`);
314
+ return [{ repeat: kwargs.mode }];
315
+ },
316
+ });
@@ -0,0 +1,21 @@
1
+ export interface SpotifyCredentials {
2
+ clientId: string;
3
+ clientSecret: string;
4
+ }
5
+ export interface SpotifyTrackSummary {
6
+ track: string;
7
+ artist: string;
8
+ album: string;
9
+ uri: string;
10
+ }
11
+ export declare function parseDotEnv(content: string): Record<string, string>;
12
+ export declare function resolveSpotifyCredentials(fileEnv: Record<string, string>, processEnv?: NodeJS.ProcessEnv): SpotifyCredentials;
13
+ export declare function isPlaceholderCredential(value: string | null | undefined): boolean;
14
+ export declare function hasConfiguredSpotifyCredentials(credentials: SpotifyCredentials): boolean;
15
+ export declare function assertSpotifyCredentialsConfigured(credentials: SpotifyCredentials, envFile: string): void;
16
+ export declare function mapSpotifyTrackResults(data: any): SpotifyTrackSummary[];
17
+ export declare function getFirstSpotifyTrack(data: any): {
18
+ uri: string;
19
+ name: string;
20
+ artist: string;
21
+ } | null;
@@ -0,0 +1,66 @@
1
+ import { CliError } from '../../errors.js';
2
+ const SPOTIFY_PLACEHOLDER_PATTERNS = [
3
+ /^your_spotify_client_id_here$/i,
4
+ /^your_spotify_client_secret_here$/i,
5
+ /^your_.+_here$/i,
6
+ ];
7
+ export function parseDotEnv(content) {
8
+ return Object.fromEntries(content
9
+ .split(/\r?\n/)
10
+ .map(line => line.trim())
11
+ .filter(line => line && !line.startsWith('#') && line.includes('='))
12
+ .map(line => {
13
+ const index = line.indexOf('=');
14
+ return [line.slice(0, index).trim(), line.slice(index + 1).trim()];
15
+ }));
16
+ }
17
+ export function resolveSpotifyCredentials(fileEnv, processEnv = process.env) {
18
+ return {
19
+ clientId: processEnv.SPOTIFY_CLIENT_ID || fileEnv.SPOTIFY_CLIENT_ID || '',
20
+ clientSecret: processEnv.SPOTIFY_CLIENT_SECRET || fileEnv.SPOTIFY_CLIENT_SECRET || '',
21
+ };
22
+ }
23
+ export function isPlaceholderCredential(value) {
24
+ const normalized = value?.trim() || '';
25
+ if (!normalized)
26
+ return false;
27
+ return SPOTIFY_PLACEHOLDER_PATTERNS.some(pattern => pattern.test(normalized));
28
+ }
29
+ export function hasConfiguredSpotifyCredentials(credentials) {
30
+ return Boolean(credentials.clientId.trim()) &&
31
+ Boolean(credentials.clientSecret.trim()) &&
32
+ !isPlaceholderCredential(credentials.clientId) &&
33
+ !isPlaceholderCredential(credentials.clientSecret);
34
+ }
35
+ export function assertSpotifyCredentialsConfigured(credentials, envFile) {
36
+ if (hasConfiguredSpotifyCredentials(credentials))
37
+ return;
38
+ throw new CliError('CONFIG', `Missing Spotify credentials.\n\n` +
39
+ `1. Go to https://developer.spotify.com/dashboard and create an app\n` +
40
+ `2. Add ${'http://127.0.0.1:8888/callback'} as a Redirect URI\n` +
41
+ `3. Copy your Client ID and Client Secret\n` +
42
+ `4. Open the file: ${envFile}\n` +
43
+ `5. Fill in SPOTIFY_CLIENT_ID and SPOTIFY_CLIENT_SECRET, then save\n` +
44
+ `6. Run: opencli spotify auth`);
45
+ }
46
+ export function mapSpotifyTrackResults(data) {
47
+ const items = data?.tracks?.items;
48
+ if (!Array.isArray(items))
49
+ return [];
50
+ return items.map((track) => ({
51
+ track: track?.name || '',
52
+ artist: Array.isArray(track?.artists) ? track.artists.map((artist) => artist.name).join(', ') : '',
53
+ album: track?.album?.name || '',
54
+ uri: track?.uri || '',
55
+ }));
56
+ }
57
+ export function getFirstSpotifyTrack(data) {
58
+ const track = mapSpotifyTrackResults(data)[0];
59
+ if (!track)
60
+ return null;
61
+ return {
62
+ uri: track.uri,
63
+ name: track.track,
64
+ artist: track.artist,
65
+ };
66
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,67 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { assertSpotifyCredentialsConfigured, getFirstSpotifyTrack, hasConfiguredSpotifyCredentials, mapSpotifyTrackResults, parseDotEnv, resolveSpotifyCredentials, } from './utils.js';
3
+ describe('spotify utils', () => {
4
+ it('parses dotenv-style credential files', () => {
5
+ const env = parseDotEnv(`
6
+ # Spotify credentials
7
+ SPOTIFY_CLIENT_ID=abc123
8
+ SPOTIFY_CLIENT_SECRET=def456
9
+ `);
10
+ expect(env).toEqual({
11
+ SPOTIFY_CLIENT_ID: 'abc123',
12
+ SPOTIFY_CLIENT_SECRET: 'def456',
13
+ });
14
+ });
15
+ it('prefers explicit process env over file values', () => {
16
+ const credentials = resolveSpotifyCredentials({
17
+ SPOTIFY_CLIENT_ID: 'file-id',
18
+ SPOTIFY_CLIENT_SECRET: 'file-secret',
19
+ }, {
20
+ SPOTIFY_CLIENT_ID: 'env-id',
21
+ SPOTIFY_CLIENT_SECRET: 'env-secret',
22
+ });
23
+ expect(credentials).toEqual({
24
+ clientId: 'env-id',
25
+ clientSecret: 'env-secret',
26
+ });
27
+ });
28
+ it('treats placeholder values as unconfigured credentials', () => {
29
+ expect(hasConfiguredSpotifyCredentials({
30
+ clientId: 'your_spotify_client_id_here',
31
+ clientSecret: 'your_spotify_client_secret_here',
32
+ })).toBe(false);
33
+ });
34
+ it('throws a helpful CONFIG error for empty or placeholder credentials', () => {
35
+ expect(() => assertSpotifyCredentialsConfigured({
36
+ clientId: '',
37
+ clientSecret: '',
38
+ }, '/tmp/spotify.env')).toThrow(/Missing Spotify credentials/);
39
+ expect(() => assertSpotifyCredentialsConfigured({
40
+ clientId: 'your_spotify_client_id_here',
41
+ clientSecret: 'real-secret',
42
+ }, '/tmp/spotify.env')).toThrow(/Fill in SPOTIFY_CLIENT_ID and SPOTIFY_CLIENT_SECRET/);
43
+ });
44
+ it('maps search payloads into stable track summaries', () => {
45
+ const results = mapSpotifyTrackResults({
46
+ tracks: {
47
+ items: [
48
+ {
49
+ name: 'Numb',
50
+ artists: [{ name: 'Linkin Park' }, { name: 'Jay-Z' }],
51
+ album: { name: 'Encore' },
52
+ uri: 'spotify:track:123',
53
+ },
54
+ ],
55
+ },
56
+ });
57
+ expect(results).toEqual([
58
+ {
59
+ track: 'Numb',
60
+ artist: 'Linkin Park, Jay-Z',
61
+ album: 'Encore',
62
+ uri: 'spotify:track:123',
63
+ },
64
+ ]);
65
+ expect(getFirstSpotifyTrack({ tracks: { items: [] } })).toBeNull();
66
+ });
67
+ });
@@ -2,3 +2,7 @@ import type { IPage } from '../../types.js';
2
2
  export declare function buildSubstackBrowseUrl(category?: string): string;
3
3
  export declare function loadSubstackFeed(page: IPage, url: string, limit: number): Promise<any[]>;
4
4
  export declare function loadSubstackArchive(page: IPage, baseUrl: string, limit: number): Promise<any[]>;
5
+ export declare const __test__: {
6
+ FEED_POST_LINK_SELECTOR: string;
7
+ ARCHIVE_POST_LINK_SELECTOR: string;
8
+ };
@@ -1,4 +1,6 @@
1
1
  import { CommandExecutionError } from '../../errors.js';
2
+ const FEED_POST_LINK_SELECTOR = 'a[href*="/home/post/"], a[href*="/p/"]';
3
+ const ARCHIVE_POST_LINK_SELECTOR = 'a[href*="/p/"]';
2
4
  export function buildSubstackBrowseUrl(category) {
3
5
  if (!category || category === 'all')
4
6
  return 'https://substack.com/';
@@ -9,7 +11,7 @@ export async function loadSubstackFeed(page, url, limit) {
9
11
  if (!page)
10
12
  throw new CommandExecutionError('Browser session required for substack feed');
11
13
  await page.goto(url);
12
- await page.wait({ selector: 'article', timeout: 5 });
14
+ await page.wait({ selector: FEED_POST_LINK_SELECTOR, timeout: 5 });
13
15
  const data = await page.evaluate(`
14
16
  (async () => {
15
17
  await new Promise((resolve) => setTimeout(resolve, 3000));
@@ -77,7 +79,7 @@ export async function loadSubstackArchive(page, baseUrl, limit) {
77
79
  if (!page)
78
80
  throw new CommandExecutionError('Browser session required for substack archive');
79
81
  await page.goto(`${baseUrl}/archive`);
80
- await page.wait({ selector: 'article', timeout: 5 });
82
+ await page.wait({ selector: ARCHIVE_POST_LINK_SELECTOR, timeout: 5 });
81
83
  const data = await page.evaluate(`
82
84
  (async () => {
83
85
  await new Promise((resolve) => setTimeout(resolve, 3000));
@@ -128,3 +130,7 @@ export async function loadSubstackArchive(page, baseUrl, limit) {
128
130
  `);
129
131
  return Array.isArray(data) ? data : [];
130
132
  }
133
+ export const __test__ = {
134
+ FEED_POST_LINK_SELECTOR,
135
+ ARCHIVE_POST_LINK_SELECTOR,
136
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,46 @@
1
+ import { describe, expect, it, vi } from 'vitest';
2
+ import { __test__, loadSubstackArchive, loadSubstackFeed } from './utils.js';
3
+ function createPageMock(evaluateResult) {
4
+ return {
5
+ goto: vi.fn().mockResolvedValue(undefined),
6
+ evaluate: vi.fn().mockResolvedValue(evaluateResult),
7
+ snapshot: vi.fn().mockResolvedValue(undefined),
8
+ click: vi.fn().mockResolvedValue(undefined),
9
+ typeText: vi.fn().mockResolvedValue(undefined),
10
+ pressKey: vi.fn().mockResolvedValue(undefined),
11
+ scrollTo: vi.fn().mockResolvedValue(undefined),
12
+ getFormState: vi.fn().mockResolvedValue({}),
13
+ wait: vi.fn().mockResolvedValue(undefined),
14
+ tabs: vi.fn().mockResolvedValue([]),
15
+ closeTab: vi.fn().mockResolvedValue(undefined),
16
+ newTab: vi.fn().mockResolvedValue(undefined),
17
+ selectTab: vi.fn().mockResolvedValue(undefined),
18
+ networkRequests: vi.fn().mockResolvedValue([]),
19
+ consoleMessages: vi.fn().mockResolvedValue([]),
20
+ scroll: vi.fn().mockResolvedValue(undefined),
21
+ autoScroll: vi.fn().mockResolvedValue(undefined),
22
+ installInterceptor: vi.fn().mockResolvedValue(undefined),
23
+ getInterceptedRequests: vi.fn().mockResolvedValue([]),
24
+ getCookies: vi.fn().mockResolvedValue([]),
25
+ screenshot: vi.fn().mockResolvedValue(''),
26
+ waitForCapture: vi.fn().mockResolvedValue(undefined),
27
+ };
28
+ }
29
+ describe('substack utils wait selectors', () => {
30
+ it('waits for both feed link shapes before scraping the feed', async () => {
31
+ const page = createPageMock([]);
32
+ await loadSubstackFeed(page, 'https://substack.com/', 5);
33
+ expect(page.wait).toHaveBeenCalledWith({
34
+ selector: __test__.FEED_POST_LINK_SELECTOR,
35
+ timeout: 5,
36
+ });
37
+ });
38
+ it('waits for archive post links before scraping archive pages', async () => {
39
+ const page = createPageMock([]);
40
+ await loadSubstackArchive(page, 'https://example.substack.com', 5);
41
+ expect(page.wait).toHaveBeenCalledWith({
42
+ selector: __test__.ARCHIVE_POST_LINK_SELECTOR,
43
+ timeout: 5,
44
+ });
45
+ });
46
+ });
@@ -0,0 +1,4 @@
1
+ import './hot.js';
2
+ import './posts.js';
3
+ import './read.js';
4
+ import './search.js';
@@ -0,0 +1,79 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { Strategy, getRegistry } from '../../registry.js';
3
+ import './hot.js';
4
+ import './posts.js';
5
+ import './read.js';
6
+ import './search.js';
7
+ describe('tieba commands', () => {
8
+ it('registers all tieba commands as TypeScript adapters', () => {
9
+ const hot = getRegistry().get('tieba/hot');
10
+ const posts = getRegistry().get('tieba/posts');
11
+ const search = getRegistry().get('tieba/search');
12
+ const read = getRegistry().get('tieba/read');
13
+ expect(hot).toBeDefined();
14
+ expect(posts).toBeDefined();
15
+ expect(search).toBeDefined();
16
+ expect(read).toBeDefined();
17
+ expect(typeof hot?.func).toBe('function');
18
+ expect(typeof posts?.func).toBe('function');
19
+ expect(typeof search?.func).toBe('function');
20
+ expect(typeof read?.func).toBe('function');
21
+ });
22
+ it('keeps the intended browser strategies', () => {
23
+ const hot = getRegistry().get('tieba/hot');
24
+ const posts = getRegistry().get('tieba/posts');
25
+ const search = getRegistry().get('tieba/search');
26
+ const read = getRegistry().get('tieba/read');
27
+ expect(hot?.strategy).toBe(Strategy.PUBLIC);
28
+ expect(posts?.strategy).toBe(Strategy.COOKIE);
29
+ expect(search?.strategy).toBe(Strategy.COOKIE);
30
+ expect(read?.strategy).toBe(Strategy.COOKIE);
31
+ expect(hot?.browser).toBe(true);
32
+ expect(posts?.browser).toBe(true);
33
+ expect(search?.browser).toBe(true);
34
+ expect(read?.browser).toBe(true);
35
+ });
36
+ it('keeps the public limit contract at 20 items for list commands', () => {
37
+ const hot = getRegistry().get('tieba/hot');
38
+ const posts = getRegistry().get('tieba/posts');
39
+ const search = getRegistry().get('tieba/search');
40
+ expect(hot?.args.find((arg) => arg.name === 'limit')?.default).toBe(20);
41
+ expect(posts?.args.find((arg) => arg.name === 'limit')?.default).toBe(20);
42
+ expect(search?.args.find((arg) => arg.name === 'limit')?.default).toBe(20);
43
+ });
44
+ it('rejects tieba read results when navigation lands on the wrong page number', async () => {
45
+ const read = getRegistry().get('tieba/read');
46
+ expect(read).toBeDefined();
47
+ expect(typeof read?.func).toBe('function');
48
+ const run = read?.func;
49
+ if (!run)
50
+ throw new Error('tieba/read did not register a handler');
51
+ const page = {
52
+ goto: async () => undefined,
53
+ evaluate: async () => ({
54
+ pageMeta: {
55
+ pathname: '/p/10163164720',
56
+ pn: '1',
57
+ },
58
+ mainPost: {
59
+ title: '测试帖子',
60
+ author: '作者',
61
+ contentText: '正文',
62
+ structuredText: '',
63
+ visibleTime: '2026-03-29 12:00',
64
+ structuredTime: 0,
65
+ hasMedia: false,
66
+ },
67
+ replies: [],
68
+ }),
69
+ };
70
+ await expect(run(page, {
71
+ id: '10163164720',
72
+ page: 2,
73
+ limit: 5,
74
+ })).rejects.toMatchObject({
75
+ code: 'EMPTY_RESULT',
76
+ hint: expect.stringMatching(/requested page/i),
77
+ });
78
+ });
79
+ });
@@ -0,0 +1 @@
1
+ export {};