@jackwener/opencli 1.4.1 → 1.5.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 (369) hide show
  1. package/.github/workflows/build-extension.yml +2 -6
  2. package/.github/workflows/ci.yml +21 -1
  3. package/README.md +35 -6
  4. package/README.zh-CN.md +12 -5
  5. package/SKILL.md +2 -0
  6. package/dist/browser/cdp.d.ts +2 -1
  7. package/dist/browser/cdp.js +5 -0
  8. package/dist/browser/discover.d.ts +4 -1
  9. package/dist/browser/discover.js +6 -2
  10. package/dist/browser/errors.d.ts +2 -2
  11. package/dist/browser/errors.js +4 -12
  12. package/dist/browser/mcp.d.ts +2 -1
  13. package/dist/browser/page.d.ts +3 -0
  14. package/dist/browser/page.js +24 -1
  15. package/dist/build-manifest.d.ts +2 -0
  16. package/dist/build-manifest.js +39 -14
  17. package/dist/build-manifest.test.js +21 -0
  18. package/dist/capabilityRouting.d.ts +2 -0
  19. package/dist/capabilityRouting.js +2 -1
  20. package/dist/cli-manifest.json +1567 -108
  21. package/dist/cli.js +68 -6
  22. package/dist/clis/36kr/article.d.ts +1 -0
  23. package/dist/clis/36kr/article.js +62 -0
  24. package/dist/clis/36kr/hot.d.ts +3 -0
  25. package/dist/clis/36kr/hot.js +80 -0
  26. package/dist/clis/36kr/hot.test.d.ts +1 -0
  27. package/dist/clis/36kr/hot.test.js +15 -0
  28. package/dist/clis/36kr/news.d.ts +1 -0
  29. package/dist/clis/36kr/news.js +51 -0
  30. package/dist/clis/36kr/news.test.d.ts +1 -0
  31. package/dist/clis/36kr/news.test.js +85 -0
  32. package/dist/clis/36kr/search.d.ts +1 -0
  33. package/dist/clis/36kr/search.js +72 -0
  34. package/dist/clis/bilibili/comments.d.ts +5 -0
  35. package/dist/clis/bilibili/comments.js +40 -0
  36. package/dist/clis/bilibili/comments.test.d.ts +1 -0
  37. package/dist/clis/bilibili/comments.test.js +82 -0
  38. package/dist/clis/bluesky/feeds.yaml +29 -0
  39. package/dist/clis/bluesky/followers.yaml +33 -0
  40. package/dist/clis/bluesky/following.yaml +33 -0
  41. package/dist/clis/bluesky/profile.yaml +27 -0
  42. package/dist/clis/bluesky/search.yaml +34 -0
  43. package/dist/clis/bluesky/starter-packs.yaml +34 -0
  44. package/dist/clis/bluesky/thread.yaml +32 -0
  45. package/dist/clis/bluesky/trending.yaml +27 -0
  46. package/dist/clis/bluesky/user.yaml +34 -0
  47. package/dist/clis/chatgpt/ask.js +29 -14
  48. package/dist/clis/chatgpt/ax.d.ts +6 -0
  49. package/dist/clis/chatgpt/ax.js +172 -1
  50. package/dist/clis/chatgpt/model.d.ts +1 -0
  51. package/dist/clis/chatgpt/model.js +24 -0
  52. package/dist/clis/chatgpt/send.js +12 -3
  53. package/dist/clis/douban/download.d.ts +1 -0
  54. package/dist/clis/douban/download.js +67 -0
  55. package/dist/clis/douban/download.test.d.ts +1 -0
  56. package/dist/clis/douban/download.test.js +170 -0
  57. package/dist/clis/douban/photos.d.ts +1 -0
  58. package/dist/clis/douban/photos.js +34 -0
  59. package/dist/clis/douban/utils.d.ts +25 -0
  60. package/dist/clis/douban/utils.js +190 -1
  61. package/dist/clis/douban/utils.test.d.ts +1 -0
  62. package/dist/clis/douban/utils.test.js +64 -0
  63. package/dist/clis/imdb/person.d.ts +1 -0
  64. package/dist/clis/imdb/person.js +203 -0
  65. package/dist/clis/imdb/reviews.d.ts +1 -0
  66. package/dist/clis/imdb/reviews.js +88 -0
  67. package/dist/clis/imdb/search.d.ts +1 -0
  68. package/dist/clis/imdb/search.js +161 -0
  69. package/dist/clis/imdb/title.d.ts +1 -0
  70. package/dist/clis/imdb/title.js +93 -0
  71. package/dist/clis/imdb/top.d.ts +1 -0
  72. package/dist/clis/imdb/top.js +53 -0
  73. package/dist/clis/imdb/trending.d.ts +1 -0
  74. package/dist/clis/imdb/trending.js +52 -0
  75. package/dist/clis/imdb/utils.d.ts +46 -0
  76. package/dist/clis/imdb/utils.js +285 -0
  77. package/dist/clis/imdb/utils.test.d.ts +1 -0
  78. package/dist/clis/imdb/utils.test.js +88 -0
  79. package/dist/clis/jd/item.d.ts +4 -0
  80. package/dist/clis/jd/item.js +16 -15
  81. package/dist/clis/jd/item.test.js +16 -1
  82. package/dist/clis/linux-do/categories.yaml +38 -9
  83. package/dist/clis/linux-do/category.d.ts +1 -0
  84. package/dist/clis/linux-do/category.js +36 -0
  85. package/dist/clis/linux-do/feed.d.ts +45 -0
  86. package/dist/clis/linux-do/feed.js +397 -0
  87. package/dist/clis/linux-do/feed.test.d.ts +1 -0
  88. package/dist/clis/linux-do/feed.test.js +118 -0
  89. package/dist/clis/linux-do/hot.d.ts +1 -0
  90. package/dist/clis/linux-do/hot.js +25 -0
  91. package/dist/clis/linux-do/latest.d.ts +1 -0
  92. package/dist/clis/linux-do/latest.js +18 -0
  93. package/dist/clis/linux-do/tags.yaml +41 -0
  94. package/dist/clis/linux-do/topic.yaml +41 -3
  95. package/dist/clis/linux-do/user-posts.yaml +67 -0
  96. package/dist/clis/linux-do/user-topics.yaml +54 -0
  97. package/dist/clis/paperreview/commands.test.d.ts +3 -0
  98. package/dist/clis/paperreview/commands.test.js +243 -0
  99. package/dist/clis/paperreview/feedback.d.ts +1 -0
  100. package/dist/clis/paperreview/feedback.js +52 -0
  101. package/dist/clis/paperreview/review.d.ts +1 -0
  102. package/dist/clis/paperreview/review.js +37 -0
  103. package/dist/clis/paperreview/submit.d.ts +1 -0
  104. package/dist/clis/paperreview/submit.js +85 -0
  105. package/dist/clis/paperreview/utils.d.ts +46 -0
  106. package/dist/clis/paperreview/utils.js +197 -0
  107. package/dist/clis/paperreview/utils.test.d.ts +1 -0
  108. package/dist/clis/paperreview/utils.test.js +49 -0
  109. package/dist/clis/producthunt/browse.d.ts +1 -0
  110. package/dist/clis/producthunt/browse.js +99 -0
  111. package/dist/clis/producthunt/hot.d.ts +1 -0
  112. package/dist/clis/producthunt/hot.js +110 -0
  113. package/dist/clis/producthunt/posts.d.ts +1 -0
  114. package/dist/clis/producthunt/posts.js +28 -0
  115. package/dist/clis/producthunt/today.d.ts +1 -0
  116. package/dist/clis/producthunt/today.js +35 -0
  117. package/dist/clis/producthunt/utils.d.ts +29 -0
  118. package/dist/clis/producthunt/utils.js +99 -0
  119. package/dist/clis/producthunt/utils.test.d.ts +1 -0
  120. package/dist/clis/producthunt/utils.test.js +64 -0
  121. package/dist/clis/twitter/article.js +4 -28
  122. package/dist/clis/twitter/likes.d.ts +24 -0
  123. package/dist/clis/twitter/likes.js +217 -0
  124. package/dist/clis/twitter/likes.test.d.ts +1 -0
  125. package/dist/clis/twitter/likes.test.js +85 -0
  126. package/dist/clis/twitter/profile.js +4 -28
  127. package/dist/clis/twitter/search.js +2 -1
  128. package/dist/clis/twitter/search.test.js +2 -0
  129. package/dist/clis/twitter/shared.d.ts +6 -0
  130. package/dist/clis/twitter/shared.js +35 -0
  131. package/dist/clis/twitter/timeline.js +2 -13
  132. package/dist/clis/twitter/trending.js +29 -61
  133. package/dist/clis/v2ex/hot.yaml +17 -3
  134. package/dist/clis/weixin/download.d.ts +17 -0
  135. package/dist/clis/weixin/download.js +88 -20
  136. package/dist/clis/weread/book.js +2 -2
  137. package/dist/clis/weread/commands.test.d.ts +3 -0
  138. package/dist/clis/weread/commands.test.js +43 -0
  139. package/dist/clis/weread/highlights.js +2 -2
  140. package/dist/clis/weread/notebooks.js +2 -2
  141. package/dist/clis/weread/notes.js +3 -3
  142. package/dist/clis/weread/shelf.js +2 -2
  143. package/dist/clis/weread/utils.d.ts +4 -4
  144. package/dist/clis/weread/utils.js +32 -14
  145. package/dist/clis/weread/utils.test.js +1 -28
  146. package/dist/clis/xiaohongshu/comments.d.ts +5 -0
  147. package/dist/clis/xiaohongshu/comments.js +74 -0
  148. package/dist/clis/xiaohongshu/comments.test.d.ts +1 -0
  149. package/dist/clis/xiaohongshu/comments.test.js +79 -0
  150. package/dist/clis/xiaohongshu/publish.js +179 -47
  151. package/dist/clis/xiaohongshu/publish.test.d.ts +1 -0
  152. package/dist/clis/xiaohongshu/publish.test.js +131 -0
  153. package/dist/clis/xiaohongshu/search.d.ts +8 -1
  154. package/dist/clis/xiaohongshu/search.js +20 -1
  155. package/dist/clis/xiaohongshu/search.test.d.ts +1 -1
  156. package/dist/clis/xiaohongshu/search.test.js +32 -1
  157. package/dist/commanderAdapter.d.ts +1 -0
  158. package/dist/commanderAdapter.js +176 -29
  159. package/dist/commanderAdapter.test.d.ts +1 -0
  160. package/dist/commanderAdapter.test.js +62 -0
  161. package/dist/daemon.js +17 -1
  162. package/dist/discovery.js +48 -42
  163. package/dist/doctor.d.ts +2 -2
  164. package/dist/doctor.js +11 -4
  165. package/dist/download/index.js +63 -51
  166. package/dist/download/index.test.js +17 -4
  167. package/dist/engine.test.js +42 -0
  168. package/dist/errors.d.ts +4 -2
  169. package/dist/errors.js +17 -34
  170. package/dist/execution.d.ts +1 -3
  171. package/dist/execution.js +66 -8
  172. package/dist/execution.test.d.ts +1 -0
  173. package/dist/execution.test.js +40 -0
  174. package/dist/external.js +6 -1
  175. package/dist/hooks.js +2 -0
  176. package/dist/main.js +6 -0
  177. package/dist/output.js +5 -1
  178. package/dist/pipeline/executor.js +3 -4
  179. package/dist/plugin-manifest.d.ts +70 -0
  180. package/dist/plugin-manifest.js +160 -0
  181. package/dist/plugin-manifest.test.d.ts +4 -0
  182. package/dist/plugin-manifest.test.js +179 -0
  183. package/dist/plugin-scaffold.d.ts +28 -0
  184. package/dist/plugin-scaffold.js +142 -0
  185. package/dist/plugin-scaffold.test.d.ts +4 -0
  186. package/dist/plugin-scaffold.test.js +83 -0
  187. package/dist/plugin.d.ts +82 -11
  188. package/dist/plugin.js +870 -84
  189. package/dist/plugin.test.js +1032 -17
  190. package/dist/registry.d.ts +4 -0
  191. package/dist/registry.js +2 -0
  192. package/dist/runtime-detect.d.ts +21 -0
  193. package/dist/runtime-detect.js +32 -0
  194. package/dist/runtime-detect.test.d.ts +1 -0
  195. package/dist/runtime-detect.test.js +27 -0
  196. package/dist/runtime.d.ts +1 -0
  197. package/dist/runtime.js +2 -2
  198. package/dist/serialization.d.ts +2 -0
  199. package/dist/serialization.js +6 -0
  200. package/dist/types.d.ts +3 -0
  201. package/dist/update-check.d.ts +22 -0
  202. package/dist/update-check.js +112 -0
  203. package/dist/weixin-download.test.d.ts +1 -0
  204. package/dist/weixin-download.test.js +30 -0
  205. package/dist/weread-private-api-regression.test.d.ts +1 -0
  206. package/dist/weread-private-api-regression.test.js +122 -0
  207. package/dist/yaml-schema.d.ts +3 -0
  208. package/dist/yaml-schema.js +18 -1
  209. package/docs/.vitepress/config.mts +4 -0
  210. package/docs/adapters/browser/36kr.md +47 -0
  211. package/docs/adapters/browser/bluesky.md +53 -0
  212. package/docs/adapters/browser/douban.md +14 -0
  213. package/docs/adapters/browser/imdb.md +47 -0
  214. package/docs/adapters/browser/jd.md +2 -2
  215. package/docs/adapters/browser/linux-do.md +181 -20
  216. package/docs/adapters/browser/paperreview.md +43 -0
  217. package/docs/adapters/browser/producthunt.md +49 -0
  218. package/docs/adapters/desktop/chatgpt.md +5 -0
  219. package/docs/adapters/index.md +6 -2
  220. package/docs/advanced/download.md +4 -0
  221. package/docs/advanced/rate-limiter-plugin.md +99 -0
  222. package/docs/guide/electron-app-cli.md +200 -0
  223. package/docs/guide/getting-started.md +1 -0
  224. package/docs/guide/plugins.md +97 -0
  225. package/docs/zh/guide/electron-app-cli.md +188 -0
  226. package/docs/zh/guide/getting-started.md +1 -0
  227. package/docs/zh/guide/plugins.md +65 -0
  228. package/extension/package.json +1 -0
  229. package/extension/scripts/package-release.mjs +179 -0
  230. package/extension/src/background.ts +2 -0
  231. package/package.json +4 -1
  232. package/scripts/postinstall.js +10 -0
  233. package/src/browser/cdp.ts +8 -1
  234. package/src/browser/discover.ts +8 -3
  235. package/src/browser/errors.ts +13 -14
  236. package/src/browser/mcp.ts +2 -1
  237. package/src/browser/page.ts +24 -1
  238. package/src/build-manifest.test.ts +23 -0
  239. package/src/build-manifest.ts +40 -15
  240. package/src/capabilityRouting.ts +2 -1
  241. package/src/cli.ts +69 -6
  242. package/src/clis/36kr/article.ts +69 -0
  243. package/src/clis/36kr/hot.test.ts +19 -0
  244. package/src/clis/36kr/hot.ts +100 -0
  245. package/src/clis/36kr/news.test.ts +90 -0
  246. package/src/clis/36kr/news.ts +54 -0
  247. package/src/clis/36kr/search.ts +78 -0
  248. package/src/clis/bilibili/comments.test.ts +102 -0
  249. package/src/clis/bilibili/comments.ts +44 -0
  250. package/src/clis/bluesky/feeds.yaml +29 -0
  251. package/src/clis/bluesky/followers.yaml +33 -0
  252. package/src/clis/bluesky/following.yaml +33 -0
  253. package/src/clis/bluesky/profile.yaml +27 -0
  254. package/src/clis/bluesky/search.yaml +34 -0
  255. package/src/clis/bluesky/starter-packs.yaml +34 -0
  256. package/src/clis/bluesky/thread.yaml +32 -0
  257. package/src/clis/bluesky/trending.yaml +27 -0
  258. package/src/clis/bluesky/user.yaml +34 -0
  259. package/src/clis/chatgpt/ask.ts +28 -14
  260. package/src/clis/chatgpt/ax.ts +180 -1
  261. package/src/clis/chatgpt/model.ts +27 -0
  262. package/src/clis/chatgpt/send.ts +16 -6
  263. package/src/clis/douban/download.test.ts +196 -0
  264. package/src/clis/douban/download.ts +78 -0
  265. package/src/clis/douban/photos.ts +36 -0
  266. package/src/clis/douban/utils.test.ts +97 -0
  267. package/src/clis/douban/utils.ts +232 -1
  268. package/src/clis/imdb/person.ts +232 -0
  269. package/src/clis/imdb/reviews.ts +111 -0
  270. package/src/clis/imdb/search.ts +179 -0
  271. package/src/clis/imdb/title.ts +121 -0
  272. package/src/clis/imdb/top.ts +67 -0
  273. package/src/clis/imdb/trending.ts +66 -0
  274. package/src/clis/imdb/utils.test.ts +117 -0
  275. package/src/clis/imdb/utils.ts +305 -0
  276. package/src/clis/jd/item.test.ts +18 -1
  277. package/src/clis/jd/item.ts +18 -15
  278. package/src/clis/linux-do/categories.yaml +38 -9
  279. package/src/clis/linux-do/category.ts +37 -0
  280. package/src/clis/linux-do/feed.test.ts +132 -0
  281. package/src/clis/linux-do/feed.ts +501 -0
  282. package/src/clis/linux-do/hot.ts +26 -0
  283. package/src/clis/linux-do/latest.ts +19 -0
  284. package/src/clis/linux-do/tags.yaml +41 -0
  285. package/src/clis/linux-do/topic.yaml +41 -3
  286. package/src/clis/linux-do/user-posts.yaml +67 -0
  287. package/src/clis/linux-do/user-topics.yaml +54 -0
  288. package/src/clis/paperreview/commands.test.ts +283 -0
  289. package/src/clis/paperreview/feedback.ts +64 -0
  290. package/src/clis/paperreview/review.ts +47 -0
  291. package/src/clis/paperreview/submit.ts +119 -0
  292. package/src/clis/paperreview/utils.test.ts +68 -0
  293. package/src/clis/paperreview/utils.ts +276 -0
  294. package/src/clis/producthunt/browse.ts +109 -0
  295. package/src/clis/producthunt/hot.ts +127 -0
  296. package/src/clis/producthunt/posts.ts +29 -0
  297. package/src/clis/producthunt/today.ts +37 -0
  298. package/src/clis/producthunt/utils.test.ts +72 -0
  299. package/src/clis/producthunt/utils.ts +122 -0
  300. package/src/clis/twitter/article.ts +5 -28
  301. package/src/clis/twitter/likes.test.ts +91 -0
  302. package/src/clis/twitter/likes.ts +256 -0
  303. package/src/clis/twitter/profile.ts +5 -28
  304. package/src/clis/twitter/search.test.ts +2 -0
  305. package/src/clis/twitter/search.ts +3 -1
  306. package/src/clis/twitter/shared.ts +45 -0
  307. package/src/clis/twitter/timeline.ts +2 -13
  308. package/src/clis/twitter/trending.ts +29 -77
  309. package/src/clis/v2ex/hot.yaml +17 -3
  310. package/src/clis/weixin/download.ts +114 -20
  311. package/src/clis/weread/book.ts +2 -2
  312. package/src/clis/weread/commands.test.ts +57 -0
  313. package/src/clis/weread/highlights.ts +2 -2
  314. package/src/clis/weread/notebooks.ts +2 -2
  315. package/src/clis/weread/notes.ts +3 -3
  316. package/src/clis/weread/shelf.ts +2 -2
  317. package/src/clis/weread/utils.test.ts +1 -32
  318. package/src/clis/weread/utils.ts +41 -16
  319. package/src/clis/xiaohongshu/comments.test.ts +96 -0
  320. package/src/clis/xiaohongshu/comments.ts +81 -0
  321. package/src/clis/xiaohongshu/publish.test.ts +151 -0
  322. package/src/clis/xiaohongshu/publish.ts +206 -54
  323. package/src/clis/xiaohongshu/search.test.ts +39 -1
  324. package/src/clis/xiaohongshu/search.ts +19 -1
  325. package/src/commanderAdapter.test.ts +78 -0
  326. package/src/commanderAdapter.ts +188 -24
  327. package/src/daemon.ts +19 -1
  328. package/src/discovery.ts +49 -48
  329. package/src/doctor.ts +15 -5
  330. package/src/download/index.test.ts +14 -4
  331. package/src/download/index.ts +67 -55
  332. package/src/engine.test.ts +38 -0
  333. package/src/errors.ts +26 -63
  334. package/src/execution.test.ts +47 -0
  335. package/src/execution.ts +67 -9
  336. package/src/external.ts +6 -1
  337. package/src/hooks.ts +1 -0
  338. package/src/main.ts +7 -0
  339. package/src/output.ts +3 -1
  340. package/src/pipeline/executor.ts +4 -6
  341. package/src/plugin-manifest.test.ts +223 -0
  342. package/src/plugin-manifest.ts +206 -0
  343. package/src/plugin-scaffold.test.ts +98 -0
  344. package/src/plugin-scaffold.ts +170 -0
  345. package/src/plugin.test.ts +1104 -17
  346. package/src/plugin.ts +1101 -86
  347. package/src/registry.ts +6 -1
  348. package/src/runtime-detect.test.ts +30 -0
  349. package/src/runtime-detect.ts +36 -0
  350. package/src/runtime.ts +3 -3
  351. package/src/serialization.ts +4 -0
  352. package/src/types.ts +3 -0
  353. package/src/update-check.ts +114 -0
  354. package/src/weixin-download.test.ts +64 -0
  355. package/src/weread-private-api-regression.test.ts +150 -0
  356. package/src/yaml-schema.ts +20 -0
  357. package/tests/e2e/browser-auth.test.ts +13 -9
  358. package/tests/e2e/browser-public-extended.test.ts +1 -1
  359. package/tests/e2e/browser-public.test.ts +62 -4
  360. package/tests/e2e/helpers.ts +2 -1
  361. package/tests/e2e/public-commands.test.ts +37 -3
  362. package/tests/smoke/api-health.test.ts +1 -1
  363. package/vitest.config.ts +10 -0
  364. package/dist/clis/linux-do/category.yaml +0 -51
  365. package/dist/clis/linux-do/hot.yaml +0 -50
  366. package/dist/clis/linux-do/latest.yaml +0 -40
  367. package/src/clis/linux-do/category.yaml +0 -51
  368. package/src/clis/linux-do/hot.yaml +0 -50
  369. package/src/clis/linux-do/latest.yaml +0 -40
@@ -0,0 +1,46 @@
1
+ import type { IPage } from '../../types.js';
2
+ /**
3
+ * Normalize an IMDb title or person input to a bare ID.
4
+ * Accepts bare IDs, desktop URLs, mobile URLs, and URLs with language prefixes or query params.
5
+ */
6
+ export declare function normalizeImdbId(input: string, prefix: 'tt' | 'nm'): string;
7
+ /**
8
+ * Convert an ISO 8601 duration string to a short human-readable format for table display.
9
+ * Example: PT2H28M -> 2h 28m.
10
+ */
11
+ export declare function formatDuration(iso: string): string;
12
+ /**
13
+ * Force an IMDb page URL to use the English language parameter,
14
+ * reducing structural differences across localized pages.
15
+ */
16
+ export declare function forceEnglishUrl(url: string): string;
17
+ /**
18
+ * Normalize IMDb title-type payloads that may be represented as an object,
19
+ * a raw string, or an empty text field with only an internal id.
20
+ */
21
+ export declare function normalizeImdbTitleType(input: unknown): string;
22
+ /**
23
+ * Extract structured JSON-LD data from the page.
24
+ * Accepts a single type string or an array of types to match against @type.
25
+ */
26
+ export declare function extractJsonLd(page: IPage, type?: string | string[]): Promise<Record<string, unknown> | null>;
27
+ /**
28
+ * Poll until the current IMDb page path matches the expected entity/search path.
29
+ */
30
+ export declare function waitForImdbPath(page: IPage, pathPattern: string, timeoutMs?: number): Promise<boolean>;
31
+ /**
32
+ * Wait until IMDb search results (or the search UI state) has rendered.
33
+ */
34
+ export declare function waitForImdbSearchReady(page: IPage, timeoutMs?: number): Promise<boolean>;
35
+ /**
36
+ * Wait until IMDb review cards (or the page review summary) has rendered.
37
+ */
38
+ export declare function waitForImdbReviewsReady(page: IPage, timeoutMs?: number): Promise<boolean>;
39
+ /**
40
+ * Read the current IMDb entity id from the page URL/canonical metadata.
41
+ */
42
+ export declare function getCurrentImdbId(page: IPage, prefix: 'tt' | 'nm'): Promise<string>;
43
+ /**
44
+ * Detect whether the current page is an IMDb bot-challenge or verification page.
45
+ */
46
+ export declare function isChallengePage(page: IPage): Promise<boolean>;
@@ -0,0 +1,285 @@
1
+ import { ArgumentError } from '../../errors.js';
2
+ /**
3
+ * Normalize an IMDb title or person input to a bare ID.
4
+ * Accepts bare IDs, desktop URLs, mobile URLs, and URLs with language prefixes or query params.
5
+ */
6
+ export function normalizeImdbId(input, prefix) {
7
+ const trimmed = input.trim();
8
+ const barePattern = new RegExp(`^${prefix}\\d{7,8}$`);
9
+ if (barePattern.test(trimmed)) {
10
+ return trimmed;
11
+ }
12
+ const pathPattern = new RegExp(`/(?:[a-z]{2}/)?(?:title|name)/(${prefix}\\d{7,8})(?:[/?#]|$)`, 'i');
13
+ const pathMatch = trimmed.match(pathPattern);
14
+ if (pathMatch) {
15
+ return pathMatch[1];
16
+ }
17
+ throw new ArgumentError(`Invalid IMDb ID: "${input}"`, `Expected ${prefix === 'tt' ? 'title' : 'name'} ID like ${prefix === 'tt' ? 'tt1375666' : 'nm0634240'} or an IMDb URL`);
18
+ }
19
+ /**
20
+ * Convert an ISO 8601 duration string to a short human-readable format for table display.
21
+ * Example: PT2H28M -> 2h 28m.
22
+ */
23
+ export function formatDuration(iso) {
24
+ if (!iso) {
25
+ return '';
26
+ }
27
+ const match = iso.match(/^PT(?:(\d+)H)?(?:(\d+)M)?$/);
28
+ if (!match) {
29
+ return '';
30
+ }
31
+ const parts = [];
32
+ if (match[1]) {
33
+ parts.push(`${match[1]}h`);
34
+ }
35
+ if (match[2]) {
36
+ parts.push(`${match[2]}m`);
37
+ }
38
+ return parts.join(' ');
39
+ }
40
+ /**
41
+ * Force an IMDb page URL to use the English language parameter,
42
+ * reducing structural differences across localized pages.
43
+ */
44
+ export function forceEnglishUrl(url) {
45
+ const parsed = new URL(url);
46
+ parsed.searchParams.set('language', 'en-US');
47
+ return parsed.toString();
48
+ }
49
+ /**
50
+ * Normalize IMDb title-type payloads that may be represented as an object,
51
+ * a raw string, or an empty text field with only an internal id.
52
+ */
53
+ export function normalizeImdbTitleType(input) {
54
+ const raw = (() => {
55
+ if (typeof input === 'string')
56
+ return input;
57
+ if (!input || typeof input !== 'object')
58
+ return '';
59
+ const value = input;
60
+ return typeof value.text === 'string' && value.text.trim()
61
+ ? value.text
62
+ : typeof value.id === 'string'
63
+ ? value.id
64
+ : '';
65
+ })().trim();
66
+ if (!raw)
67
+ return '';
68
+ const known = {
69
+ movie: 'Movie',
70
+ short: 'Short',
71
+ video: 'Video',
72
+ tvEpisode: 'TV Episode',
73
+ tvMiniSeries: 'TV Mini Series',
74
+ tvMovie: 'TV Movie',
75
+ tvSeries: 'TV Series',
76
+ tvShort: 'TV Short',
77
+ tvSpecial: 'TV Special',
78
+ videoGame: 'Video Game',
79
+ };
80
+ return known[raw] ?? raw;
81
+ }
82
+ /**
83
+ * Extract structured JSON-LD data from the page.
84
+ * Accepts a single type string or an array of types to match against @type.
85
+ */
86
+ export async function extractJsonLd(page, type) {
87
+ const filterTypes = type ? (Array.isArray(type) ? type : [type]) : [];
88
+ return page.evaluate(`
89
+ (function() {
90
+ var scripts = document.querySelectorAll('script[type="application/ld+json"]');
91
+ var wantedTypes = ${JSON.stringify(filterTypes)};
92
+
93
+ function matchesType(data) {
94
+ if (wantedTypes.length === 0) {
95
+ return true;
96
+ }
97
+ if (!data || typeof data !== 'object') {
98
+ return false;
99
+ }
100
+ if (wantedTypes.indexOf(data['@type']) !== -1) {
101
+ return true;
102
+ }
103
+ if (Array.isArray(data['@type'])) {
104
+ for (var t = 0; t < data['@type'].length; t++) {
105
+ if (wantedTypes.indexOf(data['@type'][t]) !== -1) return true;
106
+ }
107
+ }
108
+ return false;
109
+ }
110
+
111
+ function findMatch(data) {
112
+ if (Array.isArray(data)) {
113
+ for (var i = 0; i < data.length; i++) {
114
+ var itemMatch = findMatch(data[i]);
115
+ if (itemMatch) {
116
+ return itemMatch;
117
+ }
118
+ }
119
+ return null;
120
+ }
121
+
122
+ if (!data || typeof data !== 'object') {
123
+ return null;
124
+ }
125
+
126
+ if (matchesType(data)) {
127
+ return data;
128
+ }
129
+
130
+ if (Array.isArray(data['@graph'])) {
131
+ return findMatch(data['@graph']);
132
+ }
133
+
134
+ return null;
135
+ }
136
+
137
+ for (var i = 0; i < scripts.length; i++) {
138
+ try {
139
+ var parsed = JSON.parse(scripts[i].textContent || 'null');
140
+ var match = findMatch(parsed);
141
+ if (match) {
142
+ return match;
143
+ }
144
+ } catch (error) {
145
+ void error;
146
+ }
147
+ }
148
+
149
+ return null;
150
+ })()
151
+ `);
152
+ }
153
+ /**
154
+ * Poll until the current IMDb page path matches the expected entity/search path.
155
+ */
156
+ export async function waitForImdbPath(page, pathPattern, timeoutMs = 15000) {
157
+ const result = await page.evaluate(`
158
+ (async function() {
159
+ var deadline = Date.now() + ${timeoutMs};
160
+ var pattern = new RegExp(${JSON.stringify(pathPattern)}, 'i');
161
+ while (Date.now() < deadline) {
162
+ if (pattern.test(window.location.pathname)) {
163
+ return true;
164
+ }
165
+ await new Promise(function(resolve) { setTimeout(resolve, 250); });
166
+ }
167
+ return pattern.test(window.location.pathname);
168
+ })()
169
+ `);
170
+ return Boolean(result);
171
+ }
172
+ /**
173
+ * Wait until IMDb search results (or the search UI state) has rendered.
174
+ */
175
+ export async function waitForImdbSearchReady(page, timeoutMs = 15000) {
176
+ const result = await page.evaluate(`
177
+ (async function() {
178
+ var deadline = Date.now() + ${timeoutMs};
179
+
180
+ function hasSearchResults() {
181
+ var nextDataEl = document.getElementById('__NEXT_DATA__');
182
+ if (nextDataEl) {
183
+ try {
184
+ var nextData = JSON.parse(nextDataEl.textContent || 'null');
185
+ var pageProps = nextData && nextData.props && nextData.props.pageProps;
186
+ var titleResults = (pageProps && pageProps.titleResults && pageProps.titleResults.results) || [];
187
+ var nameResults = (pageProps && pageProps.nameResults && pageProps.nameResults.results) || [];
188
+ if (titleResults.length > 0 || nameResults.length > 0) {
189
+ return true;
190
+ }
191
+ } catch (error) {
192
+ void error;
193
+ }
194
+ }
195
+
196
+ if (document.querySelector('a[href*="/title/"], a[href*="/name/"]')) {
197
+ return true;
198
+ }
199
+
200
+ var body = document.body ? (document.body.textContent || '') : '';
201
+ return body.includes('No results found for') || body.includes('No exact matches');
202
+ }
203
+
204
+ while (Date.now() < deadline) {
205
+ if (hasSearchResults()) {
206
+ return true;
207
+ }
208
+ await new Promise(function(resolve) { setTimeout(resolve, 250); });
209
+ }
210
+
211
+ return hasSearchResults();
212
+ })()
213
+ `);
214
+ return Boolean(result);
215
+ }
216
+ /**
217
+ * Wait until IMDb review cards (or the page review summary) has rendered.
218
+ */
219
+ export async function waitForImdbReviewsReady(page, timeoutMs = 15000) {
220
+ const result = await page.evaluate(`
221
+ (async function() {
222
+ var deadline = Date.now() + ${timeoutMs};
223
+
224
+ function hasReviewContent() {
225
+ if (document.querySelector('article.user-review-item, [data-testid="review-card-parent"], [data-testid="tturv-total-reviews"]')) {
226
+ return true;
227
+ }
228
+ var body = document.body ? (document.body.textContent || '') : '';
229
+ return body.includes('No user reviews') || body.includes('Review this title');
230
+ }
231
+
232
+ while (Date.now() < deadline) {
233
+ if (hasReviewContent()) {
234
+ return true;
235
+ }
236
+ await new Promise(function(resolve) { setTimeout(resolve, 250); });
237
+ }
238
+
239
+ return hasReviewContent();
240
+ })()
241
+ `);
242
+ return Boolean(result);
243
+ }
244
+ /**
245
+ * Read the current IMDb entity id from the page URL/canonical metadata.
246
+ */
247
+ export async function getCurrentImdbId(page, prefix) {
248
+ const result = await page.evaluate(`
249
+ (function() {
250
+ var pattern = new RegExp('(${prefix}\\\\d{7,8})', 'i');
251
+ var candidates = [
252
+ window.location.pathname || '',
253
+ document.querySelector('link[rel="canonical"]')?.getAttribute('href') || '',
254
+ document.querySelector('meta[property="og:url"]')?.getAttribute('content') || ''
255
+ ];
256
+
257
+ for (var i = 0; i < candidates.length; i++) {
258
+ var match = candidates[i].match(pattern);
259
+ if (match) {
260
+ return match[1];
261
+ }
262
+ }
263
+ return '';
264
+ })()
265
+ `);
266
+ return typeof result === 'string' ? result : '';
267
+ }
268
+ /**
269
+ * Detect whether the current page is an IMDb bot-challenge or verification page.
270
+ */
271
+ export async function isChallengePage(page) {
272
+ const result = await page.evaluate(`
273
+ (function() {
274
+ var title = document.title || '';
275
+ var body = document.body ? (document.body.textContent || '') : '';
276
+ return title.includes('Robot Check') ||
277
+ title.includes('Are you a robot') ||
278
+ title.includes('JavaScript is disabled') ||
279
+ body.includes('captcha') ||
280
+ body.includes('verify that you are human') ||
281
+ body.includes('not a robot');
282
+ })()
283
+ `);
284
+ return Boolean(result);
285
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,88 @@
1
+ import { describe, expect, it, vi } from 'vitest';
2
+ import { extractJsonLd, forceEnglishUrl, formatDuration, getCurrentImdbId, isChallengePage, normalizeImdbTitleType, normalizeImdbId, waitForImdbPath, waitForImdbReviewsReady, waitForImdbSearchReady, } from './utils.js';
3
+ describe('normalizeImdbId', () => {
4
+ it('passes through bare ids', () => {
5
+ expect(normalizeImdbId('tt1375666', 'tt')).toBe('tt1375666');
6
+ expect(normalizeImdbId('nm0634240', 'nm')).toBe('nm0634240');
7
+ });
8
+ it('extracts ids from supported urls', () => {
9
+ expect(normalizeImdbId('https://www.imdb.com/title/tt1375666/', 'tt')).toBe('tt1375666');
10
+ expect(normalizeImdbId('https://m.imdb.com/title/tt1375666/', 'tt')).toBe('tt1375666');
11
+ expect(normalizeImdbId('https://www.imdb.com/de/title/tt1375666/?ref_=nv_sr_srsg_0', 'tt')).toBe('tt1375666');
12
+ expect(normalizeImdbId('https://www.imdb.com/name/nm0634240/', 'nm')).toBe('nm0634240');
13
+ });
14
+ it('throws on invalid or mismatched ids', () => {
15
+ expect(() => normalizeImdbId('invalid', 'tt')).toThrow('Invalid IMDb ID');
16
+ expect(() => normalizeImdbId('tt1', 'tt')).toThrow('Invalid IMDb ID');
17
+ expect(() => normalizeImdbId('nm0634240', 'tt')).toThrow('Invalid IMDb ID');
18
+ });
19
+ });
20
+ describe('formatDuration', () => {
21
+ it('converts ISO-8601 durations to a short human format', () => {
22
+ expect(formatDuration('PT2H28M')).toBe('2h 28m');
23
+ expect(formatDuration('PT1H')).toBe('1h');
24
+ expect(formatDuration('PT45M')).toBe('45m');
25
+ expect(formatDuration('PT2H')).toBe('2h');
26
+ });
27
+ it('returns an empty string for invalid input', () => {
28
+ expect(formatDuration('')).toBe('');
29
+ expect(formatDuration('invalid')).toBe('');
30
+ });
31
+ });
32
+ describe('forceEnglishUrl', () => {
33
+ it('adds the English language parameter', () => {
34
+ expect(forceEnglishUrl('https://www.imdb.com/title/tt1375666/')).toContain('language=en-US');
35
+ });
36
+ it('preserves existing query parameters', () => {
37
+ const result = forceEnglishUrl('https://www.imdb.com/title/tt1375666/?ref_=nv');
38
+ expect(result).toContain('language=en-US');
39
+ expect(result).toContain('ref_=nv');
40
+ });
41
+ });
42
+ describe('normalizeImdbTitleType', () => {
43
+ it('maps internal imdb ids to readable labels', () => {
44
+ expect(normalizeImdbTitleType({ id: 'movie', text: '' })).toBe('Movie');
45
+ expect(normalizeImdbTitleType({ id: 'tvSeries', text: '' })).toBe('TV Series');
46
+ expect(normalizeImdbTitleType('short')).toBe('Short');
47
+ });
48
+ it('preserves explicit text labels', () => {
49
+ expect(normalizeImdbTitleType({ id: 'movie', text: 'Feature Film' })).toBe('Feature Film');
50
+ });
51
+ });
52
+ describe('extractJsonLd', () => {
53
+ it('returns the evaluated JSON-LD payload', async () => {
54
+ const page = {
55
+ evaluate: vi.fn().mockResolvedValue({ '@type': 'Movie', name: 'Inception' }),
56
+ };
57
+ await expect(extractJsonLd(page, 'Movie')).resolves.toEqual({ '@type': 'Movie', name: 'Inception' });
58
+ expect(page.evaluate).toHaveBeenCalledTimes(1);
59
+ expect(page.evaluate).toHaveBeenCalledWith(expect.stringContaining('"Movie"'));
60
+ });
61
+ });
62
+ describe('isChallengePage', () => {
63
+ it('returns true when the page evaluation matches a challenge', async () => {
64
+ const page = {
65
+ evaluate: vi.fn().mockResolvedValue(true),
66
+ };
67
+ await expect(isChallengePage(page)).resolves.toBe(true);
68
+ expect(page.evaluate).toHaveBeenCalledTimes(1);
69
+ });
70
+ });
71
+ describe('imdb browser helpers', () => {
72
+ it('reads the current imdb id from page metadata', async () => {
73
+ const page = {
74
+ evaluate: vi.fn().mockResolvedValue('nm0634240'),
75
+ };
76
+ await expect(getCurrentImdbId(page, 'nm')).resolves.toBe('nm0634240');
77
+ expect(page.evaluate).toHaveBeenCalledTimes(1);
78
+ });
79
+ it('wait helpers resolve mocked readiness booleans', async () => {
80
+ const page = {
81
+ evaluate: vi.fn().mockResolvedValue(true),
82
+ };
83
+ await expect(waitForImdbPath(page, '^/find/?$')).resolves.toBe(true);
84
+ await expect(waitForImdbSearchReady(page)).resolves.toBe(true);
85
+ await expect(waitForImdbReviewsReady(page)).resolves.toBe(true);
86
+ expect(page.evaluate).toHaveBeenCalledTimes(3);
87
+ });
88
+ });
@@ -1 +1,5 @@
1
+ declare function extractAvifImages(imageUrls: string[], maxImages: number): string[];
2
+ export declare const __test__: {
3
+ extractAvifImages: typeof extractAvifImages;
4
+ };
1
5
  export {};
@@ -5,10 +5,16 @@
5
5
  * 用法: opencli jd item 100291143898
6
6
  */
7
7
  import { cli, Strategy } from '../../registry.js';
8
+ function extractAvifImages(imageUrls, maxImages) {
9
+ const unique = [...new Set(imageUrls.filter(Boolean))];
10
+ return unique
11
+ .filter((url) => url.includes('.avif') && url.includes('pcpubliccms'))
12
+ .slice(0, maxImages);
13
+ }
8
14
  cli({
9
15
  site: 'jd',
10
16
  name: 'item',
11
- description: '京东商品详情(价格、主图、详情图、规格参数)',
17
+ description: '京东商品详情(价格、店铺、规格参数、AVIF 图片)',
12
18
  domain: 'item.jd.com',
13
19
  strategy: Strategy.COOKIE,
14
20
  args: [
@@ -22,17 +28,17 @@ cli({
22
28
  name: 'images',
23
29
  type: 'int',
24
30
  default: 10,
25
- help: '详情图数量(默认10)',
31
+ help: 'AVIF 图片数量上限(默认10)',
26
32
  },
27
33
  ],
28
- columns: ['title', 'price', 'shop', 'specs', 'mainImages', 'detailImages'],
34
+ columns: ['title', 'price', 'shop', 'specs', 'avifImages'],
29
35
  func: async (page, kwargs) => {
30
36
  const sku = kwargs.sku;
31
37
  const maxImages = kwargs.images;
32
38
  const url = `https://item.jd.com/${sku}.html`;
33
39
  await page.goto(url, { waitUntil: 'load' });
34
40
  await page.wait(2);
35
- // 滚动加载详情图
41
+ // 滚动加载商品详情区域中的延迟图片
36
42
  for (let i = 0; i < 6; i++) {
37
43
  await page.evaluate(`window.scrollTo(0, ${i * 2500})`);
38
44
  await page.wait(1);
@@ -61,17 +67,9 @@ cli({
61
67
  // 所有图片
62
68
  const allImgs = Array.from(document.querySelectorAll('img[src*="360buyimg.com"]'));
63
69
  const srcs = allImgs.map(img => img.src).filter(Boolean);
64
- const unique = [...new Set(srcs)];
65
70
 
66
- // 主图
67
- const mainImgs = unique
68
- .filter(u => u.includes('/n1/') || u.includes('/n3/') || u.includes('/n4/') || u.includes('/img/'))
69
- .slice(0, maxImg);
70
-
71
- // 详情图
72
- const detailImgs = unique
73
- .filter(u => u.includes('/babel/') || u.includes('/popshop/'))
74
- .slice(0, maxImg);
71
+ // 所有 avif 图片(去重,只保留 pcpubliccms CDN)
72
+ const avifImages = ${extractAvifImages.toString()}(srcs, maxImg);
75
73
 
76
74
  // 规格参数:从页面文本提取
77
75
  const text = document.body.innerText;
@@ -88,9 +86,12 @@ cli({
88
86
  }
89
87
  }
90
88
 
91
- return { title, price, shop, specs, mainImages: mainImgs, detailImages: detailImgs, totalImages: unique.length };
89
+ return { title, price, shop, specs, avifImages, totalImages: new Set(srcs).size };
92
90
  })()
93
91
  `);
94
92
  return [data];
95
93
  },
96
94
  });
95
+ export const __test__ = {
96
+ extractAvifImages,
97
+ };
@@ -1,5 +1,6 @@
1
1
  import { describe, expect, it } from 'vitest';
2
2
  import { getRegistry } from '../../registry.js';
3
+ import { __test__ } from './item.js';
3
4
  import './item.js';
4
5
  describe('jd item adapter', () => {
5
6
  const command = getRegistry().get('jd/item');
@@ -23,6 +24,20 @@ describe('jd item adapter', () => {
23
24
  expect(imagesArg.default).toBe(10);
24
25
  });
25
26
  it('includes expected columns', () => {
26
- expect(command.columns).toEqual(expect.arrayContaining(['title', 'price', 'shop', 'specs', 'mainImages', 'detailImages']));
27
+ expect(command.columns).toEqual(expect.arrayContaining(['title', 'price', 'shop', 'specs', 'avifImages']));
28
+ });
29
+ it('extracts only pcpubliccms avif images and respects the limit', () => {
30
+ const result = __test__.extractAvifImages([
31
+ 'https://img14.360buyimg.com/n1/jfs/t1/normal.jpg',
32
+ 'https://img10.360buyimg.com/imgzone/jfs/t1/detail.avif',
33
+ 'https://pcpubliccms.jd.com/image1.avif',
34
+ 'https://pcpubliccms.jd.com/image1.avif',
35
+ 'https://pcpubliccms.jd.com/image2.avif?x=1',
36
+ 'https://example.com/not-jd.avif',
37
+ ], 2);
38
+ expect(result).toEqual([
39
+ 'https://pcpubliccms.jd.com/image1.avif',
40
+ 'https://pcpubliccms.jd.com/image2.avif?x=1',
41
+ ]);
27
42
  });
28
43
  });
@@ -2,9 +2,14 @@ site: linux-do
2
2
  name: categories
3
3
  description: linux.do 分类列表
4
4
  domain: linux.do
5
+ strategy: cookie
5
6
  browser: true
6
7
 
7
8
  args:
9
+ subcategories:
10
+ type: boolean
11
+ default: false
12
+ description: Include subcategories
8
13
  limit:
9
14
  type: int
10
15
  default: 20
@@ -20,13 +25,39 @@ pipeline:
20
25
  let data;
21
26
  try { data = await res.json(); } catch { throw new Error('响应不是有效 JSON - 请先登录 linux.do'); }
22
27
  const cats = data?.category_list?.categories || [];
23
- return cats.slice(0, ${{ args.limit }}).map(c => ({
24
- name: c.name,
25
- slug: c.slug,
26
- id: c.id,
27
- topics: c.topic_count,
28
- description: (c.description_text || '').slice(0, 80),
29
- }));
28
+ const showSub = ${{ args.subcategories }};
29
+ const results = [];
30
+ const limit = ${{ args.limit }};
31
+ for (const c of cats.slice(0, ${{ args.limit }})) {
32
+ results.push({
33
+ name: c.name,
34
+ slug: c.slug,
35
+ id: c.id,
36
+ topics: c.topic_count,
37
+ description: (c.description_text || '').slice(0, 80),
38
+ });
39
+ if (results.length >= limit) break;
40
+ if (showSub && c.subcategory_ids && c.subcategory_ids.length > 0) {
41
+ const subRes = await fetch('/categories.json?parent_category_id=' + c.id, { credentials: 'include' });
42
+ if (subRes.ok) {
43
+ let subData;
44
+ try { subData = await subRes.json(); } catch { continue; }
45
+ const subCats = subData?.category_list?.categories || [];
46
+ for (const sc of subCats) {
47
+ results.push({
48
+ name: c.name + ' / ' + sc.name,
49
+ slug: sc.slug,
50
+ id: sc.id,
51
+ topics: sc.topic_count,
52
+ description: (sc.description_text || '').slice(0, 80),
53
+ });
54
+ if (results.length >= limit) break;
55
+ }
56
+ }
57
+ }
58
+ if (results.length >= limit) break;
59
+ }
60
+ return results;
30
61
  })()
31
62
 
32
63
  - map:
@@ -36,6 +67,4 @@ pipeline:
36
67
  topics: ${{ item.topics }}
37
68
  description: ${{ item.description }}
38
69
 
39
- - limit: ${{ args.limit }}
40
-
41
70
  columns: [name, slug, id, topics, description]
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,36 @@
1
+ import { cli, Strategy } from '../../registry.js';
2
+ import { buildLinuxDoCompatFooter, executeLinuxDoFeed } from './feed.js';
3
+ cli({
4
+ site: 'linux-do',
5
+ name: 'category',
6
+ description: 'linux.do 分类内话题',
7
+ domain: 'linux.do',
8
+ strategy: Strategy.COOKIE,
9
+ browser: true,
10
+ columns: ['title', 'replies', 'created', 'likes', 'views', 'url'],
11
+ deprecated: 'opencli linux-do category is kept for backward compatibility.',
12
+ replacedBy: 'opencli linux-do feed --category <id-or-name>',
13
+ args: [
14
+ {
15
+ name: 'slug',
16
+ positional: true,
17
+ type: 'str',
18
+ required: true,
19
+ help: 'Category slug (legacy compatibility argument)',
20
+ },
21
+ {
22
+ name: 'id',
23
+ positional: true,
24
+ type: 'int',
25
+ required: true,
26
+ help: 'Category ID',
27
+ },
28
+ { name: 'limit', type: 'int', default: 20, help: 'Number of items (per_page)' },
29
+ ],
30
+ func: async (page, kwargs) => executeLinuxDoFeed(page, {
31
+ limit: kwargs.limit,
32
+ category: String(kwargs.id),
33
+ view: 'latest',
34
+ }),
35
+ footerExtra: (kwargs) => buildLinuxDoCompatFooter(`opencli linux-do feed --category ${kwargs.id ?? '<id>'}`),
36
+ });
@@ -0,0 +1,45 @@
1
+ import { type Arg, type CommandArgs } from '../../registry.js';
2
+ import type { IPage } from '../../types.js';
3
+ interface LinuxDoTagRecord {
4
+ id: number;
5
+ slug: string;
6
+ name: string;
7
+ }
8
+ interface LinuxDoCategoryRecord {
9
+ id: number;
10
+ name: string;
11
+ description: string;
12
+ slug: string;
13
+ parentCategoryId: number | null;
14
+ }
15
+ interface ResolvedLinuxDoCategory extends LinuxDoCategoryRecord {
16
+ parent: LinuxDoCategoryRecord | null;
17
+ }
18
+ interface FeedRequest {
19
+ url: string;
20
+ }
21
+ interface TopicListItem {
22
+ title: string;
23
+ replies: number;
24
+ created: string;
25
+ likes: number;
26
+ views: number;
27
+ url: string;
28
+ }
29
+ /**
30
+ * 将命令参数转换为最终请求地址
31
+ */
32
+ declare function resolveFeedRequest(page: IPage | null, kwargs: Record<string, any>): Promise<FeedRequest>;
33
+ export declare const LINUX_DO_FEED_ARGS: Arg[];
34
+ export declare function executeLinuxDoFeed(page: IPage | null, kwargs: CommandArgs): Promise<TopicListItem[]>;
35
+ export declare function buildLinuxDoCompatFooter(replacement: string): string;
36
+ export declare const __test__: {
37
+ resetMetadataCaches(): void;
38
+ setLiveMetadataForTests({ tags, categories, }: {
39
+ tags?: LinuxDoTagRecord[] | null;
40
+ categories?: ResolvedLinuxDoCategory[] | null;
41
+ }): void;
42
+ setCacheDirForTests(dir: string | null): void;
43
+ resolveFeedRequest: typeof resolveFeedRequest;
44
+ };
45
+ export {};