@ouro.bot/cli 0.1.0-alpha.55 → 0.1.0-alpha.550
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/README.md +133 -19
- package/RepairGuide.ouro/agent.json +5 -0
- package/RepairGuide.ouro/psyche/IDENTITY.md +19 -0
- package/RepairGuide.ouro/psyche/SOUL.md +55 -0
- package/RepairGuide.ouro/skills/diagnose-bootstrap-drift.md +54 -0
- package/RepairGuide.ouro/skills/diagnose-broken-remote.md +63 -0
- package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
- package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
- package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/changelog.json +3555 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +140 -0
- package/dist/arc/episodes.js +117 -0
- package/dist/arc/intentions.js +133 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +237 -0
- package/dist/arc/packets.js +193 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +837 -26
- package/dist/heart/agent-entry.js +58 -3
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/adapter.js +2 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +479 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +111 -0
- package/dist/heart/config-registry.js +304 -0
- package/dist/heart/config.js +114 -118
- package/dist/heart/core.js +925 -246
- package/dist/heart/cross-chat-delivery.js +3 -18
- package/dist/heart/daemon/agent-config-check.js +512 -0
- package/dist/heart/daemon/agent-discovery.js +102 -3
- package/dist/heart/daemon/agent-service.js +522 -0
- package/dist/heart/daemon/agentic-repair.js +554 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/boot-sync-probe.js +197 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +665 -0
- package/dist/heart/daemon/cli-exec.js +7565 -0
- package/dist/heart/daemon/cli-help.js +498 -0
- package/dist/heart/daemon/cli-parse.js +1590 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +775 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/connect-bay.js +323 -0
- package/dist/heart/daemon/daemon-cli.js +29 -1672
- package/dist/heart/daemon/daemon-entry.js +417 -2
- package/dist/heart/daemon/daemon-health.js +183 -0
- package/dist/heart/daemon/daemon-rollup.js +58 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +87 -13
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +758 -71
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +844 -0
- package/dist/heart/daemon/drift-detection.js +146 -0
- package/dist/heart/daemon/health-monitor.js +122 -1
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
- package/dist/heart/daemon/http-health-probe.js +80 -0
- package/dist/heart/daemon/human-command-screens.js +234 -0
- package/dist/heart/daemon/human-readiness.js +114 -0
- package/dist/heart/daemon/inner-status.js +102 -0
- package/dist/heart/daemon/interactive-repair.js +394 -0
- package/dist/heart/daemon/launchd.js +37 -8
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- package/dist/heart/daemon/message-router.js +2 -2
- package/dist/heart/daemon/os-cron-deps.js +135 -0
- package/dist/heart/daemon/os-cron.js +14 -12
- package/dist/heart/daemon/ouro-bot-entry.js +4 -2
- package/dist/heart/daemon/ouro-entry.js +3 -1
- package/dist/heart/daemon/process-manager.js +375 -33
- package/dist/heart/daemon/provider-discovery.js +137 -0
- package/dist/heart/daemon/provider-ping-progress.js +83 -0
- package/dist/heart/daemon/pulse.js +475 -0
- package/dist/heart/daemon/readiness-repair.js +365 -0
- package/dist/heart/daemon/run-hooks.js +2 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +3 -31
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +353 -38
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +158 -11
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +330 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +162 -17
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +1 -1
- package/dist/heart/habits/habit-migration.js +189 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-runtime-state.js +100 -0
- package/dist/heart/habits/habit-scheduler.js +372 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
- package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
- package/dist/heart/identity.js +200 -51
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +1 -1
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
- package/dist/heart/mailbox/mailbox-http-response.js +7 -0
- package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
- package/dist/heart/mailbox/mailbox-http-static.js +103 -0
- package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
- package/dist/heart/mailbox/mailbox-http.js +99 -0
- package/dist/heart/mailbox/mailbox-read.js +31 -0
- package/dist/heart/mailbox/mailbox-types.js +27 -0
- package/dist/heart/mailbox/mailbox-view.js +195 -0
- package/dist/heart/mailbox/readers/agent-machine.js +382 -0
- package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
- package/dist/heart/mailbox/readers/mail.js +362 -0
- package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
- package/dist/heart/mailbox/readers/sessions.js +232 -0
- package/dist/heart/mailbox/readers/shared.js +111 -0
- package/dist/heart/mcp/mcp-server.js +683 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +19 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +255 -0
- package/dist/heart/provider-credentials.js +425 -0
- package/dist/heart/provider-failover.js +301 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +262 -0
- package/dist/heart/provider-state.js +216 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +139 -52
- package/dist/heart/providers/azure.js +97 -13
- package/dist/heart/providers/error-classification.js +127 -0
- package/dist/heart/providers/github-copilot.js +145 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +26 -8
- package/dist/heart/providers/openai-codex.js +55 -40
- package/dist/heart/runtime-capability-check.js +170 -0
- package/dist/heart/runtime-credentials.js +367 -0
- package/dist/heart/runtime-cwd.js +87 -0
- package/dist/heart/sense-truth.js +11 -4
- package/dist/heart/session-activity.js +43 -22
- package/dist/heart/session-events.js +1149 -0
- package/dist/heart/session-playback-cli-main.js +5 -0
- package/dist/heart/session-playback-cli.js +36 -0
- package/dist/heart/session-playback.js +231 -0
- package/dist/heart/session-stats-cli-main.js +5 -0
- package/dist/heart/session-stats.js +182 -0
- package/dist/heart/session-transcript.js +243 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +44 -27
- package/dist/heart/sync-classification.js +176 -0
- package/dist/heart/sync.js +449 -0
- package/dist/heart/target-resolution.js +9 -5
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/timeouts.js +101 -0
- package/dist/heart/tool-activity-callbacks.js +59 -0
- package/dist/heart/tool-description.js +139 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +381 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +426 -0
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/mailbox-ui/assets/index-Cm51CY9W.js +61 -0
- package/dist/mailbox-ui/index.html +15 -0
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +674 -0
- package/dist/mailroom/body-cache.js +61 -0
- package/dist/mailroom/core.js +720 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +430 -0
- package/dist/mailroom/mbox-import.js +383 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +233 -0
- package/dist/mailroom/search-cache.js +256 -0
- package/dist/mailroom/search-relevance.js +319 -0
- package/dist/mailroom/smtp-ingress.js +176 -0
- package/dist/mailroom/source-state.js +176 -0
- package/dist/mailroom/thread.js +109 -0
- package/dist/mailroom/travel-extract.js +89 -0
- package/dist/mind/bundle-manifest.js +7 -1
- package/dist/mind/context.js +165 -101
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +62 -75
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/friends/channel.js +30 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +39 -3
- package/dist/mind/friends/types.js +2 -2
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +4 -0
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +995 -123
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +139 -5
- package/dist/nerves/coverage/run-artifacts.js +1 -1
- package/dist/nerves/event-buffer.js +111 -0
- package/dist/nerves/index.js +224 -4
- package/dist/nerves/observation.js +20 -0
- package/dist/nerves/redact.js +79 -0
- package/dist/nerves/review/cli-main.js +5 -0
- package/dist/nerves/review/cli.js +156 -0
- package/dist/nerves/review/core.js +152 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/repertoire/ado-client.js +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +816 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +180 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +158 -9
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +170 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +111 -0
- package/dist/repertoire/duffel-client.js +185 -0
- package/dist/repertoire/github-client.js +14 -55
- package/dist/repertoire/graph-client.js +11 -52
- package/dist/repertoire/guardrails.js +396 -0
- package/dist/repertoire/mcp-client.js +295 -0
- package/dist/repertoire/mcp-manager.js +362 -0
- package/dist/repertoire/mcp-tools.js +63 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +15 -24
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +31 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +16 -4
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -78
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-base.js +47 -1075
- package/dist/repertoire/tools-bluebubbles.js +1 -0
- package/dist/repertoire/tools-bridge.js +142 -0
- package/dist/repertoire/tools-bundle.js +984 -0
- package/dist/repertoire/tools-config.js +185 -0
- package/dist/repertoire/tools-continuity.js +248 -0
- package/dist/repertoire/tools-credential.js +381 -0
- package/dist/repertoire/tools-files.js +342 -0
- package/dist/repertoire/tools-flight.js +224 -0
- package/dist/repertoire/tools-flow.js +119 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1857 -0
- package/dist/repertoire/tools-notes.js +421 -0
- package/dist/repertoire/tools-session.js +750 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +243 -0
- package/dist/repertoire/tools-teams.js +9 -39
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +604 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools.js +108 -100
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +131 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +561 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/bluebubbles/active-turns.js +216 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
- package/dist/senses/bluebubbles/entry.js +77 -0
- package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
- package/dist/senses/bluebubbles/index.js +2305 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
- package/dist/senses/bluebubbles/processed-log.js +133 -0
- package/dist/senses/bluebubbles/replay.js +137 -0
- package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +607 -0
- package/dist/senses/cli/spinner-imperative.js +135 -0
- package/dist/senses/cli/spinner.js +101 -0
- package/dist/senses/cli/status-line.js +60 -0
- package/dist/senses/cli/streaming-markdown.js +526 -0
- package/dist/senses/cli/tool-display.js +85 -0
- package/dist/senses/cli/tool-render.js +85 -0
- package/dist/senses/cli/tui-store.js +240 -0
- package/dist/senses/cli/virtual-list.js +35 -0
- package/dist/senses/cli-entry.js +60 -8
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +520 -209
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +175 -21
- package/dist/senses/inner-dialog.js +330 -27
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +569 -182
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +248 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +387 -98
- package/dist/senses/trust-gate.js +100 -5
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +146 -0
- package/package.json +38 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +101 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/ouro-path-installer.js +0 -178
- package/dist/heart/daemon/subagent-installer.js +0 -166
- package/dist/heart/session-recall.js +0 -116
- package/dist/mind/associative-recall.js +0 -209
- package/dist/senses/bluebubbles-entry.js +0 -13
- package/dist/senses/bluebubbles.js +0 -1177
- package/dist/senses/debug-activity.js +0 -148
- package/subagents/README.md +0 -86
- package/subagents/work-doer.md +0 -237
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -390
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
- /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
|
@@ -38,10 +38,13 @@ const fs = __importStar(require("fs"));
|
|
|
38
38
|
const os = __importStar(require("os"));
|
|
39
39
|
const path = __importStar(require("path"));
|
|
40
40
|
const runtime_1 = require("../../nerves/runtime");
|
|
41
|
-
const bluebubbles_runtime_state_1 = require("../../senses/bluebubbles-runtime-state");
|
|
42
41
|
const identity_1 = require("../identity");
|
|
42
|
+
const runtime_credentials_1 = require("../runtime-credentials");
|
|
43
|
+
const provider_credentials_1 = require("../provider-credentials");
|
|
43
44
|
const sense_truth_1 = require("../sense-truth");
|
|
45
|
+
const machine_identity_1 = require("../machine-identity");
|
|
44
46
|
const process_manager_1 = require("./process-manager");
|
|
47
|
+
const http_health_probe_1 = require("./http-health-probe");
|
|
45
48
|
const DEFAULT_TEAMS_PORT = 3978;
|
|
46
49
|
const DEFAULT_BLUEBUBBLES_PORT = 18790;
|
|
47
50
|
const DEFAULT_BLUEBUBBLES_WEBHOOK_PATH = "/bluebubbles-webhook";
|
|
@@ -51,6 +54,7 @@ function defaultSenses() {
|
|
|
51
54
|
cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
|
|
52
55
|
teams: { ...identity_1.DEFAULT_AGENT_SENSES.teams },
|
|
53
56
|
bluebubbles: { ...identity_1.DEFAULT_AGENT_SENSES.bluebubbles },
|
|
57
|
+
mail: { ...identity_1.DEFAULT_AGENT_SENSES.mail },
|
|
54
58
|
};
|
|
55
59
|
}
|
|
56
60
|
function readAgentSenses(agentJsonPath) {
|
|
@@ -76,7 +80,7 @@ function readAgentSenses(agentJsonPath) {
|
|
|
76
80
|
if (!rawSenses || typeof rawSenses !== "object" || Array.isArray(rawSenses)) {
|
|
77
81
|
return defaults;
|
|
78
82
|
}
|
|
79
|
-
for (const sense of ["cli", "teams", "bluebubbles"]) {
|
|
83
|
+
for (const sense of ["cli", "teams", "bluebubbles", "mail"]) {
|
|
80
84
|
const rawSense = rawSenses[sense];
|
|
81
85
|
if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
|
|
82
86
|
continue;
|
|
@@ -88,22 +92,6 @@ function readAgentSenses(agentJsonPath) {
|
|
|
88
92
|
}
|
|
89
93
|
return defaults;
|
|
90
94
|
}
|
|
91
|
-
function readSecretsPayload(secretsPath) {
|
|
92
|
-
try {
|
|
93
|
-
const raw = fs.readFileSync(secretsPath, "utf-8");
|
|
94
|
-
const parsed = JSON.parse(raw);
|
|
95
|
-
if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
|
|
96
|
-
return { payload: {}, error: "invalid secrets.json object" };
|
|
97
|
-
}
|
|
98
|
-
return { payload: parsed, error: null };
|
|
99
|
-
}
|
|
100
|
-
catch (error) {
|
|
101
|
-
return {
|
|
102
|
-
payload: {},
|
|
103
|
-
error: error instanceof Error ? error.message : String(error),
|
|
104
|
-
};
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
95
|
function textField(record, key) {
|
|
108
96
|
const value = record?.[key];
|
|
109
97
|
return typeof value === "string" ? value.trim() : "";
|
|
@@ -112,17 +100,36 @@ function numberField(record, key, fallback) {
|
|
|
112
100
|
const value = record?.[key];
|
|
113
101
|
return typeof value === "number" && Number.isFinite(value) ? value : fallback;
|
|
114
102
|
}
|
|
115
|
-
function
|
|
103
|
+
function compactRuntimeConfigError(agent, error) {
|
|
104
|
+
const compact = error.replace(/\s+/g, " ").trim();
|
|
105
|
+
if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
|
|
106
|
+
return `vault locked; run 'ouro vault unlock --agent ${agent}' if you have the saved secret, or 'ouro vault replace --agent ${agent}' if none was saved`;
|
|
107
|
+
}
|
|
108
|
+
return compact || "unavailable";
|
|
109
|
+
}
|
|
110
|
+
function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
|
|
111
|
+
if (runtimeConfig.ok)
|
|
112
|
+
return "";
|
|
113
|
+
const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
|
|
114
|
+
if (runtimeConfig.reason === "missing")
|
|
115
|
+
return `missing vault ${itemName} (${agent})`;
|
|
116
|
+
return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
|
|
117
|
+
}
|
|
118
|
+
function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
|
|
116
119
|
const base = {
|
|
117
120
|
cli: { configured: true, detail: "local interactive terminal" },
|
|
118
121
|
teams: { configured: false, detail: "not enabled in agent.json" },
|
|
119
122
|
bluebubbles: { configured: false, detail: "not enabled in agent.json" },
|
|
123
|
+
mail: { configured: false, detail: "not enabled in agent.json" },
|
|
120
124
|
};
|
|
121
|
-
const
|
|
125
|
+
const payload = runtimeConfig.ok ? runtimeConfig.config : {};
|
|
126
|
+
const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
|
|
122
127
|
const teams = payload.teams;
|
|
123
128
|
const teamsChannel = payload.teamsChannel;
|
|
124
|
-
const
|
|
125
|
-
const
|
|
129
|
+
const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
|
|
130
|
+
const bluebubbles = machinePayload.bluebubbles;
|
|
131
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
132
|
+
const mailroom = payload.mailroom;
|
|
126
133
|
if (senses.teams.enabled) {
|
|
127
134
|
const missing = [];
|
|
128
135
|
if (!textField(teams, "clientId"))
|
|
@@ -138,9 +145,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
138
145
|
}
|
|
139
146
|
: {
|
|
140
147
|
configured: false,
|
|
141
|
-
detail:
|
|
142
|
-
? `missing
|
|
143
|
-
:
|
|
148
|
+
detail: runtimeConfig.ok
|
|
149
|
+
? `missing ${missing.join("/")}`
|
|
150
|
+
: unavailableDetail,
|
|
144
151
|
};
|
|
145
152
|
}
|
|
146
153
|
if (senses.bluebubbles.enabled) {
|
|
@@ -156,19 +163,52 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
156
163
|
}
|
|
157
164
|
: {
|
|
158
165
|
configured: false,
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
166
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
167
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
168
|
+
? "not attached on this machine"
|
|
169
|
+
: machineRuntimeConfig.ok
|
|
170
|
+
? `missing ${missing.join("/")}`
|
|
171
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
172
|
+
};
|
|
173
|
+
}
|
|
174
|
+
if (senses.mail.enabled) {
|
|
175
|
+
const privateKeys = mailroom?.privateKeys;
|
|
176
|
+
const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
|
|
177
|
+
const mailboxAddress = textField(mailroom, "mailboxAddress");
|
|
178
|
+
const missing = [];
|
|
179
|
+
if (!mailboxAddress)
|
|
180
|
+
missing.push("mailroom.mailboxAddress");
|
|
181
|
+
if (!hasPrivateKeys)
|
|
182
|
+
missing.push("mailroom.privateKeys");
|
|
183
|
+
base.mail = missing.length === 0
|
|
184
|
+
? { configured: true, detail: mailboxAddress }
|
|
185
|
+
: {
|
|
186
|
+
configured: false,
|
|
187
|
+
detail: runtimeConfig.ok
|
|
188
|
+
? `missing ${missing.join("/")}`
|
|
189
|
+
: unavailableDetail,
|
|
162
190
|
};
|
|
163
191
|
}
|
|
164
192
|
return base;
|
|
165
193
|
}
|
|
194
|
+
function senseRepairHint(agent, sense) {
|
|
195
|
+
if (sense === "teams") {
|
|
196
|
+
return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
|
|
197
|
+
}
|
|
198
|
+
if (sense === "mail") {
|
|
199
|
+
return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
|
|
200
|
+
}
|
|
201
|
+
return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
|
|
202
|
+
}
|
|
203
|
+
function currentMachineId() {
|
|
204
|
+
return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
|
|
205
|
+
}
|
|
166
206
|
function parseSenseSnapshotName(name) {
|
|
167
207
|
const parts = name.split(":");
|
|
168
208
|
if (parts.length !== 2)
|
|
169
209
|
return null;
|
|
170
210
|
const [agent, sense] = parts;
|
|
171
|
-
if (sense !== "teams" && sense !== "bluebubbles")
|
|
211
|
+
if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail")
|
|
172
212
|
return null;
|
|
173
213
|
return { agent, sense };
|
|
174
214
|
}
|
|
@@ -177,6 +217,32 @@ function runtimeInfoFor(status) {
|
|
|
177
217
|
return { runtime: "running" };
|
|
178
218
|
return { runtime: "error" };
|
|
179
219
|
}
|
|
220
|
+
function managedSenseEntry(sense) {
|
|
221
|
+
if (sense === "teams")
|
|
222
|
+
return "senses/teams-entry.js";
|
|
223
|
+
if (sense === "bluebubbles")
|
|
224
|
+
return "senses/bluebubbles/entry.js";
|
|
225
|
+
return "senses/mail-entry.js";
|
|
226
|
+
}
|
|
227
|
+
function runtimeCredentialBootstrapFor(agent, sense) {
|
|
228
|
+
const runtime = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
229
|
+
const machineId = sense === "bluebubbles" ? currentMachineId() : undefined;
|
|
230
|
+
const machine = sense === "bluebubbles" ? (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent) : null;
|
|
231
|
+
const providerPool = (0, provider_credentials_1.readProviderCredentialPool)(agent);
|
|
232
|
+
const providerCredentialRecords = providerPool.ok
|
|
233
|
+
? Object.values(providerPool.pool.providers).filter((record) => !!record)
|
|
234
|
+
: [];
|
|
235
|
+
const bootstrap = {
|
|
236
|
+
agentName: agent,
|
|
237
|
+
runtimeConfig: runtime.ok ? runtime.config : undefined,
|
|
238
|
+
machineRuntimeConfig: machine?.ok ? machine.config : undefined,
|
|
239
|
+
machineId,
|
|
240
|
+
providerCredentialRecords: providerCredentialRecords.length > 0 ? providerCredentialRecords : undefined,
|
|
241
|
+
};
|
|
242
|
+
if (!bootstrap.runtimeConfig && !bootstrap.machineRuntimeConfig && !bootstrap.providerCredentialRecords)
|
|
243
|
+
return null;
|
|
244
|
+
return bootstrap;
|
|
245
|
+
}
|
|
180
246
|
function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
181
247
|
if (!lastCheckedAt) {
|
|
182
248
|
return false;
|
|
@@ -187,53 +253,186 @@ function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
|
187
253
|
}
|
|
188
254
|
return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
|
|
189
255
|
}
|
|
256
|
+
function readBlueBubblesRuntimeJson(runtimePath) {
|
|
257
|
+
try {
|
|
258
|
+
const raw = fs.readFileSync(runtimePath, "utf-8");
|
|
259
|
+
const parsed = JSON.parse(raw);
|
|
260
|
+
/* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
|
|
261
|
+
return {
|
|
262
|
+
upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
|
|
263
|
+
? parsed.upstreamStatus
|
|
264
|
+
: "unknown",
|
|
265
|
+
detail: typeof parsed.detail === "string" && parsed.detail.trim()
|
|
266
|
+
? parsed.detail
|
|
267
|
+
: "startup health probe pending",
|
|
268
|
+
lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
|
|
269
|
+
proofMethod: typeof parsed.proofMethod === "string" && parsed.proofMethod.trim()
|
|
270
|
+
? parsed.proofMethod
|
|
271
|
+
: undefined,
|
|
272
|
+
pendingRecoveryCount: typeof parsed.pendingRecoveryCount === "number" && Number.isFinite(parsed.pendingRecoveryCount)
|
|
273
|
+
? parsed.pendingRecoveryCount
|
|
274
|
+
: 0,
|
|
275
|
+
failedRecoveryCount: typeof parsed.failedRecoveryCount === "number" && Number.isFinite(parsed.failedRecoveryCount)
|
|
276
|
+
? parsed.failedRecoveryCount
|
|
277
|
+
: 0,
|
|
278
|
+
oldestPendingRecoveryAt: typeof parsed.oldestPendingRecoveryAt === "string" ? parsed.oldestPendingRecoveryAt : undefined,
|
|
279
|
+
oldestPendingRecoveryAgeMs: typeof parsed.oldestPendingRecoveryAgeMs === "number" && Number.isFinite(parsed.oldestPendingRecoveryAgeMs)
|
|
280
|
+
? parsed.oldestPendingRecoveryAgeMs
|
|
281
|
+
: undefined,
|
|
282
|
+
activeTurnCount: typeof parsed.activeTurnCount === "number" && Number.isFinite(parsed.activeTurnCount)
|
|
283
|
+
? parsed.activeTurnCount
|
|
284
|
+
: undefined,
|
|
285
|
+
stalledTurnCount: typeof parsed.stalledTurnCount === "number" && Number.isFinite(parsed.stalledTurnCount)
|
|
286
|
+
? parsed.stalledTurnCount
|
|
287
|
+
: undefined,
|
|
288
|
+
oldestActiveTurnStartedAt: typeof parsed.oldestActiveTurnStartedAt === "string" ? parsed.oldestActiveTurnStartedAt : undefined,
|
|
289
|
+
oldestActiveTurnAgeMs: typeof parsed.oldestActiveTurnAgeMs === "number" && Number.isFinite(parsed.oldestActiveTurnAgeMs)
|
|
290
|
+
? parsed.oldestActiveTurnAgeMs
|
|
291
|
+
: undefined,
|
|
292
|
+
};
|
|
293
|
+
/* v8 ignore stop */
|
|
294
|
+
/* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
|
|
295
|
+
}
|
|
296
|
+
catch {
|
|
297
|
+
return { upstreamStatus: "unknown", detail: "startup health probe pending", pendingRecoveryCount: 0, failedRecoveryCount: 0 };
|
|
298
|
+
}
|
|
299
|
+
/* v8 ignore stop */
|
|
300
|
+
}
|
|
190
301
|
function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
|
|
191
302
|
const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
192
303
|
const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
|
|
193
304
|
if (!fs.existsSync(runtimePath)) {
|
|
194
305
|
return { runtime: snapshot?.runtime };
|
|
195
306
|
}
|
|
196
|
-
const state = (
|
|
307
|
+
const state = readBlueBubblesRuntimeJson(runtimePath);
|
|
308
|
+
const checkedAtMs = state.lastCheckedAt ? Date.parse(state.lastCheckedAt) : Number.NaN;
|
|
309
|
+
const proofFacts = {
|
|
310
|
+
proofMethod: state.proofMethod ?? "bluebubbles.checkHealth",
|
|
311
|
+
lastProofAt: state.lastCheckedAt,
|
|
312
|
+
proofAgeMs: Number.isFinite(checkedAtMs) ? Math.max(0, Date.now() - checkedAtMs) : undefined,
|
|
313
|
+
pendingRecoveryCount: state.pendingRecoveryCount,
|
|
314
|
+
failedRecoveryCount: state.failedRecoveryCount,
|
|
315
|
+
oldestPendingRecoveryAt: state.oldestPendingRecoveryAt,
|
|
316
|
+
oldestPendingRecoveryAgeMs: state.oldestPendingRecoveryAgeMs,
|
|
317
|
+
activeTurnCount: state.activeTurnCount,
|
|
318
|
+
stalledTurnCount: state.stalledTurnCount,
|
|
319
|
+
oldestActiveTurnStartedAt: state.oldestActiveTurnStartedAt,
|
|
320
|
+
oldestActiveTurnAgeMs: state.oldestActiveTurnAgeMs,
|
|
321
|
+
};
|
|
197
322
|
if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
|
|
198
|
-
return {
|
|
323
|
+
return {
|
|
324
|
+
runtime: snapshot?.runtime,
|
|
325
|
+
lastFailure: state.lastCheckedAt ? "BlueBubbles proof is stale" : undefined,
|
|
326
|
+
failureLayer: state.lastCheckedAt ? "proof_freshness" : undefined,
|
|
327
|
+
};
|
|
328
|
+
}
|
|
329
|
+
if (snapshot?.runtime !== "running") {
|
|
330
|
+
return {
|
|
331
|
+
runtime: "error",
|
|
332
|
+
detail: "BlueBubbles listener is not running",
|
|
333
|
+
...proofFacts,
|
|
334
|
+
lastFailure: "listener process is not running",
|
|
335
|
+
failureLayer: "listener",
|
|
336
|
+
recoveryAction: "daemon health monitor will restart the BlueBubbles listener when its probe fails",
|
|
337
|
+
};
|
|
199
338
|
}
|
|
200
339
|
if (state.upstreamStatus === "error") {
|
|
201
340
|
return {
|
|
202
341
|
runtime: "error",
|
|
203
342
|
detail: state.detail,
|
|
343
|
+
...proofFacts,
|
|
344
|
+
lastFailure: state.detail,
|
|
345
|
+
failureLayer: "upstream",
|
|
346
|
+
recoveryAction: "verify BlueBubbles server/app auth and local machine attachment, then retry the listener",
|
|
347
|
+
};
|
|
348
|
+
}
|
|
349
|
+
if (state.pendingRecoveryCount > 0) {
|
|
350
|
+
return {
|
|
351
|
+
runtime: "error",
|
|
352
|
+
detail: state.detail,
|
|
353
|
+
...proofFacts,
|
|
354
|
+
lastFailure: state.detail,
|
|
355
|
+
failureLayer: "recovery_queue",
|
|
356
|
+
recoveryAction: "queued recovery will retry; inspect BlueBubbles inbound/recovery sidecar logs if age keeps growing",
|
|
357
|
+
};
|
|
358
|
+
}
|
|
359
|
+
if ((state.stalledTurnCount ?? 0) > 0) {
|
|
360
|
+
return {
|
|
361
|
+
runtime: "error",
|
|
362
|
+
detail: state.detail,
|
|
363
|
+
...proofFacts,
|
|
364
|
+
lastFailure: state.detail,
|
|
365
|
+
failureLayer: "live_turn_stall",
|
|
366
|
+
recoveryAction: "live iMessage turn timeout/watchdog will release the lane and recovery will retry captured messages",
|
|
204
367
|
};
|
|
205
368
|
}
|
|
206
369
|
if (state.upstreamStatus === "ok") {
|
|
207
|
-
return {
|
|
370
|
+
return {
|
|
371
|
+
runtime: "running",
|
|
372
|
+
...proofFacts,
|
|
373
|
+
...(state.failedRecoveryCount > 0 ? { detail: state.detail } : {}),
|
|
374
|
+
...(state.failedRecoveryCount > 0 ? {
|
|
375
|
+
lastFailure: state.detail,
|
|
376
|
+
failureLayer: "recovery_quarantine",
|
|
377
|
+
recoveryAction: "inspect quarantined BlueBubbles recovery failures; live transport remains reachable",
|
|
378
|
+
} : {}),
|
|
379
|
+
};
|
|
208
380
|
}
|
|
209
|
-
return { runtime: snapshot?.runtime };
|
|
381
|
+
return { runtime: snapshot?.runtime, ...proofFacts };
|
|
210
382
|
}
|
|
211
383
|
class DaemonSenseManager {
|
|
212
384
|
processManager;
|
|
213
385
|
contexts;
|
|
386
|
+
pendingConfigRefreshes = new Set();
|
|
214
387
|
bundlesRoot;
|
|
215
388
|
constructor(options) {
|
|
216
389
|
const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
217
|
-
const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
|
|
218
390
|
this.bundlesRoot = bundlesRoot;
|
|
219
391
|
this.contexts = new Map(options.agents.map((agent) => {
|
|
220
392
|
const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
|
|
221
|
-
const facts =
|
|
393
|
+
const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
222
394
|
return [agent, { senses, facts }];
|
|
223
395
|
}));
|
|
224
396
|
const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
225
|
-
return ["teams", "bluebubbles"]
|
|
226
|
-
.filter((sense) => context.senses[sense].enabled
|
|
397
|
+
return ["teams", "bluebubbles", "mail"]
|
|
398
|
+
.filter((sense) => context.senses[sense].enabled)
|
|
227
399
|
.map((sense) => ({
|
|
228
400
|
name: `${agent}:${sense}`,
|
|
229
401
|
agentArg: agent,
|
|
230
|
-
entry: sense
|
|
402
|
+
entry: managedSenseEntry(sense),
|
|
231
403
|
channel: sense,
|
|
232
404
|
autoStart: true,
|
|
405
|
+
getRuntimeCredentialBootstrap: () => runtimeCredentialBootstrapFor(agent, sense),
|
|
233
406
|
}));
|
|
234
407
|
});
|
|
235
408
|
this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
|
|
236
409
|
agents: managedSenseAgents,
|
|
410
|
+
configCheck: async (name) => {
|
|
411
|
+
const parsed = parseSenseSnapshotName(name);
|
|
412
|
+
if (!parsed)
|
|
413
|
+
return { ok: true };
|
|
414
|
+
const context = this.contexts.get(parsed.agent);
|
|
415
|
+
if (!context)
|
|
416
|
+
return { ok: true };
|
|
417
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(parsed.agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent));
|
|
418
|
+
const fact = context.facts[parsed.sense];
|
|
419
|
+
if (fact.configured)
|
|
420
|
+
return { ok: true };
|
|
421
|
+
this.scheduleSenseConfigRefresh(name, parsed);
|
|
422
|
+
if (fact.optional) {
|
|
423
|
+
return {
|
|
424
|
+
ok: false,
|
|
425
|
+
skip: true,
|
|
426
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
|
|
427
|
+
};
|
|
428
|
+
}
|
|
429
|
+
return {
|
|
430
|
+
ok: false,
|
|
431
|
+
skip: true,
|
|
432
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
|
|
433
|
+
fix: senseRepairHint(parsed.agent, parsed.sense),
|
|
434
|
+
};
|
|
435
|
+
},
|
|
237
436
|
});
|
|
238
437
|
(0, runtime_1.emitNervesEvent)({
|
|
239
438
|
component: "channels",
|
|
@@ -245,12 +444,106 @@ class DaemonSenseManager {
|
|
|
245
444
|
},
|
|
246
445
|
});
|
|
247
446
|
}
|
|
447
|
+
scheduleSenseConfigRefresh(name, parsed) {
|
|
448
|
+
if (this.pendingConfigRefreshes.has(name))
|
|
449
|
+
return;
|
|
450
|
+
this.pendingConfigRefreshes.add(name);
|
|
451
|
+
void this.refreshSenseConfigAndRetry(name, parsed);
|
|
452
|
+
}
|
|
453
|
+
async refreshSenseConfigAndRetry(name, parsed) {
|
|
454
|
+
try {
|
|
455
|
+
const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
|
|
456
|
+
const machineRefreshed = parsed.sense === "bluebubbles"
|
|
457
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
458
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
|
|
459
|
+
const context = this.contexts.get(parsed.agent);
|
|
460
|
+
/* v8 ignore next -- defensive: config refreshes are only scheduled for known agent contexts @preserve */
|
|
461
|
+
if (!context)
|
|
462
|
+
return;
|
|
463
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
|
|
464
|
+
if (!context.facts[parsed.sense].configured)
|
|
465
|
+
return;
|
|
466
|
+
setTimeout(() => {
|
|
467
|
+
void this.processManager.startAgent?.(name).catch((error) => {
|
|
468
|
+
(0, runtime_1.emitNervesEvent)({
|
|
469
|
+
level: "error",
|
|
470
|
+
component: "channels",
|
|
471
|
+
event: "channel.daemon_sense_autostart_error",
|
|
472
|
+
message: "sense autostart failed",
|
|
473
|
+
/* v8 ignore next -- defensive: process manager rejects with Error instances in normal use @preserve */
|
|
474
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
475
|
+
});
|
|
476
|
+
});
|
|
477
|
+
}, 0);
|
|
478
|
+
}
|
|
479
|
+
catch (error) {
|
|
480
|
+
(0, runtime_1.emitNervesEvent)({
|
|
481
|
+
level: "error",
|
|
482
|
+
component: "channels",
|
|
483
|
+
event: "channel.daemon_sense_autostart_error",
|
|
484
|
+
message: "sense config refresh failed",
|
|
485
|
+
/* v8 ignore next -- defensive: runtime credential refresh rejects with Error instances in normal use @preserve */
|
|
486
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
487
|
+
});
|
|
488
|
+
}
|
|
489
|
+
finally {
|
|
490
|
+
this.pendingConfigRefreshes.delete(name);
|
|
491
|
+
}
|
|
492
|
+
}
|
|
248
493
|
async startAutoStartSenses() {
|
|
249
494
|
await this.processManager.startAutoStartAgents();
|
|
250
495
|
}
|
|
496
|
+
triggerAutoStartSenses() {
|
|
497
|
+
if (this.processManager.triggerAutoStartAgents) {
|
|
498
|
+
this.processManager.triggerAutoStartAgents();
|
|
499
|
+
return;
|
|
500
|
+
}
|
|
501
|
+
void this.processManager.startAutoStartAgents().catch((error) => {
|
|
502
|
+
(0, runtime_1.emitNervesEvent)({
|
|
503
|
+
level: "error",
|
|
504
|
+
component: "channels",
|
|
505
|
+
event: "channel.daemon_sense_autostart_error",
|
|
506
|
+
message: "sense autostart failed",
|
|
507
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
508
|
+
});
|
|
509
|
+
});
|
|
510
|
+
}
|
|
251
511
|
async stopAll() {
|
|
252
512
|
await this.processManager.stopAll();
|
|
253
513
|
}
|
|
514
|
+
/* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
|
|
515
|
+
listManagedPids() {
|
|
516
|
+
return this.processManager.listAgentSnapshots()
|
|
517
|
+
.map((s) => s.pid)
|
|
518
|
+
.filter((pid) => pid !== null && pid !== undefined);
|
|
519
|
+
}
|
|
520
|
+
/* v8 ignore stop */
|
|
521
|
+
listHealthProbes() {
|
|
522
|
+
const probes = [];
|
|
523
|
+
for (const [agent, context] of this.contexts.entries()) {
|
|
524
|
+
const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
525
|
+
const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
526
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
527
|
+
if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
|
|
528
|
+
continue;
|
|
529
|
+
}
|
|
530
|
+
const machinePayload = machineRuntimeConfig.config;
|
|
531
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
532
|
+
const port = numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT);
|
|
533
|
+
probes.push({
|
|
534
|
+
...(0, http_health_probe_1.createHttpHealthProbe)(`bluebubbles:${agent}`, port),
|
|
535
|
+
managedName: `${agent}:bluebubbles`,
|
|
536
|
+
});
|
|
537
|
+
}
|
|
538
|
+
return probes;
|
|
539
|
+
}
|
|
540
|
+
async restartSense(managedName) {
|
|
541
|
+
if (this.processManager.restartAgent) {
|
|
542
|
+
await this.processManager.restartAgent(managedName);
|
|
543
|
+
return;
|
|
544
|
+
}
|
|
545
|
+
await this.processManager.startAgent?.(managedName);
|
|
546
|
+
}
|
|
254
547
|
listSenseRows() {
|
|
255
548
|
const runtime = new Map();
|
|
256
549
|
for (const snapshot of this.processManager.listAgentSnapshots()) {
|
|
@@ -262,6 +555,7 @@ class DaemonSenseManager {
|
|
|
262
555
|
runtime.set(parsed.agent, current);
|
|
263
556
|
}
|
|
264
557
|
const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
558
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
265
559
|
const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
|
|
266
560
|
const runtimeInfo = {
|
|
267
561
|
cli: { configured: true },
|
|
@@ -271,8 +565,13 @@ class DaemonSenseManager {
|
|
|
271
565
|
},
|
|
272
566
|
bluebubbles: {
|
|
273
567
|
configured: context.facts.bluebubbles.configured,
|
|
568
|
+
optional: context.facts.bluebubbles.optional,
|
|
274
569
|
...blueBubblesRuntimeFacts,
|
|
275
570
|
},
|
|
571
|
+
mail: {
|
|
572
|
+
configured: context.facts.mail.configured,
|
|
573
|
+
...(runtime.get(agent)?.mail ?? {}),
|
|
574
|
+
},
|
|
276
575
|
};
|
|
277
576
|
const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
|
|
278
577
|
return inventory.map((entry) => ({
|
|
@@ -287,6 +586,22 @@ class DaemonSenseManager {
|
|
|
287
586
|
?? context.facts[entry.sense].detail
|
|
288
587
|
: context.facts[entry.sense].detail
|
|
289
588
|
: "not enabled in agent.json",
|
|
589
|
+
...(entry.sense === "bluebubbles" ? {
|
|
590
|
+
proofMethod: blueBubblesRuntimeFacts.proofMethod,
|
|
591
|
+
lastProofAt: blueBubblesRuntimeFacts.lastProofAt,
|
|
592
|
+
proofAgeMs: blueBubblesRuntimeFacts.proofAgeMs,
|
|
593
|
+
lastFailure: blueBubblesRuntimeFacts.lastFailure,
|
|
594
|
+
failureLayer: blueBubblesRuntimeFacts.failureLayer,
|
|
595
|
+
recoveryAction: blueBubblesRuntimeFacts.recoveryAction,
|
|
596
|
+
pendingRecoveryCount: blueBubblesRuntimeFacts.pendingRecoveryCount,
|
|
597
|
+
failedRecoveryCount: blueBubblesRuntimeFacts.failedRecoveryCount,
|
|
598
|
+
oldestPendingRecoveryAt: blueBubblesRuntimeFacts.oldestPendingRecoveryAt,
|
|
599
|
+
oldestPendingRecoveryAgeMs: blueBubblesRuntimeFacts.oldestPendingRecoveryAgeMs,
|
|
600
|
+
activeTurnCount: blueBubblesRuntimeFacts.activeTurnCount,
|
|
601
|
+
stalledTurnCount: blueBubblesRuntimeFacts.stalledTurnCount,
|
|
602
|
+
oldestActiveTurnStartedAt: blueBubblesRuntimeFacts.oldestActiveTurnStartedAt,
|
|
603
|
+
oldestActiveTurnAgeMs: blueBubblesRuntimeFacts.oldestActiveTurnAgeMs,
|
|
604
|
+
} : {}),
|
|
290
605
|
}));
|
|
291
606
|
});
|
|
292
607
|
(0, runtime_1.emitNervesEvent)({
|
|
@@ -0,0 +1,131 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Pluggable session ID resolver for MCP conversations.
|
|
3
|
+
// Tries tool-specific methods first, falls back to UUID.
|
|
4
|
+
//
|
|
5
|
+
// Claude Code: walks parent PID chain -> reads ~/.claude/sessions/{pid}.json
|
|
6
|
+
// Codex: reads thread_id from env (future)
|
|
7
|
+
// Fallback: generates UUID
|
|
8
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
9
|
+
if (k2 === undefined) k2 = k;
|
|
10
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
11
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
12
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
13
|
+
}
|
|
14
|
+
Object.defineProperty(o, k2, desc);
|
|
15
|
+
}) : (function(o, m, k, k2) {
|
|
16
|
+
if (k2 === undefined) k2 = k;
|
|
17
|
+
o[k2] = m[k];
|
|
18
|
+
}));
|
|
19
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
20
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
21
|
+
}) : function(o, v) {
|
|
22
|
+
o["default"] = v;
|
|
23
|
+
});
|
|
24
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
25
|
+
var ownKeys = function(o) {
|
|
26
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
27
|
+
var ar = [];
|
|
28
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
29
|
+
return ar;
|
|
30
|
+
};
|
|
31
|
+
return ownKeys(o);
|
|
32
|
+
};
|
|
33
|
+
return function (mod) {
|
|
34
|
+
if (mod && mod.__esModule) return mod;
|
|
35
|
+
var result = {};
|
|
36
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
37
|
+
__setModuleDefault(result, mod);
|
|
38
|
+
return result;
|
|
39
|
+
};
|
|
40
|
+
})();
|
|
41
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
42
|
+
exports.resolveSessionId = resolveSessionId;
|
|
43
|
+
const fs = __importStar(require("fs"));
|
|
44
|
+
const path = __importStar(require("path"));
|
|
45
|
+
const os = __importStar(require("os"));
|
|
46
|
+
const crypto_1 = require("crypto");
|
|
47
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
48
|
+
const DEFAULT_CLAUDE_SESSIONS_DIR = path.join(os.homedir(), ".claude", "sessions");
|
|
49
|
+
const MAX_PID_WALK_DEPTH = 10;
|
|
50
|
+
/**
|
|
51
|
+
* Try to read a Claude Code session ID from a PID-keyed session file.
|
|
52
|
+
* Returns the sessionId if found, null otherwise.
|
|
53
|
+
*/
|
|
54
|
+
function tryReadClaudeSession(sessionsDir, pid) {
|
|
55
|
+
const sessionFile = path.join(sessionsDir, `${pid}.json`);
|
|
56
|
+
if (!fs.existsSync(sessionFile))
|
|
57
|
+
return null;
|
|
58
|
+
try {
|
|
59
|
+
const raw = fs.readFileSync(sessionFile, "utf-8");
|
|
60
|
+
const data = JSON.parse(raw);
|
|
61
|
+
if (typeof data.sessionId === "string" && data.sessionId.length > 0) {
|
|
62
|
+
return data.sessionId;
|
|
63
|
+
}
|
|
64
|
+
return null;
|
|
65
|
+
}
|
|
66
|
+
catch {
|
|
67
|
+
return null;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Walk the parent PID chain looking for a Claude Code session file.
|
|
72
|
+
* Starts at the current process PID, walks up to parent, grandparent, etc.
|
|
73
|
+
* Returns the session ID from the first matching file, or null.
|
|
74
|
+
*/
|
|
75
|
+
function walkPidChain(sessionsDir) {
|
|
76
|
+
let currentPid = process.pid;
|
|
77
|
+
for (let depth = 0; depth < MAX_PID_WALK_DEPTH; depth++) {
|
|
78
|
+
const sessionId = tryReadClaudeSession(sessionsDir, currentPid);
|
|
79
|
+
if (sessionId) {
|
|
80
|
+
(0, runtime_1.emitNervesEvent)({
|
|
81
|
+
component: "daemon",
|
|
82
|
+
event: "daemon.session_id_pid_walk_hit",
|
|
83
|
+
message: "found Claude session via PID walk",
|
|
84
|
+
meta: { pid: currentPid, depth, sessionId },
|
|
85
|
+
});
|
|
86
|
+
return sessionId;
|
|
87
|
+
}
|
|
88
|
+
// Walk to parent PID
|
|
89
|
+
// On macOS/Linux, process.ppid gives the parent. For deeper ancestry,
|
|
90
|
+
// we'd need /proc/{pid}/stat or ps -o ppid=. For now, we check current + parent.
|
|
91
|
+
if (depth === 0) {
|
|
92
|
+
currentPid = process.ppid;
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
// Cannot walk further without OS-specific process tree APIs
|
|
96
|
+
break;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
return null;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Resolve a session ID for the current MCP connection.
|
|
103
|
+
* Returns a stable identifier that ties MCP tool calls to a conversation session.
|
|
104
|
+
*
|
|
105
|
+
* Resolution order:
|
|
106
|
+
* 1. Claude Code PID walk: check ~/.claude/sessions/{pid}.json for current + parent PID
|
|
107
|
+
* 2. UUID fallback: generate a random UUID
|
|
108
|
+
*/
|
|
109
|
+
function resolveSessionId(options) {
|
|
110
|
+
const sessionsDir = options?.claudeSessionsDir ?? DEFAULT_CLAUDE_SESSIONS_DIR;
|
|
111
|
+
// Try Claude Code PID walk
|
|
112
|
+
const claudeSessionId = walkPidChain(sessionsDir);
|
|
113
|
+
if (claudeSessionId) {
|
|
114
|
+
(0, runtime_1.emitNervesEvent)({
|
|
115
|
+
component: "daemon",
|
|
116
|
+
event: "daemon.session_id_resolved",
|
|
117
|
+
message: "session ID resolved via Claude Code PID walk",
|
|
118
|
+
meta: { sessionId: claudeSessionId, method: "claude-pid-walk" },
|
|
119
|
+
});
|
|
120
|
+
return claudeSessionId;
|
|
121
|
+
}
|
|
122
|
+
// Fallback: UUID
|
|
123
|
+
const sessionId = (0, crypto_1.randomUUID)();
|
|
124
|
+
(0, runtime_1.emitNervesEvent)({
|
|
125
|
+
component: "daemon",
|
|
126
|
+
event: "daemon.session_id_resolved",
|
|
127
|
+
message: "session ID resolved via UUID fallback",
|
|
128
|
+
meta: { sessionId, method: "uuid-fallback" },
|
|
129
|
+
});
|
|
130
|
+
return sessionId;
|
|
131
|
+
}
|