codepro-ia 1.0.0

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 (393) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +70 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +7 -0
  6. package/package.json +144 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +167 -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 +1127 -0
  15. package/src/acp/session.ts +101 -0
  16. package/src/acp/types.ts +22 -0
  17. package/src/agent/agent.ts +311 -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 +724 -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 +124 -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 +106 -0
  68. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +763 -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 +1090 -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 +46 -0
  121. package/src/cli/cmd/tui/routes/home.tsx +141 -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 +1936 -0
  129. package/src/cli/cmd/tui/routes/session/permission.tsx +489 -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 +144 -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 +344 -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 +88 -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 +97 -0
  159. package/src/config/config.ts +1246 -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 +2032 -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 +622 -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 +31 -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 +1220 -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 +1713 -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 +705 -0
  217. package/src/pty/index.ts +229 -0
  218. package/src/question/index.ts +171 -0
  219. package/src/server/error.ts +36 -0
  220. package/src/server/mdns.ts +59 -0
  221. package/src/server/routes/config.ts +92 -0
  222. package/src/server/routes/experimental.ts +157 -0
  223. package/src/server/routes/file.ts +197 -0
  224. package/src/server/routes/global.ts +135 -0
  225. package/src/server/routes/mcp.ts +225 -0
  226. package/src/server/routes/permission.ts +68 -0
  227. package/src/server/routes/project.ts +82 -0
  228. package/src/server/routes/provider.ts +165 -0
  229. package/src/server/routes/pty.ts +169 -0
  230. package/src/server/routes/question.ts +98 -0
  231. package/src/server/routes/session.ts +935 -0
  232. package/src/server/routes/tui.ts +377 -0
  233. package/src/server/server.ts +578 -0
  234. package/src/session/compaction.ts +225 -0
  235. package/src/session/index.ts +488 -0
  236. package/src/session/llm.ts +279 -0
  237. package/src/session/message-v2.ts +702 -0
  238. package/src/session/message.ts +189 -0
  239. package/src/session/processor.ts +406 -0
  240. package/src/session/prompt/anthropic-20250930.txt +166 -0
  241. package/src/session/prompt/anthropic.txt +105 -0
  242. package/src/session/prompt/anthropic_spoof.txt +1 -0
  243. package/src/session/prompt/beast.txt +147 -0
  244. package/src/session/prompt/build-switch.txt +5 -0
  245. package/src/session/prompt/codex.txt +72 -0
  246. package/src/session/prompt/codex_header.txt +72 -0
  247. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  248. package/src/session/prompt/gemini.txt +155 -0
  249. package/src/session/prompt/max-steps.txt +16 -0
  250. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  251. package/src/session/prompt/plan.txt +26 -0
  252. package/src/session/prompt/qwen.txt +109 -0
  253. package/src/session/prompt.ts +1792 -0
  254. package/src/session/retry.ts +90 -0
  255. package/src/session/revert.ts +108 -0
  256. package/src/session/status.ts +76 -0
  257. package/src/session/summary.ts +150 -0
  258. package/src/session/system.ts +138 -0
  259. package/src/session/todo.ts +37 -0
  260. package/src/share/share-next.ts +194 -0
  261. package/src/share/share.ts +87 -0
  262. package/src/shell/shell.ts +67 -0
  263. package/src/skill/index.ts +1 -0
  264. package/src/skill/skill.ts +136 -0
  265. package/src/snapshot/index.ts +199 -0
  266. package/src/storage/storage.ts +227 -0
  267. package/src/tool/bash.ts +258 -0
  268. package/src/tool/bash.txt +115 -0
  269. package/src/tool/batch.ts +175 -0
  270. package/src/tool/batch.txt +24 -0
  271. package/src/tool/codesearch.ts +132 -0
  272. package/src/tool/codesearch.txt +12 -0
  273. package/src/tool/edit.ts +645 -0
  274. package/src/tool/edit.txt +10 -0
  275. package/src/tool/external-directory.ts +32 -0
  276. package/src/tool/glob.ts +77 -0
  277. package/src/tool/glob.txt +6 -0
  278. package/src/tool/grep.ts +154 -0
  279. package/src/tool/grep.txt +8 -0
  280. package/src/tool/invalid.ts +17 -0
  281. package/src/tool/ls.ts +121 -0
  282. package/src/tool/ls.txt +1 -0
  283. package/src/tool/lsp.ts +96 -0
  284. package/src/tool/lsp.txt +19 -0
  285. package/src/tool/multiedit.ts +46 -0
  286. package/src/tool/multiedit.txt +41 -0
  287. package/src/tool/patch.ts +201 -0
  288. package/src/tool/patch.txt +1 -0
  289. package/src/tool/plan-enter.txt +14 -0
  290. package/src/tool/plan-exit.txt +13 -0
  291. package/src/tool/plan.ts +130 -0
  292. package/src/tool/question.ts +33 -0
  293. package/src/tool/question.txt +10 -0
  294. package/src/tool/read.ts +200 -0
  295. package/src/tool/read.txt +12 -0
  296. package/src/tool/registry.ts +143 -0
  297. package/src/tool/skill.ts +75 -0
  298. package/src/tool/task.ts +188 -0
  299. package/src/tool/task.txt +60 -0
  300. package/src/tool/todo.ts +53 -0
  301. package/src/tool/todoread.txt +14 -0
  302. package/src/tool/todowrite.txt +167 -0
  303. package/src/tool/tool.ts +88 -0
  304. package/src/tool/truncation.ts +99 -0
  305. package/src/tool/webfetch.ts +182 -0
  306. package/src/tool/webfetch.txt +13 -0
  307. package/src/tool/websearch.ts +150 -0
  308. package/src/tool/websearch.txt +14 -0
  309. package/src/tool/write.ts +80 -0
  310. package/src/tool/write.txt +8 -0
  311. package/src/util/archive.ts +16 -0
  312. package/src/util/color.ts +19 -0
  313. package/src/util/context.ts +25 -0
  314. package/src/util/defer.ts +12 -0
  315. package/src/util/eventloop.ts +20 -0
  316. package/src/util/filesystem.ts +93 -0
  317. package/src/util/fn.ts +11 -0
  318. package/src/util/format.ts +20 -0
  319. package/src/util/iife.ts +3 -0
  320. package/src/util/keybind.ts +103 -0
  321. package/src/util/lazy.ts +18 -0
  322. package/src/util/locale.ts +81 -0
  323. package/src/util/lock.ts +98 -0
  324. package/src/util/log.ts +180 -0
  325. package/src/util/queue.ts +32 -0
  326. package/src/util/rpc.ts +66 -0
  327. package/src/util/scrap.ts +10 -0
  328. package/src/util/signal.ts +12 -0
  329. package/src/util/timeout.ts +14 -0
  330. package/src/util/token.ts +7 -0
  331. package/src/util/wildcard.ts +56 -0
  332. package/src/worktree/index.ts +217 -0
  333. package/sst-env.d.ts +9 -0
  334. package/test/agent/agent.test.ts +638 -0
  335. package/test/bun.test.ts +53 -0
  336. package/test/cli/github-action.test.ts +129 -0
  337. package/test/cli/github-remote.test.ts +80 -0
  338. package/test/cli/tui/transcript.test.ts +297 -0
  339. package/test/config/agent-color.test.ts +66 -0
  340. package/test/config/config.test.ts +1376 -0
  341. package/test/config/fixtures/empty-frontmatter.md +4 -0
  342. package/test/config/fixtures/frontmatter.md +28 -0
  343. package/test/config/fixtures/no-frontmatter.md +1 -0
  344. package/test/config/markdown.test.ts +192 -0
  345. package/test/file/ignore.test.ts +10 -0
  346. package/test/file/path-traversal.test.ts +198 -0
  347. package/test/fixture/fixture.ts +45 -0
  348. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  349. package/test/ide/ide.test.ts +82 -0
  350. package/test/keybind.test.ts +421 -0
  351. package/test/lsp/client.test.ts +95 -0
  352. package/test/mcp/headers.test.ts +153 -0
  353. package/test/mcp/oauth-browser.test.ts +261 -0
  354. package/test/patch/patch.test.ts +348 -0
  355. package/test/permission/arity.test.ts +33 -0
  356. package/test/permission/next.test.ts +652 -0
  357. package/test/permission-task.test.ts +319 -0
  358. package/test/plugin/codex.test.ts +123 -0
  359. package/test/preload.ts +65 -0
  360. package/test/project/project.test.ts +120 -0
  361. package/test/provider/amazon-bedrock.test.ts +268 -0
  362. package/test/provider/gitlab-duo.test.ts +286 -0
  363. package/test/provider/provider.test.ts +2149 -0
  364. package/test/provider/transform.test.ts +1566 -0
  365. package/test/question/question.test.ts +300 -0
  366. package/test/server/session-list.test.ts +39 -0
  367. package/test/server/session-select.test.ts +78 -0
  368. package/test/session/compaction.test.ts +293 -0
  369. package/test/session/llm.test.ts +90 -0
  370. package/test/session/message-v2.test.ts +662 -0
  371. package/test/session/retry.test.ts +131 -0
  372. package/test/session/revert-compact.test.ts +285 -0
  373. package/test/session/session.test.ts +71 -0
  374. package/test/skill/skill.test.ts +185 -0
  375. package/test/snapshot/snapshot.test.ts +939 -0
  376. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  377. package/test/tool/bash.test.ts +320 -0
  378. package/test/tool/external-directory.test.ts +126 -0
  379. package/test/tool/fixtures/large-image.png +0 -0
  380. package/test/tool/fixtures/models-api.json +33453 -0
  381. package/test/tool/grep.test.ts +109 -0
  382. package/test/tool/patch.test.ts +261 -0
  383. package/test/tool/read.test.ts +303 -0
  384. package/test/tool/registry.test.ts +76 -0
  385. package/test/tool/truncation.test.ts +159 -0
  386. package/test/util/filesystem.test.ts +39 -0
  387. package/test/util/format.test.ts +59 -0
  388. package/test/util/iife.test.ts +36 -0
  389. package/test/util/lazy.test.ts +50 -0
  390. package/test/util/lock.test.ts +72 -0
  391. package/test/util/timeout.test.ts +21 -0
  392. package/test/util/wildcard.test.ts +75 -0
  393. package/tsconfig.json +16 -0
@@ -0,0 +1,705 @@
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
+ switch (model.api.npm) {
329
+ case "@openrouter/ai-sdk-provider":
330
+ if (!model.id.includes("gpt") && !model.id.includes("gemini-3") && !model.id.includes("grok-4")) return {}
331
+ return Object.fromEntries(OPENAI_EFFORTS.map((effort) => [effort, { reasoning: { effort } }]))
332
+
333
+ // TODO: YOU CANNOT SET max_tokens if this is set!!!
334
+ case "@ai-sdk/gateway":
335
+ return Object.fromEntries(OPENAI_EFFORTS.map((effort) => [effort, { reasoningEffort: effort }]))
336
+
337
+ case "@ai-sdk/cerebras":
338
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/cerebras
339
+ case "@ai-sdk/togetherai":
340
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/togetherai
341
+ case "@ai-sdk/xai":
342
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/xai
343
+ case "@ai-sdk/deepinfra":
344
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/deepinfra
345
+ case "@ai-sdk/openai-compatible":
346
+ return Object.fromEntries(WIDELY_SUPPORTED_EFFORTS.map((effort) => [effort, { reasoningEffort: effort }]))
347
+
348
+ case "@ai-sdk/azure":
349
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/azure
350
+ if (id === "o1-mini") return {}
351
+ const azureEfforts = ["low", "medium", "high"]
352
+ if (id.includes("gpt-5-") || id === "gpt-5") {
353
+ azureEfforts.unshift("minimal")
354
+ }
355
+ return Object.fromEntries(
356
+ azureEfforts.map((effort) => [
357
+ effort,
358
+ {
359
+ reasoningEffort: effort,
360
+ reasoningSummary: "auto",
361
+ include: ["reasoning.encrypted_content"],
362
+ },
363
+ ]),
364
+ )
365
+ case "@ai-sdk/openai":
366
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/openai
367
+ if (id === "gpt-5-pro") return {}
368
+ const openaiEfforts = iife(() => {
369
+ if (id.includes("codex")) {
370
+ if (id.includes("5.2")) return [...WIDELY_SUPPORTED_EFFORTS, "xhigh"]
371
+ return WIDELY_SUPPORTED_EFFORTS
372
+ }
373
+ const arr = [...WIDELY_SUPPORTED_EFFORTS]
374
+ if (id.includes("gpt-5-") || id === "gpt-5") {
375
+ arr.unshift("minimal")
376
+ }
377
+ if (model.release_date >= "2025-11-13") {
378
+ arr.unshift("none")
379
+ }
380
+ if (model.release_date >= "2025-12-04") {
381
+ arr.push("xhigh")
382
+ }
383
+ return arr
384
+ })
385
+ return Object.fromEntries(
386
+ openaiEfforts.map((effort) => [
387
+ effort,
388
+ {
389
+ reasoningEffort: effort,
390
+ reasoningSummary: "auto",
391
+ include: ["reasoning.encrypted_content"],
392
+ },
393
+ ]),
394
+ )
395
+
396
+ case "@ai-sdk/anthropic":
397
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/anthropic
398
+ return {
399
+ high: {
400
+ thinking: {
401
+ type: "enabled",
402
+ budgetTokens: 16000,
403
+ },
404
+ },
405
+ max: {
406
+ thinking: {
407
+ type: "enabled",
408
+ budgetTokens: 31999,
409
+ },
410
+ },
411
+ }
412
+
413
+ case "@ai-sdk/amazon-bedrock":
414
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/amazon-bedrock
415
+ // For Anthropic models on Bedrock, use reasoningConfig with budgetTokens
416
+ if (model.api.id.includes("anthropic")) {
417
+ return {
418
+ high: {
419
+ reasoningConfig: {
420
+ type: "enabled",
421
+ budgetTokens: 16000,
422
+ },
423
+ },
424
+ max: {
425
+ reasoningConfig: {
426
+ type: "enabled",
427
+ budgetTokens: 31999,
428
+ },
429
+ },
430
+ }
431
+ }
432
+
433
+ // For Amazon Nova models, use reasoningConfig with maxReasoningEffort
434
+ return Object.fromEntries(
435
+ WIDELY_SUPPORTED_EFFORTS.map((effort) => [
436
+ effort,
437
+ {
438
+ reasoningConfig: {
439
+ type: "enabled",
440
+ maxReasoningEffort: effort,
441
+ },
442
+ },
443
+ ]),
444
+ )
445
+
446
+ case "@ai-sdk/google-vertex":
447
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/google-vertex
448
+ case "@ai-sdk/google":
449
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/google-generative-ai
450
+ if (id.includes("2.5")) {
451
+ return {
452
+ high: {
453
+ thinkingConfig: {
454
+ includeThoughts: true,
455
+ thinkingBudget: 16000,
456
+ },
457
+ },
458
+ max: {
459
+ thinkingConfig: {
460
+ includeThoughts: true,
461
+ thinkingBudget: 24576,
462
+ },
463
+ },
464
+ }
465
+ }
466
+ return Object.fromEntries(
467
+ ["low", "high"].map((effort) => [
468
+ effort,
469
+ {
470
+ includeThoughts: true,
471
+ thinkingLevel: effort,
472
+ },
473
+ ]),
474
+ )
475
+
476
+ case "@ai-sdk/mistral":
477
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/mistral
478
+ return {}
479
+
480
+ case "@ai-sdk/cohere":
481
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/cohere
482
+ return {}
483
+
484
+ case "@ai-sdk/groq":
485
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/groq
486
+ const groqEffort = ["none", ...WIDELY_SUPPORTED_EFFORTS]
487
+ return Object.fromEntries(
488
+ groqEffort.map((effort) => [
489
+ effort,
490
+ {
491
+ includeThoughts: true,
492
+ thinkingLevel: effort,
493
+ },
494
+ ]),
495
+ )
496
+
497
+ case "@ai-sdk/perplexity":
498
+ // https://v5.ai-sdk.dev/providers/ai-sdk-providers/perplexity
499
+ return {}
500
+ }
501
+ return {}
502
+ }
503
+
504
+ export function options(input: {
505
+ model: Provider.Model
506
+ sessionID: string
507
+ providerOptions?: Record<string, any>
508
+ }): Record<string, any> {
509
+ const result: Record<string, any> = {}
510
+
511
+ // openai and providers using openai package should set store to false by default.
512
+ if (input.model.providerID === "openai" || input.model.api.npm === "@ai-sdk/openai") {
513
+ result["store"] = false
514
+ }
515
+
516
+ if (input.model.api.npm === "@openrouter/ai-sdk-provider") {
517
+ result["usage"] = {
518
+ include: true,
519
+ }
520
+ if (input.model.api.id.includes("gemini-3")) {
521
+ result["reasoning"] = { effort: "high" }
522
+ }
523
+ }
524
+
525
+ if (
526
+ input.model.providerID === "baseten" ||
527
+ (input.model.providerID === "opencode" && ["kimi-k2-thinking", "glm-4.6"].includes(input.model.api.id))
528
+ ) {
529
+ result["chat_template_args"] = { enable_thinking: true }
530
+ }
531
+
532
+ if (["zai", "zhipuai"].includes(input.model.providerID) && input.model.api.npm === "@ai-sdk/openai-compatible") {
533
+ result["thinking"] = {
534
+ type: "enabled",
535
+ clear_thinking: false,
536
+ }
537
+ }
538
+
539
+ if (input.model.providerID === "openai" || input.providerOptions?.setCacheKey) {
540
+ result["promptCacheKey"] = input.sessionID
541
+ }
542
+
543
+ if (input.model.api.npm === "@ai-sdk/google" || input.model.api.npm === "@ai-sdk/google-vertex") {
544
+ result["thinkingConfig"] = {
545
+ includeThoughts: true,
546
+ }
547
+ if (input.model.api.id.includes("gemini-3")) {
548
+ result["thinkingConfig"]["thinkingLevel"] = "high"
549
+ }
550
+ }
551
+
552
+ if (input.model.api.id.includes("gpt-5") && !input.model.api.id.includes("gpt-5-chat")) {
553
+ if (input.model.providerID.includes("codex")) {
554
+ result["store"] = false
555
+ }
556
+
557
+ if (!input.model.api.id.includes("codex") && !input.model.api.id.includes("gpt-5-pro")) {
558
+ result["reasoningEffort"] = "medium"
559
+ }
560
+
561
+ if (input.model.api.id.endsWith("gpt-5.") && input.model.providerID !== "azure") {
562
+ result["textVerbosity"] = "low"
563
+ }
564
+
565
+ if (input.model.providerID.startsWith("opencode")) {
566
+ result["promptCacheKey"] = input.sessionID
567
+ result["include"] = ["reasoning.encrypted_content"]
568
+ result["reasoningSummary"] = "auto"
569
+ }
570
+ }
571
+ return result
572
+ }
573
+
574
+ export function smallOptions(model: Provider.Model) {
575
+ if (model.providerID === "openai" || model.api.id.includes("gpt-5")) {
576
+ if (model.api.id.includes("5.")) {
577
+ return { reasoningEffort: "low" }
578
+ }
579
+ return { reasoningEffort: "minimal" }
580
+ }
581
+ if (model.providerID === "google") {
582
+ // gemini-3 uses thinkingLevel, gemini-2.5 uses thinkingBudget
583
+ if (model.api.id.includes("gemini-3")) {
584
+ return { thinkingConfig: { thinkingLevel: "minimal" } }
585
+ }
586
+ return { thinkingConfig: { thinkingBudget: 0 } }
587
+ }
588
+ if (model.providerID === "openrouter") {
589
+ if (model.api.id.includes("google")) {
590
+ return { reasoning: { enabled: false } }
591
+ }
592
+ return { reasoningEffort: "minimal" }
593
+ }
594
+ return {}
595
+ }
596
+
597
+ export function providerOptions(model: Provider.Model, options: { [x: string]: any }) {
598
+ const key = sdkKey(model.api.npm) ?? model.providerID
599
+ return { [key]: options }
600
+ }
601
+
602
+ export function maxOutputTokens(
603
+ npm: string,
604
+ options: Record<string, any>,
605
+ modelLimit: number,
606
+ globalLimit: number,
607
+ ): number {
608
+ const modelCap = modelLimit || globalLimit
609
+ const standardLimit = Math.min(modelCap, globalLimit)
610
+
611
+ if (npm === "@ai-sdk/anthropic") {
612
+ const thinking = options?.["thinking"]
613
+ const budgetTokens = typeof thinking?.["budgetTokens"] === "number" ? thinking["budgetTokens"] : 0
614
+ const enabled = thinking?.["type"] === "enabled"
615
+ if (enabled && budgetTokens > 0) {
616
+ // Return text tokens so that text + thinking <= model cap, preferring 32k text when possible.
617
+ if (budgetTokens + standardLimit <= modelCap) {
618
+ return standardLimit
619
+ }
620
+ return modelCap - budgetTokens
621
+ }
622
+ }
623
+
624
+ return standardLimit
625
+ }
626
+
627
+ export function schema(model: Provider.Model, schema: JSONSchema.BaseSchema) {
628
+ /*
629
+ if (["openai", "azure"].includes(providerID)) {
630
+ if (schema.type === "object" && schema.properties) {
631
+ for (const [key, value] of Object.entries(schema.properties)) {
632
+ if (schema.required?.includes(key)) continue
633
+ schema.properties[key] = {
634
+ anyOf: [
635
+ value as JSONSchema.JSONSchema,
636
+ {
637
+ type: "null",
638
+ },
639
+ ],
640
+ }
641
+ }
642
+ }
643
+ }
644
+ */
645
+
646
+ // Convert integer enums to string enums for Google/Gemini
647
+ if (model.providerID === "google" || model.api.id.includes("gemini")) {
648
+ const sanitizeGemini = (obj: any): any => {
649
+ if (obj === null || typeof obj !== "object") {
650
+ return obj
651
+ }
652
+
653
+ if (Array.isArray(obj)) {
654
+ return obj.map(sanitizeGemini)
655
+ }
656
+
657
+ const result: any = {}
658
+ for (const [key, value] of Object.entries(obj)) {
659
+ if (key === "enum" && Array.isArray(value)) {
660
+ // Convert all enum values to strings
661
+ result[key] = value.map((v) => String(v))
662
+ // If we have integer type with enum, change type to string
663
+ if (result.type === "integer" || result.type === "number") {
664
+ result.type = "string"
665
+ }
666
+ } else if (typeof value === "object" && value !== null) {
667
+ result[key] = sanitizeGemini(value)
668
+ } else {
669
+ result[key] = value
670
+ }
671
+ }
672
+
673
+ // Filter required array to only include fields that exist in properties
674
+ if (result.type === "object" && result.properties && Array.isArray(result.required)) {
675
+ result.required = result.required.filter((field: any) => field in result.properties)
676
+ }
677
+
678
+ if (result.type === "array" && result.items == null) {
679
+ result.items = {}
680
+ }
681
+
682
+ return result
683
+ }
684
+
685
+ schema = sanitizeGemini(schema)
686
+ }
687
+
688
+ return schema
689
+ }
690
+
691
+ export function error(providerID: string, error: APICallError) {
692
+ let message = error.message
693
+ if (providerID.includes("github-copilot") && error.statusCode === 403) {
694
+ return "Please reauthenticate with the copilot provider to ensure your credentials work properly with OpenCode."
695
+ }
696
+ if (providerID.includes("github-copilot") && message.includes("The requested model is not supported")) {
697
+ return (
698
+ message +
699
+ "\n\nMake sure the model is enabled in your copilot settings: https://github.com/settings/copilot/features"
700
+ )
701
+ }
702
+
703
+ return message
704
+ }
705
+ }