@jackwener/opencli 1.6.1 → 1.6.2

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 (384) hide show
  1. package/CONTRIBUTING.md +1 -1
  2. package/README.md +27 -45
  3. package/README.zh-CN.md +32 -34
  4. package/autoresearch/browse-tasks.json +18 -20
  5. package/autoresearch/commands/debug.ts +163 -0
  6. package/autoresearch/commands/fix.ts +145 -0
  7. package/autoresearch/commands/plan.ts +88 -0
  8. package/autoresearch/commands/run.ts +138 -0
  9. package/autoresearch/config.ts +82 -0
  10. package/autoresearch/engine.ts +359 -0
  11. package/autoresearch/eval-all.ts +127 -0
  12. package/autoresearch/eval-browse.ts +1 -1
  13. package/autoresearch/eval-publish.ts +238 -0
  14. package/autoresearch/eval-save.ts +249 -0
  15. package/autoresearch/eval-skill.ts +14 -8
  16. package/autoresearch/eval-v2ex.ts +220 -0
  17. package/autoresearch/eval-zhihu.ts +230 -0
  18. package/autoresearch/logger.ts +69 -0
  19. package/autoresearch/presets/combined-reliability.ts +27 -0
  20. package/autoresearch/presets/index.ts +23 -0
  21. package/autoresearch/presets/operate-reliability.ts +24 -0
  22. package/autoresearch/presets/save-reliability.ts +26 -0
  23. package/autoresearch/presets/skill-quality.ts +20 -0
  24. package/autoresearch/presets/v2ex-reliability.ts +24 -0
  25. package/autoresearch/presets/zhihu-reliability.ts +25 -0
  26. package/autoresearch/publish-tasks.json +345 -0
  27. package/autoresearch/run-save.sh +11 -0
  28. package/autoresearch/save-adapters/xhs-explore-deep.ts +64 -0
  29. package/autoresearch/save-adapters/xhs-note-comments.ts +61 -0
  30. package/autoresearch/save-adapters/xhs-search-full.ts +62 -0
  31. package/autoresearch/save-adapters/zhihu-hot-detail.ts +52 -0
  32. package/autoresearch/save-adapters/zhihu-question-full.ts +57 -0
  33. package/autoresearch/save-adapters/zhihu-search-detail.ts +53 -0
  34. package/autoresearch/save-tasks.json +281 -0
  35. package/autoresearch/v2ex-tasks.json +899 -0
  36. package/autoresearch/zhihu-tasks.json +848 -0
  37. package/dist/browser/base-page.d.ts +4 -2
  38. package/dist/browser/base-page.js +37 -4
  39. package/dist/browser/bridge.js +10 -8
  40. package/dist/browser/cdp.js +2 -6
  41. package/dist/browser/daemon-client.d.ts +11 -1
  42. package/dist/browser/daemon-client.js +3 -0
  43. package/dist/browser/dom-helpers.d.ts +4 -2
  44. package/dist/browser/dom-helpers.js +42 -31
  45. package/dist/browser/dom-snapshot.js +23 -1
  46. package/dist/browser/page.d.ts +7 -2
  47. package/dist/browser/page.js +112 -30
  48. package/dist/browser.test.js +1 -1
  49. package/dist/build-manifest.d.ts +1 -0
  50. package/dist/build-manifest.js +1 -0
  51. package/dist/cli-manifest.json +1135 -184
  52. package/dist/cli.d.ts +2 -0
  53. package/dist/cli.js +48 -7
  54. package/dist/cli.test.d.ts +1 -0
  55. package/dist/cli.test.js +88 -0
  56. package/dist/clis/1688/item.d.ts +70 -0
  57. package/dist/clis/1688/item.js +187 -0
  58. package/dist/clis/1688/item.test.d.ts +1 -0
  59. package/dist/clis/1688/item.test.js +67 -0
  60. package/dist/clis/1688/search.d.ts +56 -0
  61. package/dist/clis/1688/search.js +309 -0
  62. package/dist/clis/1688/search.test.d.ts +1 -0
  63. package/dist/clis/1688/search.test.js +75 -0
  64. package/dist/clis/1688/shared.d.ts +112 -0
  65. package/dist/clis/1688/shared.js +514 -0
  66. package/dist/clis/1688/shared.test.d.ts +1 -0
  67. package/dist/clis/1688/shared.test.js +57 -0
  68. package/dist/clis/1688/store.d.ts +45 -0
  69. package/dist/clis/1688/store.js +226 -0
  70. package/dist/clis/1688/store.test.d.ts +1 -0
  71. package/dist/clis/1688/store.test.js +62 -0
  72. package/dist/clis/amazon/bestsellers.d.ts +0 -20
  73. package/dist/clis/amazon/bestsellers.js +6 -129
  74. package/dist/clis/amazon/bestsellers.test.js +12 -3
  75. package/dist/clis/amazon/movers-shakers.d.ts +1 -0
  76. package/dist/clis/amazon/movers-shakers.js +7 -0
  77. package/dist/clis/amazon/new-releases.d.ts +1 -0
  78. package/dist/clis/amazon/new-releases.js +7 -0
  79. package/dist/clis/amazon/rankings.d.ts +59 -0
  80. package/dist/clis/amazon/rankings.js +226 -0
  81. package/dist/clis/amazon/rankings.test.d.ts +1 -0
  82. package/dist/clis/amazon/rankings.test.js +41 -0
  83. package/dist/clis/amazon/shared.d.ts +11 -0
  84. package/dist/clis/amazon/shared.js +121 -11
  85. package/dist/clis/amazon/shared.test.js +11 -0
  86. package/dist/clis/bilibili/comments.js +2 -2
  87. package/dist/clis/bilibili/comments.test.js +3 -2
  88. package/dist/clis/bilibili/download.js +2 -1
  89. package/dist/clis/bilibili/subtitle.js +4 -3
  90. package/dist/clis/bilibili/subtitle.test.js +2 -1
  91. package/dist/clis/bilibili/utils.d.ts +5 -0
  92. package/dist/clis/bilibili/utils.js +30 -0
  93. package/dist/clis/bilibili/utils.test.d.ts +1 -0
  94. package/dist/clis/bilibili/utils.test.js +17 -0
  95. package/dist/clis/douban/marks.js +1 -1
  96. package/dist/clis/douban/subject.yaml +50 -19
  97. package/dist/clis/doubao/utils.js +32 -12
  98. package/dist/clis/douyin/_shared/browser-fetch.test.js +0 -1
  99. package/dist/clis/douyin/_shared/transcode.test.js +0 -2
  100. package/dist/clis/douyin/draft.test.js +0 -2
  101. package/dist/clis/facebook/search.test.js +0 -2
  102. package/dist/clis/gemini/ask.js +9 -3
  103. package/dist/clis/gemini/ask.test.d.ts +1 -0
  104. package/dist/clis/gemini/ask.test.js +100 -0
  105. package/dist/clis/gemini/reply-state.test.d.ts +1 -0
  106. package/dist/clis/gemini/reply-state.test.js +641 -0
  107. package/dist/clis/gemini/utils.d.ts +44 -1
  108. package/dist/clis/gemini/utils.js +528 -61
  109. package/dist/clis/gemini/utils.test.js +149 -2
  110. package/dist/clis/hupu/detail.d.ts +1 -0
  111. package/dist/clis/hupu/detail.js +72 -0
  112. package/dist/clis/hupu/hot.yaml +43 -0
  113. package/dist/clis/hupu/like.d.ts +1 -0
  114. package/dist/clis/hupu/like.js +75 -0
  115. package/dist/clis/hupu/reply.d.ts +1 -0
  116. package/dist/clis/hupu/reply.js +71 -0
  117. package/dist/clis/hupu/search.d.ts +1 -0
  118. package/dist/clis/hupu/search.js +59 -0
  119. package/dist/clis/hupu/unlike.d.ts +1 -0
  120. package/dist/clis/hupu/unlike.js +75 -0
  121. package/dist/clis/hupu/utils.d.ts +20 -0
  122. package/dist/clis/hupu/utils.js +319 -0
  123. package/dist/clis/instagram/_shared/private-publish.d.ts +138 -0
  124. package/dist/clis/instagram/_shared/private-publish.js +1030 -0
  125. package/dist/clis/instagram/_shared/private-publish.test.d.ts +1 -0
  126. package/dist/clis/instagram/_shared/private-publish.test.js +705 -0
  127. package/dist/clis/instagram/_shared/protocol-capture.d.ts +26 -0
  128. package/dist/clis/instagram/_shared/protocol-capture.js +282 -0
  129. package/dist/clis/instagram/_shared/protocol-capture.test.d.ts +1 -0
  130. package/dist/clis/instagram/_shared/protocol-capture.test.js +114 -0
  131. package/dist/clis/instagram/_shared/runtime-info.d.ts +9 -0
  132. package/dist/clis/instagram/_shared/runtime-info.js +81 -0
  133. package/dist/clis/instagram/note.d.ts +1 -0
  134. package/dist/clis/instagram/note.js +222 -0
  135. package/dist/clis/instagram/note.test.d.ts +1 -0
  136. package/dist/clis/instagram/note.test.js +81 -0
  137. package/dist/clis/instagram/post.d.ts +4 -0
  138. package/dist/clis/instagram/post.js +1496 -0
  139. package/dist/clis/instagram/post.test.d.ts +1 -0
  140. package/dist/clis/instagram/post.test.js +1647 -0
  141. package/dist/clis/instagram/reel.d.ts +1 -0
  142. package/dist/clis/instagram/reel.js +826 -0
  143. package/dist/clis/instagram/reel.test.d.ts +1 -0
  144. package/dist/clis/instagram/reel.test.js +167 -0
  145. package/dist/clis/instagram/story.d.ts +1 -0
  146. package/dist/clis/instagram/story.js +115 -0
  147. package/dist/clis/instagram/story.test.d.ts +1 -0
  148. package/dist/clis/instagram/story.test.js +167 -0
  149. package/dist/clis/sinafinance/stock-rank.d.ts +4 -0
  150. package/dist/clis/sinafinance/stock-rank.js +65 -0
  151. package/dist/clis/substack/utils.test.js +0 -2
  152. package/dist/clis/twitter/post.js +72 -45
  153. package/dist/clis/twitter/post.test.d.ts +1 -0
  154. package/dist/clis/twitter/post.test.js +116 -0
  155. package/dist/clis/twitter/reply.d.ts +12 -0
  156. package/dist/clis/twitter/reply.js +257 -35
  157. package/dist/clis/twitter/reply.test.d.ts +1 -0
  158. package/dist/clis/twitter/reply.test.js +151 -0
  159. package/dist/clis/xianyu/chat.d.ts +7 -0
  160. package/dist/clis/xianyu/chat.js +146 -0
  161. package/dist/clis/xianyu/chat.test.d.ts +1 -0
  162. package/dist/clis/xianyu/chat.test.js +15 -0
  163. package/dist/clis/xianyu/item.d.ts +7 -0
  164. package/dist/clis/xianyu/item.js +152 -0
  165. package/dist/clis/xianyu/item.test.d.ts +1 -0
  166. package/dist/clis/xianyu/item.test.js +56 -0
  167. package/dist/clis/xianyu/search.d.ts +10 -0
  168. package/dist/clis/xianyu/search.js +134 -0
  169. package/dist/clis/xianyu/search.test.d.ts +1 -0
  170. package/dist/clis/xianyu/search.test.js +17 -0
  171. package/dist/clis/xianyu/utils.d.ts +1 -0
  172. package/dist/clis/xianyu/utils.js +8 -0
  173. package/dist/clis/xiaoe/catalog.yaml +129 -0
  174. package/dist/clis/xiaoe/content.yaml +43 -0
  175. package/dist/clis/xiaoe/courses.yaml +73 -0
  176. package/dist/clis/xiaoe/detail.yaml +39 -0
  177. package/dist/clis/xiaoe/play-url.yaml +124 -0
  178. package/dist/clis/xiaohongshu/comments.test.js +0 -2
  179. package/dist/clis/xiaohongshu/creator-note-detail.test.js +0 -2
  180. package/dist/clis/xiaohongshu/creator-notes.test.js +0 -2
  181. package/dist/clis/xiaohongshu/download.test.js +0 -2
  182. package/dist/clis/xiaohongshu/note.test.js +0 -2
  183. package/dist/clis/xiaohongshu/publish.test.js +0 -2
  184. package/dist/clis/xiaohongshu/search.js +29 -20
  185. package/dist/clis/xiaohongshu/search.test.js +56 -48
  186. package/dist/clis/yuanbao/ask.d.ts +21 -0
  187. package/dist/clis/yuanbao/ask.js +427 -0
  188. package/dist/clis/yuanbao/ask.test.d.ts +1 -0
  189. package/dist/clis/yuanbao/ask.test.js +124 -0
  190. package/dist/clis/yuanbao/new.d.ts +1 -0
  191. package/dist/clis/yuanbao/new.js +70 -0
  192. package/dist/clis/yuanbao/new.test.d.ts +1 -0
  193. package/dist/clis/yuanbao/new.test.js +30 -0
  194. package/dist/clis/yuanbao/shared.d.ts +13 -0
  195. package/dist/clis/yuanbao/shared.js +49 -0
  196. package/dist/clis/zhihu/question.js +30 -19
  197. package/dist/clis/zhihu/question.test.js +34 -16
  198. package/dist/commanderAdapter.js +8 -4
  199. package/dist/commanderAdapter.test.js +42 -0
  200. package/dist/completion.js +3 -1
  201. package/dist/completion.test.d.ts +1 -0
  202. package/dist/completion.test.js +23 -0
  203. package/dist/doctor.js +1 -1
  204. package/dist/electron-apps.d.ts +2 -0
  205. package/dist/electron-apps.js +7 -1
  206. package/dist/errors.js +1 -1
  207. package/dist/execution.js +25 -35
  208. package/dist/explore.js +1 -1
  209. package/dist/launcher.d.ts +4 -0
  210. package/dist/launcher.js +64 -8
  211. package/dist/launcher.test.js +88 -7
  212. package/dist/output.d.ts +2 -0
  213. package/dist/output.js +10 -1
  214. package/dist/output.test.d.ts +0 -3
  215. package/dist/output.test.js +59 -92
  216. package/dist/pipeline/executor.test.js +0 -2
  217. package/dist/pipeline/steps/download.test.js +0 -2
  218. package/dist/registry.d.ts +2 -0
  219. package/dist/serialization.d.ts +1 -0
  220. package/dist/serialization.js +1 -0
  221. package/dist/types.d.ts +9 -2
  222. package/docs/.vitepress/config.mts +4 -0
  223. package/docs/adapters/browser/1688.md +52 -0
  224. package/docs/adapters/browser/36kr.md +2 -1
  225. package/docs/adapters/browser/doubao.md +5 -1
  226. package/docs/adapters/browser/hupu.md +53 -0
  227. package/docs/adapters/browser/sinafinance.md +32 -2
  228. package/docs/adapters/browser/weibo.md +6 -1
  229. package/docs/adapters/browser/wikipedia.md +2 -0
  230. package/docs/adapters/browser/xianyu.md +42 -0
  231. package/docs/adapters/browser/xiaoe.md +44 -0
  232. package/docs/adapters/browser/yuanbao.md +64 -0
  233. package/docs/adapters/index.md +14 -5
  234. package/docs/comparison.md +1 -1
  235. package/docs/developer/ai-workflow.md +2 -2
  236. package/docs/developer/contributing.md +1 -1
  237. package/docs/developer/testing.md +2 -0
  238. package/docs/guide/plugins.md +1 -0
  239. package/docs/guide/troubleshooting.md +11 -0
  240. package/docs/superpowers/specs/2026-04-03-v2ex-autoresearch-design.md +41 -0
  241. package/docs/zh/guide/plugins.md +1 -0
  242. package/extension/dist/background.js +1127 -0
  243. package/extension/src/background.test.ts +39 -0
  244. package/extension/src/background.ts +223 -34
  245. package/extension/src/cdp.ts +194 -4
  246. package/extension/src/protocol.ts +22 -1
  247. package/package.json +3 -2
  248. package/scripts/postinstall.js +1 -1
  249. package/skills/opencli-explorer/SKILL.md +1 -1
  250. package/skills/opencli-oneshot/SKILL.md +2 -2
  251. package/skills/opencli-operate/SKILL.md +120 -27
  252. package/skills/opencli-usage/SKILL.md +31 -20
  253. package/skills/opencli-usage/browser.md +114 -16
  254. package/skills/opencli-usage/public-api.md +32 -3
  255. package/skills/smart-search/SKILL.md +156 -0
  256. package/skills/smart-search/references/sources-ai.md +74 -0
  257. package/skills/smart-search/references/sources-info.md +43 -0
  258. package/skills/smart-search/references/sources-media.md +50 -0
  259. package/skills/smart-search/references/sources-other.md +42 -0
  260. package/skills/smart-search/references/sources-shopping.md +31 -0
  261. package/skills/smart-search/references/sources-social.md +51 -0
  262. package/skills/smart-search/references/sources-tech.md +42 -0
  263. package/skills/smart-search/references/sources-travel.md +20 -0
  264. package/src/browser/base-page.ts +41 -6
  265. package/src/browser/bridge.ts +11 -8
  266. package/src/browser/cdp.ts +1 -8
  267. package/src/browser/daemon-client.ts +11 -1
  268. package/src/browser/dom-helpers.ts +43 -31
  269. package/src/browser/dom-snapshot.ts +23 -1
  270. package/src/browser/page.ts +115 -31
  271. package/src/browser.test.ts +1 -1
  272. package/src/build-manifest.ts +2 -0
  273. package/src/cli.test.ts +133 -0
  274. package/src/cli.ts +73 -11
  275. package/src/clis/1688/item.test.ts +69 -0
  276. package/src/clis/1688/item.ts +282 -0
  277. package/src/clis/1688/search.test.ts +81 -0
  278. package/src/clis/1688/search.ts +402 -0
  279. package/src/clis/1688/shared.test.ts +75 -0
  280. package/src/clis/1688/shared.ts +623 -0
  281. package/src/clis/1688/store.test.ts +69 -0
  282. package/src/clis/1688/store.ts +300 -0
  283. package/src/clis/amazon/bestsellers.test.ts +12 -3
  284. package/src/clis/amazon/bestsellers.ts +6 -178
  285. package/src/clis/amazon/movers-shakers.ts +8 -0
  286. package/src/clis/amazon/new-releases.ts +8 -0
  287. package/src/clis/amazon/rankings.test.ts +47 -0
  288. package/src/clis/amazon/rankings.ts +312 -0
  289. package/src/clis/amazon/shared.test.ts +16 -0
  290. package/src/clis/amazon/shared.ts +134 -12
  291. package/src/clis/bilibili/comments.test.ts +4 -3
  292. package/src/clis/bilibili/comments.ts +2 -2
  293. package/src/clis/bilibili/download.ts +2 -1
  294. package/src/clis/bilibili/subtitle.test.ts +2 -1
  295. package/src/clis/bilibili/subtitle.ts +4 -3
  296. package/src/clis/bilibili/utils.test.ts +21 -0
  297. package/src/clis/bilibili/utils.ts +27 -0
  298. package/src/clis/douban/marks.ts +1 -1
  299. package/src/clis/douban/subject.yaml +50 -19
  300. package/src/clis/doubao/utils.ts +32 -12
  301. package/src/clis/douyin/_shared/browser-fetch.test.ts +0 -1
  302. package/src/clis/douyin/_shared/transcode.test.ts +0 -2
  303. package/src/clis/douyin/draft.test.ts +0 -2
  304. package/src/clis/facebook/search.test.ts +0 -2
  305. package/src/clis/gemini/ask.test.ts +116 -0
  306. package/src/clis/gemini/ask.ts +10 -3
  307. package/src/clis/gemini/reply-state.test.ts +708 -0
  308. package/src/clis/gemini/utils.test.ts +184 -2
  309. package/src/clis/gemini/utils.ts +588 -60
  310. package/src/clis/hupu/detail.ts +126 -0
  311. package/src/clis/hupu/hot.yaml +43 -0
  312. package/src/clis/hupu/like.ts +76 -0
  313. package/src/clis/hupu/reply.ts +76 -0
  314. package/src/clis/hupu/search.ts +95 -0
  315. package/src/clis/hupu/unlike.ts +76 -0
  316. package/src/clis/hupu/utils.ts +381 -0
  317. package/src/clis/instagram/_shared/private-publish.test.ts +827 -0
  318. package/src/clis/instagram/_shared/private-publish.ts +1303 -0
  319. package/src/clis/instagram/_shared/protocol-capture.test.ts +148 -0
  320. package/src/clis/instagram/_shared/protocol-capture.ts +321 -0
  321. package/src/clis/instagram/_shared/runtime-info.ts +91 -0
  322. package/src/clis/instagram/note.test.ts +96 -0
  323. package/src/clis/instagram/note.ts +254 -0
  324. package/src/clis/instagram/post.test.ts +1716 -0
  325. package/src/clis/instagram/post.ts +1620 -0
  326. package/src/clis/instagram/reel.test.ts +191 -0
  327. package/src/clis/instagram/reel.ts +886 -0
  328. package/src/clis/instagram/story.test.ts +191 -0
  329. package/src/clis/instagram/story.ts +151 -0
  330. package/src/clis/sinafinance/stock-rank.ts +68 -0
  331. package/src/clis/substack/utils.test.ts +0 -2
  332. package/src/clis/twitter/post.test.ts +157 -0
  333. package/src/clis/twitter/post.ts +82 -48
  334. package/src/clis/twitter/reply.test.ts +177 -0
  335. package/src/clis/twitter/reply.ts +285 -39
  336. package/src/clis/xianyu/chat.test.ts +20 -0
  337. package/src/clis/xianyu/chat.ts +175 -0
  338. package/src/clis/xianyu/item.test.ts +67 -0
  339. package/src/clis/xianyu/item.ts +172 -0
  340. package/src/clis/xianyu/search.test.ts +22 -0
  341. package/src/clis/xianyu/search.ts +151 -0
  342. package/src/clis/xianyu/utils.ts +9 -0
  343. package/src/clis/xiaoe/catalog.yaml +129 -0
  344. package/src/clis/xiaoe/content.yaml +43 -0
  345. package/src/clis/xiaoe/courses.yaml +73 -0
  346. package/src/clis/xiaoe/detail.yaml +39 -0
  347. package/src/clis/xiaoe/play-url.yaml +124 -0
  348. package/src/clis/xiaohongshu/comments.test.ts +0 -2
  349. package/src/clis/xiaohongshu/creator-note-detail.test.ts +0 -2
  350. package/src/clis/xiaohongshu/creator-notes.test.ts +0 -2
  351. package/src/clis/xiaohongshu/download.test.ts +0 -2
  352. package/src/clis/xiaohongshu/note.test.ts +0 -2
  353. package/src/clis/xiaohongshu/publish.test.ts +0 -2
  354. package/src/clis/xiaohongshu/search.test.ts +59 -48
  355. package/src/clis/xiaohongshu/search.ts +31 -21
  356. package/src/clis/yuanbao/ask.test.ts +156 -0
  357. package/src/clis/yuanbao/ask.ts +522 -0
  358. package/src/clis/yuanbao/new.test.ts +36 -0
  359. package/src/clis/yuanbao/new.ts +81 -0
  360. package/src/clis/yuanbao/shared.ts +57 -0
  361. package/src/clis/zhihu/question.test.ts +42 -17
  362. package/src/clis/zhihu/question.ts +31 -26
  363. package/src/commanderAdapter.test.ts +51 -0
  364. package/src/commanderAdapter.ts +8 -4
  365. package/src/completion.test.ts +30 -0
  366. package/src/completion.ts +3 -1
  367. package/src/doctor.ts +1 -1
  368. package/src/electron-apps.ts +9 -1
  369. package/src/errors.ts +1 -1
  370. package/src/execution.ts +26 -30
  371. package/src/explore.ts +1 -1
  372. package/src/launcher.test.ts +121 -7
  373. package/src/launcher.ts +87 -9
  374. package/src/output.test.ts +50 -90
  375. package/src/output.ts +10 -1
  376. package/src/pipeline/executor.test.ts +0 -2
  377. package/src/pipeline/steps/download.test.ts +0 -2
  378. package/src/registry.ts +2 -0
  379. package/src/serialization.ts +2 -0
  380. package/src/types.ts +9 -2
  381. package/tests/e2e/browser-auth.test.ts +9 -0
  382. package/CLI-EXPLORER.md +0 -724
  383. package/CLI-ONESHOT.md +0 -216
  384. package/SKILL.md +0 -59
@@ -0,0 +1,151 @@
1
+ import * as fs from 'node:fs';
2
+ import * as os from 'node:os';
3
+ import * as path from 'node:path';
4
+ import { describe, expect, it, vi } from 'vitest';
5
+ import { getRegistry } from '../../registry.js';
6
+ import { __test__ } from './reply.js';
7
+ function createPageMock(evaluateResults, overrides = {}) {
8
+ const evaluate = vi.fn();
9
+ for (const result of evaluateResults) {
10
+ evaluate.mockResolvedValueOnce(result);
11
+ }
12
+ return {
13
+ goto: vi.fn().mockResolvedValue(undefined),
14
+ evaluate,
15
+ snapshot: vi.fn().mockResolvedValue(undefined),
16
+ click: vi.fn().mockResolvedValue(undefined),
17
+ typeText: vi.fn().mockResolvedValue(undefined),
18
+ pressKey: vi.fn().mockResolvedValue(undefined),
19
+ scrollTo: vi.fn().mockResolvedValue(undefined),
20
+ getFormState: vi.fn().mockResolvedValue({ forms: [], orphanFields: [] }),
21
+ wait: vi.fn().mockResolvedValue(undefined),
22
+ tabs: vi.fn().mockResolvedValue([]),
23
+ selectTab: vi.fn().mockResolvedValue(undefined),
24
+ networkRequests: vi.fn().mockResolvedValue([]),
25
+ consoleMessages: vi.fn().mockResolvedValue([]),
26
+ scroll: vi.fn().mockResolvedValue(undefined),
27
+ autoScroll: vi.fn().mockResolvedValue(undefined),
28
+ installInterceptor: vi.fn().mockResolvedValue(undefined),
29
+ getInterceptedRequests: vi.fn().mockResolvedValue([]),
30
+ getCookies: vi.fn().mockResolvedValue([]),
31
+ screenshot: vi.fn().mockResolvedValue(''),
32
+ waitForCapture: vi.fn().mockResolvedValue(undefined),
33
+ ...overrides,
34
+ };
35
+ }
36
+ describe('twitter reply command', () => {
37
+ it('keeps the text-only reply flow working', async () => {
38
+ const cmd = getRegistry().get('twitter/reply');
39
+ expect(cmd?.func).toBeTypeOf('function');
40
+ const page = createPageMock([
41
+ { ok: true, message: 'Reply posted successfully.' },
42
+ ]);
43
+ const result = await cmd.func(page, {
44
+ url: 'https://x.com/_kop6/status/2040254679301718161?s=20',
45
+ text: 'text-only reply',
46
+ });
47
+ expect(page.goto).toHaveBeenCalledWith('https://x.com/_kop6/status/2040254679301718161?s=20');
48
+ expect(page.wait).toHaveBeenCalledWith({ selector: '[data-testid="primaryColumn"]' });
49
+ expect(result).toEqual([
50
+ {
51
+ status: 'success',
52
+ message: 'Reply posted successfully.',
53
+ text: 'text-only reply',
54
+ },
55
+ ]);
56
+ });
57
+ it('uploads a local image through the dedicated reply composer when --image is provided', async () => {
58
+ const cmd = getRegistry().get('twitter/reply');
59
+ expect(cmd?.func).toBeTypeOf('function');
60
+ const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'opencli-twitter-reply-'));
61
+ const imagePath = path.join(tempDir, 'qr.png');
62
+ fs.writeFileSync(imagePath, Buffer.from([0x89, 0x50, 0x4e, 0x47]));
63
+ const setFileInput = vi.fn().mockResolvedValue(undefined);
64
+ const page = createPageMock([
65
+ { ok: true, previewCount: 1 },
66
+ { ok: true, message: 'Reply posted successfully.' },
67
+ ], {
68
+ setFileInput,
69
+ });
70
+ const result = await cmd.func(page, {
71
+ url: 'https://x.com/_kop6/status/2040254679301718161?s=20',
72
+ text: 'reply with image',
73
+ image: imagePath,
74
+ });
75
+ expect(page.goto).toHaveBeenCalledWith('https://x.com/compose/post?in_reply_to=2040254679301718161');
76
+ expect(page.wait).toHaveBeenNthCalledWith(1, { selector: '[data-testid="tweetTextarea_0"]' });
77
+ expect(page.wait).toHaveBeenNthCalledWith(2, { selector: 'input[type="file"][data-testid="fileInput"]', timeout: 20 });
78
+ expect(setFileInput).toHaveBeenCalledWith([imagePath], 'input[type="file"][data-testid="fileInput"]');
79
+ expect(result).toEqual([
80
+ {
81
+ status: 'success',
82
+ message: 'Reply posted successfully.',
83
+ text: 'reply with image',
84
+ image: imagePath,
85
+ },
86
+ ]);
87
+ });
88
+ it('downloads a remote image before uploading when --image-url is provided', async () => {
89
+ const cmd = getRegistry().get('twitter/reply');
90
+ expect(cmd?.func).toBeTypeOf('function');
91
+ const fetchMock = vi.fn().mockResolvedValue({
92
+ ok: true,
93
+ headers: {
94
+ get: vi.fn().mockReturnValue('image/png'),
95
+ },
96
+ arrayBuffer: vi.fn().mockResolvedValue(Uint8Array.from([0x89, 0x50, 0x4e, 0x47]).buffer),
97
+ });
98
+ vi.stubGlobal('fetch', fetchMock);
99
+ const setFileInput = vi.fn().mockResolvedValue(undefined);
100
+ const page = createPageMock([
101
+ { ok: true, previewCount: 1 },
102
+ { ok: true, message: 'Reply posted successfully.' },
103
+ ], {
104
+ setFileInput,
105
+ });
106
+ const result = await cmd.func(page, {
107
+ url: 'https://x.com/_kop6/status/2040254679301718161?s=20',
108
+ text: 'reply with remote image',
109
+ 'image-url': 'https://example.com/qr',
110
+ });
111
+ expect(fetchMock).toHaveBeenCalledWith('https://example.com/qr');
112
+ expect(setFileInput).toHaveBeenCalledTimes(1);
113
+ const uploadedPath = setFileInput.mock.calls[0][0][0];
114
+ expect(uploadedPath).toMatch(/opencli-twitter-reply-.*\/image\.png$/);
115
+ expect(fs.existsSync(uploadedPath)).toBe(false);
116
+ expect(result).toEqual([
117
+ {
118
+ status: 'success',
119
+ message: 'Reply posted successfully.',
120
+ text: 'reply with remote image',
121
+ 'image-url': 'https://example.com/qr',
122
+ },
123
+ ]);
124
+ vi.unstubAllGlobals();
125
+ });
126
+ it('rejects invalid image paths early', async () => {
127
+ await expect(() => __test__.resolveImagePath('/tmp/does-not-exist.png'))
128
+ .toThrow('Image file not found');
129
+ });
130
+ it('rejects using --image and --image-url together', async () => {
131
+ const cmd = getRegistry().get('twitter/reply');
132
+ expect(cmd?.func).toBeTypeOf('function');
133
+ const page = createPageMock([]);
134
+ await expect(cmd.func(page, {
135
+ url: 'https://x.com/_kop6/status/2040254679301718161?s=20',
136
+ text: 'nope',
137
+ image: '/tmp/a.png',
138
+ 'image-url': 'https://example.com/a.png',
139
+ })).rejects.toThrow('Use either --image or --image-url, not both.');
140
+ });
141
+ it('extracts tweet ids from both user and i/status URLs', () => {
142
+ expect(__test__.extractTweetId('https://x.com/_kop6/status/2040254679301718161?s=20')).toBe('2040254679301718161');
143
+ expect(__test__.extractTweetId('https://x.com/i/status/2040318731105313143')).toBe('2040318731105313143');
144
+ expect(__test__.buildReplyComposerUrl('https://x.com/i/status/2040318731105313143'))
145
+ .toBe('https://x.com/compose/post?in_reply_to=2040318731105313143');
146
+ });
147
+ it('prefers content-type when resolving remote image extensions', () => {
148
+ expect(__test__.resolveImageExtension('https://example.com/no-ext', 'image/webp')).toBe('.webp');
149
+ expect(__test__.resolveImageExtension('https://example.com/a.jpeg?x=1', null)).toBe('.jpeg');
150
+ });
151
+ });
@@ -0,0 +1,7 @@
1
+ import { normalizeNumericId } from './utils.js';
2
+ declare function buildChatUrl(itemId: string, peerUserId: string): string;
3
+ export declare const __test__: {
4
+ normalizeNumericId: typeof normalizeNumericId;
5
+ buildChatUrl: typeof buildChatUrl;
6
+ };
7
+ export {};
@@ -0,0 +1,146 @@
1
+ import { AuthRequiredError, SelectorError } from '../../errors.js';
2
+ import { cli, Strategy } from '../../registry.js';
3
+ import { normalizeNumericId } from './utils.js';
4
+ function buildChatUrl(itemId, peerUserId) {
5
+ return `https://www.goofish.com/im?itemId=${encodeURIComponent(itemId)}&peerUserId=${encodeURIComponent(peerUserId)}`;
6
+ }
7
+ function buildExtractChatStateEvaluate() {
8
+ return `
9
+ (() => {
10
+ const clean = (value) => (value || '').replace(/\\s+/g, ' ').trim();
11
+ const bodyText = document.body?.innerText || '';
12
+ const requiresAuth = /请先登录|登录后/.test(bodyText);
13
+
14
+ const textarea = document.querySelector('textarea');
15
+ const sendButton = Array.from(document.querySelectorAll('button'))
16
+ .find((btn) => clean(btn.textContent || '') === '发送');
17
+ const topbar = document.querySelector('[class*="message-topbar"]');
18
+ const itemCard = Array.from(document.querySelectorAll('a[href*="/item?id="]'))
19
+ .find((el) => el.closest('main'));
20
+ const itemTitleNode =
21
+ document.querySelector('[class*="container"] [class*="title"]')
22
+ || document.querySelector('[class*="item-main-info"] [class*="desc"]')
23
+ || document.querySelector('[class*="headSkuInfo"]')
24
+ || itemCard?.querySelector('[class*="title"]')
25
+ || itemCard?.previousElementSibling?.querySelector?.('[class*="title"]');
26
+
27
+ const messageRoot = document.querySelector('#message-list-scrollable');
28
+ const visibleMessages = Array.from(
29
+ (messageRoot || document).querySelectorAll('[class*="message"], [class*="msg"], [class*="bubble"]')
30
+ ).map((el) => clean(el.textContent || ''))
31
+ .filter(Boolean)
32
+ .filter((text) => !['发送', '闲鱼号', '立即购买'].includes(text))
33
+ .filter((text) => !/^消息\\d*\\+?$/.test(text))
34
+ .slice(-20);
35
+
36
+ return {
37
+ requiresAuth,
38
+ title: clean(document.title || ''),
39
+ peer_name: clean(topbar?.querySelector('[class*="text1"]')?.textContent || ''),
40
+ peer_masked_id: clean(topbar?.querySelector('[class*="text2"]')?.textContent || '').replace(/^\\(|\\)$/g, ''),
41
+ item_title: clean(itemTitleNode?.textContent || ''),
42
+ item_url: itemCard?.href || '',
43
+ price: clean(itemCard?.querySelector('[class*="money"]')?.textContent || ''),
44
+ location: clean(itemCard?.querySelector('[class*="delivery"] + [class*="delivery"], [class*="delivery"]:last-child')?.textContent || ''),
45
+ can_input: Boolean(textarea && !textarea.disabled),
46
+ can_send: Boolean(sendButton),
47
+ visible_messages: visibleMessages,
48
+ };
49
+ })()
50
+ `;
51
+ }
52
+ function buildSendMessageEvaluate(text) {
53
+ return `
54
+ (() => {
55
+ const clean = (value) => (value || '').replace(/\\s+/g, ' ').trim();
56
+ const textarea = document.querySelector('textarea');
57
+ if (!textarea || textarea.disabled) {
58
+ return { ok: false, reason: 'input-not-found' };
59
+ }
60
+
61
+ const setter = Object.getOwnPropertyDescriptor(window.HTMLTextAreaElement.prototype, 'value')?.set;
62
+ if (!setter) {
63
+ return { ok: false, reason: 'textarea-setter-not-found' };
64
+ }
65
+
66
+ textarea.focus();
67
+ setter.call(textarea, ${JSON.stringify(text)});
68
+ textarea.dispatchEvent(new Event('input', { bubbles: true }));
69
+ textarea.dispatchEvent(new Event('change', { bubbles: true }));
70
+
71
+ const sendButton = Array.from(document.querySelectorAll('button'))
72
+ .find((btn) => clean(btn.textContent || '') === '发送');
73
+ if (!sendButton) {
74
+ return { ok: false, reason: 'send-button-not-found' };
75
+ }
76
+
77
+ sendButton.click();
78
+ return { ok: true };
79
+ })()
80
+ `;
81
+ }
82
+ cli({
83
+ site: 'xianyu',
84
+ name: 'chat',
85
+ description: '打开闲鱼聊一聊会话,并可选发送消息',
86
+ domain: 'www.goofish.com',
87
+ strategy: Strategy.COOKIE,
88
+ navigateBefore: false,
89
+ browser: true,
90
+ args: [
91
+ { name: 'item_id', required: true, positional: true, help: '闲鱼商品 item_id' },
92
+ { name: 'user_id', required: true, positional: true, help: '聊一聊对方的 user_id / peerUserId' },
93
+ { name: 'text', help: 'Message to send after opening the chat' },
94
+ ],
95
+ columns: ['status', 'peer_name', 'item_title', 'price', 'location', 'message'],
96
+ func: async (page, kwargs) => {
97
+ const itemId = normalizeNumericId(kwargs.item_id, 'item_id', '1038951278192');
98
+ const userId = normalizeNumericId(kwargs.user_id, 'user_id', '3650092411');
99
+ const url = buildChatUrl(itemId, userId);
100
+ const text = String(kwargs.text || '').trim();
101
+ await page.goto(url);
102
+ await page.wait(2);
103
+ const state = await page.evaluate(buildExtractChatStateEvaluate());
104
+ if (state?.requiresAuth) {
105
+ throw new AuthRequiredError('www.goofish.com', 'Xianyu chat requires a logged-in browser session');
106
+ }
107
+ if (!state?.can_input) {
108
+ throw new SelectorError('闲鱼聊天输入框', '未找到可用的聊天输入框,请确认该会话页已正确加载');
109
+ }
110
+ if (!text) {
111
+ return [{
112
+ status: 'ready',
113
+ peer_name: state.peer_name || '',
114
+ item_title: state.item_title || '',
115
+ price: state.price || '',
116
+ location: state.location || '',
117
+ message: (state.visible_messages || []).slice(-1)[0] || '',
118
+ peer_user_id: userId,
119
+ item_id: itemId,
120
+ url,
121
+ item_url: state.item_url || '',
122
+ }];
123
+ }
124
+ const sent = await page.evaluate(buildSendMessageEvaluate(text));
125
+ if (!sent?.ok) {
126
+ throw new SelectorError('闲鱼发送按钮', `消息发送失败:${sent?.reason || 'unknown-reason'}`);
127
+ }
128
+ await page.wait(1);
129
+ return [{
130
+ status: 'sent',
131
+ peer_name: state.peer_name || '',
132
+ item_title: state.item_title || '',
133
+ price: state.price || '',
134
+ location: state.location || '',
135
+ message: text,
136
+ peer_user_id: userId,
137
+ item_id: itemId,
138
+ url,
139
+ item_url: state.item_url || '',
140
+ }];
141
+ },
142
+ });
143
+ export const __test__ = {
144
+ normalizeNumericId,
145
+ buildChatUrl,
146
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,15 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { __test__ } from './chat.js';
3
+ describe('xianyu chat helpers', () => {
4
+ it('builds goofish im urls from ids', () => {
5
+ expect(__test__.buildChatUrl('1038951278192', '3650092411')).toBe('https://www.goofish.com/im?itemId=1038951278192&peerUserId=3650092411');
6
+ });
7
+ it('normalizes numeric ids', () => {
8
+ expect(__test__.normalizeNumericId('1038951278192', 'item_id', '1038951278192')).toBe('1038951278192');
9
+ expect(__test__.normalizeNumericId(3650092411, 'user_id', '3650092411')).toBe('3650092411');
10
+ });
11
+ it('rejects non-numeric ids', () => {
12
+ expect(() => __test__.normalizeNumericId('abc', 'item_id', '1038951278192')).toThrow();
13
+ expect(() => __test__.normalizeNumericId('3650092411x', 'user_id', '3650092411')).toThrow();
14
+ });
15
+ });
@@ -0,0 +1,7 @@
1
+ import { normalizeNumericId } from './utils.js';
2
+ declare function buildItemUrl(itemId: string): string;
3
+ export declare const __test__: {
4
+ normalizeNumericId: typeof normalizeNumericId;
5
+ buildItemUrl: typeof buildItemUrl;
6
+ };
7
+ export {};
@@ -0,0 +1,152 @@
1
+ import { AuthRequiredError, EmptyResultError, SelectorError } from '../../errors.js';
2
+ import { cli, Strategy } from '../../registry.js';
3
+ import { normalizeNumericId } from './utils.js';
4
+ function buildItemUrl(itemId) {
5
+ return `https://www.goofish.com/item?id=${encodeURIComponent(itemId)}`;
6
+ }
7
+ function buildFetchItemEvaluate(itemId) {
8
+ return `
9
+ (async () => {
10
+ const clean = (value) => String(value ?? '').replace(/\\s+/g, ' ').trim();
11
+ const extractRetCode = (ret) => {
12
+ const first = Array.isArray(ret) ? ret[0] : '';
13
+ return clean(first).split('::')[0] || '';
14
+ };
15
+
16
+ const waitFor = async (predicate, timeoutMs = 5000) => {
17
+ const start = Date.now();
18
+ while (Date.now() - start < timeoutMs) {
19
+ if (predicate()) return true;
20
+ await new Promise((r) => setTimeout(r, 150));
21
+ }
22
+ return false;
23
+ };
24
+
25
+ const bodyText = document.body?.innerText || '';
26
+ if (/请先登录|登录后/.test(bodyText)) {
27
+ return { error: 'auth-required' };
28
+ }
29
+
30
+ if (/验证码|安全验证|异常访问/.test(bodyText)) {
31
+ return { error: 'blocked' };
32
+ }
33
+
34
+ await waitFor(() => window.lib?.mtop?.request);
35
+ if (!window.lib || !window.lib.mtop || typeof window.lib.mtop.request !== 'function') {
36
+ return { error: 'mtop-not-ready' };
37
+ }
38
+
39
+ let response;
40
+ try {
41
+ response = await window.lib.mtop.request({
42
+ api: 'mtop.taobao.idle.pc.detail',
43
+ data: { itemId: ${JSON.stringify(itemId)} },
44
+ type: 'POST',
45
+ v: '1.0',
46
+ dataType: 'json',
47
+ needLogin: false,
48
+ needLoginPC: false,
49
+ sessionOption: 'AutoLoginOnly',
50
+ ecode: 0,
51
+ });
52
+ } catch (error) {
53
+ const ret = error?.ret || [];
54
+ return {
55
+ error: 'mtop-request-failed',
56
+ error_code: extractRetCode(ret),
57
+ error_message: clean(Array.isArray(ret) ? ret.join(' | ') : error?.message || error),
58
+ };
59
+ }
60
+
61
+ const retCode = extractRetCode(response?.ret || []);
62
+ if (retCode && retCode !== 'SUCCESS') {
63
+ return {
64
+ error: 'mtop-response-error',
65
+ error_code: retCode,
66
+ error_message: clean((response?.ret || []).join(' | ')),
67
+ };
68
+ }
69
+
70
+ const data = response?.data || {};
71
+ const item = data.itemDO || {};
72
+ const seller = data.sellerDO || {};
73
+ const labels = Array.isArray(item.itemLabelExtList) ? item.itemLabelExtList : [];
74
+ const findLabel = (name) => labels.find((label) => clean(label.propertyText) === name)?.text || '';
75
+ const images = Array.isArray(item.imageInfos)
76
+ ? item.imageInfos.map((entry) => entry?.url).filter(Boolean)
77
+ : [];
78
+
79
+ return {
80
+ item_id: clean(item.itemId || ${JSON.stringify(itemId)}),
81
+ title: clean(item.title || ''),
82
+ description: clean(item.desc || ''),
83
+ price: clean('¥' + (item.soldPrice || item.defaultPrice || '')).replace(/^¥\\s*$/, ''),
84
+ original_price: clean(item.originalPrice || ''),
85
+ want_count: String(item.wantCnt ?? ''),
86
+ collect_count: String(item.collectCnt ?? ''),
87
+ browse_count: String(item.browseCnt ?? ''),
88
+ status: clean(item.itemStatusStr || ''),
89
+ condition: clean(findLabel('成色')),
90
+ brand: clean(findLabel('品牌')),
91
+ category: clean(findLabel('分类')),
92
+ location: clean(seller.publishCity || seller.city || ''),
93
+ seller_name: clean(seller.nick || seller.uniqueName || ''),
94
+ seller_id: String(seller.sellerId || ''),
95
+ seller_score: clean(seller.xianyuSummary || ''),
96
+ reply_ratio_24h: clean(seller.replyRatio24h || ''),
97
+ reply_interval: clean(seller.replyInterval || ''),
98
+ item_url: ${JSON.stringify(buildItemUrl(itemId))},
99
+ seller_url: seller.sellerId ? 'https://www.goofish.com/personal?userId=' + seller.sellerId : '',
100
+ image_count: String(images.length),
101
+ image_urls: images,
102
+ };
103
+ })()
104
+ `;
105
+ }
106
+ cli({
107
+ site: 'xianyu',
108
+ name: 'item',
109
+ description: '查看闲鱼商品详情',
110
+ domain: 'www.goofish.com',
111
+ strategy: Strategy.COOKIE,
112
+ navigateBefore: false,
113
+ browser: true,
114
+ args: [
115
+ { name: 'item_id', required: true, positional: true, help: '闲鱼商品 item_id' },
116
+ ],
117
+ columns: ['item_id', 'title', 'price', 'condition', 'brand', 'location', 'seller_name', 'want_count'],
118
+ func: async (page, kwargs) => {
119
+ const itemId = normalizeNumericId(kwargs.item_id, 'item_id', '1040754408976');
120
+ await page.goto(buildItemUrl(itemId));
121
+ await page.wait(2);
122
+ const result = await page.evaluate(buildFetchItemEvaluate(itemId));
123
+ if (result?.error === 'auth-required') {
124
+ throw new AuthRequiredError('www.goofish.com', 'Xianyu item detail requires a logged-in browser session');
125
+ }
126
+ if (result?.error === 'blocked') {
127
+ throw new EmptyResultError('xianyu item', 'Xianyu item detail is blocked by verification or risk control');
128
+ }
129
+ if (result?.error === 'mtop-not-ready') {
130
+ throw new SelectorError('window.lib.mtop', '闲鱼页面未完成初始化,无法调用商品详情接口');
131
+ }
132
+ if (!result || typeof result !== 'object') {
133
+ throw new EmptyResultError('xianyu item', '闲鱼商品详情接口未返回有效数据');
134
+ }
135
+ const errorCode = String(result.error_code || '');
136
+ const errorMessage = String(result.error_message || '');
137
+ if (/FAIL_SYS_SESSION_EXPIRED|SESSION_EXPIRED|FAIL_SYS/.test(errorCode) || /FAIL_SYS_SESSION_EXPIRED|SESSION_EXPIRED/.test(errorMessage)) {
138
+ throw new AuthRequiredError('www.goofish.com', 'Xianyu item detail requires a logged-in browser session');
139
+ }
140
+ if (result.error) {
141
+ throw new EmptyResultError('xianyu item', errorMessage || `Xianyu item detail request failed: ${result.error}`);
142
+ }
143
+ if (!String(result.title || '').trim()) {
144
+ throw new EmptyResultError('xianyu item', 'No item detail was returned for the specified item_id');
145
+ }
146
+ return [result];
147
+ },
148
+ });
149
+ export const __test__ = {
150
+ normalizeNumericId,
151
+ buildItemUrl,
152
+ };
@@ -0,0 +1 @@
1
+ import './item.js';
@@ -0,0 +1,56 @@
1
+ import { describe, expect, it, vi } from 'vitest';
2
+ import { AuthRequiredError, EmptyResultError, SelectorError } from '../../errors.js';
3
+ import { getRegistry } from '../../registry.js';
4
+ import { __test__ } from './item.js';
5
+ import './item.js';
6
+ function createPageMock(evaluateResult) {
7
+ return {
8
+ goto: vi.fn().mockResolvedValue(undefined),
9
+ evaluate: vi.fn().mockResolvedValue(evaluateResult),
10
+ snapshot: vi.fn().mockResolvedValue(undefined),
11
+ click: vi.fn().mockResolvedValue(undefined),
12
+ typeText: vi.fn().mockResolvedValue(undefined),
13
+ pressKey: vi.fn().mockResolvedValue(undefined),
14
+ scrollTo: vi.fn().mockResolvedValue(undefined),
15
+ getFormState: vi.fn().mockResolvedValue({ forms: [], orphanFields: [] }),
16
+ wait: vi.fn().mockResolvedValue(undefined),
17
+ tabs: vi.fn().mockResolvedValue([]),
18
+ selectTab: vi.fn().mockResolvedValue(undefined),
19
+ networkRequests: vi.fn().mockResolvedValue([]),
20
+ consoleMessages: vi.fn().mockResolvedValue([]),
21
+ scroll: vi.fn().mockResolvedValue(undefined),
22
+ autoScroll: vi.fn().mockResolvedValue(undefined),
23
+ installInterceptor: vi.fn().mockResolvedValue(undefined),
24
+ getInterceptedRequests: vi.fn().mockResolvedValue([]),
25
+ getCookies: vi.fn().mockResolvedValue([]),
26
+ screenshot: vi.fn().mockResolvedValue(''),
27
+ waitForCapture: vi.fn().mockResolvedValue(undefined),
28
+ };
29
+ }
30
+ describe('xianyu item helpers', () => {
31
+ it('normalizes numeric item ids', () => {
32
+ expect(__test__.normalizeNumericId('1040754408976', 'item_id', '1040754408976')).toBe('1040754408976');
33
+ expect(__test__.normalizeNumericId(1040754408976, 'item_id', '1040754408976')).toBe('1040754408976');
34
+ });
35
+ it('builds item urls', () => {
36
+ expect(__test__.buildItemUrl('1040754408976')).toBe('https://www.goofish.com/item?id=1040754408976');
37
+ });
38
+ it('rejects invalid item ids', () => {
39
+ expect(() => __test__.normalizeNumericId('abc', 'item_id', '1040754408976')).toThrow();
40
+ });
41
+ });
42
+ describe('xianyu item command', () => {
43
+ const command = getRegistry().get('xianyu/item');
44
+ it('throws AuthRequiredError on login wall before mtop is available', async () => {
45
+ const page = createPageMock({ error: 'auth-required' });
46
+ await expect(command.func(page, { item_id: '1040754408976' })).rejects.toBeInstanceOf(AuthRequiredError);
47
+ });
48
+ it('throws EmptyResultError on verification or risk-control pages', async () => {
49
+ const page = createPageMock({ error: 'blocked' });
50
+ await expect(command.func(page, { item_id: '1040754408976' })).rejects.toBeInstanceOf(EmptyResultError);
51
+ });
52
+ it('keeps SelectorError for true mtop initialization failures', async () => {
53
+ const page = createPageMock({ error: 'mtop-not-ready' });
54
+ await expect(command.func(page, { item_id: '1040754408976' })).rejects.toBeInstanceOf(SelectorError);
55
+ });
56
+ });
@@ -0,0 +1,10 @@
1
+ declare function normalizeLimit(value: unknown): number;
2
+ declare function buildSearchUrl(query: string): string;
3
+ declare function itemIdFromUrl(url: string): string;
4
+ export declare const __test__: {
5
+ MAX_LIMIT: number;
6
+ normalizeLimit: typeof normalizeLimit;
7
+ buildSearchUrl: typeof buildSearchUrl;
8
+ itemIdFromUrl: typeof itemIdFromUrl;
9
+ };
10
+ export {};