@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,641 @@
1
+ import { describe, expect, it, vi } from 'vitest';
2
+ import { __test__, waitForGeminiResponse, waitForGeminiSubmission } from './utils.js';
3
+ function snapshot(overrides = {}) {
4
+ return {
5
+ turns: [],
6
+ transcriptLines: [],
7
+ composerHasText: false,
8
+ isGenerating: false,
9
+ structuredTurnsTrusted: true,
10
+ ...overrides,
11
+ };
12
+ }
13
+ function createPageMock() {
14
+ return {
15
+ goto: vi.fn().mockResolvedValue(undefined),
16
+ evaluate: vi.fn(),
17
+ getCookies: vi.fn().mockResolvedValue([]),
18
+ snapshot: vi.fn().mockResolvedValue(undefined),
19
+ click: vi.fn().mockResolvedValue(undefined),
20
+ typeText: vi.fn().mockResolvedValue(undefined),
21
+ pressKey: vi.fn().mockResolvedValue(undefined),
22
+ scrollTo: vi.fn().mockResolvedValue(undefined),
23
+ getFormState: vi.fn().mockResolvedValue({}),
24
+ wait: vi.fn().mockResolvedValue(undefined),
25
+ tabs: vi.fn().mockResolvedValue([]),
26
+ selectTab: vi.fn().mockResolvedValue(undefined),
27
+ networkRequests: vi.fn().mockResolvedValue([]),
28
+ consoleMessages: vi.fn().mockResolvedValue([]),
29
+ scroll: vi.fn().mockResolvedValue(undefined),
30
+ autoScroll: vi.fn().mockResolvedValue(undefined),
31
+ installInterceptor: vi.fn().mockResolvedValue(undefined),
32
+ getInterceptedRequests: vi.fn().mockResolvedValue([]),
33
+ waitForCapture: vi.fn().mockResolvedValue(undefined),
34
+ screenshot: vi.fn().mockResolvedValue(''),
35
+ nativeType: vi.fn().mockResolvedValue(undefined),
36
+ nativeKeyPress: vi.fn().mockResolvedValue(undefined),
37
+ };
38
+ }
39
+ describe('Gemini snapshot diff helpers', () => {
40
+ it('reports appended trusted turns when the current snapshot extends the baseline', () => {
41
+ const before = snapshot({
42
+ turns: [{ Role: 'Assistant', Text: '旧回答' }],
43
+ });
44
+ const current = snapshot({
45
+ turns: [
46
+ { Role: 'Assistant', Text: '旧回答' },
47
+ { Role: 'User', Text: '请只回复:OK' },
48
+ ],
49
+ });
50
+ expect(__test__.diffTrustedStructuredTurns(before, current)).toEqual({
51
+ appendedTurns: [{ Role: 'User', Text: '请只回复:OK' }],
52
+ hasTrustedAppend: true,
53
+ hasNewUserTurn: true,
54
+ hasNewAssistantTurn: false,
55
+ });
56
+ });
57
+ it('treats restored structured turns as untrusted when the pre-send snapshot had no trustworthy turns', () => {
58
+ const before = snapshot({
59
+ turns: [],
60
+ transcriptLines: ['旧问题', '旧回答'],
61
+ structuredTurnsTrusted: false,
62
+ });
63
+ const current = snapshot({
64
+ turns: [
65
+ { Role: 'User', Text: '旧问题' },
66
+ { Role: 'Assistant', Text: '旧回答' },
67
+ ],
68
+ transcriptLines: ['旧问题', '旧回答'],
69
+ structuredTurnsTrusted: true,
70
+ });
71
+ expect(__test__.diffTrustedStructuredTurns(before, current)).toEqual({
72
+ appendedTurns: [],
73
+ hasTrustedAppend: false,
74
+ hasNewUserTurn: false,
75
+ hasNewAssistantTurn: false,
76
+ });
77
+ });
78
+ it('keeps transcript delta lines raw for later conservative fallback checks', () => {
79
+ const before = snapshot({
80
+ transcriptLines: ['baseline'],
81
+ });
82
+ const current = snapshot({
83
+ transcriptLines: ['baseline', '关于“请只回复:OK”,这里是解释。'],
84
+ structuredTurnsTrusted: false,
85
+ });
86
+ expect(__test__.diffTranscriptLines(before, current)).toEqual([
87
+ '关于“请只回复:OK”,这里是解释。',
88
+ ]);
89
+ });
90
+ });
91
+ describe('Gemini submission state', () => {
92
+ it('confirms submission from a trusted appended user turn', async () => {
93
+ const page = createPageMock();
94
+ const evaluate = vi.mocked(page.evaluate);
95
+ evaluate
96
+ .mockResolvedValueOnce('https://gemini.google.com/app')
97
+ .mockResolvedValueOnce({
98
+ turns: [
99
+ { Role: 'Assistant', Text: '旧回答' },
100
+ { Role: 'User', Text: '请只回复:OK' },
101
+ ],
102
+ transcriptLines: ['baseline', '请只回复:OK'],
103
+ composerHasText: false,
104
+ isGenerating: true,
105
+ structuredTurnsTrusted: true,
106
+ });
107
+ const result = await waitForGeminiSubmission(page, snapshot({
108
+ turns: [{ Role: 'Assistant', Text: '旧回答' }],
109
+ transcriptLines: ['baseline'],
110
+ composerHasText: true,
111
+ structuredTurnsTrusted: true,
112
+ }), 4);
113
+ expect(result).toEqual({
114
+ snapshot: {
115
+ turns: [
116
+ { Role: 'Assistant', Text: '旧回答' },
117
+ { Role: 'User', Text: '请只回复:OK' },
118
+ ],
119
+ transcriptLines: ['baseline', '请只回复:OK'],
120
+ composerHasText: false,
121
+ isGenerating: true,
122
+ structuredTurnsTrusted: true,
123
+ },
124
+ preSendAssistantCount: 1,
125
+ userAnchorTurn: { Role: 'User', Text: '请只回复:OK' },
126
+ reason: 'user_turn',
127
+ });
128
+ });
129
+ it('confirms submission from composer cleared plus generating even when transcript has not changed yet', async () => {
130
+ const page = createPageMock();
131
+ const evaluate = vi.mocked(page.evaluate);
132
+ evaluate
133
+ .mockResolvedValueOnce('https://gemini.google.com/app')
134
+ .mockResolvedValueOnce({
135
+ turns: [],
136
+ transcriptLines: ['baseline'],
137
+ composerHasText: false,
138
+ isGenerating: true,
139
+ structuredTurnsTrusted: false,
140
+ });
141
+ const result = await waitForGeminiSubmission(page, snapshot({
142
+ transcriptLines: ['baseline'],
143
+ composerHasText: true,
144
+ structuredTurnsTrusted: false,
145
+ }), 2);
146
+ expect(result).toEqual({
147
+ snapshot: {
148
+ turns: [],
149
+ transcriptLines: ['baseline'],
150
+ composerHasText: false,
151
+ isGenerating: true,
152
+ structuredTurnsTrusted: false,
153
+ },
154
+ preSendAssistantCount: 0,
155
+ userAnchorTurn: null,
156
+ reason: 'composer_generating',
157
+ });
158
+ });
159
+ it('confirms submission from generating state even when the pre-send baseline composer was empty', async () => {
160
+ const page = createPageMock();
161
+ const evaluate = vi.mocked(page.evaluate);
162
+ evaluate
163
+ .mockResolvedValueOnce('https://gemini.google.com/app')
164
+ .mockResolvedValueOnce({
165
+ turns: [
166
+ { Role: 'User', Text: '你说\n\n请只回复:DBG2' },
167
+ { Role: 'User', Text: '请只回复:DBG2' },
168
+ ],
169
+ transcriptLines: ['baseline', '请只回复:DBG2'],
170
+ composerHasText: false,
171
+ isGenerating: true,
172
+ structuredTurnsTrusted: true,
173
+ });
174
+ const result = await waitForGeminiSubmission(page, snapshot({
175
+ turns: [{ Role: 'Assistant', Text: '需要我为你做些什么?' }],
176
+ transcriptLines: ['baseline'],
177
+ composerHasText: false,
178
+ structuredTurnsTrusted: true,
179
+ }), 2);
180
+ expect(result).toEqual({
181
+ snapshot: {
182
+ turns: [
183
+ { Role: 'User', Text: '你说\n\n请只回复:DBG2' },
184
+ { Role: 'User', Text: '请只回复:DBG2' },
185
+ ],
186
+ transcriptLines: ['baseline', '请只回复:DBG2'],
187
+ composerHasText: false,
188
+ isGenerating: true,
189
+ structuredTurnsTrusted: true,
190
+ },
191
+ preSendAssistantCount: 1,
192
+ userAnchorTurn: { Role: 'User', Text: '请只回复:DBG2' },
193
+ reason: 'composer_generating',
194
+ });
195
+ });
196
+ it('confirms submission from composer cleared plus transcript growth when generation state is unavailable', async () => {
197
+ const page = createPageMock();
198
+ const evaluate = vi.mocked(page.evaluate);
199
+ // This transcript delta may be only a prompt echo. It is allowed to confirm
200
+ // submission only because the composer has already cleared, and it must never
201
+ // be reused later as reply ownership evidence.
202
+ evaluate
203
+ .mockResolvedValueOnce('https://gemini.google.com/app')
204
+ .mockResolvedValueOnce({
205
+ turns: [],
206
+ transcriptLines: ['baseline', '请只回复:OK'],
207
+ composerHasText: false,
208
+ isGenerating: false,
209
+ structuredTurnsTrusted: false,
210
+ });
211
+ const result = await waitForGeminiSubmission(page, snapshot({
212
+ transcriptLines: ['baseline'],
213
+ composerHasText: true,
214
+ structuredTurnsTrusted: false,
215
+ }), 2);
216
+ expect(result).toEqual({
217
+ snapshot: {
218
+ turns: [],
219
+ transcriptLines: ['baseline', '请只回复:OK'],
220
+ composerHasText: false,
221
+ isGenerating: false,
222
+ structuredTurnsTrusted: false,
223
+ },
224
+ preSendAssistantCount: 0,
225
+ userAnchorTurn: null,
226
+ reason: 'composer_transcript',
227
+ });
228
+ });
229
+ it('does not confirm submission when old structured turns only reappear after an untrusted pre-send snapshot', async () => {
230
+ const page = createPageMock();
231
+ const evaluate = vi.mocked(page.evaluate);
232
+ evaluate
233
+ .mockResolvedValueOnce('https://gemini.google.com/app')
234
+ .mockResolvedValueOnce({
235
+ turns: [
236
+ { Role: 'User', Text: '旧问题' },
237
+ { Role: 'Assistant', Text: '旧回答' },
238
+ ],
239
+ transcriptLines: ['旧问题', '旧回答'],
240
+ composerHasText: false,
241
+ isGenerating: false,
242
+ structuredTurnsTrusted: true,
243
+ })
244
+ .mockResolvedValueOnce('https://gemini.google.com/app')
245
+ .mockResolvedValueOnce({
246
+ turns: [
247
+ { Role: 'User', Text: '旧问题' },
248
+ { Role: 'Assistant', Text: '旧回答' },
249
+ ],
250
+ transcriptLines: ['旧问题', '旧回答'],
251
+ composerHasText: false,
252
+ isGenerating: false,
253
+ structuredTurnsTrusted: true,
254
+ });
255
+ const result = await waitForGeminiSubmission(page, snapshot({
256
+ turns: [],
257
+ transcriptLines: ['旧问题', '旧回答'],
258
+ composerHasText: true,
259
+ structuredTurnsTrusted: false,
260
+ }), 2);
261
+ expect(result).toBeNull();
262
+ });
263
+ it('does not confirm submission from transcript growth alone when the composer never clears', async () => {
264
+ const page = createPageMock();
265
+ const evaluate = vi.mocked(page.evaluate);
266
+ evaluate
267
+ .mockResolvedValueOnce('https://gemini.google.com/app')
268
+ .mockResolvedValueOnce({
269
+ turns: [],
270
+ transcriptLines: ['baseline', '请只回复:OK'],
271
+ composerHasText: true,
272
+ isGenerating: false,
273
+ structuredTurnsTrusted: false,
274
+ })
275
+ .mockResolvedValueOnce('https://gemini.google.com/app')
276
+ .mockResolvedValueOnce({
277
+ turns: [],
278
+ transcriptLines: ['baseline', '请只回复:OK'],
279
+ composerHasText: true,
280
+ isGenerating: false,
281
+ structuredTurnsTrusted: false,
282
+ });
283
+ const result = await waitForGeminiSubmission(page, snapshot({
284
+ transcriptLines: ['baseline'],
285
+ composerHasText: true,
286
+ structuredTurnsTrusted: false,
287
+ }), 2);
288
+ expect(result).toBeNull();
289
+ });
290
+ it('keeps polling past ten seconds when the overall timeout budget still allows submission confirmation', async () => {
291
+ const page = createPageMock();
292
+ const evaluate = vi.mocked(page.evaluate);
293
+ for (let index = 0; index < 10; index += 1) {
294
+ evaluate
295
+ .mockResolvedValueOnce('https://gemini.google.com/app')
296
+ .mockResolvedValueOnce({
297
+ turns: [],
298
+ transcriptLines: ['baseline'],
299
+ composerHasText: true,
300
+ isGenerating: false,
301
+ structuredTurnsTrusted: false,
302
+ });
303
+ }
304
+ evaluate
305
+ .mockResolvedValueOnce('https://gemini.google.com/app')
306
+ .mockResolvedValueOnce({
307
+ turns: [],
308
+ transcriptLines: ['baseline', '请只回复:OK'],
309
+ composerHasText: false,
310
+ isGenerating: false,
311
+ structuredTurnsTrusted: false,
312
+ });
313
+ const result = await waitForGeminiSubmission(page, snapshot({
314
+ transcriptLines: ['baseline'],
315
+ composerHasText: true,
316
+ structuredTurnsTrusted: false,
317
+ }), 12);
318
+ expect(result?.reason).toBe('composer_transcript');
319
+ });
320
+ });
321
+ describe('Gemini reply state', () => {
322
+ it('does not reuse an older identical reply when the submission baseline has no structured user anchor', async () => {
323
+ const page = createPageMock();
324
+ const evaluate = vi.mocked(page.evaluate);
325
+ evaluate
326
+ .mockResolvedValueOnce('https://gemini.google.com/app')
327
+ .mockResolvedValueOnce({
328
+ turns: [{ Role: 'Assistant', Text: 'OK' }],
329
+ transcriptLines: ['baseline', 'OK'],
330
+ composerHasText: false,
331
+ isGenerating: false,
332
+ structuredTurnsTrusted: true,
333
+ })
334
+ .mockResolvedValueOnce('https://gemini.google.com/app')
335
+ .mockResolvedValueOnce({
336
+ turns: [
337
+ { Role: 'Assistant', Text: 'OK' },
338
+ { Role: 'Assistant', Text: 'OK' },
339
+ ],
340
+ transcriptLines: ['baseline', 'OK'],
341
+ composerHasText: false,
342
+ isGenerating: false,
343
+ structuredTurnsTrusted: true,
344
+ })
345
+ .mockResolvedValueOnce('https://gemini.google.com/app')
346
+ .mockResolvedValueOnce({
347
+ turns: [
348
+ { Role: 'Assistant', Text: 'OK' },
349
+ { Role: 'Assistant', Text: 'OK' },
350
+ ],
351
+ transcriptLines: ['baseline', 'OK'],
352
+ composerHasText: false,
353
+ isGenerating: false,
354
+ structuredTurnsTrusted: true,
355
+ });
356
+ const result = await waitForGeminiResponse(page, {
357
+ snapshot: snapshot({
358
+ turns: [{ Role: 'Assistant', Text: 'OK' }],
359
+ transcriptLines: ['baseline', 'OK'],
360
+ composerHasText: false,
361
+ isGenerating: true,
362
+ structuredTurnsTrusted: true,
363
+ }),
364
+ preSendAssistantCount: 1,
365
+ userAnchorTurn: null,
366
+ reason: 'composer_generating',
367
+ }, '请只回复:OK', 6);
368
+ expect(result).toBe('OK');
369
+ });
370
+ it('does not treat prepended older history as the current round reply when reply ownership has no user anchor', async () => {
371
+ const page = createPageMock();
372
+ const evaluate = vi.mocked(page.evaluate);
373
+ evaluate
374
+ .mockResolvedValueOnce('https://gemini.google.com/app')
375
+ .mockResolvedValueOnce({
376
+ turns: [
377
+ { Role: 'Assistant', Text: '更早的问题' },
378
+ { Role: 'Assistant', Text: '旧回答' },
379
+ ],
380
+ transcriptLines: ['baseline'],
381
+ composerHasText: false,
382
+ isGenerating: false,
383
+ structuredTurnsTrusted: true,
384
+ })
385
+ .mockResolvedValueOnce('https://gemini.google.com/app')
386
+ .mockResolvedValueOnce({
387
+ turns: [
388
+ { Role: 'Assistant', Text: '更早的问题' },
389
+ { Role: 'Assistant', Text: '旧回答' },
390
+ ],
391
+ transcriptLines: ['baseline'],
392
+ composerHasText: false,
393
+ isGenerating: false,
394
+ structuredTurnsTrusted: true,
395
+ })
396
+ .mockResolvedValueOnce('https://gemini.google.com/app')
397
+ .mockResolvedValueOnce({
398
+ turns: [
399
+ { Role: 'Assistant', Text: '更早的问题' },
400
+ { Role: 'Assistant', Text: '旧回答' },
401
+ ],
402
+ transcriptLines: ['baseline'],
403
+ composerHasText: false,
404
+ isGenerating: false,
405
+ structuredTurnsTrusted: true,
406
+ });
407
+ const result = await waitForGeminiResponse(page, {
408
+ snapshot: snapshot({
409
+ turns: [{ Role: 'Assistant', Text: '旧回答' }],
410
+ transcriptLines: ['baseline'],
411
+ composerHasText: false,
412
+ isGenerating: true,
413
+ structuredTurnsTrusted: true,
414
+ }),
415
+ preSendAssistantCount: 1,
416
+ userAnchorTurn: null,
417
+ reason: 'composer_generating',
418
+ }, '请只回复:OK', 6);
419
+ expect(result).toBe('');
420
+ });
421
+ it('accepts a reply when the submission snapshot contains only the current round user turns and later appends a new assistant', async () => {
422
+ const page = createPageMock();
423
+ const evaluate = vi.mocked(page.evaluate);
424
+ evaluate
425
+ .mockResolvedValueOnce('https://gemini.google.com/app')
426
+ .mockResolvedValueOnce({
427
+ turns: [
428
+ { Role: 'User', Text: '你说\n\n请只回复:DBGREG' },
429
+ { Role: 'User', Text: '请只回复:DBGREG' },
430
+ { Role: 'Assistant', Text: 'DBGREG' },
431
+ ],
432
+ transcriptLines: ['baseline'],
433
+ composerHasText: false,
434
+ isGenerating: false,
435
+ structuredTurnsTrusted: true,
436
+ })
437
+ .mockResolvedValueOnce('https://gemini.google.com/app')
438
+ .mockResolvedValueOnce({
439
+ turns: [
440
+ { Role: 'User', Text: '你说\n\n请只回复:DBGREG' },
441
+ { Role: 'User', Text: '请只回复:DBGREG' },
442
+ { Role: 'Assistant', Text: 'DBGREG' },
443
+ ],
444
+ transcriptLines: ['baseline'],
445
+ composerHasText: false,
446
+ isGenerating: false,
447
+ structuredTurnsTrusted: true,
448
+ });
449
+ const result = await waitForGeminiResponse(page, {
450
+ snapshot: snapshot({
451
+ turns: [
452
+ { Role: 'User', Text: '你说\n\n请只回复:DBGREG' },
453
+ { Role: 'User', Text: '请只回复:DBGREG' },
454
+ ],
455
+ transcriptLines: ['baseline'],
456
+ composerHasText: false,
457
+ isGenerating: true,
458
+ structuredTurnsTrusted: true,
459
+ }),
460
+ preSendAssistantCount: 1,
461
+ userAnchorTurn: { Role: 'User', Text: '请只回复:DBGREG' },
462
+ reason: 'composer_generating',
463
+ }, '请只回复:DBGREG', 6);
464
+ expect(result).toBe('DBGREG');
465
+ });
466
+ it('does not trust an assistant-only submission snapshot without a stable post-submission owner', async () => {
467
+ const page = createPageMock();
468
+ const evaluate = vi.mocked(page.evaluate);
469
+ evaluate
470
+ .mockResolvedValueOnce('https://gemini.google.com/app')
471
+ .mockResolvedValueOnce({
472
+ turns: [{ Role: 'Assistant', Text: '完整回答' }],
473
+ transcriptLines: ['baseline'],
474
+ composerHasText: false,
475
+ isGenerating: false,
476
+ structuredTurnsTrusted: true,
477
+ })
478
+ .mockResolvedValueOnce('https://gemini.google.com/app')
479
+ .mockResolvedValueOnce({
480
+ turns: [{ Role: 'Assistant', Text: '完整回答' }],
481
+ transcriptLines: ['baseline'],
482
+ composerHasText: false,
483
+ isGenerating: false,
484
+ structuredTurnsTrusted: true,
485
+ });
486
+ const result = await waitForGeminiResponse(page, {
487
+ snapshot: snapshot({
488
+ turns: [{ Role: 'Assistant', Text: '半截回答' }],
489
+ transcriptLines: ['baseline'],
490
+ composerHasText: false,
491
+ isGenerating: true,
492
+ structuredTurnsTrusted: true,
493
+ }),
494
+ preSendAssistantCount: 0,
495
+ userAnchorTurn: null,
496
+ reason: 'composer_generating',
497
+ }, '请解释', 4);
498
+ expect(result).toBe('');
499
+ });
500
+ it('accepts an assistant reply that appears after a structured user anchor only after it stabilizes and generation stops', async () => {
501
+ const page = createPageMock();
502
+ const evaluate = vi.mocked(page.evaluate);
503
+ evaluate
504
+ .mockResolvedValueOnce('https://gemini.google.com/app')
505
+ .mockResolvedValueOnce({
506
+ turns: [
507
+ { Role: 'Assistant', Text: '旧回答' },
508
+ { Role: 'User', Text: '请解释' },
509
+ { Role: 'Assistant', Text: '半截回答' },
510
+ ],
511
+ transcriptLines: ['baseline'],
512
+ composerHasText: false,
513
+ isGenerating: true,
514
+ structuredTurnsTrusted: true,
515
+ })
516
+ .mockResolvedValueOnce('https://gemini.google.com/app')
517
+ .mockResolvedValueOnce({
518
+ turns: [
519
+ { Role: 'Assistant', Text: '旧回答' },
520
+ { Role: 'User', Text: '请解释' },
521
+ { Role: 'Assistant', Text: '完整回答' },
522
+ ],
523
+ transcriptLines: ['baseline'],
524
+ composerHasText: false,
525
+ isGenerating: true,
526
+ structuredTurnsTrusted: true,
527
+ })
528
+ .mockResolvedValueOnce('https://gemini.google.com/app')
529
+ .mockResolvedValueOnce({
530
+ turns: [
531
+ { Role: 'Assistant', Text: '旧回答' },
532
+ { Role: 'User', Text: '请解释' },
533
+ { Role: 'Assistant', Text: '完整回答' },
534
+ ],
535
+ transcriptLines: ['baseline'],
536
+ composerHasText: false,
537
+ isGenerating: false,
538
+ structuredTurnsTrusted: true,
539
+ });
540
+ const result = await waitForGeminiResponse(page, {
541
+ snapshot: snapshot({
542
+ turns: [
543
+ { Role: 'Assistant', Text: '旧回答' },
544
+ { Role: 'User', Text: '请解释' },
545
+ ],
546
+ transcriptLines: ['baseline'],
547
+ composerHasText: false,
548
+ isGenerating: true,
549
+ structuredTurnsTrusted: true,
550
+ }),
551
+ preSendAssistantCount: 1,
552
+ userAnchorTurn: { Role: 'User', Text: '请解释' },
553
+ reason: 'user_turn',
554
+ }, '请解释', 6);
555
+ expect(result).toBe('完整回答');
556
+ });
557
+ it('uses transcript fallback only after two identical post-submission deltas and after generation stops', async () => {
558
+ const page = createPageMock();
559
+ const evaluate = vi.mocked(page.evaluate);
560
+ evaluate
561
+ .mockResolvedValueOnce('https://gemini.google.com/app')
562
+ .mockResolvedValueOnce({
563
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
564
+ transcriptLines: ['baseline', 'OK'],
565
+ composerHasText: false,
566
+ isGenerating: true,
567
+ structuredTurnsTrusted: true,
568
+ })
569
+ .mockResolvedValueOnce('https://gemini.google.com/app')
570
+ .mockResolvedValueOnce({
571
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
572
+ transcriptLines: ['baseline', 'OK'],
573
+ composerHasText: false,
574
+ isGenerating: false,
575
+ structuredTurnsTrusted: true,
576
+ })
577
+ .mockResolvedValueOnce('https://gemini.google.com/app')
578
+ .mockResolvedValueOnce({
579
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
580
+ transcriptLines: ['baseline', 'OK'],
581
+ composerHasText: false,
582
+ isGenerating: false,
583
+ structuredTurnsTrusted: true,
584
+ });
585
+ const result = await waitForGeminiResponse(page, {
586
+ snapshot: snapshot({
587
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
588
+ transcriptLines: ['baseline'],
589
+ composerHasText: false,
590
+ isGenerating: true,
591
+ structuredTurnsTrusted: true,
592
+ }),
593
+ preSendAssistantCount: 0,
594
+ userAnchorTurn: { Role: 'User', Text: '请只回复:OK' },
595
+ reason: 'user_turn',
596
+ }, '请只回复:OK', 6);
597
+ expect(result).toBe('OK');
598
+ });
599
+ it('ignores transcript lines that appeared before submission confirmation and only accepts post-submission transcript deltas', async () => {
600
+ const page = createPageMock();
601
+ const evaluate = vi.mocked(page.evaluate);
602
+ evaluate
603
+ .mockResolvedValueOnce('https://gemini.google.com/app')
604
+ .mockResolvedValueOnce({
605
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
606
+ transcriptLines: ['baseline', '早到的提示词回声', 'OK'],
607
+ composerHasText: false,
608
+ isGenerating: false,
609
+ structuredTurnsTrusted: true,
610
+ })
611
+ .mockResolvedValueOnce('https://gemini.google.com/app')
612
+ .mockResolvedValueOnce({
613
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
614
+ transcriptLines: ['baseline', '早到的提示词回声', 'OK'],
615
+ composerHasText: false,
616
+ isGenerating: false,
617
+ structuredTurnsTrusted: true,
618
+ })
619
+ .mockResolvedValueOnce('https://gemini.google.com/app')
620
+ .mockResolvedValueOnce({
621
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
622
+ transcriptLines: ['baseline', '早到的提示词回声', 'OK'],
623
+ composerHasText: false,
624
+ isGenerating: false,
625
+ structuredTurnsTrusted: true,
626
+ });
627
+ const result = await waitForGeminiResponse(page, {
628
+ snapshot: snapshot({
629
+ turns: [{ Role: 'User', Text: '请只回复:OK' }],
630
+ transcriptLines: ['baseline', '早到的提示词回声'],
631
+ composerHasText: false,
632
+ isGenerating: true,
633
+ structuredTurnsTrusted: true,
634
+ }),
635
+ preSendAssistantCount: 0,
636
+ userAnchorTurn: { Role: 'User', Text: '请只回复:OK' },
637
+ reason: 'composer_transcript',
638
+ }, '请只回复:OK', 6);
639
+ expect(result).toBe('OK');
640
+ });
641
+ });