ai 6.0.33 → 6.0.35

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 (357) hide show
  1. package/CHANGELOG.md +16 -0
  2. package/dist/index.d.mts +50 -21
  3. package/dist/index.d.ts +50 -21
  4. package/dist/index.js +348 -286
  5. package/dist/index.js.map +1 -1
  6. package/dist/index.mjs +280 -219
  7. package/dist/index.mjs.map +1 -1
  8. package/dist/internal/index.js +1 -1
  9. package/dist/internal/index.mjs +1 -1
  10. package/docs/02-foundations/03-prompts.mdx +2 -2
  11. package/docs/03-ai-sdk-core/15-tools-and-tool-calling.mdx +1 -1
  12. package/docs/07-reference/01-ai-sdk-core/28-output.mdx +1 -1
  13. package/docs/07-reference/05-ai-sdk-errors/ai-ui-message-stream-error.mdx +67 -0
  14. package/package.json +6 -4
  15. package/src/agent/agent.ts +116 -0
  16. package/src/agent/create-agent-ui-stream-response.test.ts +258 -0
  17. package/src/agent/create-agent-ui-stream-response.ts +50 -0
  18. package/src/agent/create-agent-ui-stream.ts +73 -0
  19. package/src/agent/index.ts +33 -0
  20. package/src/agent/infer-agent-tools.ts +7 -0
  21. package/src/agent/infer-agent-ui-message.test-d.ts +54 -0
  22. package/src/agent/infer-agent-ui-message.ts +11 -0
  23. package/src/agent/pipe-agent-ui-stream-to-response.ts +52 -0
  24. package/src/agent/tool-loop-agent-on-finish-callback.ts +31 -0
  25. package/src/agent/tool-loop-agent-on-step-finish-callback.ts +11 -0
  26. package/src/agent/tool-loop-agent-settings.ts +182 -0
  27. package/src/agent/tool-loop-agent.test-d.ts +114 -0
  28. package/src/agent/tool-loop-agent.test.ts +442 -0
  29. package/src/agent/tool-loop-agent.ts +114 -0
  30. package/src/embed/__snapshots__/embed-many.test.ts.snap +191 -0
  31. package/src/embed/__snapshots__/embed.test.ts.snap +81 -0
  32. package/src/embed/embed-many-result.ts +53 -0
  33. package/src/embed/embed-many.test.ts +653 -0
  34. package/src/embed/embed-many.ts +378 -0
  35. package/src/embed/embed-result.ts +50 -0
  36. package/src/embed/embed.test.ts +298 -0
  37. package/src/embed/embed.ts +211 -0
  38. package/src/embed/index.ts +4 -0
  39. package/src/error/index.ts +35 -0
  40. package/src/error/invalid-argument-error.ts +34 -0
  41. package/src/error/invalid-stream-part-error.ts +28 -0
  42. package/src/error/invalid-tool-approval-error.ts +26 -0
  43. package/src/error/invalid-tool-input-error.ts +33 -0
  44. package/src/error/no-image-generated-error.ts +39 -0
  45. package/src/error/no-object-generated-error.ts +70 -0
  46. package/src/error/no-output-generated-error.ts +26 -0
  47. package/src/error/no-speech-generated-error.ts +18 -0
  48. package/src/error/no-such-tool-error.ts +35 -0
  49. package/src/error/no-transcript-generated-error.ts +20 -0
  50. package/src/error/tool-call-not-found-for-approval-error.ts +32 -0
  51. package/src/error/tool-call-repair-error.ts +30 -0
  52. package/src/error/ui-message-stream-error.ts +48 -0
  53. package/src/error/unsupported-model-version-error.ts +23 -0
  54. package/src/error/verify-no-object-generated-error.ts +27 -0
  55. package/src/generate-image/generate-image-result.ts +42 -0
  56. package/src/generate-image/generate-image.test.ts +1420 -0
  57. package/src/generate-image/generate-image.ts +360 -0
  58. package/src/generate-image/index.ts +18 -0
  59. package/src/generate-object/__snapshots__/generate-object.test.ts.snap +133 -0
  60. package/src/generate-object/__snapshots__/stream-object.test.ts.snap +297 -0
  61. package/src/generate-object/generate-object-result.ts +67 -0
  62. package/src/generate-object/generate-object.test-d.ts +49 -0
  63. package/src/generate-object/generate-object.test.ts +1191 -0
  64. package/src/generate-object/generate-object.ts +518 -0
  65. package/src/generate-object/index.ts +9 -0
  66. package/src/generate-object/inject-json-instruction.test.ts +181 -0
  67. package/src/generate-object/inject-json-instruction.ts +30 -0
  68. package/src/generate-object/output-strategy.ts +415 -0
  69. package/src/generate-object/parse-and-validate-object-result.ts +111 -0
  70. package/src/generate-object/repair-text.ts +12 -0
  71. package/src/generate-object/stream-object-result.ts +120 -0
  72. package/src/generate-object/stream-object.test-d.ts +74 -0
  73. package/src/generate-object/stream-object.test.ts +1950 -0
  74. package/src/generate-object/stream-object.ts +986 -0
  75. package/src/generate-object/validate-object-generation-input.ts +144 -0
  76. package/src/generate-speech/generate-speech-result.ts +30 -0
  77. package/src/generate-speech/generate-speech.test.ts +300 -0
  78. package/src/generate-speech/generate-speech.ts +190 -0
  79. package/src/generate-speech/generated-audio-file.ts +65 -0
  80. package/src/generate-speech/index.ts +3 -0
  81. package/src/generate-text/__snapshots__/generate-text.test.ts.snap +1872 -0
  82. package/src/generate-text/__snapshots__/stream-text.test.ts.snap +1255 -0
  83. package/src/generate-text/collect-tool-approvals.test.ts +553 -0
  84. package/src/generate-text/collect-tool-approvals.ts +116 -0
  85. package/src/generate-text/content-part.ts +25 -0
  86. package/src/generate-text/execute-tool-call.ts +129 -0
  87. package/src/generate-text/extract-reasoning-content.ts +17 -0
  88. package/src/generate-text/extract-text-content.ts +15 -0
  89. package/src/generate-text/generate-text-result.ts +168 -0
  90. package/src/generate-text/generate-text.test-d.ts +68 -0
  91. package/src/generate-text/generate-text.test.ts +7011 -0
  92. package/src/generate-text/generate-text.ts +1223 -0
  93. package/src/generate-text/generated-file.ts +70 -0
  94. package/src/generate-text/index.ts +57 -0
  95. package/src/generate-text/is-approval-needed.ts +29 -0
  96. package/src/generate-text/output-utils.ts +23 -0
  97. package/src/generate-text/output.test.ts +698 -0
  98. package/src/generate-text/output.ts +590 -0
  99. package/src/generate-text/parse-tool-call.test.ts +570 -0
  100. package/src/generate-text/parse-tool-call.ts +188 -0
  101. package/src/generate-text/prepare-step.ts +103 -0
  102. package/src/generate-text/prune-messages.test.ts +720 -0
  103. package/src/generate-text/prune-messages.ts +167 -0
  104. package/src/generate-text/reasoning-output.ts +20 -0
  105. package/src/generate-text/reasoning.ts +8 -0
  106. package/src/generate-text/response-message.ts +10 -0
  107. package/src/generate-text/run-tools-transformation.test.ts +1143 -0
  108. package/src/generate-text/run-tools-transformation.ts +420 -0
  109. package/src/generate-text/smooth-stream.test.ts +2101 -0
  110. package/src/generate-text/smooth-stream.ts +162 -0
  111. package/src/generate-text/step-result.ts +238 -0
  112. package/src/generate-text/stop-condition.ts +29 -0
  113. package/src/generate-text/stream-text-result.ts +463 -0
  114. package/src/generate-text/stream-text.test-d.ts +200 -0
  115. package/src/generate-text/stream-text.test.ts +19979 -0
  116. package/src/generate-text/stream-text.ts +2505 -0
  117. package/src/generate-text/to-response-messages.test.ts +922 -0
  118. package/src/generate-text/to-response-messages.ts +163 -0
  119. package/src/generate-text/tool-approval-request-output.ts +21 -0
  120. package/src/generate-text/tool-call-repair-function.ts +27 -0
  121. package/src/generate-text/tool-call.ts +47 -0
  122. package/src/generate-text/tool-error.ts +34 -0
  123. package/src/generate-text/tool-output-denied.ts +21 -0
  124. package/src/generate-text/tool-output.ts +7 -0
  125. package/src/generate-text/tool-result.ts +36 -0
  126. package/src/generate-text/tool-set.ts +14 -0
  127. package/src/global.ts +24 -0
  128. package/src/index.ts +50 -0
  129. package/src/logger/index.ts +6 -0
  130. package/src/logger/log-warnings.test.ts +351 -0
  131. package/src/logger/log-warnings.ts +119 -0
  132. package/src/middleware/__snapshots__/simulate-streaming-middleware.test.ts.snap +64 -0
  133. package/src/middleware/add-tool-input-examples-middleware.test.ts +476 -0
  134. package/src/middleware/add-tool-input-examples-middleware.ts +90 -0
  135. package/src/middleware/default-embedding-settings-middleware.test.ts +126 -0
  136. package/src/middleware/default-embedding-settings-middleware.ts +22 -0
  137. package/src/middleware/default-settings-middleware.test.ts +388 -0
  138. package/src/middleware/default-settings-middleware.ts +33 -0
  139. package/src/middleware/extract-json-middleware.test.ts +827 -0
  140. package/src/middleware/extract-json-middleware.ts +197 -0
  141. package/src/middleware/extract-reasoning-middleware.test.ts +1028 -0
  142. package/src/middleware/extract-reasoning-middleware.ts +238 -0
  143. package/src/middleware/index.ts +10 -0
  144. package/src/middleware/simulate-streaming-middleware.test.ts +911 -0
  145. package/src/middleware/simulate-streaming-middleware.ts +79 -0
  146. package/src/middleware/wrap-embedding-model.test.ts +358 -0
  147. package/src/middleware/wrap-embedding-model.ts +86 -0
  148. package/src/middleware/wrap-image-model.test.ts +423 -0
  149. package/src/middleware/wrap-image-model.ts +85 -0
  150. package/src/middleware/wrap-language-model.test.ts +518 -0
  151. package/src/middleware/wrap-language-model.ts +104 -0
  152. package/src/middleware/wrap-provider.test.ts +120 -0
  153. package/src/middleware/wrap-provider.ts +51 -0
  154. package/src/model/as-embedding-model-v3.test.ts +319 -0
  155. package/src/model/as-embedding-model-v3.ts +24 -0
  156. package/src/model/as-image-model-v3.test.ts +409 -0
  157. package/src/model/as-image-model-v3.ts +24 -0
  158. package/src/model/as-language-model-v3.test.ts +508 -0
  159. package/src/model/as-language-model-v3.ts +103 -0
  160. package/src/model/as-provider-v3.ts +36 -0
  161. package/src/model/as-speech-model-v3.test.ts +356 -0
  162. package/src/model/as-speech-model-v3.ts +24 -0
  163. package/src/model/as-transcription-model-v3.test.ts +529 -0
  164. package/src/model/as-transcription-model-v3.ts +24 -0
  165. package/src/model/resolve-model.test.ts +244 -0
  166. package/src/model/resolve-model.ts +126 -0
  167. package/src/prompt/call-settings.ts +148 -0
  168. package/src/prompt/content-part.ts +209 -0
  169. package/src/prompt/convert-to-language-model-prompt.test.ts +2018 -0
  170. package/src/prompt/convert-to-language-model-prompt.ts +442 -0
  171. package/src/prompt/create-tool-model-output.test.ts +508 -0
  172. package/src/prompt/create-tool-model-output.ts +34 -0
  173. package/src/prompt/data-content.test.ts +15 -0
  174. package/src/prompt/data-content.ts +134 -0
  175. package/src/prompt/index.ts +27 -0
  176. package/src/prompt/invalid-data-content-error.ts +29 -0
  177. package/src/prompt/invalid-message-role-error.ts +27 -0
  178. package/src/prompt/message-conversion-error.ts +28 -0
  179. package/src/prompt/message.ts +68 -0
  180. package/src/prompt/prepare-call-settings.test.ts +159 -0
  181. package/src/prompt/prepare-call-settings.ts +108 -0
  182. package/src/prompt/prepare-tools-and-tool-choice.test.ts +461 -0
  183. package/src/prompt/prepare-tools-and-tool-choice.ts +86 -0
  184. package/src/prompt/prompt.ts +43 -0
  185. package/src/prompt/split-data-url.ts +17 -0
  186. package/src/prompt/standardize-prompt.test.ts +82 -0
  187. package/src/prompt/standardize-prompt.ts +99 -0
  188. package/src/prompt/wrap-gateway-error.ts +29 -0
  189. package/src/registry/custom-provider.test.ts +211 -0
  190. package/src/registry/custom-provider.ts +155 -0
  191. package/src/registry/index.ts +7 -0
  192. package/src/registry/no-such-provider-error.ts +41 -0
  193. package/src/registry/provider-registry.test.ts +691 -0
  194. package/src/registry/provider-registry.ts +328 -0
  195. package/src/rerank/index.ts +2 -0
  196. package/src/rerank/rerank-result.ts +70 -0
  197. package/src/rerank/rerank.test.ts +516 -0
  198. package/src/rerank/rerank.ts +237 -0
  199. package/src/telemetry/assemble-operation-name.ts +21 -0
  200. package/src/telemetry/get-base-telemetry-attributes.ts +53 -0
  201. package/src/telemetry/get-tracer.ts +20 -0
  202. package/src/telemetry/noop-tracer.ts +69 -0
  203. package/src/telemetry/record-span.ts +63 -0
  204. package/src/telemetry/select-telemetry-attributes.ts +78 -0
  205. package/src/telemetry/select-temetry-attributes.test.ts +114 -0
  206. package/src/telemetry/stringify-for-telemetry.test.ts +114 -0
  207. package/src/telemetry/stringify-for-telemetry.ts +33 -0
  208. package/src/telemetry/telemetry-settings.ts +44 -0
  209. package/src/test/mock-embedding-model-v2.ts +35 -0
  210. package/src/test/mock-embedding-model-v3.ts +48 -0
  211. package/src/test/mock-image-model-v2.ts +28 -0
  212. package/src/test/mock-image-model-v3.ts +28 -0
  213. package/src/test/mock-language-model-v2.ts +72 -0
  214. package/src/test/mock-language-model-v3.ts +77 -0
  215. package/src/test/mock-provider-v2.ts +68 -0
  216. package/src/test/mock-provider-v3.ts +80 -0
  217. package/src/test/mock-reranking-model-v3.ts +25 -0
  218. package/src/test/mock-server-response.ts +69 -0
  219. package/src/test/mock-speech-model-v2.ts +24 -0
  220. package/src/test/mock-speech-model-v3.ts +24 -0
  221. package/src/test/mock-tracer.ts +156 -0
  222. package/src/test/mock-transcription-model-v2.ts +24 -0
  223. package/src/test/mock-transcription-model-v3.ts +24 -0
  224. package/src/test/mock-values.ts +4 -0
  225. package/src/test/not-implemented.ts +3 -0
  226. package/src/text-stream/create-text-stream-response.test.ts +38 -0
  227. package/src/text-stream/create-text-stream-response.ts +18 -0
  228. package/src/text-stream/index.ts +2 -0
  229. package/src/text-stream/pipe-text-stream-to-response.test.ts +38 -0
  230. package/src/text-stream/pipe-text-stream-to-response.ts +26 -0
  231. package/src/transcribe/index.ts +2 -0
  232. package/src/transcribe/transcribe-result.ts +60 -0
  233. package/src/transcribe/transcribe.test.ts +313 -0
  234. package/src/transcribe/transcribe.ts +173 -0
  235. package/src/types/embedding-model-middleware.ts +3 -0
  236. package/src/types/embedding-model.ts +18 -0
  237. package/src/types/image-model-middleware.ts +3 -0
  238. package/src/types/image-model-response-metadata.ts +16 -0
  239. package/src/types/image-model.ts +19 -0
  240. package/src/types/index.ts +29 -0
  241. package/src/types/json-value.ts +15 -0
  242. package/src/types/language-model-middleware.ts +3 -0
  243. package/src/types/language-model-request-metadata.ts +6 -0
  244. package/src/types/language-model-response-metadata.ts +21 -0
  245. package/src/types/language-model.ts +104 -0
  246. package/src/types/provider-metadata.ts +16 -0
  247. package/src/types/provider.ts +55 -0
  248. package/src/types/reranking-model.ts +6 -0
  249. package/src/types/speech-model-response-metadata.ts +21 -0
  250. package/src/types/speech-model.ts +6 -0
  251. package/src/types/transcription-model-response-metadata.ts +16 -0
  252. package/src/types/transcription-model.ts +9 -0
  253. package/src/types/usage.ts +200 -0
  254. package/src/types/warning.ts +7 -0
  255. package/src/ui/__snapshots__/append-response-messages.test.ts.snap +416 -0
  256. package/src/ui/__snapshots__/convert-to-model-messages.test.ts.snap +419 -0
  257. package/src/ui/__snapshots__/process-chat-text-response.test.ts.snap +142 -0
  258. package/src/ui/call-completion-api.ts +157 -0
  259. package/src/ui/chat-transport.ts +83 -0
  260. package/src/ui/chat.test-d.ts +233 -0
  261. package/src/ui/chat.test.ts +2695 -0
  262. package/src/ui/chat.ts +716 -0
  263. package/src/ui/convert-file-list-to-file-ui-parts.ts +36 -0
  264. package/src/ui/convert-to-model-messages.test.ts +2775 -0
  265. package/src/ui/convert-to-model-messages.ts +373 -0
  266. package/src/ui/default-chat-transport.ts +36 -0
  267. package/src/ui/direct-chat-transport.test.ts +446 -0
  268. package/src/ui/direct-chat-transport.ts +118 -0
  269. package/src/ui/http-chat-transport.test.ts +185 -0
  270. package/src/ui/http-chat-transport.ts +292 -0
  271. package/src/ui/index.ts +71 -0
  272. package/src/ui/last-assistant-message-is-complete-with-approval-responses.ts +44 -0
  273. package/src/ui/last-assistant-message-is-complete-with-tool-calls.test.ts +371 -0
  274. package/src/ui/last-assistant-message-is-complete-with-tool-calls.ts +39 -0
  275. package/src/ui/process-text-stream.test.ts +38 -0
  276. package/src/ui/process-text-stream.ts +16 -0
  277. package/src/ui/process-ui-message-stream.test.ts +8294 -0
  278. package/src/ui/process-ui-message-stream.ts +761 -0
  279. package/src/ui/text-stream-chat-transport.ts +23 -0
  280. package/src/ui/transform-text-to-ui-message-stream.test.ts +124 -0
  281. package/src/ui/transform-text-to-ui-message-stream.ts +27 -0
  282. package/src/ui/ui-messages.test.ts +48 -0
  283. package/src/ui/ui-messages.ts +534 -0
  284. package/src/ui/use-completion.ts +84 -0
  285. package/src/ui/validate-ui-messages.test.ts +1428 -0
  286. package/src/ui/validate-ui-messages.ts +476 -0
  287. package/src/ui-message-stream/create-ui-message-stream-response.test.ts +266 -0
  288. package/src/ui-message-stream/create-ui-message-stream-response.ts +32 -0
  289. package/src/ui-message-stream/create-ui-message-stream.test.ts +639 -0
  290. package/src/ui-message-stream/create-ui-message-stream.ts +124 -0
  291. package/src/ui-message-stream/get-response-ui-message-id.test.ts +55 -0
  292. package/src/ui-message-stream/get-response-ui-message-id.ts +24 -0
  293. package/src/ui-message-stream/handle-ui-message-stream-finish.test.ts +429 -0
  294. package/src/ui-message-stream/handle-ui-message-stream-finish.ts +135 -0
  295. package/src/ui-message-stream/index.ts +13 -0
  296. package/src/ui-message-stream/json-to-sse-transform-stream.ts +12 -0
  297. package/src/ui-message-stream/pipe-ui-message-stream-to-response.test.ts +90 -0
  298. package/src/ui-message-stream/pipe-ui-message-stream-to-response.ts +40 -0
  299. package/src/ui-message-stream/read-ui-message-stream.test.ts +122 -0
  300. package/src/ui-message-stream/read-ui-message-stream.ts +87 -0
  301. package/src/ui-message-stream/ui-message-chunks.test-d.ts +18 -0
  302. package/src/ui-message-stream/ui-message-chunks.ts +344 -0
  303. package/src/ui-message-stream/ui-message-stream-headers.ts +7 -0
  304. package/src/ui-message-stream/ui-message-stream-on-finish-callback.ts +32 -0
  305. package/src/ui-message-stream/ui-message-stream-response-init.ts +5 -0
  306. package/src/ui-message-stream/ui-message-stream-writer.ts +24 -0
  307. package/src/util/as-array.ts +3 -0
  308. package/src/util/async-iterable-stream.test.ts +241 -0
  309. package/src/util/async-iterable-stream.ts +94 -0
  310. package/src/util/consume-stream.ts +29 -0
  311. package/src/util/cosine-similarity.test.ts +57 -0
  312. package/src/util/cosine-similarity.ts +47 -0
  313. package/src/util/create-resolvable-promise.ts +30 -0
  314. package/src/util/create-stitchable-stream.test.ts +239 -0
  315. package/src/util/create-stitchable-stream.ts +112 -0
  316. package/src/util/data-url.ts +17 -0
  317. package/src/util/deep-partial.ts +84 -0
  318. package/src/util/detect-media-type.test.ts +670 -0
  319. package/src/util/detect-media-type.ts +184 -0
  320. package/src/util/download/download-function.ts +45 -0
  321. package/src/util/download/download.test.ts +69 -0
  322. package/src/util/download/download.ts +46 -0
  323. package/src/util/error-handler.ts +1 -0
  324. package/src/util/fix-json.test.ts +279 -0
  325. package/src/util/fix-json.ts +401 -0
  326. package/src/util/get-potential-start-index.test.ts +34 -0
  327. package/src/util/get-potential-start-index.ts +30 -0
  328. package/src/util/index.ts +11 -0
  329. package/src/util/is-deep-equal-data.test.ts +119 -0
  330. package/src/util/is-deep-equal-data.ts +48 -0
  331. package/src/util/is-non-empty-object.ts +5 -0
  332. package/src/util/job.ts +1 -0
  333. package/src/util/log-v2-compatibility-warning.ts +21 -0
  334. package/src/util/merge-abort-signals.test.ts +155 -0
  335. package/src/util/merge-abort-signals.ts +43 -0
  336. package/src/util/merge-objects.test.ts +118 -0
  337. package/src/util/merge-objects.ts +79 -0
  338. package/src/util/now.ts +4 -0
  339. package/src/util/parse-partial-json.test.ts +80 -0
  340. package/src/util/parse-partial-json.ts +30 -0
  341. package/src/util/prepare-headers.test.ts +51 -0
  342. package/src/util/prepare-headers.ts +14 -0
  343. package/src/util/prepare-retries.test.ts +10 -0
  344. package/src/util/prepare-retries.ts +47 -0
  345. package/src/util/retry-error.ts +41 -0
  346. package/src/util/retry-with-exponential-backoff.test.ts +446 -0
  347. package/src/util/retry-with-exponential-backoff.ts +154 -0
  348. package/src/util/serial-job-executor.test.ts +162 -0
  349. package/src/util/serial-job-executor.ts +36 -0
  350. package/src/util/simulate-readable-stream.test.ts +98 -0
  351. package/src/util/simulate-readable-stream.ts +39 -0
  352. package/src/util/split-array.test.ts +60 -0
  353. package/src/util/split-array.ts +20 -0
  354. package/src/util/value-of.ts +65 -0
  355. package/src/util/write-to-server-response.test.ts +266 -0
  356. package/src/util/write-to-server-response.ts +49 -0
  357. package/src/version.ts +5 -0
@@ -0,0 +1,344 @@
1
+ import { z } from 'zod/v4';
2
+ import {
3
+ ProviderMetadata,
4
+ providerMetadataSchema,
5
+ } from '../types/provider-metadata';
6
+ import { FinishReason } from '../types/language-model';
7
+ import {
8
+ InferUIMessageData,
9
+ InferUIMessageMetadata,
10
+ UIDataTypes,
11
+ UIMessage,
12
+ } from '../ui/ui-messages';
13
+ import { ValueOf } from '../util/value-of';
14
+ import { lazySchema, zodSchema } from '@ai-sdk/provider-utils';
15
+
16
+ export const uiMessageChunkSchema = lazySchema(() =>
17
+ zodSchema(
18
+ z.union([
19
+ z.strictObject({
20
+ type: z.literal('text-start'),
21
+ id: z.string(),
22
+ providerMetadata: providerMetadataSchema.optional(),
23
+ }),
24
+ z.strictObject({
25
+ type: z.literal('text-delta'),
26
+ id: z.string(),
27
+ delta: z.string(),
28
+ providerMetadata: providerMetadataSchema.optional(),
29
+ }),
30
+ z.strictObject({
31
+ type: z.literal('text-end'),
32
+ id: z.string(),
33
+ providerMetadata: providerMetadataSchema.optional(),
34
+ }),
35
+ z.strictObject({
36
+ type: z.literal('error'),
37
+ errorText: z.string(),
38
+ }),
39
+ z.strictObject({
40
+ type: z.literal('tool-input-start'),
41
+ toolCallId: z.string(),
42
+ toolName: z.string(),
43
+ providerExecuted: z.boolean().optional(),
44
+ dynamic: z.boolean().optional(),
45
+ title: z.string().optional(),
46
+ }),
47
+ z.strictObject({
48
+ type: z.literal('tool-input-delta'),
49
+ toolCallId: z.string(),
50
+ inputTextDelta: z.string(),
51
+ }),
52
+ z.strictObject({
53
+ type: z.literal('tool-input-available'),
54
+ toolCallId: z.string(),
55
+ toolName: z.string(),
56
+ input: z.unknown(),
57
+ providerExecuted: z.boolean().optional(),
58
+ providerMetadata: providerMetadataSchema.optional(),
59
+ dynamic: z.boolean().optional(),
60
+ title: z.string().optional(),
61
+ }),
62
+ z.strictObject({
63
+ type: z.literal('tool-input-error'),
64
+ toolCallId: z.string(),
65
+ toolName: z.string(),
66
+ input: z.unknown(),
67
+ providerExecuted: z.boolean().optional(),
68
+ providerMetadata: providerMetadataSchema.optional(),
69
+ dynamic: z.boolean().optional(),
70
+ errorText: z.string(),
71
+ title: z.string().optional(),
72
+ }),
73
+ z.strictObject({
74
+ type: z.literal('tool-approval-request'),
75
+ approvalId: z.string(),
76
+ toolCallId: z.string(),
77
+ }),
78
+ z.strictObject({
79
+ type: z.literal('tool-output-available'),
80
+ toolCallId: z.string(),
81
+ output: z.unknown(),
82
+ providerExecuted: z.boolean().optional(),
83
+ dynamic: z.boolean().optional(),
84
+ preliminary: z.boolean().optional(),
85
+ }),
86
+ z.strictObject({
87
+ type: z.literal('tool-output-error'),
88
+ toolCallId: z.string(),
89
+ errorText: z.string(),
90
+ providerExecuted: z.boolean().optional(),
91
+ dynamic: z.boolean().optional(),
92
+ }),
93
+ z.strictObject({
94
+ type: z.literal('tool-output-denied'),
95
+ toolCallId: z.string(),
96
+ }),
97
+ z.strictObject({
98
+ type: z.literal('reasoning-start'),
99
+ id: z.string(),
100
+ providerMetadata: providerMetadataSchema.optional(),
101
+ }),
102
+ z.strictObject({
103
+ type: z.literal('reasoning-delta'),
104
+ id: z.string(),
105
+ delta: z.string(),
106
+ providerMetadata: providerMetadataSchema.optional(),
107
+ }),
108
+ z.strictObject({
109
+ type: z.literal('reasoning-end'),
110
+ id: z.string(),
111
+ providerMetadata: providerMetadataSchema.optional(),
112
+ }),
113
+ z.strictObject({
114
+ type: z.literal('source-url'),
115
+ sourceId: z.string(),
116
+ url: z.string(),
117
+ title: z.string().optional(),
118
+ providerMetadata: providerMetadataSchema.optional(),
119
+ }),
120
+ z.strictObject({
121
+ type: z.literal('source-document'),
122
+ sourceId: z.string(),
123
+ mediaType: z.string(),
124
+ title: z.string(),
125
+ filename: z.string().optional(),
126
+ providerMetadata: providerMetadataSchema.optional(),
127
+ }),
128
+ z.strictObject({
129
+ type: z.literal('file'),
130
+ url: z.string(),
131
+ mediaType: z.string(),
132
+ providerMetadata: providerMetadataSchema.optional(),
133
+ }),
134
+ z.strictObject({
135
+ type: z.custom<`data-${string}`>(
136
+ (value): value is `data-${string}` =>
137
+ typeof value === 'string' && value.startsWith('data-'),
138
+ { message: 'Type must start with "data-"' },
139
+ ),
140
+ id: z.string().optional(),
141
+ data: z.unknown(),
142
+ transient: z.boolean().optional(),
143
+ }),
144
+ z.strictObject({
145
+ type: z.literal('start-step'),
146
+ }),
147
+ z.strictObject({
148
+ type: z.literal('finish-step'),
149
+ }),
150
+ z.strictObject({
151
+ type: z.literal('start'),
152
+ messageId: z.string().optional(),
153
+ messageMetadata: z.unknown().optional(),
154
+ }),
155
+ z.strictObject({
156
+ type: z.literal('finish'),
157
+ finishReason: z
158
+ .enum([
159
+ 'stop',
160
+ 'length',
161
+ 'content-filter',
162
+ 'tool-calls',
163
+ 'error',
164
+ 'other',
165
+ ] as const satisfies readonly FinishReason[])
166
+ .optional(),
167
+ messageMetadata: z.unknown().optional(),
168
+ }),
169
+ z.strictObject({
170
+ type: z.literal('abort'),
171
+ reason: z.string().optional(),
172
+ }),
173
+ z.strictObject({
174
+ type: z.literal('message-metadata'),
175
+ messageMetadata: z.unknown(),
176
+ }),
177
+ ]),
178
+ ),
179
+ );
180
+
181
+ export type DataUIMessageChunk<DATA_TYPES extends UIDataTypes> = ValueOf<{
182
+ [NAME in keyof DATA_TYPES & string]: {
183
+ type: `data-${NAME}`;
184
+ id?: string;
185
+ data: DATA_TYPES[NAME];
186
+ transient?: boolean;
187
+ };
188
+ }>;
189
+
190
+ export type UIMessageChunk<
191
+ METADATA = unknown,
192
+ DATA_TYPES extends UIDataTypes = UIDataTypes,
193
+ > =
194
+ | {
195
+ type: 'text-start';
196
+ id: string;
197
+ providerMetadata?: ProviderMetadata;
198
+ }
199
+ | {
200
+ type: 'text-delta';
201
+ delta: string;
202
+ id: string;
203
+ providerMetadata?: ProviderMetadata;
204
+ }
205
+ | {
206
+ type: 'text-end';
207
+ id: string;
208
+ providerMetadata?: ProviderMetadata;
209
+ }
210
+ | {
211
+ type: 'reasoning-start';
212
+ id: string;
213
+ providerMetadata?: ProviderMetadata;
214
+ }
215
+ | {
216
+ type: 'reasoning-delta';
217
+ id: string;
218
+ delta: string;
219
+ providerMetadata?: ProviderMetadata;
220
+ }
221
+ | {
222
+ type: 'reasoning-end';
223
+ id: string;
224
+ providerMetadata?: ProviderMetadata;
225
+ }
226
+ | {
227
+ type: 'error';
228
+ errorText: string;
229
+ }
230
+ | {
231
+ type: 'tool-input-available';
232
+ toolCallId: string;
233
+ toolName: string;
234
+ input: unknown;
235
+ providerExecuted?: boolean;
236
+ providerMetadata?: ProviderMetadata;
237
+ dynamic?: boolean;
238
+ title?: string;
239
+ }
240
+ | {
241
+ type: 'tool-input-error';
242
+ toolCallId: string;
243
+ toolName: string;
244
+ input: unknown;
245
+ providerExecuted?: boolean;
246
+ providerMetadata?: ProviderMetadata;
247
+ dynamic?: boolean;
248
+ errorText: string;
249
+ title?: string;
250
+ }
251
+ | {
252
+ type: 'tool-approval-request';
253
+ approvalId: string;
254
+ toolCallId: string;
255
+ }
256
+ | {
257
+ type: 'tool-output-available';
258
+ toolCallId: string;
259
+ output: unknown;
260
+ providerExecuted?: boolean;
261
+ dynamic?: boolean;
262
+ preliminary?: boolean;
263
+ }
264
+ | {
265
+ type: 'tool-output-error';
266
+ toolCallId: string;
267
+ errorText: string;
268
+ providerExecuted?: boolean;
269
+ dynamic?: boolean;
270
+ }
271
+ | {
272
+ type: 'tool-output-denied';
273
+ toolCallId: string;
274
+ }
275
+ | {
276
+ type: 'tool-input-start';
277
+ toolCallId: string;
278
+ toolName: string;
279
+ providerExecuted?: boolean;
280
+ dynamic?: boolean;
281
+ title?: string;
282
+ }
283
+ | {
284
+ type: 'tool-input-delta';
285
+ toolCallId: string;
286
+ inputTextDelta: string;
287
+ }
288
+ | {
289
+ type: 'source-url';
290
+ sourceId: string;
291
+ url: string;
292
+ title?: string;
293
+ providerMetadata?: ProviderMetadata;
294
+ }
295
+ | {
296
+ type: 'source-document';
297
+ sourceId: string;
298
+ mediaType: string;
299
+ title: string;
300
+ filename?: string;
301
+ providerMetadata?: ProviderMetadata;
302
+ }
303
+ | {
304
+ type: 'file';
305
+ url: string;
306
+ mediaType: string;
307
+ providerMetadata?: ProviderMetadata;
308
+ }
309
+ | DataUIMessageChunk<DATA_TYPES>
310
+ | {
311
+ type: 'start-step';
312
+ }
313
+ | {
314
+ type: 'finish-step';
315
+ }
316
+ | {
317
+ type: 'start';
318
+ messageId?: string;
319
+ messageMetadata?: METADATA;
320
+ }
321
+ | {
322
+ type: 'finish';
323
+ finishReason?: FinishReason;
324
+ messageMetadata?: METADATA;
325
+ }
326
+ | {
327
+ type: 'abort';
328
+ reason?: string;
329
+ }
330
+ | {
331
+ type: 'message-metadata';
332
+ messageMetadata: METADATA;
333
+ };
334
+
335
+ export function isDataUIMessageChunk(
336
+ chunk: UIMessageChunk,
337
+ ): chunk is DataUIMessageChunk<UIDataTypes> {
338
+ return chunk.type.startsWith('data-');
339
+ }
340
+
341
+ export type InferUIMessageChunk<T extends UIMessage> = UIMessageChunk<
342
+ InferUIMessageMetadata<T>,
343
+ InferUIMessageData<T>
344
+ >;
@@ -0,0 +1,7 @@
1
+ export const UI_MESSAGE_STREAM_HEADERS = {
2
+ 'content-type': 'text/event-stream',
3
+ 'cache-control': 'no-cache',
4
+ connection: 'keep-alive',
5
+ 'x-vercel-ai-ui-message-stream': 'v1',
6
+ 'x-accel-buffering': 'no', // disable nginx buffering
7
+ };
@@ -0,0 +1,32 @@
1
+ import { FinishReason } from '../types/language-model';
2
+ import { UIMessage } from '../ui/ui-messages';
3
+
4
+ export type UIMessageStreamOnFinishCallback<UI_MESSAGE extends UIMessage> =
5
+ (event: {
6
+ /**
7
+ * The updated list of UI messages.
8
+ */
9
+ messages: UI_MESSAGE[];
10
+
11
+ /**
12
+ * Indicates whether the response message is a continuation of the last original message,
13
+ * or if a new message was created.
14
+ */
15
+ isContinuation: boolean;
16
+
17
+ /**
18
+ * Indicates whether the stream was aborted.
19
+ */
20
+ isAborted: boolean;
21
+
22
+ /**
23
+ * The message that was sent to the client as a response
24
+ * (including the original message if it was extended).
25
+ */
26
+ responseMessage: UI_MESSAGE;
27
+
28
+ /**
29
+ * The reason why the generation finished.
30
+ */
31
+ finishReason?: FinishReason;
32
+ }) => PromiseLike<void> | void;
@@ -0,0 +1,5 @@
1
+ export type UIMessageStreamResponseInit = ResponseInit & {
2
+ consumeSseStream?: (options: {
3
+ stream: ReadableStream<string>;
4
+ }) => PromiseLike<void> | void;
5
+ };
@@ -0,0 +1,24 @@
1
+ import { UIMessage } from '../ui';
2
+ import { ErrorHandler } from '../util/error-handler';
3
+ import { InferUIMessageChunk } from './ui-message-chunks';
4
+
5
+ export interface UIMessageStreamWriter<
6
+ UI_MESSAGE extends UIMessage = UIMessage,
7
+ > {
8
+ /**
9
+ * Appends a data stream part to the stream.
10
+ */
11
+ write(part: InferUIMessageChunk<UI_MESSAGE>): void;
12
+
13
+ /**
14
+ * Merges the contents of another stream to this stream.
15
+ */
16
+ merge(stream: ReadableStream<InferUIMessageChunk<UI_MESSAGE>>): void;
17
+
18
+ /**
19
+ * Error handler that is used by the data stream writer.
20
+ * This is intended for forwarding when merging streams
21
+ * to prevent duplicated error masking.
22
+ */
23
+ onError: ErrorHandler | undefined;
24
+ }
@@ -0,0 +1,3 @@
1
+ export function asArray<T>(value: T | T[] | undefined): T[] {
2
+ return value === undefined ? [] : Array.isArray(value) ? value : [value];
3
+ }
@@ -0,0 +1,241 @@
1
+ import {
2
+ convertArrayToReadableStream,
3
+ convertAsyncIterableToArray,
4
+ convertReadableStreamToArray,
5
+ } from '@ai-sdk/provider-utils/test';
6
+ import { createAsyncIterableStream } from './async-iterable-stream';
7
+ import { describe, it, expect } from 'vitest';
8
+
9
+ describe('createAsyncIterableStream()', () => {
10
+ it('should read all chunks from a non-empty stream using async iteration', async () => {
11
+ const source = convertArrayToReadableStream(['chunk1', 'chunk2', 'chunk3']);
12
+
13
+ const asyncIterableStream = createAsyncIterableStream(source);
14
+
15
+ expect(await convertAsyncIterableToArray(asyncIterableStream)).toEqual([
16
+ 'chunk1',
17
+ 'chunk2',
18
+ 'chunk3',
19
+ ]);
20
+ });
21
+
22
+ it('should handle an empty stream gracefully', async () => {
23
+ const source = convertArrayToReadableStream<string>([]);
24
+ const asyncIterableStream = createAsyncIterableStream(source);
25
+
26
+ expect(await convertAsyncIterableToArray(asyncIterableStream)).toEqual([]);
27
+ });
28
+
29
+ it('should maintain ReadableStream functionality', async () => {
30
+ const source = convertArrayToReadableStream(['chunk1', 'chunk2', 'chunk3']);
31
+
32
+ const asyncIterableStream = createAsyncIterableStream(source);
33
+
34
+ expect(await convertReadableStreamToArray(asyncIterableStream)).toEqual([
35
+ 'chunk1',
36
+ 'chunk2',
37
+ 'chunk3',
38
+ ]);
39
+ });
40
+
41
+ it('should cancel stream on early exit from for-await loop', async () => {
42
+ let streamCancelled = false;
43
+
44
+ const source = new ReadableStream({
45
+ start(controller) {
46
+ controller.enqueue('chunk1');
47
+ controller.enqueue('chunk2');
48
+ controller.enqueue('chunk3');
49
+ },
50
+ cancel() {
51
+ streamCancelled = true;
52
+ },
53
+ });
54
+
55
+ const asyncIterableStream = createAsyncIterableStream(source);
56
+
57
+ const collected: string[] = [];
58
+ for await (const chunk of asyncIterableStream) {
59
+ collected.push(chunk);
60
+ if (chunk === 'chunk2') {
61
+ break;
62
+ }
63
+ }
64
+
65
+ expect(collected).toEqual(['chunk1', 'chunk2']);
66
+ expect(streamCancelled).toBe(true);
67
+ });
68
+
69
+ it('should cancel stream when exception thrown inside for-await loop', async () => {
70
+ let streamCancelled = false;
71
+
72
+ const source = new ReadableStream({
73
+ start(controller) {
74
+ controller.enqueue('chunk1');
75
+ controller.enqueue('chunk2');
76
+ controller.enqueue('chunk3');
77
+ },
78
+ cancel() {
79
+ streamCancelled = true;
80
+ },
81
+ });
82
+
83
+ const asyncIterableStream = createAsyncIterableStream(source);
84
+
85
+ const collected: string[] = [];
86
+ await expect(async () => {
87
+ for await (const chunk of asyncIterableStream) {
88
+ collected.push(chunk);
89
+ if (chunk === 'chunk2') {
90
+ throw new Error('Test error');
91
+ }
92
+ }
93
+ }).rejects.toThrow('Test error');
94
+
95
+ expect(collected).toEqual(['chunk1', 'chunk2']);
96
+ expect(streamCancelled).toBe(true);
97
+ });
98
+
99
+ it('should not cancel stream when exception thrown inside for-await loop', async () => {
100
+ let streamCancelled = false;
101
+
102
+ const source = new ReadableStream({
103
+ start(controller) {
104
+ controller.enqueue('chunk1');
105
+ controller.enqueue('chunk2');
106
+ controller.enqueue('chunk3');
107
+ controller.close();
108
+ },
109
+ cancel() {
110
+ streamCancelled = true;
111
+ },
112
+ });
113
+
114
+ const asyncIterableStream = createAsyncIterableStream(source);
115
+
116
+ expect(await convertAsyncIterableToArray(asyncIterableStream)).toEqual([
117
+ 'chunk1',
118
+ 'chunk2',
119
+ 'chunk3',
120
+ ]);
121
+
122
+ expect(streamCancelled).toBe(false);
123
+ });
124
+
125
+ it('should not allow iterating twice after breaking', async () => {
126
+ const source = convertArrayToReadableStream(['chunk1', 'chunk2', 'chunk3']);
127
+
128
+ const asyncIterableStream = createAsyncIterableStream(source);
129
+
130
+ const collected: string[] = [];
131
+ for await (const chunk of asyncIterableStream) {
132
+ collected.push(chunk);
133
+ if (chunk === 'chunk1') {
134
+ break;
135
+ }
136
+ }
137
+
138
+ expect(collected).toEqual(['chunk1']);
139
+
140
+ for await (const chunk of asyncIterableStream) {
141
+ collected.push(chunk);
142
+ }
143
+
144
+ expect(collected).toEqual(['chunk1']);
145
+ });
146
+
147
+ it('should propagate errors from source stream to async iterable', async () => {
148
+ let controller: ReadableStreamDefaultController<string>;
149
+ const source = new ReadableStream({
150
+ start(ctrl) {
151
+ controller = ctrl;
152
+ controller.enqueue('chunk1');
153
+ controller.enqueue('chunk2');
154
+ },
155
+ });
156
+
157
+ const asyncIterableStream = createAsyncIterableStream(source);
158
+
159
+ const collected: string[] = [];
160
+ await expect(async () => {
161
+ for await (const chunk of asyncIterableStream) {
162
+ collected.push(chunk);
163
+ if (chunk === 'chunk2') {
164
+ controller.error(new Error('Stream error'));
165
+ }
166
+ }
167
+ }).rejects.toThrow('Stream error');
168
+
169
+ expect(collected).toEqual(['chunk1', 'chunk2']);
170
+ });
171
+
172
+ it('should stop async iterable when stream is cancelled', async () => {
173
+ let iterationCompleted = false;
174
+ let errorCaught: Error | null = null;
175
+
176
+ const source = convertArrayToReadableStream(['chunk1', 'chunk2', 'chunk3']);
177
+
178
+ const asyncIterableStream = createAsyncIterableStream(source);
179
+
180
+ try {
181
+ for await (const chunk of asyncIterableStream) {
182
+ if (chunk === 'chunk1') {
183
+ await asyncIterableStream.cancel('Test cancellation');
184
+ }
185
+ }
186
+ iterationCompleted = true;
187
+ } catch (error) {
188
+ errorCaught = error as Error;
189
+ }
190
+
191
+ expect(iterationCompleted).toBe(false);
192
+ expect(errorCaught).not.toBeNull();
193
+ });
194
+
195
+ it('should not collect any chunks when iterating on already cancelled stream', async () => {
196
+ const source = convertArrayToReadableStream(['chunk1', 'chunk2', 'chunk3']);
197
+
198
+ const asyncIterableStream = createAsyncIterableStream(source);
199
+
200
+ await asyncIterableStream.cancel();
201
+
202
+ const collected: string[] = [];
203
+ for await (const chunk of asyncIterableStream) {
204
+ collected.push(chunk);
205
+ }
206
+
207
+ expect(collected).toEqual([]);
208
+ });
209
+
210
+ it('should not throw when return is called after the stream completed', async () => {
211
+ const input = ['chunk1', 'chunk2', 'chunk3'];
212
+ const source = convertArrayToReadableStream(input);
213
+
214
+ const asyncIterableStream = createAsyncIterableStream(source);
215
+
216
+ const asyncIterator = asyncIterableStream[Symbol.asyncIterator]();
217
+
218
+ const output = await (async () => {
219
+ const output: Array<string> = [];
220
+
221
+ while (true) {
222
+ const value = await asyncIterator.next();
223
+
224
+ if (value.done) {
225
+ break;
226
+ }
227
+
228
+ output.push(value.value);
229
+ }
230
+
231
+ return output;
232
+ })();
233
+
234
+ expect(output).toEqual(input);
235
+
236
+ expect(await asyncIterator.return?.()).toEqual({
237
+ done: true,
238
+ value: undefined,
239
+ });
240
+ });
241
+ });