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,828 @@
1
+ import type { APICallError, ModelMessage } from "ai"
2
+ import { mergeDeep, unique } from "remeda"
3
+ import type { JSONSchema7 } from "@ai-sdk/provider"
4
+ import type { JSONSchema } from "zod/v4/core"
5
+ import type { Provider } from "./provider"
6
+ import type { ModelsDev } from "./models"
7
+ import { iife } from "@/util/iife"
8
+
9
+ type Modality = NonNullable<ModelsDev.Model["modalities"]>["input"][number]
10
+
11
+ function mimeToModality(mime: string): Modality | undefined {
12
+ if (mime.startsWith("image/")) return "image"
13
+ if (mime.startsWith("audio/")) return "audio"
14
+ if (mime.startsWith("video/")) return "video"
15
+ if (mime === "application/pdf") return "pdf"
16
+ return undefined
17
+ }
18
+
19
+ export namespace ProviderTransform {
20
+ // Maps npm package to the key the AI SDK expects for providerOptions
21
+ function sdkKey(npm: string): string | undefined {
22
+ switch (npm) {
23
+ case "@ai-sdk/github-copilot":
24
+ return "copilot"
25
+ case "@ai-sdk/openai":
26
+ case "@ai-sdk/azure":
27
+ return "openai"
28
+ case "@ai-sdk/amazon-bedrock":
29
+ return "bedrock"
30
+ case "@ai-sdk/anthropic":
31
+ case "@ai-sdk/google-vertex/anthropic":
32
+ return "anthropic"
33
+ case "@ai-sdk/google-vertex":
34
+ case "@ai-sdk/google":
35
+ return "google"
36
+ case "@ai-sdk/gateway":
37
+ return "gateway"
38
+ case "@openrouter/ai-sdk-provider":
39
+ return "openrouter"
40
+ }
41
+ return undefined
42
+ }
43
+
44
+ function normalizeMessages(
45
+ msgs: ModelMessage[],
46
+ model: Provider.Model,
47
+ options: Record<string, unknown>,
48
+ ): ModelMessage[] {
49
+ // Anthropic rejects messages with empty content - filter out empty string messages
50
+ // and remove empty text/reasoning parts from array content
51
+ if (model.api.npm === "@ai-sdk/anthropic") {
52
+ msgs = msgs
53
+ .map((msg) => {
54
+ if (typeof msg.content === "string") {
55
+ if (msg.content === "") return undefined
56
+ return msg
57
+ }
58
+ if (!Array.isArray(msg.content)) return msg
59
+ const filtered = msg.content.filter((part) => {
60
+ if (part.type === "text" || part.type === "reasoning") {
61
+ return part.text !== ""
62
+ }
63
+ return true
64
+ })
65
+ if (filtered.length === 0) return undefined
66
+ return { ...msg, content: filtered }
67
+ })
68
+ .filter((msg): msg is ModelMessage => msg !== undefined && msg.content !== "")
69
+ }
70
+
71
+ if (model.api.id.includes("claude")) {
72
+ return msgs.map((msg) => {
73
+ if ((msg.role === "assistant" || msg.role === "tool") && Array.isArray(msg.content)) {
74
+ msg.content = msg.content.map((part) => {
75
+ if ((part.type === "tool-call" || part.type === "tool-result") && "toolCallId" in part) {
76
+ return {
77
+ ...part,
78
+ toolCallId: part.toolCallId.replace(/[^a-zA-Z0-9_-]/g, "_"),
79
+ }
80
+ }
81
+ return part
82
+ })
83
+ }
84
+ return msg
85
+ })
86
+ }
87
+ if (
88
+ model.providerID === "mistral" ||
89
+ model.api.id.toLowerCase().includes("mistral") ||
90
+ model.api.id.toLocaleLowerCase().includes("devstral")
91
+ ) {
92
+ const result: ModelMessage[] = []
93
+ for (let i = 0; i < msgs.length; i++) {
94
+ const msg = msgs[i]
95
+ const nextMsg = msgs[i + 1]
96
+
97
+ if ((msg.role === "assistant" || msg.role === "tool") && Array.isArray(msg.content)) {
98
+ msg.content = msg.content.map((part) => {
99
+ if ((part.type === "tool-call" || part.type === "tool-result") && "toolCallId" in part) {
100
+ // Mistral requires alphanumeric tool call IDs with exactly 9 characters
101
+ const normalizedId = part.toolCallId
102
+ .replace(/[^a-zA-Z0-9]/g, "") // Remove non-alphanumeric characters
103
+ .substring(0, 9) // Take first 9 characters
104
+ .padEnd(9, "0") // Pad with zeros if less than 9 characters
105
+
106
+ return {
107
+ ...part,
108
+ toolCallId: normalizedId,
109
+ }
110
+ }
111
+ return part
112
+ })
113
+ }
114
+
115
+ result.push(msg)
116
+
117
+ // Fix message sequence: tool messages cannot be followed by user messages
118
+ if (msg.role === "tool" && nextMsg?.role === "user") {
119
+ result.push({
120
+ role: "assistant",
121
+ content: [
122
+ {
123
+ type: "text",
124
+ text: "Done.",
125
+ },
126
+ ],
127
+ })
128
+ }
129
+ }
130
+ return result
131
+ }
132
+
133
+ if (typeof model.capabilities.interleaved === "object" && model.capabilities.interleaved.field) {
134
+ const field = model.capabilities.interleaved.field
135
+ return msgs.map((msg) => {
136
+ if (msg.role === "assistant" && Array.isArray(msg.content)) {
137
+ const reasoningParts = msg.content.filter((part: any) => part.type === "reasoning")
138
+ const reasoningText = reasoningParts.map((part: any) => part.text).join("")
139
+
140
+ // Filter out reasoning parts from content
141
+ const filteredContent = msg.content.filter((part: any) => part.type !== "reasoning")
142
+
143
+ // Include reasoning_content | reasoning_details directly on the message for all assistant messages
144
+ if (reasoningText) {
145
+ return {
146
+ ...msg,
147
+ content: filteredContent,
148
+ providerOptions: {
149
+ ...msg.providerOptions,
150
+ openaiCompatible: {
151
+ ...(msg.providerOptions as any)?.openaiCompatible,
152
+ [field]: reasoningText,
153
+ },
154
+ },
155
+ }
156
+ }
157
+
158
+ return {
159
+ ...msg,
160
+ content: filteredContent,
161
+ }
162
+ }
163
+
164
+ return msg
165
+ })
166
+ }
167
+
168
+ return msgs
169
+ }
170
+
171
+ function applyCaching(msgs: ModelMessage[], providerID: string): ModelMessage[] {
172
+ const system = msgs.filter((msg) => msg.role === "system").slice(0, 2)
173
+ const final = msgs.filter((msg) => msg.role !== "system").slice(-2)
174
+
175
+ const providerOptions = {
176
+ anthropic: {
177
+ cacheControl: { type: "ephemeral" },
178
+ },
179
+ openrouter: {
180
+ cacheControl: { type: "ephemeral" },
181
+ },
182
+ bedrock: {
183
+ cachePoint: { type: "default" },
184
+ },
185
+ openaiCompatible: {
186
+ cache_control: { type: "ephemeral" },
187
+ },
188
+ copilot: {
189
+ copilot_cache_control: { type: "ephemeral" },
190
+ },
191
+ }
192
+
193
+ for (const msg of unique([...system, ...final])) {
194
+ const useMessageLevelOptions = providerID === "anthropic" || providerID.includes("bedrock")
195
+ const shouldUseContentOptions = !useMessageLevelOptions && Array.isArray(msg.content) && msg.content.length > 0
196
+
197
+ if (shouldUseContentOptions) {
198
+ const lastContent = msg.content[msg.content.length - 1]
199
+ if (lastContent && typeof lastContent === "object") {
200
+ lastContent.providerOptions = mergeDeep(lastContent.providerOptions ?? {}, providerOptions)
201
+ continue
202
+ }
203
+ }
204
+
205
+ msg.providerOptions = mergeDeep(msg.providerOptions ?? {}, providerOptions)
206
+ }
207
+
208
+ return msgs
209
+ }
210
+
211
+ function unsupportedParts(msgs: ModelMessage[], model: Provider.Model): ModelMessage[] {
212
+ return msgs.map((msg) => {
213
+ if (msg.role !== "user" || !Array.isArray(msg.content)) return msg
214
+
215
+ const filtered = msg.content.map((part) => {
216
+ if (part.type !== "file" && part.type !== "image") return part
217
+
218
+ // Check for empty base64 image data
219
+ if (part.type === "image") {
220
+ const imageStr = part.image.toString()
221
+ if (imageStr.startsWith("data:")) {
222
+ const match = imageStr.match(/^data:([^;]+);base64,(.*)$/)
223
+ if (match && (!match[2] || match[2].length === 0)) {
224
+ return {
225
+ type: "text" as const,
226
+ text: "ERROR: Image file is empty or corrupted. Please provide a valid image.",
227
+ }
228
+ }
229
+ }
230
+ }
231
+
232
+ const mime = part.type === "image" ? part.image.toString().split(";")[0].replace("data:", "") : part.mediaType
233
+ const filename = part.type === "file" ? part.filename : undefined
234
+ const modality = mimeToModality(mime)
235
+ if (!modality) return part
236
+ if (model.capabilities.input[modality]) return part
237
+
238
+ const name = filename ? `"${filename}"` : modality
239
+ return {
240
+ type: "text" as const,
241
+ text: `ERROR: Cannot read ${name} (this model does not support ${modality} input). Inform the user.`,
242
+ }
243
+ })
244
+
245
+ return { ...msg, content: filtered }
246
+ })
247
+ }
248
+
249
+ export function message(msgs: ModelMessage[], model: Provider.Model, options: Record<string, unknown>) {
250
+ msgs = unsupportedParts(msgs, model)
251
+ msgs = normalizeMessages(msgs, model, options)
252
+ if (
253
+ model.providerID === "anthropic" ||
254
+ model.api.id.includes("anthropic") ||
255
+ model.api.id.includes("claude") ||
256
+ model.id.includes("anthropic") ||
257
+ model.id.includes("claude") ||
258
+ model.api.npm === "@ai-sdk/anthropic"
259
+ ) {
260
+ msgs = applyCaching(msgs, model.providerID)
261
+ }
262
+
263
+ // Remap providerOptions keys from stored providerID to expected SDK key
264
+ const key = sdkKey(model.api.npm)
265
+ if (key && key !== model.providerID && model.api.npm !== "@ai-sdk/azure") {
266
+ const remap = (opts: Record<string, any> | undefined) => {
267
+ if (!opts) return opts
268
+ if (!(model.providerID in opts)) return opts
269
+ const result = { ...opts }
270
+ result[key] = result[model.providerID]
271
+ delete result[model.providerID]
272
+ return result
273
+ }
274
+
275
+ msgs = msgs.map((msg) => {
276
+ if (!Array.isArray(msg.content)) return { ...msg, providerOptions: remap(msg.providerOptions) }
277
+ return {
278
+ ...msg,
279
+ providerOptions: remap(msg.providerOptions),
280
+ content: msg.content.map((part) => ({ ...part, providerOptions: remap(part.providerOptions) })),
281
+ } as typeof msg
282
+ })
283
+ }
284
+
285
+ return msgs
286
+ }
287
+
288
+ export function temperature(model: Provider.Model) {
289
+ const id = model.id.toLowerCase()
290
+ if (id.includes("qwen")) return 0.55
291
+ if (id.includes("claude")) return undefined
292
+ if (id.includes("gemini")) return 1.0
293
+ if (id.includes("glm-4.6")) return 1.0
294
+ if (id.includes("glm-4.7")) return 1.0
295
+ if (id.includes("minimax-m2")) return 1.0
296
+ if (id.includes("kimi-k2")) {
297
+ // kimi-k2-thinking & kimi-k2.5 && kimi-k2p5
298
+ if (id.includes("thinking") || id.includes("k2.") || id.includes("k2p")) {
299
+ return 1.0
300
+ }
301
+ return 0.6
302
+ }
303
+ return undefined
304
+ }
305
+
306
+ export function topP(model: Provider.Model) {
307
+ const id = model.id.toLowerCase()
308
+ if (id.includes("qwen")) return 1
309
+ if (id.includes("minimax-m2") || id.includes("kimi-k2.5") || id.includes("kimi-k2p5") || id.includes("gemini")) {
310
+ return 0.95
311
+ }
312
+ return undefined
313
+ }
314
+
315
+ export function topK(model: Provider.Model) {
316
+ const id = model.id.toLowerCase()
317
+ if (id.includes("minimax-m2")) {
318
+ if (id.includes("m2.1")) return 40
319
+ return 20
320
+ }
321
+ if (id.includes("gemini")) return 64
322
+ return undefined
323
+ }
324
+
325
+ const WIDELY_SUPPORTED_EFFORTS = ["low", "medium", "high"]
326
+ const OPENAI_EFFORTS = ["none", "minimal", ...WIDELY_SUPPORTED_EFFORTS, "xhigh"]
327
+
328
+ export function variants(model: Provider.Model): Record<string, Record<string, any>> {
329
+ if (!model.capabilities.reasoning) return {}
330
+
331
+ const id = model.id.toLowerCase()
332
+ if (
333
+ id.includes("deepseek") ||
334
+ id.includes("minimax") ||
335
+ id.includes("glm") ||
336
+ id.includes("mistral") ||
337
+ id.includes("kimi") ||
338
+ // TODO: Remove this after models.dev data is fixed to use "kimi-k2.5" instead of "k2p5"
339
+ id.includes("k2p5")
340
+ )
341
+ return {}
342
+
343
+ // see: https://docs.x.ai/docs/guides/reasoning#control-how-hard-the-model-thinks
344
+ if (id.includes("grok") && id.includes("grok-3-mini")) {
345
+ if (model.api.npm === "@openrouter/ai-sdk-provider") {
346
+ return {
347
+ low: { reasoning: { effort: "low" } },
348
+ high: { reasoning: { effort: "high" } },
349
+ }
350
+ }
351
+ return {
352
+ low: { reasoningEffort: "low" },
353
+ high: { reasoningEffort: "high" },
354
+ }
355
+ }
356
+ if (id.includes("grok")) return {}
357
+
358
+ switch (model.api.npm) {
359
+ case "@openrouter/ai-sdk-provider":
360
+ if (!model.id.includes("gpt") && !model.id.includes("gemini-3")) return {}
361
+ return Object.fromEntries(OPENAI_EFFORTS.map((effort) => [effort, { reasoning: { effort } }]))
362
+
363
+ // TODO: YOU CANNOT SET max_tokens if this is set!!!
364
+ case "@ai-sdk/gateway":
365
+ return Object.fromEntries(OPENAI_EFFORTS.map((effort) => [effort, { reasoningEffort: effort }]))
366
+
367
+ case "@ai-sdk/github-copilot":
368
+ if (model.id.includes("gemini")) {
369
+ // currently github copilot only returns thinking
370
+ return {}
371
+ }
372
+ if (model.id.includes("claude")) {
373
+ return {
374
+ thinking: { thinking_budget: 4000 },
375
+ }
376
+ }
377
+ const copilotEfforts = iife(() => {
378
+ if (id.includes("5.1-codex-max") || id.includes("5.2") || id.includes("5.3"))
379
+ return [...WIDELY_SUPPORTED_EFFORTS, "xhigh"]
380
+ return WIDELY_SUPPORTED_EFFORTS
381
+ })
382
+ return Object.fromEntries(
383
+ copilotEfforts.map((effort) => [
384
+ effort,
385
+ {
386
+ reasoningEffort: effort,
387
+ reasoningSummary: "auto",
388
+ include: ["reasoning.encrypted_content"],
389
+ },
390
+ ]),
391
+ )
392
+
393
+ case "@ai-sdk/cerebras":
394
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/cerebras
395
+ case "@ai-sdk/togetherai":
396
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/togetherai
397
+ case "@ai-sdk/xai":
398
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/xai
399
+ case "@ai-sdk/deepinfra":
400
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/deepinfra
401
+ case "@ai-sdk/openai-compatible":
402
+ return Object.fromEntries(WIDELY_SUPPORTED_EFFORTS.map((effort) => [effort, { reasoningEffort: effort }]))
403
+
404
+ case "@ai-sdk/azure":
405
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/azure
406
+ if (id === "o1-mini") return {}
407
+ const azureEfforts = ["low", "medium", "high"]
408
+ if (id.includes("gpt-5-") || id === "gpt-5") {
409
+ azureEfforts.unshift("minimal")
410
+ }
411
+ return Object.fromEntries(
412
+ azureEfforts.map((effort) => [
413
+ effort,
414
+ {
415
+ reasoningEffort: effort,
416
+ reasoningSummary: "auto",
417
+ include: ["reasoning.encrypted_content"],
418
+ },
419
+ ]),
420
+ )
421
+ case "@ai-sdk/openai":
422
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/openai
423
+ if (id === "gpt-5-pro") return {}
424
+ const openaiEfforts = iife(() => {
425
+ if (id.includes("codex")) {
426
+ if (id.includes("5.2") || id.includes("5.3")) return [...WIDELY_SUPPORTED_EFFORTS, "xhigh"]
427
+ return WIDELY_SUPPORTED_EFFORTS
428
+ }
429
+ const arr = [...WIDELY_SUPPORTED_EFFORTS]
430
+ if (id.includes("gpt-5-") || id === "gpt-5") {
431
+ arr.unshift("minimal")
432
+ }
433
+ if (model.release_date >= "2025-11-13") {
434
+ arr.unshift("none")
435
+ }
436
+ if (model.release_date >= "2025-12-04") {
437
+ arr.push("xhigh")
438
+ }
439
+ return arr
440
+ })
441
+ return Object.fromEntries(
442
+ openaiEfforts.map((effort) => [
443
+ effort,
444
+ {
445
+ reasoningEffort: effort,
446
+ reasoningSummary: "auto",
447
+ include: ["reasoning.encrypted_content"],
448
+ },
449
+ ]),
450
+ )
451
+
452
+ case "@ai-sdk/anthropic":
453
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/anthropic
454
+ case "@ai-sdk/google-vertex/anthropic":
455
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/google-vertex#anthropic-provider
456
+ return {
457
+ high: {
458
+ thinking: {
459
+ type: "enabled",
460
+ budgetTokens: Math.min(16_000, Math.floor(model.limit.output / 2 - 1)),
461
+ },
462
+ },
463
+ max: {
464
+ thinking: {
465
+ type: "enabled",
466
+ budgetTokens: Math.min(31_999, model.limit.output - 1),
467
+ },
468
+ },
469
+ }
470
+
471
+ case "@ai-sdk/amazon-bedrock":
472
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/amazon-bedrock
473
+ // For Anthropic models on Bedrock, use reasoningConfig with budgetTokens
474
+ if (model.api.id.includes("anthropic")) {
475
+ return {
476
+ high: {
477
+ reasoningConfig: {
478
+ type: "enabled",
479
+ budgetTokens: 16000,
480
+ },
481
+ },
482
+ max: {
483
+ reasoningConfig: {
484
+ type: "enabled",
485
+ budgetTokens: 31999,
486
+ },
487
+ },
488
+ }
489
+ }
490
+
491
+ // For Amazon Nova models, use reasoningConfig with maxReasoningEffort
492
+ return Object.fromEntries(
493
+ WIDELY_SUPPORTED_EFFORTS.map((effort) => [
494
+ effort,
495
+ {
496
+ reasoningConfig: {
497
+ type: "enabled",
498
+ maxReasoningEffort: effort,
499
+ },
500
+ },
501
+ ]),
502
+ )
503
+
504
+ case "@ai-sdk/google-vertex":
505
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/google-vertex
506
+ case "@ai-sdk/google":
507
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/google-generative-ai
508
+ if (id.includes("2.5")) {
509
+ return {
510
+ high: {
511
+ thinkingConfig: {
512
+ includeThoughts: true,
513
+ thinkingBudget: 16000,
514
+ },
515
+ },
516
+ max: {
517
+ thinkingConfig: {
518
+ includeThoughts: true,
519
+ thinkingBudget: 24576,
520
+ },
521
+ },
522
+ }
523
+ }
524
+ return Object.fromEntries(
525
+ ["low", "high"].map((effort) => [
526
+ effort,
527
+ {
528
+ includeThoughts: true,
529
+ thinkingLevel: effort,
530
+ },
531
+ ]),
532
+ )
533
+
534
+ case "@ai-sdk/mistral":
535
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/mistral
536
+ return {}
537
+
538
+ case "@ai-sdk/cohere":
539
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/cohere
540
+ return {}
541
+
542
+ case "@ai-sdk/groq":
543
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/groq
544
+ const groqEffort = ["none", ...WIDELY_SUPPORTED_EFFORTS]
545
+ return Object.fromEntries(
546
+ groqEffort.map((effort) => [
547
+ effort,
548
+ {
549
+ includeThoughts: true,
550
+ thinkingLevel: effort,
551
+ },
552
+ ]),
553
+ )
554
+
555
+ case "@ai-sdk/perplexity":
556
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/perplexity
557
+ return {}
558
+
559
+ case "@mymediset/sap-ai-provider":
560
+ case "@jerome-benoit/sap-ai-provider-v2":
561
+ if (model.api.id.includes("anthropic")) {
562
+ return {
563
+ high: {
564
+ thinking: {
565
+ type: "enabled",
566
+ budgetTokens: 16000,
567
+ },
568
+ },
569
+ max: {
570
+ thinking: {
571
+ type: "enabled",
572
+ budgetTokens: 31999,
573
+ },
574
+ },
575
+ }
576
+ }
577
+ return Object.fromEntries(WIDELY_SUPPORTED_EFFORTS.map((effort) => [effort, { reasoningEffort: effort }]))
578
+ }
579
+ return {}
580
+ }
581
+
582
+ export function options(input: {
583
+ model: Provider.Model
584
+ sessionID: string
585
+ providerOptions?: Record<string, any>
586
+ }): Record<string, any> {
587
+ const result: Record<string, any> = {}
588
+
589
+ // openai and providers using openai package should set store to false by default.
590
+ if (
591
+ input.model.providerID === "openai" ||
592
+ input.model.api.npm === "@ai-sdk/openai" ||
593
+ input.model.api.npm === "@ai-sdk/github-copilot"
594
+ ) {
595
+ result["store"] = false
596
+ }
597
+
598
+ if (input.model.api.npm === "@openrouter/ai-sdk-provider") {
599
+ result["usage"] = {
600
+ include: true,
601
+ }
602
+ if (input.model.api.id.includes("gemini-3")) {
603
+ result["reasoning"] = { effort: "high" }
604
+ }
605
+ }
606
+
607
+ if (
608
+ input.model.providerID === "baseten" ||
609
+ (input.model.providerID === "opencode" && ["kimi-k2-thinking", "glm-4.6"].includes(input.model.api.id))
610
+ ) {
611
+ result["chat_template_args"] = { enable_thinking: true }
612
+ }
613
+
614
+ if (["zai", "zhipuai"].includes(input.model.providerID) && input.model.api.npm === "@ai-sdk/openai-compatible") {
615
+ result["thinking"] = {
616
+ type: "enabled",
617
+ clear_thinking: false,
618
+ }
619
+ }
620
+
621
+ if (input.model.providerID === "openai" || input.providerOptions?.setCacheKey) {
622
+ result["promptCacheKey"] = input.sessionID
623
+ }
624
+
625
+ if (input.model.api.npm === "@ai-sdk/google" || input.model.api.npm === "@ai-sdk/google-vertex") {
626
+ result["thinkingConfig"] = {
627
+ includeThoughts: true,
628
+ }
629
+ if (input.model.api.id.includes("gemini-3")) {
630
+ result["thinkingConfig"]["thinkingLevel"] = "high"
631
+ }
632
+ }
633
+
634
+ // Enable thinking by default for kimi-k2.5/k2p5 models using anthropic SDK
635
+ const modelId = input.model.api.id.toLowerCase()
636
+ if (
637
+ (input.model.api.npm === "@ai-sdk/anthropic" || input.model.api.npm === "@ai-sdk/google-vertex/anthropic") &&
638
+ (modelId.includes("k2p5") || modelId.includes("kimi-k2.5") || modelId.includes("kimi-k2p5"))
639
+ ) {
640
+ result["thinking"] = {
641
+ type: "enabled",
642
+ budgetTokens: Math.min(16_000, Math.floor(input.model.limit.output / 2 - 1)),
643
+ }
644
+ }
645
+
646
+ if (input.model.api.id.includes("gpt-5") && !input.model.api.id.includes("gpt-5-chat")) {
647
+ if (!input.model.api.id.includes("gpt-5-pro")) {
648
+ result["reasoningEffort"] = "medium"
649
+ result["reasoningSummary"] = "auto"
650
+ }
651
+
652
+ // Only set textVerbosity for non-chat gpt-5.x models
653
+ // Chat models (e.g. gpt-5.2-chat-latest) only support "medium" verbosity
654
+ if (
655
+ input.model.api.id.includes("gpt-5.") &&
656
+ !input.model.api.id.includes("codex") &&
657
+ !input.model.api.id.includes("-chat") &&
658
+ input.model.providerID !== "azure"
659
+ ) {
660
+ result["textVerbosity"] = "low"
661
+ }
662
+
663
+ if (input.model.providerID.startsWith("opencode")) {
664
+ result["promptCacheKey"] = input.sessionID
665
+ result["include"] = ["reasoning.encrypted_content"]
666
+ result["reasoningSummary"] = "auto"
667
+ }
668
+ }
669
+
670
+ if (input.model.providerID === "venice") {
671
+ result["promptCacheKey"] = input.sessionID
672
+ }
673
+
674
+ return result
675
+ }
676
+
677
+ export function smallOptions(model: Provider.Model) {
678
+ if (
679
+ model.providerID === "openai" ||
680
+ model.api.npm === "@ai-sdk/openai" ||
681
+ model.api.npm === "@ai-sdk/github-copilot"
682
+ ) {
683
+ if (model.api.id.includes("gpt-5")) {
684
+ if (model.api.id.includes("5.")) {
685
+ return { store: false, reasoningEffort: "low" }
686
+ }
687
+ return { store: false, reasoningEffort: "minimal" }
688
+ }
689
+ return { store: false }
690
+ }
691
+ if (model.providerID === "google") {
692
+ // gemini-3 uses thinkingLevel, gemini-2.5 uses thinkingBudget
693
+ if (model.api.id.includes("gemini-3")) {
694
+ return { thinkingConfig: { thinkingLevel: "minimal" } }
695
+ }
696
+ return { thinkingConfig: { thinkingBudget: 0 } }
697
+ }
698
+ if (model.providerID === "openrouter") {
699
+ if (model.api.id.includes("google")) {
700
+ return { reasoning: { enabled: false } }
701
+ }
702
+ return { reasoningEffort: "minimal" }
703
+ }
704
+ return {}
705
+ }
706
+
707
+ export function providerOptions(model: Provider.Model, options: { [x: string]: any }) {
708
+ const key = sdkKey(model.api.npm) ?? model.providerID
709
+ return { [key]: options }
710
+ }
711
+
712
+ export function maxOutputTokens(
713
+ npm: string,
714
+ options: Record<string, any>,
715
+ modelLimit: number,
716
+ globalLimit: number,
717
+ ): number {
718
+ const modelCap = modelLimit || globalLimit
719
+ const standardLimit = Math.min(modelCap, globalLimit)
720
+
721
+ if (npm === "@ai-sdk/anthropic" || npm === "@ai-sdk/google-vertex/anthropic") {
722
+ const thinking = options?.["thinking"]
723
+ const budgetTokens = typeof thinking?.["budgetTokens"] === "number" ? thinking["budgetTokens"] : 0
724
+ const enabled = thinking?.["type"] === "enabled"
725
+ if (enabled && budgetTokens > 0) {
726
+ // Return text tokens so that text + thinking <= model cap, preferring 32k text when possible.
727
+ if (budgetTokens + standardLimit <= modelCap) {
728
+ return standardLimit
729
+ }
730
+ return modelCap - budgetTokens
731
+ }
732
+ }
733
+
734
+ return standardLimit
735
+ }
736
+
737
+ export function schema(model: Provider.Model, schema: JSONSchema.BaseSchema | JSONSchema7): JSONSchema7 {
738
+ /*
739
+ if (["openai", "azure"].includes(providerID)) {
740
+ if (schema.type === "object" && schema.properties) {
741
+ for (const [key, value] of Object.entries(schema.properties)) {
742
+ if (schema.required?.includes(key)) continue
743
+ schema.properties[key] = {
744
+ anyOf: [
745
+ value as JSONSchema.JSONSchema,
746
+ {
747
+ type: "null",
748
+ },
749
+ ],
750
+ }
751
+ }
752
+ }
753
+ }
754
+ */
755
+
756
+ // Convert integer enums to string enums for Google/Gemini
757
+ if (model.providerID === "google" || model.api.id.includes("gemini")) {
758
+ const sanitizeGemini = (obj: any): any => {
759
+ if (obj === null || typeof obj !== "object") {
760
+ return obj
761
+ }
762
+
763
+ if (Array.isArray(obj)) {
764
+ return obj.map(sanitizeGemini)
765
+ }
766
+
767
+ const result: any = {}
768
+ for (const [key, value] of Object.entries(obj)) {
769
+ if (key === "enum" && Array.isArray(value)) {
770
+ // Convert all enum values to strings
771
+ result[key] = value.map((v) => String(v))
772
+ // If we have integer type with enum, change type to string
773
+ if (result.type === "integer" || result.type === "number") {
774
+ result.type = "string"
775
+ }
776
+ } else if (typeof value === "object" && value !== null) {
777
+ result[key] = sanitizeGemini(value)
778
+ } else {
779
+ result[key] = value
780
+ }
781
+ }
782
+
783
+ // Filter required array to only include fields that exist in properties
784
+ if (result.type === "object" && result.properties && Array.isArray(result.required)) {
785
+ result.required = result.required.filter((field: any) => field in result.properties)
786
+ }
787
+
788
+ if (result.type === "array") {
789
+ if (result.items == null) {
790
+ result.items = {}
791
+ }
792
+ // Ensure items has at least a type if it's an empty object
793
+ // This handles nested arrays like { type: "array", items: { type: "array", items: {} } }
794
+ if (typeof result.items === "object" && !Array.isArray(result.items) && !result.items.type) {
795
+ result.items.type = "string"
796
+ }
797
+ }
798
+
799
+ // Remove properties/required from non-object types (Gemini rejects these)
800
+ if (result.type && result.type !== "object") {
801
+ delete result.properties
802
+ delete result.required
803
+ }
804
+
805
+ return result
806
+ }
807
+
808
+ schema = sanitizeGemini(schema)
809
+ }
810
+
811
+ return schema as JSONSchema7
812
+ }
813
+
814
+ export function error(providerID: string, error: APICallError) {
815
+ let message = error.message
816
+ if (providerID.includes("github-copilot") && error.statusCode === 403) {
817
+ return "Please reauthenticate with the copilot provider to ensure your credentials work properly with OpenCode."
818
+ }
819
+ if (providerID.includes("github-copilot") && message.includes("The requested model is not supported")) {
820
+ return (
821
+ message +
822
+ "\n\nMake sure the model is enabled in your copilot settings: https://github.com/settings/copilot/features"
823
+ )
824
+ }
825
+
826
+ return message
827
+ }
828
+ }