@jackwener/opencli 1.3.3 → 1.4.1

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 (680) hide show
  1. package/.github/actions/setup-chrome/action.yml +5 -4
  2. package/.github/pull_request_template.md +3 -1
  3. package/.github/workflows/build-extension.yml +7 -1
  4. package/.github/workflows/ci.yml +46 -6
  5. package/.github/workflows/docs.yml +1 -1
  6. package/.github/workflows/e2e-headed.yml +36 -3
  7. package/.github/workflows/release.yml +1 -1
  8. package/.github/workflows/security.yml +0 -3
  9. package/CHANGELOG.md +78 -0
  10. package/CONTRIBUTING.md +6 -3
  11. package/PRIVACY.md +57 -0
  12. package/README.md +31 -4
  13. package/README.zh-CN.md +31 -4
  14. package/SKILL.md +107 -2
  15. package/TESTING.md +1 -0
  16. package/chatwise-opencli.ps1 +82 -0
  17. package/dist/analysis.d.ts +38 -0
  18. package/dist/analysis.js +166 -0
  19. package/dist/browser/cdp.d.ts +0 -4
  20. package/dist/browser/cdp.js +53 -41
  21. package/dist/browser/cdp.test.d.ts +1 -0
  22. package/dist/browser/cdp.test.js +52 -0
  23. package/dist/browser/dom-snapshot.d.ts +2 -2
  24. package/dist/browser/dom-snapshot.js +54 -1
  25. package/dist/browser/dom-snapshot.test.js +36 -0
  26. package/dist/browser/index.d.ts +2 -2
  27. package/dist/browser/index.js +1 -1
  28. package/dist/browser/mcp.d.ts +0 -2
  29. package/dist/browser/mcp.js +2 -3
  30. package/dist/browser/page.d.ts +4 -3
  31. package/dist/browser/page.js +34 -37
  32. package/dist/browser/stealth.d.ts +0 -2
  33. package/dist/browser/stealth.js +24 -9
  34. package/dist/browser.test.js +2 -2
  35. package/dist/build-manifest.js +15 -9
  36. package/dist/build-manifest.test.js +12 -0
  37. package/dist/cascade.js +4 -2
  38. package/dist/cli-manifest.json +1325 -256
  39. package/dist/cli.js +57 -29
  40. package/dist/clis/_shared/desktop-commands.d.ts +22 -0
  41. package/dist/clis/_shared/desktop-commands.js +108 -0
  42. package/dist/clis/antigravity/serve.js +5 -2
  43. package/dist/clis/apple-podcasts/search.js +2 -1
  44. package/dist/clis/arxiv/search.js +3 -3
  45. package/dist/clis/bbc/news.js +0 -1
  46. package/dist/clis/bilibili/dynamic.test.d.ts +1 -0
  47. package/dist/clis/bilibili/dynamic.test.js +68 -0
  48. package/dist/clis/bilibili/favorite.js +4 -2
  49. package/dist/clis/bilibili/following.js +3 -2
  50. package/dist/clis/bilibili/subtitle.js +8 -7
  51. package/dist/clis/bilibili/utils.js +2 -2
  52. package/dist/clis/boss/batchgreet.js +1 -1
  53. package/dist/clis/boss/chatlist.js +1 -1
  54. package/dist/clis/boss/chatmsg.js +1 -1
  55. package/dist/clis/boss/detail.js +1 -1
  56. package/dist/clis/boss/exchange.js +1 -1
  57. package/dist/clis/boss/greet.js +1 -1
  58. package/dist/clis/boss/invite.js +1 -1
  59. package/dist/clis/boss/joblist.js +1 -1
  60. package/dist/clis/boss/mark.js +4 -3
  61. package/dist/clis/boss/recommend.js +1 -1
  62. package/dist/clis/boss/resume.js +1 -1
  63. package/dist/clis/boss/search.js +1 -1
  64. package/dist/clis/boss/send.js +5 -4
  65. package/dist/clis/boss/stats.js +1 -1
  66. package/dist/clis/chatgpt/ask.js +4 -0
  67. package/dist/clis/chatgpt/new.js +5 -1
  68. package/dist/clis/chatgpt/read.js +5 -1
  69. package/dist/clis/chatgpt/send.js +2 -1
  70. package/dist/clis/chatgpt/status.js +5 -1
  71. package/dist/clis/chatwise/ask.js +8 -2
  72. package/dist/clis/chatwise/export.js +2 -0
  73. package/dist/clis/chatwise/history.js +2 -0
  74. package/dist/clis/chatwise/model.js +8 -3
  75. package/dist/clis/chatwise/new.js +3 -18
  76. package/dist/clis/chatwise/read.js +2 -0
  77. package/dist/clis/chatwise/screenshot.js +3 -27
  78. package/dist/clis/chatwise/send.js +8 -2
  79. package/dist/clis/chatwise/shared.d.ts +2 -0
  80. package/dist/clis/chatwise/shared.js +6 -0
  81. package/dist/clis/chatwise/status.js +3 -22
  82. package/dist/clis/codex/ask.js +6 -2
  83. package/dist/clis/codex/dump.js +2 -25
  84. package/dist/clis/codex/new.js +2 -25
  85. package/dist/clis/codex/screenshot.js +2 -27
  86. package/dist/clis/codex/send.js +6 -4
  87. package/dist/clis/codex/status.js +2 -22
  88. package/dist/clis/ctrip/search.js +0 -1
  89. package/dist/clis/cursor/ask.js +2 -1
  90. package/dist/clis/cursor/composer.js +2 -1
  91. package/dist/clis/cursor/dump.js +2 -25
  92. package/dist/clis/cursor/new.js +2 -18
  93. package/dist/clis/cursor/read.js +2 -1
  94. package/dist/clis/cursor/screenshot.js +1 -30
  95. package/dist/clis/cursor/send.js +2 -1
  96. package/dist/clis/cursor/status.js +2 -21
  97. package/dist/clis/dictionary/examples.yaml +25 -0
  98. package/dist/clis/dictionary/search.yaml +27 -0
  99. package/dist/clis/dictionary/synonyms.yaml +25 -0
  100. package/dist/clis/douban/book-hot.js +1 -1
  101. package/dist/clis/douban/movie-hot.js +1 -1
  102. package/dist/clis/douban/search.js +1 -1
  103. package/dist/clis/douban/utils.d.ts +4 -1
  104. package/dist/clis/douban/utils.js +156 -1
  105. package/dist/clis/doubao/ask.js +1 -1
  106. package/dist/clis/doubao/new.js +1 -1
  107. package/dist/clis/doubao/read.js +1 -1
  108. package/dist/clis/doubao/send.js +1 -1
  109. package/dist/clis/doubao/status.js +1 -1
  110. package/dist/clis/doubao-app/ask.js +1 -1
  111. package/dist/clis/doubao-app/new.js +1 -1
  112. package/dist/clis/doubao-app/read.js +1 -1
  113. package/dist/clis/doubao-app/send.js +1 -1
  114. package/dist/clis/douyin/_shared/browser-fetch.d.ts +10 -0
  115. package/dist/clis/douyin/_shared/browser-fetch.js +30 -0
  116. package/dist/clis/douyin/_shared/browser-fetch.test.d.ts +1 -0
  117. package/dist/clis/douyin/_shared/browser-fetch.test.js +31 -0
  118. package/dist/clis/douyin/_shared/creation-id.d.ts +1 -0
  119. package/dist/clis/douyin/_shared/creation-id.js +5 -0
  120. package/dist/clis/douyin/_shared/creation-id.test.d.ts +1 -0
  121. package/dist/clis/douyin/_shared/creation-id.test.js +22 -0
  122. package/dist/clis/douyin/_shared/imagex-upload.d.ts +20 -0
  123. package/dist/clis/douyin/_shared/imagex-upload.js +53 -0
  124. package/dist/clis/douyin/_shared/imagex-upload.test.d.ts +1 -0
  125. package/dist/clis/douyin/_shared/imagex-upload.test.js +87 -0
  126. package/dist/clis/douyin/_shared/sts2.d.ts +8 -0
  127. package/dist/clis/douyin/_shared/sts2.js +15 -0
  128. package/dist/clis/douyin/_shared/text-extra.d.ts +18 -0
  129. package/dist/clis/douyin/_shared/text-extra.js +15 -0
  130. package/dist/clis/douyin/_shared/text-extra.test.d.ts +1 -0
  131. package/dist/clis/douyin/_shared/text-extra.test.js +37 -0
  132. package/dist/clis/douyin/_shared/timing.d.ts +2 -0
  133. package/dist/clis/douyin/_shared/timing.js +22 -0
  134. package/dist/clis/douyin/_shared/timing.test.d.ts +1 -0
  135. package/dist/clis/douyin/_shared/timing.test.js +28 -0
  136. package/dist/clis/douyin/_shared/tos-upload-short-read.test.d.ts +11 -0
  137. package/dist/clis/douyin/_shared/tos-upload-short-read.test.js +83 -0
  138. package/dist/clis/douyin/_shared/tos-upload.d.ts +53 -0
  139. package/dist/clis/douyin/_shared/tos-upload.js +295 -0
  140. package/dist/clis/douyin/_shared/tos-upload.test.d.ts +1 -0
  141. package/dist/clis/douyin/_shared/tos-upload.test.js +229 -0
  142. package/dist/clis/douyin/_shared/transcode.d.ts +27 -0
  143. package/dist/clis/douyin/_shared/transcode.js +45 -0
  144. package/dist/clis/douyin/_shared/transcode.test.d.ts +1 -0
  145. package/dist/clis/douyin/_shared/transcode.test.js +93 -0
  146. package/dist/clis/douyin/_shared/types.d.ts +26 -0
  147. package/dist/clis/douyin/_shared/types.js +1 -0
  148. package/dist/clis/douyin/activities.d.ts +1 -0
  149. package/dist/clis/douyin/activities.js +20 -0
  150. package/dist/clis/douyin/activities.test.d.ts +1 -0
  151. package/dist/clis/douyin/activities.test.js +22 -0
  152. package/dist/clis/douyin/collections.d.ts +1 -0
  153. package/dist/clis/douyin/collections.js +22 -0
  154. package/dist/clis/douyin/collections.test.d.ts +1 -0
  155. package/dist/clis/douyin/collections.test.js +23 -0
  156. package/dist/clis/douyin/delete.d.ts +1 -0
  157. package/dist/clis/douyin/delete.js +18 -0
  158. package/dist/clis/douyin/delete.test.d.ts +1 -0
  159. package/dist/clis/douyin/delete.test.js +11 -0
  160. package/dist/clis/douyin/draft.d.ts +14 -0
  161. package/dist/clis/douyin/draft.js +237 -0
  162. package/dist/clis/douyin/draft.test.d.ts +1 -0
  163. package/dist/clis/douyin/draft.test.js +11 -0
  164. package/dist/clis/douyin/drafts.d.ts +1 -0
  165. package/dist/clis/douyin/drafts.js +23 -0
  166. package/dist/clis/douyin/drafts.test.d.ts +1 -0
  167. package/dist/clis/douyin/drafts.test.js +11 -0
  168. package/dist/clis/douyin/hashtag.d.ts +1 -0
  169. package/dist/clis/douyin/hashtag.js +45 -0
  170. package/dist/clis/douyin/hashtag.test.d.ts +1 -0
  171. package/dist/clis/douyin/hashtag.test.js +25 -0
  172. package/dist/clis/douyin/location.d.ts +1 -0
  173. package/dist/clis/douyin/location.js +24 -0
  174. package/dist/clis/douyin/location.test.d.ts +1 -0
  175. package/dist/clis/douyin/location.test.js +23 -0
  176. package/dist/clis/douyin/profile.d.ts +1 -0
  177. package/dist/clis/douyin/profile.js +28 -0
  178. package/dist/clis/douyin/profile.test.d.ts +1 -0
  179. package/dist/clis/douyin/profile.test.js +11 -0
  180. package/dist/clis/douyin/publish.d.ts +14 -0
  181. package/dist/clis/douyin/publish.js +288 -0
  182. package/dist/clis/douyin/publish.test.d.ts +1 -0
  183. package/dist/clis/douyin/publish.test.js +38 -0
  184. package/dist/clis/douyin/stats.d.ts +1 -0
  185. package/dist/clis/douyin/stats.js +27 -0
  186. package/dist/clis/douyin/stats.test.d.ts +1 -0
  187. package/dist/clis/douyin/stats.test.js +22 -0
  188. package/dist/clis/douyin/update.d.ts +1 -0
  189. package/dist/clis/douyin/update.js +31 -0
  190. package/dist/clis/douyin/update.test.d.ts +1 -0
  191. package/dist/clis/douyin/update.test.js +11 -0
  192. package/dist/clis/douyin/videos.d.ts +1 -0
  193. package/dist/clis/douyin/videos.js +34 -0
  194. package/dist/clis/douyin/videos.test.d.ts +1 -0
  195. package/dist/clis/douyin/videos.test.js +11 -0
  196. package/dist/clis/grok/ask.d.ts +4 -0
  197. package/dist/clis/grok/ask.js +28 -10
  198. package/dist/clis/grok/ask.test.js +18 -0
  199. package/dist/clis/hackernews/search.yaml +1 -1
  200. package/dist/clis/instagram/search.yaml +2 -1
  201. package/dist/clis/jd/item.d.ts +1 -0
  202. package/dist/clis/jd/item.js +96 -0
  203. package/dist/clis/jd/item.test.d.ts +1 -0
  204. package/dist/clis/jd/item.test.js +28 -0
  205. package/dist/clis/jike/feed.js +1 -1
  206. package/dist/clis/jike/search.js +1 -1
  207. package/dist/clis/linkedin/search.js +5 -4
  208. package/dist/clis/linkedin/timeline.d.ts +21 -0
  209. package/dist/clis/linkedin/timeline.js +503 -0
  210. package/dist/clis/linkedin/timeline.test.d.ts +1 -0
  211. package/dist/clis/linkedin/timeline.test.js +81 -0
  212. package/dist/clis/linux-do/search.yaml +3 -1
  213. package/dist/clis/medium/feed.js +1 -1
  214. package/dist/clis/medium/search.js +2 -2
  215. package/dist/clis/medium/user.js +1 -1
  216. package/dist/clis/medium/{shared.js → utils.js} +2 -1
  217. package/dist/clis/pixiv/detail.yaml +49 -0
  218. package/dist/clis/pixiv/download.d.ts +7 -0
  219. package/dist/clis/pixiv/download.js +78 -0
  220. package/dist/clis/pixiv/download.test.d.ts +1 -0
  221. package/dist/clis/pixiv/download.test.js +87 -0
  222. package/dist/clis/pixiv/illusts.d.ts +8 -0
  223. package/dist/clis/pixiv/illusts.js +65 -0
  224. package/dist/clis/pixiv/illusts.test.d.ts +1 -0
  225. package/dist/clis/pixiv/illusts.test.js +99 -0
  226. package/dist/clis/pixiv/ranking.yaml +53 -0
  227. package/dist/clis/pixiv/search.d.ts +6 -0
  228. package/dist/clis/pixiv/search.js +43 -0
  229. package/dist/clis/pixiv/search.test.d.ts +1 -0
  230. package/dist/clis/pixiv/search.test.js +83 -0
  231. package/dist/clis/pixiv/test-utils.d.ts +12 -0
  232. package/dist/clis/pixiv/test-utils.js +23 -0
  233. package/dist/clis/pixiv/user.yaml +46 -0
  234. package/dist/clis/pixiv/utils.d.ts +27 -0
  235. package/dist/clis/pixiv/utils.js +49 -0
  236. package/dist/clis/reddit/comment.js +2 -1
  237. package/dist/clis/reddit/read.js +4 -3
  238. package/dist/clis/reddit/read.test.d.ts +1 -0
  239. package/dist/clis/reddit/read.test.js +28 -0
  240. package/dist/clis/reddit/save.js +2 -1
  241. package/dist/clis/reddit/saved.js +7 -3
  242. package/dist/clis/reddit/subscribe.js +2 -1
  243. package/dist/clis/reddit/upvote.js +2 -1
  244. package/dist/clis/reddit/upvoted.js +7 -3
  245. package/dist/clis/reuters/search.js +0 -1
  246. package/dist/clis/sinablog/article.js +1 -1
  247. package/dist/clis/sinablog/hot.js +1 -1
  248. package/dist/clis/sinablog/user.js +1 -1
  249. package/dist/clis/substack/feed.js +1 -1
  250. package/dist/clis/substack/publication.js +1 -1
  251. package/dist/clis/substack/search.js +3 -2
  252. package/dist/clis/substack/{shared.js → utils.js} +3 -2
  253. package/dist/clis/tiktok/search.yaml +2 -1
  254. package/dist/clis/twitter/accept.js +2 -1
  255. package/dist/clis/twitter/article.js +4 -1
  256. package/dist/clis/twitter/block.js +2 -1
  257. package/dist/clis/twitter/bookmark.js +2 -1
  258. package/dist/clis/twitter/bookmarks.js +3 -2
  259. package/dist/clis/twitter/delete.js +2 -1
  260. package/dist/clis/twitter/follow.js +2 -1
  261. package/dist/clis/twitter/followers.js +3 -2
  262. package/dist/clis/twitter/following.js +3 -2
  263. package/dist/clis/twitter/hide-reply.js +2 -1
  264. package/dist/clis/twitter/like.js +2 -1
  265. package/dist/clis/twitter/notifications.js +2 -1
  266. package/dist/clis/twitter/post.js +2 -1
  267. package/dist/clis/twitter/profile.js +5 -2
  268. package/dist/clis/twitter/reply-dm.js +2 -1
  269. package/dist/clis/twitter/reply.js +2 -1
  270. package/dist/clis/twitter/search.js +32 -13
  271. package/dist/clis/twitter/search.test.d.ts +1 -0
  272. package/dist/clis/twitter/search.test.js +156 -0
  273. package/dist/clis/twitter/thread.js +2 -2
  274. package/dist/clis/twitter/timeline.js +3 -2
  275. package/dist/clis/twitter/trending.js +3 -2
  276. package/dist/clis/twitter/unblock.js +2 -1
  277. package/dist/clis/twitter/unbookmark.js +2 -1
  278. package/dist/clis/twitter/unfollow.js +2 -1
  279. package/dist/clis/v2ex/daily.js +3 -2
  280. package/dist/clis/v2ex/me.js +3 -2
  281. package/dist/clis/v2ex/notifications.js +4 -4
  282. package/dist/clis/web/read.d.ts +16 -0
  283. package/dist/clis/web/read.js +202 -0
  284. package/dist/clis/weibo/comments.d.ts +1 -0
  285. package/dist/clis/weibo/comments.js +53 -0
  286. package/dist/clis/weibo/feed.d.ts +1 -0
  287. package/dist/clis/weibo/feed.js +56 -0
  288. package/dist/clis/weibo/hot.js +0 -1
  289. package/dist/clis/weibo/me.d.ts +1 -0
  290. package/dist/clis/weibo/me.js +76 -0
  291. package/dist/clis/weibo/post.d.ts +1 -0
  292. package/dist/clis/weibo/post.js +75 -0
  293. package/dist/clis/weibo/user.d.ts +1 -0
  294. package/dist/clis/weibo/user.js +63 -0
  295. package/dist/clis/weibo/utils.d.ts +6 -0
  296. package/dist/clis/weibo/utils.js +30 -0
  297. package/dist/clis/weread/search.js +3 -2
  298. package/dist/clis/xueqiu/danjuan-utils.d.ts +55 -0
  299. package/dist/clis/xueqiu/danjuan-utils.js +126 -0
  300. package/dist/clis/xueqiu/danjuan-utils.test.d.ts +1 -0
  301. package/dist/clis/xueqiu/danjuan-utils.test.js +41 -0
  302. package/dist/clis/xueqiu/fund-holdings.d.ts +1 -0
  303. package/dist/clis/xueqiu/fund-holdings.js +28 -0
  304. package/dist/clis/xueqiu/fund-snapshot.d.ts +1 -0
  305. package/dist/clis/xueqiu/fund-snapshot.js +25 -0
  306. package/dist/clis/xueqiu/search.yaml +2 -1
  307. package/dist/clis/yahoo-finance/quote.js +0 -1
  308. package/dist/clis/youtube/channel.d.ts +1 -0
  309. package/dist/clis/youtube/channel.js +150 -0
  310. package/dist/clis/youtube/comments.d.ts +1 -0
  311. package/dist/clis/youtube/comments.js +95 -0
  312. package/dist/clis/youtube/search.js +0 -1
  313. package/dist/clis/youtube/transcript.js +5 -4
  314. package/dist/clis/youtube/video.js +3 -2
  315. package/dist/clis/zhihu/search.yaml +2 -1
  316. package/dist/daemon.js +7 -3
  317. package/dist/discovery.js +11 -10
  318. package/dist/doctor.js +2 -1
  319. package/dist/download/index.d.ts +4 -12
  320. package/dist/download/index.js +33 -12
  321. package/dist/download/index.test.js +79 -2
  322. package/dist/download/media-download.js +4 -2
  323. package/dist/engine.test.js +76 -4
  324. package/dist/execution.d.ts +1 -9
  325. package/dist/execution.js +56 -46
  326. package/dist/explore.js +12 -111
  327. package/dist/external-clis.yaml +0 -25
  328. package/dist/external.js +7 -5
  329. package/dist/external.test.js +4 -0
  330. package/dist/generate.d.ts +0 -9
  331. package/dist/generate.js +4 -20
  332. package/dist/hooks.d.ts +46 -0
  333. package/dist/hooks.js +56 -0
  334. package/dist/hooks.test.d.ts +4 -0
  335. package/dist/hooks.test.js +92 -0
  336. package/dist/interceptor.js +70 -23
  337. package/dist/main.js +2 -0
  338. package/dist/output.js +12 -6
  339. package/dist/pipeline/executor.js +1 -1
  340. package/dist/pipeline/steps/browser.js +1 -3
  341. package/dist/pipeline/steps/download.js +42 -26
  342. package/dist/pipeline/steps/download.test.d.ts +1 -0
  343. package/dist/pipeline/steps/download.test.js +101 -0
  344. package/dist/pipeline/steps/fetch.js +40 -22
  345. package/dist/pipeline/steps/fetch.test.d.ts +1 -0
  346. package/dist/pipeline/steps/fetch.test.js +123 -0
  347. package/dist/pipeline/steps/transform.js +2 -6
  348. package/dist/pipeline/template.js +66 -52
  349. package/dist/pipeline/template.test.js +28 -0
  350. package/dist/pipeline/transform.test.js +18 -0
  351. package/dist/plugin.d.ts +40 -1
  352. package/dist/plugin.js +214 -17
  353. package/dist/plugin.test.d.ts +1 -1
  354. package/dist/plugin.test.js +219 -3
  355. package/dist/record.js +6 -98
  356. package/dist/registry-api.d.ts +2 -0
  357. package/dist/registry-api.js +1 -0
  358. package/dist/registry.d.ts +5 -2
  359. package/dist/registry.js +1 -2
  360. package/dist/runtime.d.ts +0 -1
  361. package/dist/runtime.js +14 -4
  362. package/dist/snapshotFormatter.d.ts +7 -14
  363. package/dist/snapshotFormatter.js +38 -78
  364. package/dist/utils.d.ts +9 -0
  365. package/dist/utils.js +29 -0
  366. package/dist/validate.js +3 -5
  367. package/dist/weread-search-regression.test.d.ts +1 -0
  368. package/dist/weread-search-regression.test.js +39 -0
  369. package/dist/yaml-schema.d.ts +26 -0
  370. package/dist/yaml-schema.js +5 -0
  371. package/docs/.vitepress/config.mts +16 -0
  372. package/docs/adapters/browser/dictionary.md +27 -0
  373. package/docs/adapters/browser/douyin.md +75 -0
  374. package/docs/adapters/browser/jd.md +27 -0
  375. package/docs/adapters/browser/linkedin.md +6 -0
  376. package/docs/adapters/browser/pixiv.md +92 -0
  377. package/docs/adapters/browser/twitter.md +6 -0
  378. package/docs/adapters/browser/web.md +30 -0
  379. package/docs/adapters/browser/xueqiu.md +27 -9
  380. package/docs/adapters/index.md +9 -2
  381. package/docs/comparison.md +125 -0
  382. package/docs/developer/contributing.md +21 -2
  383. package/docs/developer/testing.md +14 -8
  384. package/docs/developer/ts-adapter.md +18 -0
  385. package/docs/developer/yaml-adapter.md +16 -0
  386. package/docs/guide/plugins.md +10 -0
  387. package/docs/zh/guide/plugins.md +10 -0
  388. package/extension/dist/background.js +100 -35
  389. package/extension/manifest.json +6 -2
  390. package/extension/package.json +1 -1
  391. package/extension/popup.html +84 -0
  392. package/extension/popup.js +25 -0
  393. package/extension/src/background.test.ts +46 -1
  394. package/extension/src/background.ts +128 -34
  395. package/extension/src/cdp.ts +9 -9
  396. package/package.json +3 -2
  397. package/scripts/check-doc-coverage.sh +2 -0
  398. package/src/analysis.ts +170 -0
  399. package/src/browser/cdp.test.ts +66 -0
  400. package/src/browser/cdp.ts +59 -44
  401. package/src/browser/dom-snapshot.test.ts +42 -0
  402. package/src/browser/dom-snapshot.ts +56 -3
  403. package/src/browser/index.ts +2 -2
  404. package/src/browser/mcp.ts +2 -4
  405. package/src/browser/page.ts +34 -37
  406. package/src/browser/stealth.ts +24 -10
  407. package/src/browser.test.ts +2 -2
  408. package/src/build-manifest.test.ts +14 -0
  409. package/src/build-manifest.ts +13 -31
  410. package/src/cascade.ts +5 -3
  411. package/src/cli.ts +66 -34
  412. package/src/clis/_shared/desktop-commands.ts +121 -0
  413. package/src/clis/antigravity/serve.ts +6 -3
  414. package/src/clis/apple-podcasts/search.ts +2 -1
  415. package/src/clis/arxiv/search.ts +3 -3
  416. package/src/clis/bbc/news.ts +0 -1
  417. package/src/clis/bilibili/dynamic.test.ts +79 -0
  418. package/src/clis/bilibili/favorite.ts +5 -2
  419. package/src/clis/bilibili/following.ts +3 -2
  420. package/src/clis/bilibili/subtitle.ts +8 -7
  421. package/src/clis/bilibili/utils.ts +2 -2
  422. package/src/clis/boss/batchgreet.ts +1 -1
  423. package/src/clis/boss/chatlist.ts +1 -1
  424. package/src/clis/boss/chatmsg.ts +1 -1
  425. package/src/clis/boss/detail.ts +1 -1
  426. package/src/clis/boss/exchange.ts +1 -1
  427. package/src/clis/boss/greet.ts +1 -1
  428. package/src/clis/boss/invite.ts +1 -1
  429. package/src/clis/boss/joblist.ts +1 -1
  430. package/src/clis/boss/mark.ts +4 -3
  431. package/src/clis/boss/recommend.ts +1 -1
  432. package/src/clis/boss/resume.ts +1 -1
  433. package/src/clis/boss/search.ts +1 -1
  434. package/src/clis/boss/send.ts +5 -4
  435. package/src/clis/boss/stats.ts +1 -1
  436. package/src/clis/chatgpt/ask.ts +5 -0
  437. package/src/clis/chatgpt/new.ts +7 -2
  438. package/src/clis/chatgpt/read.ts +7 -2
  439. package/src/clis/chatgpt/send.ts +3 -2
  440. package/src/clis/chatgpt/status.ts +6 -1
  441. package/src/clis/chatwise/ask.ts +7 -2
  442. package/src/clis/chatwise/export.ts +2 -0
  443. package/src/clis/chatwise/history.ts +2 -0
  444. package/src/clis/chatwise/model.ts +7 -3
  445. package/src/clis/chatwise/new.ts +3 -20
  446. package/src/clis/chatwise/read.ts +2 -0
  447. package/src/clis/chatwise/screenshot.ts +3 -32
  448. package/src/clis/chatwise/send.ts +7 -2
  449. package/src/clis/chatwise/shared.ts +8 -0
  450. package/src/clis/chatwise/status.ts +3 -24
  451. package/src/clis/codex/ask.ts +5 -2
  452. package/src/clis/codex/dump.ts +2 -27
  453. package/src/clis/codex/new.ts +2 -28
  454. package/src/clis/codex/screenshot.ts +2 -32
  455. package/src/clis/codex/send.ts +5 -4
  456. package/src/clis/codex/status.ts +2 -24
  457. package/src/clis/ctrip/search.ts +0 -1
  458. package/src/clis/cursor/ask.ts +2 -1
  459. package/src/clis/cursor/composer.ts +2 -1
  460. package/src/clis/cursor/dump.ts +2 -27
  461. package/src/clis/cursor/new.ts +2 -20
  462. package/src/clis/cursor/read.ts +2 -1
  463. package/src/clis/cursor/screenshot.ts +1 -36
  464. package/src/clis/cursor/send.ts +2 -1
  465. package/src/clis/cursor/status.ts +2 -22
  466. package/src/clis/dictionary/examples.yaml +25 -0
  467. package/src/clis/dictionary/search.yaml +27 -0
  468. package/src/clis/dictionary/synonyms.yaml +25 -0
  469. package/src/clis/douban/book-hot.ts +1 -1
  470. package/src/clis/douban/movie-hot.ts +1 -1
  471. package/src/clis/douban/search.ts +1 -1
  472. package/src/clis/douban/utils.ts +165 -1
  473. package/src/clis/doubao/ask.ts +1 -1
  474. package/src/clis/doubao/new.ts +1 -1
  475. package/src/clis/doubao/read.ts +1 -1
  476. package/src/clis/doubao/send.ts +1 -1
  477. package/src/clis/doubao/status.ts +1 -1
  478. package/src/clis/doubao-app/ask.ts +1 -1
  479. package/src/clis/doubao-app/new.ts +1 -1
  480. package/src/clis/doubao-app/read.ts +1 -1
  481. package/src/clis/doubao-app/send.ts +1 -1
  482. package/src/clis/douyin/_shared/browser-fetch.test.ts +38 -0
  483. package/src/clis/douyin/_shared/browser-fetch.ts +45 -0
  484. package/src/clis/douyin/_shared/creation-id.test.ts +26 -0
  485. package/src/clis/douyin/_shared/creation-id.ts +8 -0
  486. package/src/clis/douyin/_shared/imagex-upload.test.ts +113 -0
  487. package/src/clis/douyin/_shared/imagex-upload.ts +76 -0
  488. package/src/clis/douyin/_shared/sts2.ts +20 -0
  489. package/src/clis/douyin/_shared/text-extra.test.ts +42 -0
  490. package/src/clis/douyin/_shared/text-extra.ts +33 -0
  491. package/src/clis/douyin/_shared/timing.test.ts +38 -0
  492. package/src/clis/douyin/_shared/timing.ts +22 -0
  493. package/src/clis/douyin/_shared/tos-upload-short-read.test.ts +102 -0
  494. package/src/clis/douyin/_shared/tos-upload.test.ts +281 -0
  495. package/src/clis/douyin/_shared/tos-upload.ts +444 -0
  496. package/src/clis/douyin/_shared/transcode.test.ts +117 -0
  497. package/src/clis/douyin/_shared/transcode.ts +78 -0
  498. package/src/clis/douyin/_shared/types.ts +29 -0
  499. package/src/clis/douyin/activities.test.ts +25 -0
  500. package/src/clis/douyin/activities.ts +23 -0
  501. package/src/clis/douyin/collections.test.ts +26 -0
  502. package/src/clis/douyin/collections.ts +25 -0
  503. package/src/clis/douyin/delete.test.ts +12 -0
  504. package/src/clis/douyin/delete.ts +20 -0
  505. package/src/clis/douyin/draft.test.ts +12 -0
  506. package/src/clis/douyin/draft.ts +282 -0
  507. package/src/clis/douyin/drafts.test.ts +12 -0
  508. package/src/clis/douyin/drafts.ts +27 -0
  509. package/src/clis/douyin/hashtag.test.ts +28 -0
  510. package/src/clis/douyin/hashtag.ts +56 -0
  511. package/src/clis/douyin/location.test.ts +26 -0
  512. package/src/clis/douyin/location.ts +27 -0
  513. package/src/clis/douyin/profile.test.ts +12 -0
  514. package/src/clis/douyin/profile.ts +37 -0
  515. package/src/clis/douyin/publish.test.ts +45 -0
  516. package/src/clis/douyin/publish.ts +340 -0
  517. package/src/clis/douyin/stats.test.ts +25 -0
  518. package/src/clis/douyin/stats.ts +30 -0
  519. package/src/clis/douyin/update.test.ts +12 -0
  520. package/src/clis/douyin/update.ts +43 -0
  521. package/src/clis/douyin/videos.test.ts +12 -0
  522. package/src/clis/douyin/videos.ts +49 -0
  523. package/src/clis/grok/ask.test.ts +25 -0
  524. package/src/clis/grok/ask.ts +25 -12
  525. package/src/clis/hackernews/search.yaml +1 -1
  526. package/src/clis/instagram/search.yaml +2 -1
  527. package/src/clis/jd/item.test.ts +35 -0
  528. package/src/clis/jd/item.ts +101 -0
  529. package/src/clis/jike/feed.ts +1 -1
  530. package/src/clis/jike/search.ts +1 -1
  531. package/src/clis/linkedin/search.ts +5 -4
  532. package/src/clis/linkedin/timeline.test.ts +99 -0
  533. package/src/clis/linkedin/timeline.ts +532 -0
  534. package/src/clis/linux-do/search.yaml +3 -1
  535. package/src/clis/medium/feed.ts +1 -1
  536. package/src/clis/medium/search.ts +2 -2
  537. package/src/clis/medium/user.ts +1 -1
  538. package/src/clis/medium/{shared.ts → utils.ts} +2 -1
  539. package/src/clis/pixiv/detail.yaml +49 -0
  540. package/src/clis/pixiv/download.test.ts +114 -0
  541. package/src/clis/pixiv/download.ts +91 -0
  542. package/src/clis/pixiv/illusts.test.ts +115 -0
  543. package/src/clis/pixiv/illusts.ts +78 -0
  544. package/src/clis/pixiv/ranking.yaml +53 -0
  545. package/src/clis/pixiv/search.test.ts +97 -0
  546. package/src/clis/pixiv/search.ts +53 -0
  547. package/src/clis/pixiv/test-utils.ts +29 -0
  548. package/src/clis/pixiv/user.yaml +46 -0
  549. package/src/clis/pixiv/utils.ts +62 -0
  550. package/src/clis/reddit/comment.ts +2 -1
  551. package/src/clis/reddit/read.test.ts +34 -0
  552. package/src/clis/reddit/read.ts +4 -3
  553. package/src/clis/reddit/save.ts +2 -1
  554. package/src/clis/reddit/saved.ts +6 -2
  555. package/src/clis/reddit/subscribe.ts +2 -1
  556. package/src/clis/reddit/upvote.ts +2 -1
  557. package/src/clis/reddit/upvoted.ts +6 -2
  558. package/src/clis/reuters/search.ts +0 -1
  559. package/src/clis/sinablog/article.ts +1 -1
  560. package/src/clis/sinablog/hot.ts +1 -1
  561. package/src/clis/sinablog/user.ts +1 -1
  562. package/src/clis/substack/feed.ts +1 -1
  563. package/src/clis/substack/publication.ts +1 -1
  564. package/src/clis/substack/search.ts +3 -2
  565. package/src/clis/substack/{shared.ts → utils.ts} +3 -2
  566. package/src/clis/tiktok/search.yaml +2 -1
  567. package/src/clis/twitter/accept.ts +2 -1
  568. package/src/clis/twitter/article.ts +3 -1
  569. package/src/clis/twitter/block.ts +2 -1
  570. package/src/clis/twitter/bookmark.ts +2 -1
  571. package/src/clis/twitter/bookmarks.ts +3 -2
  572. package/src/clis/twitter/delete.ts +2 -1
  573. package/src/clis/twitter/follow.ts +2 -1
  574. package/src/clis/twitter/followers.ts +3 -2
  575. package/src/clis/twitter/following.ts +3 -2
  576. package/src/clis/twitter/hide-reply.ts +2 -1
  577. package/src/clis/twitter/like.ts +2 -1
  578. package/src/clis/twitter/notifications.ts +2 -1
  579. package/src/clis/twitter/post.ts +2 -1
  580. package/src/clis/twitter/profile.ts +4 -2
  581. package/src/clis/twitter/reply-dm.ts +2 -1
  582. package/src/clis/twitter/reply.ts +2 -1
  583. package/src/clis/twitter/search.test.ts +180 -0
  584. package/src/clis/twitter/search.ts +40 -14
  585. package/src/clis/twitter/thread.ts +2 -2
  586. package/src/clis/twitter/timeline.ts +3 -2
  587. package/src/clis/twitter/trending.ts +3 -2
  588. package/src/clis/twitter/unblock.ts +2 -1
  589. package/src/clis/twitter/unbookmark.ts +2 -1
  590. package/src/clis/twitter/unfollow.ts +2 -1
  591. package/src/clis/v2ex/daily.ts +3 -2
  592. package/src/clis/v2ex/me.ts +3 -2
  593. package/src/clis/v2ex/notifications.ts +3 -4
  594. package/src/clis/web/read.ts +210 -0
  595. package/src/clis/weibo/comments.ts +54 -0
  596. package/src/clis/weibo/feed.ts +57 -0
  597. package/src/clis/weibo/hot.ts +0 -1
  598. package/src/clis/weibo/me.ts +77 -0
  599. package/src/clis/weibo/post.ts +77 -0
  600. package/src/clis/weibo/user.ts +64 -0
  601. package/src/clis/weibo/utils.ts +32 -0
  602. package/src/clis/weread/search.ts +3 -2
  603. package/src/clis/xueqiu/danjuan-utils.test.ts +49 -0
  604. package/src/clis/xueqiu/danjuan-utils.ts +176 -0
  605. package/src/clis/xueqiu/fund-holdings.ts +32 -0
  606. package/src/clis/xueqiu/fund-snapshot.ts +27 -0
  607. package/src/clis/xueqiu/search.yaml +2 -1
  608. package/src/clis/yahoo-finance/quote.ts +0 -1
  609. package/src/clis/youtube/channel.ts +155 -0
  610. package/src/clis/youtube/comments.ts +97 -0
  611. package/src/clis/youtube/search.ts +0 -1
  612. package/src/clis/youtube/transcript.ts +5 -4
  613. package/src/clis/youtube/video.ts +3 -2
  614. package/src/clis/zhihu/search.yaml +2 -1
  615. package/src/daemon.ts +5 -4
  616. package/src/discovery.ts +12 -34
  617. package/src/doctor.ts +3 -2
  618. package/src/download/index.test.ts +93 -2
  619. package/src/download/index.ts +44 -23
  620. package/src/download/media-download.ts +5 -3
  621. package/src/engine.test.ts +84 -3
  622. package/src/execution.ts +62 -46
  623. package/src/explore.ts +21 -90
  624. package/src/external-clis.yaml +0 -25
  625. package/src/external.test.ts +9 -0
  626. package/src/external.ts +12 -10
  627. package/src/generate.ts +4 -41
  628. package/src/hooks.test.ts +126 -0
  629. package/src/hooks.ts +90 -0
  630. package/src/interceptor.ts +73 -23
  631. package/src/main.ts +2 -0
  632. package/src/output.ts +14 -6
  633. package/src/pipeline/executor.ts +1 -1
  634. package/src/pipeline/steps/browser.ts +1 -3
  635. package/src/pipeline/steps/download.test.ts +136 -0
  636. package/src/pipeline/steps/download.ts +47 -34
  637. package/src/pipeline/steps/fetch.test.ts +179 -0
  638. package/src/pipeline/steps/fetch.ts +39 -23
  639. package/src/pipeline/steps/transform.ts +2 -6
  640. package/src/pipeline/template.test.ts +28 -0
  641. package/src/pipeline/template.ts +67 -79
  642. package/src/pipeline/transform.test.ts +20 -0
  643. package/src/plugin.test.ts +251 -3
  644. package/src/plugin.ts +265 -21
  645. package/src/record.ts +12 -84
  646. package/src/registry-api.ts +2 -0
  647. package/src/registry.ts +7 -4
  648. package/src/runtime.ts +14 -4
  649. package/src/snapshotFormatter.ts +43 -121
  650. package/src/utils.ts +39 -0
  651. package/src/validate.ts +3 -5
  652. package/src/weread-search-regression.test.ts +44 -0
  653. package/src/yaml-schema.ts +28 -0
  654. package/tests/e2e/browser-auth.test.ts +25 -0
  655. package/tests/e2e/browser-public-extended.test.ts +162 -0
  656. package/tests/e2e/browser-public.test.ts +7 -146
  657. package/tests/e2e/plugin-management.test.ts +137 -0
  658. package/tests/e2e/public-commands.test.ts +34 -1
  659. package/vitest.config.ts +33 -8
  660. package/.github/workflows/pkg-pr-new.yml +0 -30
  661. package/dist/clis/douban/shared.d.ts +0 -4
  662. package/dist/clis/douban/shared.js +0 -155
  663. package/src/clis/douban/shared.ts +0 -165
  664. /package/dist/clis/boss/{common.d.ts → utils.d.ts} +0 -0
  665. /package/dist/clis/boss/{common.js → utils.js} +0 -0
  666. /package/dist/clis/doubao/{common.d.ts → utils.d.ts} +0 -0
  667. /package/dist/clis/doubao/{common.js → utils.js} +0 -0
  668. /package/dist/clis/doubao-app/{common.d.ts → utils.d.ts} +0 -0
  669. /package/dist/clis/doubao-app/{common.js → utils.js} +0 -0
  670. /package/dist/clis/jike/{shared.d.ts → utils.d.ts} +0 -0
  671. /package/dist/clis/jike/{shared.js → utils.js} +0 -0
  672. /package/dist/clis/medium/{shared.d.ts → utils.d.ts} +0 -0
  673. /package/dist/clis/sinablog/{shared.d.ts → utils.d.ts} +0 -0
  674. /package/dist/clis/sinablog/{shared.js → utils.js} +0 -0
  675. /package/dist/clis/substack/{shared.d.ts → utils.d.ts} +0 -0
  676. /package/src/clis/boss/{common.ts → utils.ts} +0 -0
  677. /package/src/clis/doubao/{common.ts → utils.ts} +0 -0
  678. /package/src/clis/doubao-app/{common.ts → utils.ts} +0 -0
  679. /package/src/clis/jike/{shared.ts → utils.ts} +0 -0
  680. /package/src/clis/sinablog/{shared.ts → utils.ts} +0 -0
@@ -0,0 +1,45 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './publish.js';
4
+
5
+ describe('douyin publish registration', () => {
6
+ it('registers the publish command', () => {
7
+ const registry = getRegistry();
8
+ const cmds = [...registry.values()];
9
+ const cmd = cmds.find((c) => c.site === 'douyin' && c.name === 'publish');
10
+ expect(cmd).toBeDefined();
11
+ expect(cmd?.args.some((a) => a.name === 'video')).toBe(true);
12
+ expect(cmd?.args.some((a) => a.name === 'title')).toBe(true);
13
+ expect(cmd?.args.some((a) => a.name === 'schedule')).toBe(true);
14
+ });
15
+
16
+ it('has all expected args', () => {
17
+ const registry = getRegistry();
18
+ const cmd = [...registry.values()].find(
19
+ (c) => c.site === 'douyin' && c.name === 'publish',
20
+ );
21
+ const argNames = cmd?.args.map((a) => a.name) ?? [];
22
+ expect(argNames).toContain('video');
23
+ expect(argNames).toContain('title');
24
+ expect(argNames).toContain('schedule');
25
+ expect(argNames).toContain('caption');
26
+ expect(argNames).toContain('cover');
27
+ expect(argNames).toContain('visibility');
28
+ expect(argNames).toContain('allow_download');
29
+ expect(argNames).toContain('collection');
30
+ expect(argNames).toContain('activity');
31
+ expect(argNames).toContain('poi_id');
32
+ expect(argNames).toContain('poi_name');
33
+ expect(argNames).toContain('hotspot');
34
+ expect(argNames).toContain('no_safety_check');
35
+ expect(argNames).toContain('sync_toutiao');
36
+ });
37
+
38
+ it('uses COOKIE strategy', () => {
39
+ const registry = getRegistry();
40
+ const cmd = [...registry.values()].find(
41
+ (c) => c.site === 'douyin' && c.name === 'publish',
42
+ );
43
+ expect(cmd?.strategy).toBe('cookie');
44
+ });
45
+ });
@@ -0,0 +1,340 @@
1
+ /**
2
+ * Douyin publish — 8-phase pipeline for scheduling video posts.
3
+ *
4
+ * Phases:
5
+ * 1. STS2 credentials
6
+ * 2. Apply TOS upload URL
7
+ * 3. TOS multipart upload
8
+ * 4. Cover upload (optional, via ImageX)
9
+ * 5. Enable video
10
+ * 6. Poll transcode
11
+ * 7. Content safety check
12
+ * 8. create_v2 publish
13
+ */
14
+
15
+ import * as fs from 'node:fs';
16
+ import * as path from 'node:path';
17
+ import { cli, Strategy } from '../../registry.js';
18
+ import { ArgumentError, CommandExecutionError } from '../../errors.js';
19
+ import type { IPage } from '../../types.js';
20
+ import type { TosUploadInfo } from './_shared/types.js';
21
+ import { getSts2Credentials } from './_shared/sts2.js';
22
+ import { tosUpload } from './_shared/tos-upload.js';
23
+ import { imagexUpload } from './_shared/imagex-upload.js';
24
+ import { pollTranscode } from './_shared/transcode.js';
25
+ import { browserFetch } from './_shared/browser-fetch.js';
26
+ import { generateCreationId } from './_shared/creation-id.js';
27
+ import { validateTiming, toUnixSeconds } from './_shared/timing.js';
28
+ import { parseTextExtra, extractHashtagNames } from './_shared/text-extra.js';
29
+ import type { HashtagInfo } from './_shared/text-extra.js';
30
+
31
+ const VISIBILITY_MAP: Record<string, number> = {
32
+ public: 0,
33
+ friends: 1,
34
+ private: 2,
35
+ };
36
+
37
+ const IMAGEX_BASE = 'https://imagex.bytedanceapi.com';
38
+ const IMAGEX_SERVICE_ID = '1147';
39
+
40
+ const DEVICE_PARAMS =
41
+ 'aid=1128&cookie_enabled=true&screen_width=1512&screen_height=982&browser_language=zh-CN&browser_platform=MacIntel&browser_name=Mozilla&browser_online=true&timezone_name=Asia%2FTokyo&support_h265=1';
42
+
43
+ const DEFAULT_COVER_TOOLS_INFO = JSON.stringify({
44
+ video_cover_source: 2,
45
+ cover_timestamp: 0,
46
+ recommend_timestamp: 0,
47
+ is_cover_edit: 0,
48
+ is_cover_template: 0,
49
+ cover_template_id: '',
50
+ is_text_template: 0,
51
+ text_template_id: '',
52
+ text_template_content: '',
53
+ is_text: 0,
54
+ text_num: 0,
55
+ text_content: '',
56
+ is_use_sticker: 0,
57
+ sticker_id: '',
58
+ is_use_filter: 0,
59
+ filter_id: '',
60
+ is_cover_modify: 0,
61
+ to_status: 0,
62
+ cover_type: 0,
63
+ initial_cover_uri: '',
64
+ cut_coordinate: '',
65
+ });
66
+
67
+ cli({
68
+ site: 'douyin',
69
+ name: 'publish',
70
+ description: '定时发布视频到抖音(必须设置 2h ~ 14天后的发布时间)',
71
+ domain: 'creator.douyin.com',
72
+ strategy: Strategy.COOKIE,
73
+ args: [
74
+ { name: 'video', required: true, positional: true, help: '视频文件路径' },
75
+ { name: 'title', required: true, help: '视频标题(≤30字)' },
76
+ { name: 'schedule', required: true, help: '定时发布时间(ISO8601 或 Unix 秒,2h ~ 14天后)' },
77
+ { name: 'caption', default: '', help: '正文内容(≤1000字,支持 #话题)' },
78
+ { name: 'cover', default: '', help: '封面图片路径(不提供时使用视频截帧)' },
79
+ { name: 'visibility', default: 'public', choices: ['public', 'friends', 'private'] },
80
+ { name: 'allow_download', type: 'bool', default: false, help: '允许下载' },
81
+ { name: 'collection', default: '', help: '合集 ID' },
82
+ { name: 'activity', default: '', help: '活动 ID' },
83
+ { name: 'poi_id', default: '', help: '地理位置 ID' },
84
+ { name: 'poi_name', default: '', help: '地理位置名称' },
85
+ { name: 'hotspot', default: '', help: '关联热点词' },
86
+ { name: 'no_safety_check', type: 'bool', default: false, help: '跳过内容安全检测' },
87
+ { name: 'sync_toutiao', type: 'bool', default: false, help: '同步发布到头条' },
88
+ ],
89
+ columns: ['status', 'aweme_id', 'url', 'publish_time'],
90
+ func: async (page: IPage, kwargs) => {
91
+ // ── Fail-fast validation ────────────────────────────────────────────
92
+ const videoPath = path.resolve(kwargs.video as string);
93
+ if (!fs.existsSync(videoPath)) {
94
+ throw new ArgumentError(`视频文件不存在: ${videoPath}`);
95
+ }
96
+ const ext = path.extname(videoPath).toLowerCase();
97
+ if (!['.mp4', '.mov', '.avi', '.webm'].includes(ext)) {
98
+ throw new ArgumentError(`不支持的视频格式: ${ext}(支持 mp4/mov/avi/webm)`);
99
+ }
100
+ const fileSize = fs.statSync(videoPath).size;
101
+
102
+ const title = kwargs.title as string;
103
+ if (title.length > 30) {
104
+ throw new ArgumentError('标题不能超过 30 字');
105
+ }
106
+
107
+ const caption = (kwargs.caption as string) || '';
108
+ if (caption.length > 1000) {
109
+ throw new ArgumentError('正文不能超过 1000 字');
110
+ }
111
+
112
+ const timingTs = toUnixSeconds(kwargs.schedule as string | number);
113
+ validateTiming(timingTs);
114
+
115
+ const visibilityType = VISIBILITY_MAP[kwargs.visibility as string] ?? 0;
116
+
117
+ const coverPath = kwargs.cover as string;
118
+ if (coverPath) {
119
+ if (!fs.existsSync(path.resolve(coverPath))) {
120
+ throw new ArgumentError(`封面文件不存在: ${path.resolve(coverPath)}`);
121
+ }
122
+ }
123
+
124
+ // ── Phase 1: STS2 credentials ───────────────────────────────────────
125
+ const credentials = await getSts2Credentials(page);
126
+
127
+ // ── Phase 2: Apply TOS upload URL ───────────────────────────────────
128
+ const vodUrl = `https://vod.bytedanceapi.com/?Action=ApplyVideoUpload&ServiceId=1128&Version=2021-01-01&FileType=video&FileSize=${fileSize}`;
129
+ const vodJs = `fetch(${JSON.stringify(vodUrl)}, { credentials: 'include' }).then(r => r.json())`;
130
+ const vodRes = (await page.evaluate(vodJs)) as {
131
+ Result: {
132
+ UploadAddress: {
133
+ VideoId: string;
134
+ UploadHosts: string[];
135
+ StoreInfos: Array<{ Auth: string; StoreUri: string }>;
136
+ };
137
+ };
138
+ };
139
+ const { VideoId: videoId, UploadHosts, StoreInfos } = vodRes.Result.UploadAddress;
140
+ const tosUrl = `https://${UploadHosts[0]}/${StoreInfos[0].StoreUri}`;
141
+ const tosUploadInfo: TosUploadInfo = {
142
+ tos_upload_url: tosUrl,
143
+ auth: StoreInfos[0].Auth,
144
+ video_id: videoId,
145
+ };
146
+
147
+ // ── Phase 3: TOS upload ─────────────────────────────────────────────
148
+ await tosUpload({
149
+ filePath: videoPath,
150
+ uploadInfo: tosUploadInfo,
151
+ credentials,
152
+ onProgress: (uploaded, total) => {
153
+ const pct = Math.round((uploaded / total) * 100);
154
+ process.stderr.write(`\r 上传进度: ${pct}%`);
155
+ },
156
+ });
157
+ process.stderr.write('\n');
158
+
159
+ // ── Phase 4: Cover upload (optional) ────────────────────────────────
160
+ let coverUri = '';
161
+ let coverWidth = 720;
162
+ let coverHeight = 1280;
163
+
164
+ if (kwargs.cover) {
165
+ const resolvedCoverPath = path.resolve(kwargs.cover as string);
166
+
167
+ // 4A: Apply ImageX upload
168
+ const applyUrl = `${IMAGEX_BASE}/?Action=ApplyImageUpload&ServiceId=${IMAGEX_SERVICE_ID}&Version=2018-08-01&UploadNum=1`;
169
+ const applyJs = `fetch(${JSON.stringify(applyUrl)}, { credentials: 'include' }).then(r => r.json())`;
170
+ const applyRes = (await page.evaluate(applyJs)) as {
171
+ Result: {
172
+ UploadAddress: {
173
+ UploadHosts: string[];
174
+ StoreInfos: Array<{ Auth: string; StoreUri: string; UploadHost: string }>;
175
+ };
176
+ };
177
+ };
178
+ const { StoreInfos: imgStoreInfos } = applyRes.Result.UploadAddress;
179
+ const imgUploadUrl = `https://${imgStoreInfos[0].UploadHost}/${imgStoreInfos[0].StoreUri}`;
180
+
181
+ // 4B: Upload image
182
+ const coverStoreUri = await imagexUpload(resolvedCoverPath, {
183
+ upload_url: imgUploadUrl,
184
+ store_uri: imgStoreInfos[0].StoreUri,
185
+ });
186
+
187
+ // 4C: Commit ImageX upload
188
+ const commitUrl = `${IMAGEX_BASE}/?Action=CommitImageUpload&ServiceId=${IMAGEX_SERVICE_ID}&Version=2018-08-01`;
189
+ const commitBody = JSON.stringify({ SuccessObjKeys: [coverStoreUri] });
190
+ const commitJs = `
191
+ fetch(${JSON.stringify(commitUrl)}, {
192
+ method: 'POST',
193
+ credentials: 'include',
194
+ headers: { 'Content-Type': 'application/json' },
195
+ body: ${JSON.stringify(commitBody)}
196
+ }).then(r => r.json())
197
+ `;
198
+ await page.evaluate(commitJs);
199
+
200
+ coverUri = coverStoreUri;
201
+ }
202
+
203
+ // ── Phase 5: Enable video ───────────────────────────────────────────
204
+ const enableUrl = `https://creator.douyin.com/web/api/media/video/enable/?video_id=${videoId}&aid=1128`;
205
+ await browserFetch(page, 'GET', enableUrl);
206
+
207
+ // ── Phase 6: Poll transcode ─────────────────────────────────────────
208
+ const transResult = await pollTranscode(page, videoId);
209
+ coverWidth = transResult.width;
210
+ coverHeight = transResult.height;
211
+ if (!coverUri) {
212
+ coverUri = transResult.poster_uri;
213
+ }
214
+
215
+ // ── Phase 7: Content safety check ───────────────────────────────────
216
+ if (!kwargs.no_safety_check) {
217
+ const safetyUrl =
218
+ 'https://creator.douyin.com/aweme/v1/post_assistant/fast_detect/pre_check';
219
+ const safetyBody = {
220
+ video_id: videoId,
221
+ title,
222
+ desc: caption,
223
+ };
224
+ await browserFetch(page, 'POST', safetyUrl, { body: safetyBody });
225
+
226
+ const pollUrl =
227
+ 'https://creator.douyin.com/aweme/v1/post_assistant/fast_detect/poll';
228
+ const deadline = Date.now() + 30_000;
229
+ let safetyPassed = false;
230
+ while (Date.now() < deadline) {
231
+ const pollRes = (await browserFetch(page, 'POST', pollUrl, {
232
+ body: safetyBody,
233
+ })) as { status: number };
234
+ if (pollRes.status === 0) { safetyPassed = true; break; }
235
+ if (pollRes.status === 1) {
236
+ throw new CommandExecutionError(
237
+ '内容安全检测不通过,请修改后重试',
238
+ '使用 --no_safety_check 跳过',
239
+ );
240
+ }
241
+ await new Promise((r) => setTimeout(r, 2000));
242
+ }
243
+ if (!safetyPassed) {
244
+ throw new CommandExecutionError(
245
+ '内容安全检测超时(30s),请稍后重试',
246
+ '使用 --no_safety_check 跳过',
247
+ );
248
+ }
249
+ }
250
+
251
+ // ── Phase 8: create_v2 publish ──────────────────────────────────────
252
+ const hashtagNames = extractHashtagNames(caption);
253
+ const hashtags: HashtagInfo[] = [];
254
+ let searchFrom = 0;
255
+ for (const name of hashtagNames) {
256
+ const idx = caption.indexOf(`#${name}`, searchFrom);
257
+ if (idx === -1) continue;
258
+ hashtags.push({ name, id: 0, start: idx, end: idx + name.length + 1 });
259
+ searchFrom = idx + name.length + 1;
260
+ }
261
+ const textExtraArr = parseTextExtra(caption, hashtags);
262
+
263
+ const publishBody = {
264
+ item: {
265
+ common: {
266
+ text: caption,
267
+ caption: '',
268
+ item_title: title,
269
+ activity: JSON.stringify(kwargs.activity ? [kwargs.activity] : []),
270
+ text_extra: JSON.stringify(textExtraArr),
271
+ challenges: '[]',
272
+ mentions: '[]',
273
+ hashtag_source: '',
274
+ hot_sentence: (kwargs.hotspot as string) || '',
275
+ interaction_stickers: '[]',
276
+ visibility_type: visibilityType,
277
+ download: kwargs.allow_download ? 1 : 0,
278
+ timing: timingTs,
279
+ creation_id: generateCreationId(),
280
+ media_type: 4,
281
+ video_id: videoId,
282
+ music_source: 0,
283
+ music_id: null,
284
+ ...(kwargs.poi_id
285
+ ? { poi_id: kwargs.poi_id as string, poi_name: kwargs.poi_name as string }
286
+ : {}),
287
+ },
288
+ cover: {
289
+ poster: coverUri,
290
+ custom_cover_image_height: coverHeight,
291
+ custom_cover_image_width: coverWidth,
292
+ poster_delay: 0,
293
+ cover_tools_info: DEFAULT_COVER_TOOLS_INFO,
294
+ cover_tools_extend_info: '{}',
295
+ },
296
+ mix: kwargs.collection
297
+ ? { mix_id: kwargs.collection as string, mix_order: 0 }
298
+ : {},
299
+ chapter: {
300
+ chapter: JSON.stringify({
301
+ chapter_abstract: '',
302
+ chapter_details: [],
303
+ chapter_type: 0,
304
+ }),
305
+ },
306
+ anchor: {},
307
+ sync: {
308
+ should_sync: false,
309
+ sync_to_toutiao: kwargs.sync_toutiao ? 1 : 0,
310
+ },
311
+ open_platform: {},
312
+ assistant: { is_preview: 0, is_post_assistant: 1 },
313
+ declare: { user_declare_info: '{}' },
314
+ },
315
+ };
316
+
317
+ const publishUrl = `https://creator.douyin.com/web/api/media/aweme/create_v2/?read_aid=2906&${DEVICE_PARAMS}`;
318
+ const publishRes = (await browserFetch(page, 'POST', publishUrl, {
319
+ body: publishBody,
320
+ })) as { status_code: number; aweme_id: string };
321
+
322
+ const awemeId = publishRes.aweme_id;
323
+ if (!awemeId) {
324
+ throw new CommandExecutionError(`发布成功但未返回 aweme_id: ${JSON.stringify(publishRes)}`);
325
+ }
326
+ const url = `https://www.douyin.com/video/${awemeId}`;
327
+ const publishTimeStr = new Date(timingTs * 1000).toLocaleString('zh-CN', {
328
+ timeZone: 'Asia/Tokyo',
329
+ });
330
+
331
+ return [
332
+ {
333
+ status: '✅ 定时发布成功!',
334
+ aweme_id: awemeId,
335
+ url,
336
+ publish_time: publishTimeStr,
337
+ },
338
+ ];
339
+ },
340
+ });
@@ -0,0 +1,25 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './stats.js';
4
+
5
+ describe('douyin stats registration', () => {
6
+ it('registers the stats command', () => {
7
+ const registry = getRegistry();
8
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'stats');
9
+ expect(cmd).toBeDefined();
10
+ expect(cmd?.args.some(a => a.name === 'aweme_id')).toBe(true);
11
+ });
12
+
13
+ it('has expected columns', () => {
14
+ const registry = getRegistry();
15
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'stats');
16
+ expect(cmd?.columns).toContain('metric');
17
+ expect(cmd?.columns).toContain('value');
18
+ });
19
+
20
+ it('uses COOKIE strategy', () => {
21
+ const registry = getRegistry();
22
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'stats');
23
+ expect(cmd?.strategy).toBe('cookie');
24
+ });
25
+ });
@@ -0,0 +1,30 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { browserFetch } from './_shared/browser-fetch.js';
3
+
4
+ cli({
5
+ site: 'douyin',
6
+ name: 'stats',
7
+ description: '作品数据分析',
8
+ domain: 'creator.douyin.com',
9
+ strategy: Strategy.COOKIE,
10
+ args: [
11
+ { name: 'aweme_id', required: true, positional: true },
12
+ ],
13
+ columns: ['metric', 'value'],
14
+ func: async (page, kwargs) => {
15
+ const now = Math.floor(Date.now() / 1000);
16
+ const sevenDaysAgo = now - 7 * 86400;
17
+ const url = 'https://creator.douyin.com/janus/douyin/creator/data/item_analysis/metrics_trend';
18
+ const body = {
19
+ aweme_id: kwargs.aweme_id as string,
20
+ start_time: sevenDaysAgo,
21
+ end_time: now,
22
+ metrics: ['play_count', 'like_count', 'comment_count', 'share_count'],
23
+ };
24
+ const res = await browserFetch(page, 'POST', url, { body }) as {
25
+ data: Record<string, number>
26
+ };
27
+ const data = res.data ?? {};
28
+ return Object.entries(data).map(([metric, value]) => ({ metric, value }));
29
+ },
30
+ });
@@ -0,0 +1,12 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './update.js';
4
+
5
+ describe('douyin update registration', () => {
6
+ it('registers the update command', () => {
7
+ const registry = getRegistry();
8
+ const values = [...registry.values()];
9
+ const cmd = values.find(c => c.site === 'douyin' && c.name === 'update');
10
+ expect(cmd).toBeDefined();
11
+ });
12
+ });
@@ -0,0 +1,43 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { ArgumentError } from '../../errors.js';
3
+ import { browserFetch } from './_shared/browser-fetch.js';
4
+ import { toUnixSeconds, validateTiming } from './_shared/timing.js';
5
+ import type { IPage } from '../../types.js';
6
+
7
+ cli({
8
+ site: 'douyin',
9
+ name: 'update',
10
+ description: '更新视频信息',
11
+ domain: 'creator.douyin.com',
12
+ strategy: Strategy.COOKIE,
13
+ args: [
14
+ { name: 'aweme_id', required: true, positional: true },
15
+ { name: 'reschedule', default: '', help: '新的发布时间(ISO8601 或 Unix 秒)' },
16
+ { name: 'caption', default: '', help: '新的正文内容' },
17
+ ],
18
+ columns: ['status'],
19
+ func: async (page: IPage, kwargs) => {
20
+ if (!kwargs.reschedule && !kwargs.caption) {
21
+ throw new ArgumentError('必须提供 --reschedule 或 --caption');
22
+ }
23
+ if (kwargs.reschedule) {
24
+ const newTime = toUnixSeconds(kwargs.reschedule as string | number);
25
+ validateTiming(newTime);
26
+ await browserFetch(
27
+ page,
28
+ 'POST',
29
+ 'https://creator.douyin.com/web/api/media/update/timer/?aid=1128',
30
+ { body: { aweme_id: kwargs.aweme_id, publish_time: newTime } },
31
+ );
32
+ }
33
+ if (kwargs.caption) {
34
+ await browserFetch(
35
+ page,
36
+ 'POST',
37
+ 'https://creator.douyin.com/web/api/media/update/desc/?aid=1128',
38
+ { body: { aweme_id: kwargs.aweme_id, desc: kwargs.caption } },
39
+ );
40
+ }
41
+ return [{ status: '✅ 更新成功' }];
42
+ },
43
+ });
@@ -0,0 +1,12 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './videos.js';
4
+
5
+ describe('douyin videos registration', () => {
6
+ it('registers the videos command', () => {
7
+ const registry = getRegistry();
8
+ const values = [...registry.values()];
9
+ const cmd = values.find(c => c.site === 'douyin' && c.name === 'videos');
10
+ expect(cmd).toBeDefined();
11
+ });
12
+ });
@@ -0,0 +1,49 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { browserFetch } from './_shared/browser-fetch.js';
3
+ import type { IPage } from '../../types.js';
4
+
5
+ cli({
6
+ site: 'douyin',
7
+ name: 'videos',
8
+ description: '获取作品列表',
9
+ domain: 'creator.douyin.com',
10
+ strategy: Strategy.COOKIE,
11
+ args: [
12
+ { name: 'limit', type: 'int', default: 20, help: '每页数量' },
13
+ { name: 'page', type: 'int', default: 1, help: '页码' },
14
+ { name: 'status', default: 'all', choices: ['all', 'published', 'reviewing', 'scheduled'] },
15
+ ],
16
+ columns: ['aweme_id', 'title', 'status', 'play_count', 'digg_count', 'create_time'],
17
+ func: async (page: IPage, kwargs) => {
18
+ const statusMap: Record<string, number> = { all: 0, published: 1, reviewing: 3, scheduled: 0 };
19
+ const statusNum = statusMap[kwargs.status as string] ?? 0;
20
+ const url = `https://creator.douyin.com/janus/douyin/creator/pc/work_list?page_size=${kwargs.limit}&page_num=${kwargs.page}&status=${statusNum}`;
21
+ const res = (await browserFetch(page, 'GET', url)) as {
22
+ data: {
23
+ work_list: Array<{
24
+ aweme_id: string;
25
+ desc: string;
26
+ status: number;
27
+ public_time: number;
28
+ create_time: number;
29
+ statistics: { play_count: number; digg_count: number };
30
+ }>;
31
+ };
32
+ };
33
+ let items = res.data?.work_list ?? [];
34
+
35
+ // The API has a bug with status=16 for scheduled, so filter client-side
36
+ if (kwargs.status === 'scheduled') {
37
+ items = items.filter((v) => v.public_time > Date.now() / 1000);
38
+ }
39
+
40
+ return items.map((v) => ({
41
+ aweme_id: v.aweme_id,
42
+ title: v.desc,
43
+ status: v.status,
44
+ play_count: v.statistics?.play_count ?? 0,
45
+ digg_count: v.statistics?.digg_count ?? 0,
46
+ create_time: new Date(v.create_time * 1000).toLocaleString('zh-CN', { timeZone: 'Asia/Tokyo' }),
47
+ }));
48
+ },
49
+ });
@@ -1,7 +1,32 @@
1
1
  import { describe, expect, it } from 'vitest';
2
+ import type { IPage } from '../../types.js';
2
3
  import { __test__ } from './ask.js';
3
4
 
4
5
  describe('grok ask helpers', () => {
6
+ describe('isOnGrok', () => {
7
+ const fakePage = (url: string | Error): IPage =>
8
+ ({ evaluate: () => url instanceof Error ? Promise.reject(url) : Promise.resolve(url) }) as unknown as IPage;
9
+
10
+ it('returns true for grok.com URLs', async () => {
11
+ expect(await __test__.isOnGrok(fakePage('https://grok.com/'))).toBe(true);
12
+ expect(await __test__.isOnGrok(fakePage('https://grok.com/chat/abc123'))).toBe(true);
13
+ });
14
+
15
+ it('returns true for grok.com subdomains', async () => {
16
+ expect(await __test__.isOnGrok(fakePage('https://api.grok.com/v1'))).toBe(true);
17
+ });
18
+
19
+ it('returns false for non-grok domains', async () => {
20
+ expect(await __test__.isOnGrok(fakePage('https://fakegrok.com/'))).toBe(false);
21
+ expect(await __test__.isOnGrok(fakePage('https://example.com/?next=grok.com'))).toBe(false);
22
+ expect(await __test__.isOnGrok(fakePage('about:blank'))).toBe(false);
23
+ });
24
+
25
+ it('returns false when evaluate throws (detached tab)', async () => {
26
+ expect(await __test__.isOnGrok(fakePage(new Error('detached')))).toBe(false);
27
+ });
28
+ });
29
+
5
30
  it('normalizes boolean flags for explicit web routing', () => {
6
31
  expect(__test__.normalizeBooleanFlag(true)).toBe(true);
7
32
  expect(__test__.normalizeBooleanFlag('true')).toBe(true);
@@ -53,6 +53,19 @@ function updateStableState(previousText: string, stableCount: number, nextText:
53
53
  return { previousText: nextText, stableCount: 0 };
54
54
  }
55
55
 
56
+ /** Check whether the tab is already on grok.com (any path). */
57
+ async function isOnGrok(page: IPage): Promise<boolean> {
58
+ // catch handles blank tabs (about:blank) or detached pages
59
+ const url = await page.evaluate('window.location.href').catch(() => '');
60
+ if (typeof url !== 'string' || !url) return false;
61
+ try {
62
+ const hostname = new URL(url).hostname;
63
+ return hostname === 'grok.com' || hostname.endsWith('.grok.com');
64
+ } catch {
65
+ return false;
66
+ }
67
+ }
68
+
56
69
  async function runDefaultAsk(
57
70
  page: IPage,
58
71
  prompt: string,
@@ -60,21 +73,17 @@ async function runDefaultAsk(
60
73
  newChat: boolean,
61
74
  ) {
62
75
  if (newChat) {
76
+ // Explicitly start a fresh conversation via the homepage
63
77
  await page.goto(GROK_URL);
64
78
  await page.wait(2);
65
- await page.evaluate(`(() => {
66
- const btn = [...document.querySelectorAll('a, button')].find(b => {
67
- const t = (b.textContent || '').trim().toLowerCase();
68
- return t.includes('new') || b.getAttribute('href') === '/';
69
- });
70
- if (btn) btn.click();
71
- })()`);
79
+ await tryStartFreshChat(page);
72
80
  await page.wait(2);
81
+ } else if (!(await isOnGrok(page))) {
82
+ // First invocation or tab was recycled — navigate to Grok
83
+ await page.goto(GROK_URL);
84
+ await page.wait(3);
73
85
  }
74
86
 
75
- await page.goto(GROK_URL);
76
- await page.wait(3);
77
-
78
87
  const promptJson = JSON.stringify(prompt);
79
88
  const sendResult = await page.evaluate(`(async () => {
80
89
  try {
@@ -249,11 +258,14 @@ async function runExplicitWebAsk(
249
258
  timeoutMs: number,
250
259
  newChat: boolean,
251
260
  ) {
252
- await page.goto(GROK_URL, { settleMs: 2000 });
253
-
254
261
  if (newChat) {
262
+ // Navigate to homepage and start a fresh conversation
263
+ await page.goto(GROK_URL, { settleMs: 2000 });
255
264
  await tryStartFreshChat(page);
256
265
  await page.wait(2);
266
+ } else if (!(await isOnGrok(page))) {
267
+ // First invocation or tab was recycled — navigate to Grok
268
+ await page.goto(GROK_URL, { settleMs: 2000 });
257
269
  }
258
270
 
259
271
  const baselineBubbles = await getBubbleTexts(page);
@@ -318,4 +330,5 @@ export const __test__ = {
318
330
  updateStableState,
319
331
  normalizeBooleanFlag,
320
332
  normalizeBubbleText,
333
+ isOnGrok,
321
334
  };
@@ -41,4 +41,4 @@ pipeline:
41
41
 
42
42
  - limit: ${{ args.limit }}
43
43
 
44
- columns: [rank, title, score, author, comments]
44
+ columns: [rank, title, score, author, comments, url]