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,765 @@
|
|
|
1
|
+
import { existsSync, statSync, watch, openSync, readSync, closeSync } from "fs";
|
|
2
|
+
import { basename } from "path";
|
|
3
|
+
import { createHash } from "crypto";
|
|
4
|
+
import { formatForDiscord, chunk, safeCodeBlock } from "./format.mjs";
|
|
5
|
+
import { dropTrace, _dtPreview } from "./drop-trace.mjs";
|
|
6
|
+
import {
|
|
7
|
+
cwdToProjectSlug,
|
|
8
|
+
discoverCurrentClaudeSession,
|
|
9
|
+
listInteractiveClaudeSessions,
|
|
10
|
+
getLatestInteractiveClaudeSession
|
|
11
|
+
} from "./session-discovery.mjs";
|
|
12
|
+
import {
|
|
13
|
+
findLatestTranscriptByMtime,
|
|
14
|
+
sameResolvedPath,
|
|
15
|
+
detectCurrentSessionTranscript,
|
|
16
|
+
discoverSessionBoundTranscript
|
|
17
|
+
} from "./transcript-discovery.mjs";
|
|
18
|
+
import {
|
|
19
|
+
SKIP_TEXTS,
|
|
20
|
+
HIDDEN_TOOLS,
|
|
21
|
+
isRecallMemory,
|
|
22
|
+
isMemoryFile,
|
|
23
|
+
buildDedupKey,
|
|
24
|
+
buildToolLine
|
|
25
|
+
} from "./tool-format.mjs";
|
|
26
|
+
|
|
27
|
+
class OutputForwarder {
|
|
28
|
+
ownerGetter = null;
|
|
29
|
+
setOwnerGetter(fn) {
|
|
30
|
+
this.ownerGetter = fn;
|
|
31
|
+
}
|
|
32
|
+
_isOwner() {
|
|
33
|
+
if (!this.ownerGetter) return true;
|
|
34
|
+
// Fail closed: a probe exception must NOT be treated as ownership.
|
|
35
|
+
// Forwarding transcript output from a non-owner duplicates Discord
|
|
36
|
+
// sends from the previous owner process.
|
|
37
|
+
try { return !!this.ownerGetter(); } catch { return false; }
|
|
38
|
+
}
|
|
39
|
+
constructor(cb, statusState) {
|
|
40
|
+
this.cb = cb;
|
|
41
|
+
this.statusState = statusState;
|
|
42
|
+
this._persistTimer = null;
|
|
43
|
+
this._pendingPersistData = null;
|
|
44
|
+
// Best-effort final flush on process exit. The handler is sync (writeFileSync
|
|
45
|
+
// + fsyncSync inside updateState), so it actually lands on graceful exit.
|
|
46
|
+
// SIGTERM/SIGINT shutdowns that bypass 'exit' are covered by the timer
|
|
47
|
+
// unref-ing so the event loop drains it before close.
|
|
48
|
+
process.on('exit', () => { try { this._flushPersistState(); } catch {} });
|
|
49
|
+
}
|
|
50
|
+
lastHash = "";
|
|
51
|
+
sentCount = 0;
|
|
52
|
+
transcriptPath = "";
|
|
53
|
+
channelId = "";
|
|
54
|
+
userMessageId = "";
|
|
55
|
+
emoji = "";
|
|
56
|
+
lastFileSize = 0;
|
|
57
|
+
readFileSize = 0;
|
|
58
|
+
watchingPath = "";
|
|
59
|
+
watcher = null;
|
|
60
|
+
idleTimer = null;
|
|
61
|
+
onIdleCallback = null;
|
|
62
|
+
inExplorerSequence = false;
|
|
63
|
+
inRecallSequence = false;
|
|
64
|
+
hasSeenAssistant = false;
|
|
65
|
+
sending = false;
|
|
66
|
+
sendRetryTimer = null;
|
|
67
|
+
// Priority-lane queues (Tier3 split).
|
|
68
|
+
// finalLane — text items (transcript text, final answer segments)
|
|
69
|
+
// streamLane — tool-log / progress items
|
|
70
|
+
// drainQueue() picks from finalLane first so final answers are never blocked
|
|
71
|
+
// behind a batch of tool-log messages.
|
|
72
|
+
finalLane = [];
|
|
73
|
+
streamLane = [];
|
|
74
|
+
// Cap streamLane length under sustained backpressure (Discord 429 storm,
|
|
75
|
+
// network outage). When over the cap, the oldest text item is merged
|
|
76
|
+
// into the next pending text payload so we don't grow unbounded but
|
|
77
|
+
// also don't lose content.
|
|
78
|
+
static SEND_QUEUE_MAX = 200;
|
|
79
|
+
// Persisted final-flush ledger so a forwarder restart can resume final
|
|
80
|
+
// forwarding instead of giving up after 5 short retries.
|
|
81
|
+
pendingFinalFlush = false;
|
|
82
|
+
mainSessionId = "";
|
|
83
|
+
watchDebounce = null;
|
|
84
|
+
turnTextBuffer = "";
|
|
85
|
+
hasBinding() {
|
|
86
|
+
return !!this.transcriptPath;
|
|
87
|
+
}
|
|
88
|
+
/** Set context for current turn (called on user message) */
|
|
89
|
+
setContext(channelId, transcriptPath, options = {}) {
|
|
90
|
+
this.channelId = channelId;
|
|
91
|
+
if (!transcriptPath) return;
|
|
92
|
+
if (this.transcriptPath && !existsSync(this.transcriptPath)) {
|
|
93
|
+
const relocated = detectCurrentSessionTranscript()?.transcriptPath ?? findLatestTranscriptByMtime();
|
|
94
|
+
if (relocated) {
|
|
95
|
+
transcriptPath = relocated;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
if (this.transcriptPath !== transcriptPath) {
|
|
99
|
+
this.closeWatcher();
|
|
100
|
+
dropTrace("context.transcriptPath.change", { sessionId: this.mainSessionId || "(none)", oldPath: this.transcriptPath || "(none)", newPath: transcriptPath });
|
|
101
|
+
this.transcriptPath = transcriptPath;
|
|
102
|
+
this.mainSessionId = "";
|
|
103
|
+
this.sentCount = 0;
|
|
104
|
+
this.lastHash = "";
|
|
105
|
+
this.turnTextBuffer = "";
|
|
106
|
+
}
|
|
107
|
+
try {
|
|
108
|
+
const stat = existsSync(this.transcriptPath) ? statSync(this.transcriptPath) : null;
|
|
109
|
+
const currentSize = stat?.size ?? 0;
|
|
110
|
+
let fileSize;
|
|
111
|
+
if (options.replayFromStart) {
|
|
112
|
+
fileSize = 0;
|
|
113
|
+
} else if (options.catchUpFromPersisted) {
|
|
114
|
+
const persisted = this.statusState?.read?.();
|
|
115
|
+
const persistedSize = typeof persisted?.lastFileSize === "number" ? persisted.lastFileSize : -1;
|
|
116
|
+
const sameTranscript = persisted?.transcriptPath &&
|
|
117
|
+
sameResolvedPath(persisted.transcriptPath, this.transcriptPath);
|
|
118
|
+
fileSize = (sameTranscript && persistedSize >= 0)
|
|
119
|
+
? Math.min(Math.max(persistedSize, 0), currentSize)
|
|
120
|
+
: currentSize;
|
|
121
|
+
} else {
|
|
122
|
+
fileSize = currentSize;
|
|
123
|
+
}
|
|
124
|
+
this.lastFileSize = fileSize;
|
|
125
|
+
this.readFileSize = fileSize;
|
|
126
|
+
} catch {
|
|
127
|
+
this.lastFileSize = 0;
|
|
128
|
+
this.readFileSize = 0;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
/** Reset counters for new turn */
|
|
132
|
+
reset() {
|
|
133
|
+
this.sentCount = 0;
|
|
134
|
+
this.lastHash = "";
|
|
135
|
+
this.inExplorerSequence = false;
|
|
136
|
+
this.inRecallSequence = false;
|
|
137
|
+
this.hasSeenAssistant = false;
|
|
138
|
+
this.turnTextBuffer = "";
|
|
139
|
+
if (this.idleTimer) {
|
|
140
|
+
clearTimeout(this.idleTimer);
|
|
141
|
+
this.idleTimer = null;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
/** Read new bytes from transcript file since readFileSize */
|
|
145
|
+
readNewLines() {
|
|
146
|
+
if (!this.transcriptPath || !existsSync(this.transcriptPath)) {
|
|
147
|
+
return { lines: [], nextFileSize: this.readFileSize };
|
|
148
|
+
}
|
|
149
|
+
let fd = null;
|
|
150
|
+
try {
|
|
151
|
+
const stat = this._pendingStat ?? statSync(this.transcriptPath);
|
|
152
|
+
this._pendingStat = null;
|
|
153
|
+
if (stat.size <= this.readFileSize) {
|
|
154
|
+
return { lines: [], nextFileSize: this.readFileSize };
|
|
155
|
+
}
|
|
156
|
+
const startOffset = this.readFileSize;
|
|
157
|
+
fd = openSync(this.transcriptPath, "r");
|
|
158
|
+
const buf = Buffer.alloc(stat.size - startOffset);
|
|
159
|
+
readSync(fd, buf, 0, buf.length, startOffset);
|
|
160
|
+
// Only advance readFileSize to the last newline boundary. A trailing
|
|
161
|
+
// partial line (writer still appending) must be re-read next tick;
|
|
162
|
+
// otherwise the parse-failed slice is silently consumed forever.
|
|
163
|
+
const lastNl = buf.lastIndexOf(0x0a);
|
|
164
|
+
const consumed = lastNl >= 0 ? lastNl + 1 : 0;
|
|
165
|
+
const nextFileSize = startOffset + consumed;
|
|
166
|
+
this.readFileSize = nextFileSize;
|
|
167
|
+
const text = consumed > 0 ? buf.slice(0, consumed).toString("utf8") : "";
|
|
168
|
+
return {
|
|
169
|
+
lines: text ? text.split("\n").filter((l) => l.trim()) : [],
|
|
170
|
+
nextFileSize
|
|
171
|
+
};
|
|
172
|
+
} catch {
|
|
173
|
+
return { lines: [], nextFileSize: this.readFileSize };
|
|
174
|
+
} finally {
|
|
175
|
+
if (fd != null) {
|
|
176
|
+
closeSync(fd);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
/** Track last tool_use name and file path for matching with tool_result */
|
|
181
|
+
lastToolName = "";
|
|
182
|
+
lastToolFilePath = "";
|
|
183
|
+
/** Extract new assistant text + tool logs from transcript since readFileSize */
|
|
184
|
+
extractNewText() {
|
|
185
|
+
const { lines: newLines, nextFileSize } = this.readNewLines();
|
|
186
|
+
let newText = "";
|
|
187
|
+
for (const l of newLines) {
|
|
188
|
+
try {
|
|
189
|
+
const entry = JSON.parse(l);
|
|
190
|
+
if (!entry.isSidechain && entry.sessionId && !this.mainSessionId) {
|
|
191
|
+
this.mainSessionId = entry.sessionId;
|
|
192
|
+
}
|
|
193
|
+
if (entry.isSidechain) continue;
|
|
194
|
+
if (this.mainSessionId && entry.sessionId && entry.sessionId !== this.mainSessionId) continue;
|
|
195
|
+
if (entry.type === "user" && entry.message?.content?.some((c) => c.type === "tool_result")) {
|
|
196
|
+
if (OutputForwarder.isRecallMemory(this.lastToolName)) {
|
|
197
|
+
continue;
|
|
198
|
+
}
|
|
199
|
+
if (this.lastToolName === "Edit" && entry.toolUseResult && !OutputForwarder.isMemoryFile(this.lastToolFilePath)) {
|
|
200
|
+
const old = entry.toolUseResult.oldString || "";
|
|
201
|
+
const nw = entry.toolUseResult.newString || "";
|
|
202
|
+
if (old || nw) {
|
|
203
|
+
const diffLines = [];
|
|
204
|
+
for (const l2 of old.split("\n")) diffLines.push("- " + l2);
|
|
205
|
+
for (const l2 of nw.split("\n")) diffLines.push("+ " + l2);
|
|
206
|
+
const shown = diffLines.slice(0, 15);
|
|
207
|
+
let diffContent = shown.join("\n");
|
|
208
|
+
if (diffLines.length > 15) diffContent += "\n... +" + (diffLines.length - 15) + " lines";
|
|
209
|
+
const block = safeCodeBlock(diffContent, "diff");
|
|
210
|
+
newText += block + "\n";
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
continue;
|
|
214
|
+
}
|
|
215
|
+
if (entry.type === "assistant" && entry.message?.content) {
|
|
216
|
+
this.hasSeenAssistant = true;
|
|
217
|
+
const SEARCH_TOOLS = /* @__PURE__ */ new Set(["Read", "Grep", "Glob"]);
|
|
218
|
+
const parts = [];
|
|
219
|
+
for (const c of entry.message.content) {
|
|
220
|
+
if (c.type === "text" && c.text?.trim()) {
|
|
221
|
+
this.inExplorerSequence = false;
|
|
222
|
+
this.inRecallSequence = false;
|
|
223
|
+
let cleaned = OutputForwarder.stripForeignWorkerChannels(c.text.trim());
|
|
224
|
+
cleaned = cleaned.replace(/<(memory-context|system-reminder|event)\b[^>]*>[\s\S]*?<\/\1>/gi, "").trim();
|
|
225
|
+
if (cleaned) parts.push(cleaned);
|
|
226
|
+
} else if (c.type === "tool_use") {
|
|
227
|
+
this.lastToolName = c.name || "";
|
|
228
|
+
this.lastToolFilePath = c.input?.file_path || "";
|
|
229
|
+
if (OutputForwarder.isHidden(c.name)) continue;
|
|
230
|
+
if (SEARCH_TOOLS.has(c.name)) {
|
|
231
|
+
if (!this.inExplorerSequence) {
|
|
232
|
+
this.inExplorerSequence = true;
|
|
233
|
+
let target = "";
|
|
234
|
+
if (c.name === "Read") target = c.input?.file_path ? basename(c.input.file_path) : "";
|
|
235
|
+
else if (c.name === "Grep") target = '"' + (c.input?.pattern || "").substring(0, 25) + '"';
|
|
236
|
+
else if (c.name === "Glob") target = (c.input?.pattern || "").substring(0, 25);
|
|
237
|
+
if (parts.length > 0) parts.push("");
|
|
238
|
+
parts.push("\u25CF **Explorer** (" + (target || c.name) + ")");
|
|
239
|
+
}
|
|
240
|
+
continue;
|
|
241
|
+
}
|
|
242
|
+
if (OutputForwarder.isRecallMemory(c.name)) {
|
|
243
|
+
if (!this.inRecallSequence) {
|
|
244
|
+
this.inRecallSequence = true;
|
|
245
|
+
if (parts.length > 0) parts.push("");
|
|
246
|
+
parts.push("\u25CF **recall_memory**");
|
|
247
|
+
}
|
|
248
|
+
continue;
|
|
249
|
+
}
|
|
250
|
+
this.inExplorerSequence = false;
|
|
251
|
+
this.inRecallSequence = false;
|
|
252
|
+
const toolLine = OutputForwarder.buildToolLine(c.name, c.input);
|
|
253
|
+
if (toolLine) {
|
|
254
|
+
if (parts.length > 0) parts.push("");
|
|
255
|
+
parts.push(toolLine);
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
if (parts.length) newText += parts.join("\n") + "\n";
|
|
260
|
+
}
|
|
261
|
+
} catch {
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
return { text: newText.trim(), nextFileSize };
|
|
265
|
+
}
|
|
266
|
+
// ── Single-send gate ──────────────────────────────────────────────
|
|
267
|
+
// All Discord sends pass through sendOnce() so duplicate concurrent sends are avoided.
|
|
268
|
+
// Texts that should never be forwarded to Discord (Claude's internal status lines)
|
|
269
|
+
static SKIP_TEXTS = SKIP_TEXTS;
|
|
270
|
+
commitReadProgress(nextFileSize) {
|
|
271
|
+
if (nextFileSize <= this.lastFileSize) return;
|
|
272
|
+
this.lastFileSize = nextFileSize;
|
|
273
|
+
this.persistState();
|
|
274
|
+
}
|
|
275
|
+
async deliverQueueItem(item) {
|
|
276
|
+
const targetChannelId = item.channelId ?? this.channelId;
|
|
277
|
+
if (!item.text || !targetChannelId) {
|
|
278
|
+
this.commitReadProgress(item.nextFileSize);
|
|
279
|
+
return;
|
|
280
|
+
}
|
|
281
|
+
if (!item.skipHashDedup && OutputForwarder.SKIP_TEXTS.has(item.text.trim())) {
|
|
282
|
+
this.commitReadProgress(item.nextFileSize);
|
|
283
|
+
return;
|
|
284
|
+
}
|
|
285
|
+
const formatted = item.preformatted ? item.text : formatForDiscord(item.text);
|
|
286
|
+
const hash = item.skipHashDedup
|
|
287
|
+
? ""
|
|
288
|
+
: item.dedupKey
|
|
289
|
+
? item.dedupKey
|
|
290
|
+
: createHash("md5").update(formatted).digest("hex");
|
|
291
|
+
if (!item.skipHashDedup && this.lastHash === hash) {
|
|
292
|
+
this.commitReadProgress(item.nextFileSize);
|
|
293
|
+
return;
|
|
294
|
+
}
|
|
295
|
+
const chunks = chunk(formatted, 2e3);
|
|
296
|
+
const _t0Send = Date.now();
|
|
297
|
+
// Resume from _nextChunkIdx if this item is being retried after a partial send.
|
|
298
|
+
// This avoids re-sending chunks that already landed successfully.
|
|
299
|
+
if (item._chunks === undefined) {
|
|
300
|
+
item._chunks = chunks;
|
|
301
|
+
item._nextChunkIdx = 0;
|
|
302
|
+
item._sendRetries = 0;
|
|
303
|
+
}
|
|
304
|
+
for (let _ci = item._nextChunkIdx; _ci < item._chunks.length; _ci++) {
|
|
305
|
+
const c = item._chunks[_ci];
|
|
306
|
+
try {
|
|
307
|
+
await this.cb.send(targetChannelId, c);
|
|
308
|
+
item._nextChunkIdx = _ci + 1;
|
|
309
|
+
dropTrace("discord.send.ok", null);
|
|
310
|
+
} catch (err) {
|
|
311
|
+
// Discord 429 or transient error — honour Retry-After then re-throw so
|
|
312
|
+
// drainQueue's retry loop calls deliverQueueItem again. Chunk progress
|
|
313
|
+
// is stored on item so we resume from the failed chunk, not chunk 0.
|
|
314
|
+
const status = err?.status ?? err?.code ?? err?.httpStatus;
|
|
315
|
+
const retryAfter = err?.retryAfter ?? err?.retry_after
|
|
316
|
+
?? err?.headers?.["retry-after"] ?? err?.response?.headers?.["retry-after"];
|
|
317
|
+
dropTrace("discord.send.err", { channelId: this.channelId, chunkIndex: _ci, status, retryAfter: retryAfter ?? "(none)", err: String(err) });
|
|
318
|
+
item._sendRetries = (item._sendRetries || 0) + 1;
|
|
319
|
+
if (item._sendRetries >= 3) {
|
|
320
|
+
// Cap retries to avoid infinite duplicate loop — give up on this item
|
|
321
|
+
process.stderr.write(`[output-forwarder] chunk send exceeded 3 retries at chunk ${_ci}, dropping item\n`);
|
|
322
|
+
item._nextChunkIdx = item._chunks.length; // mark exhausted
|
|
323
|
+
return;
|
|
324
|
+
}
|
|
325
|
+
if (status === 429) {
|
|
326
|
+
if (retryAfter != null) {
|
|
327
|
+
const ms = Number(retryAfter) > 1000 ? Number(retryAfter) : Number(retryAfter) * 1000;
|
|
328
|
+
if (Number.isFinite(ms) && ms > 0) {
|
|
329
|
+
await new Promise((r) => setTimeout(r, Math.min(ms, 60_000)));
|
|
330
|
+
}
|
|
331
|
+
} else {
|
|
332
|
+
await new Promise((r) => setTimeout(r, 1000));
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
throw err;
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
if (!item.skipHashDedup) {
|
|
339
|
+
this.lastHash = hash;
|
|
340
|
+
}
|
|
341
|
+
const _bt = typeof item.bufferText === 'string' ? item.bufferText : '';
|
|
342
|
+
if (_bt.trim()) {
|
|
343
|
+
this.turnTextBuffer = this.turnTextBuffer ? `${this.turnTextBuffer}
|
|
344
|
+
|
|
345
|
+
${_bt.trim()}` : _bt.trim();
|
|
346
|
+
}
|
|
347
|
+
this.sentCount += chunks.length;
|
|
348
|
+
this.commitReadProgress(item.nextFileSize);
|
|
349
|
+
}
|
|
350
|
+
scheduleRetry() {
|
|
351
|
+
if (this.sendRetryTimer) return;
|
|
352
|
+
dropTrace("drain.retry.schedule", { finalLen: this.finalLane.length, streamLen: this.streamLane.length });
|
|
353
|
+
this.sendRetryTimer = setTimeout(() => {
|
|
354
|
+
this.sendRetryTimer = null;
|
|
355
|
+
this._kickDrain();
|
|
356
|
+
}, 1e3);
|
|
357
|
+
}
|
|
358
|
+
/** Forward new assistant text to Discord. Returns true if text was sent. */
|
|
359
|
+
async forwardNewText() {
|
|
360
|
+
if (!this._isOwner()) return false;
|
|
361
|
+
if (!this.channelId) return false;
|
|
362
|
+
const { text: newText, nextFileSize } = this.extractNewText();
|
|
363
|
+
if (!newText) {
|
|
364
|
+
if (!this.sending && this.finalLane.length === 0 && this.streamLane.length === 0) {
|
|
365
|
+
this.commitReadProgress(nextFileSize);
|
|
366
|
+
}
|
|
367
|
+
return false;
|
|
368
|
+
}
|
|
369
|
+
// Coalesce back-to-back text items BEFORE drain/chunking so adjacent
|
|
370
|
+
// emits merge into one chunk pass rather than waiting for SEND_QUEUE_MAX.
|
|
371
|
+
// tool-log items stay separate (preformatted) so we only merge plain-text.
|
|
372
|
+
// Issue 1: if drain is in flight finalLane[0] is being delivered; coalescing
|
|
373
|
+
// into that slot causes a concurrent mutation race. Only coalesce into the
|
|
374
|
+
// tail of finalLane when it is not the item currently being drained.
|
|
375
|
+
// finalLane[0] is being drained when this.sending; coalesce into tail only
|
|
376
|
+
// when tail is not index 0 (i.e. length >= 2) or drain is not in flight.
|
|
377
|
+
const ftLen = this.finalLane.length;
|
|
378
|
+
const ftTail = (ftLen > 0 && !(this.sending && ftLen === 1))
|
|
379
|
+
? this.finalLane[ftLen - 1] : null;
|
|
380
|
+
if (ftTail) {
|
|
381
|
+
ftTail.text = `${ftTail.text}\n\n${newText}`;
|
|
382
|
+
ftTail.bufferText = `${ftTail.bufferText}\n\n${newText}`;
|
|
383
|
+
ftTail.nextFileSize = nextFileSize;
|
|
384
|
+
this._kickDrain();
|
|
385
|
+
return true;
|
|
386
|
+
}
|
|
387
|
+
// Cap-and-merge backpressure guard on finalLane: when saturated, fold
|
|
388
|
+
// the new text into the trailing finalLane item.
|
|
389
|
+
if (this.finalLane.length >= OutputForwarder.SEND_QUEUE_MAX) {
|
|
390
|
+
// Skip index 0 while drain is in flight (same race guard as above).
|
|
391
|
+
const capEnd = this.sending ? 1 : 0;
|
|
392
|
+
for (let i = this.finalLane.length - 1; i >= capEnd; i--) {
|
|
393
|
+
const cap = this.finalLane[i];
|
|
394
|
+
cap.text = `${cap.text}\n\n${newText}`;
|
|
395
|
+
cap.bufferText = `${cap.bufferText}\n\n${newText}`;
|
|
396
|
+
cap.nextFileSize = nextFileSize;
|
|
397
|
+
this._kickDrain();
|
|
398
|
+
return true;
|
|
399
|
+
}
|
|
400
|
+
}
|
|
401
|
+
this.finalLane.push({
|
|
402
|
+
type: "text",
|
|
403
|
+
text: newText,
|
|
404
|
+
nextFileSize,
|
|
405
|
+
bufferText: newText
|
|
406
|
+
});
|
|
407
|
+
this._kickDrain();
|
|
408
|
+
return true;
|
|
409
|
+
}
|
|
410
|
+
/** Forward tool log line to Discord */
|
|
411
|
+
async forwardToolLog(toolLine, toolName, toolInput) {
|
|
412
|
+
if (!this._isOwner()) return;
|
|
413
|
+
if (!this.channelId) return;
|
|
414
|
+
// Issue 2: do NOT advance readFileSize here via stat. The stat'd size
|
|
415
|
+
// could jump past bytes that forwardNewText has not yet parsed, causing
|
|
416
|
+
// extractNewText/readNewLines to skip real content. readFileSize is
|
|
417
|
+
// advanced only inside readNewLines once bytes are actually consumed.
|
|
418
|
+
this.streamLane.push({
|
|
419
|
+
type: "toolLog",
|
|
420
|
+
text: toolLine,
|
|
421
|
+
nextFileSize: this.readFileSize,
|
|
422
|
+
preformatted: true,
|
|
423
|
+
dedupKey: OutputForwarder.buildDedupKey(toolName, toolInput)
|
|
424
|
+
});
|
|
425
|
+
this._kickDrain();
|
|
426
|
+
}
|
|
427
|
+
/** Centralised fire-and-forget drainQueue with rejection guard. */
|
|
428
|
+
_kickDrain() {
|
|
429
|
+
this.drainQueue().catch(err => process.stderr.write(`[output-forwarder] drainQueue rejected: ${err?.message || err}\n`));
|
|
430
|
+
}
|
|
431
|
+
/** Drain both priority lanes sequentially. finalLane drains first when non-empty. */
|
|
432
|
+
async drainQueue() {
|
|
433
|
+
if (this.sending) return;
|
|
434
|
+
this.sending = true;
|
|
435
|
+
try {
|
|
436
|
+
while (this.finalLane.length > 0 || this.streamLane.length > 0) {
|
|
437
|
+
// Pick from finalLane first; fall back to streamLane.
|
|
438
|
+
const fromFinal = this.finalLane.length > 0;
|
|
439
|
+
const lane = fromFinal ? this.finalLane : this.streamLane;
|
|
440
|
+
const item = lane[0];
|
|
441
|
+
try {
|
|
442
|
+
if (item.type === "text") {
|
|
443
|
+
await this.deliverQueueItem(item);
|
|
444
|
+
} else if (item.type === "toolLog") {
|
|
445
|
+
await this.processToolLog(item);
|
|
446
|
+
}
|
|
447
|
+
lane.shift();
|
|
448
|
+
} catch (err) {
|
|
449
|
+
process.stderr.write(`mixdog: send failed: ${err}
|
|
450
|
+
`);
|
|
451
|
+
dropTrace("drain.send.err", { finalLen: this.finalLane.length, streamLen: this.streamLane.length, lane: fromFinal ? "final" : "stream", itemType: item?.type, err: String(err) });
|
|
452
|
+
this.scheduleRetry();
|
|
453
|
+
break;
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
} finally {
|
|
457
|
+
this.sending = false;
|
|
458
|
+
}
|
|
459
|
+
}
|
|
460
|
+
/** Internal: process a single tool log send (extracted from old forwardToolLog) */
|
|
461
|
+
async processToolLog(item) {
|
|
462
|
+
if (this.userMessageId) {
|
|
463
|
+
const newEmoji = "\u{1F6E0}\uFE0F";
|
|
464
|
+
try {
|
|
465
|
+
if (this.emoji && this.emoji !== newEmoji) {
|
|
466
|
+
await this.cb.removeReaction(this.channelId, this.userMessageId, this.emoji);
|
|
467
|
+
}
|
|
468
|
+
await this.cb.react(this.channelId, this.userMessageId, newEmoji);
|
|
469
|
+
this.emoji = newEmoji;
|
|
470
|
+
} catch {
|
|
471
|
+
}
|
|
472
|
+
}
|
|
473
|
+
await this.deliverQueueItem(item);
|
|
474
|
+
}
|
|
475
|
+
/** Forward final text on session idle */
|
|
476
|
+
async forwardFinalText(retries = 0, pinnedChannelId = null) {
|
|
477
|
+
if (!this._isOwner()) return;
|
|
478
|
+
// Pin the target channel at call time so a rebind to a new turn's channel
|
|
479
|
+
// (which mutates this.channelId synchronously after this fire-and-forget
|
|
480
|
+
// call returns) cannot redirect the previous turn's final output to the
|
|
481
|
+
// wrong channel.
|
|
482
|
+
const channelId = pinnedChannelId ?? this.channelId;
|
|
483
|
+
if (!channelId) return;
|
|
484
|
+
if (this.sending || this.finalLane.length > 0 || this.streamLane.length > 0) {
|
|
485
|
+
// Mark a durable flush request so a process restart picks it up
|
|
486
|
+
// instead of dropping the final frame on the floor.
|
|
487
|
+
try {
|
|
488
|
+
this.pendingFinalFlush = true;
|
|
489
|
+
this.updateState((state) => { state.pendingFinalFlush = true; });
|
|
490
|
+
} catch {}
|
|
491
|
+
if (retries < 5) {
|
|
492
|
+
setTimeout(() => void this.forwardFinalText(retries + 1, channelId), 300);
|
|
493
|
+
} else {
|
|
494
|
+
dropTrace("drain.finalText.exhausted", { retries, finalLen: this.finalLane.length, streamLen: this.streamLane.length, sending: this.sending });
|
|
495
|
+
// Past the short-retry budget: schedule a longer-tail drain wait so
|
|
496
|
+
// the final frame still ships once the queue empties, instead of
|
|
497
|
+
// silently giving up.
|
|
498
|
+
const waitDrain = () => {
|
|
499
|
+
if (!this.sending && this.finalLane.length === 0 && this.streamLane.length === 0) {
|
|
500
|
+
void this.forwardFinalText(0, channelId);
|
|
501
|
+
return;
|
|
502
|
+
}
|
|
503
|
+
setTimeout(waitDrain, 1000);
|
|
504
|
+
};
|
|
505
|
+
setTimeout(waitDrain, 1000);
|
|
506
|
+
}
|
|
507
|
+
return;
|
|
508
|
+
}
|
|
509
|
+
this.sending = true;
|
|
510
|
+
try {
|
|
511
|
+
if (this.userMessageId && this.emoji) {
|
|
512
|
+
try {
|
|
513
|
+
await this.cb.removeReaction(channelId, this.userMessageId, this.emoji);
|
|
514
|
+
} catch {
|
|
515
|
+
}
|
|
516
|
+
}
|
|
517
|
+
const { text: newText, nextFileSize } = this.extractNewText();
|
|
518
|
+
if (newText) {
|
|
519
|
+
const finalItem = { type: "text", text: newText, nextFileSize, bufferText: newText, channelId };
|
|
520
|
+
try {
|
|
521
|
+
await this.deliverQueueItem(finalItem);
|
|
522
|
+
} catch (err) {
|
|
523
|
+
// Transient send failure: extractNewText already advanced the read
|
|
524
|
+
// cursor past these bytes, so dropping the item here would lose the
|
|
525
|
+
// final text. Requeue it (it retains per-chunk send progress) and let
|
|
526
|
+
// drainQueue retry instead of silently discarding. pendingFinalFlush
|
|
527
|
+
// stays set so a process restart can also resume the flush.
|
|
528
|
+
this.finalLane.push(finalItem);
|
|
529
|
+
this.scheduleRetry();
|
|
530
|
+
return;
|
|
531
|
+
}
|
|
532
|
+
} else {
|
|
533
|
+
this.commitReadProgress(nextFileSize);
|
|
534
|
+
}
|
|
535
|
+
if (this.turnTextBuffer.trim()) {
|
|
536
|
+
await this.cb.recordAssistantTurn?.({
|
|
537
|
+
channelId,
|
|
538
|
+
text: this.turnTextBuffer.trim(),
|
|
539
|
+
sessionId: this.mainSessionId || void 0
|
|
540
|
+
});
|
|
541
|
+
this.turnTextBuffer = "";
|
|
542
|
+
}
|
|
543
|
+
// Clear the durable flush marker only after delivery succeeded so a
|
|
544
|
+
// throw above leaves pendingFinalFlush=true and a process restart
|
|
545
|
+
// can resume the final forward instead of dropping the frame.
|
|
546
|
+
try {
|
|
547
|
+
this.pendingFinalFlush = false;
|
|
548
|
+
this.updateState((state) => { state.pendingFinalFlush = false; });
|
|
549
|
+
} catch {}
|
|
550
|
+
this.updateState((state) => {
|
|
551
|
+
state.sessionIdle = true;
|
|
552
|
+
});
|
|
553
|
+
} finally {
|
|
554
|
+
this.sending = false;
|
|
555
|
+
}
|
|
556
|
+
}
|
|
557
|
+
/** Hidden tools — skip both tool_use and tool_result */
|
|
558
|
+
static HIDDEN_TOOLS = HIDDEN_TOOLS;
|
|
559
|
+
/** Check if a tool name is recall_memory */
|
|
560
|
+
static isRecallMemory = isRecallMemory;
|
|
561
|
+
/** Check if a file path points to a memory file */
|
|
562
|
+
static isMemoryFile = isMemoryFile;
|
|
563
|
+
/** Check if a tool should be hidden */
|
|
564
|
+
static isHidden = (name) => {
|
|
565
|
+
// Read through OutputForwarder.HIDDEN_TOOLS so runtime reassignment of
|
|
566
|
+
// the static Set propagates into hidden-tool detection (restores the
|
|
567
|
+
// original indirect-reference semantics before the tool-format split).
|
|
568
|
+
// The non-set checks are inlined rather than delegated to the imported
|
|
569
|
+
// isHidden, because that helper would re-consult the module-local
|
|
570
|
+
// HIDDEN_TOOLS Set and ignore the OutputForwarder static.
|
|
571
|
+
if (OutputForwarder.HIDDEN_TOOLS.has(name)) return true;
|
|
572
|
+
if ((name.includes("plugin_mixdog") && !name.endsWith("recall_memory")) || name === "reply" || name === "react" || name === "edit_message" || name === "fetch" || name === "download_attachment") return true;
|
|
573
|
+
return false;
|
|
574
|
+
};
|
|
575
|
+
/** Concatenate text blocks from a transcript entry (user or assistant). */
|
|
576
|
+
static collectEntryText(entry) {
|
|
577
|
+
const parts = entry?.message?.content;
|
|
578
|
+
if (!Array.isArray(parts)) return "";
|
|
579
|
+
return parts
|
|
580
|
+
.filter((c) => c && c.type === "text" && typeof c.text === "string")
|
|
581
|
+
.map((c) => c.text)
|
|
582
|
+
.join("\n");
|
|
583
|
+
}
|
|
584
|
+
/**
|
|
585
|
+
* Remove mixdog worker/dispatch `<channel>` blocks whose `client_host_pid`
|
|
586
|
+
* does not match this owner's MIXDOG_OWNER_HOST_PID. Own-worker and legacy
|
|
587
|
+
* blocks (no client_host_pid) are left intact.
|
|
588
|
+
*/
|
|
589
|
+
static stripForeignWorkerChannels(text) {
|
|
590
|
+
if (!text || !/<channel\b/i.test(text)) return text || "";
|
|
591
|
+
const owner = Number(process.env.MIXDOG_OWNER_HOST_PID);
|
|
592
|
+
const ownerOk = Number.isFinite(owner) && owner > 0;
|
|
593
|
+
return text.replace(/<channel\b([^>]*)>[\s\S]*?<\/channel>/gi, (block, openAttrs) => {
|
|
594
|
+
if (!/\bsource\s*=\s*["'][^"']*mixdog/i.test(openAttrs)) return block;
|
|
595
|
+
const m = openAttrs.match(/\bclient_host_pid\s*=\s*["']([^"']+)["']/i);
|
|
596
|
+
if (!m) return block;
|
|
597
|
+
if (!ownerOk) return block;
|
|
598
|
+
const origin = Number(m[1]);
|
|
599
|
+
if (!Number.isFinite(origin) || origin <= 0) return block;
|
|
600
|
+
if (origin === owner) return block;
|
|
601
|
+
return "";
|
|
602
|
+
});
|
|
603
|
+
}
|
|
604
|
+
/**
|
|
605
|
+
* Build a per-call dedup key for tool-log queue items.
|
|
606
|
+
* Uses the full (unsquished) tool args so that two Reads on distinct files
|
|
607
|
+
* sharing a basename, or two Grep/Glob calls sharing only a pattern prefix,
|
|
608
|
+
* do not collapse onto the same key and suppress the second send.
|
|
609
|
+
* Returns "" to fall back to md5(formatted) at delivery time.
|
|
610
|
+
*/
|
|
611
|
+
static buildDedupKey = buildDedupKey;
|
|
612
|
+
/** Build a tool log line from the tool name and input. */
|
|
613
|
+
static buildToolLine = (name, input) => {
|
|
614
|
+
// Pass OutputForwarder.isHidden as the hidden-tool predicate so reassign-
|
|
615
|
+
// ment of the static propagates into tool-line construction (restores the
|
|
616
|
+
// original indirect-reference semantics before the tool-format split).
|
|
617
|
+
return buildToolLine(name, input, OutputForwarder.isHidden);
|
|
618
|
+
};
|
|
619
|
+
// ── File watch ─────────────────────────────────────────────────────
|
|
620
|
+
/** Set callback for idle detection (no new data for 5s after assistant entry) */
|
|
621
|
+
setOnIdle(cb) {
|
|
622
|
+
this.onIdleCallback = cb;
|
|
623
|
+
}
|
|
624
|
+
/** Start watching transcript file for changes (runs once, never stops) */
|
|
625
|
+
startWatch() {
|
|
626
|
+
if (!this.transcriptPath) return;
|
|
627
|
+
if (this.watchingPath === this.transcriptPath && this.watcher) {
|
|
628
|
+
dropTrace("watch.start.skip", { reason: "already_watching", path: this.watchingPath });
|
|
629
|
+
return;
|
|
630
|
+
}
|
|
631
|
+
this.closeWatcher();
|
|
632
|
+
this.watchingPath = this.transcriptPath;
|
|
633
|
+
dropTrace("watch.start.install", { path: this.watchingPath });
|
|
634
|
+
try {
|
|
635
|
+
this.watcher = watch(this.transcriptPath, () => this.scheduleWatchFlush());
|
|
636
|
+
this.watcher.on("error", (err) => {
|
|
637
|
+
dropTrace("watch.error", { path: this.watchingPath, err: String(err) });
|
|
638
|
+
this.closeWatcher();
|
|
639
|
+
});
|
|
640
|
+
this.watcher.on("close", () => {
|
|
641
|
+
dropTrace("watch.close.event", { path: this.watchingPath });
|
|
642
|
+
});
|
|
643
|
+
// Cover bytes written between the stat in setContext() and watch install.
|
|
644
|
+
this.scheduleWatchFlush();
|
|
645
|
+
} catch (e) {
|
|
646
|
+
dropTrace("watch.start.catch", { path: this.watchingPath, err: String(e) });
|
|
647
|
+
this.closeWatcher();
|
|
648
|
+
}
|
|
649
|
+
}
|
|
650
|
+
/** Stop watching the transcript file. Delegates to closeWatcher() so
|
|
651
|
+
* callers that invoke stopWatch() on deactivation / ownership loss
|
|
652
|
+
* actually release the fs.watch handle + debounce/retry timers
|
|
653
|
+
* instead of leaking them for the lifetime of the process. */
|
|
654
|
+
stopWatch() {
|
|
655
|
+
this.closeWatcher();
|
|
656
|
+
}
|
|
657
|
+
/** Reset the idle timer — safety net in case turn-end signal is missed */
|
|
658
|
+
resetIdleTimer() {
|
|
659
|
+
if (this.idleTimer) clearTimeout(this.idleTimer);
|
|
660
|
+
this.idleTimer = setTimeout(() => {
|
|
661
|
+
this.idleTimer = null;
|
|
662
|
+
if (this.onIdleCallback) this.onIdleCallback();
|
|
663
|
+
}, 1e3);
|
|
664
|
+
}
|
|
665
|
+
closeWatcher() {
|
|
666
|
+
dropTrace("watch.close.call", { watcher: !!this.watcher, watchDebounce: !!this.watchDebounce, sendRetryTimer: !!this.sendRetryTimer, finalLen: this.finalLane.length, streamLen: this.streamLane.length, path: this.watchingPath || "(none)" });
|
|
667
|
+
if (this.watchDebounce) {
|
|
668
|
+
clearTimeout(this.watchDebounce);
|
|
669
|
+
this.watchDebounce = null;
|
|
670
|
+
}
|
|
671
|
+
if (this.sendRetryTimer) {
|
|
672
|
+
clearTimeout(this.sendRetryTimer);
|
|
673
|
+
this.sendRetryTimer = null;
|
|
674
|
+
}
|
|
675
|
+
if (this.watcher) {
|
|
676
|
+
this.watcher.close();
|
|
677
|
+
this.watcher = null;
|
|
678
|
+
}
|
|
679
|
+
this.watchingPath = "";
|
|
680
|
+
}
|
|
681
|
+
scheduleWatchFlush() {
|
|
682
|
+
if (this.watchDebounce) clearTimeout(this.watchDebounce);
|
|
683
|
+
this.watchDebounce = setTimeout(() => {
|
|
684
|
+
this.watchDebounce = null;
|
|
685
|
+
let _wfStat = null;
|
|
686
|
+
if (this.transcriptPath) {
|
|
687
|
+
try { _wfStat = statSync(this.transcriptPath); } catch {}
|
|
688
|
+
}
|
|
689
|
+
if (this.transcriptPath && !_wfStat) {
|
|
690
|
+
const relocated = detectCurrentSessionTranscript()?.transcriptPath ?? findLatestTranscriptByMtime();
|
|
691
|
+
if (relocated && relocated !== this.transcriptPath) {
|
|
692
|
+
process.stderr.write(`mixdog: watched transcript gone during flush, relocated to ${relocated}
|
|
693
|
+
`);
|
|
694
|
+
dropTrace("watch.flush.relocate", { from: this.transcriptPath, to: relocated });
|
|
695
|
+
this.closeWatcher();
|
|
696
|
+
this.transcriptPath = relocated;
|
|
697
|
+
this.mainSessionId = "";
|
|
698
|
+
this.startWatch();
|
|
699
|
+
}
|
|
700
|
+
return;
|
|
701
|
+
}
|
|
702
|
+
this._pendingStat = _wfStat;
|
|
703
|
+
this.forwardNewText().then((hadText) => {
|
|
704
|
+
// Only trace when forwardNewText actually emitted text. hadText=false flushes
|
|
705
|
+
// fire on every poll cycle and accumulate ~1MB/hour of identical no-op rows.
|
|
706
|
+
if (hadText) dropTrace("watch.flush", { hadText, transcriptPath: this.transcriptPath || "(none)", watchingPath: this.watchingPath || "(none)" });
|
|
707
|
+
if (hadText) {
|
|
708
|
+
this.resetIdleTimer();
|
|
709
|
+
}
|
|
710
|
+
}).catch(err => process.stderr.write(`[output-forwarder] forwardNewText rejected: ${err?.message || err}\n`));
|
|
711
|
+
}, 100);
|
|
712
|
+
}
|
|
713
|
+
updateState(mutator) {
|
|
714
|
+
this.statusState.update(mutator);
|
|
715
|
+
}
|
|
716
|
+
// Debounced: every commitReadProgress() used to fire a full tmp+fsync+rename+
|
|
717
|
+
// dir-fsync cycle through state-file.mjs writeJsonFile. Under steady
|
|
718
|
+
// transcript progress (Discord forwarder following a live session) that
|
|
719
|
+
// hit disk 5–10×/sec. Coalesce updates into a single write per 1.5s
|
|
720
|
+
// window; final flush on process exit / explicit flushPersistState().
|
|
721
|
+
persistState() {
|
|
722
|
+
this._pendingPersistData = {
|
|
723
|
+
lastFileSize: this.lastFileSize,
|
|
724
|
+
sentCount: this.sentCount,
|
|
725
|
+
lastSentHash: this.lastHash,
|
|
726
|
+
lastSentTime: Date.now(),
|
|
727
|
+
emoji: this.emoji,
|
|
728
|
+
sessionIdle: false,
|
|
729
|
+
};
|
|
730
|
+
if (this._persistTimer) return;
|
|
731
|
+
this._persistTimer = setTimeout(() => {
|
|
732
|
+
this._persistTimer = null;
|
|
733
|
+
this._flushPersistState();
|
|
734
|
+
}, 1500);
|
|
735
|
+
if (this._persistTimer.unref) this._persistTimer.unref();
|
|
736
|
+
}
|
|
737
|
+
flushPersistState() { this._flushPersistState(); }
|
|
738
|
+
_flushPersistState() {
|
|
739
|
+
const data = this._pendingPersistData;
|
|
740
|
+
if (!data) return;
|
|
741
|
+
this._pendingPersistData = null;
|
|
742
|
+
if (this._persistTimer) {
|
|
743
|
+
clearTimeout(this._persistTimer);
|
|
744
|
+
this._persistTimer = null;
|
|
745
|
+
}
|
|
746
|
+
this.updateState((state) => {
|
|
747
|
+
state.lastFileSize = data.lastFileSize;
|
|
748
|
+
state.sentCount = data.sentCount;
|
|
749
|
+
state.lastSentHash = data.lastSentHash;
|
|
750
|
+
state.lastSentTime = data.lastSentTime;
|
|
751
|
+
state.emoji = data.emoji;
|
|
752
|
+
state.sessionIdle = data.sessionIdle;
|
|
753
|
+
});
|
|
754
|
+
}
|
|
755
|
+
}
|
|
756
|
+
export {
|
|
757
|
+
OutputForwarder,
|
|
758
|
+
cwdToProjectSlug,
|
|
759
|
+
detectCurrentSessionTranscript,
|
|
760
|
+
discoverCurrentClaudeSession,
|
|
761
|
+
discoverSessionBoundTranscript,
|
|
762
|
+
findLatestTranscriptByMtime,
|
|
763
|
+
getLatestInteractiveClaudeSession,
|
|
764
|
+
listInteractiveClaudeSessions
|
|
765
|
+
};
|