@ouro.bot/cli 0.1.0-alpha.65 → 0.1.0-alpha.650
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 +4151 -14
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +144 -0
- package/dist/arc/episodes.js +118 -0
- package/dist/arc/evolution.js +487 -0
- package/dist/arc/intentions.js +134 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +270 -0
- package/dist/arc/packets.js +288 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +57 -0
- package/dist/heart/active-work.js +860 -43
- package/dist/heart/agent-entry.js +69 -3
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +479 -0
- package/dist/heart/awaiting/await-alert.js +146 -0
- package/dist/heart/awaiting/await-expiry.js +108 -0
- package/dist/heart/awaiting/await-loader.js +91 -0
- package/dist/heart/awaiting/await-parser.js +141 -0
- package/dist/heart/awaiting/await-runtime-state.js +100 -0
- package/dist/heart/awaiting/await-scheduler.js +377 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bridges/manager.js +137 -17
- package/dist/heart/bridges/store.js +14 -2
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +135 -0
- package/dist/heart/config-registry.js +322 -0
- package/dist/heart/config.js +118 -119
- package/dist/heart/core.js +1123 -247
- 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 +780 -0
- package/dist/heart/daemon/cli-desk.js +322 -0
- package/dist/heart/daemon/cli-exec.js +7480 -0
- package/dist/heart/daemon/cli-help.js +505 -0
- package/dist/heart/daemon/cli-parse.js +1554 -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 -1750
- package/dist/heart/daemon/daemon-entry.js +485 -2
- package/dist/heart/daemon/daemon-health.js +176 -0
- package/dist/heart/daemon/daemon-rollup.js +57 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +88 -13
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +932 -74
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +873 -0
- package/dist/heart/daemon/health-monitor.js +122 -1
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
- package/dist/heart/daemon/http-health-probe.js +80 -0
- package/dist/heart/daemon/human-command-screens.js +234 -0
- package/dist/heart/daemon/human-readiness.js +114 -0
- package/dist/heart/daemon/inner-status.js +89 -0
- package/dist/heart/daemon/interactive-repair.js +394 -0
- package/dist/heart/daemon/launchd.js +37 -8
- package/dist/heart/daemon/log-tailer.js +79 -10
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- package/dist/heart/daemon/migrate-to-desk.js +848 -0
- package/dist/heart/daemon/os-cron-deps.js +135 -0
- package/dist/heart/daemon/os-cron.js +14 -12
- package/dist/heart/daemon/ouro-bot-entry.js +4 -2
- package/dist/heart/daemon/ouro-entry.js +3 -1
- package/dist/heart/daemon/plugin-cli.js +432 -0
- package/dist/heart/daemon/process-manager.js +510 -40
- 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 +35 -14
- package/dist/heart/daemon/runtime-metadata.js +2 -30
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +493 -38
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +1 -1
- 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 +117 -39
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +229 -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 -4
- 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 +37 -14
- package/dist/heart/identity.js +168 -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 +197 -0
- package/dist/heart/mailbox/readers/agent-machine.js +418 -0
- package/dist/heart/mailbox/readers/continuity-readers.js +319 -0
- package/dist/heart/mailbox/readers/mail.js +375 -0
- package/dist/heart/mailbox/readers/runtime-readers.js +756 -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 +692 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +19 -0
- package/dist/heart/orientation-frame.js +217 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +272 -0
- package/dist/heart/provider-credentials.js +425 -0
- package/dist/heart/provider-failover.js +311 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +262 -0
- package/dist/heart/provider-readiness-cache.js +40 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +139 -52
- package/dist/heart/providers/azure.js +23 -11
- package/dist/heart/providers/error-classification.js +127 -0
- package/dist/heart/providers/github-copilot.js +145 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +26 -8
- package/dist/heart/providers/openai-codex-token.js +349 -0
- 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 +48 -24
- package/dist/heart/session-events.js +1163 -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 +133 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +44 -27
- package/dist/heart/structured-output.js +196 -0
- package/dist/heart/sync-classification.js +176 -0
- package/dist/heart/sync.js +449 -0
- package/dist/heart/target-resolution.js +9 -5
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/timeouts.js +101 -0
- package/dist/heart/tool-activity-callbacks.js +59 -0
- package/dist/heart/tool-description.js +143 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +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 +409 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
- package/dist/heart/versioning/update-hooks.js +154 -0
- package/dist/mailbox-ui/assets/index-9-AxCxuB.js +61 -0
- package/dist/mailbox-ui/assets/index-CWzt267f.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 +715 -0
- package/dist/mailroom/body-cache.js +61 -0
- package/dist/mailroom/core.js +788 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +568 -0
- package/dist/mailroom/mbox-import.js +393 -0
- package/dist/mailroom/migration.js +164 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +233 -0
- package/dist/mailroom/search-cache.js +334 -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 +14 -1
- package/dist/mind/context.js +251 -101
- package/dist/mind/desk-section.js +310 -0
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +68 -76
- 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 +48 -4
- package/dist/mind/friends/types.js +2 -2
- package/dist/mind/journal-index.js +162 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +6 -1
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +1051 -138
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +139 -5
- package/dist/nerves/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 +16 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +1041 -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 +331 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +166 -10
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +219 -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/desk/classifier.js +362 -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 +136 -25
- package/dist/repertoire/mcp-client.js +295 -0
- package/dist/repertoire/mcp-manager.js +403 -0
- package/dist/repertoire/mcp-tools.js +83 -0
- package/dist/repertoire/plugin-mcp.js +175 -0
- package/dist/repertoire/plugins.js +253 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +48 -4
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-awaiting.js +372 -0
- package/dist/repertoire/tools-base.js +59 -1082
- package/dist/repertoire/tools-bluebubbles.js +2 -0
- package/dist/repertoire/tools-bridge.js +144 -0
- package/dist/repertoire/tools-bundle.js +993 -0
- package/dist/repertoire/tools-config.js +186 -0
- package/dist/repertoire/tools-continuity.js +252 -0
- package/dist/repertoire/tools-credential.js +383 -0
- package/dist/repertoire/tools-evolution.js +527 -0
- package/dist/repertoire/tools-files.js +344 -0
- package/dist/repertoire/tools-flight.js +227 -0
- package/dist/repertoire/tools-flow.js +119 -0
- package/dist/repertoire/tools-github.js +3 -8
- package/dist/repertoire/tools-mail.js +1975 -0
- package/dist/repertoire/tools-notes.js +438 -0
- package/dist/repertoire/tools-obligations.js +143 -0
- package/dist/repertoire/tools-orientation.js +31 -0
- package/dist/repertoire/tools-record.js +464 -0
- package/dist/repertoire/tools-runtime.js +150 -0
- package/dist/repertoire/tools-session.js +766 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +182 -0
- package/dist/repertoire/tools-surface.js +344 -0
- package/dist/repertoire/tools-teams.js +12 -39
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +982 -0
- package/dist/repertoire/tools-user-profile.js +146 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools-voice.js +145 -0
- package/dist/repertoire/tools.js +193 -77
- 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 +186 -0
- package/dist/senses/await-turn-message.js +58 -0
- package/dist/senses/bluebubbles/active-turns.js +216 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
- package/dist/senses/bluebubbles/entry.js +77 -0
- package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
- package/dist/senses/bluebubbles/index.js +2737 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -71
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
- package/dist/senses/bluebubbles/processed-log.js +133 -0
- package/dist/senses/bluebubbles/replay.js +137 -0
- package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/bluebubbles-meta-guard.js +40 -0
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +607 -0
- package/dist/senses/cli/spinner-imperative.js +135 -0
- package/dist/senses/cli/spinner.js +101 -0
- package/dist/senses/cli/status-line.js +60 -0
- package/dist/senses/cli/streaming-markdown.js +526 -0
- package/dist/senses/cli/tool-display.js +85 -0
- package/dist/senses/cli/tool-render.js +85 -0
- package/dist/senses/cli/tui-store.js +240 -0
- package/dist/senses/cli/virtual-list.js +35 -0
- package/dist/senses/cli-entry.js +60 -8
- package/dist/senses/cli-layout.js +100 -0
- package/dist/senses/cli.js +517 -204
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +254 -22
- package/dist/senses/inner-dialog.js +505 -40
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +711 -181
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +393 -0
- package/dist/senses/surface-tool.js +108 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +390 -98
- package/dist/senses/trust-gate.js +100 -5
- package/dist/senses/voice/audio-playback.js +237 -0
- package/dist/senses/voice/audio-routing.js +119 -0
- package/dist/senses/voice/elevenlabs.js +202 -0
- package/dist/senses/voice/floor-control.js +431 -0
- package/dist/senses/voice/floor-controller.js +115 -0
- package/dist/senses/voice/golden-path.js +116 -0
- package/dist/senses/voice/index.js +29 -0
- package/dist/senses/voice/meeting.js +113 -0
- package/dist/senses/voice/outbound.js +190 -0
- package/dist/senses/voice/phone.js +33 -0
- package/dist/senses/voice/playback.js +139 -0
- package/dist/senses/voice/realtime-eval.js +496 -0
- package/dist/senses/voice/realtime-trace.js +531 -0
- package/dist/senses/voice/transcript.js +70 -0
- package/dist/senses/voice/turn.js +191 -0
- package/dist/senses/voice/twilio-phone-runtime.js +807 -0
- package/dist/senses/voice/twilio-phone.js +5079 -0
- package/dist/senses/voice/types.js +2 -0
- package/dist/senses/voice/whisper.js +161 -0
- package/dist/senses/voice-entry.js +81 -0
- package/dist/senses/voice-realtime-eval-command.js +99 -0
- package/dist/senses/voice-realtime-eval-entry.js +21 -0
- package/dist/senses/voice-twilio-entry.js +87 -0
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +265 -0
- package/dist/util/frontmatter.js +53 -0
- package/package.json +53 -10
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +99 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/auth-flow.js +0 -351
- package/dist/heart/daemon/ouro-path-installer.js +0 -178
- package/dist/heart/daemon/update-hooks.js +0 -138
- package/dist/heart/safe-workspace.js +0 -228
- package/dist/heart/session-recall.js +0 -116
- package/dist/mind/associative-recall.js +0 -209
- package/dist/repertoire/tasks/board.js +0 -134
- package/dist/repertoire/tasks/index.js +0 -224
- package/dist/repertoire/tasks/lifecycle.js +0 -80
- package/dist/repertoire/tasks/middleware.js +0 -65
- package/dist/repertoire/tasks/parser.js +0 -173
- package/dist/repertoire/tasks/scanner.js +0 -132
- package/dist/repertoire/tasks/transitions.js +0 -144
- 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 -7
- /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/{repertoire/tasks/types.js → heart/attachments/sources/adapter.js} +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,83 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* First-class MCP tool integration — converts MCP server tools into ToolDefinitions
|
|
4
|
+
* so the model can call them directly without shell indirection.
|
|
5
|
+
*/
|
|
6
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
|
+
exports.mcpToolsAsDefinitions = mcpToolsAsDefinitions;
|
|
8
|
+
const runtime_1 = require("../nerves/runtime");
|
|
9
|
+
/**
|
|
10
|
+
* Convert all tools from an McpManager into ToolDefinition objects.
|
|
11
|
+
*
|
|
12
|
+
* Naming rules:
|
|
13
|
+
* - Builtin servers (agent.json `mcpServers`) — legacy `{server}_{tool}`
|
|
14
|
+
* shape (e.g., `browser_navigate`), with double-prefix avoidance when
|
|
15
|
+
* the tool already starts with the server name.
|
|
16
|
+
* - Plugin servers (from `<plugin>/.mcp.json`, W6 Unit 9) — Anthropic public
|
|
17
|
+
* convention `mcp__{server}__{tool}` (e.g., `mcp__desk__task_create`).
|
|
18
|
+
* This matches Claude Code's external naming and the on-prompt promise
|
|
19
|
+
* in `desk-section.ts` (`mcp__desk__*`).
|
|
20
|
+
*
|
|
21
|
+
* The handler always calls `mcpManager.callTool()` with the un-prefixed
|
|
22
|
+
* `(server, tool)` pair regardless of how the surfaced name was shaped.
|
|
23
|
+
*/
|
|
24
|
+
function mcpToolsAsDefinitions(mcpManager) {
|
|
25
|
+
if (!mcpManager)
|
|
26
|
+
return [];
|
|
27
|
+
return mcpManager.listAllTools().flatMap((entry) => {
|
|
28
|
+
const isPluginSourced = Boolean(entry.pluginId);
|
|
29
|
+
return entry.tools.map((tool) => ({
|
|
30
|
+
tool: {
|
|
31
|
+
type: "function",
|
|
32
|
+
function: {
|
|
33
|
+
name: isPluginSourced
|
|
34
|
+
? `mcp__${entry.server}__${tool.name}`
|
|
35
|
+
: tool.name.startsWith(`${entry.server}_`) || tool.name === entry.server
|
|
36
|
+
? tool.name
|
|
37
|
+
: `${entry.server}_${tool.name}`,
|
|
38
|
+
description: tool.description || `MCP tool: ${tool.name} (server: ${entry.server})`,
|
|
39
|
+
parameters: tool.inputSchema ?? { type: "object", properties: {} },
|
|
40
|
+
},
|
|
41
|
+
},
|
|
42
|
+
riskProfile: {
|
|
43
|
+
mutates: "external_side_effect",
|
|
44
|
+
risk: "high",
|
|
45
|
+
reason: "MCP tools may mutate external systems",
|
|
46
|
+
},
|
|
47
|
+
handler: async (args) => {
|
|
48
|
+
(0, runtime_1.emitNervesEvent)({
|
|
49
|
+
event: "mcp.tool_start",
|
|
50
|
+
component: "repertoire",
|
|
51
|
+
message: `calling MCP tool ${entry.server}/${tool.name}`,
|
|
52
|
+
meta: { server: entry.server, tool: tool.name },
|
|
53
|
+
});
|
|
54
|
+
try {
|
|
55
|
+
const result = await mcpManager.callTool(entry.server, tool.name, args);
|
|
56
|
+
const text = result.content
|
|
57
|
+
.filter((c) => c.type === "text" && c.text)
|
|
58
|
+
.map((c) => c.text)
|
|
59
|
+
.join("");
|
|
60
|
+
(0, runtime_1.emitNervesEvent)({
|
|
61
|
+
event: "mcp.tool_end",
|
|
62
|
+
component: "repertoire",
|
|
63
|
+
message: `MCP tool ${entry.server}/${tool.name} completed`,
|
|
64
|
+
meta: { server: entry.server, tool: tool.name },
|
|
65
|
+
});
|
|
66
|
+
return text;
|
|
67
|
+
}
|
|
68
|
+
catch (error) {
|
|
69
|
+
const reason = error instanceof Error ? error.message : String(error);
|
|
70
|
+
(0, runtime_1.emitNervesEvent)({
|
|
71
|
+
level: "error",
|
|
72
|
+
event: "mcp.tool_error",
|
|
73
|
+
component: "repertoire",
|
|
74
|
+
message: `MCP tool ${entry.server}/${tool.name} failed: ${reason}`,
|
|
75
|
+
meta: { server: entry.server, tool: tool.name, reason },
|
|
76
|
+
});
|
|
77
|
+
return `[mcp error] ${entry.server}/${tool.name}: ${reason}`;
|
|
78
|
+
}
|
|
79
|
+
},
|
|
80
|
+
mcpServer: entry.server,
|
|
81
|
+
}));
|
|
82
|
+
});
|
|
83
|
+
}
|
|
@@ -0,0 +1,175 @@
|
|
|
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.listPluginMcpServers = listPluginMcpServers;
|
|
37
|
+
exports.pluginMcpServerToConfig = pluginMcpServerToConfig;
|
|
38
|
+
const fs = __importStar(require("fs"));
|
|
39
|
+
const path = __importStar(require("path"));
|
|
40
|
+
const identity_1 = require("../heart/identity");
|
|
41
|
+
const runtime_1 = require("../nerves/runtime");
|
|
42
|
+
const plugins_1 = require("./plugins");
|
|
43
|
+
/**
|
|
44
|
+
* Resolve `${VAR}` and `${VAR:-default}` against the process env (with a
|
|
45
|
+
* caller-supplied override map applied first — used to inject the `DESK`
|
|
46
|
+
* fallback bound to the agent's bundle desk root).
|
|
47
|
+
*
|
|
48
|
+
* Anything that isn't a recognized `${...}` token passes through verbatim.
|
|
49
|
+
* Multiple substitutions per string are supported.
|
|
50
|
+
*/
|
|
51
|
+
function resolveVars(input, overrides) {
|
|
52
|
+
return input.replace(/\$\{([A-Za-z_][A-Za-z0-9_]*)(:-([^}]*))?\}/g, (_match, varName, _full, def) => {
|
|
53
|
+
if (overrides[varName] !== undefined)
|
|
54
|
+
return overrides[varName];
|
|
55
|
+
const fromEnv = process.env[varName];
|
|
56
|
+
if (fromEnv !== undefined)
|
|
57
|
+
return fromEnv;
|
|
58
|
+
if (def !== undefined)
|
|
59
|
+
return def;
|
|
60
|
+
return "";
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
function resolveArgs(args, overrides) {
|
|
64
|
+
return args.map((a) => resolveVars(a, overrides));
|
|
65
|
+
}
|
|
66
|
+
function resolveEnv(env, overrides) {
|
|
67
|
+
const out = {};
|
|
68
|
+
for (const [k, v] of Object.entries(env)) {
|
|
69
|
+
out[k] = resolveVars(v, overrides);
|
|
70
|
+
}
|
|
71
|
+
return out;
|
|
72
|
+
}
|
|
73
|
+
function readPluginMcpManifest(pluginRoot) {
|
|
74
|
+
const manifestPath = path.join(pluginRoot, ".mcp.json");
|
|
75
|
+
if (!fs.existsSync(manifestPath))
|
|
76
|
+
return null;
|
|
77
|
+
const raw = fs.readFileSync(manifestPath, "utf-8");
|
|
78
|
+
try {
|
|
79
|
+
const parsed = JSON.parse(raw);
|
|
80
|
+
const servers = parsed?.mcpServers;
|
|
81
|
+
if (!servers || typeof servers !== "object")
|
|
82
|
+
return {};
|
|
83
|
+
return servers;
|
|
84
|
+
}
|
|
85
|
+
catch (err) {
|
|
86
|
+
/* v8 ignore next -- err.message vs String(err) branch is defensive @preserve */
|
|
87
|
+
const reason = err instanceof Error ? err.message : String(err);
|
|
88
|
+
(0, runtime_1.emitNervesEvent)({
|
|
89
|
+
level: "error",
|
|
90
|
+
event: "plugin_mcp.parse_error",
|
|
91
|
+
component: "repertoire",
|
|
92
|
+
message: `failed to parse plugin .mcp.json at ${manifestPath}: ${reason}`,
|
|
93
|
+
meta: { manifestPath, reason },
|
|
94
|
+
});
|
|
95
|
+
return null;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* List the MCP servers declared across the current agent's enabled plugins.
|
|
100
|
+
*
|
|
101
|
+
* Returns a flat list — one entry per `(plugin, server-name)` pair — already
|
|
102
|
+
* resolved against process env + the `DESK` bundle-default. The result is
|
|
103
|
+
* suitable for direct hand-off to `McpManager.start()` after key-flattening.
|
|
104
|
+
*
|
|
105
|
+
* `homeDir` is an optional override for the `~/.ouro-cli/` root (test-only).
|
|
106
|
+
*/
|
|
107
|
+
function listPluginMcpServers(homeDir) {
|
|
108
|
+
(0, runtime_1.emitNervesEvent)({
|
|
109
|
+
event: "plugin_mcp.list_start",
|
|
110
|
+
component: "repertoire",
|
|
111
|
+
message: "discovering plugin-declared MCP servers",
|
|
112
|
+
meta: { operation: "listPluginMcpServers" },
|
|
113
|
+
});
|
|
114
|
+
const config = (0, identity_1.loadAgentConfig)();
|
|
115
|
+
const declaredPlugins = config.plugins ?? [];
|
|
116
|
+
const pluginsRoot = (0, plugins_1.getPluginsRoot)(homeDir);
|
|
117
|
+
// Per-agent override: DESK defaults to <bundleRoot>/desk/ when not explicitly set.
|
|
118
|
+
const overrides = {};
|
|
119
|
+
if (process.env.DESK === undefined) {
|
|
120
|
+
overrides.DESK = path.join((0, identity_1.getAgentRoot)(), "desk");
|
|
121
|
+
}
|
|
122
|
+
const out = [];
|
|
123
|
+
for (const plugin of declaredPlugins) {
|
|
124
|
+
if (!plugin.enabled)
|
|
125
|
+
continue;
|
|
126
|
+
const pluginDir = (0, plugins_1.getPluginDir)(plugin.id, homeDir);
|
|
127
|
+
// If the plugins root or plugin dir doesn't exist, skip (installed-but-removed
|
|
128
|
+
// or never-installed). The manifest read does its own .mcp.json existence
|
|
129
|
+
// check; we only need to avoid touching missing plugin dirs.
|
|
130
|
+
if (!fs.existsSync(pluginsRoot))
|
|
131
|
+
break;
|
|
132
|
+
if (!fs.existsSync(pluginDir))
|
|
133
|
+
continue;
|
|
134
|
+
const manifest = readPluginMcpManifest(pluginDir);
|
|
135
|
+
if (!manifest)
|
|
136
|
+
continue;
|
|
137
|
+
for (const [serverName, entry] of Object.entries(manifest)) {
|
|
138
|
+
if (!entry || typeof entry.command !== "string" || entry.command.length === 0) {
|
|
139
|
+
continue;
|
|
140
|
+
}
|
|
141
|
+
out.push({
|
|
142
|
+
pluginId: plugin.id,
|
|
143
|
+
serverName,
|
|
144
|
+
command: resolveVars(entry.command, overrides),
|
|
145
|
+
args: resolveArgs(entry.args ?? [], overrides),
|
|
146
|
+
env: resolveEnv(entry.env ?? {}, overrides),
|
|
147
|
+
cwd: pluginDir,
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
(0, runtime_1.emitNervesEvent)({
|
|
152
|
+
event: "plugin_mcp.list_end",
|
|
153
|
+
component: "repertoire",
|
|
154
|
+
message: "discovered plugin-declared MCP servers",
|
|
155
|
+
meta: {
|
|
156
|
+
operation: "listPluginMcpServers",
|
|
157
|
+
pluginCount: declaredPlugins.length,
|
|
158
|
+
serverCount: out.length,
|
|
159
|
+
},
|
|
160
|
+
});
|
|
161
|
+
return out;
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Adapter: convert a `PluginMcpServer` into the shape `McpManager.start()`
|
|
165
|
+
* already consumes (`McpServerConfig`). Used by `getSharedMcpManager()` so
|
|
166
|
+
* builtin + plugin servers can be merged into one start call.
|
|
167
|
+
*/
|
|
168
|
+
function pluginMcpServerToConfig(server) {
|
|
169
|
+
return {
|
|
170
|
+
command: server.command,
|
|
171
|
+
args: server.args,
|
|
172
|
+
env: server.env,
|
|
173
|
+
cwd: server.cwd,
|
|
174
|
+
};
|
|
175
|
+
}
|
|
@@ -0,0 +1,253 @@
|
|
|
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.getPluginsRoot = getPluginsRoot;
|
|
37
|
+
exports.getPluginDir = getPluginDir;
|
|
38
|
+
exports.getPluginSkillsDir = getPluginSkillsDir;
|
|
39
|
+
exports.listPlugins = listPlugins;
|
|
40
|
+
exports.listEnabledPlugins = listEnabledPlugins;
|
|
41
|
+
exports.listPluginSkills = listPluginSkills;
|
|
42
|
+
exports.loadPluginSkill = loadPluginSkill;
|
|
43
|
+
const fs = __importStar(require("fs"));
|
|
44
|
+
const path = __importStar(require("path"));
|
|
45
|
+
const identity_1 = require("../heart/identity");
|
|
46
|
+
const ouro_version_manager_1 = require("../heart/versioning/ouro-version-manager");
|
|
47
|
+
const runtime_1 = require("../nerves/runtime");
|
|
48
|
+
/**
|
|
49
|
+
* Plugin support for ouroboros (W5.2 of the worker-generalization extraction).
|
|
50
|
+
*
|
|
51
|
+
* Plugins are installed machine-locally at `~/.ouro-cli/plugins/<plugin-id>/`
|
|
52
|
+
* via `ouro plugin install` (W5.3). Each agent's bundle declares which plugins
|
|
53
|
+
* it has enabled via the `plugins[]` field in agent.json (W5.1).
|
|
54
|
+
*
|
|
55
|
+
* This module exposes the loader half: list installed plugins, list the skills
|
|
56
|
+
* a plugin provides, and load a specific plugin skill's body. Mirrors
|
|
57
|
+
* `src/repertoire/skills.ts` priority-chain pattern.
|
|
58
|
+
*
|
|
59
|
+
* Integration with prompt assembly (W5.4): `listPluginSkills(enabledPlugins)`
|
|
60
|
+
* is called from `src/mind/prompt.ts` alongside `listSkills()` so plugin
|
|
61
|
+
* skills appear in the agent's skill index via the same `load_skill` tool.
|
|
62
|
+
*/
|
|
63
|
+
/** ~/.ouro-cli/plugins/ — machine-scoped plugin install root. */
|
|
64
|
+
function getPluginsRoot(homeDir) {
|
|
65
|
+
return path.join((0, ouro_version_manager_1.getOuroCliHome)(homeDir), "plugins");
|
|
66
|
+
}
|
|
67
|
+
/** ~/.ouro-cli/plugins/<plugin-id>/ — install dir for a specific plugin. */
|
|
68
|
+
function getPluginDir(pluginId, homeDir) {
|
|
69
|
+
return path.join(getPluginsRoot(homeDir), pluginId);
|
|
70
|
+
}
|
|
71
|
+
/** ~/.ouro-cli/plugins/<plugin-id>/skills/ — the plugin's skill directory. */
|
|
72
|
+
function getPluginSkillsDir(pluginId, homeDir) {
|
|
73
|
+
return path.join(getPluginDir(pluginId, homeDir), "skills");
|
|
74
|
+
}
|
|
75
|
+
function listSkillDirNames(dir) {
|
|
76
|
+
// Industry-standard plugin skill layout: each skill is a directory under
|
|
77
|
+
// `skills/`, containing a `SKILL.md` (Claude Code + Copilot CLI convention;
|
|
78
|
+
// SKILL.md is the cross-vendor "agent skill" format with YAML frontmatter +
|
|
79
|
+
// markdown body). Older flat-file layouts (`skills/<name>.md`) are not part
|
|
80
|
+
// of the spec and not supported.
|
|
81
|
+
if (!fs.existsSync(dir))
|
|
82
|
+
return [];
|
|
83
|
+
return fs
|
|
84
|
+
.readdirSync(dir)
|
|
85
|
+
.filter((name) => {
|
|
86
|
+
const candidate = path.join(dir, name);
|
|
87
|
+
try {
|
|
88
|
+
if (!fs.statSync(candidate).isDirectory())
|
|
89
|
+
return false;
|
|
90
|
+
return fs.existsSync(path.join(candidate, "SKILL.md"));
|
|
91
|
+
}
|
|
92
|
+
catch {
|
|
93
|
+
return false;
|
|
94
|
+
}
|
|
95
|
+
})
|
|
96
|
+
.sort();
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* List installed plugins under ~/.ouro-cli/plugins/.
|
|
100
|
+
*
|
|
101
|
+
* Returns plugin IDs — directory names under the plugins root. Does NOT
|
|
102
|
+
* filter by enabled-for-this-agent; that's `listEnabledPlugins()`'s job.
|
|
103
|
+
*/
|
|
104
|
+
function listPlugins(homeDir) {
|
|
105
|
+
(0, runtime_1.emitNervesEvent)({
|
|
106
|
+
event: "plugins.list_start",
|
|
107
|
+
component: "repertoire",
|
|
108
|
+
message: "listing installed plugins",
|
|
109
|
+
meta: { operation: "listPlugins" },
|
|
110
|
+
});
|
|
111
|
+
const root = getPluginsRoot(homeDir);
|
|
112
|
+
if (!fs.existsSync(root)) {
|
|
113
|
+
(0, runtime_1.emitNervesEvent)({
|
|
114
|
+
event: "plugins.list_end",
|
|
115
|
+
component: "repertoire",
|
|
116
|
+
message: "no plugins root yet",
|
|
117
|
+
meta: { operation: "listPlugins", count: 0, root },
|
|
118
|
+
});
|
|
119
|
+
return [];
|
|
120
|
+
}
|
|
121
|
+
const plugins = fs
|
|
122
|
+
.readdirSync(root)
|
|
123
|
+
.filter((name) => {
|
|
124
|
+
const fullPath = path.join(root, name);
|
|
125
|
+
try {
|
|
126
|
+
return fs.statSync(fullPath).isDirectory();
|
|
127
|
+
}
|
|
128
|
+
catch {
|
|
129
|
+
return false;
|
|
130
|
+
}
|
|
131
|
+
})
|
|
132
|
+
.sort();
|
|
133
|
+
(0, runtime_1.emitNervesEvent)({
|
|
134
|
+
event: "plugins.list_end",
|
|
135
|
+
component: "repertoire",
|
|
136
|
+
message: "listed installed plugins",
|
|
137
|
+
meta: { operation: "listPlugins", count: plugins.length },
|
|
138
|
+
});
|
|
139
|
+
return plugins;
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* List plugins this agent has enabled via agent.json `plugins[]`.
|
|
143
|
+
*
|
|
144
|
+
* Reads the agent config + filters to entries with `enabled: true` whose
|
|
145
|
+
* plugin directory actually exists under ~/.ouro-cli/plugins/. Returns
|
|
146
|
+
* the matched PluginConfig entries (not just IDs) so callers have access
|
|
147
|
+
* to source + version metadata.
|
|
148
|
+
*/
|
|
149
|
+
function listEnabledPlugins(homeDir) {
|
|
150
|
+
(0, runtime_1.emitNervesEvent)({
|
|
151
|
+
event: "plugins.list_enabled_start",
|
|
152
|
+
component: "repertoire",
|
|
153
|
+
message: "listing enabled plugins for agent",
|
|
154
|
+
meta: { operation: "listEnabledPlugins" },
|
|
155
|
+
});
|
|
156
|
+
const config = (0, identity_1.loadAgentConfig)();
|
|
157
|
+
const declared = config.plugins ?? [];
|
|
158
|
+
const installed = new Set(listPlugins(homeDir));
|
|
159
|
+
const enabled = declared.filter((p) => p.enabled && installed.has(p.id));
|
|
160
|
+
(0, runtime_1.emitNervesEvent)({
|
|
161
|
+
event: "plugins.list_enabled_end",
|
|
162
|
+
component: "repertoire",
|
|
163
|
+
message: "filtered to enabled + installed plugins",
|
|
164
|
+
meta: {
|
|
165
|
+
operation: "listEnabledPlugins",
|
|
166
|
+
declared: declared.length,
|
|
167
|
+
enabled: enabled.length,
|
|
168
|
+
},
|
|
169
|
+
});
|
|
170
|
+
return enabled;
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* List skill names provided by the given enabled plugins.
|
|
174
|
+
*
|
|
175
|
+
* Walks each plugin's `skills/` directory using the industry-standard
|
|
176
|
+
* directory-per-skill layout (`skills/<name>/SKILL.md`). The SKILL.md
|
|
177
|
+
* convention is shared across Claude Code and Copilot CLI.
|
|
178
|
+
*
|
|
179
|
+
* Returns a deduplicated sorted list across all plugins. If two plugins
|
|
180
|
+
* declare a skill with the same name, the later-sorted plugin's skill wins
|
|
181
|
+
* (Set dedupe by name; callers concerned with collision can call this per
|
|
182
|
+
* plugin instead).
|
|
183
|
+
*/
|
|
184
|
+
function listPluginSkills(enabledPlugins, homeDir) {
|
|
185
|
+
(0, runtime_1.emitNervesEvent)({
|
|
186
|
+
event: "plugins.list_skills_start",
|
|
187
|
+
component: "repertoire",
|
|
188
|
+
message: "listing skills across enabled plugins",
|
|
189
|
+
meta: { operation: "listPluginSkills", pluginCount: enabledPlugins.length },
|
|
190
|
+
});
|
|
191
|
+
const all = [];
|
|
192
|
+
for (const plugin of enabledPlugins) {
|
|
193
|
+
const skillsDir = getPluginSkillsDir(plugin.id, homeDir);
|
|
194
|
+
const skills = listSkillDirNames(skillsDir);
|
|
195
|
+
all.push(...skills);
|
|
196
|
+
}
|
|
197
|
+
const deduped = [...new Set(all)].sort();
|
|
198
|
+
(0, runtime_1.emitNervesEvent)({
|
|
199
|
+
event: "plugins.list_skills_end",
|
|
200
|
+
component: "repertoire",
|
|
201
|
+
message: "listed plugin skills",
|
|
202
|
+
meta: {
|
|
203
|
+
operation: "listPluginSkills",
|
|
204
|
+
pluginCount: enabledPlugins.length,
|
|
205
|
+
skillCount: deduped.length,
|
|
206
|
+
},
|
|
207
|
+
});
|
|
208
|
+
return deduped;
|
|
209
|
+
}
|
|
210
|
+
/**
|
|
211
|
+
* Load the body of a specific skill from a specific plugin.
|
|
212
|
+
*
|
|
213
|
+
* Resolves the industry-standard directory-per-skill layout:
|
|
214
|
+
* `skills/<skillName>/SKILL.md`. Throws if not found.
|
|
215
|
+
*/
|
|
216
|
+
function loadPluginSkill(pluginId, skillName, homeDir) {
|
|
217
|
+
(0, runtime_1.emitNervesEvent)({
|
|
218
|
+
event: "plugins.load_skill_start",
|
|
219
|
+
component: "repertoire",
|
|
220
|
+
message: "loading plugin skill",
|
|
221
|
+
meta: { operation: "loadPluginSkill", plugin: pluginId, skill: skillName },
|
|
222
|
+
});
|
|
223
|
+
const skillsDir = getPluginSkillsDir(pluginId, homeDir);
|
|
224
|
+
const skillPath = path.join(skillsDir, skillName, "SKILL.md");
|
|
225
|
+
if (!fs.existsSync(skillPath)) {
|
|
226
|
+
(0, runtime_1.emitNervesEvent)({
|
|
227
|
+
level: "error",
|
|
228
|
+
event: "plugins.load_skill_error",
|
|
229
|
+
component: "repertoire",
|
|
230
|
+
message: "plugin skill not found",
|
|
231
|
+
meta: {
|
|
232
|
+
operation: "loadPluginSkill",
|
|
233
|
+
plugin: pluginId,
|
|
234
|
+
skill: skillName,
|
|
235
|
+
checkedPath: skillPath,
|
|
236
|
+
},
|
|
237
|
+
});
|
|
238
|
+
throw new Error(`plugin skill '${pluginId}:${skillName}' not found at ${skillPath}`);
|
|
239
|
+
}
|
|
240
|
+
const content = fs.readFileSync(skillPath, "utf-8");
|
|
241
|
+
(0, runtime_1.emitNervesEvent)({
|
|
242
|
+
event: "plugins.load_skill_end",
|
|
243
|
+
component: "repertoire",
|
|
244
|
+
message: "loaded plugin skill",
|
|
245
|
+
meta: {
|
|
246
|
+
operation: "loadPluginSkill",
|
|
247
|
+
plugin: pluginId,
|
|
248
|
+
skill: skillName,
|
|
249
|
+
path: skillPath,
|
|
250
|
+
},
|
|
251
|
+
});
|
|
252
|
+
return content;
|
|
253
|
+
}
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.spawnBackgroundShell = spawnBackgroundShell;
|
|
4
|
+
exports.getShellSession = getShellSession;
|
|
5
|
+
exports.listShellSessions = listShellSessions;
|
|
6
|
+
exports.tailShellSession = tailShellSession;
|
|
7
|
+
exports.resetShellSessions = resetShellSessions;
|
|
8
|
+
exports.detectDestructivePatterns = detectDestructivePatterns;
|
|
9
|
+
const child_process_1 = require("child_process");
|
|
10
|
+
const crypto_1 = require("crypto");
|
|
11
|
+
const runtime_1 = require("../nerves/runtime");
|
|
12
|
+
const sessions = new Map();
|
|
13
|
+
const MAX_OUTPUT_LINES = 200;
|
|
14
|
+
function spawnBackgroundShell(command) {
|
|
15
|
+
const id = (0, crypto_1.randomUUID)();
|
|
16
|
+
const proc = (0, child_process_1.spawn)("sh", ["-c", command], {
|
|
17
|
+
stdio: ["pipe", "pipe", "pipe"],
|
|
18
|
+
});
|
|
19
|
+
const session = {
|
|
20
|
+
process: proc,
|
|
21
|
+
info: {
|
|
22
|
+
id,
|
|
23
|
+
command,
|
|
24
|
+
status: "running",
|
|
25
|
+
exitCode: null,
|
|
26
|
+
pid: proc.pid,
|
|
27
|
+
startedAt: Date.now(),
|
|
28
|
+
output: [],
|
|
29
|
+
},
|
|
30
|
+
};
|
|
31
|
+
const appendOutput = (data) => {
|
|
32
|
+
const lines = data.toString().split("\n");
|
|
33
|
+
for (const line of lines) {
|
|
34
|
+
if (line.length > 0 || session.info.output.length > 0) {
|
|
35
|
+
session.info.output.push(line);
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
// Keep only the last MAX_OUTPUT_LINES
|
|
39
|
+
if (session.info.output.length > MAX_OUTPUT_LINES) {
|
|
40
|
+
session.info.output = session.info.output.slice(-MAX_OUTPUT_LINES);
|
|
41
|
+
}
|
|
42
|
+
};
|
|
43
|
+
proc.stdout?.on("data", appendOutput);
|
|
44
|
+
proc.stderr?.on("data", appendOutput);
|
|
45
|
+
proc.on("close", (code) => {
|
|
46
|
+
session.info.status = "exited";
|
|
47
|
+
session.info.exitCode = code;
|
|
48
|
+
(0, runtime_1.emitNervesEvent)({
|
|
49
|
+
component: "repertoire",
|
|
50
|
+
event: "repertoire.shell.process_exit",
|
|
51
|
+
message: "background shell process exited",
|
|
52
|
+
meta: { id, command, exitCode: code, pid: proc.pid },
|
|
53
|
+
});
|
|
54
|
+
});
|
|
55
|
+
sessions.set(id, session);
|
|
56
|
+
(0, runtime_1.emitNervesEvent)({
|
|
57
|
+
component: "repertoire",
|
|
58
|
+
event: "repertoire.shell.spawn",
|
|
59
|
+
message: "spawned background shell session",
|
|
60
|
+
meta: { id, command, pid: proc.pid },
|
|
61
|
+
});
|
|
62
|
+
return { ...session.info };
|
|
63
|
+
}
|
|
64
|
+
function getShellSession(id) {
|
|
65
|
+
const session = sessions.get(id);
|
|
66
|
+
if (!session)
|
|
67
|
+
return undefined;
|
|
68
|
+
(0, runtime_1.emitNervesEvent)({
|
|
69
|
+
component: "repertoire",
|
|
70
|
+
event: "repertoire.shell.status_check",
|
|
71
|
+
message: "checked shell session status",
|
|
72
|
+
meta: { id, status: session.info.status, exitCode: session.info.exitCode },
|
|
73
|
+
});
|
|
74
|
+
return { ...session.info };
|
|
75
|
+
}
|
|
76
|
+
function listShellSessions() {
|
|
77
|
+
return Array.from(sessions.values()).map((s) => ({
|
|
78
|
+
id: s.info.id,
|
|
79
|
+
command: s.info.command,
|
|
80
|
+
status: s.info.status,
|
|
81
|
+
exitCode: s.info.exitCode,
|
|
82
|
+
pid: s.info.pid,
|
|
83
|
+
startedAt: s.info.startedAt,
|
|
84
|
+
output: [], // Don't include full output in listing
|
|
85
|
+
}));
|
|
86
|
+
}
|
|
87
|
+
function tailShellSession(id, lines = 50) {
|
|
88
|
+
const session = sessions.get(id);
|
|
89
|
+
if (!session)
|
|
90
|
+
return undefined;
|
|
91
|
+
const tail = session.info.output.slice(-lines);
|
|
92
|
+
(0, runtime_1.emitNervesEvent)({
|
|
93
|
+
component: "repertoire",
|
|
94
|
+
event: "repertoire.shell.tail",
|
|
95
|
+
message: "tailed shell session output",
|
|
96
|
+
meta: { id, requestedLines: lines, returnedLines: tail.length },
|
|
97
|
+
});
|
|
98
|
+
return tail.join("\n");
|
|
99
|
+
}
|
|
100
|
+
/** Reset all sessions (for testing) */
|
|
101
|
+
function resetShellSessions() {
|
|
102
|
+
for (const session of sessions.values()) {
|
|
103
|
+
if (session.info.status === "running") {
|
|
104
|
+
session.process.kill();
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
sessions.clear();
|
|
108
|
+
}
|
|
109
|
+
const DESTRUCTIVE_PATTERNS = [
|
|
110
|
+
{ name: "rm -rf root/home", regex: /rm\s+-rf\s+[/~]/ },
|
|
111
|
+
{ name: "git push --force", regex: /git\s+push\s+--force/ },
|
|
112
|
+
{ name: "git reset --hard", regex: /git\s+reset\s+--hard/ },
|
|
113
|
+
{ name: "git clean -f", regex: /git\s+clean\s+-f/ },
|
|
114
|
+
{ name: "git branch -D", regex: /git\s+branch\s+-D/ },
|
|
115
|
+
{ name: "fork bomb", regex: /:\(\)\s*\{/ },
|
|
116
|
+
{ name: "write to raw device", regex: />\s*\/dev\/sd/ },
|
|
117
|
+
{ name: "git checkout .", regex: /git\s+checkout\s+\./ },
|
|
118
|
+
{ name: "git stash drop", regex: /git\s+stash\s+drop/ },
|
|
119
|
+
];
|
|
120
|
+
/**
|
|
121
|
+
* Detect destructive patterns in a shell command.
|
|
122
|
+
* Returns list of matched pattern names. Empty array = safe.
|
|
123
|
+
* This is a friction layer, not a hard block.
|
|
124
|
+
*/
|
|
125
|
+
function detectDestructivePatterns(command) {
|
|
126
|
+
const matched = [];
|
|
127
|
+
for (const pattern of DESTRUCTIVE_PATTERNS) {
|
|
128
|
+
if (pattern.regex.test(command)) {
|
|
129
|
+
matched.push(pattern.name);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
return matched;
|
|
133
|
+
}
|