@askexenow/exe-os 0.9.280 → 0.9.281

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 (436) hide show
  1. package/deploy/compose/.env.customer.example +1 -1
  2. package/deploy/compose/.env.default +2 -1
  3. package/deploy/compose/.env.example +1 -1
  4. package/deploy/compose/docker-compose.yml +74 -13
  5. package/deploy/compose/erp-nginx/nginx.conf +52 -0
  6. package/deploy/compose/generate-env.ts +1 -1
  7. package/deploy/compose/init-db.sql +13 -2
  8. package/deploy/stack-manifests/v0.9.json +11 -31
  9. package/dist/active-agent-6ZBHGHXF.js +26 -0
  10. package/dist/active-agent-E23SCYER.js +27 -0
  11. package/dist/active-agent-KMZT44S4.js +26 -0
  12. package/dist/active-agent-LFFTVROM.js +27 -0
  13. package/dist/agentic-ontology-PCZB5HV5.js +25 -0
  14. package/dist/agentic-ontology-PGGJN2ES.js +25 -0
  15. package/dist/assets/tmux.conf +2 -0
  16. package/dist/backfill-metadata-KQ4FEVUR.js +599 -0
  17. package/dist/backfill-metadata-Y3YWCHKJ.js +599 -0
  18. package/dist/behaviors-H4DZECKL.js +39 -0
  19. package/dist/behaviors-WIUTIJF6.js +39 -0
  20. package/dist/bin/agentic-ontology-backfill.js +5 -5
  21. package/dist/bin/agentic-reflection-backfill.js +6 -6
  22. package/dist/bin/agentic-semantic-label.js +5 -5
  23. package/dist/bin/backfill-conversations.js +4 -4
  24. package/dist/bin/backfill-responses.js +4 -4
  25. package/dist/bin/backfill-vectors.js +5 -5
  26. package/dist/bin/bulk-sync-postgres.js +7 -7
  27. package/dist/bin/cc-doctor.js +4 -4
  28. package/dist/bin/cleanup-stale-review-tasks.js +10 -10
  29. package/dist/bin/cli.js +15 -15
  30. package/dist/bin/exe-agent-config.js +2 -2
  31. package/dist/bin/exe-agent.js +4 -4
  32. package/dist/bin/exe-assign.js +5 -5
  33. package/dist/bin/exe-boot.js +17 -17
  34. package/dist/bin/exe-call.js +4 -4
  35. package/dist/bin/exe-cloud.js +5 -5
  36. package/dist/bin/exe-dispatch.js +10 -10
  37. package/dist/bin/exe-doctor.js +1 -1
  38. package/dist/bin/exe-export-behaviors.js +7 -7
  39. package/dist/bin/exe-forget.js +6 -6
  40. package/dist/bin/exe-gateway.js +7 -7
  41. package/dist/bin/exe-healthcheck.js +4 -4
  42. package/dist/bin/exe-heartbeat.js +10 -10
  43. package/dist/bin/exe-kill.js +13 -13
  44. package/dist/bin/exe-launch-agent.js +17 -17
  45. package/dist/bin/exe-new-employee.js +7 -7
  46. package/dist/bin/exe-pending-messages.js +11 -11
  47. package/dist/bin/exe-pending-notifications.js +10 -10
  48. package/dist/bin/exe-pending-reviews.js +10 -10
  49. package/dist/bin/exe-rename.js +4 -4
  50. package/dist/bin/exe-review.js +12 -12
  51. package/dist/bin/exe-search.js +5 -5
  52. package/dist/bin/exe-session-cleanup.js +15 -15
  53. package/dist/bin/exe-settings.js +5 -5
  54. package/dist/bin/exe-start-codex.js +11 -11
  55. package/dist/bin/exe-start-opencode.js +8 -8
  56. package/dist/bin/exe-status.js +11 -11
  57. package/dist/bin/exe-support.js +2 -2
  58. package/dist/bin/exe-team.js +3 -3
  59. package/dist/bin/exe-watchdog.js +17 -2
  60. package/dist/bin/git-sweep.js +11 -11
  61. package/dist/bin/graph-backfill.js +4 -4
  62. package/dist/bin/graph-export.js +5 -5
  63. package/dist/bin/import-history.js +7 -7
  64. package/dist/bin/install.js +8 -7
  65. package/dist/bin/intercom-check.js +4 -4
  66. package/dist/bin/mcp-sessions.js +2 -2
  67. package/dist/bin/orchestration-metrics.js +4 -4
  68. package/dist/bin/postgres-agentic-reflection-backfill.js +2 -2
  69. package/dist/bin/postgres-agentic-semantic-backfill.js +1 -1
  70. package/dist/bin/scan-tasks.js +10 -10
  71. package/dist/bin/setup.js +1 -1
  72. package/dist/bin/shard-migrate.js +4 -4
  73. package/dist/bin/stack-update.js +3 -3
  74. package/dist/bin/vps-health-gate.js +1 -1
  75. package/dist/capability-cards-F22XWGLB.js +88 -0
  76. package/dist/capability-cards-KCKITXXZ.js +88 -0
  77. package/dist/capacity-monitor-CHXGKVEO.js +50 -0
  78. package/dist/capacity-monitor-L3M5KB6T.js +50 -0
  79. package/dist/catchup-brief-BVZANLRZ.js +154 -0
  80. package/dist/catchup-brief-LDUGMC7S.js +154 -0
  81. package/dist/catchup-brief-NZDLL7SA.js +154 -0
  82. package/dist/catchup-brief-OOIXVGBA.js +154 -0
  83. package/dist/catchup-brief-UPXHDYTB.js +154 -0
  84. package/dist/chunk-2BI3FQKL.js +1876 -0
  85. package/dist/chunk-2CJUQGHH.js +362 -0
  86. package/dist/chunk-2EVYBMBJ.js +128 -0
  87. package/dist/chunk-2NQOZOXG.js +2113 -0
  88. package/dist/chunk-2QK5E3LB.js +128 -0
  89. package/dist/chunk-2VTCG4ZU.js +1352 -0
  90. package/dist/chunk-2YGI36DV.js +1119 -0
  91. package/dist/chunk-32BQN2QN.js +185 -0
  92. package/dist/chunk-33JMO4UV.js +157 -0
  93. package/dist/chunk-3DZAYLXY.js +377 -0
  94. package/dist/chunk-3MGBE7GR.js +76 -0
  95. package/dist/chunk-3MTV4FJL.js +271 -0
  96. package/dist/chunk-3NN7VQ27.js +1352 -0
  97. package/dist/chunk-4533HNOG.js +70 -0
  98. package/dist/chunk-456UW3MT.js +731 -0
  99. package/dist/chunk-4AU56XV2.js +58 -0
  100. package/dist/chunk-4MRP6EBR.js +280 -0
  101. package/dist/chunk-4MRWW52U.js +14219 -0
  102. package/dist/chunk-4WWECNAY.js +50 -0
  103. package/dist/chunk-4ZJDDR6L.js +171 -0
  104. package/dist/chunk-522EOPM6.js +382 -0
  105. package/dist/chunk-52UDAVZE.js +3208 -0
  106. package/dist/chunk-5FAMUB4X.js +204 -0
  107. package/dist/chunk-5PGZJQUI.js +58 -0
  108. package/dist/chunk-67TZJXNZ.js +262 -0
  109. package/dist/chunk-6FEZ7GN2.js +123 -0
  110. package/dist/chunk-6MOGND7S.js +14219 -0
  111. package/dist/chunk-6TRSVY7L.js +181 -0
  112. package/dist/chunk-6WCS7ZNK.js +85 -0
  113. package/dist/chunk-6WRYDREW.js +539 -0
  114. package/dist/chunk-72UA2FB3.js +181 -0
  115. package/dist/chunk-77DMFEOL.js +30 -0
  116. package/dist/chunk-7COXVQ5W.js +214 -0
  117. package/dist/chunk-7HLWBYH7.js +60 -0
  118. package/dist/chunk-ALSTZCWT.js +204 -0
  119. package/dist/chunk-ARUTDXZX.js +280 -0
  120. package/dist/chunk-AU47B4QY.js +129 -0
  121. package/dist/chunk-AXOREWVL.js +836 -0
  122. package/dist/chunk-B5MNC54V.js +127 -0
  123. package/dist/chunk-B74VSMKX.js +1350 -0
  124. package/dist/chunk-BHWLH44J.js +362 -0
  125. package/dist/chunk-BUPZ3HD2.js +85 -0
  126. package/dist/chunk-BWVLMA53.js +2113 -0
  127. package/dist/chunk-C5WLBKMJ.js +50 -0
  128. package/dist/chunk-C62T6R2A.js +97 -0
  129. package/dist/chunk-CV5KBAIK.js +33 -0
  130. package/dist/chunk-DAYSDWXA.js +1068 -0
  131. package/dist/chunk-DI4URIUB.js +227 -0
  132. package/dist/chunk-DNTCYFJ6.js +76 -0
  133. package/dist/chunk-DPDRRS7T.js +103 -0
  134. package/dist/chunk-DT3EV6CW.js +103 -0
  135. package/dist/chunk-DX45HDWY.js +1076 -0
  136. package/dist/chunk-E72BD6MG.js +284 -0
  137. package/dist/chunk-EAPUSVKS.js +375 -0
  138. package/dist/chunk-EFUANRRT.js +85 -0
  139. package/dist/chunk-EG2SCT5R.js +1352 -0
  140. package/dist/chunk-EJD2JU77.js +58 -0
  141. package/dist/chunk-EMXYUAVP.js +81 -0
  142. package/dist/chunk-ENM2TAAM.js +14219 -0
  143. package/dist/chunk-EPDRTPVP.js +1876 -0
  144. package/dist/chunk-EW6XDHID.js +221 -0
  145. package/dist/chunk-EYEGSAWZ.js +1094 -0
  146. package/dist/chunk-F6L33PAQ.js +231 -0
  147. package/dist/chunk-FC2SCTVE.js +38 -0
  148. package/dist/chunk-FSRKIZGZ.js +630 -0
  149. package/dist/chunk-FTG7I5CB.js +81 -0
  150. package/dist/chunk-GVFRLWX7.js +30 -0
  151. package/dist/chunk-H3XMZOWW.js +1119 -0
  152. package/dist/chunk-HAKXE6LN.js +123 -0
  153. package/dist/chunk-HLP3ZDTW.js +448 -0
  154. package/dist/chunk-HOYWKQAA.js +510 -0
  155. package/dist/chunk-HWDD64IW.js +712 -0
  156. package/dist/chunk-HWMCULHY.js +127 -0
  157. package/dist/chunk-HZZHRZPK.js +210 -0
  158. package/dist/chunk-ICRWTYNW.js +103 -0
  159. package/dist/chunk-J5MWPC33.js +167 -0
  160. package/dist/chunk-JBXANNNB.js +70 -0
  161. package/dist/chunk-JPBMIYWF.js +1352 -0
  162. package/dist/chunk-JY6EXBFI.js +373 -0
  163. package/dist/chunk-JZLIBXI7.js +14219 -0
  164. package/dist/chunk-JZPTKXJ6.js +668 -0
  165. package/dist/chunk-KEYMA4ZP.js +510 -0
  166. package/dist/chunk-KHGNN6GL.js +2078 -0
  167. package/dist/chunk-KVLB2PD6.js +97 -0
  168. package/dist/chunk-L3YBRBKL.js +1076 -0
  169. package/dist/chunk-L7VZ32NA.js +89 -0
  170. package/dist/chunk-LEHLADW4.js +221 -0
  171. package/dist/chunk-LPD4HILQ.js +262 -0
  172. package/dist/chunk-LSJ3ADDI.js +51 -0
  173. package/dist/chunk-LUMS2MAS.js +58 -0
  174. package/dist/chunk-LV4SEC6C.js +394 -0
  175. package/dist/chunk-LXXBEI4A.js +284 -0
  176. package/dist/chunk-M4NPCAIH.js +456 -0
  177. package/dist/chunk-MFJ62LQ5.js +157 -0
  178. package/dist/chunk-MGEZNKOD.js +836 -0
  179. package/dist/chunk-MJ7X5IBW.js +227 -0
  180. package/dist/chunk-MLGRWCY4.js +54 -0
  181. package/dist/chunk-MQZX57IY.js +348 -0
  182. package/dist/chunk-MY7MFF6J.js +103 -0
  183. package/dist/chunk-MYA2X5OY.js +185 -0
  184. package/dist/chunk-MYEABW5Z.js +630 -0
  185. package/dist/chunk-MZ2TDCAL.js +402 -0
  186. package/dist/chunk-NF3FRB7Z.js +271 -0
  187. package/dist/chunk-NXYIFEPV.js +539 -0
  188. package/dist/chunk-O2GVE5B5.js +58 -0
  189. package/dist/chunk-O5GUCDR2.js +456 -0
  190. package/dist/chunk-OBZNRECA.js +128 -0
  191. package/dist/chunk-OMZ2RLJG.js +214 -0
  192. package/dist/chunk-ORDHJRWN.js +299 -0
  193. package/dist/chunk-OSPIJMCD.js +210 -0
  194. package/dist/chunk-OUGWEH4J.js +240 -0
  195. package/dist/chunk-OV5MJQGC.js +1876 -0
  196. package/dist/chunk-PC635OAG.js +4318 -0
  197. package/dist/chunk-PH46R4J6.js +348 -0
  198. package/dist/chunk-PJP2EP7P.js +394 -0
  199. package/dist/chunk-PODFWH3V.js +333 -0
  200. package/dist/chunk-PPWH3SHR.js +1068 -0
  201. package/dist/chunk-PWPJK7KB.js +4318 -0
  202. package/dist/chunk-PXONZVG4.js +377 -0
  203. package/dist/chunk-QAOGJRZD.js +369 -0
  204. package/dist/chunk-QC3LAEI7.js +197 -0
  205. package/dist/chunk-QDWQDUWI.js +668 -0
  206. package/dist/chunk-QKJFD6BH.js +1350 -0
  207. package/dist/chunk-QNYVJGFM.js +345 -0
  208. package/dist/chunk-QQF3XGQ5.js +14219 -0
  209. package/dist/chunk-QRPFQNI3.js +150 -0
  210. package/dist/chunk-QSSU5XWD.js +731 -0
  211. package/dist/chunk-QXZAGVAV.js +2078 -0
  212. package/dist/chunk-RA54MW64.js +244 -0
  213. package/dist/chunk-RF7PUWXI.js +197 -0
  214. package/dist/chunk-S3INDYSO.js +244 -0
  215. package/dist/chunk-SLQVTHH5.js +369 -0
  216. package/dist/chunk-SY2B74KL.js +345 -0
  217. package/dist/chunk-T6QPXXXW.js +712 -0
  218. package/dist/chunk-TO5M5YCT.js +41 -0
  219. package/dist/chunk-TQ4VXUAF.js +129 -0
  220. package/dist/chunk-UAB7RQC4.js +41 -0
  221. package/dist/chunk-UMEIBDYW.js +97 -0
  222. package/dist/chunk-UXW5TB7Y.js +240 -0
  223. package/dist/chunk-VB2N5WOX.js +150 -0
  224. package/dist/chunk-VLZEMRG3.js +167 -0
  225. package/dist/chunk-VM3V6VK7.js +230 -0
  226. package/dist/chunk-VMCGKBHB.js +1352 -0
  227. package/dist/chunk-VNIYZAR5.js +128 -0
  228. package/dist/chunk-VYV4KOD2.js +85 -0
  229. package/dist/chunk-W4SRJBAT.js +171 -0
  230. package/dist/chunk-W5W3LZ3Q.js +54 -0
  231. package/dist/chunk-WDNZEOM3.js +38 -0
  232. package/dist/chunk-WUKEXVOR.js +3208 -0
  233. package/dist/chunk-X3Z35Q6L.js +373 -0
  234. package/dist/chunk-X4VOU6BQ.js +382 -0
  235. package/dist/chunk-X6EEVSVG.js +290 -0
  236. package/dist/chunk-XFHGWGNB.js +1094 -0
  237. package/dist/chunk-XKOLRWYA.js +33 -0
  238. package/dist/chunk-XMMIL3UD.js +402 -0
  239. package/dist/chunk-XWILC6VA.js +290 -0
  240. package/dist/chunk-Y4OQCX4C.js +97 -0
  241. package/dist/chunk-Y67VYYOA.js +231 -0
  242. package/dist/chunk-YGQCQTQH.js +230 -0
  243. package/dist/chunk-YGWFBN5A.js +299 -0
  244. package/dist/chunk-YMKUXZIG.js +379 -0
  245. package/dist/chunk-YOMLMT7E.js +230 -0
  246. package/dist/chunk-YPESIZOB.js +14219 -0
  247. package/dist/chunk-Z2CGCIU2.js +89 -0
  248. package/dist/chunk-ZLAWNHQR.js +448 -0
  249. package/dist/chunk-ZME5UQSN.js +333 -0
  250. package/dist/co-activation-NUEQYXE5.js +73 -0
  251. package/dist/co-activation-ZG5HLBCZ.js +73 -0
  252. package/dist/co-occurrence-7S5KWQB2.js +94 -0
  253. package/dist/co-occurrence-X5SWDXT2.js +94 -0
  254. package/dist/core-memory-GOPBRGGZ.js +110 -0
  255. package/dist/core-memory-XLCU6L5M.js +110 -0
  256. package/dist/crdt-sync-EPKHPGRZ.js +33 -0
  257. package/dist/crdt-sync-UIQJ5U7T.js +33 -0
  258. package/dist/crm-webhook-MKN23JNU.js +10 -0
  259. package/dist/crm-webhook-SM63BPXO.js +10 -0
  260. package/dist/cto-delegation-gate-PQY5TOVZ.js +279 -0
  261. package/dist/cto-delegation-gate-V5VVUR3G.js +279 -0
  262. package/dist/daemon-orchestration-C7AAS67Q.js +138 -0
  263. package/dist/daemon-orchestration-OBCAJB2H.js +138 -0
  264. package/dist/db-backup-F7VP4QRH.js +33 -0
  265. package/dist/db-backup-KVYC57W7.js +33 -0
  266. package/dist/doc-graph-extractor-H2ETEINP.js +132 -0
  267. package/dist/doc-graph-extractor-PCUZEYCH.js +132 -0
  268. package/dist/dreaming-4OZXSLE3.js +33 -0
  269. package/dist/dreaming-Z2RYEYNT.js +33 -0
  270. package/dist/exe-drift-GEWNIK7A.js +69 -0
  271. package/dist/exe-drift-XCGH7AFO.js +69 -0
  272. package/dist/exe-export-7DKAU5IP.js +75 -0
  273. package/dist/exe-export-BCHH6OE6.js +75 -0
  274. package/dist/exe-import-PDRIZVYF.js +78 -0
  275. package/dist/exe-import-ZCKUDFKL.js +78 -0
  276. package/dist/exe-key-FUWLLI3U.js +580 -0
  277. package/dist/exe-key-RKKNVUMP.js +580 -0
  278. package/dist/exe-snapshot-G4I5FQMK.js +337 -0
  279. package/dist/exe-snapshot-GWU7QTZK.js +337 -0
  280. package/dist/fast-db-init-6QG6YQNT.js +7 -0
  281. package/dist/fast-db-init-X2QDQUA4.js +7 -0
  282. package/dist/founder-context-TOMNUBGJ.js +96 -0
  283. package/dist/founder-context-UU3V6MAS.js +96 -0
  284. package/dist/gateway/index.js +8 -8
  285. package/dist/git-staleness-FEPFMZKF.js +111 -0
  286. package/dist/git-staleness-HYVYLCW3.js +111 -0
  287. package/dist/git-task-sweep-IRV52JIM.js +41 -0
  288. package/dist/git-task-sweep-T6BSM3GS.js +41 -0
  289. package/dist/global-procedures-3AURRMKO.js +21 -0
  290. package/dist/global-procedures-JPCYBZYC.js +21 -0
  291. package/dist/graph-auto-extract-OC3AOSMW.js +182 -0
  292. package/dist/graph-auto-extract-PVDYEJBY.js +182 -0
  293. package/dist/hooks/bug-report-worker.js +12 -12
  294. package/dist/hooks/codex-stop-task-finalizer.js +12 -12
  295. package/dist/hooks/commit-complete.js +12 -12
  296. package/dist/hooks/error-recall.js +6 -6
  297. package/dist/hooks/exe-heartbeat-hook.js +3 -3
  298. package/dist/hooks/ingest.js +6 -6
  299. package/dist/hooks/instructions-loaded.js +4 -4
  300. package/dist/hooks/manifest.json +19 -19
  301. package/dist/hooks/notification.js +4 -4
  302. package/dist/hooks/post-compact.js +11 -11
  303. package/dist/hooks/post-tool-combined.js +5 -5
  304. package/dist/hooks/pre-compact.js +12 -12
  305. package/dist/hooks/pre-tool-use.js +15 -15
  306. package/dist/hooks/prompt-submit.js +22 -22
  307. package/dist/hooks/session-end.js +16 -16
  308. package/dist/hooks/session-start.js +10 -10
  309. package/dist/hooks/stop.js +15 -15
  310. package/dist/hooks/subagent-stop.js +11 -11
  311. package/dist/hooks/summary-worker.js +15 -15
  312. package/dist/index.js +18 -18
  313. package/dist/installer-72XXLBRP.js +39 -0
  314. package/dist/installer-HDXG2BZN.js +343 -0
  315. package/dist/installer-JALMKPCS.js +297 -0
  316. package/dist/installer-Q46SNNLU.js +39 -0
  317. package/dist/installer-W7PIPRCX.js +343 -0
  318. package/dist/installer-Z7WQEOS7.js +297 -0
  319. package/dist/lib/cloud-sync.js +5 -5
  320. package/dist/lib/consolidation.js +5 -5
  321. package/dist/lib/database.js +2 -2
  322. package/dist/lib/db.js +2 -2
  323. package/dist/lib/employee-templates.js +4 -4
  324. package/dist/lib/employees.js +2 -2
  325. package/dist/lib/exe-daemon.js +45 -41
  326. package/dist/lib/hybrid-search.js +5 -5
  327. package/dist/lib/identity.js +2 -2
  328. package/dist/lib/license.js +1 -1
  329. package/dist/lib/messaging.js +10 -10
  330. package/dist/lib/reminders.js +3 -3
  331. package/dist/lib/schedules.js +5 -5
  332. package/dist/lib/session-registry.js +4 -4
  333. package/dist/lib/skill-learning.js +6 -6
  334. package/dist/lib/store.js +4 -4
  335. package/dist/lib/task-router.js +3 -3
  336. package/dist/lib/tasks.js +11 -11
  337. package/dist/lib/tmux-routing.js +9 -9
  338. package/dist/lib/token-spend.js +3 -3
  339. package/dist/license-gate-7JZCHOAG.js +14 -0
  340. package/dist/license-gate-OP4SKL4P.js +14 -0
  341. package/dist/mcp/register-tools.js +58 -58
  342. package/dist/mcp/server.js +59 -59
  343. package/dist/mcp/tools/complete-reminder.js +4 -4
  344. package/dist/mcp/tools/create-reminder.js +4 -4
  345. package/dist/mcp/tools/create-task.js +13 -13
  346. package/dist/mcp/tools/deactivate-behavior.js +7 -7
  347. package/dist/mcp/tools/list-reminders.js +4 -4
  348. package/dist/mcp/tools/list-tasks.js +13 -13
  349. package/dist/mcp/tools/send-message.js +12 -12
  350. package/dist/mcp/tools/update-task.js +12 -12
  351. package/dist/mcp-health-VULNT722.js +17 -0
  352. package/dist/mcp-health-WDOB6XUB.js +19 -0
  353. package/dist/mcp-http-config-2OZ7N74D.js +28 -0
  354. package/dist/mcp-http-config-4VXA5K73.js +28 -0
  355. package/dist/memory-cards-2K6QRZU6.js +179 -0
  356. package/dist/memory-cards-KSJF5OH2.js +179 -0
  357. package/dist/memory-graph-extractor-IJD5HWYT.js +21 -0
  358. package/dist/memory-graph-extractor-O4GAXOK5.js +21 -0
  359. package/dist/memory-poisoning-defense-2JRPWT5V.js +223 -0
  360. package/dist/memory-poisoning-defense-DH4A25NU.js +223 -0
  361. package/dist/memory-reflection-ISY2BBDB.js +243 -0
  362. package/dist/memory-reflection-Z5AQRR6H.js +243 -0
  363. package/dist/notifications-2VSWK2UJ.js +46 -0
  364. package/dist/notifications-4S253VQM.js +46 -0
  365. package/dist/oauth-server-D7D4574D.js +437 -0
  366. package/dist/oauth-server-MACN54SJ.js +437 -0
  367. package/dist/orchestration-events-BGP5RYQI.js +26 -0
  368. package/dist/orchestration-events-MDXUEVRZ.js +26 -0
  369. package/dist/orchestrator-DHK7RSSH.js +34 -0
  370. package/dist/orchestrator-R75WHQVA.js +34 -0
  371. package/dist/pipeline-router-4WUKQQEC.js +14 -0
  372. package/dist/pipeline-router-GWB2XK2Q.js +14 -0
  373. package/dist/plan-limits-NNJRAESF.js +27 -0
  374. package/dist/plan-limits-YTQW4UR4.js +27 -0
  375. package/dist/project-boot-46GZJTEX.js +299 -0
  376. package/dist/project-boot-PPHBBGIF.js +299 -0
  377. package/dist/projection-worker-UPAWXI7P.js +1034 -0
  378. package/dist/projection-worker-ZIKDYBW5.js +1034 -0
  379. package/dist/reranker-5ZBP2RRN.js +19 -0
  380. package/dist/reranker-E2MQIMJL.js +19 -0
  381. package/dist/reranker-GLSDJT3V.js +19 -0
  382. package/dist/reranker-LBBXWNOD.js +19 -0
  383. package/dist/reranker-XZ2EF4OH.js +19 -0
  384. package/dist/retrieval-health-JYRKPSII.js +7 -0
  385. package/dist/retrieval-health-OUV25J6S.js +7 -0
  386. package/dist/retrieval-health-U73JUAZL.js +7 -0
  387. package/dist/retrieval-health-WSZ7TYFF.js +7 -0
  388. package/dist/review-polling-62JV55ZT.js +125 -0
  389. package/dist/review-polling-CJXLWFWK.js +125 -0
  390. package/dist/runtime/index.js +12 -12
  391. package/dist/session-events-2ADD54VI.js +37 -0
  392. package/dist/session-events-QIJVBSKS.js +37 -0
  393. package/dist/session-kill-telemetry-HS6HD2YE.js +30 -0
  394. package/dist/session-kill-telemetry-MRT5FVSM.js +30 -0
  395. package/dist/session-scope-7ICYPC33.js +87 -0
  396. package/dist/session-scope-KMXD6EE6.js +87 -0
  397. package/dist/setup-wizard-B6GIT7YC.js +12 -0
  398. package/dist/setup-wizard-JUIJ4UZO.js +12 -0
  399. package/dist/skill-refinement-HIOX4VMC.js +158 -0
  400. package/dist/skill-refinement-T7JXRYUW.js +158 -0
  401. package/dist/stack-update-5KE6BZKQ.js +74 -0
  402. package/dist/stack-update-OP2RHP7N.js +74 -0
  403. package/dist/stack-update-VGCWDJEE.js +74 -0
  404. package/dist/steward-gate-L22WE3SY.js +14 -0
  405. package/dist/steward-gate-YKD2LUWN.js +14 -0
  406. package/dist/task-enforcement-5AOKXTY4.js +439 -0
  407. package/dist/task-enforcement-VO3YEGIO.js +439 -0
  408. package/dist/task-scope-YV2WPKRD.js +36 -0
  409. package/dist/task-scope-ZSXDZBRE.js +36 -0
  410. package/dist/tasks-crud-C6KADACT.js +78 -0
  411. package/dist/tasks-crud-NV6JEWGL.js +78 -0
  412. package/dist/tasks-notify-E22HSN6O.js +39 -0
  413. package/dist/tasks-notify-RPSEQ4WV.js +39 -0
  414. package/dist/tasks-review-V4ZLXOAZ.js +48 -0
  415. package/dist/tasks-review-ZVRI73JE.js +48 -0
  416. package/dist/telemetry-upload-LXUH7SKI.js +740 -0
  417. package/dist/telemetry-upload-TCDAZTUQ.js +740 -0
  418. package/dist/token-budget-OFBEZJTA.js +85 -0
  419. package/dist/token-budget-WAN57V6S.js +85 -0
  420. package/dist/tool-telemetry-UA3N32PK.js +17 -0
  421. package/dist/tool-telemetry-XXZJ35RR.js +17 -0
  422. package/dist/tui/App.js +17 -17
  423. package/dist/tui-data-46QLCJUE.js +259 -0
  424. package/dist/tui-data-ZDB7BLP2.js +259 -0
  425. package/dist/wiki-acl-HHSIBPF3.js +111 -0
  426. package/dist/wiki-acl-O65GZ2ZF.js +111 -0
  427. package/dist/worker-gate-27I4GAEZ.js +21 -0
  428. package/dist/worker-gate-DXU4HEPY.js +21 -0
  429. package/dist/workflow-engine-63EOEJ5Q.js +28 -0
  430. package/dist/workflow-engine-C6F2RMPN.js +28 -0
  431. package/dist/worktree-SFKKOMFD.js +27 -0
  432. package/dist/worktree-SVCE3S7X.js +27 -0
  433. package/dist/worktree-sweep-S3JHJTVP.js +20 -0
  434. package/dist/worktree-sweep-U3TIQ7WL.js +20 -0
  435. package/package.json +1 -1
  436. package/release-notes.json +88 -26
@@ -0,0 +1,284 @@
1
+ import {
2
+ routeTask
3
+ } from "./chunk-VB2N5WOX.js";
4
+ import {
5
+ createTaskCore,
6
+ employeeSessionName,
7
+ ensureEmployee,
8
+ getSessionState,
9
+ isEmployeeAlive,
10
+ sessionScopeFilter,
11
+ updateTaskStatus
12
+ } from "./chunk-PWPJK7KB.js";
13
+ import {
14
+ DEFAULT_COORDINATOR_TEMPLATE_NAME,
15
+ getCoordinatorName
16
+ } from "./chunk-WUKEXVOR.js";
17
+
18
+ // src/runtime/orchestrator.ts
19
+ var STALE_THRESHOLD_MS = 2 * 60 * 60 * 1e3;
20
+ var MultiAgentOrchestrator = class {
21
+ config;
22
+ constructor(config) {
23
+ this.config = config;
24
+ }
25
+ /**
26
+ * Route and dispatch a single task to the best-fit employee.
27
+ */
28
+ async dispatchTask(task) {
29
+ let targetEmployee;
30
+ let routingScore = 0;
31
+ if (task.assignTo) {
32
+ const found = this.config.employees.find((e) => e.name === task.assignTo);
33
+ if (!found) {
34
+ return {
35
+ employee: task.assignTo,
36
+ sessionName: "",
37
+ status: "failed",
38
+ routingScore: 0,
39
+ error: `Employee "${task.assignTo}" not found`
40
+ };
41
+ }
42
+ targetEmployee = found;
43
+ routingScore = 1;
44
+ } else {
45
+ const routed = await routeTask(
46
+ `${task.title}
47
+ ${task.context}`,
48
+ this.config.employees,
49
+ this.config.embedFn,
50
+ this.config.searchFn
51
+ );
52
+ targetEmployee = routed.employee;
53
+ routingScore = routed.score;
54
+ }
55
+ try {
56
+ await createTaskCore({
57
+ title: task.title,
58
+ assignedTo: targetEmployee.name,
59
+ assignedBy: getCoordinatorName(),
60
+ projectName: task.projectName,
61
+ priority: task.priority,
62
+ context: task.context,
63
+ baseDir: this.config.projectDir,
64
+ skipDispatch: true
65
+ });
66
+ } catch (err) {
67
+ return {
68
+ employee: targetEmployee.name,
69
+ sessionName: "",
70
+ status: "failed",
71
+ routingScore,
72
+ error: `Task creation failed: ${err instanceof Error ? err.message : String(err)}`
73
+ };
74
+ }
75
+ const result = ensureEmployee(
76
+ targetEmployee.name,
77
+ this.config.exeSession,
78
+ this.config.projectDir,
79
+ task.spawnOpts
80
+ );
81
+ return {
82
+ employee: targetEmployee.name,
83
+ sessionName: result.sessionName,
84
+ status: result.status === "failed" ? "failed" : result.status === "spawned" ? "dispatched" : "already_running",
85
+ routingScore,
86
+ error: result.error
87
+ };
88
+ }
89
+ /**
90
+ * Check health of all employees. Detect stuck and idle agents.
91
+ */
92
+ async healthCheck() {
93
+ const { getClient } = await import("./lib/database.js");
94
+ const client = getClient();
95
+ const employees = [];
96
+ const stuckAgents = [];
97
+ const idleAgents = [];
98
+ const crashedSessions = [];
99
+ let totalOpen = 0;
100
+ let totalInProgress = 0;
101
+ for (const emp of this.config.employees) {
102
+ const sessionName = employeeSessionName(emp.name, this.config.exeSession);
103
+ const alive = isEmployeeAlive(sessionName);
104
+ const state = alive ? getSessionState(sessionName) : "offline";
105
+ const oScope = sessionScopeFilter();
106
+ const tasks = await client.execute({
107
+ sql: `SELECT title, status, updated_at FROM tasks
108
+ WHERE assigned_to = ? AND status IN ('open', 'in_progress')${oScope.sql}
109
+ ORDER BY priority ASC`,
110
+ args: [emp.name, ...oScope.args]
111
+ });
112
+ const openCount = tasks.rows.filter((r) => r.status === "open").length;
113
+ const inProgressCount = tasks.rows.filter((r) => r.status === "in_progress").length;
114
+ totalOpen += openCount;
115
+ totalInProgress += inProgressCount;
116
+ const staleTasks = [];
117
+ for (const row of tasks.rows) {
118
+ if (row.status === "in_progress" && row.updated_at) {
119
+ const age = Date.now() - new Date(String(row.updated_at)).getTime();
120
+ if (age > STALE_THRESHOLD_MS) {
121
+ staleTasks.push(String(row.title));
122
+ }
123
+ }
124
+ }
125
+ const crashed = !alive && inProgressCount > 0;
126
+ if (staleTasks.length > 0) {
127
+ stuckAgents.push(emp.name);
128
+ }
129
+ if (crashed) {
130
+ crashedSessions.push(emp.name);
131
+ }
132
+ if (alive && state === "idle" && openCount + inProgressCount === 0) {
133
+ idleAgents.push(emp.name);
134
+ }
135
+ employees.push({
136
+ name: emp.name,
137
+ role: emp.role,
138
+ sessionName,
139
+ alive,
140
+ crashed,
141
+ state,
142
+ taskCount: openCount + inProgressCount,
143
+ inProgressCount,
144
+ staleTasks
145
+ });
146
+ }
147
+ return {
148
+ employees,
149
+ stuckAgents,
150
+ idleAgents,
151
+ crashedSessions,
152
+ totalOpenTasks: totalOpen,
153
+ totalInProgress
154
+ };
155
+ }
156
+ /**
157
+ * Process completed reviews. Auto-approve P2 tasks with passing results.
158
+ */
159
+ async processReviews() {
160
+ const { getClient } = await import("./lib/database.js");
161
+ const client = getClient();
162
+ const autoApproved = [];
163
+ const needsReview = [];
164
+ const coordinatorName = getCoordinatorName();
165
+ const rScope = sessionScopeFilter();
166
+ const reviews = await client.execute({
167
+ sql: `SELECT id, title, assigned_to, priority, result, task_file FROM tasks
168
+ WHERE (assigned_to = ? OR assigned_to = ?)
169
+ AND status IN ('open', 'in_progress')
170
+ AND task_file LIKE '%review-%'${rScope.sql}
171
+ ORDER BY priority ASC
172
+ LIMIT 20`,
173
+ args: [coordinatorName, DEFAULT_COORDINATOR_TEMPLATE_NAME, ...rScope.args]
174
+ });
175
+ for (const row of reviews.rows) {
176
+ const item = {
177
+ taskId: String(row.id),
178
+ title: String(row.title),
179
+ assignedTo: String(row.assigned_to),
180
+ priority: String(row.priority),
181
+ result: String(row.result ?? ""),
182
+ taskFile: String(row.task_file)
183
+ };
184
+ if (this.config.autoApproveP2 && item.priority === "p2" && /tests?\s+pass/i.test(item.result)) {
185
+ try {
186
+ await updateTaskStatus({
187
+ taskId: item.taskId,
188
+ status: "done",
189
+ result: `[auto-approved] ${item.result}`,
190
+ skipReviewCreation: true
191
+ });
192
+ } catch {
193
+ await client.execute({
194
+ sql: `UPDATE tasks SET status = 'done', result = ? WHERE id = ?`,
195
+ args: [`[auto-approved] ${item.result}`, item.taskId]
196
+ });
197
+ }
198
+ autoApproved.push(item);
199
+ } else {
200
+ needsReview.push(item);
201
+ }
202
+ }
203
+ return {
204
+ autoApproved,
205
+ needsReview,
206
+ processed: autoApproved.length + needsReview.length
207
+ };
208
+ }
209
+ /**
210
+ * Run one full orchestration cycle: health check + process reviews.
211
+ * Does NOT dispatch tasks — call dispatchTask() explicitly.
212
+ */
213
+ async tick() {
214
+ const health = await this.healthCheck();
215
+ const reviews = await this.processReviews();
216
+ for (const stuck of health.stuckAgents) {
217
+ const sessionName = employeeSessionName(stuck, this.config.exeSession);
218
+ if (isEmployeeAlive(sessionName)) {
219
+ const { sendIntercom } = await import("./lib/tmux-routing.js");
220
+ sendIntercom(sessionName);
221
+ }
222
+ }
223
+ for (const crashed of health.crashedSessions) {
224
+ this.recoverSession(crashed);
225
+ }
226
+ return {
227
+ dispatched: [],
228
+ health,
229
+ reviews,
230
+ timestamp: (/* @__PURE__ */ new Date()).toISOString()
231
+ };
232
+ }
233
+ /**
234
+ * Recover a crashed session by spawning a fresh CC session.
235
+ * The new session boots with identity via --system-prompt, finds existing in_progress tasks, and resumes.
236
+ */
237
+ recoverSession(employeeName) {
238
+ const sessionName = employeeSessionName(employeeName, this.config.exeSession);
239
+ const result = ensureEmployee(
240
+ employeeName,
241
+ this.config.exeSession,
242
+ this.config.projectDir
243
+ );
244
+ if (result.status === "failed") {
245
+ return { status: "failed", sessionName, error: result.error };
246
+ }
247
+ return { status: "recovered", sessionName: result.sessionName };
248
+ }
249
+ /**
250
+ * Dispatch a batch of tasks, auto-routing each to the best employee.
251
+ */
252
+ async dispatchBatch(tasks) {
253
+ const results = [];
254
+ for (const task of tasks) {
255
+ const result = await this.dispatchTask(task);
256
+ results.push(result);
257
+ }
258
+ return results;
259
+ }
260
+ /**
261
+ * Get a formatted status summary (matches exe boot brief style).
262
+ */
263
+ async getStatusSummary() {
264
+ const health = await this.healthCheck();
265
+ const lines = [];
266
+ lines.push("\u25B8 ORCHESTRATOR STATUS");
267
+ lines.push(` Tasks: ${health.totalOpenTasks} open, ${health.totalInProgress} in progress`);
268
+ lines.push(` Crashed: ${health.crashedSessions.length > 0 ? health.crashedSessions.join(", ") : "none"}`);
269
+ lines.push(` Stuck: ${health.stuckAgents.length > 0 ? health.stuckAgents.join(", ") : "none"}`);
270
+ lines.push(` Idle: ${health.idleAgents.length > 0 ? health.idleAgents.join(", ") : "none"}`);
271
+ lines.push("");
272
+ for (const emp of health.employees) {
273
+ const stateIcon = emp.alive ? emp.state === "idle" ? "\u{1F7E2}" : emp.state === "thinking" ? "\u{1F7E1}" : emp.state === "tool" ? "\u{1F535}" : "\u26AA" : "\u{1F534}";
274
+ const taskInfo = emp.taskCount > 0 ? `${emp.taskCount} task${emp.taskCount > 1 ? "s" : ""} (${emp.inProgressCount} active)` : "idle";
275
+ const staleWarning = emp.staleTasks.length > 0 ? ` \u26A0 stale: ${emp.staleTasks[0]}` : "";
276
+ lines.push(` ${stateIcon} ${emp.name} (${emp.role}): ${taskInfo}${staleWarning}`);
277
+ }
278
+ return lines.join("\n");
279
+ }
280
+ };
281
+
282
+ export {
283
+ MultiAgentOrchestrator
284
+ };
@@ -0,0 +1,375 @@
1
+ import {
2
+ EXE_AI_DIR
3
+ } from "./chunk-T3B5RK4H.js";
4
+
5
+ // src/lib/license.ts
6
+ import { readFileSync, writeFileSync, existsSync, mkdirSync, statSync } from "fs";
7
+ import { randomUUID } from "crypto";
8
+ import path from "path";
9
+ import { jwtVerify, importSPKI } from "jose";
10
+ var LICENSE_PATH = path.join(EXE_AI_DIR, "license.key");
11
+ var CACHE_PATH = path.join(EXE_AI_DIR, "license-cache.json");
12
+ var DEVICE_ID_PATH = path.join(EXE_AI_DIR, "device-id");
13
+ var API_BASE = process.env.EXE_CLOUD_ENDPOINT ?? "https://api.askexe.com";
14
+ var RETRY_DELAY_MS = 500;
15
+ async function fetchRetry(url, init) {
16
+ try {
17
+ return await fetch(url, init);
18
+ } catch {
19
+ await new Promise((r) => setTimeout(r, RETRY_DELAY_MS));
20
+ return fetch(url, { ...init, signal: AbortSignal.timeout(1e4) });
21
+ }
22
+ }
23
+ var LICENSE_PUBLIC_KEY_PEM = `-----BEGIN PUBLIC KEY-----
24
+ MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEeHztAMOpR/ZMh+rWuOASjEZ54CGY
25
+ 4uj+UqeKCcvtgNHKmOK278HJaJcANe9xAeji8AFYu27q3WtzCi04pHudow==
26
+ -----END PUBLIC KEY-----`;
27
+ var LICENSE_PUBLIC_KEY_PEM_PREV = `-----BEGIN PUBLIC KEY-----
28
+ MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEVlMxzSQraFaXp+qRyRwDNg25R4qH
29
+ sV3p1+vXy1GH/9W2VQyq3pUvOlAeeDMMT+hbL5kbtwCl7daoaZZ8Tv8acw==
30
+ -----END PUBLIC KEY-----`;
31
+ var LICENSE_PUBLIC_KEYS = [LICENSE_PUBLIC_KEY_PEM, LICENSE_PUBLIC_KEY_PEM_PREV];
32
+ var LICENSE_JWT_ALG = "ES256";
33
+ var PLAN_LIMITS = {
34
+ free: { devices: 1, employees: 1, memories: 5e3 },
35
+ pro: { devices: 3, employees: 5, memories: 1e5 },
36
+ team: { devices: 10, employees: 20, memories: 1e6 },
37
+ agency: { devices: 50, employees: 100, memories: 1e7 },
38
+ enterprise: { devices: -1, employees: -1, memories: -1 }
39
+ };
40
+ var FREE_LICENSE = {
41
+ valid: true,
42
+ plan: "free",
43
+ email: "",
44
+ expiresAt: null,
45
+ deviceLimit: 1,
46
+ employeeLimit: 1,
47
+ memoryLimit: 5e3
48
+ };
49
+ function loadDeviceId() {
50
+ const deviceJsonPath = path.join(EXE_AI_DIR, "device.json");
51
+ try {
52
+ if (existsSync(deviceJsonPath)) {
53
+ const data = JSON.parse(readFileSync(deviceJsonPath, "utf8"));
54
+ if (data.deviceId) return data.deviceId;
55
+ }
56
+ } catch {
57
+ }
58
+ try {
59
+ if (existsSync(DEVICE_ID_PATH)) {
60
+ const id2 = readFileSync(DEVICE_ID_PATH, "utf8").trim();
61
+ if (id2) return id2;
62
+ }
63
+ } catch {
64
+ }
65
+ const id = randomUUID();
66
+ mkdirSync(EXE_AI_DIR, { recursive: true });
67
+ writeFileSync(DEVICE_ID_PATH, id, "utf8");
68
+ return id;
69
+ }
70
+ function loadLicense() {
71
+ try {
72
+ if (!existsSync(LICENSE_PATH)) return null;
73
+ return readFileSync(LICENSE_PATH, "utf8").trim();
74
+ } catch {
75
+ return null;
76
+ }
77
+ }
78
+ function saveLicense(apiKey) {
79
+ mkdirSync(EXE_AI_DIR, { recursive: true });
80
+ writeFileSync(LICENSE_PATH, apiKey.trim(), { encoding: "utf8", mode: 384 });
81
+ }
82
+ async function verifyLicenseJwt(token) {
83
+ for (const pem of LICENSE_PUBLIC_KEYS) {
84
+ try {
85
+ const key = await importSPKI(pem, LICENSE_JWT_ALG);
86
+ const { payload } = await jwtVerify(token, key, {
87
+ algorithms: [LICENSE_JWT_ALG]
88
+ });
89
+ const plan = payload.plan ?? "free";
90
+ const email = payload.sub ?? "";
91
+ const limits = PLAN_LIMITS[plan] ?? PLAN_LIMITS.free;
92
+ return {
93
+ valid: true,
94
+ plan,
95
+ email,
96
+ expiresAt: payload.exp ? new Date(payload.exp * 1e3).toISOString() : null,
97
+ deviceLimit: limits.devices,
98
+ employeeLimit: limits.employees,
99
+ memoryLimit: limits.memories
100
+ };
101
+ } catch {
102
+ continue;
103
+ }
104
+ }
105
+ return null;
106
+ }
107
+ async function getCachedLicense() {
108
+ try {
109
+ if (!existsSync(CACHE_PATH)) return null;
110
+ const raw = JSON.parse(readFileSync(CACHE_PATH, "utf8"));
111
+ if (!raw.token || typeof raw.token !== "string") return null;
112
+ return await verifyLicenseJwt(raw.token);
113
+ } catch {
114
+ return null;
115
+ }
116
+ }
117
+ async function getGraceVerifiedLicense(graceDays = 7) {
118
+ try {
119
+ const token = readCachedLicenseToken();
120
+ if (!token) return null;
121
+ const payloadB64 = token.split(".")[1];
122
+ if (!payloadB64) return null;
123
+ const payload = JSON.parse(Buffer.from(payloadB64, "base64url").toString());
124
+ const expMs = (payload.exp ?? 0) * 1e3;
125
+ if (Date.now() >= expMs + graceDays * 24 * 60 * 60 * 1e3) return null;
126
+ let verified = null;
127
+ for (const pem of LICENSE_PUBLIC_KEYS) {
128
+ try {
129
+ const key = await importSPKI(pem, LICENSE_JWT_ALG);
130
+ const result = await jwtVerify(token, key, {
131
+ algorithms: [LICENSE_JWT_ALG],
132
+ clockTolerance: graceDays * 24 * 60 * 60
133
+ });
134
+ verified = result.payload;
135
+ break;
136
+ } catch {
137
+ continue;
138
+ }
139
+ }
140
+ if (!verified) return null;
141
+ const plan = verified.plan ?? "free";
142
+ const limits = PLAN_LIMITS[plan] ?? PLAN_LIMITS.free;
143
+ process.stderr.write(
144
+ `[license] Offline grace: verified cached plan "${plan}" (expired, within ${graceDays}d grace).
145
+ `
146
+ );
147
+ return {
148
+ valid: true,
149
+ plan,
150
+ email: verified.sub ?? "",
151
+ expiresAt: verified.exp ? new Date(verified.exp * 1e3).toISOString() : null,
152
+ deviceLimit: limits.devices,
153
+ employeeLimit: limits.employees,
154
+ memoryLimit: limits.memories
155
+ };
156
+ } catch {
157
+ return null;
158
+ }
159
+ }
160
+ function readCachedLicenseToken() {
161
+ try {
162
+ if (!existsSync(CACHE_PATH)) return null;
163
+ const raw = JSON.parse(readFileSync(CACHE_PATH, "utf8"));
164
+ return typeof raw.token === "string" ? raw.token : null;
165
+ } catch {
166
+ return null;
167
+ }
168
+ }
169
+ function cacheResponse(token) {
170
+ try {
171
+ writeFileSync(CACHE_PATH, JSON.stringify({ token }), "utf8");
172
+ } catch {
173
+ }
174
+ }
175
+ async function validateViaCFWorker(apiKey, deviceId) {
176
+ try {
177
+ const res = await fetchRetry(`${API_BASE}/auth/activate`, {
178
+ method: "POST",
179
+ headers: { "Content-Type": "application/json" },
180
+ body: JSON.stringify({ apiKey, deviceId }),
181
+ signal: AbortSignal.timeout(1e4)
182
+ });
183
+ if (!res.ok) return null;
184
+ const data = await res.json();
185
+ if (data.error === "device_limit_exceeded") return null;
186
+ if (!data.valid) return null;
187
+ if (data.token) {
188
+ cacheResponse(data.token);
189
+ const verified = await verifyLicenseJwt(data.token);
190
+ if (verified) return verified;
191
+ }
192
+ const limits = PLAN_LIMITS[data.plan] ?? PLAN_LIMITS.free;
193
+ return {
194
+ valid: data.valid,
195
+ plan: data.plan,
196
+ email: data.email,
197
+ expiresAt: data.expiresAt,
198
+ deviceLimit: limits.devices,
199
+ employeeLimit: limits.employees,
200
+ memoryLimit: limits.memories
201
+ };
202
+ } catch {
203
+ return null;
204
+ }
205
+ }
206
+ async function validateLicense(apiKey, deviceId) {
207
+ const did = deviceId ?? loadDeviceId();
208
+ const cfResult = await validateViaCFWorker(apiKey, did);
209
+ if (cfResult) return cfResult;
210
+ const cached = await getCachedLicense();
211
+ if (cached) return cached;
212
+ const graced = await getGraceVerifiedLicense();
213
+ if (graced) return graced;
214
+ return { ...FREE_LICENSE, valid: false };
215
+ }
216
+ var CACHE_MAX_AGE_MS = 36e5;
217
+ function getCacheAgeMs() {
218
+ try {
219
+ const s = statSync(CACHE_PATH);
220
+ return Date.now() - s.mtimeMs;
221
+ } catch {
222
+ return Infinity;
223
+ }
224
+ }
225
+ async function checkLicense() {
226
+ let key = loadLicense();
227
+ if (!key) {
228
+ try {
229
+ const configPath = path.join(EXE_AI_DIR, "config.json");
230
+ if (existsSync(configPath)) {
231
+ const raw = JSON.parse(readFileSync(configPath, "utf8"));
232
+ const cloud = raw.cloud;
233
+ if (cloud?.apiKey) {
234
+ key = cloud.apiKey;
235
+ saveLicense(key);
236
+ }
237
+ }
238
+ } catch {
239
+ }
240
+ }
241
+ if (!key) return FREE_LICENSE;
242
+ const cached = await getCachedLicense();
243
+ if (cached && getCacheAgeMs() < CACHE_MAX_AGE_MS) return cached;
244
+ const deviceId = loadDeviceId();
245
+ return validateLicense(key, deviceId);
246
+ }
247
+ function isFeatureAllowed(license, feature) {
248
+ switch (feature) {
249
+ case "cloud_sync":
250
+ case "external_agents":
251
+ case "wiki":
252
+ return license.plan !== "free";
253
+ case "unlimited_employees":
254
+ return license.plan === "team" || license.plan === "agency" || license.plan === "enterprise";
255
+ }
256
+ }
257
+ function mirrorLicenseKey(apiKey) {
258
+ const trimmed = apiKey.trim();
259
+ if (!trimmed) return;
260
+ saveLicense(trimmed);
261
+ }
262
+ async function assertVpsLicense(opts) {
263
+ const env = opts?.env ?? process.env;
264
+ const inProduction = env.NODE_ENV === "production";
265
+ if (!opts?.force && !inProduction) {
266
+ return { ...FREE_LICENSE, plan: "free" };
267
+ }
268
+ const envKey = env.EXE_LICENSE_KEY?.trim();
269
+ if (envKey) {
270
+ saveLicense(envKey);
271
+ }
272
+ const apiKey = envKey ?? loadLicense();
273
+ if (!apiKey) {
274
+ throw new Error(
275
+ "License required: set EXE_LICENSE_KEY env var with your exe_sk_* key. Purchase at https://askexe.com. This VPS image refuses to boot without a valid license."
276
+ );
277
+ }
278
+ const deviceId = loadDeviceId();
279
+ let backendResponse = null;
280
+ let explicitRejection = false;
281
+ let transientFailure = false;
282
+ try {
283
+ const res = await fetchRetry(`${API_BASE}/auth/activate`, {
284
+ method: "POST",
285
+ headers: { "Content-Type": "application/json" },
286
+ body: JSON.stringify({ apiKey, deviceId }),
287
+ signal: AbortSignal.timeout(1e4)
288
+ });
289
+ if (res.ok) {
290
+ backendResponse = await res.json();
291
+ if (!backendResponse.valid) explicitRejection = true;
292
+ } else if (res.status === 401 || res.status === 403) {
293
+ explicitRejection = true;
294
+ } else {
295
+ transientFailure = true;
296
+ }
297
+ } catch {
298
+ transientFailure = true;
299
+ }
300
+ if (backendResponse?.valid) {
301
+ if (backendResponse.token) {
302
+ cacheResponse(backendResponse.token);
303
+ const verified = await verifyLicenseJwt(backendResponse.token);
304
+ if (verified) return verified;
305
+ }
306
+ const limits = PLAN_LIMITS[backendResponse.plan] ?? PLAN_LIMITS.free;
307
+ return {
308
+ valid: true,
309
+ plan: backendResponse.plan,
310
+ email: backendResponse.email,
311
+ expiresAt: backendResponse.expiresAt,
312
+ deviceLimit: limits.devices,
313
+ employeeLimit: limits.employees,
314
+ memoryLimit: limits.memories
315
+ };
316
+ }
317
+ if (explicitRejection) {
318
+ throw new Error(
319
+ `License invalid or expired. Renew at https://askexe.com, then restart. Backend rejected key ending in ...${apiKey.slice(-4)}.`
320
+ );
321
+ }
322
+ if (!transientFailure) {
323
+ throw new Error(
324
+ "License validation failed: unknown backend state. Restore network connectivity to https://api.askexe.com and retry."
325
+ );
326
+ }
327
+ const fresh = await getCachedLicense();
328
+ if (fresh && fresh.valid) return fresh;
329
+ const graceDays = opts?.offlineGraceDays ?? 7;
330
+ const graced = await getGraceVerifiedLicense(graceDays);
331
+ if (graced) return graced;
332
+ throw new Error(
333
+ `License validation unreachable for more than ${graceDays} days. Restore network connectivity to https://api.askexe.com and retry. This VPS image refuses to boot after the offline grace window.`
334
+ );
335
+ }
336
+ var _revalTimer = null;
337
+ function startLicenseRevalidation(intervalMs = 36e5) {
338
+ if (_revalTimer) return;
339
+ _revalTimer = setInterval(async () => {
340
+ try {
341
+ const license = await checkLicense();
342
+ if (!license.valid) {
343
+ process.stderr.write("[exe-os] License expired or invalid \u2014 features may be restricted\n");
344
+ }
345
+ } catch {
346
+ }
347
+ }, intervalMs);
348
+ if (_revalTimer && typeof _revalTimer === "object" && "unref" in _revalTimer) {
349
+ _revalTimer.unref();
350
+ }
351
+ }
352
+ function stopLicenseRevalidation() {
353
+ if (_revalTimer) {
354
+ clearInterval(_revalTimer);
355
+ _revalTimer = null;
356
+ }
357
+ }
358
+
359
+ export {
360
+ LICENSE_PUBLIC_KEY_PEM,
361
+ PLAN_LIMITS,
362
+ loadDeviceId,
363
+ loadLicense,
364
+ saveLicense,
365
+ getCachedLicense,
366
+ getGraceVerifiedLicense,
367
+ readCachedLicenseToken,
368
+ validateLicense,
369
+ checkLicense,
370
+ isFeatureAllowed,
371
+ mirrorLicenseKey,
372
+ assertVpsLicense,
373
+ startLicenseRevalidation,
374
+ stopLicenseRevalidation
375
+ };