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,651 @@
|
|
|
1
|
+
import { readFileSync, existsSync, readdirSync } from 'fs';
|
|
2
|
+
import { join } from 'path';
|
|
3
|
+
import { homedir } from 'os';
|
|
4
|
+
import { maxMtime, maxMtimeRecursive } from '../cache-mtime.mjs';
|
|
5
|
+
// --- Agent template loading ---
|
|
6
|
+
/**
|
|
7
|
+
* Load an agent MD file (Worker.md, Reviewer.md, etc.) as session instructions.
|
|
8
|
+
* Strips frontmatter, returns the body.
|
|
9
|
+
*/
|
|
10
|
+
// Agent template cache — mtime-invalidated per (name, cwd).
|
|
11
|
+
const _agentTemplateCache = new Map();
|
|
12
|
+
export function loadAgentTemplate(name, cwd) {
|
|
13
|
+
// When cwd is null/missing (bridge maintenance callers like cycle1-agent
|
|
14
|
+
// pass cwd:null on purpose so provider-cache shards don't fork per MCP
|
|
15
|
+
// launch dir), skip the project-scoped template lookup entirely — DO NOT
|
|
16
|
+
// fall back to process.cwd(), which would leak the launcher's working
|
|
17
|
+
// directory into the cache key and fragment the shard per caller workspace.
|
|
18
|
+
const projectDir = (typeof cwd === 'string' && cwd.length > 0) ? cwd : null;
|
|
19
|
+
const key = `${name}|${projectDir ?? '__noproject__'}`;
|
|
20
|
+
// Search paths for agent files. Drop the <projectDir>/.claude/agents
|
|
21
|
+
// entry when cwd is missing; home-level + plugin walk still apply so the
|
|
22
|
+
// function's "no template found → null" contract is preserved via the
|
|
23
|
+
// normal readSafe loop below.
|
|
24
|
+
const searchPaths = [];
|
|
25
|
+
if (projectDir) {
|
|
26
|
+
searchPaths.push(join(projectDir, '.claude', 'agents', `${name}.md`));
|
|
27
|
+
}
|
|
28
|
+
searchPaths.push(join(homedir(), '.claude', 'agents', `${name}.md`));
|
|
29
|
+
// Derive the set of directories that gate freshness. Include both the
|
|
30
|
+
// containing agents/ dirs and the plugin marketplaces root so any
|
|
31
|
+
// addition/removal of nested agent files is detected.
|
|
32
|
+
const mtimePaths = [];
|
|
33
|
+
if (projectDir) mtimePaths.push(join(projectDir, '.claude', 'agents'));
|
|
34
|
+
mtimePaths.push(join(homedir(), '.claude', 'agents'));
|
|
35
|
+
// Also search plugin agent directories — precise sentinels only, no
|
|
36
|
+
// full marketplaces tree walk.
|
|
37
|
+
const pluginAgentDirs = [];
|
|
38
|
+
// 1. Cache layout: CLAUDE_PLUGIN_ROOT points to the live versioned dir.
|
|
39
|
+
const _pluginRoot = process.env.CLAUDE_PLUGIN_ROOT;
|
|
40
|
+
if (_pluginRoot) {
|
|
41
|
+
const cacheAgentsDir = join(_pluginRoot, 'agents');
|
|
42
|
+
searchPaths.push(join(cacheAgentsDir, `${name}.md`));
|
|
43
|
+
pluginAgentDirs.push(cacheAgentsDir);
|
|
44
|
+
}
|
|
45
|
+
// 2. Dev-sync / source layout: marketplaces/trib-plugin/agents.
|
|
46
|
+
const devSyncAgentsDir = join(homedir(), '.claude', 'plugins', 'marketplaces', 'trib-plugin', 'agents');
|
|
47
|
+
searchPaths.push(join(devSyncAgentsDir, `${name}.md`));
|
|
48
|
+
pluginAgentDirs.push(devSyncAgentsDir);
|
|
49
|
+
// maxMtimeRecursive only over the two precise agents/ sentinel dirs.
|
|
50
|
+
const mtime = pluginAgentDirs.length > 0
|
|
51
|
+
? Math.max(maxMtimeRecursive(mtimePaths), maxMtimeRecursive(pluginAgentDirs))
|
|
52
|
+
: maxMtimeRecursive(mtimePaths);
|
|
53
|
+
const cached = _agentTemplateCache.get(key);
|
|
54
|
+
if (cached && mtime <= cached.mtime) return cached.value;
|
|
55
|
+
for (const p of searchPaths) {
|
|
56
|
+
const content = readSafe(p);
|
|
57
|
+
if (content) {
|
|
58
|
+
// Strip YAML frontmatter
|
|
59
|
+
const stripped = content.replace(/^---\n[\s\S]*?\n---\n*/, '');
|
|
60
|
+
const body = stripped.trim();
|
|
61
|
+
_agentTemplateCache.set(key, { mtime, value: body });
|
|
62
|
+
return body;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
_agentTemplateCache.set(key, { mtime, value: null });
|
|
66
|
+
return null;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Collect available skills (frontmatter only — token efficient).
|
|
70
|
+
* Full content loaded on demand via loadSkillContent().
|
|
71
|
+
*/
|
|
72
|
+
export function collectSkills(cwd) {
|
|
73
|
+
// When cwd is null/missing (e.g. bridge maintenance callers that pass
|
|
74
|
+
// cwd:null on purpose so provider-cache shards don't fork per caller
|
|
75
|
+
// workspace), skip project-scoped skills entirely — DO NOT fall back
|
|
76
|
+
// to process.cwd(), which would leak the MCP launch dir into the
|
|
77
|
+
// shard key and fragment the cache.
|
|
78
|
+
const projectDir = (typeof cwd === 'string' && cwd.length > 0) ? cwd : null;
|
|
79
|
+
const skills = [];
|
|
80
|
+
const dirs = [
|
|
81
|
+
join(homedir(), '.claude', 'skills'),
|
|
82
|
+
];
|
|
83
|
+
if (projectDir) {
|
|
84
|
+
dirs.push(join(projectDir, '.claude', 'skills'));
|
|
85
|
+
}
|
|
86
|
+
// Plugin skill directories
|
|
87
|
+
const pluginBase = join(homedir(), '.claude', 'plugins', 'marketplaces');
|
|
88
|
+
if (existsSync(pluginBase)) {
|
|
89
|
+
try {
|
|
90
|
+
walkForSkills(pluginBase, dirs);
|
|
91
|
+
}
|
|
92
|
+
catch { /* ignore */ }
|
|
93
|
+
}
|
|
94
|
+
const seen = new Set();
|
|
95
|
+
for (const dir of dirs) {
|
|
96
|
+
if (!existsSync(dir))
|
|
97
|
+
continue;
|
|
98
|
+
try {
|
|
99
|
+
const files = readdirSync(dir, { recursive: true });
|
|
100
|
+
for (const f of files) {
|
|
101
|
+
if (!String(f).endsWith('.md'))
|
|
102
|
+
continue;
|
|
103
|
+
const filePath = join(dir, String(f));
|
|
104
|
+
const content = readSafe(filePath);
|
|
105
|
+
if (!content)
|
|
106
|
+
continue;
|
|
107
|
+
const fm = parseFrontmatter(content);
|
|
108
|
+
if (!fm.name)
|
|
109
|
+
continue;
|
|
110
|
+
if (seen.has(fm.name))
|
|
111
|
+
continue;
|
|
112
|
+
seen.add(fm.name);
|
|
113
|
+
skills.push({
|
|
114
|
+
name: fm.name,
|
|
115
|
+
description: fm.description || '',
|
|
116
|
+
filePath,
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
catch { /* ignore */ }
|
|
121
|
+
}
|
|
122
|
+
return skills;
|
|
123
|
+
}
|
|
124
|
+
// --- Skill cache (mtime-based, keyed by cwd) ---
|
|
125
|
+
const _skillsCache = new Map();
|
|
126
|
+
const _mtimeCache = new Map();
|
|
127
|
+
const _MTIME_TTL_MS = 2000;
|
|
128
|
+
export function collectSkillsCached(cwd) {
|
|
129
|
+
const key = cwd ?? '';
|
|
130
|
+
const projectDir = (typeof cwd === 'string' && cwd.length > 0) ? cwd : null;
|
|
131
|
+
const mtimePaths = [join(homedir(), '.claude', 'skills')];
|
|
132
|
+
if (projectDir) mtimePaths.push(join(projectDir, '.claude', 'skills'));
|
|
133
|
+
const pluginBase = join(homedir(), '.claude', 'plugins', 'marketplaces');
|
|
134
|
+
const skillsDirs = [...mtimePaths];
|
|
135
|
+
if (existsSync(pluginBase)) {
|
|
136
|
+
try { walkForSkills(pluginBase, skillsDirs); } catch { /* ignore */ }
|
|
137
|
+
}
|
|
138
|
+
let mtime;
|
|
139
|
+
const mtimeCached = _mtimeCache.get(key);
|
|
140
|
+
if (mtimeCached && Date.now() - mtimeCached.checkedAt < _MTIME_TTL_MS) {
|
|
141
|
+
mtime = mtimeCached.mtime;
|
|
142
|
+
} else {
|
|
143
|
+
mtime = maxMtimeRecursive(skillsDirs);
|
|
144
|
+
_mtimeCache.set(key, { mtime, checkedAt: Date.now() });
|
|
145
|
+
if (_mtimeCache.size > 16) {
|
|
146
|
+
_mtimeCache.delete(_mtimeCache.keys().next().value);
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
const entry = _skillsCache.get(key);
|
|
150
|
+
if (entry && entry.mtime >= mtime) {
|
|
151
|
+
return entry.value;
|
|
152
|
+
}
|
|
153
|
+
const skills = collectSkills(cwd);
|
|
154
|
+
_skillsCache.set(key, { value: skills, mtime });
|
|
155
|
+
if (_skillsCache.size > 16) {
|
|
156
|
+
_skillsCache.delete(_skillsCache.keys().next().value);
|
|
157
|
+
}
|
|
158
|
+
return skills;
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Load full skill content by name.
|
|
162
|
+
*/
|
|
163
|
+
export function loadSkillContent(name, cwd) {
|
|
164
|
+
const skills = collectSkillsCached(cwd);
|
|
165
|
+
const skill = skills.find(s => s.name === name);
|
|
166
|
+
if (!skill)
|
|
167
|
+
return null;
|
|
168
|
+
return readSafe(skill.filePath);
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* Build slim skill tool definitions (Hermes-style 3-tool split).
|
|
172
|
+
* The skill catalogue is served at runtime via `skills_list` rather than
|
|
173
|
+
* inlined into tool descriptions, keeping per-session schema bytes small.
|
|
174
|
+
*
|
|
175
|
+
* The structure is constant regardless of how many skills are in scope —
|
|
176
|
+
* the 3-tool shape only shows up when `skills.length > 0`, and the slot
|
|
177
|
+
* contents never change. Memoise so every createSession doesn't rebuild
|
|
178
|
+
* identical objects (trivial work, but the allocation noise shows up in
|
|
179
|
+
* repeated Pool C fan-out).
|
|
180
|
+
*/
|
|
181
|
+
let _skillToolDefsCache = null;
|
|
182
|
+
/**
|
|
183
|
+
* @param {Array} skills — discovered skill frontmatter list (may be empty)
|
|
184
|
+
* @param {object} [opts]
|
|
185
|
+
* @param {boolean} [opts.ownerIsBridge=false]
|
|
186
|
+
* Bridge sessions ALWAYS include the 3 meta-tools regardless of the current
|
|
187
|
+
* cwd's skill inventory — the concrete skill list is resolved at tool-call
|
|
188
|
+
* time (cwd-scoped) so the tool schema stays bit-identical across roles /
|
|
189
|
+
* cwds and the provider cache shard does not fragment.
|
|
190
|
+
* Non-bridge sessions keep the historical "empty when skills.length===0"
|
|
191
|
+
* behaviour.
|
|
192
|
+
*/
|
|
193
|
+
export function buildSkillToolDefs(skills, { ownerIsBridge = false } = {}) {
|
|
194
|
+
if (!ownerIsBridge && !skills.length) return [];
|
|
195
|
+
if (_skillToolDefsCache) return _skillToolDefsCache;
|
|
196
|
+
_skillToolDefsCache = [
|
|
197
|
+
{
|
|
198
|
+
name: 'skills_list',
|
|
199
|
+
description: 'List available skills with short descriptions. Call before skill_view or skill_execute.',
|
|
200
|
+
inputSchema: {
|
|
201
|
+
type: 'object',
|
|
202
|
+
properties: {},
|
|
203
|
+
required: [],
|
|
204
|
+
},
|
|
205
|
+
},
|
|
206
|
+
{
|
|
207
|
+
name: 'skill_view',
|
|
208
|
+
description: 'Return the full body of a skill by name (without executing).',
|
|
209
|
+
inputSchema: {
|
|
210
|
+
type: 'object',
|
|
211
|
+
properties: {
|
|
212
|
+
name: { type: 'string', description: 'Skill name' },
|
|
213
|
+
},
|
|
214
|
+
required: ['name'],
|
|
215
|
+
},
|
|
216
|
+
},
|
|
217
|
+
{
|
|
218
|
+
name: 'skill_execute',
|
|
219
|
+
description: 'Load and execute a skill. Skill body is injected into context.',
|
|
220
|
+
inputSchema: {
|
|
221
|
+
type: 'object',
|
|
222
|
+
properties: {
|
|
223
|
+
name: { type: 'string', description: 'Skill name' },
|
|
224
|
+
args: { type: 'object', description: 'Optional arguments passed to the skill', additionalProperties: true },
|
|
225
|
+
},
|
|
226
|
+
required: ['name'],
|
|
227
|
+
},
|
|
228
|
+
},
|
|
229
|
+
];
|
|
230
|
+
return _skillToolDefsCache;
|
|
231
|
+
}
|
|
232
|
+
// --- Collect project MD (Phase B §5) ---
|
|
233
|
+
/**
|
|
234
|
+
* Read <cwd>/PROJECT.md if present. Used to inject project-scoped guidance
|
|
235
|
+
* into Tier 3 `# project-context` without polluting Tier 2 (Pool B prefix).
|
|
236
|
+
*/
|
|
237
|
+
// PROJECT.md lookup per cwd — mtime-invalidated so edits are visible
|
|
238
|
+
// on the very next createSession call.
|
|
239
|
+
const _projectMdCache = new Map();
|
|
240
|
+
export function collectProjectMd(cwd) {
|
|
241
|
+
// When cwd is null/missing (bridge maintenance calls deliberately pass
|
|
242
|
+
// cwd:null so provider-cache shards don't fork per caller workspace),
|
|
243
|
+
// return empty — DO NOT fall back to process.cwd(). This path feeds
|
|
244
|
+
// manager.mjs BP3 / sessionMarker, so leaking process.cwd() project
|
|
245
|
+
// context into bridge maintenance sessions would fragment shards per
|
|
246
|
+
// MCP launch dir.
|
|
247
|
+
const projectDir = (typeof cwd === 'string' && cwd.length > 0) ? cwd : null;
|
|
248
|
+
if (!projectDir) return '';
|
|
249
|
+
const filePath = join(projectDir, 'PROJECT.md');
|
|
250
|
+
const mtime = maxMtime([filePath]);
|
|
251
|
+
const cached = _projectMdCache.get(projectDir);
|
|
252
|
+
if (cached && mtime <= cached.mtime) return cached.value;
|
|
253
|
+
const content = readSafe(filePath) || '';
|
|
254
|
+
_projectMdCache.set(projectDir, { mtime, value: content });
|
|
255
|
+
return content;
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
// --- Role template loading (Phase B §4 — UI-managed) ---
|
|
259
|
+
/**
|
|
260
|
+
* Read <dataDir>/roles/<role>.md, parse frontmatter (name, description,
|
|
261
|
+
* permission) and body. Returns { description, permission, body } or null.
|
|
262
|
+
*
|
|
263
|
+
* The role md is created/edited from the Config UI; runtime parses it on
|
|
264
|
+
* each spawn and injects the result into the Tier 3 system-reminder via
|
|
265
|
+
* composeSystemPrompt's `roleTemplate` slot.
|
|
266
|
+
*/
|
|
267
|
+
// Role template cache — mtime-invalidated so UI edits are visible
|
|
268
|
+
// on the very next createSession call without any TTL delay.
|
|
269
|
+
const _roleTemplateCache = new Map();
|
|
270
|
+
export function loadRoleTemplate(role, dataDir) {
|
|
271
|
+
if (!role || !dataDir) return null;
|
|
272
|
+
const key = `${role}|${dataDir}`;
|
|
273
|
+
const path = join(dataDir, 'roles', `${role}.md`);
|
|
274
|
+
const mtime = maxMtime([path]);
|
|
275
|
+
const cached = _roleTemplateCache.get(key);
|
|
276
|
+
if (cached && mtime <= cached.mtime) return cached.value;
|
|
277
|
+
const content = readSafe(path);
|
|
278
|
+
if (!content) {
|
|
279
|
+
_roleTemplateCache.set(key, { mtime, value: null });
|
|
280
|
+
return null;
|
|
281
|
+
}
|
|
282
|
+
const fm = parseFrontmatter(content);
|
|
283
|
+
const body = content.replace(/^---\n[\s\S]*?\n---\n*/, '').trim();
|
|
284
|
+
const description = (fm.description || '').trim();
|
|
285
|
+
const rawPermission = (fm.permission || '').trim().toLowerCase();
|
|
286
|
+
const VALID_ROLE_PERMISSIONS = new Set(['read', 'read-write', 'mcp', 'full']);
|
|
287
|
+
// Fail closed: unknown permission values are rejected rather than silently
|
|
288
|
+
// falling through as full access.
|
|
289
|
+
const permission = VALID_ROLE_PERMISSIONS.has(rawPermission) ? rawPermission : null;
|
|
290
|
+
const template = {
|
|
291
|
+
description: description || null,
|
|
292
|
+
permission,
|
|
293
|
+
body: body || null,
|
|
294
|
+
};
|
|
295
|
+
_roleTemplateCache.set(key, { mtime, value: template });
|
|
296
|
+
return template;
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
// --- Role-scoped catalog loader ---
|
|
300
|
+
// Emits a BP2 block scoped to the calling role:
|
|
301
|
+
// - Public/custom bridge workers: their own agents/<role>.md when present,
|
|
302
|
+
// plus the public bridge-worker contract.
|
|
303
|
+
// - Hidden roles: their own rules/bridge/<role>.md section only.
|
|
304
|
+
// - Null role: falls back to the full all-in-one block
|
|
305
|
+
// (explicit-cache unified-shard path).
|
|
306
|
+
//
|
|
307
|
+
// BP2 is no longer bit-identical cross-role — the provider cache shards by
|
|
308
|
+
// role group (public / each retrieval hidden / each maintenance hidden),
|
|
309
|
+
// trading a small number of additional shards for ~68% fewer prefix bytes
|
|
310
|
+
// on the public-role hot path. Role identity still rides the sessionMarker
|
|
311
|
+
// user message separately (see composeSystemPrompt).
|
|
312
|
+
//
|
|
313
|
+
// Classification is dynamic — hidden retrieval/maintenance sets come from the
|
|
314
|
+
// `kind` field in internal-roles.mjs. Any other non-null role is public/custom.
|
|
315
|
+
import { listHiddenRolesByKind } from '../internal-roles.mjs';
|
|
316
|
+
|
|
317
|
+
let _roleClassificationCache = null;
|
|
318
|
+
|
|
319
|
+
function loadRoleClassification() {
|
|
320
|
+
if (_roleClassificationCache) return _roleClassificationCache;
|
|
321
|
+
const value = {
|
|
322
|
+
retrieval: new Set(listHiddenRolesByKind('retrieval')),
|
|
323
|
+
maintenance: new Set(listHiddenRolesByKind('maintenance')),
|
|
324
|
+
};
|
|
325
|
+
_roleClassificationCache = value;
|
|
326
|
+
return value;
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
const _scopedRoleCatalogCache = new Map();
|
|
330
|
+
|
|
331
|
+
function loadHiddenRoleSnippets(pluginRoot) {
|
|
332
|
+
try {
|
|
333
|
+
const bridgeDir = join(pluginRoot, 'rules', 'bridge');
|
|
334
|
+
if (!existsSync(bridgeDir)) return [];
|
|
335
|
+
const files = readdirSync(bridgeDir)
|
|
336
|
+
.filter(f => f.endsWith('.md') && f !== '00-common.md')
|
|
337
|
+
.sort();
|
|
338
|
+
const pairs = [];
|
|
339
|
+
for (const f of files) {
|
|
340
|
+
const raw = readSafe(join(bridgeDir, f));
|
|
341
|
+
if (!raw) continue;
|
|
342
|
+
const body = raw.replace(/^---\n[\s\S]*?\n---\n*/, '').trim();
|
|
343
|
+
if (!body) continue;
|
|
344
|
+
const name = f.replace(/^\d+-/, '').replace(/\.md$/, '');
|
|
345
|
+
pairs.push({ name, body });
|
|
346
|
+
}
|
|
347
|
+
return pairs;
|
|
348
|
+
} catch {
|
|
349
|
+
return [];
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
function loadAgentSections(pluginRoot) {
|
|
354
|
+
const agentsDir = join(pluginRoot, 'agents');
|
|
355
|
+
const agentSections = [];
|
|
356
|
+
if (!existsSync(agentsDir)) return agentSections;
|
|
357
|
+
const files = readdirSync(agentsDir)
|
|
358
|
+
.filter(f => f.endsWith('.md'))
|
|
359
|
+
.sort();
|
|
360
|
+
for (const f of files) {
|
|
361
|
+
const raw = readSafe(join(agentsDir, f));
|
|
362
|
+
if (!raw) continue;
|
|
363
|
+
const body = raw.replace(/^---\n[\s\S]*?\n---\n*/, '').trim();
|
|
364
|
+
if (!body) continue;
|
|
365
|
+
const name = f.replace(/\.md$/, '');
|
|
366
|
+
agentSections.push(`## ${name}\n\n${body}`);
|
|
367
|
+
}
|
|
368
|
+
return agentSections;
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
// Empty by design — measurement (dev/role-shard-probe.mjs +
|
|
372
|
+
// dev/prefix-bytes-probe.mjs, 2026-04-29 trace) showed that even on
|
|
373
|
+
// providers with explicit cache breakpoints, BP2 hit-cost dominates
|
|
374
|
+
// cold-load cost at >10 calls/hr. Per-role scoped catalogs cut the
|
|
375
|
+
// average BP2 prefix from 33 KB to <1 KB (~95% reduction), at the
|
|
376
|
+
// price of ~5x more cold loads — a clear net win at observed call
|
|
377
|
+
// volumes (~791 calls/hr, distinct 5 roles/hr). Implicit-prefix-hash
|
|
378
|
+
// providers (deepseek, xai, lmstudio, ollama) prefer the smaller
|
|
379
|
+
// scoped prefix anyway. Leaving the set in place (rather than deleting
|
|
380
|
+
// the branch) so a future provider with different cache economics can
|
|
381
|
+
// be added back without re-introducing the dead code.
|
|
382
|
+
const EXPLICIT_CACHE_PROVIDERS = new Set();
|
|
383
|
+
|
|
384
|
+
// Inbound-event maintenance roles that report results back to Lead. These
|
|
385
|
+
// must follow rules/bridge/20-skip-protocol.md so genuine no-ops (label-only
|
|
386
|
+
// events, dedup, "nothing to report") prefix their output with `[meta:silent]`
|
|
387
|
+
// and the dispatch layer suppresses the Lead inject. Other roles
|
|
388
|
+
// (cycle1/cycle2 memory maintenance, retrieval roles) never emit toward Lead.
|
|
389
|
+
const INBOUND_EVENT_ROLES = new Set(['webhook-handler', 'scheduler-task']);
|
|
390
|
+
|
|
391
|
+
export function loadScopedRoleCatalog(role, provider = null) {
|
|
392
|
+
const useUnified = !!(provider && EXPLICIT_CACHE_PROVIDERS.has(provider));
|
|
393
|
+
const cacheKey = useUnified ? '__unified__' : (role || '__all__');
|
|
394
|
+
const cached = _scopedRoleCatalogCache.get(cacheKey);
|
|
395
|
+
const pluginRoot = process.env.CLAUDE_PLUGIN_ROOT;
|
|
396
|
+
// Use maxMtimeRecursive so edits to .md files inside agents/ and
|
|
397
|
+
// rules/bridge/ propagate — parent dir mtime is unchanged on
|
|
398
|
+
// Linux/macOS when only a nested file's content changes.
|
|
399
|
+
const mtime = pluginRoot ? maxMtimeRecursive([
|
|
400
|
+
join(pluginRoot, 'agents'),
|
|
401
|
+
join(pluginRoot, 'rules', 'bridge'),
|
|
402
|
+
]) : 0;
|
|
403
|
+
if (cached && mtime <= cached.mtime) {
|
|
404
|
+
return cached.value;
|
|
405
|
+
}
|
|
406
|
+
try {
|
|
407
|
+
if (!pluginRoot) return '';
|
|
408
|
+
const agentSections = loadAgentSections(pluginRoot);
|
|
409
|
+
const hiddenPairs = loadHiddenRoleSnippets(pluginRoot);
|
|
410
|
+
const classification = loadRoleClassification();
|
|
411
|
+
|
|
412
|
+
// Pick which bridge-rule sections + agents/<role>.md sections to emit
|
|
413
|
+
// based on role classification. Self-only emit keeps BP2 minimal:
|
|
414
|
+
// a public worker only sees its own agents/worker.md, not the full
|
|
415
|
+
// catalog.
|
|
416
|
+
let bridgeRuleSectionsToEmit = null; // null → drop the bridge-rule block entirely
|
|
417
|
+
let agentSectionsToEmit = agentSections; // default: full (unknown-role fallback)
|
|
418
|
+
if (useUnified) {
|
|
419
|
+
// Explicit-cache providers — every role sees the same all-in-one
|
|
420
|
+
// catalog. Cross-role calls hit the same provider-side prefix
|
|
421
|
+
// shard, eliminating the role-shard miss seen on Pool C
|
|
422
|
+
// transitions for codex/openai. BP3 sessionMarker still carries
|
|
423
|
+
// role identity, so behavior parity is preserved.
|
|
424
|
+
bridgeRuleSectionsToEmit = hiddenPairs.map(p => `## ${p.name}\n\n${p.body}`);
|
|
425
|
+
agentSectionsToEmit = agentSections;
|
|
426
|
+
} else if (role === 'explorer') {
|
|
427
|
+
// Explorer-specific contract (rules/bridge/30-explorer.md) rides
|
|
428
|
+
// BP2 — a deliberate cache-shard split from `worker`: brief-level
|
|
429
|
+
// descriptive-only constraints proved insufficient (haiku still
|
|
430
|
+
// rendered verdicts on evaluative queries), so the contract must
|
|
431
|
+
// sit at system level. agents/worker.md is still shared for the
|
|
432
|
+
// tool-use discipline.
|
|
433
|
+
const self = hiddenPairs.find(p => p.name === 'explorer');
|
|
434
|
+
bridgeRuleSectionsToEmit = self ? [`## ${self.name}\n\n${self.body}`] : [];
|
|
435
|
+
agentSectionsToEmit = agentSections.filter(s => s.startsWith('## worker\n'));
|
|
436
|
+
} else if (role && classification.maintenance.has(role)) {
|
|
437
|
+
const self = hiddenPairs.find(p => p.name === role);
|
|
438
|
+
bridgeRuleSectionsToEmit = [];
|
|
439
|
+
if (self) {
|
|
440
|
+
bridgeRuleSectionsToEmit.push(`## ${self.name}\n\n${self.body}`);
|
|
441
|
+
} else {
|
|
442
|
+
// Fallback: maintenance role without rules/bridge/*.md entry —
|
|
443
|
+
// pull self body from agents/<role>.md instead so newly-added
|
|
444
|
+
// hidden roles work without needing a duplicate bridge file.
|
|
445
|
+
const fromAgent = agentSections.find(s => s.startsWith(`## ${role}\n`));
|
|
446
|
+
if (fromAgent) bridgeRuleSectionsToEmit.push(fromAgent);
|
|
447
|
+
}
|
|
448
|
+
// Inbound-event roles also need the skip-protocol rule so they
|
|
449
|
+
// can opt their no-op outputs out of the Lead inject (BP1 would
|
|
450
|
+
// waste the bytes on unrelated bridge workers).
|
|
451
|
+
if (INBOUND_EVENT_ROLES.has(role)) {
|
|
452
|
+
const skip = hiddenPairs.find(p => p.name === 'skip-protocol');
|
|
453
|
+
if (skip) bridgeRuleSectionsToEmit.push(`## ${skip.name}\n\n${skip.body}`);
|
|
454
|
+
}
|
|
455
|
+
agentSectionsToEmit = [];
|
|
456
|
+
} else if (role) {
|
|
457
|
+
// Public/custom role — self-only agents/<role>.md when present,
|
|
458
|
+
// not the full hidden/maintenance bundle. The universal bridge
|
|
459
|
+
// contract rides BP1 (rules/bridge/00-common.md); the worker tool
|
|
460
|
+
// discipline lives in agents/worker.md.
|
|
461
|
+
bridgeRuleSectionsToEmit = [];
|
|
462
|
+
agentSectionsToEmit = agentSections.filter(s => s.startsWith(`## ${role}\n`));
|
|
463
|
+
} else {
|
|
464
|
+
// Null role — full catalog emitted (explicit-cache providers that
|
|
465
|
+
// shard by __all__ key).
|
|
466
|
+
bridgeRuleSectionsToEmit = hiddenPairs.map(p => `## ${p.name}\n\n${p.body}`);
|
|
467
|
+
// agentSectionsToEmit already set to full agentSections above.
|
|
468
|
+
}
|
|
469
|
+
|
|
470
|
+
const blocks = [];
|
|
471
|
+
if (agentSectionsToEmit.length) {
|
|
472
|
+
blocks.push(`# Agent Role Catalog\n\n${agentSectionsToEmit.join('\n\n---\n\n')}`);
|
|
473
|
+
}
|
|
474
|
+
if (bridgeRuleSectionsToEmit && bridgeRuleSectionsToEmit.length) {
|
|
475
|
+
blocks.push(`# Bridge Role Rules\n\n${bridgeRuleSectionsToEmit.join('\n\n---\n\n')}`);
|
|
476
|
+
}
|
|
477
|
+
const value = blocks.join('\n\n---\n\n');
|
|
478
|
+
_scopedRoleCatalogCache.set(cacheKey, { mtime, value });
|
|
479
|
+
return value;
|
|
480
|
+
} catch {
|
|
481
|
+
return '';
|
|
482
|
+
}
|
|
483
|
+
}
|
|
484
|
+
|
|
485
|
+
// --- Compose system prompt — 4-BP cache layout ---
|
|
486
|
+
// Returns { baseRules, roleCatalog, sessionMarker, volatileTail } mapping
|
|
487
|
+
// directly to the breakpoint plan:
|
|
488
|
+
// BP1 (1h, system block #1) = baseRules — bridge common rules, filtered
|
|
489
|
+
// BP2 (1h, system block #2) = roleCatalog — scoped role catalog + project context
|
|
490
|
+
// BP3 (1h, first <system-reminder> user) = sessionMarker (role-specific task body)
|
|
491
|
+
// BP4 (5m, messages tail) = volatileTail (role + permission + task-brief)
|
|
492
|
+
//
|
|
493
|
+
// Design note — why role/permission sit in BP4, not BP3:
|
|
494
|
+
// BP3 is reserved for role-specific task bodies that are stable within the
|
|
495
|
+
// bridge session (webhook/schedule/hidden retrieval details). A cross-role
|
|
496
|
+
// burst within the same project should still share BP1+BP2, while BP4 picks
|
|
497
|
+
// up the per-call role / permission / task variance. Tool-routing hints are
|
|
498
|
+
// static cross-role, so they live in shared bridge rules rather than being
|
|
499
|
+
// regenerated per call.
|
|
500
|
+
//
|
|
501
|
+
// BP1 inputs:
|
|
502
|
+
// - opts.bridgeRules : rules-builder buildBridgeInjectionContent output
|
|
503
|
+
// (Pool B roles share bit-identical prefix)
|
|
504
|
+
// - opts.userPrompt : explicit systemPrompt override from callsite
|
|
505
|
+
//
|
|
506
|
+
// BP2 inputs:
|
|
507
|
+
// - loadScopedRoleCatalog(opts.role, opts.provider)
|
|
508
|
+
// - opts.projectContext : cwd's PROJECT.md content, when present
|
|
509
|
+
//
|
|
510
|
+
// BP3/BP4 inputs:
|
|
511
|
+
// - opts.role : role name from user-workflow.json or hidden-role registry
|
|
512
|
+
// - opts.agentTemplate : agents/<role>.md body when authored
|
|
513
|
+
// - opts.taskBrief : Lead-issued task description (Sub only)
|
|
514
|
+
// - opts.hasSkills : true → skills_list hint
|
|
515
|
+
// - opts.memoryContext : recap / history context
|
|
516
|
+
//
|
|
517
|
+
// `profile.skip` still filters specific buckets (claudemd, skills, memory)
|
|
518
|
+
// for backward compatibility with existing profiles.
|
|
519
|
+
export function composeSystemPrompt(opts) {
|
|
520
|
+
const profile = opts.profile || null;
|
|
521
|
+
const skip = profile?.skip || {};
|
|
522
|
+
|
|
523
|
+
// ── BP1: baseRules (system block #1, 1h cache) ─────────────────────
|
|
524
|
+
// Bridge common rules + explicit systemPrompt override. Contains
|
|
525
|
+
// bridgeRules (MCP instructions, Pool B shared rules, _shared/tool
|
|
526
|
+
// efficiency). Identical across ALL roles — BP1 shared pool-wide.
|
|
527
|
+
const baseParts = [];
|
|
528
|
+
if (opts.bridgeRules) baseParts.push(opts.bridgeRules);
|
|
529
|
+
if (opts.userPrompt) baseParts.push(opts.userPrompt);
|
|
530
|
+
const baseRules = baseParts.join('\n\n---\n\n');
|
|
531
|
+
|
|
532
|
+
// ── BP2: roleCatalog (system block #2, 1h cache) ────────────────────
|
|
533
|
+
// Cross-role-stable layer: scoped agents/<role>.md catalog + project
|
|
534
|
+
// context. Role / permission markers are emitted in BP4 instead so a
|
|
535
|
+
// cross-role burst within the same project shares BP1+BP2+BP3 entirely
|
|
536
|
+
// (matches the design note above). Without this split, a worker→reviewer
|
|
537
|
+
// hand-off on the same project would churn BP2 every time the role line
|
|
538
|
+
// changed even though the catalog and project context were identical.
|
|
539
|
+
const roleCatalogScoped = loadScopedRoleCatalog(opts.role || null, opts.provider || null);
|
|
540
|
+
const catalogParts = [];
|
|
541
|
+
if (roleCatalogScoped) catalogParts.push(roleCatalogScoped);
|
|
542
|
+
if (opts.projectContext) {
|
|
543
|
+
catalogParts.push('# project-context\n' + opts.projectContext);
|
|
544
|
+
}
|
|
545
|
+
const roleCatalog = catalogParts.join('\n\n');
|
|
546
|
+
|
|
547
|
+
// ── BP3: sessionMarker (first <system-reminder> user msg, 1h cache) ─
|
|
548
|
+
// Role-specific task instructions only — webhook event body, schedule
|
|
549
|
+
// task body, hidden retrieval tool detail. The <!-- bp3-sentinel --> tag
|
|
550
|
+
// is what Anthropic's findTier3Index() matches on to claim a 1h BP3
|
|
551
|
+
// slot; without role-specific content, sessionMarker stays empty so the
|
|
552
|
+
// sentinel doesn't pin an empty user message into the 1h shard.
|
|
553
|
+
const sessionMarker = opts.roleSpecific
|
|
554
|
+
? '<!-- bp3-sentinel -->\n' + opts.roleSpecific
|
|
555
|
+
: '';
|
|
556
|
+
|
|
557
|
+
// ── BP4-adjacent: volatileTail (second user <system-reminder>, 5m) ──
|
|
558
|
+
// Per-call variance: role marker, permission, task brief. memoryContext
|
|
559
|
+
// is Lead-only (injected by hooks/session-start.cjs); bridge sessions
|
|
560
|
+
// never carry it. Keeping role/permission here (rather than in BP2)
|
|
561
|
+
// means cross-role bursts on the same project share BP1+BP2+BP3
|
|
562
|
+
// entirely — only this 5m volatile tail picks up the per-call variance.
|
|
563
|
+
const volatileParts = [];
|
|
564
|
+
if (opts.role && !opts.skipRoleReminder) {
|
|
565
|
+
volatileParts.push('# role\n' + opts.role);
|
|
566
|
+
}
|
|
567
|
+
const permission = opts.permission || opts.roleTemplate?.permission || null;
|
|
568
|
+
if (permission) {
|
|
569
|
+
let permissionLabel = String(permission);
|
|
570
|
+
let allow =
|
|
571
|
+
permission === 'read'
|
|
572
|
+
? 'read-only; write/edit/bash rejected'
|
|
573
|
+
: permission === 'read-write'
|
|
574
|
+
? 'read + write/edit/bash'
|
|
575
|
+
: permission === 'mcp'
|
|
576
|
+
? 'MCP/internal retrieval tools only; file/shell/edit tools rejected'
|
|
577
|
+
: permission === 'full'
|
|
578
|
+
? 'full — all tools'
|
|
579
|
+
: 'unknown — treat as read-only';
|
|
580
|
+
if (permission && typeof permission === 'object') {
|
|
581
|
+
const allowList = Array.isArray(permission.allow)
|
|
582
|
+
? permission.allow.map(v => String(v || '').trim()).filter(Boolean)
|
|
583
|
+
: [];
|
|
584
|
+
const denyList = Array.isArray(permission.deny)
|
|
585
|
+
? permission.deny.map(v => String(v || '').trim()).filter(Boolean)
|
|
586
|
+
: [];
|
|
587
|
+
const parts = [];
|
|
588
|
+
if (allowList.length) parts.push(`allow: ${allowList.join(', ')}`);
|
|
589
|
+
if (denyList.length) parts.push(`deny: ${denyList.join(', ')}`);
|
|
590
|
+
permissionLabel = parts.length ? parts.join('; ') : 'custom tool permission';
|
|
591
|
+
allow = allowList.length
|
|
592
|
+
? 'only listed tools are available'
|
|
593
|
+
: denyList.length
|
|
594
|
+
? 'listed tools are rejected'
|
|
595
|
+
: 'custom allow/deny policy';
|
|
596
|
+
}
|
|
597
|
+
volatileParts.push(`# permission\n${permissionLabel} — ${allow}.`);
|
|
598
|
+
}
|
|
599
|
+
if (opts.taskBrief) volatileParts.push('# task-brief\n' + opts.taskBrief);
|
|
600
|
+
if (opts.cwd && typeof opts.cwd === 'string' && opts.cwd.trim()) {
|
|
601
|
+
volatileParts.push(`# cwd\n${opts.cwd.trim()}`);
|
|
602
|
+
}
|
|
603
|
+
const volatileTail = volatileParts.length > 0
|
|
604
|
+
? volatileParts.join('\n\n')
|
|
605
|
+
: '';
|
|
606
|
+
|
|
607
|
+
return { baseRules, roleCatalog, sessionMarker, volatileTail };
|
|
608
|
+
}
|
|
609
|
+
// --- Helpers ---
|
|
610
|
+
function readSafe(path) {
|
|
611
|
+
try {
|
|
612
|
+
if (!existsSync(path))
|
|
613
|
+
return null;
|
|
614
|
+
const content = readFileSync(path, 'utf-8').trim();
|
|
615
|
+
return content || null;
|
|
616
|
+
}
|
|
617
|
+
catch {
|
|
618
|
+
return null;
|
|
619
|
+
}
|
|
620
|
+
}
|
|
621
|
+
function parseFrontmatter(content) {
|
|
622
|
+
const match = content.match(/^---\n([\s\S]*?)\n---/);
|
|
623
|
+
if (!match)
|
|
624
|
+
return {};
|
|
625
|
+
const fm = match[1];
|
|
626
|
+
const name = fm.match(/^name:\s*["']?(.+?)["']?\s*$/m)?.[1]?.trim();
|
|
627
|
+
const description = fm.match(/^description:\s*["']?(.+?)["']?\s*$/m)?.[1]?.trim();
|
|
628
|
+
const permission = fm.match(/^permission:\s*["']?(.+?)["']?\s*$/m)?.[1]?.trim();
|
|
629
|
+
return { name, description, permission };
|
|
630
|
+
}
|
|
631
|
+
// depth cap: marketplaces/<plugin>/skills/ is depth 2 from pluginBase
|
|
632
|
+
function walkForSkills(dir, result, depth = 0) {
|
|
633
|
+
if (depth > 3) return;
|
|
634
|
+
try {
|
|
635
|
+
const entries = readdirSync(dir, { withFileTypes: true });
|
|
636
|
+
for (const entry of entries) {
|
|
637
|
+
if (entry.name === 'node_modules')
|
|
638
|
+
continue;
|
|
639
|
+
const full = join(dir, entry.name);
|
|
640
|
+
if (entry.isDirectory()) {
|
|
641
|
+
if (entry.name === 'skills') {
|
|
642
|
+
result.push(full);
|
|
643
|
+
}
|
|
644
|
+
else {
|
|
645
|
+
walkForSkills(full, result, depth + 1);
|
|
646
|
+
}
|
|
647
|
+
}
|
|
648
|
+
}
|
|
649
|
+
}
|
|
650
|
+
catch { /* ignore */ }
|
|
651
|
+
}
|