@office-ai/aioncli-core 0.2.2 → 0.8.1

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 (758) hide show
  1. package/dist/index.d.ts +10 -3
  2. package/dist/index.js +10 -3
  3. package/dist/index.js.map +1 -1
  4. package/dist/src/agents/codebase-investigator.d.ts +11 -0
  5. package/dist/src/agents/codebase-investigator.js +73 -0
  6. package/dist/src/agents/codebase-investigator.js.map +1 -0
  7. package/dist/src/agents/executor.d.ts +88 -0
  8. package/dist/src/agents/executor.js +417 -0
  9. package/dist/src/agents/executor.js.map +1 -0
  10. package/dist/src/agents/executor.test.js +419 -0
  11. package/dist/src/agents/executor.test.js.map +1 -0
  12. package/dist/src/agents/invocation.d.ts +43 -0
  13. package/dist/src/agents/invocation.js +100 -0
  14. package/dist/src/agents/invocation.js.map +1 -0
  15. package/dist/src/agents/invocation.test.js +206 -0
  16. package/dist/src/agents/invocation.test.js.map +1 -0
  17. package/dist/src/agents/registry.d.ts +35 -0
  18. package/dist/src/agents/registry.js +58 -0
  19. package/dist/src/agents/registry.js.map +1 -0
  20. package/dist/src/agents/registry.test.js +146 -0
  21. package/dist/src/agents/registry.test.js.map +1 -0
  22. package/dist/src/agents/schema-utils.d.ts +39 -0
  23. package/dist/src/agents/schema-utils.js +57 -0
  24. package/dist/src/agents/schema-utils.js.map +1 -0
  25. package/dist/src/agents/schema-utils.test.d.ts +6 -0
  26. package/dist/src/agents/schema-utils.test.js +144 -0
  27. package/dist/src/agents/schema-utils.test.js.map +1 -0
  28. package/dist/src/agents/subagent-tool-wrapper.d.ts +36 -0
  29. package/dist/src/agents/subagent-tool-wrapper.js +47 -0
  30. package/dist/src/agents/subagent-tool-wrapper.js.map +1 -0
  31. package/dist/src/agents/subagent-tool-wrapper.test.d.ts +6 -0
  32. package/dist/src/agents/subagent-tool-wrapper.test.js +105 -0
  33. package/dist/src/agents/subagent-tool-wrapper.test.js.map +1 -0
  34. package/dist/src/agents/types.d.ts +116 -0
  35. package/dist/src/agents/types.js +17 -0
  36. package/dist/src/agents/types.js.map +1 -0
  37. package/dist/src/agents/utils.d.ts +15 -0
  38. package/dist/src/agents/utils.js +29 -0
  39. package/dist/src/agents/utils.js.map +1 -0
  40. package/dist/src/agents/utils.test.d.ts +6 -0
  41. package/dist/src/agents/utils.test.js +87 -0
  42. package/dist/src/agents/utils.test.js.map +1 -0
  43. package/dist/src/code_assist/codeAssist.d.ts +6 -3
  44. package/dist/src/code_assist/codeAssist.js +12 -0
  45. package/dist/src/code_assist/codeAssist.js.map +1 -1
  46. package/dist/src/code_assist/converter.d.ts +4 -1
  47. package/dist/src/code_assist/converter.js +38 -5
  48. package/dist/src/code_assist/converter.js.map +1 -1
  49. package/dist/src/code_assist/converter.test.js +93 -0
  50. package/dist/src/code_assist/converter.test.js.map +1 -1
  51. package/dist/src/code_assist/oauth-credential-storage.d.ts +25 -0
  52. package/dist/src/code_assist/oauth-credential-storage.js +109 -0
  53. package/dist/src/code_assist/oauth-credential-storage.js.map +1 -0
  54. package/dist/src/code_assist/oauth-credential-storage.test.d.ts +6 -0
  55. package/dist/src/code_assist/oauth-credential-storage.test.js +136 -0
  56. package/dist/src/code_assist/oauth-credential-storage.test.js.map +1 -0
  57. package/dist/src/code_assist/oauth2.d.ts +1 -1
  58. package/dist/src/code_assist/oauth2.js +107 -48
  59. package/dist/src/code_assist/oauth2.js.map +1 -1
  60. package/dist/src/code_assist/oauth2.test.js +735 -343
  61. package/dist/src/code_assist/oauth2.test.js.map +1 -1
  62. package/dist/src/code_assist/server.d.ts +4 -4
  63. package/dist/src/code_assist/server.js +25 -2
  64. package/dist/src/code_assist/server.js.map +1 -1
  65. package/dist/src/code_assist/server.test.js +25 -0
  66. package/dist/src/code_assist/server.test.js.map +1 -1
  67. package/dist/src/code_assist/setup.d.ts +1 -1
  68. package/dist/src/code_assist/setup.js +1 -1
  69. package/dist/src/code_assist/setup.js.map +1 -1
  70. package/dist/src/code_assist/setup.test.js.map +1 -1
  71. package/dist/src/code_assist/types.d.ts +17 -2
  72. package/dist/src/config/config.d.ts +121 -25
  73. package/dist/src/config/config.js +298 -87
  74. package/dist/src/config/config.js.map +1 -1
  75. package/dist/src/config/config.test.js +370 -131
  76. package/dist/src/config/config.test.js.map +1 -1
  77. package/dist/src/config/constants.d.ts +11 -0
  78. package/dist/src/config/constants.js +16 -0
  79. package/dist/src/config/constants.js.map +1 -0
  80. package/dist/src/config/models.d.ts +16 -0
  81. package/dist/src/config/models.js +29 -0
  82. package/dist/src/config/models.js.map +1 -1
  83. package/dist/src/config/models.test.d.ts +6 -0
  84. package/dist/src/config/models.test.js +55 -0
  85. package/dist/src/config/models.test.js.map +1 -0
  86. package/dist/src/config/storage.d.ts +34 -0
  87. package/dist/src/config/storage.js +95 -0
  88. package/dist/src/config/storage.js.map +1 -0
  89. package/dist/src/config/storage.test.d.ts +6 -0
  90. package/dist/src/config/storage.test.js +47 -0
  91. package/dist/src/config/storage.test.js.map +1 -0
  92. package/dist/src/confirmation-bus/index.d.ts +7 -0
  93. package/dist/src/confirmation-bus/index.js +8 -0
  94. package/dist/src/confirmation-bus/index.js.map +1 -0
  95. package/dist/src/confirmation-bus/message-bus.d.ts +17 -0
  96. package/dist/src/confirmation-bus/message-bus.js +81 -0
  97. package/dist/src/confirmation-bus/message-bus.js.map +1 -0
  98. package/dist/src/confirmation-bus/message-bus.test.d.ts +6 -0
  99. package/dist/src/confirmation-bus/message-bus.test.js +164 -0
  100. package/dist/src/confirmation-bus/message-bus.test.js.map +1 -0
  101. package/dist/src/confirmation-bus/types.d.ts +38 -0
  102. package/dist/src/confirmation-bus/types.js +15 -0
  103. package/dist/src/confirmation-bus/types.js.map +1 -0
  104. package/dist/src/core/baseLlmClient.d.ts +54 -0
  105. package/dist/src/core/baseLlmClient.js +190 -0
  106. package/dist/src/core/baseLlmClient.js.map +1 -0
  107. package/dist/src/core/baseLlmClient.test.d.ts +6 -0
  108. package/dist/src/core/baseLlmClient.test.js +316 -0
  109. package/dist/src/core/baseLlmClient.test.js.map +1 -0
  110. package/dist/src/core/client.d.ts +28 -28
  111. package/dist/src/core/client.js +187 -333
  112. package/dist/src/core/client.js.map +1 -1
  113. package/dist/src/core/client.test.js +745 -500
  114. package/dist/src/core/client.test.js.map +1 -1
  115. package/dist/src/core/contentGenerator.d.ts +4 -4
  116. package/dist/src/core/contentGenerator.js +6 -7
  117. package/dist/src/core/contentGenerator.js.map +1 -1
  118. package/dist/src/core/contentGenerator.test.js +1 -3
  119. package/dist/src/core/contentGenerator.test.js.map +1 -1
  120. package/dist/src/core/coreToolScheduler.d.ts +20 -7
  121. package/dist/src/core/coreToolScheduler.js +216 -53
  122. package/dist/src/core/coreToolScheduler.js.map +1 -1
  123. package/dist/src/core/coreToolScheduler.test.js +564 -88
  124. package/dist/src/core/coreToolScheduler.test.js.map +1 -1
  125. package/dist/src/core/geminiChat.d.ts +54 -43
  126. package/dist/src/core/geminiChat.js +298 -280
  127. package/dist/src/core/geminiChat.js.map +1 -1
  128. package/dist/src/core/geminiChat.test.js +1255 -321
  129. package/dist/src/core/geminiChat.test.js.map +1 -1
  130. package/dist/src/core/geminiRequest.js +1 -0
  131. package/dist/src/core/geminiRequest.js.map +1 -1
  132. package/dist/src/core/logger.d.ts +4 -2
  133. package/dist/src/core/logger.js +4 -3
  134. package/dist/src/core/logger.js.map +1 -1
  135. package/dist/src/core/logger.test.js +17 -16
  136. package/dist/src/core/logger.test.js.map +1 -1
  137. package/dist/src/core/loggingContentGenerator.d.ts +3 -3
  138. package/dist/src/core/loggingContentGenerator.js +15 -16
  139. package/dist/src/core/loggingContentGenerator.js.map +1 -1
  140. package/dist/src/core/nonInteractiveToolExecutor.d.ts +3 -5
  141. package/dist/src/core/nonInteractiveToolExecutor.js +14 -122
  142. package/dist/src/core/nonInteractiveToolExecutor.js.map +1 -1
  143. package/dist/src/core/nonInteractiveToolExecutor.test.js +158 -78
  144. package/dist/src/core/nonInteractiveToolExecutor.test.js.map +1 -1
  145. package/dist/src/core/openaiContentGenerator.d.ts +4 -3
  146. package/dist/src/core/openaiContentGenerator.js +21 -14
  147. package/dist/src/core/openaiContentGenerator.js.map +1 -1
  148. package/dist/src/core/openaiContentGenerator.test.js +1 -0
  149. package/dist/src/core/openaiContentGenerator.test.js.map +1 -1
  150. package/dist/src/core/prompts.d.ts +5 -0
  151. package/dist/src/core/prompts.js +66 -44
  152. package/dist/src/core/prompts.js.map +1 -1
  153. package/dist/src/core/prompts.test.js +130 -1
  154. package/dist/src/core/prompts.test.js.map +1 -1
  155. package/dist/src/core/subagent.d.ts +24 -18
  156. package/dist/src/core/subagent.js +125 -90
  157. package/dist/src/core/subagent.js.map +1 -1
  158. package/dist/src/core/subagent.test.js +59 -44
  159. package/dist/src/core/subagent.test.js.map +1 -1
  160. package/dist/src/core/turn.d.ts +37 -13
  161. package/dist/src/core/turn.js +63 -28
  162. package/dist/src/core/turn.js.map +1 -1
  163. package/dist/src/core/turn.test.js +359 -100
  164. package/dist/src/core/turn.test.js.map +1 -1
  165. package/dist/src/fallback/handler.d.ts +7 -0
  166. package/dist/src/fallback/handler.js +129 -0
  167. package/dist/src/fallback/handler.js.map +1 -0
  168. package/dist/src/fallback/handler.test.d.ts +6 -0
  169. package/dist/src/fallback/handler.test.js +130 -0
  170. package/dist/src/fallback/handler.test.js.map +1 -0
  171. package/dist/src/fallback/types.d.ts +14 -0
  172. package/dist/src/fallback/types.js +7 -0
  173. package/dist/src/fallback/types.js.map +1 -0
  174. package/dist/src/generated/git-commit.d.ts +1 -1
  175. package/dist/src/generated/git-commit.js +1 -1
  176. package/dist/src/ide/constants.d.ts +3 -0
  177. package/dist/src/ide/constants.js +3 -0
  178. package/dist/src/ide/constants.js.map +1 -1
  179. package/dist/src/ide/detect-ide.d.ts +48 -12
  180. package/dist/src/ide/detect-ide.js +47 -66
  181. package/dist/src/ide/detect-ide.js.map +1 -1
  182. package/dist/src/ide/detect-ide.test.js +79 -52
  183. package/dist/src/ide/detect-ide.test.js.map +1 -1
  184. package/dist/src/ide/ide-client.d.ts +69 -23
  185. package/dist/src/ide/ide-client.js +372 -78
  186. package/dist/src/ide/ide-client.js.map +1 -1
  187. package/dist/src/ide/ide-client.test.js +375 -30
  188. package/dist/src/ide/ide-client.test.js.map +1 -1
  189. package/dist/src/ide/ide-installer.d.ts +2 -2
  190. package/dist/src/ide/ide-installer.js +37 -24
  191. package/dist/src/ide/ide-installer.js.map +1 -1
  192. package/dist/src/ide/ide-installer.test.js +104 -26
  193. package/dist/src/ide/ide-installer.test.js.map +1 -1
  194. package/dist/src/ide/ideContext.d.ts +35 -365
  195. package/dist/src/ide/ideContext.js +60 -106
  196. package/dist/src/ide/ideContext.js.map +1 -1
  197. package/dist/src/ide/ideContext.test.js +152 -24
  198. package/dist/src/ide/ideContext.test.js.map +1 -1
  199. package/dist/src/ide/process-utils.d.ts +7 -5
  200. package/dist/src/ide/process-utils.js +81 -50
  201. package/dist/src/ide/process-utils.js.map +1 -1
  202. package/dist/src/ide/process-utils.test.d.ts +6 -0
  203. package/dist/src/ide/process-utils.test.js +158 -0
  204. package/dist/src/ide/process-utils.test.js.map +1 -0
  205. package/dist/src/ide/types.d.ts +486 -0
  206. package/dist/src/ide/types.js +138 -0
  207. package/dist/src/ide/types.js.map +1 -0
  208. package/dist/src/index.d.ts +20 -2
  209. package/dist/src/index.js +20 -2
  210. package/dist/src/index.js.map +1 -1
  211. package/dist/src/mcp/google-auth-provider.d.ts +3 -3
  212. package/dist/src/mcp/google-auth-provider.test.js.map +1 -1
  213. package/dist/src/mcp/oauth-provider.d.ts +17 -13
  214. package/dist/src/mcp/oauth-provider.js +81 -69
  215. package/dist/src/mcp/oauth-provider.js.map +1 -1
  216. package/dist/src/mcp/oauth-provider.test.js +212 -37
  217. package/dist/src/mcp/oauth-provider.test.js.map +1 -1
  218. package/dist/src/mcp/oauth-token-storage.d.ts +14 -32
  219. package/dist/src/mcp/oauth-token-storage.js +54 -25
  220. package/dist/src/mcp/oauth-token-storage.js.map +1 -1
  221. package/dist/src/mcp/oauth-token-storage.test.js +256 -162
  222. package/dist/src/mcp/oauth-token-storage.test.js.map +1 -1
  223. package/dist/src/mcp/oauth-utils.d.ts +9 -1
  224. package/dist/src/mcp/oauth-utils.js +42 -27
  225. package/dist/src/mcp/oauth-utils.js.map +1 -1
  226. package/dist/src/mcp/oauth-utils.test.js +41 -1
  227. package/dist/src/mcp/oauth-utils.test.js.map +1 -1
  228. package/dist/src/mcp/sa-impersonation-provider.d.ts +33 -0
  229. package/dist/src/mcp/sa-impersonation-provider.js +130 -0
  230. package/dist/src/mcp/sa-impersonation-provider.js.map +1 -0
  231. package/dist/src/mcp/sa-impersonation-provider.test.d.ts +6 -0
  232. package/dist/src/mcp/sa-impersonation-provider.test.js +117 -0
  233. package/dist/src/mcp/sa-impersonation-provider.test.js.map +1 -0
  234. package/dist/src/mcp/token-storage/base-token-storage.d.ts +19 -0
  235. package/dist/src/mcp/token-storage/base-token-storage.js +36 -0
  236. package/dist/src/mcp/token-storage/base-token-storage.js.map +1 -0
  237. package/dist/src/mcp/token-storage/base-token-storage.test.d.ts +6 -0
  238. package/dist/src/mcp/token-storage/base-token-storage.test.js +160 -0
  239. package/dist/src/mcp/token-storage/base-token-storage.test.js.map +1 -0
  240. package/dist/src/mcp/token-storage/file-token-storage.d.ts +24 -0
  241. package/dist/src/mcp/token-storage/file-token-storage.js +144 -0
  242. package/dist/src/mcp/token-storage/file-token-storage.js.map +1 -0
  243. package/dist/src/mcp/token-storage/file-token-storage.test.d.ts +6 -0
  244. package/dist/src/mcp/token-storage/file-token-storage.test.js +235 -0
  245. package/dist/src/mcp/token-storage/file-token-storage.test.js.map +1 -0
  246. package/dist/src/mcp/token-storage/hybrid-token-storage.d.ts +23 -0
  247. package/dist/src/mcp/token-storage/hybrid-token-storage.js +78 -0
  248. package/dist/src/mcp/token-storage/hybrid-token-storage.js.map +1 -0
  249. package/dist/src/mcp/token-storage/hybrid-token-storage.test.d.ts +6 -0
  250. package/dist/src/mcp/token-storage/hybrid-token-storage.test.js +193 -0
  251. package/dist/src/mcp/token-storage/hybrid-token-storage.test.js.map +1 -0
  252. package/dist/src/mcp/token-storage/index.d.ts +11 -0
  253. package/dist/src/mcp/token-storage/index.js +12 -0
  254. package/dist/src/mcp/token-storage/index.js.map +1 -0
  255. package/dist/src/mcp/token-storage/keychain-token-storage.d.ts +31 -0
  256. package/dist/src/mcp/token-storage/keychain-token-storage.js +190 -0
  257. package/dist/src/mcp/token-storage/keychain-token-storage.js.map +1 -0
  258. package/dist/src/mcp/token-storage/keychain-token-storage.test.d.ts +6 -0
  259. package/dist/src/mcp/token-storage/keychain-token-storage.test.js +254 -0
  260. package/dist/src/mcp/token-storage/keychain-token-storage.test.js.map +1 -0
  261. package/dist/src/mcp/token-storage/types.d.ts +38 -0
  262. package/dist/src/mcp/token-storage/types.js +11 -0
  263. package/dist/src/mcp/token-storage/types.js.map +1 -0
  264. package/dist/src/output/json-formatter.d.ts +11 -0
  265. package/dist/src/output/json-formatter.js +30 -0
  266. package/dist/src/output/json-formatter.js.map +1 -0
  267. package/dist/src/output/json-formatter.test.d.ts +6 -0
  268. package/dist/src/output/json-formatter.test.js +266 -0
  269. package/dist/src/output/json-formatter.test.js.map +1 -0
  270. package/dist/src/output/types.d.ts +20 -0
  271. package/dist/src/output/types.js +11 -0
  272. package/dist/src/output/types.js.map +1 -0
  273. package/dist/src/policy/index.d.ts +7 -0
  274. package/dist/src/policy/index.js +8 -0
  275. package/dist/src/policy/index.js.map +1 -0
  276. package/dist/src/policy/policy-engine.d.ts +30 -0
  277. package/dist/src/policy/policy-engine.js +92 -0
  278. package/dist/src/policy/policy-engine.js.map +1 -0
  279. package/dist/src/policy/policy-engine.test.d.ts +6 -0
  280. package/dist/src/policy/policy-engine.test.js +515 -0
  281. package/dist/src/policy/policy-engine.test.js.map +1 -0
  282. package/dist/src/policy/stable-stringify.d.ts +58 -0
  283. package/dist/src/policy/stable-stringify.js +122 -0
  284. package/dist/src/policy/stable-stringify.js.map +1 -0
  285. package/dist/src/policy/types.d.ts +47 -0
  286. package/dist/src/policy/types.js +12 -0
  287. package/dist/src/policy/types.js.map +1 -0
  288. package/dist/src/prompts/mcp-prompts.d.ts +2 -2
  289. package/dist/src/prompts/prompt-registry.d.ts +1 -1
  290. package/dist/src/routing/modelRouterService.d.ts +23 -0
  291. package/dist/src/routing/modelRouterService.js +70 -0
  292. package/dist/src/routing/modelRouterService.js.map +1 -0
  293. package/dist/src/routing/modelRouterService.test.d.ts +6 -0
  294. package/dist/src/routing/modelRouterService.test.js +98 -0
  295. package/dist/src/routing/modelRouterService.test.js.map +1 -0
  296. package/dist/src/routing/routingStrategy.d.ts +62 -0
  297. package/dist/src/routing/routingStrategy.js +7 -0
  298. package/dist/src/routing/routingStrategy.js.map +1 -0
  299. package/dist/src/routing/strategies/classifierStrategy.d.ts +12 -0
  300. package/dist/src/routing/strategies/classifierStrategy.js +173 -0
  301. package/dist/src/routing/strategies/classifierStrategy.js.map +1 -0
  302. package/dist/src/routing/strategies/classifierStrategy.test.d.ts +6 -0
  303. package/dist/src/routing/strategies/classifierStrategy.test.js +192 -0
  304. package/dist/src/routing/strategies/classifierStrategy.test.js.map +1 -0
  305. package/dist/src/routing/strategies/compositeStrategy.d.ts +26 -0
  306. package/dist/src/routing/strategies/compositeStrategy.js +68 -0
  307. package/dist/src/routing/strategies/compositeStrategy.js.map +1 -0
  308. package/dist/src/routing/strategies/compositeStrategy.test.d.ts +6 -0
  309. package/dist/src/routing/strategies/compositeStrategy.test.js +123 -0
  310. package/dist/src/routing/strategies/compositeStrategy.test.js.map +1 -0
  311. package/dist/src/routing/strategies/defaultStrategy.d.ts +12 -0
  312. package/dist/src/routing/strategies/defaultStrategy.js +20 -0
  313. package/dist/src/routing/strategies/defaultStrategy.js.map +1 -0
  314. package/dist/src/routing/strategies/defaultStrategy.test.d.ts +6 -0
  315. package/dist/src/routing/strategies/defaultStrategy.test.js +26 -0
  316. package/dist/src/routing/strategies/defaultStrategy.test.js.map +1 -0
  317. package/dist/src/routing/strategies/fallbackStrategy.d.ts +12 -0
  318. package/dist/src/routing/strategies/fallbackStrategy.js +25 -0
  319. package/dist/src/routing/strategies/fallbackStrategy.js.map +1 -0
  320. package/dist/src/routing/strategies/fallbackStrategy.test.d.ts +6 -0
  321. package/dist/src/routing/strategies/fallbackStrategy.test.js +55 -0
  322. package/dist/src/routing/strategies/fallbackStrategy.test.js.map +1 -0
  323. package/dist/src/routing/strategies/overrideStrategy.d.ts +15 -0
  324. package/dist/src/routing/strategies/overrideStrategy.js +28 -0
  325. package/dist/src/routing/strategies/overrideStrategy.js.map +1 -0
  326. package/dist/src/routing/strategies/overrideStrategy.test.d.ts +6 -0
  327. package/dist/src/routing/strategies/overrideStrategy.test.js +42 -0
  328. package/dist/src/routing/strategies/overrideStrategy.test.js.map +1 -0
  329. package/dist/src/services/chatRecordingService.d.ts +8 -14
  330. package/dist/src/services/chatRecordingService.js +33 -21
  331. package/dist/src/services/chatRecordingService.js.map +1 -1
  332. package/dist/src/services/chatRecordingService.test.js +69 -25
  333. package/dist/src/services/chatRecordingService.test.js.map +1 -1
  334. package/dist/src/services/fileDiscoveryService.d.ts +10 -0
  335. package/dist/src/services/fileDiscoveryService.js +32 -18
  336. package/dist/src/services/fileDiscoveryService.js.map +1 -1
  337. package/dist/src/services/fileDiscoveryService.test.js +3 -3
  338. package/dist/src/services/fileDiscoveryService.test.js.map +1 -1
  339. package/dist/src/services/fileSystemService.d.ts +9 -0
  340. package/dist/src/services/fileSystemService.js +12 -1
  341. package/dist/src/services/fileSystemService.js.map +1 -1
  342. package/dist/src/services/fileSystemService.test.js +1 -1
  343. package/dist/src/services/fileSystemService.test.js.map +1 -1
  344. package/dist/src/services/gitService.d.ts +3 -1
  345. package/dist/src/services/gitService.js +30 -24
  346. package/dist/src/services/gitService.js.map +1 -1
  347. package/dist/src/services/gitService.test.js +30 -37
  348. package/dist/src/services/gitService.test.js.map +1 -1
  349. package/dist/src/services/loopDetectionService.d.ts +8 -2
  350. package/dist/src/services/loopDetectionService.js +64 -24
  351. package/dist/src/services/loopDetectionService.js.map +1 -1
  352. package/dist/src/services/loopDetectionService.test.js +64 -13
  353. package/dist/src/services/loopDetectionService.test.js.map +1 -1
  354. package/dist/src/services/shellExecutionService.d.ts +36 -2
  355. package/dist/src/services/shellExecutionService.js +238 -47
  356. package/dist/src/services/shellExecutionService.js.map +1 -1
  357. package/dist/src/services/shellExecutionService.test.js +197 -58
  358. package/dist/src/services/shellExecutionService.test.js.map +1 -1
  359. package/dist/src/telemetry/activity-detector.d.ts +41 -0
  360. package/dist/src/telemetry/activity-detector.js +61 -0
  361. package/dist/src/telemetry/activity-detector.js.map +1 -0
  362. package/dist/src/telemetry/activity-detector.test.d.ts +6 -0
  363. package/dist/src/telemetry/activity-detector.test.js +136 -0
  364. package/dist/src/telemetry/activity-detector.test.js.map +1 -0
  365. package/dist/src/telemetry/activity-types.d.ts +19 -0
  366. package/dist/src/telemetry/activity-types.js +21 -0
  367. package/dist/src/telemetry/activity-types.js.map +1 -0
  368. package/dist/src/telemetry/clearcut-logger/clearcut-logger.d.ts +34 -4
  369. package/dist/src/telemetry/clearcut-logger/clearcut-logger.js +322 -15
  370. package/dist/src/telemetry/clearcut-logger/clearcut-logger.js.map +1 -1
  371. package/dist/src/telemetry/clearcut-logger/clearcut-logger.test.d.ts +1 -0
  372. package/dist/src/telemetry/clearcut-logger/clearcut-logger.test.js +321 -11
  373. package/dist/src/telemetry/clearcut-logger/clearcut-logger.test.js.map +1 -1
  374. package/dist/src/telemetry/clearcut-logger/event-metadata-key.d.ts +51 -2
  375. package/dist/src/telemetry/clearcut-logger/event-metadata-key.js +124 -2
  376. package/dist/src/telemetry/clearcut-logger/event-metadata-key.js.map +1 -1
  377. package/dist/src/telemetry/config.d.ts +31 -0
  378. package/dist/src/telemetry/config.js +76 -0
  379. package/dist/src/telemetry/config.js.map +1 -0
  380. package/dist/src/telemetry/config.test.d.ts +6 -0
  381. package/dist/src/telemetry/config.test.js +124 -0
  382. package/dist/src/telemetry/config.test.js.map +1 -0
  383. package/dist/src/telemetry/constants.d.ts +17 -7
  384. package/dist/src/telemetry/constants.js +18 -7
  385. package/dist/src/telemetry/constants.js.map +1 -1
  386. package/dist/src/telemetry/file-exporters.d.ts +5 -4
  387. package/dist/src/telemetry/file-exporters.js +1 -1
  388. package/dist/src/telemetry/file-exporters.js.map +1 -1
  389. package/dist/src/telemetry/gcp-exporters.d.ts +34 -0
  390. package/dist/src/telemetry/gcp-exporters.js +117 -0
  391. package/dist/src/telemetry/gcp-exporters.js.map +1 -0
  392. package/dist/src/telemetry/gcp-exporters.test.d.ts +6 -0
  393. package/dist/src/telemetry/gcp-exporters.test.js +318 -0
  394. package/dist/src/telemetry/gcp-exporters.test.js.map +1 -0
  395. package/dist/src/telemetry/high-water-mark-tracker.d.ts +43 -0
  396. package/dist/src/telemetry/high-water-mark-tracker.js +88 -0
  397. package/dist/src/telemetry/high-water-mark-tracker.js.map +1 -0
  398. package/dist/src/telemetry/high-water-mark-tracker.test.d.ts +6 -0
  399. package/dist/src/telemetry/high-water-mark-tracker.test.js +152 -0
  400. package/dist/src/telemetry/high-water-mark-tracker.test.js.map +1 -0
  401. package/dist/src/telemetry/index.d.ts +12 -2
  402. package/dist/src/telemetry/index.js +16 -2
  403. package/dist/src/telemetry/index.js.map +1 -1
  404. package/dist/src/telemetry/loggers.d.ts +17 -2
  405. package/dist/src/telemetry/loggers.js +316 -14
  406. package/dist/src/telemetry/loggers.js.map +1 -1
  407. package/dist/src/telemetry/loggers.test.circular.js +3 -3
  408. package/dist/src/telemetry/loggers.test.circular.js.map +1 -1
  409. package/dist/src/telemetry/loggers.test.js +452 -48
  410. package/dist/src/telemetry/loggers.test.js.map +1 -1
  411. package/dist/src/telemetry/metrics.d.ts +323 -12
  412. package/dist/src/telemetry/metrics.js +464 -83
  413. package/dist/src/telemetry/metrics.js.map +1 -1
  414. package/dist/src/telemetry/metrics.test.js +583 -38
  415. package/dist/src/telemetry/metrics.test.js.map +1 -1
  416. package/dist/src/telemetry/rate-limiter.d.ts +48 -0
  417. package/dist/src/telemetry/rate-limiter.js +100 -0
  418. package/dist/src/telemetry/rate-limiter.js.map +1 -0
  419. package/dist/src/telemetry/rate-limiter.test.d.ts +6 -0
  420. package/dist/src/telemetry/rate-limiter.test.js +207 -0
  421. package/dist/src/telemetry/rate-limiter.test.js.map +1 -0
  422. package/dist/src/telemetry/sdk.d.ts +1 -1
  423. package/dist/src/telemetry/sdk.js +20 -2
  424. package/dist/src/telemetry/sdk.js.map +1 -1
  425. package/dist/src/telemetry/sdk.test.js +108 -0
  426. package/dist/src/telemetry/sdk.test.js.map +1 -1
  427. package/dist/src/telemetry/telemetry-utils.d.ts +6 -0
  428. package/dist/src/telemetry/telemetry-utils.js +14 -0
  429. package/dist/src/telemetry/telemetry-utils.js.map +1 -0
  430. package/dist/src/telemetry/telemetry-utils.test.d.ts +6 -0
  431. package/dist/src/telemetry/telemetry-utils.test.js +40 -0
  432. package/dist/src/telemetry/telemetry-utils.test.js.map +1 -0
  433. package/dist/src/telemetry/types.d.ts +136 -8
  434. package/dist/src/telemetry/types.js +233 -11
  435. package/dist/src/telemetry/types.js.map +1 -1
  436. package/dist/src/telemetry/uiTelemetry.d.ts +3 -3
  437. package/dist/src/telemetry/uiTelemetry.js +7 -8
  438. package/dist/src/telemetry/uiTelemetry.js.map +1 -1
  439. package/dist/src/telemetry/uiTelemetry.test.js +33 -29
  440. package/dist/src/telemetry/uiTelemetry.test.js.map +1 -1
  441. package/dist/src/test-utils/config.d.ts +2 -1
  442. package/dist/src/test-utils/config.js.map +1 -1
  443. package/dist/src/test-utils/index.d.ts +6 -0
  444. package/dist/src/test-utils/index.js +7 -0
  445. package/dist/src/test-utils/index.js.map +1 -0
  446. package/dist/src/test-utils/mock-tool.d.ts +66 -0
  447. package/dist/src/test-utils/{tools.js → mock-tool.js} +45 -29
  448. package/dist/src/test-utils/mock-tool.js.map +1 -0
  449. package/dist/src/test-utils/mockWorkspaceContext.d.ts +1 -1
  450. package/dist/src/tools/diffOptions.d.ts +1 -1
  451. package/dist/src/tools/diffOptions.js +21 -13
  452. package/dist/src/tools/diffOptions.js.map +1 -1
  453. package/dist/src/tools/diffOptions.test.js +58 -22
  454. package/dist/src/tools/diffOptions.test.js.map +1 -1
  455. package/dist/src/tools/edit.d.ts +6 -5
  456. package/dist/src/tools/edit.js +58 -40
  457. package/dist/src/tools/edit.js.map +1 -1
  458. package/dist/src/tools/edit.test.js +192 -16
  459. package/dist/src/tools/edit.test.js.map +1 -1
  460. package/dist/src/tools/glob.d.ts +7 -2
  461. package/dist/src/tools/glob.js +42 -23
  462. package/dist/src/tools/glob.js.map +1 -1
  463. package/dist/src/tools/glob.test.js +80 -4
  464. package/dist/src/tools/glob.test.js.map +1 -1
  465. package/dist/src/tools/grep.d.ts +3 -2
  466. package/dist/src/tools/grep.js +35 -15
  467. package/dist/src/tools/grep.js.map +1 -1
  468. package/dist/src/tools/grep.test.js +26 -3
  469. package/dist/src/tools/grep.test.js.map +1 -1
  470. package/dist/src/tools/ls.d.ts +3 -2
  471. package/dist/src/tools/ls.js +31 -39
  472. package/dist/src/tools/ls.js.map +1 -1
  473. package/dist/src/tools/ls.test.js +145 -280
  474. package/dist/src/tools/ls.test.js.map +1 -1
  475. package/dist/src/tools/mcp-client-manager.d.ts +8 -6
  476. package/dist/src/tools/mcp-client-manager.js +13 -4
  477. package/dist/src/tools/mcp-client-manager.js.map +1 -1
  478. package/dist/src/tools/mcp-client-manager.test.js +20 -1
  479. package/dist/src/tools/mcp-client-manager.test.js.map +1 -1
  480. package/dist/src/tools/mcp-client.d.ts +18 -21
  481. package/dist/src/tools/mcp-client.js +87 -120
  482. package/dist/src/tools/mcp-client.js.map +1 -1
  483. package/dist/src/tools/mcp-client.test.js +32 -152
  484. package/dist/src/tools/mcp-client.test.js.map +1 -1
  485. package/dist/src/tools/mcp-tool.d.ts +6 -4
  486. package/dist/src/tools/mcp-tool.js +51 -13
  487. package/dist/src/tools/mcp-tool.js.map +1 -1
  488. package/dist/src/tools/mcp-tool.test.js +166 -12
  489. package/dist/src/tools/mcp-tool.test.js.map +1 -1
  490. package/dist/src/tools/memoryTool.d.ts +3 -2
  491. package/dist/src/tools/memoryTool.js +14 -37
  492. package/dist/src/tools/memoryTool.js.map +1 -1
  493. package/dist/src/tools/memoryTool.test.js +16 -4
  494. package/dist/src/tools/memoryTool.test.js.map +1 -1
  495. package/dist/src/tools/message-bus-integration.test.d.ts +6 -0
  496. package/dist/src/tools/message-bus-integration.test.js +183 -0
  497. package/dist/src/tools/message-bus-integration.test.js.map +1 -0
  498. package/dist/src/tools/modifiable-tool.d.ts +2 -2
  499. package/dist/src/tools/modifiable-tool.js +3 -3
  500. package/dist/src/tools/modifiable-tool.js.map +1 -1
  501. package/dist/src/tools/modifiable-tool.test.js +4 -4
  502. package/dist/src/tools/modifiable-tool.test.js.map +1 -1
  503. package/dist/src/tools/read-file.d.ts +3 -2
  504. package/dist/src/tools/read-file.js +33 -43
  505. package/dist/src/tools/read-file.js.map +1 -1
  506. package/dist/src/tools/read-file.test.js +39 -7
  507. package/dist/src/tools/read-file.test.js.map +1 -1
  508. package/dist/src/tools/read-many-files.d.ts +3 -2
  509. package/dist/src/tools/read-many-files.js +52 -107
  510. package/dist/src/tools/read-many-files.js.map +1 -1
  511. package/dist/src/tools/read-many-files.test.js +64 -11
  512. package/dist/src/tools/read-many-files.test.js.map +1 -1
  513. package/dist/src/tools/ripGrep.d.ts +55 -0
  514. package/dist/src/tools/ripGrep.js +393 -0
  515. package/dist/src/tools/ripGrep.js.map +1 -0
  516. package/dist/src/tools/ripGrep.test.d.ts +6 -0
  517. package/dist/src/tools/ripGrep.test.js +976 -0
  518. package/dist/src/tools/ripGrep.test.js.map +1 -0
  519. package/dist/src/tools/shell.d.ts +13 -2
  520. package/dist/src/tools/shell.js +42 -32
  521. package/dist/src/tools/shell.js.map +1 -1
  522. package/dist/src/tools/shell.test.js +57 -75
  523. package/dist/src/tools/shell.test.js.map +1 -1
  524. package/dist/src/tools/smart-edit.d.ts +91 -0
  525. package/dist/src/tools/smart-edit.js +702 -0
  526. package/dist/src/tools/smart-edit.js.map +1 -0
  527. package/dist/src/tools/smart-edit.test.d.ts +6 -0
  528. package/dist/src/tools/smart-edit.test.js +542 -0
  529. package/dist/src/tools/smart-edit.test.js.map +1 -0
  530. package/dist/src/tools/tool-error.d.ts +18 -1
  531. package/dist/src/tools/tool-error.js +27 -0
  532. package/dist/src/tools/tool-error.js.map +1 -1
  533. package/dist/src/tools/tool-registry.d.ts +10 -4
  534. package/dist/src/tools/tool-registry.js +20 -7
  535. package/dist/src/tools/tool-registry.js.map +1 -1
  536. package/dist/src/tools/tool-registry.test.js +93 -10
  537. package/dist/src/tools/tool-registry.test.js.map +1 -1
  538. package/dist/src/tools/tools.d.ts +33 -16
  539. package/dist/src/tools/tools.js +115 -5
  540. package/dist/src/tools/tools.js.map +1 -1
  541. package/dist/src/tools/tools.test.js +1 -2
  542. package/dist/src/tools/tools.test.js.map +1 -1
  543. package/dist/src/tools/web-fetch.d.ts +3 -2
  544. package/dist/src/tools/web-fetch.js +14 -10
  545. package/dist/src/tools/web-fetch.js.map +1 -1
  546. package/dist/src/tools/web-fetch.test.js +55 -16
  547. package/dist/src/tools/web-fetch.test.js.map +1 -1
  548. package/dist/src/tools/web-search.d.ts +4 -3
  549. package/dist/src/tools/web-search.js +31 -8
  550. package/dist/src/tools/web-search.js.map +1 -1
  551. package/dist/src/tools/web-search.test.js +69 -1
  552. package/dist/src/tools/web-search.test.js.map +1 -1
  553. package/dist/src/tools/write-file.d.ts +4 -3
  554. package/dist/src/tools/write-file.js +17 -18
  555. package/dist/src/tools/write-file.js.map +1 -1
  556. package/dist/src/tools/write-file.test.js +108 -24
  557. package/dist/src/tools/write-file.test.js.map +1 -1
  558. package/dist/src/tools/write-todos.d.ts +25 -0
  559. package/dist/src/tools/write-todos.js +150 -0
  560. package/dist/src/tools/write-todos.js.map +1 -0
  561. package/dist/src/tools/write-todos.test.d.ts +6 -0
  562. package/dist/src/tools/write-todos.test.js +89 -0
  563. package/dist/src/tools/write-todos.test.js.map +1 -0
  564. package/dist/src/utils/bfsFileSearch.d.ts +2 -2
  565. package/dist/src/utils/bfsFileSearch.js +13 -7
  566. package/dist/src/utils/bfsFileSearch.js.map +1 -1
  567. package/dist/src/utils/bfsFileSearch.test.js +3 -3
  568. package/dist/src/utils/bfsFileSearch.test.js.map +1 -1
  569. package/dist/src/utils/editCorrector.d.ts +9 -8
  570. package/dist/src/utils/editCorrector.js +62 -19
  571. package/dist/src/utils/editCorrector.js.map +1 -1
  572. package/dist/src/utils/editCorrector.test.js +33 -82
  573. package/dist/src/utils/editCorrector.test.js.map +1 -1
  574. package/dist/src/utils/editor.js +32 -45
  575. package/dist/src/utils/editor.js.map +1 -1
  576. package/dist/src/utils/editor.test.js +62 -76
  577. package/dist/src/utils/editor.test.js.map +1 -1
  578. package/dist/src/utils/environmentContext.d.ts +2 -2
  579. package/dist/src/utils/errorParsing.js +2 -2
  580. package/dist/src/utils/errorParsing.js.map +1 -1
  581. package/dist/src/utils/errorParsing.test.js +7 -7
  582. package/dist/src/utils/errorParsing.test.js.map +1 -1
  583. package/dist/src/utils/errorReporting.d.ts +1 -1
  584. package/dist/src/utils/errors.d.ts +25 -0
  585. package/dist/src/utils/errors.js +42 -0
  586. package/dist/src/utils/errors.js.map +1 -1
  587. package/dist/src/utils/fetch.js +1 -1
  588. package/dist/src/utils/fetch.js.map +1 -1
  589. package/dist/src/utils/fileUtils.d.ts +24 -12
  590. package/dist/src/utils/fileUtils.js +170 -79
  591. package/dist/src/utils/fileUtils.js.map +1 -1
  592. package/dist/src/utils/fileUtils.test.js +347 -29
  593. package/dist/src/utils/fileUtils.test.js.map +1 -1
  594. package/dist/src/utils/filesearch/crawler.d.ts +1 -1
  595. package/dist/src/utils/filesearch/crawler.test.js +2 -2
  596. package/dist/src/utils/filesearch/crawler.test.js.map +1 -1
  597. package/dist/src/utils/filesearch/fileSearch.d.ts +1 -0
  598. package/dist/src/utils/filesearch/fileSearch.js +14 -9
  599. package/dist/src/utils/filesearch/fileSearch.js.map +1 -1
  600. package/dist/src/utils/filesearch/fileSearch.test.js +90 -0
  601. package/dist/src/utils/filesearch/fileSearch.test.js.map +1 -1
  602. package/dist/src/utils/flashFallback.test.d.ts +6 -0
  603. package/dist/src/utils/{flashFallback.integration.test.js → flashFallback.test.js} +33 -29
  604. package/dist/src/utils/flashFallback.test.js.map +1 -0
  605. package/dist/src/utils/geminiIgnoreParser.d.ts +18 -0
  606. package/dist/src/utils/geminiIgnoreParser.js +61 -0
  607. package/dist/src/utils/geminiIgnoreParser.js.map +1 -0
  608. package/dist/src/utils/geminiIgnoreParser.test.d.ts +6 -0
  609. package/dist/src/utils/geminiIgnoreParser.test.js +50 -0
  610. package/dist/src/utils/geminiIgnoreParser.test.js.map +1 -0
  611. package/dist/src/utils/generateContentResponseUtilities.d.ts +1 -2
  612. package/dist/src/utils/generateContentResponseUtilities.js +1 -13
  613. package/dist/src/utils/generateContentResponseUtilities.js.map +1 -1
  614. package/dist/src/utils/generateContentResponseUtilities.test.js +2 -40
  615. package/dist/src/utils/generateContentResponseUtilities.test.js.map +1 -1
  616. package/dist/src/utils/getFolderStructure.d.ts +2 -2
  617. package/dist/src/utils/getFolderStructure.js +3 -3
  618. package/dist/src/utils/getFolderStructure.js.map +1 -1
  619. package/dist/src/utils/getFolderStructure.test.js +4 -4
  620. package/dist/src/utils/getFolderStructure.test.js.map +1 -1
  621. package/dist/src/utils/gitIgnoreParser.d.ts +3 -7
  622. package/dist/src/utils/gitIgnoreParser.js +126 -35
  623. package/dist/src/utils/gitIgnoreParser.js.map +1 -1
  624. package/dist/src/utils/gitIgnoreParser.test.js +69 -38
  625. package/dist/src/utils/gitIgnoreParser.test.js.map +1 -1
  626. package/dist/src/utils/gitUtils.js +2 -2
  627. package/dist/src/utils/gitUtils.js.map +1 -1
  628. package/dist/src/utils/ignorePatterns.d.ts +103 -0
  629. package/dist/src/utils/ignorePatterns.js +220 -0
  630. package/dist/src/utils/ignorePatterns.js.map +1 -0
  631. package/dist/src/utils/ignorePatterns.test.d.ts +6 -0
  632. package/dist/src/utils/ignorePatterns.test.js +250 -0
  633. package/dist/src/utils/ignorePatterns.test.js.map +1 -0
  634. package/dist/src/utils/installationManager.d.ts +16 -0
  635. package/dist/src/utils/installationManager.js +50 -0
  636. package/dist/src/utils/installationManager.js.map +1 -0
  637. package/dist/src/utils/installationManager.test.d.ts +6 -0
  638. package/dist/src/utils/installationManager.test.js +83 -0
  639. package/dist/src/utils/installationManager.test.js.map +1 -0
  640. package/dist/src/utils/language-detection.d.ts +6 -0
  641. package/dist/src/utils/language-detection.js +101 -0
  642. package/dist/src/utils/language-detection.js.map +1 -0
  643. package/dist/src/utils/llm-edit-fixer.d.ts +26 -0
  644. package/dist/src/utils/llm-edit-fixer.js +131 -0
  645. package/dist/src/utils/llm-edit-fixer.js.map +1 -0
  646. package/dist/src/utils/llm-edit-fixer.test.d.ts +6 -0
  647. package/dist/src/utils/llm-edit-fixer.test.js +186 -0
  648. package/dist/src/utils/llm-edit-fixer.test.js.map +1 -0
  649. package/dist/src/utils/memoryDiscovery.d.ts +7 -6
  650. package/dist/src/utils/memoryDiscovery.js +68 -33
  651. package/dist/src/utils/memoryDiscovery.js.map +1 -1
  652. package/dist/src/utils/memoryDiscovery.test.js +88 -26
  653. package/dist/src/utils/memoryDiscovery.test.js.map +1 -1
  654. package/dist/src/utils/memoryImportProcessor.js +15 -22
  655. package/dist/src/utils/memoryImportProcessor.js.map +1 -1
  656. package/dist/src/utils/memoryImportProcessor.test.js +16 -141
  657. package/dist/src/utils/memoryImportProcessor.test.js.map +1 -1
  658. package/dist/src/utils/messageInspectors.d.ts +1 -1
  659. package/dist/src/utils/nextSpeakerChecker.d.ts +3 -3
  660. package/dist/src/utils/nextSpeakerChecker.js +8 -2
  661. package/dist/src/utils/nextSpeakerChecker.js.map +1 -1
  662. package/dist/src/utils/nextSpeakerChecker.test.js +75 -64
  663. package/dist/src/utils/nextSpeakerChecker.test.js.map +1 -1
  664. package/dist/src/utils/partUtils.d.ts +22 -1
  665. package/dist/src/utils/partUtils.js +68 -0
  666. package/dist/src/utils/partUtils.js.map +1 -1
  667. package/dist/src/utils/partUtils.test.js +112 -1
  668. package/dist/src/utils/partUtils.test.js.map +1 -1
  669. package/dist/src/utils/pathReader.d.ts +17 -0
  670. package/dist/src/utils/pathReader.js +92 -0
  671. package/dist/src/utils/pathReader.js.map +1 -0
  672. package/dist/src/utils/pathReader.test.d.ts +6 -0
  673. package/dist/src/utils/pathReader.test.js +363 -0
  674. package/dist/src/utils/pathReader.test.js.map +1 -0
  675. package/dist/src/utils/paths.d.ts +0 -17
  676. package/dist/src/utils/paths.js +2 -28
  677. package/dist/src/utils/paths.js.map +1 -1
  678. package/dist/src/utils/promptIdContext.d.ts +7 -0
  679. package/dist/src/utils/promptIdContext.js +8 -0
  680. package/dist/src/utils/promptIdContext.js.map +1 -0
  681. package/dist/src/utils/quotaErrorDetection.d.ts +1 -1
  682. package/dist/src/utils/retry.d.ts +3 -1
  683. package/dist/src/utils/retry.js +60 -5
  684. package/dist/src/utils/retry.js.map +1 -1
  685. package/dist/src/utils/retry.test.js +35 -3
  686. package/dist/src/utils/retry.test.js.map +1 -1
  687. package/dist/src/utils/schemaValidator.js +15 -1
  688. package/dist/src/utils/schemaValidator.js.map +1 -1
  689. package/dist/src/utils/schemaValidator.test.d.ts +6 -0
  690. package/dist/src/utils/schemaValidator.test.js +113 -0
  691. package/dist/src/utils/schemaValidator.test.js.map +1 -0
  692. package/dist/src/utils/session.js +1 -1
  693. package/dist/src/utils/session.js.map +1 -1
  694. package/dist/src/utils/shell-utils.d.ts +6 -1
  695. package/dist/src/utils/shell-utils.js +51 -30
  696. package/dist/src/utils/shell-utils.js.map +1 -1
  697. package/dist/src/utils/shell-utils.test.js +9 -0
  698. package/dist/src/utils/shell-utils.test.js.map +1 -1
  699. package/dist/src/utils/summarizer.d.ts +2 -2
  700. package/dist/src/utils/summarizer.test.js.map +1 -1
  701. package/dist/src/utils/systemEncoding.js +2 -2
  702. package/dist/src/utils/systemEncoding.js.map +1 -1
  703. package/dist/src/utils/systemEncoding.test.js +2 -2
  704. package/dist/src/utils/systemEncoding.test.js.map +1 -1
  705. package/dist/src/utils/terminalSerializer.d.ts +25 -0
  706. package/dist/src/utils/terminalSerializer.js +432 -0
  707. package/dist/src/utils/terminalSerializer.js.map +1 -0
  708. package/dist/src/utils/terminalSerializer.test.d.ts +6 -0
  709. package/dist/src/utils/terminalSerializer.test.js +176 -0
  710. package/dist/src/utils/terminalSerializer.test.js.map +1 -0
  711. package/dist/src/utils/textUtils.d.ts +5 -0
  712. package/dist/src/utils/textUtils.js +14 -0
  713. package/dist/src/utils/textUtils.js.map +1 -1
  714. package/dist/src/utils/textUtils.test.d.ts +6 -0
  715. package/dist/src/utils/textUtils.test.js +59 -0
  716. package/dist/src/utils/textUtils.test.js.map +1 -0
  717. package/dist/src/utils/thoughtUtils.d.ts +21 -0
  718. package/dist/src/utils/thoughtUtils.js +39 -0
  719. package/dist/src/utils/thoughtUtils.js.map +1 -0
  720. package/dist/src/utils/thoughtUtils.test.d.ts +6 -0
  721. package/dist/src/utils/thoughtUtils.test.js +78 -0
  722. package/dist/src/utils/thoughtUtils.test.js.map +1 -0
  723. package/dist/src/utils/tool-utils.d.ts +19 -0
  724. package/dist/src/utils/tool-utils.js +58 -0
  725. package/dist/src/utils/tool-utils.js.map +1 -0
  726. package/dist/src/utils/tool-utils.test.d.ts +6 -0
  727. package/dist/src/utils/tool-utils.test.js +61 -0
  728. package/dist/src/utils/tool-utils.test.js.map +1 -0
  729. package/dist/src/utils/userAccountManager.d.ts +20 -0
  730. package/dist/src/utils/userAccountManager.js +114 -0
  731. package/dist/src/utils/userAccountManager.js.map +1 -0
  732. package/dist/src/utils/userAccountManager.test.d.ts +6 -0
  733. package/dist/src/utils/{user_account.test.js → userAccountManager.test.js} +33 -30
  734. package/dist/src/utils/userAccountManager.test.js.map +1 -0
  735. package/dist/src/utils/workspaceContext.js +13 -7
  736. package/dist/src/utils/workspaceContext.js.map +1 -1
  737. package/dist/src/utils/workspaceContext.test.js +41 -16
  738. package/dist/src/utils/workspaceContext.test.js.map +1 -1
  739. package/dist/tsconfig.tsbuildinfo +1 -1
  740. package/package.json +18 -9
  741. package/dist/src/core/modelCheck.d.ts +0 -14
  742. package/dist/src/core/modelCheck.js +0 -62
  743. package/dist/src/core/modelCheck.js.map +0 -1
  744. package/dist/src/test-utils/tools.d.ts +0 -44
  745. package/dist/src/test-utils/tools.js.map +0 -1
  746. package/dist/src/utils/flashFallback.integration.test.js.map +0 -1
  747. package/dist/src/utils/user_account.d.ts +0 -9
  748. package/dist/src/utils/user_account.js +0 -109
  749. package/dist/src/utils/user_account.js.map +0 -1
  750. package/dist/src/utils/user_account.test.js.map +0 -1
  751. package/dist/src/utils/user_id.d.ts +0 -11
  752. package/dist/src/utils/user_id.js +0 -49
  753. package/dist/src/utils/user_id.js.map +0 -1
  754. package/dist/src/utils/user_id.test.js +0 -21
  755. package/dist/src/utils/user_id.test.js.map +0 -1
  756. /package/dist/src/{utils/flashFallback.integration.test.d.ts → agents/executor.test.d.ts} +0 -0
  757. /package/dist/src/{utils/user_account.test.d.ts → agents/invocation.test.d.ts} +0 -0
  758. /package/dist/src/{utils/user_id.test.d.ts → agents/registry.test.d.ts} +0 -0
@@ -4,72 +4,502 @@
4
4
  * SPDX-License-Identifier: Apache-2.0
5
5
  */
6
6
  import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
7
- import { GeminiChat } from './geminiChat.js';
7
+ import { ApiError } from '@google/genai';
8
+ import { GeminiChat, InvalidStreamError, StreamEventType, } from './geminiChat.js';
8
9
  import { setSimulate429 } from '../utils/testUtils.js';
9
- // Mocks
10
- const mockModelsModule = {
11
- generateContent: vi.fn(),
12
- generateContentStream: vi.fn(),
13
- countTokens: vi.fn(),
14
- embedContent: vi.fn(),
15
- batchEmbedContents: vi.fn(),
16
- };
10
+ import { DEFAULT_GEMINI_FLASH_MODEL } from '../config/models.js';
11
+ import { AuthType } from './contentGenerator.js';
12
+ import {} from '../utils/retry.js';
13
+ import { Kind } from '../tools/tools.js';
14
+ import { uiTelemetryService } from '../telemetry/uiTelemetry.js';
15
+ // Mock fs module to prevent actual file system operations during tests
16
+ const mockFileSystem = new Map();
17
+ vi.mock('node:fs', () => {
18
+ const fsModule = {
19
+ mkdirSync: vi.fn(),
20
+ writeFileSync: vi.fn((path, data) => {
21
+ mockFileSystem.set(path, data);
22
+ }),
23
+ readFileSync: vi.fn((path) => {
24
+ if (mockFileSystem.has(path)) {
25
+ return mockFileSystem.get(path);
26
+ }
27
+ throw Object.assign(new Error('ENOENT: no such file or directory'), {
28
+ code: 'ENOENT',
29
+ });
30
+ }),
31
+ existsSync: vi.fn((path) => mockFileSystem.has(path)),
32
+ };
33
+ return {
34
+ default: fsModule,
35
+ ...fsModule,
36
+ };
37
+ });
38
+ const { mockHandleFallback } = vi.hoisted(() => ({
39
+ mockHandleFallback: vi.fn(),
40
+ }));
41
+ // Add mock for the retry utility
42
+ const { mockRetryWithBackoff } = vi.hoisted(() => ({
43
+ mockRetryWithBackoff: vi.fn(),
44
+ }));
45
+ vi.mock('../utils/retry.js', () => ({
46
+ retryWithBackoff: mockRetryWithBackoff,
47
+ }));
48
+ vi.mock('../fallback/handler.js', () => ({
49
+ handleFallback: mockHandleFallback,
50
+ }));
51
+ const { mockLogContentRetry, mockLogContentRetryFailure } = vi.hoisted(() => ({
52
+ mockLogContentRetry: vi.fn(),
53
+ mockLogContentRetryFailure: vi.fn(),
54
+ }));
55
+ vi.mock('../telemetry/loggers.js', () => ({
56
+ logContentRetry: mockLogContentRetry,
57
+ logContentRetryFailure: mockLogContentRetryFailure,
58
+ }));
59
+ vi.mock('../telemetry/uiTelemetry.js', () => ({
60
+ uiTelemetryService: {
61
+ setLastPromptTokenCount: vi.fn(),
62
+ },
63
+ }));
17
64
  describe('GeminiChat', () => {
65
+ let mockContentGenerator;
18
66
  let chat;
19
67
  let mockConfig;
20
68
  const config = {};
21
69
  beforeEach(() => {
22
70
  vi.clearAllMocks();
71
+ vi.mocked(uiTelemetryService.setLastPromptTokenCount).mockClear();
72
+ mockContentGenerator = {
73
+ generateContent: vi.fn(),
74
+ generateContentStream: vi.fn(),
75
+ countTokens: vi.fn(),
76
+ embedContent: vi.fn(),
77
+ batchEmbedContents: vi.fn(),
78
+ };
79
+ mockHandleFallback.mockClear();
80
+ // Default mock implementation for tests that don't care about retry logic
81
+ mockRetryWithBackoff.mockImplementation(async (apiCall) => apiCall());
23
82
  mockConfig = {
24
83
  getSessionId: () => 'test-session-id',
25
84
  getTelemetryLogPromptsEnabled: () => true,
26
85
  getUsageStatisticsEnabled: () => true,
27
86
  getDebugMode: () => false,
28
- getContentGeneratorConfig: () => ({
29
- authType: 'oauth-personal',
87
+ getContentGeneratorConfig: vi.fn().mockReturnValue({
88
+ authType: 'oauth-personal', // Ensure this is set for fallback tests
30
89
  model: 'test-model',
31
90
  }),
32
91
  getModel: vi.fn().mockReturnValue('gemini-pro'),
33
92
  setModel: vi.fn(),
93
+ isInFallbackMode: vi.fn().mockReturnValue(false),
34
94
  getQuotaErrorOccurred: vi.fn().mockReturnValue(false),
35
95
  setQuotaErrorOccurred: vi.fn(),
36
96
  flashFallbackHandler: undefined,
97
+ getProjectRoot: vi.fn().mockReturnValue('/test/project/root'),
98
+ storage: {
99
+ getProjectTempDir: vi.fn().mockReturnValue('/test/temp'),
100
+ },
101
+ getToolRegistry: vi.fn().mockReturnValue({
102
+ getTool: vi.fn(),
103
+ }),
104
+ getContentGenerator: vi.fn().mockReturnValue(mockContentGenerator),
37
105
  };
38
106
  // Disable 429 simulation for tests
39
107
  setSimulate429(false);
40
108
  // Reset history for each test by creating a new instance
41
- chat = new GeminiChat(mockConfig, mockModelsModule, config, []);
109
+ chat = new GeminiChat(mockConfig, config, []);
42
110
  });
43
111
  afterEach(() => {
44
112
  vi.restoreAllMocks();
45
113
  vi.resetAllMocks();
46
114
  });
47
- describe('sendMessage', () => {
48
- it('should call generateContent with the correct parameters', async () => {
49
- const response = {
50
- candidates: [
51
- {
52
- content: {
53
- parts: [{ text: 'response' }],
54
- role: 'model',
115
+ describe('sendMessageStream', () => {
116
+ it('should succeed if a tool call is followed by an empty part', async () => {
117
+ // 1. Mock a stream that contains a tool call, then an invalid (empty) part.
118
+ const streamWithToolCall = (async function* () {
119
+ yield {
120
+ candidates: [
121
+ {
122
+ content: {
123
+ role: 'model',
124
+ parts: [{ functionCall: { name: 'test_tool', args: {} } }],
125
+ },
55
126
  },
56
- finishReason: 'STOP',
57
- index: 0,
58
- safetyRatings: [],
127
+ ],
128
+ };
129
+ // This second chunk is invalid according to isValidResponse
130
+ yield {
131
+ candidates: [
132
+ {
133
+ content: {
134
+ role: 'model',
135
+ parts: [{ text: '' }],
136
+ },
137
+ },
138
+ ],
139
+ };
140
+ })();
141
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(streamWithToolCall);
142
+ // 2. Action & Assert: The stream processing should complete without throwing an error
143
+ // because the presence of a tool call makes the empty final chunk acceptable.
144
+ const stream = await chat.sendMessageStream('test-model', { message: 'test message' }, 'prompt-id-tool-call-empty-end');
145
+ await expect((async () => {
146
+ for await (const _ of stream) {
147
+ /* consume stream */
148
+ }
149
+ })()).resolves.not.toThrow();
150
+ // 3. Verify history was recorded correctly
151
+ const history = chat.getHistory();
152
+ expect(history.length).toBe(2); // user turn + model turn
153
+ const modelTurn = history[1];
154
+ expect(modelTurn?.parts?.length).toBe(1); // The empty part is discarded
155
+ expect(modelTurn?.parts[0].functionCall).toBeDefined();
156
+ });
157
+ it('should fail if the stream ends with an empty part and has no finishReason', async () => {
158
+ // 1. Mock a stream that ends with an invalid part and has no finish reason.
159
+ const streamWithNoFinish = (async function* () {
160
+ yield {
161
+ candidates: [
162
+ {
163
+ content: {
164
+ role: 'model',
165
+ parts: [{ text: 'Initial content...' }],
166
+ },
167
+ },
168
+ ],
169
+ };
170
+ // This second chunk is invalid and has no finishReason, so it should fail.
171
+ yield {
172
+ candidates: [
173
+ {
174
+ content: {
175
+ role: 'model',
176
+ parts: [{ text: '' }],
177
+ },
178
+ },
179
+ ],
180
+ };
181
+ })();
182
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(streamWithNoFinish);
183
+ // 2. Action & Assert: The stream should fail because there's no finish reason.
184
+ const stream = await chat.sendMessageStream('test-model', { message: 'test message' }, 'prompt-id-no-finish-empty-end');
185
+ await expect((async () => {
186
+ for await (const _ of stream) {
187
+ /* consume stream */
188
+ }
189
+ })()).rejects.toThrow(InvalidStreamError);
190
+ });
191
+ it('should succeed if the stream ends with an invalid part but has a finishReason and contained a valid part', async () => {
192
+ // 1. Mock a stream that sends a valid chunk, then an invalid one, but has a finish reason.
193
+ const streamWithInvalidEnd = (async function* () {
194
+ yield {
195
+ candidates: [
196
+ {
197
+ content: {
198
+ role: 'model',
199
+ parts: [{ text: 'Initial valid content...' }],
200
+ },
201
+ },
202
+ ],
203
+ };
204
+ // This second chunk is invalid, but the response has a finishReason.
205
+ yield {
206
+ candidates: [
207
+ {
208
+ content: {
209
+ role: 'model',
210
+ parts: [{ text: '' }], // Invalid part
211
+ },
212
+ finishReason: 'STOP',
213
+ },
214
+ ],
215
+ };
216
+ })();
217
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(streamWithInvalidEnd);
218
+ // 2. Action & Assert: The stream should complete without throwing an error.
219
+ const stream = await chat.sendMessageStream('test-model', { message: 'test message' }, 'prompt-id-valid-then-invalid-end');
220
+ await expect((async () => {
221
+ for await (const _ of stream) {
222
+ /* consume stream */
223
+ }
224
+ })()).resolves.not.toThrow();
225
+ // 3. Verify history was recorded correctly with only the valid part.
226
+ const history = chat.getHistory();
227
+ expect(history.length).toBe(2); // user turn + model turn
228
+ const modelTurn = history[1];
229
+ expect(modelTurn?.parts?.length).toBe(1);
230
+ expect(modelTurn?.parts[0].text).toBe('Initial valid content...');
231
+ });
232
+ it('should consolidate subsequent text chunks after receiving an empty text chunk', async () => {
233
+ // 1. Mock the API to return a stream where one chunk is just an empty text part.
234
+ const multiChunkStream = (async function* () {
235
+ yield {
236
+ candidates: [
237
+ { content: { role: 'model', parts: [{ text: 'Hello' }] } },
238
+ ],
239
+ };
240
+ // FIX: The original test used { text: '' }, which is invalid.
241
+ // A chunk can be empty but still valid. This chunk is now removed
242
+ // as the important part is consolidating what comes after.
243
+ yield {
244
+ candidates: [
245
+ {
246
+ content: { role: 'model', parts: [{ text: ' World!' }] },
247
+ finishReason: 'STOP',
248
+ },
249
+ ],
250
+ };
251
+ })();
252
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(multiChunkStream);
253
+ // 2. Action: Send a message and consume the stream.
254
+ const stream = await chat.sendMessageStream('test-model', { message: 'test message' }, 'prompt-id-empty-chunk-consolidation');
255
+ for await (const _ of stream) {
256
+ // Consume the stream
257
+ }
258
+ // 3. Assert: Check that the final history was correctly consolidated.
259
+ const history = chat.getHistory();
260
+ expect(history.length).toBe(2);
261
+ const modelTurn = history[1];
262
+ expect(modelTurn?.parts?.length).toBe(1);
263
+ expect(modelTurn?.parts[0].text).toBe('Hello World!');
264
+ });
265
+ it('should consolidate adjacent text parts that arrive in separate stream chunks', async () => {
266
+ // 1. Mock the API to return a stream of multiple, adjacent text chunks.
267
+ const multiChunkStream = (async function* () {
268
+ yield {
269
+ candidates: [
270
+ { content: { role: 'model', parts: [{ text: 'This is the ' }] } },
271
+ ],
272
+ };
273
+ yield {
274
+ candidates: [
275
+ { content: { role: 'model', parts: [{ text: 'first part.' }] } },
276
+ ],
277
+ };
278
+ // This function call should break the consolidation.
279
+ yield {
280
+ candidates: [
281
+ {
282
+ content: {
283
+ role: 'model',
284
+ parts: [{ functionCall: { name: 'do_stuff', args: {} } }],
285
+ },
286
+ },
287
+ ],
288
+ };
289
+ yield {
290
+ candidates: [
291
+ {
292
+ content: {
293
+ role: 'model',
294
+ parts: [{ text: 'This is the second part.' }],
295
+ },
296
+ },
297
+ ],
298
+ };
299
+ })();
300
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(multiChunkStream);
301
+ // 2. Action: Send a message and consume the stream.
302
+ const stream = await chat.sendMessageStream('test-model', { message: 'test message' }, 'prompt-id-multi-chunk');
303
+ for await (const _ of stream) {
304
+ // Consume the stream to trigger history recording.
305
+ }
306
+ // 3. Assert: Check that the final history was correctly consolidated.
307
+ const history = chat.getHistory();
308
+ // The history should contain the user's turn and ONE consolidated model turn.
309
+ expect(history.length).toBe(2);
310
+ const modelTurn = history[1];
311
+ expect(modelTurn.role).toBe('model');
312
+ // The model turn should have 3 distinct parts: the merged text, the function call, and the final text.
313
+ expect(modelTurn?.parts?.length).toBe(3);
314
+ expect(modelTurn?.parts[0].text).toBe('This is the first part.');
315
+ expect(modelTurn.parts[1].functionCall).toBeDefined();
316
+ expect(modelTurn.parts[2].text).toBe('This is the second part.');
317
+ });
318
+ it('should preserve text parts that stream in the same chunk as a thought', async () => {
319
+ // 1. Mock the API to return a single chunk containing both a thought and visible text.
320
+ const mixedContentStream = (async function* () {
321
+ yield {
322
+ candidates: [
323
+ {
324
+ content: {
325
+ role: 'model',
326
+ parts: [
327
+ { thought: 'This is a thought.' },
328
+ { text: 'This is the visible text that should not be lost.' },
329
+ ],
330
+ },
331
+ finishReason: 'STOP',
332
+ },
333
+ ],
334
+ };
335
+ })();
336
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(mixedContentStream);
337
+ // 2. Action: Send a message and fully consume the stream to trigger history recording.
338
+ const stream = await chat.sendMessageStream('test-model', { message: 'test message' }, 'prompt-id-mixed-chunk');
339
+ for await (const _ of stream) {
340
+ // This loop consumes the stream.
341
+ }
342
+ // 3. Assert: Check the final state of the history.
343
+ const history = chat.getHistory();
344
+ // The history should contain two turns: the user's message and the model's response.
345
+ expect(history.length).toBe(2);
346
+ const modelTurn = history[1];
347
+ expect(modelTurn.role).toBe('model');
348
+ // CRUCIAL ASSERTION:
349
+ // The buggy code would fail here, resulting in parts.length being 0.
350
+ // The corrected code will pass, preserving the single visible text part.
351
+ expect(modelTurn?.parts?.length).toBe(1);
352
+ expect(modelTurn?.parts[0].text).toBe('This is the visible text that should not be lost.');
353
+ });
354
+ it('should throw an error when a tool call is followed by an empty stream response', async () => {
355
+ // 1. Setup: A history where the model has just made a function call.
356
+ const initialHistory = [
357
+ {
358
+ role: 'user',
359
+ parts: [{ text: 'Find a good Italian restaurant for me.' }],
360
+ },
361
+ {
362
+ role: 'model',
363
+ parts: [
364
+ {
365
+ functionCall: {
366
+ name: 'find_restaurant',
367
+ args: { cuisine: 'Italian' },
368
+ },
369
+ },
370
+ ],
371
+ },
372
+ ];
373
+ chat.setHistory(initialHistory);
374
+ // 2. Mock the API to return an empty/thought-only stream.
375
+ const emptyStreamResponse = (async function* () {
376
+ yield {
377
+ candidates: [
378
+ {
379
+ content: { role: 'model', parts: [{ thought: true }] },
380
+ finishReason: 'STOP',
381
+ },
382
+ ],
383
+ };
384
+ })();
385
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(emptyStreamResponse);
386
+ // 3. Action: Send the function response back to the model and consume the stream.
387
+ const stream = await chat.sendMessageStream('test-model', {
388
+ message: {
389
+ functionResponse: {
390
+ name: 'find_restaurant',
391
+ response: { name: 'Vesuvio' },
59
392
  },
60
- ],
61
- text: () => 'response',
62
- };
63
- vi.mocked(mockModelsModule.generateContent).mockResolvedValue(response);
64
- await chat.sendMessage({ message: 'hello' }, 'prompt-id-1');
65
- expect(mockModelsModule.generateContent).toHaveBeenCalledWith({
66
- model: 'gemini-pro',
67
- contents: [{ role: 'user', parts: [{ text: 'hello' }] }],
68
- config: {},
69
- }, 'prompt-id-1');
393
+ },
394
+ }, 'prompt-id-stream-1');
395
+ // 4. Assert: The stream processing should throw an InvalidStreamError.
396
+ await expect((async () => {
397
+ for await (const _ of stream) {
398
+ // This loop consumes the stream to trigger the internal logic.
399
+ }
400
+ })()).rejects.toThrow(InvalidStreamError);
401
+ });
402
+ it('should succeed when there is a tool call without finish reason', async () => {
403
+ // Setup: Stream with tool call but no finish reason
404
+ const streamWithToolCall = (async function* () {
405
+ yield {
406
+ candidates: [
407
+ {
408
+ content: {
409
+ role: 'model',
410
+ parts: [
411
+ {
412
+ functionCall: {
413
+ name: 'test_function',
414
+ args: { param: 'value' },
415
+ },
416
+ },
417
+ ],
418
+ },
419
+ // No finishReason
420
+ },
421
+ ],
422
+ };
423
+ })();
424
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(streamWithToolCall);
425
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-1');
426
+ // Should not throw an error
427
+ await expect((async () => {
428
+ for await (const _ of stream) {
429
+ // consume stream
430
+ }
431
+ })()).resolves.not.toThrow();
432
+ });
433
+ it('should throw InvalidStreamError when no tool call and no finish reason', async () => {
434
+ // Setup: Stream with text but no finish reason and no tool call
435
+ const streamWithoutFinishReason = (async function* () {
436
+ yield {
437
+ candidates: [
438
+ {
439
+ content: {
440
+ role: 'model',
441
+ parts: [{ text: 'some response' }],
442
+ },
443
+ // No finishReason
444
+ },
445
+ ],
446
+ };
447
+ })();
448
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(streamWithoutFinishReason);
449
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-1');
450
+ await expect((async () => {
451
+ for await (const _ of stream) {
452
+ // consume stream
453
+ }
454
+ })()).rejects.toThrow(InvalidStreamError);
455
+ });
456
+ it('should throw InvalidStreamError when no tool call and empty response text', async () => {
457
+ // Setup: Stream with finish reason but empty response (only thoughts)
458
+ const streamWithEmptyResponse = (async function* () {
459
+ yield {
460
+ candidates: [
461
+ {
462
+ content: {
463
+ role: 'model',
464
+ parts: [{ thought: 'thinking...' }],
465
+ },
466
+ finishReason: 'STOP',
467
+ },
468
+ ],
469
+ };
470
+ })();
471
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(streamWithEmptyResponse);
472
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-1');
473
+ await expect((async () => {
474
+ for await (const _ of stream) {
475
+ // consume stream
476
+ }
477
+ })()).rejects.toThrow(InvalidStreamError);
478
+ });
479
+ it('should succeed when there is finish reason and response text', async () => {
480
+ // Setup: Stream with both finish reason and text content
481
+ const validStream = (async function* () {
482
+ yield {
483
+ candidates: [
484
+ {
485
+ content: {
486
+ role: 'model',
487
+ parts: [{ text: 'valid response' }],
488
+ },
489
+ finishReason: 'STOP',
490
+ },
491
+ ],
492
+ };
493
+ })();
494
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(validStream);
495
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-1');
496
+ // Should not throw an error
497
+ await expect((async () => {
498
+ for await (const _ of stream) {
499
+ // consume stream
500
+ }
501
+ })()).resolves.not.toThrow();
70
502
  });
71
- });
72
- describe('sendMessageStream', () => {
73
503
  it('should call generateContentStream with the correct parameters', async () => {
74
504
  const response = (async function* () {
75
505
  yield {
@@ -85,339 +515,843 @@ describe('GeminiChat', () => {
85
515
  },
86
516
  ],
87
517
  text: () => 'response',
518
+ usageMetadata: {
519
+ promptTokenCount: 42,
520
+ candidatesTokenCount: 15,
521
+ totalTokenCount: 57,
522
+ },
88
523
  };
89
524
  })();
90
- vi.mocked(mockModelsModule.generateContentStream).mockResolvedValue(response);
91
- await chat.sendMessageStream({ message: 'hello' }, 'prompt-id-1');
92
- expect(mockModelsModule.generateContentStream).toHaveBeenCalledWith({
93
- model: 'gemini-pro',
94
- contents: [{ role: 'user', parts: [{ text: 'hello' }] }],
525
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(response);
526
+ const stream = await chat.sendMessageStream('test-model', { message: 'hello' }, 'prompt-id-1');
527
+ for await (const _ of stream) {
528
+ // consume stream
529
+ }
530
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledWith({
531
+ model: 'test-model',
532
+ contents: [
533
+ {
534
+ role: 'user',
535
+ parts: [{ text: 'hello' }],
536
+ },
537
+ ],
95
538
  config: {},
96
539
  }, 'prompt-id-1');
540
+ // Verify that token counting is called when usageMetadata is present
541
+ expect(uiTelemetryService.setLastPromptTokenCount).toHaveBeenCalledWith(42);
542
+ expect(uiTelemetryService.setLastPromptTokenCount).toHaveBeenCalledTimes(1);
97
543
  });
98
544
  });
99
- describe('recordHistory', () => {
100
- const userInput = {
101
- role: 'user',
102
- parts: [{ text: 'User input' }],
103
- };
104
- it('should add user input and a single model output to history', () => {
105
- const modelOutput = [
106
- { role: 'model', parts: [{ text: 'Model output' }] },
107
- ];
108
- // @ts-expect-error Accessing private method for testing purposes
109
- chat.recordHistory(userInput, modelOutput);
545
+ describe('addHistory', () => {
546
+ it('should add a new content item to the history', () => {
547
+ const newContent = {
548
+ role: 'user',
549
+ parts: [{ text: 'A new message' }],
550
+ };
551
+ chat.addHistory(newContent);
110
552
  const history = chat.getHistory();
111
- expect(history).toEqual([userInput, modelOutput[0]]);
553
+ expect(history.length).toBe(1);
554
+ expect(history[0]).toEqual(newContent);
112
555
  });
113
- it('should consolidate adjacent model outputs', () => {
114
- const modelOutputParts = [
115
- { role: 'model', parts: [{ text: 'Model part 1' }] },
116
- { role: 'model', parts: [{ text: 'Model part 2' }] },
117
- ];
118
- // @ts-expect-error Accessing private method for testing purposes
119
- chat.recordHistory(userInput, modelOutputParts);
556
+ it('should add multiple items correctly', () => {
557
+ const content1 = {
558
+ role: 'user',
559
+ parts: [{ text: 'Message 1' }],
560
+ };
561
+ const content2 = {
562
+ role: 'model',
563
+ parts: [{ text: 'Message 2' }],
564
+ };
565
+ chat.addHistory(content1);
566
+ chat.addHistory(content2);
120
567
  const history = chat.getHistory();
121
568
  expect(history.length).toBe(2);
122
- expect(history[0]).toEqual(userInput);
123
- expect(history[1].role).toBe('model');
124
- expect(history[1].parts).toEqual([{ text: 'Model part 1Model part 2' }]);
569
+ expect(history[0]).toEqual(content1);
570
+ expect(history[1]).toEqual(content2);
125
571
  });
126
- it('should handle a mix of user and model roles in outputContents (though unusual)', () => {
127
- const mixedOutput = [
128
- { role: 'model', parts: [{ text: 'Model 1' }] },
129
- { role: 'user', parts: [{ text: 'Unexpected User' }] }, // This should be pushed as is
130
- { role: 'model', parts: [{ text: 'Model 2' }] },
131
- ];
132
- // @ts-expect-error Accessing private method for testing purposes
133
- chat.recordHistory(userInput, mixedOutput);
134
- const history = chat.getHistory();
135
- expect(history.length).toBe(4); // user, model1, user_unexpected, model2
136
- expect(history[0]).toEqual(userInput);
137
- expect(history[1]).toEqual(mixedOutput[0]);
138
- expect(history[2]).toEqual(mixedOutput[1]);
139
- expect(history[3]).toEqual(mixedOutput[2]);
572
+ });
573
+ describe('sendMessageStream with retries', () => {
574
+ it('should yield a RETRY event when an invalid stream is encountered', async () => {
575
+ // ARRANGE: Mock the stream to fail once, then succeed.
576
+ vi.mocked(mockContentGenerator.generateContentStream)
577
+ .mockImplementationOnce(async () =>
578
+ // First attempt: An invalid stream with an empty text part.
579
+ (async function* () {
580
+ yield {
581
+ candidates: [{ content: { parts: [{ text: '' }] } }],
582
+ };
583
+ })())
584
+ .mockImplementationOnce(async () =>
585
+ // Second attempt (the retry): A minimal valid stream.
586
+ (async function* () {
587
+ yield {
588
+ candidates: [
589
+ {
590
+ content: { parts: [{ text: 'Success' }] },
591
+ finishReason: 'STOP',
592
+ },
593
+ ],
594
+ };
595
+ })());
596
+ // ACT: Send a message and collect all events from the stream.
597
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-yield-retry');
598
+ const events = [];
599
+ for await (const event of stream) {
600
+ events.push(event);
601
+ }
602
+ // ASSERT: Check that a RETRY event was present in the stream's output.
603
+ const retryEvent = events.find((e) => e.type === StreamEventType.RETRY);
604
+ expect(retryEvent).toBeDefined();
605
+ expect(retryEvent?.type).toBe(StreamEventType.RETRY);
140
606
  });
141
- it('should consolidate multiple adjacent model outputs correctly', () => {
142
- const modelOutputParts = [
143
- { role: 'model', parts: [{ text: 'M1' }] },
144
- { role: 'model', parts: [{ text: 'M2' }] },
145
- { role: 'model', parts: [{ text: 'M3' }] },
146
- ];
147
- // @ts-expect-error Accessing private method for testing purposes
148
- chat.recordHistory(userInput, modelOutputParts);
607
+ it('should retry on invalid content, succeed, and report metrics', async () => {
608
+ // Use mockImplementationOnce to provide a fresh, promise-wrapped generator for each attempt.
609
+ vi.mocked(mockContentGenerator.generateContentStream)
610
+ .mockImplementationOnce(async () =>
611
+ // First call returns an invalid stream
612
+ (async function* () {
613
+ yield {
614
+ candidates: [{ content: { parts: [{ text: '' }] } }], // Invalid empty text part
615
+ };
616
+ })())
617
+ .mockImplementationOnce(async () =>
618
+ // Second call returns a valid stream
619
+ (async function* () {
620
+ yield {
621
+ candidates: [
622
+ {
623
+ content: { parts: [{ text: 'Successful response' }] },
624
+ finishReason: 'STOP',
625
+ },
626
+ ],
627
+ };
628
+ })());
629
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-retry-success');
630
+ const chunks = [];
631
+ for await (const chunk of stream) {
632
+ chunks.push(chunk);
633
+ }
634
+ // Assertions
635
+ expect(mockLogContentRetry).toHaveBeenCalledTimes(1);
636
+ expect(mockLogContentRetryFailure).not.toHaveBeenCalled();
637
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(2);
638
+ // Check for a retry event
639
+ expect(chunks.some((c) => c.type === StreamEventType.RETRY)).toBe(true);
640
+ // Check for the successful content chunk
641
+ expect(chunks.some((c) => c.type === StreamEventType.CHUNK &&
642
+ c.value.candidates?.[0]?.content?.parts?.[0]?.text ===
643
+ 'Successful response')).toBe(true);
644
+ // Check that history was recorded correctly once, with no duplicates.
149
645
  const history = chat.getHistory();
150
646
  expect(history.length).toBe(2);
151
- expect(history[1].parts).toEqual([{ text: 'M1M2M3' }]);
647
+ expect(history[0]).toEqual({
648
+ role: 'user',
649
+ parts: [{ text: 'test' }],
650
+ });
651
+ expect(history[1]).toEqual({
652
+ role: 'model',
653
+ parts: [{ text: 'Successful response' }],
654
+ });
655
+ // Verify that token counting is not called when usageMetadata is missing
656
+ expect(uiTelemetryService.setLastPromptTokenCount).not.toHaveBeenCalled();
152
657
  });
153
- it('should not consolidate if roles are different between model outputs', () => {
154
- const modelOutputParts = [
155
- { role: 'model', parts: [{ text: 'M1' }] },
156
- { role: 'user', parts: [{ text: 'Interjecting User' }] },
157
- { role: 'model', parts: [{ text: 'M2' }] },
158
- ];
159
- // @ts-expect-error Accessing private method for testing purposes
160
- chat.recordHistory(userInput, modelOutputParts);
658
+ it('should fail after all retries on persistent invalid content and report metrics', async () => {
659
+ vi.mocked(mockContentGenerator.generateContentStream).mockImplementation(async () => (async function* () {
660
+ yield {
661
+ candidates: [
662
+ {
663
+ content: {
664
+ parts: [{ text: '' }],
665
+ role: 'model',
666
+ },
667
+ },
668
+ ],
669
+ };
670
+ })());
671
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-retry-fail');
672
+ await expect(async () => {
673
+ for await (const _ of stream) {
674
+ // Must loop to trigger the internal logic that throws.
675
+ }
676
+ }).rejects.toThrow(InvalidStreamError);
677
+ // Should be called 2 times (initial + 1 retry)
678
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(2);
679
+ expect(mockLogContentRetry).toHaveBeenCalledTimes(1);
680
+ expect(mockLogContentRetryFailure).toHaveBeenCalledTimes(1);
681
+ // History should still contain the user message.
161
682
  const history = chat.getHistory();
162
- expect(history.length).toBe(4); // user, M1, Interjecting User, M2
163
- expect(history[1].parts).toEqual([{ text: 'M1' }]);
164
- expect(history[3].parts).toEqual([{ text: 'M2' }]);
165
- });
166
- it('should merge with last history entry if it is also a model output', () => {
167
- // @ts-expect-error Accessing private property for test setup
168
- chat.history = [
169
- userInput,
170
- { role: 'model', parts: [{ text: 'Initial Model Output' }] },
171
- ]; // Prime the history
172
- const newModelOutput = [
173
- { role: 'model', parts: [{ text: 'New Model Part 1' }] },
174
- { role: 'model', parts: [{ text: 'New Model Part 2' }] },
175
- ];
176
- // @ts-expect-error Accessing private method for testing purposes
177
- chat.recordHistory(userInput, newModelOutput); // userInput here is for the *next* turn, but history is already primed
178
- // Reset and set up a more realistic scenario for merging with existing history
179
- chat = new GeminiChat(mockConfig, mockModelsModule, config, []);
180
- const firstUserInput = {
683
+ expect(history.length).toBe(1);
684
+ expect(history[0]).toEqual({
181
685
  role: 'user',
182
- parts: [{ text: 'First user input' }],
686
+ parts: [{ text: 'test' }],
687
+ });
688
+ });
689
+ describe('API error retry behavior', () => {
690
+ beforeEach(() => {
691
+ // Use a more direct mock for retry testing
692
+ mockRetryWithBackoff.mockImplementation(async (apiCall, options) => {
693
+ try {
694
+ return await apiCall();
695
+ }
696
+ catch (error) {
697
+ if (options?.shouldRetryOnError &&
698
+ options.shouldRetryOnError(error)) {
699
+ // Try again
700
+ return await apiCall();
701
+ }
702
+ throw error;
703
+ }
704
+ });
705
+ });
706
+ it('should not retry on 400 Bad Request errors', async () => {
707
+ const error400 = new ApiError({ message: 'Bad Request', status: 400 });
708
+ vi.mocked(mockContentGenerator.generateContentStream).mockRejectedValue(error400);
709
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-400');
710
+ await expect((async () => {
711
+ for await (const _ of stream) {
712
+ /* consume stream */
713
+ }
714
+ })()).rejects.toThrow(error400);
715
+ // Should only be called once (no retry)
716
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(1);
717
+ });
718
+ it('should retry on 429 Rate Limit errors', async () => {
719
+ const error429 = new ApiError({ message: 'Rate Limited', status: 429 });
720
+ vi.mocked(mockContentGenerator.generateContentStream)
721
+ .mockRejectedValueOnce(error429)
722
+ .mockResolvedValueOnce((async function* () {
723
+ yield {
724
+ candidates: [
725
+ {
726
+ content: { parts: [{ text: 'Success after retry' }] },
727
+ finishReason: 'STOP',
728
+ },
729
+ ],
730
+ };
731
+ })());
732
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-429-retry');
733
+ const events = [];
734
+ for await (const event of stream) {
735
+ events.push(event);
736
+ }
737
+ // Should be called twice (initial + retry)
738
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(2);
739
+ // Should have successful content
740
+ expect(events.some((e) => e.type === StreamEventType.CHUNK &&
741
+ e.value.candidates?.[0]?.content?.parts?.[0]?.text ===
742
+ 'Success after retry')).toBe(true);
743
+ });
744
+ it('should not retry on schema depth errors', async () => {
745
+ const schemaError = new ApiError({
746
+ message: 'Request failed: maximum schema depth exceeded',
747
+ status: 500,
748
+ });
749
+ vi.mocked(mockContentGenerator.generateContentStream).mockRejectedValue(schemaError);
750
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-schema');
751
+ await expect((async () => {
752
+ for await (const _ of stream) {
753
+ /* consume stream */
754
+ }
755
+ })()).rejects.toThrow(schemaError);
756
+ // Should only be called once (no retry)
757
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(1);
758
+ });
759
+ it('should retry on 5xx server errors', async () => {
760
+ const error500 = new ApiError({
761
+ message: 'Internal Server Error 500',
762
+ status: 500,
763
+ });
764
+ vi.mocked(mockContentGenerator.generateContentStream)
765
+ .mockRejectedValueOnce(error500)
766
+ .mockResolvedValueOnce((async function* () {
767
+ yield {
768
+ candidates: [
769
+ {
770
+ content: { parts: [{ text: 'Recovered from 500' }] },
771
+ finishReason: 'STOP',
772
+ },
773
+ ],
774
+ };
775
+ })());
776
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-500-retry');
777
+ const events = [];
778
+ for await (const event of stream) {
779
+ events.push(event);
780
+ }
781
+ // Should be called twice (initial + retry)
782
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(2);
783
+ });
784
+ afterEach(() => {
785
+ // Reset to default behavior
786
+ mockRetryWithBackoff.mockImplementation(async (apiCall) => apiCall());
787
+ });
788
+ });
789
+ });
790
+ it('should correctly retry and append to an existing history mid-conversation', async () => {
791
+ // 1. Setup
792
+ const initialHistory = [
793
+ { role: 'user', parts: [{ text: 'First question' }] },
794
+ { role: 'model', parts: [{ text: 'First answer' }] },
795
+ ];
796
+ chat.setHistory(initialHistory);
797
+ // 2. Mock the API to fail once with an empty stream, then succeed.
798
+ vi.mocked(mockContentGenerator.generateContentStream)
799
+ .mockImplementationOnce(async () => (async function* () {
800
+ yield {
801
+ candidates: [{ content: { parts: [{ text: '' }] } }],
183
802
  };
184
- const firstModelOutput = [
185
- { role: 'model', parts: [{ text: 'First model response' }] },
186
- ];
187
- // @ts-expect-error Accessing private method for testing purposes
188
- chat.recordHistory(firstUserInput, firstModelOutput);
189
- const secondUserInput = {
190
- role: 'user',
191
- parts: [{ text: 'Second user input' }],
803
+ })())
804
+ .mockImplementationOnce(async () =>
805
+ // Second attempt succeeds
806
+ (async function* () {
807
+ yield {
808
+ candidates: [
809
+ {
810
+ content: { parts: [{ text: 'Second answer' }] },
811
+ finishReason: 'STOP',
812
+ },
813
+ ],
192
814
  };
193
- const secondModelOutput = [
194
- { role: 'model', parts: [{ text: 'Second model response part 1' }] },
195
- { role: 'model', parts: [{ text: 'Second model response part 2' }] },
196
- ];
197
- // @ts-expect-error Accessing private method for testing purposes
198
- chat.recordHistory(secondUserInput, secondModelOutput);
199
- const finalHistory = chat.getHistory();
200
- expect(finalHistory.length).toBe(4); // user1, model1, user2, model2(consolidated)
201
- expect(finalHistory[0]).toEqual(firstUserInput);
202
- expect(finalHistory[1]).toEqual(firstModelOutput[0]);
203
- expect(finalHistory[2]).toEqual(secondUserInput);
204
- expect(finalHistory[3].role).toBe('model');
205
- expect(finalHistory[3].parts).toEqual([
206
- { text: 'Second model response part 1Second model response part 2' },
207
- ]);
815
+ })());
816
+ // 3. Send a new message
817
+ const stream = await chat.sendMessageStream('test-model', { message: 'Second question' }, 'prompt-id-retry-existing');
818
+ for await (const _ of stream) {
819
+ // consume stream
820
+ }
821
+ // 4. Assert the final history and metrics
822
+ const history = chat.getHistory();
823
+ expect(history.length).toBe(4);
824
+ // Assert that the correct metrics were reported for one empty-stream retry
825
+ expect(mockLogContentRetry).toHaveBeenCalledTimes(1);
826
+ // Explicitly verify the structure of each part to satisfy TypeScript
827
+ const turn1 = history[0];
828
+ if (!turn1?.parts?.[0] || !('text' in turn1.parts[0])) {
829
+ throw new Error('Test setup error: First turn is not a valid text part.');
830
+ }
831
+ expect(turn1.parts[0].text).toBe('First question');
832
+ const turn2 = history[1];
833
+ if (!turn2?.parts?.[0] || !('text' in turn2.parts[0])) {
834
+ throw new Error('Test setup error: Second turn is not a valid text part.');
835
+ }
836
+ expect(turn2.parts[0].text).toBe('First answer');
837
+ const turn3 = history[2];
838
+ if (!turn3?.parts?.[0] || !('text' in turn3.parts[0])) {
839
+ throw new Error('Test setup error: Third turn is not a valid text part.');
840
+ }
841
+ expect(turn3.parts[0].text).toBe('Second question');
842
+ const turn4 = history[3];
843
+ if (!turn4?.parts?.[0] || !('text' in turn4.parts[0])) {
844
+ throw new Error('Test setup error: Fourth turn is not a valid text part.');
845
+ }
846
+ expect(turn4.parts[0].text).toBe('Second answer');
847
+ });
848
+ it('should retry if the model returns a completely empty stream (no chunks)', async () => {
849
+ // 1. Mock the API to return an empty stream first, then a valid one.
850
+ vi.mocked(mockContentGenerator.generateContentStream)
851
+ .mockImplementationOnce(
852
+ // First call resolves to an async generator that yields nothing.
853
+ async () => (async function* () { })())
854
+ .mockImplementationOnce(
855
+ // Second call returns a valid stream.
856
+ async () => (async function* () {
857
+ yield {
858
+ candidates: [
859
+ {
860
+ content: {
861
+ parts: [{ text: 'Successful response after empty' }],
862
+ },
863
+ finishReason: 'STOP',
864
+ },
865
+ ],
866
+ };
867
+ })());
868
+ // 2. Call the method and consume the stream.
869
+ const stream = await chat.sendMessageStream('test-model', { message: 'test empty stream' }, 'prompt-id-empty-stream');
870
+ const chunks = [];
871
+ for await (const chunk of stream) {
872
+ chunks.push(chunk);
873
+ }
874
+ // 3. Assert the results.
875
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(2);
876
+ expect(chunks.some((c) => c.type === StreamEventType.CHUNK &&
877
+ c.value.candidates?.[0]?.content?.parts?.[0]?.text ===
878
+ 'Successful response after empty')).toBe(true);
879
+ const history = chat.getHistory();
880
+ expect(history.length).toBe(2);
881
+ // Explicitly verify the structure of each part to satisfy TypeScript
882
+ const turn1 = history[0];
883
+ if (!turn1?.parts?.[0] || !('text' in turn1.parts[0])) {
884
+ throw new Error('Test setup error: First turn is not a valid text part.');
885
+ }
886
+ expect(turn1.parts[0].text).toBe('test empty stream');
887
+ const turn2 = history[1];
888
+ if (!turn2?.parts?.[0] || !('text' in turn2.parts[0])) {
889
+ throw new Error('Test setup error: Second turn is not a valid text part.');
890
+ }
891
+ expect(turn2.parts[0].text).toBe('Successful response after empty');
892
+ });
893
+ it('should queue a subsequent sendMessageStream call until the first stream is fully consumed', async () => {
894
+ // 1. Create a promise to manually control the stream's lifecycle
895
+ let continueFirstStream;
896
+ const firstStreamContinuePromise = new Promise((resolve) => {
897
+ continueFirstStream = resolve;
208
898
  });
209
- it('should correctly merge consolidated new output with existing model history', () => {
210
- // Setup: history ends with a model turn
211
- const initialUser = {
212
- role: 'user',
213
- parts: [{ text: 'Initial user query' }],
899
+ // 2. Mock the API to return controllable async generators
900
+ const firstStreamGenerator = (async function* () {
901
+ yield {
902
+ candidates: [
903
+ { content: { parts: [{ text: 'first response part 1' }] } },
904
+ ],
214
905
  };
215
- const initialModel = {
216
- role: 'model',
217
- parts: [{ text: 'Initial model answer.' }],
906
+ await firstStreamContinuePromise; // Pause the stream
907
+ yield {
908
+ candidates: [
909
+ {
910
+ content: { parts: [{ text: ' part 2' }] },
911
+ finishReason: 'STOP',
912
+ },
913
+ ],
218
914
  };
219
- chat = new GeminiChat(mockConfig, mockModelsModule, config, [
220
- initialUser,
221
- initialModel,
222
- ]);
223
- // New interaction
224
- const currentUserInput = {
225
- role: 'user',
226
- parts: [{ text: 'Follow-up question' }],
915
+ })();
916
+ const secondStreamGenerator = (async function* () {
917
+ yield {
918
+ candidates: [
919
+ {
920
+ content: { parts: [{ text: 'second response' }] },
921
+ finishReason: 'STOP',
922
+ },
923
+ ],
227
924
  };
228
- const newModelParts = [
229
- { role: 'model', parts: [{ text: 'Part A of new answer.' }] },
230
- { role: 'model', parts: [{ text: 'Part B of new answer.' }] },
231
- ];
232
- // @ts-expect-error Accessing private method for testing purposes
233
- chat.recordHistory(currentUserInput, newModelParts);
234
- const history = chat.getHistory();
235
- // Expected: initialUser, initialModel, currentUserInput, consolidatedNewModelParts
236
- expect(history.length).toBe(4);
237
- expect(history[0]).toEqual(initialUser);
238
- expect(history[1]).toEqual(initialModel);
239
- expect(history[2]).toEqual(currentUserInput);
240
- expect(history[3].role).toBe('model');
241
- expect(history[3].parts).toEqual([
242
- { text: 'Part A of new answer.Part B of new answer.' },
243
- ]);
925
+ })();
926
+ vi.mocked(mockContentGenerator.generateContentStream)
927
+ .mockResolvedValueOnce(firstStreamGenerator)
928
+ .mockResolvedValueOnce(secondStreamGenerator);
929
+ // 3. Start the first stream and consume only the first chunk to pause it
930
+ const firstStream = await chat.sendMessageStream('test-model', { message: 'first' }, 'prompt-1');
931
+ const firstStreamIterator = firstStream[Symbol.asyncIterator]();
932
+ await firstStreamIterator.next();
933
+ // 4. While the first stream is paused, start the second call. It will block.
934
+ const secondStreamPromise = chat.sendMessageStream('test-model', { message: 'second' }, 'prompt-2');
935
+ // 5. Assert that only one API call has been made so far.
936
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(1);
937
+ // 6. Unblock and fully consume the first stream to completion.
938
+ continueFirstStream();
939
+ await firstStreamIterator.next(); // Consume the rest of the stream
940
+ await firstStreamIterator.next(); // Finish the iterator
941
+ // 7. Now that the first stream is done, await the second promise to get its generator.
942
+ const secondStream = await secondStreamPromise;
943
+ // 8. Start consuming the second stream, which triggers its internal API call.
944
+ const secondStreamIterator = secondStream[Symbol.asyncIterator]();
945
+ await secondStreamIterator.next();
946
+ // 9. The second API call should now have been made.
947
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(2);
948
+ // 10. FIX: Fully consume the second stream to ensure recordHistory is called.
949
+ await secondStreamIterator.next(); // This finishes the iterator.
950
+ // 11. Final check on history.
951
+ const history = chat.getHistory();
952
+ expect(history.length).toBe(4);
953
+ const turn4 = history[3];
954
+ if (!turn4?.parts?.[0] || !('text' in turn4.parts[0])) {
955
+ throw new Error('Test setup error: Fourth turn is not a valid text part.');
956
+ }
957
+ expect(turn4.parts[0].text).toBe('second response');
958
+ });
959
+ describe('stopBeforeSecondMutator', () => {
960
+ beforeEach(() => {
961
+ // Common setup for these tests: mock the tool registry.
962
+ const mockToolRegistry = {
963
+ getTool: vi.fn((toolName) => {
964
+ if (toolName === 'edit') {
965
+ return { kind: Kind.Edit };
966
+ }
967
+ return { kind: Kind.Other };
968
+ }),
969
+ };
970
+ vi.mocked(mockConfig.getToolRegistry).mockReturnValue(mockToolRegistry);
244
971
  });
245
- it('should handle empty modelOutput array', () => {
246
- // @ts-expect-error Accessing private method for testing purposes
247
- chat.recordHistory(userInput, []);
972
+ it('should stop streaming before a second mutator tool call', async () => {
973
+ const responses = [
974
+ {
975
+ candidates: [
976
+ { content: { role: 'model', parts: [{ text: 'First part. ' }] } },
977
+ ],
978
+ },
979
+ {
980
+ candidates: [
981
+ {
982
+ content: {
983
+ role: 'model',
984
+ parts: [{ functionCall: { name: 'edit', args: {} } }],
985
+ },
986
+ },
987
+ ],
988
+ },
989
+ {
990
+ candidates: [
991
+ {
992
+ content: {
993
+ role: 'model',
994
+ parts: [{ functionCall: { name: 'fetch', args: {} } }],
995
+ },
996
+ },
997
+ ],
998
+ },
999
+ // This chunk contains the second mutator and should be clipped.
1000
+ {
1001
+ candidates: [
1002
+ {
1003
+ content: {
1004
+ role: 'model',
1005
+ parts: [
1006
+ { functionCall: { name: 'edit', args: {} } },
1007
+ { text: 'some trailing text' },
1008
+ ],
1009
+ },
1010
+ },
1011
+ ],
1012
+ },
1013
+ // This chunk should never be reached.
1014
+ {
1015
+ candidates: [
1016
+ {
1017
+ content: {
1018
+ role: 'model',
1019
+ parts: [{ text: 'This should not appear.' }],
1020
+ },
1021
+ },
1022
+ ],
1023
+ },
1024
+ ];
1025
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue((async function* () {
1026
+ for (const response of responses) {
1027
+ yield response;
1028
+ }
1029
+ })());
1030
+ const stream = await chat.sendMessageStream('test-model', { message: 'test message' }, 'prompt-id-mutator-test');
1031
+ for await (const _ of stream) {
1032
+ // Consume the stream to trigger history recording.
1033
+ }
248
1034
  const history = chat.getHistory();
249
- // If modelOutput is empty, it might push a default empty model part depending on isFunctionResponse
250
- // Assuming isFunctionResponse(userInput) is false for this simple text input
251
1035
  expect(history.length).toBe(2);
252
- expect(history[0]).toEqual(userInput);
253
- expect(history[1].role).toBe('model');
254
- expect(history[1].parts).toEqual([]);
1036
+ const modelTurn = history[1];
1037
+ expect(modelTurn.role).toBe('model');
1038
+ expect(modelTurn?.parts?.length).toBe(3);
1039
+ expect(modelTurn?.parts[0].text).toBe('First part. ');
1040
+ expect(modelTurn.parts[1].functionCall?.name).toBe('edit');
1041
+ expect(modelTurn.parts[2].functionCall?.name).toBe('fetch');
255
1042
  });
256
- it('should handle aggregating modelOutput', () => {
257
- const modelOutputUndefinedParts = [
258
- { role: 'model', parts: [{ text: 'First model part' }] },
259
- { role: 'model', parts: [{ text: 'Second model part' }] },
260
- { role: 'model', parts: undefined }, // Test undefined parts
261
- { role: 'model', parts: [{ text: 'Third model part' }] },
262
- { role: 'model', parts: [] }, // Test empty parts array
1043
+ it('should not stop streaming if only one mutator is present', async () => {
1044
+ const responses = [
1045
+ {
1046
+ candidates: [
1047
+ { content: { role: 'model', parts: [{ text: 'Part 1. ' }] } },
1048
+ ],
1049
+ },
1050
+ {
1051
+ candidates: [
1052
+ {
1053
+ content: {
1054
+ role: 'model',
1055
+ parts: [{ functionCall: { name: 'edit', args: {} } }],
1056
+ },
1057
+ },
1058
+ ],
1059
+ },
1060
+ {
1061
+ candidates: [
1062
+ {
1063
+ content: {
1064
+ role: 'model',
1065
+ parts: [{ text: 'Part 2.' }],
1066
+ },
1067
+ finishReason: 'STOP',
1068
+ },
1069
+ ],
1070
+ },
263
1071
  ];
264
- // @ts-expect-error Accessing private method for testing purposes
265
- chat.recordHistory(userInput, modelOutputUndefinedParts);
1072
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue((async function* () {
1073
+ for (const response of responses) {
1074
+ yield response;
1075
+ }
1076
+ })());
1077
+ const stream = await chat.sendMessageStream('test-model', { message: 'test message' }, 'prompt-id-one-mutator');
1078
+ for await (const _ of stream) {
1079
+ /* consume */
1080
+ }
266
1081
  const history = chat.getHistory();
267
- expect(history.length).toBe(5);
268
- expect(history[0]).toEqual(userInput);
269
- expect(history[1].role).toBe('model');
270
- expect(history[1].parts).toEqual([
271
- { text: 'First model partSecond model part' },
272
- ]);
273
- expect(history[2].role).toBe('model');
274
- expect(history[2].parts).toBeUndefined();
275
- expect(history[3].role).toBe('model');
276
- expect(history[3].parts).toEqual([{ text: 'Third model part' }]);
277
- expect(history[4].role).toBe('model');
278
- expect(history[4].parts).toEqual([]);
1082
+ const modelTurn = history[1];
1083
+ expect(modelTurn?.parts?.length).toBe(3);
1084
+ expect(modelTurn.parts[1].functionCall?.name).toBe('edit');
1085
+ expect(modelTurn.parts[2].text).toBe('Part 2.');
279
1086
  });
280
- it('should handle modelOutput with parts being undefined or empty (if they pass initial every check)', () => {
281
- const modelOutputUndefinedParts = [
282
- { role: 'model', parts: [{ text: 'Text part' }] },
283
- { role: 'model', parts: undefined }, // Test undefined parts
284
- { role: 'model', parts: [] }, // Test empty parts array
1087
+ it('should clip the chunk containing the second mutator, preserving prior parts', async () => {
1088
+ const responses = [
1089
+ {
1090
+ candidates: [
1091
+ {
1092
+ content: {
1093
+ role: 'model',
1094
+ parts: [{ functionCall: { name: 'edit', args: {} } }],
1095
+ },
1096
+ },
1097
+ ],
1098
+ },
1099
+ // This chunk has a valid part before the second mutator.
1100
+ // The valid part should be kept, the rest of the chunk discarded.
1101
+ {
1102
+ candidates: [
1103
+ {
1104
+ content: {
1105
+ role: 'model',
1106
+ parts: [
1107
+ { text: 'Keep this text. ' },
1108
+ { functionCall: { name: 'edit', args: {} } },
1109
+ { text: 'Discard this text.' },
1110
+ ],
1111
+ },
1112
+ finishReason: 'STOP',
1113
+ },
1114
+ ],
1115
+ },
285
1116
  ];
286
- // @ts-expect-error Accessing private method for testing purposes
287
- chat.recordHistory(userInput, modelOutputUndefinedParts);
1117
+ const stream = (async function* () {
1118
+ for (const response of responses) {
1119
+ yield response;
1120
+ }
1121
+ })();
1122
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(stream);
1123
+ const resultStream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-clip-chunk');
1124
+ for await (const _ of resultStream) {
1125
+ /* consume */
1126
+ }
288
1127
  const history = chat.getHistory();
289
- expect(history.length).toBe(4); // userInput, model1 (text), model2 (undefined parts), model3 (empty parts)
290
- expect(history[0]).toEqual(userInput);
291
- expect(history[1].role).toBe('model');
292
- expect(history[1].parts).toEqual([{ text: 'Text part' }]);
293
- expect(history[2].role).toBe('model');
294
- expect(history[2].parts).toBeUndefined();
295
- expect(history[3].role).toBe('model');
296
- expect(history[3].parts).toEqual([]);
1128
+ const modelTurn = history[1];
1129
+ expect(modelTurn?.parts?.length).toBe(2);
1130
+ expect(modelTurn.parts[0].functionCall?.name).toBe('edit');
1131
+ expect(modelTurn.parts[1].text).toBe('Keep this text. ');
297
1132
  });
298
- it('should correctly handle automaticFunctionCallingHistory', () => {
299
- const afcHistory = [
300
- { role: 'user', parts: [{ text: 'AFC User' }] },
301
- { role: 'model', parts: [{ text: 'AFC Model' }] },
302
- ];
303
- const modelOutput = [
304
- { role: 'model', parts: [{ text: 'Regular Model Output' }] },
1133
+ it('should handle two mutators in the same chunk (parallel call scenario)', async () => {
1134
+ const responses = [
1135
+ {
1136
+ candidates: [
1137
+ {
1138
+ content: {
1139
+ role: 'model',
1140
+ parts: [
1141
+ { text: 'Some text. ' },
1142
+ { functionCall: { name: 'edit', args: {} } },
1143
+ { functionCall: { name: 'edit', args: {} } },
1144
+ ],
1145
+ },
1146
+ finishReason: 'STOP',
1147
+ },
1148
+ ],
1149
+ },
305
1150
  ];
306
- // @ts-expect-error Accessing private method for testing purposes
307
- chat.recordHistory(userInput, modelOutput, afcHistory);
1151
+ const stream = (async function* () {
1152
+ for (const response of responses) {
1153
+ yield response;
1154
+ }
1155
+ })();
1156
+ vi.mocked(mockContentGenerator.generateContentStream).mockResolvedValue(stream);
1157
+ const resultStream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-parallel-mutators');
1158
+ for await (const _ of resultStream) {
1159
+ /* consume */
1160
+ }
308
1161
  const history = chat.getHistory();
309
- expect(history.length).toBe(3);
310
- expect(history[0]).toEqual(afcHistory[0]);
311
- expect(history[1]).toEqual(afcHistory[1]);
312
- expect(history[2]).toEqual(modelOutput[0]);
1162
+ const modelTurn = history[1];
1163
+ expect(modelTurn?.parts?.length).toBe(2);
1164
+ expect(modelTurn.parts[0].text).toBe('Some text. ');
1165
+ expect(modelTurn.parts[1].functionCall?.name).toBe('edit');
313
1166
  });
314
- it('should add userInput if AFC history is present but empty', () => {
315
- const modelOutput = [
316
- { role: 'model', parts: [{ text: 'Model Output' }] },
317
- ];
318
- // @ts-expect-error Accessing private method for testing purposes
319
- chat.recordHistory(userInput, modelOutput, []); // Empty AFC history
320
- const history = chat.getHistory();
321
- expect(history.length).toBe(2);
322
- expect(history[0]).toEqual(userInput);
323
- expect(history[1]).toEqual(modelOutput[0]);
1167
+ });
1168
+ describe('Model Resolution', () => {
1169
+ const mockResponse = {
1170
+ candidates: [
1171
+ {
1172
+ content: { parts: [{ text: 'response' }], role: 'model' },
1173
+ finishReason: 'STOP',
1174
+ },
1175
+ ],
1176
+ };
1177
+ it('should use the FLASH model when in fallback mode (sendMessageStream)', async () => {
1178
+ vi.mocked(mockConfig.getModel).mockReturnValue('gemini-pro');
1179
+ vi.mocked(mockConfig.isInFallbackMode).mockReturnValue(true);
1180
+ vi.mocked(mockContentGenerator.generateContentStream).mockImplementation(async () => (async function* () {
1181
+ yield mockResponse;
1182
+ })());
1183
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-res3');
1184
+ for await (const _ of stream) {
1185
+ // consume stream
1186
+ }
1187
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledWith(expect.objectContaining({
1188
+ model: DEFAULT_GEMINI_FLASH_MODEL,
1189
+ }), 'prompt-id-res3');
324
1190
  });
325
- it('should skip "thought" content from modelOutput', () => {
326
- const modelOutputWithThought = [
327
- { role: 'model', parts: [{ thought: true }, { text: 'Visible text' }] },
328
- { role: 'model', parts: [{ text: 'Another visible text' }] },
329
- ];
330
- // @ts-expect-error Accessing private method for testing purposes
331
- chat.recordHistory(userInput, modelOutputWithThought);
332
- const history = chat.getHistory();
333
- expect(history.length).toBe(2); // User input + consolidated model output
334
- expect(history[0]).toEqual(userInput);
335
- expect(history[1].role).toBe('model');
336
- // The 'thought' part is skipped, 'Another visible text' becomes the first part.
337
- expect(history[1].parts).toEqual([{ text: 'Another visible text' }]);
1191
+ });
1192
+ describe('Fallback Integration (Retries)', () => {
1193
+ const error429 = new ApiError({
1194
+ message: 'API Error 429: Quota exceeded',
1195
+ status: 429,
338
1196
  });
339
- it('should skip "thought" content even if it is the only content', () => {
340
- const modelOutputOnlyThought = [
341
- { role: 'model', parts: [{ thought: true }] },
342
- ];
343
- // @ts-expect-error Accessing private method for testing purposes
344
- chat.recordHistory(userInput, modelOutputOnlyThought);
345
- const history = chat.getHistory();
346
- expect(history.length).toBe(1); // User input + default empty model part
347
- expect(history[0]).toEqual(userInput);
1197
+ // Define the simulated behavior for retryWithBackoff for these tests.
1198
+ // This simulation tries the apiCall, if it fails, it calls the callback,
1199
+ // and then tries the apiCall again if the callback returns true.
1200
+ const simulateRetryBehavior = async (apiCall, options) => {
1201
+ try {
1202
+ return await apiCall();
1203
+ }
1204
+ catch (error) {
1205
+ if (options.onPersistent429) {
1206
+ // We simulate the "persistent" trigger here for simplicity.
1207
+ const shouldRetry = await options.onPersistent429(options.authType, error);
1208
+ if (shouldRetry) {
1209
+ return await apiCall();
1210
+ }
1211
+ }
1212
+ throw error; // Stop if callback returns false/null or doesn't exist
1213
+ }
1214
+ };
1215
+ beforeEach(() => {
1216
+ mockRetryWithBackoff.mockImplementation(simulateRetryBehavior);
348
1217
  });
349
- it('should correctly consolidate text parts when a thought part is in between', () => {
350
- const modelOutputMixed = [
351
- { role: 'model', parts: [{ text: 'Part 1.' }] },
352
- {
353
- role: 'model',
354
- parts: [{ thought: true }, { text: 'Should be skipped' }],
355
- },
356
- { role: 'model', parts: [{ text: 'Part 2.' }] },
357
- ];
358
- // @ts-expect-error Accessing private method for testing purposes
359
- chat.recordHistory(userInput, modelOutputMixed);
360
- const history = chat.getHistory();
361
- expect(history.length).toBe(2);
362
- expect(history[0]).toEqual(userInput);
363
- expect(history[1].role).toBe('model');
364
- expect(history[1].parts).toEqual([{ text: 'Part 1.Part 2.' }]);
1218
+ afterEach(() => {
1219
+ mockRetryWithBackoff.mockImplementation(async (apiCall) => apiCall());
365
1220
  });
366
- it('should handle multiple thought parts correctly', () => {
367
- const modelOutputMultipleThoughts = [
368
- { role: 'model', parts: [{ thought: true }] },
369
- { role: 'model', parts: [{ text: 'Visible 1' }] },
370
- { role: 'model', parts: [{ thought: true }] },
371
- { role: 'model', parts: [{ text: 'Visible 2' }] },
372
- ];
373
- // @ts-expect-error Accessing private method for testing purposes
374
- chat.recordHistory(userInput, modelOutputMultipleThoughts);
1221
+ it('should call handleFallback with the specific failed model and retry if handler returns true', async () => {
1222
+ const authType = AuthType.LOGIN_WITH_GOOGLE;
1223
+ vi.mocked(mockConfig.getContentGeneratorConfig).mockReturnValue({
1224
+ authType,
1225
+ });
1226
+ const isInFallbackModeSpy = vi.spyOn(mockConfig, 'isInFallbackMode');
1227
+ isInFallbackModeSpy.mockReturnValue(false);
1228
+ vi.mocked(mockContentGenerator.generateContentStream)
1229
+ .mockRejectedValueOnce(error429) // Attempt 1 fails
1230
+ .mockResolvedValueOnce(
1231
+ // Attempt 2 succeeds
1232
+ (async function* () {
1233
+ yield {
1234
+ candidates: [
1235
+ {
1236
+ content: { parts: [{ text: 'Success on retry' }] },
1237
+ finishReason: 'STOP',
1238
+ },
1239
+ ],
1240
+ };
1241
+ })());
1242
+ mockHandleFallback.mockImplementation(async () => {
1243
+ isInFallbackModeSpy.mockReturnValue(true);
1244
+ return true; // Signal retry
1245
+ });
1246
+ const stream = await chat.sendMessageStream('test-model', { message: 'trigger 429' }, 'prompt-id-fb1');
1247
+ // Consume stream to trigger logic
1248
+ for await (const _ of stream) {
1249
+ // no-op
1250
+ }
1251
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(2);
1252
+ expect(mockHandleFallback).toHaveBeenCalledTimes(1);
1253
+ expect(mockHandleFallback).toHaveBeenCalledWith(mockConfig, 'test-model', authType, error429);
375
1254
  const history = chat.getHistory();
376
- expect(history.length).toBe(2);
377
- expect(history[0]).toEqual(userInput);
378
- expect(history[1].role).toBe('model');
379
- expect(history[1].parts).toEqual([{ text: 'Visible 1Visible 2' }]);
1255
+ const modelTurn = history[1];
1256
+ expect(modelTurn.parts[0].text).toBe('Success on retry');
380
1257
  });
381
- it('should handle thought part at the end of outputContents', () => {
382
- const modelOutputThoughtAtEnd = [
383
- { role: 'model', parts: [{ text: 'Visible text' }] },
384
- { role: 'model', parts: [{ thought: true }] },
385
- ];
386
- // @ts-expect-error Accessing private method for testing purposes
387
- chat.recordHistory(userInput, modelOutputThoughtAtEnd);
388
- const history = chat.getHistory();
389
- expect(history.length).toBe(2);
390
- expect(history[0]).toEqual(userInput);
391
- expect(history[1].role).toBe('model');
392
- expect(history[1].parts).toEqual([{ text: 'Visible text' }]);
1258
+ it('should stop retrying if handleFallback returns false (e.g., auth intent)', async () => {
1259
+ vi.mocked(mockConfig.getModel).mockReturnValue('gemini-pro');
1260
+ vi.mocked(mockContentGenerator.generateContentStream).mockRejectedValue(error429);
1261
+ mockHandleFallback.mockResolvedValue(false);
1262
+ const stream = await chat.sendMessageStream('test-model', { message: 'test stop' }, 'prompt-id-fb2');
1263
+ await expect((async () => {
1264
+ for await (const _ of stream) {
1265
+ /* consume stream */
1266
+ }
1267
+ })()).rejects.toThrow(error429);
1268
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(1);
1269
+ expect(mockHandleFallback).toHaveBeenCalledTimes(1);
393
1270
  });
394
1271
  });
395
- describe('addHistory', () => {
396
- it('should add a new content item to the history', () => {
397
- const newContent = {
398
- role: 'user',
399
- parts: [{ text: 'A new message' }],
1272
+ it('should discard valid partial content from a failed attempt upon retry', async () => {
1273
+ // Mock the stream to fail on the first attempt after yielding some valid content.
1274
+ vi.mocked(mockContentGenerator.generateContentStream)
1275
+ .mockImplementationOnce(async () =>
1276
+ // First attempt: yields one valid chunk, then one invalid chunk
1277
+ (async function* () {
1278
+ yield {
1279
+ candidates: [
1280
+ {
1281
+ content: {
1282
+ parts: [{ text: 'This valid part should be discarded' }],
1283
+ },
1284
+ },
1285
+ ],
400
1286
  };
401
- chat.addHistory(newContent);
402
- const history = chat.getHistory();
403
- expect(history.length).toBe(1);
404
- expect(history[0]).toEqual(newContent);
405
- });
406
- it('should add multiple items correctly', () => {
407
- const content1 = {
408
- role: 'user',
409
- parts: [{ text: 'Message 1' }],
1287
+ yield {
1288
+ candidates: [{ content: { parts: [{ text: '' }] } }], // Invalid chunk triggers retry
410
1289
  };
411
- const content2 = {
412
- role: 'model',
413
- parts: [{ text: 'Message 2' }],
1290
+ })())
1291
+ .mockImplementationOnce(async () =>
1292
+ // Second attempt (the retry): succeeds
1293
+ (async function* () {
1294
+ yield {
1295
+ candidates: [
1296
+ {
1297
+ content: {
1298
+ parts: [{ text: 'Successful final response' }],
1299
+ },
1300
+ finishReason: 'STOP',
1301
+ },
1302
+ ],
414
1303
  };
415
- chat.addHistory(content1);
416
- chat.addHistory(content2);
417
- const history = chat.getHistory();
418
- expect(history.length).toBe(2);
419
- expect(history[0]).toEqual(content1);
420
- expect(history[1]).toEqual(content2);
1304
+ })());
1305
+ // Send a message and consume the stream
1306
+ const stream = await chat.sendMessageStream('test-model', { message: 'test' }, 'prompt-id-discard-test');
1307
+ const events = [];
1308
+ for await (const event of stream) {
1309
+ events.push(event);
1310
+ }
1311
+ // Check that a retry happened
1312
+ expect(mockContentGenerator.generateContentStream).toHaveBeenCalledTimes(2);
1313
+ expect(events.some((e) => e.type === StreamEventType.RETRY)).toBe(true);
1314
+ // Check the final recorded history
1315
+ const history = chat.getHistory();
1316
+ expect(history.length).toBe(2); // user turn + final model turn
1317
+ const modelTurn = history[1];
1318
+ // The model turn should only contain the text from the successful attempt
1319
+ expect(modelTurn.parts[0].text).toBe('Successful final response');
1320
+ // It should NOT contain any text from the failed attempt
1321
+ expect(modelTurn.parts[0].text).not.toContain('This valid part should be discarded');
1322
+ });
1323
+ describe('stripThoughtsFromHistory', () => {
1324
+ it('should strip thought signatures', () => {
1325
+ chat.setHistory([
1326
+ {
1327
+ role: 'user',
1328
+ parts: [{ text: 'hello' }],
1329
+ },
1330
+ {
1331
+ role: 'model',
1332
+ parts: [
1333
+ { text: 'thinking...', thoughtSignature: 'thought-123' },
1334
+ {
1335
+ functionCall: { name: 'test', args: {} },
1336
+ thoughtSignature: 'thought-456',
1337
+ },
1338
+ ],
1339
+ },
1340
+ ]);
1341
+ chat.stripThoughtsFromHistory();
1342
+ expect(chat.getHistory()).toEqual([
1343
+ {
1344
+ role: 'user',
1345
+ parts: [{ text: 'hello' }],
1346
+ },
1347
+ {
1348
+ role: 'model',
1349
+ parts: [
1350
+ { text: 'thinking...' },
1351
+ { functionCall: { name: 'test', args: {} } },
1352
+ ],
1353
+ },
1354
+ ]);
421
1355
  });
422
1356
  });
423
1357
  });