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,348 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Smart Bridge — Cache Registry (Phase D-2: provider × profile matrix)
|
|
3
|
+
*
|
|
4
|
+
* Each provider caches independently (Anthropic workspace+model shard,
|
|
5
|
+
* OpenAI prompt_cache_key, Gemini implicit cache observations). Tracking a
|
|
6
|
+
* single entry per profile-id — as v1 did — means a profile used across two
|
|
7
|
+
* providers silently overwrites the warm state of the other. v2 indexes by
|
|
8
|
+
* (profileId, provider), preserving per-shard hit/miss/TTL independently.
|
|
9
|
+
*
|
|
10
|
+
* Persistence: <plugin-data>/cache-registry.json. v1 files auto-migrate
|
|
11
|
+
* on load.
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import { existsSync, readFileSync, mkdirSync } from 'fs';
|
|
15
|
+
import { createHash } from 'crypto';
|
|
16
|
+
import { dirname, join } from 'path';
|
|
17
|
+
import { getPluginData } from '../config.mjs';
|
|
18
|
+
import { writeJsonAtomicSync } from '../../../shared/atomic-file.mjs';
|
|
19
|
+
|
|
20
|
+
const REGISTRY_VERSION = 2;
|
|
21
|
+
|
|
22
|
+
function registryPath() {
|
|
23
|
+
return join(getPluginData(), 'cache-registry.json');
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function emptyRegistry() {
|
|
27
|
+
return {
|
|
28
|
+
version: REGISTRY_VERSION,
|
|
29
|
+
// profiles: profileId → provider → entry
|
|
30
|
+
// entry = { prefixHash, createdAt, expiresAt, hitCount, missCount, observedOnly }
|
|
31
|
+
profiles: {},
|
|
32
|
+
openaiKeys: {}, // cacheKey → { retention, lastUsedAt }
|
|
33
|
+
updatedAt: new Date().toISOString(),
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
// shape: profiles[profileId][provider] = { prefixHash, createdAt, ... }
|
|
38
|
+
function migrateV1ToV2(raw) {
|
|
39
|
+
const migrated = emptyRegistry();
|
|
40
|
+
migrated.openaiKeys = raw.openaiKeys || {};
|
|
41
|
+
for (const [profileId, entry] of Object.entries(raw.profiles || {})) {
|
|
42
|
+
if (!entry || typeof entry !== 'object') continue;
|
|
43
|
+
const provider = entry.provider || 'unknown';
|
|
44
|
+
const { provider: _p, ...rest } = entry;
|
|
45
|
+
migrated.profiles[profileId] = { [provider]: rest };
|
|
46
|
+
}
|
|
47
|
+
return migrated;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export class CacheRegistry {
|
|
51
|
+
constructor() {
|
|
52
|
+
this.data = emptyRegistry();
|
|
53
|
+
this.loaded = false;
|
|
54
|
+
this.dirty = false;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
load() {
|
|
58
|
+
const path = registryPath();
|
|
59
|
+
if (!existsSync(path)) {
|
|
60
|
+
this.data = emptyRegistry();
|
|
61
|
+
this.loaded = true;
|
|
62
|
+
return;
|
|
63
|
+
}
|
|
64
|
+
try {
|
|
65
|
+
const raw = JSON.parse(readFileSync(path, 'utf8'));
|
|
66
|
+
if (raw.version === REGISTRY_VERSION) {
|
|
67
|
+
this.data = {
|
|
68
|
+
version: REGISTRY_VERSION,
|
|
69
|
+
profiles: raw.profiles || {},
|
|
70
|
+
openaiKeys: raw.openaiKeys || {},
|
|
71
|
+
updatedAt: raw.updatedAt || new Date().toISOString(),
|
|
72
|
+
};
|
|
73
|
+
} else if (raw.version === 1 || raw.version === undefined) {
|
|
74
|
+
process.stderr.write(`[cache-registry] migrating v${raw.version || '?'} → v${REGISTRY_VERSION}\n`);
|
|
75
|
+
this.data = migrateV1ToV2(raw);
|
|
76
|
+
this.dirty = true;
|
|
77
|
+
} else {
|
|
78
|
+
process.stderr.write(`[cache-registry] unknown version ${raw.version}, resetting\n`);
|
|
79
|
+
this.data = emptyRegistry();
|
|
80
|
+
}
|
|
81
|
+
} catch (err) {
|
|
82
|
+
process.stderr.write(`[cache-registry] load failed: ${err.message}\n`);
|
|
83
|
+
this.data = emptyRegistry();
|
|
84
|
+
}
|
|
85
|
+
this.cleanupExpired();
|
|
86
|
+
this.loaded = true;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
save() {
|
|
90
|
+
if (!this.loaded) return;
|
|
91
|
+
const path = registryPath();
|
|
92
|
+
try {
|
|
93
|
+
mkdirSync(dirname(path), { recursive: true });
|
|
94
|
+
this.data.updatedAt = new Date().toISOString();
|
|
95
|
+
writeJsonAtomicSync(path, this.data, { lock: true });
|
|
96
|
+
this.dirty = false;
|
|
97
|
+
} catch (err) {
|
|
98
|
+
process.stderr.write(`[cache-registry] save failed: ${err.message}\n`);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
_getEntry(profileId, provider) {
|
|
103
|
+
return this.data.profiles[profileId]?.[provider] || null;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
_setEntry(profileId, provider, entry) {
|
|
107
|
+
if (!this.data.profiles[profileId]) this.data.profiles[profileId] = {};
|
|
108
|
+
this.data.profiles[profileId][provider] = entry;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// --- Profile × provider cache warm state ---
|
|
112
|
+
|
|
113
|
+
/**
|
|
114
|
+
* Record that a (profile, provider) shard was just written to provider
|
|
115
|
+
* cache. prefixContent should be deterministic (system + tools + context
|
|
116
|
+
* chunks) so re-hashing produces the same hash on next run. TTL seconds
|
|
117
|
+
* is provider-specific and decided by the caller.
|
|
118
|
+
*/
|
|
119
|
+
markWarm(profileId, provider, prefixContent, ttlSeconds) {
|
|
120
|
+
if (!this.loaded) this.load();
|
|
121
|
+
if (!profileId || !provider) return;
|
|
122
|
+
const now = Date.now();
|
|
123
|
+
const prefixHash = hashContent(prefixContent);
|
|
124
|
+
const existing = this._getEntry(profileId, provider);
|
|
125
|
+
const samePrefix = existing && existing.prefixHash === prefixHash;
|
|
126
|
+
this._setEntry(profileId, provider, {
|
|
127
|
+
prefixHash,
|
|
128
|
+
createdAt: samePrefix ? (existing.createdAt || now) : now,
|
|
129
|
+
expiresAt: now + ttlSeconds * 1000,
|
|
130
|
+
hitCount: samePrefix ? (existing.hitCount || 0) : 0,
|
|
131
|
+
missCount: existing?.missCount || 0,
|
|
132
|
+
observedOnly: false,
|
|
133
|
+
});
|
|
134
|
+
this.dirty = true;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
/**
|
|
138
|
+
* Record that a provider returned cache metrics for this prefix without
|
|
139
|
+
* implying the next request can rely on a warm shard. Gemini implicit
|
|
140
|
+
* caching is the main case: hits are useful telemetry, but no explicit
|
|
141
|
+
* cache object or stable warm contract exists.
|
|
142
|
+
*/
|
|
143
|
+
markObserved(profileId, provider, prefixContent, ttlSeconds) {
|
|
144
|
+
if (!this.loaded) this.load();
|
|
145
|
+
if (!profileId || !provider) return;
|
|
146
|
+
const now = Date.now();
|
|
147
|
+
const prefixHash = hashContent(prefixContent);
|
|
148
|
+
const existing = this._getEntry(profileId, provider);
|
|
149
|
+
const samePrefix = existing && existing.prefixHash === prefixHash;
|
|
150
|
+
this._setEntry(profileId, provider, {
|
|
151
|
+
prefixHash,
|
|
152
|
+
createdAt: samePrefix ? (existing.createdAt || now) : now,
|
|
153
|
+
expiresAt: now + ttlSeconds * 1000,
|
|
154
|
+
hitCount: samePrefix ? (existing.hitCount || 0) : 0,
|
|
155
|
+
missCount: samePrefix ? (existing.missCount || 0) : 0,
|
|
156
|
+
observedOnly: true,
|
|
157
|
+
});
|
|
158
|
+
this.dirty = true;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* Check whether a (profile, provider) shard is still warm for the given
|
|
163
|
+
* prefix content. warm=true is only returned for providers we deliberately
|
|
164
|
+
* mark as reusable. Observation-only providers may have hit/miss stats,
|
|
165
|
+
* but still return warm=false because the provider has no explicit warm
|
|
166
|
+
* contract.
|
|
167
|
+
*/
|
|
168
|
+
checkWarm(profileId, provider, prefixContent) {
|
|
169
|
+
if (!this.loaded) this.load();
|
|
170
|
+
const entry = this._getEntry(profileId, provider);
|
|
171
|
+
if (!entry) return { warm: false, expiresIn: 0, reason: 'no-entry' };
|
|
172
|
+
const now = Date.now();
|
|
173
|
+
if ((entry.expiresAt || 0) < now) return { warm: false, expiresIn: 0, reason: 'expired' };
|
|
174
|
+
if (entry.observedOnly) {
|
|
175
|
+
return { warm: false, expiresIn: entry.expiresAt - now, reason: 'observed-only' };
|
|
176
|
+
}
|
|
177
|
+
const currentHash = hashContent(prefixContent);
|
|
178
|
+
if (currentHash !== entry.prefixHash) {
|
|
179
|
+
return { warm: false, expiresIn: 0, reason: 'hash-mismatch' };
|
|
180
|
+
}
|
|
181
|
+
return { warm: true, expiresIn: entry.expiresAt - now, reason: 'warm' };
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
recordHit(profileId, provider) {
|
|
185
|
+
if (!this.loaded) this.load();
|
|
186
|
+
const entry = this._getEntry(profileId, provider);
|
|
187
|
+
if (entry) {
|
|
188
|
+
entry.hitCount = (entry.hitCount || 0) + 1;
|
|
189
|
+
this.dirty = true;
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
recordMiss(profileId, provider) {
|
|
194
|
+
if (!this.loaded) this.load();
|
|
195
|
+
const entry = this._getEntry(profileId, provider);
|
|
196
|
+
if (entry) {
|
|
197
|
+
entry.missCount = (entry.missCount || 0) + 1;
|
|
198
|
+
this.dirty = true;
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
/**
|
|
203
|
+
* Invalidate a single provider shard when `provider` is given, or every
|
|
204
|
+
* provider shard under that profile when omitted.
|
|
205
|
+
*/
|
|
206
|
+
invalidate(profileId, provider) {
|
|
207
|
+
if (!profileId || !this.data.profiles[profileId]) return;
|
|
208
|
+
if (provider) {
|
|
209
|
+
if (this.data.profiles[profileId][provider]) {
|
|
210
|
+
delete this.data.profiles[profileId][provider];
|
|
211
|
+
if (Object.keys(this.data.profiles[profileId]).length === 0) {
|
|
212
|
+
delete this.data.profiles[profileId];
|
|
213
|
+
}
|
|
214
|
+
this.dirty = true;
|
|
215
|
+
}
|
|
216
|
+
} else {
|
|
217
|
+
delete this.data.profiles[profileId];
|
|
218
|
+
this.dirty = true;
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
// --- OpenAI cache_key tracking ---
|
|
223
|
+
|
|
224
|
+
trackOpenAIKey(cacheKey, retention) {
|
|
225
|
+
if (!this.loaded) this.load();
|
|
226
|
+
this.data.openaiKeys[cacheKey] = {
|
|
227
|
+
retention,
|
|
228
|
+
lastUsedAt: new Date().toISOString(),
|
|
229
|
+
};
|
|
230
|
+
this.dirty = true;
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
// --- Stats & maintenance ---
|
|
234
|
+
|
|
235
|
+
cleanupExpired() {
|
|
236
|
+
if (!this.loaded) return;
|
|
237
|
+
const now = Date.now();
|
|
238
|
+
let removed = 0;
|
|
239
|
+
for (const [profileId, providers] of Object.entries(this.data.profiles)) {
|
|
240
|
+
for (const [provider, entry] of Object.entries(providers)) {
|
|
241
|
+
if ((entry?.expiresAt || 0) < now) {
|
|
242
|
+
delete providers[provider];
|
|
243
|
+
removed += 1;
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
if (Object.keys(providers).length === 0) {
|
|
247
|
+
delete this.data.profiles[profileId];
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
if (removed > 0) this.dirty = true;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
/**
|
|
254
|
+
* Matrix-shaped stats. profiles[profileId][provider] = { hitCount,
|
|
255
|
+
* missCount, hitRate, expiresIn, prefixHash }. shardCount totals every
|
|
256
|
+
* (profile, provider) pair; profileCount counts distinct profiles.
|
|
257
|
+
*/
|
|
258
|
+
getStats() {
|
|
259
|
+
if (!this.loaded) this.load();
|
|
260
|
+
const now = Date.now();
|
|
261
|
+
const profiles = {};
|
|
262
|
+
let shardCount = 0;
|
|
263
|
+
for (const [profileId, providers] of Object.entries(this.data.profiles)) {
|
|
264
|
+
profiles[profileId] = {};
|
|
265
|
+
for (const [provider, entry] of Object.entries(providers)) {
|
|
266
|
+
const total = (entry.hitCount || 0) + (entry.missCount || 0);
|
|
267
|
+
profiles[profileId][provider] = {
|
|
268
|
+
prefixHash: entry.prefixHash || null,
|
|
269
|
+
hitCount: entry.hitCount || 0,
|
|
270
|
+
missCount: entry.missCount || 0,
|
|
271
|
+
hitRate: total > 0 ? (entry.hitCount || 0) / total : 0,
|
|
272
|
+
expiresIn: Math.max(0, (entry.expiresAt || 0) - now),
|
|
273
|
+
createdAt: entry.createdAt || null,
|
|
274
|
+
observedOnly: entry.observedOnly === true,
|
|
275
|
+
};
|
|
276
|
+
shardCount += 1;
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
return {
|
|
280
|
+
profileCount: Object.keys(this.data.profiles).length,
|
|
281
|
+
shardCount,
|
|
282
|
+
profiles,
|
|
283
|
+
openaiKeyCount: Object.keys(this.data.openaiKeys).length,
|
|
284
|
+
};
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
// --- Singleton-style access (callers import a shared instance) ---
|
|
288
|
+
|
|
289
|
+
static _shared = null;
|
|
290
|
+
static shared() {
|
|
291
|
+
if (!this._shared) {
|
|
292
|
+
this._shared = new CacheRegistry();
|
|
293
|
+
this._shared.load();
|
|
294
|
+
}
|
|
295
|
+
return this._shared;
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
// --- Helpers ---
|
|
300
|
+
|
|
301
|
+
/**
|
|
302
|
+
* Deep stable stringifier — deterministic key order at every nesting level.
|
|
303
|
+
* Arrays preserve positional order (tool order matters); objects sort keys
|
|
304
|
+
* alphabetically. Primitives and null are serialized via JSON.stringify.
|
|
305
|
+
*
|
|
306
|
+
* Replaces the previous replacer-array form, which acted as a top-level
|
|
307
|
+
* allowlist and silently emptied nested tool objects (name/description/
|
|
308
|
+
* inputSchema were dropped → falsely stable hash across schema changes).
|
|
309
|
+
*/
|
|
310
|
+
export function stableStringify(value) {
|
|
311
|
+
if (value === null || typeof value !== 'object') {
|
|
312
|
+
return JSON.stringify(value);
|
|
313
|
+
}
|
|
314
|
+
if (Array.isArray(value)) {
|
|
315
|
+
return '[' + value.map(v => stableStringify(v)).join(',') + ']';
|
|
316
|
+
}
|
|
317
|
+
const keys = Object.keys(value).sort();
|
|
318
|
+
const parts = [];
|
|
319
|
+
for (const k of keys) {
|
|
320
|
+
const v = value[k];
|
|
321
|
+
if (v === undefined) continue;
|
|
322
|
+
parts.push(JSON.stringify(k) + ':' + stableStringify(v));
|
|
323
|
+
}
|
|
324
|
+
return '{' + parts.join(',') + '}';
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
/**
|
|
328
|
+
* Stable content hash — deterministic across runs as long as structure is identical.
|
|
329
|
+
* Used to detect when a profile's prefix content has drifted.
|
|
330
|
+
*/
|
|
331
|
+
export function hashContent(content) {
|
|
332
|
+
const canonical = typeof content === 'string'
|
|
333
|
+
? content
|
|
334
|
+
: stableStringify(content);
|
|
335
|
+
return createHash('sha256').update(canonical).digest('hex').slice(0, 16);
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
/**
|
|
339
|
+
* Default TTL (seconds) per cache layer. Providers pick their own; callers
|
|
340
|
+
* that don't specify land on the shortest safe default.
|
|
341
|
+
*/
|
|
342
|
+
export const DEFAULT_TTL_SECONDS = {
|
|
343
|
+
'1h': 3600,
|
|
344
|
+
'5m': 300,
|
|
345
|
+
'24h': 86400,
|
|
346
|
+
in_memory: 600, // optimistic estimate
|
|
347
|
+
none: 0,
|
|
348
|
+
};
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Smart Bridge — shared session builder.
|
|
3
|
+
*
|
|
4
|
+
* Single source of truth for bridge session creation + role/preset
|
|
5
|
+
* telemetry. Both entry points route through this helper:
|
|
6
|
+
*
|
|
7
|
+
* - `smart-bridge/bridge-llm.mjs` — internal callers
|
|
8
|
+
* (memory-cycle, scheduler, webhook) dispatching via
|
|
9
|
+
* `makeBridgeLlm`.
|
|
10
|
+
* - `src/agent/index.mjs` `case 'bridge'` — Lead-originated MCP
|
|
11
|
+
* bridge dispatches into user-workflow roles.
|
|
12
|
+
*
|
|
13
|
+
* Before this helper, the two paths carried separate `createSession` +
|
|
14
|
+
* `traceBridgePreset` blocks. Lead-direct dispatches silently skipped
|
|
15
|
+
* the trace so cache-hit analysis missed every user-workflow role call.
|
|
16
|
+
*
|
|
17
|
+
* Preset resolution stays with each caller since they read from
|
|
18
|
+
* different sources (MCP: user-workflow.json only; Smart Bridge:
|
|
19
|
+
* hidden-role map first, then user-workflow.json). The helper takes
|
|
20
|
+
* already-resolved primitives.
|
|
21
|
+
*/
|
|
22
|
+
|
|
23
|
+
import { createSession } from '../session/manager.mjs';
|
|
24
|
+
import { traceBridgePreset } from '../bridge-trace.mjs';
|
|
25
|
+
import { resolveBridgeSessionPermission } from '../internal-roles.mjs';
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* @param {object} opts
|
|
29
|
+
* @param {string} opts.role — canonical role name ('worker', 'explorer', ...)
|
|
30
|
+
* @param {string} opts.presetName — resolved preset identifier
|
|
31
|
+
* @param {object} opts.preset — resolved preset object from agent-config
|
|
32
|
+
* @param {object} opts.runtimeSpec — resolveRuntimeSpec output; must carry .scopeKey / .lane
|
|
33
|
+
* @param {string} [opts.permission] — 'read' | 'read-write' | null (preset/full default when unset)
|
|
34
|
+
* @param {string|null} [opts.cwd] — absolute working dir; null is the fixed bridge sentinel meaning "no caller workspace context"
|
|
35
|
+
* @param {string} [opts.owner='bridge']
|
|
36
|
+
* @param {string} [opts.permissionMode] — Claude Code permissionMode forwarded from the MCP payload ('bypassPermissions', 'acceptEdits', 'plan', 'dontAsk', 'default')
|
|
37
|
+
* @param {string[]} [opts.schemaAllowedTools] — schema-level allowlist from a hidden-role toolSchemaProfile
|
|
38
|
+
* @param {string} [opts.sourceType]
|
|
39
|
+
* @param {string} [opts.sourceName]
|
|
40
|
+
* @param {string} [opts.taskType]
|
|
41
|
+
* @param {number} [opts.maxLoopIterations]
|
|
42
|
+
* @param {string} [opts.parentSessionId]
|
|
43
|
+
* @param {string|null} [opts.ownerSessionId] - owning Mixdog MCP instance id for statusline isolation
|
|
44
|
+
* @param {boolean} [opts.skipRoleReminder=false] — Pool C suppresses Tier 3 reminder
|
|
45
|
+
* @returns {{ session: object, effectiveCwd: string|null }}
|
|
46
|
+
*/
|
|
47
|
+
export function prepareBridgeSession({
|
|
48
|
+
role,
|
|
49
|
+
presetName,
|
|
50
|
+
preset,
|
|
51
|
+
runtimeSpec,
|
|
52
|
+
permission,
|
|
53
|
+
cwd,
|
|
54
|
+
owner = 'bridge',
|
|
55
|
+
permissionMode,
|
|
56
|
+
sourceType,
|
|
57
|
+
sourceName,
|
|
58
|
+
taskType,
|
|
59
|
+
maxLoopIterations,
|
|
60
|
+
parentSessionId,
|
|
61
|
+
ownerSessionId,
|
|
62
|
+
clientHostPid,
|
|
63
|
+
bridgeTag,
|
|
64
|
+
skipRoleReminder = false,
|
|
65
|
+
cacheKeyOverride,
|
|
66
|
+
schemaAllowedTools,
|
|
67
|
+
}) {
|
|
68
|
+
const effectivePermission = resolveBridgeSessionPermission(role, permission);
|
|
69
|
+
// Pass cwd through verbatim — null is the fixed bridge sentinel meaning
|
|
70
|
+
// "no caller workspace context" (cycle1-agent shards, etc). Upgrading
|
|
71
|
+
// null → process.cwd() here would defeat cache-shard fork suppression.
|
|
72
|
+
// Downstream collectors (collect.mjs) handle null as "no project cwd".
|
|
73
|
+
const effectiveCwd = cwd == null ? null : cwd;
|
|
74
|
+
const effectiveOwnerSessionId = ownerSessionId === undefined
|
|
75
|
+
? (process.env.MIXDOG_OWNER_SESSION_ID || null)
|
|
76
|
+
: ownerSessionId;
|
|
77
|
+
const sessionOpts = {
|
|
78
|
+
preset,
|
|
79
|
+
owner,
|
|
80
|
+
scopeKey: runtimeSpec.scopeKey,
|
|
81
|
+
lane: runtimeSpec.lane,
|
|
82
|
+
cwd: effectiveCwd,
|
|
83
|
+
role: role || undefined,
|
|
84
|
+
taskType: taskType || undefined,
|
|
85
|
+
maxLoopIterations: Number.isFinite(maxLoopIterations) ? maxLoopIterations : undefined,
|
|
86
|
+
sourceType: sourceType || undefined,
|
|
87
|
+
sourceName: sourceName || undefined,
|
|
88
|
+
ownerSessionId: effectiveOwnerSessionId || null,
|
|
89
|
+
clientHostPid: clientHostPid || null,
|
|
90
|
+
};
|
|
91
|
+
if (bridgeTag) sessionOpts.bridgeTag = bridgeTag;
|
|
92
|
+
if (effectivePermission) sessionOpts.permission = effectivePermission;
|
|
93
|
+
if (permissionMode) sessionOpts.permissionMode = permissionMode;
|
|
94
|
+
if (skipRoleReminder) sessionOpts.skipRoleReminder = true;
|
|
95
|
+
if (cacheKeyOverride) sessionOpts.cacheKeyOverride = cacheKeyOverride;
|
|
96
|
+
if (Array.isArray(schemaAllowedTools)) {
|
|
97
|
+
sessionOpts.schemaAllowedTools = schemaAllowedTools;
|
|
98
|
+
}
|
|
99
|
+
const session = createSession(sessionOpts);
|
|
100
|
+
try {
|
|
101
|
+
traceBridgePreset({
|
|
102
|
+
sessionId: session.id,
|
|
103
|
+
role: role || null,
|
|
104
|
+
presetName: presetName || null,
|
|
105
|
+
// runtimeSpec carries scopeKey/lane but resolveRuntimeSpec does not
|
|
106
|
+
// populate model/provider — fall back to preset fields.
|
|
107
|
+
model: runtimeSpec?.model || preset?.model || null,
|
|
108
|
+
provider: runtimeSpec?.provider || preset?.provider || null,
|
|
109
|
+
parentSessionId: parentSessionId || null,
|
|
110
|
+
permission: effectivePermission || null,
|
|
111
|
+
sourceName: sourceName || null,
|
|
112
|
+
cacheKeyOverride: cacheKeyOverride || null,
|
|
113
|
+
});
|
|
114
|
+
} catch { /* telemetry best-effort */ }
|
|
115
|
+
return { session, effectiveCwd };
|
|
116
|
+
}
|
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
import { getHiddenRole } from './internal-roles.mjs';
|
|
2
|
+
|
|
3
|
+
const SECOND_MS = 1000;
|
|
4
|
+
const MIN_PROVIDER_TIMEOUT_MS = 30_000;
|
|
5
|
+
|
|
6
|
+
export const STALL_TICK_MS = 15_000;
|
|
7
|
+
export const DEFAULT_STALL_WARN_S = 300;
|
|
8
|
+
export const DEFAULT_STALL_ABORT_S = 600;
|
|
9
|
+
// First-byte (no-stream-delta) abort for the bridge stall watchdog. A wedged
|
|
10
|
+
// socket can sit at stage=requesting with zero server events. The 30s deadline
|
|
11
|
+
// trialed here false-aborted slow high-reasoning first bytes (e.g. gpt-5.5
|
|
12
|
+
// XHIGH, which can legitimately think >30s before the first delta) and, paired
|
|
13
|
+
// with dispatch auto-retry, produced premature aborts + duplicate re-dispatches.
|
|
14
|
+
// Auto-retry is now removed (bridge-retry.mjs MAX_ATTEMPTS = 1), so a single
|
|
15
|
+
// attempt must get a generous first-byte window: 300s (5 min). Env-overridable.
|
|
16
|
+
export const DEFAULT_STALL_FIRST_BYTE_ABORT_S = (() => {
|
|
17
|
+
const raw = process.env.MIXDOG_STALL_FIRST_BYTE_ABORT_S;
|
|
18
|
+
const n = Number.parseInt(raw, 10);
|
|
19
|
+
if (Number.isFinite(n) && n > 0) return Math.min(Math.max(n, 5), 600);
|
|
20
|
+
return 300;
|
|
21
|
+
})();
|
|
22
|
+
|
|
23
|
+
export function envThresholdSeconds(env = process.env) {
|
|
24
|
+
const raw = env.STALL_TIMEOUT_S;
|
|
25
|
+
if (!raw) return null;
|
|
26
|
+
const n = Number.parseInt(raw, 10);
|
|
27
|
+
if (!Number.isFinite(n) || n <= 0) return null;
|
|
28
|
+
return n;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export function resolveBaseStallThresholds(env = process.env) {
|
|
32
|
+
const abort = envThresholdSeconds(env) ?? DEFAULT_STALL_ABORT_S;
|
|
33
|
+
const warn = abort > DEFAULT_STALL_WARN_S ? DEFAULT_STALL_WARN_S : Math.floor(abort / 2);
|
|
34
|
+
return { warn, abort };
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
const _baseThresholds = resolveBaseStallThresholds();
|
|
38
|
+
export const STALL_WARN_S = _baseThresholds.warn;
|
|
39
|
+
export const STALL_ABORT_S = _baseThresholds.abort;
|
|
40
|
+
export const STALL_WARN_MS = STALL_WARN_S * SECOND_MS;
|
|
41
|
+
export const STALL_ABORT_MS = STALL_ABORT_S * SECOND_MS;
|
|
42
|
+
|
|
43
|
+
export const PROVIDER_MAX_BEFORE_WARN_MS = Math.max(
|
|
44
|
+
SECOND_MS,
|
|
45
|
+
STALL_WARN_MS - STALL_TICK_MS,
|
|
46
|
+
);
|
|
47
|
+
|
|
48
|
+
export function resolveTimeoutMs(envNames, fallbackMs, { minMs = 1_000, maxMs = Number.POSITIVE_INFINITY, env = process.env } = {}) {
|
|
49
|
+
const names = Array.isArray(envNames) ? envNames : [envNames].filter(Boolean);
|
|
50
|
+
for (const name of names) {
|
|
51
|
+
const raw = env?.[name];
|
|
52
|
+
if (raw == null || raw === '') continue;
|
|
53
|
+
const n = Number(raw);
|
|
54
|
+
if (Number.isFinite(n) && n > 0) {
|
|
55
|
+
return Math.min(Math.max(n, minMs), maxMs);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
return Math.min(Math.max(fallbackMs, minMs), maxMs);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export const PROVIDER_FIRST_BYTE_TIMEOUT_MS = resolveTimeoutMs(
|
|
62
|
+
'MIXDOG_PROVIDER_FIRST_BYTE_TIMEOUT_MS',
|
|
63
|
+
Math.min(120_000, PROVIDER_MAX_BEFORE_WARN_MS),
|
|
64
|
+
{ minMs: MIN_PROVIDER_TIMEOUT_MS, maxMs: PROVIDER_MAX_BEFORE_WARN_MS },
|
|
65
|
+
);
|
|
66
|
+
|
|
67
|
+
export const PROVIDER_GENERATE_TOTAL_TIMEOUT_MS = resolveTimeoutMs(
|
|
68
|
+
'MIXDOG_PROVIDER_GENERATE_TOTAL_TIMEOUT_MS',
|
|
69
|
+
PROVIDER_MAX_BEFORE_WARN_MS,
|
|
70
|
+
{ minMs: PROVIDER_FIRST_BYTE_TIMEOUT_MS, maxMs: PROVIDER_MAX_BEFORE_WARN_MS },
|
|
71
|
+
);
|
|
72
|
+
|
|
73
|
+
export const PROVIDER_CACHE_CREATE_TIMEOUT_MS = resolveTimeoutMs(
|
|
74
|
+
'MIXDOG_PROVIDER_CACHE_CREATE_TIMEOUT_MS',
|
|
75
|
+
Math.min(120_000, PROVIDER_GENERATE_TOTAL_TIMEOUT_MS),
|
|
76
|
+
{ minMs: MIN_PROVIDER_TIMEOUT_MS, maxMs: PROVIDER_MAX_BEFORE_WARN_MS },
|
|
77
|
+
);
|
|
78
|
+
|
|
79
|
+
export const PROVIDER_CACHE_CREATE_TOTAL_TIMEOUT_MS = resolveTimeoutMs(
|
|
80
|
+
'MIXDOG_PROVIDER_CACHE_CREATE_TOTAL_TIMEOUT_MS',
|
|
81
|
+
Math.min(180_000, PROVIDER_GENERATE_TOTAL_TIMEOUT_MS),
|
|
82
|
+
{ minMs: MIN_PROVIDER_TIMEOUT_MS, maxMs: PROVIDER_MAX_BEFORE_WARN_MS },
|
|
83
|
+
);
|
|
84
|
+
|
|
85
|
+
export const PROVIDER_HTTP_RESPONSE_TIMEOUT_MS = resolveTimeoutMs(
|
|
86
|
+
'MIXDOG_PROVIDER_HTTP_RESPONSE_TIMEOUT_MS',
|
|
87
|
+
60_000,
|
|
88
|
+
{ minMs: 10_000, maxMs: PROVIDER_MAX_BEFORE_WARN_MS },
|
|
89
|
+
);
|
|
90
|
+
|
|
91
|
+
// Stream idle watchdog is OFF by default — matches Claude Code native
|
|
92
|
+
// behaviour (its watchdog is gated behind CLAUDE_ENABLE_STREAM_WATCHDOG).
|
|
93
|
+
// The bridge stall watchdog (STALL_ABORT_S, 600s) is the backstop for
|
|
94
|
+
// genuinely dead streams; this short inter-chunk idle watchdog only adds
|
|
95
|
+
// value when explicitly enabled, and otherwise prematurely kills slow
|
|
96
|
+
// high-reasoning streams. Enable with MIXDOG_ENABLE_STREAM_WATCHDOG=1.
|
|
97
|
+
const _sseWatchdogRaw = process.env.MIXDOG_ENABLE_STREAM_WATCHDOG;
|
|
98
|
+
export const PROVIDER_SSE_IDLE_WATCHDOG_ENABLED =
|
|
99
|
+
_sseWatchdogRaw === '1' || _sseWatchdogRaw === 'true' || _sseWatchdogRaw === 'yes';
|
|
100
|
+
|
|
101
|
+
export const PROVIDER_SSE_IDLE_TIMEOUT_MS = resolveTimeoutMs(
|
|
102
|
+
'MIXDOG_PROVIDER_SSE_IDLE_TIMEOUT_MS',
|
|
103
|
+
90_000,
|
|
104
|
+
{ minMs: 10_000, maxMs: STALL_WARN_MS },
|
|
105
|
+
);
|
|
106
|
+
|
|
107
|
+
export const PROVIDER_WS_HANDSHAKE_TIMEOUT_MS = resolveTimeoutMs(
|
|
108
|
+
'MIXDOG_PROVIDER_WS_HANDSHAKE_TIMEOUT_MS',
|
|
109
|
+
30_000,
|
|
110
|
+
{ minMs: 5_000, maxMs: PROVIDER_MAX_BEFORE_WARN_MS },
|
|
111
|
+
);
|
|
112
|
+
|
|
113
|
+
export const PROVIDER_WS_ACQUIRE_TIMEOUT_MS = resolveTimeoutMs(
|
|
114
|
+
'MIXDOG_PROVIDER_WS_ACQUIRE_TIMEOUT_MS',
|
|
115
|
+
15_000,
|
|
116
|
+
{ minMs: 5_000, maxMs: PROVIDER_MAX_BEFORE_WARN_MS },
|
|
117
|
+
);
|
|
118
|
+
|
|
119
|
+
export const PROVIDER_WS_FIRST_MEANINGFUL_TIMEOUT_MS = resolveTimeoutMs(
|
|
120
|
+
'MIXDOG_PROVIDER_WS_FIRST_MEANINGFUL_TIMEOUT_MS',
|
|
121
|
+
PROVIDER_GENERATE_TOTAL_TIMEOUT_MS,
|
|
122
|
+
{ minMs: MIN_PROVIDER_TIMEOUT_MS, maxMs: PROVIDER_MAX_BEFORE_WARN_MS },
|
|
123
|
+
);
|
|
124
|
+
|
|
125
|
+
export const PROVIDER_WS_INTER_CHUNK_TIMEOUT_MS = resolveTimeoutMs(
|
|
126
|
+
'MIXDOG_PROVIDER_WS_INTER_CHUNK_TIMEOUT_MS',
|
|
127
|
+
STALL_ABORT_MS,
|
|
128
|
+
{ minMs: STALL_WARN_MS, maxMs: STALL_ABORT_MS },
|
|
129
|
+
);
|
|
130
|
+
|
|
131
|
+
export const PROVIDER_RETRY_BACKOFF_MS = Object.freeze([0, 1000, 2000, 4000, 8000]);
|
|
132
|
+
export const PROVIDER_RETRY_MAX_ATTEMPTS = PROVIDER_RETRY_BACKOFF_MS.length;
|
|
133
|
+
export const PROVIDER_RETRY_JITTER_RATIO = (() => {
|
|
134
|
+
const raw = process.env.MIXDOG_PROVIDER_RETRY_JITTER_RATIO;
|
|
135
|
+
const n = Number(raw);
|
|
136
|
+
if (Number.isFinite(n) && n >= 0) return Math.min(n, 1);
|
|
137
|
+
return 0.2;
|
|
138
|
+
})();
|
|
139
|
+
|
|
140
|
+
export function resolveBridgeStallThresholds(role, env = process.env) {
|
|
141
|
+
const cfg = role ? getHiddenRole(role) : null;
|
|
142
|
+
const cfgAbort = cfg?.stallCap?.idleSeconds > 0 ? cfg.stallCap.idleSeconds : STALL_ABORT_S;
|
|
143
|
+
const envOverride = envThresholdSeconds(env);
|
|
144
|
+
const abort = envOverride != null ? envOverride : cfgAbort;
|
|
145
|
+
// Mid-stream "slow" warning disabled — a bridge stall now notifies ONLY at
|
|
146
|
+
// the abort deadline (10 min default). warn === abort means the watchdog's
|
|
147
|
+
// warn branch (verdict 'ok' && stale >= warn) can never fire before 'stall',
|
|
148
|
+
// so the only notification a stalled bridge worker emits is at the deadline.
|
|
149
|
+
const warn = abort;
|
|
150
|
+
// First-byte deadline: a request still in 'requesting' that never produced a
|
|
151
|
+
// single SSE delta is hung, not slow-reasoning. Abort on this shorter deadline
|
|
152
|
+
// so the lead is notified in minutes, not at the full mid-stream window.
|
|
153
|
+
const firstByteAbort = Math.min(abort, DEFAULT_STALL_FIRST_BYTE_ABORT_S);
|
|
154
|
+
return { warn, abort, firstByteAbort };
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
export function resolveBridgeToolThresholdSeconds(role, thresholdSeconds) {
|
|
158
|
+
const cfg = role ? getHiddenRole(role) : null;
|
|
159
|
+
if (cfg?.stallCap?.toolRunningSeconds > 0) return cfg.stallCap.toolRunningSeconds;
|
|
160
|
+
return thresholdSeconds;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
export function providerTimeoutError(label, timeoutMs) {
|
|
164
|
+
const err = new Error(`${label} timed out after ${timeoutMs}ms`);
|
|
165
|
+
err.name = 'ProviderTimeoutError';
|
|
166
|
+
err.code = 'EPROVIDERTIMEOUT';
|
|
167
|
+
return err;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
export function createTimeoutSignal(parentSignal, timeoutMs, label) {
|
|
171
|
+
const ac = new AbortController();
|
|
172
|
+
let timer = null;
|
|
173
|
+
let parentListener = null;
|
|
174
|
+
const cleanup = () => {
|
|
175
|
+
if (timer) { clearTimeout(timer); timer = null; }
|
|
176
|
+
if (parentListener && parentSignal) {
|
|
177
|
+
try { parentSignal.removeEventListener('abort', parentListener); } catch {}
|
|
178
|
+
parentListener = null;
|
|
179
|
+
}
|
|
180
|
+
};
|
|
181
|
+
const abort = (reason) => {
|
|
182
|
+
try { ac.abort(reason); } catch {}
|
|
183
|
+
};
|
|
184
|
+
if (parentSignal) {
|
|
185
|
+
parentListener = () => abort(parentSignal.reason);
|
|
186
|
+
if (parentSignal.aborted) {
|
|
187
|
+
parentListener();
|
|
188
|
+
} else {
|
|
189
|
+
parentSignal.addEventListener('abort', parentListener, { once: true });
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
timer = setTimeout(() => abort(providerTimeoutError(label, timeoutMs)), timeoutMs);
|
|
193
|
+
if (typeof timer.unref === 'function') timer.unref();
|
|
194
|
+
return { signal: ac.signal, cleanup };
|
|
195
|
+
}
|