@ouro.bot/cli 0.1.0-alpha.60 → 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 +127 -23
- package/RepairGuide.ouro/agent.json +5 -0
- package/RepairGuide.ouro/psyche/IDENTITY.md +19 -0
- package/RepairGuide.ouro/psyche/SOUL.md +55 -0
- package/RepairGuide.ouro/skills/diagnose-broken-remote.md +63 -0
- package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
- package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
- package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/changelog.json +3857 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +140 -0
- package/dist/arc/episodes.js +117 -0
- package/dist/arc/intentions.js +133 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +254 -0
- package/dist/arc/packets.js +193 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +837 -26
- package/dist/heart/agent-entry.js +69 -3
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/adapter.js +2 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +479 -0
- package/dist/heart/awaiting/await-alert.js +146 -0
- package/dist/heart/awaiting/await-expiry.js +108 -0
- package/dist/heart/awaiting/await-loader.js +91 -0
- package/dist/heart/awaiting/await-parser.js +141 -0
- package/dist/heart/awaiting/await-runtime-state.js +97 -0
- package/dist/heart/awaiting/await-scheduler.js +377 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +142 -0
- package/dist/heart/config-registry.js +322 -0
- package/dist/heart/config.js +114 -119
- package/dist/heart/core.js +909 -246
- package/dist/heart/cross-chat-delivery.js +3 -18
- package/dist/heart/daemon/agent-config-check.js +419 -0
- package/dist/heart/daemon/agent-discovery.js +102 -3
- package/dist/heart/daemon/agent-service.js +522 -0
- package/dist/heart/daemon/agentic-repair.js +547 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/boot-sync-probe.js +197 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +776 -0
- package/dist/heart/daemon/cli-exec.js +7571 -0
- package/dist/heart/daemon/cli-help.js +498 -0
- package/dist/heart/daemon/cli-parse.js +1599 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +763 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/connect-bay.js +323 -0
- package/dist/heart/daemon/daemon-cli.js +29 -1703
- package/dist/heart/daemon/daemon-entry.js +485 -2
- package/dist/heart/daemon/daemon-health.js +176 -0
- package/dist/heart/daemon/daemon-rollup.js +57 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +88 -13
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +842 -69
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +873 -0
- package/dist/heart/daemon/health-monitor.js +122 -1
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
- package/dist/heart/daemon/http-health-probe.js +80 -0
- package/dist/heart/daemon/human-command-screens.js +234 -0
- package/dist/heart/daemon/human-readiness.js +114 -0
- package/dist/heart/daemon/inner-status.js +89 -0
- package/dist/heart/daemon/interactive-repair.js +394 -0
- package/dist/heart/daemon/launchd.js +37 -8
- package/dist/heart/daemon/log-tailer.js +78 -9
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- package/dist/heart/daemon/os-cron-deps.js +135 -0
- package/dist/heart/daemon/os-cron.js +14 -12
- package/dist/heart/daemon/ouro-bot-entry.js +4 -2
- package/dist/heart/daemon/ouro-entry.js +3 -1
- package/dist/heart/daemon/process-manager.js +375 -33
- package/dist/heart/daemon/provider-discovery.js +137 -0
- package/dist/heart/daemon/provider-ping-progress.js +83 -0
- package/dist/heart/daemon/pulse.js +475 -0
- package/dist/heart/daemon/readiness-repair.js +365 -0
- package/dist/heart/daemon/run-hooks.js +2 -0
- package/dist/heart/daemon/runtime-logging.js +10 -2
- package/dist/heart/daemon/runtime-metadata.js +2 -30
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +462 -38
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +158 -11
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +330 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +162 -17
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +1 -1
- package/dist/heart/habits/habit-migration.js +189 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-runtime-state.js +100 -0
- package/dist/heart/habits/habit-scheduler.js +372 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +32 -56
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
- package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
- package/dist/heart/identity.js +166 -55
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +1 -1
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
- package/dist/heart/mailbox/mailbox-http-response.js +7 -0
- package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
- package/dist/heart/mailbox/mailbox-http-static.js +103 -0
- package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
- package/dist/heart/mailbox/mailbox-http.js +99 -0
- package/dist/heart/mailbox/mailbox-read.js +31 -0
- package/dist/heart/mailbox/mailbox-types.js +27 -0
- package/dist/heart/mailbox/mailbox-view.js +195 -0
- package/dist/heart/mailbox/readers/agent-machine.js +382 -0
- package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
- package/dist/heart/mailbox/readers/mail.js +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 +19 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +267 -0
- package/dist/heart/provider-credentials.js +425 -0
- package/dist/heart/provider-failover.js +301 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +262 -0
- package/dist/heart/provider-readiness-cache.js +40 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +139 -52
- package/dist/heart/providers/azure.js +23 -11
- package/dist/heart/providers/error-classification.js +127 -0
- package/dist/heart/providers/github-copilot.js +145 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +26 -8
- package/dist/heart/providers/openai-codex.js +55 -40
- package/dist/heart/runtime-capability-check.js +170 -0
- package/dist/heart/runtime-credentials.js +367 -0
- package/dist/heart/runtime-cwd.js +87 -0
- package/dist/heart/sense-truth.js +13 -4
- package/dist/heart/session-activity.js +43 -22
- package/dist/heart/session-events.js +1149 -0
- package/dist/heart/session-playback-cli-main.js +5 -0
- package/dist/heart/session-playback-cli.js +36 -0
- package/dist/heart/session-playback.js +231 -0
- package/dist/heart/session-stats-cli-main.js +5 -0
- package/dist/heart/session-stats.js +182 -0
- package/dist/heart/session-transcript.js +243 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +44 -27
- package/dist/heart/sync-classification.js +176 -0
- package/dist/heart/sync.js +449 -0
- package/dist/heart/target-resolution.js +9 -5
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/timeouts.js +101 -0
- package/dist/heart/tool-activity-callbacks.js +59 -0
- package/dist/heart/tool-description.js +143 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +421 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +426 -0
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mailbox-ui/assets/index-B-461hes.js +61 -0
- package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/mailbox-ui/index.html +15 -0
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +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 +7 -1
- package/dist/mind/context.js +165 -101
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +62 -75
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/friends/channel.js +39 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +39 -3
- package/dist/mind/friends/types.js +2 -2
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +4 -0
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +1050 -135
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +129 -5
- package/dist/nerves/event-buffer.js +111 -0
- package/dist/nerves/index.js +224 -4
- package/dist/nerves/observation.js +20 -0
- package/dist/nerves/redact.js +79 -0
- package/dist/nerves/review/cli-main.js +5 -0
- package/dist/nerves/review/cli.js +156 -0
- package/dist/nerves/review/core.js +152 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/repertoire/ado-client.js +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +997 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +180 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +158 -9
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +170 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +178 -0
- package/dist/repertoire/duffel-client.js +185 -0
- package/dist/repertoire/github-client.js +14 -55
- package/dist/repertoire/graph-client.js +11 -52
- package/dist/repertoire/guardrails.js +396 -0
- package/dist/repertoire/mcp-client.js +295 -0
- package/dist/repertoire/mcp-manager.js +362 -0
- package/dist/repertoire/mcp-tools.js +63 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +15 -24
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +31 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +16 -4
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -78
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-awaiting.js +360 -0
- package/dist/repertoire/tools-base.js +53 -1082
- package/dist/repertoire/tools-bluebubbles.js +1 -0
- package/dist/repertoire/tools-bridge.js +142 -0
- package/dist/repertoire/tools-bundle.js +984 -0
- package/dist/repertoire/tools-config.js +185 -0
- package/dist/repertoire/tools-continuity.js +248 -0
- package/dist/repertoire/tools-credential.js +381 -0
- package/dist/repertoire/tools-files.js +342 -0
- package/dist/repertoire/tools-flight.js +224 -0
- package/dist/repertoire/tools-flow.js +119 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1916 -0
- package/dist/repertoire/tools-notes.js +421 -0
- package/dist/repertoire/tools-obligations.js +142 -0
- package/dist/repertoire/tools-runtime.js +61 -0
- package/dist/repertoire/tools-session.js +809 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +345 -0
- package/dist/repertoire/tools-teams.js +9 -39
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +604 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools-voice.js +144 -0
- package/dist/repertoire/tools.js +115 -103
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +131 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +594 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/await-turn-message.js +58 -0
- package/dist/senses/bluebubbles/active-turns.js +216 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
- package/dist/senses/bluebubbles/entry.js +77 -0
- package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
- package/dist/senses/bluebubbles/index.js +2548 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -71
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
- package/dist/senses/bluebubbles/processed-log.js +133 -0
- package/dist/senses/bluebubbles/replay.js +137 -0
- package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/bluebubbles-meta-guard.js +40 -0
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +607 -0
- package/dist/senses/cli/spinner-imperative.js +135 -0
- package/dist/senses/cli/spinner.js +101 -0
- package/dist/senses/cli/status-line.js +60 -0
- package/dist/senses/cli/streaming-markdown.js +526 -0
- package/dist/senses/cli/tool-display.js +85 -0
- package/dist/senses/cli/tool-render.js +85 -0
- package/dist/senses/cli/tui-store.js +240 -0
- package/dist/senses/cli/virtual-list.js +35 -0
- package/dist/senses/cli-entry.js +60 -8
- package/dist/senses/cli-layout.js +100 -0
- package/dist/senses/cli.js +516 -204
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +185 -21
- package/dist/senses/inner-dialog.js +372 -27
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +654 -181
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +392 -0
- package/dist/senses/surface-tool.js +70 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +387 -98
- package/dist/senses/trust-gate.js +100 -5
- package/dist/senses/voice/audio-playback.js +237 -0
- package/dist/senses/voice/audio-routing.js +119 -0
- package/dist/senses/voice/elevenlabs.js +202 -0
- package/dist/senses/voice/floor-control.js +431 -0
- package/dist/senses/voice/floor-controller.js +115 -0
- package/dist/senses/voice/golden-path.js +116 -0
- package/dist/senses/voice/index.js +29 -0
- package/dist/senses/voice/meeting.js +113 -0
- package/dist/senses/voice/outbound.js +190 -0
- package/dist/senses/voice/phone.js +33 -0
- package/dist/senses/voice/playback.js +139 -0
- package/dist/senses/voice/realtime-eval.js +496 -0
- package/dist/senses/voice/realtime-trace.js +531 -0
- package/dist/senses/voice/transcript.js +70 -0
- package/dist/senses/voice/turn.js +191 -0
- package/dist/senses/voice/twilio-phone-runtime.js +807 -0
- package/dist/senses/voice/twilio-phone.js +5077 -0
- package/dist/senses/voice/types.js +2 -0
- package/dist/senses/voice/whisper.js +161 -0
- package/dist/senses/voice-entry.js +81 -0
- package/dist/senses/voice-realtime-eval-command.js +99 -0
- package/dist/senses/voice-realtime-eval-entry.js +21 -0
- package/dist/senses/voice-twilio-entry.js +87 -0
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +265 -0
- package/package.json +41 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +99 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/auth-flow.js +0 -351
- package/dist/heart/daemon/ouro-path-installer.js +0 -178
- package/dist/heart/daemon/subagent-installer.js +0 -166
- package/dist/heart/safe-workspace.js +0 -228
- package/dist/heart/session-recall.js +0 -116
- package/dist/mind/associative-recall.js +0 -209
- package/dist/senses/bluebubbles-entry.js +0 -13
- package/dist/senses/bluebubbles.js +0 -1177
- package/dist/senses/debug-activity.js +0 -148
- package/subagents/README.md +0 -86
- package/subagents/work-doer.md +0 -237
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -390
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
- /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
|
@@ -39,14 +39,29 @@ const path = __importStar(require("path"));
|
|
|
39
39
|
const process_manager_1 = require("./process-manager");
|
|
40
40
|
const daemon_1 = require("./daemon");
|
|
41
41
|
const runtime_1 = require("../../nerves/runtime");
|
|
42
|
+
const index_1 = require("../../nerves/index");
|
|
42
43
|
const message_router_1 = require("./message-router");
|
|
43
44
|
const health_monitor_1 = require("./health-monitor");
|
|
45
|
+
const daemon_health_1 = require("./daemon-health");
|
|
46
|
+
const daemon_rollup_1 = require("./daemon-rollup");
|
|
44
47
|
const task_scheduler_1 = require("./task-scheduler");
|
|
45
48
|
const runtime_logging_1 = require("./runtime-logging");
|
|
46
49
|
const sense_manager_1 = require("./sense-manager");
|
|
47
50
|
const agent_discovery_1 = require("./agent-discovery");
|
|
48
51
|
const identity_1 = require("../identity");
|
|
49
52
|
const runtime_mode_1 = require("./runtime-mode");
|
|
53
|
+
const habit_scheduler_1 = require("../habits/habit-scheduler");
|
|
54
|
+
const habit_migration_1 = require("../habits/habit-migration");
|
|
55
|
+
const await_scheduler_1 = require("../awaiting/await-scheduler");
|
|
56
|
+
const await_expiry_1 = require("../awaiting/await-expiry");
|
|
57
|
+
const os_cron_deps_1 = require("./os-cron-deps");
|
|
58
|
+
const os_cron_1 = require("./os-cron");
|
|
59
|
+
const daemon_tombstone_1 = require("./daemon-tombstone");
|
|
60
|
+
const agent_config_check_1 = require("./agent-config-check");
|
|
61
|
+
const pulse_1 = require("./pulse");
|
|
62
|
+
const socket_client_1 = require("./socket-client");
|
|
63
|
+
const bundle_manifest_1 = require("../../mind/bundle-manifest");
|
|
64
|
+
const mcp_canary_1 = require("./mcp-canary");
|
|
50
65
|
function parseSocketPath(argv) {
|
|
51
66
|
const socketIndex = argv.indexOf("--socket");
|
|
52
67
|
if (socketIndex >= 0) {
|
|
@@ -66,6 +81,16 @@ const mode = (0, runtime_mode_1.detectRuntimeMode)((0, identity_1.getRepoRoot)()
|
|
|
66
81
|
message: "starting daemon entrypoint",
|
|
67
82
|
meta: { socketPath, entryPath, mode },
|
|
68
83
|
});
|
|
84
|
+
/* v8 ignore next -- dev-mode indicator: false branch (production) tested in daemon-boot-updates.test.ts @preserve */
|
|
85
|
+
if (mode === "dev") {
|
|
86
|
+
const repoRoot = (0, identity_1.getRepoRoot)();
|
|
87
|
+
(0, runtime_1.emitNervesEvent)({
|
|
88
|
+
component: "daemon",
|
|
89
|
+
event: "daemon.dev_mode_indicator",
|
|
90
|
+
message: `[dev] running from ${repoRoot}`,
|
|
91
|
+
meta: { repoRoot },
|
|
92
|
+
});
|
|
93
|
+
}
|
|
69
94
|
const managedAgents = (0, agent_discovery_1.listEnabledBundleAgents)();
|
|
70
95
|
const processManager = new process_manager_1.DaemonProcessManager({
|
|
71
96
|
agents: managedAgents.map((agent) => ({
|
|
@@ -75,6 +100,28 @@ const processManager = new process_manager_1.DaemonProcessManager({
|
|
|
75
100
|
autoStart: true,
|
|
76
101
|
})),
|
|
77
102
|
existsSync: fs.existsSync,
|
|
103
|
+
/* v8 ignore next 4 -- wiring: delegates to checkAgentConfigWithProviderHealth which has full unit tests @preserve */
|
|
104
|
+
configCheck: async (agent) => {
|
|
105
|
+
const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
|
|
106
|
+
return (0, agent_config_check_1.checkAgentConfigWithProviderHealth)(agent, bundlesRoot);
|
|
107
|
+
},
|
|
108
|
+
/* v8 ignore start -- pulse flush wiring: integration code; flushPulse itself has full unit tests @preserve */
|
|
109
|
+
onSnapshotChange: () => {
|
|
110
|
+
(0, pulse_1.flushPulse)({
|
|
111
|
+
snapshots: processManager.listAgentSnapshots(),
|
|
112
|
+
bundlesRoot: (0, identity_1.getAgentBundlesRoot)(),
|
|
113
|
+
daemonVersion: (0, bundle_manifest_1.getPackageVersion)(),
|
|
114
|
+
now: new Date(),
|
|
115
|
+
// Default I/O wired into pulse.ts (writePulse, readPulse, etc.)
|
|
116
|
+
// Wake recipient: send inner.wake over the daemon's own socket so
|
|
117
|
+
// the recipient agent runs an inner-dialog turn that picks up the
|
|
118
|
+
// pulse alert. Catch errors silently — pulse is best-effort.
|
|
119
|
+
fireInnerWake: (agent) => {
|
|
120
|
+
(0, socket_client_1.sendDaemonCommand)(socketPath, { kind: "inner.wake", agent }).catch(() => { });
|
|
121
|
+
},
|
|
122
|
+
});
|
|
123
|
+
},
|
|
124
|
+
/* v8 ignore stop */
|
|
78
125
|
});
|
|
79
126
|
const scheduler = new task_scheduler_1.TaskDrivenScheduler({
|
|
80
127
|
agents: [...managedAgents],
|
|
@@ -86,6 +133,23 @@ const senseManager = new sense_manager_1.DaemonSenseManager({
|
|
|
86
133
|
const healthMonitor = new health_monitor_1.HealthMonitor({
|
|
87
134
|
processManager,
|
|
88
135
|
scheduler,
|
|
136
|
+
senseProbeProvider: () => [
|
|
137
|
+
...senseManager.listHealthProbes(),
|
|
138
|
+
...managedAgents.map((agent) => (0, mcp_canary_1.createMcpStatusCanaryProbe)({
|
|
139
|
+
agent,
|
|
140
|
+
socketPath,
|
|
141
|
+
command: process.execPath,
|
|
142
|
+
commandArgs: [
|
|
143
|
+
path.join(__dirname, "ouro-bot-entry.js"),
|
|
144
|
+
"mcp-serve",
|
|
145
|
+
"--agent",
|
|
146
|
+
agent,
|
|
147
|
+
"--socket",
|
|
148
|
+
socketPath,
|
|
149
|
+
],
|
|
150
|
+
ignoreOverviewHealth: true,
|
|
151
|
+
})),
|
|
152
|
+
],
|
|
89
153
|
alertSink: (message) => {
|
|
90
154
|
(0, runtime_1.emitNervesEvent)({
|
|
91
155
|
level: "error",
|
|
@@ -95,7 +159,49 @@ const healthMonitor = new health_monitor_1.HealthMonitor({
|
|
|
95
159
|
meta: { message },
|
|
96
160
|
});
|
|
97
161
|
},
|
|
162
|
+
/* v8 ignore next 3 -- wiring: delegates to processManager.restartAgent which has full unit tests @preserve */
|
|
163
|
+
onCriticalAgent: (agentName) => {
|
|
164
|
+
try {
|
|
165
|
+
processManager.restartAgent(agentName);
|
|
166
|
+
}
|
|
167
|
+
catch { /* recovery is best-effort */ }
|
|
168
|
+
},
|
|
169
|
+
/* v8 ignore next 3 -- wiring: delegates to senseManager.restartSense which has focused tests @preserve */
|
|
170
|
+
onCriticalSense: (managedName) => {
|
|
171
|
+
try {
|
|
172
|
+
void senseManager.restartSense(managedName);
|
|
173
|
+
}
|
|
174
|
+
catch { /* recovery is best-effort */ }
|
|
175
|
+
},
|
|
98
176
|
});
|
|
177
|
+
const habitSchedulers = [];
|
|
178
|
+
const awaitSchedulers = [];
|
|
179
|
+
let entryRuntimeStopping = false;
|
|
180
|
+
let stopCommandExitScheduled = false;
|
|
181
|
+
function stopEntryRuntime() {
|
|
182
|
+
if (entryRuntimeStopping)
|
|
183
|
+
return;
|
|
184
|
+
entryRuntimeStopping = true;
|
|
185
|
+
for (const s of habitSchedulers) {
|
|
186
|
+
s.stopWatching();
|
|
187
|
+
s.stop();
|
|
188
|
+
}
|
|
189
|
+
for (const s of awaitSchedulers) {
|
|
190
|
+
s.stopWatching();
|
|
191
|
+
s.stop();
|
|
192
|
+
}
|
|
193
|
+
healthMonitor.stopPeriodicChecks();
|
|
194
|
+
}
|
|
195
|
+
function scheduleCleanProcessExitAfterStopCommand() {
|
|
196
|
+
if (stopCommandExitScheduled)
|
|
197
|
+
return;
|
|
198
|
+
stopCommandExitScheduled = true;
|
|
199
|
+
// Account for the explicit daemon.stop path so the process exit catch-all
|
|
200
|
+
// does not mislabel an operator-requested stop as an unexpected clean exit.
|
|
201
|
+
_tombstoneWritten = true;
|
|
202
|
+
writeStopCommandHealthState();
|
|
203
|
+
setTimeout(() => process.exit(0), 100);
|
|
204
|
+
}
|
|
99
205
|
const daemon = new daemon_1.OuroDaemon({
|
|
100
206
|
socketPath,
|
|
101
207
|
processManager,
|
|
@@ -103,21 +209,398 @@ const daemon = new daemon_1.OuroDaemon({
|
|
|
103
209
|
scheduler,
|
|
104
210
|
healthMonitor,
|
|
105
211
|
router,
|
|
212
|
+
mode,
|
|
213
|
+
onStopCommandComplete: () => {
|
|
214
|
+
stopEntryRuntime();
|
|
215
|
+
scheduleCleanProcessExitAfterStopCommand();
|
|
216
|
+
},
|
|
106
217
|
});
|
|
107
|
-
|
|
218
|
+
const daemonStartedAt = new Date().toISOString();
|
|
219
|
+
const degradedComponents = [];
|
|
220
|
+
function buildDaemonHealthState() {
|
|
221
|
+
const snapshots = processManager.listAgentSnapshots();
|
|
222
|
+
const agentDegradedComponents = snapshots
|
|
223
|
+
.filter((snapshot) => snapshot.status !== "running")
|
|
224
|
+
.map((snapshot) => {
|
|
225
|
+
const reasonParts = [
|
|
226
|
+
snapshot.errorReason ?? `${snapshot.channel} is ${snapshot.status}`,
|
|
227
|
+
snapshot.fixHint ? `Fix: ${snapshot.fixHint}` : null,
|
|
228
|
+
].filter((part) => part !== null);
|
|
229
|
+
return {
|
|
230
|
+
component: `agent:${snapshot.name}`,
|
|
231
|
+
reason: reasonParts.join(" "),
|
|
232
|
+
since: snapshot.lastCrashAt ?? daemonStartedAt,
|
|
233
|
+
};
|
|
234
|
+
});
|
|
235
|
+
// Preserved for backwards-compatible inspection: callers (status
|
|
236
|
+
// command, mailbox surface, etc.) may still read this combined list
|
|
237
|
+
// for per-component reasons. The rollup status field above is what
|
|
238
|
+
// changed meaning — the array is still the union of bootstrap +
|
|
239
|
+
// agent-derived degradation entries.
|
|
240
|
+
const degraded = [
|
|
241
|
+
...degradedComponents.map((entry) => ({ ...entry })),
|
|
242
|
+
...agentDegradedComponents,
|
|
243
|
+
];
|
|
244
|
+
// Layer 1 rollup: project per-agent snapshots into the minimal
|
|
245
|
+
// AgentRollupInput shape and let computeDaemonRollup decide. The
|
|
246
|
+
// input is "every enabled agent" — managedAgents was filtered via
|
|
247
|
+
// listEnabledBundleAgents at module init, and snapshots only covers
|
|
248
|
+
// agents the process manager was told to manage, so by construction
|
|
249
|
+
// these entries are all enabled. The rollup function is a pure
|
|
250
|
+
// declarative function on the data we hand it.
|
|
251
|
+
//
|
|
252
|
+
// Note: safe-mode is wired as `false` here. Existing crash-loop
|
|
253
|
+
// detection (safe-mode.ts) already runs at the daemon-up boot path
|
|
254
|
+
// (cli-exec.ts), not from inside the daemon process itself. Once
|
|
255
|
+
// the daemon is up and reaching this rollup, safe mode no longer
|
|
256
|
+
// applies — the daemon is by definition past the crash-loop gate.
|
|
257
|
+
// If a future PR moves safe-mode signal into the running daemon,
|
|
258
|
+
// wire it through this third argument.
|
|
259
|
+
const rollupStatus = (0, daemon_rollup_1.computeDaemonRollup)({
|
|
260
|
+
enabledAgents: snapshots.map((snapshot) => ({
|
|
261
|
+
name: snapshot.name,
|
|
262
|
+
status: snapshot.status,
|
|
263
|
+
})),
|
|
264
|
+
bootstrapDegraded: degradedComponents,
|
|
265
|
+
safeMode: false,
|
|
266
|
+
});
|
|
267
|
+
return {
|
|
268
|
+
status: rollupStatus,
|
|
269
|
+
mode,
|
|
270
|
+
pid: process.pid,
|
|
271
|
+
startedAt: daemonStartedAt,
|
|
272
|
+
uptimeSeconds: Math.floor(process.uptime()),
|
|
273
|
+
safeMode: null,
|
|
274
|
+
degraded,
|
|
275
|
+
agents: Object.fromEntries(snapshots.map((snapshot) => [
|
|
276
|
+
snapshot.name,
|
|
277
|
+
{
|
|
278
|
+
status: snapshot.status,
|
|
279
|
+
pid: snapshot.pid,
|
|
280
|
+
crashes: snapshot.restartCount,
|
|
281
|
+
},
|
|
282
|
+
])),
|
|
283
|
+
habits: {},
|
|
284
|
+
};
|
|
285
|
+
}
|
|
286
|
+
function recordRecoverableBootstrapFailure(options) {
|
|
287
|
+
const errorMessage = options.error instanceof Error ? options.error.message : String(options.error);
|
|
288
|
+
const existing = degradedComponents.find((entry) => entry.component === options.component);
|
|
289
|
+
const reason = `${errorMessage}. ${options.guidance}`;
|
|
290
|
+
if (existing) {
|
|
291
|
+
existing.reason = reason;
|
|
292
|
+
}
|
|
293
|
+
else {
|
|
294
|
+
degradedComponents.push({
|
|
295
|
+
component: options.component,
|
|
296
|
+
reason,
|
|
297
|
+
since: new Date().toISOString(),
|
|
298
|
+
});
|
|
299
|
+
}
|
|
300
|
+
(0, runtime_1.emitNervesEvent)({
|
|
301
|
+
level: "warn",
|
|
302
|
+
component: "daemon",
|
|
303
|
+
event: "daemon.bootstrap_degraded",
|
|
304
|
+
message: "recoverable daemon bootstrap failure; daemon remains available in degraded mode",
|
|
305
|
+
meta: {
|
|
306
|
+
agent: options.agent,
|
|
307
|
+
component: options.component,
|
|
308
|
+
habitsDir: options.habitsDir,
|
|
309
|
+
error: errorMessage,
|
|
310
|
+
guidance: options.guidance,
|
|
311
|
+
},
|
|
312
|
+
});
|
|
313
|
+
}
|
|
314
|
+
function emitHabitSetupError(agent, error) {
|
|
315
|
+
const normalized = error instanceof Error ? error : new Error(String(error));
|
|
316
|
+
(0, runtime_1.emitNervesEvent)({
|
|
317
|
+
level: "error",
|
|
318
|
+
component: "daemon",
|
|
319
|
+
event: "daemon.habit_setup_error",
|
|
320
|
+
message: `habit setup failed for agent ${agent}`,
|
|
321
|
+
meta: { agent, error: normalized.message },
|
|
322
|
+
});
|
|
323
|
+
}
|
|
324
|
+
function emitAwaitSetupError(agent, error) {
|
|
325
|
+
const normalized = error instanceof Error ? error : new Error(String(error));
|
|
326
|
+
(0, runtime_1.emitNervesEvent)({
|
|
327
|
+
level: "error",
|
|
328
|
+
component: "daemon",
|
|
329
|
+
event: "daemon.await_setup_error",
|
|
330
|
+
message: `await setup failed for agent ${agent}`,
|
|
331
|
+
meta: { agent, error: normalized.message },
|
|
332
|
+
});
|
|
333
|
+
}
|
|
334
|
+
/* v8 ignore start — daemon health writer wiring, tested via daemon-health.test.ts @preserve */
|
|
335
|
+
const healthWriter = new daemon_health_1.DaemonHealthWriter((0, daemon_health_1.getDefaultHealthPath)());
|
|
336
|
+
const healthSink = (0, daemon_health_1.createHealthNervesSink)(healthWriter, buildDaemonHealthState);
|
|
337
|
+
(0, index_1.registerGlobalLogSink)(healthSink);
|
|
338
|
+
/* v8 ignore stop */
|
|
339
|
+
function writeStopCommandHealthState() {
|
|
340
|
+
try {
|
|
341
|
+
healthWriter.writeHealth({
|
|
342
|
+
...buildDaemonHealthState(),
|
|
343
|
+
status: "down",
|
|
344
|
+
uptimeSeconds: Math.floor(process.uptime()),
|
|
345
|
+
});
|
|
346
|
+
}
|
|
347
|
+
catch {
|
|
348
|
+
// Health writes are best-effort during shutdown.
|
|
349
|
+
}
|
|
350
|
+
}
|
|
351
|
+
/* v8 ignore start -- habit wiring: lambdas delegate to processManager/fs; tested via HabitScheduler unit tests @preserve */
|
|
352
|
+
void daemon.start().then(() => {
|
|
353
|
+
const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
|
|
354
|
+
const ouroPath = (0, os_cron_deps_1.resolveOuroBinaryPath)();
|
|
355
|
+
const osCronDeps = (0, os_cron_deps_1.createRealOsCronDeps)();
|
|
356
|
+
for (const agent of managedAgents) {
|
|
357
|
+
const bundleRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
358
|
+
const habitsDir = path.join(bundleRoot, "habits");
|
|
359
|
+
const degradedComponent = `habits:${agent}`;
|
|
360
|
+
try {
|
|
361
|
+
// Migrate old tasks/habits/ to habits/ at bundle root
|
|
362
|
+
(0, habit_migration_1.migrateHabitsFromTaskSystem)(bundleRoot);
|
|
363
|
+
const osCronManager = new os_cron_1.LaunchdCronManager(osCronDeps);
|
|
364
|
+
const scheduler = new habit_scheduler_1.HabitScheduler({
|
|
365
|
+
agent,
|
|
366
|
+
habitsDir,
|
|
367
|
+
osCronManager,
|
|
368
|
+
onHabitFire: (habitName) => {
|
|
369
|
+
processManager.sendToAgent(agent, { type: "habit", habitName });
|
|
370
|
+
},
|
|
371
|
+
deps: {
|
|
372
|
+
readdir: (dir) => fs.readdirSync(dir),
|
|
373
|
+
readFile: (p, enc) => fs.readFileSync(p, enc),
|
|
374
|
+
writeFile: (p, c, enc) => fs.writeFileSync(p, c, enc),
|
|
375
|
+
existsSync: (p) => fs.existsSync(p),
|
|
376
|
+
now: () => Date.now(),
|
|
377
|
+
ouroPath,
|
|
378
|
+
watch: (dir, cb) => fs.watch(dir, cb),
|
|
379
|
+
},
|
|
380
|
+
});
|
|
381
|
+
try {
|
|
382
|
+
scheduler.start();
|
|
383
|
+
scheduler.startPeriodicReconciliation();
|
|
384
|
+
scheduler.watchForChanges();
|
|
385
|
+
habitSchedulers.push(scheduler);
|
|
386
|
+
}
|
|
387
|
+
catch (error) {
|
|
388
|
+
try {
|
|
389
|
+
scheduler.stopWatching();
|
|
390
|
+
scheduler.stop();
|
|
391
|
+
}
|
|
392
|
+
catch {
|
|
393
|
+
// Cleanup is best-effort for partially initialized schedulers.
|
|
394
|
+
}
|
|
395
|
+
emitHabitSetupError(agent, error);
|
|
396
|
+
recordRecoverableBootstrapFailure({
|
|
397
|
+
agent,
|
|
398
|
+
component: degradedComponent,
|
|
399
|
+
habitsDir,
|
|
400
|
+
error,
|
|
401
|
+
guidance: `fix ${agent} habits or cron setup and rerun ouro up to restore habit automation`,
|
|
402
|
+
});
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
catch (err) {
|
|
406
|
+
const error = err instanceof Error ? err : new Error(String(err));
|
|
407
|
+
emitHabitSetupError(agent, error);
|
|
408
|
+
recordRecoverableBootstrapFailure({
|
|
409
|
+
agent,
|
|
410
|
+
component: degradedComponent,
|
|
411
|
+
habitsDir,
|
|
412
|
+
error,
|
|
413
|
+
guidance: `fix ${agent} habits or cron setup and rerun ouro up to restore habit automation`,
|
|
414
|
+
});
|
|
415
|
+
}
|
|
416
|
+
// Parallel await-condition scheduler. Uses its own OS cron manager so
|
|
417
|
+
// habits and awaits don't share label namespace and stale removals can't
|
|
418
|
+
// collide.
|
|
419
|
+
const awaitsDir = path.join(bundleRoot, "awaiting");
|
|
420
|
+
const awaitDegradedComponent = `awaits:${agent}`;
|
|
421
|
+
try {
|
|
422
|
+
const awaitOsCronManager = new os_cron_1.LaunchdCronManager(osCronDeps);
|
|
423
|
+
const awaitScheduler = new await_scheduler_1.AwaitScheduler({
|
|
424
|
+
agent,
|
|
425
|
+
awaitsDir,
|
|
426
|
+
osCronManager: awaitOsCronManager,
|
|
427
|
+
onAwaitFire: (awaitName) => {
|
|
428
|
+
processManager.sendToAgent(agent, { type: "await", awaitName });
|
|
429
|
+
},
|
|
430
|
+
onAwaitExpire: (awaitName) => {
|
|
431
|
+
void (0, await_expiry_1.archiveAndAlertExpiredAwait)({
|
|
432
|
+
agentRoot: bundleRoot,
|
|
433
|
+
agentName: agent,
|
|
434
|
+
awaitName,
|
|
435
|
+
deliveryDeps: {
|
|
436
|
+
agentName: agent,
|
|
437
|
+
queuePending: () => {
|
|
438
|
+
// Best-effort: queue inner-dialog wake so the agent processes the alert path
|
|
439
|
+
(0, socket_client_1.sendDaemonCommand)(socketPath, { kind: "inner.wake", agent }).catch(() => { });
|
|
440
|
+
},
|
|
441
|
+
},
|
|
442
|
+
}).catch((err) => {
|
|
443
|
+
(0, runtime_1.emitNervesEvent)({
|
|
444
|
+
level: "error",
|
|
445
|
+
component: "daemon",
|
|
446
|
+
event: "daemon.await_expire_error",
|
|
447
|
+
message: "await expiry handler threw",
|
|
448
|
+
meta: { agent, awaitName, error: err instanceof Error ? err.message : String(err) },
|
|
449
|
+
});
|
|
450
|
+
});
|
|
451
|
+
},
|
|
452
|
+
deps: {
|
|
453
|
+
readdir: (dir) => fs.readdirSync(dir),
|
|
454
|
+
readFile: (p, enc) => fs.readFileSync(p, enc),
|
|
455
|
+
existsSync: (p) => fs.existsSync(p),
|
|
456
|
+
mkdir: (dir) => { fs.mkdirSync(dir, { recursive: true }); },
|
|
457
|
+
now: () => Date.now(),
|
|
458
|
+
ouroPath,
|
|
459
|
+
watch: (dir, cb) => fs.watch(dir, cb),
|
|
460
|
+
},
|
|
461
|
+
});
|
|
462
|
+
try {
|
|
463
|
+
awaitScheduler.start();
|
|
464
|
+
awaitScheduler.startPeriodicReconciliation();
|
|
465
|
+
awaitScheduler.watchForChanges();
|
|
466
|
+
awaitSchedulers.push(awaitScheduler);
|
|
467
|
+
}
|
|
468
|
+
catch (error) {
|
|
469
|
+
try {
|
|
470
|
+
awaitScheduler.stopWatching();
|
|
471
|
+
awaitScheduler.stop();
|
|
472
|
+
}
|
|
473
|
+
catch {
|
|
474
|
+
// best-effort cleanup
|
|
475
|
+
}
|
|
476
|
+
emitAwaitSetupError(agent, error);
|
|
477
|
+
recordRecoverableBootstrapFailure({
|
|
478
|
+
agent,
|
|
479
|
+
component: awaitDegradedComponent,
|
|
480
|
+
habitsDir: awaitsDir,
|
|
481
|
+
error,
|
|
482
|
+
guidance: `fix ${agent} awaits or cron setup and rerun ouro up to restore await automation`,
|
|
483
|
+
});
|
|
484
|
+
}
|
|
485
|
+
}
|
|
486
|
+
catch (err) {
|
|
487
|
+
const error = err instanceof Error ? err : new Error(String(err));
|
|
488
|
+
emitAwaitSetupError(agent, error);
|
|
489
|
+
recordRecoverableBootstrapFailure({
|
|
490
|
+
agent,
|
|
491
|
+
component: awaitDegradedComponent,
|
|
492
|
+
habitsDir: awaitsDir,
|
|
493
|
+
error,
|
|
494
|
+
guidance: `fix ${agent} awaits or cron setup and rerun ouro up to restore await automation`,
|
|
495
|
+
});
|
|
496
|
+
}
|
|
497
|
+
}
|
|
498
|
+
healthMonitor.startPeriodicChecks(60_000);
|
|
499
|
+
/* v8 ignore start -- startup failure + signal handlers: call process.exit, untestable in vitest @preserve */
|
|
500
|
+
}).catch(async (err) => {
|
|
501
|
+
const error = err instanceof Error ? err : new Error(String(err));
|
|
502
|
+
_tombstoneWritten = true;
|
|
503
|
+
(0, daemon_tombstone_1.writeDaemonTombstone)("startupFailure", error);
|
|
108
504
|
(0, runtime_1.emitNervesEvent)({
|
|
109
505
|
level: "error",
|
|
110
506
|
component: "daemon",
|
|
111
507
|
event: "daemon.entry_error",
|
|
112
508
|
message: "daemon entrypoint failed",
|
|
113
|
-
meta: {},
|
|
509
|
+
meta: { error: error.message },
|
|
114
510
|
});
|
|
511
|
+
setTimeout(() => process.exit(1), 5_000).unref();
|
|
115
512
|
await daemon.stop();
|
|
116
513
|
process.exit(1);
|
|
117
514
|
});
|
|
118
515
|
process.on("SIGINT", () => {
|
|
516
|
+
// ALWAYS write a tombstone, even on signal-driven shutdown. The previous
|
|
517
|
+
// behavior was to set _gracefulShutdown=true and skip the tombstone, which
|
|
518
|
+
// meant ANY external SIGINT/SIGTERM (launchd policy, OOM killer, manual
|
|
519
|
+
// kill, killOrphanProcesses from a sibling daemon) silently disappeared
|
|
520
|
+
// from the death log. The user lost weeks of visibility into why their
|
|
521
|
+
// daemon kept dying. Tombstones are informational — having a "sigint"
|
|
522
|
+
// tombstone is strictly better than silence.
|
|
523
|
+
_tombstoneWritten = true;
|
|
524
|
+
(0, daemon_tombstone_1.writeDaemonTombstone)("sigint", new Error("daemon received SIGINT"));
|
|
525
|
+
stopEntryRuntime();
|
|
526
|
+
setTimeout(() => process.exit(1), 5_000).unref();
|
|
119
527
|
void daemon.stop().then(() => process.exit(0));
|
|
120
528
|
});
|
|
121
529
|
process.on("SIGTERM", () => {
|
|
530
|
+
_tombstoneWritten = true;
|
|
531
|
+
(0, daemon_tombstone_1.writeDaemonTombstone)("sigterm", new Error("daemon received SIGTERM"));
|
|
532
|
+
stopEntryRuntime();
|
|
533
|
+
setTimeout(() => process.exit(1), 5_000).unref();
|
|
122
534
|
void daemon.stop().then(() => process.exit(0));
|
|
123
535
|
});
|
|
536
|
+
/* v8 ignore stop */
|
|
537
|
+
// Suppress EPIPE on stdout/stderr — normal when detached daemon's parent exits
|
|
538
|
+
/* v8 ignore start -- EPIPE suppression: only fires when parent process exits @preserve */
|
|
539
|
+
process.stdout?.on?.("error", () => { });
|
|
540
|
+
process.stderr?.on?.("error", () => { });
|
|
541
|
+
/* v8 ignore stop */
|
|
542
|
+
/* v8 ignore start -- global exception handlers: genuinely untestable in vitest; exercised by real daemon crashes @preserve */
|
|
543
|
+
let _uncaughtCount = 0;
|
|
544
|
+
let _tombstoneWritten = false;
|
|
545
|
+
let _lastKnownCause = null;
|
|
546
|
+
const CIRCUIT_BREAKER_WINDOW_MS = 60_000;
|
|
547
|
+
const CIRCUIT_BREAKER_MAX = 10;
|
|
548
|
+
process.on("uncaughtException", (error) => {
|
|
549
|
+
// EPIPE is normal for detached daemon processes — parent closed the pipe
|
|
550
|
+
if (error.code === "EPIPE")
|
|
551
|
+
return;
|
|
552
|
+
_uncaughtCount++;
|
|
553
|
+
_lastKnownCause = error;
|
|
554
|
+
setTimeout(() => { _uncaughtCount--; }, CIRCUIT_BREAKER_WINDOW_MS).unref();
|
|
555
|
+
_tombstoneWritten = true;
|
|
556
|
+
(0, daemon_tombstone_1.writeDaemonTombstone)("uncaughtException", error);
|
|
557
|
+
(0, runtime_1.emitNervesEvent)({
|
|
558
|
+
level: "error",
|
|
559
|
+
component: "daemon",
|
|
560
|
+
event: "daemon.uncaught_exception",
|
|
561
|
+
message: "uncaught exception in daemon process (continuing)",
|
|
562
|
+
meta: { error: error.message, stack: error.stack ?? null, uncaughtCount: _uncaughtCount },
|
|
563
|
+
});
|
|
564
|
+
// Circuit breaker: if too many exceptions in a short window, the process
|
|
565
|
+
// is in a bad state — exit so launchd/self-spawn can restart fresh.
|
|
566
|
+
if (_uncaughtCount >= CIRCUIT_BREAKER_MAX) {
|
|
567
|
+
(0, runtime_1.emitNervesEvent)({
|
|
568
|
+
level: "error",
|
|
569
|
+
component: "daemon",
|
|
570
|
+
event: "daemon.circuit_breaker_exit",
|
|
571
|
+
message: `daemon exiting: ${_uncaughtCount} uncaught exceptions in ${CIRCUIT_BREAKER_WINDOW_MS / 1000}s`,
|
|
572
|
+
meta: { uncaughtCount: _uncaughtCount },
|
|
573
|
+
});
|
|
574
|
+
setTimeout(() => process.exit(1), 5_000).unref();
|
|
575
|
+
void daemon.stop().then(() => process.exit(1));
|
|
576
|
+
}
|
|
577
|
+
});
|
|
578
|
+
process.on("unhandledRejection", (reason) => {
|
|
579
|
+
const error = reason instanceof Error ? reason : new Error(String(reason));
|
|
580
|
+
_lastKnownCause = error;
|
|
581
|
+
_tombstoneWritten = true;
|
|
582
|
+
(0, daemon_tombstone_1.writeDaemonTombstone)("unhandledRejection", error);
|
|
583
|
+
(0, runtime_1.emitNervesEvent)({
|
|
584
|
+
level: "error",
|
|
585
|
+
component: "daemon",
|
|
586
|
+
event: "daemon.unhandled_rejection",
|
|
587
|
+
message: "unhandled promise rejection in daemon process",
|
|
588
|
+
meta: { reason: error.message, stack: error.stack ?? null },
|
|
589
|
+
});
|
|
590
|
+
});
|
|
591
|
+
// Catch-all: write tombstone on any exit where we didn't already record the cause.
|
|
592
|
+
// process.on('exit') is synchronous-only — writeDaemonTombstone uses writeFileSync, so it works.
|
|
593
|
+
//
|
|
594
|
+
// Previously this skipped writing if `_gracefulShutdown` was true, which made
|
|
595
|
+
// SIGINT/SIGTERM-driven exits invisible in the death log. The signal handlers
|
|
596
|
+
// above now always write their own tombstone before exiting, so this catch-all
|
|
597
|
+
// only runs for exits the signal handlers didn't reach (e.g. process.exit
|
|
598
|
+
// called from somewhere unexpected).
|
|
599
|
+
process.on("exit", (code) => {
|
|
600
|
+
if (_tombstoneWritten)
|
|
601
|
+
return;
|
|
602
|
+
const reason = code === 0 ? "unexpectedCleanExit" : "unexpectedExit";
|
|
603
|
+
const error = _lastKnownCause ?? new Error(`daemon exited with code ${code} (no specific cause captured)`);
|
|
604
|
+
(0, daemon_tombstone_1.writeDaemonTombstone)(reason, error);
|
|
605
|
+
});
|
|
606
|
+
/* v8 ignore stop */
|