ai 6.0.33 → 6.0.34

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (351) hide show
  1. package/CHANGELOG.md +10 -0
  2. package/dist/index.js +1 -1
  3. package/dist/index.mjs +1 -1
  4. package/dist/internal/index.js +1 -1
  5. package/dist/internal/index.mjs +1 -1
  6. package/docs/02-foundations/03-prompts.mdx +2 -2
  7. package/docs/03-ai-sdk-core/15-tools-and-tool-calling.mdx +1 -1
  8. package/docs/07-reference/01-ai-sdk-core/28-output.mdx +1 -1
  9. package/package.json +6 -4
  10. package/src/agent/agent.ts +116 -0
  11. package/src/agent/create-agent-ui-stream-response.test.ts +258 -0
  12. package/src/agent/create-agent-ui-stream-response.ts +50 -0
  13. package/src/agent/create-agent-ui-stream.ts +73 -0
  14. package/src/agent/index.ts +33 -0
  15. package/src/agent/infer-agent-tools.ts +7 -0
  16. package/src/agent/infer-agent-ui-message.test-d.ts +54 -0
  17. package/src/agent/infer-agent-ui-message.ts +11 -0
  18. package/src/agent/pipe-agent-ui-stream-to-response.ts +52 -0
  19. package/src/agent/tool-loop-agent-on-finish-callback.ts +31 -0
  20. package/src/agent/tool-loop-agent-on-step-finish-callback.ts +11 -0
  21. package/src/agent/tool-loop-agent-settings.ts +182 -0
  22. package/src/agent/tool-loop-agent.test-d.ts +114 -0
  23. package/src/agent/tool-loop-agent.test.ts +442 -0
  24. package/src/agent/tool-loop-agent.ts +114 -0
  25. package/src/embed/__snapshots__/embed-many.test.ts.snap +191 -0
  26. package/src/embed/__snapshots__/embed.test.ts.snap +81 -0
  27. package/src/embed/embed-many-result.ts +53 -0
  28. package/src/embed/embed-many.test.ts +653 -0
  29. package/src/embed/embed-many.ts +378 -0
  30. package/src/embed/embed-result.ts +50 -0
  31. package/src/embed/embed.test.ts +298 -0
  32. package/src/embed/embed.ts +211 -0
  33. package/src/embed/index.ts +4 -0
  34. package/src/error/index.ts +34 -0
  35. package/src/error/invalid-argument-error.ts +34 -0
  36. package/src/error/invalid-stream-part-error.ts +28 -0
  37. package/src/error/invalid-tool-approval-error.ts +26 -0
  38. package/src/error/invalid-tool-input-error.ts +33 -0
  39. package/src/error/no-image-generated-error.ts +39 -0
  40. package/src/error/no-object-generated-error.ts +70 -0
  41. package/src/error/no-output-generated-error.ts +26 -0
  42. package/src/error/no-speech-generated-error.ts +18 -0
  43. package/src/error/no-such-tool-error.ts +35 -0
  44. package/src/error/no-transcript-generated-error.ts +20 -0
  45. package/src/error/tool-call-not-found-for-approval-error.ts +32 -0
  46. package/src/error/tool-call-repair-error.ts +30 -0
  47. package/src/error/unsupported-model-version-error.ts +23 -0
  48. package/src/error/verify-no-object-generated-error.ts +27 -0
  49. package/src/generate-image/generate-image-result.ts +42 -0
  50. package/src/generate-image/generate-image.test.ts +1420 -0
  51. package/src/generate-image/generate-image.ts +360 -0
  52. package/src/generate-image/index.ts +18 -0
  53. package/src/generate-object/__snapshots__/generate-object.test.ts.snap +133 -0
  54. package/src/generate-object/__snapshots__/stream-object.test.ts.snap +297 -0
  55. package/src/generate-object/generate-object-result.ts +67 -0
  56. package/src/generate-object/generate-object.test-d.ts +49 -0
  57. package/src/generate-object/generate-object.test.ts +1191 -0
  58. package/src/generate-object/generate-object.ts +518 -0
  59. package/src/generate-object/index.ts +9 -0
  60. package/src/generate-object/inject-json-instruction.test.ts +181 -0
  61. package/src/generate-object/inject-json-instruction.ts +30 -0
  62. package/src/generate-object/output-strategy.ts +415 -0
  63. package/src/generate-object/parse-and-validate-object-result.ts +111 -0
  64. package/src/generate-object/repair-text.ts +12 -0
  65. package/src/generate-object/stream-object-result.ts +120 -0
  66. package/src/generate-object/stream-object.test-d.ts +74 -0
  67. package/src/generate-object/stream-object.test.ts +1950 -0
  68. package/src/generate-object/stream-object.ts +986 -0
  69. package/src/generate-object/validate-object-generation-input.ts +144 -0
  70. package/src/generate-speech/generate-speech-result.ts +30 -0
  71. package/src/generate-speech/generate-speech.test.ts +300 -0
  72. package/src/generate-speech/generate-speech.ts +190 -0
  73. package/src/generate-speech/generated-audio-file.ts +65 -0
  74. package/src/generate-speech/index.ts +3 -0
  75. package/src/generate-text/__snapshots__/generate-text.test.ts.snap +1872 -0
  76. package/src/generate-text/__snapshots__/stream-text.test.ts.snap +1255 -0
  77. package/src/generate-text/collect-tool-approvals.test.ts +553 -0
  78. package/src/generate-text/collect-tool-approvals.ts +116 -0
  79. package/src/generate-text/content-part.ts +25 -0
  80. package/src/generate-text/execute-tool-call.ts +129 -0
  81. package/src/generate-text/extract-reasoning-content.ts +17 -0
  82. package/src/generate-text/extract-text-content.ts +15 -0
  83. package/src/generate-text/generate-text-result.ts +168 -0
  84. package/src/generate-text/generate-text.test-d.ts +68 -0
  85. package/src/generate-text/generate-text.test.ts +7011 -0
  86. package/src/generate-text/generate-text.ts +1223 -0
  87. package/src/generate-text/generated-file.ts +70 -0
  88. package/src/generate-text/index.ts +57 -0
  89. package/src/generate-text/is-approval-needed.ts +29 -0
  90. package/src/generate-text/output-utils.ts +23 -0
  91. package/src/generate-text/output.test.ts +698 -0
  92. package/src/generate-text/output.ts +590 -0
  93. package/src/generate-text/parse-tool-call.test.ts +570 -0
  94. package/src/generate-text/parse-tool-call.ts +188 -0
  95. package/src/generate-text/prepare-step.ts +103 -0
  96. package/src/generate-text/prune-messages.test.ts +720 -0
  97. package/src/generate-text/prune-messages.ts +167 -0
  98. package/src/generate-text/reasoning-output.ts +20 -0
  99. package/src/generate-text/reasoning.ts +8 -0
  100. package/src/generate-text/response-message.ts +10 -0
  101. package/src/generate-text/run-tools-transformation.test.ts +1143 -0
  102. package/src/generate-text/run-tools-transformation.ts +420 -0
  103. package/src/generate-text/smooth-stream.test.ts +2101 -0
  104. package/src/generate-text/smooth-stream.ts +162 -0
  105. package/src/generate-text/step-result.ts +238 -0
  106. package/src/generate-text/stop-condition.ts +29 -0
  107. package/src/generate-text/stream-text-result.ts +463 -0
  108. package/src/generate-text/stream-text.test-d.ts +200 -0
  109. package/src/generate-text/stream-text.test.ts +19979 -0
  110. package/src/generate-text/stream-text.ts +2505 -0
  111. package/src/generate-text/to-response-messages.test.ts +922 -0
  112. package/src/generate-text/to-response-messages.ts +163 -0
  113. package/src/generate-text/tool-approval-request-output.ts +21 -0
  114. package/src/generate-text/tool-call-repair-function.ts +27 -0
  115. package/src/generate-text/tool-call.ts +47 -0
  116. package/src/generate-text/tool-error.ts +34 -0
  117. package/src/generate-text/tool-output-denied.ts +21 -0
  118. package/src/generate-text/tool-output.ts +7 -0
  119. package/src/generate-text/tool-result.ts +36 -0
  120. package/src/generate-text/tool-set.ts +14 -0
  121. package/src/global.ts +24 -0
  122. package/src/index.ts +50 -0
  123. package/src/logger/index.ts +6 -0
  124. package/src/logger/log-warnings.test.ts +351 -0
  125. package/src/logger/log-warnings.ts +119 -0
  126. package/src/middleware/__snapshots__/simulate-streaming-middleware.test.ts.snap +64 -0
  127. package/src/middleware/add-tool-input-examples-middleware.test.ts +476 -0
  128. package/src/middleware/add-tool-input-examples-middleware.ts +90 -0
  129. package/src/middleware/default-embedding-settings-middleware.test.ts +126 -0
  130. package/src/middleware/default-embedding-settings-middleware.ts +22 -0
  131. package/src/middleware/default-settings-middleware.test.ts +388 -0
  132. package/src/middleware/default-settings-middleware.ts +33 -0
  133. package/src/middleware/extract-json-middleware.test.ts +827 -0
  134. package/src/middleware/extract-json-middleware.ts +197 -0
  135. package/src/middleware/extract-reasoning-middleware.test.ts +1028 -0
  136. package/src/middleware/extract-reasoning-middleware.ts +238 -0
  137. package/src/middleware/index.ts +10 -0
  138. package/src/middleware/simulate-streaming-middleware.test.ts +911 -0
  139. package/src/middleware/simulate-streaming-middleware.ts +79 -0
  140. package/src/middleware/wrap-embedding-model.test.ts +358 -0
  141. package/src/middleware/wrap-embedding-model.ts +86 -0
  142. package/src/middleware/wrap-image-model.test.ts +423 -0
  143. package/src/middleware/wrap-image-model.ts +85 -0
  144. package/src/middleware/wrap-language-model.test.ts +518 -0
  145. package/src/middleware/wrap-language-model.ts +104 -0
  146. package/src/middleware/wrap-provider.test.ts +120 -0
  147. package/src/middleware/wrap-provider.ts +51 -0
  148. package/src/model/as-embedding-model-v3.test.ts +319 -0
  149. package/src/model/as-embedding-model-v3.ts +24 -0
  150. package/src/model/as-image-model-v3.test.ts +409 -0
  151. package/src/model/as-image-model-v3.ts +24 -0
  152. package/src/model/as-language-model-v3.test.ts +508 -0
  153. package/src/model/as-language-model-v3.ts +103 -0
  154. package/src/model/as-provider-v3.ts +36 -0
  155. package/src/model/as-speech-model-v3.test.ts +356 -0
  156. package/src/model/as-speech-model-v3.ts +24 -0
  157. package/src/model/as-transcription-model-v3.test.ts +529 -0
  158. package/src/model/as-transcription-model-v3.ts +24 -0
  159. package/src/model/resolve-model.test.ts +244 -0
  160. package/src/model/resolve-model.ts +126 -0
  161. package/src/prompt/call-settings.ts +148 -0
  162. package/src/prompt/content-part.ts +209 -0
  163. package/src/prompt/convert-to-language-model-prompt.test.ts +2018 -0
  164. package/src/prompt/convert-to-language-model-prompt.ts +442 -0
  165. package/src/prompt/create-tool-model-output.test.ts +508 -0
  166. package/src/prompt/create-tool-model-output.ts +34 -0
  167. package/src/prompt/data-content.test.ts +15 -0
  168. package/src/prompt/data-content.ts +134 -0
  169. package/src/prompt/index.ts +27 -0
  170. package/src/prompt/invalid-data-content-error.ts +29 -0
  171. package/src/prompt/invalid-message-role-error.ts +27 -0
  172. package/src/prompt/message-conversion-error.ts +28 -0
  173. package/src/prompt/message.ts +68 -0
  174. package/src/prompt/prepare-call-settings.test.ts +159 -0
  175. package/src/prompt/prepare-call-settings.ts +108 -0
  176. package/src/prompt/prepare-tools-and-tool-choice.test.ts +461 -0
  177. package/src/prompt/prepare-tools-and-tool-choice.ts +86 -0
  178. package/src/prompt/prompt.ts +43 -0
  179. package/src/prompt/split-data-url.ts +17 -0
  180. package/src/prompt/standardize-prompt.test.ts +82 -0
  181. package/src/prompt/standardize-prompt.ts +99 -0
  182. package/src/prompt/wrap-gateway-error.ts +29 -0
  183. package/src/registry/custom-provider.test.ts +211 -0
  184. package/src/registry/custom-provider.ts +155 -0
  185. package/src/registry/index.ts +7 -0
  186. package/src/registry/no-such-provider-error.ts +41 -0
  187. package/src/registry/provider-registry.test.ts +691 -0
  188. package/src/registry/provider-registry.ts +328 -0
  189. package/src/rerank/index.ts +2 -0
  190. package/src/rerank/rerank-result.ts +70 -0
  191. package/src/rerank/rerank.test.ts +516 -0
  192. package/src/rerank/rerank.ts +237 -0
  193. package/src/telemetry/assemble-operation-name.ts +21 -0
  194. package/src/telemetry/get-base-telemetry-attributes.ts +53 -0
  195. package/src/telemetry/get-tracer.ts +20 -0
  196. package/src/telemetry/noop-tracer.ts +69 -0
  197. package/src/telemetry/record-span.ts +63 -0
  198. package/src/telemetry/select-telemetry-attributes.ts +78 -0
  199. package/src/telemetry/select-temetry-attributes.test.ts +114 -0
  200. package/src/telemetry/stringify-for-telemetry.test.ts +114 -0
  201. package/src/telemetry/stringify-for-telemetry.ts +33 -0
  202. package/src/telemetry/telemetry-settings.ts +44 -0
  203. package/src/test/mock-embedding-model-v2.ts +35 -0
  204. package/src/test/mock-embedding-model-v3.ts +48 -0
  205. package/src/test/mock-image-model-v2.ts +28 -0
  206. package/src/test/mock-image-model-v3.ts +28 -0
  207. package/src/test/mock-language-model-v2.ts +72 -0
  208. package/src/test/mock-language-model-v3.ts +77 -0
  209. package/src/test/mock-provider-v2.ts +68 -0
  210. package/src/test/mock-provider-v3.ts +80 -0
  211. package/src/test/mock-reranking-model-v3.ts +25 -0
  212. package/src/test/mock-server-response.ts +69 -0
  213. package/src/test/mock-speech-model-v2.ts +24 -0
  214. package/src/test/mock-speech-model-v3.ts +24 -0
  215. package/src/test/mock-tracer.ts +156 -0
  216. package/src/test/mock-transcription-model-v2.ts +24 -0
  217. package/src/test/mock-transcription-model-v3.ts +24 -0
  218. package/src/test/mock-values.ts +4 -0
  219. package/src/test/not-implemented.ts +3 -0
  220. package/src/text-stream/create-text-stream-response.test.ts +38 -0
  221. package/src/text-stream/create-text-stream-response.ts +18 -0
  222. package/src/text-stream/index.ts +2 -0
  223. package/src/text-stream/pipe-text-stream-to-response.test.ts +38 -0
  224. package/src/text-stream/pipe-text-stream-to-response.ts +26 -0
  225. package/src/transcribe/index.ts +2 -0
  226. package/src/transcribe/transcribe-result.ts +60 -0
  227. package/src/transcribe/transcribe.test.ts +313 -0
  228. package/src/transcribe/transcribe.ts +173 -0
  229. package/src/types/embedding-model-middleware.ts +3 -0
  230. package/src/types/embedding-model.ts +18 -0
  231. package/src/types/image-model-middleware.ts +3 -0
  232. package/src/types/image-model-response-metadata.ts +16 -0
  233. package/src/types/image-model.ts +19 -0
  234. package/src/types/index.ts +29 -0
  235. package/src/types/json-value.ts +15 -0
  236. package/src/types/language-model-middleware.ts +3 -0
  237. package/src/types/language-model-request-metadata.ts +6 -0
  238. package/src/types/language-model-response-metadata.ts +21 -0
  239. package/src/types/language-model.ts +104 -0
  240. package/src/types/provider-metadata.ts +16 -0
  241. package/src/types/provider.ts +55 -0
  242. package/src/types/reranking-model.ts +6 -0
  243. package/src/types/speech-model-response-metadata.ts +21 -0
  244. package/src/types/speech-model.ts +6 -0
  245. package/src/types/transcription-model-response-metadata.ts +16 -0
  246. package/src/types/transcription-model.ts +9 -0
  247. package/src/types/usage.ts +200 -0
  248. package/src/types/warning.ts +7 -0
  249. package/src/ui/__snapshots__/append-response-messages.test.ts.snap +416 -0
  250. package/src/ui/__snapshots__/convert-to-model-messages.test.ts.snap +419 -0
  251. package/src/ui/__snapshots__/process-chat-text-response.test.ts.snap +142 -0
  252. package/src/ui/call-completion-api.ts +157 -0
  253. package/src/ui/chat-transport.ts +83 -0
  254. package/src/ui/chat.test-d.ts +233 -0
  255. package/src/ui/chat.test.ts +2695 -0
  256. package/src/ui/chat.ts +716 -0
  257. package/src/ui/convert-file-list-to-file-ui-parts.ts +36 -0
  258. package/src/ui/convert-to-model-messages.test.ts +2775 -0
  259. package/src/ui/convert-to-model-messages.ts +373 -0
  260. package/src/ui/default-chat-transport.ts +36 -0
  261. package/src/ui/direct-chat-transport.test.ts +446 -0
  262. package/src/ui/direct-chat-transport.ts +118 -0
  263. package/src/ui/http-chat-transport.test.ts +185 -0
  264. package/src/ui/http-chat-transport.ts +292 -0
  265. package/src/ui/index.ts +71 -0
  266. package/src/ui/last-assistant-message-is-complete-with-approval-responses.ts +44 -0
  267. package/src/ui/last-assistant-message-is-complete-with-tool-calls.test.ts +371 -0
  268. package/src/ui/last-assistant-message-is-complete-with-tool-calls.ts +39 -0
  269. package/src/ui/process-text-stream.test.ts +38 -0
  270. package/src/ui/process-text-stream.ts +16 -0
  271. package/src/ui/process-ui-message-stream.test.ts +8052 -0
  272. package/src/ui/process-ui-message-stream.ts +713 -0
  273. package/src/ui/text-stream-chat-transport.ts +23 -0
  274. package/src/ui/transform-text-to-ui-message-stream.test.ts +124 -0
  275. package/src/ui/transform-text-to-ui-message-stream.ts +27 -0
  276. package/src/ui/ui-messages.test.ts +48 -0
  277. package/src/ui/ui-messages.ts +534 -0
  278. package/src/ui/use-completion.ts +84 -0
  279. package/src/ui/validate-ui-messages.test.ts +1428 -0
  280. package/src/ui/validate-ui-messages.ts +476 -0
  281. package/src/ui-message-stream/create-ui-message-stream-response.test.ts +266 -0
  282. package/src/ui-message-stream/create-ui-message-stream-response.ts +32 -0
  283. package/src/ui-message-stream/create-ui-message-stream.test.ts +639 -0
  284. package/src/ui-message-stream/create-ui-message-stream.ts +124 -0
  285. package/src/ui-message-stream/get-response-ui-message-id.test.ts +55 -0
  286. package/src/ui-message-stream/get-response-ui-message-id.ts +24 -0
  287. package/src/ui-message-stream/handle-ui-message-stream-finish.test.ts +429 -0
  288. package/src/ui-message-stream/handle-ui-message-stream-finish.ts +135 -0
  289. package/src/ui-message-stream/index.ts +13 -0
  290. package/src/ui-message-stream/json-to-sse-transform-stream.ts +12 -0
  291. package/src/ui-message-stream/pipe-ui-message-stream-to-response.test.ts +90 -0
  292. package/src/ui-message-stream/pipe-ui-message-stream-to-response.ts +40 -0
  293. package/src/ui-message-stream/read-ui-message-stream.test.ts +122 -0
  294. package/src/ui-message-stream/read-ui-message-stream.ts +87 -0
  295. package/src/ui-message-stream/ui-message-chunks.test-d.ts +18 -0
  296. package/src/ui-message-stream/ui-message-chunks.ts +344 -0
  297. package/src/ui-message-stream/ui-message-stream-headers.ts +7 -0
  298. package/src/ui-message-stream/ui-message-stream-on-finish-callback.ts +32 -0
  299. package/src/ui-message-stream/ui-message-stream-response-init.ts +5 -0
  300. package/src/ui-message-stream/ui-message-stream-writer.ts +24 -0
  301. package/src/util/as-array.ts +3 -0
  302. package/src/util/async-iterable-stream.test.ts +241 -0
  303. package/src/util/async-iterable-stream.ts +94 -0
  304. package/src/util/consume-stream.ts +29 -0
  305. package/src/util/cosine-similarity.test.ts +57 -0
  306. package/src/util/cosine-similarity.ts +47 -0
  307. package/src/util/create-resolvable-promise.ts +30 -0
  308. package/src/util/create-stitchable-stream.test.ts +239 -0
  309. package/src/util/create-stitchable-stream.ts +112 -0
  310. package/src/util/data-url.ts +17 -0
  311. package/src/util/deep-partial.ts +84 -0
  312. package/src/util/detect-media-type.test.ts +670 -0
  313. package/src/util/detect-media-type.ts +184 -0
  314. package/src/util/download/download-function.ts +45 -0
  315. package/src/util/download/download.test.ts +69 -0
  316. package/src/util/download/download.ts +46 -0
  317. package/src/util/error-handler.ts +1 -0
  318. package/src/util/fix-json.test.ts +279 -0
  319. package/src/util/fix-json.ts +401 -0
  320. package/src/util/get-potential-start-index.test.ts +34 -0
  321. package/src/util/get-potential-start-index.ts +30 -0
  322. package/src/util/index.ts +11 -0
  323. package/src/util/is-deep-equal-data.test.ts +119 -0
  324. package/src/util/is-deep-equal-data.ts +48 -0
  325. package/src/util/is-non-empty-object.ts +5 -0
  326. package/src/util/job.ts +1 -0
  327. package/src/util/log-v2-compatibility-warning.ts +21 -0
  328. package/src/util/merge-abort-signals.test.ts +155 -0
  329. package/src/util/merge-abort-signals.ts +43 -0
  330. package/src/util/merge-objects.test.ts +118 -0
  331. package/src/util/merge-objects.ts +79 -0
  332. package/src/util/now.ts +4 -0
  333. package/src/util/parse-partial-json.test.ts +80 -0
  334. package/src/util/parse-partial-json.ts +30 -0
  335. package/src/util/prepare-headers.test.ts +51 -0
  336. package/src/util/prepare-headers.ts +14 -0
  337. package/src/util/prepare-retries.test.ts +10 -0
  338. package/src/util/prepare-retries.ts +47 -0
  339. package/src/util/retry-error.ts +41 -0
  340. package/src/util/retry-with-exponential-backoff.test.ts +446 -0
  341. package/src/util/retry-with-exponential-backoff.ts +154 -0
  342. package/src/util/serial-job-executor.test.ts +162 -0
  343. package/src/util/serial-job-executor.ts +36 -0
  344. package/src/util/simulate-readable-stream.test.ts +98 -0
  345. package/src/util/simulate-readable-stream.ts +39 -0
  346. package/src/util/split-array.test.ts +60 -0
  347. package/src/util/split-array.ts +20 -0
  348. package/src/util/value-of.ts +65 -0
  349. package/src/util/write-to-server-response.test.ts +266 -0
  350. package/src/util/write-to-server-response.ts +49 -0
  351. package/src/version.ts +5 -0
@@ -0,0 +1,351 @@
1
+ import type { SharedV3Warning } from '@ai-sdk/provider';
2
+ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
3
+ import { Warning } from '../types/warning';
4
+ import {
5
+ FIRST_WARNING_INFO_MESSAGE,
6
+ logWarnings,
7
+ resetLogWarningsState,
8
+ } from './log-warnings';
9
+
10
+ // Mock console.warn and console.info
11
+ const mockConsoleWarn = vi.fn();
12
+ const mockConsoleInfo = vi.fn();
13
+ vi.stubGlobal('console', { warn: mockConsoleWarn, info: mockConsoleInfo });
14
+
15
+ describe('logWarnings', () => {
16
+ beforeEach(() => {
17
+ mockConsoleWarn.mockClear();
18
+ mockConsoleInfo.mockClear();
19
+ resetLogWarningsState();
20
+ delete globalThis.AI_SDK_LOG_WARNINGS;
21
+ });
22
+
23
+ afterEach(() => {
24
+ delete globalThis.AI_SDK_LOG_WARNINGS;
25
+ });
26
+
27
+ describe('when AI_SDK_LOG_WARNINGS is false', () => {
28
+ beforeEach(() => {
29
+ globalThis.AI_SDK_LOG_WARNINGS = false;
30
+ });
31
+
32
+ it('should not log any warnings (single)', () => {
33
+ const warnings: Warning[] = [
34
+ {
35
+ type: 'other',
36
+ message: 'Test warning',
37
+ } as SharedV3Warning,
38
+ ];
39
+
40
+ logWarnings({ warnings, provider: 'providerX', model: 'modelY' });
41
+
42
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
43
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
44
+ });
45
+
46
+ it('should not log any warnings (multiple)', () => {
47
+ const warnings: Warning[] = [
48
+ {
49
+ type: 'other',
50
+ message: 'Test warning 1',
51
+ },
52
+ {
53
+ type: 'other',
54
+ message: 'Test warning 2',
55
+ },
56
+ ];
57
+
58
+ logWarnings({ warnings, provider: 'provider', model: 'model' });
59
+
60
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
61
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
62
+ });
63
+
64
+ it('should not count empty arrays as first call', () => {
65
+ logWarnings({ warnings: [], provider: 'prov', model: 'mod' });
66
+
67
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
68
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
69
+
70
+ logWarnings({
71
+ warnings: [{ type: 'other', message: 'foo' } as SharedV3Warning],
72
+ provider: 'p1',
73
+ model: 'm1',
74
+ });
75
+
76
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
77
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
78
+ });
79
+ });
80
+
81
+ describe('when AI_SDK_LOG_WARNINGS is a custom function', () => {
82
+ it('should call the custom function with warning options', () => {
83
+ const customLogger = vi.fn();
84
+ globalThis.AI_SDK_LOG_WARNINGS = customLogger;
85
+
86
+ const warnings: Warning[] = [
87
+ {
88
+ type: 'other',
89
+ message: 'Test warning',
90
+ } as SharedV3Warning,
91
+ ];
92
+
93
+ const options = { warnings, provider: 'pp', model: 'mm' };
94
+ logWarnings(options);
95
+
96
+ expect(customLogger).toHaveBeenCalledOnce();
97
+ expect(customLogger).toHaveBeenCalledWith(options);
98
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
99
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
100
+ });
101
+
102
+ it('should call the custom function with multiple warnings', () => {
103
+ const customLogger = vi.fn();
104
+ globalThis.AI_SDK_LOG_WARNINGS = customLogger;
105
+
106
+ const warnings: Warning[] = [
107
+ {
108
+ type: 'unsupported',
109
+ feature: 'temperature',
110
+ details: 'Temperature not supported',
111
+ },
112
+ {
113
+ type: 'other',
114
+ message: 'Another warning',
115
+ },
116
+ ];
117
+
118
+ const opts = { warnings, provider: 'provider', model: 'model' };
119
+ logWarnings(opts);
120
+
121
+ expect(customLogger).toHaveBeenCalledOnce();
122
+ expect(customLogger).toHaveBeenCalledWith(opts);
123
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
124
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
125
+ });
126
+
127
+ it('should not call the custom function with empty warnings array', () => {
128
+ const customLogger = vi.fn();
129
+ globalThis.AI_SDK_LOG_WARNINGS = customLogger;
130
+
131
+ const warnings: Warning[] = [];
132
+
133
+ logWarnings({ warnings, provider: 'x', model: 'y' });
134
+
135
+ expect(customLogger).not.toHaveBeenCalled();
136
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
137
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
138
+ });
139
+ });
140
+
141
+ describe('when AI_SDK_LOG_WARNINGS is unset/undefined (default behavior)', () => {
142
+ it('should show console.info once for first warning(s), then log to console.warn for each warning', () => {
143
+ const warning: SharedV3Warning = {
144
+ type: 'other',
145
+ message: 'Test warning message',
146
+ };
147
+ const warnings: Warning[] = [warning];
148
+
149
+ logWarnings({ warnings, provider: 'myProvider', model: 'myModel' });
150
+
151
+ expect(mockConsoleInfo).toHaveBeenCalledTimes(1);
152
+ expect(mockConsoleInfo).toHaveBeenCalledWith(FIRST_WARNING_INFO_MESSAGE);
153
+ expect(mockConsoleWarn).toHaveBeenCalledTimes(1);
154
+ expect(mockConsoleWarn).toHaveBeenCalledWith(
155
+ 'AI SDK Warning (myProvider / myModel): Test warning message',
156
+ );
157
+ });
158
+
159
+ it('should only show console.info on the first non-empty call', () => {
160
+ const first: Warning[] = [
161
+ { type: 'other', message: '1' } as SharedV3Warning,
162
+ ];
163
+ const second: Warning[] = [
164
+ { type: 'other', message: '2' } as SharedV3Warning,
165
+ ];
166
+
167
+ logWarnings({ warnings: first, provider: 'a', model: 'b' });
168
+ logWarnings({ warnings: second, provider: 'a', model: 'b' });
169
+
170
+ // Info on first call only
171
+ expect(mockConsoleInfo).toHaveBeenCalledTimes(1);
172
+ expect(mockConsoleInfo).toHaveBeenCalledWith(FIRST_WARNING_INFO_MESSAGE);
173
+ expect(mockConsoleWarn).toHaveBeenCalledTimes(2);
174
+ expect(mockConsoleWarn).toHaveBeenNthCalledWith(
175
+ 1,
176
+ 'AI SDK Warning (a / b): 1',
177
+ );
178
+ expect(mockConsoleWarn).toHaveBeenNthCalledWith(
179
+ 2,
180
+ 'AI SDK Warning (a / b): 2',
181
+ );
182
+ });
183
+
184
+ it('should only log for non-empty warnings', () => {
185
+ logWarnings({ warnings: [], provider: 'err', model: 'm' });
186
+
187
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
188
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
189
+
190
+ logWarnings({
191
+ warnings: [{ type: 'other', message: 't1' } as SharedV3Warning],
192
+ provider: 'prov',
193
+ model: 'mod',
194
+ });
195
+ expect(mockConsoleInfo).toHaveBeenCalledOnce();
196
+ expect(mockConsoleWarn).toHaveBeenCalledOnce();
197
+
198
+ logWarnings({ warnings: [], provider: 'prov', model: 'mod' });
199
+ expect(mockConsoleInfo).toHaveBeenCalledOnce();
200
+ expect(mockConsoleWarn).toHaveBeenCalledOnce();
201
+
202
+ logWarnings({
203
+ warnings: [{ type: 'other', message: 't2' } as SharedV3Warning],
204
+ provider: 'prov',
205
+ model: 'mod',
206
+ });
207
+ expect(mockConsoleInfo).toHaveBeenCalledOnce(); // only once
208
+ expect(mockConsoleWarn).toHaveBeenCalledTimes(2);
209
+ });
210
+
211
+ it('should handle various warning types per formatWarning', () => {
212
+ const warnings: Warning[] = [
213
+ {
214
+ type: 'unsupported',
215
+ feature: 'mediaType',
216
+ details: 'detail',
217
+ },
218
+ {
219
+ type: 'unsupported',
220
+ feature: 'voice',
221
+ details: 'detail2',
222
+ },
223
+ {
224
+ type: 'other',
225
+ message: 'other msg',
226
+ },
227
+ ];
228
+
229
+ logWarnings({ warnings, provider: 'zzz', model: 'MMM' });
230
+
231
+ expect(mockConsoleInfo).toHaveBeenCalledTimes(1);
232
+ expect(mockConsoleWarn).toHaveBeenCalledTimes(3);
233
+ expect(mockConsoleWarn).toHaveBeenNthCalledWith(
234
+ 1,
235
+ 'AI SDK Warning (zzz / MMM): ' +
236
+ 'The feature "mediaType" is not supported. detail',
237
+ );
238
+ expect(mockConsoleWarn).toHaveBeenNthCalledWith(
239
+ 2,
240
+ 'AI SDK Warning (zzz / MMM): ' +
241
+ 'The feature "voice" is not supported. detail2',
242
+ );
243
+ expect(mockConsoleWarn).toHaveBeenNthCalledWith(
244
+ 3,
245
+ 'AI SDK Warning (zzz / MMM): other msg',
246
+ );
247
+ });
248
+
249
+ it('should include warning even with "unknown provider" and "unknown model"', () => {
250
+ logWarnings({
251
+ warnings: [{ type: 'other', message: 'messx' } as SharedV3Warning],
252
+ provider: 'unknown provider',
253
+ model: 'unknown model',
254
+ });
255
+
256
+ expect(mockConsoleInfo).toHaveBeenCalledTimes(1);
257
+ expect(mockConsoleWarn).toHaveBeenCalledWith(
258
+ 'AI SDK Warning (unknown provider / unknown model): messx',
259
+ );
260
+ });
261
+ });
262
+
263
+ describe('when AI_SDK_LOG_WARNINGS is undefined (explicitly set)', () => {
264
+ beforeEach(() => {
265
+ globalThis.AI_SDK_LOG_WARNINGS = undefined;
266
+ });
267
+
268
+ it('should use default behavior and log to console.warn', () => {
269
+ const warning: SharedV3Warning = {
270
+ type: 'other',
271
+ message: 'Test warning with undefined logger',
272
+ };
273
+ const warnings: Warning[] = [warning];
274
+
275
+ logWarnings({ warnings, provider: 'p1', model: 'm1' });
276
+
277
+ expect(mockConsoleInfo).toHaveBeenCalledOnce();
278
+ expect(mockConsoleInfo).toHaveBeenCalledWith(FIRST_WARNING_INFO_MESSAGE);
279
+ expect(mockConsoleWarn).toHaveBeenCalledOnce();
280
+ expect(mockConsoleWarn).toHaveBeenCalledWith(
281
+ 'AI SDK Warning (p1 / m1): Test warning with undefined logger',
282
+ );
283
+ });
284
+ });
285
+
286
+ describe('first-time information note', () => {
287
+ it('should not display the info message for empty warnings', () => {
288
+ logWarnings({ warnings: [], provider: 'a', model: 'b' });
289
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
290
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
291
+ });
292
+
293
+ it('should display informational note only on first real call', () => {
294
+ logWarnings({ warnings: [], provider: 'a', model: 'b' });
295
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
296
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
297
+
298
+ logWarnings({ warnings: [], provider: 'a', model: 'b' });
299
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
300
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
301
+
302
+ logWarnings({
303
+ warnings: [{ type: 'other', message: 'foo' } as SharedV3Warning],
304
+ provider: 'abc',
305
+ model: 'bbb',
306
+ });
307
+ expect(mockConsoleInfo).toHaveBeenCalledTimes(1);
308
+ expect(mockConsoleWarn).toHaveBeenCalledTimes(1);
309
+
310
+ logWarnings({
311
+ warnings: [{ type: 'other', message: 'bar' } as SharedV3Warning],
312
+ provider: 'abc',
313
+ model: 'bbb',
314
+ });
315
+ expect(mockConsoleInfo).toHaveBeenCalledTimes(1);
316
+ expect(mockConsoleWarn).toHaveBeenCalledTimes(2);
317
+ });
318
+
319
+ it('should not display information note when using custom logger', () => {
320
+ const customLogger = vi.fn();
321
+ globalThis.AI_SDK_LOG_WARNINGS = customLogger;
322
+
323
+ logWarnings({
324
+ warnings: [{ type: 'other', message: 'Message' } as SharedV3Warning],
325
+ provider: 'provV',
326
+ model: 'modZ',
327
+ });
328
+
329
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
330
+ expect(customLogger).toHaveBeenCalledOnce();
331
+ });
332
+
333
+ it('should not display information note when AI_SDK_LOG_WARNINGS is false', () => {
334
+ globalThis.AI_SDK_LOG_WARNINGS = false;
335
+
336
+ logWarnings({
337
+ warnings: [
338
+ {
339
+ type: 'other',
340
+ message: 'Suppressed',
341
+ } as SharedV3Warning,
342
+ ],
343
+ provider: 'notProv',
344
+ model: 'notModel',
345
+ });
346
+
347
+ expect(mockConsoleInfo).not.toHaveBeenCalled();
348
+ expect(mockConsoleWarn).not.toHaveBeenCalled();
349
+ });
350
+ });
351
+ });
@@ -0,0 +1,119 @@
1
+ import { Warning } from '../types';
2
+
3
+ /**
4
+ * A function for logging warnings.
5
+ *
6
+ * You can assign it to the `AI_SDK_LOG_WARNINGS` global variable to use it as the default warning logger.
7
+ *
8
+ * @example
9
+ * ```ts
10
+ * globalThis.AI_SDK_LOG_WARNINGS = (options) => {
11
+ * console.log('WARNINGS:', options.warnings, options.provider, options.model);
12
+ * };
13
+ * ```
14
+ */
15
+ export type LogWarningsFunction = (options: {
16
+ /**
17
+ * The warnings returned by the model provider.
18
+ */
19
+ warnings: Warning[];
20
+
21
+ /**
22
+ * The provider id used for the call.
23
+ */
24
+ provider: string;
25
+
26
+ /**
27
+ * The model id used for the call.
28
+ */
29
+ model: string;
30
+ }) => void;
31
+
32
+ /**
33
+ * Formats a warning object into a human-readable string with clear AI SDK branding
34
+ */
35
+ function formatWarning({
36
+ warning,
37
+ provider,
38
+ model,
39
+ }: {
40
+ warning: Warning;
41
+ provider: string;
42
+ model: string;
43
+ }): string {
44
+ const prefix = `AI SDK Warning (${provider} / ${model}):`;
45
+
46
+ switch (warning.type) {
47
+ case 'unsupported': {
48
+ let message = `${prefix} The feature "${warning.feature}" is not supported.`;
49
+ if (warning.details) {
50
+ message += ` ${warning.details}`;
51
+ }
52
+ return message;
53
+ }
54
+
55
+ case 'compatibility': {
56
+ let message = `${prefix} The feature "${warning.feature}" is used in a compatibility mode.`;
57
+ if (warning.details) {
58
+ message += ` ${warning.details}`;
59
+ }
60
+ return message;
61
+ }
62
+
63
+ case 'other': {
64
+ return `${prefix} ${warning.message}`;
65
+ }
66
+
67
+ default: {
68
+ // Fallback for any unknown warning types
69
+ return `${prefix} ${JSON.stringify(warning, null, 2)}`;
70
+ }
71
+ }
72
+ }
73
+
74
+ export const FIRST_WARNING_INFO_MESSAGE =
75
+ 'AI SDK Warning System: To turn off warning logging, set the AI_SDK_LOG_WARNINGS global to false.';
76
+
77
+ let hasLoggedBefore = false;
78
+
79
+ export const logWarnings: LogWarningsFunction = options => {
80
+ // if the warnings array is empty, do nothing
81
+ if (options.warnings.length === 0) {
82
+ return;
83
+ }
84
+
85
+ const logger = globalThis.AI_SDK_LOG_WARNINGS;
86
+
87
+ // if the logger is set to false, do nothing
88
+ if (logger === false) {
89
+ return;
90
+ }
91
+
92
+ // use the provided logger if it is a function
93
+ if (typeof logger === 'function') {
94
+ logger(options);
95
+ return;
96
+ }
97
+
98
+ // display information note on first call
99
+ if (!hasLoggedBefore) {
100
+ hasLoggedBefore = true;
101
+ console.info(FIRST_WARNING_INFO_MESSAGE);
102
+ }
103
+
104
+ // default behavior: log warnings to the console
105
+ for (const warning of options.warnings) {
106
+ console.warn(
107
+ formatWarning({
108
+ warning,
109
+ provider: options.provider,
110
+ model: options.model,
111
+ }),
112
+ );
113
+ }
114
+ };
115
+
116
+ // Reset function for testing purposes
117
+ export const resetLogWarningsState = () => {
118
+ hasLoggedBefore = false;
119
+ };
@@ -0,0 +1,64 @@
1
+ // Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
2
+
3
+ exports[`simulateStreamingMiddleware > should handle empty text response 1`] = `
4
+ [
5
+ {
6
+ "type": "start",
7
+ },
8
+ {
9
+ "request": {},
10
+ "type": "start-step",
11
+ "warnings": [],
12
+ },
13
+ {
14
+ "finishReason": "stop",
15
+ "providerMetadata": undefined,
16
+ "rawFinishReason": "stop",
17
+ "response": {
18
+ "headers": undefined,
19
+ "id": "id-6",
20
+ "modelId": "mock-model-id",
21
+ "timestamp": 2025-01-01T00:00:00.000Z,
22
+ },
23
+ "type": "finish-step",
24
+ "usage": {
25
+ "cachedInputTokens": 0,
26
+ "inputTokenDetails": {
27
+ "cacheReadTokens": 0,
28
+ "cacheWriteTokens": 0,
29
+ "noCacheTokens": 5,
30
+ },
31
+ "inputTokens": 5,
32
+ "outputTokenDetails": {
33
+ "reasoningTokens": 3,
34
+ "textTokens": 10,
35
+ },
36
+ "outputTokens": 10,
37
+ "raw": undefined,
38
+ "reasoningTokens": 3,
39
+ "totalTokens": 15,
40
+ },
41
+ },
42
+ {
43
+ "finishReason": "stop",
44
+ "rawFinishReason": "stop",
45
+ "totalUsage": {
46
+ "cachedInputTokens": 0,
47
+ "inputTokenDetails": {
48
+ "cacheReadTokens": 0,
49
+ "cacheWriteTokens": 0,
50
+ "noCacheTokens": 5,
51
+ },
52
+ "inputTokens": 5,
53
+ "outputTokenDetails": {
54
+ "reasoningTokens": 3,
55
+ "textTokens": 10,
56
+ },
57
+ "outputTokens": 10,
58
+ "reasoningTokens": 3,
59
+ "totalTokens": 15,
60
+ },
61
+ "type": "finish",
62
+ },
63
+ ]
64
+ `;