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,476 @@
1
+ import { JSONObject, LanguageModelV3CallOptions } from '@ai-sdk/provider';
2
+ import { addToolInputExamplesMiddleware } from './add-tool-input-examples-middleware';
3
+ import { MockLanguageModelV3 } from '../test/mock-language-model-v3';
4
+ import { describe, it, expect } from 'vitest';
5
+
6
+ const BASE_PARAMS: LanguageModelV3CallOptions = {
7
+ prompt: [
8
+ { role: 'user', content: [{ type: 'text', text: 'Hello, world!' }] },
9
+ ],
10
+ };
11
+
12
+ const MOCK_MODEL = new MockLanguageModelV3();
13
+
14
+ describe('addToolInputExamplesMiddleware', () => {
15
+ describe('transformParams', () => {
16
+ it('should append examples to tool description', async () => {
17
+ const middleware = addToolInputExamplesMiddleware();
18
+
19
+ const result = await middleware.transformParams!({
20
+ type: 'generate',
21
+ params: {
22
+ ...BASE_PARAMS,
23
+ tools: [
24
+ {
25
+ type: 'function',
26
+ name: 'weather',
27
+ description: 'Get the weather in a location',
28
+ inputSchema: {
29
+ type: 'object',
30
+ properties: { location: { type: 'string' } },
31
+ },
32
+ inputExamples: [
33
+ { input: { location: 'San Francisco' } },
34
+ { input: { location: 'London' } },
35
+ ],
36
+ },
37
+ ],
38
+ },
39
+ model: MOCK_MODEL,
40
+ });
41
+
42
+ expect(result.tools).toMatchInlineSnapshot(`
43
+ [
44
+ {
45
+ "description": "Get the weather in a location
46
+
47
+ Input Examples:
48
+ {"location":"San Francisco"}
49
+ {"location":"London"}",
50
+ "inputExamples": undefined,
51
+ "inputSchema": {
52
+ "properties": {
53
+ "location": {
54
+ "type": "string",
55
+ },
56
+ },
57
+ "type": "object",
58
+ },
59
+ "name": "weather",
60
+ "type": "function",
61
+ },
62
+ ]
63
+ `);
64
+ });
65
+
66
+ it('should handle tool without existing description', async () => {
67
+ const middleware = addToolInputExamplesMiddleware();
68
+
69
+ const result = await middleware.transformParams!({
70
+ type: 'generate',
71
+ params: {
72
+ ...BASE_PARAMS,
73
+ tools: [
74
+ {
75
+ type: 'function',
76
+ name: 'weather',
77
+ inputSchema: {
78
+ type: 'object',
79
+ properties: { location: { type: 'string' } },
80
+ },
81
+ inputExamples: [{ input: { location: 'Berlin' } }],
82
+ },
83
+ ],
84
+ },
85
+ model: MOCK_MODEL,
86
+ });
87
+
88
+ expect(result.tools).toMatchInlineSnapshot(`
89
+ [
90
+ {
91
+ "description": "Input Examples:
92
+ {"location":"Berlin"}",
93
+ "inputExamples": undefined,
94
+ "inputSchema": {
95
+ "properties": {
96
+ "location": {
97
+ "type": "string",
98
+ },
99
+ },
100
+ "type": "object",
101
+ },
102
+ "name": "weather",
103
+ "type": "function",
104
+ },
105
+ ]
106
+ `);
107
+ });
108
+ });
109
+
110
+ describe('examplesPrefix', () => {
111
+ it('should use provided examplesPrefix as header', async () => {
112
+ const middleware = addToolInputExamplesMiddleware({
113
+ prefix: 'Here are some example inputs:',
114
+ });
115
+
116
+ const result = await middleware.transformParams!({
117
+ type: 'generate',
118
+ params: {
119
+ ...BASE_PARAMS,
120
+ tools: [
121
+ {
122
+ type: 'function',
123
+ name: 'weather',
124
+ description: 'Get the weather',
125
+ inputSchema: {
126
+ type: 'object',
127
+ properties: { location: { type: 'string' } },
128
+ },
129
+ inputExamples: [{ input: { location: 'Paris' } }],
130
+ },
131
+ ],
132
+ },
133
+ model: MOCK_MODEL,
134
+ });
135
+
136
+ expect((result.tools![0] as any).description).toMatchInlineSnapshot(`
137
+ "Get the weather
138
+
139
+ Here are some example inputs:
140
+ {"location":"Paris"}"
141
+ `);
142
+ });
143
+ });
144
+
145
+ describe('formatExample', () => {
146
+ it('should use default JSON.stringify format', async () => {
147
+ const middleware = addToolInputExamplesMiddleware();
148
+
149
+ const result = await middleware.transformParams!({
150
+ type: 'generate',
151
+ params: {
152
+ ...BASE_PARAMS,
153
+ tools: [
154
+ {
155
+ type: 'function',
156
+ name: 'search',
157
+ description: 'Search for items',
158
+ inputSchema: {
159
+ type: 'object',
160
+ properties: {
161
+ query: { type: 'string' },
162
+ limit: { type: 'number' },
163
+ },
164
+ },
165
+ inputExamples: [{ input: { query: 'test', limit: 10 } }],
166
+ },
167
+ ],
168
+ },
169
+ model: MOCK_MODEL,
170
+ });
171
+
172
+ expect((result.tools![0] as any).description).toMatchInlineSnapshot(`
173
+ "Search for items
174
+
175
+ Input Examples:
176
+ {"query":"test","limit":10}"
177
+ `);
178
+ });
179
+
180
+ it('should use custom format function', async () => {
181
+ const middleware = addToolInputExamplesMiddleware({
182
+ format: (example: { input: JSONObject }, index: number) =>
183
+ `${index + 1}. ${JSON.stringify(example.input)}`,
184
+ });
185
+
186
+ const result = await middleware.transformParams!({
187
+ type: 'generate',
188
+ params: {
189
+ ...BASE_PARAMS,
190
+ tools: [
191
+ {
192
+ type: 'function',
193
+ name: 'weather',
194
+ description: 'Get the weather',
195
+ inputSchema: {
196
+ type: 'object',
197
+ properties: { location: { type: 'string' } },
198
+ },
199
+ inputExamples: [
200
+ { input: { location: 'Paris' } },
201
+ { input: { location: 'Tokyo' } },
202
+ ],
203
+ },
204
+ ],
205
+ },
206
+ model: MOCK_MODEL,
207
+ });
208
+
209
+ expect((result.tools![0] as any).description).toMatchInlineSnapshot(`
210
+ "Get the weather
211
+
212
+ Input Examples:
213
+ 1. {"location":"Paris"}
214
+ 2. {"location":"Tokyo"}"
215
+ `);
216
+ });
217
+ });
218
+
219
+ describe('removeInputExamples', () => {
220
+ it('should remove inputExamples by default', async () => {
221
+ const middleware = addToolInputExamplesMiddleware();
222
+
223
+ const result = await middleware.transformParams!({
224
+ type: 'generate',
225
+ params: {
226
+ ...BASE_PARAMS,
227
+ tools: [
228
+ {
229
+ type: 'function',
230
+ name: 'weather',
231
+ description: 'Get the weather',
232
+ inputSchema: {
233
+ type: 'object',
234
+ properties: { location: { type: 'string' } },
235
+ },
236
+ inputExamples: [{ input: { location: 'NYC' } }],
237
+ },
238
+ ],
239
+ },
240
+ model: MOCK_MODEL,
241
+ });
242
+
243
+ expect((result.tools![0] as any).inputExamples).toBeUndefined();
244
+ });
245
+
246
+ it('should keep inputExamples when remove is false', async () => {
247
+ const middleware = addToolInputExamplesMiddleware({
248
+ remove: false,
249
+ });
250
+
251
+ const result = await middleware.transformParams!({
252
+ type: 'generate',
253
+ params: {
254
+ ...BASE_PARAMS,
255
+ tools: [
256
+ {
257
+ type: 'function',
258
+ name: 'weather',
259
+ description: 'Get the weather',
260
+ inputSchema: {
261
+ type: 'object',
262
+ properties: { location: { type: 'string' } },
263
+ },
264
+ inputExamples: [{ input: { location: 'NYC' } }],
265
+ },
266
+ ],
267
+ },
268
+ model: MOCK_MODEL,
269
+ });
270
+
271
+ expect((result.tools![0] as any).inputExamples).toMatchInlineSnapshot(`
272
+ [
273
+ {
274
+ "input": {
275
+ "location": "NYC",
276
+ },
277
+ },
278
+ ]
279
+ `);
280
+ });
281
+ });
282
+
283
+ describe('edge cases', () => {
284
+ it('should pass through tools without inputExamples', async () => {
285
+ const middleware = addToolInputExamplesMiddleware();
286
+
287
+ const result = await middleware.transformParams!({
288
+ type: 'generate',
289
+ params: {
290
+ ...BASE_PARAMS,
291
+ tools: [
292
+ {
293
+ type: 'function',
294
+ name: 'weather',
295
+ description: 'Get the weather',
296
+ inputSchema: {
297
+ type: 'object',
298
+ properties: { location: { type: 'string' } },
299
+ },
300
+ },
301
+ ],
302
+ },
303
+ model: MOCK_MODEL,
304
+ });
305
+
306
+ expect(result.tools).toMatchInlineSnapshot(`
307
+ [
308
+ {
309
+ "description": "Get the weather",
310
+ "inputSchema": {
311
+ "properties": {
312
+ "location": {
313
+ "type": "string",
314
+ },
315
+ },
316
+ "type": "object",
317
+ },
318
+ "name": "weather",
319
+ "type": "function",
320
+ },
321
+ ]
322
+ `);
323
+ });
324
+
325
+ it('should pass through tools with empty inputExamples array', async () => {
326
+ const middleware = addToolInputExamplesMiddleware();
327
+
328
+ const result = await middleware.transformParams!({
329
+ type: 'generate',
330
+ params: {
331
+ ...BASE_PARAMS,
332
+ tools: [
333
+ {
334
+ type: 'function',
335
+ name: 'weather',
336
+ description: 'Get the weather',
337
+ inputSchema: {
338
+ type: 'object',
339
+ properties: { location: { type: 'string' } },
340
+ },
341
+ inputExamples: [],
342
+ },
343
+ ],
344
+ },
345
+ model: MOCK_MODEL,
346
+ });
347
+
348
+ expect((result.tools![0] as any).description).toBe('Get the weather');
349
+ });
350
+
351
+ it('should pass through provider tools unchanged', async () => {
352
+ const middleware = addToolInputExamplesMiddleware();
353
+
354
+ const result = await middleware.transformParams!({
355
+ type: 'generate',
356
+ params: {
357
+ ...BASE_PARAMS,
358
+ tools: [
359
+ {
360
+ type: 'provider',
361
+ name: 'web_search',
362
+ id: 'anthropic.web_search_20250305',
363
+ args: { maxUses: 5 },
364
+ },
365
+ ],
366
+ },
367
+ model: MOCK_MODEL,
368
+ });
369
+
370
+ expect(result.tools).toMatchInlineSnapshot(`
371
+ [
372
+ {
373
+ "args": {
374
+ "maxUses": 5,
375
+ },
376
+ "id": "anthropic.web_search_20250305",
377
+ "name": "web_search",
378
+ "type": "provider",
379
+ },
380
+ ]
381
+ `);
382
+ });
383
+
384
+ it('should handle multiple tools with mixed examples', async () => {
385
+ const middleware = addToolInputExamplesMiddleware();
386
+
387
+ const result = await middleware.transformParams!({
388
+ type: 'generate',
389
+ params: {
390
+ ...BASE_PARAMS,
391
+ tools: [
392
+ {
393
+ type: 'function',
394
+ name: 'weather',
395
+ description: 'Get the weather',
396
+ inputSchema: {
397
+ type: 'object',
398
+ properties: { location: { type: 'string' } },
399
+ },
400
+ inputExamples: [{ input: { location: 'NYC' } }],
401
+ },
402
+ {
403
+ type: 'function',
404
+ name: 'time',
405
+ description: 'Get the current time',
406
+ inputSchema: {
407
+ type: 'object',
408
+ properties: { timezone: { type: 'string' } },
409
+ },
410
+ },
411
+ ],
412
+ },
413
+ model: MOCK_MODEL,
414
+ });
415
+
416
+ expect(result.tools).toMatchInlineSnapshot(`
417
+ [
418
+ {
419
+ "description": "Get the weather
420
+
421
+ Input Examples:
422
+ {"location":"NYC"}",
423
+ "inputExamples": undefined,
424
+ "inputSchema": {
425
+ "properties": {
426
+ "location": {
427
+ "type": "string",
428
+ },
429
+ },
430
+ "type": "object",
431
+ },
432
+ "name": "weather",
433
+ "type": "function",
434
+ },
435
+ {
436
+ "description": "Get the current time",
437
+ "inputSchema": {
438
+ "properties": {
439
+ "timezone": {
440
+ "type": "string",
441
+ },
442
+ },
443
+ "type": "object",
444
+ },
445
+ "name": "time",
446
+ "type": "function",
447
+ },
448
+ ]
449
+ `);
450
+ });
451
+
452
+ it('should handle empty tools array', async () => {
453
+ const middleware = addToolInputExamplesMiddleware();
454
+
455
+ const result = await middleware.transformParams!({
456
+ type: 'generate',
457
+ params: { ...BASE_PARAMS, tools: [] },
458
+ model: MOCK_MODEL,
459
+ });
460
+
461
+ expect(result.tools).toEqual([]);
462
+ });
463
+
464
+ it('should handle undefined tools', async () => {
465
+ const middleware = addToolInputExamplesMiddleware();
466
+
467
+ const result = await middleware.transformParams!({
468
+ type: 'generate',
469
+ params: { ...BASE_PARAMS, tools: undefined },
470
+ model: MOCK_MODEL,
471
+ });
472
+
473
+ expect(result.tools).toBeUndefined();
474
+ });
475
+ });
476
+ });
@@ -0,0 +1,90 @@
1
+ import { JSONObject, LanguageModelV3FunctionTool } from '@ai-sdk/provider';
2
+ import { LanguageModelMiddleware } from '../types';
3
+
4
+ function defaultFormatExample(example: { input: JSONObject }): string {
5
+ return JSON.stringify(example.input);
6
+ }
7
+
8
+ /**
9
+ * Middleware that appends input examples to tool descriptions.
10
+ *
11
+ * This is useful for providers that don't natively support the `inputExamples`
12
+ * property. The middleware serializes examples into the tool's description text.
13
+ *
14
+ * @param options - Configuration options for the middleware.
15
+ * @param options.prefix - A prefix to prepend before the examples. Default: 'Input Examples:'
16
+ * @param options.format - Optional custom formatter for each example.
17
+ * Receives the example object and its index. Default: JSON.stringify(example.input)
18
+ * @param options.remove - Whether to remove the inputExamples property
19
+ * after adding them to the description. Default: true
20
+ *
21
+ * @example
22
+ * ```ts
23
+ * import { wrapLanguageModel, addToolInputExamplesMiddleware } from 'ai';
24
+ *
25
+ * const model = wrapLanguageModel({
26
+ * model: yourModel,
27
+ * middleware: addToolInputExamplesMiddleware(),
28
+ * });
29
+ * ```
30
+ */
31
+ export function addToolInputExamplesMiddleware({
32
+ prefix = 'Input Examples:',
33
+ format = defaultFormatExample,
34
+ remove = true,
35
+ }: {
36
+ /**
37
+ * A prefix to prepend before the examples.
38
+ */
39
+ prefix?: string;
40
+
41
+ /**
42
+ * Optional custom formatter for each example.
43
+ * Receives the example object and its index.
44
+ * Default: JSON.stringify(example.input)
45
+ */
46
+ format?: (example: { input: JSONObject }, index: number) => string;
47
+
48
+ /**
49
+ * Whether to remove the inputExamples property after adding them to the description.
50
+ * Default: true
51
+ */
52
+ remove?: boolean;
53
+ } = {}): LanguageModelMiddleware {
54
+ return {
55
+ specificationVersion: 'v3',
56
+ transformParams: async ({ params }) => {
57
+ if (!params.tools?.length) {
58
+ return params;
59
+ }
60
+
61
+ const transformedTools = params.tools.map(tool => {
62
+ // Only transform function tools that have inputExamples
63
+ if (tool.type !== 'function' || !tool.inputExamples?.length) {
64
+ return tool;
65
+ }
66
+
67
+ const formattedExamples = tool.inputExamples
68
+ .map((example, index) => format(example, index))
69
+ .join('\n');
70
+
71
+ const examplesSection = `${prefix}\n${formattedExamples}`;
72
+
73
+ const toolDescription = tool.description
74
+ ? `${tool.description}\n\n${examplesSection}`
75
+ : examplesSection;
76
+
77
+ return {
78
+ ...tool,
79
+ description: toolDescription,
80
+ inputExamples: remove ? undefined : tool.inputExamples,
81
+ } satisfies LanguageModelV3FunctionTool;
82
+ });
83
+
84
+ return {
85
+ ...params,
86
+ tools: transformedTools,
87
+ };
88
+ },
89
+ };
90
+ }
@@ -0,0 +1,126 @@
1
+ import { EmbeddingModelV3CallOptions } from '@ai-sdk/provider';
2
+ import { defaultEmbeddingSettingsMiddleware } from './default-embedding-settings-middleware';
3
+ import { MockEmbeddingModelV3 } from '../test/mock-embedding-model-v3';
4
+ import { describe, it, expect } from 'vitest';
5
+
6
+ const params: EmbeddingModelV3CallOptions = {
7
+ values: ['hello world'],
8
+ };
9
+
10
+ const mockModel = new MockEmbeddingModelV3();
11
+
12
+ describe('headers', () => {
13
+ it('should merge headers', async () => {
14
+ const middleware = defaultEmbeddingSettingsMiddleware({
15
+ settings: {
16
+ headers: { 'X-Custom-Header': 'test', 'X-Another-Header': 'test2' },
17
+ },
18
+ });
19
+ const result = await middleware.transformParams!({
20
+ params: {
21
+ ...params,
22
+ headers: { 'X-Custom-Header': 'test2' },
23
+ },
24
+ model: mockModel,
25
+ });
26
+ expect(result.headers).toEqual({
27
+ 'X-Custom-Header': 'test2',
28
+ 'X-Another-Header': 'test2',
29
+ });
30
+ });
31
+
32
+ it('should handle empty default headers', async () => {
33
+ const middleware = defaultEmbeddingSettingsMiddleware({
34
+ settings: { headers: {} },
35
+ });
36
+ const result = await middleware.transformParams!({
37
+ params: { ...params, headers: { 'X-Param-Header': 'param' } },
38
+ model: mockModel,
39
+ });
40
+ expect(result.headers).toEqual({ 'X-Param-Header': 'param' });
41
+ });
42
+
43
+ it('should handle empty param headers', async () => {
44
+ const middleware = defaultEmbeddingSettingsMiddleware({
45
+ settings: { headers: { 'X-Default-Header': 'default' } },
46
+ });
47
+ const result = await middleware.transformParams!({
48
+ params: { ...params, headers: {} },
49
+ model: mockModel,
50
+ });
51
+ expect(result.headers).toEqual({ 'X-Default-Header': 'default' });
52
+ });
53
+
54
+ it('should handle both headers being undefined', async () => {
55
+ const middleware = defaultEmbeddingSettingsMiddleware({
56
+ settings: {},
57
+ });
58
+ const result = await middleware.transformParams!({
59
+ params: { ...params },
60
+ model: mockModel,
61
+ });
62
+ expect(result.headers).toBeUndefined();
63
+ });
64
+ });
65
+
66
+ describe('providerOptions', () => {
67
+ it('should handle empty default providerOptions', async () => {
68
+ const middleware = defaultEmbeddingSettingsMiddleware({
69
+ settings: {
70
+ providerOptions: {},
71
+ },
72
+ });
73
+ const result = await middleware.transformParams!({
74
+ params: {
75
+ ...params,
76
+ providerOptions: {
77
+ google: {
78
+ outputDimensionality: 512,
79
+ taskType: 'SEMANTIC_SIMILARITY',
80
+ },
81
+ },
82
+ },
83
+ model: mockModel,
84
+ });
85
+ expect(result.providerOptions).toEqual({
86
+ google: {
87
+ outputDimensionality: 512,
88
+ taskType: 'SEMANTIC_SIMILARITY',
89
+ },
90
+ });
91
+ });
92
+
93
+ it('should handle empty param providerOptions', async () => {
94
+ const middleware = defaultEmbeddingSettingsMiddleware({
95
+ settings: {
96
+ providerOptions: {
97
+ google: {
98
+ outputDimensionality: 512,
99
+ taskType: 'SEMANTIC_SIMILARITY',
100
+ },
101
+ },
102
+ },
103
+ });
104
+ const result = await middleware.transformParams!({
105
+ params: { ...params, providerOptions: {} },
106
+ model: mockModel,
107
+ });
108
+ expect(result.providerOptions).toEqual({
109
+ google: {
110
+ outputDimensionality: 512,
111
+ taskType: 'SEMANTIC_SIMILARITY',
112
+ },
113
+ });
114
+ });
115
+
116
+ it('should handle both providerOptions being undefined', async () => {
117
+ const middleware = defaultEmbeddingSettingsMiddleware({
118
+ settings: {},
119
+ });
120
+ const result = await middleware.transformParams!({
121
+ params: { ...params },
122
+ model: mockModel,
123
+ });
124
+ expect(result.providerOptions).toBeUndefined();
125
+ });
126
+ });
@@ -0,0 +1,22 @@
1
+ import { EmbeddingModelV3CallOptions } from '@ai-sdk/provider';
2
+ import { EmbeddingModelMiddleware } from '../types';
3
+ import { mergeObjects } from '../util/merge-objects';
4
+
5
+ /**
6
+ * Applies default settings for a embedding model.
7
+ */
8
+ export function defaultEmbeddingSettingsMiddleware({
9
+ settings,
10
+ }: {
11
+ settings: Partial<{
12
+ headers?: EmbeddingModelV3CallOptions['headers'];
13
+ providerOptions?: EmbeddingModelV3CallOptions['providerOptions'];
14
+ }>;
15
+ }): EmbeddingModelMiddleware {
16
+ return {
17
+ specificationVersion: 'v3',
18
+ transformParams: async ({ params }) => {
19
+ return mergeObjects(settings, params) as EmbeddingModelV3CallOptions;
20
+ },
21
+ };
22
+ }