@askexenow/exe-os 0.9.294 → 0.9.296
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/deploy/compose/cloudflared/config.yml.example +14 -9
- package/deploy/compose/docker-compose.yml +86 -8
- package/deploy/compose/sso-edge/default.conf.template +87 -0
- package/deploy/compose/sso-edge/entrypoint.sh +23 -0
- package/deploy/compose/sso-edge/sso-redirect.conf +63 -0
- package/deploy/stack-manifests/v0.9.json +2 -2
- package/dist/active-agent-AFX2FODG.js +28 -0
- package/dist/active-agent-E2IJA7YX.js +27 -0
- package/dist/agentic-ontology-A2YUZK5O.js +25 -0
- package/dist/assets/com.askexe.exed.plist +4 -1
- package/dist/backfill-metadata-OC7EOD5U.js +600 -0
- package/dist/behaviors-H5ZOVHDH.js +46 -0
- package/dist/bin/agentic-ontology-backfill.js +5 -5
- package/dist/bin/agentic-reflection-backfill.js +6 -6
- package/dist/bin/agentic-semantic-label.js +5 -5
- package/dist/bin/backfill-conversations.js +6 -6
- package/dist/bin/backfill-responses.js +6 -6
- package/dist/bin/backfill-vectors.js +8 -8
- package/dist/bin/bulk-sync-postgres.js +7 -7
- package/dist/bin/cc-doctor.js +4 -4
- package/dist/bin/cleanup-stale-review-tasks.js +11 -11
- package/dist/bin/cli.js +16 -16
- package/dist/bin/deferred-daemon-restart.js +1 -1
- package/dist/bin/exe-agent-config.js +2 -2
- package/dist/bin/exe-agent.js +4 -4
- package/dist/bin/exe-assign.js +8 -8
- package/dist/bin/exe-boot.js +21 -18
- package/dist/bin/exe-call.js +4 -4
- package/dist/bin/exe-cloud.js +7 -7
- package/dist/bin/exe-dispatch.js +11 -11
- package/dist/bin/exe-doctor.js +3 -2
- package/dist/bin/exe-export-behaviors.js +7 -7
- package/dist/bin/exe-forget.js +6 -6
- package/dist/bin/exe-gateway.js +7 -7
- package/dist/bin/exe-healthcheck.js +6 -4
- package/dist/bin/exe-heartbeat.js +11 -11
- package/dist/bin/exe-kill.js +14 -14
- package/dist/bin/exe-launch-agent.js +18 -18
- package/dist/bin/exe-new-employee.js +6 -6
- package/dist/bin/exe-pending-messages.js +12 -12
- package/dist/bin/exe-pending-notifications.js +11 -11
- package/dist/bin/exe-pending-reviews.js +11 -11
- package/dist/bin/exe-rename.js +4 -4
- package/dist/bin/exe-review.js +13 -13
- package/dist/bin/exe-search.js +5 -5
- package/dist/bin/exe-session-cleanup.js +16 -16
- package/dist/bin/exe-settings.js +39 -9
- package/dist/bin/exe-start-codex.js +11 -11
- package/dist/bin/exe-start-opencode.js +8 -8
- package/dist/bin/exe-status.js +12 -12
- package/dist/bin/exe-team.js +3 -3
- package/dist/bin/git-sweep.js +12 -12
- package/dist/bin/graph-backfill.js +4 -4
- package/dist/bin/graph-export.js +5 -5
- package/dist/bin/import-history.js +7 -7
- package/dist/bin/install-launchd.js +13 -6
- package/dist/bin/install.js +26 -14
- package/dist/bin/intercom-check.js +4 -4
- package/dist/bin/mcp-sessions.js +2 -2
- package/dist/bin/orchestration-metrics.js +4 -4
- package/dist/bin/postgres-agentic-reflection-backfill.js +2 -2
- package/dist/bin/postgres-agentic-semantic-backfill.js +1 -1
- package/dist/bin/scan-tasks.js +11 -11
- package/dist/bin/setup.js +1 -1
- package/dist/bin/shard-migrate.js +4 -4
- package/dist/bin/stack-update.js +2 -2
- package/dist/bin/vps-health-gate.js +1 -1
- package/dist/capability-cards-4USI7CUW.js +89 -0
- package/dist/capacity-monitor-WLCBTEYR.js +51 -0
- package/dist/catchup-brief-ZR3NX6LZ.js +175 -0
- package/dist/chunk-22TVSRQQ.js +226 -0
- package/dist/chunk-2E43UXRH.js +395 -0
- package/dist/chunk-2PIGT6UJ.js +460 -0
- package/dist/chunk-3XTMW2MZ.js +535 -0
- package/dist/chunk-465PQFTH.js +262 -0
- package/dist/chunk-5CCXU2AW.js +129 -0
- package/dist/chunk-5D6MPWR7.js +1094 -0
- package/dist/chunk-5Q4MR6SL.js +123 -0
- package/dist/chunk-6327RBWR.js +345 -0
- package/dist/chunk-6MZZREZY.js +199 -0
- package/dist/chunk-7DI2Q4O5.js +1186 -0
- package/dist/chunk-7PW5VNIY.js +122 -0
- package/dist/chunk-7T7Y56HW.js +43 -0
- package/dist/chunk-7UHCWCLT.js +128 -0
- package/dist/chunk-A2ZUMF6L.js +1350 -0
- package/dist/chunk-AKV44JEH.js +185 -0
- package/dist/chunk-ANHWGX5N.js +735 -0
- package/dist/chunk-BQ3P4TKD.js +97 -0
- package/dist/chunk-BUZMT3KZ.js +604 -0
- package/dist/chunk-C2SBESBO.js +210 -0
- package/dist/chunk-CLSXZUZW.js +51 -0
- package/dist/chunk-CONHLVAR.js +1079 -0
- package/dist/chunk-D3WTZPFX.js +456 -0
- package/dist/chunk-DE6SOIYL.js +197 -0
- package/dist/chunk-EIVNMA3Q.js +284 -0
- package/dist/chunk-EJIF4FNT.js +12 -0
- package/dist/chunk-FDFOW564.js +171 -0
- package/dist/chunk-GZUBJ5EC.js +127 -0
- package/dist/chunk-HGZITN22.js +105 -0
- package/dist/chunk-HSRKDU6X.js +362 -0
- package/dist/chunk-IIEN2PHV.js +85 -0
- package/dist/chunk-JQ56VLMM.js +567 -0
- package/dist/chunk-JVHHXRFY.js +280 -0
- package/dist/chunk-JXCXGZ3S.js +55 -0
- package/dist/chunk-K5ZO532Q.js +4388 -0
- package/dist/chunk-K6CAAMXF.js +97 -0
- package/dist/chunk-KA26YTNU.js +81 -0
- package/dist/chunk-KMUW5C3R.js +381 -0
- package/dist/chunk-KOO3J5PV.js +20 -0
- package/dist/chunk-LSV7OFIH.js +290 -0
- package/dist/chunk-LSVFDVNY.js +1158 -0
- package/dist/chunk-LXDQTW32.js +230 -0
- package/dist/chunk-MEP7OUVZ.js +181 -0
- package/dist/chunk-MN2B2LKS.js +240 -0
- package/dist/chunk-N2EAYPYQ.js +1352 -0
- package/dist/chunk-N7I2A667.js +70 -0
- package/dist/chunk-NLZHVIOP.js +630 -0
- package/dist/chunk-NUH5TRZL.js +227 -0
- package/dist/chunk-OAHEIH3G.js +167 -0
- package/dist/chunk-OBHRQGCK.js +58 -0
- package/dist/chunk-ODFA7B2V.js +54 -0
- package/dist/chunk-OSNUP45F.js +731 -0
- package/dist/chunk-OTPRHBTO.js +33 -0
- package/dist/chunk-P6MUA4QU.js +157 -0
- package/dist/chunk-PGIOFKSK.js +2093 -0
- package/dist/chunk-PSE7VHWK.js +50 -0
- package/dist/chunk-QIFUVZFW.js +331 -0
- package/dist/chunk-RDPXKTVK.js +221 -0
- package/dist/chunk-RKYTYJGB.js +76 -0
- package/dist/chunk-RXLR6EFM.js +348 -0
- package/dist/chunk-SDB67PQJ.js +159 -0
- package/dist/chunk-SF2T7MP3.js +402 -0
- package/dist/chunk-SLU3FRFQ.js +2133 -0
- package/dist/chunk-SNDZJ5IV.js +214 -0
- package/dist/chunk-STEEAABW.js +448 -0
- package/dist/chunk-TUTWNHIQ.js +244 -0
- package/dist/chunk-UDP35QBR.js +30 -0
- package/dist/chunk-UKFHNJBI.js +85 -0
- package/dist/chunk-VC2DTK2X.js +382 -0
- package/dist/chunk-VRRAE5JX.js +836 -0
- package/dist/chunk-VVJTBQPR.js +38 -0
- package/dist/chunk-W3EQ362K.js +581 -0
- package/dist/chunk-WHIXIFHC.js +2242 -0
- package/dist/chunk-WRNGJJNR.js +377 -0
- package/dist/chunk-WUKHLCBE.js +3313 -0
- package/dist/chunk-WVPLHGDG.js +150 -0
- package/dist/chunk-XJZBSTL5.js +204 -0
- package/dist/chunk-Y3PMNUM5.js +304 -0
- package/dist/chunk-YHVS4QOV.js +14597 -0
- package/dist/chunk-YJ2OYAOC.js +668 -0
- package/dist/chunk-YYAD2GXX.js +128 -0
- package/dist/chunk-ZQML7EWE.js +333 -0
- package/dist/co-activation-XJLH46OX.js +74 -0
- package/dist/co-occurrence-GNN2X526.js +95 -0
- package/dist/code-context-index-OCPRLFG5.js +30 -0
- package/dist/core-memory-J4W2IYOF.js +110 -0
- package/dist/crdt-sync-QCBTSHIH.js +33 -0
- package/dist/crm-webhook-EM442VUW.js +10 -0
- package/dist/cto-delegation-gate-MLJMVHBK.js +280 -0
- package/dist/daemon-orchestration-2VNLZVTW.js +139 -0
- package/dist/db-backup-VUGFTPJ4.js +43 -0
- package/dist/doc-graph-extractor-PNRSFPSS.js +133 -0
- package/dist/dreaming-SK5VEQRF.js +34 -0
- package/dist/entity-boost-TQWWJUC2.js +375 -0
- package/dist/exe-drift-N34UPO7S.js +70 -0
- package/dist/exe-export-KACBKGVV.js +77 -0
- package/dist/exe-import-GXGDWACG.js +80 -0
- package/dist/exe-key-XPDOZBWW.js +673 -0
- package/dist/exe-snapshot-32GQKGQ5.js +338 -0
- package/dist/fast-db-init-F3TDD5VV.js +7 -0
- package/dist/gateway/index.js +8 -8
- package/dist/git-staleness-J45WNYRF.js +112 -0
- package/dist/git-task-sweep-BTGVQPFB.js +42 -0
- package/dist/global-procedures-6JCQWU4D.js +22 -0
- package/dist/graph-auto-extract-3ZQNXTPC.js +183 -0
- package/dist/hooks/bug-report-worker.js +13 -13
- package/dist/hooks/codex-stop-task-finalizer.js +13 -13
- package/dist/hooks/commit-complete.js +13 -13
- package/dist/hooks/error-recall.js +6 -6
- package/dist/hooks/exe-heartbeat-hook.js +3 -3
- package/dist/hooks/ingest-worker.js +3 -3
- package/dist/hooks/ingest.js +6 -6
- package/dist/hooks/instructions-loaded.js +4 -4
- package/dist/hooks/manifest.json +20 -20
- package/dist/hooks/notification.js +4 -4
- package/dist/hooks/post-compact.js +12 -12
- package/dist/hooks/post-tool-combined.js +6 -6
- package/dist/hooks/pre-compact.js +16 -16
- package/dist/hooks/pre-tool-use.js +16 -16
- package/dist/hooks/prompt-submit.js +24 -24
- package/dist/hooks/session-end.js +21 -21
- package/dist/hooks/session-start.js +12 -12
- package/dist/hooks/stop.js +19 -19
- package/dist/hooks/subagent-stop.js +12 -12
- package/dist/hooks/summary-worker.js +19 -19
- package/dist/index.js +19 -19
- package/dist/installer-5VPFY7SB.js +298 -0
- package/dist/installer-OENFPMA2.js +344 -0
- package/dist/installer-OIX4QOG5.js +40 -0
- package/dist/lib/cloud-sync.js +7 -7
- package/dist/lib/consolidation.js +6 -5
- package/dist/lib/database.js +2 -2
- package/dist/lib/db-daemon-client.js +2 -2
- package/dist/lib/db.js +2 -2
- package/dist/lib/embed-worker.js +1 -0
- package/dist/lib/embedder.js +7 -3
- package/dist/lib/employee-templates.js +4 -4
- package/dist/lib/employees.js +2 -2
- package/dist/lib/exe-daemon-client.js +2 -2
- package/dist/lib/exe-daemon.js +160 -79
- package/dist/lib/hybrid-search.js +5 -5
- package/dist/lib/identity.js +2 -2
- package/dist/lib/messaging.js +11 -11
- package/dist/lib/reminders.js +3 -3
- package/dist/lib/schedules.js +5 -5
- package/dist/lib/session-registry.js +4 -4
- package/dist/lib/skill-learning.js +6 -6
- package/dist/lib/store.js +4 -4
- package/dist/lib/task-router.js +3 -3
- package/dist/lib/tasks.js +12 -12
- package/dist/lib/tmux-routing.js +12 -10
- package/dist/lib/tmux-transport.js +1 -1
- package/dist/lib/token-spend.js +3 -3
- package/dist/lib/transport.js +2 -2
- package/dist/mcp/register-tools.js +62 -61
- package/dist/mcp/server.js +63 -62
- package/dist/mcp/tools/complete-reminder.js +4 -4
- package/dist/mcp/tools/create-reminder.js +4 -4
- package/dist/mcp/tools/create-task.js +14 -14
- package/dist/mcp/tools/deactivate-behavior.js +7 -7
- package/dist/mcp/tools/list-reminders.js +4 -4
- package/dist/mcp/tools/list-tasks.js +14 -14
- package/dist/mcp/tools/send-message.js +13 -13
- package/dist/mcp/tools/update-task.js +13 -13
- package/dist/mcp-http-config-PQTOLCTP.js +29 -0
- package/dist/memory-cards-4RVDZIY2.js +180 -0
- package/dist/memory-graph-extractor-L6YC7G4M.js +22 -0
- package/dist/memory-poisoning-defense-4YVJYH4G.js +224 -0
- package/dist/memory-queue-client-MVAUOZNJ.js +16 -0
- package/dist/memory-reflection-SHHDQNOH.js +244 -0
- package/dist/message-queue-client-DCKZT6X2.js +92 -0
- package/dist/notifications-JFR3G42W.js +47 -0
- package/dist/orchestration-events-MGCGPTDN.js +27 -0
- package/dist/orchestrator-DAFL2YZB.js +35 -0
- package/dist/pipeline-router-WWSZVPCH.js +15 -0
- package/dist/plan-limits-C7XCSDZC.js +28 -0
- package/dist/project-boot-N3NTBVLE.js +299 -0
- package/dist/projection-worker-MTPAPCWX.js +1084 -0
- package/dist/prospective-memory-BTIVUJSB.js +232 -0
- package/dist/reranker-UA6WVESJ.js +19 -0
- package/dist/retrieval-health-7XNZJEBF.js +12 -0
- package/dist/review-polling-4ALGMXC3.js +126 -0
- package/dist/runtime/index.js +13 -13
- package/dist/self-query-router-MROFQLQB.js +192 -0
- package/dist/session-events-CK44XOU4.js +38 -0
- package/dist/session-kill-telemetry-MT6ITDOG.js +31 -0
- package/dist/session-scope-3XDBWV65.js +88 -0
- package/dist/setup-wizard-X6DOD7MC.js +12 -0
- package/dist/skill-refinement-G2CCY3GM.js +159 -0
- package/dist/stack-update-JF7F56AS.js +84 -0
- package/dist/steward-gate-YF2CYXE7.js +15 -0
- package/dist/task-enforcement-YN6HK7NE.js +506 -0
- package/dist/task-scope-CVK6ISCZ.js +37 -0
- package/dist/tasks-crud-NTNET4JE.js +79 -0
- package/dist/tasks-notify-4LJVFPCV.js +40 -0
- package/dist/tasks-review-3V4WOIRG.js +49 -0
- package/dist/telemetry-upload-5PNUKGTM.js +741 -0
- package/dist/token-budget-E46G7ZAQ.js +86 -0
- package/dist/tool-capability-index-JDSMKJER.js +10 -0
- package/dist/tool-telemetry-J3NLS3LJ.js +17 -0
- package/dist/tui/App.js +18 -18
- package/dist/tui-data-6DOMUUCM.js +260 -0
- package/dist/wiki-acl-5UK37LKF.js +111 -0
- package/dist/worker-gate-FM7AEC7G.js +21 -0
- package/dist/workflow-engine-2EDUHUIY.js +28 -0
- package/dist/worktree-7YKKJIYR.js +28 -0
- package/dist/worktree-sweep-C3ELFGDN.js +21 -0
- package/package.json +1 -1
- package/release-notes.json +88 -88
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getClient
|
|
3
|
+
} from "./chunk-WUKHLCBE.js";
|
|
4
|
+
|
|
5
|
+
// src/lib/session-kill-telemetry.ts
|
|
6
|
+
import crypto from "crypto";
|
|
7
|
+
import { appendFileSync } from "fs";
|
|
8
|
+
import path from "path";
|
|
9
|
+
import os from "os";
|
|
10
|
+
var KILL_FILE_LOG = path.join(os.homedir(), ".exe-os", "logs", "session-kills.jsonl");
|
|
11
|
+
var TOKENS_PER_IDLE_MINUTE = 50;
|
|
12
|
+
async function recordSessionKill(input) {
|
|
13
|
+
const killedAt = (/* @__PURE__ */ new Date()).toISOString();
|
|
14
|
+
const id = crypto.randomUUID();
|
|
15
|
+
try {
|
|
16
|
+
const logEntry = JSON.stringify({
|
|
17
|
+
id,
|
|
18
|
+
timestamp: killedAt,
|
|
19
|
+
session: input.sessionName,
|
|
20
|
+
agent: input.agentId,
|
|
21
|
+
reason: input.reason,
|
|
22
|
+
ticksIdle: input.ticksIdle ?? null,
|
|
23
|
+
tokensSaved: input.estimatedTokensSaved ?? null
|
|
24
|
+
});
|
|
25
|
+
appendFileSync(KILL_FILE_LOG, logEntry + "\n");
|
|
26
|
+
} catch {
|
|
27
|
+
process.stderr.write(
|
|
28
|
+
`[session-kill-telemetry] file log write failed for ${input.sessionName}
|
|
29
|
+
`
|
|
30
|
+
);
|
|
31
|
+
}
|
|
32
|
+
try {
|
|
33
|
+
const client = getClient();
|
|
34
|
+
await client.execute({
|
|
35
|
+
sql: `INSERT INTO session_kills
|
|
36
|
+
(id, session_name, agent_id, killed_at, reason,
|
|
37
|
+
ticks_idle, estimated_tokens_saved)
|
|
38
|
+
VALUES (?, ?, ?, ?, ?, ?, ?)`,
|
|
39
|
+
args: [
|
|
40
|
+
id,
|
|
41
|
+
input.sessionName,
|
|
42
|
+
input.agentId,
|
|
43
|
+
killedAt,
|
|
44
|
+
input.reason,
|
|
45
|
+
input.ticksIdle ?? null,
|
|
46
|
+
input.estimatedTokensSaved ?? null
|
|
47
|
+
]
|
|
48
|
+
});
|
|
49
|
+
} catch (err) {
|
|
50
|
+
process.stderr.write(
|
|
51
|
+
`[session-kill-telemetry] DB write failed: ${err instanceof Error ? err.message : String(err)}
|
|
52
|
+
`
|
|
53
|
+
);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
async function countKillsSince(sinceISO) {
|
|
57
|
+
try {
|
|
58
|
+
const client = getClient();
|
|
59
|
+
const result = await client.execute({
|
|
60
|
+
sql: `SELECT COUNT(*) AS n FROM session_kills WHERE killed_at >= ?`,
|
|
61
|
+
args: [sinceISO]
|
|
62
|
+
});
|
|
63
|
+
const row = result.rows[0];
|
|
64
|
+
return row ? Number(row.n) : 0;
|
|
65
|
+
} catch {
|
|
66
|
+
return 0;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
var IDLE_KILL_STREAK_META_KEY = "idle_kill_suspect_streak";
|
|
70
|
+
var IDLE_KILL_SUSPECT_DAY_THRESHOLD = 3;
|
|
71
|
+
var IDLE_KILL_MIN_LIVE_SESSIONS = 5;
|
|
72
|
+
function parseStreakState(raw) {
|
|
73
|
+
if (!raw) return { lastDate: null, streak: 0 };
|
|
74
|
+
try {
|
|
75
|
+
const parsed = JSON.parse(raw);
|
|
76
|
+
return {
|
|
77
|
+
lastDate: typeof parsed.lastDate === "string" ? parsed.lastDate : null,
|
|
78
|
+
streak: typeof parsed.streak === "number" ? parsed.streak : 0
|
|
79
|
+
};
|
|
80
|
+
} catch {
|
|
81
|
+
return { lastDate: null, streak: 0 };
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
function nextStreakState(prev, qualifiesToday, todayDate) {
|
|
85
|
+
if (!qualifiesToday) return { lastDate: todayDate, streak: 0 };
|
|
86
|
+
if (prev.lastDate === todayDate) return prev;
|
|
87
|
+
return { lastDate: todayDate, streak: prev.streak + 1 };
|
|
88
|
+
}
|
|
89
|
+
function computeIdleKillSuspectStreak(prev, killsToday, liveSessions, todayDate) {
|
|
90
|
+
const qualifies = killsToday === 0 && liveSessions >= IDLE_KILL_MIN_LIVE_SESSIONS;
|
|
91
|
+
const state = nextStreakState(prev, qualifies, todayDate);
|
|
92
|
+
return {
|
|
93
|
+
state,
|
|
94
|
+
suspect: state.streak >= IDLE_KILL_SUSPECT_DAY_THRESHOLD
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
async function sumTokensSavedSince(sinceISO) {
|
|
98
|
+
try {
|
|
99
|
+
const client = getClient();
|
|
100
|
+
const result = await client.execute({
|
|
101
|
+
sql: `SELECT COALESCE(SUM(estimated_tokens_saved), 0) AS total
|
|
102
|
+
FROM session_kills
|
|
103
|
+
WHERE killed_at >= ?`,
|
|
104
|
+
args: [sinceISO]
|
|
105
|
+
});
|
|
106
|
+
const row = result.rows[0];
|
|
107
|
+
return row ? Number(row.total) : 0;
|
|
108
|
+
} catch {
|
|
109
|
+
return 0;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
export {
|
|
114
|
+
TOKENS_PER_IDLE_MINUTE,
|
|
115
|
+
recordSessionKill,
|
|
116
|
+
countKillsSince,
|
|
117
|
+
IDLE_KILL_STREAK_META_KEY,
|
|
118
|
+
IDLE_KILL_SUSPECT_DAY_THRESHOLD,
|
|
119
|
+
IDLE_KILL_MIN_LIVE_SESSIONS,
|
|
120
|
+
parseStreakState,
|
|
121
|
+
computeIdleKillSuspectStreak,
|
|
122
|
+
sumTokensSavedSince
|
|
123
|
+
};
|
|
@@ -0,0 +1,345 @@
|
|
|
1
|
+
import {
|
|
2
|
+
employeeSessionName,
|
|
3
|
+
isEmployeeAlive,
|
|
4
|
+
resolveExeSession,
|
|
5
|
+
sendIntercom,
|
|
6
|
+
strictSessionScopeFilter
|
|
7
|
+
} from "./chunk-K5ZO532Q.js";
|
|
8
|
+
import {
|
|
9
|
+
parseMessage,
|
|
10
|
+
serializeMessage
|
|
11
|
+
} from "./chunk-4JERP7NT.js";
|
|
12
|
+
import {
|
|
13
|
+
recordOrchestrationEventBestEffort
|
|
14
|
+
} from "./chunk-NLZHVIOP.js";
|
|
15
|
+
import {
|
|
16
|
+
getClient
|
|
17
|
+
} from "./chunk-WUKHLCBE.js";
|
|
18
|
+
|
|
19
|
+
// src/lib/messaging.ts
|
|
20
|
+
import crypto from "crypto";
|
|
21
|
+
function generateUlid() {
|
|
22
|
+
const timestamp = Date.now().toString(36).padStart(10, "0");
|
|
23
|
+
const random = crypto.randomBytes(10).toString("hex").slice(0, 16);
|
|
24
|
+
return (timestamp + random).toUpperCase();
|
|
25
|
+
}
|
|
26
|
+
function rowToMessage(row) {
|
|
27
|
+
return {
|
|
28
|
+
id: row.id,
|
|
29
|
+
fromAgent: row.from_agent,
|
|
30
|
+
fromDevice: row.from_device,
|
|
31
|
+
targetAgent: row.target_agent,
|
|
32
|
+
targetProject: row.target_project ?? null,
|
|
33
|
+
targetDevice: row.target_device,
|
|
34
|
+
sessionScope: row.session_scope ?? null,
|
|
35
|
+
content: row.content,
|
|
36
|
+
priority: row.priority ?? "normal",
|
|
37
|
+
status: row.status ?? "pending",
|
|
38
|
+
serverSeq: row.server_seq != null ? Number(row.server_seq) : null,
|
|
39
|
+
retryCount: Number(row.retry_count ?? 0),
|
|
40
|
+
createdAt: row.created_at,
|
|
41
|
+
deliveredAt: row.delivered_at ?? null,
|
|
42
|
+
processedAt: row.processed_at ?? null,
|
|
43
|
+
failedAt: row.failed_at ?? null,
|
|
44
|
+
failureReason: row.failure_reason ?? null
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
var MAX_RETRIES = 10;
|
|
48
|
+
async function countUnreadMessages(targetAgent, sessionScope) {
|
|
49
|
+
const client = getClient();
|
|
50
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
51
|
+
const result = await client.execute({
|
|
52
|
+
sql: `SELECT COUNT(*) AS count FROM messages
|
|
53
|
+
WHERE target_agent = ? AND status IN ('pending', 'delivered')${scope.sql}`,
|
|
54
|
+
args: [targetAgent, ...scope.args]
|
|
55
|
+
});
|
|
56
|
+
return Number(result.rows[0]?.count ?? 0);
|
|
57
|
+
}
|
|
58
|
+
async function sendMessage(input) {
|
|
59
|
+
const client = getClient();
|
|
60
|
+
const id = generateUlid();
|
|
61
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
62
|
+
const targetDevice = input.targetDevice ?? "local";
|
|
63
|
+
const sessionScope = input.sessionScope === void 0 ? resolveExeSession() : input.sessionScope;
|
|
64
|
+
await client.execute({
|
|
65
|
+
sql: `INSERT INTO messages (id, from_agent, from_device, target_agent, target_project, target_device, session_scope, content, priority, status, founder_id, created_at)
|
|
66
|
+
VALUES (?, ?, 'local', ?, ?, ?, ?, ?, ?, 'pending', ?, ?)`,
|
|
67
|
+
args: [
|
|
68
|
+
id,
|
|
69
|
+
input.fromAgent,
|
|
70
|
+
input.targetAgent,
|
|
71
|
+
input.targetProject ?? null,
|
|
72
|
+
targetDevice,
|
|
73
|
+
sessionScope,
|
|
74
|
+
input.content,
|
|
75
|
+
input.priority ?? "normal",
|
|
76
|
+
input.founderId ?? null,
|
|
77
|
+
now
|
|
78
|
+
]
|
|
79
|
+
});
|
|
80
|
+
recordOrchestrationEventBestEffort({
|
|
81
|
+
eventType: "message.sent",
|
|
82
|
+
source: "messaging.sendMessage",
|
|
83
|
+
agentId: input.fromAgent,
|
|
84
|
+
payload: { targetAgent: input.targetAgent, priority: input.priority ?? "normal", targetDevice },
|
|
85
|
+
sessionScope: sessionScope ?? null
|
|
86
|
+
});
|
|
87
|
+
try {
|
|
88
|
+
if (targetDevice !== "local") {
|
|
89
|
+
await deliverCrossMachineMessage(id, targetDevice);
|
|
90
|
+
} else {
|
|
91
|
+
await deliverLocalMessage(id);
|
|
92
|
+
}
|
|
93
|
+
} catch {
|
|
94
|
+
}
|
|
95
|
+
const sentScope = strictSessionScopeFilter(sessionScope);
|
|
96
|
+
const result = await client.execute({
|
|
97
|
+
sql: `SELECT * FROM messages WHERE id = ?${sentScope.sql}`,
|
|
98
|
+
args: [id, ...sentScope.args]
|
|
99
|
+
});
|
|
100
|
+
return rowToMessage(result.rows[0]);
|
|
101
|
+
}
|
|
102
|
+
var _wsClientSend = null;
|
|
103
|
+
function setWsClientSend(fn) {
|
|
104
|
+
_wsClientSend = fn;
|
|
105
|
+
}
|
|
106
|
+
async function deliverCrossMachineMessage(messageId, targetDevice) {
|
|
107
|
+
const client = getClient();
|
|
108
|
+
const result = await client.execute({
|
|
109
|
+
sql: "SELECT * FROM messages WHERE id = ?",
|
|
110
|
+
args: [messageId]
|
|
111
|
+
});
|
|
112
|
+
if (result.rows.length === 0) return false;
|
|
113
|
+
const msg = rowToMessage(result.rows[0]);
|
|
114
|
+
if (msg.status !== "pending") return false;
|
|
115
|
+
if (!_wsClientSend) {
|
|
116
|
+
return false;
|
|
117
|
+
}
|
|
118
|
+
const payload = JSON.stringify({
|
|
119
|
+
id: msg.id,
|
|
120
|
+
fromAgent: msg.fromAgent,
|
|
121
|
+
targetAgent: msg.targetAgent,
|
|
122
|
+
targetProject: msg.targetProject,
|
|
123
|
+
sessionScope: msg.sessionScope,
|
|
124
|
+
content: msg.content,
|
|
125
|
+
priority: msg.priority,
|
|
126
|
+
createdAt: msg.createdAt
|
|
127
|
+
});
|
|
128
|
+
const sent = _wsClientSend(targetDevice, payload);
|
|
129
|
+
if (sent) {
|
|
130
|
+
await client.execute({
|
|
131
|
+
sql: "UPDATE messages SET status = 'synced' WHERE id = ?",
|
|
132
|
+
args: [messageId]
|
|
133
|
+
});
|
|
134
|
+
return true;
|
|
135
|
+
}
|
|
136
|
+
return false;
|
|
137
|
+
}
|
|
138
|
+
async function deliverLocalMessage(messageId) {
|
|
139
|
+
const client = getClient();
|
|
140
|
+
const result = await client.execute({
|
|
141
|
+
sql: "SELECT * FROM messages WHERE id = ?",
|
|
142
|
+
args: [messageId]
|
|
143
|
+
});
|
|
144
|
+
if (result.rows.length === 0) return false;
|
|
145
|
+
const msg = rowToMessage(result.rows[0]);
|
|
146
|
+
if (msg.status !== "pending") return false;
|
|
147
|
+
const targetAgent = msg.targetAgent;
|
|
148
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
149
|
+
try {
|
|
150
|
+
const exeSession = resolveExeSession();
|
|
151
|
+
if (!exeSession) {
|
|
152
|
+
throw new Error("No coordinator session found");
|
|
153
|
+
}
|
|
154
|
+
process.stderr.write(`[messaging] ENTER deliverLocal: target=${targetAgent} exeSession=${exeSession}
|
|
155
|
+
`);
|
|
156
|
+
const { isCoordinatorName: isCoord } = await import("./lib/employees.js");
|
|
157
|
+
const isCOO = isCoord(targetAgent);
|
|
158
|
+
process.stderr.write(`[messaging] COO check: targetAgent="${targetAgent}" isCOO=${isCOO}
|
|
159
|
+
`);
|
|
160
|
+
const sessionName = isCOO ? exeSession : employeeSessionName(targetAgent, exeSession);
|
|
161
|
+
process.stderr.write(`[messaging] delivery attempt: target=${targetAgent} session=${sessionName} isCOO=${isCOO}
|
|
162
|
+
`);
|
|
163
|
+
if (!isCOO && !isEmployeeAlive(sessionName)) {
|
|
164
|
+
throw new Error("Session not running \u2014 message stays queued");
|
|
165
|
+
}
|
|
166
|
+
const unreadCount = await countUnreadMessages(targetAgent, msg.sessionScope);
|
|
167
|
+
if (unreadCount <= 0) {
|
|
168
|
+
return false;
|
|
169
|
+
}
|
|
170
|
+
const intercomResult = sendIntercom(sessionName, { force: true, reason: "message" });
|
|
171
|
+
process.stderr.write(`[messaging] intercom result for ${sessionName}: ${intercomResult}
|
|
172
|
+
`);
|
|
173
|
+
await client.execute({
|
|
174
|
+
sql: "UPDATE messages SET status = 'delivered', delivered_at = ? WHERE id = ?",
|
|
175
|
+
args: [now, messageId]
|
|
176
|
+
});
|
|
177
|
+
recordOrchestrationEventBestEffort({
|
|
178
|
+
eventType: "message.delivered",
|
|
179
|
+
source: "messaging.deliverLocalMessage",
|
|
180
|
+
agentId: msg.targetAgent,
|
|
181
|
+
payload: { fromAgent: msg.fromAgent, priority: msg.priority }
|
|
182
|
+
});
|
|
183
|
+
return true;
|
|
184
|
+
} catch (err) {
|
|
185
|
+
process.stderr.write(
|
|
186
|
+
`[messaging] delivery failed for message ${messageId}: ${err instanceof Error ? err.message : String(err)}
|
|
187
|
+
`
|
|
188
|
+
);
|
|
189
|
+
const newRetryCount = msg.retryCount + 1;
|
|
190
|
+
if (newRetryCount >= MAX_RETRIES) {
|
|
191
|
+
await markFailed(messageId, "session unavailable after 10 retries", msg.sessionScope);
|
|
192
|
+
} else {
|
|
193
|
+
await client.execute({
|
|
194
|
+
sql: "UPDATE messages SET retry_count = ? WHERE id = ?",
|
|
195
|
+
args: [newRetryCount, messageId]
|
|
196
|
+
});
|
|
197
|
+
}
|
|
198
|
+
return false;
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
async function getPendingMessages(targetAgent, sessionScope) {
|
|
202
|
+
const client = getClient();
|
|
203
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
204
|
+
const result = await client.execute({
|
|
205
|
+
sql: `SELECT * FROM messages
|
|
206
|
+
WHERE target_agent = ? AND status IN ('pending', 'delivered')${scope.sql}
|
|
207
|
+
ORDER BY id`,
|
|
208
|
+
args: [targetAgent, ...scope.args]
|
|
209
|
+
});
|
|
210
|
+
return result.rows.map((row) => rowToMessage(row));
|
|
211
|
+
}
|
|
212
|
+
async function markRead(messageId, sessionScope) {
|
|
213
|
+
const client = getClient();
|
|
214
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
215
|
+
await client.execute({
|
|
216
|
+
sql: `UPDATE messages SET status = 'read'
|
|
217
|
+
WHERE id = ? AND status IN ('pending', 'delivered')${scope.sql}`,
|
|
218
|
+
args: [messageId, ...scope.args]
|
|
219
|
+
});
|
|
220
|
+
recordOrchestrationEventBestEffort({
|
|
221
|
+
eventType: "message.read",
|
|
222
|
+
source: "messaging.markRead",
|
|
223
|
+
payload: { messageId }
|
|
224
|
+
});
|
|
225
|
+
}
|
|
226
|
+
async function markAcknowledged(messageId, sessionScope) {
|
|
227
|
+
const client = getClient();
|
|
228
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
229
|
+
await client.execute({
|
|
230
|
+
sql: `UPDATE messages SET status = 'acknowledged', processed_at = ?
|
|
231
|
+
WHERE id = ? AND status = 'read'${scope.sql}`,
|
|
232
|
+
args: [(/* @__PURE__ */ new Date()).toISOString(), messageId, ...scope.args]
|
|
233
|
+
});
|
|
234
|
+
}
|
|
235
|
+
async function markProcessed(messageId, sessionScope) {
|
|
236
|
+
const client = getClient();
|
|
237
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
238
|
+
await client.execute({
|
|
239
|
+
sql: `UPDATE messages SET status = 'processed', processed_at = ?
|
|
240
|
+
WHERE id = ?${scope.sql}`,
|
|
241
|
+
args: [(/* @__PURE__ */ new Date()).toISOString(), messageId, ...scope.args]
|
|
242
|
+
});
|
|
243
|
+
}
|
|
244
|
+
async function getMessageStatus(messageId, sessionScope) {
|
|
245
|
+
const client = getClient();
|
|
246
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
247
|
+
const result = await client.execute({
|
|
248
|
+
sql: `SELECT status FROM messages WHERE id = ?${scope.sql}`,
|
|
249
|
+
args: [messageId, ...scope.args]
|
|
250
|
+
});
|
|
251
|
+
return result.rows[0]?.status ?? null;
|
|
252
|
+
}
|
|
253
|
+
async function getUnacknowledgedMessages(targetAgent, sessionScope) {
|
|
254
|
+
const client = getClient();
|
|
255
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
256
|
+
const result = await client.execute({
|
|
257
|
+
sql: `SELECT * FROM messages
|
|
258
|
+
WHERE target_agent = ? AND status IN ('pending', 'delivered', 'read')${scope.sql}
|
|
259
|
+
ORDER BY id`,
|
|
260
|
+
args: [targetAgent, ...scope.args]
|
|
261
|
+
});
|
|
262
|
+
return result.rows.map((row) => rowToMessage(row));
|
|
263
|
+
}
|
|
264
|
+
async function getReadMessages(targetAgent, sessionScope) {
|
|
265
|
+
const client = getClient();
|
|
266
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
267
|
+
const result = await client.execute({
|
|
268
|
+
sql: `SELECT * FROM messages
|
|
269
|
+
WHERE target_agent = ? AND status = 'read'${scope.sql}
|
|
270
|
+
ORDER BY id`,
|
|
271
|
+
args: [targetAgent, ...scope.args]
|
|
272
|
+
});
|
|
273
|
+
return result.rows.map((row) => rowToMessage(row));
|
|
274
|
+
}
|
|
275
|
+
async function markFailed(messageId, reason, sessionScope) {
|
|
276
|
+
const client = getClient();
|
|
277
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
278
|
+
await client.execute({
|
|
279
|
+
sql: `UPDATE messages SET status = 'failed', failed_at = ?, failure_reason = ?
|
|
280
|
+
WHERE id = ?${scope.sql}`,
|
|
281
|
+
args: [(/* @__PURE__ */ new Date()).toISOString(), reason, messageId, ...scope.args]
|
|
282
|
+
});
|
|
283
|
+
}
|
|
284
|
+
async function getFailedMessages(sessionScope) {
|
|
285
|
+
const client = getClient();
|
|
286
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
287
|
+
const result = await client.execute({
|
|
288
|
+
sql: `SELECT * FROM messages WHERE status = 'failed'${scope.sql} ORDER BY created_at DESC`,
|
|
289
|
+
args: [...scope.args]
|
|
290
|
+
});
|
|
291
|
+
return result.rows.map((row) => rowToMessage(row));
|
|
292
|
+
}
|
|
293
|
+
async function retryPendingMessages(sessionScope) {
|
|
294
|
+
const client = getClient();
|
|
295
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
296
|
+
const result = await client.execute({
|
|
297
|
+
sql: `SELECT * FROM messages
|
|
298
|
+
WHERE status = 'pending' AND retry_count < ?${scope.sql}
|
|
299
|
+
ORDER BY id`,
|
|
300
|
+
args: [MAX_RETRIES, ...scope.args]
|
|
301
|
+
});
|
|
302
|
+
let delivered = 0;
|
|
303
|
+
for (const row of result.rows) {
|
|
304
|
+
const msg = rowToMessage(row);
|
|
305
|
+
try {
|
|
306
|
+
const success = await deliverLocalMessage(msg.id);
|
|
307
|
+
if (success) delivered++;
|
|
308
|
+
} catch {
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
return delivered;
|
|
312
|
+
}
|
|
313
|
+
async function sendTypedMessage(envelope, targetAgent, opts) {
|
|
314
|
+
return sendMessage({
|
|
315
|
+
fromAgent: envelope.from,
|
|
316
|
+
targetAgent,
|
|
317
|
+
targetProject: opts?.targetProject,
|
|
318
|
+
targetDevice: opts?.targetDevice,
|
|
319
|
+
content: serializeMessage(envelope),
|
|
320
|
+
priority: opts?.priority,
|
|
321
|
+
sessionScope: opts?.sessionScope
|
|
322
|
+
});
|
|
323
|
+
}
|
|
324
|
+
function parseIncomingMessage(content) {
|
|
325
|
+
return parseMessage(content);
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
export {
|
|
329
|
+
countUnreadMessages,
|
|
330
|
+
sendMessage,
|
|
331
|
+
setWsClientSend,
|
|
332
|
+
deliverLocalMessage,
|
|
333
|
+
getPendingMessages,
|
|
334
|
+
markRead,
|
|
335
|
+
markAcknowledged,
|
|
336
|
+
markProcessed,
|
|
337
|
+
getMessageStatus,
|
|
338
|
+
getUnacknowledgedMessages,
|
|
339
|
+
getReadMessages,
|
|
340
|
+
markFailed,
|
|
341
|
+
getFailedMessages,
|
|
342
|
+
retryPendingMessages,
|
|
343
|
+
sendTypedMessage,
|
|
344
|
+
parseIncomingMessage
|
|
345
|
+
};
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
import {
|
|
2
|
+
resolveAnthropicAuth
|
|
3
|
+
} from "./chunk-EJIF4FNT.js";
|
|
4
|
+
|
|
5
|
+
// src/lib/retrieval-health.ts
|
|
6
|
+
async function getRetrievalHealth() {
|
|
7
|
+
let embedDaemon = "unavailable";
|
|
8
|
+
try {
|
|
9
|
+
const { isDaemonAlive } = await import("./mcp-health-WDOB6XUB.js");
|
|
10
|
+
const { alive } = isDaemonAlive();
|
|
11
|
+
if (alive) {
|
|
12
|
+
try {
|
|
13
|
+
const { pingDaemon } = await import("./lib/exe-daemon-client.js");
|
|
14
|
+
const health = await pingDaemon();
|
|
15
|
+
if (health) {
|
|
16
|
+
const worker = health.embedWorker;
|
|
17
|
+
const embeddingField = health.embedding;
|
|
18
|
+
if (worker?.disabledByDesign || embeddingField === "disabled") {
|
|
19
|
+
embedDaemon = "disabled";
|
|
20
|
+
} else if (worker) {
|
|
21
|
+
embedDaemon = worker.forked && worker.ready && worker.pid != null ? "running" : "stopped";
|
|
22
|
+
} else if (embeddingField === "ok") {
|
|
23
|
+
embedDaemon = "running";
|
|
24
|
+
} else {
|
|
25
|
+
embedDaemon = "stopped";
|
|
26
|
+
}
|
|
27
|
+
} else {
|
|
28
|
+
embedDaemon = "stopped";
|
|
29
|
+
}
|
|
30
|
+
} catch {
|
|
31
|
+
embedDaemon = "stopped";
|
|
32
|
+
}
|
|
33
|
+
} else {
|
|
34
|
+
embedDaemon = "unavailable";
|
|
35
|
+
}
|
|
36
|
+
} catch {
|
|
37
|
+
embedDaemon = "unavailable";
|
|
38
|
+
}
|
|
39
|
+
let configuredSearchMode = "hybrid";
|
|
40
|
+
let selfQueryRouterEnabled = true;
|
|
41
|
+
try {
|
|
42
|
+
const { loadConfigSync } = await import("./lib/config.js");
|
|
43
|
+
const cfg = loadConfigSync();
|
|
44
|
+
configuredSearchMode = cfg.searchMode ?? "hybrid";
|
|
45
|
+
selfQueryRouterEnabled = cfg.selfQueryRouter ?? true;
|
|
46
|
+
} catch {
|
|
47
|
+
}
|
|
48
|
+
let actualSearchMode;
|
|
49
|
+
if (configuredSearchMode === "fts") {
|
|
50
|
+
actualSearchMode = "fts+graph";
|
|
51
|
+
} else if (embedDaemon === "running") {
|
|
52
|
+
actualSearchMode = "hybrid (vector+fts+graph)";
|
|
53
|
+
} else {
|
|
54
|
+
actualSearchMode = "fts+graph (degraded from hybrid)";
|
|
55
|
+
}
|
|
56
|
+
let queryRouter;
|
|
57
|
+
if (!selfQueryRouterEnabled) {
|
|
58
|
+
queryRouter = "disabled";
|
|
59
|
+
} else if (embedDaemon === "running" || embedDaemon === "disabled") {
|
|
60
|
+
queryRouter = resolveAnthropicAuth() ? "active" : "fallback";
|
|
61
|
+
} else {
|
|
62
|
+
queryRouter = "fallback";
|
|
63
|
+
}
|
|
64
|
+
let totalMemories = 0;
|
|
65
|
+
let memoriesWithVectors = 0;
|
|
66
|
+
let entities = 0;
|
|
67
|
+
let relationships = 0;
|
|
68
|
+
try {
|
|
69
|
+
const { getClient } = await import("./lib/database.js");
|
|
70
|
+
const client = getClient();
|
|
71
|
+
const [totalRes, vectorRes] = await Promise.all([
|
|
72
|
+
client.execute("SELECT COUNT(*) as cnt FROM memories WHERE status IS NULL OR status != 'deleted'"),
|
|
73
|
+
client.execute("SELECT COUNT(*) as cnt FROM memories WHERE vector IS NOT NULL AND (status IS NULL OR status != 'deleted')")
|
|
74
|
+
]);
|
|
75
|
+
totalMemories = Number(totalRes.rows[0]?.cnt ?? 0);
|
|
76
|
+
memoriesWithVectors = Number(vectorRes.rows[0]?.cnt ?? 0);
|
|
77
|
+
try {
|
|
78
|
+
const { getGraphStats } = await import("./graph-query-AP5R6ZHO.js");
|
|
79
|
+
const stats = await getGraphStats(client);
|
|
80
|
+
entities = stats.entities;
|
|
81
|
+
relationships = stats.relationships;
|
|
82
|
+
} catch {
|
|
83
|
+
}
|
|
84
|
+
} catch {
|
|
85
|
+
}
|
|
86
|
+
const staleCount = totalMemories - memoriesWithVectors;
|
|
87
|
+
const coveragePct = totalMemories > 0 ? Math.round(memoriesWithVectors / totalMemories * 100) : null;
|
|
88
|
+
let dbRestore = { hasEvents: false, count: 0, last: null, quarantineFiles: 0 };
|
|
89
|
+
try {
|
|
90
|
+
const { getDbRestoreHealth } = await import("./db-restore-events-GNZS42YO.js");
|
|
91
|
+
dbRestore = getDbRestoreHealth();
|
|
92
|
+
} catch {
|
|
93
|
+
}
|
|
94
|
+
const summary = {
|
|
95
|
+
embedDaemon,
|
|
96
|
+
searchMode: {
|
|
97
|
+
configured: configuredSearchMode,
|
|
98
|
+
actual: actualSearchMode
|
|
99
|
+
},
|
|
100
|
+
queryRouter,
|
|
101
|
+
vectorBackfill: {
|
|
102
|
+
totalMemories,
|
|
103
|
+
memoriesWithVectors,
|
|
104
|
+
staleCount,
|
|
105
|
+
coveragePct
|
|
106
|
+
},
|
|
107
|
+
graph: {
|
|
108
|
+
entities,
|
|
109
|
+
relationships
|
|
110
|
+
},
|
|
111
|
+
dbRestore
|
|
112
|
+
};
|
|
113
|
+
return {
|
|
114
|
+
...summary,
|
|
115
|
+
degradations: computeDegradations(summary)
|
|
116
|
+
};
|
|
117
|
+
}
|
|
118
|
+
function computeDegradations(h) {
|
|
119
|
+
const out = [];
|
|
120
|
+
const isRuntimeDowngrade = h.searchMode.actual.includes("degraded");
|
|
121
|
+
if (isRuntimeDowngrade) {
|
|
122
|
+
out.push({
|
|
123
|
+
code: "retrieval_downgraded",
|
|
124
|
+
severity: "warn",
|
|
125
|
+
message: `Retrieval running as "${h.searchMode.actual}" (configured: "${h.searchMode.configured}")`,
|
|
126
|
+
remediation: h.embedDaemon === "disabled" ? "Embeddings are disabled. Enable them in settings if you want vector retrieval." : "Embed daemon is not running. Start the daemon / embed worker to restore hybrid retrieval."
|
|
127
|
+
});
|
|
128
|
+
} else if (h.searchMode.configured === "fts") {
|
|
129
|
+
out.push({
|
|
130
|
+
code: "embeddings_off",
|
|
131
|
+
severity: "info",
|
|
132
|
+
message: "Search mode is FTS-only by configuration (keyword + graph; no vector retrieval)",
|
|
133
|
+
remediation: 'Switch searchMode to "hybrid" in settings to enable vector retrieval.'
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
if (h.queryRouter === "fallback") {
|
|
137
|
+
out.push({
|
|
138
|
+
code: "query_router_fallback",
|
|
139
|
+
severity: "info",
|
|
140
|
+
message: "Self-query-router using rule-based fallback (LLM routing is an optional enrichment; keyword + graph retrieval unaffected)",
|
|
141
|
+
remediation: "To enable LLM-based query routing, provide an Anthropic credential: set ANTHROPIC_API_KEY, or ANTHROPIC_AUTH_TOKEN if you use Claude Code / OpenCode / a subscription-OAuth host."
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
if (h.dbRestore.hasEvents) {
|
|
145
|
+
const last = h.dbRestore.last;
|
|
146
|
+
if (last && (last.kind === "restore-failed" || last.kind === "zeroed-no-wal")) {
|
|
147
|
+
out.push({
|
|
148
|
+
code: "db_corruption_unrecovered",
|
|
149
|
+
severity: "critical",
|
|
150
|
+
message: `DB corruption could not be auto-recovered: ${last.detail ?? last.kind} (${last.at})`,
|
|
151
|
+
remediation: "Restore from a known-good backup (`exe-os cloud reupload` or restore a backup); preserved corrupt DB is in ~/.exe-os/."
|
|
152
|
+
});
|
|
153
|
+
} else if (last && (last.kind === "auto-restore" || last.kind === "wal-recovery")) {
|
|
154
|
+
out.push({
|
|
155
|
+
code: "db_auto_restored",
|
|
156
|
+
severity: "warn",
|
|
157
|
+
message: last.kind === "auto-restore" ? `DB was auto-restored from backup${last.restoredFrom ? ` (${last.restoredFrom})` : ""}${typeof last.hoursLost === "number" ? `, ~${last.hoursLost}h data-loss window` : ""} at ${last.at}` : `DB was reconstructed from WAL after a 0-byte corruption at ${last.at}`,
|
|
158
|
+
remediation: "Recent memories may be missing. Run a cloud sync to reconcile and verify recent data."
|
|
159
|
+
});
|
|
160
|
+
} else if (h.dbRestore.quarantineFiles > 0) {
|
|
161
|
+
out.push({
|
|
162
|
+
code: "db_corruption_quarantine",
|
|
163
|
+
severity: "warn",
|
|
164
|
+
message: `${h.dbRestore.quarantineFiles} quarantined corrupt DB file(s) present in ~/.exe-os/ (prior corruption event)`,
|
|
165
|
+
remediation: "Verify data integrity; remove quarantined .corrupt-*/.zeroed-* files once you've confirmed no data is needed."
|
|
166
|
+
});
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
return out;
|
|
170
|
+
}
|
|
171
|
+
var SEVERITY_ICON = {
|
|
172
|
+
info: "\u2139\uFE0F",
|
|
173
|
+
// ℹ️
|
|
174
|
+
warn: "\u{1F7E0}",
|
|
175
|
+
// 🟠
|
|
176
|
+
critical: "\u{1F534}"
|
|
177
|
+
// 🔴
|
|
178
|
+
};
|
|
179
|
+
function formatDegradations(degradations, heading = "Degradations") {
|
|
180
|
+
const lines = [];
|
|
181
|
+
lines.push(`### ${heading}`);
|
|
182
|
+
if (degradations.length === 0) {
|
|
183
|
+
lines.push("\u{1F7E2} None \u2014 all retrieval and storage subsystems nominal.");
|
|
184
|
+
return lines.join("\n");
|
|
185
|
+
}
|
|
186
|
+
const order = { critical: 0, warn: 1, info: 2 };
|
|
187
|
+
const sorted = [...degradations].sort((a, b) => order[a.severity] - order[b.severity]);
|
|
188
|
+
for (const d of sorted) {
|
|
189
|
+
lines.push(`${SEVERITY_ICON[d.severity]} ${d.message}`);
|
|
190
|
+
if (d.remediation) lines.push(` \u2192 ${d.remediation}`);
|
|
191
|
+
}
|
|
192
|
+
return lines.join("\n");
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
export {
|
|
196
|
+
getRetrievalHealth,
|
|
197
|
+
computeDegradations,
|
|
198
|
+
formatDegradations
|
|
199
|
+
};
|