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