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,713 @@
1
+ import { FlexibleSchema, validateTypes } from '@ai-sdk/provider-utils';
2
+ import { ProviderMetadata } from '../types';
3
+ import { FinishReason } from '../types/language-model';
4
+ import {
5
+ DataUIMessageChunk,
6
+ InferUIMessageChunk,
7
+ isDataUIMessageChunk,
8
+ UIMessageChunk,
9
+ } from '../ui-message-stream/ui-message-chunks';
10
+ import { ErrorHandler } from '../util/error-handler';
11
+ import { mergeObjects } from '../util/merge-objects';
12
+ import { parsePartialJson } from '../util/parse-partial-json';
13
+ import { UIDataTypesToSchemas } from './chat';
14
+ import {
15
+ DataUIPart,
16
+ DynamicToolUIPart,
17
+ getStaticToolName,
18
+ InferUIMessageData,
19
+ InferUIMessageMetadata,
20
+ InferUIMessageToolCall,
21
+ InferUIMessageTools,
22
+ isStaticToolUIPart,
23
+ isToolUIPart,
24
+ ReasoningUIPart,
25
+ TextUIPart,
26
+ ToolUIPart,
27
+ UIMessage,
28
+ UIMessagePart,
29
+ } from './ui-messages';
30
+
31
+ export type StreamingUIMessageState<UI_MESSAGE extends UIMessage> = {
32
+ message: UI_MESSAGE;
33
+ activeTextParts: Record<string, TextUIPart>;
34
+ activeReasoningParts: Record<string, ReasoningUIPart>;
35
+ partialToolCalls: Record<
36
+ string,
37
+ {
38
+ text: string;
39
+ index: number;
40
+ toolName: string;
41
+ dynamic?: boolean;
42
+ title?: string;
43
+ }
44
+ >;
45
+ finishReason?: FinishReason;
46
+ };
47
+
48
+ export function createStreamingUIMessageState<UI_MESSAGE extends UIMessage>({
49
+ lastMessage,
50
+ messageId,
51
+ }: {
52
+ lastMessage: UI_MESSAGE | undefined;
53
+ messageId: string;
54
+ }): StreamingUIMessageState<UI_MESSAGE> {
55
+ return {
56
+ message:
57
+ lastMessage?.role === 'assistant'
58
+ ? lastMessage
59
+ : ({
60
+ id: messageId,
61
+ metadata: undefined,
62
+ role: 'assistant',
63
+ parts: [] as UIMessagePart<
64
+ InferUIMessageData<UI_MESSAGE>,
65
+ InferUIMessageTools<UI_MESSAGE>
66
+ >[],
67
+ } as UI_MESSAGE),
68
+ activeTextParts: {},
69
+ activeReasoningParts: {},
70
+ partialToolCalls: {},
71
+ };
72
+ }
73
+
74
+ export function processUIMessageStream<UI_MESSAGE extends UIMessage>({
75
+ stream,
76
+ messageMetadataSchema,
77
+ dataPartSchemas,
78
+ runUpdateMessageJob,
79
+ onError,
80
+ onToolCall,
81
+ onData,
82
+ }: {
83
+ // input stream is not fully typed yet:
84
+ stream: ReadableStream<UIMessageChunk>;
85
+ messageMetadataSchema?: FlexibleSchema<InferUIMessageMetadata<UI_MESSAGE>>;
86
+ dataPartSchemas?: UIDataTypesToSchemas<InferUIMessageData<UI_MESSAGE>>;
87
+ onToolCall?: (options: {
88
+ toolCall: InferUIMessageToolCall<UI_MESSAGE>;
89
+ }) => void | PromiseLike<void>;
90
+ onData?: (dataPart: DataUIPart<InferUIMessageData<UI_MESSAGE>>) => void;
91
+ runUpdateMessageJob: (
92
+ job: (options: {
93
+ state: StreamingUIMessageState<UI_MESSAGE>;
94
+ write: () => void;
95
+ }) => Promise<void>,
96
+ ) => Promise<void>;
97
+ onError: ErrorHandler;
98
+ }): ReadableStream<InferUIMessageChunk<UI_MESSAGE>> {
99
+ return stream.pipeThrough(
100
+ new TransformStream<UIMessageChunk, InferUIMessageChunk<UI_MESSAGE>>({
101
+ async transform(chunk, controller) {
102
+ await runUpdateMessageJob(async ({ state, write }) => {
103
+ function getToolInvocation(toolCallId: string) {
104
+ const toolInvocations = state.message.parts.filter(isToolUIPart);
105
+
106
+ const toolInvocation = toolInvocations.find(
107
+ invocation => invocation.toolCallId === toolCallId,
108
+ );
109
+
110
+ if (toolInvocation == null) {
111
+ throw new Error(
112
+ `no tool invocation found for tool call ${toolCallId}`,
113
+ );
114
+ }
115
+
116
+ return toolInvocation;
117
+ }
118
+
119
+ function updateToolPart(
120
+ options: {
121
+ toolName: keyof InferUIMessageTools<UI_MESSAGE> & string;
122
+ toolCallId: string;
123
+ providerExecuted?: boolean;
124
+ title?: string;
125
+ } & (
126
+ | {
127
+ state: 'input-streaming';
128
+ input: unknown;
129
+ providerExecuted?: boolean;
130
+ }
131
+ | {
132
+ state: 'input-available';
133
+ input: unknown;
134
+ providerExecuted?: boolean;
135
+ providerMetadata?: ProviderMetadata;
136
+ }
137
+ | {
138
+ state: 'output-available';
139
+ input: unknown;
140
+ output: unknown;
141
+ providerExecuted?: boolean;
142
+ preliminary?: boolean;
143
+ }
144
+ | {
145
+ state: 'output-error';
146
+ input: unknown;
147
+ rawInput?: unknown;
148
+ errorText: string;
149
+ providerExecuted?: boolean;
150
+ providerMetadata?: ProviderMetadata;
151
+ }
152
+ ),
153
+ ) {
154
+ const part = state.message.parts.find(
155
+ part =>
156
+ isStaticToolUIPart(part) &&
157
+ part.toolCallId === options.toolCallId,
158
+ ) as ToolUIPart<InferUIMessageTools<UI_MESSAGE>> | undefined;
159
+
160
+ const anyOptions = options as any;
161
+ const anyPart = part as any;
162
+
163
+ if (part != null) {
164
+ part.state = options.state;
165
+ anyPart.input = anyOptions.input;
166
+ anyPart.output = anyOptions.output;
167
+ anyPart.errorText = anyOptions.errorText;
168
+ anyPart.rawInput = anyOptions.rawInput;
169
+ anyPart.preliminary = anyOptions.preliminary;
170
+ if (options.title !== undefined) {
171
+ anyPart.title = options.title;
172
+ }
173
+ // once providerExecuted is set, it stays for streaming
174
+ anyPart.providerExecuted =
175
+ anyOptions.providerExecuted ?? part.providerExecuted;
176
+
177
+ if (
178
+ anyOptions.providerMetadata != null &&
179
+ part.state === 'input-available'
180
+ ) {
181
+ part.callProviderMetadata = anyOptions.providerMetadata;
182
+ }
183
+ } else {
184
+ state.message.parts.push({
185
+ type: `tool-${options.toolName}`,
186
+ toolCallId: options.toolCallId,
187
+ state: options.state,
188
+ title: options.title,
189
+ input: anyOptions.input,
190
+ output: anyOptions.output,
191
+ rawInput: anyOptions.rawInput,
192
+ errorText: anyOptions.errorText,
193
+ providerExecuted: anyOptions.providerExecuted,
194
+ preliminary: anyOptions.preliminary,
195
+ ...(anyOptions.providerMetadata != null
196
+ ? { callProviderMetadata: anyOptions.providerMetadata }
197
+ : {}),
198
+ } as ToolUIPart<InferUIMessageTools<UI_MESSAGE>>);
199
+ }
200
+ }
201
+
202
+ function updateDynamicToolPart(
203
+ options: {
204
+ toolName: keyof InferUIMessageTools<UI_MESSAGE> & string;
205
+ toolCallId: string;
206
+ providerExecuted?: boolean;
207
+ title?: string;
208
+ } & (
209
+ | {
210
+ state: 'input-streaming';
211
+ input: unknown;
212
+ }
213
+ | {
214
+ state: 'input-available';
215
+ input: unknown;
216
+ providerMetadata?: ProviderMetadata;
217
+ }
218
+ | {
219
+ state: 'output-available';
220
+ input: unknown;
221
+ output: unknown;
222
+ preliminary: boolean | undefined;
223
+ }
224
+ | {
225
+ state: 'output-error';
226
+ input: unknown;
227
+ errorText: string;
228
+ providerMetadata?: ProviderMetadata;
229
+ }
230
+ ),
231
+ ) {
232
+ const part = state.message.parts.find(
233
+ part =>
234
+ part.type === 'dynamic-tool' &&
235
+ part.toolCallId === options.toolCallId,
236
+ ) as DynamicToolUIPart | undefined;
237
+
238
+ const anyOptions = options as any;
239
+ const anyPart = part as any;
240
+
241
+ if (part != null) {
242
+ part.state = options.state;
243
+ anyPart.toolName = options.toolName;
244
+ anyPart.input = anyOptions.input;
245
+ anyPart.output = anyOptions.output;
246
+ anyPart.errorText = anyOptions.errorText;
247
+ anyPart.rawInput = anyOptions.rawInput ?? anyPart.rawInput;
248
+ anyPart.preliminary = anyOptions.preliminary;
249
+ if (options.title !== undefined) {
250
+ anyPart.title = options.title;
251
+ }
252
+ // once providerExecuted is set, it stays for streaming
253
+ anyPart.providerExecuted =
254
+ anyOptions.providerExecuted ?? part.providerExecuted;
255
+
256
+ if (
257
+ anyOptions.providerMetadata != null &&
258
+ part.state === 'input-available'
259
+ ) {
260
+ part.callProviderMetadata = anyOptions.providerMetadata;
261
+ }
262
+ } else {
263
+ state.message.parts.push({
264
+ type: 'dynamic-tool',
265
+ toolName: options.toolName,
266
+ toolCallId: options.toolCallId,
267
+ state: options.state,
268
+ input: anyOptions.input,
269
+ output: anyOptions.output,
270
+ errorText: anyOptions.errorText,
271
+ preliminary: anyOptions.preliminary,
272
+ providerExecuted: anyOptions.providerExecuted,
273
+ title: options.title,
274
+ ...(anyOptions.providerMetadata != null
275
+ ? { callProviderMetadata: anyOptions.providerMetadata }
276
+ : {}),
277
+ } as DynamicToolUIPart);
278
+ }
279
+ }
280
+
281
+ async function updateMessageMetadata(metadata: unknown) {
282
+ if (metadata != null) {
283
+ const mergedMetadata =
284
+ state.message.metadata != null
285
+ ? mergeObjects(state.message.metadata, metadata)
286
+ : metadata;
287
+
288
+ if (messageMetadataSchema != null) {
289
+ await validateTypes({
290
+ value: mergedMetadata,
291
+ schema: messageMetadataSchema,
292
+ });
293
+ }
294
+
295
+ state.message.metadata =
296
+ mergedMetadata as InferUIMessageMetadata<UI_MESSAGE>;
297
+ }
298
+ }
299
+
300
+ switch (chunk.type) {
301
+ case 'text-start': {
302
+ const textPart: TextUIPart = {
303
+ type: 'text',
304
+ text: '',
305
+ providerMetadata: chunk.providerMetadata,
306
+ state: 'streaming',
307
+ };
308
+ state.activeTextParts[chunk.id] = textPart;
309
+ state.message.parts.push(textPart);
310
+ write();
311
+ break;
312
+ }
313
+
314
+ case 'text-delta': {
315
+ const textPart = state.activeTextParts[chunk.id];
316
+ textPart.text += chunk.delta;
317
+ textPart.providerMetadata =
318
+ chunk.providerMetadata ?? textPart.providerMetadata;
319
+ write();
320
+ break;
321
+ }
322
+
323
+ case 'text-end': {
324
+ const textPart = state.activeTextParts[chunk.id];
325
+ textPart.state = 'done';
326
+ textPart.providerMetadata =
327
+ chunk.providerMetadata ?? textPart.providerMetadata;
328
+ delete state.activeTextParts[chunk.id];
329
+ write();
330
+ break;
331
+ }
332
+
333
+ case 'reasoning-start': {
334
+ const reasoningPart: ReasoningUIPart = {
335
+ type: 'reasoning',
336
+ text: '',
337
+ providerMetadata: chunk.providerMetadata,
338
+ state: 'streaming',
339
+ };
340
+ state.activeReasoningParts[chunk.id] = reasoningPart;
341
+ state.message.parts.push(reasoningPart);
342
+ write();
343
+ break;
344
+ }
345
+
346
+ case 'reasoning-delta': {
347
+ const reasoningPart = state.activeReasoningParts[chunk.id];
348
+ reasoningPart.text += chunk.delta;
349
+ reasoningPart.providerMetadata =
350
+ chunk.providerMetadata ?? reasoningPart.providerMetadata;
351
+ write();
352
+ break;
353
+ }
354
+
355
+ case 'reasoning-end': {
356
+ const reasoningPart = state.activeReasoningParts[chunk.id];
357
+ reasoningPart.providerMetadata =
358
+ chunk.providerMetadata ?? reasoningPart.providerMetadata;
359
+ reasoningPart.state = 'done';
360
+ delete state.activeReasoningParts[chunk.id];
361
+
362
+ write();
363
+ break;
364
+ }
365
+
366
+ case 'file': {
367
+ state.message.parts.push({
368
+ type: 'file',
369
+ mediaType: chunk.mediaType,
370
+ url: chunk.url,
371
+ });
372
+
373
+ write();
374
+ break;
375
+ }
376
+
377
+ case 'source-url': {
378
+ state.message.parts.push({
379
+ type: 'source-url',
380
+ sourceId: chunk.sourceId,
381
+ url: chunk.url,
382
+ title: chunk.title,
383
+ providerMetadata: chunk.providerMetadata,
384
+ });
385
+
386
+ write();
387
+ break;
388
+ }
389
+
390
+ case 'source-document': {
391
+ state.message.parts.push({
392
+ type: 'source-document',
393
+ sourceId: chunk.sourceId,
394
+ mediaType: chunk.mediaType,
395
+ title: chunk.title,
396
+ filename: chunk.filename,
397
+ providerMetadata: chunk.providerMetadata,
398
+ });
399
+
400
+ write();
401
+ break;
402
+ }
403
+
404
+ case 'tool-input-start': {
405
+ const toolInvocations =
406
+ state.message.parts.filter(isStaticToolUIPart);
407
+
408
+ // add the partial tool call to the map
409
+ state.partialToolCalls[chunk.toolCallId] = {
410
+ text: '',
411
+ toolName: chunk.toolName,
412
+ index: toolInvocations.length,
413
+ dynamic: chunk.dynamic,
414
+ title: chunk.title,
415
+ };
416
+
417
+ if (chunk.dynamic) {
418
+ updateDynamicToolPart({
419
+ toolCallId: chunk.toolCallId,
420
+ toolName: chunk.toolName,
421
+ state: 'input-streaming',
422
+ input: undefined,
423
+ providerExecuted: chunk.providerExecuted,
424
+ title: chunk.title,
425
+ });
426
+ } else {
427
+ updateToolPart({
428
+ toolCallId: chunk.toolCallId,
429
+ toolName: chunk.toolName,
430
+ state: 'input-streaming',
431
+ input: undefined,
432
+ providerExecuted: chunk.providerExecuted,
433
+ title: chunk.title,
434
+ });
435
+ }
436
+
437
+ write();
438
+ break;
439
+ }
440
+
441
+ case 'tool-input-delta': {
442
+ const partialToolCall = state.partialToolCalls[chunk.toolCallId];
443
+
444
+ partialToolCall.text += chunk.inputTextDelta;
445
+
446
+ const { value: partialArgs } = await parsePartialJson(
447
+ partialToolCall.text,
448
+ );
449
+
450
+ if (partialToolCall.dynamic) {
451
+ updateDynamicToolPart({
452
+ toolCallId: chunk.toolCallId,
453
+ toolName: partialToolCall.toolName,
454
+ state: 'input-streaming',
455
+ input: partialArgs,
456
+ title: partialToolCall.title,
457
+ });
458
+ } else {
459
+ updateToolPart({
460
+ toolCallId: chunk.toolCallId,
461
+ toolName: partialToolCall.toolName,
462
+ state: 'input-streaming',
463
+ input: partialArgs,
464
+ title: partialToolCall.title,
465
+ });
466
+ }
467
+
468
+ write();
469
+ break;
470
+ }
471
+
472
+ case 'tool-input-available': {
473
+ if (chunk.dynamic) {
474
+ updateDynamicToolPart({
475
+ toolCallId: chunk.toolCallId,
476
+ toolName: chunk.toolName,
477
+ state: 'input-available',
478
+ input: chunk.input,
479
+ providerExecuted: chunk.providerExecuted,
480
+ providerMetadata: chunk.providerMetadata,
481
+ title: chunk.title,
482
+ });
483
+ } else {
484
+ updateToolPart({
485
+ toolCallId: chunk.toolCallId,
486
+ toolName: chunk.toolName,
487
+ state: 'input-available',
488
+ input: chunk.input,
489
+ providerExecuted: chunk.providerExecuted,
490
+ providerMetadata: chunk.providerMetadata,
491
+ title: chunk.title,
492
+ });
493
+ }
494
+
495
+ write();
496
+
497
+ // invoke the onToolCall callback if it exists. This is blocking.
498
+ // In the future we should make this non-blocking, which
499
+ // requires additional state management for error handling etc.
500
+ // Skip calling onToolCall for provider-executed tools since they are already executed
501
+ if (onToolCall && !chunk.providerExecuted) {
502
+ await onToolCall({
503
+ toolCall: chunk as InferUIMessageToolCall<UI_MESSAGE>,
504
+ });
505
+ }
506
+ break;
507
+ }
508
+
509
+ case 'tool-input-error': {
510
+ if (chunk.dynamic) {
511
+ updateDynamicToolPart({
512
+ toolCallId: chunk.toolCallId,
513
+ toolName: chunk.toolName,
514
+ state: 'output-error',
515
+ input: chunk.input,
516
+ errorText: chunk.errorText,
517
+ providerExecuted: chunk.providerExecuted,
518
+ providerMetadata: chunk.providerMetadata,
519
+ });
520
+ } else {
521
+ updateToolPart({
522
+ toolCallId: chunk.toolCallId,
523
+ toolName: chunk.toolName,
524
+ state: 'output-error',
525
+ input: undefined,
526
+ rawInput: chunk.input,
527
+ errorText: chunk.errorText,
528
+ providerExecuted: chunk.providerExecuted,
529
+ providerMetadata: chunk.providerMetadata,
530
+ });
531
+ }
532
+
533
+ write();
534
+ break;
535
+ }
536
+
537
+ case 'tool-approval-request': {
538
+ const toolInvocation = getToolInvocation(chunk.toolCallId);
539
+ toolInvocation.state = 'approval-requested';
540
+ toolInvocation.approval = { id: chunk.approvalId };
541
+ write();
542
+ break;
543
+ }
544
+
545
+ case 'tool-output-denied': {
546
+ const toolInvocation = getToolInvocation(chunk.toolCallId);
547
+ toolInvocation.state = 'output-denied';
548
+ write();
549
+ break;
550
+ }
551
+
552
+ case 'tool-output-available': {
553
+ const toolInvocation = getToolInvocation(chunk.toolCallId);
554
+
555
+ if (toolInvocation.type === 'dynamic-tool') {
556
+ updateDynamicToolPart({
557
+ toolCallId: chunk.toolCallId,
558
+ toolName: toolInvocation.toolName,
559
+ state: 'output-available',
560
+ input: (toolInvocation as any).input,
561
+ output: chunk.output,
562
+ preliminary: chunk.preliminary,
563
+ providerExecuted: chunk.providerExecuted,
564
+ title: toolInvocation.title,
565
+ });
566
+ } else {
567
+ updateToolPart({
568
+ toolCallId: chunk.toolCallId,
569
+ toolName: getStaticToolName(toolInvocation),
570
+ state: 'output-available',
571
+ input: (toolInvocation as any).input,
572
+ output: chunk.output,
573
+ providerExecuted: chunk.providerExecuted,
574
+ preliminary: chunk.preliminary,
575
+ title: toolInvocation.title,
576
+ });
577
+ }
578
+
579
+ write();
580
+ break;
581
+ }
582
+
583
+ case 'tool-output-error': {
584
+ const toolInvocation = getToolInvocation(chunk.toolCallId);
585
+
586
+ if (toolInvocation.type === 'dynamic-tool') {
587
+ updateDynamicToolPart({
588
+ toolCallId: chunk.toolCallId,
589
+ toolName: toolInvocation.toolName,
590
+ state: 'output-error',
591
+ input: (toolInvocation as any).input,
592
+ errorText: chunk.errorText,
593
+ providerExecuted: chunk.providerExecuted,
594
+ title: toolInvocation.title,
595
+ });
596
+ } else {
597
+ updateToolPart({
598
+ toolCallId: chunk.toolCallId,
599
+ toolName: getStaticToolName(toolInvocation),
600
+ state: 'output-error',
601
+ input: (toolInvocation as any).input,
602
+ rawInput: (toolInvocation as any).rawInput,
603
+ errorText: chunk.errorText,
604
+ providerExecuted: chunk.providerExecuted,
605
+ title: toolInvocation.title,
606
+ });
607
+ }
608
+
609
+ write();
610
+ break;
611
+ }
612
+
613
+ case 'start-step': {
614
+ // add a step boundary part to the message
615
+ state.message.parts.push({ type: 'step-start' });
616
+ break;
617
+ }
618
+
619
+ case 'finish-step': {
620
+ // reset the current text and reasoning parts
621
+ state.activeTextParts = {};
622
+ state.activeReasoningParts = {};
623
+ break;
624
+ }
625
+
626
+ case 'start': {
627
+ if (chunk.messageId != null) {
628
+ state.message.id = chunk.messageId;
629
+ }
630
+
631
+ await updateMessageMetadata(chunk.messageMetadata);
632
+
633
+ if (chunk.messageId != null || chunk.messageMetadata != null) {
634
+ write();
635
+ }
636
+ break;
637
+ }
638
+
639
+ case 'finish': {
640
+ if (chunk.finishReason != null) {
641
+ state.finishReason = chunk.finishReason;
642
+ }
643
+ await updateMessageMetadata(chunk.messageMetadata);
644
+ if (chunk.messageMetadata != null) {
645
+ write();
646
+ }
647
+ break;
648
+ }
649
+
650
+ case 'message-metadata': {
651
+ await updateMessageMetadata(chunk.messageMetadata);
652
+ if (chunk.messageMetadata != null) {
653
+ write();
654
+ }
655
+ break;
656
+ }
657
+
658
+ case 'error': {
659
+ onError?.(new Error(chunk.errorText));
660
+ break;
661
+ }
662
+
663
+ default: {
664
+ if (isDataUIMessageChunk(chunk)) {
665
+ // validate data chunk if dataPartSchemas is provided
666
+ if (dataPartSchemas?.[chunk.type] != null) {
667
+ await validateTypes({
668
+ value: chunk.data,
669
+ schema: dataPartSchemas[chunk.type],
670
+ });
671
+ }
672
+
673
+ // cast, validation is done above
674
+ const dataChunk = chunk as DataUIMessageChunk<
675
+ InferUIMessageData<UI_MESSAGE>
676
+ >;
677
+
678
+ // transient parts are not added to the message state
679
+ if (dataChunk.transient) {
680
+ onData?.(dataChunk);
681
+ break;
682
+ }
683
+
684
+ const existingUIPart =
685
+ dataChunk.id != null
686
+ ? (state.message.parts.find(
687
+ chunkArg =>
688
+ dataChunk.type === chunkArg.type &&
689
+ dataChunk.id === chunkArg.id,
690
+ ) as
691
+ | DataUIPart<InferUIMessageData<UI_MESSAGE>>
692
+ | undefined)
693
+ : undefined;
694
+
695
+ if (existingUIPart != null) {
696
+ existingUIPart.data = dataChunk.data;
697
+ } else {
698
+ state.message.parts.push(dataChunk);
699
+ }
700
+
701
+ onData?.(dataChunk);
702
+
703
+ write();
704
+ }
705
+ }
706
+ }
707
+
708
+ controller.enqueue(chunk as InferUIMessageChunk<UI_MESSAGE>);
709
+ });
710
+ },
711
+ }),
712
+ );
713
+ }