@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,369 @@
1
+ import {
2
+ resolveSupportAdminToken,
3
+ supportAdminHeaders
4
+ } from "./chunk-2XZ6X3PJ.js";
5
+ import {
6
+ markSupportReportSentSync
7
+ } from "./chunk-MMRUBN3I.js";
8
+ import {
9
+ loadLicense,
10
+ readCachedLicenseToken
11
+ } from "./chunk-EAPUSVKS.js";
12
+ import {
13
+ EXE_AI_DIR,
14
+ loadConfig
15
+ } from "./chunk-T3B5RK4H.js";
16
+
17
+ // src/bin/exe-support.ts
18
+ import { mkdirSync, readFileSync, unlinkSync, writeFileSync } from "fs";
19
+ import path from "path";
20
+ import { randomUUID } from "crypto";
21
+ var DEFAULT_BUG_ENDPOINT = "https://api.askexe.com/v1/support/bug-reports";
22
+ var DEFAULT_ADMIN_ENDPOINT = "https://api.askexe.com/admin/support/bug-reports";
23
+ async function main(argv = process.argv.slice(2)) {
24
+ const command = argv[0] && !argv[0].startsWith("--") ? argv[0] : "health";
25
+ const json = argv.includes("--json");
26
+ const project = getArg(argv, "--project") ?? "support-smoke";
27
+ if (command === "help" || argv.includes("--help") || argv.includes("-h")) {
28
+ printHelp();
29
+ return;
30
+ }
31
+ if (command === "health") {
32
+ const result = await runHealth();
33
+ output(result, json, "health");
34
+ process.exitCode = hasFailures(result) ? 1 : 0;
35
+ return;
36
+ }
37
+ if (command === "test") {
38
+ const result = await runTest(project);
39
+ output(result, json, "test");
40
+ process.exitCode = hasFailures(result) ? 1 : 0;
41
+ return;
42
+ }
43
+ if (command === "flush") {
44
+ await runFlush(json);
45
+ return;
46
+ }
47
+ if (command === "status") {
48
+ await runStatus(json);
49
+ return;
50
+ }
51
+ console.error("Usage: exe-os support health|test|flush|status [--project <name>] [--json]");
52
+ process.exitCode = 1;
53
+ }
54
+ async function runHealth() {
55
+ const checks = [];
56
+ const endpoints = await resolveEndpoints();
57
+ checks.push(checkLocalWrite());
58
+ checks.push({
59
+ check: "license_key_present",
60
+ level: loadLicense() ? "pass" : "fail",
61
+ detail: loadLicense() ? "license.key found" : "missing ~/.exe-os/license.key; run exe-os setup or exe-os cloud setup",
62
+ next: loadLicense() ? void 0 : "Run `exe-os setup` or ask AskExe for the customer license key."
63
+ });
64
+ checks.push({
65
+ check: "license_token_cached",
66
+ level: readCachedLicenseToken() ? "pass" : "warn",
67
+ detail: readCachedLicenseToken() ? "cached license token found" : "no cached token yet; support can still use license.key",
68
+ next: readCachedLicenseToken() ? void 0 : "This is OK if license.key exists. It refreshes after cloud/license validation."
69
+ });
70
+ try {
71
+ const res = await fetch(endpoints.healthEndpoint, { method: "GET", signal: AbortSignal.timeout(1e4) });
72
+ const body = await safeJson(res);
73
+ checks.push({
74
+ check: "support_health_endpoint",
75
+ level: res.ok && body?.status !== "down" ? "pass" : "fail",
76
+ detail: `${res.status} ${body?.status ?? res.statusText}`,
77
+ next: res.ok ? void 0 : "Check internet access, DNS, or AskExe support status."
78
+ });
79
+ for (const remote of body?.checks ?? []) {
80
+ const name = remote.name ?? "unknown";
81
+ checks.push({
82
+ check: `server_${name}`,
83
+ level: remote.ok === true ? "pass" : name === "admin_inbox_configured" ? "warn" : "fail",
84
+ detail: remote.detail ?? (remote.ok ? "ok" : "failed"),
85
+ next: remote.ok ? void 0 : "AskExe must fix this server-side."
86
+ });
87
+ }
88
+ } catch (err) {
89
+ checks.push({
90
+ check: "support_health_endpoint",
91
+ level: "fail",
92
+ detail: err instanceof Error ? err.message : String(err),
93
+ next: "Check internet access, then run `exe-os support health` again."
94
+ });
95
+ }
96
+ return checks;
97
+ }
98
+ async function runTest(project) {
99
+ const checks = await runHealth();
100
+ const endpoints = await resolveEndpoints();
101
+ const config = await loadConfig();
102
+ const licenseKey = loadLicense();
103
+ const licenseToken = readCachedLicenseToken();
104
+ const id = randomUUID();
105
+ const version = readPackageVersion();
106
+ const reportPath = writeLocalTestReport(id, project, version);
107
+ checks.push({ check: "local_report_file", level: "pass", detail: reportPath });
108
+ if (!licenseKey && !licenseToken) {
109
+ checks.push({
110
+ check: "upstream_post",
111
+ level: "fail",
112
+ detail: "not sent because this device has no license key/token",
113
+ next: "Run `exe-os setup` or ask AskExe to provision this customer device."
114
+ });
115
+ return checks;
116
+ }
117
+ const payload = {
118
+ id,
119
+ title: `TEST \u2014 ${project} support intake (${version})`,
120
+ classification: "unclear",
121
+ severity: "p3",
122
+ summary: "Synthetic exe-os support intake smoke test. Safe to close.",
123
+ customer_impact: "No customer impact; diagnostic only.",
124
+ reproduction_steps: ["Run exe-os support test"],
125
+ expected: "The report reaches AskExe support intake and can be triaged.",
126
+ actual: "Smoke test submitted by exe-os support test.",
127
+ package_version: `@askexenow/exe-os@${version}`,
128
+ project_name: project,
129
+ agent_id: "support-test",
130
+ agent_role: "diagnostic",
131
+ report_path: reportPath,
132
+ markdown: readFileSync(reportPath, "utf8")
133
+ };
134
+ try {
135
+ const headers = {
136
+ "content-type": "application/json",
137
+ ...supportAdminHeaders(config)
138
+ };
139
+ if (licenseKey) headers["x-exe-license-key"] = licenseKey;
140
+ if (licenseToken) headers["x-exe-license-token"] = licenseToken;
141
+ const res = await fetch(endpoints.bugEndpoint, {
142
+ method: "POST",
143
+ headers,
144
+ body: JSON.stringify(payload),
145
+ signal: AbortSignal.timeout(15e3)
146
+ });
147
+ const data = await safeJson(res);
148
+ checks.push({
149
+ check: "upstream_post",
150
+ level: res.ok ? "pass" : "fail",
151
+ detail: res.ok ? `sent id=${String(data?.id ?? id)}` : summarizeHttpFailure(res.status, data),
152
+ next: res.ok ? void 0 : nextForPostFailure(res.status)
153
+ });
154
+ if (res.ok) {
155
+ markSupportReportSentSync(reportPath);
156
+ checks.push(await maybeCloseAdmin(String(data?.id ?? id), endpoints.adminEndpoint, version, config));
157
+ }
158
+ } catch (err) {
159
+ checks.push({
160
+ check: "upstream_post",
161
+ level: "fail",
162
+ detail: err instanceof Error ? err.message : String(err),
163
+ next: "Check internet access, then run `exe-os support test` again."
164
+ });
165
+ }
166
+ return checks;
167
+ }
168
+ async function resolveEndpoints() {
169
+ const config = await loadConfig();
170
+ const bugEndpoint = process.env.EXE_SUPPORT_BUG_REPORT_ENDPOINT ?? config.support?.bugReportEndpoint ?? DEFAULT_BUG_ENDPOINT;
171
+ const healthEndpoint = process.env.EXE_SUPPORT_HEALTH_ENDPOINT ?? bugEndpoint.replace(/\/bug-reports\/?$/, "/health");
172
+ const adminEndpoint = process.env.ASKEXE_SUPPORT_ADMIN_ENDPOINT ?? process.env.EXE_SUPPORT_ADMIN_ENDPOINT ?? DEFAULT_ADMIN_ENDPOINT;
173
+ return { bugEndpoint, healthEndpoint, adminEndpoint };
174
+ }
175
+ function checkLocalWrite() {
176
+ const dir = path.join(EXE_AI_DIR, "bug-reports");
177
+ const testPath = path.join(dir, ".support-write-test");
178
+ try {
179
+ mkdirSync(dir, { recursive: true, mode: 448 });
180
+ writeFileSync(testPath, "ok\n", { mode: 384 });
181
+ unlinkSync(testPath);
182
+ return { check: "local_bug_report_dir_writable", level: "pass", detail: dir };
183
+ } catch (err) {
184
+ return {
185
+ check: "local_bug_report_dir_writable",
186
+ level: "fail",
187
+ detail: err instanceof Error ? err.message : String(err),
188
+ next: "Fix permissions on ~/.exe-os or rerun setup on a writable user account."
189
+ };
190
+ }
191
+ }
192
+ function writeLocalTestReport(id, project, version) {
193
+ const dir = path.join(EXE_AI_DIR, "bug-reports");
194
+ mkdirSync(dir, { recursive: true, mode: 448 });
195
+ const date = (/* @__PURE__ */ new Date()).toISOString().slice(0, 10);
196
+ const filePath = path.join(dir, `${date}-support-intake-test-${id.slice(0, 8)}.md`);
197
+ writeFileSync(filePath, `# TEST \u2014 ${project} support intake
198
+
199
+ Report ID: ${id}
200
+ Package: @askexenow/exe-os@${version}
201
+
202
+ Synthetic smoke test from \`exe-os support test\`. Safe to close.
203
+ `, { mode: 384 });
204
+ return filePath;
205
+ }
206
+ async function maybeCloseAdmin(id, adminEndpoint, version, config) {
207
+ const token = resolveSupportAdminToken(config);
208
+ if (!token) {
209
+ return { check: "askexe_admin_autoclose", level: "warn", detail: "skipped; admin token is AskExe-only" };
210
+ }
211
+ try {
212
+ const res = await fetch(`${adminEndpoint}/${encodeURIComponent(id)}`, {
213
+ method: "PATCH",
214
+ headers: { authorization: `Bearer ${token}`, "content-type": "application/json" },
215
+ body: JSON.stringify({
216
+ status: "closed",
217
+ triage_notes: "Auto-closed synthetic support smoke test.",
218
+ fixed_version: version
219
+ }),
220
+ signal: AbortSignal.timeout(1e4)
221
+ });
222
+ return {
223
+ check: "askexe_admin_autoclose",
224
+ level: res.ok ? "pass" : "warn",
225
+ detail: res.ok ? "closed" : `${res.status} ${await res.text()}`,
226
+ next: res.ok ? void 0 : "Report was sent; AskExe can close the smoke report manually."
227
+ };
228
+ } catch (err) {
229
+ return {
230
+ check: "askexe_admin_autoclose",
231
+ level: "warn",
232
+ detail: err instanceof Error ? err.message : String(err),
233
+ next: "Report was sent; AskExe can close the smoke report manually."
234
+ };
235
+ }
236
+ }
237
+ function readPackageVersion() {
238
+ let dir = path.dirname(new URL(import.meta.url).pathname);
239
+ for (let i = 0; i < 6; i++) {
240
+ const pkg = path.join(dir, "package.json");
241
+ try {
242
+ const parsed = JSON.parse(readFileSync(pkg, "utf8"));
243
+ if (parsed.version) return parsed.version;
244
+ } catch {
245
+ }
246
+ dir = path.dirname(dir);
247
+ }
248
+ return "unknown";
249
+ }
250
+ async function safeJson(res) {
251
+ try {
252
+ return await res.json();
253
+ } catch {
254
+ return null;
255
+ }
256
+ }
257
+ function getArg(argv, name) {
258
+ const idx = argv.indexOf(name);
259
+ if (idx >= 0) return argv[idx + 1];
260
+ const prefix = `${name}=`;
261
+ const found = argv.find((arg) => arg.startsWith(prefix));
262
+ return found?.slice(prefix.length);
263
+ }
264
+ function hasFailures(rows) {
265
+ return rows.some((row) => row.level === "fail");
266
+ }
267
+ function summarizeHttpFailure(status, data) {
268
+ const error = data?.error;
269
+ return `${status} ${error?.message ?? JSON.stringify(data)}`;
270
+ }
271
+ function nextForPostFailure(status) {
272
+ if (status === 401) return "License credentials were not sent. Run `exe-os support health` and check license_key_present.";
273
+ if (status === 403) return "License is valid locally but not accepted by support intake. AskExe must check server-side license provisioning.";
274
+ if (status >= 500) return "AskExe support intake is unhealthy server-side. Try again shortly and report the local file path.";
275
+ return "Run `exe-os support health`; if it passes, send this output to AskExe.";
276
+ }
277
+ async function runFlush(json) {
278
+ const { flushSupportOutbox } = await import("./support-outbox-YLXLMEKF.js");
279
+ const result = await flushSupportOutbox({ maxPerFlush: 10 });
280
+ if (json) {
281
+ console.log(JSON.stringify(result, null, 2));
282
+ return;
283
+ }
284
+ console.log("\nexe-os support flush\n");
285
+ console.log(` Bug reports sent: ${result.bugsSent}`);
286
+ console.log(` Feature requests sent: ${result.featuresSent}`);
287
+ console.log(` Errors: ${result.errors}`);
288
+ console.log(` Skipped: ${result.skipped}`);
289
+ console.log("");
290
+ const total = result.bugsSent + result.featuresSent;
291
+ if (total > 0) {
292
+ console.log(`Flushed ${total} item(s) to AskExe.`);
293
+ } else if (result.errors > 0) {
294
+ console.log("Nothing sent \u2014 network errors occurred. Retry later.");
295
+ } else {
296
+ console.log("Outbox is empty \u2014 nothing to flush.");
297
+ }
298
+ console.log("");
299
+ }
300
+ async function runStatus(json) {
301
+ const { getOutboxStatus } = await import("./support-outbox-YLXLMEKF.js");
302
+ const status = await getOutboxStatus();
303
+ if (json) {
304
+ console.log(JSON.stringify(status, null, 2));
305
+ return;
306
+ }
307
+ console.log("\nexe-os support status\n");
308
+ const bugs = status.bugReports;
309
+ const features = status.featureRequests;
310
+ console.log(` Bug reports: ${bugs.unsent} unsent / ${bugs.total} total${bugs.oldest ? ` (oldest: ${bugs.oldest.slice(0, 10)})` : ""}`);
311
+ console.log(` Feature requests: ${features.unsent} unsent / ${features.total} total${features.oldest ? ` (oldest: ${features.oldest.slice(0, 10)})` : ""}`);
312
+ console.log("");
313
+ const totalUnsent = bugs.unsent + features.unsent;
314
+ if (totalUnsent > 0) {
315
+ console.log(`${totalUnsent} unsent item(s). Run \`exe-os support flush\` to send manually.`);
316
+ } else {
317
+ console.log("All items sent \u2014 outbox is clear.");
318
+ }
319
+ console.log("");
320
+ }
321
+ function printHelp() {
322
+ console.log(`
323
+ exe-os support
324
+
325
+ Customer-safe diagnostics for AskExe support intake.
326
+
327
+ Commands:
328
+ exe-os support health Check local setup + AskExe support server health
329
+ exe-os support test --project hygo Send a safe smoke report end-to-end
330
+ exe-os support flush Force-flush unsent bug reports + feature requests
331
+ exe-os support status Show outbox status (unsent counts)
332
+
333
+ What success means:
334
+ health = this device looks ready
335
+ test = AskExe received a real report from this device
336
+ flush = all unsent items attempted delivery
337
+ status = visibility into what's queued
338
+ `);
339
+ }
340
+ function output(rows, json, command) {
341
+ if (json) {
342
+ console.log(JSON.stringify({ ok: !hasFailures(rows), checks: rows }, null, 2));
343
+ return;
344
+ }
345
+ console.log(`
346
+ exe-os support ${command}
347
+ `);
348
+ for (const row of rows) {
349
+ const icon = row.level === "pass" ? "\u2705" : row.level === "warn" ? "\u26A0\uFE0F " : "\u274C";
350
+ console.log(`${icon} ${row.check}: ${row.detail}`);
351
+ if (row.next) console.log(` \u2192 ${row.next}`);
352
+ }
353
+ console.log("");
354
+ if (hasFailures(rows)) {
355
+ console.log("Result: not ready. Fix the failed item(s), then rerun this command.");
356
+ } else if (rows.some((row) => row.level === "warn")) {
357
+ console.log("Result: ready with warnings. Bug reports can be sent; warnings are informational unless AskExe asked for stricter validation.");
358
+ } else {
359
+ console.log(command === "test" ? "Result: ready. AskExe received the test report." : "Result: ready. Run `exe-os support test --project <customer>` for an end-to-end send test.");
360
+ }
361
+ console.log("");
362
+ }
363
+
364
+ export {
365
+ main,
366
+ runHealth,
367
+ runTest,
368
+ hasFailures
369
+ };
@@ -0,0 +1,197 @@
1
+ import {
2
+ initStore
3
+ } from "./chunk-AXOREWVL.js";
4
+ import {
5
+ getClient,
6
+ isInitialized
7
+ } from "./chunk-WUKEXVOR.js";
8
+
9
+ // src/lib/schedules.ts
10
+ import crypto from "crypto";
11
+ import { execSync } from "child_process";
12
+ var CRON_FIELD = /^[\d*/,\-]+$/;
13
+ function isValidCron(cron) {
14
+ const fields = cron.trim().split(/\s+/);
15
+ if (fields.length !== 5) return false;
16
+ return fields.every((f) => CRON_FIELD.test(f));
17
+ }
18
+ var SAFE_ID = /^[a-zA-Z0-9_\-]+$/;
19
+ function isValidScheduleId(id) {
20
+ return SAFE_ID.test(id) && id.length <= 128;
21
+ }
22
+ async function ensureDb() {
23
+ if (!isInitialized()) {
24
+ await initStore();
25
+ }
26
+ }
27
+ function parseHumanCron(input) {
28
+ const s = input.toLowerCase().trim();
29
+ if (/^[\d*\/,-]+\s+[\d*\/,-]+\s+[\d*\/,-]+\s+[\d*\/,-]+\s+[\d*\/,-]+$/.test(s)) {
30
+ return s;
31
+ }
32
+ const dayMap = {
33
+ sunday: "0",
34
+ sun: "0",
35
+ monday: "1",
36
+ mon: "1",
37
+ tuesday: "2",
38
+ tue: "2",
39
+ wednesday: "3",
40
+ wed: "3",
41
+ thursday: "4",
42
+ thu: "4",
43
+ friday: "5",
44
+ fri: "5",
45
+ saturday: "6",
46
+ sat: "6"
47
+ };
48
+ const everyMatch = s.match(/every\s+(\d+)\s*([mh])/);
49
+ if (everyMatch) {
50
+ const n = everyMatch[1];
51
+ const unit = everyMatch[2];
52
+ if (unit === "m") return `*/${n} * * * *`;
53
+ if (unit === "h") return `0 */${n} * * *`;
54
+ }
55
+ let hour = -1;
56
+ let minute = 0;
57
+ if (/midnight/.test(s)) {
58
+ hour = 0;
59
+ } else if (/noon/.test(s)) {
60
+ hour = 12;
61
+ } else {
62
+ const timeMatch = s.match(/(\d{1,2})(?::(\d{2}))?\s*(am|pm)?/);
63
+ if (timeMatch) {
64
+ hour = parseInt(timeMatch[1], 10);
65
+ minute = timeMatch[2] ? parseInt(timeMatch[2], 10) : 0;
66
+ if (timeMatch[3] === "pm" && hour < 12) hour += 12;
67
+ if (timeMatch[3] === "am" && hour === 12) hour = 0;
68
+ }
69
+ }
70
+ if (hour === -1) hour = 0;
71
+ let dow = "*";
72
+ if (/weekday|weekdays/.test(s)) {
73
+ dow = "1-5";
74
+ } else if (/weekend|weekends/.test(s)) {
75
+ dow = "0,6";
76
+ } else {
77
+ for (const [name, val] of Object.entries(dayMap)) {
78
+ if (s.includes(name)) {
79
+ dow = val;
80
+ break;
81
+ }
82
+ }
83
+ }
84
+ return `${minute} ${hour} * * ${dow}`;
85
+ }
86
+ async function createSchedule(input) {
87
+ if (!isValidCron(input.cron)) {
88
+ throw new Error(`Invalid cron expression: ${input.cron}. Must be 5 fields with only digits, *, /, -, comma.`);
89
+ }
90
+ await ensureDb();
91
+ const client = getClient();
92
+ const id = crypto.randomUUID().slice(0, 8);
93
+ const now = (/* @__PURE__ */ new Date()).toISOString();
94
+ const prompt = input.prompt ?? input.description;
95
+ await client.execute({
96
+ sql: `INSERT INTO schedules (id, cron, description, job_type, prompt, assigned_to, project_name, active, use_crontab, created_at)
97
+ VALUES (?, ?, ?, ?, ?, ?, ?, 1, ?, ?)`,
98
+ args: [
99
+ id,
100
+ input.cron,
101
+ input.description,
102
+ input.jobType ?? "report",
103
+ prompt,
104
+ input.assignedTo ?? null,
105
+ input.projectName ?? null,
106
+ input.useCrontab ? 1 : 0,
107
+ now
108
+ ]
109
+ });
110
+ if (input.useCrontab) {
111
+ addToCrontab(id, input.cron, prompt, input.projectName);
112
+ }
113
+ return {
114
+ id,
115
+ cron: input.cron,
116
+ description: input.description,
117
+ jobType: input.jobType ?? "report",
118
+ prompt,
119
+ assignedTo: input.assignedTo,
120
+ projectName: input.projectName,
121
+ active: true,
122
+ useCrontab: input.useCrontab ?? false,
123
+ createdAt: now
124
+ };
125
+ }
126
+ async function listSchedules(activeOnly = true) {
127
+ await ensureDb();
128
+ const client = getClient();
129
+ const sql = activeOnly ? "SELECT * FROM schedules WHERE active = 1 ORDER BY created_at ASC" : "SELECT * FROM schedules ORDER BY created_at ASC";
130
+ const result = await client.execute({ sql, args: [] });
131
+ return result.rows.map((row) => ({
132
+ id: String(row.id),
133
+ cron: String(row.cron),
134
+ description: String(row.description),
135
+ jobType: String(row.job_type),
136
+ prompt: row.prompt ? String(row.prompt) : void 0,
137
+ assignedTo: row.assigned_to ? String(row.assigned_to) : void 0,
138
+ projectName: row.project_name ? String(row.project_name) : void 0,
139
+ active: Number(row.active) === 1,
140
+ useCrontab: Number(row.use_crontab) === 1,
141
+ createdAt: String(row.created_at)
142
+ }));
143
+ }
144
+ async function deleteSchedule(id) {
145
+ await ensureDb();
146
+ const client = getClient();
147
+ const existing = await client.execute({
148
+ sql: "SELECT use_crontab FROM schedules WHERE id = ?",
149
+ args: [id]
150
+ });
151
+ if (existing.rows.length === 0) return false;
152
+ const usesCrontab = Number(existing.rows[0].use_crontab) === 1;
153
+ await client.execute({
154
+ sql: "DELETE FROM schedules WHERE id = ?",
155
+ args: [id]
156
+ });
157
+ if (usesCrontab) {
158
+ removeFromCrontab(id);
159
+ }
160
+ return true;
161
+ }
162
+ function addToCrontab(id, cron, prompt, projectDir) {
163
+ if (!isValidCron(cron)) {
164
+ throw new Error(`Invalid cron expression: ${cron}`);
165
+ }
166
+ if (!isValidScheduleId(id)) {
167
+ throw new Error(`Invalid schedule ID: ${id}`);
168
+ }
169
+ try {
170
+ const cwd = projectDir ? `cd ${JSON.stringify(projectDir)} && ` : "";
171
+ const escapedPrompt = prompt.replace(/"/g, '\\"');
172
+ const entry = `${cron} ${cwd}claude -p --dangerously-skip-permissions "${escapedPrompt}" # exe-schedule:${id}`;
173
+ execSync(
174
+ `(crontab -l 2>/dev/null; echo ${JSON.stringify(entry)}) | crontab -`,
175
+ { timeout: 5e3, stdio: "ignore" }
176
+ );
177
+ } catch (err) {
178
+ if (err instanceof Error && err.message.startsWith("Invalid")) throw err;
179
+ }
180
+ }
181
+ function removeFromCrontab(id) {
182
+ if (!isValidScheduleId(id)) return;
183
+ try {
184
+ execSync(
185
+ `crontab -l 2>/dev/null | grep -v "exe-schedule:${id}" | crontab -`,
186
+ { timeout: 5e3, stdio: "ignore" }
187
+ );
188
+ } catch {
189
+ }
190
+ }
191
+
192
+ export {
193
+ parseHumanCron,
194
+ createSchedule,
195
+ listSchedules,
196
+ deleteSchedule
197
+ };