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