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,196 @@
1
+ // Server-less, preview-first symbol rename.
2
+ //
3
+ // Mixdog has no semantic LSP. This tool reuses the existing code-graph
4
+ // find_references API to collect every reference site for a symbol, then —
5
+ // BY DEFAULT — returns a PREVIEW only: the {file,line,col,context} sites that
6
+ // WOULD change, plus an explicit warning that the match is graph-based
7
+ // (approximate, not LSP-semantic) and may include same-named-but-different
8
+ // symbols. The destructive rewrite runs ONLY when apply===true, and replaces
9
+ // the symbol EXACTLY at the {line,col} spans find_references returned —
10
+ // verifying the slice equals the symbol before splicing. It never blanket-
11
+ // replaces a line, so same-line string/comment occurrences the reference
12
+ // search did NOT sanction are left untouched, `$`-identifiers can't mis-match
13
+ // inside larger identifiers, and original (possibly mixed) line endings are
14
+ // preserved verbatim.
15
+ import { readFileSync, writeFileSync, statSync } from 'fs';
16
+ import { isAbsolute, resolve as pathResolve } from 'path';
17
+ import { executeCodeGraphTool, markCodeGraphDirtyPaths } from '../code-graph.mjs';
18
+ import { invalidateBuiltinResultCache } from './cache-layers.mjs';
19
+
20
+ const APPROX_WARNING =
21
+ 'APPROXIMATE: matches come from the code-graph (text/word-boundary based), '
22
+ + 'NOT an LSP-semantic resolver. They may include same-named-but-different '
23
+ + 'symbols (shadowed locals, unrelated properties, etc.). Review every site '
24
+ + 'before applying.';
25
+
26
+ // A valid JS/TS/Py-style identifier — required so we never attempt a rewrite
27
+ // of an arbitrary string that would corrupt source via word-boundary regex.
28
+ function _isIdentifier(s) {
29
+ return typeof s === 'string' && /^[A-Za-z_$][\w$]*$/.test(s);
30
+ }
31
+
32
+ // Parse find_references text output. Each hit line is `rel:line:col context`.
33
+ // Diagnostic/footer lines (starting with `(` or `[`) and blanks are skipped.
34
+ function _parseReferenceLines(text) {
35
+ const sites = [];
36
+ if (typeof text !== 'string') return sites;
37
+ const re = /^(.+?):(\d+):(\d+)\s{2,}(.*)$/;
38
+ for (const raw of text.split(/\r?\n/)) {
39
+ const line = raw.replace(/\s+$/, '');
40
+ if (!line) continue;
41
+ if (line.startsWith('(') || line.startsWith('[') || line.startsWith('#')) continue;
42
+ const m = re.exec(line);
43
+ if (!m) continue;
44
+ sites.push({ file: m[1], line: +m[2], col: +m[3], context: m[4] });
45
+ }
46
+ return sites;
47
+ }
48
+
49
+ function _absFor(file, workDir) {
50
+ return isAbsolute(file) ? file : pathResolve(workDir, file);
51
+ }
52
+
53
+ // Split content into physical lines while preserving each line's own trailing
54
+ // terminator (\r\n, \n, or '' for the last line). This lets us rewrite a single
55
+ // line's text and re-join with the EXACT original endings — no normalization of
56
+ // mixed line endings.
57
+ function _splitKeepEol(content) {
58
+ const out = [];
59
+ const re = /\r\n|\n|\r/g;
60
+ let last = 0;
61
+ let m = null;
62
+ while ((m = re.exec(content))) {
63
+ out.push({ text: content.slice(last, m.index), eol: m[0] });
64
+ last = m.index + m[0].length;
65
+ }
66
+ out.push({ text: content.slice(last), eol: '' });
67
+ return out;
68
+ }
69
+
70
+ // Apply the rename ONLY at the exact {line,col} spans find_references returned.
71
+ // For each site we verify the slice [col-1 .. col-1+symbol.length] === symbol
72
+ // before splicing; if it doesn't match (stale graph, shifted line, or an
73
+ // occurrence the search didn't sanction) the site is recorded as skipped and
74
+ // nothing on that line is touched. Identity-check + slice splice replaces the
75
+ // previous `\b`-regex line rewrite, which mishandled `$`-identifiers and
76
+ // clobbered same-line non-reference occurrences.
77
+ function _applyRename(sites, symbol, newName, workDir) {
78
+ const byFile = new Map();
79
+ for (const s of sites) {
80
+ const abs = _absFor(s.file, workDir);
81
+ if (!byFile.has(abs)) byFile.set(abs, []);
82
+ byFile.get(abs).push(s);
83
+ }
84
+ const changed = [];
85
+ const skipped = [];
86
+ let replacements = 0;
87
+ for (const [abs, fileSites] of byFile) {
88
+ let content;
89
+ try { content = readFileSync(abs, 'utf8'); }
90
+ catch (err) {
91
+ for (const s of fileSites) skipped.push({ ...s, reason: `read failed: ${err && err.message ? err.message : String(err)}` });
92
+ continue;
93
+ }
94
+ const lines = _splitKeepEol(content);
95
+ // Replace from rightmost column first within a line so earlier splices
96
+ // don't shift the offsets of later ones on the same line.
97
+ const ordered = fileSites.slice().sort((a, b) => (a.line - b.line) || (b.col - a.col));
98
+ let fileTouched = false;
99
+ for (const s of ordered) {
100
+ const idx = s.line - 1;
101
+ if (idx < 0 || idx >= lines.length) { skipped.push({ ...s, reason: 'line out of range' }); continue; }
102
+ const start = s.col - 1;
103
+ const text = lines[idx].text;
104
+ if (start < 0 || start + symbol.length > text.length) { skipped.push({ ...s, reason: 'column out of range' }); continue; }
105
+ const slice = text.slice(start, start + symbol.length);
106
+ if (slice !== symbol) { skipped.push({ ...s, reason: `slice mismatch (got ${JSON.stringify(slice)})` }); continue; }
107
+ lines[idx].text = text.slice(0, start) + newName + text.slice(start + symbol.length);
108
+ replacements += 1;
109
+ fileTouched = true;
110
+ }
111
+ if (fileTouched) {
112
+ writeFileSync(abs, lines.map((l) => l.text + l.eol).join(''), 'utf8');
113
+ changed.push(abs);
114
+ }
115
+ }
116
+ return { changed, replacements, skipped };
117
+ }
118
+
119
+ export async function executeRenameTool(args, workDir, signal = null) {
120
+ try {
121
+ const symbol = (args && typeof args.symbol === 'string') ? args.symbol.trim() : '';
122
+ const newName = (args && typeof args.new_name === 'string') ? args.new_name.trim() : '';
123
+ const file = (args && typeof args.file === 'string' && args.file.trim()) ? args.file.trim() : '';
124
+ const apply = args && args.apply === true;
125
+ if (!symbol) return 'Error: rename requires "symbol"';
126
+ if (!newName) return 'Error: rename requires "new_name"';
127
+ if (!_isIdentifier(symbol)) return `Error: rename "symbol" must be a valid identifier (got ${JSON.stringify(args.symbol)})`;
128
+ if (!_isIdentifier(newName)) return `Error: rename "new_name" must be a valid identifier (got ${JSON.stringify(args.new_name)})`;
129
+ if (symbol === newName) return 'Error: rename "new_name" is identical to "symbol"; nothing to do';
130
+
131
+ // Collect reference sites via the code-graph references mode.
132
+ const refArgs = { mode: 'references', symbol };
133
+ if (file) refArgs.file = file;
134
+ const refText = await executeCodeGraphTool('code_graph', refArgs, workDir, signal);
135
+ if (typeof refText === 'string' && refText.startsWith('Error:')) {
136
+ return refText;
137
+ }
138
+ const sites = _parseReferenceLines(refText);
139
+
140
+ if (sites.length === 0) {
141
+ return JSON.stringify({
142
+ ok: true,
143
+ mode: apply ? 'apply' : 'preview',
144
+ symbol,
145
+ new_name: newName,
146
+ warning: APPROX_WARNING,
147
+ count: 0,
148
+ sites: [],
149
+ note: 'no reference sites found — nothing to rename',
150
+ }, null, 2);
151
+ }
152
+
153
+ // DEFAULT: preview only. Never mutate on the default call. The CORE of
154
+ // the payload is the approximate-match warning plus the change-list of
155
+ // {file,line,col,context} sites that WOULD change; ok/mode/count/hint are
156
+ // secondary metadata.
157
+ if (!apply) {
158
+ return JSON.stringify({
159
+ warning: APPROX_WARNING,
160
+ changes: sites,
161
+ ok: true,
162
+ mode: 'preview',
163
+ symbol,
164
+ new_name: newName,
165
+ count: sites.length,
166
+ hint: 'Re-run with apply:true to perform the rename at these sites.',
167
+ }, null, 2);
168
+ }
169
+
170
+ // apply===true: perform the gated rewrite at the exact sanctioned spans.
171
+ const { changed, replacements, skipped } = _applyRename(sites, symbol, newName, workDir);
172
+ if (changed.length) {
173
+ try { invalidateBuiltinResultCache(changed); } catch {}
174
+ try { markCodeGraphDirtyPaths(changed); } catch {}
175
+ }
176
+ const result = {
177
+ ok: true,
178
+ mode: 'apply',
179
+ symbol,
180
+ new_name: newName,
181
+ warning: APPROX_WARNING,
182
+ sites_considered: sites.length,
183
+ replacements,
184
+ files_changed: changed.map((p) => (p.startsWith(workDir) ? p.slice(workDir.length).replace(/^[\\/]/, '') : p)),
185
+ };
186
+ if (skipped.length) {
187
+ result.skipped = skipped;
188
+ result.skipped_warning = `${skipped.length} site(s) were NOT renamed (offset out of range or the slice did not match the symbol). They were left untouched; review them manually.`;
189
+ }
190
+ return JSON.stringify(result, null, 2);
191
+ } catch (err) {
192
+ return JSON.stringify({ ok: false, error: `rename failed: ${err && err.message ? err.message : String(err)}` }, null, 2);
193
+ }
194
+ }
195
+
196
+ export default executeRenameTool;
@@ -0,0 +1,422 @@
1
+ import { spawn, execFileSync } from 'child_process';
2
+ import { existsSync } from 'node:fs';
3
+ import { join } from 'node:path';
4
+
5
+ let _rgExecutableResolved = null;
6
+
7
+ function _resolveRgExecutable() {
8
+ const isWin = process.platform === 'win32';
9
+ try {
10
+ const cmd = isWin ? 'where' : 'which';
11
+ const out = execFileSync(cmd, ['rg'], { encoding: 'utf8', windowsHide: true }).trim();
12
+ const first = out.split(/\r?\n/).map((l) => l.trim()).find(Boolean);
13
+ if (first && existsSync(first)) return first;
14
+ } catch { /* fall through to PATH scan */ }
15
+ const pathSep = isWin ? ';' : ':';
16
+ const dirs = String(process.env.PATH || '').split(pathSep).filter(Boolean);
17
+ const pathext = isWin
18
+ ? String(process.env.PATHEXT || '.EXE;.CMD;.BAT').split(';').map((e) => e.toLowerCase())
19
+ : [''];
20
+ const names = isWin ? ['rg.exe', 'rg'] : ['rg'];
21
+ for (const dir of dirs) {
22
+ for (const name of names) {
23
+ const candidate = join(dir, name);
24
+ if (!existsSync(candidate)) continue;
25
+ if (isWin) {
26
+ const ext = candidate.slice(candidate.lastIndexOf('.')).toLowerCase();
27
+ if (ext && !pathext.includes(ext)) continue;
28
+ }
29
+ return candidate;
30
+ }
31
+ }
32
+ return 'rg';
33
+ }
34
+
35
+ function rgExecutable() {
36
+ if (_rgExecutableResolved === null) {
37
+ _rgExecutableResolved = _resolveRgExecutable();
38
+ }
39
+ return _rgExecutableResolved;
40
+ }
41
+
42
+ // When _resolveRgExecutable() exhausts `where`/`which` and the PATH scan and
43
+ // falls back to the bare 'rg', a later spawn would die with a raw ENOENT. Probe
44
+ // the fallback so we can surface a clear, actionable error instead. Success is
45
+ // cached permanently; failure is cached for only RG_FALLBACK_FAIL_TTL_MS so
46
+ // installing rg mid-session recovers without restarting the daemon.
47
+ const RG_FALLBACK_FAIL_TTL_MS = 30000;
48
+ let _rgFallbackUsable = null;
49
+ let _rgFallbackFailAt = 0;
50
+ function rgFallbackUsable() {
51
+ if (_rgFallbackUsable === true) return true;
52
+ if (_rgFallbackUsable === false && (Date.now() - _rgFallbackFailAt) < RG_FALLBACK_FAIL_TTL_MS) {
53
+ return false;
54
+ }
55
+ try {
56
+ execFileSync('rg', ['--version'], { stdio: 'ignore', windowsHide: true, timeout: 3000 });
57
+ _rgFallbackUsable = true;
58
+ } catch {
59
+ _rgFallbackUsable = false;
60
+ _rgFallbackFailAt = Date.now();
61
+ }
62
+ return _rgFallbackUsable;
63
+ }
64
+
65
+ // Throws a clear error only when the bare 'rg' fallback was reached AND it is
66
+ // not actually runnable. A real resolved path short-circuits with no probe, so
67
+ // behavior is unchanged whenever rg exists.
68
+ function assertRgAvailable() {
69
+ if (rgExecutable() !== 'rg') return;
70
+ if (rgFallbackUsable()) return;
71
+ const e = new Error('ripgrep (rg) not found on PATH — install ripgrep or add it to PATH');
72
+ e.code = 'ERG_NOT_FOUND';
73
+ throw e;
74
+ }
75
+
76
+ // Cap rg stdout accumulation so a runaway producer (huge repo, accidental
77
+ // match-all) cannot balloon the JS string heap. Mirrors CC's ripgrep.ts cap.
78
+ const MAX_RG_STDOUT_BYTES = 20 * 1024 * 1024; // 20MB
79
+
80
+ // SIGTERM → grace → force kill; hard Promise settle if 'close' never fires
81
+ // (mirrors shell-command.mjs treeKill + _treeKillForceSettle timings).
82
+ const RG_KILL_GRACE_MS = 3000;
83
+ const RG_FORCE_SETTLE_SLACK_MS = 5000;
84
+
85
+ function _rgProcGone(proc) {
86
+ return !proc || proc.exitCode != null || proc.signalCode != null;
87
+ }
88
+
89
+ function _escalateRgKill(proc) {
90
+ if (_rgProcGone(proc)) return;
91
+ const pid = proc.pid;
92
+ if (!pid) return;
93
+ try {
94
+ if (process.platform === 'win32') {
95
+ spawn('taskkill', ['/pid', String(pid), '/t', '/f'], {
96
+ windowsHide: true,
97
+ stdio: 'ignore',
98
+ });
99
+ } else {
100
+ try {
101
+ proc.kill('SIGKILL');
102
+ } catch { /* ignore */ }
103
+ }
104
+ } catch { /* ignore */ }
105
+ }
106
+
107
+ function _killRgProc(proc, timers) {
108
+ if (_rgProcGone(proc)) return;
109
+ try {
110
+ proc.kill('SIGTERM');
111
+ } catch { /* ignore */ }
112
+ if (timers.killGrace) {
113
+ clearTimeout(timers.killGrace);
114
+ timers.killGrace = null;
115
+ }
116
+ timers.killGrace = setTimeout(() => {
117
+ timers.killGrace = null;
118
+ _escalateRgKill(proc);
119
+ }, RG_KILL_GRACE_MS);
120
+ if (timers.killGrace.unref) timers.killGrace.unref();
121
+ }
122
+
123
+ function _clearRgTimeoutOnly(timers) {
124
+ if (timers.timeout) {
125
+ clearTimeout(timers.timeout);
126
+ timers.timeout = null;
127
+ }
128
+ }
129
+
130
+ function _clearRgTimers(timers) {
131
+ _clearRgTimeoutOnly(timers);
132
+ if (timers.killGrace) {
133
+ clearTimeout(timers.killGrace);
134
+ timers.killGrace = null;
135
+ }
136
+ if (timers.forceSettle) {
137
+ clearTimeout(timers.forceSettle);
138
+ timers.forceSettle = null;
139
+ }
140
+ }
141
+
142
+ function _armRgForceSettle({ timeoutMs, isSettled, timers, proc, onForceSettle }) {
143
+ if (timers.forceSettle) clearTimeout(timers.forceSettle);
144
+ timers.forceSettle = setTimeout(() => {
145
+ timers.forceSettle = null;
146
+ if (isSettled()) return;
147
+ // Hard deadline: escalate immediately — do not schedule grace then clear it.
148
+ _escalateRgKill(proc);
149
+ onForceSettle();
150
+ }, timeoutMs + RG_FORCE_SETTLE_SLACK_MS);
151
+ if (timers.forceSettle.unref) timers.forceSettle.unref();
152
+ }
153
+
154
+ // Ripgrep wrapper. Ripgrep occasionally fails with EAGAIN on Windows when
155
+ // thread/resource pressure spikes. On EAGAIN we retry once with `-j 1` to
156
+ // force single-threaded execution; rg exit code 1 is "no matches" and is
157
+ // surfaced as empty stdout so callers can render "(no matches)" uniformly.
158
+ function spawnRg(argsList, execOptions) {
159
+ const timeoutMs = Number(execOptions?.timeout ?? 20000);
160
+ return new Promise((resolve, reject) => {
161
+ const proc = spawn(rgExecutable(), argsList, {
162
+ cwd: execOptions?.cwd,
163
+ env: execOptions?.env || process.env,
164
+ windowsHide: true,
165
+ stdio: ['ignore', 'pipe', 'pipe'],
166
+ });
167
+ let stdout = '';
168
+ let stderr = '';
169
+ let timedOut = false;
170
+ /** @type {'timeout' | 'cap' | null} */
171
+ let killReason = null;
172
+ let settled = false;
173
+ let stdoutBytes = 0;
174
+ let stdoutTruncated = false;
175
+ const timers = { timeout: null, killGrace: null, forceSettle: null };
176
+ timers.timeout = setTimeout(() => {
177
+ timedOut = true;
178
+ killReason = 'timeout';
179
+ _killRgProc(proc, timers);
180
+ }, timeoutMs);
181
+ if (timers.timeout.unref) timers.timeout.unref();
182
+ _armRgForceSettle({
183
+ timeoutMs,
184
+ isSettled: () => settled,
185
+ timers,
186
+ proc,
187
+ onForceSettle: () => {
188
+ timedOut = true;
189
+ killReason = 'timeout';
190
+ const e = new Error(`rg timed out after ${timeoutMs} ms`);
191
+ e.code = 'ETIMEDOUT';
192
+ if (settled) return;
193
+ settled = true;
194
+ _clearRgTimers(timers);
195
+ reject(e);
196
+ },
197
+ });
198
+ proc.stdout.setEncoding('utf-8');
199
+ proc.stderr.setEncoding('utf-8');
200
+ proc.stdout.on('data', (d) => {
201
+ if (stdoutTruncated) return;
202
+ // Account by UTF-8 byte length (d is a decoded string here), not by
203
+ // string length, so non-ASCII output cannot overshoot the byte cap.
204
+ const dBytes = Buffer.byteLength(d);
205
+ const remaining = MAX_RG_STDOUT_BYTES - stdoutBytes;
206
+ if (dBytes >= remaining) {
207
+ // Slice by BYTES (not chars) and cut on a UTF-8 codepoint
208
+ // boundary: a char-count slice overshoots on multibyte output,
209
+ // and even a raw byte slice would let a split trailing codepoint
210
+ // decode to U+FFFD (3 bytes) and exceed the cap. Back `end` over
211
+ // continuation bytes (0b10xxxxxx) so the kept bytes are <= cap.
212
+ const buf = Buffer.from(d, 'utf8');
213
+ let end = Math.max(0, Math.min(remaining, buf.length));
214
+ while (end > 0 && end < buf.length && (buf[end] & 0xC0) === 0x80) end--;
215
+ stdout += buf.subarray(0, end).toString('utf8');
216
+ stdoutBytes = MAX_RG_STDOUT_BYTES;
217
+ stdoutTruncated = true;
218
+ if (killReason !== 'timeout') killReason = 'cap';
219
+ _clearRgTimeoutOnly(timers);
220
+ _killRgProc(proc, timers);
221
+ return;
222
+ }
223
+ stdout += d;
224
+ stdoutBytes += dBytes;
225
+ });
226
+ proc.stderr.on('data', (d) => { stderr += d; });
227
+ proc.on('error', (err) => {
228
+ if (settled) return;
229
+ settled = true;
230
+ _clearRgTimers(timers);
231
+ reject(err);
232
+ });
233
+ proc.on('close', (code) => {
234
+ if (settled) return;
235
+ settled = true;
236
+ _clearRgTimers(timers);
237
+ if (timedOut && killReason === 'timeout') {
238
+ const e = new Error(`rg timed out after ${timeoutMs} ms`);
239
+ e.code = 'ETIMEDOUT';
240
+ return reject(e);
241
+ }
242
+ const wrap = (s) => {
243
+ if (!stdoutTruncated) return s;
244
+ const boxed = new String(s);
245
+ try { boxed.truncated = true; } catch { /* ignore */ }
246
+ return boxed;
247
+ };
248
+ const boxPartialStdout = (s, rgStderrText) => {
249
+ const boxed = new String(s);
250
+ try {
251
+ if (stdoutTruncated) boxed.truncated = true;
252
+ boxed.partial = true;
253
+ boxed.rgStderr = rgStderrText;
254
+ } catch { /* ignore */ }
255
+ return boxed;
256
+ };
257
+ if (code === 0) return resolve(wrap(stdout));
258
+ if (code === 1) return resolve(wrap(''));
259
+ // SIGTERM after our own truncation kill: surface accumulated stdout.
260
+ if (stdoutTruncated) return resolve(wrap(stdout));
261
+ // Exit 2 (e.g. permission denied on some paths) may still emit matches.
262
+ if (code === 2 && stdout.length > 0) {
263
+ return resolve(boxPartialStdout(stdout, stderr.trim()));
264
+ }
265
+ const e = new Error(`rg exited with code ${code}: ${stderr.trim()}`);
266
+ e.code = code;
267
+ e.stderr = stderr;
268
+ reject(e);
269
+ });
270
+ });
271
+ }
272
+
273
+ export async function runRg(argsList, execOptions = {}) {
274
+ assertRgAvailable();
275
+ try {
276
+ return await spawnRg(argsList, execOptions);
277
+ } catch (err) {
278
+ const msg = String(err?.message || err?.stderr || '');
279
+ if (/EAGAIN/i.test(msg) && !argsList.includes('-j')) {
280
+ return spawnRg(['-j', '1', ...argsList], execOptions);
281
+ }
282
+ throw err;
283
+ }
284
+ }
285
+
286
+ function spawnRgWindowedLines(argsList, execOptions, opts = {}) {
287
+ const timeoutMs = Number(execOptions?.timeout ?? 20000);
288
+ const offset = Math.max(0, Math.floor(Number(opts.offset) || 0));
289
+ const lineLimit = Number.isFinite(opts.limit) ? Math.max(0, Math.floor(Number(opts.limit) || 0)) : Infinity;
290
+ const summaryLimit = Math.max(0, Math.floor(Number(opts.summaryLimit) || 0));
291
+ const collectLimit = lineLimit === Infinity ? Infinity : Math.max(lineLimit, summaryLimit);
292
+ return new Promise((resolve, reject) => {
293
+ const proc = spawn(rgExecutable(), argsList, {
294
+ cwd: execOptions?.cwd,
295
+ env: execOptions?.env || process.env,
296
+ windowsHide: true,
297
+ stdio: ['ignore', 'pipe', 'pipe'],
298
+ });
299
+ let buffer = '';
300
+ let stderr = '';
301
+ let skipped = 0;
302
+ let seenAfterOffset = 0;
303
+ let stoppedEarly = false;
304
+ let timedOut = false;
305
+ /** @type {'timeout' | 'early' | null} */
306
+ let killReason = null;
307
+ let settled = false;
308
+ const timers = { timeout: null, killGrace: null, forceSettle: null };
309
+ const lines = [];
310
+ timers.timeout = setTimeout(() => {
311
+ timedOut = true;
312
+ killReason = 'timeout';
313
+ _killRgProc(proc, timers);
314
+ }, timeoutMs);
315
+ if (timers.timeout.unref) timers.timeout.unref();
316
+ _armRgForceSettle({
317
+ timeoutMs,
318
+ isSettled: () => settled,
319
+ timers,
320
+ proc,
321
+ onForceSettle: () => {
322
+ timedOut = true;
323
+ killReason = 'timeout';
324
+ const e = new Error(`rg timed out after ${timeoutMs} ms`);
325
+ e.code = 'ETIMEDOUT';
326
+ if (settled) return;
327
+ settled = true;
328
+ _clearRgTimers(timers);
329
+ reject(e);
330
+ },
331
+ });
332
+ const stopEarly = () => {
333
+ if (stoppedEarly) return;
334
+ stoppedEarly = true;
335
+ buffer = '';
336
+ if (killReason !== 'timeout') killReason = 'early';
337
+ _clearRgTimeoutOnly(timers);
338
+ _killRgProc(proc, timers);
339
+ };
340
+ const pushLine = (raw) => {
341
+ if (stoppedEarly || raw.length === 0) return;
342
+ const line = raw.endsWith('\r') ? raw.slice(0, -1) : raw;
343
+ if (line.length === 0) return;
344
+ if (skipped < offset) {
345
+ skipped++;
346
+ return;
347
+ }
348
+ seenAfterOffset++;
349
+ if (seenAfterOffset <= collectLimit) {
350
+ lines.push(line);
351
+ return;
352
+ }
353
+ stopEarly();
354
+ };
355
+ proc.stdout.setEncoding('utf-8');
356
+ proc.stderr.setEncoding('utf-8');
357
+ proc.stdout.on('data', (chunk) => {
358
+ if (stoppedEarly) return;
359
+ buffer += chunk;
360
+ let start = 0;
361
+ let idx = buffer.indexOf('\n', start);
362
+ while (!stoppedEarly && idx !== -1) {
363
+ pushLine(buffer.slice(start, idx));
364
+ start = idx + 1;
365
+ idx = buffer.indexOf('\n', start);
366
+ }
367
+ buffer = stoppedEarly ? '' : buffer.slice(start);
368
+ });
369
+ proc.stderr.on('data', (d) => {
370
+ if (stderr.length < 4096) stderr += d;
371
+ });
372
+ proc.on('error', (err) => {
373
+ if (settled) return;
374
+ settled = true;
375
+ _clearRgTimers(timers);
376
+ reject(err);
377
+ });
378
+ proc.on('close', (code) => {
379
+ if (settled) return;
380
+ settled = true;
381
+ _clearRgTimers(timers);
382
+ if (timedOut && killReason === 'timeout') {
383
+ const e = new Error(`rg timed out after ${timeoutMs} ms`);
384
+ e.code = 'ETIMEDOUT';
385
+ return reject(e);
386
+ }
387
+ if (!stoppedEarly && buffer.length > 0) pushLine(buffer);
388
+ if (stoppedEarly) {
389
+ return resolve({ lines, complete: false, totalSeen: seenAfterOffset });
390
+ }
391
+ if (code === 0 || code === 1) {
392
+ return resolve({ lines, complete: true, totalSeen: seenAfterOffset });
393
+ }
394
+ if (code === 2 && lines.length > 0) {
395
+ return resolve({
396
+ lines,
397
+ complete: false,
398
+ totalSeen: seenAfterOffset,
399
+ partial: true,
400
+ rgStderr: stderr.trim(),
401
+ });
402
+ }
403
+ const e = new Error(`rg exited with code ${code}: ${stderr.trim()}`);
404
+ e.code = code;
405
+ e.stderr = stderr;
406
+ reject(e);
407
+ });
408
+ });
409
+ }
410
+
411
+ export async function runRgWindowedLines(argsList, execOptions = {}, opts = {}) {
412
+ assertRgAvailable();
413
+ try {
414
+ return await spawnRgWindowedLines(argsList, execOptions, opts);
415
+ } catch (err) {
416
+ const msg = String(err?.message || err?.stderr || '');
417
+ if (/EAGAIN/i.test(msg) && !argsList.includes('-j')) {
418
+ return spawnRgWindowedLines(['-j', '1', ...argsList], execOptions, opts);
419
+ }
420
+ throw err;
421
+ }
422
+ }