@jackwener/opencli 1.6.0 → 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 (390) hide show
  1. package/CHANGELOG.md +8 -0
  2. package/CONTRIBUTING.md +1 -1
  3. package/README.md +27 -45
  4. package/README.zh-CN.md +32 -34
  5. package/autoresearch/browse-tasks.json +18 -20
  6. package/autoresearch/commands/debug.ts +163 -0
  7. package/autoresearch/commands/fix.ts +145 -0
  8. package/autoresearch/commands/plan.ts +88 -0
  9. package/autoresearch/commands/run.ts +138 -0
  10. package/autoresearch/config.ts +82 -0
  11. package/autoresearch/engine.ts +359 -0
  12. package/autoresearch/eval-all.ts +127 -0
  13. package/autoresearch/eval-browse.ts +1 -1
  14. package/autoresearch/eval-publish.ts +238 -0
  15. package/autoresearch/eval-save.ts +249 -0
  16. package/autoresearch/eval-skill.ts +14 -8
  17. package/autoresearch/eval-v2ex.ts +220 -0
  18. package/autoresearch/eval-zhihu.ts +230 -0
  19. package/autoresearch/logger.ts +69 -0
  20. package/autoresearch/presets/combined-reliability.ts +27 -0
  21. package/autoresearch/presets/index.ts +23 -0
  22. package/autoresearch/presets/operate-reliability.ts +24 -0
  23. package/autoresearch/presets/save-reliability.ts +26 -0
  24. package/autoresearch/presets/skill-quality.ts +20 -0
  25. package/autoresearch/presets/v2ex-reliability.ts +24 -0
  26. package/autoresearch/presets/zhihu-reliability.ts +25 -0
  27. package/autoresearch/publish-tasks.json +345 -0
  28. package/autoresearch/run-save.sh +11 -0
  29. package/autoresearch/save-adapters/xhs-explore-deep.ts +64 -0
  30. package/autoresearch/save-adapters/xhs-note-comments.ts +61 -0
  31. package/autoresearch/save-adapters/xhs-search-full.ts +62 -0
  32. package/autoresearch/save-adapters/zhihu-hot-detail.ts +52 -0
  33. package/autoresearch/save-adapters/zhihu-question-full.ts +57 -0
  34. package/autoresearch/save-adapters/zhihu-search-detail.ts +53 -0
  35. package/autoresearch/save-tasks.json +281 -0
  36. package/autoresearch/v2ex-tasks.json +899 -0
  37. package/autoresearch/zhihu-tasks.json +848 -0
  38. package/bun.lock +615 -0
  39. package/dist/browser/base-page.d.ts +4 -2
  40. package/dist/browser/base-page.js +37 -4
  41. package/dist/browser/bridge.js +10 -8
  42. package/dist/browser/cdp.js +2 -6
  43. package/dist/browser/daemon-client.d.ts +11 -1
  44. package/dist/browser/daemon-client.js +3 -0
  45. package/dist/browser/dom-helpers.d.ts +4 -2
  46. package/dist/browser/dom-helpers.js +42 -31
  47. package/dist/browser/dom-snapshot.js +23 -1
  48. package/dist/browser/page.d.ts +7 -2
  49. package/dist/browser/page.js +112 -30
  50. package/dist/browser.test.js +1 -1
  51. package/dist/build-manifest.d.ts +1 -0
  52. package/dist/build-manifest.js +1 -0
  53. package/dist/cli-manifest.json +1133 -182
  54. package/dist/cli.d.ts +2 -0
  55. package/dist/cli.js +48 -7
  56. package/dist/cli.test.d.ts +1 -0
  57. package/dist/cli.test.js +88 -0
  58. package/dist/clis/1688/item.d.ts +70 -0
  59. package/dist/clis/1688/item.js +187 -0
  60. package/dist/clis/1688/item.test.d.ts +1 -0
  61. package/dist/clis/1688/item.test.js +67 -0
  62. package/dist/clis/1688/search.d.ts +56 -0
  63. package/dist/clis/1688/search.js +309 -0
  64. package/dist/clis/1688/search.test.d.ts +1 -0
  65. package/dist/clis/1688/search.test.js +75 -0
  66. package/dist/clis/1688/shared.d.ts +112 -0
  67. package/dist/clis/1688/shared.js +514 -0
  68. package/dist/clis/1688/shared.test.d.ts +1 -0
  69. package/dist/clis/1688/shared.test.js +57 -0
  70. package/dist/clis/1688/store.d.ts +45 -0
  71. package/dist/clis/1688/store.js +226 -0
  72. package/dist/clis/1688/store.test.d.ts +1 -0
  73. package/dist/clis/1688/store.test.js +62 -0
  74. package/dist/clis/amazon/bestsellers.d.ts +0 -20
  75. package/dist/clis/amazon/bestsellers.js +6 -129
  76. package/dist/clis/amazon/bestsellers.test.js +12 -3
  77. package/dist/clis/amazon/movers-shakers.d.ts +1 -0
  78. package/dist/clis/amazon/movers-shakers.js +7 -0
  79. package/dist/clis/amazon/new-releases.d.ts +1 -0
  80. package/dist/clis/amazon/new-releases.js +7 -0
  81. package/dist/clis/amazon/rankings.d.ts +59 -0
  82. package/dist/clis/amazon/rankings.js +226 -0
  83. package/dist/clis/amazon/rankings.test.d.ts +1 -0
  84. package/dist/clis/amazon/rankings.test.js +41 -0
  85. package/dist/clis/amazon/shared.d.ts +11 -0
  86. package/dist/clis/amazon/shared.js +121 -11
  87. package/dist/clis/amazon/shared.test.js +11 -0
  88. package/dist/clis/bilibili/comments.js +2 -2
  89. package/dist/clis/bilibili/comments.test.js +3 -2
  90. package/dist/clis/bilibili/download.js +2 -1
  91. package/dist/clis/bilibili/subtitle.js +4 -3
  92. package/dist/clis/bilibili/subtitle.test.js +2 -1
  93. package/dist/clis/bilibili/utils.d.ts +5 -0
  94. package/dist/clis/bilibili/utils.js +30 -0
  95. package/dist/clis/bilibili/utils.test.d.ts +1 -0
  96. package/dist/clis/bilibili/utils.test.js +17 -0
  97. package/dist/clis/douban/marks.js +1 -1
  98. package/dist/clis/douban/subject.yaml +50 -19
  99. package/dist/clis/doubao/utils.js +32 -12
  100. package/dist/clis/douyin/_shared/browser-fetch.test.js +0 -1
  101. package/dist/clis/douyin/_shared/transcode.test.js +0 -2
  102. package/dist/clis/douyin/draft.test.js +0 -2
  103. package/dist/clis/facebook/search.test.js +0 -2
  104. package/dist/clis/gemini/ask.js +9 -3
  105. package/dist/clis/gemini/ask.test.d.ts +1 -0
  106. package/dist/clis/gemini/ask.test.js +100 -0
  107. package/dist/clis/gemini/reply-state.test.d.ts +1 -0
  108. package/dist/clis/gemini/reply-state.test.js +641 -0
  109. package/dist/clis/gemini/utils.d.ts +44 -1
  110. package/dist/clis/gemini/utils.js +528 -61
  111. package/dist/clis/gemini/utils.test.js +149 -2
  112. package/dist/clis/hupu/detail.d.ts +1 -0
  113. package/dist/clis/hupu/detail.js +72 -0
  114. package/dist/clis/hupu/hot.yaml +43 -0
  115. package/dist/clis/hupu/like.d.ts +1 -0
  116. package/dist/clis/hupu/like.js +75 -0
  117. package/dist/clis/hupu/reply.d.ts +1 -0
  118. package/dist/clis/hupu/reply.js +71 -0
  119. package/dist/clis/hupu/search.d.ts +1 -0
  120. package/dist/clis/hupu/search.js +59 -0
  121. package/dist/clis/hupu/unlike.d.ts +1 -0
  122. package/dist/clis/hupu/unlike.js +75 -0
  123. package/dist/clis/hupu/utils.d.ts +20 -0
  124. package/dist/clis/hupu/utils.js +319 -0
  125. package/dist/clis/instagram/_shared/private-publish.d.ts +138 -0
  126. package/dist/clis/instagram/_shared/private-publish.js +1030 -0
  127. package/dist/clis/instagram/_shared/private-publish.test.d.ts +1 -0
  128. package/dist/clis/instagram/_shared/private-publish.test.js +705 -0
  129. package/dist/clis/instagram/_shared/protocol-capture.d.ts +26 -0
  130. package/dist/clis/instagram/_shared/protocol-capture.js +282 -0
  131. package/dist/clis/instagram/_shared/protocol-capture.test.d.ts +1 -0
  132. package/dist/clis/instagram/_shared/protocol-capture.test.js +114 -0
  133. package/dist/clis/instagram/_shared/runtime-info.d.ts +9 -0
  134. package/dist/clis/instagram/_shared/runtime-info.js +81 -0
  135. package/dist/clis/instagram/note.d.ts +1 -0
  136. package/dist/clis/instagram/note.js +222 -0
  137. package/dist/clis/instagram/note.test.d.ts +1 -0
  138. package/dist/clis/instagram/note.test.js +81 -0
  139. package/dist/clis/instagram/post.d.ts +4 -0
  140. package/dist/clis/instagram/post.js +1496 -0
  141. package/dist/clis/instagram/post.test.d.ts +1 -0
  142. package/dist/clis/instagram/post.test.js +1647 -0
  143. package/dist/clis/instagram/reel.d.ts +1 -0
  144. package/dist/clis/instagram/reel.js +826 -0
  145. package/dist/clis/instagram/reel.test.d.ts +1 -0
  146. package/dist/clis/instagram/reel.test.js +167 -0
  147. package/dist/clis/instagram/story.d.ts +1 -0
  148. package/dist/clis/instagram/story.js +115 -0
  149. package/dist/clis/instagram/story.test.d.ts +1 -0
  150. package/dist/clis/instagram/story.test.js +167 -0
  151. package/dist/clis/sinafinance/stock-rank.d.ts +4 -0
  152. package/dist/clis/sinafinance/stock-rank.js +65 -0
  153. package/dist/clis/substack/utils.test.js +0 -2
  154. package/dist/clis/twitter/post.js +72 -45
  155. package/dist/clis/twitter/post.test.d.ts +1 -0
  156. package/dist/clis/twitter/post.test.js +116 -0
  157. package/dist/clis/twitter/reply.d.ts +12 -0
  158. package/dist/clis/twitter/reply.js +257 -35
  159. package/dist/clis/twitter/reply.test.d.ts +1 -0
  160. package/dist/clis/twitter/reply.test.js +151 -0
  161. package/dist/clis/twitter/search.js +67 -5
  162. package/dist/clis/twitter/search.test.js +83 -5
  163. package/dist/clis/xianyu/chat.d.ts +7 -0
  164. package/dist/clis/xianyu/chat.js +146 -0
  165. package/dist/clis/xianyu/chat.test.d.ts +1 -0
  166. package/dist/clis/xianyu/chat.test.js +15 -0
  167. package/dist/clis/xianyu/item.d.ts +7 -0
  168. package/dist/clis/xianyu/item.js +152 -0
  169. package/dist/clis/xianyu/item.test.d.ts +1 -0
  170. package/dist/clis/xianyu/item.test.js +56 -0
  171. package/dist/clis/xianyu/search.d.ts +10 -0
  172. package/dist/clis/xianyu/search.js +134 -0
  173. package/dist/clis/xianyu/search.test.d.ts +1 -0
  174. package/dist/clis/xianyu/search.test.js +17 -0
  175. package/dist/clis/xianyu/utils.d.ts +1 -0
  176. package/dist/clis/xianyu/utils.js +8 -0
  177. package/dist/clis/xiaoe/catalog.yaml +129 -0
  178. package/dist/clis/xiaoe/content.yaml +43 -0
  179. package/dist/clis/xiaoe/courses.yaml +73 -0
  180. package/dist/clis/xiaoe/detail.yaml +39 -0
  181. package/dist/clis/xiaoe/play-url.yaml +124 -0
  182. package/dist/clis/xiaohongshu/comments.test.js +0 -2
  183. package/dist/clis/xiaohongshu/creator-note-detail.test.js +0 -2
  184. package/dist/clis/xiaohongshu/creator-notes.test.js +0 -2
  185. package/dist/clis/xiaohongshu/download.test.js +0 -2
  186. package/dist/clis/xiaohongshu/note.test.js +0 -2
  187. package/dist/clis/xiaohongshu/publish.test.js +0 -2
  188. package/dist/clis/xiaohongshu/search.js +29 -20
  189. package/dist/clis/xiaohongshu/search.test.js +56 -48
  190. package/dist/clis/yuanbao/ask.d.ts +21 -0
  191. package/dist/clis/yuanbao/ask.js +427 -0
  192. package/dist/clis/yuanbao/ask.test.d.ts +1 -0
  193. package/dist/clis/yuanbao/ask.test.js +124 -0
  194. package/dist/clis/yuanbao/new.d.ts +1 -0
  195. package/dist/clis/yuanbao/new.js +70 -0
  196. package/dist/clis/yuanbao/new.test.d.ts +1 -0
  197. package/dist/clis/yuanbao/new.test.js +30 -0
  198. package/dist/clis/yuanbao/shared.d.ts +13 -0
  199. package/dist/clis/yuanbao/shared.js +49 -0
  200. package/dist/clis/zhihu/question.js +30 -19
  201. package/dist/clis/zhihu/question.test.js +34 -16
  202. package/dist/commanderAdapter.js +8 -4
  203. package/dist/commanderAdapter.test.js +42 -0
  204. package/dist/completion.js +3 -1
  205. package/dist/completion.test.d.ts +1 -0
  206. package/dist/completion.test.js +23 -0
  207. package/dist/doctor.js +1 -1
  208. package/dist/electron-apps.d.ts +2 -0
  209. package/dist/electron-apps.js +7 -1
  210. package/dist/errors.js +1 -1
  211. package/dist/execution.js +25 -35
  212. package/dist/explore.js +1 -1
  213. package/dist/launcher.d.ts +4 -0
  214. package/dist/launcher.js +64 -8
  215. package/dist/launcher.test.js +88 -7
  216. package/dist/output.d.ts +2 -0
  217. package/dist/output.js +10 -1
  218. package/dist/output.test.d.ts +0 -3
  219. package/dist/output.test.js +59 -92
  220. package/dist/pipeline/executor.test.js +0 -2
  221. package/dist/pipeline/steps/download.test.js +0 -2
  222. package/dist/registry.d.ts +2 -0
  223. package/dist/serialization.d.ts +1 -0
  224. package/dist/serialization.js +1 -0
  225. package/dist/types.d.ts +9 -2
  226. package/docs/.vitepress/config.mts +4 -0
  227. package/docs/adapters/browser/1688.md +52 -0
  228. package/docs/adapters/browser/36kr.md +2 -1
  229. package/docs/adapters/browser/doubao.md +5 -1
  230. package/docs/adapters/browser/hupu.md +53 -0
  231. package/docs/adapters/browser/sinafinance.md +32 -2
  232. package/docs/adapters/browser/weibo.md +6 -1
  233. package/docs/adapters/browser/wikipedia.md +2 -0
  234. package/docs/adapters/browser/xianyu.md +42 -0
  235. package/docs/adapters/browser/xiaoe.md +44 -0
  236. package/docs/adapters/browser/yuanbao.md +64 -0
  237. package/docs/adapters/index.md +14 -5
  238. package/docs/comparison.md +1 -1
  239. package/docs/developer/ai-workflow.md +2 -2
  240. package/docs/developer/contributing.md +1 -1
  241. package/docs/developer/testing.md +2 -0
  242. package/docs/guide/plugins.md +1 -0
  243. package/docs/guide/troubleshooting.md +11 -0
  244. package/docs/superpowers/specs/2026-04-03-v2ex-autoresearch-design.md +41 -0
  245. package/docs/zh/guide/plugins.md +1 -0
  246. package/extension/dist/background.js +1127 -0
  247. package/extension/src/background.test.ts +39 -0
  248. package/extension/src/background.ts +223 -34
  249. package/extension/src/cdp.ts +194 -4
  250. package/extension/src/protocol.ts +22 -1
  251. package/package.json +3 -2
  252. package/scripts/postinstall.js +1 -1
  253. package/skills/opencli-explorer/SKILL.md +1 -1
  254. package/skills/opencli-oneshot/SKILL.md +2 -2
  255. package/skills/opencli-operate/SKILL.md +120 -27
  256. package/skills/opencli-usage/SKILL.md +31 -20
  257. package/skills/opencli-usage/browser.md +114 -16
  258. package/skills/opencli-usage/public-api.md +32 -3
  259. package/skills/smart-search/SKILL.md +156 -0
  260. package/skills/smart-search/references/sources-ai.md +74 -0
  261. package/skills/smart-search/references/sources-info.md +43 -0
  262. package/skills/smart-search/references/sources-media.md +50 -0
  263. package/skills/smart-search/references/sources-other.md +42 -0
  264. package/skills/smart-search/references/sources-shopping.md +31 -0
  265. package/skills/smart-search/references/sources-social.md +51 -0
  266. package/skills/smart-search/references/sources-tech.md +42 -0
  267. package/skills/smart-search/references/sources-travel.md +20 -0
  268. package/src/browser/base-page.ts +41 -6
  269. package/src/browser/bridge.ts +11 -8
  270. package/src/browser/cdp.ts +1 -8
  271. package/src/browser/daemon-client.ts +11 -1
  272. package/src/browser/dom-helpers.ts +43 -31
  273. package/src/browser/dom-snapshot.ts +23 -1
  274. package/src/browser/page.ts +115 -31
  275. package/src/browser.test.ts +1 -1
  276. package/src/build-manifest.ts +2 -0
  277. package/src/cli.test.ts +133 -0
  278. package/src/cli.ts +73 -11
  279. package/src/clis/1688/item.test.ts +69 -0
  280. package/src/clis/1688/item.ts +282 -0
  281. package/src/clis/1688/search.test.ts +81 -0
  282. package/src/clis/1688/search.ts +402 -0
  283. package/src/clis/1688/shared.test.ts +75 -0
  284. package/src/clis/1688/shared.ts +623 -0
  285. package/src/clis/1688/store.test.ts +69 -0
  286. package/src/clis/1688/store.ts +300 -0
  287. package/src/clis/amazon/bestsellers.test.ts +12 -3
  288. package/src/clis/amazon/bestsellers.ts +6 -178
  289. package/src/clis/amazon/movers-shakers.ts +8 -0
  290. package/src/clis/amazon/new-releases.ts +8 -0
  291. package/src/clis/amazon/rankings.test.ts +47 -0
  292. package/src/clis/amazon/rankings.ts +312 -0
  293. package/src/clis/amazon/shared.test.ts +16 -0
  294. package/src/clis/amazon/shared.ts +134 -12
  295. package/src/clis/bilibili/comments.test.ts +4 -3
  296. package/src/clis/bilibili/comments.ts +2 -2
  297. package/src/clis/bilibili/download.ts +2 -1
  298. package/src/clis/bilibili/subtitle.test.ts +2 -1
  299. package/src/clis/bilibili/subtitle.ts +4 -3
  300. package/src/clis/bilibili/utils.test.ts +21 -0
  301. package/src/clis/bilibili/utils.ts +27 -0
  302. package/src/clis/douban/marks.ts +1 -1
  303. package/src/clis/douban/subject.yaml +50 -19
  304. package/src/clis/doubao/utils.ts +32 -12
  305. package/src/clis/douyin/_shared/browser-fetch.test.ts +0 -1
  306. package/src/clis/douyin/_shared/transcode.test.ts +0 -2
  307. package/src/clis/douyin/draft.test.ts +0 -2
  308. package/src/clis/facebook/search.test.ts +0 -2
  309. package/src/clis/gemini/ask.test.ts +116 -0
  310. package/src/clis/gemini/ask.ts +10 -3
  311. package/src/clis/gemini/reply-state.test.ts +708 -0
  312. package/src/clis/gemini/utils.test.ts +184 -2
  313. package/src/clis/gemini/utils.ts +588 -60
  314. package/src/clis/hupu/detail.ts +126 -0
  315. package/src/clis/hupu/hot.yaml +43 -0
  316. package/src/clis/hupu/like.ts +76 -0
  317. package/src/clis/hupu/reply.ts +76 -0
  318. package/src/clis/hupu/search.ts +95 -0
  319. package/src/clis/hupu/unlike.ts +76 -0
  320. package/src/clis/hupu/utils.ts +381 -0
  321. package/src/clis/instagram/_shared/private-publish.test.ts +827 -0
  322. package/src/clis/instagram/_shared/private-publish.ts +1303 -0
  323. package/src/clis/instagram/_shared/protocol-capture.test.ts +148 -0
  324. package/src/clis/instagram/_shared/protocol-capture.ts +321 -0
  325. package/src/clis/instagram/_shared/runtime-info.ts +91 -0
  326. package/src/clis/instagram/note.test.ts +96 -0
  327. package/src/clis/instagram/note.ts +254 -0
  328. package/src/clis/instagram/post.test.ts +1716 -0
  329. package/src/clis/instagram/post.ts +1620 -0
  330. package/src/clis/instagram/reel.test.ts +191 -0
  331. package/src/clis/instagram/reel.ts +886 -0
  332. package/src/clis/instagram/story.test.ts +191 -0
  333. package/src/clis/instagram/story.ts +151 -0
  334. package/src/clis/sinafinance/stock-rank.ts +68 -0
  335. package/src/clis/substack/utils.test.ts +0 -2
  336. package/src/clis/twitter/post.test.ts +157 -0
  337. package/src/clis/twitter/post.ts +82 -48
  338. package/src/clis/twitter/reply.test.ts +177 -0
  339. package/src/clis/twitter/reply.ts +285 -39
  340. package/src/clis/twitter/search.test.ts +88 -5
  341. package/src/clis/twitter/search.ts +68 -5
  342. package/src/clis/xianyu/chat.test.ts +20 -0
  343. package/src/clis/xianyu/chat.ts +175 -0
  344. package/src/clis/xianyu/item.test.ts +67 -0
  345. package/src/clis/xianyu/item.ts +172 -0
  346. package/src/clis/xianyu/search.test.ts +22 -0
  347. package/src/clis/xianyu/search.ts +151 -0
  348. package/src/clis/xianyu/utils.ts +9 -0
  349. package/src/clis/xiaoe/catalog.yaml +129 -0
  350. package/src/clis/xiaoe/content.yaml +43 -0
  351. package/src/clis/xiaoe/courses.yaml +73 -0
  352. package/src/clis/xiaoe/detail.yaml +39 -0
  353. package/src/clis/xiaoe/play-url.yaml +124 -0
  354. package/src/clis/xiaohongshu/comments.test.ts +0 -2
  355. package/src/clis/xiaohongshu/creator-note-detail.test.ts +0 -2
  356. package/src/clis/xiaohongshu/creator-notes.test.ts +0 -2
  357. package/src/clis/xiaohongshu/download.test.ts +0 -2
  358. package/src/clis/xiaohongshu/note.test.ts +0 -2
  359. package/src/clis/xiaohongshu/publish.test.ts +0 -2
  360. package/src/clis/xiaohongshu/search.test.ts +59 -48
  361. package/src/clis/xiaohongshu/search.ts +31 -21
  362. package/src/clis/yuanbao/ask.test.ts +156 -0
  363. package/src/clis/yuanbao/ask.ts +522 -0
  364. package/src/clis/yuanbao/new.test.ts +36 -0
  365. package/src/clis/yuanbao/new.ts +81 -0
  366. package/src/clis/yuanbao/shared.ts +57 -0
  367. package/src/clis/zhihu/question.test.ts +42 -17
  368. package/src/clis/zhihu/question.ts +31 -26
  369. package/src/commanderAdapter.test.ts +51 -0
  370. package/src/commanderAdapter.ts +8 -4
  371. package/src/completion.test.ts +30 -0
  372. package/src/completion.ts +3 -1
  373. package/src/doctor.ts +1 -1
  374. package/src/electron-apps.ts +9 -1
  375. package/src/errors.ts +1 -1
  376. package/src/execution.ts +26 -30
  377. package/src/explore.ts +1 -1
  378. package/src/launcher.test.ts +121 -7
  379. package/src/launcher.ts +87 -9
  380. package/src/output.test.ts +50 -90
  381. package/src/output.ts +10 -1
  382. package/src/pipeline/executor.test.ts +0 -2
  383. package/src/pipeline/steps/download.test.ts +0 -2
  384. package/src/registry.ts +2 -0
  385. package/src/serialization.ts +2 -0
  386. package/src/types.ts +9 -2
  387. package/tests/e2e/browser-auth.test.ts +9 -0
  388. package/CLI-EXPLORER.md +0 -724
  389. package/CLI-ONESHOT.md +0 -216
  390. package/SKILL.md +0 -59
@@ -0,0 +1,708 @@
1
+ import { describe, expect, it, vi } from 'vitest';
2
+ import type { IPage } from '../../types.js';
3
+ import type { GeminiSnapshot } from './utils.js';
4
+ import { __test__, waitForGeminiResponse, waitForGeminiSubmission } from './utils.js';
5
+
6
+ function snapshot(overrides: Partial<GeminiSnapshot> = {}): GeminiSnapshot {
7
+ return {
8
+ turns: [],
9
+ transcriptLines: [],
10
+ composerHasText: false,
11
+ isGenerating: false,
12
+ structuredTurnsTrusted: true,
13
+ ...overrides,
14
+ };
15
+ }
16
+
17
+ function createPageMock(): IPage {
18
+ return {
19
+ goto: vi.fn().mockResolvedValue(undefined),
20
+ evaluate: vi.fn(),
21
+ getCookies: vi.fn().mockResolvedValue([]),
22
+ snapshot: vi.fn().mockResolvedValue(undefined),
23
+ click: vi.fn().mockResolvedValue(undefined),
24
+ typeText: vi.fn().mockResolvedValue(undefined),
25
+ pressKey: vi.fn().mockResolvedValue(undefined),
26
+ scrollTo: vi.fn().mockResolvedValue(undefined),
27
+ getFormState: vi.fn().mockResolvedValue({}),
28
+ wait: vi.fn().mockResolvedValue(undefined),
29
+ tabs: vi.fn().mockResolvedValue([]),
30
+ selectTab: vi.fn().mockResolvedValue(undefined),
31
+ networkRequests: vi.fn().mockResolvedValue([]),
32
+ consoleMessages: vi.fn().mockResolvedValue([]),
33
+ scroll: vi.fn().mockResolvedValue(undefined),
34
+ autoScroll: vi.fn().mockResolvedValue(undefined),
35
+ installInterceptor: vi.fn().mockResolvedValue(undefined),
36
+ getInterceptedRequests: vi.fn().mockResolvedValue([]),
37
+ waitForCapture: vi.fn().mockResolvedValue(undefined),
38
+ screenshot: vi.fn().mockResolvedValue(''),
39
+ nativeType: vi.fn().mockResolvedValue(undefined),
40
+ nativeKeyPress: vi.fn().mockResolvedValue(undefined),
41
+ } as unknown as IPage;
42
+ }
43
+
44
+ describe('Gemini snapshot diff helpers', () => {
45
+ it('reports appended trusted turns when the current snapshot extends the baseline', () => {
46
+ const before = snapshot({
47
+ turns: [{ Role: 'Assistant', Text: '旧回答' }],
48
+ });
49
+ const current = snapshot({
50
+ turns: [
51
+ { Role: 'Assistant', Text: '旧回答' },
52
+ { Role: 'User', Text: '请只回复:OK' },
53
+ ],
54
+ });
55
+
56
+ expect(__test__.diffTrustedStructuredTurns(before, current)).toEqual({
57
+ appendedTurns: [{ Role: 'User', Text: '请只回复:OK' }],
58
+ hasTrustedAppend: true,
59
+ hasNewUserTurn: true,
60
+ hasNewAssistantTurn: false,
61
+ });
62
+ });
63
+
64
+ it('treats restored structured turns as untrusted when the pre-send snapshot had no trustworthy turns', () => {
65
+ const before = snapshot({
66
+ turns: [],
67
+ transcriptLines: ['旧问题', '旧回答'],
68
+ structuredTurnsTrusted: false,
69
+ });
70
+ const current = snapshot({
71
+ turns: [
72
+ { Role: 'User', Text: '旧问题' },
73
+ { Role: 'Assistant', Text: '旧回答' },
74
+ ],
75
+ transcriptLines: ['旧问题', '旧回答'],
76
+ structuredTurnsTrusted: true,
77
+ });
78
+
79
+ expect(__test__.diffTrustedStructuredTurns(before, current)).toEqual({
80
+ appendedTurns: [],
81
+ hasTrustedAppend: false,
82
+ hasNewUserTurn: false,
83
+ hasNewAssistantTurn: false,
84
+ });
85
+ });
86
+
87
+ it('keeps transcript delta lines raw for later conservative fallback checks', () => {
88
+ const before = snapshot({
89
+ transcriptLines: ['baseline'],
90
+ });
91
+ const current = snapshot({
92
+ transcriptLines: ['baseline', '关于“请只回复:OK”,这里是解释。'],
93
+ structuredTurnsTrusted: false,
94
+ });
95
+
96
+ expect(__test__.diffTranscriptLines(before, current)).toEqual([
97
+ '关于“请只回复:OK”,这里是解释。',
98
+ ]);
99
+ });
100
+ });
101
+
102
+ describe('Gemini submission state', () => {
103
+ it('confirms submission from a trusted appended user turn', async () => {
104
+ const page = createPageMock();
105
+ const evaluate = vi.mocked(page.evaluate);
106
+
107
+ evaluate
108
+ .mockResolvedValueOnce('https://gemini.google.com/app')
109
+ .mockResolvedValueOnce({
110
+ turns: [
111
+ { Role: 'Assistant', Text: '旧回答' },
112
+ { Role: 'User', Text: '请只回复:OK' },
113
+ ],
114
+ transcriptLines: ['baseline', '请只回复:OK'],
115
+ composerHasText: false,
116
+ isGenerating: true,
117
+ structuredTurnsTrusted: true,
118
+ });
119
+
120
+ const result = await waitForGeminiSubmission(page, snapshot({
121
+ turns: [{ Role: 'Assistant', Text: '旧回答' }],
122
+ transcriptLines: ['baseline'],
123
+ composerHasText: true,
124
+ structuredTurnsTrusted: true,
125
+ }), 4);
126
+
127
+ expect(result).toEqual({
128
+ snapshot: {
129
+ turns: [
130
+ { Role: 'Assistant', Text: '旧回答' },
131
+ { Role: 'User', Text: '请只回复:OK' },
132
+ ],
133
+ transcriptLines: ['baseline', '请只回复:OK'],
134
+ composerHasText: false,
135
+ isGenerating: true,
136
+ structuredTurnsTrusted: true,
137
+ },
138
+ preSendAssistantCount: 1,
139
+ userAnchorTurn: { Role: 'User', Text: '请只回复:OK' },
140
+ reason: 'user_turn',
141
+ });
142
+ });
143
+
144
+ it('confirms submission from composer cleared plus generating even when transcript has not changed yet', async () => {
145
+ const page = createPageMock();
146
+ const evaluate = vi.mocked(page.evaluate);
147
+
148
+ evaluate
149
+ .mockResolvedValueOnce('https://gemini.google.com/app')
150
+ .mockResolvedValueOnce({
151
+ turns: [],
152
+ transcriptLines: ['baseline'],
153
+ composerHasText: false,
154
+ isGenerating: true,
155
+ structuredTurnsTrusted: false,
156
+ });
157
+
158
+ const result = await waitForGeminiSubmission(page, snapshot({
159
+ transcriptLines: ['baseline'],
160
+ composerHasText: true,
161
+ structuredTurnsTrusted: false,
162
+ }), 2);
163
+
164
+ expect(result).toEqual({
165
+ snapshot: {
166
+ turns: [],
167
+ transcriptLines: ['baseline'],
168
+ composerHasText: false,
169
+ isGenerating: true,
170
+ structuredTurnsTrusted: false,
171
+ },
172
+ preSendAssistantCount: 0,
173
+ userAnchorTurn: null,
174
+ reason: 'composer_generating',
175
+ });
176
+ });
177
+
178
+ it('confirms submission from generating state even when the pre-send baseline composer was empty', async () => {
179
+ const page = createPageMock();
180
+ const evaluate = vi.mocked(page.evaluate);
181
+
182
+ evaluate
183
+ .mockResolvedValueOnce('https://gemini.google.com/app')
184
+ .mockResolvedValueOnce({
185
+ turns: [
186
+ { Role: 'User', Text: '你说\n\n请只回复:DBG2' },
187
+ { Role: 'User', Text: '请只回复:DBG2' },
188
+ ],
189
+ transcriptLines: ['baseline', '请只回复:DBG2'],
190
+ composerHasText: false,
191
+ isGenerating: true,
192
+ structuredTurnsTrusted: true,
193
+ });
194
+
195
+ const result = await waitForGeminiSubmission(page, snapshot({
196
+ turns: [{ Role: 'Assistant', Text: '需要我为你做些什么?' }],
197
+ transcriptLines: ['baseline'],
198
+ composerHasText: false,
199
+ structuredTurnsTrusted: true,
200
+ }), 2);
201
+
202
+ expect(result).toEqual({
203
+ snapshot: {
204
+ turns: [
205
+ { Role: 'User', Text: '你说\n\n请只回复:DBG2' },
206
+ { Role: 'User', Text: '请只回复:DBG2' },
207
+ ],
208
+ transcriptLines: ['baseline', '请只回复:DBG2'],
209
+ composerHasText: false,
210
+ isGenerating: true,
211
+ structuredTurnsTrusted: true,
212
+ },
213
+ preSendAssistantCount: 1,
214
+ userAnchorTurn: { Role: 'User', Text: '请只回复:DBG2' },
215
+ reason: 'composer_generating',
216
+ });
217
+ });
218
+
219
+ it('confirms submission from composer cleared plus transcript growth when generation state is unavailable', async () => {
220
+ const page = createPageMock();
221
+ const evaluate = vi.mocked(page.evaluate);
222
+
223
+ // This transcript delta may be only a prompt echo. It is allowed to confirm
224
+ // submission only because the composer has already cleared, and it must never
225
+ // be reused later as reply ownership evidence.
226
+ evaluate
227
+ .mockResolvedValueOnce('https://gemini.google.com/app')
228
+ .mockResolvedValueOnce({
229
+ turns: [],
230
+ transcriptLines: ['baseline', '请只回复:OK'],
231
+ composerHasText: false,
232
+ isGenerating: false,
233
+ structuredTurnsTrusted: false,
234
+ });
235
+
236
+ const result = await waitForGeminiSubmission(page, snapshot({
237
+ transcriptLines: ['baseline'],
238
+ composerHasText: true,
239
+ structuredTurnsTrusted: false,
240
+ }), 2);
241
+
242
+ expect(result).toEqual({
243
+ snapshot: {
244
+ turns: [],
245
+ transcriptLines: ['baseline', '请只回复:OK'],
246
+ composerHasText: false,
247
+ isGenerating: false,
248
+ structuredTurnsTrusted: false,
249
+ },
250
+ preSendAssistantCount: 0,
251
+ userAnchorTurn: null,
252
+ reason: 'composer_transcript',
253
+ });
254
+ });
255
+
256
+ it('does not confirm submission when old structured turns only reappear after an untrusted pre-send snapshot', async () => {
257
+ const page = createPageMock();
258
+ const evaluate = vi.mocked(page.evaluate);
259
+
260
+ evaluate
261
+ .mockResolvedValueOnce('https://gemini.google.com/app')
262
+ .mockResolvedValueOnce({
263
+ turns: [
264
+ { Role: 'User', Text: '旧问题' },
265
+ { Role: 'Assistant', Text: '旧回答' },
266
+ ],
267
+ transcriptLines: ['旧问题', '旧回答'],
268
+ composerHasText: false,
269
+ isGenerating: false,
270
+ structuredTurnsTrusted: true,
271
+ })
272
+ .mockResolvedValueOnce('https://gemini.google.com/app')
273
+ .mockResolvedValueOnce({
274
+ turns: [
275
+ { Role: 'User', Text: '旧问题' },
276
+ { Role: 'Assistant', Text: '旧回答' },
277
+ ],
278
+ transcriptLines: ['旧问题', '旧回答'],
279
+ composerHasText: false,
280
+ isGenerating: false,
281
+ structuredTurnsTrusted: true,
282
+ });
283
+
284
+ const result = await waitForGeminiSubmission(page, snapshot({
285
+ turns: [],
286
+ transcriptLines: ['旧问题', '旧回答'],
287
+ composerHasText: true,
288
+ structuredTurnsTrusted: false,
289
+ }), 2);
290
+
291
+ expect(result).toBeNull();
292
+ });
293
+
294
+ it('does not confirm submission from transcript growth alone when the composer never clears', async () => {
295
+ const page = createPageMock();
296
+ const evaluate = vi.mocked(page.evaluate);
297
+
298
+ evaluate
299
+ .mockResolvedValueOnce('https://gemini.google.com/app')
300
+ .mockResolvedValueOnce({
301
+ turns: [],
302
+ transcriptLines: ['baseline', '请只回复:OK'],
303
+ composerHasText: true,
304
+ isGenerating: false,
305
+ structuredTurnsTrusted: false,
306
+ })
307
+ .mockResolvedValueOnce('https://gemini.google.com/app')
308
+ .mockResolvedValueOnce({
309
+ turns: [],
310
+ transcriptLines: ['baseline', '请只回复:OK'],
311
+ composerHasText: true,
312
+ isGenerating: false,
313
+ structuredTurnsTrusted: false,
314
+ });
315
+
316
+ const result = await waitForGeminiSubmission(page, snapshot({
317
+ transcriptLines: ['baseline'],
318
+ composerHasText: true,
319
+ structuredTurnsTrusted: false,
320
+ }), 2);
321
+
322
+ expect(result).toBeNull();
323
+ });
324
+
325
+ it('keeps polling past ten seconds when the overall timeout budget still allows submission confirmation', async () => {
326
+ const page = createPageMock();
327
+ const evaluate = vi.mocked(page.evaluate);
328
+
329
+ for (let index = 0; index < 10; index += 1) {
330
+ evaluate
331
+ .mockResolvedValueOnce('https://gemini.google.com/app')
332
+ .mockResolvedValueOnce({
333
+ turns: [],
334
+ transcriptLines: ['baseline'],
335
+ composerHasText: true,
336
+ isGenerating: false,
337
+ structuredTurnsTrusted: false,
338
+ });
339
+ }
340
+
341
+ evaluate
342
+ .mockResolvedValueOnce('https://gemini.google.com/app')
343
+ .mockResolvedValueOnce({
344
+ turns: [],
345
+ transcriptLines: ['baseline', '请只回复:OK'],
346
+ composerHasText: false,
347
+ isGenerating: false,
348
+ structuredTurnsTrusted: false,
349
+ });
350
+
351
+ const result = await waitForGeminiSubmission(page, snapshot({
352
+ transcriptLines: ['baseline'],
353
+ composerHasText: true,
354
+ structuredTurnsTrusted: false,
355
+ }), 12);
356
+
357
+ expect(result?.reason).toBe('composer_transcript');
358
+ });
359
+ });
360
+
361
+ describe('Gemini reply state', () => {
362
+ it('does not reuse an older identical reply when the submission baseline has no structured user anchor', async () => {
363
+ const page = createPageMock();
364
+ const evaluate = vi.mocked(page.evaluate);
365
+
366
+ evaluate
367
+ .mockResolvedValueOnce('https://gemini.google.com/app')
368
+ .mockResolvedValueOnce({
369
+ turns: [{ Role: 'Assistant', Text: 'OK' }],
370
+ transcriptLines: ['baseline', 'OK'],
371
+ composerHasText: false,
372
+ isGenerating: false,
373
+ structuredTurnsTrusted: true,
374
+ })
375
+ .mockResolvedValueOnce('https://gemini.google.com/app')
376
+ .mockResolvedValueOnce({
377
+ turns: [
378
+ { Role: 'Assistant', Text: 'OK' },
379
+ { Role: 'Assistant', Text: 'OK' },
380
+ ],
381
+ transcriptLines: ['baseline', 'OK'],
382
+ composerHasText: false,
383
+ isGenerating: false,
384
+ structuredTurnsTrusted: true,
385
+ })
386
+ .mockResolvedValueOnce('https://gemini.google.com/app')
387
+ .mockResolvedValueOnce({
388
+ turns: [
389
+ { Role: 'Assistant', Text: 'OK' },
390
+ { Role: 'Assistant', Text: 'OK' },
391
+ ],
392
+ transcriptLines: ['baseline', 'OK'],
393
+ composerHasText: false,
394
+ isGenerating: false,
395
+ structuredTurnsTrusted: true,
396
+ });
397
+
398
+ const result = await waitForGeminiResponse(page, {
399
+ snapshot: snapshot({
400
+ turns: [{ Role: 'Assistant', Text: 'OK' }],
401
+ transcriptLines: ['baseline', 'OK'],
402
+ composerHasText: false,
403
+ isGenerating: true,
404
+ structuredTurnsTrusted: true,
405
+ }),
406
+ preSendAssistantCount: 1,
407
+ userAnchorTurn: null,
408
+ reason: 'composer_generating',
409
+ }, '请只回复:OK', 6);
410
+
411
+ expect(result).toBe('OK');
412
+ });
413
+
414
+ it('does not treat prepended older history as the current round reply when reply ownership has no user anchor', async () => {
415
+ const page = createPageMock();
416
+ const evaluate = vi.mocked(page.evaluate);
417
+
418
+ evaluate
419
+ .mockResolvedValueOnce('https://gemini.google.com/app')
420
+ .mockResolvedValueOnce({
421
+ turns: [
422
+ { Role: 'Assistant', Text: '更早的问题' },
423
+ { Role: 'Assistant', Text: '旧回答' },
424
+ ],
425
+ transcriptLines: ['baseline'],
426
+ composerHasText: false,
427
+ isGenerating: false,
428
+ structuredTurnsTrusted: true,
429
+ })
430
+ .mockResolvedValueOnce('https://gemini.google.com/app')
431
+ .mockResolvedValueOnce({
432
+ turns: [
433
+ { Role: 'Assistant', Text: '更早的问题' },
434
+ { Role: 'Assistant', Text: '旧回答' },
435
+ ],
436
+ transcriptLines: ['baseline'],
437
+ composerHasText: false,
438
+ isGenerating: false,
439
+ structuredTurnsTrusted: true,
440
+ })
441
+ .mockResolvedValueOnce('https://gemini.google.com/app')
442
+ .mockResolvedValueOnce({
443
+ turns: [
444
+ { Role: 'Assistant', Text: '更早的问题' },
445
+ { Role: 'Assistant', Text: '旧回答' },
446
+ ],
447
+ transcriptLines: ['baseline'],
448
+ composerHasText: false,
449
+ isGenerating: false,
450
+ structuredTurnsTrusted: true,
451
+ });
452
+
453
+ const result = await waitForGeminiResponse(page, {
454
+ snapshot: snapshot({
455
+ turns: [{ Role: 'Assistant', Text: '旧回答' }],
456
+ transcriptLines: ['baseline'],
457
+ composerHasText: false,
458
+ isGenerating: true,
459
+ structuredTurnsTrusted: true,
460
+ }),
461
+ preSendAssistantCount: 1,
462
+ userAnchorTurn: null,
463
+ reason: 'composer_generating',
464
+ }, '请只回复:OK', 6);
465
+
466
+ expect(result).toBe('');
467
+ });
468
+
469
+ it('accepts a reply when the submission snapshot contains only the current round user turns and later appends a new assistant', async () => {
470
+ const page = createPageMock();
471
+ const evaluate = vi.mocked(page.evaluate);
472
+
473
+ evaluate
474
+ .mockResolvedValueOnce('https://gemini.google.com/app')
475
+ .mockResolvedValueOnce({
476
+ turns: [
477
+ { Role: 'User', Text: '你说\n\n请只回复:DBGREG' },
478
+ { Role: 'User', Text: '请只回复:DBGREG' },
479
+ { Role: 'Assistant', Text: 'DBGREG' },
480
+ ],
481
+ transcriptLines: ['baseline'],
482
+ composerHasText: false,
483
+ isGenerating: false,
484
+ structuredTurnsTrusted: true,
485
+ })
486
+ .mockResolvedValueOnce('https://gemini.google.com/app')
487
+ .mockResolvedValueOnce({
488
+ turns: [
489
+ { Role: 'User', Text: '你说\n\n请只回复:DBGREG' },
490
+ { Role: 'User', Text: '请只回复:DBGREG' },
491
+ { Role: 'Assistant', Text: 'DBGREG' },
492
+ ],
493
+ transcriptLines: ['baseline'],
494
+ composerHasText: false,
495
+ isGenerating: false,
496
+ structuredTurnsTrusted: true,
497
+ });
498
+
499
+ const result = await waitForGeminiResponse(page, {
500
+ snapshot: snapshot({
501
+ turns: [
502
+ { Role: 'User', Text: '你说\n\n请只回复:DBGREG' },
503
+ { Role: 'User', Text: '请只回复:DBGREG' },
504
+ ],
505
+ transcriptLines: ['baseline'],
506
+ composerHasText: false,
507
+ isGenerating: true,
508
+ structuredTurnsTrusted: true,
509
+ }),
510
+ preSendAssistantCount: 1,
511
+ userAnchorTurn: { Role: 'User', Text: '请只回复:DBGREG' },
512
+ reason: 'composer_generating',
513
+ }, '请只回复:DBGREG', 6);
514
+
515
+ expect(result).toBe('DBGREG');
516
+ });
517
+
518
+ it('does not trust an assistant-only submission snapshot without a stable post-submission owner', async () => {
519
+ const page = createPageMock();
520
+ const evaluate = vi.mocked(page.evaluate);
521
+
522
+ evaluate
523
+ .mockResolvedValueOnce('https://gemini.google.com/app')
524
+ .mockResolvedValueOnce({
525
+ turns: [{ Role: 'Assistant', Text: '完整回答' }],
526
+ transcriptLines: ['baseline'],
527
+ composerHasText: false,
528
+ isGenerating: false,
529
+ structuredTurnsTrusted: true,
530
+ })
531
+ .mockResolvedValueOnce('https://gemini.google.com/app')
532
+ .mockResolvedValueOnce({
533
+ turns: [{ Role: 'Assistant', Text: '完整回答' }],
534
+ transcriptLines: ['baseline'],
535
+ composerHasText: false,
536
+ isGenerating: false,
537
+ structuredTurnsTrusted: true,
538
+ });
539
+
540
+ const result = await waitForGeminiResponse(page, {
541
+ snapshot: snapshot({
542
+ turns: [{ Role: 'Assistant', Text: '半截回答' }],
543
+ transcriptLines: ['baseline'],
544
+ composerHasText: false,
545
+ isGenerating: true,
546
+ structuredTurnsTrusted: true,
547
+ }),
548
+ preSendAssistantCount: 0,
549
+ userAnchorTurn: null,
550
+ reason: 'composer_generating',
551
+ }, '请解释', 4);
552
+
553
+ expect(result).toBe('');
554
+ });
555
+
556
+ it('accepts an assistant reply that appears after a structured user anchor only after it stabilizes and generation stops', async () => {
557
+ const page = createPageMock();
558
+ const evaluate = vi.mocked(page.evaluate);
559
+
560
+ evaluate
561
+ .mockResolvedValueOnce('https://gemini.google.com/app')
562
+ .mockResolvedValueOnce({
563
+ turns: [
564
+ { Role: 'Assistant', Text: '旧回答' },
565
+ { Role: 'User', Text: '请解释' },
566
+ { Role: 'Assistant', Text: '半截回答' },
567
+ ],
568
+ transcriptLines: ['baseline'],
569
+ composerHasText: false,
570
+ isGenerating: true,
571
+ structuredTurnsTrusted: true,
572
+ })
573
+ .mockResolvedValueOnce('https://gemini.google.com/app')
574
+ .mockResolvedValueOnce({
575
+ turns: [
576
+ { Role: 'Assistant', Text: '旧回答' },
577
+ { Role: 'User', Text: '请解释' },
578
+ { Role: 'Assistant', Text: '完整回答' },
579
+ ],
580
+ transcriptLines: ['baseline'],
581
+ composerHasText: false,
582
+ isGenerating: true,
583
+ structuredTurnsTrusted: true,
584
+ })
585
+ .mockResolvedValueOnce('https://gemini.google.com/app')
586
+ .mockResolvedValueOnce({
587
+ turns: [
588
+ { Role: 'Assistant', Text: '旧回答' },
589
+ { Role: 'User', Text: '请解释' },
590
+ { Role: 'Assistant', Text: '完整回答' },
591
+ ],
592
+ transcriptLines: ['baseline'],
593
+ composerHasText: false,
594
+ isGenerating: false,
595
+ structuredTurnsTrusted: true,
596
+ });
597
+
598
+ const result = await waitForGeminiResponse(page, {
599
+ snapshot: snapshot({
600
+ turns: [
601
+ { Role: 'Assistant', Text: '旧回答' },
602
+ { Role: 'User', Text: '请解释' },
603
+ ],
604
+ transcriptLines: ['baseline'],
605
+ composerHasText: false,
606
+ isGenerating: true,
607
+ structuredTurnsTrusted: true,
608
+ }),
609
+ preSendAssistantCount: 1,
610
+ userAnchorTurn: { Role: 'User', Text: '请解释' },
611
+ reason: 'user_turn',
612
+ }, '请解释', 6);
613
+
614
+ expect(result).toBe('完整回答');
615
+ });
616
+
617
+ it('uses transcript fallback only after two identical post-submission deltas and after generation stops', async () => {
618
+ const page = createPageMock();
619
+ const evaluate = vi.mocked(page.evaluate);
620
+
621
+ evaluate
622
+ .mockResolvedValueOnce('https://gemini.google.com/app')
623
+ .mockResolvedValueOnce({
624
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
625
+ transcriptLines: ['baseline', 'OK'],
626
+ composerHasText: false,
627
+ isGenerating: true,
628
+ structuredTurnsTrusted: true,
629
+ })
630
+ .mockResolvedValueOnce('https://gemini.google.com/app')
631
+ .mockResolvedValueOnce({
632
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
633
+ transcriptLines: ['baseline', 'OK'],
634
+ composerHasText: false,
635
+ isGenerating: false,
636
+ structuredTurnsTrusted: true,
637
+ })
638
+ .mockResolvedValueOnce('https://gemini.google.com/app')
639
+ .mockResolvedValueOnce({
640
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
641
+ transcriptLines: ['baseline', 'OK'],
642
+ composerHasText: false,
643
+ isGenerating: false,
644
+ structuredTurnsTrusted: true,
645
+ });
646
+
647
+ const result = await waitForGeminiResponse(page, {
648
+ snapshot: snapshot({
649
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
650
+ transcriptLines: ['baseline'],
651
+ composerHasText: false,
652
+ isGenerating: true,
653
+ structuredTurnsTrusted: true,
654
+ }),
655
+ preSendAssistantCount: 0,
656
+ userAnchorTurn: { Role: 'User', Text: '请只回复:OK' },
657
+ reason: 'user_turn',
658
+ }, '请只回复:OK', 6);
659
+
660
+ expect(result).toBe('OK');
661
+ });
662
+
663
+ it('ignores transcript lines that appeared before submission confirmation and only accepts post-submission transcript deltas', async () => {
664
+ const page = createPageMock();
665
+ const evaluate = vi.mocked(page.evaluate);
666
+
667
+ evaluate
668
+ .mockResolvedValueOnce('https://gemini.google.com/app')
669
+ .mockResolvedValueOnce({
670
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
671
+ transcriptLines: ['baseline', '早到的提示词回声', 'OK'],
672
+ composerHasText: false,
673
+ isGenerating: false,
674
+ structuredTurnsTrusted: true,
675
+ })
676
+ .mockResolvedValueOnce('https://gemini.google.com/app')
677
+ .mockResolvedValueOnce({
678
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
679
+ transcriptLines: ['baseline', '早到的提示词回声', 'OK'],
680
+ composerHasText: false,
681
+ isGenerating: false,
682
+ structuredTurnsTrusted: true,
683
+ })
684
+ .mockResolvedValueOnce('https://gemini.google.com/app')
685
+ .mockResolvedValueOnce({
686
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
687
+ transcriptLines: ['baseline', '早到的提示词回声', 'OK'],
688
+ composerHasText: false,
689
+ isGenerating: false,
690
+ structuredTurnsTrusted: true,
691
+ });
692
+
693
+ const result = await waitForGeminiResponse(page, {
694
+ snapshot: snapshot({
695
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
696
+ transcriptLines: ['baseline', '早到的提示词回声'],
697
+ composerHasText: false,
698
+ isGenerating: true,
699
+ structuredTurnsTrusted: true,
700
+ }),
701
+ preSendAssistantCount: 0,
702
+ userAnchorTurn: { Role: 'User', Text: '请只回复:OK' },
703
+ reason: 'composer_transcript',
704
+ }, '请只回复:OK', 6);
705
+
706
+ expect(result).toBe('OK');
707
+ });
708
+ });