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,356 +0,0 @@
|
|
|
1
|
-
import { Database } from "bun:sqlite"
|
|
2
|
-
import { mkdir, symlink } from "node:fs/promises"
|
|
3
|
-
import os from "node:os"
|
|
4
|
-
import path from "node:path"
|
|
5
|
-
import { expect, spyOn, test } from "bun:test"
|
|
6
|
-
import { offsetToPosition, resolveZedDbPath, resolveZedSelection } from "../../../src/cli/cmd/tui/context/editor-zed"
|
|
7
|
-
import { tmpdir } from "../../fixture/fixture"
|
|
8
|
-
|
|
9
|
-
type ZedFixtureOptions = {
|
|
10
|
-
workspacePaths?: string | null
|
|
11
|
-
itemKind?: string
|
|
12
|
-
editor?: boolean
|
|
13
|
-
selectionStart?: number | null
|
|
14
|
-
selectionEnd?: number | null
|
|
15
|
-
selections?: Array<{ start: number | null; end: number | null }>
|
|
16
|
-
contents?: string
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
async function writeZedFixture(dir: string, options: ZedFixtureOptions = {}) {
|
|
20
|
-
const dbPath = path.join(dir, "zed.sqlite")
|
|
21
|
-
const filePath = path.join(dir, "file.ts")
|
|
22
|
-
const contents = options.contents ?? "one\ntwo\nthree"
|
|
23
|
-
await Bun.write(filePath, contents)
|
|
24
|
-
|
|
25
|
-
const db = new Database(dbPath)
|
|
26
|
-
db.run("create table workspaces (workspace_id integer, paths text, timestamp text)")
|
|
27
|
-
db.run("create table panes (pane_id integer, workspace_id integer, active integer)")
|
|
28
|
-
db.run("create table items (item_id integer, workspace_id integer, pane_id integer, active integer, kind text)")
|
|
29
|
-
db.run("create table editors (item_id integer, workspace_id integer, buffer_path text, contents text)")
|
|
30
|
-
db.run("create table editor_selections (editor_id integer, workspace_id integer, start integer, end integer)")
|
|
31
|
-
db.run("insert into workspaces values (1, ?, ?)", [options.workspacePaths ?? JSON.stringify([dir]), "2026-04-27"])
|
|
32
|
-
db.run("insert into panes values (1, 1, 1)")
|
|
33
|
-
db.run("insert into items values (1, 1, 1, 1, ?)", [options.itemKind ?? "Editor"])
|
|
34
|
-
if (options.editor !== false) {
|
|
35
|
-
db.run("insert into editors values (1, 1, ?, ?)", [filePath, contents])
|
|
36
|
-
;(
|
|
37
|
-
options.selections ?? [
|
|
38
|
-
{
|
|
39
|
-
start: options.selectionStart === undefined ? 4 : options.selectionStart,
|
|
40
|
-
end: options.selectionEnd === undefined ? 7 : options.selectionEnd,
|
|
41
|
-
},
|
|
42
|
-
]
|
|
43
|
-
).forEach((selection) =>
|
|
44
|
-
db.run("insert into editor_selections values (1, 1, ?, ?)", [selection.start, selection.end]),
|
|
45
|
-
)
|
|
46
|
-
}
|
|
47
|
-
db.close()
|
|
48
|
-
|
|
49
|
-
return { dbPath, filePath }
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
function utf8ByteOffset(text: string, offset: number) {
|
|
53
|
-
return new TextEncoder().encode(text.slice(0, offset)).length
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
test("offsetToPosition converts Zed offsets to 1-based editor positions", () => {
|
|
57
|
-
expect(offsetToPosition("one\ntwo\nthree", 0)).toEqual({ line: 1, character: 1 })
|
|
58
|
-
expect(offsetToPosition("one\ntwo\nthree", 4)).toEqual({ line: 2, character: 1 })
|
|
59
|
-
expect(offsetToPosition("one\ntwo\nthree", 6)).toEqual({ line: 2, character: 3 })
|
|
60
|
-
expect(offsetToPosition("one\ntwo\nthree", 100)).toEqual({ line: 3, character: 6 })
|
|
61
|
-
expect(offsetToPosition("Ж\nabc", utf8ByteOffset("Ж\nabc", "Ж\nabc".indexOf("a")))).toEqual({
|
|
62
|
-
line: 2,
|
|
63
|
-
character: 1,
|
|
64
|
-
})
|
|
65
|
-
expect(offsetToPosition("😀\nabc", utf8ByteOffset("😀\nabc", "😀\nabc".indexOf("a")))).toEqual({
|
|
66
|
-
line: 2,
|
|
67
|
-
character: 1,
|
|
68
|
-
})
|
|
69
|
-
})
|
|
70
|
-
|
|
71
|
-
test("resolveZedDbPath skips candidates that cannot be stated", async () => {
|
|
72
|
-
await using tmp = await tmpdir()
|
|
73
|
-
const loop = path.join(tmp.path, "loop")
|
|
74
|
-
await symlink(loop, loop)
|
|
75
|
-
const home = spyOn(os, "homedir").mockImplementation(() => tmp.path)
|
|
76
|
-
const previous = process.env.SAEEOL_ZED_DB
|
|
77
|
-
process.env.SAEEOL_ZED_DB = loop
|
|
78
|
-
|
|
79
|
-
try {
|
|
80
|
-
expect(resolveZedDbPath()).toBeUndefined()
|
|
81
|
-
} finally {
|
|
82
|
-
if (previous === undefined) delete process.env.SAEEOL_ZED_DB
|
|
83
|
-
else process.env.SAEEOL_ZED_DB = previous
|
|
84
|
-
home.mockRestore()
|
|
85
|
-
}
|
|
86
|
-
})
|
|
87
|
-
|
|
88
|
-
test("resolveZedSelection returns active editor selection", async () => {
|
|
89
|
-
await using tmp = await tmpdir()
|
|
90
|
-
const fixture = await writeZedFixture(tmp.path)
|
|
91
|
-
|
|
92
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({
|
|
93
|
-
type: "selection",
|
|
94
|
-
selection: {
|
|
95
|
-
filePath: fixture.filePath,
|
|
96
|
-
source: "zed",
|
|
97
|
-
ranges: [
|
|
98
|
-
{
|
|
99
|
-
text: "two",
|
|
100
|
-
selection: {
|
|
101
|
-
start: { line: 2, character: 1 },
|
|
102
|
-
end: { line: 2, character: 4 },
|
|
103
|
-
},
|
|
104
|
-
},
|
|
105
|
-
],
|
|
106
|
-
},
|
|
107
|
-
})
|
|
108
|
-
})
|
|
109
|
-
|
|
110
|
-
test("resolveZedSelection returns all active editor selections sorted by offset", async () => {
|
|
111
|
-
await using tmp = await tmpdir()
|
|
112
|
-
const contents = "one\ntwo\nthree\nfour"
|
|
113
|
-
const fixture = await writeZedFixture(tmp.path, {
|
|
114
|
-
contents,
|
|
115
|
-
selections: [
|
|
116
|
-
{
|
|
117
|
-
start: utf8ByteOffset(contents, contents.indexOf("four")),
|
|
118
|
-
end: utf8ByteOffset(contents, contents.indexOf("four") + 4),
|
|
119
|
-
},
|
|
120
|
-
{
|
|
121
|
-
start: utf8ByteOffset(contents, contents.indexOf("two")),
|
|
122
|
-
end: utf8ByteOffset(contents, contents.indexOf("two") + 3),
|
|
123
|
-
},
|
|
124
|
-
],
|
|
125
|
-
})
|
|
126
|
-
|
|
127
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({
|
|
128
|
-
type: "selection",
|
|
129
|
-
selection: {
|
|
130
|
-
filePath: fixture.filePath,
|
|
131
|
-
source: "zed",
|
|
132
|
-
ranges: [
|
|
133
|
-
{
|
|
134
|
-
text: "two",
|
|
135
|
-
selection: {
|
|
136
|
-
start: { line: 2, character: 1 },
|
|
137
|
-
end: { line: 2, character: 4 },
|
|
138
|
-
},
|
|
139
|
-
},
|
|
140
|
-
{
|
|
141
|
-
text: "four",
|
|
142
|
-
selection: {
|
|
143
|
-
start: { line: 4, character: 1 },
|
|
144
|
-
end: { line: 4, character: 5 },
|
|
145
|
-
},
|
|
146
|
-
},
|
|
147
|
-
],
|
|
148
|
-
},
|
|
149
|
-
})
|
|
150
|
-
})
|
|
151
|
-
|
|
152
|
-
test("resolveZedSelection converts Zed UTF-8 byte offsets to string offsets", async () => {
|
|
153
|
-
await using tmp = await tmpdir()
|
|
154
|
-
const contents = "a\nЖЖЖЖЖЖЖЖЖЖ\nb\nTARGET\nz"
|
|
155
|
-
const start = contents.indexOf("TARGET")
|
|
156
|
-
const fixture = await writeZedFixture(tmp.path, {
|
|
157
|
-
contents,
|
|
158
|
-
selectionStart: utf8ByteOffset(contents, start),
|
|
159
|
-
selectionEnd: utf8ByteOffset(contents, start + "TARGET".length),
|
|
160
|
-
})
|
|
161
|
-
|
|
162
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({
|
|
163
|
-
type: "selection",
|
|
164
|
-
selection: {
|
|
165
|
-
filePath: fixture.filePath,
|
|
166
|
-
source: "zed",
|
|
167
|
-
ranges: [
|
|
168
|
-
{
|
|
169
|
-
text: "TARGET",
|
|
170
|
-
selection: {
|
|
171
|
-
start: { line: 4, character: 1 },
|
|
172
|
-
end: { line: 4, character: 7 },
|
|
173
|
-
},
|
|
174
|
-
},
|
|
175
|
-
],
|
|
176
|
-
},
|
|
177
|
-
})
|
|
178
|
-
})
|
|
179
|
-
|
|
180
|
-
test("resolveZedSelection handles non-ASCII text inside the selected range", async () => {
|
|
181
|
-
await using tmp = await tmpdir()
|
|
182
|
-
const contents = "a\npre\nвыбор\nz"
|
|
183
|
-
const start = contents.indexOf("выбор")
|
|
184
|
-
const fixture = await writeZedFixture(tmp.path, {
|
|
185
|
-
contents,
|
|
186
|
-
selectionStart: utf8ByteOffset(contents, start),
|
|
187
|
-
selectionEnd: utf8ByteOffset(contents, start + "выбор".length),
|
|
188
|
-
})
|
|
189
|
-
|
|
190
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({
|
|
191
|
-
type: "selection",
|
|
192
|
-
selection: {
|
|
193
|
-
filePath: fixture.filePath,
|
|
194
|
-
source: "zed",
|
|
195
|
-
ranges: [
|
|
196
|
-
{
|
|
197
|
-
text: "выбор",
|
|
198
|
-
selection: {
|
|
199
|
-
start: { line: 3, character: 1 },
|
|
200
|
-
end: { line: 3, character: 6 },
|
|
201
|
-
},
|
|
202
|
-
},
|
|
203
|
-
],
|
|
204
|
-
},
|
|
205
|
-
})
|
|
206
|
-
})
|
|
207
|
-
|
|
208
|
-
test("resolveZedSelection handles emoji before the selected range", async () => {
|
|
209
|
-
await using tmp = await tmpdir()
|
|
210
|
-
const contents = "😀\nTARGET\nz"
|
|
211
|
-
const start = contents.indexOf("TARGET")
|
|
212
|
-
const fixture = await writeZedFixture(tmp.path, {
|
|
213
|
-
contents,
|
|
214
|
-
selectionStart: utf8ByteOffset(contents, start),
|
|
215
|
-
selectionEnd: utf8ByteOffset(contents, start + "TARGET".length),
|
|
216
|
-
})
|
|
217
|
-
|
|
218
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({
|
|
219
|
-
type: "selection",
|
|
220
|
-
selection: {
|
|
221
|
-
filePath: fixture.filePath,
|
|
222
|
-
source: "zed",
|
|
223
|
-
ranges: [
|
|
224
|
-
{
|
|
225
|
-
text: "TARGET",
|
|
226
|
-
selection: {
|
|
227
|
-
start: { line: 2, character: 1 },
|
|
228
|
-
end: { line: 2, character: 7 },
|
|
229
|
-
},
|
|
230
|
-
},
|
|
231
|
-
],
|
|
232
|
-
},
|
|
233
|
-
})
|
|
234
|
-
})
|
|
235
|
-
|
|
236
|
-
test("resolveZedSelection handles reversed Zed byte offsets", async () => {
|
|
237
|
-
await using tmp = await tmpdir()
|
|
238
|
-
const contents = "a\nЖЖЖ\nTARGET\nz"
|
|
239
|
-
const start = contents.indexOf("TARGET")
|
|
240
|
-
const fixture = await writeZedFixture(tmp.path, {
|
|
241
|
-
contents,
|
|
242
|
-
selectionStart: utf8ByteOffset(contents, start + "TARGET".length),
|
|
243
|
-
selectionEnd: utf8ByteOffset(contents, start),
|
|
244
|
-
})
|
|
245
|
-
|
|
246
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({
|
|
247
|
-
type: "selection",
|
|
248
|
-
selection: {
|
|
249
|
-
filePath: fixture.filePath,
|
|
250
|
-
source: "zed",
|
|
251
|
-
ranges: [
|
|
252
|
-
{
|
|
253
|
-
text: "TARGET",
|
|
254
|
-
selection: {
|
|
255
|
-
start: { line: 3, character: 1 },
|
|
256
|
-
end: { line: 3, character: 7 },
|
|
257
|
-
},
|
|
258
|
-
},
|
|
259
|
-
],
|
|
260
|
-
},
|
|
261
|
-
})
|
|
262
|
-
})
|
|
263
|
-
|
|
264
|
-
test("resolveZedSelection returns empty when no workspace matches", async () => {
|
|
265
|
-
await using tmp = await tmpdir()
|
|
266
|
-
const fixture = await writeZedFixture(tmp.path, {
|
|
267
|
-
workspacePaths: JSON.stringify([path.join(path.dirname(tmp.path), "other-workspace")]),
|
|
268
|
-
})
|
|
269
|
-
|
|
270
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({ type: "empty" })
|
|
271
|
-
})
|
|
272
|
-
|
|
273
|
-
test("resolveZedSelection matches a Zed workspace that contains the session directory", async () => {
|
|
274
|
-
await using tmp = await tmpdir()
|
|
275
|
-
const fixture = await writeZedFixture(tmp.path)
|
|
276
|
-
|
|
277
|
-
expect(await resolveZedSelection(fixture.dbPath, path.join(tmp.path, "packages", "app"))).toEqual({
|
|
278
|
-
type: "selection",
|
|
279
|
-
selection: {
|
|
280
|
-
filePath: fixture.filePath,
|
|
281
|
-
source: "zed",
|
|
282
|
-
ranges: [
|
|
283
|
-
{
|
|
284
|
-
text: "two",
|
|
285
|
-
selection: {
|
|
286
|
-
start: { line: 2, character: 1 },
|
|
287
|
-
end: { line: 2, character: 4 },
|
|
288
|
-
},
|
|
289
|
-
},
|
|
290
|
-
],
|
|
291
|
-
},
|
|
292
|
-
})
|
|
293
|
-
})
|
|
294
|
-
|
|
295
|
-
test("resolveZedSelection prefers the most specific containing Zed workspace", async () => {
|
|
296
|
-
await using tmp = await tmpdir()
|
|
297
|
-
const fixture = await writeZedFixture(tmp.path)
|
|
298
|
-
const child = path.join(tmp.path, "packages")
|
|
299
|
-
const childFile = path.join(child, "child.ts")
|
|
300
|
-
await mkdir(child, { recursive: true })
|
|
301
|
-
await Bun.write(childFile, "child")
|
|
302
|
-
|
|
303
|
-
const db = new Database(fixture.dbPath)
|
|
304
|
-
db.run("insert into workspaces values (2, ?, ?)", [JSON.stringify([child]), "2026-01-01"])
|
|
305
|
-
db.run("insert into panes values (2, 2, 1)")
|
|
306
|
-
db.run("insert into items values (2, 2, 2, 1, ?)", ["Editor"])
|
|
307
|
-
db.run("insert into editors values (2, 2, ?, ?)", [childFile, "child"])
|
|
308
|
-
db.run("insert into editor_selections values (2, 2, 0, 5)")
|
|
309
|
-
db.close()
|
|
310
|
-
|
|
311
|
-
expect(await resolveZedSelection(fixture.dbPath, path.join(child, "app"))).toEqual({
|
|
312
|
-
type: "selection",
|
|
313
|
-
selection: {
|
|
314
|
-
filePath: childFile,
|
|
315
|
-
source: "zed",
|
|
316
|
-
ranges: [
|
|
317
|
-
{
|
|
318
|
-
text: "child",
|
|
319
|
-
selection: {
|
|
320
|
-
start: { line: 1, character: 1 },
|
|
321
|
-
end: { line: 1, character: 6 },
|
|
322
|
-
},
|
|
323
|
-
},
|
|
324
|
-
],
|
|
325
|
-
},
|
|
326
|
-
})
|
|
327
|
-
})
|
|
328
|
-
|
|
329
|
-
test("resolveZedSelection ignores a Zed workspace nested inside the session directory", async () => {
|
|
330
|
-
await using tmp = await tmpdir()
|
|
331
|
-
const child = path.join(tmp.path, "effect-lab")
|
|
332
|
-
await mkdir(child, { recursive: true })
|
|
333
|
-
const fixture = await writeZedFixture(child)
|
|
334
|
-
|
|
335
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({ type: "empty" })
|
|
336
|
-
})
|
|
337
|
-
|
|
338
|
-
test("resolveZedSelection returns unavailable when a Zed terminal is active", async () => {
|
|
339
|
-
await using tmp = await tmpdir()
|
|
340
|
-
const fixture = await writeZedFixture(tmp.path, { itemKind: "Terminal", editor: false })
|
|
341
|
-
|
|
342
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({ type: "unavailable" })
|
|
343
|
-
})
|
|
344
|
-
|
|
345
|
-
test("resolveZedSelection returns unavailable when the database cannot be queried", async () => {
|
|
346
|
-
await using tmp = await tmpdir()
|
|
347
|
-
|
|
348
|
-
expect(await resolveZedSelection(path.join(tmp.path, "missing.sqlite"), tmp.path)).toEqual({ type: "unavailable" })
|
|
349
|
-
})
|
|
350
|
-
|
|
351
|
-
test("resolveZedSelection returns unavailable when active selection is missing offsets", async () => {
|
|
352
|
-
await using tmp = await tmpdir()
|
|
353
|
-
const fixture = await writeZedFixture(tmp.path, { selectionStart: null, selectionEnd: null })
|
|
354
|
-
|
|
355
|
-
expect(await resolveZedSelection(fixture.dbPath, tmp.path)).toEqual({ type: "unavailable" })
|
|
356
|
-
})
|
|
@@ -1,228 +0,0 @@
|
|
|
1
|
-
import { mkdir, writeFile } from "node:fs/promises"
|
|
2
|
-
import os from "node:os"
|
|
3
|
-
import path from "node:path"
|
|
4
|
-
import { afterEach, expect, spyOn, test } from "bun:test"
|
|
5
|
-
import { createRoot } from "solid-js"
|
|
6
|
-
import { EditorContextProvider, useEditorContext } from "../../../src/cli/cmd/tui/context/editor"
|
|
7
|
-
import { tmpdir } from "../../fixture/fixture"
|
|
8
|
-
import { FakeWebSocket } from "../../lib/websocket"
|
|
9
|
-
|
|
10
|
-
const originalClaudePort = process.env.CLAUDE_CODE_SSE_PORT
|
|
11
|
-
const originalEditorPort = process.env.SAEEOL_EDITOR_SSE_PORT
|
|
12
|
-
|
|
13
|
-
afterEach(() => {
|
|
14
|
-
process.env.CLAUDE_CODE_SSE_PORT = originalClaudePort
|
|
15
|
-
process.env.SAEEOL_EDITOR_SSE_PORT = originalEditorPort
|
|
16
|
-
})
|
|
17
|
-
|
|
18
|
-
function nextTick() {
|
|
19
|
-
return new Promise<void>((resolve) => queueMicrotask(resolve))
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
function mountEditorContext(WebSocketImpl?: typeof WebSocket) {
|
|
23
|
-
let editor!: ReturnType<typeof useEditorContext>
|
|
24
|
-
let dispose!: () => void
|
|
25
|
-
|
|
26
|
-
createRoot((nextDispose) => {
|
|
27
|
-
dispose = nextDispose
|
|
28
|
-
|
|
29
|
-
const Consumer = () => {
|
|
30
|
-
editor = useEditorContext()
|
|
31
|
-
return null
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
return (
|
|
35
|
-
<EditorContextProvider WebSocketImpl={WebSocketImpl}>
|
|
36
|
-
<Consumer />
|
|
37
|
-
</EditorContextProvider>
|
|
38
|
-
)
|
|
39
|
-
})
|
|
40
|
-
|
|
41
|
-
return {
|
|
42
|
-
editor,
|
|
43
|
-
dispose,
|
|
44
|
-
}
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
function createWebSocketImpl(...sockets: FakeWebSocket[]) {
|
|
48
|
-
let index = 0
|
|
49
|
-
|
|
50
|
-
return class {
|
|
51
|
-
constructor(url: string, options?: { headers?: Record<string, string> }) {
|
|
52
|
-
const socket = sockets[index]
|
|
53
|
-
index += 1
|
|
54
|
-
expect(socket).toBeDefined()
|
|
55
|
-
expect(url).toBe(socket!.url)
|
|
56
|
-
expect(options).toEqual(socket!.options)
|
|
57
|
-
return socket as unknown as object
|
|
58
|
-
}
|
|
59
|
-
} as unknown as typeof WebSocket
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
test("useEditorContext reconnect switches editor server by session directory", async () => {
|
|
63
|
-
await using tmp = await tmpdir()
|
|
64
|
-
const startupDirectory = path.join(tmp.path, "startup")
|
|
65
|
-
const sessionDirectory = path.join(tmp.path, "session")
|
|
66
|
-
const ideDirectory = path.join(tmp.path, ".claude", "ide")
|
|
67
|
-
await mkdir(startupDirectory, { recursive: true })
|
|
68
|
-
await mkdir(sessionDirectory, { recursive: true })
|
|
69
|
-
await mkdir(ideDirectory, { recursive: true })
|
|
70
|
-
await writeFile(
|
|
71
|
-
path.join(ideDirectory, "3001.lock"),
|
|
72
|
-
JSON.stringify({
|
|
73
|
-
transport: "ws",
|
|
74
|
-
workspaceFolders: [startupDirectory],
|
|
75
|
-
}),
|
|
76
|
-
)
|
|
77
|
-
await writeFile(
|
|
78
|
-
path.join(ideDirectory, "3002.lock"),
|
|
79
|
-
JSON.stringify({
|
|
80
|
-
transport: "ws",
|
|
81
|
-
workspaceFolders: [sessionDirectory],
|
|
82
|
-
}),
|
|
83
|
-
)
|
|
84
|
-
|
|
85
|
-
process.env.CLAUDE_CODE_SSE_PORT = undefined
|
|
86
|
-
process.env.SAEEOL_EDITOR_SSE_PORT = undefined
|
|
87
|
-
spyOn(process, "cwd").mockImplementation(() => startupDirectory)
|
|
88
|
-
spyOn(os, "homedir").mockImplementation(() => tmp.path)
|
|
89
|
-
const firstSocket = new FakeWebSocket("ws://127.0.0.1:3001")
|
|
90
|
-
const secondSocket = new FakeWebSocket("ws://127.0.0.1:3002")
|
|
91
|
-
|
|
92
|
-
const mounted = mountEditorContext(createWebSocketImpl(firstSocket, secondSocket))
|
|
93
|
-
await nextTick()
|
|
94
|
-
|
|
95
|
-
expect(firstSocket.closed).toBeFalse()
|
|
96
|
-
|
|
97
|
-
mounted.editor.reconnect(sessionDirectory)
|
|
98
|
-
await nextTick()
|
|
99
|
-
|
|
100
|
-
expect(firstSocket.closed).toBeTrue()
|
|
101
|
-
expect(secondSocket.closed).toBeFalse()
|
|
102
|
-
|
|
103
|
-
mounted.dispose()
|
|
104
|
-
})
|
|
105
|
-
|
|
106
|
-
test("useEditorContext favors configured port over lock files", async () => {
|
|
107
|
-
await using tmp = await tmpdir()
|
|
108
|
-
const startupDirectory = path.join(tmp.path, "startup")
|
|
109
|
-
const ideDirectory = path.join(tmp.path, ".claude", "ide")
|
|
110
|
-
await mkdir(startupDirectory, { recursive: true })
|
|
111
|
-
await mkdir(ideDirectory, { recursive: true })
|
|
112
|
-
await writeFile(
|
|
113
|
-
path.join(ideDirectory, "3001.lock"),
|
|
114
|
-
JSON.stringify({
|
|
115
|
-
transport: "ws",
|
|
116
|
-
workspaceFolders: [startupDirectory],
|
|
117
|
-
}),
|
|
118
|
-
)
|
|
119
|
-
|
|
120
|
-
process.env.CLAUDE_CODE_SSE_PORT = "4010"
|
|
121
|
-
process.env.SAEEOL_EDITOR_SSE_PORT = undefined
|
|
122
|
-
spyOn(process, "cwd").mockImplementation(() => startupDirectory)
|
|
123
|
-
spyOn(os, "homedir").mockImplementation(() => tmp.path)
|
|
124
|
-
const socket = new FakeWebSocket("ws://127.0.0.1:4010")
|
|
125
|
-
|
|
126
|
-
const mounted = mountEditorContext(createWebSocketImpl(socket))
|
|
127
|
-
await nextTick()
|
|
128
|
-
|
|
129
|
-
expect(socket.closed).toBeFalse()
|
|
130
|
-
|
|
131
|
-
mounted.dispose()
|
|
132
|
-
})
|
|
133
|
-
|
|
134
|
-
test("useEditorContext resets selection when reconnecting", async () => {
|
|
135
|
-
await using tmp = await tmpdir()
|
|
136
|
-
const startupDirectory = path.join(tmp.path, "startup")
|
|
137
|
-
const ideDirectory = path.join(tmp.path, ".claude", "ide")
|
|
138
|
-
await mkdir(startupDirectory, { recursive: true })
|
|
139
|
-
await mkdir(ideDirectory, { recursive: true })
|
|
140
|
-
await writeFile(
|
|
141
|
-
path.join(ideDirectory, "3001.lock"),
|
|
142
|
-
JSON.stringify({
|
|
143
|
-
transport: "ws",
|
|
144
|
-
workspaceFolders: [startupDirectory],
|
|
145
|
-
}),
|
|
146
|
-
)
|
|
147
|
-
|
|
148
|
-
process.env.CLAUDE_CODE_SSE_PORT = undefined
|
|
149
|
-
process.env.SAEEOL_EDITOR_SSE_PORT = undefined
|
|
150
|
-
spyOn(process, "cwd").mockImplementation(() => startupDirectory)
|
|
151
|
-
spyOn(os, "homedir").mockImplementation(() => tmp.path)
|
|
152
|
-
const socket = new FakeWebSocket("ws://127.0.0.1:3001")
|
|
153
|
-
|
|
154
|
-
const mounted = mountEditorContext(createWebSocketImpl(socket))
|
|
155
|
-
await nextTick()
|
|
156
|
-
|
|
157
|
-
expect(socket.closed).toBeFalse()
|
|
158
|
-
expect(mounted.editor.selection()).toBeUndefined()
|
|
159
|
-
expect(mounted.editor.connected()).toBeFalse()
|
|
160
|
-
|
|
161
|
-
socket.open()
|
|
162
|
-
socket.message(
|
|
163
|
-
JSON.stringify({
|
|
164
|
-
jsonrpc: "2.0",
|
|
165
|
-
id: 1,
|
|
166
|
-
result: {
|
|
167
|
-
protocolVersion: "2025-11-25",
|
|
168
|
-
serverInfo: { name: "test", version: "0.0.0" },
|
|
169
|
-
},
|
|
170
|
-
}),
|
|
171
|
-
)
|
|
172
|
-
socket.message(
|
|
173
|
-
JSON.stringify({
|
|
174
|
-
jsonrpc: "2.0",
|
|
175
|
-
method: "selection_changed",
|
|
176
|
-
params: {
|
|
177
|
-
text: "foo",
|
|
178
|
-
filePath: path.join(startupDirectory, "file.ts"),
|
|
179
|
-
selection: {
|
|
180
|
-
start: { line: 1, character: 1 },
|
|
181
|
-
end: { line: 1, character: 4 },
|
|
182
|
-
},
|
|
183
|
-
},
|
|
184
|
-
}),
|
|
185
|
-
)
|
|
186
|
-
|
|
187
|
-
expect(mounted.editor.connected()).toBeTrue()
|
|
188
|
-
expect(mounted.editor.server()).toEqual({
|
|
189
|
-
protocolVersion: "2025-11-25",
|
|
190
|
-
serverInfo: { name: "test", version: "0.0.0" },
|
|
191
|
-
})
|
|
192
|
-
expect(mounted.editor.selection()).toEqual({
|
|
193
|
-
filePath: path.join(startupDirectory, "file.ts"),
|
|
194
|
-
source: "websocket",
|
|
195
|
-
ranges: [
|
|
196
|
-
{
|
|
197
|
-
text: "foo",
|
|
198
|
-
selection: {
|
|
199
|
-
start: { line: 1, character: 1 },
|
|
200
|
-
end: { line: 1, character: 4 },
|
|
201
|
-
},
|
|
202
|
-
},
|
|
203
|
-
],
|
|
204
|
-
})
|
|
205
|
-
|
|
206
|
-
mounted.editor.reconnect(startupDirectory)
|
|
207
|
-
|
|
208
|
-
expect(socket.closed).toBeFalse()
|
|
209
|
-
expect(mounted.editor.connected()).toBeTrue()
|
|
210
|
-
expect(mounted.editor.selection()).toBeUndefined()
|
|
211
|
-
|
|
212
|
-
mounted.dispose()
|
|
213
|
-
})
|
|
214
|
-
|
|
215
|
-
test("useEditorContext connects with SAEEOL_EDITOR_SSE_PORT", async () => {
|
|
216
|
-
await using tmp = await tmpdir()
|
|
217
|
-
process.env.CLAUDE_CODE_SSE_PORT = undefined
|
|
218
|
-
process.env.SAEEOL_EDITOR_SSE_PORT = "4020"
|
|
219
|
-
spyOn(process, "cwd").mockImplementation(() => tmp.path)
|
|
220
|
-
const socket = new FakeWebSocket("ws://127.0.0.1:4020")
|
|
221
|
-
|
|
222
|
-
const mounted = mountEditorContext(createWebSocketImpl(socket))
|
|
223
|
-
await nextTick()
|
|
224
|
-
|
|
225
|
-
expect(socket.closed).toBeFalse()
|
|
226
|
-
|
|
227
|
-
mounted.dispose()
|
|
228
|
-
})
|
|
@@ -1,90 +0,0 @@
|
|
|
1
|
-
import { describe, expect, test } from "bun:test"
|
|
2
|
-
import type { ParsedKey } from "@opentui/core"
|
|
3
|
-
import { createPluginKeybind } from "../../../src/cli/cmd/tui/context/plugin-keybinds"
|
|
4
|
-
|
|
5
|
-
describe("createPluginKeybind", () => {
|
|
6
|
-
const defaults = {
|
|
7
|
-
open: "ctrl+o",
|
|
8
|
-
close: "escape",
|
|
9
|
-
}
|
|
10
|
-
|
|
11
|
-
test("uses defaults when overrides are missing", () => {
|
|
12
|
-
const api = {
|
|
13
|
-
match: () => false,
|
|
14
|
-
print: (key: string) => key,
|
|
15
|
-
}
|
|
16
|
-
const bind = createPluginKeybind(api, defaults)
|
|
17
|
-
|
|
18
|
-
expect(bind.all).toEqual(defaults)
|
|
19
|
-
expect(bind.get("open")).toBe("ctrl+o")
|
|
20
|
-
expect(bind.get("close")).toBe("escape")
|
|
21
|
-
})
|
|
22
|
-
|
|
23
|
-
test("applies valid overrides", () => {
|
|
24
|
-
const api = {
|
|
25
|
-
match: () => false,
|
|
26
|
-
print: (key: string) => key,
|
|
27
|
-
}
|
|
28
|
-
const bind = createPluginKeybind(api, defaults, {
|
|
29
|
-
open: "ctrl+alt+o",
|
|
30
|
-
close: "q",
|
|
31
|
-
})
|
|
32
|
-
|
|
33
|
-
expect(bind.all).toEqual({
|
|
34
|
-
open: "ctrl+alt+o",
|
|
35
|
-
close: "q",
|
|
36
|
-
})
|
|
37
|
-
})
|
|
38
|
-
|
|
39
|
-
test("ignores invalid overrides", () => {
|
|
40
|
-
const api = {
|
|
41
|
-
match: () => false,
|
|
42
|
-
print: (key: string) => key,
|
|
43
|
-
}
|
|
44
|
-
const bind = createPluginKeybind(api, defaults, {
|
|
45
|
-
open: " ",
|
|
46
|
-
close: 1,
|
|
47
|
-
extra: "ctrl+x",
|
|
48
|
-
})
|
|
49
|
-
|
|
50
|
-
expect(bind.all).toEqual(defaults)
|
|
51
|
-
expect(bind.get("extra")).toBe("extra")
|
|
52
|
-
})
|
|
53
|
-
|
|
54
|
-
test("resolves names for match", () => {
|
|
55
|
-
const list: string[] = []
|
|
56
|
-
const api = {
|
|
57
|
-
match: (key: string) => {
|
|
58
|
-
list.push(key)
|
|
59
|
-
return true
|
|
60
|
-
},
|
|
61
|
-
print: (key: string) => key,
|
|
62
|
-
}
|
|
63
|
-
const bind = createPluginKeybind(api, defaults, {
|
|
64
|
-
open: "ctrl+shift+o",
|
|
65
|
-
})
|
|
66
|
-
|
|
67
|
-
bind.match("open", { name: "x" } as ParsedKey)
|
|
68
|
-
bind.match("ctrl+k", { name: "x" } as ParsedKey)
|
|
69
|
-
|
|
70
|
-
expect(list).toEqual(["ctrl+shift+o", "ctrl+k"])
|
|
71
|
-
})
|
|
72
|
-
|
|
73
|
-
test("resolves names for print", () => {
|
|
74
|
-
const list: string[] = []
|
|
75
|
-
const api = {
|
|
76
|
-
match: () => false,
|
|
77
|
-
print: (key: string) => {
|
|
78
|
-
list.push(key)
|
|
79
|
-
return `print:${key}`
|
|
80
|
-
},
|
|
81
|
-
}
|
|
82
|
-
const bind = createPluginKeybind(api, defaults, {
|
|
83
|
-
close: "q",
|
|
84
|
-
})
|
|
85
|
-
|
|
86
|
-
expect(bind.print("close")).toBe("print:q")
|
|
87
|
-
expect(bind.print("ctrl+p")).toBe("print:ctrl+p")
|
|
88
|
-
expect(list).toEqual(["q", "ctrl+p"])
|
|
89
|
-
})
|
|
90
|
-
})
|