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