saeeol 1.3.0 → 1.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (537) hide show
  1. package/.turbo/turbo-typecheck.log +1 -0
  2. package/AGENTS.md +72 -0
  3. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  4. package/Dockerfile +18 -0
  5. package/assets/saeeol.ico +0 -0
  6. package/bin/saeeol.cjs +0 -0
  7. package/database.db +0 -0
  8. package/drizzle.config.ts +10 -0
  9. package/git +0 -0
  10. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  11. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  12. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  13. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  14. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  15. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  16. package/migration/20260225215848_workspace/migration.sql +7 -0
  17. package/migration/20260225215848_workspace/snapshot.json +959 -0
  18. package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
  19. package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
  20. package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
  21. package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
  22. package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
  23. package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
  24. package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
  25. package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
  26. package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
  27. package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
  28. package/migration/20260323234822_events/migration.sql +13 -0
  29. package/migration/20260323234822_events/snapshot.json +1271 -0
  30. package/migration/20260410174513_workspace-name/migration.sql +16 -0
  31. package/migration/20260410174513_workspace-name/snapshot.json +1271 -0
  32. package/migration/20260413175956_chief_energizer/migration.sql +13 -0
  33. package/migration/20260413175956_chief_energizer/snapshot.json +1399 -0
  34. package/migration/20260423070820_add_icon_url_override/migration.sql +2 -0
  35. package/migration/20260423070820_add_icon_url_override/snapshot.json +1409 -0
  36. package/migration/20260428004200_add_session_path/migration.sql +1 -0
  37. package/migration/20260428004200_add_session_path/snapshot.json +1419 -0
  38. package/npm/bin/saeeol +42 -0
  39. package/npm/package.json +39 -0
  40. package/npm/postinstall.js +162 -0
  41. package/package.json +201 -207
  42. package/parsers-config.ts +289 -0
  43. package/script/build.ts +393 -0
  44. package/script/check-migrations.ts +16 -0
  45. package/script/fix-node-pty.ts +34 -0
  46. package/script/generate.ts +23 -0
  47. package/script/postinstall.mjs +189 -0
  48. package/script/publish.ts +200 -0
  49. package/script/run-workspace-server +106 -0
  50. package/script/schema.ts +63 -0
  51. package/script/test-runner.ts +420 -0
  52. package/script/time.ts +6 -0
  53. package/script/trace-imports.ts +153 -0
  54. package/script/upgrade-opentui.ts +64 -0
  55. package/scripts/diff-sdk-types.sh +52 -0
  56. package/specs/effect/facades.md +221 -0
  57. package/specs/effect/http-api.md +401 -0
  58. package/specs/effect/instance-context.md +309 -0
  59. package/specs/effect/loose-ends.md +34 -0
  60. package/specs/effect/migration.md +299 -0
  61. package/specs/effect/routes.md +64 -0
  62. package/specs/effect/schema.md +399 -0
  63. package/specs/effect/server-package.md +668 -0
  64. package/specs/effect/tools.md +90 -0
  65. package/specs/tui-plugins.md +433 -0
  66. package/specs/v2/api.ts +67 -0
  67. package/specs/v2/keymappings.md +10 -0
  68. package/specs/v2/message-shape.md +136 -0
  69. package/src/acp/agent-message.ts +1 -1
  70. package/src/acp/agent-utils.ts +1 -1
  71. package/src/boxes/ansi.ts +17 -0
  72. package/src/boxes/atomic-write.ts +35 -0
  73. package/src/boxes/b64.ts +58 -0
  74. package/src/boxes/bash-security.ts +129 -0
  75. package/src/boxes/bom.ts +18 -0
  76. package/src/boxes/cancel.ts +16 -0
  77. package/src/boxes/chop.ts +12 -0
  78. package/src/boxes/clamp.ts +3 -0
  79. package/src/boxes/compact.ts +9 -0
  80. package/src/boxes/cost-tracker.ts +116 -0
  81. package/src/boxes/dataurl.ts +29 -0
  82. package/src/boxes/delay.ts +27 -0
  83. package/src/boxes/diff-apply.ts +53 -0
  84. package/src/boxes/disposable.ts +13 -0
  85. package/src/boxes/err.ts +34 -0
  86. package/src/boxes/human.ts +47 -0
  87. package/src/boxes/iife.ts +9 -0
  88. package/src/boxes/latch.ts +8 -0
  89. package/src/boxes/memory.ts +198 -0
  90. package/src/boxes/net.ts +16 -0
  91. package/src/boxes/plural.ts +4 -0
  92. package/src/boxes/puny.ts +21 -0
  93. package/src/boxes/retry.ts +49 -0
  94. package/src/boxes/rwlock.ts +41 -0
  95. package/src/boxes/schedule.ts +71 -0
  96. package/src/boxes/scope.ts +21 -0
  97. package/src/boxes/tokens.ts +9 -0
  98. package/src/boxes/ttl-cache.ts +63 -0
  99. package/src/boxes/typed-event.ts +51 -0
  100. package/src/boxes/uid.ts +50 -0
  101. package/src/boxes/wave6.test.ts +296 -0
  102. package/src/boxes/wildcard.ts +58 -0
  103. package/src/bus/global.ts +1 -1
  104. package/src/cli/cmd/github-run-api.ts +2 -2
  105. package/src/cli/cmd/run-events.ts +2 -2
  106. package/src/cli/cmd/tui/component/logo.tsx +1 -1
  107. package/src/cli/cmd/tui/component/prompt/use-prompt-memos.ts +2 -2
  108. package/src/cli/cmd/tui/context/app/editor-zed.ts +1 -1
  109. package/src/cli/cmd/tui/context/app/editor.ts +1 -1
  110. package/src/cli/cmd/tui/context/app/theme.tsx +1 -1
  111. package/src/cli/cmd/tui/util/revert-diff.ts +1 -1
  112. package/src/overlay/cli/cmd/roll-call-call.ts +1 -1
  113. package/src/overlay/cost-tracker/format.ts +1 -1
  114. package/src/overlay/cost-tracker/index.ts +4 -4
  115. package/src/overlay/cost-tracker/state.ts +2 -2
  116. package/src/overlay/cost-tracker/types.ts +2 -2
  117. package/src/overlay/memory/age.ts +1 -1
  118. package/src/overlay/memory/index.ts +4 -4
  119. package/src/overlay/memory/paths.ts +2 -2
  120. package/src/overlay/memory/scan.ts +1 -1
  121. package/src/overlay/memory/types.ts +2 -2
  122. package/src/overlay/tool/bash-security.ts +3 -3
  123. package/src/overlay/util/url.ts +1 -1
  124. package/src/plugin/codex-auth.ts +1 -1
  125. package/src/provider/model-cache.ts +2 -2
  126. package/src/provider/provider-resolve.ts +3 -3
  127. package/src/provider/transform-message.ts +1 -1
  128. package/src/server/routes/game.ts +284 -0
  129. package/src/server/server.ts +2 -0
  130. package/src/session/core/compaction/compaction-helpers.ts +1 -1
  131. package/src/session/core/compaction/compaction.ts +1 -1
  132. package/src/session/core/session.ts +2 -0
  133. package/src/sessions/ingest-queue.ts +2 -2
  134. package/src/sessions/remote-ws.ts +1 -1
  135. package/src/tool/workflow/question.ts +1 -1
  136. package/src/util/abort.ts +1 -1
  137. package/src/util/bom.ts +2 -2
  138. package/src/util/color.ts +1 -1
  139. package/src/util/data-url.ts +1 -1
  140. package/src/util/defer.ts +1 -1
  141. package/src/util/error.ts +2 -2
  142. package/src/util/filesystem.ts +2 -2
  143. package/src/util/format.ts +1 -1
  144. package/src/util/iife.ts +1 -1
  145. package/src/util/local-context.ts +1 -1
  146. package/src/util/locale.ts +2 -2
  147. package/src/util/lock.ts +1 -1
  148. package/src/util/network.ts +1 -1
  149. package/src/util/signal.ts +1 -1
  150. package/src/util/token.ts +1 -1
  151. package/src/util/wildcard.ts +1 -1
  152. package/sst-env.d.ts +10 -0
  153. package/test/AGENTS.md +133 -0
  154. package/test/account/repo.test.ts +352 -0
  155. package/test/account/service.test.ts +456 -0
  156. package/test/acp/agent-interface.test.ts +51 -0
  157. package/test/acp/event-subscription.test.ts +725 -0
  158. package/test/agent/agent.test.ts +890 -0
  159. package/test/auth/auth.test.ts +86 -0
  160. package/test/bun/registry.test.ts +75 -0
  161. package/test/bus/bus-effect.test.ts +161 -0
  162. package/test/bus/bus-integration.test.ts +87 -0
  163. package/test/bus/bus.test.ts +219 -0
  164. package/test/cli/account.test.ts +26 -0
  165. package/test/cli/auto-mode.test.ts +75 -0
  166. package/test/cli/bin-saeeol.test.ts +8 -0
  167. package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
  168. package/test/cli/cmd/tui/prompt-traits.test.ts +38 -0
  169. package/test/cli/cmd/tui/sync.test.tsx +159 -0
  170. package/test/cli/error.test.ts +18 -0
  171. package/test/cli/github-action.test.ts +198 -0
  172. package/test/cli/github-remote.test.ts +85 -0
  173. package/test/cli/import.test.ts +97 -0
  174. package/test/cli/install-artifact.test.ts +72 -0
  175. package/test/cli/plugin-auth-picker.test.ts +120 -0
  176. package/test/cli/pr.test.ts +59 -0
  177. package/test/cli/tui/editor-context-zed.test.ts +356 -0
  178. package/test/cli/tui/editor-context.test.tsx +228 -0
  179. package/test/cli/tui/keybind-plugin.test.ts +90 -0
  180. package/test/cli/tui/markdown.test.ts +161 -0
  181. package/test/cli/tui/plugin-add.test.ts +111 -0
  182. package/test/cli/tui/plugin-install.test.ts +87 -0
  183. package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
  184. package/test/cli/tui/plugin-loader-entrypoint.test.ts +484 -0
  185. package/test/cli/tui/plugin-loader-pure.test.ts +71 -0
  186. package/test/cli/tui/plugin-loader.test.ts +816 -0
  187. package/test/cli/tui/plugin-toggle.test.ts +157 -0
  188. package/test/cli/tui/revert-diff.test.ts +35 -0
  189. package/test/cli/tui/slot-replace.test.tsx +47 -0
  190. package/test/cli/tui/theme-store.test.ts +54 -0
  191. package/test/cli/tui/thread.test.ts +28 -0
  192. package/test/cli/tui/transcript.test.ts +426 -0
  193. package/test/cli/tui/usage.test.ts +60 -0
  194. package/test/cli/tui/use-event.test.tsx +175 -0
  195. package/test/config/agent-color.test.ts +67 -0
  196. package/test/config/config.test.ts +2544 -0
  197. package/test/config/fixtures/empty-frontmatter.md +4 -0
  198. package/test/config/fixtures/frontmatter.md +28 -0
  199. package/test/config/fixtures/markdown-header.md +11 -0
  200. package/test/config/fixtures/no-frontmatter.md +1 -0
  201. package/test/config/fixtures/weird-model-id.md +13 -0
  202. package/test/config/lsp.test.ts +87 -0
  203. package/test/config/markdown.test.ts +228 -0
  204. package/test/config/plugin.test.ts +0 -0
  205. package/test/config/tui.test.ts +624 -0
  206. package/test/control-plane/adapters.test.ts +71 -0
  207. package/test/control-plane/workspace.test.ts +1526 -0
  208. package/test/effect/app-runtime-logger.test.ts +98 -0
  209. package/test/effect/config-service.test.ts +65 -0
  210. package/test/effect/instance-state.test.ts +394 -0
  211. package/test/effect/run-service.test.ts +89 -0
  212. package/test/effect/runner.test.ts +523 -0
  213. package/test/fake/provider.ts +82 -0
  214. package/test/file/fsmonitor.test.ts +68 -0
  215. package/test/file/ignore.test.ts +10 -0
  216. package/test/file/index.test.ts +954 -0
  217. package/test/file/path-traversal.test.ts +205 -0
  218. package/test/file/ripgrep.test.ts +226 -0
  219. package/test/file/watcher.test.ts +249 -0
  220. package/test/filesystem/filesystem.test.ts +319 -0
  221. package/test/fixture/db.ts +11 -0
  222. package/test/fixture/fixture.test.ts +26 -0
  223. package/test/fixture/fixture.ts +175 -0
  224. package/test/fixture/flock-worker.ts +72 -0
  225. package/test/fixture/log-init-worker.ts +62 -0
  226. package/test/fixture/lsp/fake-lsp-server.js +249 -0
  227. package/test/fixture/plug-worker.ts +93 -0
  228. package/test/fixture/plugin-meta-worker.ts +19 -0
  229. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  230. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  231. package/test/fixture/skills/index.json +6 -0
  232. package/test/fixture/tui-plugin.ts +323 -0
  233. package/test/fixture/tui-runtime.ts +31 -0
  234. package/test/format/format.test.ts +272 -0
  235. package/test/git/git.test.ts +128 -0
  236. package/test/ide/ide.test.ts +82 -0
  237. package/test/installation/installation.test.ts +168 -0
  238. package/test/keybind.test.ts +421 -0
  239. package/test/lib/effect.ts +53 -0
  240. package/test/lib/filesystem.ts +10 -0
  241. package/test/lib/llm-server.ts +778 -0
  242. package/test/lib/websocket.ts +46 -0
  243. package/test/lsp/client.test.ts +482 -0
  244. package/test/lsp/index.test.ts +160 -0
  245. package/test/lsp/launch.test.ts +22 -0
  246. package/test/lsp/lifecycle.test.ts +184 -0
  247. package/test/ltm/ltm.test.ts +230 -0
  248. package/test/mcp/headers.test.ts +178 -0
  249. package/test/mcp/lifecycle.test.ts +787 -0
  250. package/test/mcp/oauth-auto-connect.test.ts +311 -0
  251. package/test/mcp/oauth-browser.test.ts +276 -0
  252. package/test/mcp/oauth-callback.test.ts +34 -0
  253. package/test/memory/abort-leak-webfetch.ts +49 -0
  254. package/test/memory/abort-leak.test.ts +128 -0
  255. package/test/patch/patch.test.ts +348 -0
  256. package/test/permission/arity.test.ts +33 -0
  257. package/test/permission/next.test.ts +1227 -0
  258. package/test/permission/next.toConfig.test.ts +110 -0
  259. package/test/permission-task.test.ts +326 -0
  260. package/test/plugin/auth-override.test.ts +79 -0
  261. package/test/plugin/cloudflare.test.ts +68 -0
  262. package/test/plugin/codex.test.ts +123 -0
  263. package/test/plugin/github-copilot-models.test.ts +261 -0
  264. package/test/plugin/install-concurrency.test.ts +140 -0
  265. package/test/plugin/install.test.ts +570 -0
  266. package/test/plugin/loader-shared.test.ts +1169 -0
  267. package/test/plugin/meta.test.ts +137 -0
  268. package/test/plugin/plugin-contract.test.ts +291 -0
  269. package/test/plugin/shared.test.ts +88 -0
  270. package/test/plugin/trigger.test.ts +102 -0
  271. package/test/plugin/workspace-adapter.test.ts +109 -0
  272. package/test/preload.ts +77 -0
  273. package/test/project/instance.test.ts +276 -0
  274. package/test/project/migrate-global.test.ts +152 -0
  275. package/test/project/project.test.ts +600 -0
  276. package/test/project/vcs.test.ts +286 -0
  277. package/test/project/worktree-remove.test.ts +126 -0
  278. package/test/project/worktree.test.ts +223 -0
  279. package/test/provider/amazon-bedrock.test.ts +462 -0
  280. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  281. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  282. package/test/provider/gitlab-duo.test.ts +413 -0
  283. package/test/provider/local.test.ts +208 -0
  284. package/test/provider/models.test.ts +261 -0
  285. package/test/provider/provider-category.test.ts +190 -0
  286. package/test/provider/provider.test.ts +2758 -0
  287. package/test/provider/transform.test.ts +3681 -0
  288. package/test/pty/pty-output-isolation.test.ts +147 -0
  289. package/test/pty/pty-session.test.ts +102 -0
  290. package/test/pty/pty-shell.test.ts +104 -0
  291. package/test/question/question.test.ts +490 -0
  292. package/test/saeeol/agent-global-config-dirs.test.ts +24 -0
  293. package/test/saeeol/agent-manager-tool.test.ts +71 -0
  294. package/test/saeeol/agent-permission-overrides.test.ts +75 -0
  295. package/test/saeeol/agent-skill-permissions.test.ts +37 -0
  296. package/test/saeeol/ask-agent-permissions.test.ts +303 -0
  297. package/test/saeeol/bash-hierarchy.test.ts +64 -0
  298. package/test/saeeol/bash-permission-metadata.test.ts +66 -0
  299. package/test/saeeol/bash-security-extended.test.ts +243 -0
  300. package/test/saeeol/bedrock-claude-empty-content.test.ts +138 -0
  301. package/test/saeeol/boxes-integration.test.ts +415 -0
  302. package/test/saeeol/builtin-skills.test.ts +75 -0
  303. package/test/saeeol/cleanup.ts +28 -0
  304. package/test/saeeol/cli/dev-setup.test.ts +74 -0
  305. package/test/saeeol/cli/roll-call.test.ts +161 -0
  306. package/test/saeeol/cli-run-auto-helper.test.ts +58 -0
  307. package/test/saeeol/codex-auth-refresh.test.ts +124 -0
  308. package/test/saeeol/commit-message/generate.test.ts +188 -0
  309. package/test/saeeol/commit-message/git-context.test.ts +303 -0
  310. package/test/saeeol/commit-message-windows.test.ts +38 -0
  311. package/test/saeeol/compaction-payload-recovery.test.ts +406 -0
  312. package/test/saeeol/compaction-preservation-audit.test.ts +122 -0
  313. package/test/saeeol/compaction-skip-guard.test.ts +224 -0
  314. package/test/saeeol/compaction-smart-select.test.ts +100 -0
  315. package/test/saeeol/config/config.test.ts +166 -0
  316. package/test/saeeol/config/indexing-default-plugin.test.ts +82 -0
  317. package/test/saeeol/config/opentelemetry-default.test.ts +29 -0
  318. package/test/saeeol/config-gitignore.test.ts +70 -0
  319. package/test/saeeol/config-injector.test.ts +305 -0
  320. package/test/saeeol/config-resilience.test.ts +234 -0
  321. package/test/saeeol/config-validation.test.ts +183 -0
  322. package/test/saeeol/cost-propagation.test.ts +94 -0
  323. package/test/saeeol/cost-tracker-extended.test.ts +141 -0
  324. package/test/saeeol/cost-tracker.test.ts +64 -0
  325. package/test/saeeol/custom-provider-delete.test.ts +149 -0
  326. package/test/saeeol/diff-full.test.ts +226 -0
  327. package/test/saeeol/edit-permission-filediff.test.ts +223 -0
  328. package/test/saeeol/encoding.test.ts +364 -0
  329. package/test/saeeol/enhance-prompt.test.ts +61 -0
  330. package/test/saeeol/ensure-plan-dir.test.ts +32 -0
  331. package/test/saeeol/errors.test.ts +144 -0
  332. package/test/saeeol/external-directory-boundary.test.ts +96 -0
  333. package/test/saeeol/gateway-headers.test.ts +88 -0
  334. package/test/saeeol/help.test.ts +191 -0
  335. package/test/saeeol/ignore-migrator.test.ts +308 -0
  336. package/test/saeeol/indexing-auth.test.ts +45 -0
  337. package/test/saeeol/indexing-feature.test.ts +44 -0
  338. package/test/saeeol/indexing-label.test.ts +70 -0
  339. package/test/saeeol/indexing-startup.test.ts +381 -0
  340. package/test/saeeol/indexing-worktree.test.ts +73 -0
  341. package/test/saeeol/instruction.test.ts +136 -0
  342. package/test/saeeol/lancedb-runtime.test.ts +116 -0
  343. package/test/saeeol/loader-auth.test.ts +168 -0
  344. package/test/saeeol/local-model.test.ts +621 -0
  345. package/test/saeeol/logo.test.ts +31 -0
  346. package/test/saeeol/lsp-typescript-lightweight.test.ts +89 -0
  347. package/test/saeeol/mcp-branding.test.ts +33 -0
  348. package/test/saeeol/mcp-docker-rm.test.ts +32 -0
  349. package/test/saeeol/mcp-migrator.test.ts +736 -0
  350. package/test/saeeol/mcp-oauth-callback.test.ts +33 -0
  351. package/test/saeeol/memory-io.test.ts +198 -0
  352. package/test/saeeol/memory-paths.test.ts +87 -0
  353. package/test/saeeol/memory-security.test.ts +166 -0
  354. package/test/saeeol/model-cache-org.test.ts +164 -0
  355. package/test/saeeol/model-info-panel-utils.test.ts +52 -0
  356. package/test/saeeol/model-info-panel.types.test.ts +7 -0
  357. package/test/saeeol/models-401-fallback.test.ts +52 -0
  358. package/test/saeeol/modes-migrator.test.ts +320 -0
  359. package/test/saeeol/nvidia-headers.test.ts +74 -0
  360. package/test/saeeol/patch-jsonc.test.ts +73 -0
  361. package/test/saeeol/patch.test.ts +172 -0
  362. package/test/saeeol/paths.test.ts +265 -0
  363. package/test/saeeol/permission/config-paths.test.ts +174 -0
  364. package/test/saeeol/permission/env-read.test.ts +149 -0
  365. package/test/saeeol/permission/external-directory-allow.test.ts +327 -0
  366. package/test/saeeol/permission/next.always-rules.test.ts +882 -0
  367. package/test/saeeol/permission/next.reply-http.test.ts +205 -0
  368. package/test/saeeol/permission/next.reply-routing.test.ts +184 -0
  369. package/test/saeeol/plan-exit-detection.test.ts +494 -0
  370. package/test/saeeol/plan-followup.test.ts +1376 -0
  371. package/test/saeeol/project-config-update.test.ts +120 -0
  372. package/test/saeeol/project-id.test.ts +455 -0
  373. package/test/saeeol/provider-cost.test.ts +171 -0
  374. package/test/saeeol/provider-list-failed-state.test.ts +100 -0
  375. package/test/saeeol/question-dismiss-all.test.ts +174 -0
  376. package/test/saeeol/read-directory.test.ts +116 -0
  377. package/test/saeeol/rules-migrator.test.ts +257 -0
  378. package/test/saeeol/run-auto.test.ts +176 -0
  379. package/test/saeeol/run-network.test.ts +224 -0
  380. package/test/saeeol/semantic-search.test.ts +186 -0
  381. package/test/saeeol/server/permission-allow-everything.test.ts +125 -0
  382. package/test/saeeol/session/instruction-substitution.test.ts +72 -0
  383. package/test/saeeol/session/platform-attribution.test.ts +118 -0
  384. package/test/saeeol/session/session.test.ts +105 -0
  385. package/test/saeeol/session-compaction-cap.test.ts +399 -0
  386. package/test/saeeol/session-compaction-chunks.test.ts +501 -0
  387. package/test/saeeol/session-compaction-safety.test.ts +481 -0
  388. package/test/saeeol/session-fork-remap.test.ts +251 -0
  389. package/test/saeeol/session-import-service.test.ts +114 -0
  390. package/test/saeeol/session-list.test.ts +47 -0
  391. package/test/saeeol/session-message-metadata.test.ts +128 -0
  392. package/test/saeeol/session-overflow.test.ts +78 -0
  393. package/test/saeeol/session-processor-empty-tool-calls.test.ts +571 -0
  394. package/test/saeeol/session-processor-network-offline.test.ts +204 -0
  395. package/test/saeeol/session-processor-retry-limit.test.ts +238 -0
  396. package/test/saeeol/session-processor-review-telemetry.test.ts +82 -0
  397. package/test/saeeol/session-prompt-compaction-safety.test.ts +517 -0
  398. package/test/saeeol/session-prompt-queue.test.ts +815 -0
  399. package/test/saeeol/sessions/inflight-cache.test.ts +157 -0
  400. package/test/saeeol/sessions/ingest-queue.test.ts +402 -0
  401. package/test/saeeol/sessions/remote-protocol.test.ts +258 -0
  402. package/test/saeeol/sessions/remote-sender.test.ts +1036 -0
  403. package/test/saeeol/sessions/remote-ws.test.ts +367 -0
  404. package/test/saeeol/sessions/sessions-enable-remote.test.disable +181 -0
  405. package/test/saeeol/slot-prop-reactivity.test.ts +142 -0
  406. package/test/saeeol/snapshot-cache.test.ts +84 -0
  407. package/test/saeeol/snapshot-freeze-repro.test.ts +100 -0
  408. package/test/saeeol/snapshot-track-timeout.test.ts +519 -0
  409. package/test/saeeol/stats-subagent-cost.test.ts +123 -0
  410. package/test/saeeol/suggestion/auto-dismiss.test.ts +65 -0
  411. package/test/saeeol/suggestion/suggestion.test.ts +145 -0
  412. package/test/saeeol/suggestion/tool.test.ts +298 -0
  413. package/test/saeeol/summary-file-diff.test.ts +28 -0
  414. package/test/saeeol/system-prompt.test.ts +142 -0
  415. package/test/saeeol/task-nesting.test.ts +193 -0
  416. package/test/saeeol/telemetry/feedback.test.ts +8 -0
  417. package/test/saeeol/todo-view.test.ts +57 -0
  418. package/test/saeeol/tool-encoding.test.ts +455 -0
  419. package/test/saeeol/tool-registry-indexing-import-failure.test.ts +49 -0
  420. package/test/saeeol/tool-registry-indexing.test.ts +236 -0
  421. package/test/saeeol/tool-registry-semantic-import-failure.test.ts +55 -0
  422. package/test/saeeol/tool-task-model.test.ts +352 -0
  423. package/test/saeeol/transform-opus-4.7.test.ts +89 -0
  424. package/test/saeeol/tui-diff.test.ts +91 -0
  425. package/test/saeeol/tui-sync.test.ts +80 -0
  426. package/test/saeeol/util/url.test.ts +141 -0
  427. package/test/saeeol/workflows-migrator.test.ts +261 -0
  428. package/test/saeeol/worktree-diff-summary.test.ts +64 -0
  429. package/test/saeeol/worktree-diff.test.ts +223 -0
  430. package/test/saeeol/worktree-remove-lock.test.ts +82 -0
  431. package/test/server/AGENTS.md +15 -0
  432. package/test/server/contract.test.ts +357 -0
  433. package/test/server/experimental-session-list.test.ts +157 -0
  434. package/test/server/global-session-list.test.ts +155 -0
  435. package/test/server/httpapi-authorization.test.ts +103 -0
  436. package/test/server/httpapi-bridge.test.ts +440 -0
  437. package/test/server/httpapi-config.test.ts +67 -0
  438. package/test/server/httpapi-cors.test.ts +89 -0
  439. package/test/server/httpapi-event.test.ts +57 -0
  440. package/test/server/httpapi-experimental.test.ts +219 -0
  441. package/test/server/httpapi-file.test.ts +79 -0
  442. package/test/server/httpapi-instance-context.test.ts +237 -0
  443. package/test/server/httpapi-instance.legacy.test.ts +140 -0
  444. package/test/server/httpapi-instance.test.ts +83 -0
  445. package/test/server/httpapi-json-parity.test.ts +263 -0
  446. package/test/server/httpapi-mcp-oauth.test.ts +76 -0
  447. package/test/server/httpapi-mcp.test.ts +189 -0
  448. package/test/server/httpapi-provider.test.ts +153 -0
  449. package/test/server/httpapi-pty-websocket.test.ts +16 -0
  450. package/test/server/httpapi-pty.test.ts +175 -0
  451. package/test/server/httpapi-raw-route-auth.test.ts +89 -0
  452. package/test/server/httpapi-sdk.test.ts +681 -0
  453. package/test/server/httpapi-session.test.ts +464 -0
  454. package/test/server/httpapi-sync.test.ts +130 -0
  455. package/test/server/httpapi-tui.test.ts +121 -0
  456. package/test/server/httpapi-workspace-routing.test.ts +471 -0
  457. package/test/server/httpapi-workspace.test.ts +427 -0
  458. package/test/server/lib/conformance.ts +88 -0
  459. package/test/server/lib/stateful.ts +112 -0
  460. package/test/server/project-init-git.test.ts +113 -0
  461. package/test/server/proxy-util.test.ts +113 -0
  462. package/test/server/session-actions.test.ts +49 -0
  463. package/test/server/session-list.test.ts +238 -0
  464. package/test/server/session-messages.test.ts +167 -0
  465. package/test/server/session-select.test.ts +100 -0
  466. package/test/server/trace-attributes.test.ts +76 -0
  467. package/test/server/workspace-proxy.test.ts +165 -0
  468. package/test/server/workspace-routing.test.ts +85 -0
  469. package/test/session/compaction.test.ts +2420 -0
  470. package/test/session/instruction.test.ts +247 -0
  471. package/test/session/llm.test.ts +1273 -0
  472. package/test/session/message-v2.test.ts +1291 -0
  473. package/test/session/messages-pagination.test.ts +1173 -0
  474. package/test/session/network.test.ts +249 -0
  475. package/test/session/processor-effect.test.ts +847 -0
  476. package/test/session/prompt.test.ts +2131 -0
  477. package/test/session/retry.test.ts +340 -0
  478. package/test/session/revert-compact.test.ts +639 -0
  479. package/test/session/schema-decoding.test.ts +311 -0
  480. package/test/session/session-entry-stepper.test.ts +917 -0
  481. package/test/session/session-schema.test.ts +76 -0
  482. package/test/session/snapshot-tool-race.test.ts +257 -0
  483. package/test/session/structured-output-integration.test.ts +265 -0
  484. package/test/session/structured-output.test.ts +381 -0
  485. package/test/session/system.test.ts +73 -0
  486. package/test/share/share-next.test.ts +333 -0
  487. package/test/shell/shell.test.ts +99 -0
  488. package/test/skill/discovery.test.ts +116 -0
  489. package/test/skill/skill.test.ts +393 -0
  490. package/test/snapshot/snapshot.test.ts +1531 -0
  491. package/test/storage/db.test.ts +23 -0
  492. package/test/storage/json-migration.test.ts +832 -0
  493. package/test/storage/storage.test.ts +293 -0
  494. package/test/suggestion/suggestion.test.ts +1 -0
  495. package/test/sync/index.test.ts +256 -0
  496. package/test/tool/__snapshots__/parameters.test.ts.snap +500 -0
  497. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  498. package/test/tool/apply_patch.test.ts +614 -0
  499. package/test/tool/bash.test.ts +1225 -0
  500. package/test/tool/diagnostics-filter.test.ts +55 -0
  501. package/test/tool/edit.test.ts +754 -0
  502. package/test/tool/external-directory.test.ts +169 -0
  503. package/test/tool/fixtures/large-image.png +0 -0
  504. package/test/tool/fixtures/models-api.json +65179 -0
  505. package/test/tool/glob.test.ts +107 -0
  506. package/test/tool/grep.test.ts +114 -0
  507. package/test/tool/lsp.test.ts +187 -0
  508. package/test/tool/parameters.test.ts +243 -0
  509. package/test/tool/question.test.ts +129 -0
  510. package/test/tool/read.test.ts +500 -0
  511. package/test/tool/recall.test.ts +151 -0
  512. package/test/tool/registry.test.ts +203 -0
  513. package/test/tool/skill.test.ts +135 -0
  514. package/test/tool/suggest.test.ts +1 -0
  515. package/test/tool/task.test.ts +612 -0
  516. package/test/tool/tool-define.test.ts +99 -0
  517. package/test/tool/truncation.test.ts +260 -0
  518. package/test/tool/webfetch.test.ts +103 -0
  519. package/test/tool/write.test.ts +291 -0
  520. package/test/util/data-url.test.ts +14 -0
  521. package/test/util/effect-zod.test.ts +754 -0
  522. package/test/util/error.test.ts +38 -0
  523. package/test/util/filesystem.test.ts +656 -0
  524. package/test/util/format.test.ts +59 -0
  525. package/test/util/glob.test.ts +164 -0
  526. package/test/util/iife.test.ts +36 -0
  527. package/test/util/lazy.test.ts +50 -0
  528. package/test/util/lock.test.ts +72 -0
  529. package/test/util/log.test.ts +86 -0
  530. package/test/util/module.test.ts +59 -0
  531. package/test/util/process.test.ts +128 -0
  532. package/test/util/timeout.test.ts +21 -0
  533. package/test/util/which.test.ts +100 -0
  534. package/test/util/wildcard.test.ts +90 -0
  535. package/test/workspace/workspace-restore.test.ts +296 -0
  536. package/src/provider/models-snapshot.d.ts +0 -2
  537. package/src/provider/models-snapshot.js +0 -3
@@ -0,0 +1,2544 @@
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
+ })