@zenalexa/unicli 0.219.0 → 0.220.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 (430) hide show
  1. package/AGENTS.md +9 -9
  2. package/README.md +55 -249
  3. package/README.zh-CN.md +55 -249
  4. package/dist/adapters/1point3acres/forum.d.ts +58 -0
  5. package/dist/adapters/1point3acres/forum.d.ts.map +1 -0
  6. package/dist/adapters/1point3acres/forum.js +708 -0
  7. package/dist/adapters/1point3acres/forum.js.map +1 -0
  8. package/dist/adapters/aibase/news.d.ts +21 -0
  9. package/dist/adapters/aibase/news.d.ts.map +1 -0
  10. package/dist/adapters/aibase/news.js +96 -0
  11. package/dist/adapters/aibase/news.js.map +1 -0
  12. package/dist/adapters/arxiv/papers.d.ts +27 -0
  13. package/dist/adapters/arxiv/papers.d.ts.map +1 -0
  14. package/dist/adapters/arxiv/papers.js +193 -0
  15. package/dist/adapters/arxiv/papers.js.map +1 -0
  16. package/dist/adapters/bbc/topic.d.ts +24 -0
  17. package/dist/adapters/bbc/topic.d.ts.map +1 -0
  18. package/dist/adapters/bbc/topic.js +122 -0
  19. package/dist/adapters/bbc/topic.js.map +1 -0
  20. package/dist/adapters/chatgpt/web.d.ts +24 -0
  21. package/dist/adapters/chatgpt/web.d.ts.map +1 -0
  22. package/dist/adapters/chatgpt/web.js +242 -0
  23. package/dist/adapters/chatgpt/web.js.map +1 -0
  24. package/dist/adapters/claude/web.d.ts +24 -0
  25. package/dist/adapters/claude/web.d.ts.map +1 -0
  26. package/dist/adapters/claude/web.js +575 -0
  27. package/dist/adapters/claude/web.js.map +1 -0
  28. package/dist/adapters/codex/projects.d.ts +27 -0
  29. package/dist/adapters/codex/projects.d.ts.map +1 -0
  30. package/dist/adapters/codex/projects.js +147 -0
  31. package/dist/adapters/codex/projects.js.map +1 -0
  32. package/dist/adapters/coingecko/markets.d.ts +19 -0
  33. package/dist/adapters/coingecko/markets.d.ts.map +1 -0
  34. package/dist/adapters/coingecko/markets.js +474 -0
  35. package/dist/adapters/coingecko/markets.js.map +1 -0
  36. package/dist/adapters/coupang/product.d.ts +27 -0
  37. package/dist/adapters/coupang/product.d.ts.map +1 -0
  38. package/dist/adapters/coupang/product.js +211 -0
  39. package/dist/adapters/coupang/product.js.map +1 -0
  40. package/dist/adapters/crates/registry.d.ts +44 -0
  41. package/dist/adapters/crates/registry.d.ts.map +1 -0
  42. package/dist/adapters/crates/registry.js +186 -0
  43. package/dist/adapters/crates/registry.js.map +1 -0
  44. package/dist/adapters/ctrip/travel.d.ts +83 -0
  45. package/dist/adapters/ctrip/travel.d.ts.map +1 -0
  46. package/dist/adapters/ctrip/travel.js +630 -0
  47. package/dist/adapters/ctrip/travel.js.map +1 -0
  48. package/dist/adapters/dblp/publications.d.ts +41 -0
  49. package/dist/adapters/dblp/publications.d.ts.map +1 -0
  50. package/dist/adapters/dblp/publications.js +409 -0
  51. package/dist/adapters/dblp/publications.js.map +1 -0
  52. package/dist/adapters/deepseek/web.d.ts +1 -1
  53. package/dist/adapters/deepseek/web.d.ts.map +1 -1
  54. package/dist/adapters/deepseek/web.js +66 -1
  55. package/dist/adapters/deepseek/web.js.map +1 -1
  56. package/dist/adapters/defillama/protocols.d.ts +13 -0
  57. package/dist/adapters/defillama/protocols.d.ts.map +1 -0
  58. package/dist/adapters/defillama/protocols.js +218 -0
  59. package/dist/adapters/defillama/protocols.js.map +1 -0
  60. package/dist/adapters/devto/read.d.ts +26 -0
  61. package/dist/adapters/devto/read.d.ts.map +1 -0
  62. package/dist/adapters/devto/read.js +110 -0
  63. package/dist/adapters/devto/read.js.map +1 -0
  64. package/dist/adapters/dianping/shop.d.ts +38 -0
  65. package/dist/adapters/dianping/shop.d.ts.map +1 -0
  66. package/dist/adapters/dianping/shop.js +194 -0
  67. package/dist/adapters/dianping/shop.js.map +1 -0
  68. package/dist/adapters/dockerhub/registry.d.ts +36 -0
  69. package/dist/adapters/dockerhub/registry.d.ts.map +1 -0
  70. package/dist/adapters/dockerhub/registry.js +172 -0
  71. package/dist/adapters/dockerhub/registry.js.map +1 -0
  72. package/dist/adapters/endoflife/product.d.ts +11 -0
  73. package/dist/adapters/endoflife/product.d.ts.map +1 -0
  74. package/dist/adapters/endoflife/product.js +113 -0
  75. package/dist/adapters/endoflife/product.js.map +1 -0
  76. package/dist/adapters/facebook/marketplace-extra.d.ts +9 -0
  77. package/dist/adapters/facebook/marketplace-extra.d.ts.map +1 -0
  78. package/dist/adapters/facebook/marketplace-extra.js +170 -0
  79. package/dist/adapters/facebook/marketplace-extra.js.map +1 -0
  80. package/dist/adapters/flathub/apps.d.ts +17 -0
  81. package/dist/adapters/flathub/apps.d.ts.map +1 -0
  82. package/dist/adapters/flathub/apps.js +220 -0
  83. package/dist/adapters/flathub/apps.js.map +1 -0
  84. package/dist/adapters/goproxy/module.d.ts +24 -0
  85. package/dist/adapters/goproxy/module.d.ts.map +1 -0
  86. package/dist/adapters/goproxy/module.js +221 -0
  87. package/dist/adapters/goproxy/module.js.map +1 -0
  88. package/dist/adapters/grok/web.d.ts +29 -0
  89. package/dist/adapters/grok/web.d.ts.map +1 -0
  90. package/dist/adapters/grok/web.js +553 -0
  91. package/dist/adapters/grok/web.js.map +1 -0
  92. package/dist/adapters/hackernews/read.d.ts +31 -0
  93. package/dist/adapters/hackernews/read.d.ts.map +1 -0
  94. package/dist/adapters/hackernews/read.js +201 -0
  95. package/dist/adapters/hackernews/read.js.map +1 -0
  96. package/dist/adapters/hf/paper.d.ts +22 -0
  97. package/dist/adapters/hf/paper.d.ts.map +1 -0
  98. package/dist/adapters/hf/paper.js +112 -0
  99. package/dist/adapters/hf/paper.js.map +1 -0
  100. package/dist/adapters/homebrew/packages.d.ts +52 -0
  101. package/dist/adapters/homebrew/packages.d.ts.map +1 -0
  102. package/dist/adapters/homebrew/packages.js +240 -0
  103. package/dist/adapters/homebrew/packages.js.map +1 -0
  104. package/dist/adapters/indeed/jobs.d.ts +38 -0
  105. package/dist/adapters/indeed/jobs.d.ts.map +1 -0
  106. package/dist/adapters/indeed/jobs.js +300 -0
  107. package/dist/adapters/indeed/jobs.js.map +1 -0
  108. package/dist/adapters/instagram/collections.d.ts +9 -0
  109. package/dist/adapters/instagram/collections.d.ts.map +1 -0
  110. package/dist/adapters/instagram/collections.js +174 -0
  111. package/dist/adapters/instagram/collections.js.map +1 -0
  112. package/dist/adapters/lichess/players.d.ts +46 -0
  113. package/dist/adapters/lichess/players.d.ts.map +1 -0
  114. package/dist/adapters/lichess/players.js +221 -0
  115. package/dist/adapters/lichess/players.js.map +1 -0
  116. package/dist/adapters/lobsters/read-domain.d.ts +35 -0
  117. package/dist/adapters/lobsters/read-domain.d.ts.map +1 -0
  118. package/dist/adapters/lobsters/read-domain.js +306 -0
  119. package/dist/adapters/lobsters/read-domain.js.map +1 -0
  120. package/dist/adapters/maven/artifact.d.ts +30 -0
  121. package/dist/adapters/maven/artifact.d.ts.map +1 -0
  122. package/dist/adapters/maven/artifact.js +121 -0
  123. package/dist/adapters/maven/artifact.js.map +1 -0
  124. package/dist/adapters/mdn/search.d.ts +11 -0
  125. package/dist/adapters/mdn/search.d.ts.map +1 -0
  126. package/dist/adapters/mdn/search.js +115 -0
  127. package/dist/adapters/mdn/search.js.map +1 -0
  128. package/dist/adapters/medium/tag.d.ts +15 -0
  129. package/dist/adapters/medium/tag.d.ts.map +1 -0
  130. package/dist/adapters/medium/tag.js +148 -0
  131. package/dist/adapters/medium/tag.js.map +1 -0
  132. package/dist/adapters/npm/package.d.ts +32 -0
  133. package/dist/adapters/npm/package.d.ts.map +1 -0
  134. package/dist/adapters/npm/package.js +141 -0
  135. package/dist/adapters/npm/package.js.map +1 -0
  136. package/dist/adapters/nuget/package.d.ts +34 -0
  137. package/dist/adapters/nuget/package.d.ts.map +1 -0
  138. package/dist/adapters/nuget/package.js +135 -0
  139. package/dist/adapters/nuget/package.js.map +1 -0
  140. package/dist/adapters/nvd/cve.d.ts +42 -0
  141. package/dist/adapters/nvd/cve.d.ts.map +1 -0
  142. package/dist/adapters/nvd/cve.js +132 -0
  143. package/dist/adapters/nvd/cve.js.map +1 -0
  144. package/dist/adapters/oeis/sequences.d.ts +14 -0
  145. package/dist/adapters/oeis/sequences.d.ts.map +1 -0
  146. package/dist/adapters/oeis/sequences.js +219 -0
  147. package/dist/adapters/oeis/sequences.js.map +1 -0
  148. package/dist/adapters/openalex/works.d.ts +43 -0
  149. package/dist/adapters/openalex/works.d.ts.map +1 -0
  150. package/dist/adapters/openalex/works.js +267 -0
  151. package/dist/adapters/openalex/works.js.map +1 -0
  152. package/dist/adapters/openfda/records.d.ts +18 -0
  153. package/dist/adapters/openfda/records.d.ts.map +1 -0
  154. package/dist/adapters/openfda/records.js +209 -0
  155. package/dist/adapters/openfda/records.js.map +1 -0
  156. package/dist/adapters/openreview/papers.d.ts +34 -0
  157. package/dist/adapters/openreview/papers.d.ts.map +1 -0
  158. package/dist/adapters/openreview/papers.js +463 -0
  159. package/dist/adapters/openreview/papers.js.map +1 -0
  160. package/dist/adapters/osv/security.d.ts +36 -0
  161. package/dist/adapters/osv/security.d.ts.map +1 -0
  162. package/dist/adapters/osv/security.js +247 -0
  163. package/dist/adapters/osv/security.js.map +1 -0
  164. package/dist/adapters/packagist/package.d.ts +31 -0
  165. package/dist/adapters/packagist/package.d.ts.map +1 -0
  166. package/dist/adapters/packagist/package.js +108 -0
  167. package/dist/adapters/packagist/package.js.map +1 -0
  168. package/dist/adapters/pubmed/articles.d.ts +31 -0
  169. package/dist/adapters/pubmed/articles.d.ts.map +1 -0
  170. package/dist/adapters/pubmed/articles.js +385 -0
  171. package/dist/adapters/pubmed/articles.js.map +1 -0
  172. package/dist/adapters/pypi/package.d.ts +38 -0
  173. package/dist/adapters/pypi/package.d.ts.map +1 -0
  174. package/dist/adapters/pypi/package.js +235 -0
  175. package/dist/adapters/pypi/package.js.map +1 -0
  176. package/dist/adapters/qwen/web.d.ts +26 -0
  177. package/dist/adapters/qwen/web.d.ts.map +1 -0
  178. package/dist/adapters/qwen/web.js +672 -0
  179. package/dist/adapters/qwen/web.js.map +1 -0
  180. package/dist/adapters/reddit/account.d.ts +12 -0
  181. package/dist/adapters/reddit/account.d.ts.map +1 -0
  182. package/dist/adapters/reddit/account.js +409 -0
  183. package/dist/adapters/reddit/account.js.map +1 -0
  184. package/dist/adapters/rednote/web.d.ts +30 -0
  185. package/dist/adapters/rednote/web.d.ts.map +1 -0
  186. package/dist/adapters/rednote/web.js +858 -0
  187. package/dist/adapters/rednote/web.js.map +1 -0
  188. package/dist/adapters/rest-countries/countries.d.ts +14 -0
  189. package/dist/adapters/rest-countries/countries.d.ts.map +1 -0
  190. package/dist/adapters/rest-countries/countries.js +231 -0
  191. package/dist/adapters/rest-countries/countries.js.map +1 -0
  192. package/dist/adapters/reuters/article-detail.d.ts +37 -0
  193. package/dist/adapters/reuters/article-detail.d.ts.map +1 -0
  194. package/dist/adapters/reuters/article-detail.js +139 -0
  195. package/dist/adapters/reuters/article-detail.js.map +1 -0
  196. package/dist/adapters/rfc/rfc.d.ts +11 -0
  197. package/dist/adapters/rfc/rfc.d.ts.map +1 -0
  198. package/dist/adapters/rfc/rfc.js +121 -0
  199. package/dist/adapters/rfc/rfc.js.map +1 -0
  200. package/dist/adapters/rubygems/gem.d.ts +26 -0
  201. package/dist/adapters/rubygems/gem.d.ts.map +1 -0
  202. package/dist/adapters/rubygems/gem.js +96 -0
  203. package/dist/adapters/rubygems/gem.js.map +1 -0
  204. package/dist/adapters/stackoverflow/questions.d.ts +79 -0
  205. package/dist/adapters/stackoverflow/questions.d.ts.map +1 -0
  206. package/dist/adapters/stackoverflow/questions.js +504 -0
  207. package/dist/adapters/stackoverflow/questions.js.map +1 -0
  208. package/dist/adapters/steam/app.d.ts +39 -0
  209. package/dist/adapters/steam/app.d.ts.map +1 -0
  210. package/dist/adapters/steam/app.js +165 -0
  211. package/dist/adapters/steam/app.js.map +1 -0
  212. package/dist/adapters/tiktok/creator-videos.d.ts +52 -0
  213. package/dist/adapters/tiktok/creator-videos.d.ts.map +1 -0
  214. package/dist/adapters/tiktok/creator-videos.js +267 -0
  215. package/dist/adapters/tiktok/creator-videos.js.map +1 -0
  216. package/dist/adapters/tvmaze/shows.d.ts +13 -0
  217. package/dist/adapters/tvmaze/shows.d.ts.map +1 -0
  218. package/dist/adapters/tvmaze/shows.js +240 -0
  219. package/dist/adapters/tvmaze/shows.js.map +1 -0
  220. package/dist/adapters/twitter/bookmark-folders.d.ts +33 -0
  221. package/dist/adapters/twitter/bookmark-folders.d.ts.map +1 -0
  222. package/dist/adapters/twitter/bookmark-folders.js +290 -0
  223. package/dist/adapters/twitter/bookmark-folders.js.map +1 -0
  224. package/dist/adapters/twitter/quote.d.ts +18 -0
  225. package/dist/adapters/twitter/quote.d.ts.map +1 -0
  226. package/dist/adapters/twitter/quote.js +285 -0
  227. package/dist/adapters/twitter/quote.js.map +1 -0
  228. package/dist/adapters/twitter/tweet-actions.d.ts +12 -0
  229. package/dist/adapters/twitter/tweet-actions.d.ts.map +1 -0
  230. package/dist/adapters/twitter/tweet-actions.js +145 -0
  231. package/dist/adapters/twitter/tweet-actions.js.map +1 -0
  232. package/dist/adapters/twitter/tweet-url.d.ts +15 -0
  233. package/dist/adapters/twitter/tweet-url.d.ts.map +1 -0
  234. package/dist/adapters/twitter/tweet-url.js +57 -0
  235. package/dist/adapters/twitter/tweet-url.js.map +1 -0
  236. package/dist/adapters/uisdc/news.d.ts +22 -0
  237. package/dist/adapters/uisdc/news.d.ts.map +1 -0
  238. package/dist/adapters/uisdc/news.js +91 -0
  239. package/dist/adapters/uisdc/news.js.map +1 -0
  240. package/dist/adapters/weibo/favorites-publish.d.ts +28 -0
  241. package/dist/adapters/weibo/favorites-publish.d.ts.map +1 -0
  242. package/dist/adapters/weibo/favorites-publish.js +356 -0
  243. package/dist/adapters/weibo/favorites-publish.js.map +1 -0
  244. package/dist/adapters/wikidata/entities.d.ts +15 -0
  245. package/dist/adapters/wikidata/entities.d.ts.map +1 -0
  246. package/dist/adapters/wikidata/entities.js +219 -0
  247. package/dist/adapters/wikidata/entities.js.map +1 -0
  248. package/dist/adapters/wikipedia/page.d.ts +21 -0
  249. package/dist/adapters/wikipedia/page.d.ts.map +1 -0
  250. package/dist/adapters/wikipedia/page.js +116 -0
  251. package/dist/adapters/wikipedia/page.js.map +1 -0
  252. package/dist/adapters/wttr/weather.d.ts +12 -0
  253. package/dist/adapters/wttr/weather.d.ts.map +1 -0
  254. package/dist/adapters/wttr/weather.js +207 -0
  255. package/dist/adapters/wttr/weather.js.map +1 -0
  256. package/dist/adapters/xianyu/publish.d.ts +31 -0
  257. package/dist/adapters/xianyu/publish.d.ts.map +1 -0
  258. package/dist/adapters/xianyu/publish.js +349 -0
  259. package/dist/adapters/xianyu/publish.js.map +1 -0
  260. package/dist/adapters/xiaohongshu/user-helpers.d.ts +2 -2
  261. package/dist/adapters/xiaohongshu/user-helpers.d.ts.map +1 -1
  262. package/dist/adapters/xiaohongshu/user-helpers.js +5 -4
  263. package/dist/adapters/xiaohongshu/user-helpers.js.map +1 -1
  264. package/dist/adapters/yuanbao/web.d.ts +27 -0
  265. package/dist/adapters/yuanbao/web.d.ts.map +1 -0
  266. package/dist/adapters/yuanbao/web.js +365 -0
  267. package/dist/adapters/yuanbao/web.js.map +1 -0
  268. package/dist/adapters/zhihu/collection.d.ts +33 -0
  269. package/dist/adapters/zhihu/collection.d.ts.map +1 -0
  270. package/dist/adapters/zhihu/collection.js +185 -0
  271. package/dist/adapters/zhihu/collection.js.map +1 -0
  272. package/dist/adapters/zlibrary/web.d.ts +19 -0
  273. package/dist/adapters/zlibrary/web.d.ts.map +1 -0
  274. package/dist/adapters/zlibrary/web.js +153 -0
  275. package/dist/adapters/zlibrary/web.js.map +1 -0
  276. package/dist/browser/daemon-client.js +2 -2
  277. package/dist/browser/daemon-client.js.map +1 -1
  278. package/dist/discovery/macos-dynamic.d.ts.map +1 -1
  279. package/dist/discovery/macos-dynamic.js +17 -3
  280. package/dist/discovery/macos-dynamic.js.map +1 -1
  281. package/dist/manifest-compact.txt +10 -10
  282. package/dist/manifest-search.json +1 -1
  283. package/dist/manifest.json +5130 -112
  284. package/dist/transport/refs.d.ts +5 -3
  285. package/dist/transport/refs.d.ts.map +1 -1
  286. package/dist/transport/refs.js +8 -1
  287. package/dist/transport/refs.js.map +1 -1
  288. package/dist/transport/sidecar-binary.d.ts +7 -0
  289. package/dist/transport/sidecar-binary.d.ts.map +1 -1
  290. package/dist/transport/sidecar-binary.js +28 -8
  291. package/dist/transport/sidecar-binary.js.map +1 -1
  292. package/package.json +4 -3
  293. package/server.json +3 -3
  294. package/skills/unicli/SKILL.md +1 -1
  295. package/skills/unicli-claude-code/SKILL.md +1 -1
  296. package/skills/unicli-hermes/SKILL.md +1 -1
  297. package/src/adapters/1point3acres/forum.test.ts +300 -0
  298. package/src/adapters/1point3acres/forum.ts +852 -0
  299. package/src/adapters/aibase/news.test.ts +42 -0
  300. package/src/adapters/aibase/news.ts +118 -0
  301. package/src/adapters/arxiv/papers.test.ts +59 -0
  302. package/src/adapters/arxiv/papers.ts +226 -0
  303. package/src/adapters/bbc/topic.test.ts +52 -0
  304. package/src/adapters/bbc/topic.ts +149 -0
  305. package/src/adapters/chatgpt/web.test.ts +121 -0
  306. package/src/adapters/chatgpt/web.ts +286 -0
  307. package/src/adapters/claude/web.test.ts +206 -0
  308. package/src/adapters/claude/web.ts +684 -0
  309. package/src/adapters/codex/projects.test.ts +77 -0
  310. package/src/adapters/codex/projects.ts +178 -0
  311. package/src/adapters/coingecko/markets.test.ts +156 -0
  312. package/src/adapters/coingecko/markets.ts +574 -0
  313. package/src/adapters/coupang/product.test.ts +111 -0
  314. package/src/adapters/coupang/product.ts +256 -0
  315. package/src/adapters/crates/registry.test.ts +89 -0
  316. package/src/adapters/crates/registry.ts +247 -0
  317. package/src/adapters/ctrip/travel.test.ts +359 -0
  318. package/src/adapters/ctrip/travel.ts +792 -0
  319. package/src/adapters/dblp/publications.test.ts +123 -0
  320. package/src/adapters/dblp/publications.ts +494 -0
  321. package/src/adapters/deepseek/web.test.ts +69 -0
  322. package/src/adapters/deepseek/web.ts +78 -1
  323. package/src/adapters/defillama/protocols.test.ts +75 -0
  324. package/src/adapters/defillama/protocols.ts +253 -0
  325. package/src/adapters/devto/read.test.ts +49 -0
  326. package/src/adapters/devto/read.ts +145 -0
  327. package/src/adapters/dianping/shop.test.ts +134 -0
  328. package/src/adapters/dianping/shop.ts +261 -0
  329. package/src/adapters/dockerhub/registry.test.ts +97 -0
  330. package/src/adapters/dockerhub/registry.ts +223 -0
  331. package/src/adapters/endoflife/product.test.ts +50 -0
  332. package/src/adapters/endoflife/product.ts +128 -0
  333. package/src/adapters/facebook/marketplace-extra.test.ts +132 -0
  334. package/src/adapters/facebook/marketplace-extra.ts +213 -0
  335. package/src/adapters/flathub/apps.test.ts +85 -0
  336. package/src/adapters/flathub/apps.ts +254 -0
  337. package/src/adapters/goproxy/module.test.ts +72 -0
  338. package/src/adapters/goproxy/module.ts +258 -0
  339. package/src/adapters/grok/web.test.ts +181 -0
  340. package/src/adapters/grok/web.ts +640 -0
  341. package/src/adapters/hackernews/read.test.ts +68 -0
  342. package/src/adapters/hackernews/read.ts +265 -0
  343. package/src/adapters/hf/paper.test.ts +48 -0
  344. package/src/adapters/hf/paper.ts +138 -0
  345. package/src/adapters/homebrew/packages.test.ts +109 -0
  346. package/src/adapters/homebrew/packages.ts +304 -0
  347. package/src/adapters/indeed/jobs.test.ts +230 -0
  348. package/src/adapters/indeed/jobs.ts +375 -0
  349. package/src/adapters/instagram/collections.test.ts +94 -0
  350. package/src/adapters/instagram/collections.ts +206 -0
  351. package/src/adapters/lichess/players.test.ts +99 -0
  352. package/src/adapters/lichess/players.ts +277 -0
  353. package/src/adapters/lobsters/read-domain.test.ts +121 -0
  354. package/src/adapters/lobsters/read-domain.ts +400 -0
  355. package/src/adapters/maven/artifact.test.ts +67 -0
  356. package/src/adapters/maven/artifact.ts +155 -0
  357. package/src/adapters/mdn/search.test.ts +39 -0
  358. package/src/adapters/mdn/search.ts +133 -0
  359. package/src/adapters/medium/tag.test.ts +64 -0
  360. package/src/adapters/medium/tag.ts +164 -0
  361. package/src/adapters/npm/package.test.ts +53 -0
  362. package/src/adapters/npm/package.ts +177 -0
  363. package/src/adapters/nuget/package.test.ts +102 -0
  364. package/src/adapters/nuget/package.ts +193 -0
  365. package/src/adapters/nvd/cve.test.ts +66 -0
  366. package/src/adapters/nvd/cve.ts +182 -0
  367. package/src/adapters/oeis/sequences.test.ts +71 -0
  368. package/src/adapters/oeis/sequences.ts +234 -0
  369. package/src/adapters/openalex/works.test.ts +99 -0
  370. package/src/adapters/openalex/works.ts +319 -0
  371. package/src/adapters/openfda/records.test.ts +90 -0
  372. package/src/adapters/openfda/records.ts +239 -0
  373. package/src/adapters/openreview/papers.test.ts +139 -0
  374. package/src/adapters/openreview/papers.ts +560 -0
  375. package/src/adapters/osv/security.test.ts +91 -0
  376. package/src/adapters/osv/security.ts +298 -0
  377. package/src/adapters/packagist/package.test.ts +62 -0
  378. package/src/adapters/packagist/package.ts +146 -0
  379. package/src/adapters/pubmed/articles.test.ts +96 -0
  380. package/src/adapters/pubmed/articles.ts +497 -0
  381. package/src/adapters/pypi/package.test.ts +131 -0
  382. package/src/adapters/pypi/package.ts +297 -0
  383. package/src/adapters/qwen/web.test.ts +176 -0
  384. package/src/adapters/qwen/web.ts +758 -0
  385. package/src/adapters/reddit/account.test.ts +56 -0
  386. package/src/adapters/reddit/account.ts +493 -0
  387. package/src/adapters/rednote/web.test.ts +354 -0
  388. package/src/adapters/rednote/web.ts +968 -0
  389. package/src/adapters/rest-countries/countries.test.ts +80 -0
  390. package/src/adapters/rest-countries/countries.ts +271 -0
  391. package/src/adapters/reuters/article-detail.test.ts +65 -0
  392. package/src/adapters/reuters/article-detail.ts +186 -0
  393. package/src/adapters/rfc/rfc.test.ts +37 -0
  394. package/src/adapters/rfc/rfc.ts +133 -0
  395. package/src/adapters/rubygems/gem.test.ts +43 -0
  396. package/src/adapters/rubygems/gem.ts +126 -0
  397. package/src/adapters/stackoverflow/questions.test.ts +207 -0
  398. package/src/adapters/stackoverflow/questions.ts +765 -0
  399. package/src/adapters/steam/app.test.ts +68 -0
  400. package/src/adapters/steam/app.ts +218 -0
  401. package/src/adapters/tiktok/creator-videos.test.ts +158 -0
  402. package/src/adapters/tiktok/creator-videos.ts +370 -0
  403. package/src/adapters/tvmaze/shows.test.ts +93 -0
  404. package/src/adapters/tvmaze/shows.ts +271 -0
  405. package/src/adapters/twitter/bookmark-folders.test.ts +164 -0
  406. package/src/adapters/twitter/bookmark-folders.ts +366 -0
  407. package/src/adapters/twitter/quote.test.ts +157 -0
  408. package/src/adapters/twitter/quote.ts +332 -0
  409. package/src/adapters/twitter/tweet-actions.test.ts +51 -0
  410. package/src/adapters/twitter/tweet-actions.ts +187 -0
  411. package/src/adapters/twitter/tweet-url.ts +65 -0
  412. package/src/adapters/uisdc/news.test.ts +46 -0
  413. package/src/adapters/uisdc/news.ts +111 -0
  414. package/src/adapters/weibo/favorites-publish.test.ts +177 -0
  415. package/src/adapters/weibo/favorites-publish.ts +426 -0
  416. package/src/adapters/wikidata/entities.test.ts +103 -0
  417. package/src/adapters/wikidata/entities.ts +253 -0
  418. package/src/adapters/wikipedia/page.test.ts +49 -0
  419. package/src/adapters/wikipedia/page.ts +158 -0
  420. package/src/adapters/wttr/weather.test.ts +99 -0
  421. package/src/adapters/wttr/weather.ts +239 -0
  422. package/src/adapters/xianyu/publish.test.ts +210 -0
  423. package/src/adapters/xianyu/publish.ts +420 -0
  424. package/src/adapters/xiaohongshu/user-helpers.ts +5 -2
  425. package/src/adapters/yuanbao/web.test.ts +144 -0
  426. package/src/adapters/yuanbao/web.ts +435 -0
  427. package/src/adapters/zhihu/collection.test.ts +96 -0
  428. package/src/adapters/zhihu/collection.ts +239 -0
  429. package/src/adapters/zlibrary/web.test.ts +104 -0
  430. package/src/adapters/zlibrary/web.ts +192 -0
@@ -0,0 +1,256 @@
1
+ /**
2
+ * @owner src/adapters/coupang/product.ts
3
+ * @does Register agent-facing Coupang product detail extraction implemented with site-specific safety checks.
4
+ * @needs Coupang product page DOM and optional logged-in browser session for restricted products.
5
+ * @feeds surface coverage ledger and Coupang search-to-detail workflows.
6
+ * @breaks Coupang product page redirects, anti-bot pages, or product DOM/bootstrap schema drift can hide detail fields.
7
+ */
8
+
9
+ import { cli, Strategy } from "../../registry.js";
10
+ import type { IPage } from "../../types.js";
11
+ import { js, str } from "../_shared/browser-tools.js";
12
+
13
+ const COUPANG_ORIGIN = "https://www.coupang.com";
14
+ const PRODUCT_ID_RE = /^\d{6,}$/;
15
+
16
+ interface CoupangProductData {
17
+ product_id?: unknown;
18
+ title?: unknown;
19
+ price?: unknown;
20
+ original_price?: unknown;
21
+ discount_rate?: unknown;
22
+ rating?: unknown;
23
+ review_count?: unknown;
24
+ seller?: unknown;
25
+ brand?: unknown;
26
+ rocket?: unknown;
27
+ delivery_promise?: unknown;
28
+ image_url?: unknown;
29
+ }
30
+
31
+ interface CoupangProductResult {
32
+ ok?: unknown;
33
+ reason?: unknown;
34
+ currentProductId?: unknown;
35
+ loginHints?: { hasLoginLink?: unknown; hasMyCoupang?: unknown };
36
+ data?: CoupangProductData;
37
+ }
38
+
39
+ export function normalizeCoupangProductId(value: unknown): string {
40
+ const raw = str(value).trim();
41
+ const match = raw.match(/(?:^|\/vp\/products\/)(\d{6,})(?:[/?#]|$)/);
42
+ return match?.[1] ?? (PRODUCT_ID_RE.test(raw) ? raw : "");
43
+ }
44
+
45
+ export function requireCoupangProductId(value: unknown, label: string): string {
46
+ const raw = str(value).trim();
47
+ let id = "";
48
+ if (label === "url") {
49
+ let url: URL;
50
+ try {
51
+ url = new URL(raw.startsWith("http") ? raw : `${COUPANG_ORIGIN}${raw}`);
52
+ } catch {
53
+ throw new Error(
54
+ "Coupang url must be a product URL containing /vp/products/<id>.",
55
+ );
56
+ }
57
+ const host = url.hostname.toLowerCase();
58
+ const match = url.pathname.match(/^\/vp\/products\/(\d{6,})(?:\/|$)/);
59
+ if ((host === "coupang.com" || host.endsWith(".coupang.com")) && match)
60
+ id = match[1];
61
+ } else {
62
+ id = normalizeCoupangProductId(raw);
63
+ }
64
+ if (!id)
65
+ throw new Error(
66
+ `Coupang ${label} must be a numeric product ID or product URL.`,
67
+ );
68
+ return id;
69
+ }
70
+
71
+ export function canonicalizeCoupangProductUrl(
72
+ value: unknown,
73
+ productId: string,
74
+ ): string {
75
+ const raw = str(value).trim();
76
+ if (raw) {
77
+ try {
78
+ const url = new URL(
79
+ raw.startsWith("http") ? raw : `${COUPANG_ORIGIN}${raw}`,
80
+ );
81
+ const host = url.hostname.toLowerCase();
82
+ if (host !== "coupang.com" && !host.endsWith(".coupang.com")) return "";
83
+ const id =
84
+ normalizeCoupangProductId(url.pathname) ||
85
+ normalizeCoupangProductId(productId);
86
+ return id ? `${COUPANG_ORIGIN}/vp/products/${id}` : url.toString();
87
+ } catch {
88
+ return "";
89
+ }
90
+ }
91
+ return productId ? `${COUPANG_ORIGIN}/vp/products/${productId}` : "";
92
+ }
93
+
94
+ function numberOrNull(value: unknown): number | null {
95
+ if (typeof value === "number" && Number.isFinite(value)) return value;
96
+ const text = str(value).replace(/[^\d.]/g, "");
97
+ if (!text) return null;
98
+ const n = Number(text);
99
+ return Number.isFinite(n) ? n : null;
100
+ }
101
+
102
+ export function mapCoupangProductRow(
103
+ productId: string,
104
+ data: CoupangProductData,
105
+ url: string,
106
+ ): Record<string, unknown> {
107
+ const actualId = normalizeCoupangProductId(data.product_id) || productId;
108
+ return {
109
+ product_id: actualId,
110
+ title: str(data.title).trim() || null,
111
+ price: numberOrNull(data.price),
112
+ original_price: numberOrNull(data.original_price),
113
+ discount_rate: numberOrNull(data.discount_rate),
114
+ rating: numberOrNull(data.rating),
115
+ review_count: numberOrNull(data.review_count),
116
+ seller: str(data.seller).trim() || null,
117
+ brand: str(data.brand).trim() || null,
118
+ rocket: str(data.rocket).trim() || null,
119
+ delivery_promise: str(data.delivery_promise).trim() || null,
120
+ image_url: str(data.image_url).trim() || null,
121
+ url: canonicalizeCoupangProductUrl("", actualId) || url,
122
+ };
123
+ }
124
+
125
+ function buildProductEvaluate(expectedProductId: string): string {
126
+ return `(() => {
127
+ const expectedProductId = ${js(expectedProductId)};
128
+ const normalizeText = (value) => value == null ? '' : String(value).trim();
129
+ const parseNum = (value) => {
130
+ const text = normalizeText(value).replace(/[^\\d.]/g, '');
131
+ if (!text) return null;
132
+ const num = Number(text);
133
+ return Number.isFinite(num) ? num : null;
134
+ };
135
+ const loginHints = {
136
+ hasLoginLink: Boolean(document.querySelector('a[href*="login"], a[title*="로그인"]')),
137
+ hasMyCoupang: /마이쿠팡/.test(document.body.innerText || ''),
138
+ };
139
+ const pathMatch = location.pathname.match(/\\/vp\\/products\\/(\\d+)/);
140
+ const currentProductId = pathMatch?.[1] || '';
141
+ if (expectedProductId && currentProductId && expectedProductId !== currentProductId) {
142
+ return { ok: false, reason: 'PRODUCT_MISMATCH', currentProductId, loginHints };
143
+ }
144
+ const jsonLd = (() => {
145
+ const scripts = Array.from(document.querySelectorAll('script[type="application/ld+json"]'));
146
+ for (const script of scripts) {
147
+ try {
148
+ const docs = JSON.parse(script.textContent || 'null');
149
+ const items = Array.isArray(docs) ? docs : [docs];
150
+ for (const doc of items) {
151
+ if (!doc || typeof doc !== 'object') continue;
152
+ const types = Array.isArray(doc['@type']) ? doc['@type'] : [doc['@type']];
153
+ if (!types.some((type) => /Product/i.test(String(type || '')))) continue;
154
+ const offer = Array.isArray(doc.offers) ? doc.offers[0] : doc.offers;
155
+ return {
156
+ title: normalizeText(doc.name),
157
+ brand: normalizeText(doc.brand?.name || doc.brand),
158
+ image_url: normalizeText(Array.isArray(doc.image) ? doc.image[0] : doc.image),
159
+ price: parseNum(offer?.price),
160
+ rating: parseNum(doc.aggregateRating?.ratingValue),
161
+ review_count: parseNum(doc.aggregateRating?.reviewCount),
162
+ seller: normalizeText(offer?.seller?.name),
163
+ };
164
+ }
165
+ } catch {}
166
+ }
167
+ return null;
168
+ })();
169
+ const dom = (() => {
170
+ const title = document.querySelector('.prod-buy-header__title, h1[class*="prod-buy-header"], h1[class*="ProductName"], h1[class*="product-name"]');
171
+ const price = document.querySelector('.total-price strong, .prod-sale-price strong, [class*="finalPrice"], [class*="sellingPrice"], [class*="price-value"]');
172
+ const original = document.querySelector('.origin-price, .base-price, del[class*="origin"], del[class*="base"], [class*="strike"], [class*="origin-price"]');
173
+ const discount = document.querySelector('.discount-percentage, [class*="discount"][class*="percent"], [class*="discountRate"]');
174
+ const rating = document.querySelector('.rating-star-num, [class*="ratingStar"], [class*="rating-star"], [class*="rating-num"], [class*="ProductRating"]');
175
+ const reviews = document.querySelector('.count, .rating-total-count, [class*="reviewCount"], [class*="review-count"]');
176
+ const seller = document.querySelector('.prod-sale-vendor-name, [class*="vendor-name"], [class*="vendorName"], [class*="sellerName"]');
177
+ const image = document.querySelector('.prod-image__detail, [class*="prod-image"] img, [class*="ProductImage"] img');
178
+ return {
179
+ title: normalizeText(title?.textContent),
180
+ price: parseNum(price?.textContent),
181
+ original_price: parseNum(original?.textContent),
182
+ discount_rate: parseNum(discount?.textContent),
183
+ rating: parseNum(rating?.getAttribute?.('aria-label') || rating?.textContent),
184
+ review_count: parseNum(reviews?.textContent),
185
+ seller: normalizeText(seller?.textContent),
186
+ image_url: normalizeText(image?.getAttribute?.('src') || image?.getAttribute?.('data-src')),
187
+ };
188
+ })();
189
+ const data = { ...(jsonLd || {}) };
190
+ for (const [key, value] of Object.entries(dom)) {
191
+ if (data[key] == null || data[key] === '') data[key] = value;
192
+ }
193
+ if (!data.title && data.price == null) {
194
+ return { ok: false, reason: 'NO_DATA_EXTRACTED', currentProductId, loginHints };
195
+ }
196
+ return { ok: true, currentProductId, loginHints, data };
197
+ })()`;
198
+ }
199
+
200
+ cli({
201
+ site: "coupang",
202
+ name: "product",
203
+ description: "Read full product detail for a Coupang product",
204
+ domain: "www.coupang.com",
205
+ strategy: Strategy.COOKIE,
206
+ browser: true,
207
+ args: [
208
+ { name: "product-id", type: "str", required: false, positional: true },
209
+ { name: "url", type: "str", required: false },
210
+ ],
211
+ columns: [
212
+ "product_id",
213
+ "title",
214
+ "price",
215
+ "original_price",
216
+ "discount_rate",
217
+ "rating",
218
+ "review_count",
219
+ "seller",
220
+ "brand",
221
+ "rocket",
222
+ "delivery_promise",
223
+ "image_url",
224
+ "url",
225
+ ],
226
+ func: async (page, kwargs) => {
227
+ const rawProductId = kwargs["product-id"];
228
+ if (!rawProductId && !kwargs.url)
229
+ throw new Error("Either product-id or url is required.");
230
+ const productId = rawProductId
231
+ ? requireCoupangProductId(rawProductId, "product-id")
232
+ : requireCoupangProductId(kwargs.url, "url");
233
+ const url =
234
+ canonicalizeCoupangProductUrl(kwargs.url, productId) ||
235
+ canonicalizeCoupangProductUrl("", productId);
236
+ const p = page as IPage;
237
+ await p.goto(url, { waitUntil: "load", settleMs: 3000 });
238
+ await p.wait(2);
239
+ const result = (await p.evaluate(
240
+ buildProductEvaluate(productId),
241
+ )) as CoupangProductResult;
242
+ const loginHints = result?.loginHints ?? {};
243
+ if (loginHints.hasLoginLink && !loginHints.hasMyCoupang) {
244
+ throw new Error("Coupang login is required.");
245
+ }
246
+ if (result?.reason === "PRODUCT_MISMATCH") {
247
+ throw new Error(
248
+ `Coupang product page redirected away from expected product ${productId}.`,
249
+ );
250
+ }
251
+ if (!result?.ok || !result.data) {
252
+ throw new Error(`No Coupang product data extracted from ${url}.`);
253
+ }
254
+ return [mapCoupangProductRow(productId, result.data, url)];
255
+ },
256
+ });
@@ -0,0 +1,89 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import {
3
+ mapCratesDetailRow,
4
+ mapCratesSearchRows,
5
+ requireRegistryLimit,
6
+ requireRegistryString,
7
+ } from "./registry.js";
8
+
9
+ describe("crates agent-facing registry commands", () => {
10
+ it("validates shared registry args", () => {
11
+ expect(requireRegistryString(" serde ", "query")).toBe("serde");
12
+ expect(() => requireRegistryString("", "query")).toThrow(
13
+ "query is required",
14
+ );
15
+ expect(requireRegistryLimit(undefined, 20)).toBe(20);
16
+ expect(requireRegistryLimit("3", 20)).toBe(3);
17
+ expect(() => requireRegistryLimit("0", 20)).toThrow(
18
+ "limit must be an integer",
19
+ );
20
+ expect(() => requireRegistryLimit("101", 20)).toThrow(
21
+ "limit must be an integer",
22
+ );
23
+ });
24
+
25
+ it("maps crates.io search rows to agent-facing columns", () => {
26
+ expect(
27
+ mapCratesSearchRows(
28
+ [
29
+ {
30
+ name: "serde",
31
+ newest_version: "1.0.0",
32
+ description: "Serialize",
33
+ downloads: 10,
34
+ recent_downloads: 2,
35
+ repository: "https://github.com/serde-rs/serde",
36
+ updated_at: "2026-05-01T00:00:00Z",
37
+ },
38
+ ],
39
+ 20,
40
+ ),
41
+ ).toEqual([
42
+ {
43
+ rank: 1,
44
+ name: "serde",
45
+ latestVersion: "1.0.0",
46
+ description: "Serialize",
47
+ downloads: 10,
48
+ recentDownloads: 2,
49
+ repository: "https://github.com/serde-rs/serde",
50
+ updated: "2026-05-01",
51
+ url: "https://crates.io/crates/serde",
52
+ },
53
+ ]);
54
+ });
55
+
56
+ it("maps crates.io detail rows without sentinel output", () => {
57
+ expect(
58
+ mapCratesDetailRow({
59
+ crate: {
60
+ name: "tokio",
61
+ newest_version: "1.2.3",
62
+ description: "Runtime",
63
+ downloads: 100,
64
+ recent_downloads: 9,
65
+ num_versions: 42,
66
+ homepage: "https://tokio.rs",
67
+ documentation: "https://docs.rs/tokio",
68
+ repository: "https://github.com/tokio-rs/tokio",
69
+ created_at: "2020-01-01T00:00:00Z",
70
+ updated_at: "2026-05-02T00:00:00Z",
71
+ },
72
+ versions: [{ num: "1.2.3", license: "MIT" }],
73
+ keywords: [{ keyword: "async" }],
74
+ categories: [{ category: "asynchronous" }],
75
+ }),
76
+ ).toMatchObject({
77
+ name: "tokio",
78
+ latestVersion: "1.2.3",
79
+ license: "MIT",
80
+ keywords: "async",
81
+ categories: "asynchronous",
82
+ created: "2020-01-01",
83
+ updated: "2026-05-02",
84
+ });
85
+ expect(() => mapCratesDetailRow({})).toThrow(
86
+ "crates.io returned no crate metadata",
87
+ );
88
+ });
89
+ });
@@ -0,0 +1,247 @@
1
+ /**
2
+ * @owner src/adapters/crates/registry.ts
3
+ * @does Register agent-facing crates.io search and crate metadata commands.
4
+ * @needs Public crates.io API, TypeScript adapter loader, bounded argument parsing.
5
+ * @feeds surface coverage ledger, package registry command surface, agent-readable registry rows.
6
+ * @breaks crates.io API envelope drift, weak query validation, or silent empty results hide registry lookup failures.
7
+ */
8
+
9
+ import { cli, Strategy } from "../../registry.js";
10
+
11
+ const CRATES_BASE = "https://crates.io";
12
+
13
+ interface CratesSearchItem {
14
+ id?: unknown;
15
+ name?: unknown;
16
+ newest_version?: unknown;
17
+ max_stable_version?: unknown;
18
+ max_version?: unknown;
19
+ description?: unknown;
20
+ downloads?: unknown;
21
+ recent_downloads?: unknown;
22
+ repository?: unknown;
23
+ homepage?: unknown;
24
+ updated_at?: unknown;
25
+ }
26
+
27
+ interface CratesDetailBody {
28
+ crate?: CratesSearchItem & {
29
+ num_versions?: unknown;
30
+ created_at?: unknown;
31
+ };
32
+ versions?: Array<{
33
+ num?: unknown;
34
+ license?: unknown;
35
+ }>;
36
+ keywords?: Array<{
37
+ keyword?: unknown;
38
+ id?: unknown;
39
+ }>;
40
+ categories?: Array<{
41
+ category?: unknown;
42
+ slug?: unknown;
43
+ }>;
44
+ }
45
+
46
+ function stringField(value: unknown): string {
47
+ return typeof value === "string" ? value : "";
48
+ }
49
+
50
+ function numberField(value: unknown): number | null {
51
+ return typeof value === "number" && Number.isFinite(value) ? value : null;
52
+ }
53
+
54
+ export function requireRegistryString(value: unknown, name: string): string {
55
+ const text = String(value ?? "").trim();
56
+ if (!text) throw new Error(`${name} is required.`);
57
+ return text;
58
+ }
59
+
60
+ export function requireRegistryLimit(value: unknown, fallback: number): number {
61
+ if (value === undefined || value === null || value === "") return fallback;
62
+ const n = Number(value);
63
+ if (!Number.isInteger(n) || n < 1 || n > 100) {
64
+ throw new Error(
65
+ `limit must be an integer in [1, 100]. Got: ${String(value)}`,
66
+ );
67
+ }
68
+ return n;
69
+ }
70
+
71
+ async function fetchCratesJson(url: string): Promise<unknown> {
72
+ const response = await fetch(url, {
73
+ headers: {
74
+ "User-Agent": "unicli (https://github.com/olo-dot-io/Uni-CLI)",
75
+ Accept: "application/json",
76
+ },
77
+ });
78
+ if (!response.ok) {
79
+ throw new Error(`crates.io request failed: HTTP ${response.status}`);
80
+ }
81
+ return response.json();
82
+ }
83
+
84
+ export function mapCratesSearchRows(
85
+ items: CratesSearchItem[],
86
+ limit: number,
87
+ ): Array<Record<string, unknown>> {
88
+ return items.slice(0, limit).map((crate, index) => {
89
+ const name = stringField(crate.name) || stringField(crate.id);
90
+ return {
91
+ rank: index + 1,
92
+ name,
93
+ latestVersion:
94
+ stringField(crate.newest_version) ||
95
+ stringField(crate.max_stable_version) ||
96
+ stringField(crate.max_version),
97
+ description: stringField(crate.description).trim(),
98
+ downloads: numberField(crate.downloads),
99
+ recentDownloads: numberField(crate.recent_downloads),
100
+ repository: stringField(crate.repository) || stringField(crate.homepage),
101
+ updated: stringField(crate.updated_at).slice(0, 10),
102
+ url: name ? `${CRATES_BASE}/crates/${name}` : "",
103
+ };
104
+ });
105
+ }
106
+
107
+ export function mapCratesDetailRow(
108
+ body: CratesDetailBody,
109
+ ): Record<string, unknown> {
110
+ const crate = body.crate;
111
+ if (!crate || !(stringField(crate.name) || stringField(crate.id))) {
112
+ throw new Error("crates.io returned no crate metadata.");
113
+ }
114
+ const versions = Array.isArray(body.versions) ? body.versions : [];
115
+ const latestVersion =
116
+ stringField(crate.newest_version) ||
117
+ stringField(crate.max_stable_version) ||
118
+ stringField(crate.max_version);
119
+ const latestRow =
120
+ versions.find((version) => stringField(version.num) === latestVersion) ??
121
+ versions[0] ??
122
+ {};
123
+ const name = stringField(crate.name) || stringField(crate.id);
124
+ const keywords = Array.isArray(body.keywords)
125
+ ? body.keywords
126
+ .map((item) => stringField(item.keyword) || stringField(item.id))
127
+ .filter(Boolean)
128
+ .join(", ")
129
+ : "";
130
+ const categories = Array.isArray(body.categories)
131
+ ? body.categories
132
+ .map((item) => stringField(item.category) || stringField(item.slug))
133
+ .filter(Boolean)
134
+ .join(", ")
135
+ : "";
136
+
137
+ return {
138
+ name,
139
+ latestVersion,
140
+ description: stringField(crate.description).trim(),
141
+ downloads: numberField(crate.downloads),
142
+ recentDownloads: numberField(crate.recent_downloads),
143
+ versions: numberField(crate.num_versions) ?? versions.length,
144
+ license: stringField(latestRow.license),
145
+ homepage: stringField(crate.homepage),
146
+ documentation: stringField(
147
+ (crate as { documentation?: unknown }).documentation,
148
+ ),
149
+ repository: stringField(crate.repository),
150
+ keywords,
151
+ categories,
152
+ created: stringField(crate.created_at).slice(0, 10),
153
+ updated: stringField(crate.updated_at).slice(0, 10),
154
+ url: `${CRATES_BASE}/crates/${name}`,
155
+ };
156
+ }
157
+
158
+ cli({
159
+ site: "crates",
160
+ name: "search",
161
+ description: "Search the public crates.io registry by keyword",
162
+ domain: "crates.io",
163
+ strategy: Strategy.PUBLIC,
164
+ args: [
165
+ {
166
+ name: "query",
167
+ type: "str",
168
+ required: true,
169
+ positional: true,
170
+ description: "Search keyword",
171
+ },
172
+ {
173
+ name: "limit",
174
+ type: "int",
175
+ default: 20,
176
+ description: "Max results",
177
+ },
178
+ ],
179
+ columns: [
180
+ "rank",
181
+ "name",
182
+ "latestVersion",
183
+ "description",
184
+ "downloads",
185
+ "recentDownloads",
186
+ "repository",
187
+ "updated",
188
+ "url",
189
+ ],
190
+ func: async (_page, kwargs) => {
191
+ const query = requireRegistryString(kwargs.query, "query");
192
+ const limit = requireRegistryLimit(kwargs.limit, 20);
193
+ const url = `${CRATES_BASE}/api/v1/crates?q=${encodeURIComponent(query)}&per_page=${limit}`;
194
+ const body = (await fetchCratesJson(url)) as {
195
+ crates?: CratesSearchItem[];
196
+ };
197
+ const rows = mapCratesSearchRows(
198
+ Array.isArray(body.crates) ? body.crates : [],
199
+ limit,
200
+ );
201
+ if (rows.length === 0) {
202
+ throw new Error(`No crates.io results matched "${query}".`);
203
+ }
204
+ return rows;
205
+ },
206
+ });
207
+
208
+ cli({
209
+ site: "crates",
210
+ name: "crate",
211
+ description: "Single crates.io crate metadata",
212
+ domain: "crates.io",
213
+ strategy: Strategy.PUBLIC,
214
+ args: [
215
+ {
216
+ name: "name",
217
+ type: "str",
218
+ required: true,
219
+ positional: true,
220
+ description: "crates.io crate name",
221
+ },
222
+ ],
223
+ columns: [
224
+ "name",
225
+ "latestVersion",
226
+ "description",
227
+ "downloads",
228
+ "recentDownloads",
229
+ "versions",
230
+ "license",
231
+ "homepage",
232
+ "documentation",
233
+ "repository",
234
+ "keywords",
235
+ "categories",
236
+ "created",
237
+ "updated",
238
+ "url",
239
+ ],
240
+ func: async (_page, kwargs) => {
241
+ const name = requireRegistryString(kwargs.name, "name");
242
+ const body = (await fetchCratesJson(
243
+ `${CRATES_BASE}/api/v1/crates/${encodeURIComponent(name)}`,
244
+ )) as CratesDetailBody;
245
+ return [mapCratesDetailRow(body)];
246
+ },
247
+ });