@ouro.bot/cli 0.1.0-alpha.52 → 0.1.0-alpha.521
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 +133 -19
- 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-bootstrap-drift.md +54 -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 +3361 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +140 -0
- package/dist/arc/episodes.js +117 -0
- package/dist/arc/intentions.js +133 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +237 -0
- package/dist/arc/packets.js +193 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +837 -26
- package/dist/heart/agent-entry.js +58 -3
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/adapter.js +2 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +427 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +111 -0
- package/dist/heart/config-registry.js +304 -0
- package/dist/heart/config.js +119 -129
- package/dist/heart/core.js +948 -243
- package/dist/heart/cross-chat-delivery.js +3 -18
- package/dist/heart/daemon/agent-config-check.js +512 -0
- package/dist/heart/daemon/agent-discovery.js +102 -3
- package/dist/heart/daemon/agent-service.js +360 -0
- package/dist/heart/daemon/agentic-repair.js +554 -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 +643 -0
- package/dist/heart/daemon/cli-exec.js +7476 -0
- package/dist/heart/daemon/cli-help.js +493 -0
- package/dist/heart/daemon/cli-parse.js +1557 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +649 -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 -1631
- package/dist/heart/daemon/daemon-entry.js +404 -3
- package/dist/heart/daemon/daemon-health.js +183 -0
- package/dist/heart/daemon/daemon-rollup.js +58 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +190 -12
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +758 -60
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +837 -0
- package/dist/heart/daemon/drift-detection.js +146 -0
- package/dist/heart/daemon/health-monitor.js +92 -1
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
- package/dist/heart/daemon/http-health-probe.js +80 -0
- package/dist/heart/daemon/human-command-screens.js +234 -0
- package/dist/heart/daemon/human-readiness.js +114 -0
- package/dist/heart/daemon/inner-status.js +102 -0
- package/dist/heart/daemon/interactive-repair.js +394 -0
- package/dist/heart/daemon/launchd.js +25 -5
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/message-router.js +2 -2
- package/dist/heart/daemon/os-cron-deps.js +134 -0
- 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 +2 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +73 -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 +259 -37
- 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 +109 -4
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +330 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +162 -17
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +1 -1
- package/dist/heart/habits/habit-migration.js +189 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-runtime-state.js +100 -0
- package/dist/heart/habits/habit-scheduler.js +372 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
- package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
- package/dist/heart/identity.js +205 -66
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +1 -1
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mcp/mcp-server.js +653 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +19 -0
- package/dist/heart/outlook/outlook-http-hooks.js +66 -0
- package/dist/heart/outlook/outlook-http-response.js +7 -0
- package/dist/heart/outlook/outlook-http-routes.js +244 -0
- package/dist/heart/outlook/outlook-http-static.js +103 -0
- package/dist/heart/outlook/outlook-http-transport.js +116 -0
- package/dist/heart/outlook/outlook-http.js +99 -0
- package/dist/heart/outlook/outlook-read.js +31 -0
- package/dist/heart/outlook/outlook-types.js +27 -0
- package/dist/heart/outlook/outlook-view.js +195 -0
- package/dist/heart/outlook/readers/agent-machine.js +382 -0
- package/dist/heart/outlook/readers/continuity-readers.js +336 -0
- package/dist/heart/outlook/readers/mail.js +362 -0
- package/dist/heart/outlook/readers/runtime-readers.js +650 -0
- package/dist/heart/outlook/readers/sessions.js +232 -0
- package/dist/heart/outlook/readers/shared.js +111 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +255 -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-state.js +216 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +139 -52
- package/dist/heart/providers/azure.js +97 -13
- package/dist/heart/providers/error-classification.js +127 -0
- package/dist/heart/providers/github-copilot.js +145 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +26 -8
- package/dist/heart/providers/openai-codex.js +55 -40
- package/dist/heart/runtime-capability-check.js +170 -0
- package/dist/heart/runtime-credentials.js +367 -0
- package/dist/heart/sense-truth.js +11 -4
- package/dist/heart/session-activity.js +43 -22
- package/dist/heart/session-events.js +1149 -0
- package/dist/heart/session-playback-cli-main.js +5 -0
- package/dist/heart/session-playback-cli.js +36 -0
- package/dist/heart/session-playback.js +231 -0
- package/dist/heart/session-stats-cli-main.js +5 -0
- package/dist/heart/session-stats.js +182 -0
- package/dist/heart/session-transcript.js +243 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +44 -27
- package/dist/heart/sync-classification.js +176 -0
- package/dist/heart/sync.js +449 -0
- package/dist/heart/target-resolution.js +9 -5
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/timeouts.js +101 -0
- package/dist/heart/tool-activity-callbacks.js +36 -0
- package/dist/heart/tool-description.js +139 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +381 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +425 -0
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +5 -1
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +606 -0
- package/dist/mailroom/body-cache.js +61 -0
- package/dist/mailroom/core.js +672 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +426 -0
- package/dist/mailroom/mbox-import.js +382 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +228 -0
- package/dist/mailroom/search-cache.js +182 -0
- package/dist/mailroom/search-relevance.js +319 -0
- package/dist/mailroom/smtp-ingress.js +176 -0
- package/dist/mailroom/source-state.js +176 -0
- package/dist/mailroom/thread.js +109 -0
- package/dist/mailroom/travel-extract.js +89 -0
- package/dist/mind/bundle-manifest.js +7 -1
- package/dist/mind/context.js +165 -101
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +74 -93
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/friends/channel.js +30 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +39 -3
- package/dist/mind/friends/types.js +2 -2
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +4 -0
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +942 -122
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +139 -5
- package/dist/nerves/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/outlook-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
- package/dist/outlook-ui/index.html +15 -0
- package/dist/repertoire/ado-client.js +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +816 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +180 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +158 -9
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +170 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +111 -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 +255 -0
- package/dist/repertoire/mcp-manager.js +305 -0
- package/dist/repertoire/mcp-tools.js +63 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +15 -24
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +31 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +16 -4
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -78
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-base.js +47 -1075
- package/dist/repertoire/tools-bluebubbles.js +1 -0
- package/dist/repertoire/tools-bridge.js +142 -0
- package/dist/repertoire/tools-bundle.js +984 -0
- package/dist/repertoire/tools-config.js +185 -0
- package/dist/repertoire/tools-continuity.js +248 -0
- package/dist/repertoire/tools-credential.js +381 -0
- package/dist/repertoire/tools-files.js +342 -0
- package/dist/repertoire/tools-flight.js +224 -0
- package/dist/repertoire/tools-flow.js +119 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1477 -0
- package/dist/repertoire/tools-notes.js +421 -0
- package/dist/repertoire/tools-session.js +750 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +243 -0
- package/dist/repertoire/tools-teams.js +9 -39
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +422 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools.js +108 -100
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +131 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +561 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/bluebubbles/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 +1947 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
- package/dist/senses/bluebubbles/processed-log.js +111 -0
- package/dist/senses/bluebubbles/replay.js +129 -0
- package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +2 -2
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +607 -0
- package/dist/senses/cli/spinner-imperative.js +135 -0
- package/dist/senses/cli/spinner.js +101 -0
- package/dist/senses/cli/status-line.js +60 -0
- package/dist/senses/cli/streaming-markdown.js +526 -0
- package/dist/senses/cli/tool-display.js +85 -0
- package/dist/senses/cli/tool-render.js +85 -0
- package/dist/senses/cli/tui-store.js +240 -0
- package/dist/senses/cli/virtual-list.js +35 -0
- package/dist/senses/cli-entry.js +60 -8
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +520 -209
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +175 -21
- package/dist/senses/inner-dialog.js +330 -27
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +569 -182
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +248 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +387 -98
- package/dist/senses/trust-gate.js +100 -5
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +146 -0
- package/package.json +38 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +101 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/ouro-path-installer.js +0 -178
- package/dist/heart/daemon/subagent-installer.js +0 -166
- package/dist/heart/session-recall.js +0 -116
- package/dist/mind/associative-recall.js +0 -209
- package/dist/senses/bluebubbles-entry.js +0 -13
- package/dist/senses/bluebubbles.js +0 -1177
- package/dist/senses/debug-activity.js +0 -148
- package/subagents/README.md +0 -86
- package/subagents/work-doer.md +0 -237
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -390
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
- /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
|
@@ -0,0 +1,161 @@
|
|
|
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.SAFE_MODE_OVERRIDE_FILENAME = void 0;
|
|
37
|
+
exports.detectSafeMode = detectSafeMode;
|
|
38
|
+
exports.pruneOldCrashes = pruneOldCrashes;
|
|
39
|
+
const fs = __importStar(require("fs"));
|
|
40
|
+
const path = __importStar(require("path"));
|
|
41
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
42
|
+
exports.SAFE_MODE_OVERRIDE_FILENAME = "safe-mode-override.json";
|
|
43
|
+
/** 3+ crashes within this window triggers safe mode */
|
|
44
|
+
const CRASH_WINDOW_MS = 5 * 60 * 1000; // 5 minutes
|
|
45
|
+
const CRASH_THRESHOLD = 3;
|
|
46
|
+
/**
|
|
47
|
+
* Reads crash history from the tombstone file and determines if safe mode should be active.
|
|
48
|
+
* Returns active if 3+ crashes occurred within 5 minutes.
|
|
49
|
+
*
|
|
50
|
+
* Safe mode is bypassed when:
|
|
51
|
+
* - devMode is true
|
|
52
|
+
* - A safe-mode override file exists (written by `ouro up --force`)
|
|
53
|
+
*/
|
|
54
|
+
function detectSafeMode(tombstonePath, options) {
|
|
55
|
+
const inactive = { active: false, reason: "", enteredAt: "" };
|
|
56
|
+
// Dev mode: never enter safe mode
|
|
57
|
+
if (options?.devMode) {
|
|
58
|
+
return inactive;
|
|
59
|
+
}
|
|
60
|
+
// Check for override file (--force)
|
|
61
|
+
const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
|
|
62
|
+
try {
|
|
63
|
+
if (fs.existsSync(overridePath)) {
|
|
64
|
+
return inactive;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
catch {
|
|
68
|
+
// Best-effort check
|
|
69
|
+
}
|
|
70
|
+
// Read tombstone
|
|
71
|
+
let parsed;
|
|
72
|
+
try {
|
|
73
|
+
const raw = fs.readFileSync(tombstonePath, "utf-8");
|
|
74
|
+
parsed = JSON.parse(raw);
|
|
75
|
+
}
|
|
76
|
+
catch {
|
|
77
|
+
return inactive;
|
|
78
|
+
}
|
|
79
|
+
// Extract recentCrashes
|
|
80
|
+
if (!Array.isArray(parsed.recentCrashes)) {
|
|
81
|
+
return inactive;
|
|
82
|
+
}
|
|
83
|
+
const nowMs = options?.now ? options.now() : Date.now();
|
|
84
|
+
const windowStart = nowMs - CRASH_WINDOW_MS;
|
|
85
|
+
// Filter to valid string timestamps within the crash window
|
|
86
|
+
const recentInWindow = parsed.recentCrashes.filter((entry) => {
|
|
87
|
+
if (typeof entry !== "string")
|
|
88
|
+
return false;
|
|
89
|
+
const ts = new Date(entry).getTime();
|
|
90
|
+
if (isNaN(ts))
|
|
91
|
+
return false;
|
|
92
|
+
return ts >= windowStart;
|
|
93
|
+
});
|
|
94
|
+
if (recentInWindow.length < CRASH_THRESHOLD) {
|
|
95
|
+
return inactive;
|
|
96
|
+
}
|
|
97
|
+
const result = {
|
|
98
|
+
active: true,
|
|
99
|
+
reason: `crash loop detected: ${recentInWindow.length} crashes in last 5 minutes`,
|
|
100
|
+
enteredAt: new Date(nowMs).toISOString(),
|
|
101
|
+
};
|
|
102
|
+
(0, runtime_1.emitNervesEvent)({
|
|
103
|
+
level: "error",
|
|
104
|
+
component: "daemon",
|
|
105
|
+
event: "daemon.safe_mode_entered",
|
|
106
|
+
message: result.reason,
|
|
107
|
+
meta: {
|
|
108
|
+
crashCount: recentInWindow.length,
|
|
109
|
+
windowMs: CRASH_WINDOW_MS,
|
|
110
|
+
tombstonePath,
|
|
111
|
+
},
|
|
112
|
+
});
|
|
113
|
+
return result;
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Prunes crash entries older than 5 minutes from the tombstone's recentCrashes.
|
|
117
|
+
* Also removes the safe-mode override file if present.
|
|
118
|
+
* Called after successful startup (uptime > stability threshold).
|
|
119
|
+
*/
|
|
120
|
+
function pruneOldCrashes(tombstonePath, options) {
|
|
121
|
+
// Remove override file
|
|
122
|
+
const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
|
|
123
|
+
try {
|
|
124
|
+
if (fs.existsSync(overridePath)) {
|
|
125
|
+
fs.unlinkSync(overridePath);
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
catch {
|
|
129
|
+
// Best-effort
|
|
130
|
+
}
|
|
131
|
+
// Read existing tombstone
|
|
132
|
+
let parsed;
|
|
133
|
+
try {
|
|
134
|
+
const raw = fs.readFileSync(tombstonePath, "utf-8");
|
|
135
|
+
parsed = JSON.parse(raw);
|
|
136
|
+
}
|
|
137
|
+
catch {
|
|
138
|
+
return;
|
|
139
|
+
}
|
|
140
|
+
if (!Array.isArray(parsed.recentCrashes)) {
|
|
141
|
+
return;
|
|
142
|
+
}
|
|
143
|
+
const nowMs = options?.now ? options.now() : Date.now();
|
|
144
|
+
const windowStart = nowMs - CRASH_WINDOW_MS;
|
|
145
|
+
// Keep only entries within the window
|
|
146
|
+
const pruned = parsed.recentCrashes.filter((entry) => {
|
|
147
|
+
if (typeof entry !== "string")
|
|
148
|
+
return false;
|
|
149
|
+
const ts = new Date(entry).getTime();
|
|
150
|
+
if (isNaN(ts))
|
|
151
|
+
return false;
|
|
152
|
+
return ts >= windowStart;
|
|
153
|
+
});
|
|
154
|
+
parsed.recentCrashes = pruned;
|
|
155
|
+
try {
|
|
156
|
+
fs.writeFileSync(tombstonePath, JSON.stringify(parsed, null, 2) + "\n", "utf-8");
|
|
157
|
+
}
|
|
158
|
+
catch {
|
|
159
|
+
// Best-effort
|
|
160
|
+
}
|
|
161
|
+
}
|
|
@@ -38,18 +38,23 @@ const fs = __importStar(require("fs"));
|
|
|
38
38
|
const os = __importStar(require("os"));
|
|
39
39
|
const path = __importStar(require("path"));
|
|
40
40
|
const runtime_1 = require("../../nerves/runtime");
|
|
41
|
-
const bluebubbles_runtime_state_1 = require("../../senses/bluebubbles-runtime-state");
|
|
42
41
|
const identity_1 = require("../identity");
|
|
42
|
+
const runtime_credentials_1 = require("../runtime-credentials");
|
|
43
|
+
const provider_credentials_1 = require("../provider-credentials");
|
|
43
44
|
const sense_truth_1 = require("../sense-truth");
|
|
45
|
+
const machine_identity_1 = require("../machine-identity");
|
|
44
46
|
const process_manager_1 = require("./process-manager");
|
|
47
|
+
const http_health_probe_1 = require("./http-health-probe");
|
|
45
48
|
const DEFAULT_TEAMS_PORT = 3978;
|
|
46
49
|
const DEFAULT_BLUEBUBBLES_PORT = 18790;
|
|
47
50
|
const DEFAULT_BLUEBUBBLES_WEBHOOK_PATH = "/bluebubbles-webhook";
|
|
51
|
+
const BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS = 90_000;
|
|
48
52
|
function defaultSenses() {
|
|
49
53
|
return {
|
|
50
54
|
cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
|
|
51
55
|
teams: { ...identity_1.DEFAULT_AGENT_SENSES.teams },
|
|
52
56
|
bluebubbles: { ...identity_1.DEFAULT_AGENT_SENSES.bluebubbles },
|
|
57
|
+
mail: { ...identity_1.DEFAULT_AGENT_SENSES.mail },
|
|
53
58
|
};
|
|
54
59
|
}
|
|
55
60
|
function readAgentSenses(agentJsonPath) {
|
|
@@ -75,7 +80,7 @@ function readAgentSenses(agentJsonPath) {
|
|
|
75
80
|
if (!rawSenses || typeof rawSenses !== "object" || Array.isArray(rawSenses)) {
|
|
76
81
|
return defaults;
|
|
77
82
|
}
|
|
78
|
-
for (const sense of ["cli", "teams", "bluebubbles"]) {
|
|
83
|
+
for (const sense of ["cli", "teams", "bluebubbles", "mail"]) {
|
|
79
84
|
const rawSense = rawSenses[sense];
|
|
80
85
|
if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
|
|
81
86
|
continue;
|
|
@@ -87,22 +92,6 @@ function readAgentSenses(agentJsonPath) {
|
|
|
87
92
|
}
|
|
88
93
|
return defaults;
|
|
89
94
|
}
|
|
90
|
-
function readSecretsPayload(secretsPath) {
|
|
91
|
-
try {
|
|
92
|
-
const raw = fs.readFileSync(secretsPath, "utf-8");
|
|
93
|
-
const parsed = JSON.parse(raw);
|
|
94
|
-
if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
|
|
95
|
-
return { payload: {}, error: "invalid secrets.json object" };
|
|
96
|
-
}
|
|
97
|
-
return { payload: parsed, error: null };
|
|
98
|
-
}
|
|
99
|
-
catch (error) {
|
|
100
|
-
return {
|
|
101
|
-
payload: {},
|
|
102
|
-
error: error instanceof Error ? error.message : String(error),
|
|
103
|
-
};
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
95
|
function textField(record, key) {
|
|
107
96
|
const value = record?.[key];
|
|
108
97
|
return typeof value === "string" ? value.trim() : "";
|
|
@@ -111,17 +100,36 @@ function numberField(record, key, fallback) {
|
|
|
111
100
|
const value = record?.[key];
|
|
112
101
|
return typeof value === "number" && Number.isFinite(value) ? value : fallback;
|
|
113
102
|
}
|
|
114
|
-
function
|
|
103
|
+
function compactRuntimeConfigError(agent, error) {
|
|
104
|
+
const compact = error.replace(/\s+/g, " ").trim();
|
|
105
|
+
if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
|
|
106
|
+
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`;
|
|
107
|
+
}
|
|
108
|
+
return compact || "unavailable";
|
|
109
|
+
}
|
|
110
|
+
function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
|
|
111
|
+
if (runtimeConfig.ok)
|
|
112
|
+
return "";
|
|
113
|
+
const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
|
|
114
|
+
if (runtimeConfig.reason === "missing")
|
|
115
|
+
return `missing vault ${itemName} (${agent})`;
|
|
116
|
+
return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
|
|
117
|
+
}
|
|
118
|
+
function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
|
|
115
119
|
const base = {
|
|
116
120
|
cli: { configured: true, detail: "local interactive terminal" },
|
|
117
121
|
teams: { configured: false, detail: "not enabled in agent.json" },
|
|
118
122
|
bluebubbles: { configured: false, detail: "not enabled in agent.json" },
|
|
123
|
+
mail: { configured: false, detail: "not enabled in agent.json" },
|
|
119
124
|
};
|
|
120
|
-
const
|
|
125
|
+
const payload = runtimeConfig.ok ? runtimeConfig.config : {};
|
|
126
|
+
const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
|
|
121
127
|
const teams = payload.teams;
|
|
122
128
|
const teamsChannel = payload.teamsChannel;
|
|
123
|
-
const
|
|
124
|
-
const
|
|
129
|
+
const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
|
|
130
|
+
const bluebubbles = machinePayload.bluebubbles;
|
|
131
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
132
|
+
const mailroom = payload.mailroom;
|
|
125
133
|
if (senses.teams.enabled) {
|
|
126
134
|
const missing = [];
|
|
127
135
|
if (!textField(teams, "clientId"))
|
|
@@ -137,9 +145,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
137
145
|
}
|
|
138
146
|
: {
|
|
139
147
|
configured: false,
|
|
140
|
-
detail:
|
|
141
|
-
? `missing
|
|
142
|
-
:
|
|
148
|
+
detail: runtimeConfig.ok
|
|
149
|
+
? `missing ${missing.join("/")}`
|
|
150
|
+
: unavailableDetail,
|
|
143
151
|
};
|
|
144
152
|
}
|
|
145
153
|
if (senses.bluebubbles.enabled) {
|
|
@@ -155,19 +163,52 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
155
163
|
}
|
|
156
164
|
: {
|
|
157
165
|
configured: false,
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
166
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
167
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
168
|
+
? "not attached on this machine"
|
|
169
|
+
: machineRuntimeConfig.ok
|
|
170
|
+
? `missing ${missing.join("/")}`
|
|
171
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
172
|
+
};
|
|
173
|
+
}
|
|
174
|
+
if (senses.mail.enabled) {
|
|
175
|
+
const privateKeys = mailroom?.privateKeys;
|
|
176
|
+
const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
|
|
177
|
+
const mailboxAddress = textField(mailroom, "mailboxAddress");
|
|
178
|
+
const missing = [];
|
|
179
|
+
if (!mailboxAddress)
|
|
180
|
+
missing.push("mailroom.mailboxAddress");
|
|
181
|
+
if (!hasPrivateKeys)
|
|
182
|
+
missing.push("mailroom.privateKeys");
|
|
183
|
+
base.mail = missing.length === 0
|
|
184
|
+
? { configured: true, detail: mailboxAddress }
|
|
185
|
+
: {
|
|
186
|
+
configured: false,
|
|
187
|
+
detail: runtimeConfig.ok
|
|
188
|
+
? `missing ${missing.join("/")}`
|
|
189
|
+
: unavailableDetail,
|
|
161
190
|
};
|
|
162
191
|
}
|
|
163
192
|
return base;
|
|
164
193
|
}
|
|
194
|
+
function senseRepairHint(agent, sense) {
|
|
195
|
+
if (sense === "teams") {
|
|
196
|
+
return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
|
|
197
|
+
}
|
|
198
|
+
if (sense === "mail") {
|
|
199
|
+
return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
|
|
200
|
+
}
|
|
201
|
+
return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
|
|
202
|
+
}
|
|
203
|
+
function currentMachineId() {
|
|
204
|
+
return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
|
|
205
|
+
}
|
|
165
206
|
function parseSenseSnapshotName(name) {
|
|
166
207
|
const parts = name.split(":");
|
|
167
208
|
if (parts.length !== 2)
|
|
168
209
|
return null;
|
|
169
210
|
const [agent, sense] = parts;
|
|
170
|
-
if (sense !== "teams" && sense !== "bluebubbles")
|
|
211
|
+
if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail")
|
|
171
212
|
return null;
|
|
172
213
|
return { agent, sense };
|
|
173
214
|
}
|
|
@@ -176,47 +217,138 @@ function runtimeInfoFor(status) {
|
|
|
176
217
|
return { runtime: "running" };
|
|
177
218
|
return { runtime: "error" };
|
|
178
219
|
}
|
|
220
|
+
function managedSenseEntry(sense) {
|
|
221
|
+
if (sense === "teams")
|
|
222
|
+
return "senses/teams-entry.js";
|
|
223
|
+
if (sense === "bluebubbles")
|
|
224
|
+
return "senses/bluebubbles/entry.js";
|
|
225
|
+
return "senses/mail-entry.js";
|
|
226
|
+
}
|
|
227
|
+
function runtimeCredentialBootstrapFor(agent, sense) {
|
|
228
|
+
const runtime = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
229
|
+
const machineId = sense === "bluebubbles" ? currentMachineId() : undefined;
|
|
230
|
+
const machine = sense === "bluebubbles" ? (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent) : null;
|
|
231
|
+
const providerPool = (0, provider_credentials_1.readProviderCredentialPool)(agent);
|
|
232
|
+
const providerCredentialRecords = providerPool.ok
|
|
233
|
+
? Object.values(providerPool.pool.providers).filter((record) => !!record)
|
|
234
|
+
: [];
|
|
235
|
+
const bootstrap = {
|
|
236
|
+
agentName: agent,
|
|
237
|
+
runtimeConfig: runtime.ok ? runtime.config : undefined,
|
|
238
|
+
machineRuntimeConfig: machine?.ok ? machine.config : undefined,
|
|
239
|
+
machineId,
|
|
240
|
+
providerCredentialRecords: providerCredentialRecords.length > 0 ? providerCredentialRecords : undefined,
|
|
241
|
+
};
|
|
242
|
+
if (!bootstrap.runtimeConfig && !bootstrap.machineRuntimeConfig && !bootstrap.providerCredentialRecords)
|
|
243
|
+
return null;
|
|
244
|
+
return bootstrap;
|
|
245
|
+
}
|
|
246
|
+
function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
247
|
+
if (!lastCheckedAt) {
|
|
248
|
+
return false;
|
|
249
|
+
}
|
|
250
|
+
const checkedAt = Date.parse(lastCheckedAt);
|
|
251
|
+
if (!Number.isFinite(checkedAt)) {
|
|
252
|
+
return false;
|
|
253
|
+
}
|
|
254
|
+
return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
|
|
255
|
+
}
|
|
256
|
+
function readBlueBubblesRuntimeJson(runtimePath) {
|
|
257
|
+
try {
|
|
258
|
+
const raw = fs.readFileSync(runtimePath, "utf-8");
|
|
259
|
+
const parsed = JSON.parse(raw);
|
|
260
|
+
/* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
|
|
261
|
+
return {
|
|
262
|
+
upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
|
|
263
|
+
? parsed.upstreamStatus
|
|
264
|
+
: "unknown",
|
|
265
|
+
detail: typeof parsed.detail === "string" && parsed.detail.trim()
|
|
266
|
+
? parsed.detail
|
|
267
|
+
: "startup health probe pending",
|
|
268
|
+
lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
|
|
269
|
+
};
|
|
270
|
+
/* v8 ignore stop */
|
|
271
|
+
/* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
|
|
272
|
+
}
|
|
273
|
+
catch {
|
|
274
|
+
return { upstreamStatus: "unknown", detail: "startup health probe pending" };
|
|
275
|
+
}
|
|
276
|
+
/* v8 ignore stop */
|
|
277
|
+
}
|
|
179
278
|
function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
|
|
180
279
|
const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
181
280
|
const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
|
|
182
|
-
if (
|
|
281
|
+
if (!fs.existsSync(runtimePath)) {
|
|
282
|
+
return { runtime: snapshot?.runtime };
|
|
283
|
+
}
|
|
284
|
+
const state = readBlueBubblesRuntimeJson(runtimePath);
|
|
285
|
+
if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
|
|
183
286
|
return { runtime: snapshot?.runtime };
|
|
184
287
|
}
|
|
185
|
-
const state = (0, bluebubbles_runtime_state_1.readBlueBubblesRuntimeState)(agent, agentRoot);
|
|
186
288
|
if (state.upstreamStatus === "error") {
|
|
187
289
|
return {
|
|
188
290
|
runtime: "error",
|
|
189
291
|
detail: state.detail,
|
|
190
292
|
};
|
|
191
293
|
}
|
|
192
|
-
|
|
294
|
+
if (state.upstreamStatus === "ok") {
|
|
295
|
+
return { runtime: "running" };
|
|
296
|
+
}
|
|
297
|
+
return { runtime: snapshot?.runtime };
|
|
193
298
|
}
|
|
194
299
|
class DaemonSenseManager {
|
|
195
300
|
processManager;
|
|
196
301
|
contexts;
|
|
302
|
+
pendingConfigRefreshes = new Set();
|
|
197
303
|
bundlesRoot;
|
|
198
304
|
constructor(options) {
|
|
199
305
|
const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
200
|
-
const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
|
|
201
306
|
this.bundlesRoot = bundlesRoot;
|
|
202
307
|
this.contexts = new Map(options.agents.map((agent) => {
|
|
203
308
|
const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
|
|
204
|
-
const facts =
|
|
309
|
+
const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
205
310
|
return [agent, { senses, facts }];
|
|
206
311
|
}));
|
|
207
312
|
const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
208
|
-
return ["teams", "bluebubbles"]
|
|
209
|
-
.filter((sense) => context.senses[sense].enabled
|
|
313
|
+
return ["teams", "bluebubbles", "mail"]
|
|
314
|
+
.filter((sense) => context.senses[sense].enabled)
|
|
210
315
|
.map((sense) => ({
|
|
211
316
|
name: `${agent}:${sense}`,
|
|
212
317
|
agentArg: agent,
|
|
213
|
-
entry: sense
|
|
318
|
+
entry: managedSenseEntry(sense),
|
|
214
319
|
channel: sense,
|
|
215
320
|
autoStart: true,
|
|
321
|
+
getRuntimeCredentialBootstrap: () => runtimeCredentialBootstrapFor(agent, sense),
|
|
216
322
|
}));
|
|
217
323
|
});
|
|
218
324
|
this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
|
|
219
325
|
agents: managedSenseAgents,
|
|
326
|
+
configCheck: async (name) => {
|
|
327
|
+
const parsed = parseSenseSnapshotName(name);
|
|
328
|
+
if (!parsed)
|
|
329
|
+
return { ok: true };
|
|
330
|
+
const context = this.contexts.get(parsed.agent);
|
|
331
|
+
if (!context)
|
|
332
|
+
return { ok: true };
|
|
333
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(parsed.agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent));
|
|
334
|
+
const fact = context.facts[parsed.sense];
|
|
335
|
+
if (fact.configured)
|
|
336
|
+
return { ok: true };
|
|
337
|
+
this.scheduleSenseConfigRefresh(name, parsed);
|
|
338
|
+
if (fact.optional) {
|
|
339
|
+
return {
|
|
340
|
+
ok: false,
|
|
341
|
+
skip: true,
|
|
342
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
|
|
343
|
+
};
|
|
344
|
+
}
|
|
345
|
+
return {
|
|
346
|
+
ok: false,
|
|
347
|
+
skip: true,
|
|
348
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
|
|
349
|
+
fix: senseRepairHint(parsed.agent, parsed.sense),
|
|
350
|
+
};
|
|
351
|
+
},
|
|
220
352
|
});
|
|
221
353
|
(0, runtime_1.emitNervesEvent)({
|
|
222
354
|
component: "channels",
|
|
@@ -228,12 +360,96 @@ class DaemonSenseManager {
|
|
|
228
360
|
},
|
|
229
361
|
});
|
|
230
362
|
}
|
|
363
|
+
scheduleSenseConfigRefresh(name, parsed) {
|
|
364
|
+
if (this.pendingConfigRefreshes.has(name))
|
|
365
|
+
return;
|
|
366
|
+
this.pendingConfigRefreshes.add(name);
|
|
367
|
+
void this.refreshSenseConfigAndRetry(name, parsed);
|
|
368
|
+
}
|
|
369
|
+
async refreshSenseConfigAndRetry(name, parsed) {
|
|
370
|
+
try {
|
|
371
|
+
const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
|
|
372
|
+
const machineRefreshed = parsed.sense === "bluebubbles"
|
|
373
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
374
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
|
|
375
|
+
const context = this.contexts.get(parsed.agent);
|
|
376
|
+
/* v8 ignore next -- defensive: config refreshes are only scheduled for known agent contexts @preserve */
|
|
377
|
+
if (!context)
|
|
378
|
+
return;
|
|
379
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
|
|
380
|
+
if (!context.facts[parsed.sense].configured)
|
|
381
|
+
return;
|
|
382
|
+
setTimeout(() => {
|
|
383
|
+
void this.processManager.startAgent?.(name).catch((error) => {
|
|
384
|
+
(0, runtime_1.emitNervesEvent)({
|
|
385
|
+
level: "error",
|
|
386
|
+
component: "channels",
|
|
387
|
+
event: "channel.daemon_sense_autostart_error",
|
|
388
|
+
message: "sense autostart failed",
|
|
389
|
+
/* v8 ignore next -- defensive: process manager rejects with Error instances in normal use @preserve */
|
|
390
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
391
|
+
});
|
|
392
|
+
});
|
|
393
|
+
}, 0);
|
|
394
|
+
}
|
|
395
|
+
catch (error) {
|
|
396
|
+
(0, runtime_1.emitNervesEvent)({
|
|
397
|
+
level: "error",
|
|
398
|
+
component: "channels",
|
|
399
|
+
event: "channel.daemon_sense_autostart_error",
|
|
400
|
+
message: "sense config refresh failed",
|
|
401
|
+
/* v8 ignore next -- defensive: runtime credential refresh rejects with Error instances in normal use @preserve */
|
|
402
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
403
|
+
});
|
|
404
|
+
}
|
|
405
|
+
finally {
|
|
406
|
+
this.pendingConfigRefreshes.delete(name);
|
|
407
|
+
}
|
|
408
|
+
}
|
|
231
409
|
async startAutoStartSenses() {
|
|
232
410
|
await this.processManager.startAutoStartAgents();
|
|
233
411
|
}
|
|
412
|
+
triggerAutoStartSenses() {
|
|
413
|
+
if (this.processManager.triggerAutoStartAgents) {
|
|
414
|
+
this.processManager.triggerAutoStartAgents();
|
|
415
|
+
return;
|
|
416
|
+
}
|
|
417
|
+
void this.processManager.startAutoStartAgents().catch((error) => {
|
|
418
|
+
(0, runtime_1.emitNervesEvent)({
|
|
419
|
+
level: "error",
|
|
420
|
+
component: "channels",
|
|
421
|
+
event: "channel.daemon_sense_autostart_error",
|
|
422
|
+
message: "sense autostart failed",
|
|
423
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
424
|
+
});
|
|
425
|
+
});
|
|
426
|
+
}
|
|
234
427
|
async stopAll() {
|
|
235
428
|
await this.processManager.stopAll();
|
|
236
429
|
}
|
|
430
|
+
/* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
|
|
431
|
+
listManagedPids() {
|
|
432
|
+
return this.processManager.listAgentSnapshots()
|
|
433
|
+
.map((s) => s.pid)
|
|
434
|
+
.filter((pid) => pid !== null && pid !== undefined);
|
|
435
|
+
}
|
|
436
|
+
/* v8 ignore stop */
|
|
437
|
+
listHealthProbes() {
|
|
438
|
+
const probes = [];
|
|
439
|
+
for (const [agent, context] of this.contexts.entries()) {
|
|
440
|
+
const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
|
|
441
|
+
const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
|
|
442
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
|
|
443
|
+
if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
|
|
444
|
+
continue;
|
|
445
|
+
}
|
|
446
|
+
const machinePayload = machineRuntimeConfig.config;
|
|
447
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
448
|
+
const port = numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT);
|
|
449
|
+
probes.push((0, http_health_probe_1.createHttpHealthProbe)(`bluebubbles:${agent}`, port));
|
|
450
|
+
}
|
|
451
|
+
return probes;
|
|
452
|
+
}
|
|
237
453
|
listSenseRows() {
|
|
238
454
|
const runtime = new Map();
|
|
239
455
|
for (const snapshot of this.processManager.listAgentSnapshots()) {
|
|
@@ -245,6 +461,7 @@ class DaemonSenseManager {
|
|
|
245
461
|
runtime.set(parsed.agent, current);
|
|
246
462
|
}
|
|
247
463
|
const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
464
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
248
465
|
const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
|
|
249
466
|
const runtimeInfo = {
|
|
250
467
|
cli: { configured: true },
|
|
@@ -254,8 +471,13 @@ class DaemonSenseManager {
|
|
|
254
471
|
},
|
|
255
472
|
bluebubbles: {
|
|
256
473
|
configured: context.facts.bluebubbles.configured,
|
|
474
|
+
optional: context.facts.bluebubbles.optional,
|
|
257
475
|
...blueBubblesRuntimeFacts,
|
|
258
476
|
},
|
|
477
|
+
mail: {
|
|
478
|
+
configured: context.facts.mail.configured,
|
|
479
|
+
...(runtime.get(agent)?.mail ?? {}),
|
|
480
|
+
},
|
|
259
481
|
};
|
|
260
482
|
const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
|
|
261
483
|
return inventory.map((entry) => ({
|