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,827 @@
1
+ import { LanguageModelV3Usage } from '@ai-sdk/provider';
2
+ import {
3
+ convertArrayToReadableStream,
4
+ convertAsyncIterableToArray,
5
+ } from '@ai-sdk/provider-utils/test';
6
+ import { describe, expect, it } from 'vitest';
7
+ import { generateText, streamText } from '../generate-text';
8
+ import { wrapLanguageModel } from '../middleware/wrap-language-model';
9
+ import { MockLanguageModelV3 } from '../test/mock-language-model-v3';
10
+ import { extractJsonMiddleware } from './extract-json-middleware';
11
+
12
+ const testUsage: LanguageModelV3Usage = {
13
+ inputTokens: {
14
+ total: 5,
15
+ noCache: 5,
16
+ cacheRead: 0,
17
+ cacheWrite: 0,
18
+ },
19
+ outputTokens: {
20
+ total: 10,
21
+ text: 10,
22
+ reasoning: 0,
23
+ },
24
+ };
25
+
26
+ describe('extractJsonMiddleware', () => {
27
+ describe('wrapGenerate', () => {
28
+ it('should strip markdown json fence from text content', async () => {
29
+ const mockModel = new MockLanguageModelV3({
30
+ async doGenerate() {
31
+ return {
32
+ content: [
33
+ {
34
+ type: 'text',
35
+ text: '```json\n{"value": "test"}\n```',
36
+ },
37
+ ],
38
+ finishReason: { unified: 'stop', raw: 'stop' },
39
+ usage: testUsage,
40
+ warnings: [],
41
+ };
42
+ },
43
+ });
44
+
45
+ const result = await generateText({
46
+ model: wrapLanguageModel({
47
+ model: mockModel,
48
+ middleware: extractJsonMiddleware(),
49
+ }),
50
+ prompt: 'Generate JSON',
51
+ });
52
+
53
+ expect(result.text).toBe('{"value": "test"}');
54
+ });
55
+
56
+ it('should strip markdown fence without json tag', async () => {
57
+ const mockModel = new MockLanguageModelV3({
58
+ async doGenerate() {
59
+ return {
60
+ content: [
61
+ {
62
+ type: 'text',
63
+ text: '```\n{"value": "test"}\n```',
64
+ },
65
+ ],
66
+ finishReason: { unified: 'stop', raw: 'stop' },
67
+ usage: testUsage,
68
+ warnings: [],
69
+ };
70
+ },
71
+ });
72
+
73
+ const result = await generateText({
74
+ model: wrapLanguageModel({
75
+ model: mockModel,
76
+ middleware: extractJsonMiddleware(),
77
+ }),
78
+ prompt: 'Generate JSON',
79
+ });
80
+
81
+ expect(result.text).toBe('{"value": "test"}');
82
+ });
83
+
84
+ it('should leave text without fences unchanged', async () => {
85
+ const mockModel = new MockLanguageModelV3({
86
+ async doGenerate() {
87
+ return {
88
+ content: [
89
+ {
90
+ type: 'text',
91
+ text: '{"value": "test"}',
92
+ },
93
+ ],
94
+ finishReason: { unified: 'stop', raw: 'stop' },
95
+ usage: testUsage,
96
+ warnings: [],
97
+ };
98
+ },
99
+ });
100
+
101
+ const result = await generateText({
102
+ model: wrapLanguageModel({
103
+ model: mockModel,
104
+ middleware: extractJsonMiddleware(),
105
+ }),
106
+ prompt: 'Generate JSON',
107
+ });
108
+
109
+ expect(result.text).toBe('{"value": "test"}');
110
+ });
111
+
112
+ it('should use custom transform function when provided', async () => {
113
+ const mockModel = new MockLanguageModelV3({
114
+ async doGenerate() {
115
+ return {
116
+ content: [
117
+ {
118
+ type: 'text',
119
+ text: 'PREFIX{"value": "test"}SUFFIX',
120
+ },
121
+ ],
122
+ finishReason: { unified: 'stop', raw: 'stop' },
123
+ usage: testUsage,
124
+ warnings: [],
125
+ };
126
+ },
127
+ });
128
+
129
+ const result = await generateText({
130
+ model: wrapLanguageModel({
131
+ model: mockModel,
132
+ middleware: extractJsonMiddleware({
133
+ transform: text => text.replace('PREFIX', '').replace('SUFFIX', ''),
134
+ }),
135
+ }),
136
+ prompt: 'Generate JSON',
137
+ });
138
+
139
+ expect(result.text).toBe('{"value": "test"}');
140
+ });
141
+
142
+ it('should preserve non-text content parts', async () => {
143
+ const mockModel = new MockLanguageModelV3({
144
+ async doGenerate() {
145
+ return {
146
+ content: [
147
+ {
148
+ type: 'text',
149
+ text: '```json\n{"value": "test"}\n```',
150
+ },
151
+ {
152
+ type: 'tool-call',
153
+ toolCallId: 'call-1',
154
+ toolName: 'testTool',
155
+ input: '{"foo": "bar"}',
156
+ },
157
+ ],
158
+ finishReason: { unified: 'stop', raw: 'stop' },
159
+ usage: testUsage,
160
+ warnings: [],
161
+ };
162
+ },
163
+ });
164
+
165
+ const result = await generateText({
166
+ model: wrapLanguageModel({
167
+ model: mockModel,
168
+ middleware: extractJsonMiddleware(),
169
+ }),
170
+ prompt: 'Generate JSON',
171
+ });
172
+
173
+ expect(result.text).toBe('{"value": "test"}');
174
+ expect(result.content).toHaveLength(3);
175
+ expect(result.content[1]).toMatchObject({
176
+ type: 'tool-call',
177
+ toolCallId: 'call-1',
178
+ });
179
+ });
180
+ });
181
+
182
+ describe('wrapStream', () => {
183
+ it('should strip markdown json fence from streamed text', async () => {
184
+ const mockModel = new MockLanguageModelV3({
185
+ async doStream() {
186
+ return {
187
+ stream: convertArrayToReadableStream([
188
+ {
189
+ type: 'response-metadata',
190
+ id: 'id-0',
191
+ modelId: 'mock-model-id',
192
+ timestamp: new Date(0),
193
+ },
194
+ { type: 'text-start', id: '1' },
195
+ { type: 'text-delta', id: '1', delta: '```json\n' },
196
+ { type: 'text-delta', id: '1', delta: '{"value": "test"}' },
197
+ { type: 'text-delta', id: '1', delta: '\n```' },
198
+ { type: 'text-end', id: '1' },
199
+ {
200
+ type: 'finish',
201
+ finishReason: { unified: 'stop', raw: 'stop' },
202
+ usage: testUsage,
203
+ },
204
+ ]),
205
+ };
206
+ },
207
+ });
208
+
209
+ const result = streamText({
210
+ model: wrapLanguageModel({
211
+ model: mockModel,
212
+ middleware: extractJsonMiddleware(),
213
+ }),
214
+ prompt: 'Generate JSON',
215
+ });
216
+
217
+ expect(await result.text).toBe('{"value": "test"}');
218
+ });
219
+
220
+ it('should strip markdown fence without json tag from streamed text', async () => {
221
+ const mockModel = new MockLanguageModelV3({
222
+ async doStream() {
223
+ return {
224
+ stream: convertArrayToReadableStream([
225
+ {
226
+ type: 'response-metadata',
227
+ id: 'id-0',
228
+ modelId: 'mock-model-id',
229
+ timestamp: new Date(0),
230
+ },
231
+ { type: 'text-start', id: '1' },
232
+ { type: 'text-delta', id: '1', delta: '```\n' },
233
+ { type: 'text-delta', id: '1', delta: '{"value": "test"}' },
234
+ { type: 'text-delta', id: '1', delta: '\n```' },
235
+ { type: 'text-end', id: '1' },
236
+ {
237
+ type: 'finish',
238
+ finishReason: { unified: 'stop', raw: 'stop' },
239
+ usage: testUsage,
240
+ },
241
+ ]),
242
+ };
243
+ },
244
+ });
245
+
246
+ const result = streamText({
247
+ model: wrapLanguageModel({
248
+ model: mockModel,
249
+ middleware: extractJsonMiddleware(),
250
+ }),
251
+ prompt: 'Generate JSON',
252
+ });
253
+
254
+ expect(await result.text).toBe('{"value": "test"}');
255
+ });
256
+
257
+ it('should leave text without fences unchanged in stream', async () => {
258
+ const mockModel = new MockLanguageModelV3({
259
+ async doStream() {
260
+ return {
261
+ stream: convertArrayToReadableStream([
262
+ {
263
+ type: 'response-metadata',
264
+ id: 'id-0',
265
+ modelId: 'mock-model-id',
266
+ timestamp: new Date(0),
267
+ },
268
+ { type: 'text-start', id: '1' },
269
+ { type: 'text-delta', id: '1', delta: '{"value": "test"}' },
270
+ { type: 'text-end', id: '1' },
271
+ {
272
+ type: 'finish',
273
+ finishReason: { unified: 'stop', raw: 'stop' },
274
+ usage: testUsage,
275
+ },
276
+ ]),
277
+ };
278
+ },
279
+ });
280
+
281
+ const result = streamText({
282
+ model: wrapLanguageModel({
283
+ model: mockModel,
284
+ middleware: extractJsonMiddleware(),
285
+ }),
286
+ prompt: 'Generate JSON',
287
+ });
288
+
289
+ expect(await result.text).toBe('{"value": "test"}');
290
+ });
291
+
292
+ it('should handle fence split across multiple deltas', async () => {
293
+ const mockModel = new MockLanguageModelV3({
294
+ async doStream() {
295
+ return {
296
+ stream: convertArrayToReadableStream([
297
+ {
298
+ type: 'response-metadata',
299
+ id: 'id-0',
300
+ modelId: 'mock-model-id',
301
+ timestamp: new Date(0),
302
+ },
303
+ { type: 'text-start', id: '1' },
304
+ { type: 'text-delta', id: '1', delta: '`' },
305
+ { type: 'text-delta', id: '1', delta: '``' },
306
+ { type: 'text-delta', id: '1', delta: 'json\n' },
307
+ { type: 'text-delta', id: '1', delta: '{"value": "test"}' },
308
+ { type: 'text-delta', id: '1', delta: '\n`' },
309
+ { type: 'text-delta', id: '1', delta: '``' },
310
+ { type: 'text-end', id: '1' },
311
+ {
312
+ type: 'finish',
313
+ finishReason: { unified: 'stop', raw: 'stop' },
314
+ usage: testUsage,
315
+ },
316
+ ]),
317
+ };
318
+ },
319
+ });
320
+
321
+ const result = streamText({
322
+ model: wrapLanguageModel({
323
+ model: mockModel,
324
+ middleware: extractJsonMiddleware(),
325
+ }),
326
+ prompt: 'Generate JSON',
327
+ });
328
+
329
+ expect(await result.text).toBe('{"value": "test"}');
330
+ });
331
+
332
+ it('should handle content that starts with backtick but is not a fence', async () => {
333
+ const mockModel = new MockLanguageModelV3({
334
+ async doStream() {
335
+ return {
336
+ stream: convertArrayToReadableStream([
337
+ {
338
+ type: 'response-metadata',
339
+ id: 'id-0',
340
+ modelId: 'mock-model-id',
341
+ timestamp: new Date(0),
342
+ },
343
+ { type: 'text-start', id: '1' },
344
+ { type: 'text-delta', id: '1', delta: '`code`' },
345
+ { type: 'text-end', id: '1' },
346
+ {
347
+ type: 'finish',
348
+ finishReason: { unified: 'stop', raw: 'stop' },
349
+ usage: testUsage,
350
+ },
351
+ ]),
352
+ };
353
+ },
354
+ });
355
+
356
+ const result = streamText({
357
+ model: wrapLanguageModel({
358
+ model: mockModel,
359
+ middleware: extractJsonMiddleware(),
360
+ }),
361
+ prompt: 'Generate JSON',
362
+ });
363
+
364
+ expect(await result.text).toBe('`code`');
365
+ });
366
+
367
+ it('should pass through non-text chunks unchanged', async () => {
368
+ const mockModel = new MockLanguageModelV3({
369
+ async doStream() {
370
+ return {
371
+ stream: convertArrayToReadableStream([
372
+ {
373
+ type: 'response-metadata',
374
+ id: 'id-0',
375
+ modelId: 'mock-model-id',
376
+ timestamp: new Date(0),
377
+ },
378
+ { type: 'text-start', id: '1' },
379
+ { type: 'text-delta', id: '1', delta: '```json\n' },
380
+ { type: 'text-delta', id: '1', delta: '{"value": "test"}' },
381
+ { type: 'text-delta', id: '1', delta: '\n```' },
382
+ { type: 'text-end', id: '1' },
383
+ {
384
+ type: 'tool-input-start',
385
+ id: 'tool-1',
386
+ toolName: 'testTool',
387
+ },
388
+ {
389
+ type: 'tool-input-delta',
390
+ id: 'tool-1',
391
+ delta: '{"arg": "value"}',
392
+ },
393
+ { type: 'tool-input-end', id: 'tool-1' },
394
+ {
395
+ type: 'finish',
396
+ finishReason: { unified: 'stop', raw: 'stop' },
397
+ usage: testUsage,
398
+ },
399
+ ]),
400
+ };
401
+ },
402
+ });
403
+
404
+ const result = streamText({
405
+ model: wrapLanguageModel({
406
+ model: mockModel,
407
+ middleware: extractJsonMiddleware(),
408
+ }),
409
+ prompt: 'Generate JSON',
410
+ });
411
+
412
+ const fullStream = await convertAsyncIterableToArray(result.fullStream);
413
+ const toolInputStart = fullStream.find(
414
+ chunk => chunk.type === 'tool-input-start',
415
+ );
416
+ expect(toolInputStart).toBeDefined();
417
+ });
418
+
419
+ it('should handle multiple text blocks with different IDs', async () => {
420
+ const mockModel = new MockLanguageModelV3({
421
+ async doStream() {
422
+ return {
423
+ stream: convertArrayToReadableStream([
424
+ {
425
+ type: 'response-metadata',
426
+ id: 'id-0',
427
+ modelId: 'mock-model-id',
428
+ timestamp: new Date(0),
429
+ },
430
+ { type: 'text-start', id: '1' },
431
+ { type: 'text-delta', id: '1', delta: '```json\n' },
432
+ { type: 'text-delta', id: '1', delta: '{"first": true}' },
433
+ { type: 'text-delta', id: '1', delta: '\n```' },
434
+ { type: 'text-end', id: '1' },
435
+ { type: 'text-start', id: '2' },
436
+ { type: 'text-delta', id: '2', delta: '```json\n' },
437
+ { type: 'text-delta', id: '2', delta: '{"second": true}' },
438
+ { type: 'text-delta', id: '2', delta: '\n```' },
439
+ { type: 'text-end', id: '2' },
440
+ {
441
+ type: 'finish',
442
+ finishReason: { unified: 'stop', raw: 'stop' },
443
+ usage: testUsage,
444
+ },
445
+ ]),
446
+ };
447
+ },
448
+ });
449
+
450
+ const result = streamText({
451
+ model: wrapLanguageModel({
452
+ model: mockModel,
453
+ middleware: extractJsonMiddleware(),
454
+ }),
455
+ prompt: 'Generate JSON',
456
+ });
457
+
458
+ const fullStream = await convertAsyncIterableToArray(result.fullStream);
459
+ const textDeltas = fullStream.filter(
460
+ chunk => chunk.type === 'text-delta',
461
+ );
462
+
463
+ const allText = textDeltas.map(d => d.text).join('');
464
+ expect(allText).toContain('{"first": true}');
465
+ expect(allText).toContain('{"second": true}');
466
+ expect(allText).not.toContain('```');
467
+ });
468
+
469
+ it('should handle text-delta without prior text-start', async () => {
470
+ const mockModel = new MockLanguageModelV3({
471
+ async doStream() {
472
+ return {
473
+ stream: convertArrayToReadableStream([
474
+ {
475
+ type: 'response-metadata',
476
+ id: 'id-0',
477
+ modelId: 'mock-model-id',
478
+ timestamp: new Date(0),
479
+ },
480
+ // text-delta without text-start (edge case)
481
+ { type: 'text-delta', id: 'unknown', delta: 'some text' },
482
+ {
483
+ type: 'finish',
484
+ finishReason: { unified: 'stop', raw: 'stop' },
485
+ usage: testUsage,
486
+ },
487
+ ]),
488
+ };
489
+ },
490
+ });
491
+
492
+ const result = streamText({
493
+ model: wrapLanguageModel({
494
+ model: mockModel,
495
+ middleware: extractJsonMiddleware(),
496
+ }),
497
+ prompt: 'Generate JSON',
498
+ });
499
+
500
+ const fullStream = await convertAsyncIterableToArray(result.fullStream);
501
+ const textDeltas = fullStream.filter(
502
+ chunk => chunk.type === 'text-delta',
503
+ );
504
+ expect(textDeltas.length).toBeGreaterThanOrEqual(1);
505
+ });
506
+
507
+ it('should emit text-start when stream ends while still in prefix phase', async () => {
508
+ const mockModel = new MockLanguageModelV3({
509
+ async doStream() {
510
+ return {
511
+ stream: convertArrayToReadableStream([
512
+ {
513
+ type: 'response-metadata',
514
+ id: 'id-0',
515
+ modelId: 'mock-model-id',
516
+ timestamp: new Date(0),
517
+ },
518
+ { type: 'text-start', id: '1' },
519
+ // Very short content that stays in prefix buffer
520
+ { type: 'text-delta', id: '1', delta: '``' },
521
+ { type: 'text-end', id: '1' },
522
+ {
523
+ type: 'finish',
524
+ finishReason: { unified: 'stop', raw: 'stop' },
525
+ usage: testUsage,
526
+ },
527
+ ]),
528
+ };
529
+ },
530
+ });
531
+
532
+ const result = streamText({
533
+ model: wrapLanguageModel({
534
+ model: mockModel,
535
+ middleware: extractJsonMiddleware(),
536
+ }),
537
+ prompt: 'Generate JSON',
538
+ });
539
+
540
+ const fullStream = await convertAsyncIterableToArray(result.fullStream);
541
+ const textStarts = fullStream.filter(
542
+ chunk => chunk.type === 'text-start',
543
+ );
544
+ const textEnds = fullStream.filter(chunk => chunk.type === 'text-end');
545
+
546
+ expect(textStarts.length).toBe(1);
547
+ expect(textEnds.length).toBe(1);
548
+ });
549
+
550
+ it('should apply custom transform to streamed content', async () => {
551
+ const mockModel = new MockLanguageModelV3({
552
+ async doStream() {
553
+ return {
554
+ stream: convertArrayToReadableStream([
555
+ {
556
+ type: 'response-metadata',
557
+ id: 'id-0',
558
+ modelId: 'mock-model-id',
559
+ timestamp: new Date(0),
560
+ },
561
+ { type: 'text-start', id: '1' },
562
+ { type: 'text-delta', id: '1', delta: 'PREFIX' },
563
+ { type: 'text-delta', id: '1', delta: '{"value": "test"}' },
564
+ { type: 'text-delta', id: '1', delta: 'SUFFIX' },
565
+ { type: 'text-end', id: '1' },
566
+ {
567
+ type: 'finish',
568
+ finishReason: { unified: 'stop', raw: 'stop' },
569
+ usage: testUsage,
570
+ },
571
+ ]),
572
+ };
573
+ },
574
+ });
575
+
576
+ const result = streamText({
577
+ model: wrapLanguageModel({
578
+ model: mockModel,
579
+ middleware: extractJsonMiddleware({
580
+ transform: text => text.replace('PREFIX', '').replace('SUFFIX', ''),
581
+ }),
582
+ }),
583
+ prompt: 'Generate JSON',
584
+ });
585
+
586
+ expect(await result.text).toBe('{"value": "test"}');
587
+ });
588
+
589
+ it('should handle large content exceeding suffix buffer', async () => {
590
+ const largeJson = JSON.stringify({
591
+ data: 'x'.repeat(100),
592
+ nested: { values: Array.from({ length: 10 }, (_, i) => i) },
593
+ });
594
+
595
+ const mockModel = new MockLanguageModelV3({
596
+ async doStream() {
597
+ return {
598
+ stream: convertArrayToReadableStream([
599
+ {
600
+ type: 'response-metadata',
601
+ id: 'id-0',
602
+ modelId: 'mock-model-id',
603
+ timestamp: new Date(0),
604
+ },
605
+ { type: 'text-start', id: '1' },
606
+ { type: 'text-delta', id: '1', delta: '```json\n' },
607
+ { type: 'text-delta', id: '1', delta: largeJson },
608
+ { type: 'text-delta', id: '1', delta: '\n```' },
609
+ { type: 'text-end', id: '1' },
610
+ {
611
+ type: 'finish',
612
+ finishReason: { unified: 'stop', raw: 'stop' },
613
+ usage: testUsage,
614
+ },
615
+ ]),
616
+ };
617
+ },
618
+ });
619
+
620
+ const result = streamText({
621
+ model: wrapLanguageModel({
622
+ model: mockModel,
623
+ middleware: extractJsonMiddleware(),
624
+ }),
625
+ prompt: 'Generate JSON',
626
+ });
627
+
628
+ expect(await result.text).toBe(largeJson);
629
+ });
630
+
631
+ it('should handle content arriving character by character', async () => {
632
+ const chars = [...'```json\n{"value": "test"}\n```'];
633
+
634
+ const mockModel = new MockLanguageModelV3({
635
+ async doStream() {
636
+ return {
637
+ stream: convertArrayToReadableStream([
638
+ {
639
+ type: 'response-metadata',
640
+ id: 'id-0',
641
+ modelId: 'mock-model-id',
642
+ timestamp: new Date(0),
643
+ },
644
+ { type: 'text-start', id: '1' },
645
+ ...chars.map(char => ({
646
+ type: 'text-delta' as const,
647
+ id: '1',
648
+ delta: char,
649
+ })),
650
+ { type: 'text-end', id: '1' },
651
+ {
652
+ type: 'finish',
653
+ finishReason: { unified: 'stop', raw: 'stop' },
654
+ usage: testUsage,
655
+ },
656
+ ]),
657
+ };
658
+ },
659
+ });
660
+
661
+ const result = streamText({
662
+ model: wrapLanguageModel({
663
+ model: mockModel,
664
+ middleware: extractJsonMiddleware(),
665
+ }),
666
+ prompt: 'Generate JSON',
667
+ });
668
+
669
+ expect(await result.text).toBe('{"value": "test"}');
670
+ });
671
+
672
+ it('should handle fence with extra whitespace', async () => {
673
+ const mockModel = new MockLanguageModelV3({
674
+ async doStream() {
675
+ return {
676
+ stream: convertArrayToReadableStream([
677
+ {
678
+ type: 'response-metadata',
679
+ id: 'id-0',
680
+ modelId: 'mock-model-id',
681
+ timestamp: new Date(0),
682
+ },
683
+ { type: 'text-start', id: '1' },
684
+ { type: 'text-delta', id: '1', delta: '```json \n' },
685
+ { type: 'text-delta', id: '1', delta: '{"value": "test"}' },
686
+ { type: 'text-delta', id: '1', delta: '\n``` ' },
687
+ { type: 'text-end', id: '1' },
688
+ {
689
+ type: 'finish',
690
+ finishReason: { unified: 'stop', raw: 'stop' },
691
+ usage: testUsage,
692
+ },
693
+ ]),
694
+ };
695
+ },
696
+ });
697
+
698
+ const result = streamText({
699
+ model: wrapLanguageModel({
700
+ model: mockModel,
701
+ middleware: extractJsonMiddleware(),
702
+ }),
703
+ prompt: 'Generate JSON',
704
+ });
705
+
706
+ expect(await result.text).toBe('{"value": "test"}');
707
+ });
708
+
709
+ it('should verify stream output matches expected structure', async () => {
710
+ const mockModel = new MockLanguageModelV3({
711
+ async doStream() {
712
+ return {
713
+ stream: convertArrayToReadableStream([
714
+ {
715
+ type: 'response-metadata',
716
+ id: 'id-0',
717
+ modelId: 'mock-model-id',
718
+ timestamp: new Date(0),
719
+ },
720
+ { type: 'text-start', id: '1' },
721
+ { type: 'text-delta', id: '1', delta: '```json\n' },
722
+ { type: 'text-delta', id: '1', delta: '{"value": "test"}' },
723
+ { type: 'text-delta', id: '1', delta: '\n```' },
724
+ { type: 'text-end', id: '1' },
725
+ {
726
+ type: 'finish',
727
+ finishReason: { unified: 'stop', raw: 'stop' },
728
+ usage: testUsage,
729
+ },
730
+ ]),
731
+ };
732
+ },
733
+ });
734
+
735
+ const result = streamText({
736
+ model: wrapLanguageModel({
737
+ model: mockModel,
738
+ middleware: extractJsonMiddleware(),
739
+ }),
740
+ prompt: 'Generate JSON',
741
+ });
742
+
743
+ const fullStream = await convertAsyncIterableToArray(result.fullStream);
744
+
745
+ expect(fullStream.find(c => c.type === 'start')).toBeDefined();
746
+ expect(fullStream.find(c => c.type === 'text-start')).toBeDefined();
747
+ expect(fullStream.find(c => c.type === 'text-end')).toBeDefined();
748
+ expect(fullStream.find(c => c.type === 'finish')).toBeDefined();
749
+
750
+ const textDeltas = fullStream.filter(c => c.type === 'text-delta');
751
+ const combinedText = textDeltas.map(d => d.text).join('');
752
+ expect(combinedText).toBe('{"value": "test"}');
753
+ });
754
+
755
+ it('should handle empty content between fences', async () => {
756
+ const mockModel = new MockLanguageModelV3({
757
+ async doStream() {
758
+ return {
759
+ stream: convertArrayToReadableStream([
760
+ {
761
+ type: 'response-metadata',
762
+ id: 'id-0',
763
+ modelId: 'mock-model-id',
764
+ timestamp: new Date(0),
765
+ },
766
+ { type: 'text-start', id: '1' },
767
+ { type: 'text-delta', id: '1', delta: '```json\n```' },
768
+ { type: 'text-end', id: '1' },
769
+ {
770
+ type: 'finish',
771
+ finishReason: { unified: 'stop', raw: 'stop' },
772
+ usage: testUsage,
773
+ },
774
+ ]),
775
+ };
776
+ },
777
+ });
778
+
779
+ const result = streamText({
780
+ model: wrapLanguageModel({
781
+ model: mockModel,
782
+ middleware: extractJsonMiddleware(),
783
+ }),
784
+ prompt: 'Generate JSON',
785
+ });
786
+
787
+ expect(await result.text).toBe('');
788
+ });
789
+
790
+ it('should handle content starting without backtick quickly switching to streaming', async () => {
791
+ const mockModel = new MockLanguageModelV3({
792
+ async doStream() {
793
+ return {
794
+ stream: convertArrayToReadableStream([
795
+ {
796
+ type: 'response-metadata',
797
+ id: 'id-0',
798
+ modelId: 'mock-model-id',
799
+ timestamp: new Date(0),
800
+ },
801
+ { type: 'text-start', id: '1' },
802
+ { type: 'text-delta', id: '1', delta: '{' },
803
+ { type: 'text-delta', id: '1', delta: '"value": "test"' },
804
+ { type: 'text-delta', id: '1', delta: '}' },
805
+ { type: 'text-end', id: '1' },
806
+ {
807
+ type: 'finish',
808
+ finishReason: { unified: 'stop', raw: 'stop' },
809
+ usage: testUsage,
810
+ },
811
+ ]),
812
+ };
813
+ },
814
+ });
815
+
816
+ const result = streamText({
817
+ model: wrapLanguageModel({
818
+ model: mockModel,
819
+ middleware: extractJsonMiddleware(),
820
+ }),
821
+ prompt: 'Generate JSON',
822
+ });
823
+
824
+ expect(await result.text).toBe('{"value": "test"}');
825
+ });
826
+ });
827
+ });