woozlit 2.3.0 → 2.3.2

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 (1118) hide show
  1. package/{src/skill/compose/LICENSE-superpowers → LICENSE} +21 -26
  2. package/package.json +18 -191
  3. package/AGENTS.md +0 -134
  4. package/Dockerfile +0 -18
  5. package/README.md +0 -15
  6. package/bunfig.toml +0 -7
  7. package/drizzle.config.ts +0 -10
  8. package/git +0 -0
  9. package/migration/20260127222353_familiar_lady_ursula/migration.sql +0 -90
  10. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +0 -796
  11. package/migration/20260211171708_add_project_commands/migration.sql +0 -1
  12. package/migration/20260211171708_add_project_commands/snapshot.json +0 -806
  13. package/migration/20260213144116_wakeful_the_professor/migration.sql +0 -11
  14. package/migration/20260213144116_wakeful_the_professor/snapshot.json +0 -897
  15. package/migration/20260225215848_workspace/migration.sql +0 -7
  16. package/migration/20260225215848_workspace/snapshot.json +0 -959
  17. package/migration/20260227213759_add_session_workspace_id/migration.sql +0 -2
  18. package/migration/20260227213759_add_session_workspace_id/snapshot.json +0 -983
  19. package/migration/20260228203230_blue_harpoon/migration.sql +0 -17
  20. package/migration/20260228203230_blue_harpoon/snapshot.json +0 -1102
  21. package/migration/20260303231226_add_workspace_fields/migration.sql +0 -5
  22. package/migration/20260303231226_add_workspace_fields/snapshot.json +0 -1013
  23. package/migration/20260309230000_move_org_to_state/migration.sql +0 -3
  24. package/migration/20260309230000_move_org_to_state/snapshot.json +0 -1156
  25. package/migration/20260312043431_session_message_cursor/migration.sql +0 -4
  26. package/migration/20260312043431_session_message_cursor/snapshot.json +0 -1168
  27. package/migration/20260323234822_events/migration.sql +0 -13
  28. package/migration/20260323234822_events/snapshot.json +0 -1271
  29. package/migration/20260410174513_workspace-name/migration.sql +0 -16
  30. package/migration/20260410174513_workspace-name/snapshot.json +0 -1271
  31. package/migration/20260413175956_chief_energizer/migration.sql +0 -13
  32. package/migration/20260413175956_chief_energizer/snapshot.json +0 -1399
  33. package/migration/20260422160000_context_inheritance/migration.sql +0 -3
  34. package/migration/20260422170000_task_registry/migration.sql +0 -18
  35. package/migration/20260423145421_remove_session_entry/migration.sql +0 -4
  36. package/migration/20260515000000_actor_rename/migration.sql +0 -7
  37. package/migration/20260515010000_memory_fts/migration.sql +0 -33
  38. package/migration/20260515020000_user_task/migration.sql +0 -29
  39. package/migration/20260519000000_last_checkpoint_message_id/migration.sql +0 -1
  40. package/migration/20260521000000_message_agent_id/migration.sql +0 -2
  41. package/migration/20260521000100_actor_registry_v6/migration.sql +0 -25
  42. package/migration/20260521010000_memory_fts_v6/migration.sql +0 -33
  43. package/migration/20260521020000_memory_fts_triggers/migration.sql +0 -17
  44. package/migration/20260526000000_agent_id_main/migration.sql +0 -14
  45. package/migration/20260527000000_actor_lifecycle/migration.sql +0 -8
  46. package/migration/20260527000100_inbox/migration.sql +0 -12
  47. package/migration/20260529000000_task_todo_redesign/migration.sql +0 -16
  48. package/migration/20260603000000_task_in_progress_owner/migration.sql +0 -1
  49. package/migration/20260603000000_workflow_run/migration.sql +0 -17
  50. package/migration/20260604000000_workflow_script_sha/migration.sql +0 -1
  51. package/migration/20260608000000_claude_import/migration.sql +0 -7
  52. package/migration/20260608010000_claude_import_message_ids/migration.sql +0 -1
  53. package/migration/20260609000000_history_fts/migration.sql +0 -29
  54. package/migration/20260609230000_workflow_agent_timeout/migration.sql +0 -1
  55. package/parsers-config.ts +0 -290
  56. package/script/build.ts +0 -269
  57. package/script/check-migrations.ts +0 -16
  58. package/script/fix-node-pty.ts +0 -28
  59. package/script/generate.ts +0 -23
  60. package/script/publish.ts +0 -75
  61. package/script/run-workspace-server +0 -106
  62. package/script/schema.ts +0 -63
  63. package/script/time.ts +0 -6
  64. package/script/trace-imports.ts +0 -153
  65. package/script/upgrade-opentui.ts +0 -64
  66. package/src/account/account.sql.ts +0 -39
  67. package/src/account/account.ts +0 -456
  68. package/src/account/repo.ts +0 -166
  69. package/src/account/schema.ts +0 -99
  70. package/src/account/url.ts +0 -8
  71. package/src/acp/README.md +0 -174
  72. package/src/acp/agent.ts +0 -1783
  73. package/src/acp/session.ts +0 -116
  74. package/src/acp/types.ts +0 -24
  75. package/src/actor/actor.sql.ts +0 -38
  76. package/src/actor/events.ts +0 -67
  77. package/src/actor/index.ts +0 -2
  78. package/src/actor/registry.ts +0 -412
  79. package/src/actor/return-header.ts +0 -24
  80. package/src/actor/schema.ts +0 -47
  81. package/src/actor/spawn-ref.ts +0 -16
  82. package/src/actor/spawn.ts +0 -741
  83. package/src/actor/turn.ts +0 -49
  84. package/src/actor/waiter.ts +0 -166
  85. package/src/agent/agent.ts +0 -554
  86. package/src/agent/config.ts +0 -5
  87. package/src/agent/generate.txt +0 -75
  88. package/src/agent/prompt/checkpoint-writer.txt +0 -167
  89. package/src/agent/prompt/compaction.txt +0 -9
  90. package/src/agent/prompt/distill.txt +0 -199
  91. package/src/agent/prompt/dream.txt +0 -155
  92. package/src/agent/prompt/explore.txt +0 -18
  93. package/src/agent/prompt/summary.txt +0 -11
  94. package/src/agent/prompt/title.txt +0 -44
  95. package/src/audio.d.ts +0 -9
  96. package/src/auth/index.ts +0 -106
  97. package/src/bus/bus-event.ts +0 -33
  98. package/src/bus/global.ts +0 -12
  99. package/src/bus/index.ts +0 -193
  100. package/src/cli/bootstrap.ts +0 -33
  101. package/src/cli/cmd/account.ts +0 -258
  102. package/src/cli/cmd/acp.ts +0 -70
  103. package/src/cli/cmd/agent.ts +0 -248
  104. package/src/cli/cmd/cmd.ts +0 -7
  105. package/src/cli/cmd/db.ts +0 -120
  106. package/src/cli/cmd/debug/agent.ts +0 -192
  107. package/src/cli/cmd/debug/config.ts +0 -17
  108. package/src/cli/cmd/debug/file.ts +0 -100
  109. package/src/cli/cmd/debug/index.ts +0 -48
  110. package/src/cli/cmd/debug/lsp.ts +0 -61
  111. package/src/cli/cmd/debug/ripgrep.ts +0 -105
  112. package/src/cli/cmd/debug/scrap.ts +0 -16
  113. package/src/cli/cmd/debug/skill.ts +0 -23
  114. package/src/cli/cmd/debug/snapshot.ts +0 -53
  115. package/src/cli/cmd/export.ts +0 -306
  116. package/src/cli/cmd/generate.ts +0 -50
  117. package/src/cli/cmd/github.ts +0 -1647
  118. package/src/cli/cmd/import.ts +0 -208
  119. package/src/cli/cmd/mcp.ts +0 -812
  120. package/src/cli/cmd/models.ts +0 -88
  121. package/src/cli/cmd/plug.ts +0 -233
  122. package/src/cli/cmd/pr.ts +0 -138
  123. package/src/cli/cmd/providers.ts +0 -689
  124. package/src/cli/cmd/run-completion.ts +0 -77
  125. package/src/cli/cmd/run.ts +0 -694
  126. package/src/cli/cmd/serve.ts +0 -21
  127. package/src/cli/cmd/session.ts +0 -181
  128. package/src/cli/cmd/stats.ts +0 -413
  129. package/src/cli/cmd/tui/app.tsx +0 -1349
  130. package/src/cli/cmd/tui/asset/TEN_VAD_LICENSE +0 -12
  131. package/src/cli/cmd/tui/asset/charge.wav +0 -0
  132. package/src/cli/cmd/tui/asset/pulse-a.wav +0 -0
  133. package/src/cli/cmd/tui/asset/pulse-b.wav +0 -0
  134. package/src/cli/cmd/tui/asset/pulse-c.wav +0 -0
  135. package/src/cli/cmd/tui/asset/ten_vad.wasm +0 -0
  136. package/src/cli/cmd/tui/asset/ten_vad_loader.js +0 -30
  137. package/src/cli/cmd/tui/attach.ts +0 -83
  138. package/src/cli/cmd/tui/component/background-image.tsx +0 -150
  139. package/src/cli/cmd/tui/component/bg-pulse.tsx +0 -130
  140. package/src/cli/cmd/tui/component/border.tsx +0 -21
  141. package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
  142. package/src/cli/cmd/tui/component/dialog-command.tsx +0 -208
  143. package/src/cli/cmd/tui/component/dialog-console-org.tsx +0 -103
  144. package/src/cli/cmd/tui/component/dialog-go-upsell.tsx +0 -157
  145. package/src/cli/cmd/tui/component/dialog-image-list.tsx +0 -111
  146. package/src/cli/cmd/tui/component/dialog-logo-design.tsx +0 -37
  147. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  148. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -260
  149. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -454
  150. package/src/cli/cmd/tui/component/dialog-session-delete-failed.tsx +0 -101
  151. package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -269
  152. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
  153. package/src/cli/cmd/tui/component/dialog-skill.tsx +0 -42
  154. package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -87
  155. package/src/cli/cmd/tui/component/dialog-status.tsx +0 -170
  156. package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
  157. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
  158. package/src/cli/cmd/tui/component/dialog-variant.tsx +0 -39
  159. package/src/cli/cmd/tui/component/dialog-woozlit-login.tsx +0 -206
  160. package/src/cli/cmd/tui/component/dialog-workflows.tsx +0 -62
  161. package/src/cli/cmd/tui/component/dialog-workspace-create.tsx +0 -289
  162. package/src/cli/cmd/tui/component/dialog-workspace-unavailable.tsx +0 -81
  163. package/src/cli/cmd/tui/component/dialog-worktree.tsx +0 -90
  164. package/src/cli/cmd/tui/component/error-component.tsx +0 -92
  165. package/src/cli/cmd/tui/component/logo.tsx +0 -961
  166. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +0 -14
  167. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -684
  168. package/src/cli/cmd/tui/component/prompt/cwd.ts +0 -0
  169. package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -90
  170. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
  171. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1829
  172. package/src/cli/cmd/tui/component/prompt/part.ts +0 -16
  173. package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
  174. package/src/cli/cmd/tui/component/spinner.tsx +0 -24
  175. package/src/cli/cmd/tui/component/starry-background.tsx +0 -305
  176. package/src/cli/cmd/tui/component/startup-loading.tsx +0 -67
  177. package/src/cli/cmd/tui/component/task-item.tsx +0 -63
  178. package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
  179. package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
  180. package/src/cli/cmd/tui/config/cwd.ts +0 -5
  181. package/src/cli/cmd/tui/config/tui-migrate.ts +0 -151
  182. package/src/cli/cmd/tui/config/tui-schema.ts +0 -38
  183. package/src/cli/cmd/tui/config/tui.ts +0 -219
  184. package/src/cli/cmd/tui/context/args.tsx +0 -16
  185. package/src/cli/cmd/tui/context/directory.ts +0 -15
  186. package/src/cli/cmd/tui/context/event.ts +0 -45
  187. package/src/cli/cmd/tui/context/exit.tsx +0 -65
  188. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  189. package/src/cli/cmd/tui/context/keybind.tsx +0 -105
  190. package/src/cli/cmd/tui/context/kv.tsx +0 -76
  191. package/src/cli/cmd/tui/context/language.tsx +0 -91
  192. package/src/cli/cmd/tui/context/local.tsx +0 -455
  193. package/src/cli/cmd/tui/context/plugin-keybinds.ts +0 -41
  194. package/src/cli/cmd/tui/context/project.tsx +0 -109
  195. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  196. package/src/cli/cmd/tui/context/route.tsx +0 -61
  197. package/src/cli/cmd/tui/context/sdk.tsx +0 -150
  198. package/src/cli/cmd/tui/context/sync.tsx +0 -828
  199. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  200. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  201. package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
  202. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -230
  203. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -230
  204. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  205. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -225
  206. package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
  207. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  208. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  209. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  210. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  211. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -242
  212. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  213. package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -232
  214. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  215. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  216. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  217. package/src/cli/cmd/tui/context/theme/mimocode.json +0 -245
  218. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  219. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  220. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  221. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  222. package/src/cli/cmd/tui/context/theme/orng.json +0 -247
  223. package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
  224. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  225. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  226. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  227. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  228. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  229. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  230. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  231. package/src/cli/cmd/tui/context/theme/woozlit.json +0 -245
  232. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  233. package/src/cli/cmd/tui/context/theme.tsx +0 -1298
  234. package/src/cli/cmd/tui/context/thinking.ts +0 -48
  235. package/src/cli/cmd/tui/context/tui-config.tsx +0 -9
  236. package/src/cli/cmd/tui/event.ts +0 -56
  237. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +0 -93
  238. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +0 -193
  239. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +0 -54
  240. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +0 -169
  241. package/src/cli/cmd/tui/feature-plugins/sidebar/cwd.tsx +0 -45
  242. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +0 -62
  243. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +0 -93
  244. package/src/cli/cmd/tui/feature-plugins/sidebar/goal.tsx +0 -84
  245. package/src/cli/cmd/tui/feature-plugins/sidebar/instructions.tsx +0 -54
  246. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +0 -66
  247. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +0 -98
  248. package/src/cli/cmd/tui/feature-plugins/sidebar/task.tsx +0 -95
  249. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +0 -51
  250. package/src/cli/cmd/tui/feature-plugins/sidebar/tps.ts +0 -31
  251. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +0 -274
  252. package/src/cli/cmd/tui/i18n/en.ts +0 -397
  253. package/src/cli/cmd/tui/i18n/es.ts +0 -433
  254. package/src/cli/cmd/tui/i18n/fr.ts +0 -440
  255. package/src/cli/cmd/tui/i18n/ja.ts +0 -392
  256. package/src/cli/cmd/tui/i18n/locales.ts +0 -82
  257. package/src/cli/cmd/tui/i18n/ru.ts +0 -452
  258. package/src/cli/cmd/tui/i18n/zh.ts +0 -390
  259. package/src/cli/cmd/tui/i18n/zht.ts +0 -360
  260. package/src/cli/cmd/tui/layer.ts +0 -6
  261. package/src/cli/cmd/tui/plugin/api.tsx +0 -405
  262. package/src/cli/cmd/tui/plugin/index.ts +0 -3
  263. package/src/cli/cmd/tui/plugin/internal.ts +0 -35
  264. package/src/cli/cmd/tui/plugin/runtime.ts +0 -1030
  265. package/src/cli/cmd/tui/plugin/slots.tsx +0 -60
  266. package/src/cli/cmd/tui/routes/home.tsx +0 -172
  267. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -76
  268. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -116
  269. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -47
  270. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
  271. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -91
  272. package/src/cli/cmd/tui/routes/session/index.tsx +0 -2532
  273. package/src/cli/cmd/tui/routes/session/permission.tsx +0 -691
  274. package/src/cli/cmd/tui/routes/session/question.tsx +0 -488
  275. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -97
  276. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +0 -142
  277. package/src/cli/cmd/tui/thread.ts +0 -246
  278. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -61
  279. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -95
  280. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -223
  281. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -42
  282. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -123
  283. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -452
  284. package/src/cli/cmd/tui/ui/dialog.tsx +0 -207
  285. package/src/cli/cmd/tui/ui/link.tsx +0 -28
  286. package/src/cli/cmd/tui/ui/spinner.ts +0 -378
  287. package/src/cli/cmd/tui/ui/toast.tsx +0 -102
  288. package/src/cli/cmd/tui/util/clipboard.ts +0 -203
  289. package/src/cli/cmd/tui/util/editor.ts +0 -35
  290. package/src/cli/cmd/tui/util/image-protocol.ts +0 -35
  291. package/src/cli/cmd/tui/util/index.ts +0 -6
  292. package/src/cli/cmd/tui/util/model.ts +0 -23
  293. package/src/cli/cmd/tui/util/provider-origin.ts +0 -7
  294. package/src/cli/cmd/tui/util/revert-diff.ts +0 -18
  295. package/src/cli/cmd/tui/util/scroll.ts +0 -23
  296. package/src/cli/cmd/tui/util/selection.ts +0 -23
  297. package/src/cli/cmd/tui/util/signal.ts +0 -41
  298. package/src/cli/cmd/tui/util/sound.ts +0 -154
  299. package/src/cli/cmd/tui/util/system-locale.ts +0 -209
  300. package/src/cli/cmd/tui/util/terminal.ts +0 -110
  301. package/src/cli/cmd/tui/util/transcript.ts +0 -112
  302. package/src/cli/cmd/tui/util/vad.ts +0 -229
  303. package/src/cli/cmd/tui/util/voice.ts +0 -360
  304. package/src/cli/cmd/tui/win32.ts +0 -130
  305. package/src/cli/cmd/tui/worker.ts +0 -104
  306. package/src/cli/cmd/uninstall.ts +0 -351
  307. package/src/cli/cmd/upgrade.ts +0 -79
  308. package/src/cli/cmd/web.ts +0 -81
  309. package/src/cli/effect/prompt.ts +0 -25
  310. package/src/cli/error.ts +0 -82
  311. package/src/cli/heap.ts +0 -59
  312. package/src/cli/i18n.ts +0 -15
  313. package/src/cli/logo.ts +0 -59
  314. package/src/cli/network.ts +0 -62
  315. package/src/cli/ui.ts +0 -133
  316. package/src/cli/upgrade.ts +0 -41
  317. package/src/command/index.ts +0 -276
  318. package/src/command/template/initialize.txt +0 -66
  319. package/src/command/template/review.txt +0 -101
  320. package/src/config/agent.ts +0 -197
  321. package/src/config/command.ts +0 -69
  322. package/src/config/config.ts +0 -1024
  323. package/src/config/console-state.ts +0 -16
  324. package/src/config/entry-name.ts +0 -16
  325. package/src/config/error.ts +0 -21
  326. package/src/config/formatter.ts +0 -17
  327. package/src/config/history.ts +0 -21
  328. package/src/config/index.ts +0 -16
  329. package/src/config/keybinds.ts +0 -127
  330. package/src/config/layout.ts +0 -10
  331. package/src/config/lsp.ts +0 -45
  332. package/src/config/managed.ts +0 -70
  333. package/src/config/markdown.ts +0 -97
  334. package/src/config/mcp.ts +0 -172
  335. package/src/config/model-id.ts +0 -14
  336. package/src/config/parse.ts +0 -44
  337. package/src/config/paths.ts +0 -73
  338. package/src/config/permission.ts +0 -76
  339. package/src/config/plugin.ts +0 -88
  340. package/src/config/provider.ts +0 -118
  341. package/src/config/server.ts +0 -20
  342. package/src/config/skills.ts +0 -16
  343. package/src/config/variable.ts +0 -90
  344. package/src/control-plane/adaptors/index.ts +0 -52
  345. package/src/control-plane/adaptors/worktree.ts +0 -47
  346. package/src/control-plane/dev/debug-workspace-plugin.ts +0 -73
  347. package/src/control-plane/schema.ts +0 -19
  348. package/src/control-plane/sse.ts +0 -66
  349. package/src/control-plane/types.ts +0 -34
  350. package/src/control-plane/util.ts +0 -37
  351. package/src/control-plane/workspace-context.ts +0 -26
  352. package/src/control-plane/workspace.sql.ts +0 -17
  353. package/src/control-plane/workspace.ts +0 -615
  354. package/src/effect/app-runtime.ts +0 -146
  355. package/src/effect/bootstrap-runtime.ts +0 -33
  356. package/src/effect/bridge.ts +0 -48
  357. package/src/effect/cross-spawn-spawner.ts +0 -514
  358. package/src/effect/index.ts +0 -5
  359. package/src/effect/instance-ref.ts +0 -11
  360. package/src/effect/instance-registry.ts +0 -12
  361. package/src/effect/instance-state.ts +0 -81
  362. package/src/effect/logger.ts +0 -73
  363. package/src/effect/memo-map.ts +0 -3
  364. package/src/effect/observability.ts +0 -107
  365. package/src/effect/run-service.ts +0 -52
  366. package/src/effect/runner.ts +0 -210
  367. package/src/effect/runtime.ts +0 -19
  368. package/src/env/index.ts +0 -37
  369. package/src/file/ignore.ts +0 -81
  370. package/src/file/index.ts +0 -664
  371. package/src/file/protected.ts +0 -59
  372. package/src/file/ripgrep.ts +0 -485
  373. package/src/file/watcher.ts +0 -163
  374. package/src/flag/flag.ts +0 -172
  375. package/src/format/formatter.ts +0 -403
  376. package/src/format/index.ts +0 -203
  377. package/src/git/index.ts +0 -260
  378. package/src/global/index.ts +0 -54
  379. package/src/history/backfill.ts +0 -162
  380. package/src/history/extract.ts +0 -67
  381. package/src/history/fts-query.ts +0 -15
  382. package/src/history/fts.sql.ts +0 -20
  383. package/src/history/index.ts +0 -10
  384. package/src/history/resolve.ts +0 -65
  385. package/src/history/service.ts +0 -258
  386. package/src/history/writer.ts +0 -112
  387. package/src/id/id.ts +0 -87
  388. package/src/ide/index.ts +0 -73
  389. package/src/inbox/inbox-ref.ts +0 -38
  390. package/src/inbox/inbox.sql.ts +0 -26
  391. package/src/inbox/inbox.ts +0 -223
  392. package/src/inbox/index.ts +0 -3
  393. package/src/inbox/render.ts +0 -40
  394. package/src/index.ts +0 -260
  395. package/src/installation/index.ts +0 -351
  396. package/src/installation/version.ts +0 -8
  397. package/src/lsp/client.ts +0 -249
  398. package/src/lsp/diagnostic.ts +0 -29
  399. package/src/lsp/index.ts +0 -3
  400. package/src/lsp/language.ts +0 -120
  401. package/src/lsp/launch.ts +0 -21
  402. package/src/lsp/lsp.ts +0 -519
  403. package/src/lsp/server.ts +0 -1956
  404. package/src/mcp/auth.ts +0 -144
  405. package/src/mcp/index.ts +0 -944
  406. package/src/mcp/oauth-callback.ts +0 -232
  407. package/src/mcp/oauth-provider.ts +0 -214
  408. package/src/memory/fts-query.ts +0 -37
  409. package/src/memory/fts.sql.ts +0 -19
  410. package/src/memory/index.ts +0 -1
  411. package/src/memory/paths.ts +0 -116
  412. package/src/memory/reconcile.ts +0 -144
  413. package/src/memory/service.ts +0 -144
  414. package/src/metrics/client.ts +0 -35
  415. package/src/metrics/event.ts +0 -43
  416. package/src/metrics/index.ts +0 -5
  417. package/src/metrics/installation.ts +0 -18
  418. package/src/metrics/subscriber.ts +0 -58
  419. package/src/metrics/util.ts +0 -9
  420. package/src/node.ts +0 -6
  421. package/src/npm/config.ts +0 -0
  422. package/src/npm/index.ts +0 -293
  423. package/src/npmcli-config.d.ts +0 -43
  424. package/src/patch/index.ts +0 -680
  425. package/src/permission/arity.ts +0 -163
  426. package/src/permission/evaluate.ts +0 -15
  427. package/src/permission/index.ts +0 -379
  428. package/src/permission/schema.ts +0 -17
  429. package/src/plugin/checkpoint-splitover.ts +0 -60
  430. package/src/plugin/cloudflare.ts +0 -76
  431. package/src/plugin/codex.ts +0 -607
  432. package/src/plugin/github-copilot/copilot.ts +0 -368
  433. package/src/plugin/github-copilot/models.ts +0 -153
  434. package/src/plugin/index.ts +0 -500
  435. package/src/plugin/install.ts +0 -439
  436. package/src/plugin/loader.ts +0 -216
  437. package/src/plugin/matcher.ts +0 -33
  438. package/src/plugin/meta.ts +0 -188
  439. package/src/plugin/shared.ts +0 -323
  440. package/src/plugin/subagent-progress-checker.ts +0 -147
  441. package/src/plugin/woozlit-free.ts +0 -164
  442. package/src/plugin/woozlit.ts +0 -440
  443. package/src/project/bootstrap.ts +0 -59
  444. package/src/project/index.ts +0 -2
  445. package/src/project/instance.ts +0 -190
  446. package/src/project/project-id.ts +0 -48
  447. package/src/project/project.sql.ts +0 -16
  448. package/src/project/project.ts +0 -501
  449. package/src/project/schema.ts +0 -15
  450. package/src/project/vcs.ts +0 -227
  451. package/src/provider/auth.ts +0 -234
  452. package/src/provider/error.ts +0 -216
  453. package/src/provider/index.ts +0 -5
  454. package/src/provider/models.ts +0 -180
  455. package/src/provider/provider.ts +0 -1788
  456. package/src/provider/schema.ts +0 -36
  457. package/src/provider/sdk/copilot/README.md +0 -5
  458. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +0 -170
  459. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +0 -15
  460. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +0 -19
  461. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +0 -64
  462. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +0 -815
  463. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +0 -28
  464. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +0 -44
  465. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +0 -83
  466. package/src/provider/sdk/copilot/copilot-provider.ts +0 -100
  467. package/src/provider/sdk/copilot/index.ts +0 -2
  468. package/src/provider/sdk/copilot/openai-compatible-error.ts +0 -27
  469. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +0 -335
  470. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +0 -22
  471. package/src/provider/sdk/copilot/responses/openai-config.ts +0 -18
  472. package/src/provider/sdk/copilot/responses/openai-error.ts +0 -22
  473. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +0 -214
  474. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +0 -1770
  475. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +0 -173
  476. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +0 -1
  477. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +0 -87
  478. package/src/provider/sdk/copilot/responses/tool/file-search.ts +0 -127
  479. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +0 -114
  480. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +0 -64
  481. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +0 -103
  482. package/src/provider/sdk/copilot/responses/tool/web-search.ts +0 -102
  483. package/src/provider/transform.ts +0 -1350
  484. package/src/pty/index.ts +0 -364
  485. package/src/pty/pty.bun.ts +0 -26
  486. package/src/pty/pty.node.ts +0 -27
  487. package/src/pty/pty.ts +0 -25
  488. package/src/pty/schema.ts +0 -17
  489. package/src/question/index.ts +0 -252
  490. package/src/question/schema.ts +0 -17
  491. package/src/server/adapter.bun.ts +0 -40
  492. package/src/server/adapter.node.ts +0 -66
  493. package/src/server/adapter.ts +0 -21
  494. package/src/server/error.ts +0 -53
  495. package/src/server/event.ts +0 -7
  496. package/src/server/fence.ts +0 -81
  497. package/src/server/mdns.ts +0 -60
  498. package/src/server/middleware.ts +0 -92
  499. package/src/server/projectors.ts +0 -28
  500. package/src/server/proxy.ts +0 -171
  501. package/src/server/routes/control/index.ts +0 -218
  502. package/src/server/routes/control/workspace.ts +0 -203
  503. package/src/server/routes/global.ts +0 -287
  504. package/src/server/routes/instance/bash-interactive.ts +0 -82
  505. package/src/server/routes/instance/config.ts +0 -89
  506. package/src/server/routes/instance/event.ts +0 -88
  507. package/src/server/routes/instance/experimental.ts +0 -408
  508. package/src/server/routes/instance/file.ts +0 -190
  509. package/src/server/routes/instance/httpapi/config.ts +0 -51
  510. package/src/server/routes/instance/httpapi/permission.ts +0 -72
  511. package/src/server/routes/instance/httpapi/project.ts +0 -62
  512. package/src/server/routes/instance/httpapi/provider.ts +0 -150
  513. package/src/server/routes/instance/httpapi/question.ts +0 -121
  514. package/src/server/routes/instance/httpapi/server.ts +0 -136
  515. package/src/server/routes/instance/index.ts +0 -301
  516. package/src/server/routes/instance/mcp.ts +0 -260
  517. package/src/server/routes/instance/middleware.ts +0 -35
  518. package/src/server/routes/instance/permission.ts +0 -73
  519. package/src/server/routes/instance/project.ts +0 -122
  520. package/src/server/routes/instance/provider.ts +0 -158
  521. package/src/server/routes/instance/pty.ts +0 -247
  522. package/src/server/routes/instance/question.ts +0 -162
  523. package/src/server/routes/instance/session.ts +0 -1296
  524. package/src/server/routes/instance/sync.ts +0 -143
  525. package/src/server/routes/instance/trace.ts +0 -59
  526. package/src/server/routes/instance/tui.ts +0 -384
  527. package/src/server/routes/instance/workflows.ts +0 -72
  528. package/src/server/routes/ui.ts +0 -55
  529. package/src/server/server.ts +0 -136
  530. package/src/server/workspace.ts +0 -122
  531. package/src/session/auto-dream.ts +0 -123
  532. package/src/session/boundary.ts +0 -77
  533. package/src/session/budgeted-read.ts +0 -118
  534. package/src/session/checkpoint-align.ts +0 -29
  535. package/src/session/checkpoint-context.ts +0 -36
  536. package/src/session/checkpoint-paths.ts +0 -86
  537. package/src/session/checkpoint-progress-reconcile.ts +0 -111
  538. package/src/session/checkpoint-retry.ts +0 -192
  539. package/src/session/checkpoint-templates.ts +0 -114
  540. package/src/session/checkpoint-validator.ts +0 -259
  541. package/src/session/checkpoint.ts +0 -1478
  542. package/src/session/classify.ts +0 -92
  543. package/src/session/claude-import.sql.ts +0 -13
  544. package/src/session/claude-import.ts +0 -379
  545. package/src/session/compaction.ts +0 -543
  546. package/src/session/goal.ts +0 -232
  547. package/src/session/index.ts +0 -1
  548. package/src/session/instruction.ts +0 -314
  549. package/src/session/last-message-info.ts +0 -32
  550. package/src/session/llm-request-prefix.ts +0 -82
  551. package/src/session/llm.ts +0 -735
  552. package/src/session/max-mode.ts +0 -397
  553. package/src/session/message-v2.ts +0 -1136
  554. package/src/session/message.ts +0 -191
  555. package/src/session/overflow.ts +0 -53
  556. package/src/session/prefix-capture-ref.ts +0 -48
  557. package/src/session/processor.ts +0 -972
  558. package/src/session/projectors.ts +0 -137
  559. package/src/session/prompt/anthropic.txt +0 -154
  560. package/src/session/prompt/beast.txt +0 -155
  561. package/src/session/prompt/build-switch.txt +0 -5
  562. package/src/session/prompt/codex.txt +0 -79
  563. package/src/session/prompt/compose.txt +0 -115
  564. package/src/session/prompt/copilot-gpt-5.txt +0 -143
  565. package/src/session/prompt/default.txt +0 -151
  566. package/src/session/prompt/gemini.txt +0 -155
  567. package/src/session/prompt/gpt.txt +0 -107
  568. package/src/session/prompt/kimi.txt +0 -95
  569. package/src/session/prompt/max-steps.txt +0 -16
  570. package/src/session/prompt/trinity.txt +0 -97
  571. package/src/session/prompt.ts +0 -3362
  572. package/src/session/prune.ts +0 -481
  573. package/src/session/retry.ts +0 -166
  574. package/src/session/revert.ts +0 -161
  575. package/src/session/run-state.ts +0 -135
  576. package/src/session/schema.ts +0 -36
  577. package/src/session/session.sql.ts +0 -110
  578. package/src/session/session.ts +0 -908
  579. package/src/session/status.ts +0 -89
  580. package/src/session/summary.ts +0 -163
  581. package/src/session/system.ts +0 -86
  582. package/src/session/todo.ts +0 -77
  583. package/src/share/index.ts +0 -2
  584. package/src/share/session.ts +0 -57
  585. package/src/share/share-next.ts +0 -381
  586. package/src/share/share.sql.ts +0 -13
  587. package/src/shell/shell.ts +0 -110
  588. package/src/skill/compose/.bundle/ask/SKILL.md +0 -58
  589. package/src/skill/compose/.bundle/brainstorm/SKILL.md +0 -220
  590. package/src/skill/compose/.bundle/brainstorm/scripts/frame-template.html +0 -214
  591. package/src/skill/compose/.bundle/brainstorm/scripts/helper.js +0 -88
  592. package/src/skill/compose/.bundle/brainstorm/scripts/server.cjs +0 -354
  593. package/src/skill/compose/.bundle/brainstorm/scripts/start-server.sh +0 -148
  594. package/src/skill/compose/.bundle/brainstorm/scripts/stop-server.sh +0 -56
  595. package/src/skill/compose/.bundle/brainstorm/spec-document-reviewer-prompt.md +0 -50
  596. package/src/skill/compose/.bundle/brainstorm/visual-companion.md +0 -287
  597. package/src/skill/compose/.bundle/debug/CREATION-LOG.md +0 -119
  598. package/src/skill/compose/.bundle/debug/SKILL.md +0 -297
  599. package/src/skill/compose/.bundle/debug/condition-based-waiting-example.ts +0 -158
  600. package/src/skill/compose/.bundle/debug/condition-based-waiting.md +0 -115
  601. package/src/skill/compose/.bundle/debug/defense-in-depth.md +0 -122
  602. package/src/skill/compose/.bundle/debug/find-polluter.sh +0 -63
  603. package/src/skill/compose/.bundle/debug/root-cause-tracing.md +0 -169
  604. package/src/skill/compose/.bundle/debug/test-academic.md +0 -14
  605. package/src/skill/compose/.bundle/debug/test-pressure-1.md +0 -58
  606. package/src/skill/compose/.bundle/debug/test-pressure-2.md +0 -68
  607. package/src/skill/compose/.bundle/debug/test-pressure-3.md +0 -69
  608. package/src/skill/compose/.bundle/execute/SKILL.md +0 -71
  609. package/src/skill/compose/.bundle/feedback/SKILL.md +0 -214
  610. package/src/skill/compose/.bundle/merge/SKILL.md +0 -252
  611. package/src/skill/compose/.bundle/new-skill/SKILL.md +0 -656
  612. package/src/skill/compose/.bundle/new-skill/anthropic-best-practices.md +0 -1150
  613. package/src/skill/compose/.bundle/new-skill/examples/CLAUDE_MD_TESTING.md +0 -189
  614. package/src/skill/compose/.bundle/new-skill/graphviz-conventions.dot +0 -172
  615. package/src/skill/compose/.bundle/new-skill/persuasion-principles.md +0 -187
  616. package/src/skill/compose/.bundle/new-skill/render-graphs.js +0 -168
  617. package/src/skill/compose/.bundle/new-skill/testing-skills-with-subagents.md +0 -384
  618. package/src/skill/compose/.bundle/parallel/SKILL.md +0 -182
  619. package/src/skill/compose/.bundle/plan/SKILL.md +0 -161
  620. package/src/skill/compose/.bundle/plan/plan-document-reviewer-prompt.md +0 -50
  621. package/src/skill/compose/.bundle/report/SKILL.md +0 -180
  622. package/src/skill/compose/.bundle/review/SKILL.md +0 -104
  623. package/src/skill/compose/.bundle/review/code-reviewer.md +0 -171
  624. package/src/skill/compose/.bundle/subagent/SKILL.md +0 -344
  625. package/src/skill/compose/.bundle/subagent/code-quality-reviewer-prompt.md +0 -24
  626. package/src/skill/compose/.bundle/subagent/implementer-prompt.md +0 -126
  627. package/src/skill/compose/.bundle/subagent/spec-reviewer-prompt.md +0 -112
  628. package/src/skill/compose/.bundle/tdd/SKILL.md +0 -372
  629. package/src/skill/compose/.bundle/tdd/testing-anti-patterns.md +0 -299
  630. package/src/skill/compose/.bundle/verify/SKILL.md +0 -140
  631. package/src/skill/compose/.bundle/worktree/SKILL.md +0 -234
  632. package/src/skill/compose/LICENSE-karpathy +0 -28
  633. package/src/skill/compose/bundle.macro.ts +0 -30
  634. package/src/skill/compose/extract.ts +0 -85
  635. package/src/skill/discovery.ts +0 -116
  636. package/src/skill/index.ts +0 -311
  637. package/src/snapshot/index.ts +0 -777
  638. package/src/sql.d.ts +0 -4
  639. package/src/storage/db.bun.ts +0 -8
  640. package/src/storage/db.node.ts +0 -8
  641. package/src/storage/db.ts +0 -172
  642. package/src/storage/index.ts +0 -26
  643. package/src/storage/json-migration.ts +0 -426
  644. package/src/storage/schema.sql.ts +0 -10
  645. package/src/storage/schema.ts +0 -7
  646. package/src/storage/storage.ts +0 -331
  647. package/src/sync/README.md +0 -179
  648. package/src/sync/event.sql.ts +0 -16
  649. package/src/sync/index.ts +0 -278
  650. package/src/sync/schema.ts +0 -14
  651. package/src/task/events.ts +0 -28
  652. package/src/task/gate-state.ts +0 -54
  653. package/src/task/gate.ts +0 -116
  654. package/src/task/index.ts +0 -1
  655. package/src/task/registry.ts +0 -387
  656. package/src/task/schema.ts +0 -43
  657. package/src/task/task.sql.ts +0 -50
  658. package/src/team/events.ts +0 -22
  659. package/src/team/index.ts +0 -113
  660. package/src/team/schema.ts +0 -31
  661. package/src/temporary.ts +0 -33
  662. package/src/tool/actor.shell.txt +0 -72
  663. package/src/tool/actor.ts +0 -803
  664. package/src/tool/actor.txt +0 -103
  665. package/src/tool/apply_patch.ts +0 -308
  666. package/src/tool/apply_patch.txt +0 -33
  667. package/src/tool/bash-interactive.ts +0 -183
  668. package/src/tool/bash.ts +0 -696
  669. package/src/tool/bash.txt +0 -123
  670. package/src/tool/change-directory.ts +0 -91
  671. package/src/tool/codesearch.ts +0 -63
  672. package/src/tool/codesearch.txt +0 -12
  673. package/src/tool/edit.ts +0 -685
  674. package/src/tool/edit.txt +0 -10
  675. package/src/tool/external-directory.ts +0 -132
  676. package/src/tool/glob.ts +0 -100
  677. package/src/tool/glob.txt +0 -6
  678. package/src/tool/grep.ts +0 -145
  679. package/src/tool/grep.txt +0 -8
  680. package/src/tool/history.ts +0 -146
  681. package/src/tool/history.txt +0 -17
  682. package/src/tool/index.ts +0 -4
  683. package/src/tool/invalid.ts +0 -20
  684. package/src/tool/invocation-style.ts +0 -17
  685. package/src/tool/lsp.ts +0 -91
  686. package/src/tool/lsp.txt +0 -19
  687. package/src/tool/mcp-exa.ts +0 -78
  688. package/src/tool/memory-path-guard.ts +0 -162
  689. package/src/tool/memory.ts +0 -81
  690. package/src/tool/memory.txt +0 -69
  691. package/src/tool/multiedit.ts +0 -61
  692. package/src/tool/multiedit.txt +0 -41
  693. package/src/tool/plan-enter.txt +0 -14
  694. package/src/tool/plan-exit.txt +0 -13
  695. package/src/tool/plan.ts +0 -90
  696. package/src/tool/question.ts +0 -67
  697. package/src/tool/question.txt +0 -10
  698. package/src/tool/read.ts +0 -327
  699. package/src/tool/read.txt +0 -14
  700. package/src/tool/registry.ts +0 -415
  701. package/src/tool/schema.ts +0 -17
  702. package/src/tool/session-cwd.ts +0 -35
  703. package/src/tool/shell-tokenize.ts +0 -346
  704. package/src/tool/shell-wrap.ts +0 -190
  705. package/src/tool/skill.ts +0 -76
  706. package/src/tool/skill.txt +0 -5
  707. package/src/tool/task.shell.txt +0 -57
  708. package/src/tool/task.ts +0 -456
  709. package/src/tool/task.txt +0 -56
  710. package/src/tool/tool.ts +0 -153
  711. package/src/tool/truncate.ts +0 -201
  712. package/src/tool/truncation-dir.ts +0 -4
  713. package/src/tool/webfetch.ts +0 -199
  714. package/src/tool/webfetch.txt +0 -13
  715. package/src/tool/websearch/index.ts +0 -111
  716. package/src/tool/websearch/mimo.ts +0 -120
  717. package/src/tool/websearch/websearch.txt +0 -14
  718. package/src/tool/workflow.ts +0 -164
  719. package/src/tool/workflow.txt +0 -25
  720. package/src/tool/write.ts +0 -88
  721. package/src/tool/write.txt +0 -9
  722. package/src/util/abort.ts +0 -35
  723. package/src/util/archive.ts +0 -15
  724. package/src/util/color.ts +0 -17
  725. package/src/util/data-url.ts +0 -9
  726. package/src/util/defer.ts +0 -10
  727. package/src/util/effect-http-client.ts +0 -11
  728. package/src/util/effect-zod.ts +0 -367
  729. package/src/util/error.ts +0 -78
  730. package/src/util/filesystem.ts +0 -243
  731. package/src/util/fn.ts +0 -21
  732. package/src/util/format.ts +0 -20
  733. package/src/util/iife.ts +0 -3
  734. package/src/util/index.ts +0 -12
  735. package/src/util/keybind.ts +0 -101
  736. package/src/util/lazy.ts +0 -18
  737. package/src/util/local-context.ts +0 -23
  738. package/src/util/locale.ts +0 -79
  739. package/src/util/lock.ts +0 -96
  740. package/src/util/log.ts +0 -197
  741. package/src/util/media.ts +0 -26
  742. package/src/util/mimo-process.ts +0 -24
  743. package/src/util/network.ts +0 -9
  744. package/src/util/process.ts +0 -174
  745. package/src/util/queue.ts +0 -32
  746. package/src/util/record.ts +0 -3
  747. package/src/util/rpc.ts +0 -64
  748. package/src/util/schema.ts +0 -53
  749. package/src/util/scrap.ts +0 -10
  750. package/src/util/signal.ts +0 -12
  751. package/src/util/timeout.ts +0 -14
  752. package/src/util/token.ts +0 -5
  753. package/src/util/update-schema.ts +0 -13
  754. package/src/util/which.ts +0 -14
  755. package/src/util/wildcard.ts +0 -57
  756. package/src/workflow/builtin/deep-research.js +0 -391
  757. package/src/workflow/builtin.ts +0 -54
  758. package/src/workflow/events.ts +0 -72
  759. package/src/workflow/meta.ts +0 -335
  760. package/src/workflow/persistence.ts +0 -312
  761. package/src/workflow/resolve.ts +0 -45
  762. package/src/workflow/runtime-ref.ts +0 -18
  763. package/src/workflow/runtime.ts +0 -1234
  764. package/src/workflow/sandbox.ts +0 -280
  765. package/src/workflow/workflow.sql.ts +0 -31
  766. package/src/workflow/workspace.ts +0 -69
  767. package/src/worktree/index.ts +0 -614
  768. package/sst-env.d.ts +0 -10
  769. package/test/AGENTS.md +0 -133
  770. package/test/account/repo.test.ts +0 -352
  771. package/test/account/service.test.ts +0 -456
  772. package/test/acp/agent-interface.test.ts +0 -51
  773. package/test/acp/event-subscription.test.ts +0 -725
  774. package/test/actor/cancel-cascade.test.ts +0 -432
  775. package/test/actor/no-completion-listener.test.ts +0 -41
  776. package/test/actor/poststop-progress-write-permission.repro.test.ts +0 -414
  777. package/test/actor/registry-render.test.ts +0 -113
  778. package/test/actor/registry-status.test.ts +0 -111
  779. package/test/actor/registry.test.ts +0 -619
  780. package/test/actor/return-header.test.ts +0 -40
  781. package/test/actor/spawn-lifecycle.test.ts +0 -346
  782. package/test/actor/spawn-no-deadlock.test.ts +0 -340
  783. package/test/actor/spawn-notification.test.ts +0 -393
  784. package/test/actor/spawn-task-autostart.test.ts +0 -530
  785. package/test/actor/spawn.test.ts +0 -1072
  786. package/test/actor/status-event-payload.test.ts +0 -132
  787. package/test/actor/terminology.test.ts +0 -39
  788. package/test/actor/turn.test.ts +0 -125
  789. package/test/actor/waiter.test.ts +0 -246
  790. package/test/agent/agent.test.ts +0 -874
  791. package/test/agent/allowlist.test.ts +0 -45
  792. package/test/auth/auth.test.ts +0 -86
  793. package/test/bus/bus-effect.test.ts +0 -162
  794. package/test/bus/bus-integration.test.ts +0 -87
  795. package/test/bus/bus.test.ts +0 -219
  796. package/test/cli/account.test.ts +0 -26
  797. package/test/cli/cmd/tui/prompt-part.test.ts +0 -47
  798. package/test/cli/error.test.ts +0 -18
  799. package/test/cli/github-action.test.ts +0 -198
  800. package/test/cli/github-remote.test.ts +0 -80
  801. package/test/cli/import.test.ts +0 -54
  802. package/test/cli/plugin-auth-picker.test.ts +0 -120
  803. package/test/cli/run-completion.test.ts +0 -131
  804. package/test/cli/tui/keybind-plugin.test.ts +0 -90
  805. package/test/cli/tui/plugin-add.test.ts +0 -111
  806. package/test/cli/tui/plugin-install.test.ts +0 -87
  807. package/test/cli/tui/plugin-lifecycle.test.ts +0 -224
  808. package/test/cli/tui/plugin-loader-entrypoint.test.ts +0 -484
  809. package/test/cli/tui/plugin-loader-pure.test.ts +0 -71
  810. package/test/cli/tui/plugin-loader.test.ts +0 -816
  811. package/test/cli/tui/plugin-toggle.test.ts +0 -157
  812. package/test/cli/tui/revert-diff.test.ts +0 -35
  813. package/test/cli/tui/route-agent-id.test.ts +0 -26
  814. package/test/cli/tui/sidebar-tps.test.ts +0 -63
  815. package/test/cli/tui/slot-replace.test.tsx +0 -47
  816. package/test/cli/tui/sync-bucket.test.ts +0 -29
  817. package/test/cli/tui/theme-store.test.ts +0 -51
  818. package/test/cli/tui/thread.test.ts +0 -121
  819. package/test/cli/tui/transcript.test.ts +0 -426
  820. package/test/cli/tui/use-event.test.tsx +0 -175
  821. package/test/cli/tui/voice.test.ts +0 -269
  822. package/test/command/deep-research-command.test.ts +0 -16
  823. package/test/config/agent-color.test.ts +0 -77
  824. package/test/config/checkpoint-fork.test.ts +0 -21
  825. package/test/config/config.test.ts +0 -2577
  826. package/test/config/fixtures/empty-frontmatter.md +0 -4
  827. package/test/config/fixtures/frontmatter.md +0 -28
  828. package/test/config/fixtures/markdown-header.md +0 -11
  829. package/test/config/fixtures/no-frontmatter.md +0 -1
  830. package/test/config/fixtures/weird-model-id.md +0 -13
  831. package/test/config/lsp.test.ts +0 -87
  832. package/test/config/markdown.test.ts +0 -228
  833. package/test/config/plugin.test.ts +0 -0
  834. package/test/config/tui.test.ts +0 -627
  835. package/test/control-plane/adaptors.test.ts +0 -71
  836. package/test/control-plane/sse.test.ts +0 -56
  837. package/test/effect/app-runtime-logger.test.ts +0 -92
  838. package/test/effect/cross-spawn-spawner.test.ts +0 -411
  839. package/test/effect/instance-state.test.ts +0 -482
  840. package/test/effect/observability.test.ts +0 -46
  841. package/test/effect/run-service.test.ts +0 -46
  842. package/test/effect/runner-warn-log.test.ts +0 -111
  843. package/test/effect/runner.test.ts +0 -494
  844. package/test/fake/provider.ts +0 -90
  845. package/test/file/fsmonitor.test.ts +0 -68
  846. package/test/file/ignore.test.ts +0 -10
  847. package/test/file/index.test.ts +0 -956
  848. package/test/file/path-traversal.test.ts +0 -204
  849. package/test/file/ripgrep.test.ts +0 -214
  850. package/test/file/watcher.test.ts +0 -249
  851. package/test/filesystem/filesystem.test.ts +0 -319
  852. package/test/fixture/db.ts +0 -11
  853. package/test/fixture/fixture.test.ts +0 -58
  854. package/test/fixture/fixture.ts +0 -190
  855. package/test/fixture/flock-worker.ts +0 -72
  856. package/test/fixture/lsp/fake-lsp-server.js +0 -75
  857. package/test/fixture/plug-worker.ts +0 -93
  858. package/test/fixture/plugin-meta-worker.ts +0 -19
  859. package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
  860. package/test/fixture/skills/agents-sdk/references/callable.md +0 -92
  861. package/test/fixture/skills/cloudflare/SKILL.md +0 -211
  862. package/test/fixture/skills/index.json +0 -6
  863. package/test/fixture/tui-plugin.ts +0 -329
  864. package/test/fixture/tui-runtime.ts +0 -31
  865. package/test/format/format.test.ts +0 -244
  866. package/test/git/git.test.ts +0 -128
  867. package/test/global/fixture/global-paths-worker.ts +0 -17
  868. package/test/global/mimocode-home.test.ts +0 -143
  869. package/test/history/backfill.test.ts +0 -149
  870. package/test/history/extract.test.ts +0 -106
  871. package/test/history/fts-query.test.ts +0 -30
  872. package/test/history/resolve.test.ts +0 -130
  873. package/test/history/service.test.ts +0 -210
  874. package/test/history/writer.test.ts +0 -163
  875. package/test/ide/ide.test.ts +0 -82
  876. package/test/inbox/drain-in-loop.test.ts +0 -230
  877. package/test/inbox/fork-agent-compat.test.ts +0 -387
  878. package/test/inbox/gc-on-init.test.ts +0 -167
  879. package/test/inbox/send-no-block.test.ts +0 -120
  880. package/test/inbox/sender-cancel-independence.test.ts +0 -160
  881. package/test/inbox/wake-matrix.test.ts +0 -141
  882. package/test/installation/installation.test.ts +0 -226
  883. package/test/keybind.test.ts +0 -421
  884. package/test/lib/effect.ts +0 -53
  885. package/test/lib/filesystem.ts +0 -10
  886. package/test/lib/llm-server.ts +0 -770
  887. package/test/lib/scripted-llm-server.ts +0 -245
  888. package/test/lsp/client.test.ts +0 -98
  889. package/test/lsp/index.test.ts +0 -109
  890. package/test/lsp/launch.test.ts +0 -22
  891. package/test/lsp/lifecycle.test.ts +0 -184
  892. package/test/mcp/headers.test.ts +0 -178
  893. package/test/mcp/lifecycle.test.ts +0 -824
  894. package/test/mcp/oauth-auto-connect.test.ts +0 -281
  895. package/test/mcp/oauth-browser.test.ts +0 -268
  896. package/test/mcp/oauth-callback.test.ts +0 -34
  897. package/test/memory/abort-leak-webfetch.ts +0 -49
  898. package/test/memory/abort-leak.test.ts +0 -127
  899. package/test/memory/cc-frontmatter.test.ts +0 -85
  900. package/test/memory/cc-paths.test.ts +0 -60
  901. package/test/memory/cc-reconcile.test.ts +0 -239
  902. package/test/memory/cc-search.test.ts +0 -151
  903. package/test/memory/fts-query.test.ts +0 -48
  904. package/test/memory/fts-rowid-stability.test.ts +0 -271
  905. package/test/memory/paths.test.ts +0 -210
  906. package/test/memory/reconcile.test.ts +0 -115
  907. package/test/memory/service.test.ts +0 -169
  908. package/test/npm.test.ts +0 -18
  909. package/test/patch/patch.test.ts +0 -348
  910. package/test/permission/abort.test.ts +0 -116
  911. package/test/permission/arity.test.ts +0 -33
  912. package/test/permission/disabled.test.ts +0 -51
  913. package/test/permission/next.test.ts +0 -1080
  914. package/test/permission/non-interactive.test.ts +0 -55
  915. package/test/permission-task.test.ts +0 -326
  916. package/test/plugin/actor-hooks.test.ts +0 -1471
  917. package/test/plugin/auth-override.test.ts +0 -79
  918. package/test/plugin/checkpoint-splitover.test.ts +0 -434
  919. package/test/plugin/cloudflare.test.ts +0 -68
  920. package/test/plugin/codex.test.ts +0 -123
  921. package/test/plugin/github-copilot-models.test.ts +0 -163
  922. package/test/plugin/install-concurrency.test.ts +0 -140
  923. package/test/plugin/install.test.ts +0 -570
  924. package/test/plugin/loader-shared.test.ts +0 -1169
  925. package/test/plugin/matcher.test.ts +0 -97
  926. package/test/plugin/meta.test.ts +0 -137
  927. package/test/plugin/shared.test.ts +0 -88
  928. package/test/plugin/subagent-progress-checker.test.ts +0 -227
  929. package/test/plugin/trigger.test.ts +0 -116
  930. package/test/plugin/woozlit.test.ts +0 -68
  931. package/test/plugin/workspace-adaptor.test.ts +0 -109
  932. package/test/preload.ts +0 -102
  933. package/test/project/migrate-global.test.ts +0 -150
  934. package/test/project/project-id.test.ts +0 -64
  935. package/test/project/project.test.ts +0 -481
  936. package/test/project/vcs.test.ts +0 -286
  937. package/test/project/worktree-remove.test.ts +0 -126
  938. package/test/project/worktree.test.ts +0 -214
  939. package/test/provider/amazon-bedrock.test.ts +0 -462
  940. package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
  941. package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
  942. package/test/provider/error.test.ts +0 -160
  943. package/test/provider/gitlab-duo.test.ts +0 -413
  944. package/test/provider/model-groups.test.ts +0 -389
  945. package/test/provider/provider-chunk-timeout.test.ts +0 -23
  946. package/test/provider/provider.test.ts +0 -2648
  947. package/test/provider/transform.test.ts +0 -3379
  948. package/test/pty/pty-output-isolation.test.ts +0 -146
  949. package/test/pty/pty-session.test.ts +0 -102
  950. package/test/pty/pty-shell.test.ts +0 -69
  951. package/test/question/question.test.ts +0 -464
  952. package/test/server/global-session-list.test.ts +0 -105
  953. package/test/server/project-init-git.test.ts +0 -122
  954. package/test/server/session-actions.test.ts +0 -49
  955. package/test/server/session-list.test.ts +0 -110
  956. package/test/server/session-messages.test.ts +0 -220
  957. package/test/server/session-prompt-busy.test.ts +0 -146
  958. package/test/server/session-select.test.ts +0 -100
  959. package/test/server/session-task-route.test.ts +0 -165
  960. package/test/server/summarize-route-main-slice.test.ts +0 -99
  961. package/test/server/trace-attributes.test.ts +0 -76
  962. package/test/server/workflows-route.test.ts +0 -279
  963. package/test/session/bootstrap-skip-system.test.ts +0 -121
  964. package/test/session/boundary.test.ts +0 -33
  965. package/test/session/budgeted-read.test.ts +0 -74
  966. package/test/session/checkpoint-align.test.ts +0 -58
  967. package/test/session/checkpoint-boundary.test.ts +0 -186
  968. package/test/session/checkpoint-child-session.test.ts +0 -508
  969. package/test/session/checkpoint-context.test.ts +0 -141
  970. package/test/session/checkpoint-drain.test.ts +0 -188
  971. package/test/session/checkpoint-extract-titles.test.ts +0 -58
  972. package/test/session/checkpoint-fork-mode.test.ts +0 -576
  973. package/test/session/checkpoint-main-slice.test.ts +0 -259
  974. package/test/session/checkpoint-paths.test.ts +0 -78
  975. package/test/session/checkpoint-permission.test.ts +0 -136
  976. package/test/session/checkpoint-progress-reconcile.test.ts +0 -219
  977. package/test/session/checkpoint-rebuild-unify.test.ts +0 -143
  978. package/test/session/checkpoint-rebuild-v3.test.ts +0 -248
  979. package/test/session/checkpoint-render-verify.test.ts +0 -512
  980. package/test/session/checkpoint-retry.test.ts +0 -150
  981. package/test/session/checkpoint-splitover-integration.test.ts +0 -533
  982. package/test/session/checkpoint-templates.test.ts +0 -51
  983. package/test/session/checkpoint-thresholds.test.ts +0 -120
  984. package/test/session/checkpoint-validator.test.ts +0 -189
  985. package/test/session/classify-integration.test.ts +0 -476
  986. package/test/session/classify.test.ts +0 -335
  987. package/test/session/compaction-agent-scope.test.ts +0 -164
  988. package/test/session/context-inheritance.test.ts +0 -46
  989. package/test/session/fork-prefix-invariant.test.ts +0 -116
  990. package/test/session/goal.test.ts +0 -106
  991. package/test/session/instruction.test.ts +0 -387
  992. package/test/session/invalid-output-continuation.test.ts +0 -150
  993. package/test/session/last-message-info.test.ts +0 -47
  994. package/test/session/length-tool-safety.test.ts +0 -121
  995. package/test/session/llm-request-prefix.test.ts +0 -197
  996. package/test/session/llm-retry.test.ts +0 -59
  997. package/test/session/llm-system-prompt.test.ts +0 -479
  998. package/test/session/llm.test.ts +0 -1272
  999. package/test/session/main-lifecycle.test.ts +0 -51
  1000. package/test/session/main-runloop-history-invariant.test.ts +0 -182
  1001. package/test/session/max-mode-econnreset.test.ts +0 -229
  1002. package/test/session/max-mode.test.ts +0 -54
  1003. package/test/session/message-v2-filter.test.ts +0 -197
  1004. package/test/session/message-v2.test.ts +0 -1119
  1005. package/test/session/messages-default-main.test.ts +0 -105
  1006. package/test/session/messages-pagination.test.ts +0 -888
  1007. package/test/session/overflow.test.ts +0 -576
  1008. package/test/session/processor-effect.test.ts +0 -853
  1009. package/test/session/prompt-effect.test.ts +0 -1574
  1010. package/test/session/prompt-rebuild-loop.test.ts +0 -108
  1011. package/test/session/prompt-rebuild-reset.test.ts +0 -67
  1012. package/test/session/prompt-sweep.test.ts +0 -145
  1013. package/test/session/prompt-task-gate.test.ts +0 -127
  1014. package/test/session/prompt.test.ts +0 -703
  1015. package/test/session/prune-main-slice.test.ts +0 -272
  1016. package/test/session/prune-skip-system.test.ts +0 -346
  1017. package/test/session/prune.test.ts +0 -419
  1018. package/test/session/rebuild-microcompact.test.ts +0 -318
  1019. package/test/session/recall-reminder.test.ts +0 -37
  1020. package/test/session/retry.test.ts +0 -410
  1021. package/test/session/revert-compact.test.ts +0 -639
  1022. package/test/session/run-state-tuple-key.test.ts +0 -152
  1023. package/test/session/session-create-registers-main.test.ts +0 -70
  1024. package/test/session/session.test.ts +0 -181
  1025. package/test/session/snapshot-tool-race.test.ts +0 -301
  1026. package/test/session/structured-output-integration.test.ts +0 -264
  1027. package/test/session/structured-output-retry.test.ts +0 -127
  1028. package/test/session/structured-output.test.ts +0 -397
  1029. package/test/session/summary-main-slice.test.ts +0 -170
  1030. package/test/session/system.test.ts +0 -72
  1031. package/test/share/share-next.test.ts +0 -332
  1032. package/test/shell/shell.test.ts +0 -73
  1033. package/test/skill/compose-review.test.ts +0 -141
  1034. package/test/skill/discovery.test.ts +0 -116
  1035. package/test/skill/skill.test.ts +0 -465
  1036. package/test/snapshot/snapshot.test.ts +0 -1531
  1037. package/test/storage/db.test.ts +0 -16
  1038. package/test/storage/json-migration.test.ts +0 -831
  1039. package/test/storage/storage.test.ts +0 -293
  1040. package/test/sync/index.test.ts +0 -237
  1041. package/test/task/gate-state.test.ts +0 -66
  1042. package/test/task/gate.test.ts +0 -167
  1043. package/test/task/registry.test.ts +0 -152
  1044. package/test/task/state-machine.test.ts +0 -292
  1045. package/test/team/migrate-to-inbox.test.ts +0 -124
  1046. package/test/team/team.test.ts +0 -75
  1047. package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
  1048. package/test/tool/actor-cancel.test.ts +0 -206
  1049. package/test/tool/actor-recover.test.ts +0 -50
  1050. package/test/tool/actor-send.test.ts +0 -200
  1051. package/test/tool/actor-status.test.ts +0 -296
  1052. package/test/tool/actor-wait.test.ts +0 -193
  1053. package/test/tool/actor.shell.test.ts +0 -250
  1054. package/test/tool/actor.test.ts +0 -748
  1055. package/test/tool/apply_patch.test.ts +0 -626
  1056. package/test/tool/bash.test.ts +0 -1195
  1057. package/test/tool/describe-workflow.test.ts +0 -12
  1058. package/test/tool/edit.test.ts +0 -691
  1059. package/test/tool/external-directory.test.ts +0 -207
  1060. package/test/tool/fixtures/large-image.png +0 -0
  1061. package/test/tool/fixtures/models-api.json +0 -65179
  1062. package/test/tool/glob.test.ts +0 -81
  1063. package/test/tool/grep.test.ts +0 -114
  1064. package/test/tool/history.test.ts +0 -144
  1065. package/test/tool/invocation-style.test.ts +0 -30
  1066. package/test/tool/memory-edit-ask-skip.test.ts +0 -62
  1067. package/test/tool/memory-path-guard.test.ts +0 -594
  1068. package/test/tool/memory.test.ts +0 -71
  1069. package/test/tool/question.test.ts +0 -167
  1070. package/test/tool/read.test.ts +0 -483
  1071. package/test/tool/registry-invocation-style.test.ts +0 -121
  1072. package/test/tool/registry.test.ts +0 -164
  1073. package/test/tool/shell-tokenize.test.ts +0 -273
  1074. package/test/tool/shell-wrap-missing-script.test.ts +0 -128
  1075. package/test/tool/shell-wrap.test.ts +0 -257
  1076. package/test/tool/skill.test.ts +0 -99
  1077. package/test/tool/task-recover.test.ts +0 -36
  1078. package/test/tool/task.shell.test.ts +0 -234
  1079. package/test/tool/task.test.ts +0 -296
  1080. package/test/tool/tool-def-shell-shape.test.ts +0 -23
  1081. package/test/tool/tool-define.test.ts +0 -59
  1082. package/test/tool/truncation.test.ts +0 -253
  1083. package/test/tool/webfetch.test.ts +0 -103
  1084. package/test/tool/whitelist.test.ts +0 -373
  1085. package/test/tool/write.test.ts +0 -244
  1086. package/test/util/data-url.test.ts +0 -14
  1087. package/test/util/effect-zod.test.ts +0 -869
  1088. package/test/util/error.test.ts +0 -38
  1089. package/test/util/filesystem.test.ts +0 -656
  1090. package/test/util/format.test.ts +0 -59
  1091. package/test/util/glob.test.ts +0 -164
  1092. package/test/util/iife.test.ts +0 -36
  1093. package/test/util/lazy.test.ts +0 -50
  1094. package/test/util/lock.test.ts +0 -72
  1095. package/test/util/log.test.ts +0 -44
  1096. package/test/util/module.test.ts +0 -59
  1097. package/test/util/process.test.ts +0 -128
  1098. package/test/util/timeout.test.ts +0 -21
  1099. package/test/util/which.test.ts +0 -100
  1100. package/test/util/wildcard.test.ts +0 -90
  1101. package/test/workflow/builtin.test.ts +0 -22
  1102. package/test/workflow/deep-research-cluster.test.ts +0 -47
  1103. package/test/workflow/lib.ts +0 -243
  1104. package/test/workflow/meta.test.ts +0 -142
  1105. package/test/workflow/model-routing.test.ts +0 -68
  1106. package/test/workflow/persistence.test.ts +0 -229
  1107. package/test/workflow/resolve.test.ts +0 -37
  1108. package/test/workflow/runtime-nested.test.ts +0 -419
  1109. package/test/workflow/runtime-worktree.test.ts +0 -261
  1110. package/test/workflow/runtime.test.ts +0 -1078
  1111. package/test/workflow/sandbox.test.ts +0 -259
  1112. package/test/workflow/tool.test.ts +0 -473
  1113. package/test/workflow/verify-wow.test.ts +0 -144
  1114. package/test/workflow/workspace.test.ts +0 -88
  1115. package/test/workspace/workspace-restore.test.ts +0 -281
  1116. package/test/worktree/index.test.ts +0 -30
  1117. package/tsconfig.json +0 -24
  1118. /package/{script/postinstall.mjs → postinstall.mjs} +0 -0
@@ -1,2648 +0,0 @@
1
- import { test, expect } from "bun:test"
2
- import { mkdir } from "fs/promises"
3
- import path from "path"
4
-
5
- import { tmpdir } from "../fixture/fixture"
6
- import { Instance } from "../../src/project/instance"
7
- import { Plugin } from "../../src/plugin/index"
8
- import { ModelsDev } from "../../src/provider"
9
- import { Provider } from "../../src/provider"
10
- import { ProviderID, ModelID } from "../../src/provider/schema"
11
- import { Env } from "../../src/env"
12
- import { Effect } from "effect"
13
- import { AppRuntime } from "../../src/effect/app-runtime"
14
- import { makeRuntime } from "../../src/effect/run-service"
15
-
16
- const env = makeRuntime(Env.Service, Env.defaultLayer)
17
- const set = (k: string, v: string) => env.runSync((svc) => svc.set(k, v))
18
-
19
- async function run<A, E>(fn: (provider: Provider.Interface) => Effect.Effect<A, E, never>) {
20
- return AppRuntime.runPromise(
21
- Effect.gen(function* () {
22
- const provider = yield* Provider.Service
23
- return yield* fn(provider)
24
- }),
25
- )
26
- }
27
-
28
- async function list() {
29
- return run((provider) => provider.list())
30
- }
31
-
32
- async function getProvider(providerID: ProviderID) {
33
- return run((provider) => provider.getProvider(providerID))
34
- }
35
-
36
- async function getModel(providerID: ProviderID, modelID: ModelID) {
37
- return run((provider) => provider.getModel(providerID, modelID))
38
- }
39
-
40
- async function getLanguage(model: Provider.Model) {
41
- return run((provider) => provider.getLanguage(model))
42
- }
43
-
44
- async function closest(providerID: ProviderID, query: string[]) {
45
- return run((provider) => provider.closest(providerID, query))
46
- }
47
-
48
- async function getSmallModel(providerID: ProviderID) {
49
- return run((provider) => provider.getSmallModel(providerID))
50
- }
51
-
52
- async function defaultModel() {
53
- return run((provider) => provider.defaultModel())
54
- }
55
-
56
- function opencodeProviderPresent(providers: Awaited<ReturnType<typeof list>>): boolean {
57
- return providers[ProviderID.make("opencode")] !== undefined
58
- }
59
-
60
- test("provider loaded from env variable", async () => {
61
- await using tmp = await tmpdir({
62
- init: async (dir) => {
63
- await Bun.write(
64
- path.join(dir, "WOOZLIT_CODE.json"),
65
- JSON.stringify({
66
- $schema: "https://opencode.ai/config.json",
67
- }),
68
- )
69
- },
70
- })
71
- await Instance.provide({
72
- directory: tmp.path,
73
- init: async () => {
74
- set("ANTHROPIC_API_KEY", "test-api-key")
75
- },
76
- fn: async () => {
77
- const providers = await list()
78
- expect(providers[ProviderID.anthropic]).toBeDefined()
79
- // Provider should retain its connection source even if custom loaders
80
- // merge additional options.
81
- expect(providers[ProviderID.anthropic].source).toBe("env")
82
- expect(providers[ProviderID.anthropic].options.headers["anthropic-beta"]).toBeDefined()
83
- },
84
- })
85
- })
86
-
87
- test("provider loaded from config with apiKey option", async () => {
88
- await using tmp = await tmpdir({
89
- init: async (dir) => {
90
- await Bun.write(
91
- path.join(dir, "WOOZLIT_CODE.json"),
92
- JSON.stringify({
93
- $schema: "https://opencode.ai/config.json",
94
- provider: {
95
- anthropic: {
96
- options: {
97
- apiKey: "config-api-key",
98
- },
99
- },
100
- },
101
- }),
102
- )
103
- },
104
- })
105
- await Instance.provide({
106
- directory: tmp.path,
107
- fn: async () => {
108
- const providers = await list()
109
- expect(providers[ProviderID.anthropic]).toBeDefined()
110
- },
111
- })
112
- })
113
-
114
- test("disabled_providers excludes provider", async () => {
115
- await using tmp = await tmpdir({
116
- init: async (dir) => {
117
- await Bun.write(
118
- path.join(dir, "WOOZLIT_CODE.json"),
119
- JSON.stringify({
120
- $schema: "https://opencode.ai/config.json",
121
- disabled_providers: ["anthropic"],
122
- }),
123
- )
124
- },
125
- })
126
- await Instance.provide({
127
- directory: tmp.path,
128
- init: async () => {
129
- set("ANTHROPIC_API_KEY", "test-api-key")
130
- },
131
- fn: async () => {
132
- const providers = await list()
133
- expect(providers[ProviderID.anthropic]).toBeUndefined()
134
- },
135
- })
136
- })
137
-
138
- test("enabled_providers restricts to only listed providers", async () => {
139
- await using tmp = await tmpdir({
140
- init: async (dir) => {
141
- await Bun.write(
142
- path.join(dir, "WOOZLIT_CODE.json"),
143
- JSON.stringify({
144
- $schema: "https://opencode.ai/config.json",
145
- enabled_providers: ["anthropic"],
146
- }),
147
- )
148
- },
149
- })
150
- await Instance.provide({
151
- directory: tmp.path,
152
- init: async () => {
153
- set("ANTHROPIC_API_KEY", "test-api-key")
154
- set("OPENAI_API_KEY", "test-openai-key")
155
- },
156
- fn: async () => {
157
- const providers = await list()
158
- expect(providers[ProviderID.anthropic]).toBeDefined()
159
- expect(providers[ProviderID.openai]).toBeUndefined()
160
- },
161
- })
162
- })
163
-
164
- test("model whitelist filters models for provider", async () => {
165
- await using tmp = await tmpdir({
166
- init: async (dir) => {
167
- await Bun.write(
168
- path.join(dir, "WOOZLIT_CODE.json"),
169
- JSON.stringify({
170
- $schema: "https://opencode.ai/config.json",
171
- provider: {
172
- anthropic: {
173
- whitelist: ["claude-sonnet-4-20250514"],
174
- },
175
- },
176
- }),
177
- )
178
- },
179
- })
180
- await Instance.provide({
181
- directory: tmp.path,
182
- init: async () => {
183
- set("ANTHROPIC_API_KEY", "test-api-key")
184
- },
185
- fn: async () => {
186
- const providers = await list()
187
- expect(providers[ProviderID.anthropic]).toBeDefined()
188
- const models = Object.keys(providers[ProviderID.anthropic].models)
189
- expect(models).toContain("claude-sonnet-4-20250514")
190
- expect(models.length).toBe(1)
191
- },
192
- })
193
- })
194
-
195
- test("model blacklist excludes specific models", async () => {
196
- await using tmp = await tmpdir({
197
- init: async (dir) => {
198
- await Bun.write(
199
- path.join(dir, "WOOZLIT_CODE.json"),
200
- JSON.stringify({
201
- $schema: "https://opencode.ai/config.json",
202
- provider: {
203
- anthropic: {
204
- blacklist: ["claude-sonnet-4-20250514"],
205
- },
206
- },
207
- }),
208
- )
209
- },
210
- })
211
- await Instance.provide({
212
- directory: tmp.path,
213
- init: async () => {
214
- set("ANTHROPIC_API_KEY", "test-api-key")
215
- },
216
- fn: async () => {
217
- const providers = await list()
218
- expect(providers[ProviderID.anthropic]).toBeDefined()
219
- const models = Object.keys(providers[ProviderID.anthropic].models)
220
- expect(models).not.toContain("claude-sonnet-4-20250514")
221
- },
222
- })
223
- })
224
-
225
- test("custom model alias via config", async () => {
226
- await using tmp = await tmpdir({
227
- init: async (dir) => {
228
- await Bun.write(
229
- path.join(dir, "WOOZLIT_CODE.json"),
230
- JSON.stringify({
231
- $schema: "https://opencode.ai/config.json",
232
- provider: {
233
- anthropic: {
234
- models: {
235
- "my-alias": {
236
- id: "claude-sonnet-4-20250514",
237
- name: "My Custom Alias",
238
- },
239
- },
240
- },
241
- },
242
- }),
243
- )
244
- },
245
- })
246
- await Instance.provide({
247
- directory: tmp.path,
248
- init: async () => {
249
- set("ANTHROPIC_API_KEY", "test-api-key")
250
- },
251
- fn: async () => {
252
- const providers = await list()
253
- expect(providers[ProviderID.anthropic]).toBeDefined()
254
- expect(providers[ProviderID.anthropic].models["my-alias"]).toBeDefined()
255
- expect(providers[ProviderID.anthropic].models["my-alias"].name).toBe("My Custom Alias")
256
- },
257
- })
258
- })
259
-
260
- test("custom provider with npm package", async () => {
261
- await using tmp = await tmpdir({
262
- init: async (dir) => {
263
- await Bun.write(
264
- path.join(dir, "WOOZLIT_CODE.json"),
265
- JSON.stringify({
266
- $schema: "https://opencode.ai/config.json",
267
- provider: {
268
- "custom-provider": {
269
- name: "Custom Provider",
270
- npm: "@ai-sdk/openai-compatible",
271
- api: "https://api.custom.com/v1",
272
- env: ["CUSTOM_API_KEY"],
273
- models: {
274
- "custom-model": {
275
- name: "Custom Model",
276
- tool_call: true,
277
- limit: {
278
- context: 128000,
279
- output: 4096,
280
- },
281
- },
282
- },
283
- options: {
284
- apiKey: "custom-key",
285
- },
286
- },
287
- },
288
- }),
289
- )
290
- },
291
- })
292
- await Instance.provide({
293
- directory: tmp.path,
294
- fn: async () => {
295
- const providers = await list()
296
- expect(providers[ProviderID.make("custom-provider")]).toBeDefined()
297
- expect(providers[ProviderID.make("custom-provider")].name).toBe("Custom Provider")
298
- expect(providers[ProviderID.make("custom-provider")].models["custom-model"]).toBeDefined()
299
- },
300
- })
301
- })
302
-
303
- test("custom DeepSeek openai-compatible model defaults interleaved reasoning field", async () => {
304
- await using tmp = await tmpdir({
305
- init: async (dir) => {
306
- await Bun.write(
307
- path.join(dir, "WOOZLIT_CODE.json"),
308
- JSON.stringify({
309
- $schema: "https://opencode.ai/config.json",
310
- provider: {
311
- "custom-provider": {
312
- name: "Custom Provider",
313
- npm: "@ai-sdk/openai-compatible",
314
- api: "https://api.custom.com/v1",
315
- models: {
316
- "deepseek-r1": {
317
- name: "DeepSeek R1",
318
- },
319
- "deepseek-details": {
320
- name: "DeepSeek Details",
321
- interleaved: { field: "reasoning_details" },
322
- },
323
- "custom-model": {
324
- name: "Custom Model",
325
- },
326
- },
327
- options: {
328
- apiKey: "custom-key",
329
- },
330
- },
331
- "custom-anthropic-provider": {
332
- name: "Custom Anthropic Provider",
333
- npm: "@ai-sdk/anthropic",
334
- api: "https://api.custom.com/v1",
335
- models: {
336
- "deepseek-r1": {
337
- name: "DeepSeek R1",
338
- },
339
- },
340
- options: {
341
- apiKey: "custom-key",
342
- },
343
- },
344
- },
345
- }),
346
- )
347
- },
348
- })
349
- await Instance.provide({
350
- directory: tmp.path,
351
- fn: async () => {
352
- const providers = await list()
353
- const provider = providers[ProviderID.make("custom-provider")]
354
- expect(provider.models["deepseek-r1"].capabilities.interleaved).toEqual({ field: "reasoning_content" })
355
- expect(provider.models["deepseek-details"].capabilities.interleaved).toEqual({ field: "reasoning_details" })
356
- expect(provider.models["custom-model"].capabilities.interleaved).toBe(false)
357
- expect(
358
- providers[ProviderID.make("custom-anthropic-provider")].models["deepseek-r1"].capabilities.interleaved,
359
- ).toBe(false)
360
- },
361
- })
362
- })
363
-
364
- test("env variable takes precedence, config merges options", async () => {
365
- await using tmp = await tmpdir({
366
- init: async (dir) => {
367
- await Bun.write(
368
- path.join(dir, "WOOZLIT_CODE.json"),
369
- JSON.stringify({
370
- $schema: "https://opencode.ai/config.json",
371
- provider: {
372
- anthropic: {
373
- options: {
374
- timeout: 60000,
375
- chunkTimeout: 15000,
376
- },
377
- },
378
- },
379
- }),
380
- )
381
- },
382
- })
383
- await Instance.provide({
384
- directory: tmp.path,
385
- init: async () => {
386
- set("ANTHROPIC_API_KEY", "env-api-key")
387
- },
388
- fn: async () => {
389
- const providers = await list()
390
- expect(providers[ProviderID.anthropic]).toBeDefined()
391
- // Config options should be merged
392
- expect(providers[ProviderID.anthropic].options.timeout).toBe(60000)
393
- expect(providers[ProviderID.anthropic].options.chunkTimeout).toBe(15000)
394
- },
395
- })
396
- })
397
-
398
- test("getModel returns model for valid provider/model", async () => {
399
- await using tmp = await tmpdir({
400
- init: async (dir) => {
401
- await Bun.write(
402
- path.join(dir, "WOOZLIT_CODE.json"),
403
- JSON.stringify({
404
- $schema: "https://opencode.ai/config.json",
405
- }),
406
- )
407
- },
408
- })
409
- await Instance.provide({
410
- directory: tmp.path,
411
- init: async () => {
412
- set("ANTHROPIC_API_KEY", "test-api-key")
413
- },
414
- fn: async () => {
415
- const model = await getModel(ProviderID.anthropic, ModelID.make("claude-sonnet-4-20250514"))
416
- expect(model).toBeDefined()
417
- expect(String(model.providerID)).toBe("anthropic")
418
- expect(String(model.id)).toBe("claude-sonnet-4-20250514")
419
- const language = await getLanguage(model)
420
- expect(language).toBeDefined()
421
- },
422
- })
423
- })
424
-
425
- test("getModel throws ModelNotFoundError for invalid model", async () => {
426
- await using tmp = await tmpdir({
427
- init: async (dir) => {
428
- await Bun.write(
429
- path.join(dir, "WOOZLIT_CODE.json"),
430
- JSON.stringify({
431
- $schema: "https://opencode.ai/config.json",
432
- }),
433
- )
434
- },
435
- })
436
- await Instance.provide({
437
- directory: tmp.path,
438
- init: async () => {
439
- set("ANTHROPIC_API_KEY", "test-api-key")
440
- },
441
- fn: async () => {
442
- expect(getModel(ProviderID.anthropic, ModelID.make("nonexistent-model"))).rejects.toThrow()
443
- },
444
- })
445
- })
446
-
447
- test("getModel throws ModelNotFoundError for invalid provider", async () => {
448
- await using tmp = await tmpdir({
449
- init: async (dir) => {
450
- await Bun.write(
451
- path.join(dir, "WOOZLIT_CODE.json"),
452
- JSON.stringify({
453
- $schema: "https://opencode.ai/config.json",
454
- }),
455
- )
456
- },
457
- })
458
- await Instance.provide({
459
- directory: tmp.path,
460
- fn: async () => {
461
- expect(getModel(ProviderID.make("nonexistent-provider"), ModelID.make("some-model"))).rejects.toThrow()
462
- },
463
- })
464
- })
465
-
466
- test("parseModel correctly parses provider/model string", () => {
467
- const result = Provider.parseModel("anthropic/claude-sonnet-4")
468
- expect(String(result.providerID)).toBe("anthropic")
469
- expect(String(result.modelID)).toBe("claude-sonnet-4")
470
- })
471
-
472
- test("parseModel handles model IDs with slashes", () => {
473
- const result = Provider.parseModel("openrouter/anthropic/claude-3-opus")
474
- expect(String(result.providerID)).toBe("openrouter")
475
- expect(String(result.modelID)).toBe("anthropic/claude-3-opus")
476
- })
477
-
478
- test("defaultModel returns first available model when no config set", async () => {
479
- await using tmp = await tmpdir({
480
- init: async (dir) => {
481
- await Bun.write(
482
- path.join(dir, "WOOZLIT_CODE.json"),
483
- JSON.stringify({
484
- $schema: "https://opencode.ai/config.json",
485
- }),
486
- )
487
- },
488
- })
489
- await Instance.provide({
490
- directory: tmp.path,
491
- init: async () => {
492
- set("ANTHROPIC_API_KEY", "test-api-key")
493
- },
494
- fn: async () => {
495
- const model = await defaultModel()
496
- expect(model.providerID).toBeDefined()
497
- expect(model.modelID).toBeDefined()
498
- },
499
- })
500
- })
501
-
502
- test("defaultModel respects config model setting", async () => {
503
- await using tmp = await tmpdir({
504
- init: async (dir) => {
505
- await Bun.write(
506
- path.join(dir, "WOOZLIT_CODE.json"),
507
- JSON.stringify({
508
- $schema: "https://opencode.ai/config.json",
509
- model: "anthropic/claude-sonnet-4-20250514",
510
- }),
511
- )
512
- },
513
- })
514
- await Instance.provide({
515
- directory: tmp.path,
516
- init: async () => {
517
- set("ANTHROPIC_API_KEY", "test-api-key")
518
- },
519
- fn: async () => {
520
- const model = await defaultModel()
521
- expect(String(model.providerID)).toBe("anthropic")
522
- expect(String(model.modelID)).toBe("claude-sonnet-4-20250514")
523
- },
524
- })
525
- })
526
-
527
- test("provider with baseURL from config", async () => {
528
- await using tmp = await tmpdir({
529
- init: async (dir) => {
530
- await Bun.write(
531
- path.join(dir, "WOOZLIT_CODE.json"),
532
- JSON.stringify({
533
- $schema: "https://opencode.ai/config.json",
534
- provider: {
535
- "custom-openai": {
536
- name: "Custom OpenAI",
537
- npm: "@ai-sdk/openai-compatible",
538
- env: [],
539
- models: {
540
- "gpt-4": {
541
- name: "GPT-4",
542
- tool_call: true,
543
- limit: { context: 128000, output: 4096 },
544
- },
545
- },
546
- options: {
547
- apiKey: "test-key",
548
- baseURL: "https://custom.openai.com/v1",
549
- },
550
- },
551
- },
552
- }),
553
- )
554
- },
555
- })
556
- await Instance.provide({
557
- directory: tmp.path,
558
- fn: async () => {
559
- const providers = await list()
560
- expect(providers[ProviderID.make("custom-openai")]).toBeDefined()
561
- expect(providers[ProviderID.make("custom-openai")].options.baseURL).toBe("https://custom.openai.com/v1")
562
- },
563
- })
564
- })
565
-
566
- test("model cost defaults to zero when not specified", async () => {
567
- await using tmp = await tmpdir({
568
- init: async (dir) => {
569
- await Bun.write(
570
- path.join(dir, "WOOZLIT_CODE.json"),
571
- JSON.stringify({
572
- $schema: "https://opencode.ai/config.json",
573
- provider: {
574
- "test-provider": {
575
- name: "Test Provider",
576
- npm: "@ai-sdk/openai-compatible",
577
- env: [],
578
- models: {
579
- "test-model": {
580
- name: "Test Model",
581
- tool_call: true,
582
- limit: { context: 128000, output: 4096 },
583
- },
584
- },
585
- options: {
586
- apiKey: "test-key",
587
- },
588
- },
589
- },
590
- }),
591
- )
592
- },
593
- })
594
- await Instance.provide({
595
- directory: tmp.path,
596
- fn: async () => {
597
- const providers = await list()
598
- const model = providers[ProviderID.make("test-provider")].models["test-model"]
599
- expect(model.cost.input).toBe(0)
600
- expect(model.cost.output).toBe(0)
601
- expect(model.cost.cache.read).toBe(0)
602
- expect(model.cost.cache.write).toBe(0)
603
- },
604
- })
605
- })
606
-
607
- test("model options are merged from existing model", async () => {
608
- await using tmp = await tmpdir({
609
- init: async (dir) => {
610
- await Bun.write(
611
- path.join(dir, "WOOZLIT_CODE.json"),
612
- JSON.stringify({
613
- $schema: "https://opencode.ai/config.json",
614
- provider: {
615
- anthropic: {
616
- models: {
617
- "claude-sonnet-4-20250514": {
618
- options: {
619
- customOption: "custom-value",
620
- },
621
- },
622
- },
623
- },
624
- },
625
- }),
626
- )
627
- },
628
- })
629
- await Instance.provide({
630
- directory: tmp.path,
631
- init: async () => {
632
- set("ANTHROPIC_API_KEY", "test-api-key")
633
- },
634
- fn: async () => {
635
- const providers = await list()
636
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
637
- expect(model.options.customOption).toBe("custom-value")
638
- },
639
- })
640
- })
641
-
642
- test("provider removed when all models filtered out", async () => {
643
- await using tmp = await tmpdir({
644
- init: async (dir) => {
645
- await Bun.write(
646
- path.join(dir, "WOOZLIT_CODE.json"),
647
- JSON.stringify({
648
- $schema: "https://opencode.ai/config.json",
649
- provider: {
650
- anthropic: {
651
- whitelist: ["nonexistent-model"],
652
- },
653
- },
654
- }),
655
- )
656
- },
657
- })
658
- await Instance.provide({
659
- directory: tmp.path,
660
- init: async () => {
661
- set("ANTHROPIC_API_KEY", "test-api-key")
662
- },
663
- fn: async () => {
664
- const providers = await list()
665
- expect(providers[ProviderID.anthropic]).toBeUndefined()
666
- },
667
- })
668
- })
669
-
670
- test("closest finds model by partial match", async () => {
671
- await using tmp = await tmpdir({
672
- init: async (dir) => {
673
- await Bun.write(
674
- path.join(dir, "WOOZLIT_CODE.json"),
675
- JSON.stringify({
676
- $schema: "https://opencode.ai/config.json",
677
- }),
678
- )
679
- },
680
- })
681
- await Instance.provide({
682
- directory: tmp.path,
683
- init: async () => {
684
- set("ANTHROPIC_API_KEY", "test-api-key")
685
- },
686
- fn: async () => {
687
- const result = await closest(ProviderID.anthropic, ["sonnet-4"])
688
- expect(result).toBeDefined()
689
- expect(String(result?.providerID)).toBe("anthropic")
690
- expect(String(result?.modelID)).toContain("sonnet-4")
691
- },
692
- })
693
- })
694
-
695
- test("closest returns undefined for nonexistent provider", async () => {
696
- await using tmp = await tmpdir({
697
- init: async (dir) => {
698
- await Bun.write(
699
- path.join(dir, "WOOZLIT_CODE.json"),
700
- JSON.stringify({
701
- $schema: "https://opencode.ai/config.json",
702
- }),
703
- )
704
- },
705
- })
706
- await Instance.provide({
707
- directory: tmp.path,
708
- fn: async () => {
709
- const result = await closest(ProviderID.make("nonexistent"), ["model"])
710
- expect(result).toBeUndefined()
711
- },
712
- })
713
- })
714
-
715
- test("getModel uses realIdByKey for aliased models", async () => {
716
- await using tmp = await tmpdir({
717
- init: async (dir) => {
718
- await Bun.write(
719
- path.join(dir, "WOOZLIT_CODE.json"),
720
- JSON.stringify({
721
- $schema: "https://opencode.ai/config.json",
722
- provider: {
723
- anthropic: {
724
- models: {
725
- "my-sonnet": {
726
- id: "claude-sonnet-4-20250514",
727
- name: "My Sonnet Alias",
728
- },
729
- },
730
- },
731
- },
732
- }),
733
- )
734
- },
735
- })
736
- await Instance.provide({
737
- directory: tmp.path,
738
- init: async () => {
739
- set("ANTHROPIC_API_KEY", "test-api-key")
740
- },
741
- fn: async () => {
742
- const providers = await list()
743
- expect(providers[ProviderID.anthropic].models["my-sonnet"]).toBeDefined()
744
-
745
- const model = await getModel(ProviderID.anthropic, ModelID.make("my-sonnet"))
746
- expect(model).toBeDefined()
747
- expect(String(model.id)).toBe("my-sonnet")
748
- expect(model.name).toBe("My Sonnet Alias")
749
- },
750
- })
751
- })
752
-
753
- test("provider api field sets model api.url", async () => {
754
- await using tmp = await tmpdir({
755
- init: async (dir) => {
756
- await Bun.write(
757
- path.join(dir, "WOOZLIT_CODE.json"),
758
- JSON.stringify({
759
- $schema: "https://opencode.ai/config.json",
760
- provider: {
761
- "custom-api": {
762
- name: "Custom API",
763
- npm: "@ai-sdk/openai-compatible",
764
- api: "https://api.example.com/v1",
765
- env: [],
766
- models: {
767
- "model-1": {
768
- name: "Model 1",
769
- tool_call: true,
770
- limit: { context: 8000, output: 2000 },
771
- },
772
- },
773
- options: {
774
- apiKey: "test-key",
775
- },
776
- },
777
- },
778
- }),
779
- )
780
- },
781
- })
782
- await Instance.provide({
783
- directory: tmp.path,
784
- fn: async () => {
785
- const providers = await list()
786
- // api field is stored on model.api.url, used by getSDK to set baseURL
787
- expect(providers[ProviderID.make("custom-api")].models["model-1"].api.url).toBe("https://api.example.com/v1")
788
- },
789
- })
790
- })
791
-
792
- test("explicit baseURL overrides api field", async () => {
793
- await using tmp = await tmpdir({
794
- init: async (dir) => {
795
- await Bun.write(
796
- path.join(dir, "WOOZLIT_CODE.json"),
797
- JSON.stringify({
798
- $schema: "https://opencode.ai/config.json",
799
- provider: {
800
- "custom-api": {
801
- name: "Custom API",
802
- npm: "@ai-sdk/openai-compatible",
803
- api: "https://api.example.com/v1",
804
- env: [],
805
- models: {
806
- "model-1": {
807
- name: "Model 1",
808
- tool_call: true,
809
- limit: { context: 8000, output: 2000 },
810
- },
811
- },
812
- options: {
813
- apiKey: "test-key",
814
- baseURL: "https://custom.override.com/v1",
815
- },
816
- },
817
- },
818
- }),
819
- )
820
- },
821
- })
822
- await Instance.provide({
823
- directory: tmp.path,
824
- fn: async () => {
825
- const providers = await list()
826
- expect(providers[ProviderID.make("custom-api")].options.baseURL).toBe("https://custom.override.com/v1")
827
- },
828
- })
829
- })
830
-
831
- test("model inherits properties from existing database model", async () => {
832
- await using tmp = await tmpdir({
833
- init: async (dir) => {
834
- await Bun.write(
835
- path.join(dir, "WOOZLIT_CODE.json"),
836
- JSON.stringify({
837
- $schema: "https://opencode.ai/config.json",
838
- provider: {
839
- anthropic: {
840
- models: {
841
- "claude-sonnet-4-20250514": {
842
- name: "Custom Name for Sonnet",
843
- },
844
- },
845
- },
846
- },
847
- }),
848
- )
849
- },
850
- })
851
- await Instance.provide({
852
- directory: tmp.path,
853
- init: async () => {
854
- set("ANTHROPIC_API_KEY", "test-api-key")
855
- },
856
- fn: async () => {
857
- const providers = await list()
858
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
859
- expect(model.name).toBe("Custom Name for Sonnet")
860
- expect(model.capabilities.toolcall).toBe(true)
861
- expect(model.capabilities.attachment).toBe(true)
862
- expect(model.limit.context).toBeGreaterThan(0)
863
- },
864
- })
865
- })
866
-
867
- test("disabled_providers prevents loading even with env var", async () => {
868
- await using tmp = await tmpdir({
869
- init: async (dir) => {
870
- await Bun.write(
871
- path.join(dir, "WOOZLIT_CODE.json"),
872
- JSON.stringify({
873
- $schema: "https://opencode.ai/config.json",
874
- disabled_providers: ["openai"],
875
- }),
876
- )
877
- },
878
- })
879
- await Instance.provide({
880
- directory: tmp.path,
881
- init: async () => {
882
- set("OPENAI_API_KEY", "test-openai-key")
883
- },
884
- fn: async () => {
885
- const providers = await list()
886
- expect(providers[ProviderID.openai]).toBeUndefined()
887
- },
888
- })
889
- })
890
-
891
- test("enabled_providers with empty array allows no providers", async () => {
892
- await using tmp = await tmpdir({
893
- init: async (dir) => {
894
- await Bun.write(
895
- path.join(dir, "WOOZLIT_CODE.json"),
896
- JSON.stringify({
897
- $schema: "https://opencode.ai/config.json",
898
- enabled_providers: [],
899
- }),
900
- )
901
- },
902
- })
903
- await Instance.provide({
904
- directory: tmp.path,
905
- init: async () => {
906
- set("ANTHROPIC_API_KEY", "test-api-key")
907
- set("OPENAI_API_KEY", "test-openai-key")
908
- },
909
- fn: async () => {
910
- const providers = await list()
911
- expect(Object.keys(providers).length).toBe(0)
912
- },
913
- })
914
- })
915
-
916
- test("whitelist and blacklist can be combined", async () => {
917
- await using tmp = await tmpdir({
918
- init: async (dir) => {
919
- await Bun.write(
920
- path.join(dir, "WOOZLIT_CODE.json"),
921
- JSON.stringify({
922
- $schema: "https://opencode.ai/config.json",
923
- provider: {
924
- anthropic: {
925
- whitelist: ["claude-sonnet-4-20250514", "claude-opus-4-20250514"],
926
- blacklist: ["claude-opus-4-20250514"],
927
- },
928
- },
929
- }),
930
- )
931
- },
932
- })
933
- await Instance.provide({
934
- directory: tmp.path,
935
- init: async () => {
936
- set("ANTHROPIC_API_KEY", "test-api-key")
937
- },
938
- fn: async () => {
939
- const providers = await list()
940
- expect(providers[ProviderID.anthropic]).toBeDefined()
941
- const models = Object.keys(providers[ProviderID.anthropic].models)
942
- expect(models).toContain("claude-sonnet-4-20250514")
943
- expect(models).not.toContain("claude-opus-4-20250514")
944
- expect(models.length).toBe(1)
945
- },
946
- })
947
- })
948
-
949
- test("model modalities default correctly", async () => {
950
- await using tmp = await tmpdir({
951
- init: async (dir) => {
952
- await Bun.write(
953
- path.join(dir, "WOOZLIT_CODE.json"),
954
- JSON.stringify({
955
- $schema: "https://opencode.ai/config.json",
956
- provider: {
957
- "test-provider": {
958
- name: "Test",
959
- npm: "@ai-sdk/openai-compatible",
960
- env: [],
961
- models: {
962
- "test-model": {
963
- name: "Test Model",
964
- tool_call: true,
965
- limit: { context: 8000, output: 2000 },
966
- },
967
- },
968
- options: { apiKey: "test" },
969
- },
970
- },
971
- }),
972
- )
973
- },
974
- })
975
- await Instance.provide({
976
- directory: tmp.path,
977
- fn: async () => {
978
- const providers = await list()
979
- const model = providers[ProviderID.make("test-provider")].models["test-model"]
980
- expect(model.capabilities.input.text).toBe(true)
981
- expect(model.capabilities.output.text).toBe(true)
982
- },
983
- })
984
- })
985
-
986
- test("model with custom cost values", async () => {
987
- await using tmp = await tmpdir({
988
- init: async (dir) => {
989
- await Bun.write(
990
- path.join(dir, "WOOZLIT_CODE.json"),
991
- JSON.stringify({
992
- $schema: "https://opencode.ai/config.json",
993
- provider: {
994
- "test-provider": {
995
- name: "Test",
996
- npm: "@ai-sdk/openai-compatible",
997
- env: [],
998
- models: {
999
- "test-model": {
1000
- name: "Test Model",
1001
- tool_call: true,
1002
- limit: { context: 8000, output: 2000 },
1003
- cost: {
1004
- input: 5,
1005
- output: 15,
1006
- cache_read: 2.5,
1007
- cache_write: 7.5,
1008
- },
1009
- },
1010
- },
1011
- options: { apiKey: "test" },
1012
- },
1013
- },
1014
- }),
1015
- )
1016
- },
1017
- })
1018
- await Instance.provide({
1019
- directory: tmp.path,
1020
- fn: async () => {
1021
- const providers = await list()
1022
- const model = providers[ProviderID.make("test-provider")].models["test-model"]
1023
- expect(model.cost.input).toBe(5)
1024
- expect(model.cost.output).toBe(15)
1025
- expect(model.cost.cache.read).toBe(2.5)
1026
- expect(model.cost.cache.write).toBe(7.5)
1027
- },
1028
- })
1029
- })
1030
-
1031
- // getSmallModel routes through the `lite` tier: an explicit `small_model`
1032
- // literal wins first (back-compat), otherwise it resolves resolveModelRef("lite")
1033
- // — a configured `lite` group (provider-aware), else the built-in tier fallback
1034
- // to the default model. The old hardcoded heuristic (haiku/flash/nano scan) is
1035
- // gone. Both tests use a fully config-declared provider so they don't depend on
1036
- // env-keyed autoload of models.dev catalogs (flaky/unavailable in this sandbox).
1037
- const SMALL_MODEL_PROVIDER = {
1038
- small: {
1039
- name: "Small",
1040
- npm: "@ai-sdk/openai-compatible",
1041
- env: [],
1042
- models: {
1043
- "small-lite": { name: "Small Lite", tool_call: true, limit: { context: 8000, output: 2000 } },
1044
- "small-big": { name: "Small Big", tool_call: true, limit: { context: 8000, output: 2000 } },
1045
- },
1046
- options: { apiKey: "test-key" },
1047
- },
1048
- }
1049
-
1050
- test("getSmallModel resolves via the lite group", async () => {
1051
- await using tmp = await tmpdir({
1052
- init: async (dir) => {
1053
- await Bun.write(
1054
- path.join(dir, "WOOZLIT_CODE.json"),
1055
- JSON.stringify({
1056
- $schema: "https://opencode.ai/config.json",
1057
- provider: SMALL_MODEL_PROVIDER,
1058
- model_groups: {
1059
- lite: "small/small-lite",
1060
- },
1061
- }),
1062
- )
1063
- },
1064
- })
1065
- await Instance.provide({
1066
- directory: tmp.path,
1067
- fn: async () => {
1068
- const model = await getSmallModel(ProviderID.make("small"))
1069
- expect(model).toBeDefined()
1070
- expect(String(model?.providerID)).toBe("small")
1071
- expect(String(model?.id)).toBe("small-lite")
1072
- },
1073
- })
1074
- })
1075
-
1076
- test("getSmallModel respects config small_model override", async () => {
1077
- await using tmp = await tmpdir({
1078
- init: async (dir) => {
1079
- await Bun.write(
1080
- path.join(dir, "WOOZLIT_CODE.json"),
1081
- JSON.stringify({
1082
- $schema: "https://opencode.ai/config.json",
1083
- provider: SMALL_MODEL_PROVIDER,
1084
- small_model: "small/small-big",
1085
- // A lite group exists too — small_model must still win (first branch).
1086
- model_groups: {
1087
- lite: "small/small-lite",
1088
- },
1089
- }),
1090
- )
1091
- },
1092
- })
1093
- await Instance.provide({
1094
- directory: tmp.path,
1095
- fn: async () => {
1096
- const model = await getSmallModel(ProviderID.make("small"))
1097
- expect(model).toBeDefined()
1098
- expect(String(model?.providerID)).toBe("small")
1099
- expect(String(model?.id)).toBe("small-big")
1100
- },
1101
- })
1102
- })
1103
-
1104
- test("provider.sort prioritizes preferred models", () => {
1105
- const models = [
1106
- { id: "random-model", name: "Random" },
1107
- { id: "claude-sonnet-4-latest", name: "Claude Sonnet 4" },
1108
- { id: "gpt-5-turbo", name: "GPT-5 Turbo" },
1109
- { id: "other-model", name: "Other" },
1110
- ] as any[]
1111
-
1112
- const sorted = Provider.sort(models)
1113
- expect(sorted[0].id).toContain("sonnet-4")
1114
- expect(sorted[0].id).toContain("latest")
1115
- expect(sorted[sorted.length - 1].id).not.toContain("gpt-5")
1116
- expect(sorted[sorted.length - 1].id).not.toContain("sonnet-4")
1117
- })
1118
-
1119
- test("multiple providers can be configured simultaneously", async () => {
1120
- await using tmp = await tmpdir({
1121
- init: async (dir) => {
1122
- await Bun.write(
1123
- path.join(dir, "WOOZLIT_CODE.json"),
1124
- JSON.stringify({
1125
- $schema: "https://opencode.ai/config.json",
1126
- provider: {
1127
- anthropic: {
1128
- options: { timeout: 30000 },
1129
- },
1130
- openai: {
1131
- options: { timeout: 60000 },
1132
- },
1133
- },
1134
- }),
1135
- )
1136
- },
1137
- })
1138
- await Instance.provide({
1139
- directory: tmp.path,
1140
- init: async () => {
1141
- set("ANTHROPIC_API_KEY", "test-anthropic-key")
1142
- set("OPENAI_API_KEY", "test-openai-key")
1143
- },
1144
- fn: async () => {
1145
- const providers = await list()
1146
- expect(providers[ProviderID.anthropic]).toBeDefined()
1147
- expect(providers[ProviderID.openai]).toBeDefined()
1148
- expect(providers[ProviderID.anthropic].options.timeout).toBe(30000)
1149
- expect(providers[ProviderID.openai].options.timeout).toBe(60000)
1150
- },
1151
- })
1152
- })
1153
-
1154
- test("provider with custom npm package", async () => {
1155
- await using tmp = await tmpdir({
1156
- init: async (dir) => {
1157
- await Bun.write(
1158
- path.join(dir, "WOOZLIT_CODE.json"),
1159
- JSON.stringify({
1160
- $schema: "https://opencode.ai/config.json",
1161
- provider: {
1162
- "local-llm": {
1163
- name: "Local LLM",
1164
- npm: "@ai-sdk/openai-compatible",
1165
- env: [],
1166
- models: {
1167
- "llama-3": {
1168
- name: "Llama 3",
1169
- tool_call: true,
1170
- limit: { context: 8192, output: 2048 },
1171
- },
1172
- },
1173
- options: {
1174
- apiKey: "not-needed",
1175
- baseURL: "http://localhost:11434/v1",
1176
- },
1177
- },
1178
- },
1179
- }),
1180
- )
1181
- },
1182
- })
1183
- await Instance.provide({
1184
- directory: tmp.path,
1185
- fn: async () => {
1186
- const providers = await list()
1187
- expect(providers[ProviderID.make("local-llm")]).toBeDefined()
1188
- expect(providers[ProviderID.make("local-llm")].models["llama-3"].api.npm).toBe("@ai-sdk/openai-compatible")
1189
- expect(providers[ProviderID.make("local-llm")].options.baseURL).toBe("http://localhost:11434/v1")
1190
- },
1191
- })
1192
- })
1193
-
1194
- // Edge cases for model configuration
1195
-
1196
- test("model alias name defaults to alias key when id differs", async () => {
1197
- await using tmp = await tmpdir({
1198
- init: async (dir) => {
1199
- await Bun.write(
1200
- path.join(dir, "WOOZLIT_CODE.json"),
1201
- JSON.stringify({
1202
- $schema: "https://opencode.ai/config.json",
1203
- provider: {
1204
- anthropic: {
1205
- models: {
1206
- sonnet: {
1207
- id: "claude-sonnet-4-20250514",
1208
- // no name specified - should default to "sonnet" (the key)
1209
- },
1210
- },
1211
- },
1212
- },
1213
- }),
1214
- )
1215
- },
1216
- })
1217
- await Instance.provide({
1218
- directory: tmp.path,
1219
- init: async () => {
1220
- set("ANTHROPIC_API_KEY", "test-api-key")
1221
- },
1222
- fn: async () => {
1223
- const providers = await list()
1224
- expect(providers[ProviderID.anthropic].models["sonnet"].name).toBe("sonnet")
1225
- },
1226
- })
1227
- })
1228
-
1229
- test("provider with multiple env var options only includes apiKey when single env", async () => {
1230
- await using tmp = await tmpdir({
1231
- init: async (dir) => {
1232
- await Bun.write(
1233
- path.join(dir, "WOOZLIT_CODE.json"),
1234
- JSON.stringify({
1235
- $schema: "https://opencode.ai/config.json",
1236
- provider: {
1237
- "multi-env": {
1238
- name: "Multi Env Provider",
1239
- npm: "@ai-sdk/openai-compatible",
1240
- env: ["MULTI_ENV_KEY_1", "MULTI_ENV_KEY_2"],
1241
- models: {
1242
- "model-1": {
1243
- name: "Model 1",
1244
- tool_call: true,
1245
- limit: { context: 8000, output: 2000 },
1246
- },
1247
- },
1248
- options: {
1249
- baseURL: "https://api.example.com/v1",
1250
- },
1251
- },
1252
- },
1253
- }),
1254
- )
1255
- },
1256
- })
1257
- await Instance.provide({
1258
- directory: tmp.path,
1259
- init: async () => {
1260
- set("MULTI_ENV_KEY_1", "test-key")
1261
- },
1262
- fn: async () => {
1263
- const providers = await list()
1264
- expect(providers[ProviderID.make("multi-env")]).toBeDefined()
1265
- // When multiple env options exist, key should NOT be auto-set
1266
- expect(providers[ProviderID.make("multi-env")].key).toBeUndefined()
1267
- },
1268
- })
1269
- })
1270
-
1271
- test("provider with single env var includes apiKey automatically", async () => {
1272
- await using tmp = await tmpdir({
1273
- init: async (dir) => {
1274
- await Bun.write(
1275
- path.join(dir, "WOOZLIT_CODE.json"),
1276
- JSON.stringify({
1277
- $schema: "https://opencode.ai/config.json",
1278
- provider: {
1279
- "single-env": {
1280
- name: "Single Env Provider",
1281
- npm: "@ai-sdk/openai-compatible",
1282
- env: ["SINGLE_ENV_KEY"],
1283
- models: {
1284
- "model-1": {
1285
- name: "Model 1",
1286
- tool_call: true,
1287
- limit: { context: 8000, output: 2000 },
1288
- },
1289
- },
1290
- options: {
1291
- baseURL: "https://api.example.com/v1",
1292
- },
1293
- },
1294
- },
1295
- }),
1296
- )
1297
- },
1298
- })
1299
- await Instance.provide({
1300
- directory: tmp.path,
1301
- init: async () => {
1302
- set("SINGLE_ENV_KEY", "my-api-key")
1303
- },
1304
- fn: async () => {
1305
- const providers = await list()
1306
- expect(providers[ProviderID.make("single-env")]).toBeDefined()
1307
- // Single env option should auto-set key
1308
- expect(providers[ProviderID.make("single-env")].key).toBe("my-api-key")
1309
- },
1310
- })
1311
- })
1312
-
1313
- test("model cost overrides existing cost values", async () => {
1314
- await using tmp = await tmpdir({
1315
- init: async (dir) => {
1316
- await Bun.write(
1317
- path.join(dir, "WOOZLIT_CODE.json"),
1318
- JSON.stringify({
1319
- $schema: "https://opencode.ai/config.json",
1320
- provider: {
1321
- anthropic: {
1322
- models: {
1323
- "claude-sonnet-4-20250514": {
1324
- cost: {
1325
- input: 999,
1326
- output: 888,
1327
- },
1328
- },
1329
- },
1330
- },
1331
- },
1332
- }),
1333
- )
1334
- },
1335
- })
1336
- await Instance.provide({
1337
- directory: tmp.path,
1338
- init: async () => {
1339
- set("ANTHROPIC_API_KEY", "test-api-key")
1340
- },
1341
- fn: async () => {
1342
- const providers = await list()
1343
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
1344
- expect(model.cost.input).toBe(999)
1345
- expect(model.cost.output).toBe(888)
1346
- },
1347
- })
1348
- })
1349
-
1350
- test("completely new provider not in database can be configured", async () => {
1351
- await using tmp = await tmpdir({
1352
- init: async (dir) => {
1353
- await Bun.write(
1354
- path.join(dir, "WOOZLIT_CODE.json"),
1355
- JSON.stringify({
1356
- $schema: "https://opencode.ai/config.json",
1357
- provider: {
1358
- "brand-new-provider": {
1359
- name: "Brand New",
1360
- npm: "@ai-sdk/openai-compatible",
1361
- env: [],
1362
- api: "https://new-api.com/v1",
1363
- models: {
1364
- "new-model": {
1365
- name: "New Model",
1366
- tool_call: true,
1367
- reasoning: true,
1368
- attachment: true,
1369
- temperature: true,
1370
- limit: { context: 32000, output: 8000 },
1371
- modalities: {
1372
- input: ["text", "image"],
1373
- output: ["text"],
1374
- },
1375
- },
1376
- },
1377
- options: {
1378
- apiKey: "new-key",
1379
- },
1380
- },
1381
- },
1382
- }),
1383
- )
1384
- },
1385
- })
1386
- await Instance.provide({
1387
- directory: tmp.path,
1388
- fn: async () => {
1389
- const providers = await list()
1390
- expect(providers[ProviderID.make("brand-new-provider")]).toBeDefined()
1391
- expect(providers[ProviderID.make("brand-new-provider")].name).toBe("Brand New")
1392
- const model = providers[ProviderID.make("brand-new-provider")].models["new-model"]
1393
- expect(model.capabilities.reasoning).toBe(true)
1394
- expect(model.capabilities.attachment).toBe(true)
1395
- expect(model.capabilities.input.image).toBe(true)
1396
- },
1397
- })
1398
- })
1399
-
1400
- test("disabled_providers and enabled_providers interaction", async () => {
1401
- await using tmp = await tmpdir({
1402
- init: async (dir) => {
1403
- await Bun.write(
1404
- path.join(dir, "WOOZLIT_CODE.json"),
1405
- JSON.stringify({
1406
- $schema: "https://opencode.ai/config.json",
1407
- // enabled_providers takes precedence - only these are considered
1408
- enabled_providers: ["anthropic", "openai"],
1409
- // Then disabled_providers filters from the enabled set
1410
- disabled_providers: ["openai"],
1411
- }),
1412
- )
1413
- },
1414
- })
1415
- await Instance.provide({
1416
- directory: tmp.path,
1417
- init: async () => {
1418
- set("ANTHROPIC_API_KEY", "test-anthropic")
1419
- set("OPENAI_API_KEY", "test-openai")
1420
- set("GOOGLE_GENERATIVE_AI_API_KEY", "test-google")
1421
- },
1422
- fn: async () => {
1423
- const providers = await list()
1424
- // anthropic: in enabled, not in disabled = allowed
1425
- expect(providers[ProviderID.anthropic]).toBeDefined()
1426
- // openai: in enabled, but also in disabled = NOT allowed
1427
- expect(providers[ProviderID.openai]).toBeUndefined()
1428
- // google: not in enabled = NOT allowed (even though not disabled)
1429
- expect(providers[ProviderID.google]).toBeUndefined()
1430
- },
1431
- })
1432
- })
1433
-
1434
- test("model with tool_call false", async () => {
1435
- await using tmp = await tmpdir({
1436
- init: async (dir) => {
1437
- await Bun.write(
1438
- path.join(dir, "WOOZLIT_CODE.json"),
1439
- JSON.stringify({
1440
- $schema: "https://opencode.ai/config.json",
1441
- provider: {
1442
- "no-tools": {
1443
- name: "No Tools Provider",
1444
- npm: "@ai-sdk/openai-compatible",
1445
- env: [],
1446
- models: {
1447
- "basic-model": {
1448
- name: "Basic Model",
1449
- tool_call: false,
1450
- limit: { context: 4000, output: 1000 },
1451
- },
1452
- },
1453
- options: { apiKey: "test" },
1454
- },
1455
- },
1456
- }),
1457
- )
1458
- },
1459
- })
1460
- await Instance.provide({
1461
- directory: tmp.path,
1462
- fn: async () => {
1463
- const providers = await list()
1464
- expect(providers[ProviderID.make("no-tools")].models["basic-model"].capabilities.toolcall).toBe(false)
1465
- },
1466
- })
1467
- })
1468
-
1469
- test("model defaults tool_call to true when not specified", async () => {
1470
- await using tmp = await tmpdir({
1471
- init: async (dir) => {
1472
- await Bun.write(
1473
- path.join(dir, "WOOZLIT_CODE.json"),
1474
- JSON.stringify({
1475
- $schema: "https://opencode.ai/config.json",
1476
- provider: {
1477
- "default-tools": {
1478
- name: "Default Tools Provider",
1479
- npm: "@ai-sdk/openai-compatible",
1480
- env: [],
1481
- models: {
1482
- model: {
1483
- name: "Model",
1484
- // tool_call not specified
1485
- limit: { context: 4000, output: 1000 },
1486
- },
1487
- },
1488
- options: { apiKey: "test" },
1489
- },
1490
- },
1491
- }),
1492
- )
1493
- },
1494
- })
1495
- await Instance.provide({
1496
- directory: tmp.path,
1497
- fn: async () => {
1498
- const providers = await list()
1499
- expect(providers[ProviderID.make("default-tools")].models["model"].capabilities.toolcall).toBe(true)
1500
- },
1501
- })
1502
- })
1503
-
1504
- test("model headers are preserved", async () => {
1505
- await using tmp = await tmpdir({
1506
- init: async (dir) => {
1507
- await Bun.write(
1508
- path.join(dir, "WOOZLIT_CODE.json"),
1509
- JSON.stringify({
1510
- $schema: "https://opencode.ai/config.json",
1511
- provider: {
1512
- "headers-provider": {
1513
- name: "Headers Provider",
1514
- npm: "@ai-sdk/openai-compatible",
1515
- env: [],
1516
- models: {
1517
- model: {
1518
- name: "Model",
1519
- tool_call: true,
1520
- limit: { context: 4000, output: 1000 },
1521
- headers: {
1522
- "X-Custom-Header": "custom-value",
1523
- Authorization: "Bearer special-token",
1524
- },
1525
- },
1526
- },
1527
- options: { apiKey: "test" },
1528
- },
1529
- },
1530
- }),
1531
- )
1532
- },
1533
- })
1534
- await Instance.provide({
1535
- directory: tmp.path,
1536
- fn: async () => {
1537
- const providers = await list()
1538
- const model = providers[ProviderID.make("headers-provider")].models["model"]
1539
- expect(model.headers).toEqual({
1540
- "X-Custom-Header": "custom-value",
1541
- Authorization: "Bearer special-token",
1542
- })
1543
- },
1544
- })
1545
- })
1546
-
1547
- test("provider env fallback - second env var used if first missing", async () => {
1548
- await using tmp = await tmpdir({
1549
- init: async (dir) => {
1550
- await Bun.write(
1551
- path.join(dir, "WOOZLIT_CODE.json"),
1552
- JSON.stringify({
1553
- $schema: "https://opencode.ai/config.json",
1554
- provider: {
1555
- "fallback-env": {
1556
- name: "Fallback Env Provider",
1557
- npm: "@ai-sdk/openai-compatible",
1558
- env: ["PRIMARY_KEY", "FALLBACK_KEY"],
1559
- models: {
1560
- model: {
1561
- name: "Model",
1562
- tool_call: true,
1563
- limit: { context: 4000, output: 1000 },
1564
- },
1565
- },
1566
- options: { baseURL: "https://api.example.com" },
1567
- },
1568
- },
1569
- }),
1570
- )
1571
- },
1572
- })
1573
- await Instance.provide({
1574
- directory: tmp.path,
1575
- init: async () => {
1576
- // Only set fallback, not primary
1577
- set("FALLBACK_KEY", "fallback-api-key")
1578
- },
1579
- fn: async () => {
1580
- const providers = await list()
1581
- // Provider should load because fallback env var is set
1582
- expect(providers[ProviderID.make("fallback-env")]).toBeDefined()
1583
- },
1584
- })
1585
- })
1586
-
1587
- test("getModel returns consistent results", async () => {
1588
- await using tmp = await tmpdir({
1589
- init: async (dir) => {
1590
- await Bun.write(
1591
- path.join(dir, "WOOZLIT_CODE.json"),
1592
- JSON.stringify({
1593
- $schema: "https://opencode.ai/config.json",
1594
- }),
1595
- )
1596
- },
1597
- })
1598
- await Instance.provide({
1599
- directory: tmp.path,
1600
- init: async () => {
1601
- set("ANTHROPIC_API_KEY", "test-api-key")
1602
- },
1603
- fn: async () => {
1604
- const model1 = await getModel(ProviderID.anthropic, ModelID.make("claude-sonnet-4-20250514"))
1605
- const model2 = await getModel(ProviderID.anthropic, ModelID.make("claude-sonnet-4-20250514"))
1606
- expect(model1.providerID).toEqual(model2.providerID)
1607
- expect(model1.id).toEqual(model2.id)
1608
- expect(model1).toEqual(model2)
1609
- },
1610
- })
1611
- })
1612
-
1613
- test("provider name defaults to id when not in database", async () => {
1614
- await using tmp = await tmpdir({
1615
- init: async (dir) => {
1616
- await Bun.write(
1617
- path.join(dir, "WOOZLIT_CODE.json"),
1618
- JSON.stringify({
1619
- $schema: "https://opencode.ai/config.json",
1620
- provider: {
1621
- "my-custom-id": {
1622
- // no name specified
1623
- npm: "@ai-sdk/openai-compatible",
1624
- env: [],
1625
- models: {
1626
- model: {
1627
- name: "Model",
1628
- tool_call: true,
1629
- limit: { context: 4000, output: 1000 },
1630
- },
1631
- },
1632
- options: { apiKey: "test" },
1633
- },
1634
- },
1635
- }),
1636
- )
1637
- },
1638
- })
1639
- await Instance.provide({
1640
- directory: tmp.path,
1641
- fn: async () => {
1642
- const providers = await list()
1643
- expect(providers[ProviderID.make("my-custom-id")].name).toBe("my-custom-id")
1644
- },
1645
- })
1646
- })
1647
-
1648
- test("ModelNotFoundError includes suggestions for typos", async () => {
1649
- await using tmp = await tmpdir({
1650
- init: async (dir) => {
1651
- await Bun.write(
1652
- path.join(dir, "WOOZLIT_CODE.json"),
1653
- JSON.stringify({
1654
- $schema: "https://opencode.ai/config.json",
1655
- }),
1656
- )
1657
- },
1658
- })
1659
- await Instance.provide({
1660
- directory: tmp.path,
1661
- init: async () => {
1662
- set("ANTHROPIC_API_KEY", "test-api-key")
1663
- },
1664
- fn: async () => {
1665
- try {
1666
- await getModel(ProviderID.anthropic, ModelID.make("claude-sonet-4")) // typo: sonet instead of sonnet
1667
- expect(true).toBe(false) // Should not reach here
1668
- } catch (e: any) {
1669
- expect(e.data.suggestions).toBeDefined()
1670
- expect(e.data.suggestions.length).toBeGreaterThan(0)
1671
- }
1672
- },
1673
- })
1674
- })
1675
-
1676
- test("ModelNotFoundError for provider includes suggestions", async () => {
1677
- await using tmp = await tmpdir({
1678
- init: async (dir) => {
1679
- await Bun.write(
1680
- path.join(dir, "WOOZLIT_CODE.json"),
1681
- JSON.stringify({
1682
- $schema: "https://opencode.ai/config.json",
1683
- }),
1684
- )
1685
- },
1686
- })
1687
- await Instance.provide({
1688
- directory: tmp.path,
1689
- init: async () => {
1690
- set("ANTHROPIC_API_KEY", "test-api-key")
1691
- },
1692
- fn: async () => {
1693
- try {
1694
- await getModel(ProviderID.make("antropic"), ModelID.make("claude-sonnet-4")) // typo: antropic
1695
- expect(true).toBe(false) // Should not reach here
1696
- } catch (e: any) {
1697
- expect(e.data.suggestions).toBeDefined()
1698
- expect(e.data.suggestions).toContain("anthropic")
1699
- }
1700
- },
1701
- })
1702
- })
1703
-
1704
- test("getProvider returns undefined for nonexistent provider", async () => {
1705
- await using tmp = await tmpdir({
1706
- init: async (dir) => {
1707
- await Bun.write(
1708
- path.join(dir, "WOOZLIT_CODE.json"),
1709
- JSON.stringify({
1710
- $schema: "https://opencode.ai/config.json",
1711
- }),
1712
- )
1713
- },
1714
- })
1715
- await Instance.provide({
1716
- directory: tmp.path,
1717
- fn: async () => {
1718
- const provider = await getProvider(ProviderID.make("nonexistent"))
1719
- expect(provider).toBeUndefined()
1720
- },
1721
- })
1722
- })
1723
-
1724
- test("getProvider returns provider info", async () => {
1725
- await using tmp = await tmpdir({
1726
- init: async (dir) => {
1727
- await Bun.write(
1728
- path.join(dir, "WOOZLIT_CODE.json"),
1729
- JSON.stringify({
1730
- $schema: "https://opencode.ai/config.json",
1731
- }),
1732
- )
1733
- },
1734
- })
1735
- await Instance.provide({
1736
- directory: tmp.path,
1737
- init: async () => {
1738
- set("ANTHROPIC_API_KEY", "test-api-key")
1739
- },
1740
- fn: async () => {
1741
- const provider = await getProvider(ProviderID.anthropic)
1742
- expect(provider).toBeDefined()
1743
- expect(String(provider?.id)).toBe("anthropic")
1744
- },
1745
- })
1746
- })
1747
-
1748
- test("closest returns undefined when no partial match found", async () => {
1749
- await using tmp = await tmpdir({
1750
- init: async (dir) => {
1751
- await Bun.write(
1752
- path.join(dir, "WOOZLIT_CODE.json"),
1753
- JSON.stringify({
1754
- $schema: "https://opencode.ai/config.json",
1755
- }),
1756
- )
1757
- },
1758
- })
1759
- await Instance.provide({
1760
- directory: tmp.path,
1761
- init: async () => {
1762
- set("ANTHROPIC_API_KEY", "test-api-key")
1763
- },
1764
- fn: async () => {
1765
- const result = await closest(ProviderID.anthropic, ["nonexistent-xyz-model"])
1766
- expect(result).toBeUndefined()
1767
- },
1768
- })
1769
- })
1770
-
1771
- test("closest checks multiple query terms in order", async () => {
1772
- await using tmp = await tmpdir({
1773
- init: async (dir) => {
1774
- await Bun.write(
1775
- path.join(dir, "WOOZLIT_CODE.json"),
1776
- JSON.stringify({
1777
- $schema: "https://opencode.ai/config.json",
1778
- }),
1779
- )
1780
- },
1781
- })
1782
- await Instance.provide({
1783
- directory: tmp.path,
1784
- init: async () => {
1785
- set("ANTHROPIC_API_KEY", "test-api-key")
1786
- },
1787
- fn: async () => {
1788
- // First term won't match, second will
1789
- const result = await closest(ProviderID.anthropic, ["nonexistent", "haiku"])
1790
- expect(result).toBeDefined()
1791
- expect(result?.modelID).toContain("haiku")
1792
- },
1793
- })
1794
- })
1795
-
1796
- test("model limit defaults to DEFAULT_CONTEXT_WINDOW (200K) when not specified (F41)", async () => {
1797
- await using tmp = await tmpdir({
1798
- init: async (dir) => {
1799
- await Bun.write(
1800
- path.join(dir, "WOOZLIT_CODE.json"),
1801
- JSON.stringify({
1802
- $schema: "https://opencode.ai/config.json",
1803
- provider: {
1804
- "no-limit": {
1805
- name: "No Limit Provider",
1806
- npm: "@ai-sdk/openai-compatible",
1807
- env: [],
1808
- models: {
1809
- model: {
1810
- name: "Model",
1811
- tool_call: true,
1812
- // no limit specified
1813
- },
1814
- },
1815
- options: { apiKey: "test" },
1816
- },
1817
- },
1818
- }),
1819
- )
1820
- },
1821
- })
1822
- await Instance.provide({
1823
- directory: tmp.path,
1824
- fn: async () => {
1825
- const providers = await list()
1826
- const model = providers[ProviderID.make("no-limit")].models["model"]
1827
- expect(model.limit.context).toBe(200_000)
1828
- expect(model.limit.output).toBe(0)
1829
- },
1830
- })
1831
- })
1832
-
1833
- test("provider options are deeply merged", async () => {
1834
- await using tmp = await tmpdir({
1835
- init: async (dir) => {
1836
- await Bun.write(
1837
- path.join(dir, "WOOZLIT_CODE.json"),
1838
- JSON.stringify({
1839
- $schema: "https://opencode.ai/config.json",
1840
- provider: {
1841
- anthropic: {
1842
- options: {
1843
- headers: {
1844
- "X-Custom": "custom-value",
1845
- },
1846
- timeout: 30000,
1847
- },
1848
- },
1849
- },
1850
- }),
1851
- )
1852
- },
1853
- })
1854
- await Instance.provide({
1855
- directory: tmp.path,
1856
- init: async () => {
1857
- set("ANTHROPIC_API_KEY", "test-api-key")
1858
- },
1859
- fn: async () => {
1860
- const providers = await list()
1861
- // Custom options should be merged
1862
- expect(providers[ProviderID.anthropic].options.timeout).toBe(30000)
1863
- expect(providers[ProviderID.anthropic].options.headers["X-Custom"]).toBe("custom-value")
1864
- // anthropic custom loader adds its own headers, they should coexist
1865
- expect(providers[ProviderID.anthropic].options.headers["anthropic-beta"]).toBeDefined()
1866
- },
1867
- })
1868
- })
1869
-
1870
- test("custom model inherits npm package from models.dev provider config", async () => {
1871
- await using tmp = await tmpdir({
1872
- init: async (dir) => {
1873
- await Bun.write(
1874
- path.join(dir, "WOOZLIT_CODE.json"),
1875
- JSON.stringify({
1876
- $schema: "https://opencode.ai/config.json",
1877
- provider: {
1878
- openai: {
1879
- models: {
1880
- "my-custom-model": {
1881
- name: "My Custom Model",
1882
- tool_call: true,
1883
- limit: { context: 8000, output: 2000 },
1884
- },
1885
- },
1886
- },
1887
- },
1888
- }),
1889
- )
1890
- },
1891
- })
1892
- await Instance.provide({
1893
- directory: tmp.path,
1894
- init: async () => {
1895
- set("OPENAI_API_KEY", "test-api-key")
1896
- },
1897
- fn: async () => {
1898
- const providers = await list()
1899
- const model = providers[ProviderID.openai].models["my-custom-model"]
1900
- expect(model).toBeDefined()
1901
- expect(model.api.npm).toBe("@ai-sdk/openai")
1902
- },
1903
- })
1904
- })
1905
-
1906
- test("custom model inherits api.url from models.dev provider", async () => {
1907
- await using tmp = await tmpdir({
1908
- init: async (dir) => {
1909
- await Bun.write(
1910
- path.join(dir, "WOOZLIT_CODE.json"),
1911
- JSON.stringify({
1912
- $schema: "https://opencode.ai/config.json",
1913
- provider: {
1914
- openrouter: {
1915
- models: {
1916
- "prime-intellect/intellect-3": {},
1917
- "deepseek/deepseek-r1-0528": {
1918
- name: "DeepSeek R1",
1919
- },
1920
- },
1921
- },
1922
- },
1923
- }),
1924
- )
1925
- },
1926
- })
1927
- await Instance.provide({
1928
- directory: tmp.path,
1929
- init: async () => {
1930
- set("OPENROUTER_API_KEY", "test-api-key")
1931
- },
1932
- fn: async () => {
1933
- const providers = await list()
1934
- expect(providers[ProviderID.openrouter]).toBeDefined()
1935
-
1936
- // New model not in database should inherit api.url from provider
1937
- const intellect = providers[ProviderID.openrouter].models["prime-intellect/intellect-3"]
1938
- expect(intellect).toBeDefined()
1939
- expect(intellect.api.url).toBe("https://openrouter.ai/api/v1")
1940
-
1941
- // Another new model should also inherit api.url
1942
- const deepseek = providers[ProviderID.openrouter].models["deepseek/deepseek-r1-0528"]
1943
- expect(deepseek).toBeDefined()
1944
- expect(deepseek.api.url).toBe("https://openrouter.ai/api/v1")
1945
- expect(deepseek.name).toBe("DeepSeek R1")
1946
- },
1947
- })
1948
- })
1949
-
1950
- test("mode cost preserves over-200k pricing from base model", () => {
1951
- const provider = {
1952
- id: "openai",
1953
- name: "OpenAI",
1954
- env: [],
1955
- api: "https://api.openai.com/v1",
1956
- models: {
1957
- "gpt-5.4": {
1958
- id: "gpt-5.4",
1959
- name: "GPT-5.4",
1960
- family: "gpt",
1961
- release_date: "2026-03-05",
1962
- attachment: true,
1963
- reasoning: true,
1964
- temperature: false,
1965
- tool_call: true,
1966
- cost: {
1967
- input: 2.5,
1968
- output: 15,
1969
- cache_read: 0.25,
1970
- context_over_200k: {
1971
- input: 5,
1972
- output: 22.5,
1973
- cache_read: 0.5,
1974
- },
1975
- },
1976
- limit: {
1977
- context: 1_050_000,
1978
- input: 922_000,
1979
- output: 128_000,
1980
- },
1981
- experimental: {
1982
- modes: {
1983
- fast: {
1984
- cost: {
1985
- input: 5,
1986
- output: 30,
1987
- cache_read: 0.5,
1988
- },
1989
- provider: {
1990
- body: {
1991
- service_tier: "priority",
1992
- },
1993
- },
1994
- },
1995
- },
1996
- },
1997
- },
1998
- },
1999
- } as unknown as ModelsDev.Provider
2000
-
2001
- const model = Provider.fromModelsDevProvider(provider).models["gpt-5.4-fast"]
2002
- expect(model.cost.input).toEqual(5)
2003
- expect(model.cost.output).toEqual(30)
2004
- expect(model.cost.cache.read).toEqual(0.5)
2005
- expect(model.cost.cache.write).toEqual(0)
2006
- expect(model.options["serviceTier"]).toEqual("priority")
2007
- expect(model.cost.experimentalOver200K).toEqual({
2008
- input: 5,
2009
- output: 22.5,
2010
- cache: {
2011
- read: 0.5,
2012
- write: 0,
2013
- },
2014
- })
2015
- })
2016
-
2017
- test("models.dev normalization fills required response fields", () => {
2018
- const provider = {
2019
- id: "gateway",
2020
- name: "Gateway",
2021
- env: [],
2022
- models: {
2023
- "gpt-5.4": {
2024
- id: "gpt-5.4",
2025
- name: "GPT-5.4",
2026
- family: "gpt",
2027
- cost: {
2028
- input: 2.5,
2029
- output: 15,
2030
- },
2031
- limit: {
2032
- context: 1_050_000,
2033
- input: 922_000,
2034
- output: 128_000,
2035
- },
2036
- },
2037
- },
2038
- } as unknown as ModelsDev.Provider
2039
-
2040
- const model = Provider.fromModelsDevProvider(provider).models["gpt-5.4"]
2041
- expect(model.api.url).toBe("")
2042
- expect(model.capabilities.temperature).toBe(false)
2043
- expect(model.capabilities.reasoning).toBe(false)
2044
- expect(model.capabilities.attachment).toBe(false)
2045
- expect(model.capabilities.toolcall).toBe(true)
2046
- expect(model.release_date).toBe("")
2047
- })
2048
-
2049
- test("model variants are generated for reasoning models", async () => {
2050
- await using tmp = await tmpdir({
2051
- init: async (dir) => {
2052
- await Bun.write(
2053
- path.join(dir, "WOOZLIT_CODE.json"),
2054
- JSON.stringify({
2055
- $schema: "https://opencode.ai/config.json",
2056
- }),
2057
- )
2058
- },
2059
- })
2060
- await Instance.provide({
2061
- directory: tmp.path,
2062
- init: async () => {
2063
- set("ANTHROPIC_API_KEY", "test-api-key")
2064
- },
2065
- fn: async () => {
2066
- const providers = await list()
2067
- // Claude sonnet 4 has reasoning capability
2068
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
2069
- expect(model.capabilities.reasoning).toBe(true)
2070
- expect(model.variants).toBeDefined()
2071
- expect(Object.keys(model.variants!).length).toBeGreaterThan(0)
2072
- },
2073
- })
2074
- })
2075
-
2076
- test("model variants can be disabled via config", async () => {
2077
- await using tmp = await tmpdir({
2078
- init: async (dir) => {
2079
- await Bun.write(
2080
- path.join(dir, "WOOZLIT_CODE.json"),
2081
- JSON.stringify({
2082
- $schema: "https://opencode.ai/config.json",
2083
- provider: {
2084
- anthropic: {
2085
- models: {
2086
- "claude-sonnet-4-20250514": {
2087
- variants: {
2088
- high: { disabled: true },
2089
- },
2090
- },
2091
- },
2092
- },
2093
- },
2094
- }),
2095
- )
2096
- },
2097
- })
2098
- await Instance.provide({
2099
- directory: tmp.path,
2100
- init: async () => {
2101
- set("ANTHROPIC_API_KEY", "test-api-key")
2102
- },
2103
- fn: async () => {
2104
- const providers = await list()
2105
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
2106
- expect(model.variants).toBeDefined()
2107
- expect(model.variants!["high"]).toBeUndefined()
2108
- // max variant should still exist
2109
- expect(model.variants!["max"]).toBeDefined()
2110
- },
2111
- })
2112
- })
2113
-
2114
- test("model variants can be customized via config", async () => {
2115
- await using tmp = await tmpdir({
2116
- init: async (dir) => {
2117
- await Bun.write(
2118
- path.join(dir, "WOOZLIT_CODE.json"),
2119
- JSON.stringify({
2120
- $schema: "https://opencode.ai/config.json",
2121
- provider: {
2122
- anthropic: {
2123
- models: {
2124
- "claude-sonnet-4-20250514": {
2125
- variants: {
2126
- high: {
2127
- thinking: {
2128
- type: "enabled",
2129
- budgetTokens: 20000,
2130
- },
2131
- },
2132
- },
2133
- },
2134
- },
2135
- },
2136
- },
2137
- }),
2138
- )
2139
- },
2140
- })
2141
- await Instance.provide({
2142
- directory: tmp.path,
2143
- init: async () => {
2144
- set("ANTHROPIC_API_KEY", "test-api-key")
2145
- },
2146
- fn: async () => {
2147
- const providers = await list()
2148
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
2149
- expect(model.variants!["high"]).toBeDefined()
2150
- expect(model.variants!["high"].thinking.budgetTokens).toBe(20000)
2151
- },
2152
- })
2153
- })
2154
-
2155
- test("disabled key is stripped from variant config", async () => {
2156
- await using tmp = await tmpdir({
2157
- init: async (dir) => {
2158
- await Bun.write(
2159
- path.join(dir, "WOOZLIT_CODE.json"),
2160
- JSON.stringify({
2161
- $schema: "https://opencode.ai/config.json",
2162
- provider: {
2163
- anthropic: {
2164
- models: {
2165
- "claude-sonnet-4-20250514": {
2166
- variants: {
2167
- max: {
2168
- disabled: false,
2169
- customField: "test",
2170
- },
2171
- },
2172
- },
2173
- },
2174
- },
2175
- },
2176
- }),
2177
- )
2178
- },
2179
- })
2180
- await Instance.provide({
2181
- directory: tmp.path,
2182
- init: async () => {
2183
- set("ANTHROPIC_API_KEY", "test-api-key")
2184
- },
2185
- fn: async () => {
2186
- const providers = await list()
2187
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
2188
- expect(model.variants!["max"]).toBeDefined()
2189
- expect(model.variants!["max"].disabled).toBeUndefined()
2190
- expect(model.variants!["max"].customField).toBe("test")
2191
- },
2192
- })
2193
- })
2194
-
2195
- test("all variants can be disabled via config", async () => {
2196
- await using tmp = await tmpdir({
2197
- init: async (dir) => {
2198
- await Bun.write(
2199
- path.join(dir, "WOOZLIT_CODE.json"),
2200
- JSON.stringify({
2201
- $schema: "https://opencode.ai/config.json",
2202
- provider: {
2203
- anthropic: {
2204
- models: {
2205
- "claude-sonnet-4-20250514": {
2206
- variants: {
2207
- high: { disabled: true },
2208
- max: { disabled: true },
2209
- },
2210
- },
2211
- },
2212
- },
2213
- },
2214
- }),
2215
- )
2216
- },
2217
- })
2218
- await Instance.provide({
2219
- directory: tmp.path,
2220
- init: async () => {
2221
- set("ANTHROPIC_API_KEY", "test-api-key")
2222
- },
2223
- fn: async () => {
2224
- const providers = await list()
2225
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
2226
- expect(model.variants).toBeDefined()
2227
- expect(Object.keys(model.variants!).length).toBe(0)
2228
- },
2229
- })
2230
- })
2231
-
2232
- test("variant config merges with generated variants", async () => {
2233
- await using tmp = await tmpdir({
2234
- init: async (dir) => {
2235
- await Bun.write(
2236
- path.join(dir, "WOOZLIT_CODE.json"),
2237
- JSON.stringify({
2238
- $schema: "https://opencode.ai/config.json",
2239
- provider: {
2240
- anthropic: {
2241
- models: {
2242
- "claude-sonnet-4-20250514": {
2243
- variants: {
2244
- high: {
2245
- extraOption: "custom-value",
2246
- },
2247
- },
2248
- },
2249
- },
2250
- },
2251
- },
2252
- }),
2253
- )
2254
- },
2255
- })
2256
- await Instance.provide({
2257
- directory: tmp.path,
2258
- init: async () => {
2259
- set("ANTHROPIC_API_KEY", "test-api-key")
2260
- },
2261
- fn: async () => {
2262
- const providers = await list()
2263
- const model = providers[ProviderID.anthropic].models["claude-sonnet-4-20250514"]
2264
- expect(model.variants!["high"]).toBeDefined()
2265
- // Should have both the generated thinking config and the custom option
2266
- expect(model.variants!["high"].thinking).toBeDefined()
2267
- expect(model.variants!["high"].extraOption).toBe("custom-value")
2268
- },
2269
- })
2270
- })
2271
-
2272
- test("variants filtered in second pass for database models", async () => {
2273
- await using tmp = await tmpdir({
2274
- init: async (dir) => {
2275
- await Bun.write(
2276
- path.join(dir, "WOOZLIT_CODE.json"),
2277
- JSON.stringify({
2278
- $schema: "https://opencode.ai/config.json",
2279
- provider: {
2280
- openai: {
2281
- models: {
2282
- "gpt-5": {
2283
- variants: {
2284
- high: { disabled: true },
2285
- },
2286
- },
2287
- },
2288
- },
2289
- },
2290
- }),
2291
- )
2292
- },
2293
- })
2294
- await Instance.provide({
2295
- directory: tmp.path,
2296
- init: async () => {
2297
- set("OPENAI_API_KEY", "test-api-key")
2298
- },
2299
- fn: async () => {
2300
- const providers = await list()
2301
- const model = providers[ProviderID.openai].models["gpt-5"]
2302
- expect(model.variants).toBeDefined()
2303
- expect(model.variants!["high"]).toBeUndefined()
2304
- // Other variants should still exist
2305
- expect(model.variants!["medium"]).toBeDefined()
2306
- },
2307
- })
2308
- })
2309
-
2310
- test("custom model with variants enabled and disabled", async () => {
2311
- await using tmp = await tmpdir({
2312
- init: async (dir) => {
2313
- await Bun.write(
2314
- path.join(dir, "WOOZLIT_CODE.json"),
2315
- JSON.stringify({
2316
- $schema: "https://opencode.ai/config.json",
2317
- provider: {
2318
- "custom-reasoning": {
2319
- name: "Custom Reasoning Provider",
2320
- npm: "@ai-sdk/openai-compatible",
2321
- env: [],
2322
- models: {
2323
- "reasoning-model": {
2324
- name: "Reasoning Model",
2325
- tool_call: true,
2326
- reasoning: true,
2327
- limit: { context: 128000, output: 16000 },
2328
- variants: {
2329
- low: { reasoningEffort: "low" },
2330
- medium: { reasoningEffort: "medium" },
2331
- high: { reasoningEffort: "high", disabled: true },
2332
- custom: { reasoningEffort: "custom", budgetTokens: 5000 },
2333
- },
2334
- },
2335
- },
2336
- options: { apiKey: "test-key" },
2337
- },
2338
- },
2339
- }),
2340
- )
2341
- },
2342
- })
2343
- await Instance.provide({
2344
- directory: tmp.path,
2345
- fn: async () => {
2346
- const providers = await list()
2347
- const model = providers[ProviderID.make("custom-reasoning")].models["reasoning-model"]
2348
- expect(model.variants).toBeDefined()
2349
- // Enabled variants should exist
2350
- expect(model.variants!["low"]).toBeDefined()
2351
- expect(model.variants!["low"].reasoningEffort).toBe("low")
2352
- expect(model.variants!["medium"]).toBeDefined()
2353
- expect(model.variants!["medium"].reasoningEffort).toBe("medium")
2354
- expect(model.variants!["custom"]).toBeDefined()
2355
- expect(model.variants!["custom"].reasoningEffort).toBe("custom")
2356
- expect(model.variants!["custom"].budgetTokens).toBe(5000)
2357
- // Disabled variant should not exist
2358
- expect(model.variants!["high"]).toBeUndefined()
2359
- // disabled key should be stripped from all variants
2360
- expect(model.variants!["low"].disabled).toBeUndefined()
2361
- expect(model.variants!["medium"].disabled).toBeUndefined()
2362
- expect(model.variants!["custom"].disabled).toBeUndefined()
2363
- },
2364
- })
2365
- })
2366
-
2367
- test("Google Vertex: retains baseURL for custom proxy", async () => {
2368
- await using tmp = await tmpdir({
2369
- init: async (dir) => {
2370
- await Bun.write(
2371
- path.join(dir, "WOOZLIT_CODE.json"),
2372
- JSON.stringify({
2373
- $schema: "https://opencode.ai/config.json",
2374
- provider: {
2375
- "vertex-proxy": {
2376
- name: "Vertex Proxy",
2377
- npm: "@ai-sdk/google-vertex",
2378
- api: "https://my-proxy.com/v1",
2379
- env: ["GOOGLE_APPLICATION_CREDENTIALS"], // Mock env var requirement
2380
- models: {
2381
- "gemini-pro": {
2382
- name: "Gemini Pro",
2383
- tool_call: true,
2384
- },
2385
- },
2386
- options: {
2387
- project: "test-project",
2388
- location: "us-central1",
2389
- baseURL: "https://my-proxy.com/v1", // Should be retained
2390
- },
2391
- },
2392
- },
2393
- }),
2394
- )
2395
- },
2396
- })
2397
-
2398
- await Instance.provide({
2399
- directory: tmp.path,
2400
- init: async () => {
2401
- set("GOOGLE_APPLICATION_CREDENTIALS", "test-creds")
2402
- },
2403
- fn: async () => {
2404
- const providers = await list()
2405
- expect(providers[ProviderID.make("vertex-proxy")]).toBeDefined()
2406
- expect(providers[ProviderID.make("vertex-proxy")].options.baseURL).toBe("https://my-proxy.com/v1")
2407
- },
2408
- })
2409
- })
2410
-
2411
- test("Google Vertex: supports OpenAI compatible models", async () => {
2412
- await using tmp = await tmpdir({
2413
- init: async (dir) => {
2414
- await Bun.write(
2415
- path.join(dir, "WOOZLIT_CODE.json"),
2416
- JSON.stringify({
2417
- $schema: "https://opencode.ai/config.json",
2418
- provider: {
2419
- "vertex-openai": {
2420
- name: "Vertex OpenAI",
2421
- npm: "@ai-sdk/google-vertex",
2422
- env: ["GOOGLE_APPLICATION_CREDENTIALS"],
2423
- models: {
2424
- "gpt-4": {
2425
- name: "GPT-4",
2426
- provider: {
2427
- npm: "@ai-sdk/openai-compatible",
2428
- api: "https://api.openai.com/v1",
2429
- },
2430
- },
2431
- },
2432
- options: {
2433
- project: "test-project",
2434
- location: "us-central1",
2435
- },
2436
- },
2437
- },
2438
- }),
2439
- )
2440
- },
2441
- })
2442
-
2443
- await Instance.provide({
2444
- directory: tmp.path,
2445
- init: async () => {
2446
- set("GOOGLE_APPLICATION_CREDENTIALS", "test-creds")
2447
- },
2448
- fn: async () => {
2449
- const providers = await list()
2450
- const model = providers[ProviderID.make("vertex-openai")].models["gpt-4"]
2451
-
2452
- expect(model).toBeDefined()
2453
- expect(model.api.npm).toBe("@ai-sdk/openai-compatible")
2454
- },
2455
- })
2456
- })
2457
-
2458
- test("cloudflare-ai-gateway loads with env variables", async () => {
2459
- await using tmp = await tmpdir({
2460
- init: async (dir) => {
2461
- await Bun.write(
2462
- path.join(dir, "WOOZLIT_CODE.json"),
2463
- JSON.stringify({
2464
- $schema: "https://opencode.ai/config.json",
2465
- }),
2466
- )
2467
- },
2468
- })
2469
- await Instance.provide({
2470
- directory: tmp.path,
2471
- init: async () => {
2472
- set("CLOUDFLARE_ACCOUNT_ID", "test-account")
2473
- set("CLOUDFLARE_GATEWAY_ID", "test-gateway")
2474
- set("CLOUDFLARE_API_TOKEN", "test-token")
2475
- },
2476
- fn: async () => {
2477
- const providers = await list()
2478
- expect(providers[ProviderID.make("cloudflare-ai-gateway")]).toBeDefined()
2479
- },
2480
- })
2481
- })
2482
-
2483
- test("cloudflare-ai-gateway forwards config metadata options", async () => {
2484
- await using tmp = await tmpdir({
2485
- init: async (dir) => {
2486
- await Bun.write(
2487
- path.join(dir, "WOOZLIT_CODE.json"),
2488
- JSON.stringify({
2489
- $schema: "https://opencode.ai/config.json",
2490
- provider: {
2491
- "cloudflare-ai-gateway": {
2492
- options: {
2493
- metadata: { invoked_by: "test", project: "opencode" },
2494
- },
2495
- },
2496
- },
2497
- }),
2498
- )
2499
- },
2500
- })
2501
- await Instance.provide({
2502
- directory: tmp.path,
2503
- init: async () => {
2504
- set("CLOUDFLARE_ACCOUNT_ID", "test-account")
2505
- set("CLOUDFLARE_GATEWAY_ID", "test-gateway")
2506
- set("CLOUDFLARE_API_TOKEN", "test-token")
2507
- },
2508
- fn: async () => {
2509
- const providers = await list()
2510
- expect(providers[ProviderID.make("cloudflare-ai-gateway")]).toBeDefined()
2511
- expect(providers[ProviderID.make("cloudflare-ai-gateway")].options.metadata).toEqual({
2512
- invoked_by: "test",
2513
- project: "opencode",
2514
- })
2515
- },
2516
- })
2517
- })
2518
-
2519
- test("plugin config providers persist after instance dispose", async () => {
2520
- await using tmp = await tmpdir({
2521
- init: async (dir) => {
2522
- const root = path.join(dir, ".WOOZLIT_CODE", "plugin")
2523
- await mkdir(root, { recursive: true })
2524
- await Bun.write(
2525
- path.join(root, "demo-provider.ts"),
2526
- [
2527
- "export default {",
2528
- ' id: "demo.plugin-provider",',
2529
- " server: async () => ({",
2530
- " async config(cfg) {",
2531
- " cfg.provider ??= {}",
2532
- " cfg.provider.demo = {",
2533
- ' name: "Demo Provider",',
2534
- ' npm: "@ai-sdk/openai-compatible",',
2535
- ' api: "https://example.com/v1",',
2536
- " models: {",
2537
- " chat: {",
2538
- ' name: "Demo Chat",',
2539
- " tool_call: true,",
2540
- " limit: { context: 128000, output: 4096 },",
2541
- " },",
2542
- " },",
2543
- " }",
2544
- " },",
2545
- " }),",
2546
- "}",
2547
- "",
2548
- ].join("\n"),
2549
- )
2550
- },
2551
- })
2552
-
2553
- const first = await Instance.provide({
2554
- directory: tmp.path,
2555
- fn: async () =>
2556
- AppRuntime.runPromise(
2557
- Effect.gen(function* () {
2558
- const plugin = yield* Plugin.Service
2559
- const provider = yield* Provider.Service
2560
- yield* plugin.init()
2561
- return yield* provider.list()
2562
- }),
2563
- ),
2564
- })
2565
- expect(first[ProviderID.make("demo")]).toBeDefined()
2566
- expect(first[ProviderID.make("demo")].models[ModelID.make("chat")]).toBeDefined()
2567
-
2568
- await Instance.disposeAll()
2569
-
2570
- const second = await Instance.provide({
2571
- directory: tmp.path,
2572
- fn: async () => list(),
2573
- })
2574
- expect(second[ProviderID.make("demo")]).toBeDefined()
2575
- expect(second[ProviderID.make("demo")].models[ModelID.make("chat")]).toBeDefined()
2576
- })
2577
-
2578
- test("plugin config enabled and disabled providers are honored", async () => {
2579
- await using tmp = await tmpdir({
2580
- init: async (dir) => {
2581
- const root = path.join(dir, ".WOOZLIT_CODE", "plugin")
2582
- await mkdir(root, { recursive: true })
2583
- await Bun.write(
2584
- path.join(root, "provider-filter.ts"),
2585
- [
2586
- "export default {",
2587
- ' id: "demo.provider-filter",',
2588
- " server: async () => ({",
2589
- " async config(cfg) {",
2590
- ' cfg.enabled_providers = ["anthropic", "openai"]',
2591
- ' cfg.disabled_providers = ["openai"]',
2592
- " },",
2593
- " }),",
2594
- "}",
2595
- "",
2596
- ].join("\n"),
2597
- )
2598
- },
2599
- })
2600
-
2601
- await Instance.provide({
2602
- directory: tmp.path,
2603
- init: async () => {
2604
- set("ANTHROPIC_API_KEY", "test-anthropic-key")
2605
- set("OPENAI_API_KEY", "test-openai-key")
2606
- },
2607
- fn: async () => {
2608
- const providers = await list()
2609
- expect(providers[ProviderID.anthropic]).toBeDefined()
2610
- expect(providers[ProviderID.openai]).toBeUndefined()
2611
- },
2612
- })
2613
- })
2614
-
2615
- test("opencode and opencode-go providers are disabled by MimoFreeAuthPlugin", async () => {
2616
- await using base = await tmpdir({
2617
- init: async (dir) => {
2618
- await Bun.write(
2619
- path.join(dir, "WOOZLIT_CODE.json"),
2620
- JSON.stringify({
2621
- $schema: "https://opencode.ai/config.json",
2622
- provider: {
2623
- opencode: {
2624
- options: {
2625
- apiKey: "test-key",
2626
- },
2627
- },
2628
- },
2629
- }),
2630
- )
2631
- },
2632
- })
2633
-
2634
- const providers = await Instance.provide({
2635
- directory: base.path,
2636
- fn: async () => list(),
2637
- })
2638
-
2639
- // MimoFreeAuthPlugin always pushes opencode/opencode-go into disabled_providers,
2640
- // so they should not appear even when the user supplies an apiKey or auth record.
2641
- expect(opencodeProviderPresent(providers)).toBe(false)
2642
- expect(providers[ProviderID.make("opencode-go")]).toBeUndefined()
2643
- // The replacement free provider should be present.
2644
- expect(providers[ProviderID.make("mimo")]).toBeDefined()
2645
- expect(providers[ProviderID.make("mimo")].models[ModelID.make("mimo-auto")]).toBeDefined()
2646
- expect(providers[ProviderID.make("mimo")].models[ModelID.make("mimo-auto")].limit.context).toBe(1_000_000)
2647
- expect(providers[ProviderID.make("mimo")].models[ModelID.make("mimo-auto")].limit.output).toBe(128_000)
2648
- })