@ouro.bot/cli 0.1.0-alpha.56 → 0.1.0-alpha.561
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 +3604 -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 +251 -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/audio-routing.js +119 -0
- package/dist/senses/voice/elevenlabs.js +178 -0
- package/dist/senses/voice/golden-path.js +116 -0
- package/dist/senses/voice/index.js +26 -0
- package/dist/senses/voice/meeting.js +113 -0
- package/dist/senses/voice/playback.js +139 -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 +161 -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
|
@@ -38,31 +38,16 @@ async function deliverCrossChatMessage(request, deps) {
|
|
|
38
38
|
authorizingTrustLevel: request.authorizingSession?.trustLevel ?? null,
|
|
39
39
|
},
|
|
40
40
|
});
|
|
41
|
-
if (request.intent
|
|
42
|
-
const result = queueForLater(request, deps, "generic outreach stays queued until the target session is next active");
|
|
43
|
-
(0, runtime_1.emitNervesEvent)({
|
|
44
|
-
component: "engine",
|
|
45
|
-
event: "engine.cross_chat_delivery_end",
|
|
46
|
-
message: "queued generic outreach for later delivery",
|
|
47
|
-
meta: {
|
|
48
|
-
friendId: request.friendId,
|
|
49
|
-
channel: request.channel,
|
|
50
|
-
key: request.key,
|
|
51
|
-
status: result.status,
|
|
52
|
-
},
|
|
53
|
-
});
|
|
54
|
-
return result;
|
|
55
|
-
}
|
|
56
|
-
if (!isExplicitlyAuthorized(request)) {
|
|
41
|
+
if (!isExplicitlyAuthorized(request) && request.intent !== "generic_outreach") {
|
|
57
42
|
const result = {
|
|
58
43
|
status: "blocked",
|
|
59
|
-
detail: "
|
|
44
|
+
detail: "cross-chat delivery requires a trusted asking session or generic outreach intent",
|
|
60
45
|
};
|
|
61
46
|
(0, runtime_1.emitNervesEvent)({
|
|
62
47
|
level: "warn",
|
|
63
48
|
component: "engine",
|
|
64
49
|
event: "engine.cross_chat_delivery_end",
|
|
65
|
-
message: "blocked
|
|
50
|
+
message: "blocked cross-chat delivery",
|
|
66
51
|
meta: {
|
|
67
52
|
friendId: request.friendId,
|
|
68
53
|
channel: request.channel,
|
|
@@ -0,0 +1,419 @@
|
|
|
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.vaultUnlockOrRecoverFix = vaultUnlockOrRecoverFix;
|
|
37
|
+
exports.checkAgentConfig = checkAgentConfig;
|
|
38
|
+
exports.checkAgentConfigWithProviderHealth = checkAgentConfigWithProviderHealth;
|
|
39
|
+
const fs = __importStar(require("fs"));
|
|
40
|
+
const path = __importStar(require("path"));
|
|
41
|
+
const identity_1 = require("../identity");
|
|
42
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
43
|
+
const provider_models_1 = require("../provider-models");
|
|
44
|
+
const provider_credentials_1 = require("../provider-credentials");
|
|
45
|
+
const provider_readiness_cache_1 = require("../provider-readiness-cache");
|
|
46
|
+
const vault_unlock_1 = require("../../repertoire/vault-unlock");
|
|
47
|
+
const readiness_repair_1 = require("./readiness-repair");
|
|
48
|
+
const provider_ping_progress_1 = require("./provider-ping-progress");
|
|
49
|
+
const LANES = [
|
|
50
|
+
{ lane: "outward", facing: "humanFacing" },
|
|
51
|
+
{ lane: "inner", facing: "agentFacing" },
|
|
52
|
+
];
|
|
53
|
+
function isAgentProvider(value) {
|
|
54
|
+
return Object.prototype.hasOwnProperty.call(identity_1.PROVIDER_CREDENTIALS, value);
|
|
55
|
+
}
|
|
56
|
+
function agentRootFor(agentName, bundlesRoot) {
|
|
57
|
+
return path.join(bundlesRoot, `${agentName}.ouro`);
|
|
58
|
+
}
|
|
59
|
+
function configPathFor(agentName, bundlesRoot) {
|
|
60
|
+
return path.join(agentRootFor(agentName, bundlesRoot), "agent.json");
|
|
61
|
+
}
|
|
62
|
+
function laneForFacing(facing) {
|
|
63
|
+
return facing === "humanFacing" ? "outward" : "inner";
|
|
64
|
+
}
|
|
65
|
+
function resolveFacingProvider(parsed, facing, agentName, agentJsonPath) {
|
|
66
|
+
const raw = parsed[facing];
|
|
67
|
+
if (!raw || typeof raw !== "object" || Array.isArray(raw)) {
|
|
68
|
+
return {
|
|
69
|
+
ok: false,
|
|
70
|
+
result: {
|
|
71
|
+
ok: false,
|
|
72
|
+
error: `agent.json for '${agentName}' is missing ${facing}.provider`,
|
|
73
|
+
fix: `Add ${facing}: { provider, model } to ${agentJsonPath}. Valid providers: ${Object.keys(identity_1.PROVIDER_CREDENTIALS).join(", ")}`,
|
|
74
|
+
},
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
const provider = raw.provider;
|
|
78
|
+
if (typeof provider !== "string" || provider.length === 0) {
|
|
79
|
+
return {
|
|
80
|
+
ok: false,
|
|
81
|
+
result: {
|
|
82
|
+
ok: false,
|
|
83
|
+
error: `agent.json for '${agentName}' is missing ${facing}.provider`,
|
|
84
|
+
fix: `Set ${facing}.provider in ${agentJsonPath}. Valid providers: ${Object.keys(identity_1.PROVIDER_CREDENTIALS).join(", ")}`,
|
|
85
|
+
},
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
if (!isAgentProvider(provider)) {
|
|
89
|
+
return {
|
|
90
|
+
ok: false,
|
|
91
|
+
result: {
|
|
92
|
+
ok: false,
|
|
93
|
+
error: `Unknown provider '${provider}' in ${facing}.provider for '${agentName}'`,
|
|
94
|
+
fix: `Set ${facing}.provider to one of: ${Object.keys(identity_1.PROVIDER_CREDENTIALS).join(", ")}`,
|
|
95
|
+
},
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
const rawModel = raw.model;
|
|
99
|
+
const model = typeof rawModel === "string" && rawModel.trim().length > 0
|
|
100
|
+
? rawModel.trim()
|
|
101
|
+
: (0, provider_models_1.getDefaultModelForProvider)(provider);
|
|
102
|
+
return { ok: true, selected: { facing, lane: laneForFacing(facing), provider, model } };
|
|
103
|
+
}
|
|
104
|
+
function readAgentConfigForProviderCheck(agentName, bundlesRoot) {
|
|
105
|
+
const agentJsonPath = configPathFor(agentName, bundlesRoot);
|
|
106
|
+
let raw;
|
|
107
|
+
try {
|
|
108
|
+
raw = fs.readFileSync(agentJsonPath, "utf-8");
|
|
109
|
+
}
|
|
110
|
+
catch {
|
|
111
|
+
return {
|
|
112
|
+
ok: false,
|
|
113
|
+
result: {
|
|
114
|
+
ok: false,
|
|
115
|
+
error: `agent.json not found at ${agentJsonPath}`,
|
|
116
|
+
fix: `Run 'ouro hatch ${agentName}' to create the agent bundle, or verify that ${bundlesRoot}/${agentName}.ouro/ exists.`,
|
|
117
|
+
},
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
let parsed;
|
|
121
|
+
try {
|
|
122
|
+
parsed = JSON.parse(raw);
|
|
123
|
+
}
|
|
124
|
+
catch {
|
|
125
|
+
return {
|
|
126
|
+
ok: false,
|
|
127
|
+
result: {
|
|
128
|
+
ok: false,
|
|
129
|
+
error: `agent.json at ${agentJsonPath} contains invalid JSON`,
|
|
130
|
+
fix: `Open ${agentJsonPath} and fix the JSON syntax.`,
|
|
131
|
+
},
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
if (parsed.enabled === false) {
|
|
135
|
+
return { ok: true, disabled: true, agentJsonPath, parsed };
|
|
136
|
+
}
|
|
137
|
+
return { ok: true, disabled: false, agentJsonPath, parsed };
|
|
138
|
+
}
|
|
139
|
+
function readProviderSelectionForCheck(agentName, bundlesRoot) {
|
|
140
|
+
const configResult = readAgentConfigForProviderCheck(agentName, bundlesRoot);
|
|
141
|
+
if (!configResult.ok)
|
|
142
|
+
return { ok: false, result: configResult.result };
|
|
143
|
+
if (configResult.disabled)
|
|
144
|
+
return { ok: true, disabled: true };
|
|
145
|
+
const bindings = {};
|
|
146
|
+
for (const { lane, facing } of LANES) {
|
|
147
|
+
const selected = resolveFacingProvider(configResult.parsed, facing, agentName, configResult.agentJsonPath);
|
|
148
|
+
if (!selected.ok)
|
|
149
|
+
return { ok: false, result: selected.result };
|
|
150
|
+
bindings[lane] = selected.selected;
|
|
151
|
+
}
|
|
152
|
+
return {
|
|
153
|
+
ok: true,
|
|
154
|
+
disabled: false,
|
|
155
|
+
agentRoot: agentRootFor(agentName, bundlesRoot),
|
|
156
|
+
bindings,
|
|
157
|
+
};
|
|
158
|
+
}
|
|
159
|
+
function providerCredentialConfig(record) {
|
|
160
|
+
return {
|
|
161
|
+
...record.credentials,
|
|
162
|
+
...record.config,
|
|
163
|
+
};
|
|
164
|
+
}
|
|
165
|
+
function pingAttemptCount(result) {
|
|
166
|
+
if (Array.isArray(result.attempts))
|
|
167
|
+
return result.attempts.length;
|
|
168
|
+
return undefined;
|
|
169
|
+
}
|
|
170
|
+
function missingCredentialResult(agentName, lane, provider, model, credentialPath) {
|
|
171
|
+
return {
|
|
172
|
+
ok: false,
|
|
173
|
+
error: `${lane} provider ${provider} model ${model} has no credentials in ${agentName}'s vault at ${credentialPath}`,
|
|
174
|
+
fix: `Run 'ouro auth --agent ${agentName} --provider ${provider}' to authenticate.`,
|
|
175
|
+
issue: (0, readiness_repair_1.providerCredentialMissingIssue)({
|
|
176
|
+
agentName,
|
|
177
|
+
lane,
|
|
178
|
+
provider,
|
|
179
|
+
model,
|
|
180
|
+
credentialPath,
|
|
181
|
+
}),
|
|
182
|
+
};
|
|
183
|
+
}
|
|
184
|
+
function isTransientVaultError(error) {
|
|
185
|
+
const normalized = error.toLowerCase();
|
|
186
|
+
return (normalized.includes("timed out") ||
|
|
187
|
+
normalized.includes("econnrefused") ||
|
|
188
|
+
normalized.includes("socket hang up") ||
|
|
189
|
+
normalized.includes("etimedout"));
|
|
190
|
+
}
|
|
191
|
+
function invalidPoolResult(agentName, lane, provider, model, pool) {
|
|
192
|
+
if (pool.reason === "unavailable" && isTransientVaultError(pool.error)) {
|
|
193
|
+
return {
|
|
194
|
+
ok: false,
|
|
195
|
+
error: `${lane} provider ${provider} model ${model} cannot read provider credentials from ${agentName}'s vault: ${pool.error}`,
|
|
196
|
+
fix: `Vault read timed out -- this usually resolves on retry. Run 'ouro up' again.`,
|
|
197
|
+
};
|
|
198
|
+
}
|
|
199
|
+
if (pool.reason === "unavailable" && isVaultLockedError(pool.error)) {
|
|
200
|
+
return {
|
|
201
|
+
ok: false,
|
|
202
|
+
error: `${lane} provider ${provider} model ${model} cannot read provider credentials because ${agentName}'s credential vault is locked on this machine.`,
|
|
203
|
+
fix: vaultUnlockOrRecoverFix(agentName),
|
|
204
|
+
issue: (0, readiness_repair_1.vaultLockedIssue)(agentName),
|
|
205
|
+
};
|
|
206
|
+
}
|
|
207
|
+
if (pool.reason === "unavailable" && (0, vault_unlock_1.isCredentialVaultNotConfiguredError)(pool.error)) {
|
|
208
|
+
return {
|
|
209
|
+
ok: false,
|
|
210
|
+
error: `${lane} provider ${provider} model ${model} cannot read provider credentials because ${agentName}'s credential vault is not configured in agent.json.`,
|
|
211
|
+
fix: (0, vault_unlock_1.vaultCreateRecoverFix)(agentName, `Then run 'ouro auth --agent ${agentName} --provider ${provider}' and rerun 'ouro up'.`),
|
|
212
|
+
issue: (0, readiness_repair_1.vaultUnconfiguredIssue)(agentName),
|
|
213
|
+
};
|
|
214
|
+
}
|
|
215
|
+
if (pool.reason === "invalid") {
|
|
216
|
+
return {
|
|
217
|
+
ok: false,
|
|
218
|
+
error: `${lane} provider ${provider} model ${model} cannot read provider credentials from ${agentName}'s vault at ${pool.poolPath}: ${pool.error}`,
|
|
219
|
+
fix: `Run 'ouro auth --agent ${agentName} --provider ${provider}' to rewrite this provider credential, then run 'ouro up' again.`,
|
|
220
|
+
};
|
|
221
|
+
}
|
|
222
|
+
return {
|
|
223
|
+
ok: false,
|
|
224
|
+
error: `${lane} provider ${provider} model ${model} cannot read provider credentials from ${agentName}'s vault at ${pool.poolPath}: ${pool.error}`,
|
|
225
|
+
fix: vaultUnlockOrRecoverFix(agentName, `Then run 'ouro up' again. If the credential is missing or stale after unlock or recovery, run 'ouro auth --agent ${agentName} --provider ${provider}'.`),
|
|
226
|
+
};
|
|
227
|
+
}
|
|
228
|
+
function isVaultLockedError(error) {
|
|
229
|
+
const normalized = error.toLowerCase();
|
|
230
|
+
return /(?:ouro )?credential vault is locked|vault(?: is)? locked/.test(normalized);
|
|
231
|
+
}
|
|
232
|
+
function vaultUnlockOrRecoverFix(agentName, nextStep = "Then run 'ouro up' again.") {
|
|
233
|
+
return (0, vault_unlock_1.vaultUnlockReplaceRecoverFix)(agentName, nextStep);
|
|
234
|
+
}
|
|
235
|
+
function failedPingResult(agentName, lane, provider, model, result) {
|
|
236
|
+
return {
|
|
237
|
+
ok: false,
|
|
238
|
+
error: `${lane} provider ${provider} model ${model} failed live check: ${result.message}`,
|
|
239
|
+
fix: (0, readiness_repair_1.providerLiveCheckFix)({
|
|
240
|
+
agentName,
|
|
241
|
+
lane,
|
|
242
|
+
provider,
|
|
243
|
+
classification: result.classification,
|
|
244
|
+
}),
|
|
245
|
+
issue: (0, readiness_repair_1.providerLiveCheckFailedIssue)({
|
|
246
|
+
agentName,
|
|
247
|
+
lane,
|
|
248
|
+
provider,
|
|
249
|
+
model,
|
|
250
|
+
classification: result.classification,
|
|
251
|
+
message: result.message,
|
|
252
|
+
}),
|
|
253
|
+
};
|
|
254
|
+
}
|
|
255
|
+
function credentialRecordForLane(pool, provider) {
|
|
256
|
+
return pool.providers[provider];
|
|
257
|
+
}
|
|
258
|
+
function laneAudienceLabel(lane) {
|
|
259
|
+
return lane === "outward" ? "chat" : "inner dialog";
|
|
260
|
+
}
|
|
261
|
+
function bindingLabel(binding) {
|
|
262
|
+
return `${binding.provider} / ${binding.model}`;
|
|
263
|
+
}
|
|
264
|
+
function selectedProviderPlan(agentName, bindings) {
|
|
265
|
+
return [
|
|
266
|
+
`${agentName}: checking the providers in agent.json`,
|
|
267
|
+
...LANES.map(({ lane }) => `- ${laneAudienceLabel(lane)}: ${bindingLabel(bindings[lane])}`),
|
|
268
|
+
].join("\n");
|
|
269
|
+
}
|
|
270
|
+
function selectedProvidersForBindings(bindings) {
|
|
271
|
+
return [...new Set(LANES.map(({ lane }) => bindings[lane].provider))];
|
|
272
|
+
}
|
|
273
|
+
function mapVaultRefreshProgress(agentName, onProgress) {
|
|
274
|
+
return (message) => {
|
|
275
|
+
if (message.startsWith("reading vault items for ")) {
|
|
276
|
+
onProgress(`${agentName}: opening saved provider credentials in the vault`);
|
|
277
|
+
return;
|
|
278
|
+
}
|
|
279
|
+
const providerRead = message.match(/^reading ([a-z0-9-]+) credentials\.\.\.$/i);
|
|
280
|
+
if (providerRead) {
|
|
281
|
+
onProgress(`${agentName}: reading saved ${providerRead[1]} credentials`);
|
|
282
|
+
return;
|
|
283
|
+
}
|
|
284
|
+
if (message === "parsing provider credentials...") {
|
|
285
|
+
onProgress(`${agentName}: organizing saved provider credentials`);
|
|
286
|
+
}
|
|
287
|
+
};
|
|
288
|
+
}
|
|
289
|
+
function providerPingSubject(agentName, lanes) {
|
|
290
|
+
const laneList = lanes.map((lane) => laneAudienceLabel(lane)).join(" + ");
|
|
291
|
+
return `${agentName} (${laneList})`;
|
|
292
|
+
}
|
|
293
|
+
/**
|
|
294
|
+
* Structural validation only. Live provider credential validation belongs to
|
|
295
|
+
* checkAgentConfigWithProviderHealth(), which reads the agent vault and pings.
|
|
296
|
+
*/
|
|
297
|
+
function checkAgentConfig(agentName, bundlesRoot) {
|
|
298
|
+
const selectionResult = readProviderSelectionForCheck(agentName, bundlesRoot);
|
|
299
|
+
if (!selectionResult.ok)
|
|
300
|
+
return selectionResult.result;
|
|
301
|
+
if (selectionResult.disabled)
|
|
302
|
+
return { ok: true };
|
|
303
|
+
(0, runtime_1.emitNervesEvent)({
|
|
304
|
+
component: "daemon",
|
|
305
|
+
event: "daemon.agent_config_valid",
|
|
306
|
+
message: "agent config validation passed",
|
|
307
|
+
meta: {
|
|
308
|
+
agent: agentName,
|
|
309
|
+
providers: selectedProvidersForBindings(selectionResult.bindings),
|
|
310
|
+
liveProviderCheck: false,
|
|
311
|
+
},
|
|
312
|
+
});
|
|
313
|
+
return { ok: true };
|
|
314
|
+
}
|
|
315
|
+
async function checkAgentConfigWithProviderHealth(agentName, bundlesRoot, deps = {}) {
|
|
316
|
+
const selectionResult = readProviderSelectionForCheck(agentName, bundlesRoot);
|
|
317
|
+
if (!selectionResult.ok)
|
|
318
|
+
return selectionResult.result;
|
|
319
|
+
if (selectionResult.disabled)
|
|
320
|
+
return { ok: true };
|
|
321
|
+
deps.onProgress?.(selectedProviderPlan(agentName, selectionResult.bindings));
|
|
322
|
+
const ping = deps.pingProvider ?? (await Promise.resolve().then(() => __importStar(require("../provider-ping")))).pingProvider;
|
|
323
|
+
const providers = selectedProvidersForBindings(selectionResult.bindings);
|
|
324
|
+
const poolResult = await (0, provider_credentials_1.refreshProviderCredentialPool)(agentName, {
|
|
325
|
+
...(deps.onProgress ? { onProgress: mapVaultRefreshProgress(agentName, deps.onProgress) } : {}),
|
|
326
|
+
providers,
|
|
327
|
+
preserveCachedOnFailure: true,
|
|
328
|
+
});
|
|
329
|
+
const pingGroups = new Map();
|
|
330
|
+
for (const { lane } of LANES) {
|
|
331
|
+
const binding = selectionResult.bindings[lane];
|
|
332
|
+
if (!poolResult.ok) {
|
|
333
|
+
if (poolResult.reason === "missing") {
|
|
334
|
+
return missingCredentialResult(agentName, lane, binding.provider, binding.model, poolResult.poolPath);
|
|
335
|
+
}
|
|
336
|
+
return invalidPoolResult(agentName, lane, binding.provider, binding.model, {
|
|
337
|
+
...poolResult,
|
|
338
|
+
reason: poolResult.reason,
|
|
339
|
+
});
|
|
340
|
+
}
|
|
341
|
+
const record = credentialRecordForLane(poolResult.pool, binding.provider);
|
|
342
|
+
if (!record) {
|
|
343
|
+
return missingCredentialResult(agentName, lane, binding.provider, binding.model, poolResult.poolPath);
|
|
344
|
+
}
|
|
345
|
+
const key = `${binding.provider}\0${binding.model}\0${record.revision}`;
|
|
346
|
+
const group = pingGroups.get(key);
|
|
347
|
+
if (group) {
|
|
348
|
+
group.lanes.push(lane);
|
|
349
|
+
}
|
|
350
|
+
else {
|
|
351
|
+
pingGroups.set(key, {
|
|
352
|
+
provider: binding.provider,
|
|
353
|
+
model: binding.model,
|
|
354
|
+
record,
|
|
355
|
+
lanes: [lane],
|
|
356
|
+
});
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
const groups = [...pingGroups.values()];
|
|
360
|
+
const pingResults = await Promise.all(groups.map(async (group) => {
|
|
361
|
+
const result = await ping(group.provider, providerCredentialConfig(group.record), {
|
|
362
|
+
model: group.model,
|
|
363
|
+
...(deps.providerPingOptions ?? {}),
|
|
364
|
+
...(deps.onProgress
|
|
365
|
+
? (0, provider_ping_progress_1.createProviderPingProgressReporter)({
|
|
366
|
+
provider: group.provider,
|
|
367
|
+
model: group.model,
|
|
368
|
+
subject: providerPingSubject(agentName, group.lanes),
|
|
369
|
+
}, deps.onProgress)
|
|
370
|
+
: {}),
|
|
371
|
+
});
|
|
372
|
+
return { group, result };
|
|
373
|
+
}));
|
|
374
|
+
let firstFailure = null;
|
|
375
|
+
for (const { group, result } of pingResults) {
|
|
376
|
+
if (!result.ok) {
|
|
377
|
+
for (const lane of group.lanes) {
|
|
378
|
+
(0, provider_readiness_cache_1.recordProviderLaneReadiness)({
|
|
379
|
+
agentName,
|
|
380
|
+
lane,
|
|
381
|
+
provider: group.provider,
|
|
382
|
+
model: group.model,
|
|
383
|
+
credentialRevision: group.record.revision,
|
|
384
|
+
status: "failed",
|
|
385
|
+
checkedAt: new Date().toISOString(),
|
|
386
|
+
error: result.message,
|
|
387
|
+
attempts: pingAttemptCount(result),
|
|
388
|
+
});
|
|
389
|
+
}
|
|
390
|
+
firstFailure ??= failedPingResult(agentName, group.lanes[0], group.provider, group.model, result);
|
|
391
|
+
continue;
|
|
392
|
+
}
|
|
393
|
+
for (const lane of group.lanes) {
|
|
394
|
+
(0, provider_readiness_cache_1.recordProviderLaneReadiness)({
|
|
395
|
+
agentName,
|
|
396
|
+
lane,
|
|
397
|
+
provider: group.provider,
|
|
398
|
+
model: group.model,
|
|
399
|
+
credentialRevision: group.record.revision,
|
|
400
|
+
status: "ready",
|
|
401
|
+
checkedAt: new Date().toISOString(),
|
|
402
|
+
attempts: pingAttemptCount(result),
|
|
403
|
+
});
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
if (firstFailure)
|
|
407
|
+
return firstFailure;
|
|
408
|
+
(0, runtime_1.emitNervesEvent)({
|
|
409
|
+
component: "daemon",
|
|
410
|
+
event: "daemon.agent_config_valid",
|
|
411
|
+
message: "agent config validation passed",
|
|
412
|
+
meta: {
|
|
413
|
+
agent: agentName,
|
|
414
|
+
providers: [...new Set([...pingGroups.values()].map((group) => group.provider))],
|
|
415
|
+
liveProviderCheck: true,
|
|
416
|
+
},
|
|
417
|
+
});
|
|
418
|
+
return { ok: true };
|
|
419
|
+
}
|
|
@@ -33,12 +33,34 @@ var __importStar = (this && this.__importStar) || (function () {
|
|
|
33
33
|
};
|
|
34
34
|
})();
|
|
35
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.isLibraryKind = isLibraryKind;
|
|
37
|
+
exports.listAllBundleAgents = listAllBundleAgents;
|
|
36
38
|
exports.listEnabledBundleAgents = listEnabledBundleAgents;
|
|
39
|
+
exports.listBundleSyncRows = listBundleSyncRows;
|
|
37
40
|
const fs = __importStar(require("fs"));
|
|
38
41
|
const path = __importStar(require("path"));
|
|
42
|
+
const child_process_1 = require("child_process");
|
|
39
43
|
const identity_1 = require("../identity");
|
|
40
44
|
const runtime_1 = require("../../nerves/runtime");
|
|
41
|
-
|
|
45
|
+
/**
|
|
46
|
+
* True when the value is the string `"library"`. Library bundles are
|
|
47
|
+
* content-only resources — never run as agents, never appear in sync surfaces.
|
|
48
|
+
*/
|
|
49
|
+
function isLibraryKind(kind) {
|
|
50
|
+
return kind === "library";
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* Walk the bundles root and return one row per `<name>.ouro` directory whose
|
|
54
|
+
* `agent.json` is readable and parseable. Includes both enabled and disabled
|
|
55
|
+
* agents AND library-kind bundles — callers that need only real agents should
|
|
56
|
+
* use `listEnabledBundleAgents` (which filters both `enabled` and `kind`).
|
|
57
|
+
*
|
|
58
|
+
* Bundles whose `agent.json` is missing, malformed, or unreadable are skipped
|
|
59
|
+
* silently (they aren't real agents from the harness's perspective).
|
|
60
|
+
*
|
|
61
|
+
* Sorted alphabetically by name for stable display.
|
|
62
|
+
*/
|
|
63
|
+
function listAllBundleAgents(options = {}) {
|
|
42
64
|
const bundlesRoot = options.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
|
|
43
65
|
const readdirSync = options.readdirSync ?? fs.readdirSync;
|
|
44
66
|
const readFileSync = options.readFileSync ?? fs.readFileSync;
|
|
@@ -63,19 +85,96 @@ function listEnabledBundleAgents(options = {}) {
|
|
|
63
85
|
const agentName = entry.name.slice(0, -5);
|
|
64
86
|
const configPath = path.join(bundlesRoot, entry.name, "agent.json");
|
|
65
87
|
let enabled = true;
|
|
88
|
+
let kind;
|
|
66
89
|
try {
|
|
67
90
|
const raw = readFileSync(configPath, "utf-8");
|
|
68
91
|
const parsed = JSON.parse(raw);
|
|
69
92
|
if (typeof parsed.enabled === "boolean") {
|
|
70
93
|
enabled = parsed.enabled;
|
|
71
94
|
}
|
|
95
|
+
if (typeof parsed.kind === "string") {
|
|
96
|
+
kind = parsed.kind;
|
|
97
|
+
}
|
|
72
98
|
}
|
|
73
99
|
catch {
|
|
74
100
|
continue;
|
|
75
101
|
}
|
|
102
|
+
const row = { name: agentName, enabled };
|
|
103
|
+
if (kind !== undefined)
|
|
104
|
+
row.kind = kind;
|
|
105
|
+
discovered.push(row);
|
|
106
|
+
}
|
|
107
|
+
return discovered.sort((left, right) => left.name.localeCompare(right.name));
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Real agents only — excludes both disabled bundles and library-kind bundles.
|
|
111
|
+
* Library bundles (SerpentGuide, RepairGuide, …) are content-only and must
|
|
112
|
+
* never appear in spawn lists, status rollups, or sync rows.
|
|
113
|
+
*/
|
|
114
|
+
function listEnabledBundleAgents(options = {}) {
|
|
115
|
+
return listAllBundleAgents(options)
|
|
116
|
+
.filter((row) => row.enabled && !isLibraryKind(row.kind))
|
|
117
|
+
.map((row) => row.name);
|
|
118
|
+
}
|
|
119
|
+
/**
|
|
120
|
+
* Read the per-agent sync block from each enabled bundle's agent.json.
|
|
121
|
+
* Used by the daemon (and stopped-state status renderer) to build per-agent
|
|
122
|
+
* sync rows without depending on argv-derived global identity. Bundles that
|
|
123
|
+
* cannot be read are skipped silently.
|
|
124
|
+
*
|
|
125
|
+
* For rows with sync enabled, also checks whether the bundle is a git repo
|
|
126
|
+
* (via .git directory presence) and resolves the remote URL via
|
|
127
|
+
* `git remote get-url <remote>`. On any error the URL is left undefined and
|
|
128
|
+
* the status renderer falls back to "local only" or "not a git repo".
|
|
129
|
+
*/
|
|
130
|
+
function listBundleSyncRows(options = {}) {
|
|
131
|
+
const bundlesRoot = options.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
|
|
132
|
+
const readFileSync = options.readFileSync ?? fs.readFileSync;
|
|
133
|
+
const execFileSync = options.execFileSync ?? child_process_1.execFileSync;
|
|
134
|
+
const existsSync = options.existsSync ?? fs.existsSync;
|
|
135
|
+
const agents = listEnabledBundleAgents(options);
|
|
136
|
+
const rows = [];
|
|
137
|
+
for (const agent of agents) {
|
|
138
|
+
const bundleRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
139
|
+
const configPath = path.join(bundleRoot, "agent.json");
|
|
140
|
+
let enabled = false;
|
|
141
|
+
let remote = "origin";
|
|
142
|
+
try {
|
|
143
|
+
const raw = readFileSync(configPath, "utf-8");
|
|
144
|
+
const parsed = JSON.parse(raw);
|
|
145
|
+
if (parsed.sync && typeof parsed.sync === "object") {
|
|
146
|
+
if (typeof parsed.sync.enabled === "boolean")
|
|
147
|
+
enabled = parsed.sync.enabled;
|
|
148
|
+
if (typeof parsed.sync.remote === "string" && parsed.sync.remote.length > 0) {
|
|
149
|
+
remote = parsed.sync.remote;
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
catch {
|
|
154
|
+
// Best-effort: bundle without readable config still gets a row with defaults
|
|
155
|
+
}
|
|
156
|
+
const row = { agent, enabled, remote };
|
|
76
157
|
if (enabled) {
|
|
77
|
-
|
|
158
|
+
// Only meaningful when sync is enabled — we don't care about disabled bundles'
|
|
159
|
+
// git state. Check .git presence before attempting any git invocation.
|
|
160
|
+
const gitInitialized = existsSync(path.join(bundleRoot, ".git"));
|
|
161
|
+
row.gitInitialized = gitInitialized;
|
|
162
|
+
if (gitInitialized) {
|
|
163
|
+
try {
|
|
164
|
+
const out = execFileSync("git", ["remote", "get-url", remote], {
|
|
165
|
+
cwd: bundleRoot,
|
|
166
|
+
stdio: "pipe",
|
|
167
|
+
timeout: 5000,
|
|
168
|
+
}).toString().trim();
|
|
169
|
+
if (out.length > 0)
|
|
170
|
+
row.remoteUrl = out;
|
|
171
|
+
}
|
|
172
|
+
catch {
|
|
173
|
+
// No remote configured or git missing — leave remoteUrl undefined.
|
|
174
|
+
}
|
|
175
|
+
}
|
|
78
176
|
}
|
|
177
|
+
rows.push(row);
|
|
79
178
|
}
|
|
80
|
-
return
|
|
179
|
+
return rows;
|
|
81
180
|
}
|