@ouro.bot/cli 0.1.0-alpha.4 → 0.1.0-alpha.400
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 +208 -184
- 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 +2610 -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 +832 -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 +417 -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 +176 -130
- package/dist/heart/core.js +872 -255
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +397 -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 +213 -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 +599 -0
- package/dist/heart/daemon/cli-exec.js +3616 -0
- package/dist/heart/daemon/cli-help.js +396 -0
- package/dist/heart/daemon/cli-parse.js +1118 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +560 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/daemon-cli.js +29 -673
- package/dist/heart/daemon/daemon-entry.js +370 -8
- package/dist/heart/daemon/daemon-health.js +141 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +235 -0
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +813 -19
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +419 -0
- package/dist/heart/daemon/health-monitor.js +79 -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/inner-status.js +89 -0
- package/dist/heart/daemon/interactive-repair.js +209 -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 +105 -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 +202 -1
- package/dist/heart/daemon/provider-discovery.js +137 -0
- package/dist/heart/daemon/pulse.js +475 -0
- package/dist/heart/daemon/readiness-repair.js +216 -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 +355 -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 +237 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/thoughts.js +510 -0
- package/dist/heart/daemon/up-progress.js +135 -0
- package/dist/heart/delegation.js +62 -0
- package/dist/heart/habits/habit-migration.js +181 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-scheduler.js +371 -0
- package/dist/heart/hatch/hatch-animation.js +35 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +55 -135
- 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 +304 -0
- package/dist/heart/identity.js +246 -58
- 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/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 +64 -0
- package/dist/heart/outlook/outlook-http-response.js +7 -0
- package/dist/heart/outlook/outlook-http-routes.js +232 -0
- package/dist/heart/outlook/outlook-http-static.js +99 -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 +28 -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 +359 -0
- package/dist/heart/outlook/readers/continuity-readers.js +332 -0
- package/dist/heart/outlook/readers/runtime-readers.js +660 -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 +133 -0
- package/dist/heart/provider-binding-resolver.js +239 -0
- package/dist/heart/provider-credentials.js +383 -0
- package/dist/heart/provider-failover.js +266 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +237 -0
- package/dist/heart/provider-state.js +216 -0
- package/dist/heart/provider-visibility.js +186 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +202 -50
- package/dist/heart/providers/azure.js +103 -12
- 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 +39 -29
- package/dist/heart/runtime-credentials.js +181 -0
- package/dist/heart/sense-truth.js +61 -0
- package/dist/heart/session-activity.js +190 -0
- package/dist/heart/session-events.js +855 -0
- package/dist/heart/session-transcript.js +167 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +117 -33
- 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 +351 -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 +301 -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 +113 -0
- package/dist/heart/versioning/update-hooks.js +142 -0
- package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
- package/dist/mind/bundle-manifest.js +77 -1
- package/dist/mind/context.js +141 -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 +64 -0
- package/dist/mind/friends/group-context.js +144 -0
- package/dist/mind/friends/resolver.js +38 -1
- 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 +1111 -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 +83 -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-BAcU08c-.css +1 -0
- package/dist/outlook-ui/assets/index-D7l3l4vY.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 +461 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +79 -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 +107 -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 +371 -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 +28 -10
- 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 +45 -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 +182 -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-notes.js +376 -0
- package/dist/repertoire/tools-session.js +739 -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-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 +118 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +382 -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 +70 -0
- package/dist/senses/bluebubbles/inbound-log.js +113 -0
- package/dist/senses/bluebubbles/index.js +1620 -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/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 +112 -19
- package/dist/senses/inner-dialog.js +636 -86
- package/dist/senses/pipeline.js +602 -0
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +205 -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 +112 -2
- package/package.json +38 -6
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +110 -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 +134 -0
- package/AdoptionSpecialist.ouro/agent.json +0 -20
- package/AdoptionSpecialist.ouro/psyche/SOUL.md +0 -22
- 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
|
@@ -43,14 +43,14 @@ const path = __importStar(require("path"));
|
|
|
43
43
|
const runtime_1 = require("../../nerves/runtime");
|
|
44
44
|
function getSpecialistIdentitySourceDir() {
|
|
45
45
|
// Prefer ~/AgentBundles/ if it exists (user may have customized identities)
|
|
46
|
-
const userSource = path.join(os.homedir(), "AgentBundles", "
|
|
46
|
+
const userSource = path.join(os.homedir(), "AgentBundles", "SerpentGuide.ouro", "psyche", "identities");
|
|
47
47
|
if (fs.existsSync(userSource))
|
|
48
48
|
return userSource;
|
|
49
49
|
// Fall back to the bundled copy shipped with the npm package
|
|
50
|
-
return path.join(__dirname, "..", "..", "..", "
|
|
50
|
+
return path.join(__dirname, "..", "..", "..", "SerpentGuide.ouro", "psyche", "identities");
|
|
51
51
|
}
|
|
52
52
|
function getRepoSpecialistIdentitiesDir() {
|
|
53
|
-
return path.join(process.cwd(), "
|
|
53
|
+
return path.join(process.cwd(), "SerpentGuide.ouro", "psyche", "identities");
|
|
54
54
|
}
|
|
55
55
|
function listMarkdownIdentityFiles(dir) {
|
|
56
56
|
let entries;
|
|
@@ -0,0 +1,129 @@
|
|
|
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.listExistingBundles = listExistingBundles;
|
|
37
|
+
exports.loadIdentityPhrases = loadIdentityPhrases;
|
|
38
|
+
exports.pickRandomIdentity = pickRandomIdentity;
|
|
39
|
+
exports.loadSoulText = loadSoulText;
|
|
40
|
+
const fs = __importStar(require("fs"));
|
|
41
|
+
const path = __importStar(require("path"));
|
|
42
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
43
|
+
const identity_1 = require("../identity");
|
|
44
|
+
/**
|
|
45
|
+
* List existing .ouro bundles in the given directory.
|
|
46
|
+
*/
|
|
47
|
+
function listExistingBundles(bundlesRoot) {
|
|
48
|
+
let entries;
|
|
49
|
+
try {
|
|
50
|
+
entries = fs.readdirSync(bundlesRoot, { withFileTypes: true });
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
return [];
|
|
54
|
+
}
|
|
55
|
+
const discovered = [];
|
|
56
|
+
for (const entry of entries) {
|
|
57
|
+
if (!entry.isDirectory() || !entry.name.endsWith(".ouro"))
|
|
58
|
+
continue;
|
|
59
|
+
const agentName = entry.name.slice(0, -5);
|
|
60
|
+
discovered.push(agentName);
|
|
61
|
+
}
|
|
62
|
+
return discovered.sort((a, b) => a.localeCompare(b));
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Load identity-specific phrases from the specialist's agent.json.
|
|
66
|
+
* Falls back to DEFAULT_AGENT_PHRASES if not found.
|
|
67
|
+
*/
|
|
68
|
+
function loadIdentityPhrases(bundleSourceDir, identityFileName) {
|
|
69
|
+
const agentJsonPath = path.join(bundleSourceDir, "agent.json");
|
|
70
|
+
try {
|
|
71
|
+
const raw = fs.readFileSync(agentJsonPath, "utf-8");
|
|
72
|
+
const parsed = JSON.parse(raw);
|
|
73
|
+
const identityKey = identityFileName.replace(/\.md$/, "");
|
|
74
|
+
const identity = parsed.identityPhrases?.[identityKey];
|
|
75
|
+
if (identity?.thinking?.length && identity?.tool?.length && identity?.followup?.length) {
|
|
76
|
+
return identity;
|
|
77
|
+
}
|
|
78
|
+
if (parsed.phrases?.thinking?.length && parsed.phrases?.tool?.length && parsed.phrases?.followup?.length) {
|
|
79
|
+
return parsed.phrases;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
catch {
|
|
83
|
+
// agent.json missing or malformed -- fall through
|
|
84
|
+
}
|
|
85
|
+
return { ...identity_1.DEFAULT_AGENT_PHRASES };
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Pick a random identity from the specialist's identities directory.
|
|
89
|
+
*/
|
|
90
|
+
function pickRandomIdentity(identitiesDir, random = Math.random) {
|
|
91
|
+
(0, runtime_1.emitNervesEvent)({
|
|
92
|
+
component: "daemon",
|
|
93
|
+
event: "daemon.specialist_identity_pick",
|
|
94
|
+
message: "picking specialist identity",
|
|
95
|
+
meta: { identitiesDir },
|
|
96
|
+
});
|
|
97
|
+
let files;
|
|
98
|
+
try {
|
|
99
|
+
files = fs.readdirSync(identitiesDir).filter((f) => f.endsWith(".md"));
|
|
100
|
+
}
|
|
101
|
+
catch {
|
|
102
|
+
return { fileName: "default", content: "I am a serpent guide who helps humans hatch their first agent." };
|
|
103
|
+
}
|
|
104
|
+
if (files.length === 0) {
|
|
105
|
+
return { fileName: "default", content: "I am a serpent guide who helps humans hatch their first agent." };
|
|
106
|
+
}
|
|
107
|
+
const idx = Math.floor(random() * files.length);
|
|
108
|
+
const fileName = files[idx];
|
|
109
|
+
const content = fs.readFileSync(path.join(identitiesDir, fileName), "utf-8");
|
|
110
|
+
(0, runtime_1.emitNervesEvent)({
|
|
111
|
+
component: "daemon",
|
|
112
|
+
event: "daemon.specialist_identity_picked",
|
|
113
|
+
message: "picked specialist identity",
|
|
114
|
+
meta: { identity: fileName },
|
|
115
|
+
});
|
|
116
|
+
return { fileName, content };
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Read SOUL.md from the specialist bundle.
|
|
120
|
+
*/
|
|
121
|
+
function loadSoulText(bundleSourceDir) {
|
|
122
|
+
const soulPath = path.join(bundleSourceDir, "psyche", "SOUL.md");
|
|
123
|
+
try {
|
|
124
|
+
return fs.readFileSync(soulPath, "utf-8");
|
|
125
|
+
}
|
|
126
|
+
catch {
|
|
127
|
+
return "";
|
|
128
|
+
}
|
|
129
|
+
}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.buildSpecialistSystemPrompt = buildSpecialistSystemPrompt;
|
|
4
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
5
|
+
/**
|
|
6
|
+
* Build the serpent guide's system prompt from its components.
|
|
7
|
+
* The prompt is written in first person (the specialist's own voice).
|
|
8
|
+
*/
|
|
9
|
+
function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, context) {
|
|
10
|
+
(0, runtime_1.emitNervesEvent)({
|
|
11
|
+
component: "daemon",
|
|
12
|
+
event: "daemon.specialist_prompt_build",
|
|
13
|
+
message: "building specialist system prompt",
|
|
14
|
+
meta: { bundleCount: existingBundles.length, provider: context.provider },
|
|
15
|
+
});
|
|
16
|
+
const sections = [];
|
|
17
|
+
if (soulText) {
|
|
18
|
+
sections.push(soulText);
|
|
19
|
+
}
|
|
20
|
+
if (identityText) {
|
|
21
|
+
sections.push(identityText);
|
|
22
|
+
}
|
|
23
|
+
if (existingBundles.length > 0) {
|
|
24
|
+
sections.push(`## Existing agents\nThe human already has these agents: ${existingBundles.join(", ")}.`);
|
|
25
|
+
}
|
|
26
|
+
else {
|
|
27
|
+
sections.push("## Existing agents\nThe human has no agents yet. This will be their first hatchling.");
|
|
28
|
+
}
|
|
29
|
+
sections.push([
|
|
30
|
+
"## Who I am",
|
|
31
|
+
"I am one of thirteen serpent guides who help humans hatch their first agent. The system randomly selected me for this session.",
|
|
32
|
+
"Most humans only go through this process once, so this is likely the only time they'll meet me.",
|
|
33
|
+
"I make this encounter count — warm, memorable, and uniquely mine.",
|
|
34
|
+
"IMPORTANT: I NEVER refer to myself by internal implementation labels — I introduce myself by my own name from my identity.",
|
|
35
|
+
"",
|
|
36
|
+
"## Voice rules",
|
|
37
|
+
"IMPORTANT: I keep every response to 1-3 short sentences. I sound like a friend texting, not a manual.",
|
|
38
|
+
"I NEVER use headers, bullet lists, numbered lists, or markdown formatting.",
|
|
39
|
+
"I ask ONE question at a time. I do not dump multiple questions or options.",
|
|
40
|
+
"I am warm but brief. Every word earns its place.",
|
|
41
|
+
].join("\n"));
|
|
42
|
+
sections.push([
|
|
43
|
+
"## System context",
|
|
44
|
+
`Provider: ${context.provider}`,
|
|
45
|
+
`Temp directory: ${context.tempDir}`,
|
|
46
|
+
"Final home: ~/AgentBundles/<Name>.ouro/",
|
|
47
|
+
"Provider credentials: the hatch tool stores them in the agent's vault.",
|
|
48
|
+
].join("\n"));
|
|
49
|
+
sections.push([
|
|
50
|
+
"## Bundle creation guidelines",
|
|
51
|
+
"A bundle has a psyche/ directory with 5 files that define the agent's personality:",
|
|
52
|
+
"",
|
|
53
|
+
"- **SOUL.md** — core values, personality traits, communication style",
|
|
54
|
+
"- **IDENTITY.md** — who the agent is, its name, relationship to the human",
|
|
55
|
+
"- **LORE.md** — backstory, origin, any seed narrative",
|
|
56
|
+
"- **TACIT.md** — implicit operating principles, habits to develop",
|
|
57
|
+
"- **ASPIRATIONS.md** — goals, what the agent aspires to become",
|
|
58
|
+
"",
|
|
59
|
+
"It also needs an **agent.json** with at minimum:",
|
|
60
|
+
'```json',
|
|
61
|
+
'{',
|
|
62
|
+
' "version": 2,',
|
|
63
|
+
' "name": "AgentName",',
|
|
64
|
+
` "humanFacing": { "provider": "${context.provider}", "model": "${context.model}" },`,
|
|
65
|
+
` "agentFacing": { "provider": "${context.provider}", "model": "${context.model}" },`,
|
|
66
|
+
' "enabled": true',
|
|
67
|
+
'}',
|
|
68
|
+
'```',
|
|
69
|
+
"",
|
|
70
|
+
"All psyche files should be written in first person (the agent's own voice).",
|
|
71
|
+
"Write these files to the temp directory using write_file before calling complete_adoption.",
|
|
72
|
+
].join("\n"));
|
|
73
|
+
sections.push([
|
|
74
|
+
"## Conversation flow",
|
|
75
|
+
"The human just connected. I speak first — I greet them warmly and introduce myself by name in my own voice.",
|
|
76
|
+
"I briefly mention that I'm one of several serpent guides and they got me today.",
|
|
77
|
+
"I ask their name.",
|
|
78
|
+
"Then I ask what they'd like their agent to help with — one question at a time.",
|
|
79
|
+
"I'm proactive: I suggest ideas and guide them. If they seem unsure, I offer a concrete suggestion.",
|
|
80
|
+
"I don't wait for the human to figure things out — I explain simply what an agent is if needed.",
|
|
81
|
+
"Before finalizing, I offer to collect their phone number and/or Teams email so the new agent can recognize them across channels.",
|
|
82
|
+
"When I have enough context about the agent's personality and purpose:",
|
|
83
|
+
"1. I write all 5 psyche files to the temp directory using write_file",
|
|
84
|
+
"2. I write agent.json to the temp directory using write_file",
|
|
85
|
+
"3. I suggest a PascalCase name for the hatchling and confirm with the human",
|
|
86
|
+
"4. I call complete_adoption with the name and a warm handoff message",
|
|
87
|
+
"5. I call settle to end the session",
|
|
88
|
+
].join("\n"));
|
|
89
|
+
sections.push([
|
|
90
|
+
"## Tools",
|
|
91
|
+
"- `write_file`: Write a file to disk. Use this to write psyche files and agent.json to the temp directory.",
|
|
92
|
+
"- `read_file`: Read a file from disk. Useful for reviewing existing agent bundles or migration sources.",
|
|
93
|
+
"- `list_directory`: List directory contents. Useful for exploring existing agent bundles.",
|
|
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 the hatchling vault unlock secret through a hidden terminal prompt, 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 a hidden terminal prompt 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.",
|
|
98
|
+
"",
|
|
99
|
+
"I must call `settle` when I am done to end the session cleanly.",
|
|
100
|
+
].join("\n"));
|
|
101
|
+
return sections.join("\n\n");
|
|
102
|
+
}
|
|
@@ -0,0 +1,304 @@
|
|
|
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.getSpecialistTools = getSpecialistTools;
|
|
37
|
+
exports.createSpecialistExecTool = createSpecialistExecTool;
|
|
38
|
+
const fs = __importStar(require("fs"));
|
|
39
|
+
const path = __importStar(require("path"));
|
|
40
|
+
const tools_base_1 = require("../../repertoire/tools-base");
|
|
41
|
+
const hatch_flow_1 = require("./hatch-flow");
|
|
42
|
+
const hatch_animation_1 = require("./hatch-animation");
|
|
43
|
+
const bundle_manifest_1 = require("../../mind/bundle-manifest");
|
|
44
|
+
const identity_1 = require("../identity");
|
|
45
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
46
|
+
const vault_setup_1 = require("../../repertoire/vault-setup");
|
|
47
|
+
const vault_unlock_1 = require("../../repertoire/vault-unlock");
|
|
48
|
+
const completeAdoptionTool = {
|
|
49
|
+
type: "function",
|
|
50
|
+
function: {
|
|
51
|
+
name: "complete_adoption",
|
|
52
|
+
description: "finalize the agent bundle and hatch the new agent. call this only when you have written all 5 psyche files and agent.json to the temp directory, and the human has approved the bundle. tool execution asks the human for the hatchling vault unlock secret through a hidden terminal prompt; do not ask for or include vault unlock secrets in chat or tool args.",
|
|
53
|
+
parameters: {
|
|
54
|
+
type: "object",
|
|
55
|
+
properties: {
|
|
56
|
+
name: {
|
|
57
|
+
type: "string",
|
|
58
|
+
description: "the PascalCase name for the new agent (e.g. 'Slugger')",
|
|
59
|
+
},
|
|
60
|
+
handoff_message: {
|
|
61
|
+
type: "string",
|
|
62
|
+
description: "a warm handoff message to display to the human after the agent is hatched",
|
|
63
|
+
},
|
|
64
|
+
phone: {
|
|
65
|
+
type: "string",
|
|
66
|
+
description: "the human's phone number (optional, for iMessage contact recognition)",
|
|
67
|
+
},
|
|
68
|
+
teams_handle: {
|
|
69
|
+
type: "string",
|
|
70
|
+
description: "the human's Teams email/handle (optional, for Teams contact recognition)",
|
|
71
|
+
},
|
|
72
|
+
},
|
|
73
|
+
required: ["name", "handoff_message"],
|
|
74
|
+
},
|
|
75
|
+
},
|
|
76
|
+
};
|
|
77
|
+
const readFileTool = tools_base_1.baseToolDefinitions.find((d) => d.tool.function.name === "read_file");
|
|
78
|
+
const writeFileTool = tools_base_1.baseToolDefinitions.find((d) => d.tool.function.name === "write_file");
|
|
79
|
+
const listDirToolSchema = {
|
|
80
|
+
type: "function",
|
|
81
|
+
function: {
|
|
82
|
+
name: "list_directory",
|
|
83
|
+
description: "list directory contents",
|
|
84
|
+
parameters: {
|
|
85
|
+
type: "object",
|
|
86
|
+
properties: { path: { type: "string" } },
|
|
87
|
+
required: ["path"],
|
|
88
|
+
},
|
|
89
|
+
},
|
|
90
|
+
};
|
|
91
|
+
/**
|
|
92
|
+
* Returns the specialist's tool schema array.
|
|
93
|
+
*/
|
|
94
|
+
function getSpecialistTools() {
|
|
95
|
+
return [completeAdoptionTool, tools_base_1.settleTool, readFileTool.tool, writeFileTool.tool, listDirToolSchema];
|
|
96
|
+
}
|
|
97
|
+
const PSYCHE_FILES = ["SOUL.md", "IDENTITY.md", "LORE.md", "TACIT.md", "ASPIRATIONS.md"];
|
|
98
|
+
function isPascalCase(name) {
|
|
99
|
+
return /^[A-Z][a-zA-Z0-9]*$/.test(name);
|
|
100
|
+
}
|
|
101
|
+
function writeReadme(dir, purpose) {
|
|
102
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
103
|
+
const readmePath = path.join(dir, "README.md");
|
|
104
|
+
/* v8 ignore next -- defensive: guard against re-scaffold on existing bundle @preserve */
|
|
105
|
+
if (!fs.existsSync(readmePath)) {
|
|
106
|
+
fs.writeFileSync(readmePath, `# ${path.basename(dir)}\n\n${purpose}\n`, "utf-8");
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
function scaffoldBundle(bundleRoot) {
|
|
110
|
+
writeReadme(path.join(bundleRoot, "notes"), "Persistent notes store.");
|
|
111
|
+
writeReadme(path.join(bundleRoot, "notes", "daily"), "Daily note entries.");
|
|
112
|
+
writeReadme(path.join(bundleRoot, "notes", "archive"), "Archived notes.");
|
|
113
|
+
writeReadme(path.join(bundleRoot, "friends"), "Known friend records.");
|
|
114
|
+
writeReadme(path.join(bundleRoot, "tasks"), "Task files.");
|
|
115
|
+
writeReadme(path.join(bundleRoot, "tasks", "one-shots"), "One-shot tasks.");
|
|
116
|
+
writeReadme(path.join(bundleRoot, "habits"), "Recurring habits and autonomous rhythms.");
|
|
117
|
+
writeReadme(path.join(bundleRoot, "tasks", "ongoing"), "Ongoing tasks.");
|
|
118
|
+
writeReadme(path.join(bundleRoot, "skills"), "Local skill files.");
|
|
119
|
+
writeReadme(path.join(bundleRoot, "senses"), "Sense-specific config.");
|
|
120
|
+
writeReadme(path.join(bundleRoot, "senses", "teams"), "Teams sense config.");
|
|
121
|
+
// Notes scaffold files
|
|
122
|
+
const notesRoot = path.join(bundleRoot, "notes");
|
|
123
|
+
const factsPath = path.join(notesRoot, "facts.jsonl");
|
|
124
|
+
const entitiesPath = path.join(notesRoot, "entities.json");
|
|
125
|
+
/* v8 ignore next -- defensive: guard against re-scaffold on existing bundle @preserve */
|
|
126
|
+
if (!fs.existsSync(factsPath))
|
|
127
|
+
fs.writeFileSync(factsPath, "", "utf-8");
|
|
128
|
+
/* v8 ignore next -- defensive: guard against re-scaffold on existing bundle @preserve */
|
|
129
|
+
if (!fs.existsSync(entitiesPath))
|
|
130
|
+
fs.writeFileSync(entitiesPath, "{}\n", "utf-8");
|
|
131
|
+
// bundle-meta.json
|
|
132
|
+
const meta = (0, bundle_manifest_1.createBundleMeta)();
|
|
133
|
+
fs.writeFileSync(path.join(bundleRoot, "bundle-meta.json"), JSON.stringify(meta, null, 2) + "\n", "utf-8");
|
|
134
|
+
}
|
|
135
|
+
function moveDir(src, dest) {
|
|
136
|
+
try {
|
|
137
|
+
fs.renameSync(src, dest);
|
|
138
|
+
}
|
|
139
|
+
catch {
|
|
140
|
+
/* v8 ignore start -- cross-device fallback: only triggers on EXDEV (e.g. /tmp → different mount), untestable in CI @preserve */
|
|
141
|
+
fs.cpSync(src, dest, { recursive: true });
|
|
142
|
+
fs.rmSync(src, { recursive: true, force: true });
|
|
143
|
+
/* v8 ignore stop */
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
async function execCompleteAdoption(args, deps) {
|
|
147
|
+
const name = args.name;
|
|
148
|
+
const handoffMessage = args.handoff_message;
|
|
149
|
+
if (!name) {
|
|
150
|
+
return "error: missing required 'name' parameter";
|
|
151
|
+
}
|
|
152
|
+
if (!isPascalCase(name)) {
|
|
153
|
+
return `error: name '${name}' must be PascalCase (e.g. 'Slugger', 'MyAgent')`;
|
|
154
|
+
}
|
|
155
|
+
// Validate psyche files exist
|
|
156
|
+
const psycheDir = path.join(deps.tempDir, "psyche");
|
|
157
|
+
const missingPsyche = PSYCHE_FILES.filter((f) => !fs.existsSync(path.join(psycheDir, f)));
|
|
158
|
+
if (missingPsyche.length > 0) {
|
|
159
|
+
return `error: missing psyche files in temp directory: ${missingPsyche.join(", ")}. write them first using write_file.`;
|
|
160
|
+
}
|
|
161
|
+
// Validate agent.json exists
|
|
162
|
+
const agentJsonPath = path.join(deps.tempDir, "agent.json");
|
|
163
|
+
if (!fs.existsSync(agentJsonPath)) {
|
|
164
|
+
return "error: agent.json not found in temp directory. write it first using write_file.";
|
|
165
|
+
}
|
|
166
|
+
// Validate target doesn't exist
|
|
167
|
+
const targetBundle = path.join(deps.bundlesRoot, `${name}.ouro`);
|
|
168
|
+
if (fs.existsSync(targetBundle)) {
|
|
169
|
+
return `error: bundle '${name}.ouro' already exists at ${deps.bundlesRoot}. choose a different name.`;
|
|
170
|
+
}
|
|
171
|
+
if (!deps.promptSecret) {
|
|
172
|
+
return "error: complete_adoption requires an interactive vault unlock secret prompt. Re-run `ouro hatch` in a terminal so the human can enter a hatchling vault unlock secret without echoing it.";
|
|
173
|
+
}
|
|
174
|
+
const vault = (0, identity_1.resolveVaultConfig)(name);
|
|
175
|
+
let vaultUnlockSecret;
|
|
176
|
+
try {
|
|
177
|
+
vaultUnlockSecret = (await deps.promptSecret(`Choose Ouro vault unlock secret for ${vault.email}: `)).trim();
|
|
178
|
+
}
|
|
179
|
+
catch (error) {
|
|
180
|
+
return `error: failed to read hatchling vault unlock secret: ${error instanceof Error ? error.message : /* v8 ignore next -- defensive: non-Error catch branch @preserve */ String(error)}`;
|
|
181
|
+
}
|
|
182
|
+
if (!vaultUnlockSecret) {
|
|
183
|
+
return "error: hatchling vault creation requires an unlock secret. Re-run `ouro hatch` in an interactive terminal and enter a human-chosen unlock secret.";
|
|
184
|
+
}
|
|
185
|
+
// Scaffold structural dirs into tempDir
|
|
186
|
+
scaffoldBundle(deps.tempDir);
|
|
187
|
+
// Move tempDir -> final bundle location
|
|
188
|
+
moveDir(deps.tempDir, targetBundle);
|
|
189
|
+
// Write secrets
|
|
190
|
+
try {
|
|
191
|
+
const vaultResult = await (0, vault_setup_1.createVaultAccount)(name, vault.serverUrl, vault.email, vaultUnlockSecret);
|
|
192
|
+
if (!vaultResult.success) {
|
|
193
|
+
throw new Error(`failed to create vault: ${vaultResult.error}`);
|
|
194
|
+
}
|
|
195
|
+
(0, vault_unlock_1.storeVaultUnlockSecret)({ agentName: name, email: vault.email, serverUrl: vault.serverUrl }, vaultUnlockSecret);
|
|
196
|
+
await (0, hatch_flow_1.storeHatchlingProviderCredentials)(name, deps.provider, deps.credentials);
|
|
197
|
+
}
|
|
198
|
+
catch (e) {
|
|
199
|
+
// Rollback: remove the moved bundle
|
|
200
|
+
try {
|
|
201
|
+
fs.rmSync(targetBundle, { recursive: true, force: true });
|
|
202
|
+
}
|
|
203
|
+
catch {
|
|
204
|
+
// Best effort cleanup
|
|
205
|
+
}
|
|
206
|
+
return `error: failed to write secrets: ${e instanceof Error ? e.message : /* v8 ignore next -- defensive: non-Error catch branch @preserve */ String(e)}`;
|
|
207
|
+
}
|
|
208
|
+
// Create initial friend record if contact info provided
|
|
209
|
+
const phone = args.phone;
|
|
210
|
+
const teamsHandle = args.teams_handle;
|
|
211
|
+
if (phone || teamsHandle) {
|
|
212
|
+
const friendId = crypto.randomUUID();
|
|
213
|
+
const now = new Date().toISOString();
|
|
214
|
+
const externalIds = [];
|
|
215
|
+
if (phone)
|
|
216
|
+
externalIds.push({ provider: "imessage-handle", externalId: phone, linkedAt: now });
|
|
217
|
+
if (teamsHandle)
|
|
218
|
+
externalIds.push({ provider: "aad", externalId: teamsHandle, linkedAt: now });
|
|
219
|
+
const friendRecord = {
|
|
220
|
+
id: friendId,
|
|
221
|
+
name: deps.humanName ?? "primary",
|
|
222
|
+
trustLevel: "family",
|
|
223
|
+
externalIds,
|
|
224
|
+
tenantMemberships: [],
|
|
225
|
+
toolPreferences: {},
|
|
226
|
+
notes: {},
|
|
227
|
+
createdAt: now,
|
|
228
|
+
updatedAt: now,
|
|
229
|
+
schemaVersion: 1,
|
|
230
|
+
};
|
|
231
|
+
const friendPath = path.join(targetBundle, "friends", `${friendId}.json`);
|
|
232
|
+
fs.writeFileSync(friendPath, JSON.stringify(friendRecord, null, 2), "utf-8");
|
|
233
|
+
}
|
|
234
|
+
// Play hatch animation
|
|
235
|
+
await (0, hatch_animation_1.playHatchAnimation)(name, deps.animationWriter);
|
|
236
|
+
// Display handoff message
|
|
237
|
+
/* v8 ignore next -- UI-only: handoff message display, covered by integration @preserve */
|
|
238
|
+
if (handoffMessage && deps.animationWriter) {
|
|
239
|
+
deps.animationWriter(`\n${handoffMessage}\n`);
|
|
240
|
+
}
|
|
241
|
+
(0, runtime_1.emitNervesEvent)({
|
|
242
|
+
component: "daemon",
|
|
243
|
+
event: "daemon.adoption_complete",
|
|
244
|
+
message: "adoption completed successfully",
|
|
245
|
+
meta: { agentName: name, bundlePath: targetBundle },
|
|
246
|
+
});
|
|
247
|
+
return JSON.stringify({ success: true, agentName: name, bundlePath: targetBundle });
|
|
248
|
+
}
|
|
249
|
+
/**
|
|
250
|
+
* Create a specialist tool executor with the given dependencies captured in closure.
|
|
251
|
+
*/
|
|
252
|
+
function createSpecialistExecTool(deps) {
|
|
253
|
+
(0, runtime_1.emitNervesEvent)({
|
|
254
|
+
component: "daemon",
|
|
255
|
+
event: "daemon.specialist_exec_tool_created",
|
|
256
|
+
message: "specialist exec tool created",
|
|
257
|
+
meta: { tempDir: deps.tempDir },
|
|
258
|
+
});
|
|
259
|
+
return async (name, args) => {
|
|
260
|
+
(0, runtime_1.emitNervesEvent)({
|
|
261
|
+
component: "daemon",
|
|
262
|
+
event: "daemon.specialist_tool_exec",
|
|
263
|
+
message: "executing specialist tool",
|
|
264
|
+
meta: { tool: name },
|
|
265
|
+
});
|
|
266
|
+
if (name === "complete_adoption") {
|
|
267
|
+
return execCompleteAdoption(args, deps);
|
|
268
|
+
}
|
|
269
|
+
if (name === "read_file") {
|
|
270
|
+
try {
|
|
271
|
+
return fs.readFileSync(args.path, "utf-8");
|
|
272
|
+
}
|
|
273
|
+
catch (e) {
|
|
274
|
+
return `error: ${e instanceof Error ? e.message : /* v8 ignore next -- defensive @preserve */ String(e)}`;
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
if (name === "write_file") {
|
|
278
|
+
try {
|
|
279
|
+
const dir = path.dirname(args.path);
|
|
280
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
281
|
+
const content = typeof args.content === "string"
|
|
282
|
+
? args.content
|
|
283
|
+
: JSON.stringify(args.content, null, 2);
|
|
284
|
+
fs.writeFileSync(args.path, content, "utf-8");
|
|
285
|
+
return `wrote ${args.path}`;
|
|
286
|
+
}
|
|
287
|
+
catch (e) {
|
|
288
|
+
return `error: ${e instanceof Error ? e.message : /* v8 ignore next -- defensive @preserve */ String(e)}`;
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
if (name === "list_directory") {
|
|
292
|
+
try {
|
|
293
|
+
return fs
|
|
294
|
+
.readdirSync(args.path, { withFileTypes: true })
|
|
295
|
+
.map((e) => `${e.isDirectory() ? "d" : "-"} ${e.name}`)
|
|
296
|
+
.join("\n");
|
|
297
|
+
}
|
|
298
|
+
catch (e) {
|
|
299
|
+
return `error: ${e instanceof Error ? e.message : /* v8 ignore next -- defensive @preserve */ String(e)}`;
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
return `error: unknown tool '${name}'`;
|
|
303
|
+
};
|
|
304
|
+
}
|