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