@ouro.bot/cli 0.1.0-alpha.5 → 0.1.0-alpha.51
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/AdoptionSpecialist.ouro/agent.json +70 -9
- package/AdoptionSpecialist.ouro/psyche/SOUL.md +5 -2
- package/AdoptionSpecialist.ouro/psyche/identities/monty.md +2 -2
- package/README.md +117 -188
- package/assets/ouroboros.png +0 -0
- package/changelog.json +252 -0
- package/dist/heart/active-work.js +157 -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/config.js +81 -8
- package/dist/heart/core.js +172 -52
- package/dist/heart/daemon/agent-discovery.js +81 -0
- package/dist/heart/daemon/daemon-cli.js +1099 -164
- package/dist/heart/daemon/daemon-entry.js +14 -5
- package/dist/heart/daemon/daemon-runtime-sync.js +90 -0
- package/dist/heart/daemon/daemon.js +184 -9
- package/dist/heart/daemon/hatch-animation.js +10 -3
- package/dist/heart/daemon/hatch-flow.js +3 -20
- package/dist/heart/daemon/hooks/bundle-meta.js +92 -0
- package/dist/heart/daemon/launchd.js +151 -0
- package/dist/heart/daemon/message-router.js +15 -6
- package/dist/heart/daemon/ouro-bot-entry.js +0 -0
- package/dist/heart/daemon/ouro-bot-global-installer.js +128 -0
- package/dist/heart/daemon/ouro-entry.js +0 -0
- package/dist/heart/daemon/ouro-path-installer.js +178 -0
- package/dist/heart/daemon/ouro-uti.js +11 -2
- package/dist/heart/daemon/process-manager.js +1 -1
- package/dist/heart/daemon/run-hooks.js +37 -0
- package/dist/heart/daemon/runtime-metadata.js +118 -0
- package/dist/heart/daemon/sense-manager.js +290 -0
- package/dist/heart/daemon/socket-client.js +202 -0
- package/dist/heart/daemon/specialist-orchestrator.js +53 -84
- package/dist/heart/daemon/specialist-prompt.js +64 -5
- package/dist/heart/daemon/specialist-tools.js +213 -58
- package/dist/heart/daemon/staged-restart.js +114 -0
- package/dist/heart/daemon/subagent-installer.js +48 -7
- package/dist/heart/daemon/thoughts.js +379 -0
- package/dist/heart/daemon/update-checker.js +111 -0
- package/dist/heart/daemon/update-hooks.js +138 -0
- package/dist/heart/daemon/wrapper-publish-guard.js +86 -0
- package/dist/heart/delegation.js +62 -0
- package/dist/heart/identity.js +82 -4
- package/dist/heart/kicks.js +1 -19
- package/dist/heart/model-capabilities.js +40 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/providers/anthropic.js +72 -7
- package/dist/heart/providers/azure.js +8 -1
- package/dist/heart/providers/minimax.js +4 -0
- package/dist/heart/providers/openai-codex.js +10 -1
- package/dist/heart/sense-truth.js +61 -0
- package/dist/heart/session-activity.js +169 -0
- package/dist/heart/session-recall.js +116 -0
- package/dist/heart/streaming.js +100 -22
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/mind/associative-recall.js +14 -2
- package/dist/mind/bundle-manifest.js +70 -0
- package/dist/mind/context.js +27 -11
- package/dist/mind/first-impressions.js +16 -2
- package/dist/mind/friends/channel.js +43 -0
- package/dist/mind/friends/store-file.js +19 -0
- package/dist/mind/friends/types.js +9 -1
- package/dist/mind/memory.js +10 -3
- package/dist/mind/pending.js +72 -9
- package/dist/mind/phrases.js +1 -0
- package/dist/mind/prompt.js +275 -77
- package/dist/mind/token-estimate.js +8 -12
- package/dist/nerves/cli-logging.js +15 -2
- package/dist/repertoire/ado-client.js +4 -2
- package/dist/repertoire/coding/feedback.js +134 -0
- package/dist/repertoire/coding/index.js +4 -1
- package/dist/repertoire/coding/manager.js +62 -4
- package/dist/repertoire/coding/spawner.js +3 -3
- package/dist/repertoire/coding/tools.js +41 -2
- package/dist/repertoire/data/ado-endpoints.json +188 -0
- package/dist/repertoire/tasks/board.js +12 -0
- package/dist/repertoire/tasks/index.js +23 -9
- package/dist/repertoire/tasks/transitions.js +1 -2
- package/dist/repertoire/tools-base.js +496 -245
- package/dist/repertoire/tools-bluebubbles.js +93 -0
- package/dist/repertoire/tools-teams.js +58 -25
- package/dist/repertoire/tools.js +93 -49
- package/dist/senses/bluebubbles-client.js +484 -0
- package/dist/senses/bluebubbles-entry.js +13 -0
- package/dist/senses/bluebubbles-inbound-log.js +109 -0
- package/dist/senses/bluebubbles-media.js +338 -0
- package/dist/senses/bluebubbles-model.js +261 -0
- package/dist/senses/bluebubbles-mutation-log.js +116 -0
- package/dist/senses/bluebubbles-runtime-state.js +109 -0
- package/dist/senses/bluebubbles-session-cleanup.js +72 -0
- package/dist/senses/bluebubbles.js +1142 -0
- package/dist/senses/cli.js +340 -138
- package/dist/senses/continuity.js +94 -0
- package/dist/senses/debug-activity.js +148 -0
- package/dist/senses/inner-dialog-worker.js +47 -18
- package/dist/senses/inner-dialog.js +330 -84
- package/dist/senses/pipeline.js +256 -0
- package/dist/senses/teams.js +541 -129
- package/dist/senses/trust-gate.js +112 -2
- package/package.json +14 -3
- package/subagents/README.md +46 -33
- package/subagents/work-doer.md +28 -24
- package/subagents/work-merger.md +24 -30
- package/subagents/work-planner.md +44 -27
- package/dist/heart/daemon/specialist-session.js +0 -142
- package/dist/inner-worker-entry.js +0 -4
|
@@ -0,0 +1,118 @@
|
|
|
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.getRuntimeMetadata = getRuntimeMetadata;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
const childProcess = __importStar(require("child_process"));
|
|
40
|
+
const identity_1 = require("../identity");
|
|
41
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
42
|
+
const UNKNOWN_METADATA = "unknown";
|
|
43
|
+
function optionalFunction(target, key) {
|
|
44
|
+
try {
|
|
45
|
+
const candidate = target[key];
|
|
46
|
+
return typeof candidate === "function" ? candidate : null;
|
|
47
|
+
}
|
|
48
|
+
catch {
|
|
49
|
+
return null;
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
function readVersion(packageJsonPath, readFileSyncImpl) {
|
|
53
|
+
try {
|
|
54
|
+
const parsed = JSON.parse(readFileSyncImpl(packageJsonPath, "utf-8"));
|
|
55
|
+
return typeof parsed.version === "string" && parsed.version.trim().length > 0
|
|
56
|
+
? parsed.version.trim()
|
|
57
|
+
: UNKNOWN_METADATA;
|
|
58
|
+
}
|
|
59
|
+
catch {
|
|
60
|
+
return UNKNOWN_METADATA;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
function readLastUpdated(repoRoot, packageJsonPath, statSyncImpl, execFileSyncImpl) {
|
|
64
|
+
try {
|
|
65
|
+
const raw = execFileSyncImpl("git", ["log", "-1", "--format=%cI"], {
|
|
66
|
+
cwd: repoRoot,
|
|
67
|
+
encoding: "utf-8",
|
|
68
|
+
stdio: ["ignore", "pipe", "ignore"],
|
|
69
|
+
}).trim();
|
|
70
|
+
if (raw.length > 0) {
|
|
71
|
+
return { value: raw, source: "git" };
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
catch {
|
|
75
|
+
// fall through to mtime fallback
|
|
76
|
+
}
|
|
77
|
+
try {
|
|
78
|
+
const stats = statSyncImpl(packageJsonPath);
|
|
79
|
+
return {
|
|
80
|
+
value: stats.mtime.toISOString(),
|
|
81
|
+
source: "package-json-mtime",
|
|
82
|
+
};
|
|
83
|
+
}
|
|
84
|
+
catch {
|
|
85
|
+
return { value: UNKNOWN_METADATA, source: "unknown" };
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
function getRuntimeMetadata(deps = {}) {
|
|
89
|
+
const repoRoot = deps.repoRoot ?? (0, identity_1.getRepoRoot)();
|
|
90
|
+
const readFileSyncImpl = deps.readFileSync ?? optionalFunction(fs, "readFileSync")?.bind(fs) ?? null;
|
|
91
|
+
const statSyncImpl = deps.statSync ?? optionalFunction(fs, "statSync")?.bind(fs) ?? null;
|
|
92
|
+
const execFileSyncImpl = deps.execFileSync
|
|
93
|
+
?? optionalFunction(childProcess, "execFileSync")?.bind(childProcess)
|
|
94
|
+
?? null;
|
|
95
|
+
const packageJsonPath = path.join(repoRoot, "package.json");
|
|
96
|
+
const version = readFileSyncImpl
|
|
97
|
+
? readVersion(packageJsonPath, readFileSyncImpl)
|
|
98
|
+
: UNKNOWN_METADATA;
|
|
99
|
+
const lastUpdated = statSyncImpl
|
|
100
|
+
? readLastUpdated(repoRoot, packageJsonPath, statSyncImpl, execFileSyncImpl ?? (() => {
|
|
101
|
+
throw new Error("git unavailable");
|
|
102
|
+
}))
|
|
103
|
+
: { value: UNKNOWN_METADATA, source: "unknown" };
|
|
104
|
+
(0, runtime_1.emitNervesEvent)({
|
|
105
|
+
component: "daemon",
|
|
106
|
+
event: "daemon.runtime_metadata_read",
|
|
107
|
+
message: "read runtime metadata",
|
|
108
|
+
meta: {
|
|
109
|
+
version,
|
|
110
|
+
lastUpdated: lastUpdated.value,
|
|
111
|
+
lastUpdatedSource: lastUpdated.source,
|
|
112
|
+
},
|
|
113
|
+
});
|
|
114
|
+
return {
|
|
115
|
+
version,
|
|
116
|
+
lastUpdated: lastUpdated.value,
|
|
117
|
+
};
|
|
118
|
+
}
|
|
@@ -0,0 +1,290 @@
|
|
|
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.DaemonSenseManager = void 0;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const os = __importStar(require("os"));
|
|
39
|
+
const path = __importStar(require("path"));
|
|
40
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
41
|
+
const bluebubbles_runtime_state_1 = require("../../senses/bluebubbles-runtime-state");
|
|
42
|
+
const identity_1 = require("../identity");
|
|
43
|
+
const sense_truth_1 = require("../sense-truth");
|
|
44
|
+
const process_manager_1 = require("./process-manager");
|
|
45
|
+
const DEFAULT_TEAMS_PORT = 3978;
|
|
46
|
+
const DEFAULT_BLUEBUBBLES_PORT = 18790;
|
|
47
|
+
const DEFAULT_BLUEBUBBLES_WEBHOOK_PATH = "/bluebubbles-webhook";
|
|
48
|
+
function defaultSenses() {
|
|
49
|
+
return {
|
|
50
|
+
cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
|
|
51
|
+
teams: { ...identity_1.DEFAULT_AGENT_SENSES.teams },
|
|
52
|
+
bluebubbles: { ...identity_1.DEFAULT_AGENT_SENSES.bluebubbles },
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
function readAgentSenses(agentJsonPath) {
|
|
56
|
+
const defaults = defaultSenses();
|
|
57
|
+
let parsed;
|
|
58
|
+
try {
|
|
59
|
+
parsed = JSON.parse(fs.readFileSync(agentJsonPath, "utf-8"));
|
|
60
|
+
}
|
|
61
|
+
catch (error) {
|
|
62
|
+
(0, runtime_1.emitNervesEvent)({
|
|
63
|
+
level: "warn",
|
|
64
|
+
component: "channels",
|
|
65
|
+
event: "channel.daemon_sense_agent_config_fallback",
|
|
66
|
+
message: "using default senses because agent config could not be read",
|
|
67
|
+
meta: {
|
|
68
|
+
path: agentJsonPath,
|
|
69
|
+
reason: error instanceof Error ? error.message : String(error),
|
|
70
|
+
},
|
|
71
|
+
});
|
|
72
|
+
return defaults;
|
|
73
|
+
}
|
|
74
|
+
const rawSenses = parsed.senses;
|
|
75
|
+
if (!rawSenses || typeof rawSenses !== "object" || Array.isArray(rawSenses)) {
|
|
76
|
+
return defaults;
|
|
77
|
+
}
|
|
78
|
+
for (const sense of ["cli", "teams", "bluebubbles"]) {
|
|
79
|
+
const rawSense = rawSenses[sense];
|
|
80
|
+
if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
|
|
81
|
+
continue;
|
|
82
|
+
}
|
|
83
|
+
const enabled = rawSense.enabled;
|
|
84
|
+
if (typeof enabled === "boolean") {
|
|
85
|
+
defaults[sense] = { enabled };
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
return defaults;
|
|
89
|
+
}
|
|
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
|
+
function textField(record, key) {
|
|
107
|
+
const value = record?.[key];
|
|
108
|
+
return typeof value === "string" ? value.trim() : "";
|
|
109
|
+
}
|
|
110
|
+
function numberField(record, key, fallback) {
|
|
111
|
+
const value = record?.[key];
|
|
112
|
+
return typeof value === "number" && Number.isFinite(value) ? value : fallback;
|
|
113
|
+
}
|
|
114
|
+
function senseFactsFromSecrets(agent, senses, secretsPath) {
|
|
115
|
+
const base = {
|
|
116
|
+
cli: { configured: true, detail: "local interactive terminal" },
|
|
117
|
+
teams: { configured: false, detail: "not enabled in agent.json" },
|
|
118
|
+
bluebubbles: { configured: false, detail: "not enabled in agent.json" },
|
|
119
|
+
};
|
|
120
|
+
const { payload, error } = readSecretsPayload(secretsPath);
|
|
121
|
+
const teams = payload.teams;
|
|
122
|
+
const teamsChannel = payload.teamsChannel;
|
|
123
|
+
const bluebubbles = payload.bluebubbles;
|
|
124
|
+
const bluebubblesChannel = payload.bluebubblesChannel;
|
|
125
|
+
if (senses.teams.enabled) {
|
|
126
|
+
const missing = [];
|
|
127
|
+
if (!textField(teams, "clientId"))
|
|
128
|
+
missing.push("teams.clientId");
|
|
129
|
+
if (!textField(teams, "clientSecret"))
|
|
130
|
+
missing.push("teams.clientSecret");
|
|
131
|
+
if (!textField(teams, "tenantId"))
|
|
132
|
+
missing.push("teams.tenantId");
|
|
133
|
+
base.teams = missing.length === 0
|
|
134
|
+
? {
|
|
135
|
+
configured: true,
|
|
136
|
+
detail: `:${numberField(teamsChannel, "port", DEFAULT_TEAMS_PORT)}`,
|
|
137
|
+
}
|
|
138
|
+
: {
|
|
139
|
+
configured: false,
|
|
140
|
+
detail: error && !fs.existsSync(secretsPath)
|
|
141
|
+
? `missing secrets.json (${agent})`
|
|
142
|
+
: `missing ${missing.join("/")}`,
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
if (senses.bluebubbles.enabled) {
|
|
146
|
+
const missing = [];
|
|
147
|
+
if (!textField(bluebubbles, "serverUrl"))
|
|
148
|
+
missing.push("bluebubbles.serverUrl");
|
|
149
|
+
if (!textField(bluebubbles, "password"))
|
|
150
|
+
missing.push("bluebubbles.password");
|
|
151
|
+
base.bluebubbles = missing.length === 0
|
|
152
|
+
? {
|
|
153
|
+
configured: true,
|
|
154
|
+
detail: `:${numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT)} ${textField(bluebubblesChannel, "webhookPath") || DEFAULT_BLUEBUBBLES_WEBHOOK_PATH}`,
|
|
155
|
+
}
|
|
156
|
+
: {
|
|
157
|
+
configured: false,
|
|
158
|
+
detail: error && !fs.existsSync(secretsPath)
|
|
159
|
+
? `missing secrets.json (${agent})`
|
|
160
|
+
: `missing ${missing.join("/")}`,
|
|
161
|
+
};
|
|
162
|
+
}
|
|
163
|
+
return base;
|
|
164
|
+
}
|
|
165
|
+
function parseSenseSnapshotName(name) {
|
|
166
|
+
const parts = name.split(":");
|
|
167
|
+
if (parts.length !== 2)
|
|
168
|
+
return null;
|
|
169
|
+
const [agent, sense] = parts;
|
|
170
|
+
if (sense !== "teams" && sense !== "bluebubbles")
|
|
171
|
+
return null;
|
|
172
|
+
return { agent, sense };
|
|
173
|
+
}
|
|
174
|
+
function runtimeInfoFor(status) {
|
|
175
|
+
if (status === "running")
|
|
176
|
+
return { runtime: "running" };
|
|
177
|
+
return { runtime: "error" };
|
|
178
|
+
}
|
|
179
|
+
function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
|
|
180
|
+
const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
|
|
181
|
+
const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
|
|
182
|
+
if (snapshot?.runtime !== "running" || !fs.existsSync(runtimePath)) {
|
|
183
|
+
return { runtime: snapshot?.runtime };
|
|
184
|
+
}
|
|
185
|
+
const state = (0, bluebubbles_runtime_state_1.readBlueBubblesRuntimeState)(agent, agentRoot);
|
|
186
|
+
if (state.upstreamStatus === "error") {
|
|
187
|
+
return {
|
|
188
|
+
runtime: "error",
|
|
189
|
+
detail: state.detail,
|
|
190
|
+
};
|
|
191
|
+
}
|
|
192
|
+
return { runtime: snapshot.runtime };
|
|
193
|
+
}
|
|
194
|
+
class DaemonSenseManager {
|
|
195
|
+
processManager;
|
|
196
|
+
contexts;
|
|
197
|
+
bundlesRoot;
|
|
198
|
+
constructor(options) {
|
|
199
|
+
const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
|
|
200
|
+
const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
|
|
201
|
+
this.bundlesRoot = bundlesRoot;
|
|
202
|
+
this.contexts = new Map(options.agents.map((agent) => {
|
|
203
|
+
const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
|
|
204
|
+
const facts = senseFactsFromSecrets(agent, senses, path.join(secretsRoot, agent, "secrets.json"));
|
|
205
|
+
return [agent, { senses, facts }];
|
|
206
|
+
}));
|
|
207
|
+
const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
208
|
+
return ["teams", "bluebubbles"]
|
|
209
|
+
.filter((sense) => context.senses[sense].enabled && context.facts[sense].configured)
|
|
210
|
+
.map((sense) => ({
|
|
211
|
+
name: `${agent}:${sense}`,
|
|
212
|
+
agentArg: agent,
|
|
213
|
+
entry: sense === "teams" ? "senses/teams-entry.js" : "senses/bluebubbles-entry.js",
|
|
214
|
+
channel: sense,
|
|
215
|
+
autoStart: true,
|
|
216
|
+
}));
|
|
217
|
+
});
|
|
218
|
+
this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
|
|
219
|
+
agents: managedSenseAgents,
|
|
220
|
+
});
|
|
221
|
+
(0, runtime_1.emitNervesEvent)({
|
|
222
|
+
component: "channels",
|
|
223
|
+
event: "channel.daemon_sense_manager_init",
|
|
224
|
+
message: "initialized daemon sense manager",
|
|
225
|
+
meta: {
|
|
226
|
+
agents: options.agents,
|
|
227
|
+
managedSenseProcesses: managedSenseAgents.map((entry) => entry.name),
|
|
228
|
+
},
|
|
229
|
+
});
|
|
230
|
+
}
|
|
231
|
+
async startAutoStartSenses() {
|
|
232
|
+
await this.processManager.startAutoStartAgents();
|
|
233
|
+
}
|
|
234
|
+
async stopAll() {
|
|
235
|
+
await this.processManager.stopAll();
|
|
236
|
+
}
|
|
237
|
+
listSenseRows() {
|
|
238
|
+
const runtime = new Map();
|
|
239
|
+
for (const snapshot of this.processManager.listAgentSnapshots()) {
|
|
240
|
+
const parsed = parseSenseSnapshotName(snapshot.name);
|
|
241
|
+
if (!parsed)
|
|
242
|
+
continue;
|
|
243
|
+
const current = runtime.get(parsed.agent) ?? {};
|
|
244
|
+
current[parsed.sense] = runtimeInfoFor(snapshot.status);
|
|
245
|
+
runtime.set(parsed.agent, current);
|
|
246
|
+
}
|
|
247
|
+
const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
|
|
248
|
+
const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
|
|
249
|
+
const runtimeInfo = {
|
|
250
|
+
cli: { configured: true },
|
|
251
|
+
teams: {
|
|
252
|
+
configured: context.facts.teams.configured,
|
|
253
|
+
...(runtime.get(agent)?.teams ?? {}),
|
|
254
|
+
},
|
|
255
|
+
bluebubbles: {
|
|
256
|
+
configured: context.facts.bluebubbles.configured,
|
|
257
|
+
...blueBubblesRuntimeFacts,
|
|
258
|
+
},
|
|
259
|
+
};
|
|
260
|
+
const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
|
|
261
|
+
return inventory.map((entry) => ({
|
|
262
|
+
agent,
|
|
263
|
+
sense: entry.sense,
|
|
264
|
+
label: entry.label,
|
|
265
|
+
enabled: entry.enabled,
|
|
266
|
+
status: entry.status,
|
|
267
|
+
detail: entry.enabled
|
|
268
|
+
? entry.sense === "bluebubbles"
|
|
269
|
+
? blueBubblesRuntimeFacts.detail
|
|
270
|
+
?? context.facts[entry.sense].detail
|
|
271
|
+
: context.facts[entry.sense].detail
|
|
272
|
+
: "not enabled in agent.json",
|
|
273
|
+
}));
|
|
274
|
+
});
|
|
275
|
+
(0, runtime_1.emitNervesEvent)({
|
|
276
|
+
component: "channels",
|
|
277
|
+
event: "channel.daemon_sense_rows_built",
|
|
278
|
+
message: "built daemon sense status rows",
|
|
279
|
+
meta: {
|
|
280
|
+
rows: rows.map((row) => ({
|
|
281
|
+
agent: row.agent,
|
|
282
|
+
sense: row.sense,
|
|
283
|
+
status: row.status,
|
|
284
|
+
})),
|
|
285
|
+
},
|
|
286
|
+
});
|
|
287
|
+
return rows;
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
exports.DaemonSenseManager = DaemonSenseManager;
|
|
@@ -0,0 +1,202 @@
|
|
|
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.DEFAULT_DAEMON_SOCKET_PATH = void 0;
|
|
37
|
+
exports.sendDaemonCommand = sendDaemonCommand;
|
|
38
|
+
exports.checkDaemonSocketAlive = checkDaemonSocketAlive;
|
|
39
|
+
exports.requestInnerWake = requestInnerWake;
|
|
40
|
+
const fs = __importStar(require("fs"));
|
|
41
|
+
const net = __importStar(require("net"));
|
|
42
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
43
|
+
exports.DEFAULT_DAEMON_SOCKET_PATH = "/tmp/ouroboros-daemon.sock";
|
|
44
|
+
function sendDaemonCommand(socketPath, command) {
|
|
45
|
+
(0, runtime_1.emitNervesEvent)({
|
|
46
|
+
component: "daemon",
|
|
47
|
+
event: "daemon.socket_command_start",
|
|
48
|
+
message: "sending daemon command over socket",
|
|
49
|
+
meta: { socketPath, kind: command.kind },
|
|
50
|
+
});
|
|
51
|
+
return new Promise((resolve, reject) => {
|
|
52
|
+
const client = net.createConnection(socketPath);
|
|
53
|
+
let raw = "";
|
|
54
|
+
client.on("connect", () => {
|
|
55
|
+
client.write(JSON.stringify(command));
|
|
56
|
+
client.end();
|
|
57
|
+
});
|
|
58
|
+
client.on("data", (chunk) => {
|
|
59
|
+
raw += chunk.toString("utf-8");
|
|
60
|
+
});
|
|
61
|
+
client.on("error", (error) => {
|
|
62
|
+
(0, runtime_1.emitNervesEvent)({
|
|
63
|
+
level: "error",
|
|
64
|
+
component: "daemon",
|
|
65
|
+
event: "daemon.socket_command_error",
|
|
66
|
+
message: "daemon socket command failed",
|
|
67
|
+
meta: {
|
|
68
|
+
socketPath,
|
|
69
|
+
kind: command.kind,
|
|
70
|
+
error: error.message,
|
|
71
|
+
},
|
|
72
|
+
});
|
|
73
|
+
reject(error);
|
|
74
|
+
});
|
|
75
|
+
client.on("end", () => {
|
|
76
|
+
const trimmed = raw.trim();
|
|
77
|
+
if (trimmed.length === 0 && command.kind === "daemon.stop") {
|
|
78
|
+
(0, runtime_1.emitNervesEvent)({
|
|
79
|
+
component: "daemon",
|
|
80
|
+
event: "daemon.socket_command_end",
|
|
81
|
+
message: "daemon socket command completed",
|
|
82
|
+
meta: { socketPath, kind: command.kind, ok: true },
|
|
83
|
+
});
|
|
84
|
+
resolve({ ok: true, message: "daemon stopped" });
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
if (trimmed.length === 0) {
|
|
88
|
+
const error = new Error("Daemon returned empty response.");
|
|
89
|
+
(0, runtime_1.emitNervesEvent)({
|
|
90
|
+
level: "error",
|
|
91
|
+
component: "daemon",
|
|
92
|
+
event: "daemon.socket_command_error",
|
|
93
|
+
message: "daemon socket command returned empty response",
|
|
94
|
+
meta: {
|
|
95
|
+
socketPath,
|
|
96
|
+
kind: command.kind,
|
|
97
|
+
error: error.message,
|
|
98
|
+
},
|
|
99
|
+
});
|
|
100
|
+
reject(error);
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
try {
|
|
104
|
+
const parsed = JSON.parse(trimmed);
|
|
105
|
+
(0, runtime_1.emitNervesEvent)({
|
|
106
|
+
component: "daemon",
|
|
107
|
+
event: "daemon.socket_command_end",
|
|
108
|
+
message: "daemon socket command completed",
|
|
109
|
+
meta: {
|
|
110
|
+
socketPath,
|
|
111
|
+
kind: command.kind,
|
|
112
|
+
ok: parsed.ok,
|
|
113
|
+
},
|
|
114
|
+
});
|
|
115
|
+
resolve(parsed);
|
|
116
|
+
}
|
|
117
|
+
catch (error) {
|
|
118
|
+
(0, runtime_1.emitNervesEvent)({
|
|
119
|
+
level: "error",
|
|
120
|
+
component: "daemon",
|
|
121
|
+
event: "daemon.socket_command_error",
|
|
122
|
+
message: "daemon socket command returned invalid JSON",
|
|
123
|
+
meta: {
|
|
124
|
+
socketPath,
|
|
125
|
+
kind: command.kind,
|
|
126
|
+
error: error instanceof Error ? error.message : String(error),
|
|
127
|
+
},
|
|
128
|
+
});
|
|
129
|
+
reject(error);
|
|
130
|
+
}
|
|
131
|
+
});
|
|
132
|
+
});
|
|
133
|
+
}
|
|
134
|
+
function checkDaemonSocketAlive(socketPath) {
|
|
135
|
+
(0, runtime_1.emitNervesEvent)({
|
|
136
|
+
component: "daemon",
|
|
137
|
+
event: "daemon.socket_alive_check_start",
|
|
138
|
+
message: "checking daemon socket health",
|
|
139
|
+
meta: { socketPath },
|
|
140
|
+
});
|
|
141
|
+
return new Promise((resolve) => {
|
|
142
|
+
const client = net.createConnection(socketPath);
|
|
143
|
+
let raw = "";
|
|
144
|
+
let done = false;
|
|
145
|
+
const finalize = (alive) => {
|
|
146
|
+
if (done)
|
|
147
|
+
return;
|
|
148
|
+
done = true;
|
|
149
|
+
(0, runtime_1.emitNervesEvent)({
|
|
150
|
+
component: "daemon",
|
|
151
|
+
event: "daemon.socket_alive_check_end",
|
|
152
|
+
message: "daemon socket health check completed",
|
|
153
|
+
meta: { socketPath, alive },
|
|
154
|
+
});
|
|
155
|
+
resolve(alive);
|
|
156
|
+
};
|
|
157
|
+
if ("setTimeout" in client && typeof client.setTimeout === "function") {
|
|
158
|
+
client.setTimeout(800, () => {
|
|
159
|
+
client.destroy();
|
|
160
|
+
finalize(false);
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
client.on("connect", () => {
|
|
164
|
+
client.write(JSON.stringify({ kind: "daemon.status" }));
|
|
165
|
+
client.end();
|
|
166
|
+
});
|
|
167
|
+
client.on("data", (chunk) => {
|
|
168
|
+
raw += chunk.toString("utf-8");
|
|
169
|
+
});
|
|
170
|
+
client.on("error", () => finalize(false));
|
|
171
|
+
client.on("end", () => {
|
|
172
|
+
if (raw.trim().length === 0) {
|
|
173
|
+
finalize(false);
|
|
174
|
+
return;
|
|
175
|
+
}
|
|
176
|
+
try {
|
|
177
|
+
JSON.parse(raw);
|
|
178
|
+
finalize(true);
|
|
179
|
+
}
|
|
180
|
+
catch {
|
|
181
|
+
finalize(false);
|
|
182
|
+
}
|
|
183
|
+
});
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
async function requestInnerWake(agent, socketPath = exports.DEFAULT_DAEMON_SOCKET_PATH) {
|
|
187
|
+
const socketAvailable = fs.existsSync(socketPath);
|
|
188
|
+
(0, runtime_1.emitNervesEvent)({
|
|
189
|
+
component: "daemon",
|
|
190
|
+
event: "daemon.inner_wake_request",
|
|
191
|
+
message: "requesting daemon-managed inner wake",
|
|
192
|
+
meta: {
|
|
193
|
+
agent,
|
|
194
|
+
socketPath,
|
|
195
|
+
socketAvailable,
|
|
196
|
+
},
|
|
197
|
+
});
|
|
198
|
+
if (!socketAvailable) {
|
|
199
|
+
return null;
|
|
200
|
+
}
|
|
201
|
+
return sendDaemonCommand(socketPath, { kind: "inner.wake", agent });
|
|
202
|
+
}
|