@ouro.bot/cli 0.1.0-alpha.5 → 0.1.0-alpha.500
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 +226 -183
- package/SerpentGuide.ouro/agent.json +82 -0
- package/SerpentGuide.ouro/psyche/SOUL.md +25 -0
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/assets/ouroboros.png +0 -0
- package/changelog.json +3418 -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 +989 -0
- 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 +426 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bridges/manager.js +358 -0
- package/dist/heart/bridges/state-machine.js +135 -0
- package/dist/heart/bridges/store.js +123 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +111 -0
- package/dist/heart/config-registry.js +304 -0
- package/dist/heart/config.js +193 -130
- package/dist/heart/core.js +1010 -261
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +490 -0
- package/dist/heart/daemon/agent-discovery.js +157 -0
- package/dist/heart/daemon/agent-service.js +360 -0
- package/dist/heart/daemon/agentic-repair.js +216 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +640 -0
- package/dist/heart/daemon/cli-exec.js +7239 -0
- package/dist/heart/daemon/cli-help.js +493 -0
- package/dist/heart/daemon/cli-parse.js +1533 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +561 -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 +30 -697
- package/dist/heart/daemon/daemon-entry.js +359 -8
- package/dist/heart/daemon/daemon-health.js +141 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +268 -0
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +813 -19
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +615 -0
- package/dist/heart/daemon/health-monitor.js +92 -1
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +206 -0
- 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 +171 -0
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/message-router.js +17 -8
- package/dist/heart/daemon/os-cron-deps.js +134 -0
- 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 +215 -1
- 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 +39 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +191 -0
- package/dist/heart/daemon/runtime-mode.js +67 -0
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +431 -0
- 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 +307 -0
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +264 -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 +524 -0
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +62 -0
- 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-animation.js +10 -3
- package/dist/heart/{daemon → hatch}/hatch-flow.js +54 -136
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
- package/dist/heart/hatch/specialist-orchestrator.js +129 -0
- package/dist/heart/hatch/specialist-prompt.js +102 -0
- package/dist/heart/hatch/specialist-tools.js +306 -0
- package/dist/heart/identity.js +274 -61
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +2 -20
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mcp/mcp-server.js +653 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +59 -0
- package/dist/heart/outlook/outlook-http-hooks.js +66 -0
- package/dist/heart/outlook/outlook-http-response.js +7 -0
- package/dist/heart/outlook/outlook-http-routes.js +244 -0
- package/dist/heart/outlook/outlook-http-static.js +103 -0
- package/dist/heart/outlook/outlook-http-transport.js +116 -0
- package/dist/heart/outlook/outlook-http.js +99 -0
- package/dist/heart/outlook/outlook-read.js +31 -0
- package/dist/heart/outlook/outlook-types.js +27 -0
- package/dist/heart/outlook/outlook-view.js +195 -0
- package/dist/heart/outlook/readers/agent-machine.js +382 -0
- package/dist/heart/outlook/readers/continuity-readers.js +336 -0
- package/dist/heart/outlook/readers/mail.js +362 -0
- package/dist/heart/outlook/readers/runtime-readers.js +644 -0
- package/dist/heart/outlook/readers/sessions.js +232 -0
- package/dist/heart/outlook/readers/shared.js +111 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +255 -0
- package/dist/heart/provider-credentials.js +424 -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-state.js +216 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +202 -50
- package/dist/heart/providers/azure.js +104 -13
- package/dist/heart/providers/error-classification.js +63 -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 +29 -7
- package/dist/heart/providers/openai-codex.js +63 -39
- package/dist/heart/runtime-capability-check.js +170 -0
- package/dist/heart/runtime-credentials.js +260 -0
- package/dist/heart/sense-truth.js +68 -0
- package/dist/heart/session-activity.js +190 -0
- package/dist/heart/session-events.js +1089 -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-transcript.js +167 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +129 -34
- package/dist/heart/sync.js +332 -0
- package/dist/heart/target-resolution.js +127 -0
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/tool-activity-callbacks.js +36 -0
- package/dist/heart/tool-description.js +135 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +372 -0
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/heart/versioning/ouro-bot-global-installer.js +128 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +425 -0
- package/dist/heart/{daemon → versioning}/ouro-uti.js +11 -2
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/versioning/staged-restart.js +146 -0
- package/dist/heart/versioning/update-checker.js +115 -0
- package/dist/heart/versioning/update-hooks.js +142 -0
- package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +606 -0
- package/dist/mailroom/core.js +672 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +426 -0
- package/dist/mailroom/mbox-import.js +382 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +219 -0
- package/dist/mailroom/search-cache.js +182 -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 +77 -1
- package/dist/mind/context.js +173 -94
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +84 -96
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/first-impressions.js +16 -2
- package/dist/mind/friends/channel.js +73 -0
- package/dist/mind/friends/group-context.js +144 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +58 -3
- package/dist/mind/friends/trust-explanation.js +74 -0
- package/dist/mind/friends/types.js +10 -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 +76 -9
- package/dist/mind/phrases.js +1 -0
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +1144 -117
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/mind/token-estimate.js +8 -12
- package/dist/nerves/cli-logging.js +22 -3
- 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 +101 -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/runtime.js +5 -1
- package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
- package/dist/outlook-ui/index.html +15 -0
- package/dist/repertoire/ado-client.js +17 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +774 -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 +301 -0
- package/dist/repertoire/coding/index.js +4 -1
- package/dist/repertoire/coding/manager.js +220 -13
- package/dist/repertoire/coding/spawner.js +58 -12
- package/dist/repertoire/coding/tools.js +209 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +111 -0
- package/dist/repertoire/data/ado-endpoints.json +188 -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 +255 -0
- package/dist/repertoire/mcp-manager.js +305 -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 +43 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +39 -13
- 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 -79
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-base.js +49 -707
- package/dist/repertoire/tools-bluebubbles.js +94 -0
- package/dist/repertoire/tools-bridge.js +141 -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 +105 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1377 -0
- package/dist/repertoire/tools-notes.js +376 -0
- package/dist/repertoire/tools-session.js +749 -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 +64 -61
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +356 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools.js +149 -98
- 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 +561 -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/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/bluebubbles/client.js +685 -0
- package/dist/senses/bluebubbles/entry.js +73 -0
- package/dist/senses/bluebubbles/inbound-log.js +126 -0
- package/dist/senses/bluebubbles/index.js +1881 -0
- package/dist/senses/bluebubbles/media.js +389 -0
- package/dist/senses/bluebubbles/model.js +282 -0
- package/dist/senses/bluebubbles/mutation-log.js +116 -0
- package/dist/senses/bluebubbles/processed-log.js +111 -0
- package/dist/senses/bluebubbles/replay.js +129 -0
- package/dist/senses/bluebubbles/runtime-state.js +109 -0
- package/dist/senses/bluebubbles/session-cleanup.js +72 -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 +605 -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 +83 -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 +768 -264
- package/dist/senses/commands.js +66 -3
- package/dist/senses/continuity.js +94 -0
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +199 -16
- package/dist/senses/inner-dialog.js +640 -91
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +665 -0
- 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 +844 -197
- package/dist/senses/trust-gate.js +207 -2
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +146 -0
- package/package.json +47 -6
- 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/AdoptionSpecialist.ouro/agent.json +0 -20
- package/AdoptionSpecialist.ouro/psyche/SOUL.md +0 -22
- package/dist/heart/daemon/specialist-orchestrator.js +0 -160
- package/dist/heart/daemon/specialist-prompt.js +0 -40
- package/dist/heart/daemon/specialist-session.js +0 -142
- package/dist/heart/daemon/specialist-tools.js +0 -128
- package/dist/heart/daemon/subagent-installer.js +0 -125
- package/dist/inner-worker-entry.js +0 -4
- package/dist/mind/associative-recall.js +0 -197
- package/subagents/README.md +0 -73
- package/subagents/work-doer.md +0 -233
- package/subagents/work-merger.md +0 -624
- package/subagents/work-planner.md +0 -373
- /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/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
|
@@ -0,0 +1,366 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* UpProgress — accumulated-checklist progress renderer.
|
|
4
|
+
*
|
|
5
|
+
* Displays completed phases with checkmarks, the current phase with a
|
|
6
|
+
* spinner and elapsed time, and pending phases as plain text. Uses ANSI
|
|
7
|
+
* cursor control for in-place overwriting in TTY mode, and falls back to
|
|
8
|
+
* static line-per-phase output in non-TTY mode.
|
|
9
|
+
*
|
|
10
|
+
* The caller can drive animation by calling `render(now)`. In production CLI
|
|
11
|
+
* use, `autoRender` starts a short-lived timer while a TTY phase is active so
|
|
12
|
+
* long operations never leave a dead-looking cursor.
|
|
13
|
+
*/
|
|
14
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
15
|
+
exports.CommandProgress = exports.UpProgress = void 0;
|
|
16
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
17
|
+
const terminal_ui_1 = require("./terminal-ui");
|
|
18
|
+
// ── ANSI constants (shared with startup-tui.ts pattern) ──
|
|
19
|
+
const SPINNER_FRAMES = "\u280B\u2819\u2839\u2838\u283C\u2834\u2826\u2827\u2807\u280F";
|
|
20
|
+
const RESET = "\x1b[0m";
|
|
21
|
+
const BOLD = "\x1b[1m";
|
|
22
|
+
const DIM = "\x1b[2m";
|
|
23
|
+
const GREEN = "\x1b[38;2;46;204;64m";
|
|
24
|
+
const RED = "\x1b[38;2;255;106;106m";
|
|
25
|
+
const BASE_UP_PHASE_PLAN = [
|
|
26
|
+
"update check",
|
|
27
|
+
"system setup",
|
|
28
|
+
"provider checks",
|
|
29
|
+
"starting daemon",
|
|
30
|
+
"final daemon check",
|
|
31
|
+
];
|
|
32
|
+
const FRIENDLY_UP_PHASE_LABELS = {
|
|
33
|
+
"update check": "Check for updates",
|
|
34
|
+
"system setup": "Prepare this machine",
|
|
35
|
+
"agent updates": "Update installed agents",
|
|
36
|
+
"bundle cleanup": "Clean up stale bundles",
|
|
37
|
+
"provider checks": "Check the providers your agents use right now",
|
|
38
|
+
"starting daemon": "Start the background service",
|
|
39
|
+
"final daemon check": "Confirm the background service stayed up",
|
|
40
|
+
};
|
|
41
|
+
function splitDetailLines(detail) {
|
|
42
|
+
if (!detail)
|
|
43
|
+
return [];
|
|
44
|
+
return detail
|
|
45
|
+
.split(/\r?\n/)
|
|
46
|
+
.map((line) => line.trimEnd())
|
|
47
|
+
.filter((line) => line.length > 0);
|
|
48
|
+
}
|
|
49
|
+
// ── UpProgress class ──
|
|
50
|
+
class UpProgress {
|
|
51
|
+
write;
|
|
52
|
+
isTTY;
|
|
53
|
+
columns;
|
|
54
|
+
now;
|
|
55
|
+
autoRender;
|
|
56
|
+
renderIntervalMs;
|
|
57
|
+
setTimer;
|
|
58
|
+
clearTimer;
|
|
59
|
+
eventScope;
|
|
60
|
+
commandName;
|
|
61
|
+
completed = [];
|
|
62
|
+
currentPhase = null;
|
|
63
|
+
currentDetail = null;
|
|
64
|
+
upPhasePlan = BASE_UP_PHASE_PLAN;
|
|
65
|
+
prevLineCount = 0;
|
|
66
|
+
ended = false;
|
|
67
|
+
renderTimer = null;
|
|
68
|
+
constructor(options) {
|
|
69
|
+
/* v8 ignore next -- thin wrapper: raw process.stdout.write for ANSI cursor control @preserve */
|
|
70
|
+
this.write = options?.write ?? ((text) => process.stdout.write(text));
|
|
71
|
+
/* v8 ignore next -- thin wrapper: real isTTY check injected for testability @preserve */
|
|
72
|
+
this.isTTY = options?.isTTY ?? (process.stdout.isTTY === true);
|
|
73
|
+
this.columns = options?.columns;
|
|
74
|
+
/* v8 ignore next -- thin wrapper: real Date.now injected for testability @preserve */
|
|
75
|
+
this.now = options?.now ?? (() => Date.now());
|
|
76
|
+
this.autoRender = options?.autoRender ?? false;
|
|
77
|
+
this.renderIntervalMs = options?.renderIntervalMs ?? 80;
|
|
78
|
+
/* v8 ignore start -- real timers are injected in tests when needed @preserve */
|
|
79
|
+
this.setTimer = options?.setInterval ?? ((callback, ms) => setInterval(callback, ms));
|
|
80
|
+
this.clearTimer = options?.clearInterval ?? ((handle) => clearInterval(handle));
|
|
81
|
+
/* v8 ignore stop */
|
|
82
|
+
this.eventScope = options?.eventScope ?? "up";
|
|
83
|
+
this.commandName = options?.commandName ?? null;
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Begin a new phase with spinner. If a phase is already active, it is
|
|
87
|
+
* auto-completed (no detail text).
|
|
88
|
+
*/
|
|
89
|
+
startPhase(label) {
|
|
90
|
+
if (this.currentPhase) {
|
|
91
|
+
this.completePhase(this.currentPhase.label);
|
|
92
|
+
}
|
|
93
|
+
this.currentPhase = { label, startedAt: this.now() };
|
|
94
|
+
this.currentDetail = null;
|
|
95
|
+
if (this.isTTY) {
|
|
96
|
+
this.ensureAutoRender();
|
|
97
|
+
this.flushRender();
|
|
98
|
+
}
|
|
99
|
+
else {
|
|
100
|
+
this.write(` ... ${label}\n`);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Emit a one-line status breadcrumb in non-TTY mode without affecting the
|
|
105
|
+
* accumulated checklist state. Used for daemon startup sub-steps.
|
|
106
|
+
*/
|
|
107
|
+
announceStep(label) {
|
|
108
|
+
if (this.currentPhase) {
|
|
109
|
+
this.updateDetail(label);
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
if (this.isTTY)
|
|
113
|
+
return;
|
|
114
|
+
this.write(` ${label}\n`);
|
|
115
|
+
}
|
|
116
|
+
setPhasePlan(labels) {
|
|
117
|
+
const nextPlan = [...new Set(labels.map((label) => label.trim()).filter((label) => label.length > 0))];
|
|
118
|
+
this.upPhasePlan = nextPlan.length > 0 ? nextPlan : BASE_UP_PHASE_PLAN;
|
|
119
|
+
if (this.isTTY && this.eventScope === "up") {
|
|
120
|
+
this.flushRender();
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Update the sub-step detail on the current spinner phase. Rendered as
|
|
125
|
+
* "label (Xs) -- detail" in TTY mode. In non-TTY mode, writes changed
|
|
126
|
+
* detail lines so long operations remain visible in logs and captured output.
|
|
127
|
+
*/
|
|
128
|
+
updateDetail(detail) {
|
|
129
|
+
if (!this.currentPhase || detail === this.currentDetail)
|
|
130
|
+
return;
|
|
131
|
+
this.currentDetail = detail;
|
|
132
|
+
this.currentPhase.detail = detail;
|
|
133
|
+
if (this.isTTY) {
|
|
134
|
+
this.flushRender();
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
for (const line of splitDetailLines(detail)) {
|
|
138
|
+
this.write(` ${line}\n`);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Mark the current phase as done. In non-TTY mode, immediately writes
|
|
143
|
+
* a static line. Emits a nerves event for observability.
|
|
144
|
+
*/
|
|
145
|
+
completePhase(label, detail) {
|
|
146
|
+
if (!this.currentPhase) {
|
|
147
|
+
return;
|
|
148
|
+
}
|
|
149
|
+
const elapsedMs = this.now() - this.currentPhase.startedAt;
|
|
150
|
+
this.completed.push({ status: "success", label, detail });
|
|
151
|
+
this.currentPhase = null;
|
|
152
|
+
this.currentDetail = null;
|
|
153
|
+
this.stopAutoRender();
|
|
154
|
+
if (this.eventScope === "command") {
|
|
155
|
+
(0, runtime_1.emitNervesEvent)({
|
|
156
|
+
component: "daemon",
|
|
157
|
+
event: "daemon.cli_progress_phase_complete",
|
|
158
|
+
message: `phase complete: ${label}`,
|
|
159
|
+
meta: { command: this.commandName, phase: label, detail: detail ?? null, elapsedMs },
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
else {
|
|
163
|
+
(0, runtime_1.emitNervesEvent)({
|
|
164
|
+
component: "daemon",
|
|
165
|
+
event: "daemon.up_phase_complete",
|
|
166
|
+
message: `phase complete: ${label}`,
|
|
167
|
+
meta: { phase: label, detail: detail ?? null, elapsedMs },
|
|
168
|
+
});
|
|
169
|
+
}
|
|
170
|
+
if (this.isTTY) {
|
|
171
|
+
this.flushRender();
|
|
172
|
+
}
|
|
173
|
+
else {
|
|
174
|
+
const detailStr = detail ? ` \u2014 ${detail}` : "";
|
|
175
|
+
this.write(` \u2713 ${label}${detailStr}\n`);
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
failPhase(label, detail) {
|
|
179
|
+
if (!this.currentPhase) {
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
const elapsedMs = this.now() - this.currentPhase.startedAt;
|
|
183
|
+
this.completed.push({ status: "failure", label, detail });
|
|
184
|
+
this.currentPhase = null;
|
|
185
|
+
this.currentDetail = null;
|
|
186
|
+
this.stopAutoRender();
|
|
187
|
+
if (this.eventScope === "command") {
|
|
188
|
+
(0, runtime_1.emitNervesEvent)({
|
|
189
|
+
level: "warn",
|
|
190
|
+
component: "daemon",
|
|
191
|
+
event: "daemon.cli_progress_phase_failed",
|
|
192
|
+
message: `phase failed: ${label}`,
|
|
193
|
+
meta: { command: this.commandName, phase: label, detail: detail ?? null, elapsedMs },
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
else {
|
|
197
|
+
(0, runtime_1.emitNervesEvent)({
|
|
198
|
+
level: "warn",
|
|
199
|
+
component: "daemon",
|
|
200
|
+
event: "daemon.up_phase_failed",
|
|
201
|
+
message: `phase failed: ${label}`,
|
|
202
|
+
meta: { phase: label, detail: detail ?? null, elapsedMs },
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
if (this.isTTY) {
|
|
206
|
+
this.flushRender();
|
|
207
|
+
}
|
|
208
|
+
else {
|
|
209
|
+
const detailStr = detail ? ` \u2014 ${detail}` : "";
|
|
210
|
+
this.write(` \u2717 ${label}${detailStr}\n`);
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* Build an ANSI string for in-place terminal display. Returns empty
|
|
215
|
+
* string in non-TTY mode (output is written eagerly in completePhase).
|
|
216
|
+
*/
|
|
217
|
+
render(now) {
|
|
218
|
+
if (!this.isTTY) {
|
|
219
|
+
return "";
|
|
220
|
+
}
|
|
221
|
+
const lines = this.renderLines(now);
|
|
222
|
+
const output = (0, terminal_ui_1.renderOverwriteFrame)(lines, this.prevLineCount, true);
|
|
223
|
+
this.prevLineCount = lines.length;
|
|
224
|
+
return output;
|
|
225
|
+
}
|
|
226
|
+
/**
|
|
227
|
+
* Finalize the progress display. Clears the current phase (if any) and
|
|
228
|
+
* writes the final checklist state. Idempotent.
|
|
229
|
+
*/
|
|
230
|
+
end() {
|
|
231
|
+
if (this.ended) {
|
|
232
|
+
return;
|
|
233
|
+
}
|
|
234
|
+
this.ended = true;
|
|
235
|
+
if (this.currentPhase) {
|
|
236
|
+
this.currentPhase = null;
|
|
237
|
+
this.currentDetail = null;
|
|
238
|
+
}
|
|
239
|
+
this.stopAutoRender();
|
|
240
|
+
if (this.isTTY) {
|
|
241
|
+
this.flushRender();
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
ensureAutoRender() {
|
|
245
|
+
if (!this.autoRender || !this.isTTY || this.renderTimer !== null) {
|
|
246
|
+
return;
|
|
247
|
+
}
|
|
248
|
+
this.renderTimer = this.setTimer(() => this.flushRender(), this.renderIntervalMs);
|
|
249
|
+
}
|
|
250
|
+
stopAutoRender() {
|
|
251
|
+
if (this.renderTimer === null) {
|
|
252
|
+
return;
|
|
253
|
+
}
|
|
254
|
+
this.clearTimer(this.renderTimer);
|
|
255
|
+
this.renderTimer = null;
|
|
256
|
+
}
|
|
257
|
+
flushRender() {
|
|
258
|
+
const output = this.render(this.now());
|
|
259
|
+
this.write(output);
|
|
260
|
+
}
|
|
261
|
+
renderLines(now) {
|
|
262
|
+
if (this.eventScope === "up") {
|
|
263
|
+
return this.renderUpScreen(now);
|
|
264
|
+
}
|
|
265
|
+
const lines = [];
|
|
266
|
+
for (const phase of this.completed) {
|
|
267
|
+
const detailStr = phase.detail ? ` ${DIM}\u2014 ${phase.detail}${RESET}` : "";
|
|
268
|
+
if (phase.status === "failure") {
|
|
269
|
+
lines.push(` ${RED}\u2717${RESET} ${phase.label}${detailStr}`);
|
|
270
|
+
}
|
|
271
|
+
else {
|
|
272
|
+
lines.push(` ${GREEN}\u2713${RESET} ${phase.label}${detailStr}`);
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
if (this.currentPhase) {
|
|
276
|
+
const elapsed = now - this.currentPhase.startedAt;
|
|
277
|
+
const elapsedSec = (elapsed / 1000).toFixed(1);
|
|
278
|
+
const frameIndex = Math.floor(elapsed / 80) % SPINNER_FRAMES.length;
|
|
279
|
+
const spinner = SPINNER_FRAMES[frameIndex];
|
|
280
|
+
lines.push(` ${BOLD}${spinner}${RESET} ${this.currentPhase.label} ${DIM}(${elapsedSec}s)${RESET}`);
|
|
281
|
+
for (const detailLine of splitDetailLines(this.currentPhase.detail)) {
|
|
282
|
+
lines.push(` ${DIM}${detailLine}${RESET}`);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
return lines;
|
|
286
|
+
}
|
|
287
|
+
renderUpStepLabel(label) {
|
|
288
|
+
return FRIENDLY_UP_PHASE_LABELS[label] ?? label;
|
|
289
|
+
}
|
|
290
|
+
renderUpScreen(now) {
|
|
291
|
+
const seenLabels = new Set();
|
|
292
|
+
const completedByLabel = new Map(this.completed.map((phase) => [phase.label, phase]));
|
|
293
|
+
const steps = [];
|
|
294
|
+
let currentStepLabel = this.completed.some((phase) => phase.status === "failure")
|
|
295
|
+
? "Boot paused."
|
|
296
|
+
: this.completed.length > 0
|
|
297
|
+
? "Boot checklist complete."
|
|
298
|
+
: "Waiting to begin.";
|
|
299
|
+
let currentStepDetails = [];
|
|
300
|
+
if (this.currentPhase) {
|
|
301
|
+
const elapsed = now - this.currentPhase.startedAt;
|
|
302
|
+
const elapsedSec = (elapsed / 1000).toFixed(1);
|
|
303
|
+
const frameIndex = Math.floor(elapsed / 80) % SPINNER_FRAMES.length;
|
|
304
|
+
const spinner = SPINNER_FRAMES[frameIndex];
|
|
305
|
+
currentStepLabel = `${spinner} ${this.renderUpStepLabel(this.currentPhase.label)} (${elapsedSec}s)`;
|
|
306
|
+
currentStepDetails = splitDetailLines(this.currentPhase.detail);
|
|
307
|
+
}
|
|
308
|
+
for (const label of this.upPhasePlan) {
|
|
309
|
+
seenLabels.add(label);
|
|
310
|
+
const completedPhase = completedByLabel.get(label);
|
|
311
|
+
if (completedPhase) {
|
|
312
|
+
steps.push({
|
|
313
|
+
label: this.renderUpStepLabel(label),
|
|
314
|
+
status: completedPhase.status === "failure" ? "failed" : "done",
|
|
315
|
+
detail: completedPhase.detail,
|
|
316
|
+
});
|
|
317
|
+
continue;
|
|
318
|
+
}
|
|
319
|
+
if (this.currentPhase?.label === label) {
|
|
320
|
+
steps.push({
|
|
321
|
+
label: this.renderUpStepLabel(label),
|
|
322
|
+
status: "active",
|
|
323
|
+
});
|
|
324
|
+
continue;
|
|
325
|
+
}
|
|
326
|
+
steps.push({
|
|
327
|
+
label: this.renderUpStepLabel(label),
|
|
328
|
+
status: "pending",
|
|
329
|
+
});
|
|
330
|
+
}
|
|
331
|
+
for (const phase of this.completed) {
|
|
332
|
+
if (!seenLabels.has(phase.label)) {
|
|
333
|
+
steps.push({
|
|
334
|
+
label: this.renderUpStepLabel(phase.label),
|
|
335
|
+
status: phase.status === "failure" ? "failed" : "done",
|
|
336
|
+
detail: phase.detail,
|
|
337
|
+
});
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
if (this.currentPhase && !seenLabels.has(this.currentPhase.label)) {
|
|
341
|
+
steps.push({
|
|
342
|
+
label: this.renderUpStepLabel(this.currentPhase.label),
|
|
343
|
+
status: "active",
|
|
344
|
+
});
|
|
345
|
+
}
|
|
346
|
+
return (0, terminal_ui_1.renderTerminalOperation)({
|
|
347
|
+
isTTY: true,
|
|
348
|
+
columns: this.columns,
|
|
349
|
+
masthead: {
|
|
350
|
+
subtitle: "Booting the local agent runtime.",
|
|
351
|
+
},
|
|
352
|
+
title: "Starting Ouro",
|
|
353
|
+
summary: "Ouro is bringing the local agent runtime online and will stop here if anything needs attention.",
|
|
354
|
+
currentStep: {
|
|
355
|
+
label: currentStepLabel,
|
|
356
|
+
detailLines: currentStepDetails,
|
|
357
|
+
},
|
|
358
|
+
steps,
|
|
359
|
+
currentTitle: "Doing now",
|
|
360
|
+
stepsTitle: "Boot checklist",
|
|
361
|
+
suppressEvent: true,
|
|
362
|
+
}).trimEnd().split("\n");
|
|
363
|
+
}
|
|
364
|
+
}
|
|
365
|
+
exports.UpProgress = UpProgress;
|
|
366
|
+
exports.CommandProgress = UpProgress;
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.PORKBUN_OPS_COMPATIBILITY_ALIAS = exports.PORKBUN_OPS_CREDENTIAL_PREFIX = void 0;
|
|
4
|
+
exports.isVaultItemTemplate = isVaultItemTemplate;
|
|
5
|
+
exports.normalizeVaultItemName = normalizeVaultItemName;
|
|
6
|
+
exports.normalizeVaultItemFieldName = normalizeVaultItemFieldName;
|
|
7
|
+
exports.vaultItemTemplateSecretFields = vaultItemTemplateSecretFields;
|
|
8
|
+
exports.normalizePorkbunOpsAccount = normalizePorkbunOpsAccount;
|
|
9
|
+
exports.porkbunOpsCredentialItemName = porkbunOpsCredentialItemName;
|
|
10
|
+
exports.porkbunOpsAccountFromItemName = porkbunOpsAccountFromItemName;
|
|
11
|
+
exports.requireVaultItemSecret = requireVaultItemSecret;
|
|
12
|
+
exports.PORKBUN_OPS_CREDENTIAL_PREFIX = "ops/registrars/porkbun/accounts";
|
|
13
|
+
exports.PORKBUN_OPS_COMPATIBILITY_ALIAS = "vault ops porkbun";
|
|
14
|
+
const VAULT_ITEM_NAME_FORBIDDEN = /[\r\n\t]/;
|
|
15
|
+
const VAULT_ITEM_FIELD_FORBIDDEN = /[\r\n\t=]/;
|
|
16
|
+
const PORKBUN_OPS_ACCOUNT_FORBIDDEN = /[\/\r\n\t]/;
|
|
17
|
+
function isVaultItemTemplate(value) {
|
|
18
|
+
return value === "porkbun-api";
|
|
19
|
+
}
|
|
20
|
+
function normalizeVaultItemName(item) {
|
|
21
|
+
const normalized = item?.trim() ?? "";
|
|
22
|
+
if (!normalized || VAULT_ITEM_NAME_FORBIDDEN.test(normalized) || normalized.startsWith("/") || normalized.endsWith("/")) {
|
|
23
|
+
throw new Error("Vault item name/path must be non-empty, relative, and free of control characters.");
|
|
24
|
+
}
|
|
25
|
+
return normalized;
|
|
26
|
+
}
|
|
27
|
+
function normalizeVaultItemFieldName(field) {
|
|
28
|
+
const normalized = field?.trim() ?? "";
|
|
29
|
+
if (!normalized || VAULT_ITEM_FIELD_FORBIDDEN.test(normalized)) {
|
|
30
|
+
throw new Error("Vault item field names must be non-empty and free of control characters or '='.");
|
|
31
|
+
}
|
|
32
|
+
return normalized;
|
|
33
|
+
}
|
|
34
|
+
function vaultItemTemplateSecretFields(_template) {
|
|
35
|
+
return ["apiKey", "secretApiKey"];
|
|
36
|
+
}
|
|
37
|
+
function normalizePorkbunOpsAccount(account) {
|
|
38
|
+
const normalized = account?.trim() ?? "";
|
|
39
|
+
if (!normalized || PORKBUN_OPS_ACCOUNT_FORBIDDEN.test(normalized)) {
|
|
40
|
+
throw new Error("Porkbun account must be a non-empty account label without slashes or control characters.");
|
|
41
|
+
}
|
|
42
|
+
return normalized;
|
|
43
|
+
}
|
|
44
|
+
function porkbunOpsCredentialItemName(account) {
|
|
45
|
+
return `${exports.PORKBUN_OPS_CREDENTIAL_PREFIX}/${normalizePorkbunOpsAccount(account)}`;
|
|
46
|
+
}
|
|
47
|
+
function porkbunOpsAccountFromItemName(itemName) {
|
|
48
|
+
const prefix = `${exports.PORKBUN_OPS_CREDENTIAL_PREFIX}/`;
|
|
49
|
+
return itemName.startsWith(prefix) ? itemName.slice(prefix.length) : undefined;
|
|
50
|
+
}
|
|
51
|
+
function requireVaultItemSecret(value, label) {
|
|
52
|
+
const trimmed = value.trim();
|
|
53
|
+
if (!trimmed)
|
|
54
|
+
throw new Error(`${label} cannot be blank`);
|
|
55
|
+
return trimmed;
|
|
56
|
+
}
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.decideDelegation = decideDelegation;
|
|
4
|
+
const runtime_1 = require("../nerves/runtime");
|
|
5
|
+
const CROSS_SESSION_TOOLS = new Set(["query_session", "send_message", "bridge_manage"]);
|
|
6
|
+
const FAST_PATH_TOOLS = new Set(["settle"]);
|
|
7
|
+
const REFLECTION_PATTERN = /\b(think|reflect|ponder|surface|surfaces|surfaced|sit with|metaboli[sz]e)\b/i;
|
|
8
|
+
const CROSS_SESSION_PATTERN = /\b(other chat|other session|across chats?|across sessions?|keep .* aligned|relay|carry .* across)\b/i;
|
|
9
|
+
function hasExplicitReflection(ingressTexts) {
|
|
10
|
+
return ingressTexts.some((text) => REFLECTION_PATTERN.test(text));
|
|
11
|
+
}
|
|
12
|
+
function hasCrossSessionPressure(ingressTexts, requestedToolNames) {
|
|
13
|
+
if (requestedToolNames.some((name) => CROSS_SESSION_TOOLS.has(name))) {
|
|
14
|
+
return true;
|
|
15
|
+
}
|
|
16
|
+
return ingressTexts.some((text) => CROSS_SESSION_PATTERN.test(text));
|
|
17
|
+
}
|
|
18
|
+
function hasNonFastPathToolRequest(requestedToolNames) {
|
|
19
|
+
return requestedToolNames.some((name) => !FAST_PATH_TOOLS.has(name));
|
|
20
|
+
}
|
|
21
|
+
function decideDelegation(input) {
|
|
22
|
+
const requestedToolNames = (input.requestedToolNames ?? [])
|
|
23
|
+
.map((name) => name.trim())
|
|
24
|
+
.filter((name) => name.length > 0);
|
|
25
|
+
const reasons = [];
|
|
26
|
+
if (hasExplicitReflection(input.ingressTexts)) {
|
|
27
|
+
reasons.push("explicit_reflection");
|
|
28
|
+
}
|
|
29
|
+
if (hasCrossSessionPressure(input.ingressTexts, requestedToolNames)) {
|
|
30
|
+
reasons.push("cross_session");
|
|
31
|
+
}
|
|
32
|
+
if (input.activeWork.centerOfGravity === "shared-work" || input.activeWork.bridges.some((bridge) => bridge.lifecycle === "active")) {
|
|
33
|
+
reasons.push("bridge_state");
|
|
34
|
+
}
|
|
35
|
+
if (input.activeWork.taskPressure.liveTaskNames.length > 0) {
|
|
36
|
+
reasons.push("task_state");
|
|
37
|
+
}
|
|
38
|
+
if (hasNonFastPathToolRequest(requestedToolNames)) {
|
|
39
|
+
reasons.push("non_fast_path_tool");
|
|
40
|
+
}
|
|
41
|
+
if (input.mustResolveBeforeHandoff || input.activeWork.mustResolveBeforeHandoff) {
|
|
42
|
+
reasons.push("unresolved_obligation");
|
|
43
|
+
}
|
|
44
|
+
const target = reasons.length === 0 ? "fast-path" : "delegate-inward";
|
|
45
|
+
const decision = {
|
|
46
|
+
target,
|
|
47
|
+
reasons,
|
|
48
|
+
outwardClosureRequired: target === "delegate-inward" && input.channel !== "inner",
|
|
49
|
+
};
|
|
50
|
+
(0, runtime_1.emitNervesEvent)({
|
|
51
|
+
component: "engine",
|
|
52
|
+
event: "engine.delegation_decide",
|
|
53
|
+
message: "computed delegation hint",
|
|
54
|
+
meta: {
|
|
55
|
+
channel: input.channel,
|
|
56
|
+
target: decision.target,
|
|
57
|
+
reasons: decision.reasons,
|
|
58
|
+
outwardClosureRequired: decision.outwardClosureRequired,
|
|
59
|
+
},
|
|
60
|
+
});
|
|
61
|
+
return decision;
|
|
62
|
+
}
|
|
@@ -0,0 +1,189 @@
|
|
|
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.migrateHabitsFromTaskSystem = migrateHabitsFromTaskSystem;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
40
|
+
const habit_parser_1 = require("./habit-parser");
|
|
41
|
+
const habit_runtime_state_1 = require("./habit-runtime-state");
|
|
42
|
+
const parser_1 = require("../../repertoire/tasks/parser");
|
|
43
|
+
/** Fields that belong to the task system and should be stripped from migrated habits. */
|
|
44
|
+
const TASK_ONLY_FIELDS = new Set([
|
|
45
|
+
"type",
|
|
46
|
+
"category",
|
|
47
|
+
"requester",
|
|
48
|
+
"validator",
|
|
49
|
+
"scheduledAt",
|
|
50
|
+
"updated",
|
|
51
|
+
"depends_on",
|
|
52
|
+
"parent_task",
|
|
53
|
+
"artifacts",
|
|
54
|
+
]);
|
|
55
|
+
/** Regex matching the YYYY-MM-DD-HHMM- timestamp prefix in task filenames. */
|
|
56
|
+
const TIMESTAMP_PREFIX = /^\d{4}-\d{2}-\d{2}-\d{4}-/;
|
|
57
|
+
/** Map old task statuses to habit statuses. */
|
|
58
|
+
function mapStatus(taskStatus) {
|
|
59
|
+
if (taskStatus === "processing")
|
|
60
|
+
return "active";
|
|
61
|
+
if (taskStatus === "paused")
|
|
62
|
+
return "paused";
|
|
63
|
+
if (taskStatus === "done")
|
|
64
|
+
return null; // skip done habits
|
|
65
|
+
return "active"; // default to active for unknown statuses
|
|
66
|
+
}
|
|
67
|
+
/** Strip timestamp prefix from filename to get the slug name. */
|
|
68
|
+
function stripTimestampPrefix(filename) {
|
|
69
|
+
return filename.replace(TIMESTAMP_PREFIX, "");
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Migrate habit files from the old `tasks/habits/` location to the new `habits/` bundle root.
|
|
73
|
+
* - Strips timestamp prefix from filenames
|
|
74
|
+
* - Maps task statuses to habit statuses
|
|
75
|
+
* - Strips task-only frontmatter fields
|
|
76
|
+
* - Preserves body text
|
|
77
|
+
* - Skips done habits, README files, and already-migrated habits
|
|
78
|
+
* - No-op if `tasks/habits/` does not exist
|
|
79
|
+
*/
|
|
80
|
+
function migrateHabitsFromTaskSystem(bundleRoot) {
|
|
81
|
+
const oldHabitsDir = path.join(bundleRoot, "tasks", "habits");
|
|
82
|
+
if (!fs.existsSync(oldHabitsDir)) {
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
let files;
|
|
86
|
+
try {
|
|
87
|
+
files = fs.readdirSync(oldHabitsDir);
|
|
88
|
+
}
|
|
89
|
+
catch {
|
|
90
|
+
/* v8 ignore next -- race condition: dir removed between existsSync and readdirSync @preserve */
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
const mdFiles = files.filter((f) => f.endsWith(".md") && f !== "README.md");
|
|
94
|
+
if (mdFiles.length === 0)
|
|
95
|
+
return;
|
|
96
|
+
const newHabitsDir = path.join(bundleRoot, "habits");
|
|
97
|
+
fs.mkdirSync(newHabitsDir, { recursive: true });
|
|
98
|
+
let migratedCount = 0;
|
|
99
|
+
for (const file of mdFiles) {
|
|
100
|
+
const slugName = stripTimestampPrefix(file);
|
|
101
|
+
const targetPath = path.join(newHabitsDir, slugName);
|
|
102
|
+
// Skip if already migrated
|
|
103
|
+
if (fs.existsSync(targetPath)) {
|
|
104
|
+
(0, runtime_1.emitNervesEvent)({
|
|
105
|
+
component: "daemon",
|
|
106
|
+
event: "daemon.habit_migration_skip",
|
|
107
|
+
message: "habit already exists at target, skipping",
|
|
108
|
+
meta: { file, targetPath },
|
|
109
|
+
});
|
|
110
|
+
continue;
|
|
111
|
+
}
|
|
112
|
+
const sourcePath = path.join(oldHabitsDir, file);
|
|
113
|
+
let content;
|
|
114
|
+
try {
|
|
115
|
+
content = fs.readFileSync(sourcePath, "utf-8");
|
|
116
|
+
}
|
|
117
|
+
catch {
|
|
118
|
+
continue;
|
|
119
|
+
}
|
|
120
|
+
// Parse frontmatter and body
|
|
121
|
+
const lines = content.split(/\r?\n/);
|
|
122
|
+
if (lines[0]?.trim() !== "---")
|
|
123
|
+
continue;
|
|
124
|
+
const closing = lines.findIndex((line, index) => index > 0 && line.trim() === "---");
|
|
125
|
+
if (closing === -1)
|
|
126
|
+
continue;
|
|
127
|
+
const rawFrontmatter = lines.slice(1, closing).join("\n");
|
|
128
|
+
const body = lines.slice(closing + 1).join("\n").trim();
|
|
129
|
+
const frontmatter = (0, parser_1.parseFrontmatter)(rawFrontmatter);
|
|
130
|
+
// Check status — skip done habits
|
|
131
|
+
const rawStatus = typeof frontmatter.status === "string" ? frontmatter.status : "processing";
|
|
132
|
+
const habitStatus = mapStatus(rawStatus);
|
|
133
|
+
if (habitStatus === null) {
|
|
134
|
+
(0, runtime_1.emitNervesEvent)({
|
|
135
|
+
component: "daemon",
|
|
136
|
+
event: "daemon.habit_migration_skip",
|
|
137
|
+
message: "skipping done habit",
|
|
138
|
+
meta: { file, status: rawStatus },
|
|
139
|
+
});
|
|
140
|
+
continue;
|
|
141
|
+
}
|
|
142
|
+
const legacyLastRun = typeof frontmatter.lastRun === "string" && frontmatter.lastRun !== "null"
|
|
143
|
+
? frontmatter.lastRun
|
|
144
|
+
: typeof frontmatter.last_run === "string" && frontmatter.last_run !== "null"
|
|
145
|
+
? frontmatter.last_run
|
|
146
|
+
: null;
|
|
147
|
+
// Build new frontmatter, stripping task-only fields
|
|
148
|
+
const newFrontmatter = {};
|
|
149
|
+
if (typeof frontmatter.title === "string")
|
|
150
|
+
newFrontmatter.title = frontmatter.title;
|
|
151
|
+
if (typeof frontmatter.cadence === "string")
|
|
152
|
+
newFrontmatter.cadence = frontmatter.cadence;
|
|
153
|
+
newFrontmatter.status = habitStatus;
|
|
154
|
+
newFrontmatter.created = typeof frontmatter.created === "string" ? frontmatter.created : "null";
|
|
155
|
+
// Add any other non-task fields from original
|
|
156
|
+
for (const [key, value] of Object.entries(frontmatter)) {
|
|
157
|
+
if (TASK_ONLY_FIELDS.has(key))
|
|
158
|
+
continue;
|
|
159
|
+
if (key === "lastRun" || key === "last_run")
|
|
160
|
+
continue;
|
|
161
|
+
if (key in newFrontmatter)
|
|
162
|
+
continue;
|
|
163
|
+
/* v8 ignore next -- dead code: status is caught by `key in newFrontmatter` above since newFrontmatter.status is always set @preserve */
|
|
164
|
+
if (key === "status")
|
|
165
|
+
continue; // already mapped
|
|
166
|
+
newFrontmatter[key] = value;
|
|
167
|
+
}
|
|
168
|
+
const rendered = (0, habit_parser_1.renderHabitFile)(newFrontmatter, body);
|
|
169
|
+
fs.writeFileSync(targetPath, rendered, "utf-8");
|
|
170
|
+
if (legacyLastRun) {
|
|
171
|
+
(0, habit_runtime_state_1.writeHabitLastRun)(bundleRoot, path.basename(slugName, ".md"), legacyLastRun);
|
|
172
|
+
}
|
|
173
|
+
migratedCount++;
|
|
174
|
+
(0, runtime_1.emitNervesEvent)({
|
|
175
|
+
component: "daemon",
|
|
176
|
+
event: "daemon.habit_migrated",
|
|
177
|
+
message: "migrated habit from task system",
|
|
178
|
+
meta: { from: sourcePath, to: targetPath },
|
|
179
|
+
});
|
|
180
|
+
}
|
|
181
|
+
if (migratedCount > 0) {
|
|
182
|
+
(0, runtime_1.emitNervesEvent)({
|
|
183
|
+
component: "daemon",
|
|
184
|
+
event: "daemon.habit_migration_complete",
|
|
185
|
+
message: "habit migration complete",
|
|
186
|
+
meta: { bundleRoot, count: migratedCount },
|
|
187
|
+
});
|
|
188
|
+
}
|
|
189
|
+
}
|