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,298 @@
1
+ // cwd-tool.mjs — session-cwd management MCP tool.
2
+ //
3
+ // Backs the `cwd` tool exposed via tools.json. Three actions:
4
+ //
5
+ // get — report the current effective cwd (pwd()) plus its resolved
6
+ // project_id (via resolveProjectId).
7
+ // set — validate `args.path` against the filesystem and stash the
8
+ // canonical absolute form into process.env.MIXDOG_SESSION_CWD.
9
+ // captureOriginalUserCwd() consults that env var first, so
10
+ // every downstream tool (read/grep/glob/find_symbol/bash/...)
11
+ // picks up the change without a per-tool plumbing change.
12
+ // list — comprehensive `.git` repository scan rooted at $HOME and
13
+ // the rawUserCwd() session sentinel. Cached in-process; pass
14
+ // `refresh:true` to force a rebuild.
15
+ //
16
+ // list policy:
17
+ // - depth-bounded walk (maxDepth 4) using the shared walkDir helper.
18
+ // - prunes NOISE_DIR_NAMES + a small AppData cache-dir set to keep
19
+ // the scan responsive on Windows.
20
+ // - once a directory is identified as a `.git`-bearing repo, it is
21
+ // recorded and its children are NOT descended (no nested worktree
22
+ // scans, no submodule recursion).
23
+
24
+ import { resolve as pathResolve, join as pathJoin } from 'path'
25
+ import { homedir } from 'os'
26
+ import { existsSync, statSync, readFileSync, writeFileSync } from 'fs'
27
+ import { execFileSync } from 'child_process'
28
+ import { pwd, rawUserCwd, writeLastSessionCwd } from '../../../shared/user-cwd.mjs'
29
+ import { resolveOptionalCwd } from './builtin/cwd-utils.mjs'
30
+ import { walkDir, NOISE_DIR_NAMES } from './builtin/glob-walk.mjs'
31
+ import { resolveProjectId } from '../../../memory/lib/project-id-resolver.mjs'
32
+ import { resolvePluginData } from '../../../shared/plugin-paths.mjs'
33
+
34
+ // Canonical absolute-path form. Mirrors code-graph's `_canonicalGraphCwd`:
35
+ // resolve to absolute, lower-case on win32 so case drift in user input
36
+ // does not produce duplicate session-cwd values.
37
+ function _canonicalSessionCwd(cwd) {
38
+ const full = pathResolve(cwd)
39
+ return process.platform === 'win32' ? full.toLowerCase() : full
40
+ }
41
+
42
+ // Extra AppData cache dir names commonly found under %USERPROFILE% on
43
+ // Windows. They are not in NOISE_DIR_NAMES because they're user-data
44
+ // roots elsewhere, but we never want to spider them during a repo scan.
45
+ const EXTRA_NOISE_DIR_NAMES = new Set([
46
+ 'AppData', 'Application Data', 'Local Settings',
47
+ '.npm', '.yarn', '.pnpm-store', '.nuget', '.cargo', '.rustup',
48
+ '.gradle', '.m2', '.ivy2', '.android', '.docker', '.vscode',
49
+ '.cursor', 'Library',
50
+ // NOTE: do NOT prune '.claude' — the mixdog repo lives under
51
+ // ~/.claude/plugins/marketplaces/trib-plugin (depth 4 from home), so
52
+ // pruning it would hide the user's primary project from `list`. The
53
+ // maxDepth:4 cap + stop-at-.git keep the deep version cache
54
+ // (~/.claude/plugins/cache/.../0.5.x = depth 6) out of the scan.
55
+ ])
56
+
57
+ const PRUNED_DIR_NAMES = new Set([...NOISE_DIR_NAMES, ...EXTRA_NOISE_DIR_NAMES])
58
+
59
+ let _listCache = null
60
+
61
+ // Per-scan set used by the visitor to mark repos whose subtree must not
62
+ // be descended. walkDir does not natively support "stop descent into
63
+ // this dir but keep the walk going", so we approximate by checking
64
+ // ancestry on every visit and bailing early when the entry lives under
65
+ // a previously-recorded repo.
66
+ const _recordedRepos = new Set()
67
+
68
+ function _scanReposUnderFiltered(root) {
69
+ _recordedRepos.clear()
70
+ const repos = []
71
+ if (!root || !existsSync(root)) return repos
72
+ try {
73
+ const st = statSync(root)
74
+ if (!st.isDirectory()) return repos
75
+ } catch { return repos }
76
+
77
+ if (existsSync(`${root}/.git`)) {
78
+ repos.push(root)
79
+ return repos
80
+ }
81
+
82
+ walkDir(root, {
83
+ hidden: true,
84
+ maxDepth: 4,
85
+ excludeDirNames: PRUNED_DIR_NAMES,
86
+ visit: (ent, entPath) => {
87
+ if (!ent.isDirectory()) return
88
+ // Skip subtree of a previously-recorded repo. walkDir invokes
89
+ // visit before recursing, so flipping ent.isDirectory() here
90
+ // would not stop descent. Instead, check ancestry on every
91
+ // visit and bail early.
92
+ for (const recorded of _recordedRepos) {
93
+ if (entPath === recorded || entPath.startsWith(recorded + '/') || entPath.startsWith(recorded + '\\')) {
94
+ return
95
+ }
96
+ }
97
+ try {
98
+ if (existsSync(`${entPath}/.git`)) {
99
+ repos.push(entPath)
100
+ _recordedRepos.add(entPath)
101
+ }
102
+ } catch { /* ignore stat races */ }
103
+ },
104
+ })
105
+
106
+ return repos
107
+ }
108
+
109
+ // ── Owned-project filtering (config: `cwd` section) ──────────────────
110
+ // `cwd list` scans ONLY under the session entry directory (rawUserCwd) plus
111
+ // any extraRoots — never $HOME. Claude-Code-managed plugin clones live under
112
+ // ~/.claude/plugins (i.e. $HOME), so they fall outside the scan with no
113
+ // manifest/path heuristic. Repos are then kept by ONE explicit signal:
114
+ // - identityEmails: a repo is kept only when >=1 commit is authored by one
115
+ // of these. Empty set = no filter (fail open) so the list never silently
116
+ // goes blank.
117
+ // extraRoots widens the scan beyond the session cwd.
118
+ function _cwdConfig() {
119
+ // Read the `cwd` section directly (no config.mjs import: its module-load
120
+ // resolvePluginData() throws when CLAUDE_PLUGIN_DATA is unset, which would
121
+ // break tool-manifest builds that merely import this file for its defs).
122
+ let raw = {}
123
+ try {
124
+ const all = JSON.parse(readFileSync(pathJoin(resolvePluginData(), 'mixdog-config.json'), 'utf8'))
125
+ if (all && typeof all.cwd === 'object' && all.cwd) raw = all.cwd
126
+ } catch { raw = {} }
127
+ const identityEmails = Array.isArray(raw.identityEmails)
128
+ ? raw.identityEmails.filter(e => typeof e === 'string' && e.trim()).map(e => e.trim())
129
+ : []
130
+ const extraRoots = Array.isArray(raw.extraRoots)
131
+ ? raw.extraRoots.filter(r => typeof r === 'string' && r.trim()).map(r => r.trim())
132
+ : []
133
+ return { identityEmails, extraRoots }
134
+ }
135
+
136
+ function _escapeRe(s) {
137
+ return s.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')
138
+ }
139
+
140
+ // True when >=1 commit across all refs is authored by one of the identity
141
+ // emails. One git spawn per repo; multiple --author flags are OR-combined
142
+ // by git, and -1 exits on the first match.
143
+ function _authoredByIdentity(repoPath, emails) {
144
+ try {
145
+ const args = ['-C', repoPath, 'log', '--all', '-1', '--format=%H']
146
+ for (const e of emails) { args.push('--author', _escapeRe(e)) }
147
+ const out = execFileSync('git', args, { encoding: 'utf8', timeout: 5000, stdio: ['ignore', 'pipe', 'ignore'], windowsHide: true })
148
+ return out.trim().length > 0
149
+ } catch {
150
+ return false
151
+ }
152
+ }
153
+
154
+ function _buildRepoList(includeAll = false) {
155
+ const cfg = _cwdConfig()
156
+ const user = rawUserCwd()
157
+ const roots = []
158
+ const seen = new Set()
159
+ // Scan ONLY the session entry dir (+ extraRoots), never $HOME — this keeps
160
+ // Claude-Code-managed plugin clones under ~/.claude/plugins out of the scan
161
+ // entirely, with no manifest/path heuristic needed.
162
+ for (const r of [user, ...cfg.extraRoots]) {
163
+ if (!r) continue
164
+ const key = _canonicalSessionCwd(r)
165
+ if (seen.has(key)) continue
166
+ seen.add(key)
167
+ roots.push(r)
168
+ }
169
+
170
+ const repoSet = new Map() // canonical → original path
171
+ for (const root of roots) {
172
+ for (const repo of _scanReposUnderFiltered(root)) {
173
+ const key = _canonicalSessionCwd(repo)
174
+ if (!repoSet.has(key)) repoSet.set(key, repo)
175
+ }
176
+ }
177
+
178
+ let paths = [...repoSet.values()]
179
+ // Keep only repos the user has actually committed to (identityEmails). Empty
180
+ // set = no filter (fail open). No plugin filtering needed — plugin clones
181
+ // live under $HOME, which is no longer scanned.
182
+ if (!includeAll && cfg.identityEmails.length) {
183
+ paths = paths.filter(p => _authoredByIdentity(p, cfg.identityEmails))
184
+ }
185
+
186
+ const out = paths.map(repoPath => ({ path: repoPath, projectId: resolveProjectId(repoPath) }))
187
+ // Stable ordering: alphabetical by path.
188
+ out.sort((a, b) => a.path.localeCompare(b.path))
189
+ // Persist the filtered list so the SessionStart hook can inject a
190
+ // "## Projects" block without re-running the scan. Best-effort.
191
+ if (!includeAll) {
192
+ try { writeFileSync(pathJoin(resolvePluginData(), 'cwd-projects.json'), JSON.stringify({ projects: out })) } catch { /* best-effort */ }
193
+ }
194
+ return out
195
+ }
196
+
197
+ function _formatList(entries) {
198
+ if (entries.length === 0) return '[cwd] no .git repositories found under home / user-cwd roots'
199
+ const lines = entries.map(e => {
200
+ const pid = e.projectId ? ` project=${e.projectId}` : ' project=<unset>'
201
+ return `${e.path}${pid}`
202
+ })
203
+ return `[cwd] ${entries.length} repo${entries.length === 1 ? '' : 's'}:\n` + lines.join('\n')
204
+ }
205
+
206
+ /**
207
+ * MCP entry point. Returns a plain string; the server wraps it into the
208
+ * standard `{content:[{type:'text',text:...}]}` envelope.
209
+ */
210
+ export async function executeCwdTool(name, args, callerCwd, opts = {}) {
211
+ const action = (typeof args?.action === 'string' && args.action) || 'get'
212
+
213
+ if (action === 'get') {
214
+ const effective = opts.session ? opts.session.resolveCwd() : pwd()
215
+ const projectId = resolveProjectId(effective)
216
+ return `[cwd] effective=${effective}\nproject=${projectId || '<unset>'}`
217
+ }
218
+
219
+ if (action === 'set') {
220
+ const baseCwd = (typeof callerCwd === 'string' && callerCwd) ? callerCwd : pwd()
221
+ const resolved = resolveOptionalCwd(args?.path, baseCwd)
222
+ if (resolved.error) {
223
+ return resolved.error
224
+ }
225
+ // resolveOptionalCwd returns baseCwd when args.path is empty/missing;
226
+ // that is an explicit "no path provided" failure for set.
227
+ if (!args?.path || (typeof args.path === 'string' && args.path.trim() === '')) {
228
+ return '[cwd] set requires a non-empty `path` argument'
229
+ }
230
+ const canonical = _canonicalSessionCwd(resolved.cwd)
231
+ // Per-session cwd write. In daemon mode opts.session stores it on the
232
+ // session (no cross-session leak); otherwise fall back to the global env.
233
+ if (opts.session) opts.session.setCwd(canonical)
234
+ else process.env.MIXDOG_SESSION_CWD = canonical
235
+ // Key the per-terminal sentinel by THIS connection's leadPid so a shared
236
+ // daemon (one process, N terminals) keeps each terminal's `cwd set` in its
237
+ // own session-cwd-<leadPid>.txt. Falls back to MIXDOG_SUPERVISOR_PID when
238
+ // no session leadPid is present (boot/stdio path — behaviour-identical).
239
+ writeLastSessionCwd(canonical, opts.session?.leadPid)
240
+ // ③ Fire-and-forget code-graph prewarm for the new cwd so the first
241
+ // unscoped find_symbol/references/callers hits a warm cache instead of the
242
+ // cold build on the query's critical path. Guarded inside
243
+ // prewarmCodeGraphIfProject: only validated project roots are prewarmed
244
+ // (arbitrary trees skipped), and it prewarms the detected ROOT — matching
245
+ // the re-root executeCodeGraphTool applies. Dynamic import avoids a static
246
+ // dependency on the heavy code-graph module; never blocks or fails the set.
247
+ import('./code-graph.mjs')
248
+ .then((m) => { try { m.prewarmCodeGraphIfProject?.(canonical) } catch { /* best-effort */ } })
249
+ .catch(() => { /* code-graph unavailable — skip prewarm */ })
250
+ // Invalidate the in-process repo list — set() typically follows a
251
+ // list() call when the user picks one of its entries.
252
+ _listCache = null
253
+ const projectId = resolveProjectId(canonical)
254
+ return `[cwd] set effective=${canonical}\nproject=${projectId || '<unset>'}`
255
+ }
256
+
257
+ if (action === 'list') {
258
+ // `all:true` bypasses the owned-project filter and transient prune — a
259
+ // raw scan of every .git repo under the roots. Not cached (rare).
260
+ if (args?.all === true) return _formatList(_buildRepoList(true))
261
+ if (args?.refresh === true) _listCache = null
262
+ if (!_listCache) _listCache = _buildRepoList(false)
263
+ return _formatList(_listCache)
264
+ }
265
+
266
+ return `[cwd] unknown action="${action}" (expected get|set|list)`
267
+ }
268
+
269
+ // Canonical tool definition consumed by dev/scripts/build-tools-manifest.mjs.
270
+ // The handler lives above; this array is the source for the `cwd` entry in
271
+ // tools.json. The `module: 'cwd'` field is injected at build time, so it is
272
+ // intentionally absent here (mirrors every other module's source array).
273
+ export const CWD_TOOL_DEFS = [
274
+ {
275
+ name: 'cwd',
276
+ title: 'Session CWD',
277
+ annotations: { title: 'Session CWD', readOnlyHint: false, destructiveHint: false, idempotentHint: true, openWorldHint: false },
278
+ description: 'Session cwd for relative paths (not a sandbox): get/set/list. set updates session-wide cwd for tools and bridge workers — set on project switch. list: repos you commit to (cwd.identityEmails), minus plugin-cache/codex; all:true for full scan.',
279
+ inputSchema: {
280
+ type: 'object',
281
+ properties: {
282
+ action: { type: 'string', enum: ['get', 'set', 'list'] },
283
+ path: { type: 'string' },
284
+ refresh: { type: 'boolean' },
285
+ all: { type: 'boolean' },
286
+ },
287
+ },
288
+ },
289
+ ]
290
+
291
+ // Background-warm the default (filtered) repo list shortly after module
292
+ // load so the first `cwd list` returns an already-built cache. Deferred
293
+ // and unref'd so it never blocks startup or keeps the process alive.
294
+ try {
295
+ setTimeout(() => {
296
+ try { if (!_listCache) _listCache = _buildRepoList(false) } catch { /* best-effort */ }
297
+ }, 1000).unref?.()
298
+ } catch { /* environments without setTimeout */ }
@@ -0,0 +1,323 @@
1
+ 'use strict';
2
+ // Destructive command detector.
3
+ //
4
+ // Returns a short human-readable warning string when the command matches
5
+ // a known data-loss / hard-to-reverse pattern. Purely informational —
6
+ // the caller (case 'bash' in builtin.mjs) prepends the warning to the result
7
+ // envelope so the agent sees the risk inline. Does NOT block execution;
8
+ // hard blocks remain in BLOCKED_PATTERNS in builtin.mjs / bash-session.mjs.
9
+
10
+ import { SHELL_NAMES as _SHELL_NAMES, WRAPPER_NAMES as _WRAPPER_NAMES } from './shell-policy.mjs';
11
+ import { extractPowerShellCommandInner } from './shell-command.mjs';
12
+
13
+ export function stripQuotedAndHeredoc(s) { return _stripQuotedSpans(s); }
14
+ export function extractShellCInner(s) { return _extractShellCInner(s); }
15
+
16
+ function _stripQuotedSpans(s) {
17
+ return String(s || '')
18
+ .replace(/<<-?\s*['"]?(\w+)['"]?[\s\S]*?\n\1\b/g, '<<HEREDOC>>')
19
+ .replace(/'[^']*'/g, "''")
20
+ .replace(/"(?:[^"\\]|\\.)*"/g, '""')
21
+ .replace(/#[^\n]*/g, '');
22
+ }
23
+
24
+ // Heredoc bodies (shell-exec contexts route the body through scanning).
25
+ function _extractHeredocBodies(s) {
26
+ const out = [];
27
+ const re = /<<-?\s*['"]?(\w+)['"]?([\s\S]*?)\n\1\b/g;
28
+ let m;
29
+ while ((m = re.exec(String(s || ''))) !== null) out.push(m[2]);
30
+ return out;
31
+ }
32
+
33
+ // Shell-aware tokenizer. Quoted spans stay intact; separators become
34
+ // their own token so callers can split pipeline segments cleanly.
35
+ function _tokenize(s) {
36
+ const t = [], src = String(s || '');
37
+ let cur = '', i = 0;
38
+ while (i < src.length) {
39
+ const c = src[i];
40
+ if (c === "'") { const e = src.indexOf("'", i + 1); if (e === -1) { cur += src.slice(i); break; } cur += src.slice(i, e + 1); i = e + 1; continue; }
41
+ if (c === '"') { let j = i + 1; while (j < src.length) { if (src[j] === '\\' && j + 1 < src.length) { j += 2; continue; } if (src[j] === '"') break; j++; } if (j >= src.length) { cur += src.slice(i); break; } cur += src.slice(i, j + 1); i = j + 1; continue; }
42
+ if (c === '\\' && i + 1 < src.length) { cur += src[i] + src[i + 1]; i += 2; continue; }
43
+ if (c === ' ' || c === '\t') { if (cur) { t.push(cur); cur = ''; } i++; continue; }
44
+ if (c === '\n' || c === ';') { if (cur) { t.push(cur); cur = ''; } t.push(c); i++; continue; }
45
+ if (c === '&' || c === '|') { if (cur) { t.push(cur); cur = ''; } if (src[i + 1] === c) { t.push(c + c); i += 2; } else { t.push(c); i++; } continue; }
46
+ cur += c; i++;
47
+ }
48
+ if (cur) t.push(cur);
49
+ return t;
50
+ }
51
+
52
+ function _splitSegments(tokens) {
53
+ const segs = [], SEP = new Set([';', '&', '&&', '|', '||', '\n']);
54
+ let cur = [];
55
+ for (const t of tokens) { if (SEP.has(t)) { if (cur.length) segs.push(cur); cur = []; } else cur.push(t); }
56
+ if (cur.length) segs.push(cur);
57
+ return segs;
58
+ }
59
+
60
+ function _stripQuotes(t) {
61
+ if (t.length >= 2) { const a = t[0], b = t[t.length - 1]; if ((a === "'" && b === "'") || (a === '"' && b === '"')) return t.slice(1, -1); }
62
+ return t;
63
+ }
64
+
65
+ // Skip env-var assignments and known wrapper commands (with their
66
+ // option arguments) so the underlying program reaches classification.
67
+ function _peelWrappers(tokens) {
68
+ let i = 0;
69
+ while (i < tokens.length) {
70
+ const t = tokens[i];
71
+ if (/^[A-Za-z_][A-Za-z0-9_]*=/.test(t)) { i++; continue; }
72
+ if (_WRAPPER_NAMES.has(t)) {
73
+ i++;
74
+ while (i < tokens.length && (/^[-+]/.test(tokens[i]) || /^\d+[smhd]?$/.test(tokens[i]) || /^\d+m\d+s?$/.test(tokens[i]))) {
75
+ // Long option `--key value` — consume the value too when not `=`-joined and the value is non-flag / non-assignment.
76
+ if (/^--[A-Za-z0-9][\w-]*$/.test(tokens[i]) && i + 1 < tokens.length && !/^[-+]/.test(tokens[i + 1]) && !/^[A-Za-z_][A-Za-z0-9_]*=/.test(tokens[i + 1])) { i += 2; continue; }
77
+ i++;
78
+ }
79
+ continue;
80
+ }
81
+ break;
82
+ }
83
+ return tokens.slice(i);
84
+ }
85
+
86
+ // Extract `-c <payload>` (and `--command <payload>`) for shell invocations.
87
+ // Walks options including combined short flags like `-lc`, `-ic`, `-ec`.
88
+ function _extractShellCInner(s) {
89
+ const out = [];
90
+ const tokens = _tokenize(s);
91
+ const segs = _splitSegments(tokens);
92
+ for (const seg of segs) {
93
+ const peeled = _peelWrappers(seg);
94
+ if (!peeled.length) continue;
95
+ if (!_SHELL_NAMES.has(peeled[0])) {
96
+ // Extended: leaf commands that embed an inner command/payload.
97
+ // The caller re-scans each entry via isBlockedCommand, so we just
98
+ // surface the payload as a synthetic command string.
99
+ const leaf = peeled[0];
100
+ if (leaf === 'xargs' || leaf === 'parallel') {
101
+ let j = 1;
102
+ while (j < peeled.length && /^[-+]/.test(peeled[j])) {
103
+ if (/^--[A-Za-z0-9][\w-]*$/.test(peeled[j]) && j + 1 < peeled.length && !/^[-+]/.test(peeled[j + 1])) { j += 2; continue; }
104
+ j++;
105
+ }
106
+ if (j < peeled.length) out.push(peeled.slice(j).map(_stripQuotes).join(' '));
107
+ } else if (leaf === 'find') {
108
+ for (let j = 1; j < peeled.length; j++) {
109
+ if (peeled[j] === '-exec' || peeled[j] === '-execdir') {
110
+ const cmd = [];
111
+ let k = j + 1;
112
+ while (k < peeled.length && peeled[k] !== ';' && peeled[k] !== '\\;' && peeled[k] !== '+') { cmd.push(_stripQuotes(peeled[k])); k++; }
113
+ if (cmd.length) out.push(cmd.join(' '));
114
+ j = k;
115
+ }
116
+ }
117
+ } else if (leaf === 'awk') {
118
+ for (let j = 1; j < peeled.length; j++) {
119
+ const t = peeled[j];
120
+ if (t === '-f' || t === '-v' || t === '-F') { j++; continue; }
121
+ if (t.startsWith('-')) continue;
122
+ const body = _stripQuotes(t);
123
+ const m = body.match(/system\s*\(\s*(['"])([\s\S]*?)\1\s*\)/);
124
+ if (m) out.push(m[2]);
125
+ break;
126
+ }
127
+ } else if (leaf === 'perl' || leaf === 'node' || leaf === 'python' || leaf === 'python3') {
128
+ const flag = (leaf === 'python' || leaf === 'python3') ? '-c' : '-e';
129
+ for (let j = 1; j < peeled.length; j++) {
130
+ const t = peeled[j];
131
+ if (t === flag || (leaf === 'perl' && t === '-E')) { const arg = peeled[j + 1]; if (arg) out.push(_stripQuotes(arg)); break; }
132
+ if (t.startsWith('-')) continue;
133
+ break;
134
+ }
135
+ }
136
+ continue;
137
+ }
138
+ for (let i = 1; i < peeled.length; i++) {
139
+ const t = peeled[i];
140
+ if (t === '-c' || t === '--command' || /^-[a-zA-Z]+c$/.test(t)) {
141
+ const arg = peeled[i + 1];
142
+ if (arg) out.push(_stripQuotes(arg));
143
+ break;
144
+ }
145
+ if (t === '--rcfile' || t === '--init-file' || t === '-O' || t === '+O') { i++; continue; }
146
+ if (t.startsWith('-') || t.startsWith('+')) continue;
147
+ break;
148
+ }
149
+ }
150
+ return out;
151
+ }
152
+
153
+ // rm: detect -r/-R/--recursive AND -f/--force across split or combined
154
+ // short-flag tokens.
155
+ function _classifyRm(args) {
156
+ let r = false, f = false;
157
+ for (const t of args) {
158
+ if (t === '--') break;
159
+ if (t === '--recursive' || t === '-r' || t === '-R') { r = true; continue; }
160
+ if (t === '--force' || t === '-f') { f = true; continue; }
161
+ if (/^-[a-zA-Z]+$/.test(t)) { if (/[rR]/.test(t)) r = true; if (/f/.test(t)) f = true; continue; }
162
+ if (t.startsWith('-')) continue;
163
+ break;
164
+ }
165
+ if (r && f) return 'may recursively force-remove files';
166
+ if (r) return 'may recursively remove files';
167
+ if (f) return 'may force-remove files';
168
+ return null;
169
+ }
170
+
171
+ // PowerShell Remove-Item (and aliases): -Recurse/-Force, including :$true forms.
172
+ // Parity with shell-policy.mjs _removeItemRecursiveForceUnsafe switch matching.
173
+ const _PS_REMOVE_CMDS = new Set(['remove-item', 'ri', 'del', 'erase', 'rd', 'rmdir']);
174
+
175
+ function _classifyRemoveItem(args) {
176
+ let r = false, f = false;
177
+ for (const t of args) {
178
+ if (t === '--') break;
179
+ const low = String(t).toLowerCase();
180
+ if (low.startsWith('-')) {
181
+ if (/^-r(ec(urse)?)?$/.test(low) || /^-r(ec(urse)?)?:\$true$/i.test(low)) { r = true; continue; }
182
+ if (/^-fo(rce)?$/.test(low) || /^-fo(rce)?:\$true$/i.test(low)) { f = true; continue; }
183
+ if (low === '-recurse' || low === '-recursive') { r = true; continue; }
184
+ if (low === '-force') { f = true; continue; }
185
+ continue;
186
+ }
187
+ break;
188
+ }
189
+ if (r && f) return 'may recursively force-remove files';
190
+ if (r) return 'may recursively remove files';
191
+ if (f) return 'may force-remove files';
192
+ return null;
193
+ }
194
+
195
+ // git: skip global options (-C path, -c key=val, --git-dir=..., --no-pager
196
+ // etc.) before reading the subcommand.
197
+ function _classifyGit(args) {
198
+ let i = 0;
199
+ while (i < args.length) {
200
+ const t = args[i];
201
+ if (t === '-C' || t === '-c') { i += 2; continue; }
202
+ if (/^--(git-dir|work-tree|namespace)(=|$)/.test(t)) { i += t.includes('=') ? 1 : 2; continue; }
203
+ if (t === '--no-pager' || t === '--paginate' || t === '--bare' || t === '--exec-path' || /^--literal-pathspecs|--glob-pathspecs|--noglob-pathspecs|--icase-pathspecs$/.test(t)) { i++; continue; }
204
+ if (/^--exec-path=|^--list-cmds=/.test(t)) { i++; continue; }
205
+ break;
206
+ }
207
+ const sub = args[i]; if (!sub) return null;
208
+ const rest = args.slice(i + 1);
209
+ if (sub === 'reset' && rest.includes('--hard')) return 'may discard uncommitted changes';
210
+ if (sub === 'push' && rest.some(t => t === '--force' || t === '-f' || t === '--force-with-lease' || t.startsWith('--force-with-lease=') || /^\+[\w/.-]+/.test(t))) return 'may overwrite remote history';
211
+ if (sub === 'clean') {
212
+ const dry = rest.some(t => t === '-n' || t === '--dry-run' || /^-[a-zA-Z]*n[a-zA-Z]*$/.test(t));
213
+ const force = rest.some(t => t === '-f' || t === '--force' || /^-[a-zA-Z]*f[a-zA-Z]*$/.test(t));
214
+ if (!dry && force) return 'may permanently delete untracked files';
215
+ }
216
+ if ((sub === 'checkout' || sub === 'restore') && rest.includes('.')) return 'may discard all working tree changes';
217
+ if (sub === 'stash' && (rest[0] === 'drop' || rest[0] === 'clear')) return 'may permanently remove stashed changes';
218
+ if (sub === 'branch' && (rest.includes('-D') || (rest.includes('--delete') && rest.includes('--force')))) return 'may force-delete a branch';
219
+ if ((sub === 'commit' || sub === 'push' || sub === 'merge') && rest.includes('--no-verify')) return 'may skip safety hooks';
220
+ if (sub === 'commit' && rest.includes('--amend')) return 'may rewrite the last commit';
221
+ return null;
222
+ }
223
+
224
+ const _KUBECTL_VAL = new Set(['--context','--cluster','--namespace','-n','--user','--kubeconfig','--token','--server','--as','--as-group','--certificate-authority','--client-certificate','--client-key','--request-timeout','--cache-dir','--v','-v','--profile','--profile-output']);
225
+
226
+ function _classifyKubectl(args) {
227
+ let i = 0;
228
+ while (i < args.length) {
229
+ const t = args[i];
230
+ if (_KUBECTL_VAL.has(t)) { i += 2; continue; }
231
+ if (t.startsWith('--') && t.includes('=')) { i++; continue; }
232
+ if (t.startsWith('-')) { i++; continue; }
233
+ break;
234
+ }
235
+ return args[i] === 'delete' ? 'may delete Kubernetes resources' : null;
236
+ }
237
+
238
+ function _classifyTerraform(args) {
239
+ let i = 0;
240
+ while (i < args.length) {
241
+ const t = args[i];
242
+ if (/^-chdir=/.test(t) || t === '-help' || t === '-version' || t === '-h') { i++; continue; }
243
+ break;
244
+ }
245
+ return args[i] === 'destroy' ? 'may destroy Terraform infrastructure' : null;
246
+ }
247
+
248
+ function _classifyDd(args) {
249
+ for (const t of args) {
250
+ if (/^if=\/dev\//.test(t)) return 'may read from a raw device';
251
+ if (/^of=\/dev\//.test(t)) return 'may write to a raw device';
252
+ }
253
+ return null;
254
+ }
255
+
256
+ // DB destructive pattern baseline — security policy, not a heuristic
257
+ // classifier. DROP/TRUNCATE/DELETE without WHERE are unambiguously
258
+ // data-destructive regardless of context; this list is a floor, not a
259
+ // complete SQL auditor.
260
+ const _DB_PATTERNS = [
261
+ [/\b(DROP|TRUNCATE)\s+(TABLE|DATABASE|SCHEMA)\b/i, 'may drop or truncate database objects'],
262
+ [/\bDELETE\s+FROM\s+\w+[ \t]*(;|"|'|\n|$)/i, 'may delete all rows from a database table'],
263
+ ];
264
+
265
+ function _classifySegment(tokens) {
266
+ const peeled = _peelWrappers(tokens);
267
+ if (!peeled.length) return null;
268
+ const cmd = peeled[0], rest = peeled.slice(1);
269
+ const cmdLow = cmd.toLowerCase();
270
+ if (cmd === 'rm') return _classifyRm(rest);
271
+ if (_PS_REMOVE_CMDS.has(cmdLow)) {
272
+ const w = _classifyRemoveItem(rest);
273
+ if (w) return w;
274
+ }
275
+ if (cmd === 'git') return _classifyGit(rest);
276
+ if (cmd === 'kubectl') return _classifyKubectl(rest);
277
+ if (cmd === 'terraform') return _classifyTerraform(rest);
278
+ if (cmd === 'dd') return _classifyDd(rest);
279
+ if (cmd === 'rmdir' || cmd === 'rd') {
280
+ const hasS = rest.some(t => /^\/s$/i.test(t));
281
+ const hasDriveRoot = rest.some(t => /^[A-Za-z]:\\?$/.test(t));
282
+ if (hasS && hasDriveRoot) return 'may recursively remove a drive root';
283
+ }
284
+ return null;
285
+ }
286
+
287
+ export function getDestructiveCommandWarning(command) {
288
+ const raw = String(command || '');
289
+ // Per-statement DB scan (split on `;`, `&&`, `||`, `|`, `\n`) so a
290
+ // destructive statement past a separator still surfaces.
291
+ const cleaned = _stripQuotedSpans(raw);
292
+ for (const stmt of cleaned.split(/[;&|\n]+/)) {
293
+ for (const [re, warning] of _DB_PATTERNS) if (re.test(stmt)) return warning;
294
+ }
295
+ // Tokenized walk per pipeline segment.
296
+ const segs = _splitSegments(_tokenize(raw));
297
+ for (const seg of segs) {
298
+ const peeled = _peelWrappers(seg);
299
+ if (!peeled.length) continue;
300
+ if (_SHELL_NAMES.has(peeled[0])) {
301
+ for (const inner of _extractShellCInner(seg.join(' '))) {
302
+ const w = getDestructiveCommandWarning(inner);
303
+ if (w) return w;
304
+ }
305
+ for (const body of _extractHeredocBodies(raw)) {
306
+ const w = getDestructiveCommandWarning(body);
307
+ if (w) return w;
308
+ }
309
+ continue;
310
+ }
311
+ const w = _classifySegment(seg);
312
+ if (w) return w;
313
+ }
314
+ for (const inner of _extractShellCInner(raw)) {
315
+ const w = getDestructiveCommandWarning(inner);
316
+ if (w) return w;
317
+ }
318
+ for (const inner of extractPowerShellCommandInner(raw)) {
319
+ const w = getDestructiveCommandWarning(inner);
320
+ if (w) return w;
321
+ }
322
+ return null;
323
+ }