mixdog 0.7.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude-plugin/marketplace.json +31 -0
- package/.claude-plugin/plugin.json +20 -0
- package/.gitattributes +34 -0
- package/.mcp.json +14 -0
- package/ARCHITECTURE.md +77 -0
- package/CHANGELOG.md +7 -0
- package/CONTRIBUTING.md +45 -0
- package/DATA-FLOW.md +79 -0
- package/LICENSE +21 -0
- package/README.md +389 -0
- package/SECURITY.md +138 -0
- package/UNINSTALL.md +112 -0
- package/agents/maintenance.md +5 -0
- package/agents/memory-classification.md +30 -0
- package/agents/scheduler-task.md +18 -0
- package/agents/webhook-handler.md +27 -0
- package/agents/worker.md +24 -0
- package/bin/bridge +133 -0
- package/bin/statusline-launcher.mjs +78 -0
- package/bin/statusline-lib.mjs +550 -0
- package/bin/statusline.mjs +607 -0
- package/bun.lock +802 -0
- package/commands/config.md +16 -0
- package/commands/doctor.md +13 -0
- package/commands/setup.md +17 -0
- package/defaults/cycle3-review-prompt.md +90 -0
- package/defaults/hidden-roles.json +65 -0
- package/defaults/memory-chunk-prompt.md +63 -0
- package/defaults/memory-promote-prompt.md +135 -0
- package/defaults/mixdog-config.template.json +27 -0
- package/defaults/user-workflow.json +8 -0
- package/defaults/user-workflow.md +12 -0
- package/hooks/hooks.json +73 -0
- package/hooks/lib/active-instance.cjs +77 -0
- package/hooks/lib/permission-evaluator.cjs +411 -0
- package/hooks/lib/permission-route.cjs +63 -0
- package/hooks/lib/permission-rules.cjs +170 -0
- package/hooks/lib/settings-loader.cjs +116 -0
- package/hooks/post-tool-use.cjs +84 -0
- package/hooks/pre-mcp-sandbox.cjs +158 -0
- package/hooks/pre-tool-subagent.cjs +253 -0
- package/hooks/session-start.cjs +1372 -0
- package/hooks/turn-timer.cjs +82 -0
- package/lib/claude-md-writer.cjs +386 -0
- package/lib/config-cjs.cjs +61 -0
- package/lib/hook-pipe-path.cjs +10 -0
- package/lib/keychain-cjs.cjs +263 -0
- package/lib/plugin-paths.cjs +61 -0
- package/lib/rules-builder.cjs +241 -0
- package/lib/text-utils.cjs +61 -0
- package/native/README.md +117 -0
- package/native/prebuilt/linux-aarch64/mixdog-shim +0 -0
- package/native/prebuilt/linux-x86_64/mixdog-shim +0 -0
- package/native/prebuilt/macos-aarch64/mixdog-shim +0 -0
- package/native/prebuilt/macos-x86_64/mixdog-shim +0 -0
- package/native/prebuilt/windows-x86_64/mixdog-shim.exe +0 -0
- package/package.json +107 -0
- package/prompts/code-review.txt +16 -0
- package/prompts/security-audit.txt +17 -0
- package/rules/bridge/00-common.md +39 -0
- package/rules/bridge/20-skip-protocol.md +18 -0
- package/rules/bridge/30-explorer.md +33 -0
- package/rules/bridge/40-cycle1-agent.md +52 -0
- package/rules/bridge/41-cycle2-agent.md +62 -0
- package/rules/bridge/42-cycle3-agent.md +44 -0
- package/rules/lead/00-tool-lead.md +61 -0
- package/rules/lead/01-general.md +23 -0
- package/rules/lead/02-channels.md +49 -0
- package/rules/lead/03-team.md +27 -0
- package/rules/lead/04-workflow.md +20 -0
- package/rules/shared/00-language.md +14 -0
- package/rules/shared/01-tool.md +138 -0
- package/scripts/bootstrap.mjs +184 -0
- package/scripts/bridge-unify-smoke.mjs +308 -0
- package/scripts/build-runtime-linux.sh +348 -0
- package/scripts/build-runtime-macos.sh +217 -0
- package/scripts/build-runtime-windows.ps1 +242 -0
- package/scripts/builtin-utils-smoke.mjs +392 -0
- package/scripts/check-json.mjs +45 -0
- package/scripts/check-syntax-changed.mjs +102 -0
- package/scripts/check-syntax.mjs +58 -0
- package/scripts/code-graph-batch.test.mjs +33 -0
- package/scripts/config-preserve-smoke.mjs +180 -0
- package/scripts/doctor.mjs +484 -0
- package/scripts/edit-normalize-fuzz.mjs +130 -0
- package/scripts/edit-normalize-smoke.mjs +401 -0
- package/scripts/edit-operation-smoke.mjs +369 -0
- package/scripts/edit2-smoke.mjs +63 -0
- package/scripts/fuzzy-e2e.mjs +28 -0
- package/scripts/fuzzy-smoke.mjs +26 -0
- package/scripts/generate-runtime-manifest.mjs +166 -0
- package/scripts/guard-smoke.mjs +66 -0
- package/scripts/hidden-role-schema-smoke.mjs +162 -0
- package/scripts/hook-routing-smoke.mjs +29 -0
- package/scripts/inject-input.ps1 +204 -0
- package/scripts/io-complex-smoke.mjs +667 -0
- package/scripts/io-explore-bench.mjs +424 -0
- package/scripts/io-guardrails-smoke.mjs +205 -0
- package/scripts/io-mini-bench-baseline.json +11 -0
- package/scripts/io-mini-bench.mjs +216 -0
- package/scripts/io-route-harness.mjs +933 -0
- package/scripts/io-telemetry-report.mjs +691 -0
- package/scripts/mutation-bench.mjs +564 -0
- package/scripts/mutation-io-smoke.mjs +1081 -0
- package/scripts/native-patch-bridge-smoke.mjs +288 -0
- package/scripts/native-patch-smoke.mjs +304 -0
- package/scripts/patch-interior-context-smoke.mjs +49 -0
- package/scripts/patch-newline-utf8-smoke.mjs +157 -0
- package/scripts/perf-hook-smoke.mjs +71 -0
- package/scripts/permission-eval-smoke.mjs +426 -0
- package/scripts/prep-patch.mjs +53 -0
- package/scripts/prep-shim.mjs +96 -0
- package/scripts/provider-cache-smoke.mjs +687 -0
- package/scripts/report-runtime-health.mjs +132 -0
- package/scripts/run-mcp.mjs +1547 -0
- package/scripts/salvage-v4a-shatter.test.mjs +58 -0
- package/scripts/scoped-cache-io-smoke.mjs +103 -0
- package/scripts/shell-policy-round3-smoke.mjs +46 -0
- package/scripts/smoke-runtime-negative.ps1 +100 -0
- package/scripts/smoke-runtime-negative.sh +95 -0
- package/scripts/stall-policy-smoke.mjs +50 -0
- package/scripts/start-memory-worker.mjs +23 -0
- package/scripts/statusline-launcher-smoke.mjs +82 -0
- package/scripts/stress-atomic-write.mjs +1028 -0
- package/scripts/test-config-rmw-restore.mjs +122 -0
- package/scripts/test-fault-inject.mjs +164 -0
- package/scripts/test-large-file.mjs +174 -0
- package/scripts/tool-edge-smoke.mjs +209 -0
- package/scripts/uninstall.mjs +201 -0
- package/scripts/webhook-selfheal-smoke.mjs +29 -0
- package/scripts/write-overwrite-guard-smoke.mjs +56 -0
- package/server-main.mjs +3055 -0
- package/server.mjs +468 -0
- package/setup/config-merge.mjs +254 -0
- package/setup/install.mjs +120 -0
- package/setup/launch-core.mjs +507 -0
- package/setup/launch.mjs +101 -0
- package/setup/setup-server.mjs +3206 -0
- package/setup/setup.html +3693 -0
- package/skills/retro-skill-proposer/SKILL.md +92 -0
- package/skills/schedule-add/SKILL.md +77 -0
- package/skills/setup/SKILL.md +346 -0
- package/skills/webhook-add/SKILL.md +81 -0
- package/src/agent/bridge-stall-watchdog.mjs +337 -0
- package/src/agent/index.mjs +2138 -0
- package/src/agent/orchestrator/activity-bus.mjs +38 -0
- package/src/agent/orchestrator/ai-wrapped-dispatch.mjs +1010 -0
- package/src/agent/orchestrator/bridge-retry.mjs +220 -0
- package/src/agent/orchestrator/bridge-trace.mjs +583 -0
- package/src/agent/orchestrator/cache-mtime.mjs +58 -0
- package/src/agent/orchestrator/config.mjs +358 -0
- package/src/agent/orchestrator/context/collect.mjs +651 -0
- package/src/agent/orchestrator/dispatch-persist.mjs +549 -0
- package/src/agent/orchestrator/drain-registry.mjs +50 -0
- package/src/agent/orchestrator/explore-validator.mjs +8 -0
- package/src/agent/orchestrator/internal-roles.mjs +118 -0
- package/src/agent/orchestrator/internal-tools.mjs +88 -0
- package/src/agent/orchestrator/jobs.mjs +116 -0
- package/src/agent/orchestrator/mcp/client.mjs +364 -0
- package/src/agent/orchestrator/providers/anthropic-betas.mjs +21 -0
- package/src/agent/orchestrator/providers/anthropic-oauth.mjs +1745 -0
- package/src/agent/orchestrator/providers/anthropic.mjs +437 -0
- package/src/agent/orchestrator/providers/gemini.mjs +1175 -0
- package/src/agent/orchestrator/providers/grok-oauth.mjs +782 -0
- package/src/agent/orchestrator/providers/model-catalog.mjs +241 -0
- package/src/agent/orchestrator/providers/openai-compat.mjs +1467 -0
- package/src/agent/orchestrator/providers/openai-oauth-ws.mjs +1890 -0
- package/src/agent/orchestrator/providers/openai-oauth.mjs +1307 -0
- package/src/agent/orchestrator/providers/openai-ws.mjs +104 -0
- package/src/agent/orchestrator/providers/registry.mjs +192 -0
- package/src/agent/orchestrator/providers/retry-classifier.mjs +325 -0
- package/src/agent/orchestrator/session/abort-lookup.mjs +13 -0
- package/src/agent/orchestrator/session/cache/post-edit-marks.mjs +42 -0
- package/src/agent/orchestrator/session/cache/prefetch-cache.mjs +142 -0
- package/src/agent/orchestrator/session/cache/read-cache.mjs +319 -0
- package/src/agent/orchestrator/session/cache/scoped-cache-outcome.mjs +11 -0
- package/src/agent/orchestrator/session/cache/scoped-cache.mjs +361 -0
- package/src/agent/orchestrator/session/cache/util.mjs +49 -0
- package/src/agent/orchestrator/session/loop.mjs +1478 -0
- package/src/agent/orchestrator/session/manager.mjs +1975 -0
- package/src/agent/orchestrator/session/read-dedup.mjs +6 -0
- package/src/agent/orchestrator/session/result-classification.mjs +65 -0
- package/src/agent/orchestrator/session/save-session-worker.mjs +18 -0
- package/src/agent/orchestrator/session/store.mjs +624 -0
- package/src/agent/orchestrator/session/stream-watchdog.mjs +130 -0
- package/src/agent/orchestrator/session/tool-result-offload.mjs +166 -0
- package/src/agent/orchestrator/session/trim.mjs +491 -0
- package/src/agent/orchestrator/smart-bridge/CACHE-SHARD.md +115 -0
- package/src/agent/orchestrator/smart-bridge/bridge-llm.mjs +327 -0
- package/src/agent/orchestrator/smart-bridge/cache-obs.mjs +150 -0
- package/src/agent/orchestrator/smart-bridge/cache-strategy.mjs +228 -0
- package/src/agent/orchestrator/smart-bridge/index.mjs +215 -0
- package/src/agent/orchestrator/smart-bridge/profiles.mjs +37 -0
- package/src/agent/orchestrator/smart-bridge/registry.mjs +348 -0
- package/src/agent/orchestrator/smart-bridge/session-builder.mjs +116 -0
- package/src/agent/orchestrator/stall-policy.mjs +195 -0
- package/src/agent/orchestrator/tool-loop-guard.mjs +75 -0
- package/src/agent/orchestrator/tools/bash-policy-scan.mjs +77 -0
- package/src/agent/orchestrator/tools/bash-session.mjs +721 -0
- package/src/agent/orchestrator/tools/builtin/advisory-lock.mjs +171 -0
- package/src/agent/orchestrator/tools/builtin/arg-guard.mjs +455 -0
- package/src/agent/orchestrator/tools/builtin/atomic-write.mjs +236 -0
- package/src/agent/orchestrator/tools/builtin/bash-tool.mjs +480 -0
- package/src/agent/orchestrator/tools/builtin/binary-file.mjs +76 -0
- package/src/agent/orchestrator/tools/builtin/builtin-tools.mjs +256 -0
- package/src/agent/orchestrator/tools/builtin/cache-layers.mjs +386 -0
- package/src/agent/orchestrator/tools/builtin/cwd-utils.mjs +37 -0
- package/src/agent/orchestrator/tools/builtin/device-paths.mjs +154 -0
- package/src/agent/orchestrator/tools/builtin/diagnostics-tool.mjs +292 -0
- package/src/agent/orchestrator/tools/builtin/diff-utils.mjs +109 -0
- package/src/agent/orchestrator/tools/builtin/edit-base-guard.mjs +58 -0
- package/src/agent/orchestrator/tools/builtin/edit-byte-plan.mjs +240 -0
- package/src/agent/orchestrator/tools/builtin/edit-byte-utils.mjs +113 -0
- package/src/agent/orchestrator/tools/builtin/edit-commit.mjs +74 -0
- package/src/agent/orchestrator/tools/builtin/edit-context-utils.mjs +242 -0
- package/src/agent/orchestrator/tools/builtin/edit-diagnostics.mjs +211 -0
- package/src/agent/orchestrator/tools/builtin/edit-engine.mjs +1364 -0
- package/src/agent/orchestrator/tools/builtin/edit-failure-context.mjs +126 -0
- package/src/agent/orchestrator/tools/builtin/edit-hint.mjs +141 -0
- package/src/agent/orchestrator/tools/builtin/edit-match-utils.mjs +194 -0
- package/src/agent/orchestrator/tools/builtin/edit-partial-write.mjs +60 -0
- package/src/agent/orchestrator/tools/builtin/edit-stale-refresh.mjs +168 -0
- package/src/agent/orchestrator/tools/builtin/edit-tool.mjs +173 -0
- package/src/agent/orchestrator/tools/builtin/edit-utf8-guard.mjs +48 -0
- package/src/agent/orchestrator/tools/builtin/fs-reachability.mjs +48 -0
- package/src/agent/orchestrator/tools/builtin/fuzzy-match.mjs +99 -0
- package/src/agent/orchestrator/tools/builtin/glob-walk.mjs +170 -0
- package/src/agent/orchestrator/tools/builtin/grep-formatting.mjs +113 -0
- package/src/agent/orchestrator/tools/builtin/hash-utils.mjs +6 -0
- package/src/agent/orchestrator/tools/builtin/list-formatting.mjs +7 -0
- package/src/agent/orchestrator/tools/builtin/list-tool.mjs +593 -0
- package/src/agent/orchestrator/tools/builtin/native-edit-runner.mjs +89 -0
- package/src/agent/orchestrator/tools/builtin/notebook-edit-tool.mjs +300 -0
- package/src/agent/orchestrator/tools/builtin/open-config-tool.mjs +26 -0
- package/src/agent/orchestrator/tools/builtin/path-diagnostics.mjs +152 -0
- package/src/agent/orchestrator/tools/builtin/path-locks.mjs +35 -0
- package/src/agent/orchestrator/tools/builtin/path-utils.mjs +201 -0
- package/src/agent/orchestrator/tools/builtin/read-args.mjs +103 -0
- package/src/agent/orchestrator/tools/builtin/read-batch.mjs +172 -0
- package/src/agent/orchestrator/tools/builtin/read-constants.mjs +40 -0
- package/src/agent/orchestrator/tools/builtin/read-formatting.mjs +118 -0
- package/src/agent/orchestrator/tools/builtin/read-image-resize.mjs +189 -0
- package/src/agent/orchestrator/tools/builtin/read-image.mjs +88 -0
- package/src/agent/orchestrator/tools/builtin/read-lines.mjs +12 -0
- package/src/agent/orchestrator/tools/builtin/read-mode-tool.mjs +455 -0
- package/src/agent/orchestrator/tools/builtin/read-open.mjs +190 -0
- package/src/agent/orchestrator/tools/builtin/read-range-index.mjs +271 -0
- package/src/agent/orchestrator/tools/builtin/read-ranges.mjs +26 -0
- package/src/agent/orchestrator/tools/builtin/read-single-tool.mjs +728 -0
- package/src/agent/orchestrator/tools/builtin/read-snapshot-runtime.mjs +173 -0
- package/src/agent/orchestrator/tools/builtin/read-special-files.mjs +268 -0
- package/src/agent/orchestrator/tools/builtin/read-streaming.mjs +602 -0
- package/src/agent/orchestrator/tools/builtin/read-tool.mjs +530 -0
- package/src/agent/orchestrator/tools/builtin/read-windows.mjs +107 -0
- package/src/agent/orchestrator/tools/builtin/rename-tool.mjs +196 -0
- package/src/agent/orchestrator/tools/builtin/rg-runner.mjs +422 -0
- package/src/agent/orchestrator/tools/builtin/search-builders.mjs +158 -0
- package/src/agent/orchestrator/tools/builtin/search-tool.mjs +869 -0
- package/src/agent/orchestrator/tools/builtin/shell-analysis.mjs +653 -0
- package/src/agent/orchestrator/tools/builtin/shell-jobs.mjs +936 -0
- package/src/agent/orchestrator/tools/builtin/shell-output.mjs +36 -0
- package/src/agent/orchestrator/tools/builtin/shell-runtime.mjs +214 -0
- package/src/agent/orchestrator/tools/builtin/snapshot-helpers.mjs +143 -0
- package/src/agent/orchestrator/tools/builtin/snapshot-store.mjs +206 -0
- package/src/agent/orchestrator/tools/builtin/snapshot-validation.mjs +98 -0
- package/src/agent/orchestrator/tools/builtin/text-stats.mjs +69 -0
- package/src/agent/orchestrator/tools/builtin/windows-roots.mjs +23 -0
- package/src/agent/orchestrator/tools/builtin/write-tool.mjs +401 -0
- package/src/agent/orchestrator/tools/builtin.mjs +500 -0
- package/src/agent/orchestrator/tools/code-graph-prewarm-worker.mjs +39 -0
- package/src/agent/orchestrator/tools/code-graph-tool-defs.mjs +24 -0
- package/src/agent/orchestrator/tools/code-graph.mjs +4095 -0
- package/src/agent/orchestrator/tools/cwd-tool.mjs +298 -0
- package/src/agent/orchestrator/tools/destructive-warning.mjs +323 -0
- package/src/agent/orchestrator/tools/edit-normalize.mjs +603 -0
- package/src/agent/orchestrator/tools/env-scrub.mjs +100 -0
- package/src/agent/orchestrator/tools/graph-binary-fetcher.mjs +144 -0
- package/src/agent/orchestrator/tools/graph-manifest.json +26 -0
- package/src/agent/orchestrator/tools/host-input.mjs +204 -0
- package/src/agent/orchestrator/tools/mutation-content-cache.mjs +67 -0
- package/src/agent/orchestrator/tools/mutation-planner.mjs +75 -0
- package/src/agent/orchestrator/tools/next-call-utils.mjs +48 -0
- package/src/agent/orchestrator/tools/patch-binary-fetcher.mjs +133 -0
- package/src/agent/orchestrator/tools/patch-manifest.json +26 -0
- package/src/agent/orchestrator/tools/patch-tool-defs.mjs +20 -0
- package/src/agent/orchestrator/tools/patch.mjs +2754 -0
- package/src/agent/orchestrator/tools/progress-message.mjs +118 -0
- package/src/agent/orchestrator/tools/result-compression.mjs +279 -0
- package/src/agent/orchestrator/tools/shell-command.mjs +865 -0
- package/src/agent/orchestrator/tools/shell-exec-policy.mjs +89 -0
- package/src/agent/orchestrator/tools/shell-policy-danger-target.mjs +27 -0
- package/src/agent/orchestrator/tools/shell-policy-imports.mjs +7 -0
- package/src/agent/orchestrator/tools/shell-policy.mjs +345 -0
- package/src/agent/orchestrator/tools/shell-snapshot.mjs +313 -0
- package/src/agent/orchestrator/workflow-store.mjs +93 -0
- package/src/agent/tool-defs.mjs +103 -0
- package/src/channels/backends/discord.mjs +784 -0
- package/src/channels/data/voice-runtime-manifest.json +138 -0
- package/src/channels/index.mjs +3229 -0
- package/src/channels/lib/cli-worker-host.mjs +12 -0
- package/src/channels/lib/config-lock.mjs +13 -0
- package/src/channels/lib/config.mjs +292 -0
- package/src/channels/lib/drop-trace.mjs +71 -0
- package/src/channels/lib/event-pipeline.mjs +81 -0
- package/src/channels/lib/event-queue.mjs +345 -0
- package/src/channels/lib/executor.mjs +168 -0
- package/src/channels/lib/format.mjs +188 -0
- package/src/channels/lib/holidays.mjs +138 -0
- package/src/channels/lib/hook-pipe-server.mjs +802 -0
- package/src/channels/lib/interaction-workflows.mjs +184 -0
- package/src/channels/lib/memory-client.mjs +149 -0
- package/src/channels/lib/output-forwarder.mjs +765 -0
- package/src/channels/lib/runtime-paths.mjs +479 -0
- package/src/channels/lib/scheduler.mjs +723 -0
- package/src/channels/lib/session-control.mjs +36 -0
- package/src/channels/lib/session-discovery.mjs +103 -0
- package/src/channels/lib/settings.mjs +11 -0
- package/src/channels/lib/state-file.mjs +68 -0
- package/src/channels/lib/status-snapshot.mjs +219 -0
- package/src/channels/lib/tool-format.mjs +140 -0
- package/src/channels/lib/transcript-discovery.mjs +195 -0
- package/src/channels/lib/voice-runtime-fetcher.mjs +734 -0
- package/src/channels/lib/webhook.mjs +1179 -0
- package/src/channels/lib/whisper-server.mjs +477 -0
- package/src/channels/tool-defs.mjs +170 -0
- package/src/daemon/host.mjs +118 -0
- package/src/daemon/mcp-transport.mjs +47 -0
- package/src/daemon/session.mjs +100 -0
- package/src/daemon/thin-client.mjs +71 -0
- package/src/daemon/transport.mjs +163 -0
- package/src/memory/data/runtime-manifest.json +40 -0
- package/src/memory/index.mjs +3305 -0
- package/src/memory/lib/agent-ipc.mjs +93 -0
- package/src/memory/lib/bridge-trace-queries.mjs +120 -0
- package/src/memory/lib/core-memory-store.mjs +330 -0
- package/src/memory/lib/embedding-provider.mjs +269 -0
- package/src/memory/lib/embedding-worker.mjs +323 -0
- package/src/memory/lib/llm-worker-host.mjs +17 -0
- package/src/memory/lib/memory-cycle.mjs +11 -0
- package/src/memory/lib/memory-cycle1.mjs +641 -0
- package/src/memory/lib/memory-cycle2.mjs +1284 -0
- package/src/memory/lib/memory-cycle3.mjs +540 -0
- package/src/memory/lib/memory-embed.mjs +299 -0
- package/src/memory/lib/memory-extraction.mjs +5 -0
- package/src/memory/lib/memory-maintenance-store.mjs +32 -0
- package/src/memory/lib/memory-ops-policy.mjs +190 -0
- package/src/memory/lib/memory-recall-id-patch.mjs +15 -0
- package/src/memory/lib/memory-recall-read-query.mjs +7 -0
- package/src/memory/lib/memory-recall-scope-filter.mjs +63 -0
- package/src/memory/lib/memory-recall-store.mjs +621 -0
- package/src/memory/lib/memory-retrievers.mjs +112 -0
- package/src/memory/lib/memory-score.mjs +71 -0
- package/src/memory/lib/memory-text-utils.mjs +58 -0
- package/src/memory/lib/memory.mjs +412 -0
- package/src/memory/lib/model-profile.mjs +85 -0
- package/src/memory/lib/pg/adapter.mjs +308 -0
- package/src/memory/lib/pg/process.mjs +360 -0
- package/src/memory/lib/pg/supervisor.mjs +396 -0
- package/src/memory/lib/project-id-resolver.mjs +86 -0
- package/src/memory/lib/runtime-fetcher.mjs +442 -0
- package/src/memory/lib/trace-store.mjs +728 -0
- package/src/memory/tool-defs.mjs +79 -0
- package/src/search/index.mjs +1173 -0
- package/src/search/lib/backends/anthropic-oauth.mjs +98 -0
- package/src/search/lib/backends/exa.mjs +50 -0
- package/src/search/lib/backends/firecrawl.mjs +61 -0
- package/src/search/lib/backends/gemini-api.mjs +83 -0
- package/src/search/lib/backends/grok-oauth.mjs +86 -0
- package/src/search/lib/backends/index.mjs +150 -0
- package/src/search/lib/backends/openai-api.mjs +144 -0
- package/src/search/lib/backends/openai-oauth.mjs +98 -0
- package/src/search/lib/backends/openai-web-search.mjs +76 -0
- package/src/search/lib/backends/tavily.mjs +55 -0
- package/src/search/lib/backends/xai-api.mjs +113 -0
- package/src/search/lib/cache.mjs +131 -0
- package/src/search/lib/config.mjs +192 -0
- package/src/search/lib/formatter.mjs +115 -0
- package/src/search/lib/provider-usage.mjs +67 -0
- package/src/search/lib/providers.mjs +47 -0
- package/src/search/lib/search-intent.mjs +109 -0
- package/src/search/lib/setup-handler.mjs +261 -0
- package/src/search/lib/state.mjs +201 -0
- package/src/search/lib/web-tools.mjs +1207 -0
- package/src/search/tool-defs.mjs +83 -0
- package/src/setup/defender-exclusion.mjs +183 -0
- package/src/shared/abort-controller.mjs +15 -0
- package/src/shared/atomic-file.mjs +420 -0
- package/src/shared/config.mjs +350 -0
- package/src/shared/daemon-recycle.mjs +108 -0
- package/src/shared/disable-claude-builtins.mjs +88 -0
- package/src/shared/err-text.mjs +12 -0
- package/src/shared/llm/cost.mjs +66 -0
- package/src/shared/llm/http-agent.mjs +123 -0
- package/src/shared/llm/index.mjs +41 -0
- package/src/shared/llm/pid-cleanup.mjs +27 -0
- package/src/shared/llm/usage-log.mjs +47 -0
- package/src/shared/plugin-paths.mjs +58 -0
- package/src/shared/schedules-store.mjs +70 -0
- package/src/shared/seed.mjs +119 -0
- package/src/shared/user-cwd.mjs +213 -0
- package/src/shared/user-data-guard.mjs +238 -0
- package/src/status/aggregator.mjs +584 -0
- package/src/status/server.mjs +413 -0
- package/tools.json +1653 -0
|
@@ -0,0 +1,667 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* Complex I/O smoke suite for Mixdog builtin read/edit behavior.
|
|
4
|
+
*
|
|
5
|
+
* Usage:
|
|
6
|
+
* node scripts/io-complex-smoke.mjs
|
|
7
|
+
* MIXDOG_SMOKE_ROOT=C:\path\to\cache\mixdog\0.x.y node scripts/io-complex-smoke.mjs
|
|
8
|
+
*
|
|
9
|
+
* The suite imports builtin.mjs from MIXDOG_SMOKE_ROOT when provided, while
|
|
10
|
+
* keeping all fixture files and read snapshots inside fresh temp directories.
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
import { mkdirSync, readFileSync, rmSync, writeFileSync } from 'node:fs';
|
|
14
|
+
import { tmpdir } from 'node:os';
|
|
15
|
+
import { dirname, join, resolve } from 'node:path';
|
|
16
|
+
import { fileURLToPath, pathToFileURL } from 'node:url';
|
|
17
|
+
|
|
18
|
+
const HERE = dirname(fileURLToPath(import.meta.url));
|
|
19
|
+
const SOURCE_ROOT = resolve(HERE, '..');
|
|
20
|
+
const PLUGIN_ROOT = resolve(process.env.MIXDOG_SMOKE_ROOT || SOURCE_ROOT);
|
|
21
|
+
const WORKDIR = join(tmpdir(), `mixdog-io-complex-work-${process.pid}-${Date.now()}`);
|
|
22
|
+
const DATA_DIR = join(tmpdir(), `mixdog-io-complex-data-${process.pid}-${Date.now()}`);
|
|
23
|
+
const BASE_SCOPE = `io-complex-${process.pid}-${Date.now()}`;
|
|
24
|
+
|
|
25
|
+
process.env.CLAUDE_PLUGIN_ROOT = PLUGIN_ROOT;
|
|
26
|
+
process.env.CLAUDE_PLUGIN_DATA = DATA_DIR;
|
|
27
|
+
process.env.MIXDOG_HINT_LEVEL ||= 'normal';
|
|
28
|
+
|
|
29
|
+
const { parseJsonNextCalls } = await import(pathToFileURL(join(SOURCE_ROOT, 'src/agent/orchestrator/tools/next-call-utils.mjs')).href);
|
|
30
|
+
const builtinUrl = pathToFileURL(join(PLUGIN_ROOT, 'src/agent/orchestrator/tools/builtin.mjs')).href;
|
|
31
|
+
const { executeBuiltinTool } = await import(builtinUrl);
|
|
32
|
+
const patchUrl = pathToFileURL(join(PLUGIN_ROOT, 'src/agent/orchestrator/tools/patch.mjs')).href;
|
|
33
|
+
const { executePatchTool, closeNativePatchServerForTests } = await import(patchUrl);
|
|
34
|
+
const readDedupUrl = pathToFileURL(join(PLUGIN_ROOT, 'src/agent/orchestrator/session/read-dedup.mjs')).href;
|
|
35
|
+
const { extractTouchedPathsFromPatch } = await import(readDedupUrl);
|
|
36
|
+
|
|
37
|
+
let passed = 0;
|
|
38
|
+
let failed = 0;
|
|
39
|
+
|
|
40
|
+
mkdirSync(WORKDIR, { recursive: true });
|
|
41
|
+
mkdirSync(DATA_DIR, { recursive: true });
|
|
42
|
+
|
|
43
|
+
console.log(`[io-smoke] pluginRoot=${PLUGIN_ROOT}`);
|
|
44
|
+
console.log(`[io-smoke] workdir=${WORKDIR}`);
|
|
45
|
+
|
|
46
|
+
await scenario('parallel read array', async (scope) => {
|
|
47
|
+
put('read-array/a.txt', 'one\ntwo\nthree\n');
|
|
48
|
+
const out = await tool('read', { path: ['read-array/a.txt', 'read-array/a.txt'] }, scope);
|
|
49
|
+
expectIncludes(out, 'one');
|
|
50
|
+
expectIncludes(out, 'three');
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
await scenario('object-array same-file ranges', async (scope) => {
|
|
54
|
+
put('ranges.txt', rangeLines('range-line', 8));
|
|
55
|
+
const out = await tool('read', {
|
|
56
|
+
path: [
|
|
57
|
+
{ path: 'ranges.txt', offset: 0, limit: 2 },
|
|
58
|
+
{ path: 'ranges.txt', offset: 4, limit: 2 },
|
|
59
|
+
],
|
|
60
|
+
}, scope);
|
|
61
|
+
expectIncludes(out, 'range-line-1');
|
|
62
|
+
expectIncludes(out, 'range-line-5');
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
await scenario('short unseen gap edit now succeeds (read-window gate retired)', async (scope) => {
|
|
66
|
+
put('gap-short.txt', linesWith(30, 20, 'xx'));
|
|
67
|
+
await tool('read', { path: 'gap-short.txt', offset: 0, limit: 2 }, scope);
|
|
68
|
+
const out = await tool('edit', { path: 'gap-short.txt', old_string: 'xx', new_string: 'yy' }, scope);
|
|
69
|
+
expectIncludes(out, 'Edited:');
|
|
70
|
+
expectTrue(!String(out).includes('next_call:'), out);
|
|
71
|
+
expectIncludes(get('gap-short.txt'), 'yy');
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
await scenario('strong unique unseen gap auto-covers', async (scope) => {
|
|
75
|
+
const oldText = 'AUTO_COVER_TARGET_alpha_beta_gamma_1234567890';
|
|
76
|
+
put('gap-strong.txt', linesWith(30, 20, oldText));
|
|
77
|
+
await tool('read', { path: 'gap-strong.txt', offset: 0, limit: 2 }, scope);
|
|
78
|
+
const out = await tool('edit', { path: 'gap-strong.txt', old_string: oldText, new_string: 'AUTO_COVER_REPLACED' }, scope);
|
|
79
|
+
expectIncludes(out, 'Edited:');
|
|
80
|
+
expectIncludes(get('gap-strong.txt'), 'AUTO_COVER_REPLACED');
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
await scenario('large edit lands without route hint', async (scope) => {
|
|
84
|
+
const oldBlock = rangeLines('large-old', 34) + 'large-old-target\n';
|
|
85
|
+
const newBlock = oldBlock.replace('large-old-target', 'large-new-target');
|
|
86
|
+
put('large-edit-auto.txt', `before\n${oldBlock}after\n`);
|
|
87
|
+
const out = await tool('edit', {
|
|
88
|
+
path: 'large-edit-auto.txt',
|
|
89
|
+
old_string: oldBlock,
|
|
90
|
+
new_string: newBlock,
|
|
91
|
+
}, scope);
|
|
92
|
+
expectIncludes(out, 'Edited: large-edit-auto.txt');
|
|
93
|
+
expectIncludes(get('large-edit-auto.txt'), 'large-new-target');
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
await scenario('large delete lands without route hint', async (scope) => {
|
|
97
|
+
const oldBlock = rangeLines('delete-large', 31);
|
|
98
|
+
put('large-delete-auto.txt', `before\n${oldBlock}after\n`);
|
|
99
|
+
const out = await tool('edit', {
|
|
100
|
+
path: 'large-delete-auto.txt',
|
|
101
|
+
old_string: oldBlock,
|
|
102
|
+
new_string: '',
|
|
103
|
+
}, scope);
|
|
104
|
+
expectIncludes(out, 'Edited: large-delete-auto.txt');
|
|
105
|
+
expectEquals(get('large-delete-auto.txt'), 'before\nafter\n');
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
await scenario('replace_all unseen gap edit now succeeds (read-window gate retired)', async (scope) => {
|
|
109
|
+
const oldText = 'REPLACE_ALL_TARGET_alpha_beta_gamma_1234567890';
|
|
110
|
+
put('gap-replace-all.txt', linesWith(30, 20, oldText));
|
|
111
|
+
await tool('read', { path: 'gap-replace-all.txt', offset: 0, limit: 2 }, scope);
|
|
112
|
+
const out = await tool('edit', {
|
|
113
|
+
path: 'gap-replace-all.txt',
|
|
114
|
+
old_string: oldText,
|
|
115
|
+
new_string: 'REPLACE_ALL_CHANGED',
|
|
116
|
+
replace_all: true,
|
|
117
|
+
}, scope);
|
|
118
|
+
expectIncludes(out, 'Edited:');
|
|
119
|
+
expectIncludes(get('gap-replace-all.txt'), 'REPLACE_ALL_CHANGED');
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
await scenario('cross-file batch preflight rollback', async (scope) => {
|
|
123
|
+
put('batch-rb/a.txt', 'alpha\n');
|
|
124
|
+
put('batch-rb/b.txt', 'beta\n');
|
|
125
|
+
await tool('read', { path: ['batch-rb/a.txt', 'batch-rb/b.txt'] }, scope);
|
|
126
|
+
const out = await tool('edit', {
|
|
127
|
+
edits: [
|
|
128
|
+
{ path: 'batch-rb/a.txt', old_string: 'alpha', new_string: 'ALPHA' },
|
|
129
|
+
{ path: 'batch-rb/b.txt', old_string: 'missing', new_string: 'BETA' },
|
|
130
|
+
],
|
|
131
|
+
}, scope);
|
|
132
|
+
expectIncludes(out, 'batch edit preflight failed');
|
|
133
|
+
expectEquals(get('batch-rb/a.txt'), 'alpha\n');
|
|
134
|
+
expectEquals(get('batch-rb/b.txt'), 'beta\n');
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
await scenario('cross-file batch success', async (scope) => {
|
|
138
|
+
put('batch-ok/a.txt', 'alpha\n');
|
|
139
|
+
put('batch-ok/b.txt', 'beta\n');
|
|
140
|
+
await tool('read', { path: ['batch-ok/a.txt', 'batch-ok/b.txt'] }, scope);
|
|
141
|
+
const out = await tool('edit', {
|
|
142
|
+
edits: [
|
|
143
|
+
{ path: 'batch-ok/a.txt', old_string: 'alpha', new_string: 'ALPHA' },
|
|
144
|
+
{ path: 'batch-ok/b.txt', old_string: 'beta', new_string: 'BETA' },
|
|
145
|
+
],
|
|
146
|
+
}, scope);
|
|
147
|
+
expectIncludes(out, 'OK batch-ok/a.txt');
|
|
148
|
+
expectIncludes(out, 'OK batch-ok/b.txt');
|
|
149
|
+
expectEquals(get('batch-ok/a.txt'), 'ALPHA\n');
|
|
150
|
+
expectEquals(get('batch-ok/b.txt'), 'BETA\n');
|
|
151
|
+
});
|
|
152
|
+
|
|
153
|
+
await scenario('same-file sequential multi-edit', async (scope) => {
|
|
154
|
+
put('multi-seq.txt', 'first\nsecond\n');
|
|
155
|
+
await tool('read', { path: 'multi-seq.txt' }, scope);
|
|
156
|
+
const out = await tool('edit', {
|
|
157
|
+
path: 'multi-seq.txt',
|
|
158
|
+
edits: [
|
|
159
|
+
{ old_string: 'first', new_string: 'FIRST' },
|
|
160
|
+
{ old_string: 'second', new_string: 'SECOND' },
|
|
161
|
+
],
|
|
162
|
+
}, scope);
|
|
163
|
+
expectIncludes(out, 'Edited:');
|
|
164
|
+
expectEquals(get('multi-seq.txt'), 'FIRST\nSECOND\n');
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
await scenario('same-file multi-edit atomic failure', async (scope) => {
|
|
168
|
+
put('multi-fail.txt', 'first\nsecond\n');
|
|
169
|
+
await tool('read', { path: 'multi-fail.txt' }, scope);
|
|
170
|
+
const out = await tool('edit', {
|
|
171
|
+
path: 'multi-fail.txt',
|
|
172
|
+
edits: [
|
|
173
|
+
{ old_string: 'first', new_string: 'FIRST' },
|
|
174
|
+
{ old_string: 'missing', new_string: 'MISSING' },
|
|
175
|
+
],
|
|
176
|
+
}, scope);
|
|
177
|
+
expectIncludes(out, 'Error [code 8]');
|
|
178
|
+
expectEquals(get('multi-fail.txt'), 'first\nsecond\n');
|
|
179
|
+
});
|
|
180
|
+
|
|
181
|
+
await scenario('code8 exposes compact current candidate', async (scope) => {
|
|
182
|
+
put('candidate.txt', 'root\n value: 1\nend\n');
|
|
183
|
+
await tool('read', { path: 'candidate.txt' }, scope);
|
|
184
|
+
const out = await tool('edit', { path: 'candidate.txt', old_string: 'value: 1\nmissing-end', new_string: 'value: 2\nmissing-end' }, scope);
|
|
185
|
+
expectIncludes(out, 'Error [code 8]');
|
|
186
|
+
expectTrue(!String(out).includes('retry_call:'), out);
|
|
187
|
+
expectTrue(!String(out).includes('current old_string candidate'), out);
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
await scenario('parallel different-file edits', async (scope) => {
|
|
191
|
+
for (const name of ['a', 'b', 'c']) put(`parallel-diff/${name}.txt`, `${name}-old\n`);
|
|
192
|
+
await tool('read', { path: ['parallel-diff/a.txt', 'parallel-diff/b.txt', 'parallel-diff/c.txt'] }, scope);
|
|
193
|
+
const outs = await Promise.all(['a', 'b', 'c'].map((name) =>
|
|
194
|
+
tool('edit', { path: `parallel-diff/${name}.txt`, old_string: `${name}-old`, new_string: `${name}-new` }, scope)
|
|
195
|
+
));
|
|
196
|
+
for (const out of outs) expectIncludes(out, 'Edited:');
|
|
197
|
+
for (const name of ['a', 'b', 'c']) expectEquals(get(`parallel-diff/${name}.txt`), `${name}-new\n`);
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
await scenario('parallel same-file different-target multi-edits serialize', async (scope) => {
|
|
201
|
+
put('parallel-same-diff.txt', 'left\nmiddle\nright\n');
|
|
202
|
+
await tool('read', { path: 'parallel-same-diff.txt' }, scope);
|
|
203
|
+
const outs = await Promise.all([
|
|
204
|
+
tool('edit', { edits: [{ path: 'parallel-same-diff.txt', old_string: 'left', new_string: 'LEFT' }] }, scope),
|
|
205
|
+
tool('edit', { edits: [{ path: 'parallel-same-diff.txt', old_string: 'right', new_string: 'RIGHT' }] }, scope),
|
|
206
|
+
]);
|
|
207
|
+
for (const out of outs) expectIncludes(out, 'Edited:');
|
|
208
|
+
expectEquals(get('parallel-same-diff.txt'), 'LEFT\nmiddle\nRIGHT\n');
|
|
209
|
+
});
|
|
210
|
+
|
|
211
|
+
await scenario('parallel same-file same-target conflicts cleanly', async (scope) => {
|
|
212
|
+
put('parallel-same-target.txt', 'same\nother\n');
|
|
213
|
+
await tool('read', { path: 'parallel-same-target.txt' }, scope);
|
|
214
|
+
const outs = await Promise.all([
|
|
215
|
+
tool('edit', { edits: [{ path: 'parallel-same-target.txt', old_string: 'same', new_string: 'SAME_A' }] }, scope),
|
|
216
|
+
tool('edit', { edits: [{ path: 'parallel-same-target.txt', old_string: 'same', new_string: 'SAME_B' }] }, scope),
|
|
217
|
+
]);
|
|
218
|
+
expectEquals(outs.filter((out) => String(out).includes('Edited:')).length, 1);
|
|
219
|
+
expectEquals(outs.filter((out) => String(out).includes('Error [code 8]')).length, 1);
|
|
220
|
+
const finalText = get('parallel-same-target.txt');
|
|
221
|
+
expectTrue(finalText === 'SAME_A\nother\n' || finalText === 'SAME_B\nother\n', finalText);
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
await scenario('concurrent cross-file batches avoid mixed partial state', async (scope) => {
|
|
225
|
+
put('batch-race/a.txt', 'a1\n');
|
|
226
|
+
put('batch-race/b.txt', 'b1\n');
|
|
227
|
+
await tool('read', { path: ['batch-race/a.txt', 'batch-race/b.txt'] }, scope);
|
|
228
|
+
const batchA = tool('edit', {
|
|
229
|
+
edits: [
|
|
230
|
+
{ path: 'batch-race/a.txt', old_string: 'a1', new_string: 'a2' },
|
|
231
|
+
{ path: 'batch-race/b.txt', old_string: 'b1', new_string: 'b2' },
|
|
232
|
+
],
|
|
233
|
+
}, scope);
|
|
234
|
+
const batchB = tool('edit', {
|
|
235
|
+
edits: [
|
|
236
|
+
{ path: 'batch-race/a.txt', old_string: 'a1', new_string: 'a3' },
|
|
237
|
+
{ path: 'batch-race/b.txt', old_string: 'b1', new_string: 'b3' },
|
|
238
|
+
],
|
|
239
|
+
}, scope);
|
|
240
|
+
const outs = await Promise.all([batchA, batchB]);
|
|
241
|
+
expectEquals(outs.filter((out) => !String(out).startsWith('Error:')).length, 1);
|
|
242
|
+
const a = get('batch-race/a.txt');
|
|
243
|
+
const b = get('batch-race/b.txt');
|
|
244
|
+
expectTrue((a === 'a2\n' && b === 'b2\n') || (a === 'a3\n' && b === 'b3\n'), `a=${JSON.stringify(a)} b=${JSON.stringify(b)}`);
|
|
245
|
+
});
|
|
246
|
+
|
|
247
|
+
// c14b3d0 friction fix: a stale snapshot whose old_string is also absent from
|
|
248
|
+
// CURRENT disk bytes reports plain not-found against current content (code 8,
|
|
249
|
+
// with recovery context) instead of forcing a re-read turn via code 7.
|
|
250
|
+
await scenario('stale snapshot with vanished old_string returns code 8', async (scope) => {
|
|
251
|
+
put('stale.txt', 'original\n');
|
|
252
|
+
await tool('read', { path: 'stale.txt' }, scope);
|
|
253
|
+
await sleep(50);
|
|
254
|
+
put('stale.txt', 'changed\n');
|
|
255
|
+
await sleep(50);
|
|
256
|
+
const out = await tool('edit', { path: 'stale.txt', old_string: 'original', new_string: 'edited' }, scope);
|
|
257
|
+
expectIncludes(out, 'Error [code 8]');
|
|
258
|
+
expectIncludes(out, 'file changed since read');
|
|
259
|
+
});
|
|
260
|
+
|
|
261
|
+
await scenario('stale snapshot unique target auto-refreshes', async (scope) => {
|
|
262
|
+
put('stale-preview.txt', 'keep\nTARGET\n');
|
|
263
|
+
await tool('read', { path: 'stale-preview.txt' }, scope);
|
|
264
|
+
await sleep(50);
|
|
265
|
+
put('stale-preview.txt', 'keep\nTARGET\nexternal\n');
|
|
266
|
+
await sleep(50);
|
|
267
|
+
const out = await tool('edit', { path: 'stale-preview.txt', old_string: 'TARGET', new_string: 'DONE' }, scope);
|
|
268
|
+
expectIncludes(out, 'Edited:');
|
|
269
|
+
expectIncludes(get('stale-preview.txt'), 'DONE');
|
|
270
|
+
});
|
|
271
|
+
|
|
272
|
+
await scenario('CRLF-normalized edit preserves intent', async (scope) => {
|
|
273
|
+
put('crlf.txt', 'a\r\nb\r\n');
|
|
274
|
+
await tool('read', { path: 'crlf.txt' }, scope);
|
|
275
|
+
const out = await tool('edit', { path: 'crlf.txt', old_string: 'a\nb', new_string: 'x\ny' }, scope);
|
|
276
|
+
expectIncludes(out, 'Edited:');
|
|
277
|
+
expectEquals(get('crlf.txt'), 'x\r\ny\r\n');
|
|
278
|
+
});
|
|
279
|
+
|
|
280
|
+
await scenario('smart-truncated read middle edit now succeeds (read-window gate retired)', async (scope) => {
|
|
281
|
+
const rows = Array.from({ length: 700 }, (_, i) => i === 349 ? 'M' : `smart-line-${i + 1}`);
|
|
282
|
+
put('smart-large.txt', rows.join('\n') + '\n');
|
|
283
|
+
const readOut = await tool('read', { path: 'smart-large.txt' }, scope);
|
|
284
|
+
expectIncludes(readOut, 'TRUNCATED');
|
|
285
|
+
expectTrue(!String(readOut).includes('next_call:'), readOut);
|
|
286
|
+
const editOut = await tool('edit', { path: 'smart-large.txt', old_string: 'M', new_string: 'N' }, scope);
|
|
287
|
+
expectIncludes(editOut, 'Edited:');
|
|
288
|
+
expectIncludes(get('smart-large.txt'), '\nN\n');
|
|
289
|
+
});
|
|
290
|
+
|
|
291
|
+
await scenario('path offset is 0-based and file_path offset is 1-based', async (scope) => {
|
|
292
|
+
put('offset-path.txt', 'line-1\nline-2\nline-3\n');
|
|
293
|
+
put('offset-file-path.txt', 'line-1\nline-2\nline-3\n');
|
|
294
|
+
const pathOut = await tool('read', { path: 'offset-path.txt', offset: 1, limit: 1 }, scope);
|
|
295
|
+
const filePathOut = await tool('read', { file_path: 'offset-file-path.txt', offset: 2, limit: 1 }, scope);
|
|
296
|
+
expectIncludes(pathOut, 'line-2');
|
|
297
|
+
expectIncludes(filePathOut, 'line-2');
|
|
298
|
+
});
|
|
299
|
+
|
|
300
|
+
await scenario('read object-array supports file_path alias', async (scope) => {
|
|
301
|
+
put('object-file-path.txt', 'line-1\nline-2\nline-3\n');
|
|
302
|
+
const out = await tool('read', { path: [{ file_path: 'object-file-path.txt', offset: 1, limit: 1 }] }, scope);
|
|
303
|
+
expectIncludes(out, 'line-1');
|
|
304
|
+
});
|
|
305
|
+
|
|
306
|
+
await scenario('wide default read has no context advisory', async (scope) => {
|
|
307
|
+
const rows = Array.from({ length: 260 }, (_, i) => `wide-${i + 1}-${'x'.repeat(72)}`);
|
|
308
|
+
put('wide-read.txt', rows.join('\n') + '\n');
|
|
309
|
+
const fullOut = await tool('read', { path: 'wide-read.txt' }, scope);
|
|
310
|
+
expectTrue(!String(fullOut).includes('[context-advisory]'), fullOut);
|
|
311
|
+
expectTrue(!String(fullOut).includes('next_call:'), fullOut);
|
|
312
|
+
const compactOut = await tool('read', { path: 'wide-read.txt', budget: 'compact' }, scope);
|
|
313
|
+
expectIncludes(compactOut, 'lines\t260');
|
|
314
|
+
expectTrue(!String(compactOut).includes('wide-130'), compactOut);
|
|
315
|
+
const rangeOut = await tool('read', { path: 'wide-read.txt', offset: 10, limit: 5 }, scope);
|
|
316
|
+
expectTrue(!String(rangeOut).includes('[context-advisory]'), rangeOut);
|
|
317
|
+
});
|
|
318
|
+
|
|
319
|
+
await scenario('truncated range read stays compact', async (scope) => {
|
|
320
|
+
const rows = Array.from({ length: 2200 }, (_, i) => `huge-${i + 1}-${'z'.repeat(80)}`);
|
|
321
|
+
put('truncated-range.txt', rows.join('\n') + '\n');
|
|
322
|
+
const out = await tool('read', { path: 'truncated-range.txt', offset: 0, limit: 2000 }, scope);
|
|
323
|
+
expectIncludes(out, '[output truncated');
|
|
324
|
+
expectTrue(!String(out).includes('next_call:'), out);
|
|
325
|
+
});
|
|
326
|
+
|
|
327
|
+
await scenario('summary read emits outline, hidden-body edit now succeeds (read-window gate retired)', async (scope) => {
|
|
328
|
+
put('summary-read.mjs', [
|
|
329
|
+
'import x from "x";',
|
|
330
|
+
'export function alpha() {',
|
|
331
|
+
' return 1;',
|
|
332
|
+
'}',
|
|
333
|
+
'const hiddenBody = "do not stream everything";',
|
|
334
|
+
'class Beta {}',
|
|
335
|
+
'',
|
|
336
|
+
].join('\n'));
|
|
337
|
+
const out = await tool('read', { path: 'summary-read.mjs', mode: 'summary', n: 5 }, scope);
|
|
338
|
+
expectIncludes(out, 'summary summary-read.mjs');
|
|
339
|
+
expectIncludes(out, 'symbols\t');
|
|
340
|
+
expectIncludes(out, 'export function alpha');
|
|
341
|
+
expectIncludes(out, 'class Beta');
|
|
342
|
+
expectTrue(!String(out).includes('next_call:'), out);
|
|
343
|
+
const editOut = await tool('edit', { path: 'summary-read.mjs', old_string: 'hiddenBody', new_string: 'VISIBLE_BODY' }, scope);
|
|
344
|
+
expectIncludes(editOut, 'Edited:');
|
|
345
|
+
expectTrue(!String(editOut).includes('next_call:'), editOut);
|
|
346
|
+
expectIncludes(get('summary-read.mjs'), 'VISIBLE_BODY');
|
|
347
|
+
});
|
|
348
|
+
|
|
349
|
+
await scenario('summary recognizes common language symbols', async (scope) => {
|
|
350
|
+
put('summary-langs.txt', [
|
|
351
|
+
'pub fn rust_entry() {}',
|
|
352
|
+
'func GoEntry() {}',
|
|
353
|
+
'public class CsEntry {}',
|
|
354
|
+
'public void JavaStyle() {}',
|
|
355
|
+
'',
|
|
356
|
+
].join('\n'));
|
|
357
|
+
const out = await tool('read', { path: 'summary-langs.txt', mode: 'summary', n: 10 }, scope);
|
|
358
|
+
expectIncludes(out, 'pub fn rust_entry');
|
|
359
|
+
expectIncludes(out, 'func GoEntry');
|
|
360
|
+
expectIncludes(out, 'public class CsEntry');
|
|
361
|
+
expectIncludes(out, 'public void JavaStyle');
|
|
362
|
+
expectTrue(!String(out).includes('next_call:'), out);
|
|
363
|
+
});
|
|
364
|
+
|
|
365
|
+
await scenario('missing grep and glob arguments stay concise', async (scope) => {
|
|
366
|
+
const grepOut = await tool('grep', {}, scope);
|
|
367
|
+
expectIncludes(grepOut, 'Error: grep requires pattern');
|
|
368
|
+
expectTrue(!String(grepOut).includes('next_call:'), grepOut);
|
|
369
|
+
const globOut = await tool('glob', {}, scope);
|
|
370
|
+
expectIncludes(globOut, 'Error: glob requires pattern');
|
|
371
|
+
expectTrue(!String(globOut).includes('next_call:'), globOut);
|
|
372
|
+
});
|
|
373
|
+
|
|
374
|
+
await scenario('grep regex error stays concise', async (scope) => {
|
|
375
|
+
const out = await tool('grep', { pattern: '(', path: '.' }, scope);
|
|
376
|
+
expectIncludes(out, 'Error:');
|
|
377
|
+
expectTrue(!String(out).includes('hint:'), out);
|
|
378
|
+
expectTrue(!String(out).includes('next_call:'), out);
|
|
379
|
+
});
|
|
380
|
+
|
|
381
|
+
await scenario('grep aliases land one-shot', async (scope) => {
|
|
382
|
+
put('grep-alias/a.mjs', 'const needleValue = 1;\n');
|
|
383
|
+
put('grep-alias/b.txt', 'needleValue in text\n');
|
|
384
|
+
const out = await tool('grep', {
|
|
385
|
+
query: 'needleValue',
|
|
386
|
+
root: 'grep-alias',
|
|
387
|
+
file_pattern: '*.mjs',
|
|
388
|
+
mode: 'content',
|
|
389
|
+
}, scope);
|
|
390
|
+
expectIncludes(out, 'a.mjs');
|
|
391
|
+
expectIncludes(out, 'needleValue');
|
|
392
|
+
expectTrue(!String(out).includes('next_call:'), out);
|
|
393
|
+
expectTrue(!String(out).includes('b.txt'), out);
|
|
394
|
+
});
|
|
395
|
+
|
|
396
|
+
await scenario('grep and glob typo tool names canonicalize', async (scope) => {
|
|
397
|
+
put('typo-tools/a.mjs', 'const typoNeedle = 1;\n');
|
|
398
|
+
put('typo-tools/b.txt', 'typoNeedle in text\n');
|
|
399
|
+
const grepOut = await tool('grop', {
|
|
400
|
+
pattern: 'typoNeedle',
|
|
401
|
+
path: 'typo-tools',
|
|
402
|
+
glob: '*.mjs',
|
|
403
|
+
output_mode: 'content',
|
|
404
|
+
}, scope);
|
|
405
|
+
expectIncludes(grepOut, 'typoNeedle');
|
|
406
|
+
expectTrue(!String(grepOut).includes('next_call:'), grepOut);
|
|
407
|
+
const globOut = await tool('glbo', { glob: '*.mjs', root: 'typo-tools' }, scope);
|
|
408
|
+
expectIncludes(globOut, 'typo-tools/a.mjs');
|
|
409
|
+
expectTrue(!String(globOut).includes('next_call:'), globOut);
|
|
410
|
+
});
|
|
411
|
+
|
|
412
|
+
await scenario('grep file and count modes stay concise', async (scope) => {
|
|
413
|
+
put('grep-follow/a.mjs', 'const followNeedle = 1;\nconst followNeedleAgain = 2;\n');
|
|
414
|
+
put('grep-follow/b.txt', 'no match\n');
|
|
415
|
+
const filesOut = await tool('grep', {
|
|
416
|
+
pattern: 'followNeedle',
|
|
417
|
+
path: 'grep-follow',
|
|
418
|
+
glob: '*.mjs',
|
|
419
|
+
}, scope);
|
|
420
|
+
expectIncludes(filesOut, 'grep-follow/a.mjs');
|
|
421
|
+
expectTrue(!String(filesOut).includes('next_call:'), filesOut);
|
|
422
|
+
const countOut = await tool('grep', {
|
|
423
|
+
pattern: 'followNeedle',
|
|
424
|
+
path: 'grep-follow',
|
|
425
|
+
glob: '*.mjs',
|
|
426
|
+
mode: 'count',
|
|
427
|
+
}, scope);
|
|
428
|
+
expectIncludes(countOut, 'grep-follow/a.mjs:2');
|
|
429
|
+
expectTrue(!String(countOut).includes('next_call:'), countOut);
|
|
430
|
+
});
|
|
431
|
+
|
|
432
|
+
await scenario('glob aliases land one-shot', async (scope) => {
|
|
433
|
+
put('glob-alias/a.mjs', 'a\n');
|
|
434
|
+
put('glob-alias/b.txt', 'b\n');
|
|
435
|
+
const out = await tool('glob', { glob: '*.mjs', root: 'glob-alias' }, scope);
|
|
436
|
+
expectIncludes(out, 'glob-alias/a.mjs');
|
|
437
|
+
expectTrue(!String(out).includes('next_call:'), out);
|
|
438
|
+
expectTrue(!String(out).includes('glob-alias/b.txt'), out);
|
|
439
|
+
});
|
|
440
|
+
|
|
441
|
+
await scenario('V4A patch touched paths extracted', async () => {
|
|
442
|
+
const touched = extractTouchedPathsFromPatch([
|
|
443
|
+
'*** Begin Patch',
|
|
444
|
+
'*** Update File: "src/a name.mjs"\t2026-05-21',
|
|
445
|
+
'@@ anchor',
|
|
446
|
+
'-old',
|
|
447
|
+
'+new',
|
|
448
|
+
'*** Add File: src/b.mjs',
|
|
449
|
+
'+new file',
|
|
450
|
+
'*** Delete File: src/c.mjs',
|
|
451
|
+
'*** End Patch',
|
|
452
|
+
].join('\n'));
|
|
453
|
+
expectEquals(JSON.stringify(touched), JSON.stringify(['src/a name.mjs', 'src/b.mjs', 'src/c.mjs']));
|
|
454
|
+
});
|
|
455
|
+
|
|
456
|
+
await scenario('unified patch touched paths ignore tab metadata', async () => {
|
|
457
|
+
const touched = extractTouchedPathsFromPatch([
|
|
458
|
+
'--- a/src/a.mjs\t2026-05-21 10:00:00',
|
|
459
|
+
'+++ b/src/a.mjs\t2026-05-21 10:01:00',
|
|
460
|
+
'@@ -1 +1 @@',
|
|
461
|
+
'-old',
|
|
462
|
+
'+new',
|
|
463
|
+
].join('\n'));
|
|
464
|
+
expectEquals(JSON.stringify(touched), JSON.stringify(['src/a.mjs']));
|
|
465
|
+
});
|
|
466
|
+
|
|
467
|
+
await scenario('apply_patch rejected hunk returns compact context guidance', async (scope) => {
|
|
468
|
+
put('patch-guidance.txt', 'alpha\nbeta\ngamma\n');
|
|
469
|
+
const out = await patchTool({
|
|
470
|
+
patch: [
|
|
471
|
+
'--- a/patch-guidance.txt',
|
|
472
|
+
'+++ b/patch-guidance.txt',
|
|
473
|
+
'@@ -1,3 +1,3 @@',
|
|
474
|
+
' alpha',
|
|
475
|
+
'-missing',
|
|
476
|
+
'+BETA',
|
|
477
|
+
' gamma',
|
|
478
|
+
].join('\n') + '\n',
|
|
479
|
+
dry_run: true,
|
|
480
|
+
fuzzy: false,
|
|
481
|
+
}, scope);
|
|
482
|
+
expectIncludes(out, 'hunk rejected');
|
|
483
|
+
expectIncludes(out, 'expected first old/context line');
|
|
484
|
+
expectIncludes(out, 'nearest line');
|
|
485
|
+
});
|
|
486
|
+
|
|
487
|
+
await scenario('apply_patch count-mismatch unified falls back through V4A', async (scope) => {
|
|
488
|
+
put('patch-count-fallback.txt', 'a\nb\nc\nd\n');
|
|
489
|
+
const patch = [
|
|
490
|
+
'--- a/patch-count-fallback.txt',
|
|
491
|
+
'+++ b/patch-count-fallback.txt',
|
|
492
|
+
'@@ -1,1 +1,1 @@',
|
|
493
|
+
'-a',
|
|
494
|
+
'+A',
|
|
495
|
+
'+extra',
|
|
496
|
+
'@@ -4,1 +4,1 @@',
|
|
497
|
+
'-d',
|
|
498
|
+
'+D',
|
|
499
|
+
].join('\n') + '\n';
|
|
500
|
+
const out = await patchTool({ patch }, scope);
|
|
501
|
+
expectIncludes(out, 'mutation_route: source=apply_patch engine=v4a-patch reason=unified-count-fallback');
|
|
502
|
+
expectIncludes(out, 'applied 1');
|
|
503
|
+
expectEquals(get('patch-count-fallback.txt'), 'A\nextra\nb\nc\nD\n');
|
|
504
|
+
|
|
505
|
+
put('patch-count-strict.txt', 'a\nb\n');
|
|
506
|
+
let strictError = '';
|
|
507
|
+
try {
|
|
508
|
+
await patchTool({
|
|
509
|
+
patch: [
|
|
510
|
+
'--- a/patch-count-strict.txt',
|
|
511
|
+
'+++ b/patch-count-strict.txt',
|
|
512
|
+
'@@ -1,1 +1,1 @@',
|
|
513
|
+
'-a',
|
|
514
|
+
'+A',
|
|
515
|
+
'+extra',
|
|
516
|
+
].join('\n') + '\n',
|
|
517
|
+
format: 'unified',
|
|
518
|
+
}, scope);
|
|
519
|
+
} catch (err) {
|
|
520
|
+
strictError = String(err?.message || err);
|
|
521
|
+
}
|
|
522
|
+
expectIncludes(strictError, 'parse failed');
|
|
523
|
+
});
|
|
524
|
+
|
|
525
|
+
await scenario('apply_patch success reports shared mutation route', async (scope) => {
|
|
526
|
+
put('patch-route.txt', 'before\n');
|
|
527
|
+
const out = await patchTool({
|
|
528
|
+
patch: [
|
|
529
|
+
'--- a/patch-route.txt',
|
|
530
|
+
'+++ b/patch-route.txt',
|
|
531
|
+
'@@ -1,1 +1,1 @@',
|
|
532
|
+
'-before',
|
|
533
|
+
'+after',
|
|
534
|
+
].join('\n') + '\n',
|
|
535
|
+
}, scope);
|
|
536
|
+
expectIncludes(out, 'mutation_route: source=apply_patch engine=unified-patch reason=direct');
|
|
537
|
+
expectIncludes(out, 'applied 1');
|
|
538
|
+
expectEquals(get('patch-route.txt'), 'after\n');
|
|
539
|
+
});
|
|
540
|
+
|
|
541
|
+
await scenario('apply_patch long failed hunk stays compact', async (scope) => {
|
|
542
|
+
put('patch-guidance-long.txt', Array.from({ length: 14 }, (_, i) => `line-${i + 1}`).join('\n') + '\n');
|
|
543
|
+
const patchLines = [
|
|
544
|
+
'--- a/patch-guidance-long.txt',
|
|
545
|
+
'+++ b/patch-guidance-long.txt',
|
|
546
|
+
'@@ -1,14 +1,14 @@',
|
|
547
|
+
' line-1',
|
|
548
|
+
' line-2',
|
|
549
|
+
' line-3',
|
|
550
|
+
' line-4',
|
|
551
|
+
' line-5',
|
|
552
|
+
' line-6',
|
|
553
|
+
' line-7',
|
|
554
|
+
' line-8',
|
|
555
|
+
' line-9',
|
|
556
|
+
'-missing',
|
|
557
|
+
'+LINE-10',
|
|
558
|
+
' line-11',
|
|
559
|
+
' line-12',
|
|
560
|
+
' line-13',
|
|
561
|
+
' line-14',
|
|
562
|
+
];
|
|
563
|
+
const out = await patchTool({ patch: patchLines.join('\n') + '\n', dry_run: true, fuzzy: false }, scope);
|
|
564
|
+
expectIncludes(out, 'hunk rejected');
|
|
565
|
+
expectIncludes(out, 'expected first old/context line');
|
|
566
|
+
});
|
|
567
|
+
|
|
568
|
+
await scenario('long foreground bash watch is rejected compactly', async (scope) => {
|
|
569
|
+
const out = await tool('bash', { command: 'gh run watch 123456 --exit-status', timeout: 600000 }, scope);
|
|
570
|
+
expectIncludes(out, 'long foreground command detected');
|
|
571
|
+
expectTrue(!String(out).includes('run_in_background:true'), out);
|
|
572
|
+
});
|
|
573
|
+
|
|
574
|
+
await scenario('long foreground PowerShell sleep is rejected compactly', async (scope) => {
|
|
575
|
+
const out = await tool('bash', { command: 'Start-Sleep -Seconds 75; "chrome now"', timeout: 90000 }, scope);
|
|
576
|
+
expectIncludes(out, 'long foreground command detected');
|
|
577
|
+
});
|
|
578
|
+
|
|
579
|
+
if (failed > 0) {
|
|
580
|
+
console.error(`[io-smoke] failed=${failed} passed=${passed}`);
|
|
581
|
+
console.error(`[io-smoke] fixtures preserved at ${WORKDIR}`);
|
|
582
|
+
await closeNativePatchServerForTests().catch(() => {});
|
|
583
|
+
process.exit(1);
|
|
584
|
+
}
|
|
585
|
+
|
|
586
|
+
console.log(`[io-smoke] all ${passed} scenarios passed`);
|
|
587
|
+
await closeNativePatchServerForTests().catch(() => {});
|
|
588
|
+
try { rmSync(WORKDIR, { recursive: true, force: true }); } catch {}
|
|
589
|
+
try { rmSync(DATA_DIR, { recursive: true, force: true }); } catch {}
|
|
590
|
+
|
|
591
|
+
async function scenario(name, fn) {
|
|
592
|
+
const scope = `${BASE_SCOPE}-${passed + failed + 1}`;
|
|
593
|
+
try {
|
|
594
|
+
await fn(scope);
|
|
595
|
+
passed += 1;
|
|
596
|
+
console.log(`[io-smoke] PASS ${passed + failed}: ${name}`);
|
|
597
|
+
} catch (err) {
|
|
598
|
+
failed += 1;
|
|
599
|
+
console.error(`[io-smoke] FAIL ${passed + failed}: ${name}`);
|
|
600
|
+
console.error(err?.stack || String(err));
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
|
|
604
|
+
async function tool(name, args, scope) {
|
|
605
|
+
return executeBuiltinTool(name, args, WORKDIR, { readStateScope: scope, sessionId: scope });
|
|
606
|
+
}
|
|
607
|
+
|
|
608
|
+
async function patchTool(args, scope) {
|
|
609
|
+
return executePatchTool('apply_patch', args, WORKDIR, { readStateScope: scope, sessionId: scope });
|
|
610
|
+
}
|
|
611
|
+
|
|
612
|
+
function abs(rel) {
|
|
613
|
+
return join(WORKDIR, rel);
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
function put(rel, content) {
|
|
617
|
+
const path = abs(rel);
|
|
618
|
+
mkdirSync(dirname(path), { recursive: true });
|
|
619
|
+
writeFileSync(path, content);
|
|
620
|
+
}
|
|
621
|
+
|
|
622
|
+
function get(rel) {
|
|
623
|
+
return readFileSync(abs(rel), 'utf8');
|
|
624
|
+
}
|
|
625
|
+
|
|
626
|
+
function rangeLines(prefix, count) {
|
|
627
|
+
return Array.from({ length: count }, (_, i) => `${prefix}-${i + 1}`).join('\n') + '\n';
|
|
628
|
+
}
|
|
629
|
+
|
|
630
|
+
function linesWith(count, lineNumber, value) {
|
|
631
|
+
return Array.from({ length: count }, (_, i) => (i + 1 === lineNumber ? value : `line-${i + 1}`)).join('\n') + '\n';
|
|
632
|
+
}
|
|
633
|
+
|
|
634
|
+
function expectIncludes(actual, needle) {
|
|
635
|
+
if (!String(actual).includes(needle)) {
|
|
636
|
+
throw new Error(`expected output to include ${JSON.stringify(needle)}, got ${JSON.stringify(String(actual).slice(0, 500))}`);
|
|
637
|
+
}
|
|
638
|
+
}
|
|
639
|
+
|
|
640
|
+
function objectContains(actual, expected) {
|
|
641
|
+
for (const [key, value] of Object.entries(expected)) {
|
|
642
|
+
if (actual?.[key] !== value) return false;
|
|
643
|
+
}
|
|
644
|
+
return true;
|
|
645
|
+
}
|
|
646
|
+
|
|
647
|
+
function expectNextCall(actual, toolName, expectedArgs) {
|
|
648
|
+
const calls = parseJsonNextCalls(actual);
|
|
649
|
+
const found = calls.find((call) => call.tool === toolName && objectContains(call.args, expectedArgs));
|
|
650
|
+
if (!found) {
|
|
651
|
+
throw new Error(`expected next_call ${toolName}(${JSON.stringify(expectedArgs)}) in ${JSON.stringify(String(actual).slice(0, 800))}; parsed=${JSON.stringify(calls)}`);
|
|
652
|
+
}
|
|
653
|
+
}
|
|
654
|
+
|
|
655
|
+
function expectEquals(actual, expected) {
|
|
656
|
+
if (actual !== expected) {
|
|
657
|
+
throw new Error(`expected ${JSON.stringify(expected)}, got ${JSON.stringify(actual)}`);
|
|
658
|
+
}
|
|
659
|
+
}
|
|
660
|
+
|
|
661
|
+
function expectTrue(value, detail = '') {
|
|
662
|
+
if (!value) throw new Error(`expected truthy value${detail ? `: ${detail}` : ''}`);
|
|
663
|
+
}
|
|
664
|
+
|
|
665
|
+
function sleep(ms) {
|
|
666
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
667
|
+
}
|