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,135 @@
1
+ import {
2
+ createStreamingUIMessageState,
3
+ processUIMessageStream,
4
+ StreamingUIMessageState,
5
+ } from '../ui/process-ui-message-stream';
6
+ import { UIMessage } from '../ui/ui-messages';
7
+ import { ErrorHandler } from '../util/error-handler';
8
+ import { InferUIMessageChunk, UIMessageChunk } from './ui-message-chunks';
9
+ import { UIMessageStreamOnFinishCallback } from './ui-message-stream-on-finish-callback';
10
+
11
+ export function handleUIMessageStreamFinish<UI_MESSAGE extends UIMessage>({
12
+ messageId,
13
+ originalMessages = [],
14
+ onFinish,
15
+ onError,
16
+ stream,
17
+ }: {
18
+ stream: ReadableStream<InferUIMessageChunk<UI_MESSAGE>>;
19
+
20
+ /**
21
+ * The message ID to use for the response message.
22
+ * If not provided, no id will be set for the response message.
23
+ */
24
+ messageId?: string;
25
+
26
+ /**
27
+ * The original messages.
28
+ */
29
+ originalMessages?: UI_MESSAGE[];
30
+
31
+ onError: ErrorHandler;
32
+
33
+ onFinish?: UIMessageStreamOnFinishCallback<UI_MESSAGE>;
34
+ }): ReadableStream<InferUIMessageChunk<UI_MESSAGE>> {
35
+ // last message is only relevant for assistant messages
36
+ let lastMessage: UI_MESSAGE | undefined =
37
+ originalMessages?.[originalMessages.length - 1];
38
+ if (lastMessage?.role !== 'assistant') {
39
+ lastMessage = undefined;
40
+ } else {
41
+ // appending to the last message, so we need to use the same id
42
+ messageId = lastMessage.id;
43
+ }
44
+
45
+ let isAborted = false;
46
+
47
+ const idInjectedStream = stream.pipeThrough(
48
+ new TransformStream<
49
+ InferUIMessageChunk<UI_MESSAGE>,
50
+ InferUIMessageChunk<UI_MESSAGE>
51
+ >({
52
+ transform(chunk, controller) {
53
+ // when there is no messageId in the start chunk,
54
+ // but the user checked for persistence,
55
+ // inject the messageId into the chunk
56
+ if (chunk.type === 'start') {
57
+ const startChunk = chunk as UIMessageChunk & { type: 'start' };
58
+ if (startChunk.messageId == null && messageId != null) {
59
+ startChunk.messageId = messageId;
60
+ }
61
+ }
62
+
63
+ if (chunk.type === 'abort') {
64
+ isAborted = true;
65
+ }
66
+
67
+ controller.enqueue(chunk);
68
+ },
69
+ }),
70
+ );
71
+
72
+ if (onFinish == null) {
73
+ return idInjectedStream;
74
+ }
75
+
76
+ const state = createStreamingUIMessageState<UI_MESSAGE>({
77
+ lastMessage: lastMessage
78
+ ? (structuredClone(lastMessage) as UI_MESSAGE)
79
+ : undefined,
80
+ messageId: messageId ?? '', // will be overridden by the stream
81
+ });
82
+
83
+ const runUpdateMessageJob = async (
84
+ job: (options: {
85
+ state: StreamingUIMessageState<UI_MESSAGE>;
86
+ write: () => void;
87
+ }) => Promise<void>,
88
+ ) => {
89
+ await job({ state, write: () => {} });
90
+ };
91
+
92
+ let finishCalled = false;
93
+
94
+ const callOnFinish = async () => {
95
+ if (finishCalled || !onFinish) {
96
+ return;
97
+ }
98
+ finishCalled = true;
99
+
100
+ const isContinuation = state.message.id === lastMessage?.id;
101
+ await onFinish({
102
+ isAborted,
103
+ isContinuation,
104
+ responseMessage: state.message as UI_MESSAGE,
105
+ messages: [
106
+ ...(isContinuation ? originalMessages.slice(0, -1) : originalMessages),
107
+ state.message,
108
+ ] as UI_MESSAGE[],
109
+ finishReason: state.finishReason,
110
+ });
111
+ };
112
+
113
+ return processUIMessageStream<UI_MESSAGE>({
114
+ stream: idInjectedStream,
115
+ runUpdateMessageJob,
116
+ onError,
117
+ }).pipeThrough(
118
+ new TransformStream<
119
+ InferUIMessageChunk<UI_MESSAGE>,
120
+ InferUIMessageChunk<UI_MESSAGE>
121
+ >({
122
+ transform(chunk, controller) {
123
+ controller.enqueue(chunk);
124
+ },
125
+ // @ts-expect-error cancel is still new and missing from types https://developer.mozilla.org/en-US/docs/Web/API/TransformStream#browser_compatibility
126
+ async cancel() {
127
+ await callOnFinish();
128
+ },
129
+
130
+ async flush() {
131
+ await callOnFinish();
132
+ },
133
+ }),
134
+ );
135
+ }
@@ -0,0 +1,13 @@
1
+ export { createUIMessageStream } from './create-ui-message-stream';
2
+ export { createUIMessageStreamResponse } from './create-ui-message-stream-response';
3
+ export { JsonToSseTransformStream } from './json-to-sse-transform-stream';
4
+ export { pipeUIMessageStreamToResponse } from './pipe-ui-message-stream-to-response';
5
+ export { readUIMessageStream } from './read-ui-message-stream';
6
+ export {
7
+ uiMessageChunkSchema,
8
+ type InferUIMessageChunk,
9
+ type UIMessageChunk,
10
+ } from './ui-message-chunks';
11
+ export { UI_MESSAGE_STREAM_HEADERS } from './ui-message-stream-headers';
12
+ export type { UIMessageStreamOnFinishCallback } from './ui-message-stream-on-finish-callback';
13
+ export type { UIMessageStreamWriter } from './ui-message-stream-writer';
@@ -0,0 +1,12 @@
1
+ export class JsonToSseTransformStream extends TransformStream<unknown, string> {
2
+ constructor() {
3
+ super({
4
+ transform(part, controller) {
5
+ controller.enqueue(`data: ${JSON.stringify(part)}\n\n`);
6
+ },
7
+ flush(controller) {
8
+ controller.enqueue('data: [DONE]\n\n');
9
+ },
10
+ });
11
+ }
12
+ }
@@ -0,0 +1,90 @@
1
+ import { convertArrayToReadableStream } from '@ai-sdk/provider-utils/test';
2
+ import { createMockServerResponse } from '../test/mock-server-response';
3
+ import { pipeUIMessageStreamToResponse } from './pipe-ui-message-stream-to-response';
4
+ import { describe, it, expect } from 'vitest';
5
+
6
+ describe('pipeUIMessageStreamToResponse', () => {
7
+ it('should write to ServerResponse with correct headers and encoded stream', async () => {
8
+ const mockResponse = createMockServerResponse();
9
+
10
+ pipeUIMessageStreamToResponse({
11
+ response: mockResponse,
12
+ status: 200,
13
+ statusText: 'OK',
14
+ headers: {
15
+ 'Custom-Header': 'test',
16
+ },
17
+ stream: convertArrayToReadableStream([
18
+ { type: 'text-start', id: '1' },
19
+ { type: 'text-delta', id: '1', delta: 'test-data' },
20
+ { type: 'text-end', id: '1' },
21
+ ]),
22
+ });
23
+
24
+ // Wait for the stream to finish writing
25
+ await mockResponse.waitForEnd();
26
+
27
+ // Verify response properties
28
+ expect(mockResponse.statusCode).toBe(200);
29
+ expect(mockResponse.statusMessage).toBe('OK');
30
+
31
+ // Verify headers
32
+ expect(mockResponse.headers).toMatchInlineSnapshot(`
33
+ {
34
+ "cache-control": "no-cache",
35
+ "connection": "keep-alive",
36
+ "content-type": "text/event-stream",
37
+ "custom-header": "test",
38
+ "x-accel-buffering": "no",
39
+ "x-vercel-ai-ui-message-stream": "v1",
40
+ }
41
+ `);
42
+
43
+ // Verify written data using decoded chunks
44
+ const decodedChunks = mockResponse.getDecodedChunks();
45
+ expect(decodedChunks).toMatchInlineSnapshot(`
46
+ [
47
+ "data: {"type":"text-start","id":"1"}
48
+
49
+ ",
50
+ "data: {"type":"text-delta","id":"1","delta":"test-data"}
51
+
52
+ ",
53
+ "data: {"type":"text-end","id":"1"}
54
+
55
+ ",
56
+ "data: [DONE]
57
+
58
+ ",
59
+ ]
60
+ `);
61
+ });
62
+
63
+ it('should handle errors in the stream', async () => {
64
+ const mockResponse = createMockServerResponse();
65
+
66
+ pipeUIMessageStreamToResponse({
67
+ response: mockResponse,
68
+ status: 200,
69
+ stream: convertArrayToReadableStream([
70
+ { type: 'error', errorText: 'Custom error message' },
71
+ ]),
72
+ });
73
+
74
+ // Wait for the stream to finish writing
75
+ await mockResponse.waitForEnd();
76
+
77
+ // Verify error handling using decoded chunks
78
+ const decodedChunks = mockResponse.getDecodedChunks();
79
+ expect(decodedChunks).toMatchInlineSnapshot(`
80
+ [
81
+ "data: {"type":"error","errorText":"Custom error message"}
82
+
83
+ ",
84
+ "data: [DONE]
85
+
86
+ ",
87
+ ]
88
+ `);
89
+ });
90
+ });
@@ -0,0 +1,40 @@
1
+ import { ServerResponse } from 'node:http';
2
+ import { prepareHeaders } from '../util/prepare-headers';
3
+ import { writeToServerResponse } from '../util/write-to-server-response';
4
+ import { JsonToSseTransformStream } from './json-to-sse-transform-stream';
5
+ import { UI_MESSAGE_STREAM_HEADERS } from './ui-message-stream-headers';
6
+ import { UIMessageChunk } from './ui-message-chunks';
7
+ import { UIMessageStreamResponseInit } from './ui-message-stream-response-init';
8
+
9
+ export function pipeUIMessageStreamToResponse({
10
+ response,
11
+ status,
12
+ statusText,
13
+ headers,
14
+ stream,
15
+ consumeSseStream,
16
+ }: {
17
+ response: ServerResponse;
18
+ stream: ReadableStream<UIMessageChunk>;
19
+ } & UIMessageStreamResponseInit): void {
20
+ let sseStream = stream.pipeThrough(new JsonToSseTransformStream());
21
+
22
+ // when the consumeSseStream is provided, we need to tee the stream
23
+ // and send the second part to the consumeSseStream function
24
+ // so that it can be consumed by the client independently
25
+ if (consumeSseStream) {
26
+ const [stream1, stream2] = sseStream.tee();
27
+ sseStream = stream1;
28
+ consumeSseStream({ stream: stream2 }); // no await (do not block the response)
29
+ }
30
+
31
+ writeToServerResponse({
32
+ response,
33
+ status,
34
+ statusText,
35
+ headers: Object.fromEntries(
36
+ prepareHeaders(headers, UI_MESSAGE_STREAM_HEADERS).entries(),
37
+ ),
38
+ stream: sseStream.pipeThrough(new TextEncoderStream()),
39
+ });
40
+ }
@@ -0,0 +1,122 @@
1
+ import {
2
+ convertArrayToReadableStream,
3
+ convertAsyncIterableToArray,
4
+ } from '@ai-sdk/provider-utils/test';
5
+ import { UIMessageChunk } from './ui-message-chunks';
6
+ import { readUIMessageStream } from './read-ui-message-stream';
7
+ import { describe, it, expect } from 'vitest';
8
+
9
+ function createUIMessageStream(parts: UIMessageChunk[]) {
10
+ return convertArrayToReadableStream(parts);
11
+ }
12
+
13
+ describe('readUIMessageStream', () => {
14
+ it('should return a ui message object stream for a basic input stream', async () => {
15
+ const stream = createUIMessageStream([
16
+ { type: 'start', messageId: 'msg-123' },
17
+ { type: 'start-step' },
18
+ { type: 'text-start', id: 'text-1' },
19
+ { type: 'text-delta', id: 'text-1', delta: 'Hello, ' },
20
+ { type: 'text-delta', id: 'text-1', delta: 'world!' },
21
+ { type: 'text-end', id: 'text-1' },
22
+ { type: 'finish-step' },
23
+ { type: 'finish' },
24
+ ]);
25
+
26
+ const uiMessages = readUIMessageStream({ stream });
27
+
28
+ expect(await convertAsyncIterableToArray(uiMessages))
29
+ .toMatchInlineSnapshot(`
30
+ [
31
+ {
32
+ "id": "msg-123",
33
+ "metadata": undefined,
34
+ "parts": [],
35
+ "role": "assistant",
36
+ },
37
+ {
38
+ "id": "msg-123",
39
+ "metadata": undefined,
40
+ "parts": [
41
+ {
42
+ "type": "step-start",
43
+ },
44
+ {
45
+ "providerMetadata": undefined,
46
+ "state": "streaming",
47
+ "text": "",
48
+ "type": "text",
49
+ },
50
+ ],
51
+ "role": "assistant",
52
+ },
53
+ {
54
+ "id": "msg-123",
55
+ "metadata": undefined,
56
+ "parts": [
57
+ {
58
+ "type": "step-start",
59
+ },
60
+ {
61
+ "providerMetadata": undefined,
62
+ "state": "streaming",
63
+ "text": "Hello, ",
64
+ "type": "text",
65
+ },
66
+ ],
67
+ "role": "assistant",
68
+ },
69
+ {
70
+ "id": "msg-123",
71
+ "metadata": undefined,
72
+ "parts": [
73
+ {
74
+ "type": "step-start",
75
+ },
76
+ {
77
+ "providerMetadata": undefined,
78
+ "state": "streaming",
79
+ "text": "Hello, world!",
80
+ "type": "text",
81
+ },
82
+ ],
83
+ "role": "assistant",
84
+ },
85
+ {
86
+ "id": "msg-123",
87
+ "metadata": undefined,
88
+ "parts": [
89
+ {
90
+ "type": "step-start",
91
+ },
92
+ {
93
+ "providerMetadata": undefined,
94
+ "state": "done",
95
+ "text": "Hello, world!",
96
+ "type": "text",
97
+ },
98
+ ],
99
+ "role": "assistant",
100
+ },
101
+ ]
102
+ `);
103
+ });
104
+
105
+ it('should throw an error when encountering an error UI stream part', async () => {
106
+ const stream = createUIMessageStream([
107
+ { type: 'start', messageId: 'msg-123' },
108
+ { type: 'text-start', id: 'text-1' },
109
+ { type: 'text-delta', id: 'text-1', delta: 'Hello' },
110
+ { type: 'error', errorText: 'Test error message' },
111
+ ]);
112
+
113
+ const uiMessages = readUIMessageStream({
114
+ stream,
115
+ terminateOnError: true,
116
+ });
117
+
118
+ await expect(convertAsyncIterableToArray(uiMessages)).rejects.toThrow(
119
+ 'Test error message',
120
+ );
121
+ });
122
+ });
@@ -0,0 +1,87 @@
1
+ import { UIMessage } from '../ui/ui-messages';
2
+ import { UIMessageChunk } from './ui-message-chunks';
3
+ import {
4
+ createStreamingUIMessageState,
5
+ processUIMessageStream,
6
+ StreamingUIMessageState,
7
+ } from '../ui/process-ui-message-stream';
8
+ import {
9
+ AsyncIterableStream,
10
+ createAsyncIterableStream,
11
+ } from '../util/async-iterable-stream';
12
+ import { consumeStream } from '../util/consume-stream';
13
+
14
+ /**
15
+ * Transforms a stream of `UIMessageChunk`s into an `AsyncIterableStream` of `UIMessage`s.
16
+ *
17
+ * @param options.message - The last assistant message to use as a starting point when the conversation is resumed. Otherwise undefined.
18
+ * @param options.stream - The stream of `UIMessageChunk`s to read.
19
+ * @param options.terminateOnError - Whether to terminate the stream if an error occurs.
20
+ * @param options.onError - A function that is called when an error occurs.
21
+ *
22
+ * @returns An `AsyncIterableStream` of `UIMessage`s. Each stream part is a different state of the same message
23
+ * as it is being completed.
24
+ */
25
+ export function readUIMessageStream<UI_MESSAGE extends UIMessage>({
26
+ message,
27
+ stream,
28
+ onError,
29
+ terminateOnError = false,
30
+ }: {
31
+ message?: UI_MESSAGE;
32
+ stream: ReadableStream<UIMessageChunk>;
33
+ onError?: (error: unknown) => void;
34
+ terminateOnError?: boolean;
35
+ }): AsyncIterableStream<UI_MESSAGE> {
36
+ let controller: ReadableStreamDefaultController<UI_MESSAGE> | undefined;
37
+ let hasErrored = false;
38
+
39
+ const outputStream = new ReadableStream<UI_MESSAGE>({
40
+ start(controllerParam) {
41
+ controller = controllerParam;
42
+ },
43
+ });
44
+
45
+ const state = createStreamingUIMessageState<UI_MESSAGE>({
46
+ messageId: message?.id ?? '',
47
+ lastMessage: message,
48
+ });
49
+
50
+ const handleError = (error: unknown) => {
51
+ onError?.(error);
52
+
53
+ if (!hasErrored && terminateOnError) {
54
+ hasErrored = true;
55
+ controller?.error(error);
56
+ }
57
+ };
58
+
59
+ consumeStream({
60
+ stream: processUIMessageStream({
61
+ stream,
62
+ runUpdateMessageJob(
63
+ job: (options: {
64
+ state: StreamingUIMessageState<UI_MESSAGE>;
65
+ write: () => void;
66
+ }) => Promise<void>,
67
+ ) {
68
+ return job({
69
+ state,
70
+ write: () => {
71
+ controller?.enqueue(structuredClone(state.message));
72
+ },
73
+ });
74
+ },
75
+ onError: handleError,
76
+ }),
77
+ onError: handleError,
78
+ }).finally(() => {
79
+ // Only close if no error occurred. Calling close() on an errored controller
80
+ // throws "Invalid state: Controller is already closed" TypeError.
81
+ if (!hasErrored) {
82
+ controller?.close();
83
+ }
84
+ });
85
+
86
+ return createAsyncIterableStream(outputStream);
87
+ }
@@ -0,0 +1,18 @@
1
+ import { describe, expectTypeOf, it } from 'vitest';
2
+ import { UIMessageChunk, uiMessageChunkSchema } from './ui-message-chunks';
3
+ import { validateTypes } from '@ai-sdk/provider-utils';
4
+
5
+ describe('UI message chunks type', () => {
6
+ it('parsed UI message chunk should have UIMessageChunk type', async () => {
7
+ const chunk = await validateTypes({
8
+ schema: uiMessageChunkSchema,
9
+ value: {
10
+ type: 'text-delta',
11
+ delta: 'Hello, world!',
12
+ id: '123',
13
+ },
14
+ });
15
+
16
+ expectTypeOf(chunk).toEqualTypeOf<UIMessageChunk>();
17
+ });
18
+ });