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,100 @@
|
|
|
1
|
+
// R11: scrub dangerous loader/execution env vars from a model-spawned
|
|
2
|
+
// subprocess environment. Defense-in-depth on top of R5's secret scrub
|
|
3
|
+
// (provider/cloud tokens). These vars let an attacker (or model error)
|
|
4
|
+
// inject code before the user's command runs — NODE_OPTIONS can preload
|
|
5
|
+
// arbitrary --require modules, LD_PRELOAD / DYLD_INSERT_LIBRARIES inject
|
|
6
|
+
// shared libraries, PYTHONPATH / RUBYLIB / PERL5LIB hijack imports,
|
|
7
|
+
// BASH_ENV / ENV runs a script on shell startup, SHELLOPTS toggles
|
|
8
|
+
// dangerous shell flags, GLOBIGNORE silently hides files from expansion.
|
|
9
|
+
// CDPATH redirects `cd` to attacker-controlled dirs, PROMPT_COMMAND runs
|
|
10
|
+
// arbitrary code on every prompt, BASHOPTS toggles shell options, IFS
|
|
11
|
+
// rewrites word splitting, SSH_AUTH_SOCK / GPG_AGENT_INFO / GNUPGHOME
|
|
12
|
+
// expose live agent credentials to the child.
|
|
13
|
+
// PATH is intentionally preserved — subprocesses need it to find tools.
|
|
14
|
+
const LOADER_VARS = [
|
|
15
|
+
'NODE_OPTIONS',
|
|
16
|
+
'NODE_PATH',
|
|
17
|
+
'LD_PRELOAD',
|
|
18
|
+
'LD_LIBRARY_PATH',
|
|
19
|
+
'DYLD_INSERT_LIBRARIES',
|
|
20
|
+
'DYLD_LIBRARY_PATH',
|
|
21
|
+
'PYTHONPATH',
|
|
22
|
+
'RUBYLIB',
|
|
23
|
+
'PERL5LIB',
|
|
24
|
+
'BASH_ENV',
|
|
25
|
+
'ENV',
|
|
26
|
+
'SHELLOPTS',
|
|
27
|
+
'GLOBIGNORE',
|
|
28
|
+
'CDPATH',
|
|
29
|
+
'PROMPT_COMMAND',
|
|
30
|
+
'BASHOPTS',
|
|
31
|
+
'IFS',
|
|
32
|
+
'SSH_AUTH_SOCK',
|
|
33
|
+
'GPG_AGENT_INFO',
|
|
34
|
+
'GNUPGHOME',
|
|
35
|
+
];
|
|
36
|
+
|
|
37
|
+
// R5: provider / cloud / secret-family scrub. Shared across all spawn
|
|
38
|
+
// sites so the suffix and exact lists never drift. Broad provider-family
|
|
39
|
+
// prefixes (AWS_/GITHUB_/NPM_/VERCEL_/…) were removed: they nuked
|
|
40
|
+
// non-secret config like AWS_REGION, GITHUB_WORKSPACE, NPM_CONFIG_CACHE,
|
|
41
|
+
// VERCEL_ENV and broke subprocess tooling. Match instead by secret-
|
|
42
|
+
// shaped SUFFIX (covers `*_API_KEY`, `*_SECRET`, `*_TOKEN`, `*_PASSWORD`,
|
|
43
|
+
// `*_CREDENTIALS`, `*_PRIVATE_KEY`, `*_KEY`, …) and an explicit EXACT
|
|
44
|
+
// set for non-suffix secrets (AWS access ids, DATABASE_URL, named
|
|
45
|
+
// provider keys, GOOGLE_APPLICATION_CREDENTIALS).
|
|
46
|
+
// Secret-shaped suffixes. The bare `_KEY` suffix was dropped — it deleted
|
|
47
|
+
// non-secret public keys (GPG_KEY, NEXT_PUBLIC_*_KEY). `_API_KEY` is kept for
|
|
48
|
+
// broad provider coverage (e.g. XAI_API_KEY) but is guarded by PUBLIC_PREFIX_RE
|
|
49
|
+
// so client-exposed build vars (VITE_FIREBASE_API_KEY, NEXT_PUBLIC_*) survive.
|
|
50
|
+
const SECRET_SUFFIX_RE = /(_SECRET_ACCESS_KEY|_ACCESS_KEY|_SESSION_TOKEN|_AUTH_TOKEN|_API_KEY|_TOKEN|_SECRET|_PASSWORD|_CREDENTIALS|_PRIVATE_KEY|_PAT)$/;
|
|
51
|
+
// By-convention client-PUBLIC env prefixes: these are intentionally bundled to
|
|
52
|
+
// the browser and must never be scrubbed as secrets even with an _API_KEY/_KEY
|
|
53
|
+
// shape. Excludes them from the secret match below.
|
|
54
|
+
const PUBLIC_PREFIX_RE = /^(?:NEXT_PUBLIC_|NUXT_PUBLIC_|VITE_|REACT_APP_|VUE_APP_|EXPO_PUBLIC_|GATSBY_|STORYBOOK_|PUBLIC_)/;
|
|
55
|
+
const SECRET_EXACT = new Set([
|
|
56
|
+
'AWS_ACCESS_KEY_ID',
|
|
57
|
+
'AWS_SECRET_ACCESS_KEY',
|
|
58
|
+
'AWS_SESSION_TOKEN',
|
|
59
|
+
'AWS_WEB_IDENTITY_TOKEN_FILE',
|
|
60
|
+
'AWS_CONTAINER_CREDENTIALS_RELATIVE_URI',
|
|
61
|
+
'AWS_CONTAINER_CREDENTIALS_FULL_URI',
|
|
62
|
+
'DATABASE_URL',
|
|
63
|
+
'ANTHROPIC_API_KEY',
|
|
64
|
+
'OPENAI_API_KEY',
|
|
65
|
+
'GEMINI_API_KEY',
|
|
66
|
+
'GOOGLE_API_KEY',
|
|
67
|
+
'GOOGLE_APPLICATION_CREDENTIALS',
|
|
68
|
+
'GITHUB_PAT',
|
|
69
|
+
'STRIPE_SECRET_KEY',
|
|
70
|
+
'SUPABASE_SERVICE_ROLE_KEY',
|
|
71
|
+
'NPM_CONFIG__AUTH',
|
|
72
|
+
'npm_config__auth',
|
|
73
|
+
]);
|
|
74
|
+
|
|
75
|
+
export function scrubLoaderVars(env) {
|
|
76
|
+
if (!env || typeof env !== 'object') return env;
|
|
77
|
+
for (const k of LOADER_VARS) delete env[k];
|
|
78
|
+
// Wildcard sweep: the exact-name list covers the common loader vars but
|
|
79
|
+
// the DYLD_/LD_ families have many siblings (DYLD_FRAMEWORK_PATH,
|
|
80
|
+
// DYLD_FALLBACK_LIBRARY_PATH, LD_AUDIT, LD_BIND_NOW, …). Delete every
|
|
81
|
+
// key under those prefixes so a new variant doesn't sneak through.
|
|
82
|
+
for (const k of Object.keys(env)) {
|
|
83
|
+
if (/^DYLD_/.test(k) || /^LD_/.test(k)) delete env[k];
|
|
84
|
+
}
|
|
85
|
+
return env;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
// R5: strip provider/cloud/secret-family keys from a spawn env in place.
|
|
89
|
+
// Shared by bash-session, shell-jobs, shell-snapshot, and bash-tool so
|
|
90
|
+
// the prefix/suffix lists never drift across spawn sites. PATH is not
|
|
91
|
+
// matched by any family here and is preserved.
|
|
92
|
+
export function scrubProviderSecrets(env) {
|
|
93
|
+
if (!env || typeof env !== 'object') return env;
|
|
94
|
+
for (const k of Object.keys(env)) {
|
|
95
|
+
if (!PUBLIC_PREFIX_RE.test(k) && (SECRET_EXACT.has(k) || SECRET_SUFFIX_RE.test(k))) {
|
|
96
|
+
delete env[k];
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
return env;
|
|
100
|
+
}
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
// graph-binary-fetcher.mjs — fetches the prebuilt mixdog-graph native binary
|
|
2
|
+
// from the GitHub release manifest. The code graph has NO JS parse fallback,
|
|
3
|
+
// so the binary is required; ensureGraphBinary downloads + sha256-verifies it
|
|
4
|
+
// on first use and caches it under <dataDir>/graph-bin/. Mirrors the runtime
|
|
5
|
+
// fetcher pattern (memory/lib/runtime-fetcher.mjs) but for a single binary —
|
|
6
|
+
// no tar extraction.
|
|
7
|
+
//
|
|
8
|
+
// Public API:
|
|
9
|
+
// ensureGraphBinary(dataDir) -> absolute path to the verified binary.
|
|
10
|
+
// Throws if no asset matches the platform, or download/verify fails.
|
|
11
|
+
// findCachedGraphBinary(dataDir) -> path | null (sync, no network).
|
|
12
|
+
|
|
13
|
+
import { createHash } from 'node:crypto';
|
|
14
|
+
import {
|
|
15
|
+
chmodSync, createWriteStream, existsSync, mkdirSync,
|
|
16
|
+
readFileSync, readdirSync, renameSync, rmSync,
|
|
17
|
+
} from 'node:fs';
|
|
18
|
+
import { readFile } from 'node:fs/promises';
|
|
19
|
+
import { join } from 'node:path';
|
|
20
|
+
import { fileURLToPath } from 'node:url';
|
|
21
|
+
import { pipeline } from 'node:stream/promises';
|
|
22
|
+
|
|
23
|
+
// Bundled fallback manifest shipped with the plugin. CI rewrites this on each
|
|
24
|
+
// release with the per-platform asset URLs + sha256.
|
|
25
|
+
const BUNDLED_MANIFEST_PATH = fileURLToPath(new URL('./graph-manifest.json', import.meta.url));
|
|
26
|
+
|
|
27
|
+
// GitHub raw fallback — only consulted when neither cached nor bundled exists.
|
|
28
|
+
const MANIFEST_URL = 'https://raw.githubusercontent.com/trib-plugin/mixdog/main/src/agent/orchestrator/tools/graph-manifest.json';
|
|
29
|
+
|
|
30
|
+
function binSuffix() {
|
|
31
|
+
return process.platform === 'win32' ? '.exe' : '';
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
function platformKey() {
|
|
35
|
+
const os = process.platform === 'win32' ? 'win32' : process.platform;
|
|
36
|
+
return `${os}-${process.arch}`;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
function graphBinDir(dataDir) {
|
|
40
|
+
return join(dataDir, 'graph-bin');
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
async function loadManifest(dataDir) {
|
|
44
|
+
const cached = join(graphBinDir(dataDir), 'manifest.json');
|
|
45
|
+
if (existsSync(cached)) {
|
|
46
|
+
try { return JSON.parse(readFileSync(cached, 'utf8')); } catch { /* fall through */ }
|
|
47
|
+
}
|
|
48
|
+
if (existsSync(BUNDLED_MANIFEST_PATH)) {
|
|
49
|
+
return JSON.parse(readFileSync(BUNDLED_MANIFEST_PATH, 'utf8'));
|
|
50
|
+
}
|
|
51
|
+
const res = await fetch(MANIFEST_URL, { signal: AbortSignal.timeout(30_000) });
|
|
52
|
+
if (!res.ok) throw new Error(`[graph-fetcher] manifest fetch failed: ${res.status} ${res.statusText}`);
|
|
53
|
+
return res.json();
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
async function sha256File(filePath) {
|
|
57
|
+
return createHash('sha256').update(await readFile(filePath)).digest('hex');
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
async function downloadWithRetry(url, destPath) {
|
|
61
|
+
// 4 attempts total (1 + 3 retries); backoff 1s/3s/9s. 4xx is terminal.
|
|
62
|
+
const delays = [1000, 3000, 9000];
|
|
63
|
+
let lastErr;
|
|
64
|
+
for (let attempt = 0; attempt < 4; attempt++) {
|
|
65
|
+
try {
|
|
66
|
+
const res = await fetch(url, { signal: AbortSignal.timeout(180_000) });
|
|
67
|
+
if (res.status >= 400 && res.status < 500) {
|
|
68
|
+
throw new Error(`[graph-fetcher] asset HTTP ${res.status} (terminal) — ${url}`);
|
|
69
|
+
}
|
|
70
|
+
if (!res.ok) throw new Error(`[graph-fetcher] asset HTTP ${res.status} — ${url}`);
|
|
71
|
+
await pipeline(res.body, createWriteStream(destPath));
|
|
72
|
+
return;
|
|
73
|
+
} catch (err) {
|
|
74
|
+
lastErr = err;
|
|
75
|
+
if (String(err?.message || '').includes('(terminal)')) throw err;
|
|
76
|
+
if (attempt < 3) {
|
|
77
|
+
process.stderr.write(`[graph-fetcher] download attempt ${attempt + 1} failed (${err?.message}), retrying in ${delays[attempt]}ms…\n`);
|
|
78
|
+
await new Promise((r) => setTimeout(r, delays[attempt]));
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
throw lastErr;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// Remove stale binaries + tmp files, keeping the active one.
|
|
86
|
+
function gcGraphBin(dir, keepFile) {
|
|
87
|
+
try {
|
|
88
|
+
for (const name of readdirSync(dir)) {
|
|
89
|
+
if (name === 'manifest.json' || name === keepFile) continue;
|
|
90
|
+
if (name.startsWith('mixdog-graph')) {
|
|
91
|
+
try { rmSync(join(dir, name), { force: true }); } catch { /* best-effort */ }
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
} catch { /* dir may not exist yet */ }
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// Sync, network-free lookup of an already-cached binary. Used by the sync
|
|
98
|
+
// _graphBinaryPath() resolver before falling back to an async fetch.
|
|
99
|
+
export function findCachedGraphBinary(dataDir) {
|
|
100
|
+
try {
|
|
101
|
+
const dir = graphBinDir(dataDir);
|
|
102
|
+
const hit = readdirSync(dir).find(
|
|
103
|
+
(n) => n.startsWith('mixdog-graph') && !n.endsWith('.json') && !n.includes('.tmp-'),
|
|
104
|
+
);
|
|
105
|
+
return hit ? join(dir, hit) : null;
|
|
106
|
+
} catch {
|
|
107
|
+
return null;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
let _inflight = null;
|
|
112
|
+
|
|
113
|
+
export function ensureGraphBinary(dataDir) {
|
|
114
|
+
// Single-flight: concurrent callers (prewarm + cache-miss) share one download.
|
|
115
|
+
if (_inflight) return _inflight;
|
|
116
|
+
_inflight = (async () => {
|
|
117
|
+
const manifest = await loadManifest(dataDir);
|
|
118
|
+
const pkey = platformKey();
|
|
119
|
+
const asset = manifest.assets?.[pkey];
|
|
120
|
+
if (!asset || !asset.url || !asset.sha256) {
|
|
121
|
+
throw new Error(`[graph-fetcher] no prebuilt mixdog-graph asset for platform ${pkey}`);
|
|
122
|
+
}
|
|
123
|
+
const version = String(manifest.version || '0');
|
|
124
|
+
const dir = graphBinDir(dataDir);
|
|
125
|
+
mkdirSync(dir, { recursive: true });
|
|
126
|
+
const fileName = `mixdog-graph-${version}${binSuffix()}`;
|
|
127
|
+
const destPath = join(dir, fileName);
|
|
128
|
+
if (existsSync(destPath)) {
|
|
129
|
+
try { if (await sha256File(destPath) === asset.sha256) return destPath; } catch { /* re-download */ }
|
|
130
|
+
}
|
|
131
|
+
const tmpPath = `${destPath}.tmp-${process.pid}-${Date.now()}`;
|
|
132
|
+
await downloadWithRetry(asset.url, tmpPath);
|
|
133
|
+
const actual = await sha256File(tmpPath);
|
|
134
|
+
if (actual !== asset.sha256) {
|
|
135
|
+
try { rmSync(tmpPath, { force: true }); } catch { /* best-effort */ }
|
|
136
|
+
throw new Error(`[graph-fetcher] sha256 mismatch for ${pkey}: expected ${asset.sha256}, got ${actual}`);
|
|
137
|
+
}
|
|
138
|
+
renameSync(tmpPath, destPath);
|
|
139
|
+
if (process.platform !== 'win32') { try { chmodSync(destPath, 0o755); } catch { /* best-effort */ } }
|
|
140
|
+
gcGraphBin(dir, fileName);
|
|
141
|
+
return destPath;
|
|
142
|
+
})().finally(() => { _inflight = null; });
|
|
143
|
+
return _inflight;
|
|
144
|
+
}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": "0.6.5",
|
|
3
|
+
"_comment": "Rewritten by .github/workflows/graph-release.yml on each tagged release. assets maps platformKey (process.platform-process.arch, e.g. win32-x64, linux-x64, darwin-arm64) to { url, sha256 } of the mixdog-graph binary on the GitHub release. A local cargo build under native/mixdog-graph/target/release always takes precedence at runtime. (v0.5.236 entries were filled manually after CI's commit step hit detached HEAD; the workflow now checks out ref: main so future releases self-update.)",
|
|
4
|
+
"assets": {
|
|
5
|
+
"darwin-arm64": {
|
|
6
|
+
"url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-darwin-arm64",
|
|
7
|
+
"sha256": "7016c273a07d19ca9e2f56e8fa7f273fdd40fc41bdc7fef206bf23e31a21a736"
|
|
8
|
+
},
|
|
9
|
+
"darwin-x64": {
|
|
10
|
+
"url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-darwin-x64",
|
|
11
|
+
"sha256": "d076e97da4420f49a6c726bc088a3321e2e7f6a9bfb32d39162c8c53045cfcdb"
|
|
12
|
+
},
|
|
13
|
+
"linux-arm64": {
|
|
14
|
+
"url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-linux-arm64",
|
|
15
|
+
"sha256": "74754562b3c080868738c032c5b6e0e13bc53d7a5277002176b036f8d6681f39"
|
|
16
|
+
},
|
|
17
|
+
"linux-x64": {
|
|
18
|
+
"url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-linux-x64",
|
|
19
|
+
"sha256": "0d8e8bbdd49b18746ed3f972fc3719731a1143ee03ac9e6d86586788b0b431f8"
|
|
20
|
+
},
|
|
21
|
+
"win32-x64": {
|
|
22
|
+
"url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-win32-x64.exe",
|
|
23
|
+
"sha256": "1a671558e5a5f13c7429ff9987a46ad72a71e52241e200d2da820a13d7cbdae7"
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
}
|
|
@@ -0,0 +1,204 @@
|
|
|
1
|
+
// Host-terminal input injection.
|
|
2
|
+
//
|
|
3
|
+
// Exposes `inject_input` as an MCP tool that types text into the *host*
|
|
4
|
+
// terminal currently running Claude Code (and therefore this MCP server,
|
|
5
|
+
// which Claude Code spawned as a child).
|
|
6
|
+
//
|
|
7
|
+
// Strategy:
|
|
8
|
+
// 1. Walk the parent-process chain starting at this Node process.
|
|
9
|
+
// 2. Find the first ancestor whose image name identifies a supported
|
|
10
|
+
// terminal host (currently: powershell / pwsh).
|
|
11
|
+
// 3. Dispatch via a per-host handler map. Adding cmd / Windows Terminal /
|
|
12
|
+
// VS Code / Claude Desktop later means dropping a new entry into
|
|
13
|
+
// `HOST_HANDLERS` — no branching surgery in the dispatcher.
|
|
14
|
+
//
|
|
15
|
+
// The PowerShell handler reuses the proven helper script
|
|
16
|
+
// `scripts/inject-input.ps1` (AttachConsole + CreateFileW("CONIN$") +
|
|
17
|
+
// WriteConsoleInputW, with the stale-STD_INPUT_HANDLE workaround). We
|
|
18
|
+
// shell out to the script rather than reimplementing the FFI in Node so
|
|
19
|
+
// the C# console-input typedef stays in one place.
|
|
20
|
+
//
|
|
21
|
+
// Returned JSON shape: { ok, host, pid, exitCode, stderr? }.
|
|
22
|
+
|
|
23
|
+
import { spawnSync, execFileSync } from 'node:child_process'
|
|
24
|
+
import { writeFileSync, mkdtempSync, rmSync } from 'node:fs'
|
|
25
|
+
import { join, dirname } from 'node:path'
|
|
26
|
+
import { tmpdir } from 'node:os'
|
|
27
|
+
import { fileURLToPath } from 'node:url'
|
|
28
|
+
|
|
29
|
+
const __dirname = dirname(fileURLToPath(import.meta.url))
|
|
30
|
+
// trib-plugin/src/agent/orchestrator/tools/host-input.mjs
|
|
31
|
+
// __dirname = .../trib-plugin/src/agent/orchestrator/tools
|
|
32
|
+
// → trib-plugin requires four `..` (orchestrator → agent → src → root).
|
|
33
|
+
const PLUGIN_ROOT = join(__dirname, '..', '..', '..', '..')
|
|
34
|
+
const PS_HELPER = join(PLUGIN_ROOT, 'scripts', 'inject-input.ps1')
|
|
35
|
+
|
|
36
|
+
// ── Parent-chain walker (Windows) ─────────────────────────────────────
|
|
37
|
+
//
|
|
38
|
+
// Uses one wmic / Get-CimInstance call to read the full Win32_Process
|
|
39
|
+
// table once, then walks PPID links in-process. Keeps the cost to one
|
|
40
|
+
// process spawn regardless of chain depth.
|
|
41
|
+
function readProcessTable() {
|
|
42
|
+
// Prefer PowerShell + Get-CimInstance — wmic.exe is deprecated and
|
|
43
|
+
// missing on newer Windows builds.
|
|
44
|
+
try {
|
|
45
|
+
const out = execFileSync(
|
|
46
|
+
'powershell.exe',
|
|
47
|
+
[
|
|
48
|
+
'-NoProfile', '-NonInteractive', '-Command',
|
|
49
|
+
"Get-CimInstance Win32_Process | Select-Object ProcessId,ParentProcessId,Name | ConvertTo-Json -Compress",
|
|
50
|
+
],
|
|
51
|
+
{ encoding: 'utf8', windowsHide: true, timeout: 8000, maxBuffer: 16 * 1024 * 1024 },
|
|
52
|
+
)
|
|
53
|
+
const parsed = JSON.parse(out)
|
|
54
|
+
const rows = Array.isArray(parsed) ? parsed : [parsed]
|
|
55
|
+
const byPid = new Map()
|
|
56
|
+
for (const row of rows) {
|
|
57
|
+
if (!row || row.ProcessId == null) continue
|
|
58
|
+
byPid.set(Number(row.ProcessId), {
|
|
59
|
+
pid: Number(row.ProcessId),
|
|
60
|
+
ppid: Number(row.ParentProcessId ?? 0),
|
|
61
|
+
name: String(row.Name ?? ''),
|
|
62
|
+
})
|
|
63
|
+
}
|
|
64
|
+
return byPid
|
|
65
|
+
} catch (e) {
|
|
66
|
+
throw new Error(`failed to read process table: ${e?.message || e}`)
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
// Image-name → host-tag map. Lower-case match.
|
|
71
|
+
const HOST_BY_IMAGE = {
|
|
72
|
+
'pwsh.exe': 'powershell',
|
|
73
|
+
'powershell.exe': 'powershell',
|
|
74
|
+
// Future:
|
|
75
|
+
// 'cmd.exe': 'cmd',
|
|
76
|
+
// 'windowsterminal.exe': 'windows-terminal',
|
|
77
|
+
// 'code.exe': 'vscode-terminal',
|
|
78
|
+
// 'claude.exe': 'claude-desktop',
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
function classifyHost(imageName) {
|
|
82
|
+
if (!imageName) return null
|
|
83
|
+
return HOST_BY_IMAGE[imageName.toLowerCase()] ?? null
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Walk parent chain from the given pid, returning the first ancestor whose
|
|
87
|
+
// image maps to a known host. Bounded to 32 hops and skips PID 0 / orphan.
|
|
88
|
+
function findHostAncestor(startPid) {
|
|
89
|
+
const table = readProcessTable()
|
|
90
|
+
let cur = table.get(Number(startPid))
|
|
91
|
+
let hops = 0
|
|
92
|
+
const trail = []
|
|
93
|
+
while (cur && hops < 32) {
|
|
94
|
+
trail.push({ pid: cur.pid, name: cur.name })
|
|
95
|
+
const tag = classifyHost(cur.name)
|
|
96
|
+
if (tag) return { host: tag, pid: cur.pid, name: cur.name, trail }
|
|
97
|
+
if (!cur.ppid || cur.ppid === cur.pid) break
|
|
98
|
+
cur = table.get(cur.ppid)
|
|
99
|
+
hops += 1
|
|
100
|
+
}
|
|
101
|
+
return { host: null, pid: null, name: null, trail }
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// ── Per-host handlers ─────────────────────────────────────────────────
|
|
105
|
+
|
|
106
|
+
function handlePowerShell(text, pid) {
|
|
107
|
+
// Pass the payload via a temp file rather than a command-line arg so
|
|
108
|
+
// arbitrary characters (quotes, backticks, $, newlines) survive without
|
|
109
|
+
// any shell or PowerShell parsing pitfalls. The .ps1 helper is invoked
|
|
110
|
+
// directly via powershell.exe -File, no -Command interpolation.
|
|
111
|
+
const tmpDir = mkdtempSync(join(tmpdir(), 'mixdog-inject-'))
|
|
112
|
+
const payloadPath = join(tmpDir, 'payload.txt')
|
|
113
|
+
let result
|
|
114
|
+
try {
|
|
115
|
+
writeFileSync(payloadPath, String(text), 'utf8')
|
|
116
|
+
// The helper script's -Text param expects a single string. Read the
|
|
117
|
+
// payload file inside a tiny -Command shim so we don't have to touch
|
|
118
|
+
// the existing helper. -Raw preserves embedded newlines and trailing
|
|
119
|
+
// whitespace exactly as the caller wrote them.
|
|
120
|
+
const psCommand =
|
|
121
|
+
`$ErrorActionPreference='Stop'; ` +
|
|
122
|
+
`$txt = Get-Content -LiteralPath ${psQuote(payloadPath)} -Raw -Encoding UTF8; ` +
|
|
123
|
+
`& ${psQuote(PS_HELPER)} -TargetPid ${Number(pid)} -Text $txt`
|
|
124
|
+
result = spawnSync(
|
|
125
|
+
'powershell.exe',
|
|
126
|
+
['-NoProfile', '-NonInteractive', '-ExecutionPolicy', 'Bypass', '-Command', psCommand],
|
|
127
|
+
{ encoding: 'utf8', windowsHide: true, timeout: 15000 },
|
|
128
|
+
)
|
|
129
|
+
} finally {
|
|
130
|
+
try { rmSync(tmpDir, { recursive: true, force: true }) } catch {}
|
|
131
|
+
}
|
|
132
|
+
const exitCode = result.status ?? -1
|
|
133
|
+
const stderr = (result.stderr || '').trim()
|
|
134
|
+
const ok = exitCode === 0
|
|
135
|
+
const out = { ok, exitCode }
|
|
136
|
+
if (!ok && stderr) out.stderr = stderr
|
|
137
|
+
return out
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
// PowerShell single-quoted literal: escape ' as ''. Wraps the whole value
|
|
141
|
+
// in single quotes so $ / backtick / parentheses are taken literally.
|
|
142
|
+
function psQuote(s) {
|
|
143
|
+
return `'${String(s).replace(/'/g, "''")}'`
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
const HOST_HANDLERS = {
|
|
147
|
+
powershell: handlePowerShell,
|
|
148
|
+
// cmd: handleCmd,
|
|
149
|
+
// 'windows-terminal': handleWT,
|
|
150
|
+
// 'vscode-terminal': handleVSCode,
|
|
151
|
+
// 'claude-desktop': handleClaudeDesktop,
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// ── Public entry ──────────────────────────────────────────────────────
|
|
155
|
+
|
|
156
|
+
export const HOST_INPUT_TOOL_DEFS = [
|
|
157
|
+
{
|
|
158
|
+
name: 'inject_input',
|
|
159
|
+
title: 'Inject Input',
|
|
160
|
+
annotations: { title: 'Inject Input', readOnlyHint: false, destructiveHint: true, idempotentHint: false, openWorldHint: true, compressible: false },
|
|
161
|
+
description: 'Inject text into the host session as if the user typed it at the prompt. Requires text.',
|
|
162
|
+
inputSchema: {
|
|
163
|
+
type: 'object',
|
|
164
|
+
properties: {
|
|
165
|
+
text: { type: 'string', minLength: 1 },
|
|
166
|
+
},
|
|
167
|
+
required: ['text'],
|
|
168
|
+
additionalProperties: false,
|
|
169
|
+
},
|
|
170
|
+
},
|
|
171
|
+
]
|
|
172
|
+
|
|
173
|
+
export async function executeHostInputTool(name, args /*, cwd */) {
|
|
174
|
+
if (name !== 'inject_input') throw new Error(`Unknown host-input tool: ${name}`)
|
|
175
|
+
const text = args?.text
|
|
176
|
+
if (typeof text !== 'string' || text.length === 0) {
|
|
177
|
+
throw new Error('inject_input: `text` (string, non-empty) is required')
|
|
178
|
+
}
|
|
179
|
+
if (process.platform !== 'win32') {
|
|
180
|
+
throw new Error(`inject_input: only supported on Windows (got platform=${process.platform})`)
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// Always submit: append a trailing newline if the caller didn't.
|
|
184
|
+
const payload = text.endsWith('\n') ? text : text + '\n'
|
|
185
|
+
|
|
186
|
+
const { host, pid, name: imageName, trail } = findHostAncestor(process.pid)
|
|
187
|
+
if (!host) {
|
|
188
|
+
const trailStr = trail.map(t => `${t.pid}:${t.name}`).join(' → ')
|
|
189
|
+
throw new Error(`inject_input: no supported terminal host found in parent chain (${trailStr || '<empty>'})`)
|
|
190
|
+
}
|
|
191
|
+
const handler = HOST_HANDLERS[host]
|
|
192
|
+
if (typeof handler !== 'function') {
|
|
193
|
+
throw new Error(`inject_input: host "${host}" not supported yet (resolved image=${imageName} pid=${pid})`)
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
const result = handler(payload, pid)
|
|
197
|
+
return JSON.stringify({
|
|
198
|
+
ok: !!result.ok,
|
|
199
|
+
host,
|
|
200
|
+
pid,
|
|
201
|
+
exitCode: result.exitCode,
|
|
202
|
+
...(result.stderr ? { stderr: result.stderr } : {}),
|
|
203
|
+
})
|
|
204
|
+
}
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
import * as nodeBuffer from 'node:buffer';
|
|
2
|
+
import { readFileSync } from 'node:fs';
|
|
3
|
+
import { readFile as readFileAsync } from 'node:fs/promises';
|
|
4
|
+
import { decodeRawBufferForSnapshotCheck } from './builtin/snapshot-helpers.mjs';
|
|
5
|
+
|
|
6
|
+
function keyForPath(fullPath) {
|
|
7
|
+
const text = String(fullPath || '');
|
|
8
|
+
return process.platform === 'win32' ? text.toLowerCase() : text;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export function isValidUtf8Buffer(buf) {
|
|
12
|
+
if (typeof nodeBuffer.isUtf8 === 'function') return nodeBuffer.isUtf8(buf);
|
|
13
|
+
if (typeof Buffer.isUtf8 === 'function') return Buffer.isUtf8(buf);
|
|
14
|
+
return Buffer.from(buf.toString('utf-8'), 'utf-8').equals(buf);
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export function createMutationContentCache() {
|
|
18
|
+
const entries = new Map();
|
|
19
|
+
|
|
20
|
+
function getOrCreate(fullPath) {
|
|
21
|
+
const key = keyForPath(fullPath);
|
|
22
|
+
let entry = entries.get(key);
|
|
23
|
+
if (!entry) {
|
|
24
|
+
entry = { fullPath, rawBuf: null, content: null };
|
|
25
|
+
entries.set(key, entry);
|
|
26
|
+
}
|
|
27
|
+
return entry;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
function seedBuffer(fullPath, rawBuf) {
|
|
31
|
+
const entry = getOrCreate(fullPath);
|
|
32
|
+
entry.rawBuf = rawBuf;
|
|
33
|
+
entry.content = null;
|
|
34
|
+
return entry;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function readBufferSync(fullPath) {
|
|
38
|
+
const entry = getOrCreate(fullPath);
|
|
39
|
+
if (!Buffer.isBuffer(entry.rawBuf)) entry.rawBuf = readFileSync(fullPath);
|
|
40
|
+
return entry.rawBuf;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
async function readBuffer(fullPath) {
|
|
44
|
+
const entry = getOrCreate(fullPath);
|
|
45
|
+
if (!Buffer.isBuffer(entry.rawBuf)) entry.rawBuf = await readFileAsync(fullPath);
|
|
46
|
+
return entry.rawBuf;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
function readTextSync(fullPath) {
|
|
50
|
+
const entry = getOrCreate(fullPath);
|
|
51
|
+
if (typeof entry.content !== 'string') {
|
|
52
|
+
entry.content = decodeRawBufferForSnapshotCheck(readBufferSync(fullPath));
|
|
53
|
+
}
|
|
54
|
+
return entry.content;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
function getEntry(fullPath) {
|
|
58
|
+
return entries.get(keyForPath(fullPath)) || null;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
function clear(fullPath = null) {
|
|
62
|
+
if (fullPath) entries.delete(keyForPath(fullPath));
|
|
63
|
+
else entries.clear();
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
return { seedBuffer, readBuffer, readBufferSync, readTextSync, getEntry, clear };
|
|
67
|
+
}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
// NOTE: post-edit V4A fallback was removed.
|
|
2
|
+
//
|
|
3
|
+
// The removed V4A edit reroute used to emit every `old_string` line as a
|
|
4
|
+
// `-` with zero context_before/after,
|
|
5
|
+
// which meant the V4A matcher had to find the same byte slice the exact
|
|
6
|
+
// edit just failed on. Code 8 ("old_string not found") is by definition a
|
|
7
|
+
// byte/context mismatch (tab-vs-space drift, stale snapshot, fold-tier
|
|
8
|
+
// miss), so the V4A engine fails for the same reason and the user sees a
|
|
9
|
+
// noisy double-error block. The exact-unique byte path now handles large
|
|
10
|
+
// chunks natively, so the V4A pre-route is dead and has been removed.
|
|
11
|
+
|
|
12
|
+
export function planEditMutationRoute() {
|
|
13
|
+
// Large-chunk V4A reroute removed: an EXACT-UNIQUE byte match is safe
|
|
14
|
+
// at any size, so the edit-exact tier handles >=30-line chunks
|
|
15
|
+
// natively (size gate now guards only the fold/fuzzy fallback). The
|
|
16
|
+
// V4A pre-route used to fail anchor-not-found on bytes the exact tier
|
|
17
|
+
// WOULD match, then priorResult short-circuited and surfaced a noisy
|
|
18
|
+
// double-error block. Always route through edit-exact.
|
|
19
|
+
return { engine: 'edit-exact', reason: 'default' };
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export function isMutationPlanRoutable(plan) {
|
|
23
|
+
return plan?.engine === 'v4a-patch' && plan?.patchArgs?.patch;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function formatMutationRouteLine(plan = {}, extras = {}) {
|
|
27
|
+
const source = plan.sourceTool || extras.sourceTool || 'unknown';
|
|
28
|
+
const engine = plan.engine || extras.engine || 'unknown';
|
|
29
|
+
const reason = plan.reason || extras.reason || 'unknown';
|
|
30
|
+
const suffix = Object.entries(extras)
|
|
31
|
+
.filter(([key, value]) => key !== 'sourceTool' && key !== 'engine' && key !== 'reason' && value !== undefined && value !== null && value !== '')
|
|
32
|
+
.map(([key, value]) => `${key}=${String(value).replace(/\s+/g, '-')}`)
|
|
33
|
+
.join(' ');
|
|
34
|
+
return `mutation_route: source=${source} engine=${engine} reason=${reason}${suffix ? ` ${suffix}` : ''}`;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export function wrapMutationRouteOutput(text, plan = {}, extras = {}) {
|
|
38
|
+
const body = String(text ?? '');
|
|
39
|
+
if (/^mutation_route:/i.test(body.trimStart())) return body;
|
|
40
|
+
return `${formatMutationRouteLine(plan, extras)}\n${body}`;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export async function executeMutationPlan(plan, context = {}) {
|
|
44
|
+
if (!isMutationPlanRoutable(plan)) return null;
|
|
45
|
+
const { executePatchTool } = await import('./patch.mjs');
|
|
46
|
+
let out;
|
|
47
|
+
try {
|
|
48
|
+
out = await executePatchTool('apply_patch', {
|
|
49
|
+
base_path: context.workDir,
|
|
50
|
+
...plan.patchArgs,
|
|
51
|
+
}, context.workDir, {
|
|
52
|
+
...context.options,
|
|
53
|
+
readStateScope: context.readStateScope,
|
|
54
|
+
sessionId: context.options?.sessionId,
|
|
55
|
+
mutationPlan: plan,
|
|
56
|
+
});
|
|
57
|
+
} catch (err) {
|
|
58
|
+
return { ok: false, text: err?.message || String(err), plan };
|
|
59
|
+
}
|
|
60
|
+
const text = String(out ?? '');
|
|
61
|
+
if (/^Error:/i.test(text.trimStart())) {
|
|
62
|
+
return { ok: false, text: out, plan };
|
|
63
|
+
}
|
|
64
|
+
return {
|
|
65
|
+
ok: true,
|
|
66
|
+
text: wrapMutationRouteOutput(text, plan),
|
|
67
|
+
plan,
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
export function appendMutationPlanFailure(originalResult, plannedResult) {
|
|
72
|
+
if (!plannedResult || plannedResult.ok) return originalResult;
|
|
73
|
+
const plan = plannedResult.plan || {};
|
|
74
|
+
return `${originalResult}\nmutation_route: source=${plan.sourceTool || 'unknown'} engine=${plan.engine || 'unknown'} reason=${plan.reason || 'unknown'} failed\n${plannedResult.text}`;
|
|
75
|
+
}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
export function parseJsonNextCalls(text) {
|
|
2
|
+
const input = String(text ?? '');
|
|
3
|
+
if (!input.includes('next_call:')) return [];
|
|
4
|
+
const out = [];
|
|
5
|
+
const re = /next_call:\s*([A-Za-z_][\w]*)\(/g;
|
|
6
|
+
for (const match of input.matchAll(re)) {
|
|
7
|
+
const parsed = parseNextCallArgs(input, match.index + match[0].length);
|
|
8
|
+
if (!parsed) continue;
|
|
9
|
+
out.push({ tool: match[1], args: parsed.args, start: match.index, end: parsed.end });
|
|
10
|
+
}
|
|
11
|
+
return out;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export function countJsonNextCalls(text) {
|
|
15
|
+
return parseJsonNextCalls(text).length;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
function parseNextCallArgs(text, start) {
|
|
19
|
+
let i = start;
|
|
20
|
+
while (i < text.length && /\s/.test(text[i])) i++;
|
|
21
|
+
if (text[i] !== '{') return null;
|
|
22
|
+
let depth = 0;
|
|
23
|
+
let inString = false;
|
|
24
|
+
let escaped = false;
|
|
25
|
+
for (; i < text.length; i++) {
|
|
26
|
+
const ch = text[i];
|
|
27
|
+
if (inString) {
|
|
28
|
+
if (escaped) escaped = false;
|
|
29
|
+
else if (ch === '\\') escaped = true;
|
|
30
|
+
else if (ch === '"') inString = false;
|
|
31
|
+
continue;
|
|
32
|
+
}
|
|
33
|
+
if (ch === '"') inString = true;
|
|
34
|
+
else if (ch === '{') depth++;
|
|
35
|
+
else if (ch === '}') {
|
|
36
|
+
depth--;
|
|
37
|
+
if (depth === 0) {
|
|
38
|
+
const raw = text.slice(start, i + 1).trim();
|
|
39
|
+
let j = i + 1;
|
|
40
|
+
while (j < text.length && /\s/.test(text[j])) j++;
|
|
41
|
+
if (text[j] !== ')') return null;
|
|
42
|
+
try { return { args: JSON.parse(raw), end: j + 1 }; }
|
|
43
|
+
catch { return null; }
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
return null;
|
|
48
|
+
}
|