@ouro.bot/cli 0.1.0-alpha.45 → 0.1.0-alpha.450
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +127 -19
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/changelog.json +2853 -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 +426 -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 +110 -128
- package/dist/heart/core.js +745 -227
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +490 -0
- package/dist/heart/daemon/agent-discovery.js +79 -3
- package/dist/heart/daemon/agent-service.js +360 -0
- package/dist/heart/daemon/agentic-repair.js +216 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +631 -0
- package/dist/heart/daemon/cli-exec.js +5805 -0
- package/dist/heart/daemon/cli-help.js +428 -0
- package/dist/heart/daemon/cli-parse.js +1156 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +561 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/connect-bay.js +323 -0
- package/dist/heart/daemon/daemon-cli.js +29 -1617
- package/dist/heart/daemon/daemon-entry.js +356 -3
- package/dist/heart/daemon/daemon-health.js +141 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +190 -12
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +684 -58
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +427 -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/human-command-screens.js +234 -0
- package/dist/heart/daemon/human-readiness.js +114 -0
- package/dist/heart/daemon/inner-status.js +89 -0
- package/dist/heart/daemon/interactive-repair.js +394 -0
- package/dist/heart/daemon/launchd.js +25 -5
- 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 +2 -2
- 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 +214 -0
- package/dist/heart/daemon/provider-discovery.js +137 -0
- package/dist/heart/daemon/provider-ping-progress.js +83 -0
- package/dist/heart/daemon/pulse.js +475 -0
- package/dist/heart/daemon/readiness-repair.js +365 -0
- package/dist/heart/daemon/run-hooks.js +2 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +73 -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 +123 -34
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +307 -0
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +264 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +510 -0
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/delegation.js +62 -0
- package/dist/heart/habits/habit-migration.js +189 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-runtime-state.js +100 -0
- package/dist/heart/habits/habit-scheduler.js +372 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
- package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
- package/dist/heart/identity.js +197 -65
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +1 -1
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/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 +644 -0
- package/dist/heart/outlook/readers/sessions.js +232 -0
- package/dist/heart/outlook/readers/shared.js +111 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +255 -0
- package/dist/heart/provider-credentials.js +424 -0
- package/dist/heart/provider-failover.js +266 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +262 -0
- package/dist/heart/provider-state.js +216 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +193 -55
- 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 +62 -38
- package/dist/heart/runtime-capability-check.js +170 -0
- package/dist/heart/runtime-credentials.js +260 -0
- package/dist/heart/sense-truth.js +3 -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 +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 +351 -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/versioning/ouro-path-installer.js +425 -0
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +5 -1
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mind/bundle-manifest.js +7 -1
- package/dist/mind/context.js +132 -93
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +74 -93
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/friends/channel.js +21 -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 +39 -3
- package/dist/mind/friends/trust-explanation.js +74 -0
- package/dist/mind/friends/types.js +1 -1
- 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 +66 -7
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +947 -165
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +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 +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +774 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +180 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +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 +111 -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 +37 -4
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -78
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-base.js +42 -690
- 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 +361 -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 +9 -39
- 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 +144 -113
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +131 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +561 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
- package/dist/senses/bluebubbles/entry.js +73 -0
- package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +7 -3
- package/dist/senses/bluebubbles/index.js +1620 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
- package/dist/senses/bluebubbles/replay.js +129 -0
- package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +2 -2
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +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 +516 -211
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +102 -19
- package/dist/senses/inner-dialog.js +597 -95
- package/dist/senses/pipeline.js +533 -72
- 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 +413 -163
- package/dist/senses/trust-gate.js +5 -5
- package/package.json +32 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +101 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/ouro-path-installer.js +0 -178
- package/dist/heart/daemon/subagent-installer.js +0 -166
- package/dist/mind/associative-recall.js +0 -209
- package/dist/senses/bluebubbles-entry.js +0 -13
- package/dist/senses/bluebubbles.js +0 -1028
- package/dist/senses/debug-activity.js +0 -127
- package/subagents/README.md +0 -86
- package/subagents/work-doer.md +0 -237
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -390
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
- /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
|
@@ -0,0 +1,161 @@
|
|
|
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.SAFE_MODE_OVERRIDE_FILENAME = void 0;
|
|
37
|
+
exports.detectSafeMode = detectSafeMode;
|
|
38
|
+
exports.pruneOldCrashes = pruneOldCrashes;
|
|
39
|
+
const fs = __importStar(require("fs"));
|
|
40
|
+
const path = __importStar(require("path"));
|
|
41
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
42
|
+
exports.SAFE_MODE_OVERRIDE_FILENAME = "safe-mode-override.json";
|
|
43
|
+
/** 3+ crashes within this window triggers safe mode */
|
|
44
|
+
const CRASH_WINDOW_MS = 5 * 60 * 1000; // 5 minutes
|
|
45
|
+
const CRASH_THRESHOLD = 3;
|
|
46
|
+
/**
|
|
47
|
+
* Reads crash history from the tombstone file and determines if safe mode should be active.
|
|
48
|
+
* Returns active if 3+ crashes occurred within 5 minutes.
|
|
49
|
+
*
|
|
50
|
+
* Safe mode is bypassed when:
|
|
51
|
+
* - devMode is true
|
|
52
|
+
* - A safe-mode override file exists (written by `ouro up --force`)
|
|
53
|
+
*/
|
|
54
|
+
function detectSafeMode(tombstonePath, options) {
|
|
55
|
+
const inactive = { active: false, reason: "", enteredAt: "" };
|
|
56
|
+
// Dev mode: never enter safe mode
|
|
57
|
+
if (options?.devMode) {
|
|
58
|
+
return inactive;
|
|
59
|
+
}
|
|
60
|
+
// Check for override file (--force)
|
|
61
|
+
const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
|
|
62
|
+
try {
|
|
63
|
+
if (fs.existsSync(overridePath)) {
|
|
64
|
+
return inactive;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
catch {
|
|
68
|
+
// Best-effort check
|
|
69
|
+
}
|
|
70
|
+
// Read tombstone
|
|
71
|
+
let parsed;
|
|
72
|
+
try {
|
|
73
|
+
const raw = fs.readFileSync(tombstonePath, "utf-8");
|
|
74
|
+
parsed = JSON.parse(raw);
|
|
75
|
+
}
|
|
76
|
+
catch {
|
|
77
|
+
return inactive;
|
|
78
|
+
}
|
|
79
|
+
// Extract recentCrashes
|
|
80
|
+
if (!Array.isArray(parsed.recentCrashes)) {
|
|
81
|
+
return inactive;
|
|
82
|
+
}
|
|
83
|
+
const nowMs = options?.now ? options.now() : Date.now();
|
|
84
|
+
const windowStart = nowMs - CRASH_WINDOW_MS;
|
|
85
|
+
// Filter to valid string timestamps within the crash window
|
|
86
|
+
const recentInWindow = parsed.recentCrashes.filter((entry) => {
|
|
87
|
+
if (typeof entry !== "string")
|
|
88
|
+
return false;
|
|
89
|
+
const ts = new Date(entry).getTime();
|
|
90
|
+
if (isNaN(ts))
|
|
91
|
+
return false;
|
|
92
|
+
return ts >= windowStart;
|
|
93
|
+
});
|
|
94
|
+
if (recentInWindow.length < CRASH_THRESHOLD) {
|
|
95
|
+
return inactive;
|
|
96
|
+
}
|
|
97
|
+
const result = {
|
|
98
|
+
active: true,
|
|
99
|
+
reason: `crash loop detected: ${recentInWindow.length} crashes in last 5 minutes`,
|
|
100
|
+
enteredAt: new Date(nowMs).toISOString(),
|
|
101
|
+
};
|
|
102
|
+
(0, runtime_1.emitNervesEvent)({
|
|
103
|
+
level: "error",
|
|
104
|
+
component: "daemon",
|
|
105
|
+
event: "daemon.safe_mode_entered",
|
|
106
|
+
message: result.reason,
|
|
107
|
+
meta: {
|
|
108
|
+
crashCount: recentInWindow.length,
|
|
109
|
+
windowMs: CRASH_WINDOW_MS,
|
|
110
|
+
tombstonePath,
|
|
111
|
+
},
|
|
112
|
+
});
|
|
113
|
+
return result;
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Prunes crash entries older than 5 minutes from the tombstone's recentCrashes.
|
|
117
|
+
* Also removes the safe-mode override file if present.
|
|
118
|
+
* Called after successful startup (uptime > stability threshold).
|
|
119
|
+
*/
|
|
120
|
+
function pruneOldCrashes(tombstonePath, options) {
|
|
121
|
+
// Remove override file
|
|
122
|
+
const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
|
|
123
|
+
try {
|
|
124
|
+
if (fs.existsSync(overridePath)) {
|
|
125
|
+
fs.unlinkSync(overridePath);
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
catch {
|
|
129
|
+
// Best-effort
|
|
130
|
+
}
|
|
131
|
+
// Read existing tombstone
|
|
132
|
+
let parsed;
|
|
133
|
+
try {
|
|
134
|
+
const raw = fs.readFileSync(tombstonePath, "utf-8");
|
|
135
|
+
parsed = JSON.parse(raw);
|
|
136
|
+
}
|
|
137
|
+
catch {
|
|
138
|
+
return;
|
|
139
|
+
}
|
|
140
|
+
if (!Array.isArray(parsed.recentCrashes)) {
|
|
141
|
+
return;
|
|
142
|
+
}
|
|
143
|
+
const nowMs = options?.now ? options.now() : Date.now();
|
|
144
|
+
const windowStart = nowMs - CRASH_WINDOW_MS;
|
|
145
|
+
// Keep only entries within the window
|
|
146
|
+
const pruned = parsed.recentCrashes.filter((entry) => {
|
|
147
|
+
if (typeof entry !== "string")
|
|
148
|
+
return false;
|
|
149
|
+
const ts = new Date(entry).getTime();
|
|
150
|
+
if (isNaN(ts))
|
|
151
|
+
return false;
|
|
152
|
+
return ts >= windowStart;
|
|
153
|
+
});
|
|
154
|
+
parsed.recentCrashes = pruned;
|
|
155
|
+
try {
|
|
156
|
+
fs.writeFileSync(tombstonePath, JSON.stringify(parsed, null, 2) + "\n", "utf-8");
|
|
157
|
+
}
|
|
158
|
+
catch {
|
|
159
|
+
// Best-effort
|
|
160
|
+
}
|
|
161
|
+
}
|
|
@@ -38,13 +38,15 @@ const fs = __importStar(require("fs"));
|
|
|
38
38
|
const os = __importStar(require("os"));
|
|
39
39
|
const path = __importStar(require("path"));
|
|
40
40
|
const runtime_1 = require("../../nerves/runtime");
|
|
41
|
-
const bluebubbles_runtime_state_1 = require("../../senses/bluebubbles-runtime-state");
|
|
42
41
|
const identity_1 = require("../identity");
|
|
42
|
+
const runtime_credentials_1 = require("../runtime-credentials");
|
|
43
43
|
const sense_truth_1 = require("../sense-truth");
|
|
44
|
+
const machine_identity_1 = require("../machine-identity");
|
|
44
45
|
const process_manager_1 = require("./process-manager");
|
|
45
46
|
const DEFAULT_TEAMS_PORT = 3978;
|
|
46
47
|
const DEFAULT_BLUEBUBBLES_PORT = 18790;
|
|
47
48
|
const DEFAULT_BLUEBUBBLES_WEBHOOK_PATH = "/bluebubbles-webhook";
|
|
49
|
+
const BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS = 90_000;
|
|
48
50
|
function defaultSenses() {
|
|
49
51
|
return {
|
|
50
52
|
cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
|
|
@@ -87,22 +89,6 @@ function readAgentSenses(agentJsonPath) {
|
|
|
87
89
|
}
|
|
88
90
|
return defaults;
|
|
89
91
|
}
|
|
90
|
-
function readSecretsPayload(secretsPath) {
|
|
91
|
-
try {
|
|
92
|
-
const raw = fs.readFileSync(secretsPath, "utf-8");
|
|
93
|
-
const parsed = JSON.parse(raw);
|
|
94
|
-
if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
|
|
95
|
-
return { payload: {}, error: "invalid secrets.json object" };
|
|
96
|
-
}
|
|
97
|
-
return { payload: parsed, error: null };
|
|
98
|
-
}
|
|
99
|
-
catch (error) {
|
|
100
|
-
return {
|
|
101
|
-
payload: {},
|
|
102
|
-
error: error instanceof Error ? error.message : String(error),
|
|
103
|
-
};
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
92
|
function textField(record, key) {
|
|
107
93
|
const value = record?.[key];
|
|
108
94
|
return typeof value === "string" ? value.trim() : "";
|
|
@@ -111,17 +97,34 @@ function numberField(record, key, fallback) {
|
|
|
111
97
|
const value = record?.[key];
|
|
112
98
|
return typeof value === "number" && Number.isFinite(value) ? value : fallback;
|
|
113
99
|
}
|
|
114
|
-
function
|
|
100
|
+
function compactRuntimeConfigError(agent, error) {
|
|
101
|
+
const compact = error.replace(/\s+/g, " ").trim();
|
|
102
|
+
if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
|
|
103
|
+
return `vault locked; run 'ouro vault unlock --agent ${agent}' if you have the saved secret, or 'ouro vault replace --agent ${agent}' if none was saved`;
|
|
104
|
+
}
|
|
105
|
+
return compact || "unavailable";
|
|
106
|
+
}
|
|
107
|
+
function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
|
|
108
|
+
if (runtimeConfig.ok)
|
|
109
|
+
return "";
|
|
110
|
+
const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
|
|
111
|
+
if (runtimeConfig.reason === "missing")
|
|
112
|
+
return `missing vault ${itemName} (${agent})`;
|
|
113
|
+
return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
|
|
114
|
+
}
|
|
115
|
+
function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
|
|
115
116
|
const base = {
|
|
116
117
|
cli: { configured: true, detail: "local interactive terminal" },
|
|
117
118
|
teams: { configured: false, detail: "not enabled in agent.json" },
|
|
118
119
|
bluebubbles: { configured: false, detail: "not enabled in agent.json" },
|
|
119
120
|
};
|
|
120
|
-
const
|
|
121
|
+
const payload = runtimeConfig.ok ? runtimeConfig.config : {};
|
|
122
|
+
const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
|
|
121
123
|
const teams = payload.teams;
|
|
122
124
|
const teamsChannel = payload.teamsChannel;
|
|
123
|
-
const
|
|
124
|
-
const
|
|
125
|
+
const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
|
|
126
|
+
const bluebubbles = machinePayload.bluebubbles;
|
|
127
|
+
const bluebubblesChannel = machinePayload.bluebubblesChannel;
|
|
125
128
|
if (senses.teams.enabled) {
|
|
126
129
|
const missing = [];
|
|
127
130
|
if (!textField(teams, "clientId"))
|
|
@@ -137,9 +140,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
137
140
|
}
|
|
138
141
|
: {
|
|
139
142
|
configured: false,
|
|
140
|
-
detail:
|
|
141
|
-
? `missing
|
|
142
|
-
:
|
|
143
|
+
detail: runtimeConfig.ok
|
|
144
|
+
? `missing ${missing.join("/")}`
|
|
145
|
+
: unavailableDetail,
|
|
143
146
|
};
|
|
144
147
|
}
|
|
145
148
|
if (senses.bluebubbles.enabled) {
|
|
@@ -155,13 +158,25 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
|
155
158
|
}
|
|
156
159
|
: {
|
|
157
160
|
configured: false,
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
+
optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
|
|
162
|
+
detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
|
|
163
|
+
? "not attached on this machine"
|
|
164
|
+
: machineRuntimeConfig.ok
|
|
165
|
+
? `missing ${missing.join("/")}`
|
|
166
|
+
: runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
|
|
161
167
|
};
|
|
162
168
|
}
|
|
163
169
|
return base;
|
|
164
170
|
}
|
|
171
|
+
function senseRepairHint(agent, sense) {
|
|
172
|
+
if (sense === "teams") {
|
|
173
|
+
return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
|
|
174
|
+
}
|
|
175
|
+
return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
|
|
176
|
+
}
|
|
177
|
+
function currentMachineId() {
|
|
178
|
+
return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
|
|
179
|
+
}
|
|
165
180
|
function parseSenseSnapshotName(name) {
|
|
166
181
|
const parts = name.split(":");
|
|
167
182
|
if (parts.length !== 2)
|
|
@@ -176,20 +191,58 @@ function runtimeInfoFor(status) {
|
|
|
176
191
|
return { runtime: "running" };
|
|
177
192
|
return { runtime: "error" };
|
|
178
193
|
}
|
|
194
|
+
function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
|
|
195
|
+
if (!lastCheckedAt) {
|
|
196
|
+
return false;
|
|
197
|
+
}
|
|
198
|
+
const checkedAt = Date.parse(lastCheckedAt);
|
|
199
|
+
if (!Number.isFinite(checkedAt)) {
|
|
200
|
+
return false;
|
|
201
|
+
}
|
|
202
|
+
return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
|
|
203
|
+
}
|
|
204
|
+
function readBlueBubblesRuntimeJson(runtimePath) {
|
|
205
|
+
try {
|
|
206
|
+
const raw = fs.readFileSync(runtimePath, "utf-8");
|
|
207
|
+
const parsed = JSON.parse(raw);
|
|
208
|
+
/* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
|
|
209
|
+
return {
|
|
210
|
+
upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
|
|
211
|
+
? parsed.upstreamStatus
|
|
212
|
+
: "unknown",
|
|
213
|
+
detail: typeof parsed.detail === "string" && parsed.detail.trim()
|
|
214
|
+
? parsed.detail
|
|
215
|
+
: "startup health probe pending",
|
|
216
|
+
lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
|
|
217
|
+
};
|
|
218
|
+
/* v8 ignore stop */
|
|
219
|
+
/* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
|
|
220
|
+
}
|
|
221
|
+
catch {
|
|
222
|
+
return { upstreamStatus: "unknown", detail: "startup health probe pending" };
|
|
223
|
+
}
|
|
224
|
+
/* v8 ignore stop */
|
|
225
|
+
}
|
|
179
226
|
function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
|
|
180
227
|
const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
181
228
|
const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
|
|
182
|
-
if (
|
|
229
|
+
if (!fs.existsSync(runtimePath)) {
|
|
230
|
+
return { runtime: snapshot?.runtime };
|
|
231
|
+
}
|
|
232
|
+
const state = readBlueBubblesRuntimeJson(runtimePath);
|
|
233
|
+
if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
|
|
183
234
|
return { runtime: snapshot?.runtime };
|
|
184
235
|
}
|
|
185
|
-
const state = (0, bluebubbles_runtime_state_1.readBlueBubblesRuntimeState)(agent, agentRoot);
|
|
186
236
|
if (state.upstreamStatus === "error") {
|
|
187
237
|
return {
|
|
188
238
|
runtime: "error",
|
|
189
239
|
detail: state.detail,
|
|
190
240
|
};
|
|
191
241
|
}
|
|
192
|
-
|
|
242
|
+
if (state.upstreamStatus === "ok") {
|
|
243
|
+
return { runtime: "running" };
|
|
244
|
+
}
|
|
245
|
+
return { runtime: snapshot?.runtime };
|
|
193
246
|
}
|
|
194
247
|
class DaemonSenseManager {
|
|
195
248
|
processManager;
|
|
@@ -197,26 +250,53 @@ class DaemonSenseManager {
|
|
|
197
250
|
bundlesRoot;
|
|
198
251
|
constructor(options) {
|
|
199
252
|
const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
200
|
-
const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
|
|
201
253
|
this.bundlesRoot = bundlesRoot;
|
|
202
254
|
this.contexts = new Map(options.agents.map((agent) => {
|
|
203
255
|
const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
|
|
204
|
-
const facts =
|
|
256
|
+
const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
205
257
|
return [agent, { senses, facts }];
|
|
206
258
|
}));
|
|
207
259
|
const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
208
260
|
return ["teams", "bluebubbles"]
|
|
209
|
-
.filter((sense) => context.senses[sense].enabled
|
|
261
|
+
.filter((sense) => context.senses[sense].enabled)
|
|
210
262
|
.map((sense) => ({
|
|
211
263
|
name: `${agent}:${sense}`,
|
|
212
264
|
agentArg: agent,
|
|
213
|
-
entry: sense === "teams" ? "senses/teams-entry.js" : "senses/bluebubbles
|
|
265
|
+
entry: sense === "teams" ? "senses/teams-entry.js" : "senses/bluebubbles/entry.js",
|
|
214
266
|
channel: sense,
|
|
215
267
|
autoStart: true,
|
|
216
268
|
}));
|
|
217
269
|
});
|
|
218
270
|
this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
|
|
219
271
|
agents: managedSenseAgents,
|
|
272
|
+
configCheck: async (name) => {
|
|
273
|
+
const parsed = parseSenseSnapshotName(name);
|
|
274
|
+
if (!parsed)
|
|
275
|
+
return { ok: true };
|
|
276
|
+
const context = this.contexts.get(parsed.agent);
|
|
277
|
+
if (!context)
|
|
278
|
+
return { ok: true };
|
|
279
|
+
const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
|
|
280
|
+
const machineRefreshed = parsed.sense === "bluebubbles"
|
|
281
|
+
? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
|
|
282
|
+
: (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
|
|
283
|
+
context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
|
|
284
|
+
const fact = context.facts[parsed.sense];
|
|
285
|
+
if (fact.configured)
|
|
286
|
+
return { ok: true };
|
|
287
|
+
if (fact.optional) {
|
|
288
|
+
return {
|
|
289
|
+
ok: false,
|
|
290
|
+
skip: true,
|
|
291
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
|
|
292
|
+
};
|
|
293
|
+
}
|
|
294
|
+
return {
|
|
295
|
+
ok: false,
|
|
296
|
+
error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
|
|
297
|
+
fix: senseRepairHint(parsed.agent, parsed.sense),
|
|
298
|
+
};
|
|
299
|
+
},
|
|
220
300
|
});
|
|
221
301
|
(0, runtime_1.emitNervesEvent)({
|
|
222
302
|
component: "channels",
|
|
@@ -234,6 +314,13 @@ class DaemonSenseManager {
|
|
|
234
314
|
async stopAll() {
|
|
235
315
|
await this.processManager.stopAll();
|
|
236
316
|
}
|
|
317
|
+
/* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
|
|
318
|
+
listManagedPids() {
|
|
319
|
+
return this.processManager.listAgentSnapshots()
|
|
320
|
+
.map((s) => s.pid)
|
|
321
|
+
.filter((pid) => pid !== null && pid !== undefined);
|
|
322
|
+
}
|
|
323
|
+
/* v8 ignore stop */
|
|
237
324
|
listSenseRows() {
|
|
238
325
|
const runtime = new Map();
|
|
239
326
|
for (const snapshot of this.processManager.listAgentSnapshots()) {
|
|
@@ -245,6 +332,7 @@ class DaemonSenseManager {
|
|
|
245
332
|
runtime.set(parsed.agent, current);
|
|
246
333
|
}
|
|
247
334
|
const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
335
|
+
context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
|
|
248
336
|
const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
|
|
249
337
|
const runtimeInfo = {
|
|
250
338
|
cli: { configured: true },
|
|
@@ -254,6 +342,7 @@ class DaemonSenseManager {
|
|
|
254
342
|
},
|
|
255
343
|
bluebubbles: {
|
|
256
344
|
configured: context.facts.bluebubbles.configured,
|
|
345
|
+
optional: context.facts.bluebubbles.optional,
|
|
257
346
|
...blueBubblesRuntimeFacts,
|
|
258
347
|
},
|
|
259
348
|
};
|
|
@@ -0,0 +1,131 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Pluggable session ID resolver for MCP conversations.
|
|
3
|
+
// Tries tool-specific methods first, falls back to UUID.
|
|
4
|
+
//
|
|
5
|
+
// Claude Code: walks parent PID chain -> reads ~/.claude/sessions/{pid}.json
|
|
6
|
+
// Codex: reads thread_id from env (future)
|
|
7
|
+
// Fallback: generates UUID
|
|
8
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
9
|
+
if (k2 === undefined) k2 = k;
|
|
10
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
11
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
12
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
13
|
+
}
|
|
14
|
+
Object.defineProperty(o, k2, desc);
|
|
15
|
+
}) : (function(o, m, k, k2) {
|
|
16
|
+
if (k2 === undefined) k2 = k;
|
|
17
|
+
o[k2] = m[k];
|
|
18
|
+
}));
|
|
19
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
20
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
21
|
+
}) : function(o, v) {
|
|
22
|
+
o["default"] = v;
|
|
23
|
+
});
|
|
24
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
25
|
+
var ownKeys = function(o) {
|
|
26
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
27
|
+
var ar = [];
|
|
28
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
29
|
+
return ar;
|
|
30
|
+
};
|
|
31
|
+
return ownKeys(o);
|
|
32
|
+
};
|
|
33
|
+
return function (mod) {
|
|
34
|
+
if (mod && mod.__esModule) return mod;
|
|
35
|
+
var result = {};
|
|
36
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
37
|
+
__setModuleDefault(result, mod);
|
|
38
|
+
return result;
|
|
39
|
+
};
|
|
40
|
+
})();
|
|
41
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
42
|
+
exports.resolveSessionId = resolveSessionId;
|
|
43
|
+
const fs = __importStar(require("fs"));
|
|
44
|
+
const path = __importStar(require("path"));
|
|
45
|
+
const os = __importStar(require("os"));
|
|
46
|
+
const crypto_1 = require("crypto");
|
|
47
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
48
|
+
const DEFAULT_CLAUDE_SESSIONS_DIR = path.join(os.homedir(), ".claude", "sessions");
|
|
49
|
+
const MAX_PID_WALK_DEPTH = 10;
|
|
50
|
+
/**
|
|
51
|
+
* Try to read a Claude Code session ID from a PID-keyed session file.
|
|
52
|
+
* Returns the sessionId if found, null otherwise.
|
|
53
|
+
*/
|
|
54
|
+
function tryReadClaudeSession(sessionsDir, pid) {
|
|
55
|
+
const sessionFile = path.join(sessionsDir, `${pid}.json`);
|
|
56
|
+
if (!fs.existsSync(sessionFile))
|
|
57
|
+
return null;
|
|
58
|
+
try {
|
|
59
|
+
const raw = fs.readFileSync(sessionFile, "utf-8");
|
|
60
|
+
const data = JSON.parse(raw);
|
|
61
|
+
if (typeof data.sessionId === "string" && data.sessionId.length > 0) {
|
|
62
|
+
return data.sessionId;
|
|
63
|
+
}
|
|
64
|
+
return null;
|
|
65
|
+
}
|
|
66
|
+
catch {
|
|
67
|
+
return null;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Walk the parent PID chain looking for a Claude Code session file.
|
|
72
|
+
* Starts at the current process PID, walks up to parent, grandparent, etc.
|
|
73
|
+
* Returns the session ID from the first matching file, or null.
|
|
74
|
+
*/
|
|
75
|
+
function walkPidChain(sessionsDir) {
|
|
76
|
+
let currentPid = process.pid;
|
|
77
|
+
for (let depth = 0; depth < MAX_PID_WALK_DEPTH; depth++) {
|
|
78
|
+
const sessionId = tryReadClaudeSession(sessionsDir, currentPid);
|
|
79
|
+
if (sessionId) {
|
|
80
|
+
(0, runtime_1.emitNervesEvent)({
|
|
81
|
+
component: "daemon",
|
|
82
|
+
event: "daemon.session_id_pid_walk_hit",
|
|
83
|
+
message: "found Claude session via PID walk",
|
|
84
|
+
meta: { pid: currentPid, depth, sessionId },
|
|
85
|
+
});
|
|
86
|
+
return sessionId;
|
|
87
|
+
}
|
|
88
|
+
// Walk to parent PID
|
|
89
|
+
// On macOS/Linux, process.ppid gives the parent. For deeper ancestry,
|
|
90
|
+
// we'd need /proc/{pid}/stat or ps -o ppid=. For now, we check current + parent.
|
|
91
|
+
if (depth === 0) {
|
|
92
|
+
currentPid = process.ppid;
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
// Cannot walk further without OS-specific process tree APIs
|
|
96
|
+
break;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
return null;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Resolve a session ID for the current MCP connection.
|
|
103
|
+
* Returns a stable identifier that ties MCP tool calls to a conversation session.
|
|
104
|
+
*
|
|
105
|
+
* Resolution order:
|
|
106
|
+
* 1. Claude Code PID walk: check ~/.claude/sessions/{pid}.json for current + parent PID
|
|
107
|
+
* 2. UUID fallback: generate a random UUID
|
|
108
|
+
*/
|
|
109
|
+
function resolveSessionId(options) {
|
|
110
|
+
const sessionsDir = options?.claudeSessionsDir ?? DEFAULT_CLAUDE_SESSIONS_DIR;
|
|
111
|
+
// Try Claude Code PID walk
|
|
112
|
+
const claudeSessionId = walkPidChain(sessionsDir);
|
|
113
|
+
if (claudeSessionId) {
|
|
114
|
+
(0, runtime_1.emitNervesEvent)({
|
|
115
|
+
component: "daemon",
|
|
116
|
+
event: "daemon.session_id_resolved",
|
|
117
|
+
message: "session ID resolved via Claude Code PID walk",
|
|
118
|
+
meta: { sessionId: claudeSessionId, method: "claude-pid-walk" },
|
|
119
|
+
});
|
|
120
|
+
return claudeSessionId;
|
|
121
|
+
}
|
|
122
|
+
// Fallback: UUID
|
|
123
|
+
const sessionId = (0, crypto_1.randomUUID)();
|
|
124
|
+
(0, runtime_1.emitNervesEvent)({
|
|
125
|
+
component: "daemon",
|
|
126
|
+
event: "daemon.session_id_resolved",
|
|
127
|
+
message: "session ID resolved via UUID fallback",
|
|
128
|
+
meta: { sessionId, method: "uuid-fallback" },
|
|
129
|
+
});
|
|
130
|
+
return sessionId;
|
|
131
|
+
}
|
|
@@ -0,0 +1,94 @@
|
|
|
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.ensureSkillManagement = ensureSkillManagement;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
40
|
+
const identity_1 = require("../identity");
|
|
41
|
+
const SKILL_MANAGEMENT_URL = "https://raw.githubusercontent.com/ouroborosbot/ouroboros-skills/main/skills/skill-management/SKILL.md";
|
|
42
|
+
async function ensureSkillManagement() {
|
|
43
|
+
const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
|
|
44
|
+
if (!fs.existsSync(bundlesRoot))
|
|
45
|
+
return;
|
|
46
|
+
// Find all agent bundles
|
|
47
|
+
const entries = fs.readdirSync(bundlesRoot).filter(e => e.endsWith(".ouro"));
|
|
48
|
+
if (entries.length === 0)
|
|
49
|
+
return;
|
|
50
|
+
// Check if ANY bundle is missing the skill
|
|
51
|
+
const missing = entries.filter(e => {
|
|
52
|
+
const targetPath = path.join(bundlesRoot, e, "skills", "skill-management.md");
|
|
53
|
+
return !fs.existsSync(targetPath);
|
|
54
|
+
});
|
|
55
|
+
if (missing.length === 0)
|
|
56
|
+
return;
|
|
57
|
+
// eslint-disable-next-line no-console -- terminal UX: visible install status
|
|
58
|
+
console.log("installing skill-management from ouroboros-skills...");
|
|
59
|
+
try {
|
|
60
|
+
const response = await fetch(SKILL_MANAGEMENT_URL);
|
|
61
|
+
if (!response.ok) {
|
|
62
|
+
// eslint-disable-next-line no-console -- terminal UX: visible install status
|
|
63
|
+
console.error(`✗ failed to fetch skill-management (HTTP ${response.status})`);
|
|
64
|
+
(0, runtime_1.emitNervesEvent)({
|
|
65
|
+
level: "warn",
|
|
66
|
+
component: "daemon",
|
|
67
|
+
event: "daemon.skill_management_install_error",
|
|
68
|
+
message: "failed to fetch skill-management from GitHub",
|
|
69
|
+
meta: { status: response.status, url: SKILL_MANAGEMENT_URL },
|
|
70
|
+
});
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
const content = await response.text();
|
|
74
|
+
for (const bundle of missing) {
|
|
75
|
+
const skillsDir = path.join(bundlesRoot, bundle, "skills");
|
|
76
|
+
const targetPath = path.join(skillsDir, "skill-management.md");
|
|
77
|
+
fs.mkdirSync(skillsDir, { recursive: true });
|
|
78
|
+
fs.writeFileSync(targetPath, content, "utf-8");
|
|
79
|
+
}
|
|
80
|
+
// eslint-disable-next-line no-console -- terminal UX: visible install status
|
|
81
|
+
console.log(`✓ installed skill-management (${missing.length} agent${missing.length > 1 ? "s" : ""})`);
|
|
82
|
+
}
|
|
83
|
+
catch (error) {
|
|
84
|
+
// eslint-disable-next-line no-console -- terminal UX: visible install status
|
|
85
|
+
console.error(`✗ failed to install skill-management: ${error instanceof Error ? error.message : String(error)}`);
|
|
86
|
+
(0, runtime_1.emitNervesEvent)({
|
|
87
|
+
level: "warn",
|
|
88
|
+
component: "daemon",
|
|
89
|
+
event: "daemon.skill_management_install_error",
|
|
90
|
+
message: "failed to install skill-management skill",
|
|
91
|
+
meta: { error: error instanceof Error ? error.message : String(error) },
|
|
92
|
+
});
|
|
93
|
+
}
|
|
94
|
+
}
|