@ouro.bot/cli 0.1.0-alpha.6 → 0.1.0-alpha.600
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 +229 -183
- 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/SerpentGuide.ouro/agent.json +83 -0
- package/SerpentGuide.ouro/psyche/SOUL.md +25 -0
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/assets/ouroboros.png +0 -0
- package/changelog.json +4182 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +140 -0
- package/dist/arc/episodes.js +117 -0
- package/dist/arc/intentions.js +133 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +254 -0
- package/dist/arc/packets.js +193 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +989 -0
- package/dist/heart/agent-entry.js +69 -3
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/adapter.js +2 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +479 -0
- package/dist/heart/awaiting/await-alert.js +146 -0
- package/dist/heart/awaiting/await-expiry.js +108 -0
- package/dist/heart/awaiting/await-loader.js +91 -0
- package/dist/heart/awaiting/await-parser.js +141 -0
- package/dist/heart/awaiting/await-runtime-state.js +97 -0
- package/dist/heart/awaiting/await-scheduler.js +377 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bridges/manager.js +358 -0
- package/dist/heart/bridges/state-machine.js +135 -0
- package/dist/heart/bridges/store.js +123 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +142 -0
- package/dist/heart/config-registry.js +322 -0
- package/dist/heart/config.js +164 -135
- package/dist/heart/core.js +1069 -260
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +419 -0
- package/dist/heart/daemon/agent-discovery.js +180 -0
- package/dist/heart/daemon/agent-service.js +522 -0
- package/dist/heart/daemon/agentic-repair.js +547 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/boot-sync-probe.js +197 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +776 -0
- package/dist/heart/daemon/cli-exec.js +7571 -0
- package/dist/heart/daemon/cli-help.js +498 -0
- package/dist/heart/daemon/cli-parse.js +1599 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +763 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/connect-bay.js +323 -0
- package/dist/heart/daemon/daemon-cli.js +30 -758
- package/dist/heart/daemon/daemon-entry.js +540 -8
- 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 +287 -0
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +972 -20
- 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 +206 -0
- 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 +188 -0
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- package/dist/heart/daemon/message-router.js +17 -8
- package/dist/heart/daemon/os-cron-deps.js +135 -0
- package/dist/heart/daemon/os-cron.js +14 -12
- package/dist/heart/daemon/ouro-bot-entry.js +4 -2
- package/dist/heart/daemon/ouro-entry.js +3 -1
- package/dist/heart/daemon/process-manager.js +381 -26
- 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 +39 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +191 -0
- package/dist/heart/daemon/runtime-mode.js +67 -0
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +731 -0
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +349 -0
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +330 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +524 -0
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +62 -0
- 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-animation.js +10 -3
- package/dist/heart/{daemon → hatch}/hatch-flow.js +34 -136
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
- package/dist/heart/hatch/specialist-orchestrator.js +129 -0
- package/dist/heart/hatch/specialist-prompt.js +102 -0
- package/dist/heart/hatch/specialist-tools.js +306 -0
- package/dist/heart/identity.js +281 -67
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +2 -20
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
- package/dist/heart/mailbox/mailbox-http-response.js +7 -0
- package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
- package/dist/heart/mailbox/mailbox-http-static.js +103 -0
- package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
- package/dist/heart/mailbox/mailbox-http.js +99 -0
- package/dist/heart/mailbox/mailbox-read.js +31 -0
- package/dist/heart/mailbox/mailbox-types.js +27 -0
- package/dist/heart/mailbox/mailbox-view.js +195 -0
- package/dist/heart/mailbox/readers/agent-machine.js +382 -0
- package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
- package/dist/heart/mailbox/readers/mail.js +367 -0
- package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
- package/dist/heart/mailbox/readers/sessions.js +232 -0
- package/dist/heart/mailbox/readers/shared.js +111 -0
- package/dist/heart/mcp/mcp-server.js +656 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +59 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +267 -0
- package/dist/heart/provider-credentials.js +425 -0
- package/dist/heart/provider-failover.js +301 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +262 -0
- package/dist/heart/provider-readiness-cache.js +40 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +202 -50
- package/dist/heart/providers/azure.js +104 -13
- package/dist/heart/providers/error-classification.js +127 -0
- package/dist/heart/providers/github-copilot.js +145 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +29 -7
- package/dist/heart/providers/openai-codex.js +63 -39
- 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 +70 -0
- package/dist/heart/session-activity.js +190 -0
- package/dist/heart/session-events.js +1149 -0
- package/dist/heart/session-playback-cli-main.js +5 -0
- package/dist/heart/session-playback-cli.js +36 -0
- package/dist/heart/session-playback.js +231 -0
- package/dist/heart/session-stats-cli-main.js +5 -0
- package/dist/heart/session-stats.js +182 -0
- package/dist/heart/session-transcript.js +243 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +129 -34
- package/dist/heart/sync-classification.js +176 -0
- package/dist/heart/sync.js +449 -0
- package/dist/heart/target-resolution.js +127 -0
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/timeouts.js +101 -0
- package/dist/heart/tool-activity-callbacks.js +59 -0
- package/dist/heart/tool-description.js +143 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +421 -0
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/heart/versioning/ouro-bot-global-installer.js +129 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +426 -0
- package/dist/heart/{daemon → versioning}/ouro-uti.js +11 -2
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/versioning/staged-restart.js +146 -0
- package/dist/heart/versioning/update-checker.js +116 -0
- package/dist/heart/versioning/update-hooks.js +142 -0
- package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
- package/dist/mailbox-ui/assets/index-B-461hes.js +61 -0
- package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/mailbox-ui/index.html +15 -0
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +700 -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 +457 -0
- package/dist/mailroom/mbox-import.js +393 -0
- package/dist/mailroom/migration.js +164 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +233 -0
- package/dist/mailroom/search-cache.js +268 -0
- package/dist/mailroom/search-relevance.js +319 -0
- package/dist/mailroom/smtp-ingress.js +176 -0
- package/dist/mailroom/source-state.js +176 -0
- package/dist/mailroom/thread.js +109 -0
- package/dist/mailroom/travel-extract.js +89 -0
- package/dist/mind/bundle-manifest.js +77 -1
- package/dist/mind/context.js +174 -94
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +84 -96
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/first-impressions.js +16 -2
- package/dist/mind/friends/channel.js +74 -0
- package/dist/mind/friends/group-context.js +144 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +58 -3
- package/dist/mind/friends/trust-explanation.js +74 -0
- package/dist/mind/friends/types.js +10 -2
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +76 -9
- package/dist/mind/phrases.js +1 -0
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +1267 -130
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/mind/token-estimate.js +8 -12
- package/dist/nerves/cli-logging.js +22 -3
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +129 -5
- package/dist/nerves/coverage/run-artifacts.js +1 -1
- package/dist/nerves/event-buffer.js +111 -0
- package/dist/nerves/index.js +224 -4
- package/dist/nerves/observation.js +20 -0
- package/dist/nerves/redact.js +79 -0
- package/dist/nerves/review/cli-main.js +5 -0
- package/dist/nerves/review/cli.js +156 -0
- package/dist/nerves/review/core.js +152 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/repertoire/ado-client.js +17 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +997 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +180 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +301 -0
- package/dist/repertoire/coding/index.js +4 -1
- package/dist/repertoire/coding/manager.js +220 -13
- package/dist/repertoire/coding/spawner.js +58 -12
- package/dist/repertoire/coding/tools.js +209 -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/data/ado-endpoints.json +188 -0
- package/dist/repertoire/duffel-client.js +185 -0
- package/dist/repertoire/github-client.js +14 -55
- package/dist/repertoire/graph-client.js +11 -52
- package/dist/repertoire/guardrails.js +396 -0
- package/dist/repertoire/mcp-client.js +295 -0
- package/dist/repertoire/mcp-manager.js +362 -0
- package/dist/repertoire/mcp-tools.js +63 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +15 -24
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +43 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +39 -13
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -79
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-awaiting.js +360 -0
- package/dist/repertoire/tools-base.js +56 -707
- package/dist/repertoire/tools-bluebubbles.js +94 -0
- package/dist/repertoire/tools-bridge.js +142 -0
- package/dist/repertoire/tools-bundle.js +984 -0
- package/dist/repertoire/tools-config.js +185 -0
- package/dist/repertoire/tools-continuity.js +248 -0
- package/dist/repertoire/tools-credential.js +381 -0
- package/dist/repertoire/tools-files.js +342 -0
- package/dist/repertoire/tools-flight.js +224 -0
- package/dist/repertoire/tools-flow.js +119 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1916 -0
- package/dist/repertoire/tools-notes.js +421 -0
- package/dist/repertoire/tools-obligations.js +142 -0
- package/dist/repertoire/tools-runtime.js +61 -0
- package/dist/repertoire/tools-session.js +809 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +345 -0
- package/dist/repertoire/tools-teams.js +64 -61
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +604 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools-voice.js +144 -0
- package/dist/repertoire/tools.js +154 -98
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +131 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +594 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/await-turn-message.js +58 -0
- package/dist/senses/bluebubbles/active-turns.js +216 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/bluebubbles/client.js +685 -0
- package/dist/senses/bluebubbles/entry.js +77 -0
- package/dist/senses/bluebubbles/inbound-log.js +126 -0
- package/dist/senses/bluebubbles/index.js +2548 -0
- package/dist/senses/bluebubbles/media.js +389 -0
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +45 -16
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
- package/dist/senses/bluebubbles/processed-log.js +133 -0
- package/dist/senses/bluebubbles/replay.js +137 -0
- package/dist/senses/bluebubbles/runtime-state.js +137 -0
- package/dist/senses/bluebubbles/session-cleanup.js +72 -0
- 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 +187 -0
- package/dist/senses/cli.js +777 -264
- package/dist/senses/commands.js +66 -3
- package/dist/senses/continuity.js +94 -0
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +209 -16
- package/dist/senses/inner-dialog.js +682 -91
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +751 -0
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +392 -0
- package/dist/senses/surface-tool.js +70 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +925 -195
- package/dist/senses/trust-gate.js +207 -2
- package/dist/senses/voice/audio-playback.js +237 -0
- package/dist/senses/voice/audio-routing.js +119 -0
- package/dist/senses/voice/elevenlabs.js +202 -0
- package/dist/senses/voice/floor-control.js +431 -0
- package/dist/senses/voice/floor-controller.js +115 -0
- package/dist/senses/voice/golden-path.js +116 -0
- package/dist/senses/voice/index.js +29 -0
- package/dist/senses/voice/meeting.js +113 -0
- package/dist/senses/voice/outbound.js +190 -0
- package/dist/senses/voice/phone.js +33 -0
- package/dist/senses/voice/playback.js +139 -0
- package/dist/senses/voice/realtime-eval.js +496 -0
- package/dist/senses/voice/realtime-trace.js +531 -0
- package/dist/senses/voice/transcript.js +70 -0
- package/dist/senses/voice/turn.js +191 -0
- package/dist/senses/voice/twilio-phone-runtime.js +807 -0
- package/dist/senses/voice/twilio-phone.js +5077 -0
- package/dist/senses/voice/types.js +2 -0
- package/dist/senses/voice/whisper.js +161 -0
- package/dist/senses/voice-entry.js +81 -0
- package/dist/senses/voice-realtime-eval-command.js +99 -0
- package/dist/senses/voice-realtime-eval-entry.js +21 -0
- package/dist/senses/voice-twilio-entry.js +87 -0
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +265 -0
- package/package.json +52 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +99 -0
- package/skills/travel-planning.md +138 -0
- package/AdoptionSpecialist.ouro/agent.json +0 -20
- package/AdoptionSpecialist.ouro/psyche/SOUL.md +0 -22
- package/dist/heart/daemon/specialist-orchestrator.js +0 -160
- package/dist/heart/daemon/specialist-prompt.js +0 -40
- package/dist/heart/daemon/specialist-session.js +0 -142
- package/dist/heart/daemon/specialist-tools.js +0 -128
- package/dist/heart/daemon/subagent-installer.js +0 -125
- package/dist/inner-worker-entry.js +0 -4
- package/dist/mind/associative-recall.js +0 -197
- package/dist/senses/bluebubbles-client.js +0 -279
- package/dist/senses/bluebubbles-entry.js +0 -11
- package/dist/senses/bluebubbles.js +0 -332
- package/subagents/README.md +0 -73
- package/subagents/work-doer.md +0 -233
- package/subagents/work-merger.md +0 -624
- package/subagents/work-planner.md +0 -373
- /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/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
|
@@ -0,0 +1,731 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.DaemonSenseManager = void 0;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const os = __importStar(require("os"));
|
|
39
|
+
const path = __importStar(require("path"));
|
|
40
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
41
|
+
const identity_1 = require("../identity");
|
|
42
|
+
const runtime_credentials_1 = require("../runtime-credentials");
|
|
43
|
+
const provider_credentials_1 = require("../provider-credentials");
|
|
44
|
+
const sense_truth_1 = require("../sense-truth");
|
|
45
|
+
const machine_identity_1 = require("../machine-identity");
|
|
46
|
+
const process_manager_1 = require("./process-manager");
|
|
47
|
+
const http_health_probe_1 = require("./http-health-probe");
|
|
48
|
+
const DEFAULT_TEAMS_PORT = 3978;
|
|
49
|
+
const DEFAULT_BLUEBUBBLES_PORT = 18790;
|
|
50
|
+
const DEFAULT_BLUEBUBBLES_WEBHOOK_PATH = "/bluebubbles-webhook";
|
|
51
|
+
const BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS = 90_000;
|
|
52
|
+
function defaultSenses() {
|
|
53
|
+
return {
|
|
54
|
+
cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
|
|
55
|
+
teams: { ...identity_1.DEFAULT_AGENT_SENSES.teams },
|
|
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
|
+
};
|
|
60
|
+
}
|
|
61
|
+
function readAgentSenses(agentJsonPath) {
|
|
62
|
+
const defaults = defaultSenses();
|
|
63
|
+
let parsed;
|
|
64
|
+
try {
|
|
65
|
+
parsed = JSON.parse(fs.readFileSync(agentJsonPath, "utf-8"));
|
|
66
|
+
}
|
|
67
|
+
catch (error) {
|
|
68
|
+
(0, runtime_1.emitNervesEvent)({
|
|
69
|
+
level: "warn",
|
|
70
|
+
component: "channels",
|
|
71
|
+
event: "channel.daemon_sense_agent_config_fallback",
|
|
72
|
+
message: "using default senses because agent config could not be read",
|
|
73
|
+
meta: {
|
|
74
|
+
path: agentJsonPath,
|
|
75
|
+
reason: error instanceof Error ? error.message : String(error),
|
|
76
|
+
},
|
|
77
|
+
});
|
|
78
|
+
return defaults;
|
|
79
|
+
}
|
|
80
|
+
const rawSenses = parsed.senses;
|
|
81
|
+
if (!rawSenses || typeof rawSenses !== "object" || Array.isArray(rawSenses)) {
|
|
82
|
+
return defaults;
|
|
83
|
+
}
|
|
84
|
+
for (const sense of ["cli", "teams", "bluebubbles", "mail", "voice"]) {
|
|
85
|
+
const rawSense = rawSenses[sense];
|
|
86
|
+
if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
|
|
87
|
+
continue;
|
|
88
|
+
}
|
|
89
|
+
const enabled = rawSense.enabled;
|
|
90
|
+
if (typeof enabled === "boolean") {
|
|
91
|
+
defaults[sense] = { enabled };
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
return defaults;
|
|
95
|
+
}
|
|
96
|
+
function textField(record, key) {
|
|
97
|
+
const value = record?.[key];
|
|
98
|
+
return typeof value === "string" ? value.trim() : "";
|
|
99
|
+
}
|
|
100
|
+
function booleanField(record, key) {
|
|
101
|
+
return record?.[key] === true;
|
|
102
|
+
}
|
|
103
|
+
function numberField(record, key, fallback) {
|
|
104
|
+
const value = record?.[key];
|
|
105
|
+
return typeof value === "number" && Number.isFinite(value) ? value : fallback;
|
|
106
|
+
}
|
|
107
|
+
function compactRuntimeConfigError(agent, error) {
|
|
108
|
+
const compact = error.replace(/\s+/g, " ").trim();
|
|
109
|
+
if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
|
|
110
|
+
return `vault locked; run 'ouro vault unlock --agent ${agent}' if you have the saved secret, or 'ouro vault replace --agent ${agent}' if none was saved`;
|
|
111
|
+
}
|
|
112
|
+
return compact || "unavailable";
|
|
113
|
+
}
|
|
114
|
+
function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
|
|
115
|
+
if (runtimeConfig.ok)
|
|
116
|
+
return "";
|
|
117
|
+
const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
|
|
118
|
+
if (runtimeConfig.reason === "missing")
|
|
119
|
+
return `missing vault ${itemName} (${agent})`;
|
|
120
|
+
return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
|
|
121
|
+
}
|
|
122
|
+
function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
|
|
123
|
+
const base = {
|
|
124
|
+
cli: { configured: true, detail: "local interactive terminal" },
|
|
125
|
+
teams: { configured: false, detail: "not enabled in agent.json" },
|
|
126
|
+
bluebubbles: { configured: false, detail: "not enabled in agent.json" },
|
|
127
|
+
mail: { configured: false, detail: "not enabled in agent.json" },
|
|
128
|
+
voice: { configured: false, detail: "not enabled in agent.json" },
|
|
129
|
+
};
|
|
130
|
+
const payload = runtimeConfig.ok ? runtimeConfig.config : {};
|
|
131
|
+
const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
|
|
132
|
+
const teams = payload.teams;
|
|
133
|
+
const teamsChannel = payload.teamsChannel;
|
|
134
|
+
const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
|
|
135
|
+
const bluebubbles = machinePayload.bluebubbles;
|
|
136
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
137
|
+
const mailroom = payload.mailroom;
|
|
138
|
+
const integrations = payload.integrations;
|
|
139
|
+
const voice = machinePayload.voice;
|
|
140
|
+
if (senses.teams.enabled) {
|
|
141
|
+
const missing = [];
|
|
142
|
+
if (!textField(teams, "clientId"))
|
|
143
|
+
missing.push("teams.clientId");
|
|
144
|
+
if (!textField(teams, "clientSecret"))
|
|
145
|
+
missing.push("teams.clientSecret");
|
|
146
|
+
if (!textField(teams, "tenantId"))
|
|
147
|
+
missing.push("teams.tenantId");
|
|
148
|
+
base.teams = missing.length === 0
|
|
149
|
+
? {
|
|
150
|
+
configured: true,
|
|
151
|
+
detail: `:${numberField(teamsChannel, "port", DEFAULT_TEAMS_PORT)}`,
|
|
152
|
+
}
|
|
153
|
+
: {
|
|
154
|
+
configured: false,
|
|
155
|
+
detail: runtimeConfig.ok
|
|
156
|
+
? `missing ${missing.join("/")}`
|
|
157
|
+
: unavailableDetail,
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
if (senses.bluebubbles.enabled) {
|
|
161
|
+
const missing = [];
|
|
162
|
+
if (!textField(bluebubbles, "serverUrl"))
|
|
163
|
+
missing.push("bluebubbles.serverUrl");
|
|
164
|
+
if (!textField(bluebubbles, "password"))
|
|
165
|
+
missing.push("bluebubbles.password");
|
|
166
|
+
base.bluebubbles = missing.length === 0
|
|
167
|
+
? {
|
|
168
|
+
configured: true,
|
|
169
|
+
detail: `:${numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT)} ${textField(bluebubblesChannel, "webhookPath") || DEFAULT_BLUEBUBBLES_WEBHOOK_PATH}`,
|
|
170
|
+
}
|
|
171
|
+
: {
|
|
172
|
+
configured: false,
|
|
173
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
174
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
175
|
+
? "not attached on this machine"
|
|
176
|
+
: machineRuntimeConfig.ok
|
|
177
|
+
? `missing ${missing.join("/")}`
|
|
178
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
if (senses.mail.enabled) {
|
|
182
|
+
const privateKeys = mailroom?.privateKeys;
|
|
183
|
+
const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
|
|
184
|
+
const mailboxAddress = textField(mailroom, "mailboxAddress");
|
|
185
|
+
const missing = [];
|
|
186
|
+
if (!mailboxAddress)
|
|
187
|
+
missing.push("mailroom.mailboxAddress");
|
|
188
|
+
if (!hasPrivateKeys)
|
|
189
|
+
missing.push("mailroom.privateKeys");
|
|
190
|
+
base.mail = missing.length === 0
|
|
191
|
+
? { configured: true, detail: mailboxAddress }
|
|
192
|
+
: {
|
|
193
|
+
configured: false,
|
|
194
|
+
detail: runtimeConfig.ok
|
|
195
|
+
? `missing ${missing.join("/")}`
|
|
196
|
+
: unavailableDetail,
|
|
197
|
+
};
|
|
198
|
+
}
|
|
199
|
+
if (senses.voice.enabled) {
|
|
200
|
+
const portableVoice = payload.voice;
|
|
201
|
+
const conversationEngine = (textField(voice, "twilioConversationEngine")
|
|
202
|
+
|| textField(voice, "conversationEngine")
|
|
203
|
+
|| textField(portableVoice, "twilioConversationEngine")
|
|
204
|
+
|| textField(portableVoice, "conversationEngine")
|
|
205
|
+
|| "cascade").toLowerCase();
|
|
206
|
+
const twilioTransportMode = (textField(voice, "twilioTransportMode") || "record-play").toLowerCase();
|
|
207
|
+
const twilioPublicUrl = textField(voice, "twilioPublicUrl");
|
|
208
|
+
const hasOpenAIRealtimeKey = !!(textField(portableVoice, "openaiRealtimeApiKey")
|
|
209
|
+
|| textField(integrations, "openaiApiKey")
|
|
210
|
+
|| textField(integrations, "openaiEmbeddingsApiKey"));
|
|
211
|
+
const missing = [];
|
|
212
|
+
/* v8 ignore start -- voice setup missing-field matrix is enforced by the voice runtime resolver; sense-manager only renders human-readable readiness facts @preserve */
|
|
213
|
+
if (conversationEngine === "openai-realtime" || conversationEngine === "openai-sip") {
|
|
214
|
+
if (conversationEngine === "openai-realtime" && twilioTransportMode !== "media-stream") {
|
|
215
|
+
missing.push("voice.twilioTransportMode=media-stream");
|
|
216
|
+
}
|
|
217
|
+
if (!hasOpenAIRealtimeKey) {
|
|
218
|
+
missing.push("voice.openaiRealtimeApiKey");
|
|
219
|
+
}
|
|
220
|
+
if (conversationEngine === "openai-sip") {
|
|
221
|
+
if (!textField(portableVoice, "openaiSipProjectId") && !textField(voice, "openaiSipProjectId")) {
|
|
222
|
+
missing.push("voice.openaiSipProjectId");
|
|
223
|
+
}
|
|
224
|
+
const allowUnsignedWebhooks = booleanField(portableVoice, "openaiSipAllowUnsignedWebhooks")
|
|
225
|
+
|| booleanField(voice, "openaiSipAllowUnsignedWebhooks");
|
|
226
|
+
if (!allowUnsignedWebhooks && !textField(portableVoice, "openaiSipWebhookSecret") && !textField(voice, "openaiSipWebhookSecret")) {
|
|
227
|
+
missing.push("voice.openaiSipWebhookSecret");
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
else {
|
|
232
|
+
if (!textField(integrations, "elevenLabsApiKey"))
|
|
233
|
+
missing.push("integrations.elevenLabsApiKey");
|
|
234
|
+
if (!textField(integrations, "elevenLabsVoiceId") && !textField(portableVoice, "elevenLabsVoiceId")) {
|
|
235
|
+
missing.push("integrations.elevenLabsVoiceId");
|
|
236
|
+
}
|
|
237
|
+
if (!textField(voice, "whisperCliPath"))
|
|
238
|
+
missing.push("voice.whisperCliPath");
|
|
239
|
+
if (!textField(voice, "whisperModelPath"))
|
|
240
|
+
missing.push("voice.whisperModelPath");
|
|
241
|
+
}
|
|
242
|
+
/* v8 ignore stop */
|
|
243
|
+
base.voice = missing.length === 0
|
|
244
|
+
? {
|
|
245
|
+
configured: true,
|
|
246
|
+
/* v8 ignore start -- voice detail copy mirrors runtime resolver modes; resolver tests own the matrix @preserve */
|
|
247
|
+
detail: conversationEngine === "openai-sip"
|
|
248
|
+
? twilioPublicUrl
|
|
249
|
+
? "OpenAI Realtime SIP speech-to-speech; Twilio phone transport attached"
|
|
250
|
+
: "OpenAI Realtime SIP speech-to-speech"
|
|
251
|
+
: conversationEngine === "openai-realtime"
|
|
252
|
+
? twilioPublicUrl
|
|
253
|
+
? "OpenAI Realtime speech-to-speech; Twilio phone transport attached"
|
|
254
|
+
: "OpenAI Realtime speech-to-speech"
|
|
255
|
+
: twilioPublicUrl
|
|
256
|
+
? "local Whisper.cpp STT + ElevenLabs TTS; Twilio phone transport attached"
|
|
257
|
+
: "local Whisper.cpp STT + ElevenLabs TTS",
|
|
258
|
+
/* v8 ignore stop */
|
|
259
|
+
}
|
|
260
|
+
: {
|
|
261
|
+
configured: false,
|
|
262
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
263
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
264
|
+
? "not attached on this machine"
|
|
265
|
+
: runtimeConfig.ok && machineRuntimeConfig.ok
|
|
266
|
+
? `missing ${missing.join("/")}`
|
|
267
|
+
: !runtimeConfig.ok
|
|
268
|
+
? unavailableDetail
|
|
269
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
270
|
+
};
|
|
271
|
+
}
|
|
272
|
+
return base;
|
|
273
|
+
}
|
|
274
|
+
function senseRepairHint(agent, sense) {
|
|
275
|
+
if (sense === "teams") {
|
|
276
|
+
return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
|
|
277
|
+
}
|
|
278
|
+
if (sense === "mail") {
|
|
279
|
+
return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
|
|
280
|
+
}
|
|
281
|
+
if (sense === "voice") {
|
|
282
|
+
return `Agent-runnable: run 'ouro connect voice --agent ${agent}' for config guidance; use voice.twilioConversationEngine=openai-sip with voice.openaiRealtimeApiKey, voice.openaiSipProjectId, and voice.openaiSipWebhookSecret for preferred SIP phone voice; use openai-realtime for Media Streams fallback, or save ElevenLabs and local Whisper.cpp settings for cascade fallback; then run 'ouro up' again.`;
|
|
283
|
+
}
|
|
284
|
+
return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
|
|
285
|
+
}
|
|
286
|
+
function currentMachineId() {
|
|
287
|
+
return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
|
|
288
|
+
}
|
|
289
|
+
function parseSenseSnapshotName(name) {
|
|
290
|
+
const parts = name.split(":");
|
|
291
|
+
if (parts.length !== 2)
|
|
292
|
+
return null;
|
|
293
|
+
const [agent, sense] = parts;
|
|
294
|
+
if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail" && sense !== "voice")
|
|
295
|
+
return null;
|
|
296
|
+
return { agent, sense };
|
|
297
|
+
}
|
|
298
|
+
function runtimeInfoFor(status) {
|
|
299
|
+
if (status === "running")
|
|
300
|
+
return { runtime: "running" };
|
|
301
|
+
return { runtime: "error" };
|
|
302
|
+
}
|
|
303
|
+
function managedSenseEntry(sense) {
|
|
304
|
+
if (sense === "teams")
|
|
305
|
+
return "senses/teams-entry.js";
|
|
306
|
+
if (sense === "bluebubbles")
|
|
307
|
+
return "senses/bluebubbles/entry.js";
|
|
308
|
+
if (sense === "voice")
|
|
309
|
+
return "senses/voice-entry.js";
|
|
310
|
+
return "senses/mail-entry.js";
|
|
311
|
+
}
|
|
312
|
+
function runtimeCredentialBootstrapFor(agent, sense) {
|
|
313
|
+
const runtime = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
314
|
+
const machineId = sense === "bluebubbles" || sense === "voice" ? currentMachineId() : undefined;
|
|
315
|
+
const machine = sense === "bluebubbles" || sense === "voice" ? (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent) : null;
|
|
316
|
+
const providerPool = (0, provider_credentials_1.readProviderCredentialPool)(agent);
|
|
317
|
+
const providerCredentialRecords = providerPool.ok
|
|
318
|
+
? Object.values(providerPool.pool.providers).filter((record) => !!record)
|
|
319
|
+
: [];
|
|
320
|
+
const bootstrap = {
|
|
321
|
+
agentName: agent,
|
|
322
|
+
runtimeConfig: runtime.ok ? runtime.config : undefined,
|
|
323
|
+
machineRuntimeConfig: machine?.ok ? machine.config : undefined,
|
|
324
|
+
machineId,
|
|
325
|
+
providerCredentialRecords: providerCredentialRecords.length > 0 ? providerCredentialRecords : undefined,
|
|
326
|
+
};
|
|
327
|
+
if (!bootstrap.runtimeConfig && !bootstrap.machineRuntimeConfig && !bootstrap.providerCredentialRecords)
|
|
328
|
+
return null;
|
|
329
|
+
return bootstrap;
|
|
330
|
+
}
|
|
331
|
+
function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
332
|
+
if (!lastCheckedAt) {
|
|
333
|
+
return false;
|
|
334
|
+
}
|
|
335
|
+
const checkedAt = Date.parse(lastCheckedAt);
|
|
336
|
+
if (!Number.isFinite(checkedAt)) {
|
|
337
|
+
return false;
|
|
338
|
+
}
|
|
339
|
+
return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
|
|
340
|
+
}
|
|
341
|
+
function readBlueBubblesRuntimeJson(runtimePath) {
|
|
342
|
+
try {
|
|
343
|
+
const raw = fs.readFileSync(runtimePath, "utf-8");
|
|
344
|
+
const parsed = JSON.parse(raw);
|
|
345
|
+
/* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
|
|
346
|
+
return {
|
|
347
|
+
upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
|
|
348
|
+
? parsed.upstreamStatus
|
|
349
|
+
: "unknown",
|
|
350
|
+
detail: typeof parsed.detail === "string" && parsed.detail.trim()
|
|
351
|
+
? parsed.detail
|
|
352
|
+
: "startup health probe pending",
|
|
353
|
+
lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
|
|
354
|
+
proofMethod: typeof parsed.proofMethod === "string" && parsed.proofMethod.trim()
|
|
355
|
+
? parsed.proofMethod
|
|
356
|
+
: undefined,
|
|
357
|
+
pendingRecoveryCount: typeof parsed.pendingRecoveryCount === "number" && Number.isFinite(parsed.pendingRecoveryCount)
|
|
358
|
+
? parsed.pendingRecoveryCount
|
|
359
|
+
: 0,
|
|
360
|
+
failedRecoveryCount: typeof parsed.failedRecoveryCount === "number" && Number.isFinite(parsed.failedRecoveryCount)
|
|
361
|
+
? parsed.failedRecoveryCount
|
|
362
|
+
: 0,
|
|
363
|
+
oldestPendingRecoveryAt: typeof parsed.oldestPendingRecoveryAt === "string" ? parsed.oldestPendingRecoveryAt : undefined,
|
|
364
|
+
oldestPendingRecoveryAgeMs: typeof parsed.oldestPendingRecoveryAgeMs === "number" && Number.isFinite(parsed.oldestPendingRecoveryAgeMs)
|
|
365
|
+
? parsed.oldestPendingRecoveryAgeMs
|
|
366
|
+
: undefined,
|
|
367
|
+
activeTurnCount: typeof parsed.activeTurnCount === "number" && Number.isFinite(parsed.activeTurnCount)
|
|
368
|
+
? parsed.activeTurnCount
|
|
369
|
+
: undefined,
|
|
370
|
+
stalledTurnCount: typeof parsed.stalledTurnCount === "number" && Number.isFinite(parsed.stalledTurnCount)
|
|
371
|
+
? parsed.stalledTurnCount
|
|
372
|
+
: undefined,
|
|
373
|
+
oldestActiveTurnStartedAt: typeof parsed.oldestActiveTurnStartedAt === "string" ? parsed.oldestActiveTurnStartedAt : undefined,
|
|
374
|
+
oldestActiveTurnAgeMs: typeof parsed.oldestActiveTurnAgeMs === "number" && Number.isFinite(parsed.oldestActiveTurnAgeMs)
|
|
375
|
+
? parsed.oldestActiveTurnAgeMs
|
|
376
|
+
: undefined,
|
|
377
|
+
};
|
|
378
|
+
/* v8 ignore stop */
|
|
379
|
+
/* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
|
|
380
|
+
}
|
|
381
|
+
catch {
|
|
382
|
+
return { upstreamStatus: "unknown", detail: "startup health probe pending", pendingRecoveryCount: 0, failedRecoveryCount: 0 };
|
|
383
|
+
}
|
|
384
|
+
/* v8 ignore stop */
|
|
385
|
+
}
|
|
386
|
+
function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
|
|
387
|
+
const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
388
|
+
const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
|
|
389
|
+
if (!fs.existsSync(runtimePath)) {
|
|
390
|
+
return { runtime: snapshot?.runtime };
|
|
391
|
+
}
|
|
392
|
+
const state = readBlueBubblesRuntimeJson(runtimePath);
|
|
393
|
+
const checkedAtMs = state.lastCheckedAt ? Date.parse(state.lastCheckedAt) : Number.NaN;
|
|
394
|
+
const proofFacts = {
|
|
395
|
+
proofMethod: state.proofMethod ?? "bluebubbles.checkHealth",
|
|
396
|
+
lastProofAt: state.lastCheckedAt,
|
|
397
|
+
proofAgeMs: Number.isFinite(checkedAtMs) ? Math.max(0, Date.now() - checkedAtMs) : undefined,
|
|
398
|
+
pendingRecoveryCount: state.pendingRecoveryCount,
|
|
399
|
+
failedRecoveryCount: state.failedRecoveryCount,
|
|
400
|
+
oldestPendingRecoveryAt: state.oldestPendingRecoveryAt,
|
|
401
|
+
oldestPendingRecoveryAgeMs: state.oldestPendingRecoveryAgeMs,
|
|
402
|
+
activeTurnCount: state.activeTurnCount,
|
|
403
|
+
stalledTurnCount: state.stalledTurnCount,
|
|
404
|
+
oldestActiveTurnStartedAt: state.oldestActiveTurnStartedAt,
|
|
405
|
+
oldestActiveTurnAgeMs: state.oldestActiveTurnAgeMs,
|
|
406
|
+
};
|
|
407
|
+
if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
|
|
408
|
+
return {
|
|
409
|
+
runtime: snapshot?.runtime,
|
|
410
|
+
lastFailure: state.lastCheckedAt ? "BlueBubbles proof is stale" : undefined,
|
|
411
|
+
failureLayer: state.lastCheckedAt ? "proof_freshness" : undefined,
|
|
412
|
+
};
|
|
413
|
+
}
|
|
414
|
+
if (snapshot?.runtime !== "running") {
|
|
415
|
+
return {
|
|
416
|
+
runtime: "error",
|
|
417
|
+
detail: "BlueBubbles listener is not running",
|
|
418
|
+
...proofFacts,
|
|
419
|
+
lastFailure: "listener process is not running",
|
|
420
|
+
failureLayer: "listener",
|
|
421
|
+
recoveryAction: "daemon health monitor will restart the BlueBubbles listener when its probe fails",
|
|
422
|
+
};
|
|
423
|
+
}
|
|
424
|
+
if (state.upstreamStatus === "error") {
|
|
425
|
+
return {
|
|
426
|
+
runtime: "error",
|
|
427
|
+
detail: state.detail,
|
|
428
|
+
...proofFacts,
|
|
429
|
+
lastFailure: state.detail,
|
|
430
|
+
failureLayer: "upstream",
|
|
431
|
+
recoveryAction: "verify BlueBubbles server/app auth and local machine attachment, then retry the listener",
|
|
432
|
+
};
|
|
433
|
+
}
|
|
434
|
+
if (state.pendingRecoveryCount > 0) {
|
|
435
|
+
return {
|
|
436
|
+
runtime: "error",
|
|
437
|
+
detail: state.detail,
|
|
438
|
+
...proofFacts,
|
|
439
|
+
lastFailure: state.detail,
|
|
440
|
+
failureLayer: "recovery_queue",
|
|
441
|
+
recoveryAction: "queued recovery will retry; inspect BlueBubbles inbound/recovery sidecar logs if age keeps growing",
|
|
442
|
+
};
|
|
443
|
+
}
|
|
444
|
+
if ((state.stalledTurnCount ?? 0) > 0) {
|
|
445
|
+
return {
|
|
446
|
+
runtime: "error",
|
|
447
|
+
detail: state.detail,
|
|
448
|
+
...proofFacts,
|
|
449
|
+
lastFailure: state.detail,
|
|
450
|
+
failureLayer: "live_turn_stall",
|
|
451
|
+
recoveryAction: "live iMessage turn timeout/watchdog will release the lane and recovery will retry captured messages",
|
|
452
|
+
};
|
|
453
|
+
}
|
|
454
|
+
if (state.upstreamStatus === "ok") {
|
|
455
|
+
return {
|
|
456
|
+
runtime: "running",
|
|
457
|
+
...proofFacts,
|
|
458
|
+
...(state.failedRecoveryCount > 0 ? { detail: state.detail } : {}),
|
|
459
|
+
...(state.failedRecoveryCount > 0 ? {
|
|
460
|
+
lastFailure: state.detail,
|
|
461
|
+
failureLayer: "recovery_quarantine",
|
|
462
|
+
recoveryAction: "inspect quarantined BlueBubbles recovery failures; live transport remains reachable",
|
|
463
|
+
} : {}),
|
|
464
|
+
};
|
|
465
|
+
}
|
|
466
|
+
return { runtime: snapshot?.runtime, ...proofFacts };
|
|
467
|
+
}
|
|
468
|
+
class DaemonSenseManager {
|
|
469
|
+
processManager;
|
|
470
|
+
contexts;
|
|
471
|
+
pendingConfigRefreshes = new Set();
|
|
472
|
+
bundlesRoot;
|
|
473
|
+
constructor(options) {
|
|
474
|
+
const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
475
|
+
this.bundlesRoot = bundlesRoot;
|
|
476
|
+
this.contexts = new Map(options.agents.map((agent) => {
|
|
477
|
+
const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
|
|
478
|
+
const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
479
|
+
return [agent, { senses, facts }];
|
|
480
|
+
}));
|
|
481
|
+
const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
482
|
+
return ["teams", "bluebubbles", "mail", "voice"]
|
|
483
|
+
.filter((sense) => context.senses[sense].enabled)
|
|
484
|
+
.map((sense) => ({
|
|
485
|
+
name: `${agent}:${sense}`,
|
|
486
|
+
agentArg: agent,
|
|
487
|
+
entry: managedSenseEntry(sense),
|
|
488
|
+
channel: sense,
|
|
489
|
+
autoStart: true,
|
|
490
|
+
getRuntimeCredentialBootstrap: () => runtimeCredentialBootstrapFor(agent, sense),
|
|
491
|
+
}));
|
|
492
|
+
});
|
|
493
|
+
this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
|
|
494
|
+
agents: managedSenseAgents,
|
|
495
|
+
configCheck: async (name) => {
|
|
496
|
+
const parsed = parseSenseSnapshotName(name);
|
|
497
|
+
if (!parsed)
|
|
498
|
+
return { ok: true };
|
|
499
|
+
const context = this.contexts.get(parsed.agent);
|
|
500
|
+
if (!context)
|
|
501
|
+
return { ok: true };
|
|
502
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(parsed.agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent));
|
|
503
|
+
const fact = context.facts[parsed.sense];
|
|
504
|
+
if (fact.configured)
|
|
505
|
+
return { ok: true };
|
|
506
|
+
this.scheduleSenseConfigRefresh(name, parsed);
|
|
507
|
+
if (fact.optional) {
|
|
508
|
+
return {
|
|
509
|
+
ok: false,
|
|
510
|
+
skip: true,
|
|
511
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
|
|
512
|
+
};
|
|
513
|
+
}
|
|
514
|
+
return {
|
|
515
|
+
ok: false,
|
|
516
|
+
skip: true,
|
|
517
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
|
|
518
|
+
fix: senseRepairHint(parsed.agent, parsed.sense),
|
|
519
|
+
};
|
|
520
|
+
},
|
|
521
|
+
});
|
|
522
|
+
(0, runtime_1.emitNervesEvent)({
|
|
523
|
+
component: "channels",
|
|
524
|
+
event: "channel.daemon_sense_manager_init",
|
|
525
|
+
message: "initialized daemon sense manager",
|
|
526
|
+
meta: {
|
|
527
|
+
agents: options.agents,
|
|
528
|
+
managedSenseProcesses: managedSenseAgents.map((entry) => entry.name),
|
|
529
|
+
},
|
|
530
|
+
});
|
|
531
|
+
}
|
|
532
|
+
scheduleSenseConfigRefresh(name, parsed) {
|
|
533
|
+
if (this.pendingConfigRefreshes.has(name))
|
|
534
|
+
return;
|
|
535
|
+
this.pendingConfigRefreshes.add(name);
|
|
536
|
+
void this.refreshSenseConfigAndRetry(name, parsed);
|
|
537
|
+
}
|
|
538
|
+
async refreshSenseConfigAndRetry(name, parsed) {
|
|
539
|
+
try {
|
|
540
|
+
const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
|
|
541
|
+
const machineRefreshed = parsed.sense === "bluebubbles" || parsed.sense === "voice"
|
|
542
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
543
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
|
|
544
|
+
const context = this.contexts.get(parsed.agent);
|
|
545
|
+
/* v8 ignore next -- defensive: config refreshes are only scheduled for known agent contexts @preserve */
|
|
546
|
+
if (!context)
|
|
547
|
+
return;
|
|
548
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
|
|
549
|
+
if (!context.facts[parsed.sense].configured)
|
|
550
|
+
return;
|
|
551
|
+
setTimeout(() => {
|
|
552
|
+
void this.processManager.startAgent?.(name).catch((error) => {
|
|
553
|
+
(0, runtime_1.emitNervesEvent)({
|
|
554
|
+
level: "error",
|
|
555
|
+
component: "channels",
|
|
556
|
+
event: "channel.daemon_sense_autostart_error",
|
|
557
|
+
message: "sense autostart failed",
|
|
558
|
+
/* v8 ignore next -- defensive: process manager rejects with Error instances in normal use @preserve */
|
|
559
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
560
|
+
});
|
|
561
|
+
});
|
|
562
|
+
}, 0);
|
|
563
|
+
}
|
|
564
|
+
catch (error) {
|
|
565
|
+
(0, runtime_1.emitNervesEvent)({
|
|
566
|
+
level: "error",
|
|
567
|
+
component: "channels",
|
|
568
|
+
event: "channel.daemon_sense_autostart_error",
|
|
569
|
+
message: "sense config refresh failed",
|
|
570
|
+
/* v8 ignore next -- defensive: runtime credential refresh rejects with Error instances in normal use @preserve */
|
|
571
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
572
|
+
});
|
|
573
|
+
}
|
|
574
|
+
finally {
|
|
575
|
+
this.pendingConfigRefreshes.delete(name);
|
|
576
|
+
}
|
|
577
|
+
}
|
|
578
|
+
async refreshEnabledSenseConfigs() {
|
|
579
|
+
const refreshes = [...this.contexts.entries()].map(async ([agent, context]) => {
|
|
580
|
+
const enabledManagedSenses = ["teams", "bluebubbles", "mail", "voice"]
|
|
581
|
+
.filter((sense) => context.senses[sense].enabled);
|
|
582
|
+
/* v8 ignore next -- periodic refresh work only exists when a managed background sense is enabled @preserve */
|
|
583
|
+
if (enabledManagedSenses.length === 0)
|
|
584
|
+
return;
|
|
585
|
+
/* v8 ignore start -- periodic freshness refresh uses the same runtime readers covered by startup integration tests @preserve */
|
|
586
|
+
const runtimeConfig = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(agent, { preserveCachedOnFailure: true });
|
|
587
|
+
const needsMachineConfig = enabledManagedSenses.some((sense) => sense === "bluebubbles" || sense === "voice");
|
|
588
|
+
const machineRuntimeConfig = needsMachineConfig
|
|
589
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
590
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
591
|
+
/* v8 ignore stop */
|
|
592
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
593
|
+
});
|
|
594
|
+
await Promise.all(refreshes);
|
|
595
|
+
}
|
|
596
|
+
async startAutoStartSenses() {
|
|
597
|
+
await this.refreshEnabledSenseConfigs();
|
|
598
|
+
await this.processManager.startAutoStartAgents();
|
|
599
|
+
}
|
|
600
|
+
triggerAutoStartSenses() {
|
|
601
|
+
if (this.processManager.triggerAutoStartAgents) {
|
|
602
|
+
this.processManager.triggerAutoStartAgents();
|
|
603
|
+
return;
|
|
604
|
+
}
|
|
605
|
+
void this.processManager.startAutoStartAgents().catch((error) => {
|
|
606
|
+
(0, runtime_1.emitNervesEvent)({
|
|
607
|
+
level: "error",
|
|
608
|
+
component: "channels",
|
|
609
|
+
event: "channel.daemon_sense_autostart_error",
|
|
610
|
+
message: "sense autostart failed",
|
|
611
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
612
|
+
});
|
|
613
|
+
});
|
|
614
|
+
}
|
|
615
|
+
async stopAll() {
|
|
616
|
+
await this.processManager.stopAll();
|
|
617
|
+
}
|
|
618
|
+
/* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
|
|
619
|
+
listManagedPids() {
|
|
620
|
+
return this.processManager.listAgentSnapshots()
|
|
621
|
+
.map((s) => s.pid)
|
|
622
|
+
.filter((pid) => pid !== null && pid !== undefined);
|
|
623
|
+
}
|
|
624
|
+
/* v8 ignore stop */
|
|
625
|
+
listHealthProbes() {
|
|
626
|
+
const probes = [];
|
|
627
|
+
for (const [agent, context] of this.contexts.entries()) {
|
|
628
|
+
const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
629
|
+
const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
630
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
631
|
+
if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
|
|
632
|
+
continue;
|
|
633
|
+
}
|
|
634
|
+
const machinePayload = machineRuntimeConfig.config;
|
|
635
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
636
|
+
const port = numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT);
|
|
637
|
+
probes.push({
|
|
638
|
+
...(0, http_health_probe_1.createHttpHealthProbe)(`bluebubbles:${agent}`, port),
|
|
639
|
+
managedName: `${agent}:bluebubbles`,
|
|
640
|
+
});
|
|
641
|
+
}
|
|
642
|
+
return probes;
|
|
643
|
+
}
|
|
644
|
+
async restartSense(managedName) {
|
|
645
|
+
if (this.processManager.restartAgent) {
|
|
646
|
+
await this.processManager.restartAgent(managedName);
|
|
647
|
+
return;
|
|
648
|
+
}
|
|
649
|
+
await this.processManager.startAgent?.(managedName);
|
|
650
|
+
}
|
|
651
|
+
listSenseRows() {
|
|
652
|
+
const runtime = new Map();
|
|
653
|
+
for (const snapshot of this.processManager.listAgentSnapshots()) {
|
|
654
|
+
const parsed = parseSenseSnapshotName(snapshot.name);
|
|
655
|
+
if (!parsed)
|
|
656
|
+
continue;
|
|
657
|
+
const current = runtime.get(parsed.agent) ?? {};
|
|
658
|
+
current[parsed.sense] = runtimeInfoFor(snapshot.status);
|
|
659
|
+
runtime.set(parsed.agent, current);
|
|
660
|
+
}
|
|
661
|
+
const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
662
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
663
|
+
const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
|
|
664
|
+
const runtimeInfo = {
|
|
665
|
+
cli: { configured: true },
|
|
666
|
+
teams: {
|
|
667
|
+
configured: context.facts.teams.configured,
|
|
668
|
+
...(runtime.get(agent)?.teams ?? {}),
|
|
669
|
+
},
|
|
670
|
+
bluebubbles: {
|
|
671
|
+
configured: context.facts.bluebubbles.configured,
|
|
672
|
+
optional: context.facts.bluebubbles.optional,
|
|
673
|
+
...blueBubblesRuntimeFacts,
|
|
674
|
+
},
|
|
675
|
+
mail: {
|
|
676
|
+
configured: context.facts.mail.configured,
|
|
677
|
+
...(runtime.get(agent)?.mail ?? {}),
|
|
678
|
+
},
|
|
679
|
+
voice: {
|
|
680
|
+
configured: context.facts.voice.configured,
|
|
681
|
+
optional: context.facts.voice.optional,
|
|
682
|
+
...(runtime.get(agent)?.voice ?? {}),
|
|
683
|
+
},
|
|
684
|
+
};
|
|
685
|
+
const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
|
|
686
|
+
return inventory.map((entry) => ({
|
|
687
|
+
agent,
|
|
688
|
+
sense: entry.sense,
|
|
689
|
+
label: entry.label,
|
|
690
|
+
enabled: entry.enabled,
|
|
691
|
+
status: entry.status,
|
|
692
|
+
detail: entry.enabled
|
|
693
|
+
? entry.sense === "bluebubbles"
|
|
694
|
+
? blueBubblesRuntimeFacts.detail
|
|
695
|
+
?? context.facts[entry.sense].detail
|
|
696
|
+
: context.facts[entry.sense].detail
|
|
697
|
+
: "not enabled in agent.json",
|
|
698
|
+
...(entry.sense === "bluebubbles" ? {
|
|
699
|
+
proofMethod: blueBubblesRuntimeFacts.proofMethod,
|
|
700
|
+
lastProofAt: blueBubblesRuntimeFacts.lastProofAt,
|
|
701
|
+
proofAgeMs: blueBubblesRuntimeFacts.proofAgeMs,
|
|
702
|
+
lastFailure: blueBubblesRuntimeFacts.lastFailure,
|
|
703
|
+
failureLayer: blueBubblesRuntimeFacts.failureLayer,
|
|
704
|
+
recoveryAction: blueBubblesRuntimeFacts.recoveryAction,
|
|
705
|
+
pendingRecoveryCount: blueBubblesRuntimeFacts.pendingRecoveryCount,
|
|
706
|
+
failedRecoveryCount: blueBubblesRuntimeFacts.failedRecoveryCount,
|
|
707
|
+
oldestPendingRecoveryAt: blueBubblesRuntimeFacts.oldestPendingRecoveryAt,
|
|
708
|
+
oldestPendingRecoveryAgeMs: blueBubblesRuntimeFacts.oldestPendingRecoveryAgeMs,
|
|
709
|
+
activeTurnCount: blueBubblesRuntimeFacts.activeTurnCount,
|
|
710
|
+
stalledTurnCount: blueBubblesRuntimeFacts.stalledTurnCount,
|
|
711
|
+
oldestActiveTurnStartedAt: blueBubblesRuntimeFacts.oldestActiveTurnStartedAt,
|
|
712
|
+
oldestActiveTurnAgeMs: blueBubblesRuntimeFacts.oldestActiveTurnAgeMs,
|
|
713
|
+
} : {}),
|
|
714
|
+
}));
|
|
715
|
+
});
|
|
716
|
+
(0, runtime_1.emitNervesEvent)({
|
|
717
|
+
component: "channels",
|
|
718
|
+
event: "channel.daemon_sense_rows_built",
|
|
719
|
+
message: "built daemon sense status rows",
|
|
720
|
+
meta: {
|
|
721
|
+
rows: rows.map((row) => ({
|
|
722
|
+
agent: row.agent,
|
|
723
|
+
sense: row.sense,
|
|
724
|
+
status: row.status,
|
|
725
|
+
})),
|
|
726
|
+
},
|
|
727
|
+
});
|
|
728
|
+
return rows;
|
|
729
|
+
}
|
|
730
|
+
}
|
|
731
|
+
exports.DaemonSenseManager = DaemonSenseManager;
|