@ouro.bot/cli 0.1.0-alpha.62 → 0.1.0-alpha.637
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 +4087 -13
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +144 -0
- package/dist/arc/episodes.js +118 -0
- package/dist/arc/evolution.js +487 -0
- package/dist/arc/intentions.js +134 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +270 -0
- package/dist/arc/packets.js +288 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +57 -0
- package/dist/heart/active-work.js +860 -43
- 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/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 +100 -0
- package/dist/heart/awaiting/await-scheduler.js +377 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bridges/manager.js +137 -17
- package/dist/heart/bridges/store.js +14 -2
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +135 -0
- package/dist/heart/config-registry.js +322 -0
- package/dist/heart/config.js +114 -119
- package/dist/heart/core.js +1028 -248
- 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-desk.js +322 -0
- package/dist/heart/daemon/cli-exec.js +7468 -0
- package/dist/heart/daemon/cli-help.js +505 -0
- package/dist/heart/daemon/cli-parse.js +1554 -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 -1700
- 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 +906 -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 +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 +79 -10
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- package/dist/heart/daemon/migrate-to-desk.js +848 -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/plugin-cli.js +432 -0
- package/dist/heart/daemon/process-manager.js +501 -35
- 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 +11 -3
- 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 +493 -38
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +22 -9
- 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 +117 -39
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +229 -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 -4
- 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 +37 -14
- package/dist/heart/identity.js +168 -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 +197 -0
- package/dist/heart/mailbox/readers/agent-machine.js +418 -0
- package/dist/heart/mailbox/readers/continuity-readers.js +319 -0
- package/dist/heart/mailbox/readers/mail.js +375 -0
- package/dist/heart/mailbox/readers/runtime-readers.js +756 -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 +692 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +19 -0
- package/dist/heart/orientation-frame.js +217 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +272 -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 +48 -24
- package/dist/heart/session-events.js +1163 -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 +133 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +44 -27
- package/dist/heart/structured-output.js +196 -0
- 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 +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-9-AxCxuB.js +61 -0
- package/dist/mailbox-ui/assets/index-CWzt267f.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 +715 -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 +568 -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 +334 -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 +14 -1
- package/dist/mind/context.js +251 -101
- package/dist/mind/desk-section.js +310 -0
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +68 -76
- 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 +48 -4
- package/dist/mind/friends/types.js +2 -2
- package/dist/mind/journal-index.js +162 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +6 -1
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +1075 -146
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +139 -5
- package/dist/nerves/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 +16 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +1040 -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 +331 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +166 -10
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +219 -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/desk/classifier.js +362 -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 +385 -0
- package/dist/repertoire/mcp-client.js +295 -0
- package/dist/repertoire/mcp-manager.js +403 -0
- package/dist/repertoire/mcp-tools.js +83 -0
- package/dist/repertoire/plugin-mcp.js +175 -0
- package/dist/repertoire/plugins.js +253 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +48 -4
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-awaiting.js +372 -0
- package/dist/repertoire/tools-base.js +59 -1082
- package/dist/repertoire/tools-bluebubbles.js +2 -0
- package/dist/repertoire/tools-bridge.js +144 -0
- package/dist/repertoire/tools-bundle.js +993 -0
- package/dist/repertoire/tools-config.js +186 -0
- package/dist/repertoire/tools-continuity.js +252 -0
- package/dist/repertoire/tools-credential.js +383 -0
- package/dist/repertoire/tools-evolution.js +527 -0
- package/dist/repertoire/tools-files.js +344 -0
- package/dist/repertoire/tools-flight.js +227 -0
- package/dist/repertoire/tools-flow.js +119 -0
- package/dist/repertoire/tools-github.js +3 -8
- package/dist/repertoire/tools-mail.js +1975 -0
- package/dist/repertoire/tools-notes.js +438 -0
- package/dist/repertoire/tools-obligations.js +143 -0
- package/dist/repertoire/tools-orientation.js +31 -0
- package/dist/repertoire/tools-record.js +464 -0
- package/dist/repertoire/tools-runtime.js +150 -0
- package/dist/repertoire/tools-session.js +766 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +182 -0
- package/dist/repertoire/tools-surface.js +344 -0
- package/dist/repertoire/tools-teams.js +12 -39
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +982 -0
- package/dist/repertoire/tools-user-profile.js +146 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools-voice.js +145 -0
- package/dist/repertoire/tools.js +215 -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 +186 -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 +2737 -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 +517 -204
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +254 -22
- package/dist/senses/inner-dialog.js +505 -40
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +666 -181
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +393 -0
- package/dist/senses/surface-tool.js +108 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +388 -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 +5079 -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/dist/util/frontmatter.js +53 -0
- package/package.json +48 -8
- 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/safe-workspace.js +0 -228
- package/dist/heart/session-recall.js +0 -116
- package/dist/mind/associative-recall.js +0 -209
- package/dist/repertoire/tasks/board.js +0 -134
- package/dist/repertoire/tasks/index.js +0 -224
- package/dist/repertoire/tasks/lifecycle.js +0 -80
- package/dist/repertoire/tasks/middleware.js +0 -65
- package/dist/repertoire/tasks/parser.js +0 -173
- package/dist/repertoire/tasks/scanner.js +0 -132
- package/dist/repertoire/tasks/transitions.js +0 -144
- 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 -7
- /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/{repertoire/tasks/types.js → heart/attachments/sources/adapter.js} +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,9 +38,11 @@ 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");
|
|
45
47
|
const DEFAULT_TEAMS_PORT = 3978;
|
|
46
48
|
const DEFAULT_BLUEBUBBLES_PORT = 18790;
|
|
@@ -51,6 +53,8 @@ function defaultSenses() {
|
|
|
51
53
|
cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
|
|
52
54
|
teams: { ...identity_1.DEFAULT_AGENT_SENSES.teams },
|
|
53
55
|
bluebubbles: { ...identity_1.DEFAULT_AGENT_SENSES.bluebubbles },
|
|
56
|
+
mail: { ...identity_1.DEFAULT_AGENT_SENSES.mail },
|
|
57
|
+
voice: { ...identity_1.DEFAULT_AGENT_SENSES.voice },
|
|
54
58
|
};
|
|
55
59
|
}
|
|
56
60
|
function readAgentSenses(agentJsonPath) {
|
|
@@ -76,7 +80,7 @@ function readAgentSenses(agentJsonPath) {
|
|
|
76
80
|
if (!rawSenses || typeof rawSenses !== "object" || Array.isArray(rawSenses)) {
|
|
77
81
|
return defaults;
|
|
78
82
|
}
|
|
79
|
-
for (const sense of ["cli", "teams", "bluebubbles"]) {
|
|
83
|
+
for (const sense of ["cli", "teams", "bluebubbles", "mail", "voice"]) {
|
|
80
84
|
const rawSense = rawSenses[sense];
|
|
81
85
|
if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
|
|
82
86
|
continue;
|
|
@@ -88,41 +92,50 @@ function readAgentSenses(agentJsonPath) {
|
|
|
88
92
|
}
|
|
89
93
|
return defaults;
|
|
90
94
|
}
|
|
91
|
-
function readSecretsPayload(secretsPath) {
|
|
92
|
-
try {
|
|
93
|
-
const raw = fs.readFileSync(secretsPath, "utf-8");
|
|
94
|
-
const parsed = JSON.parse(raw);
|
|
95
|
-
if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
|
|
96
|
-
return { payload: {}, error: "invalid secrets.json object" };
|
|
97
|
-
}
|
|
98
|
-
return { payload: parsed, error: null };
|
|
99
|
-
}
|
|
100
|
-
catch (error) {
|
|
101
|
-
return {
|
|
102
|
-
payload: {},
|
|
103
|
-
error: error instanceof Error ? error.message : String(error),
|
|
104
|
-
};
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
95
|
function textField(record, key) {
|
|
108
96
|
const value = record?.[key];
|
|
109
97
|
return typeof value === "string" ? value.trim() : "";
|
|
110
98
|
}
|
|
99
|
+
function booleanField(record, key) {
|
|
100
|
+
return record?.[key] === true;
|
|
101
|
+
}
|
|
111
102
|
function numberField(record, key, fallback) {
|
|
112
103
|
const value = record?.[key];
|
|
113
104
|
return typeof value === "number" && Number.isFinite(value) ? value : fallback;
|
|
114
105
|
}
|
|
115
|
-
function
|
|
106
|
+
function compactRuntimeConfigError(agent, error) {
|
|
107
|
+
const compact = error.replace(/\s+/g, " ").trim();
|
|
108
|
+
if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
|
|
109
|
+
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`;
|
|
110
|
+
}
|
|
111
|
+
return compact || "unavailable";
|
|
112
|
+
}
|
|
113
|
+
function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
|
|
114
|
+
if (runtimeConfig.ok)
|
|
115
|
+
return "";
|
|
116
|
+
const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
|
|
117
|
+
if (runtimeConfig.reason === "missing")
|
|
118
|
+
return `missing vault ${itemName} (${agent})`;
|
|
119
|
+
return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
|
|
120
|
+
}
|
|
121
|
+
function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
|
|
116
122
|
const base = {
|
|
117
123
|
cli: { configured: true, detail: "local interactive terminal" },
|
|
118
124
|
teams: { configured: false, detail: "not enabled in agent.json" },
|
|
119
125
|
bluebubbles: { configured: false, detail: "not enabled in agent.json" },
|
|
126
|
+
mail: { configured: false, detail: "not enabled in agent.json" },
|
|
127
|
+
voice: { configured: false, detail: "not enabled in agent.json" },
|
|
120
128
|
};
|
|
121
|
-
const
|
|
129
|
+
const payload = runtimeConfig.ok ? runtimeConfig.config : {};
|
|
130
|
+
const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
|
|
122
131
|
const teams = payload.teams;
|
|
123
132
|
const teamsChannel = payload.teamsChannel;
|
|
124
|
-
const
|
|
125
|
-
const
|
|
133
|
+
const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
|
|
134
|
+
const bluebubbles = machinePayload.bluebubbles;
|
|
135
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
136
|
+
const mailroom = payload.mailroom;
|
|
137
|
+
const integrations = payload.integrations;
|
|
138
|
+
const voice = machinePayload.voice;
|
|
126
139
|
if (senses.teams.enabled) {
|
|
127
140
|
const missing = [];
|
|
128
141
|
if (!textField(teams, "clientId"))
|
|
@@ -138,9 +151,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
138
151
|
}
|
|
139
152
|
: {
|
|
140
153
|
configured: false,
|
|
141
|
-
detail:
|
|
142
|
-
? `missing
|
|
143
|
-
:
|
|
154
|
+
detail: runtimeConfig.ok
|
|
155
|
+
? `missing ${missing.join("/")}`
|
|
156
|
+
: unavailableDetail,
|
|
144
157
|
};
|
|
145
158
|
}
|
|
146
159
|
if (senses.bluebubbles.enabled) {
|
|
@@ -156,19 +169,128 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
156
169
|
}
|
|
157
170
|
: {
|
|
158
171
|
configured: false,
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
172
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
173
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
174
|
+
? "not attached on this machine"
|
|
175
|
+
: machineRuntimeConfig.ok
|
|
176
|
+
? `missing ${missing.join("/")}`
|
|
177
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
178
|
+
};
|
|
179
|
+
}
|
|
180
|
+
if (senses.mail.enabled) {
|
|
181
|
+
const privateKeys = mailroom?.privateKeys;
|
|
182
|
+
const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
|
|
183
|
+
const mailboxAddress = textField(mailroom, "mailboxAddress");
|
|
184
|
+
const missing = [];
|
|
185
|
+
if (!mailboxAddress)
|
|
186
|
+
missing.push("mailroom.mailboxAddress");
|
|
187
|
+
if (!hasPrivateKeys)
|
|
188
|
+
missing.push("mailroom.privateKeys");
|
|
189
|
+
base.mail = missing.length === 0
|
|
190
|
+
? { configured: true, detail: mailboxAddress }
|
|
191
|
+
: {
|
|
192
|
+
configured: false,
|
|
193
|
+
detail: runtimeConfig.ok
|
|
194
|
+
? `missing ${missing.join("/")}`
|
|
195
|
+
: unavailableDetail,
|
|
196
|
+
};
|
|
197
|
+
}
|
|
198
|
+
if (senses.voice.enabled) {
|
|
199
|
+
const portableVoice = payload.voice;
|
|
200
|
+
const conversationEngine = (textField(voice, "twilioConversationEngine")
|
|
201
|
+
|| textField(voice, "conversationEngine")
|
|
202
|
+
|| textField(portableVoice, "twilioConversationEngine")
|
|
203
|
+
|| textField(portableVoice, "conversationEngine")
|
|
204
|
+
|| "cascade").toLowerCase();
|
|
205
|
+
const twilioTransportMode = (textField(voice, "twilioTransportMode") || "record-play").toLowerCase();
|
|
206
|
+
const twilioPublicUrl = textField(voice, "twilioPublicUrl");
|
|
207
|
+
const hasOpenAIRealtimeKey = !!(textField(portableVoice, "openaiRealtimeApiKey")
|
|
208
|
+
|| textField(integrations, "openaiApiKey")
|
|
209
|
+
|| textField(integrations, "openaiEmbeddingsApiKey"));
|
|
210
|
+
const missing = [];
|
|
211
|
+
/* v8 ignore start -- voice setup missing-field matrix is enforced by the voice runtime resolver; sense-manager only renders human-readable readiness facts @preserve */
|
|
212
|
+
if (conversationEngine === "openai-realtime" || conversationEngine === "openai-sip") {
|
|
213
|
+
if (conversationEngine === "openai-realtime" && twilioTransportMode !== "media-stream") {
|
|
214
|
+
missing.push("voice.twilioTransportMode=media-stream");
|
|
215
|
+
}
|
|
216
|
+
if (!hasOpenAIRealtimeKey) {
|
|
217
|
+
missing.push("voice.openaiRealtimeApiKey");
|
|
218
|
+
}
|
|
219
|
+
if (conversationEngine === "openai-sip") {
|
|
220
|
+
if (!textField(portableVoice, "openaiSipProjectId") && !textField(voice, "openaiSipProjectId")) {
|
|
221
|
+
missing.push("voice.openaiSipProjectId");
|
|
222
|
+
}
|
|
223
|
+
const allowUnsignedWebhooks = booleanField(portableVoice, "openaiSipAllowUnsignedWebhooks")
|
|
224
|
+
|| booleanField(voice, "openaiSipAllowUnsignedWebhooks");
|
|
225
|
+
if (!allowUnsignedWebhooks && !textField(portableVoice, "openaiSipWebhookSecret") && !textField(voice, "openaiSipWebhookSecret")) {
|
|
226
|
+
missing.push("voice.openaiSipWebhookSecret");
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
else {
|
|
231
|
+
if (!textField(integrations, "elevenLabsApiKey"))
|
|
232
|
+
missing.push("integrations.elevenLabsApiKey");
|
|
233
|
+
if (!textField(integrations, "elevenLabsVoiceId") && !textField(portableVoice, "elevenLabsVoiceId")) {
|
|
234
|
+
missing.push("integrations.elevenLabsVoiceId");
|
|
235
|
+
}
|
|
236
|
+
if (!textField(voice, "whisperCliPath"))
|
|
237
|
+
missing.push("voice.whisperCliPath");
|
|
238
|
+
if (!textField(voice, "whisperModelPath"))
|
|
239
|
+
missing.push("voice.whisperModelPath");
|
|
240
|
+
}
|
|
241
|
+
/* v8 ignore stop */
|
|
242
|
+
base.voice = missing.length === 0
|
|
243
|
+
? {
|
|
244
|
+
configured: true,
|
|
245
|
+
/* v8 ignore start -- voice detail copy mirrors runtime resolver modes; resolver tests own the matrix @preserve */
|
|
246
|
+
detail: conversationEngine === "openai-sip"
|
|
247
|
+
? twilioPublicUrl
|
|
248
|
+
? "OpenAI Realtime SIP speech-to-speech; Twilio phone transport attached"
|
|
249
|
+
: "OpenAI Realtime SIP speech-to-speech"
|
|
250
|
+
: conversationEngine === "openai-realtime"
|
|
251
|
+
? twilioPublicUrl
|
|
252
|
+
? "OpenAI Realtime speech-to-speech; Twilio phone transport attached"
|
|
253
|
+
: "OpenAI Realtime speech-to-speech"
|
|
254
|
+
: twilioPublicUrl
|
|
255
|
+
? "local Whisper.cpp STT + ElevenLabs TTS; Twilio phone transport attached"
|
|
256
|
+
: "local Whisper.cpp STT + ElevenLabs TTS",
|
|
257
|
+
/* v8 ignore stop */
|
|
258
|
+
}
|
|
259
|
+
: {
|
|
260
|
+
configured: false,
|
|
261
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
262
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
263
|
+
? "not attached on this machine"
|
|
264
|
+
: runtimeConfig.ok && machineRuntimeConfig.ok
|
|
265
|
+
? `missing ${missing.join("/")}`
|
|
266
|
+
: !runtimeConfig.ok
|
|
267
|
+
? unavailableDetail
|
|
268
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
162
269
|
};
|
|
163
270
|
}
|
|
164
271
|
return base;
|
|
165
272
|
}
|
|
273
|
+
function senseRepairHint(agent, sense) {
|
|
274
|
+
if (sense === "teams") {
|
|
275
|
+
return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
|
|
276
|
+
}
|
|
277
|
+
if (sense === "mail") {
|
|
278
|
+
return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
|
|
279
|
+
}
|
|
280
|
+
if (sense === "voice") {
|
|
281
|
+
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.`;
|
|
282
|
+
}
|
|
283
|
+
return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
|
|
284
|
+
}
|
|
285
|
+
function currentMachineId() {
|
|
286
|
+
return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
|
|
287
|
+
}
|
|
166
288
|
function parseSenseSnapshotName(name) {
|
|
167
289
|
const parts = name.split(":");
|
|
168
290
|
if (parts.length !== 2)
|
|
169
291
|
return null;
|
|
170
292
|
const [agent, sense] = parts;
|
|
171
|
-
if (sense !== "teams" && sense !== "bluebubbles")
|
|
293
|
+
if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail" && sense !== "voice")
|
|
172
294
|
return null;
|
|
173
295
|
return { agent, sense };
|
|
174
296
|
}
|
|
@@ -177,6 +299,34 @@ function runtimeInfoFor(status) {
|
|
|
177
299
|
return { runtime: "running" };
|
|
178
300
|
return { runtime: "error" };
|
|
179
301
|
}
|
|
302
|
+
function managedSenseEntry(sense) {
|
|
303
|
+
if (sense === "teams")
|
|
304
|
+
return "senses/teams-entry.js";
|
|
305
|
+
if (sense === "bluebubbles")
|
|
306
|
+
return "senses/bluebubbles/entry.js";
|
|
307
|
+
if (sense === "voice")
|
|
308
|
+
return "senses/voice-entry.js";
|
|
309
|
+
return "senses/mail-entry.js";
|
|
310
|
+
}
|
|
311
|
+
function runtimeCredentialBootstrapFor(agent, sense) {
|
|
312
|
+
const runtime = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
313
|
+
const machineId = sense === "bluebubbles" || sense === "voice" ? currentMachineId() : undefined;
|
|
314
|
+
const machine = sense === "bluebubbles" || sense === "voice" ? (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent) : null;
|
|
315
|
+
const providerPool = (0, provider_credentials_1.readProviderCredentialPool)(agent);
|
|
316
|
+
const providerCredentialRecords = providerPool.ok
|
|
317
|
+
? Object.values(providerPool.pool.providers).filter((record) => !!record)
|
|
318
|
+
: [];
|
|
319
|
+
const bootstrap = {
|
|
320
|
+
agentName: agent,
|
|
321
|
+
runtimeConfig: runtime.ok ? runtime.config : undefined,
|
|
322
|
+
machineRuntimeConfig: machine?.ok ? machine.config : undefined,
|
|
323
|
+
machineId,
|
|
324
|
+
providerCredentialRecords: providerCredentialRecords.length > 0 ? providerCredentialRecords : undefined,
|
|
325
|
+
};
|
|
326
|
+
if (!bootstrap.runtimeConfig && !bootstrap.machineRuntimeConfig && !bootstrap.providerCredentialRecords)
|
|
327
|
+
return null;
|
|
328
|
+
return bootstrap;
|
|
329
|
+
}
|
|
180
330
|
function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
181
331
|
if (!lastCheckedAt) {
|
|
182
332
|
return false;
|
|
@@ -187,53 +337,186 @@ function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
|
187
337
|
}
|
|
188
338
|
return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
|
|
189
339
|
}
|
|
340
|
+
function readBlueBubblesRuntimeJson(runtimePath) {
|
|
341
|
+
try {
|
|
342
|
+
const raw = fs.readFileSync(runtimePath, "utf-8");
|
|
343
|
+
const parsed = JSON.parse(raw);
|
|
344
|
+
/* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
|
|
345
|
+
return {
|
|
346
|
+
upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
|
|
347
|
+
? parsed.upstreamStatus
|
|
348
|
+
: "unknown",
|
|
349
|
+
detail: typeof parsed.detail === "string" && parsed.detail.trim()
|
|
350
|
+
? parsed.detail
|
|
351
|
+
: "startup health probe pending",
|
|
352
|
+
lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
|
|
353
|
+
proofMethod: typeof parsed.proofMethod === "string" && parsed.proofMethod.trim()
|
|
354
|
+
? parsed.proofMethod
|
|
355
|
+
: undefined,
|
|
356
|
+
pendingRecoveryCount: typeof parsed.pendingRecoveryCount === "number" && Number.isFinite(parsed.pendingRecoveryCount)
|
|
357
|
+
? parsed.pendingRecoveryCount
|
|
358
|
+
: 0,
|
|
359
|
+
failedRecoveryCount: typeof parsed.failedRecoveryCount === "number" && Number.isFinite(parsed.failedRecoveryCount)
|
|
360
|
+
? parsed.failedRecoveryCount
|
|
361
|
+
: 0,
|
|
362
|
+
oldestPendingRecoveryAt: typeof parsed.oldestPendingRecoveryAt === "string" ? parsed.oldestPendingRecoveryAt : undefined,
|
|
363
|
+
oldestPendingRecoveryAgeMs: typeof parsed.oldestPendingRecoveryAgeMs === "number" && Number.isFinite(parsed.oldestPendingRecoveryAgeMs)
|
|
364
|
+
? parsed.oldestPendingRecoveryAgeMs
|
|
365
|
+
: undefined,
|
|
366
|
+
activeTurnCount: typeof parsed.activeTurnCount === "number" && Number.isFinite(parsed.activeTurnCount)
|
|
367
|
+
? parsed.activeTurnCount
|
|
368
|
+
: undefined,
|
|
369
|
+
stalledTurnCount: typeof parsed.stalledTurnCount === "number" && Number.isFinite(parsed.stalledTurnCount)
|
|
370
|
+
? parsed.stalledTurnCount
|
|
371
|
+
: undefined,
|
|
372
|
+
oldestActiveTurnStartedAt: typeof parsed.oldestActiveTurnStartedAt === "string" ? parsed.oldestActiveTurnStartedAt : undefined,
|
|
373
|
+
oldestActiveTurnAgeMs: typeof parsed.oldestActiveTurnAgeMs === "number" && Number.isFinite(parsed.oldestActiveTurnAgeMs)
|
|
374
|
+
? parsed.oldestActiveTurnAgeMs
|
|
375
|
+
: undefined,
|
|
376
|
+
};
|
|
377
|
+
/* v8 ignore stop */
|
|
378
|
+
/* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
|
|
379
|
+
}
|
|
380
|
+
catch {
|
|
381
|
+
return { upstreamStatus: "unknown", detail: "startup health probe pending", pendingRecoveryCount: 0, failedRecoveryCount: 0 };
|
|
382
|
+
}
|
|
383
|
+
/* v8 ignore stop */
|
|
384
|
+
}
|
|
190
385
|
function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
|
|
191
386
|
const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
192
387
|
const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
|
|
193
388
|
if (!fs.existsSync(runtimePath)) {
|
|
194
389
|
return { runtime: snapshot?.runtime };
|
|
195
390
|
}
|
|
196
|
-
const state = (
|
|
391
|
+
const state = readBlueBubblesRuntimeJson(runtimePath);
|
|
392
|
+
const checkedAtMs = state.lastCheckedAt ? Date.parse(state.lastCheckedAt) : Number.NaN;
|
|
393
|
+
const proofFacts = {
|
|
394
|
+
proofMethod: state.proofMethod ?? "bluebubbles.checkHealth",
|
|
395
|
+
lastProofAt: state.lastCheckedAt,
|
|
396
|
+
proofAgeMs: Number.isFinite(checkedAtMs) ? Math.max(0, Date.now() - checkedAtMs) : undefined,
|
|
397
|
+
pendingRecoveryCount: state.pendingRecoveryCount,
|
|
398
|
+
failedRecoveryCount: state.failedRecoveryCount,
|
|
399
|
+
oldestPendingRecoveryAt: state.oldestPendingRecoveryAt,
|
|
400
|
+
oldestPendingRecoveryAgeMs: state.oldestPendingRecoveryAgeMs,
|
|
401
|
+
activeTurnCount: state.activeTurnCount,
|
|
402
|
+
stalledTurnCount: state.stalledTurnCount,
|
|
403
|
+
oldestActiveTurnStartedAt: state.oldestActiveTurnStartedAt,
|
|
404
|
+
oldestActiveTurnAgeMs: state.oldestActiveTurnAgeMs,
|
|
405
|
+
};
|
|
197
406
|
if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
|
|
198
|
-
return {
|
|
407
|
+
return {
|
|
408
|
+
runtime: snapshot?.runtime,
|
|
409
|
+
lastFailure: state.lastCheckedAt ? "BlueBubbles proof is stale" : undefined,
|
|
410
|
+
failureLayer: state.lastCheckedAt ? "proof_freshness" : undefined,
|
|
411
|
+
};
|
|
412
|
+
}
|
|
413
|
+
if (snapshot?.runtime !== "running") {
|
|
414
|
+
return {
|
|
415
|
+
runtime: "error",
|
|
416
|
+
detail: "BlueBubbles listener is not running",
|
|
417
|
+
...proofFacts,
|
|
418
|
+
lastFailure: "listener process is not running",
|
|
419
|
+
failureLayer: "listener",
|
|
420
|
+
recoveryAction: "daemon health monitor will restart the BlueBubbles listener when its probe fails",
|
|
421
|
+
};
|
|
199
422
|
}
|
|
200
423
|
if (state.upstreamStatus === "error") {
|
|
201
424
|
return {
|
|
202
425
|
runtime: "error",
|
|
203
426
|
detail: state.detail,
|
|
427
|
+
...proofFacts,
|
|
428
|
+
lastFailure: state.detail,
|
|
429
|
+
failureLayer: "upstream",
|
|
430
|
+
recoveryAction: "verify BlueBubbles server/app auth and local machine attachment, then retry the listener",
|
|
431
|
+
};
|
|
432
|
+
}
|
|
433
|
+
if (state.pendingRecoveryCount > 0) {
|
|
434
|
+
return {
|
|
435
|
+
runtime: "error",
|
|
436
|
+
detail: state.detail,
|
|
437
|
+
...proofFacts,
|
|
438
|
+
lastFailure: state.detail,
|
|
439
|
+
failureLayer: "recovery_queue",
|
|
440
|
+
recoveryAction: "queued recovery will retry; inspect BlueBubbles inbound/recovery sidecar logs if age keeps growing",
|
|
441
|
+
};
|
|
442
|
+
}
|
|
443
|
+
if ((state.stalledTurnCount ?? 0) > 0) {
|
|
444
|
+
return {
|
|
445
|
+
runtime: "error",
|
|
446
|
+
detail: state.detail,
|
|
447
|
+
...proofFacts,
|
|
448
|
+
lastFailure: state.detail,
|
|
449
|
+
failureLayer: "live_turn_stall",
|
|
450
|
+
recoveryAction: "live iMessage turn timeout/watchdog will release the lane and recovery will retry captured messages",
|
|
204
451
|
};
|
|
205
452
|
}
|
|
206
453
|
if (state.upstreamStatus === "ok") {
|
|
207
|
-
return {
|
|
454
|
+
return {
|
|
455
|
+
runtime: "running",
|
|
456
|
+
...proofFacts,
|
|
457
|
+
...(state.failedRecoveryCount > 0 ? { detail: state.detail } : {}),
|
|
458
|
+
...(state.failedRecoveryCount > 0 ? {
|
|
459
|
+
lastFailure: state.detail,
|
|
460
|
+
failureLayer: "recovery_quarantine",
|
|
461
|
+
recoveryAction: "inspect quarantined BlueBubbles recovery failures; live transport remains reachable",
|
|
462
|
+
} : {}),
|
|
463
|
+
};
|
|
208
464
|
}
|
|
209
|
-
return { runtime: snapshot?.runtime };
|
|
465
|
+
return { runtime: snapshot?.runtime, ...proofFacts };
|
|
210
466
|
}
|
|
211
467
|
class DaemonSenseManager {
|
|
212
468
|
processManager;
|
|
213
469
|
contexts;
|
|
470
|
+
pendingConfigRefreshes = new Set();
|
|
214
471
|
bundlesRoot;
|
|
215
472
|
constructor(options) {
|
|
216
473
|
const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
217
|
-
const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
|
|
218
474
|
this.bundlesRoot = bundlesRoot;
|
|
219
475
|
this.contexts = new Map(options.agents.map((agent) => {
|
|
220
476
|
const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
|
|
221
|
-
const facts =
|
|
477
|
+
const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
222
478
|
return [agent, { senses, facts }];
|
|
223
479
|
}));
|
|
224
480
|
const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
225
|
-
return ["teams", "bluebubbles"]
|
|
226
|
-
.filter((sense) => context.senses[sense].enabled
|
|
481
|
+
return ["teams", "bluebubbles", "mail", "voice"]
|
|
482
|
+
.filter((sense) => context.senses[sense].enabled)
|
|
227
483
|
.map((sense) => ({
|
|
228
484
|
name: `${agent}:${sense}`,
|
|
229
485
|
agentArg: agent,
|
|
230
|
-
entry: sense
|
|
486
|
+
entry: managedSenseEntry(sense),
|
|
231
487
|
channel: sense,
|
|
232
488
|
autoStart: true,
|
|
489
|
+
getRuntimeCredentialBootstrap: () => runtimeCredentialBootstrapFor(agent, sense),
|
|
233
490
|
}));
|
|
234
491
|
});
|
|
235
492
|
this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
|
|
236
493
|
agents: managedSenseAgents,
|
|
494
|
+
configCheck: async (name) => {
|
|
495
|
+
const parsed = parseSenseSnapshotName(name);
|
|
496
|
+
if (!parsed)
|
|
497
|
+
return { ok: true };
|
|
498
|
+
const context = this.contexts.get(parsed.agent);
|
|
499
|
+
if (!context)
|
|
500
|
+
return { ok: true };
|
|
501
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(parsed.agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent));
|
|
502
|
+
const fact = context.facts[parsed.sense];
|
|
503
|
+
if (fact.configured)
|
|
504
|
+
return { ok: true };
|
|
505
|
+
this.scheduleSenseConfigRefresh(name, parsed);
|
|
506
|
+
if (fact.optional) {
|
|
507
|
+
return {
|
|
508
|
+
ok: false,
|
|
509
|
+
skip: true,
|
|
510
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
|
|
511
|
+
};
|
|
512
|
+
}
|
|
513
|
+
return {
|
|
514
|
+
ok: false,
|
|
515
|
+
skip: true,
|
|
516
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
|
|
517
|
+
fix: senseRepairHint(parsed.agent, parsed.sense),
|
|
518
|
+
};
|
|
519
|
+
},
|
|
237
520
|
});
|
|
238
521
|
(0, runtime_1.emitNervesEvent)({
|
|
239
522
|
component: "channels",
|
|
@@ -245,12 +528,157 @@ class DaemonSenseManager {
|
|
|
245
528
|
},
|
|
246
529
|
});
|
|
247
530
|
}
|
|
531
|
+
scheduleSenseConfigRefresh(name, parsed) {
|
|
532
|
+
if (this.pendingConfigRefreshes.has(name))
|
|
533
|
+
return;
|
|
534
|
+
this.pendingConfigRefreshes.add(name);
|
|
535
|
+
void this.refreshSenseConfigAndRetry(name, parsed);
|
|
536
|
+
}
|
|
537
|
+
async refreshSenseConfigAndRetry(name, parsed) {
|
|
538
|
+
try {
|
|
539
|
+
const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
|
|
540
|
+
const machineRefreshed = parsed.sense === "bluebubbles" || parsed.sense === "voice"
|
|
541
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
542
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
|
|
543
|
+
const context = this.contexts.get(parsed.agent);
|
|
544
|
+
/* v8 ignore next -- defensive: config refreshes are only scheduled for known agent contexts @preserve */
|
|
545
|
+
if (!context)
|
|
546
|
+
return;
|
|
547
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
|
|
548
|
+
if (!context.facts[parsed.sense].configured)
|
|
549
|
+
return;
|
|
550
|
+
setTimeout(() => {
|
|
551
|
+
void this.processManager.startAgent?.(name).catch((error) => {
|
|
552
|
+
(0, runtime_1.emitNervesEvent)({
|
|
553
|
+
level: "error",
|
|
554
|
+
component: "channels",
|
|
555
|
+
event: "channel.daemon_sense_autostart_error",
|
|
556
|
+
message: "sense autostart failed",
|
|
557
|
+
/* v8 ignore next -- defensive: process manager rejects with Error instances in normal use @preserve */
|
|
558
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
559
|
+
});
|
|
560
|
+
});
|
|
561
|
+
}, 0);
|
|
562
|
+
}
|
|
563
|
+
catch (error) {
|
|
564
|
+
(0, runtime_1.emitNervesEvent)({
|
|
565
|
+
level: "error",
|
|
566
|
+
component: "channels",
|
|
567
|
+
event: "channel.daemon_sense_autostart_error",
|
|
568
|
+
message: "sense config refresh failed",
|
|
569
|
+
/* v8 ignore next -- defensive: runtime credential refresh rejects with Error instances in normal use @preserve */
|
|
570
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
571
|
+
});
|
|
572
|
+
}
|
|
573
|
+
finally {
|
|
574
|
+
this.pendingConfigRefreshes.delete(name);
|
|
575
|
+
}
|
|
576
|
+
}
|
|
577
|
+
async refreshEnabledSenseConfigs() {
|
|
578
|
+
const refreshes = [...this.contexts.entries()].map(async ([agent, context]) => {
|
|
579
|
+
const enabledManagedSenses = ["teams", "bluebubbles", "mail", "voice"]
|
|
580
|
+
.filter((sense) => context.senses[sense].enabled);
|
|
581
|
+
/* v8 ignore next -- periodic refresh work only exists when a managed background sense is enabled @preserve */
|
|
582
|
+
if (enabledManagedSenses.length === 0)
|
|
583
|
+
return;
|
|
584
|
+
/* v8 ignore start -- periodic freshness refresh uses the same runtime readers covered by startup integration tests @preserve */
|
|
585
|
+
const runtimeConfig = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(agent, { preserveCachedOnFailure: true });
|
|
586
|
+
const needsMachineConfig = enabledManagedSenses.some((sense) => sense === "bluebubbles" || sense === "voice");
|
|
587
|
+
const machineRuntimeConfig = needsMachineConfig
|
|
588
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
589
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
590
|
+
/* v8 ignore stop */
|
|
591
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
592
|
+
});
|
|
593
|
+
await Promise.all(refreshes);
|
|
594
|
+
}
|
|
248
595
|
async startAutoStartSenses() {
|
|
596
|
+
await this.refreshEnabledSenseConfigs();
|
|
249
597
|
await this.processManager.startAutoStartAgents();
|
|
250
598
|
}
|
|
599
|
+
triggerAutoStartSenses() {
|
|
600
|
+
if (this.processManager.triggerAutoStartAgents) {
|
|
601
|
+
this.processManager.triggerAutoStartAgents();
|
|
602
|
+
return;
|
|
603
|
+
}
|
|
604
|
+
void this.processManager.startAutoStartAgents().catch((error) => {
|
|
605
|
+
(0, runtime_1.emitNervesEvent)({
|
|
606
|
+
level: "error",
|
|
607
|
+
component: "channels",
|
|
608
|
+
event: "channel.daemon_sense_autostart_error",
|
|
609
|
+
message: "sense autostart failed",
|
|
610
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
611
|
+
});
|
|
612
|
+
});
|
|
613
|
+
}
|
|
251
614
|
async stopAll() {
|
|
252
615
|
await this.processManager.stopAll();
|
|
253
616
|
}
|
|
617
|
+
/* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
|
|
618
|
+
listManagedPids() {
|
|
619
|
+
return this.processManager.listAgentSnapshots()
|
|
620
|
+
.map((s) => s.pid)
|
|
621
|
+
.filter((pid) => pid !== null && pid !== undefined);
|
|
622
|
+
}
|
|
623
|
+
/* v8 ignore stop */
|
|
624
|
+
listHealthProbes() {
|
|
625
|
+
const probes = [];
|
|
626
|
+
for (const [agent, context] of this.contexts.entries()) {
|
|
627
|
+
const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
628
|
+
const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
629
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
630
|
+
if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
|
|
631
|
+
continue;
|
|
632
|
+
}
|
|
633
|
+
// DELIBERATELY no HTTP health probe for BlueBubbles.
|
|
634
|
+
//
|
|
635
|
+
// We used to register `createHttpHealthProbe(...)` here, which GETs the
|
|
636
|
+
// sense's /health endpoint every ~60s with a 5s timeout. On 2026-05-11
|
|
637
|
+
// that caused a death spiral:
|
|
638
|
+
// 1. Sense gets busy with real work (e.g. VLM image describe → 20+s)
|
|
639
|
+
// 2. /health probe times out at 5s
|
|
640
|
+
// 3. Daemon declares the sense "critical" → SIGTERMs it mid-work
|
|
641
|
+
// 4. Sense respawns, recovery loop replays the same inbound message
|
|
642
|
+
// into the agent's BB session (visible side-effect — an affected
|
|
643
|
+
// agent saw the same user text injected 76 times)
|
|
644
|
+
// 5. New sense hits the same VLM call, gets killed at 5s, repeat
|
|
645
|
+
//
|
|
646
|
+
// The probe was redundant supervision: dead processes are already
|
|
647
|
+
// recaptured by `processManager`'s child-process exit handler. The
|
|
648
|
+
// probe specifically caught "alive but hung" cases — but the cost
|
|
649
|
+
// (killing genuinely-busy processes and replaying messages) far
|
|
650
|
+
// outweighed the benefit. For "alive but hung" detection we now
|
|
651
|
+
// rely on the agent's own awareness: BB sense's runtime.json carries
|
|
652
|
+
// pendingRecoveryCount + lastRecoveredAt, surfaced in the agent
|
|
653
|
+
// prompt. If recovery has been wedged for too long, the agent can
|
|
654
|
+
// call `restart_runtime` itself (see alpha.598 / PR #723).
|
|
655
|
+
//
|
|
656
|
+
// The respawn-loop guard in processManager is the backstop: even if
|
|
657
|
+
// something else triggers a tight respawn cycle for any reason, the
|
|
658
|
+
// guard fires and refuses further restarts after N attempts in M
|
|
659
|
+
// minutes, so we can never re-enter the 2026-05-11 spiral.
|
|
660
|
+
}
|
|
661
|
+
return probes;
|
|
662
|
+
}
|
|
663
|
+
async restartSense(managedName) {
|
|
664
|
+
if (this.processManager.restartAgent) {
|
|
665
|
+
await this.processManager.restartAgent(managedName);
|
|
666
|
+
return;
|
|
667
|
+
}
|
|
668
|
+
await this.processManager.startAgent?.(managedName);
|
|
669
|
+
}
|
|
670
|
+
async reviveSense(agent, sense) {
|
|
671
|
+
const parsed = parseSenseSnapshotName(`${agent}:${sense}`);
|
|
672
|
+
if (!parsed)
|
|
673
|
+
return null;
|
|
674
|
+
const context = this.contexts.get(parsed.agent);
|
|
675
|
+
if (!context || !context.senses[parsed.sense].enabled)
|
|
676
|
+
return null;
|
|
677
|
+
const managedName = `${parsed.agent}:${parsed.sense}`;
|
|
678
|
+
this.processManager.resetAgentFailureState?.(managedName);
|
|
679
|
+
await this.processManager.startAgent?.(managedName);
|
|
680
|
+
return this.listSenseRows().find((row) => row.agent === parsed.agent && row.sense === parsed.sense) ?? null;
|
|
681
|
+
}
|
|
254
682
|
listSenseRows() {
|
|
255
683
|
const runtime = new Map();
|
|
256
684
|
for (const snapshot of this.processManager.listAgentSnapshots()) {
|
|
@@ -262,6 +690,7 @@ class DaemonSenseManager {
|
|
|
262
690
|
runtime.set(parsed.agent, current);
|
|
263
691
|
}
|
|
264
692
|
const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
693
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
265
694
|
const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
|
|
266
695
|
const runtimeInfo = {
|
|
267
696
|
cli: { configured: true },
|
|
@@ -271,8 +700,18 @@ class DaemonSenseManager {
|
|
|
271
700
|
},
|
|
272
701
|
bluebubbles: {
|
|
273
702
|
configured: context.facts.bluebubbles.configured,
|
|
703
|
+
optional: context.facts.bluebubbles.optional,
|
|
274
704
|
...blueBubblesRuntimeFacts,
|
|
275
705
|
},
|
|
706
|
+
mail: {
|
|
707
|
+
configured: context.facts.mail.configured,
|
|
708
|
+
...(runtime.get(agent)?.mail ?? {}),
|
|
709
|
+
},
|
|
710
|
+
voice: {
|
|
711
|
+
configured: context.facts.voice.configured,
|
|
712
|
+
optional: context.facts.voice.optional,
|
|
713
|
+
...(runtime.get(agent)?.voice ?? {}),
|
|
714
|
+
},
|
|
276
715
|
};
|
|
277
716
|
const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
|
|
278
717
|
return inventory.map((entry) => ({
|
|
@@ -287,6 +726,22 @@ class DaemonSenseManager {
|
|
|
287
726
|
?? context.facts[entry.sense].detail
|
|
288
727
|
: context.facts[entry.sense].detail
|
|
289
728
|
: "not enabled in agent.json",
|
|
729
|
+
...(entry.sense === "bluebubbles" ? {
|
|
730
|
+
proofMethod: blueBubblesRuntimeFacts.proofMethod,
|
|
731
|
+
lastProofAt: blueBubblesRuntimeFacts.lastProofAt,
|
|
732
|
+
proofAgeMs: blueBubblesRuntimeFacts.proofAgeMs,
|
|
733
|
+
lastFailure: blueBubblesRuntimeFacts.lastFailure,
|
|
734
|
+
failureLayer: blueBubblesRuntimeFacts.failureLayer,
|
|
735
|
+
recoveryAction: blueBubblesRuntimeFacts.recoveryAction,
|
|
736
|
+
pendingRecoveryCount: blueBubblesRuntimeFacts.pendingRecoveryCount,
|
|
737
|
+
failedRecoveryCount: blueBubblesRuntimeFacts.failedRecoveryCount,
|
|
738
|
+
oldestPendingRecoveryAt: blueBubblesRuntimeFacts.oldestPendingRecoveryAt,
|
|
739
|
+
oldestPendingRecoveryAgeMs: blueBubblesRuntimeFacts.oldestPendingRecoveryAgeMs,
|
|
740
|
+
activeTurnCount: blueBubblesRuntimeFacts.activeTurnCount,
|
|
741
|
+
stalledTurnCount: blueBubblesRuntimeFacts.stalledTurnCount,
|
|
742
|
+
oldestActiveTurnStartedAt: blueBubblesRuntimeFacts.oldestActiveTurnStartedAt,
|
|
743
|
+
oldestActiveTurnAgeMs: blueBubblesRuntimeFacts.oldestActiveTurnAgeMs,
|
|
744
|
+
} : {}),
|
|
290
745
|
}));
|
|
291
746
|
});
|
|
292
747
|
(0, runtime_1.emitNervesEvent)({
|