saeeol 1.2.4 → 1.2.7

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