@jackwener/opencli 1.4.0 → 1.5.0

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 (514) hide show
  1. package/.github/actions/setup-chrome/action.yml +5 -4
  2. package/.github/workflows/build-extension.yml +2 -6
  3. package/.github/workflows/ci.yml +37 -3
  4. package/.github/workflows/e2e-headed.yml +16 -3
  5. package/CHANGELOG.md +23 -0
  6. package/PRIVACY.md +57 -0
  7. package/README.md +36 -7
  8. package/README.zh-CN.md +13 -6
  9. package/SKILL.md +103 -2
  10. package/dist/browser/cdp.d.ts +2 -1
  11. package/dist/browser/discover.d.ts +4 -1
  12. package/dist/browser/discover.js +6 -2
  13. package/dist/browser/errors.d.ts +2 -2
  14. package/dist/browser/errors.js +4 -12
  15. package/dist/browser/mcp.d.ts +2 -1
  16. package/dist/build-manifest.d.ts +2 -0
  17. package/dist/build-manifest.js +39 -14
  18. package/dist/build-manifest.test.js +21 -0
  19. package/dist/capabilityRouting.d.ts +2 -0
  20. package/dist/capabilityRouting.js +2 -1
  21. package/dist/cli-manifest.json +1838 -151
  22. package/dist/cli.js +34 -3
  23. package/dist/clis/36kr/article.d.ts +1 -0
  24. package/dist/clis/36kr/article.js +62 -0
  25. package/dist/clis/36kr/hot.d.ts +3 -0
  26. package/dist/clis/36kr/hot.js +80 -0
  27. package/dist/clis/36kr/hot.test.d.ts +1 -0
  28. package/dist/clis/36kr/hot.test.js +15 -0
  29. package/dist/clis/36kr/news.d.ts +1 -0
  30. package/dist/clis/36kr/news.js +51 -0
  31. package/dist/clis/36kr/news.test.d.ts +1 -0
  32. package/dist/clis/36kr/news.test.js +85 -0
  33. package/dist/clis/36kr/search.d.ts +1 -0
  34. package/dist/clis/36kr/search.js +72 -0
  35. package/dist/clis/apple-podcasts/search.js +2 -1
  36. package/dist/clis/arxiv/search.js +2 -2
  37. package/dist/clis/bbc/news.js +0 -1
  38. package/dist/clis/bilibili/comments.d.ts +5 -0
  39. package/dist/clis/bilibili/comments.js +40 -0
  40. package/dist/clis/bilibili/comments.test.d.ts +1 -0
  41. package/dist/clis/bilibili/comments.test.js +82 -0
  42. package/dist/clis/chatgpt/ask.js +29 -14
  43. package/dist/clis/chatgpt/ax.d.ts +6 -0
  44. package/dist/clis/chatgpt/ax.js +172 -1
  45. package/dist/clis/chatgpt/model.d.ts +1 -0
  46. package/dist/clis/chatgpt/model.js +24 -0
  47. package/dist/clis/chatgpt/send.js +12 -3
  48. package/dist/clis/ctrip/search.js +0 -1
  49. package/dist/clis/douban/download.d.ts +1 -0
  50. package/dist/clis/douban/download.js +67 -0
  51. package/dist/clis/douban/download.test.d.ts +1 -0
  52. package/dist/clis/douban/download.test.js +170 -0
  53. package/dist/clis/douban/photos.d.ts +1 -0
  54. package/dist/clis/douban/photos.js +34 -0
  55. package/dist/clis/douban/utils.d.ts +25 -0
  56. package/dist/clis/douban/utils.js +190 -1
  57. package/dist/clis/douban/utils.test.d.ts +1 -0
  58. package/dist/clis/douban/utils.test.js +64 -0
  59. package/dist/clis/douyin/_shared/browser-fetch.d.ts +10 -0
  60. package/dist/clis/douyin/_shared/browser-fetch.js +30 -0
  61. package/dist/clis/douyin/_shared/browser-fetch.test.d.ts +1 -0
  62. package/dist/clis/douyin/_shared/browser-fetch.test.js +31 -0
  63. package/dist/clis/douyin/_shared/creation-id.d.ts +1 -0
  64. package/dist/clis/douyin/_shared/creation-id.js +5 -0
  65. package/dist/clis/douyin/_shared/creation-id.test.d.ts +1 -0
  66. package/dist/clis/douyin/_shared/creation-id.test.js +22 -0
  67. package/dist/clis/douyin/_shared/imagex-upload.d.ts +20 -0
  68. package/dist/clis/douyin/_shared/imagex-upload.js +53 -0
  69. package/dist/clis/douyin/_shared/imagex-upload.test.d.ts +1 -0
  70. package/dist/clis/douyin/_shared/imagex-upload.test.js +87 -0
  71. package/dist/clis/douyin/_shared/sts2.d.ts +8 -0
  72. package/dist/clis/douyin/_shared/sts2.js +15 -0
  73. package/dist/clis/douyin/_shared/text-extra.d.ts +18 -0
  74. package/dist/clis/douyin/_shared/text-extra.js +15 -0
  75. package/dist/clis/douyin/_shared/text-extra.test.d.ts +1 -0
  76. package/dist/clis/douyin/_shared/text-extra.test.js +37 -0
  77. package/dist/clis/douyin/_shared/timing.d.ts +2 -0
  78. package/dist/clis/douyin/_shared/timing.js +22 -0
  79. package/dist/clis/douyin/_shared/timing.test.d.ts +1 -0
  80. package/dist/clis/douyin/_shared/timing.test.js +28 -0
  81. package/dist/clis/douyin/_shared/tos-upload-short-read.test.d.ts +11 -0
  82. package/dist/clis/douyin/_shared/tos-upload-short-read.test.js +83 -0
  83. package/dist/clis/douyin/_shared/tos-upload.d.ts +53 -0
  84. package/dist/clis/douyin/_shared/tos-upload.js +295 -0
  85. package/dist/clis/douyin/_shared/tos-upload.test.d.ts +1 -0
  86. package/dist/clis/douyin/_shared/tos-upload.test.js +229 -0
  87. package/dist/clis/douyin/_shared/transcode.d.ts +27 -0
  88. package/dist/clis/douyin/_shared/transcode.js +45 -0
  89. package/dist/clis/douyin/_shared/transcode.test.d.ts +1 -0
  90. package/dist/clis/douyin/_shared/transcode.test.js +93 -0
  91. package/dist/clis/douyin/_shared/types.d.ts +26 -0
  92. package/dist/clis/douyin/_shared/types.js +1 -0
  93. package/dist/clis/douyin/activities.d.ts +1 -0
  94. package/dist/clis/douyin/activities.js +20 -0
  95. package/dist/clis/douyin/activities.test.d.ts +1 -0
  96. package/dist/clis/douyin/activities.test.js +22 -0
  97. package/dist/clis/douyin/collections.d.ts +1 -0
  98. package/dist/clis/douyin/collections.js +22 -0
  99. package/dist/clis/douyin/collections.test.d.ts +1 -0
  100. package/dist/clis/douyin/collections.test.js +23 -0
  101. package/dist/clis/douyin/delete.d.ts +1 -0
  102. package/dist/clis/douyin/delete.js +18 -0
  103. package/dist/clis/douyin/delete.test.d.ts +1 -0
  104. package/dist/clis/douyin/delete.test.js +11 -0
  105. package/dist/clis/douyin/draft.d.ts +14 -0
  106. package/dist/clis/douyin/draft.js +237 -0
  107. package/dist/clis/douyin/draft.test.d.ts +1 -0
  108. package/dist/clis/douyin/draft.test.js +11 -0
  109. package/dist/clis/douyin/drafts.d.ts +1 -0
  110. package/dist/clis/douyin/drafts.js +23 -0
  111. package/dist/clis/douyin/drafts.test.d.ts +1 -0
  112. package/dist/clis/douyin/drafts.test.js +11 -0
  113. package/dist/clis/douyin/hashtag.d.ts +1 -0
  114. package/dist/clis/douyin/hashtag.js +45 -0
  115. package/dist/clis/douyin/hashtag.test.d.ts +1 -0
  116. package/dist/clis/douyin/hashtag.test.js +25 -0
  117. package/dist/clis/douyin/location.d.ts +1 -0
  118. package/dist/clis/douyin/location.js +24 -0
  119. package/dist/clis/douyin/location.test.d.ts +1 -0
  120. package/dist/clis/douyin/location.test.js +23 -0
  121. package/dist/clis/douyin/profile.d.ts +1 -0
  122. package/dist/clis/douyin/profile.js +28 -0
  123. package/dist/clis/douyin/profile.test.d.ts +1 -0
  124. package/dist/clis/douyin/profile.test.js +11 -0
  125. package/dist/clis/douyin/publish.d.ts +14 -0
  126. package/dist/clis/douyin/publish.js +288 -0
  127. package/dist/clis/douyin/publish.test.d.ts +1 -0
  128. package/dist/clis/douyin/publish.test.js +38 -0
  129. package/dist/clis/douyin/stats.d.ts +1 -0
  130. package/dist/clis/douyin/stats.js +27 -0
  131. package/dist/clis/douyin/stats.test.d.ts +1 -0
  132. package/dist/clis/douyin/stats.test.js +22 -0
  133. package/dist/clis/douyin/update.d.ts +1 -0
  134. package/dist/clis/douyin/update.js +31 -0
  135. package/dist/clis/douyin/update.test.d.ts +1 -0
  136. package/dist/clis/douyin/update.test.js +11 -0
  137. package/dist/clis/douyin/videos.d.ts +1 -0
  138. package/dist/clis/douyin/videos.js +34 -0
  139. package/dist/clis/douyin/videos.test.d.ts +1 -0
  140. package/dist/clis/douyin/videos.test.js +11 -0
  141. package/dist/clis/hackernews/search.yaml +1 -1
  142. package/dist/clis/imdb/person.d.ts +1 -0
  143. package/dist/clis/imdb/person.js +203 -0
  144. package/dist/clis/imdb/reviews.d.ts +1 -0
  145. package/dist/clis/imdb/reviews.js +88 -0
  146. package/dist/clis/imdb/search.d.ts +1 -0
  147. package/dist/clis/imdb/search.js +161 -0
  148. package/dist/clis/imdb/title.d.ts +1 -0
  149. package/dist/clis/imdb/title.js +93 -0
  150. package/dist/clis/imdb/top.d.ts +1 -0
  151. package/dist/clis/imdb/top.js +53 -0
  152. package/dist/clis/imdb/trending.d.ts +1 -0
  153. package/dist/clis/imdb/trending.js +52 -0
  154. package/dist/clis/imdb/utils.d.ts +46 -0
  155. package/dist/clis/imdb/utils.js +285 -0
  156. package/dist/clis/imdb/utils.test.d.ts +1 -0
  157. package/dist/clis/imdb/utils.test.js +88 -0
  158. package/dist/clis/instagram/search.yaml +2 -1
  159. package/dist/clis/jd/item.d.ts +4 -0
  160. package/dist/clis/jd/item.js +16 -15
  161. package/dist/clis/jd/item.test.js +16 -1
  162. package/dist/clis/linux-do/categories.yaml +38 -9
  163. package/dist/clis/linux-do/category.d.ts +1 -0
  164. package/dist/clis/linux-do/category.js +36 -0
  165. package/dist/clis/linux-do/feed.d.ts +45 -0
  166. package/dist/clis/linux-do/feed.js +397 -0
  167. package/dist/clis/linux-do/feed.test.d.ts +1 -0
  168. package/dist/clis/linux-do/feed.test.js +118 -0
  169. package/dist/clis/linux-do/hot.d.ts +1 -0
  170. package/dist/clis/linux-do/hot.js +25 -0
  171. package/dist/clis/linux-do/latest.d.ts +1 -0
  172. package/dist/clis/linux-do/latest.js +18 -0
  173. package/dist/clis/linux-do/search.yaml +3 -1
  174. package/dist/clis/linux-do/tags.yaml +41 -0
  175. package/dist/clis/linux-do/topic.yaml +41 -3
  176. package/dist/clis/linux-do/user-posts.yaml +67 -0
  177. package/dist/clis/linux-do/user-topics.yaml +54 -0
  178. package/dist/clis/medium/search.js +1 -1
  179. package/dist/clis/paperreview/commands.test.d.ts +3 -0
  180. package/dist/clis/paperreview/commands.test.js +243 -0
  181. package/dist/clis/paperreview/feedback.d.ts +1 -0
  182. package/dist/clis/paperreview/feedback.js +52 -0
  183. package/dist/clis/paperreview/review.d.ts +1 -0
  184. package/dist/clis/paperreview/review.js +37 -0
  185. package/dist/clis/paperreview/submit.d.ts +1 -0
  186. package/dist/clis/paperreview/submit.js +85 -0
  187. package/dist/clis/paperreview/utils.d.ts +46 -0
  188. package/dist/clis/paperreview/utils.js +197 -0
  189. package/dist/clis/paperreview/utils.test.d.ts +1 -0
  190. package/dist/clis/paperreview/utils.test.js +49 -0
  191. package/dist/clis/producthunt/browse.d.ts +1 -0
  192. package/dist/clis/producthunt/browse.js +99 -0
  193. package/dist/clis/producthunt/hot.d.ts +1 -0
  194. package/dist/clis/producthunt/hot.js +110 -0
  195. package/dist/clis/producthunt/posts.d.ts +1 -0
  196. package/dist/clis/producthunt/posts.js +28 -0
  197. package/dist/clis/producthunt/today.d.ts +1 -0
  198. package/dist/clis/producthunt/today.js +35 -0
  199. package/dist/clis/producthunt/utils.d.ts +29 -0
  200. package/dist/clis/producthunt/utils.js +99 -0
  201. package/dist/clis/producthunt/utils.test.d.ts +1 -0
  202. package/dist/clis/producthunt/utils.test.js +64 -0
  203. package/dist/clis/reuters/search.js +0 -1
  204. package/dist/clis/twitter/article.js +4 -28
  205. package/dist/clis/twitter/likes.d.ts +24 -0
  206. package/dist/clis/twitter/likes.js +217 -0
  207. package/dist/clis/twitter/likes.test.d.ts +1 -0
  208. package/dist/clis/twitter/likes.test.js +85 -0
  209. package/dist/clis/twitter/profile.js +4 -28
  210. package/dist/clis/twitter/search.js +7 -4
  211. package/dist/clis/twitter/search.test.js +56 -2
  212. package/dist/clis/twitter/shared.d.ts +6 -0
  213. package/dist/clis/twitter/shared.js +35 -0
  214. package/dist/clis/twitter/timeline.js +2 -13
  215. package/dist/clis/weibo/comments.d.ts +1 -0
  216. package/dist/clis/weibo/comments.js +53 -0
  217. package/dist/clis/weibo/feed.d.ts +1 -0
  218. package/dist/clis/weibo/feed.js +56 -0
  219. package/dist/clis/weibo/hot.js +0 -1
  220. package/dist/clis/weibo/me.d.ts +1 -0
  221. package/dist/clis/weibo/me.js +76 -0
  222. package/dist/clis/weibo/post.d.ts +1 -0
  223. package/dist/clis/weibo/post.js +75 -0
  224. package/dist/clis/weibo/user.d.ts +1 -0
  225. package/dist/clis/weibo/user.js +63 -0
  226. package/dist/clis/weibo/utils.d.ts +6 -0
  227. package/dist/clis/weibo/utils.js +30 -0
  228. package/dist/clis/weixin/download.d.ts +17 -0
  229. package/dist/clis/weixin/download.js +88 -20
  230. package/dist/clis/weread/book.js +2 -2
  231. package/dist/clis/weread/commands.test.d.ts +3 -0
  232. package/dist/clis/weread/commands.test.js +43 -0
  233. package/dist/clis/weread/highlights.js +2 -2
  234. package/dist/clis/weread/notebooks.js +2 -2
  235. package/dist/clis/weread/notes.js +3 -3
  236. package/dist/clis/weread/search.js +3 -2
  237. package/dist/clis/weread/shelf.js +2 -2
  238. package/dist/clis/weread/utils.d.ts +4 -4
  239. package/dist/clis/weread/utils.js +32 -14
  240. package/dist/clis/weread/utils.test.js +1 -28
  241. package/dist/clis/xiaohongshu/comments.d.ts +5 -0
  242. package/dist/clis/xiaohongshu/comments.js +74 -0
  243. package/dist/clis/xiaohongshu/comments.test.d.ts +1 -0
  244. package/dist/clis/xiaohongshu/comments.test.js +79 -0
  245. package/dist/clis/xiaohongshu/publish.js +114 -18
  246. package/dist/clis/xiaohongshu/publish.test.d.ts +1 -0
  247. package/dist/clis/xiaohongshu/publish.test.js +119 -0
  248. package/dist/clis/xueqiu/search.yaml +2 -1
  249. package/dist/clis/yahoo-finance/quote.js +0 -1
  250. package/dist/clis/youtube/channel.d.ts +1 -0
  251. package/dist/clis/youtube/channel.js +150 -0
  252. package/dist/clis/youtube/comments.d.ts +1 -0
  253. package/dist/clis/youtube/comments.js +95 -0
  254. package/dist/clis/youtube/search.js +0 -1
  255. package/dist/clis/zhihu/search.yaml +2 -1
  256. package/dist/commanderAdapter.d.ts +1 -0
  257. package/dist/commanderAdapter.js +176 -29
  258. package/dist/commanderAdapter.test.d.ts +1 -0
  259. package/dist/commanderAdapter.test.js +62 -0
  260. package/dist/daemon.js +17 -1
  261. package/dist/discovery.js +8 -14
  262. package/dist/doctor.d.ts +1 -0
  263. package/dist/doctor.js +9 -2
  264. package/dist/download/index.js +63 -51
  265. package/dist/download/index.test.js +17 -4
  266. package/dist/errors.d.ts +3 -1
  267. package/dist/errors.js +15 -32
  268. package/dist/execution.d.ts +1 -3
  269. package/dist/execution.js +21 -1
  270. package/dist/external-clis.yaml +0 -17
  271. package/dist/hooks.js +2 -0
  272. package/dist/main.js +5 -0
  273. package/dist/output.js +5 -1
  274. package/dist/pipeline/executor.js +3 -4
  275. package/dist/plugin-manifest.d.ts +70 -0
  276. package/dist/plugin-manifest.js +160 -0
  277. package/dist/plugin-manifest.test.d.ts +4 -0
  278. package/dist/plugin-manifest.test.js +179 -0
  279. package/dist/plugin.d.ts +38 -5
  280. package/dist/plugin.js +267 -33
  281. package/dist/plugin.test.js +220 -3
  282. package/dist/registry.d.ts +4 -0
  283. package/dist/registry.js +2 -0
  284. package/dist/runtime-detect.d.ts +21 -0
  285. package/dist/runtime-detect.js +32 -0
  286. package/dist/runtime-detect.test.d.ts +1 -0
  287. package/dist/runtime-detect.test.js +27 -0
  288. package/dist/runtime.js +1 -1
  289. package/dist/serialization.d.ts +2 -0
  290. package/dist/serialization.js +6 -0
  291. package/dist/types.d.ts +1 -0
  292. package/dist/update-check.d.ts +22 -0
  293. package/dist/update-check.js +112 -0
  294. package/dist/weixin-download.test.d.ts +1 -0
  295. package/dist/weixin-download.test.js +30 -0
  296. package/dist/weread-private-api-regression.test.d.ts +1 -0
  297. package/dist/weread-private-api-regression.test.js +122 -0
  298. package/dist/weread-search-regression.test.d.ts +1 -0
  299. package/dist/weread-search-regression.test.js +39 -0
  300. package/dist/yaml-schema.d.ts +3 -0
  301. package/dist/yaml-schema.js +18 -1
  302. package/docs/.vitepress/config.mts +17 -0
  303. package/docs/adapters/browser/36kr.md +47 -0
  304. package/docs/adapters/browser/douban.md +14 -0
  305. package/docs/adapters/browser/douyin.md +75 -0
  306. package/docs/adapters/browser/imdb.md +47 -0
  307. package/docs/adapters/browser/jd.md +2 -2
  308. package/docs/adapters/browser/linux-do.md +181 -20
  309. package/docs/adapters/browser/paperreview.md +43 -0
  310. package/docs/adapters/browser/producthunt.md +49 -0
  311. package/docs/adapters/browser/twitter.md +6 -0
  312. package/docs/adapters/desktop/chatgpt.md +5 -0
  313. package/docs/adapters/index.md +12 -3
  314. package/docs/advanced/download.md +4 -0
  315. package/docs/advanced/rate-limiter-plugin.md +99 -0
  316. package/docs/guide/electron-app-cli.md +200 -0
  317. package/docs/guide/getting-started.md +1 -0
  318. package/docs/guide/plugins.md +87 -0
  319. package/docs/zh/guide/electron-app-cli.md +188 -0
  320. package/docs/zh/guide/getting-started.md +1 -0
  321. package/docs/zh/guide/plugins.md +65 -0
  322. package/extension/dist/background.js +508 -518
  323. package/extension/manifest.json +6 -2
  324. package/extension/package.json +2 -1
  325. package/extension/popup.html +84 -0
  326. package/extension/popup.js +25 -0
  327. package/extension/scripts/package-release.mjs +179 -0
  328. package/extension/src/background.ts +22 -1
  329. package/package.json +4 -1
  330. package/scripts/postinstall.js +10 -0
  331. package/src/browser/cdp.ts +2 -1
  332. package/src/browser/discover.ts +8 -3
  333. package/src/browser/errors.ts +13 -14
  334. package/src/browser/mcp.ts +2 -1
  335. package/src/build-manifest.test.ts +23 -0
  336. package/src/build-manifest.ts +40 -15
  337. package/src/capabilityRouting.ts +2 -1
  338. package/src/cli.ts +35 -3
  339. package/src/clis/36kr/article.ts +69 -0
  340. package/src/clis/36kr/hot.test.ts +19 -0
  341. package/src/clis/36kr/hot.ts +100 -0
  342. package/src/clis/36kr/news.test.ts +90 -0
  343. package/src/clis/36kr/news.ts +54 -0
  344. package/src/clis/36kr/search.ts +78 -0
  345. package/src/clis/apple-podcasts/search.ts +2 -1
  346. package/src/clis/arxiv/search.ts +2 -2
  347. package/src/clis/bbc/news.ts +0 -1
  348. package/src/clis/bilibili/comments.test.ts +102 -0
  349. package/src/clis/bilibili/comments.ts +44 -0
  350. package/src/clis/chatgpt/ask.ts +28 -14
  351. package/src/clis/chatgpt/ax.ts +180 -1
  352. package/src/clis/chatgpt/model.ts +27 -0
  353. package/src/clis/chatgpt/send.ts +16 -6
  354. package/src/clis/ctrip/search.ts +0 -1
  355. package/src/clis/douban/download.test.ts +196 -0
  356. package/src/clis/douban/download.ts +78 -0
  357. package/src/clis/douban/photos.ts +36 -0
  358. package/src/clis/douban/utils.test.ts +97 -0
  359. package/src/clis/douban/utils.ts +232 -1
  360. package/src/clis/douyin/_shared/browser-fetch.test.ts +38 -0
  361. package/src/clis/douyin/_shared/browser-fetch.ts +45 -0
  362. package/src/clis/douyin/_shared/creation-id.test.ts +26 -0
  363. package/src/clis/douyin/_shared/creation-id.ts +8 -0
  364. package/src/clis/douyin/_shared/imagex-upload.test.ts +113 -0
  365. package/src/clis/douyin/_shared/imagex-upload.ts +76 -0
  366. package/src/clis/douyin/_shared/sts2.ts +20 -0
  367. package/src/clis/douyin/_shared/text-extra.test.ts +42 -0
  368. package/src/clis/douyin/_shared/text-extra.ts +33 -0
  369. package/src/clis/douyin/_shared/timing.test.ts +38 -0
  370. package/src/clis/douyin/_shared/timing.ts +22 -0
  371. package/src/clis/douyin/_shared/tos-upload-short-read.test.ts +102 -0
  372. package/src/clis/douyin/_shared/tos-upload.test.ts +281 -0
  373. package/src/clis/douyin/_shared/tos-upload.ts +444 -0
  374. package/src/clis/douyin/_shared/transcode.test.ts +117 -0
  375. package/src/clis/douyin/_shared/transcode.ts +78 -0
  376. package/src/clis/douyin/_shared/types.ts +29 -0
  377. package/src/clis/douyin/activities.test.ts +25 -0
  378. package/src/clis/douyin/activities.ts +23 -0
  379. package/src/clis/douyin/collections.test.ts +26 -0
  380. package/src/clis/douyin/collections.ts +25 -0
  381. package/src/clis/douyin/delete.test.ts +12 -0
  382. package/src/clis/douyin/delete.ts +20 -0
  383. package/src/clis/douyin/draft.test.ts +12 -0
  384. package/src/clis/douyin/draft.ts +282 -0
  385. package/src/clis/douyin/drafts.test.ts +12 -0
  386. package/src/clis/douyin/drafts.ts +27 -0
  387. package/src/clis/douyin/hashtag.test.ts +28 -0
  388. package/src/clis/douyin/hashtag.ts +56 -0
  389. package/src/clis/douyin/location.test.ts +26 -0
  390. package/src/clis/douyin/location.ts +27 -0
  391. package/src/clis/douyin/profile.test.ts +12 -0
  392. package/src/clis/douyin/profile.ts +37 -0
  393. package/src/clis/douyin/publish.test.ts +45 -0
  394. package/src/clis/douyin/publish.ts +340 -0
  395. package/src/clis/douyin/stats.test.ts +25 -0
  396. package/src/clis/douyin/stats.ts +30 -0
  397. package/src/clis/douyin/update.test.ts +12 -0
  398. package/src/clis/douyin/update.ts +43 -0
  399. package/src/clis/douyin/videos.test.ts +12 -0
  400. package/src/clis/douyin/videos.ts +49 -0
  401. package/src/clis/hackernews/search.yaml +1 -1
  402. package/src/clis/imdb/person.ts +232 -0
  403. package/src/clis/imdb/reviews.ts +111 -0
  404. package/src/clis/imdb/search.ts +179 -0
  405. package/src/clis/imdb/title.ts +121 -0
  406. package/src/clis/imdb/top.ts +67 -0
  407. package/src/clis/imdb/trending.ts +66 -0
  408. package/src/clis/imdb/utils.test.ts +117 -0
  409. package/src/clis/imdb/utils.ts +305 -0
  410. package/src/clis/instagram/search.yaml +2 -1
  411. package/src/clis/jd/item.test.ts +18 -1
  412. package/src/clis/jd/item.ts +18 -15
  413. package/src/clis/linux-do/categories.yaml +38 -9
  414. package/src/clis/linux-do/category.ts +37 -0
  415. package/src/clis/linux-do/feed.test.ts +132 -0
  416. package/src/clis/linux-do/feed.ts +501 -0
  417. package/src/clis/linux-do/hot.ts +26 -0
  418. package/src/clis/linux-do/latest.ts +19 -0
  419. package/src/clis/linux-do/search.yaml +3 -1
  420. package/src/clis/linux-do/tags.yaml +41 -0
  421. package/src/clis/linux-do/topic.yaml +41 -3
  422. package/src/clis/linux-do/user-posts.yaml +67 -0
  423. package/src/clis/linux-do/user-topics.yaml +54 -0
  424. package/src/clis/medium/search.ts +1 -1
  425. package/src/clis/paperreview/commands.test.ts +283 -0
  426. package/src/clis/paperreview/feedback.ts +64 -0
  427. package/src/clis/paperreview/review.ts +47 -0
  428. package/src/clis/paperreview/submit.ts +119 -0
  429. package/src/clis/paperreview/utils.test.ts +68 -0
  430. package/src/clis/paperreview/utils.ts +276 -0
  431. package/src/clis/producthunt/browse.ts +109 -0
  432. package/src/clis/producthunt/hot.ts +127 -0
  433. package/src/clis/producthunt/posts.ts +29 -0
  434. package/src/clis/producthunt/today.ts +37 -0
  435. package/src/clis/producthunt/utils.test.ts +72 -0
  436. package/src/clis/producthunt/utils.ts +122 -0
  437. package/src/clis/reuters/search.ts +0 -1
  438. package/src/clis/twitter/article.ts +5 -28
  439. package/src/clis/twitter/likes.test.ts +91 -0
  440. package/src/clis/twitter/likes.ts +256 -0
  441. package/src/clis/twitter/profile.ts +5 -28
  442. package/src/clis/twitter/search.test.ts +71 -2
  443. package/src/clis/twitter/search.ts +8 -4
  444. package/src/clis/twitter/shared.ts +45 -0
  445. package/src/clis/twitter/timeline.ts +2 -13
  446. package/src/clis/weibo/comments.ts +54 -0
  447. package/src/clis/weibo/feed.ts +57 -0
  448. package/src/clis/weibo/hot.ts +0 -1
  449. package/src/clis/weibo/me.ts +77 -0
  450. package/src/clis/weibo/post.ts +77 -0
  451. package/src/clis/weibo/user.ts +64 -0
  452. package/src/clis/weibo/utils.ts +32 -0
  453. package/src/clis/weixin/download.ts +114 -20
  454. package/src/clis/weread/book.ts +2 -2
  455. package/src/clis/weread/commands.test.ts +57 -0
  456. package/src/clis/weread/highlights.ts +2 -2
  457. package/src/clis/weread/notebooks.ts +2 -2
  458. package/src/clis/weread/notes.ts +3 -3
  459. package/src/clis/weread/search.ts +3 -2
  460. package/src/clis/weread/shelf.ts +2 -2
  461. package/src/clis/weread/utils.test.ts +1 -32
  462. package/src/clis/weread/utils.ts +41 -16
  463. package/src/clis/xiaohongshu/comments.test.ts +96 -0
  464. package/src/clis/xiaohongshu/comments.ts +81 -0
  465. package/src/clis/xiaohongshu/publish.test.ts +137 -0
  466. package/src/clis/xiaohongshu/publish.ts +129 -18
  467. package/src/clis/xueqiu/search.yaml +2 -1
  468. package/src/clis/yahoo-finance/quote.ts +0 -1
  469. package/src/clis/youtube/channel.ts +155 -0
  470. package/src/clis/youtube/comments.ts +97 -0
  471. package/src/clis/youtube/search.ts +0 -1
  472. package/src/clis/zhihu/search.yaml +2 -1
  473. package/src/commanderAdapter.test.ts +78 -0
  474. package/src/commanderAdapter.ts +188 -24
  475. package/src/daemon.ts +19 -1
  476. package/src/discovery.ts +8 -15
  477. package/src/doctor.ts +13 -2
  478. package/src/download/index.test.ts +14 -4
  479. package/src/download/index.ts +67 -55
  480. package/src/errors.ts +25 -66
  481. package/src/execution.ts +28 -3
  482. package/src/external-clis.yaml +0 -17
  483. package/src/hooks.ts +1 -0
  484. package/src/main.ts +6 -0
  485. package/src/output.ts +3 -1
  486. package/src/pipeline/executor.ts +4 -6
  487. package/src/plugin-manifest.test.ts +223 -0
  488. package/src/plugin-manifest.ts +206 -0
  489. package/src/plugin.test.ts +246 -2
  490. package/src/plugin.ts +338 -36
  491. package/src/registry.ts +6 -1
  492. package/src/runtime-detect.test.ts +30 -0
  493. package/src/runtime-detect.ts +36 -0
  494. package/src/runtime.ts +1 -1
  495. package/src/serialization.ts +4 -0
  496. package/src/types.ts +1 -0
  497. package/src/update-check.ts +114 -0
  498. package/src/weixin-download.test.ts +64 -0
  499. package/src/weread-private-api-regression.test.ts +150 -0
  500. package/src/weread-search-regression.test.ts +44 -0
  501. package/src/yaml-schema.ts +20 -0
  502. package/tests/e2e/browser-auth.test.ts +13 -9
  503. package/tests/e2e/browser-public-extended.test.ts +162 -0
  504. package/tests/e2e/browser-public.test.ts +55 -136
  505. package/tests/e2e/helpers.ts +2 -1
  506. package/tests/e2e/public-commands.test.ts +37 -3
  507. package/tests/smoke/api-health.test.ts +1 -1
  508. package/vitest.config.ts +34 -17
  509. package/dist/clis/linux-do/category.yaml +0 -51
  510. package/dist/clis/linux-do/hot.yaml +0 -50
  511. package/dist/clis/linux-do/latest.yaml +0 -40
  512. package/src/clis/linux-do/category.yaml +0 -51
  513. package/src/clis/linux-do/hot.yaml +0 -50
  514. package/src/clis/linux-do/latest.yaml +0 -40
@@ -0,0 +1,23 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { browserFetch } from './_shared/browser-fetch.js';
3
+
4
+ cli({
5
+ site: 'douyin',
6
+ name: 'activities',
7
+ description: '官方活动列表',
8
+ domain: 'creator.douyin.com',
9
+ strategy: Strategy.COOKIE,
10
+ args: [],
11
+ columns: ['activity_id', 'title', 'end_time'],
12
+ func: async (page, _kwargs) => {
13
+ const url = 'https://creator.douyin.com/web/api/media/activity/get/?aid=1128';
14
+ const res = await browserFetch(page, 'GET', url) as {
15
+ activity_list: Array<{ activity_id: string; title: string; end_time: number }>
16
+ };
17
+ return (res.activity_list ?? []).map(a => ({
18
+ activity_id: a.activity_id,
19
+ title: a.title,
20
+ end_time: new Date(a.end_time * 1000).toLocaleString('zh-CN', { timeZone: 'Asia/Tokyo' }),
21
+ }));
22
+ },
23
+ });
@@ -0,0 +1,26 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './collections.js';
4
+
5
+ describe('douyin collections registration', () => {
6
+ it('registers the collections command', () => {
7
+ const registry = getRegistry();
8
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'collections');
9
+ expect(cmd).toBeDefined();
10
+ expect(cmd?.args.some(a => a.name === 'limit')).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 === 'collections');
16
+ expect(cmd?.columns).toContain('mix_id');
17
+ expect(cmd?.columns).toContain('name');
18
+ expect(cmd?.columns).toContain('item_count');
19
+ });
20
+
21
+ it('uses COOKIE strategy', () => {
22
+ const registry = getRegistry();
23
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'collections');
24
+ expect(cmd?.strategy).toBe('cookie');
25
+ });
26
+ });
@@ -0,0 +1,25 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { browserFetch } from './_shared/browser-fetch.js';
3
+
4
+ cli({
5
+ site: 'douyin',
6
+ name: 'collections',
7
+ description: '合集列表',
8
+ domain: 'creator.douyin.com',
9
+ strategy: Strategy.COOKIE,
10
+ args: [
11
+ { name: 'limit', type: 'int', default: 20 },
12
+ ],
13
+ columns: ['mix_id', 'name', 'item_count'],
14
+ func: async (page, kwargs) => {
15
+ const url = `https://creator.douyin.com/web/api/mix/list/?aid=1128&count=${kwargs.limit}`;
16
+ const res = await browserFetch(page, 'GET', url) as {
17
+ mix_list: Array<{ mix_id: string; mix_name: string; item_count: number }>
18
+ };
19
+ return (res.mix_list ?? []).map(m => ({
20
+ mix_id: m.mix_id,
21
+ name: m.mix_name,
22
+ item_count: m.item_count,
23
+ }));
24
+ },
25
+ });
@@ -0,0 +1,12 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './delete.js';
4
+
5
+ describe('douyin delete registration', () => {
6
+ it('registers the delete command', () => {
7
+ const registry = getRegistry();
8
+ const values = [...registry.values()];
9
+ const cmd = values.find(c => c.site === 'douyin' && c.name === 'delete');
10
+ expect(cmd).toBeDefined();
11
+ });
12
+ });
@@ -0,0 +1,20 @@
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: 'delete',
8
+ description: '删除作品',
9
+ domain: 'creator.douyin.com',
10
+ strategy: Strategy.COOKIE,
11
+ args: [
12
+ { name: 'aweme_id', required: true, positional: true, help: '作品 ID' },
13
+ ],
14
+ columns: ['status'],
15
+ func: async (page: IPage, kwargs) => {
16
+ const url = 'https://creator.douyin.com/web/api/media/aweme/delete/?aid=1128';
17
+ await browserFetch(page, 'POST', url, { body: { aweme_id: kwargs.aweme_id } });
18
+ return [{ status: `✅ 已删除 ${kwargs.aweme_id}` }];
19
+ },
20
+ });
@@ -0,0 +1,12 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './draft.js';
4
+
5
+ describe('douyin draft registration', () => {
6
+ it('registers the draft command', () => {
7
+ const registry = getRegistry();
8
+ const values = [...registry.values()];
9
+ const cmd = values.find(c => c.site === 'douyin' && c.name === 'draft');
10
+ expect(cmd).toBeDefined();
11
+ });
12
+ });
@@ -0,0 +1,282 @@
1
+ /**
2
+ * Douyin draft — 6-phase pipeline for saving video as draft.
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. (skipped — no safety check for drafts)
12
+ * 8. create_v2 with is_draft: 1
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 { parseTextExtra, extractHashtagNames } from './_shared/text-extra.js';
28
+ import type { HashtagInfo } from './_shared/text-extra.js';
29
+
30
+ const VISIBILITY_MAP: Record<string, number> = {
31
+ public: 0,
32
+ friends: 1,
33
+ private: 2,
34
+ };
35
+
36
+ const IMAGEX_BASE = 'https://imagex.bytedanceapi.com';
37
+ const IMAGEX_SERVICE_ID = '1147';
38
+
39
+ const DEVICE_PARAMS =
40
+ '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';
41
+
42
+ const DEFAULT_COVER_TOOLS_INFO = JSON.stringify({
43
+ video_cover_source: 2,
44
+ cover_timestamp: 0,
45
+ recommend_timestamp: 0,
46
+ is_cover_edit: 0,
47
+ is_cover_template: 0,
48
+ cover_template_id: '',
49
+ is_text_template: 0,
50
+ text_template_id: '',
51
+ text_template_content: '',
52
+ is_text: 0,
53
+ text_num: 0,
54
+ text_content: '',
55
+ is_use_sticker: 0,
56
+ sticker_id: '',
57
+ is_use_filter: 0,
58
+ filter_id: '',
59
+ is_cover_modify: 0,
60
+ to_status: 0,
61
+ cover_type: 0,
62
+ initial_cover_uri: '',
63
+ cut_coordinate: '',
64
+ });
65
+
66
+ cli({
67
+ site: 'douyin',
68
+ name: 'draft',
69
+ description: '上传视频并保存为草稿',
70
+ domain: 'creator.douyin.com',
71
+ strategy: Strategy.COOKIE,
72
+ args: [
73
+ { name: 'video', required: true, positional: true, help: '视频文件路径' },
74
+ { name: 'title', required: true, help: '视频标题(≤30字)' },
75
+ { name: 'caption', default: '', help: '正文内容(≤1000字,支持 #话题)' },
76
+ { name: 'cover', default: '', help: '封面图片路径' },
77
+ { name: 'visibility', default: 'public', choices: ['public', 'friends', 'private'] },
78
+ ],
79
+ columns: ['status', 'aweme_id'],
80
+ func: async (page: IPage, kwargs) => {
81
+ // ── Fail-fast validation ────────────────────────────────────────────
82
+ const videoPath = path.resolve(kwargs.video as string);
83
+ if (!fs.existsSync(videoPath)) {
84
+ throw new ArgumentError(`视频文件不存在: ${videoPath}`);
85
+ }
86
+ const ext = path.extname(videoPath).toLowerCase();
87
+ if (!['.mp4', '.mov', '.avi', '.webm'].includes(ext)) {
88
+ throw new ArgumentError(`不支持的视频格式: ${ext}(支持 mp4/mov/avi/webm)`);
89
+ }
90
+ const fileSize = fs.statSync(videoPath).size;
91
+
92
+ const title = kwargs.title as string;
93
+ if (title.length > 30) {
94
+ throw new ArgumentError('标题不能超过 30 字');
95
+ }
96
+
97
+ const caption = (kwargs.caption as string) || '';
98
+ if (caption.length > 1000) {
99
+ throw new ArgumentError('正文不能超过 1000 字');
100
+ }
101
+
102
+ const visibilityType = VISIBILITY_MAP[kwargs.visibility as string] ?? 0;
103
+
104
+ const coverPath = kwargs.cover as string;
105
+ if (coverPath) {
106
+ if (!fs.existsSync(path.resolve(coverPath))) {
107
+ throw new ArgumentError(`封面文件不存在: ${path.resolve(coverPath)}`);
108
+ }
109
+ }
110
+
111
+ // ── Phase 1: STS2 credentials ───────────────────────────────────────
112
+ const credentials = await getSts2Credentials(page);
113
+
114
+ // ── Phase 2: Apply TOS upload URL ───────────────────────────────────
115
+ const vodUrl = `https://vod.bytedanceapi.com/?Action=ApplyVideoUpload&ServiceId=1128&Version=2021-01-01&FileType=video&FileSize=${fileSize}`;
116
+ const vodJs = `fetch(${JSON.stringify(vodUrl)}, { credentials: 'include' }).then(r => r.json())`;
117
+ const vodRes = (await page.evaluate(vodJs)) as {
118
+ Result: {
119
+ UploadAddress: {
120
+ VideoId: string;
121
+ UploadHosts: string[];
122
+ StoreInfos: Array<{ Auth: string; StoreUri: string }>;
123
+ };
124
+ };
125
+ };
126
+ const { VideoId: videoId, UploadHosts, StoreInfos } = vodRes.Result.UploadAddress;
127
+ const tosUrl = `https://${UploadHosts[0]}/${StoreInfos[0].StoreUri}`;
128
+ const tosUploadInfo: TosUploadInfo = {
129
+ tos_upload_url: tosUrl,
130
+ auth: StoreInfos[0].Auth,
131
+ video_id: videoId,
132
+ };
133
+
134
+ // ── Phase 3: TOS upload ─────────────────────────────────────────────
135
+ await tosUpload({
136
+ filePath: videoPath,
137
+ uploadInfo: tosUploadInfo,
138
+ credentials,
139
+ onProgress: (uploaded, total) => {
140
+ const pct = Math.round((uploaded / total) * 100);
141
+ process.stderr.write(`\r 上传进度: ${pct}%`);
142
+ },
143
+ });
144
+ process.stderr.write('\n');
145
+
146
+ // ── Phase 4: Cover upload (optional) ────────────────────────────────
147
+ let coverUri = '';
148
+ let coverWidth = 720;
149
+ let coverHeight = 1280;
150
+
151
+ if (kwargs.cover) {
152
+ const resolvedCoverPath = path.resolve(kwargs.cover as string);
153
+
154
+ // 4A: Apply ImageX upload
155
+ const applyUrl = `${IMAGEX_BASE}/?Action=ApplyImageUpload&ServiceId=${IMAGEX_SERVICE_ID}&Version=2018-08-01&UploadNum=1`;
156
+ const applyJs = `fetch(${JSON.stringify(applyUrl)}, { credentials: 'include' }).then(r => r.json())`;
157
+ const applyRes = (await page.evaluate(applyJs)) as {
158
+ Result: {
159
+ UploadAddress: {
160
+ UploadHosts: string[];
161
+ StoreInfos: Array<{ Auth: string; StoreUri: string; UploadHost: string }>;
162
+ };
163
+ };
164
+ };
165
+ const { StoreInfos: imgStoreInfos } = applyRes.Result.UploadAddress;
166
+ const imgUploadUrl = `https://${imgStoreInfos[0].UploadHost}/${imgStoreInfos[0].StoreUri}`;
167
+
168
+ // 4B: Upload image
169
+ const coverStoreUri = await imagexUpload(resolvedCoverPath, {
170
+ upload_url: imgUploadUrl,
171
+ store_uri: imgStoreInfos[0].StoreUri,
172
+ });
173
+
174
+ // 4C: Commit ImageX upload
175
+ const commitUrl = `${IMAGEX_BASE}/?Action=CommitImageUpload&ServiceId=${IMAGEX_SERVICE_ID}&Version=2018-08-01`;
176
+ const commitBody = JSON.stringify({ SuccessObjKeys: [coverStoreUri] });
177
+ const commitJs = `
178
+ fetch(${JSON.stringify(commitUrl)}, {
179
+ method: 'POST',
180
+ credentials: 'include',
181
+ headers: { 'Content-Type': 'application/json' },
182
+ body: ${JSON.stringify(commitBody)}
183
+ }).then(r => r.json())
184
+ `;
185
+ await page.evaluate(commitJs);
186
+
187
+ coverUri = coverStoreUri;
188
+ }
189
+
190
+ // ── Phase 5: Enable video ───────────────────────────────────────────
191
+ const enableUrl = `https://creator.douyin.com/web/api/media/video/enable/?video_id=${videoId}&aid=1128`;
192
+ await browserFetch(page, 'GET', enableUrl);
193
+
194
+ // ── Phase 6: Poll transcode ─────────────────────────────────────────
195
+ const transResult = await pollTranscode(page, videoId);
196
+ coverWidth = transResult.width;
197
+ coverHeight = transResult.height;
198
+ if (!coverUri) {
199
+ coverUri = transResult.poster_uri;
200
+ }
201
+
202
+ // ── Phase 7: SKIP (no safety check for drafts) ──────────────────────
203
+
204
+ // ── Phase 8: create_v2 with is_draft: 1 ────────────────────────────
205
+ const hashtagNames = extractHashtagNames(caption);
206
+ const hashtags: HashtagInfo[] = [];
207
+ let searchFrom = 0;
208
+ for (const name of hashtagNames) {
209
+ const idx = caption.indexOf(`#${name}`, searchFrom);
210
+ if (idx === -1) continue;
211
+ hashtags.push({ name, id: 0, start: idx, end: idx + name.length + 1 });
212
+ searchFrom = idx + name.length + 1;
213
+ }
214
+ const textExtraArr = parseTextExtra(caption, hashtags);
215
+
216
+ const publishBody = {
217
+ item: {
218
+ common: {
219
+ text: caption,
220
+ caption: '',
221
+ item_title: title,
222
+ activity: '[]',
223
+ text_extra: JSON.stringify(textExtraArr),
224
+ challenges: '[]',
225
+ mentions: '[]',
226
+ hashtag_source: '',
227
+ hot_sentence: '',
228
+ interaction_stickers: '[]',
229
+ visibility_type: visibilityType,
230
+ download: 0,
231
+ is_draft: 1,
232
+ creation_id: generateCreationId(),
233
+ media_type: 4,
234
+ video_id: videoId,
235
+ music_source: 0,
236
+ music_id: null,
237
+ },
238
+ cover: {
239
+ poster: coverUri,
240
+ custom_cover_image_height: coverHeight,
241
+ custom_cover_image_width: coverWidth,
242
+ poster_delay: 0,
243
+ cover_tools_info: DEFAULT_COVER_TOOLS_INFO,
244
+ cover_tools_extend_info: '{}',
245
+ },
246
+ mix: {},
247
+ chapter: {
248
+ chapter: JSON.stringify({
249
+ chapter_abstract: '',
250
+ chapter_details: [],
251
+ chapter_type: 0,
252
+ }),
253
+ },
254
+ anchor: {},
255
+ sync: {
256
+ should_sync: false,
257
+ sync_to_toutiao: 0,
258
+ },
259
+ open_platform: {},
260
+ assistant: { is_preview: 0, is_post_assistant: 1 },
261
+ declare: { user_declare_info: '{}' },
262
+ },
263
+ };
264
+
265
+ const publishUrl = `https://creator.douyin.com/web/api/media/aweme/create_v2/?read_aid=2906&${DEVICE_PARAMS}`;
266
+ const publishRes = (await browserFetch(page, 'POST', publishUrl, {
267
+ body: publishBody,
268
+ })) as { status_code: number; aweme_id: string };
269
+
270
+ const awemeId = publishRes.aweme_id;
271
+ if (!awemeId) {
272
+ throw new CommandExecutionError(`草稿保存成功但未返回 aweme_id: ${JSON.stringify(publishRes)}`);
273
+ }
274
+
275
+ return [
276
+ {
277
+ status: '✅ 草稿保存成功!',
278
+ aweme_id: awemeId,
279
+ },
280
+ ];
281
+ },
282
+ });
@@ -0,0 +1,12 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './drafts.js';
4
+
5
+ describe('douyin drafts registration', () => {
6
+ it('registers the drafts command', () => {
7
+ const registry = getRegistry();
8
+ const values = [...registry.values()];
9
+ const cmd = values.find(c => c.site === 'douyin' && c.name === 'drafts');
10
+ expect(cmd).toBeDefined();
11
+ });
12
+ });
@@ -0,0 +1,27 @@
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: 'drafts',
8
+ description: '获取草稿列表',
9
+ domain: 'creator.douyin.com',
10
+ strategy: Strategy.COOKIE,
11
+ args: [
12
+ { name: 'limit', type: 'int', default: 20 },
13
+ ],
14
+ columns: ['aweme_id', 'title', 'create_time'],
15
+ func: async (page: IPage, kwargs) => {
16
+ const url = 'https://creator.douyin.com/web/api/media/aweme/draft/?aid=1128';
17
+ const res = (await browserFetch(page, 'GET', url)) as {
18
+ aweme_list: Array<{ aweme_id: string; desc: string; create_time: number }>;
19
+ };
20
+ const items = (res.aweme_list ?? []).slice(0, kwargs.limit as number);
21
+ return items.map((v) => ({
22
+ aweme_id: v.aweme_id,
23
+ title: v.desc,
24
+ create_time: new Date(v.create_time * 1000).toLocaleString('zh-CN', { timeZone: 'Asia/Tokyo' }),
25
+ }));
26
+ },
27
+ });
@@ -0,0 +1,28 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './hashtag.js';
4
+
5
+ describe('douyin hashtag registration', () => {
6
+ it('registers the hashtag command', () => {
7
+ const registry = getRegistry();
8
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'hashtag');
9
+ expect(cmd).toBeDefined();
10
+ expect(cmd?.args.some(a => a.name === 'action')).toBe(true);
11
+ });
12
+
13
+ it('has all expected args', () => {
14
+ const registry = getRegistry();
15
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'hashtag');
16
+ const argNames = cmd?.args.map(a => a.name) ?? [];
17
+ expect(argNames).toContain('action');
18
+ expect(argNames).toContain('keyword');
19
+ expect(argNames).toContain('cover');
20
+ expect(argNames).toContain('limit');
21
+ });
22
+
23
+ it('uses COOKIE strategy', () => {
24
+ const registry = getRegistry();
25
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'hashtag');
26
+ expect(cmd?.strategy).toBe('cookie');
27
+ });
28
+ });
@@ -0,0 +1,56 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { browserFetch } from './_shared/browser-fetch.js';
3
+ import { ArgumentError } from '../../errors.js';
4
+
5
+ cli({
6
+ site: 'douyin',
7
+ name: 'hashtag',
8
+ description: '话题搜索 / AI推荐 / 热点词',
9
+ domain: 'creator.douyin.com',
10
+ strategy: Strategy.COOKIE,
11
+ args: [
12
+ { name: 'action', required: true, positional: true, choices: ['search', 'suggest', 'hot'], help: 'search=关键词搜索 suggest=AI推荐 hot=热点词' },
13
+ { name: 'keyword', default: '', help: '搜索关键词(search/hot 使用)' },
14
+ { name: 'cover', default: '', help: '封面 URI(suggest 使用)' },
15
+ { name: 'limit', type: 'int', default: 10 },
16
+ ],
17
+ columns: ['name', 'id', 'view_count'],
18
+ func: async (page, kwargs) => {
19
+ const action = kwargs.action as string;
20
+
21
+ if (action === 'search') {
22
+ const url = `https://creator.douyin.com/aweme/v1/challenge/search/?keyword=${encodeURIComponent(kwargs.keyword as string)}&count=${kwargs.limit}&aid=1128`;
23
+ const res = await browserFetch(page, 'GET', url) as {
24
+ challenge_list: Array<{ challenge_info: { cid: string; cha_name: string; view_count: number } }>
25
+ };
26
+ return (res.challenge_list ?? []).map(c => ({
27
+ name: c.challenge_info.cha_name,
28
+ id: c.challenge_info.cid,
29
+ view_count: c.challenge_info.view_count,
30
+ }));
31
+ }
32
+
33
+ if (action === 'suggest') {
34
+ const url = `https://creator.douyin.com/web/api/media/hashtag/rec/?cover_uri=${encodeURIComponent(kwargs.cover as string)}&aid=1128`;
35
+ const res = await browserFetch(page, 'GET', url) as {
36
+ hashtag_list: Array<{ name: string; id: string; view_count: number }>
37
+ };
38
+ return (res.hashtag_list ?? []).map(h => ({ name: h.name, id: h.id, view_count: h.view_count }));
39
+ }
40
+
41
+ if (action === 'hot') {
42
+ const kw = kwargs.keyword as string;
43
+ const url = `https://creator.douyin.com/aweme/v1/hotspot/recommend/?${kw ? `keyword=${encodeURIComponent(kw)}&` : ''}aid=1128`;
44
+ const res = await browserFetch(page, 'GET', url) as {
45
+ hotspot_list: Array<{ sentence: string; hot_value: number }>
46
+ };
47
+ return (res.hotspot_list ?? []).slice(0, kwargs.limit as number).map(h => ({
48
+ name: h.sentence,
49
+ id: '',
50
+ view_count: h.hot_value,
51
+ }));
52
+ }
53
+
54
+ throw new ArgumentError(`未知的 action: ${action}`);
55
+ },
56
+ });
@@ -0,0 +1,26 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './location.js';
4
+
5
+ describe('douyin location registration', () => {
6
+ it('registers the location command', () => {
7
+ const registry = getRegistry();
8
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'location');
9
+ expect(cmd).toBeDefined();
10
+ expect(cmd?.args.some(a => a.name === 'query')).toBe(true);
11
+ });
12
+
13
+ it('has all expected args', () => {
14
+ const registry = getRegistry();
15
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'location');
16
+ const argNames = cmd?.args.map(a => a.name) ?? [];
17
+ expect(argNames).toContain('query');
18
+ expect(argNames).toContain('limit');
19
+ });
20
+
21
+ it('uses COOKIE strategy', () => {
22
+ const registry = getRegistry();
23
+ const cmd = [...registry.values()].find(c => c.site === 'douyin' && c.name === 'location');
24
+ expect(cmd?.strategy).toBe('cookie');
25
+ });
26
+ });
@@ -0,0 +1,27 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { browserFetch } from './_shared/browser-fetch.js';
3
+
4
+ cli({
5
+ site: 'douyin',
6
+ name: 'location',
7
+ description: '地理位置 POI 搜索',
8
+ domain: 'creator.douyin.com',
9
+ strategy: Strategy.COOKIE,
10
+ args: [
11
+ { name: 'query', required: true, positional: true, help: '地名关键词' },
12
+ { name: 'limit', type: 'int', default: 20 },
13
+ ],
14
+ columns: ['poi_id', 'name', 'address', 'city'],
15
+ func: async (page, kwargs) => {
16
+ const url = `https://creator.douyin.com/aweme/v1/life/video_api/search/poi/?keyword=${encodeURIComponent(kwargs.query as string)}&count=${kwargs.limit}&aid=1128`;
17
+ const res = await browserFetch(page, 'GET', url) as {
18
+ poi_list: Array<{ poi_id: string; poi_name: string; address: string; city_name: string }>
19
+ };
20
+ return (res.poi_list ?? []).map(p => ({
21
+ poi_id: p.poi_id,
22
+ name: p.poi_name,
23
+ address: p.address,
24
+ city: p.city_name,
25
+ }));
26
+ },
27
+ });
@@ -0,0 +1,12 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { getRegistry } from '../../registry.js';
3
+ import './profile.js';
4
+
5
+ describe('douyin profile registration', () => {
6
+ it('registers the profile command', () => {
7
+ const registry = getRegistry();
8
+ const values = [...registry.values()];
9
+ const cmd = values.find(c => c.site === 'douyin' && c.name === 'profile');
10
+ expect(cmd).toBeDefined();
11
+ });
12
+ });
@@ -0,0 +1,37 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { browserFetch } from './_shared/browser-fetch.js';
3
+ import { CommandExecutionError } from '../../errors.js';
4
+ import type { IPage } from '../../types.js';
5
+
6
+ cli({
7
+ site: 'douyin',
8
+ name: 'profile',
9
+ description: '获取账号信息',
10
+ domain: 'creator.douyin.com',
11
+ strategy: Strategy.COOKIE,
12
+ args: [],
13
+ columns: ['uid', 'nickname', 'follower_count', 'following_count', 'aweme_count'],
14
+ func: async (page: IPage, _kwargs) => {
15
+ const url = 'https://creator.douyin.com/web/api/media/user/info/?aid=1128';
16
+ const res = (await browserFetch(page, 'GET', url)) as {
17
+ user_info: {
18
+ uid: string;
19
+ nickname: string;
20
+ follower_count: number;
21
+ following_count: number;
22
+ aweme_count: number;
23
+ };
24
+ };
25
+ const u = res.user_info;
26
+ if (!u) throw new CommandExecutionError('用户信息获取失败,请确认已登录 creator.douyin.com');
27
+ return [
28
+ {
29
+ uid: u.uid,
30
+ nickname: u.nickname,
31
+ follower_count: u.follower_count,
32
+ following_count: u.following_count,
33
+ aweme_count: u.aweme_count,
34
+ },
35
+ ];
36
+ },
37
+ });