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