nuwaxcode 1.1.25

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 (397) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/nuwaxcode +84 -0
  5. package/bunfig.toml +7 -0
  6. package/package.json +123 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +172 -0
  9. package/script/postinstall.mjs +125 -0
  10. package/script/publish-registries.ts +187 -0
  11. package/script/publish.ts +70 -0
  12. package/script/schema.ts +47 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +1280 -0
  15. package/src/acp/session.ts +111 -0
  16. package/src/acp/types.ts +24 -0
  17. package/src/agent/agent.ts +332 -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 +43 -0
  23. package/src/auth/index.ts +73 -0
  24. package/src/bun/index.ts +134 -0
  25. package/src/bus/bus-event.ts +43 -0
  26. package/src/bus/global.ts +10 -0
  27. package/src/bus/index.ts +105 -0
  28. package/src/cli/bootstrap.ts +17 -0
  29. package/src/cli/cmd/acp.ts +69 -0
  30. package/src/cli/cmd/agent.ts +257 -0
  31. package/src/cli/cmd/auth.ts +400 -0
  32. package/src/cli/cmd/cmd.ts +7 -0
  33. package/src/cli/cmd/debug/agent.ts +166 -0
  34. package/src/cli/cmd/debug/config.ts +16 -0
  35. package/src/cli/cmd/debug/file.ts +97 -0
  36. package/src/cli/cmd/debug/index.ts +48 -0
  37. package/src/cli/cmd/debug/lsp.ts +52 -0
  38. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  39. package/src/cli/cmd/debug/scrap.ts +16 -0
  40. package/src/cli/cmd/debug/skill.ts +16 -0
  41. package/src/cli/cmd/debug/snapshot.ts +52 -0
  42. package/src/cli/cmd/export.ts +88 -0
  43. package/src/cli/cmd/generate.ts +38 -0
  44. package/src/cli/cmd/github.ts +1548 -0
  45. package/src/cli/cmd/import.ts +98 -0
  46. package/src/cli/cmd/mcp.ts +755 -0
  47. package/src/cli/cmd/models.ts +77 -0
  48. package/src/cli/cmd/pr.ts +112 -0
  49. package/src/cli/cmd/run.ts +395 -0
  50. package/src/cli/cmd/serve.ts +20 -0
  51. package/src/cli/cmd/session.ts +135 -0
  52. package/src/cli/cmd/stats.ts +402 -0
  53. package/src/cli/cmd/tui/app.tsx +761 -0
  54. package/src/cli/cmd/tui/attach.ts +31 -0
  55. package/src/cli/cmd/tui/component/border.tsx +21 -0
  56. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  57. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  58. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  59. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  60. package/src/cli/cmd/tui/component/dialog-provider.tsx +256 -0
  61. package/src/cli/cmd/tui/component/dialog-session-list.tsx +114 -0
  62. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  63. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  64. package/src/cli/cmd/tui/component/dialog-status.tsx +164 -0
  65. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  66. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  67. package/src/cli/cmd/tui/component/logo.tsx +88 -0
  68. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +632 -0
  69. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  70. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  71. package/src/cli/cmd/tui/component/prompt/index.tsx +1096 -0
  72. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  73. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  74. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  75. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  76. package/src/cli/cmd/tui/context/args.tsx +14 -0
  77. package/src/cli/cmd/tui/context/directory.ts +13 -0
  78. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  79. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  80. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  81. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  82. package/src/cli/cmd/tui/context/local.tsx +402 -0
  83. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  84. package/src/cli/cmd/tui/context/route.tsx +46 -0
  85. package/src/cli/cmd/tui/context/sdk.tsx +94 -0
  86. package/src/cli/cmd/tui/context/sync.tsx +427 -0
  87. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  88. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  89. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  90. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  91. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  93. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  94. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  95. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  96. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  97. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  98. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  99. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  100. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  101. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  102. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  103. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  105. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  106. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  107. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  108. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  109. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  110. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  111. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  112. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  113. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  114. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  115. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  116. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  117. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  118. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  119. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  120. package/src/cli/cmd/tui/event.ts +48 -0
  121. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  122. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  123. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  126. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  127. package/src/cli/cmd/tui/routes/session/header.tsx +136 -0
  128. package/src/cli/cmd/tui/routes/session/index.tsx +2050 -0
  129. package/src/cli/cmd/tui/routes/session/permission.tsx +495 -0
  130. package/src/cli/cmd/tui/routes/session/question.tsx +435 -0
  131. package/src/cli/cmd/tui/routes/session/sidebar.tsx +312 -0
  132. package/src/cli/cmd/tui/thread.ts +165 -0
  133. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  134. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  135. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  136. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  137. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  138. package/src/cli/cmd/tui/ui/dialog-select.tsx +354 -0
  139. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  140. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  141. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  142. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  143. package/src/cli/cmd/tui/util/clipboard.ts +160 -0
  144. package/src/cli/cmd/tui/util/editor.ts +32 -0
  145. package/src/cli/cmd/tui/util/signal.ts +7 -0
  146. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  147. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  148. package/src/cli/cmd/tui/worker.ts +152 -0
  149. package/src/cli/cmd/uninstall.ts +357 -0
  150. package/src/cli/cmd/upgrade.ts +73 -0
  151. package/src/cli/cmd/web.ts +81 -0
  152. package/src/cli/error.ts +57 -0
  153. package/src/cli/network.ts +53 -0
  154. package/src/cli/ui.ts +84 -0
  155. package/src/cli/upgrade.ts +25 -0
  156. package/src/command/index.ts +131 -0
  157. package/src/command/template/initialize.txt +10 -0
  158. package/src/command/template/review.txt +99 -0
  159. package/src/config/config.ts +1255 -0
  160. package/src/config/markdown.ts +93 -0
  161. package/src/env/index.ts +26 -0
  162. package/src/file/ignore.ts +83 -0
  163. package/src/file/index.ts +411 -0
  164. package/src/file/ripgrep.ts +409 -0
  165. package/src/file/time.ts +64 -0
  166. package/src/file/watcher.ts +118 -0
  167. package/src/flag/flag.ts +54 -0
  168. package/src/format/formatter.ts +359 -0
  169. package/src/format/index.ts +137 -0
  170. package/src/global/index.ts +55 -0
  171. package/src/id/id.ts +83 -0
  172. package/src/ide/index.ts +76 -0
  173. package/src/index.ts +159 -0
  174. package/src/installation/index.ts +246 -0
  175. package/src/lsp/client.ts +252 -0
  176. package/src/lsp/index.ts +485 -0
  177. package/src/lsp/language.ts +119 -0
  178. package/src/lsp/server.ts +2046 -0
  179. package/src/mcp/auth.ts +135 -0
  180. package/src/mcp/index.ts +926 -0
  181. package/src/mcp/oauth-callback.ts +200 -0
  182. package/src/mcp/oauth-provider.ts +154 -0
  183. package/src/patch/index.ts +680 -0
  184. package/src/permission/arity.ts +163 -0
  185. package/src/permission/index.ts +210 -0
  186. package/src/permission/next.ts +269 -0
  187. package/src/plugin/codex.ts +493 -0
  188. package/src/plugin/copilot.ts +269 -0
  189. package/src/plugin/index.ts +135 -0
  190. package/src/project/bootstrap.ts +35 -0
  191. package/src/project/instance.ts +91 -0
  192. package/src/project/project.ts +320 -0
  193. package/src/project/state.ts +66 -0
  194. package/src/project/vcs.ts +76 -0
  195. package/src/provider/auth.ts +147 -0
  196. package/src/provider/models-macro.ts +11 -0
  197. package/src/provider/models.ts +112 -0
  198. package/src/provider/provider.ts +1219 -0
  199. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  200. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  201. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  202. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  203. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  204. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  205. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  206. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  207. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
  208. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  209. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  210. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  211. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  212. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  213. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  214. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  215. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  216. package/src/provider/transform.ts +724 -0
  217. package/src/pty/index.ts +229 -0
  218. package/src/question/index.ts +171 -0
  219. package/src/scheduler/index.ts +61 -0
  220. package/src/server/error.ts +36 -0
  221. package/src/server/mdns.ts +59 -0
  222. package/src/server/routes/config.ts +92 -0
  223. package/src/server/routes/experimental.ts +157 -0
  224. package/src/server/routes/file.ts +197 -0
  225. package/src/server/routes/global.ts +135 -0
  226. package/src/server/routes/mcp.ts +225 -0
  227. package/src/server/routes/permission.ts +68 -0
  228. package/src/server/routes/project.ts +82 -0
  229. package/src/server/routes/provider.ts +165 -0
  230. package/src/server/routes/pty.ts +169 -0
  231. package/src/server/routes/question.ts +98 -0
  232. package/src/server/routes/session.ts +935 -0
  233. package/src/server/routes/tui.ts +379 -0
  234. package/src/server/server.ts +578 -0
  235. package/src/session/compaction.ts +225 -0
  236. package/src/session/index.ts +488 -0
  237. package/src/session/llm.ts +279 -0
  238. package/src/session/message-v2.ts +702 -0
  239. package/src/session/message.ts +189 -0
  240. package/src/session/processor.ts +406 -0
  241. package/src/session/prompt/anthropic-20250930.txt +166 -0
  242. package/src/session/prompt/anthropic.txt +105 -0
  243. package/src/session/prompt/anthropic_spoof.txt +1 -0
  244. package/src/session/prompt/beast.txt +147 -0
  245. package/src/session/prompt/build-switch.txt +5 -0
  246. package/src/session/prompt/codex.txt +73 -0
  247. package/src/session/prompt/codex_header.txt +72 -0
  248. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  249. package/src/session/prompt/gemini.txt +155 -0
  250. package/src/session/prompt/max-steps.txt +16 -0
  251. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  252. package/src/session/prompt/plan.txt +26 -0
  253. package/src/session/prompt/qwen.txt +109 -0
  254. package/src/session/prompt.ts +1805 -0
  255. package/src/session/retry.ts +90 -0
  256. package/src/session/revert.ts +108 -0
  257. package/src/session/status.ts +76 -0
  258. package/src/session/summary.ts +150 -0
  259. package/src/session/system.ts +138 -0
  260. package/src/session/todo.ts +37 -0
  261. package/src/share/share-next.ts +194 -0
  262. package/src/share/share.ts +87 -0
  263. package/src/shell/shell.ts +67 -0
  264. package/src/skill/index.ts +1 -0
  265. package/src/skill/skill.ts +136 -0
  266. package/src/snapshot/index.ts +236 -0
  267. package/src/storage/storage.ts +227 -0
  268. package/src/tool/apply_patch.ts +277 -0
  269. package/src/tool/apply_patch.txt +1 -0
  270. package/src/tool/bash.ts +258 -0
  271. package/src/tool/bash.txt +115 -0
  272. package/src/tool/batch.ts +175 -0
  273. package/src/tool/batch.txt +24 -0
  274. package/src/tool/codesearch.ts +132 -0
  275. package/src/tool/codesearch.txt +12 -0
  276. package/src/tool/edit.ts +645 -0
  277. package/src/tool/edit.txt +10 -0
  278. package/src/tool/external-directory.ts +32 -0
  279. package/src/tool/glob.ts +77 -0
  280. package/src/tool/glob.txt +6 -0
  281. package/src/tool/grep.ts +154 -0
  282. package/src/tool/grep.txt +8 -0
  283. package/src/tool/invalid.ts +17 -0
  284. package/src/tool/ls.ts +121 -0
  285. package/src/tool/ls.txt +1 -0
  286. package/src/tool/lsp.ts +96 -0
  287. package/src/tool/lsp.txt +19 -0
  288. package/src/tool/multiedit.ts +46 -0
  289. package/src/tool/multiedit.txt +41 -0
  290. package/src/tool/plan-enter.txt +14 -0
  291. package/src/tool/plan-exit.txt +13 -0
  292. package/src/tool/plan.ts +130 -0
  293. package/src/tool/question.ts +33 -0
  294. package/src/tool/question.txt +10 -0
  295. package/src/tool/read.ts +202 -0
  296. package/src/tool/read.txt +12 -0
  297. package/src/tool/registry.ts +158 -0
  298. package/src/tool/skill.ts +75 -0
  299. package/src/tool/task.ts +188 -0
  300. package/src/tool/task.txt +60 -0
  301. package/src/tool/todo.ts +53 -0
  302. package/src/tool/todoread.txt +14 -0
  303. package/src/tool/todowrite.txt +167 -0
  304. package/src/tool/tool.ts +88 -0
  305. package/src/tool/truncation.ts +106 -0
  306. package/src/tool/webfetch.ts +182 -0
  307. package/src/tool/webfetch.txt +13 -0
  308. package/src/tool/websearch.ts +150 -0
  309. package/src/tool/websearch.txt +14 -0
  310. package/src/tool/write.ts +80 -0
  311. package/src/tool/write.txt +8 -0
  312. package/src/util/archive.ts +16 -0
  313. package/src/util/color.ts +19 -0
  314. package/src/util/context.ts +25 -0
  315. package/src/util/defer.ts +12 -0
  316. package/src/util/eventloop.ts +20 -0
  317. package/src/util/filesystem.ts +93 -0
  318. package/src/util/fn.ts +11 -0
  319. package/src/util/format.ts +20 -0
  320. package/src/util/iife.ts +3 -0
  321. package/src/util/keybind.ts +103 -0
  322. package/src/util/lazy.ts +18 -0
  323. package/src/util/locale.ts +81 -0
  324. package/src/util/lock.ts +98 -0
  325. package/src/util/log.ts +180 -0
  326. package/src/util/queue.ts +32 -0
  327. package/src/util/rpc.ts +66 -0
  328. package/src/util/scrap.ts +10 -0
  329. package/src/util/signal.ts +12 -0
  330. package/src/util/timeout.ts +14 -0
  331. package/src/util/token.ts +7 -0
  332. package/src/util/wildcard.ts +56 -0
  333. package/src/worktree/index.ts +217 -0
  334. package/sst-env.d.ts +9 -0
  335. package/test/acp/event-subscription.test.ts +436 -0
  336. package/test/agent/agent.test.ts +638 -0
  337. package/test/bun.test.ts +53 -0
  338. package/test/cli/github-action.test.ts +129 -0
  339. package/test/cli/github-remote.test.ts +80 -0
  340. package/test/cli/tui/transcript.test.ts +297 -0
  341. package/test/config/agent-color.test.ts +66 -0
  342. package/test/config/config.test.ts +1414 -0
  343. package/test/config/fixtures/empty-frontmatter.md +4 -0
  344. package/test/config/fixtures/frontmatter.md +28 -0
  345. package/test/config/fixtures/no-frontmatter.md +1 -0
  346. package/test/config/markdown.test.ts +192 -0
  347. package/test/file/ignore.test.ts +10 -0
  348. package/test/file/path-traversal.test.ts +198 -0
  349. package/test/fixture/fixture.ts +45 -0
  350. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  351. package/test/ide/ide.test.ts +82 -0
  352. package/test/keybind.test.ts +421 -0
  353. package/test/lsp/client.test.ts +95 -0
  354. package/test/mcp/headers.test.ts +153 -0
  355. package/test/mcp/oauth-browser.test.ts +261 -0
  356. package/test/patch/patch.test.ts +348 -0
  357. package/test/permission/arity.test.ts +33 -0
  358. package/test/permission/next.test.ts +652 -0
  359. package/test/permission-task.test.ts +319 -0
  360. package/test/plugin/codex.test.ts +123 -0
  361. package/test/preload.ts +65 -0
  362. package/test/project/project.test.ts +120 -0
  363. package/test/provider/amazon-bedrock.test.ts +268 -0
  364. package/test/provider/gitlab-duo.test.ts +286 -0
  365. package/test/provider/provider.test.ts +2149 -0
  366. package/test/provider/transform.test.ts +1596 -0
  367. package/test/question/question.test.ts +300 -0
  368. package/test/scheduler.test.ts +73 -0
  369. package/test/server/session-list.test.ts +39 -0
  370. package/test/server/session-select.test.ts +78 -0
  371. package/test/session/compaction.test.ts +293 -0
  372. package/test/session/llm.test.ts +90 -0
  373. package/test/session/message-v2.test.ts +662 -0
  374. package/test/session/retry.test.ts +131 -0
  375. package/test/session/revert-compact.test.ts +285 -0
  376. package/test/session/session.test.ts +71 -0
  377. package/test/skill/skill.test.ts +185 -0
  378. package/test/snapshot/snapshot.test.ts +939 -0
  379. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  380. package/test/tool/apply_patch.test.ts +515 -0
  381. package/test/tool/bash.test.ts +320 -0
  382. package/test/tool/external-directory.test.ts +126 -0
  383. package/test/tool/fixtures/large-image.png +0 -0
  384. package/test/tool/fixtures/models-api.json +33453 -0
  385. package/test/tool/grep.test.ts +109 -0
  386. package/test/tool/question.test.ts +105 -0
  387. package/test/tool/read.test.ts +332 -0
  388. package/test/tool/registry.test.ts +76 -0
  389. package/test/tool/truncation.test.ts +159 -0
  390. package/test/util/filesystem.test.ts +39 -0
  391. package/test/util/format.test.ts +59 -0
  392. package/test/util/iife.test.ts +36 -0
  393. package/test/util/lazy.test.ts +50 -0
  394. package/test/util/lock.test.ts +72 -0
  395. package/test/util/timeout.test.ts +21 -0
  396. package/test/util/wildcard.test.ts +75 -0
  397. package/tsconfig.json +16 -0
@@ -0,0 +1,724 @@
1
+ import type { APICallError, ModelMessage } from "ai"
2
+ import { unique } from "remeda"
3
+ import type { JSONSchema } from "zod/v4/core"
4
+ import type { Provider } from "./provider"
5
+ import type { ModelsDev } from "./models"
6
+ import { iife } from "@/util/iife"
7
+
8
+ type Modality = NonNullable<ModelsDev.Model["modalities"]>["input"][number]
9
+
10
+ function mimeToModality(mime: string): Modality | undefined {
11
+ if (mime.startsWith("image/")) return "image"
12
+ if (mime.startsWith("audio/")) return "audio"
13
+ if (mime.startsWith("video/")) return "video"
14
+ if (mime === "application/pdf") return "pdf"
15
+ return undefined
16
+ }
17
+
18
+ export namespace ProviderTransform {
19
+ // Maps npm package to the key the AI SDK expects for providerOptions
20
+ function sdkKey(npm: string): string | undefined {
21
+ switch (npm) {
22
+ case "@ai-sdk/github-copilot":
23
+ case "@ai-sdk/openai":
24
+ case "@ai-sdk/azure":
25
+ return "openai"
26
+ case "@ai-sdk/amazon-bedrock":
27
+ return "bedrock"
28
+ case "@ai-sdk/anthropic":
29
+ return "anthropic"
30
+ case "@ai-sdk/google-vertex":
31
+ case "@ai-sdk/google":
32
+ return "google"
33
+ case "@ai-sdk/gateway":
34
+ return "gateway"
35
+ case "@openrouter/ai-sdk-provider":
36
+ return "openrouter"
37
+ }
38
+ return undefined
39
+ }
40
+
41
+ function normalizeMessages(
42
+ msgs: ModelMessage[],
43
+ model: Provider.Model,
44
+ options: Record<string, unknown>,
45
+ ): ModelMessage[] {
46
+ // Anthropic rejects messages with empty content - filter out empty string messages
47
+ // and remove empty text/reasoning parts from array content
48
+ if (model.api.npm === "@ai-sdk/anthropic") {
49
+ msgs = msgs
50
+ .map((msg) => {
51
+ if (typeof msg.content === "string") {
52
+ if (msg.content === "") return undefined
53
+ return msg
54
+ }
55
+ if (!Array.isArray(msg.content)) return msg
56
+ const filtered = msg.content.filter((part) => {
57
+ if (part.type === "text" || part.type === "reasoning") {
58
+ return part.text !== ""
59
+ }
60
+ return true
61
+ })
62
+ if (filtered.length === 0) return undefined
63
+ return { ...msg, content: filtered }
64
+ })
65
+ .filter((msg): msg is ModelMessage => msg !== undefined && msg.content !== "")
66
+ }
67
+
68
+ if (model.api.id.includes("claude")) {
69
+ return msgs.map((msg) => {
70
+ if ((msg.role === "assistant" || msg.role === "tool") && Array.isArray(msg.content)) {
71
+ msg.content = msg.content.map((part) => {
72
+ if ((part.type === "tool-call" || part.type === "tool-result") && "toolCallId" in part) {
73
+ return {
74
+ ...part,
75
+ toolCallId: part.toolCallId.replace(/[^a-zA-Z0-9_-]/g, "_"),
76
+ }
77
+ }
78
+ return part
79
+ })
80
+ }
81
+ return msg
82
+ })
83
+ }
84
+ if (model.providerID === "mistral" || model.api.id.toLowerCase().includes("mistral")) {
85
+ const result: ModelMessage[] = []
86
+ for (let i = 0; i < msgs.length; i++) {
87
+ const msg = msgs[i]
88
+ const nextMsg = msgs[i + 1]
89
+
90
+ if ((msg.role === "assistant" || msg.role === "tool") && Array.isArray(msg.content)) {
91
+ msg.content = msg.content.map((part) => {
92
+ if ((part.type === "tool-call" || part.type === "tool-result") && "toolCallId" in part) {
93
+ // Mistral requires alphanumeric tool call IDs with exactly 9 characters
94
+ const normalizedId = part.toolCallId
95
+ .replace(/[^a-zA-Z0-9]/g, "") // Remove non-alphanumeric characters
96
+ .substring(0, 9) // Take first 9 characters
97
+ .padEnd(9, "0") // Pad with zeros if less than 9 characters
98
+
99
+ return {
100
+ ...part,
101
+ toolCallId: normalizedId,
102
+ }
103
+ }
104
+ return part
105
+ })
106
+ }
107
+
108
+ result.push(msg)
109
+
110
+ // Fix message sequence: tool messages cannot be followed by user messages
111
+ if (msg.role === "tool" && nextMsg?.role === "user") {
112
+ result.push({
113
+ role: "assistant",
114
+ content: [
115
+ {
116
+ type: "text",
117
+ text: "Done.",
118
+ },
119
+ ],
120
+ })
121
+ }
122
+ }
123
+ return result
124
+ }
125
+
126
+ if (
127
+ model.capabilities.interleaved &&
128
+ typeof model.capabilities.interleaved === "object" &&
129
+ model.capabilities.interleaved.field === "reasoning_content"
130
+ ) {
131
+ return msgs.map((msg) => {
132
+ if (msg.role === "assistant" && Array.isArray(msg.content)) {
133
+ const reasoningParts = msg.content.filter((part: any) => part.type === "reasoning")
134
+ const reasoningText = reasoningParts.map((part: any) => part.text).join("")
135
+
136
+ // Filter out reasoning parts from content
137
+ const filteredContent = msg.content.filter((part: any) => part.type !== "reasoning")
138
+
139
+ // Include reasoning_content directly on the message for all assistant messages
140
+ if (reasoningText) {
141
+ return {
142
+ ...msg,
143
+ content: filteredContent,
144
+ providerOptions: {
145
+ ...msg.providerOptions,
146
+ openaiCompatible: {
147
+ ...(msg.providerOptions as any)?.openaiCompatible,
148
+ reasoning_content: reasoningText,
149
+ },
150
+ },
151
+ }
152
+ }
153
+
154
+ return {
155
+ ...msg,
156
+ content: filteredContent,
157
+ }
158
+ }
159
+
160
+ return msg
161
+ })
162
+ }
163
+
164
+ return msgs
165
+ }
166
+
167
+ function applyCaching(msgs: ModelMessage[], providerID: string): ModelMessage[] {
168
+ const system = msgs.filter((msg) => msg.role === "system").slice(0, 2)
169
+ const final = msgs.filter((msg) => msg.role !== "system").slice(-2)
170
+
171
+ const providerOptions = {
172
+ anthropic: {
173
+ cacheControl: { type: "ephemeral" },
174
+ },
175
+ openrouter: {
176
+ cacheControl: { type: "ephemeral" },
177
+ },
178
+ bedrock: {
179
+ cachePoint: { type: "ephemeral" },
180
+ },
181
+ openaiCompatible: {
182
+ cache_control: { type: "ephemeral" },
183
+ },
184
+ }
185
+
186
+ for (const msg of unique([...system, ...final])) {
187
+ const shouldUseContentOptions = providerID !== "anthropic" && Array.isArray(msg.content) && msg.content.length > 0
188
+
189
+ if (shouldUseContentOptions) {
190
+ const lastContent = msg.content[msg.content.length - 1]
191
+ if (lastContent && typeof lastContent === "object") {
192
+ lastContent.providerOptions = {
193
+ ...lastContent.providerOptions,
194
+ ...providerOptions,
195
+ }
196
+ continue
197
+ }
198
+ }
199
+
200
+ msg.providerOptions = {
201
+ ...msg.providerOptions,
202
+ ...providerOptions,
203
+ }
204
+ }
205
+
206
+ return msgs
207
+ }
208
+
209
+ function unsupportedParts(msgs: ModelMessage[], model: Provider.Model): ModelMessage[] {
210
+ return msgs.map((msg) => {
211
+ if (msg.role !== "user" || !Array.isArray(msg.content)) return msg
212
+
213
+ const filtered = msg.content.map((part) => {
214
+ if (part.type !== "file" && part.type !== "image") return part
215
+
216
+ // Check for empty base64 image data
217
+ if (part.type === "image") {
218
+ const imageStr = part.image.toString()
219
+ if (imageStr.startsWith("data:")) {
220
+ const match = imageStr.match(/^data:([^;]+);base64,(.*)$/)
221
+ if (match && (!match[2] || match[2].length === 0)) {
222
+ return {
223
+ type: "text" as const,
224
+ text: "ERROR: Image file is empty or corrupted. Please provide a valid image.",
225
+ }
226
+ }
227
+ }
228
+ }
229
+
230
+ const mime = part.type === "image" ? part.image.toString().split(";")[0].replace("data:", "") : part.mediaType
231
+ const filename = part.type === "file" ? part.filename : undefined
232
+ const modality = mimeToModality(mime)
233
+ if (!modality) return part
234
+ if (model.capabilities.input[modality]) return part
235
+
236
+ const name = filename ? `"${filename}"` : modality
237
+ return {
238
+ type: "text" as const,
239
+ text: `ERROR: Cannot read ${name} (this model does not support ${modality} input). Inform the user.`,
240
+ }
241
+ })
242
+
243
+ return { ...msg, content: filtered }
244
+ })
245
+ }
246
+
247
+ export function message(msgs: ModelMessage[], model: Provider.Model, options: Record<string, unknown>) {
248
+ msgs = unsupportedParts(msgs, model)
249
+ msgs = normalizeMessages(msgs, model, options)
250
+ if (
251
+ model.providerID === "anthropic" ||
252
+ model.api.id.includes("anthropic") ||
253
+ model.api.id.includes("claude") ||
254
+ model.api.npm === "@ai-sdk/anthropic"
255
+ ) {
256
+ msgs = applyCaching(msgs, model.providerID)
257
+ }
258
+
259
+ // Remap providerOptions keys from stored providerID to expected SDK key
260
+ const key = sdkKey(model.api.npm)
261
+ if (key && key !== model.providerID && model.api.npm !== "@ai-sdk/azure") {
262
+ const remap = (opts: Record<string, any> | undefined) => {
263
+ if (!opts) return opts
264
+ if (!(model.providerID in opts)) return opts
265
+ const result = { ...opts }
266
+ result[key] = result[model.providerID]
267
+ delete result[model.providerID]
268
+ return result
269
+ }
270
+
271
+ msgs = msgs.map((msg) => {
272
+ if (!Array.isArray(msg.content)) return { ...msg, providerOptions: remap(msg.providerOptions) }
273
+ return {
274
+ ...msg,
275
+ providerOptions: remap(msg.providerOptions),
276
+ content: msg.content.map((part) => ({ ...part, providerOptions: remap(part.providerOptions) })),
277
+ } as typeof msg
278
+ })
279
+ }
280
+
281
+ return msgs
282
+ }
283
+
284
+ export function temperature(model: Provider.Model) {
285
+ const id = model.id.toLowerCase()
286
+ if (id.includes("qwen")) return 0.55
287
+ if (id.includes("claude")) return undefined
288
+ if (id.includes("gemini")) return 1.0
289
+ if (id.includes("glm-4.6")) return 1.0
290
+ if (id.includes("glm-4.7")) return 1.0
291
+ if (id.includes("minimax-m2")) return 1.0
292
+ if (id.includes("kimi-k2")) {
293
+ if (id.includes("thinking")) return 1.0
294
+ return 0.6
295
+ }
296
+ return undefined
297
+ }
298
+
299
+ export function topP(model: Provider.Model) {
300
+ const id = model.id.toLowerCase()
301
+ if (id.includes("qwen")) return 1
302
+ if (id.includes("minimax-m2")) {
303
+ return 0.95
304
+ }
305
+ if (id.includes("gemini")) return 0.95
306
+ return undefined
307
+ }
308
+
309
+ export function topK(model: Provider.Model) {
310
+ const id = model.id.toLowerCase()
311
+ if (id.includes("minimax-m2")) {
312
+ if (id.includes("m2.1")) return 40
313
+ return 20
314
+ }
315
+ if (id.includes("gemini")) return 64
316
+ return undefined
317
+ }
318
+
319
+ const WIDELY_SUPPORTED_EFFORTS = ["low", "medium", "high"]
320
+ const OPENAI_EFFORTS = ["none", "minimal", ...WIDELY_SUPPORTED_EFFORTS, "xhigh"]
321
+
322
+ export function variants(model: Provider.Model): Record<string, Record<string, any>> {
323
+ if (!model.capabilities.reasoning) return {}
324
+
325
+ const id = model.id.toLowerCase()
326
+ if (id.includes("deepseek") || id.includes("minimax") || id.includes("glm") || id.includes("mistral")) return {}
327
+
328
+ // see: https://docs.x.ai/docs/guides/reasoning#control-how-hard-the-model-thinks
329
+ if (id.includes("grok") && id.includes("grok-3-mini")) {
330
+ if (model.api.npm === "@openrouter/ai-sdk-provider") {
331
+ return {
332
+ low: { reasoning: { effort: "low" } },
333
+ high: { reasoning: { effort: "high" } },
334
+ }
335
+ }
336
+ return {
337
+ low: { reasoningEffort: "low" },
338
+ high: { reasoningEffort: "high" },
339
+ }
340
+ }
341
+ if (id.includes("grok")) return {}
342
+
343
+ switch (model.api.npm) {
344
+ case "@openrouter/ai-sdk-provider":
345
+ if (!model.id.includes("gpt") && !model.id.includes("gemini-3")) return {}
346
+ return Object.fromEntries(OPENAI_EFFORTS.map((effort) => [effort, { reasoning: { effort } }]))
347
+
348
+ // TODO: YOU CANNOT SET max_tokens if this is set!!!
349
+ case "@ai-sdk/gateway":
350
+ return Object.fromEntries(OPENAI_EFFORTS.map((effort) => [effort, { reasoningEffort: effort }]))
351
+
352
+ case "@ai-sdk/cerebras":
353
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/cerebras
354
+ case "@ai-sdk/togetherai":
355
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/togetherai
356
+ case "@ai-sdk/xai":
357
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/xai
358
+ case "@ai-sdk/deepinfra":
359
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/deepinfra
360
+ case "@ai-sdk/openai-compatible":
361
+ return Object.fromEntries(WIDELY_SUPPORTED_EFFORTS.map((effort) => [effort, { reasoningEffort: effort }]))
362
+
363
+ case "@ai-sdk/azure":
364
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/azure
365
+ if (id === "o1-mini") return {}
366
+ const azureEfforts = ["low", "medium", "high"]
367
+ if (id.includes("gpt-5-") || id === "gpt-5") {
368
+ azureEfforts.unshift("minimal")
369
+ }
370
+ return Object.fromEntries(
371
+ azureEfforts.map((effort) => [
372
+ effort,
373
+ {
374
+ reasoningEffort: effort,
375
+ reasoningSummary: "auto",
376
+ include: ["reasoning.encrypted_content"],
377
+ },
378
+ ]),
379
+ )
380
+ case "@ai-sdk/openai":
381
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/openai
382
+ if (id === "gpt-5-pro") return {}
383
+ const openaiEfforts = iife(() => {
384
+ if (id.includes("codex")) {
385
+ if (id.includes("5.2")) return [...WIDELY_SUPPORTED_EFFORTS, "xhigh"]
386
+ return WIDELY_SUPPORTED_EFFORTS
387
+ }
388
+ const arr = [...WIDELY_SUPPORTED_EFFORTS]
389
+ if (id.includes("gpt-5-") || id === "gpt-5") {
390
+ arr.unshift("minimal")
391
+ }
392
+ if (model.release_date >= "2025-11-13") {
393
+ arr.unshift("none")
394
+ }
395
+ if (model.release_date >= "2025-12-04") {
396
+ arr.push("xhigh")
397
+ }
398
+ return arr
399
+ })
400
+ return Object.fromEntries(
401
+ openaiEfforts.map((effort) => [
402
+ effort,
403
+ {
404
+ reasoningEffort: effort,
405
+ reasoningSummary: "auto",
406
+ include: ["reasoning.encrypted_content"],
407
+ },
408
+ ]),
409
+ )
410
+
411
+ case "@ai-sdk/anthropic":
412
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/anthropic
413
+ return {
414
+ high: {
415
+ thinking: {
416
+ type: "enabled",
417
+ budgetTokens: 16000,
418
+ },
419
+ },
420
+ max: {
421
+ thinking: {
422
+ type: "enabled",
423
+ budgetTokens: 31999,
424
+ },
425
+ },
426
+ }
427
+
428
+ case "@ai-sdk/amazon-bedrock":
429
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/amazon-bedrock
430
+ // For Anthropic models on Bedrock, use reasoningConfig with budgetTokens
431
+ if (model.api.id.includes("anthropic")) {
432
+ return {
433
+ high: {
434
+ reasoningConfig: {
435
+ type: "enabled",
436
+ budgetTokens: 16000,
437
+ },
438
+ },
439
+ max: {
440
+ reasoningConfig: {
441
+ type: "enabled",
442
+ budgetTokens: 31999,
443
+ },
444
+ },
445
+ }
446
+ }
447
+
448
+ // For Amazon Nova models, use reasoningConfig with maxReasoningEffort
449
+ return Object.fromEntries(
450
+ WIDELY_SUPPORTED_EFFORTS.map((effort) => [
451
+ effort,
452
+ {
453
+ reasoningConfig: {
454
+ type: "enabled",
455
+ maxReasoningEffort: effort,
456
+ },
457
+ },
458
+ ]),
459
+ )
460
+
461
+ case "@ai-sdk/google-vertex":
462
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/google-vertex
463
+ case "@ai-sdk/google":
464
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/google-generative-ai
465
+ if (id.includes("2.5")) {
466
+ return {
467
+ high: {
468
+ thinkingConfig: {
469
+ includeThoughts: true,
470
+ thinkingBudget: 16000,
471
+ },
472
+ },
473
+ max: {
474
+ thinkingConfig: {
475
+ includeThoughts: true,
476
+ thinkingBudget: 24576,
477
+ },
478
+ },
479
+ }
480
+ }
481
+ return Object.fromEntries(
482
+ ["low", "high"].map((effort) => [
483
+ effort,
484
+ {
485
+ includeThoughts: true,
486
+ thinkingLevel: effort,
487
+ },
488
+ ]),
489
+ )
490
+
491
+ case "@ai-sdk/mistral":
492
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/mistral
493
+ return {}
494
+
495
+ case "@ai-sdk/cohere":
496
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/cohere
497
+ return {}
498
+
499
+ case "@ai-sdk/groq":
500
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/groq
501
+ const groqEffort = ["none", ...WIDELY_SUPPORTED_EFFORTS]
502
+ return Object.fromEntries(
503
+ groqEffort.map((effort) => [
504
+ effort,
505
+ {
506
+ includeThoughts: true,
507
+ thinkingLevel: effort,
508
+ },
509
+ ]),
510
+ )
511
+
512
+ case "@ai-sdk/perplexity":
513
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/perplexity
514
+ return {}
515
+ }
516
+ return {}
517
+ }
518
+
519
+ export function options(input: {
520
+ model: Provider.Model
521
+ sessionID: string
522
+ providerOptions?: Record<string, any>
523
+ }): Record<string, any> {
524
+ const result: Record<string, any> = {}
525
+
526
+ // openai and providers using openai package should set store to false by default.
527
+ if (
528
+ input.model.providerID === "openai" ||
529
+ input.model.api.npm === "@ai-sdk/openai" ||
530
+ input.model.api.npm === "@ai-sdk/github-copilot"
531
+ ) {
532
+ result["store"] = false
533
+ }
534
+
535
+ if (input.model.api.npm === "@openrouter/ai-sdk-provider") {
536
+ result["usage"] = {
537
+ include: true,
538
+ }
539
+ if (input.model.api.id.includes("gemini-3")) {
540
+ result["reasoning"] = { effort: "high" }
541
+ }
542
+ }
543
+
544
+ if (
545
+ input.model.providerID === "baseten" ||
546
+ (input.model.providerID === "opencode" && ["kimi-k2-thinking", "glm-4.6"].includes(input.model.api.id))
547
+ ) {
548
+ result["chat_template_args"] = { enable_thinking: true }
549
+ }
550
+
551
+ if (["zai", "zhipuai"].includes(input.model.providerID) && input.model.api.npm === "@ai-sdk/openai-compatible") {
552
+ result["thinking"] = {
553
+ type: "enabled",
554
+ clear_thinking: false,
555
+ }
556
+ }
557
+
558
+ if (input.model.providerID === "openai" || input.providerOptions?.setCacheKey) {
559
+ result["promptCacheKey"] = input.sessionID
560
+ }
561
+
562
+ if (input.model.api.npm === "@ai-sdk/google" || input.model.api.npm === "@ai-sdk/google-vertex") {
563
+ result["thinkingConfig"] = {
564
+ includeThoughts: true,
565
+ }
566
+ if (input.model.api.id.includes("gemini-3")) {
567
+ result["thinkingConfig"]["thinkingLevel"] = "high"
568
+ }
569
+ }
570
+
571
+ if (input.model.api.id.includes("gpt-5") && !input.model.api.id.includes("gpt-5-chat")) {
572
+ if (input.model.providerID.includes("codex")) {
573
+ result["store"] = false
574
+ }
575
+
576
+ if (!input.model.api.id.includes("codex") && !input.model.api.id.includes("gpt-5-pro")) {
577
+ result["reasoningEffort"] = "medium"
578
+ }
579
+
580
+ if (input.model.api.id.endsWith("gpt-5.") && input.model.providerID !== "azure") {
581
+ result["textVerbosity"] = "low"
582
+ }
583
+
584
+ if (input.model.providerID.startsWith("opencode")) {
585
+ result["promptCacheKey"] = input.sessionID
586
+ result["include"] = ["reasoning.encrypted_content"]
587
+ result["reasoningSummary"] = "auto"
588
+ }
589
+ }
590
+ return result
591
+ }
592
+
593
+ export function smallOptions(model: Provider.Model) {
594
+ if (model.providerID === "openai" || model.api.id.includes("gpt-5")) {
595
+ if (model.api.id.includes("5.")) {
596
+ return { reasoningEffort: "low" }
597
+ }
598
+ return { reasoningEffort: "minimal" }
599
+ }
600
+ if (model.providerID === "google") {
601
+ // gemini-3 uses thinkingLevel, gemini-2.5 uses thinkingBudget
602
+ if (model.api.id.includes("gemini-3")) {
603
+ return { thinkingConfig: { thinkingLevel: "minimal" } }
604
+ }
605
+ return { thinkingConfig: { thinkingBudget: 0 } }
606
+ }
607
+ if (model.providerID === "openrouter") {
608
+ if (model.api.id.includes("google")) {
609
+ return { reasoning: { enabled: false } }
610
+ }
611
+ return { reasoningEffort: "minimal" }
612
+ }
613
+ return {}
614
+ }
615
+
616
+ export function providerOptions(model: Provider.Model, options: { [x: string]: any }) {
617
+ const key = sdkKey(model.api.npm) ?? model.providerID
618
+ return { [key]: options }
619
+ }
620
+
621
+ export function maxOutputTokens(
622
+ npm: string,
623
+ options: Record<string, any>,
624
+ modelLimit: number,
625
+ globalLimit: number,
626
+ ): number {
627
+ const modelCap = modelLimit || globalLimit
628
+ const standardLimit = Math.min(modelCap, globalLimit)
629
+
630
+ if (npm === "@ai-sdk/anthropic") {
631
+ const thinking = options?.["thinking"]
632
+ const budgetTokens = typeof thinking?.["budgetTokens"] === "number" ? thinking["budgetTokens"] : 0
633
+ const enabled = thinking?.["type"] === "enabled"
634
+ if (enabled && budgetTokens > 0) {
635
+ // Return text tokens so that text + thinking <= model cap, preferring 32k text when possible.
636
+ if (budgetTokens + standardLimit <= modelCap) {
637
+ return standardLimit
638
+ }
639
+ return modelCap - budgetTokens
640
+ }
641
+ }
642
+
643
+ return standardLimit
644
+ }
645
+
646
+ export function schema(model: Provider.Model, schema: JSONSchema.BaseSchema) {
647
+ /*
648
+ if (["openai", "azure"].includes(providerID)) {
649
+ if (schema.type === "object" && schema.properties) {
650
+ for (const [key, value] of Object.entries(schema.properties)) {
651
+ if (schema.required?.includes(key)) continue
652
+ schema.properties[key] = {
653
+ anyOf: [
654
+ value as JSONSchema.JSONSchema,
655
+ {
656
+ type: "null",
657
+ },
658
+ ],
659
+ }
660
+ }
661
+ }
662
+ }
663
+ */
664
+
665
+ // Convert integer enums to string enums for Google/Gemini
666
+ if (model.providerID === "google" || model.api.id.includes("gemini")) {
667
+ const sanitizeGemini = (obj: any): any => {
668
+ if (obj === null || typeof obj !== "object") {
669
+ return obj
670
+ }
671
+
672
+ if (Array.isArray(obj)) {
673
+ return obj.map(sanitizeGemini)
674
+ }
675
+
676
+ const result: any = {}
677
+ for (const [key, value] of Object.entries(obj)) {
678
+ if (key === "enum" && Array.isArray(value)) {
679
+ // Convert all enum values to strings
680
+ result[key] = value.map((v) => String(v))
681
+ // If we have integer type with enum, change type to string
682
+ if (result.type === "integer" || result.type === "number") {
683
+ result.type = "string"
684
+ }
685
+ } else if (typeof value === "object" && value !== null) {
686
+ result[key] = sanitizeGemini(value)
687
+ } else {
688
+ result[key] = value
689
+ }
690
+ }
691
+
692
+ // Filter required array to only include fields that exist in properties
693
+ if (result.type === "object" && result.properties && Array.isArray(result.required)) {
694
+ result.required = result.required.filter((field: any) => field in result.properties)
695
+ }
696
+
697
+ if (result.type === "array" && result.items == null) {
698
+ result.items = {}
699
+ }
700
+
701
+ return result
702
+ }
703
+
704
+ schema = sanitizeGemini(schema)
705
+ }
706
+
707
+ return schema
708
+ }
709
+
710
+ export function error(providerID: string, error: APICallError) {
711
+ let message = error.message
712
+ if (providerID.includes("github-copilot") && error.statusCode === 403) {
713
+ return "Please reauthenticate with the copilot provider to ensure your credentials work properly with OpenCode."
714
+ }
715
+ if (providerID.includes("github-copilot") && message.includes("The requested model is not supported")) {
716
+ return (
717
+ message +
718
+ "\n\nMake sure the model is enabled in your copilot settings: https://github.com/settings/copilot/features"
719
+ )
720
+ }
721
+
722
+ return message
723
+ }
724
+ }