@ouro.bot/cli 0.1.0-alpha.60 → 0.1.0-alpha.601
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 +3863 -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 +254 -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 +69 -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/awaiting/await-alert.js +146 -0
- package/dist/heart/awaiting/await-expiry.js +108 -0
- package/dist/heart/awaiting/await-loader.js +91 -0
- package/dist/heart/awaiting/await-parser.js +141 -0
- package/dist/heart/awaiting/await-runtime-state.js +97 -0
- package/dist/heart/awaiting/await-scheduler.js +377 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +142 -0
- package/dist/heart/config-registry.js +322 -0
- package/dist/heart/config.js +114 -119
- package/dist/heart/core.js +909 -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 +7571 -0
- package/dist/heart/daemon/cli-help.js +498 -0
- package/dist/heart/daemon/cli-parse.js +1599 -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 -1703
- package/dist/heart/daemon/daemon-entry.js +485 -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 +842 -69
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +873 -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 +78 -9
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- 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 +10 -2
- package/dist/heart/daemon/runtime-metadata.js +2 -30
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +462 -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 +166 -55
- 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 +375 -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 +656 -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 +23 -11
- 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 +143 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +421 -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 +712 -0
- package/dist/mailroom/body-cache.js +61 -0
- package/dist/mailroom/core.js +788 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +460 -0
- package/dist/mailroom/mbox-import.js +393 -0
- package/dist/mailroom/migration.js +164 -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 +268 -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 +1050 -135
- 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/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 +997 -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-awaiting.js +360 -0
- package/dist/repertoire/tools-base.js +53 -1082
- 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 +1916 -0
- package/dist/repertoire/tools-notes.js +421 -0
- package/dist/repertoire/tools-obligations.js +142 -0
- package/dist/repertoire/tools-runtime.js +61 -0
- package/dist/repertoire/tools-session.js +809 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +345 -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-voice.js +144 -0
- package/dist/repertoire/tools.js +115 -103
- 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/await-turn-message.js +58 -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 +2548 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -71
- 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/bluebubbles-meta-guard.js +40 -0
- 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 +100 -0
- package/dist/senses/cli.js +516 -204
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +185 -21
- package/dist/senses/inner-dialog.js +372 -27
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +654 -181
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +392 -0
- package/dist/senses/surface-tool.js +70 -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-playback.js +237 -0
- package/dist/senses/voice/audio-routing.js +119 -0
- package/dist/senses/voice/elevenlabs.js +202 -0
- package/dist/senses/voice/floor-control.js +431 -0
- package/dist/senses/voice/floor-controller.js +115 -0
- package/dist/senses/voice/golden-path.js +116 -0
- package/dist/senses/voice/index.js +29 -0
- package/dist/senses/voice/meeting.js +113 -0
- package/dist/senses/voice/outbound.js +190 -0
- package/dist/senses/voice/phone.js +33 -0
- package/dist/senses/voice/playback.js +139 -0
- package/dist/senses/voice/realtime-eval.js +496 -0
- package/dist/senses/voice/realtime-trace.js +531 -0
- package/dist/senses/voice/transcript.js +70 -0
- package/dist/senses/voice/turn.js +191 -0
- package/dist/senses/voice/twilio-phone-runtime.js +807 -0
- package/dist/senses/voice/twilio-phone.js +5077 -0
- package/dist/senses/voice/types.js +2 -0
- package/dist/senses/voice/whisper.js +161 -0
- package/dist/senses/voice-entry.js +81 -0
- package/dist/senses/voice-realtime-eval-command.js +99 -0
- package/dist/senses/voice-realtime-eval-entry.js +21 -0
- package/dist/senses/voice-twilio-entry.js +87 -0
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +265 -0
- package/package.json +41 -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 +99 -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/safe-workspace.js +0 -228
- 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,41 +93,50 @@ 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() : "";
|
|
110
99
|
}
|
|
100
|
+
function booleanField(record, key) {
|
|
101
|
+
return record?.[key] === true;
|
|
102
|
+
}
|
|
111
103
|
function numberField(record, key, fallback) {
|
|
112
104
|
const value = record?.[key];
|
|
113
105
|
return typeof value === "number" && Number.isFinite(value) ? value : fallback;
|
|
114
106
|
}
|
|
115
|
-
function
|
|
107
|
+
function compactRuntimeConfigError(agent, error) {
|
|
108
|
+
const compact = error.replace(/\s+/g, " ").trim();
|
|
109
|
+
if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
|
|
110
|
+
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`;
|
|
111
|
+
}
|
|
112
|
+
return compact || "unavailable";
|
|
113
|
+
}
|
|
114
|
+
function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
|
|
115
|
+
if (runtimeConfig.ok)
|
|
116
|
+
return "";
|
|
117
|
+
const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
|
|
118
|
+
if (runtimeConfig.reason === "missing")
|
|
119
|
+
return `missing vault ${itemName} (${agent})`;
|
|
120
|
+
return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
|
|
121
|
+
}
|
|
122
|
+
function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
|
|
116
123
|
const base = {
|
|
117
124
|
cli: { configured: true, detail: "local interactive terminal" },
|
|
118
125
|
teams: { configured: false, detail: "not enabled in agent.json" },
|
|
119
126
|
bluebubbles: { configured: false, detail: "not enabled in agent.json" },
|
|
127
|
+
mail: { configured: false, detail: "not enabled in agent.json" },
|
|
128
|
+
voice: { configured: false, detail: "not enabled in agent.json" },
|
|
120
129
|
};
|
|
121
|
-
const
|
|
130
|
+
const payload = runtimeConfig.ok ? runtimeConfig.config : {};
|
|
131
|
+
const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
|
|
122
132
|
const teams = payload.teams;
|
|
123
133
|
const teamsChannel = payload.teamsChannel;
|
|
124
|
-
const
|
|
125
|
-
const
|
|
134
|
+
const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
|
|
135
|
+
const bluebubbles = machinePayload.bluebubbles;
|
|
136
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
137
|
+
const mailroom = payload.mailroom;
|
|
138
|
+
const integrations = payload.integrations;
|
|
139
|
+
const voice = machinePayload.voice;
|
|
126
140
|
if (senses.teams.enabled) {
|
|
127
141
|
const missing = [];
|
|
128
142
|
if (!textField(teams, "clientId"))
|
|
@@ -138,9 +152,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
138
152
|
}
|
|
139
153
|
: {
|
|
140
154
|
configured: false,
|
|
141
|
-
detail:
|
|
142
|
-
? `missing
|
|
143
|
-
:
|
|
155
|
+
detail: runtimeConfig.ok
|
|
156
|
+
? `missing ${missing.join("/")}`
|
|
157
|
+
: unavailableDetail,
|
|
144
158
|
};
|
|
145
159
|
}
|
|
146
160
|
if (senses.bluebubbles.enabled) {
|
|
@@ -156,19 +170,128 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
156
170
|
}
|
|
157
171
|
: {
|
|
158
172
|
configured: false,
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
173
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
174
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
175
|
+
? "not attached on this machine"
|
|
176
|
+
: machineRuntimeConfig.ok
|
|
177
|
+
? `missing ${missing.join("/")}`
|
|
178
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
if (senses.mail.enabled) {
|
|
182
|
+
const privateKeys = mailroom?.privateKeys;
|
|
183
|
+
const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
|
|
184
|
+
const mailboxAddress = textField(mailroom, "mailboxAddress");
|
|
185
|
+
const missing = [];
|
|
186
|
+
if (!mailboxAddress)
|
|
187
|
+
missing.push("mailroom.mailboxAddress");
|
|
188
|
+
if (!hasPrivateKeys)
|
|
189
|
+
missing.push("mailroom.privateKeys");
|
|
190
|
+
base.mail = missing.length === 0
|
|
191
|
+
? { configured: true, detail: mailboxAddress }
|
|
192
|
+
: {
|
|
193
|
+
configured: false,
|
|
194
|
+
detail: runtimeConfig.ok
|
|
195
|
+
? `missing ${missing.join("/")}`
|
|
196
|
+
: unavailableDetail,
|
|
197
|
+
};
|
|
198
|
+
}
|
|
199
|
+
if (senses.voice.enabled) {
|
|
200
|
+
const portableVoice = payload.voice;
|
|
201
|
+
const conversationEngine = (textField(voice, "twilioConversationEngine")
|
|
202
|
+
|| textField(voice, "conversationEngine")
|
|
203
|
+
|| textField(portableVoice, "twilioConversationEngine")
|
|
204
|
+
|| textField(portableVoice, "conversationEngine")
|
|
205
|
+
|| "cascade").toLowerCase();
|
|
206
|
+
const twilioTransportMode = (textField(voice, "twilioTransportMode") || "record-play").toLowerCase();
|
|
207
|
+
const twilioPublicUrl = textField(voice, "twilioPublicUrl");
|
|
208
|
+
const hasOpenAIRealtimeKey = !!(textField(portableVoice, "openaiRealtimeApiKey")
|
|
209
|
+
|| textField(integrations, "openaiApiKey")
|
|
210
|
+
|| textField(integrations, "openaiEmbeddingsApiKey"));
|
|
211
|
+
const missing = [];
|
|
212
|
+
/* v8 ignore start -- voice setup missing-field matrix is enforced by the voice runtime resolver; sense-manager only renders human-readable readiness facts @preserve */
|
|
213
|
+
if (conversationEngine === "openai-realtime" || conversationEngine === "openai-sip") {
|
|
214
|
+
if (conversationEngine === "openai-realtime" && twilioTransportMode !== "media-stream") {
|
|
215
|
+
missing.push("voice.twilioTransportMode=media-stream");
|
|
216
|
+
}
|
|
217
|
+
if (!hasOpenAIRealtimeKey) {
|
|
218
|
+
missing.push("voice.openaiRealtimeApiKey");
|
|
219
|
+
}
|
|
220
|
+
if (conversationEngine === "openai-sip") {
|
|
221
|
+
if (!textField(portableVoice, "openaiSipProjectId") && !textField(voice, "openaiSipProjectId")) {
|
|
222
|
+
missing.push("voice.openaiSipProjectId");
|
|
223
|
+
}
|
|
224
|
+
const allowUnsignedWebhooks = booleanField(portableVoice, "openaiSipAllowUnsignedWebhooks")
|
|
225
|
+
|| booleanField(voice, "openaiSipAllowUnsignedWebhooks");
|
|
226
|
+
if (!allowUnsignedWebhooks && !textField(portableVoice, "openaiSipWebhookSecret") && !textField(voice, "openaiSipWebhookSecret")) {
|
|
227
|
+
missing.push("voice.openaiSipWebhookSecret");
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
else {
|
|
232
|
+
if (!textField(integrations, "elevenLabsApiKey"))
|
|
233
|
+
missing.push("integrations.elevenLabsApiKey");
|
|
234
|
+
if (!textField(integrations, "elevenLabsVoiceId") && !textField(portableVoice, "elevenLabsVoiceId")) {
|
|
235
|
+
missing.push("integrations.elevenLabsVoiceId");
|
|
236
|
+
}
|
|
237
|
+
if (!textField(voice, "whisperCliPath"))
|
|
238
|
+
missing.push("voice.whisperCliPath");
|
|
239
|
+
if (!textField(voice, "whisperModelPath"))
|
|
240
|
+
missing.push("voice.whisperModelPath");
|
|
241
|
+
}
|
|
242
|
+
/* v8 ignore stop */
|
|
243
|
+
base.voice = missing.length === 0
|
|
244
|
+
? {
|
|
245
|
+
configured: true,
|
|
246
|
+
/* v8 ignore start -- voice detail copy mirrors runtime resolver modes; resolver tests own the matrix @preserve */
|
|
247
|
+
detail: conversationEngine === "openai-sip"
|
|
248
|
+
? twilioPublicUrl
|
|
249
|
+
? "OpenAI Realtime SIP speech-to-speech; Twilio phone transport attached"
|
|
250
|
+
: "OpenAI Realtime SIP speech-to-speech"
|
|
251
|
+
: conversationEngine === "openai-realtime"
|
|
252
|
+
? twilioPublicUrl
|
|
253
|
+
? "OpenAI Realtime speech-to-speech; Twilio phone transport attached"
|
|
254
|
+
: "OpenAI Realtime speech-to-speech"
|
|
255
|
+
: twilioPublicUrl
|
|
256
|
+
? "local Whisper.cpp STT + ElevenLabs TTS; Twilio phone transport attached"
|
|
257
|
+
: "local Whisper.cpp STT + ElevenLabs TTS",
|
|
258
|
+
/* v8 ignore stop */
|
|
259
|
+
}
|
|
260
|
+
: {
|
|
261
|
+
configured: false,
|
|
262
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
263
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
264
|
+
? "not attached on this machine"
|
|
265
|
+
: runtimeConfig.ok && machineRuntimeConfig.ok
|
|
266
|
+
? `missing ${missing.join("/")}`
|
|
267
|
+
: !runtimeConfig.ok
|
|
268
|
+
? unavailableDetail
|
|
269
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
162
270
|
};
|
|
163
271
|
}
|
|
164
272
|
return base;
|
|
165
273
|
}
|
|
274
|
+
function senseRepairHint(agent, sense) {
|
|
275
|
+
if (sense === "teams") {
|
|
276
|
+
return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
|
|
277
|
+
}
|
|
278
|
+
if (sense === "mail") {
|
|
279
|
+
return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
|
|
280
|
+
}
|
|
281
|
+
if (sense === "voice") {
|
|
282
|
+
return `Agent-runnable: run 'ouro connect voice --agent ${agent}' for config guidance; use voice.twilioConversationEngine=openai-sip with voice.openaiRealtimeApiKey, voice.openaiSipProjectId, and voice.openaiSipWebhookSecret for preferred SIP phone voice; use openai-realtime for Media Streams fallback, or save ElevenLabs and local Whisper.cpp settings for cascade fallback; then run 'ouro up' again.`;
|
|
283
|
+
}
|
|
284
|
+
return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
|
|
285
|
+
}
|
|
286
|
+
function currentMachineId() {
|
|
287
|
+
return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
|
|
288
|
+
}
|
|
166
289
|
function parseSenseSnapshotName(name) {
|
|
167
290
|
const parts = name.split(":");
|
|
168
291
|
if (parts.length !== 2)
|
|
169
292
|
return null;
|
|
170
293
|
const [agent, sense] = parts;
|
|
171
|
-
if (sense !== "teams" && sense !== "bluebubbles")
|
|
294
|
+
if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail" && sense !== "voice")
|
|
172
295
|
return null;
|
|
173
296
|
return { agent, sense };
|
|
174
297
|
}
|
|
@@ -177,6 +300,34 @@ function runtimeInfoFor(status) {
|
|
|
177
300
|
return { runtime: "running" };
|
|
178
301
|
return { runtime: "error" };
|
|
179
302
|
}
|
|
303
|
+
function managedSenseEntry(sense) {
|
|
304
|
+
if (sense === "teams")
|
|
305
|
+
return "senses/teams-entry.js";
|
|
306
|
+
if (sense === "bluebubbles")
|
|
307
|
+
return "senses/bluebubbles/entry.js";
|
|
308
|
+
if (sense === "voice")
|
|
309
|
+
return "senses/voice-entry.js";
|
|
310
|
+
return "senses/mail-entry.js";
|
|
311
|
+
}
|
|
312
|
+
function runtimeCredentialBootstrapFor(agent, sense) {
|
|
313
|
+
const runtime = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
314
|
+
const machineId = sense === "bluebubbles" || sense === "voice" ? currentMachineId() : undefined;
|
|
315
|
+
const machine = sense === "bluebubbles" || sense === "voice" ? (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent) : null;
|
|
316
|
+
const providerPool = (0, provider_credentials_1.readProviderCredentialPool)(agent);
|
|
317
|
+
const providerCredentialRecords = providerPool.ok
|
|
318
|
+
? Object.values(providerPool.pool.providers).filter((record) => !!record)
|
|
319
|
+
: [];
|
|
320
|
+
const bootstrap = {
|
|
321
|
+
agentName: agent,
|
|
322
|
+
runtimeConfig: runtime.ok ? runtime.config : undefined,
|
|
323
|
+
machineRuntimeConfig: machine?.ok ? machine.config : undefined,
|
|
324
|
+
machineId,
|
|
325
|
+
providerCredentialRecords: providerCredentialRecords.length > 0 ? providerCredentialRecords : undefined,
|
|
326
|
+
};
|
|
327
|
+
if (!bootstrap.runtimeConfig && !bootstrap.machineRuntimeConfig && !bootstrap.providerCredentialRecords)
|
|
328
|
+
return null;
|
|
329
|
+
return bootstrap;
|
|
330
|
+
}
|
|
180
331
|
function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
181
332
|
if (!lastCheckedAt) {
|
|
182
333
|
return false;
|
|
@@ -187,53 +338,186 @@ function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
|
187
338
|
}
|
|
188
339
|
return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
|
|
189
340
|
}
|
|
341
|
+
function readBlueBubblesRuntimeJson(runtimePath) {
|
|
342
|
+
try {
|
|
343
|
+
const raw = fs.readFileSync(runtimePath, "utf-8");
|
|
344
|
+
const parsed = JSON.parse(raw);
|
|
345
|
+
/* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
|
|
346
|
+
return {
|
|
347
|
+
upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
|
|
348
|
+
? parsed.upstreamStatus
|
|
349
|
+
: "unknown",
|
|
350
|
+
detail: typeof parsed.detail === "string" && parsed.detail.trim()
|
|
351
|
+
? parsed.detail
|
|
352
|
+
: "startup health probe pending",
|
|
353
|
+
lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
|
|
354
|
+
proofMethod: typeof parsed.proofMethod === "string" && parsed.proofMethod.trim()
|
|
355
|
+
? parsed.proofMethod
|
|
356
|
+
: undefined,
|
|
357
|
+
pendingRecoveryCount: typeof parsed.pendingRecoveryCount === "number" && Number.isFinite(parsed.pendingRecoveryCount)
|
|
358
|
+
? parsed.pendingRecoveryCount
|
|
359
|
+
: 0,
|
|
360
|
+
failedRecoveryCount: typeof parsed.failedRecoveryCount === "number" && Number.isFinite(parsed.failedRecoveryCount)
|
|
361
|
+
? parsed.failedRecoveryCount
|
|
362
|
+
: 0,
|
|
363
|
+
oldestPendingRecoveryAt: typeof parsed.oldestPendingRecoveryAt === "string" ? parsed.oldestPendingRecoveryAt : undefined,
|
|
364
|
+
oldestPendingRecoveryAgeMs: typeof parsed.oldestPendingRecoveryAgeMs === "number" && Number.isFinite(parsed.oldestPendingRecoveryAgeMs)
|
|
365
|
+
? parsed.oldestPendingRecoveryAgeMs
|
|
366
|
+
: undefined,
|
|
367
|
+
activeTurnCount: typeof parsed.activeTurnCount === "number" && Number.isFinite(parsed.activeTurnCount)
|
|
368
|
+
? parsed.activeTurnCount
|
|
369
|
+
: undefined,
|
|
370
|
+
stalledTurnCount: typeof parsed.stalledTurnCount === "number" && Number.isFinite(parsed.stalledTurnCount)
|
|
371
|
+
? parsed.stalledTurnCount
|
|
372
|
+
: undefined,
|
|
373
|
+
oldestActiveTurnStartedAt: typeof parsed.oldestActiveTurnStartedAt === "string" ? parsed.oldestActiveTurnStartedAt : undefined,
|
|
374
|
+
oldestActiveTurnAgeMs: typeof parsed.oldestActiveTurnAgeMs === "number" && Number.isFinite(parsed.oldestActiveTurnAgeMs)
|
|
375
|
+
? parsed.oldestActiveTurnAgeMs
|
|
376
|
+
: undefined,
|
|
377
|
+
};
|
|
378
|
+
/* v8 ignore stop */
|
|
379
|
+
/* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
|
|
380
|
+
}
|
|
381
|
+
catch {
|
|
382
|
+
return { upstreamStatus: "unknown", detail: "startup health probe pending", pendingRecoveryCount: 0, failedRecoveryCount: 0 };
|
|
383
|
+
}
|
|
384
|
+
/* v8 ignore stop */
|
|
385
|
+
}
|
|
190
386
|
function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
|
|
191
387
|
const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
192
388
|
const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
|
|
193
389
|
if (!fs.existsSync(runtimePath)) {
|
|
194
390
|
return { runtime: snapshot?.runtime };
|
|
195
391
|
}
|
|
196
|
-
const state = (
|
|
392
|
+
const state = readBlueBubblesRuntimeJson(runtimePath);
|
|
393
|
+
const checkedAtMs = state.lastCheckedAt ? Date.parse(state.lastCheckedAt) : Number.NaN;
|
|
394
|
+
const proofFacts = {
|
|
395
|
+
proofMethod: state.proofMethod ?? "bluebubbles.checkHealth",
|
|
396
|
+
lastProofAt: state.lastCheckedAt,
|
|
397
|
+
proofAgeMs: Number.isFinite(checkedAtMs) ? Math.max(0, Date.now() - checkedAtMs) : undefined,
|
|
398
|
+
pendingRecoveryCount: state.pendingRecoveryCount,
|
|
399
|
+
failedRecoveryCount: state.failedRecoveryCount,
|
|
400
|
+
oldestPendingRecoveryAt: state.oldestPendingRecoveryAt,
|
|
401
|
+
oldestPendingRecoveryAgeMs: state.oldestPendingRecoveryAgeMs,
|
|
402
|
+
activeTurnCount: state.activeTurnCount,
|
|
403
|
+
stalledTurnCount: state.stalledTurnCount,
|
|
404
|
+
oldestActiveTurnStartedAt: state.oldestActiveTurnStartedAt,
|
|
405
|
+
oldestActiveTurnAgeMs: state.oldestActiveTurnAgeMs,
|
|
406
|
+
};
|
|
197
407
|
if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
|
|
198
|
-
return {
|
|
408
|
+
return {
|
|
409
|
+
runtime: snapshot?.runtime,
|
|
410
|
+
lastFailure: state.lastCheckedAt ? "BlueBubbles proof is stale" : undefined,
|
|
411
|
+
failureLayer: state.lastCheckedAt ? "proof_freshness" : undefined,
|
|
412
|
+
};
|
|
413
|
+
}
|
|
414
|
+
if (snapshot?.runtime !== "running") {
|
|
415
|
+
return {
|
|
416
|
+
runtime: "error",
|
|
417
|
+
detail: "BlueBubbles listener is not running",
|
|
418
|
+
...proofFacts,
|
|
419
|
+
lastFailure: "listener process is not running",
|
|
420
|
+
failureLayer: "listener",
|
|
421
|
+
recoveryAction: "daemon health monitor will restart the BlueBubbles listener when its probe fails",
|
|
422
|
+
};
|
|
199
423
|
}
|
|
200
424
|
if (state.upstreamStatus === "error") {
|
|
201
425
|
return {
|
|
202
426
|
runtime: "error",
|
|
203
427
|
detail: state.detail,
|
|
428
|
+
...proofFacts,
|
|
429
|
+
lastFailure: state.detail,
|
|
430
|
+
failureLayer: "upstream",
|
|
431
|
+
recoveryAction: "verify BlueBubbles server/app auth and local machine attachment, then retry the listener",
|
|
432
|
+
};
|
|
433
|
+
}
|
|
434
|
+
if (state.pendingRecoveryCount > 0) {
|
|
435
|
+
return {
|
|
436
|
+
runtime: "error",
|
|
437
|
+
detail: state.detail,
|
|
438
|
+
...proofFacts,
|
|
439
|
+
lastFailure: state.detail,
|
|
440
|
+
failureLayer: "recovery_queue",
|
|
441
|
+
recoveryAction: "queued recovery will retry; inspect BlueBubbles inbound/recovery sidecar logs if age keeps growing",
|
|
442
|
+
};
|
|
443
|
+
}
|
|
444
|
+
if ((state.stalledTurnCount ?? 0) > 0) {
|
|
445
|
+
return {
|
|
446
|
+
runtime: "error",
|
|
447
|
+
detail: state.detail,
|
|
448
|
+
...proofFacts,
|
|
449
|
+
lastFailure: state.detail,
|
|
450
|
+
failureLayer: "live_turn_stall",
|
|
451
|
+
recoveryAction: "live iMessage turn timeout/watchdog will release the lane and recovery will retry captured messages",
|
|
204
452
|
};
|
|
205
453
|
}
|
|
206
454
|
if (state.upstreamStatus === "ok") {
|
|
207
|
-
return {
|
|
455
|
+
return {
|
|
456
|
+
runtime: "running",
|
|
457
|
+
...proofFacts,
|
|
458
|
+
...(state.failedRecoveryCount > 0 ? { detail: state.detail } : {}),
|
|
459
|
+
...(state.failedRecoveryCount > 0 ? {
|
|
460
|
+
lastFailure: state.detail,
|
|
461
|
+
failureLayer: "recovery_quarantine",
|
|
462
|
+
recoveryAction: "inspect quarantined BlueBubbles recovery failures; live transport remains reachable",
|
|
463
|
+
} : {}),
|
|
464
|
+
};
|
|
208
465
|
}
|
|
209
|
-
return { runtime: snapshot?.runtime };
|
|
466
|
+
return { runtime: snapshot?.runtime, ...proofFacts };
|
|
210
467
|
}
|
|
211
468
|
class DaemonSenseManager {
|
|
212
469
|
processManager;
|
|
213
470
|
contexts;
|
|
471
|
+
pendingConfigRefreshes = new Set();
|
|
214
472
|
bundlesRoot;
|
|
215
473
|
constructor(options) {
|
|
216
474
|
const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
217
|
-
const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
|
|
218
475
|
this.bundlesRoot = bundlesRoot;
|
|
219
476
|
this.contexts = new Map(options.agents.map((agent) => {
|
|
220
477
|
const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
|
|
221
|
-
const facts =
|
|
478
|
+
const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
222
479
|
return [agent, { senses, facts }];
|
|
223
480
|
}));
|
|
224
481
|
const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
225
|
-
return ["teams", "bluebubbles"]
|
|
226
|
-
.filter((sense) => context.senses[sense].enabled
|
|
482
|
+
return ["teams", "bluebubbles", "mail", "voice"]
|
|
483
|
+
.filter((sense) => context.senses[sense].enabled)
|
|
227
484
|
.map((sense) => ({
|
|
228
485
|
name: `${agent}:${sense}`,
|
|
229
486
|
agentArg: agent,
|
|
230
|
-
entry: sense
|
|
487
|
+
entry: managedSenseEntry(sense),
|
|
231
488
|
channel: sense,
|
|
232
489
|
autoStart: true,
|
|
490
|
+
getRuntimeCredentialBootstrap: () => runtimeCredentialBootstrapFor(agent, sense),
|
|
233
491
|
}));
|
|
234
492
|
});
|
|
235
493
|
this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
|
|
236
494
|
agents: managedSenseAgents,
|
|
495
|
+
configCheck: async (name) => {
|
|
496
|
+
const parsed = parseSenseSnapshotName(name);
|
|
497
|
+
if (!parsed)
|
|
498
|
+
return { ok: true };
|
|
499
|
+
const context = this.contexts.get(parsed.agent);
|
|
500
|
+
if (!context)
|
|
501
|
+
return { ok: true };
|
|
502
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(parsed.agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent));
|
|
503
|
+
const fact = context.facts[parsed.sense];
|
|
504
|
+
if (fact.configured)
|
|
505
|
+
return { ok: true };
|
|
506
|
+
this.scheduleSenseConfigRefresh(name, parsed);
|
|
507
|
+
if (fact.optional) {
|
|
508
|
+
return {
|
|
509
|
+
ok: false,
|
|
510
|
+
skip: true,
|
|
511
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
|
|
512
|
+
};
|
|
513
|
+
}
|
|
514
|
+
return {
|
|
515
|
+
ok: false,
|
|
516
|
+
skip: true,
|
|
517
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
|
|
518
|
+
fix: senseRepairHint(parsed.agent, parsed.sense),
|
|
519
|
+
};
|
|
520
|
+
},
|
|
237
521
|
});
|
|
238
522
|
(0, runtime_1.emitNervesEvent)({
|
|
239
523
|
component: "channels",
|
|
@@ -245,12 +529,125 @@ class DaemonSenseManager {
|
|
|
245
529
|
},
|
|
246
530
|
});
|
|
247
531
|
}
|
|
532
|
+
scheduleSenseConfigRefresh(name, parsed) {
|
|
533
|
+
if (this.pendingConfigRefreshes.has(name))
|
|
534
|
+
return;
|
|
535
|
+
this.pendingConfigRefreshes.add(name);
|
|
536
|
+
void this.refreshSenseConfigAndRetry(name, parsed);
|
|
537
|
+
}
|
|
538
|
+
async refreshSenseConfigAndRetry(name, parsed) {
|
|
539
|
+
try {
|
|
540
|
+
const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
|
|
541
|
+
const machineRefreshed = parsed.sense === "bluebubbles" || parsed.sense === "voice"
|
|
542
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
543
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
|
|
544
|
+
const context = this.contexts.get(parsed.agent);
|
|
545
|
+
/* v8 ignore next -- defensive: config refreshes are only scheduled for known agent contexts @preserve */
|
|
546
|
+
if (!context)
|
|
547
|
+
return;
|
|
548
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
|
|
549
|
+
if (!context.facts[parsed.sense].configured)
|
|
550
|
+
return;
|
|
551
|
+
setTimeout(() => {
|
|
552
|
+
void this.processManager.startAgent?.(name).catch((error) => {
|
|
553
|
+
(0, runtime_1.emitNervesEvent)({
|
|
554
|
+
level: "error",
|
|
555
|
+
component: "channels",
|
|
556
|
+
event: "channel.daemon_sense_autostart_error",
|
|
557
|
+
message: "sense autostart failed",
|
|
558
|
+
/* v8 ignore next -- defensive: process manager rejects with Error instances in normal use @preserve */
|
|
559
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
560
|
+
});
|
|
561
|
+
});
|
|
562
|
+
}, 0);
|
|
563
|
+
}
|
|
564
|
+
catch (error) {
|
|
565
|
+
(0, runtime_1.emitNervesEvent)({
|
|
566
|
+
level: "error",
|
|
567
|
+
component: "channels",
|
|
568
|
+
event: "channel.daemon_sense_autostart_error",
|
|
569
|
+
message: "sense config refresh failed",
|
|
570
|
+
/* v8 ignore next -- defensive: runtime credential refresh rejects with Error instances in normal use @preserve */
|
|
571
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
572
|
+
});
|
|
573
|
+
}
|
|
574
|
+
finally {
|
|
575
|
+
this.pendingConfigRefreshes.delete(name);
|
|
576
|
+
}
|
|
577
|
+
}
|
|
578
|
+
async refreshEnabledSenseConfigs() {
|
|
579
|
+
const refreshes = [...this.contexts.entries()].map(async ([agent, context]) => {
|
|
580
|
+
const enabledManagedSenses = ["teams", "bluebubbles", "mail", "voice"]
|
|
581
|
+
.filter((sense) => context.senses[sense].enabled);
|
|
582
|
+
/* v8 ignore next -- periodic refresh work only exists when a managed background sense is enabled @preserve */
|
|
583
|
+
if (enabledManagedSenses.length === 0)
|
|
584
|
+
return;
|
|
585
|
+
/* v8 ignore start -- periodic freshness refresh uses the same runtime readers covered by startup integration tests @preserve */
|
|
586
|
+
const runtimeConfig = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(agent, { preserveCachedOnFailure: true });
|
|
587
|
+
const needsMachineConfig = enabledManagedSenses.some((sense) => sense === "bluebubbles" || sense === "voice");
|
|
588
|
+
const machineRuntimeConfig = needsMachineConfig
|
|
589
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
590
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
591
|
+
/* v8 ignore stop */
|
|
592
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
593
|
+
});
|
|
594
|
+
await Promise.all(refreshes);
|
|
595
|
+
}
|
|
248
596
|
async startAutoStartSenses() {
|
|
597
|
+
await this.refreshEnabledSenseConfigs();
|
|
249
598
|
await this.processManager.startAutoStartAgents();
|
|
250
599
|
}
|
|
600
|
+
triggerAutoStartSenses() {
|
|
601
|
+
if (this.processManager.triggerAutoStartAgents) {
|
|
602
|
+
this.processManager.triggerAutoStartAgents();
|
|
603
|
+
return;
|
|
604
|
+
}
|
|
605
|
+
void this.processManager.startAutoStartAgents().catch((error) => {
|
|
606
|
+
(0, runtime_1.emitNervesEvent)({
|
|
607
|
+
level: "error",
|
|
608
|
+
component: "channels",
|
|
609
|
+
event: "channel.daemon_sense_autostart_error",
|
|
610
|
+
message: "sense autostart failed",
|
|
611
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
612
|
+
});
|
|
613
|
+
});
|
|
614
|
+
}
|
|
251
615
|
async stopAll() {
|
|
252
616
|
await this.processManager.stopAll();
|
|
253
617
|
}
|
|
618
|
+
/* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
|
|
619
|
+
listManagedPids() {
|
|
620
|
+
return this.processManager.listAgentSnapshots()
|
|
621
|
+
.map((s) => s.pid)
|
|
622
|
+
.filter((pid) => pid !== null && pid !== undefined);
|
|
623
|
+
}
|
|
624
|
+
/* v8 ignore stop */
|
|
625
|
+
listHealthProbes() {
|
|
626
|
+
const probes = [];
|
|
627
|
+
for (const [agent, context] of this.contexts.entries()) {
|
|
628
|
+
const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
629
|
+
const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
630
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
631
|
+
if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
|
|
632
|
+
continue;
|
|
633
|
+
}
|
|
634
|
+
const machinePayload = machineRuntimeConfig.config;
|
|
635
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
636
|
+
const port = numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT);
|
|
637
|
+
probes.push({
|
|
638
|
+
...(0, http_health_probe_1.createHttpHealthProbe)(`bluebubbles:${agent}`, port),
|
|
639
|
+
managedName: `${agent}:bluebubbles`,
|
|
640
|
+
});
|
|
641
|
+
}
|
|
642
|
+
return probes;
|
|
643
|
+
}
|
|
644
|
+
async restartSense(managedName) {
|
|
645
|
+
if (this.processManager.restartAgent) {
|
|
646
|
+
await this.processManager.restartAgent(managedName);
|
|
647
|
+
return;
|
|
648
|
+
}
|
|
649
|
+
await this.processManager.startAgent?.(managedName);
|
|
650
|
+
}
|
|
254
651
|
listSenseRows() {
|
|
255
652
|
const runtime = new Map();
|
|
256
653
|
for (const snapshot of this.processManager.listAgentSnapshots()) {
|
|
@@ -262,6 +659,7 @@ class DaemonSenseManager {
|
|
|
262
659
|
runtime.set(parsed.agent, current);
|
|
263
660
|
}
|
|
264
661
|
const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
662
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
265
663
|
const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
|
|
266
664
|
const runtimeInfo = {
|
|
267
665
|
cli: { configured: true },
|
|
@@ -271,8 +669,18 @@ class DaemonSenseManager {
|
|
|
271
669
|
},
|
|
272
670
|
bluebubbles: {
|
|
273
671
|
configured: context.facts.bluebubbles.configured,
|
|
672
|
+
optional: context.facts.bluebubbles.optional,
|
|
274
673
|
...blueBubblesRuntimeFacts,
|
|
275
674
|
},
|
|
675
|
+
mail: {
|
|
676
|
+
configured: context.facts.mail.configured,
|
|
677
|
+
...(runtime.get(agent)?.mail ?? {}),
|
|
678
|
+
},
|
|
679
|
+
voice: {
|
|
680
|
+
configured: context.facts.voice.configured,
|
|
681
|
+
optional: context.facts.voice.optional,
|
|
682
|
+
...(runtime.get(agent)?.voice ?? {}),
|
|
683
|
+
},
|
|
276
684
|
};
|
|
277
685
|
const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
|
|
278
686
|
return inventory.map((entry) => ({
|
|
@@ -287,6 +695,22 @@ class DaemonSenseManager {
|
|
|
287
695
|
?? context.facts[entry.sense].detail
|
|
288
696
|
: context.facts[entry.sense].detail
|
|
289
697
|
: "not enabled in agent.json",
|
|
698
|
+
...(entry.sense === "bluebubbles" ? {
|
|
699
|
+
proofMethod: blueBubblesRuntimeFacts.proofMethod,
|
|
700
|
+
lastProofAt: blueBubblesRuntimeFacts.lastProofAt,
|
|
701
|
+
proofAgeMs: blueBubblesRuntimeFacts.proofAgeMs,
|
|
702
|
+
lastFailure: blueBubblesRuntimeFacts.lastFailure,
|
|
703
|
+
failureLayer: blueBubblesRuntimeFacts.failureLayer,
|
|
704
|
+
recoveryAction: blueBubblesRuntimeFacts.recoveryAction,
|
|
705
|
+
pendingRecoveryCount: blueBubblesRuntimeFacts.pendingRecoveryCount,
|
|
706
|
+
failedRecoveryCount: blueBubblesRuntimeFacts.failedRecoveryCount,
|
|
707
|
+
oldestPendingRecoveryAt: blueBubblesRuntimeFacts.oldestPendingRecoveryAt,
|
|
708
|
+
oldestPendingRecoveryAgeMs: blueBubblesRuntimeFacts.oldestPendingRecoveryAgeMs,
|
|
709
|
+
activeTurnCount: blueBubblesRuntimeFacts.activeTurnCount,
|
|
710
|
+
stalledTurnCount: blueBubblesRuntimeFacts.stalledTurnCount,
|
|
711
|
+
oldestActiveTurnStartedAt: blueBubblesRuntimeFacts.oldestActiveTurnStartedAt,
|
|
712
|
+
oldestActiveTurnAgeMs: blueBubblesRuntimeFacts.oldestActiveTurnAgeMs,
|
|
713
|
+
} : {}),
|
|
290
714
|
}));
|
|
291
715
|
});
|
|
292
716
|
(0, runtime_1.emitNervesEvent)({
|