@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.
Files changed (305) hide show
  1. package/dist/active-agent-BDYXURXQ.js +26 -0
  2. package/dist/active-agent-YWBGAKGU.js +25 -0
  3. package/dist/agentic-ontology-56VHSVS3.js +25 -0
  4. package/dist/backfill-metadata-G46ABBVR.js +597 -0
  5. package/dist/backfill-metadata-VAV27KJK.js +597 -0
  6. package/dist/behaviors-USUTDXVA.js +25 -0
  7. package/dist/bin/agentic-ontology-backfill.js +5 -5
  8. package/dist/bin/agentic-reflection-backfill.js +6 -6
  9. package/dist/bin/agentic-semantic-label.js +5 -5
  10. package/dist/bin/backfill-conversations.js +4 -4
  11. package/dist/bin/backfill-responses.js +4 -4
  12. package/dist/bin/backfill-vectors.js +5 -5
  13. package/dist/bin/bulk-sync-postgres.js +6 -6
  14. package/dist/bin/cc-doctor.js +4 -4
  15. package/dist/bin/cleanup-stale-review-tasks.js +10 -10
  16. package/dist/bin/cli.js +16 -16
  17. package/dist/bin/exe-agent-config.js +3 -3
  18. package/dist/bin/exe-agent.js +4 -4
  19. package/dist/bin/exe-assign.js +5 -5
  20. package/dist/bin/exe-boot.js +17 -17
  21. package/dist/bin/exe-call.js +4 -4
  22. package/dist/bin/exe-cloud.js +4 -4
  23. package/dist/bin/exe-dispatch.js +10 -10
  24. package/dist/bin/exe-doctor.js +1 -1
  25. package/dist/bin/exe-export-behaviors.js +7 -7
  26. package/dist/bin/exe-forget.js +6 -6
  27. package/dist/bin/exe-gateway.js +7 -7
  28. package/dist/bin/exe-healthcheck.js +4 -4
  29. package/dist/bin/exe-heartbeat.js +10 -10
  30. package/dist/bin/exe-kill.js +13 -13
  31. package/dist/bin/exe-launch-agent.js +37 -19
  32. package/dist/bin/exe-new-employee.js +6 -6
  33. package/dist/bin/exe-pending-messages.js +11 -11
  34. package/dist/bin/exe-pending-notifications.js +10 -10
  35. package/dist/bin/exe-pending-reviews.js +10 -10
  36. package/dist/bin/exe-rename.js +4 -4
  37. package/dist/bin/exe-review.js +12 -12
  38. package/dist/bin/exe-search.js +5 -5
  39. package/dist/bin/exe-session-cleanup.js +15 -15
  40. package/dist/bin/exe-settings.js +5 -5
  41. package/dist/bin/exe-start-codex.js +11 -11
  42. package/dist/bin/exe-start-opencode.js +8 -8
  43. package/dist/bin/exe-status.js +11 -11
  44. package/dist/bin/exe-team.js +3 -3
  45. package/dist/bin/git-sweep.js +11 -11
  46. package/dist/bin/graph-backfill.js +4 -4
  47. package/dist/bin/graph-export.js +5 -5
  48. package/dist/bin/import-history.js +7 -7
  49. package/dist/bin/install.js +6 -6
  50. package/dist/bin/intercom-check.js +4 -4
  51. package/dist/bin/mcp-sessions.js +2 -2
  52. package/dist/bin/orchestration-metrics.js +4 -4
  53. package/dist/bin/postgres-agentic-reflection-backfill.js +2 -2
  54. package/dist/bin/postgres-agentic-semantic-backfill.js +1 -1
  55. package/dist/bin/scan-tasks.js +10 -10
  56. package/dist/bin/setup.js +1 -1
  57. package/dist/bin/shard-migrate.js +4 -4
  58. package/dist/capacity-monitor-IFVRCIM7.js +49 -0
  59. package/dist/capacity-monitor-Q47GBDSY.js +49 -0
  60. package/dist/catchup-brief-RP4QHXNT.js +151 -0
  61. package/dist/catchup-brief-TKA6TEK4.js +151 -0
  62. package/dist/chunk-23KJ2LXY.js +58 -0
  63. package/dist/chunk-2NQQP3FF.js +630 -0
  64. package/dist/chunk-3FU5I3KV.js +526 -0
  65. package/dist/chunk-46IEEKPU.js +13696 -0
  66. package/dist/chunk-57UAFTO2.js +3958 -0
  67. package/dist/chunk-5AS622MM.js +3958 -0
  68. package/dist/chunk-62DEE65H.js +371 -0
  69. package/dist/chunk-64T6DFSS.js +447 -0
  70. package/dist/chunk-6BWDP63Z.js +197 -0
  71. package/dist/chunk-6GPYL7TX.js +214 -0
  72. package/dist/chunk-6KWLUVFL.js +54 -0
  73. package/dist/chunk-6N5ISWBF.js +1148 -0
  74. package/dist/chunk-6OD7PVMC.js +333 -0
  75. package/dist/chunk-7ET5CYTD.js +382 -0
  76. package/dist/chunk-7IWLKR6N.js +76 -0
  77. package/dist/chunk-7OEUOJL5.js +1021 -0
  78. package/dist/chunk-AIRJTKDK.js +204 -0
  79. package/dist/chunk-ATJ3NXDP.js +244 -0
  80. package/dist/chunk-B5IS7LE4.js +128 -0
  81. package/dist/chunk-BKINEQVI.js +244 -0
  82. package/dist/chunk-BOJV6NI3.js +128 -0
  83. package/dist/chunk-BPHWI6N2.js +284 -0
  84. package/dist/chunk-BXCQWWJP.js +185 -0
  85. package/dist/chunk-BZ6K7AY3.js +50 -0
  86. package/dist/chunk-C54KIFLS.js +214 -0
  87. package/dist/chunk-C6OYEJJI.js +260 -0
  88. package/dist/chunk-CHBHR5W6.js +3556 -0
  89. package/dist/chunk-D2T3272U.js +171 -0
  90. package/dist/chunk-DCHEIVGT.js +221 -0
  91. package/dist/chunk-DHIBLMSP.js +30 -0
  92. package/dist/chunk-E2AF2WYY.js +346 -0
  93. package/dist/chunk-E2KZEZZW.js +1090 -0
  94. package/dist/chunk-E4KWB4WM.js +348 -0
  95. package/dist/chunk-ENU7URWK.js +1073 -0
  96. package/dist/chunk-EZ7KAZMC.js +132 -0
  97. package/dist/chunk-F4FSSHR4.js +1073 -0
  98. package/dist/chunk-FPXU56FG.js +346 -0
  99. package/dist/chunk-FY7HHR5I.js +128 -0
  100. package/dist/chunk-G2S2UMU4.js +159 -0
  101. package/dist/chunk-G33BHQCO.js +70 -0
  102. package/dist/chunk-GJQTL7RX.js +127 -0
  103. package/dist/chunk-GLCKDEM2.js +97 -0
  104. package/dist/chunk-GMA34SXV.js +240 -0
  105. package/dist/chunk-GVAVEBYR.js +2091 -0
  106. package/dist/chunk-HOSJTLBQ.js +513 -0
  107. package/dist/chunk-IC6HVAS3.js +56 -0
  108. package/dist/chunk-IDFJNO44.js +1051 -0
  109. package/dist/chunk-ILFJMEY5.js +97 -0
  110. package/dist/chunk-ISQAOSL3.js +1921 -0
  111. package/dist/chunk-J6V2DCZK.js +382 -0
  112. package/dist/chunk-JP4CLFLR.js +1148 -0
  113. package/dist/chunk-JTIOZHWG.js +58 -0
  114. package/dist/chunk-KDICWAYV.js +1345 -0
  115. package/dist/chunk-KOBIB6WG.js +159 -0
  116. package/dist/chunk-KQFDDQB6.js +13696 -0
  117. package/dist/chunk-KZNSOHCB.js +280 -0
  118. package/dist/chunk-LVMBYP3C.js +171 -0
  119. package/dist/chunk-M2WQW5NC.js +227 -0
  120. package/dist/chunk-MY6SP5NZ.js +551 -0
  121. package/dist/chunk-N2ACW2ZG.js +363 -0
  122. package/dist/chunk-NSMJDATI.js +495 -0
  123. package/dist/chunk-NSQ5JE23.js +1090 -0
  124. package/dist/chunk-NZL567WG.js +81 -0
  125. package/dist/chunk-O5OMH6LI.js +244 -0
  126. package/dist/chunk-OBUV3W7L.js +163 -0
  127. package/dist/chunk-OLDS7LJN.js +495 -0
  128. package/dist/chunk-OO2I22RX.js +38 -0
  129. package/dist/chunk-OPUUT33V.js +447 -0
  130. package/dist/chunk-OT3VMTKB.js +50 -0
  131. package/dist/chunk-P6RVIOVA.js +157 -0
  132. package/dist/chunk-PUA5564C.js +210 -0
  133. package/dist/chunk-PUQLKLQX.js +731 -0
  134. package/dist/chunk-QROKS65G.js +76 -0
  135. package/dist/chunk-R54I2N2T.js +818 -0
  136. package/dist/chunk-RCFYQHUP.js +818 -0
  137. package/dist/chunk-RJTND4YS.js +284 -0
  138. package/dist/chunk-SBLHQMMZ.js +81 -0
  139. package/dist/chunk-SG2ANG5C.js +123 -0
  140. package/dist/chunk-SVFNKSZV.js +333 -0
  141. package/dist/chunk-TAQT2DC7.js +330 -0
  142. package/dist/chunk-TB7HFW7M.js +127 -0
  143. package/dist/chunk-UUKDAIH2.js +731 -0
  144. package/dist/chunk-V6VEFEEH.js +1345 -0
  145. package/dist/chunk-VIO2ALGH.js +290 -0
  146. package/dist/chunk-VKCUSNJW.js +377 -0
  147. package/dist/chunk-VRPPJFIQ.js +1921 -0
  148. package/dist/chunk-WP3PVBBP.js +204 -0
  149. package/dist/chunk-WQEUY7DC.js +129 -0
  150. package/dist/chunk-WXMXUKCA.js +262 -0
  151. package/dist/chunk-X2WBH2IO.js +297 -0
  152. package/dist/chunk-X33TSJNO.js +394 -0
  153. package/dist/chunk-X7MMI2UI.js +89 -0
  154. package/dist/chunk-XG3BQZIK.js +85 -0
  155. package/dist/chunk-XIKBIAOS.js +75 -0
  156. package/dist/chunk-XPEB545Q.js +54 -0
  157. package/dist/chunk-XWH2MLWS.js +330 -0
  158. package/dist/chunk-YH7V73XW.js +89 -0
  159. package/dist/chunk-YMLM5D65.js +135 -0
  160. package/dist/chunk-YNJPRQ6J.js +377 -0
  161. package/dist/chunk-YSNEHBI6.js +551 -0
  162. package/dist/chunk-ZD6BMW2K.js +33 -0
  163. package/dist/chunk-ZKG5IYCG.js +668 -0
  164. package/dist/chunk-ZU4K7ZNX.js +197 -0
  165. package/dist/co-activation-HZMJC34P.js +72 -0
  166. package/dist/co-occurrence-AVYXRV4L.js +74 -0
  167. package/dist/core-memory-NPJCVUMF.js +110 -0
  168. package/dist/core-memory-OKGXL33Z.js +110 -0
  169. package/dist/crdt-sync-ZCH55JNR.js +33 -0
  170. package/dist/crm-webhook-6OMVUUGR.js +10 -0
  171. package/dist/crm-webhook-UCWF3XDB.js +10 -0
  172. package/dist/cto-delegation-gate-JFZFZGC2.js +206 -0
  173. package/dist/cto-delegation-gate-K32M4GVM.js +206 -0
  174. package/dist/daemon-orchestration-2Q7BYOHC.js +135 -0
  175. package/dist/daemon-orchestration-4RJ2CZJL.js +135 -0
  176. package/dist/db-backup-5GA2YFDX.js +33 -0
  177. package/dist/dreaming-I6KXO6E2.js +32 -0
  178. package/dist/dreaming-NJBK5ILR.js +32 -0
  179. package/dist/exe-drift-VSMIMHL4.js +68 -0
  180. package/dist/exe-export-DVHHIA6Y.js +73 -0
  181. package/dist/exe-export-GIVQDENS.js +73 -0
  182. package/dist/exe-import-7N46LSMQ.js +76 -0
  183. package/dist/exe-import-FINYUV5T.js +76 -0
  184. package/dist/exe-key-H45JY44F.js +579 -0
  185. package/dist/exe-key-MAEQGTB7.js +579 -0
  186. package/dist/exe-snapshot-3TEM3BFD.js +164 -0
  187. package/dist/exe-snapshot-HECGUHL3.js +164 -0
  188. package/dist/fast-db-init-3CNTADVO.js +7 -0
  189. package/dist/fast-db-init-HXCS2AP5.js +7 -0
  190. package/dist/gateway/index.js +8 -8
  191. package/dist/git-staleness-YCEBBIVK.js +110 -0
  192. package/dist/git-task-sweep-H34STRNT.js +40 -0
  193. package/dist/git-task-sweep-YL7NLDCK.js +40 -0
  194. package/dist/global-procedures-IHZM6C2K.js +20 -0
  195. package/dist/graph-auto-extract-RZQ3MHP2.js +162 -0
  196. package/dist/hooks/bug-report-worker.js +12 -12
  197. package/dist/hooks/codex-stop-task-finalizer.js +12 -12
  198. package/dist/hooks/commit-complete.js +12 -12
  199. package/dist/hooks/error-recall.js +6 -6
  200. package/dist/hooks/exe-heartbeat-hook.js +3 -3
  201. package/dist/hooks/ingest.js +6 -6
  202. package/dist/hooks/instructions-loaded.js +4 -4
  203. package/dist/hooks/manifest.json +19 -19
  204. package/dist/hooks/notification.js +4 -4
  205. package/dist/hooks/post-compact.js +11 -11
  206. package/dist/hooks/post-tool-combined.js +5 -5
  207. package/dist/hooks/pre-compact.js +12 -12
  208. package/dist/hooks/pre-tool-use.js +15 -15
  209. package/dist/hooks/prompt-submit.js +21 -21
  210. package/dist/hooks/session-end.js +16 -16
  211. package/dist/hooks/session-start.js +10 -10
  212. package/dist/hooks/stop.js +15 -15
  213. package/dist/hooks/subagent-stop.js +11 -11
  214. package/dist/hooks/summary-worker.js +15 -15
  215. package/dist/index.js +18 -18
  216. package/dist/installer-4EW5ZDGD.js +296 -0
  217. package/dist/installer-B2JTQO55.js +38 -0
  218. package/dist/installer-MIL352T7.js +342 -0
  219. package/dist/lib/agent-config.js +9 -3
  220. package/dist/lib/cloud-sync.js +4 -4
  221. package/dist/lib/consolidation.js +5 -5
  222. package/dist/lib/database.js +2 -2
  223. package/dist/lib/db.js +2 -2
  224. package/dist/lib/employee-templates.js +4 -4
  225. package/dist/lib/employees.js +2 -2
  226. package/dist/lib/exe-daemon.js +34 -34
  227. package/dist/lib/hybrid-search.js +5 -5
  228. package/dist/lib/identity.js +2 -2
  229. package/dist/lib/messaging.js +10 -10
  230. package/dist/lib/reminders.js +3 -3
  231. package/dist/lib/schedules.js +5 -5
  232. package/dist/lib/session-registry.js +4 -4
  233. package/dist/lib/skill-learning.js +4 -4
  234. package/dist/lib/store.js +4 -4
  235. package/dist/lib/task-router.js +3 -3
  236. package/dist/lib/tasks.js +11 -11
  237. package/dist/lib/tmux-routing.js +9 -9
  238. package/dist/lib/token-spend.js +3 -3
  239. package/dist/mcp/register-tools.js +54 -54
  240. package/dist/mcp/server.js +55 -55
  241. package/dist/mcp/tools/complete-reminder.js +4 -4
  242. package/dist/mcp/tools/create-reminder.js +4 -4
  243. package/dist/mcp/tools/create-task.js +13 -13
  244. package/dist/mcp/tools/deactivate-behavior.js +5 -5
  245. package/dist/mcp/tools/list-reminders.js +4 -4
  246. package/dist/mcp/tools/list-tasks.js +13 -13
  247. package/dist/mcp/tools/send-message.js +12 -12
  248. package/dist/mcp/tools/update-task.js +12 -12
  249. package/dist/mcp-http-config-OJQR246S.js +27 -0
  250. package/dist/memory-cards-IPULSQFA.js +174 -0
  251. package/dist/memory-graph-extractor-3TZZOKHY.js +17 -0
  252. package/dist/memory-poisoning-defense-SGUGR5YJ.js +225 -0
  253. package/dist/memory-reflection-H3WGCEM6.js +238 -0
  254. package/dist/notifications-VWPO6NJF.js +45 -0
  255. package/dist/notifications-WCSRQN2V.js +45 -0
  256. package/dist/orchestration-events-O5PSDEIO.js +25 -0
  257. package/dist/orchestrator-RAPEJUOI.js +33 -0
  258. package/dist/orchestrator-XPG6LJAI.js +33 -0
  259. package/dist/pipeline-router-5NT6FUC3.js +13 -0
  260. package/dist/pipeline-router-KSUXONDT.js +13 -0
  261. package/dist/plan-limits-53NXLNDQ.js +26 -0
  262. package/dist/project-boot-ITN3FZMM.js +299 -0
  263. package/dist/projection-worker-27XX5M2W.js +964 -0
  264. package/dist/reranker-GU7L2PJX.js +19 -0
  265. package/dist/reranker-TZEXIJAN.js +19 -0
  266. package/dist/review-polling-FA2J2Q5O.js +124 -0
  267. package/dist/review-polling-MLS4BQ3N.js +124 -0
  268. package/dist/runtime/index.js +12 -12
  269. package/dist/session-events-WWGF3B2N.js +36 -0
  270. package/dist/session-events-ZHXXAH6B.js +36 -0
  271. package/dist/session-kill-telemetry-O4TJHHOZ.js +29 -0
  272. package/dist/session-scope-CQXB7VMH.js +86 -0
  273. package/dist/session-scope-HHUMJYF6.js +86 -0
  274. package/dist/setup-wizard-UM2RHSBJ.js +12 -0
  275. package/dist/skill-refinement-MJPOHYD5.js +157 -0
  276. package/dist/skill-refinement-NVUBRK22.js +157 -0
  277. package/dist/stack-release-BAPCXMXW.js +713 -0
  278. package/dist/stack-release-W4TWTEZP.js +731 -0
  279. package/dist/steward-gate-VLE7OCKO.js +13 -0
  280. package/dist/task-enforcement-FUHDL6UR.js +391 -0
  281. package/dist/task-enforcement-QL3K4N3F.js +391 -0
  282. package/dist/task-scope-TZYMB634.js +35 -0
  283. package/dist/task-scope-ZVLUBS4C.js +35 -0
  284. package/dist/tasks-crud-4MSLJWXE.js +77 -0
  285. package/dist/tasks-crud-HIPXKRKX.js +77 -0
  286. package/dist/tasks-notify-7JBUNE7R.js +38 -0
  287. package/dist/tasks-notify-UPIJ3L4O.js +38 -0
  288. package/dist/tasks-review-5SJSFTUB.js +47 -0
  289. package/dist/tasks-review-JHSYBR5I.js +47 -0
  290. package/dist/telemetry-upload-BSGOXGUP.js +739 -0
  291. package/dist/telemetry-upload-LTX3C5HZ.js +739 -0
  292. package/dist/token-budget-2CDWQU3Q.js +84 -0
  293. package/dist/tool-telemetry-7YS7EN7B.js +17 -0
  294. package/dist/tui/App.js +17 -17
  295. package/dist/tui-data-GDGBOS6G.js +258 -0
  296. package/dist/tui-data-VXF2RBVM.js +258 -0
  297. package/dist/wiki-acl-MJIMXRQV.js +111 -0
  298. package/dist/worker-gate-WQGTZOSM.js +21 -0
  299. package/dist/worker-gate-X2YDTKTL.js +21 -0
  300. package/dist/workflow-engine-CYXRZXBM.js +28 -0
  301. package/dist/workflow-engine-KMLAXVA4.js +28 -0
  302. package/dist/worktree-NLSKVRNC.js +26 -0
  303. package/dist/worktree-sweep-44TMEPLE.js +19 -0
  304. package/package.json +1 -1
  305. package/release-notes.json +24 -19
@@ -0,0 +1,185 @@
1
+ import {
2
+ getSessionKey
3
+ } from "./chunk-CVYC6DUW.js";
4
+ import {
5
+ getAgentContext
6
+ } from "./chunk-GJV3WDWM.js";
7
+ import {
8
+ DEFAULT_COORDINATOR_TEMPLATE_NAME,
9
+ getCoordinatorEmployee,
10
+ getEmployee,
11
+ loadEmployeesSync
12
+ } from "./chunk-CHBHR5W6.js";
13
+ import {
14
+ EXE_AI_DIR
15
+ } from "./chunk-VXIMSRTO.js";
16
+
17
+ // src/lib/active-agent.ts
18
+ import { readFileSync, writeFileSync, mkdirSync, unlinkSync, readdirSync } from "fs";
19
+ import { execSync } from "child_process";
20
+ import path from "path";
21
+ var CACHE_DIR = path.join(EXE_AI_DIR, "session-cache");
22
+ var STALE_MS = 24 * 60 * 60 * 1e3;
23
+ function isNameWithOptionalInstance(candidate, baseName) {
24
+ if (candidate === baseName) return true;
25
+ if (!candidate.startsWith(baseName)) return false;
26
+ return /^\d+$/.test(candidate.slice(baseName.length));
27
+ }
28
+ function resolveEmployeeFromSessionPrefix(prefix, employees) {
29
+ const sorted = [...employees].sort((a, b) => b.name.length - a.name.length);
30
+ for (const employee of sorted) {
31
+ if (isNameWithOptionalInstance(prefix, employee.name)) {
32
+ return { agentId: employee.name, agentRole: employee.role };
33
+ }
34
+ }
35
+ return null;
36
+ }
37
+ function resolveActiveAgentFromTmuxSession(sessionName) {
38
+ const employees = loadEmployeesSync();
39
+ const coordinator = getCoordinatorEmployee(employees);
40
+ const coordinatorName = coordinator?.name ?? DEFAULT_COORDINATOR_TEMPLATE_NAME;
41
+ if (isNameWithOptionalInstance(sessionName, coordinatorName)) {
42
+ return {
43
+ agentId: coordinatorName,
44
+ agentRole: coordinator?.role ?? "COO"
45
+ };
46
+ }
47
+ if (isNameWithOptionalInstance(sessionName, DEFAULT_COORDINATOR_TEMPLATE_NAME)) {
48
+ return {
49
+ agentId: coordinator?.name ?? DEFAULT_COORDINATOR_TEMPLATE_NAME,
50
+ agentRole: coordinator?.role ?? "COO"
51
+ };
52
+ }
53
+ if (sessionName.includes("-")) {
54
+ const prefix = sessionName.split("-")[0] ?? "";
55
+ const employee = resolveEmployeeFromSessionPrefix(prefix, employees);
56
+ if (employee) return employee;
57
+ const legacy = prefix.match(/^([a-zA-Z]+)\d*$/);
58
+ if (legacy?.[1] && legacy[1] !== DEFAULT_COORDINATOR_TEMPLATE_NAME) {
59
+ const emp = getEmployee(employees, legacy[1]);
60
+ return { agentId: emp?.name ?? legacy[1], agentRole: emp?.role ?? "employee" };
61
+ }
62
+ }
63
+ return null;
64
+ }
65
+ function getMarkerPath() {
66
+ return path.join(CACHE_DIR, `active-agent-${getSessionKey()}.json`);
67
+ }
68
+ function writeActiveAgent(agentId, agentRole) {
69
+ try {
70
+ mkdirSync(CACHE_DIR, { recursive: true });
71
+ writeFileSync(
72
+ getMarkerPath(),
73
+ JSON.stringify({ agentId, agentRole, startedAt: (/* @__PURE__ */ new Date()).toISOString() })
74
+ );
75
+ } catch {
76
+ }
77
+ }
78
+ function clearActiveAgent() {
79
+ try {
80
+ unlinkSync(getMarkerPath());
81
+ } catch {
82
+ }
83
+ }
84
+ function getActiveAgent() {
85
+ const httpCtx = getAgentContext();
86
+ if (httpCtx) return httpCtx;
87
+ try {
88
+ const markerPath = getMarkerPath();
89
+ const raw = readFileSync(markerPath, "utf8");
90
+ const data = JSON.parse(raw);
91
+ if (data.agentId) {
92
+ if (data.startedAt) {
93
+ const age = Date.now() - new Date(data.startedAt).getTime();
94
+ if (age > STALE_MS) {
95
+ try {
96
+ unlinkSync(markerPath);
97
+ } catch {
98
+ }
99
+ } else {
100
+ return {
101
+ agentId: data.agentId,
102
+ agentRole: data.agentRole || "employee"
103
+ };
104
+ }
105
+ } else {
106
+ return {
107
+ agentId: data.agentId,
108
+ agentRole: data.agentRole || "employee"
109
+ };
110
+ }
111
+ }
112
+ } catch {
113
+ }
114
+ if (process.env.EXE_TEST_HERMETIC !== "1") {
115
+ try {
116
+ const sessionName = execSync(
117
+ "tmux display-message -p '#{session_name}' 2>/dev/null",
118
+ { encoding: "utf8", timeout: 2e3 }
119
+ ).trim();
120
+ const resolved = resolveActiveAgentFromTmuxSession(sessionName);
121
+ if (resolved) return resolved;
122
+ } catch {
123
+ }
124
+ }
125
+ return {
126
+ agentId: process.env.AGENT_ID || "default",
127
+ agentRole: process.env.AGENT_ROLE || "employee"
128
+ };
129
+ }
130
+ function getAllActiveAgents() {
131
+ try {
132
+ const files = readdirSync(CACHE_DIR);
133
+ const sessions = [];
134
+ for (const file of files) {
135
+ if (!file.startsWith("active-agent-") || !file.endsWith(".json")) continue;
136
+ const key = file.slice("active-agent-".length, -".json".length);
137
+ if (key === "undefined") continue;
138
+ try {
139
+ const raw = readFileSync(path.join(CACHE_DIR, file), "utf8");
140
+ const data = JSON.parse(raw);
141
+ if (!data.agentId) continue;
142
+ if (data.startedAt) {
143
+ const age = Date.now() - new Date(data.startedAt).getTime();
144
+ if (age > STALE_MS) {
145
+ try {
146
+ unlinkSync(path.join(CACHE_DIR, file));
147
+ } catch {
148
+ }
149
+ continue;
150
+ }
151
+ }
152
+ sessions.push({
153
+ agentId: data.agentId,
154
+ agentRole: data.agentRole || "employee",
155
+ startedAt: data.startedAt || (/* @__PURE__ */ new Date()).toISOString(),
156
+ sessionKey: key
157
+ });
158
+ } catch {
159
+ }
160
+ }
161
+ return sessions;
162
+ } catch {
163
+ return [];
164
+ }
165
+ }
166
+ function cleanupSessionMarkers() {
167
+ const key = getSessionKey();
168
+ try {
169
+ unlinkSync(path.join(CACHE_DIR, `active-agent-${key}.json`));
170
+ } catch {
171
+ }
172
+ try {
173
+ unlinkSync(path.join(CACHE_DIR, "active-agent-undefined.json"));
174
+ } catch {
175
+ }
176
+ }
177
+
178
+ export {
179
+ resolveActiveAgentFromTmuxSession,
180
+ writeActiveAgent,
181
+ clearActiveAgent,
182
+ getActiveAgent,
183
+ getAllActiveAgents,
184
+ cleanupSessionMarkers
185
+ };
@@ -0,0 +1,50 @@
1
+ import {
2
+ sendMessage
3
+ } from "./chunk-TAQT2DC7.js";
4
+ import {
5
+ getActiveAgent
6
+ } from "./chunk-BXCQWWJP.js";
7
+
8
+ // src/mcp/tools/send-message.ts
9
+ import { z } from "zod";
10
+ function registerSendMessage(server) {
11
+ server.registerTool(
12
+ "send_message",
13
+ {
14
+ title: "Send Message",
15
+ description: "Send a structured message to another agent. Messages are queued and delivered via intercom. NOTE: messages are fire-and-forget \u2014 do NOT use for actionable work dispatch. Use create_task instead to assign work to employees.",
16
+ inputSchema: {
17
+ target_agent: z.string().describe("Recipient agent name"),
18
+ content: z.string().describe("Message content"),
19
+ target_project: z.string().optional().describe("Project context (optional)"),
20
+ priority: z.enum(["normal", "urgent"]).default("normal").describe("Message priority (default: normal)"),
21
+ target_device: z.string().optional().describe("Target device ID for cross-machine delivery (default: local)")
22
+ }
23
+ },
24
+ async ({ target_agent, content, target_project, priority, target_device }) => {
25
+ const { agentId } = getActiveAgent();
26
+ const msg = await sendMessage({
27
+ fromAgent: agentId,
28
+ targetAgent: target_agent,
29
+ targetProject: target_project,
30
+ targetDevice: target_device,
31
+ content,
32
+ priority
33
+ });
34
+ const statusText = msg.status === "delivered" ? "delivered" : msg.status === "failed" ? `failed: ${msg.failureReason}` : "queued (pending delivery)";
35
+ return {
36
+ content: [
37
+ {
38
+ type: "text",
39
+ text: `Message sent to ${target_agent}: ${statusText}
40
+ ID: ${msg.id}`
41
+ }
42
+ ]
43
+ };
44
+ }
45
+ );
46
+ }
47
+
48
+ export {
49
+ registerSendMessage
50
+ };
@@ -0,0 +1,214 @@
1
+ import {
2
+ sessionScopeFilter
3
+ } from "./chunk-57UAFTO2.js";
4
+
5
+ // src/lib/git-task-sweep.ts
6
+ import { execSync } from "child_process";
7
+ var DEFAULT_COMMIT_LIMIT = 50;
8
+ var DEFAULT_STALE_MINUTES = 10;
9
+ var AUTO_ESCALATE_THRESHOLD = 0.6;
10
+ var EXACT_UUID_SCORE = 1;
11
+ var TITLE_KEYWORD_SCORE = 0.5;
12
+ var FILE_PATH_SCORE = 0.3;
13
+ var MIN_KEYWORD_OVERLAP = 3;
14
+ var STOP_WORDS = /* @__PURE__ */ new Set([
15
+ "a",
16
+ "an",
17
+ "the",
18
+ "and",
19
+ "or",
20
+ "but",
21
+ "in",
22
+ "on",
23
+ "at",
24
+ "to",
25
+ "for",
26
+ "of",
27
+ "with",
28
+ "by",
29
+ "from",
30
+ "is",
31
+ "it",
32
+ "as",
33
+ "be",
34
+ "was",
35
+ "are",
36
+ "this",
37
+ "that",
38
+ "not",
39
+ "no",
40
+ "if",
41
+ "so",
42
+ "do",
43
+ "up"
44
+ ]);
45
+ function extractKeywords(text) {
46
+ return text.toLowerCase().replace(/[^a-z0-9\s-]/g, " ").split(/\s+/).filter((w) => w.length >= 3 && !STOP_WORDS.has(w));
47
+ }
48
+ function matchScore(task, commitMessage, changedFiles) {
49
+ if (task.id.length >= 8 && commitMessage.includes(task.id)) {
50
+ return EXACT_UUID_SCORE;
51
+ }
52
+ let score = 0;
53
+ const titleWords = extractKeywords(task.title);
54
+ const commitWords = new Set(extractKeywords(commitMessage));
55
+ const overlap = titleWords.filter((w) => commitWords.has(w));
56
+ if (overlap.length >= MIN_KEYWORD_OVERLAP) {
57
+ score += TITLE_KEYWORD_SCORE;
58
+ }
59
+ if (task.context && changedFiles.length > 0) {
60
+ const contextLower = task.context.toLowerCase();
61
+ const hasFileMatch = changedFiles.some(
62
+ (f) => contextLower.includes(f.toLowerCase())
63
+ );
64
+ if (hasFileMatch) {
65
+ score += FILE_PATH_SCORE;
66
+ }
67
+ }
68
+ return score;
69
+ }
70
+ function getRecentCommits(limit = DEFAULT_COMMIT_LIMIT) {
71
+ try {
72
+ const SEPARATOR = "<<SEP>>";
73
+ const output = execSync(
74
+ `git log --format="%h${SEPARATOR}%s${SEPARATOR}%aI" --name-only -n ${limit} -z`,
75
+ { encoding: "utf8", timeout: 1e4 }
76
+ );
77
+ const entries = output.split("\0").filter(Boolean);
78
+ const commits = [];
79
+ let current = null;
80
+ for (const entry of entries) {
81
+ if (entry.includes(SEPARATOR)) {
82
+ const lines = entry.split("\n");
83
+ const headerLine = lines[0];
84
+ const parts = headerLine.split(SEPARATOR);
85
+ if (parts.length >= 3) {
86
+ if (current) commits.push(current);
87
+ current = {
88
+ hash: parts[0],
89
+ message: parts[1],
90
+ files: lines.slice(1).filter(Boolean),
91
+ date: new Date(parts[2])
92
+ };
93
+ }
94
+ } else if (current) {
95
+ const files = entry.split("\n").filter(Boolean);
96
+ current.files.push(...files);
97
+ }
98
+ }
99
+ if (current) commits.push(current);
100
+ return commits;
101
+ } catch {
102
+ return [];
103
+ }
104
+ }
105
+ function isStale(updatedAt, staleMinutes) {
106
+ const updated = new Date(updatedAt).getTime();
107
+ const threshold = Date.now() - staleMinutes * 6e4;
108
+ return updated < threshold;
109
+ }
110
+ function findBestMatch(task, commits) {
111
+ let best = null;
112
+ for (const commit of commits) {
113
+ const score = matchScore(task, commit.message, commit.files);
114
+ if (score >= AUTO_ESCALATE_THRESHOLD && (!best || score > best.score)) {
115
+ best = { commit, score };
116
+ }
117
+ }
118
+ return best;
119
+ }
120
+ async function sweepTasks(projectName, options = {}) {
121
+ const commitLimit = options.commitLimit ?? DEFAULT_COMMIT_LIMIT;
122
+ const staleMinutes = options.staleMinutes ?? DEFAULT_STALE_MINUTES;
123
+ const dryRun = options.dryRun ?? false;
124
+ const result = { escalated: [], unchanged: 0, errors: [] };
125
+ const commits = getRecentCommits(commitLimit);
126
+ if (commits.length === 0) {
127
+ result.errors.push("No git commits found (not a git repo or empty history)");
128
+ return result;
129
+ }
130
+ let tasks;
131
+ try {
132
+ const { initStore } = await import("./lib/store.js");
133
+ await initStore();
134
+ const { getClient } = await import("./lib/database.js");
135
+ const client = getClient();
136
+ const conditions = ["status = 'in_progress'"];
137
+ const args = [];
138
+ if (projectName) {
139
+ conditions.push("project_name = ?");
140
+ args.push(projectName);
141
+ }
142
+ const swScope = sessionScopeFilter();
143
+ if (swScope.sql) {
144
+ conditions.push("(session_scope IS NULL OR session_scope = ?)");
145
+ args.push(...swScope.args);
146
+ }
147
+ const queryResult = await client.execute({
148
+ sql: `SELECT id, title, assigned_to, project_name, status, updated_at, context
149
+ FROM tasks WHERE ${conditions.join(" AND ")}
150
+ ORDER BY updated_at ASC`,
151
+ args
152
+ });
153
+ tasks = queryResult.rows.map((r) => ({
154
+ id: String(r.id),
155
+ title: String(r.title),
156
+ assignedTo: String(r.assigned_to),
157
+ projectName: String(r.project_name),
158
+ status: String(r.status),
159
+ updatedAt: String(r.updated_at),
160
+ context: r.context ? String(r.context) : void 0
161
+ }));
162
+ } catch (err) {
163
+ result.errors.push(`DB query failed: ${err instanceof Error ? err.message : String(err)}`);
164
+ return result;
165
+ }
166
+ if (tasks.length === 0) {
167
+ return result;
168
+ }
169
+ for (const task of tasks) {
170
+ if (!isStale(task.updatedAt, staleMinutes)) {
171
+ result.unchanged++;
172
+ continue;
173
+ }
174
+ const match = findBestMatch(task, commits);
175
+ if (!match) {
176
+ result.unchanged++;
177
+ continue;
178
+ }
179
+ if (!dryRun) {
180
+ try {
181
+ const { updateTaskStatus } = await import("./tasks-crud-4MSLJWXE.js");
182
+ await updateTaskStatus({
183
+ taskId: task.id,
184
+ status: "needs_review",
185
+ result: `Auto-escalated by git-sweep: matching commit ${match.commit.hash} found (score: ${match.score.toFixed(2)})`
186
+ });
187
+ } catch (err) {
188
+ result.errors.push(
189
+ `Failed to escalate task ${task.id}: ${err instanceof Error ? err.message : String(err)}`
190
+ );
191
+ continue;
192
+ }
193
+ }
194
+ result.escalated.push({
195
+ taskId: task.id,
196
+ title: task.title,
197
+ matchedCommit: match.commit.hash,
198
+ score: match.score
199
+ });
200
+ process.stderr.write(
201
+ `[git-sweep] ${dryRun ? "WOULD escalate" : "Escalated"} task ${task.id} \u2192 commit ${match.commit.hash} (score: ${match.score.toFixed(2)})
202
+ `
203
+ );
204
+ }
205
+ return result;
206
+ }
207
+
208
+ export {
209
+ extractKeywords,
210
+ matchScore,
211
+ getRecentCommits,
212
+ findBestMatch,
213
+ sweepTasks
214
+ };
@@ -0,0 +1,260 @@
1
+ import {
2
+ EXE_AI_DIR
3
+ } from "./chunk-VXIMSRTO.js";
4
+
5
+ // src/lib/db-backup.ts
6
+ import { existsSync, readdirSync, unlinkSync, statSync, statfsSync } from "fs";
7
+ import { copyFile, mkdir, unlink } from "fs/promises";
8
+ import path from "path";
9
+ var BACKUP_DIR = path.join(EXE_AI_DIR, "backups");
10
+ var DEFAULT_KEEP_DAYS = 3;
11
+ var DB_NAMES = ["memories.db", "exe-mem.db", "exe-os.db", "exe.db"];
12
+ var REASON_COUNT_CAP = {
13
+ "pre-restart": 5,
14
+ "pre-consolidation": 5,
15
+ "pre-fix": 5,
16
+ "pre-restore": 5
17
+ };
18
+ var MIN_FREE_BYTES = 5 * 1024 * 1024 * 1024;
19
+ function findActiveDb() {
20
+ for (const name of DB_NAMES) {
21
+ const p = path.join(EXE_AI_DIR, name);
22
+ if (existsSync(p)) return p;
23
+ }
24
+ return null;
25
+ }
26
+ async function createBackup(reason = "manual") {
27
+ return createBackupAsync(reason);
28
+ }
29
+ async function createBackupAsync(reason = "manual") {
30
+ const dbPath = findActiveDb();
31
+ if (!dbPath) return null;
32
+ await mkdir(BACKUP_DIR, { recursive: true });
33
+ try {
34
+ let dbSize = 0;
35
+ try {
36
+ dbSize = statSync(dbPath).size;
37
+ } catch {
38
+ }
39
+ const freeBytes = getFreeBytes(BACKUP_DIR);
40
+ if (!hasEnoughDiskSpace(freeBytes, dbSize)) {
41
+ process.stderr.write(
42
+ `[db-backup] DISK GUARD: skipping "${reason}" backup \u2014 ${Math.round(freeBytes / 1024 / 1024)}MB free, need >${Math.round(Math.max(MIN_FREE_BYTES, dbSize * 2) / 1024 / 1024)}MB. Disk is nearly full; refusing to make it worse.
43
+ `
44
+ );
45
+ return null;
46
+ }
47
+ } catch {
48
+ }
49
+ const dbName = path.basename(dbPath, ".db");
50
+ const timestamp = (/* @__PURE__ */ new Date()).toISOString().replace(/[:.]/g, "-").slice(0, 19);
51
+ const backupName = `${dbName}-${reason}-${timestamp}.db`;
52
+ const backupPath = path.join(BACKUP_DIR, backupName);
53
+ try {
54
+ const { getClient } = await import("./lib/database.js");
55
+ const client = getClient();
56
+ await client.execute("PRAGMA wal_checkpoint(TRUNCATE)");
57
+ } catch {
58
+ try {
59
+ const { execFile: execFileCb } = await import("child_process");
60
+ const { promisify } = await import("util");
61
+ const execFileAsync = promisify(execFileCb);
62
+ await execFileAsync("sqlite3", [dbPath, "PRAGMA wal_checkpoint(TRUNCATE);"], { timeout: 3e4 });
63
+ } catch {
64
+ process.stderr.write(`[db-backup] WAL checkpoint skipped \u2014 proceeding with copy (backup may include stale WAL)
65
+ `);
66
+ }
67
+ }
68
+ await copyFile(dbPath, backupPath);
69
+ try {
70
+ const { getClient } = await import("./lib/database.js");
71
+ const client = getClient();
72
+ const chk = await client.execute("PRAGMA quick_check;");
73
+ const result = String(chk.rows[0]?.quick_check ?? chk.rows[0]?.[0] ?? "").trim();
74
+ if (result && result !== "ok") {
75
+ process.stderr.write(`[db-backup] WARNING: backup source failed quick_check (${reason}): ${result}
76
+ `);
77
+ try {
78
+ await unlink(backupPath);
79
+ } catch {
80
+ }
81
+ return null;
82
+ }
83
+ } catch {
84
+ process.stderr.write(`[db-backup] Backup validation skipped (encrypted DB or CLI context) \u2014 proceeding with checkpoint-consistent copy.
85
+ `);
86
+ }
87
+ const walPath = dbPath + "-wal";
88
+ if (existsSync(walPath)) {
89
+ try {
90
+ await copyFile(walPath, backupPath + "-wal");
91
+ } catch {
92
+ }
93
+ }
94
+ const shmPath = dbPath + "-shm";
95
+ if (existsSync(shmPath)) {
96
+ try {
97
+ await copyFile(shmPath, backupPath + "-shm");
98
+ } catch {
99
+ }
100
+ }
101
+ return backupPath;
102
+ }
103
+ function rotateBackups(keepDays = DEFAULT_KEEP_DAYS) {
104
+ if (!existsSync(BACKUP_DIR)) return 0;
105
+ const cutoff = Date.now() - keepDays * 24 * 60 * 60 * 1e3;
106
+ let deleted = 0;
107
+ try {
108
+ const files = readdirSync(BACKUP_DIR);
109
+ for (const file of files) {
110
+ if (!file.endsWith(".db") && !file.endsWith(".db-wal") && !file.endsWith(".db-shm")) continue;
111
+ const filePath = path.join(BACKUP_DIR, file);
112
+ try {
113
+ const stat = statSync(filePath);
114
+ if (stat.mtimeMs < cutoff) {
115
+ unlinkSync(filePath);
116
+ deleted++;
117
+ }
118
+ } catch {
119
+ }
120
+ }
121
+ } catch {
122
+ }
123
+ return deleted;
124
+ }
125
+ function selectBackupsToDelete(backups, keep) {
126
+ if (keep < 0) keep = 0;
127
+ const sorted = [...backups].sort((a, b) => b.mtimeMs - a.mtimeMs);
128
+ return sorted.slice(keep).map((b) => b.path);
129
+ }
130
+ function hasEnoughDiskSpace(freeBytes, backupSizeBytes, minFreeBytes = MIN_FREE_BYTES) {
131
+ const required = Math.max(minFreeBytes, backupSizeBytes * 2);
132
+ return freeBytes >= required;
133
+ }
134
+ function getFreeBytes(dir) {
135
+ try {
136
+ const s = statfsSync(dir);
137
+ return s.bavail * s.bsize;
138
+ } catch {
139
+ return Number.POSITIVE_INFINITY;
140
+ }
141
+ }
142
+ function rotateBackupsByReason(reason, keep) {
143
+ if (!existsSync(BACKUP_DIR)) return 0;
144
+ let deleted = 0;
145
+ try {
146
+ const files = readdirSync(BACKUP_DIR);
147
+ const matches = [];
148
+ for (const file of files) {
149
+ if (!file.endsWith(".db")) continue;
150
+ if (!file.includes(`-${reason}-`)) continue;
151
+ const filePath = path.join(BACKUP_DIR, file);
152
+ try {
153
+ matches.push({ path: filePath, mtimeMs: statSync(filePath).mtimeMs });
154
+ } catch {
155
+ }
156
+ }
157
+ for (const target of selectBackupsToDelete(matches, keep)) {
158
+ for (const p of [target, target + "-wal", target + "-shm"]) {
159
+ try {
160
+ if (existsSync(p)) {
161
+ unlinkSync(p);
162
+ if (p === target) deleted++;
163
+ }
164
+ } catch {
165
+ }
166
+ }
167
+ }
168
+ } catch {
169
+ }
170
+ return deleted;
171
+ }
172
+ function enforceRetention(reason, keepDays = DEFAULT_KEEP_DAYS) {
173
+ let deleted = rotateBackups(keepDays);
174
+ const cap = REASON_COUNT_CAP[reason];
175
+ if (typeof cap === "number") {
176
+ deleted += rotateBackupsByReason(reason, cap);
177
+ }
178
+ return deleted;
179
+ }
180
+ function listBackups() {
181
+ if (!existsSync(BACKUP_DIR)) return [];
182
+ try {
183
+ const files = readdirSync(BACKUP_DIR).filter((f) => f.endsWith(".db") && !f.endsWith("-wal") && !f.endsWith("-shm"));
184
+ return files.map((name) => {
185
+ const p = path.join(BACKUP_DIR, name);
186
+ const stat = statSync(p);
187
+ return { path: p, name, size: stat.size, date: stat.mtime };
188
+ }).sort((a, b) => b.date.getTime() - a.date.getTime());
189
+ } catch {
190
+ return [];
191
+ }
192
+ }
193
+ function hasBackupToday(reason) {
194
+ const today = (/* @__PURE__ */ new Date()).toISOString().slice(0, 10);
195
+ const backups = listBackups();
196
+ return backups.some((b) => b.name.includes(reason) && b.name.includes(today.replace(/-/g, "-")));
197
+ }
198
+ function getLatestBackup() {
199
+ const backups = listBackups();
200
+ return backups.length > 0 ? backups[0].path : null;
201
+ }
202
+ async function restoreBackup(backupPath) {
203
+ const dbPath = findActiveDb();
204
+ if (!dbPath) {
205
+ const target = path.join(EXE_AI_DIR, DB_NAMES[0]);
206
+ const source2 = backupPath ?? getLatestBackup();
207
+ if (!source2 || !existsSync(source2)) return null;
208
+ await copyFile(source2, target);
209
+ return { restored: path.basename(source2), preRestoreBackup: null };
210
+ }
211
+ const source = backupPath ?? getLatestBackup();
212
+ if (!source || !existsSync(source)) return null;
213
+ if (path.resolve(source) === path.resolve(dbPath)) return null;
214
+ const preRestoreBackup = await createBackup("pre-restore");
215
+ await copyFile(source, dbPath);
216
+ const walPath = source + "-wal";
217
+ const shmPath = source + "-shm";
218
+ if (existsSync(walPath)) {
219
+ try {
220
+ await copyFile(walPath, dbPath + "-wal");
221
+ } catch {
222
+ }
223
+ } else {
224
+ try {
225
+ unlinkSync(dbPath + "-wal");
226
+ } catch {
227
+ }
228
+ }
229
+ if (existsSync(shmPath)) {
230
+ try {
231
+ await copyFile(shmPath, dbPath + "-shm");
232
+ } catch {
233
+ }
234
+ } else {
235
+ try {
236
+ unlinkSync(dbPath + "-shm");
237
+ } catch {
238
+ }
239
+ }
240
+ return { restored: path.basename(source), preRestoreBackup };
241
+ }
242
+ function getBackupDir() {
243
+ return BACKUP_DIR;
244
+ }
245
+
246
+ export {
247
+ findActiveDb,
248
+ createBackup,
249
+ createBackupAsync,
250
+ rotateBackups,
251
+ selectBackupsToDelete,
252
+ hasEnoughDiskSpace,
253
+ rotateBackupsByReason,
254
+ enforceRetention,
255
+ listBackups,
256
+ hasBackupToday,
257
+ getLatestBackup,
258
+ restoreBackup,
259
+ getBackupDir
260
+ };