saeeol 1.2.5 → 1.2.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +15 -12
- package/src/ltm/config.ts +15 -15
- package/src/ltm/events.ts +1 -1
- package/src/ltm/index.ts +1 -1
- package/src/ltm/pipeline.ts +22 -22
- package/src/ltm/scheduler.ts +20 -20
- package/src/ltm/store.ts +9 -7
- package/src/ltm/types.ts +16 -16
- package/src/provider/local/embedder.ts +21 -21
- package/src/provider/models-snapshot.d.ts +2 -0
- package/src/provider/models-snapshot.js +3 -0
- package/src/session/message/message-errors.ts +1 -1
- package/src/session/message/message-parts.ts +1 -1
- package/src/session/message/message-transform.ts +1 -1
- package/src/session/message/message-types.ts +1 -1
- package/src/tool/core/tool.ts +1 -1
- package/AGENTS.md +0 -72
- package/BUN_SHELL_MIGRATION_PLAN.md +0 -136
- package/Dockerfile +0 -18
- package/assets/saeeol.ico +0 -0
- package/bin/saeeol +0 -187
- package/bunfig.toml +0 -7
- package/database.db +0 -0
- package/drizzle.config.ts +0 -10
- package/git +0 -0
- package/migration/20260127222353_familiar_lady_ursula/migration.sql +0 -90
- package/migration/20260127222353_familiar_lady_ursula/snapshot.json +0 -796
- package/migration/20260211171708_add_project_commands/migration.sql +0 -1
- package/migration/20260211171708_add_project_commands/snapshot.json +0 -806
- package/migration/20260213144116_wakeful_the_professor/migration.sql +0 -11
- package/migration/20260213144116_wakeful_the_professor/snapshot.json +0 -897
- package/migration/20260225215848_workspace/migration.sql +0 -7
- package/migration/20260225215848_workspace/snapshot.json +0 -959
- package/migration/20260227213759_add_session_workspace_id/migration.sql +0 -2
- package/migration/20260227213759_add_session_workspace_id/snapshot.json +0 -983
- package/migration/20260228203230_blue_harpoon/migration.sql +0 -17
- package/migration/20260228203230_blue_harpoon/snapshot.json +0 -1102
- package/migration/20260303231226_add_workspace_fields/migration.sql +0 -5
- package/migration/20260303231226_add_workspace_fields/snapshot.json +0 -1013
- package/migration/20260309230000_move_org_to_state/migration.sql +0 -3
- package/migration/20260309230000_move_org_to_state/snapshot.json +0 -1156
- package/migration/20260312043431_session_message_cursor/migration.sql +0 -4
- package/migration/20260312043431_session_message_cursor/snapshot.json +0 -1168
- package/migration/20260323234822_events/migration.sql +0 -13
- package/migration/20260323234822_events/snapshot.json +0 -1271
- package/migration/20260410174513_workspace-name/migration.sql +0 -16
- package/migration/20260410174513_workspace-name/snapshot.json +0 -1271
- package/migration/20260413175956_chief_energizer/migration.sql +0 -13
- package/migration/20260413175956_chief_energizer/snapshot.json +0 -1399
- package/migration/20260423070820_add_icon_url_override/migration.sql +0 -2
- package/migration/20260423070820_add_icon_url_override/snapshot.json +0 -1409
- package/migration/20260428004200_add_session_path/migration.sql +0 -1
- package/migration/20260428004200_add_session_path/snapshot.json +0 -1419
- package/npm/bin/saeeol +0 -42
- package/npm/package.json +0 -39
- package/npm/postinstall.js +0 -162
- package/parsers-config.ts +0 -289
- package/script/build.ts +0 -393
- package/script/check-migrations.ts +0 -16
- package/script/fix-node-pty.ts +0 -34
- package/script/generate.ts +0 -23
- package/script/postinstall.mjs +0 -189
- package/script/publish.ts +0 -200
- package/script/run-workspace-server +0 -106
- package/script/schema.ts +0 -63
- package/script/test-runner.ts +0 -420
- package/script/time.ts +0 -6
- package/script/trace-imports.ts +0 -153
- package/script/upgrade-opentui.ts +0 -64
- package/scripts/diff-sdk-types.sh +0 -52
- package/specs/effect/facades.md +0 -221
- package/specs/effect/http-api.md +0 -401
- package/specs/effect/instance-context.md +0 -309
- package/specs/effect/loose-ends.md +0 -34
- package/specs/effect/migration.md +0 -299
- package/specs/effect/routes.md +0 -64
- package/specs/effect/schema.md +0 -399
- package/specs/effect/server-package.md +0 -668
- package/specs/effect/tools.md +0 -90
- package/specs/tui-plugins.md +0 -433
- package/specs/v2/api.ts +0 -67
- package/specs/v2/keymappings.md +0 -10
- package/specs/v2/message-shape.md +0 -136
- package/src/tool/apply_patch.txt +0 -33
- package/src/tool/bash.txt +0 -119
- package/src/tool/edit.txt +0 -10
- package/src/tool/glob.txt +0 -6
- package/src/tool/grep.txt +0 -8
- package/src/tool/lsp.txt +0 -24
- package/src/tool/plan-enter.txt +0 -14
- package/src/tool/plan-exit.txt +0 -13
- package/src/tool/question.txt +0 -11
- package/src/tool/read.txt +0 -14
- package/src/tool/recall.txt +0 -12
- package/src/tool/skill.txt +0 -5
- package/src/tool/task.txt +0 -57
- package/src/tool/todowrite.txt +0 -167
- package/src/tool/warpgrep.txt +0 -10
- package/src/tool/webfetch.txt +0 -13
- package/src/tool/websearch.txt +0 -14
- package/src/tool/write.txt +0 -8
- package/sst-env.d.ts +0 -10
- package/test/AGENTS.md +0 -133
- package/test/account/repo.test.ts +0 -352
- package/test/account/service.test.ts +0 -456
- package/test/acp/agent-interface.test.ts +0 -51
- package/test/acp/event-subscription.test.ts +0 -725
- package/test/agent/agent.test.ts +0 -890
- package/test/auth/auth.test.ts +0 -86
- package/test/bun/registry.test.ts +0 -75
- package/test/bus/bus-effect.test.ts +0 -161
- package/test/bus/bus-integration.test.ts +0 -87
- package/test/bus/bus.test.ts +0 -219
- package/test/cli/account.test.ts +0 -26
- package/test/cli/auto-mode.test.ts +0 -75
- package/test/cli/bin-saeeol.test.ts +0 -8
- package/test/cli/cmd/tui/prompt-part.test.ts +0 -47
- package/test/cli/cmd/tui/prompt-traits.test.ts +0 -38
- package/test/cli/cmd/tui/sync.test.tsx +0 -159
- package/test/cli/error.test.ts +0 -18
- package/test/cli/github-action.test.ts +0 -198
- package/test/cli/github-remote.test.ts +0 -85
- package/test/cli/import.test.ts +0 -97
- package/test/cli/install-artifact.test.ts +0 -72
- package/test/cli/plugin-auth-picker.test.ts +0 -120
- package/test/cli/pr.test.ts +0 -59
- package/test/cli/tui/editor-context-zed.test.ts +0 -356
- package/test/cli/tui/editor-context.test.tsx +0 -228
- package/test/cli/tui/keybind-plugin.test.ts +0 -90
- package/test/cli/tui/markdown.test.ts +0 -161
- package/test/cli/tui/plugin-add.test.ts +0 -111
- package/test/cli/tui/plugin-install.test.ts +0 -87
- package/test/cli/tui/plugin-lifecycle.test.ts +0 -224
- package/test/cli/tui/plugin-loader-entrypoint.test.ts +0 -484
- package/test/cli/tui/plugin-loader-pure.test.ts +0 -71
- package/test/cli/tui/plugin-loader.test.ts +0 -816
- package/test/cli/tui/plugin-toggle.test.ts +0 -157
- package/test/cli/tui/revert-diff.test.ts +0 -35
- package/test/cli/tui/slot-replace.test.tsx +0 -47
- package/test/cli/tui/theme-store.test.ts +0 -54
- package/test/cli/tui/thread.test.ts +0 -28
- package/test/cli/tui/transcript.test.ts +0 -426
- package/test/cli/tui/usage.test.ts +0 -60
- package/test/cli/tui/use-event.test.tsx +0 -175
- package/test/config/agent-color.test.ts +0 -67
- package/test/config/config.test.ts +0 -2544
- package/test/config/fixtures/empty-frontmatter.md +0 -4
- package/test/config/fixtures/frontmatter.md +0 -28
- package/test/config/fixtures/markdown-header.md +0 -11
- package/test/config/fixtures/no-frontmatter.md +0 -1
- package/test/config/fixtures/weird-model-id.md +0 -13
- package/test/config/lsp.test.ts +0 -87
- package/test/config/markdown.test.ts +0 -228
- package/test/config/plugin.test.ts +0 -0
- package/test/config/tui.test.ts +0 -624
- package/test/control-plane/adapters.test.ts +0 -71
- package/test/control-plane/workspace.test.ts +0 -1526
- package/test/effect/app-runtime-logger.test.ts +0 -98
- package/test/effect/config-service.test.ts +0 -65
- package/test/effect/instance-state.test.ts +0 -394
- package/test/effect/run-service.test.ts +0 -89
- package/test/effect/runner.test.ts +0 -523
- package/test/fake/provider.ts +0 -82
- package/test/file/fsmonitor.test.ts +0 -68
- package/test/file/ignore.test.ts +0 -10
- package/test/file/index.test.ts +0 -954
- package/test/file/path-traversal.test.ts +0 -205
- package/test/file/ripgrep.test.ts +0 -226
- package/test/file/watcher.test.ts +0 -249
- package/test/filesystem/filesystem.test.ts +0 -319
- package/test/fixture/db.ts +0 -11
- package/test/fixture/fixture.test.ts +0 -26
- package/test/fixture/fixture.ts +0 -175
- package/test/fixture/flock-worker.ts +0 -72
- package/test/fixture/log-init-worker.ts +0 -62
- package/test/fixture/lsp/fake-lsp-server.js +0 -249
- package/test/fixture/plug-worker.ts +0 -93
- package/test/fixture/plugin-meta-worker.ts +0 -19
- package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
- package/test/fixture/skills/cloudflare/SKILL.md +0 -211
- package/test/fixture/skills/index.json +0 -6
- package/test/fixture/tui-plugin.ts +0 -323
- package/test/fixture/tui-runtime.ts +0 -31
- package/test/format/format.test.ts +0 -272
- package/test/git/git.test.ts +0 -128
- package/test/ide/ide.test.ts +0 -82
- package/test/installation/installation.test.ts +0 -168
- package/test/keybind.test.ts +0 -421
- package/test/lib/effect.ts +0 -53
- package/test/lib/filesystem.ts +0 -10
- package/test/lib/llm-server.ts +0 -778
- package/test/lib/websocket.ts +0 -46
- package/test/lsp/client.test.ts +0 -482
- package/test/lsp/index.test.ts +0 -160
- package/test/lsp/launch.test.ts +0 -22
- package/test/lsp/lifecycle.test.ts +0 -184
- package/test/ltm/ltm.test.ts +0 -230
- package/test/mcp/headers.test.ts +0 -178
- package/test/mcp/lifecycle.test.ts +0 -787
- package/test/mcp/oauth-auto-connect.test.ts +0 -311
- package/test/mcp/oauth-browser.test.ts +0 -276
- package/test/mcp/oauth-callback.test.ts +0 -34
- package/test/memory/abort-leak-webfetch.ts +0 -49
- package/test/memory/abort-leak.test.ts +0 -128
- package/test/patch/patch.test.ts +0 -348
- package/test/permission/arity.test.ts +0 -33
- package/test/permission/next.test.ts +0 -1227
- package/test/permission/next.toConfig.test.ts +0 -110
- package/test/permission-task.test.ts +0 -326
- package/test/plugin/auth-override.test.ts +0 -79
- package/test/plugin/cloudflare.test.ts +0 -68
- package/test/plugin/codex.test.ts +0 -123
- package/test/plugin/github-copilot-models.test.ts +0 -261
- package/test/plugin/install-concurrency.test.ts +0 -140
- package/test/plugin/install.test.ts +0 -570
- package/test/plugin/loader-shared.test.ts +0 -1169
- package/test/plugin/meta.test.ts +0 -137
- package/test/plugin/shared.test.ts +0 -88
- package/test/plugin/trigger.test.ts +0 -102
- package/test/plugin/workspace-adapter.test.ts +0 -109
- package/test/preload.ts +0 -77
- package/test/project/instance.test.ts +0 -276
- package/test/project/migrate-global.test.ts +0 -152
- package/test/project/project.test.ts +0 -600
- package/test/project/vcs.test.ts +0 -286
- package/test/project/worktree-remove.test.ts +0 -126
- package/test/project/worktree.test.ts +0 -223
- package/test/provider/amazon-bedrock.test.ts +0 -462
- package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
- package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
- package/test/provider/gitlab-duo.test.ts +0 -413
- package/test/provider/local.test.ts +0 -208
- package/test/provider/models.test.ts +0 -261
- package/test/provider/provider-category.test.ts +0 -190
- package/test/provider/provider.test.ts +0 -2758
- package/test/provider/transform.test.ts +0 -3681
- package/test/pty/pty-output-isolation.test.ts +0 -147
- package/test/pty/pty-session.test.ts +0 -102
- package/test/pty/pty-shell.test.ts +0 -104
- package/test/question/question.test.ts +0 -490
- package/test/saeeol/agent-global-config-dirs.test.ts +0 -24
- package/test/saeeol/agent-manager-tool.test.ts +0 -71
- package/test/saeeol/agent-permission-overrides.test.ts +0 -75
- package/test/saeeol/agent-skill-permissions.test.ts +0 -37
- package/test/saeeol/ask-agent-permissions.test.ts +0 -303
- package/test/saeeol/bash-hierarchy.test.ts +0 -64
- package/test/saeeol/bash-permission-metadata.test.ts +0 -66
- package/test/saeeol/bash-security-extended.test.ts +0 -243
- package/test/saeeol/bedrock-claude-empty-content.test.ts +0 -138
- package/test/saeeol/boxes-integration.test.ts +0 -415
- package/test/saeeol/builtin-skills.test.ts +0 -75
- package/test/saeeol/cleanup.ts +0 -28
- package/test/saeeol/cli/dev-setup.test.ts +0 -74
- package/test/saeeol/cli/roll-call.test.ts +0 -161
- package/test/saeeol/cli-run-auto-helper.test.ts +0 -58
- package/test/saeeol/codex-auth-refresh.test.ts +0 -124
- package/test/saeeol/commit-message/generate.test.ts +0 -188
- package/test/saeeol/commit-message/git-context.test.ts +0 -303
- package/test/saeeol/commit-message-windows.test.ts +0 -38
- package/test/saeeol/compaction-payload-recovery.test.ts +0 -406
- package/test/saeeol/compaction-preservation-audit.test.ts +0 -122
- package/test/saeeol/compaction-skip-guard.test.ts +0 -224
- package/test/saeeol/compaction-smart-select.test.ts +0 -100
- package/test/saeeol/config/config.test.ts +0 -166
- package/test/saeeol/config/indexing-default-plugin.test.ts +0 -82
- package/test/saeeol/config/opentelemetry-default.test.ts +0 -29
- package/test/saeeol/config-gitignore.test.ts +0 -70
- package/test/saeeol/config-injector.test.ts +0 -305
- package/test/saeeol/config-resilience.test.ts +0 -234
- package/test/saeeol/config-validation.test.ts +0 -183
- package/test/saeeol/cost-propagation.test.ts +0 -94
- package/test/saeeol/cost-tracker-extended.test.ts +0 -141
- package/test/saeeol/cost-tracker.test.ts +0 -64
- package/test/saeeol/custom-provider-delete.test.ts +0 -149
- package/test/saeeol/diff-full.test.ts +0 -226
- package/test/saeeol/edit-permission-filediff.test.ts +0 -223
- package/test/saeeol/encoding.test.ts +0 -364
- package/test/saeeol/enhance-prompt.test.ts +0 -61
- package/test/saeeol/ensure-plan-dir.test.ts +0 -32
- package/test/saeeol/errors.test.ts +0 -144
- package/test/saeeol/external-directory-boundary.test.ts +0 -96
- package/test/saeeol/gateway-headers.test.ts +0 -88
- package/test/saeeol/help.test.ts +0 -191
- package/test/saeeol/ignore-migrator.test.ts +0 -308
- package/test/saeeol/indexing-auth.test.ts +0 -45
- package/test/saeeol/indexing-feature.test.ts +0 -44
- package/test/saeeol/indexing-label.test.ts +0 -70
- package/test/saeeol/indexing-startup.test.ts +0 -381
- package/test/saeeol/indexing-worktree.test.ts +0 -73
- package/test/saeeol/instruction.test.ts +0 -136
- package/test/saeeol/lancedb-runtime.test.ts +0 -116
- package/test/saeeol/loader-auth.test.ts +0 -168
- package/test/saeeol/local-model.test.ts +0 -621
- package/test/saeeol/logo.test.ts +0 -31
- package/test/saeeol/lsp-typescript-lightweight.test.ts +0 -89
- package/test/saeeol/mcp-branding.test.ts +0 -33
- package/test/saeeol/mcp-docker-rm.test.ts +0 -32
- package/test/saeeol/mcp-migrator.test.ts +0 -736
- package/test/saeeol/mcp-oauth-callback.test.ts +0 -33
- package/test/saeeol/memory-io.test.ts +0 -198
- package/test/saeeol/memory-paths.test.ts +0 -87
- package/test/saeeol/memory-security.test.ts +0 -166
- package/test/saeeol/model-cache-org.test.ts +0 -164
- package/test/saeeol/model-info-panel-utils.test.ts +0 -52
- package/test/saeeol/model-info-panel.types.test.ts +0 -7
- package/test/saeeol/models-401-fallback.test.ts +0 -52
- package/test/saeeol/modes-migrator.test.ts +0 -320
- package/test/saeeol/nvidia-headers.test.ts +0 -74
- package/test/saeeol/patch-jsonc.test.ts +0 -73
- package/test/saeeol/patch.test.ts +0 -172
- package/test/saeeol/paths.test.ts +0 -265
- package/test/saeeol/permission/config-paths.test.ts +0 -174
- package/test/saeeol/permission/env-read.test.ts +0 -149
- package/test/saeeol/permission/external-directory-allow.test.ts +0 -327
- package/test/saeeol/permission/next.always-rules.test.ts +0 -882
- package/test/saeeol/permission/next.reply-http.test.ts +0 -205
- package/test/saeeol/permission/next.reply-routing.test.ts +0 -184
- package/test/saeeol/plan-exit-detection.test.ts +0 -494
- package/test/saeeol/plan-followup.test.ts +0 -1376
- package/test/saeeol/project-config-update.test.ts +0 -120
- package/test/saeeol/project-id.test.ts +0 -455
- package/test/saeeol/provider-cost.test.ts +0 -171
- package/test/saeeol/provider-list-failed-state.test.ts +0 -100
- package/test/saeeol/question-dismiss-all.test.ts +0 -174
- package/test/saeeol/read-directory.test.ts +0 -116
- package/test/saeeol/rules-migrator.test.ts +0 -257
- package/test/saeeol/run-auto.test.ts +0 -176
- package/test/saeeol/run-network.test.ts +0 -224
- package/test/saeeol/semantic-search.test.ts +0 -186
- package/test/saeeol/server/permission-allow-everything.test.ts +0 -125
- package/test/saeeol/session/instruction-substitution.test.ts +0 -72
- package/test/saeeol/session/platform-attribution.test.ts +0 -118
- package/test/saeeol/session/session.test.ts +0 -105
- package/test/saeeol/session-compaction-cap.test.ts +0 -399
- package/test/saeeol/session-compaction-chunks.test.ts +0 -501
- package/test/saeeol/session-compaction-safety.test.ts +0 -481
- package/test/saeeol/session-fork-remap.test.ts +0 -251
- package/test/saeeol/session-import-service.test.ts +0 -114
- package/test/saeeol/session-list.test.ts +0 -47
- package/test/saeeol/session-message-metadata.test.ts +0 -128
- package/test/saeeol/session-overflow.test.ts +0 -78
- package/test/saeeol/session-processor-empty-tool-calls.test.ts +0 -571
- package/test/saeeol/session-processor-network-offline.test.ts +0 -204
- package/test/saeeol/session-processor-retry-limit.test.ts +0 -238
- package/test/saeeol/session-processor-review-telemetry.test.ts +0 -82
- package/test/saeeol/session-prompt-compaction-safety.test.ts +0 -517
- package/test/saeeol/session-prompt-queue.test.ts +0 -815
- package/test/saeeol/sessions/inflight-cache.test.ts +0 -157
- package/test/saeeol/sessions/ingest-queue.test.ts +0 -402
- package/test/saeeol/sessions/remote-protocol.test.ts +0 -258
- package/test/saeeol/sessions/remote-sender.test.ts +0 -1036
- package/test/saeeol/sessions/remote-ws.test.ts +0 -367
- package/test/saeeol/sessions/sessions-enable-remote.test.disable +0 -181
- package/test/saeeol/slot-prop-reactivity.test.ts +0 -142
- package/test/saeeol/snapshot-cache.test.ts +0 -84
- package/test/saeeol/snapshot-freeze-repro.test.ts +0 -100
- package/test/saeeol/snapshot-track-timeout.test.ts +0 -519
- package/test/saeeol/stats-subagent-cost.test.ts +0 -123
- package/test/saeeol/suggestion/auto-dismiss.test.ts +0 -65
- package/test/saeeol/suggestion/suggestion.test.ts +0 -145
- package/test/saeeol/suggestion/tool.test.ts +0 -298
- package/test/saeeol/summary-file-diff.test.ts +0 -28
- package/test/saeeol/system-prompt.test.ts +0 -142
- package/test/saeeol/task-nesting.test.ts +0 -193
- package/test/saeeol/telemetry/feedback.test.ts +0 -8
- package/test/saeeol/todo-view.test.ts +0 -57
- package/test/saeeol/tool-encoding.test.ts +0 -455
- package/test/saeeol/tool-registry-indexing-import-failure.test.ts +0 -49
- package/test/saeeol/tool-registry-indexing.test.ts +0 -236
- package/test/saeeol/tool-registry-semantic-import-failure.test.ts +0 -55
- package/test/saeeol/tool-task-model.test.ts +0 -352
- package/test/saeeol/transform-opus-4.7.test.ts +0 -89
- package/test/saeeol/tui-diff.test.ts +0 -91
- package/test/saeeol/tui-sync.test.ts +0 -80
- package/test/saeeol/util/url.test.ts +0 -141
- package/test/saeeol/workflows-migrator.test.ts +0 -261
- package/test/saeeol/worktree-diff-summary.test.ts +0 -64
- package/test/saeeol/worktree-diff.test.ts +0 -223
- package/test/saeeol/worktree-remove-lock.test.ts +0 -82
- package/test/server/AGENTS.md +0 -15
- package/test/server/contract.test.ts +0 -231
- package/test/server/experimental-session-list.test.ts +0 -157
- package/test/server/global-session-list.test.ts +0 -155
- package/test/server/httpapi-authorization.test.ts +0 -103
- package/test/server/httpapi-bridge.test.ts +0 -440
- package/test/server/httpapi-config.test.ts +0 -67
- package/test/server/httpapi-cors.test.ts +0 -89
- package/test/server/httpapi-event.test.ts +0 -57
- package/test/server/httpapi-experimental.test.ts +0 -219
- package/test/server/httpapi-file.test.ts +0 -79
- package/test/server/httpapi-instance-context.test.ts +0 -237
- package/test/server/httpapi-instance.legacy.test.ts +0 -140
- package/test/server/httpapi-instance.test.ts +0 -83
- package/test/server/httpapi-json-parity.test.ts +0 -263
- package/test/server/httpapi-mcp-oauth.test.ts +0 -76
- package/test/server/httpapi-mcp.test.ts +0 -189
- package/test/server/httpapi-provider.test.ts +0 -153
- package/test/server/httpapi-pty-websocket.test.ts +0 -16
- package/test/server/httpapi-pty.test.ts +0 -175
- package/test/server/httpapi-raw-route-auth.test.ts +0 -89
- package/test/server/httpapi-sdk.test.ts +0 -679
- package/test/server/httpapi-session.test.ts +0 -464
- package/test/server/httpapi-sync.test.ts +0 -130
- package/test/server/httpapi-tui.test.ts +0 -121
- package/test/server/httpapi-workspace-routing.test.ts +0 -471
- package/test/server/httpapi-workspace.test.ts +0 -427
- package/test/server/project-init-git.test.ts +0 -113
- package/test/server/proxy-util.test.ts +0 -113
- package/test/server/session-actions.test.ts +0 -49
- package/test/server/session-list.test.ts +0 -238
- package/test/server/session-messages.test.ts +0 -167
- package/test/server/session-select.test.ts +0 -100
- package/test/server/trace-attributes.test.ts +0 -76
- package/test/server/workspace-proxy.test.ts +0 -165
- package/test/server/workspace-routing.test.ts +0 -85
- package/test/session/compaction.test.ts +0 -2420
- package/test/session/instruction.test.ts +0 -247
- package/test/session/llm.test.ts +0 -1273
- package/test/session/message-v2.test.ts +0 -1291
- package/test/session/messages-pagination.test.ts +0 -1173
- package/test/session/network.test.ts +0 -249
- package/test/session/processor-effect.test.ts +0 -847
- package/test/session/prompt.test.ts +0 -2131
- package/test/session/retry.test.ts +0 -340
- package/test/session/revert-compact.test.ts +0 -639
- package/test/session/schema-decoding.test.ts +0 -311
- package/test/session/session-entry-stepper.test.ts +0 -917
- package/test/session/session-schema.test.ts +0 -76
- package/test/session/snapshot-tool-race.test.ts +0 -257
- package/test/session/structured-output-integration.test.ts +0 -265
- package/test/session/structured-output.test.ts +0 -381
- package/test/session/system.test.ts +0 -73
- package/test/share/share-next.test.ts +0 -333
- package/test/shell/shell.test.ts +0 -99
- package/test/skill/discovery.test.ts +0 -116
- package/test/skill/skill.test.ts +0 -393
- package/test/snapshot/snapshot.test.ts +0 -1531
- package/test/storage/db.test.ts +0 -23
- package/test/storage/json-migration.test.ts +0 -832
- package/test/storage/storage.test.ts +0 -293
- package/test/suggestion/suggestion.test.ts +0 -1
- package/test/sync/index.test.ts +0 -256
- package/test/tool/__snapshots__/parameters.test.ts.snap +0 -500
- package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
- package/test/tool/apply_patch.test.ts +0 -614
- package/test/tool/bash.test.ts +0 -1225
- package/test/tool/diagnostics-filter.test.ts +0 -55
- package/test/tool/edit.test.ts +0 -754
- package/test/tool/external-directory.test.ts +0 -169
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +0 -65179
- package/test/tool/glob.test.ts +0 -107
- package/test/tool/grep.test.ts +0 -114
- package/test/tool/lsp.test.ts +0 -187
- package/test/tool/parameters.test.ts +0 -243
- package/test/tool/question.test.ts +0 -129
- package/test/tool/read.test.ts +0 -500
- package/test/tool/recall.test.ts +0 -151
- package/test/tool/registry.test.ts +0 -203
- package/test/tool/skill.test.ts +0 -135
- package/test/tool/suggest.test.ts +0 -1
- package/test/tool/task.test.ts +0 -612
- package/test/tool/tool-define.test.ts +0 -99
- package/test/tool/truncation.test.ts +0 -260
- package/test/tool/webfetch.test.ts +0 -103
- package/test/tool/write.test.ts +0 -291
- package/test/util/data-url.test.ts +0 -14
- package/test/util/effect-zod.test.ts +0 -754
- package/test/util/error.test.ts +0 -38
- package/test/util/filesystem.test.ts +0 -656
- package/test/util/format.test.ts +0 -59
- package/test/util/glob.test.ts +0 -164
- package/test/util/iife.test.ts +0 -36
- package/test/util/lazy.test.ts +0 -50
- package/test/util/lock.test.ts +0 -72
- package/test/util/log.test.ts +0 -86
- package/test/util/module.test.ts +0 -59
- package/test/util/process.test.ts +0 -128
- package/test/util/timeout.test.ts +0 -21
- package/test/util/which.test.ts +0 -100
- package/test/util/wildcard.test.ts +0 -90
- package/test/workspace/workspace-restore.test.ts +0 -296
- package/tsconfig.json +0 -19
|
@@ -1,64 +0,0 @@
|
|
|
1
|
-
import { describe, expect, test } from "bun:test"
|
|
2
|
-
import { $ } from "bun"
|
|
3
|
-
import fs from "node:fs/promises"
|
|
4
|
-
import path from "node:path"
|
|
5
|
-
import { tmpdir } from "../fixture/fixture"
|
|
6
|
-
import { WorktreeDiff } from "../../src/overlay/review/worktree-diff"
|
|
7
|
-
|
|
8
|
-
describe("WorktreeDiff summary", () => {
|
|
9
|
-
async function setup() {
|
|
10
|
-
return await tmpdir({
|
|
11
|
-
git: true,
|
|
12
|
-
init: async (dir) => {
|
|
13
|
-
await Bun.write(path.join(dir, "tracked.txt"), "hello\n")
|
|
14
|
-
await $`git add tracked.txt`.cwd(dir).quiet()
|
|
15
|
-
await $`git commit -m "init"`.cwd(dir).quiet()
|
|
16
|
-
},
|
|
17
|
-
})
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
test("summary returns metadata without loading contents", async () => {
|
|
21
|
-
await using tmp = await setup()
|
|
22
|
-
await Bun.write(path.join(tmp.path, "tracked.txt"), "hello\nworld\n")
|
|
23
|
-
await fs.mkdir(path.join(tmp.path, "node_modules", "pkg"), { recursive: true })
|
|
24
|
-
await Bun.write(path.join(tmp.path, "node_modules", "pkg", "index.js"), "module.exports = 1\n")
|
|
25
|
-
|
|
26
|
-
const diffs = await WorktreeDiff.summary({ dir: tmp.path, base: "HEAD" })
|
|
27
|
-
const tracked = diffs.find((diff) => diff.file === "tracked.txt")
|
|
28
|
-
const generated = diffs.find((diff) => diff.file === "node_modules/pkg/index.js")
|
|
29
|
-
|
|
30
|
-
expect(tracked).toBeDefined()
|
|
31
|
-
expect(tracked?.tracked).toBe(true)
|
|
32
|
-
expect(tracked?.summarized).toBe(true)
|
|
33
|
-
expect(tracked?.before).toBe("")
|
|
34
|
-
expect(tracked?.after).toBe("")
|
|
35
|
-
expect(tracked?.status).toBe("modified")
|
|
36
|
-
|
|
37
|
-
expect(generated).toBeDefined()
|
|
38
|
-
expect(generated?.tracked).toBe(false)
|
|
39
|
-
expect(generated?.generatedLike).toBe(true)
|
|
40
|
-
expect(generated?.summarized).toBe(true)
|
|
41
|
-
expect(generated?.before).toBe("")
|
|
42
|
-
expect(generated?.after).toBe("")
|
|
43
|
-
expect(generated?.additions).toBe(1)
|
|
44
|
-
expect(generated?.status).toBe("added")
|
|
45
|
-
expect(generated?.stamp?.length).toBeGreaterThan(0)
|
|
46
|
-
})
|
|
47
|
-
|
|
48
|
-
test("detail loads one file and computes untracked line counts", async () => {
|
|
49
|
-
await using tmp = await setup()
|
|
50
|
-
await fs.mkdir(path.join(tmp.path, "vendor"), { recursive: true })
|
|
51
|
-
await Bun.write(path.join(tmp.path, "vendor", "bundle.js"), "one\ntwo\nthree\n")
|
|
52
|
-
|
|
53
|
-
const diff = await WorktreeDiff.detail({ dir: tmp.path, base: "HEAD", file: "vendor/bundle.js" })
|
|
54
|
-
|
|
55
|
-
expect(diff).toBeDefined()
|
|
56
|
-
expect(diff?.summarized).toBe(false)
|
|
57
|
-
expect(diff?.tracked).toBe(false)
|
|
58
|
-
expect(diff?.generatedLike).toBe(true)
|
|
59
|
-
expect(diff?.before).toBe("")
|
|
60
|
-
expect(diff?.after).toBe("one\ntwo\nthree\n")
|
|
61
|
-
expect(diff?.additions).toBe(3)
|
|
62
|
-
expect(diff?.status).toBe("added")
|
|
63
|
-
})
|
|
64
|
-
})
|
|
@@ -1,223 +0,0 @@
|
|
|
1
|
-
import { test, expect, describe } from "bun:test"
|
|
2
|
-
import { $ } from "bun"
|
|
3
|
-
import { tmpdir } from "../fixture/fixture"
|
|
4
|
-
import path from "path"
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* Tests for the worktree diff logic used by GET /experimental/worktree/diff.
|
|
8
|
-
* Reproduces the exact git commands from the endpoint to verify they work
|
|
9
|
-
* for tracked, staged, and untracked files.
|
|
10
|
-
*/
|
|
11
|
-
describe("worktree diff git commands", () => {
|
|
12
|
-
async function setupRepo() {
|
|
13
|
-
const tmp = await tmpdir({
|
|
14
|
-
git: true,
|
|
15
|
-
init: async (dir) => {
|
|
16
|
-
// Create an initial file and commit it so we have a base
|
|
17
|
-
await Bun.write(path.join(dir, "existing.txt"), "hello\n")
|
|
18
|
-
await $`git add .`.cwd(dir).quiet()
|
|
19
|
-
await $`git commit -m "add existing.txt"`.cwd(dir).quiet()
|
|
20
|
-
},
|
|
21
|
-
})
|
|
22
|
-
return tmp
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
test("git diff sees committed changes but NOT untracked files", async () => {
|
|
26
|
-
await using tmp = await setupRepo()
|
|
27
|
-
const dir = tmp.path
|
|
28
|
-
|
|
29
|
-
// Get the current HEAD as our "ancestor" (simulating merge-base)
|
|
30
|
-
const headResult = await $`git rev-parse HEAD`.cwd(dir).quiet()
|
|
31
|
-
const ancestor = headResult.stdout.toString().trim()
|
|
32
|
-
|
|
33
|
-
// Create an untracked file (agent writes a file but doesn't stage it)
|
|
34
|
-
await Bun.write(path.join(dir, "life.py"), 'print("hello world")\n')
|
|
35
|
-
|
|
36
|
-
// Verify the file exists
|
|
37
|
-
const exists = await Bun.file(path.join(dir, "life.py")).exists()
|
|
38
|
-
expect(exists).toBe(true)
|
|
39
|
-
|
|
40
|
-
// git diff --name-status does NOT see untracked files
|
|
41
|
-
const nameStatus = await $`git -c core.quotepath=false diff --name-status --no-renames ${ancestor}`
|
|
42
|
-
.cwd(dir)
|
|
43
|
-
.quiet()
|
|
44
|
-
.nothrow()
|
|
45
|
-
const nameStatusOutput = nameStatus.stdout.toString().trim()
|
|
46
|
-
console.log("git diff --name-status output:", JSON.stringify(nameStatusOutput))
|
|
47
|
-
expect(nameStatusOutput).toBe("") // empty — life.py is untracked
|
|
48
|
-
|
|
49
|
-
// git ls-files --others DOES see untracked files
|
|
50
|
-
const untracked = await $`git ls-files --others --exclude-standard`.cwd(dir).quiet().nothrow()
|
|
51
|
-
const untrackedOutput = untracked.stdout.toString().trim()
|
|
52
|
-
console.log("git ls-files --others output:", JSON.stringify(untrackedOutput))
|
|
53
|
-
expect(untrackedOutput).toContain("life.py")
|
|
54
|
-
})
|
|
55
|
-
|
|
56
|
-
test("git diff sees staged (added) files", async () => {
|
|
57
|
-
await using tmp = await setupRepo()
|
|
58
|
-
const dir = tmp.path
|
|
59
|
-
|
|
60
|
-
const headResult = await $`git rev-parse HEAD`.cwd(dir).quiet()
|
|
61
|
-
const ancestor = headResult.stdout.toString().trim()
|
|
62
|
-
|
|
63
|
-
// Create and stage a new file
|
|
64
|
-
await Bun.write(path.join(dir, "staged.py"), 'print("staged")\n')
|
|
65
|
-
await $`git add staged.py`.cwd(dir).quiet()
|
|
66
|
-
|
|
67
|
-
const nameStatus = await $`git -c core.quotepath=false diff --name-status --no-renames ${ancestor}`
|
|
68
|
-
.cwd(dir)
|
|
69
|
-
.quiet()
|
|
70
|
-
.nothrow()
|
|
71
|
-
const nameStatusOutput = nameStatus.stdout.toString().trim()
|
|
72
|
-
console.log("git diff --name-status (staged):", JSON.stringify(nameStatusOutput))
|
|
73
|
-
// git diff <ancestor> (no --cached) compares ancestor to working tree,
|
|
74
|
-
// which includes staged changes
|
|
75
|
-
expect(nameStatusOutput).toContain("staged.py")
|
|
76
|
-
})
|
|
77
|
-
|
|
78
|
-
test("git diff sees modifications to tracked files", async () => {
|
|
79
|
-
await using tmp = await setupRepo()
|
|
80
|
-
const dir = tmp.path
|
|
81
|
-
|
|
82
|
-
const headResult = await $`git rev-parse HEAD`.cwd(dir).quiet()
|
|
83
|
-
const ancestor = headResult.stdout.toString().trim()
|
|
84
|
-
|
|
85
|
-
// Modify existing tracked file without staging
|
|
86
|
-
await Bun.write(path.join(dir, "existing.txt"), "hello\nmodified\n")
|
|
87
|
-
|
|
88
|
-
const nameStatus = await $`git -c core.quotepath=false diff --name-status --no-renames ${ancestor}`
|
|
89
|
-
.cwd(dir)
|
|
90
|
-
.quiet()
|
|
91
|
-
.nothrow()
|
|
92
|
-
const nameStatusOutput = nameStatus.stdout.toString().trim()
|
|
93
|
-
console.log("git diff --name-status (modified):", JSON.stringify(nameStatusOutput))
|
|
94
|
-
expect(nameStatusOutput).toContain("existing.txt")
|
|
95
|
-
})
|
|
96
|
-
|
|
97
|
-
test("full diff pipeline: tracked + untracked combined", async () => {
|
|
98
|
-
await using tmp = await setupRepo()
|
|
99
|
-
const dir = tmp.path
|
|
100
|
-
|
|
101
|
-
const headResult = await $`git rev-parse HEAD`.cwd(dir).quiet()
|
|
102
|
-
const ancestor = headResult.stdout.toString().trim()
|
|
103
|
-
|
|
104
|
-
// Modify existing file (tracked change)
|
|
105
|
-
await Bun.write(path.join(dir, "existing.txt"), "hello\nmodified\n")
|
|
106
|
-
// Create untracked file
|
|
107
|
-
await Bun.write(path.join(dir, "new-file.py"), 'print("new")\n')
|
|
108
|
-
|
|
109
|
-
// Step 1: git diff for tracked changes
|
|
110
|
-
const nameStatus = await $`git -c core.quotepath=false diff --name-status --no-renames ${ancestor}`
|
|
111
|
-
.cwd(dir)
|
|
112
|
-
.quiet()
|
|
113
|
-
.nothrow()
|
|
114
|
-
const tracked = new Set<string>()
|
|
115
|
-
const trackedFiles: string[] = []
|
|
116
|
-
for (const line of nameStatus.stdout.toString().trim().split("\n")) {
|
|
117
|
-
if (!line) continue
|
|
118
|
-
const parts = line.split("\t")
|
|
119
|
-
const file = parts.slice(1).join("\t")
|
|
120
|
-
if (file) {
|
|
121
|
-
tracked.add(file)
|
|
122
|
-
trackedFiles.push(file)
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
// Step 2: git ls-files for untracked
|
|
127
|
-
const untrackedResult = await $`git ls-files --others --exclude-standard`.cwd(dir).quiet().nothrow()
|
|
128
|
-
const untrackedFiles: string[] = []
|
|
129
|
-
for (const file of untrackedResult.stdout.toString().trim().split("\n")) {
|
|
130
|
-
if (!file || tracked.has(file)) continue
|
|
131
|
-
untrackedFiles.push(file)
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
console.log("tracked files:", trackedFiles)
|
|
135
|
-
console.log("untracked files:", untrackedFiles)
|
|
136
|
-
|
|
137
|
-
expect(trackedFiles).toContain("existing.txt")
|
|
138
|
-
expect(trackedFiles).not.toContain("new-file.py")
|
|
139
|
-
expect(untrackedFiles).toContain("new-file.py")
|
|
140
|
-
expect(untrackedFiles).not.toContain("existing.txt")
|
|
141
|
-
|
|
142
|
-
// Combined = both
|
|
143
|
-
const allFiles = [...trackedFiles, ...untrackedFiles]
|
|
144
|
-
expect(allFiles).toContain("existing.txt")
|
|
145
|
-
expect(allFiles).toContain("new-file.py")
|
|
146
|
-
})
|
|
147
|
-
|
|
148
|
-
test("worktree scenario: branch with no new commits, only untracked files", async () => {
|
|
149
|
-
// This is the exact scenario from the screenshot:
|
|
150
|
-
// - Worktree created from main
|
|
151
|
-
// - Agent writes life.py (never committed/staged)
|
|
152
|
-
// - merge-base HEAD main = HEAD (no divergence)
|
|
153
|
-
// - git diff shows nothing, git ls-files --others shows life.py
|
|
154
|
-
await using tmp = await setupRepo()
|
|
155
|
-
const dir = tmp.path
|
|
156
|
-
|
|
157
|
-
// Simulate: worktree is on same commit as base (no new commits)
|
|
158
|
-
// merge-base HEAD HEAD = HEAD
|
|
159
|
-
const mergeBase = await $`git merge-base HEAD HEAD`.cwd(dir).quiet()
|
|
160
|
-
const ancestor = mergeBase.stdout.toString().trim()
|
|
161
|
-
console.log("ancestor (same as HEAD):", ancestor)
|
|
162
|
-
|
|
163
|
-
// Agent writes a file
|
|
164
|
-
await Bun.write(
|
|
165
|
-
path.join(dir, "life.py"),
|
|
166
|
-
`
|
|
167
|
-
import random
|
|
168
|
-
import time
|
|
169
|
-
import os
|
|
170
|
-
|
|
171
|
-
def create_board(rows, cols):
|
|
172
|
-
return [[random.choice([0, 1]) for _ in range(cols)] for _ in range(rows)]
|
|
173
|
-
|
|
174
|
-
print("Game of Life")
|
|
175
|
-
`,
|
|
176
|
-
)
|
|
177
|
-
|
|
178
|
-
// git diff: nothing (HEAD == ancestor, no tracked changes)
|
|
179
|
-
const nameStatus = await $`git -c core.quotepath=false diff --name-status --no-renames ${ancestor}`
|
|
180
|
-
.cwd(dir)
|
|
181
|
-
.quiet()
|
|
182
|
-
.nothrow()
|
|
183
|
-
const nameStatusRaw = nameStatus.stdout.toString().trim()
|
|
184
|
-
console.log("nameStatus raw:", JSON.stringify(nameStatusRaw))
|
|
185
|
-
|
|
186
|
-
// git ls-files --others: should find life.py
|
|
187
|
-
const untrackedResult = await $`git ls-files --others --exclude-standard`.cwd(dir).quiet().nothrow()
|
|
188
|
-
const untrackedRaw = untrackedResult.stdout.toString().trim()
|
|
189
|
-
console.log("untracked raw:", JSON.stringify(untrackedRaw))
|
|
190
|
-
|
|
191
|
-
expect(untrackedRaw).toContain("life.py")
|
|
192
|
-
|
|
193
|
-
// Now simulate the full endpoint logic
|
|
194
|
-
const seen = new Set<string>()
|
|
195
|
-
const diffs: { file: string; status: string }[] = []
|
|
196
|
-
|
|
197
|
-
// Process tracked changes
|
|
198
|
-
for (const line of nameStatusRaw.split("\n")) {
|
|
199
|
-
if (!line) continue
|
|
200
|
-
const parts = line.split("\t")
|
|
201
|
-
const file = parts.slice(1).join("\t")
|
|
202
|
-
if (file) {
|
|
203
|
-
seen.add(file)
|
|
204
|
-
diffs.push({ file, status: "modified" })
|
|
205
|
-
}
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
// Process untracked files
|
|
209
|
-
if (untrackedResult.exitCode === 0) {
|
|
210
|
-
for (const file of untrackedRaw.split("\n")) {
|
|
211
|
-
if (!file || seen.has(file)) continue
|
|
212
|
-
const f = Bun.file(path.join(dir, file))
|
|
213
|
-
if (!(await f.exists())) continue
|
|
214
|
-
diffs.push({ file, status: "added" })
|
|
215
|
-
}
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
console.log("final diffs:", diffs)
|
|
219
|
-
expect(diffs.length).toBe(1)
|
|
220
|
-
expect(diffs[0]!.file).toBe("life.py")
|
|
221
|
-
expect(diffs[0]!.status).toBe("added")
|
|
222
|
-
})
|
|
223
|
-
})
|
|
@@ -1,82 +0,0 @@
|
|
|
1
|
-
import { $ } from "bun"
|
|
2
|
-
import { describe, expect } from "bun:test"
|
|
3
|
-
import * as fs from "fs/promises"
|
|
4
|
-
import path from "path"
|
|
5
|
-
import { Effect, Layer } from "effect"
|
|
6
|
-
import * as CrossSpawnSpawner from "@saeeol/core/cross-spawn-spawner"
|
|
7
|
-
import { Worktree } from "../../src/worktree"
|
|
8
|
-
import { provideTmpdirInstance } from "../fixture/fixture"
|
|
9
|
-
import { testEffect } from "../lib/effect"
|
|
10
|
-
|
|
11
|
-
const it = testEffect(Layer.mergeAll(Worktree.defaultLayer, CrossSpawnSpawner.defaultLayer))
|
|
12
|
-
|
|
13
|
-
describe("Worktree.remove lock retries", () => {
|
|
14
|
-
it.live("retries transient git remove lock failures", () =>
|
|
15
|
-
provideTmpdirInstance(
|
|
16
|
-
(root) =>
|
|
17
|
-
Effect.gen(function* () {
|
|
18
|
-
const svc = yield* Worktree.Service
|
|
19
|
-
const name = `remove-retry-${Date.now().toString(36)}`
|
|
20
|
-
const branch = `saeeol/${name}`
|
|
21
|
-
const dir = path.join(root, "..", name)
|
|
22
|
-
|
|
23
|
-
yield* Effect.promise(() => $`git worktree add --no-checkout -b ${branch} ${dir}`.cwd(root).quiet())
|
|
24
|
-
yield* Effect.promise(() => $`git reset --hard`.cwd(dir).quiet())
|
|
25
|
-
|
|
26
|
-
const real = (yield* Effect.promise(() => $`which git`.quiet().text())).trim()
|
|
27
|
-
expect(real).toBeTruthy()
|
|
28
|
-
|
|
29
|
-
const bin = path.join(root, "bin")
|
|
30
|
-
const shim = path.join(bin, "git")
|
|
31
|
-
const state = path.join(bin, "attempt")
|
|
32
|
-
yield* Effect.promise(() => fs.mkdir(bin, { recursive: true }))
|
|
33
|
-
yield* Effect.promise(() =>
|
|
34
|
-
Bun.write(
|
|
35
|
-
shim,
|
|
36
|
-
[
|
|
37
|
-
"#!/bin/bash",
|
|
38
|
-
`REAL_GIT=${JSON.stringify(real)}`,
|
|
39
|
-
`STATE=${JSON.stringify(state)}`,
|
|
40
|
-
'if [ "$1" = "worktree" ] && [ "$2" = "remove" ] && [ ! -f "$STATE" ]; then',
|
|
41
|
-
' touch "$STATE"',
|
|
42
|
-
' echo "fatal: EBUSY: resource busy or locked, rmdir $4" >&2',
|
|
43
|
-
" exit 1",
|
|
44
|
-
"fi",
|
|
45
|
-
'exec "$REAL_GIT" "$@"',
|
|
46
|
-
].join("\n"),
|
|
47
|
-
),
|
|
48
|
-
)
|
|
49
|
-
yield* Effect.promise(() => fs.chmod(shim, 0o755))
|
|
50
|
-
|
|
51
|
-
const prev = yield* Effect.acquireRelease(
|
|
52
|
-
Effect.sync(() => {
|
|
53
|
-
const prev = process.env.PATH ?? ""
|
|
54
|
-
process.env.PATH = `${bin}${path.delimiter}${prev}`
|
|
55
|
-
return prev
|
|
56
|
-
}),
|
|
57
|
-
(prev) =>
|
|
58
|
-
Effect.sync(() => {
|
|
59
|
-
process.env.PATH = prev
|
|
60
|
-
}),
|
|
61
|
-
)
|
|
62
|
-
void prev
|
|
63
|
-
|
|
64
|
-
const ok = yield* svc.remove({ directory: dir })
|
|
65
|
-
|
|
66
|
-
expect(ok).toBe(true)
|
|
67
|
-
expect(
|
|
68
|
-
yield* Effect.promise(() =>
|
|
69
|
-
fs
|
|
70
|
-
.stat(dir)
|
|
71
|
-
.then(() => true)
|
|
72
|
-
.catch(() => false),
|
|
73
|
-
),
|
|
74
|
-
).toBe(false)
|
|
75
|
-
|
|
76
|
-
const list = yield* Effect.promise(() => $`git worktree list --porcelain`.cwd(root).quiet().text())
|
|
77
|
-
expect(list).not.toContain(`worktree ${dir}`)
|
|
78
|
-
}),
|
|
79
|
-
{ git: true },
|
|
80
|
-
),
|
|
81
|
-
)
|
|
82
|
-
})
|
package/test/server/AGENTS.md
DELETED
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
# Server Test Guide
|
|
2
|
-
|
|
3
|
-
Use these patterns for server and HttpApi middleware tests in this directory.
|
|
4
|
-
|
|
5
|
-
- Prefer focused middleware tests with tiny fake routes over full API route trees when testing routing, context, proxying, or middleware policy.
|
|
6
|
-
- Use `testEffect(...)` with `NodeHttpServer.layerTest` for the primary in-test server and make relative `HttpClient` requests against it.
|
|
7
|
-
- Use `HttpRouter.add(...)` probe routes that expose the context under test, such as `WorkspaceRouteContext`, `InstanceRef`, or `WorkspaceRef`.
|
|
8
|
-
- Compose middleware in the same order as production when testing interactions, for example `instanceRouterMiddleware.combine(workspaceRouterMiddleware)`.
|
|
9
|
-
- For secondary upstream servers, build Effect `NodeHttpServer.layer(...)` into the current test scope with `Layer.build(...)` so the listener stays alive until the test scope exits.
|
|
10
|
-
- Avoid `Bun.serve` when testing Effect HTTP middleware. Keep the test in the Effect HTTP stack unless the production path being tested is Bun-specific.
|
|
11
|
-
- For WebSocket paths, use `Socket.makeWebSocket(...)` from the test client and assert protocol forwarding or frame relay when relevant.
|
|
12
|
-
- Use scoped test layers for flags, database reset, and other global mutable state. Restore flags and reset state in finalizers.
|
|
13
|
-
- Use `tmpdirScoped({ git: true })` plus `Project.use.fromDirectory(dir)` for project-backed requests.
|
|
14
|
-
- If a test needs persisted state without matching runtime state, keep direct database setup inside a narrowly named helper that explains that state.
|
|
15
|
-
- Add comments for non-obvious test topology, especially tests involving both the local test server and a fake upstream server.
|
|
@@ -1,231 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* contract.test.ts — SAEEOL 서버 계약 테스트
|
|
3
|
-
*
|
|
4
|
-
* closebook 패턴 기반:
|
|
5
|
-
* - response-conformance: Zod 스키마로 응답 바디 적합성 검사
|
|
6
|
-
* - consumer-contract: SDK/클라이언트가 기대하는 API 응답 스키마 검증
|
|
7
|
-
* - stateful-api-testing: 세션 생성 → 목록 조회 순차 테스트
|
|
8
|
-
*
|
|
9
|
-
* Hono app.request()로 실제 HTTP 서버 없이 테스트.
|
|
10
|
-
* Bun test + Effect + zod 검증.
|
|
11
|
-
*/
|
|
12
|
-
|
|
13
|
-
import { afterEach, describe, expect, test } from "bun:test"
|
|
14
|
-
import { Effect } from "effect"
|
|
15
|
-
import { Flag } from "@saeeol/core/flag/flag"
|
|
16
|
-
import { Server } from "../../src/server/server"
|
|
17
|
-
import { resetDatabase } from "../fixture/db"
|
|
18
|
-
import { disposeAllInstances, tmpdir } from "../fixture/fixture"
|
|
19
|
-
import * as Log from "@saeeol/core/util/log"
|
|
20
|
-
import { z } from "zod"
|
|
21
|
-
|
|
22
|
-
void Log.init({ print: false })
|
|
23
|
-
|
|
24
|
-
const original = Flag.SAEEOL_EXPERIMENTAL_HTTPAPI
|
|
25
|
-
|
|
26
|
-
function app(experimental = true) {
|
|
27
|
-
Flag.SAEEOL_EXPERIMENTAL_HTTPAPI = experimental
|
|
28
|
-
return experimental ? Server.Default().app : Server.Legacy().app
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
// ── Zod 스키마 (계약: 클라이언트가 기대하는 응답 형태) ──
|
|
32
|
-
|
|
33
|
-
const HealthSchema = z.object({
|
|
34
|
-
healthy: z.boolean(),
|
|
35
|
-
version: z.string(),
|
|
36
|
-
})
|
|
37
|
-
|
|
38
|
-
const ConfigSchema = z.object({
|
|
39
|
-
model: z.string().optional(),
|
|
40
|
-
username: z.string().optional(),
|
|
41
|
-
}).passthrough()
|
|
42
|
-
|
|
43
|
-
const SessionListSchema = z.array(z.object({
|
|
44
|
-
id: z.string(),
|
|
45
|
-
title: z.string().optional(),
|
|
46
|
-
createdAt: z.number().optional(),
|
|
47
|
-
}).passthrough())
|
|
48
|
-
|
|
49
|
-
const SessionCreateSchema = z.object({
|
|
50
|
-
id: z.string(),
|
|
51
|
-
}).passthrough()
|
|
52
|
-
|
|
53
|
-
const AgentListSchema = z.array(z.unknown())
|
|
54
|
-
|
|
55
|
-
// ── 응답 적합성 검증 (closebook: response-conformance) ──
|
|
56
|
-
|
|
57
|
-
function checkConformance<T extends z.ZodType>(
|
|
58
|
-
body: unknown,
|
|
59
|
-
status: number,
|
|
60
|
-
schema: T,
|
|
61
|
-
allowedStatuses: number[] = [200],
|
|
62
|
-
): { ok: boolean; violations: string[] } {
|
|
63
|
-
const violations: string[] = []
|
|
64
|
-
|
|
65
|
-
if (!allowedStatuses.includes(status)) {
|
|
66
|
-
violations.push(`status: expected ${allowedStatuses.join("/")}, got ${status}`)
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
const parsed = schema.safeParse(body)
|
|
70
|
-
if (!parsed.success) {
|
|
71
|
-
for (const issue of parsed.error.issues) {
|
|
72
|
-
violations.push(`${issue.path.join(".") || "(root)"}: ${issue.message}`)
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
return { ok: violations.length === 0, violations }
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
// ── 테스트 ──
|
|
80
|
-
|
|
81
|
-
describe("SAEEOL server contract tests", () => {
|
|
82
|
-
afterEach(async () => {
|
|
83
|
-
Flag.SAEEOL_EXPERIMENTAL_HTTPAPI = original
|
|
84
|
-
await disposeAllInstances()
|
|
85
|
-
})
|
|
86
|
-
|
|
87
|
-
describe("GET /global/health", () => {
|
|
88
|
-
test("returns valid health response (legacy)", async () => {
|
|
89
|
-
const res = await app(false).request("/global/health")
|
|
90
|
-
const body = await res.json()
|
|
91
|
-
expect(res.status).toBe(200)
|
|
92
|
-
const { ok, violations } = checkConformance(body, res.status, HealthSchema)
|
|
93
|
-
expect(ok).toBe(true)
|
|
94
|
-
if (!ok) console.error("Health violations:", violations)
|
|
95
|
-
})
|
|
96
|
-
|
|
97
|
-
test("returns valid health response (experimental)", async () => {
|
|
98
|
-
const res = await app(true).request("/global/health")
|
|
99
|
-
const body = await res.json()
|
|
100
|
-
expect(res.status).toBe(200)
|
|
101
|
-
const { ok, violations } = checkConformance(body, res.status, HealthSchema)
|
|
102
|
-
expect(ok).toBe(true)
|
|
103
|
-
if (!ok) console.error("Health violations:", violations)
|
|
104
|
-
})
|
|
105
|
-
})
|
|
106
|
-
|
|
107
|
-
describe("GET /global/config", () => {
|
|
108
|
-
test("returns valid config response", async () => {
|
|
109
|
-
await using tmp = await tmpdir({ config: {} })
|
|
110
|
-
const res = await app(false).request("/global/config", {
|
|
111
|
-
headers: { "x-saeeol-directory": tmp.path },
|
|
112
|
-
})
|
|
113
|
-
const body = await res.json()
|
|
114
|
-
const { ok, violations } = checkConformance(body, res.status, ConfigSchema)
|
|
115
|
-
expect(ok).toBe(true)
|
|
116
|
-
if (!ok) console.error("Config violations:", violations)
|
|
117
|
-
})
|
|
118
|
-
})
|
|
119
|
-
|
|
120
|
-
describe("GET /agent", () => {
|
|
121
|
-
test("returns agent list", async () => {
|
|
122
|
-
await using tmp = await tmpdir({ config: {} })
|
|
123
|
-
const res = await app(false).request("/agent", {
|
|
124
|
-
headers: { "x-saeeol-directory": tmp.path },
|
|
125
|
-
})
|
|
126
|
-
const body = await res.json()
|
|
127
|
-
const { ok, violations } = checkConformance(body, res.status, AgentListSchema)
|
|
128
|
-
expect(ok).toBe(true)
|
|
129
|
-
if (!ok) console.error("Agent violations:", violations)
|
|
130
|
-
})
|
|
131
|
-
})
|
|
132
|
-
|
|
133
|
-
// ── 상태 순차 테스트 (closebook: stateful-api-testing) ──
|
|
134
|
-
|
|
135
|
-
describe("session lifecycle", () => {
|
|
136
|
-
test("create → list flow", async () => {
|
|
137
|
-
await using tmp = await tmpdir({ git: true, config: {} })
|
|
138
|
-
const headers = { "x-saeeol-directory": tmp.path, "Content-Type": "application/json" }
|
|
139
|
-
|
|
140
|
-
// Step 1: 세션 생성
|
|
141
|
-
const createRes = await app(false).request("/session", {
|
|
142
|
-
method: "POST",
|
|
143
|
-
headers,
|
|
144
|
-
body: JSON.stringify({ title: "contract-test" }),
|
|
145
|
-
})
|
|
146
|
-
const createBody = await createRes.json()
|
|
147
|
-
const { ok: createOk, violations: createViolations } = checkConformance(
|
|
148
|
-
createBody,
|
|
149
|
-
createRes.status,
|
|
150
|
-
SessionCreateSchema,
|
|
151
|
-
)
|
|
152
|
-
expect(createOk).toBe(true)
|
|
153
|
-
if (!createOk) console.error("Session create violations:", createViolations)
|
|
154
|
-
|
|
155
|
-
// Step 2: 세션 목록
|
|
156
|
-
const listRes = await app(false).request("/session", { headers })
|
|
157
|
-
const listBody = await listRes.json()
|
|
158
|
-
const { ok: listOk, violations: listViolations } = checkConformance(
|
|
159
|
-
listBody,
|
|
160
|
-
listRes.status,
|
|
161
|
-
SessionListSchema,
|
|
162
|
-
)
|
|
163
|
-
expect(listOk).toBe(true)
|
|
164
|
-
if (!listOk) console.error("Session list violations:", listViolations)
|
|
165
|
-
expect(listBody.length).toBeGreaterThanOrEqual(1)
|
|
166
|
-
})
|
|
167
|
-
})
|
|
168
|
-
|
|
169
|
-
describe("unknown route", () => {
|
|
170
|
-
test("returns 404 for unknown endpoints", async () => {
|
|
171
|
-
const res = await app(false).request("/nonexistent")
|
|
172
|
-
expect(res.status).toBe(404)
|
|
173
|
-
})
|
|
174
|
-
})
|
|
175
|
-
|
|
176
|
-
// ── 계약 파일 생성 (closebook: consumer-contract) ──
|
|
177
|
-
|
|
178
|
-
describe("pact artifact generation", () => {
|
|
179
|
-
test("collects verified interactions into contract", async () => {
|
|
180
|
-
await using tmp = await tmpdir({ config: {} })
|
|
181
|
-
const headers = { "x-saeeol-directory": tmp.path }
|
|
182
|
-
const interactions = []
|
|
183
|
-
|
|
184
|
-
// Health 인터랙션
|
|
185
|
-
const healthRes = await app(false).request("/global/health")
|
|
186
|
-
const healthBody = await healthRes.json()
|
|
187
|
-
interactions.push({
|
|
188
|
-
description: "health check",
|
|
189
|
-
providerStates: [],
|
|
190
|
-
request: { method: "GET", path: "/global/health" },
|
|
191
|
-
response: { status: healthRes.status, body: healthBody },
|
|
192
|
-
})
|
|
193
|
-
|
|
194
|
-
// Config 인터랙션
|
|
195
|
-
const configRes = await app(false).request("/global/config", { headers })
|
|
196
|
-
const configBody = await configRes.json()
|
|
197
|
-
interactions.push({
|
|
198
|
-
description: "global config",
|
|
199
|
-
providerStates: [],
|
|
200
|
-
request: { method: "GET", path: "/global/config" },
|
|
201
|
-
response: { status: configRes.status, body: configBody },
|
|
202
|
-
})
|
|
203
|
-
|
|
204
|
-
// Agent 인터랙션
|
|
205
|
-
const agentRes = await app(false).request("/agent", { headers })
|
|
206
|
-
const agentBody = await agentRes.json()
|
|
207
|
-
interactions.push({
|
|
208
|
-
description: "agent list",
|
|
209
|
-
providerStates: [],
|
|
210
|
-
request: { method: "GET", path: "/agent" },
|
|
211
|
-
response: { status: agentRes.status, body: agentBody },
|
|
212
|
-
})
|
|
213
|
-
|
|
214
|
-
// 계약 파일 검증
|
|
215
|
-
const pact = {
|
|
216
|
-
consumer: { name: "saeeol-vscode" },
|
|
217
|
-
provider: { name: "saeeol-server" },
|
|
218
|
-
interactions,
|
|
219
|
-
metadata: { version: "1.0.0" },
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
expect(pact.interactions.length).toBe(3)
|
|
223
|
-
for (const ix of pact.interactions) {
|
|
224
|
-
expect(ix).toHaveProperty("description")
|
|
225
|
-
expect(ix).toHaveProperty("request")
|
|
226
|
-
expect(ix).toHaveProperty("response")
|
|
227
|
-
expect(ix.response.status).toBeLessThan(500)
|
|
228
|
-
}
|
|
229
|
-
})
|
|
230
|
-
})
|
|
231
|
-
})
|