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,442 @@
1
+ import { LanguageModelV3CallOptions } from '@ai-sdk/provider';
2
+ import { convertArrayToReadableStream } from '@ai-sdk/provider-utils/test';
3
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
4
+ import { MockLanguageModelV3 } from '../test/mock-language-model-v3';
5
+ import { ToolLoopAgent } from './tool-loop-agent';
6
+
7
+ describe('ToolLoopAgent', () => {
8
+ describe('generate', () => {
9
+ let doGenerateOptions: LanguageModelV3CallOptions | undefined;
10
+ let mockModel: MockLanguageModelV3;
11
+
12
+ beforeEach(() => {
13
+ doGenerateOptions = undefined;
14
+ mockModel = new MockLanguageModelV3({
15
+ doGenerate: async options => {
16
+ doGenerateOptions = options;
17
+ return {
18
+ content: [{ type: 'text', text: 'reply' }],
19
+ finishReason: { unified: 'stop', raw: 'stop' },
20
+ usage: {
21
+ cachedInputTokens: undefined,
22
+ inputTokens: {
23
+ total: 3,
24
+ noCache: 3,
25
+ cacheRead: undefined,
26
+ cacheWrite: undefined,
27
+ },
28
+ outputTokens: {
29
+ total: 10,
30
+ text: 10,
31
+ reasoning: undefined,
32
+ },
33
+ },
34
+ warnings: [],
35
+ };
36
+ },
37
+ });
38
+ });
39
+
40
+ it('should use prepareCall', async () => {
41
+ const agent = new ToolLoopAgent<{ value: string }>({
42
+ model: mockModel,
43
+ prepareCall: ({ options, ...rest }) => {
44
+ return {
45
+ ...rest,
46
+ providerOptions: {
47
+ test: { value: options.value },
48
+ },
49
+ };
50
+ },
51
+ });
52
+
53
+ await agent.generate({
54
+ prompt: 'Hello, world!',
55
+ options: { value: 'test' },
56
+ });
57
+
58
+ expect(doGenerateOptions?.providerOptions).toMatchInlineSnapshot(`
59
+ {
60
+ "test": {
61
+ "value": "test",
62
+ },
63
+ }
64
+ `);
65
+ });
66
+
67
+ it('should pass abortSignal to generateText', async () => {
68
+ const abortController = new AbortController();
69
+
70
+ const agent = new ToolLoopAgent({ model: mockModel });
71
+
72
+ await agent.generate({
73
+ prompt: 'Hello, world!',
74
+ abortSignal: abortController.signal,
75
+ });
76
+
77
+ expect(doGenerateOptions?.abortSignal).toBe(abortController.signal);
78
+ });
79
+
80
+ it('should pass timeout to generateText', async () => {
81
+ const agent = new ToolLoopAgent({ model: mockModel });
82
+
83
+ await agent.generate({
84
+ prompt: 'Hello, world!',
85
+ timeout: 5000,
86
+ });
87
+
88
+ // timeout is merged into abortSignal, so we check that an abort signal was created
89
+ expect(doGenerateOptions?.abortSignal).toBeDefined();
90
+ });
91
+
92
+ it('should pass experimental_download to generateText', async () => {
93
+ const downloadFunction = vi
94
+ .fn()
95
+ .mockResolvedValue([
96
+ { data: new Uint8Array([1, 2, 3]), mediaType: 'image/png' },
97
+ ]);
98
+
99
+ const agent = new ToolLoopAgent({
100
+ model: mockModel,
101
+ experimental_download: downloadFunction,
102
+ });
103
+
104
+ await agent.generate({
105
+ prompt: [
106
+ {
107
+ role: 'user',
108
+ content: [
109
+ {
110
+ type: 'image',
111
+ image: new URL('https://example.com/image.png'),
112
+ },
113
+ ],
114
+ },
115
+ ],
116
+ });
117
+
118
+ expect(downloadFunction).toHaveBeenCalledWith([
119
+ {
120
+ url: new URL('https://example.com/image.png'),
121
+ isUrlSupportedByModel: false,
122
+ },
123
+ ]);
124
+ });
125
+
126
+ describe('instructions', () => {
127
+ it('should pass string instructions', async () => {
128
+ const agent = new ToolLoopAgent({
129
+ model: mockModel,
130
+ instructions: 'INSTRUCTIONS',
131
+ });
132
+
133
+ await agent.generate({
134
+ prompt: 'Hello, world!',
135
+ });
136
+
137
+ expect(doGenerateOptions?.prompt).toMatchInlineSnapshot(`
138
+ [
139
+ {
140
+ "content": "INSTRUCTIONS",
141
+ "role": "system",
142
+ },
143
+ {
144
+ "content": [
145
+ {
146
+ "text": "Hello, world!",
147
+ "type": "text",
148
+ },
149
+ ],
150
+ "providerOptions": undefined,
151
+ "role": "user",
152
+ },
153
+ ]
154
+ `);
155
+ });
156
+
157
+ it('should pass system message instructions', async () => {
158
+ const agent = new ToolLoopAgent({
159
+ model: mockModel,
160
+ instructions: {
161
+ role: 'system',
162
+ content: 'INSTRUCTIONS',
163
+ providerOptions: { test: { value: 'test' } },
164
+ },
165
+ });
166
+
167
+ await agent.generate({
168
+ prompt: 'Hello, world!',
169
+ });
170
+
171
+ expect(doGenerateOptions?.prompt).toMatchInlineSnapshot(`
172
+ [
173
+ {
174
+ "content": "INSTRUCTIONS",
175
+ "providerOptions": {
176
+ "test": {
177
+ "value": "test",
178
+ },
179
+ },
180
+ "role": "system",
181
+ },
182
+ {
183
+ "content": [
184
+ {
185
+ "text": "Hello, world!",
186
+ "type": "text",
187
+ },
188
+ ],
189
+ "providerOptions": undefined,
190
+ "role": "user",
191
+ },
192
+ ]
193
+ `);
194
+ });
195
+
196
+ it('should pass array of system message instructions', async () => {
197
+ const agent = new ToolLoopAgent({
198
+ model: mockModel,
199
+ instructions: [
200
+ {
201
+ role: 'system',
202
+ content: 'INSTRUCTIONS',
203
+ providerOptions: { test: { value: 'test' } },
204
+ },
205
+ {
206
+ role: 'system',
207
+ content: 'INSTRUCTIONS 2',
208
+ providerOptions: { test: { value: 'test 2' } },
209
+ },
210
+ ],
211
+ });
212
+
213
+ await agent.generate({
214
+ prompt: 'Hello, world!',
215
+ });
216
+
217
+ expect(doGenerateOptions?.prompt).toMatchInlineSnapshot(`
218
+ [
219
+ {
220
+ "content": "INSTRUCTIONS",
221
+ "providerOptions": {
222
+ "test": {
223
+ "value": "test",
224
+ },
225
+ },
226
+ "role": "system",
227
+ },
228
+ {
229
+ "content": "INSTRUCTIONS 2",
230
+ "providerOptions": {
231
+ "test": {
232
+ "value": "test 2",
233
+ },
234
+ },
235
+ "role": "system",
236
+ },
237
+ {
238
+ "content": [
239
+ {
240
+ "text": "Hello, world!",
241
+ "type": "text",
242
+ },
243
+ ],
244
+ "providerOptions": undefined,
245
+ "role": "user",
246
+ },
247
+ ]
248
+ `);
249
+ });
250
+ });
251
+ });
252
+
253
+ describe('stream', () => {
254
+ let doStreamOptions: LanguageModelV3CallOptions | undefined;
255
+ let mockModel: MockLanguageModelV3;
256
+
257
+ beforeEach(() => {
258
+ doStreamOptions = undefined;
259
+ mockModel = new MockLanguageModelV3({
260
+ doStream: async options => {
261
+ doStreamOptions = options;
262
+ return {
263
+ stream: convertArrayToReadableStream([
264
+ {
265
+ type: 'stream-start',
266
+ warnings: [],
267
+ },
268
+ {
269
+ type: 'response-metadata',
270
+ id: 'id-0',
271
+ modelId: 'mock-model-id',
272
+ timestamp: new Date(0),
273
+ },
274
+ { type: 'text-start', id: '1' },
275
+ { type: 'text-delta', id: '1', delta: 'Hello' },
276
+ { type: 'text-delta', id: '1', delta: ', ' },
277
+ { type: 'text-delta', id: '1', delta: `world!` },
278
+ { type: 'text-end', id: '1' },
279
+ {
280
+ type: 'finish',
281
+ finishReason: { unified: 'stop', raw: 'stop' },
282
+ usage: {
283
+ inputTokens: {
284
+ total: 3,
285
+ noCache: 3,
286
+ cacheRead: undefined,
287
+ cacheWrite: undefined,
288
+ },
289
+ outputTokens: {
290
+ total: 10,
291
+ text: 10,
292
+ reasoning: undefined,
293
+ },
294
+ },
295
+ providerMetadata: {
296
+ testProvider: { testKey: 'testValue' },
297
+ },
298
+ },
299
+ ]),
300
+ };
301
+ },
302
+ });
303
+ });
304
+
305
+ it('should use prepareCall', async () => {
306
+ const agent = new ToolLoopAgent<{ value: string }>({
307
+ model: mockModel,
308
+ prepareCall: ({ options, ...rest }) => {
309
+ return {
310
+ ...rest,
311
+ providerOptions: {
312
+ test: { value: options.value },
313
+ },
314
+ };
315
+ },
316
+ });
317
+
318
+ const result = await agent.stream({
319
+ prompt: 'Hello, world!',
320
+ options: { value: 'test' },
321
+ });
322
+
323
+ await result.consumeStream();
324
+
325
+ expect(doStreamOptions?.providerOptions).toMatchInlineSnapshot(
326
+ `
327
+ {
328
+ "test": {
329
+ "value": "test",
330
+ },
331
+ }
332
+ `,
333
+ );
334
+ });
335
+
336
+ it('should pass abortSignal to streamText', async () => {
337
+ const abortController = new AbortController();
338
+
339
+ const agent = new ToolLoopAgent({
340
+ model: mockModel,
341
+ });
342
+
343
+ const result = await agent.stream({
344
+ prompt: 'Hello, world!',
345
+ abortSignal: abortController.signal,
346
+ });
347
+
348
+ await result.consumeStream();
349
+
350
+ expect(doStreamOptions?.abortSignal).toBe(abortController.signal);
351
+ });
352
+
353
+ it('should pass timeout to streamText', async () => {
354
+ const agent = new ToolLoopAgent({
355
+ model: mockModel,
356
+ });
357
+
358
+ const result = await agent.stream({
359
+ prompt: 'Hello, world!',
360
+ timeout: 5000,
361
+ });
362
+
363
+ await result.consumeStream();
364
+
365
+ // timeout is merged into abortSignal, so we check that an abort signal was created
366
+ expect(doStreamOptions?.abortSignal).toBeDefined();
367
+ });
368
+
369
+ it('should pass string instructions', async () => {
370
+ const agent = new ToolLoopAgent({
371
+ model: mockModel,
372
+ instructions: 'INSTRUCTIONS',
373
+ });
374
+
375
+ const result = await agent.stream({
376
+ prompt: 'Hello, world!',
377
+ });
378
+
379
+ await result.consumeStream();
380
+
381
+ expect(doStreamOptions?.prompt).toMatchInlineSnapshot(`
382
+ [
383
+ {
384
+ "content": "INSTRUCTIONS",
385
+ "role": "system",
386
+ },
387
+ {
388
+ "content": [
389
+ {
390
+ "text": "Hello, world!",
391
+ "type": "text",
392
+ },
393
+ ],
394
+ "providerOptions": undefined,
395
+ "role": "user",
396
+ },
397
+ ]
398
+ `);
399
+ });
400
+
401
+ it('should pass system message instructions', async () => {
402
+ const agent = new ToolLoopAgent({
403
+ model: mockModel,
404
+ instructions: {
405
+ role: 'system',
406
+ content: 'INSTRUCTIONS',
407
+ providerOptions: { test: { value: 'test' } },
408
+ },
409
+ });
410
+
411
+ const result = await agent.stream({
412
+ prompt: 'Hello, world!',
413
+ });
414
+
415
+ await result.consumeStream();
416
+
417
+ expect(doStreamOptions?.prompt).toMatchInlineSnapshot(`
418
+ [
419
+ {
420
+ "content": "INSTRUCTIONS",
421
+ "providerOptions": {
422
+ "test": {
423
+ "value": "test",
424
+ },
425
+ },
426
+ "role": "system",
427
+ },
428
+ {
429
+ "content": [
430
+ {
431
+ "text": "Hello, world!",
432
+ "type": "text",
433
+ },
434
+ ],
435
+ "providerOptions": undefined,
436
+ "role": "user",
437
+ },
438
+ ]
439
+ `);
440
+ });
441
+ });
442
+ });
@@ -0,0 +1,114 @@
1
+ import { generateText } from '../generate-text/generate-text';
2
+ import { GenerateTextResult } from '../generate-text/generate-text-result';
3
+ import { Output } from '../generate-text/output';
4
+ import { stepCountIs } from '../generate-text/stop-condition';
5
+ import { streamText } from '../generate-text/stream-text';
6
+ import { StreamTextResult } from '../generate-text/stream-text-result';
7
+ import { ToolSet } from '../generate-text/tool-set';
8
+ import { Prompt } from '../prompt';
9
+ import { Agent, AgentCallParameters, AgentStreamParameters } from './agent';
10
+ import { ToolLoopAgentSettings } from './tool-loop-agent-settings';
11
+
12
+ /**
13
+ * A tool loop agent is an agent that runs tools in a loop. In each step,
14
+ * it calls the LLM, and if there are tool calls, it executes the tools
15
+ * and calls the LLM again in a new step with the tool results.
16
+ *
17
+ * The loop continues until:
18
+ * - A finish reasoning other than tool-calls is returned, or
19
+ * - A tool that is invoked does not have an execute function, or
20
+ * - A tool call needs approval, or
21
+ * - A stop condition is met (default stop condition is stepCountIs(20))
22
+ */
23
+ export class ToolLoopAgent<
24
+ CALL_OPTIONS = never,
25
+ TOOLS extends ToolSet = {},
26
+ OUTPUT extends Output = never,
27
+ > implements Agent<CALL_OPTIONS, TOOLS, OUTPUT>
28
+ {
29
+ readonly version = 'agent-v1';
30
+
31
+ private readonly settings: ToolLoopAgentSettings<CALL_OPTIONS, TOOLS, OUTPUT>;
32
+
33
+ constructor(settings: ToolLoopAgentSettings<CALL_OPTIONS, TOOLS, OUTPUT>) {
34
+ this.settings = settings;
35
+ }
36
+
37
+ /**
38
+ * The id of the agent.
39
+ */
40
+ get id(): string | undefined {
41
+ return this.settings.id;
42
+ }
43
+
44
+ /**
45
+ * The tools that the agent can use.
46
+ */
47
+ get tools(): TOOLS {
48
+ return this.settings.tools as TOOLS;
49
+ }
50
+
51
+ private async prepareCall(
52
+ options: AgentCallParameters<CALL_OPTIONS>,
53
+ ): Promise<
54
+ Omit<
55
+ ToolLoopAgentSettings<CALL_OPTIONS, TOOLS, OUTPUT>,
56
+ 'prepareCall' | 'instructions'
57
+ > &
58
+ Prompt
59
+ > {
60
+ const baseCallArgs = {
61
+ ...this.settings,
62
+ stopWhen: this.settings.stopWhen ?? stepCountIs(20),
63
+ ...options,
64
+ };
65
+
66
+ const preparedCallArgs =
67
+ (await this.settings.prepareCall?.(baseCallArgs)) ?? baseCallArgs;
68
+
69
+ const { instructions, messages, prompt, ...callArgs } = preparedCallArgs;
70
+
71
+ return {
72
+ ...callArgs,
73
+
74
+ // restore prompt types
75
+ ...({ system: instructions, messages, prompt } as Prompt),
76
+ };
77
+ }
78
+
79
+ /**
80
+ * Generates an output from the agent (non-streaming).
81
+ */
82
+ async generate({
83
+ abortSignal,
84
+ timeout,
85
+ ...options
86
+ }: AgentCallParameters<CALL_OPTIONS>): Promise<
87
+ GenerateTextResult<TOOLS, OUTPUT>
88
+ > {
89
+ return generateText({
90
+ ...(await this.prepareCall(options)),
91
+ abortSignal,
92
+ timeout,
93
+ });
94
+ }
95
+
96
+ /**
97
+ * Streams an output from the agent (streaming).
98
+ */
99
+ async stream({
100
+ abortSignal,
101
+ timeout,
102
+ experimental_transform,
103
+ ...options
104
+ }: AgentStreamParameters<CALL_OPTIONS, TOOLS>): Promise<
105
+ StreamTextResult<TOOLS, OUTPUT>
106
+ > {
107
+ return streamText({
108
+ ...(await this.prepareCall(options)),
109
+ abortSignal,
110
+ timeout,
111
+ experimental_transform,
112
+ });
113
+ }
114
+ }