@ouro.bot/cli 0.1.0-alpha.56 → 0.1.0-alpha.560
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 +3596 -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 +479 -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 +322 -0
- package/dist/heart/config.js +114 -118
- package/dist/heart/core.js +913 -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 +7457 -0
- package/dist/heart/daemon/cli-help.js +498 -0
- package/dist/heart/daemon/cli-parse.js +1592 -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 -1698
- package/dist/heart/daemon/daemon-entry.js +387 -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 +796 -71
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +826 -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 +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- package/dist/heart/daemon/message-router.js +2 -2
- 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 +67 -16
- package/dist/heart/daemon/runtime-metadata.js +3 -31
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +389 -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 +203 -57
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +1 -1
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
- package/dist/heart/mailbox/mailbox-http-response.js +7 -0
- package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
- package/dist/heart/mailbox/mailbox-http-static.js +103 -0
- package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
- package/dist/heart/mailbox/mailbox-http.js +99 -0
- package/dist/heart/mailbox/mailbox-read.js +31 -0
- package/dist/heart/mailbox/mailbox-types.js +27 -0
- package/dist/heart/mailbox/mailbox-view.js +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 +362 -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 +683 -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 +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/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 +139 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +389 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +426 -0
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mailbox-ui/assets/index-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 +674 -0
- package/dist/mailroom/body-cache.js +61 -0
- package/dist/mailroom/core.js +720 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +430 -0
- package/dist/mailroom/mbox-import.js +383 -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 +256 -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 +1011 -123
- 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/coverage/run-artifacts.js +1 -1
- package/dist/nerves/event-buffer.js +111 -0
- package/dist/nerves/index.js +224 -4
- package/dist/nerves/observation.js +20 -0
- package/dist/nerves/redact.js +79 -0
- package/dist/nerves/review/cli-main.js +5 -0
- package/dist/nerves/review/cli.js +156 -0
- package/dist/nerves/review/core.js +152 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/repertoire/ado-client.js +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +963 -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-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 +1857 -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 +604 -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 +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/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 +2305 -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 +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/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 +549 -181
- 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/senses/voice/elevenlabs.js +125 -0
- package/dist/senses/voice/index.js +22 -0
- package/dist/senses/voice/transcript.js +70 -0
- package/dist/senses/voice/turn.js +85 -0
- package/dist/senses/voice/types.js +2 -0
- package/dist/senses/voice/whisper.js +133 -0
- package/dist/senses/voice-entry.js +80 -0
- 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/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/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,131 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Pluggable session ID resolver for MCP conversations.
|
|
3
|
+
// Tries tool-specific methods first, falls back to UUID.
|
|
4
|
+
//
|
|
5
|
+
// Claude Code: walks parent PID chain -> reads ~/.claude/sessions/{pid}.json
|
|
6
|
+
// Codex: reads thread_id from env (future)
|
|
7
|
+
// Fallback: generates UUID
|
|
8
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
9
|
+
if (k2 === undefined) k2 = k;
|
|
10
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
11
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
12
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
13
|
+
}
|
|
14
|
+
Object.defineProperty(o, k2, desc);
|
|
15
|
+
}) : (function(o, m, k, k2) {
|
|
16
|
+
if (k2 === undefined) k2 = k;
|
|
17
|
+
o[k2] = m[k];
|
|
18
|
+
}));
|
|
19
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
20
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
21
|
+
}) : function(o, v) {
|
|
22
|
+
o["default"] = v;
|
|
23
|
+
});
|
|
24
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
25
|
+
var ownKeys = function(o) {
|
|
26
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
27
|
+
var ar = [];
|
|
28
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
29
|
+
return ar;
|
|
30
|
+
};
|
|
31
|
+
return ownKeys(o);
|
|
32
|
+
};
|
|
33
|
+
return function (mod) {
|
|
34
|
+
if (mod && mod.__esModule) return mod;
|
|
35
|
+
var result = {};
|
|
36
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
37
|
+
__setModuleDefault(result, mod);
|
|
38
|
+
return result;
|
|
39
|
+
};
|
|
40
|
+
})();
|
|
41
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
42
|
+
exports.resolveSessionId = resolveSessionId;
|
|
43
|
+
const fs = __importStar(require("fs"));
|
|
44
|
+
const path = __importStar(require("path"));
|
|
45
|
+
const os = __importStar(require("os"));
|
|
46
|
+
const crypto_1 = require("crypto");
|
|
47
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
48
|
+
const DEFAULT_CLAUDE_SESSIONS_DIR = path.join(os.homedir(), ".claude", "sessions");
|
|
49
|
+
const MAX_PID_WALK_DEPTH = 10;
|
|
50
|
+
/**
|
|
51
|
+
* Try to read a Claude Code session ID from a PID-keyed session file.
|
|
52
|
+
* Returns the sessionId if found, null otherwise.
|
|
53
|
+
*/
|
|
54
|
+
function tryReadClaudeSession(sessionsDir, pid) {
|
|
55
|
+
const sessionFile = path.join(sessionsDir, `${pid}.json`);
|
|
56
|
+
if (!fs.existsSync(sessionFile))
|
|
57
|
+
return null;
|
|
58
|
+
try {
|
|
59
|
+
const raw = fs.readFileSync(sessionFile, "utf-8");
|
|
60
|
+
const data = JSON.parse(raw);
|
|
61
|
+
if (typeof data.sessionId === "string" && data.sessionId.length > 0) {
|
|
62
|
+
return data.sessionId;
|
|
63
|
+
}
|
|
64
|
+
return null;
|
|
65
|
+
}
|
|
66
|
+
catch {
|
|
67
|
+
return null;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Walk the parent PID chain looking for a Claude Code session file.
|
|
72
|
+
* Starts at the current process PID, walks up to parent, grandparent, etc.
|
|
73
|
+
* Returns the session ID from the first matching file, or null.
|
|
74
|
+
*/
|
|
75
|
+
function walkPidChain(sessionsDir) {
|
|
76
|
+
let currentPid = process.pid;
|
|
77
|
+
for (let depth = 0; depth < MAX_PID_WALK_DEPTH; depth++) {
|
|
78
|
+
const sessionId = tryReadClaudeSession(sessionsDir, currentPid);
|
|
79
|
+
if (sessionId) {
|
|
80
|
+
(0, runtime_1.emitNervesEvent)({
|
|
81
|
+
component: "daemon",
|
|
82
|
+
event: "daemon.session_id_pid_walk_hit",
|
|
83
|
+
message: "found Claude session via PID walk",
|
|
84
|
+
meta: { pid: currentPid, depth, sessionId },
|
|
85
|
+
});
|
|
86
|
+
return sessionId;
|
|
87
|
+
}
|
|
88
|
+
// Walk to parent PID
|
|
89
|
+
// On macOS/Linux, process.ppid gives the parent. For deeper ancestry,
|
|
90
|
+
// we'd need /proc/{pid}/stat or ps -o ppid=. For now, we check current + parent.
|
|
91
|
+
if (depth === 0) {
|
|
92
|
+
currentPid = process.ppid;
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
// Cannot walk further without OS-specific process tree APIs
|
|
96
|
+
break;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
return null;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Resolve a session ID for the current MCP connection.
|
|
103
|
+
* Returns a stable identifier that ties MCP tool calls to a conversation session.
|
|
104
|
+
*
|
|
105
|
+
* Resolution order:
|
|
106
|
+
* 1. Claude Code PID walk: check ~/.claude/sessions/{pid}.json for current + parent PID
|
|
107
|
+
* 2. UUID fallback: generate a random UUID
|
|
108
|
+
*/
|
|
109
|
+
function resolveSessionId(options) {
|
|
110
|
+
const sessionsDir = options?.claudeSessionsDir ?? DEFAULT_CLAUDE_SESSIONS_DIR;
|
|
111
|
+
// Try Claude Code PID walk
|
|
112
|
+
const claudeSessionId = walkPidChain(sessionsDir);
|
|
113
|
+
if (claudeSessionId) {
|
|
114
|
+
(0, runtime_1.emitNervesEvent)({
|
|
115
|
+
component: "daemon",
|
|
116
|
+
event: "daemon.session_id_resolved",
|
|
117
|
+
message: "session ID resolved via Claude Code PID walk",
|
|
118
|
+
meta: { sessionId: claudeSessionId, method: "claude-pid-walk" },
|
|
119
|
+
});
|
|
120
|
+
return claudeSessionId;
|
|
121
|
+
}
|
|
122
|
+
// Fallback: UUID
|
|
123
|
+
const sessionId = (0, crypto_1.randomUUID)();
|
|
124
|
+
(0, runtime_1.emitNervesEvent)({
|
|
125
|
+
component: "daemon",
|
|
126
|
+
event: "daemon.session_id_resolved",
|
|
127
|
+
message: "session ID resolved via UUID fallback",
|
|
128
|
+
meta: { sessionId, method: "uuid-fallback" },
|
|
129
|
+
});
|
|
130
|
+
return sessionId;
|
|
131
|
+
}
|
|
@@ -0,0 +1,94 @@
|
|
|
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.ensureSkillManagement = ensureSkillManagement;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
40
|
+
const identity_1 = require("../identity");
|
|
41
|
+
const SKILL_MANAGEMENT_URL = "https://raw.githubusercontent.com/ouroborosbot/ouroboros-skills/main/skills/skill-management/SKILL.md";
|
|
42
|
+
async function ensureSkillManagement() {
|
|
43
|
+
const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
|
|
44
|
+
if (!fs.existsSync(bundlesRoot))
|
|
45
|
+
return;
|
|
46
|
+
// Find all agent bundles
|
|
47
|
+
const entries = fs.readdirSync(bundlesRoot).filter(e => e.endsWith(".ouro"));
|
|
48
|
+
if (entries.length === 0)
|
|
49
|
+
return;
|
|
50
|
+
// Check if ANY bundle is missing the skill
|
|
51
|
+
const missing = entries.filter(e => {
|
|
52
|
+
const targetPath = path.join(bundlesRoot, e, "skills", "skill-management.md");
|
|
53
|
+
return !fs.existsSync(targetPath);
|
|
54
|
+
});
|
|
55
|
+
if (missing.length === 0)
|
|
56
|
+
return;
|
|
57
|
+
// eslint-disable-next-line no-console -- terminal UX: visible install status
|
|
58
|
+
console.log("installing skill-management from ouroboros-skills...");
|
|
59
|
+
try {
|
|
60
|
+
const response = await fetch(SKILL_MANAGEMENT_URL);
|
|
61
|
+
if (!response.ok) {
|
|
62
|
+
// eslint-disable-next-line no-console -- terminal UX: visible install status
|
|
63
|
+
console.error(`✗ failed to fetch skill-management (HTTP ${response.status})`);
|
|
64
|
+
(0, runtime_1.emitNervesEvent)({
|
|
65
|
+
level: "warn",
|
|
66
|
+
component: "daemon",
|
|
67
|
+
event: "daemon.skill_management_install_error",
|
|
68
|
+
message: "failed to fetch skill-management from GitHub",
|
|
69
|
+
meta: { status: response.status, url: SKILL_MANAGEMENT_URL },
|
|
70
|
+
});
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
const content = await response.text();
|
|
74
|
+
for (const bundle of missing) {
|
|
75
|
+
const skillsDir = path.join(bundlesRoot, bundle, "skills");
|
|
76
|
+
const targetPath = path.join(skillsDir, "skill-management.md");
|
|
77
|
+
fs.mkdirSync(skillsDir, { recursive: true });
|
|
78
|
+
fs.writeFileSync(targetPath, content, "utf-8");
|
|
79
|
+
}
|
|
80
|
+
// eslint-disable-next-line no-console -- terminal UX: visible install status
|
|
81
|
+
console.log(`✓ installed skill-management (${missing.length} agent${missing.length > 1 ? "s" : ""})`);
|
|
82
|
+
}
|
|
83
|
+
catch (error) {
|
|
84
|
+
// eslint-disable-next-line no-console -- terminal UX: visible install status
|
|
85
|
+
console.error(`✗ failed to install skill-management: ${error instanceof Error ? error.message : String(error)}`);
|
|
86
|
+
(0, runtime_1.emitNervesEvent)({
|
|
87
|
+
level: "warn",
|
|
88
|
+
component: "daemon",
|
|
89
|
+
event: "daemon.skill_management_install_error",
|
|
90
|
+
message: "failed to install skill-management skill",
|
|
91
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
92
|
+
});
|
|
93
|
+
}
|
|
94
|
+
}
|
|
@@ -33,7 +33,8 @@ var __importStar = (this && this.__importStar) || (function () {
|
|
|
33
33
|
};
|
|
34
34
|
})();
|
|
35
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
-
exports.DEFAULT_DAEMON_SOCKET_PATH = void 0;
|
|
36
|
+
exports.DEFAULT_DAEMON_COMMAND_TIMEOUT_MS = exports.DEFAULT_DAEMON_SOCKET_PATH = void 0;
|
|
37
|
+
exports.__bypassVitestGuardForTests = __bypassVitestGuardForTests;
|
|
37
38
|
exports.sendDaemonCommand = sendDaemonCommand;
|
|
38
39
|
exports.checkDaemonSocketAlive = checkDaemonSocketAlive;
|
|
39
40
|
exports.requestInnerWake = requestInnerWake;
|
|
@@ -41,7 +42,82 @@ const fs = __importStar(require("fs"));
|
|
|
41
42
|
const net = __importStar(require("net"));
|
|
42
43
|
const runtime_1 = require("../../nerves/runtime");
|
|
43
44
|
exports.DEFAULT_DAEMON_SOCKET_PATH = "/tmp/ouroboros-daemon.sock";
|
|
44
|
-
|
|
45
|
+
exports.DEFAULT_DAEMON_COMMAND_TIMEOUT_MS = 10 * 60 * 1000;
|
|
46
|
+
/**
|
|
47
|
+
* Defense-in-depth: detect if we're running under vitest. Tests that forget
|
|
48
|
+
* to `vi.mock("../../heart/daemon/socket-client")` would otherwise leak real
|
|
49
|
+
* `inner.wake` commands (with whatever literal agent name the test uses,
|
|
50
|
+
* commonly "testagent") into whichever real daemon happens to be running on
|
|
51
|
+
* the developer's machine. That has caused real outages when test loops
|
|
52
|
+
* flooded the production-on-localhost daemon.
|
|
53
|
+
*
|
|
54
|
+
* We detect vitest via `process.argv` (not env vars — the project bans those)
|
|
55
|
+
* and convert all socket operations into safe no-ops. Individual tests are
|
|
56
|
+
* still expected to vi.mock this module so they can assert on call counts,
|
|
57
|
+
* but this guard guarantees that "I forgot to mock" can never again take
|
|
58
|
+
* down the daemon.
|
|
59
|
+
*
|
|
60
|
+
* The socket-client's OWN tests (which exercise the real functions against a
|
|
61
|
+
* test socket) call `__bypassVitestGuardForTests(true)` to opt out of the
|
|
62
|
+
* guard. We persist that flag on globalThis so it survives `vi.resetModules`
|
|
63
|
+
* (which clears the module cache between tests).
|
|
64
|
+
*
|
|
65
|
+
* HARDENING (cross-file leak protection): the bypass flag lives on globalThis
|
|
66
|
+
* and is process-wide. With vitest's default `fileParallelism: true`,
|
|
67
|
+
* concurrently-running test files in the same worker can have the bypass on
|
|
68
|
+
* even though they didn't ask for it — and any test that uses
|
|
69
|
+
* `name: "testagent"` and exercises a code path through this module will
|
|
70
|
+
* leak `inner.wake testagent` to the production daemon socket. We saw this
|
|
71
|
+
* cause a real outage on 2026-04-08 (daemon SIGTERMed mid-validation).
|
|
72
|
+
*
|
|
73
|
+
* Defense: even when the bypass flag is set, we ALWAYS hard-block calls that
|
|
74
|
+
* target the production daemon socket path. Test files that legitimately
|
|
75
|
+
* exercise the real socket-client code paths use a synthetic socket path like
|
|
76
|
+
* `/tmp/daemon.sock` or `/tmp/some-real-daemon.sock` — those keep working.
|
|
77
|
+
* Calls to `/tmp/ouroboros-daemon.sock` (DEFAULT_DAEMON_SOCKET_PATH) under
|
|
78
|
+
* vitest are unconditionally blocked, regardless of bypass state. This makes
|
|
79
|
+
* the cross-file leak vector impossible without restricting legitimate
|
|
80
|
+
* socket-client unit tests.
|
|
81
|
+
*/
|
|
82
|
+
const BYPASS_KEY = "__ouro_socket_client_bypass_vitest_guard__";
|
|
83
|
+
/** ONLY for socket-client.test.ts. Do not call from any other test or production code. */
|
|
84
|
+
function __bypassVitestGuardForTests(bypass) {
|
|
85
|
+
;
|
|
86
|
+
globalThis[BYPASS_KEY] = bypass;
|
|
87
|
+
}
|
|
88
|
+
function isVitestProcess() {
|
|
89
|
+
/* v8 ignore next -- defensive: process and process.argv always exist in node @preserve */
|
|
90
|
+
if (typeof process === "undefined" || !Array.isArray(process.argv))
|
|
91
|
+
return false;
|
|
92
|
+
return process.argv.some((arg) => typeof arg === "string" && arg.includes("vitest"));
|
|
93
|
+
}
|
|
94
|
+
function isProductionDaemonSocket(socketPath) {
|
|
95
|
+
return socketPath === exports.DEFAULT_DAEMON_SOCKET_PATH;
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Returns true when this socket call should be converted into a safe no-op
|
|
99
|
+
* because we're under vitest and the call would otherwise leak into a real
|
|
100
|
+
* daemon. The bypass flag short-circuits the guard for non-production socket
|
|
101
|
+
* paths only — production socket paths are ALWAYS blocked under vitest.
|
|
102
|
+
*/
|
|
103
|
+
function shouldSuppressSocketCall(socketPath) {
|
|
104
|
+
if (!isVitestProcess())
|
|
105
|
+
return false;
|
|
106
|
+
if (isProductionDaemonSocket(socketPath))
|
|
107
|
+
return true;
|
|
108
|
+
return globalThis[BYPASS_KEY] !== true;
|
|
109
|
+
}
|
|
110
|
+
function sendDaemonCommand(socketPath, command, options = {}) {
|
|
111
|
+
if (shouldSuppressSocketCall(socketPath)) {
|
|
112
|
+
(0, runtime_1.emitNervesEvent)({
|
|
113
|
+
level: "warn",
|
|
114
|
+
component: "daemon",
|
|
115
|
+
event: "daemon.socket_command_test_blocked",
|
|
116
|
+
message: "blocked socket command from leaking into real daemon under vitest",
|
|
117
|
+
meta: { socketPath, kind: command.kind, isProductionSocket: isProductionDaemonSocket(socketPath) },
|
|
118
|
+
});
|
|
119
|
+
return Promise.resolve({ ok: true, message: "test mode: socket call suppressed" });
|
|
120
|
+
}
|
|
45
121
|
(0, runtime_1.emitNervesEvent)({
|
|
46
122
|
component: "daemon",
|
|
47
123
|
event: "daemon.socket_command_start",
|
|
@@ -51,14 +127,67 @@ function sendDaemonCommand(socketPath, command) {
|
|
|
51
127
|
return new Promise((resolve, reject) => {
|
|
52
128
|
const client = net.createConnection(socketPath);
|
|
53
129
|
let raw = "";
|
|
130
|
+
let settled = false;
|
|
131
|
+
const timeoutMs = options.timeoutMs ?? exports.DEFAULT_DAEMON_COMMAND_TIMEOUT_MS;
|
|
132
|
+
const resolveOnce = (response) => {
|
|
133
|
+
/* v8 ignore next -- duplicate settlement guard; callers also guard event handlers before reaching this helper @preserve */
|
|
134
|
+
if (settled)
|
|
135
|
+
return;
|
|
136
|
+
settled = true;
|
|
137
|
+
resolve(response);
|
|
138
|
+
};
|
|
139
|
+
const rejectOnce = (error) => {
|
|
140
|
+
/* v8 ignore next -- duplicate timeout/error races should not re-reject the command promise @preserve */
|
|
141
|
+
if (settled)
|
|
142
|
+
return;
|
|
143
|
+
settled = true;
|
|
144
|
+
reject(error);
|
|
145
|
+
};
|
|
146
|
+
if ("setTimeout" in client && typeof client.setTimeout === "function") {
|
|
147
|
+
client.setTimeout(timeoutMs, () => {
|
|
148
|
+
const error = new Error(`Daemon command ${command.kind} timed out after ${timeoutMs}ms waiting for a response.`);
|
|
149
|
+
(0, runtime_1.emitNervesEvent)({
|
|
150
|
+
level: "error",
|
|
151
|
+
component: "daemon",
|
|
152
|
+
event: "daemon.socket_command_timeout",
|
|
153
|
+
message: "daemon socket command timed out",
|
|
154
|
+
meta: {
|
|
155
|
+
socketPath,
|
|
156
|
+
kind: command.kind,
|
|
157
|
+
timeoutMs,
|
|
158
|
+
error: error.message,
|
|
159
|
+
},
|
|
160
|
+
});
|
|
161
|
+
client.destroy();
|
|
162
|
+
rejectOnce(error);
|
|
163
|
+
});
|
|
164
|
+
}
|
|
54
165
|
client.on("connect", () => {
|
|
55
|
-
client.
|
|
56
|
-
|
|
166
|
+
// Write the command + newline delimiter. DO NOT call client.end()
|
|
167
|
+
// afterwards — the server closes the connection once it has written
|
|
168
|
+
// its response, which triggers our on("end") handler with the full
|
|
169
|
+
// response in `raw`.
|
|
170
|
+
//
|
|
171
|
+
// Calling client.end() here half-closes the TCP connection. The
|
|
172
|
+
// daemon server uses net.createServer() with the default
|
|
173
|
+
// allowHalfOpen: false, so when the server sees the client's FIN it
|
|
174
|
+
// auto-closes its own writable side — and any response the server
|
|
175
|
+
// tries to write after processing a long-running command (like
|
|
176
|
+
// agent.senseTurn, which runs a full LLM turn) gets dropped on the
|
|
177
|
+
// floor. Verified via repro: with client.end(), agent.senseTurn
|
|
178
|
+
// returns empty in ~149ms; without it, the same call returns the real
|
|
179
|
+
// response in ~5.8s. This is a regression of the fix in #303 that
|
|
180
|
+
// was silently reverted by 253e4b1f (commit titled "socket half-close
|
|
181
|
+
// fix" but actually *added* the half-close back).
|
|
182
|
+
client.write(JSON.stringify(command) + "\n");
|
|
57
183
|
});
|
|
58
184
|
client.on("data", (chunk) => {
|
|
59
185
|
raw += chunk.toString("utf-8");
|
|
60
186
|
});
|
|
61
187
|
client.on("error", (error) => {
|
|
188
|
+
/* v8 ignore next -- duplicate post-settlement socket errors are ignored defensively @preserve */
|
|
189
|
+
if (settled)
|
|
190
|
+
return;
|
|
62
191
|
(0, runtime_1.emitNervesEvent)({
|
|
63
192
|
level: "error",
|
|
64
193
|
component: "daemon",
|
|
@@ -70,9 +199,12 @@ function sendDaemonCommand(socketPath, command) {
|
|
|
70
199
|
error: error.message,
|
|
71
200
|
},
|
|
72
201
|
});
|
|
73
|
-
|
|
202
|
+
rejectOnce(error);
|
|
74
203
|
});
|
|
75
204
|
client.on("end", () => {
|
|
205
|
+
/* v8 ignore next -- duplicate post-settlement socket end events are ignored defensively @preserve */
|
|
206
|
+
if (settled)
|
|
207
|
+
return;
|
|
76
208
|
const trimmed = raw.trim();
|
|
77
209
|
if (trimmed.length === 0 && command.kind === "daemon.stop") {
|
|
78
210
|
(0, runtime_1.emitNervesEvent)({
|
|
@@ -81,7 +213,7 @@ function sendDaemonCommand(socketPath, command) {
|
|
|
81
213
|
message: "daemon socket command completed",
|
|
82
214
|
meta: { socketPath, kind: command.kind, ok: true },
|
|
83
215
|
});
|
|
84
|
-
|
|
216
|
+
resolveOnce({ ok: true, message: "daemon stopped" });
|
|
85
217
|
return;
|
|
86
218
|
}
|
|
87
219
|
if (trimmed.length === 0) {
|
|
@@ -97,7 +229,7 @@ function sendDaemonCommand(socketPath, command) {
|
|
|
97
229
|
error: error.message,
|
|
98
230
|
},
|
|
99
231
|
});
|
|
100
|
-
|
|
232
|
+
rejectOnce(error);
|
|
101
233
|
return;
|
|
102
234
|
}
|
|
103
235
|
try {
|
|
@@ -112,7 +244,7 @@ function sendDaemonCommand(socketPath, command) {
|
|
|
112
244
|
ok: parsed.ok,
|
|
113
245
|
},
|
|
114
246
|
});
|
|
115
|
-
|
|
247
|
+
resolveOnce(parsed);
|
|
116
248
|
}
|
|
117
249
|
catch (error) {
|
|
118
250
|
(0, runtime_1.emitNervesEvent)({
|
|
@@ -126,12 +258,15 @@ function sendDaemonCommand(socketPath, command) {
|
|
|
126
258
|
error: error instanceof Error ? error.message : String(error),
|
|
127
259
|
},
|
|
128
260
|
});
|
|
129
|
-
|
|
261
|
+
rejectOnce(error);
|
|
130
262
|
}
|
|
131
263
|
});
|
|
132
264
|
});
|
|
133
265
|
}
|
|
134
266
|
function checkDaemonSocketAlive(socketPath) {
|
|
267
|
+
if (shouldSuppressSocketCall(socketPath)) {
|
|
268
|
+
return Promise.resolve(false);
|
|
269
|
+
}
|
|
135
270
|
(0, runtime_1.emitNervesEvent)({
|
|
136
271
|
component: "daemon",
|
|
137
272
|
event: "daemon.socket_alive_check_start",
|
|
@@ -161,8 +296,10 @@ function checkDaemonSocketAlive(socketPath) {
|
|
|
161
296
|
});
|
|
162
297
|
}
|
|
163
298
|
client.on("connect", () => {
|
|
164
|
-
|
|
165
|
-
client.end()
|
|
299
|
+
// Same half-close rationale as sendDaemonCommand: do NOT call
|
|
300
|
+
// client.end() here. The server closes after responding and that
|
|
301
|
+
// triggers the on("end") handler below.
|
|
302
|
+
client.write(JSON.stringify({ kind: "daemon.status" }) + "\n");
|
|
166
303
|
});
|
|
167
304
|
client.on("data", (chunk) => {
|
|
168
305
|
raw += chunk.toString("utf-8");
|
|
@@ -184,6 +321,16 @@ function checkDaemonSocketAlive(socketPath) {
|
|
|
184
321
|
});
|
|
185
322
|
}
|
|
186
323
|
async function requestInnerWake(agent, socketPath = exports.DEFAULT_DAEMON_SOCKET_PATH) {
|
|
324
|
+
if (shouldSuppressSocketCall(socketPath)) {
|
|
325
|
+
(0, runtime_1.emitNervesEvent)({
|
|
326
|
+
level: "warn",
|
|
327
|
+
component: "daemon",
|
|
328
|
+
event: "daemon.inner_wake_test_blocked",
|
|
329
|
+
message: "blocked inner wake from leaking into real daemon under vitest",
|
|
330
|
+
meta: { agent, socketPath, isProductionSocket: isProductionDaemonSocket(socketPath) },
|
|
331
|
+
});
|
|
332
|
+
return null;
|
|
333
|
+
}
|
|
187
334
|
const socketAvailable = fs.existsSync(socketPath);
|
|
188
335
|
(0, runtime_1.emitNervesEvent)({
|
|
189
336
|
component: "daemon",
|
|
@@ -0,0 +1,96 @@
|
|
|
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.pruneStaleEphemeralBundles = pruneStaleEphemeralBundles;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
const identity_1 = require("../identity");
|
|
40
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
41
|
+
/**
|
|
42
|
+
* Scan the bundles root for `.ouro` directories that have no `agent.json`
|
|
43
|
+
* file (definitively dead ephemeral bundles) and delete them.
|
|
44
|
+
*
|
|
45
|
+
* Returns a list of pruned bundle directory names (e.g. `["stale.ouro"]`)
|
|
46
|
+
* for display purposes. Bundles that have `agent.json` -- even if disabled
|
|
47
|
+
* -- are never deleted. Errors on individual bundles are swallowed so that
|
|
48
|
+
* one permission-denied doesn't block pruning the rest.
|
|
49
|
+
*/
|
|
50
|
+
function pruneStaleEphemeralBundles(deps = {}) {
|
|
51
|
+
const bundlesRoot = deps.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
|
|
52
|
+
const readdirSync = deps.readdirSync ?? fs.readdirSync;
|
|
53
|
+
const existsSync = deps.existsSync ?? fs.existsSync;
|
|
54
|
+
const rmSync = deps.rmSync ?? fs.rmSync;
|
|
55
|
+
let entries;
|
|
56
|
+
try {
|
|
57
|
+
entries = readdirSync(bundlesRoot, { withFileTypes: true });
|
|
58
|
+
}
|
|
59
|
+
catch {
|
|
60
|
+
return [];
|
|
61
|
+
}
|
|
62
|
+
const pruned = [];
|
|
63
|
+
for (const entry of entries) {
|
|
64
|
+
if (!entry.isDirectory() || !entry.name.endsWith(".ouro"))
|
|
65
|
+
continue;
|
|
66
|
+
const bundlePath = path.join(bundlesRoot, entry.name);
|
|
67
|
+
const agentJsonPath = path.join(bundlePath, "agent.json");
|
|
68
|
+
if (existsSync(agentJsonPath))
|
|
69
|
+
continue;
|
|
70
|
+
try {
|
|
71
|
+
rmSync(bundlePath, { recursive: true, force: true });
|
|
72
|
+
pruned.push(entry.name);
|
|
73
|
+
(0, runtime_1.emitNervesEvent)({
|
|
74
|
+
level: "info",
|
|
75
|
+
component: "daemon",
|
|
76
|
+
event: "daemon.stale_bundle_pruned",
|
|
77
|
+
message: `pruned stale ephemeral bundle: ${entry.name}`,
|
|
78
|
+
meta: { bundle: entry.name, bundlePath },
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
catch (error) {
|
|
82
|
+
(0, runtime_1.emitNervesEvent)({
|
|
83
|
+
level: "warn",
|
|
84
|
+
component: "daemon",
|
|
85
|
+
event: "daemon.stale_bundle_prune_error",
|
|
86
|
+
message: `failed to prune stale bundle: ${entry.name}`,
|
|
87
|
+
meta: {
|
|
88
|
+
bundle: entry.name,
|
|
89
|
+
bundlePath,
|
|
90
|
+
error: error instanceof Error ? error.message : String(error),
|
|
91
|
+
},
|
|
92
|
+
});
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
return pruned;
|
|
96
|
+
}
|