@farhanic017/octocode 3.5.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 (716) hide show
  1. package/.swarm_lessons.json +85 -0
  2. package/AGENTS.md +151 -0
  3. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  4. package/Dockerfile +18 -0
  5. package/README.md +15 -0
  6. package/bin/octocode +31 -0
  7. package/bunfig.toml +7 -0
  8. package/git +0 -0
  9. package/package.json +151 -0
  10. package/parsers-config.ts +1 -0
  11. package/script/bench-search.ts +115 -0
  12. package/script/bench-test-suite.ts +52 -0
  13. package/script/build.ts +243 -0
  14. package/script/generate.ts +14 -0
  15. package/script/httpapi-exercise.ts +1 -0
  16. package/script/octo-shim.mjs +34 -0
  17. package/script/postinstall.mjs +198 -0
  18. package/script/profile-test-files.ts +42 -0
  19. package/script/publish.ts +211 -0
  20. package/script/run-workspace-server +106 -0
  21. package/script/schema.ts +77 -0
  22. package/script/time.ts +6 -0
  23. package/script/trace-imports.ts +154 -0
  24. package/specs/effect/error-boundaries-plan.md +235 -0
  25. package/specs/effect/errors.md +207 -0
  26. package/specs/effect/facades.md +218 -0
  27. package/specs/effect/guide.md +247 -0
  28. package/specs/effect/instance-context.md +13 -0
  29. package/specs/effect/loose-ends.md +30 -0
  30. package/specs/effect/migration.md +62 -0
  31. package/specs/effect/routes.md +61 -0
  32. package/specs/effect/schema.md +88 -0
  33. package/specs/effect/server-package.md +58 -0
  34. package/specs/effect/todo.md +241 -0
  35. package/specs/effect/tools.md +88 -0
  36. package/specs/openapi-translation-cleanup.md +204 -0
  37. package/specs/tui-plugins.md +544 -0
  38. package/specs/v2/api.ts +67 -0
  39. package/specs/v2/message-shape.md +136 -0
  40. package/specs/v2/notifications.md +13 -0
  41. package/specs/v2/tui-command-shim.md +67 -0
  42. package/src/account/account.ts +459 -0
  43. package/src/account/repo.ts +170 -0
  44. package/src/account/schema.ts +99 -0
  45. package/src/account/url.ts +8 -0
  46. package/src/acp/agent.ts +95 -0
  47. package/src/acp/config-option.ts +203 -0
  48. package/src/acp/content.ts +250 -0
  49. package/src/acp/directory.ts +210 -0
  50. package/src/acp/error.ts +90 -0
  51. package/src/acp/event.ts +345 -0
  52. package/src/acp/permission.ts +145 -0
  53. package/src/acp/profile.ts +42 -0
  54. package/src/acp/service.ts +1062 -0
  55. package/src/acp/tool.ts +321 -0
  56. package/src/acp/usage.ts +239 -0
  57. package/src/agent/agent.ts +451 -0
  58. package/src/agent/generate.txt +75 -0
  59. package/src/agent/prompt/compaction.txt +30 -0
  60. package/src/agent/prompt/explore.txt +18 -0
  61. package/src/agent/prompt/summary.txt +11 -0
  62. package/src/agent/prompt/title.txt +50 -0
  63. package/src/agent/subagent-permissions.ts +35 -0
  64. package/src/audio.d.ts +14 -0
  65. package/src/auth/index.ts +96 -0
  66. package/src/background/job.ts +36 -0
  67. package/src/bus/global.ts +22 -0
  68. package/src/cli/bootstrap.ts +11 -0
  69. package/src/cli/cmd/account.ts +264 -0
  70. package/src/cli/cmd/acp.ts +76 -0
  71. package/src/cli/cmd/agent.ts +259 -0
  72. package/src/cli/cmd/attach.ts +97 -0
  73. package/src/cli/cmd/cmd.ts +7 -0
  74. package/src/cli/cmd/db.ts +62 -0
  75. package/src/cli/cmd/debug/agent.handler.ts +188 -0
  76. package/src/cli/cmd/debug/agent.ts +27 -0
  77. package/src/cli/cmd/debug/config.ts +14 -0
  78. package/src/cli/cmd/debug/file.ts +87 -0
  79. package/src/cli/cmd/debug/index.ts +87 -0
  80. package/src/cli/cmd/debug/lsp.ts +51 -0
  81. package/src/cli/cmd/debug/ripgrep.ts +99 -0
  82. package/src/cli/cmd/debug/scrap.ts +18 -0
  83. package/src/cli/cmd/debug/skill.ts +15 -0
  84. package/src/cli/cmd/debug/snapshot.ts +50 -0
  85. package/src/cli/cmd/debug/startup.ts +11 -0
  86. package/src/cli/cmd/debug/v2.ts +46 -0
  87. package/src/cli/cmd/export.ts +292 -0
  88. package/src/cli/cmd/generate.ts +54 -0
  89. package/src/cli/cmd/github.handler.ts +1594 -0
  90. package/src/cli/cmd/github.shared.ts +30 -0
  91. package/src/cli/cmd/github.ts +42 -0
  92. package/src/cli/cmd/import.ts +224 -0
  93. package/src/cli/cmd/mcp.ts +859 -0
  94. package/src/cli/cmd/models.ts +66 -0
  95. package/src/cli/cmd/plug.ts +230 -0
  96. package/src/cli/cmd/pr.ts +115 -0
  97. package/src/cli/cmd/prompt-display.ts +1 -0
  98. package/src/cli/cmd/providers.ts +550 -0
  99. package/src/cli/cmd/run/demo.ts +1274 -0
  100. package/src/cli/cmd/run/entry.body.ts +205 -0
  101. package/src/cli/cmd/run/footer.command.tsx +1088 -0
  102. package/src/cli/cmd/run/footer.menu.tsx +353 -0
  103. package/src/cli/cmd/run/footer.permission.tsx +482 -0
  104. package/src/cli/cmd/run/footer.prompt.tsx +1351 -0
  105. package/src/cli/cmd/run/footer.question.tsx +580 -0
  106. package/src/cli/cmd/run/footer.subagent.tsx +180 -0
  107. package/src/cli/cmd/run/footer.ts +1153 -0
  108. package/src/cli/cmd/run/footer.view.tsx +992 -0
  109. package/src/cli/cmd/run/footer.width.ts +27 -0
  110. package/src/cli/cmd/run/otel.ts +117 -0
  111. package/src/cli/cmd/run/permission.shared.ts +256 -0
  112. package/src/cli/cmd/run/prompt.editor.ts +157 -0
  113. package/src/cli/cmd/run/prompt.shared.ts +153 -0
  114. package/src/cli/cmd/run/question.shared.ts +340 -0
  115. package/src/cli/cmd/run/runtime.boot.ts +202 -0
  116. package/src/cli/cmd/run/runtime.lifecycle.ts +431 -0
  117. package/src/cli/cmd/run/runtime.queue.ts +349 -0
  118. package/src/cli/cmd/run/runtime.shared.ts +17 -0
  119. package/src/cli/cmd/run/runtime.stdin.ts +37 -0
  120. package/src/cli/cmd/run/runtime.ts +905 -0
  121. package/src/cli/cmd/run/scrollback.shared.ts +92 -0
  122. package/src/cli/cmd/run/scrollback.surface.ts +448 -0
  123. package/src/cli/cmd/run/scrollback.writer.tsx +353 -0
  124. package/src/cli/cmd/run/splash.ts +284 -0
  125. package/src/cli/cmd/run/stream.transport.ts +1465 -0
  126. package/src/cli/cmd/run/stream.ts +175 -0
  127. package/src/cli/cmd/run/subagent-data.ts +876 -0
  128. package/src/cli/cmd/run/theme.ts +690 -0
  129. package/src/cli/cmd/run/tool.ts +1489 -0
  130. package/src/cli/cmd/run/trace.ts +94 -0
  131. package/src/cli/cmd/run/turn-summary.ts +47 -0
  132. package/src/cli/cmd/run/types.ts +350 -0
  133. package/src/cli/cmd/run/variant.shared.ts +215 -0
  134. package/src/cli/cmd/run.ts +897 -0
  135. package/src/cli/cmd/serve.ts +24 -0
  136. package/src/cli/cmd/stats.ts +393 -0
  137. package/src/cli/cmd/tui.ts +264 -0
  138. package/src/cli/cmd/uninstall.ts +353 -0
  139. package/src/cli/cmd/upgrade.ts +74 -0
  140. package/src/cli/cmd/web.ts +84 -0
  141. package/src/cli/effect/prompt.ts +37 -0
  142. package/src/cli/effect-cmd.ts +96 -0
  143. package/src/cli/error.ts +118 -0
  144. package/src/cli/heap.ts +66 -0
  145. package/src/cli/logo.ts +1 -0
  146. package/src/cli/network.ts +64 -0
  147. package/src/cli/tui/layer.ts +7 -0
  148. package/src/cli/tui/photon_rs_bg-bq08arze.wasm +0 -0
  149. package/src/cli/tui/tree-sitter-3jzf13jk.wasm +0 -0
  150. package/src/cli/tui/tree-sitter-bash-hq5s6fxb.wasm +0 -0
  151. package/src/cli/tui/tree-sitter-powershell-ryb2ffqs.wasm +0 -0
  152. package/src/cli/tui/worker.ts +99 -0
  153. package/src/cli/ui.ts +144 -0
  154. package/src/cli/upgrade.ts +53 -0
  155. package/src/command/index.ts +189 -0
  156. package/src/command/template/initialize.txt +66 -0
  157. package/src/command/template/review.txt +101 -0
  158. package/src/command/terminal-ai-commands.ts +567 -0
  159. package/src/config/agent.ts +68 -0
  160. package/src/config/command.ts +45 -0
  161. package/src/config/config.ts +686 -0
  162. package/src/config/entry-name.ts +19 -0
  163. package/src/config/managed.ts +77 -0
  164. package/src/config/markdown.ts +36 -0
  165. package/src/config/parse.ts +79 -0
  166. package/src/config/paths.ts +47 -0
  167. package/src/config/plugin.ts +79 -0
  168. package/src/config/reference.ts +48 -0
  169. package/src/config/tui-cwd.ts +5 -0
  170. package/src/config/tui-host-attention.ts +21 -0
  171. package/src/config/tui-migrate.ts +156 -0
  172. package/src/config/tui.ts +294 -0
  173. package/src/config/variable.ts +91 -0
  174. package/src/control-plane/adapters/index.ts +41 -0
  175. package/src/control-plane/adapters/worktree.ts +96 -0
  176. package/src/control-plane/dev/README.md +19 -0
  177. package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
  178. package/src/control-plane/types.ts +59 -0
  179. package/src/control-plane/util.ts +39 -0
  180. package/src/control-plane/workspace-adapter-runtime.ts +51 -0
  181. package/src/control-plane/workspace-context.ts +26 -0
  182. package/src/control-plane/workspace.ts +1075 -0
  183. package/src/desktop/agent.ts +147 -0
  184. package/src/desktop/index.ts +247 -0
  185. package/src/desktop/platform.ts +97 -0
  186. package/src/desktop/stream.ts +64 -0
  187. package/src/desktop/vision.ts +52 -0
  188. package/src/desktop/websocket.ts +120 -0
  189. package/src/effect/app-runtime.ts +133 -0
  190. package/src/effect/bootstrap-runtime.ts +23 -0
  191. package/src/effect/bridge.ts +84 -0
  192. package/src/effect/config-service.ts +67 -0
  193. package/src/effect/instance-ref.ts +11 -0
  194. package/src/effect/instance-registry.ts +12 -0
  195. package/src/effect/instance-state.ts +72 -0
  196. package/src/effect/promise.ts +17 -0
  197. package/src/effect/run-service.ts +47 -0
  198. package/src/effect/runner.ts +217 -0
  199. package/src/effect/runtime-flags.ts +76 -0
  200. package/src/env/index.ts +40 -0
  201. package/src/event-v2-bridge.ts +76 -0
  202. package/src/format/formatter.ts +404 -0
  203. package/src/format/index.ts +212 -0
  204. package/src/git/index.ts +347 -0
  205. package/src/id/id.ts +80 -0
  206. package/src/ide/index.ts +70 -0
  207. package/src/image/image.ts +177 -0
  208. package/src/index.ts +208 -0
  209. package/src/installation/index.ts +350 -0
  210. package/src/lsp/client.ts +686 -0
  211. package/src/lsp/diagnostic.ts +29 -0
  212. package/src/lsp/language.ts +121 -0
  213. package/src/lsp/launch.ts +21 -0
  214. package/src/lsp/lsp.ts +517 -0
  215. package/src/lsp/server.ts +2116 -0
  216. package/src/markdown.d.ts +4 -0
  217. package/src/mcp/auth.ts +171 -0
  218. package/src/mcp/builtin.ts +92 -0
  219. package/src/mcp/index.ts +1000 -0
  220. package/src/mcp/oauth-callback.ts +232 -0
  221. package/src/mcp/oauth-provider.ts +217 -0
  222. package/src/node.ts +5 -0
  223. package/src/patch/index.ts +689 -0
  224. package/src/permission/arity.ts +163 -0
  225. package/src/permission/evaluate.ts +1 -0
  226. package/src/permission/index.ts +230 -0
  227. package/src/plugin/azure.ts +26 -0
  228. package/src/plugin/cloudflare.ts +76 -0
  229. package/src/plugin/digitalocean.ts +391 -0
  230. package/src/plugin/github-copilot/copilot.ts +417 -0
  231. package/src/plugin/github-copilot/models.ts +246 -0
  232. package/src/plugin/index.ts +320 -0
  233. package/src/plugin/install.ts +439 -0
  234. package/src/plugin/loader.ts +237 -0
  235. package/src/plugin/meta.ts +188 -0
  236. package/src/plugin/openai/README.md +31 -0
  237. package/src/plugin/openai/codex.ts +647 -0
  238. package/src/plugin/openai/ws-pool.ts +290 -0
  239. package/src/plugin/openai/ws.ts +381 -0
  240. package/src/plugin/shared.ts +323 -0
  241. package/src/plugin/tui/internal.ts +12 -0
  242. package/src/plugin/tui/runtime.ts +1174 -0
  243. package/src/plugin/xai.ts +742 -0
  244. package/src/project/bootstrap-service.ts +9 -0
  245. package/src/project/bootstrap.ts +80 -0
  246. package/src/project/instance-context.ts +24 -0
  247. package/src/project/instance-layer.ts +11 -0
  248. package/src/project/instance-runtime.ts +16 -0
  249. package/src/project/instance-store.ts +207 -0
  250. package/src/project/project.ts +520 -0
  251. package/src/project/vcs.ts +435 -0
  252. package/src/provider/auth.ts +230 -0
  253. package/src/provider/error.ts +188 -0
  254. package/src/provider/model-status.ts +8 -0
  255. package/src/provider/provider.ts +2037 -0
  256. package/src/provider/transform.ts +1367 -0
  257. package/src/pty-preparation.ts +30 -0
  258. package/src/question/index.ts +229 -0
  259. package/src/question/schema.ts +10 -0
  260. package/src/reference/reference.ts +239 -0
  261. package/src/reference/repository-cache.ts +320 -0
  262. package/src/server/auth.ts +48 -0
  263. package/src/server/cors.ts +34 -0
  264. package/src/server/event.ts +13 -0
  265. package/src/server/global-lifecycle.ts +37 -0
  266. package/src/server/init-projectors.ts +3 -0
  267. package/src/server/mdns.ts +60 -0
  268. package/src/server/projectors.ts +1 -0
  269. package/src/server/proxy-util.ts +48 -0
  270. package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
  271. package/src/server/routes/instance/httpapi/api.ts +80 -0
  272. package/src/server/routes/instance/httpapi/errors.ts +193 -0
  273. package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
  274. package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
  275. package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
  276. package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
  277. package/src/server/routes/instance/httpapi/groups/experimental.ts +260 -0
  278. package/src/server/routes/instance/httpapi/groups/file.ts +172 -0
  279. package/src/server/routes/instance/httpapi/groups/global.ts +138 -0
  280. package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
  281. package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
  282. package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
  283. package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
  284. package/src/server/routes/instance/httpapi/groups/project-copy.ts +88 -0
  285. package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
  286. package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
  287. package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
  288. package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
  289. package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
  290. package/src/server/routes/instance/httpapi/groups/reference.ts +60 -0
  291. package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
  292. package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
  293. package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
  294. package/src/server/routes/instance/httpapi/handlers/config.ts +34 -0
  295. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
  296. package/src/server/routes/instance/httpapi/handlers/control.ts +37 -0
  297. package/src/server/routes/instance/httpapi/handlers/event.ts +102 -0
  298. package/src/server/routes/instance/httpapi/handlers/experimental.ts +187 -0
  299. package/src/server/routes/instance/httpapi/handlers/file.ts +128 -0
  300. package/src/server/routes/instance/httpapi/handlers/global.ts +157 -0
  301. package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
  302. package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
  303. package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
  304. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +157 -0
  305. package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
  306. package/src/server/routes/instance/httpapi/handlers/provider.ts +113 -0
  307. package/src/server/routes/instance/httpapi/handlers/pty.ts +258 -0
  308. package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
  309. package/src/server/routes/instance/httpapi/handlers/reference.ts +27 -0
  310. package/src/server/routes/instance/httpapi/handlers/sync.ts +95 -0
  311. package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
  312. package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
  313. package/src/server/routes/instance/httpapi/lifecycle.ts +57 -0
  314. package/src/server/routes/instance/httpapi/middleware/authorization.ts +150 -0
  315. package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
  316. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
  317. package/src/server/routes/instance/httpapi/middleware/error.ts +36 -0
  318. package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
  319. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
  320. package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
  321. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +42 -0
  322. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
  323. package/src/server/routes/instance/httpapi/public.ts +535 -0
  324. package/src/server/routes/instance/httpapi/server.ts +281 -0
  325. package/src/server/routes/instance/httpapi/websocket-tracker.ts +57 -0
  326. package/src/server/server.ts +218 -0
  327. package/src/server/shared/fence.ts +68 -0
  328. package/src/server/shared/pty-ticket.ts +15 -0
  329. package/src/server/shared/public-ui.ts +12 -0
  330. package/src/server/shared/tui-control.ts +28 -0
  331. package/src/server/shared/ui.ts +108 -0
  332. package/src/server/shared/workspace-routing.ts +38 -0
  333. package/src/server/tui-event.ts +53 -0
  334. package/src/share/share-next.ts +379 -0
  335. package/src/shell/shell.ts +215 -0
  336. package/src/skill/discovery.ts +115 -0
  337. package/src/skill/index.ts +382 -0
  338. package/src/snapshot/index.ts +762 -0
  339. package/src/sql.d.ts +4 -0
  340. package/src/storage/schema.ts +5 -0
  341. package/src/storage/storage.ts +329 -0
  342. package/src/swarm/evolution.ts +304 -0
  343. package/src/swarm/index.ts +7 -0
  344. package/src/swarm/learning.ts +736 -0
  345. package/src/swarm/vision.ts +131 -0
  346. package/src/sync/README.md +179 -0
  347. package/src/sync/schema.ts +11 -0
  348. package/src/temporary.ts +33 -0
  349. package/src/tool/apply_patch.ts +313 -0
  350. package/src/tool/apply_patch.txt +33 -0
  351. package/src/tool/browser_click.ts +51 -0
  352. package/src/tool/browser_drag.ts +76 -0
  353. package/src/tool/browser_evaluate.ts +51 -0
  354. package/src/tool/browser_hover.ts +51 -0
  355. package/src/tool/browser_navigate.ts +45 -0
  356. package/src/tool/browser_screenshot.ts +66 -0
  357. package/src/tool/browser_select.ts +52 -0
  358. package/src/tool/browser_type.ts +52 -0
  359. package/src/tool/browser_wait.ts +60 -0
  360. package/src/tool/desktop_clipboard.ts +82 -0
  361. package/src/tool/desktop_control.ts +160 -0
  362. package/src/tool/desktop_record.ts +93 -0
  363. package/src/tool/desktop_replay.ts +99 -0
  364. package/src/tool/desktop_screen_record.ts +143 -0
  365. package/src/tool/desktop_window.ts +142 -0
  366. package/src/tool/desktop_workflow.ts +81 -0
  367. package/src/tool/edit.ts +737 -0
  368. package/src/tool/edit.txt +10 -0
  369. package/src/tool/external-directory.ts +49 -0
  370. package/src/tool/glob.ts +84 -0
  371. package/src/tool/glob.txt +6 -0
  372. package/src/tool/grep.ts +140 -0
  373. package/src/tool/grep.txt +8 -0
  374. package/src/tool/invalid.ts +21 -0
  375. package/src/tool/json-schema.ts +164 -0
  376. package/src/tool/lsp.ts +113 -0
  377. package/src/tool/lsp.txt +24 -0
  378. package/src/tool/mcp-websearch.ts +96 -0
  379. package/src/tool/open_app.ts +55 -0
  380. package/src/tool/open_terminal.ts +69 -0
  381. package/src/tool/plan-enter.txt +14 -0
  382. package/src/tool/plan-exit.txt +13 -0
  383. package/src/tool/plan.ts +79 -0
  384. package/src/tool/question.ts +44 -0
  385. package/src/tool/question.txt +10 -0
  386. package/src/tool/read.ts +392 -0
  387. package/src/tool/read.txt +14 -0
  388. package/src/tool/registry.ts +527 -0
  389. package/src/tool/schema.ts +14 -0
  390. package/src/tool/screenshot.ts +72 -0
  391. package/src/tool/shell/id.ts +19 -0
  392. package/src/tool/shell/prompt.ts +307 -0
  393. package/src/tool/shell/shell.txt +21 -0
  394. package/src/tool/shell.ts +669 -0
  395. package/src/tool/skill.ts +72 -0
  396. package/src/tool/skill.txt +5 -0
  397. package/src/tool/task.ts +339 -0
  398. package/src/tool/task.txt +19 -0
  399. package/src/tool/todo.ts +57 -0
  400. package/src/tool/todowrite.txt +44 -0
  401. package/src/tool/tool.ts +180 -0
  402. package/src/tool/truncate.ts +160 -0
  403. package/src/tool/truncation-dir.ts +4 -0
  404. package/src/tool/visual_qa.ts +77 -0
  405. package/src/tool/webfetch.ts +192 -0
  406. package/src/tool/webfetch.txt +13 -0
  407. package/src/tool/websearch.ts +143 -0
  408. package/src/tool/websearch.txt +14 -0
  409. package/src/tool/write.ts +104 -0
  410. package/src/tool/write.txt +8 -0
  411. package/src/util/archive.ts +17 -0
  412. package/src/util/bom.ts +27 -0
  413. package/src/util/data-url.ts +9 -0
  414. package/src/util/defer.ts +10 -0
  415. package/src/util/effect-http-client.ts +11 -0
  416. package/src/util/error.ts +1 -0
  417. package/src/util/filesystem.ts +251 -0
  418. package/src/util/iife.ts +3 -0
  419. package/src/util/lazy.ts +20 -0
  420. package/src/util/local-context.ts +25 -0
  421. package/src/util/locale.ts +2 -0
  422. package/src/util/media.ts +26 -0
  423. package/src/util/process.ts +177 -0
  424. package/src/util/proxy-env.ts +72 -0
  425. package/src/util/queue.ts +32 -0
  426. package/src/util/record.ts +1 -0
  427. package/src/util/repository.ts +232 -0
  428. package/src/util/rpc.ts +66 -0
  429. package/src/util/signal.ts +12 -0
  430. package/src/util/sys-monitor.ts +223 -0
  431. package/src/util/timeout.ts +13 -0
  432. package/src/util/wildcard.ts +59 -0
  433. package/src/worktree/index.ts +645 -0
  434. package/sst-env.d.ts +10 -0
  435. package/test/AGENTS.md +204 -0
  436. package/test/EFFECT_TEST_MIGRATION.md +169 -0
  437. package/test/account/repo.test.ts +353 -0
  438. package/test/account/service.test.ts +453 -0
  439. package/test/acp/config-option.test.ts +229 -0
  440. package/test/acp/content.test.ts +201 -0
  441. package/test/acp/directory.test.ts +186 -0
  442. package/test/acp/error.test.ts +67 -0
  443. package/test/acp/event.test.ts +743 -0
  444. package/test/acp/permission.test.ts +273 -0
  445. package/test/acp/tool.test.ts +210 -0
  446. package/test/acp/usage.test.ts +315 -0
  447. package/test/agent/agent.test.ts +711 -0
  448. package/test/agent/plan-mode-subagent-bypass.test.ts +213 -0
  449. package/test/agent/plugin-agent-regression.test.ts +62 -0
  450. package/test/auth/auth.test.ts +77 -0
  451. package/test/background/job.test.ts +243 -0
  452. package/test/cli/account.test.ts +30 -0
  453. package/test/cli/acp/acp-test-client.ts +97 -0
  454. package/test/cli/acp/config-options.test.ts +103 -0
  455. package/test/cli/acp/helpers.ts +96 -0
  456. package/test/cli/acp/initialize-auth.test.ts +61 -0
  457. package/test/cli/acp/lifecycle.test.ts +118 -0
  458. package/test/cli/acp/prompt-content.test.ts +97 -0
  459. package/test/cli/acp/skills.test.ts +38 -0
  460. package/test/cli/cmd/tui/attention.test.ts +484 -0
  461. package/test/cli/effect-cmd-instance-als.test.ts +39 -0
  462. package/test/cli/error.test.ts +95 -0
  463. package/test/cli/github-action.test.ts +199 -0
  464. package/test/cli/github-remote.test.ts +90 -0
  465. package/test/cli/help/__snapshots__/help-snapshots.test.ts.snap +631 -0
  466. package/test/cli/help/help-snapshots.test.ts +137 -0
  467. package/test/cli/import.test.ts +54 -0
  468. package/test/cli/mcp-add.test.ts +74 -0
  469. package/test/cli/plugin-auth-picker.test.ts +120 -0
  470. package/test/cli/run/entry.body.test.ts +536 -0
  471. package/test/cli/run/footer.menu.test.ts +43 -0
  472. package/test/cli/run/footer.view.test.tsx +1336 -0
  473. package/test/cli/run/footer.width.test.ts +35 -0
  474. package/test/cli/run/permission.shared.test.ts +144 -0
  475. package/test/cli/run/prompt.editor.test.ts +101 -0
  476. package/test/cli/run/prompt.shared.test.ts +101 -0
  477. package/test/cli/run/question.shared.test.ts +115 -0
  478. package/test/cli/run/run-process.test.ts +84 -0
  479. package/test/cli/run/runtime.boot.test.ts +283 -0
  480. package/test/cli/run/runtime.queue.test.ts +481 -0
  481. package/test/cli/run/runtime.stdin.test.ts +71 -0
  482. package/test/cli/run/runtime.test.ts +238 -0
  483. package/test/cli/run/scrollback.surface.test.ts +1065 -0
  484. package/test/cli/run/stream.test.ts +56 -0
  485. package/test/cli/run/stream.transport.test.ts +2363 -0
  486. package/test/cli/run/subagent-data.test.ts +547 -0
  487. package/test/cli/run/theme.test.ts +177 -0
  488. package/test/cli/run/variant.shared.test.ts +217 -0
  489. package/test/cli/serve/serve-process.test.ts +61 -0
  490. package/test/cli/smokes/read-only.test.ts +115 -0
  491. package/test/cli/tui/attach.test.ts +11 -0
  492. package/test/cli/tui/editor-context-zed.test.ts +379 -0
  493. package/test/cli/tui/editor-context.test.tsx +297 -0
  494. package/test/cli/tui/plugin-add.test.ts +110 -0
  495. package/test/cli/tui/plugin-install.test.ts +87 -0
  496. package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
  497. package/test/cli/tui/plugin-loader-entrypoint.test.ts +485 -0
  498. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  499. package/test/cli/tui/plugin-loader.test.ts +1332 -0
  500. package/test/cli/tui/plugin-toggle.test.ts +264 -0
  501. package/test/cli/tui/thread.test.ts +36 -0
  502. package/test/cli.test.ts +7 -0
  503. package/test/command/acp-slash-detect.test.ts +202 -0
  504. package/test/command/demo-slash.test.ts +213 -0
  505. package/test/command/slash-head.test.ts +232 -0
  506. package/test/command/slash-parsing.test.ts +193 -0
  507. package/test/command/terminal-ai-commands.test.ts +536 -0
  508. package/test/config/agent-color.test.ts +47 -0
  509. package/test/config/config.test.ts +1994 -0
  510. package/test/config/entry-name.test.ts +57 -0
  511. package/test/config/fixtures/empty-frontmatter.md +4 -0
  512. package/test/config/fixtures/frontmatter.md +28 -0
  513. package/test/config/fixtures/markdown-header.md +11 -0
  514. package/test/config/fixtures/no-frontmatter.md +1 -0
  515. package/test/config/fixtures/weird-model-id.md +13 -0
  516. package/test/config/lsp.test.ts +69 -0
  517. package/test/config/markdown.test.ts +228 -0
  518. package/test/config/plugin.test.ts +0 -0
  519. package/test/config/tui.test.ts +886 -0
  520. package/test/control-plane/adapters.test.ts +71 -0
  521. package/test/control-plane/workspace.test.ts +1704 -0
  522. package/test/effect/app-runtime-logger.test.ts +105 -0
  523. package/test/effect/config-service.test.ts +65 -0
  524. package/test/effect/instance-state.test.ts +391 -0
  525. package/test/effect/run-service.test.ts +89 -0
  526. package/test/effect/runner.test.ts +514 -0
  527. package/test/effect/runtime-flags.test.ts +373 -0
  528. package/test/fake/account.ts +9 -0
  529. package/test/fake/auth.ts +8 -0
  530. package/test/fake/npm.ts +8 -0
  531. package/test/fake/provider.ts +82 -0
  532. package/test/fake/skill.ts +8 -0
  533. package/test/filesystem/filesystem.test.ts +319 -0
  534. package/test/fixture/agent-plugin.constants.ts +6 -0
  535. package/test/fixture/agent-plugin.ts +12 -0
  536. package/test/fixture/config.ts +23 -0
  537. package/test/fixture/db.ts +11 -0
  538. package/test/fixture/fixture.test.ts +26 -0
  539. package/test/fixture/fixture.ts +224 -0
  540. package/test/fixture/flag.ts +20 -0
  541. package/test/fixture/flock-worker.ts +72 -0
  542. package/test/fixture/lsp/fake-lsp-server.js +249 -0
  543. package/test/fixture/plug-worker.ts +96 -0
  544. package/test/fixture/plugin-meta-worker.ts +19 -0
  545. package/test/fixture/plugin.ts +10 -0
  546. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  547. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  548. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  549. package/test/fixture/skills/index.json +6 -0
  550. package/test/fixture/tui-environment.tsx +32 -0
  551. package/test/fixture/tui-plugin.ts +357 -0
  552. package/test/fixture/tui-runtime.ts +56 -0
  553. package/test/fixture/tui-sdk.ts +82 -0
  554. package/test/fixture/workspace.ts +30 -0
  555. package/test/format/format.test.ts +228 -0
  556. package/test/git/git.test.ts +178 -0
  557. package/test/ide/ide.test.ts +82 -0
  558. package/test/image/fixtures/picture-5mb-base64.png +0 -0
  559. package/test/image/image.test.ts +123 -0
  560. package/test/installation/installation.test.ts +231 -0
  561. package/test/lib/cli-process.ts +459 -0
  562. package/test/lib/effect.ts +177 -0
  563. package/test/lib/filesystem.ts +10 -0
  564. package/test/lib/llm-server.ts +771 -0
  565. package/test/lib/snapshot.ts +73 -0
  566. package/test/lib/test-provider.ts +37 -0
  567. package/test/lib/websocket.ts +46 -0
  568. package/test/lsp/client.test.ts +493 -0
  569. package/test/lsp/index.test.ts +232 -0
  570. package/test/lsp/jdtls-root.test.ts +459 -0
  571. package/test/lsp/launch.test.ts +22 -0
  572. package/test/lsp/lifecycle.test.ts +160 -0
  573. package/test/mcp/auth.test.ts +78 -0
  574. package/test/mcp/builtin.test.ts +95 -0
  575. package/test/mcp/headers.test.ts +126 -0
  576. package/test/mcp/lifecycle.test.ts +999 -0
  577. package/test/mcp/oauth-auto-connect.test.ts +274 -0
  578. package/test/mcp/oauth-browser.test.ts +232 -0
  579. package/test/mcp/oauth-callback.test.ts +34 -0
  580. package/test/mcp/oauth-provider.test.ts +61 -0
  581. package/test/patch/patch.test.ts +383 -0
  582. package/test/permission/arity.test.ts +33 -0
  583. package/test/permission/next.test.ts +1176 -0
  584. package/test/permission-task.test.ts +318 -0
  585. package/test/plugin/auth-override.test.ts +105 -0
  586. package/test/plugin/cloudflare.test.ts +68 -0
  587. package/test/plugin/codex.test.ts +247 -0
  588. package/test/plugin/github-copilot-models.test.ts +332 -0
  589. package/test/plugin/install-concurrency.test.ts +140 -0
  590. package/test/plugin/install.test.ts +573 -0
  591. package/test/plugin/loader-shared.test.ts +1303 -0
  592. package/test/plugin/meta.test.ts +137 -0
  593. package/test/plugin/openai-rollout.test.ts +17 -0
  594. package/test/plugin/openai-ws.test.ts +877 -0
  595. package/test/plugin/shared.test.ts +88 -0
  596. package/test/plugin/trigger.test.ts +120 -0
  597. package/test/plugin/workspace-adapter.test.ts +137 -0
  598. package/test/plugin/xai.test.ts +634 -0
  599. package/test/preload.ts +99 -0
  600. package/test/project/instance-bootstrap.test.ts +110 -0
  601. package/test/project/instance.test.ts +245 -0
  602. package/test/project/migrate-global.test.ts +170 -0
  603. package/test/project/project-directory.test.ts +169 -0
  604. package/test/project/project.test.ts +818 -0
  605. package/test/project/vcs.test.ts +336 -0
  606. package/test/project/worktree-remove.test.ts +126 -0
  607. package/test/project/worktree.test.ts +320 -0
  608. package/test/provider/amazon-bedrock.test.ts +360 -0
  609. package/test/provider/cf-ai-gateway-e2e.test.ts +132 -0
  610. package/test/provider/digitalocean.test.ts +123 -0
  611. package/test/provider/gitlab-duo.test.ts +412 -0
  612. package/test/provider/header-timeout.test.ts +233 -0
  613. package/test/provider/model-status.test.ts +61 -0
  614. package/test/provider/provider.test.ts +1795 -0
  615. package/test/provider/transform.test.ts +3937 -0
  616. package/test/pty/pty-shell.test.ts +102 -0
  617. package/test/question/question.test.ts +465 -0
  618. package/test/reference/reference.test.ts +311 -0
  619. package/test/server/AGENTS.md +15 -0
  620. package/test/server/auth.test.ts +59 -0
  621. package/test/server/global-bus.ts +31 -0
  622. package/test/server/httpapi-authorization.test.ts +174 -0
  623. package/test/server/httpapi-compression.test.ts +154 -0
  624. package/test/server/httpapi-config.test.ts +113 -0
  625. package/test/server/httpapi-control-plane.test.ts +63 -0
  626. package/test/server/httpapi-cors-vary.test.ts +66 -0
  627. package/test/server/httpapi-cors.test.ts +122 -0
  628. package/test/server/httpapi-error-middleware.test.ts +96 -0
  629. package/test/server/httpapi-event.test.ts +97 -0
  630. package/test/server/httpapi-exercise/assertions.ts +64 -0
  631. package/test/server/httpapi-exercise/backend.ts +144 -0
  632. package/test/server/httpapi-exercise/dsl.ts +210 -0
  633. package/test/server/httpapi-exercise/environment.ts +40 -0
  634. package/test/server/httpapi-exercise/index.ts +1538 -0
  635. package/test/server/httpapi-exercise/report.ts +66 -0
  636. package/test/server/httpapi-exercise/routing.ts +96 -0
  637. package/test/server/httpapi-exercise/runner.ts +267 -0
  638. package/test/server/httpapi-exercise/runtime.ts +52 -0
  639. package/test/server/httpapi-exercise/types.ts +123 -0
  640. package/test/server/httpapi-experimental.test.ts +300 -0
  641. package/test/server/httpapi-file.test.ts +76 -0
  642. package/test/server/httpapi-global.test.ts +66 -0
  643. package/test/server/httpapi-instance-context.test.ts +347 -0
  644. package/test/server/httpapi-instance-route-auth.test.ts +84 -0
  645. package/test/server/httpapi-instance.test.ts +265 -0
  646. package/test/server/httpapi-layer.ts +33 -0
  647. package/test/server/httpapi-listen.test.ts +415 -0
  648. package/test/server/httpapi-mcp-oauth.test.ts +73 -0
  649. package/test/server/httpapi-mcp.test.ts +234 -0
  650. package/test/server/httpapi-mdns.test.ts +82 -0
  651. package/test/server/httpapi-promptasync-context.test.ts +222 -0
  652. package/test/server/httpapi-provider.test.ts +403 -0
  653. package/test/server/httpapi-pty.test.ts +275 -0
  654. package/test/server/httpapi-public-openapi.test.ts +297 -0
  655. package/test/server/httpapi-query-schema-drift.test.ts +330 -0
  656. package/test/server/httpapi-reference.test.ts +56 -0
  657. package/test/server/httpapi-schema-error-body.test.ts +165 -0
  658. package/test/server/httpapi-sdk.test.ts +909 -0
  659. package/test/server/httpapi-sync.test.ts +154 -0
  660. package/test/server/httpapi-ui.test.ts +456 -0
  661. package/test/server/httpapi-v2-location.test.ts +85 -0
  662. package/test/server/httpapi-workspace-routing.test.ts +554 -0
  663. package/test/server/httpapi-workspace.test.ts +515 -0
  664. package/test/server/project-copy.test.ts +101 -0
  665. package/test/server/project-init-git.test.ts +117 -0
  666. package/test/server/proxy-util.test.ts +113 -0
  667. package/test/server/sdk-error-shape.test.ts +84 -0
  668. package/test/server/sdk-v1-smoke.test.ts +60 -0
  669. package/test/server/workspace-proxy.test.ts +181 -0
  670. package/test/server/workspace-routing.test.ts +94 -0
  671. package/test/server/worktree-endpoint-repro.test.ts +307 -0
  672. package/test/share/share-next.test.ts +344 -0
  673. package/test/shell/shell.test.ts +99 -0
  674. package/test/skill/discovery.test.ts +139 -0
  675. package/test/skill/skill.test.ts +571 -0
  676. package/test/snapshot/snapshot.test.ts +1121 -0
  677. package/test/storage/storage.test.ts +296 -0
  678. package/test/storage/workspace-time-migration.test.ts +50 -0
  679. package/test/tool/__snapshots__/parameters.test.ts.snap +484 -0
  680. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  681. package/test/tool/apply_patch.test.ts +533 -0
  682. package/test/tool/browser.integration.test.ts +141 -0
  683. package/test/tool/desktop.integration.test.ts +129 -0
  684. package/test/tool/desktop.test.ts +85 -0
  685. package/test/tool/edit.test.ts +578 -0
  686. package/test/tool/external-directory.test.ts +167 -0
  687. package/test/tool/fixtures/large-image.png +0 -0
  688. package/test/tool/fixtures/models-api.json +117299 -0
  689. package/test/tool/glob.test.ts +188 -0
  690. package/test/tool/grep.test.ts +266 -0
  691. package/test/tool/lsp.test.ts +181 -0
  692. package/test/tool/parameters.test.ts +293 -0
  693. package/test/tool/question.test.ts +138 -0
  694. package/test/tool/read.test.ts +659 -0
  695. package/test/tool/registry.test.ts +539 -0
  696. package/test/tool/shell.test.ts +1256 -0
  697. package/test/tool/skill.test.ts +135 -0
  698. package/test/tool/task.test.ts +901 -0
  699. package/test/tool/tool-define.test.ts +153 -0
  700. package/test/tool/truncation.test.ts +266 -0
  701. package/test/tool/webfetch.test.ts +113 -0
  702. package/test/tool/websearch.test.ts +99 -0
  703. package/test/tool/write.test.ts +276 -0
  704. package/test/util/data-url.test.ts +14 -0
  705. package/test/util/error.test.ts +16 -0
  706. package/test/util/filesystem.test.ts +656 -0
  707. package/test/util/glob.test.ts +164 -0
  708. package/test/util/iife.test.ts +36 -0
  709. package/test/util/lazy.test.ts +50 -0
  710. package/test/util/log.test.ts +77 -0
  711. package/test/util/module.test.ts +59 -0
  712. package/test/util/process.test.ts +128 -0
  713. package/test/util/repository.test.ts +93 -0
  714. package/test/util/timeout.test.ts +21 -0
  715. package/test/util/wildcard.test.ts +90 -0
  716. package/tsconfig.json +16 -0
@@ -0,0 +1,1795 @@
1
+ import { afterEach, expect, test } from "bun:test"
2
+ import { mkdir, unlink } from "fs/promises"
3
+ import path from "path"
4
+ import { Effect, Layer } from "effect"
5
+ import { ModelsDev } from "@octocode-ai/core/models-dev"
6
+ import { FSUtil } from "@octocode-ai/core/fs-util"
7
+ import { CrossSpawnSpawner } from "@octocode-ai/core/cross-spawn-spawner"
8
+ import { Global } from "@octocode-ai/core/global"
9
+ import { disposeAllInstances, provideInstanceEffect, tmpdirScoped, TestInstance } from "../fixture/fixture"
10
+ import { markPluginDependenciesReady } from "../fixture/plugin"
11
+ import { Auth } from "@/auth"
12
+ import { Config } from "@/config/config"
13
+ import { Env } from "../../src/env"
14
+ import { Plugin } from "../../src/plugin/index"
15
+ import { Provider } from "@/provider/provider"
16
+
17
+ import { RuntimeFlags } from "@/effect/runtime-flags"
18
+ import { Filesystem } from "@/util/filesystem"
19
+ import { InstanceLayer } from "@/project/instance-layer"
20
+ import { testEffect } from "../lib/effect"
21
+ import { ProviderV2 } from "@octocode-ai/core/provider"
22
+ import { ModelV2 } from "@octocode-ai/core/model"
23
+
24
+ const originalEnv = new Map<string, string | undefined>()
25
+
26
+ const rememberEnv = (k: string) => {
27
+ if (!originalEnv.has(k)) originalEnv.set(k, process.env[k])
28
+ }
29
+
30
+ const setProcessEnv = (k: string, v: string) =>
31
+ Effect.sync(() => {
32
+ rememberEnv(k)
33
+ process.env[k] = v
34
+ })
35
+
36
+ const set = (k: string, v: string) =>
37
+ Effect.gen(function* () {
38
+ rememberEnv(k)
39
+ process.env[k] = v
40
+ yield* Env.use.set(k, v)
41
+ })
42
+
43
+ const remove = (k: string) =>
44
+ Effect.gen(function* () {
45
+ rememberEnv(k)
46
+ delete process.env[k]
47
+ yield* Env.use.remove(k)
48
+ })
49
+
50
+ afterEach(async () => {
51
+ for (const [key, value] of originalEnv) {
52
+ if (value === undefined) delete process.env[key]
53
+ else process.env[key] = value
54
+ }
55
+ originalEnv.clear()
56
+ await disposeAllInstances()
57
+ })
58
+
59
+ const providerLayer = (flags: Partial<RuntimeFlags.Info> = {}) =>
60
+ Provider.layer.pipe(
61
+ Layer.provide(FSUtil.defaultLayer),
62
+ Layer.provide(Env.defaultLayer),
63
+ Layer.provide(Config.defaultLayer),
64
+ Layer.provide(Auth.defaultLayer),
65
+ Layer.provide(Plugin.defaultLayer),
66
+ Layer.provide(ModelsDev.defaultLayer),
67
+ Layer.provide(RuntimeFlags.layer(flags)),
68
+ )
69
+
70
+ const list = Provider.use.list()
71
+
72
+ const paid = (providers: Record<string, { models: Record<string, { cost: { input: number } }> }>) => {
73
+ let count = 0
74
+ for (const provider of Object.values(providers)) {
75
+ count += Object.values(provider.models).filter((model) => model.cost.input > 0).length
76
+ }
77
+ return count
78
+ }
79
+
80
+ const languageBaseURL = (language: unknown) => (language as { config: { baseURL: string } }).config.baseURL
81
+
82
+ const it = testEffect(Layer.mergeAll(Provider.defaultLayer, Env.defaultLayer, Plugin.defaultLayer))
83
+ const experimentalModels = testEffect(providerLayer({ enableExperimentalModels: true }))
84
+
85
+ const alphaProviderConfig = {
86
+ provider: {
87
+ "custom-provider": {
88
+ name: "Custom Provider",
89
+ npm: "@ai-sdk/openai-compatible",
90
+ api: "https://api.custom.com/v1",
91
+ models: {
92
+ "active-model": {
93
+ name: "Active Model",
94
+ },
95
+ "alpha-model": {
96
+ name: "Alpha Model",
97
+ status: "alpha" as const,
98
+ },
99
+ },
100
+ options: {
101
+ apiKey: "custom-key",
102
+ },
103
+ },
104
+ },
105
+ }
106
+
107
+ it.instance("provider loaded from env variable", () =>
108
+ Effect.gen(function* () {
109
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
110
+ const providers = yield* list
111
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
112
+ // Provider should retain its connection source even if custom loaders
113
+ // merge additional options.
114
+ expect(providers[ProviderV2.ID.anthropic].source).toBe("env")
115
+ expect(providers[ProviderV2.ID.anthropic].options.headers["anthropic-beta"]).toBeDefined()
116
+ }),
117
+ )
118
+
119
+ it.instance(
120
+ "provider loaded from config with apiKey option",
121
+ Effect.gen(function* () {
122
+ const providers = yield* list
123
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
124
+ }),
125
+ { config: { provider: { anthropic: { options: { apiKey: "config-api-key" } } } } },
126
+ )
127
+
128
+ it.instance(
129
+ "disabled_providers excludes provider",
130
+ Effect.gen(function* () {
131
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
132
+ const providers = yield* list
133
+ expect(providers[ProviderV2.ID.anthropic]).toBeUndefined()
134
+ }),
135
+ { config: { disabled_providers: ["anthropic"] } },
136
+ )
137
+
138
+ it.instance(
139
+ "enabled_providers restricts to only listed providers",
140
+ Effect.gen(function* () {
141
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
142
+ yield* setProcessEnv("OPENAI_API_KEY", "test-openai-key")
143
+ const providers = yield* list
144
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
145
+ expect(providers[ProviderV2.ID.openai]).toBeUndefined()
146
+ }),
147
+ { config: { enabled_providers: ["anthropic"] } },
148
+ )
149
+
150
+ it.instance(
151
+ "model whitelist filters models for provider",
152
+ Effect.gen(function* () {
153
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
154
+ const providers = yield* list
155
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
156
+ const models = Object.keys(providers[ProviderV2.ID.anthropic].models)
157
+ expect(models).toContain("claude-sonnet-4-20250514")
158
+ expect(models.length).toBe(1)
159
+ }),
160
+ { config: { provider: { anthropic: { whitelist: ["claude-sonnet-4-20250514"] } } } },
161
+ )
162
+
163
+ it.instance(
164
+ "model blacklist excludes specific models",
165
+ Effect.gen(function* () {
166
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
167
+ const providers = yield* list
168
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
169
+ const models = Object.keys(providers[ProviderV2.ID.anthropic].models)
170
+ expect(models).not.toContain("claude-sonnet-4-20250514")
171
+ }),
172
+ { config: { provider: { anthropic: { blacklist: ["claude-sonnet-4-20250514"] } } } },
173
+ )
174
+
175
+ it.instance(
176
+ "custom model alias via config",
177
+ Effect.gen(function* () {
178
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
179
+ const providers = yield* list
180
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
181
+ expect(providers[ProviderV2.ID.anthropic].models["my-alias"]).toBeDefined()
182
+ expect(providers[ProviderV2.ID.anthropic].models["my-alias"].name).toBe("My Custom Alias")
183
+ }),
184
+ {
185
+ config: {
186
+ provider: {
187
+ anthropic: { models: { "my-alias": { id: "claude-sonnet-4-20250514", name: "My Custom Alias" } } },
188
+ },
189
+ },
190
+ },
191
+ )
192
+
193
+ it.instance(
194
+ "custom provider with npm package",
195
+ Effect.gen(function* () {
196
+ const providers = yield* list
197
+ expect(providers[ProviderV2.ID.make("custom-provider")]).toBeDefined()
198
+ expect(providers[ProviderV2.ID.make("custom-provider")].name).toBe("Custom Provider")
199
+ expect(providers[ProviderV2.ID.make("custom-provider")].models["custom-model"]).toBeDefined()
200
+ }),
201
+ {
202
+ config: {
203
+ provider: {
204
+ "custom-provider": {
205
+ name: "Custom Provider",
206
+ npm: "@ai-sdk/openai-compatible",
207
+ api: "https://api.custom.com/v1",
208
+ env: ["CUSTOM_API_KEY"],
209
+ models: {
210
+ "custom-model": {
211
+ name: "Custom Model",
212
+ tool_call: true,
213
+ limit: { context: 128000, output: 4096 },
214
+ },
215
+ },
216
+ options: { apiKey: "custom-key" },
217
+ },
218
+ },
219
+ },
220
+ },
221
+ )
222
+
223
+ it.instance(
224
+ "filters alpha provider models by default",
225
+ Effect.gen(function* () {
226
+ const providers = yield* list
227
+ expect(providers[ProviderV2.ID.make("custom-provider")].models["active-model"]).toBeDefined()
228
+ expect(providers[ProviderV2.ID.make("custom-provider")].models["alpha-model"]).toBeUndefined()
229
+ }),
230
+ { config: alphaProviderConfig },
231
+ )
232
+
233
+ experimentalModels.instance(
234
+ "includes alpha provider models when experimental models are enabled",
235
+ Effect.gen(function* () {
236
+ const providers = yield* list
237
+ expect(providers[ProviderV2.ID.make("custom-provider")].models["active-model"]).toBeDefined()
238
+ expect(providers[ProviderV2.ID.make("custom-provider")].models["alpha-model"]).toBeDefined()
239
+ }),
240
+ { config: alphaProviderConfig },
241
+ )
242
+
243
+ it.instance(
244
+ "custom DeepSeek openai-compatible model defaults interleaved reasoning field",
245
+ Effect.gen(function* () {
246
+ const providers = yield* list
247
+ const provider = providers[ProviderV2.ID.make("custom-provider")]
248
+ expect(provider.models["deepseek-r1"].capabilities.interleaved).toEqual({ field: "reasoning_content" })
249
+ expect(provider.models["deepseek-details"].capabilities.interleaved).toEqual({ field: "reasoning_details" })
250
+ expect(provider.models["custom-model"].capabilities.interleaved).toBe(false)
251
+ expect(
252
+ providers[ProviderV2.ID.make("custom-anthropic-provider")].models["deepseek-r1"].capabilities.interleaved,
253
+ ).toBe(false)
254
+ }),
255
+ {
256
+ config: {
257
+ provider: {
258
+ "custom-provider": {
259
+ name: "Custom Provider",
260
+ npm: "@ai-sdk/openai-compatible",
261
+ api: "https://api.custom.com/v1",
262
+ models: {
263
+ "deepseek-r1": { name: "DeepSeek R1" },
264
+ "deepseek-details": { name: "DeepSeek Details", interleaved: { field: "reasoning_details" } },
265
+ "custom-model": { name: "Custom Model" },
266
+ },
267
+ options: { apiKey: "custom-key" },
268
+ },
269
+ "custom-anthropic-provider": {
270
+ name: "Custom Anthropic Provider",
271
+ npm: "@ai-sdk/anthropic",
272
+ api: "https://api.custom.com/v1",
273
+ models: { "deepseek-r1": { name: "DeepSeek R1" } },
274
+ options: { apiKey: "custom-key" },
275
+ },
276
+ },
277
+ },
278
+ },
279
+ )
280
+
281
+ it.instance(
282
+ "env variable takes precedence, config merges options",
283
+ Effect.gen(function* () {
284
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "env-api-key")
285
+ const providers = yield* list
286
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
287
+ // Config options should be merged
288
+ expect(providers[ProviderV2.ID.anthropic].options.timeout).toBe(60000)
289
+ expect(providers[ProviderV2.ID.anthropic].options.headerTimeout).toBe(10000)
290
+ expect(providers[ProviderV2.ID.anthropic].options.chunkTimeout).toBe(15000)
291
+ }),
292
+ { config: { provider: { anthropic: { options: { timeout: 60000, headerTimeout: 10000, chunkTimeout: 15000 } } } } },
293
+ )
294
+
295
+ it.instance("getModel returns model for valid provider/model", () =>
296
+ Effect.gen(function* () {
297
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
298
+ const provider = yield* Provider.Service
299
+ const model = yield* provider.getModel(ProviderV2.ID.anthropic, ModelV2.ID.make("claude-sonnet-4-20250514"))
300
+ expect(model).toBeDefined()
301
+ expect(String(model.providerID)).toBe("anthropic")
302
+ expect(String(model.id)).toBe("claude-sonnet-4-20250514")
303
+ const language = yield* provider.getLanguage(model)
304
+ expect(language).toBeDefined()
305
+ }),
306
+ )
307
+
308
+ it.instance("getModel throws ModelNotFoundError for invalid model", () =>
309
+ Effect.gen(function* () {
310
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
311
+ const exit = yield* Provider.use
312
+ .getModel(ProviderV2.ID.anthropic, ModelV2.ID.make("nonexistent-model"))
313
+ .pipe(Effect.exit)
314
+ expect(exit._tag).toBe("Failure")
315
+ }),
316
+ )
317
+
318
+ it.instance("getModel throws ModelNotFoundError for invalid provider", () =>
319
+ Effect.gen(function* () {
320
+ const exit = yield* Provider.use
321
+ .getModel(ProviderV2.ID.make("nonexistent-provider"), ModelV2.ID.make("some-model"))
322
+ .pipe(Effect.exit)
323
+ expect(exit._tag).toBe("Failure")
324
+ }),
325
+ )
326
+
327
+ // Pure synchronous unit tests — no Effect runtime needed.
328
+
329
+ test("parseModel correctly parses provider/model string", () => {
330
+ const result = Provider.parseModel("anthropic/claude-sonnet-4")
331
+ expect(String(result.providerID)).toBe("anthropic")
332
+ expect(String(result.modelID)).toBe("claude-sonnet-4")
333
+ })
334
+
335
+ test("parseModel handles model IDs with slashes", () => {
336
+ const result = Provider.parseModel("openrouter/anthropic/claude-3-opus")
337
+ expect(String(result.providerID)).toBe("openrouter")
338
+ expect(String(result.modelID)).toBe("anthropic/claude-3-opus")
339
+ })
340
+
341
+ it.instance("defaultModel returns first available model when no config set", () =>
342
+ Effect.gen(function* () {
343
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
344
+ const model = yield* Provider.use.defaultModel()
345
+ expect(model.providerID).toBeDefined()
346
+ expect(model.modelID).toBeDefined()
347
+ }),
348
+ )
349
+
350
+ it.instance(
351
+ "defaultModel respects config model setting",
352
+ Effect.gen(function* () {
353
+ yield* setProcessEnv("ANTHROPIC_API_KEY", "test-api-key")
354
+ const model = yield* Provider.use.defaultModel()
355
+ expect(String(model.providerID)).toBe("anthropic")
356
+ expect(String(model.modelID)).toBe("claude-sonnet-4-20250514")
357
+ }),
358
+ { config: { model: "anthropic/claude-sonnet-4-20250514" } },
359
+ )
360
+
361
+ it.instance(
362
+ "defaultModel returns a typed error when config excludes every provider",
363
+ Effect.gen(function* () {
364
+ const error = yield* Provider.use.defaultModel().pipe(Effect.flip)
365
+ expect(error).toBeInstanceOf(Provider.NoProvidersError)
366
+ expect(error._tag).toBe("ProviderNoProvidersError")
367
+ }),
368
+ { config: { enabled_providers: [] } },
369
+ )
370
+
371
+ it.instance(
372
+ "provider with baseURL from config",
373
+ Effect.gen(function* () {
374
+ const providers = yield* list
375
+ expect(providers[ProviderV2.ID.make("custom-openai")]).toBeDefined()
376
+ expect(providers[ProviderV2.ID.make("custom-openai")].options.baseURL).toBe("https://custom.openai.com/v1")
377
+ }),
378
+ {
379
+ config: {
380
+ provider: {
381
+ "custom-openai": {
382
+ name: "Custom OpenAI",
383
+ npm: "@ai-sdk/openai-compatible",
384
+ env: [],
385
+ models: { "gpt-4": { name: "GPT-4", tool_call: true, limit: { context: 128000, output: 4096 } } },
386
+ options: { apiKey: "test-key", baseURL: "https://custom.openai.com/v1" },
387
+ },
388
+ },
389
+ },
390
+ },
391
+ )
392
+
393
+ it.instance(
394
+ "model cost defaults to zero when not specified",
395
+ Effect.gen(function* () {
396
+ const providers = yield* list
397
+ const model = providers[ProviderV2.ID.make("test-provider")].models["test-model"]
398
+ expect(model.cost.input).toBe(0)
399
+ expect(model.cost.output).toBe(0)
400
+ expect(model.cost.cache.read).toBe(0)
401
+ expect(model.cost.cache.write).toBe(0)
402
+ }),
403
+ {
404
+ config: {
405
+ provider: {
406
+ "test-provider": {
407
+ name: "Test Provider",
408
+ npm: "@ai-sdk/openai-compatible",
409
+ env: [],
410
+ models: { "test-model": { name: "Test Model", tool_call: true, limit: { context: 128000, output: 4096 } } },
411
+ options: { apiKey: "test-key" },
412
+ },
413
+ },
414
+ },
415
+ },
416
+ )
417
+
418
+ it.instance(
419
+ "model options are merged from existing model",
420
+ Effect.gen(function* () {
421
+ const providers = yield* list
422
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
423
+ expect(model.options.customOption).toBe("custom-value")
424
+ }),
425
+ {
426
+ config: {
427
+ provider: {
428
+ anthropic: {
429
+ options: { apiKey: "test-api-key" },
430
+ models: { "claude-sonnet-4-20250514": { options: { customOption: "custom-value" } } },
431
+ },
432
+ },
433
+ },
434
+ },
435
+ )
436
+
437
+ it.instance(
438
+ "provider removed when all models filtered out",
439
+ Effect.gen(function* () {
440
+ const providers = yield* list
441
+ expect(providers[ProviderV2.ID.anthropic]).toBeUndefined()
442
+ }),
443
+ { config: { provider: { anthropic: { options: { apiKey: "test-api-key" }, whitelist: ["nonexistent-model"] } } } },
444
+ )
445
+
446
+ it.instance("closest finds model by partial match", () =>
447
+ Effect.gen(function* () {
448
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
449
+ const result = yield* Provider.use.closest(ProviderV2.ID.anthropic, ["sonnet-4"])
450
+ expect(result).toBeDefined()
451
+ expect(String(result?.providerID)).toBe("anthropic")
452
+ expect(String(result?.modelID)).toContain("sonnet-4")
453
+ }),
454
+ )
455
+
456
+ it.instance("closest returns undefined for nonexistent provider", () =>
457
+ Effect.gen(function* () {
458
+ const result = yield* Provider.use.closest(ProviderV2.ID.make("nonexistent"), ["model"])
459
+ expect(result).toBeUndefined()
460
+ }),
461
+ )
462
+
463
+ it.instance(
464
+ "getModel uses realIdByKey for aliased models",
465
+ Effect.gen(function* () {
466
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
467
+ const providers = yield* list
468
+ expect(providers[ProviderV2.ID.anthropic].models["my-sonnet"]).toBeDefined()
469
+
470
+ const model = yield* Provider.use.getModel(ProviderV2.ID.anthropic, ModelV2.ID.make("my-sonnet"))
471
+ expect(model).toBeDefined()
472
+ expect(String(model.id)).toBe("my-sonnet")
473
+ expect(model.name).toBe("My Sonnet Alias")
474
+ }),
475
+ {
476
+ config: {
477
+ provider: {
478
+ anthropic: {
479
+ models: { "my-sonnet": { id: "claude-sonnet-4-20250514", name: "My Sonnet Alias" } },
480
+ },
481
+ },
482
+ },
483
+ },
484
+ )
485
+
486
+ it.instance(
487
+ "provider api field sets model api.url",
488
+ Effect.gen(function* () {
489
+ const providers = yield* list
490
+ // api field is stored on model.api.url, used by getSDK to set baseURL
491
+ expect(providers[ProviderV2.ID.make("custom-api")].models["model-1"].api.url).toBe("https://api.example.com/v1")
492
+ }),
493
+ {
494
+ config: {
495
+ provider: {
496
+ "custom-api": {
497
+ name: "Custom API",
498
+ npm: "@ai-sdk/openai-compatible",
499
+ api: "https://api.example.com/v1",
500
+ env: [],
501
+ models: { "model-1": { name: "Model 1", tool_call: true, limit: { context: 8000, output: 2000 } } },
502
+ options: { apiKey: "test-key" },
503
+ },
504
+ },
505
+ },
506
+ },
507
+ )
508
+
509
+ it.instance(
510
+ "explicit baseURL overrides api field",
511
+ Effect.gen(function* () {
512
+ const providers = yield* list
513
+ expect(providers[ProviderV2.ID.make("custom-api")].options.baseURL).toBe("https://custom.override.com/v1")
514
+ }),
515
+ {
516
+ config: {
517
+ provider: {
518
+ "custom-api": {
519
+ name: "Custom API",
520
+ npm: "@ai-sdk/openai-compatible",
521
+ api: "https://api.example.com/v1",
522
+ env: [],
523
+ models: { "model-1": { name: "Model 1", tool_call: true, limit: { context: 8000, output: 2000 } } },
524
+ options: { apiKey: "test-key", baseURL: "https://custom.override.com/v1" },
525
+ },
526
+ },
527
+ },
528
+ },
529
+ )
530
+
531
+ it.instance(
532
+ "model inherits properties from existing database model",
533
+ Effect.gen(function* () {
534
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
535
+ const providers = yield* list
536
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
537
+ expect(model.name).toBe("Custom Name for Sonnet")
538
+ expect(model.capabilities.toolcall).toBe(true)
539
+ expect(model.capabilities.attachment).toBe(true)
540
+ expect(model.limit.context).toBeGreaterThan(0)
541
+ }),
542
+ {
543
+ config: {
544
+ provider: { anthropic: { models: { "claude-sonnet-4-20250514": { name: "Custom Name for Sonnet" } } } },
545
+ },
546
+ },
547
+ )
548
+
549
+ it.instance(
550
+ "disabled_providers prevents loading even with env var",
551
+ Effect.gen(function* () {
552
+ yield* set("OPENAI_API_KEY", "test-openai-key")
553
+ const providers = yield* list
554
+ expect(providers[ProviderV2.ID.openai]).toBeUndefined()
555
+ }),
556
+ { config: { disabled_providers: ["openai"] } },
557
+ )
558
+
559
+ it.instance(
560
+ "enabled_providers with empty array allows no providers",
561
+ Effect.gen(function* () {
562
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
563
+ yield* set("OPENAI_API_KEY", "test-openai-key")
564
+ const providers = yield* list
565
+ expect(Object.keys(providers).length).toBe(0)
566
+ }),
567
+ { config: { enabled_providers: [] } },
568
+ )
569
+
570
+ it.instance(
571
+ "whitelist and blacklist can be combined",
572
+ Effect.gen(function* () {
573
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
574
+ const providers = yield* list
575
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
576
+ const models = Object.keys(providers[ProviderV2.ID.anthropic].models)
577
+ expect(models).toContain("claude-sonnet-4-20250514")
578
+ expect(models).not.toContain("claude-opus-4-20250514")
579
+ expect(models.length).toBe(1)
580
+ }),
581
+ {
582
+ config: {
583
+ provider: {
584
+ anthropic: {
585
+ whitelist: ["claude-sonnet-4-20250514", "claude-opus-4-20250514"],
586
+ blacklist: ["claude-opus-4-20250514"],
587
+ },
588
+ },
589
+ },
590
+ },
591
+ )
592
+
593
+ it.instance(
594
+ "model modalities default correctly",
595
+ Effect.gen(function* () {
596
+ const providers = yield* list
597
+ const model = providers[ProviderV2.ID.make("test-provider")].models["test-model"]
598
+ expect(model.capabilities.input.text).toBe(true)
599
+ expect(model.capabilities.output.text).toBe(true)
600
+ }),
601
+ {
602
+ config: {
603
+ provider: {
604
+ "test-provider": {
605
+ name: "Test",
606
+ npm: "@ai-sdk/openai-compatible",
607
+ env: [],
608
+ models: { "test-model": { name: "Test Model", tool_call: true, limit: { context: 8000, output: 2000 } } },
609
+ options: { apiKey: "test" },
610
+ },
611
+ },
612
+ },
613
+ },
614
+ )
615
+
616
+ it.instance(
617
+ "model with custom cost values",
618
+ Effect.gen(function* () {
619
+ const providers = yield* list
620
+ const model = providers[ProviderV2.ID.make("test-provider")].models["test-model"]
621
+ expect(model.cost.input).toBe(5)
622
+ expect(model.cost.output).toBe(15)
623
+ expect(model.cost.cache.read).toBe(2.5)
624
+ expect(model.cost.cache.write).toBe(7.5)
625
+ }),
626
+ {
627
+ config: {
628
+ provider: {
629
+ "test-provider": {
630
+ name: "Test",
631
+ npm: "@ai-sdk/openai-compatible",
632
+ env: [],
633
+ models: {
634
+ "test-model": {
635
+ name: "Test Model",
636
+ tool_call: true,
637
+ limit: { context: 8000, output: 2000 },
638
+ cost: { input: 5, output: 15, cache_read: 2.5, cache_write: 7.5 },
639
+ },
640
+ },
641
+ options: { apiKey: "test" },
642
+ },
643
+ },
644
+ },
645
+ },
646
+ )
647
+
648
+ it.instance("getSmallModel returns appropriate small model", () =>
649
+ Effect.gen(function* () {
650
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
651
+ const model = yield* Provider.use.getSmallModel(ProviderV2.ID.anthropic)
652
+ expect(model).toBeDefined()
653
+ expect(model?.id).toContain("haiku")
654
+ }),
655
+ )
656
+
657
+ it.instance(
658
+ "getSmallModel respects config small_model override",
659
+ Effect.gen(function* () {
660
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
661
+ const model = yield* Provider.use.getSmallModel(ProviderV2.ID.anthropic)
662
+ expect(model).toBeDefined()
663
+ expect(String(model?.providerID)).toBe("anthropic")
664
+ expect(String(model?.id)).toBe("claude-sonnet-4-20250514")
665
+ }),
666
+ { config: { small_model: "anthropic/claude-sonnet-4-20250514" } },
667
+ )
668
+
669
+ it.instance(
670
+ "getSmallModel ignores invalid config small_model",
671
+ Effect.gen(function* () {
672
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
673
+ const model = yield* Provider.use.getSmallModel(ProviderV2.ID.anthropic)
674
+ expect(model).toBeUndefined()
675
+ }),
676
+ { config: { small_model: "anthropic/not-a-real-model" } },
677
+ )
678
+
679
+ test("provider.sort prioritizes preferred models", () => {
680
+ const models = [
681
+ { id: "random-model", name: "Random" },
682
+ { id: "claude-sonnet-4-latest", name: "Claude Sonnet 4" },
683
+ { id: "gpt-5-turbo", name: "GPT-5 Turbo" },
684
+ { id: "other-model", name: "Other" },
685
+ ] as any[]
686
+
687
+ const sorted = Provider.sort(models)
688
+ expect(sorted[0].id).toContain("sonnet-4")
689
+ expect(sorted[0].id).toContain("latest")
690
+ expect(sorted[sorted.length - 1].id).not.toContain("gpt-5")
691
+ expect(sorted[sorted.length - 1].id).not.toContain("sonnet-4")
692
+ })
693
+
694
+ it.instance(
695
+ "multiple providers can be configured simultaneously",
696
+ Effect.gen(function* () {
697
+ yield* set("ANTHROPIC_API_KEY", "test-anthropic-key")
698
+ yield* set("OPENAI_API_KEY", "test-openai-key")
699
+ const providers = yield* list
700
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
701
+ expect(providers[ProviderV2.ID.openai]).toBeDefined()
702
+ expect(providers[ProviderV2.ID.anthropic].options.timeout).toBe(30000)
703
+ expect(providers[ProviderV2.ID.openai].options.timeout).toBe(60000)
704
+ }),
705
+ {
706
+ config: {
707
+ provider: {
708
+ anthropic: { options: { timeout: 30000 } },
709
+ openai: { options: { timeout: 60000 } },
710
+ },
711
+ },
712
+ },
713
+ )
714
+
715
+ it.instance(
716
+ "provider with custom npm package",
717
+ Effect.gen(function* () {
718
+ const providers = yield* list
719
+ expect(providers[ProviderV2.ID.make("local-llm")]).toBeDefined()
720
+ expect(providers[ProviderV2.ID.make("local-llm")].models["llama-3"].api.npm).toBe("@ai-sdk/openai-compatible")
721
+ expect(providers[ProviderV2.ID.make("local-llm")].options.baseURL).toBe("http://localhost:11434/v1")
722
+ }),
723
+ {
724
+ config: {
725
+ provider: {
726
+ "local-llm": {
727
+ name: "Local LLM",
728
+ npm: "@ai-sdk/openai-compatible",
729
+ env: [],
730
+ models: { "llama-3": { name: "Llama 3", tool_call: true, limit: { context: 8192, output: 2048 } } },
731
+ options: { apiKey: "not-needed", baseURL: "http://localhost:11434/v1" },
732
+ },
733
+ },
734
+ },
735
+ },
736
+ )
737
+
738
+ // Edge cases for model configuration
739
+
740
+ it.instance(
741
+ "model alias name defaults to alias key when id differs",
742
+ Effect.gen(function* () {
743
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
744
+ const providers = yield* list
745
+ expect(providers[ProviderV2.ID.anthropic].models["sonnet"].name).toBe("sonnet")
746
+ }),
747
+ {
748
+ config: {
749
+ provider: {
750
+ anthropic: {
751
+ models: { sonnet: { id: "claude-sonnet-4-20250514" } },
752
+ },
753
+ },
754
+ },
755
+ },
756
+ )
757
+
758
+ it.instance(
759
+ "provider with multiple env var options only includes apiKey when single env",
760
+ Effect.gen(function* () {
761
+ yield* set("MULTI_ENV_KEY_1", "test-key")
762
+ const providers = yield* list
763
+ expect(providers[ProviderV2.ID.make("multi-env")]).toBeDefined()
764
+ // When multiple env options exist, key should NOT be auto-set
765
+ expect(providers[ProviderV2.ID.make("multi-env")].key).toBeUndefined()
766
+ }),
767
+ {
768
+ config: {
769
+ provider: {
770
+ "multi-env": {
771
+ name: "Multi Env Provider",
772
+ npm: "@ai-sdk/openai-compatible",
773
+ env: ["MULTI_ENV_KEY_1", "MULTI_ENV_KEY_2"],
774
+ models: { "model-1": { name: "Model 1", tool_call: true, limit: { context: 8000, output: 2000 } } },
775
+ options: { baseURL: "https://api.example.com/v1" },
776
+ },
777
+ },
778
+ },
779
+ },
780
+ )
781
+
782
+ it.instance(
783
+ "provider with single env var includes apiKey automatically",
784
+ Effect.gen(function* () {
785
+ yield* set("SINGLE_ENV_KEY", "my-api-key")
786
+ const providers = yield* list
787
+ expect(providers[ProviderV2.ID.make("single-env")]).toBeDefined()
788
+ // Single env option should auto-set key
789
+ expect(providers[ProviderV2.ID.make("single-env")].key).toBe("my-api-key")
790
+ }),
791
+ {
792
+ config: {
793
+ provider: {
794
+ "single-env": {
795
+ name: "Single Env Provider",
796
+ npm: "@ai-sdk/openai-compatible",
797
+ env: ["SINGLE_ENV_KEY"],
798
+ models: { "model-1": { name: "Model 1", tool_call: true, limit: { context: 8000, output: 2000 } } },
799
+ options: { baseURL: "https://api.example.com/v1" },
800
+ },
801
+ },
802
+ },
803
+ },
804
+ )
805
+
806
+ it.instance(
807
+ "model cost overrides existing cost values",
808
+ Effect.gen(function* () {
809
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
810
+ const providers = yield* list
811
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
812
+ expect(model.cost.input).toBe(999)
813
+ expect(model.cost.output).toBe(888)
814
+ }),
815
+ {
816
+ config: {
817
+ provider: {
818
+ anthropic: {
819
+ models: { "claude-sonnet-4-20250514": { cost: { input: 999, output: 888 } } },
820
+ },
821
+ },
822
+ },
823
+ },
824
+ )
825
+
826
+ it.instance(
827
+ "completely new provider not in database can be configured",
828
+ Effect.gen(function* () {
829
+ const providers = yield* list
830
+ expect(providers[ProviderV2.ID.make("brand-new-provider")]).toBeDefined()
831
+ expect(providers[ProviderV2.ID.make("brand-new-provider")].name).toBe("Brand New")
832
+ const model = providers[ProviderV2.ID.make("brand-new-provider")].models["new-model"]
833
+ expect(model.capabilities.reasoning).toBe(true)
834
+ expect(model.capabilities.attachment).toBe(true)
835
+ expect(model.capabilities.input.image).toBe(true)
836
+ }),
837
+ {
838
+ config: {
839
+ provider: {
840
+ "brand-new-provider": {
841
+ name: "Brand New",
842
+ npm: "@ai-sdk/openai-compatible",
843
+ env: [],
844
+ api: "https://new-api.com/v1",
845
+ models: {
846
+ "new-model": {
847
+ name: "New Model",
848
+ tool_call: true,
849
+ reasoning: true,
850
+ attachment: true,
851
+ temperature: true,
852
+ limit: { context: 32000, output: 8000 },
853
+ modalities: { input: ["text", "image"], output: ["text"] },
854
+ },
855
+ },
856
+ options: { apiKey: "new-key" },
857
+ },
858
+ },
859
+ },
860
+ },
861
+ )
862
+
863
+ it.instance(
864
+ "disabled_providers and enabled_providers interaction",
865
+ Effect.gen(function* () {
866
+ yield* set("ANTHROPIC_API_KEY", "test-anthropic")
867
+ yield* set("OPENAI_API_KEY", "test-openai")
868
+ yield* set("GOOGLE_GENERATIVE_AI_API_KEY", "test-google")
869
+ const providers = yield* list
870
+ // anthropic: in enabled, not in disabled = allowed
871
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
872
+ // openai: in enabled, but also in disabled = NOT allowed
873
+ expect(providers[ProviderV2.ID.openai]).toBeUndefined()
874
+ // google: not in enabled = NOT allowed (even though not disabled)
875
+ expect(providers[ProviderV2.ID.google]).toBeUndefined()
876
+ }),
877
+ {
878
+ // enabled_providers takes precedence — only these are considered
879
+ // Then disabled_providers filters from the enabled set
880
+ config: { enabled_providers: ["anthropic", "openai"], disabled_providers: ["openai"] },
881
+ },
882
+ )
883
+
884
+ it.instance(
885
+ "model with tool_call false",
886
+ Effect.gen(function* () {
887
+ const providers = yield* list
888
+ expect(providers[ProviderV2.ID.make("no-tools")].models["basic-model"].capabilities.toolcall).toBe(false)
889
+ }),
890
+ {
891
+ config: {
892
+ provider: {
893
+ "no-tools": {
894
+ name: "No Tools Provider",
895
+ npm: "@ai-sdk/openai-compatible",
896
+ env: [],
897
+ models: { "basic-model": { name: "Basic Model", tool_call: false, limit: { context: 4000, output: 1000 } } },
898
+ options: { apiKey: "test" },
899
+ },
900
+ },
901
+ },
902
+ },
903
+ )
904
+
905
+ it.instance(
906
+ "model defaults tool_call to true when not specified",
907
+ Effect.gen(function* () {
908
+ const providers = yield* list
909
+ expect(providers[ProviderV2.ID.make("default-tools")].models["model"].capabilities.toolcall).toBe(true)
910
+ }),
911
+ {
912
+ config: {
913
+ provider: {
914
+ "default-tools": {
915
+ name: "Default Tools Provider",
916
+ npm: "@ai-sdk/openai-compatible",
917
+ env: [],
918
+ models: { model: { name: "Model", limit: { context: 4000, output: 1000 } } },
919
+ options: { apiKey: "test" },
920
+ },
921
+ },
922
+ },
923
+ },
924
+ )
925
+
926
+ it.instance(
927
+ "model headers are preserved",
928
+ Effect.gen(function* () {
929
+ const providers = yield* list
930
+ const model = providers[ProviderV2.ID.make("headers-provider")].models["model"]
931
+ expect(model.headers).toEqual({
932
+ "X-Custom-Header": "custom-value",
933
+ Authorization: "Bearer special-token",
934
+ })
935
+ }),
936
+ {
937
+ config: {
938
+ provider: {
939
+ "headers-provider": {
940
+ name: "Headers Provider",
941
+ npm: "@ai-sdk/openai-compatible",
942
+ env: [],
943
+ models: {
944
+ model: {
945
+ name: "Model",
946
+ tool_call: true,
947
+ limit: { context: 4000, output: 1000 },
948
+ headers: { "X-Custom-Header": "custom-value", Authorization: "Bearer special-token" },
949
+ },
950
+ },
951
+ options: { apiKey: "test" },
952
+ },
953
+ },
954
+ },
955
+ },
956
+ )
957
+
958
+ it.instance(
959
+ "provider env fallback - second env var used if first missing",
960
+ Effect.gen(function* () {
961
+ // Only set fallback, not primary
962
+ yield* set("FALLBACK_KEY", "fallback-api-key")
963
+ const providers = yield* list
964
+ // Provider should load because fallback env var is set
965
+ expect(providers[ProviderV2.ID.make("fallback-env")]).toBeDefined()
966
+ }),
967
+ {
968
+ config: {
969
+ provider: {
970
+ "fallback-env": {
971
+ name: "Fallback Env Provider",
972
+ npm: "@ai-sdk/openai-compatible",
973
+ env: ["PRIMARY_KEY", "FALLBACK_KEY"],
974
+ models: { model: { name: "Model", tool_call: true, limit: { context: 4000, output: 1000 } } },
975
+ options: { baseURL: "https://api.example.com" },
976
+ },
977
+ },
978
+ },
979
+ },
980
+ )
981
+
982
+ it.instance("getModel returns consistent results", () =>
983
+ Effect.gen(function* () {
984
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
985
+ const model1 = yield* Provider.use.getModel(ProviderV2.ID.anthropic, ModelV2.ID.make("claude-sonnet-4-20250514"))
986
+ const model2 = yield* Provider.use.getModel(ProviderV2.ID.anthropic, ModelV2.ID.make("claude-sonnet-4-20250514"))
987
+ expect(model1.providerID).toEqual(model2.providerID)
988
+ expect(model1.id).toEqual(model2.id)
989
+ expect(model1).toEqual(model2)
990
+ }),
991
+ )
992
+
993
+ it.instance(
994
+ "provider name defaults to id when not in database",
995
+ Effect.gen(function* () {
996
+ const providers = yield* list
997
+ expect(providers[ProviderV2.ID.make("my-custom-id")].name).toBe("my-custom-id")
998
+ }),
999
+ {
1000
+ config: {
1001
+ provider: {
1002
+ "my-custom-id": {
1003
+ npm: "@ai-sdk/openai-compatible",
1004
+ env: [],
1005
+ models: { model: { name: "Model", tool_call: true, limit: { context: 4000, output: 1000 } } },
1006
+ options: { apiKey: "test" },
1007
+ },
1008
+ },
1009
+ },
1010
+ },
1011
+ )
1012
+
1013
+ it.instance("ModelNotFoundError includes suggestions for typos", () =>
1014
+ Effect.gen(function* () {
1015
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1016
+ const error = yield* Provider.use
1017
+ .getModel(ProviderV2.ID.anthropic, ModelV2.ID.make("claude-sonet-4"))
1018
+ .pipe(Effect.flip)
1019
+ expect(error.suggestions).toBeDefined()
1020
+ expect((error.suggestions ?? []).length).toBeGreaterThan(0)
1021
+ }),
1022
+ )
1023
+
1024
+ it.instance("ModelNotFoundError for provider includes suggestions", () =>
1025
+ Effect.gen(function* () {
1026
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1027
+ const error = yield* Provider.use
1028
+ .getModel(ProviderV2.ID.make("antropic"), ModelV2.ID.make("claude-sonnet-4"))
1029
+ .pipe(Effect.flip)
1030
+ expect(error.suggestions).toBeDefined()
1031
+ expect(error.suggestions).toContain("anthropic")
1032
+ }),
1033
+ )
1034
+
1035
+ it.instance("ModelNotFoundError suggests catalog models for unloaded providers", () =>
1036
+ Effect.gen(function* () {
1037
+ yield* remove("OCTOCODE_API_KEY")
1038
+ const error = yield* Provider.use
1039
+ .getModel(ProviderV2.ID.octocode, ModelV2.ID.make("claude-haiku-fake-model"))
1040
+ .pipe(Effect.flip)
1041
+ if (!Provider.ModelNotFoundError.isInstance(error)) throw error
1042
+ expect(error.suggestions ?? []).toContain("claude-haiku-4-5")
1043
+ }),
1044
+ )
1045
+
1046
+ it.instance("getProvider returns undefined for nonexistent provider", () =>
1047
+ Effect.gen(function* () {
1048
+ const provider = yield* Provider.Service.use((svc) => svc.getProvider(ProviderV2.ID.make("nonexistent")))
1049
+ expect(provider).toBeUndefined()
1050
+ }),
1051
+ )
1052
+
1053
+ it.instance("getProvider returns provider info", () =>
1054
+ Effect.gen(function* () {
1055
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1056
+ const provider = yield* Provider.use.getProvider(ProviderV2.ID.anthropic)
1057
+ expect(provider).toBeDefined()
1058
+ expect(String(provider?.id)).toBe("anthropic")
1059
+ }),
1060
+ )
1061
+
1062
+ it.instance("closest returns undefined when no partial match found", () =>
1063
+ Effect.gen(function* () {
1064
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1065
+ const result = yield* Provider.use.closest(ProviderV2.ID.anthropic, ["nonexistent-xyz-model"])
1066
+ expect(result).toBeUndefined()
1067
+ }),
1068
+ )
1069
+
1070
+ it.instance("closest checks multiple query terms in order", () =>
1071
+ Effect.gen(function* () {
1072
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1073
+ // First term won't match, second will
1074
+ const result = yield* Provider.use.closest(ProviderV2.ID.anthropic, ["nonexistent", "haiku"])
1075
+ expect(result).toBeDefined()
1076
+ expect(result?.modelID).toContain("haiku")
1077
+ }),
1078
+ )
1079
+
1080
+ it.instance(
1081
+ "model limit defaults to zero when not specified",
1082
+ Effect.gen(function* () {
1083
+ const providers = yield* list
1084
+ const model = providers[ProviderV2.ID.make("no-limit")].models["model"]
1085
+ expect(model.limit.context).toBe(0)
1086
+ expect(model.limit.output).toBe(0)
1087
+ }),
1088
+ {
1089
+ config: {
1090
+ provider: {
1091
+ "no-limit": {
1092
+ name: "No Limit Provider",
1093
+ npm: "@ai-sdk/openai-compatible",
1094
+ env: [],
1095
+ models: { model: { name: "Model", tool_call: true } },
1096
+ options: { apiKey: "test" },
1097
+ },
1098
+ },
1099
+ },
1100
+ },
1101
+ )
1102
+
1103
+ it.instance(
1104
+ "provider options are deeply merged",
1105
+ Effect.gen(function* () {
1106
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1107
+ const providers = yield* list
1108
+ // Custom options should be merged
1109
+ expect(providers[ProviderV2.ID.anthropic].options.timeout).toBe(30000)
1110
+ expect(providers[ProviderV2.ID.anthropic].options.headers["X-Custom"]).toBe("custom-value")
1111
+ // anthropic custom loader adds its own headers, they should coexist
1112
+ expect(providers[ProviderV2.ID.anthropic].options.headers["anthropic-beta"]).toBeDefined()
1113
+ }),
1114
+ {
1115
+ config: {
1116
+ provider: { anthropic: { options: { headers: { "X-Custom": "custom-value" }, timeout: 30000 } } },
1117
+ },
1118
+ },
1119
+ )
1120
+
1121
+ it.instance(
1122
+ "hosted nvidia provider adds billing origin header",
1123
+ Effect.gen(function* () {
1124
+ const providers = yield* list
1125
+ expect(providers[ProviderV2.ID.make("nvidia")].options.headers).toEqual({
1126
+ "HTTP-Referer": "https://octocode.ai/",
1127
+ "X-Title": "octo",
1128
+ "X-BILLING-INVOKE-ORIGIN": "OctoCode",
1129
+ })
1130
+ }),
1131
+ { config: { provider: { nvidia: { options: { apiKey: "test-api-key" } } } } },
1132
+ )
1133
+
1134
+ it.instance(
1135
+ "custom nvidia baseURL adds billing origin header",
1136
+ Effect.gen(function* () {
1137
+ const providers = yield* list
1138
+ expect(providers[ProviderV2.ID.make("nvidia")].options.headers).toEqual({
1139
+ "HTTP-Referer": "https://octocode.ai/",
1140
+ "X-Title": "octo",
1141
+ "X-BILLING-INVOKE-ORIGIN": "OctoCode",
1142
+ })
1143
+ }),
1144
+ { config: { provider: { nvidia: { options: { apiKey: "test-api-key", baseURL: "http://localhost:8000/v1" } } } } },
1145
+ )
1146
+
1147
+ it.instance(
1148
+ "explicit nvidia billing origin header is preserved",
1149
+ Effect.gen(function* () {
1150
+ const providers = yield* list
1151
+ expect(providers[ProviderV2.ID.make("nvidia")].options.headers["X-BILLING-INVOKE-ORIGIN"]).toBe("CustomOrigin")
1152
+ }),
1153
+ {
1154
+ config: {
1155
+ provider: {
1156
+ nvidia: {
1157
+ options: {
1158
+ apiKey: "test-api-key",
1159
+ baseURL: "http://localhost:8000/v1",
1160
+ headers: { "X-BILLING-INVOKE-ORIGIN": "CustomOrigin" },
1161
+ },
1162
+ },
1163
+ },
1164
+ },
1165
+ },
1166
+ )
1167
+
1168
+ it.instance(
1169
+ "custom model inherits npm package from models.dev provider config",
1170
+ Effect.gen(function* () {
1171
+ yield* set("OPENAI_API_KEY", "test-api-key")
1172
+ const providers = yield* list
1173
+ const model = providers[ProviderV2.ID.openai].models["my-custom-model"]
1174
+ expect(model).toBeDefined()
1175
+ expect(model.api.npm).toBe("@ai-sdk/openai")
1176
+ }),
1177
+ {
1178
+ config: {
1179
+ provider: {
1180
+ openai: {
1181
+ models: {
1182
+ "my-custom-model": {
1183
+ name: "My Custom Model",
1184
+ tool_call: true,
1185
+ limit: { context: 8000, output: 2000 },
1186
+ },
1187
+ },
1188
+ },
1189
+ },
1190
+ },
1191
+ },
1192
+ )
1193
+
1194
+ it.instance(
1195
+ "custom model inherits api.url from models.dev provider",
1196
+ Effect.gen(function* () {
1197
+ yield* set("OPENROUTER_API_KEY", "test-api-key")
1198
+ const providers = yield* list
1199
+ expect(providers[ProviderV2.ID.openrouter]).toBeDefined()
1200
+
1201
+ // New model not in database should inherit api.url from provider
1202
+ const intellect = providers[ProviderV2.ID.openrouter].models["prime-intellect/intellect-3"]
1203
+ expect(intellect).toBeDefined()
1204
+ expect(intellect.api.url).toBe("https://openrouter.ai/api/v1")
1205
+
1206
+ // Another new model should also inherit api.url
1207
+ const deepseek = providers[ProviderV2.ID.openrouter].models["deepseek/deepseek-r1-0528"]
1208
+ expect(deepseek).toBeDefined()
1209
+ expect(deepseek.api.url).toBe("https://openrouter.ai/api/v1")
1210
+ expect(deepseek.name).toBe("DeepSeek R1")
1211
+ }),
1212
+ {
1213
+ config: {
1214
+ provider: {
1215
+ openrouter: {
1216
+ models: {
1217
+ "prime-intellect/intellect-3": {},
1218
+ "deepseek/deepseek-r1-0528": { name: "DeepSeek R1" },
1219
+ },
1220
+ },
1221
+ },
1222
+ },
1223
+ },
1224
+ )
1225
+
1226
+ test("mode cost preserves over-200k pricing from base model", () => {
1227
+ const provider = {
1228
+ id: "openai",
1229
+ name: "OpenAI",
1230
+ env: [],
1231
+ api: "https://api.openai.com/v1",
1232
+ models: {
1233
+ "gpt-5.4": {
1234
+ id: "gpt-5.4",
1235
+ name: "GPT-5.4",
1236
+ family: "gpt",
1237
+ release_date: "2026-03-05",
1238
+ attachment: true,
1239
+ reasoning: true,
1240
+ temperature: false,
1241
+ tool_call: true,
1242
+ cost: {
1243
+ input: 2.5,
1244
+ output: 15,
1245
+ cache_read: 0.25,
1246
+ context_over_200k: {
1247
+ input: 5,
1248
+ output: 22.5,
1249
+ cache_read: 0.5,
1250
+ },
1251
+ },
1252
+ limit: {
1253
+ context: 1_050_000,
1254
+ input: 922_000,
1255
+ output: 128_000,
1256
+ },
1257
+ experimental: {
1258
+ modes: {
1259
+ fast: {
1260
+ cost: {
1261
+ input: 5,
1262
+ output: 30,
1263
+ cache_read: 0.5,
1264
+ },
1265
+ provider: {
1266
+ body: {
1267
+ service_tier: "priority",
1268
+ },
1269
+ },
1270
+ },
1271
+ },
1272
+ },
1273
+ },
1274
+ },
1275
+ } as unknown as ModelsDev.Provider
1276
+
1277
+ const model = Provider.fromModelsDevProvider(provider).models["gpt-5.4-fast"]
1278
+ expect(model.cost.input).toEqual(5)
1279
+ expect(model.cost.output).toEqual(30)
1280
+ expect(model.cost.cache.read).toEqual(0.5)
1281
+ expect(model.cost.cache.write).toEqual(0)
1282
+ expect(model.options["serviceTier"]).toEqual("priority")
1283
+ expect(model.cost.experimentalOver200K).toEqual({
1284
+ input: 5,
1285
+ output: 22.5,
1286
+ cache: { read: 0.5, write: 0 },
1287
+ })
1288
+ })
1289
+
1290
+ test("models.dev normalization fills required response fields", () => {
1291
+ const provider = {
1292
+ id: "gateway",
1293
+ name: "Gateway",
1294
+ env: [],
1295
+ models: {
1296
+ "gpt-5.4": {
1297
+ id: "gpt-5.4",
1298
+ name: "GPT-5.4",
1299
+ family: "gpt",
1300
+ cost: { input: 2.5, output: 15 },
1301
+ limit: { context: 1_050_000, input: 922_000, output: 128_000 },
1302
+ },
1303
+ },
1304
+ } as unknown as ModelsDev.Provider
1305
+
1306
+ const model = Provider.fromModelsDevProvider(provider).models["gpt-5.4"]
1307
+ expect(model.api.url).toBe("")
1308
+ expect(model.capabilities.temperature).toBe(false)
1309
+ expect(model.capabilities.reasoning).toBe(false)
1310
+ expect(model.capabilities.attachment).toBe(false)
1311
+ expect(model.capabilities.toolcall).toBe(true)
1312
+ expect(model.release_date).toBe("")
1313
+ })
1314
+
1315
+ it.instance("model variants are generated for reasoning models", () =>
1316
+ Effect.gen(function* () {
1317
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1318
+ const providers = yield* list
1319
+ // Claude sonnet 4 has reasoning capability
1320
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
1321
+ expect(model.capabilities.reasoning).toBe(true)
1322
+ expect(model.variants).toBeDefined()
1323
+ expect(Object.keys(model.variants!).length).toBeGreaterThan(0)
1324
+ }),
1325
+ )
1326
+
1327
+ it.instance(
1328
+ "model variants can be disabled via config",
1329
+ Effect.gen(function* () {
1330
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1331
+ const providers = yield* list
1332
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
1333
+ expect(model.variants).toBeDefined()
1334
+ expect(model.variants!["high"]).toBeUndefined()
1335
+ // max variant should still exist
1336
+ expect(model.variants!["max"]).toBeDefined()
1337
+ }),
1338
+ {
1339
+ config: {
1340
+ provider: {
1341
+ anthropic: {
1342
+ models: { "claude-sonnet-4-20250514": { variants: { high: { disabled: true } } } },
1343
+ },
1344
+ },
1345
+ },
1346
+ },
1347
+ )
1348
+
1349
+ it.instance(
1350
+ "model variants can be customized via config",
1351
+ Effect.gen(function* () {
1352
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1353
+ const providers = yield* list
1354
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
1355
+ expect(model.variants!["high"]).toBeDefined()
1356
+ expect(model.variants!["high"].thinking.budgetTokens).toBe(20000)
1357
+ }),
1358
+ {
1359
+ config: {
1360
+ provider: {
1361
+ anthropic: {
1362
+ models: {
1363
+ "claude-sonnet-4-20250514": {
1364
+ variants: { high: { thinking: { type: "enabled", budgetTokens: 20000 } } },
1365
+ },
1366
+ },
1367
+ },
1368
+ },
1369
+ },
1370
+ },
1371
+ )
1372
+
1373
+ it.instance(
1374
+ "disabled key is stripped from variant config",
1375
+ Effect.gen(function* () {
1376
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1377
+ const providers = yield* list
1378
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
1379
+ expect(model.variants!["max"]).toBeDefined()
1380
+ expect(model.variants!["max"].disabled).toBeUndefined()
1381
+ expect(model.variants!["max"].customField).toBe("test")
1382
+ }),
1383
+ {
1384
+ config: {
1385
+ provider: {
1386
+ anthropic: {
1387
+ models: {
1388
+ "claude-sonnet-4-20250514": {
1389
+ variants: { max: { disabled: false, customField: "test" } },
1390
+ },
1391
+ },
1392
+ },
1393
+ },
1394
+ },
1395
+ },
1396
+ )
1397
+
1398
+ it.instance(
1399
+ "all variants can be disabled via config",
1400
+ Effect.gen(function* () {
1401
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1402
+ const providers = yield* list
1403
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
1404
+ expect(model.variants).toBeDefined()
1405
+ expect(Object.keys(model.variants!).length).toBe(0)
1406
+ }),
1407
+ {
1408
+ config: {
1409
+ provider: {
1410
+ anthropic: {
1411
+ models: {
1412
+ "claude-sonnet-4-20250514": {
1413
+ variants: { high: { disabled: true }, max: { disabled: true } },
1414
+ },
1415
+ },
1416
+ },
1417
+ },
1418
+ },
1419
+ },
1420
+ )
1421
+
1422
+ it.instance(
1423
+ "variant config merges with generated variants",
1424
+ Effect.gen(function* () {
1425
+ yield* set("ANTHROPIC_API_KEY", "test-api-key")
1426
+ const providers = yield* list
1427
+ const model = providers[ProviderV2.ID.anthropic].models["claude-sonnet-4-20250514"]
1428
+ expect(model.variants!["high"]).toBeDefined()
1429
+ // Should have both the generated thinking config and the custom option
1430
+ expect(model.variants!["high"].thinking).toBeDefined()
1431
+ expect(model.variants!["high"].extraOption).toBe("custom-value")
1432
+ }),
1433
+ {
1434
+ config: {
1435
+ provider: {
1436
+ anthropic: {
1437
+ models: {
1438
+ "claude-sonnet-4-20250514": { variants: { high: { extraOption: "custom-value" } } },
1439
+ },
1440
+ },
1441
+ },
1442
+ },
1443
+ },
1444
+ )
1445
+
1446
+ it.instance(
1447
+ "variants filtered in second pass for database models",
1448
+ Effect.gen(function* () {
1449
+ yield* set("OPENAI_API_KEY", "test-api-key")
1450
+ const providers = yield* list
1451
+ const model = providers[ProviderV2.ID.openai].models["gpt-5"]
1452
+ expect(model.variants).toBeDefined()
1453
+ expect(model.variants!["high"]).toBeUndefined()
1454
+ // Other variants should still exist
1455
+ expect(model.variants!["medium"]).toBeDefined()
1456
+ }),
1457
+ {
1458
+ config: {
1459
+ provider: { openai: { models: { "gpt-5": { variants: { high: { disabled: true } } } } } },
1460
+ },
1461
+ },
1462
+ )
1463
+
1464
+ it.instance(
1465
+ "custom model with variants enabled and disabled",
1466
+ Effect.gen(function* () {
1467
+ const providers = yield* list
1468
+ const model = providers[ProviderV2.ID.make("custom-reasoning")].models["reasoning-model"]
1469
+ expect(model.variants).toBeDefined()
1470
+ // Enabled variants should exist
1471
+ expect(model.variants!["low"]).toBeDefined()
1472
+ expect(model.variants!["low"].reasoningEffort).toBe("low")
1473
+ expect(model.variants!["medium"]).toBeDefined()
1474
+ expect(model.variants!["medium"].reasoningEffort).toBe("medium")
1475
+ expect(model.variants!["custom"]).toBeDefined()
1476
+ expect(model.variants!["custom"].reasoningEffort).toBe("custom")
1477
+ expect(model.variants!["custom"].budgetTokens).toBe(5000)
1478
+ // Disabled variant should not exist
1479
+ expect(model.variants!["high"]).toBeUndefined()
1480
+ // disabled key should be stripped from all variants
1481
+ expect(model.variants!["low"].disabled).toBeUndefined()
1482
+ expect(model.variants!["medium"].disabled).toBeUndefined()
1483
+ expect(model.variants!["custom"].disabled).toBeUndefined()
1484
+ }),
1485
+ {
1486
+ config: {
1487
+ provider: {
1488
+ "custom-reasoning": {
1489
+ name: "Custom Reasoning Provider",
1490
+ npm: "@ai-sdk/openai-compatible",
1491
+ env: [],
1492
+ models: {
1493
+ "reasoning-model": {
1494
+ name: "Reasoning Model",
1495
+ tool_call: true,
1496
+ reasoning: true,
1497
+ limit: { context: 128000, output: 16000 },
1498
+ variants: {
1499
+ low: { reasoningEffort: "low" },
1500
+ medium: { reasoningEffort: "medium" },
1501
+ high: { reasoningEffort: "high", disabled: true },
1502
+ custom: { reasoningEffort: "custom", budgetTokens: 5000 },
1503
+ },
1504
+ },
1505
+ },
1506
+ options: { apiKey: "test-key" },
1507
+ },
1508
+ },
1509
+ },
1510
+ },
1511
+ )
1512
+
1513
+ it.instance(
1514
+ "Google Vertex: retains baseURL for custom proxy",
1515
+ Effect.gen(function* () {
1516
+ yield* set("GOOGLE_APPLICATION_CREDENTIALS", "test-creds")
1517
+ const providers = yield* list
1518
+ expect(providers[ProviderV2.ID.make("vertex-proxy")]).toBeDefined()
1519
+ expect(providers[ProviderV2.ID.make("vertex-proxy")].options.baseURL).toBe("https://my-proxy.com/v1")
1520
+ }),
1521
+ {
1522
+ config: {
1523
+ provider: {
1524
+ "vertex-proxy": {
1525
+ name: "Vertex Proxy",
1526
+ npm: "@ai-sdk/google-vertex",
1527
+ api: "https://my-proxy.com/v1",
1528
+ env: ["GOOGLE_APPLICATION_CREDENTIALS"],
1529
+ models: { "gemini-pro": { name: "Gemini Pro", tool_call: true } },
1530
+ options: {
1531
+ project: "test-project",
1532
+ location: "us-central1",
1533
+ baseURL: "https://my-proxy.com/v1",
1534
+ },
1535
+ },
1536
+ },
1537
+ },
1538
+ },
1539
+ )
1540
+
1541
+ it.instance(
1542
+ "Google Vertex: supports OpenAI compatible models",
1543
+ Effect.gen(function* () {
1544
+ yield* set("GOOGLE_APPLICATION_CREDENTIALS", "test-creds")
1545
+ const providers = yield* list
1546
+ const model = providers[ProviderV2.ID.make("vertex-openai")].models["gpt-4"]
1547
+ expect(model).toBeDefined()
1548
+ expect(model.api.npm).toBe("@ai-sdk/openai-compatible")
1549
+ }),
1550
+ {
1551
+ config: {
1552
+ provider: {
1553
+ "vertex-openai": {
1554
+ name: "Vertex OpenAI",
1555
+ npm: "@ai-sdk/google-vertex",
1556
+ env: ["GOOGLE_APPLICATION_CREDENTIALS"],
1557
+ models: {
1558
+ "gpt-4": {
1559
+ name: "GPT-4",
1560
+ provider: { npm: "@ai-sdk/openai-compatible", api: "https://api.openai.com/v1" },
1561
+ },
1562
+ },
1563
+ options: { project: "test-project", location: "us-central1" },
1564
+ },
1565
+ },
1566
+ },
1567
+ },
1568
+ )
1569
+
1570
+ it.instance("Google Vertex: uses REP endpoint for Claude continental multi-regions", () =>
1571
+ Effect.gen(function* () {
1572
+ yield* set("GOOGLE_CLOUD_PROJECT", "test-project")
1573
+ yield* set("VERTEX_LOCATION", "eu")
1574
+ const provider = yield* Provider.Service
1575
+ const model = yield* provider.getModel(
1576
+ ProviderV2.ID.make("google-vertex"),
1577
+ ModelV2.ID.make("claude-sonnet-4-6@default"),
1578
+ )
1579
+ const language = yield* provider.getLanguage(model)
1580
+ expect(languageBaseURL(language)).toBe(
1581
+ "https://aiplatform.eu.rep.googleapis.com/v1/projects/test-project/locations/eu/publishers/anthropic/models",
1582
+ )
1583
+ }),
1584
+ )
1585
+
1586
+ it.instance("Google Vertex Anthropic: uses REP endpoint for continental multi-regions", () =>
1587
+ Effect.gen(function* () {
1588
+ yield* set("GOOGLE_CLOUD_PROJECT", "test-project")
1589
+ yield* set("VERTEX_LOCATION", "us")
1590
+ const provider = yield* Provider.Service
1591
+ const model = yield* provider.getModel(
1592
+ ProviderV2.ID.make("google-vertex-anthropic"),
1593
+ ModelV2.ID.make("claude-sonnet-4-6@default"),
1594
+ )
1595
+ const language = yield* provider.getLanguage(model)
1596
+ expect(languageBaseURL(language)).toBe(
1597
+ "https://aiplatform.us.rep.googleapis.com/v1/projects/test-project/locations/us/publishers/anthropic/models",
1598
+ )
1599
+ }),
1600
+ )
1601
+
1602
+ it.instance("Google Vertex: keeps regional Claude endpoints unchanged", () =>
1603
+ Effect.gen(function* () {
1604
+ yield* set("GOOGLE_CLOUD_PROJECT", "test-project")
1605
+ yield* set("VERTEX_LOCATION", "europe-west1")
1606
+ const provider = yield* Provider.Service
1607
+ const model = yield* provider.getModel(
1608
+ ProviderV2.ID.make("google-vertex"),
1609
+ ModelV2.ID.make("claude-sonnet-4-6@default"),
1610
+ )
1611
+ const language = yield* provider.getLanguage(model)
1612
+ expect(languageBaseURL(language)).toBe(
1613
+ "https://europe-west1-aiplatform.googleapis.com/v1/projects/test-project/locations/europe-west1/publishers/anthropic/models",
1614
+ )
1615
+ }),
1616
+ )
1617
+
1618
+ it.instance("cloudflare-ai-gateway loads with env variables", () =>
1619
+ Effect.gen(function* () {
1620
+ yield* set("CLOUDFLARE_ACCOUNT_ID", "test-account")
1621
+ yield* set("CLOUDFLARE_GATEWAY_ID", "test-gateway")
1622
+ yield* set("CLOUDFLARE_API_TOKEN", "test-token")
1623
+ const providers = yield* list
1624
+ expect(providers[ProviderV2.ID.make("cloudflare-ai-gateway")]).toBeDefined()
1625
+ }),
1626
+ )
1627
+
1628
+ it.instance(
1629
+ "cloudflare-ai-gateway forwards config metadata options",
1630
+ Effect.gen(function* () {
1631
+ yield* set("CLOUDFLARE_ACCOUNT_ID", "test-account")
1632
+ yield* set("CLOUDFLARE_GATEWAY_ID", "test-gateway")
1633
+ yield* set("CLOUDFLARE_API_TOKEN", "test-token")
1634
+ const providers = yield* list
1635
+ expect(providers[ProviderV2.ID.make("cloudflare-ai-gateway")]).toBeDefined()
1636
+ expect(providers[ProviderV2.ID.make("cloudflare-ai-gateway")].options.metadata).toEqual({
1637
+ invoked_by: "test",
1638
+ project: "octo",
1639
+ })
1640
+ }),
1641
+ {
1642
+ config: {
1643
+ provider: { "cloudflare-ai-gateway": { options: { metadata: { invoked_by: "test", project: "octo" } } } },
1644
+ },
1645
+ },
1646
+ )
1647
+
1648
+ // Tests that need plugin file setup or multi-instance flows fall back to a
1649
+ // scoped tmpdir + provideInstance pattern via it.effect.
1650
+
1651
+ const provideMultiInstance = <A, E, R>(eff: Effect.Effect<A, E, R>) =>
1652
+ eff.pipe(Effect.provide(InstanceLayer.layer), Effect.provide(CrossSpawnSpawner.defaultLayer))
1653
+
1654
+ it.effect("plugin config providers persist after instance dispose", () =>
1655
+ Effect.gen(function* () {
1656
+ const dir = yield* tmpdirScoped()
1657
+ const configDir = path.join(dir, ".octocode")
1658
+ const root = path.join(configDir, "plugin")
1659
+ yield* Effect.promise(() => mkdir(root, { recursive: true }))
1660
+ yield* Effect.promise(() => markPluginDependenciesReady(configDir))
1661
+ yield* Effect.promise(() => markPluginDependenciesReady(Global.Path.config))
1662
+ yield* Effect.promise(() =>
1663
+ Bun.write(
1664
+ path.join(root, "demo-provider.ts"),
1665
+ [
1666
+ "export default {",
1667
+ ' id: "demo.plugin-provider",',
1668
+ " server: async () => ({",
1669
+ " async config(cfg) {",
1670
+ " cfg.provider ??= {}",
1671
+ " cfg.provider.demo = {",
1672
+ ' name: "Demo Provider",',
1673
+ ' npm: "@ai-sdk/openai-compatible",',
1674
+ ' api: "https://example.com/v1",',
1675
+ " models: {",
1676
+ " chat: {",
1677
+ ' name: "Demo Chat",',
1678
+ " tool_call: true,",
1679
+ " limit: { context: 128000, output: 4096 },",
1680
+ " },",
1681
+ " },",
1682
+ " }",
1683
+ " },",
1684
+ " }),",
1685
+ "}",
1686
+ "",
1687
+ ].join("\n"),
1688
+ ),
1689
+ )
1690
+
1691
+ const loadAndList = Effect.gen(function* () {
1692
+ const plugin = yield* Plugin.Service
1693
+ const provider = yield* Provider.Service
1694
+ yield* plugin.init()
1695
+ return yield* provider.list()
1696
+ }).pipe(provideInstanceEffect(dir))
1697
+
1698
+ const first = yield* loadAndList
1699
+ expect(first[ProviderV2.ID.make("demo")]).toBeDefined()
1700
+ expect(first[ProviderV2.ID.make("demo")].models[ModelV2.ID.make("chat")]).toBeDefined()
1701
+
1702
+ yield* Effect.promise(() => disposeAllInstances())
1703
+
1704
+ const second = yield* loadAndList
1705
+ expect(second[ProviderV2.ID.make("demo")]).toBeDefined()
1706
+ expect(second[ProviderV2.ID.make("demo")].models[ModelV2.ID.make("chat")]).toBeDefined()
1707
+ }).pipe(provideMultiInstance),
1708
+ )
1709
+
1710
+ it.instance(
1711
+ "plugin config enabled and disabled providers are honored",
1712
+ Effect.gen(function* () {
1713
+ const instance = yield* TestInstance
1714
+ const configDir = path.join(instance.directory, ".octocode")
1715
+ const root = path.join(configDir, "plugin")
1716
+ yield* Effect.promise(() => mkdir(root, { recursive: true }))
1717
+ yield* Effect.promise(() => markPluginDependenciesReady(configDir))
1718
+ yield* Effect.promise(() =>
1719
+ Bun.write(
1720
+ path.join(root, "provider-filter.ts"),
1721
+ [
1722
+ "export default {",
1723
+ ' id: "demo.provider-filter",',
1724
+ " server: async () => ({",
1725
+ " async config(cfg) {",
1726
+ ' cfg.enabled_providers = ["anthropic", "openai"]',
1727
+ ' cfg.disabled_providers = ["openai"]',
1728
+ " },",
1729
+ " }),",
1730
+ "}",
1731
+ "",
1732
+ ].join("\n"),
1733
+ ),
1734
+ )
1735
+
1736
+ yield* set("ANTHROPIC_API_KEY", "test-anthropic-key")
1737
+ yield* set("OPENAI_API_KEY", "test-openai-key")
1738
+ const providers = yield* list
1739
+ expect(providers[ProviderV2.ID.anthropic]).toBeDefined()
1740
+ expect(providers[ProviderV2.ID.openai]).toBeUndefined()
1741
+ }),
1742
+ )
1743
+
1744
+ it.effect.skip("octocode loader keeps paid models when config apiKey is present", () =>
1745
+ Effect.gen(function* () {
1746
+ const noneDir = yield* tmpdirScoped()
1747
+ const keyedDir = yield* tmpdirScoped({
1748
+ config: { provider: { octocode: { options: { apiKey: "test-key" } } } },
1749
+ })
1750
+
1751
+ const listIn = (directory: string) =>
1752
+ Provider.use
1753
+ .list()
1754
+ .pipe(provideInstanceEffect(directory))
1755
+ .pipe(Effect.provide(InstanceLayer.layer), Effect.provide(CrossSpawnSpawner.defaultLayer))
1756
+
1757
+ const none = paid(yield* listIn(noneDir))
1758
+ const keyedCount = paid(yield* listIn(keyedDir))
1759
+
1760
+ expect(none).toBe(0)
1761
+ expect(keyedCount).toBeGreaterThan(0)
1762
+ }).pipe(provideMultiInstance),
1763
+ )
1764
+
1765
+ it.effect.skip("octocode loader keeps paid models when auth exists", () =>
1766
+ Effect.gen(function* () {
1767
+ const noneDir = yield* tmpdirScoped()
1768
+ const keyedDir = yield* tmpdirScoped()
1769
+
1770
+ const listIn = (directory: string) =>
1771
+ Provider.use
1772
+ .list()
1773
+ .pipe(provideInstanceEffect(directory))
1774
+ .pipe(Effect.provide(InstanceLayer.layer), Effect.provide(CrossSpawnSpawner.defaultLayer))
1775
+
1776
+ const none = paid(yield* listIn(noneDir))
1777
+
1778
+ const authPath = path.join(Global.Path.data, "auth.json")
1779
+ const original = yield* Effect.promise(() => Filesystem.readText(authPath).catch(() => undefined))
1780
+
1781
+ yield* Effect.acquireRelease(
1782
+ Effect.promise(() => Filesystem.write(authPath, JSON.stringify({ octocode: { type: "api", key: "test-key" } }))),
1783
+ () =>
1784
+ Effect.promise(async () => {
1785
+ if (original !== undefined) await Filesystem.write(authPath, original)
1786
+ else await unlink(authPath).catch(() => undefined)
1787
+ }),
1788
+ )
1789
+
1790
+ const keyedCount = paid(yield* listIn(keyedDir))
1791
+
1792
+ expect(none).toBe(0)
1793
+ expect(keyedCount).toBeGreaterThan(0)
1794
+ }).pipe(provideMultiInstance),
1795
+ )