@ouro.bot/cli 0.1.0-alpha.34 → 0.1.0-alpha.340
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 +188 -187
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +1 -1
- package/changelog.json +2031 -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 +37 -2
- 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 +463 -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 +53 -21
- package/dist/heart/core.js +695 -195
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +292 -0
- package/dist/heart/daemon/agent-discovery.js +79 -3
- package/dist/heart/daemon/agent-service.js +360 -0
- package/dist/heart/daemon/agentic-repair.js +170 -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 +591 -0
- package/dist/heart/daemon/cli-exec.js +2297 -0
- package/dist/heart/daemon/cli-help.js +306 -0
- package/dist/heart/daemon/cli-parse.js +824 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +512 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/daemon-cli.js +30 -1171
- package/dist/heart/daemon/daemon-entry.js +358 -3
- package/dist/heart/daemon/daemon-health.js +141 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +157 -12
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +751 -58
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +401 -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 +115 -1
- 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 +91 -0
- package/dist/heart/daemon/launchd.js +46 -9
- 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 +1 -1
- package/dist/heart/daemon/process-manager.js +201 -0
- package/dist/heart/daemon/provider-discovery.js +105 -0
- package/dist/heart/daemon/pulse.js +463 -0
- package/dist/heart/daemon/run-hooks.js +2 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +101 -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 +72 -3
- 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/{daemon → hatch}/hatch-flow.js +32 -120
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +10 -7
- package/dist/heart/{daemon → hatch}/specialist-tools.js +49 -3
- package/dist/heart/identity.js +154 -59
- package/dist/heart/kicks.js +2 -20
- package/dist/heart/mcp/mcp-server.js +653 -0
- package/dist/heart/migrate-config.js +127 -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 +194 -0
- package/dist/heart/outlook/readers/agent-machine.js +355 -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 +231 -0
- package/dist/heart/outlook/readers/shared.js +111 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/provider-failover.js +135 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +234 -0
- package/dist/heart/providers/anthropic-token.js +163 -0
- package/dist/heart/providers/anthropic.js +171 -50
- package/dist/heart/providers/azure.js +97 -11
- package/dist/heart/providers/error-classification.js +63 -0
- package/dist/heart/providers/github-copilot.js +135 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +23 -6
- package/dist/heart/providers/openai-codex.js +33 -23
- package/dist/heart/session-activity.js +190 -0
- package/dist/heart/session-events.js +726 -0
- package/dist/heart/session-recall.js +162 -0
- package/dist/heart/start-of-turn-packet.js +341 -0
- package/dist/heart/streaming.js +36 -27
- 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 +358 -0
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/{daemon → versioning}/ouro-path-installer.js +78 -35
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +12 -2
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mind/associative-recall.js +137 -66
- package/dist/mind/bundle-manifest.js +7 -1
- package/dist/mind/context.js +89 -93
- 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 +14 -1
- package/dist/mind/friends/channel.js +56 -0
- package/dist/mind/friends/group-context.js +144 -0
- package/dist/mind/friends/resolver.js +37 -0
- package/dist/mind/friends/store-file.js +58 -3
- package/dist/mind/friends/trust-explanation.js +74 -0
- package/dist/mind/friends/types.js +8 -0
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +74 -7
- package/dist/mind/prompt.js +999 -111
- 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 +7 -1
- package/dist/nerves/coverage/audit.js +1 -1
- 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-DC7sZefn.js +61 -0
- package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
- package/dist/outlook-ui/index.html +15 -0
- package/dist/repertoire/ado-client.js +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +319 -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 +197 -30
- package/dist/repertoire/coding/manager.js +158 -9
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +170 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +527 -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 +375 -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 +14 -23
- 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 +316 -0
- package/dist/repertoire/tools-base.js +45 -771
- package/dist/repertoire/tools-bluebubbles.js +1 -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-memory.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 +12 -62
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +110 -0
- package/dist/repertoire/tools.js +144 -138
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +118 -0
- package/dist/repertoire/vault-setup.js +241 -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 → bluebubbles/client.js} +225 -9
- package/dist/senses/bluebubbles/entry.js +13 -0
- package/dist/senses/bluebubbles/inbound-log.js +113 -0
- package/dist/senses/bluebubbles/index.js +1590 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
- package/dist/senses/bluebubbles/replay.js +129 -0
- package/dist/senses/bluebubbles/runtime-state.js +109 -0
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +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 +1 -1
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +595 -246
- package/dist/senses/commands.js +65 -1
- 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 +633 -86
- package/dist/senses/pipeline.js +567 -0
- package/dist/senses/shared-turn.js +199 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams.js +665 -160
- package/dist/senses/trust-gate.js +112 -2
- package/package.json +29 -7
- 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 +81 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/subagent-installer.js +0 -134
- package/dist/senses/bluebubbles-entry.js +0 -11
- package/dist/senses/bluebubbles.js +0 -547
- package/dist/senses/debug-activity.js +0 -124
- package/subagents/README.md +0 -73
- package/subagents/work-doer.md +0 -235
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -382
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
- /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
|
@@ -0,0 +1,371 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.HabitScheduler = void 0;
|
|
37
|
+
const path = __importStar(require("path"));
|
|
38
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
39
|
+
const habit_parser_1 = require("./habit-parser");
|
|
40
|
+
const cadence_1 = require("../daemon/cadence");
|
|
41
|
+
const WATCH_DEBOUNCE_MS = 200;
|
|
42
|
+
class HabitScheduler {
|
|
43
|
+
agent;
|
|
44
|
+
habitsDir;
|
|
45
|
+
osCronManager;
|
|
46
|
+
onHabitFire;
|
|
47
|
+
deps;
|
|
48
|
+
execForVerify;
|
|
49
|
+
platform;
|
|
50
|
+
watcher = null;
|
|
51
|
+
debounceTimer = null;
|
|
52
|
+
parseErrors = [];
|
|
53
|
+
timerFallbacks = new Map();
|
|
54
|
+
degradedHabitNames = new Map();
|
|
55
|
+
periodicTimer = null;
|
|
56
|
+
constructor(options) {
|
|
57
|
+
this.agent = options.agent;
|
|
58
|
+
this.habitsDir = options.habitsDir;
|
|
59
|
+
this.osCronManager = options.osCronManager;
|
|
60
|
+
this.onHabitFire = options.onHabitFire;
|
|
61
|
+
this.deps = options.deps;
|
|
62
|
+
this.execForVerify = options.execForVerify;
|
|
63
|
+
this.platform = options.platform ?? process.platform;
|
|
64
|
+
}
|
|
65
|
+
start() {
|
|
66
|
+
(0, runtime_1.emitNervesEvent)({
|
|
67
|
+
component: "daemon",
|
|
68
|
+
event: "daemon.habit_scheduler_start",
|
|
69
|
+
message: "habit scheduler starting",
|
|
70
|
+
meta: { agent: this.agent, habitsDir: this.habitsDir },
|
|
71
|
+
});
|
|
72
|
+
const habits = this.scanHabits();
|
|
73
|
+
const jobs = this.buildJobs(habits);
|
|
74
|
+
this.osCronManager.sync(jobs);
|
|
75
|
+
this.verifyCronAndCreateFallbacks(jobs);
|
|
76
|
+
this.fireOverdueHabits(habits);
|
|
77
|
+
}
|
|
78
|
+
reconcile() {
|
|
79
|
+
(0, runtime_1.emitNervesEvent)({
|
|
80
|
+
component: "daemon",
|
|
81
|
+
event: "daemon.habit_scheduler_reconcile",
|
|
82
|
+
message: "habit scheduler reconciling",
|
|
83
|
+
meta: { agent: this.agent },
|
|
84
|
+
});
|
|
85
|
+
// Clear ALL existing timers FIRST to prevent overlap window
|
|
86
|
+
this.clearAllTimerFallbacks();
|
|
87
|
+
const habits = this.scanHabits();
|
|
88
|
+
const jobs = this.buildJobs(habits);
|
|
89
|
+
this.osCronManager.sync(jobs);
|
|
90
|
+
this.verifyCronAndCreateFallbacks(jobs);
|
|
91
|
+
this.fireOverdueHabits(habits);
|
|
92
|
+
}
|
|
93
|
+
fireOverdueHabits(habits) {
|
|
94
|
+
for (const habit of habits) {
|
|
95
|
+
if (habit.status !== "active")
|
|
96
|
+
continue;
|
|
97
|
+
if (!habit.cadence)
|
|
98
|
+
continue;
|
|
99
|
+
const cadenceMs = (0, cadence_1.parseCadenceToMs)(habit.cadence);
|
|
100
|
+
if (cadenceMs === null)
|
|
101
|
+
continue;
|
|
102
|
+
const nowMs = this.deps.now();
|
|
103
|
+
if (habit.lastRun === null) {
|
|
104
|
+
(0, runtime_1.emitNervesEvent)({
|
|
105
|
+
component: "daemon",
|
|
106
|
+
event: "daemon.habit_fire",
|
|
107
|
+
message: "firing overdue habit (never run)",
|
|
108
|
+
meta: { habitName: habit.name, agent: this.agent },
|
|
109
|
+
});
|
|
110
|
+
this.onHabitFire(habit.name);
|
|
111
|
+
continue;
|
|
112
|
+
}
|
|
113
|
+
const lastRunMs = new Date(habit.lastRun).getTime();
|
|
114
|
+
const elapsed = nowMs - lastRunMs;
|
|
115
|
+
if (elapsed >= cadenceMs) {
|
|
116
|
+
(0, runtime_1.emitNervesEvent)({
|
|
117
|
+
component: "daemon",
|
|
118
|
+
event: "daemon.habit_fire",
|
|
119
|
+
message: "firing overdue habit",
|
|
120
|
+
meta: { habitName: habit.name, agent: this.agent, elapsedMs: elapsed },
|
|
121
|
+
});
|
|
122
|
+
this.onHabitFire(habit.name);
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
stop() {
|
|
127
|
+
// `_end` (not `_stop`) to pair with `daemon.habit_scheduler_start`
|
|
128
|
+
// under the nerves audit start/end pairing rule.
|
|
129
|
+
(0, runtime_1.emitNervesEvent)({
|
|
130
|
+
component: "daemon",
|
|
131
|
+
event: "daemon.habit_scheduler_end",
|
|
132
|
+
message: "habit scheduler stopping",
|
|
133
|
+
meta: { agent: this.agent },
|
|
134
|
+
});
|
|
135
|
+
this.stopPeriodicReconciliation();
|
|
136
|
+
this.clearAllTimerFallbacks();
|
|
137
|
+
this.osCronManager.removeAll();
|
|
138
|
+
}
|
|
139
|
+
listOverdueHabits() {
|
|
140
|
+
const habits = this.scanHabits();
|
|
141
|
+
const nowMs = this.deps.now();
|
|
142
|
+
const overdue = [];
|
|
143
|
+
for (const habit of habits) {
|
|
144
|
+
if (habit.status !== "active")
|
|
145
|
+
continue;
|
|
146
|
+
if (!habit.cadence)
|
|
147
|
+
continue;
|
|
148
|
+
const cadenceMs = (0, cadence_1.parseCadenceToMs)(habit.cadence);
|
|
149
|
+
if (cadenceMs === null)
|
|
150
|
+
continue;
|
|
151
|
+
if (habit.lastRun === null) {
|
|
152
|
+
overdue.push({ name: habit.name, elapsedMs: Infinity });
|
|
153
|
+
continue;
|
|
154
|
+
}
|
|
155
|
+
const lastRunMs = new Date(habit.lastRun).getTime();
|
|
156
|
+
const elapsed = nowMs - lastRunMs;
|
|
157
|
+
if (elapsed >= cadenceMs) {
|
|
158
|
+
overdue.push({ name: habit.name, elapsedMs: elapsed });
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
return overdue;
|
|
162
|
+
}
|
|
163
|
+
getParseErrors() {
|
|
164
|
+
return [...this.parseErrors];
|
|
165
|
+
}
|
|
166
|
+
getHabitFile(name) {
|
|
167
|
+
const filePath = path.join(this.habitsDir, `${name}.md`);
|
|
168
|
+
try {
|
|
169
|
+
const content = this.deps.readFile(filePath, "utf-8");
|
|
170
|
+
return (0, habit_parser_1.parseHabitFile)(content, filePath);
|
|
171
|
+
}
|
|
172
|
+
catch {
|
|
173
|
+
return null;
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
watchForChanges() {
|
|
177
|
+
const watchFn = this.deps.watch;
|
|
178
|
+
if (!watchFn)
|
|
179
|
+
return;
|
|
180
|
+
// Ensure habits directory exists before watching — agents may not have one yet
|
|
181
|
+
try {
|
|
182
|
+
this.watcher = watchFn(this.habitsDir, (_event, _filename) => {
|
|
183
|
+
if (this.debounceTimer !== null) {
|
|
184
|
+
clearTimeout(this.debounceTimer);
|
|
185
|
+
}
|
|
186
|
+
this.debounceTimer = setTimeout(() => {
|
|
187
|
+
this.debounceTimer = null;
|
|
188
|
+
this.reconcile();
|
|
189
|
+
}, WATCH_DEBOUNCE_MS);
|
|
190
|
+
});
|
|
191
|
+
/* v8 ignore start — ENOENT catch requires real missing directory @preserve */
|
|
192
|
+
}
|
|
193
|
+
catch {
|
|
194
|
+
// habits directory may not exist for all agents — skip watching silently
|
|
195
|
+
}
|
|
196
|
+
/* v8 ignore stop */
|
|
197
|
+
}
|
|
198
|
+
stopWatching() {
|
|
199
|
+
if (this.debounceTimer !== null) {
|
|
200
|
+
clearTimeout(this.debounceTimer);
|
|
201
|
+
this.debounceTimer = null;
|
|
202
|
+
}
|
|
203
|
+
if (this.watcher !== null) {
|
|
204
|
+
this.watcher.close();
|
|
205
|
+
this.watcher = null;
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
getDegradedHabits() {
|
|
209
|
+
const result = [];
|
|
210
|
+
for (const [name, reason] of this.degradedHabitNames) {
|
|
211
|
+
result.push({ name, reason });
|
|
212
|
+
}
|
|
213
|
+
return result;
|
|
214
|
+
}
|
|
215
|
+
static DEFAULT_PERIODIC_INTERVAL_MS = 300_000; // 5 minutes
|
|
216
|
+
static INITIAL_RECONCILIATION_DELAY_MS = 30_000; // 30 seconds
|
|
217
|
+
startPeriodicReconciliation(intervalMs) {
|
|
218
|
+
const interval = intervalMs ?? HabitScheduler.DEFAULT_PERIODIC_INTERVAL_MS;
|
|
219
|
+
// First reconciliation after a short delay (30s)
|
|
220
|
+
this.periodicTimer = setTimeout(() => {
|
|
221
|
+
this.reconcile();
|
|
222
|
+
this.scheduleNextReconciliation(interval);
|
|
223
|
+
}, HabitScheduler.INITIAL_RECONCILIATION_DELAY_MS);
|
|
224
|
+
}
|
|
225
|
+
stopPeriodicReconciliation() {
|
|
226
|
+
if (this.periodicTimer !== null) {
|
|
227
|
+
clearTimeout(this.periodicTimer);
|
|
228
|
+
this.periodicTimer = null;
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
scheduleNextReconciliation(intervalMs) {
|
|
232
|
+
this.periodicTimer = setTimeout(() => {
|
|
233
|
+
this.reconcile();
|
|
234
|
+
this.scheduleNextReconciliation(intervalMs);
|
|
235
|
+
}, intervalMs);
|
|
236
|
+
}
|
|
237
|
+
verifyCronAndCreateFallbacks(jobs) {
|
|
238
|
+
if (!this.execForVerify)
|
|
239
|
+
return;
|
|
240
|
+
const verifiedLabels = this.verifyCronEntries();
|
|
241
|
+
for (const job of jobs) {
|
|
242
|
+
const label = `bot.ouro.${job.agent}.${job.taskId}`;
|
|
243
|
+
const isVerified = this.platform === "darwin"
|
|
244
|
+
? verifiedLabels.has(label)
|
|
245
|
+
: verifiedLabels.has(job.taskId);
|
|
246
|
+
if (!isVerified) {
|
|
247
|
+
(0, runtime_1.emitNervesEvent)({
|
|
248
|
+
component: "daemon",
|
|
249
|
+
event: "daemon.habit_cron_verification_failed",
|
|
250
|
+
message: `cron verification failed for habit: ${job.taskId}`,
|
|
251
|
+
meta: { habitName: job.taskId, agent: job.agent, label },
|
|
252
|
+
});
|
|
253
|
+
// Parse cadence from the original habit file for timer interval
|
|
254
|
+
const habitFile = this.getHabitFile(job.taskId);
|
|
255
|
+
const ms = habitFile?.cadence ? (0, cadence_1.parseCadenceToMs)(habitFile.cadence) : null;
|
|
256
|
+
if (ms !== null) {
|
|
257
|
+
this.createTimerFallback(job.taskId, ms);
|
|
258
|
+
}
|
|
259
|
+
this.degradedHabitNames.set(job.taskId, "cron registration failed — using timer fallback");
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
verifyCronEntries() {
|
|
264
|
+
const verified = new Set();
|
|
265
|
+
try {
|
|
266
|
+
if (this.platform === "darwin") {
|
|
267
|
+
const output = this.execForVerify("launchctl list");
|
|
268
|
+
const lines = output.split("\n");
|
|
269
|
+
for (const line of lines) {
|
|
270
|
+
const match = line.match(/bot\.ouro\.\S+\.\S+/);
|
|
271
|
+
if (match) {
|
|
272
|
+
verified.add(match[0]);
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
else {
|
|
277
|
+
const output = this.execForVerify("crontab -l");
|
|
278
|
+
const lines = output.split("\n");
|
|
279
|
+
for (const line of lines) {
|
|
280
|
+
const match = line.match(/ouro poke \S+ --habit (\S+)/);
|
|
281
|
+
if (match) {
|
|
282
|
+
verified.add(match[1]);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
catch {
|
|
288
|
+
// Verification command failed — return empty set (all habits unverified)
|
|
289
|
+
}
|
|
290
|
+
return verified;
|
|
291
|
+
}
|
|
292
|
+
createTimerFallback(habitName, cadenceMs) {
|
|
293
|
+
const schedule = () => {
|
|
294
|
+
const timer = setTimeout(() => {
|
|
295
|
+
this.onHabitFire(habitName);
|
|
296
|
+
schedule();
|
|
297
|
+
}, cadenceMs);
|
|
298
|
+
this.timerFallbacks.set(habitName, timer);
|
|
299
|
+
};
|
|
300
|
+
schedule();
|
|
301
|
+
}
|
|
302
|
+
clearAllTimerFallbacks() {
|
|
303
|
+
for (const timer of this.timerFallbacks.values()) {
|
|
304
|
+
clearTimeout(timer);
|
|
305
|
+
}
|
|
306
|
+
this.timerFallbacks.clear();
|
|
307
|
+
this.degradedHabitNames.clear();
|
|
308
|
+
}
|
|
309
|
+
scanHabits() {
|
|
310
|
+
let files;
|
|
311
|
+
try {
|
|
312
|
+
files = this.deps.readdir(this.habitsDir);
|
|
313
|
+
}
|
|
314
|
+
catch {
|
|
315
|
+
this.parseErrors = [];
|
|
316
|
+
return [];
|
|
317
|
+
}
|
|
318
|
+
const habits = [];
|
|
319
|
+
const errors = [];
|
|
320
|
+
for (const file of files) {
|
|
321
|
+
if (!file.endsWith(".md"))
|
|
322
|
+
continue;
|
|
323
|
+
const filePath = path.join(this.habitsDir, file);
|
|
324
|
+
try {
|
|
325
|
+
const content = this.deps.readFile(filePath, "utf-8");
|
|
326
|
+
const habit = (0, habit_parser_1.parseHabitFile)(content, filePath);
|
|
327
|
+
habits.push(habit);
|
|
328
|
+
}
|
|
329
|
+
catch (error) {
|
|
330
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
331
|
+
errors.push({ file, error: errorMessage });
|
|
332
|
+
(0, runtime_1.emitNervesEvent)({
|
|
333
|
+
level: "error",
|
|
334
|
+
component: "daemon",
|
|
335
|
+
event: "daemon.habit_parse_error",
|
|
336
|
+
message: "failed to parse habit file",
|
|
337
|
+
meta: {
|
|
338
|
+
file,
|
|
339
|
+
error: errorMessage,
|
|
340
|
+
agent: this.agent,
|
|
341
|
+
},
|
|
342
|
+
});
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
this.parseErrors = errors;
|
|
346
|
+
return habits;
|
|
347
|
+
}
|
|
348
|
+
buildJobs(habits) {
|
|
349
|
+
const jobs = [];
|
|
350
|
+
for (const habit of habits) {
|
|
351
|
+
if (habit.status !== "active")
|
|
352
|
+
continue;
|
|
353
|
+
if (!habit.cadence)
|
|
354
|
+
continue;
|
|
355
|
+
const cronSchedule = (0, cadence_1.parseCadenceToCron)(habit.cadence);
|
|
356
|
+
if (cronSchedule === null)
|
|
357
|
+
continue;
|
|
358
|
+
jobs.push({
|
|
359
|
+
id: `${this.agent}:${habit.name}:cadence`,
|
|
360
|
+
agent: this.agent,
|
|
361
|
+
taskId: habit.name,
|
|
362
|
+
schedule: cronSchedule,
|
|
363
|
+
lastRun: habit.lastRun,
|
|
364
|
+
command: `${this.deps.ouroPath} poke ${this.agent} --habit ${habit.name}`,
|
|
365
|
+
taskPath: path.join(this.habitsDir, `${habit.name}.md`),
|
|
366
|
+
});
|
|
367
|
+
}
|
|
368
|
+
return jobs;
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
exports.HabitScheduler = HabitScheduler;
|
|
@@ -39,16 +39,14 @@ const fs = __importStar(require("fs"));
|
|
|
39
39
|
const os = __importStar(require("os"));
|
|
40
40
|
const path = __importStar(require("path"));
|
|
41
41
|
const identity_1 = require("../identity");
|
|
42
|
+
const config_1 = require("../config");
|
|
42
43
|
const runtime_1 = require("../../nerves/runtime");
|
|
44
|
+
const auth_flow_1 = require("../auth/auth-flow");
|
|
45
|
+
const provider_models_1 = require("../provider-models");
|
|
46
|
+
const habit_parser_1 = require("../habits/habit-parser");
|
|
43
47
|
const hatch_specialist_1 = require("./hatch-specialist");
|
|
44
48
|
function requiredCredentialKeys(provider) {
|
|
45
|
-
|
|
46
|
-
return ["setupToken"];
|
|
47
|
-
if (provider === "openai-codex")
|
|
48
|
-
return ["oauthAccessToken"];
|
|
49
|
-
if (provider === "minimax")
|
|
50
|
-
return ["apiKey"];
|
|
51
|
-
return ["apiKey", "endpoint", "deployment"];
|
|
49
|
+
return identity_1.PROVIDER_CREDENTIALS[provider].required;
|
|
52
50
|
}
|
|
53
51
|
function validateCredentials(provider, credentials) {
|
|
54
52
|
const missing = requiredCredentialKeys(provider).filter((key) => {
|
|
@@ -66,70 +64,8 @@ function validateCredentials(provider, credentials) {
|
|
|
66
64
|
throw new Error(`Missing required credentials for ${provider}: ${missing.join(", ")}`);
|
|
67
65
|
}
|
|
68
66
|
}
|
|
69
|
-
function buildSecretsTemplate() {
|
|
70
|
-
return {
|
|
71
|
-
providers: {
|
|
72
|
-
azure: {
|
|
73
|
-
modelName: "gpt-4o-mini",
|
|
74
|
-
apiKey: "",
|
|
75
|
-
endpoint: "",
|
|
76
|
-
deployment: "",
|
|
77
|
-
apiVersion: "2025-04-01-preview",
|
|
78
|
-
},
|
|
79
|
-
minimax: {
|
|
80
|
-
model: "minimax-text-01",
|
|
81
|
-
apiKey: "",
|
|
82
|
-
},
|
|
83
|
-
anthropic: {
|
|
84
|
-
model: "claude-opus-4-6",
|
|
85
|
-
setupToken: "",
|
|
86
|
-
},
|
|
87
|
-
"openai-codex": {
|
|
88
|
-
model: "gpt-5.4",
|
|
89
|
-
oauthAccessToken: "",
|
|
90
|
-
},
|
|
91
|
-
},
|
|
92
|
-
teams: {
|
|
93
|
-
clientId: "",
|
|
94
|
-
clientSecret: "",
|
|
95
|
-
tenantId: "",
|
|
96
|
-
},
|
|
97
|
-
oauth: {
|
|
98
|
-
graphConnectionName: "graph",
|
|
99
|
-
adoConnectionName: "ado",
|
|
100
|
-
githubConnectionName: "",
|
|
101
|
-
},
|
|
102
|
-
teamsChannel: {
|
|
103
|
-
skipConfirmation: true,
|
|
104
|
-
port: 3978,
|
|
105
|
-
},
|
|
106
|
-
integrations: {
|
|
107
|
-
perplexityApiKey: "",
|
|
108
|
-
openaiEmbeddingsApiKey: "",
|
|
109
|
-
},
|
|
110
|
-
};
|
|
111
|
-
}
|
|
112
67
|
function writeSecretsFile(agentName, provider, credentials, secretsRoot) {
|
|
113
|
-
|
|
114
|
-
if (provider === "anthropic") {
|
|
115
|
-
secrets.providers.anthropic.setupToken = credentials.setupToken.trim();
|
|
116
|
-
}
|
|
117
|
-
else if (provider === "openai-codex") {
|
|
118
|
-
secrets.providers["openai-codex"].oauthAccessToken = credentials.oauthAccessToken.trim();
|
|
119
|
-
}
|
|
120
|
-
else if (provider === "minimax") {
|
|
121
|
-
secrets.providers.minimax.apiKey = credentials.apiKey.trim();
|
|
122
|
-
}
|
|
123
|
-
else {
|
|
124
|
-
secrets.providers.azure.apiKey = credentials.apiKey.trim();
|
|
125
|
-
secrets.providers.azure.endpoint = credentials.endpoint.trim();
|
|
126
|
-
secrets.providers.azure.deployment = credentials.deployment.trim();
|
|
127
|
-
}
|
|
128
|
-
const secretsDir = path.join(secretsRoot, agentName);
|
|
129
|
-
fs.mkdirSync(secretsDir, { recursive: true });
|
|
130
|
-
const secretsPath = path.join(secretsDir, "secrets.json");
|
|
131
|
-
fs.writeFileSync(secretsPath, `${JSON.stringify(secrets, null, 2)}\n`, "utf-8");
|
|
132
|
-
return secretsPath;
|
|
68
|
+
return (0, auth_flow_1.writeProviderCredentials)(agentName, provider, credentials, { secretsRoot }).secretsPath;
|
|
133
69
|
}
|
|
134
70
|
function writeReadme(dir, purpose) {
|
|
135
71
|
fs.mkdirSync(dir, { recursive: true });
|
|
@@ -138,52 +74,25 @@ function writeReadme(dir, purpose) {
|
|
|
138
74
|
fs.writeFileSync(readmePath, `# ${path.basename(dir)}\n\n${purpose}\n`, "utf-8");
|
|
139
75
|
}
|
|
140
76
|
}
|
|
141
|
-
function
|
|
142
|
-
const
|
|
143
|
-
const slug = trimmed
|
|
144
|
-
.replace(/[^a-z0-9]+/g, "-")
|
|
145
|
-
.replace(/^-+/, "")
|
|
146
|
-
.replace(/-+$/, "");
|
|
147
|
-
return slug || "friend";
|
|
148
|
-
}
|
|
149
|
-
function pad(value) {
|
|
150
|
-
return String(value).padStart(2, "0");
|
|
151
|
-
}
|
|
152
|
-
function formatTaskStem(now) {
|
|
153
|
-
return `${now.getFullYear()}-${pad(now.getMonth() + 1)}-${pad(now.getDate())}-${pad(now.getHours())}${pad(now.getMinutes())}`;
|
|
154
|
-
}
|
|
155
|
-
function writeHeartbeatTask(bundleRoot, now) {
|
|
156
|
-
const habitsDir = path.join(bundleRoot, "tasks", "habits");
|
|
77
|
+
function writeHeartbeatHabit(bundleRoot, now) {
|
|
78
|
+
const habitsDir = path.join(bundleRoot, "habits");
|
|
157
79
|
fs.mkdirSync(habitsDir, { recursive: true });
|
|
158
|
-
const
|
|
159
|
-
const
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
"
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
"status: processing",
|
|
167
|
-
`created: ${iso}`,
|
|
168
|
-
`updated: ${iso}`,
|
|
169
|
-
"requester: system",
|
|
170
|
-
"validator: null",
|
|
171
|
-
"cadence: \"30m\"",
|
|
172
|
-
"scheduledAt: null",
|
|
173
|
-
"lastRun: null",
|
|
174
|
-
"---",
|
|
175
|
-
"",
|
|
176
|
-
"Run a lightweight heartbeat cycle. Review task board and inbox.",
|
|
177
|
-
"",
|
|
178
|
-
].join("\n");
|
|
80
|
+
const filePath = path.join(habitsDir, "heartbeat.md");
|
|
81
|
+
const content = (0, habit_parser_1.renderHabitFile)({
|
|
82
|
+
title: "Heartbeat check-in",
|
|
83
|
+
cadence: "30m",
|
|
84
|
+
status: "active",
|
|
85
|
+
lastRun: "null",
|
|
86
|
+
created: now.toISOString(),
|
|
87
|
+
}, "Run a lightweight heartbeat cycle. Review task board and inbox.\nCheck on pending obligations. Journal anything important.");
|
|
179
88
|
fs.writeFileSync(filePath, content, "utf-8");
|
|
180
89
|
}
|
|
181
90
|
function writeFriendImprint(bundleRoot, humanName, now) {
|
|
182
91
|
const friendsDir = path.join(bundleRoot, "friends");
|
|
183
92
|
fs.mkdirSync(friendsDir, { recursive: true });
|
|
184
93
|
const nowIso = now.toISOString();
|
|
185
|
-
const id = `friend-${slugify(humanName)}`;
|
|
186
|
-
const localExternalId =
|
|
94
|
+
const id = `friend-${(0, config_1.slugify)(humanName) || "friend"}`;
|
|
95
|
+
const localExternalId = os.userInfo().username;
|
|
187
96
|
const record = {
|
|
188
97
|
id,
|
|
189
98
|
name: humanName,
|
|
@@ -207,16 +116,19 @@ function writeFriendImprint(bundleRoot, humanName, now) {
|
|
|
207
116
|
};
|
|
208
117
|
fs.writeFileSync(path.join(friendsDir, `${id}.json`), `${JSON.stringify(record, null, 2)}\n`, "utf-8");
|
|
209
118
|
}
|
|
210
|
-
function
|
|
211
|
-
const
|
|
212
|
-
fs.mkdirSync(path.join(
|
|
213
|
-
fs.mkdirSync(path.join(
|
|
214
|
-
fs.writeFileSync(path.join(
|
|
215
|
-
fs.writeFileSync(path.join(
|
|
119
|
+
function writeDiaryScaffold(bundleRoot) {
|
|
120
|
+
const diaryRoot = path.join(bundleRoot, "diary");
|
|
121
|
+
fs.mkdirSync(path.join(diaryRoot, "daily"), { recursive: true });
|
|
122
|
+
fs.mkdirSync(path.join(diaryRoot, "archive"), { recursive: true });
|
|
123
|
+
fs.writeFileSync(path.join(diaryRoot, "facts.jsonl"), "", "utf-8");
|
|
124
|
+
fs.writeFileSync(path.join(diaryRoot, "entities.json"), "{}\n", "utf-8");
|
|
216
125
|
}
|
|
217
126
|
function writeHatchlingAgentConfig(bundleRoot, input) {
|
|
218
127
|
const template = (0, identity_1.buildDefaultAgentTemplate)(input.agentName);
|
|
128
|
+
const model = (0, provider_models_1.getDefaultModelForProvider)(input.provider);
|
|
219
129
|
template.provider = input.provider;
|
|
130
|
+
template.humanFacing = { provider: input.provider, model };
|
|
131
|
+
template.agentFacing = { provider: input.provider, model };
|
|
220
132
|
template.enabled = true;
|
|
221
133
|
fs.writeFileSync(path.join(bundleRoot, "agent.json"), `${JSON.stringify(template, null, 2)}\n`, "utf-8");
|
|
222
134
|
}
|
|
@@ -242,25 +154,25 @@ async function runHatchFlow(input, deps = {}) {
|
|
|
242
154
|
identitiesDir: targetIdentities,
|
|
243
155
|
random,
|
|
244
156
|
});
|
|
245
|
-
const specialistSecretsPath = writeSecretsFile("
|
|
157
|
+
const specialistSecretsPath = writeSecretsFile("SerpentGuide", input.provider, input.credentials, secretsRoot);
|
|
246
158
|
const hatchlingSecretsPath = writeSecretsFile(input.agentName, input.provider, input.credentials, secretsRoot);
|
|
247
159
|
const bundleRoot = path.join(bundlesRoot, `${input.agentName}.ouro`);
|
|
248
160
|
fs.mkdirSync(bundleRoot, { recursive: true });
|
|
249
161
|
writeReadme(bundleRoot, "Root of this agent bundle.");
|
|
250
162
|
writeReadme(path.join(bundleRoot, "psyche"), "Identity and behavior files.");
|
|
251
|
-
writeReadme(path.join(bundleRoot, "
|
|
163
|
+
writeReadme(path.join(bundleRoot, "diary"), "Persistent diary — things I've learned and remember.");
|
|
252
164
|
writeReadme(path.join(bundleRoot, "friends"), "Known friend records.");
|
|
253
165
|
writeReadme(path.join(bundleRoot, "tasks"), "Task files.");
|
|
254
|
-
writeReadme(path.join(bundleRoot, "tasks", "habits"), "Recurring tasks.");
|
|
255
166
|
writeReadme(path.join(bundleRoot, "tasks", "one-shots"), "One-shot tasks.");
|
|
256
167
|
writeReadme(path.join(bundleRoot, "tasks", "ongoing"), "Ongoing tasks.");
|
|
168
|
+
writeReadme(path.join(bundleRoot, "habits"), "Recurring habits and autonomous rhythms.");
|
|
257
169
|
writeReadme(path.join(bundleRoot, "skills"), "Local skill files.");
|
|
258
170
|
writeReadme(path.join(bundleRoot, "senses"), "Sense-specific config.");
|
|
259
171
|
writeReadme(path.join(bundleRoot, "senses", "teams"), "Teams sense config.");
|
|
260
172
|
writeHatchlingAgentConfig(bundleRoot, input);
|
|
261
|
-
|
|
173
|
+
writeDiaryScaffold(bundleRoot);
|
|
262
174
|
writeFriendImprint(bundleRoot, input.humanName, now);
|
|
263
|
-
|
|
175
|
+
writeHeartbeatHabit(bundleRoot, now);
|
|
264
176
|
(0, runtime_1.emitNervesEvent)({
|
|
265
177
|
component: "daemon",
|
|
266
178
|
event: "daemon.hatch_flow_end",
|
|
@@ -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;
|
|
@@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
exports.buildSpecialistSystemPrompt = buildSpecialistSystemPrompt;
|
|
4
4
|
const runtime_1 = require("../../nerves/runtime");
|
|
5
5
|
/**
|
|
6
|
-
* Build the
|
|
6
|
+
* Build the serpent guide's system prompt from its components.
|
|
7
7
|
* The prompt is written in first person (the specialist's own voice).
|
|
8
8
|
*/
|
|
9
9
|
function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, context) {
|
|
@@ -31,7 +31,7 @@ function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, co
|
|
|
31
31
|
"I am one of thirteen serpent guides who help humans hatch their first agent. The system randomly selected me for this session.",
|
|
32
32
|
"Most humans only go through this process once, so this is likely the only time they'll meet me.",
|
|
33
33
|
"I make this encounter count — warm, memorable, and uniquely mine.",
|
|
34
|
-
"IMPORTANT: I NEVER refer to myself
|
|
34
|
+
"IMPORTANT: I NEVER refer to myself by internal implementation labels — I introduce myself by my own name from my identity.",
|
|
35
35
|
"",
|
|
36
36
|
"## Voice rules",
|
|
37
37
|
"IMPORTANT: I keep every response to 1-3 short sentences. I sound like a friend texting, not a manual.",
|
|
@@ -59,8 +59,10 @@ function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, co
|
|
|
59
59
|
"It also needs an **agent.json** with at minimum:",
|
|
60
60
|
'```json',
|
|
61
61
|
'{',
|
|
62
|
+
' "version": 2,',
|
|
62
63
|
' "name": "AgentName",',
|
|
63
|
-
` "provider": "${context.provider}",`,
|
|
64
|
+
` "humanFacing": { "provider": "${context.provider}", "model": "${context.model}" },`,
|
|
65
|
+
` "agentFacing": { "provider": "${context.provider}", "model": "${context.model}" },`,
|
|
64
66
|
' "enabled": true',
|
|
65
67
|
'}',
|
|
66
68
|
'```',
|
|
@@ -76,23 +78,24 @@ function buildSpecialistSystemPrompt(soulText, identityText, existingBundles, co
|
|
|
76
78
|
"Then I ask what they'd like their agent to help with — one question at a time.",
|
|
77
79
|
"I'm proactive: I suggest ideas and guide them. If they seem unsure, I offer a concrete suggestion.",
|
|
78
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.",
|
|
79
82
|
"When I have enough context about the agent's personality and purpose:",
|
|
80
83
|
"1. I write all 5 psyche files to the temp directory using write_file",
|
|
81
84
|
"2. I write agent.json to the temp directory using write_file",
|
|
82
85
|
"3. I suggest a PascalCase name for the hatchling and confirm with the human",
|
|
83
86
|
"4. I call complete_adoption with the name and a warm handoff message",
|
|
84
|
-
"5. I call
|
|
87
|
+
"5. I call settle to end the session",
|
|
85
88
|
].join("\n"));
|
|
86
89
|
sections.push([
|
|
87
90
|
"## Tools",
|
|
88
91
|
"- `write_file`: Write a file to disk. Use this to write psyche files and agent.json to the temp directory.",
|
|
89
92
|
"- `read_file`: Read a file from disk. Useful for reviewing existing agent bundles or migration sources.",
|
|
90
93
|
"- `list_directory`: List directory contents. Useful for exploring existing agent bundles.",
|
|
91
|
-
"- I also have the normal local harness tools when useful here, including `shell`, task
|
|
94
|
+
"- I also have the normal local harness tools when useful here, including `shell`, `ouro task create`, `ouro reminder create`, memory tools, coding tools, and repo helpers.",
|
|
92
95
|
"- `complete_adoption`: Finalize the bundle. Validates, scaffolds structural dirs, moves to ~/AgentBundles/, writes secrets, plays hatch animation. I call this with `name` (PascalCase) and `handoff_message` (warm message for the human).",
|
|
93
|
-
"- `
|
|
96
|
+
"- `settle`: End the conversation with a final message. I call this after complete_adoption succeeds.",
|
|
94
97
|
"",
|
|
95
|
-
"I must call `
|
|
98
|
+
"I must call `settle` when I am done to end the session cleanly.",
|
|
96
99
|
].join("\n"));
|
|
97
100
|
return sections.join("\n\n");
|
|
98
101
|
}
|