@askexenow/exe-os 0.9.294 → 0.9.296

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (279) hide show
  1. package/deploy/compose/cloudflared/config.yml.example +14 -9
  2. package/deploy/compose/docker-compose.yml +86 -8
  3. package/deploy/compose/sso-edge/default.conf.template +87 -0
  4. package/deploy/compose/sso-edge/entrypoint.sh +23 -0
  5. package/deploy/compose/sso-edge/sso-redirect.conf +63 -0
  6. package/deploy/stack-manifests/v0.9.json +2 -2
  7. package/dist/active-agent-AFX2FODG.js +28 -0
  8. package/dist/active-agent-E2IJA7YX.js +27 -0
  9. package/dist/agentic-ontology-A2YUZK5O.js +25 -0
  10. package/dist/assets/com.askexe.exed.plist +4 -1
  11. package/dist/backfill-metadata-OC7EOD5U.js +600 -0
  12. package/dist/behaviors-H5ZOVHDH.js +46 -0
  13. package/dist/bin/agentic-ontology-backfill.js +5 -5
  14. package/dist/bin/agentic-reflection-backfill.js +6 -6
  15. package/dist/bin/agentic-semantic-label.js +5 -5
  16. package/dist/bin/backfill-conversations.js +6 -6
  17. package/dist/bin/backfill-responses.js +6 -6
  18. package/dist/bin/backfill-vectors.js +8 -8
  19. package/dist/bin/bulk-sync-postgres.js +7 -7
  20. package/dist/bin/cc-doctor.js +4 -4
  21. package/dist/bin/cleanup-stale-review-tasks.js +11 -11
  22. package/dist/bin/cli.js +16 -16
  23. package/dist/bin/deferred-daemon-restart.js +1 -1
  24. package/dist/bin/exe-agent-config.js +2 -2
  25. package/dist/bin/exe-agent.js +4 -4
  26. package/dist/bin/exe-assign.js +8 -8
  27. package/dist/bin/exe-boot.js +21 -18
  28. package/dist/bin/exe-call.js +4 -4
  29. package/dist/bin/exe-cloud.js +7 -7
  30. package/dist/bin/exe-dispatch.js +11 -11
  31. package/dist/bin/exe-doctor.js +3 -2
  32. package/dist/bin/exe-export-behaviors.js +7 -7
  33. package/dist/bin/exe-forget.js +6 -6
  34. package/dist/bin/exe-gateway.js +7 -7
  35. package/dist/bin/exe-healthcheck.js +6 -4
  36. package/dist/bin/exe-heartbeat.js +11 -11
  37. package/dist/bin/exe-kill.js +14 -14
  38. package/dist/bin/exe-launch-agent.js +18 -18
  39. package/dist/bin/exe-new-employee.js +6 -6
  40. package/dist/bin/exe-pending-messages.js +12 -12
  41. package/dist/bin/exe-pending-notifications.js +11 -11
  42. package/dist/bin/exe-pending-reviews.js +11 -11
  43. package/dist/bin/exe-rename.js +4 -4
  44. package/dist/bin/exe-review.js +13 -13
  45. package/dist/bin/exe-search.js +5 -5
  46. package/dist/bin/exe-session-cleanup.js +16 -16
  47. package/dist/bin/exe-settings.js +39 -9
  48. package/dist/bin/exe-start-codex.js +11 -11
  49. package/dist/bin/exe-start-opencode.js +8 -8
  50. package/dist/bin/exe-status.js +12 -12
  51. package/dist/bin/exe-team.js +3 -3
  52. package/dist/bin/git-sweep.js +12 -12
  53. package/dist/bin/graph-backfill.js +4 -4
  54. package/dist/bin/graph-export.js +5 -5
  55. package/dist/bin/import-history.js +7 -7
  56. package/dist/bin/install-launchd.js +13 -6
  57. package/dist/bin/install.js +26 -14
  58. package/dist/bin/intercom-check.js +4 -4
  59. package/dist/bin/mcp-sessions.js +2 -2
  60. package/dist/bin/orchestration-metrics.js +4 -4
  61. package/dist/bin/postgres-agentic-reflection-backfill.js +2 -2
  62. package/dist/bin/postgres-agentic-semantic-backfill.js +1 -1
  63. package/dist/bin/scan-tasks.js +11 -11
  64. package/dist/bin/setup.js +1 -1
  65. package/dist/bin/shard-migrate.js +4 -4
  66. package/dist/bin/stack-update.js +2 -2
  67. package/dist/bin/vps-health-gate.js +1 -1
  68. package/dist/capability-cards-4USI7CUW.js +89 -0
  69. package/dist/capacity-monitor-WLCBTEYR.js +51 -0
  70. package/dist/catchup-brief-ZR3NX6LZ.js +175 -0
  71. package/dist/chunk-22TVSRQQ.js +226 -0
  72. package/dist/chunk-2E43UXRH.js +395 -0
  73. package/dist/chunk-2PIGT6UJ.js +460 -0
  74. package/dist/chunk-3XTMW2MZ.js +535 -0
  75. package/dist/chunk-465PQFTH.js +262 -0
  76. package/dist/chunk-5CCXU2AW.js +129 -0
  77. package/dist/chunk-5D6MPWR7.js +1094 -0
  78. package/dist/chunk-5Q4MR6SL.js +123 -0
  79. package/dist/chunk-6327RBWR.js +345 -0
  80. package/dist/chunk-6MZZREZY.js +199 -0
  81. package/dist/chunk-7DI2Q4O5.js +1186 -0
  82. package/dist/chunk-7PW5VNIY.js +122 -0
  83. package/dist/chunk-7T7Y56HW.js +43 -0
  84. package/dist/chunk-7UHCWCLT.js +128 -0
  85. package/dist/chunk-A2ZUMF6L.js +1350 -0
  86. package/dist/chunk-AKV44JEH.js +185 -0
  87. package/dist/chunk-ANHWGX5N.js +735 -0
  88. package/dist/chunk-BQ3P4TKD.js +97 -0
  89. package/dist/chunk-BUZMT3KZ.js +604 -0
  90. package/dist/chunk-C2SBESBO.js +210 -0
  91. package/dist/chunk-CLSXZUZW.js +51 -0
  92. package/dist/chunk-CONHLVAR.js +1079 -0
  93. package/dist/chunk-D3WTZPFX.js +456 -0
  94. package/dist/chunk-DE6SOIYL.js +197 -0
  95. package/dist/chunk-EIVNMA3Q.js +284 -0
  96. package/dist/chunk-EJIF4FNT.js +12 -0
  97. package/dist/chunk-FDFOW564.js +171 -0
  98. package/dist/chunk-GZUBJ5EC.js +127 -0
  99. package/dist/chunk-HGZITN22.js +105 -0
  100. package/dist/chunk-HSRKDU6X.js +362 -0
  101. package/dist/chunk-IIEN2PHV.js +85 -0
  102. package/dist/chunk-JQ56VLMM.js +567 -0
  103. package/dist/chunk-JVHHXRFY.js +280 -0
  104. package/dist/chunk-JXCXGZ3S.js +55 -0
  105. package/dist/chunk-K5ZO532Q.js +4388 -0
  106. package/dist/chunk-K6CAAMXF.js +97 -0
  107. package/dist/chunk-KA26YTNU.js +81 -0
  108. package/dist/chunk-KMUW5C3R.js +381 -0
  109. package/dist/chunk-KOO3J5PV.js +20 -0
  110. package/dist/chunk-LSV7OFIH.js +290 -0
  111. package/dist/chunk-LSVFDVNY.js +1158 -0
  112. package/dist/chunk-LXDQTW32.js +230 -0
  113. package/dist/chunk-MEP7OUVZ.js +181 -0
  114. package/dist/chunk-MN2B2LKS.js +240 -0
  115. package/dist/chunk-N2EAYPYQ.js +1352 -0
  116. package/dist/chunk-N7I2A667.js +70 -0
  117. package/dist/chunk-NLZHVIOP.js +630 -0
  118. package/dist/chunk-NUH5TRZL.js +227 -0
  119. package/dist/chunk-OAHEIH3G.js +167 -0
  120. package/dist/chunk-OBHRQGCK.js +58 -0
  121. package/dist/chunk-ODFA7B2V.js +54 -0
  122. package/dist/chunk-OSNUP45F.js +731 -0
  123. package/dist/chunk-OTPRHBTO.js +33 -0
  124. package/dist/chunk-P6MUA4QU.js +157 -0
  125. package/dist/chunk-PGIOFKSK.js +2093 -0
  126. package/dist/chunk-PSE7VHWK.js +50 -0
  127. package/dist/chunk-QIFUVZFW.js +331 -0
  128. package/dist/chunk-RDPXKTVK.js +221 -0
  129. package/dist/chunk-RKYTYJGB.js +76 -0
  130. package/dist/chunk-RXLR6EFM.js +348 -0
  131. package/dist/chunk-SDB67PQJ.js +159 -0
  132. package/dist/chunk-SF2T7MP3.js +402 -0
  133. package/dist/chunk-SLU3FRFQ.js +2133 -0
  134. package/dist/chunk-SNDZJ5IV.js +214 -0
  135. package/dist/chunk-STEEAABW.js +448 -0
  136. package/dist/chunk-TUTWNHIQ.js +244 -0
  137. package/dist/chunk-UDP35QBR.js +30 -0
  138. package/dist/chunk-UKFHNJBI.js +85 -0
  139. package/dist/chunk-VC2DTK2X.js +382 -0
  140. package/dist/chunk-VRRAE5JX.js +836 -0
  141. package/dist/chunk-VVJTBQPR.js +38 -0
  142. package/dist/chunk-W3EQ362K.js +581 -0
  143. package/dist/chunk-WHIXIFHC.js +2242 -0
  144. package/dist/chunk-WRNGJJNR.js +377 -0
  145. package/dist/chunk-WUKHLCBE.js +3313 -0
  146. package/dist/chunk-WVPLHGDG.js +150 -0
  147. package/dist/chunk-XJZBSTL5.js +204 -0
  148. package/dist/chunk-Y3PMNUM5.js +304 -0
  149. package/dist/chunk-YHVS4QOV.js +14597 -0
  150. package/dist/chunk-YJ2OYAOC.js +668 -0
  151. package/dist/chunk-YYAD2GXX.js +128 -0
  152. package/dist/chunk-ZQML7EWE.js +333 -0
  153. package/dist/co-activation-XJLH46OX.js +74 -0
  154. package/dist/co-occurrence-GNN2X526.js +95 -0
  155. package/dist/code-context-index-OCPRLFG5.js +30 -0
  156. package/dist/core-memory-J4W2IYOF.js +110 -0
  157. package/dist/crdt-sync-QCBTSHIH.js +33 -0
  158. package/dist/crm-webhook-EM442VUW.js +10 -0
  159. package/dist/cto-delegation-gate-MLJMVHBK.js +280 -0
  160. package/dist/daemon-orchestration-2VNLZVTW.js +139 -0
  161. package/dist/db-backup-VUGFTPJ4.js +43 -0
  162. package/dist/doc-graph-extractor-PNRSFPSS.js +133 -0
  163. package/dist/dreaming-SK5VEQRF.js +34 -0
  164. package/dist/entity-boost-TQWWJUC2.js +375 -0
  165. package/dist/exe-drift-N34UPO7S.js +70 -0
  166. package/dist/exe-export-KACBKGVV.js +77 -0
  167. package/dist/exe-import-GXGDWACG.js +80 -0
  168. package/dist/exe-key-XPDOZBWW.js +673 -0
  169. package/dist/exe-snapshot-32GQKGQ5.js +338 -0
  170. package/dist/fast-db-init-F3TDD5VV.js +7 -0
  171. package/dist/gateway/index.js +8 -8
  172. package/dist/git-staleness-J45WNYRF.js +112 -0
  173. package/dist/git-task-sweep-BTGVQPFB.js +42 -0
  174. package/dist/global-procedures-6JCQWU4D.js +22 -0
  175. package/dist/graph-auto-extract-3ZQNXTPC.js +183 -0
  176. package/dist/hooks/bug-report-worker.js +13 -13
  177. package/dist/hooks/codex-stop-task-finalizer.js +13 -13
  178. package/dist/hooks/commit-complete.js +13 -13
  179. package/dist/hooks/error-recall.js +6 -6
  180. package/dist/hooks/exe-heartbeat-hook.js +3 -3
  181. package/dist/hooks/ingest-worker.js +3 -3
  182. package/dist/hooks/ingest.js +6 -6
  183. package/dist/hooks/instructions-loaded.js +4 -4
  184. package/dist/hooks/manifest.json +20 -20
  185. package/dist/hooks/notification.js +4 -4
  186. package/dist/hooks/post-compact.js +12 -12
  187. package/dist/hooks/post-tool-combined.js +6 -6
  188. package/dist/hooks/pre-compact.js +16 -16
  189. package/dist/hooks/pre-tool-use.js +16 -16
  190. package/dist/hooks/prompt-submit.js +24 -24
  191. package/dist/hooks/session-end.js +21 -21
  192. package/dist/hooks/session-start.js +12 -12
  193. package/dist/hooks/stop.js +19 -19
  194. package/dist/hooks/subagent-stop.js +12 -12
  195. package/dist/hooks/summary-worker.js +19 -19
  196. package/dist/index.js +19 -19
  197. package/dist/installer-5VPFY7SB.js +298 -0
  198. package/dist/installer-OENFPMA2.js +344 -0
  199. package/dist/installer-OIX4QOG5.js +40 -0
  200. package/dist/lib/cloud-sync.js +7 -7
  201. package/dist/lib/consolidation.js +6 -5
  202. package/dist/lib/database.js +2 -2
  203. package/dist/lib/db-daemon-client.js +2 -2
  204. package/dist/lib/db.js +2 -2
  205. package/dist/lib/embed-worker.js +1 -0
  206. package/dist/lib/embedder.js +7 -3
  207. package/dist/lib/employee-templates.js +4 -4
  208. package/dist/lib/employees.js +2 -2
  209. package/dist/lib/exe-daemon-client.js +2 -2
  210. package/dist/lib/exe-daemon.js +160 -79
  211. package/dist/lib/hybrid-search.js +5 -5
  212. package/dist/lib/identity.js +2 -2
  213. package/dist/lib/messaging.js +11 -11
  214. package/dist/lib/reminders.js +3 -3
  215. package/dist/lib/schedules.js +5 -5
  216. package/dist/lib/session-registry.js +4 -4
  217. package/dist/lib/skill-learning.js +6 -6
  218. package/dist/lib/store.js +4 -4
  219. package/dist/lib/task-router.js +3 -3
  220. package/dist/lib/tasks.js +12 -12
  221. package/dist/lib/tmux-routing.js +12 -10
  222. package/dist/lib/tmux-transport.js +1 -1
  223. package/dist/lib/token-spend.js +3 -3
  224. package/dist/lib/transport.js +2 -2
  225. package/dist/mcp/register-tools.js +62 -61
  226. package/dist/mcp/server.js +63 -62
  227. package/dist/mcp/tools/complete-reminder.js +4 -4
  228. package/dist/mcp/tools/create-reminder.js +4 -4
  229. package/dist/mcp/tools/create-task.js +14 -14
  230. package/dist/mcp/tools/deactivate-behavior.js +7 -7
  231. package/dist/mcp/tools/list-reminders.js +4 -4
  232. package/dist/mcp/tools/list-tasks.js +14 -14
  233. package/dist/mcp/tools/send-message.js +13 -13
  234. package/dist/mcp/tools/update-task.js +13 -13
  235. package/dist/mcp-http-config-PQTOLCTP.js +29 -0
  236. package/dist/memory-cards-4RVDZIY2.js +180 -0
  237. package/dist/memory-graph-extractor-L6YC7G4M.js +22 -0
  238. package/dist/memory-poisoning-defense-4YVJYH4G.js +224 -0
  239. package/dist/memory-queue-client-MVAUOZNJ.js +16 -0
  240. package/dist/memory-reflection-SHHDQNOH.js +244 -0
  241. package/dist/message-queue-client-DCKZT6X2.js +92 -0
  242. package/dist/notifications-JFR3G42W.js +47 -0
  243. package/dist/orchestration-events-MGCGPTDN.js +27 -0
  244. package/dist/orchestrator-DAFL2YZB.js +35 -0
  245. package/dist/pipeline-router-WWSZVPCH.js +15 -0
  246. package/dist/plan-limits-C7XCSDZC.js +28 -0
  247. package/dist/project-boot-N3NTBVLE.js +299 -0
  248. package/dist/projection-worker-MTPAPCWX.js +1084 -0
  249. package/dist/prospective-memory-BTIVUJSB.js +232 -0
  250. package/dist/reranker-UA6WVESJ.js +19 -0
  251. package/dist/retrieval-health-7XNZJEBF.js +12 -0
  252. package/dist/review-polling-4ALGMXC3.js +126 -0
  253. package/dist/runtime/index.js +13 -13
  254. package/dist/self-query-router-MROFQLQB.js +192 -0
  255. package/dist/session-events-CK44XOU4.js +38 -0
  256. package/dist/session-kill-telemetry-MT6ITDOG.js +31 -0
  257. package/dist/session-scope-3XDBWV65.js +88 -0
  258. package/dist/setup-wizard-X6DOD7MC.js +12 -0
  259. package/dist/skill-refinement-G2CCY3GM.js +159 -0
  260. package/dist/stack-update-JF7F56AS.js +84 -0
  261. package/dist/steward-gate-YF2CYXE7.js +15 -0
  262. package/dist/task-enforcement-YN6HK7NE.js +506 -0
  263. package/dist/task-scope-CVK6ISCZ.js +37 -0
  264. package/dist/tasks-crud-NTNET4JE.js +79 -0
  265. package/dist/tasks-notify-4LJVFPCV.js +40 -0
  266. package/dist/tasks-review-3V4WOIRG.js +49 -0
  267. package/dist/telemetry-upload-5PNUKGTM.js +741 -0
  268. package/dist/token-budget-E46G7ZAQ.js +86 -0
  269. package/dist/tool-capability-index-JDSMKJER.js +10 -0
  270. package/dist/tool-telemetry-J3NLS3LJ.js +17 -0
  271. package/dist/tui/App.js +18 -18
  272. package/dist/tui-data-6DOMUUCM.js +260 -0
  273. package/dist/wiki-acl-5UK37LKF.js +111 -0
  274. package/dist/worker-gate-FM7AEC7G.js +21 -0
  275. package/dist/workflow-engine-2EDUHUIY.js +28 -0
  276. package/dist/worktree-7YKKJIYR.js +28 -0
  277. package/dist/worktree-sweep-C3ELFGDN.js +21 -0
  278. package/package.json +1 -1
  279. package/release-notes.json +88 -88
@@ -0,0 +1,227 @@
1
+ import {
2
+ getClient
3
+ } from "./chunk-WUKHLCBE.js";
4
+
5
+ // src/lib/token-spend.ts
6
+ import { readdir, stat } from "fs/promises";
7
+ import { createReadStream } from "fs";
8
+ import { createInterface } from "readline";
9
+ import path from "path";
10
+ import os from "os";
11
+ var MODEL_PRICING = {
12
+ // Opus 4.5+ ($5/$25 — Anthropic price drop from original Opus 4)
13
+ "claude-opus-4-7": { input: 5 / 1e6, output: 25 / 1e6, cacheRead: 0.5 / 1e6, cacheWrite: 6.25 / 1e6 },
14
+ "claude-opus-4-6": { input: 5 / 1e6, output: 25 / 1e6, cacheRead: 0.5 / 1e6, cacheWrite: 6.25 / 1e6 },
15
+ "claude-opus-4-5": { input: 5 / 1e6, output: 25 / 1e6, cacheRead: 0.5 / 1e6, cacheWrite: 6.25 / 1e6 },
16
+ // Opus 4.0/4.1 (legacy $15/$75)
17
+ "claude-opus-4-1": { input: 15 / 1e6, output: 75 / 1e6, cacheRead: 1.5 / 1e6, cacheWrite: 18.75 / 1e6 },
18
+ "claude-opus-4": { input: 15 / 1e6, output: 75 / 1e6, cacheRead: 1.5 / 1e6, cacheWrite: 18.75 / 1e6 },
19
+ // Sonnet 4.x
20
+ "claude-sonnet-4": { input: 3 / 1e6, output: 15 / 1e6, cacheRead: 0.3 / 1e6, cacheWrite: 3.75 / 1e6 },
21
+ // Sonnet 3.7/3.5
22
+ "claude-3-7-sonnet": { input: 3 / 1e6, output: 15 / 1e6, cacheRead: 0.3 / 1e6, cacheWrite: 3.75 / 1e6 },
23
+ "claude-3-5-sonnet": { input: 3 / 1e6, output: 15 / 1e6, cacheRead: 0.3 / 1e6, cacheWrite: 3.75 / 1e6 },
24
+ // Haiku 4.5
25
+ "claude-haiku-4-5": { input: 1 / 1e6, output: 5 / 1e6, cacheRead: 0.1 / 1e6, cacheWrite: 1.25 / 1e6 },
26
+ // Haiku 3.5
27
+ "claude-3-5-haiku": { input: 0.8 / 1e6, output: 4 / 1e6, cacheRead: 0.08 / 1e6, cacheWrite: 1 / 1e6 },
28
+ // Opus 3
29
+ "claude-3-opus": { input: 15 / 1e6, output: 75 / 1e6, cacheRead: 1.5 / 1e6, cacheWrite: 18.75 / 1e6 },
30
+ // Sonnet 3
31
+ "claude-3-sonnet": { input: 3 / 1e6, output: 15 / 1e6, cacheRead: 0.3 / 1e6, cacheWrite: 3.75 / 1e6 },
32
+ // Haiku 3
33
+ "claude-3-haiku": { input: 0.25 / 1e6, output: 1.25 / 1e6, cacheRead: 0.03 / 1e6, cacheWrite: 0.3 / 1e6 }
34
+ };
35
+ var DEFAULT_PRICING = MODEL_PRICING["claude-sonnet-4"];
36
+ var CACHE_TTL_MS = 5 * 60 * 1e3;
37
+ var JSONL_INDEX_TTL_MS = 5 * 60 * 1e3;
38
+ var USAGE_CACHE_MAX_ENTRIES = 1e4;
39
+ var _spendCache = /* @__PURE__ */ new Map();
40
+ var _spendInFlight = /* @__PURE__ */ new Map();
41
+ var _usageCache = /* @__PURE__ */ new Map();
42
+ var _usageInFlight = /* @__PURE__ */ new Map();
43
+ var _jsonlPathIndex = null;
44
+ function yieldToEventLoop() {
45
+ return new Promise((resolve) => setImmediate(resolve));
46
+ }
47
+ function getPricing(model) {
48
+ if (MODEL_PRICING[model]) return MODEL_PRICING[model];
49
+ const stripped = model.replace(/-\d{8}$/, "");
50
+ if (MODEL_PRICING[stripped]) return MODEL_PRICING[stripped];
51
+ const sortedKeys = Object.keys(MODEL_PRICING).sort((a, b) => b.length - a.length);
52
+ for (const key of sortedKeys) {
53
+ if (model.includes(key)) return MODEL_PRICING[key];
54
+ }
55
+ return DEFAULT_PRICING;
56
+ }
57
+ async function getAgentSpend(period = "7d") {
58
+ const cached = _spendCache.get(period);
59
+ if (cached && Date.now() < cached.expires) {
60
+ return cached.result;
61
+ }
62
+ const inFlight = _spendInFlight.get(period);
63
+ if (inFlight) return inFlight;
64
+ const promise = computeAgentSpend(period).finally(() => _spendInFlight.delete(period));
65
+ _spendInFlight.set(period, promise);
66
+ return promise;
67
+ }
68
+ async function computeAgentSpend(period) {
69
+ const cutoff = periodToCutoff(period);
70
+ const client = getClient();
71
+ const dbResult = await client.execute({
72
+ sql: `SELECT session_uuid, agent_id FROM session_agent_map WHERE started_at >= ?`,
73
+ args: [cutoff]
74
+ });
75
+ if (dbResult.rows.length === 0) return [];
76
+ const sessionAgent = /* @__PURE__ */ new Map();
77
+ for (const row of dbResult.rows) {
78
+ sessionAgent.set(row.session_uuid, row.agent_id);
79
+ }
80
+ const jsonlPaths = await getClaudeSessionJsonlIndex();
81
+ if (jsonlPaths.size === 0) return [];
82
+ const agentTotals = /* @__PURE__ */ new Map();
83
+ let processed = 0;
84
+ for (const [sessionUuid, agentId] of sessionAgent) {
85
+ const jsonlPath = jsonlPaths.get(sessionUuid);
86
+ if (!jsonlPath) continue;
87
+ try {
88
+ const usage = await extractSessionUsageCached(jsonlPath);
89
+ if (usage.input === 0 && usage.output === 0) continue;
90
+ const totals = agentTotals.get(agentId) ?? {
91
+ input: 0,
92
+ output: 0,
93
+ cacheRead: 0,
94
+ cacheCreate: 0,
95
+ costUSD: 0,
96
+ sessions: /* @__PURE__ */ new Set()
97
+ };
98
+ totals.input += usage.input;
99
+ totals.output += usage.output;
100
+ totals.cacheRead += usage.cacheRead;
101
+ totals.cacheCreate += usage.cacheCreate;
102
+ totals.costUSD += usage.costUSD;
103
+ totals.sessions.add(sessionUuid);
104
+ agentTotals.set(agentId, totals);
105
+ } catch {
106
+ }
107
+ processed++;
108
+ if (processed % 25 === 0) await yieldToEventLoop();
109
+ }
110
+ const result = Array.from(agentTotals.entries()).map(([agentId, t]) => ({
111
+ agentId,
112
+ inputTokens: t.input,
113
+ outputTokens: t.output,
114
+ cacheReadTokens: t.cacheRead,
115
+ cacheCreationTokens: t.cacheCreate,
116
+ costUSD: t.costUSD,
117
+ sessions: t.sessions.size,
118
+ period
119
+ })).sort((a, b) => b.costUSD - a.costUSD);
120
+ _spendCache.set(period, { result, expires: Date.now() + CACHE_TTL_MS });
121
+ return result;
122
+ }
123
+ async function getClaudeSessionJsonlIndex() {
124
+ if (_jsonlPathIndex && Date.now() < _jsonlPathIndex.expires) {
125
+ return _jsonlPathIndex.paths;
126
+ }
127
+ const claudeDir = path.join(os.homedir(), ".claude", "projects");
128
+ const paths = /* @__PURE__ */ new Map();
129
+ let projectDirs = [];
130
+ try {
131
+ const entries = await readdir(claudeDir, { withFileTypes: true });
132
+ projectDirs = entries.filter((entry) => entry.isDirectory()).map((entry) => path.join(claudeDir, entry.name));
133
+ } catch {
134
+ _jsonlPathIndex = { expires: Date.now() + JSONL_INDEX_TTL_MS, paths };
135
+ return paths;
136
+ }
137
+ let scanned = 0;
138
+ for (const dir of projectDirs) {
139
+ try {
140
+ const files = await readdir(dir);
141
+ for (const file of files) {
142
+ if (!file.endsWith(".jsonl")) continue;
143
+ paths.set(file.slice(0, -".jsonl".length), path.join(dir, file));
144
+ }
145
+ } catch {
146
+ }
147
+ scanned++;
148
+ if (scanned % 25 === 0) await yieldToEventLoop();
149
+ }
150
+ _jsonlPathIndex = { expires: Date.now() + JSONL_INDEX_TTL_MS, paths };
151
+ return paths;
152
+ }
153
+ async function extractSessionUsageCached(jsonlPath) {
154
+ const st = await stat(jsonlPath);
155
+ const cached = _usageCache.get(jsonlPath);
156
+ if (cached && cached.mtimeMs === st.mtimeMs && cached.size === st.size) {
157
+ return cached.usage;
158
+ }
159
+ const existing = _usageInFlight.get(jsonlPath);
160
+ if (existing) return existing;
161
+ const promise = extractSessionUsage(jsonlPath).then((usage) => {
162
+ _usageCache.set(jsonlPath, { mtimeMs: st.mtimeMs, size: st.size, usage });
163
+ pruneUsageCache();
164
+ return usage;
165
+ }).finally(() => _usageInFlight.delete(jsonlPath));
166
+ _usageInFlight.set(jsonlPath, promise);
167
+ return promise;
168
+ }
169
+ function pruneUsageCache() {
170
+ if (_usageCache.size <= USAGE_CACHE_MAX_ENTRIES) return;
171
+ const overflow = _usageCache.size - USAGE_CACHE_MAX_ENTRIES;
172
+ let removed = 0;
173
+ for (const key of _usageCache.keys()) {
174
+ _usageCache.delete(key);
175
+ removed++;
176
+ if (removed >= overflow) break;
177
+ }
178
+ }
179
+ async function extractSessionUsage(jsonlPath) {
180
+ let input = 0;
181
+ let output = 0;
182
+ let cacheRead = 0;
183
+ let cacheCreate = 0;
184
+ let costUSD = 0;
185
+ const seenMessageIds = /* @__PURE__ */ new Set();
186
+ const rl = createInterface({
187
+ input: createReadStream(jsonlPath, { encoding: "utf8" }),
188
+ crlfDelay: Infinity
189
+ });
190
+ for await (const line of rl) {
191
+ if (!line.includes('"type":"assistant"')) continue;
192
+ try {
193
+ const record = JSON.parse(line);
194
+ if (record.type !== "assistant") continue;
195
+ const messageId = record.message?.id;
196
+ if (messageId) {
197
+ if (seenMessageIds.has(messageId)) continue;
198
+ seenMessageIds.add(messageId);
199
+ }
200
+ const usage = record.message?.usage;
201
+ if (!usage) continue;
202
+ const model = record.message?.model ?? "";
203
+ const pricing = getPricing(model);
204
+ const inp = usage.input_tokens ?? 0;
205
+ const out = usage.output_tokens ?? 0;
206
+ const cr = usage.cache_read_input_tokens ?? 0;
207
+ const cc = usage.cache_creation_input_tokens ?? 0;
208
+ input += inp;
209
+ output += out;
210
+ cacheRead += cr;
211
+ cacheCreate += cc;
212
+ if (pricing) {
213
+ costUSD += inp * pricing.input + out * pricing.output + cr * pricing.cacheRead + cc * pricing.cacheWrite;
214
+ }
215
+ } catch {
216
+ }
217
+ }
218
+ return { input, output, cacheRead, cacheCreate, costUSD };
219
+ }
220
+ function periodToCutoff(period) {
221
+ const ms = { "24h": 864e5, "7d": 6048e5, "30d": 2592e6 }[period];
222
+ return new Date(Date.now() - ms).toISOString();
223
+ }
224
+
225
+ export {
226
+ getAgentSpend
227
+ };
@@ -0,0 +1,167 @@
1
+ import {
2
+ EXE_AI_DIR
3
+ } from "./chunk-R36FAN53.js";
4
+
5
+ // src/lib/worker-gate.ts
6
+ import { readdirSync, writeFileSync, unlinkSync, mkdirSync, existsSync, readFileSync } from "fs";
7
+ import path from "path";
8
+ import { totalmem, cpus } from "os";
9
+ var WORKER_PID_DIR = path.join(EXE_AI_DIR, "worker-pids");
10
+ function computeMaxWorkers() {
11
+ const totalGB = totalmem() / 1024 ** 3;
12
+ const coreCount = cpus().length;
13
+ const byRam = Math.floor(totalGB / 4);
14
+ const byCpu = Math.floor(coreCount / 2);
15
+ return Math.max(2, Math.min(8, Math.min(byRam, byCpu)));
16
+ }
17
+ var MAX_CONCURRENT_WORKERS = process.env.EXE_MAX_WORKERS ? Math.max(1, parseInt(process.env.EXE_MAX_WORKERS, 10) || 3) : computeMaxWorkers();
18
+ function tryAcquireWorkerSlot() {
19
+ try {
20
+ mkdirSync(WORKER_PID_DIR, { recursive: true });
21
+ const reservationId = `res-${process.pid}-${Date.now()}`;
22
+ const reservationPath = path.join(WORKER_PID_DIR, `${reservationId}.pid`);
23
+ writeFileSync(reservationPath, String(process.pid));
24
+ const files = readdirSync(WORKER_PID_DIR);
25
+ let alive = 0;
26
+ for (const f of files) {
27
+ if (!f.endsWith(".pid")) continue;
28
+ if (f.startsWith("res-")) {
29
+ const resParts = f.replace(".pid", "").split("-");
30
+ const resPid = parseInt(resParts[1] ?? "", 10);
31
+ if (!isNaN(resPid) && resPid > 0) {
32
+ try {
33
+ process.kill(resPid, 0);
34
+ alive++;
35
+ } catch {
36
+ try {
37
+ unlinkSync(path.join(WORKER_PID_DIR, f));
38
+ } catch {
39
+ }
40
+ }
41
+ } else {
42
+ alive++;
43
+ }
44
+ continue;
45
+ }
46
+ const dashIdx = f.lastIndexOf("-");
47
+ const pid = parseInt(f.slice(dashIdx + 1).replace(".pid", ""), 10);
48
+ if (isNaN(pid)) continue;
49
+ try {
50
+ process.kill(pid, 0);
51
+ alive++;
52
+ } catch {
53
+ try {
54
+ unlinkSync(path.join(WORKER_PID_DIR, f));
55
+ } catch {
56
+ }
57
+ }
58
+ }
59
+ if (alive >= MAX_CONCURRENT_WORKERS) {
60
+ try {
61
+ unlinkSync(reservationPath);
62
+ } catch {
63
+ }
64
+ return false;
65
+ }
66
+ try {
67
+ unlinkSync(reservationPath);
68
+ } catch {
69
+ }
70
+ return true;
71
+ } catch {
72
+ return true;
73
+ }
74
+ }
75
+ function registerWorkerPid(pid) {
76
+ try {
77
+ mkdirSync(WORKER_PID_DIR, { recursive: true });
78
+ writeFileSync(path.join(WORKER_PID_DIR, `worker-${pid}.pid`), String(pid));
79
+ } catch {
80
+ }
81
+ }
82
+ function cleanupWorkerPid() {
83
+ try {
84
+ unlinkSync(path.join(WORKER_PID_DIR, `worker-${process.pid}.pid`));
85
+ } catch {
86
+ }
87
+ }
88
+ var BACKFILL_LOCK = path.join(WORKER_PID_DIR, "backfill.lock");
89
+ function tryAcquireBackfillLock() {
90
+ try {
91
+ mkdirSync(WORKER_PID_DIR, { recursive: true });
92
+ if (existsSync(BACKFILL_LOCK)) {
93
+ try {
94
+ const pid = parseInt(
95
+ readFileSync(BACKFILL_LOCK, "utf8").trim(),
96
+ 10
97
+ );
98
+ if (!isNaN(pid) && pid > 0) {
99
+ try {
100
+ process.kill(pid, 0);
101
+ return false;
102
+ } catch {
103
+ }
104
+ }
105
+ } catch {
106
+ }
107
+ }
108
+ writeFileSync(BACKFILL_LOCK, String(process.pid));
109
+ return true;
110
+ } catch {
111
+ return true;
112
+ }
113
+ }
114
+ function releaseBackfillLock() {
115
+ try {
116
+ unlinkSync(BACKFILL_LOCK);
117
+ } catch {
118
+ }
119
+ }
120
+ async function getTaskAwareCapacity() {
121
+ const { getClient } = await import("./lib/database.js");
122
+ const { sessionScopeFilter } = await import("./task-scope-CVK6ISCZ.js");
123
+ const client = getClient();
124
+ const scope = sessionScopeFilter();
125
+ const result = await client.execute({
126
+ sql: `SELECT assigned_to, COUNT(*) as cnt
127
+ FROM tasks
128
+ WHERE status IN ('open', 'in_progress')${scope.sql}
129
+ GROUP BY assigned_to`,
130
+ args: [...scope.args]
131
+ });
132
+ const busyAgents = [];
133
+ let totalInProgress = 0;
134
+ for (const row of result.rows) {
135
+ const agent = String(row.assigned_to ?? "");
136
+ const count = Number(row.cnt ?? 0);
137
+ if (agent && count > 0) {
138
+ busyAgents.push(agent);
139
+ totalInProgress += count;
140
+ }
141
+ }
142
+ const idleAgents = [];
143
+ try {
144
+ const { listTmuxSessions } = await import("./lib/tmux-status.js");
145
+ const { isExeSession } = await import("./lib/tmux-routing.js");
146
+ const { baseAgentName } = await import("./lib/employees.js");
147
+ const liveSessions = listTmuxSessions().filter((s) => !isExeSession(s) && s.includes("-"));
148
+ const liveAgentNames = new Set(liveSessions.map((s) => baseAgentName(s.split("-")[0] ?? "")));
149
+ for (const name of liveAgentNames) {
150
+ if (name && !busyAgents.includes(name)) {
151
+ idleAgents.push(name);
152
+ }
153
+ }
154
+ } catch {
155
+ }
156
+ return { busyAgents, idleAgents, totalInProgress };
157
+ }
158
+
159
+ export {
160
+ MAX_CONCURRENT_WORKERS,
161
+ tryAcquireWorkerSlot,
162
+ registerWorkerPid,
163
+ cleanupWorkerPid,
164
+ tryAcquireBackfillLock,
165
+ releaseBackfillLock,
166
+ getTaskAwareCapacity
167
+ };
@@ -0,0 +1,58 @@
1
+ import {
2
+ MODELS_DIR
3
+ } from "./chunk-R36FAN53.js";
4
+
5
+ // src/lib/reranker.ts
6
+ import path from "path";
7
+ import { existsSync } from "fs";
8
+ var RERANKER_MODEL_FILE = process.env.EXE_RERANK_MODEL_FILE ?? "exe-reranker-v1-q4_k_m.gguf";
9
+ function isRerankerAvailable() {
10
+ return existsSync(path.join(MODELS_DIR, RERANKER_MODEL_FILE));
11
+ }
12
+ function getRerankerModelPath() {
13
+ return path.join(MODELS_DIR, RERANKER_MODEL_FILE);
14
+ }
15
+ async function disposeReranker() {
16
+ }
17
+ async function rerankWithScores(query, texts, topK) {
18
+ if (texts.length === 0) return [];
19
+ const { rerankViaWorker } = await import("./lib/exe-daemon.js");
20
+ const scored = await rerankViaWorker(query, texts, topK);
21
+ return scored.map((s) => ({
22
+ text: texts[s.index] ?? "",
23
+ score: s.score,
24
+ index: s.index
25
+ }));
26
+ }
27
+ async function rerank(query, candidates, topK = 5) {
28
+ if (candidates.length === 0) return [];
29
+ if (candidates.length <= topK) return candidates;
30
+ const scored = await rerankWithScores(
31
+ query,
32
+ candidates.map((c) => c.raw_text),
33
+ topK
34
+ );
35
+ return scored.map((s) => candidates[s.index]);
36
+ }
37
+ async function rerankWithContext(query, candidates, topK) {
38
+ if (candidates.length === 0) return [];
39
+ const formattedTexts = candidates.map(
40
+ (c) => c.context ? `[${c.context}] ${c.text.slice(0, 460)}` : c.text.slice(0, 512)
41
+ );
42
+ const { rerankViaWorker } = await import("./lib/exe-daemon.js");
43
+ const scored = await rerankViaWorker(query, formattedTexts, topK);
44
+ return scored.map((s) => ({
45
+ text: candidates[s.index]?.text ?? "",
46
+ score: s.score,
47
+ index: s.index
48
+ }));
49
+ }
50
+
51
+ export {
52
+ isRerankerAvailable,
53
+ getRerankerModelPath,
54
+ disposeReranker,
55
+ rerankWithScores,
56
+ rerank,
57
+ rerankWithContext
58
+ };
@@ -0,0 +1,54 @@
1
+ import {
2
+ processCRMEvent
3
+ } from "./chunk-VC2DTK2X.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
+ };