@jackwener/opencli 1.7.11 → 1.7.12

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 (730) hide show
  1. package/README.zh-CN.md +2 -1
  2. package/cli-manifest.json +1417 -24
  3. package/clis/1688/assets.js +1 -0
  4. package/clis/1688/download.js +1 -0
  5. package/clis/1688/item.js +1 -0
  6. package/clis/1688/search.js +2 -1
  7. package/clis/1688/store.js +1 -0
  8. package/clis/36kr/article.js +1 -0
  9. package/clis/36kr/hot.js +1 -0
  10. package/clis/36kr/news.js +1 -0
  11. package/clis/36kr/search.js +1 -0
  12. package/clis/51job/company.js +1 -0
  13. package/clis/51job/detail.js +1 -0
  14. package/clis/51job/hot.js +1 -0
  15. package/clis/51job/search.js +1 -0
  16. package/clis/amazon/bestsellers.js +1 -0
  17. package/clis/amazon/discussion.js +1 -0
  18. package/clis/amazon/movers-shakers.js +1 -0
  19. package/clis/amazon/new-releases.js +1 -0
  20. package/clis/amazon/offer.js +1 -0
  21. package/clis/amazon/product.js +1 -0
  22. package/clis/amazon/rankings.js +1 -0
  23. package/clis/amazon/search.js +1 -0
  24. package/clis/antigravity/dump.js +1 -0
  25. package/clis/antigravity/extract-code.js +1 -0
  26. package/clis/antigravity/model.js +1 -0
  27. package/clis/antigravity/new.js +1 -0
  28. package/clis/antigravity/read.js +1 -0
  29. package/clis/antigravity/send.js +1 -0
  30. package/clis/antigravity/status.js +1 -0
  31. package/clis/antigravity/watch.js +1 -0
  32. package/clis/apple-podcasts/episodes.js +1 -0
  33. package/clis/apple-podcasts/search.js +1 -0
  34. package/clis/apple-podcasts/top.js +1 -0
  35. package/clis/arxiv/arxiv.test.js +112 -0
  36. package/clis/arxiv/paper.js +4 -3
  37. package/clis/arxiv/recent.js +33 -0
  38. package/clis/arxiv/search.js +19 -7
  39. package/clis/arxiv/utils.js +68 -5
  40. package/clis/baidu-scholar/search.js +1 -0
  41. package/clis/band/bands.js +1 -0
  42. package/clis/band/mentions.js +1 -0
  43. package/clis/band/post.js +1 -0
  44. package/clis/band/posts.js +1 -0
  45. package/clis/barchart/flow.js +1 -0
  46. package/clis/barchart/greeks.js +1 -0
  47. package/clis/barchart/options.js +1 -0
  48. package/clis/barchart/quote.js +1 -0
  49. package/clis/bbc/news.js +1 -0
  50. package/clis/bilibili/comments.js +1 -0
  51. package/clis/bilibili/download.js +1 -0
  52. package/clis/bilibili/dynamic.js +1 -0
  53. package/clis/bilibili/favorite.js +1 -0
  54. package/clis/bilibili/feed.js +2 -0
  55. package/clis/bilibili/following.js +1 -0
  56. package/clis/bilibili/history.js +1 -0
  57. package/clis/bilibili/hot.js +6 -1
  58. package/clis/bilibili/hot.test.js +17 -0
  59. package/clis/bilibili/me.js +1 -1
  60. package/clis/bilibili/ranking.js +1 -0
  61. package/clis/bilibili/search.js +1 -1
  62. package/clis/bilibili/subtitle.js +1 -0
  63. package/clis/bilibili/user-videos.js +1 -0
  64. package/clis/bilibili/video.js +1 -0
  65. package/clis/binance/asks.js +1 -0
  66. package/clis/binance/depth.js +1 -0
  67. package/clis/binance/gainers.js +1 -0
  68. package/clis/binance/klines.js +1 -0
  69. package/clis/binance/losers.js +1 -0
  70. package/clis/binance/pairs.js +1 -0
  71. package/clis/binance/price.js +1 -0
  72. package/clis/binance/prices.js +1 -0
  73. package/clis/binance/ticker.js +1 -0
  74. package/clis/binance/top.js +1 -0
  75. package/clis/binance/trades.js +1 -0
  76. package/clis/bloomberg/businessweek.js +1 -0
  77. package/clis/bloomberg/economics.js +1 -0
  78. package/clis/bloomberg/feeds.js +1 -0
  79. package/clis/bloomberg/industries.js +1 -0
  80. package/clis/bloomberg/main.js +1 -0
  81. package/clis/bloomberg/markets.js +1 -0
  82. package/clis/bloomberg/news.js +1 -0
  83. package/clis/bloomberg/opinions.js +1 -0
  84. package/clis/bloomberg/politics.js +1 -0
  85. package/clis/bloomberg/tech.js +1 -0
  86. package/clis/bluesky/feeds.js +1 -0
  87. package/clis/bluesky/followers.js +1 -0
  88. package/clis/bluesky/following.js +1 -0
  89. package/clis/bluesky/profile.js +1 -0
  90. package/clis/bluesky/search.js +1 -0
  91. package/clis/bluesky/starter-packs.js +1 -0
  92. package/clis/bluesky/thread.js +1 -0
  93. package/clis/bluesky/trending.js +1 -0
  94. package/clis/bluesky/user.js +3 -1
  95. package/clis/boss/batchgreet.js +1 -0
  96. package/clis/boss/chatlist.js +1 -0
  97. package/clis/boss/chatmsg.js +1 -0
  98. package/clis/boss/detail.js +1 -0
  99. package/clis/boss/exchange.js +1 -0
  100. package/clis/boss/greet.js +1 -0
  101. package/clis/boss/invite.js +1 -0
  102. package/clis/boss/joblist.js +1 -0
  103. package/clis/boss/mark.js +1 -0
  104. package/clis/boss/recommend.js +1 -0
  105. package/clis/boss/resume.js +1 -0
  106. package/clis/boss/search.js +1 -0
  107. package/clis/boss/send.js +1 -0
  108. package/clis/boss/stats.js +1 -0
  109. package/clis/chaoxing/assignments.js +1 -0
  110. package/clis/chaoxing/exams.js +1 -0
  111. package/clis/chatgpt/image.js +1 -0
  112. package/clis/chatgpt-app/ask.js +1 -0
  113. package/clis/chatgpt-app/model.js +1 -0
  114. package/clis/chatgpt-app/new.js +1 -0
  115. package/clis/chatgpt-app/read.js +1 -0
  116. package/clis/chatgpt-app/send.js +1 -0
  117. package/clis/chatgpt-app/status.js +1 -0
  118. package/clis/chatwise/ask.js +1 -0
  119. package/clis/chatwise/export.js +1 -0
  120. package/clis/chatwise/history.js +1 -0
  121. package/clis/chatwise/model.js +1 -0
  122. package/clis/chatwise/read.js +1 -0
  123. package/clis/chatwise/send.js +1 -0
  124. package/clis/claude/ask.js +1 -0
  125. package/clis/claude/detail.js +1 -0
  126. package/clis/claude/history.js +1 -0
  127. package/clis/claude/new.js +1 -0
  128. package/clis/claude/read.js +1 -0
  129. package/clis/claude/send.js +1 -0
  130. package/clis/claude/status.js +1 -0
  131. package/clis/cnki/search.js +1 -0
  132. package/clis/codex/ask.js +1 -0
  133. package/clis/codex/export.js +1 -0
  134. package/clis/codex/extract-diff.js +1 -0
  135. package/clis/codex/history.js +1 -0
  136. package/clis/codex/model.js +1 -0
  137. package/clis/codex/read.js +1 -0
  138. package/clis/codex/send.js +1 -0
  139. package/clis/coupang/add-to-cart.js +1 -0
  140. package/clis/coupang/search.js +1 -0
  141. package/clis/ctrip/search.js +1 -0
  142. package/clis/cursor/ask.js +1 -0
  143. package/clis/cursor/composer.js +1 -0
  144. package/clis/cursor/export.js +1 -0
  145. package/clis/cursor/extract-code.js +1 -0
  146. package/clis/cursor/history.js +1 -0
  147. package/clis/cursor/model.js +1 -0
  148. package/clis/cursor/read.js +1 -0
  149. package/clis/cursor/send.js +1 -0
  150. package/clis/dblp/dblp.test.js +397 -0
  151. package/clis/dblp/paper.js +40 -0
  152. package/clis/dblp/search.js +45 -0
  153. package/clis/dblp/utils.js +290 -0
  154. package/clis/deepseek/ask.js +1 -0
  155. package/clis/deepseek/history.js +1 -0
  156. package/clis/deepseek/new.js +1 -0
  157. package/clis/deepseek/read.js +1 -0
  158. package/clis/deepseek/status.js +1 -0
  159. package/clis/devto/devto.test.js +236 -0
  160. package/clis/devto/read.js +103 -0
  161. package/clis/devto/tag.js +5 -1
  162. package/clis/devto/top.js +5 -1
  163. package/clis/devto/user.js +5 -1
  164. package/clis/dianping/__fixtures__/search.html +168 -0
  165. package/clis/dianping/__fixtures__/shop.html +6 -0
  166. package/clis/dianping/dianping.test.js +424 -0
  167. package/clis/dianping/search.js +154 -0
  168. package/clis/dianping/shop.js +173 -0
  169. package/clis/dianping/utils.js +157 -0
  170. package/clis/dictionary/examples.js +1 -0
  171. package/clis/dictionary/search.js +1 -0
  172. package/clis/dictionary/synonyms.js +1 -0
  173. package/clis/discord-app/channels.js +1 -0
  174. package/clis/discord-app/delete.js +1 -0
  175. package/clis/discord-app/members.js +1 -0
  176. package/clis/discord-app/read.js +1 -0
  177. package/clis/discord-app/search.js +1 -0
  178. package/clis/discord-app/send.js +1 -0
  179. package/clis/discord-app/servers.js +1 -0
  180. package/clis/discord-app/status.js +1 -0
  181. package/clis/douban/book-hot.js +1 -0
  182. package/clis/douban/download.js +1 -0
  183. package/clis/douban/marks.js +1 -0
  184. package/clis/douban/movie-hot.js +2 -1
  185. package/clis/douban/movie-hot.test.js +14 -0
  186. package/clis/douban/photos.js +2 -1
  187. package/clis/douban/reviews.js +1 -0
  188. package/clis/douban/search.js +1 -0
  189. package/clis/douban/subject.js +1 -0
  190. package/clis/douban/top250.js +1 -0
  191. package/clis/douban/utils.js +11 -13
  192. package/clis/douban/utils.test.js +79 -0
  193. package/clis/doubao/ask.js +1 -0
  194. package/clis/doubao/detail.js +1 -0
  195. package/clis/doubao/history.js +1 -0
  196. package/clis/doubao/meeting-summary.js +1 -0
  197. package/clis/doubao/meeting-transcript.js +1 -0
  198. package/clis/doubao/new.js +1 -0
  199. package/clis/doubao/read.js +1 -0
  200. package/clis/doubao/send.js +1 -0
  201. package/clis/doubao/status.js +1 -0
  202. package/clis/doubao-app/ask.js +1 -0
  203. package/clis/doubao-app/dump.js +1 -0
  204. package/clis/doubao-app/new.js +1 -0
  205. package/clis/doubao-app/read.js +1 -0
  206. package/clis/doubao-app/screenshot.js +1 -0
  207. package/clis/doubao-app/send.js +1 -0
  208. package/clis/doubao-app/status.js +1 -0
  209. package/clis/douyin/activities.js +1 -0
  210. package/clis/douyin/collections.js +1 -0
  211. package/clis/douyin/delete.js +1 -0
  212. package/clis/douyin/draft.js +1 -0
  213. package/clis/douyin/drafts.js +1 -0
  214. package/clis/douyin/hashtag.js +1 -0
  215. package/clis/douyin/location.js +1 -0
  216. package/clis/douyin/profile.js +1 -0
  217. package/clis/douyin/publish.js +1 -0
  218. package/clis/douyin/stats.js +1 -0
  219. package/clis/douyin/update.js +1 -0
  220. package/clis/douyin/user-videos.js +1 -0
  221. package/clis/douyin/videos.js +1 -0
  222. package/clis/eastmoney/announcement.js +1 -0
  223. package/clis/eastmoney/convertible.js +1 -0
  224. package/clis/eastmoney/etf.js +1 -0
  225. package/clis/eastmoney/holders.js +1 -0
  226. package/clis/eastmoney/hot-rank.js +1 -0
  227. package/clis/eastmoney/index-board.js +1 -0
  228. package/clis/eastmoney/kline.js +1 -0
  229. package/clis/eastmoney/kuaixun.js +1 -0
  230. package/clis/eastmoney/longhu.js +1 -0
  231. package/clis/eastmoney/money-flow.js +1 -0
  232. package/clis/eastmoney/northbound.js +1 -0
  233. package/clis/eastmoney/quote.js +1 -0
  234. package/clis/eastmoney/rank.js +1 -0
  235. package/clis/eastmoney/sectors.js +1 -0
  236. package/clis/facebook/add-friend.js +1 -0
  237. package/clis/facebook/events.js +1 -0
  238. package/clis/facebook/feed.js +1 -0
  239. package/clis/facebook/friends.js +1 -0
  240. package/clis/facebook/groups.js +1 -0
  241. package/clis/facebook/join-group.js +1 -0
  242. package/clis/facebook/marketplace-inbox.js +1 -0
  243. package/clis/facebook/marketplace-listings.js +1 -0
  244. package/clis/facebook/memories.js +1 -0
  245. package/clis/facebook/notifications.js +1 -0
  246. package/clis/facebook/profile.js +1 -0
  247. package/clis/facebook/search.js +1 -0
  248. package/clis/gemini/ask.js +1 -0
  249. package/clis/gemini/deep-research-result.js +1 -0
  250. package/clis/gemini/deep-research.js +1 -0
  251. package/clis/gemini/image.js +1 -0
  252. package/clis/gemini/new.js +1 -0
  253. package/clis/gitee/search.js +1 -0
  254. package/clis/gitee/trending.js +1 -0
  255. package/clis/gitee/user.js +1 -0
  256. package/clis/google/news.js +1 -0
  257. package/clis/google/search.js +1 -0
  258. package/clis/google/suggest.js +1 -0
  259. package/clis/google/trends.js +1 -0
  260. package/clis/google-scholar/cite.js +1 -0
  261. package/clis/google-scholar/profile.js +1 -0
  262. package/clis/google-scholar/search.js +1 -0
  263. package/clis/gov-law/recent.js +1 -0
  264. package/clis/gov-law/search.js +1 -0
  265. package/clis/gov-policy/recent.js +1 -0
  266. package/clis/gov-policy/search.js +1 -0
  267. package/clis/grok/ask.js +1 -0
  268. package/clis/grok/image.ts +1 -0
  269. package/clis/hackernews/ask.js +3 -1
  270. package/clis/hackernews/best.js +3 -1
  271. package/clis/hackernews/hackernews.test.js +132 -0
  272. package/clis/hackernews/jobs.js +3 -1
  273. package/clis/hackernews/new.js +3 -1
  274. package/clis/hackernews/read.js +188 -0
  275. package/clis/hackernews/search.js +3 -1
  276. package/clis/hackernews/show.js +3 -1
  277. package/clis/hackernews/top.js +3 -1
  278. package/clis/hackernews/user.js +1 -0
  279. package/clis/hf/top.js +1 -0
  280. package/clis/hupu/detail.js +1 -0
  281. package/clis/hupu/hot.js +3 -1
  282. package/clis/hupu/like.js +1 -0
  283. package/clis/hupu/mentions.js +2 -1
  284. package/clis/hupu/reply.js +1 -0
  285. package/clis/hupu/search.js +3 -1
  286. package/clis/hupu/unlike.js +1 -0
  287. package/clis/imdb/person.js +1 -0
  288. package/clis/imdb/reviews.js +1 -0
  289. package/clis/imdb/search.js +1 -0
  290. package/clis/imdb/title.js +1 -0
  291. package/clis/imdb/top.js +1 -0
  292. package/clis/imdb/trending.js +1 -0
  293. package/clis/indeed/indeed.test.js +375 -0
  294. package/clis/indeed/job.js +86 -0
  295. package/clis/indeed/search.js +110 -0
  296. package/clis/indeed/utils.js +152 -0
  297. package/clis/instagram/collection-create.js +1 -0
  298. package/clis/instagram/collection-delete.js +1 -0
  299. package/clis/instagram/comment.js +1 -0
  300. package/clis/instagram/download.js +1 -0
  301. package/clis/instagram/explore.js +1 -0
  302. package/clis/instagram/follow.js +1 -0
  303. package/clis/instagram/followers.js +1 -0
  304. package/clis/instagram/following.js +1 -0
  305. package/clis/instagram/like.js +1 -0
  306. package/clis/instagram/note.js +1 -0
  307. package/clis/instagram/post.js +1 -0
  308. package/clis/instagram/profile.js +1 -0
  309. package/clis/instagram/reel.js +1 -0
  310. package/clis/instagram/save.js +1 -0
  311. package/clis/instagram/saved.js +1 -0
  312. package/clis/instagram/search.js +1 -0
  313. package/clis/instagram/story.js +1 -0
  314. package/clis/instagram/unfollow.js +1 -0
  315. package/clis/instagram/unlike.js +1 -0
  316. package/clis/instagram/unsave.js +1 -0
  317. package/clis/instagram/user.js +1 -0
  318. package/clis/jd/add-cart.js +1 -0
  319. package/clis/jd/cart.js +1 -0
  320. package/clis/jd/detail.js +1 -0
  321. package/clis/jd/item.js +1 -0
  322. package/clis/jd/reviews.js +1 -0
  323. package/clis/jd/search.js +1 -0
  324. package/clis/jianyu/detail.js +1 -0
  325. package/clis/jianyu/search.js +1 -0
  326. package/clis/jike/comment.js +1 -0
  327. package/clis/jike/create.js +1 -0
  328. package/clis/jike/feed.js +3 -1
  329. package/clis/jike/like.js +1 -0
  330. package/clis/jike/notifications.js +1 -0
  331. package/clis/jike/post.js +1 -0
  332. package/clis/jike/repost.js +1 -0
  333. package/clis/jike/search.js +3 -1
  334. package/clis/jike/topic.js +1 -0
  335. package/clis/jike/user.js +3 -1
  336. package/clis/jimeng/generate.js +1 -0
  337. package/clis/jimeng/history.js +1 -0
  338. package/clis/jimeng/new.js +1 -0
  339. package/clis/jimeng/workspaces.js +1 -0
  340. package/clis/ke/chengjiao.js +1 -0
  341. package/clis/ke/ershoufang.js +1 -0
  342. package/clis/ke/xiaoqu.js +1 -0
  343. package/clis/ke/zufang.js +1 -0
  344. package/clis/lesswrong/comments.js +1 -0
  345. package/clis/lesswrong/curated.js +1 -0
  346. package/clis/lesswrong/frontpage.js +1 -0
  347. package/clis/lesswrong/new.js +1 -0
  348. package/clis/lesswrong/read.js +1 -0
  349. package/clis/lesswrong/sequences.js +1 -0
  350. package/clis/lesswrong/shortform.js +1 -0
  351. package/clis/lesswrong/tag.js +1 -0
  352. package/clis/lesswrong/tags.js +1 -0
  353. package/clis/lesswrong/top-month.js +1 -0
  354. package/clis/lesswrong/top-week.js +1 -0
  355. package/clis/lesswrong/top-year.js +1 -0
  356. package/clis/lesswrong/top.js +1 -0
  357. package/clis/lesswrong/user-posts.js +1 -0
  358. package/clis/lesswrong/user.js +1 -0
  359. package/clis/linkedin/search.js +1 -0
  360. package/clis/linkedin/timeline.js +1 -0
  361. package/clis/linux-do/categories.js +1 -0
  362. package/clis/linux-do/category.js +1 -0
  363. package/clis/linux-do/feed.js +1 -0
  364. package/clis/linux-do/hot.js +1 -0
  365. package/clis/linux-do/latest.js +1 -0
  366. package/clis/linux-do/search.js +1 -0
  367. package/clis/linux-do/tags.js +2 -1
  368. package/clis/linux-do/topic-content.js +1 -0
  369. package/clis/linux-do/topic.js +1 -0
  370. package/clis/linux-do/user-posts.js +1 -0
  371. package/clis/linux-do/user-topics.js +1 -0
  372. package/clis/lobsters/active.js +4 -1
  373. package/clis/lobsters/hot.js +4 -1
  374. package/clis/lobsters/lobsters.test.js +169 -0
  375. package/clis/lobsters/newest.js +4 -1
  376. package/clis/lobsters/read.js +196 -0
  377. package/clis/lobsters/tag.js +4 -1
  378. package/clis/maimai/search-talents.js +1 -0
  379. package/clis/medium/feed.js +1 -0
  380. package/clis/medium/search.js +1 -0
  381. package/clis/medium/user.js +1 -0
  382. package/clis/mubu/doc.js +1 -0
  383. package/clis/mubu/docs.js +1 -0
  384. package/clis/mubu/notes.js +1 -0
  385. package/clis/mubu/recent.js +1 -0
  386. package/clis/mubu/search.js +1 -0
  387. package/clis/notebooklm/current.js +1 -0
  388. package/clis/notebooklm/get.js +1 -0
  389. package/clis/notebooklm/history.js +1 -0
  390. package/clis/notebooklm/list.js +1 -0
  391. package/clis/notebooklm/note-list.js +1 -0
  392. package/clis/notebooklm/notes-get.js +1 -0
  393. package/clis/notebooklm/open.js +1 -0
  394. package/clis/notebooklm/source-fulltext.js +1 -0
  395. package/clis/notebooklm/source-get.js +1 -0
  396. package/clis/notebooklm/source-guide.js +1 -0
  397. package/clis/notebooklm/source-list.js +1 -0
  398. package/clis/notebooklm/status.js +1 -0
  399. package/clis/notebooklm/summary.js +1 -0
  400. package/clis/notion/export.js +1 -0
  401. package/clis/notion/favorites.js +1 -0
  402. package/clis/notion/new.js +1 -0
  403. package/clis/notion/read.js +1 -0
  404. package/clis/notion/search.js +1 -0
  405. package/clis/notion/sidebar.js +1 -0
  406. package/clis/notion/status.js +1 -0
  407. package/clis/notion/write.js +1 -0
  408. package/clis/nowcoder/companies.js +1 -0
  409. package/clis/nowcoder/creators.js +1 -0
  410. package/clis/nowcoder/detail.js +1 -0
  411. package/clis/nowcoder/experience.js +1 -0
  412. package/clis/nowcoder/hot.js +1 -0
  413. package/clis/nowcoder/jobs.js +1 -0
  414. package/clis/nowcoder/notifications.js +1 -0
  415. package/clis/nowcoder/papers.js +1 -0
  416. package/clis/nowcoder/practice.js +1 -0
  417. package/clis/nowcoder/recommend.js +1 -0
  418. package/clis/nowcoder/referral.js +1 -0
  419. package/clis/nowcoder/salary.js +1 -0
  420. package/clis/nowcoder/search.js +1 -0
  421. package/clis/nowcoder/suggest.js +1 -0
  422. package/clis/nowcoder/topics.js +1 -0
  423. package/clis/nowcoder/trending.js +1 -0
  424. package/clis/ones/login.js +1 -0
  425. package/clis/ones/logout.js +1 -0
  426. package/clis/ones/me.js +1 -0
  427. package/clis/ones/my-tasks.js +1 -0
  428. package/clis/ones/task.js +1 -0
  429. package/clis/ones/tasks.js +1 -0
  430. package/clis/ones/token-info.js +1 -0
  431. package/clis/ones/worklog.js +1 -0
  432. package/clis/openreview/openreview.test.js +345 -0
  433. package/clis/openreview/paper.js +43 -0
  434. package/clis/openreview/reviews.js +131 -0
  435. package/clis/openreview/search.js +46 -0
  436. package/clis/openreview/utils.js +158 -0
  437. package/clis/openreview/venue.js +63 -0
  438. package/clis/paperreview/feedback.js +1 -0
  439. package/clis/paperreview/review.js +1 -0
  440. package/clis/paperreview/submit.js +1 -0
  441. package/clis/pixiv/detail.js +1 -0
  442. package/clis/pixiv/download.js +1 -0
  443. package/clis/pixiv/illusts.js +2 -1
  444. package/clis/pixiv/ranking.js +2 -1
  445. package/clis/pixiv/search.js +2 -1
  446. package/clis/pixiv/user.js +2 -0
  447. package/clis/powerchina/search.js +1 -0
  448. package/clis/producthunt/browse.js +1 -0
  449. package/clis/producthunt/hot.js +1 -0
  450. package/clis/producthunt/posts.js +1 -0
  451. package/clis/producthunt/today.js +1 -0
  452. package/clis/quark/ls.js +1 -0
  453. package/clis/quark/mkdir.js +1 -0
  454. package/clis/quark/mv.js +1 -0
  455. package/clis/quark/rename.js +1 -0
  456. package/clis/quark/rm.js +1 -0
  457. package/clis/quark/save.js +1 -0
  458. package/clis/quark/share-tree.js +1 -0
  459. package/clis/reddit/comment.js +1 -0
  460. package/clis/reddit/frontpage.js +1 -0
  461. package/clis/reddit/hot.js +6 -1
  462. package/clis/reddit/hot.test.js +18 -0
  463. package/clis/reddit/popular.js +1 -0
  464. package/clis/reddit/read.js +1 -0
  465. package/clis/reddit/save.js +1 -0
  466. package/clis/reddit/saved.js +1 -0
  467. package/clis/reddit/search.js +1 -0
  468. package/clis/reddit/subreddit.js +1 -0
  469. package/clis/reddit/subscribe.js +1 -0
  470. package/clis/reddit/upvote.js +1 -0
  471. package/clis/reddit/upvoted.js +1 -0
  472. package/clis/reddit/user-comments.js +1 -0
  473. package/clis/reddit/user-posts.js +1 -0
  474. package/clis/reddit/user.js +1 -0
  475. package/clis/reuters/search.js +1 -0
  476. package/clis/sinablog/article.js +1 -0
  477. package/clis/sinablog/hot.js +1 -0
  478. package/clis/sinablog/search.js +1 -0
  479. package/clis/sinablog/user.js +1 -0
  480. package/clis/sinafinance/news.js +1 -0
  481. package/clis/sinafinance/rolling-news.js +1 -0
  482. package/clis/sinafinance/stock-rank.js +1 -0
  483. package/clis/sinafinance/stock.js +1 -0
  484. package/clis/smzdm/search.js +1 -0
  485. package/clis/spotify/spotify.js +11 -0
  486. package/clis/stackoverflow/bounties.js +11 -3
  487. package/clis/stackoverflow/hot.js +10 -2
  488. package/clis/stackoverflow/read.js +314 -0
  489. package/clis/stackoverflow/search.js +10 -2
  490. package/clis/stackoverflow/stackoverflow.test.js +346 -0
  491. package/clis/stackoverflow/unanswered.js +9 -2
  492. package/clis/steam/top-sellers.js +1 -0
  493. package/clis/substack/feed.js +1 -0
  494. package/clis/substack/publication.js +1 -0
  495. package/clis/substack/search.js +1 -0
  496. package/clis/taobao/add-cart.js +1 -0
  497. package/clis/taobao/cart.js +1 -0
  498. package/clis/taobao/detail.js +1 -0
  499. package/clis/taobao/reviews.js +1 -0
  500. package/clis/taobao/search.js +1 -0
  501. package/clis/tdx/hot-rank.js +1 -0
  502. package/clis/ths/hot-rank.js +1 -0
  503. package/clis/tieba/hot.js +2 -1
  504. package/clis/tieba/posts.js +1 -0
  505. package/clis/tieba/read.js +1 -0
  506. package/clis/tieba/search.js +2 -1
  507. package/clis/tiktok/comment.js +1 -0
  508. package/clis/tiktok/explore.js +1 -0
  509. package/clis/tiktok/follow.js +1 -0
  510. package/clis/tiktok/following.js +1 -0
  511. package/clis/tiktok/friends.js +1 -0
  512. package/clis/tiktok/like.js +1 -0
  513. package/clis/tiktok/live.js +1 -0
  514. package/clis/tiktok/notifications.js +1 -0
  515. package/clis/tiktok/profile.js +1 -0
  516. package/clis/tiktok/save.js +1 -0
  517. package/clis/tiktok/search.js +1 -0
  518. package/clis/tiktok/unfollow.js +1 -0
  519. package/clis/tiktok/unlike.js +1 -0
  520. package/clis/tiktok/unsave.js +1 -0
  521. package/clis/tiktok/user.js +1 -0
  522. package/clis/toutiao/articles.js +1 -0
  523. package/clis/twitter/accept.js +1 -0
  524. package/clis/twitter/article.js +1 -0
  525. package/clis/twitter/block.js +1 -0
  526. package/clis/twitter/bookmark.js +1 -0
  527. package/clis/twitter/bookmarks.js +2 -1
  528. package/clis/twitter/delete.js +1 -0
  529. package/clis/twitter/download.js +1 -0
  530. package/clis/twitter/follow.js +1 -0
  531. package/clis/twitter/followers.js +1 -0
  532. package/clis/twitter/following.js +1 -0
  533. package/clis/twitter/hide-reply.js +1 -0
  534. package/clis/twitter/like.js +1 -0
  535. package/clis/twitter/likes.js +2 -1
  536. package/clis/twitter/list-add.js +1 -0
  537. package/clis/twitter/list-remove.js +1 -0
  538. package/clis/twitter/list-tweets.js +1 -0
  539. package/clis/twitter/lists.js +1 -0
  540. package/clis/twitter/notifications.js +1 -0
  541. package/clis/twitter/post.js +1 -0
  542. package/clis/twitter/profile.js +1 -0
  543. package/clis/twitter/reply-dm.js +1 -0
  544. package/clis/twitter/reply.js +1 -0
  545. package/clis/twitter/search.js +1 -0
  546. package/clis/twitter/thread.js +1 -0
  547. package/clis/twitter/timeline.js +1 -0
  548. package/clis/twitter/trending.js +11 -12
  549. package/clis/twitter/trending.test.js +15 -0
  550. package/clis/twitter/tweets.js +2 -1
  551. package/clis/twitter/tweets.test.js +2 -2
  552. package/clis/twitter/unblock.js +1 -0
  553. package/clis/twitter/unbookmark.js +1 -0
  554. package/clis/twitter/unfollow.js +1 -0
  555. package/clis/uiverse/code.js +1 -0
  556. package/clis/uiverse/preview.js +1 -0
  557. package/clis/v2ex/daily.js +1 -0
  558. package/clis/v2ex/hot.js +1 -0
  559. package/clis/v2ex/latest.js +1 -0
  560. package/clis/v2ex/me.js +1 -0
  561. package/clis/v2ex/member.js +1 -0
  562. package/clis/v2ex/node.js +1 -0
  563. package/clis/v2ex/nodes.js +1 -0
  564. package/clis/v2ex/notifications.js +1 -0
  565. package/clis/v2ex/replies.js +1 -0
  566. package/clis/v2ex/topic.js +1 -0
  567. package/clis/v2ex/user.js +1 -0
  568. package/clis/wanfang/search.js +1 -0
  569. package/clis/web/read.js +1 -0
  570. package/clis/weibo/comments.js +1 -0
  571. package/clis/weibo/favorites.js +1 -0
  572. package/clis/weibo/feed.js +3 -1
  573. package/clis/weibo/hot.js +1 -0
  574. package/clis/weibo/me.js +1 -0
  575. package/clis/weibo/post.js +1 -0
  576. package/clis/weibo/publish.js +1 -0
  577. package/clis/weibo/search.js +8 -2
  578. package/clis/weibo/user.js +1 -0
  579. package/clis/weixin/create-draft.js +1 -0
  580. package/clis/weixin/download.js +1 -0
  581. package/clis/weixin/drafts.js +1 -0
  582. package/clis/weread/ai-outline.js +1 -0
  583. package/clis/weread/book.js +1 -0
  584. package/clis/weread/highlights.js +1 -0
  585. package/clis/weread/notebooks.js +1 -0
  586. package/clis/weread/notes.js +1 -0
  587. package/clis/weread/ranking.js +1 -0
  588. package/clis/weread/search.js +1 -0
  589. package/clis/weread/shelf.js +1 -0
  590. package/clis/wikipedia/random.js +1 -0
  591. package/clis/wikipedia/search.js +1 -0
  592. package/clis/wikipedia/summary.js +1 -0
  593. package/clis/wikipedia/trending.js +1 -0
  594. package/clis/xianyu/chat.js +1 -0
  595. package/clis/xianyu/item.js +1 -0
  596. package/clis/xianyu/search.js +1 -0
  597. package/clis/xiaoe/catalog.js +2 -1
  598. package/clis/xiaoe/content.js +1 -0
  599. package/clis/xiaoe/courses.js +1 -0
  600. package/clis/xiaoe/detail.js +1 -0
  601. package/clis/xiaoe/play-url.js +1 -0
  602. package/clis/xiaohongshu/comments.js +1 -0
  603. package/clis/xiaohongshu/creator-note-detail.js +1 -0
  604. package/clis/xiaohongshu/creator-notes-summary.js +1 -0
  605. package/clis/xiaohongshu/creator-notes.js +1 -0
  606. package/clis/xiaohongshu/creator-profile.js +1 -0
  607. package/clis/xiaohongshu/creator-stats.js +1 -0
  608. package/clis/xiaohongshu/download.js +1 -0
  609. package/clis/xiaohongshu/feed.js +2 -1
  610. package/clis/xiaohongshu/note.js +1 -0
  611. package/clis/xiaohongshu/notifications.js +1 -0
  612. package/clis/xiaohongshu/publish.js +1 -0
  613. package/clis/xiaohongshu/search.js +1 -0
  614. package/clis/xiaohongshu/user.js +1 -0
  615. package/clis/xiaoyuzhou/download.js +1 -0
  616. package/clis/xiaoyuzhou/episode.js +1 -0
  617. package/clis/xiaoyuzhou/podcast-episodes.js +1 -0
  618. package/clis/xiaoyuzhou/podcast.js +1 -0
  619. package/clis/xiaoyuzhou/transcript.js +1 -0
  620. package/clis/xueqiu/comments.js +1 -0
  621. package/clis/xueqiu/earnings-date.js +1 -0
  622. package/clis/xueqiu/feed.js +1 -0
  623. package/clis/xueqiu/fund-holdings.js +1 -0
  624. package/clis/xueqiu/fund-snapshot.js +1 -0
  625. package/clis/xueqiu/groups.js +1 -0
  626. package/clis/xueqiu/hot-stock.js +1 -0
  627. package/clis/xueqiu/hot.js +1 -0
  628. package/clis/xueqiu/kline.js +1 -0
  629. package/clis/xueqiu/search.js +1 -0
  630. package/clis/xueqiu/stock.js +1 -0
  631. package/clis/xueqiu/watchlist.js +1 -0
  632. package/clis/yahoo-finance/quote.js +1 -0
  633. package/clis/yollomi/background.js +1 -0
  634. package/clis/yollomi/edit.js +1 -0
  635. package/clis/yollomi/face-swap.js +1 -0
  636. package/clis/yollomi/generate.js +1 -0
  637. package/clis/yollomi/models.js +1 -0
  638. package/clis/yollomi/object-remover.js +1 -0
  639. package/clis/yollomi/remove-bg.js +1 -0
  640. package/clis/yollomi/restore.js +1 -0
  641. package/clis/yollomi/try-on.js +1 -0
  642. package/clis/yollomi/upload.js +1 -0
  643. package/clis/yollomi/upscale.js +1 -0
  644. package/clis/yollomi/video.js +1 -0
  645. package/clis/youtube/channel.js +1 -0
  646. package/clis/youtube/comments.js +1 -0
  647. package/clis/youtube/feed.js +8 -7
  648. package/clis/youtube/feed.test.js +131 -0
  649. package/clis/youtube/history.js +1 -0
  650. package/clis/youtube/like.js +1 -0
  651. package/clis/youtube/playlist.js +1 -0
  652. package/clis/youtube/search.js +1 -0
  653. package/clis/youtube/subscribe.js +1 -0
  654. package/clis/youtube/subscriptions.js +1 -0
  655. package/clis/youtube/transcript.js +1 -0
  656. package/clis/youtube/unlike.js +1 -0
  657. package/clis/youtube/unsubscribe.js +1 -0
  658. package/clis/youtube/video.js +1 -0
  659. package/clis/youtube/watch-later.js +1 -0
  660. package/clis/yuanbao/ask.js +1 -0
  661. package/clis/yuanbao/new.js +1 -0
  662. package/clis/zhihu/answer.js +1 -0
  663. package/clis/zhihu/collection.js +1 -0
  664. package/clis/zhihu/collections.js +1 -0
  665. package/clis/zhihu/comment.js +1 -0
  666. package/clis/zhihu/download.js +1 -0
  667. package/clis/zhihu/favorite.js +1 -0
  668. package/clis/zhihu/follow.js +1 -0
  669. package/clis/zhihu/hot.js +1 -0
  670. package/clis/zhihu/like.js +1 -0
  671. package/clis/zhihu/question.js +1 -0
  672. package/clis/zhihu/search.js +1 -0
  673. package/clis/zlibrary/info.js +1 -0
  674. package/clis/zlibrary/search.js +1 -0
  675. package/clis/zsxq/dynamics.js +1 -0
  676. package/clis/zsxq/groups.js +1 -0
  677. package/clis/zsxq/search.js +1 -0
  678. package/clis/zsxq/topic.js +1 -0
  679. package/clis/zsxq/topics.js +1 -0
  680. package/dist/src/adapter-source.test.js +1 -1
  681. package/dist/src/browser/analyze.test.js +2 -0
  682. package/dist/src/browser/base-page.d.ts +9 -0
  683. package/dist/src/browser/base-page.js +72 -3
  684. package/dist/src/browser/base-page.test.js +42 -0
  685. package/dist/src/browser/cdp.js +6 -0
  686. package/dist/src/browser/cdp.test.js +3 -0
  687. package/dist/src/browser/page.d.ts +1 -0
  688. package/dist/src/browser/page.js +6 -0
  689. package/dist/src/browser/page.test.js +17 -0
  690. package/dist/src/browser/target-resolver.d.ts +10 -3
  691. package/dist/src/browser/target-resolver.js +16 -10
  692. package/dist/src/browser/verify-fixture.d.ts +6 -1
  693. package/dist/src/browser/verify-fixture.js +87 -0
  694. package/dist/src/browser/verify-fixture.test.js +44 -1
  695. package/dist/src/build-manifest.js +3 -0
  696. package/dist/src/build-manifest.test.js +18 -12
  697. package/dist/src/capabilityRouting.test.js +1 -1
  698. package/dist/src/cli.js +141 -5
  699. package/dist/src/cli.test.js +179 -0
  700. package/dist/src/commanderAdapter.d.ts +1 -1
  701. package/dist/src/commanderAdapter.js +17 -7
  702. package/dist/src/commanderAdapter.test.js +7 -6
  703. package/dist/src/convention-audit.d.ts +50 -0
  704. package/dist/src/convention-audit.js +546 -0
  705. package/dist/src/convention-audit.test.d.ts +1 -0
  706. package/dist/src/convention-audit.test.js +226 -0
  707. package/dist/src/discovery.js +2 -0
  708. package/dist/src/doctor.js +0 -1
  709. package/dist/src/doctor.test.js +1 -1
  710. package/dist/src/engine.test.js +10 -10
  711. package/dist/src/execution.js +1 -1
  712. package/dist/src/execution.test.js +9 -9
  713. package/dist/src/help.d.ts +15 -0
  714. package/dist/src/help.js +149 -0
  715. package/dist/src/main.js +13 -7
  716. package/dist/src/manifest-types.d.ts +2 -0
  717. package/dist/src/plugin.test.js +26 -26
  718. package/dist/src/registry.d.ts +5 -0
  719. package/dist/src/registry.js +8 -0
  720. package/dist/src/registry.test.js +27 -20
  721. package/dist/src/serialization.d.ts +4 -0
  722. package/dist/src/serialization.js +27 -1
  723. package/dist/src/serialization.test.js +24 -2
  724. package/dist/src/types.d.ts +2 -0
  725. package/package.json +4 -1
  726. package/scripts/check-listing-id-pairing.mjs +193 -0
  727. package/scripts/check-silent-column-drop.mjs +105 -0
  728. package/scripts/check-typed-error-lint.mjs +118 -0
  729. package/scripts/silent-column-drop-baseline.json +962 -0
  730. package/scripts/typed-error-lint-baseline.json +1586 -0
@@ -4,6 +4,7 @@ import { getOnesBaseUrl, onesFetchInPage } from './common.js';
4
4
  cli({
5
5
  site: 'ones',
6
6
  name: 'login',
7
+ access: 'write',
7
8
  description: 'ONES Project API — login via Chrome Bridge (POST auth/login); stderr prints export hints for ONES_USER_ID / TOKEN',
8
9
  domain: 'ones.cn',
9
10
  strategy: Strategy.COOKIE,
@@ -3,6 +3,7 @@ import { onesFetchInPage } from './common.js';
3
3
  cli({
4
4
  site: 'ones',
5
5
  name: 'logout',
6
+ access: 'write',
6
7
  description: 'ONES Project API — invalidate current token (GET auth/logout) via Chrome Bridge',
7
8
  domain: 'ones.cn',
8
9
  strategy: Strategy.COOKIE,
package/clis/ones/me.js CHANGED
@@ -4,6 +4,7 @@ import { onesFetchInPage } from './common.js';
4
4
  cli({
5
5
  site: 'ones',
6
6
  name: 'me',
7
+ access: 'read',
7
8
  description: 'ONES Project API — current user (GET users/me) via Chrome Bridge',
8
9
  domain: 'ones.cn',
9
10
  strategy: Strategy.COOKIE,
@@ -39,6 +39,7 @@ async function peekTasks(page, team, query, cap) {
39
39
  cli({
40
40
  site: 'ones',
41
41
  name: 'my-tasks',
42
+ access: 'read',
42
43
  description: 'ONES — my work items (filters/peek + strict must query). Default: assignee=me. Use --mode if your site uses field004 for assignee.',
43
44
  domain: 'ones.cn',
44
45
  strategy: Strategy.COOKIE,
package/clis/ones/task.js CHANGED
@@ -10,6 +10,7 @@ import { formatStamp } from './task-helpers.js';
10
10
  cli({
11
11
  site: 'ones',
12
12
  name: 'task',
13
+ access: 'read',
13
14
  description: 'ONES — work item detail (GET team/:team/task/:id/info); id is URL segment after …/task/',
14
15
  domain: 'ones.cn',
15
16
  strategy: Strategy.COOKIE,
@@ -20,6 +20,7 @@ function buildQuery(project, assign) {
20
20
  cli({
21
21
  site: 'ones',
22
22
  name: 'tasks',
23
+ access: 'read',
23
24
  description: 'ONES Project API — list work items (POST team/:team/filters/peek); use token-info -f json for team uuid',
24
25
  domain: 'ones.cn',
25
26
  strategy: Strategy.COOKIE,
@@ -4,6 +4,7 @@ import { onesFetchInPage } from './common.js';
4
4
  cli({
5
5
  site: 'ones',
6
6
  name: 'token-info',
7
+ access: 'read',
7
8
  description: 'ONES Project API — session detail (GET auth/token_info) via Chrome Bridge: user, teams, org',
8
9
  domain: 'ones.cn',
9
10
  strategy: Strategy.COOKIE,
@@ -99,6 +99,7 @@ export function buildAddManhourGraphqlBody(input) {
99
99
  cli({
100
100
  site: 'ones',
101
101
  name: 'worklog',
102
+ access: 'write',
102
103
  description: 'ONES — log work hours on a task (defaults to today; use --date to backfill; endpoint falls back by deployment).',
103
104
  domain: 'ones.cn',
104
105
  strategy: Strategy.COOKIE,
@@ -0,0 +1,345 @@
1
+ import { afterEach, describe, expect, it, vi } from 'vitest';
2
+ import { getRegistry } from '@jackwener/opencli/registry';
3
+ import {
4
+ absolutePdf,
5
+ coerceInt,
6
+ formatDate,
7
+ noteToRow,
8
+ requireBoundedInt,
9
+ requireForumId,
10
+ requireNonNegativeInt,
11
+ } from './utils.js';
12
+ import './search.js';
13
+ import './venue.js';
14
+ import './paper.js';
15
+ import './reviews.js';
16
+
17
+ const SAMPLE_NOTE = {
18
+ id: 'abc123XYZ_',
19
+ forum: 'abc123XYZ_',
20
+ pdate: 1727524853394,
21
+ cdate: 1727524853394,
22
+ content: {
23
+ title: { value: 'Test Paper Title with spaces' },
24
+ authors: { value: ['Alice Smith', 'Bob Jones'] },
25
+ authorids: { value: ['~Alice_Smith1', '~Bob_Jones2'] },
26
+ keywords: { value: ['transformer', 'attention'] },
27
+ abstract: { value: 'A long abstract\n with newlines.' },
28
+ venue: { value: 'ICLR 2024 oral' },
29
+ venueid: { value: 'ICLR.cc/2024/Conference' },
30
+ primary_area: { value: 'foundation models' },
31
+ pdf: { value: '/pdf/abc.pdf' },
32
+ },
33
+ };
34
+
35
+ afterEach(() => {
36
+ vi.unstubAllGlobals();
37
+ });
38
+
39
+ describe('openreview adapter', () => {
40
+ it('registers all four commands with the expected columns', () => {
41
+ const search = getRegistry().get('openreview/search');
42
+ const venue = getRegistry().get('openreview/venue');
43
+ const paper = getRegistry().get('openreview/paper');
44
+ const reviews = getRegistry().get('openreview/reviews');
45
+
46
+ expect(search).toBeDefined();
47
+ expect(venue).toBeDefined();
48
+ expect(paper).toBeDefined();
49
+ expect(reviews).toBeDefined();
50
+
51
+ expect(search.columns).toEqual(['rank', 'id', 'title', 'authors', 'venue', 'pdate', 'url']);
52
+ expect(venue.columns).toEqual(['rank', 'id', 'title', 'authors', 'keywords', 'primary_area', 'pdate', 'pdf', 'url']);
53
+ expect(paper.columns).toEqual(['id', 'title', 'authors', 'keywords', 'venue', 'venueid', 'primary_area', 'abstract', 'pdate', 'pdf', 'url']);
54
+ expect(reviews.columns).toEqual(['type', 'author', 'rating', 'confidence', 'text']);
55
+ });
56
+
57
+ it('noteToRow extracts every wrapped v2 field, joins lists, and builds absolute URLs', () => {
58
+ const row = noteToRow(SAMPLE_NOTE);
59
+ expect(row.id).toBe('abc123XYZ_');
60
+ expect(row.title).toBe('Test Paper Title with spaces');
61
+ expect(row.authors).toBe('Alice Smith, Bob Jones');
62
+ expect(row.keywords).toBe('transformer, attention');
63
+ expect(row.abstract).toBe('A long abstract with newlines.');
64
+ expect(row.venue).toBe('ICLR 2024 oral');
65
+ expect(row.venueid).toBe('ICLR.cc/2024/Conference');
66
+ expect(row.primary_area).toBe('foundation models');
67
+ expect(row.pdf).toBe('https://openreview.net/pdf/abc.pdf');
68
+ expect(row.url).toBe('https://openreview.net/forum?id=abc123XYZ_');
69
+ expect(row.pdate).toBe('2024-09-28');
70
+ });
71
+
72
+ it('noteToRow falls back to author IDs when authors field is missing', () => {
73
+ const note = { ...SAMPLE_NOTE, content: { ...SAMPLE_NOTE.content, authors: undefined } };
74
+ expect(noteToRow(note).authors).toBe('Alice Smith, Bob Jones');
75
+ });
76
+
77
+ it('coerceInt accepts numeric strings, rejects floats and NaN', () => {
78
+ expect(coerceInt(10)).toBe(10);
79
+ expect(coerceInt('25')).toBe(25);
80
+ expect(Number.isNaN(coerceInt(1.5))).toBe(true);
81
+ expect(Number.isNaN(coerceInt('abc'))).toBe(true);
82
+ expect(Number.isNaN(coerceInt(undefined))).toBe(true);
83
+ expect(Number.isNaN(coerceInt(''))).toBe(true);
84
+ });
85
+
86
+ it('requireBoundedInt rejects non-positive, non-integer and over-cap values', () => {
87
+ expect(requireBoundedInt(10, 25, 50)).toBe(10);
88
+ expect(requireBoundedInt(undefined, 25, 50)).toBe(25);
89
+ expect(() => requireBoundedInt(0, 25, 50)).toThrow('positive integer');
90
+ expect(() => requireBoundedInt(1.5, 25, 50)).toThrow('positive integer');
91
+ expect(() => requireBoundedInt(51, 25, 50)).toThrow('<= 50');
92
+ expect(() => requireBoundedInt('abc', 25, 50)).toThrow('positive integer');
93
+ });
94
+
95
+ it('requireNonNegativeInt accepts 0 and rejects negatives', () => {
96
+ expect(requireNonNegativeInt(0, 0)).toBe(0);
97
+ expect(requireNonNegativeInt(50, 0)).toBe(50);
98
+ expect(() => requireNonNegativeInt(-1, 0)).toThrow('non-negative integer');
99
+ expect(() => requireNonNegativeInt(1.5, 0)).toThrow('non-negative integer');
100
+ });
101
+
102
+ it('requireForumId rejects empty and malformed ids', () => {
103
+ expect(requireForumId('5sRnsubyAK')).toBe('5sRnsubyAK');
104
+ expect(requireForumId('abc-def_12')).toBe('abc-def_12');
105
+ expect(() => requireForumId('')).toThrow('required');
106
+ expect(() => requireForumId(' ')).toThrow('required');
107
+ expect(() => requireForumId('has space')).toThrow('not a valid forum id');
108
+ expect(() => requireForumId('a/b')).toThrow('not a valid forum id');
109
+ expect(() => requireForumId('short')).toThrow('not a valid forum id');
110
+ });
111
+
112
+ it('formatDate handles ms-since-epoch and rejects invalid input', () => {
113
+ expect(formatDate(1727524853394)).toBe('2024-09-28');
114
+ expect(formatDate(0)).toBe('');
115
+ expect(formatDate(undefined)).toBe('');
116
+ expect(formatDate('abc')).toBe('');
117
+ });
118
+
119
+ it('absolutePdf prefixes relative paths and preserves absolute URLs', () => {
120
+ expect(absolutePdf('/pdf/abc.pdf')).toBe('https://openreview.net/pdf/abc.pdf');
121
+ expect(absolutePdf('http://arxiv.org/pdf/1.pdf')).toBe('http://arxiv.org/pdf/1.pdf');
122
+ expect(absolutePdf('https://example.com/x.pdf')).toBe('https://example.com/x.pdf');
123
+ expect(absolutePdf('')).toBe('');
124
+ expect(absolutePdf(undefined)).toBe('');
125
+ });
126
+
127
+ it('search throws ArgumentError on empty query', async () => {
128
+ const search = getRegistry().get('openreview/search');
129
+ await expect(search.func({ query: '', limit: 10 })).rejects.toMatchObject({ code: 'ARGUMENT' });
130
+ await expect(search.func({ query: ' ', limit: 10 })).rejects.toMatchObject({ code: 'ARGUMENT' });
131
+ });
132
+
133
+ it('search throws EmptyResult when API returns no notes', async () => {
134
+ vi.stubGlobal('fetch', vi.fn().mockResolvedValue(new Response(JSON.stringify({ notes: [] }), { status: 200 })));
135
+ const search = getRegistry().get('openreview/search');
136
+ await expect(search.func({ query: 'no-such-paper-xyz', limit: 5 })).rejects.toMatchObject({ code: 'EMPTY_RESULT' });
137
+ });
138
+
139
+ it('search wraps non-200 responses as CommandExecutionError', async () => {
140
+ vi.stubGlobal('fetch', vi.fn().mockResolvedValue(new Response('rate limited', { status: 429 })));
141
+ const search = getRegistry().get('openreview/search');
142
+ await expect(search.func({ query: 'transformer', limit: 5 })).rejects.toMatchObject({ code: 'COMMAND_EXEC' });
143
+ });
144
+
145
+ it('search wraps fetch network errors', async () => {
146
+ vi.stubGlobal('fetch', vi.fn().mockRejectedValue(new Error('ECONNRESET')));
147
+ const search = getRegistry().get('openreview/search');
148
+ await expect(search.func({ query: 'transformer', limit: 5 })).rejects.toMatchObject({
149
+ code: 'COMMAND_EXEC',
150
+ message: expect.stringContaining('Network failure'),
151
+ });
152
+ });
153
+
154
+ it('search wraps malformed JSON', async () => {
155
+ vi.stubGlobal('fetch', vi.fn().mockResolvedValue(new Response('not-json', { status: 200 })));
156
+ const search = getRegistry().get('openreview/search');
157
+ await expect(search.func({ query: 'transformer', limit: 5 })).rejects.toMatchObject({
158
+ code: 'COMMAND_EXEC',
159
+ message: expect.stringContaining('Malformed JSON'),
160
+ });
161
+ });
162
+
163
+ it('search wraps OpenReview in-band error envelopes as CommandExecutionError', async () => {
164
+ vi.stubGlobal('fetch', vi.fn().mockResolvedValue(new Response(JSON.stringify({
165
+ errors: [{ message: 'bad term' }],
166
+ }), { status: 200 })));
167
+ const search = getRegistry().get('openreview/search');
168
+ await expect(search.func({ query: 'transformer', limit: 5 })).rejects.toMatchObject({
169
+ code: 'COMMAND_EXEC',
170
+ message: expect.stringContaining('bad term'),
171
+ });
172
+ });
173
+
174
+ it('search returns rank-ordered rows with the expected shape', async () => {
175
+ vi.stubGlobal('fetch', vi.fn().mockResolvedValue(new Response(JSON.stringify({ notes: [SAMPLE_NOTE, SAMPLE_NOTE] }), { status: 200 })));
176
+ const search = getRegistry().get('openreview/search');
177
+ const rows = await search.func({ query: 'transformer', limit: 5 });
178
+ expect(rows).toHaveLength(2);
179
+ expect(rows[0]).toEqual({
180
+ rank: 1,
181
+ id: 'abc123XYZ_',
182
+ title: 'Test Paper Title with spaces',
183
+ authors: 'Alice Smith, Bob Jones',
184
+ venue: 'ICLR 2024 oral',
185
+ pdate: '2024-09-28',
186
+ url: 'https://openreview.net/forum?id=abc123XYZ_',
187
+ });
188
+ expect(rows[1].rank).toBe(2);
189
+ });
190
+
191
+ it('venue dispatches invitation vs venue-text via /-/ heuristic', async () => {
192
+ // Each call must return a fresh Response — bodies can only be read once.
193
+ const fetchMock = vi.fn().mockImplementation(() => Promise.resolve(new Response(JSON.stringify({ notes: [SAMPLE_NOTE] }), { status: 200 })));
194
+ vi.stubGlobal('fetch', fetchMock);
195
+ const venue = getRegistry().get('openreview/venue');
196
+ await venue.func({ venue: 'ICLR.cc/2025/Conference/-/Submission', limit: 1, offset: 0 });
197
+ expect(fetchMock.mock.calls[0][0]).toContain('invitation=ICLR.cc');
198
+ await venue.func({ venue: 'ICLR 2024 oral', limit: 1, offset: 0 });
199
+ expect(fetchMock.mock.calls[1][0]).toContain('content.venue=ICLR%202024%20oral');
200
+ });
201
+
202
+ it('venue applies offset to the rank column so paginated results stay ordered', async () => {
203
+ vi.stubGlobal('fetch', vi.fn().mockResolvedValue(new Response(JSON.stringify({ notes: [SAMPLE_NOTE, SAMPLE_NOTE] }), { status: 200 })));
204
+ const venue = getRegistry().get('openreview/venue');
205
+ const rows = await venue.func({ venue: 'ICLR 2024 oral', limit: 2, offset: 50 });
206
+ expect(rows[0].rank).toBe(51);
207
+ expect(rows[1].rank).toBe(52);
208
+ });
209
+
210
+ it('paper rejects invalid ids before calling the network', async () => {
211
+ const paper = getRegistry().get('openreview/paper');
212
+ await expect(paper.func({ id: '' })).rejects.toMatchObject({ code: 'ARGUMENT' });
213
+ await expect(paper.func({ id: 'has space' })).rejects.toMatchObject({ code: 'ARGUMENT' });
214
+ });
215
+
216
+ it('paper throws EmptyResult when the note id is unknown', async () => {
217
+ vi.stubGlobal('fetch', vi.fn().mockResolvedValue(new Response(JSON.stringify({ notes: [] }), { status: 200 })));
218
+ const paper = getRegistry().get('openreview/paper');
219
+ await expect(paper.func({ id: 'abc123XYZ_' })).rejects.toMatchObject({ code: 'EMPTY_RESULT' });
220
+ });
221
+
222
+ it('paper returns a single row with the full abstract intact', async () => {
223
+ vi.stubGlobal('fetch', vi.fn().mockResolvedValue(new Response(JSON.stringify({ notes: [SAMPLE_NOTE] }), { status: 200 })));
224
+ const paper = getRegistry().get('openreview/paper');
225
+ const rows = await paper.func({ id: 'abc123XYZ_' });
226
+ expect(rows).toHaveLength(1);
227
+ expect(rows[0].abstract).toBe('A long abstract with newlines.');
228
+ expect(rows[0].pdf).toBe('https://openreview.net/pdf/abc.pdf');
229
+ });
230
+
231
+ it('reviews rejects max-length below 200', async () => {
232
+ const reviews = getRegistry().get('openreview/reviews');
233
+ await expect(reviews.func({ forum: 'abc123XYZ_', 'max-length': 100 })).rejects.toMatchObject({ code: 'ARGUMENT' });
234
+ });
235
+
236
+ it('reviews emits PAPER first then sorts by cdate, classifies invitations, and joins sections', async () => {
237
+ const replyReview = {
238
+ id: 'review1aaa',
239
+ forum: 'abc123XYZ_',
240
+ replyto: 'abc123XYZ_',
241
+ cdate: 1727524900000,
242
+ invitations: ['ICLR.cc/2025/Conference/Submission1/-/Official_Review'],
243
+ signatures: ['ICLR.cc/2025/Conference/Submission1/Reviewer_uVwr'],
244
+ content: {
245
+ summary: { value: 'Short summary' },
246
+ strengths: { value: 'Solid baselines' },
247
+ weaknesses: { value: 'Limited novelty' },
248
+ rating: { value: 5 },
249
+ confidence: { value: 4 },
250
+ },
251
+ };
252
+ const replyDecision = {
253
+ id: 'decision01',
254
+ forum: 'abc123XYZ_',
255
+ replyto: 'abc123XYZ_',
256
+ cdate: 1727525000000,
257
+ invitations: ['ICLR.cc/2025/Conference/-/Decision'],
258
+ signatures: ['ICLR.cc/2025/Conference/Program_Chairs'],
259
+ content: {
260
+ decision: { value: 'Accept (poster)' },
261
+ },
262
+ };
263
+ const replyMetaReview = {
264
+ id: 'meta000001',
265
+ forum: 'abc123XYZ_',
266
+ replyto: 'abc123XYZ_',
267
+ cdate: 1727524950000,
268
+ invitations: ['ICLR.cc/2025/Conference/Submission1/-/Meta_Review'],
269
+ signatures: ['ICLR.cc/2025/Conference/Area_Chair_1'],
270
+ content: {
271
+ summary: { value: 'Meta summary' },
272
+ },
273
+ };
274
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(async (url) => {
275
+ const href = String(url);
276
+ if (href.includes('/notes?id=')) {
277
+ return new Response(JSON.stringify({ notes: [SAMPLE_NOTE] }), { status: 200 });
278
+ }
279
+ return new Response(JSON.stringify({ notes: [replyDecision, replyReview, { ...SAMPLE_NOTE }, replyMetaReview] }), { status: 200 });
280
+ }));
281
+ const reviews = getRegistry().get('openreview/reviews');
282
+ const rows = await reviews.func({ forum: 'abc123XYZ_', 'max-length': 4000 });
283
+ expect(rows[0].type).toBe('PAPER');
284
+ expect(rows.filter((row) => row.type === 'PAPER')).toHaveLength(1);
285
+ expect(rows[1].type).toBe('REVIEW');
286
+ expect(rows[1].author).toBe('Reviewer_uVwr');
287
+ expect(rows[1].rating).toBe('5');
288
+ expect(rows[1].confidence).toBe('4');
289
+ expect(rows[1].text).toContain('Summary: Short summary');
290
+ expect(rows[1].text).toContain('Strengths: Solid baselines');
291
+ expect(rows[2].type).toBe('META_REVIEW');
292
+ expect(rows[2].author).toBe('Area_Chair_1');
293
+ expect(rows[2].text).toContain('Summary: Meta summary');
294
+ expect(rows[3].type).toBe('DECISION');
295
+ expect(rows[3].author).toBe('Program_Chairs');
296
+ expect(rows[3].text).toContain('Decision: Accept (poster)');
297
+ });
298
+
299
+ it('reviews still emits PAPER row 0 when forum replies response omits the root note', async () => {
300
+ const replyReview = {
301
+ id: 'review1aaa',
302
+ forum: 'abc123XYZ_',
303
+ replyto: 'abc123XYZ_',
304
+ cdate: 1727524900000,
305
+ invitations: ['ICLR.cc/2025/Conference/Submission1/-/Official_Review'],
306
+ signatures: ['ICLR.cc/2025/Conference/Submission1/Reviewer_uVwr'],
307
+ content: { summary: { value: 'Short summary' } },
308
+ };
309
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(async (url) => {
310
+ const href = String(url);
311
+ if (href.includes('/notes?id=')) {
312
+ return new Response(JSON.stringify({ notes: [SAMPLE_NOTE] }), { status: 200 });
313
+ }
314
+ return new Response(JSON.stringify({ notes: [replyReview] }), { status: 200 });
315
+ }));
316
+ const reviews = getRegistry().get('openreview/reviews');
317
+ const rows = await reviews.func({ forum: 'abc123XYZ_', 'max-length': 4000 });
318
+ expect(rows[0].type).toBe('PAPER');
319
+ expect(rows[0].author).toBe('');
320
+ expect(rows[1].type).toBe('REVIEW');
321
+ });
322
+
323
+ it('reviews truncates long text to max-length with ellipsis', async () => {
324
+ const longReview = {
325
+ id: 'longreview1',
326
+ forum: 'abc123XYZ_',
327
+ replyto: 'abc123XYZ_',
328
+ cdate: 1727524900000,
329
+ invitations: ['ICLR.cc/2025/Conference/Submission1/-/Official_Review'],
330
+ signatures: ['ICLR.cc/2025/Conference/Submission1/Reviewer_xxxx'],
331
+ content: { summary: { value: 'x'.repeat(5000) } },
332
+ };
333
+ vi.stubGlobal('fetch', vi.fn().mockImplementation(async (url) => {
334
+ const href = String(url);
335
+ if (href.includes('/notes?id=')) {
336
+ return new Response(JSON.stringify({ notes: [SAMPLE_NOTE] }), { status: 200 });
337
+ }
338
+ return new Response(JSON.stringify({ notes: [longReview] }), { status: 200 });
339
+ }));
340
+ const reviews = getRegistry().get('openreview/reviews');
341
+ const rows = await reviews.func({ forum: 'abc123XYZ_', 'max-length': 500 });
342
+ expect(rows[1].text.length).toBe(500);
343
+ expect(rows[1].text.endsWith('...')).toBe(true);
344
+ });
345
+ });
@@ -0,0 +1,43 @@
1
+ /**
2
+ * OpenReview single paper detail (full abstract + metadata).
3
+ */
4
+ import { cli, Strategy } from '@jackwener/opencli/registry';
5
+ import { EmptyResultError } from '@jackwener/opencli/errors';
6
+ import { noteToRow, openreviewFetch, requireForumId } from './utils.js';
7
+
8
+ cli({
9
+ site: 'openreview',
10
+ name: 'paper',
11
+ access: 'read',
12
+ description: 'Show full metadata for a single OpenReview paper',
13
+ domain: 'openreview.net',
14
+ strategy: Strategy.PUBLIC,
15
+ browser: false,
16
+ args: [
17
+ { name: 'id', positional: true, required: true, help: 'OpenReview note id (e.g. "5sRnsubyAK")' },
18
+ ],
19
+ columns: ['id', 'title', 'authors', 'keywords', 'venue', 'venueid', 'primary_area', 'abstract', 'pdate', 'pdf', 'url'],
20
+ func: async (args) => {
21
+ const id = requireForumId(args.id);
22
+ const path = `/notes?id=${encodeURIComponent(id)}`;
23
+ const json = await openreviewFetch(path, `openreview paper ${id}`);
24
+ const notes = Array.isArray(json?.notes) ? json.notes : [];
25
+ if (!notes.length) {
26
+ throw new EmptyResultError('openreview', `No paper found with id "${id}". Confirm the forum/note id from openreview.net.`);
27
+ }
28
+ const row = noteToRow(notes[0]);
29
+ return [{
30
+ id: row.id,
31
+ title: row.title,
32
+ authors: row.authors,
33
+ keywords: row.keywords,
34
+ venue: row.venue,
35
+ venueid: row.venueid,
36
+ primary_area: row.primary_area,
37
+ abstract: row.abstract,
38
+ pdate: row.pdate,
39
+ pdf: row.pdf,
40
+ url: row.url,
41
+ }];
42
+ },
43
+ });
@@ -0,0 +1,131 @@
1
+ /**
2
+ * OpenReview paper + threaded reviews/decisions/comments.
3
+ *
4
+ * Walks the forum tree once and emits one row per note in chronological order:
5
+ * PAPER → REVIEW (one per reviewer) → REBUTTAL/COMMENT → DECISION → WITHDRAWAL.
6
+ *
7
+ * Each row carries `rating` / `confidence` so reviewers stand out at a glance.
8
+ */
9
+ import { cli, Strategy } from '@jackwener/opencli/registry';
10
+ import { ArgumentError, EmptyResultError } from '@jackwener/opencli/errors';
11
+ import { coerceInt, openreviewFetch, readContent, requireForumId } from './utils.js';
12
+
13
+ const SECTION_FIELDS = [
14
+ ['summary', 'Summary'],
15
+ ['strengths', 'Strengths'],
16
+ ['weaknesses', 'Weaknesses'],
17
+ ['questions', 'Questions'],
18
+ ['comment', 'Comment'],
19
+ ['rebuttal', 'Rebuttal'],
20
+ ['decision', 'Decision'],
21
+ ['recommendation', 'Recommendation'],
22
+ ['title', 'Title'],
23
+ ['abstract', 'Abstract'],
24
+ ['withdrawal_confirmation', 'Withdrawal confirmation'],
25
+ ];
26
+
27
+ /** Pull the trailing segment of an invitation id (e.g. ".../-/Official_Review" → "Official_Review"). */
28
+ function invitationTail(invitations) {
29
+ if (!Array.isArray(invitations)) return '';
30
+ for (const inv of invitations) {
31
+ const m = String(inv).match(/\/-\/([^/]+)$/);
32
+ if (m) return m[1];
33
+ }
34
+ return '';
35
+ }
36
+
37
+ /** Map a note's invitation tail to a row type label. */
38
+ function classifyNote(note, isRoot) {
39
+ if (isRoot) return 'PAPER';
40
+ const tail = invitationTail(note?.invitations).toLowerCase();
41
+ if (tail.includes('decision')) return 'DECISION';
42
+ if (tail.includes('withdrawal')) return 'WITHDRAWAL';
43
+ if (tail.includes('rebuttal')) return 'REBUTTAL';
44
+ if (tail.includes('meta')) return 'META_REVIEW';
45
+ if (tail.includes('review')) return 'REVIEW';
46
+ if (tail.includes('comment')) return 'COMMENT';
47
+ return tail ? tail.toUpperCase() : 'NOTE';
48
+ }
49
+
50
+ /**
51
+ * Pick the most informative signature segment, e.g.:
52
+ * "ICLR.cc/2025/Conference/Submission14296/Reviewer_uVwr" → "Reviewer_uVwr"
53
+ * "ICLR.cc/2025/Conference/Program_Chairs" → "Program_Chairs"
54
+ * "~Pedro_Jose_Moreno_Mengibar1" → "Pedro Jose Moreno Mengibar"
55
+ */
56
+ function authorFromSignatures(signatures) {
57
+ if (!Array.isArray(signatures) || !signatures.length) return '';
58
+ const sig = String(signatures[0]);
59
+ if (sig.startsWith('~')) {
60
+ return sig.replace(/^~/, '').replace(/\d+$/, '').replace(/_/g, ' ').trim();
61
+ }
62
+ const parts = sig.split('/');
63
+ return parts[parts.length - 1] || sig;
64
+ }
65
+
66
+ function joinSections(content) {
67
+ const parts = [];
68
+ for (const [key, label] of SECTION_FIELDS) {
69
+ const value = readContent(content, key);
70
+ if (value === undefined || value === null) continue;
71
+ const text = Array.isArray(value) ? value.join(', ') : String(value);
72
+ const trimmed = text.replace(/\r\n/g, '\n').trim();
73
+ if (!trimmed) continue;
74
+ parts.push(`${label}: ${trimmed}`);
75
+ }
76
+ return parts.join('\n\n');
77
+ }
78
+
79
+ function truncate(text, maxLength) {
80
+ if (text.length <= maxLength) return text;
81
+ return `${text.slice(0, maxLength - 3)}...`;
82
+ }
83
+
84
+ cli({
85
+ site: 'openreview',
86
+ name: 'reviews',
87
+ access: 'read',
88
+ description: 'Show full review thread (paper + reviews + decisions) for an OpenReview forum',
89
+ domain: 'openreview.net',
90
+ strategy: Strategy.PUBLIC,
91
+ browser: false,
92
+ args: [
93
+ { name: 'forum', positional: true, required: true, help: 'OpenReview forum id (same as paper id)' },
94
+ { name: 'max-length', type: 'int', default: 4000, help: 'Per-row text truncation (min 200)' },
95
+ ],
96
+ columns: ['type', 'author', 'rating', 'confidence', 'text'],
97
+ func: async (args) => {
98
+ const forum = requireForumId(args.forum, 'forum');
99
+ const rawMax = args['max-length'] ?? args.maxLength ?? 4000;
100
+ const maxLength = coerceInt(rawMax);
101
+ if (!Number.isInteger(maxLength) || maxLength < 200) {
102
+ throw new ArgumentError('openreview reviews max-length must be an integer >= 200');
103
+ }
104
+ const rootJson = await openreviewFetch(`/notes?id=${encodeURIComponent(forum)}`, `openreview paper ${forum}`);
105
+ const rootNotes = Array.isArray(rootJson?.notes) ? rootJson.notes : [];
106
+ const root = rootNotes[0];
107
+ if (!root) {
108
+ throw new EmptyResultError('openreview', `No forum found with id "${forum}". Confirm the forum id from openreview.net.`);
109
+ }
110
+ const repliesJson = await openreviewFetch(`/notes?forum=${encodeURIComponent(forum)}&details=replies&limit=1000`, `openreview reviews ${forum}`);
111
+ const replies = Array.isArray(repliesJson?.notes) ? repliesJson.notes.filter(note => note?.id !== forum) : [];
112
+ // Sort by cdate (creation time) so ordering is deterministic regardless of API order.
113
+ const sorted = [...replies].sort((a, b) => (a?.cdate ?? 0) - (b?.cdate ?? 0));
114
+ const ordered = [root, ...sorted];
115
+ return ordered.map((note) => {
116
+ const isRoot = note?.id === forum;
117
+ const type = classifyNote(note, isRoot);
118
+ const author = authorFromSignatures(note?.signatures);
119
+ const rating = readContent(note?.content, 'rating');
120
+ const confidence = readContent(note?.content, 'confidence');
121
+ const text = truncate(joinSections(note?.content), maxLength);
122
+ return {
123
+ type,
124
+ author,
125
+ rating: rating === undefined || rating === null ? '' : String(rating),
126
+ confidence: confidence === undefined || confidence === null ? '' : String(confidence),
127
+ text,
128
+ };
129
+ });
130
+ },
131
+ });
@@ -0,0 +1,46 @@
1
+ /**
2
+ * OpenReview full-text search.
3
+ */
4
+ import { cli, Strategy } from '@jackwener/opencli/registry';
5
+ import { ArgumentError, EmptyResultError } from '@jackwener/opencli/errors';
6
+ import { noteToRow, openreviewFetch, requireBoundedInt } from './utils.js';
7
+
8
+ cli({
9
+ site: 'openreview',
10
+ name: 'search',
11
+ access: 'read',
12
+ description: 'Search OpenReview papers by free-text query',
13
+ domain: 'openreview.net',
14
+ strategy: Strategy.PUBLIC,
15
+ browser: false,
16
+ args: [
17
+ { name: 'query', positional: true, required: true, help: 'Search keyword (e.g. "diffusion model")' },
18
+ { name: 'limit', type: 'int', default: 25, help: 'Max results (max 50)' },
19
+ ],
20
+ columns: ['rank', 'id', 'title', 'authors', 'venue', 'pdate', 'url'],
21
+ func: async (args) => {
22
+ const term = String(args.query ?? '').trim();
23
+ if (!term) {
24
+ throw new ArgumentError('openreview search query cannot be empty');
25
+ }
26
+ const limit = requireBoundedInt(args.limit, 25, 50);
27
+ const path = `/notes/search?term=${encodeURIComponent(term)}&type=terms&limit=${limit}`;
28
+ const json = await openreviewFetch(path, 'openreview search');
29
+ const notes = Array.isArray(json?.notes) ? json.notes : [];
30
+ if (!notes.length) {
31
+ throw new EmptyResultError('openreview', `No papers found for "${term}". Try a different keyword.`);
32
+ }
33
+ return notes.slice(0, limit).map((note, i) => {
34
+ const row = noteToRow(note);
35
+ return {
36
+ rank: i + 1,
37
+ id: row.id,
38
+ title: row.title,
39
+ authors: row.authors,
40
+ venue: row.venue,
41
+ pdate: row.pdate,
42
+ url: row.url,
43
+ };
44
+ });
45
+ },
46
+ });