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,196 @@
|
|
|
1
|
+
// Server-less, preview-first symbol rename.
|
|
2
|
+
//
|
|
3
|
+
// Mixdog has no semantic LSP. This tool reuses the existing code-graph
|
|
4
|
+
// find_references API to collect every reference site for a symbol, then —
|
|
5
|
+
// BY DEFAULT — returns a PREVIEW only: the {file,line,col,context} sites that
|
|
6
|
+
// WOULD change, plus an explicit warning that the match is graph-based
|
|
7
|
+
// (approximate, not LSP-semantic) and may include same-named-but-different
|
|
8
|
+
// symbols. The destructive rewrite runs ONLY when apply===true, and replaces
|
|
9
|
+
// the symbol EXACTLY at the {line,col} spans find_references returned —
|
|
10
|
+
// verifying the slice equals the symbol before splicing. It never blanket-
|
|
11
|
+
// replaces a line, so same-line string/comment occurrences the reference
|
|
12
|
+
// search did NOT sanction are left untouched, `$`-identifiers can't mis-match
|
|
13
|
+
// inside larger identifiers, and original (possibly mixed) line endings are
|
|
14
|
+
// preserved verbatim.
|
|
15
|
+
import { readFileSync, writeFileSync, statSync } from 'fs';
|
|
16
|
+
import { isAbsolute, resolve as pathResolve } from 'path';
|
|
17
|
+
import { executeCodeGraphTool, markCodeGraphDirtyPaths } from '../code-graph.mjs';
|
|
18
|
+
import { invalidateBuiltinResultCache } from './cache-layers.mjs';
|
|
19
|
+
|
|
20
|
+
const APPROX_WARNING =
|
|
21
|
+
'APPROXIMATE: matches come from the code-graph (text/word-boundary based), '
|
|
22
|
+
+ 'NOT an LSP-semantic resolver. They may include same-named-but-different '
|
|
23
|
+
+ 'symbols (shadowed locals, unrelated properties, etc.). Review every site '
|
|
24
|
+
+ 'before applying.';
|
|
25
|
+
|
|
26
|
+
// A valid JS/TS/Py-style identifier — required so we never attempt a rewrite
|
|
27
|
+
// of an arbitrary string that would corrupt source via word-boundary regex.
|
|
28
|
+
function _isIdentifier(s) {
|
|
29
|
+
return typeof s === 'string' && /^[A-Za-z_$][\w$]*$/.test(s);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// Parse find_references text output. Each hit line is `rel:line:col context`.
|
|
33
|
+
// Diagnostic/footer lines (starting with `(` or `[`) and blanks are skipped.
|
|
34
|
+
function _parseReferenceLines(text) {
|
|
35
|
+
const sites = [];
|
|
36
|
+
if (typeof text !== 'string') return sites;
|
|
37
|
+
const re = /^(.+?):(\d+):(\d+)\s{2,}(.*)$/;
|
|
38
|
+
for (const raw of text.split(/\r?\n/)) {
|
|
39
|
+
const line = raw.replace(/\s+$/, '');
|
|
40
|
+
if (!line) continue;
|
|
41
|
+
if (line.startsWith('(') || line.startsWith('[') || line.startsWith('#')) continue;
|
|
42
|
+
const m = re.exec(line);
|
|
43
|
+
if (!m) continue;
|
|
44
|
+
sites.push({ file: m[1], line: +m[2], col: +m[3], context: m[4] });
|
|
45
|
+
}
|
|
46
|
+
return sites;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
function _absFor(file, workDir) {
|
|
50
|
+
return isAbsolute(file) ? file : pathResolve(workDir, file);
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// Split content into physical lines while preserving each line's own trailing
|
|
54
|
+
// terminator (\r\n, \n, or '' for the last line). This lets us rewrite a single
|
|
55
|
+
// line's text and re-join with the EXACT original endings — no normalization of
|
|
56
|
+
// mixed line endings.
|
|
57
|
+
function _splitKeepEol(content) {
|
|
58
|
+
const out = [];
|
|
59
|
+
const re = /\r\n|\n|\r/g;
|
|
60
|
+
let last = 0;
|
|
61
|
+
let m = null;
|
|
62
|
+
while ((m = re.exec(content))) {
|
|
63
|
+
out.push({ text: content.slice(last, m.index), eol: m[0] });
|
|
64
|
+
last = m.index + m[0].length;
|
|
65
|
+
}
|
|
66
|
+
out.push({ text: content.slice(last), eol: '' });
|
|
67
|
+
return out;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
// Apply the rename ONLY at the exact {line,col} spans find_references returned.
|
|
71
|
+
// For each site we verify the slice [col-1 .. col-1+symbol.length] === symbol
|
|
72
|
+
// before splicing; if it doesn't match (stale graph, shifted line, or an
|
|
73
|
+
// occurrence the search didn't sanction) the site is recorded as skipped and
|
|
74
|
+
// nothing on that line is touched. Identity-check + slice splice replaces the
|
|
75
|
+
// previous `\b`-regex line rewrite, which mishandled `$`-identifiers and
|
|
76
|
+
// clobbered same-line non-reference occurrences.
|
|
77
|
+
function _applyRename(sites, symbol, newName, workDir) {
|
|
78
|
+
const byFile = new Map();
|
|
79
|
+
for (const s of sites) {
|
|
80
|
+
const abs = _absFor(s.file, workDir);
|
|
81
|
+
if (!byFile.has(abs)) byFile.set(abs, []);
|
|
82
|
+
byFile.get(abs).push(s);
|
|
83
|
+
}
|
|
84
|
+
const changed = [];
|
|
85
|
+
const skipped = [];
|
|
86
|
+
let replacements = 0;
|
|
87
|
+
for (const [abs, fileSites] of byFile) {
|
|
88
|
+
let content;
|
|
89
|
+
try { content = readFileSync(abs, 'utf8'); }
|
|
90
|
+
catch (err) {
|
|
91
|
+
for (const s of fileSites) skipped.push({ ...s, reason: `read failed: ${err && err.message ? err.message : String(err)}` });
|
|
92
|
+
continue;
|
|
93
|
+
}
|
|
94
|
+
const lines = _splitKeepEol(content);
|
|
95
|
+
// Replace from rightmost column first within a line so earlier splices
|
|
96
|
+
// don't shift the offsets of later ones on the same line.
|
|
97
|
+
const ordered = fileSites.slice().sort((a, b) => (a.line - b.line) || (b.col - a.col));
|
|
98
|
+
let fileTouched = false;
|
|
99
|
+
for (const s of ordered) {
|
|
100
|
+
const idx = s.line - 1;
|
|
101
|
+
if (idx < 0 || idx >= lines.length) { skipped.push({ ...s, reason: 'line out of range' }); continue; }
|
|
102
|
+
const start = s.col - 1;
|
|
103
|
+
const text = lines[idx].text;
|
|
104
|
+
if (start < 0 || start + symbol.length > text.length) { skipped.push({ ...s, reason: 'column out of range' }); continue; }
|
|
105
|
+
const slice = text.slice(start, start + symbol.length);
|
|
106
|
+
if (slice !== symbol) { skipped.push({ ...s, reason: `slice mismatch (got ${JSON.stringify(slice)})` }); continue; }
|
|
107
|
+
lines[idx].text = text.slice(0, start) + newName + text.slice(start + symbol.length);
|
|
108
|
+
replacements += 1;
|
|
109
|
+
fileTouched = true;
|
|
110
|
+
}
|
|
111
|
+
if (fileTouched) {
|
|
112
|
+
writeFileSync(abs, lines.map((l) => l.text + l.eol).join(''), 'utf8');
|
|
113
|
+
changed.push(abs);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
return { changed, replacements, skipped };
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
export async function executeRenameTool(args, workDir, signal = null) {
|
|
120
|
+
try {
|
|
121
|
+
const symbol = (args && typeof args.symbol === 'string') ? args.symbol.trim() : '';
|
|
122
|
+
const newName = (args && typeof args.new_name === 'string') ? args.new_name.trim() : '';
|
|
123
|
+
const file = (args && typeof args.file === 'string' && args.file.trim()) ? args.file.trim() : '';
|
|
124
|
+
const apply = args && args.apply === true;
|
|
125
|
+
if (!symbol) return 'Error: rename requires "symbol"';
|
|
126
|
+
if (!newName) return 'Error: rename requires "new_name"';
|
|
127
|
+
if (!_isIdentifier(symbol)) return `Error: rename "symbol" must be a valid identifier (got ${JSON.stringify(args.symbol)})`;
|
|
128
|
+
if (!_isIdentifier(newName)) return `Error: rename "new_name" must be a valid identifier (got ${JSON.stringify(args.new_name)})`;
|
|
129
|
+
if (symbol === newName) return 'Error: rename "new_name" is identical to "symbol"; nothing to do';
|
|
130
|
+
|
|
131
|
+
// Collect reference sites via the code-graph references mode.
|
|
132
|
+
const refArgs = { mode: 'references', symbol };
|
|
133
|
+
if (file) refArgs.file = file;
|
|
134
|
+
const refText = await executeCodeGraphTool('code_graph', refArgs, workDir, signal);
|
|
135
|
+
if (typeof refText === 'string' && refText.startsWith('Error:')) {
|
|
136
|
+
return refText;
|
|
137
|
+
}
|
|
138
|
+
const sites = _parseReferenceLines(refText);
|
|
139
|
+
|
|
140
|
+
if (sites.length === 0) {
|
|
141
|
+
return JSON.stringify({
|
|
142
|
+
ok: true,
|
|
143
|
+
mode: apply ? 'apply' : 'preview',
|
|
144
|
+
symbol,
|
|
145
|
+
new_name: newName,
|
|
146
|
+
warning: APPROX_WARNING,
|
|
147
|
+
count: 0,
|
|
148
|
+
sites: [],
|
|
149
|
+
note: 'no reference sites found — nothing to rename',
|
|
150
|
+
}, null, 2);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
// DEFAULT: preview only. Never mutate on the default call. The CORE of
|
|
154
|
+
// the payload is the approximate-match warning plus the change-list of
|
|
155
|
+
// {file,line,col,context} sites that WOULD change; ok/mode/count/hint are
|
|
156
|
+
// secondary metadata.
|
|
157
|
+
if (!apply) {
|
|
158
|
+
return JSON.stringify({
|
|
159
|
+
warning: APPROX_WARNING,
|
|
160
|
+
changes: sites,
|
|
161
|
+
ok: true,
|
|
162
|
+
mode: 'preview',
|
|
163
|
+
symbol,
|
|
164
|
+
new_name: newName,
|
|
165
|
+
count: sites.length,
|
|
166
|
+
hint: 'Re-run with apply:true to perform the rename at these sites.',
|
|
167
|
+
}, null, 2);
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
// apply===true: perform the gated rewrite at the exact sanctioned spans.
|
|
171
|
+
const { changed, replacements, skipped } = _applyRename(sites, symbol, newName, workDir);
|
|
172
|
+
if (changed.length) {
|
|
173
|
+
try { invalidateBuiltinResultCache(changed); } catch {}
|
|
174
|
+
try { markCodeGraphDirtyPaths(changed); } catch {}
|
|
175
|
+
}
|
|
176
|
+
const result = {
|
|
177
|
+
ok: true,
|
|
178
|
+
mode: 'apply',
|
|
179
|
+
symbol,
|
|
180
|
+
new_name: newName,
|
|
181
|
+
warning: APPROX_WARNING,
|
|
182
|
+
sites_considered: sites.length,
|
|
183
|
+
replacements,
|
|
184
|
+
files_changed: changed.map((p) => (p.startsWith(workDir) ? p.slice(workDir.length).replace(/^[\\/]/, '') : p)),
|
|
185
|
+
};
|
|
186
|
+
if (skipped.length) {
|
|
187
|
+
result.skipped = skipped;
|
|
188
|
+
result.skipped_warning = `${skipped.length} site(s) were NOT renamed (offset out of range or the slice did not match the symbol). They were left untouched; review them manually.`;
|
|
189
|
+
}
|
|
190
|
+
return JSON.stringify(result, null, 2);
|
|
191
|
+
} catch (err) {
|
|
192
|
+
return JSON.stringify({ ok: false, error: `rename failed: ${err && err.message ? err.message : String(err)}` }, null, 2);
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
export default executeRenameTool;
|
|
@@ -0,0 +1,422 @@
|
|
|
1
|
+
import { spawn, execFileSync } from 'child_process';
|
|
2
|
+
import { existsSync } from 'node:fs';
|
|
3
|
+
import { join } from 'node:path';
|
|
4
|
+
|
|
5
|
+
let _rgExecutableResolved = null;
|
|
6
|
+
|
|
7
|
+
function _resolveRgExecutable() {
|
|
8
|
+
const isWin = process.platform === 'win32';
|
|
9
|
+
try {
|
|
10
|
+
const cmd = isWin ? 'where' : 'which';
|
|
11
|
+
const out = execFileSync(cmd, ['rg'], { encoding: 'utf8', windowsHide: true }).trim();
|
|
12
|
+
const first = out.split(/\r?\n/).map((l) => l.trim()).find(Boolean);
|
|
13
|
+
if (first && existsSync(first)) return first;
|
|
14
|
+
} catch { /* fall through to PATH scan */ }
|
|
15
|
+
const pathSep = isWin ? ';' : ':';
|
|
16
|
+
const dirs = String(process.env.PATH || '').split(pathSep).filter(Boolean);
|
|
17
|
+
const pathext = isWin
|
|
18
|
+
? String(process.env.PATHEXT || '.EXE;.CMD;.BAT').split(';').map((e) => e.toLowerCase())
|
|
19
|
+
: [''];
|
|
20
|
+
const names = isWin ? ['rg.exe', 'rg'] : ['rg'];
|
|
21
|
+
for (const dir of dirs) {
|
|
22
|
+
for (const name of names) {
|
|
23
|
+
const candidate = join(dir, name);
|
|
24
|
+
if (!existsSync(candidate)) continue;
|
|
25
|
+
if (isWin) {
|
|
26
|
+
const ext = candidate.slice(candidate.lastIndexOf('.')).toLowerCase();
|
|
27
|
+
if (ext && !pathext.includes(ext)) continue;
|
|
28
|
+
}
|
|
29
|
+
return candidate;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
return 'rg';
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function rgExecutable() {
|
|
36
|
+
if (_rgExecutableResolved === null) {
|
|
37
|
+
_rgExecutableResolved = _resolveRgExecutable();
|
|
38
|
+
}
|
|
39
|
+
return _rgExecutableResolved;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// When _resolveRgExecutable() exhausts `where`/`which` and the PATH scan and
|
|
43
|
+
// falls back to the bare 'rg', a later spawn would die with a raw ENOENT. Probe
|
|
44
|
+
// the fallback so we can surface a clear, actionable error instead. Success is
|
|
45
|
+
// cached permanently; failure is cached for only RG_FALLBACK_FAIL_TTL_MS so
|
|
46
|
+
// installing rg mid-session recovers without restarting the daemon.
|
|
47
|
+
const RG_FALLBACK_FAIL_TTL_MS = 30000;
|
|
48
|
+
let _rgFallbackUsable = null;
|
|
49
|
+
let _rgFallbackFailAt = 0;
|
|
50
|
+
function rgFallbackUsable() {
|
|
51
|
+
if (_rgFallbackUsable === true) return true;
|
|
52
|
+
if (_rgFallbackUsable === false && (Date.now() - _rgFallbackFailAt) < RG_FALLBACK_FAIL_TTL_MS) {
|
|
53
|
+
return false;
|
|
54
|
+
}
|
|
55
|
+
try {
|
|
56
|
+
execFileSync('rg', ['--version'], { stdio: 'ignore', windowsHide: true, timeout: 3000 });
|
|
57
|
+
_rgFallbackUsable = true;
|
|
58
|
+
} catch {
|
|
59
|
+
_rgFallbackUsable = false;
|
|
60
|
+
_rgFallbackFailAt = Date.now();
|
|
61
|
+
}
|
|
62
|
+
return _rgFallbackUsable;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// Throws a clear error only when the bare 'rg' fallback was reached AND it is
|
|
66
|
+
// not actually runnable. A real resolved path short-circuits with no probe, so
|
|
67
|
+
// behavior is unchanged whenever rg exists.
|
|
68
|
+
function assertRgAvailable() {
|
|
69
|
+
if (rgExecutable() !== 'rg') return;
|
|
70
|
+
if (rgFallbackUsable()) return;
|
|
71
|
+
const e = new Error('ripgrep (rg) not found on PATH — install ripgrep or add it to PATH');
|
|
72
|
+
e.code = 'ERG_NOT_FOUND';
|
|
73
|
+
throw e;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// Cap rg stdout accumulation so a runaway producer (huge repo, accidental
|
|
77
|
+
// match-all) cannot balloon the JS string heap. Mirrors CC's ripgrep.ts cap.
|
|
78
|
+
const MAX_RG_STDOUT_BYTES = 20 * 1024 * 1024; // 20MB
|
|
79
|
+
|
|
80
|
+
// SIGTERM → grace → force kill; hard Promise settle if 'close' never fires
|
|
81
|
+
// (mirrors shell-command.mjs treeKill + _treeKillForceSettle timings).
|
|
82
|
+
const RG_KILL_GRACE_MS = 3000;
|
|
83
|
+
const RG_FORCE_SETTLE_SLACK_MS = 5000;
|
|
84
|
+
|
|
85
|
+
function _rgProcGone(proc) {
|
|
86
|
+
return !proc || proc.exitCode != null || proc.signalCode != null;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
function _escalateRgKill(proc) {
|
|
90
|
+
if (_rgProcGone(proc)) return;
|
|
91
|
+
const pid = proc.pid;
|
|
92
|
+
if (!pid) return;
|
|
93
|
+
try {
|
|
94
|
+
if (process.platform === 'win32') {
|
|
95
|
+
spawn('taskkill', ['/pid', String(pid), '/t', '/f'], {
|
|
96
|
+
windowsHide: true,
|
|
97
|
+
stdio: 'ignore',
|
|
98
|
+
});
|
|
99
|
+
} else {
|
|
100
|
+
try {
|
|
101
|
+
proc.kill('SIGKILL');
|
|
102
|
+
} catch { /* ignore */ }
|
|
103
|
+
}
|
|
104
|
+
} catch { /* ignore */ }
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
function _killRgProc(proc, timers) {
|
|
108
|
+
if (_rgProcGone(proc)) return;
|
|
109
|
+
try {
|
|
110
|
+
proc.kill('SIGTERM');
|
|
111
|
+
} catch { /* ignore */ }
|
|
112
|
+
if (timers.killGrace) {
|
|
113
|
+
clearTimeout(timers.killGrace);
|
|
114
|
+
timers.killGrace = null;
|
|
115
|
+
}
|
|
116
|
+
timers.killGrace = setTimeout(() => {
|
|
117
|
+
timers.killGrace = null;
|
|
118
|
+
_escalateRgKill(proc);
|
|
119
|
+
}, RG_KILL_GRACE_MS);
|
|
120
|
+
if (timers.killGrace.unref) timers.killGrace.unref();
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
function _clearRgTimeoutOnly(timers) {
|
|
124
|
+
if (timers.timeout) {
|
|
125
|
+
clearTimeout(timers.timeout);
|
|
126
|
+
timers.timeout = null;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
function _clearRgTimers(timers) {
|
|
131
|
+
_clearRgTimeoutOnly(timers);
|
|
132
|
+
if (timers.killGrace) {
|
|
133
|
+
clearTimeout(timers.killGrace);
|
|
134
|
+
timers.killGrace = null;
|
|
135
|
+
}
|
|
136
|
+
if (timers.forceSettle) {
|
|
137
|
+
clearTimeout(timers.forceSettle);
|
|
138
|
+
timers.forceSettle = null;
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
function _armRgForceSettle({ timeoutMs, isSettled, timers, proc, onForceSettle }) {
|
|
143
|
+
if (timers.forceSettle) clearTimeout(timers.forceSettle);
|
|
144
|
+
timers.forceSettle = setTimeout(() => {
|
|
145
|
+
timers.forceSettle = null;
|
|
146
|
+
if (isSettled()) return;
|
|
147
|
+
// Hard deadline: escalate immediately — do not schedule grace then clear it.
|
|
148
|
+
_escalateRgKill(proc);
|
|
149
|
+
onForceSettle();
|
|
150
|
+
}, timeoutMs + RG_FORCE_SETTLE_SLACK_MS);
|
|
151
|
+
if (timers.forceSettle.unref) timers.forceSettle.unref();
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// Ripgrep wrapper. Ripgrep occasionally fails with EAGAIN on Windows when
|
|
155
|
+
// thread/resource pressure spikes. On EAGAIN we retry once with `-j 1` to
|
|
156
|
+
// force single-threaded execution; rg exit code 1 is "no matches" and is
|
|
157
|
+
// surfaced as empty stdout so callers can render "(no matches)" uniformly.
|
|
158
|
+
function spawnRg(argsList, execOptions) {
|
|
159
|
+
const timeoutMs = Number(execOptions?.timeout ?? 20000);
|
|
160
|
+
return new Promise((resolve, reject) => {
|
|
161
|
+
const proc = spawn(rgExecutable(), argsList, {
|
|
162
|
+
cwd: execOptions?.cwd,
|
|
163
|
+
env: execOptions?.env || process.env,
|
|
164
|
+
windowsHide: true,
|
|
165
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
166
|
+
});
|
|
167
|
+
let stdout = '';
|
|
168
|
+
let stderr = '';
|
|
169
|
+
let timedOut = false;
|
|
170
|
+
/** @type {'timeout' | 'cap' | null} */
|
|
171
|
+
let killReason = null;
|
|
172
|
+
let settled = false;
|
|
173
|
+
let stdoutBytes = 0;
|
|
174
|
+
let stdoutTruncated = false;
|
|
175
|
+
const timers = { timeout: null, killGrace: null, forceSettle: null };
|
|
176
|
+
timers.timeout = setTimeout(() => {
|
|
177
|
+
timedOut = true;
|
|
178
|
+
killReason = 'timeout';
|
|
179
|
+
_killRgProc(proc, timers);
|
|
180
|
+
}, timeoutMs);
|
|
181
|
+
if (timers.timeout.unref) timers.timeout.unref();
|
|
182
|
+
_armRgForceSettle({
|
|
183
|
+
timeoutMs,
|
|
184
|
+
isSettled: () => settled,
|
|
185
|
+
timers,
|
|
186
|
+
proc,
|
|
187
|
+
onForceSettle: () => {
|
|
188
|
+
timedOut = true;
|
|
189
|
+
killReason = 'timeout';
|
|
190
|
+
const e = new Error(`rg timed out after ${timeoutMs} ms`);
|
|
191
|
+
e.code = 'ETIMEDOUT';
|
|
192
|
+
if (settled) return;
|
|
193
|
+
settled = true;
|
|
194
|
+
_clearRgTimers(timers);
|
|
195
|
+
reject(e);
|
|
196
|
+
},
|
|
197
|
+
});
|
|
198
|
+
proc.stdout.setEncoding('utf-8');
|
|
199
|
+
proc.stderr.setEncoding('utf-8');
|
|
200
|
+
proc.stdout.on('data', (d) => {
|
|
201
|
+
if (stdoutTruncated) return;
|
|
202
|
+
// Account by UTF-8 byte length (d is a decoded string here), not by
|
|
203
|
+
// string length, so non-ASCII output cannot overshoot the byte cap.
|
|
204
|
+
const dBytes = Buffer.byteLength(d);
|
|
205
|
+
const remaining = MAX_RG_STDOUT_BYTES - stdoutBytes;
|
|
206
|
+
if (dBytes >= remaining) {
|
|
207
|
+
// Slice by BYTES (not chars) and cut on a UTF-8 codepoint
|
|
208
|
+
// boundary: a char-count slice overshoots on multibyte output,
|
|
209
|
+
// and even a raw byte slice would let a split trailing codepoint
|
|
210
|
+
// decode to U+FFFD (3 bytes) and exceed the cap. Back `end` over
|
|
211
|
+
// continuation bytes (0b10xxxxxx) so the kept bytes are <= cap.
|
|
212
|
+
const buf = Buffer.from(d, 'utf8');
|
|
213
|
+
let end = Math.max(0, Math.min(remaining, buf.length));
|
|
214
|
+
while (end > 0 && end < buf.length && (buf[end] & 0xC0) === 0x80) end--;
|
|
215
|
+
stdout += buf.subarray(0, end).toString('utf8');
|
|
216
|
+
stdoutBytes = MAX_RG_STDOUT_BYTES;
|
|
217
|
+
stdoutTruncated = true;
|
|
218
|
+
if (killReason !== 'timeout') killReason = 'cap';
|
|
219
|
+
_clearRgTimeoutOnly(timers);
|
|
220
|
+
_killRgProc(proc, timers);
|
|
221
|
+
return;
|
|
222
|
+
}
|
|
223
|
+
stdout += d;
|
|
224
|
+
stdoutBytes += dBytes;
|
|
225
|
+
});
|
|
226
|
+
proc.stderr.on('data', (d) => { stderr += d; });
|
|
227
|
+
proc.on('error', (err) => {
|
|
228
|
+
if (settled) return;
|
|
229
|
+
settled = true;
|
|
230
|
+
_clearRgTimers(timers);
|
|
231
|
+
reject(err);
|
|
232
|
+
});
|
|
233
|
+
proc.on('close', (code) => {
|
|
234
|
+
if (settled) return;
|
|
235
|
+
settled = true;
|
|
236
|
+
_clearRgTimers(timers);
|
|
237
|
+
if (timedOut && killReason === 'timeout') {
|
|
238
|
+
const e = new Error(`rg timed out after ${timeoutMs} ms`);
|
|
239
|
+
e.code = 'ETIMEDOUT';
|
|
240
|
+
return reject(e);
|
|
241
|
+
}
|
|
242
|
+
const wrap = (s) => {
|
|
243
|
+
if (!stdoutTruncated) return s;
|
|
244
|
+
const boxed = new String(s);
|
|
245
|
+
try { boxed.truncated = true; } catch { /* ignore */ }
|
|
246
|
+
return boxed;
|
|
247
|
+
};
|
|
248
|
+
const boxPartialStdout = (s, rgStderrText) => {
|
|
249
|
+
const boxed = new String(s);
|
|
250
|
+
try {
|
|
251
|
+
if (stdoutTruncated) boxed.truncated = true;
|
|
252
|
+
boxed.partial = true;
|
|
253
|
+
boxed.rgStderr = rgStderrText;
|
|
254
|
+
} catch { /* ignore */ }
|
|
255
|
+
return boxed;
|
|
256
|
+
};
|
|
257
|
+
if (code === 0) return resolve(wrap(stdout));
|
|
258
|
+
if (code === 1) return resolve(wrap(''));
|
|
259
|
+
// SIGTERM after our own truncation kill: surface accumulated stdout.
|
|
260
|
+
if (stdoutTruncated) return resolve(wrap(stdout));
|
|
261
|
+
// Exit 2 (e.g. permission denied on some paths) may still emit matches.
|
|
262
|
+
if (code === 2 && stdout.length > 0) {
|
|
263
|
+
return resolve(boxPartialStdout(stdout, stderr.trim()));
|
|
264
|
+
}
|
|
265
|
+
const e = new Error(`rg exited with code ${code}: ${stderr.trim()}`);
|
|
266
|
+
e.code = code;
|
|
267
|
+
e.stderr = stderr;
|
|
268
|
+
reject(e);
|
|
269
|
+
});
|
|
270
|
+
});
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
export async function runRg(argsList, execOptions = {}) {
|
|
274
|
+
assertRgAvailable();
|
|
275
|
+
try {
|
|
276
|
+
return await spawnRg(argsList, execOptions);
|
|
277
|
+
} catch (err) {
|
|
278
|
+
const msg = String(err?.message || err?.stderr || '');
|
|
279
|
+
if (/EAGAIN/i.test(msg) && !argsList.includes('-j')) {
|
|
280
|
+
return spawnRg(['-j', '1', ...argsList], execOptions);
|
|
281
|
+
}
|
|
282
|
+
throw err;
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
function spawnRgWindowedLines(argsList, execOptions, opts = {}) {
|
|
287
|
+
const timeoutMs = Number(execOptions?.timeout ?? 20000);
|
|
288
|
+
const offset = Math.max(0, Math.floor(Number(opts.offset) || 0));
|
|
289
|
+
const lineLimit = Number.isFinite(opts.limit) ? Math.max(0, Math.floor(Number(opts.limit) || 0)) : Infinity;
|
|
290
|
+
const summaryLimit = Math.max(0, Math.floor(Number(opts.summaryLimit) || 0));
|
|
291
|
+
const collectLimit = lineLimit === Infinity ? Infinity : Math.max(lineLimit, summaryLimit);
|
|
292
|
+
return new Promise((resolve, reject) => {
|
|
293
|
+
const proc = spawn(rgExecutable(), argsList, {
|
|
294
|
+
cwd: execOptions?.cwd,
|
|
295
|
+
env: execOptions?.env || process.env,
|
|
296
|
+
windowsHide: true,
|
|
297
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
298
|
+
});
|
|
299
|
+
let buffer = '';
|
|
300
|
+
let stderr = '';
|
|
301
|
+
let skipped = 0;
|
|
302
|
+
let seenAfterOffset = 0;
|
|
303
|
+
let stoppedEarly = false;
|
|
304
|
+
let timedOut = false;
|
|
305
|
+
/** @type {'timeout' | 'early' | null} */
|
|
306
|
+
let killReason = null;
|
|
307
|
+
let settled = false;
|
|
308
|
+
const timers = { timeout: null, killGrace: null, forceSettle: null };
|
|
309
|
+
const lines = [];
|
|
310
|
+
timers.timeout = setTimeout(() => {
|
|
311
|
+
timedOut = true;
|
|
312
|
+
killReason = 'timeout';
|
|
313
|
+
_killRgProc(proc, timers);
|
|
314
|
+
}, timeoutMs);
|
|
315
|
+
if (timers.timeout.unref) timers.timeout.unref();
|
|
316
|
+
_armRgForceSettle({
|
|
317
|
+
timeoutMs,
|
|
318
|
+
isSettled: () => settled,
|
|
319
|
+
timers,
|
|
320
|
+
proc,
|
|
321
|
+
onForceSettle: () => {
|
|
322
|
+
timedOut = true;
|
|
323
|
+
killReason = 'timeout';
|
|
324
|
+
const e = new Error(`rg timed out after ${timeoutMs} ms`);
|
|
325
|
+
e.code = 'ETIMEDOUT';
|
|
326
|
+
if (settled) return;
|
|
327
|
+
settled = true;
|
|
328
|
+
_clearRgTimers(timers);
|
|
329
|
+
reject(e);
|
|
330
|
+
},
|
|
331
|
+
});
|
|
332
|
+
const stopEarly = () => {
|
|
333
|
+
if (stoppedEarly) return;
|
|
334
|
+
stoppedEarly = true;
|
|
335
|
+
buffer = '';
|
|
336
|
+
if (killReason !== 'timeout') killReason = 'early';
|
|
337
|
+
_clearRgTimeoutOnly(timers);
|
|
338
|
+
_killRgProc(proc, timers);
|
|
339
|
+
};
|
|
340
|
+
const pushLine = (raw) => {
|
|
341
|
+
if (stoppedEarly || raw.length === 0) return;
|
|
342
|
+
const line = raw.endsWith('\r') ? raw.slice(0, -1) : raw;
|
|
343
|
+
if (line.length === 0) return;
|
|
344
|
+
if (skipped < offset) {
|
|
345
|
+
skipped++;
|
|
346
|
+
return;
|
|
347
|
+
}
|
|
348
|
+
seenAfterOffset++;
|
|
349
|
+
if (seenAfterOffset <= collectLimit) {
|
|
350
|
+
lines.push(line);
|
|
351
|
+
return;
|
|
352
|
+
}
|
|
353
|
+
stopEarly();
|
|
354
|
+
};
|
|
355
|
+
proc.stdout.setEncoding('utf-8');
|
|
356
|
+
proc.stderr.setEncoding('utf-8');
|
|
357
|
+
proc.stdout.on('data', (chunk) => {
|
|
358
|
+
if (stoppedEarly) return;
|
|
359
|
+
buffer += chunk;
|
|
360
|
+
let start = 0;
|
|
361
|
+
let idx = buffer.indexOf('\n', start);
|
|
362
|
+
while (!stoppedEarly && idx !== -1) {
|
|
363
|
+
pushLine(buffer.slice(start, idx));
|
|
364
|
+
start = idx + 1;
|
|
365
|
+
idx = buffer.indexOf('\n', start);
|
|
366
|
+
}
|
|
367
|
+
buffer = stoppedEarly ? '' : buffer.slice(start);
|
|
368
|
+
});
|
|
369
|
+
proc.stderr.on('data', (d) => {
|
|
370
|
+
if (stderr.length < 4096) stderr += d;
|
|
371
|
+
});
|
|
372
|
+
proc.on('error', (err) => {
|
|
373
|
+
if (settled) return;
|
|
374
|
+
settled = true;
|
|
375
|
+
_clearRgTimers(timers);
|
|
376
|
+
reject(err);
|
|
377
|
+
});
|
|
378
|
+
proc.on('close', (code) => {
|
|
379
|
+
if (settled) return;
|
|
380
|
+
settled = true;
|
|
381
|
+
_clearRgTimers(timers);
|
|
382
|
+
if (timedOut && killReason === 'timeout') {
|
|
383
|
+
const e = new Error(`rg timed out after ${timeoutMs} ms`);
|
|
384
|
+
e.code = 'ETIMEDOUT';
|
|
385
|
+
return reject(e);
|
|
386
|
+
}
|
|
387
|
+
if (!stoppedEarly && buffer.length > 0) pushLine(buffer);
|
|
388
|
+
if (stoppedEarly) {
|
|
389
|
+
return resolve({ lines, complete: false, totalSeen: seenAfterOffset });
|
|
390
|
+
}
|
|
391
|
+
if (code === 0 || code === 1) {
|
|
392
|
+
return resolve({ lines, complete: true, totalSeen: seenAfterOffset });
|
|
393
|
+
}
|
|
394
|
+
if (code === 2 && lines.length > 0) {
|
|
395
|
+
return resolve({
|
|
396
|
+
lines,
|
|
397
|
+
complete: false,
|
|
398
|
+
totalSeen: seenAfterOffset,
|
|
399
|
+
partial: true,
|
|
400
|
+
rgStderr: stderr.trim(),
|
|
401
|
+
});
|
|
402
|
+
}
|
|
403
|
+
const e = new Error(`rg exited with code ${code}: ${stderr.trim()}`);
|
|
404
|
+
e.code = code;
|
|
405
|
+
e.stderr = stderr;
|
|
406
|
+
reject(e);
|
|
407
|
+
});
|
|
408
|
+
});
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
export async function runRgWindowedLines(argsList, execOptions = {}, opts = {}) {
|
|
412
|
+
assertRgAvailable();
|
|
413
|
+
try {
|
|
414
|
+
return await spawnRgWindowedLines(argsList, execOptions, opts);
|
|
415
|
+
} catch (err) {
|
|
416
|
+
const msg = String(err?.message || err?.stderr || '');
|
|
417
|
+
if (/EAGAIN/i.test(msg) && !argsList.includes('-j')) {
|
|
418
|
+
return spawnRgWindowedLines(['-j', '1', ...argsList], execOptions, opts);
|
|
419
|
+
}
|
|
420
|
+
throw err;
|
|
421
|
+
}
|
|
422
|
+
}
|