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,765 @@
1
+ import { existsSync, statSync, watch, openSync, readSync, closeSync } from "fs";
2
+ import { basename } from "path";
3
+ import { createHash } from "crypto";
4
+ import { formatForDiscord, chunk, safeCodeBlock } from "./format.mjs";
5
+ import { dropTrace, _dtPreview } from "./drop-trace.mjs";
6
+ import {
7
+ cwdToProjectSlug,
8
+ discoverCurrentClaudeSession,
9
+ listInteractiveClaudeSessions,
10
+ getLatestInteractiveClaudeSession
11
+ } from "./session-discovery.mjs";
12
+ import {
13
+ findLatestTranscriptByMtime,
14
+ sameResolvedPath,
15
+ detectCurrentSessionTranscript,
16
+ discoverSessionBoundTranscript
17
+ } from "./transcript-discovery.mjs";
18
+ import {
19
+ SKIP_TEXTS,
20
+ HIDDEN_TOOLS,
21
+ isRecallMemory,
22
+ isMemoryFile,
23
+ buildDedupKey,
24
+ buildToolLine
25
+ } from "./tool-format.mjs";
26
+
27
+ class OutputForwarder {
28
+ ownerGetter = null;
29
+ setOwnerGetter(fn) {
30
+ this.ownerGetter = fn;
31
+ }
32
+ _isOwner() {
33
+ if (!this.ownerGetter) return true;
34
+ // Fail closed: a probe exception must NOT be treated as ownership.
35
+ // Forwarding transcript output from a non-owner duplicates Discord
36
+ // sends from the previous owner process.
37
+ try { return !!this.ownerGetter(); } catch { return false; }
38
+ }
39
+ constructor(cb, statusState) {
40
+ this.cb = cb;
41
+ this.statusState = statusState;
42
+ this._persistTimer = null;
43
+ this._pendingPersistData = null;
44
+ // Best-effort final flush on process exit. The handler is sync (writeFileSync
45
+ // + fsyncSync inside updateState), so it actually lands on graceful exit.
46
+ // SIGTERM/SIGINT shutdowns that bypass 'exit' are covered by the timer
47
+ // unref-ing so the event loop drains it before close.
48
+ process.on('exit', () => { try { this._flushPersistState(); } catch {} });
49
+ }
50
+ lastHash = "";
51
+ sentCount = 0;
52
+ transcriptPath = "";
53
+ channelId = "";
54
+ userMessageId = "";
55
+ emoji = "";
56
+ lastFileSize = 0;
57
+ readFileSize = 0;
58
+ watchingPath = "";
59
+ watcher = null;
60
+ idleTimer = null;
61
+ onIdleCallback = null;
62
+ inExplorerSequence = false;
63
+ inRecallSequence = false;
64
+ hasSeenAssistant = false;
65
+ sending = false;
66
+ sendRetryTimer = null;
67
+ // Priority-lane queues (Tier3 split).
68
+ // finalLane — text items (transcript text, final answer segments)
69
+ // streamLane — tool-log / progress items
70
+ // drainQueue() picks from finalLane first so final answers are never blocked
71
+ // behind a batch of tool-log messages.
72
+ finalLane = [];
73
+ streamLane = [];
74
+ // Cap streamLane length under sustained backpressure (Discord 429 storm,
75
+ // network outage). When over the cap, the oldest text item is merged
76
+ // into the next pending text payload so we don't grow unbounded but
77
+ // also don't lose content.
78
+ static SEND_QUEUE_MAX = 200;
79
+ // Persisted final-flush ledger so a forwarder restart can resume final
80
+ // forwarding instead of giving up after 5 short retries.
81
+ pendingFinalFlush = false;
82
+ mainSessionId = "";
83
+ watchDebounce = null;
84
+ turnTextBuffer = "";
85
+ hasBinding() {
86
+ return !!this.transcriptPath;
87
+ }
88
+ /** Set context for current turn (called on user message) */
89
+ setContext(channelId, transcriptPath, options = {}) {
90
+ this.channelId = channelId;
91
+ if (!transcriptPath) return;
92
+ if (this.transcriptPath && !existsSync(this.transcriptPath)) {
93
+ const relocated = detectCurrentSessionTranscript()?.transcriptPath ?? findLatestTranscriptByMtime();
94
+ if (relocated) {
95
+ transcriptPath = relocated;
96
+ }
97
+ }
98
+ if (this.transcriptPath !== transcriptPath) {
99
+ this.closeWatcher();
100
+ dropTrace("context.transcriptPath.change", { sessionId: this.mainSessionId || "(none)", oldPath: this.transcriptPath || "(none)", newPath: transcriptPath });
101
+ this.transcriptPath = transcriptPath;
102
+ this.mainSessionId = "";
103
+ this.sentCount = 0;
104
+ this.lastHash = "";
105
+ this.turnTextBuffer = "";
106
+ }
107
+ try {
108
+ const stat = existsSync(this.transcriptPath) ? statSync(this.transcriptPath) : null;
109
+ const currentSize = stat?.size ?? 0;
110
+ let fileSize;
111
+ if (options.replayFromStart) {
112
+ fileSize = 0;
113
+ } else if (options.catchUpFromPersisted) {
114
+ const persisted = this.statusState?.read?.();
115
+ const persistedSize = typeof persisted?.lastFileSize === "number" ? persisted.lastFileSize : -1;
116
+ const sameTranscript = persisted?.transcriptPath &&
117
+ sameResolvedPath(persisted.transcriptPath, this.transcriptPath);
118
+ fileSize = (sameTranscript && persistedSize >= 0)
119
+ ? Math.min(Math.max(persistedSize, 0), currentSize)
120
+ : currentSize;
121
+ } else {
122
+ fileSize = currentSize;
123
+ }
124
+ this.lastFileSize = fileSize;
125
+ this.readFileSize = fileSize;
126
+ } catch {
127
+ this.lastFileSize = 0;
128
+ this.readFileSize = 0;
129
+ }
130
+ }
131
+ /** Reset counters for new turn */
132
+ reset() {
133
+ this.sentCount = 0;
134
+ this.lastHash = "";
135
+ this.inExplorerSequence = false;
136
+ this.inRecallSequence = false;
137
+ this.hasSeenAssistant = false;
138
+ this.turnTextBuffer = "";
139
+ if (this.idleTimer) {
140
+ clearTimeout(this.idleTimer);
141
+ this.idleTimer = null;
142
+ }
143
+ }
144
+ /** Read new bytes from transcript file since readFileSize */
145
+ readNewLines() {
146
+ if (!this.transcriptPath || !existsSync(this.transcriptPath)) {
147
+ return { lines: [], nextFileSize: this.readFileSize };
148
+ }
149
+ let fd = null;
150
+ try {
151
+ const stat = this._pendingStat ?? statSync(this.transcriptPath);
152
+ this._pendingStat = null;
153
+ if (stat.size <= this.readFileSize) {
154
+ return { lines: [], nextFileSize: this.readFileSize };
155
+ }
156
+ const startOffset = this.readFileSize;
157
+ fd = openSync(this.transcriptPath, "r");
158
+ const buf = Buffer.alloc(stat.size - startOffset);
159
+ readSync(fd, buf, 0, buf.length, startOffset);
160
+ // Only advance readFileSize to the last newline boundary. A trailing
161
+ // partial line (writer still appending) must be re-read next tick;
162
+ // otherwise the parse-failed slice is silently consumed forever.
163
+ const lastNl = buf.lastIndexOf(0x0a);
164
+ const consumed = lastNl >= 0 ? lastNl + 1 : 0;
165
+ const nextFileSize = startOffset + consumed;
166
+ this.readFileSize = nextFileSize;
167
+ const text = consumed > 0 ? buf.slice(0, consumed).toString("utf8") : "";
168
+ return {
169
+ lines: text ? text.split("\n").filter((l) => l.trim()) : [],
170
+ nextFileSize
171
+ };
172
+ } catch {
173
+ return { lines: [], nextFileSize: this.readFileSize };
174
+ } finally {
175
+ if (fd != null) {
176
+ closeSync(fd);
177
+ }
178
+ }
179
+ }
180
+ /** Track last tool_use name and file path for matching with tool_result */
181
+ lastToolName = "";
182
+ lastToolFilePath = "";
183
+ /** Extract new assistant text + tool logs from transcript since readFileSize */
184
+ extractNewText() {
185
+ const { lines: newLines, nextFileSize } = this.readNewLines();
186
+ let newText = "";
187
+ for (const l of newLines) {
188
+ try {
189
+ const entry = JSON.parse(l);
190
+ if (!entry.isSidechain && entry.sessionId && !this.mainSessionId) {
191
+ this.mainSessionId = entry.sessionId;
192
+ }
193
+ if (entry.isSidechain) continue;
194
+ if (this.mainSessionId && entry.sessionId && entry.sessionId !== this.mainSessionId) continue;
195
+ if (entry.type === "user" && entry.message?.content?.some((c) => c.type === "tool_result")) {
196
+ if (OutputForwarder.isRecallMemory(this.lastToolName)) {
197
+ continue;
198
+ }
199
+ if (this.lastToolName === "Edit" && entry.toolUseResult && !OutputForwarder.isMemoryFile(this.lastToolFilePath)) {
200
+ const old = entry.toolUseResult.oldString || "";
201
+ const nw = entry.toolUseResult.newString || "";
202
+ if (old || nw) {
203
+ const diffLines = [];
204
+ for (const l2 of old.split("\n")) diffLines.push("- " + l2);
205
+ for (const l2 of nw.split("\n")) diffLines.push("+ " + l2);
206
+ const shown = diffLines.slice(0, 15);
207
+ let diffContent = shown.join("\n");
208
+ if (diffLines.length > 15) diffContent += "\n... +" + (diffLines.length - 15) + " lines";
209
+ const block = safeCodeBlock(diffContent, "diff");
210
+ newText += block + "\n";
211
+ }
212
+ }
213
+ continue;
214
+ }
215
+ if (entry.type === "assistant" && entry.message?.content) {
216
+ this.hasSeenAssistant = true;
217
+ const SEARCH_TOOLS = /* @__PURE__ */ new Set(["Read", "Grep", "Glob"]);
218
+ const parts = [];
219
+ for (const c of entry.message.content) {
220
+ if (c.type === "text" && c.text?.trim()) {
221
+ this.inExplorerSequence = false;
222
+ this.inRecallSequence = false;
223
+ let cleaned = OutputForwarder.stripForeignWorkerChannels(c.text.trim());
224
+ cleaned = cleaned.replace(/<(memory-context|system-reminder|event)\b[^>]*>[\s\S]*?<\/\1>/gi, "").trim();
225
+ if (cleaned) parts.push(cleaned);
226
+ } else if (c.type === "tool_use") {
227
+ this.lastToolName = c.name || "";
228
+ this.lastToolFilePath = c.input?.file_path || "";
229
+ if (OutputForwarder.isHidden(c.name)) continue;
230
+ if (SEARCH_TOOLS.has(c.name)) {
231
+ if (!this.inExplorerSequence) {
232
+ this.inExplorerSequence = true;
233
+ let target = "";
234
+ if (c.name === "Read") target = c.input?.file_path ? basename(c.input.file_path) : "";
235
+ else if (c.name === "Grep") target = '"' + (c.input?.pattern || "").substring(0, 25) + '"';
236
+ else if (c.name === "Glob") target = (c.input?.pattern || "").substring(0, 25);
237
+ if (parts.length > 0) parts.push("");
238
+ parts.push("\u25CF **Explorer** (" + (target || c.name) + ")");
239
+ }
240
+ continue;
241
+ }
242
+ if (OutputForwarder.isRecallMemory(c.name)) {
243
+ if (!this.inRecallSequence) {
244
+ this.inRecallSequence = true;
245
+ if (parts.length > 0) parts.push("");
246
+ parts.push("\u25CF **recall_memory**");
247
+ }
248
+ continue;
249
+ }
250
+ this.inExplorerSequence = false;
251
+ this.inRecallSequence = false;
252
+ const toolLine = OutputForwarder.buildToolLine(c.name, c.input);
253
+ if (toolLine) {
254
+ if (parts.length > 0) parts.push("");
255
+ parts.push(toolLine);
256
+ }
257
+ }
258
+ }
259
+ if (parts.length) newText += parts.join("\n") + "\n";
260
+ }
261
+ } catch {
262
+ }
263
+ }
264
+ return { text: newText.trim(), nextFileSize };
265
+ }
266
+ // ── Single-send gate ──────────────────────────────────────────────
267
+ // All Discord sends pass through sendOnce() so duplicate concurrent sends are avoided.
268
+ // Texts that should never be forwarded to Discord (Claude's internal status lines)
269
+ static SKIP_TEXTS = SKIP_TEXTS;
270
+ commitReadProgress(nextFileSize) {
271
+ if (nextFileSize <= this.lastFileSize) return;
272
+ this.lastFileSize = nextFileSize;
273
+ this.persistState();
274
+ }
275
+ async deliverQueueItem(item) {
276
+ const targetChannelId = item.channelId ?? this.channelId;
277
+ if (!item.text || !targetChannelId) {
278
+ this.commitReadProgress(item.nextFileSize);
279
+ return;
280
+ }
281
+ if (!item.skipHashDedup && OutputForwarder.SKIP_TEXTS.has(item.text.trim())) {
282
+ this.commitReadProgress(item.nextFileSize);
283
+ return;
284
+ }
285
+ const formatted = item.preformatted ? item.text : formatForDiscord(item.text);
286
+ const hash = item.skipHashDedup
287
+ ? ""
288
+ : item.dedupKey
289
+ ? item.dedupKey
290
+ : createHash("md5").update(formatted).digest("hex");
291
+ if (!item.skipHashDedup && this.lastHash === hash) {
292
+ this.commitReadProgress(item.nextFileSize);
293
+ return;
294
+ }
295
+ const chunks = chunk(formatted, 2e3);
296
+ const _t0Send = Date.now();
297
+ // Resume from _nextChunkIdx if this item is being retried after a partial send.
298
+ // This avoids re-sending chunks that already landed successfully.
299
+ if (item._chunks === undefined) {
300
+ item._chunks = chunks;
301
+ item._nextChunkIdx = 0;
302
+ item._sendRetries = 0;
303
+ }
304
+ for (let _ci = item._nextChunkIdx; _ci < item._chunks.length; _ci++) {
305
+ const c = item._chunks[_ci];
306
+ try {
307
+ await this.cb.send(targetChannelId, c);
308
+ item._nextChunkIdx = _ci + 1;
309
+ dropTrace("discord.send.ok", null);
310
+ } catch (err) {
311
+ // Discord 429 or transient error — honour Retry-After then re-throw so
312
+ // drainQueue's retry loop calls deliverQueueItem again. Chunk progress
313
+ // is stored on item so we resume from the failed chunk, not chunk 0.
314
+ const status = err?.status ?? err?.code ?? err?.httpStatus;
315
+ const retryAfter = err?.retryAfter ?? err?.retry_after
316
+ ?? err?.headers?.["retry-after"] ?? err?.response?.headers?.["retry-after"];
317
+ dropTrace("discord.send.err", { channelId: this.channelId, chunkIndex: _ci, status, retryAfter: retryAfter ?? "(none)", err: String(err) });
318
+ item._sendRetries = (item._sendRetries || 0) + 1;
319
+ if (item._sendRetries >= 3) {
320
+ // Cap retries to avoid infinite duplicate loop — give up on this item
321
+ process.stderr.write(`[output-forwarder] chunk send exceeded 3 retries at chunk ${_ci}, dropping item\n`);
322
+ item._nextChunkIdx = item._chunks.length; // mark exhausted
323
+ return;
324
+ }
325
+ if (status === 429) {
326
+ if (retryAfter != null) {
327
+ const ms = Number(retryAfter) > 1000 ? Number(retryAfter) : Number(retryAfter) * 1000;
328
+ if (Number.isFinite(ms) && ms > 0) {
329
+ await new Promise((r) => setTimeout(r, Math.min(ms, 60_000)));
330
+ }
331
+ } else {
332
+ await new Promise((r) => setTimeout(r, 1000));
333
+ }
334
+ }
335
+ throw err;
336
+ }
337
+ }
338
+ if (!item.skipHashDedup) {
339
+ this.lastHash = hash;
340
+ }
341
+ const _bt = typeof item.bufferText === 'string' ? item.bufferText : '';
342
+ if (_bt.trim()) {
343
+ this.turnTextBuffer = this.turnTextBuffer ? `${this.turnTextBuffer}
344
+
345
+ ${_bt.trim()}` : _bt.trim();
346
+ }
347
+ this.sentCount += chunks.length;
348
+ this.commitReadProgress(item.nextFileSize);
349
+ }
350
+ scheduleRetry() {
351
+ if (this.sendRetryTimer) return;
352
+ dropTrace("drain.retry.schedule", { finalLen: this.finalLane.length, streamLen: this.streamLane.length });
353
+ this.sendRetryTimer = setTimeout(() => {
354
+ this.sendRetryTimer = null;
355
+ this._kickDrain();
356
+ }, 1e3);
357
+ }
358
+ /** Forward new assistant text to Discord. Returns true if text was sent. */
359
+ async forwardNewText() {
360
+ if (!this._isOwner()) return false;
361
+ if (!this.channelId) return false;
362
+ const { text: newText, nextFileSize } = this.extractNewText();
363
+ if (!newText) {
364
+ if (!this.sending && this.finalLane.length === 0 && this.streamLane.length === 0) {
365
+ this.commitReadProgress(nextFileSize);
366
+ }
367
+ return false;
368
+ }
369
+ // Coalesce back-to-back text items BEFORE drain/chunking so adjacent
370
+ // emits merge into one chunk pass rather than waiting for SEND_QUEUE_MAX.
371
+ // tool-log items stay separate (preformatted) so we only merge plain-text.
372
+ // Issue 1: if drain is in flight finalLane[0] is being delivered; coalescing
373
+ // into that slot causes a concurrent mutation race. Only coalesce into the
374
+ // tail of finalLane when it is not the item currently being drained.
375
+ // finalLane[0] is being drained when this.sending; coalesce into tail only
376
+ // when tail is not index 0 (i.e. length >= 2) or drain is not in flight.
377
+ const ftLen = this.finalLane.length;
378
+ const ftTail = (ftLen > 0 && !(this.sending && ftLen === 1))
379
+ ? this.finalLane[ftLen - 1] : null;
380
+ if (ftTail) {
381
+ ftTail.text = `${ftTail.text}\n\n${newText}`;
382
+ ftTail.bufferText = `${ftTail.bufferText}\n\n${newText}`;
383
+ ftTail.nextFileSize = nextFileSize;
384
+ this._kickDrain();
385
+ return true;
386
+ }
387
+ // Cap-and-merge backpressure guard on finalLane: when saturated, fold
388
+ // the new text into the trailing finalLane item.
389
+ if (this.finalLane.length >= OutputForwarder.SEND_QUEUE_MAX) {
390
+ // Skip index 0 while drain is in flight (same race guard as above).
391
+ const capEnd = this.sending ? 1 : 0;
392
+ for (let i = this.finalLane.length - 1; i >= capEnd; i--) {
393
+ const cap = this.finalLane[i];
394
+ cap.text = `${cap.text}\n\n${newText}`;
395
+ cap.bufferText = `${cap.bufferText}\n\n${newText}`;
396
+ cap.nextFileSize = nextFileSize;
397
+ this._kickDrain();
398
+ return true;
399
+ }
400
+ }
401
+ this.finalLane.push({
402
+ type: "text",
403
+ text: newText,
404
+ nextFileSize,
405
+ bufferText: newText
406
+ });
407
+ this._kickDrain();
408
+ return true;
409
+ }
410
+ /** Forward tool log line to Discord */
411
+ async forwardToolLog(toolLine, toolName, toolInput) {
412
+ if (!this._isOwner()) return;
413
+ if (!this.channelId) return;
414
+ // Issue 2: do NOT advance readFileSize here via stat. The stat'd size
415
+ // could jump past bytes that forwardNewText has not yet parsed, causing
416
+ // extractNewText/readNewLines to skip real content. readFileSize is
417
+ // advanced only inside readNewLines once bytes are actually consumed.
418
+ this.streamLane.push({
419
+ type: "toolLog",
420
+ text: toolLine,
421
+ nextFileSize: this.readFileSize,
422
+ preformatted: true,
423
+ dedupKey: OutputForwarder.buildDedupKey(toolName, toolInput)
424
+ });
425
+ this._kickDrain();
426
+ }
427
+ /** Centralised fire-and-forget drainQueue with rejection guard. */
428
+ _kickDrain() {
429
+ this.drainQueue().catch(err => process.stderr.write(`[output-forwarder] drainQueue rejected: ${err?.message || err}\n`));
430
+ }
431
+ /** Drain both priority lanes sequentially. finalLane drains first when non-empty. */
432
+ async drainQueue() {
433
+ if (this.sending) return;
434
+ this.sending = true;
435
+ try {
436
+ while (this.finalLane.length > 0 || this.streamLane.length > 0) {
437
+ // Pick from finalLane first; fall back to streamLane.
438
+ const fromFinal = this.finalLane.length > 0;
439
+ const lane = fromFinal ? this.finalLane : this.streamLane;
440
+ const item = lane[0];
441
+ try {
442
+ if (item.type === "text") {
443
+ await this.deliverQueueItem(item);
444
+ } else if (item.type === "toolLog") {
445
+ await this.processToolLog(item);
446
+ }
447
+ lane.shift();
448
+ } catch (err) {
449
+ process.stderr.write(`mixdog: send failed: ${err}
450
+ `);
451
+ dropTrace("drain.send.err", { finalLen: this.finalLane.length, streamLen: this.streamLane.length, lane: fromFinal ? "final" : "stream", itemType: item?.type, err: String(err) });
452
+ this.scheduleRetry();
453
+ break;
454
+ }
455
+ }
456
+ } finally {
457
+ this.sending = false;
458
+ }
459
+ }
460
+ /** Internal: process a single tool log send (extracted from old forwardToolLog) */
461
+ async processToolLog(item) {
462
+ if (this.userMessageId) {
463
+ const newEmoji = "\u{1F6E0}\uFE0F";
464
+ try {
465
+ if (this.emoji && this.emoji !== newEmoji) {
466
+ await this.cb.removeReaction(this.channelId, this.userMessageId, this.emoji);
467
+ }
468
+ await this.cb.react(this.channelId, this.userMessageId, newEmoji);
469
+ this.emoji = newEmoji;
470
+ } catch {
471
+ }
472
+ }
473
+ await this.deliverQueueItem(item);
474
+ }
475
+ /** Forward final text on session idle */
476
+ async forwardFinalText(retries = 0, pinnedChannelId = null) {
477
+ if (!this._isOwner()) return;
478
+ // Pin the target channel at call time so a rebind to a new turn's channel
479
+ // (which mutates this.channelId synchronously after this fire-and-forget
480
+ // call returns) cannot redirect the previous turn's final output to the
481
+ // wrong channel.
482
+ const channelId = pinnedChannelId ?? this.channelId;
483
+ if (!channelId) return;
484
+ if (this.sending || this.finalLane.length > 0 || this.streamLane.length > 0) {
485
+ // Mark a durable flush request so a process restart picks it up
486
+ // instead of dropping the final frame on the floor.
487
+ try {
488
+ this.pendingFinalFlush = true;
489
+ this.updateState((state) => { state.pendingFinalFlush = true; });
490
+ } catch {}
491
+ if (retries < 5) {
492
+ setTimeout(() => void this.forwardFinalText(retries + 1, channelId), 300);
493
+ } else {
494
+ dropTrace("drain.finalText.exhausted", { retries, finalLen: this.finalLane.length, streamLen: this.streamLane.length, sending: this.sending });
495
+ // Past the short-retry budget: schedule a longer-tail drain wait so
496
+ // the final frame still ships once the queue empties, instead of
497
+ // silently giving up.
498
+ const waitDrain = () => {
499
+ if (!this.sending && this.finalLane.length === 0 && this.streamLane.length === 0) {
500
+ void this.forwardFinalText(0, channelId);
501
+ return;
502
+ }
503
+ setTimeout(waitDrain, 1000);
504
+ };
505
+ setTimeout(waitDrain, 1000);
506
+ }
507
+ return;
508
+ }
509
+ this.sending = true;
510
+ try {
511
+ if (this.userMessageId && this.emoji) {
512
+ try {
513
+ await this.cb.removeReaction(channelId, this.userMessageId, this.emoji);
514
+ } catch {
515
+ }
516
+ }
517
+ const { text: newText, nextFileSize } = this.extractNewText();
518
+ if (newText) {
519
+ const finalItem = { type: "text", text: newText, nextFileSize, bufferText: newText, channelId };
520
+ try {
521
+ await this.deliverQueueItem(finalItem);
522
+ } catch (err) {
523
+ // Transient send failure: extractNewText already advanced the read
524
+ // cursor past these bytes, so dropping the item here would lose the
525
+ // final text. Requeue it (it retains per-chunk send progress) and let
526
+ // drainQueue retry instead of silently discarding. pendingFinalFlush
527
+ // stays set so a process restart can also resume the flush.
528
+ this.finalLane.push(finalItem);
529
+ this.scheduleRetry();
530
+ return;
531
+ }
532
+ } else {
533
+ this.commitReadProgress(nextFileSize);
534
+ }
535
+ if (this.turnTextBuffer.trim()) {
536
+ await this.cb.recordAssistantTurn?.({
537
+ channelId,
538
+ text: this.turnTextBuffer.trim(),
539
+ sessionId: this.mainSessionId || void 0
540
+ });
541
+ this.turnTextBuffer = "";
542
+ }
543
+ // Clear the durable flush marker only after delivery succeeded so a
544
+ // throw above leaves pendingFinalFlush=true and a process restart
545
+ // can resume the final forward instead of dropping the frame.
546
+ try {
547
+ this.pendingFinalFlush = false;
548
+ this.updateState((state) => { state.pendingFinalFlush = false; });
549
+ } catch {}
550
+ this.updateState((state) => {
551
+ state.sessionIdle = true;
552
+ });
553
+ } finally {
554
+ this.sending = false;
555
+ }
556
+ }
557
+ /** Hidden tools — skip both tool_use and tool_result */
558
+ static HIDDEN_TOOLS = HIDDEN_TOOLS;
559
+ /** Check if a tool name is recall_memory */
560
+ static isRecallMemory = isRecallMemory;
561
+ /** Check if a file path points to a memory file */
562
+ static isMemoryFile = isMemoryFile;
563
+ /** Check if a tool should be hidden */
564
+ static isHidden = (name) => {
565
+ // Read through OutputForwarder.HIDDEN_TOOLS so runtime reassignment of
566
+ // the static Set propagates into hidden-tool detection (restores the
567
+ // original indirect-reference semantics before the tool-format split).
568
+ // The non-set checks are inlined rather than delegated to the imported
569
+ // isHidden, because that helper would re-consult the module-local
570
+ // HIDDEN_TOOLS Set and ignore the OutputForwarder static.
571
+ if (OutputForwarder.HIDDEN_TOOLS.has(name)) return true;
572
+ if ((name.includes("plugin_mixdog") && !name.endsWith("recall_memory")) || name === "reply" || name === "react" || name === "edit_message" || name === "fetch" || name === "download_attachment") return true;
573
+ return false;
574
+ };
575
+ /** Concatenate text blocks from a transcript entry (user or assistant). */
576
+ static collectEntryText(entry) {
577
+ const parts = entry?.message?.content;
578
+ if (!Array.isArray(parts)) return "";
579
+ return parts
580
+ .filter((c) => c && c.type === "text" && typeof c.text === "string")
581
+ .map((c) => c.text)
582
+ .join("\n");
583
+ }
584
+ /**
585
+ * Remove mixdog worker/dispatch `<channel>` blocks whose `client_host_pid`
586
+ * does not match this owner's MIXDOG_OWNER_HOST_PID. Own-worker and legacy
587
+ * blocks (no client_host_pid) are left intact.
588
+ */
589
+ static stripForeignWorkerChannels(text) {
590
+ if (!text || !/<channel\b/i.test(text)) return text || "";
591
+ const owner = Number(process.env.MIXDOG_OWNER_HOST_PID);
592
+ const ownerOk = Number.isFinite(owner) && owner > 0;
593
+ return text.replace(/<channel\b([^>]*)>[\s\S]*?<\/channel>/gi, (block, openAttrs) => {
594
+ if (!/\bsource\s*=\s*["'][^"']*mixdog/i.test(openAttrs)) return block;
595
+ const m = openAttrs.match(/\bclient_host_pid\s*=\s*["']([^"']+)["']/i);
596
+ if (!m) return block;
597
+ if (!ownerOk) return block;
598
+ const origin = Number(m[1]);
599
+ if (!Number.isFinite(origin) || origin <= 0) return block;
600
+ if (origin === owner) return block;
601
+ return "";
602
+ });
603
+ }
604
+ /**
605
+ * Build a per-call dedup key for tool-log queue items.
606
+ * Uses the full (unsquished) tool args so that two Reads on distinct files
607
+ * sharing a basename, or two Grep/Glob calls sharing only a pattern prefix,
608
+ * do not collapse onto the same key and suppress the second send.
609
+ * Returns "" to fall back to md5(formatted) at delivery time.
610
+ */
611
+ static buildDedupKey = buildDedupKey;
612
+ /** Build a tool log line from the tool name and input. */
613
+ static buildToolLine = (name, input) => {
614
+ // Pass OutputForwarder.isHidden as the hidden-tool predicate so reassign-
615
+ // ment of the static propagates into tool-line construction (restores the
616
+ // original indirect-reference semantics before the tool-format split).
617
+ return buildToolLine(name, input, OutputForwarder.isHidden);
618
+ };
619
+ // ── File watch ─────────────────────────────────────────────────────
620
+ /** Set callback for idle detection (no new data for 5s after assistant entry) */
621
+ setOnIdle(cb) {
622
+ this.onIdleCallback = cb;
623
+ }
624
+ /** Start watching transcript file for changes (runs once, never stops) */
625
+ startWatch() {
626
+ if (!this.transcriptPath) return;
627
+ if (this.watchingPath === this.transcriptPath && this.watcher) {
628
+ dropTrace("watch.start.skip", { reason: "already_watching", path: this.watchingPath });
629
+ return;
630
+ }
631
+ this.closeWatcher();
632
+ this.watchingPath = this.transcriptPath;
633
+ dropTrace("watch.start.install", { path: this.watchingPath });
634
+ try {
635
+ this.watcher = watch(this.transcriptPath, () => this.scheduleWatchFlush());
636
+ this.watcher.on("error", (err) => {
637
+ dropTrace("watch.error", { path: this.watchingPath, err: String(err) });
638
+ this.closeWatcher();
639
+ });
640
+ this.watcher.on("close", () => {
641
+ dropTrace("watch.close.event", { path: this.watchingPath });
642
+ });
643
+ // Cover bytes written between the stat in setContext() and watch install.
644
+ this.scheduleWatchFlush();
645
+ } catch (e) {
646
+ dropTrace("watch.start.catch", { path: this.watchingPath, err: String(e) });
647
+ this.closeWatcher();
648
+ }
649
+ }
650
+ /** Stop watching the transcript file. Delegates to closeWatcher() so
651
+ * callers that invoke stopWatch() on deactivation / ownership loss
652
+ * actually release the fs.watch handle + debounce/retry timers
653
+ * instead of leaking them for the lifetime of the process. */
654
+ stopWatch() {
655
+ this.closeWatcher();
656
+ }
657
+ /** Reset the idle timer — safety net in case turn-end signal is missed */
658
+ resetIdleTimer() {
659
+ if (this.idleTimer) clearTimeout(this.idleTimer);
660
+ this.idleTimer = setTimeout(() => {
661
+ this.idleTimer = null;
662
+ if (this.onIdleCallback) this.onIdleCallback();
663
+ }, 1e3);
664
+ }
665
+ closeWatcher() {
666
+ dropTrace("watch.close.call", { watcher: !!this.watcher, watchDebounce: !!this.watchDebounce, sendRetryTimer: !!this.sendRetryTimer, finalLen: this.finalLane.length, streamLen: this.streamLane.length, path: this.watchingPath || "(none)" });
667
+ if (this.watchDebounce) {
668
+ clearTimeout(this.watchDebounce);
669
+ this.watchDebounce = null;
670
+ }
671
+ if (this.sendRetryTimer) {
672
+ clearTimeout(this.sendRetryTimer);
673
+ this.sendRetryTimer = null;
674
+ }
675
+ if (this.watcher) {
676
+ this.watcher.close();
677
+ this.watcher = null;
678
+ }
679
+ this.watchingPath = "";
680
+ }
681
+ scheduleWatchFlush() {
682
+ if (this.watchDebounce) clearTimeout(this.watchDebounce);
683
+ this.watchDebounce = setTimeout(() => {
684
+ this.watchDebounce = null;
685
+ let _wfStat = null;
686
+ if (this.transcriptPath) {
687
+ try { _wfStat = statSync(this.transcriptPath); } catch {}
688
+ }
689
+ if (this.transcriptPath && !_wfStat) {
690
+ const relocated = detectCurrentSessionTranscript()?.transcriptPath ?? findLatestTranscriptByMtime();
691
+ if (relocated && relocated !== this.transcriptPath) {
692
+ process.stderr.write(`mixdog: watched transcript gone during flush, relocated to ${relocated}
693
+ `);
694
+ dropTrace("watch.flush.relocate", { from: this.transcriptPath, to: relocated });
695
+ this.closeWatcher();
696
+ this.transcriptPath = relocated;
697
+ this.mainSessionId = "";
698
+ this.startWatch();
699
+ }
700
+ return;
701
+ }
702
+ this._pendingStat = _wfStat;
703
+ this.forwardNewText().then((hadText) => {
704
+ // Only trace when forwardNewText actually emitted text. hadText=false flushes
705
+ // fire on every poll cycle and accumulate ~1MB/hour of identical no-op rows.
706
+ if (hadText) dropTrace("watch.flush", { hadText, transcriptPath: this.transcriptPath || "(none)", watchingPath: this.watchingPath || "(none)" });
707
+ if (hadText) {
708
+ this.resetIdleTimer();
709
+ }
710
+ }).catch(err => process.stderr.write(`[output-forwarder] forwardNewText rejected: ${err?.message || err}\n`));
711
+ }, 100);
712
+ }
713
+ updateState(mutator) {
714
+ this.statusState.update(mutator);
715
+ }
716
+ // Debounced: every commitReadProgress() used to fire a full tmp+fsync+rename+
717
+ // dir-fsync cycle through state-file.mjs writeJsonFile. Under steady
718
+ // transcript progress (Discord forwarder following a live session) that
719
+ // hit disk 5–10×/sec. Coalesce updates into a single write per 1.5s
720
+ // window; final flush on process exit / explicit flushPersistState().
721
+ persistState() {
722
+ this._pendingPersistData = {
723
+ lastFileSize: this.lastFileSize,
724
+ sentCount: this.sentCount,
725
+ lastSentHash: this.lastHash,
726
+ lastSentTime: Date.now(),
727
+ emoji: this.emoji,
728
+ sessionIdle: false,
729
+ };
730
+ if (this._persistTimer) return;
731
+ this._persistTimer = setTimeout(() => {
732
+ this._persistTimer = null;
733
+ this._flushPersistState();
734
+ }, 1500);
735
+ if (this._persistTimer.unref) this._persistTimer.unref();
736
+ }
737
+ flushPersistState() { this._flushPersistState(); }
738
+ _flushPersistState() {
739
+ const data = this._pendingPersistData;
740
+ if (!data) return;
741
+ this._pendingPersistData = null;
742
+ if (this._persistTimer) {
743
+ clearTimeout(this._persistTimer);
744
+ this._persistTimer = null;
745
+ }
746
+ this.updateState((state) => {
747
+ state.lastFileSize = data.lastFileSize;
748
+ state.sentCount = data.sentCount;
749
+ state.lastSentHash = data.lastSentHash;
750
+ state.lastSentTime = data.lastSentTime;
751
+ state.emoji = data.emoji;
752
+ state.sessionIdle = data.sessionIdle;
753
+ });
754
+ }
755
+ }
756
+ export {
757
+ OutputForwarder,
758
+ cwdToProjectSlug,
759
+ detectCurrentSessionTranscript,
760
+ discoverCurrentClaudeSession,
761
+ discoverSessionBoundTranscript,
762
+ findLatestTranscriptByMtime,
763
+ getLatestInteractiveClaudeSession,
764
+ listInteractiveClaudeSessions
765
+ };