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.
Files changed (404) hide show
  1. package/.claude-plugin/marketplace.json +31 -0
  2. package/.claude-plugin/plugin.json +20 -0
  3. package/.gitattributes +34 -0
  4. package/.mcp.json +14 -0
  5. package/ARCHITECTURE.md +77 -0
  6. package/CHANGELOG.md +7 -0
  7. package/CONTRIBUTING.md +45 -0
  8. package/DATA-FLOW.md +79 -0
  9. package/LICENSE +21 -0
  10. package/README.md +389 -0
  11. package/SECURITY.md +138 -0
  12. package/UNINSTALL.md +112 -0
  13. package/agents/maintenance.md +5 -0
  14. package/agents/memory-classification.md +30 -0
  15. package/agents/scheduler-task.md +18 -0
  16. package/agents/webhook-handler.md +27 -0
  17. package/agents/worker.md +24 -0
  18. package/bin/bridge +133 -0
  19. package/bin/statusline-launcher.mjs +78 -0
  20. package/bin/statusline-lib.mjs +550 -0
  21. package/bin/statusline.mjs +607 -0
  22. package/bun.lock +802 -0
  23. package/commands/config.md +16 -0
  24. package/commands/doctor.md +13 -0
  25. package/commands/setup.md +17 -0
  26. package/defaults/cycle3-review-prompt.md +90 -0
  27. package/defaults/hidden-roles.json +65 -0
  28. package/defaults/memory-chunk-prompt.md +63 -0
  29. package/defaults/memory-promote-prompt.md +135 -0
  30. package/defaults/mixdog-config.template.json +27 -0
  31. package/defaults/user-workflow.json +8 -0
  32. package/defaults/user-workflow.md +12 -0
  33. package/hooks/hooks.json +73 -0
  34. package/hooks/lib/active-instance.cjs +77 -0
  35. package/hooks/lib/permission-evaluator.cjs +411 -0
  36. package/hooks/lib/permission-route.cjs +63 -0
  37. package/hooks/lib/permission-rules.cjs +170 -0
  38. package/hooks/lib/settings-loader.cjs +116 -0
  39. package/hooks/post-tool-use.cjs +84 -0
  40. package/hooks/pre-mcp-sandbox.cjs +158 -0
  41. package/hooks/pre-tool-subagent.cjs +253 -0
  42. package/hooks/session-start.cjs +1372 -0
  43. package/hooks/turn-timer.cjs +82 -0
  44. package/lib/claude-md-writer.cjs +386 -0
  45. package/lib/config-cjs.cjs +61 -0
  46. package/lib/hook-pipe-path.cjs +10 -0
  47. package/lib/keychain-cjs.cjs +263 -0
  48. package/lib/plugin-paths.cjs +61 -0
  49. package/lib/rules-builder.cjs +241 -0
  50. package/lib/text-utils.cjs +61 -0
  51. package/native/README.md +117 -0
  52. package/native/prebuilt/linux-aarch64/mixdog-shim +0 -0
  53. package/native/prebuilt/linux-x86_64/mixdog-shim +0 -0
  54. package/native/prebuilt/macos-aarch64/mixdog-shim +0 -0
  55. package/native/prebuilt/macos-x86_64/mixdog-shim +0 -0
  56. package/native/prebuilt/windows-x86_64/mixdog-shim.exe +0 -0
  57. package/package.json +107 -0
  58. package/prompts/code-review.txt +16 -0
  59. package/prompts/security-audit.txt +17 -0
  60. package/rules/bridge/00-common.md +39 -0
  61. package/rules/bridge/20-skip-protocol.md +18 -0
  62. package/rules/bridge/30-explorer.md +33 -0
  63. package/rules/bridge/40-cycle1-agent.md +52 -0
  64. package/rules/bridge/41-cycle2-agent.md +62 -0
  65. package/rules/bridge/42-cycle3-agent.md +44 -0
  66. package/rules/lead/00-tool-lead.md +61 -0
  67. package/rules/lead/01-general.md +23 -0
  68. package/rules/lead/02-channels.md +49 -0
  69. package/rules/lead/03-team.md +27 -0
  70. package/rules/lead/04-workflow.md +20 -0
  71. package/rules/shared/00-language.md +14 -0
  72. package/rules/shared/01-tool.md +138 -0
  73. package/scripts/bootstrap.mjs +184 -0
  74. package/scripts/bridge-unify-smoke.mjs +308 -0
  75. package/scripts/build-runtime-linux.sh +348 -0
  76. package/scripts/build-runtime-macos.sh +217 -0
  77. package/scripts/build-runtime-windows.ps1 +242 -0
  78. package/scripts/builtin-utils-smoke.mjs +392 -0
  79. package/scripts/check-json.mjs +45 -0
  80. package/scripts/check-syntax-changed.mjs +102 -0
  81. package/scripts/check-syntax.mjs +58 -0
  82. package/scripts/code-graph-batch.test.mjs +33 -0
  83. package/scripts/config-preserve-smoke.mjs +180 -0
  84. package/scripts/doctor.mjs +484 -0
  85. package/scripts/edit-normalize-fuzz.mjs +130 -0
  86. package/scripts/edit-normalize-smoke.mjs +401 -0
  87. package/scripts/edit-operation-smoke.mjs +369 -0
  88. package/scripts/edit2-smoke.mjs +63 -0
  89. package/scripts/fuzzy-e2e.mjs +28 -0
  90. package/scripts/fuzzy-smoke.mjs +26 -0
  91. package/scripts/generate-runtime-manifest.mjs +166 -0
  92. package/scripts/guard-smoke.mjs +66 -0
  93. package/scripts/hidden-role-schema-smoke.mjs +162 -0
  94. package/scripts/hook-routing-smoke.mjs +29 -0
  95. package/scripts/inject-input.ps1 +204 -0
  96. package/scripts/io-complex-smoke.mjs +667 -0
  97. package/scripts/io-explore-bench.mjs +424 -0
  98. package/scripts/io-guardrails-smoke.mjs +205 -0
  99. package/scripts/io-mini-bench-baseline.json +11 -0
  100. package/scripts/io-mini-bench.mjs +216 -0
  101. package/scripts/io-route-harness.mjs +933 -0
  102. package/scripts/io-telemetry-report.mjs +691 -0
  103. package/scripts/mutation-bench.mjs +564 -0
  104. package/scripts/mutation-io-smoke.mjs +1081 -0
  105. package/scripts/native-patch-bridge-smoke.mjs +288 -0
  106. package/scripts/native-patch-smoke.mjs +304 -0
  107. package/scripts/patch-interior-context-smoke.mjs +49 -0
  108. package/scripts/patch-newline-utf8-smoke.mjs +157 -0
  109. package/scripts/perf-hook-smoke.mjs +71 -0
  110. package/scripts/permission-eval-smoke.mjs +426 -0
  111. package/scripts/prep-patch.mjs +53 -0
  112. package/scripts/prep-shim.mjs +96 -0
  113. package/scripts/provider-cache-smoke.mjs +687 -0
  114. package/scripts/report-runtime-health.mjs +132 -0
  115. package/scripts/run-mcp.mjs +1547 -0
  116. package/scripts/salvage-v4a-shatter.test.mjs +58 -0
  117. package/scripts/scoped-cache-io-smoke.mjs +103 -0
  118. package/scripts/shell-policy-round3-smoke.mjs +46 -0
  119. package/scripts/smoke-runtime-negative.ps1 +100 -0
  120. package/scripts/smoke-runtime-negative.sh +95 -0
  121. package/scripts/stall-policy-smoke.mjs +50 -0
  122. package/scripts/start-memory-worker.mjs +23 -0
  123. package/scripts/statusline-launcher-smoke.mjs +82 -0
  124. package/scripts/stress-atomic-write.mjs +1028 -0
  125. package/scripts/test-config-rmw-restore.mjs +122 -0
  126. package/scripts/test-fault-inject.mjs +164 -0
  127. package/scripts/test-large-file.mjs +174 -0
  128. package/scripts/tool-edge-smoke.mjs +209 -0
  129. package/scripts/uninstall.mjs +201 -0
  130. package/scripts/webhook-selfheal-smoke.mjs +29 -0
  131. package/scripts/write-overwrite-guard-smoke.mjs +56 -0
  132. package/server-main.mjs +3055 -0
  133. package/server.mjs +468 -0
  134. package/setup/config-merge.mjs +254 -0
  135. package/setup/install.mjs +120 -0
  136. package/setup/launch-core.mjs +507 -0
  137. package/setup/launch.mjs +101 -0
  138. package/setup/setup-server.mjs +3206 -0
  139. package/setup/setup.html +3693 -0
  140. package/skills/retro-skill-proposer/SKILL.md +92 -0
  141. package/skills/schedule-add/SKILL.md +77 -0
  142. package/skills/setup/SKILL.md +346 -0
  143. package/skills/webhook-add/SKILL.md +81 -0
  144. package/src/agent/bridge-stall-watchdog.mjs +337 -0
  145. package/src/agent/index.mjs +2138 -0
  146. package/src/agent/orchestrator/activity-bus.mjs +38 -0
  147. package/src/agent/orchestrator/ai-wrapped-dispatch.mjs +1010 -0
  148. package/src/agent/orchestrator/bridge-retry.mjs +220 -0
  149. package/src/agent/orchestrator/bridge-trace.mjs +583 -0
  150. package/src/agent/orchestrator/cache-mtime.mjs +58 -0
  151. package/src/agent/orchestrator/config.mjs +358 -0
  152. package/src/agent/orchestrator/context/collect.mjs +651 -0
  153. package/src/agent/orchestrator/dispatch-persist.mjs +549 -0
  154. package/src/agent/orchestrator/drain-registry.mjs +50 -0
  155. package/src/agent/orchestrator/explore-validator.mjs +8 -0
  156. package/src/agent/orchestrator/internal-roles.mjs +118 -0
  157. package/src/agent/orchestrator/internal-tools.mjs +88 -0
  158. package/src/agent/orchestrator/jobs.mjs +116 -0
  159. package/src/agent/orchestrator/mcp/client.mjs +364 -0
  160. package/src/agent/orchestrator/providers/anthropic-betas.mjs +21 -0
  161. package/src/agent/orchestrator/providers/anthropic-oauth.mjs +1745 -0
  162. package/src/agent/orchestrator/providers/anthropic.mjs +437 -0
  163. package/src/agent/orchestrator/providers/gemini.mjs +1175 -0
  164. package/src/agent/orchestrator/providers/grok-oauth.mjs +782 -0
  165. package/src/agent/orchestrator/providers/model-catalog.mjs +241 -0
  166. package/src/agent/orchestrator/providers/openai-compat.mjs +1467 -0
  167. package/src/agent/orchestrator/providers/openai-oauth-ws.mjs +1890 -0
  168. package/src/agent/orchestrator/providers/openai-oauth.mjs +1307 -0
  169. package/src/agent/orchestrator/providers/openai-ws.mjs +104 -0
  170. package/src/agent/orchestrator/providers/registry.mjs +192 -0
  171. package/src/agent/orchestrator/providers/retry-classifier.mjs +325 -0
  172. package/src/agent/orchestrator/session/abort-lookup.mjs +13 -0
  173. package/src/agent/orchestrator/session/cache/post-edit-marks.mjs +42 -0
  174. package/src/agent/orchestrator/session/cache/prefetch-cache.mjs +142 -0
  175. package/src/agent/orchestrator/session/cache/read-cache.mjs +319 -0
  176. package/src/agent/orchestrator/session/cache/scoped-cache-outcome.mjs +11 -0
  177. package/src/agent/orchestrator/session/cache/scoped-cache.mjs +361 -0
  178. package/src/agent/orchestrator/session/cache/util.mjs +49 -0
  179. package/src/agent/orchestrator/session/loop.mjs +1478 -0
  180. package/src/agent/orchestrator/session/manager.mjs +1975 -0
  181. package/src/agent/orchestrator/session/read-dedup.mjs +6 -0
  182. package/src/agent/orchestrator/session/result-classification.mjs +65 -0
  183. package/src/agent/orchestrator/session/save-session-worker.mjs +18 -0
  184. package/src/agent/orchestrator/session/store.mjs +624 -0
  185. package/src/agent/orchestrator/session/stream-watchdog.mjs +130 -0
  186. package/src/agent/orchestrator/session/tool-result-offload.mjs +166 -0
  187. package/src/agent/orchestrator/session/trim.mjs +491 -0
  188. package/src/agent/orchestrator/smart-bridge/CACHE-SHARD.md +115 -0
  189. package/src/agent/orchestrator/smart-bridge/bridge-llm.mjs +327 -0
  190. package/src/agent/orchestrator/smart-bridge/cache-obs.mjs +150 -0
  191. package/src/agent/orchestrator/smart-bridge/cache-strategy.mjs +228 -0
  192. package/src/agent/orchestrator/smart-bridge/index.mjs +215 -0
  193. package/src/agent/orchestrator/smart-bridge/profiles.mjs +37 -0
  194. package/src/agent/orchestrator/smart-bridge/registry.mjs +348 -0
  195. package/src/agent/orchestrator/smart-bridge/session-builder.mjs +116 -0
  196. package/src/agent/orchestrator/stall-policy.mjs +195 -0
  197. package/src/agent/orchestrator/tool-loop-guard.mjs +75 -0
  198. package/src/agent/orchestrator/tools/bash-policy-scan.mjs +77 -0
  199. package/src/agent/orchestrator/tools/bash-session.mjs +721 -0
  200. package/src/agent/orchestrator/tools/builtin/advisory-lock.mjs +171 -0
  201. package/src/agent/orchestrator/tools/builtin/arg-guard.mjs +455 -0
  202. package/src/agent/orchestrator/tools/builtin/atomic-write.mjs +236 -0
  203. package/src/agent/orchestrator/tools/builtin/bash-tool.mjs +480 -0
  204. package/src/agent/orchestrator/tools/builtin/binary-file.mjs +76 -0
  205. package/src/agent/orchestrator/tools/builtin/builtin-tools.mjs +256 -0
  206. package/src/agent/orchestrator/tools/builtin/cache-layers.mjs +386 -0
  207. package/src/agent/orchestrator/tools/builtin/cwd-utils.mjs +37 -0
  208. package/src/agent/orchestrator/tools/builtin/device-paths.mjs +154 -0
  209. package/src/agent/orchestrator/tools/builtin/diagnostics-tool.mjs +292 -0
  210. package/src/agent/orchestrator/tools/builtin/diff-utils.mjs +109 -0
  211. package/src/agent/orchestrator/tools/builtin/edit-base-guard.mjs +58 -0
  212. package/src/agent/orchestrator/tools/builtin/edit-byte-plan.mjs +240 -0
  213. package/src/agent/orchestrator/tools/builtin/edit-byte-utils.mjs +113 -0
  214. package/src/agent/orchestrator/tools/builtin/edit-commit.mjs +74 -0
  215. package/src/agent/orchestrator/tools/builtin/edit-context-utils.mjs +242 -0
  216. package/src/agent/orchestrator/tools/builtin/edit-diagnostics.mjs +211 -0
  217. package/src/agent/orchestrator/tools/builtin/edit-engine.mjs +1364 -0
  218. package/src/agent/orchestrator/tools/builtin/edit-failure-context.mjs +126 -0
  219. package/src/agent/orchestrator/tools/builtin/edit-hint.mjs +141 -0
  220. package/src/agent/orchestrator/tools/builtin/edit-match-utils.mjs +194 -0
  221. package/src/agent/orchestrator/tools/builtin/edit-partial-write.mjs +60 -0
  222. package/src/agent/orchestrator/tools/builtin/edit-stale-refresh.mjs +168 -0
  223. package/src/agent/orchestrator/tools/builtin/edit-tool.mjs +173 -0
  224. package/src/agent/orchestrator/tools/builtin/edit-utf8-guard.mjs +48 -0
  225. package/src/agent/orchestrator/tools/builtin/fs-reachability.mjs +48 -0
  226. package/src/agent/orchestrator/tools/builtin/fuzzy-match.mjs +99 -0
  227. package/src/agent/orchestrator/tools/builtin/glob-walk.mjs +170 -0
  228. package/src/agent/orchestrator/tools/builtin/grep-formatting.mjs +113 -0
  229. package/src/agent/orchestrator/tools/builtin/hash-utils.mjs +6 -0
  230. package/src/agent/orchestrator/tools/builtin/list-formatting.mjs +7 -0
  231. package/src/agent/orchestrator/tools/builtin/list-tool.mjs +593 -0
  232. package/src/agent/orchestrator/tools/builtin/native-edit-runner.mjs +89 -0
  233. package/src/agent/orchestrator/tools/builtin/notebook-edit-tool.mjs +300 -0
  234. package/src/agent/orchestrator/tools/builtin/open-config-tool.mjs +26 -0
  235. package/src/agent/orchestrator/tools/builtin/path-diagnostics.mjs +152 -0
  236. package/src/agent/orchestrator/tools/builtin/path-locks.mjs +35 -0
  237. package/src/agent/orchestrator/tools/builtin/path-utils.mjs +201 -0
  238. package/src/agent/orchestrator/tools/builtin/read-args.mjs +103 -0
  239. package/src/agent/orchestrator/tools/builtin/read-batch.mjs +172 -0
  240. package/src/agent/orchestrator/tools/builtin/read-constants.mjs +40 -0
  241. package/src/agent/orchestrator/tools/builtin/read-formatting.mjs +118 -0
  242. package/src/agent/orchestrator/tools/builtin/read-image-resize.mjs +189 -0
  243. package/src/agent/orchestrator/tools/builtin/read-image.mjs +88 -0
  244. package/src/agent/orchestrator/tools/builtin/read-lines.mjs +12 -0
  245. package/src/agent/orchestrator/tools/builtin/read-mode-tool.mjs +455 -0
  246. package/src/agent/orchestrator/tools/builtin/read-open.mjs +190 -0
  247. package/src/agent/orchestrator/tools/builtin/read-range-index.mjs +271 -0
  248. package/src/agent/orchestrator/tools/builtin/read-ranges.mjs +26 -0
  249. package/src/agent/orchestrator/tools/builtin/read-single-tool.mjs +728 -0
  250. package/src/agent/orchestrator/tools/builtin/read-snapshot-runtime.mjs +173 -0
  251. package/src/agent/orchestrator/tools/builtin/read-special-files.mjs +268 -0
  252. package/src/agent/orchestrator/tools/builtin/read-streaming.mjs +602 -0
  253. package/src/agent/orchestrator/tools/builtin/read-tool.mjs +530 -0
  254. package/src/agent/orchestrator/tools/builtin/read-windows.mjs +107 -0
  255. package/src/agent/orchestrator/tools/builtin/rename-tool.mjs +196 -0
  256. package/src/agent/orchestrator/tools/builtin/rg-runner.mjs +422 -0
  257. package/src/agent/orchestrator/tools/builtin/search-builders.mjs +158 -0
  258. package/src/agent/orchestrator/tools/builtin/search-tool.mjs +869 -0
  259. package/src/agent/orchestrator/tools/builtin/shell-analysis.mjs +653 -0
  260. package/src/agent/orchestrator/tools/builtin/shell-jobs.mjs +936 -0
  261. package/src/agent/orchestrator/tools/builtin/shell-output.mjs +36 -0
  262. package/src/agent/orchestrator/tools/builtin/shell-runtime.mjs +214 -0
  263. package/src/agent/orchestrator/tools/builtin/snapshot-helpers.mjs +143 -0
  264. package/src/agent/orchestrator/tools/builtin/snapshot-store.mjs +206 -0
  265. package/src/agent/orchestrator/tools/builtin/snapshot-validation.mjs +98 -0
  266. package/src/agent/orchestrator/tools/builtin/text-stats.mjs +69 -0
  267. package/src/agent/orchestrator/tools/builtin/windows-roots.mjs +23 -0
  268. package/src/agent/orchestrator/tools/builtin/write-tool.mjs +401 -0
  269. package/src/agent/orchestrator/tools/builtin.mjs +500 -0
  270. package/src/agent/orchestrator/tools/code-graph-prewarm-worker.mjs +39 -0
  271. package/src/agent/orchestrator/tools/code-graph-tool-defs.mjs +24 -0
  272. package/src/agent/orchestrator/tools/code-graph.mjs +4095 -0
  273. package/src/agent/orchestrator/tools/cwd-tool.mjs +298 -0
  274. package/src/agent/orchestrator/tools/destructive-warning.mjs +323 -0
  275. package/src/agent/orchestrator/tools/edit-normalize.mjs +603 -0
  276. package/src/agent/orchestrator/tools/env-scrub.mjs +100 -0
  277. package/src/agent/orchestrator/tools/graph-binary-fetcher.mjs +144 -0
  278. package/src/agent/orchestrator/tools/graph-manifest.json +26 -0
  279. package/src/agent/orchestrator/tools/host-input.mjs +204 -0
  280. package/src/agent/orchestrator/tools/mutation-content-cache.mjs +67 -0
  281. package/src/agent/orchestrator/tools/mutation-planner.mjs +75 -0
  282. package/src/agent/orchestrator/tools/next-call-utils.mjs +48 -0
  283. package/src/agent/orchestrator/tools/patch-binary-fetcher.mjs +133 -0
  284. package/src/agent/orchestrator/tools/patch-manifest.json +26 -0
  285. package/src/agent/orchestrator/tools/patch-tool-defs.mjs +20 -0
  286. package/src/agent/orchestrator/tools/patch.mjs +2754 -0
  287. package/src/agent/orchestrator/tools/progress-message.mjs +118 -0
  288. package/src/agent/orchestrator/tools/result-compression.mjs +279 -0
  289. package/src/agent/orchestrator/tools/shell-command.mjs +865 -0
  290. package/src/agent/orchestrator/tools/shell-exec-policy.mjs +89 -0
  291. package/src/agent/orchestrator/tools/shell-policy-danger-target.mjs +27 -0
  292. package/src/agent/orchestrator/tools/shell-policy-imports.mjs +7 -0
  293. package/src/agent/orchestrator/tools/shell-policy.mjs +345 -0
  294. package/src/agent/orchestrator/tools/shell-snapshot.mjs +313 -0
  295. package/src/agent/orchestrator/workflow-store.mjs +93 -0
  296. package/src/agent/tool-defs.mjs +103 -0
  297. package/src/channels/backends/discord.mjs +784 -0
  298. package/src/channels/data/voice-runtime-manifest.json +138 -0
  299. package/src/channels/index.mjs +3229 -0
  300. package/src/channels/lib/cli-worker-host.mjs +12 -0
  301. package/src/channels/lib/config-lock.mjs +13 -0
  302. package/src/channels/lib/config.mjs +292 -0
  303. package/src/channels/lib/drop-trace.mjs +71 -0
  304. package/src/channels/lib/event-pipeline.mjs +81 -0
  305. package/src/channels/lib/event-queue.mjs +345 -0
  306. package/src/channels/lib/executor.mjs +168 -0
  307. package/src/channels/lib/format.mjs +188 -0
  308. package/src/channels/lib/holidays.mjs +138 -0
  309. package/src/channels/lib/hook-pipe-server.mjs +802 -0
  310. package/src/channels/lib/interaction-workflows.mjs +184 -0
  311. package/src/channels/lib/memory-client.mjs +149 -0
  312. package/src/channels/lib/output-forwarder.mjs +765 -0
  313. package/src/channels/lib/runtime-paths.mjs +479 -0
  314. package/src/channels/lib/scheduler.mjs +723 -0
  315. package/src/channels/lib/session-control.mjs +36 -0
  316. package/src/channels/lib/session-discovery.mjs +103 -0
  317. package/src/channels/lib/settings.mjs +11 -0
  318. package/src/channels/lib/state-file.mjs +68 -0
  319. package/src/channels/lib/status-snapshot.mjs +219 -0
  320. package/src/channels/lib/tool-format.mjs +140 -0
  321. package/src/channels/lib/transcript-discovery.mjs +195 -0
  322. package/src/channels/lib/voice-runtime-fetcher.mjs +734 -0
  323. package/src/channels/lib/webhook.mjs +1179 -0
  324. package/src/channels/lib/whisper-server.mjs +477 -0
  325. package/src/channels/tool-defs.mjs +170 -0
  326. package/src/daemon/host.mjs +118 -0
  327. package/src/daemon/mcp-transport.mjs +47 -0
  328. package/src/daemon/session.mjs +100 -0
  329. package/src/daemon/thin-client.mjs +71 -0
  330. package/src/daemon/transport.mjs +163 -0
  331. package/src/memory/data/runtime-manifest.json +40 -0
  332. package/src/memory/index.mjs +3305 -0
  333. package/src/memory/lib/agent-ipc.mjs +93 -0
  334. package/src/memory/lib/bridge-trace-queries.mjs +120 -0
  335. package/src/memory/lib/core-memory-store.mjs +330 -0
  336. package/src/memory/lib/embedding-provider.mjs +269 -0
  337. package/src/memory/lib/embedding-worker.mjs +323 -0
  338. package/src/memory/lib/llm-worker-host.mjs +17 -0
  339. package/src/memory/lib/memory-cycle.mjs +11 -0
  340. package/src/memory/lib/memory-cycle1.mjs +641 -0
  341. package/src/memory/lib/memory-cycle2.mjs +1284 -0
  342. package/src/memory/lib/memory-cycle3.mjs +540 -0
  343. package/src/memory/lib/memory-embed.mjs +299 -0
  344. package/src/memory/lib/memory-extraction.mjs +5 -0
  345. package/src/memory/lib/memory-maintenance-store.mjs +32 -0
  346. package/src/memory/lib/memory-ops-policy.mjs +190 -0
  347. package/src/memory/lib/memory-recall-id-patch.mjs +15 -0
  348. package/src/memory/lib/memory-recall-read-query.mjs +7 -0
  349. package/src/memory/lib/memory-recall-scope-filter.mjs +63 -0
  350. package/src/memory/lib/memory-recall-store.mjs +621 -0
  351. package/src/memory/lib/memory-retrievers.mjs +112 -0
  352. package/src/memory/lib/memory-score.mjs +71 -0
  353. package/src/memory/lib/memory-text-utils.mjs +58 -0
  354. package/src/memory/lib/memory.mjs +412 -0
  355. package/src/memory/lib/model-profile.mjs +85 -0
  356. package/src/memory/lib/pg/adapter.mjs +308 -0
  357. package/src/memory/lib/pg/process.mjs +360 -0
  358. package/src/memory/lib/pg/supervisor.mjs +396 -0
  359. package/src/memory/lib/project-id-resolver.mjs +86 -0
  360. package/src/memory/lib/runtime-fetcher.mjs +442 -0
  361. package/src/memory/lib/trace-store.mjs +728 -0
  362. package/src/memory/tool-defs.mjs +79 -0
  363. package/src/search/index.mjs +1173 -0
  364. package/src/search/lib/backends/anthropic-oauth.mjs +98 -0
  365. package/src/search/lib/backends/exa.mjs +50 -0
  366. package/src/search/lib/backends/firecrawl.mjs +61 -0
  367. package/src/search/lib/backends/gemini-api.mjs +83 -0
  368. package/src/search/lib/backends/grok-oauth.mjs +86 -0
  369. package/src/search/lib/backends/index.mjs +150 -0
  370. package/src/search/lib/backends/openai-api.mjs +144 -0
  371. package/src/search/lib/backends/openai-oauth.mjs +98 -0
  372. package/src/search/lib/backends/openai-web-search.mjs +76 -0
  373. package/src/search/lib/backends/tavily.mjs +55 -0
  374. package/src/search/lib/backends/xai-api.mjs +113 -0
  375. package/src/search/lib/cache.mjs +131 -0
  376. package/src/search/lib/config.mjs +192 -0
  377. package/src/search/lib/formatter.mjs +115 -0
  378. package/src/search/lib/provider-usage.mjs +67 -0
  379. package/src/search/lib/providers.mjs +47 -0
  380. package/src/search/lib/search-intent.mjs +109 -0
  381. package/src/search/lib/setup-handler.mjs +261 -0
  382. package/src/search/lib/state.mjs +201 -0
  383. package/src/search/lib/web-tools.mjs +1207 -0
  384. package/src/search/tool-defs.mjs +83 -0
  385. package/src/setup/defender-exclusion.mjs +183 -0
  386. package/src/shared/abort-controller.mjs +15 -0
  387. package/src/shared/atomic-file.mjs +420 -0
  388. package/src/shared/config.mjs +350 -0
  389. package/src/shared/daemon-recycle.mjs +108 -0
  390. package/src/shared/disable-claude-builtins.mjs +88 -0
  391. package/src/shared/err-text.mjs +12 -0
  392. package/src/shared/llm/cost.mjs +66 -0
  393. package/src/shared/llm/http-agent.mjs +123 -0
  394. package/src/shared/llm/index.mjs +41 -0
  395. package/src/shared/llm/pid-cleanup.mjs +27 -0
  396. package/src/shared/llm/usage-log.mjs +47 -0
  397. package/src/shared/plugin-paths.mjs +58 -0
  398. package/src/shared/schedules-store.mjs +70 -0
  399. package/src/shared/seed.mjs +119 -0
  400. package/src/shared/user-cwd.mjs +213 -0
  401. package/src/shared/user-data-guard.mjs +238 -0
  402. package/src/status/aggregator.mjs +584 -0
  403. package/src/status/server.mjs +413 -0
  404. 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
+ }