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,350 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Unified config reader/writer.
|
|
3
|
+
* Single file: mixdog-config.json with sections: channels, agent, memory, search.
|
|
4
|
+
*/
|
|
5
|
+
import { readFileSync, mkdirSync, existsSync } from 'fs'
|
|
6
|
+
import { join, dirname } from 'path'
|
|
7
|
+
import { createRequire } from 'module'
|
|
8
|
+
import { resolvePluginData } from './plugin-paths.mjs'
|
|
9
|
+
import { renameWithRetrySync, writeJsonAtomicSync, withFileLockSync } from './atomic-file.mjs'
|
|
10
|
+
import {
|
|
11
|
+
backupUserData,
|
|
12
|
+
markUserDataInitialized,
|
|
13
|
+
loadLatestMixdogConfigFromBackup,
|
|
14
|
+
hasUserDataInitMarker,
|
|
15
|
+
} from './user-data-guard.mjs'
|
|
16
|
+
|
|
17
|
+
const _require = createRequire(import.meta.url)
|
|
18
|
+
const { getSecret: _getSecret, setSecret: _setSecret, deleteSecret: _deleteSecret } = _require('../../lib/keychain-cjs.cjs')
|
|
19
|
+
|
|
20
|
+
const DATA_DIR = resolvePluginData()
|
|
21
|
+
|
|
22
|
+
const CONFIG_PATH = join(DATA_DIR, 'mixdog-config.json')
|
|
23
|
+
|
|
24
|
+
const GENERATED_KEY = '_generated'
|
|
25
|
+
|
|
26
|
+
function isPlainObject(value) {
|
|
27
|
+
return !!value && typeof value === 'object' && !Array.isArray(value)
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export function stripGeneratedMarker(data) {
|
|
31
|
+
if (!isPlainObject(data) || !Object.prototype.hasOwnProperty.call(data, GENERATED_KEY)) return data
|
|
32
|
+
const { [GENERATED_KEY]: _generated, ...rest } = data
|
|
33
|
+
return rest
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function readJsonFile(path) {
|
|
37
|
+
let raw
|
|
38
|
+
try {
|
|
39
|
+
raw = readFileSync(path, 'utf8')
|
|
40
|
+
} catch (err) {
|
|
41
|
+
if (err.code === 'ENOENT') return null
|
|
42
|
+
// Fail closed on unknown read errors (EACCES, EIO, …). Returning {}
|
|
43
|
+
// here would let a subsequent writeSection() serialize an empty
|
|
44
|
+
// object back over an existing-but-temporarily-unreadable config and
|
|
45
|
+
// erase every other section. Better to surface the read error and
|
|
46
|
+
// abort the RMW than silently destroy data.
|
|
47
|
+
process.stderr.write(`[config] readJsonFile: unexpected read error for ${path}: ${err.message}\n`)
|
|
48
|
+
throw err
|
|
49
|
+
}
|
|
50
|
+
try {
|
|
51
|
+
return JSON.parse(raw)
|
|
52
|
+
} catch (err) {
|
|
53
|
+
// Quarantine a malformed mixdog-config.json so the next boot starts fresh
|
|
54
|
+
// instead of looping on a broken file.
|
|
55
|
+
if (path === CONFIG_PATH) {
|
|
56
|
+
const corrupt = `${path}.corrupt-${Date.now()}`
|
|
57
|
+
try { renameWithRetrySync(path, corrupt) } catch {}
|
|
58
|
+
process.stderr.write(`[config] mixdog-config.json is malformed (${err.message}). Renamed to ${corrupt}. Restore it or delete to start fresh.\n`)
|
|
59
|
+
}
|
|
60
|
+
return null
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
function writeJsonFile(path, data) {
|
|
65
|
+
// R4 data-at-rest: mixdog-config.json holds provider apiKeys; clamp to
|
|
66
|
+
// owner-only on POSIX via 0o600/0o700 mode bits, AND fail-closed on
|
|
67
|
+
// Windows where those bits are advisory — `secret: true` makes the
|
|
68
|
+
// atomic writer apply an owner-only NTFS ACL (icacls) to the file,
|
|
69
|
+
// temp, lock, and parent dir, throwing if the ACL cannot be enforced
|
|
70
|
+
// so the key is never left world-readable. 0o700 on the parent dir
|
|
71
|
+
// restricts directory traversal in shared-home setups on POSIX.
|
|
72
|
+
mkdirSync(dirname(path), { recursive: true, mode: 0o700 })
|
|
73
|
+
// NOTE: lock:false here — callers that perform read-modify-write
|
|
74
|
+
// (writeSection/updateSection) hold the lock at the outer RMW
|
|
75
|
+
// boundary, so the inner write must not try to re-acquire the same
|
|
76
|
+
// lock file (would self-deadlock on `openSync('wx')`). Direct
|
|
77
|
+
// whole-config writers go through `writeAllLocked` below.
|
|
78
|
+
if (path === CONFIG_PATH) {
|
|
79
|
+
try { backupUserData(DATA_DIR, 'pre-config-write') } catch {}
|
|
80
|
+
}
|
|
81
|
+
writeJsonAtomicSync(path, data, { lock: false, fsyncDir: true, mode: 0o600, secret: true })
|
|
82
|
+
if (path === CONFIG_PATH) {
|
|
83
|
+
try { markUserDataInitialized(DATA_DIR) } catch {}
|
|
84
|
+
try { backupUserData(DATA_DIR, 'post-config-write') } catch {}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
function readAll() {
|
|
89
|
+
const parsed = readJsonFile(CONFIG_PATH)
|
|
90
|
+
if (parsed != null) return parsed
|
|
91
|
+
// Symmetric with readAllForRmW(): a missing/unreadable config on a data
|
|
92
|
+
// dir that was previously initialized means the file was LOST (deleted,
|
|
93
|
+
// failed write), not a fresh install. Self-heal from the newest
|
|
94
|
+
// structurally-complete backup instead of silently collapsing to {} —
|
|
95
|
+
// which callers (readSection/getCapabilities) merge with in-memory
|
|
96
|
+
// defaults, dropping the user's presets/roles/sections. In-memory only:
|
|
97
|
+
// readAll() runs OUTSIDE the config lock (unlike the RMW path), so we do
|
|
98
|
+
// not write here to avoid an unlocked-write race; the next writeSection
|
|
99
|
+
// re-persists the file under the lock.
|
|
100
|
+
if (hasUserDataInitMarker(DATA_DIR)) {
|
|
101
|
+
const restored = loadLatestMixdogConfigFromBackup(DATA_DIR)
|
|
102
|
+
if (restored && isPlainObject(restored)) {
|
|
103
|
+
process.stderr.write('[config] read: restored mixdog-config.json from latest user-data backup (missing after init)\n')
|
|
104
|
+
return restored
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
return {}
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
function quarantineMalformedConfig(parseErr) {
|
|
111
|
+
const corrupt = `${CONFIG_PATH}.corrupt-${Date.now()}`
|
|
112
|
+
try {
|
|
113
|
+
if (existsSync(CONFIG_PATH)) renameWithRetrySync(CONFIG_PATH, corrupt)
|
|
114
|
+
} catch {}
|
|
115
|
+
process.stderr.write(
|
|
116
|
+
`[config] mixdog-config.json is malformed (${parseErr.message}). Renamed to ${corrupt}. Restore it or delete to start fresh.\n`,
|
|
117
|
+
)
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
function restoreAllForRmWOrThrow(reason) {
|
|
121
|
+
const restored = loadLatestMixdogConfigFromBackup(DATA_DIR)
|
|
122
|
+
if (restored && isPlainObject(restored)) {
|
|
123
|
+
process.stderr.write(`[config] RMW read: restored mixdog-config.json from latest user-data backup (${reason})\n`)
|
|
124
|
+
return restored
|
|
125
|
+
}
|
|
126
|
+
throw new Error(
|
|
127
|
+
`[config] mixdog-config.json is unreadable and no valid backup was found (${reason}); refusing section write that would wipe other sections`,
|
|
128
|
+
)
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Read-modify-write baseline. ENOENT on a never-initialized data dir → {}.
|
|
133
|
+
* Malformed on-disk config (or missing config after prior init) → restore
|
|
134
|
+
* from backup or throw; never silently collapse to a one-section overwrite.
|
|
135
|
+
*/
|
|
136
|
+
function readAllForRmW() {
|
|
137
|
+
let raw
|
|
138
|
+
try {
|
|
139
|
+
raw = readFileSync(CONFIG_PATH, 'utf8')
|
|
140
|
+
} catch (err) {
|
|
141
|
+
if (err.code === 'ENOENT') {
|
|
142
|
+
if (hasUserDataInitMarker(DATA_DIR)) {
|
|
143
|
+
return restoreAllForRmWOrThrow('config file missing after user-data was initialized')
|
|
144
|
+
}
|
|
145
|
+
return {}
|
|
146
|
+
}
|
|
147
|
+
process.stderr.write(`[config] readAllForRmW: unexpected read error for ${CONFIG_PATH}: ${err.message}\n`)
|
|
148
|
+
throw err
|
|
149
|
+
}
|
|
150
|
+
try {
|
|
151
|
+
const parsed = JSON.parse(raw)
|
|
152
|
+
if (!isPlainObject(parsed)) throw new SyntaxError('config root must be a JSON object')
|
|
153
|
+
return parsed
|
|
154
|
+
} catch (parseErr) {
|
|
155
|
+
quarantineMalformedConfig(parseErr)
|
|
156
|
+
return restoreAllForRmWOrThrow(parseErr.message)
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
function writeAll(data) {
|
|
161
|
+
writeJsonFile(CONFIG_PATH, data)
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
// Serialize a read-modify-write under the same file lock. Concurrent
|
|
165
|
+
// processes used to race here: each read the old config, each computed
|
|
166
|
+
// `all[section] = …`, each atomic-wrote — the later writer would
|
|
167
|
+
// silently clobber the earlier section update. Holding the lock across
|
|
168
|
+
// read+modify+write keeps RMW linearizable.
|
|
169
|
+
function withConfigLock(fn) {
|
|
170
|
+
// secret:true clamps the lock file (which sits beside the API-key
|
|
171
|
+
// config in a possibly shared home dir) to an owner-only ACL on win32,
|
|
172
|
+
// fail-closed. POSIX is unaffected.
|
|
173
|
+
return withFileLockSync(`${CONFIG_PATH}.lock`, fn, { secret: true })
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
export function readSection(section) {
|
|
177
|
+
return stripGeneratedMarker(readAll()[section] ?? null) ?? {}
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
export function writeSection(section, data) {
|
|
181
|
+
withConfigLock(() => {
|
|
182
|
+
const all = readAllForRmW()
|
|
183
|
+
all[section] = stripGeneratedMarker(data)
|
|
184
|
+
writeAll(all)
|
|
185
|
+
})
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
export function updateSection(section, updater) {
|
|
189
|
+
withConfigLock(() => {
|
|
190
|
+
const all = readAllForRmW()
|
|
191
|
+
const current = stripGeneratedMarker(all[section] || {})
|
|
192
|
+
all[section] = stripGeneratedMarker(typeof updater === 'function' ? updater(current) : updater)
|
|
193
|
+
writeAll(all)
|
|
194
|
+
})
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
// ── Capabilities (B2 central path policy) ───────────────────────────
|
|
198
|
+
// Top-level `capabilities` section in mixdog-config.json. Safe defaults
|
|
199
|
+
// win on missing/malformed input — every cap is OFF unless explicitly
|
|
200
|
+
// enabled. Settings round-trip through the setup UI; the in-process
|
|
201
|
+
// path gate reads them via `getCapabilities()`.
|
|
202
|
+
//
|
|
203
|
+
// homeAccess: when true, file tools may write anywhere under $HOME. When
|
|
204
|
+
// false (default), file tools are cwd-scoped — matches the setup UI's
|
|
205
|
+
// out-of-the-box "OFF" toggle so a fresh install is restrictive until the
|
|
206
|
+
// user explicitly opts in. This ONLY controls the main-agent path gate —
|
|
207
|
+
// bridge role Edit/Write to HOME paths always go through Discord approval
|
|
208
|
+
// regardless (enforced in hooks/pre-tool-subagent.cjs).
|
|
209
|
+
const CAPABILITY_DEFAULTS = Object.freeze({ homeAccess: false })
|
|
210
|
+
|
|
211
|
+
function readCapabilities() {
|
|
212
|
+
const raw = readAll().capabilities
|
|
213
|
+
const out = { ...CAPABILITY_DEFAULTS }
|
|
214
|
+
if (raw && typeof raw === 'object') {
|
|
215
|
+
if (raw.homeAccess === true) out.homeAccess = true
|
|
216
|
+
else if (raw.homeAccess === false) out.homeAccess = false
|
|
217
|
+
}
|
|
218
|
+
return out
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
// Convenience alias requested by B2 call-site plumbing. Returns the
|
|
222
|
+
// same object shape as readCapabilities(); callers that only need a
|
|
223
|
+
// boolean can read `.homeAccess` directly.
|
|
224
|
+
export function getCapabilities() {
|
|
225
|
+
return readCapabilities()
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
// ── Secret account names ─────────────────────────────────────────────────────
|
|
229
|
+
// Canonical account strings stored in the OS keychain. Must stay in sync with
|
|
230
|
+
// the migration logic in seed.mjs.
|
|
231
|
+
export const SECRET_ACCOUNTS = Object.freeze({
|
|
232
|
+
discordToken: 'discord.token',
|
|
233
|
+
webhookAuth: 'webhook.authtoken',
|
|
234
|
+
searchApiKey: (provider) => `search.${provider}.apiKey`,
|
|
235
|
+
agentApiKey: (provider) => `agent.${provider}.apiKey`,
|
|
236
|
+
})
|
|
237
|
+
|
|
238
|
+
export function isDiscordSnowflake(value) {
|
|
239
|
+
return /^\d{17,20}$/.test(String(value || '').trim())
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
export function diagnoseDiscordTokenValue(value, config = {}) {
|
|
243
|
+
const token = String(value || '').trim()
|
|
244
|
+
if (!token) return null
|
|
245
|
+
const discord = config?.discord && typeof config.discord === 'object' ? config.discord : {}
|
|
246
|
+
const appId = String(discord.applicationId || '').trim()
|
|
247
|
+
if (appId && token === appId) return 'Bot token field contains the Application ID, not the bot token.'
|
|
248
|
+
const channels = config?.channelsConfig && typeof config.channelsConfig === 'object' ? config.channelsConfig : {}
|
|
249
|
+
for (const ch of Object.values(channels)) {
|
|
250
|
+
const channelId = String(ch?.channelId || '').trim()
|
|
251
|
+
if (channelId && token === channelId) return 'Bot token field contains a Channel ID, not the bot token.'
|
|
252
|
+
}
|
|
253
|
+
if (isDiscordSnowflake(token)) return 'Bot token field contains a numeric Discord ID, not the bot token.'
|
|
254
|
+
return null
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
// ── Secret-aware getters ─────────────────────────────────────────────────────
|
|
258
|
+
// Read order: ENV MIXDOG_<UPPER_SNAKE> → OS keychain → null.
|
|
259
|
+
|
|
260
|
+
function _envKey(account) {
|
|
261
|
+
// 'discord.token' → 'MIXDOG_DISCORD_TOKEN'
|
|
262
|
+
return 'MIXDOG_' + account.replace(/[.\s]+/g, '_').toUpperCase()
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
function _readSecret(account) {
|
|
266
|
+
const envVal = process.env[_envKey(account)]
|
|
267
|
+
if (envVal) return envVal
|
|
268
|
+
try { return _getSecret(account) } catch { return null }
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
/**
|
|
272
|
+
* Returns the Discord bot token.
|
|
273
|
+
* Priority: MIXDOG_DISCORD_TOKEN → keychain('discord.token') → null
|
|
274
|
+
*/
|
|
275
|
+
export function getDiscordToken() {
|
|
276
|
+
return _readSecret(SECRET_ACCOUNTS.discordToken)
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
/**
|
|
280
|
+
* Returns the ngrok/webhook authtoken.
|
|
281
|
+
* Priority: MIXDOG_WEBHOOK_AUTHTOKEN → keychain('webhook.authtoken') → null
|
|
282
|
+
*/
|
|
283
|
+
export function getWebhookAuthtoken() {
|
|
284
|
+
return _readSecret(SECRET_ACCOUNTS.webhookAuth)
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
const SEARCH_PROVIDERS = ['firecrawl', 'tavily', 'exa']
|
|
288
|
+
|
|
289
|
+
/**
|
|
290
|
+
* Returns the API key for a search provider.
|
|
291
|
+
* Priority: MIXDOG_SEARCH_<PROVIDER>_APIKEY → keychain('search.<provider>.apiKey') → null
|
|
292
|
+
*/
|
|
293
|
+
export function getSearchApiKey(provider) {
|
|
294
|
+
if (!SEARCH_PROVIDERS.includes(provider)) return null
|
|
295
|
+
return _readSecret(SECRET_ACCOUNTS.searchApiKey(provider))
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
// Standard provider env names take precedence so existing OPENAI_API_KEY-style
|
|
299
|
+
// exports keep working, then MIXDOG_AGENT_<P>_APIKEY, then the OS keychain.
|
|
300
|
+
const AGENT_PROVIDER_ENV = Object.freeze({
|
|
301
|
+
openai: 'OPENAI_API_KEY', anthropic: 'ANTHROPIC_API_KEY', gemini: 'GEMINI_API_KEY',
|
|
302
|
+
deepseek: 'DEEPSEEK_API_KEY', xai: 'XAI_API_KEY', nvidia: 'NVIDIA_API_KEY',
|
|
303
|
+
})
|
|
304
|
+
|
|
305
|
+
// Last-resort env aliases honored AFTER the standard env / MIXDOG_AGENT_* /
|
|
306
|
+
// keychain sources. GROK_API_KEY is the established xAI alias elsewhere in the
|
|
307
|
+
// repo (search discovery, xai-api/grok-oauth backends), so honoring it here
|
|
308
|
+
// keeps provider discovery and dispatch resolving the same credential.
|
|
309
|
+
const AGENT_PROVIDER_ENV_ALIASES = Object.freeze({
|
|
310
|
+
xai: ['GROK_API_KEY'],
|
|
311
|
+
})
|
|
312
|
+
|
|
313
|
+
/**
|
|
314
|
+
* Returns the API key for an agent provider.
|
|
315
|
+
* Priority: <PROVIDER>_API_KEY env -> MIXDOG_AGENT_<PROVIDER>_APIKEY -> keychain('agent.<provider>.apiKey') -> alias env (e.g. GROK_API_KEY for xai) -> null.
|
|
316
|
+
* Never reads mixdog-config.json — provider keys are keychain-only.
|
|
317
|
+
*/
|
|
318
|
+
export function getAgentApiKey(provider) {
|
|
319
|
+
const std = AGENT_PROVIDER_ENV[provider]
|
|
320
|
+
if (std && process.env[std]) return process.env[std]
|
|
321
|
+
const fromStore = _readSecret(SECRET_ACCOUNTS.agentApiKey(provider))
|
|
322
|
+
if (fromStore) return fromStore
|
|
323
|
+
for (const alias of AGENT_PROVIDER_ENV_ALIASES[provider] || []) {
|
|
324
|
+
if (process.env[alias]) return process.env[alias]
|
|
325
|
+
}
|
|
326
|
+
return null
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
/**
|
|
330
|
+
* Persist a secret to the OS keychain. Throws on failure.
|
|
331
|
+
* Never writes to mixdog-config.json.
|
|
332
|
+
*/
|
|
333
|
+
export function saveSecret(account, value) {
|
|
334
|
+
_setSecret(account, value)
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
export function deleteSecret(account) {
|
|
338
|
+
_deleteSecret(account)
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
/**
|
|
342
|
+
* Whether a secret is stored in the OS keychain for `account` (ignores env).
|
|
343
|
+
* Lets the setup UI show "Set" WITHOUT ever sending the secret value to the
|
|
344
|
+
* browser.
|
|
345
|
+
*/
|
|
346
|
+
export function hasStoredSecret(account) {
|
|
347
|
+
try { return !!_getSecret(account) } catch { return false }
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
export { CONFIG_PATH }
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
// Daemon recycle barrier — closes the dev-sync full-restart "stale-serving"
|
|
2
|
+
// race.
|
|
3
|
+
//
|
|
4
|
+
// When dev-sync runs in-band (invoked from inside the live daemon's bash
|
|
5
|
+
// tool) it cannot kill the shared daemon synchronously — that would sever
|
|
6
|
+
// its own process before the response flushes. It instead SCHEDULES a
|
|
7
|
+
// detached kill a couple seconds out and returns immediately. During that
|
|
8
|
+
// kill-delay plus the supervisor's respawn time the OLD daemon stays alive
|
|
9
|
+
// and keeps accepting work, so a bridge worker spawned in that window binds
|
|
10
|
+
// to about-to-die, stale-code/schema daemon (observed: a worker received a
|
|
11
|
+
// pre-edit tool schema right after a forced restart).
|
|
12
|
+
//
|
|
13
|
+
// Fix: at kill-schedule time dev-sync writes a sentinel naming the EXACT
|
|
14
|
+
// daemon being recycled by its (server_pid, server_started_at) identity.
|
|
15
|
+
// The bridge spawn-admission path refuses to start a worker while the live
|
|
16
|
+
// advert still names that same daemon, so the dispatch is deferred to the
|
|
17
|
+
// fresh daemon instead of served stale. This is identity-based, not a timer:
|
|
18
|
+
// the gate compares the sentinel against the current advert, and the moment
|
|
19
|
+
// the fresh daemon rewrites the advert with a new server_pid the match
|
|
20
|
+
// breaks and spawns resume — no clock, no heuristic window. server_started_at
|
|
21
|
+
// guards the rare case of an OS PID reuse aliasing the fresh daemon onto the
|
|
22
|
+
// doomed identity.
|
|
23
|
+
//
|
|
24
|
+
// Neutral location (src/shared) so both the agent orchestrator (spawn gate)
|
|
25
|
+
// and the dev-sync script (sentinel writer) import the SAME path + schema —
|
|
26
|
+
// the agent layer never imports the channels layer where the rest of the
|
|
27
|
+
// runtime-state lives.
|
|
28
|
+
|
|
29
|
+
import { tmpdir } from 'node:os';
|
|
30
|
+
import { join, resolve } from 'node:path';
|
|
31
|
+
import { existsSync, mkdirSync, readFileSync, renameSync, rmSync, writeFileSync } from 'node:fs';
|
|
32
|
+
|
|
33
|
+
// Mirror runtime-paths.mjs RUNTIME_ROOT resolution so the sentinel lands
|
|
34
|
+
// alongside active-instance.json regardless of MIXDOG_RUNTIME_ROOT override.
|
|
35
|
+
const RUNTIME_ROOT = process.env.MIXDOG_RUNTIME_ROOT
|
|
36
|
+
? resolve(process.env.MIXDOG_RUNTIME_ROOT)
|
|
37
|
+
: join(tmpdir(), 'mixdog');
|
|
38
|
+
const RECYCLE_FILE = join(RUNTIME_ROOT, 'daemon-recycle.json');
|
|
39
|
+
const ACTIVE_INSTANCE_FILE = join(RUNTIME_ROOT, 'active-instance.json');
|
|
40
|
+
|
|
41
|
+
function _readJson(path) {
|
|
42
|
+
try { return JSON.parse(readFileSync(path, 'utf8')); }
|
|
43
|
+
catch { return null; }
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
function _pid(value) {
|
|
47
|
+
const n = Number(value);
|
|
48
|
+
return Number.isFinite(n) && n > 0 ? n : null;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function readDaemonRecycleSentinel() {
|
|
52
|
+
return _readJson(RECYCLE_FILE);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// Called by dev-sync when it schedules a forced daemon (server_pid) kill.
|
|
56
|
+
// The full (server_pid, server_started_at) identity is REQUIRED: a sentinel
|
|
57
|
+
// without a verifiable start time could falsely block a PID-reused fresh
|
|
58
|
+
// daemon, so we decline to arm the barrier rather than arm it partially
|
|
59
|
+
// (degrades to prior behavior — no heuristic PID-only match).
|
|
60
|
+
function writeDaemonRecycleSentinel({ doomedServerPid, doomedServerStartedAt } = {}) {
|
|
61
|
+
const pid = _pid(doomedServerPid);
|
|
62
|
+
const startedAt = Number(doomedServerStartedAt);
|
|
63
|
+
if (pid === null || !Number.isFinite(startedAt)) return false;
|
|
64
|
+
try {
|
|
65
|
+
if (!existsSync(RUNTIME_ROOT)) mkdirSync(RUNTIME_ROOT, { recursive: true });
|
|
66
|
+
// Atomic publish: write a temp file then rename, so a concurrent
|
|
67
|
+
// isCurrentDaemonDoomed() never observes a partial/empty sentinel and
|
|
68
|
+
// falsely allows a spawn on the doomed daemon. Mirrors active-instance.json.
|
|
69
|
+
const tmp = `${RECYCLE_FILE}.${process.pid}.tmp`;
|
|
70
|
+
writeFileSync(tmp, JSON.stringify({ doomedServerPid: pid, doomedServerStartedAt: startedAt }));
|
|
71
|
+
renameSync(tmp, RECYCLE_FILE);
|
|
72
|
+
return true;
|
|
73
|
+
} catch { return false; }
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
function clearDaemonRecycleSentinel() {
|
|
77
|
+
try { if (existsSync(RECYCLE_FILE)) rmSync(RECYCLE_FILE, { force: true }); }
|
|
78
|
+
catch { /* best-effort */ }
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// True when the live advert (this daemon's own active-instance.json) names
|
|
82
|
+
// the daemon dev-sync flagged for recycle. PID identity is primary; when the
|
|
83
|
+
// sentinel recorded server_started_at it must also match the advert so a
|
|
84
|
+
// reused PID cannot alias a freshly-booted daemon onto the doomed identity.
|
|
85
|
+
// Reads nothing when no sentinel exists (the common case) — one stat/read.
|
|
86
|
+
function isCurrentDaemonDoomed() {
|
|
87
|
+
const recycle = readDaemonRecycleSentinel();
|
|
88
|
+
if (!recycle) return false;
|
|
89
|
+
const advert = _readJson(ACTIVE_INSTANCE_FILE);
|
|
90
|
+
if (!advert) return false;
|
|
91
|
+
const doomedPid = _pid(recycle.doomedServerPid);
|
|
92
|
+
const curPid = _pid(advert.server_pid);
|
|
93
|
+
if (doomedPid === null || curPid === null || doomedPid !== curPid) return false;
|
|
94
|
+
// Strict (server_pid, server_started_at) identity — no PID-only fallback,
|
|
95
|
+
// so a reused PID on a fresh daemon (different start time) never matches.
|
|
96
|
+
const doomedStart = Number(recycle.doomedServerStartedAt);
|
|
97
|
+
const curStart = Number(advert.server_started_at);
|
|
98
|
+
return Number.isFinite(doomedStart) && Number.isFinite(curStart) && doomedStart === curStart;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
export {
|
|
102
|
+
RECYCLE_FILE,
|
|
103
|
+
ACTIVE_INSTANCE_FILE,
|
|
104
|
+
readDaemonRecycleSentinel,
|
|
105
|
+
writeDaemonRecycleSentinel,
|
|
106
|
+
clearDaemonRecycleSentinel,
|
|
107
|
+
isCurrentDaemonDoomed,
|
|
108
|
+
};
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import { existsSync, mkdirSync, readFileSync } from 'fs';
|
|
2
|
+
import { dirname, join } from 'path';
|
|
3
|
+
import { homedir } from 'os';
|
|
4
|
+
import { writeFileAtomicSync } from './atomic-file.mjs';
|
|
5
|
+
import { getBackupRoot } from './user-data-guard.mjs';
|
|
6
|
+
|
|
7
|
+
// On FIRST install only (gated by the caller via the fresh creation of
|
|
8
|
+
// mixdog-config.json), disable Claude Code's built-in auto-memory and away
|
|
9
|
+
// summary so mixdog's own memory/recap is authoritative. The user's prior
|
|
10
|
+
// values are captured to an install-restore snapshot first so the original
|
|
11
|
+
// behaviour can be restored. The createOnly gate in seed.mjs guarantees this
|
|
12
|
+
// runs exactly once, so we never reapply on later boots.
|
|
13
|
+
|
|
14
|
+
// Settings path is a parameter (defaulting to ~/.claude/settings.json) so it
|
|
15
|
+
// can be redirected via MIXDOG_CLAUDE_SETTINGS_PATH for testing without a real
|
|
16
|
+
// homedir.
|
|
17
|
+
export function resolveClaudeSettingsPath() {
|
|
18
|
+
return process.env.MIXDOG_CLAUDE_SETTINGS_PATH
|
|
19
|
+
|| join(homedir(), '.claude', 'settings.json');
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function readJsonOrNull(filePath) {
|
|
23
|
+
try {
|
|
24
|
+
return JSON.parse(readFileSync(filePath, 'utf8'));
|
|
25
|
+
} catch {
|
|
26
|
+
return null;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
function isPlainObject(value) {
|
|
31
|
+
return !!value && typeof value === 'object' && !Array.isArray(value);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Disable Claude Code's built-in memory/recap on first install. Best-effort:
|
|
36
|
+
* any failure is warned to stderr and swallowed so seeding is never broken.
|
|
37
|
+
*
|
|
38
|
+
* @param {string} settingsPath path to settings.json (default ~/.claude/settings.json)
|
|
39
|
+
* @returns {{ snapshot: boolean, merged: boolean }}
|
|
40
|
+
*/
|
|
41
|
+
export function disableClaudeBuiltinsOnFirstInstall(settingsPath = resolveClaudeSettingsPath()) {
|
|
42
|
+
const result = { snapshot: false, merged: false };
|
|
43
|
+
try {
|
|
44
|
+
let settings = {};
|
|
45
|
+
if (existsSync(settingsPath)) {
|
|
46
|
+
// File exists: parse it. An unparseable existing file may hold raw,
|
|
47
|
+
// hand-fixable user content — never overwrite it. Warn and skip the
|
|
48
|
+
// whole merge (no reliable priors → no snapshot either). Only a truly
|
|
49
|
+
// missing file is treated as `{}` and gets the disable flags written.
|
|
50
|
+
const parsed = readJsonOrNull(settingsPath);
|
|
51
|
+
if (parsed === null) {
|
|
52
|
+
process.stderr.write(`[seed] claude settings.json at ${settingsPath} is unparseable; leaving untouched\n`);
|
|
53
|
+
return result;
|
|
54
|
+
}
|
|
55
|
+
// Valid JSON with a non-object top level (array/string/number) also
|
|
56
|
+
// holds user content we cannot merge into — skip, same as unparseable.
|
|
57
|
+
if (!isPlainObject(parsed)) {
|
|
58
|
+
process.stderr.write(`[seed] claude settings.json at ${settingsPath} is not a JSON object; leaving untouched\n`);
|
|
59
|
+
return result;
|
|
60
|
+
}
|
|
61
|
+
settings = parsed;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
// 1. Restore snapshot FIRST, capturing prior values (null when absent).
|
|
65
|
+
const snapshotPath = join(getBackupRoot(), 'install-restore', 'claude-settings-original.json');
|
|
66
|
+
if (!existsSync(snapshotPath)) {
|
|
67
|
+
const snapshot = {
|
|
68
|
+
autoMemoryEnabled: Object.prototype.hasOwnProperty.call(settings, 'autoMemoryEnabled')
|
|
69
|
+
? settings.autoMemoryEnabled : null,
|
|
70
|
+
awaySummaryEnabled: Object.prototype.hasOwnProperty.call(settings, 'awaySummaryEnabled')
|
|
71
|
+
? settings.awaySummaryEnabled : null,
|
|
72
|
+
capturedAt: new Date().toISOString(),
|
|
73
|
+
};
|
|
74
|
+
mkdirSync(dirname(snapshotPath), { recursive: true });
|
|
75
|
+
writeFileAtomicSync(snapshotPath, JSON.stringify(snapshot, null, 2) + '\n', { fsyncDir: true });
|
|
76
|
+
result.snapshot = true;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// 2. Merge the disables, preserving all other keys, atomic write.
|
|
80
|
+
const merged = { ...settings, autoMemoryEnabled: false, awaySummaryEnabled: false };
|
|
81
|
+
mkdirSync(dirname(settingsPath), { recursive: true });
|
|
82
|
+
writeFileAtomicSync(settingsPath, JSON.stringify(merged, null, 2) + '\n', { fsyncDir: true });
|
|
83
|
+
result.merged = true;
|
|
84
|
+
} catch (e) {
|
|
85
|
+
process.stderr.write(`[seed] disable claude built-ins failed: ${e.message}\n`);
|
|
86
|
+
}
|
|
87
|
+
return result;
|
|
88
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
export function errText(e) {
|
|
2
|
+
if (e == null) return String(e);
|
|
3
|
+
if (typeof e === 'string') return e;
|
|
4
|
+
if (e instanceof Error) return e.message || e.name || String(e);
|
|
5
|
+
// ErrorEvent / event-like / plain object (NOT instanceof Error)
|
|
6
|
+
if (typeof e.message === 'string' && e.message) return e.message;
|
|
7
|
+
if (e.error != null && e.error !== e) return errText(e.error);
|
|
8
|
+
if (e.reason != null && e.reason !== e) return errText(e.reason);
|
|
9
|
+
if (typeof e.type === 'string' && e.type) return `${e.type} event`;
|
|
10
|
+
try { const j = JSON.stringify(e); if (j && j !== '{}' && j !== 'null') return j; } catch {}
|
|
11
|
+
return String(e);
|
|
12
|
+
}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Per-call cost estimator for bridge-trace usage rows.
|
|
3
|
+
*
|
|
4
|
+
* Pricing is pulled from the LiteLLM catalog (already warmed by providers/
|
|
5
|
+
* agent bootstrap). All four token slots — input / output / cacheRead /
|
|
6
|
+
* cacheWrite — are multiplied by their matching $/M rate from the catalog
|
|
7
|
+
* and summed. Missing rates are treated as 0 (no extrapolation).
|
|
8
|
+
*
|
|
9
|
+
* The catalog is looked up synchronously: if it has not been warmed yet
|
|
10
|
+
* (fresh process, first call), this returns 0 without blocking. The next
|
|
11
|
+
* call will pick up the cache.
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import { getModelMetadataSync } from '../../agent/orchestrator/providers/model-catalog.mjs';
|
|
15
|
+
|
|
16
|
+
// OpenAI / Codex / Gemini report `input_tokens` as the total prompt token
|
|
17
|
+
// count *including* the cached portion (inclusive). Anthropic reports the
|
|
18
|
+
// uncached remainder only and bills cached_read / cached_write as separate
|
|
19
|
+
// additive slots (additive). Cost and prompt-total math has to branch on this.
|
|
20
|
+
// OpenAI-compatible direct providers (deepseek / nvidia / ollama / lmstudio)
|
|
21
|
+
// go through the OpenAI SDK and likewise report an inclusive prompt_tokens
|
|
22
|
+
// with a separate cached-tokens detail — so they are inclusive too. Omitting
|
|
23
|
+
// them bills the cached portion at the full input rate AND re-adds it as a
|
|
24
|
+
// cacheRead slot, double-billing the cache (e.g. a ~10k-token cached system
|
|
25
|
+
// prompt charged ~25x its real cost on every DeepSeek call).
|
|
26
|
+
export function isInclusiveProvider(provider) {
|
|
27
|
+
if (!provider) return false;
|
|
28
|
+
const p = String(provider).toLowerCase();
|
|
29
|
+
// 'grok' covers grok-oauth, which delegates inference to the xai compat
|
|
30
|
+
// provider (inclusive input_tokens) but reports its own provider id on
|
|
31
|
+
// usage rows — without it, cached tokens would be double-billed in the
|
|
32
|
+
// cost fallback and prompt totals.
|
|
33
|
+
return p.includes('openai') || p.includes('codex') || p.includes('gemini') || p.includes('google') || p.includes('xai') || p.includes('grok')
|
|
34
|
+
|| p.includes('deepseek') || p.includes('nvidia') || p.includes('ollama') || p.includes('lmstudio') || p.includes('groq') || p.includes('openrouter');
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* @param {object} args
|
|
39
|
+
* @param {string} args.model
|
|
40
|
+
* @param {string} [args.provider]
|
|
41
|
+
* @param {number} [args.inputTokens]
|
|
42
|
+
* @param {number} [args.outputTokens]
|
|
43
|
+
* @param {number} [args.cacheReadTokens]
|
|
44
|
+
* @param {number} [args.cacheWriteTokens]
|
|
45
|
+
* @returns {number} USD, rounded to 6 decimal places.
|
|
46
|
+
*/
|
|
47
|
+
export function computeCostUsd(args) {
|
|
48
|
+
const meta = getModelMetadataSync(args?.model);
|
|
49
|
+
if (!meta) return 0;
|
|
50
|
+
const inputTokens = args.inputTokens || 0;
|
|
51
|
+
const outputTokens = args.outputTokens || 0;
|
|
52
|
+
const cacheReadTokens = args.cacheReadTokens || 0;
|
|
53
|
+
const cacheWriteTokens = args.cacheWriteTokens || 0;
|
|
54
|
+
const billableInput = isInclusiveProvider(args.provider)
|
|
55
|
+
? Math.max(inputTokens - cacheReadTokens - cacheWriteTokens, 0)
|
|
56
|
+
: inputTokens;
|
|
57
|
+
const parts = [
|
|
58
|
+
billableInput * (meta.inputCostPerM || 0),
|
|
59
|
+
outputTokens * (meta.outputCostPerM || 0),
|
|
60
|
+
cacheReadTokens * (meta.cacheReadCostPerM || 0),
|
|
61
|
+
cacheWriteTokens * (meta.cacheWriteCostPerM || 0),
|
|
62
|
+
];
|
|
63
|
+
const total = parts.reduce((s, x) => s + x, 0) / 1_000_000;
|
|
64
|
+
if (!Number.isFinite(total) || total <= 0) return 0;
|
|
65
|
+
return Number(total.toFixed(6));
|
|
66
|
+
}
|