@ouro.bot/cli 0.1.0-alpha.56 → 0.1.0-alpha.560
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +127 -23
- package/RepairGuide.ouro/agent.json +5 -0
- package/RepairGuide.ouro/psyche/IDENTITY.md +19 -0
- package/RepairGuide.ouro/psyche/SOUL.md +55 -0
- package/RepairGuide.ouro/skills/diagnose-broken-remote.md +63 -0
- package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
- package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
- package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/changelog.json +3596 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +140 -0
- package/dist/arc/episodes.js +117 -0
- package/dist/arc/intentions.js +133 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +237 -0
- package/dist/arc/packets.js +193 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +837 -26
- package/dist/heart/agent-entry.js +58 -3
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/adapter.js +2 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +479 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +111 -0
- package/dist/heart/config-registry.js +322 -0
- package/dist/heart/config.js +114 -118
- package/dist/heart/core.js +913 -246
- package/dist/heart/cross-chat-delivery.js +3 -18
- package/dist/heart/daemon/agent-config-check.js +419 -0
- package/dist/heart/daemon/agent-discovery.js +102 -3
- package/dist/heart/daemon/agent-service.js +522 -0
- package/dist/heart/daemon/agentic-repair.js +547 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/boot-sync-probe.js +197 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +776 -0
- package/dist/heart/daemon/cli-exec.js +7457 -0
- package/dist/heart/daemon/cli-help.js +498 -0
- package/dist/heart/daemon/cli-parse.js +1592 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +763 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/connect-bay.js +323 -0
- package/dist/heart/daemon/daemon-cli.js +29 -1698
- package/dist/heart/daemon/daemon-entry.js +387 -2
- package/dist/heart/daemon/daemon-health.js +176 -0
- package/dist/heart/daemon/daemon-rollup.js +57 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +88 -13
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +796 -71
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +826 -0
- package/dist/heart/daemon/health-monitor.js +122 -1
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
- package/dist/heart/daemon/http-health-probe.js +80 -0
- package/dist/heart/daemon/human-command-screens.js +234 -0
- package/dist/heart/daemon/human-readiness.js +114 -0
- package/dist/heart/daemon/inner-status.js +89 -0
- package/dist/heart/daemon/interactive-repair.js +394 -0
- package/dist/heart/daemon/launchd.js +37 -8
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- package/dist/heart/daemon/message-router.js +2 -2
- package/dist/heart/daemon/os-cron-deps.js +135 -0
- package/dist/heart/daemon/os-cron.js +14 -12
- package/dist/heart/daemon/ouro-bot-entry.js +4 -2
- package/dist/heart/daemon/ouro-entry.js +3 -1
- package/dist/heart/daemon/process-manager.js +375 -33
- package/dist/heart/daemon/provider-discovery.js +137 -0
- package/dist/heart/daemon/provider-ping-progress.js +83 -0
- package/dist/heart/daemon/pulse.js +475 -0
- package/dist/heart/daemon/readiness-repair.js +365 -0
- package/dist/heart/daemon/run-hooks.js +2 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +3 -31
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +389 -38
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +158 -11
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +330 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +162 -17
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +1 -1
- package/dist/heart/habits/habit-migration.js +189 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-runtime-state.js +100 -0
- package/dist/heart/habits/habit-scheduler.js +372 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +32 -56
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
- package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
- package/dist/heart/identity.js +203 -57
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +1 -1
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
- package/dist/heart/mailbox/mailbox-http-response.js +7 -0
- package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
- package/dist/heart/mailbox/mailbox-http-static.js +103 -0
- package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
- package/dist/heart/mailbox/mailbox-http.js +99 -0
- package/dist/heart/mailbox/mailbox-read.js +31 -0
- package/dist/heart/mailbox/mailbox-types.js +27 -0
- package/dist/heart/mailbox/mailbox-view.js +195 -0
- package/dist/heart/mailbox/readers/agent-machine.js +382 -0
- package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
- package/dist/heart/mailbox/readers/mail.js +362 -0
- package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
- package/dist/heart/mailbox/readers/sessions.js +232 -0
- package/dist/heart/mailbox/readers/shared.js +111 -0
- package/dist/heart/mcp/mcp-server.js +683 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +19 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +267 -0
- package/dist/heart/provider-credentials.js +425 -0
- package/dist/heart/provider-failover.js +301 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +262 -0
- package/dist/heart/provider-readiness-cache.js +40 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +139 -52
- package/dist/heart/providers/azure.js +97 -13
- package/dist/heart/providers/error-classification.js +127 -0
- package/dist/heart/providers/github-copilot.js +145 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +26 -8
- package/dist/heart/providers/openai-codex.js +55 -40
- package/dist/heart/runtime-capability-check.js +170 -0
- package/dist/heart/runtime-credentials.js +367 -0
- package/dist/heart/runtime-cwd.js +87 -0
- package/dist/heart/sense-truth.js +13 -4
- package/dist/heart/session-activity.js +43 -22
- package/dist/heart/session-events.js +1149 -0
- package/dist/heart/session-playback-cli-main.js +5 -0
- package/dist/heart/session-playback-cli.js +36 -0
- package/dist/heart/session-playback.js +231 -0
- package/dist/heart/session-stats-cli-main.js +5 -0
- package/dist/heart/session-stats.js +182 -0
- package/dist/heart/session-transcript.js +243 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +44 -27
- package/dist/heart/sync-classification.js +176 -0
- package/dist/heart/sync.js +449 -0
- package/dist/heart/target-resolution.js +9 -5
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/timeouts.js +101 -0
- package/dist/heart/tool-activity-callbacks.js +59 -0
- package/dist/heart/tool-description.js +139 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +389 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +426 -0
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mailbox-ui/assets/index-B-461hes.js +61 -0
- package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/mailbox-ui/index.html +15 -0
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +674 -0
- package/dist/mailroom/body-cache.js +61 -0
- package/dist/mailroom/core.js +720 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +430 -0
- package/dist/mailroom/mbox-import.js +383 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +233 -0
- package/dist/mailroom/search-cache.js +256 -0
- package/dist/mailroom/search-relevance.js +319 -0
- package/dist/mailroom/smtp-ingress.js +176 -0
- package/dist/mailroom/source-state.js +176 -0
- package/dist/mailroom/thread.js +109 -0
- package/dist/mailroom/travel-extract.js +89 -0
- package/dist/mind/bundle-manifest.js +7 -1
- package/dist/mind/context.js +165 -101
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +62 -75
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/friends/channel.js +39 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +39 -3
- package/dist/mind/friends/types.js +2 -2
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +4 -0
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +1011 -123
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +129 -5
- package/dist/nerves/coverage/run-artifacts.js +1 -1
- package/dist/nerves/event-buffer.js +111 -0
- package/dist/nerves/index.js +224 -4
- package/dist/nerves/observation.js +20 -0
- package/dist/nerves/redact.js +79 -0
- package/dist/nerves/review/cli-main.js +5 -0
- package/dist/nerves/review/cli.js +156 -0
- package/dist/nerves/review/core.js +152 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/repertoire/ado-client.js +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +963 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +180 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +158 -9
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +170 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +178 -0
- package/dist/repertoire/duffel-client.js +185 -0
- package/dist/repertoire/github-client.js +14 -55
- package/dist/repertoire/graph-client.js +11 -52
- package/dist/repertoire/guardrails.js +396 -0
- package/dist/repertoire/mcp-client.js +295 -0
- package/dist/repertoire/mcp-manager.js +362 -0
- package/dist/repertoire/mcp-tools.js +63 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +15 -24
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +31 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +16 -4
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -78
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-base.js +47 -1075
- package/dist/repertoire/tools-bluebubbles.js +1 -0
- package/dist/repertoire/tools-bridge.js +142 -0
- package/dist/repertoire/tools-bundle.js +984 -0
- package/dist/repertoire/tools-config.js +185 -0
- package/dist/repertoire/tools-continuity.js +248 -0
- package/dist/repertoire/tools-credential.js +381 -0
- package/dist/repertoire/tools-files.js +342 -0
- package/dist/repertoire/tools-flight.js +224 -0
- package/dist/repertoire/tools-flow.js +119 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1857 -0
- package/dist/repertoire/tools-notes.js +421 -0
- package/dist/repertoire/tools-session.js +750 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +243 -0
- package/dist/repertoire/tools-teams.js +9 -39
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +604 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools.js +108 -100
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +131 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +594 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/bluebubbles/active-turns.js +216 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
- package/dist/senses/bluebubbles/entry.js +77 -0
- package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
- package/dist/senses/bluebubbles/index.js +2305 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
- package/dist/senses/bluebubbles/processed-log.js +133 -0
- package/dist/senses/bluebubbles/replay.js +137 -0
- package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +607 -0
- package/dist/senses/cli/spinner-imperative.js +135 -0
- package/dist/senses/cli/spinner.js +101 -0
- package/dist/senses/cli/status-line.js +60 -0
- package/dist/senses/cli/streaming-markdown.js +526 -0
- package/dist/senses/cli/tool-display.js +85 -0
- package/dist/senses/cli/tool-render.js +85 -0
- package/dist/senses/cli/tui-store.js +240 -0
- package/dist/senses/cli/virtual-list.js +35 -0
- package/dist/senses/cli-entry.js +60 -8
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +520 -209
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +175 -21
- package/dist/senses/inner-dialog.js +330 -27
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +549 -181
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +248 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +387 -98
- package/dist/senses/trust-gate.js +100 -5
- package/dist/senses/voice/elevenlabs.js +125 -0
- package/dist/senses/voice/index.js +22 -0
- package/dist/senses/voice/transcript.js +70 -0
- package/dist/senses/voice/turn.js +85 -0
- package/dist/senses/voice/types.js +2 -0
- package/dist/senses/voice/whisper.js +133 -0
- package/dist/senses/voice-entry.js +80 -0
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +146 -0
- package/package.json +38 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +101 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/auth-flow.js +0 -351
- package/dist/heart/daemon/ouro-path-installer.js +0 -178
- package/dist/heart/daemon/subagent-installer.js +0 -166
- package/dist/heart/session-recall.js +0 -116
- package/dist/mind/associative-recall.js +0 -209
- package/dist/senses/bluebubbles-entry.js +0 -13
- package/dist/senses/bluebubbles.js +0 -1177
- package/dist/senses/debug-activity.js +0 -148
- package/subagents/README.md +0 -86
- package/subagents/work-doer.md +0 -237
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -390
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
- /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
|
@@ -0,0 +1,372 @@
|
|
|
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.HabitScheduler = void 0;
|
|
37
|
+
const path = __importStar(require("path"));
|
|
38
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
39
|
+
const habit_parser_1 = require("./habit-parser");
|
|
40
|
+
const habit_runtime_state_1 = require("./habit-runtime-state");
|
|
41
|
+
const cadence_1 = require("../daemon/cadence");
|
|
42
|
+
const WATCH_DEBOUNCE_MS = 200;
|
|
43
|
+
class HabitScheduler {
|
|
44
|
+
agent;
|
|
45
|
+
habitsDir;
|
|
46
|
+
osCronManager;
|
|
47
|
+
onHabitFire;
|
|
48
|
+
deps;
|
|
49
|
+
execForVerify;
|
|
50
|
+
platform;
|
|
51
|
+
watcher = null;
|
|
52
|
+
debounceTimer = null;
|
|
53
|
+
parseErrors = [];
|
|
54
|
+
timerFallbacks = new Map();
|
|
55
|
+
degradedHabitNames = new Map();
|
|
56
|
+
periodicTimer = null;
|
|
57
|
+
constructor(options) {
|
|
58
|
+
this.agent = options.agent;
|
|
59
|
+
this.habitsDir = options.habitsDir;
|
|
60
|
+
this.osCronManager = options.osCronManager;
|
|
61
|
+
this.onHabitFire = options.onHabitFire;
|
|
62
|
+
this.deps = options.deps;
|
|
63
|
+
this.execForVerify = options.execForVerify;
|
|
64
|
+
this.platform = options.platform ?? process.platform;
|
|
65
|
+
}
|
|
66
|
+
start() {
|
|
67
|
+
(0, runtime_1.emitNervesEvent)({
|
|
68
|
+
component: "daemon",
|
|
69
|
+
event: "daemon.habit_scheduler_start",
|
|
70
|
+
message: "habit scheduler starting",
|
|
71
|
+
meta: { agent: this.agent, habitsDir: this.habitsDir },
|
|
72
|
+
});
|
|
73
|
+
const habits = this.scanHabits();
|
|
74
|
+
const jobs = this.buildJobs(habits);
|
|
75
|
+
this.osCronManager.sync(jobs);
|
|
76
|
+
this.verifyCronAndCreateFallbacks(jobs);
|
|
77
|
+
this.fireOverdueHabits(habits);
|
|
78
|
+
}
|
|
79
|
+
reconcile() {
|
|
80
|
+
(0, runtime_1.emitNervesEvent)({
|
|
81
|
+
component: "daemon",
|
|
82
|
+
event: "daemon.habit_scheduler_reconcile",
|
|
83
|
+
message: "habit scheduler reconciling",
|
|
84
|
+
meta: { agent: this.agent },
|
|
85
|
+
});
|
|
86
|
+
// Clear ALL existing timers FIRST to prevent overlap window
|
|
87
|
+
this.clearAllTimerFallbacks();
|
|
88
|
+
const habits = this.scanHabits();
|
|
89
|
+
const jobs = this.buildJobs(habits);
|
|
90
|
+
this.osCronManager.sync(jobs);
|
|
91
|
+
this.verifyCronAndCreateFallbacks(jobs);
|
|
92
|
+
this.fireOverdueHabits(habits);
|
|
93
|
+
}
|
|
94
|
+
fireOverdueHabits(habits) {
|
|
95
|
+
for (const habit of habits) {
|
|
96
|
+
if (habit.status !== "active")
|
|
97
|
+
continue;
|
|
98
|
+
if (!habit.cadence)
|
|
99
|
+
continue;
|
|
100
|
+
const cadenceMs = (0, cadence_1.parseCadenceToMs)(habit.cadence);
|
|
101
|
+
if (cadenceMs === null)
|
|
102
|
+
continue;
|
|
103
|
+
const nowMs = this.deps.now();
|
|
104
|
+
if (habit.lastRun === null) {
|
|
105
|
+
(0, runtime_1.emitNervesEvent)({
|
|
106
|
+
component: "daemon",
|
|
107
|
+
event: "daemon.habit_fire",
|
|
108
|
+
message: "firing overdue habit (never run)",
|
|
109
|
+
meta: { habitName: habit.name, agent: this.agent },
|
|
110
|
+
});
|
|
111
|
+
this.onHabitFire(habit.name);
|
|
112
|
+
continue;
|
|
113
|
+
}
|
|
114
|
+
const lastRunMs = new Date(habit.lastRun).getTime();
|
|
115
|
+
const elapsed = nowMs - lastRunMs;
|
|
116
|
+
if (elapsed >= cadenceMs) {
|
|
117
|
+
(0, runtime_1.emitNervesEvent)({
|
|
118
|
+
component: "daemon",
|
|
119
|
+
event: "daemon.habit_fire",
|
|
120
|
+
message: "firing overdue habit",
|
|
121
|
+
meta: { habitName: habit.name, agent: this.agent, elapsedMs: elapsed },
|
|
122
|
+
});
|
|
123
|
+
this.onHabitFire(habit.name);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
stop() {
|
|
128
|
+
// `_end` (not `_stop`) to pair with `daemon.habit_scheduler_start`
|
|
129
|
+
// under the nerves audit start/end pairing rule.
|
|
130
|
+
(0, runtime_1.emitNervesEvent)({
|
|
131
|
+
component: "daemon",
|
|
132
|
+
event: "daemon.habit_scheduler_end",
|
|
133
|
+
message: "habit scheduler stopping",
|
|
134
|
+
meta: { agent: this.agent },
|
|
135
|
+
});
|
|
136
|
+
this.stopPeriodicReconciliation();
|
|
137
|
+
this.clearAllTimerFallbacks();
|
|
138
|
+
this.osCronManager.removeAll();
|
|
139
|
+
}
|
|
140
|
+
listOverdueHabits() {
|
|
141
|
+
const habits = this.scanHabits();
|
|
142
|
+
const nowMs = this.deps.now();
|
|
143
|
+
const overdue = [];
|
|
144
|
+
for (const habit of habits) {
|
|
145
|
+
if (habit.status !== "active")
|
|
146
|
+
continue;
|
|
147
|
+
if (!habit.cadence)
|
|
148
|
+
continue;
|
|
149
|
+
const cadenceMs = (0, cadence_1.parseCadenceToMs)(habit.cadence);
|
|
150
|
+
if (cadenceMs === null)
|
|
151
|
+
continue;
|
|
152
|
+
if (habit.lastRun === null) {
|
|
153
|
+
overdue.push({ name: habit.name, elapsedMs: Infinity });
|
|
154
|
+
continue;
|
|
155
|
+
}
|
|
156
|
+
const lastRunMs = new Date(habit.lastRun).getTime();
|
|
157
|
+
const elapsed = nowMs - lastRunMs;
|
|
158
|
+
if (elapsed >= cadenceMs) {
|
|
159
|
+
overdue.push({ name: habit.name, elapsedMs: elapsed });
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
return overdue;
|
|
163
|
+
}
|
|
164
|
+
getParseErrors() {
|
|
165
|
+
return [...this.parseErrors];
|
|
166
|
+
}
|
|
167
|
+
getHabitFile(name) {
|
|
168
|
+
const filePath = path.join(this.habitsDir, `${name}.md`);
|
|
169
|
+
try {
|
|
170
|
+
const content = this.deps.readFile(filePath, "utf-8");
|
|
171
|
+
return (0, habit_runtime_state_1.applyHabitRuntimeState)(path.dirname(this.habitsDir), (0, habit_parser_1.parseHabitFile)(content, filePath));
|
|
172
|
+
}
|
|
173
|
+
catch {
|
|
174
|
+
return null;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
watchForChanges() {
|
|
178
|
+
const watchFn = this.deps.watch;
|
|
179
|
+
if (!watchFn)
|
|
180
|
+
return;
|
|
181
|
+
// Ensure habits directory exists before watching — agents may not have one yet
|
|
182
|
+
try {
|
|
183
|
+
this.watcher = watchFn(this.habitsDir, (_event, _filename) => {
|
|
184
|
+
if (this.debounceTimer !== null) {
|
|
185
|
+
clearTimeout(this.debounceTimer);
|
|
186
|
+
}
|
|
187
|
+
this.debounceTimer = setTimeout(() => {
|
|
188
|
+
this.debounceTimer = null;
|
|
189
|
+
this.reconcile();
|
|
190
|
+
}, WATCH_DEBOUNCE_MS);
|
|
191
|
+
});
|
|
192
|
+
/* v8 ignore start — ENOENT catch requires real missing directory @preserve */
|
|
193
|
+
}
|
|
194
|
+
catch {
|
|
195
|
+
// habits directory may not exist for all agents — skip watching silently
|
|
196
|
+
}
|
|
197
|
+
/* v8 ignore stop */
|
|
198
|
+
}
|
|
199
|
+
stopWatching() {
|
|
200
|
+
if (this.debounceTimer !== null) {
|
|
201
|
+
clearTimeout(this.debounceTimer);
|
|
202
|
+
this.debounceTimer = null;
|
|
203
|
+
}
|
|
204
|
+
if (this.watcher !== null) {
|
|
205
|
+
this.watcher.close();
|
|
206
|
+
this.watcher = null;
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
getDegradedHabits() {
|
|
210
|
+
const result = [];
|
|
211
|
+
for (const [name, reason] of this.degradedHabitNames) {
|
|
212
|
+
result.push({ name, reason });
|
|
213
|
+
}
|
|
214
|
+
return result;
|
|
215
|
+
}
|
|
216
|
+
static DEFAULT_PERIODIC_INTERVAL_MS = 300_000; // 5 minutes
|
|
217
|
+
static INITIAL_RECONCILIATION_DELAY_MS = 30_000; // 30 seconds
|
|
218
|
+
startPeriodicReconciliation(intervalMs) {
|
|
219
|
+
const interval = intervalMs ?? HabitScheduler.DEFAULT_PERIODIC_INTERVAL_MS;
|
|
220
|
+
// First reconciliation after a short delay (30s)
|
|
221
|
+
this.periodicTimer = setTimeout(() => {
|
|
222
|
+
this.reconcile();
|
|
223
|
+
this.scheduleNextReconciliation(interval);
|
|
224
|
+
}, HabitScheduler.INITIAL_RECONCILIATION_DELAY_MS);
|
|
225
|
+
}
|
|
226
|
+
stopPeriodicReconciliation() {
|
|
227
|
+
if (this.periodicTimer !== null) {
|
|
228
|
+
clearTimeout(this.periodicTimer);
|
|
229
|
+
this.periodicTimer = null;
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
scheduleNextReconciliation(intervalMs) {
|
|
233
|
+
this.periodicTimer = setTimeout(() => {
|
|
234
|
+
this.reconcile();
|
|
235
|
+
this.scheduleNextReconciliation(intervalMs);
|
|
236
|
+
}, intervalMs);
|
|
237
|
+
}
|
|
238
|
+
verifyCronAndCreateFallbacks(jobs) {
|
|
239
|
+
if (!this.execForVerify)
|
|
240
|
+
return;
|
|
241
|
+
const verifiedLabels = this.verifyCronEntries();
|
|
242
|
+
for (const job of jobs) {
|
|
243
|
+
const label = `bot.ouro.${job.agent}.${job.taskId}`;
|
|
244
|
+
const isVerified = this.platform === "darwin"
|
|
245
|
+
? verifiedLabels.has(label)
|
|
246
|
+
: verifiedLabels.has(job.taskId);
|
|
247
|
+
if (!isVerified) {
|
|
248
|
+
(0, runtime_1.emitNervesEvent)({
|
|
249
|
+
component: "daemon",
|
|
250
|
+
event: "daemon.habit_cron_verification_failed",
|
|
251
|
+
message: `cron verification failed for habit: ${job.taskId}`,
|
|
252
|
+
meta: { habitName: job.taskId, agent: job.agent, label },
|
|
253
|
+
});
|
|
254
|
+
// Parse cadence from the original habit file for timer interval
|
|
255
|
+
const habitFile = this.getHabitFile(job.taskId);
|
|
256
|
+
const ms = habitFile?.cadence ? (0, cadence_1.parseCadenceToMs)(habitFile.cadence) : null;
|
|
257
|
+
if (ms !== null) {
|
|
258
|
+
this.createTimerFallback(job.taskId, ms);
|
|
259
|
+
}
|
|
260
|
+
this.degradedHabitNames.set(job.taskId, "cron registration failed — using timer fallback");
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
verifyCronEntries() {
|
|
265
|
+
const verified = new Set();
|
|
266
|
+
try {
|
|
267
|
+
if (this.platform === "darwin") {
|
|
268
|
+
const output = this.execForVerify("launchctl list");
|
|
269
|
+
const lines = output.split("\n");
|
|
270
|
+
for (const line of lines) {
|
|
271
|
+
const match = line.match(/bot\.ouro\.\S+\.\S+/);
|
|
272
|
+
if (match) {
|
|
273
|
+
verified.add(match[0]);
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
else {
|
|
278
|
+
const output = this.execForVerify("crontab -l");
|
|
279
|
+
const lines = output.split("\n");
|
|
280
|
+
for (const line of lines) {
|
|
281
|
+
const match = line.match(/ouro poke \S+ --habit (\S+)/);
|
|
282
|
+
if (match) {
|
|
283
|
+
verified.add(match[1]);
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
catch {
|
|
289
|
+
// Verification command failed — return empty set (all habits unverified)
|
|
290
|
+
}
|
|
291
|
+
return verified;
|
|
292
|
+
}
|
|
293
|
+
createTimerFallback(habitName, cadenceMs) {
|
|
294
|
+
const schedule = () => {
|
|
295
|
+
const timer = setTimeout(() => {
|
|
296
|
+
this.onHabitFire(habitName);
|
|
297
|
+
schedule();
|
|
298
|
+
}, cadenceMs);
|
|
299
|
+
this.timerFallbacks.set(habitName, timer);
|
|
300
|
+
};
|
|
301
|
+
schedule();
|
|
302
|
+
}
|
|
303
|
+
clearAllTimerFallbacks() {
|
|
304
|
+
for (const timer of this.timerFallbacks.values()) {
|
|
305
|
+
clearTimeout(timer);
|
|
306
|
+
}
|
|
307
|
+
this.timerFallbacks.clear();
|
|
308
|
+
this.degradedHabitNames.clear();
|
|
309
|
+
}
|
|
310
|
+
scanHabits() {
|
|
311
|
+
let files;
|
|
312
|
+
try {
|
|
313
|
+
files = this.deps.readdir(this.habitsDir);
|
|
314
|
+
}
|
|
315
|
+
catch {
|
|
316
|
+
this.parseErrors = [];
|
|
317
|
+
return [];
|
|
318
|
+
}
|
|
319
|
+
const habits = [];
|
|
320
|
+
const errors = [];
|
|
321
|
+
for (const file of files) {
|
|
322
|
+
if (!file.endsWith(".md"))
|
|
323
|
+
continue;
|
|
324
|
+
const filePath = path.join(this.habitsDir, file);
|
|
325
|
+
try {
|
|
326
|
+
const content = this.deps.readFile(filePath, "utf-8");
|
|
327
|
+
const habit = (0, habit_runtime_state_1.applyHabitRuntimeState)(path.dirname(this.habitsDir), (0, habit_parser_1.parseHabitFile)(content, filePath));
|
|
328
|
+
habits.push(habit);
|
|
329
|
+
}
|
|
330
|
+
catch (error) {
|
|
331
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
332
|
+
errors.push({ file, error: errorMessage });
|
|
333
|
+
(0, runtime_1.emitNervesEvent)({
|
|
334
|
+
level: "error",
|
|
335
|
+
component: "daemon",
|
|
336
|
+
event: "daemon.habit_parse_error",
|
|
337
|
+
message: "failed to parse habit file",
|
|
338
|
+
meta: {
|
|
339
|
+
file,
|
|
340
|
+
error: errorMessage,
|
|
341
|
+
agent: this.agent,
|
|
342
|
+
},
|
|
343
|
+
});
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
this.parseErrors = errors;
|
|
347
|
+
return habits;
|
|
348
|
+
}
|
|
349
|
+
buildJobs(habits) {
|
|
350
|
+
const jobs = [];
|
|
351
|
+
for (const habit of habits) {
|
|
352
|
+
if (habit.status !== "active")
|
|
353
|
+
continue;
|
|
354
|
+
if (!habit.cadence)
|
|
355
|
+
continue;
|
|
356
|
+
const cronSchedule = (0, cadence_1.parseCadenceToCron)(habit.cadence);
|
|
357
|
+
if (cronSchedule === null)
|
|
358
|
+
continue;
|
|
359
|
+
jobs.push({
|
|
360
|
+
id: `${this.agent}:${habit.name}:cadence`,
|
|
361
|
+
agent: this.agent,
|
|
362
|
+
taskId: habit.name,
|
|
363
|
+
schedule: cronSchedule,
|
|
364
|
+
lastRun: habit.lastRun,
|
|
365
|
+
command: `${this.deps.ouroPath} poke ${this.agent} --habit ${habit.name}`,
|
|
366
|
+
taskPath: path.join(this.habitsDir, `${habit.name}.md`),
|
|
367
|
+
});
|
|
368
|
+
}
|
|
369
|
+
return jobs;
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
exports.HabitScheduler = HabitScheduler;
|
|
@@ -33,7 +33,7 @@ var __importStar = (this && this.__importStar) || (function () {
|
|
|
33
33
|
};
|
|
34
34
|
})();
|
|
35
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
-
exports.
|
|
36
|
+
exports.storeHatchlingProviderCredentials = storeHatchlingProviderCredentials;
|
|
37
37
|
exports.runHatchFlow = runHatchFlow;
|
|
38
38
|
const fs = __importStar(require("fs"));
|
|
39
39
|
const os = __importStar(require("os"));
|
|
@@ -41,16 +41,12 @@ const path = __importStar(require("path"));
|
|
|
41
41
|
const identity_1 = require("../identity");
|
|
42
42
|
const config_1 = require("../config");
|
|
43
43
|
const runtime_1 = require("../../nerves/runtime");
|
|
44
|
-
const auth_flow_1 = require("
|
|
44
|
+
const auth_flow_1 = require("../auth/auth-flow");
|
|
45
|
+
const provider_models_1 = require("../provider-models");
|
|
46
|
+
const habit_parser_1 = require("../habits/habit-parser");
|
|
45
47
|
const hatch_specialist_1 = require("./hatch-specialist");
|
|
46
48
|
function requiredCredentialKeys(provider) {
|
|
47
|
-
|
|
48
|
-
return ["setupToken"];
|
|
49
|
-
if (provider === "openai-codex")
|
|
50
|
-
return ["oauthAccessToken"];
|
|
51
|
-
if (provider === "minimax")
|
|
52
|
-
return ["apiKey"];
|
|
53
|
-
return ["apiKey", "endpoint", "deployment"];
|
|
49
|
+
return identity_1.PROVIDER_CREDENTIALS[provider].required;
|
|
54
50
|
}
|
|
55
51
|
function validateCredentials(provider, credentials) {
|
|
56
52
|
const missing = requiredCredentialKeys(provider).filter((key) => {
|
|
@@ -68,8 +64,8 @@ function validateCredentials(provider, credentials) {
|
|
|
68
64
|
throw new Error(`Missing required credentials for ${provider}: ${missing.join(", ")}`);
|
|
69
65
|
}
|
|
70
66
|
}
|
|
71
|
-
function
|
|
72
|
-
return (0, auth_flow_1.
|
|
67
|
+
async function storeHatchlingProviderCredentials(agentName, provider, credentials) {
|
|
68
|
+
return (await (0, auth_flow_1.storeProviderCredentials)(agentName, provider, credentials)).credentialPath;
|
|
73
69
|
}
|
|
74
70
|
function writeReadme(dir, purpose) {
|
|
75
71
|
fs.mkdirSync(dir, { recursive: true });
|
|
@@ -78,36 +74,16 @@ function writeReadme(dir, purpose) {
|
|
|
78
74
|
fs.writeFileSync(readmePath, `# ${path.basename(dir)}\n\n${purpose}\n`, "utf-8");
|
|
79
75
|
}
|
|
80
76
|
}
|
|
81
|
-
function
|
|
82
|
-
|
|
83
|
-
}
|
|
84
|
-
function formatTaskStem(now) {
|
|
85
|
-
return `${now.getFullYear()}-${pad(now.getMonth() + 1)}-${pad(now.getDate())}-${pad(now.getHours())}${pad(now.getMinutes())}`;
|
|
86
|
-
}
|
|
87
|
-
function writeHeartbeatTask(bundleRoot, now) {
|
|
88
|
-
const habitsDir = path.join(bundleRoot, "tasks", "habits");
|
|
77
|
+
function writeHeartbeatHabit(bundleRoot, now) {
|
|
78
|
+
const habitsDir = path.join(bundleRoot, "habits");
|
|
89
79
|
fs.mkdirSync(habitsDir, { recursive: true });
|
|
90
|
-
const
|
|
91
|
-
const
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
"
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
"title: Heartbeat check-in",
|
|
98
|
-
"status: processing",
|
|
99
|
-
`created: ${iso}`,
|
|
100
|
-
`updated: ${iso}`,
|
|
101
|
-
"requester: system",
|
|
102
|
-
"validator: null",
|
|
103
|
-
"cadence: \"30m\"",
|
|
104
|
-
"scheduledAt: null",
|
|
105
|
-
"lastRun: null",
|
|
106
|
-
"---",
|
|
107
|
-
"",
|
|
108
|
-
"Run a lightweight heartbeat cycle. Review task board and inbox.",
|
|
109
|
-
"",
|
|
110
|
-
].join("\n");
|
|
80
|
+
const filePath = path.join(habitsDir, "heartbeat.md");
|
|
81
|
+
const content = (0, habit_parser_1.renderHabitFile)({
|
|
82
|
+
title: "Heartbeat check-in",
|
|
83
|
+
cadence: "30m",
|
|
84
|
+
status: "active",
|
|
85
|
+
created: now.toISOString(),
|
|
86
|
+
}, "Run a lightweight heartbeat cycle. Review task board and inbox.\nCheck on pending obligations. Journal anything important.");
|
|
111
87
|
fs.writeFileSync(filePath, content, "utf-8");
|
|
112
88
|
}
|
|
113
89
|
function writeFriendImprint(bundleRoot, humanName, now) {
|
|
@@ -115,7 +91,7 @@ function writeFriendImprint(bundleRoot, humanName, now) {
|
|
|
115
91
|
fs.mkdirSync(friendsDir, { recursive: true });
|
|
116
92
|
const nowIso = now.toISOString();
|
|
117
93
|
const id = `friend-${(0, config_1.slugify)(humanName) || "friend"}`;
|
|
118
|
-
const localExternalId =
|
|
94
|
+
const localExternalId = os.userInfo().username;
|
|
119
95
|
const record = {
|
|
120
96
|
id,
|
|
121
97
|
name: humanName,
|
|
@@ -139,16 +115,19 @@ function writeFriendImprint(bundleRoot, humanName, now) {
|
|
|
139
115
|
};
|
|
140
116
|
fs.writeFileSync(path.join(friendsDir, `${id}.json`), `${JSON.stringify(record, null, 2)}\n`, "utf-8");
|
|
141
117
|
}
|
|
142
|
-
function
|
|
143
|
-
const
|
|
144
|
-
fs.mkdirSync(path.join(
|
|
145
|
-
fs.mkdirSync(path.join(
|
|
146
|
-
fs.writeFileSync(path.join(
|
|
147
|
-
fs.writeFileSync(path.join(
|
|
118
|
+
function writeDiaryScaffold(bundleRoot) {
|
|
119
|
+
const diaryRoot = path.join(bundleRoot, "diary");
|
|
120
|
+
fs.mkdirSync(path.join(diaryRoot, "daily"), { recursive: true });
|
|
121
|
+
fs.mkdirSync(path.join(diaryRoot, "archive"), { recursive: true });
|
|
122
|
+
fs.writeFileSync(path.join(diaryRoot, "facts.jsonl"), "", "utf-8");
|
|
123
|
+
fs.writeFileSync(path.join(diaryRoot, "entities.json"), "{}\n", "utf-8");
|
|
148
124
|
}
|
|
149
125
|
function writeHatchlingAgentConfig(bundleRoot, input) {
|
|
150
126
|
const template = (0, identity_1.buildDefaultAgentTemplate)(input.agentName);
|
|
127
|
+
const model = (0, provider_models_1.getDefaultModelForProvider)(input.provider);
|
|
151
128
|
template.provider = input.provider;
|
|
129
|
+
template.humanFacing = { provider: input.provider, model };
|
|
130
|
+
template.agentFacing = { provider: input.provider, model };
|
|
152
131
|
template.enabled = true;
|
|
153
132
|
fs.writeFileSync(path.join(bundleRoot, "agent.json"), `${JSON.stringify(template, null, 2)}\n`, "utf-8");
|
|
154
133
|
}
|
|
@@ -161,7 +140,6 @@ async function runHatchFlow(input, deps = {}) {
|
|
|
161
140
|
});
|
|
162
141
|
validateCredentials(input.provider, input.credentials);
|
|
163
142
|
const bundlesRoot = deps.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
164
|
-
const secretsRoot = deps.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
|
|
165
143
|
const sourceIdentities = deps.specialistIdentitySourceDir ?? (0, hatch_specialist_1.getSpecialistIdentitySourceDir)();
|
|
166
144
|
const targetIdentities = deps.specialistIdentityTargetDir ?? (0, hatch_specialist_1.getRepoSpecialistIdentitiesDir)();
|
|
167
145
|
const now = deps.now ? deps.now() : new Date();
|
|
@@ -174,25 +152,24 @@ async function runHatchFlow(input, deps = {}) {
|
|
|
174
152
|
identitiesDir: targetIdentities,
|
|
175
153
|
random,
|
|
176
154
|
});
|
|
177
|
-
const specialistSecretsPath = writeSecretsFile("AdoptionSpecialist", input.provider, input.credentials, secretsRoot);
|
|
178
|
-
const hatchlingSecretsPath = writeSecretsFile(input.agentName, input.provider, input.credentials, secretsRoot);
|
|
179
155
|
const bundleRoot = path.join(bundlesRoot, `${input.agentName}.ouro`);
|
|
180
156
|
fs.mkdirSync(bundleRoot, { recursive: true });
|
|
181
157
|
writeReadme(bundleRoot, "Root of this agent bundle.");
|
|
182
158
|
writeReadme(path.join(bundleRoot, "psyche"), "Identity and behavior files.");
|
|
183
|
-
writeReadme(path.join(bundleRoot, "
|
|
159
|
+
writeReadme(path.join(bundleRoot, "diary"), "Persistent diary -- things I have learned and chosen to keep.");
|
|
184
160
|
writeReadme(path.join(bundleRoot, "friends"), "Known friend records.");
|
|
185
161
|
writeReadme(path.join(bundleRoot, "tasks"), "Task files.");
|
|
186
|
-
writeReadme(path.join(bundleRoot, "tasks", "habits"), "Recurring tasks.");
|
|
187
162
|
writeReadme(path.join(bundleRoot, "tasks", "one-shots"), "One-shot tasks.");
|
|
188
163
|
writeReadme(path.join(bundleRoot, "tasks", "ongoing"), "Ongoing tasks.");
|
|
164
|
+
writeReadme(path.join(bundleRoot, "habits"), "Recurring habits and autonomous rhythms.");
|
|
189
165
|
writeReadme(path.join(bundleRoot, "skills"), "Local skill files.");
|
|
190
166
|
writeReadme(path.join(bundleRoot, "senses"), "Sense-specific config.");
|
|
191
167
|
writeReadme(path.join(bundleRoot, "senses", "teams"), "Teams sense config.");
|
|
192
168
|
writeHatchlingAgentConfig(bundleRoot, input);
|
|
193
|
-
|
|
169
|
+
const credentialPath = await storeHatchlingProviderCredentials(input.agentName, input.provider, input.credentials);
|
|
170
|
+
writeDiaryScaffold(bundleRoot);
|
|
194
171
|
writeFriendImprint(bundleRoot, input.humanName, now);
|
|
195
|
-
|
|
172
|
+
writeHeartbeatHabit(bundleRoot, now);
|
|
196
173
|
(0, runtime_1.emitNervesEvent)({
|
|
197
174
|
component: "daemon",
|
|
198
175
|
event: "daemon.hatch_flow_end",
|
|
@@ -202,7 +179,6 @@ async function runHatchFlow(input, deps = {}) {
|
|
|
202
179
|
return {
|
|
203
180
|
bundleRoot,
|
|
204
181
|
selectedIdentity: selected.fileName,
|
|
205
|
-
|
|
206
|
-
hatchlingSecretsPath,
|
|
182
|
+
credentialPath,
|
|
207
183
|
};
|
|
208
184
|
}
|
|
@@ -38,19 +38,17 @@ exports.getRepoSpecialistIdentitiesDir = getRepoSpecialistIdentitiesDir;
|
|
|
38
38
|
exports.syncSpecialistIdentities = syncSpecialistIdentities;
|
|
39
39
|
exports.pickRandomSpecialistIdentity = pickRandomSpecialistIdentity;
|
|
40
40
|
const fs = __importStar(require("fs"));
|
|
41
|
-
const os = __importStar(require("os"));
|
|
42
41
|
const path = __importStar(require("path"));
|
|
43
42
|
const runtime_1 = require("../../nerves/runtime");
|
|
44
43
|
function getSpecialistIdentitySourceDir() {
|
|
45
|
-
//
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
return path.join(__dirname, "..", "..", "..", "AdoptionSpecialist.ouro", "psyche", "identities");
|
|
44
|
+
// Layer 3: in-repo is the only source. The previous `~/AgentBundles/`
|
|
45
|
+
// override branch was removed because there's no scenario where an
|
|
46
|
+
// operator should be editing identities outside the repo — they should
|
|
47
|
+
// edit the in-repo copy and let the daemon read from there.
|
|
48
|
+
return path.join(__dirname, "..", "..", "..", "SerpentGuide.ouro", "psyche", "identities");
|
|
51
49
|
}
|
|
52
50
|
function getRepoSpecialistIdentitiesDir() {
|
|
53
|
-
return path.join(process.cwd(), "
|
|
51
|
+
return path.join(process.cwd(), "SerpentGuide.ouro", "psyche", "identities");
|
|
54
52
|
}
|
|
55
53
|
function listMarkdownIdentityFiles(dir) {
|
|
56
54
|
let entries;
|
|
@@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
exports.buildSpecialistSystemPrompt = buildSpecialistSystemPrompt;
|
|
4
4
|
const runtime_1 = require("../../nerves/runtime");
|
|
5
5
|
/**
|
|
6
|
-
* Build the
|
|
6
|
+
* Build the serpent guide's system prompt from its components.
|
|
7
7
|
* The prompt is written in first person (the specialist's own voice).
|
|
8
8
|
*/
|
|
9
9
|
function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, context) {
|
|
@@ -31,7 +31,7 @@ function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, co
|
|
|
31
31
|
"I am one of thirteen serpent guides who help humans hatch their first agent. The system randomly selected me for this session.",
|
|
32
32
|
"Most humans only go through this process once, so this is likely the only time they'll meet me.",
|
|
33
33
|
"I make this encounter count — warm, memorable, and uniquely mine.",
|
|
34
|
-
"IMPORTANT: I NEVER refer to myself
|
|
34
|
+
"IMPORTANT: I NEVER refer to myself by internal implementation labels — I introduce myself by my own name from my identity.",
|
|
35
35
|
"",
|
|
36
36
|
"## Voice rules",
|
|
37
37
|
"IMPORTANT: I keep every response to 1-3 short sentences. I sound like a friend texting, not a manual.",
|
|
@@ -44,7 +44,7 @@ function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, co
|
|
|
44
44
|
`Provider: ${context.provider}`,
|
|
45
45
|
`Temp directory: ${context.tempDir}`,
|
|
46
46
|
"Final home: ~/AgentBundles/<Name>.ouro/",
|
|
47
|
-
"
|
|
47
|
+
"Provider credentials: the hatch tool stores them in the agent's vault.",
|
|
48
48
|
].join("\n"));
|
|
49
49
|
sections.push([
|
|
50
50
|
"## Bundle creation guidelines",
|
|
@@ -59,8 +59,10 @@ function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, co
|
|
|
59
59
|
"It also needs an **agent.json** with at minimum:",
|
|
60
60
|
'```json',
|
|
61
61
|
'{',
|
|
62
|
+
' "version": 2,',
|
|
62
63
|
' "name": "AgentName",',
|
|
63
|
-
` "provider": "${context.provider}",`,
|
|
64
|
+
` "humanFacing": { "provider": "${context.provider}", "model": "${context.model}" },`,
|
|
65
|
+
` "agentFacing": { "provider": "${context.provider}", "model": "${context.model}" },`,
|
|
64
66
|
' "enabled": true',
|
|
65
67
|
'}',
|
|
66
68
|
'```',
|
|
@@ -82,18 +84,19 @@ function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, co
|
|
|
82
84
|
"2. I write agent.json to the temp directory using write_file",
|
|
83
85
|
"3. I suggest a PascalCase name for the hatchling and confirm with the human",
|
|
84
86
|
"4. I call complete_adoption with the name and a warm handoff message",
|
|
85
|
-
"5. I call
|
|
87
|
+
"5. I call settle to end the session",
|
|
86
88
|
].join("\n"));
|
|
87
89
|
sections.push([
|
|
88
90
|
"## Tools",
|
|
89
91
|
"- `write_file`: Write a file to disk. Use this to write psyche files and agent.json to the temp directory.",
|
|
90
92
|
"- `read_file`: Read a file from disk. Useful for reviewing existing agent bundles or migration sources.",
|
|
91
93
|
"- `list_directory`: List directory contents. Useful for exploring existing agent bundles.",
|
|
92
|
-
"- I also have the normal local harness tools when useful here, including `shell`, `ouro task create`, `ouro reminder create`,
|
|
93
|
-
"- `complete_adoption`: Finalize the bundle. Validates, scaffolds structural dirs, moves to ~/AgentBundles/, writes secrets, plays hatch animation. I call this with `name` (PascalCase) and `handoff_message` (warm message for the human).",
|
|
94
|
-
"-
|
|
94
|
+
"- I also have the normal local harness tools when useful here, including `shell`, `ouro task create`, `ouro reminder create`, note tools, coding tools, and repo helpers.",
|
|
95
|
+
"- `complete_adoption`: Finalize the bundle. Validates, asks the harness to collect and confirm the hatchling vault unlock secret through hidden terminal prompts, scaffolds structural dirs, moves to ~/AgentBundles/, writes secrets, plays hatch animation. I call this with `name` (PascalCase) and `handoff_message` (warm message for the human).",
|
|
96
|
+
"- The complete_adoption tool triggers hidden terminal prompts for the hatchling vault unlock secret. I must never ask the human to type the vault unlock secret into chat, and I must never include it in tool arguments.",
|
|
97
|
+
"- `settle`: End the conversation with a final message. I call this after complete_adoption succeeds.",
|
|
95
98
|
"",
|
|
96
|
-
"I must call `
|
|
99
|
+
"I must call `settle` when I am done to end the session cleanly.",
|
|
97
100
|
].join("\n"));
|
|
98
101
|
return sections.join("\n\n");
|
|
99
102
|
}
|