@jackwener/opencli 1.7.12 → 1.7.13

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 (407) hide show
  1. package/README.md +8 -7
  2. package/README.zh-CN.md +9 -8
  3. package/cli-manifest.json +12194 -6843
  4. package/clis/1point3acres/digest.js +35 -0
  5. package/clis/1point3acres/forum.js +51 -0
  6. package/clis/1point3acres/forums.js +44 -0
  7. package/clis/1point3acres/hot.js +35 -0
  8. package/clis/1point3acres/latest.js +35 -0
  9. package/clis/1point3acres/notifications.js +64 -0
  10. package/clis/1point3acres/search.js +71 -0
  11. package/clis/1point3acres/thread.js +117 -0
  12. package/clis/1point3acres/user.js +77 -0
  13. package/clis/1point3acres/utils.js +247 -0
  14. package/clis/_shared/desktop-commands.js +4 -0
  15. package/clis/aibase/news.js +110 -0
  16. package/clis/aibase/news.test.js +59 -0
  17. package/clis/amazon/discussion.test.js +1 -28
  18. package/clis/antigravity/watch.js +3 -2
  19. package/clis/arxiv/author.js +44 -0
  20. package/clis/baidu-scholar/search.js +0 -1
  21. package/clis/bbc/topic.js +57 -0
  22. package/clis/bbc/utils.js +79 -0
  23. package/clis/chaoxing/assignments.js +1 -1
  24. package/clis/chaoxing/exams.js +1 -1
  25. package/clis/chatgpt/ask.js +57 -0
  26. package/clis/chatgpt/commands.test.js +45 -0
  27. package/clis/chatgpt/detail.js +46 -0
  28. package/clis/chatgpt/history.js +39 -0
  29. package/clis/chatgpt/image.js +12 -11
  30. package/clis/chatgpt/image.test.js +23 -0
  31. package/clis/chatgpt/new.js +25 -0
  32. package/clis/chatgpt/read.js +43 -0
  33. package/clis/chatgpt/send.js +46 -0
  34. package/clis/chatgpt/status.js +29 -0
  35. package/clis/chatgpt/utils.js +294 -4
  36. package/clis/chatgpt/utils.test.js +13 -0
  37. package/clis/chatgpt-app/ask.js +6 -3
  38. package/clis/chatwise/ask.js +16 -43
  39. package/clis/chatwise/composer.test.js +186 -0
  40. package/clis/chatwise/send.js +2 -24
  41. package/clis/chatwise/utils.js +143 -0
  42. package/clis/claude/ask.js +1 -1
  43. package/clis/claude/detail.js +1 -0
  44. package/clis/claude/history.js +1 -0
  45. package/clis/claude/new.js +1 -0
  46. package/clis/claude/read.js +1 -0
  47. package/clis/claude/send.js +1 -0
  48. package/clis/claude/status.js +1 -0
  49. package/clis/codex/ask.js +15 -9
  50. package/clis/codex/history.js +16 -33
  51. package/clis/codex/projects.js +28 -0
  52. package/clis/codex/read.js +10 -4
  53. package/clis/codex/send.js +10 -3
  54. package/clis/codex/sidebar.js +356 -0
  55. package/clis/codex/sidebar.test.js +329 -0
  56. package/clis/coingecko/categories.js +75 -0
  57. package/clis/coingecko/coin.js +107 -0
  58. package/clis/coingecko/coingecko.test.js +109 -0
  59. package/clis/coingecko/derivatives.js +84 -0
  60. package/clis/coingecko/exchanges.js +74 -0
  61. package/clis/coingecko/global.js +71 -0
  62. package/clis/coingecko/top.js +64 -0
  63. package/clis/coingecko/trending.js +55 -0
  64. package/clis/coupang/add-to-cart.js +21 -13
  65. package/clis/coupang/coupang.test.js +159 -0
  66. package/clis/coupang/product.js +257 -0
  67. package/clis/coupang/search.js +38 -16
  68. package/clis/coupang/utils.js +55 -1
  69. package/clis/crates/crate.js +62 -0
  70. package/clis/crates/search.js +44 -0
  71. package/clis/crates/utils.js +72 -0
  72. package/clis/ctrip/ctrip.test.js +234 -0
  73. package/clis/ctrip/hotel-suggest.js +45 -0
  74. package/clis/ctrip/search.js +22 -68
  75. package/clis/ctrip/utils.js +175 -0
  76. package/clis/cursor/ask.js +6 -3
  77. package/clis/dblp/author.js +133 -0
  78. package/clis/dblp/venue.js +64 -0
  79. package/clis/deepseek/ask.js +12 -7
  80. package/clis/deepseek/ask.test.js +13 -13
  81. package/clis/deepseek/detail.js +38 -0
  82. package/clis/deepseek/detail.test.js +81 -0
  83. package/clis/deepseek/history.js +1 -0
  84. package/clis/deepseek/new.js +1 -0
  85. package/clis/deepseek/read.js +1 -0
  86. package/clis/deepseek/send.js +140 -0
  87. package/clis/deepseek/send.test.js +107 -0
  88. package/clis/deepseek/status.js +1 -0
  89. package/clis/deepseek/utils.js +66 -0
  90. package/clis/deepseek/utils.test.js +107 -1
  91. package/clis/defillama/defillama.test.js +99 -0
  92. package/clis/defillama/protocol.js +84 -0
  93. package/clis/defillama/protocols.js +55 -0
  94. package/clis/defillama/utils.js +99 -0
  95. package/clis/devto/latest.js +74 -0
  96. package/clis/dockerhub/image.js +52 -0
  97. package/clis/dockerhub/search.js +47 -0
  98. package/clis/dockerhub/utils.js +100 -0
  99. package/clis/doubao/ask.js +7 -3
  100. package/clis/doubao/detail.js +1 -0
  101. package/clis/doubao/history.js +1 -0
  102. package/clis/doubao/meeting-summary.js +1 -0
  103. package/clis/doubao/meeting-transcript.js +1 -0
  104. package/clis/doubao/new.js +1 -0
  105. package/clis/doubao/read.js +1 -0
  106. package/clis/doubao/send.js +1 -0
  107. package/clis/doubao/status.js +1 -0
  108. package/clis/douyin/draft.test.js +1 -30
  109. package/clis/endoflife/endoflife.test.js +51 -0
  110. package/clis/endoflife/product.js +55 -0
  111. package/clis/endoflife/utils.js +89 -0
  112. package/clis/facebook/__fixtures__/notifications-page.html +13 -0
  113. package/clis/facebook/notifications.js +326 -30
  114. package/clis/facebook/notifications.test.js +458 -0
  115. package/clis/flathub/app.js +71 -0
  116. package/clis/flathub/flathub.test.js +90 -0
  117. package/clis/flathub/search.js +80 -0
  118. package/clis/flathub/utils.js +114 -0
  119. package/clis/gemini/ask.js +7 -3
  120. package/clis/gemini/ask.test.js +2 -2
  121. package/clis/gemini/deep-research-result.js +6 -2
  122. package/clis/gemini/deep-research-result.test.js +15 -14
  123. package/clis/gemini/deep-research.js +8 -4
  124. package/clis/gemini/deep-research.test.js +15 -18
  125. package/clis/gemini/image.js +7 -2
  126. package/clis/gemini/new.js +1 -0
  127. package/clis/gemini/utils.js +0 -4
  128. package/clis/google-scholar/cite.js +0 -1
  129. package/clis/google-scholar/profile.js +0 -1
  130. package/clis/google-scholar/search.js +0 -1
  131. package/clis/goproxy/goproxy.test.js +103 -0
  132. package/clis/goproxy/module.js +47 -0
  133. package/clis/goproxy/utils.js +165 -0
  134. package/clis/goproxy/versions.js +59 -0
  135. package/clis/gov-law/recent.js +0 -1
  136. package/clis/gov-law/search.js +0 -1
  137. package/clis/gov-policy/__fixtures__/recent.html +16 -0
  138. package/clis/gov-policy/__fixtures__/search.html +41 -0
  139. package/clis/gov-policy/gov-policy.test.js +224 -0
  140. package/clis/gov-policy/recent.js +66 -24
  141. package/clis/gov-policy/search.js +65 -23
  142. package/clis/gov-policy/utils.js +54 -0
  143. package/clis/grok/ask.js +49 -265
  144. package/clis/grok/ask.test.js +21 -46
  145. package/clis/grok/detail.js +60 -0
  146. package/clis/grok/history.js +48 -0
  147. package/clis/grok/{image.ts → image.js} +56 -70
  148. package/clis/grok/image.test.ts +20 -0
  149. package/clis/grok/new.js +20 -0
  150. package/clis/grok/read.js +39 -0
  151. package/clis/grok/send.js +50 -0
  152. package/clis/grok/status.js +41 -0
  153. package/clis/grok/utils.js +326 -0
  154. package/clis/grok/utils.test.js +103 -0
  155. package/clis/hf/datasets.js +88 -0
  156. package/clis/hf/hf.test.js +16 -0
  157. package/clis/hf/models.js +91 -0
  158. package/clis/hf/paper.js +79 -0
  159. package/clis/hf/spaces.js +101 -0
  160. package/clis/hf/top.js +1 -0
  161. package/clis/homebrew/cask.js +39 -0
  162. package/clis/homebrew/formula.js +41 -0
  163. package/clis/homebrew/popular.js +54 -0
  164. package/clis/homebrew/utils.js +100 -0
  165. package/clis/hupu/__fixtures__/hot-home.html +64 -0
  166. package/clis/hupu/detail.js +0 -1
  167. package/clis/hupu/hot.js +156 -35
  168. package/clis/hupu/hot.test.js +224 -0
  169. package/clis/hupu/search.js +0 -1
  170. package/clis/instagram/note.js +1 -1
  171. package/clis/instagram/note.test.js +1 -29
  172. package/clis/instagram/post.js +1 -1
  173. package/clis/instagram/post.test.js +1 -1
  174. package/clis/instagram/reel.js +1 -1
  175. package/clis/instagram/story.js +1 -1
  176. package/clis/instagram/story.test.js +1 -34
  177. package/clis/jd/commands.test.js +1 -24
  178. package/clis/lichess/lichess.test.js +85 -0
  179. package/clis/lichess/top.js +46 -0
  180. package/clis/lichess/user.js +91 -0
  181. package/clis/lichess/utils.js +97 -0
  182. package/clis/linkedin/search.js +107 -10
  183. package/clis/linkedin/search.test.js +222 -0
  184. package/clis/linux-do/feed.js +2 -5
  185. package/clis/linux-do/feed.test.js +35 -0
  186. package/clis/lobsters/domain.js +92 -0
  187. package/clis/maven/artifact.js +49 -0
  188. package/clis/maven/search.js +51 -0
  189. package/clis/maven/utils.js +110 -0
  190. package/clis/mdn/search.js +97 -0
  191. package/clis/medium/tag.js +135 -0
  192. package/clis/npm/downloads.js +59 -0
  193. package/clis/npm/package.js +70 -0
  194. package/clis/npm/search.js +49 -0
  195. package/clis/npm/utils.js +76 -0
  196. package/clis/nuget/nuget.test.js +111 -0
  197. package/clis/nuget/package.js +101 -0
  198. package/clis/nuget/search.js +69 -0
  199. package/clis/nuget/utils.js +87 -0
  200. package/clis/nvd/cve.js +121 -0
  201. package/clis/oeis/oeis.test.js +88 -0
  202. package/clis/oeis/search.js +63 -0
  203. package/clis/oeis/sequence.js +71 -0
  204. package/clis/oeis/utils.js +88 -0
  205. package/clis/openalex/search.js +69 -0
  206. package/clis/openalex/utils.js +160 -0
  207. package/clis/openalex/work.js +65 -0
  208. package/clis/openfda/drug-label.js +74 -0
  209. package/clis/openfda/food-recall.js +65 -0
  210. package/clis/openfda/openfda.test.js +114 -0
  211. package/clis/openfda/utils.js +67 -0
  212. package/clis/osv/osv.test.js +97 -0
  213. package/clis/osv/query.js +72 -0
  214. package/clis/osv/utils.js +169 -0
  215. package/clis/osv/vulnerability.js +54 -0
  216. package/clis/packagist/package.js +49 -0
  217. package/clis/packagist/search.js +43 -0
  218. package/clis/packagist/utils.js +113 -0
  219. package/clis/paperreview/feedback.js +1 -1
  220. package/clis/paperreview/review.js +1 -1
  221. package/clis/paperreview/submit.js +1 -1
  222. package/clis/pixiv/download.test.js +1 -1
  223. package/clis/pixiv/illusts.test.js +1 -1
  224. package/clis/pixiv/search.test.js +1 -1
  225. package/clis/pubmed/article.js +50 -0
  226. package/clis/pubmed/author.js +64 -0
  227. package/clis/pubmed/citations.js +36 -0
  228. package/clis/pubmed/pubmed.test.js +276 -0
  229. package/clis/pubmed/related.js +45 -0
  230. package/clis/pubmed/search.js +75 -0
  231. package/clis/pubmed/utils.js +309 -0
  232. package/clis/pypi/downloads.js +66 -0
  233. package/clis/pypi/package.js +79 -0
  234. package/clis/pypi/utils.js +55 -0
  235. package/clis/quark/mv.js +1 -1
  236. package/clis/quark/save.js +1 -1
  237. package/clis/qwen/ask.js +85 -0
  238. package/clis/qwen/detail.js +62 -0
  239. package/clis/qwen/history.js +61 -0
  240. package/clis/qwen/image.js +179 -0
  241. package/clis/qwen/new.js +23 -0
  242. package/clis/qwen/read.js +41 -0
  243. package/clis/qwen/send.js +55 -0
  244. package/clis/qwen/status.js +37 -0
  245. package/clis/qwen/utils.js +409 -0
  246. package/clis/qwen/utils.test.js +45 -0
  247. package/clis/rest-countries/country.js +65 -0
  248. package/clis/rest-countries/region.js +64 -0
  249. package/clis/rest-countries/rest-countries.test.js +83 -0
  250. package/clis/rest-countries/utils.js +126 -0
  251. package/clis/reuters/article-detail.js +53 -0
  252. package/clis/reuters/reuters.test.js +299 -0
  253. package/clis/reuters/search.js +45 -34
  254. package/clis/reuters/utils.js +159 -0
  255. package/clis/rfc/rfc.js +52 -0
  256. package/clis/rfc/rfc.test.js +74 -0
  257. package/clis/rfc/utils.js +72 -0
  258. package/clis/rubygems/gem.js +42 -0
  259. package/clis/rubygems/search.js +47 -0
  260. package/clis/rubygems/utils.js +86 -0
  261. package/clis/stackoverflow/related.js +66 -0
  262. package/clis/stackoverflow/stackoverflow.test.js +58 -0
  263. package/clis/stackoverflow/tag.js +60 -0
  264. package/clis/stackoverflow/user.js +50 -0
  265. package/clis/stackoverflow/utils.js +118 -0
  266. package/clis/steam/app.js +67 -0
  267. package/clis/steam/search.js +58 -0
  268. package/clis/steam/steam.test.js +46 -0
  269. package/clis/steam/utils.js +107 -0
  270. package/clis/taobao/commands.test.js +1 -24
  271. package/clis/test-utils.js +61 -0
  272. package/clis/tieba/hot.js +0 -1
  273. package/clis/tiktok/comment.js +128 -41
  274. package/clis/tiktok/creator-videos.js +270 -0
  275. package/clis/tiktok/creator-videos.test.js +113 -0
  276. package/clis/tiktok/explore.js +137 -29
  277. package/clis/tiktok/follow.js +115 -33
  278. package/clis/tiktok/following.js +157 -36
  279. package/clis/tiktok/friends.js +139 -37
  280. package/clis/tiktok/live.js +137 -41
  281. package/clis/tiktok/notifications.js +141 -38
  282. package/clis/tiktok/refactor.test.js +389 -0
  283. package/clis/tiktok/unfollow.js +124 -38
  284. package/clis/tiktok/user.js +203 -29
  285. package/clis/tiktok/utils.js +505 -0
  286. package/clis/tiktok/write-refactor.test.js +370 -0
  287. package/clis/toutiao/articles.js +36 -62
  288. package/clis/toutiao/hot.js +63 -0
  289. package/clis/toutiao/toutiao.test.js +378 -0
  290. package/clis/toutiao/utils.js +161 -0
  291. package/clis/tvmaze/search.js +61 -0
  292. package/clis/tvmaze/show.js +60 -0
  293. package/clis/tvmaze/tvmaze.test.js +93 -0
  294. package/clis/tvmaze/utils.js +110 -0
  295. package/clis/twitter/accept.js +1 -1
  296. package/clis/twitter/followers.js +134 -69
  297. package/clis/twitter/reply-dm.js +1 -1
  298. package/clis/twitter/reply.test.js +1 -29
  299. package/clis/uisdc/news.js +105 -0
  300. package/clis/uisdc/news.test.js +66 -0
  301. package/clis/wanfang/search.js +0 -1
  302. package/clis/web/read.js +47 -17
  303. package/clis/web/read.test.js +101 -1
  304. package/clis/weixin/create-draft.js +1 -1
  305. package/clis/weixin/drafts.js +1 -1
  306. package/clis/weixin/drafts.test.js +5 -1
  307. package/clis/weixin/search.js +157 -0
  308. package/clis/weixin/search.test.js +227 -0
  309. package/clis/wikidata/entity.js +60 -0
  310. package/clis/wikidata/search.js +50 -0
  311. package/clis/wikidata/utils.js +117 -0
  312. package/clis/wikidata/wikidata.test.js +83 -0
  313. package/clis/wikipedia/page.js +95 -0
  314. package/clis/wttr/current.js +63 -0
  315. package/clis/wttr/forecast.js +71 -0
  316. package/clis/wttr/utils.js +50 -0
  317. package/clis/wttr/wttr.test.js +84 -0
  318. package/clis/xianyu/chat.js +16 -4
  319. package/clis/xianyu/chat.test.js +64 -0
  320. package/clis/xianyu/publish.js +485 -0
  321. package/clis/xianyu/publish.test.js +220 -0
  322. package/clis/xiaoe/catalog.js +105 -40
  323. package/clis/xiaoe/content.js +164 -29
  324. package/clis/xiaoe/courses.js +86 -29
  325. package/clis/xiaoe/xiaoe.test.js +486 -0
  326. package/clis/xiaohongshu/creator-notes-summary.js +1 -1
  327. package/clis/xiaohongshu/publish.js +16 -3
  328. package/clis/xiaohongshu/publish.test.js +46 -1
  329. package/clis/youtube/transcript.js +13 -19
  330. package/clis/youtube/transcript.test.js +17 -0
  331. package/clis/yuanbao/ask.js +17 -66
  332. package/clis/yuanbao/ask.test.js +5 -5
  333. package/clis/yuanbao/detail.js +65 -0
  334. package/clis/yuanbao/history.js +51 -0
  335. package/clis/yuanbao/new.js +1 -0
  336. package/clis/yuanbao/read.js +38 -0
  337. package/clis/yuanbao/send.js +57 -0
  338. package/clis/yuanbao/shared.js +297 -5
  339. package/clis/yuanbao/shared.test.js +80 -0
  340. package/clis/yuanbao/status.js +44 -0
  341. package/clis/zlibrary/commands.test.js +1 -11
  342. package/dist/src/browser/base-page.d.ts +9 -0
  343. package/dist/src/browser/base-page.js +44 -1
  344. package/dist/src/browser/base-page.test.js +66 -0
  345. package/dist/src/browser/cdp.d.ts +1 -0
  346. package/dist/src/browser/cdp.js +51 -9
  347. package/dist/src/browser/daemon-client.d.ts +4 -0
  348. package/dist/src/browser/errors.js +1 -1
  349. package/dist/src/browser/page.d.ts +1 -1
  350. package/dist/src/browser/page.js +3 -1
  351. package/dist/src/browser/page.test.js +29 -0
  352. package/dist/src/browser/target-errors.d.ts +2 -1
  353. package/dist/src/browser/target-errors.js +1 -0
  354. package/dist/src/browser/target-resolver.d.ts +25 -0
  355. package/dist/src/browser/target-resolver.js +43 -0
  356. package/dist/src/build-manifest.js +9 -4
  357. package/dist/src/build-manifest.test.js +2 -8
  358. package/dist/src/capabilityRouting.d.ts +16 -1
  359. package/dist/src/capabilityRouting.js +24 -1
  360. package/dist/src/capabilityRouting.test.js +19 -1
  361. package/dist/src/cli.js +76 -11
  362. package/dist/src/cli.test.js +150 -0
  363. package/dist/src/commanderAdapter.js +0 -5
  364. package/dist/src/commanderAdapter.test.js +0 -1
  365. package/dist/src/discovery.js +2 -5
  366. package/dist/src/errors.js +1 -1
  367. package/dist/src/execution.d.ts +1 -1
  368. package/dist/src/execution.js +111 -27
  369. package/dist/src/execution.test.js +326 -17
  370. package/dist/src/help.d.ts +23 -2
  371. package/dist/src/help.js +41 -19
  372. package/dist/src/help.test.d.ts +1 -0
  373. package/dist/src/help.test.js +54 -0
  374. package/dist/src/main.js +14 -1
  375. package/dist/src/manifest-types.d.ts +5 -3
  376. package/dist/src/pipeline/executor.js +1 -1
  377. package/dist/src/pipeline/executor.test.js +8 -0
  378. package/dist/src/pipeline/registry.d.ts +9 -0
  379. package/dist/src/pipeline/registry.js +13 -1
  380. package/dist/src/pipeline/steps/browser.d.ts +1 -0
  381. package/dist/src/pipeline/steps/browser.js +10 -0
  382. package/dist/src/pipeline/steps/download.test.js +1 -0
  383. package/dist/src/registry-api.d.ts +1 -1
  384. package/dist/src/registry.d.ts +12 -11
  385. package/dist/src/registry.js +16 -6
  386. package/dist/src/registry.test.js +2 -2
  387. package/dist/src/runtime.d.ts +2 -1
  388. package/dist/src/runtime.js +1 -1
  389. package/dist/src/serialization.d.ts +2 -2
  390. package/dist/src/serialization.js +4 -6
  391. package/dist/src/serialization.test.js +17 -0
  392. package/dist/src/types.d.ts +17 -0
  393. package/dist/src/validate.js +15 -11
  394. package/dist/src/validate.test.d.ts +9 -0
  395. package/dist/src/validate.test.js +90 -0
  396. package/package.json +1 -1
  397. package/scripts/fetch-adapters.js +1 -1
  398. package/scripts/typed-error-lint-baseline.json +5 -77
  399. package/clis/ctrip/search.test.js +0 -64
  400. package/clis/gov-policy/commands.test.js +0 -27
  401. package/clis/linux-do/category.js +0 -37
  402. package/clis/linux-do/hot.js +0 -26
  403. package/clis/linux-do/latest.js +0 -19
  404. package/clis/pixiv/test-utils.js +0 -23
  405. package/clis/toutiao/articles.test.js +0 -30
  406. package/dist/src/analysis.d.ts +0 -40
  407. package/dist/src/analysis.js +0 -172
@@ -0,0 +1,378 @@
1
+ import { afterEach, describe, expect, it, vi } from 'vitest';
2
+ import { getRegistry } from '@jackwener/opencli/registry';
3
+ import { ArgumentError, AuthRequiredError, CommandExecutionError, EmptyResultError } from '@jackwener/opencli/errors';
4
+
5
+ import {
6
+ HOT_BOARD_URL,
7
+ __test__,
8
+ mapHotRow,
9
+ looksToutiaoAuthWallText,
10
+ parseArticlesPage,
11
+ parseHotLimit,
12
+ parseToutiaoArticlesText,
13
+ } from './utils.js';
14
+
15
+ import './articles.js';
16
+ import './hot.js';
17
+
18
+ afterEach(() => {
19
+ vi.unstubAllGlobals();
20
+ vi.restoreAllMocks();
21
+ });
22
+
23
+ describe('toutiao parseArticlesPage', () => {
24
+ it('falls back to default for empty / undefined / null', () => {
25
+ expect(parseArticlesPage(undefined)).toBe(1);
26
+ expect(parseArticlesPage(null)).toBe(1);
27
+ expect(parseArticlesPage('')).toBe(1);
28
+ expect(parseArticlesPage(undefined, 3)).toBe(3);
29
+ });
30
+
31
+ it('parses valid integer pages in [1, 4]', () => {
32
+ expect(parseArticlesPage(1)).toBe(1);
33
+ expect(parseArticlesPage('2')).toBe(2);
34
+ expect(parseArticlesPage(4)).toBe(4);
35
+ });
36
+
37
+ it('rejects non-integer pages with ArgumentError', () => {
38
+ expect(() => parseArticlesPage('abc')).toThrow(ArgumentError);
39
+ expect(() => parseArticlesPage(1.5)).toThrow(ArgumentError);
40
+ });
41
+
42
+ it('rejects out-of-range pages with ArgumentError (no silent clamp)', () => {
43
+ expect(() => parseArticlesPage(0)).toThrow(ArgumentError);
44
+ expect(() => parseArticlesPage(5)).toThrow(ArgumentError);
45
+ expect(() => parseArticlesPage(-1)).toThrow(ArgumentError);
46
+ expect(() => parseArticlesPage(99)).toThrow(ArgumentError);
47
+ });
48
+ });
49
+
50
+ describe('toutiao parseHotLimit', () => {
51
+ it('falls back to default for empty / undefined / null', () => {
52
+ expect(parseHotLimit(undefined)).toBe(30);
53
+ expect(parseHotLimit(null)).toBe(30);
54
+ expect(parseHotLimit('')).toBe(30);
55
+ expect(parseHotLimit(undefined, 10)).toBe(10);
56
+ });
57
+
58
+ it('parses valid integer limits in [1, 50]', () => {
59
+ expect(parseHotLimit(1)).toBe(1);
60
+ expect(parseHotLimit('30')).toBe(30);
61
+ expect(parseHotLimit(50)).toBe(50);
62
+ });
63
+
64
+ it('rejects out-of-range limits with ArgumentError (no silent clamp)', () => {
65
+ expect(() => parseHotLimit(0)).toThrow(ArgumentError);
66
+ expect(() => parseHotLimit(51)).toThrow(ArgumentError);
67
+ expect(() => parseHotLimit(-1)).toThrow(ArgumentError);
68
+ expect(() => parseHotLimit(1.5)).toThrow(ArgumentError);
69
+ });
70
+
71
+ it('rejects non-numeric limits with ArgumentError', () => {
72
+ expect(() => parseHotLimit('abc')).toThrow(ArgumentError);
73
+ });
74
+ });
75
+
76
+ describe('toutiao parseToutiaoArticlesText (silent column drop fix)', () => {
77
+ const rowWithStats = [
78
+ '短标题',
79
+ '04-20 20:30',
80
+ '已发布',
81
+ '展现 8 阅读 0 点赞 0 评论 0',
82
+ ].join('\n');
83
+
84
+ it('keeps short chinese titles and emits full row when stats present', () => {
85
+ expect(parseToutiaoArticlesText(rowWithStats)).toEqual([
86
+ {
87
+ title: '短标题',
88
+ date: '04-20 20:30',
89
+ status: '已发布',
90
+ '展现': '8',
91
+ '阅读': '0',
92
+ '点赞': '0',
93
+ '评论': '0',
94
+ },
95
+ ]);
96
+ });
97
+
98
+ it('still emits partial row with null stat columns when stats line failed to render', () => {
99
+ // Regression: previously `if (title && stats) push` silently dropped
100
+ // rows that had a title but where the stats span had not finished
101
+ // rendering, masking creator-backend slow-render bugs.
102
+ const partial = [
103
+ '部分行 (stats 慢渲染)',
104
+ '04-20 20:30',
105
+ '已发布',
106
+ // stats line absent on purpose
107
+ ].join('\n');
108
+ const out = parseToutiaoArticlesText(partial);
109
+ expect(out).toEqual([
110
+ {
111
+ title: '部分行 (stats 慢渲染)',
112
+ date: '04-20 20:30',
113
+ status: '已发布',
114
+ '展现': null,
115
+ '阅读': null,
116
+ '点赞': null,
117
+ '评论': null,
118
+ },
119
+ ]);
120
+ });
121
+
122
+ it('returns empty array for non-text input', () => {
123
+ expect(parseToutiaoArticlesText('')).toEqual([]);
124
+ expect(parseToutiaoArticlesText(null)).toEqual([]);
125
+ expect(parseToutiaoArticlesText(undefined)).toEqual([]);
126
+ });
127
+
128
+ it('handles multi-row input without losing rows', () => {
129
+ const text = [
130
+ '第一篇文章',
131
+ '04-20 20:30',
132
+ '已发布',
133
+ '展现 100 阅读 50 点赞 5 评论 1',
134
+ '',
135
+ '第二篇文章',
136
+ '04-21 09:15',
137
+ '已发布',
138
+ '展现 8 阅读 0 点赞 0 评论 0',
139
+ ].join('\n');
140
+ expect(parseToutiaoArticlesText(text)).toHaveLength(2);
141
+ });
142
+ });
143
+
144
+ describe('toutiao auth wall detection', () => {
145
+ it('recognizes creator-login/captcha pages', () => {
146
+ expect(looksToutiaoAuthWallText('账号登录 请登录 mp.toutiao.com/profile_v4/login')).toBe(true);
147
+ expect(looksToutiaoAuthWallText('安全验证 captcha')).toBe(true);
148
+ expect(looksToutiaoAuthWallText('短标题 04-20 20:30 已发布 展现 1 阅读 1')).toBe(false);
149
+ });
150
+ });
151
+
152
+ describe('toutiao mapHotRow', () => {
153
+ it('projects upstream item to stable 8-column shape', () => {
154
+ const upstream = {
155
+ ClusterId: 12345,
156
+ ClusterIdStr: '12345',
157
+ Title: '某热点新闻',
158
+ QueryWord: '某热点',
159
+ HotValue: '987654',
160
+ Label: '热',
161
+ Url: 'https://www.toutiao.com/trending/12345/',
162
+ Image: { url: 'https://p.image/x.jpg', url_list: ['https://p.image/x.jpg'] },
163
+ };
164
+ expect(mapHotRow(upstream, 0)).toEqual({
165
+ rank: 1,
166
+ group_id: '12345',
167
+ title: '某热点新闻',
168
+ query: '某热点',
169
+ hot_value: 987654,
170
+ label: '热',
171
+ url: 'https://www.toutiao.com/trending/12345/',
172
+ image_url: 'https://p.image/x.jpg',
173
+ });
174
+ });
175
+
176
+ it('falls back to ClusterId numeric when ClusterIdStr missing', () => {
177
+ const out = mapHotRow({ ClusterId: 7, Title: 'X' }, 5);
178
+ expect(out.group_id).toBe('7');
179
+ expect(out.rank).toBe(6);
180
+ });
181
+
182
+ it('falls back QueryWord → Title when query missing', () => {
183
+ const out = mapHotRow({ Title: '只有标题' }, 0);
184
+ expect(out.query).toBe('只有标题');
185
+ });
186
+
187
+ it('handles Image as url_list-only shape', () => {
188
+ const out = mapHotRow({
189
+ Title: 'Y',
190
+ Image: { url_list: ['https://list.image/y.png'] },
191
+ }, 0);
192
+ expect(out.image_url).toBe('https://list.image/y.png');
193
+ });
194
+
195
+ it('handles live Image.url_list entries shaped as objects', () => {
196
+ const out = mapHotRow({
197
+ Title: 'Y',
198
+ Image: { url_list: [{ url: 'https://list.image/y.png' }] },
199
+ }, 0);
200
+ expect(out.image_url).toBe('https://list.image/y.png');
201
+ });
202
+
203
+ it('returns null image when Image missing or all variants empty', () => {
204
+ expect(mapHotRow({ Title: 'Z' }, 0).image_url).toBeNull();
205
+ expect(mapHotRow({ Title: 'Z', Image: {} }, 0).image_url).toBeNull();
206
+ expect(mapHotRow({ Title: 'Z', Image: { url_list: [] } }, 0).image_url).toBeNull();
207
+ });
208
+
209
+ it('drops untitled rows (returns null) instead of emitting empty-title row', () => {
210
+ expect(mapHotRow({ Title: '', ClusterId: 1 }, 0)).toBeNull();
211
+ expect(mapHotRow({ Title: ' ', ClusterId: 1 }, 0)).toBeNull();
212
+ expect(mapHotRow(null, 0)).toBeNull();
213
+ expect(mapHotRow('not-an-object', 0)).toBeNull();
214
+ });
215
+
216
+ it('returns null hot_value for non-numeric HotValue (no silent 0 sentinel)', () => {
217
+ expect(mapHotRow({ Title: 'Q', HotValue: 'NaN' }, 0).hot_value).toBeNull();
218
+ expect(mapHotRow({ Title: 'Q', HotValue: '-1' }, 0).hot_value).toBeNull();
219
+ });
220
+ });
221
+
222
+ describe('toutiao registry shape', () => {
223
+ const articles = getRegistry().get('toutiao/articles');
224
+ const hot = getRegistry().get('toutiao/hot');
225
+
226
+ it('articles is registered as browser/cookie read adapter', () => {
227
+ expect(articles).toBeTruthy();
228
+ expect(articles.access).toBe('read');
229
+ expect(articles.browser).toBe(true);
230
+ expect(articles.columns).toEqual(['title', 'date', 'status', '展现', '阅读', '点赞', '评论']);
231
+ });
232
+
233
+ it('hot is registered as public non-browser read adapter', () => {
234
+ expect(hot).toBeTruthy();
235
+ expect(hot.access).toBe('read');
236
+ expect(hot.browser).toBe(false);
237
+ expect(hot.columns).toEqual(['rank', 'group_id', 'title', 'query', 'hot_value', 'label', 'url', 'image_url']);
238
+ });
239
+ });
240
+
241
+ describe('toutiao articles adapter (registry func)', () => {
242
+ const cmd = getRegistry().get('toutiao/articles');
243
+
244
+ it('rejects invalid page before browser navigation', async () => {
245
+ const page = { goto: vi.fn(), wait: vi.fn(), evaluate: vi.fn() };
246
+
247
+ await expect(cmd.func(page, { page: 0 })).rejects.toThrow(ArgumentError);
248
+ await expect(cmd.func(page, { page: 5 })).rejects.toThrow(ArgumentError);
249
+ expect(page.goto).not.toHaveBeenCalled();
250
+ });
251
+
252
+ it('throws AuthRequiredError when creator backend renders a login wall', async () => {
253
+ const page = {
254
+ goto: vi.fn().mockResolvedValue(undefined),
255
+ wait: vi.fn().mockResolvedValue(undefined),
256
+ evaluate: vi.fn().mockResolvedValue('账号登录 请登录'),
257
+ };
258
+
259
+ await expect(cmd.func(page, { page: 1 })).rejects.toBeInstanceOf(AuthRequiredError);
260
+ });
261
+
262
+ it('wraps render failures as CommandExecutionError', async () => {
263
+ const page = {
264
+ goto: vi.fn().mockRejectedValue(new Error('browser down')),
265
+ wait: vi.fn(),
266
+ evaluate: vi.fn(),
267
+ };
268
+
269
+ await expect(cmd.func(page, { page: 1 })).rejects.toBeInstanceOf(CommandExecutionError);
270
+ });
271
+
272
+ it('throws EmptyResultError when rendered article text has no rows', async () => {
273
+ const page = {
274
+ goto: vi.fn().mockResolvedValue(undefined),
275
+ wait: vi.fn().mockResolvedValue(undefined),
276
+ evaluate: vi.fn().mockResolvedValue('没有文章'),
277
+ };
278
+
279
+ await expect(cmd.func(page, { page: 1 })).rejects.toBeInstanceOf(EmptyResultError);
280
+ });
281
+ });
282
+
283
+ describe('toutiao hot adapter (registry func)', () => {
284
+ const cmd = getRegistry().get('toutiao/hot');
285
+
286
+ it('rejects invalid limit before fetching (no silent clamp)', async () => {
287
+ const fetchMock = vi.fn();
288
+ vi.stubGlobal('fetch', fetchMock);
289
+
290
+ await expect(cmd.func(null, { limit: 0 })).rejects.toThrow(ArgumentError);
291
+ await expect(cmd.func(null, { limit: 51 })).rejects.toThrow(ArgumentError);
292
+ await expect(cmd.func(null, { limit: 'abc' })).rejects.toThrow(ArgumentError);
293
+ expect(fetchMock).not.toHaveBeenCalled();
294
+ });
295
+
296
+ it('hits the public hot-board URL with default limit', async () => {
297
+ const fetchMock = vi.fn().mockImplementation(() => Promise.resolve(
298
+ new Response(JSON.stringify({
299
+ data: [
300
+ { ClusterIdStr: '1', Title: 'A', HotValue: '100' },
301
+ { ClusterIdStr: '2', Title: 'B', HotValue: '200' },
302
+ ],
303
+ }), { status: 200 }),
304
+ ));
305
+ vi.stubGlobal('fetch', fetchMock);
306
+
307
+ const rows = await cmd.func(null, {});
308
+ expect(fetchMock).toHaveBeenCalledTimes(1);
309
+ expect(fetchMock.mock.calls[0][0]).toBe(HOT_BOARD_URL);
310
+ expect(rows).toHaveLength(2);
311
+ expect(rows[0]).toMatchObject({ rank: 1, group_id: '1', title: 'A', hot_value: 100 });
312
+ expect(rows[1]).toMatchObject({ rank: 2, group_id: '2', title: 'B', hot_value: 200 });
313
+ });
314
+
315
+ it('throws CommandExecutionError on non-OK HTTP', async () => {
316
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(() => Promise.resolve(
317
+ new Response('', { status: 503 }),
318
+ )));
319
+
320
+ await expect(cmd.func(null, { limit: 5 })).rejects.toThrow(CommandExecutionError);
321
+ });
322
+
323
+ it('throws CommandExecutionError on malformed JSON', async () => {
324
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(() => Promise.resolve(
325
+ new Response('not-json{{{', { status: 200 }),
326
+ )));
327
+
328
+ await expect(cmd.func(null, { limit: 5 })).rejects.toThrow(CommandExecutionError);
329
+ });
330
+
331
+ it('throws CommandExecutionError on in-band error envelope', async () => {
332
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(() => Promise.resolve(
333
+ new Response(JSON.stringify({ status: 'error', message: 'rate limited' }), { status: 200 }),
334
+ )));
335
+
336
+ await expect(cmd.func(null, { limit: 5 })).rejects.toThrow(CommandExecutionError);
337
+ });
338
+
339
+ it('throws CommandExecutionError when fetch rejects (no silent catch)', async () => {
340
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(() => Promise.reject(new Error('network down'))));
341
+
342
+ await expect(cmd.func(null, { limit: 5 })).rejects.toThrow(CommandExecutionError);
343
+ });
344
+
345
+ it('throws EmptyResultError when upstream payload is empty', async () => {
346
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(() => Promise.resolve(
347
+ new Response(JSON.stringify({ data: [] }), { status: 200 }),
348
+ )));
349
+
350
+ await expect(cmd.func(null, { limit: 5 })).rejects.toThrow(EmptyResultError);
351
+ });
352
+
353
+ it('caps result rows at limit and dense-ranks (1..N) after filtering', async () => {
354
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(() => Promise.resolve(
355
+ new Response(JSON.stringify({
356
+ data: [
357
+ { ClusterIdStr: '1', Title: 'kept' },
358
+ { ClusterIdStr: '2', Title: '' }, // empty title → dropped
359
+ { ClusterIdStr: '3', Title: 'also kept' },
360
+ { ClusterIdStr: '4', Title: 'over-limit' },
361
+ ],
362
+ }), { status: 200 }),
363
+ )));
364
+
365
+ const rows = await cmd.func(null, { limit: 2 });
366
+ expect(rows.map(r => r.title)).toEqual(['kept', 'also kept']);
367
+ expect(rows.map(r => r.rank)).toEqual([1, 2]); // dense rank after empty-title drop
368
+ });
369
+ });
370
+
371
+ describe('toutiao __test__ contract', () => {
372
+ it('exports min/max bounds for documentation parity', () => {
373
+ expect(__test__.ARTICLES_MIN_PAGE).toBe(1);
374
+ expect(__test__.ARTICLES_MAX_PAGE).toBe(4);
375
+ expect(__test__.HOT_MIN_LIMIT).toBe(1);
376
+ expect(__test__.HOT_MAX_LIMIT).toBe(50);
377
+ });
378
+ });
@@ -0,0 +1,161 @@
1
+ /**
2
+ * Shared helpers for the toutiao adapter.
3
+ */
4
+ import { ArgumentError } from '@jackwener/opencli/errors';
5
+
6
+ const ARTICLES_MIN_PAGE = 1;
7
+ const ARTICLES_MAX_PAGE = 4;
8
+ const HOT_MIN_LIMIT = 1;
9
+ const HOT_MAX_LIMIT = 50;
10
+
11
+ export function parseArticlesPage(raw, fallback = 1) {
12
+ if (raw === undefined || raw === null || raw === '') return fallback;
13
+ const parsed = Number(raw);
14
+ if (!Number.isFinite(parsed) || !Number.isInteger(parsed)) {
15
+ throw new ArgumentError(`--page must be an integer between ${ARTICLES_MIN_PAGE} and ${ARTICLES_MAX_PAGE}, got ${JSON.stringify(raw)}`);
16
+ }
17
+ if (parsed < ARTICLES_MIN_PAGE || parsed > ARTICLES_MAX_PAGE) {
18
+ throw new ArgumentError(`--page must be between ${ARTICLES_MIN_PAGE} and ${ARTICLES_MAX_PAGE}, got ${parsed}`);
19
+ }
20
+ return parsed;
21
+ }
22
+
23
+ export function parseHotLimit(raw, fallback = 30) {
24
+ if (raw === undefined || raw === null || raw === '') return fallback;
25
+ const parsed = Number(raw);
26
+ if (!Number.isFinite(parsed) || !Number.isInteger(parsed)) {
27
+ throw new ArgumentError(`--limit must be an integer between ${HOT_MIN_LIMIT} and ${HOT_MAX_LIMIT}, got ${JSON.stringify(raw)}`);
28
+ }
29
+ if (parsed < HOT_MIN_LIMIT || parsed > HOT_MAX_LIMIT) {
30
+ throw new ArgumentError(`--limit must be between ${HOT_MIN_LIMIT} and ${HOT_MAX_LIMIT}, got ${parsed}`);
31
+ }
32
+ return parsed;
33
+ }
34
+
35
+ const NON_TITLE_LINES = new Set([
36
+ '展现', '阅读', '点赞', '评论',
37
+ '查看数据', '查看评论', '修改', '更多', '首发',
38
+ '已发布', '定时发布', '定时发布中', '由文章生成', '审核中',
39
+ ]);
40
+
41
+ const STATS_RE = /展现\s*([\d,]+)\s*阅读\s*([\d,]+)\s*点赞\s*([\d,]+)\s*评论\s*([\d,]*)/;
42
+
43
+ /**
44
+ * Extract creator-backend article rows from the rendered text dump.
45
+ *
46
+ * Surfaces every row anchored on a `MM-DD HH:MM` line; if the matching stats
47
+ * line never came through (slow render / missing element), the row is still
48
+ * emitted with `null` for stat columns rather than silently dropped.
49
+ */
50
+ export function parseToutiaoArticlesText(text) {
51
+ const lines = String(text || '').split('\n').map((line) => line.trim()).filter(Boolean);
52
+ const results = [];
53
+
54
+ for (let i = 0; i < lines.length; i++) {
55
+ const line = lines[i];
56
+ if (!/^\d{2}-\d{2}\s+\d{2}:\d{2}$/.test(line)) continue;
57
+
58
+ const date = line;
59
+ let title = null;
60
+ let status = null;
61
+ let stats = null;
62
+
63
+ for (let back = 3; back >= 1; back--) {
64
+ const prev = lines[i - back] || '';
65
+ if (!prev || prev.length >= 100 || /^\d+$/.test(prev) || NON_TITLE_LINES.has(prev)) continue;
66
+ title = prev;
67
+ break;
68
+ }
69
+
70
+ for (let fwd = 1; fwd < 8; fwd++) {
71
+ const fwdLine = lines[i + fwd] || '';
72
+ if (fwdLine === '已发布' || fwdLine === '定时发布中' || fwdLine === '审核中' || fwdLine === '由文章生成') {
73
+ status = fwdLine;
74
+ }
75
+ if (fwdLine.includes('展现') && fwdLine.includes('阅读')) {
76
+ const match = fwdLine.match(STATS_RE);
77
+ if (match) {
78
+ stats = {
79
+ '展现': match[1],
80
+ '阅读': match[2],
81
+ '点赞': match[3],
82
+ '评论': match[4] || '0',
83
+ };
84
+ }
85
+ }
86
+ }
87
+
88
+ if (!title) continue;
89
+
90
+ if (stats) {
91
+ results.push({ title, date, status, ...stats });
92
+ } else {
93
+ // Surface partial rows so callers can see they exist (was previously
94
+ // silently dropped — masking creator-backend slow-render bugs).
95
+ results.push({
96
+ title,
97
+ date,
98
+ status,
99
+ '展现': null,
100
+ '阅读': null,
101
+ '点赞': null,
102
+ '评论': null,
103
+ });
104
+ }
105
+ }
106
+
107
+ return results;
108
+ }
109
+
110
+ function trimOrNull(v) {
111
+ const s = String(v ?? '').trim();
112
+ return s ? s : null;
113
+ }
114
+
115
+ function pickImage(item) {
116
+ const url = item?.Image?.url;
117
+ if (typeof url === 'string' && url) return url;
118
+ const firstFromList = Array.isArray(item?.Image?.url_list)
119
+ ? item.Image.url_list
120
+ .map((entry) => typeof entry === 'string' ? entry : entry?.url)
121
+ .find((u) => typeof u === 'string' && u)
122
+ : null;
123
+ return firstFromList || null;
124
+ }
125
+
126
+ function parseHot(v) {
127
+ const n = Number(v);
128
+ return Number.isFinite(n) && n >= 0 ? n : null;
129
+ }
130
+
131
+ /**
132
+ * Project a row from the public toutiao hot-board API into stable shape.
133
+ */
134
+ export function mapHotRow(item, index) {
135
+ if (!item || typeof item !== 'object') return null;
136
+ const groupId = trimOrNull(item.ClusterIdStr || (item.ClusterId != null ? String(item.ClusterId) : null));
137
+ const title = trimOrNull(item.Title);
138
+ if (!title) return null;
139
+ return {
140
+ rank: index + 1,
141
+ group_id: groupId,
142
+ title,
143
+ query: trimOrNull(item.QueryWord) || title,
144
+ hot_value: parseHot(item.HotValue),
145
+ label: trimOrNull(item.Label),
146
+ url: trimOrNull(item.Url),
147
+ image_url: pickImage(item),
148
+ };
149
+ }
150
+
151
+ export const HOT_BOARD_URL = 'https://www.toutiao.com/hot-event/hot-board/?origin=toutiao_pc';
152
+
153
+ export function looksToutiaoAuthWallText(value) {
154
+ const text = String(value || '').replace(/\s+/g, ' ').trim().toLowerCase();
155
+ if (!text) return false;
156
+ return /登录|请登录|账号登录|扫码登录|安全验证|验证码|captcha/.test(text) ||
157
+ /\b(login|sign in|captcha|verification required)\b/.test(text) ||
158
+ /mp\.toutiao\.com\/profile_v4\/login/.test(text);
159
+ }
160
+
161
+ export const __test__ = { ARTICLES_MIN_PAGE, ARTICLES_MAX_PAGE, HOT_MIN_LIMIT, HOT_MAX_LIMIT };
@@ -0,0 +1,61 @@
1
+ // tvmaze search — TV show search by title.
2
+ //
3
+ // Hits `https://api.tvmaze.com/search/shows?q=<query>` and returns one row per
4
+ // match. Includes the show id (round-tripable into `tvmaze show <id>`),
5
+ // premiered/ended dates, network, and TVmaze rating so agents can rank.
6
+ import { cli, Strategy } from '@jackwener/opencli/registry';
7
+ import { EmptyResultError } from '@jackwener/opencli/errors';
8
+ import {
9
+ TVMAZE_BASE, joinList, requireBoundedInt, requireString, stripHtml, tvmazeFetch,
10
+ } from './utils.js';
11
+
12
+ cli({
13
+ site: 'tvmaze',
14
+ name: 'search',
15
+ access: 'read',
16
+ description: 'TVmaze TV show search by title (returns id, name, network, premiered/ended, rating)',
17
+ domain: 'tvmaze.com',
18
+ strategy: Strategy.PUBLIC,
19
+ browser: false,
20
+ args: [
21
+ { name: 'query', positional: true, type: 'string', required: true, help: 'TV show title or fragment to search for' },
22
+ { name: 'limit', type: 'int', default: 20, help: 'Max rows to return (1-50)' },
23
+ ],
24
+ columns: [
25
+ 'rank', 'id', 'name', 'type', 'language', 'genres',
26
+ 'status', 'premiered', 'ended', 'network', 'rating',
27
+ 'matchScore', 'summary', 'url',
28
+ ],
29
+ func: async (args) => {
30
+ const query = requireString(args.query, 'query');
31
+ const limit = requireBoundedInt(args.limit, 20, 50, 'limit');
32
+ const list = await tvmazeFetch(
33
+ `${TVMAZE_BASE}/search/shows?q=${encodeURIComponent(query)}`,
34
+ `tvmaze search ${query}`,
35
+ );
36
+ if (!Array.isArray(list) || list.length === 0) {
37
+ throw new EmptyResultError('tvmaze search', `TVmaze returned no shows matching "${query}".`);
38
+ }
39
+ const rows = list.slice(0, limit).map((entry, i) => {
40
+ const show = entry?.show ?? {};
41
+ const network = show.network?.name ?? show.webChannel?.name ?? '';
42
+ return {
43
+ rank: i + 1,
44
+ id: typeof show.id === 'number' ? show.id : null,
45
+ name: String(show.name ?? '').trim(),
46
+ type: String(show.type ?? '').trim(),
47
+ language: String(show.language ?? '').trim(),
48
+ genres: joinList(show.genres),
49
+ status: String(show.status ?? '').trim(),
50
+ premiered: typeof show.premiered === 'string' ? show.premiered : null,
51
+ ended: typeof show.ended === 'string' ? show.ended : null,
52
+ network: String(network).trim(),
53
+ rating: show.rating?.average == null ? null : Number(show.rating.average),
54
+ matchScore: typeof entry?.score === 'number' ? entry.score : null,
55
+ summary: stripHtml(show.summary),
56
+ url: String(show.url ?? '').trim(),
57
+ };
58
+ });
59
+ return rows;
60
+ },
61
+ });
@@ -0,0 +1,60 @@
1
+ // tvmaze show — full TV show details by TVmaze id.
2
+ //
3
+ // Hits `https://api.tvmaze.com/shows/<id>`. Returns one row with name, status,
4
+ // premiered/ended dates, network, runtime, rating, official site, IMDB / TheTVDB
5
+ // cross-refs (so agents can hop into other adapters), and a plain-text summary.
6
+ import { cli, Strategy } from '@jackwener/opencli/registry';
7
+ import { EmptyResultError } from '@jackwener/opencli/errors';
8
+ import { TVMAZE_BASE, joinList, requireShowId, stripHtml, tvmazeFetch } from './utils.js';
9
+
10
+ cli({
11
+ site: 'tvmaze',
12
+ name: 'show',
13
+ access: 'read',
14
+ description: 'Single TVmaze TV show detail by id (network, schedule, rating, IMDB/TheTVDB cross-refs)',
15
+ domain: 'tvmaze.com',
16
+ strategy: Strategy.PUBLIC,
17
+ browser: false,
18
+ args: [
19
+ { name: 'id', positional: true, type: 'int', required: true, help: 'TVmaze show id (positive integer)' },
20
+ ],
21
+ columns: [
22
+ 'id', 'name', 'type', 'language', 'genres', 'status',
23
+ 'premiered', 'ended', 'runtime', 'averageRuntime', 'network',
24
+ 'country', 'schedule', 'rating', 'imdb', 'thetvdb',
25
+ 'officialSite', 'summary', 'url',
26
+ ],
27
+ func: async (args) => {
28
+ const id = requireShowId(args.id);
29
+ const show = await tvmazeFetch(`${TVMAZE_BASE}/shows/${id}`, `tvmaze show ${id}`);
30
+ if (!show || show.id == null) {
31
+ throw new EmptyResultError('tvmaze show', `TVmaze returned no show for id ${id}.`);
32
+ }
33
+ const network = show.network?.name ?? show.webChannel?.name ?? '';
34
+ const country = show.network?.country?.name ?? show.webChannel?.country?.name ?? '';
35
+ const days = Array.isArray(show.schedule?.days) ? show.schedule.days.join(', ') : '';
36
+ const time = String(show.schedule?.time ?? '').trim();
37
+ const schedule = days || time ? `${days}${days && time ? ' ' : ''}${time}`.trim() : '';
38
+ return [{
39
+ id: Number(show.id),
40
+ name: String(show.name ?? '').trim(),
41
+ type: String(show.type ?? '').trim(),
42
+ language: String(show.language ?? '').trim(),
43
+ genres: joinList(show.genres),
44
+ status: String(show.status ?? '').trim(),
45
+ premiered: typeof show.premiered === 'string' ? show.premiered : null,
46
+ ended: typeof show.ended === 'string' ? show.ended : null,
47
+ runtime: show.runtime == null ? null : Number(show.runtime),
48
+ averageRuntime: show.averageRuntime == null ? null : Number(show.averageRuntime),
49
+ network: String(network).trim(),
50
+ country: String(country).trim(),
51
+ schedule,
52
+ rating: show.rating?.average == null ? null : Number(show.rating.average),
53
+ imdb: String(show.externals?.imdb ?? '').trim(),
54
+ thetvdb: show.externals?.thetvdb == null ? null : Number(show.externals.thetvdb),
55
+ officialSite: String(show.officialSite ?? '').trim(),
56
+ summary: stripHtml(show.summary),
57
+ url: String(show.url ?? '').trim(),
58
+ }];
59
+ },
60
+ });