@askexenow/exe-os 0.9.265 → 0.9.266
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/.env.example +4 -2
- package/deploy/compose/docker-compose.yml +6 -4
- package/deploy/compose/generate-env.ts +5 -2
- package/dist/active-agent-2XJJZFEW.js +25 -0
- package/dist/active-agent-JZ4PHIAK.js +26 -0
- package/dist/agentic-ontology-J4B525DT.js +25 -0
- package/dist/backfill-metadata-TWEC6BJD.js +597 -0
- package/dist/backfill-metadata-VGZLCJWZ.js +597 -0
- package/dist/behaviors-BPIVFAGY.js +25 -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 +4 -4
- package/dist/bin/backfill-responses.js +4 -4
- package/dist/bin/backfill-vectors.js +5 -5
- package/dist/bin/bulk-sync-postgres.js +6 -6
- package/dist/bin/cc-doctor.js +6 -1
- package/dist/bin/cleanup-stale-review-tasks.js +9 -9
- package/dist/bin/cli.js +15 -15
- package/dist/bin/exe-agent-config.js +2 -2
- package/dist/bin/exe-agent.js +4 -4
- package/dist/bin/exe-assign.js +5 -5
- package/dist/bin/exe-boot.js +16 -16
- package/dist/bin/exe-call.js +4 -4
- package/dist/bin/exe-cloud.js +4 -4
- package/dist/bin/exe-dispatch.js +9 -9
- package/dist/bin/exe-doctor.js +1 -1
- 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 -1
- package/dist/bin/exe-heartbeat.js +9 -9
- package/dist/bin/exe-kill.js +12 -12
- package/dist/bin/exe-launch-agent.js +16 -16
- package/dist/bin/exe-new-employee.js +6 -6
- package/dist/bin/exe-pending-messages.js +10 -10
- package/dist/bin/exe-pending-notifications.js +9 -9
- package/dist/bin/exe-pending-reviews.js +9 -9
- package/dist/bin/exe-rename.js +4 -4
- package/dist/bin/exe-review.js +11 -11
- package/dist/bin/exe-search.js +5 -5
- package/dist/bin/exe-session-cleanup.js +14 -14
- package/dist/bin/exe-settings.js +4 -4
- package/dist/bin/exe-start-codex.js +11 -11
- package/dist/bin/exe-start-opencode.js +8 -8
- package/dist/bin/exe-status.js +10 -10
- package/dist/bin/exe-team.js +3 -3
- package/dist/bin/git-sweep.js +10 -10
- 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.js +6 -6
- 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 +9 -9
- package/dist/bin/setup.js +1 -1
- package/dist/bin/shard-migrate.js +4 -4
- package/dist/capacity-monitor-3QORC56J.js +49 -0
- package/dist/capacity-monitor-FWRBCZCZ.js +49 -0
- package/dist/catchup-brief-242H63OP.js +151 -0
- package/dist/catchup-brief-T7W2ARRB.js +151 -0
- package/dist/catchup-brief-VJDLXJHJ.js +151 -0
- package/dist/chunk-2FDYA3GS.js +75 -0
- package/dist/chunk-2V33IVZF.js +1148 -0
- package/dist/chunk-2W5NQOQJ.js +333 -0
- package/dist/chunk-32QR5EHE.js +244 -0
- package/dist/chunk-3DBROKA6.js +382 -0
- package/dist/chunk-3DS2YHAT.js +204 -0
- package/dist/chunk-3O2ZD7YV.js +227 -0
- package/dist/chunk-43O2PZH7.js +214 -0
- package/dist/chunk-4AHL6I7X.js +284 -0
- package/dist/chunk-5GBPXJIH.js +50 -0
- package/dist/chunk-5IXN2532.js +58 -0
- package/dist/chunk-5SH5TR45.js +731 -0
- package/dist/chunk-5XGMTWKW.js +128 -0
- package/dist/chunk-6AQAILCS.js +13696 -0
- package/dist/chunk-72KQJZBH.js +382 -0
- package/dist/chunk-7B2AENJ3.js +1345 -0
- package/dist/chunk-7GKMYFVR.js +551 -0
- package/dist/chunk-7HUGVJHW.js +1021 -0
- package/dist/chunk-7HWBGXDV.js +333 -0
- package/dist/chunk-7ROGPDW6.js +197 -0
- package/dist/chunk-7ZMLNURG.js +513 -0
- package/dist/chunk-A3G4UXNA.js +280 -0
- package/dist/chunk-A4T6U3XZ.js +371 -0
- package/dist/chunk-ADJNRFEK.js +13696 -0
- package/dist/chunk-AVJTW67M.js +513 -0
- package/dist/chunk-B6RC7DAC.js +1090 -0
- package/dist/chunk-BAOJLZVC.js +244 -0
- package/dist/chunk-BCUMPA7J.js +159 -0
- package/dist/chunk-BH2ST7KK.js +183 -0
- package/dist/chunk-C4BIYP5C.js +129 -0
- package/dist/chunk-CBUZG6OC.js +89 -0
- package/dist/chunk-CGQRWCCP.js +81 -0
- package/dist/chunk-CLX4WQHT.js +214 -0
- package/dist/chunk-D4CQZL46.js +76 -0
- package/dist/chunk-DEBLYXA2.js +56 -0
- package/dist/chunk-EDNUE5L3.js +38 -0
- package/dist/chunk-ESNHLTLC.js +1073 -0
- package/dist/chunk-FICUAZEA.js +128 -0
- package/dist/chunk-FP6U5PSY.js +818 -0
- package/dist/chunk-G55UTKFR.js +89 -0
- package/dist/chunk-GIJUXLK5.js +330 -0
- package/dist/chunk-GVSSEGIV.js +348 -0
- package/dist/chunk-GYOOZ725.js +330 -0
- package/dist/chunk-HCULUL4D.js +447 -0
- package/dist/chunk-HDRQKH7R.js +1051 -0
- package/dist/chunk-HHBN7Y7R.js +197 -0
- package/dist/chunk-IQBXQ7GF.js +1073 -0
- package/dist/chunk-ITF6GEJI.js +2091 -0
- package/dist/chunk-IXHVSWLM.js +58 -0
- package/dist/chunk-J3ZFONT4.js +1921 -0
- package/dist/chunk-JEOFG274.js +394 -0
- package/dist/chunk-JH7ISOIJ.js +33 -0
- package/dist/chunk-JVRMAVPN.js +731 -0
- package/dist/chunk-KBYRYSXT.js +3951 -0
- package/dist/chunk-KECNXWD6.js +1051 -0
- package/dist/chunk-KIYCHS35.js +128 -0
- package/dist/chunk-KUP3D6G2.js +1345 -0
- package/dist/chunk-KXS5HHDH.js +54 -0
- package/dist/chunk-LVZW4WQY.js +221 -0
- package/dist/chunk-LWOCCDSB.js +240 -0
- package/dist/chunk-M4HRKRSE.js +30 -0
- package/dist/chunk-MPG5DR3K.js +244 -0
- package/dist/chunk-NOG2LVBA.js +159 -0
- package/dist/chunk-NP3CYOEY.js +290 -0
- package/dist/chunk-OVQOKLYA.js +210 -0
- package/dist/chunk-OW26ML36.js +1148 -0
- package/dist/chunk-P2DOIWQG.js +260 -0
- package/dist/chunk-P2QTTEYP.js +3951 -0
- package/dist/chunk-P5IKFL7B.js +85 -0
- package/dist/chunk-PD2LUPHD.js +377 -0
- package/dist/chunk-PF5EFIAH.js +818 -0
- package/dist/chunk-PK3NFO76.js +551 -0
- package/dist/chunk-QKMFMJTU.js +171 -0
- package/dist/chunk-R77M74QM.js +13696 -0
- package/dist/chunk-RHZ2CRJU.js +3556 -0
- package/dist/chunk-RLR4LVO5.js +1090 -0
- package/dist/chunk-RMDDSWFH.js +495 -0
- package/dist/chunk-RPLEZ74B.js +95 -0
- package/dist/chunk-RXMVPGH2.js +346 -0
- package/dist/chunk-S2ARWNS6.js +135 -0
- package/dist/chunk-SHPIATX7.js +90 -0
- package/dist/chunk-SQUZ3TI2.js +668 -0
- package/dist/chunk-SUGBKORD.js +1148 -0
- package/dist/chunk-T2XD7XL2.js +346 -0
- package/dist/chunk-TAIYLROT.js +127 -0
- package/dist/chunk-TCD4XCBS.js +495 -0
- package/dist/chunk-TSBMSHMM.js +163 -0
- package/dist/chunk-U776PBU6.js +488 -0
- package/dist/chunk-UTVNIAOF.js +284 -0
- package/dist/chunk-VF6VT4MR.js +70 -0
- package/dist/chunk-VZXLYUKS.js +171 -0
- package/dist/chunk-WIAHPJDS.js +58 -0
- package/dist/chunk-WM5ELLT4.js +54 -0
- package/dist/chunk-WUUSPOCC.js +1921 -0
- package/dist/chunk-WXHAMPK4.js +81 -0
- package/dist/chunk-WYNZ23XH.js +262 -0
- package/dist/chunk-WZOQLU2Y.js +127 -0
- package/dist/chunk-X24GBZKR.js +157 -0
- package/dist/chunk-X5FRF5MG.js +13696 -0
- package/dist/chunk-XBPLYFBA.js +204 -0
- package/dist/chunk-XDGKFXFW.js +363 -0
- package/dist/chunk-XIT6ONPY.js +630 -0
- package/dist/chunk-XNRJ5JHU.js +377 -0
- package/dist/chunk-XREWTPZM.js +123 -0
- package/dist/chunk-XURCA5VD.js +297 -0
- package/dist/chunk-Y55VFZJB.js +50 -0
- package/dist/chunk-YIH2YMWP.js +76 -0
- package/dist/chunk-YZMCCT73.js +97 -0
- package/dist/chunk-ZTP3VV23.js +447 -0
- package/dist/co-activation-JOYR73H2.js +72 -0
- package/dist/co-occurrence-G7RRJ3DB.js +74 -0
- package/dist/core-memory-6KK55Z65.js +110 -0
- package/dist/core-memory-QRS6GBXL.js +110 -0
- package/dist/crdt-sync-QWK65Z5A.js +33 -0
- package/dist/crm-webhook-BNUS3USD.js +10 -0
- package/dist/crm-webhook-ZTIRFVAH.js +10 -0
- package/dist/cto-delegation-gate-DNZJGBS7.js +206 -0
- package/dist/cto-delegation-gate-YI3LK7VY.js +206 -0
- package/dist/daemon-orchestration-7LVOHYKA.js +135 -0
- package/dist/daemon-orchestration-SDQNWI3K.js +135 -0
- package/dist/db-backup-5ENHDNZI.js +33 -0
- package/dist/dreaming-PGJOCGAM.js +32 -0
- package/dist/dreaming-UNHHGOU3.js +32 -0
- package/dist/exe-drift-QQJMUQXA.js +68 -0
- package/dist/exe-export-BEZZSQB4.js +73 -0
- package/dist/exe-export-TQGI2VBT.js +73 -0
- package/dist/exe-import-2SR2UP3D.js +76 -0
- package/dist/exe-import-73Q4VLFK.js +76 -0
- package/dist/exe-key-6JHEINOM.js +579 -0
- package/dist/exe-key-RIRPIFK6.js +579 -0
- package/dist/exe-snapshot-NDMDQJOH.js +164 -0
- package/dist/exe-snapshot-PVCBPMPJ.js +164 -0
- package/dist/fast-db-init-22OBOPTN.js +7 -0
- package/dist/fast-db-init-M6ZYYGAL.js +7 -0
- package/dist/gateway/index.js +8 -8
- package/dist/git-staleness-4DGYC3RY.js +110 -0
- package/dist/git-task-sweep-CUR7R2ZM.js +40 -0
- package/dist/git-task-sweep-YG3IVGK3.js +40 -0
- package/dist/global-procedures-TYDNKM7V.js +20 -0
- package/dist/graph-auto-extract-I6B5HWTP.js +162 -0
- package/dist/hooks/bug-report-worker.js +11 -11
- package/dist/hooks/codex-stop-task-finalizer.js +11 -11
- package/dist/hooks/commit-complete.js +11 -11
- package/dist/hooks/error-recall.js +6 -6
- package/dist/hooks/exe-heartbeat-hook.js +3 -3
- package/dist/hooks/ingest.js +6 -6
- package/dist/hooks/instructions-loaded.js +4 -4
- package/dist/hooks/manifest.json +19 -19
- package/dist/hooks/notification.js +4 -4
- package/dist/hooks/post-compact.js +10 -10
- package/dist/hooks/post-tool-combined.js +5 -5
- package/dist/hooks/pre-compact.js +11 -11
- package/dist/hooks/pre-tool-use.js +14 -14
- package/dist/hooks/prompt-submit.js +20 -20
- package/dist/hooks/session-end.js +15 -15
- package/dist/hooks/session-start.js +10 -10
- package/dist/hooks/stop.js +14 -14
- package/dist/hooks/subagent-stop.js +10 -10
- package/dist/hooks/summary-worker.js +14 -14
- package/dist/index.js +17 -17
- package/dist/installer-4LP73C3M.js +38 -0
- package/dist/installer-AUVLNS4D.js +296 -0
- package/dist/installer-KCRLQSQH.js +38 -0
- package/dist/installer-KGXYUOIV.js +296 -0
- package/dist/installer-UI6TNJGO.js +342 -0
- package/dist/installer-Z7NPD4PD.js +342 -0
- package/dist/lib/cloud-sync.js +4 -4
- package/dist/lib/consolidation.js +5 -5
- package/dist/lib/database.js +2 -2
- package/dist/lib/db-daemon-client.js +6 -0
- package/dist/lib/db.js +2 -2
- package/dist/lib/employee-templates.js +4 -4
- package/dist/lib/employees.js +2 -2
- package/dist/lib/exe-daemon.js +39 -38
- package/dist/lib/hybrid-search.js +5 -5
- package/dist/lib/identity.js +2 -2
- package/dist/lib/messaging.js +9 -9
- 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 +4 -4
- package/dist/lib/store.js +4 -4
- package/dist/lib/task-router.js +3 -3
- package/dist/lib/tasks.js +10 -10
- package/dist/lib/tmux-routing.js +8 -8
- package/dist/lib/token-spend.js +3 -3
- package/dist/mcp/register-tools.js +53 -52
- package/dist/mcp/server.js +54 -53
- 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 +12 -12
- package/dist/mcp/tools/deactivate-behavior.js +5 -5
- package/dist/mcp/tools/list-reminders.js +4 -4
- package/dist/mcp/tools/list-tasks.js +12 -12
- package/dist/mcp/tools/send-message.js +11 -11
- package/dist/mcp/tools/update-task.js +11 -11
- package/dist/mcp-http-config-LN5ITKGS.js +27 -0
- package/dist/mcp-http-config-V43VHNEH.js +27 -0
- package/dist/memory-cards-S36DCEY4.js +174 -0
- package/dist/memory-graph-extractor-66YL4V4B.js +17 -0
- package/dist/memory-poisoning-defense-MXWE6CTY.js +225 -0
- package/dist/memory-reflection-W676LUX5.js +238 -0
- package/dist/notifications-OFLXZTDZ.js +45 -0
- package/dist/notifications-XAV4V7T5.js +45 -0
- package/dist/orchestration-events-MJLWJ2H2.js +25 -0
- package/dist/orchestrator-AIAKV5ST.js +33 -0
- package/dist/orchestrator-VK7D6I36.js +33 -0
- package/dist/pipeline-router-DXH5QVUH.js +13 -0
- package/dist/pipeline-router-EV6C5S33.js +13 -0
- package/dist/plan-limits-TQMLKZBG.js +26 -0
- package/dist/project-boot-4XTXHDG4.js +299 -0
- package/dist/projection-worker-NAYVNSI4.js +964 -0
- package/dist/reranker-KHZI7HQK.js +19 -0
- package/dist/reranker-UPMS3OX7.js +19 -0
- package/dist/reranker-ZN73HWSB.js +19 -0
- package/dist/review-polling-3UEZSPNT.js +124 -0
- package/dist/review-polling-KBDX3ILL.js +124 -0
- package/dist/runtime/index.js +11 -11
- package/dist/session-events-2CHOLA6D.js +36 -0
- package/dist/session-events-HM22KEHN.js +36 -0
- package/dist/session-kill-telemetry-LUPVYGZX.js +29 -0
- package/dist/session-scope-EPZZZHHJ.js +86 -0
- package/dist/session-scope-ZGE6GHMS.js +86 -0
- package/dist/setup-wizard-KNWH54HW.js +12 -0
- package/dist/skill-refinement-C4GNWV4F.js +157 -0
- package/dist/skill-refinement-CAUIE4YM.js +157 -0
- package/dist/steward-gate-MZ2DV36N.js +13 -0
- package/dist/task-enforcement-6OMXG6EA.js +333 -0
- package/dist/task-enforcement-JFSF6QDD.js +333 -0
- package/dist/task-scope-4HJSCJ7L.js +35 -0
- package/dist/task-scope-7HYE4AS6.js +35 -0
- package/dist/tasks-crud-JJL6MMJF.js +77 -0
- package/dist/tasks-crud-KMIPDHBZ.js +77 -0
- package/dist/tasks-notify-5D6LF52K.js +38 -0
- package/dist/tasks-notify-JNGGXFUY.js +38 -0
- package/dist/tasks-review-N6I5UC7G.js +47 -0
- package/dist/tasks-review-USHAMO5N.js +47 -0
- package/dist/telemetry-upload-BV2M2NND.js +739 -0
- package/dist/telemetry-upload-Y7QPDJW4.js +739 -0
- package/dist/token-budget-2LUZOFO7.js +84 -0
- package/dist/tool-telemetry-5JT6DPVK.js +17 -0
- package/dist/tui/App.js +16 -16
- package/dist/tui-data-DMAKZMGD.js +258 -0
- package/dist/tui-data-X7NI6ZQ3.js +258 -0
- package/dist/wiki-acl-QHGGWX63.js +111 -0
- package/dist/worker-gate-4XKUOFWO.js +21 -0
- package/dist/worker-gate-TPZBOXOA.js +21 -0
- package/dist/workflow-engine-F3URLOPA.js +28 -0
- package/dist/workflow-engine-OFOL446Z.js +28 -0
- package/dist/worktree-V3NVP7MV.js +26 -0
- package/dist/worktree-sweep-2CADL43Y.js +19 -0
- package/package.json +1 -1
- package/release-notes.json +16 -15
|
@@ -0,0 +1,348 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getIdentity,
|
|
3
|
+
loadEmployeesSync
|
|
4
|
+
} from "./chunk-RHZ2CRJU.js";
|
|
5
|
+
|
|
6
|
+
// src/lib/drift-probes.ts
|
|
7
|
+
var DRIFT_THRESHOLD = 80;
|
|
8
|
+
async function getRecentMemories(agentId, limit) {
|
|
9
|
+
try {
|
|
10
|
+
const { getClient, isInitialized } = await import("./lib/database.js");
|
|
11
|
+
if (!isInitialized()) return { count: 0, texts: [] };
|
|
12
|
+
const client = getClient();
|
|
13
|
+
const result = await client.execute({
|
|
14
|
+
sql: `SELECT text FROM memories WHERE agent_id = ? ORDER BY created_at DESC LIMIT ?`,
|
|
15
|
+
args: [agentId, limit]
|
|
16
|
+
});
|
|
17
|
+
const texts = result.rows.map((r) => String(r.text ?? ""));
|
|
18
|
+
return { count: texts.length, texts };
|
|
19
|
+
} catch {
|
|
20
|
+
return { count: 0, texts: [] };
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
async function getDecisionCount(agentId) {
|
|
24
|
+
try {
|
|
25
|
+
const { getClient, isInitialized } = await import("./lib/database.js");
|
|
26
|
+
if (!isInitialized()) return 0;
|
|
27
|
+
const client = getClient();
|
|
28
|
+
const result = await client.execute({
|
|
29
|
+
sql: `SELECT COUNT(*) as cnt FROM memories WHERE agent_id = ? AND memory_type = 'decision'`,
|
|
30
|
+
args: [agentId]
|
|
31
|
+
});
|
|
32
|
+
return Number(result.rows[0]?.cnt ?? 0);
|
|
33
|
+
} catch {
|
|
34
|
+
return 0;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
function probeContinuity(ctx) {
|
|
38
|
+
const memCount = ctx.recentMemoryCount;
|
|
39
|
+
let score;
|
|
40
|
+
let detail;
|
|
41
|
+
if (memCount >= 20) {
|
|
42
|
+
score = 95;
|
|
43
|
+
detail = `${memCount} recent memories \u2014 strong continuity`;
|
|
44
|
+
} else if (memCount >= 10) {
|
|
45
|
+
score = 85;
|
|
46
|
+
detail = `${memCount} recent memories \u2014 adequate continuity`;
|
|
47
|
+
} else if (memCount >= 3) {
|
|
48
|
+
score = 70;
|
|
49
|
+
detail = `${memCount} recent memories \u2014 limited continuity`;
|
|
50
|
+
} else if (memCount >= 1) {
|
|
51
|
+
score = 50;
|
|
52
|
+
detail = `${memCount} recent memories \u2014 weak continuity`;
|
|
53
|
+
} else {
|
|
54
|
+
score = 30;
|
|
55
|
+
detail = "No recent memories \u2014 agent may lack session context";
|
|
56
|
+
}
|
|
57
|
+
if (ctx.storedDecisionCount > 0) {
|
|
58
|
+
score = Math.min(100, score + 5);
|
|
59
|
+
detail += ` (+${ctx.storedDecisionCount} decisions)`;
|
|
60
|
+
}
|
|
61
|
+
return { axis: "continuity", score, detail };
|
|
62
|
+
}
|
|
63
|
+
function probeConsistency(ctx) {
|
|
64
|
+
if (!ctx.identityBody || ctx.recentMemoryTexts.length === 0) {
|
|
65
|
+
return {
|
|
66
|
+
axis: "consistency",
|
|
67
|
+
score: ctx.identityBody ? 80 : 50,
|
|
68
|
+
detail: ctx.identityBody ? "No memories to check against identity" : "No identity file \u2014 cannot assess consistency"
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
const identityTerms = extractKeyTerms(ctx.identityBody);
|
|
72
|
+
if (identityTerms.length === 0) {
|
|
73
|
+
return { axis: "consistency", score: 75, detail: "Identity has no extractable key terms" };
|
|
74
|
+
}
|
|
75
|
+
const memoryText = ctx.recentMemoryTexts.join(" ").toLowerCase();
|
|
76
|
+
let matched = 0;
|
|
77
|
+
for (const term of identityTerms) {
|
|
78
|
+
if (memoryText.includes(term.toLowerCase())) matched++;
|
|
79
|
+
}
|
|
80
|
+
const ratio = matched / identityTerms.length;
|
|
81
|
+
const score = Math.round(50 + ratio * 50);
|
|
82
|
+
const detail = `${matched}/${identityTerms.length} identity terms found in recent memories`;
|
|
83
|
+
return { axis: "consistency", score, detail };
|
|
84
|
+
}
|
|
85
|
+
function probeRoleFidelity(ctx) {
|
|
86
|
+
const flags = [];
|
|
87
|
+
if (!ctx.identityBody) {
|
|
88
|
+
return {
|
|
89
|
+
axis: "role-fidelity",
|
|
90
|
+
score: 40,
|
|
91
|
+
detail: `No identity file for ${ctx.agent.name}`
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
let score = 100;
|
|
95
|
+
const identityLower = ctx.identityBody.toLowerCase();
|
|
96
|
+
const rosterRole = ctx.agent.role.toLowerCase();
|
|
97
|
+
if (!identityLower.includes(rosterRole)) {
|
|
98
|
+
score -= 15;
|
|
99
|
+
flags.push(`Identity does not mention role "${ctx.agent.role}"`);
|
|
100
|
+
}
|
|
101
|
+
if (!identityLower.includes(ctx.agent.name.toLowerCase())) {
|
|
102
|
+
score -= 10;
|
|
103
|
+
flags.push(`Identity does not mention agent name "${ctx.agent.name}"`);
|
|
104
|
+
}
|
|
105
|
+
const hasScopeBoundary = identityLower.includes("do not") || identityLower.includes("don't") || identityLower.includes("not your") || identityLower.includes("outside your") || identityLower.includes("you do not");
|
|
106
|
+
if (!hasScopeBoundary) {
|
|
107
|
+
score -= 10;
|
|
108
|
+
flags.push("Identity lacks explicit scope boundaries (what NOT to do)");
|
|
109
|
+
}
|
|
110
|
+
if (ctx.agent.systemPrompt) {
|
|
111
|
+
const promptTerms = extractKeyTerms(ctx.agent.systemPrompt);
|
|
112
|
+
const identityText = ctx.identityBody.toLowerCase();
|
|
113
|
+
let promptMatched = 0;
|
|
114
|
+
for (const term of promptTerms.slice(0, 20)) {
|
|
115
|
+
if (identityText.includes(term.toLowerCase())) promptMatched++;
|
|
116
|
+
}
|
|
117
|
+
const promptRatio = promptTerms.length > 0 ? promptMatched / Math.min(promptTerms.length, 20) : 1;
|
|
118
|
+
if (promptRatio < 0.3) {
|
|
119
|
+
score -= 15;
|
|
120
|
+
flags.push("Identity content diverges significantly from roster systemPrompt");
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
if (ctx.recentMemoryTexts.length > 5) {
|
|
124
|
+
const roleKeywords = extractRoleKeywords(ctx.agent.role);
|
|
125
|
+
const memoryText = ctx.recentMemoryTexts.join(" ").toLowerCase();
|
|
126
|
+
let domainHits = 0;
|
|
127
|
+
for (const kw of roleKeywords) {
|
|
128
|
+
if (memoryText.includes(kw)) domainHits++;
|
|
129
|
+
}
|
|
130
|
+
const domainRatio = roleKeywords.length > 0 ? domainHits / roleKeywords.length : 1;
|
|
131
|
+
if (domainRatio < 0.2) {
|
|
132
|
+
score -= 10;
|
|
133
|
+
flags.push(`Recent work shows little overlap with ${ctx.agent.role} domain keywords`);
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
score = Math.max(0, Math.min(100, score));
|
|
137
|
+
const detail = flags.length > 0 ? flags.join("; ") : "Identity aligns with roster definition";
|
|
138
|
+
return { axis: "role-fidelity", score, detail };
|
|
139
|
+
}
|
|
140
|
+
function probeWorldModel(ctx) {
|
|
141
|
+
if (!ctx.identityBody) {
|
|
142
|
+
return {
|
|
143
|
+
axis: "world-model",
|
|
144
|
+
score: 50,
|
|
145
|
+
detail: "No identity file \u2014 cannot assess org awareness"
|
|
146
|
+
};
|
|
147
|
+
}
|
|
148
|
+
const identityLower = ctx.identityBody.toLowerCase();
|
|
149
|
+
const flags = [];
|
|
150
|
+
let score = 100;
|
|
151
|
+
const otherAgents = ctx.allEmployees.filter((e) => e.name !== ctx.agent.name);
|
|
152
|
+
let mentionedAgents = 0;
|
|
153
|
+
for (const other of otherAgents) {
|
|
154
|
+
if (identityLower.includes(other.name.toLowerCase())) mentionedAgents++;
|
|
155
|
+
}
|
|
156
|
+
if (otherAgents.length > 0) {
|
|
157
|
+
const mentionRatio = mentionedAgents / otherAgents.length;
|
|
158
|
+
if (mentionRatio < 0.3) {
|
|
159
|
+
score -= 15;
|
|
160
|
+
flags.push(`Identity mentions ${mentionedAgents}/${otherAgents.length} team members`);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
const hasReportingChain = identityLower.includes("report") || identityLower.includes("manager") || identityLower.includes("coo") || identityLower.includes("coordinator");
|
|
164
|
+
if (!hasReportingChain) {
|
|
165
|
+
score -= 10;
|
|
166
|
+
flags.push("Identity does not reference reporting chain");
|
|
167
|
+
}
|
|
168
|
+
const orgTerms = ["exe-os", "exe-wiki", "exe-crm", "askexe"];
|
|
169
|
+
let orgMentions = 0;
|
|
170
|
+
for (const term of orgTerms) {
|
|
171
|
+
if (identityLower.includes(term)) orgMentions++;
|
|
172
|
+
}
|
|
173
|
+
if (orgMentions === 0) {
|
|
174
|
+
score -= 5;
|
|
175
|
+
flags.push("Identity does not reference any org products");
|
|
176
|
+
}
|
|
177
|
+
score = Math.max(0, Math.min(100, score));
|
|
178
|
+
const detail = flags.length > 0 ? flags.join("; ") : "Identity reflects org structure";
|
|
179
|
+
return { axis: "world-model", score, detail };
|
|
180
|
+
}
|
|
181
|
+
function extractKeyTerms(text) {
|
|
182
|
+
const stopwords = /* @__PURE__ */ new Set([
|
|
183
|
+
"the",
|
|
184
|
+
"and",
|
|
185
|
+
"for",
|
|
186
|
+
"are",
|
|
187
|
+
"but",
|
|
188
|
+
"not",
|
|
189
|
+
"you",
|
|
190
|
+
"all",
|
|
191
|
+
"any",
|
|
192
|
+
"can",
|
|
193
|
+
"had",
|
|
194
|
+
"her",
|
|
195
|
+
"was",
|
|
196
|
+
"one",
|
|
197
|
+
"our",
|
|
198
|
+
"out",
|
|
199
|
+
"has",
|
|
200
|
+
"his",
|
|
201
|
+
"how",
|
|
202
|
+
"its",
|
|
203
|
+
"may",
|
|
204
|
+
"new",
|
|
205
|
+
"now",
|
|
206
|
+
"old",
|
|
207
|
+
"see",
|
|
208
|
+
"way",
|
|
209
|
+
"who",
|
|
210
|
+
"did",
|
|
211
|
+
"get",
|
|
212
|
+
"got",
|
|
213
|
+
"let",
|
|
214
|
+
"say",
|
|
215
|
+
"she",
|
|
216
|
+
"too",
|
|
217
|
+
"use",
|
|
218
|
+
"with",
|
|
219
|
+
"this",
|
|
220
|
+
"that",
|
|
221
|
+
"from",
|
|
222
|
+
"they",
|
|
223
|
+
"been",
|
|
224
|
+
"have",
|
|
225
|
+
"will",
|
|
226
|
+
"your",
|
|
227
|
+
"what",
|
|
228
|
+
"when",
|
|
229
|
+
"make",
|
|
230
|
+
"like",
|
|
231
|
+
"just",
|
|
232
|
+
"over",
|
|
233
|
+
"such",
|
|
234
|
+
"take",
|
|
235
|
+
"than",
|
|
236
|
+
"them",
|
|
237
|
+
"very",
|
|
238
|
+
"some",
|
|
239
|
+
"could",
|
|
240
|
+
"into",
|
|
241
|
+
"then",
|
|
242
|
+
"more",
|
|
243
|
+
"also",
|
|
244
|
+
"after",
|
|
245
|
+
"should",
|
|
246
|
+
"would",
|
|
247
|
+
"about",
|
|
248
|
+
"their",
|
|
249
|
+
"which",
|
|
250
|
+
"these",
|
|
251
|
+
"other",
|
|
252
|
+
"every",
|
|
253
|
+
"does",
|
|
254
|
+
"being",
|
|
255
|
+
"those",
|
|
256
|
+
"never",
|
|
257
|
+
"before",
|
|
258
|
+
"through"
|
|
259
|
+
]);
|
|
260
|
+
const words = text.replace(/[^\w\s-]/g, " ").split(/\s+/).filter((w) => w.length > 3 && !stopwords.has(w.toLowerCase()));
|
|
261
|
+
return [...new Set(words.map((w) => w.toLowerCase()))].slice(0, 50);
|
|
262
|
+
}
|
|
263
|
+
function extractRoleKeywords(role) {
|
|
264
|
+
const roleKeywordMap = {
|
|
265
|
+
"coo": ["coordinate", "review", "status", "team", "task", "priority", "dispatch"],
|
|
266
|
+
"cto": ["architecture", "code", "technical", "system", "design", "review", "security"],
|
|
267
|
+
"cmo": ["marketing", "brand", "content", "design", "seo", "social", "campaign"],
|
|
268
|
+
"principal engineer": ["code", "implement", "test", "fix", "feature", "refactor", "build"],
|
|
269
|
+
"staff code reviewer": ["review", "code", "quality", "issue", "fix", "pattern"],
|
|
270
|
+
"content production specialist": ["video", "image", "render", "content", "produce", "media"],
|
|
271
|
+
"ai product lead": ["competitive", "analysis", "feature", "product", "research", "repo"]
|
|
272
|
+
};
|
|
273
|
+
const normalized = role.toLowerCase();
|
|
274
|
+
return roleKeywordMap[normalized] ?? normalized.split(/\s+/).filter((w) => w.length > 2);
|
|
275
|
+
}
|
|
276
|
+
async function runDriftProbes(options = {}) {
|
|
277
|
+
const employees = loadEmployeesSync();
|
|
278
|
+
const targetAgents = options.agentId ? employees.filter((e) => e.name === options.agentId) : employees;
|
|
279
|
+
if (targetAgents.length === 0) {
|
|
280
|
+
return [];
|
|
281
|
+
}
|
|
282
|
+
const axes = options.axes ?? ["continuity", "consistency", "role-fidelity", "world-model"];
|
|
283
|
+
const results = [];
|
|
284
|
+
for (const agent of targetAgents) {
|
|
285
|
+
const identity = getIdentity(agent.name);
|
|
286
|
+
const { count: memCount, texts: memTexts } = await getRecentMemories(agent.name, 50);
|
|
287
|
+
const decisionCount = await getDecisionCount(agent.name);
|
|
288
|
+
const ctx = {
|
|
289
|
+
agent,
|
|
290
|
+
identityBody: identity?.body ?? null,
|
|
291
|
+
identityRole: identity?.frontmatter.role ?? null,
|
|
292
|
+
recentMemoryCount: memCount,
|
|
293
|
+
recentMemoryTexts: memTexts,
|
|
294
|
+
storedDecisionCount: decisionCount,
|
|
295
|
+
allEmployees: employees
|
|
296
|
+
};
|
|
297
|
+
const probes = [];
|
|
298
|
+
const probeFns = {
|
|
299
|
+
continuity: probeContinuity,
|
|
300
|
+
consistency: probeConsistency,
|
|
301
|
+
"role-fidelity": probeRoleFidelity,
|
|
302
|
+
"world-model": probeWorldModel
|
|
303
|
+
};
|
|
304
|
+
for (const axis of axes) {
|
|
305
|
+
probes.push(probeFns[axis](ctx));
|
|
306
|
+
}
|
|
307
|
+
const scores = {
|
|
308
|
+
continuity: 0,
|
|
309
|
+
consistency: 0,
|
|
310
|
+
"role-fidelity": 0,
|
|
311
|
+
"world-model": 0
|
|
312
|
+
};
|
|
313
|
+
for (const probe of probes) {
|
|
314
|
+
scores[probe.axis] = probe.score;
|
|
315
|
+
}
|
|
316
|
+
const weights = {
|
|
317
|
+
continuity: 0.2,
|
|
318
|
+
consistency: 0.2,
|
|
319
|
+
"role-fidelity": 0.35,
|
|
320
|
+
"world-model": 0.25
|
|
321
|
+
};
|
|
322
|
+
let weightedSum = 0;
|
|
323
|
+
let weightTotal = 0;
|
|
324
|
+
for (const axis of axes) {
|
|
325
|
+
weightedSum += scores[axis] * weights[axis];
|
|
326
|
+
weightTotal += weights[axis];
|
|
327
|
+
}
|
|
328
|
+
const overall = Math.round(weightTotal > 0 ? weightedSum / weightTotal : 0);
|
|
329
|
+
const flags = [];
|
|
330
|
+
for (const probe of probes) {
|
|
331
|
+
if (probe.score < DRIFT_THRESHOLD) {
|
|
332
|
+
flags.push(`${probe.axis} at ${probe.score}% \u2014 ${probe.detail}`);
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
results.push({
|
|
336
|
+
agent: agent.name,
|
|
337
|
+
scores,
|
|
338
|
+
overall,
|
|
339
|
+
drifting: flags.length > 0,
|
|
340
|
+
flags
|
|
341
|
+
});
|
|
342
|
+
}
|
|
343
|
+
return results;
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
export {
|
|
347
|
+
runDriftProbes
|
|
348
|
+
};
|
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
import {
|
|
2
|
+
employeeSessionName,
|
|
3
|
+
isEmployeeAlive,
|
|
4
|
+
resolveExeSession,
|
|
5
|
+
sendIntercom,
|
|
6
|
+
strictSessionScopeFilter
|
|
7
|
+
} from "./chunk-P2QTTEYP.js";
|
|
8
|
+
import {
|
|
9
|
+
parseMessage,
|
|
10
|
+
serializeMessage
|
|
11
|
+
} from "./chunk-4JERP7NT.js";
|
|
12
|
+
import {
|
|
13
|
+
recordOrchestrationEventBestEffort
|
|
14
|
+
} from "./chunk-XIT6ONPY.js";
|
|
15
|
+
import {
|
|
16
|
+
getClient
|
|
17
|
+
} from "./chunk-RHZ2CRJU.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 sendMessage(input) {
|
|
49
|
+
const client = getClient();
|
|
50
|
+
const id = generateUlid();
|
|
51
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
52
|
+
const targetDevice = input.targetDevice ?? "local";
|
|
53
|
+
const sessionScope = input.sessionScope === void 0 ? resolveExeSession() : input.sessionScope;
|
|
54
|
+
await client.execute({
|
|
55
|
+
sql: `INSERT INTO messages (id, from_agent, from_device, target_agent, target_project, target_device, session_scope, content, priority, status, founder_id, created_at)
|
|
56
|
+
VALUES (?, ?, 'local', ?, ?, ?, ?, ?, ?, 'pending', ?, ?)`,
|
|
57
|
+
args: [
|
|
58
|
+
id,
|
|
59
|
+
input.fromAgent,
|
|
60
|
+
input.targetAgent,
|
|
61
|
+
input.targetProject ?? null,
|
|
62
|
+
targetDevice,
|
|
63
|
+
sessionScope,
|
|
64
|
+
input.content,
|
|
65
|
+
input.priority ?? "normal",
|
|
66
|
+
input.founderId ?? null,
|
|
67
|
+
now
|
|
68
|
+
]
|
|
69
|
+
});
|
|
70
|
+
recordOrchestrationEventBestEffort({
|
|
71
|
+
eventType: "message.sent",
|
|
72
|
+
source: "messaging.sendMessage",
|
|
73
|
+
agentId: input.fromAgent,
|
|
74
|
+
payload: { targetAgent: input.targetAgent, priority: input.priority ?? "normal", targetDevice },
|
|
75
|
+
sessionScope: sessionScope ?? null
|
|
76
|
+
});
|
|
77
|
+
try {
|
|
78
|
+
if (targetDevice !== "local") {
|
|
79
|
+
await deliverCrossMachineMessage(id, targetDevice);
|
|
80
|
+
} else {
|
|
81
|
+
await deliverLocalMessage(id);
|
|
82
|
+
}
|
|
83
|
+
} catch {
|
|
84
|
+
}
|
|
85
|
+
const sentScope = strictSessionScopeFilter(sessionScope);
|
|
86
|
+
const result = await client.execute({
|
|
87
|
+
sql: `SELECT * FROM messages WHERE id = ?${sentScope.sql}`,
|
|
88
|
+
args: [id, ...sentScope.args]
|
|
89
|
+
});
|
|
90
|
+
return rowToMessage(result.rows[0]);
|
|
91
|
+
}
|
|
92
|
+
var _wsClientSend = null;
|
|
93
|
+
function setWsClientSend(fn) {
|
|
94
|
+
_wsClientSend = fn;
|
|
95
|
+
}
|
|
96
|
+
async function deliverCrossMachineMessage(messageId, targetDevice) {
|
|
97
|
+
const client = getClient();
|
|
98
|
+
const result = await client.execute({
|
|
99
|
+
sql: "SELECT * FROM messages WHERE id = ?",
|
|
100
|
+
args: [messageId]
|
|
101
|
+
});
|
|
102
|
+
if (result.rows.length === 0) return false;
|
|
103
|
+
const msg = rowToMessage(result.rows[0]);
|
|
104
|
+
if (msg.status !== "pending") return false;
|
|
105
|
+
if (!_wsClientSend) {
|
|
106
|
+
return false;
|
|
107
|
+
}
|
|
108
|
+
const payload = JSON.stringify({
|
|
109
|
+
id: msg.id,
|
|
110
|
+
fromAgent: msg.fromAgent,
|
|
111
|
+
targetAgent: msg.targetAgent,
|
|
112
|
+
targetProject: msg.targetProject,
|
|
113
|
+
sessionScope: msg.sessionScope,
|
|
114
|
+
content: msg.content,
|
|
115
|
+
priority: msg.priority,
|
|
116
|
+
createdAt: msg.createdAt
|
|
117
|
+
});
|
|
118
|
+
const sent = _wsClientSend(targetDevice, payload);
|
|
119
|
+
if (sent) {
|
|
120
|
+
await client.execute({
|
|
121
|
+
sql: "UPDATE messages SET status = 'synced' WHERE id = ?",
|
|
122
|
+
args: [messageId]
|
|
123
|
+
});
|
|
124
|
+
return true;
|
|
125
|
+
}
|
|
126
|
+
return false;
|
|
127
|
+
}
|
|
128
|
+
async function deliverLocalMessage(messageId) {
|
|
129
|
+
const client = getClient();
|
|
130
|
+
const result = await client.execute({
|
|
131
|
+
sql: "SELECT * FROM messages WHERE id = ?",
|
|
132
|
+
args: [messageId]
|
|
133
|
+
});
|
|
134
|
+
if (result.rows.length === 0) return false;
|
|
135
|
+
const msg = rowToMessage(result.rows[0]);
|
|
136
|
+
if (msg.status !== "pending") return false;
|
|
137
|
+
const targetAgent = msg.targetAgent;
|
|
138
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
139
|
+
try {
|
|
140
|
+
const exeSession = resolveExeSession();
|
|
141
|
+
if (!exeSession) {
|
|
142
|
+
throw new Error("No coordinator session found");
|
|
143
|
+
}
|
|
144
|
+
process.stderr.write(`[messaging] ENTER deliverLocal: target=${targetAgent} exeSession=${exeSession}
|
|
145
|
+
`);
|
|
146
|
+
const { isCoordinatorName: isCoord } = await import("./lib/employees.js");
|
|
147
|
+
const isCOO = isCoord(targetAgent);
|
|
148
|
+
process.stderr.write(`[messaging] COO check: targetAgent="${targetAgent}" isCOO=${isCOO}
|
|
149
|
+
`);
|
|
150
|
+
const sessionName = isCOO ? exeSession : employeeSessionName(targetAgent, exeSession);
|
|
151
|
+
process.stderr.write(`[messaging] delivery attempt: target=${targetAgent} session=${sessionName} isCOO=${isCOO}
|
|
152
|
+
`);
|
|
153
|
+
if (!isCOO && !isEmployeeAlive(sessionName)) {
|
|
154
|
+
throw new Error("Session not running \u2014 message stays queued");
|
|
155
|
+
}
|
|
156
|
+
const intercomResult = sendIntercom(sessionName, { force: true, reason: "message" });
|
|
157
|
+
process.stderr.write(`[messaging] intercom result for ${sessionName}: ${intercomResult}
|
|
158
|
+
`);
|
|
159
|
+
await client.execute({
|
|
160
|
+
sql: "UPDATE messages SET status = 'delivered', delivered_at = ? WHERE id = ?",
|
|
161
|
+
args: [now, messageId]
|
|
162
|
+
});
|
|
163
|
+
recordOrchestrationEventBestEffort({
|
|
164
|
+
eventType: "message.delivered",
|
|
165
|
+
source: "messaging.deliverLocalMessage",
|
|
166
|
+
agentId: msg.targetAgent,
|
|
167
|
+
payload: { fromAgent: msg.fromAgent, priority: msg.priority }
|
|
168
|
+
});
|
|
169
|
+
return true;
|
|
170
|
+
} catch (err) {
|
|
171
|
+
process.stderr.write(
|
|
172
|
+
`[messaging] delivery failed for message ${messageId}: ${err instanceof Error ? err.message : String(err)}
|
|
173
|
+
`
|
|
174
|
+
);
|
|
175
|
+
const newRetryCount = msg.retryCount + 1;
|
|
176
|
+
if (newRetryCount >= MAX_RETRIES) {
|
|
177
|
+
await markFailed(messageId, "session unavailable after 10 retries", msg.sessionScope);
|
|
178
|
+
} else {
|
|
179
|
+
await client.execute({
|
|
180
|
+
sql: "UPDATE messages SET retry_count = ? WHERE id = ?",
|
|
181
|
+
args: [newRetryCount, messageId]
|
|
182
|
+
});
|
|
183
|
+
}
|
|
184
|
+
return false;
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
async function getPendingMessages(targetAgent, sessionScope) {
|
|
188
|
+
const client = getClient();
|
|
189
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
190
|
+
const result = await client.execute({
|
|
191
|
+
sql: `SELECT * FROM messages
|
|
192
|
+
WHERE target_agent = ? AND status IN ('pending', 'delivered')${scope.sql}
|
|
193
|
+
ORDER BY id`,
|
|
194
|
+
args: [targetAgent, ...scope.args]
|
|
195
|
+
});
|
|
196
|
+
return result.rows.map((row) => rowToMessage(row));
|
|
197
|
+
}
|
|
198
|
+
async function markRead(messageId, sessionScope) {
|
|
199
|
+
const client = getClient();
|
|
200
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
201
|
+
await client.execute({
|
|
202
|
+
sql: `UPDATE messages SET status = 'read'
|
|
203
|
+
WHERE id = ? AND status IN ('pending', 'delivered')${scope.sql}`,
|
|
204
|
+
args: [messageId, ...scope.args]
|
|
205
|
+
});
|
|
206
|
+
recordOrchestrationEventBestEffort({
|
|
207
|
+
eventType: "message.read",
|
|
208
|
+
source: "messaging.markRead",
|
|
209
|
+
payload: { messageId }
|
|
210
|
+
});
|
|
211
|
+
}
|
|
212
|
+
async function markAcknowledged(messageId, sessionScope) {
|
|
213
|
+
const client = getClient();
|
|
214
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
215
|
+
await client.execute({
|
|
216
|
+
sql: `UPDATE messages SET status = 'acknowledged', processed_at = ?
|
|
217
|
+
WHERE id = ? AND status = 'read'${scope.sql}`,
|
|
218
|
+
args: [(/* @__PURE__ */ new Date()).toISOString(), messageId, ...scope.args]
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
async function markProcessed(messageId, sessionScope) {
|
|
222
|
+
const client = getClient();
|
|
223
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
224
|
+
await client.execute({
|
|
225
|
+
sql: `UPDATE messages SET status = 'processed', processed_at = ?
|
|
226
|
+
WHERE id = ?${scope.sql}`,
|
|
227
|
+
args: [(/* @__PURE__ */ new Date()).toISOString(), messageId, ...scope.args]
|
|
228
|
+
});
|
|
229
|
+
}
|
|
230
|
+
async function getMessageStatus(messageId, sessionScope) {
|
|
231
|
+
const client = getClient();
|
|
232
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
233
|
+
const result = await client.execute({
|
|
234
|
+
sql: `SELECT status FROM messages WHERE id = ?${scope.sql}`,
|
|
235
|
+
args: [messageId, ...scope.args]
|
|
236
|
+
});
|
|
237
|
+
return result.rows[0]?.status ?? null;
|
|
238
|
+
}
|
|
239
|
+
async function getUnacknowledgedMessages(targetAgent, sessionScope) {
|
|
240
|
+
const client = getClient();
|
|
241
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
242
|
+
const result = await client.execute({
|
|
243
|
+
sql: `SELECT * FROM messages
|
|
244
|
+
WHERE target_agent = ? AND status IN ('pending', 'delivered', 'read')${scope.sql}
|
|
245
|
+
ORDER BY id`,
|
|
246
|
+
args: [targetAgent, ...scope.args]
|
|
247
|
+
});
|
|
248
|
+
return result.rows.map((row) => rowToMessage(row));
|
|
249
|
+
}
|
|
250
|
+
async function getReadMessages(targetAgent, sessionScope) {
|
|
251
|
+
const client = getClient();
|
|
252
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
253
|
+
const result = await client.execute({
|
|
254
|
+
sql: `SELECT * FROM messages
|
|
255
|
+
WHERE target_agent = ? AND status = 'read'${scope.sql}
|
|
256
|
+
ORDER BY id`,
|
|
257
|
+
args: [targetAgent, ...scope.args]
|
|
258
|
+
});
|
|
259
|
+
return result.rows.map((row) => rowToMessage(row));
|
|
260
|
+
}
|
|
261
|
+
async function markFailed(messageId, reason, sessionScope) {
|
|
262
|
+
const client = getClient();
|
|
263
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
264
|
+
await client.execute({
|
|
265
|
+
sql: `UPDATE messages SET status = 'failed', failed_at = ?, failure_reason = ?
|
|
266
|
+
WHERE id = ?${scope.sql}`,
|
|
267
|
+
args: [(/* @__PURE__ */ new Date()).toISOString(), reason, messageId, ...scope.args]
|
|
268
|
+
});
|
|
269
|
+
}
|
|
270
|
+
async function getFailedMessages(sessionScope) {
|
|
271
|
+
const client = getClient();
|
|
272
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
273
|
+
const result = await client.execute({
|
|
274
|
+
sql: `SELECT * FROM messages WHERE status = 'failed'${scope.sql} ORDER BY created_at DESC`,
|
|
275
|
+
args: [...scope.args]
|
|
276
|
+
});
|
|
277
|
+
return result.rows.map((row) => rowToMessage(row));
|
|
278
|
+
}
|
|
279
|
+
async function retryPendingMessages(sessionScope) {
|
|
280
|
+
const client = getClient();
|
|
281
|
+
const scope = strictSessionScopeFilter(sessionScope);
|
|
282
|
+
const result = await client.execute({
|
|
283
|
+
sql: `SELECT * FROM messages
|
|
284
|
+
WHERE status = 'pending' AND retry_count < ?${scope.sql}
|
|
285
|
+
ORDER BY id`,
|
|
286
|
+
args: [MAX_RETRIES, ...scope.args]
|
|
287
|
+
});
|
|
288
|
+
let delivered = 0;
|
|
289
|
+
for (const row of result.rows) {
|
|
290
|
+
const msg = rowToMessage(row);
|
|
291
|
+
try {
|
|
292
|
+
const success = await deliverLocalMessage(msg.id);
|
|
293
|
+
if (success) delivered++;
|
|
294
|
+
} catch {
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
return delivered;
|
|
298
|
+
}
|
|
299
|
+
async function sendTypedMessage(envelope, targetAgent, opts) {
|
|
300
|
+
return sendMessage({
|
|
301
|
+
fromAgent: envelope.from,
|
|
302
|
+
targetAgent,
|
|
303
|
+
targetProject: opts?.targetProject,
|
|
304
|
+
targetDevice: opts?.targetDevice,
|
|
305
|
+
content: serializeMessage(envelope),
|
|
306
|
+
priority: opts?.priority,
|
|
307
|
+
sessionScope: opts?.sessionScope
|
|
308
|
+
});
|
|
309
|
+
}
|
|
310
|
+
function parseIncomingMessage(content) {
|
|
311
|
+
return parseMessage(content);
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
export {
|
|
315
|
+
sendMessage,
|
|
316
|
+
setWsClientSend,
|
|
317
|
+
deliverLocalMessage,
|
|
318
|
+
getPendingMessages,
|
|
319
|
+
markRead,
|
|
320
|
+
markAcknowledged,
|
|
321
|
+
markProcessed,
|
|
322
|
+
getMessageStatus,
|
|
323
|
+
getUnacknowledgedMessages,
|
|
324
|
+
getReadMessages,
|
|
325
|
+
markFailed,
|
|
326
|
+
getFailedMessages,
|
|
327
|
+
retryPendingMessages,
|
|
328
|
+
sendTypedMessage,
|
|
329
|
+
parseIncomingMessage
|
|
330
|
+
};
|