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