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,104 @@
1
+ /**
2
+ * OpenAI Direct API — WebSocket transport via Responses API.
3
+ *
4
+ * Uses the same `sendViaWebSocket` plumbing as openai-oauth (Codex), with two
5
+ * differences encoded in the `auth.type === 'openai-direct'` branch inside
6
+ * openai-oauth-ws.mjs:
7
+ * 1. Authorization header: Bearer <OPENAI_API_KEY> (no account_id, no
8
+ * originator).
9
+ * 2. Endpoint: wss://api.openai.com/v1/responses.
10
+ *
11
+ * The Responses API request body is reused from openai-oauth (`buildRequestBody`)
12
+ * so prompt_cache_key, reasoning effort, and tool wiring stay byte-identical
13
+ * across the two providers — only the transport endpoint and auth header change.
14
+ */
15
+ import { sendViaWebSocket } from './openai-oauth-ws.mjs';
16
+ import { buildRequestBody } from './openai-oauth.mjs';
17
+ import { resolveProviderCacheKey } from '../smart-bridge/cache-strategy.mjs';
18
+
19
+ export class OpenAIDirectProvider {
20
+ // input_tokens INCLUDES cached tokens (OpenAI convention). See registry.mjs.
21
+ static inputExcludesCache = false;
22
+ name = 'openai';
23
+ config;
24
+ constructor(config) {
25
+ this.config = config || {};
26
+ }
27
+ _ensureKey() {
28
+ const k = this.config.apiKey;
29
+ if (!k) throw new Error('OPENAI_API_KEY not configured (providers.openai.apiKey)');
30
+ return k;
31
+ }
32
+ async send(messages, model, tools, sendOpts) {
33
+ const opts = sendOpts || {};
34
+ const onStageChange = typeof opts.onStageChange === 'function' ? opts.onStageChange : null;
35
+ const onStreamDelta = typeof opts.onStreamDelta === 'function' ? opts.onStreamDelta : null;
36
+ const onToolCall = typeof opts.onToolCall === 'function' ? opts.onToolCall : null;
37
+ const externalSignal = opts.signal || null;
38
+ const apiKey = this._ensureKey();
39
+ const useModel = model || 'gpt-5.5';
40
+ const body = buildRequestBody(messages, useModel, tools, sendOpts);
41
+ // Public Responses API supports prompt_cache_retention='24h' at no
42
+ // extra cost (same cached_input_tokens billing as the default 5–10
43
+ // min in-memory cache). Codex/oauth rejects the parameter, so it's
44
+ // injected only on the direct path. See openai-oauth.mjs:290-294
45
+ // for the rationale.
46
+ body.prompt_cache_retention = '24h';
47
+ // poolKey MUST be sessionId-only. Falling back to promptCacheKey would
48
+ // let unrelated raw sessions sharing the same provider-scoped cache
49
+ // bucket reuse each other's pooled socket and inherit lastResponseId
50
+ // delta state, producing cross-session prompt corruption.
51
+ const poolKey = opts.sessionId || null;
52
+ // cacheKey (prompt_cache_key) only groups the server-side prefix-cache
53
+ // shard — safe to share across sessions, unlike the sessionId poolKey
54
+ // above. Resolver lands on the shared 'mixdog-openai' shard for the
55
+ // public direct path (never sessionId), mirroring the codex path.
56
+ const cacheKey = resolveProviderCacheKey(opts, 'openai');
57
+ // Force explicit cache grouping so back-to-back calls don't race the
58
+ // server's implicit prefix-hash registration (codex path already does
59
+ // this — see openai-oauth.mjs:281-283).
60
+ if (cacheKey) body.prompt_cache_key = String(cacheKey).slice(0, 64);
61
+ const iteration = Number.isFinite(Number(opts.iteration)) ? Number(opts.iteration) : null;
62
+ const auth = { type: 'openai-direct', apiKey };
63
+ return sendViaWebSocket({
64
+ auth,
65
+ body,
66
+ sendOpts: opts,
67
+ onStreamDelta,
68
+ onToolCall,
69
+ onStageChange,
70
+ externalSignal,
71
+ poolKey,
72
+ cacheKey,
73
+ iteration,
74
+ useModel,
75
+ displayModel: (id) => id,
76
+ });
77
+ }
78
+ async listModels() {
79
+ try {
80
+ const apiKey = this._ensureKey();
81
+ const res = await fetch('https://api.openai.com/v1/models', {
82
+ signal: AbortSignal.timeout(10_000),
83
+ headers: { 'Authorization': `Bearer ${apiKey}` },
84
+ });
85
+ if (!res.ok) return [];
86
+ const j = await res.json();
87
+ return (j.data || []).map((m) => ({
88
+ id: m.id,
89
+ name: m.id,
90
+ provider: 'openai',
91
+ contextWindow: 200000,
92
+ // Preserve release timestamp from OpenAI so downstream
93
+ // freshness filters (e.g. setup UI's 6-month coding-model
94
+ // cutoff) can drop deprecated generations.
95
+ created: typeof m.created === 'number' ? m.created : null,
96
+ }));
97
+ } catch {
98
+ return [];
99
+ }
100
+ }
101
+ async isAvailable() {
102
+ return !!this.config.apiKey;
103
+ }
104
+ }
@@ -0,0 +1,192 @@
1
+ import { OpenAICompatProvider, OPENAI_COMPAT_PRESETS } from './openai-compat.mjs';
2
+ import { AnthropicProvider } from './anthropic.mjs';
3
+ import { GeminiProvider } from './gemini.mjs';
4
+ import { OpenAIOAuthProvider, hasOpenAIOAuthCredentials } from './openai-oauth.mjs';
5
+ import { AnthropicOAuthProvider, hasAnthropicOAuthCredentials } from './anthropic-oauth.mjs';
6
+ import { GrokOAuthProvider, hasGrokOAuthCredentials } from './grok-oauth.mjs';
7
+ import { OpenAIDirectProvider } from './openai-ws.mjs';
8
+ import { refreshCatalog as refreshMetadataCatalog } from './model-catalog.mjs';
9
+ // OpenAI-compat provider names are self-declared by openai-compat.mjs via
10
+ // OPENAI_COMPAT_PRESETS. No parallel list maintained here.
11
+ const providers = new Map();
12
+ export async function initProviders(config) {
13
+ // Invariant: never wipe the live registry based on an empty / all-disabled
14
+ // config. Without this guard, a stale `loadAgentConfig()` (e.g. mid-reload
15
+ // or a transient FS hiccup) would land here as `{}` or `{...,enabled:false}`,
16
+ // and the `providers.clear()` at the bottom would erase every previously
17
+ // registered provider. The owner process then stays alive returning
18
+ // `Provider "<name>" not found or not enabled` until restart. Throwing
19
+ // here preserves whatever was already registered.
20
+ const entries = Object.entries(config || {});
21
+ if (entries.length === 0) {
22
+ throw new Error('[provider] initProviders called with empty config — refusing to clear registry');
23
+ }
24
+ const next = new Map();
25
+ for (const [name, cfg] of entries) {
26
+ if (!cfg.enabled)
27
+ continue;
28
+ try {
29
+ if (name === 'anthropic') {
30
+ next.set(name, new AnthropicProvider(cfg));
31
+ }
32
+ else if (name === 'gemini') {
33
+ next.set(name, new GeminiProvider(cfg));
34
+ }
35
+ else if (name === 'openai-oauth') {
36
+ next.set(name, new OpenAIOAuthProvider(cfg));
37
+ }
38
+ else if (name === 'anthropic-oauth') {
39
+ next.set(name, new AnthropicOAuthProvider(cfg));
40
+ }
41
+ else if (name === 'grok-oauth') {
42
+ next.set(name, new GrokOAuthProvider(cfg));
43
+ }
44
+ else if (name === 'openai') {
45
+ next.set(name, new OpenAIDirectProvider(cfg));
46
+ }
47
+ else if (Object.prototype.hasOwnProperty.call(OPENAI_COMPAT_PRESETS, name)) {
48
+ next.set(name, new OpenAICompatProvider(name, cfg));
49
+ }
50
+ else {
51
+ throw new Error(`unknown enabled provider: ${name}`);
52
+ }
53
+ }
54
+ catch (err) {
55
+ const msg = err instanceof Error ? err.message : String(err);
56
+ throw new Error(`[provider] Failed to init "${name}": ${msg}`);
57
+ }
58
+ }
59
+ // Second guard: every entry was disabled. Same reasoning — keep the
60
+ // existing registry rather than going dark.
61
+ if (next.size === 0) {
62
+ throw new Error('[provider] all providers disabled in config — refusing to clear registry');
63
+ }
64
+ // OAuth preservation guard. anthropic-oauth / openai-oauth are NOT stored
65
+ // in mixdog-config.json — buildDefaultConfig injects them at runtime by
66
+ // calling hasAnthropic/OpenAIOAuthCredentials(), which reads the on-disk
67
+ // credentials file each call. A transient ENOENT / partial-write / JSON
68
+ // parse failure quietly returns false, the OAuth entry lands in next as
69
+ // disabled (silently skipped by the `!cfg.enabled` continue above), and
70
+ // the `providers.clear()` below would erase the previously registered
71
+ // instance permanently — the process then returns
72
+ // `Provider "anthropic-oauth" not found or not enabled` for the rest of
73
+ // its lifetime even though the credential file is fine again. Carry
74
+ // forward the prior instance instead.
75
+ for (const name of ['anthropic-oauth', 'openai-oauth', 'grok-oauth']) {
76
+ if (!next.has(name) && providers.has(name)) {
77
+ next.set(name, providers.get(name));
78
+ }
79
+ }
80
+ providers.clear();
81
+ for (const [k, v] of next) providers.set(k, v);
82
+ }
83
+ export function getProvider(name) {
84
+ const cached = providers.get(name);
85
+ if (cached) return cached;
86
+ // OAuth lazy fallback. Covers the boot-time race where
87
+ // hasAnthropic/OpenAIOAuthCredentials() returned false the first time
88
+ // (credential file mid-write, lock contention, or a transient parse
89
+ // failure) — initProviders then skipped the entry entirely so there is
90
+ // nothing for the preservation guard to carry forward. Re-probe the
91
+ // credential each miss: if the credential is now valid, register the
92
+ // instance on the spot so subsequent calls hit the cached entry.
93
+ if (name === 'anthropic-oauth' && hasAnthropicOAuthCredentials()) {
94
+ const inst = new AnthropicOAuthProvider({});
95
+ providers.set(name, inst);
96
+ return inst;
97
+ }
98
+ if (name === 'openai-oauth' && hasOpenAIOAuthCredentials()) {
99
+ const inst = new OpenAIOAuthProvider({});
100
+ providers.set(name, inst);
101
+ return inst;
102
+ }
103
+ if (name === 'grok-oauth' && hasGrokOAuthCredentials()) {
104
+ const inst = new GrokOAuthProvider({});
105
+ providers.set(name, inst);
106
+ return inst;
107
+ }
108
+ return undefined;
109
+ }
110
+ // Whether a provider reports usage.input_tokens EXCLUDING cached tokens
111
+ // (Anthropic) rather than INCLUDING them (openai / gemini / grok). Used to
112
+ // normalize the live "context window" footprint in session metrics: for a
113
+ // cache-excluding provider the cache_read count must be added back to reflect
114
+ // what the model actually saw last turn. The convention is declared as a
115
+ // static `inputExcludesCache` on each provider class, so a newly added
116
+ // provider states its own answer — no central regex to keep in sync. Unknown /
117
+ // unregistered providers default to false (the openai/gemini majority).
118
+ export function providerInputExcludesCache(name) {
119
+ const p = getProvider(name);
120
+ return p?.constructor?.inputExcludesCache === true;
121
+ }
122
+ export function getAllProviders() {
123
+ // Defensive copy — callers must not mutate the live registry or retain
124
+ // stale entries across re-init (initProviders rebuilds the map in place).
125
+ return new Map(providers);
126
+ }
127
+ // Background catalog warm-up. Each provider's listModels() either hits its
128
+ // own cached model list (no-op) or fires a single HTTP refresh. Called from
129
+ // agent.init() after providers are registered so the first bridge LLM call
130
+ // (e.g. cycle1 on session start) does not pay the catalog refresh latency
131
+ // inline. Fire-and-forget: failures are logged inside each provider.
132
+ export function warmupCatalogs() {
133
+ for (const [name, provider] of providers) {
134
+ if (typeof provider?.listModels !== 'function') continue;
135
+ Promise.resolve()
136
+ .then(() => provider.listModels())
137
+ .catch((err) => {
138
+ const msg = err instanceof Error ? err.message : String(err);
139
+ process.stderr.write(`[provider:${name}] catalog warm-up failed: ${msg}\n`);
140
+ });
141
+ }
142
+ }
143
+
144
+ // Force-refresh each provider's /models catalog on every MCP start. Unlike
145
+ // warmupCatalogs (which calls listModels() and so respects the 24h provider
146
+ // TTL → no-op when the cache is fresh), this bypasses the TTL via
147
+ // _refreshModelCache so a model released since the last refresh is picked up
148
+ // at startup instead of waiting for TTL expiry. Deliberately does NOT touch
149
+ // the shared LiteLLM metadata catalog (refreshMetadataCatalog) — pricing /
150
+ // context metadata stays on its own 24h TTL. Fire-and-forget: never awaited,
151
+ // per-provider failures logged to stderr like warmupCatalogs.
152
+ export function refreshProviderCatalogsOnStartup() {
153
+ for (const [name, provider] of providers) {
154
+ const refreshFn = typeof provider?._refreshModelCache === 'function'
155
+ ? () => provider._refreshModelCache()
156
+ : (typeof provider?.listModels === 'function' ? () => provider.listModels() : null);
157
+ if (!refreshFn) continue;
158
+ Promise.resolve()
159
+ .then(() => refreshFn())
160
+ .catch((err) => {
161
+ const msg = err instanceof Error ? err.message : String(err);
162
+ process.stderr.write(`[provider:${name}] startup catalog refresh failed: ${msg}\n`);
163
+ });
164
+ }
165
+ }
166
+
167
+ // Force-refresh provider catalogs after an operator changes model/provider
168
+ // configuration. This bypasses the 24h provider TTL where supported and warms
169
+ // the shared LiteLLM metadata cache first so context/pricing metadata follows
170
+ // newly released models without waiting for the next process restart.
171
+ export function refreshCatalogs() {
172
+ const metadataReady = Promise.resolve()
173
+ .then(() => refreshMetadataCatalog())
174
+ .catch((err) => {
175
+ const msg = err instanceof Error ? err.message : String(err);
176
+ process.stderr.write(`[model-catalog] metadata refresh failed: ${msg}\n`);
177
+ return null;
178
+ });
179
+ for (const [name, provider] of providers) {
180
+ const refreshFn = typeof provider?._refreshModelCache === 'function'
181
+ ? () => provider._refreshModelCache()
182
+ : (typeof provider?.listModels === 'function' ? () => provider.listModels() : null);
183
+ if (!refreshFn) continue;
184
+ Promise.resolve()
185
+ .then(() => metadataReady)
186
+ .then(() => refreshFn())
187
+ .catch((err) => {
188
+ const msg = err instanceof Error ? err.message : String(err);
189
+ process.stderr.write(`[provider:${name}] catalog refresh failed: ${msg}\n`);
190
+ });
191
+ }
192
+ }
@@ -0,0 +1,325 @@
1
+ /**
2
+ * retry-classifier.mjs — shared transient/permanent error classifier
3
+ *
4
+ * Single source of truth across every provider (openai-oauth-ws, openai-oauth,
5
+ * anthropic-oauth, anthropic, gemini, openai-ws, openai-compat).
6
+ *
7
+ * Goal: when a provider returns a transient server-side condition we should
8
+ * retry; when it returns a deterministic refusal (auth, permission, quota)
9
+ * we should fail fast. Mid-stream WS events (server-supplied error / response
10
+ * .failed messages) historically lost their HTTP status because the message
11
+ * was wrapped without classification — that left "Our servers are currently
12
+ * overloaded" indistinguishable from a permanent failure to the retry layer.
13
+ *
14
+ * Usage:
15
+ * import { classifyError, populateHttpStatusFromMessage } from './retry-classifier.mjs'
16
+ * const kind = classifyError(err) // 'auth' | 'permanent' | 'transient' | 'unknown'
17
+ * populateHttpStatusFromMessage(err) // mutates err.httpStatus if message hints at one
18
+ */
19
+
20
+ import {
21
+ PROVIDER_MAX_BEFORE_WARN_MS,
22
+ PROVIDER_RETRY_BACKOFF_MS,
23
+ PROVIDER_RETRY_JITTER_RATIO,
24
+ PROVIDER_RETRY_MAX_ATTEMPTS,
25
+ createTimeoutSignal,
26
+ } from '../stall-policy.mjs'
27
+
28
+ // HTTP statuses considered transient — safe to retry with backoff.
29
+ // 408 — request timeout
30
+ // 429 — rate limit (caller may still respect Retry-After, but the kind
31
+ // classification here only signals "retryable"; sub-error in the
32
+ // provider can still treat 429 as permanent for quota-exhausted)
33
+ // 500/502/503/504 — server errors (overload / bad gateway / timeout)
34
+ const TRANSIENT_STATUSES = new Set([408, 500, 502, 503, 504])
35
+
36
+ // HTTP statuses that mean "permanent: stop retrying, surface to caller".
37
+ // 401/403 — auth issue
38
+ // 404 — not found
39
+ // 400/422 — bad request (deterministic)
40
+ const AUTH_STATUSES = new Set([401, 403])
41
+ const PERMANENT_STATUSES = new Set([400, 404, 405, 410, 415, 422])
42
+
43
+ // Server-message text patterns. Used when a WS / SSE event carries an error
44
+ // payload but no explicit status code — we sniff the text and assign the most
45
+ // likely HTTP equivalent so the retry layer can use the same rules.
46
+ const MESSAGE_PATTERNS = [
47
+ // Overload / transient 5xx — server is asking us to back off. The `\b`
48
+ // anchor is intentionally OMITTED on the trailing side of `overload` so
49
+ // "overloaded" / "overloading" both match (inflected forms are common in
50
+ // server error text).
51
+ { regex: /(?:overload(?:ed|ing)?|temporarily unavailable|try again later|service unavailable|bad gateway|gateway timeout)/i, status: 503 },
52
+ // Explicit 5xx mention.
53
+ { regex: /\b(?:5\d\d|http 5\d\d)\b/i, status: 503 },
54
+ // Rate limit / quota — same retry posture as 429 but treat as permanent
55
+ // by classifyError because per-call retry won't change the answer (the
56
+ // window must elapse). Providers that want time-bounded retry should
57
+ // honor Retry-After on the original response, not loop here.
58
+ { regex: /(?:rate ?limit|quota)/i, status: 429 },
59
+ // Auth — never retryable from our side.
60
+ { regex: /\b(?:unauthorized|unauthorised|authentication|not authenticated|token expired|access token|invalid api key)\b/i, status: 401 },
61
+ { regex: /\b(?:forbidden|permission denied|policy violation)\b/i, status: 403 },
62
+ ]
63
+
64
+ /**
65
+ * Inspect `err.message` (or the explicit `msg` argument) and, if the text
66
+ * matches one of the known transient/permanent patterns, set `err.httpStatus`
67
+ * to the corresponding code. No-op when httpStatus is already set.
68
+ *
69
+ * Returns the resolved httpStatus (existing or newly assigned), or 0 when
70
+ * nothing matched.
71
+ */
72
+ export function populateHttpStatusFromMessage(err, msg = null) {
73
+ if (!err || typeof err !== 'object') return 0
74
+ if (Number(err.httpStatus) > 0) return Number(err.httpStatus)
75
+ const text = String(msg ?? err.message ?? '')
76
+ if (!text) return 0
77
+ for (const { regex, status } of MESSAGE_PATTERNS) {
78
+ if (regex.test(text)) {
79
+ err.httpStatus = status
80
+ return status
81
+ }
82
+ }
83
+ return 0
84
+ }
85
+
86
+ /**
87
+ * Classify an error for retry policy. Combines HTTP status (when set) and
88
+ * message-text fallback so message-only errors (mid-stream WS error events)
89
+ * route through the same logic as fetch responses.
90
+ *
91
+ * 'auth' — 401/403 — invalid credentials / forbidden, fail fast.
92
+ * 'permanent' — 4xx (non-auth) or quota — caller decision is final.
93
+ * 'transient' — 5xx/408 or socket-level transient codes — retry with backoff.
94
+ * 'unknown' — neither; default to permanent in safety-critical paths,
95
+ * or retry once in best-effort paths.
96
+ */
97
+ export function classifyError(err) {
98
+ if (!err) return 'unknown'
99
+ // Truncated SSE stream (message_start without message_stop). These are
100
+ // idempotent to retry: the partial result is discarded, and a pendingToolUse
101
+ // means the tool_use input JSON never completed, so re-requesting is safe.
102
+ // Checked BEFORE HTTP-status classification so a truncation error that also
103
+ // carries a 4xx/429 status still classifies transient per the contract.
104
+ // Treating this as transient is what lets every withRetry / mid-stream-loop
105
+ // consumer recover a cut-off stream uniformly.
106
+ if (err.truncatedStream === true || err.code === 'TRUNCATED_STREAM') return 'transient'
107
+
108
+ // Honor explicit httpStatus first, then sniff message text.
109
+ const status = Number(err.httpStatus || err.status || err.response?.status || 0) || populateHttpStatusFromMessage(err)
110
+ if (AUTH_STATUSES.has(status)) return 'auth'
111
+ if (status === 429) return 'permanent'
112
+ if (TRANSIENT_STATUSES.has(status)) return 'transient'
113
+ if (PERMANENT_STATUSES.has(status)) return 'permanent'
114
+
115
+ // Socket-level codes (Node errno) — DNS / reset / refused / timeout are all
116
+ // transient: we can retry the same request and may succeed.
117
+ const code = String(err.code || '')
118
+ if (code === 'ECONNRESET' || code === 'ETIMEDOUT' || code === 'ESOCKETTIMEDOUT'
119
+ || code === 'EAI_AGAIN' || code === 'ENOTFOUND' || code === 'EAI_NODATA'
120
+ || code === 'ECONNREFUSED' || code === 'ENETUNREACH' || code === 'EHOSTUNREACH'
121
+ || code === 'EPIPE'
122
+ || code === 'EPROVIDERTIMEOUT' || code === 'EGEMINITIMEOUT'
123
+ || code === 'EWSACQUIRETIMEOUT') {
124
+ return 'transient'
125
+ }
126
+
127
+ return 'unknown'
128
+ }
129
+
130
+ // Provider error-text signatures for a context-window / input-too-large
131
+ // rejection. These are DETERMINISTIC refusals (the request is simply too big)
132
+ // — not transient faults — so they must never be routed through the
133
+ // network/stall retry path. The fix is to shrink the payload (trim harder)
134
+ // and re-send, which the agent loop's send path does once before surfacing.
135
+ // Patterns cover OpenAI ("maximum context length", "reduce the length"),
136
+ // Anthropic ("prompt is too long"), and generic "input exceeds the context
137
+ // window" phrasing. Match is case-insensitive over err.message.
138
+ const CONTEXT_OVERFLOW_PATTERNS = [
139
+ /input (?:length|tokens?) exceeds? the context window/i,
140
+ /exceeds? the (?:maximum )?context (?:window|length)/i,
141
+ /maximum context length/i,
142
+ /context[_ ]length[_ ]exceeded/i,
143
+ /prompt is too long/i,
144
+ /reduce the length of (?:the )?(?:messages|input|prompt)/i,
145
+ ]
146
+
147
+ /**
148
+ * True when `err` is a context-window-exceeded provider rejection. Walks
149
+ * err.cause / err.response.data up to depth 2 so SDK-wrapped errors are
150
+ * detected. Deterministic: the same request will always be rejected, so
151
+ * callers must shrink the payload (trim harder) before re-sending rather
152
+ * than blindly retrying against the same input.
153
+ * @param {unknown} err
154
+ * @returns {boolean}
155
+ */
156
+ export function isContextOverflowError(err, _depth = 0) {
157
+ if (!err || _depth > 2) return false
158
+ const msg = (err instanceof Error ? err.message : (typeof err === 'string' ? err : err?.message)) || ''
159
+ if (msg && CONTEXT_OVERFLOW_PATTERNS.some((re) => re.test(msg))) return true
160
+ if (err.cause != null && err.cause !== err) return isContextOverflowError(err.cause, _depth + 1)
161
+ if (err.response?.data != null) return isContextOverflowError(err.response.data, _depth + 1)
162
+ return false
163
+ }
164
+
165
+ function _headerValue(headers, name) {
166
+ if (!headers) return null
167
+ const lower = name.toLowerCase()
168
+ if (typeof headers.get === 'function') return headers.get(name) || headers.get(lower)
169
+ for (const [k, v] of Object.entries(headers)) {
170
+ if (String(k).toLowerCase() === lower) return Array.isArray(v) ? v[0] : v
171
+ }
172
+ return null
173
+ }
174
+
175
+ export function retryAfterMsFromError(err) {
176
+ const headers = err?.headers || err?.response?.headers || err?.data?.responseHeaders || null
177
+ const retryAfterMs = _headerValue(headers, 'retry-after-ms')
178
+ if (retryAfterMs != null && retryAfterMs !== '') {
179
+ const n = Number(retryAfterMs)
180
+ if (Number.isFinite(n) && n >= 0) return n
181
+ }
182
+ const retryAfter = _headerValue(headers, 'retry-after')
183
+ if (retryAfter == null || retryAfter === '') return null
184
+ const seconds = Number(retryAfter)
185
+ if (Number.isFinite(seconds) && seconds >= 0) return Math.ceil(seconds * 1000)
186
+ const dateMs = Date.parse(String(retryAfter))
187
+ if (!Number.isNaN(dateMs)) return Math.max(0, dateMs - Date.now())
188
+ return null
189
+ }
190
+
191
+ /**
192
+ * Convenience predicate: should this error be retried at the request level?
193
+ * Wraps classifyError() with the standard "transient = retry, otherwise no"
194
+ * policy. Callers that have provider-specific retry budgets (e.g. anthropic-
195
+ * oauth's MAX_ATTEMPTS, openai-oauth-ws's mid-stream classifier) still gate
196
+ * on attempt count separately; this helper only answers the kind question.
197
+ */
198
+ export function isRetryable(err) {
199
+ return classifyError(err) === 'transient'
200
+ }
201
+
202
+ // Default backoff schedule used by withRetry when caller does not override.
203
+ // Mirrors anthropic-oauth's 5-attempt curve (immediate + 1s/2s/4s/8s) so the
204
+ // total cap stays under 15s. Total upper bound = sum = 15s.
205
+ const DEFAULT_BACKOFF_MS = PROVIDER_RETRY_BACKOFF_MS
206
+ const DEFAULT_MAX_ATTEMPTS = PROVIDER_RETRY_MAX_ATTEMPTS
207
+
208
+ export function jitterDelayMs(ms, ratio = PROVIDER_RETRY_JITTER_RATIO, mode = 'symmetric') {
209
+ const base = Number(ms) || 0
210
+ if (base <= 0) return 0
211
+ const r = Math.min(Math.max(Number(ratio) || 0, 0), 1)
212
+ if (!r) return Math.round(base)
213
+ const spread = base * r
214
+ const offset = mode === 'positive'
215
+ ? Math.random() * spread
216
+ : (Math.random() * 2 - 1) * spread
217
+ return Math.max(0, Math.round(base + offset))
218
+ }
219
+
220
+ /**
221
+ * Run an async function with exponential-backoff retry on transient errors.
222
+ *
223
+ * Behavior:
224
+ * - Calls `fn()` up to `maxAttempts` times.
225
+ * - Between attempts, sleeps `backoffMs[attemptIndex]`.
226
+ * - Honors `signal` (AbortSignal): aborts current attempt's wait and re-
227
+ * throws caller's reason. Does NOT abort an in-flight call — that's
228
+ * the provider's own responsibility via its native abort plumbing.
229
+ * - Uses classifyError() to decide retry. 'transient' → retry,
230
+ * 'auth' / 'permanent' / 'unknown' → throw immediately.
231
+ * - populateHttpStatusFromMessage(err) is called on every caught error so
232
+ * server-text errors (e.g. "Our servers are currently overloaded")
233
+ * resolve to httpStatus before classification.
234
+ *
235
+ * Returns whatever `fn()` resolves to. Throws the last error if every retry
236
+ * is exhausted, or the first error if it's classified non-transient.
237
+ */
238
+ export async function withRetry(fn, opts = {}) {
239
+ const maxAttempts = Number(opts.maxAttempts ?? DEFAULT_MAX_ATTEMPTS)
240
+ const backoffMs = Array.isArray(opts.backoffMs) ? opts.backoffMs : DEFAULT_BACKOFF_MS
241
+ const signal = opts.signal || null
242
+ const onRetry = typeof opts.onRetry === 'function' ? opts.onRetry : null
243
+ const perAttemptTimeoutMs = Number(opts.perAttemptTimeoutMs || 0)
244
+ const perAttemptLabel = opts.perAttemptLabel || 'provider request'
245
+ const maxRetryAfterMs = Number(opts.maxRetryAfterMs ?? PROVIDER_MAX_BEFORE_WARN_MS)
246
+ const retryJitterRatio = Number(opts.retryJitterRatio ?? PROVIDER_RETRY_JITTER_RATIO)
247
+
248
+ let lastErr = null
249
+ let nextDelayMs = null
250
+ let nextDelayReason = null
251
+ for (let attempt = 0; attempt < maxAttempts; attempt++) {
252
+ if (signal?.aborted) {
253
+ const reason = signal.reason
254
+ throw reason instanceof Error ? reason : new Error('withRetry: aborted')
255
+ }
256
+ if (attempt > 0) {
257
+ const rawWait = nextDelayMs ?? backoffMs[Math.min(attempt, backoffMs.length - 1)] ?? 0
258
+ const wait = jitterDelayMs(
259
+ rawWait,
260
+ retryJitterRatio,
261
+ nextDelayReason === 'retry-after' ? 'positive' : 'symmetric',
262
+ )
263
+ const boundedWait = nextDelayReason === 'retry-after' && Number.isFinite(maxRetryAfterMs)
264
+ ? Math.min(wait, maxRetryAfterMs)
265
+ : wait
266
+ onRetry?.({ attempt, lastErr, delayMs: boundedWait, delayReason: nextDelayReason })
267
+ if (boundedWait > 0) await _sleepWithAbort(boundedWait, signal)
268
+ if (signal?.aborted) {
269
+ const reason = signal.reason
270
+ throw reason instanceof Error ? reason : new Error('withRetry: aborted')
271
+ }
272
+ nextDelayMs = null
273
+ nextDelayReason = null
274
+ }
275
+ const attemptTimeout = perAttemptTimeoutMs > 0
276
+ ? createTimeoutSignal(signal, perAttemptTimeoutMs, `${perAttemptLabel} attempt ${attempt + 1}`)
277
+ : null
278
+ const attemptSignal = attemptTimeout?.signal || signal
279
+ try {
280
+ return await fn({ attempt, signal: attemptSignal })
281
+ } catch (err) {
282
+ let caught = err
283
+ if (!signal?.aborted && attemptSignal?.aborted && attemptSignal.reason instanceof Error) {
284
+ caught = attemptSignal.reason
285
+ }
286
+ if (signal?.aborted) {
287
+ const reason = signal.reason
288
+ throw reason instanceof Error ? reason : new Error('withRetry: aborted')
289
+ }
290
+ lastErr = caught
291
+ populateHttpStatusFromMessage(caught)
292
+ const retryAfterMs = retryAfterMsFromError(caught)
293
+ const status = Number(caught?.httpStatus || caught?.status || caught?.response?.status || 0)
294
+ const kind = classifyError(caught)
295
+ const retryableRateLimit = status === 429 && retryAfterMs != null
296
+ if (kind !== 'transient' && !retryableRateLimit) throw caught
297
+ // Last attempt failed transiently — propagate to caller.
298
+ if (attempt === maxAttempts - 1) throw caught
299
+ if (retryAfterMs != null) {
300
+ nextDelayMs = Math.max(0, Math.min(retryAfterMs, maxRetryAfterMs))
301
+ nextDelayReason = 'retry-after'
302
+ }
303
+ } finally {
304
+ attemptTimeout?.cleanup()
305
+ }
306
+ }
307
+ // Defensive — loop above always returns or throws.
308
+ throw lastErr || new Error('withRetry: exhausted with no error captured')
309
+ }
310
+
311
+ function _sleepWithAbort(ms, signal) {
312
+ return new Promise((resolve) => {
313
+ let onAbort = null
314
+ const t = setTimeout(() => {
315
+ if (signal && onAbort) {
316
+ try { signal.removeEventListener('abort', onAbort) } catch {}
317
+ }
318
+ resolve()
319
+ }, ms)
320
+ if (!signal) return
321
+ if (signal.aborted) { clearTimeout(t); resolve(); return }
322
+ onAbort = () => { clearTimeout(t); resolve() }
323
+ signal.addEventListener('abort', onAbort, { once: true })
324
+ })
325
+ }
@@ -0,0 +1,13 @@
1
+ // Lazy lookup of a session's AbortSignal without import-cycling against
2
+ // session/manager.mjs. The session manager is loaded once on first call;
3
+ // further lookups hit the cached resolver.
4
+ let _resolver = null;
5
+
6
+ export async function getAbortSignalForSession(sessionId) {
7
+ if (!sessionId) return null;
8
+ if (!_resolver) {
9
+ const mod = await import('./manager.mjs');
10
+ _resolver = typeof mod.getSessionAbortSignal === 'function' ? mod.getSessionAbortSignal : null;
11
+ }
12
+ return _resolver ? _resolver(sessionId) : null;
13
+ }
@@ -0,0 +1,42 @@
1
+ // Post-edit advisory marks: one-shot sidecar on the next read after a write.
2
+ import { _normalizeAbs } from './util.mjs';
3
+
4
+ // sessionId -> Map<absPath, { ts, toolName }>
5
+ const _postEditBySession = new Map();
6
+
7
+ /**
8
+ * Mark `path` as just-edited for `sessionId`. Caller invokes this after a
9
+ * successful write/edit/apply_patch so the next read on the same path can
10
+ * receive a one-shot advisory sidecar.
11
+ */
12
+ export function markPostEdit({ sessionId, path, cwd, toolName }) {
13
+ if (!sessionId) return;
14
+ const abs = _normalizeAbs(path, cwd);
15
+ if (!abs) return;
16
+ let m = _postEditBySession.get(sessionId);
17
+ if (!m) { m = new Map(); _postEditBySession.set(sessionId, m); }
18
+ m.set(abs, { ts: Date.now(), toolName: String(toolName || 'edit') });
19
+ }
20
+
21
+ /**
22
+ * One-shot read of a post-edit mark for `sessionId`+`path`. Returns the
23
+ * mark info on hit and removes it (so the advisory only fires once per
24
+ * edit). Returns null on miss.
25
+ */
26
+ export function consumePostEditMark({ sessionId, path, cwd }) {
27
+ if (!sessionId) return null;
28
+ const abs = _normalizeAbs(path, cwd);
29
+ if (!abs) return null;
30
+ const m = _postEditBySession.get(sessionId);
31
+ if (!m) return null;
32
+ const entry = m.get(abs);
33
+ if (!entry) return null;
34
+ m.delete(abs);
35
+ return entry;
36
+ }
37
+
38
+ /** Drop all post-edit marks for a session on close. */
39
+ export function clearPostEditMarks(sessionId) {
40
+ if (!sessionId) return;
41
+ _postEditBySession.delete(sessionId);
42
+ }