ai 6.0.32 → 6.0.34

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 (353) hide show
  1. package/CHANGELOG.md +16 -0
  2. package/dist/index.js +12 -2
  3. package/dist/index.js.map +1 -1
  4. package/dist/index.mjs +12 -2
  5. package/dist/index.mjs.map +1 -1
  6. package/dist/internal/index.js +1 -1
  7. package/dist/internal/index.mjs +1 -1
  8. package/docs/02-foundations/03-prompts.mdx +2 -2
  9. package/docs/03-ai-sdk-core/15-tools-and-tool-calling.mdx +1 -1
  10. package/docs/07-reference/01-ai-sdk-core/28-output.mdx +1 -1
  11. package/package.json +6 -4
  12. package/src/agent/agent.ts +116 -0
  13. package/src/agent/create-agent-ui-stream-response.test.ts +258 -0
  14. package/src/agent/create-agent-ui-stream-response.ts +50 -0
  15. package/src/agent/create-agent-ui-stream.ts +73 -0
  16. package/src/agent/index.ts +33 -0
  17. package/src/agent/infer-agent-tools.ts +7 -0
  18. package/src/agent/infer-agent-ui-message.test-d.ts +54 -0
  19. package/src/agent/infer-agent-ui-message.ts +11 -0
  20. package/src/agent/pipe-agent-ui-stream-to-response.ts +52 -0
  21. package/src/agent/tool-loop-agent-on-finish-callback.ts +31 -0
  22. package/src/agent/tool-loop-agent-on-step-finish-callback.ts +11 -0
  23. package/src/agent/tool-loop-agent-settings.ts +182 -0
  24. package/src/agent/tool-loop-agent.test-d.ts +114 -0
  25. package/src/agent/tool-loop-agent.test.ts +442 -0
  26. package/src/agent/tool-loop-agent.ts +114 -0
  27. package/src/embed/__snapshots__/embed-many.test.ts.snap +191 -0
  28. package/src/embed/__snapshots__/embed.test.ts.snap +81 -0
  29. package/src/embed/embed-many-result.ts +53 -0
  30. package/src/embed/embed-many.test.ts +653 -0
  31. package/src/embed/embed-many.ts +378 -0
  32. package/src/embed/embed-result.ts +50 -0
  33. package/src/embed/embed.test.ts +298 -0
  34. package/src/embed/embed.ts +211 -0
  35. package/src/embed/index.ts +4 -0
  36. package/src/error/index.ts +34 -0
  37. package/src/error/invalid-argument-error.ts +34 -0
  38. package/src/error/invalid-stream-part-error.ts +28 -0
  39. package/src/error/invalid-tool-approval-error.ts +26 -0
  40. package/src/error/invalid-tool-input-error.ts +33 -0
  41. package/src/error/no-image-generated-error.ts +39 -0
  42. package/src/error/no-object-generated-error.ts +70 -0
  43. package/src/error/no-output-generated-error.ts +26 -0
  44. package/src/error/no-speech-generated-error.ts +18 -0
  45. package/src/error/no-such-tool-error.ts +35 -0
  46. package/src/error/no-transcript-generated-error.ts +20 -0
  47. package/src/error/tool-call-not-found-for-approval-error.ts +32 -0
  48. package/src/error/tool-call-repair-error.ts +30 -0
  49. package/src/error/unsupported-model-version-error.ts +23 -0
  50. package/src/error/verify-no-object-generated-error.ts +27 -0
  51. package/src/generate-image/generate-image-result.ts +42 -0
  52. package/src/generate-image/generate-image.test.ts +1420 -0
  53. package/src/generate-image/generate-image.ts +360 -0
  54. package/src/generate-image/index.ts +18 -0
  55. package/src/generate-object/__snapshots__/generate-object.test.ts.snap +133 -0
  56. package/src/generate-object/__snapshots__/stream-object.test.ts.snap +297 -0
  57. package/src/generate-object/generate-object-result.ts +67 -0
  58. package/src/generate-object/generate-object.test-d.ts +49 -0
  59. package/src/generate-object/generate-object.test.ts +1191 -0
  60. package/src/generate-object/generate-object.ts +518 -0
  61. package/src/generate-object/index.ts +9 -0
  62. package/src/generate-object/inject-json-instruction.test.ts +181 -0
  63. package/src/generate-object/inject-json-instruction.ts +30 -0
  64. package/src/generate-object/output-strategy.ts +415 -0
  65. package/src/generate-object/parse-and-validate-object-result.ts +111 -0
  66. package/src/generate-object/repair-text.ts +12 -0
  67. package/src/generate-object/stream-object-result.ts +120 -0
  68. package/src/generate-object/stream-object.test-d.ts +74 -0
  69. package/src/generate-object/stream-object.test.ts +1950 -0
  70. package/src/generate-object/stream-object.ts +986 -0
  71. package/src/generate-object/validate-object-generation-input.ts +144 -0
  72. package/src/generate-speech/generate-speech-result.ts +30 -0
  73. package/src/generate-speech/generate-speech.test.ts +300 -0
  74. package/src/generate-speech/generate-speech.ts +190 -0
  75. package/src/generate-speech/generated-audio-file.ts +65 -0
  76. package/src/generate-speech/index.ts +3 -0
  77. package/src/generate-text/__snapshots__/generate-text.test.ts.snap +1872 -0
  78. package/src/generate-text/__snapshots__/stream-text.test.ts.snap +1255 -0
  79. package/src/generate-text/collect-tool-approvals.test.ts +553 -0
  80. package/src/generate-text/collect-tool-approvals.ts +116 -0
  81. package/src/generate-text/content-part.ts +25 -0
  82. package/src/generate-text/execute-tool-call.ts +129 -0
  83. package/src/generate-text/extract-reasoning-content.ts +17 -0
  84. package/src/generate-text/extract-text-content.ts +15 -0
  85. package/src/generate-text/generate-text-result.ts +168 -0
  86. package/src/generate-text/generate-text.test-d.ts +68 -0
  87. package/src/generate-text/generate-text.test.ts +7011 -0
  88. package/src/generate-text/generate-text.ts +1223 -0
  89. package/src/generate-text/generated-file.ts +70 -0
  90. package/src/generate-text/index.ts +57 -0
  91. package/src/generate-text/is-approval-needed.ts +29 -0
  92. package/src/generate-text/output-utils.ts +23 -0
  93. package/src/generate-text/output.test.ts +698 -0
  94. package/src/generate-text/output.ts +590 -0
  95. package/src/generate-text/parse-tool-call.test.ts +570 -0
  96. package/src/generate-text/parse-tool-call.ts +188 -0
  97. package/src/generate-text/prepare-step.ts +103 -0
  98. package/src/generate-text/prune-messages.test.ts +720 -0
  99. package/src/generate-text/prune-messages.ts +167 -0
  100. package/src/generate-text/reasoning-output.ts +20 -0
  101. package/src/generate-text/reasoning.ts +8 -0
  102. package/src/generate-text/response-message.ts +10 -0
  103. package/src/generate-text/run-tools-transformation.test.ts +1143 -0
  104. package/src/generate-text/run-tools-transformation.ts +420 -0
  105. package/src/generate-text/smooth-stream.test.ts +2101 -0
  106. package/src/generate-text/smooth-stream.ts +162 -0
  107. package/src/generate-text/step-result.ts +238 -0
  108. package/src/generate-text/stop-condition.ts +29 -0
  109. package/src/generate-text/stream-text-result.ts +463 -0
  110. package/src/generate-text/stream-text.test-d.ts +200 -0
  111. package/src/generate-text/stream-text.test.ts +19979 -0
  112. package/src/generate-text/stream-text.ts +2505 -0
  113. package/src/generate-text/to-response-messages.test.ts +922 -0
  114. package/src/generate-text/to-response-messages.ts +163 -0
  115. package/src/generate-text/tool-approval-request-output.ts +21 -0
  116. package/src/generate-text/tool-call-repair-function.ts +27 -0
  117. package/src/generate-text/tool-call.ts +47 -0
  118. package/src/generate-text/tool-error.ts +34 -0
  119. package/src/generate-text/tool-output-denied.ts +21 -0
  120. package/src/generate-text/tool-output.ts +7 -0
  121. package/src/generate-text/tool-result.ts +36 -0
  122. package/src/generate-text/tool-set.ts +14 -0
  123. package/src/global.ts +24 -0
  124. package/src/index.ts +50 -0
  125. package/src/logger/index.ts +6 -0
  126. package/src/logger/log-warnings.test.ts +351 -0
  127. package/src/logger/log-warnings.ts +119 -0
  128. package/src/middleware/__snapshots__/simulate-streaming-middleware.test.ts.snap +64 -0
  129. package/src/middleware/add-tool-input-examples-middleware.test.ts +476 -0
  130. package/src/middleware/add-tool-input-examples-middleware.ts +90 -0
  131. package/src/middleware/default-embedding-settings-middleware.test.ts +126 -0
  132. package/src/middleware/default-embedding-settings-middleware.ts +22 -0
  133. package/src/middleware/default-settings-middleware.test.ts +388 -0
  134. package/src/middleware/default-settings-middleware.ts +33 -0
  135. package/src/middleware/extract-json-middleware.test.ts +827 -0
  136. package/src/middleware/extract-json-middleware.ts +197 -0
  137. package/src/middleware/extract-reasoning-middleware.test.ts +1028 -0
  138. package/src/middleware/extract-reasoning-middleware.ts +238 -0
  139. package/src/middleware/index.ts +10 -0
  140. package/src/middleware/simulate-streaming-middleware.test.ts +911 -0
  141. package/src/middleware/simulate-streaming-middleware.ts +79 -0
  142. package/src/middleware/wrap-embedding-model.test.ts +358 -0
  143. package/src/middleware/wrap-embedding-model.ts +86 -0
  144. package/src/middleware/wrap-image-model.test.ts +423 -0
  145. package/src/middleware/wrap-image-model.ts +85 -0
  146. package/src/middleware/wrap-language-model.test.ts +518 -0
  147. package/src/middleware/wrap-language-model.ts +104 -0
  148. package/src/middleware/wrap-provider.test.ts +120 -0
  149. package/src/middleware/wrap-provider.ts +51 -0
  150. package/src/model/as-embedding-model-v3.test.ts +319 -0
  151. package/src/model/as-embedding-model-v3.ts +24 -0
  152. package/src/model/as-image-model-v3.test.ts +409 -0
  153. package/src/model/as-image-model-v3.ts +24 -0
  154. package/src/model/as-language-model-v3.test.ts +508 -0
  155. package/src/model/as-language-model-v3.ts +103 -0
  156. package/src/model/as-provider-v3.ts +36 -0
  157. package/src/model/as-speech-model-v3.test.ts +356 -0
  158. package/src/model/as-speech-model-v3.ts +24 -0
  159. package/src/model/as-transcription-model-v3.test.ts +529 -0
  160. package/src/model/as-transcription-model-v3.ts +24 -0
  161. package/src/model/resolve-model.test.ts +244 -0
  162. package/src/model/resolve-model.ts +126 -0
  163. package/src/prompt/call-settings.ts +148 -0
  164. package/src/prompt/content-part.ts +209 -0
  165. package/src/prompt/convert-to-language-model-prompt.test.ts +2018 -0
  166. package/src/prompt/convert-to-language-model-prompt.ts +442 -0
  167. package/src/prompt/create-tool-model-output.test.ts +508 -0
  168. package/src/prompt/create-tool-model-output.ts +34 -0
  169. package/src/prompt/data-content.test.ts +15 -0
  170. package/src/prompt/data-content.ts +134 -0
  171. package/src/prompt/index.ts +27 -0
  172. package/src/prompt/invalid-data-content-error.ts +29 -0
  173. package/src/prompt/invalid-message-role-error.ts +27 -0
  174. package/src/prompt/message-conversion-error.ts +28 -0
  175. package/src/prompt/message.ts +68 -0
  176. package/src/prompt/prepare-call-settings.test.ts +159 -0
  177. package/src/prompt/prepare-call-settings.ts +108 -0
  178. package/src/prompt/prepare-tools-and-tool-choice.test.ts +461 -0
  179. package/src/prompt/prepare-tools-and-tool-choice.ts +86 -0
  180. package/src/prompt/prompt.ts +43 -0
  181. package/src/prompt/split-data-url.ts +17 -0
  182. package/src/prompt/standardize-prompt.test.ts +82 -0
  183. package/src/prompt/standardize-prompt.ts +99 -0
  184. package/src/prompt/wrap-gateway-error.ts +29 -0
  185. package/src/registry/custom-provider.test.ts +211 -0
  186. package/src/registry/custom-provider.ts +155 -0
  187. package/src/registry/index.ts +7 -0
  188. package/src/registry/no-such-provider-error.ts +41 -0
  189. package/src/registry/provider-registry.test.ts +691 -0
  190. package/src/registry/provider-registry.ts +328 -0
  191. package/src/rerank/index.ts +2 -0
  192. package/src/rerank/rerank-result.ts +70 -0
  193. package/src/rerank/rerank.test.ts +516 -0
  194. package/src/rerank/rerank.ts +237 -0
  195. package/src/telemetry/assemble-operation-name.ts +21 -0
  196. package/src/telemetry/get-base-telemetry-attributes.ts +53 -0
  197. package/src/telemetry/get-tracer.ts +20 -0
  198. package/src/telemetry/noop-tracer.ts +69 -0
  199. package/src/telemetry/record-span.ts +63 -0
  200. package/src/telemetry/select-telemetry-attributes.ts +78 -0
  201. package/src/telemetry/select-temetry-attributes.test.ts +114 -0
  202. package/src/telemetry/stringify-for-telemetry.test.ts +114 -0
  203. package/src/telemetry/stringify-for-telemetry.ts +33 -0
  204. package/src/telemetry/telemetry-settings.ts +44 -0
  205. package/src/test/mock-embedding-model-v2.ts +35 -0
  206. package/src/test/mock-embedding-model-v3.ts +48 -0
  207. package/src/test/mock-image-model-v2.ts +28 -0
  208. package/src/test/mock-image-model-v3.ts +28 -0
  209. package/src/test/mock-language-model-v2.ts +72 -0
  210. package/src/test/mock-language-model-v3.ts +77 -0
  211. package/src/test/mock-provider-v2.ts +68 -0
  212. package/src/test/mock-provider-v3.ts +80 -0
  213. package/src/test/mock-reranking-model-v3.ts +25 -0
  214. package/src/test/mock-server-response.ts +69 -0
  215. package/src/test/mock-speech-model-v2.ts +24 -0
  216. package/src/test/mock-speech-model-v3.ts +24 -0
  217. package/src/test/mock-tracer.ts +156 -0
  218. package/src/test/mock-transcription-model-v2.ts +24 -0
  219. package/src/test/mock-transcription-model-v3.ts +24 -0
  220. package/src/test/mock-values.ts +4 -0
  221. package/src/test/not-implemented.ts +3 -0
  222. package/src/text-stream/create-text-stream-response.test.ts +38 -0
  223. package/src/text-stream/create-text-stream-response.ts +18 -0
  224. package/src/text-stream/index.ts +2 -0
  225. package/src/text-stream/pipe-text-stream-to-response.test.ts +38 -0
  226. package/src/text-stream/pipe-text-stream-to-response.ts +26 -0
  227. package/src/transcribe/index.ts +2 -0
  228. package/src/transcribe/transcribe-result.ts +60 -0
  229. package/src/transcribe/transcribe.test.ts +313 -0
  230. package/src/transcribe/transcribe.ts +173 -0
  231. package/src/types/embedding-model-middleware.ts +3 -0
  232. package/src/types/embedding-model.ts +18 -0
  233. package/src/types/image-model-middleware.ts +3 -0
  234. package/src/types/image-model-response-metadata.ts +16 -0
  235. package/src/types/image-model.ts +19 -0
  236. package/src/types/index.ts +29 -0
  237. package/src/types/json-value.ts +15 -0
  238. package/src/types/language-model-middleware.ts +3 -0
  239. package/src/types/language-model-request-metadata.ts +6 -0
  240. package/src/types/language-model-response-metadata.ts +21 -0
  241. package/src/types/language-model.ts +104 -0
  242. package/src/types/provider-metadata.ts +16 -0
  243. package/src/types/provider.ts +55 -0
  244. package/src/types/reranking-model.ts +6 -0
  245. package/src/types/speech-model-response-metadata.ts +21 -0
  246. package/src/types/speech-model.ts +6 -0
  247. package/src/types/transcription-model-response-metadata.ts +16 -0
  248. package/src/types/transcription-model.ts +9 -0
  249. package/src/types/usage.ts +200 -0
  250. package/src/types/warning.ts +7 -0
  251. package/src/ui/__snapshots__/append-response-messages.test.ts.snap +416 -0
  252. package/src/ui/__snapshots__/convert-to-model-messages.test.ts.snap +419 -0
  253. package/src/ui/__snapshots__/process-chat-text-response.test.ts.snap +142 -0
  254. package/src/ui/call-completion-api.ts +157 -0
  255. package/src/ui/chat-transport.ts +83 -0
  256. package/src/ui/chat.test-d.ts +233 -0
  257. package/src/ui/chat.test.ts +2695 -0
  258. package/src/ui/chat.ts +716 -0
  259. package/src/ui/convert-file-list-to-file-ui-parts.ts +36 -0
  260. package/src/ui/convert-to-model-messages.test.ts +2775 -0
  261. package/src/ui/convert-to-model-messages.ts +373 -0
  262. package/src/ui/default-chat-transport.ts +36 -0
  263. package/src/ui/direct-chat-transport.test.ts +446 -0
  264. package/src/ui/direct-chat-transport.ts +118 -0
  265. package/src/ui/http-chat-transport.test.ts +185 -0
  266. package/src/ui/http-chat-transport.ts +292 -0
  267. package/src/ui/index.ts +71 -0
  268. package/src/ui/last-assistant-message-is-complete-with-approval-responses.ts +44 -0
  269. package/src/ui/last-assistant-message-is-complete-with-tool-calls.test.ts +371 -0
  270. package/src/ui/last-assistant-message-is-complete-with-tool-calls.ts +39 -0
  271. package/src/ui/process-text-stream.test.ts +38 -0
  272. package/src/ui/process-text-stream.ts +16 -0
  273. package/src/ui/process-ui-message-stream.test.ts +8052 -0
  274. package/src/ui/process-ui-message-stream.ts +713 -0
  275. package/src/ui/text-stream-chat-transport.ts +23 -0
  276. package/src/ui/transform-text-to-ui-message-stream.test.ts +124 -0
  277. package/src/ui/transform-text-to-ui-message-stream.ts +27 -0
  278. package/src/ui/ui-messages.test.ts +48 -0
  279. package/src/ui/ui-messages.ts +534 -0
  280. package/src/ui/use-completion.ts +84 -0
  281. package/src/ui/validate-ui-messages.test.ts +1428 -0
  282. package/src/ui/validate-ui-messages.ts +476 -0
  283. package/src/ui-message-stream/create-ui-message-stream-response.test.ts +266 -0
  284. package/src/ui-message-stream/create-ui-message-stream-response.ts +32 -0
  285. package/src/ui-message-stream/create-ui-message-stream.test.ts +639 -0
  286. package/src/ui-message-stream/create-ui-message-stream.ts +124 -0
  287. package/src/ui-message-stream/get-response-ui-message-id.test.ts +55 -0
  288. package/src/ui-message-stream/get-response-ui-message-id.ts +24 -0
  289. package/src/ui-message-stream/handle-ui-message-stream-finish.test.ts +429 -0
  290. package/src/ui-message-stream/handle-ui-message-stream-finish.ts +135 -0
  291. package/src/ui-message-stream/index.ts +13 -0
  292. package/src/ui-message-stream/json-to-sse-transform-stream.ts +12 -0
  293. package/src/ui-message-stream/pipe-ui-message-stream-to-response.test.ts +90 -0
  294. package/src/ui-message-stream/pipe-ui-message-stream-to-response.ts +40 -0
  295. package/src/ui-message-stream/read-ui-message-stream.test.ts +122 -0
  296. package/src/ui-message-stream/read-ui-message-stream.ts +87 -0
  297. package/src/ui-message-stream/ui-message-chunks.test-d.ts +18 -0
  298. package/src/ui-message-stream/ui-message-chunks.ts +344 -0
  299. package/src/ui-message-stream/ui-message-stream-headers.ts +7 -0
  300. package/src/ui-message-stream/ui-message-stream-on-finish-callback.ts +32 -0
  301. package/src/ui-message-stream/ui-message-stream-response-init.ts +5 -0
  302. package/src/ui-message-stream/ui-message-stream-writer.ts +24 -0
  303. package/src/util/as-array.ts +3 -0
  304. package/src/util/async-iterable-stream.test.ts +241 -0
  305. package/src/util/async-iterable-stream.ts +94 -0
  306. package/src/util/consume-stream.ts +29 -0
  307. package/src/util/cosine-similarity.test.ts +57 -0
  308. package/src/util/cosine-similarity.ts +47 -0
  309. package/src/util/create-resolvable-promise.ts +30 -0
  310. package/src/util/create-stitchable-stream.test.ts +239 -0
  311. package/src/util/create-stitchable-stream.ts +112 -0
  312. package/src/util/data-url.ts +17 -0
  313. package/src/util/deep-partial.ts +84 -0
  314. package/src/util/detect-media-type.test.ts +670 -0
  315. package/src/util/detect-media-type.ts +184 -0
  316. package/src/util/download/download-function.ts +45 -0
  317. package/src/util/download/download.test.ts +69 -0
  318. package/src/util/download/download.ts +46 -0
  319. package/src/util/error-handler.ts +1 -0
  320. package/src/util/fix-json.test.ts +279 -0
  321. package/src/util/fix-json.ts +401 -0
  322. package/src/util/get-potential-start-index.test.ts +34 -0
  323. package/src/util/get-potential-start-index.ts +30 -0
  324. package/src/util/index.ts +11 -0
  325. package/src/util/is-deep-equal-data.test.ts +119 -0
  326. package/src/util/is-deep-equal-data.ts +48 -0
  327. package/src/util/is-non-empty-object.ts +5 -0
  328. package/src/util/job.ts +1 -0
  329. package/src/util/log-v2-compatibility-warning.ts +21 -0
  330. package/src/util/merge-abort-signals.test.ts +155 -0
  331. package/src/util/merge-abort-signals.ts +43 -0
  332. package/src/util/merge-objects.test.ts +118 -0
  333. package/src/util/merge-objects.ts +79 -0
  334. package/src/util/now.ts +4 -0
  335. package/src/util/parse-partial-json.test.ts +80 -0
  336. package/src/util/parse-partial-json.ts +30 -0
  337. package/src/util/prepare-headers.test.ts +51 -0
  338. package/src/util/prepare-headers.ts +14 -0
  339. package/src/util/prepare-retries.test.ts +10 -0
  340. package/src/util/prepare-retries.ts +47 -0
  341. package/src/util/retry-error.ts +41 -0
  342. package/src/util/retry-with-exponential-backoff.test.ts +446 -0
  343. package/src/util/retry-with-exponential-backoff.ts +154 -0
  344. package/src/util/serial-job-executor.test.ts +162 -0
  345. package/src/util/serial-job-executor.ts +36 -0
  346. package/src/util/simulate-readable-stream.test.ts +98 -0
  347. package/src/util/simulate-readable-stream.ts +39 -0
  348. package/src/util/split-array.test.ts +60 -0
  349. package/src/util/split-array.ts +20 -0
  350. package/src/util/value-of.ts +65 -0
  351. package/src/util/write-to-server-response.test.ts +266 -0
  352. package/src/util/write-to-server-response.ts +49 -0
  353. package/src/version.ts +5 -0
@@ -0,0 +1,1143 @@
1
+ import {
2
+ LanguageModelV3StreamPart,
3
+ LanguageModelV3Usage,
4
+ } from '@ai-sdk/provider';
5
+ import { delay, tool } from '@ai-sdk/provider-utils';
6
+ import {
7
+ convertArrayToReadableStream,
8
+ convertReadableStreamToArray,
9
+ mockId,
10
+ } from '@ai-sdk/provider-utils/test';
11
+ import { describe, expect, it } from 'vitest';
12
+ import { z } from 'zod/v4';
13
+ import { NoSuchToolError } from '../error/no-such-tool-error';
14
+ import { MockTracer } from '../test/mock-tracer';
15
+ import { runToolsTransformation } from './run-tools-transformation';
16
+
17
+ const testUsage: LanguageModelV3Usage = {
18
+ inputTokens: {
19
+ total: 3,
20
+ noCache: 3,
21
+ cacheRead: undefined,
22
+ cacheWrite: undefined,
23
+ },
24
+ outputTokens: {
25
+ total: 10,
26
+ text: 10,
27
+ reasoning: undefined,
28
+ },
29
+ };
30
+
31
+ describe('runToolsTransformation', () => {
32
+ it('should forward text parts', async () => {
33
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
34
+ convertArrayToReadableStream([
35
+ { type: 'text-start', id: '1' },
36
+ { type: 'text-delta', id: '1', delta: 'text' },
37
+ { type: 'text-end', id: '1' },
38
+ {
39
+ type: 'finish',
40
+ finishReason: { unified: 'stop', raw: 'stop' },
41
+ usage: testUsage,
42
+ },
43
+ ]);
44
+
45
+ const transformedStream = runToolsTransformation({
46
+ generateId: mockId({ prefix: 'id' }),
47
+ tools: undefined,
48
+ generatorStream: inputStream,
49
+ tracer: new MockTracer(),
50
+ telemetry: undefined,
51
+ messages: [],
52
+ system: undefined,
53
+ abortSignal: undefined,
54
+ repairToolCall: undefined,
55
+ experimental_context: undefined,
56
+ });
57
+
58
+ const result = await convertReadableStreamToArray(transformedStream);
59
+
60
+ expect(result).toMatchInlineSnapshot(`
61
+ [
62
+ {
63
+ "id": "1",
64
+ "type": "text-start",
65
+ },
66
+ {
67
+ "delta": "text",
68
+ "id": "1",
69
+ "type": "text-delta",
70
+ },
71
+ {
72
+ "id": "1",
73
+ "type": "text-end",
74
+ },
75
+ {
76
+ "finishReason": "stop",
77
+ "providerMetadata": undefined,
78
+ "rawFinishReason": "stop",
79
+ "type": "finish",
80
+ "usage": {
81
+ "cachedInputTokens": undefined,
82
+ "inputTokenDetails": {
83
+ "cacheReadTokens": undefined,
84
+ "cacheWriteTokens": undefined,
85
+ "noCacheTokens": 3,
86
+ },
87
+ "inputTokens": 3,
88
+ "outputTokenDetails": {
89
+ "reasoningTokens": undefined,
90
+ "textTokens": 10,
91
+ },
92
+ "outputTokens": 10,
93
+ "raw": undefined,
94
+ "reasoningTokens": undefined,
95
+ "totalTokens": 13,
96
+ },
97
+ },
98
+ ]
99
+ `);
100
+ });
101
+
102
+ it('should handle async tool execution', async () => {
103
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
104
+ convertArrayToReadableStream([
105
+ {
106
+ type: 'tool-call',
107
+ toolCallId: 'call-1',
108
+ toolName: 'syncTool',
109
+ input: `{ "value": "test" }`,
110
+ },
111
+ {
112
+ type: 'finish',
113
+ finishReason: { unified: 'stop', raw: 'stop' },
114
+ usage: testUsage,
115
+ },
116
+ ]);
117
+
118
+ const transformedStream = runToolsTransformation({
119
+ generateId: mockId({ prefix: 'id' }),
120
+ tools: {
121
+ syncTool: {
122
+ title: 'Sync Tool',
123
+ inputSchema: z.object({ value: z.string() }),
124
+ execute: async ({ value }) => `${value}-sync-result`,
125
+ },
126
+ },
127
+ generatorStream: inputStream,
128
+ tracer: new MockTracer(),
129
+ telemetry: undefined,
130
+ messages: [],
131
+ system: undefined,
132
+ abortSignal: undefined,
133
+ repairToolCall: undefined,
134
+ experimental_context: undefined,
135
+ });
136
+
137
+ expect(await convertReadableStreamToArray(transformedStream))
138
+ .toMatchInlineSnapshot(`
139
+ [
140
+ {
141
+ "input": {
142
+ "value": "test",
143
+ },
144
+ "providerExecuted": undefined,
145
+ "providerMetadata": undefined,
146
+ "title": "Sync Tool",
147
+ "toolCallId": "call-1",
148
+ "toolName": "syncTool",
149
+ "type": "tool-call",
150
+ },
151
+ {
152
+ "dynamic": false,
153
+ "input": {
154
+ "value": "test",
155
+ },
156
+ "output": "test-sync-result",
157
+ "toolCallId": "call-1",
158
+ "toolName": "syncTool",
159
+ "type": "tool-result",
160
+ },
161
+ {
162
+ "finishReason": "stop",
163
+ "providerMetadata": undefined,
164
+ "rawFinishReason": "stop",
165
+ "type": "finish",
166
+ "usage": {
167
+ "cachedInputTokens": undefined,
168
+ "inputTokenDetails": {
169
+ "cacheReadTokens": undefined,
170
+ "cacheWriteTokens": undefined,
171
+ "noCacheTokens": 3,
172
+ },
173
+ "inputTokens": 3,
174
+ "outputTokenDetails": {
175
+ "reasoningTokens": undefined,
176
+ "textTokens": 10,
177
+ },
178
+ "outputTokens": 10,
179
+ "raw": undefined,
180
+ "reasoningTokens": undefined,
181
+ "totalTokens": 13,
182
+ },
183
+ },
184
+ ]
185
+ `);
186
+ });
187
+
188
+ it('should handle sync tool execution', async () => {
189
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
190
+ convertArrayToReadableStream([
191
+ {
192
+ type: 'tool-call',
193
+ toolCallId: 'call-1',
194
+ toolName: 'syncTool',
195
+ input: `{ "value": "test" }`,
196
+ },
197
+ {
198
+ type: 'finish',
199
+ finishReason: { unified: 'stop', raw: 'stop' },
200
+ usage: testUsage,
201
+ },
202
+ ]);
203
+
204
+ const transformedStream = runToolsTransformation({
205
+ generateId: mockId({ prefix: 'id' }),
206
+ tools: {
207
+ syncTool: {
208
+ title: 'Sync Tool',
209
+ inputSchema: z.object({ value: z.string() }),
210
+ execute: ({ value }) => `${value}-sync-result`,
211
+ },
212
+ },
213
+ generatorStream: inputStream,
214
+ tracer: new MockTracer(),
215
+ telemetry: undefined,
216
+ messages: [],
217
+ system: undefined,
218
+ abortSignal: undefined,
219
+ repairToolCall: undefined,
220
+ experimental_context: undefined,
221
+ });
222
+
223
+ expect(await convertReadableStreamToArray(transformedStream))
224
+ .toMatchInlineSnapshot(`
225
+ [
226
+ {
227
+ "input": {
228
+ "value": "test",
229
+ },
230
+ "providerExecuted": undefined,
231
+ "providerMetadata": undefined,
232
+ "title": "Sync Tool",
233
+ "toolCallId": "call-1",
234
+ "toolName": "syncTool",
235
+ "type": "tool-call",
236
+ },
237
+ {
238
+ "dynamic": false,
239
+ "input": {
240
+ "value": "test",
241
+ },
242
+ "output": "test-sync-result",
243
+ "toolCallId": "call-1",
244
+ "toolName": "syncTool",
245
+ "type": "tool-result",
246
+ },
247
+ {
248
+ "finishReason": "stop",
249
+ "providerMetadata": undefined,
250
+ "rawFinishReason": "stop",
251
+ "type": "finish",
252
+ "usage": {
253
+ "cachedInputTokens": undefined,
254
+ "inputTokenDetails": {
255
+ "cacheReadTokens": undefined,
256
+ "cacheWriteTokens": undefined,
257
+ "noCacheTokens": 3,
258
+ },
259
+ "inputTokens": 3,
260
+ "outputTokenDetails": {
261
+ "reasoningTokens": undefined,
262
+ "textTokens": 10,
263
+ },
264
+ "outputTokens": 10,
265
+ "raw": undefined,
266
+ "reasoningTokens": undefined,
267
+ "totalTokens": 13,
268
+ },
269
+ },
270
+ ]
271
+ `);
272
+ });
273
+
274
+ it('should hold off on sending finish until the delayed tool result is received', async () => {
275
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
276
+ convertArrayToReadableStream([
277
+ {
278
+ type: 'tool-call',
279
+ toolCallId: 'call-1',
280
+ toolName: 'delayedTool',
281
+ input: `{ "value": "test" }`,
282
+ },
283
+ {
284
+ type: 'finish',
285
+ finishReason: { unified: 'stop', raw: 'stop' },
286
+ usage: testUsage,
287
+ },
288
+ ]);
289
+
290
+ const transformedStream = runToolsTransformation({
291
+ generateId: mockId({ prefix: 'id' }),
292
+ tools: {
293
+ delayedTool: {
294
+ title: 'Delayed Tool',
295
+ inputSchema: z.object({ value: z.string() }),
296
+ execute: async ({ value }) => {
297
+ await delay(0); // Simulate delayed execution
298
+ return `${value}-delayed-result`;
299
+ },
300
+ },
301
+ },
302
+ generatorStream: inputStream,
303
+ tracer: new MockTracer(),
304
+ telemetry: undefined,
305
+ messages: [],
306
+ system: undefined,
307
+ abortSignal: undefined,
308
+ repairToolCall: undefined,
309
+ experimental_context: undefined,
310
+ });
311
+
312
+ const result = await convertReadableStreamToArray(transformedStream);
313
+
314
+ expect(result).toMatchInlineSnapshot(`
315
+ [
316
+ {
317
+ "input": {
318
+ "value": "test",
319
+ },
320
+ "providerExecuted": undefined,
321
+ "providerMetadata": undefined,
322
+ "title": "Delayed Tool",
323
+ "toolCallId": "call-1",
324
+ "toolName": "delayedTool",
325
+ "type": "tool-call",
326
+ },
327
+ {
328
+ "dynamic": false,
329
+ "input": {
330
+ "value": "test",
331
+ },
332
+ "output": "test-delayed-result",
333
+ "toolCallId": "call-1",
334
+ "toolName": "delayedTool",
335
+ "type": "tool-result",
336
+ },
337
+ {
338
+ "finishReason": "stop",
339
+ "providerMetadata": undefined,
340
+ "rawFinishReason": "stop",
341
+ "type": "finish",
342
+ "usage": {
343
+ "cachedInputTokens": undefined,
344
+ "inputTokenDetails": {
345
+ "cacheReadTokens": undefined,
346
+ "cacheWriteTokens": undefined,
347
+ "noCacheTokens": 3,
348
+ },
349
+ "inputTokens": 3,
350
+ "outputTokenDetails": {
351
+ "reasoningTokens": undefined,
352
+ "textTokens": 10,
353
+ },
354
+ "outputTokens": 10,
355
+ "raw": undefined,
356
+ "reasoningTokens": undefined,
357
+ "totalTokens": 13,
358
+ },
359
+ },
360
+ ]
361
+ `);
362
+ });
363
+
364
+ it('should try to repair tool call when the tool name is not found', async () => {
365
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
366
+ convertArrayToReadableStream([
367
+ {
368
+ type: 'tool-call',
369
+ toolCallId: 'call-1',
370
+ toolName: 'unknownTool',
371
+ input: `{ "value": "test" }`,
372
+ },
373
+ {
374
+ type: 'finish',
375
+ finishReason: { unified: 'stop', raw: 'stop' },
376
+ usage: testUsage,
377
+ },
378
+ ]);
379
+
380
+ const transformedStream = runToolsTransformation({
381
+ generateId: mockId({ prefix: 'id' }),
382
+ generatorStream: inputStream,
383
+ tracer: new MockTracer(),
384
+ telemetry: undefined,
385
+ messages: [],
386
+ system: undefined,
387
+ abortSignal: undefined,
388
+ tools: {
389
+ correctTool: {
390
+ inputSchema: z.object({ value: z.string() }),
391
+ execute: async ({ value }) => `${value}-result`,
392
+ },
393
+ },
394
+ repairToolCall: async ({ toolCall, tools, inputSchema, error }) => {
395
+ expect(NoSuchToolError.isInstance(error)).toBe(true);
396
+ expect(toolCall).toStrictEqual({
397
+ type: 'tool-call',
398
+ toolCallId: 'call-1',
399
+ toolName: 'unknownTool',
400
+ input: `{ "value": "test" }`,
401
+ });
402
+
403
+ return { ...toolCall, toolName: 'correctTool' };
404
+ },
405
+ experimental_context: undefined,
406
+ });
407
+
408
+ expect(await convertReadableStreamToArray(transformedStream))
409
+ .toMatchInlineSnapshot(`
410
+ [
411
+ {
412
+ "input": {
413
+ "value": "test",
414
+ },
415
+ "providerExecuted": undefined,
416
+ "providerMetadata": undefined,
417
+ "title": undefined,
418
+ "toolCallId": "call-1",
419
+ "toolName": "correctTool",
420
+ "type": "tool-call",
421
+ },
422
+ {
423
+ "dynamic": false,
424
+ "input": {
425
+ "value": "test",
426
+ },
427
+ "output": "test-result",
428
+ "toolCallId": "call-1",
429
+ "toolName": "correctTool",
430
+ "type": "tool-result",
431
+ },
432
+ {
433
+ "finishReason": "stop",
434
+ "providerMetadata": undefined,
435
+ "rawFinishReason": "stop",
436
+ "type": "finish",
437
+ "usage": {
438
+ "cachedInputTokens": undefined,
439
+ "inputTokenDetails": {
440
+ "cacheReadTokens": undefined,
441
+ "cacheWriteTokens": undefined,
442
+ "noCacheTokens": 3,
443
+ },
444
+ "inputTokens": 3,
445
+ "outputTokenDetails": {
446
+ "reasoningTokens": undefined,
447
+ "textTokens": 10,
448
+ },
449
+ "outputTokens": 10,
450
+ "raw": undefined,
451
+ "reasoningTokens": undefined,
452
+ "totalTokens": 13,
453
+ },
454
+ },
455
+ ]
456
+ `);
457
+ });
458
+
459
+ it('should not call execute for provider-executed tool calls', async () => {
460
+ let toolExecuted = false;
461
+
462
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
463
+ convertArrayToReadableStream([
464
+ {
465
+ type: 'tool-call',
466
+ toolCallId: 'call-1',
467
+ toolName: 'providerTool',
468
+ input: `{ "value": "test" }`,
469
+ providerExecuted: true,
470
+ },
471
+ {
472
+ type: 'tool-result',
473
+ toolCallId: 'call-1',
474
+ toolName: 'providerTool',
475
+ providerExecuted: true,
476
+ result: { example: 'example' },
477
+ },
478
+ {
479
+ type: 'finish',
480
+ finishReason: { unified: 'stop', raw: 'stop' },
481
+ usage: testUsage,
482
+ },
483
+ ]);
484
+
485
+ const transformedStream = runToolsTransformation({
486
+ generateId: mockId({ prefix: 'id' }),
487
+ tools: {
488
+ providerTool: {
489
+ inputSchema: z.object({ value: z.string() }),
490
+ execute: async ({ value }) => {
491
+ toolExecuted = true;
492
+ return `${value}-should-not-execute`;
493
+ },
494
+ },
495
+ },
496
+ generatorStream: inputStream,
497
+ tracer: new MockTracer(),
498
+ telemetry: undefined,
499
+ messages: [],
500
+ system: undefined,
501
+ abortSignal: undefined,
502
+ repairToolCall: undefined,
503
+ experimental_context: undefined,
504
+ });
505
+
506
+ await convertReadableStreamToArray(transformedStream);
507
+
508
+ expect(toolExecuted).toBe(false);
509
+ });
510
+
511
+ describe('provider-emitted tool-approval-request (MCP flow)', () => {
512
+ it('should forward provider-emitted tool-approval-request with the correct tool call', async () => {
513
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
514
+ convertArrayToReadableStream([
515
+ {
516
+ type: 'tool-call',
517
+ toolCallId: 'mcp-call-1',
518
+ toolName: 'mcp_tool',
519
+ input: `{ "query": "test" }`,
520
+ providerExecuted: true,
521
+ },
522
+ {
523
+ type: 'tool-approval-request',
524
+ approvalId: 'mcp-approval-1',
525
+ toolCallId: 'mcp-call-1',
526
+ },
527
+ {
528
+ type: 'finish',
529
+ finishReason: { unified: 'tool-calls', raw: undefined },
530
+ usage: testUsage,
531
+ },
532
+ ]);
533
+
534
+ const transformedStream = runToolsTransformation({
535
+ generateId: mockId({ prefix: 'id' }),
536
+ tools: {
537
+ mcp_tool: {
538
+ type: 'provider',
539
+ id: 'mcp.mcp_tool',
540
+ inputSchema: z.object({ query: z.string() }),
541
+ args: {},
542
+ },
543
+ },
544
+ generatorStream: inputStream,
545
+ tracer: new MockTracer(),
546
+ telemetry: undefined,
547
+ messages: [],
548
+ system: undefined,
549
+ abortSignal: undefined,
550
+ repairToolCall: undefined,
551
+ experimental_context: undefined,
552
+ });
553
+
554
+ const result = await convertReadableStreamToArray(transformedStream);
555
+
556
+ expect(result).toMatchInlineSnapshot(`
557
+ [
558
+ {
559
+ "input": {
560
+ "query": "test",
561
+ },
562
+ "providerExecuted": true,
563
+ "providerMetadata": undefined,
564
+ "title": undefined,
565
+ "toolCallId": "mcp-call-1",
566
+ "toolName": "mcp_tool",
567
+ "type": "tool-call",
568
+ },
569
+ {
570
+ "approvalId": "mcp-approval-1",
571
+ "toolCall": {
572
+ "input": {
573
+ "query": "test",
574
+ },
575
+ "providerExecuted": true,
576
+ "providerMetadata": undefined,
577
+ "title": undefined,
578
+ "toolCallId": "mcp-call-1",
579
+ "toolName": "mcp_tool",
580
+ "type": "tool-call",
581
+ },
582
+ "type": "tool-approval-request",
583
+ },
584
+ {
585
+ "finishReason": "tool-calls",
586
+ "providerMetadata": undefined,
587
+ "rawFinishReason": undefined,
588
+ "type": "finish",
589
+ "usage": {
590
+ "cachedInputTokens": undefined,
591
+ "inputTokenDetails": {
592
+ "cacheReadTokens": undefined,
593
+ "cacheWriteTokens": undefined,
594
+ "noCacheTokens": 3,
595
+ },
596
+ "inputTokens": 3,
597
+ "outputTokenDetails": {
598
+ "reasoningTokens": undefined,
599
+ "textTokens": 10,
600
+ },
601
+ "outputTokens": 10,
602
+ "raw": undefined,
603
+ "reasoningTokens": undefined,
604
+ "totalTokens": 13,
605
+ },
606
+ },
607
+ ]
608
+ `);
609
+ });
610
+
611
+ it('should emit error when tool call is not found for provider approval request', async () => {
612
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
613
+ convertArrayToReadableStream([
614
+ // No tool-call part before the approval request
615
+ {
616
+ type: 'tool-approval-request',
617
+ approvalId: 'mcp-approval-1',
618
+ toolCallId: 'non-existent-call',
619
+ },
620
+ {
621
+ type: 'finish',
622
+ finishReason: { unified: 'stop', raw: undefined },
623
+ usage: testUsage,
624
+ },
625
+ ]);
626
+
627
+ const transformedStream = runToolsTransformation({
628
+ generateId: mockId({ prefix: 'id' }),
629
+ tools: undefined,
630
+ generatorStream: inputStream,
631
+ tracer: new MockTracer(),
632
+ telemetry: undefined,
633
+ messages: [],
634
+ system: undefined,
635
+ abortSignal: undefined,
636
+ repairToolCall: undefined,
637
+ experimental_context: undefined,
638
+ });
639
+
640
+ const result = await convertReadableStreamToArray(transformedStream);
641
+
642
+ expect(result).toMatchInlineSnapshot(`
643
+ [
644
+ {
645
+ "error": [AI_ToolCallNotFoundForApprovalError: Tool call "non-existent-call" not found for approval request "mcp-approval-1".],
646
+ "type": "error",
647
+ },
648
+ {
649
+ "finishReason": "stop",
650
+ "providerMetadata": undefined,
651
+ "rawFinishReason": undefined,
652
+ "type": "finish",
653
+ "usage": {
654
+ "cachedInputTokens": undefined,
655
+ "inputTokenDetails": {
656
+ "cacheReadTokens": undefined,
657
+ "cacheWriteTokens": undefined,
658
+ "noCacheTokens": 3,
659
+ },
660
+ "inputTokens": 3,
661
+ "outputTokenDetails": {
662
+ "reasoningTokens": undefined,
663
+ "textTokens": 10,
664
+ },
665
+ "outputTokens": 10,
666
+ "raw": undefined,
667
+ "reasoningTokens": undefined,
668
+ "totalTokens": 13,
669
+ },
670
+ },
671
+ ]
672
+ `);
673
+ });
674
+
675
+ it('should handle multiple provider-executed tool calls with approval requests', async () => {
676
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
677
+ convertArrayToReadableStream([
678
+ {
679
+ type: 'tool-call',
680
+ toolCallId: 'mcp-call-1',
681
+ toolName: 'mcp_search',
682
+ input: `{ "query": "first" }`,
683
+ providerExecuted: true,
684
+ },
685
+ {
686
+ type: 'tool-call',
687
+ toolCallId: 'mcp-call-2',
688
+ toolName: 'mcp_execute',
689
+ input: `{ "command": "ls" }`,
690
+ providerExecuted: true,
691
+ },
692
+ {
693
+ type: 'tool-approval-request',
694
+ approvalId: 'approval-1',
695
+ toolCallId: 'mcp-call-1',
696
+ },
697
+ {
698
+ type: 'tool-approval-request',
699
+ approvalId: 'approval-2',
700
+ toolCallId: 'mcp-call-2',
701
+ },
702
+ {
703
+ type: 'finish',
704
+ finishReason: { unified: 'tool-calls', raw: undefined },
705
+ usage: testUsage,
706
+ },
707
+ ]);
708
+
709
+ const transformedStream = runToolsTransformation({
710
+ generateId: mockId({ prefix: 'id' }),
711
+ tools: {
712
+ mcp_search: {
713
+ type: 'provider',
714
+ id: 'mcp.mcp_search',
715
+ inputSchema: z.object({ query: z.string() }),
716
+ args: {},
717
+ },
718
+ mcp_execute: {
719
+ type: 'provider',
720
+ id: 'mcp.mcp_execute',
721
+ inputSchema: z.object({ command: z.string() }),
722
+ args: {},
723
+ },
724
+ },
725
+ generatorStream: inputStream,
726
+ tracer: new MockTracer(),
727
+ telemetry: undefined,
728
+ messages: [],
729
+ system: undefined,
730
+ abortSignal: undefined,
731
+ repairToolCall: undefined,
732
+ experimental_context: undefined,
733
+ });
734
+
735
+ const result = await convertReadableStreamToArray(transformedStream);
736
+
737
+ expect(result).toMatchInlineSnapshot(`
738
+ [
739
+ {
740
+ "input": {
741
+ "query": "first",
742
+ },
743
+ "providerExecuted": true,
744
+ "providerMetadata": undefined,
745
+ "title": undefined,
746
+ "toolCallId": "mcp-call-1",
747
+ "toolName": "mcp_search",
748
+ "type": "tool-call",
749
+ },
750
+ {
751
+ "input": {
752
+ "command": "ls",
753
+ },
754
+ "providerExecuted": true,
755
+ "providerMetadata": undefined,
756
+ "title": undefined,
757
+ "toolCallId": "mcp-call-2",
758
+ "toolName": "mcp_execute",
759
+ "type": "tool-call",
760
+ },
761
+ {
762
+ "approvalId": "approval-1",
763
+ "toolCall": {
764
+ "input": {
765
+ "query": "first",
766
+ },
767
+ "providerExecuted": true,
768
+ "providerMetadata": undefined,
769
+ "title": undefined,
770
+ "toolCallId": "mcp-call-1",
771
+ "toolName": "mcp_search",
772
+ "type": "tool-call",
773
+ },
774
+ "type": "tool-approval-request",
775
+ },
776
+ {
777
+ "approvalId": "approval-2",
778
+ "toolCall": {
779
+ "input": {
780
+ "command": "ls",
781
+ },
782
+ "providerExecuted": true,
783
+ "providerMetadata": undefined,
784
+ "title": undefined,
785
+ "toolCallId": "mcp-call-2",
786
+ "toolName": "mcp_execute",
787
+ "type": "tool-call",
788
+ },
789
+ "type": "tool-approval-request",
790
+ },
791
+ {
792
+ "finishReason": "tool-calls",
793
+ "providerMetadata": undefined,
794
+ "rawFinishReason": undefined,
795
+ "type": "finish",
796
+ "usage": {
797
+ "cachedInputTokens": undefined,
798
+ "inputTokenDetails": {
799
+ "cacheReadTokens": undefined,
800
+ "cacheWriteTokens": undefined,
801
+ "noCacheTokens": 3,
802
+ },
803
+ "inputTokens": 3,
804
+ "outputTokenDetails": {
805
+ "reasoningTokens": undefined,
806
+ "textTokens": 10,
807
+ },
808
+ "outputTokens": 10,
809
+ "raw": undefined,
810
+ "reasoningTokens": undefined,
811
+ "totalTokens": 13,
812
+ },
813
+ },
814
+ ]
815
+ `);
816
+ });
817
+ });
818
+
819
+ describe('Tool.onInputAvailable', () => {
820
+ it('should call onInputAvailable before the tool call is executed', async () => {
821
+ const output: unknown[] = [];
822
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
823
+ convertArrayToReadableStream([
824
+ {
825
+ type: 'tool-call',
826
+ toolCallId: 'call-1',
827
+ toolName: 'onInputAvailableTool',
828
+ input: `{ "value": "test" }`,
829
+ },
830
+ {
831
+ type: 'finish',
832
+ finishReason: { unified: 'stop', raw: 'stop' },
833
+ usage: testUsage,
834
+ },
835
+ ]);
836
+
837
+ const transformedStream = runToolsTransformation({
838
+ generateId: mockId({ prefix: 'id' }),
839
+ tools: {
840
+ onInputAvailableTool: tool({
841
+ inputSchema: z.object({ value: z.string() }),
842
+ onInputAvailable: async ({ input }) => {
843
+ output.push({ type: 'onInputAvailable', input });
844
+ },
845
+ }),
846
+ },
847
+ generatorStream: inputStream,
848
+ tracer: new MockTracer(),
849
+ telemetry: undefined,
850
+ messages: [],
851
+ system: undefined,
852
+ abortSignal: undefined,
853
+ repairToolCall: undefined,
854
+ experimental_context: undefined,
855
+ });
856
+
857
+ // consume each chunk to maintain order
858
+ const reader = transformedStream.getReader();
859
+ while (true) {
860
+ const { done, value } = await reader.read();
861
+ if (done) break;
862
+ output.push(value);
863
+ }
864
+
865
+ expect(output).toMatchInlineSnapshot(`
866
+ [
867
+ {
868
+ "input": {
869
+ "value": "test",
870
+ },
871
+ "type": "onInputAvailable",
872
+ },
873
+ {
874
+ "input": {
875
+ "value": "test",
876
+ },
877
+ "providerExecuted": undefined,
878
+ "providerMetadata": undefined,
879
+ "title": undefined,
880
+ "toolCallId": "call-1",
881
+ "toolName": "onInputAvailableTool",
882
+ "type": "tool-call",
883
+ },
884
+ {
885
+ "finishReason": "stop",
886
+ "providerMetadata": undefined,
887
+ "rawFinishReason": "stop",
888
+ "type": "finish",
889
+ "usage": {
890
+ "cachedInputTokens": undefined,
891
+ "inputTokenDetails": {
892
+ "cacheReadTokens": undefined,
893
+ "cacheWriteTokens": undefined,
894
+ "noCacheTokens": 3,
895
+ },
896
+ "inputTokens": 3,
897
+ "outputTokenDetails": {
898
+ "reasoningTokens": undefined,
899
+ "textTokens": 10,
900
+ },
901
+ "outputTokens": 10,
902
+ "raw": undefined,
903
+ "reasoningTokens": undefined,
904
+ "totalTokens": 13,
905
+ },
906
+ },
907
+ ]
908
+ `);
909
+ });
910
+
911
+ it('should call onInputAvailable when the tool needs approval', async () => {
912
+ const output: unknown[] = [];
913
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
914
+ convertArrayToReadableStream([
915
+ {
916
+ type: 'tool-call',
917
+ toolCallId: 'call-1',
918
+ toolName: 'onInputAvailableTool',
919
+ input: `{ "value": "test" }`,
920
+ },
921
+ {
922
+ type: 'finish',
923
+ finishReason: { unified: 'stop', raw: 'stop' },
924
+ usage: testUsage,
925
+ },
926
+ ]);
927
+
928
+ const transformedStream = runToolsTransformation({
929
+ generateId: mockId({ prefix: 'id' }),
930
+ tools: {
931
+ onInputAvailableTool: tool({
932
+ inputSchema: z.object({ value: z.string() }),
933
+ onInputAvailable: async ({ input }) => {
934
+ output.push({ type: 'onInputAvailable', input });
935
+ },
936
+ needsApproval: true,
937
+ }),
938
+ },
939
+ generatorStream: inputStream,
940
+ tracer: new MockTracer(),
941
+ telemetry: undefined,
942
+ messages: [],
943
+ system: undefined,
944
+ abortSignal: undefined,
945
+ repairToolCall: undefined,
946
+ experimental_context: undefined,
947
+ });
948
+
949
+ // consume each chunk to maintain order
950
+ const reader = transformedStream.getReader();
951
+ while (true) {
952
+ const { done, value } = await reader.read();
953
+ if (done) break;
954
+ output.push(value);
955
+ }
956
+
957
+ expect(output).toMatchInlineSnapshot(`
958
+ [
959
+ {
960
+ "input": {
961
+ "value": "test",
962
+ },
963
+ "type": "onInputAvailable",
964
+ },
965
+ {
966
+ "input": {
967
+ "value": "test",
968
+ },
969
+ "providerExecuted": undefined,
970
+ "providerMetadata": undefined,
971
+ "title": undefined,
972
+ "toolCallId": "call-1",
973
+ "toolName": "onInputAvailableTool",
974
+ "type": "tool-call",
975
+ },
976
+ {
977
+ "approvalId": "id-0",
978
+ "toolCall": {
979
+ "input": {
980
+ "value": "test",
981
+ },
982
+ "providerExecuted": undefined,
983
+ "providerMetadata": undefined,
984
+ "title": undefined,
985
+ "toolCallId": "call-1",
986
+ "toolName": "onInputAvailableTool",
987
+ "type": "tool-call",
988
+ },
989
+ "type": "tool-approval-request",
990
+ },
991
+ {
992
+ "finishReason": "stop",
993
+ "providerMetadata": undefined,
994
+ "rawFinishReason": "stop",
995
+ "type": "finish",
996
+ "usage": {
997
+ "cachedInputTokens": undefined,
998
+ "inputTokenDetails": {
999
+ "cacheReadTokens": undefined,
1000
+ "cacheWriteTokens": undefined,
1001
+ "noCacheTokens": 3,
1002
+ },
1003
+ "inputTokens": 3,
1004
+ "outputTokenDetails": {
1005
+ "reasoningTokens": undefined,
1006
+ "textTokens": 10,
1007
+ },
1008
+ "outputTokens": 10,
1009
+ "raw": undefined,
1010
+ "reasoningTokens": undefined,
1011
+ "totalTokens": 13,
1012
+ },
1013
+ },
1014
+ ]
1015
+ `);
1016
+ });
1017
+ });
1018
+
1019
+ describe('tool execution error handling', () => {
1020
+ it('should handle error thrown in async tool execution', async () => {
1021
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
1022
+ convertArrayToReadableStream([
1023
+ {
1024
+ type: 'tool-call',
1025
+ toolCallId: 'call-1',
1026
+ toolName: 'failingTool',
1027
+ input: `{ "value": "test" }`,
1028
+ },
1029
+ {
1030
+ type: 'finish',
1031
+ finishReason: { unified: 'stop', raw: 'stop' },
1032
+ usage: testUsage,
1033
+ },
1034
+ ]);
1035
+
1036
+ const toolError = new Error('Tool execution failed!');
1037
+
1038
+ const transformedStream = runToolsTransformation({
1039
+ generateId: mockId({ prefix: 'id' }),
1040
+ tools: {
1041
+ failingTool: {
1042
+ title: 'Failing Tool',
1043
+ inputSchema: z.object({ value: z.string() }),
1044
+ execute: async ({ value }) => {
1045
+ await delay(10);
1046
+ throw toolError;
1047
+ },
1048
+ },
1049
+ },
1050
+ generatorStream: inputStream,
1051
+ tracer: new MockTracer(),
1052
+ telemetry: undefined,
1053
+ messages: [],
1054
+ system: undefined,
1055
+ abortSignal: undefined,
1056
+ repairToolCall: undefined,
1057
+ experimental_context: undefined,
1058
+ });
1059
+
1060
+ const result = await convertReadableStreamToArray(transformedStream);
1061
+
1062
+ // tool-call should come first
1063
+ expect(result[0]).toMatchObject({
1064
+ type: 'tool-call',
1065
+ toolCallId: 'call-1',
1066
+ toolName: 'failingTool',
1067
+ });
1068
+
1069
+ // tool-error should be included
1070
+ expect(result).toContainEqual(
1071
+ expect.objectContaining({
1072
+ type: 'tool-error',
1073
+ toolCallId: 'call-1',
1074
+ toolName: 'failingTool',
1075
+ error: toolError,
1076
+ }),
1077
+ );
1078
+
1079
+ // finish should come last (stream closes properly)
1080
+ expect(result[result.length - 1]).toMatchObject({
1081
+ type: 'finish',
1082
+ finishReason: 'stop',
1083
+ });
1084
+ });
1085
+
1086
+ it('should handle error thrown in sync tool execution', async () => {
1087
+ const inputStream: ReadableStream<LanguageModelV3StreamPart> =
1088
+ convertArrayToReadableStream([
1089
+ {
1090
+ type: 'tool-call',
1091
+ toolCallId: 'call-1',
1092
+ toolName: 'failingTool',
1093
+ input: `{ "value": "test" }`,
1094
+ },
1095
+ {
1096
+ type: 'finish',
1097
+ finishReason: { unified: 'stop', raw: 'stop' },
1098
+ usage: testUsage,
1099
+ },
1100
+ ]);
1101
+
1102
+ const toolError = new Error('Sync tool failed!');
1103
+
1104
+ const transformedStream = runToolsTransformation({
1105
+ generateId: mockId({ prefix: 'id' }),
1106
+ tools: {
1107
+ failingTool: {
1108
+ title: 'Failing Tool',
1109
+ inputSchema: z.object({ value: z.string() }),
1110
+ execute: ({ value }) => {
1111
+ throw toolError;
1112
+ },
1113
+ },
1114
+ },
1115
+ generatorStream: inputStream,
1116
+ tracer: new MockTracer(),
1117
+ telemetry: undefined,
1118
+ messages: [],
1119
+ system: undefined,
1120
+ abortSignal: undefined,
1121
+ repairToolCall: undefined,
1122
+ experimental_context: undefined,
1123
+ });
1124
+
1125
+ const result = await convertReadableStreamToArray(transformedStream);
1126
+
1127
+ // tool-error should be included
1128
+ expect(result).toContainEqual(
1129
+ expect.objectContaining({
1130
+ type: 'tool-error',
1131
+ toolCallId: 'call-1',
1132
+ toolName: 'failingTool',
1133
+ error: toolError,
1134
+ }),
1135
+ );
1136
+
1137
+ // stream should close properly
1138
+ expect(result[result.length - 1]).toMatchObject({
1139
+ type: 'finish',
1140
+ });
1141
+ });
1142
+ });
1143
+ });