woozlit 2.3.0 → 2.3.1

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,2577 +0,0 @@
1
- import { test, expect, describe, mock, afterEach, beforeEach } from "bun:test"
2
- import { Effect, Layer, Option } from "effect"
3
- import { NodeFileSystem, NodePath } from "@effect/platform-node"
4
- import { Config, ConfigManaged } from "../../src/config"
5
- import { ConfigParse } from "../../src/config/parse"
6
- import { EffectFlock } from "@woozlit/shared/util/effect-flock"
7
-
8
- import { Instance } from "../../src/project/instance"
9
- import { Auth } from "../../src/auth"
10
- import { Account } from "../../src/account/account"
11
- import { AccessToken, AccountID, OrgID } from "../../src/account/schema"
12
- import { AppFileSystem } from "@woozlit/shared/filesystem"
13
- import { Env } from "../../src/env"
14
- import { provideTmpdirInstance } from "../fixture/fixture"
15
- import { tmpdir } from "../fixture/fixture"
16
- import * as CrossSpawnSpawner from "../../src/effect/cross-spawn-spawner"
17
- import { testEffect } from "../lib/effect"
18
-
19
- /** Infra layer that provides FileSystem, Path, ChildProcessSpawner for test fixtures */
20
- const infra = CrossSpawnSpawner.defaultLayer.pipe(
21
- Layer.provideMerge(Layer.mergeAll(NodeFileSystem.layer, NodePath.layer)),
22
- )
23
- import path from "path"
24
- import fs from "fs/promises"
25
- import { pathToFileURL } from "url"
26
- import { Global } from "../../src/global"
27
- import { ProjectID } from "../../src/project/schema"
28
- import { Filesystem } from "../../src/util"
29
- import { ConfigPlugin } from "@/config/plugin"
30
- import { Npm } from "@/npm"
31
-
32
- const emptyAccount = Layer.mock(Account.Service)({
33
- active: () => Effect.succeed(Option.none()),
34
- activeOrg: () => Effect.succeed(Option.none()),
35
- })
36
-
37
- const emptyAuth = Layer.mock(Auth.Service)({
38
- all: () => Effect.succeed({}),
39
- })
40
-
41
- const testFlock = EffectFlock.defaultLayer
42
-
43
- const layer = Config.layer.pipe(
44
- Layer.provide(testFlock),
45
- Layer.provide(AppFileSystem.defaultLayer),
46
- Layer.provide(Env.defaultLayer),
47
- Layer.provide(emptyAuth),
48
- Layer.provide(emptyAccount),
49
- Layer.provideMerge(infra),
50
- Layer.provide(Npm.defaultLayer),
51
- )
52
-
53
- const it = testEffect(layer)
54
-
55
- const load = () => Effect.runPromise(Config.Service.use((svc) => svc.get()).pipe(Effect.scoped, Effect.provide(layer)))
56
- const save = (config: Config.Info) =>
57
- Effect.runPromise(Config.Service.use((svc) => svc.update(config)).pipe(Effect.scoped, Effect.provide(layer)))
58
- const clear = (wait = false) =>
59
- Effect.runPromise(Config.Service.use((svc) => svc.invalidate(wait)).pipe(Effect.scoped, Effect.provide(layer)))
60
- const listDirs = () =>
61
- Effect.runPromise(Config.Service.use((svc) => svc.directories()).pipe(Effect.scoped, Effect.provide(layer)))
62
- const ready = () =>
63
- Effect.runPromise(Config.Service.use((svc) => svc.waitForDependencies()).pipe(Effect.scoped, Effect.provide(layer)))
64
-
65
- // Get managed config directory from environment (set in preload.ts)
66
- const managedConfigDir = process.env.WOOZLIT_CODE_TEST_MANAGED_CONFIG_DIR!
67
-
68
- beforeEach(async () => {
69
- await clear(true)
70
- })
71
-
72
- afterEach(async () => {
73
- await fs.rm(path.join(Global.Path.home, ".claude.json"), { force: true }).catch(() => {})
74
- await fs.rm(managedConfigDir, { force: true, recursive: true }).catch(() => {})
75
- await clear(true)
76
- })
77
-
78
- async function writeManagedSettings(settings: object, filename = "WOOZLIT_CODE.json") {
79
- await fs.mkdir(managedConfigDir, { recursive: true })
80
- await Filesystem.write(path.join(managedConfigDir, filename), JSON.stringify(settings))
81
- }
82
-
83
- async function writeConfig(dir: string, config: object, name = "WOOZLIT_CODE.json") {
84
- await Filesystem.write(path.join(dir, name), JSON.stringify(config))
85
- }
86
-
87
- async function writeClaudeConfig(file: string, config: object) {
88
- await Filesystem.write(file, JSON.stringify(config))
89
- }
90
-
91
- async function check(map: (dir: string) => string) {
92
- if (process.platform !== "win32") return
93
- await using globalTmp = await tmpdir()
94
- await using tmp = await tmpdir({ git: true, config: { snapshot: true } })
95
- const prev = Global.Path.config
96
- ;(Global.Path as { config: string }).config = globalTmp.path
97
- await clear()
98
- try {
99
- await writeConfig(globalTmp.path, {
100
- $schema: "https://opencode.ai/config.json",
101
- snapshot: false,
102
- })
103
- await Instance.provide({
104
- directory: map(tmp.path),
105
- fn: async () => {
106
- const cfg = await load()
107
- expect(cfg.snapshot).toBe(true)
108
- expect(Instance.directory).toBe(Filesystem.resolve(tmp.path))
109
- expect(Instance.project.id).not.toBe(ProjectID.global)
110
- },
111
- })
112
- } finally {
113
- await Instance.disposeAll()
114
- ;(Global.Path as { config: string }).config = prev
115
- await clear()
116
- }
117
- }
118
-
119
- test("loads config with defaults when no files exist", async () => {
120
- await using tmp = await tmpdir()
121
- await Instance.provide({
122
- directory: tmp.path,
123
- fn: async () => {
124
- const config = await load()
125
- expect(config.username).toBeDefined()
126
- },
127
- })
128
- })
129
-
130
- test("loads JSON config file", async () => {
131
- await using tmp = await tmpdir({
132
- init: async (dir) => {
133
- await writeConfig(dir, {
134
- $schema: "https://opencode.ai/config.json",
135
- model: "test/model",
136
- username: "testuser",
137
- })
138
- },
139
- })
140
- await Instance.provide({
141
- directory: tmp.path,
142
- fn: async () => {
143
- const config = await load()
144
- expect(config.model).toBe("test/model")
145
- expect(config.username).toBe("testuser")
146
- },
147
- })
148
- })
149
-
150
- test("loads Claude Code MCP servers from home and project config", async () => {
151
- await writeClaudeConfig(path.join(Global.Path.home, ".claude.json"), {
152
- mcpServers: {
153
- context7: {
154
- type: "http",
155
- url: "https://mcp.context7.com/mcp",
156
- headers: {
157
- Authorization: "Bearer ${CONTEXT7_API_KEY}",
158
- },
159
- },
160
- filesystem: {
161
- command: "npx",
162
- args: ["-y", "@modelcontextprotocol/server-filesystem", "/tmp/project"],
163
- env: {
164
- DEBUG: "1",
165
- },
166
- },
167
- },
168
- })
169
-
170
- await using tmp = await tmpdir({
171
- init: async (dir) => {
172
- await writeConfig(dir, {
173
- $schema: "https://opencode.ai/config.json",
174
- mcp: {
175
- context7: {
176
- type: "remote",
177
- url: "https://native.example.com/mcp",
178
- },
179
- },
180
- })
181
- await writeClaudeConfig(path.join(dir, ".claude.json"), {
182
- mcpServers: {
183
- project: {
184
- type: "streamable-http",
185
- url: "https://project.example.com/mcp",
186
- disabled: true,
187
- },
188
- },
189
- })
190
- },
191
- })
192
-
193
- await Instance.provide({
194
- directory: tmp.path,
195
- fn: async () => {
196
- const config = await load()
197
- expect(config.mcp?.context7).toEqual({
198
- type: "remote",
199
- url: "https://native.example.com/mcp",
200
- })
201
- expect(config.mcp?.filesystem).toEqual({
202
- type: "local",
203
- command: ["npx", "-y", "@modelcontextprotocol/server-filesystem", "/tmp/project"],
204
- environment: {
205
- DEBUG: "1",
206
- },
207
- enabled: true,
208
- })
209
- expect(config.mcp?.project).toEqual({
210
- type: "remote",
211
- url: "https://project.example.com/mcp",
212
- enabled: false,
213
- })
214
- expect(config.mcp_origins?.context7?.type).toBe("opencode")
215
- expect(config.mcp_origins?.filesystem).toEqual({
216
- type: "claude",
217
- source: path.join(Global.Path.home, ".claude.json"),
218
- })
219
- expect(config.mcp_origins?.project).toEqual({
220
- type: "claude",
221
- source: path.join(tmp.path, ".claude.json"),
222
- })
223
- },
224
- })
225
- })
226
-
227
- test("skips unsupported Claude Code MCP servers", async () => {
228
- await writeClaudeConfig(path.join(Global.Path.home, ".claude.json"), {
229
- mcpServers: {
230
- legacy: {
231
- type: "sse",
232
- url: "https://example.com/sse",
233
- },
234
- badArgs: {
235
- command: "npx",
236
- args: "--bad",
237
- },
238
- good: {
239
- type: "http",
240
- url: "https://good.example.com/mcp",
241
- },
242
- },
243
- })
244
-
245
- await using tmp = await tmpdir()
246
- await Instance.provide({
247
- directory: tmp.path,
248
- fn: async () => {
249
- const config = await load()
250
- expect(config.mcp?.good).toEqual({
251
- type: "remote",
252
- url: "https://good.example.com/mcp",
253
- enabled: true,
254
- })
255
- expect(config.mcp?.legacy).toBeUndefined()
256
- expect(config.mcp?.badArgs).toBeUndefined()
257
- },
258
- })
259
- })
260
-
261
- test("loads formatter boolean config", async () => {
262
- await using tmp = await tmpdir({
263
- init: async (dir) => {
264
- await writeConfig(dir, {
265
- $schema: "https://opencode.ai/config.json",
266
- formatter: true,
267
- })
268
- },
269
- })
270
- await Instance.provide({
271
- directory: tmp.path,
272
- fn: async () => {
273
- const config = await load()
274
- expect(config.formatter).toBe(true)
275
- },
276
- })
277
- })
278
-
279
- test("loads lsp boolean config", async () => {
280
- await using tmp = await tmpdir({
281
- init: async (dir) => {
282
- await writeConfig(dir, {
283
- $schema: "https://opencode.ai/config.json",
284
- lsp: true,
285
- })
286
- },
287
- })
288
- await Instance.provide({
289
- directory: tmp.path,
290
- fn: async () => {
291
- const config = await load()
292
- expect(config.lsp).toBe(true)
293
- },
294
- })
295
- })
296
-
297
- test("loads project config from Git Bash and MSYS2 paths on Windows", async () => {
298
- // Git Bash and MSYS2 both use /<drive>/... paths on Windows.
299
- await check((dir) => {
300
- const drive = dir[0].toLowerCase()
301
- const rest = dir.slice(2).replaceAll("\\", "/")
302
- return `/${drive}${rest}`
303
- })
304
- })
305
-
306
- test("loads project config from Cygwin paths on Windows", async () => {
307
- await check((dir) => {
308
- const drive = dir[0].toLowerCase()
309
- const rest = dir.slice(2).replaceAll("\\", "/")
310
- return `/cygdrive/${drive}${rest}`
311
- })
312
- })
313
-
314
- test("ignores legacy tui keys in opencode config", async () => {
315
- await using tmp = await tmpdir({
316
- init: async (dir) => {
317
- await writeConfig(dir, {
318
- $schema: "https://opencode.ai/config.json",
319
- model: "test/model",
320
- theme: "legacy",
321
- tui: { scroll_speed: 4 },
322
- })
323
- },
324
- })
325
- await Instance.provide({
326
- directory: tmp.path,
327
- fn: async () => {
328
- const config = await load()
329
- expect(config.model).toBe("test/model")
330
- expect((config as Record<string, unknown>).theme).toBeUndefined()
331
- expect((config as Record<string, unknown>).tui).toBeUndefined()
332
- },
333
- })
334
- })
335
-
336
- test("loads JSONC config file", async () => {
337
- await using tmp = await tmpdir({
338
- init: async (dir) => {
339
- await Filesystem.write(
340
- path.join(dir, "WOOZLIT_CODE.jsonc"),
341
- `{
342
- // This is a comment
343
- "$schema": "https://opencode.ai/config.json",
344
- "model": "test/model",
345
- "username": "testuser"
346
- }`,
347
- )
348
- },
349
- })
350
- await Instance.provide({
351
- directory: tmp.path,
352
- fn: async () => {
353
- const config = await load()
354
- expect(config.model).toBe("test/model")
355
- expect(config.username).toBe("testuser")
356
- },
357
- })
358
- })
359
-
360
- test("jsonc overrides json in the same directory", async () => {
361
- await using tmp = await tmpdir({
362
- init: async (dir) => {
363
- await writeConfig(
364
- dir,
365
- {
366
- $schema: "https://opencode.ai/config.json",
367
- model: "base",
368
- username: "base",
369
- },
370
- "WOOZLIT_CODE.jsonc",
371
- )
372
- await writeConfig(dir, {
373
- $schema: "https://opencode.ai/config.json",
374
- model: "override",
375
- })
376
- },
377
- })
378
- await Instance.provide({
379
- directory: tmp.path,
380
- fn: async () => {
381
- const config = await load()
382
- expect(config.model).toBe("base")
383
- expect(config.username).toBe("base")
384
- },
385
- })
386
- })
387
-
388
- test("handles environment variable substitution", async () => {
389
- const originalEnv = process.env["TEST_VAR"]
390
- process.env["TEST_VAR"] = "test-user"
391
-
392
- try {
393
- await using tmp = await tmpdir({
394
- init: async (dir) => {
395
- await writeConfig(dir, {
396
- $schema: "https://opencode.ai/config.json",
397
- username: "{env:TEST_VAR}",
398
- })
399
- },
400
- })
401
- await Instance.provide({
402
- directory: tmp.path,
403
- fn: async () => {
404
- const config = await load()
405
- expect(config.username).toBe("test-user")
406
- },
407
- })
408
- } finally {
409
- if (originalEnv !== undefined) {
410
- process.env["TEST_VAR"] = originalEnv
411
- } else {
412
- delete process.env["TEST_VAR"]
413
- }
414
- }
415
- })
416
-
417
- test("preserves env variables when adding $schema to config", async () => {
418
- const originalEnv = process.env["PRESERVE_VAR"]
419
- process.env["PRESERVE_VAR"] = "secret_value"
420
-
421
- try {
422
- await using tmp = await tmpdir({
423
- init: async (dir) => {
424
- // Config without $schema - should trigger auto-add
425
- await Filesystem.write(
426
- path.join(dir, "WOOZLIT_CODE.json"),
427
- JSON.stringify({
428
- username: "{env:PRESERVE_VAR}",
429
- }),
430
- )
431
- },
432
- })
433
- await Instance.provide({
434
- directory: tmp.path,
435
- fn: async () => {
436
- const config = await load()
437
- expect(config.username).toBe("secret_value")
438
-
439
- // Read the file to verify the env variable was preserved
440
- const content = await Filesystem.readText(path.join(tmp.path, "WOOZLIT_CODE.json"))
441
- expect(content).toContain("{env:PRESERVE_VAR}")
442
- expect(content).not.toContain("secret_value")
443
- expect(content).toContain("$schema")
444
- },
445
- })
446
- } finally {
447
- if (originalEnv !== undefined) {
448
- process.env["PRESERVE_VAR"] = originalEnv
449
- } else {
450
- delete process.env["PRESERVE_VAR"]
451
- }
452
- }
453
- })
454
-
455
- test("resolves env templates in account config with account token", async () => {
456
- const originalControlToken = process.env["WOOZLIT_CODE_CONSOLE_TOKEN"]
457
-
458
- const fakeAccount = Layer.mock(Account.Service)({
459
- active: () =>
460
- Effect.succeed(
461
- Option.some({
462
- id: AccountID.make("account-1"),
463
- email: "user@example.com",
464
- url: "https://control.example.com",
465
- active_org_id: OrgID.make("org-1"),
466
- }),
467
- ),
468
- activeOrg: () =>
469
- Effect.succeed(
470
- Option.some({
471
- account: {
472
- id: AccountID.make("account-1"),
473
- email: "user@example.com",
474
- url: "https://control.example.com",
475
- active_org_id: OrgID.make("org-1"),
476
- },
477
- org: {
478
- id: OrgID.make("org-1"),
479
- name: "Example Org",
480
- },
481
- }),
482
- ),
483
- config: () =>
484
- Effect.succeed(
485
- Option.some({
486
- provider: { opencode: { options: { apiKey: "{env:WOOZLIT_CODE_CONSOLE_TOKEN}" } } },
487
- }),
488
- ),
489
- token: () => Effect.succeed(Option.some(AccessToken.make("st_test_token"))),
490
- })
491
-
492
- const layer = Config.layer.pipe(
493
- Layer.provide(testFlock),
494
- Layer.provide(AppFileSystem.defaultLayer),
495
- Layer.provide(Env.defaultLayer),
496
- Layer.provide(emptyAuth),
497
- Layer.provide(fakeAccount),
498
- Layer.provideMerge(infra),
499
- )
500
-
501
- try {
502
- await provideTmpdirInstance(() =>
503
- Config.Service.use((svc) =>
504
- Effect.gen(function* () {
505
- const config = yield* svc.get()
506
- expect(config.provider?.["opencode"]?.options?.apiKey).toBe("st_test_token")
507
- }),
508
- ),
509
- ).pipe(Effect.scoped, Effect.provide(layer), Effect.provide(Npm.defaultLayer), Effect.runPromise)
510
- } finally {
511
- if (originalControlToken !== undefined) {
512
- process.env["WOOZLIT_CODE_CONSOLE_TOKEN"] = originalControlToken
513
- } else {
514
- delete process.env["WOOZLIT_CODE_CONSOLE_TOKEN"]
515
- }
516
- }
517
- })
518
-
519
- test("handles file inclusion substitution", async () => {
520
- await using tmp = await tmpdir({
521
- init: async (dir) => {
522
- await Filesystem.write(path.join(dir, "included.txt"), "test-user")
523
- await writeConfig(dir, {
524
- $schema: "https://opencode.ai/config.json",
525
- username: "{file:included.txt}",
526
- })
527
- },
528
- })
529
- await Instance.provide({
530
- directory: tmp.path,
531
- fn: async () => {
532
- const config = await load()
533
- expect(config.username).toBe("test-user")
534
- },
535
- })
536
- })
537
-
538
- test("handles file inclusion with replacement tokens", async () => {
539
- await using tmp = await tmpdir({
540
- init: async (dir) => {
541
- await Filesystem.write(path.join(dir, "included.md"), "const out = await Bun.$`echo hi`")
542
- await writeConfig(dir, {
543
- $schema: "https://opencode.ai/config.json",
544
- username: "{file:included.md}",
545
- })
546
- },
547
- })
548
- await Instance.provide({
549
- directory: tmp.path,
550
- fn: async () => {
551
- const config = await load()
552
- expect(config.username).toBe("const out = await Bun.$`echo hi`")
553
- },
554
- })
555
- })
556
-
557
- test("validates config schema and throws on invalid fields", async () => {
558
- await using tmp = await tmpdir({
559
- init: async (dir) => {
560
- await writeConfig(dir, {
561
- $schema: "https://opencode.ai/config.json",
562
- invalid_field: "should cause error",
563
- })
564
- },
565
- })
566
- await Instance.provide({
567
- directory: tmp.path,
568
- fn: async () => {
569
- // Strict schema should throw an error for invalid fields
570
- await expect(load()).rejects.toThrow()
571
- },
572
- })
573
- })
574
-
575
- test("throws error for invalid JSON", async () => {
576
- await using tmp = await tmpdir({
577
- init: async (dir) => {
578
- await Filesystem.write(path.join(dir, "WOOZLIT_CODE.json"), "{ invalid json }")
579
- },
580
- })
581
- await Instance.provide({
582
- directory: tmp.path,
583
- fn: async () => {
584
- await expect(load()).rejects.toThrow()
585
- },
586
- })
587
- })
588
-
589
- test("handles agent configuration", async () => {
590
- await using tmp = await tmpdir({
591
- init: async (dir) => {
592
- await writeConfig(dir, {
593
- $schema: "https://opencode.ai/config.json",
594
- agent: {
595
- test_agent: {
596
- model: "test/model",
597
- temperature: 0.7,
598
- description: "test agent",
599
- },
600
- },
601
- })
602
- },
603
- })
604
- await Instance.provide({
605
- directory: tmp.path,
606
- fn: async () => {
607
- const config = await load()
608
- expect(config.agent?.["test_agent"]).toEqual(
609
- expect.objectContaining({
610
- model: "test/model",
611
- temperature: 0.7,
612
- description: "test agent",
613
- }),
614
- )
615
- },
616
- })
617
- })
618
-
619
- test("treats agent variant as model-scoped setting (not provider option)", async () => {
620
- await using tmp = await tmpdir({
621
- init: async (dir) => {
622
- await writeConfig(dir, {
623
- $schema: "https://opencode.ai/config.json",
624
- agent: {
625
- test_agent: {
626
- model: "openai/gpt-5.2",
627
- variant: "xhigh",
628
- max_tokens: 123,
629
- },
630
- },
631
- })
632
- },
633
- })
634
-
635
- await Instance.provide({
636
- directory: tmp.path,
637
- fn: async () => {
638
- const config = await load()
639
- const agent = config.agent?.["test_agent"]
640
-
641
- expect(agent?.variant).toBe("xhigh")
642
- expect(agent?.options).toMatchObject({
643
- max_tokens: 123,
644
- })
645
- expect(agent?.options).not.toHaveProperty("variant")
646
- },
647
- })
648
- })
649
-
650
- test("handles command configuration", async () => {
651
- await using tmp = await tmpdir({
652
- init: async (dir) => {
653
- await writeConfig(dir, {
654
- $schema: "https://opencode.ai/config.json",
655
- command: {
656
- test_command: {
657
- template: "test template",
658
- description: "test command",
659
- agent: "test_agent",
660
- },
661
- },
662
- })
663
- },
664
- })
665
- await Instance.provide({
666
- directory: tmp.path,
667
- fn: async () => {
668
- const config = await load()
669
- expect(config.command?.["test_command"]).toEqual({
670
- template: "test template",
671
- description: "test command",
672
- agent: "test_agent",
673
- })
674
- },
675
- })
676
- })
677
-
678
- test("migrates autoshare to share field", async () => {
679
- await using tmp = await tmpdir({
680
- init: async (dir) => {
681
- await Filesystem.write(
682
- path.join(dir, "WOOZLIT_CODE.json"),
683
- JSON.stringify({
684
- $schema: "https://opencode.ai/config.json",
685
- autoshare: true,
686
- }),
687
- )
688
- },
689
- })
690
- await Instance.provide({
691
- directory: tmp.path,
692
- fn: async () => {
693
- const config = await load()
694
- expect(config.share).toBe("auto")
695
- expect(config.autoshare).toBe(true)
696
- },
697
- })
698
- })
699
-
700
- test("migrates mode field to agent field", async () => {
701
- await using tmp = await tmpdir({
702
- init: async (dir) => {
703
- await Filesystem.write(
704
- path.join(dir, "WOOZLIT_CODE.json"),
705
- JSON.stringify({
706
- $schema: "https://opencode.ai/config.json",
707
- mode: {
708
- test_mode: {
709
- model: "test/model",
710
- temperature: 0.5,
711
- },
712
- },
713
- }),
714
- )
715
- },
716
- })
717
- await Instance.provide({
718
- directory: tmp.path,
719
- fn: async () => {
720
- const config = await load()
721
- expect(config.agent?.["test_mode"]).toEqual({
722
- model: "test/model",
723
- temperature: 0.5,
724
- mode: "primary",
725
- options: {},
726
- permission: {},
727
- })
728
- },
729
- })
730
- })
731
-
732
- test("loads config from .WOOZLIT_CODE directory", async () => {
733
- await using tmp = await tmpdir({
734
- init: async (dir) => {
735
- const opencodeDir = path.join(dir, ".WOOZLIT_CODE")
736
- await fs.mkdir(opencodeDir, { recursive: true })
737
- const agentDir = path.join(opencodeDir, "agent")
738
- await fs.mkdir(agentDir, { recursive: true })
739
-
740
- await Filesystem.write(
741
- path.join(agentDir, "test.md"),
742
- `---
743
- model: test/model
744
- ---
745
- Test agent prompt`,
746
- )
747
- },
748
- })
749
- await Instance.provide({
750
- directory: tmp.path,
751
- fn: async () => {
752
- const config = await load()
753
- expect(config.agent?.["test"]).toEqual(
754
- expect.objectContaining({
755
- name: "test",
756
- model: "test/model",
757
- prompt: "Test agent prompt",
758
- }),
759
- )
760
- },
761
- })
762
- })
763
-
764
- test("loads agents from .WOOZLIT_CODE/agents (plural)", async () => {
765
- await using tmp = await tmpdir({
766
- init: async (dir) => {
767
- const opencodeDir = path.join(dir, ".WOOZLIT_CODE")
768
- await fs.mkdir(opencodeDir, { recursive: true })
769
-
770
- const agentsDir = path.join(opencodeDir, "agents")
771
- await fs.mkdir(path.join(agentsDir, "nested"), { recursive: true })
772
-
773
- await Filesystem.write(
774
- path.join(agentsDir, "helper.md"),
775
- `---
776
- model: test/model
777
- mode: subagent
778
- ---
779
- Helper agent prompt`,
780
- )
781
-
782
- await Filesystem.write(
783
- path.join(agentsDir, "nested", "child.md"),
784
- `---
785
- model: test/model
786
- mode: subagent
787
- ---
788
- Nested agent prompt`,
789
- )
790
- },
791
- })
792
-
793
- await Instance.provide({
794
- directory: tmp.path,
795
- fn: async () => {
796
- const config = await load()
797
-
798
- expect(config.agent?.["helper"]).toMatchObject({
799
- name: "helper",
800
- model: "test/model",
801
- mode: "subagent",
802
- prompt: "Helper agent prompt",
803
- })
804
-
805
- expect(config.agent?.["nested/child"]).toMatchObject({
806
- name: "nested/child",
807
- model: "test/model",
808
- mode: "subagent",
809
- prompt: "Nested agent prompt",
810
- })
811
- },
812
- })
813
- })
814
-
815
- test("loads commands from .WOOZLIT_CODE/command (singular)", async () => {
816
- await using tmp = await tmpdir({
817
- init: async (dir) => {
818
- const opencodeDir = path.join(dir, ".WOOZLIT_CODE")
819
- await fs.mkdir(opencodeDir, { recursive: true })
820
-
821
- const commandDir = path.join(opencodeDir, "command")
822
- await fs.mkdir(path.join(commandDir, "nested"), { recursive: true })
823
-
824
- await Filesystem.write(
825
- path.join(commandDir, "hello.md"),
826
- `---
827
- description: Test command
828
- ---
829
- Hello from singular command`,
830
- )
831
-
832
- await Filesystem.write(
833
- path.join(commandDir, "nested", "child.md"),
834
- `---
835
- description: Nested command
836
- ---
837
- Nested command template`,
838
- )
839
- },
840
- })
841
-
842
- await Instance.provide({
843
- directory: tmp.path,
844
- fn: async () => {
845
- const config = await load()
846
-
847
- expect(config.command?.["hello"]).toEqual({
848
- description: "Test command",
849
- template: "Hello from singular command",
850
- })
851
-
852
- expect(config.command?.["nested/child"]).toEqual({
853
- description: "Nested command",
854
- template: "Nested command template",
855
- })
856
- },
857
- })
858
- })
859
-
860
- test("loads commands from .WOOZLIT_CODE/commands (plural)", async () => {
861
- await using tmp = await tmpdir({
862
- init: async (dir) => {
863
- const opencodeDir = path.join(dir, ".WOOZLIT_CODE")
864
- await fs.mkdir(opencodeDir, { recursive: true })
865
-
866
- const commandsDir = path.join(opencodeDir, "commands")
867
- await fs.mkdir(path.join(commandsDir, "nested"), { recursive: true })
868
-
869
- await Filesystem.write(
870
- path.join(commandsDir, "hello.md"),
871
- `---
872
- description: Test command
873
- ---
874
- Hello from plural commands`,
875
- )
876
-
877
- await Filesystem.write(
878
- path.join(commandsDir, "nested", "child.md"),
879
- `---
880
- description: Nested command
881
- ---
882
- Nested command template`,
883
- )
884
- },
885
- })
886
-
887
- await Instance.provide({
888
- directory: tmp.path,
889
- fn: async () => {
890
- const config = await load()
891
-
892
- expect(config.command?.["hello"]).toEqual({
893
- description: "Test command",
894
- template: "Hello from plural commands",
895
- })
896
-
897
- expect(config.command?.["nested/child"]).toEqual({
898
- description: "Nested command",
899
- template: "Nested command template",
900
- })
901
- },
902
- })
903
- })
904
-
905
- test("loads commands from project .claude/commands", async () => {
906
- await using tmp = await tmpdir({
907
- init: async (dir) => {
908
- const commandsDir = path.join(dir, ".claude", "commands")
909
- await fs.mkdir(path.join(commandsDir, "nested"), { recursive: true })
910
-
911
- await Filesystem.write(
912
- path.join(commandsDir, "hello.md"),
913
- `---
914
- description: Claude command
915
- ---
916
- Hello from claude commands`,
917
- )
918
-
919
- await Filesystem.write(
920
- path.join(commandsDir, "nested", "child.md"),
921
- `---
922
- description: Nested claude command
923
- ---
924
- Nested claude template`,
925
- )
926
- },
927
- })
928
-
929
- await Instance.provide({
930
- directory: tmp.path,
931
- fn: async () => {
932
- const config = await load()
933
-
934
- expect(config.command?.["hello"]).toEqual({
935
- description: "Claude command",
936
- template: "Hello from claude commands",
937
- })
938
-
939
- expect(config.command?.["nested/child"]).toEqual({
940
- description: "Nested claude command",
941
- template: "Nested claude template",
942
- })
943
- },
944
- })
945
- })
946
-
947
- test("WOOZLIT_CODE command overrides .claude command on name collision", async () => {
948
- await using tmp = await tmpdir({
949
- init: async (dir) => {
950
- await Filesystem.write(
951
- path.join(dir, ".claude", "commands", "dup.md"),
952
- `---
953
- description: claude version
954
- ---
955
- from claude`,
956
- )
957
-
958
- await Filesystem.write(
959
- path.join(dir, ".WOOZLIT_CODE", "command", "dup.md"),
960
- `---
961
- description: WOOZLIT_CODE version
962
- ---
963
- from WOOZLIT_CODE`,
964
- )
965
- },
966
- })
967
-
968
- await Instance.provide({
969
- directory: tmp.path,
970
- fn: async () => {
971
- const config = await load()
972
-
973
- expect(config.command?.["dup"]).toEqual({
974
- description: "WOOZLIT_CODE version",
975
- template: "from WOOZLIT_CODE",
976
- })
977
- },
978
- })
979
- })
980
-
981
- test("updates config and writes to file", async () => {
982
- await using tmp = await tmpdir()
983
- await Instance.provide({
984
- directory: tmp.path,
985
- fn: async () => {
986
- const newConfig = { model: "updated/model" }
987
- await save(newConfig as any)
988
-
989
- const writtenConfig = await Filesystem.readJson<{ model: string }>(path.join(tmp.path, "config.json"))
990
- expect(writtenConfig.model).toBe("updated/model")
991
- },
992
- })
993
- })
994
-
995
- test("gets config directories", async () => {
996
- await using tmp = await tmpdir()
997
- await Instance.provide({
998
- directory: tmp.path,
999
- fn: async () => {
1000
- const dirs = await listDirs()
1001
- expect(dirs.length).toBeGreaterThanOrEqual(1)
1002
- },
1003
- })
1004
- })
1005
-
1006
- test("does not try to install dependencies in read-only WOOZLIT_CODE_CONFIG_DIR", async () => {
1007
- if (process.platform === "win32") return
1008
-
1009
- await using tmp = await tmpdir<string>({
1010
- init: async (dir) => {
1011
- const ro = path.join(dir, "readonly")
1012
- await fs.mkdir(ro, { recursive: true })
1013
- await fs.chmod(ro, 0o555)
1014
- return ro
1015
- },
1016
- dispose: async (dir) => {
1017
- const ro = path.join(dir, "readonly")
1018
- await fs.chmod(ro, 0o755).catch(() => {})
1019
- return ro
1020
- },
1021
- })
1022
-
1023
- const prev = process.env.WOOZLIT_CODE_CONFIG_DIR
1024
- process.env.WOOZLIT_CODE_CONFIG_DIR = tmp.extra
1025
-
1026
- try {
1027
- await Instance.provide({
1028
- directory: tmp.path,
1029
- fn: async () => {
1030
- await load()
1031
- },
1032
- })
1033
- } finally {
1034
- if (prev === undefined) delete process.env.WOOZLIT_CODE_CONFIG_DIR
1035
- else process.env.WOOZLIT_CODE_CONFIG_DIR = prev
1036
- }
1037
- })
1038
-
1039
- test("installs dependencies in writable WOOZLIT_CODE_CONFIG_DIR", async () => {
1040
- await using tmp = await tmpdir<string>({
1041
- init: async (dir) => {
1042
- const cfg = path.join(dir, "configdir")
1043
- await fs.mkdir(cfg, { recursive: true })
1044
- return cfg
1045
- },
1046
- })
1047
-
1048
- const prev = process.env.WOOZLIT_CODE_CONFIG_DIR
1049
- process.env.WOOZLIT_CODE_CONFIG_DIR = tmp.extra
1050
-
1051
- const noopNpm = Layer.mock(Npm.Service)({
1052
- install: () => Effect.void,
1053
- add: () => Effect.die("not implemented"),
1054
- outdated: () => Effect.succeed(false),
1055
- which: () => Effect.succeed(Option.none()),
1056
- })
1057
- const testLayer = Config.layer.pipe(
1058
- Layer.provide(testFlock),
1059
- Layer.provide(AppFileSystem.defaultLayer),
1060
- Layer.provide(Env.defaultLayer),
1061
- Layer.provide(emptyAuth),
1062
- Layer.provide(emptyAccount),
1063
- Layer.provideMerge(infra),
1064
- Layer.provide(noopNpm),
1065
- )
1066
-
1067
- try {
1068
- await Instance.provide({
1069
- directory: tmp.path,
1070
- fn: async () => {
1071
- await Effect.runPromise(Config.Service.use((svc) => svc.get()).pipe(Effect.scoped, Effect.provide(testLayer)))
1072
- await Effect.runPromise(
1073
- Config.Service.use((svc) => svc.waitForDependencies()).pipe(Effect.scoped, Effect.provide(testLayer)),
1074
- )
1075
- },
1076
- })
1077
-
1078
- // TODO: this is a hack to wait for backgruounded gitignore
1079
- await new Promise((resolve) => setTimeout(resolve, 1000))
1080
-
1081
- expect(await Filesystem.exists(path.join(tmp.extra, ".gitignore"))).toBe(true)
1082
- expect(await Filesystem.readText(path.join(tmp.extra, ".gitignore"))).toContain("package-lock.json")
1083
- } finally {
1084
- if (prev === undefined) delete process.env.WOOZLIT_CODE_CONFIG_DIR
1085
- else process.env.WOOZLIT_CODE_CONFIG_DIR = prev
1086
- }
1087
- })
1088
-
1089
- // Note: deduplication and serialization of npm installs is now handled by the
1090
- // shared Npm.Service (via EffectFlock). Those behaviors are tested in the shared
1091
- // package's npm tests, not here.
1092
-
1093
- test("resolves scoped npm plugins in config", async () => {
1094
- await using tmp = await tmpdir({
1095
- init: async (dir) => {
1096
- const pluginDir = path.join(dir, "node_modules", "@scope", "plugin")
1097
- await fs.mkdir(pluginDir, { recursive: true })
1098
-
1099
- await Filesystem.write(
1100
- path.join(dir, "package.json"),
1101
- JSON.stringify({ name: "config-fixture", version: "1.0.0", type: "module" }, null, 2),
1102
- )
1103
-
1104
- await Filesystem.write(
1105
- path.join(pluginDir, "package.json"),
1106
- JSON.stringify(
1107
- {
1108
- name: "@scope/plugin",
1109
- version: "1.0.0",
1110
- type: "module",
1111
- main: "./index.js",
1112
- },
1113
- null,
1114
- 2,
1115
- ),
1116
- )
1117
-
1118
- await Filesystem.write(path.join(pluginDir, "index.js"), "export default {}\n")
1119
-
1120
- await Filesystem.write(
1121
- path.join(dir, "WOOZLIT_CODE.json"),
1122
- JSON.stringify({ $schema: "https://opencode.ai/config.json", plugin: ["@scope/plugin"] }, null, 2),
1123
- )
1124
- },
1125
- })
1126
-
1127
- await Instance.provide({
1128
- directory: tmp.path,
1129
- fn: async () => {
1130
- const config = await load()
1131
- const pluginEntries = config.plugin ?? []
1132
- expect(pluginEntries).toContain("@scope/plugin")
1133
- },
1134
- })
1135
- })
1136
-
1137
- test("merges plugin arrays from global and local configs", async () => {
1138
- await using tmp = await tmpdir({
1139
- init: async (dir) => {
1140
- // Create a nested project structure with local .WOOZLIT_CODE config
1141
- const projectDir = path.join(dir, "project")
1142
- const opencodeDir = path.join(projectDir, ".WOOZLIT_CODE")
1143
- await fs.mkdir(opencodeDir, { recursive: true })
1144
-
1145
- // Global config with plugins
1146
- await Filesystem.write(
1147
- path.join(dir, "WOOZLIT_CODE.json"),
1148
- JSON.stringify({
1149
- $schema: "https://opencode.ai/config.json",
1150
- plugin: ["global-plugin-1", "global-plugin-2"],
1151
- }),
1152
- )
1153
-
1154
- // Local .WOOZLIT_CODE config with different plugins
1155
- await Filesystem.write(
1156
- path.join(opencodeDir, "WOOZLIT_CODE.json"),
1157
- JSON.stringify({
1158
- $schema: "https://opencode.ai/config.json",
1159
- plugin: ["local-plugin-1"],
1160
- }),
1161
- )
1162
- },
1163
- })
1164
-
1165
- await Instance.provide({
1166
- directory: path.join(tmp.path, "project"),
1167
- fn: async () => {
1168
- const config = await load()
1169
- const plugins = config.plugin ?? []
1170
-
1171
- // Should contain both global and local plugins
1172
- expect(plugins.some((p) => p.includes("global-plugin-1"))).toBe(true)
1173
- expect(plugins.some((p) => p.includes("global-plugin-2"))).toBe(true)
1174
- expect(plugins.some((p) => p.includes("local-plugin-1"))).toBe(true)
1175
-
1176
- // Should have all 3 plugins (not replaced, but merged)
1177
- const pluginNames = plugins.filter((p) => p.includes("global-plugin") || p.includes("local-plugin"))
1178
- expect(pluginNames.length).toBeGreaterThanOrEqual(3)
1179
- },
1180
- })
1181
- })
1182
-
1183
- test("does not error when only custom agent is a subagent", async () => {
1184
- await using tmp = await tmpdir({
1185
- init: async (dir) => {
1186
- const opencodeDir = path.join(dir, ".WOOZLIT_CODE")
1187
- await fs.mkdir(opencodeDir, { recursive: true })
1188
- const agentDir = path.join(opencodeDir, "agent")
1189
- await fs.mkdir(agentDir, { recursive: true })
1190
-
1191
- await Filesystem.write(
1192
- path.join(agentDir, "helper.md"),
1193
- `---
1194
- model: test/model
1195
- mode: subagent
1196
- ---
1197
- Helper subagent prompt`,
1198
- )
1199
- },
1200
- })
1201
- await Instance.provide({
1202
- directory: tmp.path,
1203
- fn: async () => {
1204
- const config = await load()
1205
- expect(config.agent?.["helper"]).toMatchObject({
1206
- name: "helper",
1207
- model: "test/model",
1208
- mode: "subagent",
1209
- prompt: "Helper subagent prompt",
1210
- })
1211
- },
1212
- })
1213
- })
1214
-
1215
- test("merges instructions arrays from global and local configs", async () => {
1216
- await using tmp = await tmpdir({
1217
- init: async (dir) => {
1218
- const projectDir = path.join(dir, "project")
1219
- const opencodeDir = path.join(projectDir, ".WOOZLIT_CODE")
1220
- await fs.mkdir(opencodeDir, { recursive: true })
1221
-
1222
- await Filesystem.write(
1223
- path.join(dir, "WOOZLIT_CODE.json"),
1224
- JSON.stringify({
1225
- $schema: "https://opencode.ai/config.json",
1226
- instructions: ["global-instructions.md", "shared-rules.md"],
1227
- }),
1228
- )
1229
-
1230
- await Filesystem.write(
1231
- path.join(opencodeDir, "WOOZLIT_CODE.json"),
1232
- JSON.stringify({
1233
- $schema: "https://opencode.ai/config.json",
1234
- instructions: ["local-instructions.md"],
1235
- }),
1236
- )
1237
- },
1238
- })
1239
-
1240
- await Instance.provide({
1241
- directory: path.join(tmp.path, "project"),
1242
- fn: async () => {
1243
- const config = await load()
1244
- const instructions = config.instructions ?? []
1245
-
1246
- expect(instructions).toContain("global-instructions.md")
1247
- expect(instructions).toContain("shared-rules.md")
1248
- expect(instructions).toContain("local-instructions.md")
1249
- expect(instructions.length).toBe(3)
1250
- },
1251
- })
1252
- })
1253
-
1254
- test("deduplicates duplicate instructions from global and local configs", async () => {
1255
- await using tmp = await tmpdir({
1256
- init: async (dir) => {
1257
- const projectDir = path.join(dir, "project")
1258
- const opencodeDir = path.join(projectDir, ".WOOZLIT_CODE")
1259
- await fs.mkdir(opencodeDir, { recursive: true })
1260
-
1261
- await Filesystem.write(
1262
- path.join(dir, "WOOZLIT_CODE.json"),
1263
- JSON.stringify({
1264
- $schema: "https://opencode.ai/config.json",
1265
- instructions: ["duplicate.md", "global-only.md"],
1266
- }),
1267
- )
1268
-
1269
- await Filesystem.write(
1270
- path.join(opencodeDir, "WOOZLIT_CODE.json"),
1271
- JSON.stringify({
1272
- $schema: "https://opencode.ai/config.json",
1273
- instructions: ["duplicate.md", "local-only.md"],
1274
- }),
1275
- )
1276
- },
1277
- })
1278
-
1279
- await Instance.provide({
1280
- directory: path.join(tmp.path, "project"),
1281
- fn: async () => {
1282
- const config = await load()
1283
- const instructions = config.instructions ?? []
1284
-
1285
- expect(instructions).toContain("global-only.md")
1286
- expect(instructions).toContain("local-only.md")
1287
- expect(instructions).toContain("duplicate.md")
1288
-
1289
- const duplicates = instructions.filter((i) => i === "duplicate.md")
1290
- expect(duplicates.length).toBe(1)
1291
- expect(instructions.length).toBe(3)
1292
- },
1293
- })
1294
- })
1295
-
1296
- test("deduplicates duplicate plugins from global and local configs", async () => {
1297
- await using tmp = await tmpdir({
1298
- init: async (dir) => {
1299
- // Create a nested project structure with local .WOOZLIT_CODE config
1300
- const projectDir = path.join(dir, "project")
1301
- const opencodeDir = path.join(projectDir, ".WOOZLIT_CODE")
1302
- await fs.mkdir(opencodeDir, { recursive: true })
1303
-
1304
- // Global config with plugins
1305
- await Filesystem.write(
1306
- path.join(dir, "WOOZLIT_CODE.json"),
1307
- JSON.stringify({
1308
- $schema: "https://opencode.ai/config.json",
1309
- plugin: ["duplicate-plugin", "global-plugin-1"],
1310
- }),
1311
- )
1312
-
1313
- // Local .WOOZLIT_CODE config with some overlapping plugins
1314
- await Filesystem.write(
1315
- path.join(opencodeDir, "WOOZLIT_CODE.json"),
1316
- JSON.stringify({
1317
- $schema: "https://opencode.ai/config.json",
1318
- plugin: ["duplicate-plugin", "local-plugin-1"],
1319
- }),
1320
- )
1321
- },
1322
- })
1323
-
1324
- await Instance.provide({
1325
- directory: path.join(tmp.path, "project"),
1326
- fn: async () => {
1327
- const config = await load()
1328
- const plugins = config.plugin ?? []
1329
-
1330
- // Should contain all unique plugins
1331
- expect(plugins.some((p) => p.includes("global-plugin-1"))).toBe(true)
1332
- expect(plugins.some((p) => p.includes("local-plugin-1"))).toBe(true)
1333
- expect(plugins.some((p) => p.includes("duplicate-plugin"))).toBe(true)
1334
-
1335
- // Should deduplicate the duplicate plugin
1336
- const duplicatePlugins = plugins.filter((p) => p.includes("duplicate-plugin"))
1337
- expect(duplicatePlugins.length).toBe(1)
1338
-
1339
- // Should have exactly 3 unique plugins
1340
- const pluginNames = plugins.filter(
1341
- (p) => p.includes("global-plugin") || p.includes("local-plugin") || p.includes("duplicate-plugin"),
1342
- )
1343
- expect(pluginNames.length).toBe(3)
1344
- },
1345
- })
1346
- })
1347
-
1348
- test("keeps plugin origins aligned with merged plugin list", async () => {
1349
- await using tmp = await tmpdir({
1350
- init: async (dir) => {
1351
- const project = path.join(dir, "project")
1352
- const local = path.join(project, ".WOOZLIT_CODE")
1353
- await fs.mkdir(local, { recursive: true })
1354
-
1355
- await Filesystem.write(
1356
- path.join(dir, "WOOZLIT_CODE.json"),
1357
- JSON.stringify({
1358
- $schema: "https://opencode.ai/config.json",
1359
- plugin: [["shared-plugin@1.0.0", { source: "global" }], "global-only@1.0.0"],
1360
- }),
1361
- )
1362
-
1363
- await Filesystem.write(
1364
- path.join(local, "WOOZLIT_CODE.json"),
1365
- JSON.stringify({
1366
- $schema: "https://opencode.ai/config.json",
1367
- plugin: [["shared-plugin@2.0.0", { source: "local" }], "local-only@1.0.0"],
1368
- }),
1369
- )
1370
- },
1371
- })
1372
-
1373
- await Instance.provide({
1374
- directory: path.join(tmp.path, "project"),
1375
- fn: async () => {
1376
- const cfg = await load()
1377
- const plugins = cfg.plugin ?? []
1378
- const origins = cfg.plugin_origins ?? []
1379
- const names = plugins.map((item) => ConfigPlugin.pluginSpecifier(item))
1380
-
1381
- expect(names).toContain("shared-plugin@2.0.0")
1382
- expect(names).not.toContain("shared-plugin@1.0.0")
1383
- expect(names).toContain("global-only@1.0.0")
1384
- expect(names).toContain("local-only@1.0.0")
1385
-
1386
- expect(origins.map((item) => item.spec)).toEqual(plugins)
1387
- const hit = origins.find((item) => ConfigPlugin.pluginSpecifier(item.spec) === "shared-plugin@2.0.0")
1388
- expect(hit?.scope).toBe("local")
1389
- },
1390
- })
1391
- })
1392
-
1393
- // Legacy tools migration tests
1394
-
1395
- test("migrates legacy tools config to permissions - allow", async () => {
1396
- await using tmp = await tmpdir({
1397
- init: async (dir) => {
1398
- await Filesystem.write(
1399
- path.join(dir, "WOOZLIT_CODE.json"),
1400
- JSON.stringify({
1401
- $schema: "https://opencode.ai/config.json",
1402
- agent: {
1403
- test: {
1404
- tools: {
1405
- bash: true,
1406
- read: true,
1407
- },
1408
- },
1409
- },
1410
- }),
1411
- )
1412
- },
1413
- })
1414
- await Instance.provide({
1415
- directory: tmp.path,
1416
- fn: async () => {
1417
- const config = await load()
1418
- expect(config.agent?.["test"]?.permission).toEqual({
1419
- bash: "allow",
1420
- read: "allow",
1421
- })
1422
- },
1423
- })
1424
- })
1425
-
1426
- test("migrates legacy tools config to permissions - deny", async () => {
1427
- await using tmp = await tmpdir({
1428
- init: async (dir) => {
1429
- await Filesystem.write(
1430
- path.join(dir, "WOOZLIT_CODE.json"),
1431
- JSON.stringify({
1432
- $schema: "https://opencode.ai/config.json",
1433
- agent: {
1434
- test: {
1435
- tools: {
1436
- bash: false,
1437
- webfetch: false,
1438
- },
1439
- },
1440
- },
1441
- }),
1442
- )
1443
- },
1444
- })
1445
- await Instance.provide({
1446
- directory: tmp.path,
1447
- fn: async () => {
1448
- const config = await load()
1449
- expect(config.agent?.["test"]?.permission).toEqual({
1450
- bash: "deny",
1451
- webfetch: "deny",
1452
- })
1453
- },
1454
- })
1455
- })
1456
-
1457
- test("migrates legacy write tool to edit permission", async () => {
1458
- await using tmp = await tmpdir({
1459
- init: async (dir) => {
1460
- await Filesystem.write(
1461
- path.join(dir, "WOOZLIT_CODE.json"),
1462
- JSON.stringify({
1463
- $schema: "https://opencode.ai/config.json",
1464
- agent: {
1465
- test: {
1466
- tools: {
1467
- write: true,
1468
- },
1469
- },
1470
- },
1471
- }),
1472
- )
1473
- },
1474
- })
1475
- await Instance.provide({
1476
- directory: tmp.path,
1477
- fn: async () => {
1478
- const config = await load()
1479
- expect(config.agent?.["test"]?.permission).toEqual({
1480
- edit: "allow",
1481
- })
1482
- },
1483
- })
1484
- })
1485
-
1486
- // Managed settings tests
1487
- // Note: preload.ts sets WOOZLIT_CODE_TEST_MANAGED_CONFIG which Global.Path.managedConfig uses
1488
-
1489
- test("managed settings override user settings", async () => {
1490
- await using tmp = await tmpdir({
1491
- init: async (dir) => {
1492
- await writeConfig(dir, {
1493
- $schema: "https://opencode.ai/config.json",
1494
- model: "user/model",
1495
- share: "auto",
1496
- username: "testuser",
1497
- })
1498
- },
1499
- })
1500
-
1501
- await writeManagedSettings({
1502
- $schema: "https://opencode.ai/config.json",
1503
- model: "managed/model",
1504
- share: "disabled",
1505
- })
1506
-
1507
- await Instance.provide({
1508
- directory: tmp.path,
1509
- fn: async () => {
1510
- const config = await load()
1511
- expect(config.model).toBe("managed/model")
1512
- expect(config.share).toBe("disabled")
1513
- expect(config.username).toBe("testuser")
1514
- },
1515
- })
1516
- })
1517
-
1518
- test("managed settings override project settings", async () => {
1519
- await using tmp = await tmpdir({
1520
- init: async (dir) => {
1521
- await writeConfig(dir, {
1522
- $schema: "https://opencode.ai/config.json",
1523
- autoupdate: true,
1524
- disabled_providers: [],
1525
- })
1526
- },
1527
- })
1528
-
1529
- await writeManagedSettings({
1530
- $schema: "https://opencode.ai/config.json",
1531
- autoupdate: false,
1532
- disabled_providers: ["openai"],
1533
- })
1534
-
1535
- await Instance.provide({
1536
- directory: tmp.path,
1537
- fn: async () => {
1538
- const config = await load()
1539
- expect(config.autoupdate).toBe(false)
1540
- expect(config.disabled_providers).toEqual(["openai"])
1541
- },
1542
- })
1543
- })
1544
-
1545
- test("missing managed settings file is not an error", async () => {
1546
- await using tmp = await tmpdir({
1547
- init: async (dir) => {
1548
- await writeConfig(dir, {
1549
- $schema: "https://opencode.ai/config.json",
1550
- model: "user/model",
1551
- })
1552
- },
1553
- })
1554
-
1555
- await Instance.provide({
1556
- directory: tmp.path,
1557
- fn: async () => {
1558
- const config = await load()
1559
- expect(config.model).toBe("user/model")
1560
- },
1561
- })
1562
- })
1563
-
1564
- test("migrates legacy edit tool to edit permission", async () => {
1565
- await using tmp = await tmpdir({
1566
- init: async (dir) => {
1567
- await Filesystem.write(
1568
- path.join(dir, "WOOZLIT_CODE.json"),
1569
- JSON.stringify({
1570
- $schema: "https://opencode.ai/config.json",
1571
- agent: {
1572
- test: {
1573
- tools: {
1574
- edit: false,
1575
- },
1576
- },
1577
- },
1578
- }),
1579
- )
1580
- },
1581
- })
1582
- await Instance.provide({
1583
- directory: tmp.path,
1584
- fn: async () => {
1585
- const config = await load()
1586
- expect(config.agent?.["test"]?.permission).toEqual({
1587
- edit: "deny",
1588
- })
1589
- },
1590
- })
1591
- })
1592
-
1593
- test("migrates legacy patch tool to edit permission", async () => {
1594
- await using tmp = await tmpdir({
1595
- init: async (dir) => {
1596
- await Filesystem.write(
1597
- path.join(dir, "WOOZLIT_CODE.json"),
1598
- JSON.stringify({
1599
- $schema: "https://opencode.ai/config.json",
1600
- agent: {
1601
- test: {
1602
- tools: {
1603
- patch: true,
1604
- },
1605
- },
1606
- },
1607
- }),
1608
- )
1609
- },
1610
- })
1611
- await Instance.provide({
1612
- directory: tmp.path,
1613
- fn: async () => {
1614
- const config = await load()
1615
- expect(config.agent?.["test"]?.permission).toEqual({
1616
- edit: "allow",
1617
- })
1618
- },
1619
- })
1620
- })
1621
-
1622
- test("migrates legacy multiedit tool to edit permission", async () => {
1623
- await using tmp = await tmpdir({
1624
- init: async (dir) => {
1625
- await Filesystem.write(
1626
- path.join(dir, "WOOZLIT_CODE.json"),
1627
- JSON.stringify({
1628
- $schema: "https://opencode.ai/config.json",
1629
- agent: {
1630
- test: {
1631
- tools: {
1632
- multiedit: false,
1633
- },
1634
- },
1635
- },
1636
- }),
1637
- )
1638
- },
1639
- })
1640
- await Instance.provide({
1641
- directory: tmp.path,
1642
- fn: async () => {
1643
- const config = await load()
1644
- expect(config.agent?.["test"]?.permission).toEqual({
1645
- edit: "deny",
1646
- })
1647
- },
1648
- })
1649
- })
1650
-
1651
- test("migrates mixed legacy tools config", async () => {
1652
- await using tmp = await tmpdir({
1653
- init: async (dir) => {
1654
- await Filesystem.write(
1655
- path.join(dir, "WOOZLIT_CODE.json"),
1656
- JSON.stringify({
1657
- $schema: "https://opencode.ai/config.json",
1658
- agent: {
1659
- test: {
1660
- tools: {
1661
- bash: true,
1662
- write: true,
1663
- read: false,
1664
- webfetch: true,
1665
- },
1666
- },
1667
- },
1668
- }),
1669
- )
1670
- },
1671
- })
1672
- await Instance.provide({
1673
- directory: tmp.path,
1674
- fn: async () => {
1675
- const config = await load()
1676
- expect(config.agent?.["test"]?.permission).toEqual({
1677
- bash: "allow",
1678
- edit: "allow",
1679
- read: "deny",
1680
- webfetch: "allow",
1681
- })
1682
- },
1683
- })
1684
- })
1685
-
1686
- test("merges legacy tools with existing permission config", async () => {
1687
- await using tmp = await tmpdir({
1688
- init: async (dir) => {
1689
- await Filesystem.write(
1690
- path.join(dir, "WOOZLIT_CODE.json"),
1691
- JSON.stringify({
1692
- $schema: "https://opencode.ai/config.json",
1693
- agent: {
1694
- test: {
1695
- permission: {
1696
- glob: "allow",
1697
- },
1698
- tools: {
1699
- bash: true,
1700
- },
1701
- },
1702
- },
1703
- }),
1704
- )
1705
- },
1706
- })
1707
- await Instance.provide({
1708
- directory: tmp.path,
1709
- fn: async () => {
1710
- const config = await load()
1711
- expect(config.agent?.["test"]?.permission).toEqual({
1712
- glob: "allow",
1713
- bash: "allow",
1714
- })
1715
- },
1716
- })
1717
- })
1718
-
1719
- test("permission config preserves key order", async () => {
1720
- await using tmp = await tmpdir({
1721
- init: async (dir) => {
1722
- await Filesystem.write(
1723
- path.join(dir, "WOOZLIT_CODE.json"),
1724
- JSON.stringify({
1725
- $schema: "https://opencode.ai/config.json",
1726
- permission: {
1727
- "*": "deny",
1728
- edit: "ask",
1729
- write: "ask",
1730
- external_directory: "ask",
1731
- read: "allow",
1732
- todowrite: "allow",
1733
- "thoughts_*": "allow",
1734
- "reasoning_model_*": "allow",
1735
- "tools_*": "allow",
1736
- "pr_comments_*": "allow",
1737
- },
1738
- }),
1739
- )
1740
- },
1741
- })
1742
- await Instance.provide({
1743
- directory: tmp.path,
1744
- fn: async () => {
1745
- const config = await load()
1746
- expect(Object.keys(config.permission!)).toEqual([
1747
- "*",
1748
- "edit",
1749
- "write",
1750
- "external_directory",
1751
- "read",
1752
- "todowrite",
1753
- "thoughts_*",
1754
- "reasoning_model_*",
1755
- "tools_*",
1756
- "pr_comments_*",
1757
- ])
1758
- },
1759
- })
1760
- })
1761
-
1762
- // MCP config merging tests
1763
-
1764
- test("project config can override MCP server enabled status", async () => {
1765
- await using tmp = await tmpdir({
1766
- init: async (dir) => {
1767
- // Simulates a base config (like from remote .well-known) with disabled MCP
1768
- await Filesystem.write(
1769
- path.join(dir, "WOOZLIT_CODE.json"),
1770
- JSON.stringify({
1771
- $schema: "https://opencode.ai/config.json",
1772
- mcp: {
1773
- jira: {
1774
- type: "remote",
1775
- url: "https://jira.example.com/mcp",
1776
- enabled: false,
1777
- },
1778
- wiki: {
1779
- type: "remote",
1780
- url: "https://wiki.example.com/mcp",
1781
- enabled: false,
1782
- },
1783
- },
1784
- }),
1785
- )
1786
- // Project config enables just jira
1787
- await Filesystem.write(
1788
- path.join(dir, "WOOZLIT_CODE.jsonc"),
1789
- JSON.stringify({
1790
- $schema: "https://opencode.ai/config.json",
1791
- mcp: {
1792
- jira: {
1793
- type: "remote",
1794
- url: "https://jira.example.com/mcp",
1795
- enabled: true,
1796
- },
1797
- },
1798
- }),
1799
- )
1800
- },
1801
- })
1802
- await Instance.provide({
1803
- directory: tmp.path,
1804
- fn: async () => {
1805
- const config = await load()
1806
- // jira should be enabled (overridden by project config)
1807
- expect(config.mcp?.jira).toEqual({
1808
- type: "remote",
1809
- url: "https://jira.example.com/mcp",
1810
- enabled: true,
1811
- })
1812
- // wiki should still be disabled (not overridden)
1813
- expect(config.mcp?.wiki).toEqual({
1814
- type: "remote",
1815
- url: "https://wiki.example.com/mcp",
1816
- enabled: false,
1817
- })
1818
- },
1819
- })
1820
- })
1821
-
1822
- test("MCP config deep merges preserving base config properties", async () => {
1823
- await using tmp = await tmpdir({
1824
- init: async (dir) => {
1825
- // Base config with full MCP definition
1826
- await Filesystem.write(
1827
- path.join(dir, "WOOZLIT_CODE.json"),
1828
- JSON.stringify({
1829
- $schema: "https://opencode.ai/config.json",
1830
- mcp: {
1831
- myserver: {
1832
- type: "remote",
1833
- url: "https://myserver.example.com/mcp",
1834
- enabled: false,
1835
- headers: {
1836
- "X-Custom-Header": "value",
1837
- },
1838
- },
1839
- },
1840
- }),
1841
- )
1842
- // Override just enables it, should preserve other properties
1843
- await Filesystem.write(
1844
- path.join(dir, "WOOZLIT_CODE.jsonc"),
1845
- JSON.stringify({
1846
- $schema: "https://opencode.ai/config.json",
1847
- mcp: {
1848
- myserver: {
1849
- type: "remote",
1850
- url: "https://myserver.example.com/mcp",
1851
- enabled: true,
1852
- },
1853
- },
1854
- }),
1855
- )
1856
- },
1857
- })
1858
- await Instance.provide({
1859
- directory: tmp.path,
1860
- fn: async () => {
1861
- const config = await load()
1862
- expect(config.mcp?.myserver).toEqual({
1863
- type: "remote",
1864
- url: "https://myserver.example.com/mcp",
1865
- enabled: true,
1866
- headers: {
1867
- "X-Custom-Header": "value",
1868
- },
1869
- })
1870
- },
1871
- })
1872
- })
1873
-
1874
- test("local .WOOZLIT_CODE config can override MCP from project config", async () => {
1875
- await using tmp = await tmpdir({
1876
- init: async (dir) => {
1877
- // Project config with disabled MCP
1878
- await Filesystem.write(
1879
- path.join(dir, "WOOZLIT_CODE.json"),
1880
- JSON.stringify({
1881
- $schema: "https://opencode.ai/config.json",
1882
- mcp: {
1883
- docs: {
1884
- type: "remote",
1885
- url: "https://docs.example.com/mcp",
1886
- enabled: false,
1887
- },
1888
- },
1889
- }),
1890
- )
1891
- // Local .WOOZLIT_CODE directory config enables it
1892
- const opencodeDir = path.join(dir, ".WOOZLIT_CODE")
1893
- await fs.mkdir(opencodeDir, { recursive: true })
1894
- await Filesystem.write(
1895
- path.join(opencodeDir, "WOOZLIT_CODE.json"),
1896
- JSON.stringify({
1897
- $schema: "https://opencode.ai/config.json",
1898
- mcp: {
1899
- docs: {
1900
- type: "remote",
1901
- url: "https://docs.example.com/mcp",
1902
- enabled: true,
1903
- },
1904
- },
1905
- }),
1906
- )
1907
- },
1908
- })
1909
- await Instance.provide({
1910
- directory: tmp.path,
1911
- fn: async () => {
1912
- const config = await load()
1913
- expect(config.mcp?.docs?.enabled).toBe(true)
1914
- },
1915
- })
1916
- })
1917
-
1918
- test("project config overrides remote well-known config", async () => {
1919
- const originalFetch = globalThis.fetch
1920
- let fetchedUrl: string | undefined
1921
- globalThis.fetch = mock((url: string | URL | Request) => {
1922
- const urlStr = url instanceof Request ? url.url : url instanceof URL ? url.href : url
1923
- if (urlStr.includes(".well-known/opencode")) {
1924
- fetchedUrl = urlStr
1925
- return Promise.resolve(
1926
- new Response(
1927
- JSON.stringify({
1928
- config: {
1929
- mcp: { jira: { type: "remote", url: "https://jira.example.com/mcp", enabled: false } },
1930
- },
1931
- }),
1932
- { status: 200 },
1933
- ),
1934
- )
1935
- }
1936
- return originalFetch(url)
1937
- }) as unknown as typeof fetch
1938
-
1939
- const fakeAuth = Layer.mock(Auth.Service)({
1940
- all: () =>
1941
- Effect.succeed({
1942
- "https://example.com": new Auth.WellKnown({ type: "wellknown", key: "TEST_TOKEN", token: "test-token" }),
1943
- }),
1944
- })
1945
-
1946
- const layer = Config.layer.pipe(
1947
- Layer.provide(testFlock),
1948
- Layer.provide(AppFileSystem.defaultLayer),
1949
- Layer.provide(Env.defaultLayer),
1950
- Layer.provide(fakeAuth),
1951
- Layer.provide(emptyAccount),
1952
- Layer.provideMerge(infra),
1953
- Layer.provide(Npm.defaultLayer),
1954
- )
1955
-
1956
- try {
1957
- await provideTmpdirInstance(
1958
- () =>
1959
- Config.Service.use((svc) =>
1960
- Effect.gen(function* () {
1961
- const config = yield* svc.get()
1962
- expect(fetchedUrl).toBe("https://example.com/.well-known/opencode")
1963
- expect(config.mcp?.jira?.enabled).toBe(true)
1964
- }),
1965
- ),
1966
- {
1967
- git: true,
1968
- config: { mcp: { jira: { type: "remote", url: "https://jira.example.com/mcp", enabled: true } } },
1969
- },
1970
- ).pipe(Effect.scoped, Effect.provide(layer), Effect.runPromise)
1971
- } finally {
1972
- globalThis.fetch = originalFetch
1973
- }
1974
- })
1975
-
1976
- test("wellknown URL with trailing slash is normalized", async () => {
1977
- const originalFetch = globalThis.fetch
1978
- let fetchedUrl: string | undefined
1979
- globalThis.fetch = mock((url: string | URL | Request) => {
1980
- const urlStr = url instanceof Request ? url.url : url instanceof URL ? url.href : url
1981
- if (urlStr.includes(".well-known/opencode")) {
1982
- fetchedUrl = urlStr
1983
- return Promise.resolve(
1984
- new Response(
1985
- JSON.stringify({
1986
- config: {
1987
- mcp: { slack: { type: "remote", url: "https://slack.example.com/mcp", enabled: true } },
1988
- },
1989
- }),
1990
- { status: 200 },
1991
- ),
1992
- )
1993
- }
1994
- return originalFetch(url)
1995
- }) as unknown as typeof fetch
1996
-
1997
- const fakeAuth = Layer.mock(Auth.Service)({
1998
- all: () =>
1999
- Effect.succeed({
2000
- "https://example.com/": new Auth.WellKnown({ type: "wellknown", key: "TEST_TOKEN", token: "test-token" }),
2001
- }),
2002
- })
2003
-
2004
- const layer = Config.layer.pipe(
2005
- Layer.provide(testFlock),
2006
- Layer.provide(AppFileSystem.defaultLayer),
2007
- Layer.provide(Env.defaultLayer),
2008
- Layer.provide(fakeAuth),
2009
- Layer.provide(emptyAccount),
2010
- Layer.provideMerge(infra),
2011
- Layer.provide(Npm.defaultLayer),
2012
- )
2013
-
2014
- try {
2015
- await provideTmpdirInstance(
2016
- () =>
2017
- Config.Service.use((svc) =>
2018
- Effect.gen(function* () {
2019
- yield* svc.get()
2020
- expect(fetchedUrl).toBe("https://example.com/.well-known/opencode")
2021
- }),
2022
- ),
2023
- { git: true },
2024
- ).pipe(Effect.scoped, Effect.provide(layer), Effect.runPromise)
2025
- } finally {
2026
- globalThis.fetch = originalFetch
2027
- }
2028
- })
2029
-
2030
- describe("resolvePluginSpec", () => {
2031
- test("keeps package specs unchanged", async () => {
2032
- await using tmp = await tmpdir()
2033
- const file = path.join(tmp.path, "WOOZLIT_CODE.json")
2034
- expect(await ConfigPlugin.resolvePluginSpec("oh-my-opencode@2.4.3", file)).toBe("oh-my-opencode@2.4.3")
2035
- expect(await ConfigPlugin.resolvePluginSpec("@scope/pkg", file)).toBe("@scope/pkg")
2036
- })
2037
-
2038
- test("resolves windows-style relative plugin directory specs", async () => {
2039
- if (process.platform !== "win32") return
2040
-
2041
- await using tmp = await tmpdir({
2042
- init: async (dir) => {
2043
- const plugin = path.join(dir, "plugin")
2044
- await fs.mkdir(plugin, { recursive: true })
2045
- await Filesystem.write(path.join(plugin, "index.ts"), "export default {}")
2046
- },
2047
- })
2048
-
2049
- const file = path.join(tmp.path, "WOOZLIT_CODE.json")
2050
- const hit = await ConfigPlugin.resolvePluginSpec(".\\plugin", file)
2051
- expect(ConfigPlugin.pluginSpecifier(hit)).toBe(pathToFileURL(path.join(tmp.path, "plugin", "index.ts")).href)
2052
- })
2053
-
2054
- test("resolves relative file plugin paths to file urls", async () => {
2055
- await using tmp = await tmpdir({
2056
- init: async (dir) => {
2057
- await Filesystem.write(path.join(dir, "plugin.ts"), "export default {}")
2058
- },
2059
- })
2060
-
2061
- const file = path.join(tmp.path, "WOOZLIT_CODE.json")
2062
- const hit = await ConfigPlugin.resolvePluginSpec("./plugin.ts", file)
2063
- expect(ConfigPlugin.pluginSpecifier(hit)).toBe(pathToFileURL(path.join(tmp.path, "plugin.ts")).href)
2064
- })
2065
-
2066
- test("resolves plugin directory paths to directory urls", async () => {
2067
- await using tmp = await tmpdir({
2068
- init: async (dir) => {
2069
- const plugin = path.join(dir, "plugin")
2070
- await fs.mkdir(plugin, { recursive: true })
2071
- await Filesystem.writeJson(path.join(plugin, "package.json"), {
2072
- name: "demo-plugin",
2073
- type: "module",
2074
- main: "./index.ts",
2075
- })
2076
- await Filesystem.write(path.join(plugin, "index.ts"), "export default {}")
2077
- },
2078
- })
2079
-
2080
- const file = path.join(tmp.path, "WOOZLIT_CODE.json")
2081
- const hit = await ConfigPlugin.resolvePluginSpec("./plugin", file)
2082
- expect(ConfigPlugin.pluginSpecifier(hit)).toBe(pathToFileURL(path.join(tmp.path, "plugin")).href)
2083
- })
2084
-
2085
- test("resolves plugin directories without package.json to index.ts", async () => {
2086
- await using tmp = await tmpdir({
2087
- init: async (dir) => {
2088
- const plugin = path.join(dir, "plugin")
2089
- await fs.mkdir(plugin, { recursive: true })
2090
- await Filesystem.write(path.join(plugin, "index.ts"), "export default {}")
2091
- },
2092
- })
2093
-
2094
- const file = path.join(tmp.path, "WOOZLIT_CODE.json")
2095
- const hit = await ConfigPlugin.resolvePluginSpec("./plugin", file)
2096
- expect(ConfigPlugin.pluginSpecifier(hit)).toBe(pathToFileURL(path.join(tmp.path, "plugin", "index.ts")).href)
2097
- })
2098
- })
2099
-
2100
- describe("deduplicatePluginOrigins", () => {
2101
- const dedupe = (plugins: ConfigPlugin.Spec[]) =>
2102
- ConfigPlugin.deduplicatePluginOrigins(
2103
- plugins.map((spec) => ({
2104
- spec,
2105
- source: "",
2106
- scope: "global" as const,
2107
- })),
2108
- ).map((item) => item.spec)
2109
-
2110
- test("removes duplicates keeping higher priority (later entries)", () => {
2111
- const plugins = ["global-plugin@1.0.0", "shared-plugin@1.0.0", "local-plugin@2.0.0", "shared-plugin@2.0.0"]
2112
-
2113
- const result = dedupe(plugins)
2114
-
2115
- expect(result).toContain("global-plugin@1.0.0")
2116
- expect(result).toContain("local-plugin@2.0.0")
2117
- expect(result).toContain("shared-plugin@2.0.0")
2118
- expect(result).not.toContain("shared-plugin@1.0.0")
2119
- expect(result.length).toBe(3)
2120
- })
2121
-
2122
- test("keeps path plugins separate from package plugins", () => {
2123
- const plugins = ["oh-my-opencode@2.4.3", "file:///project/.WOOZLIT_CODE/plugin/oh-my-opencode.js"]
2124
-
2125
- const result = dedupe(plugins)
2126
-
2127
- expect(result).toEqual(plugins)
2128
- })
2129
-
2130
- test("deduplicates direct path plugins by exact spec", () => {
2131
- const plugins = ["file:///project/.WOOZLIT_CODE/plugin/demo.ts", "file:///project/.WOOZLIT_CODE/plugin/demo.ts"]
2132
-
2133
- const result = dedupe(plugins)
2134
-
2135
- expect(result).toEqual(["file:///project/.WOOZLIT_CODE/plugin/demo.ts"])
2136
- })
2137
-
2138
- test("preserves order of remaining plugins", () => {
2139
- const plugins = ["a-plugin@1.0.0", "b-plugin@1.0.0", "c-plugin@1.0.0"]
2140
-
2141
- const result = dedupe(plugins)
2142
-
2143
- expect(result).toEqual(["a-plugin@1.0.0", "b-plugin@1.0.0", "c-plugin@1.0.0"])
2144
- })
2145
-
2146
- test("loads auto-discovered local plugins as file urls", async () => {
2147
- await using tmp = await tmpdir({
2148
- init: async (dir) => {
2149
- const projectDir = path.join(dir, "project")
2150
- const opencodeDir = path.join(projectDir, ".WOOZLIT_CODE")
2151
- const pluginDir = path.join(opencodeDir, "plugin")
2152
- await fs.mkdir(pluginDir, { recursive: true })
2153
-
2154
- await Filesystem.write(
2155
- path.join(dir, "WOOZLIT_CODE.json"),
2156
- JSON.stringify({
2157
- $schema: "https://opencode.ai/config.json",
2158
- plugin: ["my-plugin@1.0.0"],
2159
- }),
2160
- )
2161
-
2162
- await Filesystem.write(path.join(pluginDir, "my-plugin.js"), "export default {}")
2163
- },
2164
- })
2165
-
2166
- await Instance.provide({
2167
- directory: path.join(tmp.path, "project"),
2168
- fn: async () => {
2169
- const config = await load()
2170
- const plugins = config.plugin ?? []
2171
-
2172
- expect(plugins.some((p) => ConfigPlugin.pluginSpecifier(p) === "my-plugin@1.0.0")).toBe(true)
2173
- expect(plugins.some((p) => ConfigPlugin.pluginSpecifier(p).startsWith("file://"))).toBe(true)
2174
- },
2175
- })
2176
- })
2177
- })
2178
-
2179
- describe("WOOZLIT_CODE_DISABLE_PROJECT_CONFIG", () => {
2180
- test("skips project config files when flag is set", async () => {
2181
- const originalEnv = process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2182
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = "true"
2183
-
2184
- try {
2185
- await using tmp = await tmpdir({
2186
- init: async (dir) => {
2187
- // Create a project config that would normally be loaded
2188
- await Filesystem.write(
2189
- path.join(dir, "WOOZLIT_CODE.json"),
2190
- JSON.stringify({
2191
- $schema: "https://opencode.ai/config.json",
2192
- model: "project/model",
2193
- username: "project-user",
2194
- }),
2195
- )
2196
- },
2197
- })
2198
- await Instance.provide({
2199
- directory: tmp.path,
2200
- fn: async () => {
2201
- const config = await load()
2202
- // Project config should NOT be loaded - model should be default, not "project/model"
2203
- expect(config.model).not.toBe("project/model")
2204
- expect(config.username).not.toBe("project-user")
2205
- },
2206
- })
2207
- } finally {
2208
- if (originalEnv === undefined) {
2209
- delete process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2210
- } else {
2211
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = originalEnv
2212
- }
2213
- }
2214
- })
2215
-
2216
- test("skips project .WOOZLIT_CODE/ directories when flag is set", async () => {
2217
- const originalEnv = process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2218
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = "true"
2219
-
2220
- try {
2221
- await using tmp = await tmpdir({
2222
- init: async (dir) => {
2223
- // Create a .WOOZLIT_CODE directory with a command
2224
- const opencodeDir = path.join(dir, ".WOOZLIT_CODE", "command")
2225
- await fs.mkdir(opencodeDir, { recursive: true })
2226
- await Filesystem.write(path.join(opencodeDir, "test-cmd.md"), "# Test Command\nThis is a test command.")
2227
- },
2228
- })
2229
- await Instance.provide({
2230
- directory: tmp.path,
2231
- fn: async () => {
2232
- const directories = await listDirs()
2233
- // Project .WOOZLIT_CODE should NOT be in directories list
2234
- const hasProjectOpencode = directories.some((d) => d.startsWith(tmp.path))
2235
- expect(hasProjectOpencode).toBe(false)
2236
- },
2237
- })
2238
- } finally {
2239
- if (originalEnv === undefined) {
2240
- delete process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2241
- } else {
2242
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = originalEnv
2243
- }
2244
- }
2245
- })
2246
-
2247
- test("still loads global config when flag is set", async () => {
2248
- const originalEnv = process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2249
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = "true"
2250
-
2251
- try {
2252
- await using tmp = await tmpdir()
2253
- await Instance.provide({
2254
- directory: tmp.path,
2255
- fn: async () => {
2256
- // Should still get default config (from global or defaults)
2257
- const config = await load()
2258
- expect(config).toBeDefined()
2259
- expect(config.username).toBeDefined()
2260
- },
2261
- })
2262
- } finally {
2263
- if (originalEnv === undefined) {
2264
- delete process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2265
- } else {
2266
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = originalEnv
2267
- }
2268
- }
2269
- })
2270
-
2271
- test("skips relative instructions with warning when flag is set but no config dir", async () => {
2272
- const originalDisable = process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2273
- const originalConfigDir = process.env["WOOZLIT_CODE_CONFIG_DIR"]
2274
-
2275
- try {
2276
- // Ensure no config dir is set
2277
- delete process.env["WOOZLIT_CODE_CONFIG_DIR"]
2278
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = "true"
2279
-
2280
- await using tmp = await tmpdir({
2281
- init: async (dir) => {
2282
- // Create a config with relative instruction path
2283
- await Filesystem.write(
2284
- path.join(dir, "WOOZLIT_CODE.json"),
2285
- JSON.stringify({
2286
- $schema: "https://opencode.ai/config.json",
2287
- instructions: ["./CUSTOM.md"],
2288
- }),
2289
- )
2290
- // Create the instruction file (should be skipped)
2291
- await Filesystem.write(path.join(dir, "CUSTOM.md"), "# Custom Instructions")
2292
- },
2293
- })
2294
-
2295
- await Instance.provide({
2296
- directory: tmp.path,
2297
- fn: async () => {
2298
- // The relative instruction should be skipped without error
2299
- // We're mainly verifying this doesn't throw and the config loads
2300
- const config = await load()
2301
- expect(config).toBeDefined()
2302
- // The instruction should have been skipped (warning logged)
2303
- // We can't easily test the warning was logged, but we verify
2304
- // the relative path didn't cause an error
2305
- },
2306
- })
2307
- } finally {
2308
- if (originalDisable === undefined) {
2309
- delete process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2310
- } else {
2311
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = originalDisable
2312
- }
2313
- if (originalConfigDir === undefined) {
2314
- delete process.env["WOOZLIT_CODE_CONFIG_DIR"]
2315
- } else {
2316
- process.env["WOOZLIT_CODE_CONFIG_DIR"] = originalConfigDir
2317
- }
2318
- }
2319
- })
2320
-
2321
- test("WOOZLIT_CODE_CONFIG_DIR still works when flag is set", async () => {
2322
- const originalDisable = process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2323
- const originalConfigDir = process.env["WOOZLIT_CODE_CONFIG_DIR"]
2324
-
2325
- try {
2326
- await using configDirTmp = await tmpdir({
2327
- init: async (dir) => {
2328
- // Create config in the custom config dir
2329
- await Filesystem.write(
2330
- path.join(dir, "WOOZLIT_CODE.json"),
2331
- JSON.stringify({
2332
- $schema: "https://opencode.ai/config.json",
2333
- model: "configdir/model",
2334
- }),
2335
- )
2336
- },
2337
- })
2338
-
2339
- await using projectTmp = await tmpdir({
2340
- init: async (dir) => {
2341
- // Create config in project (should be ignored)
2342
- await Filesystem.write(
2343
- path.join(dir, "WOOZLIT_CODE.json"),
2344
- JSON.stringify({
2345
- $schema: "https://opencode.ai/config.json",
2346
- model: "project/model",
2347
- }),
2348
- )
2349
- },
2350
- })
2351
-
2352
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = "true"
2353
- process.env["WOOZLIT_CODE_CONFIG_DIR"] = configDirTmp.path
2354
-
2355
- await Instance.provide({
2356
- directory: projectTmp.path,
2357
- fn: async () => {
2358
- const config = await load()
2359
- // Should load from WOOZLIT_CODE_CONFIG_DIR, not project
2360
- expect(config.model).toBe("configdir/model")
2361
- },
2362
- })
2363
- } finally {
2364
- if (originalDisable === undefined) {
2365
- delete process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"]
2366
- } else {
2367
- process.env["WOOZLIT_CODE_DISABLE_PROJECT_CONFIG"] = originalDisable
2368
- }
2369
- if (originalConfigDir === undefined) {
2370
- delete process.env["WOOZLIT_CODE_CONFIG_DIR"]
2371
- } else {
2372
- process.env["WOOZLIT_CODE_CONFIG_DIR"] = originalConfigDir
2373
- }
2374
- }
2375
- })
2376
- })
2377
-
2378
- describe("WOOZLIT_CODE_CONFIG_CONTENT token substitution", () => {
2379
- test("substitutes {env:} tokens in WOOZLIT_CODE_CONFIG_CONTENT", async () => {
2380
- const originalEnv = process.env["WOOZLIT_CODE_CONFIG_CONTENT"]
2381
- const originalTestVar = process.env["TEST_CONFIG_VAR"]
2382
- process.env["TEST_CONFIG_VAR"] = "test_api_key_12345"
2383
- process.env["WOOZLIT_CODE_CONFIG_CONTENT"] = JSON.stringify({
2384
- $schema: "https://opencode.ai/config.json",
2385
- username: "{env:TEST_CONFIG_VAR}",
2386
- })
2387
-
2388
- try {
2389
- await using tmp = await tmpdir()
2390
- await Instance.provide({
2391
- directory: tmp.path,
2392
- fn: async () => {
2393
- const config = await load()
2394
- expect(config.username).toBe("test_api_key_12345")
2395
- },
2396
- })
2397
- } finally {
2398
- if (originalEnv !== undefined) {
2399
- process.env["WOOZLIT_CODE_CONFIG_CONTENT"] = originalEnv
2400
- } else {
2401
- delete process.env["WOOZLIT_CODE_CONFIG_CONTENT"]
2402
- }
2403
- if (originalTestVar !== undefined) {
2404
- process.env["TEST_CONFIG_VAR"] = originalTestVar
2405
- } else {
2406
- delete process.env["TEST_CONFIG_VAR"]
2407
- }
2408
- }
2409
- })
2410
-
2411
- test("substitutes {file:} tokens in WOOZLIT_CODE_CONFIG_CONTENT", async () => {
2412
- const originalEnv = process.env["WOOZLIT_CODE_CONFIG_CONTENT"]
2413
-
2414
- try {
2415
- await using tmp = await tmpdir({
2416
- init: async (dir) => {
2417
- await Filesystem.write(path.join(dir, "api_key.txt"), "secret_key_from_file")
2418
- process.env["WOOZLIT_CODE_CONFIG_CONTENT"] = JSON.stringify({
2419
- $schema: "https://opencode.ai/config.json",
2420
- username: "{file:./api_key.txt}",
2421
- })
2422
- },
2423
- })
2424
- await Instance.provide({
2425
- directory: tmp.path,
2426
- fn: async () => {
2427
- const config = await load()
2428
- expect(config.username).toBe("secret_key_from_file")
2429
- },
2430
- })
2431
- } finally {
2432
- if (originalEnv !== undefined) {
2433
- process.env["WOOZLIT_CODE_CONFIG_CONTENT"] = originalEnv
2434
- } else {
2435
- delete process.env["WOOZLIT_CODE_CONFIG_CONTENT"]
2436
- }
2437
- }
2438
- })
2439
- })
2440
-
2441
- // parseManagedPlist unit tests — pure function, no OS interaction
2442
-
2443
- test("parseManagedPlist strips MDM metadata keys", async () => {
2444
- const config = ConfigParse.schema(
2445
- Config.Info,
2446
- ConfigParse.jsonc(
2447
- await ConfigManaged.parseManagedPlist(
2448
- JSON.stringify({
2449
- PayloadDisplayName: "OpenCode Managed",
2450
- PayloadIdentifier: "ai.WOOZLIT_CODE.managed.test",
2451
- PayloadType: "ai.WOOZLIT_CODE.managed",
2452
- PayloadUUID: "AAAA-BBBB-CCCC",
2453
- PayloadVersion: 1,
2454
- _manualProfile: true,
2455
- share: "disabled",
2456
- model: "mdm/model",
2457
- }),
2458
- ),
2459
- "test:mobileconfig",
2460
- ),
2461
- "test:mobileconfig",
2462
- )
2463
- expect(config.share).toBe("disabled")
2464
- expect(config.model).toBe("mdm/model")
2465
- // MDM keys must not leak into the parsed config
2466
- expect((config as any).PayloadUUID).toBeUndefined()
2467
- expect((config as any).PayloadType).toBeUndefined()
2468
- expect((config as any)._manualProfile).toBeUndefined()
2469
- })
2470
-
2471
- test("parseManagedPlist parses server settings", async () => {
2472
- const config = ConfigParse.schema(
2473
- Config.Info,
2474
- ConfigParse.jsonc(
2475
- await ConfigManaged.parseManagedPlist(
2476
- JSON.stringify({
2477
- $schema: "https://opencode.ai/config.json",
2478
- server: { hostname: "127.0.0.1", mdns: false },
2479
- autoupdate: true,
2480
- }),
2481
- ),
2482
- "test:mobileconfig",
2483
- ),
2484
- "test:mobileconfig",
2485
- )
2486
- expect(config.server?.hostname).toBe("127.0.0.1")
2487
- expect(config.server?.mdns).toBe(false)
2488
- expect(config.autoupdate).toBe(true)
2489
- })
2490
-
2491
- test("parseManagedPlist parses permission rules", async () => {
2492
- const config = ConfigParse.schema(
2493
- Config.Info,
2494
- ConfigParse.jsonc(
2495
- await ConfigManaged.parseManagedPlist(
2496
- JSON.stringify({
2497
- $schema: "https://opencode.ai/config.json",
2498
- permission: {
2499
- "*": "ask",
2500
- bash: { "*": "ask", "rm -rf *": "deny", "curl *": "deny" },
2501
- grep: "allow",
2502
- glob: "allow",
2503
- webfetch: "ask",
2504
- "~/.ssh/*": "deny",
2505
- },
2506
- }),
2507
- ),
2508
- "test:mobileconfig",
2509
- ),
2510
- "test:mobileconfig",
2511
- )
2512
- expect(config.permission?.["*"]).toBe("ask")
2513
- expect(config.permission?.grep).toBe("allow")
2514
- expect(config.permission?.webfetch).toBe("ask")
2515
- expect(config.permission?.["~/.ssh/*"]).toBe("deny")
2516
- const bash = config.permission?.bash as Record<string, string>
2517
- expect(bash?.["rm -rf *"]).toBe("deny")
2518
- expect(bash?.["curl *"]).toBe("deny")
2519
- })
2520
-
2521
- test("parseManagedPlist parses enabled_providers", async () => {
2522
- const config = ConfigParse.schema(
2523
- Config.Info,
2524
- ConfigParse.jsonc(
2525
- await ConfigManaged.parseManagedPlist(
2526
- JSON.stringify({
2527
- $schema: "https://opencode.ai/config.json",
2528
- enabled_providers: ["anthropic", "google"],
2529
- }),
2530
- ),
2531
- "test:mobileconfig",
2532
- ),
2533
- "test:mobileconfig",
2534
- )
2535
- expect(config.enabled_providers).toEqual(["anthropic", "google"])
2536
- })
2537
-
2538
- test("parseManagedPlist handles empty config", async () => {
2539
- const config = ConfigParse.schema(
2540
- Config.Info,
2541
- ConfigParse.jsonc(
2542
- await ConfigManaged.parseManagedPlist(JSON.stringify({ $schema: "https://opencode.ai/config.json" })),
2543
- "test:mobileconfig",
2544
- ),
2545
- "test:mobileconfig",
2546
- )
2547
- expect(config.$schema).toBe("https://opencode.ai/config.json")
2548
- })
2549
-
2550
- describe("tool config inline struct", () => {
2551
- test("accepts invocation_style and invocation_style_by_tool", () => {
2552
- const config = Config.Info.parse({
2553
- $schema: "https://opencode.ai/config.json",
2554
- tool: { invocation_style: "shell", invocation_style_by_tool: { task: "shell", read: "json" } },
2555
- })
2556
- expect(config.tool?.invocation_style).toBe("shell")
2557
- expect(config.tool?.invocation_style_by_tool).toEqual({ task: "shell", read: "json" })
2558
- })
2559
-
2560
- test("accepts empty tool object (all fields optional)", () => {
2561
- const config = Config.Info.parse({
2562
- $schema: "https://opencode.ai/config.json",
2563
- tool: {},
2564
- })
2565
- expect(config.tool?.invocation_style).toBeUndefined()
2566
- expect(config.tool?.invocation_style_by_tool).toBeUndefined()
2567
- })
2568
-
2569
- test("rejects unknown invocation_style value", () => {
2570
- expect(() =>
2571
- Config.Info.parse({
2572
- $schema: "https://opencode.ai/config.json",
2573
- tool: { invocation_style: "verb" },
2574
- }),
2575
- ).toThrow()
2576
- })
2577
- })