ai 6.0.33 → 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 (351) hide show
  1. package/CHANGELOG.md +10 -0
  2. package/dist/index.js +1 -1
  3. package/dist/index.mjs +1 -1
  4. package/dist/internal/index.js +1 -1
  5. package/dist/internal/index.mjs +1 -1
  6. package/docs/02-foundations/03-prompts.mdx +2 -2
  7. package/docs/03-ai-sdk-core/15-tools-and-tool-calling.mdx +1 -1
  8. package/docs/07-reference/01-ai-sdk-core/28-output.mdx +1 -1
  9. package/package.json +6 -4
  10. package/src/agent/agent.ts +116 -0
  11. package/src/agent/create-agent-ui-stream-response.test.ts +258 -0
  12. package/src/agent/create-agent-ui-stream-response.ts +50 -0
  13. package/src/agent/create-agent-ui-stream.ts +73 -0
  14. package/src/agent/index.ts +33 -0
  15. package/src/agent/infer-agent-tools.ts +7 -0
  16. package/src/agent/infer-agent-ui-message.test-d.ts +54 -0
  17. package/src/agent/infer-agent-ui-message.ts +11 -0
  18. package/src/agent/pipe-agent-ui-stream-to-response.ts +52 -0
  19. package/src/agent/tool-loop-agent-on-finish-callback.ts +31 -0
  20. package/src/agent/tool-loop-agent-on-step-finish-callback.ts +11 -0
  21. package/src/agent/tool-loop-agent-settings.ts +182 -0
  22. package/src/agent/tool-loop-agent.test-d.ts +114 -0
  23. package/src/agent/tool-loop-agent.test.ts +442 -0
  24. package/src/agent/tool-loop-agent.ts +114 -0
  25. package/src/embed/__snapshots__/embed-many.test.ts.snap +191 -0
  26. package/src/embed/__snapshots__/embed.test.ts.snap +81 -0
  27. package/src/embed/embed-many-result.ts +53 -0
  28. package/src/embed/embed-many.test.ts +653 -0
  29. package/src/embed/embed-many.ts +378 -0
  30. package/src/embed/embed-result.ts +50 -0
  31. package/src/embed/embed.test.ts +298 -0
  32. package/src/embed/embed.ts +211 -0
  33. package/src/embed/index.ts +4 -0
  34. package/src/error/index.ts +34 -0
  35. package/src/error/invalid-argument-error.ts +34 -0
  36. package/src/error/invalid-stream-part-error.ts +28 -0
  37. package/src/error/invalid-tool-approval-error.ts +26 -0
  38. package/src/error/invalid-tool-input-error.ts +33 -0
  39. package/src/error/no-image-generated-error.ts +39 -0
  40. package/src/error/no-object-generated-error.ts +70 -0
  41. package/src/error/no-output-generated-error.ts +26 -0
  42. package/src/error/no-speech-generated-error.ts +18 -0
  43. package/src/error/no-such-tool-error.ts +35 -0
  44. package/src/error/no-transcript-generated-error.ts +20 -0
  45. package/src/error/tool-call-not-found-for-approval-error.ts +32 -0
  46. package/src/error/tool-call-repair-error.ts +30 -0
  47. package/src/error/unsupported-model-version-error.ts +23 -0
  48. package/src/error/verify-no-object-generated-error.ts +27 -0
  49. package/src/generate-image/generate-image-result.ts +42 -0
  50. package/src/generate-image/generate-image.test.ts +1420 -0
  51. package/src/generate-image/generate-image.ts +360 -0
  52. package/src/generate-image/index.ts +18 -0
  53. package/src/generate-object/__snapshots__/generate-object.test.ts.snap +133 -0
  54. package/src/generate-object/__snapshots__/stream-object.test.ts.snap +297 -0
  55. package/src/generate-object/generate-object-result.ts +67 -0
  56. package/src/generate-object/generate-object.test-d.ts +49 -0
  57. package/src/generate-object/generate-object.test.ts +1191 -0
  58. package/src/generate-object/generate-object.ts +518 -0
  59. package/src/generate-object/index.ts +9 -0
  60. package/src/generate-object/inject-json-instruction.test.ts +181 -0
  61. package/src/generate-object/inject-json-instruction.ts +30 -0
  62. package/src/generate-object/output-strategy.ts +415 -0
  63. package/src/generate-object/parse-and-validate-object-result.ts +111 -0
  64. package/src/generate-object/repair-text.ts +12 -0
  65. package/src/generate-object/stream-object-result.ts +120 -0
  66. package/src/generate-object/stream-object.test-d.ts +74 -0
  67. package/src/generate-object/stream-object.test.ts +1950 -0
  68. package/src/generate-object/stream-object.ts +986 -0
  69. package/src/generate-object/validate-object-generation-input.ts +144 -0
  70. package/src/generate-speech/generate-speech-result.ts +30 -0
  71. package/src/generate-speech/generate-speech.test.ts +300 -0
  72. package/src/generate-speech/generate-speech.ts +190 -0
  73. package/src/generate-speech/generated-audio-file.ts +65 -0
  74. package/src/generate-speech/index.ts +3 -0
  75. package/src/generate-text/__snapshots__/generate-text.test.ts.snap +1872 -0
  76. package/src/generate-text/__snapshots__/stream-text.test.ts.snap +1255 -0
  77. package/src/generate-text/collect-tool-approvals.test.ts +553 -0
  78. package/src/generate-text/collect-tool-approvals.ts +116 -0
  79. package/src/generate-text/content-part.ts +25 -0
  80. package/src/generate-text/execute-tool-call.ts +129 -0
  81. package/src/generate-text/extract-reasoning-content.ts +17 -0
  82. package/src/generate-text/extract-text-content.ts +15 -0
  83. package/src/generate-text/generate-text-result.ts +168 -0
  84. package/src/generate-text/generate-text.test-d.ts +68 -0
  85. package/src/generate-text/generate-text.test.ts +7011 -0
  86. package/src/generate-text/generate-text.ts +1223 -0
  87. package/src/generate-text/generated-file.ts +70 -0
  88. package/src/generate-text/index.ts +57 -0
  89. package/src/generate-text/is-approval-needed.ts +29 -0
  90. package/src/generate-text/output-utils.ts +23 -0
  91. package/src/generate-text/output.test.ts +698 -0
  92. package/src/generate-text/output.ts +590 -0
  93. package/src/generate-text/parse-tool-call.test.ts +570 -0
  94. package/src/generate-text/parse-tool-call.ts +188 -0
  95. package/src/generate-text/prepare-step.ts +103 -0
  96. package/src/generate-text/prune-messages.test.ts +720 -0
  97. package/src/generate-text/prune-messages.ts +167 -0
  98. package/src/generate-text/reasoning-output.ts +20 -0
  99. package/src/generate-text/reasoning.ts +8 -0
  100. package/src/generate-text/response-message.ts +10 -0
  101. package/src/generate-text/run-tools-transformation.test.ts +1143 -0
  102. package/src/generate-text/run-tools-transformation.ts +420 -0
  103. package/src/generate-text/smooth-stream.test.ts +2101 -0
  104. package/src/generate-text/smooth-stream.ts +162 -0
  105. package/src/generate-text/step-result.ts +238 -0
  106. package/src/generate-text/stop-condition.ts +29 -0
  107. package/src/generate-text/stream-text-result.ts +463 -0
  108. package/src/generate-text/stream-text.test-d.ts +200 -0
  109. package/src/generate-text/stream-text.test.ts +19979 -0
  110. package/src/generate-text/stream-text.ts +2505 -0
  111. package/src/generate-text/to-response-messages.test.ts +922 -0
  112. package/src/generate-text/to-response-messages.ts +163 -0
  113. package/src/generate-text/tool-approval-request-output.ts +21 -0
  114. package/src/generate-text/tool-call-repair-function.ts +27 -0
  115. package/src/generate-text/tool-call.ts +47 -0
  116. package/src/generate-text/tool-error.ts +34 -0
  117. package/src/generate-text/tool-output-denied.ts +21 -0
  118. package/src/generate-text/tool-output.ts +7 -0
  119. package/src/generate-text/tool-result.ts +36 -0
  120. package/src/generate-text/tool-set.ts +14 -0
  121. package/src/global.ts +24 -0
  122. package/src/index.ts +50 -0
  123. package/src/logger/index.ts +6 -0
  124. package/src/logger/log-warnings.test.ts +351 -0
  125. package/src/logger/log-warnings.ts +119 -0
  126. package/src/middleware/__snapshots__/simulate-streaming-middleware.test.ts.snap +64 -0
  127. package/src/middleware/add-tool-input-examples-middleware.test.ts +476 -0
  128. package/src/middleware/add-tool-input-examples-middleware.ts +90 -0
  129. package/src/middleware/default-embedding-settings-middleware.test.ts +126 -0
  130. package/src/middleware/default-embedding-settings-middleware.ts +22 -0
  131. package/src/middleware/default-settings-middleware.test.ts +388 -0
  132. package/src/middleware/default-settings-middleware.ts +33 -0
  133. package/src/middleware/extract-json-middleware.test.ts +827 -0
  134. package/src/middleware/extract-json-middleware.ts +197 -0
  135. package/src/middleware/extract-reasoning-middleware.test.ts +1028 -0
  136. package/src/middleware/extract-reasoning-middleware.ts +238 -0
  137. package/src/middleware/index.ts +10 -0
  138. package/src/middleware/simulate-streaming-middleware.test.ts +911 -0
  139. package/src/middleware/simulate-streaming-middleware.ts +79 -0
  140. package/src/middleware/wrap-embedding-model.test.ts +358 -0
  141. package/src/middleware/wrap-embedding-model.ts +86 -0
  142. package/src/middleware/wrap-image-model.test.ts +423 -0
  143. package/src/middleware/wrap-image-model.ts +85 -0
  144. package/src/middleware/wrap-language-model.test.ts +518 -0
  145. package/src/middleware/wrap-language-model.ts +104 -0
  146. package/src/middleware/wrap-provider.test.ts +120 -0
  147. package/src/middleware/wrap-provider.ts +51 -0
  148. package/src/model/as-embedding-model-v3.test.ts +319 -0
  149. package/src/model/as-embedding-model-v3.ts +24 -0
  150. package/src/model/as-image-model-v3.test.ts +409 -0
  151. package/src/model/as-image-model-v3.ts +24 -0
  152. package/src/model/as-language-model-v3.test.ts +508 -0
  153. package/src/model/as-language-model-v3.ts +103 -0
  154. package/src/model/as-provider-v3.ts +36 -0
  155. package/src/model/as-speech-model-v3.test.ts +356 -0
  156. package/src/model/as-speech-model-v3.ts +24 -0
  157. package/src/model/as-transcription-model-v3.test.ts +529 -0
  158. package/src/model/as-transcription-model-v3.ts +24 -0
  159. package/src/model/resolve-model.test.ts +244 -0
  160. package/src/model/resolve-model.ts +126 -0
  161. package/src/prompt/call-settings.ts +148 -0
  162. package/src/prompt/content-part.ts +209 -0
  163. package/src/prompt/convert-to-language-model-prompt.test.ts +2018 -0
  164. package/src/prompt/convert-to-language-model-prompt.ts +442 -0
  165. package/src/prompt/create-tool-model-output.test.ts +508 -0
  166. package/src/prompt/create-tool-model-output.ts +34 -0
  167. package/src/prompt/data-content.test.ts +15 -0
  168. package/src/prompt/data-content.ts +134 -0
  169. package/src/prompt/index.ts +27 -0
  170. package/src/prompt/invalid-data-content-error.ts +29 -0
  171. package/src/prompt/invalid-message-role-error.ts +27 -0
  172. package/src/prompt/message-conversion-error.ts +28 -0
  173. package/src/prompt/message.ts +68 -0
  174. package/src/prompt/prepare-call-settings.test.ts +159 -0
  175. package/src/prompt/prepare-call-settings.ts +108 -0
  176. package/src/prompt/prepare-tools-and-tool-choice.test.ts +461 -0
  177. package/src/prompt/prepare-tools-and-tool-choice.ts +86 -0
  178. package/src/prompt/prompt.ts +43 -0
  179. package/src/prompt/split-data-url.ts +17 -0
  180. package/src/prompt/standardize-prompt.test.ts +82 -0
  181. package/src/prompt/standardize-prompt.ts +99 -0
  182. package/src/prompt/wrap-gateway-error.ts +29 -0
  183. package/src/registry/custom-provider.test.ts +211 -0
  184. package/src/registry/custom-provider.ts +155 -0
  185. package/src/registry/index.ts +7 -0
  186. package/src/registry/no-such-provider-error.ts +41 -0
  187. package/src/registry/provider-registry.test.ts +691 -0
  188. package/src/registry/provider-registry.ts +328 -0
  189. package/src/rerank/index.ts +2 -0
  190. package/src/rerank/rerank-result.ts +70 -0
  191. package/src/rerank/rerank.test.ts +516 -0
  192. package/src/rerank/rerank.ts +237 -0
  193. package/src/telemetry/assemble-operation-name.ts +21 -0
  194. package/src/telemetry/get-base-telemetry-attributes.ts +53 -0
  195. package/src/telemetry/get-tracer.ts +20 -0
  196. package/src/telemetry/noop-tracer.ts +69 -0
  197. package/src/telemetry/record-span.ts +63 -0
  198. package/src/telemetry/select-telemetry-attributes.ts +78 -0
  199. package/src/telemetry/select-temetry-attributes.test.ts +114 -0
  200. package/src/telemetry/stringify-for-telemetry.test.ts +114 -0
  201. package/src/telemetry/stringify-for-telemetry.ts +33 -0
  202. package/src/telemetry/telemetry-settings.ts +44 -0
  203. package/src/test/mock-embedding-model-v2.ts +35 -0
  204. package/src/test/mock-embedding-model-v3.ts +48 -0
  205. package/src/test/mock-image-model-v2.ts +28 -0
  206. package/src/test/mock-image-model-v3.ts +28 -0
  207. package/src/test/mock-language-model-v2.ts +72 -0
  208. package/src/test/mock-language-model-v3.ts +77 -0
  209. package/src/test/mock-provider-v2.ts +68 -0
  210. package/src/test/mock-provider-v3.ts +80 -0
  211. package/src/test/mock-reranking-model-v3.ts +25 -0
  212. package/src/test/mock-server-response.ts +69 -0
  213. package/src/test/mock-speech-model-v2.ts +24 -0
  214. package/src/test/mock-speech-model-v3.ts +24 -0
  215. package/src/test/mock-tracer.ts +156 -0
  216. package/src/test/mock-transcription-model-v2.ts +24 -0
  217. package/src/test/mock-transcription-model-v3.ts +24 -0
  218. package/src/test/mock-values.ts +4 -0
  219. package/src/test/not-implemented.ts +3 -0
  220. package/src/text-stream/create-text-stream-response.test.ts +38 -0
  221. package/src/text-stream/create-text-stream-response.ts +18 -0
  222. package/src/text-stream/index.ts +2 -0
  223. package/src/text-stream/pipe-text-stream-to-response.test.ts +38 -0
  224. package/src/text-stream/pipe-text-stream-to-response.ts +26 -0
  225. package/src/transcribe/index.ts +2 -0
  226. package/src/transcribe/transcribe-result.ts +60 -0
  227. package/src/transcribe/transcribe.test.ts +313 -0
  228. package/src/transcribe/transcribe.ts +173 -0
  229. package/src/types/embedding-model-middleware.ts +3 -0
  230. package/src/types/embedding-model.ts +18 -0
  231. package/src/types/image-model-middleware.ts +3 -0
  232. package/src/types/image-model-response-metadata.ts +16 -0
  233. package/src/types/image-model.ts +19 -0
  234. package/src/types/index.ts +29 -0
  235. package/src/types/json-value.ts +15 -0
  236. package/src/types/language-model-middleware.ts +3 -0
  237. package/src/types/language-model-request-metadata.ts +6 -0
  238. package/src/types/language-model-response-metadata.ts +21 -0
  239. package/src/types/language-model.ts +104 -0
  240. package/src/types/provider-metadata.ts +16 -0
  241. package/src/types/provider.ts +55 -0
  242. package/src/types/reranking-model.ts +6 -0
  243. package/src/types/speech-model-response-metadata.ts +21 -0
  244. package/src/types/speech-model.ts +6 -0
  245. package/src/types/transcription-model-response-metadata.ts +16 -0
  246. package/src/types/transcription-model.ts +9 -0
  247. package/src/types/usage.ts +200 -0
  248. package/src/types/warning.ts +7 -0
  249. package/src/ui/__snapshots__/append-response-messages.test.ts.snap +416 -0
  250. package/src/ui/__snapshots__/convert-to-model-messages.test.ts.snap +419 -0
  251. package/src/ui/__snapshots__/process-chat-text-response.test.ts.snap +142 -0
  252. package/src/ui/call-completion-api.ts +157 -0
  253. package/src/ui/chat-transport.ts +83 -0
  254. package/src/ui/chat.test-d.ts +233 -0
  255. package/src/ui/chat.test.ts +2695 -0
  256. package/src/ui/chat.ts +716 -0
  257. package/src/ui/convert-file-list-to-file-ui-parts.ts +36 -0
  258. package/src/ui/convert-to-model-messages.test.ts +2775 -0
  259. package/src/ui/convert-to-model-messages.ts +373 -0
  260. package/src/ui/default-chat-transport.ts +36 -0
  261. package/src/ui/direct-chat-transport.test.ts +446 -0
  262. package/src/ui/direct-chat-transport.ts +118 -0
  263. package/src/ui/http-chat-transport.test.ts +185 -0
  264. package/src/ui/http-chat-transport.ts +292 -0
  265. package/src/ui/index.ts +71 -0
  266. package/src/ui/last-assistant-message-is-complete-with-approval-responses.ts +44 -0
  267. package/src/ui/last-assistant-message-is-complete-with-tool-calls.test.ts +371 -0
  268. package/src/ui/last-assistant-message-is-complete-with-tool-calls.ts +39 -0
  269. package/src/ui/process-text-stream.test.ts +38 -0
  270. package/src/ui/process-text-stream.ts +16 -0
  271. package/src/ui/process-ui-message-stream.test.ts +8052 -0
  272. package/src/ui/process-ui-message-stream.ts +713 -0
  273. package/src/ui/text-stream-chat-transport.ts +23 -0
  274. package/src/ui/transform-text-to-ui-message-stream.test.ts +124 -0
  275. package/src/ui/transform-text-to-ui-message-stream.ts +27 -0
  276. package/src/ui/ui-messages.test.ts +48 -0
  277. package/src/ui/ui-messages.ts +534 -0
  278. package/src/ui/use-completion.ts +84 -0
  279. package/src/ui/validate-ui-messages.test.ts +1428 -0
  280. package/src/ui/validate-ui-messages.ts +476 -0
  281. package/src/ui-message-stream/create-ui-message-stream-response.test.ts +266 -0
  282. package/src/ui-message-stream/create-ui-message-stream-response.ts +32 -0
  283. package/src/ui-message-stream/create-ui-message-stream.test.ts +639 -0
  284. package/src/ui-message-stream/create-ui-message-stream.ts +124 -0
  285. package/src/ui-message-stream/get-response-ui-message-id.test.ts +55 -0
  286. package/src/ui-message-stream/get-response-ui-message-id.ts +24 -0
  287. package/src/ui-message-stream/handle-ui-message-stream-finish.test.ts +429 -0
  288. package/src/ui-message-stream/handle-ui-message-stream-finish.ts +135 -0
  289. package/src/ui-message-stream/index.ts +13 -0
  290. package/src/ui-message-stream/json-to-sse-transform-stream.ts +12 -0
  291. package/src/ui-message-stream/pipe-ui-message-stream-to-response.test.ts +90 -0
  292. package/src/ui-message-stream/pipe-ui-message-stream-to-response.ts +40 -0
  293. package/src/ui-message-stream/read-ui-message-stream.test.ts +122 -0
  294. package/src/ui-message-stream/read-ui-message-stream.ts +87 -0
  295. package/src/ui-message-stream/ui-message-chunks.test-d.ts +18 -0
  296. package/src/ui-message-stream/ui-message-chunks.ts +344 -0
  297. package/src/ui-message-stream/ui-message-stream-headers.ts +7 -0
  298. package/src/ui-message-stream/ui-message-stream-on-finish-callback.ts +32 -0
  299. package/src/ui-message-stream/ui-message-stream-response-init.ts +5 -0
  300. package/src/ui-message-stream/ui-message-stream-writer.ts +24 -0
  301. package/src/util/as-array.ts +3 -0
  302. package/src/util/async-iterable-stream.test.ts +241 -0
  303. package/src/util/async-iterable-stream.ts +94 -0
  304. package/src/util/consume-stream.ts +29 -0
  305. package/src/util/cosine-similarity.test.ts +57 -0
  306. package/src/util/cosine-similarity.ts +47 -0
  307. package/src/util/create-resolvable-promise.ts +30 -0
  308. package/src/util/create-stitchable-stream.test.ts +239 -0
  309. package/src/util/create-stitchable-stream.ts +112 -0
  310. package/src/util/data-url.ts +17 -0
  311. package/src/util/deep-partial.ts +84 -0
  312. package/src/util/detect-media-type.test.ts +670 -0
  313. package/src/util/detect-media-type.ts +184 -0
  314. package/src/util/download/download-function.ts +45 -0
  315. package/src/util/download/download.test.ts +69 -0
  316. package/src/util/download/download.ts +46 -0
  317. package/src/util/error-handler.ts +1 -0
  318. package/src/util/fix-json.test.ts +279 -0
  319. package/src/util/fix-json.ts +401 -0
  320. package/src/util/get-potential-start-index.test.ts +34 -0
  321. package/src/util/get-potential-start-index.ts +30 -0
  322. package/src/util/index.ts +11 -0
  323. package/src/util/is-deep-equal-data.test.ts +119 -0
  324. package/src/util/is-deep-equal-data.ts +48 -0
  325. package/src/util/is-non-empty-object.ts +5 -0
  326. package/src/util/job.ts +1 -0
  327. package/src/util/log-v2-compatibility-warning.ts +21 -0
  328. package/src/util/merge-abort-signals.test.ts +155 -0
  329. package/src/util/merge-abort-signals.ts +43 -0
  330. package/src/util/merge-objects.test.ts +118 -0
  331. package/src/util/merge-objects.ts +79 -0
  332. package/src/util/now.ts +4 -0
  333. package/src/util/parse-partial-json.test.ts +80 -0
  334. package/src/util/parse-partial-json.ts +30 -0
  335. package/src/util/prepare-headers.test.ts +51 -0
  336. package/src/util/prepare-headers.ts +14 -0
  337. package/src/util/prepare-retries.test.ts +10 -0
  338. package/src/util/prepare-retries.ts +47 -0
  339. package/src/util/retry-error.ts +41 -0
  340. package/src/util/retry-with-exponential-backoff.test.ts +446 -0
  341. package/src/util/retry-with-exponential-backoff.ts +154 -0
  342. package/src/util/serial-job-executor.test.ts +162 -0
  343. package/src/util/serial-job-executor.ts +36 -0
  344. package/src/util/simulate-readable-stream.test.ts +98 -0
  345. package/src/util/simulate-readable-stream.ts +39 -0
  346. package/src/util/split-array.test.ts +60 -0
  347. package/src/util/split-array.ts +20 -0
  348. package/src/util/value-of.ts +65 -0
  349. package/src/util/write-to-server-response.test.ts +266 -0
  350. package/src/util/write-to-server-response.ts +49 -0
  351. package/src/version.ts +5 -0
@@ -0,0 +1,420 @@
1
+ import { LanguageModelV3StreamPart, SharedV3Warning } from '@ai-sdk/provider';
2
+ import {
3
+ getErrorMessage,
4
+ IdGenerator,
5
+ ModelMessage,
6
+ SystemModelMessage,
7
+ } from '@ai-sdk/provider-utils';
8
+ import { Tracer } from '@opentelemetry/api';
9
+ import { ToolCallNotFoundForApprovalError } from '../error/tool-call-not-found-for-approval-error';
10
+ import { TelemetrySettings } from '../telemetry/telemetry-settings';
11
+ import { FinishReason, LanguageModelUsage, ProviderMetadata } from '../types';
12
+ import { Source } from '../types/language-model';
13
+ import { asLanguageModelUsage } from '../types/usage';
14
+ import { executeToolCall } from './execute-tool-call';
15
+ import { DefaultGeneratedFileWithType, GeneratedFile } from './generated-file';
16
+ import { isApprovalNeeded } from './is-approval-needed';
17
+ import { parseToolCall } from './parse-tool-call';
18
+ import { ToolApprovalRequestOutput } from './tool-approval-request-output';
19
+ import { TypedToolCall } from './tool-call';
20
+ import { ToolCallRepairFunction } from './tool-call-repair-function';
21
+ import { TypedToolError } from './tool-error';
22
+ import { TypedToolResult } from './tool-result';
23
+ import { ToolSet } from './tool-set';
24
+
25
+ export type SingleRequestTextStreamPart<TOOLS extends ToolSet> =
26
+ // Text blocks:
27
+ | {
28
+ type: 'text-start';
29
+ providerMetadata?: ProviderMetadata;
30
+ id: string;
31
+ }
32
+ | {
33
+ type: 'text-delta';
34
+ id: string;
35
+ providerMetadata?: ProviderMetadata;
36
+ delta: string;
37
+ }
38
+ | {
39
+ type: 'text-end';
40
+ providerMetadata?: ProviderMetadata;
41
+ id: string;
42
+ }
43
+
44
+ // Reasoning blocks:
45
+ | {
46
+ type: 'reasoning-start';
47
+ providerMetadata?: ProviderMetadata;
48
+ id: string;
49
+ }
50
+ | {
51
+ type: 'reasoning-delta';
52
+ id: string;
53
+ providerMetadata?: ProviderMetadata;
54
+ delta: string;
55
+ }
56
+ | {
57
+ type: 'reasoning-end';
58
+ id: string;
59
+ providerMetadata?: ProviderMetadata;
60
+ }
61
+
62
+ // Tool calls:
63
+ | {
64
+ type: 'tool-input-start';
65
+ id: string;
66
+ toolName: string;
67
+ providerMetadata?: ProviderMetadata;
68
+ dynamic?: boolean;
69
+ title?: string;
70
+ }
71
+ | {
72
+ type: 'tool-input-delta';
73
+ id: string;
74
+ delta: string;
75
+ providerMetadata?: ProviderMetadata;
76
+ }
77
+ | {
78
+ type: 'tool-input-end';
79
+ id: string;
80
+ providerMetadata?: ProviderMetadata;
81
+ }
82
+ | ToolApprovalRequestOutput<TOOLS>
83
+
84
+ // Other types:
85
+ | ({ type: 'source' } & Source)
86
+ | { type: 'file'; file: GeneratedFile } // different because of GeneratedFile object
87
+ | ({ type: 'tool-call' } & TypedToolCall<TOOLS>)
88
+ | ({ type: 'tool-result' } & TypedToolResult<TOOLS>)
89
+ | ({ type: 'tool-error' } & TypedToolError<TOOLS>)
90
+ | { type: 'file'; file: GeneratedFile } // different because of GeneratedFile object
91
+ | { type: 'stream-start'; warnings: SharedV3Warning[] }
92
+ | {
93
+ type: 'response-metadata';
94
+ id?: string;
95
+ timestamp?: Date;
96
+ modelId?: string;
97
+ }
98
+ | {
99
+ type: 'finish';
100
+ finishReason: FinishReason;
101
+ rawFinishReason: string | undefined;
102
+ usage: LanguageModelUsage;
103
+ providerMetadata?: ProviderMetadata;
104
+ }
105
+ | { type: 'error'; error: unknown }
106
+ | { type: 'raw'; rawValue: unknown };
107
+
108
+ export function runToolsTransformation<TOOLS extends ToolSet>({
109
+ tools,
110
+ generatorStream,
111
+ tracer,
112
+ telemetry,
113
+ system,
114
+ messages,
115
+ abortSignal,
116
+ repairToolCall,
117
+ experimental_context,
118
+ generateId,
119
+ }: {
120
+ tools: TOOLS | undefined;
121
+ generatorStream: ReadableStream<LanguageModelV3StreamPart>;
122
+ tracer: Tracer;
123
+ telemetry: TelemetrySettings | undefined;
124
+ system: string | SystemModelMessage | Array<SystemModelMessage> | undefined;
125
+ messages: ModelMessage[];
126
+ abortSignal: AbortSignal | undefined;
127
+ repairToolCall: ToolCallRepairFunction<TOOLS> | undefined;
128
+ experimental_context: unknown;
129
+ generateId: IdGenerator;
130
+ }): ReadableStream<SingleRequestTextStreamPart<TOOLS>> {
131
+ // tool results stream
132
+ let toolResultsStreamController: ReadableStreamDefaultController<
133
+ SingleRequestTextStreamPart<TOOLS>
134
+ > | null = null;
135
+ const toolResultsStream = new ReadableStream<
136
+ SingleRequestTextStreamPart<TOOLS>
137
+ >({
138
+ start(controller) {
139
+ toolResultsStreamController = controller;
140
+ },
141
+ });
142
+
143
+ // keep track of outstanding tool results for stream closing:
144
+ const outstandingToolResults = new Set<string>();
145
+
146
+ // keep track of tool inputs for provider-side tool results
147
+ const toolInputs = new Map<string, unknown>();
148
+
149
+ // keep track of parsed tool calls so provider-emitted approval requests can reference them
150
+ const toolCallsByToolCallId = new Map<string, TypedToolCall<TOOLS>>();
151
+
152
+ let canClose = false;
153
+ let finishChunk:
154
+ | (SingleRequestTextStreamPart<TOOLS> & { type: 'finish' })
155
+ | undefined = undefined;
156
+
157
+ function attemptClose() {
158
+ // close the tool results controller if no more outstanding tool calls
159
+ if (canClose && outstandingToolResults.size === 0) {
160
+ // we delay sending the finish chunk until all tool results (incl. delayed ones)
161
+ // are received to ensure that the frontend receives tool results before a message
162
+ // finish event arrives.
163
+ if (finishChunk != null) {
164
+ toolResultsStreamController!.enqueue(finishChunk);
165
+ }
166
+
167
+ toolResultsStreamController!.close();
168
+ }
169
+ }
170
+
171
+ // forward stream
172
+ const forwardStream = new TransformStream<
173
+ LanguageModelV3StreamPart,
174
+ SingleRequestTextStreamPart<TOOLS>
175
+ >({
176
+ async transform(
177
+ chunk: LanguageModelV3StreamPart,
178
+ controller: TransformStreamDefaultController<
179
+ SingleRequestTextStreamPart<TOOLS>
180
+ >,
181
+ ) {
182
+ const chunkType = chunk.type;
183
+
184
+ switch (chunkType) {
185
+ // forward:
186
+ case 'stream-start':
187
+ case 'text-start':
188
+ case 'text-delta':
189
+ case 'text-end':
190
+ case 'reasoning-start':
191
+ case 'reasoning-delta':
192
+ case 'reasoning-end':
193
+ case 'tool-input-start':
194
+ case 'tool-input-delta':
195
+ case 'tool-input-end':
196
+ case 'source':
197
+ case 'response-metadata':
198
+ case 'error':
199
+ case 'raw': {
200
+ controller.enqueue(chunk);
201
+ break;
202
+ }
203
+
204
+ case 'file': {
205
+ controller.enqueue({
206
+ type: 'file',
207
+ file: new DefaultGeneratedFileWithType({
208
+ data: chunk.data,
209
+ mediaType: chunk.mediaType,
210
+ }),
211
+ });
212
+ break;
213
+ }
214
+
215
+ case 'finish': {
216
+ finishChunk = {
217
+ type: 'finish',
218
+ finishReason: chunk.finishReason.unified,
219
+ rawFinishReason: chunk.finishReason.raw,
220
+ usage: asLanguageModelUsage(chunk.usage),
221
+ providerMetadata: chunk.providerMetadata,
222
+ };
223
+ break;
224
+ }
225
+
226
+ case 'tool-approval-request': {
227
+ const toolCall = toolCallsByToolCallId.get(chunk.toolCallId);
228
+ if (toolCall == null) {
229
+ toolResultsStreamController!.enqueue({
230
+ type: 'error',
231
+ error: new ToolCallNotFoundForApprovalError({
232
+ toolCallId: chunk.toolCallId,
233
+ approvalId: chunk.approvalId,
234
+ }),
235
+ });
236
+ break;
237
+ }
238
+
239
+ controller.enqueue({
240
+ type: 'tool-approval-request',
241
+ approvalId: chunk.approvalId,
242
+ toolCall,
243
+ });
244
+ break;
245
+ }
246
+
247
+ // process tool call:
248
+ case 'tool-call': {
249
+ try {
250
+ const toolCall = await parseToolCall({
251
+ toolCall: chunk,
252
+ tools,
253
+ repairToolCall,
254
+ system,
255
+ messages,
256
+ });
257
+
258
+ toolCallsByToolCallId.set(toolCall.toolCallId, toolCall);
259
+ controller.enqueue(toolCall);
260
+
261
+ if (toolCall.invalid) {
262
+ toolResultsStreamController!.enqueue({
263
+ type: 'tool-error',
264
+ toolCallId: toolCall.toolCallId,
265
+ toolName: toolCall.toolName,
266
+ input: toolCall.input,
267
+ error: getErrorMessage(toolCall.error!),
268
+ dynamic: true,
269
+ title: toolCall.title,
270
+ });
271
+ break;
272
+ }
273
+
274
+ const tool = tools?.[toolCall.toolName];
275
+
276
+ if (tool == null) {
277
+ // ignore tool calls for tools that are not available,
278
+ // e.g. provider-executed dynamic tools
279
+ break;
280
+ }
281
+
282
+ if (tool.onInputAvailable != null) {
283
+ await tool.onInputAvailable({
284
+ input: toolCall.input,
285
+ toolCallId: toolCall.toolCallId,
286
+ messages,
287
+ abortSignal,
288
+ experimental_context,
289
+ });
290
+ }
291
+
292
+ if (
293
+ await isApprovalNeeded({
294
+ tool,
295
+ toolCall,
296
+ messages,
297
+ experimental_context,
298
+ })
299
+ ) {
300
+ toolResultsStreamController!.enqueue({
301
+ type: 'tool-approval-request',
302
+ approvalId: generateId(),
303
+ toolCall,
304
+ });
305
+ break;
306
+ }
307
+
308
+ toolInputs.set(toolCall.toolCallId, toolCall.input);
309
+
310
+ // Only execute tools that are not provider-executed:
311
+ if (tool.execute != null && toolCall.providerExecuted !== true) {
312
+ const toolExecutionId = generateId(); // use our own id to guarantee uniqueness
313
+ outstandingToolResults.add(toolExecutionId);
314
+
315
+ // Note: we don't await the tool execution here (by leaving out 'await' on recordSpan),
316
+ // because we want to process the next chunk as soon as possible.
317
+ // This is important for the case where the tool execution takes a long time.
318
+ executeToolCall({
319
+ toolCall,
320
+ tools,
321
+ tracer,
322
+ telemetry,
323
+ messages,
324
+ abortSignal,
325
+ experimental_context,
326
+ onPreliminaryToolResult: result => {
327
+ toolResultsStreamController!.enqueue(result);
328
+ },
329
+ })
330
+ .then(result => {
331
+ toolResultsStreamController!.enqueue(result);
332
+ })
333
+ .catch(error => {
334
+ toolResultsStreamController!.enqueue({
335
+ type: 'error',
336
+ error,
337
+ });
338
+ })
339
+ .finally(() => {
340
+ outstandingToolResults.delete(toolExecutionId);
341
+ attemptClose();
342
+ });
343
+ }
344
+ } catch (error) {
345
+ toolResultsStreamController!.enqueue({ type: 'error', error });
346
+ }
347
+
348
+ break;
349
+ }
350
+
351
+ case 'tool-result': {
352
+ const toolName = chunk.toolName as keyof TOOLS & string;
353
+
354
+ if (chunk.isError) {
355
+ toolResultsStreamController!.enqueue({
356
+ type: 'tool-error',
357
+ toolCallId: chunk.toolCallId,
358
+ toolName,
359
+ input: toolInputs.get(chunk.toolCallId),
360
+ providerExecuted: true,
361
+ error: chunk.result,
362
+ dynamic: chunk.dynamic,
363
+ } as TypedToolError<TOOLS>);
364
+ } else {
365
+ controller.enqueue({
366
+ type: 'tool-result',
367
+ toolCallId: chunk.toolCallId,
368
+ toolName,
369
+ input: toolInputs.get(chunk.toolCallId),
370
+ output: chunk.result,
371
+ providerExecuted: true,
372
+ dynamic: chunk.dynamic,
373
+ } as TypedToolResult<TOOLS>);
374
+ }
375
+ break;
376
+ }
377
+
378
+ default: {
379
+ const _exhaustiveCheck: never = chunkType;
380
+ throw new Error(`Unhandled chunk type: ${_exhaustiveCheck}`);
381
+ }
382
+ }
383
+ },
384
+
385
+ flush() {
386
+ canClose = true;
387
+ attemptClose();
388
+ },
389
+ });
390
+
391
+ // combine the generator stream and the tool results stream
392
+ return new ReadableStream<SingleRequestTextStreamPart<TOOLS>>({
393
+ async start(controller) {
394
+ // need to wait for both pipes so there are no dangling promises that
395
+ // can cause uncaught promise rejections when the stream is aborted
396
+ return Promise.all([
397
+ generatorStream.pipeThrough(forwardStream).pipeTo(
398
+ new WritableStream({
399
+ write(chunk) {
400
+ controller.enqueue(chunk);
401
+ },
402
+ close() {
403
+ // the generator stream controller is automatically closed when it's consumed
404
+ },
405
+ }),
406
+ ),
407
+ toolResultsStream.pipeTo(
408
+ new WritableStream({
409
+ write(chunk) {
410
+ controller.enqueue(chunk);
411
+ },
412
+ close() {
413
+ controller.close();
414
+ },
415
+ }),
416
+ ),
417
+ ]);
418
+ },
419
+ });
420
+ }