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,298 @@
|
|
|
1
|
+
// cwd-tool.mjs — session-cwd management MCP tool.
|
|
2
|
+
//
|
|
3
|
+
// Backs the `cwd` tool exposed via tools.json. Three actions:
|
|
4
|
+
//
|
|
5
|
+
// get — report the current effective cwd (pwd()) plus its resolved
|
|
6
|
+
// project_id (via resolveProjectId).
|
|
7
|
+
// set — validate `args.path` against the filesystem and stash the
|
|
8
|
+
// canonical absolute form into process.env.MIXDOG_SESSION_CWD.
|
|
9
|
+
// captureOriginalUserCwd() consults that env var first, so
|
|
10
|
+
// every downstream tool (read/grep/glob/find_symbol/bash/...)
|
|
11
|
+
// picks up the change without a per-tool plumbing change.
|
|
12
|
+
// list — comprehensive `.git` repository scan rooted at $HOME and
|
|
13
|
+
// the rawUserCwd() session sentinel. Cached in-process; pass
|
|
14
|
+
// `refresh:true` to force a rebuild.
|
|
15
|
+
//
|
|
16
|
+
// list policy:
|
|
17
|
+
// - depth-bounded walk (maxDepth 4) using the shared walkDir helper.
|
|
18
|
+
// - prunes NOISE_DIR_NAMES + a small AppData cache-dir set to keep
|
|
19
|
+
// the scan responsive on Windows.
|
|
20
|
+
// - once a directory is identified as a `.git`-bearing repo, it is
|
|
21
|
+
// recorded and its children are NOT descended (no nested worktree
|
|
22
|
+
// scans, no submodule recursion).
|
|
23
|
+
|
|
24
|
+
import { resolve as pathResolve, join as pathJoin } from 'path'
|
|
25
|
+
import { homedir } from 'os'
|
|
26
|
+
import { existsSync, statSync, readFileSync, writeFileSync } from 'fs'
|
|
27
|
+
import { execFileSync } from 'child_process'
|
|
28
|
+
import { pwd, rawUserCwd, writeLastSessionCwd } from '../../../shared/user-cwd.mjs'
|
|
29
|
+
import { resolveOptionalCwd } from './builtin/cwd-utils.mjs'
|
|
30
|
+
import { walkDir, NOISE_DIR_NAMES } from './builtin/glob-walk.mjs'
|
|
31
|
+
import { resolveProjectId } from '../../../memory/lib/project-id-resolver.mjs'
|
|
32
|
+
import { resolvePluginData } from '../../../shared/plugin-paths.mjs'
|
|
33
|
+
|
|
34
|
+
// Canonical absolute-path form. Mirrors code-graph's `_canonicalGraphCwd`:
|
|
35
|
+
// resolve to absolute, lower-case on win32 so case drift in user input
|
|
36
|
+
// does not produce duplicate session-cwd values.
|
|
37
|
+
function _canonicalSessionCwd(cwd) {
|
|
38
|
+
const full = pathResolve(cwd)
|
|
39
|
+
return process.platform === 'win32' ? full.toLowerCase() : full
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Extra AppData cache dir names commonly found under %USERPROFILE% on
|
|
43
|
+
// Windows. They are not in NOISE_DIR_NAMES because they're user-data
|
|
44
|
+
// roots elsewhere, but we never want to spider them during a repo scan.
|
|
45
|
+
const EXTRA_NOISE_DIR_NAMES = new Set([
|
|
46
|
+
'AppData', 'Application Data', 'Local Settings',
|
|
47
|
+
'.npm', '.yarn', '.pnpm-store', '.nuget', '.cargo', '.rustup',
|
|
48
|
+
'.gradle', '.m2', '.ivy2', '.android', '.docker', '.vscode',
|
|
49
|
+
'.cursor', 'Library',
|
|
50
|
+
// NOTE: do NOT prune '.claude' — the mixdog repo lives under
|
|
51
|
+
// ~/.claude/plugins/marketplaces/trib-plugin (depth 4 from home), so
|
|
52
|
+
// pruning it would hide the user's primary project from `list`. The
|
|
53
|
+
// maxDepth:4 cap + stop-at-.git keep the deep version cache
|
|
54
|
+
// (~/.claude/plugins/cache/.../0.5.x = depth 6) out of the scan.
|
|
55
|
+
])
|
|
56
|
+
|
|
57
|
+
const PRUNED_DIR_NAMES = new Set([...NOISE_DIR_NAMES, ...EXTRA_NOISE_DIR_NAMES])
|
|
58
|
+
|
|
59
|
+
let _listCache = null
|
|
60
|
+
|
|
61
|
+
// Per-scan set used by the visitor to mark repos whose subtree must not
|
|
62
|
+
// be descended. walkDir does not natively support "stop descent into
|
|
63
|
+
// this dir but keep the walk going", so we approximate by checking
|
|
64
|
+
// ancestry on every visit and bailing early when the entry lives under
|
|
65
|
+
// a previously-recorded repo.
|
|
66
|
+
const _recordedRepos = new Set()
|
|
67
|
+
|
|
68
|
+
function _scanReposUnderFiltered(root) {
|
|
69
|
+
_recordedRepos.clear()
|
|
70
|
+
const repos = []
|
|
71
|
+
if (!root || !existsSync(root)) return repos
|
|
72
|
+
try {
|
|
73
|
+
const st = statSync(root)
|
|
74
|
+
if (!st.isDirectory()) return repos
|
|
75
|
+
} catch { return repos }
|
|
76
|
+
|
|
77
|
+
if (existsSync(`${root}/.git`)) {
|
|
78
|
+
repos.push(root)
|
|
79
|
+
return repos
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
walkDir(root, {
|
|
83
|
+
hidden: true,
|
|
84
|
+
maxDepth: 4,
|
|
85
|
+
excludeDirNames: PRUNED_DIR_NAMES,
|
|
86
|
+
visit: (ent, entPath) => {
|
|
87
|
+
if (!ent.isDirectory()) return
|
|
88
|
+
// Skip subtree of a previously-recorded repo. walkDir invokes
|
|
89
|
+
// visit before recursing, so flipping ent.isDirectory() here
|
|
90
|
+
// would not stop descent. Instead, check ancestry on every
|
|
91
|
+
// visit and bail early.
|
|
92
|
+
for (const recorded of _recordedRepos) {
|
|
93
|
+
if (entPath === recorded || entPath.startsWith(recorded + '/') || entPath.startsWith(recorded + '\\')) {
|
|
94
|
+
return
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
try {
|
|
98
|
+
if (existsSync(`${entPath}/.git`)) {
|
|
99
|
+
repos.push(entPath)
|
|
100
|
+
_recordedRepos.add(entPath)
|
|
101
|
+
}
|
|
102
|
+
} catch { /* ignore stat races */ }
|
|
103
|
+
},
|
|
104
|
+
})
|
|
105
|
+
|
|
106
|
+
return repos
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// ── Owned-project filtering (config: `cwd` section) ──────────────────
|
|
110
|
+
// `cwd list` scans ONLY under the session entry directory (rawUserCwd) plus
|
|
111
|
+
// any extraRoots — never $HOME. Claude-Code-managed plugin clones live under
|
|
112
|
+
// ~/.claude/plugins (i.e. $HOME), so they fall outside the scan with no
|
|
113
|
+
// manifest/path heuristic. Repos are then kept by ONE explicit signal:
|
|
114
|
+
// - identityEmails: a repo is kept only when >=1 commit is authored by one
|
|
115
|
+
// of these. Empty set = no filter (fail open) so the list never silently
|
|
116
|
+
// goes blank.
|
|
117
|
+
// extraRoots widens the scan beyond the session cwd.
|
|
118
|
+
function _cwdConfig() {
|
|
119
|
+
// Read the `cwd` section directly (no config.mjs import: its module-load
|
|
120
|
+
// resolvePluginData() throws when CLAUDE_PLUGIN_DATA is unset, which would
|
|
121
|
+
// break tool-manifest builds that merely import this file for its defs).
|
|
122
|
+
let raw = {}
|
|
123
|
+
try {
|
|
124
|
+
const all = JSON.parse(readFileSync(pathJoin(resolvePluginData(), 'mixdog-config.json'), 'utf8'))
|
|
125
|
+
if (all && typeof all.cwd === 'object' && all.cwd) raw = all.cwd
|
|
126
|
+
} catch { raw = {} }
|
|
127
|
+
const identityEmails = Array.isArray(raw.identityEmails)
|
|
128
|
+
? raw.identityEmails.filter(e => typeof e === 'string' && e.trim()).map(e => e.trim())
|
|
129
|
+
: []
|
|
130
|
+
const extraRoots = Array.isArray(raw.extraRoots)
|
|
131
|
+
? raw.extraRoots.filter(r => typeof r === 'string' && r.trim()).map(r => r.trim())
|
|
132
|
+
: []
|
|
133
|
+
return { identityEmails, extraRoots }
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
function _escapeRe(s) {
|
|
137
|
+
return s.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
// True when >=1 commit across all refs is authored by one of the identity
|
|
141
|
+
// emails. One git spawn per repo; multiple --author flags are OR-combined
|
|
142
|
+
// by git, and -1 exits on the first match.
|
|
143
|
+
function _authoredByIdentity(repoPath, emails) {
|
|
144
|
+
try {
|
|
145
|
+
const args = ['-C', repoPath, 'log', '--all', '-1', '--format=%H']
|
|
146
|
+
for (const e of emails) { args.push('--author', _escapeRe(e)) }
|
|
147
|
+
const out = execFileSync('git', args, { encoding: 'utf8', timeout: 5000, stdio: ['ignore', 'pipe', 'ignore'], windowsHide: true })
|
|
148
|
+
return out.trim().length > 0
|
|
149
|
+
} catch {
|
|
150
|
+
return false
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
function _buildRepoList(includeAll = false) {
|
|
155
|
+
const cfg = _cwdConfig()
|
|
156
|
+
const user = rawUserCwd()
|
|
157
|
+
const roots = []
|
|
158
|
+
const seen = new Set()
|
|
159
|
+
// Scan ONLY the session entry dir (+ extraRoots), never $HOME — this keeps
|
|
160
|
+
// Claude-Code-managed plugin clones under ~/.claude/plugins out of the scan
|
|
161
|
+
// entirely, with no manifest/path heuristic needed.
|
|
162
|
+
for (const r of [user, ...cfg.extraRoots]) {
|
|
163
|
+
if (!r) continue
|
|
164
|
+
const key = _canonicalSessionCwd(r)
|
|
165
|
+
if (seen.has(key)) continue
|
|
166
|
+
seen.add(key)
|
|
167
|
+
roots.push(r)
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
const repoSet = new Map() // canonical → original path
|
|
171
|
+
for (const root of roots) {
|
|
172
|
+
for (const repo of _scanReposUnderFiltered(root)) {
|
|
173
|
+
const key = _canonicalSessionCwd(repo)
|
|
174
|
+
if (!repoSet.has(key)) repoSet.set(key, repo)
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
let paths = [...repoSet.values()]
|
|
179
|
+
// Keep only repos the user has actually committed to (identityEmails). Empty
|
|
180
|
+
// set = no filter (fail open). No plugin filtering needed — plugin clones
|
|
181
|
+
// live under $HOME, which is no longer scanned.
|
|
182
|
+
if (!includeAll && cfg.identityEmails.length) {
|
|
183
|
+
paths = paths.filter(p => _authoredByIdentity(p, cfg.identityEmails))
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
const out = paths.map(repoPath => ({ path: repoPath, projectId: resolveProjectId(repoPath) }))
|
|
187
|
+
// Stable ordering: alphabetical by path.
|
|
188
|
+
out.sort((a, b) => a.path.localeCompare(b.path))
|
|
189
|
+
// Persist the filtered list so the SessionStart hook can inject a
|
|
190
|
+
// "## Projects" block without re-running the scan. Best-effort.
|
|
191
|
+
if (!includeAll) {
|
|
192
|
+
try { writeFileSync(pathJoin(resolvePluginData(), 'cwd-projects.json'), JSON.stringify({ projects: out })) } catch { /* best-effort */ }
|
|
193
|
+
}
|
|
194
|
+
return out
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
function _formatList(entries) {
|
|
198
|
+
if (entries.length === 0) return '[cwd] no .git repositories found under home / user-cwd roots'
|
|
199
|
+
const lines = entries.map(e => {
|
|
200
|
+
const pid = e.projectId ? ` project=${e.projectId}` : ' project=<unset>'
|
|
201
|
+
return `${e.path}${pid}`
|
|
202
|
+
})
|
|
203
|
+
return `[cwd] ${entries.length} repo${entries.length === 1 ? '' : 's'}:\n` + lines.join('\n')
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
/**
|
|
207
|
+
* MCP entry point. Returns a plain string; the server wraps it into the
|
|
208
|
+
* standard `{content:[{type:'text',text:...}]}` envelope.
|
|
209
|
+
*/
|
|
210
|
+
export async function executeCwdTool(name, args, callerCwd, opts = {}) {
|
|
211
|
+
const action = (typeof args?.action === 'string' && args.action) || 'get'
|
|
212
|
+
|
|
213
|
+
if (action === 'get') {
|
|
214
|
+
const effective = opts.session ? opts.session.resolveCwd() : pwd()
|
|
215
|
+
const projectId = resolveProjectId(effective)
|
|
216
|
+
return `[cwd] effective=${effective}\nproject=${projectId || '<unset>'}`
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
if (action === 'set') {
|
|
220
|
+
const baseCwd = (typeof callerCwd === 'string' && callerCwd) ? callerCwd : pwd()
|
|
221
|
+
const resolved = resolveOptionalCwd(args?.path, baseCwd)
|
|
222
|
+
if (resolved.error) {
|
|
223
|
+
return resolved.error
|
|
224
|
+
}
|
|
225
|
+
// resolveOptionalCwd returns baseCwd when args.path is empty/missing;
|
|
226
|
+
// that is an explicit "no path provided" failure for set.
|
|
227
|
+
if (!args?.path || (typeof args.path === 'string' && args.path.trim() === '')) {
|
|
228
|
+
return '[cwd] set requires a non-empty `path` argument'
|
|
229
|
+
}
|
|
230
|
+
const canonical = _canonicalSessionCwd(resolved.cwd)
|
|
231
|
+
// Per-session cwd write. In daemon mode opts.session stores it on the
|
|
232
|
+
// session (no cross-session leak); otherwise fall back to the global env.
|
|
233
|
+
if (opts.session) opts.session.setCwd(canonical)
|
|
234
|
+
else process.env.MIXDOG_SESSION_CWD = canonical
|
|
235
|
+
// Key the per-terminal sentinel by THIS connection's leadPid so a shared
|
|
236
|
+
// daemon (one process, N terminals) keeps each terminal's `cwd set` in its
|
|
237
|
+
// own session-cwd-<leadPid>.txt. Falls back to MIXDOG_SUPERVISOR_PID when
|
|
238
|
+
// no session leadPid is present (boot/stdio path — behaviour-identical).
|
|
239
|
+
writeLastSessionCwd(canonical, opts.session?.leadPid)
|
|
240
|
+
// ③ Fire-and-forget code-graph prewarm for the new cwd so the first
|
|
241
|
+
// unscoped find_symbol/references/callers hits a warm cache instead of the
|
|
242
|
+
// cold build on the query's critical path. Guarded inside
|
|
243
|
+
// prewarmCodeGraphIfProject: only validated project roots are prewarmed
|
|
244
|
+
// (arbitrary trees skipped), and it prewarms the detected ROOT — matching
|
|
245
|
+
// the re-root executeCodeGraphTool applies. Dynamic import avoids a static
|
|
246
|
+
// dependency on the heavy code-graph module; never blocks or fails the set.
|
|
247
|
+
import('./code-graph.mjs')
|
|
248
|
+
.then((m) => { try { m.prewarmCodeGraphIfProject?.(canonical) } catch { /* best-effort */ } })
|
|
249
|
+
.catch(() => { /* code-graph unavailable — skip prewarm */ })
|
|
250
|
+
// Invalidate the in-process repo list — set() typically follows a
|
|
251
|
+
// list() call when the user picks one of its entries.
|
|
252
|
+
_listCache = null
|
|
253
|
+
const projectId = resolveProjectId(canonical)
|
|
254
|
+
return `[cwd] set effective=${canonical}\nproject=${projectId || '<unset>'}`
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
if (action === 'list') {
|
|
258
|
+
// `all:true` bypasses the owned-project filter and transient prune — a
|
|
259
|
+
// raw scan of every .git repo under the roots. Not cached (rare).
|
|
260
|
+
if (args?.all === true) return _formatList(_buildRepoList(true))
|
|
261
|
+
if (args?.refresh === true) _listCache = null
|
|
262
|
+
if (!_listCache) _listCache = _buildRepoList(false)
|
|
263
|
+
return _formatList(_listCache)
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
return `[cwd] unknown action="${action}" (expected get|set|list)`
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
// Canonical tool definition consumed by dev/scripts/build-tools-manifest.mjs.
|
|
270
|
+
// The handler lives above; this array is the source for the `cwd` entry in
|
|
271
|
+
// tools.json. The `module: 'cwd'` field is injected at build time, so it is
|
|
272
|
+
// intentionally absent here (mirrors every other module's source array).
|
|
273
|
+
export const CWD_TOOL_DEFS = [
|
|
274
|
+
{
|
|
275
|
+
name: 'cwd',
|
|
276
|
+
title: 'Session CWD',
|
|
277
|
+
annotations: { title: 'Session CWD', readOnlyHint: false, destructiveHint: false, idempotentHint: true, openWorldHint: false },
|
|
278
|
+
description: 'Session cwd for relative paths (not a sandbox): get/set/list. set updates session-wide cwd for tools and bridge workers — set on project switch. list: repos you commit to (cwd.identityEmails), minus plugin-cache/codex; all:true for full scan.',
|
|
279
|
+
inputSchema: {
|
|
280
|
+
type: 'object',
|
|
281
|
+
properties: {
|
|
282
|
+
action: { type: 'string', enum: ['get', 'set', 'list'] },
|
|
283
|
+
path: { type: 'string' },
|
|
284
|
+
refresh: { type: 'boolean' },
|
|
285
|
+
all: { type: 'boolean' },
|
|
286
|
+
},
|
|
287
|
+
},
|
|
288
|
+
},
|
|
289
|
+
]
|
|
290
|
+
|
|
291
|
+
// Background-warm the default (filtered) repo list shortly after module
|
|
292
|
+
// load so the first `cwd list` returns an already-built cache. Deferred
|
|
293
|
+
// and unref'd so it never blocks startup or keeps the process alive.
|
|
294
|
+
try {
|
|
295
|
+
setTimeout(() => {
|
|
296
|
+
try { if (!_listCache) _listCache = _buildRepoList(false) } catch { /* best-effort */ }
|
|
297
|
+
}, 1000).unref?.()
|
|
298
|
+
} catch { /* environments without setTimeout */ }
|
|
@@ -0,0 +1,323 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
// Destructive command detector.
|
|
3
|
+
//
|
|
4
|
+
// Returns a short human-readable warning string when the command matches
|
|
5
|
+
// a known data-loss / hard-to-reverse pattern. Purely informational —
|
|
6
|
+
// the caller (case 'bash' in builtin.mjs) prepends the warning to the result
|
|
7
|
+
// envelope so the agent sees the risk inline. Does NOT block execution;
|
|
8
|
+
// hard blocks remain in BLOCKED_PATTERNS in builtin.mjs / bash-session.mjs.
|
|
9
|
+
|
|
10
|
+
import { SHELL_NAMES as _SHELL_NAMES, WRAPPER_NAMES as _WRAPPER_NAMES } from './shell-policy.mjs';
|
|
11
|
+
import { extractPowerShellCommandInner } from './shell-command.mjs';
|
|
12
|
+
|
|
13
|
+
export function stripQuotedAndHeredoc(s) { return _stripQuotedSpans(s); }
|
|
14
|
+
export function extractShellCInner(s) { return _extractShellCInner(s); }
|
|
15
|
+
|
|
16
|
+
function _stripQuotedSpans(s) {
|
|
17
|
+
return String(s || '')
|
|
18
|
+
.replace(/<<-?\s*['"]?(\w+)['"]?[\s\S]*?\n\1\b/g, '<<HEREDOC>>')
|
|
19
|
+
.replace(/'[^']*'/g, "''")
|
|
20
|
+
.replace(/"(?:[^"\\]|\\.)*"/g, '""')
|
|
21
|
+
.replace(/#[^\n]*/g, '');
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// Heredoc bodies (shell-exec contexts route the body through scanning).
|
|
25
|
+
function _extractHeredocBodies(s) {
|
|
26
|
+
const out = [];
|
|
27
|
+
const re = /<<-?\s*['"]?(\w+)['"]?([\s\S]*?)\n\1\b/g;
|
|
28
|
+
let m;
|
|
29
|
+
while ((m = re.exec(String(s || ''))) !== null) out.push(m[2]);
|
|
30
|
+
return out;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
// Shell-aware tokenizer. Quoted spans stay intact; separators become
|
|
34
|
+
// their own token so callers can split pipeline segments cleanly.
|
|
35
|
+
function _tokenize(s) {
|
|
36
|
+
const t = [], src = String(s || '');
|
|
37
|
+
let cur = '', i = 0;
|
|
38
|
+
while (i < src.length) {
|
|
39
|
+
const c = src[i];
|
|
40
|
+
if (c === "'") { const e = src.indexOf("'", i + 1); if (e === -1) { cur += src.slice(i); break; } cur += src.slice(i, e + 1); i = e + 1; continue; }
|
|
41
|
+
if (c === '"') { let j = i + 1; while (j < src.length) { if (src[j] === '\\' && j + 1 < src.length) { j += 2; continue; } if (src[j] === '"') break; j++; } if (j >= src.length) { cur += src.slice(i); break; } cur += src.slice(i, j + 1); i = j + 1; continue; }
|
|
42
|
+
if (c === '\\' && i + 1 < src.length) { cur += src[i] + src[i + 1]; i += 2; continue; }
|
|
43
|
+
if (c === ' ' || c === '\t') { if (cur) { t.push(cur); cur = ''; } i++; continue; }
|
|
44
|
+
if (c === '\n' || c === ';') { if (cur) { t.push(cur); cur = ''; } t.push(c); i++; continue; }
|
|
45
|
+
if (c === '&' || c === '|') { if (cur) { t.push(cur); cur = ''; } if (src[i + 1] === c) { t.push(c + c); i += 2; } else { t.push(c); i++; } continue; }
|
|
46
|
+
cur += c; i++;
|
|
47
|
+
}
|
|
48
|
+
if (cur) t.push(cur);
|
|
49
|
+
return t;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
function _splitSegments(tokens) {
|
|
53
|
+
const segs = [], SEP = new Set([';', '&', '&&', '|', '||', '\n']);
|
|
54
|
+
let cur = [];
|
|
55
|
+
for (const t of tokens) { if (SEP.has(t)) { if (cur.length) segs.push(cur); cur = []; } else cur.push(t); }
|
|
56
|
+
if (cur.length) segs.push(cur);
|
|
57
|
+
return segs;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
function _stripQuotes(t) {
|
|
61
|
+
if (t.length >= 2) { const a = t[0], b = t[t.length - 1]; if ((a === "'" && b === "'") || (a === '"' && b === '"')) return t.slice(1, -1); }
|
|
62
|
+
return t;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// Skip env-var assignments and known wrapper commands (with their
|
|
66
|
+
// option arguments) so the underlying program reaches classification.
|
|
67
|
+
function _peelWrappers(tokens) {
|
|
68
|
+
let i = 0;
|
|
69
|
+
while (i < tokens.length) {
|
|
70
|
+
const t = tokens[i];
|
|
71
|
+
if (/^[A-Za-z_][A-Za-z0-9_]*=/.test(t)) { i++; continue; }
|
|
72
|
+
if (_WRAPPER_NAMES.has(t)) {
|
|
73
|
+
i++;
|
|
74
|
+
while (i < tokens.length && (/^[-+]/.test(tokens[i]) || /^\d+[smhd]?$/.test(tokens[i]) || /^\d+m\d+s?$/.test(tokens[i]))) {
|
|
75
|
+
// Long option `--key value` — consume the value too when not `=`-joined and the value is non-flag / non-assignment.
|
|
76
|
+
if (/^--[A-Za-z0-9][\w-]*$/.test(tokens[i]) && i + 1 < tokens.length && !/^[-+]/.test(tokens[i + 1]) && !/^[A-Za-z_][A-Za-z0-9_]*=/.test(tokens[i + 1])) { i += 2; continue; }
|
|
77
|
+
i++;
|
|
78
|
+
}
|
|
79
|
+
continue;
|
|
80
|
+
}
|
|
81
|
+
break;
|
|
82
|
+
}
|
|
83
|
+
return tokens.slice(i);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Extract `-c <payload>` (and `--command <payload>`) for shell invocations.
|
|
87
|
+
// Walks options including combined short flags like `-lc`, `-ic`, `-ec`.
|
|
88
|
+
function _extractShellCInner(s) {
|
|
89
|
+
const out = [];
|
|
90
|
+
const tokens = _tokenize(s);
|
|
91
|
+
const segs = _splitSegments(tokens);
|
|
92
|
+
for (const seg of segs) {
|
|
93
|
+
const peeled = _peelWrappers(seg);
|
|
94
|
+
if (!peeled.length) continue;
|
|
95
|
+
if (!_SHELL_NAMES.has(peeled[0])) {
|
|
96
|
+
// Extended: leaf commands that embed an inner command/payload.
|
|
97
|
+
// The caller re-scans each entry via isBlockedCommand, so we just
|
|
98
|
+
// surface the payload as a synthetic command string.
|
|
99
|
+
const leaf = peeled[0];
|
|
100
|
+
if (leaf === 'xargs' || leaf === 'parallel') {
|
|
101
|
+
let j = 1;
|
|
102
|
+
while (j < peeled.length && /^[-+]/.test(peeled[j])) {
|
|
103
|
+
if (/^--[A-Za-z0-9][\w-]*$/.test(peeled[j]) && j + 1 < peeled.length && !/^[-+]/.test(peeled[j + 1])) { j += 2; continue; }
|
|
104
|
+
j++;
|
|
105
|
+
}
|
|
106
|
+
if (j < peeled.length) out.push(peeled.slice(j).map(_stripQuotes).join(' '));
|
|
107
|
+
} else if (leaf === 'find') {
|
|
108
|
+
for (let j = 1; j < peeled.length; j++) {
|
|
109
|
+
if (peeled[j] === '-exec' || peeled[j] === '-execdir') {
|
|
110
|
+
const cmd = [];
|
|
111
|
+
let k = j + 1;
|
|
112
|
+
while (k < peeled.length && peeled[k] !== ';' && peeled[k] !== '\\;' && peeled[k] !== '+') { cmd.push(_stripQuotes(peeled[k])); k++; }
|
|
113
|
+
if (cmd.length) out.push(cmd.join(' '));
|
|
114
|
+
j = k;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
} else if (leaf === 'awk') {
|
|
118
|
+
for (let j = 1; j < peeled.length; j++) {
|
|
119
|
+
const t = peeled[j];
|
|
120
|
+
if (t === '-f' || t === '-v' || t === '-F') { j++; continue; }
|
|
121
|
+
if (t.startsWith('-')) continue;
|
|
122
|
+
const body = _stripQuotes(t);
|
|
123
|
+
const m = body.match(/system\s*\(\s*(['"])([\s\S]*?)\1\s*\)/);
|
|
124
|
+
if (m) out.push(m[2]);
|
|
125
|
+
break;
|
|
126
|
+
}
|
|
127
|
+
} else if (leaf === 'perl' || leaf === 'node' || leaf === 'python' || leaf === 'python3') {
|
|
128
|
+
const flag = (leaf === 'python' || leaf === 'python3') ? '-c' : '-e';
|
|
129
|
+
for (let j = 1; j < peeled.length; j++) {
|
|
130
|
+
const t = peeled[j];
|
|
131
|
+
if (t === flag || (leaf === 'perl' && t === '-E')) { const arg = peeled[j + 1]; if (arg) out.push(_stripQuotes(arg)); break; }
|
|
132
|
+
if (t.startsWith('-')) continue;
|
|
133
|
+
break;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
continue;
|
|
137
|
+
}
|
|
138
|
+
for (let i = 1; i < peeled.length; i++) {
|
|
139
|
+
const t = peeled[i];
|
|
140
|
+
if (t === '-c' || t === '--command' || /^-[a-zA-Z]+c$/.test(t)) {
|
|
141
|
+
const arg = peeled[i + 1];
|
|
142
|
+
if (arg) out.push(_stripQuotes(arg));
|
|
143
|
+
break;
|
|
144
|
+
}
|
|
145
|
+
if (t === '--rcfile' || t === '--init-file' || t === '-O' || t === '+O') { i++; continue; }
|
|
146
|
+
if (t.startsWith('-') || t.startsWith('+')) continue;
|
|
147
|
+
break;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
return out;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
// rm: detect -r/-R/--recursive AND -f/--force across split or combined
|
|
154
|
+
// short-flag tokens.
|
|
155
|
+
function _classifyRm(args) {
|
|
156
|
+
let r = false, f = false;
|
|
157
|
+
for (const t of args) {
|
|
158
|
+
if (t === '--') break;
|
|
159
|
+
if (t === '--recursive' || t === '-r' || t === '-R') { r = true; continue; }
|
|
160
|
+
if (t === '--force' || t === '-f') { f = true; continue; }
|
|
161
|
+
if (/^-[a-zA-Z]+$/.test(t)) { if (/[rR]/.test(t)) r = true; if (/f/.test(t)) f = true; continue; }
|
|
162
|
+
if (t.startsWith('-')) continue;
|
|
163
|
+
break;
|
|
164
|
+
}
|
|
165
|
+
if (r && f) return 'may recursively force-remove files';
|
|
166
|
+
if (r) return 'may recursively remove files';
|
|
167
|
+
if (f) return 'may force-remove files';
|
|
168
|
+
return null;
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
// PowerShell Remove-Item (and aliases): -Recurse/-Force, including :$true forms.
|
|
172
|
+
// Parity with shell-policy.mjs _removeItemRecursiveForceUnsafe switch matching.
|
|
173
|
+
const _PS_REMOVE_CMDS = new Set(['remove-item', 'ri', 'del', 'erase', 'rd', 'rmdir']);
|
|
174
|
+
|
|
175
|
+
function _classifyRemoveItem(args) {
|
|
176
|
+
let r = false, f = false;
|
|
177
|
+
for (const t of args) {
|
|
178
|
+
if (t === '--') break;
|
|
179
|
+
const low = String(t).toLowerCase();
|
|
180
|
+
if (low.startsWith('-')) {
|
|
181
|
+
if (/^-r(ec(urse)?)?$/.test(low) || /^-r(ec(urse)?)?:\$true$/i.test(low)) { r = true; continue; }
|
|
182
|
+
if (/^-fo(rce)?$/.test(low) || /^-fo(rce)?:\$true$/i.test(low)) { f = true; continue; }
|
|
183
|
+
if (low === '-recurse' || low === '-recursive') { r = true; continue; }
|
|
184
|
+
if (low === '-force') { f = true; continue; }
|
|
185
|
+
continue;
|
|
186
|
+
}
|
|
187
|
+
break;
|
|
188
|
+
}
|
|
189
|
+
if (r && f) return 'may recursively force-remove files';
|
|
190
|
+
if (r) return 'may recursively remove files';
|
|
191
|
+
if (f) return 'may force-remove files';
|
|
192
|
+
return null;
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
// git: skip global options (-C path, -c key=val, --git-dir=..., --no-pager
|
|
196
|
+
// etc.) before reading the subcommand.
|
|
197
|
+
function _classifyGit(args) {
|
|
198
|
+
let i = 0;
|
|
199
|
+
while (i < args.length) {
|
|
200
|
+
const t = args[i];
|
|
201
|
+
if (t === '-C' || t === '-c') { i += 2; continue; }
|
|
202
|
+
if (/^--(git-dir|work-tree|namespace)(=|$)/.test(t)) { i += t.includes('=') ? 1 : 2; continue; }
|
|
203
|
+
if (t === '--no-pager' || t === '--paginate' || t === '--bare' || t === '--exec-path' || /^--literal-pathspecs|--glob-pathspecs|--noglob-pathspecs|--icase-pathspecs$/.test(t)) { i++; continue; }
|
|
204
|
+
if (/^--exec-path=|^--list-cmds=/.test(t)) { i++; continue; }
|
|
205
|
+
break;
|
|
206
|
+
}
|
|
207
|
+
const sub = args[i]; if (!sub) return null;
|
|
208
|
+
const rest = args.slice(i + 1);
|
|
209
|
+
if (sub === 'reset' && rest.includes('--hard')) return 'may discard uncommitted changes';
|
|
210
|
+
if (sub === 'push' && rest.some(t => t === '--force' || t === '-f' || t === '--force-with-lease' || t.startsWith('--force-with-lease=') || /^\+[\w/.-]+/.test(t))) return 'may overwrite remote history';
|
|
211
|
+
if (sub === 'clean') {
|
|
212
|
+
const dry = rest.some(t => t === '-n' || t === '--dry-run' || /^-[a-zA-Z]*n[a-zA-Z]*$/.test(t));
|
|
213
|
+
const force = rest.some(t => t === '-f' || t === '--force' || /^-[a-zA-Z]*f[a-zA-Z]*$/.test(t));
|
|
214
|
+
if (!dry && force) return 'may permanently delete untracked files';
|
|
215
|
+
}
|
|
216
|
+
if ((sub === 'checkout' || sub === 'restore') && rest.includes('.')) return 'may discard all working tree changes';
|
|
217
|
+
if (sub === 'stash' && (rest[0] === 'drop' || rest[0] === 'clear')) return 'may permanently remove stashed changes';
|
|
218
|
+
if (sub === 'branch' && (rest.includes('-D') || (rest.includes('--delete') && rest.includes('--force')))) return 'may force-delete a branch';
|
|
219
|
+
if ((sub === 'commit' || sub === 'push' || sub === 'merge') && rest.includes('--no-verify')) return 'may skip safety hooks';
|
|
220
|
+
if (sub === 'commit' && rest.includes('--amend')) return 'may rewrite the last commit';
|
|
221
|
+
return null;
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
const _KUBECTL_VAL = new Set(['--context','--cluster','--namespace','-n','--user','--kubeconfig','--token','--server','--as','--as-group','--certificate-authority','--client-certificate','--client-key','--request-timeout','--cache-dir','--v','-v','--profile','--profile-output']);
|
|
225
|
+
|
|
226
|
+
function _classifyKubectl(args) {
|
|
227
|
+
let i = 0;
|
|
228
|
+
while (i < args.length) {
|
|
229
|
+
const t = args[i];
|
|
230
|
+
if (_KUBECTL_VAL.has(t)) { i += 2; continue; }
|
|
231
|
+
if (t.startsWith('--') && t.includes('=')) { i++; continue; }
|
|
232
|
+
if (t.startsWith('-')) { i++; continue; }
|
|
233
|
+
break;
|
|
234
|
+
}
|
|
235
|
+
return args[i] === 'delete' ? 'may delete Kubernetes resources' : null;
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
function _classifyTerraform(args) {
|
|
239
|
+
let i = 0;
|
|
240
|
+
while (i < args.length) {
|
|
241
|
+
const t = args[i];
|
|
242
|
+
if (/^-chdir=/.test(t) || t === '-help' || t === '-version' || t === '-h') { i++; continue; }
|
|
243
|
+
break;
|
|
244
|
+
}
|
|
245
|
+
return args[i] === 'destroy' ? 'may destroy Terraform infrastructure' : null;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
function _classifyDd(args) {
|
|
249
|
+
for (const t of args) {
|
|
250
|
+
if (/^if=\/dev\//.test(t)) return 'may read from a raw device';
|
|
251
|
+
if (/^of=\/dev\//.test(t)) return 'may write to a raw device';
|
|
252
|
+
}
|
|
253
|
+
return null;
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
// DB destructive pattern baseline — security policy, not a heuristic
|
|
257
|
+
// classifier. DROP/TRUNCATE/DELETE without WHERE are unambiguously
|
|
258
|
+
// data-destructive regardless of context; this list is a floor, not a
|
|
259
|
+
// complete SQL auditor.
|
|
260
|
+
const _DB_PATTERNS = [
|
|
261
|
+
[/\b(DROP|TRUNCATE)\s+(TABLE|DATABASE|SCHEMA)\b/i, 'may drop or truncate database objects'],
|
|
262
|
+
[/\bDELETE\s+FROM\s+\w+[ \t]*(;|"|'|\n|$)/i, 'may delete all rows from a database table'],
|
|
263
|
+
];
|
|
264
|
+
|
|
265
|
+
function _classifySegment(tokens) {
|
|
266
|
+
const peeled = _peelWrappers(tokens);
|
|
267
|
+
if (!peeled.length) return null;
|
|
268
|
+
const cmd = peeled[0], rest = peeled.slice(1);
|
|
269
|
+
const cmdLow = cmd.toLowerCase();
|
|
270
|
+
if (cmd === 'rm') return _classifyRm(rest);
|
|
271
|
+
if (_PS_REMOVE_CMDS.has(cmdLow)) {
|
|
272
|
+
const w = _classifyRemoveItem(rest);
|
|
273
|
+
if (w) return w;
|
|
274
|
+
}
|
|
275
|
+
if (cmd === 'git') return _classifyGit(rest);
|
|
276
|
+
if (cmd === 'kubectl') return _classifyKubectl(rest);
|
|
277
|
+
if (cmd === 'terraform') return _classifyTerraform(rest);
|
|
278
|
+
if (cmd === 'dd') return _classifyDd(rest);
|
|
279
|
+
if (cmd === 'rmdir' || cmd === 'rd') {
|
|
280
|
+
const hasS = rest.some(t => /^\/s$/i.test(t));
|
|
281
|
+
const hasDriveRoot = rest.some(t => /^[A-Za-z]:\\?$/.test(t));
|
|
282
|
+
if (hasS && hasDriveRoot) return 'may recursively remove a drive root';
|
|
283
|
+
}
|
|
284
|
+
return null;
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
export function getDestructiveCommandWarning(command) {
|
|
288
|
+
const raw = String(command || '');
|
|
289
|
+
// Per-statement DB scan (split on `;`, `&&`, `||`, `|`, `\n`) so a
|
|
290
|
+
// destructive statement past a separator still surfaces.
|
|
291
|
+
const cleaned = _stripQuotedSpans(raw);
|
|
292
|
+
for (const stmt of cleaned.split(/[;&|\n]+/)) {
|
|
293
|
+
for (const [re, warning] of _DB_PATTERNS) if (re.test(stmt)) return warning;
|
|
294
|
+
}
|
|
295
|
+
// Tokenized walk per pipeline segment.
|
|
296
|
+
const segs = _splitSegments(_tokenize(raw));
|
|
297
|
+
for (const seg of segs) {
|
|
298
|
+
const peeled = _peelWrappers(seg);
|
|
299
|
+
if (!peeled.length) continue;
|
|
300
|
+
if (_SHELL_NAMES.has(peeled[0])) {
|
|
301
|
+
for (const inner of _extractShellCInner(seg.join(' '))) {
|
|
302
|
+
const w = getDestructiveCommandWarning(inner);
|
|
303
|
+
if (w) return w;
|
|
304
|
+
}
|
|
305
|
+
for (const body of _extractHeredocBodies(raw)) {
|
|
306
|
+
const w = getDestructiveCommandWarning(body);
|
|
307
|
+
if (w) return w;
|
|
308
|
+
}
|
|
309
|
+
continue;
|
|
310
|
+
}
|
|
311
|
+
const w = _classifySegment(seg);
|
|
312
|
+
if (w) return w;
|
|
313
|
+
}
|
|
314
|
+
for (const inner of _extractShellCInner(raw)) {
|
|
315
|
+
const w = getDestructiveCommandWarning(inner);
|
|
316
|
+
if (w) return w;
|
|
317
|
+
}
|
|
318
|
+
for (const inner of extractPowerShellCommandInner(raw)) {
|
|
319
|
+
const w = getDestructiveCommandWarning(inner);
|
|
320
|
+
if (w) return w;
|
|
321
|
+
}
|
|
322
|
+
return null;
|
|
323
|
+
}
|