saeeol 1.0.8 → 1.1.0

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