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.
- package/.claude-plugin/marketplace.json +31 -0
- package/.claude-plugin/plugin.json +20 -0
- package/.gitattributes +34 -0
- package/.mcp.json +14 -0
- package/ARCHITECTURE.md +77 -0
- package/CHANGELOG.md +7 -0
- package/CONTRIBUTING.md +45 -0
- package/DATA-FLOW.md +79 -0
- package/LICENSE +21 -0
- package/README.md +389 -0
- package/SECURITY.md +138 -0
- package/UNINSTALL.md +112 -0
- package/agents/maintenance.md +5 -0
- package/agents/memory-classification.md +30 -0
- package/agents/scheduler-task.md +18 -0
- package/agents/webhook-handler.md +27 -0
- package/agents/worker.md +24 -0
- package/bin/bridge +133 -0
- package/bin/statusline-launcher.mjs +78 -0
- package/bin/statusline-lib.mjs +550 -0
- package/bin/statusline.mjs +607 -0
- package/bun.lock +802 -0
- package/commands/config.md +16 -0
- package/commands/doctor.md +13 -0
- package/commands/setup.md +17 -0
- package/defaults/cycle3-review-prompt.md +90 -0
- package/defaults/hidden-roles.json +65 -0
- package/defaults/memory-chunk-prompt.md +63 -0
- package/defaults/memory-promote-prompt.md +135 -0
- package/defaults/mixdog-config.template.json +27 -0
- package/defaults/user-workflow.json +8 -0
- package/defaults/user-workflow.md +12 -0
- package/hooks/hooks.json +73 -0
- package/hooks/lib/active-instance.cjs +77 -0
- package/hooks/lib/permission-evaluator.cjs +411 -0
- package/hooks/lib/permission-route.cjs +63 -0
- package/hooks/lib/permission-rules.cjs +170 -0
- package/hooks/lib/settings-loader.cjs +116 -0
- package/hooks/post-tool-use.cjs +84 -0
- package/hooks/pre-mcp-sandbox.cjs +158 -0
- package/hooks/pre-tool-subagent.cjs +253 -0
- package/hooks/session-start.cjs +1372 -0
- package/hooks/turn-timer.cjs +82 -0
- package/lib/claude-md-writer.cjs +386 -0
- package/lib/config-cjs.cjs +61 -0
- package/lib/hook-pipe-path.cjs +10 -0
- package/lib/keychain-cjs.cjs +263 -0
- package/lib/plugin-paths.cjs +61 -0
- package/lib/rules-builder.cjs +241 -0
- package/lib/text-utils.cjs +61 -0
- package/native/README.md +117 -0
- package/native/prebuilt/linux-aarch64/mixdog-shim +0 -0
- package/native/prebuilt/linux-x86_64/mixdog-shim +0 -0
- package/native/prebuilt/macos-aarch64/mixdog-shim +0 -0
- package/native/prebuilt/macos-x86_64/mixdog-shim +0 -0
- package/native/prebuilt/windows-x86_64/mixdog-shim.exe +0 -0
- package/package.json +107 -0
- package/prompts/code-review.txt +16 -0
- package/prompts/security-audit.txt +17 -0
- package/rules/bridge/00-common.md +39 -0
- package/rules/bridge/20-skip-protocol.md +18 -0
- package/rules/bridge/30-explorer.md +33 -0
- package/rules/bridge/40-cycle1-agent.md +52 -0
- package/rules/bridge/41-cycle2-agent.md +62 -0
- package/rules/bridge/42-cycle3-agent.md +44 -0
- package/rules/lead/00-tool-lead.md +61 -0
- package/rules/lead/01-general.md +23 -0
- package/rules/lead/02-channels.md +49 -0
- package/rules/lead/03-team.md +27 -0
- package/rules/lead/04-workflow.md +20 -0
- package/rules/shared/00-language.md +14 -0
- package/rules/shared/01-tool.md +138 -0
- package/scripts/bootstrap.mjs +184 -0
- package/scripts/bridge-unify-smoke.mjs +308 -0
- package/scripts/build-runtime-linux.sh +348 -0
- package/scripts/build-runtime-macos.sh +217 -0
- package/scripts/build-runtime-windows.ps1 +242 -0
- package/scripts/builtin-utils-smoke.mjs +392 -0
- package/scripts/check-json.mjs +45 -0
- package/scripts/check-syntax-changed.mjs +102 -0
- package/scripts/check-syntax.mjs +58 -0
- package/scripts/code-graph-batch.test.mjs +33 -0
- package/scripts/config-preserve-smoke.mjs +180 -0
- package/scripts/doctor.mjs +484 -0
- package/scripts/edit-normalize-fuzz.mjs +130 -0
- package/scripts/edit-normalize-smoke.mjs +401 -0
- package/scripts/edit-operation-smoke.mjs +369 -0
- package/scripts/edit2-smoke.mjs +63 -0
- package/scripts/fuzzy-e2e.mjs +28 -0
- package/scripts/fuzzy-smoke.mjs +26 -0
- package/scripts/generate-runtime-manifest.mjs +166 -0
- package/scripts/guard-smoke.mjs +66 -0
- package/scripts/hidden-role-schema-smoke.mjs +162 -0
- package/scripts/hook-routing-smoke.mjs +29 -0
- package/scripts/inject-input.ps1 +204 -0
- package/scripts/io-complex-smoke.mjs +667 -0
- package/scripts/io-explore-bench.mjs +424 -0
- package/scripts/io-guardrails-smoke.mjs +205 -0
- package/scripts/io-mini-bench-baseline.json +11 -0
- package/scripts/io-mini-bench.mjs +216 -0
- package/scripts/io-route-harness.mjs +933 -0
- package/scripts/io-telemetry-report.mjs +691 -0
- package/scripts/mutation-bench.mjs +564 -0
- package/scripts/mutation-io-smoke.mjs +1081 -0
- package/scripts/native-patch-bridge-smoke.mjs +288 -0
- package/scripts/native-patch-smoke.mjs +304 -0
- package/scripts/patch-interior-context-smoke.mjs +49 -0
- package/scripts/patch-newline-utf8-smoke.mjs +157 -0
- package/scripts/perf-hook-smoke.mjs +71 -0
- package/scripts/permission-eval-smoke.mjs +426 -0
- package/scripts/prep-patch.mjs +53 -0
- package/scripts/prep-shim.mjs +96 -0
- package/scripts/provider-cache-smoke.mjs +687 -0
- package/scripts/report-runtime-health.mjs +132 -0
- package/scripts/run-mcp.mjs +1547 -0
- package/scripts/salvage-v4a-shatter.test.mjs +58 -0
- package/scripts/scoped-cache-io-smoke.mjs +103 -0
- package/scripts/shell-policy-round3-smoke.mjs +46 -0
- package/scripts/smoke-runtime-negative.ps1 +100 -0
- package/scripts/smoke-runtime-negative.sh +95 -0
- package/scripts/stall-policy-smoke.mjs +50 -0
- package/scripts/start-memory-worker.mjs +23 -0
- package/scripts/statusline-launcher-smoke.mjs +82 -0
- package/scripts/stress-atomic-write.mjs +1028 -0
- package/scripts/test-config-rmw-restore.mjs +122 -0
- package/scripts/test-fault-inject.mjs +164 -0
- package/scripts/test-large-file.mjs +174 -0
- package/scripts/tool-edge-smoke.mjs +209 -0
- package/scripts/uninstall.mjs +201 -0
- package/scripts/webhook-selfheal-smoke.mjs +29 -0
- package/scripts/write-overwrite-guard-smoke.mjs +56 -0
- package/server-main.mjs +3055 -0
- package/server.mjs +468 -0
- package/setup/config-merge.mjs +254 -0
- package/setup/install.mjs +120 -0
- package/setup/launch-core.mjs +507 -0
- package/setup/launch.mjs +101 -0
- package/setup/setup-server.mjs +3206 -0
- package/setup/setup.html +3693 -0
- package/skills/retro-skill-proposer/SKILL.md +92 -0
- package/skills/schedule-add/SKILL.md +77 -0
- package/skills/setup/SKILL.md +346 -0
- package/skills/webhook-add/SKILL.md +81 -0
- package/src/agent/bridge-stall-watchdog.mjs +337 -0
- package/src/agent/index.mjs +2138 -0
- package/src/agent/orchestrator/activity-bus.mjs +38 -0
- package/src/agent/orchestrator/ai-wrapped-dispatch.mjs +1010 -0
- package/src/agent/orchestrator/bridge-retry.mjs +220 -0
- package/src/agent/orchestrator/bridge-trace.mjs +583 -0
- package/src/agent/orchestrator/cache-mtime.mjs +58 -0
- package/src/agent/orchestrator/config.mjs +358 -0
- package/src/agent/orchestrator/context/collect.mjs +651 -0
- package/src/agent/orchestrator/dispatch-persist.mjs +549 -0
- package/src/agent/orchestrator/drain-registry.mjs +50 -0
- package/src/agent/orchestrator/explore-validator.mjs +8 -0
- package/src/agent/orchestrator/internal-roles.mjs +118 -0
- package/src/agent/orchestrator/internal-tools.mjs +88 -0
- package/src/agent/orchestrator/jobs.mjs +116 -0
- package/src/agent/orchestrator/mcp/client.mjs +364 -0
- package/src/agent/orchestrator/providers/anthropic-betas.mjs +21 -0
- package/src/agent/orchestrator/providers/anthropic-oauth.mjs +1745 -0
- package/src/agent/orchestrator/providers/anthropic.mjs +437 -0
- package/src/agent/orchestrator/providers/gemini.mjs +1175 -0
- package/src/agent/orchestrator/providers/grok-oauth.mjs +782 -0
- package/src/agent/orchestrator/providers/model-catalog.mjs +241 -0
- package/src/agent/orchestrator/providers/openai-compat.mjs +1467 -0
- package/src/agent/orchestrator/providers/openai-oauth-ws.mjs +1890 -0
- package/src/agent/orchestrator/providers/openai-oauth.mjs +1307 -0
- package/src/agent/orchestrator/providers/openai-ws.mjs +104 -0
- package/src/agent/orchestrator/providers/registry.mjs +192 -0
- package/src/agent/orchestrator/providers/retry-classifier.mjs +325 -0
- package/src/agent/orchestrator/session/abort-lookup.mjs +13 -0
- package/src/agent/orchestrator/session/cache/post-edit-marks.mjs +42 -0
- package/src/agent/orchestrator/session/cache/prefetch-cache.mjs +142 -0
- package/src/agent/orchestrator/session/cache/read-cache.mjs +319 -0
- package/src/agent/orchestrator/session/cache/scoped-cache-outcome.mjs +11 -0
- package/src/agent/orchestrator/session/cache/scoped-cache.mjs +361 -0
- package/src/agent/orchestrator/session/cache/util.mjs +49 -0
- package/src/agent/orchestrator/session/loop.mjs +1478 -0
- package/src/agent/orchestrator/session/manager.mjs +1975 -0
- package/src/agent/orchestrator/session/read-dedup.mjs +6 -0
- package/src/agent/orchestrator/session/result-classification.mjs +65 -0
- package/src/agent/orchestrator/session/save-session-worker.mjs +18 -0
- package/src/agent/orchestrator/session/store.mjs +624 -0
- package/src/agent/orchestrator/session/stream-watchdog.mjs +130 -0
- package/src/agent/orchestrator/session/tool-result-offload.mjs +166 -0
- package/src/agent/orchestrator/session/trim.mjs +491 -0
- package/src/agent/orchestrator/smart-bridge/CACHE-SHARD.md +115 -0
- package/src/agent/orchestrator/smart-bridge/bridge-llm.mjs +327 -0
- package/src/agent/orchestrator/smart-bridge/cache-obs.mjs +150 -0
- package/src/agent/orchestrator/smart-bridge/cache-strategy.mjs +228 -0
- package/src/agent/orchestrator/smart-bridge/index.mjs +215 -0
- package/src/agent/orchestrator/smart-bridge/profiles.mjs +37 -0
- package/src/agent/orchestrator/smart-bridge/registry.mjs +348 -0
- package/src/agent/orchestrator/smart-bridge/session-builder.mjs +116 -0
- package/src/agent/orchestrator/stall-policy.mjs +195 -0
- package/src/agent/orchestrator/tool-loop-guard.mjs +75 -0
- package/src/agent/orchestrator/tools/bash-policy-scan.mjs +77 -0
- package/src/agent/orchestrator/tools/bash-session.mjs +721 -0
- package/src/agent/orchestrator/tools/builtin/advisory-lock.mjs +171 -0
- package/src/agent/orchestrator/tools/builtin/arg-guard.mjs +455 -0
- package/src/agent/orchestrator/tools/builtin/atomic-write.mjs +236 -0
- package/src/agent/orchestrator/tools/builtin/bash-tool.mjs +480 -0
- package/src/agent/orchestrator/tools/builtin/binary-file.mjs +76 -0
- package/src/agent/orchestrator/tools/builtin/builtin-tools.mjs +256 -0
- package/src/agent/orchestrator/tools/builtin/cache-layers.mjs +386 -0
- package/src/agent/orchestrator/tools/builtin/cwd-utils.mjs +37 -0
- package/src/agent/orchestrator/tools/builtin/device-paths.mjs +154 -0
- package/src/agent/orchestrator/tools/builtin/diagnostics-tool.mjs +292 -0
- package/src/agent/orchestrator/tools/builtin/diff-utils.mjs +109 -0
- package/src/agent/orchestrator/tools/builtin/edit-base-guard.mjs +58 -0
- package/src/agent/orchestrator/tools/builtin/edit-byte-plan.mjs +240 -0
- package/src/agent/orchestrator/tools/builtin/edit-byte-utils.mjs +113 -0
- package/src/agent/orchestrator/tools/builtin/edit-commit.mjs +74 -0
- package/src/agent/orchestrator/tools/builtin/edit-context-utils.mjs +242 -0
- package/src/agent/orchestrator/tools/builtin/edit-diagnostics.mjs +211 -0
- package/src/agent/orchestrator/tools/builtin/edit-engine.mjs +1364 -0
- package/src/agent/orchestrator/tools/builtin/edit-failure-context.mjs +126 -0
- package/src/agent/orchestrator/tools/builtin/edit-hint.mjs +141 -0
- package/src/agent/orchestrator/tools/builtin/edit-match-utils.mjs +194 -0
- package/src/agent/orchestrator/tools/builtin/edit-partial-write.mjs +60 -0
- package/src/agent/orchestrator/tools/builtin/edit-stale-refresh.mjs +168 -0
- package/src/agent/orchestrator/tools/builtin/edit-tool.mjs +173 -0
- package/src/agent/orchestrator/tools/builtin/edit-utf8-guard.mjs +48 -0
- package/src/agent/orchestrator/tools/builtin/fs-reachability.mjs +48 -0
- package/src/agent/orchestrator/tools/builtin/fuzzy-match.mjs +99 -0
- package/src/agent/orchestrator/tools/builtin/glob-walk.mjs +170 -0
- package/src/agent/orchestrator/tools/builtin/grep-formatting.mjs +113 -0
- package/src/agent/orchestrator/tools/builtin/hash-utils.mjs +6 -0
- package/src/agent/orchestrator/tools/builtin/list-formatting.mjs +7 -0
- package/src/agent/orchestrator/tools/builtin/list-tool.mjs +593 -0
- package/src/agent/orchestrator/tools/builtin/native-edit-runner.mjs +89 -0
- package/src/agent/orchestrator/tools/builtin/notebook-edit-tool.mjs +300 -0
- package/src/agent/orchestrator/tools/builtin/open-config-tool.mjs +26 -0
- package/src/agent/orchestrator/tools/builtin/path-diagnostics.mjs +152 -0
- package/src/agent/orchestrator/tools/builtin/path-locks.mjs +35 -0
- package/src/agent/orchestrator/tools/builtin/path-utils.mjs +201 -0
- package/src/agent/orchestrator/tools/builtin/read-args.mjs +103 -0
- package/src/agent/orchestrator/tools/builtin/read-batch.mjs +172 -0
- package/src/agent/orchestrator/tools/builtin/read-constants.mjs +40 -0
- package/src/agent/orchestrator/tools/builtin/read-formatting.mjs +118 -0
- package/src/agent/orchestrator/tools/builtin/read-image-resize.mjs +189 -0
- package/src/agent/orchestrator/tools/builtin/read-image.mjs +88 -0
- package/src/agent/orchestrator/tools/builtin/read-lines.mjs +12 -0
- package/src/agent/orchestrator/tools/builtin/read-mode-tool.mjs +455 -0
- package/src/agent/orchestrator/tools/builtin/read-open.mjs +190 -0
- package/src/agent/orchestrator/tools/builtin/read-range-index.mjs +271 -0
- package/src/agent/orchestrator/tools/builtin/read-ranges.mjs +26 -0
- package/src/agent/orchestrator/tools/builtin/read-single-tool.mjs +728 -0
- package/src/agent/orchestrator/tools/builtin/read-snapshot-runtime.mjs +173 -0
- package/src/agent/orchestrator/tools/builtin/read-special-files.mjs +268 -0
- package/src/agent/orchestrator/tools/builtin/read-streaming.mjs +602 -0
- package/src/agent/orchestrator/tools/builtin/read-tool.mjs +530 -0
- package/src/agent/orchestrator/tools/builtin/read-windows.mjs +107 -0
- package/src/agent/orchestrator/tools/builtin/rename-tool.mjs +196 -0
- package/src/agent/orchestrator/tools/builtin/rg-runner.mjs +422 -0
- package/src/agent/orchestrator/tools/builtin/search-builders.mjs +158 -0
- package/src/agent/orchestrator/tools/builtin/search-tool.mjs +869 -0
- package/src/agent/orchestrator/tools/builtin/shell-analysis.mjs +653 -0
- package/src/agent/orchestrator/tools/builtin/shell-jobs.mjs +936 -0
- package/src/agent/orchestrator/tools/builtin/shell-output.mjs +36 -0
- package/src/agent/orchestrator/tools/builtin/shell-runtime.mjs +214 -0
- package/src/agent/orchestrator/tools/builtin/snapshot-helpers.mjs +143 -0
- package/src/agent/orchestrator/tools/builtin/snapshot-store.mjs +206 -0
- package/src/agent/orchestrator/tools/builtin/snapshot-validation.mjs +98 -0
- package/src/agent/orchestrator/tools/builtin/text-stats.mjs +69 -0
- package/src/agent/orchestrator/tools/builtin/windows-roots.mjs +23 -0
- package/src/agent/orchestrator/tools/builtin/write-tool.mjs +401 -0
- package/src/agent/orchestrator/tools/builtin.mjs +500 -0
- package/src/agent/orchestrator/tools/code-graph-prewarm-worker.mjs +39 -0
- package/src/agent/orchestrator/tools/code-graph-tool-defs.mjs +24 -0
- package/src/agent/orchestrator/tools/code-graph.mjs +4095 -0
- package/src/agent/orchestrator/tools/cwd-tool.mjs +298 -0
- package/src/agent/orchestrator/tools/destructive-warning.mjs +323 -0
- package/src/agent/orchestrator/tools/edit-normalize.mjs +603 -0
- package/src/agent/orchestrator/tools/env-scrub.mjs +100 -0
- package/src/agent/orchestrator/tools/graph-binary-fetcher.mjs +144 -0
- package/src/agent/orchestrator/tools/graph-manifest.json +26 -0
- package/src/agent/orchestrator/tools/host-input.mjs +204 -0
- package/src/agent/orchestrator/tools/mutation-content-cache.mjs +67 -0
- package/src/agent/orchestrator/tools/mutation-planner.mjs +75 -0
- package/src/agent/orchestrator/tools/next-call-utils.mjs +48 -0
- package/src/agent/orchestrator/tools/patch-binary-fetcher.mjs +133 -0
- package/src/agent/orchestrator/tools/patch-manifest.json +26 -0
- package/src/agent/orchestrator/tools/patch-tool-defs.mjs +20 -0
- package/src/agent/orchestrator/tools/patch.mjs +2754 -0
- package/src/agent/orchestrator/tools/progress-message.mjs +118 -0
- package/src/agent/orchestrator/tools/result-compression.mjs +279 -0
- package/src/agent/orchestrator/tools/shell-command.mjs +865 -0
- package/src/agent/orchestrator/tools/shell-exec-policy.mjs +89 -0
- package/src/agent/orchestrator/tools/shell-policy-danger-target.mjs +27 -0
- package/src/agent/orchestrator/tools/shell-policy-imports.mjs +7 -0
- package/src/agent/orchestrator/tools/shell-policy.mjs +345 -0
- package/src/agent/orchestrator/tools/shell-snapshot.mjs +313 -0
- package/src/agent/orchestrator/workflow-store.mjs +93 -0
- package/src/agent/tool-defs.mjs +103 -0
- package/src/channels/backends/discord.mjs +784 -0
- package/src/channels/data/voice-runtime-manifest.json +138 -0
- package/src/channels/index.mjs +3229 -0
- package/src/channels/lib/cli-worker-host.mjs +12 -0
- package/src/channels/lib/config-lock.mjs +13 -0
- package/src/channels/lib/config.mjs +292 -0
- package/src/channels/lib/drop-trace.mjs +71 -0
- package/src/channels/lib/event-pipeline.mjs +81 -0
- package/src/channels/lib/event-queue.mjs +345 -0
- package/src/channels/lib/executor.mjs +168 -0
- package/src/channels/lib/format.mjs +188 -0
- package/src/channels/lib/holidays.mjs +138 -0
- package/src/channels/lib/hook-pipe-server.mjs +802 -0
- package/src/channels/lib/interaction-workflows.mjs +184 -0
- package/src/channels/lib/memory-client.mjs +149 -0
- package/src/channels/lib/output-forwarder.mjs +765 -0
- package/src/channels/lib/runtime-paths.mjs +479 -0
- package/src/channels/lib/scheduler.mjs +723 -0
- package/src/channels/lib/session-control.mjs +36 -0
- package/src/channels/lib/session-discovery.mjs +103 -0
- package/src/channels/lib/settings.mjs +11 -0
- package/src/channels/lib/state-file.mjs +68 -0
- package/src/channels/lib/status-snapshot.mjs +219 -0
- package/src/channels/lib/tool-format.mjs +140 -0
- package/src/channels/lib/transcript-discovery.mjs +195 -0
- package/src/channels/lib/voice-runtime-fetcher.mjs +734 -0
- package/src/channels/lib/webhook.mjs +1179 -0
- package/src/channels/lib/whisper-server.mjs +477 -0
- package/src/channels/tool-defs.mjs +170 -0
- package/src/daemon/host.mjs +118 -0
- package/src/daemon/mcp-transport.mjs +47 -0
- package/src/daemon/session.mjs +100 -0
- package/src/daemon/thin-client.mjs +71 -0
- package/src/daemon/transport.mjs +163 -0
- package/src/memory/data/runtime-manifest.json +40 -0
- package/src/memory/index.mjs +3305 -0
- package/src/memory/lib/agent-ipc.mjs +93 -0
- package/src/memory/lib/bridge-trace-queries.mjs +120 -0
- package/src/memory/lib/core-memory-store.mjs +330 -0
- package/src/memory/lib/embedding-provider.mjs +269 -0
- package/src/memory/lib/embedding-worker.mjs +323 -0
- package/src/memory/lib/llm-worker-host.mjs +17 -0
- package/src/memory/lib/memory-cycle.mjs +11 -0
- package/src/memory/lib/memory-cycle1.mjs +641 -0
- package/src/memory/lib/memory-cycle2.mjs +1284 -0
- package/src/memory/lib/memory-cycle3.mjs +540 -0
- package/src/memory/lib/memory-embed.mjs +299 -0
- package/src/memory/lib/memory-extraction.mjs +5 -0
- package/src/memory/lib/memory-maintenance-store.mjs +32 -0
- package/src/memory/lib/memory-ops-policy.mjs +190 -0
- package/src/memory/lib/memory-recall-id-patch.mjs +15 -0
- package/src/memory/lib/memory-recall-read-query.mjs +7 -0
- package/src/memory/lib/memory-recall-scope-filter.mjs +63 -0
- package/src/memory/lib/memory-recall-store.mjs +621 -0
- package/src/memory/lib/memory-retrievers.mjs +112 -0
- package/src/memory/lib/memory-score.mjs +71 -0
- package/src/memory/lib/memory-text-utils.mjs +58 -0
- package/src/memory/lib/memory.mjs +412 -0
- package/src/memory/lib/model-profile.mjs +85 -0
- package/src/memory/lib/pg/adapter.mjs +308 -0
- package/src/memory/lib/pg/process.mjs +360 -0
- package/src/memory/lib/pg/supervisor.mjs +396 -0
- package/src/memory/lib/project-id-resolver.mjs +86 -0
- package/src/memory/lib/runtime-fetcher.mjs +442 -0
- package/src/memory/lib/trace-store.mjs +728 -0
- package/src/memory/tool-defs.mjs +79 -0
- package/src/search/index.mjs +1173 -0
- package/src/search/lib/backends/anthropic-oauth.mjs +98 -0
- package/src/search/lib/backends/exa.mjs +50 -0
- package/src/search/lib/backends/firecrawl.mjs +61 -0
- package/src/search/lib/backends/gemini-api.mjs +83 -0
- package/src/search/lib/backends/grok-oauth.mjs +86 -0
- package/src/search/lib/backends/index.mjs +150 -0
- package/src/search/lib/backends/openai-api.mjs +144 -0
- package/src/search/lib/backends/openai-oauth.mjs +98 -0
- package/src/search/lib/backends/openai-web-search.mjs +76 -0
- package/src/search/lib/backends/tavily.mjs +55 -0
- package/src/search/lib/backends/xai-api.mjs +113 -0
- package/src/search/lib/cache.mjs +131 -0
- package/src/search/lib/config.mjs +192 -0
- package/src/search/lib/formatter.mjs +115 -0
- package/src/search/lib/provider-usage.mjs +67 -0
- package/src/search/lib/providers.mjs +47 -0
- package/src/search/lib/search-intent.mjs +109 -0
- package/src/search/lib/setup-handler.mjs +261 -0
- package/src/search/lib/state.mjs +201 -0
- package/src/search/lib/web-tools.mjs +1207 -0
- package/src/search/tool-defs.mjs +83 -0
- package/src/setup/defender-exclusion.mjs +183 -0
- package/src/shared/abort-controller.mjs +15 -0
- package/src/shared/atomic-file.mjs +420 -0
- package/src/shared/config.mjs +350 -0
- package/src/shared/daemon-recycle.mjs +108 -0
- package/src/shared/disable-claude-builtins.mjs +88 -0
- package/src/shared/err-text.mjs +12 -0
- package/src/shared/llm/cost.mjs +66 -0
- package/src/shared/llm/http-agent.mjs +123 -0
- package/src/shared/llm/index.mjs +41 -0
- package/src/shared/llm/pid-cleanup.mjs +27 -0
- package/src/shared/llm/usage-log.mjs +47 -0
- package/src/shared/plugin-paths.mjs +58 -0
- package/src/shared/schedules-store.mjs +70 -0
- package/src/shared/seed.mjs +119 -0
- package/src/shared/user-cwd.mjs +213 -0
- package/src/shared/user-data-guard.mjs +238 -0
- package/src/status/aggregator.mjs +584 -0
- package/src/status/server.mjs +413 -0
- 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
|
+
}
|