saeeol 1.3.0 → 1.4.0
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/AGENTS.md +72 -0
- package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
- package/Dockerfile +18 -0
- package/assets/saeeol.ico +0 -0
- package/bin/saeeol.cjs +3 -1
- package/bunfig.toml +7 -0
- package/database.db +0 -0
- package/drizzle.config.ts +10 -0
- package/git +0 -0
- package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
- package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
- package/migration/20260211171708_add_project_commands/migration.sql +1 -0
- package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
- package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
- package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
- package/migration/20260225215848_workspace/migration.sql +7 -0
- package/migration/20260225215848_workspace/snapshot.json +959 -0
- package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
- package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
- package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
- package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
- package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
- package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
- package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
- package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
- package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
- package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
- package/migration/20260323234822_events/migration.sql +13 -0
- package/migration/20260323234822_events/snapshot.json +1271 -0
- package/migration/20260410174513_workspace-name/migration.sql +16 -0
- package/migration/20260410174513_workspace-name/snapshot.json +1271 -0
- package/migration/20260413175956_chief_energizer/migration.sql +13 -0
- package/migration/20260413175956_chief_energizer/snapshot.json +1399 -0
- package/migration/20260423070820_add_icon_url_override/migration.sql +2 -0
- package/migration/20260423070820_add_icon_url_override/snapshot.json +1409 -0
- package/migration/20260428004200_add_session_path/migration.sql +1 -0
- package/migration/20260428004200_add_session_path/snapshot.json +1419 -0
- package/npm/bin/saeeol +42 -0
- package/npm/package.json +39 -0
- package/npm/postinstall.js +162 -0
- package/package.json +201 -207
- package/parsers-config.ts +289 -0
- package/script/build.ts +393 -0
- package/script/check-migrations.ts +16 -0
- package/script/fix-node-pty.ts +34 -0
- package/script/generate.ts +23 -0
- package/script/postinstall.mjs +189 -0
- package/script/publish.ts +200 -0
- package/script/run-workspace-server +106 -0
- package/script/schema.ts +63 -0
- package/script/test-runner.ts +420 -0
- package/script/time.ts +6 -0
- package/script/trace-imports.ts +153 -0
- package/script/upgrade-opentui.ts +64 -0
- package/scripts/diff-sdk-types.sh +52 -0
- package/specs/effect/facades.md +221 -0
- package/specs/effect/http-api.md +401 -0
- package/specs/effect/instance-context.md +309 -0
- package/specs/effect/loose-ends.md +34 -0
- package/specs/effect/migration.md +299 -0
- package/specs/effect/routes.md +64 -0
- package/specs/effect/schema.md +399 -0
- package/specs/effect/server-package.md +668 -0
- package/specs/effect/tools.md +90 -0
- package/specs/tui-plugins.md +433 -0
- package/specs/v2/api.ts +67 -0
- package/specs/v2/keymappings.md +10 -0
- package/specs/v2/message-shape.md +136 -0
- package/src/acp/agent-message.ts +1 -1
- package/src/acp/agent-utils.ts +1 -1
- package/src/boxes/ansi.ts +17 -0
- package/src/boxes/atomic-write.ts +35 -0
- package/src/boxes/b64.ts +58 -0
- package/src/boxes/bash-security.ts +129 -0
- package/src/boxes/bom.ts +18 -0
- package/src/boxes/cancel.ts +16 -0
- package/src/boxes/chop.ts +12 -0
- package/src/boxes/clamp.ts +3 -0
- package/src/boxes/compact.ts +9 -0
- package/src/boxes/cost-tracker.ts +116 -0
- package/src/boxes/dataurl.ts +29 -0
- package/src/boxes/delay.ts +27 -0
- package/src/boxes/diff-apply.ts +53 -0
- package/src/boxes/disposable.ts +13 -0
- package/src/boxes/err.ts +34 -0
- package/src/boxes/human.ts +47 -0
- package/src/boxes/iife.ts +9 -0
- package/src/boxes/latch.ts +8 -0
- package/src/boxes/memory.ts +198 -0
- package/src/boxes/net.ts +16 -0
- package/src/boxes/plural.ts +4 -0
- package/src/boxes/puny.ts +21 -0
- package/src/boxes/retry.ts +49 -0
- package/src/boxes/rwlock.ts +41 -0
- package/src/boxes/schedule.ts +71 -0
- package/src/boxes/scope.ts +21 -0
- package/src/boxes/tokens.ts +9 -0
- package/src/boxes/ttl-cache.ts +63 -0
- package/src/boxes/typed-event.ts +51 -0
- package/src/boxes/uid.ts +50 -0
- package/src/boxes/wave6.test.ts +296 -0
- package/src/boxes/wildcard.ts +58 -0
- package/src/bus/global.ts +1 -1
- package/src/cli/cmd/github-run-api.ts +2 -2
- package/src/cli/cmd/run-events.ts +2 -2
- package/src/cli/cmd/tui/component/logo.tsx +1 -1
- package/src/cli/cmd/tui/component/prompt/use-prompt-memos.ts +2 -2
- package/src/cli/cmd/tui/context/app/editor-zed.ts +1 -1
- package/src/cli/cmd/tui/context/app/editor.ts +1 -1
- package/src/cli/cmd/tui/context/app/theme.tsx +1 -1
- package/src/cli/cmd/tui/preflight.ts +138 -0
- package/src/cli/cmd/tui/thread.ts +20 -0
- package/src/cli/cmd/tui/util/revert-diff.ts +1 -1
- package/src/overlay/cli/cmd/roll-call-call.ts +1 -1
- package/src/overlay/cost-tracker/format.ts +1 -1
- package/src/overlay/cost-tracker/index.ts +4 -4
- package/src/overlay/cost-tracker/state.ts +2 -2
- package/src/overlay/cost-tracker/types.ts +2 -2
- package/src/overlay/memory/age.ts +1 -1
- package/src/overlay/memory/index.ts +4 -4
- package/src/overlay/memory/paths.ts +2 -2
- package/src/overlay/memory/scan.ts +1 -1
- package/src/overlay/memory/types.ts +2 -2
- package/src/overlay/tool/bash-security.ts +3 -3
- package/src/overlay/util/url.ts +1 -1
- package/src/plugin/codex-auth.ts +1 -1
- package/src/provider/model-cache.ts +2 -2
- package/src/provider/provider-resolve.ts +3 -3
- package/src/provider/transform-message.ts +1 -1
- package/src/server/routes/game.ts +284 -0
- package/src/server/server.ts +2 -0
- package/src/session/core/compaction/compaction-helpers.ts +1 -1
- package/src/session/core/compaction/compaction.ts +1 -1
- package/src/session/core/session.ts +2 -0
- package/src/sessions/ingest-queue.ts +2 -2
- package/src/sessions/remote-ws.ts +1 -1
- package/src/tool/workflow/question.ts +1 -1
- package/src/util/abort.ts +1 -1
- package/src/util/bom.ts +2 -2
- package/src/util/color.ts +1 -1
- package/src/util/data-url.ts +1 -1
- package/src/util/defer.ts +1 -1
- package/src/util/error.ts +2 -2
- package/src/util/filesystem.ts +2 -2
- package/src/util/format.ts +1 -1
- package/src/util/iife.ts +1 -1
- package/src/util/local-context.ts +1 -1
- package/src/util/locale.ts +2 -2
- package/src/util/lock.ts +1 -1
- package/src/util/network.ts +1 -1
- package/src/util/signal.ts +1 -1
- package/src/util/token.ts +1 -1
- package/src/util/wildcard.ts +1 -1
- package/sst-env.d.ts +10 -0
- package/test/AGENTS.md +133 -0
- package/test/account/repo.test.ts +352 -0
- package/test/account/service.test.ts +456 -0
- package/test/acp/agent-interface.test.ts +51 -0
- package/test/acp/event-subscription.test.ts +725 -0
- package/test/agent/agent.test.ts +890 -0
- package/test/auth/auth.test.ts +86 -0
- package/test/bun/registry.test.ts +75 -0
- package/test/bus/bus-effect.test.ts +161 -0
- package/test/bus/bus-integration.test.ts +87 -0
- package/test/bus/bus.test.ts +219 -0
- package/test/cli/account.test.ts +26 -0
- package/test/cli/auto-mode.test.ts +75 -0
- package/test/cli/bin-saeeol.test.ts +8 -0
- package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
- package/test/cli/cmd/tui/prompt-traits.test.ts +38 -0
- package/test/cli/cmd/tui/sync.test.tsx +159 -0
- package/test/cli/error.test.ts +18 -0
- package/test/cli/github-action.test.ts +198 -0
- package/test/cli/github-remote.test.ts +85 -0
- package/test/cli/import.test.ts +97 -0
- package/test/cli/install-artifact.test.ts +72 -0
- package/test/cli/plugin-auth-picker.test.ts +120 -0
- package/test/cli/pr.test.ts +59 -0
- package/test/cli/tui/editor-context-zed.test.ts +356 -0
- package/test/cli/tui/editor-context.test.tsx +228 -0
- package/test/cli/tui/keybind-plugin.test.ts +90 -0
- package/test/cli/tui/markdown.test.ts +161 -0
- package/test/cli/tui/plugin-add.test.ts +111 -0
- package/test/cli/tui/plugin-install.test.ts +87 -0
- package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
- package/test/cli/tui/plugin-loader-entrypoint.test.ts +484 -0
- package/test/cli/tui/plugin-loader-pure.test.ts +71 -0
- package/test/cli/tui/plugin-loader.test.ts +816 -0
- package/test/cli/tui/plugin-toggle.test.ts +157 -0
- package/test/cli/tui/revert-diff.test.ts +35 -0
- package/test/cli/tui/slot-replace.test.tsx +47 -0
- package/test/cli/tui/theme-store.test.ts +54 -0
- package/test/cli/tui/thread.test.ts +28 -0
- package/test/cli/tui/transcript.test.ts +426 -0
- package/test/cli/tui/usage.test.ts +60 -0
- package/test/cli/tui/use-event.test.tsx +175 -0
- package/test/config/agent-color.test.ts +67 -0
- package/test/config/config.test.ts +2544 -0
- package/test/config/fixtures/empty-frontmatter.md +4 -0
- package/test/config/fixtures/frontmatter.md +28 -0
- package/test/config/fixtures/markdown-header.md +11 -0
- package/test/config/fixtures/no-frontmatter.md +1 -0
- package/test/config/fixtures/weird-model-id.md +13 -0
- package/test/config/lsp.test.ts +87 -0
- package/test/config/markdown.test.ts +228 -0
- package/test/config/plugin.test.ts +0 -0
- package/test/config/tui.test.ts +624 -0
- package/test/control-plane/adapters.test.ts +71 -0
- package/test/control-plane/workspace.test.ts +1526 -0
- package/test/effect/app-runtime-logger.test.ts +98 -0
- package/test/effect/config-service.test.ts +65 -0
- package/test/effect/instance-state.test.ts +394 -0
- package/test/effect/run-service.test.ts +89 -0
- package/test/effect/runner.test.ts +523 -0
- package/test/fake/provider.ts +82 -0
- package/test/file/fsmonitor.test.ts +68 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/file/index.test.ts +954 -0
- package/test/file/path-traversal.test.ts +205 -0
- package/test/file/ripgrep.test.ts +226 -0
- package/test/file/watcher.test.ts +249 -0
- package/test/filesystem/filesystem.test.ts +319 -0
- package/test/fixture/db.ts +11 -0
- package/test/fixture/fixture.test.ts +26 -0
- package/test/fixture/fixture.ts +175 -0
- package/test/fixture/flock-worker.ts +72 -0
- package/test/fixture/log-init-worker.ts +62 -0
- package/test/fixture/lsp/fake-lsp-server.js +249 -0
- package/test/fixture/plug-worker.ts +93 -0
- package/test/fixture/plugin-meta-worker.ts +19 -0
- package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
- package/test/fixture/skills/cloudflare/SKILL.md +211 -0
- package/test/fixture/skills/index.json +6 -0
- package/test/fixture/tui-plugin.ts +323 -0
- package/test/fixture/tui-runtime.ts +31 -0
- package/test/format/format.test.ts +272 -0
- package/test/git/git.test.ts +128 -0
- package/test/ide/ide.test.ts +82 -0
- package/test/installation/installation.test.ts +168 -0
- package/test/keybind.test.ts +421 -0
- package/test/lib/effect.ts +53 -0
- package/test/lib/filesystem.ts +10 -0
- package/test/lib/llm-server.ts +778 -0
- package/test/lib/websocket.ts +46 -0
- package/test/lsp/client.test.ts +482 -0
- package/test/lsp/index.test.ts +160 -0
- package/test/lsp/launch.test.ts +22 -0
- package/test/lsp/lifecycle.test.ts +184 -0
- package/test/ltm/ltm.test.ts +230 -0
- package/test/mcp/headers.test.ts +178 -0
- package/test/mcp/lifecycle.test.ts +787 -0
- package/test/mcp/oauth-auto-connect.test.ts +311 -0
- package/test/mcp/oauth-browser.test.ts +276 -0
- package/test/mcp/oauth-callback.test.ts +34 -0
- package/test/memory/abort-leak-webfetch.ts +49 -0
- package/test/memory/abort-leak.test.ts +128 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/next.test.ts +1227 -0
- package/test/permission/next.toConfig.test.ts +110 -0
- package/test/permission-task.test.ts +326 -0
- package/test/plugin/auth-override.test.ts +79 -0
- package/test/plugin/cloudflare.test.ts +68 -0
- package/test/plugin/codex.test.ts +123 -0
- package/test/plugin/github-copilot-models.test.ts +261 -0
- package/test/plugin/install-concurrency.test.ts +140 -0
- package/test/plugin/install.test.ts +570 -0
- package/test/plugin/loader-shared.test.ts +1169 -0
- package/test/plugin/meta.test.ts +137 -0
- package/test/plugin/plugin-contract.test.ts +291 -0
- package/test/plugin/shared.test.ts +88 -0
- package/test/plugin/trigger.test.ts +102 -0
- package/test/plugin/workspace-adapter.test.ts +109 -0
- package/test/preload.ts +77 -0
- package/test/project/instance.test.ts +276 -0
- package/test/project/migrate-global.test.ts +152 -0
- package/test/project/project.test.ts +600 -0
- package/test/project/vcs.test.ts +286 -0
- package/test/project/worktree-remove.test.ts +126 -0
- package/test/project/worktree.test.ts +223 -0
- package/test/provider/amazon-bedrock.test.ts +462 -0
- package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
- package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
- package/test/provider/gitlab-duo.test.ts +413 -0
- package/test/provider/local.test.ts +208 -0
- package/test/provider/models.test.ts +261 -0
- package/test/provider/provider-category.test.ts +190 -0
- package/test/provider/provider.test.ts +2758 -0
- package/test/provider/transform.test.ts +3681 -0
- package/test/pty/pty-output-isolation.test.ts +147 -0
- package/test/pty/pty-session.test.ts +102 -0
- package/test/pty/pty-shell.test.ts +104 -0
- package/test/question/question.test.ts +490 -0
- package/test/saeeol/agent-global-config-dirs.test.ts +24 -0
- package/test/saeeol/agent-manager-tool.test.ts +71 -0
- package/test/saeeol/agent-permission-overrides.test.ts +75 -0
- package/test/saeeol/agent-skill-permissions.test.ts +37 -0
- package/test/saeeol/ask-agent-permissions.test.ts +303 -0
- package/test/saeeol/bash-hierarchy.test.ts +64 -0
- package/test/saeeol/bash-permission-metadata.test.ts +66 -0
- package/test/saeeol/bash-security-extended.test.ts +243 -0
- package/test/saeeol/bedrock-claude-empty-content.test.ts +138 -0
- package/test/saeeol/boxes-integration.test.ts +415 -0
- package/test/saeeol/builtin-skills.test.ts +75 -0
- package/test/saeeol/cleanup.ts +28 -0
- package/test/saeeol/cli/dev-setup.test.ts +74 -0
- package/test/saeeol/cli/roll-call.test.ts +161 -0
- package/test/saeeol/cli-run-auto-helper.test.ts +58 -0
- package/test/saeeol/codex-auth-refresh.test.ts +124 -0
- package/test/saeeol/commit-message/generate.test.ts +188 -0
- package/test/saeeol/commit-message/git-context.test.ts +303 -0
- package/test/saeeol/commit-message-windows.test.ts +38 -0
- package/test/saeeol/compaction-payload-recovery.test.ts +406 -0
- package/test/saeeol/compaction-preservation-audit.test.ts +122 -0
- package/test/saeeol/compaction-skip-guard.test.ts +224 -0
- package/test/saeeol/compaction-smart-select.test.ts +100 -0
- package/test/saeeol/config/config.test.ts +166 -0
- package/test/saeeol/config/indexing-default-plugin.test.ts +82 -0
- package/test/saeeol/config/opentelemetry-default.test.ts +29 -0
- package/test/saeeol/config-gitignore.test.ts +70 -0
- package/test/saeeol/config-injector.test.ts +305 -0
- package/test/saeeol/config-resilience.test.ts +234 -0
- package/test/saeeol/config-validation.test.ts +183 -0
- package/test/saeeol/cost-propagation.test.ts +94 -0
- package/test/saeeol/cost-tracker-extended.test.ts +141 -0
- package/test/saeeol/cost-tracker.test.ts +64 -0
- package/test/saeeol/custom-provider-delete.test.ts +149 -0
- package/test/saeeol/diff-full.test.ts +226 -0
- package/test/saeeol/edit-permission-filediff.test.ts +223 -0
- package/test/saeeol/encoding.test.ts +364 -0
- package/test/saeeol/enhance-prompt.test.ts +61 -0
- package/test/saeeol/ensure-plan-dir.test.ts +32 -0
- package/test/saeeol/errors.test.ts +144 -0
- package/test/saeeol/external-directory-boundary.test.ts +96 -0
- package/test/saeeol/gateway-headers.test.ts +88 -0
- package/test/saeeol/help.test.ts +191 -0
- package/test/saeeol/ignore-migrator.test.ts +308 -0
- package/test/saeeol/indexing-auth.test.ts +45 -0
- package/test/saeeol/indexing-feature.test.ts +44 -0
- package/test/saeeol/indexing-label.test.ts +70 -0
- package/test/saeeol/indexing-startup.test.ts +381 -0
- package/test/saeeol/indexing-worktree.test.ts +73 -0
- package/test/saeeol/instruction.test.ts +136 -0
- package/test/saeeol/lancedb-runtime.test.ts +116 -0
- package/test/saeeol/loader-auth.test.ts +168 -0
- package/test/saeeol/local-model.test.ts +621 -0
- package/test/saeeol/logo.test.ts +31 -0
- package/test/saeeol/lsp-typescript-lightweight.test.ts +89 -0
- package/test/saeeol/mcp-branding.test.ts +33 -0
- package/test/saeeol/mcp-docker-rm.test.ts +32 -0
- package/test/saeeol/mcp-migrator.test.ts +736 -0
- package/test/saeeol/mcp-oauth-callback.test.ts +33 -0
- package/test/saeeol/memory-io.test.ts +198 -0
- package/test/saeeol/memory-paths.test.ts +87 -0
- package/test/saeeol/memory-security.test.ts +166 -0
- package/test/saeeol/model-cache-org.test.ts +164 -0
- package/test/saeeol/model-info-panel-utils.test.ts +52 -0
- package/test/saeeol/model-info-panel.types.test.ts +7 -0
- package/test/saeeol/models-401-fallback.test.ts +52 -0
- package/test/saeeol/modes-migrator.test.ts +320 -0
- package/test/saeeol/nvidia-headers.test.ts +74 -0
- package/test/saeeol/patch-jsonc.test.ts +73 -0
- package/test/saeeol/patch.test.ts +172 -0
- package/test/saeeol/paths.test.ts +265 -0
- package/test/saeeol/permission/config-paths.test.ts +174 -0
- package/test/saeeol/permission/env-read.test.ts +149 -0
- package/test/saeeol/permission/external-directory-allow.test.ts +327 -0
- package/test/saeeol/permission/next.always-rules.test.ts +882 -0
- package/test/saeeol/permission/next.reply-http.test.ts +205 -0
- package/test/saeeol/permission/next.reply-routing.test.ts +184 -0
- package/test/saeeol/plan-exit-detection.test.ts +494 -0
- package/test/saeeol/plan-followup.test.ts +1376 -0
- package/test/saeeol/project-config-update.test.ts +120 -0
- package/test/saeeol/project-id.test.ts +455 -0
- package/test/saeeol/provider-cost.test.ts +171 -0
- package/test/saeeol/provider-list-failed-state.test.ts +100 -0
- package/test/saeeol/question-dismiss-all.test.ts +174 -0
- package/test/saeeol/read-directory.test.ts +116 -0
- package/test/saeeol/rules-migrator.test.ts +257 -0
- package/test/saeeol/run-auto.test.ts +176 -0
- package/test/saeeol/run-network.test.ts +224 -0
- package/test/saeeol/semantic-search.test.ts +186 -0
- package/test/saeeol/server/permission-allow-everything.test.ts +125 -0
- package/test/saeeol/session/instruction-substitution.test.ts +72 -0
- package/test/saeeol/session/platform-attribution.test.ts +118 -0
- package/test/saeeol/session/session.test.ts +105 -0
- package/test/saeeol/session-compaction-cap.test.ts +399 -0
- package/test/saeeol/session-compaction-chunks.test.ts +501 -0
- package/test/saeeol/session-compaction-safety.test.ts +481 -0
- package/test/saeeol/session-fork-remap.test.ts +251 -0
- package/test/saeeol/session-import-service.test.ts +114 -0
- package/test/saeeol/session-list.test.ts +47 -0
- package/test/saeeol/session-message-metadata.test.ts +128 -0
- package/test/saeeol/session-overflow.test.ts +78 -0
- package/test/saeeol/session-processor-empty-tool-calls.test.ts +571 -0
- package/test/saeeol/session-processor-network-offline.test.ts +204 -0
- package/test/saeeol/session-processor-retry-limit.test.ts +238 -0
- package/test/saeeol/session-processor-review-telemetry.test.ts +82 -0
- package/test/saeeol/session-prompt-compaction-safety.test.ts +517 -0
- package/test/saeeol/session-prompt-queue.test.ts +815 -0
- package/test/saeeol/sessions/inflight-cache.test.ts +157 -0
- package/test/saeeol/sessions/ingest-queue.test.ts +402 -0
- package/test/saeeol/sessions/remote-protocol.test.ts +258 -0
- package/test/saeeol/sessions/remote-sender.test.ts +1036 -0
- package/test/saeeol/sessions/remote-ws.test.ts +367 -0
- package/test/saeeol/sessions/sessions-enable-remote.test.disable +181 -0
- package/test/saeeol/slot-prop-reactivity.test.ts +142 -0
- package/test/saeeol/snapshot-cache.test.ts +84 -0
- package/test/saeeol/snapshot-freeze-repro.test.ts +100 -0
- package/test/saeeol/snapshot-track-timeout.test.ts +519 -0
- package/test/saeeol/stats-subagent-cost.test.ts +123 -0
- package/test/saeeol/suggestion/auto-dismiss.test.ts +65 -0
- package/test/saeeol/suggestion/suggestion.test.ts +145 -0
- package/test/saeeol/suggestion/tool.test.ts +298 -0
- package/test/saeeol/summary-file-diff.test.ts +28 -0
- package/test/saeeol/system-prompt.test.ts +142 -0
- package/test/saeeol/task-nesting.test.ts +193 -0
- package/test/saeeol/telemetry/feedback.test.ts +8 -0
- package/test/saeeol/todo-view.test.ts +57 -0
- package/test/saeeol/tool-encoding.test.ts +455 -0
- package/test/saeeol/tool-registry-indexing-import-failure.test.ts +49 -0
- package/test/saeeol/tool-registry-indexing.test.ts +236 -0
- package/test/saeeol/tool-registry-semantic-import-failure.test.ts +55 -0
- package/test/saeeol/tool-task-model.test.ts +352 -0
- package/test/saeeol/transform-opus-4.7.test.ts +89 -0
- package/test/saeeol/tui-diff.test.ts +91 -0
- package/test/saeeol/tui-sync.test.ts +80 -0
- package/test/saeeol/util/url.test.ts +141 -0
- package/test/saeeol/workflows-migrator.test.ts +261 -0
- package/test/saeeol/worktree-diff-summary.test.ts +64 -0
- package/test/saeeol/worktree-diff.test.ts +223 -0
- package/test/saeeol/worktree-remove-lock.test.ts +82 -0
- package/test/server/AGENTS.md +15 -0
- package/test/server/contract.test.ts +357 -0
- package/test/server/experimental-session-list.test.ts +157 -0
- package/test/server/global-session-list.test.ts +155 -0
- package/test/server/httpapi-authorization.test.ts +103 -0
- package/test/server/httpapi-bridge.test.ts +440 -0
- package/test/server/httpapi-config.test.ts +67 -0
- package/test/server/httpapi-cors.test.ts +89 -0
- package/test/server/httpapi-event.test.ts +57 -0
- package/test/server/httpapi-experimental.test.ts +219 -0
- package/test/server/httpapi-file.test.ts +79 -0
- package/test/server/httpapi-instance-context.test.ts +237 -0
- package/test/server/httpapi-instance.legacy.test.ts +140 -0
- package/test/server/httpapi-instance.test.ts +83 -0
- package/test/server/httpapi-json-parity.test.ts +263 -0
- package/test/server/httpapi-mcp-oauth.test.ts +76 -0
- package/test/server/httpapi-mcp.test.ts +189 -0
- package/test/server/httpapi-provider.test.ts +153 -0
- package/test/server/httpapi-pty-websocket.test.ts +16 -0
- package/test/server/httpapi-pty.test.ts +175 -0
- package/test/server/httpapi-raw-route-auth.test.ts +89 -0
- package/test/server/httpapi-sdk.test.ts +681 -0
- package/test/server/httpapi-session.test.ts +464 -0
- package/test/server/httpapi-sync.test.ts +130 -0
- package/test/server/httpapi-tui.test.ts +121 -0
- package/test/server/httpapi-workspace-routing.test.ts +471 -0
- package/test/server/httpapi-workspace.test.ts +427 -0
- package/test/server/lib/conformance.ts +88 -0
- package/test/server/lib/stateful.ts +112 -0
- package/test/server/project-init-git.test.ts +113 -0
- package/test/server/proxy-util.test.ts +113 -0
- package/test/server/session-actions.test.ts +49 -0
- package/test/server/session-list.test.ts +238 -0
- package/test/server/session-messages.test.ts +167 -0
- package/test/server/session-select.test.ts +100 -0
- package/test/server/trace-attributes.test.ts +76 -0
- package/test/server/workspace-proxy.test.ts +165 -0
- package/test/server/workspace-routing.test.ts +85 -0
- package/test/session/compaction.test.ts +2420 -0
- package/test/session/instruction.test.ts +247 -0
- package/test/session/llm.test.ts +1273 -0
- package/test/session/message-v2.test.ts +1291 -0
- package/test/session/messages-pagination.test.ts +1173 -0
- package/test/session/network.test.ts +249 -0
- package/test/session/processor-effect.test.ts +847 -0
- package/test/session/prompt.test.ts +2131 -0
- package/test/session/retry.test.ts +340 -0
- package/test/session/revert-compact.test.ts +639 -0
- package/test/session/schema-decoding.test.ts +311 -0
- package/test/session/session-entry-stepper.test.ts +917 -0
- package/test/session/session-schema.test.ts +76 -0
- package/test/session/snapshot-tool-race.test.ts +257 -0
- package/test/session/structured-output-integration.test.ts +265 -0
- package/test/session/structured-output.test.ts +381 -0
- package/test/session/system.test.ts +73 -0
- package/test/share/share-next.test.ts +333 -0
- package/test/shell/shell.test.ts +99 -0
- package/test/skill/discovery.test.ts +116 -0
- package/test/skill/skill.test.ts +393 -0
- package/test/smoke/.tui-debug-output.txt +1 -0
- package/test/smoke/.tui-debug-plain.txt +1 -0
- package/test/smoke/.tui-walkthrough-report.txt +122 -0
- package/test/smoke/smoke-tui-pty.test.ts +123 -0
- package/test/smoke/smoke-tui.mjs +83 -0
- package/test/smoke/tui-walkthrough.test.ts +520 -0
- package/test/snapshot/snapshot.test.ts +1531 -0
- package/test/storage/db.test.ts +23 -0
- package/test/storage/json-migration.test.ts +832 -0
- package/test/storage/storage.test.ts +293 -0
- package/test/suggestion/suggestion.test.ts +1 -0
- package/test/sync/index.test.ts +256 -0
- package/test/tool/__snapshots__/parameters.test.ts.snap +500 -0
- package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
- package/test/tool/apply_patch.test.ts +614 -0
- package/test/tool/bash.test.ts +1225 -0
- package/test/tool/diagnostics-filter.test.ts +55 -0
- package/test/tool/edit.test.ts +754 -0
- package/test/tool/external-directory.test.ts +169 -0
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +65179 -0
- package/test/tool/glob.test.ts +107 -0
- package/test/tool/grep.test.ts +114 -0
- package/test/tool/lsp.test.ts +187 -0
- package/test/tool/parameters.test.ts +243 -0
- package/test/tool/question.test.ts +129 -0
- package/test/tool/read.test.ts +500 -0
- package/test/tool/recall.test.ts +151 -0
- package/test/tool/registry.test.ts +203 -0
- package/test/tool/skill.test.ts +135 -0
- package/test/tool/suggest.test.ts +1 -0
- package/test/tool/task.test.ts +612 -0
- package/test/tool/tool-define.test.ts +99 -0
- package/test/tool/truncation.test.ts +260 -0
- package/test/tool/webfetch.test.ts +103 -0
- package/test/tool/write.test.ts +291 -0
- package/test/util/data-url.test.ts +14 -0
- package/test/util/effect-zod.test.ts +754 -0
- package/test/util/error.test.ts +38 -0
- package/test/util/filesystem.test.ts +656 -0
- package/test/util/format.test.ts +59 -0
- package/test/util/glob.test.ts +164 -0
- package/test/util/iife.test.ts +36 -0
- package/test/util/lazy.test.ts +50 -0
- package/test/util/lock.test.ts +72 -0
- package/test/util/log.test.ts +86 -0
- package/test/util/module.test.ts +59 -0
- package/test/util/process.test.ts +128 -0
- package/test/util/timeout.test.ts +21 -0
- package/test/util/which.test.ts +100 -0
- package/test/util/wildcard.test.ts +90 -0
- package/test/workspace/workspace-restore.test.ts +296 -0
- package/src/provider/models-snapshot.d.ts +0 -2
- package/src/provider/models-snapshot.js +0 -3
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import { formatDuration } from "../../src/util/format"
|
|
3
|
+
|
|
4
|
+
describe("util.format", () => {
|
|
5
|
+
describe("formatDuration", () => {
|
|
6
|
+
test("returns empty string for zero or negative values", () => {
|
|
7
|
+
expect(formatDuration(0)).toBe("")
|
|
8
|
+
expect(formatDuration(-1)).toBe("")
|
|
9
|
+
expect(formatDuration(-100)).toBe("")
|
|
10
|
+
})
|
|
11
|
+
|
|
12
|
+
test("formats seconds under a minute", () => {
|
|
13
|
+
expect(formatDuration(1)).toBe("1s")
|
|
14
|
+
expect(formatDuration(30)).toBe("30s")
|
|
15
|
+
expect(formatDuration(59)).toBe("59s")
|
|
16
|
+
})
|
|
17
|
+
|
|
18
|
+
test("formats minutes under an hour", () => {
|
|
19
|
+
expect(formatDuration(60)).toBe("1m")
|
|
20
|
+
expect(formatDuration(61)).toBe("1m 1s")
|
|
21
|
+
expect(formatDuration(90)).toBe("1m 30s")
|
|
22
|
+
expect(formatDuration(120)).toBe("2m")
|
|
23
|
+
expect(formatDuration(330)).toBe("5m 30s")
|
|
24
|
+
expect(formatDuration(3599)).toBe("59m 59s")
|
|
25
|
+
})
|
|
26
|
+
|
|
27
|
+
test("formats hours under a day", () => {
|
|
28
|
+
expect(formatDuration(3600)).toBe("1h")
|
|
29
|
+
expect(formatDuration(3660)).toBe("1h 1m")
|
|
30
|
+
expect(formatDuration(7200)).toBe("2h")
|
|
31
|
+
expect(formatDuration(8100)).toBe("2h 15m")
|
|
32
|
+
expect(formatDuration(86399)).toBe("23h 59m")
|
|
33
|
+
})
|
|
34
|
+
|
|
35
|
+
test("formats days under a week", () => {
|
|
36
|
+
expect(formatDuration(86400)).toBe("~1 day")
|
|
37
|
+
expect(formatDuration(172800)).toBe("~2 days")
|
|
38
|
+
expect(formatDuration(259200)).toBe("~3 days")
|
|
39
|
+
expect(formatDuration(604799)).toBe("~6 days")
|
|
40
|
+
})
|
|
41
|
+
|
|
42
|
+
test("formats weeks", () => {
|
|
43
|
+
expect(formatDuration(604800)).toBe("~1 week")
|
|
44
|
+
expect(formatDuration(1209600)).toBe("~2 weeks")
|
|
45
|
+
expect(formatDuration(1609200)).toBe("~2 weeks")
|
|
46
|
+
})
|
|
47
|
+
|
|
48
|
+
test("handles boundary values correctly", () => {
|
|
49
|
+
expect(formatDuration(59)).toBe("59s")
|
|
50
|
+
expect(formatDuration(60)).toBe("1m")
|
|
51
|
+
expect(formatDuration(3599)).toBe("59m 59s")
|
|
52
|
+
expect(formatDuration(3600)).toBe("1h")
|
|
53
|
+
expect(formatDuration(86399)).toBe("23h 59m")
|
|
54
|
+
expect(formatDuration(86400)).toBe("~1 day")
|
|
55
|
+
expect(formatDuration(604799)).toBe("~6 days")
|
|
56
|
+
expect(formatDuration(604800)).toBe("~1 week")
|
|
57
|
+
})
|
|
58
|
+
})
|
|
59
|
+
})
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
import { describe, test, expect } from "bun:test"
|
|
2
|
+
import path from "path"
|
|
3
|
+
import fs from "fs/promises"
|
|
4
|
+
import { Glob } from "@saeeol/core/util/glob"
|
|
5
|
+
import { tmpdir } from "../fixture/fixture"
|
|
6
|
+
|
|
7
|
+
describe("Glob", () => {
|
|
8
|
+
describe("scan()", () => {
|
|
9
|
+
test("finds files matching pattern", async () => {
|
|
10
|
+
await using tmp = await tmpdir()
|
|
11
|
+
await fs.writeFile(path.join(tmp.path, "a.txt"), "", "utf-8")
|
|
12
|
+
await fs.writeFile(path.join(tmp.path, "b.txt"), "", "utf-8")
|
|
13
|
+
await fs.writeFile(path.join(tmp.path, "c.md"), "", "utf-8")
|
|
14
|
+
|
|
15
|
+
const results = await Glob.scan("*.txt", { cwd: tmp.path })
|
|
16
|
+
|
|
17
|
+
expect(results.sort()).toEqual(["a.txt", "b.txt"])
|
|
18
|
+
})
|
|
19
|
+
|
|
20
|
+
test("returns absolute paths when absolute option is true", async () => {
|
|
21
|
+
await using tmp = await tmpdir()
|
|
22
|
+
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
23
|
+
|
|
24
|
+
const results = await Glob.scan("*.txt", { cwd: tmp.path, absolute: true })
|
|
25
|
+
|
|
26
|
+
expect(results[0]).toBe(path.join(tmp.path, "file.txt"))
|
|
27
|
+
})
|
|
28
|
+
|
|
29
|
+
test("excludes directories by default", async () => {
|
|
30
|
+
await using tmp = await tmpdir()
|
|
31
|
+
await fs.mkdir(path.join(tmp.path, "subdir"))
|
|
32
|
+
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
33
|
+
|
|
34
|
+
const results = await Glob.scan("*", { cwd: tmp.path })
|
|
35
|
+
|
|
36
|
+
expect(results).toEqual(["file.txt"])
|
|
37
|
+
})
|
|
38
|
+
|
|
39
|
+
test("excludes directories when include is 'file'", async () => {
|
|
40
|
+
await using tmp = await tmpdir()
|
|
41
|
+
await fs.mkdir(path.join(tmp.path, "subdir"))
|
|
42
|
+
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
43
|
+
|
|
44
|
+
const results = await Glob.scan("*", { cwd: tmp.path, include: "file" })
|
|
45
|
+
|
|
46
|
+
expect(results).toEqual(["file.txt"])
|
|
47
|
+
})
|
|
48
|
+
|
|
49
|
+
test("includes directories when include is 'all'", async () => {
|
|
50
|
+
await using tmp = await tmpdir()
|
|
51
|
+
await fs.mkdir(path.join(tmp.path, "subdir"))
|
|
52
|
+
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
53
|
+
|
|
54
|
+
const results = await Glob.scan("*", { cwd: tmp.path, include: "all" })
|
|
55
|
+
|
|
56
|
+
expect(results.sort()).toEqual(["file.txt", "subdir"])
|
|
57
|
+
})
|
|
58
|
+
|
|
59
|
+
test("handles nested patterns", async () => {
|
|
60
|
+
await using tmp = await tmpdir()
|
|
61
|
+
await fs.mkdir(path.join(tmp.path, "nested"), { recursive: true })
|
|
62
|
+
await fs.writeFile(path.join(tmp.path, "nested", "deep.txt"), "", "utf-8")
|
|
63
|
+
|
|
64
|
+
const results = await Glob.scan("**/*.txt", { cwd: tmp.path })
|
|
65
|
+
|
|
66
|
+
expect(results).toEqual([path.join("nested", "deep.txt")])
|
|
67
|
+
})
|
|
68
|
+
|
|
69
|
+
test("returns empty array for no matches", async () => {
|
|
70
|
+
await using tmp = await tmpdir()
|
|
71
|
+
|
|
72
|
+
const results = await Glob.scan("*.nonexistent", { cwd: tmp.path })
|
|
73
|
+
|
|
74
|
+
expect(results).toEqual([])
|
|
75
|
+
})
|
|
76
|
+
|
|
77
|
+
test("does not follow symlinks by default", async () => {
|
|
78
|
+
await using tmp = await tmpdir()
|
|
79
|
+
await fs.mkdir(path.join(tmp.path, "realdir"))
|
|
80
|
+
await fs.writeFile(path.join(tmp.path, "realdir", "file.txt"), "", "utf-8")
|
|
81
|
+
await fs.symlink(path.join(tmp.path, "realdir"), path.join(tmp.path, "linkdir"))
|
|
82
|
+
|
|
83
|
+
const results = await Glob.scan("**/*.txt", { cwd: tmp.path })
|
|
84
|
+
|
|
85
|
+
expect(results).toEqual([path.join("realdir", "file.txt")])
|
|
86
|
+
})
|
|
87
|
+
|
|
88
|
+
test("follows symlinks when symlink option is true", async () => {
|
|
89
|
+
await using tmp = await tmpdir()
|
|
90
|
+
await fs.mkdir(path.join(tmp.path, "realdir"))
|
|
91
|
+
await fs.writeFile(path.join(tmp.path, "realdir", "file.txt"), "", "utf-8")
|
|
92
|
+
await fs.symlink(path.join(tmp.path, "realdir"), path.join(tmp.path, "linkdir"))
|
|
93
|
+
|
|
94
|
+
const results = await Glob.scan("**/*.txt", { cwd: tmp.path, symlink: true })
|
|
95
|
+
|
|
96
|
+
expect(results.sort()).toEqual([path.join("linkdir", "file.txt"), path.join("realdir", "file.txt")])
|
|
97
|
+
})
|
|
98
|
+
|
|
99
|
+
test("includes dotfiles when dot option is true", async () => {
|
|
100
|
+
await using tmp = await tmpdir()
|
|
101
|
+
await fs.writeFile(path.join(tmp.path, ".hidden"), "", "utf-8")
|
|
102
|
+
await fs.writeFile(path.join(tmp.path, "visible"), "", "utf-8")
|
|
103
|
+
|
|
104
|
+
const results = await Glob.scan("*", { cwd: tmp.path, dot: true })
|
|
105
|
+
|
|
106
|
+
expect(results.sort()).toEqual([".hidden", "visible"])
|
|
107
|
+
})
|
|
108
|
+
|
|
109
|
+
test("excludes dotfiles when dot option is false", async () => {
|
|
110
|
+
await using tmp = await tmpdir()
|
|
111
|
+
await fs.writeFile(path.join(tmp.path, ".hidden"), "", "utf-8")
|
|
112
|
+
await fs.writeFile(path.join(tmp.path, "visible"), "", "utf-8")
|
|
113
|
+
|
|
114
|
+
const results = await Glob.scan("*", { cwd: tmp.path, dot: false })
|
|
115
|
+
|
|
116
|
+
expect(results).toEqual(["visible"])
|
|
117
|
+
})
|
|
118
|
+
})
|
|
119
|
+
|
|
120
|
+
describe("scanSync()", () => {
|
|
121
|
+
test("finds files matching pattern synchronously", async () => {
|
|
122
|
+
await using tmp = await tmpdir()
|
|
123
|
+
await fs.writeFile(path.join(tmp.path, "a.txt"), "", "utf-8")
|
|
124
|
+
await fs.writeFile(path.join(tmp.path, "b.txt"), "", "utf-8")
|
|
125
|
+
|
|
126
|
+
const results = Glob.scanSync("*.txt", { cwd: tmp.path })
|
|
127
|
+
|
|
128
|
+
expect(results.sort()).toEqual(["a.txt", "b.txt"])
|
|
129
|
+
})
|
|
130
|
+
|
|
131
|
+
test("respects options", async () => {
|
|
132
|
+
await using tmp = await tmpdir()
|
|
133
|
+
await fs.mkdir(path.join(tmp.path, "subdir"))
|
|
134
|
+
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
135
|
+
|
|
136
|
+
const results = Glob.scanSync("*", { cwd: tmp.path, include: "all" })
|
|
137
|
+
|
|
138
|
+
expect(results.sort()).toEqual(["file.txt", "subdir"])
|
|
139
|
+
})
|
|
140
|
+
})
|
|
141
|
+
|
|
142
|
+
describe("match()", () => {
|
|
143
|
+
test("matches simple patterns", () => {
|
|
144
|
+
expect(Glob.match("*.txt", "file.txt")).toBe(true)
|
|
145
|
+
expect(Glob.match("*.txt", "file.js")).toBe(false)
|
|
146
|
+
})
|
|
147
|
+
|
|
148
|
+
test("matches directory patterns", () => {
|
|
149
|
+
expect(Glob.match("**/*.js", "src/index.js")).toBe(true)
|
|
150
|
+
expect(Glob.match("**/*.js", "src/index.ts")).toBe(false)
|
|
151
|
+
})
|
|
152
|
+
|
|
153
|
+
test("matches dot files", () => {
|
|
154
|
+
expect(Glob.match(".*", ".gitignore")).toBe(true)
|
|
155
|
+
expect(Glob.match("**/*.md", ".github/README.md")).toBe(true)
|
|
156
|
+
})
|
|
157
|
+
|
|
158
|
+
test("matches brace expansion", () => {
|
|
159
|
+
expect(Glob.match("*.{js,ts}", "file.js")).toBe(true)
|
|
160
|
+
expect(Glob.match("*.{js,ts}", "file.ts")).toBe(true)
|
|
161
|
+
expect(Glob.match("*.{js,ts}", "file.py")).toBe(false)
|
|
162
|
+
})
|
|
163
|
+
})
|
|
164
|
+
})
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import { iife } from "../../src/util/iife"
|
|
3
|
+
|
|
4
|
+
describe("util.iife", () => {
|
|
5
|
+
test("should execute function immediately and return result", () => {
|
|
6
|
+
let called = false
|
|
7
|
+
const result = iife(() => {
|
|
8
|
+
called = true
|
|
9
|
+
return 42
|
|
10
|
+
})
|
|
11
|
+
|
|
12
|
+
expect(called).toBe(true)
|
|
13
|
+
expect(result).toBe(42)
|
|
14
|
+
})
|
|
15
|
+
|
|
16
|
+
test("should work with async functions", async () => {
|
|
17
|
+
let called = false
|
|
18
|
+
const result = await iife(async () => {
|
|
19
|
+
called = true
|
|
20
|
+
return "async result"
|
|
21
|
+
})
|
|
22
|
+
|
|
23
|
+
expect(called).toBe(true)
|
|
24
|
+
expect(result).toBe("async result")
|
|
25
|
+
})
|
|
26
|
+
|
|
27
|
+
test("should handle functions with no return value", () => {
|
|
28
|
+
let called = false
|
|
29
|
+
const result = iife(() => {
|
|
30
|
+
called = true
|
|
31
|
+
})
|
|
32
|
+
|
|
33
|
+
expect(called).toBe(true)
|
|
34
|
+
expect(result).toBeUndefined()
|
|
35
|
+
})
|
|
36
|
+
})
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import { lazy } from "../../src/util/lazy"
|
|
3
|
+
|
|
4
|
+
describe("util.lazy", () => {
|
|
5
|
+
test("should call function only once", () => {
|
|
6
|
+
let callCount = 0
|
|
7
|
+
const getValue = () => {
|
|
8
|
+
callCount++
|
|
9
|
+
return "expensive value"
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
const lazyValue = lazy(getValue)
|
|
13
|
+
|
|
14
|
+
expect(callCount).toBe(0)
|
|
15
|
+
|
|
16
|
+
const result1 = lazyValue()
|
|
17
|
+
expect(result1).toBe("expensive value")
|
|
18
|
+
expect(callCount).toBe(1)
|
|
19
|
+
|
|
20
|
+
const result2 = lazyValue()
|
|
21
|
+
expect(result2).toBe("expensive value")
|
|
22
|
+
expect(callCount).toBe(1)
|
|
23
|
+
})
|
|
24
|
+
|
|
25
|
+
test("should preserve the same reference", () => {
|
|
26
|
+
const obj = { value: 42 }
|
|
27
|
+
const lazyObj = lazy(() => obj)
|
|
28
|
+
|
|
29
|
+
const result1 = lazyObj()
|
|
30
|
+
const result2 = lazyObj()
|
|
31
|
+
|
|
32
|
+
expect(result1).toBe(obj)
|
|
33
|
+
expect(result2).toBe(obj)
|
|
34
|
+
expect(result1).toBe(result2)
|
|
35
|
+
})
|
|
36
|
+
|
|
37
|
+
test("should work with different return types", () => {
|
|
38
|
+
const lazyString = lazy(() => "string")
|
|
39
|
+
const lazyNumber = lazy(() => 123)
|
|
40
|
+
const lazyBoolean = lazy(() => true)
|
|
41
|
+
const lazyNull = lazy(() => null)
|
|
42
|
+
const lazyUndefined = lazy(() => undefined)
|
|
43
|
+
|
|
44
|
+
expect(lazyString()).toBe("string")
|
|
45
|
+
expect(lazyNumber()).toBe(123)
|
|
46
|
+
expect(lazyBoolean()).toBe(true)
|
|
47
|
+
expect(lazyNull()).toBe(null)
|
|
48
|
+
expect(lazyUndefined()).toBe(undefined)
|
|
49
|
+
})
|
|
50
|
+
})
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import { Lock } from "@/util/lock"
|
|
3
|
+
|
|
4
|
+
function tick() {
|
|
5
|
+
return new Promise<void>((r) => queueMicrotask(r))
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
async function flush(n = 5) {
|
|
9
|
+
for (let i = 0; i < n; i++) await tick()
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
describe("util.lock", () => {
|
|
13
|
+
test("writer exclusivity: blocks reads and other writes while held", async () => {
|
|
14
|
+
const key = "lock:" + Math.random().toString(36).slice(2)
|
|
15
|
+
|
|
16
|
+
const state = {
|
|
17
|
+
writer2: false,
|
|
18
|
+
reader: false,
|
|
19
|
+
writers: 0,
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
// Acquire writer1
|
|
23
|
+
using writer1 = await Lock.write(key)
|
|
24
|
+
state.writers++
|
|
25
|
+
expect(state.writers).toBe(1)
|
|
26
|
+
|
|
27
|
+
// Start writer2 candidate (should block)
|
|
28
|
+
const writer2Task = (async () => {
|
|
29
|
+
const w = await Lock.write(key)
|
|
30
|
+
state.writers++
|
|
31
|
+
expect(state.writers).toBe(1)
|
|
32
|
+
state.writer2 = true
|
|
33
|
+
// Hold for a tick so reader cannot slip in
|
|
34
|
+
await tick()
|
|
35
|
+
return w
|
|
36
|
+
})()
|
|
37
|
+
|
|
38
|
+
// Start reader candidate (should block)
|
|
39
|
+
const readerTask = (async () => {
|
|
40
|
+
const r = await Lock.read(key)
|
|
41
|
+
state.reader = true
|
|
42
|
+
return r
|
|
43
|
+
})()
|
|
44
|
+
|
|
45
|
+
// Flush microtasks and assert neither acquired
|
|
46
|
+
await flush()
|
|
47
|
+
expect(state.writer2).toBe(false)
|
|
48
|
+
expect(state.reader).toBe(false)
|
|
49
|
+
|
|
50
|
+
// Release writer1
|
|
51
|
+
writer1[Symbol.dispose]()
|
|
52
|
+
state.writers--
|
|
53
|
+
|
|
54
|
+
// writer2 should acquire next
|
|
55
|
+
const writer2 = await writer2Task
|
|
56
|
+
expect(state.writer2).toBe(true)
|
|
57
|
+
|
|
58
|
+
// Reader still blocked while writer2 held
|
|
59
|
+
await flush()
|
|
60
|
+
expect(state.reader).toBe(false)
|
|
61
|
+
|
|
62
|
+
// Release writer2
|
|
63
|
+
writer2[Symbol.dispose]()
|
|
64
|
+
state.writers--
|
|
65
|
+
|
|
66
|
+
// Reader should now acquire
|
|
67
|
+
const reader = await readerTask
|
|
68
|
+
expect(state.reader).toBe(true)
|
|
69
|
+
|
|
70
|
+
reader[Symbol.dispose]()
|
|
71
|
+
})
|
|
72
|
+
})
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
import { afterEach, expect, test } from "bun:test"
|
|
2
|
+
import fs from "fs/promises"
|
|
3
|
+
import path from "path"
|
|
4
|
+
import { Global } from "@saeeol/core/global"
|
|
5
|
+
import * as Log from "@saeeol/core/util/log"
|
|
6
|
+
import * as Process from "../../src/util/process"
|
|
7
|
+
import { tmpdir } from "../fixture/fixture"
|
|
8
|
+
|
|
9
|
+
const log = Global.Path.log
|
|
10
|
+
|
|
11
|
+
afterEach(() => {
|
|
12
|
+
Global.Path.log = log
|
|
13
|
+
})
|
|
14
|
+
|
|
15
|
+
async function files(dir: string) {
|
|
16
|
+
let last = ""
|
|
17
|
+
let same = 0
|
|
18
|
+
|
|
19
|
+
for (let i = 0; i < 50; i++) {
|
|
20
|
+
const list = (await fs.readdir(dir)).sort()
|
|
21
|
+
const next = JSON.stringify(list)
|
|
22
|
+
same = next === last ? same + 1 : 0
|
|
23
|
+
if (same >= 2 && list.length === 11) return list
|
|
24
|
+
last = next
|
|
25
|
+
await Bun.sleep(10)
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
return (await fs.readdir(dir)).sort()
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
test("init cleanup keeps the newest timestamped logs", async () => {
|
|
32
|
+
await using tmp = await tmpdir()
|
|
33
|
+
Global.Path.log = tmp.path
|
|
34
|
+
|
|
35
|
+
const list = Array.from({ length: 12 }, (_, i) => `2000-01-${String(i + 1).padStart(2, "0")}T000000.log`)
|
|
36
|
+
|
|
37
|
+
await Promise.all(list.map((file) => fs.writeFile(path.join(tmp.path, file), file)))
|
|
38
|
+
|
|
39
|
+
await Log.init({ print: false, dev: false })
|
|
40
|
+
|
|
41
|
+
const next = await files(tmp.path)
|
|
42
|
+
|
|
43
|
+
expect(next).not.toContain(list[0]!)
|
|
44
|
+
expect(next).toContain(list.at(-1)!)
|
|
45
|
+
})
|
|
46
|
+
const root = path.join(import.meta.dir, "../..")
|
|
47
|
+
const worker = path.join(import.meta.dir, "../fixture/log-init-worker.ts")
|
|
48
|
+
|
|
49
|
+
test("uses single log directory for rotation history", async () => {
|
|
50
|
+
await using tmp = await tmpdir()
|
|
51
|
+
|
|
52
|
+
const out = await Process.run([process.execPath, "--conditions=browser", worker, tmp.path], {
|
|
53
|
+
cwd: root,
|
|
54
|
+
nothrow: true,
|
|
55
|
+
})
|
|
56
|
+
|
|
57
|
+
const dir = path.join(tmp.path, "share", "saeeol", "log")
|
|
58
|
+
const history = path.join(dir, ".log-history")
|
|
59
|
+
|
|
60
|
+
expect(out.code).toBe(0)
|
|
61
|
+
expect(out.stderr.toString()).not.toContain("log stream error:")
|
|
62
|
+
expect(out.stdout.toString()).toBe(path.join(dir, "dev.log"))
|
|
63
|
+
|
|
64
|
+
const stat = await fs.stat(history)
|
|
65
|
+
expect(stat.isFile()).toBe(true)
|
|
66
|
+
})
|
|
67
|
+
|
|
68
|
+
test("skips rotation rename when active log file is missing", async () => {
|
|
69
|
+
await using tmp = await tmpdir()
|
|
70
|
+
|
|
71
|
+
const out = await Process.run([process.execPath, "--conditions=browser", worker, tmp.path, "missing"], {
|
|
72
|
+
cwd: root,
|
|
73
|
+
nothrow: true,
|
|
74
|
+
})
|
|
75
|
+
|
|
76
|
+
const dir = path.join(tmp.path, "share", "saeeol", "log")
|
|
77
|
+
const list = (await fs.readdir(dir)).sort()
|
|
78
|
+
const next = list.filter((file) => /^\d{8}-\d{4}-\d{2}-dev\.log$/.test(file))
|
|
79
|
+
const stat = await fs.stat(path.join(dir, next[0]!))
|
|
80
|
+
|
|
81
|
+
expect(out.code).toBe(0)
|
|
82
|
+
expect(out.stderr.toString()).not.toContain("log stream error:")
|
|
83
|
+
expect(list).toContain("dev.log")
|
|
84
|
+
expect(next).toHaveLength(1)
|
|
85
|
+
expect(stat.size).toBe(0)
|
|
86
|
+
})
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import path from "path"
|
|
3
|
+
import { Module } from "@saeeol/core/util/module"
|
|
4
|
+
import { Filesystem } from "@/util/filesystem"
|
|
5
|
+
import { tmpdir } from "../fixture/fixture"
|
|
6
|
+
|
|
7
|
+
describe("util.module", () => {
|
|
8
|
+
test("resolves package subpaths from the provided dir", async () => {
|
|
9
|
+
await using tmp = await tmpdir()
|
|
10
|
+
const root = path.join(tmp.path, "proj")
|
|
11
|
+
const file = path.join(root, "node_modules/typescript/lib/tsserver.js")
|
|
12
|
+
await Filesystem.write(file, "export {}\n")
|
|
13
|
+
await Filesystem.writeJson(path.join(root, "node_modules/typescript/package.json"), { name: "typescript" })
|
|
14
|
+
|
|
15
|
+
expect(Module.resolve("typescript/lib/tsserver.js", root)).toBe(file)
|
|
16
|
+
})
|
|
17
|
+
|
|
18
|
+
test("resolves packages through ancestor node_modules", async () => {
|
|
19
|
+
await using tmp = await tmpdir()
|
|
20
|
+
const root = path.join(tmp.path, "proj")
|
|
21
|
+
const cwd = path.join(root, "apps/web")
|
|
22
|
+
const file = path.join(root, "node_modules/eslint/lib/api.js")
|
|
23
|
+
await Filesystem.write(file, "export {}\n")
|
|
24
|
+
await Filesystem.writeJson(path.join(root, "node_modules/eslint/package.json"), {
|
|
25
|
+
name: "eslint",
|
|
26
|
+
main: "lib/api.js",
|
|
27
|
+
})
|
|
28
|
+
await Filesystem.write(path.join(cwd, ".keep"), "")
|
|
29
|
+
|
|
30
|
+
expect(Module.resolve("eslint", cwd)).toBe(file)
|
|
31
|
+
})
|
|
32
|
+
|
|
33
|
+
test("resolves relative to the provided dir", async () => {
|
|
34
|
+
await using tmp = await tmpdir()
|
|
35
|
+
const a = path.join(tmp.path, "a")
|
|
36
|
+
const b = path.join(tmp.path, "b")
|
|
37
|
+
const left = path.join(a, "node_modules/biome/index.js")
|
|
38
|
+
const right = path.join(b, "node_modules/biome/index.js")
|
|
39
|
+
await Filesystem.write(left, "export {}\n")
|
|
40
|
+
await Filesystem.write(right, "export {}\n")
|
|
41
|
+
await Filesystem.writeJson(path.join(a, "node_modules/biome/package.json"), {
|
|
42
|
+
name: "biome",
|
|
43
|
+
main: "index.js",
|
|
44
|
+
})
|
|
45
|
+
await Filesystem.writeJson(path.join(b, "node_modules/biome/package.json"), {
|
|
46
|
+
name: "biome",
|
|
47
|
+
main: "index.js",
|
|
48
|
+
})
|
|
49
|
+
|
|
50
|
+
expect(Module.resolve("biome", a)).toBe(left)
|
|
51
|
+
expect(Module.resolve("biome", b)).toBe(right)
|
|
52
|
+
expect(Module.resolve("biome", a)).not.toBe(Module.resolve("biome", b))
|
|
53
|
+
})
|
|
54
|
+
|
|
55
|
+
test("returns undefined when resolution fails", async () => {
|
|
56
|
+
await using tmp = await tmpdir()
|
|
57
|
+
expect(Module.resolve("missing-package", tmp.path)).toBeUndefined()
|
|
58
|
+
})
|
|
59
|
+
})
|
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import fs from "fs/promises"
|
|
3
|
+
import path from "path"
|
|
4
|
+
import { Process } from "@/util/process"
|
|
5
|
+
import { tmpdir } from "../fixture/fixture"
|
|
6
|
+
|
|
7
|
+
function node(script: string) {
|
|
8
|
+
return [process.execPath, "-e", script]
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
describe("util.process", () => {
|
|
12
|
+
test("captures stdout and stderr", async () => {
|
|
13
|
+
const out = await Process.run(node('process.stdout.write("out");process.stderr.write("err")'))
|
|
14
|
+
expect(out.code).toBe(0)
|
|
15
|
+
expect(out.stdout.toString()).toBe("out")
|
|
16
|
+
expect(out.stderr.toString()).toBe("err")
|
|
17
|
+
})
|
|
18
|
+
|
|
19
|
+
test("returns code when nothrow is enabled", async () => {
|
|
20
|
+
const out = await Process.run(node("process.exit(7)"), { nothrow: true })
|
|
21
|
+
expect(out.code).toBe(7)
|
|
22
|
+
})
|
|
23
|
+
|
|
24
|
+
test("throws RunFailedError on non-zero exit", async () => {
|
|
25
|
+
const err = await Process.run(node('process.stderr.write("bad");process.exit(3)')).catch((error) => error)
|
|
26
|
+
expect(err).toBeInstanceOf(Process.RunFailedError)
|
|
27
|
+
if (!(err instanceof Process.RunFailedError)) throw err
|
|
28
|
+
expect(err.code).toBe(3)
|
|
29
|
+
expect(err.stderr.toString()).toBe("bad")
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
test("aborts a running process", async () => {
|
|
33
|
+
const abort = new AbortController()
|
|
34
|
+
const started = Date.now()
|
|
35
|
+
setTimeout(() => abort.abort(), 25)
|
|
36
|
+
|
|
37
|
+
const out = await Process.run(node("setInterval(() => {}, 1000)"), {
|
|
38
|
+
abort: abort.signal,
|
|
39
|
+
nothrow: true,
|
|
40
|
+
})
|
|
41
|
+
|
|
42
|
+
expect(out.code).not.toBe(0)
|
|
43
|
+
expect(Date.now() - started).toBeLessThan(1000)
|
|
44
|
+
}, 3000)
|
|
45
|
+
|
|
46
|
+
test("kills after timeout when process ignores terminate signal", async () => {
|
|
47
|
+
if (process.platform === "win32") return
|
|
48
|
+
|
|
49
|
+
const abort = new AbortController()
|
|
50
|
+
const started = Date.now()
|
|
51
|
+
setTimeout(() => abort.abort(), 25)
|
|
52
|
+
|
|
53
|
+
const out = await Process.run(node('process.on("SIGTERM", () => {}); setInterval(() => {}, 1000)'), {
|
|
54
|
+
abort: abort.signal,
|
|
55
|
+
nothrow: true,
|
|
56
|
+
timeout: 25,
|
|
57
|
+
})
|
|
58
|
+
|
|
59
|
+
expect(out.code).not.toBe(0)
|
|
60
|
+
expect(Date.now() - started).toBeLessThan(1000)
|
|
61
|
+
}, 3000)
|
|
62
|
+
|
|
63
|
+
test("uses cwd when spawning commands", async () => {
|
|
64
|
+
await using tmp = await tmpdir()
|
|
65
|
+
const out = await Process.run(node("process.stdout.write(process.cwd())"), {
|
|
66
|
+
cwd: tmp.path,
|
|
67
|
+
})
|
|
68
|
+
expect(out.stdout.toString()).toBe(tmp.path)
|
|
69
|
+
})
|
|
70
|
+
|
|
71
|
+
test("merges environment overrides", async () => {
|
|
72
|
+
const out = await Process.run(node('process.stdout.write(process.env.SAEEOL_TEST ?? "")'), {
|
|
73
|
+
env: {
|
|
74
|
+
SAEEOL_TEST: "set",
|
|
75
|
+
},
|
|
76
|
+
})
|
|
77
|
+
expect(out.stdout.toString()).toBe("set")
|
|
78
|
+
})
|
|
79
|
+
|
|
80
|
+
test("uses shell in run on Windows", async () => {
|
|
81
|
+
if (process.platform !== "win32") return
|
|
82
|
+
|
|
83
|
+
const out = await Process.run(["set", "SAEEOL_TEST_SHELL"], {
|
|
84
|
+
shell: true,
|
|
85
|
+
env: {
|
|
86
|
+
SAEEOL_TEST_SHELL: "ok",
|
|
87
|
+
},
|
|
88
|
+
})
|
|
89
|
+
|
|
90
|
+
expect(out.code).toBe(0)
|
|
91
|
+
expect(out.stdout.toString()).toContain("SAEEOL_TEST_SHELL=ok")
|
|
92
|
+
})
|
|
93
|
+
|
|
94
|
+
test("runs cmd scripts with spaces on Windows without shell", async () => {
|
|
95
|
+
if (process.platform !== "win32") return
|
|
96
|
+
|
|
97
|
+
await using tmp = await tmpdir()
|
|
98
|
+
const dir = path.join(tmp.path, "with space")
|
|
99
|
+
const file = path.join(dir, "echo cmd.cmd")
|
|
100
|
+
|
|
101
|
+
await fs.mkdir(dir, { recursive: true })
|
|
102
|
+
await Bun.write(file, "@echo off\r\nif %~1==--stdio exit /b 0\r\nexit /b 7\r\n")
|
|
103
|
+
|
|
104
|
+
const proc = Process.spawn([file, "--stdio"], {
|
|
105
|
+
stdin: "pipe",
|
|
106
|
+
stdout: "pipe",
|
|
107
|
+
stderr: "pipe",
|
|
108
|
+
})
|
|
109
|
+
|
|
110
|
+
expect(await proc.exited).toBe(0)
|
|
111
|
+
})
|
|
112
|
+
|
|
113
|
+
test("rejects missing commands without leaking unhandled errors", async () => {
|
|
114
|
+
await using tmp = await tmpdir()
|
|
115
|
+
const cmd = path.join(tmp.path, "missing" + (process.platform === "win32" ? ".cmd" : ""))
|
|
116
|
+
const err = await Process.spawn([cmd], {
|
|
117
|
+
stdin: "pipe",
|
|
118
|
+
stdout: "pipe",
|
|
119
|
+
stderr: "pipe",
|
|
120
|
+
}).exited.catch((err) => err)
|
|
121
|
+
|
|
122
|
+
expect(err).toBeInstanceOf(Error)
|
|
123
|
+
if (!(err instanceof Error)) throw err
|
|
124
|
+
expect(err).toMatchObject({
|
|
125
|
+
code: "ENOENT",
|
|
126
|
+
})
|
|
127
|
+
})
|
|
128
|
+
})
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import { withTimeout } from "../../src/util/timeout"
|
|
3
|
+
|
|
4
|
+
describe("util.timeout", () => {
|
|
5
|
+
test("should resolve when promise completes before timeout", async () => {
|
|
6
|
+
const fastPromise = new Promise<string>((resolve) => {
|
|
7
|
+
setTimeout(() => resolve("fast"), 10)
|
|
8
|
+
})
|
|
9
|
+
|
|
10
|
+
const result = await withTimeout(fastPromise, 100)
|
|
11
|
+
expect(result).toBe("fast")
|
|
12
|
+
})
|
|
13
|
+
|
|
14
|
+
test("should reject when promise exceeds timeout", async () => {
|
|
15
|
+
const slowPromise = new Promise<string>((resolve) => {
|
|
16
|
+
setTimeout(() => resolve("slow"), 200)
|
|
17
|
+
})
|
|
18
|
+
|
|
19
|
+
await expect(withTimeout(slowPromise, 50)).rejects.toThrow("Operation timed out after 50ms")
|
|
20
|
+
})
|
|
21
|
+
})
|