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,100 @@
1
+ // R11: scrub dangerous loader/execution env vars from a model-spawned
2
+ // subprocess environment. Defense-in-depth on top of R5's secret scrub
3
+ // (provider/cloud tokens). These vars let an attacker (or model error)
4
+ // inject code before the user's command runs — NODE_OPTIONS can preload
5
+ // arbitrary --require modules, LD_PRELOAD / DYLD_INSERT_LIBRARIES inject
6
+ // shared libraries, PYTHONPATH / RUBYLIB / PERL5LIB hijack imports,
7
+ // BASH_ENV / ENV runs a script on shell startup, SHELLOPTS toggles
8
+ // dangerous shell flags, GLOBIGNORE silently hides files from expansion.
9
+ // CDPATH redirects `cd` to attacker-controlled dirs, PROMPT_COMMAND runs
10
+ // arbitrary code on every prompt, BASHOPTS toggles shell options, IFS
11
+ // rewrites word splitting, SSH_AUTH_SOCK / GPG_AGENT_INFO / GNUPGHOME
12
+ // expose live agent credentials to the child.
13
+ // PATH is intentionally preserved — subprocesses need it to find tools.
14
+ const LOADER_VARS = [
15
+ 'NODE_OPTIONS',
16
+ 'NODE_PATH',
17
+ 'LD_PRELOAD',
18
+ 'LD_LIBRARY_PATH',
19
+ 'DYLD_INSERT_LIBRARIES',
20
+ 'DYLD_LIBRARY_PATH',
21
+ 'PYTHONPATH',
22
+ 'RUBYLIB',
23
+ 'PERL5LIB',
24
+ 'BASH_ENV',
25
+ 'ENV',
26
+ 'SHELLOPTS',
27
+ 'GLOBIGNORE',
28
+ 'CDPATH',
29
+ 'PROMPT_COMMAND',
30
+ 'BASHOPTS',
31
+ 'IFS',
32
+ 'SSH_AUTH_SOCK',
33
+ 'GPG_AGENT_INFO',
34
+ 'GNUPGHOME',
35
+ ];
36
+
37
+ // R5: provider / cloud / secret-family scrub. Shared across all spawn
38
+ // sites so the suffix and exact lists never drift. Broad provider-family
39
+ // prefixes (AWS_/GITHUB_/NPM_/VERCEL_/…) were removed: they nuked
40
+ // non-secret config like AWS_REGION, GITHUB_WORKSPACE, NPM_CONFIG_CACHE,
41
+ // VERCEL_ENV and broke subprocess tooling. Match instead by secret-
42
+ // shaped SUFFIX (covers `*_API_KEY`, `*_SECRET`, `*_TOKEN`, `*_PASSWORD`,
43
+ // `*_CREDENTIALS`, `*_PRIVATE_KEY`, `*_KEY`, …) and an explicit EXACT
44
+ // set for non-suffix secrets (AWS access ids, DATABASE_URL, named
45
+ // provider keys, GOOGLE_APPLICATION_CREDENTIALS).
46
+ // Secret-shaped suffixes. The bare `_KEY` suffix was dropped — it deleted
47
+ // non-secret public keys (GPG_KEY, NEXT_PUBLIC_*_KEY). `_API_KEY` is kept for
48
+ // broad provider coverage (e.g. XAI_API_KEY) but is guarded by PUBLIC_PREFIX_RE
49
+ // so client-exposed build vars (VITE_FIREBASE_API_KEY, NEXT_PUBLIC_*) survive.
50
+ const SECRET_SUFFIX_RE = /(_SECRET_ACCESS_KEY|_ACCESS_KEY|_SESSION_TOKEN|_AUTH_TOKEN|_API_KEY|_TOKEN|_SECRET|_PASSWORD|_CREDENTIALS|_PRIVATE_KEY|_PAT)$/;
51
+ // By-convention client-PUBLIC env prefixes: these are intentionally bundled to
52
+ // the browser and must never be scrubbed as secrets even with an _API_KEY/_KEY
53
+ // shape. Excludes them from the secret match below.
54
+ const PUBLIC_PREFIX_RE = /^(?:NEXT_PUBLIC_|NUXT_PUBLIC_|VITE_|REACT_APP_|VUE_APP_|EXPO_PUBLIC_|GATSBY_|STORYBOOK_|PUBLIC_)/;
55
+ const SECRET_EXACT = new Set([
56
+ 'AWS_ACCESS_KEY_ID',
57
+ 'AWS_SECRET_ACCESS_KEY',
58
+ 'AWS_SESSION_TOKEN',
59
+ 'AWS_WEB_IDENTITY_TOKEN_FILE',
60
+ 'AWS_CONTAINER_CREDENTIALS_RELATIVE_URI',
61
+ 'AWS_CONTAINER_CREDENTIALS_FULL_URI',
62
+ 'DATABASE_URL',
63
+ 'ANTHROPIC_API_KEY',
64
+ 'OPENAI_API_KEY',
65
+ 'GEMINI_API_KEY',
66
+ 'GOOGLE_API_KEY',
67
+ 'GOOGLE_APPLICATION_CREDENTIALS',
68
+ 'GITHUB_PAT',
69
+ 'STRIPE_SECRET_KEY',
70
+ 'SUPABASE_SERVICE_ROLE_KEY',
71
+ 'NPM_CONFIG__AUTH',
72
+ 'npm_config__auth',
73
+ ]);
74
+
75
+ export function scrubLoaderVars(env) {
76
+ if (!env || typeof env !== 'object') return env;
77
+ for (const k of LOADER_VARS) delete env[k];
78
+ // Wildcard sweep: the exact-name list covers the common loader vars but
79
+ // the DYLD_/LD_ families have many siblings (DYLD_FRAMEWORK_PATH,
80
+ // DYLD_FALLBACK_LIBRARY_PATH, LD_AUDIT, LD_BIND_NOW, …). Delete every
81
+ // key under those prefixes so a new variant doesn't sneak through.
82
+ for (const k of Object.keys(env)) {
83
+ if (/^DYLD_/.test(k) || /^LD_/.test(k)) delete env[k];
84
+ }
85
+ return env;
86
+ }
87
+
88
+ // R5: strip provider/cloud/secret-family keys from a spawn env in place.
89
+ // Shared by bash-session, shell-jobs, shell-snapshot, and bash-tool so
90
+ // the prefix/suffix lists never drift across spawn sites. PATH is not
91
+ // matched by any family here and is preserved.
92
+ export function scrubProviderSecrets(env) {
93
+ if (!env || typeof env !== 'object') return env;
94
+ for (const k of Object.keys(env)) {
95
+ if (!PUBLIC_PREFIX_RE.test(k) && (SECRET_EXACT.has(k) || SECRET_SUFFIX_RE.test(k))) {
96
+ delete env[k];
97
+ }
98
+ }
99
+ return env;
100
+ }
@@ -0,0 +1,144 @@
1
+ // graph-binary-fetcher.mjs — fetches the prebuilt mixdog-graph native binary
2
+ // from the GitHub release manifest. The code graph has NO JS parse fallback,
3
+ // so the binary is required; ensureGraphBinary downloads + sha256-verifies it
4
+ // on first use and caches it under <dataDir>/graph-bin/. Mirrors the runtime
5
+ // fetcher pattern (memory/lib/runtime-fetcher.mjs) but for a single binary —
6
+ // no tar extraction.
7
+ //
8
+ // Public API:
9
+ // ensureGraphBinary(dataDir) -> absolute path to the verified binary.
10
+ // Throws if no asset matches the platform, or download/verify fails.
11
+ // findCachedGraphBinary(dataDir) -> path | null (sync, no network).
12
+
13
+ import { createHash } from 'node:crypto';
14
+ import {
15
+ chmodSync, createWriteStream, existsSync, mkdirSync,
16
+ readFileSync, readdirSync, renameSync, rmSync,
17
+ } from 'node:fs';
18
+ import { readFile } from 'node:fs/promises';
19
+ import { join } from 'node:path';
20
+ import { fileURLToPath } from 'node:url';
21
+ import { pipeline } from 'node:stream/promises';
22
+
23
+ // Bundled fallback manifest shipped with the plugin. CI rewrites this on each
24
+ // release with the per-platform asset URLs + sha256.
25
+ const BUNDLED_MANIFEST_PATH = fileURLToPath(new URL('./graph-manifest.json', import.meta.url));
26
+
27
+ // GitHub raw fallback — only consulted when neither cached nor bundled exists.
28
+ const MANIFEST_URL = 'https://raw.githubusercontent.com/trib-plugin/mixdog/main/src/agent/orchestrator/tools/graph-manifest.json';
29
+
30
+ function binSuffix() {
31
+ return process.platform === 'win32' ? '.exe' : '';
32
+ }
33
+
34
+ function platformKey() {
35
+ const os = process.platform === 'win32' ? 'win32' : process.platform;
36
+ return `${os}-${process.arch}`;
37
+ }
38
+
39
+ function graphBinDir(dataDir) {
40
+ return join(dataDir, 'graph-bin');
41
+ }
42
+
43
+ async function loadManifest(dataDir) {
44
+ const cached = join(graphBinDir(dataDir), 'manifest.json');
45
+ if (existsSync(cached)) {
46
+ try { return JSON.parse(readFileSync(cached, 'utf8')); } catch { /* fall through */ }
47
+ }
48
+ if (existsSync(BUNDLED_MANIFEST_PATH)) {
49
+ return JSON.parse(readFileSync(BUNDLED_MANIFEST_PATH, 'utf8'));
50
+ }
51
+ const res = await fetch(MANIFEST_URL, { signal: AbortSignal.timeout(30_000) });
52
+ if (!res.ok) throw new Error(`[graph-fetcher] manifest fetch failed: ${res.status} ${res.statusText}`);
53
+ return res.json();
54
+ }
55
+
56
+ async function sha256File(filePath) {
57
+ return createHash('sha256').update(await readFile(filePath)).digest('hex');
58
+ }
59
+
60
+ async function downloadWithRetry(url, destPath) {
61
+ // 4 attempts total (1 + 3 retries); backoff 1s/3s/9s. 4xx is terminal.
62
+ const delays = [1000, 3000, 9000];
63
+ let lastErr;
64
+ for (let attempt = 0; attempt < 4; attempt++) {
65
+ try {
66
+ const res = await fetch(url, { signal: AbortSignal.timeout(180_000) });
67
+ if (res.status >= 400 && res.status < 500) {
68
+ throw new Error(`[graph-fetcher] asset HTTP ${res.status} (terminal) — ${url}`);
69
+ }
70
+ if (!res.ok) throw new Error(`[graph-fetcher] asset HTTP ${res.status} — ${url}`);
71
+ await pipeline(res.body, createWriteStream(destPath));
72
+ return;
73
+ } catch (err) {
74
+ lastErr = err;
75
+ if (String(err?.message || '').includes('(terminal)')) throw err;
76
+ if (attempt < 3) {
77
+ process.stderr.write(`[graph-fetcher] download attempt ${attempt + 1} failed (${err?.message}), retrying in ${delays[attempt]}ms…\n`);
78
+ await new Promise((r) => setTimeout(r, delays[attempt]));
79
+ }
80
+ }
81
+ }
82
+ throw lastErr;
83
+ }
84
+
85
+ // Remove stale binaries + tmp files, keeping the active one.
86
+ function gcGraphBin(dir, keepFile) {
87
+ try {
88
+ for (const name of readdirSync(dir)) {
89
+ if (name === 'manifest.json' || name === keepFile) continue;
90
+ if (name.startsWith('mixdog-graph')) {
91
+ try { rmSync(join(dir, name), { force: true }); } catch { /* best-effort */ }
92
+ }
93
+ }
94
+ } catch { /* dir may not exist yet */ }
95
+ }
96
+
97
+ // Sync, network-free lookup of an already-cached binary. Used by the sync
98
+ // _graphBinaryPath() resolver before falling back to an async fetch.
99
+ export function findCachedGraphBinary(dataDir) {
100
+ try {
101
+ const dir = graphBinDir(dataDir);
102
+ const hit = readdirSync(dir).find(
103
+ (n) => n.startsWith('mixdog-graph') && !n.endsWith('.json') && !n.includes('.tmp-'),
104
+ );
105
+ return hit ? join(dir, hit) : null;
106
+ } catch {
107
+ return null;
108
+ }
109
+ }
110
+
111
+ let _inflight = null;
112
+
113
+ export function ensureGraphBinary(dataDir) {
114
+ // Single-flight: concurrent callers (prewarm + cache-miss) share one download.
115
+ if (_inflight) return _inflight;
116
+ _inflight = (async () => {
117
+ const manifest = await loadManifest(dataDir);
118
+ const pkey = platformKey();
119
+ const asset = manifest.assets?.[pkey];
120
+ if (!asset || !asset.url || !asset.sha256) {
121
+ throw new Error(`[graph-fetcher] no prebuilt mixdog-graph asset for platform ${pkey}`);
122
+ }
123
+ const version = String(manifest.version || '0');
124
+ const dir = graphBinDir(dataDir);
125
+ mkdirSync(dir, { recursive: true });
126
+ const fileName = `mixdog-graph-${version}${binSuffix()}`;
127
+ const destPath = join(dir, fileName);
128
+ if (existsSync(destPath)) {
129
+ try { if (await sha256File(destPath) === asset.sha256) return destPath; } catch { /* re-download */ }
130
+ }
131
+ const tmpPath = `${destPath}.tmp-${process.pid}-${Date.now()}`;
132
+ await downloadWithRetry(asset.url, tmpPath);
133
+ const actual = await sha256File(tmpPath);
134
+ if (actual !== asset.sha256) {
135
+ try { rmSync(tmpPath, { force: true }); } catch { /* best-effort */ }
136
+ throw new Error(`[graph-fetcher] sha256 mismatch for ${pkey}: expected ${asset.sha256}, got ${actual}`);
137
+ }
138
+ renameSync(tmpPath, destPath);
139
+ if (process.platform !== 'win32') { try { chmodSync(destPath, 0o755); } catch { /* best-effort */ } }
140
+ gcGraphBin(dir, fileName);
141
+ return destPath;
142
+ })().finally(() => { _inflight = null; });
143
+ return _inflight;
144
+ }
@@ -0,0 +1,26 @@
1
+ {
2
+ "version": "0.6.5",
3
+ "_comment": "Rewritten by .github/workflows/graph-release.yml on each tagged release. assets maps platformKey (process.platform-process.arch, e.g. win32-x64, linux-x64, darwin-arm64) to { url, sha256 } of the mixdog-graph binary on the GitHub release. A local cargo build under native/mixdog-graph/target/release always takes precedence at runtime. (v0.5.236 entries were filled manually after CI's commit step hit detached HEAD; the workflow now checks out ref: main so future releases self-update.)",
4
+ "assets": {
5
+ "darwin-arm64": {
6
+ "url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-darwin-arm64",
7
+ "sha256": "7016c273a07d19ca9e2f56e8fa7f273fdd40fc41bdc7fef206bf23e31a21a736"
8
+ },
9
+ "darwin-x64": {
10
+ "url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-darwin-x64",
11
+ "sha256": "d076e97da4420f49a6c726bc088a3321e2e7f6a9bfb32d39162c8c53045cfcdb"
12
+ },
13
+ "linux-arm64": {
14
+ "url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-linux-arm64",
15
+ "sha256": "74754562b3c080868738c032c5b6e0e13bc53d7a5277002176b036f8d6681f39"
16
+ },
17
+ "linux-x64": {
18
+ "url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-linux-x64",
19
+ "sha256": "0d8e8bbdd49b18746ed3f972fc3719731a1143ee03ac9e6d86586788b0b431f8"
20
+ },
21
+ "win32-x64": {
22
+ "url": "https://github.com/trib-plugin/mixdog/releases/download/v0.6.5/mixdog-graph-win32-x64.exe",
23
+ "sha256": "1a671558e5a5f13c7429ff9987a46ad72a71e52241e200d2da820a13d7cbdae7"
24
+ }
25
+ }
26
+ }
@@ -0,0 +1,204 @@
1
+ // Host-terminal input injection.
2
+ //
3
+ // Exposes `inject_input` as an MCP tool that types text into the *host*
4
+ // terminal currently running Claude Code (and therefore this MCP server,
5
+ // which Claude Code spawned as a child).
6
+ //
7
+ // Strategy:
8
+ // 1. Walk the parent-process chain starting at this Node process.
9
+ // 2. Find the first ancestor whose image name identifies a supported
10
+ // terminal host (currently: powershell / pwsh).
11
+ // 3. Dispatch via a per-host handler map. Adding cmd / Windows Terminal /
12
+ // VS Code / Claude Desktop later means dropping a new entry into
13
+ // `HOST_HANDLERS` — no branching surgery in the dispatcher.
14
+ //
15
+ // The PowerShell handler reuses the proven helper script
16
+ // `scripts/inject-input.ps1` (AttachConsole + CreateFileW("CONIN$") +
17
+ // WriteConsoleInputW, with the stale-STD_INPUT_HANDLE workaround). We
18
+ // shell out to the script rather than reimplementing the FFI in Node so
19
+ // the C# console-input typedef stays in one place.
20
+ //
21
+ // Returned JSON shape: { ok, host, pid, exitCode, stderr? }.
22
+
23
+ import { spawnSync, execFileSync } from 'node:child_process'
24
+ import { writeFileSync, mkdtempSync, rmSync } from 'node:fs'
25
+ import { join, dirname } from 'node:path'
26
+ import { tmpdir } from 'node:os'
27
+ import { fileURLToPath } from 'node:url'
28
+
29
+ const __dirname = dirname(fileURLToPath(import.meta.url))
30
+ // trib-plugin/src/agent/orchestrator/tools/host-input.mjs
31
+ // __dirname = .../trib-plugin/src/agent/orchestrator/tools
32
+ // → trib-plugin requires four `..` (orchestrator → agent → src → root).
33
+ const PLUGIN_ROOT = join(__dirname, '..', '..', '..', '..')
34
+ const PS_HELPER = join(PLUGIN_ROOT, 'scripts', 'inject-input.ps1')
35
+
36
+ // ── Parent-chain walker (Windows) ─────────────────────────────────────
37
+ //
38
+ // Uses one wmic / Get-CimInstance call to read the full Win32_Process
39
+ // table once, then walks PPID links in-process. Keeps the cost to one
40
+ // process spawn regardless of chain depth.
41
+ function readProcessTable() {
42
+ // Prefer PowerShell + Get-CimInstance — wmic.exe is deprecated and
43
+ // missing on newer Windows builds.
44
+ try {
45
+ const out = execFileSync(
46
+ 'powershell.exe',
47
+ [
48
+ '-NoProfile', '-NonInteractive', '-Command',
49
+ "Get-CimInstance Win32_Process | Select-Object ProcessId,ParentProcessId,Name | ConvertTo-Json -Compress",
50
+ ],
51
+ { encoding: 'utf8', windowsHide: true, timeout: 8000, maxBuffer: 16 * 1024 * 1024 },
52
+ )
53
+ const parsed = JSON.parse(out)
54
+ const rows = Array.isArray(parsed) ? parsed : [parsed]
55
+ const byPid = new Map()
56
+ for (const row of rows) {
57
+ if (!row || row.ProcessId == null) continue
58
+ byPid.set(Number(row.ProcessId), {
59
+ pid: Number(row.ProcessId),
60
+ ppid: Number(row.ParentProcessId ?? 0),
61
+ name: String(row.Name ?? ''),
62
+ })
63
+ }
64
+ return byPid
65
+ } catch (e) {
66
+ throw new Error(`failed to read process table: ${e?.message || e}`)
67
+ }
68
+ }
69
+
70
+ // Image-name → host-tag map. Lower-case match.
71
+ const HOST_BY_IMAGE = {
72
+ 'pwsh.exe': 'powershell',
73
+ 'powershell.exe': 'powershell',
74
+ // Future:
75
+ // 'cmd.exe': 'cmd',
76
+ // 'windowsterminal.exe': 'windows-terminal',
77
+ // 'code.exe': 'vscode-terminal',
78
+ // 'claude.exe': 'claude-desktop',
79
+ }
80
+
81
+ function classifyHost(imageName) {
82
+ if (!imageName) return null
83
+ return HOST_BY_IMAGE[imageName.toLowerCase()] ?? null
84
+ }
85
+
86
+ // Walk parent chain from the given pid, returning the first ancestor whose
87
+ // image maps to a known host. Bounded to 32 hops and skips PID 0 / orphan.
88
+ function findHostAncestor(startPid) {
89
+ const table = readProcessTable()
90
+ let cur = table.get(Number(startPid))
91
+ let hops = 0
92
+ const trail = []
93
+ while (cur && hops < 32) {
94
+ trail.push({ pid: cur.pid, name: cur.name })
95
+ const tag = classifyHost(cur.name)
96
+ if (tag) return { host: tag, pid: cur.pid, name: cur.name, trail }
97
+ if (!cur.ppid || cur.ppid === cur.pid) break
98
+ cur = table.get(cur.ppid)
99
+ hops += 1
100
+ }
101
+ return { host: null, pid: null, name: null, trail }
102
+ }
103
+
104
+ // ── Per-host handlers ─────────────────────────────────────────────────
105
+
106
+ function handlePowerShell(text, pid) {
107
+ // Pass the payload via a temp file rather than a command-line arg so
108
+ // arbitrary characters (quotes, backticks, $, newlines) survive without
109
+ // any shell or PowerShell parsing pitfalls. The .ps1 helper is invoked
110
+ // directly via powershell.exe -File, no -Command interpolation.
111
+ const tmpDir = mkdtempSync(join(tmpdir(), 'mixdog-inject-'))
112
+ const payloadPath = join(tmpDir, 'payload.txt')
113
+ let result
114
+ try {
115
+ writeFileSync(payloadPath, String(text), 'utf8')
116
+ // The helper script's -Text param expects a single string. Read the
117
+ // payload file inside a tiny -Command shim so we don't have to touch
118
+ // the existing helper. -Raw preserves embedded newlines and trailing
119
+ // whitespace exactly as the caller wrote them.
120
+ const psCommand =
121
+ `$ErrorActionPreference='Stop'; ` +
122
+ `$txt = Get-Content -LiteralPath ${psQuote(payloadPath)} -Raw -Encoding UTF8; ` +
123
+ `& ${psQuote(PS_HELPER)} -TargetPid ${Number(pid)} -Text $txt`
124
+ result = spawnSync(
125
+ 'powershell.exe',
126
+ ['-NoProfile', '-NonInteractive', '-ExecutionPolicy', 'Bypass', '-Command', psCommand],
127
+ { encoding: 'utf8', windowsHide: true, timeout: 15000 },
128
+ )
129
+ } finally {
130
+ try { rmSync(tmpDir, { recursive: true, force: true }) } catch {}
131
+ }
132
+ const exitCode = result.status ?? -1
133
+ const stderr = (result.stderr || '').trim()
134
+ const ok = exitCode === 0
135
+ const out = { ok, exitCode }
136
+ if (!ok && stderr) out.stderr = stderr
137
+ return out
138
+ }
139
+
140
+ // PowerShell single-quoted literal: escape ' as ''. Wraps the whole value
141
+ // in single quotes so $ / backtick / parentheses are taken literally.
142
+ function psQuote(s) {
143
+ return `'${String(s).replace(/'/g, "''")}'`
144
+ }
145
+
146
+ const HOST_HANDLERS = {
147
+ powershell: handlePowerShell,
148
+ // cmd: handleCmd,
149
+ // 'windows-terminal': handleWT,
150
+ // 'vscode-terminal': handleVSCode,
151
+ // 'claude-desktop': handleClaudeDesktop,
152
+ }
153
+
154
+ // ── Public entry ──────────────────────────────────────────────────────
155
+
156
+ export const HOST_INPUT_TOOL_DEFS = [
157
+ {
158
+ name: 'inject_input',
159
+ title: 'Inject Input',
160
+ annotations: { title: 'Inject Input', readOnlyHint: false, destructiveHint: true, idempotentHint: false, openWorldHint: true, compressible: false },
161
+ description: 'Inject text into the host session as if the user typed it at the prompt. Requires text.',
162
+ inputSchema: {
163
+ type: 'object',
164
+ properties: {
165
+ text: { type: 'string', minLength: 1 },
166
+ },
167
+ required: ['text'],
168
+ additionalProperties: false,
169
+ },
170
+ },
171
+ ]
172
+
173
+ export async function executeHostInputTool(name, args /*, cwd */) {
174
+ if (name !== 'inject_input') throw new Error(`Unknown host-input tool: ${name}`)
175
+ const text = args?.text
176
+ if (typeof text !== 'string' || text.length === 0) {
177
+ throw new Error('inject_input: `text` (string, non-empty) is required')
178
+ }
179
+ if (process.platform !== 'win32') {
180
+ throw new Error(`inject_input: only supported on Windows (got platform=${process.platform})`)
181
+ }
182
+
183
+ // Always submit: append a trailing newline if the caller didn't.
184
+ const payload = text.endsWith('\n') ? text : text + '\n'
185
+
186
+ const { host, pid, name: imageName, trail } = findHostAncestor(process.pid)
187
+ if (!host) {
188
+ const trailStr = trail.map(t => `${t.pid}:${t.name}`).join(' → ')
189
+ throw new Error(`inject_input: no supported terminal host found in parent chain (${trailStr || '<empty>'})`)
190
+ }
191
+ const handler = HOST_HANDLERS[host]
192
+ if (typeof handler !== 'function') {
193
+ throw new Error(`inject_input: host "${host}" not supported yet (resolved image=${imageName} pid=${pid})`)
194
+ }
195
+
196
+ const result = handler(payload, pid)
197
+ return JSON.stringify({
198
+ ok: !!result.ok,
199
+ host,
200
+ pid,
201
+ exitCode: result.exitCode,
202
+ ...(result.stderr ? { stderr: result.stderr } : {}),
203
+ })
204
+ }
@@ -0,0 +1,67 @@
1
+ import * as nodeBuffer from 'node:buffer';
2
+ import { readFileSync } from 'node:fs';
3
+ import { readFile as readFileAsync } from 'node:fs/promises';
4
+ import { decodeRawBufferForSnapshotCheck } from './builtin/snapshot-helpers.mjs';
5
+
6
+ function keyForPath(fullPath) {
7
+ const text = String(fullPath || '');
8
+ return process.platform === 'win32' ? text.toLowerCase() : text;
9
+ }
10
+
11
+ export function isValidUtf8Buffer(buf) {
12
+ if (typeof nodeBuffer.isUtf8 === 'function') return nodeBuffer.isUtf8(buf);
13
+ if (typeof Buffer.isUtf8 === 'function') return Buffer.isUtf8(buf);
14
+ return Buffer.from(buf.toString('utf-8'), 'utf-8').equals(buf);
15
+ }
16
+
17
+ export function createMutationContentCache() {
18
+ const entries = new Map();
19
+
20
+ function getOrCreate(fullPath) {
21
+ const key = keyForPath(fullPath);
22
+ let entry = entries.get(key);
23
+ if (!entry) {
24
+ entry = { fullPath, rawBuf: null, content: null };
25
+ entries.set(key, entry);
26
+ }
27
+ return entry;
28
+ }
29
+
30
+ function seedBuffer(fullPath, rawBuf) {
31
+ const entry = getOrCreate(fullPath);
32
+ entry.rawBuf = rawBuf;
33
+ entry.content = null;
34
+ return entry;
35
+ }
36
+
37
+ function readBufferSync(fullPath) {
38
+ const entry = getOrCreate(fullPath);
39
+ if (!Buffer.isBuffer(entry.rawBuf)) entry.rawBuf = readFileSync(fullPath);
40
+ return entry.rawBuf;
41
+ }
42
+
43
+ async function readBuffer(fullPath) {
44
+ const entry = getOrCreate(fullPath);
45
+ if (!Buffer.isBuffer(entry.rawBuf)) entry.rawBuf = await readFileAsync(fullPath);
46
+ return entry.rawBuf;
47
+ }
48
+
49
+ function readTextSync(fullPath) {
50
+ const entry = getOrCreate(fullPath);
51
+ if (typeof entry.content !== 'string') {
52
+ entry.content = decodeRawBufferForSnapshotCheck(readBufferSync(fullPath));
53
+ }
54
+ return entry.content;
55
+ }
56
+
57
+ function getEntry(fullPath) {
58
+ return entries.get(keyForPath(fullPath)) || null;
59
+ }
60
+
61
+ function clear(fullPath = null) {
62
+ if (fullPath) entries.delete(keyForPath(fullPath));
63
+ else entries.clear();
64
+ }
65
+
66
+ return { seedBuffer, readBuffer, readBufferSync, readTextSync, getEntry, clear };
67
+ }
@@ -0,0 +1,75 @@
1
+ // NOTE: post-edit V4A fallback was removed.
2
+ //
3
+ // The removed V4A edit reroute used to emit every `old_string` line as a
4
+ // `-` with zero context_before/after,
5
+ // which meant the V4A matcher had to find the same byte slice the exact
6
+ // edit just failed on. Code 8 ("old_string not found") is by definition a
7
+ // byte/context mismatch (tab-vs-space drift, stale snapshot, fold-tier
8
+ // miss), so the V4A engine fails for the same reason and the user sees a
9
+ // noisy double-error block. The exact-unique byte path now handles large
10
+ // chunks natively, so the V4A pre-route is dead and has been removed.
11
+
12
+ export function planEditMutationRoute() {
13
+ // Large-chunk V4A reroute removed: an EXACT-UNIQUE byte match is safe
14
+ // at any size, so the edit-exact tier handles >=30-line chunks
15
+ // natively (size gate now guards only the fold/fuzzy fallback). The
16
+ // V4A pre-route used to fail anchor-not-found on bytes the exact tier
17
+ // WOULD match, then priorResult short-circuited and surfaced a noisy
18
+ // double-error block. Always route through edit-exact.
19
+ return { engine: 'edit-exact', reason: 'default' };
20
+ }
21
+
22
+ export function isMutationPlanRoutable(plan) {
23
+ return plan?.engine === 'v4a-patch' && plan?.patchArgs?.patch;
24
+ }
25
+
26
+ export function formatMutationRouteLine(plan = {}, extras = {}) {
27
+ const source = plan.sourceTool || extras.sourceTool || 'unknown';
28
+ const engine = plan.engine || extras.engine || 'unknown';
29
+ const reason = plan.reason || extras.reason || 'unknown';
30
+ const suffix = Object.entries(extras)
31
+ .filter(([key, value]) => key !== 'sourceTool' && key !== 'engine' && key !== 'reason' && value !== undefined && value !== null && value !== '')
32
+ .map(([key, value]) => `${key}=${String(value).replace(/\s+/g, '-')}`)
33
+ .join(' ');
34
+ return `mutation_route: source=${source} engine=${engine} reason=${reason}${suffix ? ` ${suffix}` : ''}`;
35
+ }
36
+
37
+ export function wrapMutationRouteOutput(text, plan = {}, extras = {}) {
38
+ const body = String(text ?? '');
39
+ if (/^mutation_route:/i.test(body.trimStart())) return body;
40
+ return `${formatMutationRouteLine(plan, extras)}\n${body}`;
41
+ }
42
+
43
+ export async function executeMutationPlan(plan, context = {}) {
44
+ if (!isMutationPlanRoutable(plan)) return null;
45
+ const { executePatchTool } = await import('./patch.mjs');
46
+ let out;
47
+ try {
48
+ out = await executePatchTool('apply_patch', {
49
+ base_path: context.workDir,
50
+ ...plan.patchArgs,
51
+ }, context.workDir, {
52
+ ...context.options,
53
+ readStateScope: context.readStateScope,
54
+ sessionId: context.options?.sessionId,
55
+ mutationPlan: plan,
56
+ });
57
+ } catch (err) {
58
+ return { ok: false, text: err?.message || String(err), plan };
59
+ }
60
+ const text = String(out ?? '');
61
+ if (/^Error:/i.test(text.trimStart())) {
62
+ return { ok: false, text: out, plan };
63
+ }
64
+ return {
65
+ ok: true,
66
+ text: wrapMutationRouteOutput(text, plan),
67
+ plan,
68
+ };
69
+ }
70
+
71
+ export function appendMutationPlanFailure(originalResult, plannedResult) {
72
+ if (!plannedResult || plannedResult.ok) return originalResult;
73
+ const plan = plannedResult.plan || {};
74
+ return `${originalResult}\nmutation_route: source=${plan.sourceTool || 'unknown'} engine=${plan.engine || 'unknown'} reason=${plan.reason || 'unknown'} failed\n${plannedResult.text}`;
75
+ }
@@ -0,0 +1,48 @@
1
+ export function parseJsonNextCalls(text) {
2
+ const input = String(text ?? '');
3
+ if (!input.includes('next_call:')) return [];
4
+ const out = [];
5
+ const re = /next_call:\s*([A-Za-z_][\w]*)\(/g;
6
+ for (const match of input.matchAll(re)) {
7
+ const parsed = parseNextCallArgs(input, match.index + match[0].length);
8
+ if (!parsed) continue;
9
+ out.push({ tool: match[1], args: parsed.args, start: match.index, end: parsed.end });
10
+ }
11
+ return out;
12
+ }
13
+
14
+ export function countJsonNextCalls(text) {
15
+ return parseJsonNextCalls(text).length;
16
+ }
17
+
18
+ function parseNextCallArgs(text, start) {
19
+ let i = start;
20
+ while (i < text.length && /\s/.test(text[i])) i++;
21
+ if (text[i] !== '{') return null;
22
+ let depth = 0;
23
+ let inString = false;
24
+ let escaped = false;
25
+ for (; i < text.length; i++) {
26
+ const ch = text[i];
27
+ if (inString) {
28
+ if (escaped) escaped = false;
29
+ else if (ch === '\\') escaped = true;
30
+ else if (ch === '"') inString = false;
31
+ continue;
32
+ }
33
+ if (ch === '"') inString = true;
34
+ else if (ch === '{') depth++;
35
+ else if (ch === '}') {
36
+ depth--;
37
+ if (depth === 0) {
38
+ const raw = text.slice(start, i + 1).trim();
39
+ let j = i + 1;
40
+ while (j < text.length && /\s/.test(text[j])) j++;
41
+ if (text[j] !== ')') return null;
42
+ try { return { args: JSON.parse(raw), end: j + 1 }; }
43
+ catch { return null; }
44
+ }
45
+ }
46
+ }
47
+ return null;
48
+ }