@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,765 @@
1
+ /**
2
+ * @owner src/adapters/stackoverflow/questions.ts
3
+ * @does Register agent-facing Stack Overflow read, related, tag, and user commands.
4
+ * @needs Public Stack Exchange API, strict numeric ids, bounded pagination, HTML-to-text conversion.
5
+ * @feeds surface coverage ledger, Stack Overflow research workflows, question/answer readers.
6
+ * @breaks Stack Exchange envelope drift, partial comment pagination, or weak entity decoding corrupts reads.
7
+ */
8
+
9
+ import { cli, Strategy } from "../../registry.js";
10
+
11
+ const STACK_API = "https://api.stackexchange.com/2.3";
12
+ const STACK_SITE = "stackoverflow";
13
+ const STACK_MAX_PAGE_SIZE = 100;
14
+ const RELATED_SORTS = ["rank", "activity", "votes", "creation"] as const;
15
+ const TAG_SORTS = [
16
+ "activity",
17
+ "votes",
18
+ "creation",
19
+ "hot",
20
+ "week",
21
+ "month",
22
+ ] as const;
23
+
24
+ interface StackOwner {
25
+ display_name?: unknown;
26
+ user_id?: unknown;
27
+ }
28
+
29
+ interface StackQuestion {
30
+ question_id?: unknown;
31
+ accepted_answer_id?: unknown;
32
+ title?: unknown;
33
+ body?: unknown;
34
+ score?: unknown;
35
+ answer_count?: unknown;
36
+ view_count?: unknown;
37
+ is_answered?: unknown;
38
+ tags?: unknown;
39
+ owner?: StackOwner;
40
+ creation_date?: unknown;
41
+ last_activity_date?: unknown;
42
+ link?: unknown;
43
+ }
44
+
45
+ interface StackAnswer {
46
+ answer_id?: unknown;
47
+ is_accepted?: unknown;
48
+ body?: unknown;
49
+ score?: unknown;
50
+ owner?: StackOwner;
51
+ }
52
+
53
+ interface StackComment {
54
+ post_id?: unknown;
55
+ body?: unknown;
56
+ score?: unknown;
57
+ owner?: StackOwner;
58
+ }
59
+
60
+ interface StackUser {
61
+ user_id?: unknown;
62
+ display_name?: unknown;
63
+ reputation?: unknown;
64
+ badge_counts?: {
65
+ gold?: unknown;
66
+ silver?: unknown;
67
+ bronze?: unknown;
68
+ };
69
+ location?: unknown;
70
+ creation_date?: unknown;
71
+ last_access_date?: unknown;
72
+ link?: unknown;
73
+ }
74
+
75
+ interface StackEnvelope<T> {
76
+ items?: T[];
77
+ has_more?: unknown;
78
+ error_id?: unknown;
79
+ error_name?: unknown;
80
+ error_message?: unknown;
81
+ }
82
+
83
+ type FetchStackJson = <T>(
84
+ path: string,
85
+ params: Record<string, unknown>,
86
+ label: string,
87
+ ) => Promise<StackEnvelope<T>>;
88
+
89
+ function stringField(value: unknown): string {
90
+ return typeof value === "string" ? value : "";
91
+ }
92
+
93
+ function numberField(value: unknown): number {
94
+ return typeof value === "number" && Number.isFinite(value) ? value : 0;
95
+ }
96
+
97
+ function boolField(value: unknown): boolean {
98
+ return value === true;
99
+ }
100
+
101
+ export function requireStackQuestionId(value: unknown, label = "id"): string {
102
+ const id = String(value ?? "").trim();
103
+ if (!/^\d+$/.test(id)) {
104
+ throw new Error(`stackoverflow ${label} must be a numeric question id.`);
105
+ }
106
+ return id;
107
+ }
108
+
109
+ export function requireStackString(value: unknown, label: string): string {
110
+ const raw = String(value ?? "").trim();
111
+ if (!raw) throw new Error(`stackoverflow ${label} cannot be empty.`);
112
+ return raw;
113
+ }
114
+
115
+ export function requireStackLimit(
116
+ value: unknown,
117
+ fallback: number,
118
+ max: number,
119
+ label: string,
120
+ ): number {
121
+ const raw =
122
+ value === undefined || value === null || value === "" ? fallback : value;
123
+ const limit = Number(raw);
124
+ if (!Number.isInteger(limit) || limit < 1 || limit > max) {
125
+ throw new Error(
126
+ `stackoverflow ${label} must be an integer in [1, ${max}].`,
127
+ );
128
+ }
129
+ return limit;
130
+ }
131
+
132
+ export function requireStackMinInt(
133
+ value: unknown,
134
+ fallback: number,
135
+ min: number,
136
+ label: string,
137
+ ): number {
138
+ const raw =
139
+ value === undefined || value === null || value === "" ? fallback : value;
140
+ const n = Number(raw);
141
+ if (!Number.isInteger(n) || n < min) {
142
+ throw new Error(`stackoverflow ${label} must be an integer >= ${min}.`);
143
+ }
144
+ return n;
145
+ }
146
+
147
+ function requireSort<T extends readonly string[]>(
148
+ value: unknown,
149
+ fallback: T[number],
150
+ allowed: T,
151
+ label: string,
152
+ ): T[number] {
153
+ const sort = String(value ?? fallback).toLowerCase();
154
+ if (!allowed.includes(sort)) {
155
+ throw new Error(
156
+ `stackoverflow ${label} must be one of ${allowed.join(", ")}.`,
157
+ );
158
+ }
159
+ return sort;
160
+ }
161
+
162
+ export function stackEpochToDate(value: unknown): string {
163
+ const n = Number(value);
164
+ return Number.isFinite(n) && n > 0
165
+ ? new Date(n * 1000).toISOString().slice(0, 10)
166
+ : "";
167
+ }
168
+
169
+ const NAMED_ENTITIES: Record<string, string> = {
170
+ amp: "&",
171
+ lt: "<",
172
+ gt: ">",
173
+ quot: '"',
174
+ apos: "'",
175
+ "#39": "'",
176
+ nbsp: " ",
177
+ hellip: "...",
178
+ mdash: "-",
179
+ ndash: "-",
180
+ laquo: "<<",
181
+ raquo: ">>",
182
+ copy: "(c)",
183
+ reg: "(R)",
184
+ trade: "(TM)",
185
+ euro: "EUR",
186
+ pound: "GBP",
187
+ yen: "JPY",
188
+ rsquo: "'",
189
+ lsquo: "'",
190
+ rdquo: '"',
191
+ ldquo: '"',
192
+ };
193
+
194
+ export function decodeStackHtmlEntities(value: unknown): string {
195
+ return String(value ?? "")
196
+ .replace(/&#x([0-9a-fA-F]+);/g, (_, hex: string) => {
197
+ const code = Number.parseInt(hex, 16);
198
+ return Number.isFinite(code) ? String.fromCodePoint(code) : "";
199
+ })
200
+ .replace(/&#(\d+);/g, (_, dec: string) => {
201
+ const code = Number.parseInt(dec, 10);
202
+ return Number.isFinite(code) ? String.fromCodePoint(code) : "";
203
+ })
204
+ .replace(/&([a-zA-Z]+|#39);/g, (match, name: string) => {
205
+ return NAMED_ENTITIES[name] ?? match;
206
+ });
207
+ }
208
+
209
+ export function stackHtmlToText(value: unknown): string {
210
+ return decodeStackHtmlEntities(
211
+ String(value ?? "")
212
+ .replace(/<pre[^>]*><code[^>]*>([\s\S]*?)<\/code><\/pre>/gi, "\n$1\n")
213
+ .replace(/<code[^>]*>(.*?)<\/code>/gi, "`$1`")
214
+ .replace(/<p[^>]*>/gi, "\n\n")
215
+ .replace(/<\/p>/gi, "")
216
+ .replace(/<br\s*\/?>/gi, "\n")
217
+ .replace(/<li[^>]*>/gi, "\n- ")
218
+ .replace(/<\/li>/gi, "")
219
+ .replace(/<a[^>]*href="([^"]*)"[^>]*>(.*?)<\/a>/gi, "$2 ($1)")
220
+ .replace(/<[^>]+>/g, ""),
221
+ )
222
+ .replace(/\n{3,}/g, "\n\n")
223
+ .trim();
224
+ }
225
+
226
+ function stackAuthor(owner: StackOwner | undefined): string {
227
+ return decodeStackHtmlEntities(owner?.display_name) || "[deleted]";
228
+ }
229
+
230
+ function truncate(text: string, maxLength: number): string {
231
+ return text.length > maxLength
232
+ ? `${text.slice(0, maxLength)} ... [truncated]`
233
+ : text;
234
+ }
235
+
236
+ function indentLines(text: string, depth: number): string {
237
+ if (depth <= 0) return text;
238
+ const prefix = `${" ".repeat(depth)}> `;
239
+ return text
240
+ .split("\n")
241
+ .map((line) => `${prefix}${line}`)
242
+ .join("\n");
243
+ }
244
+
245
+ function stackItems<T>(envelope: StackEnvelope<T>, label: string): T[] {
246
+ const items = Array.isArray(envelope.items) ? envelope.items : [];
247
+ if (items.length === 0) {
248
+ throw new Error(`${label} returned no items.`);
249
+ }
250
+ return items;
251
+ }
252
+
253
+ async function fetchStackJson<T>(
254
+ path: string,
255
+ params: Record<string, unknown>,
256
+ label: string,
257
+ ): Promise<StackEnvelope<T>> {
258
+ const url = new URL(`${STACK_API}${path}`);
259
+ url.searchParams.set("site", STACK_SITE);
260
+ for (const [key, value] of Object.entries(params)) {
261
+ if (value !== undefined && value !== null && value !== "") {
262
+ url.searchParams.set(key, String(value));
263
+ }
264
+ }
265
+ const response = await fetch(url, {
266
+ headers: {
267
+ Accept: "application/json",
268
+ "User-Agent":
269
+ "unicli-stackoverflow/1.0 (https://github.com/olo-dot-io/Uni-CLI)",
270
+ },
271
+ });
272
+ if (!response.ok) {
273
+ throw new Error(
274
+ `Stack Exchange API returned HTTP ${response.status} for ${label}.`,
275
+ );
276
+ }
277
+ const envelope = (await response.json()) as StackEnvelope<T>;
278
+ if (envelope.error_id) {
279
+ throw new Error(
280
+ `Stack Exchange API error ${String(envelope.error_id)} for ${label}: ${String(
281
+ envelope.error_message ?? envelope.error_name ?? "",
282
+ )}`,
283
+ );
284
+ }
285
+ return envelope;
286
+ }
287
+
288
+ export function mapStackQuestionRows(
289
+ questions: StackQuestion[],
290
+ ): Array<Record<string, unknown>> {
291
+ return questions.map((question, index) => {
292
+ const id = numberField(question.question_id);
293
+ return {
294
+ rank: index + 1,
295
+ id,
296
+ title: decodeStackHtmlEntities(question.title),
297
+ score: numberField(question.score),
298
+ answers: numberField(question.answer_count),
299
+ views: numberField(question.view_count),
300
+ isAnswered: boolField(question.is_answered),
301
+ tags: Array.isArray(question.tags)
302
+ ? question.tags.map(String).join(", ")
303
+ : "",
304
+ author: decodeStackHtmlEntities(question.owner?.display_name),
305
+ createdAt: stackEpochToDate(question.creation_date),
306
+ lastActivityAt: stackEpochToDate(question.last_activity_date),
307
+ url:
308
+ stringField(question.link) ||
309
+ (id ? `https://stackoverflow.com/questions/${id}` : ""),
310
+ };
311
+ });
312
+ }
313
+
314
+ export function mapStackUserRows(
315
+ users: StackUser[],
316
+ ): Array<Record<string, unknown>> {
317
+ return users.map((user) => {
318
+ const id = numberField(user.user_id);
319
+ return {
320
+ userId: id,
321
+ displayName: decodeStackHtmlEntities(user.display_name),
322
+ reputation: numberField(user.reputation),
323
+ goldBadges: numberField(user.badge_counts?.gold),
324
+ silverBadges: numberField(user.badge_counts?.silver),
325
+ bronzeBadges: numberField(user.badge_counts?.bronze),
326
+ location: decodeStackHtmlEntities(user.location),
327
+ createdAt: stackEpochToDate(user.creation_date),
328
+ lastAccessAt: stackEpochToDate(user.last_access_date),
329
+ url:
330
+ stringField(user.link) ||
331
+ (id ? `https://stackoverflow.com/users/${id}` : ""),
332
+ };
333
+ });
334
+ }
335
+
336
+ function acceptedAnswerId(question: StackQuestion): number {
337
+ return numberField(question.accepted_answer_id);
338
+ }
339
+
340
+ function answerId(answer: StackAnswer): number {
341
+ return numberField(answer.answer_id);
342
+ }
343
+
344
+ export function sortStackAnswers(
345
+ question: StackQuestion,
346
+ answers: StackAnswer[],
347
+ ): StackAnswer[] {
348
+ const accepted = acceptedAnswerId(question);
349
+ return answers.slice().sort((a, b) => {
350
+ const aAccepted = boolField(a.is_accepted) || answerId(a) === accepted;
351
+ const bAccepted = boolField(b.is_accepted) || answerId(b) === accepted;
352
+ if (aAccepted !== bAccepted) return aAccepted ? -1 : 1;
353
+ return numberField(b.score) - numberField(a.score);
354
+ });
355
+ }
356
+
357
+ async function fetchAcceptedAnswerIfMissing(
358
+ question: StackQuestion,
359
+ answers: StackAnswer[],
360
+ fetchJson: FetchStackJson,
361
+ label: string,
362
+ ): Promise<StackAnswer[]> {
363
+ const accepted = acceptedAnswerId(question);
364
+ if (!accepted || answers.some((answer) => answerId(answer) === accepted)) {
365
+ return answers;
366
+ }
367
+ const envelope = await fetchJson<StackAnswer>(
368
+ `/answers/${accepted}`,
369
+ { filter: "withbody" },
370
+ `${label}/accepted-answer`,
371
+ );
372
+ const [answer] = envelope.items ?? [];
373
+ return answer ? answers.concat(answer) : answers;
374
+ }
375
+
376
+ async function fetchAnswerCommentsByAnswerId(
377
+ answers: StackAnswer[],
378
+ commentsLimit: number,
379
+ fetchJson: FetchStackJson,
380
+ label: string,
381
+ ): Promise<Map<number, StackComment[]>> {
382
+ const commentsByAnswer = new Map<number, StackComment[]>();
383
+ if (answers.length === 0) return commentsByAnswer;
384
+ const ids = answers.map(answerId).filter(Boolean).join(";");
385
+ if (!ids) return commentsByAnswer;
386
+ const pageSize = Math.min(
387
+ STACK_MAX_PAGE_SIZE,
388
+ answers.length * commentsLimit,
389
+ );
390
+ const envelope = await fetchJson<StackComment>(
391
+ `/answers/${ids}/comments`,
392
+ {
393
+ filter: "withbody",
394
+ order: "asc",
395
+ sort: "creation",
396
+ pagesize: pageSize,
397
+ },
398
+ `${label}/answer-comments`,
399
+ );
400
+ for (const comment of envelope.items ?? []) {
401
+ const postId = numberField(comment.post_id);
402
+ if (!postId) continue;
403
+ const comments = commentsByAnswer.get(postId) ?? [];
404
+ comments.push(comment);
405
+ commentsByAnswer.set(postId, comments);
406
+ }
407
+ if (envelope.has_more) {
408
+ const selectedAnswerIsPartial = answers.some((answer) => {
409
+ return (
410
+ (commentsByAnswer.get(answerId(answer)) ?? []).length < commentsLimit
411
+ );
412
+ });
413
+ if (selectedAnswerIsPartial) {
414
+ throw new Error(
415
+ `Stack Exchange answer comments for ${label} exceed one API page.`,
416
+ );
417
+ }
418
+ }
419
+ return commentsByAnswer;
420
+ }
421
+
422
+ export async function buildStackReadRows(
423
+ question: StackQuestion | undefined,
424
+ questionComments: StackComment[],
425
+ answers: StackAnswer[],
426
+ fetchJson: FetchStackJson,
427
+ options: {
428
+ answersLimit: number;
429
+ commentsLimit: number;
430
+ maxLength: number;
431
+ label: string;
432
+ },
433
+ ): Promise<Array<Record<string, unknown>>> {
434
+ if (!question) throw new Error(`${options.label} question not found.`);
435
+ const withAccepted = await fetchAcceptedAnswerIfMissing(
436
+ question,
437
+ answers,
438
+ fetchJson,
439
+ options.label,
440
+ );
441
+ const orderedAnswers = sortStackAnswers(question, withAccepted).slice(
442
+ 0,
443
+ options.answersLimit,
444
+ );
445
+ const answerComments = await fetchAnswerCommentsByAnswerId(
446
+ orderedAnswers,
447
+ options.commentsLimit,
448
+ fetchJson,
449
+ options.label,
450
+ );
451
+ const rows: Array<Record<string, unknown>> = [];
452
+ const questionBody = stackHtmlToText(question.body);
453
+ const questionText = [
454
+ decodeStackHtmlEntities(question.title),
455
+ questionBody,
456
+ stringField(question.link),
457
+ ]
458
+ .filter(Boolean)
459
+ .join("\n\n");
460
+ rows.push({
461
+ type: "POST",
462
+ author: stackAuthor(question.owner),
463
+ score: numberField(question.score),
464
+ accepted: "",
465
+ text: truncate(questionText, options.maxLength),
466
+ });
467
+ for (const comment of questionComments.slice(0, options.commentsLimit)) {
468
+ rows.push({
469
+ type: "Q-COMMENT",
470
+ author: stackAuthor(comment.owner),
471
+ score: numberField(comment.score),
472
+ accepted: "",
473
+ text: truncate(
474
+ indentLines(stackHtmlToText(comment.body), 1),
475
+ options.maxLength,
476
+ ),
477
+ });
478
+ }
479
+ for (const answer of orderedAnswers) {
480
+ rows.push({
481
+ type: "ANSWER",
482
+ author: stackAuthor(answer.owner),
483
+ score: numberField(answer.score),
484
+ accepted: boolField(answer.is_accepted) ? "true" : "",
485
+ text: truncate(stackHtmlToText(answer.body), options.maxLength),
486
+ });
487
+ for (const comment of (answerComments.get(answerId(answer)) ?? []).slice(
488
+ 0,
489
+ options.commentsLimit,
490
+ )) {
491
+ rows.push({
492
+ type: "A-COMMENT",
493
+ author: stackAuthor(comment.owner),
494
+ score: numberField(comment.score),
495
+ accepted: "",
496
+ text: truncate(
497
+ indentLines(stackHtmlToText(comment.body), 1),
498
+ options.maxLength,
499
+ ),
500
+ });
501
+ }
502
+ }
503
+ return rows;
504
+ }
505
+
506
+ cli({
507
+ site: "stackoverflow",
508
+ name: "read",
509
+ description: "Read a Stack Overflow question with answers and comments",
510
+ domain: "stackoverflow.com",
511
+ strategy: Strategy.PUBLIC,
512
+ browser: false,
513
+ args: [
514
+ {
515
+ name: "id",
516
+ type: "str",
517
+ required: true,
518
+ positional: true,
519
+ description: "Stack Overflow question id",
520
+ },
521
+ {
522
+ name: "answers-limit",
523
+ type: "int",
524
+ default: 10,
525
+ description: "Max answers to include",
526
+ },
527
+ {
528
+ name: "comments-limit",
529
+ type: "int",
530
+ default: 5,
531
+ description: "Max comments per question or answer",
532
+ },
533
+ {
534
+ name: "max-length",
535
+ type: "int",
536
+ default: 4000,
537
+ description: "Max characters per body",
538
+ },
539
+ ],
540
+ columns: ["type", "author", "score", "accepted", "text"],
541
+ func: async (_page, kwargs) => {
542
+ const id = requireStackQuestionId(kwargs.id);
543
+ const answersLimit = requireStackLimit(
544
+ kwargs["answers-limit"] ?? kwargs.answersLimit,
545
+ 10,
546
+ STACK_MAX_PAGE_SIZE,
547
+ "read answers-limit",
548
+ );
549
+ const commentsLimit = requireStackLimit(
550
+ kwargs["comments-limit"] ?? kwargs.commentsLimit,
551
+ 5,
552
+ STACK_MAX_PAGE_SIZE,
553
+ "read comments-limit",
554
+ );
555
+ const maxLength = requireStackMinInt(
556
+ kwargs["max-length"] ?? kwargs.maxLength,
557
+ 4000,
558
+ 100,
559
+ "read max-length",
560
+ );
561
+ const label = `stackoverflow/${id}`;
562
+ const [questionEnvelope, commentsEnvelope, answersEnvelope] =
563
+ await Promise.all([
564
+ fetchStackJson<StackQuestion>(
565
+ `/questions/${id}`,
566
+ { filter: "withbody" },
567
+ label,
568
+ ),
569
+ fetchStackJson<StackComment>(
570
+ `/questions/${id}/comments`,
571
+ {
572
+ filter: "withbody",
573
+ order: "asc",
574
+ sort: "creation",
575
+ pagesize: commentsLimit,
576
+ },
577
+ `${label}/comments`,
578
+ ),
579
+ fetchStackJson<StackAnswer>(
580
+ `/questions/${id}/answers`,
581
+ {
582
+ filter: "withbody",
583
+ order: "desc",
584
+ sort: "votes",
585
+ pagesize: answersLimit,
586
+ },
587
+ `${label}/answers`,
588
+ ),
589
+ ]);
590
+ return buildStackReadRows(
591
+ (questionEnvelope.items ?? [])[0],
592
+ commentsEnvelope.items ?? [],
593
+ answersEnvelope.items ?? [],
594
+ fetchStackJson,
595
+ { answersLimit, commentsLimit, maxLength, label },
596
+ );
597
+ },
598
+ });
599
+
600
+ cli({
601
+ site: "stackoverflow",
602
+ name: "related",
603
+ description: "List Stack Overflow questions related to a question id",
604
+ domain: "stackoverflow.com",
605
+ strategy: Strategy.PUBLIC,
606
+ browser: false,
607
+ args: [
608
+ {
609
+ name: "id",
610
+ type: "str",
611
+ required: true,
612
+ positional: true,
613
+ description: "Stack Overflow question id",
614
+ },
615
+ {
616
+ name: "sort",
617
+ type: "str",
618
+ default: "rank",
619
+ description: "Sort key: rank, activity, votes, creation",
620
+ },
621
+ { name: "limit", type: "int", default: 20, description: "Max questions" },
622
+ ],
623
+ columns: [
624
+ "rank",
625
+ "id",
626
+ "title",
627
+ "score",
628
+ "answers",
629
+ "views",
630
+ "isAnswered",
631
+ "tags",
632
+ "author",
633
+ "createdAt",
634
+ "lastActivityAt",
635
+ "url",
636
+ ],
637
+ func: async (_page, kwargs) => {
638
+ const id = requireStackQuestionId(kwargs.id);
639
+ const sort = requireSort(
640
+ kwargs.sort,
641
+ "rank",
642
+ RELATED_SORTS,
643
+ "related sort",
644
+ );
645
+ const limit = requireStackLimit(
646
+ kwargs.limit,
647
+ 20,
648
+ STACK_MAX_PAGE_SIZE,
649
+ "related limit",
650
+ );
651
+ const envelope = await fetchStackJson<StackQuestion>(
652
+ `/questions/${id}/related`,
653
+ { order: "desc", sort, pagesize: limit },
654
+ `stackoverflow related ${id}`,
655
+ );
656
+ return mapStackQuestionRows(
657
+ stackItems(envelope, `stackoverflow related ${id}`).slice(0, limit),
658
+ );
659
+ },
660
+ });
661
+
662
+ cli({
663
+ site: "stackoverflow",
664
+ name: "tag",
665
+ description: "List Stack Overflow questions tagged with a given tag",
666
+ domain: "stackoverflow.com",
667
+ strategy: Strategy.PUBLIC,
668
+ browser: false,
669
+ args: [
670
+ {
671
+ name: "tag",
672
+ type: "str",
673
+ required: true,
674
+ positional: true,
675
+ description: "Tag slug",
676
+ },
677
+ {
678
+ name: "sort",
679
+ type: "str",
680
+ default: "activity",
681
+ description: "Sort key: activity, votes, creation, hot, week, month",
682
+ },
683
+ { name: "limit", type: "int", default: 20, description: "Max questions" },
684
+ ],
685
+ columns: [
686
+ "rank",
687
+ "id",
688
+ "title",
689
+ "score",
690
+ "answers",
691
+ "views",
692
+ "isAnswered",
693
+ "tags",
694
+ "author",
695
+ "createdAt",
696
+ "lastActivityAt",
697
+ "url",
698
+ ],
699
+ func: async (_page, kwargs) => {
700
+ const tag = requireStackString(kwargs.tag, "tag").toLowerCase();
701
+ const sort = requireSort(kwargs.sort, "activity", TAG_SORTS, "tag sort");
702
+ const limit = requireStackLimit(
703
+ kwargs.limit,
704
+ 20,
705
+ STACK_MAX_PAGE_SIZE,
706
+ "tag limit",
707
+ );
708
+ const envelope = await fetchStackJson<StackQuestion>(
709
+ "/questions",
710
+ { tagged: tag, order: "desc", sort, pagesize: limit },
711
+ `stackoverflow tag ${tag}`,
712
+ );
713
+ return mapStackQuestionRows(
714
+ stackItems(envelope, `stackoverflow tag ${tag}`).slice(0, limit),
715
+ );
716
+ },
717
+ });
718
+
719
+ cli({
720
+ site: "stackoverflow",
721
+ name: "user",
722
+ description: "Find Stack Overflow users by display name",
723
+ domain: "stackoverflow.com",
724
+ strategy: Strategy.PUBLIC,
725
+ browser: false,
726
+ args: [
727
+ {
728
+ name: "name",
729
+ type: "str",
730
+ required: true,
731
+ positional: true,
732
+ description: "Display name or substring to search",
733
+ },
734
+ { name: "limit", type: "int", default: 10, description: "Max users" },
735
+ ],
736
+ columns: [
737
+ "userId",
738
+ "displayName",
739
+ "reputation",
740
+ "goldBadges",
741
+ "silverBadges",
742
+ "bronzeBadges",
743
+ "location",
744
+ "createdAt",
745
+ "lastAccessAt",
746
+ "url",
747
+ ],
748
+ func: async (_page, kwargs) => {
749
+ const name = requireStackString(kwargs.name, "name");
750
+ const limit = requireStackLimit(
751
+ kwargs.limit,
752
+ 10,
753
+ STACK_MAX_PAGE_SIZE,
754
+ "user limit",
755
+ );
756
+ const envelope = await fetchStackJson<StackUser>(
757
+ "/users",
758
+ { inname: name, order: "desc", sort: "reputation", pagesize: limit },
759
+ "stackoverflow user",
760
+ );
761
+ return mapStackUserRows(
762
+ stackItems(envelope, "stackoverflow user").slice(0, limit),
763
+ );
764
+ },
765
+ });