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,784 @@
1
+
2
+ // discord.js is loaded lazily so that importing this module does not pay
3
+ // the discord.js initialization cost at top level.
4
+ let _discord = null;
5
+ let _ChannelType = null;
6
+ async function ensureDiscord() {
7
+ if (_discord) return _discord;
8
+ _discord = await import("discord.js");
9
+ _ChannelType = _discord.ChannelType;
10
+ return _discord;
11
+ }
12
+ import {
13
+ readFileSync,
14
+ writeFileSync,
15
+ mkdirSync,
16
+ readdirSync,
17
+ rmSync,
18
+ statSync,
19
+ realpathSync
20
+ } from "fs";
21
+ import { join, sep } from "path";
22
+ import { chunk } from "../lib/format.mjs";
23
+ import { withConfigLock } from "../lib/config-lock.mjs";
24
+ import { readSection, updateSection } from "../../shared/config.mjs";
25
+ const MAX_CHUNK_LIMIT = 2e3;
26
+ const MAX_ATTACHMENT_BYTES = 25 * 1024 * 1024;
27
+ const RECENT_SENT_CAP = 200;
28
+ function defaultAccess() {
29
+ return {
30
+ dmPolicy: "allowlist",
31
+ allowFrom: [],
32
+ channels: {}
33
+ };
34
+ }
35
+ function normalizeAccess(parsed) {
36
+ const defaults = defaultAccess();
37
+ return {
38
+ // Legacy "pairing" policy was removed (its approval flow was never
39
+ // completable); normalize it to "allowlist" on load.
40
+ dmPolicy: parsed?.dmPolicy === "pairing" ? "allowlist" : (parsed?.dmPolicy ?? defaults.dmPolicy),
41
+ allowFrom: parsed?.allowFrom ?? defaults.allowFrom,
42
+ channels: parsed?.channels ?? defaults.channels,
43
+ mentionPatterns: parsed?.mentionPatterns,
44
+ // Setup UI historically saved a boolean toggle; runtime needs an emoji
45
+ // string for msg.react(). true → default emoji, non-string → off.
46
+ ackReaction: parsed?.ackReaction === true
47
+ ? "✅"
48
+ : (typeof parsed?.ackReaction === "string" && parsed.ackReaction ? parsed.ackReaction : undefined),
49
+ replyToMode: parsed?.replyToMode,
50
+ textChunkLimit: parsed?.textChunkLimit,
51
+ };
52
+ }
53
+ function safeAttName(att) {
54
+ return (att.name ?? att.id).replace(/[\[\]\r\n;]/g, "_");
55
+ }
56
+ class DiscordBackend {
57
+ name = "discord";
58
+ onMessage = null;
59
+ onInteraction = null;
60
+ onModalRequest = null;
61
+ onCustomCommand = null;
62
+ client;
63
+ stateDir;
64
+ configFile;
65
+ inboxDir;
66
+ token;
67
+ isStatic;
68
+ bootAccess = null;
69
+ initialAccess;
70
+ recentSentIds = /* @__PURE__ */ new Set();
71
+ sendCount = 0;
72
+ typingIntervals = /* @__PURE__ */ new Map();
73
+ constructor(config, stateDir) {
74
+ this.token = config.token;
75
+ this.mainChannelId = config.mainChannelId ?? "";
76
+ this.stateDir = stateDir;
77
+ this.configFile = config.configPath ?? "";
78
+ this.inboxDir = join(stateDir, "inbox");
79
+ this.isStatic = config.accessMode === "static";
80
+ this.initialAccess = normalizeAccess(config.access);
81
+ this.client = null;
82
+ }
83
+ // ── Lifecycle ──────────────────────────────────────────────────────
84
+ async connect() {
85
+ // Re-entry guard: if a connect() is already in-flight or completed, return
86
+ // the same promise / no-op so concurrent ownership-timer fires cannot
87
+ // overwrite this.client, duplicate listeners, or trigger duplicate logins.
88
+ if (this._connectPromise) return this._connectPromise;
89
+ this._connectPromise = this._connectInner().catch((err) => {
90
+ this._connectPromise = null;
91
+ throw err;
92
+ });
93
+ return this._connectPromise;
94
+ }
95
+ async _connectInner() {
96
+ await this._buildClient();
97
+ this._applyStaticAccessOverride();
98
+ this._registerEventListeners();
99
+ this._registerSlashCommands();
100
+ this._registerShardListeners();
101
+ try {
102
+ await this._awaitLogin();
103
+ } catch (err) {
104
+ // Destroy the partial Client to free the listeners/handles it already
105
+ // attached. Without this, a ready-timeout retry leaks every listener
106
+ // set by _registerEventListeners/_registerSlashCommands/_registerShardListeners.
107
+ try { this.client?.destroy?.(); } catch {}
108
+ this.client = null;
109
+ throw err;
110
+ }
111
+ this.persistAccessFromChannelsConfig();
112
+ }
113
+ async _buildClient() {
114
+ const { Client, GatewayIntentBits, Partials } = await ensureDiscord();
115
+ this.client = new Client({
116
+ intents: [
117
+ GatewayIntentBits.DirectMessages,
118
+ GatewayIntentBits.Guilds,
119
+ GatewayIntentBits.GuildMessages,
120
+ GatewayIntentBits.MessageContent
121
+ ],
122
+ partials: [Partials.Channel]
123
+ });
124
+ }
125
+ _applyStaticAccessOverride() {
126
+ if (this.isStatic) {
127
+ this.bootAccess = this.loadAccess();
128
+ }
129
+ }
130
+ _registerEventListeners() {
131
+ this.client.on("error", (err) => {
132
+ process.stderr.write(`mixdog discord: client error: ${err}
133
+ `);
134
+ });
135
+ this.client.on("messageCreate", (msg) => {
136
+ if (msg.author.id === this.client.user?.id) {
137
+ return;
138
+ }
139
+ if (msg.author.bot) return;
140
+ this.handleInbound(msg, Date.now()).catch(
141
+ (e) => process.stderr.write(`mixdog discord: handleInbound failed: ${e}
142
+ `)
143
+ );
144
+ });
145
+ this.client.on("interactionCreate", async (interaction) => {
146
+ try {
147
+ // Trust gate for interactions. Buttons / selects / modal submits used to
148
+ // reach onInteraction / onModalRequest without passing through the
149
+ // message gate(), so a configured allowFrom never applied to them
150
+ // (schedule/quiet/profile modals + perm approvals were openable by any
151
+ // user in the channel). Apply the same allowFrom decision here; an empty
152
+ // allowFrom stays open so current configs are unaffected.
153
+ if (!this._interactionAllowed(interaction.channelId ?? "", interaction.user?.id, interaction)) {
154
+ try {
155
+ if (typeof interaction.reply === "function" && !interaction.replied && !interaction.deferred) {
156
+ await interaction.reply({ content: "⛔ Not authorized for this action.", ephemeral: true });
157
+ } else if (typeof interaction.deferUpdate === "function") {
158
+ await interaction.deferUpdate();
159
+ }
160
+ } catch {}
161
+ return;
162
+ }
163
+ if (interaction.isChatInputCommand() && interaction.commandName === "stop") {
164
+ await interaction.reply({ content: "\u23F9 Stopping...", ephemeral: true });
165
+ if (this.onInteraction) {
166
+ this.onInteraction({
167
+ type: "button",
168
+ customId: "stop_task",
169
+ userId: interaction.user.id,
170
+ channelId: interaction.channelId ?? ""
171
+ });
172
+ }
173
+ return;
174
+ }
175
+ if (interaction.isModalSubmit()) {
176
+ if (this.onInteraction) {
177
+ const fields = {};
178
+ for (const row of interaction.components) {
179
+ for (const comp of row.components ?? []) {
180
+ if (comp.customId && comp.value != null) fields[comp.customId] = String(comp.value);
181
+ }
182
+ }
183
+ this.onInteraction({
184
+ type: "modal",
185
+ customId: interaction.customId,
186
+ userId: interaction.user.id,
187
+ channelId: interaction.channelId ?? "",
188
+ fields,
189
+ message: interaction.message ? { id: interaction.message.id } : void 0
190
+ });
191
+ }
192
+ await interaction.deferUpdate().catch(() => {
193
+ });
194
+ return;
195
+ }
196
+ if (interaction.isButton() || interaction.isStringSelectMenu() || interaction.isRoleSelectMenu() || interaction.isUserSelectMenu() || interaction.isChannelSelectMenu()) {
197
+ const needsModal = interaction.isButton() && (interaction.customId === "sched_add_next" || interaction.customId === "sched_edit_next" || interaction.customId === "quiet_set_next" || interaction.customId === "activity_add_next" || interaction.customId === "profile_edit");
198
+ if (needsModal) {
199
+ if (this.onModalRequest) {
200
+ await Promise.resolve(this.onModalRequest(interaction)).catch((err) => {
201
+ process.stderr.write(`mixdog discord: onModalRequest failed: ${err}\n`);
202
+ });
203
+ }
204
+ return;
205
+ }
206
+ await interaction.deferUpdate().catch(() => {
207
+ });
208
+ if (this.onInteraction) {
209
+ this.onInteraction({
210
+ type: interaction.isButton() ? "button" : "select",
211
+ customId: interaction.customId,
212
+ userId: interaction.user.id,
213
+ channelId: interaction.channelId,
214
+ values: interaction.isStringSelectMenu() ? interaction.values : void 0,
215
+ message: interaction.message ? { id: interaction.message.id } : void 0
216
+ });
217
+ }
218
+ }
219
+ } catch (err) {
220
+ process.stderr.write(`mixdog discord: interaction error: ${err}
221
+ `);
222
+ }
223
+ });
224
+ }
225
+ _registerSlashCommands() {
226
+ this.client.on(_discord.Events.ClientReady, async (c) => {
227
+ process.stderr.write(`mixdog discord: gateway connected as ${c.user.tag}
228
+ `);
229
+ try {
230
+ // Plugin registers no global commands; clear the global slot so any
231
+ // pre-existing entry from prior installs is not surfaced to users.
232
+ await c.application?.commands.set([]);
233
+ process.stderr.write(`mixdog discord: global application commands cleared
234
+ `);
235
+
236
+ // Replace each guild's command set with just /stop. set() overwrites,
237
+ // so the desired set is the only one that survives.
238
+ const desiredCommands = [
239
+ { name: "stop", description: "Stop the current Claude Code response" },
240
+ ];
241
+ for (const [guildId] of c.guilds.cache) {
242
+ await c.application?.commands.set(desiredCommands, guildId);
243
+ }
244
+ // Register /stop globally so it is available in DM bridge contexts
245
+ // where there is no guild scope.
246
+ try {
247
+ await c.application?.commands.set(desiredCommands);
248
+ } catch (e) {
249
+ process.stderr.write(`mixdog discord: global /stop register failed: ${e?.message}\n`);
250
+ }
251
+ process.stderr.write(`mixdog discord: /stop command registered (${c.guilds.cache.size} guild(s))
252
+ `);
253
+ } catch (err) {
254
+ process.stderr.write(`mixdog discord: slash command registration failed: ${err}
255
+ `);
256
+ }
257
+ });
258
+ }
259
+ _registerShardListeners() {
260
+ this.client.on("shardDisconnect", (ev, id) => {
261
+ process.stderr.write(`mixdog discord: shard ${id} disconnected (code ${ev.code}). Will auto-reconnect.
262
+ `);
263
+ });
264
+ this.client.on("shardReconnecting", (id) => {
265
+ process.stderr.write(`mixdog discord: shard ${id} reconnecting...
266
+ `);
267
+ });
268
+ this.client.on("shardResume", (id, replayedEvents) => {
269
+ process.stderr.write(`mixdog discord: shard ${id} resumed (replayed ${replayedEvents} events)
270
+ `);
271
+ });
272
+ this.client.on("warn", (msg) => {
273
+ process.stderr.write(`mixdog discord: warn: ${msg}
274
+ `);
275
+ });
276
+ }
277
+ async _awaitLogin() {
278
+ let readyTimeout;
279
+ const readyPromise = new Promise((resolve, reject) => {
280
+ readyTimeout = setTimeout(() => reject(new Error("discord ready timeout (30s)")), 3e4);
281
+ this.client.once(_discord.Events.ClientReady, () => {
282
+ clearTimeout(readyTimeout);
283
+ resolve();
284
+ });
285
+ });
286
+ try {
287
+ await this.client.login(this.token);
288
+ } catch (err) {
289
+ clearTimeout(readyTimeout);
290
+ throw err;
291
+ }
292
+ await readyPromise;
293
+ }
294
+ async disconnect() {
295
+ for (const interval of this.typingIntervals.values()) {
296
+ clearInterval(interval);
297
+ }
298
+ this.typingIntervals.clear();
299
+ if (this.client) this.client.destroy();
300
+ this._connectPromise = null;
301
+ }
302
+ resetSendCount() {
303
+ this.sendCount = 0;
304
+ }
305
+ startTyping(channelId) {
306
+ this.stopTyping(channelId);
307
+ const ch = this.client.channels.cache.get(channelId);
308
+ if (ch && "sendTyping" in ch) {
309
+ void ch.sendTyping().catch(() => {
310
+ });
311
+ const interval = setInterval(() => {
312
+ if ("sendTyping" in ch) {
313
+ ch.sendTyping().catch(() => {
314
+ });
315
+ }
316
+ }, 9e3);
317
+ this.typingIntervals.set(channelId, interval);
318
+ }
319
+ }
320
+ stopTyping(channelId) {
321
+ const interval = this.typingIntervals.get(channelId);
322
+ if (interval) {
323
+ clearInterval(interval);
324
+ this.typingIntervals.delete(channelId);
325
+ }
326
+ }
327
+ // ── Outbound operations ────────────────────────────────────────────
328
+ async sendMessage(chatId, text, opts) {
329
+ const ch = await this.fetchAllowedChannel(chatId);
330
+ if (!("send" in ch)) throw new Error("channel is not sendable");
331
+ const files = opts?.files ?? [];
332
+ const replyTo = opts?.replyTo;
333
+ for (const f of files) {
334
+ this.assertSendable(f);
335
+ const st = statSync(f);
336
+ if (st.size > MAX_ATTACHMENT_BYTES) {
337
+ throw new Error(`file too large: ${f} (${(st.size / 1024 / 1024).toFixed(1)}MB, max 25MB)`);
338
+ }
339
+ }
340
+ if (files.length > 10) throw new Error("max 10 attachments per message");
341
+ if (text && this.sendCount > 0) {
342
+ text = "\u3164\n" + text;
343
+ }
344
+ const access = this.loadAccess();
345
+ const limit = Math.max(1, Math.min(access.textChunkLimit ?? MAX_CHUNK_LIMIT, MAX_CHUNK_LIMIT));
346
+ const replyMode = access.replyToMode ?? "off";
347
+ const chunks = chunk(text, limit);
348
+ const sentIds = [];
349
+ try {
350
+ for (let i = 0; i < chunks.length; i++) {
351
+ const shouldReplyTo = replyTo != null && replyMode !== "off" && (replyMode === "all" || i === 0);
352
+ const embeds = i === 0 ? opts?.embeds ?? [] : [];
353
+ const components = i === 0 ? opts?.components ?? [] : [];
354
+ const sent = await ch.send({
355
+ content: chunks[i],
356
+ ...embeds.length > 0 ? { embeds } : {},
357
+ ...components.length > 0 ? { components } : {},
358
+ ...i === 0 && files.length > 0 ? { files } : {},
359
+ ...shouldReplyTo ? { reply: { messageReference: replyTo, failIfNotExists: false } } : {}
360
+ });
361
+ this.noteSent(sent.id);
362
+ sentIds.push(sent.id);
363
+ }
364
+ this.sendCount += sentIds.length;
365
+ } catch (err) {
366
+ const msg = err instanceof Error ? err.message : String(err);
367
+ throw new Error(`send failed after ${sentIds.length}/${chunks.length} chunk(s): ${msg}`);
368
+ }
369
+ return { sentIds };
370
+ }
371
+ async fetchMessages(channelId, limit) {
372
+ const ch = await this.fetchAllowedChannel(channelId);
373
+ const capped = Math.min(limit, 100);
374
+ const msgs = await ch.messages.fetch({ limit: capped });
375
+ const me = this.client.user?.id;
376
+ return [...msgs.values()].reverse().map((m) => ({
377
+ id: m.id,
378
+ user: m.author.id === me ? "me" : m.author.username,
379
+ text: m.content.replace(/[\r\n]+/g, " \u23CE "),
380
+ ts: m.createdAt.toISOString(),
381
+ isMe: m.author.id === me,
382
+ attachmentCount: m.attachments.size
383
+ }));
384
+ }
385
+ async react(chatId, messageId, emoji) {
386
+ const ch = await this.fetchAllowedChannel(chatId);
387
+ const msg = await ch.messages.fetch(messageId);
388
+ await msg.react(emoji);
389
+ }
390
+ async removeReaction(chatId, messageId, emoji) {
391
+ const ch = await this.fetchAllowedChannel(chatId);
392
+ const msg = await ch.messages.fetch(messageId);
393
+ const me = this.client.user?.id;
394
+ if (me) {
395
+ const reaction = msg.reactions.cache.get(emoji);
396
+ if (reaction) await reaction.users.remove(me);
397
+ }
398
+ }
399
+ async editMessage(chatId, messageId, text, opts) {
400
+ const ch = await this.fetchAllowedChannel(chatId);
401
+ const msg = await ch.messages.fetch(messageId);
402
+ const access = this.loadAccess();
403
+ const limit = Math.max(1, Math.min(access.textChunkLimit ?? MAX_CHUNK_LIMIT, MAX_CHUNK_LIMIT));
404
+ const chunks = chunk(text, limit);
405
+ const edited = await msg.edit({
406
+ content: chunks[0] || null,
407
+ ...opts?.embeds ? { embeds: opts.embeds } : {},
408
+ ...opts?.components ? { components: opts.components } : {}
409
+ });
410
+ const sentIds = [edited.id];
411
+ // Idempotent overflow: reuse previously-sent overflow messages, replacing
412
+ // rather than appending on every edit call.
413
+ if (!this.editOverflow) this.editOverflow = Object.create(null);
414
+ const prevOverflow = this.editOverflow[messageId] ?? [];
415
+ const newOverflow = [];
416
+ for (let i = 1; i < chunks.length; i++) {
417
+ const prevId = prevOverflow[i - 1];
418
+ if (prevId) {
419
+ try {
420
+ const prevMsg = await ch.messages.fetch(prevId);
421
+ await prevMsg.edit({ content: chunks[i] });
422
+ newOverflow.push(prevId);
423
+ sentIds.push(prevId);
424
+ continue;
425
+ } catch { /* message deleted externally — fall through to send */ }
426
+ }
427
+ const sent = await ch.send({ content: chunks[i] });
428
+ this.noteSent(sent.id);
429
+ newOverflow.push(sent.id);
430
+ sentIds.push(sent.id);
431
+ }
432
+ // Delete leftover overflow messages from a prior longer edit.
433
+ for (let j = chunks.length - 1; j < prevOverflow.length; j++) {
434
+ try { const m = await ch.messages.fetch(prevOverflow[j]); await m.delete(); } catch { /* already gone */ }
435
+ }
436
+ this.editOverflow[messageId] = newOverflow;
437
+ return sentIds[0];
438
+ }
439
+ async deleteMessage(chatId, messageId) {
440
+ const ch = await this.fetchAllowedChannel(chatId);
441
+ const msg = await ch.messages.fetch(messageId);
442
+ await msg.delete();
443
+ }
444
+ async downloadAttachment(chatId, messageId) {
445
+ const ch = await this.fetchAllowedChannel(chatId);
446
+ const msg = await ch.messages.fetch(messageId);
447
+ if (msg.attachments.size === 0) return [];
448
+ const results = [];
449
+ for (const att of msg.attachments.values()) {
450
+ const path = await this.downloadSingleAttachment(att);
451
+ results.push({
452
+ id: att.id,
453
+ path,
454
+ name: safeAttName(att),
455
+ contentType: att.contentType ?? "unknown",
456
+ size: att.size
457
+ });
458
+ }
459
+ return results;
460
+ }
461
+ async validateChannel(chatId) {
462
+ await this.fetchAllowedChannel(chatId);
463
+ }
464
+ // ── Access control ─────────────────────────────────────────────────
465
+ readConfigAccess() {
466
+ try {
467
+ const parsed = readSection("channels");
468
+ const access = normalizeAccess(parsed.access ?? this.initialAccess);
469
+ if (parsed.channelsConfig) {
470
+ for (const entry of Object.values(parsed.channelsConfig)) {
471
+ if (typeof entry === "object" && entry !== null) {
472
+ const id = entry.channelId;
473
+ if (id && !(id in access.channels)) {
474
+ access.channels[id] = { requireMention: false, allowFrom: [] };
475
+ }
476
+ }
477
+ }
478
+ }
479
+ return access;
480
+ } catch {
481
+ return this.initialAccess;
482
+ }
483
+ }
484
+ persistAccessFromChannelsConfig() {
485
+ if (this.isStatic || !this.configFile) return;
486
+ try {
487
+ const parsed = readSection("channels");
488
+ if (!parsed.channelsConfig) return;
489
+ const access = normalizeAccess(parsed.access);
490
+ let changed = false;
491
+ for (const entry of Object.values(parsed.channelsConfig)) {
492
+ if (typeof entry === "object" && entry !== null) {
493
+ const id = entry.channelId;
494
+ if (id && !(id in access.channels)) {
495
+ access.channels[id] = { requireMention: false, allowFrom: [] };
496
+ changed = true;
497
+ }
498
+ }
499
+ }
500
+ if (changed) this.saveAccess(access);
501
+ } catch (err) {
502
+ process.stderr.write(`mixdog discord: persistAccessFromChannelsConfig failed: ${err}\n`);
503
+ }
504
+ }
505
+ loadAccess() {
506
+ const a = this.bootAccess ?? this.readConfigAccess();
507
+ // Single-source channel setup: auto-allow the configured main channel so
508
+ // both inbound (gate) and outbound (fetchAllowedChannel) accept it from a
509
+ // single channelsConfig.main.channelId — no separate access.channels entry.
510
+ // An explicit entry for the same channel is preserved (not overwritten).
511
+ if (this.mainChannelId && a && !(this.mainChannelId in (a.channels ?? {}))) {
512
+ return { ...a, channels: { ...(a.channels ?? {}), [this.mainChannelId]: { allowFrom: [], requireMention: false } } };
513
+ }
514
+ return a;
515
+ }
516
+ saveAccess(a) {
517
+ if (this.isStatic) return;
518
+ if (!this.configFile) return;
519
+ return withConfigLock(() => {
520
+ mkdirSync(this.stateDir, { recursive: true, mode: 448 });
521
+ const access = {
522
+ dmPolicy: a.dmPolicy,
523
+ allowFrom: a.allowFrom,
524
+ channels: a.channels,
525
+ ...a.mentionPatterns ? { mentionPatterns: a.mentionPatterns } : {},
526
+ ...a.ackReaction ? { ackReaction: a.ackReaction } : {},
527
+ ...a.replyToMode ? { replyToMode: a.replyToMode } : {},
528
+ ...a.textChunkLimit ? { textChunkLimit: a.textChunkLimit } : {}
529
+ };
530
+ updateSection("channels", (channels) => ({
531
+ ...channels,
532
+ access
533
+ }));
534
+ });
535
+ }
536
+ // Trust decision for component/modal interactions, which otherwise bypass
537
+ // gate() entirely. Mirrors gate()'s channel branch: a channel (or, when no
538
+ // channel policy applies, the global) allowFrom list is enforced when set,
539
+ // while an empty allowFrom stays open so existing configs are unchanged.
540
+ // requireMention is N/A for a click; dmPolicy "disabled" drops all.
541
+ _interactionAllowed(channelId, userId, interaction) {
542
+ if (!userId) return false;
543
+ const access = this.loadAccess();
544
+ if (!access || access.dmPolicy === "disabled") return false;
545
+ // Mirror gate(): normalize threads to their parent channel so component
546
+ // clicks in threads inherit the parent's access policy instead of falling
547
+ // back to the global allowFrom.
548
+ let resolvedChannelId = channelId;
549
+ let isDM = false;
550
+ try {
551
+ const ch = interaction?.channel;
552
+ if (ch?.isThread?.()) resolvedChannelId = ch.parentId ?? channelId;
553
+ // Detect DM interactions. gate() honors the DM allowlist; a blanket
554
+ // non-DM fail-closed would deny e.g. global /stop.
555
+ if (ch?.type === _ChannelType.DM || ch?.isDMBased?.()) isDM = true;
556
+ } catch {}
557
+ if (isDM) {
558
+ // Mirror gate()'s DM path exactly: deliver iff the sender is in the
559
+ // global allowFrom. dmPolicy "disabled" was already filtered above.
560
+ if (access.allowFrom?.includes?.(userId)) return true;
561
+ return false;
562
+ }
563
+ const policy = resolvedChannelId ? access.channels?.[resolvedChannelId] : null;
564
+ // Mirror gate(): when this looks like a configured guild channel and no
565
+ // per-channel policy exists, fail closed (drop) instead of falling back
566
+ // to the global allowFrom. Component/modal interactions outside any
567
+ // configured channel policy should be treated the same as messages there.
568
+ if (!policy && resolvedChannelId) return false;
569
+ const allowFrom = (policy ? policy.allowFrom : access.allowFrom) ?? [];
570
+ if (allowFrom.length > 0 && !allowFrom.includes(userId)) return false;
571
+ return true;
572
+ }
573
+ async gate(msg) {
574
+ const access = this.loadAccess();
575
+ if (access.dmPolicy === "disabled") return { action: "drop" };
576
+ const senderId = msg.author.id;
577
+ const isDM = msg.channel.type === _ChannelType.DM;
578
+ if (isDM) {
579
+ if (access.allowFrom.includes(senderId)) return { action: "deliver", access };
580
+ return { action: "drop" };
581
+ }
582
+ const channelId = msg.channel.isThread() ? msg.channel.parentId ?? msg.channelId : msg.channelId;
583
+ const policy = access.channels[channelId];
584
+ if (!policy) return { action: "drop" };
585
+ const channelAllowFrom = policy.allowFrom ?? [];
586
+ const requireMention = policy.requireMention ?? false;
587
+ if (channelAllowFrom.length > 0 && !channelAllowFrom.includes(senderId)) {
588
+ return { action: "drop" };
589
+ }
590
+ if (requireMention && !await this.isMentioned(msg, access.mentionPatterns)) {
591
+ return { action: "drop" };
592
+ }
593
+ return { action: "deliver", access };
594
+ }
595
+ async isMentioned(msg, extraPatterns) {
596
+ if (this.client.user && msg.mentions.has(this.client.user)) return true;
597
+ const refId = msg.reference?.messageId;
598
+ if (refId) {
599
+ if (this.recentSentIds.has(refId)) return true;
600
+ try {
601
+ const ref = await msg.fetchReference();
602
+ if (ref.author.id === this.client.user?.id) return true;
603
+ } catch {
604
+ }
605
+ }
606
+ const text = msg.content;
607
+ for (const pat of extraPatterns ?? []) {
608
+ if (typeof pat !== "string" || pat.length === 0 || pat.length > 128) continue;
609
+ // Reject known catastrophic-backtracking shapes: nested quantifiers
610
+ // like (x+)+, (x*)*, (x+)*, (x*)+ on grouped subexpressions.
611
+ if (/\([^)]*[+*]\)[+*]/.test(pat)) continue;
612
+ try {
613
+ if (new RegExp(pat, "i").test(text)) return true;
614
+ } catch {
615
+ throw new Error(`[discord] invalid mention pattern: ${pat}`);
616
+ }
617
+ }
618
+ return false;
619
+ }
620
+ // ── Inbound handling ───────────────────────────────────────────────
621
+ async handleInbound(msg, receivedAtMs = Date.now()) {
622
+ const result = await this.gate(msg);
623
+ if (result.action === "drop") return;
624
+ if (result.access.ackReaction) {
625
+ void msg.react(result.access.ackReaction).catch(() => {
626
+ });
627
+ }
628
+ const atts = [];
629
+ for (const att of msg.attachments.values()) {
630
+ atts.push({
631
+ id: att.id,
632
+ name: safeAttName(att),
633
+ contentType: att.contentType ?? "unknown",
634
+ size: att.size
635
+ });
636
+ }
637
+ const text = msg.content || (atts.length > 0 ? "(attachment)" : "");
638
+ if (text.match(/^\/(bot|profile)\s*\(/) && this.onCustomCommand) {
639
+ const replyFn = async (reply, opts) => {
640
+ try {
641
+ const ch = await this.fetchAllowedChannel(msg.channelId);
642
+ if ("send" in ch) {
643
+ await ch.send({
644
+ ...reply ? { content: reply } : {},
645
+ ...opts?.embeds?.length ? { embeds: opts.embeds } : {},
646
+ ...opts?.components?.length ? { components: opts.components } : {}
647
+ });
648
+ }
649
+ } catch (err) {
650
+ process.stderr.write(`mixdog discord: custom command reply failed: ${err}
651
+ `);
652
+ }
653
+ };
654
+ this.onCustomCommand(text, msg.channelId, msg.author.id, replyFn);
655
+ return;
656
+ }
657
+ if (this.onMessage) {
658
+ // Thread messages were gated against the parent channel's policy (see
659
+ // gate() above), but routing also needs the parent id to find labels/
660
+ // modes in channelsConfig. Surface parentChatId so resolveInboundRoute
661
+ // can fall back to the parent when the thread id has no entry.
662
+ const isThread = (() => { try { return !!msg.channel?.isThread?.(); } catch { return false; } })();
663
+ const parentChatId = isThread ? (msg.channel.parentId ?? null) : null;
664
+ this.onMessage({
665
+ chatId: msg.channelId,
666
+ parentChatId,
667
+ messageId: msg.id,
668
+ receivedAtMs,
669
+ discordCreatedAtMs: msg.createdTimestamp ?? msg.createdAt?.getTime?.() ?? null,
670
+ user: msg.author.username,
671
+ userId: msg.author.id,
672
+ text,
673
+ ts: msg.createdAt.toISOString(),
674
+ attachments: atts
675
+ });
676
+ }
677
+ }
678
+ // ── Channel helpers ────────────────────────────────────────────────
679
+ async fetchTextChannel(id) {
680
+ const ch = await this.client.channels.fetch(id);
681
+ if (!ch || !ch.isTextBased()) {
682
+ throw new Error(`channel ${id} not found or not text-based`);
683
+ }
684
+ return ch;
685
+ }
686
+ async fetchAllowedChannel(id) {
687
+ const ch = await this.fetchTextChannel(id);
688
+ const access = this.loadAccess();
689
+ if (ch.type === _ChannelType.DM) {
690
+ let recipientId = ch.recipientId;
691
+ if (!recipientId && ch.partial) {
692
+ const fetched = await ch.fetch();
693
+ recipientId = fetched.recipientId;
694
+ }
695
+ if (recipientId && access.allowFrom.includes(recipientId)) return ch;
696
+ } else {
697
+ const key = ch.isThread() ? ch.parentId ?? ch.id : ch.id;
698
+ if (key in access.channels) return ch;
699
+ }
700
+ throw new Error(`channel ${id} is not allowlisted: add it via the Setup UI channels panel`);
701
+ }
702
+ noteSent(id) {
703
+ this.recentSentIds.add(id);
704
+ if (this.recentSentIds.size > RECENT_SENT_CAP) {
705
+ const first = this.recentSentIds.values().next().value;
706
+ if (first) this.recentSentIds.delete(first);
707
+ }
708
+ }
709
+ assertSendable(f) {
710
+ let real, stateReal;
711
+ try {
712
+ real = realpathSync(f);
713
+ stateReal = realpathSync(this.stateDir);
714
+ } catch (err) {
715
+ // Fail closed: state dir is created at boot so realpath should succeed
716
+ // invariantly; a missing `f` would fail downstream anyway. Skipping the
717
+ // state-guard on realpath failure was a bypass for symlinked attachments.
718
+ throw new Error(`refusing to send: cannot resolve real path for ${f} (${err?.message || err})`);
719
+ }
720
+ const inbox = join(stateReal, "inbox");
721
+ if (real.startsWith(stateReal + sep) && !real.startsWith(inbox + sep)) {
722
+ throw new Error(`refusing to send channel state: ${f}`);
723
+ }
724
+ }
725
+ async downloadSingleAttachment(att) {
726
+ if (att.size > MAX_ATTACHMENT_BYTES) {
727
+ throw new Error(
728
+ `attachment too large: ${(att.size / 1024 / 1024).toFixed(1)}MB, max ${MAX_ATTACHMENT_BYTES / 1024 / 1024}MB`
729
+ );
730
+ }
731
+ const res = await fetch(att.url, { signal: AbortSignal.timeout(180_000) });
732
+ if (!res.ok) {
733
+ throw new Error(`attachment download failed: HTTP ${res.status}`);
734
+ }
735
+ if (!res.body) {
736
+ throw new Error(`attachment download returned empty body: ${att.name ?? att.id}`);
737
+ }
738
+ // Stream the response so an oversized payload is aborted before the
739
+ // full body lands in memory. Buffering via arrayBuffer() first would
740
+ // already exceed MAX_ATTACHMENT_BYTES by the time we checked length.
741
+ const reader = res.body.getReader();
742
+ const chunks = [];
743
+ let received = 0;
744
+ try {
745
+ while (true) {
746
+ const { done, value } = await reader.read();
747
+ if (done) break;
748
+ if (!value) continue;
749
+ received += value.byteLength;
750
+ if (received > MAX_ATTACHMENT_BYTES) {
751
+ try { await reader.cancel(); } catch {}
752
+ throw new Error(
753
+ `attachment payload too large: exceeded ${MAX_ATTACHMENT_BYTES / 1024 / 1024}MB while streaming (${att.name ?? att.id})`
754
+ );
755
+ }
756
+ chunks.push(value);
757
+ }
758
+ } finally {
759
+ try { reader.releaseLock(); } catch {}
760
+ }
761
+ if (received === 0) {
762
+ throw new Error(`attachment download returned empty buffer: ${att.name ?? att.id}`);
763
+ }
764
+ const buf = Buffer.concat(chunks.map((c) => Buffer.from(c.buffer, c.byteOffset, c.byteLength)), received);
765
+ if (att.size > 0 && buf.length !== att.size) {
766
+ process.stderr.write(`mixdog discord: attachment size mismatch: expected ${att.size} got ${buf.length} (${att.name ?? att.id})\n`);
767
+ }
768
+ const name = att.name ?? `${att.id}`;
769
+ const rawExt = name.includes(".") ? name.slice(name.lastIndexOf(".") + 1) : "bin";
770
+ const ext = rawExt.replace(/[^a-zA-Z0-9]/g, "") || "bin";
771
+ const safeId = String(att.id).replace(/[^a-zA-Z0-9_-]/g, "_");
772
+ mkdirSync(this.inboxDir, { recursive: true });
773
+ const candidate = join(this.inboxDir, `${Date.now()}-${safeId}.${ext}`);
774
+ const resolvedInbox = realpathSync(this.inboxDir);
775
+ if (!candidate.startsWith(resolvedInbox + sep)) {
776
+ throw new Error(`attachment path traversal rejected: ${candidate}`);
777
+ }
778
+ writeFileSync(candidate, buf);
779
+ return candidate;
780
+ }
781
+ }
782
+ export {
783
+ DiscordBackend
784
+ };