opencode-v2 1.1.53

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 (439) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +5 -0
  6. package/package.json +126 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +193 -0
  9. package/script/postinstall.mjs +125 -0
  10. package/script/publish.ts +181 -0
  11. package/script/schema.ts +47 -0
  12. package/script/seed-e2e.ts +50 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +1676 -0
  15. package/src/acp/session.ts +117 -0
  16. package/src/acp/types.ts +23 -0
  17. package/src/agent/agent.ts +414 -0
  18. package/src/agent/generate.txt +75 -0
  19. package/src/agent/prompt/compaction.txt +12 -0
  20. package/src/agent/prompt/explore.txt +18 -0
  21. package/src/agent/prompt/summary.txt +11 -0
  22. package/src/agent/prompt/title.txt +44 -0
  23. package/src/auth/index.ts +70 -0
  24. package/src/bun/index.ts +137 -0
  25. package/src/bun/registry.ts +48 -0
  26. package/src/bus/bus-event.ts +43 -0
  27. package/src/bus/global.ts +10 -0
  28. package/src/bus/index.ts +105 -0
  29. package/src/cli/bootstrap.ts +17 -0
  30. package/src/cli/cmd/acp.ts +70 -0
  31. package/src/cli/cmd/agent.ts +257 -0
  32. package/src/cli/cmd/auth.ts +400 -0
  33. package/src/cli/cmd/cmd.ts +7 -0
  34. package/src/cli/cmd/debug/agent.ts +167 -0
  35. package/src/cli/cmd/debug/config.ts +16 -0
  36. package/src/cli/cmd/debug/file.ts +97 -0
  37. package/src/cli/cmd/debug/index.ts +48 -0
  38. package/src/cli/cmd/debug/lsp.ts +52 -0
  39. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  40. package/src/cli/cmd/debug/scrap.ts +16 -0
  41. package/src/cli/cmd/debug/skill.ts +16 -0
  42. package/src/cli/cmd/debug/snapshot.ts +52 -0
  43. package/src/cli/cmd/export.ts +88 -0
  44. package/src/cli/cmd/generate.ts +38 -0
  45. package/src/cli/cmd/github.ts +1540 -0
  46. package/src/cli/cmd/import.ts +147 -0
  47. package/src/cli/cmd/mcp.ts +755 -0
  48. package/src/cli/cmd/models.ts +77 -0
  49. package/src/cli/cmd/pr.ts +112 -0
  50. package/src/cli/cmd/run.ts +617 -0
  51. package/src/cli/cmd/serve.ts +20 -0
  52. package/src/cli/cmd/session.ts +135 -0
  53. package/src/cli/cmd/stats.ts +426 -0
  54. package/src/cli/cmd/tui/app.tsx +801 -0
  55. package/src/cli/cmd/tui/attach.ts +52 -0
  56. package/src/cli/cmd/tui/component/border.tsx +21 -0
  57. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  58. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  59. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  60. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  61. package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
  62. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  63. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  64. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  70. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  71. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  76. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  77. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  78. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  79. package/src/cli/cmd/tui/context/args.tsx +15 -0
  80. package/src/cli/cmd/tui/context/directory.ts +13 -0
  81. package/src/cli/cmd/tui/context/exit.tsx +52 -0
  82. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  83. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  84. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  85. package/src/cli/cmd/tui/context/local.tsx +409 -0
  86. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  87. package/src/cli/cmd/tui/context/route.tsx +46 -0
  88. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  89. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  90. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  91. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  92. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  96. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  97. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  98. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  99. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  100. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  101. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  102. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  103. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  105. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  106. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  107. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  108. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  110. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  111. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  112. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  113. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  114. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  115. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  116. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  117. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  118. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  119. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  120. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  121. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  122. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  123. package/src/cli/cmd/tui/event.ts +48 -0
  124. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  129. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  130. package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
  131. package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
  132. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  133. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  134. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  135. package/src/cli/cmd/tui/thread.ts +175 -0
  136. package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
  137. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
  138. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
  139. package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
  140. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
  141. package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
  142. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  143. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  144. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  145. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  146. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  147. package/src/cli/cmd/tui/util/editor.ts +32 -0
  148. package/src/cli/cmd/tui/util/signal.ts +7 -0
  149. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  150. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  151. package/src/cli/cmd/tui/worker.ts +152 -0
  152. package/src/cli/cmd/uninstall.ts +357 -0
  153. package/src/cli/cmd/upgrade.ts +73 -0
  154. package/src/cli/cmd/web.ts +81 -0
  155. package/src/cli/error.ts +57 -0
  156. package/src/cli/logo.ts +6 -0
  157. package/src/cli/network.ts +60 -0
  158. package/src/cli/ui.ts +113 -0
  159. package/src/cli/upgrade.ts +25 -0
  160. package/src/command/index.ts +150 -0
  161. package/src/command/template/initialize.txt +10 -0
  162. package/src/command/template/review.txt +99 -0
  163. package/src/config/config.ts +1477 -0
  164. package/src/config/markdown.ts +98 -0
  165. package/src/env/index.ts +28 -0
  166. package/src/file/ignore.ts +83 -0
  167. package/src/file/index.ts +583 -0
  168. package/src/file/ripgrep.ts +375 -0
  169. package/src/file/time.ts +69 -0
  170. package/src/file/watcher.ts +127 -0
  171. package/src/flag/flag.ts +97 -0
  172. package/src/format/formatter.ts +366 -0
  173. package/src/format/index.ts +137 -0
  174. package/src/global/index.ts +55 -0
  175. package/src/id/id.ts +83 -0
  176. package/src/ide/index.ts +76 -0
  177. package/src/index.ts +159 -0
  178. package/src/installation/index.ts +246 -0
  179. package/src/lsp/client.ts +252 -0
  180. package/src/lsp/index.ts +485 -0
  181. package/src/lsp/language.ts +119 -0
  182. package/src/lsp/server.ts +2046 -0
  183. package/src/mcp/auth.ts +132 -0
  184. package/src/mcp/index.ts +934 -0
  185. package/src/mcp/oauth-callback.ts +200 -0
  186. package/src/mcp/oauth-provider.ts +154 -0
  187. package/src/patch/index.ts +680 -0
  188. package/src/permission/arity.ts +163 -0
  189. package/src/permission/index.ts +210 -0
  190. package/src/permission/next.ts +280 -0
  191. package/src/plugin/codex.ts +624 -0
  192. package/src/plugin/copilot.ts +327 -0
  193. package/src/plugin/index.ts +138 -0
  194. package/src/project/bootstrap.ts +35 -0
  195. package/src/project/instance.ts +114 -0
  196. package/src/project/project.ts +371 -0
  197. package/src/project/state.ts +70 -0
  198. package/src/project/vcs.ts +76 -0
  199. package/src/provider/auth.ts +147 -0
  200. package/src/provider/models.ts +133 -0
  201. package/src/provider/provider.ts +1262 -0
  202. package/src/provider/sdk/copilot/README.md +5 -0
  203. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  204. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  205. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  206. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  207. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  208. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  209. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  211. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  212. package/src/provider/sdk/copilot/index.ts +2 -0
  213. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  214. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  215. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  216. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  217. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  218. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  219. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  220. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  221. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  222. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  223. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  224. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  225. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  226. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  227. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  228. package/src/provider/transform.ts +828 -0
  229. package/src/pty/index.ts +250 -0
  230. package/src/question/index.ts +171 -0
  231. package/src/scheduler/index.ts +61 -0
  232. package/src/server/error.ts +36 -0
  233. package/src/server/event.ts +7 -0
  234. package/src/server/mdns.ts +60 -0
  235. package/src/server/routes/config.ts +92 -0
  236. package/src/server/routes/experimental.ts +208 -0
  237. package/src/server/routes/file.ts +197 -0
  238. package/src/server/routes/global.ts +183 -0
  239. package/src/server/routes/mcp.ts +225 -0
  240. package/src/server/routes/permission.ts +68 -0
  241. package/src/server/routes/project.ts +82 -0
  242. package/src/server/routes/provider.ts +165 -0
  243. package/src/server/routes/pty.ts +169 -0
  244. package/src/server/routes/question.ts +98 -0
  245. package/src/server/routes/session.ts +939 -0
  246. package/src/server/routes/tui.ts +379 -0
  247. package/src/server/server.ts +613 -0
  248. package/src/session/compaction.ts +226 -0
  249. package/src/session/index.ts +524 -0
  250. package/src/session/instruction.ts +197 -0
  251. package/src/session/llm.ts +289 -0
  252. package/src/session/message-v2.ts +802 -0
  253. package/src/session/message.ts +189 -0
  254. package/src/session/processor.ts +407 -0
  255. package/src/session/prompt/agent.txt +43 -0
  256. package/src/session/prompt/anthropic-20250930.txt +166 -0
  257. package/src/session/prompt/anthropic.txt +105 -0
  258. package/src/session/prompt/beast.txt +147 -0
  259. package/src/session/prompt/build-switch.txt +5 -0
  260. package/src/session/prompt/codex_header.txt +79 -0
  261. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  262. package/src/session/prompt/gemini.txt +155 -0
  263. package/src/session/prompt/max-steps.txt +16 -0
  264. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  265. package/src/session/prompt/plan.txt +26 -0
  266. package/src/session/prompt/qwen.txt +109 -0
  267. package/src/session/prompt/research.txt +81 -0
  268. package/src/session/prompt/trinity.txt +97 -0
  269. package/src/session/prompt.ts +1952 -0
  270. package/src/session/retry.ts +97 -0
  271. package/src/session/revert.ts +121 -0
  272. package/src/session/status.ts +76 -0
  273. package/src/session/summary.ts +217 -0
  274. package/src/session/system.ts +54 -0
  275. package/src/session/todo.ts +37 -0
  276. package/src/share/share-next.ts +200 -0
  277. package/src/share/share.ts +92 -0
  278. package/src/shell/shell.ts +67 -0
  279. package/src/skill/discovery.ts +97 -0
  280. package/src/skill/index.ts +1 -0
  281. package/src/skill/skill.ts +188 -0
  282. package/src/snapshot/index.ts +255 -0
  283. package/src/storage/storage.ts +227 -0
  284. package/src/tool/agent-enter.txt +1 -0
  285. package/src/tool/agent-exit.txt +1 -0
  286. package/src/tool/agent.ts +237 -0
  287. package/src/tool/apply_patch.ts +281 -0
  288. package/src/tool/apply_patch.txt +33 -0
  289. package/src/tool/bash.ts +269 -0
  290. package/src/tool/bash.txt +115 -0
  291. package/src/tool/batch.ts +175 -0
  292. package/src/tool/batch.txt +24 -0
  293. package/src/tool/chat-enter.txt +15 -0
  294. package/src/tool/chat-exit.txt +7 -0
  295. package/src/tool/chat.ts +217 -0
  296. package/src/tool/codesearch.ts +132 -0
  297. package/src/tool/codesearch.txt +12 -0
  298. package/src/tool/edit.ts +655 -0
  299. package/src/tool/edit.txt +10 -0
  300. package/src/tool/external-directory.ts +32 -0
  301. package/src/tool/glob.ts +78 -0
  302. package/src/tool/glob.txt +6 -0
  303. package/src/tool/grep.ts +147 -0
  304. package/src/tool/grep.txt +8 -0
  305. package/src/tool/invalid.ts +17 -0
  306. package/src/tool/ls.ts +121 -0
  307. package/src/tool/ls.txt +1 -0
  308. package/src/tool/lsp.ts +96 -0
  309. package/src/tool/lsp.txt +19 -0
  310. package/src/tool/multiedit.ts +46 -0
  311. package/src/tool/multiedit.txt +41 -0
  312. package/src/tool/plan-enter.txt +14 -0
  313. package/src/tool/plan-exit.txt +13 -0
  314. package/src/tool/plan.ts +130 -0
  315. package/src/tool/question.ts +33 -0
  316. package/src/tool/question.txt +10 -0
  317. package/src/tool/read.ts +211 -0
  318. package/src/tool/read.txt +12 -0
  319. package/src/tool/registry.ts +167 -0
  320. package/src/tool/research-enter.txt +1 -0
  321. package/src/tool/research-exit.txt +1 -0
  322. package/src/tool/research.ts +134 -0
  323. package/src/tool/skill.ts +123 -0
  324. package/src/tool/task.ts +165 -0
  325. package/src/tool/task.txt +60 -0
  326. package/src/tool/todo.ts +53 -0
  327. package/src/tool/todoread.txt +14 -0
  328. package/src/tool/todowrite.txt +167 -0
  329. package/src/tool/tool.ts +89 -0
  330. package/src/tool/truncation.ts +106 -0
  331. package/src/tool/webfetch.ts +186 -0
  332. package/src/tool/webfetch.txt +13 -0
  333. package/src/tool/websearch.ts +150 -0
  334. package/src/tool/websearch.txt +14 -0
  335. package/src/tool/write.ts +85 -0
  336. package/src/tool/write.txt +8 -0
  337. package/src/util/abort.ts +35 -0
  338. package/src/util/archive.ts +16 -0
  339. package/src/util/color.ts +19 -0
  340. package/src/util/context.ts +25 -0
  341. package/src/util/defer.ts +12 -0
  342. package/src/util/eventloop.ts +20 -0
  343. package/src/util/filesystem.ts +93 -0
  344. package/src/util/fn.ts +11 -0
  345. package/src/util/format.ts +20 -0
  346. package/src/util/iife.ts +3 -0
  347. package/src/util/keybind.ts +103 -0
  348. package/src/util/lazy.ts +18 -0
  349. package/src/util/locale.ts +81 -0
  350. package/src/util/lock.ts +98 -0
  351. package/src/util/log.ts +180 -0
  352. package/src/util/proxied.ts +3 -0
  353. package/src/util/queue.ts +32 -0
  354. package/src/util/rpc.ts +66 -0
  355. package/src/util/scrap.ts +10 -0
  356. package/src/util/signal.ts +12 -0
  357. package/src/util/timeout.ts +14 -0
  358. package/src/util/token.ts +7 -0
  359. package/src/util/wildcard.ts +56 -0
  360. package/src/worktree/index.ts +574 -0
  361. package/sst-env.d.ts +9 -0
  362. package/test/acp/agent-interface.test.ts +51 -0
  363. package/test/acp/event-subscription.test.ts +436 -0
  364. package/test/agent/agent.test.ts +675 -0
  365. package/test/bun.test.ts +53 -0
  366. package/test/cli/github-action.test.ts +161 -0
  367. package/test/cli/github-remote.test.ts +80 -0
  368. package/test/cli/import.test.ts +38 -0
  369. package/test/cli/tui/transcript.test.ts +322 -0
  370. package/test/config/agent-color.test.ts +71 -0
  371. package/test/config/config.test.ts +1802 -0
  372. package/test/config/fixtures/empty-frontmatter.md +4 -0
  373. package/test/config/fixtures/frontmatter.md +28 -0
  374. package/test/config/fixtures/markdown-header.md +11 -0
  375. package/test/config/fixtures/no-frontmatter.md +1 -0
  376. package/test/config/fixtures/weird-model-id.md +13 -0
  377. package/test/config/markdown.test.ts +228 -0
  378. package/test/file/ignore.test.ts +10 -0
  379. package/test/file/path-traversal.test.ts +198 -0
  380. package/test/file/ripgrep.test.ts +39 -0
  381. package/test/fixture/fixture.ts +45 -0
  382. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  383. package/test/ide/ide.test.ts +82 -0
  384. package/test/keybind.test.ts +421 -0
  385. package/test/lsp/client.test.ts +95 -0
  386. package/test/mcp/headers.test.ts +153 -0
  387. package/test/mcp/oauth-browser.test.ts +249 -0
  388. package/test/memory/abort-leak.test.ts +136 -0
  389. package/test/patch/patch.test.ts +348 -0
  390. package/test/permission/arity.test.ts +33 -0
  391. package/test/permission/next.test.ts +690 -0
  392. package/test/permission-task.test.ts +319 -0
  393. package/test/plugin/auth-override.test.ts +44 -0
  394. package/test/plugin/codex.test.ts +123 -0
  395. package/test/preload.ts +63 -0
  396. package/test/project/project.test.ts +120 -0
  397. package/test/provider/amazon-bedrock.test.ts +445 -0
  398. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  399. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  400. package/test/provider/gitlab-duo.test.ts +262 -0
  401. package/test/provider/provider.test.ts +2129 -0
  402. package/test/provider/transform.test.ts +2022 -0
  403. package/test/question/question.test.ts +300 -0
  404. package/test/scheduler.test.ts +73 -0
  405. package/test/server/session-list.test.ts +39 -0
  406. package/test/server/session-select.test.ts +78 -0
  407. package/test/session/compaction.test.ts +293 -0
  408. package/test/session/instruction.test.ts +170 -0
  409. package/test/session/llm.test.ts +691 -0
  410. package/test/session/message-v2.test.ts +786 -0
  411. package/test/session/prompt-missing-file.test.ts +53 -0
  412. package/test/session/prompt-special-chars.test.ts +56 -0
  413. package/test/session/prompt-variant.test.ts +60 -0
  414. package/test/session/retry.test.ts +179 -0
  415. package/test/session/revert-compact.test.ts +285 -0
  416. package/test/session/session.test.ts +71 -0
  417. package/test/skill/discovery.test.ts +60 -0
  418. package/test/skill/skill.test.ts +388 -0
  419. package/test/snapshot/snapshot.test.ts +1040 -0
  420. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  421. package/test/tool/apply_patch.test.ts +559 -0
  422. package/test/tool/bash.test.ts +399 -0
  423. package/test/tool/external-directory.test.ts +127 -0
  424. package/test/tool/fixtures/large-image.png +0 -0
  425. package/test/tool/fixtures/models-api.json +38413 -0
  426. package/test/tool/grep.test.ts +110 -0
  427. package/test/tool/question.test.ts +107 -0
  428. package/test/tool/read.test.ts +358 -0
  429. package/test/tool/registry.test.ts +122 -0
  430. package/test/tool/skill.test.ts +112 -0
  431. package/test/tool/truncation.test.ts +159 -0
  432. package/test/util/filesystem.test.ts +39 -0
  433. package/test/util/format.test.ts +59 -0
  434. package/test/util/iife.test.ts +36 -0
  435. package/test/util/lazy.test.ts +50 -0
  436. package/test/util/lock.test.ts +72 -0
  437. package/test/util/timeout.test.ts +21 -0
  438. package/test/util/wildcard.test.ts +75 -0
  439. package/tsconfig.json +16 -0
@@ -0,0 +1,28 @@
1
+ import { z } from "zod/v4"
2
+
3
+ export type OpenAICompatibleChatModelId = string
4
+
5
+ export const openaiCompatibleProviderOptions = z.object({
6
+ /**
7
+ * A unique identifier representing your end-user, which can help the provider to
8
+ * monitor and detect abuse.
9
+ */
10
+ user: z.string().optional(),
11
+
12
+ /**
13
+ * Reasoning effort for reasoning models. Defaults to `medium`.
14
+ */
15
+ reasoningEffort: z.string().optional(),
16
+
17
+ /**
18
+ * Controls the verbosity of the generated text. Defaults to `medium`.
19
+ */
20
+ textVerbosity: z.string().optional(),
21
+
22
+ /**
23
+ * Copilot thinking_budget used for Anthropic models.
24
+ */
25
+ thinking_budget: z.number().optional(),
26
+ })
27
+
28
+ export type OpenAICompatibleProviderOptions = z.infer<typeof openaiCompatibleProviderOptions>
@@ -0,0 +1,44 @@
1
+ import type { SharedV2ProviderMetadata } from "@ai-sdk/provider"
2
+
3
+ /**
4
+ Extracts provider-specific metadata from API responses.
5
+ Used to standardize metadata handling across different LLM providers while allowing
6
+ provider-specific metadata to be captured.
7
+ */
8
+ export type MetadataExtractor = {
9
+ /**
10
+ * Extracts provider metadata from a complete, non-streaming response.
11
+ *
12
+ * @param parsedBody - The parsed response JSON body from the provider's API.
13
+ *
14
+ * @returns Provider-specific metadata or undefined if no metadata is available.
15
+ * The metadata should be under a key indicating the provider id.
16
+ */
17
+ extractMetadata: ({ parsedBody }: { parsedBody: unknown }) => Promise<SharedV2ProviderMetadata | undefined>
18
+
19
+ /**
20
+ * Creates an extractor for handling streaming responses. The returned object provides
21
+ * methods to process individual chunks and build the final metadata from the accumulated
22
+ * stream data.
23
+ *
24
+ * @returns An object with methods to process chunks and build metadata from a stream
25
+ */
26
+ createStreamExtractor: () => {
27
+ /**
28
+ * Process an individual chunk from the stream. Called for each chunk in the response stream
29
+ * to accumulate metadata throughout the streaming process.
30
+ *
31
+ * @param parsedChunk - The parsed JSON response chunk from the provider's API
32
+ */
33
+ processChunk(parsedChunk: unknown): void
34
+
35
+ /**
36
+ * Builds the metadata object after all chunks have been processed.
37
+ * Called at the end of the stream to generate the complete provider metadata.
38
+ *
39
+ * @returns Provider-specific metadata or undefined if no metadata is available.
40
+ * The metadata should be under a key indicating the provider id.
41
+ */
42
+ buildMetadata(): SharedV2ProviderMetadata | undefined
43
+ }
44
+ }
@@ -0,0 +1,87 @@
1
+ import {
2
+ type LanguageModelV2CallOptions,
3
+ type LanguageModelV2CallWarning,
4
+ UnsupportedFunctionalityError,
5
+ } from "@ai-sdk/provider"
6
+
7
+ export function prepareTools({
8
+ tools,
9
+ toolChoice,
10
+ }: {
11
+ tools: LanguageModelV2CallOptions["tools"]
12
+ toolChoice?: LanguageModelV2CallOptions["toolChoice"]
13
+ }): {
14
+ tools:
15
+ | undefined
16
+ | Array<{
17
+ type: "function"
18
+ function: {
19
+ name: string
20
+ description: string | undefined
21
+ parameters: unknown
22
+ }
23
+ }>
24
+ toolChoice: { type: "function"; function: { name: string } } | "auto" | "none" | "required" | undefined
25
+ toolWarnings: LanguageModelV2CallWarning[]
26
+ } {
27
+ // when the tools array is empty, change it to undefined to prevent errors:
28
+ tools = tools?.length ? tools : undefined
29
+
30
+ const toolWarnings: LanguageModelV2CallWarning[] = []
31
+
32
+ if (tools == null) {
33
+ return { tools: undefined, toolChoice: undefined, toolWarnings }
34
+ }
35
+
36
+ const openaiCompatTools: Array<{
37
+ type: "function"
38
+ function: {
39
+ name: string
40
+ description: string | undefined
41
+ parameters: unknown
42
+ }
43
+ }> = []
44
+
45
+ for (const tool of tools) {
46
+ if (tool.type === "provider-defined") {
47
+ toolWarnings.push({ type: "unsupported-tool", tool })
48
+ } else {
49
+ openaiCompatTools.push({
50
+ type: "function",
51
+ function: {
52
+ name: tool.name,
53
+ description: tool.description,
54
+ parameters: tool.inputSchema,
55
+ },
56
+ })
57
+ }
58
+ }
59
+
60
+ if (toolChoice == null) {
61
+ return { tools: openaiCompatTools, toolChoice: undefined, toolWarnings }
62
+ }
63
+
64
+ const type = toolChoice.type
65
+
66
+ switch (type) {
67
+ case "auto":
68
+ case "none":
69
+ case "required":
70
+ return { tools: openaiCompatTools, toolChoice: type, toolWarnings }
71
+ case "tool":
72
+ return {
73
+ tools: openaiCompatTools,
74
+ toolChoice: {
75
+ type: "function",
76
+ function: { name: toolChoice.toolName },
77
+ },
78
+ toolWarnings,
79
+ }
80
+ default: {
81
+ const _exhaustiveCheck: never = type
82
+ throw new UnsupportedFunctionalityError({
83
+ functionality: `tool choice type: ${_exhaustiveCheck}`,
84
+ })
85
+ }
86
+ }
87
+ }
@@ -0,0 +1,100 @@
1
+ import type { LanguageModelV2 } from "@ai-sdk/provider"
2
+ import { type FetchFunction, withoutTrailingSlash, withUserAgentSuffix } from "@ai-sdk/provider-utils"
3
+ import { OpenAICompatibleChatLanguageModel } from "./chat/openai-compatible-chat-language-model"
4
+ import { OpenAIResponsesLanguageModel } from "./responses/openai-responses-language-model"
5
+
6
+ // Import the version or define it
7
+ const VERSION = "0.1.0"
8
+
9
+ export type OpenaiCompatibleModelId = string
10
+
11
+ export interface OpenaiCompatibleProviderSettings {
12
+ /**
13
+ * API key for authenticating requests.
14
+ */
15
+ apiKey?: string
16
+
17
+ /**
18
+ * Base URL for the OpenAI Compatible API calls.
19
+ */
20
+ baseURL?: string
21
+
22
+ /**
23
+ * Name of the provider.
24
+ */
25
+ name?: string
26
+
27
+ /**
28
+ * Custom headers to include in the requests.
29
+ */
30
+ headers?: Record<string, string>
31
+
32
+ /**
33
+ * Custom fetch implementation.
34
+ */
35
+ fetch?: FetchFunction
36
+ }
37
+
38
+ export interface OpenaiCompatibleProvider {
39
+ (modelId: OpenaiCompatibleModelId): LanguageModelV2
40
+ chat(modelId: OpenaiCompatibleModelId): LanguageModelV2
41
+ responses(modelId: OpenaiCompatibleModelId): LanguageModelV2
42
+ languageModel(modelId: OpenaiCompatibleModelId): LanguageModelV2
43
+
44
+ // embeddingModel(modelId: any): EmbeddingModelV2
45
+
46
+ // imageModel(modelId: any): ImageModelV2
47
+ }
48
+
49
+ /**
50
+ * Create an OpenAI Compatible provider instance.
51
+ */
52
+ export function createOpenaiCompatible(options: OpenaiCompatibleProviderSettings = {}): OpenaiCompatibleProvider {
53
+ const baseURL = withoutTrailingSlash(options.baseURL ?? "https://api.openai.com/v1")
54
+
55
+ if (!baseURL) {
56
+ throw new Error("baseURL is required")
57
+ }
58
+
59
+ // Merge headers: defaults first, then user overrides
60
+ const headers = {
61
+ // Default OpenAI Compatible headers (can be overridden by user)
62
+ ...(options.apiKey && { Authorization: `Bearer ${options.apiKey}` }),
63
+ ...options.headers,
64
+ }
65
+
66
+ const getHeaders = () => withUserAgentSuffix(headers, `ai-sdk/openai-compatible/${VERSION}`)
67
+
68
+ const createChatModel = (modelId: OpenaiCompatibleModelId) => {
69
+ return new OpenAICompatibleChatLanguageModel(modelId, {
70
+ provider: `${options.name ?? "openai-compatible"}.chat`,
71
+ headers: getHeaders,
72
+ url: ({ path }) => `${baseURL}${path}`,
73
+ fetch: options.fetch,
74
+ })
75
+ }
76
+
77
+ const createResponsesModel = (modelId: OpenaiCompatibleModelId) => {
78
+ return new OpenAIResponsesLanguageModel(modelId, {
79
+ provider: `${options.name ?? "openai-compatible"}.responses`,
80
+ headers: getHeaders,
81
+ url: ({ path }) => `${baseURL}${path}`,
82
+ fetch: options.fetch,
83
+ })
84
+ }
85
+
86
+ const createLanguageModel = (modelId: OpenaiCompatibleModelId) => createChatModel(modelId)
87
+
88
+ const provider = function (modelId: OpenaiCompatibleModelId) {
89
+ return createChatModel(modelId)
90
+ }
91
+
92
+ provider.languageModel = createLanguageModel
93
+ provider.chat = createChatModel
94
+ provider.responses = createResponsesModel
95
+
96
+ return provider as OpenaiCompatibleProvider
97
+ }
98
+
99
+ // Default OpenAI Compatible provider instance
100
+ export const openaiCompatible = createOpenaiCompatible()
@@ -0,0 +1,2 @@
1
+ export { createOpenaiCompatible, openaiCompatible } from "./copilot-provider"
2
+ export type { OpenaiCompatibleProvider, OpenaiCompatibleProviderSettings } from "./copilot-provider"
@@ -0,0 +1,27 @@
1
+ import { z, type ZodType } from "zod/v4"
2
+
3
+ export const openaiCompatibleErrorDataSchema = z.object({
4
+ error: z.object({
5
+ message: z.string(),
6
+
7
+ // The additional information below is handled loosely to support
8
+ // OpenAI-compatible providers that have slightly different error
9
+ // responses:
10
+ type: z.string().nullish(),
11
+ param: z.any().nullish(),
12
+ code: z.union([z.string(), z.number()]).nullish(),
13
+ }),
14
+ })
15
+
16
+ export type OpenAICompatibleErrorData = z.infer<typeof openaiCompatibleErrorDataSchema>
17
+
18
+ export type ProviderErrorStructure<T> = {
19
+ errorSchema: ZodType<T>
20
+ errorToMessage: (error: T) => string
21
+ isRetryable?: (response: Response, error?: T) => boolean
22
+ }
23
+
24
+ export const defaultOpenAICompatibleErrorStructure: ProviderErrorStructure<OpenAICompatibleErrorData> = {
25
+ errorSchema: openaiCompatibleErrorDataSchema,
26
+ errorToMessage: (data) => data.error.message,
27
+ }
@@ -0,0 +1,303 @@
1
+ import {
2
+ type LanguageModelV2CallWarning,
3
+ type LanguageModelV2Prompt,
4
+ type LanguageModelV2ToolCallPart,
5
+ UnsupportedFunctionalityError,
6
+ } from "@ai-sdk/provider"
7
+ import { convertToBase64, parseProviderOptions } from "@ai-sdk/provider-utils"
8
+ import { z } from "zod/v4"
9
+ import type { OpenAIResponsesInput, OpenAIResponsesReasoning } from "./openai-responses-api-types"
10
+ import { localShellInputSchema, localShellOutputSchema } from "./tool/local-shell"
11
+
12
+ /**
13
+ * Check if a string is a file ID based on the given prefixes
14
+ * Returns false if prefixes is undefined (disables file ID detection)
15
+ */
16
+ function isFileId(data: string, prefixes?: readonly string[]): boolean {
17
+ if (!prefixes) return false
18
+ return prefixes.some((prefix) => data.startsWith(prefix))
19
+ }
20
+
21
+ export async function convertToOpenAIResponsesInput({
22
+ prompt,
23
+ systemMessageMode,
24
+ fileIdPrefixes,
25
+ store,
26
+ hasLocalShellTool = false,
27
+ }: {
28
+ prompt: LanguageModelV2Prompt
29
+ systemMessageMode: "system" | "developer" | "remove"
30
+ fileIdPrefixes?: readonly string[]
31
+ store: boolean
32
+ hasLocalShellTool?: boolean
33
+ }): Promise<{
34
+ input: OpenAIResponsesInput
35
+ warnings: Array<LanguageModelV2CallWarning>
36
+ }> {
37
+ const input: OpenAIResponsesInput = []
38
+ const warnings: Array<LanguageModelV2CallWarning> = []
39
+
40
+ for (const { role, content } of prompt) {
41
+ switch (role) {
42
+ case "system": {
43
+ switch (systemMessageMode) {
44
+ case "system": {
45
+ input.push({ role: "system", content })
46
+ break
47
+ }
48
+ case "developer": {
49
+ input.push({ role: "developer", content })
50
+ break
51
+ }
52
+ case "remove": {
53
+ warnings.push({
54
+ type: "other",
55
+ message: "system messages are removed for this model",
56
+ })
57
+ break
58
+ }
59
+ default: {
60
+ const _exhaustiveCheck: never = systemMessageMode
61
+ throw new Error(`Unsupported system message mode: ${_exhaustiveCheck}`)
62
+ }
63
+ }
64
+ break
65
+ }
66
+
67
+ case "user": {
68
+ input.push({
69
+ role: "user",
70
+ content: content.map((part, index) => {
71
+ switch (part.type) {
72
+ case "text": {
73
+ return { type: "input_text", text: part.text }
74
+ }
75
+ case "file": {
76
+ if (part.mediaType.startsWith("image/")) {
77
+ const mediaType = part.mediaType === "image/*" ? "image/jpeg" : part.mediaType
78
+
79
+ return {
80
+ type: "input_image",
81
+ ...(part.data instanceof URL
82
+ ? { image_url: part.data.toString() }
83
+ : typeof part.data === "string" && isFileId(part.data, fileIdPrefixes)
84
+ ? { file_id: part.data }
85
+ : {
86
+ image_url: `data:${mediaType};base64,${convertToBase64(part.data)}`,
87
+ }),
88
+ detail: part.providerOptions?.openai?.imageDetail,
89
+ }
90
+ } else if (part.mediaType === "application/pdf") {
91
+ if (part.data instanceof URL) {
92
+ return {
93
+ type: "input_file",
94
+ file_url: part.data.toString(),
95
+ }
96
+ }
97
+ return {
98
+ type: "input_file",
99
+ ...(typeof part.data === "string" && isFileId(part.data, fileIdPrefixes)
100
+ ? { file_id: part.data }
101
+ : {
102
+ filename: part.filename ?? `part-${index}.pdf`,
103
+ file_data: `data:application/pdf;base64,${convertToBase64(part.data)}`,
104
+ }),
105
+ }
106
+ } else {
107
+ throw new UnsupportedFunctionalityError({
108
+ functionality: `file part media type ${part.mediaType}`,
109
+ })
110
+ }
111
+ }
112
+ }
113
+ }),
114
+ })
115
+
116
+ break
117
+ }
118
+
119
+ case "assistant": {
120
+ const reasoningMessages: Record<string, OpenAIResponsesReasoning> = {}
121
+ const toolCallParts: Record<string, LanguageModelV2ToolCallPart> = {}
122
+
123
+ for (const part of content) {
124
+ switch (part.type) {
125
+ case "text": {
126
+ input.push({
127
+ role: "assistant",
128
+ content: [{ type: "output_text", text: part.text }],
129
+ id: (part.providerOptions?.openai?.itemId as string) ?? undefined,
130
+ })
131
+ break
132
+ }
133
+ case "tool-call": {
134
+ toolCallParts[part.toolCallId] = part
135
+
136
+ if (part.providerExecuted) {
137
+ break
138
+ }
139
+
140
+ if (hasLocalShellTool && part.toolName === "local_shell") {
141
+ const parsedInput = localShellInputSchema.parse(part.input)
142
+ input.push({
143
+ type: "local_shell_call",
144
+ call_id: part.toolCallId,
145
+ id: (part.providerOptions?.openai?.itemId as string) ?? undefined,
146
+ action: {
147
+ type: "exec",
148
+ command: parsedInput.action.command,
149
+ timeout_ms: parsedInput.action.timeoutMs,
150
+ user: parsedInput.action.user,
151
+ working_directory: parsedInput.action.workingDirectory,
152
+ env: parsedInput.action.env,
153
+ },
154
+ })
155
+
156
+ break
157
+ }
158
+
159
+ input.push({
160
+ type: "function_call",
161
+ call_id: part.toolCallId,
162
+ name: part.toolName,
163
+ arguments: JSON.stringify(part.input),
164
+ id: (part.providerOptions?.openai?.itemId as string) ?? undefined,
165
+ })
166
+ break
167
+ }
168
+
169
+ // assistant tool result parts are from provider-executed tools:
170
+ case "tool-result": {
171
+ if (store) {
172
+ // use item references to refer to tool results from built-in tools
173
+ input.push({ type: "item_reference", id: part.toolCallId })
174
+ } else {
175
+ warnings.push({
176
+ type: "other",
177
+ message: `Results for OpenAI tool ${part.toolName} are not sent to the API when store is false`,
178
+ })
179
+ }
180
+
181
+ break
182
+ }
183
+
184
+ case "reasoning": {
185
+ const providerOptions = await parseProviderOptions({
186
+ provider: "copilot",
187
+ providerOptions: part.providerOptions,
188
+ schema: openaiResponsesReasoningProviderOptionsSchema,
189
+ })
190
+
191
+ const reasoningId = providerOptions?.itemId
192
+
193
+ if (reasoningId != null) {
194
+ const reasoningMessage = reasoningMessages[reasoningId]
195
+
196
+ if (store) {
197
+ if (reasoningMessage === undefined) {
198
+ // use item references to refer to reasoning (single reference)
199
+ input.push({ type: "item_reference", id: reasoningId })
200
+
201
+ // store unused reasoning message to mark id as used
202
+ reasoningMessages[reasoningId] = {
203
+ type: "reasoning",
204
+ id: reasoningId,
205
+ summary: [],
206
+ }
207
+ }
208
+ } else {
209
+ const summaryParts: Array<{
210
+ type: "summary_text"
211
+ text: string
212
+ }> = []
213
+
214
+ if (part.text.length > 0) {
215
+ summaryParts.push({
216
+ type: "summary_text",
217
+ text: part.text,
218
+ })
219
+ } else if (reasoningMessage !== undefined) {
220
+ warnings.push({
221
+ type: "other",
222
+ message: `Cannot append empty reasoning part to existing reasoning sequence. Skipping reasoning part: ${JSON.stringify(part)}.`,
223
+ })
224
+ }
225
+
226
+ if (reasoningMessage === undefined) {
227
+ reasoningMessages[reasoningId] = {
228
+ type: "reasoning",
229
+ id: reasoningId,
230
+ encrypted_content: providerOptions?.reasoningEncryptedContent,
231
+ summary: summaryParts,
232
+ }
233
+ input.push(reasoningMessages[reasoningId])
234
+ } else {
235
+ reasoningMessage.summary.push(...summaryParts)
236
+ }
237
+ }
238
+ } else {
239
+ warnings.push({
240
+ type: "other",
241
+ message: `Non-OpenAI reasoning parts are not supported. Skipping reasoning part: ${JSON.stringify(part)}.`,
242
+ })
243
+ }
244
+ break
245
+ }
246
+ }
247
+ }
248
+
249
+ break
250
+ }
251
+
252
+ case "tool": {
253
+ for (const part of content) {
254
+ const output = part.output
255
+
256
+ if (hasLocalShellTool && part.toolName === "local_shell" && output.type === "json") {
257
+ input.push({
258
+ type: "local_shell_call_output",
259
+ call_id: part.toolCallId,
260
+ output: localShellOutputSchema.parse(output.value).output,
261
+ })
262
+ break
263
+ }
264
+
265
+ let contentValue: string
266
+ switch (output.type) {
267
+ case "text":
268
+ case "error-text":
269
+ contentValue = output.value
270
+ break
271
+ case "content":
272
+ case "json":
273
+ case "error-json":
274
+ contentValue = JSON.stringify(output.value)
275
+ break
276
+ }
277
+
278
+ input.push({
279
+ type: "function_call_output",
280
+ call_id: part.toolCallId,
281
+ output: contentValue,
282
+ })
283
+ }
284
+
285
+ break
286
+ }
287
+
288
+ default: {
289
+ const _exhaustiveCheck: never = role
290
+ throw new Error(`Unsupported role: ${_exhaustiveCheck}`)
291
+ }
292
+ }
293
+ }
294
+
295
+ return { input, warnings }
296
+ }
297
+
298
+ const openaiResponsesReasoningProviderOptionsSchema = z.object({
299
+ itemId: z.string().nullish(),
300
+ reasoningEncryptedContent: z.string().nullish(),
301
+ })
302
+
303
+ export type OpenAIResponsesReasoningProviderOptions = z.infer<typeof openaiResponsesReasoningProviderOptionsSchema>
@@ -0,0 +1,22 @@
1
+ import type { LanguageModelV2FinishReason } from "@ai-sdk/provider"
2
+
3
+ export function mapOpenAIResponseFinishReason({
4
+ finishReason,
5
+ hasFunctionCall,
6
+ }: {
7
+ finishReason: string | null | undefined
8
+ // flag that checks if there have been client-side tool calls (not executed by openai)
9
+ hasFunctionCall: boolean
10
+ }): LanguageModelV2FinishReason {
11
+ switch (finishReason) {
12
+ case undefined:
13
+ case null:
14
+ return hasFunctionCall ? "tool-calls" : "stop"
15
+ case "max_output_tokens":
16
+ return "length"
17
+ case "content_filter":
18
+ return "content-filter"
19
+ default:
20
+ return hasFunctionCall ? "tool-calls" : "unknown"
21
+ }
22
+ }
@@ -0,0 +1,18 @@
1
+ import type { FetchFunction } from "@ai-sdk/provider-utils"
2
+
3
+ export type OpenAIConfig = {
4
+ provider: string
5
+ url: (options: { modelId: string; path: string }) => string
6
+ headers: () => Record<string, string | undefined>
7
+ fetch?: FetchFunction
8
+ generateId?: () => string
9
+ /**
10
+ * File ID prefixes used to identify file IDs in Responses API.
11
+ * When undefined, all file data is treated as base64 content.
12
+ *
13
+ * Examples:
14
+ * - OpenAI: ['file-'] for IDs like 'file-abc123'
15
+ * - Azure OpenAI: ['assistant-'] for IDs like 'assistant-abc123'
16
+ */
17
+ fileIdPrefixes?: readonly string[]
18
+ }
@@ -0,0 +1,22 @@
1
+ import { z } from "zod/v4"
2
+ import { createJsonErrorResponseHandler } from "@ai-sdk/provider-utils"
3
+
4
+ export const openaiErrorDataSchema = z.object({
5
+ error: z.object({
6
+ message: z.string(),
7
+
8
+ // The additional information below is handled loosely to support
9
+ // OpenAI-compatible providers that have slightly different error
10
+ // responses:
11
+ type: z.string().nullish(),
12
+ param: z.any().nullish(),
13
+ code: z.union([z.string(), z.number()]).nullish(),
14
+ }),
15
+ })
16
+
17
+ export type OpenAIErrorData = z.infer<typeof openaiErrorDataSchema>
18
+
19
+ export const openaiFailedResponseHandler: any = createJsonErrorResponseHandler({
20
+ errorSchema: openaiErrorDataSchema,
21
+ errorToMessage: (data) => data.error.message,
22
+ })