@askexenow/exe-os 0.9.271 → 0.9.272
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/dist/active-agent-BDYXURXQ.js +26 -0
- package/dist/active-agent-YWBGAKGU.js +25 -0
- package/dist/agentic-ontology-56VHSVS3.js +25 -0
- package/dist/backfill-metadata-G46ABBVR.js +597 -0
- package/dist/backfill-metadata-VAV27KJK.js +597 -0
- package/dist/behaviors-USUTDXVA.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 +4 -4
- package/dist/bin/cleanup-stale-review-tasks.js +10 -10
- package/dist/bin/cli.js +16 -16
- package/dist/bin/exe-agent-config.js +3 -3
- package/dist/bin/exe-agent.js +4 -4
- package/dist/bin/exe-assign.js +5 -5
- package/dist/bin/exe-boot.js +17 -17
- package/dist/bin/exe-call.js +4 -4
- package/dist/bin/exe-cloud.js +4 -4
- package/dist/bin/exe-dispatch.js +10 -10
- 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 +4 -4
- package/dist/bin/exe-heartbeat.js +10 -10
- package/dist/bin/exe-kill.js +13 -13
- package/dist/bin/exe-launch-agent.js +37 -19
- package/dist/bin/exe-new-employee.js +6 -6
- package/dist/bin/exe-pending-messages.js +11 -11
- package/dist/bin/exe-pending-notifications.js +10 -10
- package/dist/bin/exe-pending-reviews.js +10 -10
- package/dist/bin/exe-rename.js +4 -4
- package/dist/bin/exe-review.js +12 -12
- package/dist/bin/exe-search.js +5 -5
- package/dist/bin/exe-session-cleanup.js +15 -15
- package/dist/bin/exe-settings.js +5 -5
- package/dist/bin/exe-start-codex.js +11 -11
- package/dist/bin/exe-start-opencode.js +8 -8
- package/dist/bin/exe-status.js +11 -11
- package/dist/bin/exe-team.js +3 -3
- package/dist/bin/git-sweep.js +11 -11
- 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 +10 -10
- package/dist/bin/setup.js +1 -1
- package/dist/bin/shard-migrate.js +4 -4
- package/dist/capacity-monitor-IFVRCIM7.js +49 -0
- package/dist/capacity-monitor-Q47GBDSY.js +49 -0
- package/dist/catchup-brief-RP4QHXNT.js +151 -0
- package/dist/catchup-brief-TKA6TEK4.js +151 -0
- package/dist/chunk-23KJ2LXY.js +58 -0
- package/dist/chunk-2NQQP3FF.js +630 -0
- package/dist/chunk-3FU5I3KV.js +526 -0
- package/dist/chunk-46IEEKPU.js +13696 -0
- package/dist/chunk-57UAFTO2.js +3958 -0
- package/dist/chunk-5AS622MM.js +3958 -0
- package/dist/chunk-62DEE65H.js +371 -0
- package/dist/chunk-64T6DFSS.js +447 -0
- package/dist/chunk-6BWDP63Z.js +197 -0
- package/dist/chunk-6GPYL7TX.js +214 -0
- package/dist/chunk-6KWLUVFL.js +54 -0
- package/dist/chunk-6N5ISWBF.js +1148 -0
- package/dist/chunk-6OD7PVMC.js +333 -0
- package/dist/chunk-7ET5CYTD.js +382 -0
- package/dist/chunk-7IWLKR6N.js +76 -0
- package/dist/chunk-7OEUOJL5.js +1021 -0
- package/dist/chunk-AIRJTKDK.js +204 -0
- package/dist/chunk-ATJ3NXDP.js +244 -0
- package/dist/chunk-B5IS7LE4.js +128 -0
- package/dist/chunk-BKINEQVI.js +244 -0
- package/dist/chunk-BOJV6NI3.js +128 -0
- package/dist/chunk-BPHWI6N2.js +284 -0
- package/dist/chunk-BXCQWWJP.js +185 -0
- package/dist/chunk-BZ6K7AY3.js +50 -0
- package/dist/chunk-C54KIFLS.js +214 -0
- package/dist/chunk-C6OYEJJI.js +260 -0
- package/dist/chunk-CHBHR5W6.js +3556 -0
- package/dist/chunk-D2T3272U.js +171 -0
- package/dist/chunk-DCHEIVGT.js +221 -0
- package/dist/chunk-DHIBLMSP.js +30 -0
- package/dist/chunk-E2AF2WYY.js +346 -0
- package/dist/chunk-E2KZEZZW.js +1090 -0
- package/dist/chunk-E4KWB4WM.js +348 -0
- package/dist/chunk-ENU7URWK.js +1073 -0
- package/dist/chunk-EZ7KAZMC.js +132 -0
- package/dist/chunk-F4FSSHR4.js +1073 -0
- package/dist/chunk-FPXU56FG.js +346 -0
- package/dist/chunk-FY7HHR5I.js +128 -0
- package/dist/chunk-G2S2UMU4.js +159 -0
- package/dist/chunk-G33BHQCO.js +70 -0
- package/dist/chunk-GJQTL7RX.js +127 -0
- package/dist/chunk-GLCKDEM2.js +97 -0
- package/dist/chunk-GMA34SXV.js +240 -0
- package/dist/chunk-GVAVEBYR.js +2091 -0
- package/dist/chunk-HOSJTLBQ.js +513 -0
- package/dist/chunk-IC6HVAS3.js +56 -0
- package/dist/chunk-IDFJNO44.js +1051 -0
- package/dist/chunk-ILFJMEY5.js +97 -0
- package/dist/chunk-ISQAOSL3.js +1921 -0
- package/dist/chunk-J6V2DCZK.js +382 -0
- package/dist/chunk-JP4CLFLR.js +1148 -0
- package/dist/chunk-JTIOZHWG.js +58 -0
- package/dist/chunk-KDICWAYV.js +1345 -0
- package/dist/chunk-KOBIB6WG.js +159 -0
- package/dist/chunk-KQFDDQB6.js +13696 -0
- package/dist/chunk-KZNSOHCB.js +280 -0
- package/dist/chunk-LVMBYP3C.js +171 -0
- package/dist/chunk-M2WQW5NC.js +227 -0
- package/dist/chunk-MY6SP5NZ.js +551 -0
- package/dist/chunk-N2ACW2ZG.js +363 -0
- package/dist/chunk-NSMJDATI.js +495 -0
- package/dist/chunk-NSQ5JE23.js +1090 -0
- package/dist/chunk-NZL567WG.js +81 -0
- package/dist/chunk-O5OMH6LI.js +244 -0
- package/dist/chunk-OBUV3W7L.js +163 -0
- package/dist/chunk-OLDS7LJN.js +495 -0
- package/dist/chunk-OO2I22RX.js +38 -0
- package/dist/chunk-OPUUT33V.js +447 -0
- package/dist/chunk-OT3VMTKB.js +50 -0
- package/dist/chunk-P6RVIOVA.js +157 -0
- package/dist/chunk-PUA5564C.js +210 -0
- package/dist/chunk-PUQLKLQX.js +731 -0
- package/dist/chunk-QROKS65G.js +76 -0
- package/dist/chunk-R54I2N2T.js +818 -0
- package/dist/chunk-RCFYQHUP.js +818 -0
- package/dist/chunk-RJTND4YS.js +284 -0
- package/dist/chunk-SBLHQMMZ.js +81 -0
- package/dist/chunk-SG2ANG5C.js +123 -0
- package/dist/chunk-SVFNKSZV.js +333 -0
- package/dist/chunk-TAQT2DC7.js +330 -0
- package/dist/chunk-TB7HFW7M.js +127 -0
- package/dist/chunk-UUKDAIH2.js +731 -0
- package/dist/chunk-V6VEFEEH.js +1345 -0
- package/dist/chunk-VIO2ALGH.js +290 -0
- package/dist/chunk-VKCUSNJW.js +377 -0
- package/dist/chunk-VRPPJFIQ.js +1921 -0
- package/dist/chunk-WP3PVBBP.js +204 -0
- package/dist/chunk-WQEUY7DC.js +129 -0
- package/dist/chunk-WXMXUKCA.js +262 -0
- package/dist/chunk-X2WBH2IO.js +297 -0
- package/dist/chunk-X33TSJNO.js +394 -0
- package/dist/chunk-X7MMI2UI.js +89 -0
- package/dist/chunk-XG3BQZIK.js +85 -0
- package/dist/chunk-XIKBIAOS.js +75 -0
- package/dist/chunk-XPEB545Q.js +54 -0
- package/dist/chunk-XWH2MLWS.js +330 -0
- package/dist/chunk-YH7V73XW.js +89 -0
- package/dist/chunk-YMLM5D65.js +135 -0
- package/dist/chunk-YNJPRQ6J.js +377 -0
- package/dist/chunk-YSNEHBI6.js +551 -0
- package/dist/chunk-ZD6BMW2K.js +33 -0
- package/dist/chunk-ZKG5IYCG.js +668 -0
- package/dist/chunk-ZU4K7ZNX.js +197 -0
- package/dist/co-activation-HZMJC34P.js +72 -0
- package/dist/co-occurrence-AVYXRV4L.js +74 -0
- package/dist/core-memory-NPJCVUMF.js +110 -0
- package/dist/core-memory-OKGXL33Z.js +110 -0
- package/dist/crdt-sync-ZCH55JNR.js +33 -0
- package/dist/crm-webhook-6OMVUUGR.js +10 -0
- package/dist/crm-webhook-UCWF3XDB.js +10 -0
- package/dist/cto-delegation-gate-JFZFZGC2.js +206 -0
- package/dist/cto-delegation-gate-K32M4GVM.js +206 -0
- package/dist/daemon-orchestration-2Q7BYOHC.js +135 -0
- package/dist/daemon-orchestration-4RJ2CZJL.js +135 -0
- package/dist/db-backup-5GA2YFDX.js +33 -0
- package/dist/dreaming-I6KXO6E2.js +32 -0
- package/dist/dreaming-NJBK5ILR.js +32 -0
- package/dist/exe-drift-VSMIMHL4.js +68 -0
- package/dist/exe-export-DVHHIA6Y.js +73 -0
- package/dist/exe-export-GIVQDENS.js +73 -0
- package/dist/exe-import-7N46LSMQ.js +76 -0
- package/dist/exe-import-FINYUV5T.js +76 -0
- package/dist/exe-key-H45JY44F.js +579 -0
- package/dist/exe-key-MAEQGTB7.js +579 -0
- package/dist/exe-snapshot-3TEM3BFD.js +164 -0
- package/dist/exe-snapshot-HECGUHL3.js +164 -0
- package/dist/fast-db-init-3CNTADVO.js +7 -0
- package/dist/fast-db-init-HXCS2AP5.js +7 -0
- package/dist/gateway/index.js +8 -8
- package/dist/git-staleness-YCEBBIVK.js +110 -0
- package/dist/git-task-sweep-H34STRNT.js +40 -0
- package/dist/git-task-sweep-YL7NLDCK.js +40 -0
- package/dist/global-procedures-IHZM6C2K.js +20 -0
- package/dist/graph-auto-extract-RZQ3MHP2.js +162 -0
- package/dist/hooks/bug-report-worker.js +12 -12
- package/dist/hooks/codex-stop-task-finalizer.js +12 -12
- package/dist/hooks/commit-complete.js +12 -12
- 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 +11 -11
- package/dist/hooks/post-tool-combined.js +5 -5
- package/dist/hooks/pre-compact.js +12 -12
- package/dist/hooks/pre-tool-use.js +15 -15
- package/dist/hooks/prompt-submit.js +21 -21
- package/dist/hooks/session-end.js +16 -16
- package/dist/hooks/session-start.js +10 -10
- package/dist/hooks/stop.js +15 -15
- package/dist/hooks/subagent-stop.js +11 -11
- package/dist/hooks/summary-worker.js +15 -15
- package/dist/index.js +18 -18
- package/dist/installer-4EW5ZDGD.js +296 -0
- package/dist/installer-B2JTQO55.js +38 -0
- package/dist/installer-MIL352T7.js +342 -0
- package/dist/lib/agent-config.js +9 -3
- 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.js +2 -2
- package/dist/lib/employee-templates.js +4 -4
- package/dist/lib/employees.js +2 -2
- package/dist/lib/exe-daemon.js +34 -34
- package/dist/lib/hybrid-search.js +5 -5
- package/dist/lib/identity.js +2 -2
- package/dist/lib/messaging.js +10 -10
- 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 +11 -11
- package/dist/lib/tmux-routing.js +9 -9
- package/dist/lib/token-spend.js +3 -3
- package/dist/mcp/register-tools.js +54 -54
- package/dist/mcp/server.js +55 -55
- 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 +13 -13
- 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 +13 -13
- package/dist/mcp/tools/send-message.js +12 -12
- package/dist/mcp/tools/update-task.js +12 -12
- package/dist/mcp-http-config-OJQR246S.js +27 -0
- package/dist/memory-cards-IPULSQFA.js +174 -0
- package/dist/memory-graph-extractor-3TZZOKHY.js +17 -0
- package/dist/memory-poisoning-defense-SGUGR5YJ.js +225 -0
- package/dist/memory-reflection-H3WGCEM6.js +238 -0
- package/dist/notifications-VWPO6NJF.js +45 -0
- package/dist/notifications-WCSRQN2V.js +45 -0
- package/dist/orchestration-events-O5PSDEIO.js +25 -0
- package/dist/orchestrator-RAPEJUOI.js +33 -0
- package/dist/orchestrator-XPG6LJAI.js +33 -0
- package/dist/pipeline-router-5NT6FUC3.js +13 -0
- package/dist/pipeline-router-KSUXONDT.js +13 -0
- package/dist/plan-limits-53NXLNDQ.js +26 -0
- package/dist/project-boot-ITN3FZMM.js +299 -0
- package/dist/projection-worker-27XX5M2W.js +964 -0
- package/dist/reranker-GU7L2PJX.js +19 -0
- package/dist/reranker-TZEXIJAN.js +19 -0
- package/dist/review-polling-FA2J2Q5O.js +124 -0
- package/dist/review-polling-MLS4BQ3N.js +124 -0
- package/dist/runtime/index.js +12 -12
- package/dist/session-events-WWGF3B2N.js +36 -0
- package/dist/session-events-ZHXXAH6B.js +36 -0
- package/dist/session-kill-telemetry-O4TJHHOZ.js +29 -0
- package/dist/session-scope-CQXB7VMH.js +86 -0
- package/dist/session-scope-HHUMJYF6.js +86 -0
- package/dist/setup-wizard-UM2RHSBJ.js +12 -0
- package/dist/skill-refinement-MJPOHYD5.js +157 -0
- package/dist/skill-refinement-NVUBRK22.js +157 -0
- package/dist/stack-release-BAPCXMXW.js +713 -0
- package/dist/stack-release-W4TWTEZP.js +731 -0
- package/dist/steward-gate-VLE7OCKO.js +13 -0
- package/dist/task-enforcement-FUHDL6UR.js +391 -0
- package/dist/task-enforcement-QL3K4N3F.js +391 -0
- package/dist/task-scope-TZYMB634.js +35 -0
- package/dist/task-scope-ZVLUBS4C.js +35 -0
- package/dist/tasks-crud-4MSLJWXE.js +77 -0
- package/dist/tasks-crud-HIPXKRKX.js +77 -0
- package/dist/tasks-notify-7JBUNE7R.js +38 -0
- package/dist/tasks-notify-UPIJ3L4O.js +38 -0
- package/dist/tasks-review-5SJSFTUB.js +47 -0
- package/dist/tasks-review-JHSYBR5I.js +47 -0
- package/dist/telemetry-upload-BSGOXGUP.js +739 -0
- package/dist/telemetry-upload-LTX3C5HZ.js +739 -0
- package/dist/token-budget-2CDWQU3Q.js +84 -0
- package/dist/tool-telemetry-7YS7EN7B.js +17 -0
- package/dist/tui/App.js +17 -17
- package/dist/tui-data-GDGBOS6G.js +258 -0
- package/dist/tui-data-VXF2RBVM.js +258 -0
- package/dist/wiki-acl-MJIMXRQV.js +111 -0
- package/dist/worker-gate-WQGTZOSM.js +21 -0
- package/dist/worker-gate-X2YDTKTL.js +21 -0
- package/dist/workflow-engine-CYXRZXBM.js +28 -0
- package/dist/workflow-engine-KMLAXVA4.js +28 -0
- package/dist/worktree-NLSKVRNC.js +26 -0
- package/dist/worktree-sweep-44TMEPLE.js +19 -0
- package/package.json +1 -1
- package/release-notes.json +24 -19
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getActiveAgent
|
|
3
|
+
} from "./chunk-BXCQWWJP.js";
|
|
4
|
+
import {
|
|
5
|
+
listTasks
|
|
6
|
+
} from "./chunk-5AS622MM.js";
|
|
7
|
+
import {
|
|
8
|
+
getAgentContext
|
|
9
|
+
} from "./chunk-GJV3WDWM.js";
|
|
10
|
+
import {
|
|
11
|
+
isCoordinatorName
|
|
12
|
+
} from "./chunk-CHBHR5W6.js";
|
|
13
|
+
|
|
14
|
+
// src/mcp/tools/list-tasks.ts
|
|
15
|
+
import { z } from "zod";
|
|
16
|
+
function registerListTasks(server) {
|
|
17
|
+
server.registerTool(
|
|
18
|
+
"list_tasks",
|
|
19
|
+
{
|
|
20
|
+
title: "List Tasks",
|
|
21
|
+
description: "Query tasks by assignee, status, project, or priority. Defaults to current project. Pass project_name='all' for all projects. When querying your own tasks, project filter is skipped automatically.",
|
|
22
|
+
inputSchema: {
|
|
23
|
+
assigned_to: z.string().optional().describe("Filter by employee name"),
|
|
24
|
+
status: z.enum(["open", "in_progress", "done", "needs_review", "blocked", "cancelled", "closed"]).optional().describe("Filter by status. Default: active tasks only (excludes closed/cancelled)"),
|
|
25
|
+
project_name: z.string().optional().describe("Project name. Defaults to current project. Pass 'all' for all projects."),
|
|
26
|
+
priority: z.enum(["p0", "p1", "p2"]).optional().describe("Filter by priority"),
|
|
27
|
+
cross_session: z.boolean().optional().describe("When true, return tasks from ALL coordinator sessions (read-only). Default: false.")
|
|
28
|
+
}
|
|
29
|
+
},
|
|
30
|
+
async ({ assigned_to, status, project_name, priority, cross_session }) => {
|
|
31
|
+
try {
|
|
32
|
+
const { agentId } = getActiveAgent();
|
|
33
|
+
const isReviewQueue = status === "needs_review";
|
|
34
|
+
const shouldDefaultToOwnQueue = !assigned_to && !isReviewQueue && !isCoordinatorName(agentId);
|
|
35
|
+
const effectiveAssignedTo = assigned_to ?? (shouldDefaultToOwnQueue ? agentId : void 0);
|
|
36
|
+
const effectiveReviewer = !assigned_to && isReviewQueue ? agentId : void 0;
|
|
37
|
+
const isOwnQuery = effectiveAssignedTo === agentId;
|
|
38
|
+
const callerSession = getAgentContext()?.sessionHint || process.env.EXE_SESSION_NAME || process.env.EXE_SESSION || null;
|
|
39
|
+
const resolvedProject = project_name && project_name !== "all" ? project_name : null;
|
|
40
|
+
const effectiveCrossSession = cross_session === true;
|
|
41
|
+
const tasks = await listTasks({
|
|
42
|
+
assignedTo: effectiveAssignedTo,
|
|
43
|
+
reviewer: effectiveReviewer,
|
|
44
|
+
status,
|
|
45
|
+
projectName: resolvedProject,
|
|
46
|
+
priority,
|
|
47
|
+
crossSession: effectiveCrossSession,
|
|
48
|
+
isOwnQuery: !!isOwnQuery,
|
|
49
|
+
callerSession
|
|
50
|
+
});
|
|
51
|
+
if (tasks.length === 0) {
|
|
52
|
+
return {
|
|
53
|
+
content: [{ type: "text", text: "No tasks found." }]
|
|
54
|
+
};
|
|
55
|
+
}
|
|
56
|
+
const lines = tasks.map((t) => {
|
|
57
|
+
const cpIndicator = t.checkpointCount && t.checkpointCount > 0 ? ` [cp:${t.checkpointCount}]` : "";
|
|
58
|
+
let budgetNote = "";
|
|
59
|
+
if (t.budgetTokens !== null) {
|
|
60
|
+
const pct = Math.round(t.tokensUsed / t.budgetTokens * 100);
|
|
61
|
+
budgetNote = ` [${t.tokensUsed}/${t.budgetTokens} tokens, ${pct}%]`;
|
|
62
|
+
}
|
|
63
|
+
const shortId = t.id.slice(0, 8);
|
|
64
|
+
const sessionNote = cross_session && t.sessionScope ? ` [session:${t.sessionScope}]` : "";
|
|
65
|
+
return `- [${t.priority.toUpperCase()}] ${t.title} (${t.projectName}) \u2014 ${t.status}${cpIndicator}${budgetNote}${sessionNote} \u2192 ${t.assignedTo} [id:${shortId}]`;
|
|
66
|
+
});
|
|
67
|
+
return {
|
|
68
|
+
content: [
|
|
69
|
+
{
|
|
70
|
+
type: "text",
|
|
71
|
+
text: `${tasks.length} task(s):
|
|
72
|
+
${lines.join("\n")}`
|
|
73
|
+
}
|
|
74
|
+
]
|
|
75
|
+
};
|
|
76
|
+
} catch (err) {
|
|
77
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
78
|
+
return {
|
|
79
|
+
content: [{ type: "text", text: `Failed to list tasks: ${msg}` }],
|
|
80
|
+
isError: true
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
);
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
export {
|
|
88
|
+
registerListTasks
|
|
89
|
+
};
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import {
|
|
2
|
+
deactivateBehavior
|
|
3
|
+
} from "./chunk-O5OMH6LI.js";
|
|
4
|
+
import {
|
|
5
|
+
getActiveAgent
|
|
6
|
+
} from "./chunk-BXCQWWJP.js";
|
|
7
|
+
import {
|
|
8
|
+
canCoordinate,
|
|
9
|
+
getClient
|
|
10
|
+
} from "./chunk-CHBHR5W6.js";
|
|
11
|
+
|
|
12
|
+
// src/mcp/tools/deactivate-behavior.ts
|
|
13
|
+
import { z } from "zod";
|
|
14
|
+
function registerDeactivateBehavior(server) {
|
|
15
|
+
server.registerTool(
|
|
16
|
+
"deactivate_behavior",
|
|
17
|
+
{
|
|
18
|
+
title: "Deactivate Behavior",
|
|
19
|
+
description: "Soft-delete a behavior by setting active = 0. RESTRICTED: only coordinator or founder sessions can use this. Use list_behaviors to find the behavior ID first.",
|
|
20
|
+
inputSchema: {
|
|
21
|
+
behavior_id: z.string().describe("UUID of the behavior to deactivate")
|
|
22
|
+
}
|
|
23
|
+
},
|
|
24
|
+
async ({ behavior_id }) => {
|
|
25
|
+
const caller = getActiveAgent();
|
|
26
|
+
try {
|
|
27
|
+
const { StewardGate } = await import("./steward-gate-VLE7OCKO.js");
|
|
28
|
+
const steward = new StewardGate();
|
|
29
|
+
const gateDecision = await steward.checkAuthority(caller.agentId || "unknown", "deactivate_behavior", behavior_id);
|
|
30
|
+
if (!gateDecision.allowed) {
|
|
31
|
+
return { content: [{ type: "text", text: `Steward gate denied: ${gateDecision.reason}` }], isError: true };
|
|
32
|
+
}
|
|
33
|
+
} catch {
|
|
34
|
+
}
|
|
35
|
+
const allowed = canCoordinate(caller.agentId, caller.agentRole);
|
|
36
|
+
if (!allowed) {
|
|
37
|
+
return {
|
|
38
|
+
content: [{
|
|
39
|
+
type: "text",
|
|
40
|
+
text: `Permission denied. Only the coordinator or founder sessions can deactivate behaviors. You are "${caller.agentId}".`
|
|
41
|
+
}],
|
|
42
|
+
isError: true
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
const client = getClient();
|
|
46
|
+
const result = await client.execute({
|
|
47
|
+
sql: `SELECT id, agent_id, content, domain, priority FROM behaviors WHERE id = ?`,
|
|
48
|
+
args: [behavior_id]
|
|
49
|
+
});
|
|
50
|
+
if (result.rows.length === 0) {
|
|
51
|
+
return {
|
|
52
|
+
content: [{
|
|
53
|
+
type: "text",
|
|
54
|
+
text: `Behavior not found: ${behavior_id}`
|
|
55
|
+
}],
|
|
56
|
+
isError: true
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
const row = result.rows[0];
|
|
60
|
+
const wasActive = await deactivateBehavior(behavior_id);
|
|
61
|
+
if (!wasActive) {
|
|
62
|
+
return {
|
|
63
|
+
content: [{
|
|
64
|
+
type: "text",
|
|
65
|
+
text: `Behavior ${behavior_id} was already inactive.`
|
|
66
|
+
}]
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
return {
|
|
70
|
+
content: [{
|
|
71
|
+
type: "text",
|
|
72
|
+
text: `Deactivated behavior for ${row.agent_id}.
|
|
73
|
+
ID: ${row.id}
|
|
74
|
+
Domain: ${row.domain ?? "none"}
|
|
75
|
+
Priority: ${row.priority}
|
|
76
|
+
Content: ${row.content}`
|
|
77
|
+
}]
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
export {
|
|
84
|
+
registerDeactivateBehavior
|
|
85
|
+
};
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import {
|
|
2
|
+
PLATFORM_PROCEDURES,
|
|
3
|
+
PLATFORM_PROCEDURE_TITLES
|
|
4
|
+
} from "./chunk-K77WC6HA.js";
|
|
5
|
+
import {
|
|
6
|
+
getClient
|
|
7
|
+
} from "./chunk-CHBHR5W6.js";
|
|
8
|
+
|
|
9
|
+
// src/lib/global-procedures.ts
|
|
10
|
+
import { randomUUID } from "crypto";
|
|
11
|
+
var _customerCache = "";
|
|
12
|
+
var _cacheLoaded = false;
|
|
13
|
+
var _platformCache = PLATFORM_PROCEDURES.map((p) => `### ${p.title}
|
|
14
|
+
${p.content}`).join("\n\n");
|
|
15
|
+
async function loadGlobalProcedures() {
|
|
16
|
+
const client = getClient();
|
|
17
|
+
const result = await client.execute({
|
|
18
|
+
sql: "SELECT * FROM company_procedures WHERE active = 1 ORDER BY priority ASC, created_at ASC",
|
|
19
|
+
args: []
|
|
20
|
+
});
|
|
21
|
+
const allRows = result.rows;
|
|
22
|
+
const customerOnly = allRows.filter((p) => !PLATFORM_PROCEDURE_TITLES.has(p.title));
|
|
23
|
+
const values = customerOnly.filter((p) => p.kind === "value");
|
|
24
|
+
const procedures = customerOnly.filter((p) => p.kind !== "value");
|
|
25
|
+
const parts = [];
|
|
26
|
+
if (values.length > 0) {
|
|
27
|
+
parts.push("### Company Values\n" + values.map((v) => `- **${v.title}:** ${v.content}`).join("\n"));
|
|
28
|
+
}
|
|
29
|
+
if (procedures.length > 0) {
|
|
30
|
+
parts.push(procedures.map((p) => `### ${p.title}
|
|
31
|
+
${p.content}`).join("\n\n"));
|
|
32
|
+
}
|
|
33
|
+
_customerCache = parts.join("\n\n");
|
|
34
|
+
_cacheLoaded = true;
|
|
35
|
+
return customerOnly;
|
|
36
|
+
}
|
|
37
|
+
function getGlobalProceduresBlock() {
|
|
38
|
+
const sections = [];
|
|
39
|
+
if (_platformCache) sections.push(_platformCache);
|
|
40
|
+
if (_cacheLoaded && _customerCache) sections.push(_customerCache);
|
|
41
|
+
if (sections.length === 0) return "";
|
|
42
|
+
return `## Organization-Wide Procedures (MANDATORY \u2014 supersedes all other rules)
|
|
43
|
+
|
|
44
|
+
${sections.join("\n\n")}
|
|
45
|
+
`;
|
|
46
|
+
}
|
|
47
|
+
async function storeGlobalProcedure(input) {
|
|
48
|
+
const id = randomUUID();
|
|
49
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
50
|
+
const client = getClient();
|
|
51
|
+
await client.execute({
|
|
52
|
+
sql: `INSERT INTO company_procedures (id, title, content, priority, domain, kind, active, created_at, updated_at)
|
|
53
|
+
VALUES (?, ?, ?, ?, ?, ?, 1, ?, ?)`,
|
|
54
|
+
args: [id, input.title, input.content, input.priority ?? "p0", input.domain ?? null, input.kind ?? "procedure", now, now]
|
|
55
|
+
});
|
|
56
|
+
await loadGlobalProcedures();
|
|
57
|
+
return id;
|
|
58
|
+
}
|
|
59
|
+
async function deactivateGlobalProcedure(id) {
|
|
60
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
61
|
+
const client = getClient();
|
|
62
|
+
const result = await client.execute({
|
|
63
|
+
sql: "UPDATE company_procedures SET active = 0, updated_at = ? WHERE id = ?",
|
|
64
|
+
args: [now, id]
|
|
65
|
+
});
|
|
66
|
+
await loadGlobalProcedures();
|
|
67
|
+
return result.rowsAffected > 0;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
export {
|
|
71
|
+
loadGlobalProcedures,
|
|
72
|
+
getGlobalProceduresBlock,
|
|
73
|
+
storeGlobalProcedure,
|
|
74
|
+
deactivateGlobalProcedure
|
|
75
|
+
};
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import {
|
|
2
|
+
processCRMEvent
|
|
3
|
+
} from "./chunk-J6V2DCZK.js";
|
|
4
|
+
|
|
5
|
+
// src/gateway/adapters/crm-webhook.ts
|
|
6
|
+
function parseTwentyWebhook(payload) {
|
|
7
|
+
if (!payload || typeof payload !== "object") return null;
|
|
8
|
+
const p = payload;
|
|
9
|
+
const eventName = p.eventName;
|
|
10
|
+
if (!eventName || typeof eventName !== "string") return null;
|
|
11
|
+
const dotIndex = eventName.indexOf(".");
|
|
12
|
+
if (dotIndex === -1) return null;
|
|
13
|
+
const objectType = eventName.slice(0, dotIndex);
|
|
14
|
+
const eventType = eventName.slice(dotIndex + 1);
|
|
15
|
+
if (!objectType || !eventType) return null;
|
|
16
|
+
const normalizedObjectType = objectType.charAt(0).toUpperCase() + objectType.slice(1);
|
|
17
|
+
const record = p.record ?? {};
|
|
18
|
+
if (typeof record !== "object") return null;
|
|
19
|
+
return {
|
|
20
|
+
eventType,
|
|
21
|
+
objectType: normalizedObjectType,
|
|
22
|
+
record
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
function createCRMWebhookHandler(options) {
|
|
26
|
+
return async (body) => {
|
|
27
|
+
const event = parseTwentyWebhook(body);
|
|
28
|
+
if (!event) {
|
|
29
|
+
console.warn(
|
|
30
|
+
"[crm-webhook] Malformed CRM webhook payload \u2014 skipping"
|
|
31
|
+
);
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
console.log(
|
|
35
|
+
`[crm-webhook] ${event.objectType}.${event.eventType} received`
|
|
36
|
+
);
|
|
37
|
+
try {
|
|
38
|
+
const logs = await processCRMEvent(event, options?.executor, options?.triggers);
|
|
39
|
+
if (logs.length > 0 && options?.onFired) {
|
|
40
|
+
options.onFired(logs);
|
|
41
|
+
}
|
|
42
|
+
} catch (err) {
|
|
43
|
+
console.error(
|
|
44
|
+
"[crm-webhook] Error processing CRM event:",
|
|
45
|
+
err instanceof Error ? err.message : err
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
export {
|
|
52
|
+
parseTwentyWebhook,
|
|
53
|
+
createCRMWebhookHandler
|
|
54
|
+
};
|
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
import {
|
|
2
|
+
employeeSessionName,
|
|
3
|
+
isEmployeeAlive,
|
|
4
|
+
resolveExeSession,
|
|
5
|
+
sendIntercom,
|
|
6
|
+
strictSessionScopeFilter
|
|
7
|
+
} from "./chunk-57UAFTO2.js";
|
|
8
|
+
import {
|
|
9
|
+
parseMessage,
|
|
10
|
+
serializeMessage
|
|
11
|
+
} from "./chunk-4JERP7NT.js";
|
|
12
|
+
import {
|
|
13
|
+
recordOrchestrationEventBestEffort
|
|
14
|
+
} from "./chunk-JP45GCVR.js";
|
|
15
|
+
import {
|
|
16
|
+
getClient
|
|
17
|
+
} from "./chunk-UNCGZXPU.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
|
+
};
|