@ouro.bot/cli 0.1.0-alpha.56 → 0.1.0-alpha.561
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 +127 -23
- 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-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 +3604 -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 +322 -0
- package/dist/heart/config.js +114 -118
- package/dist/heart/core.js +913 -246
- package/dist/heart/cross-chat-delivery.js +3 -18
- package/dist/heart/daemon/agent-config-check.js +419 -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 +547 -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 +776 -0
- package/dist/heart/daemon/cli-exec.js +7457 -0
- package/dist/heart/daemon/cli-help.js +498 -0
- package/dist/heart/daemon/cli-parse.js +1592 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +763 -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 -1698
- package/dist/heart/daemon/daemon-entry.js +387 -2
- package/dist/heart/daemon/daemon-health.js +176 -0
- package/dist/heart/daemon/daemon-rollup.js +57 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +88 -13
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +796 -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 +826 -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 +89 -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 +389 -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 +32 -56
- 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 +203 -57
- 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 +267 -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-readiness-cache.js +40 -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 +13 -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 +389 -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-B-461hes.js +61 -0
- package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -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 +39 -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 +1011 -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 +129 -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 +963 -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 +178 -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 +594 -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 +549 -181
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +251 -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/senses/voice/audio-routing.js +119 -0
- package/dist/senses/voice/elevenlabs.js +178 -0
- package/dist/senses/voice/golden-path.js +116 -0
- package/dist/senses/voice/index.js +26 -0
- package/dist/senses/voice/meeting.js +113 -0
- package/dist/senses/voice/playback.js +139 -0
- package/dist/senses/voice/transcript.js +70 -0
- package/dist/senses/voice/turn.js +85 -0
- package/dist/senses/voice/types.js +2 -0
- package/dist/senses/voice/whisper.js +161 -0
- package/dist/senses/voice-entry.js +80 -0
- 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/auth-flow.js +0 -351
- 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,8 @@ 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 },
|
|
58
|
+
voice: { ...identity_1.DEFAULT_AGENT_SENSES.voice },
|
|
54
59
|
};
|
|
55
60
|
}
|
|
56
61
|
function readAgentSenses(agentJsonPath) {
|
|
@@ -76,7 +81,7 @@ function readAgentSenses(agentJsonPath) {
|
|
|
76
81
|
if (!rawSenses || typeof rawSenses !== "object" || Array.isArray(rawSenses)) {
|
|
77
82
|
return defaults;
|
|
78
83
|
}
|
|
79
|
-
for (const sense of ["cli", "teams", "bluebubbles"]) {
|
|
84
|
+
for (const sense of ["cli", "teams", "bluebubbles", "mail", "voice"]) {
|
|
80
85
|
const rawSense = rawSenses[sense];
|
|
81
86
|
if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
|
|
82
87
|
continue;
|
|
@@ -88,22 +93,6 @@ function readAgentSenses(agentJsonPath) {
|
|
|
88
93
|
}
|
|
89
94
|
return defaults;
|
|
90
95
|
}
|
|
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
96
|
function textField(record, key) {
|
|
108
97
|
const value = record?.[key];
|
|
109
98
|
return typeof value === "string" ? value.trim() : "";
|
|
@@ -112,17 +101,39 @@ function numberField(record, key, fallback) {
|
|
|
112
101
|
const value = record?.[key];
|
|
113
102
|
return typeof value === "number" && Number.isFinite(value) ? value : fallback;
|
|
114
103
|
}
|
|
115
|
-
function
|
|
104
|
+
function compactRuntimeConfigError(agent, error) {
|
|
105
|
+
const compact = error.replace(/\s+/g, " ").trim();
|
|
106
|
+
if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
|
|
107
|
+
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`;
|
|
108
|
+
}
|
|
109
|
+
return compact || "unavailable";
|
|
110
|
+
}
|
|
111
|
+
function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
|
|
112
|
+
if (runtimeConfig.ok)
|
|
113
|
+
return "";
|
|
114
|
+
const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
|
|
115
|
+
if (runtimeConfig.reason === "missing")
|
|
116
|
+
return `missing vault ${itemName} (${agent})`;
|
|
117
|
+
return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
|
|
118
|
+
}
|
|
119
|
+
function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
|
|
116
120
|
const base = {
|
|
117
121
|
cli: { configured: true, detail: "local interactive terminal" },
|
|
118
122
|
teams: { configured: false, detail: "not enabled in agent.json" },
|
|
119
123
|
bluebubbles: { configured: false, detail: "not enabled in agent.json" },
|
|
124
|
+
mail: { configured: false, detail: "not enabled in agent.json" },
|
|
125
|
+
voice: { configured: false, detail: "not enabled in agent.json" },
|
|
120
126
|
};
|
|
121
|
-
const
|
|
127
|
+
const payload = runtimeConfig.ok ? runtimeConfig.config : {};
|
|
128
|
+
const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
|
|
122
129
|
const teams = payload.teams;
|
|
123
130
|
const teamsChannel = payload.teamsChannel;
|
|
124
|
-
const
|
|
125
|
-
const
|
|
131
|
+
const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
|
|
132
|
+
const bluebubbles = machinePayload.bluebubbles;
|
|
133
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
134
|
+
const mailroom = payload.mailroom;
|
|
135
|
+
const integrations = payload.integrations;
|
|
136
|
+
const voice = machinePayload.voice;
|
|
126
137
|
if (senses.teams.enabled) {
|
|
127
138
|
const missing = [];
|
|
128
139
|
if (!textField(teams, "clientId"))
|
|
@@ -138,9 +149,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
138
149
|
}
|
|
139
150
|
: {
|
|
140
151
|
configured: false,
|
|
141
|
-
detail:
|
|
142
|
-
? `missing
|
|
143
|
-
:
|
|
152
|
+
detail: runtimeConfig.ok
|
|
153
|
+
? `missing ${missing.join("/")}`
|
|
154
|
+
: unavailableDetail,
|
|
144
155
|
};
|
|
145
156
|
}
|
|
146
157
|
if (senses.bluebubbles.enabled) {
|
|
@@ -156,19 +167,77 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
156
167
|
}
|
|
157
168
|
: {
|
|
158
169
|
configured: false,
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
170
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
171
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
172
|
+
? "not attached on this machine"
|
|
173
|
+
: machineRuntimeConfig.ok
|
|
174
|
+
? `missing ${missing.join("/")}`
|
|
175
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
176
|
+
};
|
|
177
|
+
}
|
|
178
|
+
if (senses.mail.enabled) {
|
|
179
|
+
const privateKeys = mailroom?.privateKeys;
|
|
180
|
+
const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
|
|
181
|
+
const mailboxAddress = textField(mailroom, "mailboxAddress");
|
|
182
|
+
const missing = [];
|
|
183
|
+
if (!mailboxAddress)
|
|
184
|
+
missing.push("mailroom.mailboxAddress");
|
|
185
|
+
if (!hasPrivateKeys)
|
|
186
|
+
missing.push("mailroom.privateKeys");
|
|
187
|
+
base.mail = missing.length === 0
|
|
188
|
+
? { configured: true, detail: mailboxAddress }
|
|
189
|
+
: {
|
|
190
|
+
configured: false,
|
|
191
|
+
detail: runtimeConfig.ok
|
|
192
|
+
? `missing ${missing.join("/")}`
|
|
193
|
+
: unavailableDetail,
|
|
194
|
+
};
|
|
195
|
+
}
|
|
196
|
+
if (senses.voice.enabled) {
|
|
197
|
+
const missing = [];
|
|
198
|
+
if (!textField(integrations, "elevenLabsApiKey"))
|
|
199
|
+
missing.push("integrations.elevenLabsApiKey");
|
|
200
|
+
if (!textField(voice, "whisperCliPath"))
|
|
201
|
+
missing.push("voice.whisperCliPath");
|
|
202
|
+
if (!textField(voice, "whisperModelPath"))
|
|
203
|
+
missing.push("voice.whisperModelPath");
|
|
204
|
+
base.voice = missing.length === 0
|
|
205
|
+
? { configured: true, detail: "local Whisper.cpp STT + ElevenLabs TTS" }
|
|
206
|
+
: {
|
|
207
|
+
configured: false,
|
|
208
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
209
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
210
|
+
? "not attached on this machine"
|
|
211
|
+
: runtimeConfig.ok && machineRuntimeConfig.ok
|
|
212
|
+
? `missing ${missing.join("/")}`
|
|
213
|
+
: !runtimeConfig.ok
|
|
214
|
+
? unavailableDetail
|
|
215
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
162
216
|
};
|
|
163
217
|
}
|
|
164
218
|
return base;
|
|
165
219
|
}
|
|
220
|
+
function senseRepairHint(agent, sense) {
|
|
221
|
+
if (sense === "teams") {
|
|
222
|
+
return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
|
|
223
|
+
}
|
|
224
|
+
if (sense === "mail") {
|
|
225
|
+
return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
|
|
226
|
+
}
|
|
227
|
+
if (sense === "voice") {
|
|
228
|
+
return `Agent-runnable: run 'ouro connect voice --agent ${agent}' for config guidance, save ElevenLabs and local Whisper.cpp settings, then run 'ouro up' again.`;
|
|
229
|
+
}
|
|
230
|
+
return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
|
|
231
|
+
}
|
|
232
|
+
function currentMachineId() {
|
|
233
|
+
return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
|
|
234
|
+
}
|
|
166
235
|
function parseSenseSnapshotName(name) {
|
|
167
236
|
const parts = name.split(":");
|
|
168
237
|
if (parts.length !== 2)
|
|
169
238
|
return null;
|
|
170
239
|
const [agent, sense] = parts;
|
|
171
|
-
if (sense !== "teams" && sense !== "bluebubbles")
|
|
240
|
+
if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail" && sense !== "voice")
|
|
172
241
|
return null;
|
|
173
242
|
return { agent, sense };
|
|
174
243
|
}
|
|
@@ -177,6 +246,34 @@ function runtimeInfoFor(status) {
|
|
|
177
246
|
return { runtime: "running" };
|
|
178
247
|
return { runtime: "error" };
|
|
179
248
|
}
|
|
249
|
+
function managedSenseEntry(sense) {
|
|
250
|
+
if (sense === "teams")
|
|
251
|
+
return "senses/teams-entry.js";
|
|
252
|
+
if (sense === "bluebubbles")
|
|
253
|
+
return "senses/bluebubbles/entry.js";
|
|
254
|
+
if (sense === "voice")
|
|
255
|
+
return "senses/voice-entry.js";
|
|
256
|
+
return "senses/mail-entry.js";
|
|
257
|
+
}
|
|
258
|
+
function runtimeCredentialBootstrapFor(agent, sense) {
|
|
259
|
+
const runtime = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
260
|
+
const machineId = sense === "bluebubbles" || sense === "voice" ? currentMachineId() : undefined;
|
|
261
|
+
const machine = sense === "bluebubbles" || sense === "voice" ? (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent) : null;
|
|
262
|
+
const providerPool = (0, provider_credentials_1.readProviderCredentialPool)(agent);
|
|
263
|
+
const providerCredentialRecords = providerPool.ok
|
|
264
|
+
? Object.values(providerPool.pool.providers).filter((record) => !!record)
|
|
265
|
+
: [];
|
|
266
|
+
const bootstrap = {
|
|
267
|
+
agentName: agent,
|
|
268
|
+
runtimeConfig: runtime.ok ? runtime.config : undefined,
|
|
269
|
+
machineRuntimeConfig: machine?.ok ? machine.config : undefined,
|
|
270
|
+
machineId,
|
|
271
|
+
providerCredentialRecords: providerCredentialRecords.length > 0 ? providerCredentialRecords : undefined,
|
|
272
|
+
};
|
|
273
|
+
if (!bootstrap.runtimeConfig && !bootstrap.machineRuntimeConfig && !bootstrap.providerCredentialRecords)
|
|
274
|
+
return null;
|
|
275
|
+
return bootstrap;
|
|
276
|
+
}
|
|
180
277
|
function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
181
278
|
if (!lastCheckedAt) {
|
|
182
279
|
return false;
|
|
@@ -187,53 +284,186 @@ function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
|
187
284
|
}
|
|
188
285
|
return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
|
|
189
286
|
}
|
|
287
|
+
function readBlueBubblesRuntimeJson(runtimePath) {
|
|
288
|
+
try {
|
|
289
|
+
const raw = fs.readFileSync(runtimePath, "utf-8");
|
|
290
|
+
const parsed = JSON.parse(raw);
|
|
291
|
+
/* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
|
|
292
|
+
return {
|
|
293
|
+
upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
|
|
294
|
+
? parsed.upstreamStatus
|
|
295
|
+
: "unknown",
|
|
296
|
+
detail: typeof parsed.detail === "string" && parsed.detail.trim()
|
|
297
|
+
? parsed.detail
|
|
298
|
+
: "startup health probe pending",
|
|
299
|
+
lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
|
|
300
|
+
proofMethod: typeof parsed.proofMethod === "string" && parsed.proofMethod.trim()
|
|
301
|
+
? parsed.proofMethod
|
|
302
|
+
: undefined,
|
|
303
|
+
pendingRecoveryCount: typeof parsed.pendingRecoveryCount === "number" && Number.isFinite(parsed.pendingRecoveryCount)
|
|
304
|
+
? parsed.pendingRecoveryCount
|
|
305
|
+
: 0,
|
|
306
|
+
failedRecoveryCount: typeof parsed.failedRecoveryCount === "number" && Number.isFinite(parsed.failedRecoveryCount)
|
|
307
|
+
? parsed.failedRecoveryCount
|
|
308
|
+
: 0,
|
|
309
|
+
oldestPendingRecoveryAt: typeof parsed.oldestPendingRecoveryAt === "string" ? parsed.oldestPendingRecoveryAt : undefined,
|
|
310
|
+
oldestPendingRecoveryAgeMs: typeof parsed.oldestPendingRecoveryAgeMs === "number" && Number.isFinite(parsed.oldestPendingRecoveryAgeMs)
|
|
311
|
+
? parsed.oldestPendingRecoveryAgeMs
|
|
312
|
+
: undefined,
|
|
313
|
+
activeTurnCount: typeof parsed.activeTurnCount === "number" && Number.isFinite(parsed.activeTurnCount)
|
|
314
|
+
? parsed.activeTurnCount
|
|
315
|
+
: undefined,
|
|
316
|
+
stalledTurnCount: typeof parsed.stalledTurnCount === "number" && Number.isFinite(parsed.stalledTurnCount)
|
|
317
|
+
? parsed.stalledTurnCount
|
|
318
|
+
: undefined,
|
|
319
|
+
oldestActiveTurnStartedAt: typeof parsed.oldestActiveTurnStartedAt === "string" ? parsed.oldestActiveTurnStartedAt : undefined,
|
|
320
|
+
oldestActiveTurnAgeMs: typeof parsed.oldestActiveTurnAgeMs === "number" && Number.isFinite(parsed.oldestActiveTurnAgeMs)
|
|
321
|
+
? parsed.oldestActiveTurnAgeMs
|
|
322
|
+
: undefined,
|
|
323
|
+
};
|
|
324
|
+
/* v8 ignore stop */
|
|
325
|
+
/* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
|
|
326
|
+
}
|
|
327
|
+
catch {
|
|
328
|
+
return { upstreamStatus: "unknown", detail: "startup health probe pending", pendingRecoveryCount: 0, failedRecoveryCount: 0 };
|
|
329
|
+
}
|
|
330
|
+
/* v8 ignore stop */
|
|
331
|
+
}
|
|
190
332
|
function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
|
|
191
333
|
const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
192
334
|
const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
|
|
193
335
|
if (!fs.existsSync(runtimePath)) {
|
|
194
336
|
return { runtime: snapshot?.runtime };
|
|
195
337
|
}
|
|
196
|
-
const state = (
|
|
338
|
+
const state = readBlueBubblesRuntimeJson(runtimePath);
|
|
339
|
+
const checkedAtMs = state.lastCheckedAt ? Date.parse(state.lastCheckedAt) : Number.NaN;
|
|
340
|
+
const proofFacts = {
|
|
341
|
+
proofMethod: state.proofMethod ?? "bluebubbles.checkHealth",
|
|
342
|
+
lastProofAt: state.lastCheckedAt,
|
|
343
|
+
proofAgeMs: Number.isFinite(checkedAtMs) ? Math.max(0, Date.now() - checkedAtMs) : undefined,
|
|
344
|
+
pendingRecoveryCount: state.pendingRecoveryCount,
|
|
345
|
+
failedRecoveryCount: state.failedRecoveryCount,
|
|
346
|
+
oldestPendingRecoveryAt: state.oldestPendingRecoveryAt,
|
|
347
|
+
oldestPendingRecoveryAgeMs: state.oldestPendingRecoveryAgeMs,
|
|
348
|
+
activeTurnCount: state.activeTurnCount,
|
|
349
|
+
stalledTurnCount: state.stalledTurnCount,
|
|
350
|
+
oldestActiveTurnStartedAt: state.oldestActiveTurnStartedAt,
|
|
351
|
+
oldestActiveTurnAgeMs: state.oldestActiveTurnAgeMs,
|
|
352
|
+
};
|
|
197
353
|
if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
|
|
198
|
-
return {
|
|
354
|
+
return {
|
|
355
|
+
runtime: snapshot?.runtime,
|
|
356
|
+
lastFailure: state.lastCheckedAt ? "BlueBubbles proof is stale" : undefined,
|
|
357
|
+
failureLayer: state.lastCheckedAt ? "proof_freshness" : undefined,
|
|
358
|
+
};
|
|
359
|
+
}
|
|
360
|
+
if (snapshot?.runtime !== "running") {
|
|
361
|
+
return {
|
|
362
|
+
runtime: "error",
|
|
363
|
+
detail: "BlueBubbles listener is not running",
|
|
364
|
+
...proofFacts,
|
|
365
|
+
lastFailure: "listener process is not running",
|
|
366
|
+
failureLayer: "listener",
|
|
367
|
+
recoveryAction: "daemon health monitor will restart the BlueBubbles listener when its probe fails",
|
|
368
|
+
};
|
|
199
369
|
}
|
|
200
370
|
if (state.upstreamStatus === "error") {
|
|
201
371
|
return {
|
|
202
372
|
runtime: "error",
|
|
203
373
|
detail: state.detail,
|
|
374
|
+
...proofFacts,
|
|
375
|
+
lastFailure: state.detail,
|
|
376
|
+
failureLayer: "upstream",
|
|
377
|
+
recoveryAction: "verify BlueBubbles server/app auth and local machine attachment, then retry the listener",
|
|
378
|
+
};
|
|
379
|
+
}
|
|
380
|
+
if (state.pendingRecoveryCount > 0) {
|
|
381
|
+
return {
|
|
382
|
+
runtime: "error",
|
|
383
|
+
detail: state.detail,
|
|
384
|
+
...proofFacts,
|
|
385
|
+
lastFailure: state.detail,
|
|
386
|
+
failureLayer: "recovery_queue",
|
|
387
|
+
recoveryAction: "queued recovery will retry; inspect BlueBubbles inbound/recovery sidecar logs if age keeps growing",
|
|
388
|
+
};
|
|
389
|
+
}
|
|
390
|
+
if ((state.stalledTurnCount ?? 0) > 0) {
|
|
391
|
+
return {
|
|
392
|
+
runtime: "error",
|
|
393
|
+
detail: state.detail,
|
|
394
|
+
...proofFacts,
|
|
395
|
+
lastFailure: state.detail,
|
|
396
|
+
failureLayer: "live_turn_stall",
|
|
397
|
+
recoveryAction: "live iMessage turn timeout/watchdog will release the lane and recovery will retry captured messages",
|
|
204
398
|
};
|
|
205
399
|
}
|
|
206
400
|
if (state.upstreamStatus === "ok") {
|
|
207
|
-
return {
|
|
401
|
+
return {
|
|
402
|
+
runtime: "running",
|
|
403
|
+
...proofFacts,
|
|
404
|
+
...(state.failedRecoveryCount > 0 ? { detail: state.detail } : {}),
|
|
405
|
+
...(state.failedRecoveryCount > 0 ? {
|
|
406
|
+
lastFailure: state.detail,
|
|
407
|
+
failureLayer: "recovery_quarantine",
|
|
408
|
+
recoveryAction: "inspect quarantined BlueBubbles recovery failures; live transport remains reachable",
|
|
409
|
+
} : {}),
|
|
410
|
+
};
|
|
208
411
|
}
|
|
209
|
-
return { runtime: snapshot?.runtime };
|
|
412
|
+
return { runtime: snapshot?.runtime, ...proofFacts };
|
|
210
413
|
}
|
|
211
414
|
class DaemonSenseManager {
|
|
212
415
|
processManager;
|
|
213
416
|
contexts;
|
|
417
|
+
pendingConfigRefreshes = new Set();
|
|
214
418
|
bundlesRoot;
|
|
215
419
|
constructor(options) {
|
|
216
420
|
const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
217
|
-
const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
|
|
218
421
|
this.bundlesRoot = bundlesRoot;
|
|
219
422
|
this.contexts = new Map(options.agents.map((agent) => {
|
|
220
423
|
const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
|
|
221
|
-
const facts =
|
|
424
|
+
const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
222
425
|
return [agent, { senses, facts }];
|
|
223
426
|
}));
|
|
224
427
|
const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
225
|
-
return ["teams", "bluebubbles"]
|
|
226
|
-
.filter((sense) => context.senses[sense].enabled
|
|
428
|
+
return ["teams", "bluebubbles", "mail", "voice"]
|
|
429
|
+
.filter((sense) => context.senses[sense].enabled)
|
|
227
430
|
.map((sense) => ({
|
|
228
431
|
name: `${agent}:${sense}`,
|
|
229
432
|
agentArg: agent,
|
|
230
|
-
entry: sense
|
|
433
|
+
entry: managedSenseEntry(sense),
|
|
231
434
|
channel: sense,
|
|
232
435
|
autoStart: true,
|
|
436
|
+
getRuntimeCredentialBootstrap: () => runtimeCredentialBootstrapFor(agent, sense),
|
|
233
437
|
}));
|
|
234
438
|
});
|
|
235
439
|
this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
|
|
236
440
|
agents: managedSenseAgents,
|
|
441
|
+
configCheck: async (name) => {
|
|
442
|
+
const parsed = parseSenseSnapshotName(name);
|
|
443
|
+
if (!parsed)
|
|
444
|
+
return { ok: true };
|
|
445
|
+
const context = this.contexts.get(parsed.agent);
|
|
446
|
+
if (!context)
|
|
447
|
+
return { ok: true };
|
|
448
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(parsed.agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent));
|
|
449
|
+
const fact = context.facts[parsed.sense];
|
|
450
|
+
if (fact.configured)
|
|
451
|
+
return { ok: true };
|
|
452
|
+
this.scheduleSenseConfigRefresh(name, parsed);
|
|
453
|
+
if (fact.optional) {
|
|
454
|
+
return {
|
|
455
|
+
ok: false,
|
|
456
|
+
skip: true,
|
|
457
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
|
|
458
|
+
};
|
|
459
|
+
}
|
|
460
|
+
return {
|
|
461
|
+
ok: false,
|
|
462
|
+
skip: true,
|
|
463
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
|
|
464
|
+
fix: senseRepairHint(parsed.agent, parsed.sense),
|
|
465
|
+
};
|
|
466
|
+
},
|
|
237
467
|
});
|
|
238
468
|
(0, runtime_1.emitNervesEvent)({
|
|
239
469
|
component: "channels",
|
|
@@ -245,12 +475,106 @@ class DaemonSenseManager {
|
|
|
245
475
|
},
|
|
246
476
|
});
|
|
247
477
|
}
|
|
478
|
+
scheduleSenseConfigRefresh(name, parsed) {
|
|
479
|
+
if (this.pendingConfigRefreshes.has(name))
|
|
480
|
+
return;
|
|
481
|
+
this.pendingConfigRefreshes.add(name);
|
|
482
|
+
void this.refreshSenseConfigAndRetry(name, parsed);
|
|
483
|
+
}
|
|
484
|
+
async refreshSenseConfigAndRetry(name, parsed) {
|
|
485
|
+
try {
|
|
486
|
+
const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
|
|
487
|
+
const machineRefreshed = parsed.sense === "bluebubbles" || parsed.sense === "voice"
|
|
488
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
489
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
|
|
490
|
+
const context = this.contexts.get(parsed.agent);
|
|
491
|
+
/* v8 ignore next -- defensive: config refreshes are only scheduled for known agent contexts @preserve */
|
|
492
|
+
if (!context)
|
|
493
|
+
return;
|
|
494
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
|
|
495
|
+
if (!context.facts[parsed.sense].configured)
|
|
496
|
+
return;
|
|
497
|
+
setTimeout(() => {
|
|
498
|
+
void this.processManager.startAgent?.(name).catch((error) => {
|
|
499
|
+
(0, runtime_1.emitNervesEvent)({
|
|
500
|
+
level: "error",
|
|
501
|
+
component: "channels",
|
|
502
|
+
event: "channel.daemon_sense_autostart_error",
|
|
503
|
+
message: "sense autostart failed",
|
|
504
|
+
/* v8 ignore next -- defensive: process manager rejects with Error instances in normal use @preserve */
|
|
505
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
506
|
+
});
|
|
507
|
+
});
|
|
508
|
+
}, 0);
|
|
509
|
+
}
|
|
510
|
+
catch (error) {
|
|
511
|
+
(0, runtime_1.emitNervesEvent)({
|
|
512
|
+
level: "error",
|
|
513
|
+
component: "channels",
|
|
514
|
+
event: "channel.daemon_sense_autostart_error",
|
|
515
|
+
message: "sense config refresh failed",
|
|
516
|
+
/* v8 ignore next -- defensive: runtime credential refresh rejects with Error instances in normal use @preserve */
|
|
517
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
518
|
+
});
|
|
519
|
+
}
|
|
520
|
+
finally {
|
|
521
|
+
this.pendingConfigRefreshes.delete(name);
|
|
522
|
+
}
|
|
523
|
+
}
|
|
248
524
|
async startAutoStartSenses() {
|
|
249
525
|
await this.processManager.startAutoStartAgents();
|
|
250
526
|
}
|
|
527
|
+
triggerAutoStartSenses() {
|
|
528
|
+
if (this.processManager.triggerAutoStartAgents) {
|
|
529
|
+
this.processManager.triggerAutoStartAgents();
|
|
530
|
+
return;
|
|
531
|
+
}
|
|
532
|
+
void this.processManager.startAutoStartAgents().catch((error) => {
|
|
533
|
+
(0, runtime_1.emitNervesEvent)({
|
|
534
|
+
level: "error",
|
|
535
|
+
component: "channels",
|
|
536
|
+
event: "channel.daemon_sense_autostart_error",
|
|
537
|
+
message: "sense autostart failed",
|
|
538
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
539
|
+
});
|
|
540
|
+
});
|
|
541
|
+
}
|
|
251
542
|
async stopAll() {
|
|
252
543
|
await this.processManager.stopAll();
|
|
253
544
|
}
|
|
545
|
+
/* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
|
|
546
|
+
listManagedPids() {
|
|
547
|
+
return this.processManager.listAgentSnapshots()
|
|
548
|
+
.map((s) => s.pid)
|
|
549
|
+
.filter((pid) => pid !== null && pid !== undefined);
|
|
550
|
+
}
|
|
551
|
+
/* v8 ignore stop */
|
|
552
|
+
listHealthProbes() {
|
|
553
|
+
const probes = [];
|
|
554
|
+
for (const [agent, context] of this.contexts.entries()) {
|
|
555
|
+
const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
556
|
+
const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
557
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
558
|
+
if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
|
|
559
|
+
continue;
|
|
560
|
+
}
|
|
561
|
+
const machinePayload = machineRuntimeConfig.config;
|
|
562
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
563
|
+
const port = numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT);
|
|
564
|
+
probes.push({
|
|
565
|
+
...(0, http_health_probe_1.createHttpHealthProbe)(`bluebubbles:${agent}`, port),
|
|
566
|
+
managedName: `${agent}:bluebubbles`,
|
|
567
|
+
});
|
|
568
|
+
}
|
|
569
|
+
return probes;
|
|
570
|
+
}
|
|
571
|
+
async restartSense(managedName) {
|
|
572
|
+
if (this.processManager.restartAgent) {
|
|
573
|
+
await this.processManager.restartAgent(managedName);
|
|
574
|
+
return;
|
|
575
|
+
}
|
|
576
|
+
await this.processManager.startAgent?.(managedName);
|
|
577
|
+
}
|
|
254
578
|
listSenseRows() {
|
|
255
579
|
const runtime = new Map();
|
|
256
580
|
for (const snapshot of this.processManager.listAgentSnapshots()) {
|
|
@@ -262,6 +586,7 @@ class DaemonSenseManager {
|
|
|
262
586
|
runtime.set(parsed.agent, current);
|
|
263
587
|
}
|
|
264
588
|
const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
589
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
265
590
|
const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
|
|
266
591
|
const runtimeInfo = {
|
|
267
592
|
cli: { configured: true },
|
|
@@ -271,8 +596,18 @@ class DaemonSenseManager {
|
|
|
271
596
|
},
|
|
272
597
|
bluebubbles: {
|
|
273
598
|
configured: context.facts.bluebubbles.configured,
|
|
599
|
+
optional: context.facts.bluebubbles.optional,
|
|
274
600
|
...blueBubblesRuntimeFacts,
|
|
275
601
|
},
|
|
602
|
+
mail: {
|
|
603
|
+
configured: context.facts.mail.configured,
|
|
604
|
+
...(runtime.get(agent)?.mail ?? {}),
|
|
605
|
+
},
|
|
606
|
+
voice: {
|
|
607
|
+
configured: context.facts.voice.configured,
|
|
608
|
+
optional: context.facts.voice.optional,
|
|
609
|
+
...(runtime.get(agent)?.voice ?? {}),
|
|
610
|
+
},
|
|
276
611
|
};
|
|
277
612
|
const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
|
|
278
613
|
return inventory.map((entry) => ({
|
|
@@ -287,6 +622,22 @@ class DaemonSenseManager {
|
|
|
287
622
|
?? context.facts[entry.sense].detail
|
|
288
623
|
: context.facts[entry.sense].detail
|
|
289
624
|
: "not enabled in agent.json",
|
|
625
|
+
...(entry.sense === "bluebubbles" ? {
|
|
626
|
+
proofMethod: blueBubblesRuntimeFacts.proofMethod,
|
|
627
|
+
lastProofAt: blueBubblesRuntimeFacts.lastProofAt,
|
|
628
|
+
proofAgeMs: blueBubblesRuntimeFacts.proofAgeMs,
|
|
629
|
+
lastFailure: blueBubblesRuntimeFacts.lastFailure,
|
|
630
|
+
failureLayer: blueBubblesRuntimeFacts.failureLayer,
|
|
631
|
+
recoveryAction: blueBubblesRuntimeFacts.recoveryAction,
|
|
632
|
+
pendingRecoveryCount: blueBubblesRuntimeFacts.pendingRecoveryCount,
|
|
633
|
+
failedRecoveryCount: blueBubblesRuntimeFacts.failedRecoveryCount,
|
|
634
|
+
oldestPendingRecoveryAt: blueBubblesRuntimeFacts.oldestPendingRecoveryAt,
|
|
635
|
+
oldestPendingRecoveryAgeMs: blueBubblesRuntimeFacts.oldestPendingRecoveryAgeMs,
|
|
636
|
+
activeTurnCount: blueBubblesRuntimeFacts.activeTurnCount,
|
|
637
|
+
stalledTurnCount: blueBubblesRuntimeFacts.stalledTurnCount,
|
|
638
|
+
oldestActiveTurnStartedAt: blueBubblesRuntimeFacts.oldestActiveTurnStartedAt,
|
|
639
|
+
oldestActiveTurnAgeMs: blueBubblesRuntimeFacts.oldestActiveTurnAgeMs,
|
|
640
|
+
} : {}),
|
|
290
641
|
}));
|
|
291
642
|
});
|
|
292
643
|
(0, runtime_1.emitNervesEvent)({
|