@askexenow/exe-os 0.9.286 → 0.9.288
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/deploy/compose/.env.customer.example +10 -1
- package/deploy/compose/.env.example +8 -1
- package/deploy/compose/README.md +21 -0
- package/deploy/compose/docker-compose.yml +24 -1
- package/deploy/compose/generate-env.ts +16 -2
- package/deploy/compose/setup.sh +63 -0
- package/deploy/stack-manifests/v0.9.json +1 -1
- package/dist/active-agent-KB4346K6.js +26 -0
- package/dist/active-agent-LTYLGBJX.js +27 -0
- package/dist/active-agent-V4W557J4.js +26 -0
- package/dist/active-agent-XZVU5XBO.js +27 -0
- package/dist/agentic-ontology-4PNQX4YW.js +25 -0
- package/dist/agentic-ontology-WOTOTSR7.js +25 -0
- package/dist/backfill-metadata-O2BQIEVT.js +599 -0
- package/dist/backfill-metadata-PHXNTBNR.js +599 -0
- package/dist/background-jobs-RIECIYW5.js +25 -0
- package/dist/behaviors-MO5BWFPW.js +39 -0
- package/dist/behaviors-TMA2HRTL.js +39 -0
- package/dist/bin/age-ontology-load.js +2 -2
- package/dist/bin/agentic-ontology-backfill.js +7 -7
- package/dist/bin/agentic-reflection-backfill.js +8 -8
- package/dist/bin/agentic-semantic-label.js +7 -7
- package/dist/bin/backfill-conversations.js +7 -7
- package/dist/bin/backfill-responses.js +7 -7
- package/dist/bin/backfill-vectors.js +10 -10
- package/dist/bin/bulk-sync-postgres.js +8 -8
- package/dist/bin/cc-doctor.js +7 -6
- package/dist/bin/cleanup-stale-review-tasks.js +13 -13
- package/dist/bin/cli.js +20 -20
- package/dist/bin/exe-agent-config.js +4 -4
- package/dist/bin/exe-agent.js +9 -9
- package/dist/bin/exe-assign.js +9 -9
- package/dist/bin/exe-boot.js +32 -21
- package/dist/bin/exe-call.js +5 -5
- package/dist/bin/exe-cloud.js +7 -7
- package/dist/bin/exe-dispatch.js +13 -13
- package/dist/bin/exe-doctor.js +4 -1
- package/dist/bin/exe-export-behaviors.js +8 -8
- package/dist/bin/exe-forget.js +7 -7
- package/dist/bin/exe-gateway.js +8 -8
- package/dist/bin/exe-healthcheck.js +9 -6
- package/dist/bin/exe-heartbeat.js +13 -13
- package/dist/bin/exe-kill.js +16 -16
- package/dist/bin/exe-launch-agent.js +20 -20
- package/dist/bin/exe-new-employee.js +8 -8
- package/dist/bin/exe-pending-messages.js +14 -14
- package/dist/bin/exe-pending-notifications.js +13 -13
- package/dist/bin/exe-pending-reviews.js +13 -13
- package/dist/bin/exe-rename.js +5 -5
- package/dist/bin/exe-review.js +15 -15
- package/dist/bin/exe-search.js +6 -6
- package/dist/bin/exe-session-cleanup.js +18 -18
- package/dist/bin/exe-settings.js +7 -7
- package/dist/bin/exe-start-codex.js +12 -12
- package/dist/bin/exe-start-opencode.js +9 -9
- package/dist/bin/exe-status.js +14 -14
- package/dist/bin/exe-support.js +3 -3
- package/dist/bin/exe-team.js +4 -4
- package/dist/bin/git-sweep.js +14 -14
- package/dist/bin/graph-backfill.js +8 -8
- package/dist/bin/graph-export.js +6 -6
- package/dist/bin/import-history.js +10 -10
- package/dist/bin/install.js +7 -7
- package/dist/bin/intercom-check.js +4 -4
- package/dist/bin/mcp-sessions.js +2 -2
- package/dist/bin/orchestration-metrics.js +5 -5
- package/dist/bin/postgres-agentic-reflection-backfill.js +4 -4
- package/dist/bin/postgres-agentic-semantic-backfill.js +3 -3
- package/dist/bin/scan-tasks.js +13 -13
- package/dist/bin/setup.js +2 -2
- package/dist/bin/shard-migrate.js +6 -6
- package/dist/bin/stack-update.js +4 -4
- package/dist/bin/vps-health-gate.js +1 -1
- package/dist/branding-6PY5DNDC.js +97 -0
- package/dist/capability-cards-4NKVY6GD.js +88 -0
- package/dist/capability-cards-UD5TCSAZ.js +88 -0
- package/dist/capacity-monitor-AU7MVZED.js +50 -0
- package/dist/capacity-monitor-CV7PWPYR.js +50 -0
- package/dist/catchup-brief-AZND3EET.js +174 -0
- package/dist/catchup-brief-TFL6MYOZ.js +154 -0
- package/dist/chunk-22R3G3GD.js +297 -0
- package/dist/chunk-2CO47E5Q.js +210 -0
- package/dist/chunk-2F5UVNYP.js +167 -0
- package/dist/chunk-2IJCGYY2.js +30 -0
- package/dist/chunk-2V52EJI7.js +14365 -0
- package/dist/chunk-2WCXJVSN.js +456 -0
- package/dist/chunk-2X52UGP7.js +190 -0
- package/dist/chunk-3H5FNTXS.js +85 -0
- package/dist/chunk-3OC6MJIX.js +210 -0
- package/dist/chunk-3OSDE4Y2.js +1119 -0
- package/dist/chunk-43QFBQHT.js +1094 -0
- package/dist/chunk-4EBW3URQ.js +4318 -0
- package/dist/chunk-4KVMYS3U.js +280 -0
- package/dist/chunk-4UTH3H4H.js +128 -0
- package/dist/chunk-5BKYHLA3.js +411 -0
- package/dist/chunk-5RRYMKXN.js +150 -0
- package/dist/chunk-5TYYOPTD.js +50 -0
- package/dist/chunk-5XOAYMSU.js +4318 -0
- package/dist/chunk-6DQNLI55.js +240 -0
- package/dist/chunk-6GOT6FNJ.js +1352 -0
- package/dist/chunk-6KH24PJ7.js +221 -0
- package/dist/chunk-6TV66RAI.js +192 -0
- package/dist/chunk-6UD5CM2H.js +244 -0
- package/dist/chunk-6YR3J6WG.js +203 -0
- package/dist/chunk-77HA7ATH.js +89 -0
- package/dist/chunk-7G52QPUX.js +271 -0
- package/dist/chunk-7M5ZHD4E.js +214 -0
- package/dist/chunk-A3GC762G.js +333 -0
- package/dist/chunk-ALTFDYI6.js +97 -0
- package/dist/chunk-AN25OIRX.js +38 -0
- package/dist/chunk-AOWC2J6S.js +14225 -0
- package/dist/chunk-ASL5PHCT.js +1934 -0
- package/dist/chunk-AVTDW4L6.js +562 -0
- package/dist/chunk-BEQTE7GH.js +668 -0
- package/dist/chunk-BJTBA4XS.js +290 -0
- package/dist/chunk-BQBPNRUW.js +231 -0
- package/dist/chunk-BQD4XYWX.js +159 -0
- package/dist/chunk-BVFCZNJH.js +41 -0
- package/dist/chunk-BW7HTZMW.js +240 -0
- package/dist/chunk-BZP3ZSRI.js +97 -0
- package/dist/chunk-C2NCSKRZ.js +373 -0
- package/dist/chunk-C3AH5JAS.js +510 -0
- package/dist/chunk-CEKZ77KA.js +471 -0
- package/dist/chunk-CNTBRVYW.js +54 -0
- package/dist/chunk-CTJQX44O.js +271 -0
- package/dist/chunk-CVG2PBCO.js +1105 -0
- package/dist/chunk-CZ47BFAT.js +299 -0
- package/dist/chunk-CZHC3HIO.js +362 -0
- package/dist/chunk-DIZOFRZO.js +284 -0
- package/dist/chunk-DRX46W5O.js +1068 -0
- package/dist/chunk-DYICAAAM.js +630 -0
- package/dist/chunk-E4BXNYLD.js +128 -0
- package/dist/chunk-EKXOBH7F.js +227 -0
- package/dist/chunk-ELQZYKVN.js +394 -0
- package/dist/chunk-EVM22AGQ.js +382 -0
- package/dist/chunk-EY7TBDZ5.js +38 -0
- package/dist/chunk-FDOEMBUB.js +345 -0
- package/dist/chunk-FG2OXKU5.js +539 -0
- package/dist/chunk-FP6TDXIL.js +456 -0
- package/dist/chunk-FVWATWI2.js +377 -0
- package/dist/chunk-G2OW4LUS.js +55 -0
- package/dist/chunk-GAYD7KBH.js +836 -0
- package/dist/chunk-GETGWHCO.js +227 -0
- package/dist/chunk-GGLUKNFC.js +1350 -0
- package/dist/chunk-GLEE2MUJ.js +262 -0
- package/dist/chunk-GUODMEAR.js +42 -0
- package/dist/chunk-H6LA4UV5.js +735 -0
- package/dist/chunk-H6PW4LNY.js +129 -0
- package/dist/chunk-HEMJ5RDU.js +345 -0
- package/dist/chunk-HMP6KB4V.js +382 -0
- package/dist/chunk-I35KAK3E.js +458 -0
- package/dist/chunk-I4DA5D5Q.js +1922 -0
- package/dist/chunk-I53O26HI.js +1119 -0
- package/dist/chunk-IBLEQQDQ.js +181 -0
- package/dist/chunk-IS2BEXUC.js +2113 -0
- package/dist/chunk-ISCEGQHA.js +244 -0
- package/dist/chunk-JCERHH3L.js +85 -0
- package/dist/chunk-JCNCPBWU.js +97 -0
- package/dist/chunk-JEMLH7H2.js +1350 -0
- package/dist/chunk-JMR2X5FJ.js +630 -0
- package/dist/chunk-JR3MYBWG.js +2078 -0
- package/dist/chunk-JSEICUAL.js +197 -0
- package/dist/chunk-K3KOLZTH.js +123 -0
- package/dist/chunk-K4OH3B2I.js +14248 -0
- package/dist/chunk-KQ4Z2G24.js +260 -0
- package/dist/chunk-KVH5DSTA.js +369 -0
- package/dist/chunk-L4R5NOVT.js +81 -0
- package/dist/chunk-LCB6YCE6.js +70 -0
- package/dist/chunk-M543SLJZ.js +1352 -0
- package/dist/chunk-M6KDMU5P.js +2113 -0
- package/dist/chunk-MDAGRPZG.js +197 -0
- package/dist/chunk-MLNOSXRZ.js +402 -0
- package/dist/chunk-MNI5K34D.js +290 -0
- package/dist/chunk-MUDH7ZLR.js +129 -0
- package/dist/chunk-MUMBWV57.js +379 -0
- package/dist/chunk-NAQLGLOG.js +176 -0
- package/dist/chunk-NIBCUAEI.js +123 -0
- package/dist/chunk-NIZCTXYW.js +89 -0
- package/dist/chunk-NS5G36TN.js +712 -0
- package/dist/chunk-NSKQWLQL.js +76 -0
- package/dist/chunk-NUA3WRTO.js +1157 -0
- package/dist/chunk-O3IESEF6.js +85 -0
- package/dist/chunk-OAO2BK25.js +81 -0
- package/dist/chunk-OFY7RGLL.js +348 -0
- package/dist/chunk-OICU3UHF.js +181 -0
- package/dist/chunk-OK6HCC2Z.js +731 -0
- package/dist/chunk-ONTUBOYP.js +230 -0
- package/dist/chunk-OQB7TERT.js +54 -0
- package/dist/chunk-P362EYLO.js +106 -0
- package/dist/chunk-P4T4JOCE.js +192 -0
- package/dist/chunk-QAOWXQKA.js +333 -0
- package/dist/chunk-QEFT6G6J.js +2078 -0
- package/dist/chunk-QFVY5MHR.js +402 -0
- package/dist/chunk-QLMSROQC.js +604 -0
- package/dist/chunk-QNR4ZQDU.js +14365 -0
- package/dist/chunk-QQ3T45AO.js +348 -0
- package/dist/chunk-QV6HNEMH.js +230 -0
- package/dist/chunk-R5LCP56X.js +299 -0
- package/dist/chunk-RHTZA6X6.js +1922 -0
- package/dist/chunk-RL4RQIEI.js +731 -0
- package/dist/chunk-RW4EMKQF.js +185 -0
- package/dist/chunk-S35K4C6F.js +127 -0
- package/dist/chunk-SFQYQH5Z.js +50 -0
- package/dist/chunk-SLFB2RY6.js +133 -0
- package/dist/chunk-SSC7EKY2.js +133 -0
- package/dist/chunk-SWFFGWAF.js +262 -0
- package/dist/chunk-SXZCFZ5T.js +157 -0
- package/dist/chunk-T5LT62O5.js +14365 -0
- package/dist/chunk-TDGPZF5T.js +539 -0
- package/dist/chunk-TF6DEHEL.js +167 -0
- package/dist/chunk-TH7ZNVGB.js +33 -0
- package/dist/chunk-TTNYB6OH.js +33 -0
- package/dist/chunk-TUHR7Z7R.js +221 -0
- package/dist/chunk-UHKXYDBM.js +668 -0
- package/dist/chunk-UQBD2L52.js +128 -0
- package/dist/chunk-UTARH2WY.js +76 -0
- package/dist/chunk-UXYAQCIB.js +30 -0
- package/dist/chunk-UZHCPLB2.js +127 -0
- package/dist/chunk-VNJKA3K5.js +171 -0
- package/dist/chunk-VP4MDSUA.js +1156 -0
- package/dist/chunk-VWA7OCXI.js +58 -0
- package/dist/chunk-VWZZF3YB.js +70 -0
- package/dist/chunk-VYPKS352.js +214 -0
- package/dist/chunk-W6JB72KR.js +448 -0
- package/dist/chunk-WG4PSSI5.js +510 -0
- package/dist/chunk-WJPZPASS.js +373 -0
- package/dist/chunk-WLDBPKRL.js +3246 -0
- package/dist/chunk-WO2SPDUE.js +231 -0
- package/dist/chunk-WXEPPHDU.js +1094 -0
- package/dist/chunk-WXNSOT6K.js +82 -0
- package/dist/chunk-X2PSVVHB.js +448 -0
- package/dist/chunk-X5Y4JSD4.js +128 -0
- package/dist/chunk-X7VPB6Z5.js +621 -0
- package/dist/chunk-XA2GMH53.js +85 -0
- package/dist/chunk-XC2ZMYDN.js +204 -0
- package/dist/chunk-XFQJH67J.js +157 -0
- package/dist/chunk-XLF5F55U.js +230 -0
- package/dist/chunk-XQ5LT6UM.js +362 -0
- package/dist/chunk-XRMBKPIP.js +204 -0
- package/dist/chunk-XUC7YYW5.js +171 -0
- package/dist/chunk-XVDWHYIK.js +3276 -0
- package/dist/chunk-Y6CVAZ7T.js +97 -0
- package/dist/chunk-YA7WVZF5.js +377 -0
- package/dist/chunk-YACAOZVG.js +185 -0
- package/dist/chunk-YEHV3FN5.js +836 -0
- package/dist/chunk-YMZ6IGC3.js +284 -0
- package/dist/chunk-Z5P7AUFM.js +280 -0
- package/dist/chunk-Z5YFGOLZ.js +1068 -0
- package/dist/chunk-ZLYKY6GP.js +150 -0
- package/dist/chunk-ZOP7NAM2.js +58 -0
- package/dist/chunk-ZUNYZYKV.js +1157 -0
- package/dist/co-activation-54LO5GAA.js +73 -0
- package/dist/co-activation-ESPLEOMH.js +73 -0
- package/dist/co-occurrence-52ZTRX2I.js +94 -0
- package/dist/co-occurrence-O4G6VQUU.js +94 -0
- package/dist/code-context-index-OX3LN34G.js +30 -0
- package/dist/conversation-wiki-populator-IWSKIHQK.js +105 -0
- package/dist/core-memory-J6AEOYFN.js +110 -0
- package/dist/core-memory-XF4CLLDR.js +110 -0
- package/dist/cpu-steal-5YKTUPWF.js +23 -0
- package/dist/crdt-sync-5JOW6T2I.js +33 -0
- package/dist/crdt-sync-6L4OI4OA.js +33 -0
- package/dist/crm-webhook-BAXEYDPD.js +10 -0
- package/dist/crm-webhook-KVBGOUHU.js +10 -0
- package/dist/cto-delegation-gate-3WUTWBCT.js +279 -0
- package/dist/cto-delegation-gate-DZF3Z66I.js +279 -0
- package/dist/daemon-auth-M2G5PBME.js +13 -0
- package/dist/daemon-orchestration-3ANZVE7Y.js +138 -0
- package/dist/daemon-orchestration-APVUUNFB.js +138 -0
- package/dist/daemon-restart-report-F3P5AQS4.js +9 -0
- package/dist/db-backup-J7253HNV.js +33 -0
- package/dist/db-backup-M733ORYE.js +33 -0
- package/dist/db-restore-events-AIL6GSEK.js +76 -0
- package/dist/db-restore-events-LB3O46HH.js +76 -0
- package/dist/doc-graph-extractor-CF2WBSGL.js +132 -0
- package/dist/doc-graph-extractor-RSYHLOKD.js +132 -0
- package/dist/dreaming-L4FLJJO2.js +33 -0
- package/dist/dreaming-YKEKUE55.js +33 -0
- package/dist/entity-boost-ZKACOJKH.js +375 -0
- package/dist/exe-drift-AVJRXHCB.js +69 -0
- package/dist/exe-drift-MYT5NQ52.js +69 -0
- package/dist/exe-export-T2FHC5GC.js +75 -0
- package/dist/exe-export-TBV2JI4P.js +75 -0
- package/dist/exe-import-6UAKJF3Y.js +78 -0
- package/dist/exe-import-KJQO4PJX.js +78 -0
- package/dist/exe-key-KGQUC27D.js +580 -0
- package/dist/exe-key-Q772PMBM.js +580 -0
- package/dist/exe-org-KK22SSB3.js +73 -0
- package/dist/exe-snapshot-KF3HPOWJ.js +337 -0
- package/dist/exe-snapshot-SK6PMOFA.js +337 -0
- package/dist/fast-db-init-EXNYB2HK.js +7 -0
- package/dist/fast-db-init-FMECTLNS.js +7 -0
- package/dist/founder-context-N2VZ5MY2.js +96 -0
- package/dist/gateway/index.js +12 -12
- package/dist/git-staleness-2EOHIOMZ.js +111 -0
- package/dist/git-staleness-ZEYJXPFP.js +111 -0
- package/dist/git-task-sweep-I74JXGNT.js +41 -0
- package/dist/git-task-sweep-YKLOPYYV.js +41 -0
- package/dist/global-procedures-B23F4IQ3.js +21 -0
- package/dist/global-procedures-LUBLZ6WN.js +21 -0
- package/dist/graph-auto-extract-CSLZXRDY.js +182 -0
- package/dist/graph-auto-extract-DZRIETGW.js +182 -0
- package/dist/hook-integrity-HJBXTLQD.js +89 -0
- package/dist/hooks/bug-report-worker.js +15 -15
- package/dist/hooks/codex-stop-task-finalizer.js +15 -15
- package/dist/hooks/commit-complete.js +16 -16
- package/dist/hooks/error-recall.js +8 -8
- package/dist/hooks/exe-heartbeat-hook.js +5 -5
- package/dist/hooks/ingest-worker.js +5 -5
- package/dist/hooks/ingest.js +12 -12
- package/dist/hooks/instructions-loaded.js +6 -6
- package/dist/hooks/manifest.json +20 -20
- package/dist/hooks/notification.js +6 -6
- package/dist/hooks/post-compact.js +15 -15
- package/dist/hooks/post-tool-combined.js +7 -7
- package/dist/hooks/pre-compact.js +20 -20
- package/dist/hooks/pre-tool-use.js +19 -19
- package/dist/hooks/prompt-submit.js +27 -27
- package/dist/hooks/session-end.js +25 -25
- package/dist/hooks/session-start.js +37 -13
- package/dist/hooks/stop.js +22 -22
- package/dist/hooks/subagent-stop.js +15 -15
- package/dist/hooks/summary-worker.js +22 -22
- package/dist/index.js +23 -23
- package/dist/installer-AU5VMP4U.js +39 -0
- package/dist/installer-H7BIYSIK.js +39 -0
- package/dist/installer-L4NJWZGV.js +297 -0
- package/dist/installer-QE3LBMVF.js +343 -0
- package/dist/installer-QH75OOGL.js +297 -0
- package/dist/installer-XREYYGF2.js +343 -0
- package/dist/key-backup-status-IJCQT4NV.js +39 -0
- package/dist/lib/agent-config.js +2 -2
- package/dist/lib/cloud-sync.js +6 -6
- package/dist/lib/config.js +7 -3
- package/dist/lib/consolidation.js +8 -8
- package/dist/lib/database.js +3 -3
- package/dist/lib/db-daemon-client.js +3 -3
- package/dist/lib/db.js +3 -3
- package/dist/lib/device-registry.js +1 -1
- package/dist/lib/embed-worker.js +1 -1
- package/dist/lib/embedder.js +4 -4
- package/dist/lib/employee-templates.js +5 -5
- package/dist/lib/employees.js +3 -3
- package/dist/lib/exe-daemon-client.js +3 -3
- package/dist/lib/exe-daemon.js +51 -51
- package/dist/lib/hybrid-search.js +6 -6
- package/dist/lib/identity.js +3 -3
- package/dist/lib/license.js +2 -2
- package/dist/lib/messaging.js +13 -13
- package/dist/lib/reminders.js +4 -4
- package/dist/lib/schedules.js +6 -6
- package/dist/lib/session-registry.js +5 -5
- package/dist/lib/skill-learning.js +7 -7
- package/dist/lib/status-brief.js +1 -1
- package/dist/lib/store.js +5 -5
- package/dist/lib/task-router.js +4 -4
- package/dist/lib/tasks.js +14 -14
- package/dist/lib/tmux-routing.js +12 -12
- package/dist/lib/token-spend.js +4 -4
- package/dist/license-gate-X57APN2E.js +14 -0
- package/dist/mcp/register-tools.js +71 -69
- package/dist/mcp/server.js +73 -71
- package/dist/mcp/tools/complete-reminder.js +5 -5
- package/dist/mcp/tools/create-reminder.js +5 -5
- package/dist/mcp/tools/create-task.js +16 -16
- package/dist/mcp/tools/deactivate-behavior.js +8 -8
- package/dist/mcp/tools/list-reminders.js +5 -5
- package/dist/mcp/tools/list-tasks.js +16 -16
- package/dist/mcp/tools/send-message.js +15 -15
- package/dist/mcp/tools/update-task.js +15 -15
- package/dist/mcp-http-config-KMP55RXI.js +28 -0
- package/dist/mcp-http-config-XYHPQVRA.js +28 -0
- package/dist/memory-cards-AULW7ITU.js +179 -0
- package/dist/memory-cards-NLTRIBSD.js +179 -0
- package/dist/memory-graph-extractor-NGJQIKAA.js +21 -0
- package/dist/memory-graph-extractor-QWFPKSX5.js +21 -0
- package/dist/memory-poisoning-defense-5WWOLHUQ.js +223 -0
- package/dist/memory-poisoning-defense-742X4H52.js +223 -0
- package/dist/memory-queue-OXHHNDOC.js +19 -0
- package/dist/memory-queue-client-P4TZNVQB.js +16 -0
- package/dist/memory-reflection-6GF5OK2W.js +243 -0
- package/dist/memory-reflection-SGIE7OJ7.js +243 -0
- package/dist/message-queue-client-ELWCWKBO.js +92 -0
- package/dist/notifications-67M5SPLO.js +46 -0
- package/dist/notifications-HUH2H5WZ.js +46 -0
- package/dist/oauth-server-RKCMPBIP.js +437 -0
- package/dist/orchestration-events-MC3IZ46I.js +26 -0
- package/dist/orchestration-events-NZCUTMCJ.js +26 -0
- package/dist/orchestration-phase-FPAVQ57X.js +23 -0
- package/dist/orchestrator-3MJ6YJKT.js +34 -0
- package/dist/orchestrator-HOXFT4PP.js +34 -0
- package/dist/pipeline-router-NCUDK335.js +14 -0
- package/dist/pipeline-router-POLBFEBM.js +14 -0
- package/dist/plan-limits-DGAFE7V6.js +27 -0
- package/dist/plan-limits-J4WXJVDS.js +27 -0
- package/dist/project-boot-FRCQHQTF.js +299 -0
- package/dist/project-boot-QHXOUJGT.js +299 -0
- package/dist/projection-worker-QZW4DOY3.js +1034 -0
- package/dist/projection-worker-XDKC3GIZ.js +1034 -0
- package/dist/prospective-memory-JIGVZM2I.js +231 -0
- package/dist/push-notifications-ERC4SQSQ.js +15 -0
- package/dist/reranker-SF3LMYGY.js +19 -0
- package/dist/reranker-SOM2WMD6.js +19 -0
- package/dist/retrieval-health-7FRD2KLM.js +11 -0
- package/dist/retrieval-health-P6QHP263.js +11 -0
- package/dist/review-polling-66EJAMR3.js +125 -0
- package/dist/review-polling-SYKYNFRK.js +125 -0
- package/dist/runtime/index.js +19 -19
- package/dist/session-events-A6EDTM27.js +37 -0
- package/dist/session-events-DONPZ7DR.js +37 -0
- package/dist/session-kill-telemetry-NV2S2YZ2.js +30 -0
- package/dist/session-kill-telemetry-Y47QAFHP.js +30 -0
- package/dist/session-scope-H3FJVQ6E.js +87 -0
- package/dist/session-scope-I56DVQEY.js +87 -0
- package/dist/setup-wizard-MX3S3DVH.js +12 -0
- package/dist/setup-wizard-YRBCVPCU.js +12 -0
- package/dist/shard-manager-AX4W5B3N.js +30 -0
- package/dist/skill-refinement-57DCIDKT.js +158 -0
- package/dist/skill-refinement-XTBCHOMO.js +158 -0
- package/dist/stack-update-45SNTDZ6.js +76 -0
- package/dist/stack-update-CZ2QHS6T.js +76 -0
- package/dist/stack-update-XNEO4WQH.js +76 -0
- package/dist/steward-gate-J7VMPHEN.js +14 -0
- package/dist/steward-gate-PVUKWYED.js +14 -0
- package/dist/task-enforcement-G4MYUJ2E.js +439 -0
- package/dist/task-enforcement-QBF44BXN.js +439 -0
- package/dist/task-scope-D2VOYU3P.js +36 -0
- package/dist/task-scope-HGL6VACL.js +36 -0
- package/dist/tasks-crud-DFE4XOW7.js +78 -0
- package/dist/tasks-crud-J5TSLFQN.js +78 -0
- package/dist/tasks-notify-2VRKOK74.js +39 -0
- package/dist/tasks-notify-Q23MIKDA.js +39 -0
- package/dist/tasks-review-NRSVHN7V.js +48 -0
- package/dist/tasks-review-VV3RLPLT.js +48 -0
- package/dist/telemetry-upload-ISI6TEX4.js +740 -0
- package/dist/telemetry-upload-XML6WVPW.js +740 -0
- package/dist/token-budget-KVRHMY7V.js +85 -0
- package/dist/token-budget-WV2GIFYJ.js +85 -0
- package/dist/tool-capability-index-NOMFY27N.js +10 -0
- package/dist/tool-telemetry-6DGIQUV3.js +17 -0
- package/dist/tool-telemetry-NXPM2T5E.js +17 -0
- package/dist/tui/App.js +80 -38
- package/dist/tui-data-C4474APN.js +259 -0
- package/dist/tui-data-KP66NOW3.js +259 -0
- package/dist/webhook-pipe-ZAEPBTDB.js +114 -0
- package/dist/wiki-acl-4A3ZUPJK.js +111 -0
- package/dist/wiki-acl-DRTEVIHT.js +111 -0
- package/dist/wiki-client-AN52BINX.js +157 -0
- package/dist/worker-gate-EVSFKALM.js +21 -0
- package/dist/worker-gate-O2KXQN5T.js +21 -0
- package/dist/workflow-engine-MUO435MX.js +28 -0
- package/dist/workflow-engine-RV6JLGHQ.js +28 -0
- package/dist/worktree-36S7LYES.js +27 -0
- package/dist/worktree-WTNPY42O.js +27 -0
- package/dist/worktree-sweep-M5AXCSU7.js +20 -0
- package/dist/worktree-sweep-NKEGKQT3.js +20 -0
- package/package.json +1 -1
- package/release-notes.json +20 -20
|
@@ -0,0 +1,379 @@
|
|
|
1
|
+
import {
|
|
2
|
+
EXE_AI_DIR
|
|
3
|
+
} from "./chunk-CEKZ77KA.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_KEY_PEM_2026_06_17 = `-----BEGIN PUBLIC KEY-----
|
|
32
|
+
MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEYdtXo668/3uHX+8DY9MOgm0HzVrP
|
|
33
|
+
oR8NJBuiITKaerulCXiSiJMX1HP0+h5dHBVfCWQfIVk5BMdsheAMaP806w==
|
|
34
|
+
-----END PUBLIC KEY-----`;
|
|
35
|
+
var LICENSE_PUBLIC_KEYS = [LICENSE_PUBLIC_KEY_PEM_2026_06_17, LICENSE_PUBLIC_KEY_PEM, LICENSE_PUBLIC_KEY_PEM_PREV];
|
|
36
|
+
var LICENSE_JWT_ALG = "ES256";
|
|
37
|
+
var PLAN_LIMITS = {
|
|
38
|
+
free: { devices: 1, employees: 1, memories: 5e3 },
|
|
39
|
+
pro: { devices: 3, employees: 5, memories: 1e5 },
|
|
40
|
+
team: { devices: 10, employees: 20, memories: 1e6 },
|
|
41
|
+
agency: { devices: 50, employees: 100, memories: 1e7 },
|
|
42
|
+
enterprise: { devices: -1, employees: -1, memories: -1 }
|
|
43
|
+
};
|
|
44
|
+
var FREE_LICENSE = {
|
|
45
|
+
valid: true,
|
|
46
|
+
plan: "free",
|
|
47
|
+
email: "",
|
|
48
|
+
expiresAt: null,
|
|
49
|
+
deviceLimit: 1,
|
|
50
|
+
employeeLimit: 1,
|
|
51
|
+
memoryLimit: 5e3
|
|
52
|
+
};
|
|
53
|
+
function loadDeviceId() {
|
|
54
|
+
const deviceJsonPath = path.join(EXE_AI_DIR, "device.json");
|
|
55
|
+
try {
|
|
56
|
+
if (existsSync(deviceJsonPath)) {
|
|
57
|
+
const data = JSON.parse(readFileSync(deviceJsonPath, "utf8"));
|
|
58
|
+
if (data.deviceId) return data.deviceId;
|
|
59
|
+
}
|
|
60
|
+
} catch {
|
|
61
|
+
}
|
|
62
|
+
try {
|
|
63
|
+
if (existsSync(DEVICE_ID_PATH)) {
|
|
64
|
+
const id2 = readFileSync(DEVICE_ID_PATH, "utf8").trim();
|
|
65
|
+
if (id2) return id2;
|
|
66
|
+
}
|
|
67
|
+
} catch {
|
|
68
|
+
}
|
|
69
|
+
const id = randomUUID();
|
|
70
|
+
mkdirSync(EXE_AI_DIR, { recursive: true });
|
|
71
|
+
writeFileSync(DEVICE_ID_PATH, id, "utf8");
|
|
72
|
+
return id;
|
|
73
|
+
}
|
|
74
|
+
function loadLicense() {
|
|
75
|
+
try {
|
|
76
|
+
if (!existsSync(LICENSE_PATH)) return null;
|
|
77
|
+
return readFileSync(LICENSE_PATH, "utf8").trim();
|
|
78
|
+
} catch {
|
|
79
|
+
return null;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
function saveLicense(apiKey) {
|
|
83
|
+
mkdirSync(EXE_AI_DIR, { recursive: true });
|
|
84
|
+
writeFileSync(LICENSE_PATH, apiKey.trim(), { encoding: "utf8", mode: 384 });
|
|
85
|
+
}
|
|
86
|
+
async function verifyLicenseJwt(token) {
|
|
87
|
+
for (const pem of LICENSE_PUBLIC_KEYS) {
|
|
88
|
+
try {
|
|
89
|
+
const key = await importSPKI(pem, LICENSE_JWT_ALG);
|
|
90
|
+
const { payload } = await jwtVerify(token, key, {
|
|
91
|
+
algorithms: [LICENSE_JWT_ALG]
|
|
92
|
+
});
|
|
93
|
+
const plan = payload.plan ?? "free";
|
|
94
|
+
const email = payload.sub ?? "";
|
|
95
|
+
const limits = PLAN_LIMITS[plan] ?? PLAN_LIMITS.free;
|
|
96
|
+
return {
|
|
97
|
+
valid: true,
|
|
98
|
+
plan,
|
|
99
|
+
email,
|
|
100
|
+
expiresAt: payload.exp ? new Date(payload.exp * 1e3).toISOString() : null,
|
|
101
|
+
deviceLimit: limits.devices,
|
|
102
|
+
employeeLimit: limits.employees,
|
|
103
|
+
memoryLimit: limits.memories
|
|
104
|
+
};
|
|
105
|
+
} catch {
|
|
106
|
+
continue;
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
return null;
|
|
110
|
+
}
|
|
111
|
+
async function getCachedLicense() {
|
|
112
|
+
try {
|
|
113
|
+
if (!existsSync(CACHE_PATH)) return null;
|
|
114
|
+
const raw = JSON.parse(readFileSync(CACHE_PATH, "utf8"));
|
|
115
|
+
if (!raw.token || typeof raw.token !== "string") return null;
|
|
116
|
+
return await verifyLicenseJwt(raw.token);
|
|
117
|
+
} catch {
|
|
118
|
+
return null;
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
async function getGraceVerifiedLicense(graceDays = 7) {
|
|
122
|
+
try {
|
|
123
|
+
const token = readCachedLicenseToken();
|
|
124
|
+
if (!token) return null;
|
|
125
|
+
const payloadB64 = token.split(".")[1];
|
|
126
|
+
if (!payloadB64) return null;
|
|
127
|
+
const payload = JSON.parse(Buffer.from(payloadB64, "base64url").toString());
|
|
128
|
+
const expMs = (payload.exp ?? 0) * 1e3;
|
|
129
|
+
if (Date.now() >= expMs + graceDays * 24 * 60 * 60 * 1e3) return null;
|
|
130
|
+
let verified = null;
|
|
131
|
+
for (const pem of LICENSE_PUBLIC_KEYS) {
|
|
132
|
+
try {
|
|
133
|
+
const key = await importSPKI(pem, LICENSE_JWT_ALG);
|
|
134
|
+
const result = await jwtVerify(token, key, {
|
|
135
|
+
algorithms: [LICENSE_JWT_ALG],
|
|
136
|
+
clockTolerance: graceDays * 24 * 60 * 60
|
|
137
|
+
});
|
|
138
|
+
verified = result.payload;
|
|
139
|
+
break;
|
|
140
|
+
} catch {
|
|
141
|
+
continue;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
if (!verified) return null;
|
|
145
|
+
const plan = verified.plan ?? "free";
|
|
146
|
+
const limits = PLAN_LIMITS[plan] ?? PLAN_LIMITS.free;
|
|
147
|
+
process.stderr.write(
|
|
148
|
+
`[license] Offline grace: verified cached plan "${plan}" (expired, within ${graceDays}d grace).
|
|
149
|
+
`
|
|
150
|
+
);
|
|
151
|
+
return {
|
|
152
|
+
valid: true,
|
|
153
|
+
plan,
|
|
154
|
+
email: verified.sub ?? "",
|
|
155
|
+
expiresAt: verified.exp ? new Date(verified.exp * 1e3).toISOString() : null,
|
|
156
|
+
deviceLimit: limits.devices,
|
|
157
|
+
employeeLimit: limits.employees,
|
|
158
|
+
memoryLimit: limits.memories
|
|
159
|
+
};
|
|
160
|
+
} catch {
|
|
161
|
+
return null;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
function readCachedLicenseToken() {
|
|
165
|
+
try {
|
|
166
|
+
if (!existsSync(CACHE_PATH)) return null;
|
|
167
|
+
const raw = JSON.parse(readFileSync(CACHE_PATH, "utf8"));
|
|
168
|
+
return typeof raw.token === "string" ? raw.token : null;
|
|
169
|
+
} catch {
|
|
170
|
+
return null;
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
function cacheResponse(token) {
|
|
174
|
+
try {
|
|
175
|
+
writeFileSync(CACHE_PATH, JSON.stringify({ token }), "utf8");
|
|
176
|
+
} catch {
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
async function validateViaCFWorker(apiKey, deviceId) {
|
|
180
|
+
try {
|
|
181
|
+
const res = await fetchRetry(`${API_BASE}/auth/activate`, {
|
|
182
|
+
method: "POST",
|
|
183
|
+
headers: { "Content-Type": "application/json" },
|
|
184
|
+
body: JSON.stringify({ apiKey, deviceId }),
|
|
185
|
+
signal: AbortSignal.timeout(1e4)
|
|
186
|
+
});
|
|
187
|
+
if (!res.ok) return null;
|
|
188
|
+
const data = await res.json();
|
|
189
|
+
if (data.error === "device_limit_exceeded") return null;
|
|
190
|
+
if (!data.valid) return null;
|
|
191
|
+
if (data.token) {
|
|
192
|
+
cacheResponse(data.token);
|
|
193
|
+
const verified = await verifyLicenseJwt(data.token);
|
|
194
|
+
if (verified) return verified;
|
|
195
|
+
}
|
|
196
|
+
const limits = PLAN_LIMITS[data.plan] ?? PLAN_LIMITS.free;
|
|
197
|
+
return {
|
|
198
|
+
valid: data.valid,
|
|
199
|
+
plan: data.plan,
|
|
200
|
+
email: data.email,
|
|
201
|
+
expiresAt: data.expiresAt,
|
|
202
|
+
deviceLimit: limits.devices,
|
|
203
|
+
employeeLimit: limits.employees,
|
|
204
|
+
memoryLimit: limits.memories
|
|
205
|
+
};
|
|
206
|
+
} catch {
|
|
207
|
+
return null;
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
async function validateLicense(apiKey, deviceId) {
|
|
211
|
+
const did = deviceId ?? loadDeviceId();
|
|
212
|
+
const cfResult = await validateViaCFWorker(apiKey, did);
|
|
213
|
+
if (cfResult) return cfResult;
|
|
214
|
+
const cached = await getCachedLicense();
|
|
215
|
+
if (cached) return cached;
|
|
216
|
+
const graced = await getGraceVerifiedLicense();
|
|
217
|
+
if (graced) return graced;
|
|
218
|
+
return { ...FREE_LICENSE, valid: false };
|
|
219
|
+
}
|
|
220
|
+
var CACHE_MAX_AGE_MS = 36e5;
|
|
221
|
+
function getCacheAgeMs() {
|
|
222
|
+
try {
|
|
223
|
+
const s = statSync(CACHE_PATH);
|
|
224
|
+
return Date.now() - s.mtimeMs;
|
|
225
|
+
} catch {
|
|
226
|
+
return Infinity;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
async function checkLicense() {
|
|
230
|
+
let key = loadLicense();
|
|
231
|
+
if (!key) {
|
|
232
|
+
try {
|
|
233
|
+
const configPath = path.join(EXE_AI_DIR, "config.json");
|
|
234
|
+
if (existsSync(configPath)) {
|
|
235
|
+
const raw = JSON.parse(readFileSync(configPath, "utf8"));
|
|
236
|
+
const cloud = raw.cloud;
|
|
237
|
+
if (cloud?.apiKey) {
|
|
238
|
+
key = cloud.apiKey;
|
|
239
|
+
saveLicense(key);
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
} catch {
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
if (!key) return FREE_LICENSE;
|
|
246
|
+
const cached = await getCachedLicense();
|
|
247
|
+
if (cached && getCacheAgeMs() < CACHE_MAX_AGE_MS) return cached;
|
|
248
|
+
const deviceId = loadDeviceId();
|
|
249
|
+
return validateLicense(key, deviceId);
|
|
250
|
+
}
|
|
251
|
+
function isFeatureAllowed(license, feature) {
|
|
252
|
+
switch (feature) {
|
|
253
|
+
case "cloud_sync":
|
|
254
|
+
case "external_agents":
|
|
255
|
+
case "wiki":
|
|
256
|
+
return license.plan !== "free";
|
|
257
|
+
case "unlimited_employees":
|
|
258
|
+
return license.plan === "team" || license.plan === "agency" || license.plan === "enterprise";
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
function mirrorLicenseKey(apiKey) {
|
|
262
|
+
const trimmed = apiKey.trim();
|
|
263
|
+
if (!trimmed) return;
|
|
264
|
+
saveLicense(trimmed);
|
|
265
|
+
}
|
|
266
|
+
async function assertVpsLicense(opts) {
|
|
267
|
+
const env = opts?.env ?? process.env;
|
|
268
|
+
const inProduction = env.NODE_ENV === "production";
|
|
269
|
+
if (!opts?.force && !inProduction) {
|
|
270
|
+
return { ...FREE_LICENSE, plan: "free" };
|
|
271
|
+
}
|
|
272
|
+
const envKey = env.EXE_LICENSE_KEY?.trim();
|
|
273
|
+
if (envKey) {
|
|
274
|
+
saveLicense(envKey);
|
|
275
|
+
}
|
|
276
|
+
const apiKey = envKey ?? loadLicense();
|
|
277
|
+
if (!apiKey) {
|
|
278
|
+
throw new Error(
|
|
279
|
+
"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."
|
|
280
|
+
);
|
|
281
|
+
}
|
|
282
|
+
const deviceId = loadDeviceId();
|
|
283
|
+
let backendResponse = null;
|
|
284
|
+
let explicitRejection = false;
|
|
285
|
+
let transientFailure = false;
|
|
286
|
+
try {
|
|
287
|
+
const res = await fetchRetry(`${API_BASE}/auth/activate`, {
|
|
288
|
+
method: "POST",
|
|
289
|
+
headers: { "Content-Type": "application/json" },
|
|
290
|
+
body: JSON.stringify({ apiKey, deviceId }),
|
|
291
|
+
signal: AbortSignal.timeout(1e4)
|
|
292
|
+
});
|
|
293
|
+
if (res.ok) {
|
|
294
|
+
backendResponse = await res.json();
|
|
295
|
+
if (!backendResponse.valid) explicitRejection = true;
|
|
296
|
+
} else if (res.status === 401 || res.status === 403) {
|
|
297
|
+
explicitRejection = true;
|
|
298
|
+
} else {
|
|
299
|
+
transientFailure = true;
|
|
300
|
+
}
|
|
301
|
+
} catch {
|
|
302
|
+
transientFailure = true;
|
|
303
|
+
}
|
|
304
|
+
if (backendResponse?.valid) {
|
|
305
|
+
if (backendResponse.token) {
|
|
306
|
+
cacheResponse(backendResponse.token);
|
|
307
|
+
const verified = await verifyLicenseJwt(backendResponse.token);
|
|
308
|
+
if (verified) return verified;
|
|
309
|
+
}
|
|
310
|
+
const limits = PLAN_LIMITS[backendResponse.plan] ?? PLAN_LIMITS.free;
|
|
311
|
+
return {
|
|
312
|
+
valid: true,
|
|
313
|
+
plan: backendResponse.plan,
|
|
314
|
+
email: backendResponse.email,
|
|
315
|
+
expiresAt: backendResponse.expiresAt,
|
|
316
|
+
deviceLimit: limits.devices,
|
|
317
|
+
employeeLimit: limits.employees,
|
|
318
|
+
memoryLimit: limits.memories
|
|
319
|
+
};
|
|
320
|
+
}
|
|
321
|
+
if (explicitRejection) {
|
|
322
|
+
throw new Error(
|
|
323
|
+
`License invalid or expired. Renew at https://askexe.com, then restart. Backend rejected key ending in ...${apiKey.slice(-4)}.`
|
|
324
|
+
);
|
|
325
|
+
}
|
|
326
|
+
if (!transientFailure) {
|
|
327
|
+
throw new Error(
|
|
328
|
+
"License validation failed: unknown backend state. Restore network connectivity to https://api.askexe.com and retry."
|
|
329
|
+
);
|
|
330
|
+
}
|
|
331
|
+
const fresh = await getCachedLicense();
|
|
332
|
+
if (fresh && fresh.valid) return fresh;
|
|
333
|
+
const graceDays = opts?.offlineGraceDays ?? 7;
|
|
334
|
+
const graced = await getGraceVerifiedLicense(graceDays);
|
|
335
|
+
if (graced) return graced;
|
|
336
|
+
throw new Error(
|
|
337
|
+
`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.`
|
|
338
|
+
);
|
|
339
|
+
}
|
|
340
|
+
var _revalTimer = null;
|
|
341
|
+
function startLicenseRevalidation(intervalMs = 36e5) {
|
|
342
|
+
if (_revalTimer) return;
|
|
343
|
+
_revalTimer = setInterval(async () => {
|
|
344
|
+
try {
|
|
345
|
+
const license = await checkLicense();
|
|
346
|
+
if (!license.valid) {
|
|
347
|
+
process.stderr.write("[exe-os] License expired or invalid \u2014 features may be restricted\n");
|
|
348
|
+
}
|
|
349
|
+
} catch {
|
|
350
|
+
}
|
|
351
|
+
}, intervalMs);
|
|
352
|
+
if (_revalTimer && typeof _revalTimer === "object" && "unref" in _revalTimer) {
|
|
353
|
+
_revalTimer.unref();
|
|
354
|
+
}
|
|
355
|
+
}
|
|
356
|
+
function stopLicenseRevalidation() {
|
|
357
|
+
if (_revalTimer) {
|
|
358
|
+
clearInterval(_revalTimer);
|
|
359
|
+
_revalTimer = null;
|
|
360
|
+
}
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
export {
|
|
364
|
+
LICENSE_PUBLIC_KEY_PEM,
|
|
365
|
+
PLAN_LIMITS,
|
|
366
|
+
loadDeviceId,
|
|
367
|
+
loadLicense,
|
|
368
|
+
saveLicense,
|
|
369
|
+
getCachedLicense,
|
|
370
|
+
getGraceVerifiedLicense,
|
|
371
|
+
readCachedLicenseToken,
|
|
372
|
+
validateLicense,
|
|
373
|
+
checkLicense,
|
|
374
|
+
isFeatureAllowed,
|
|
375
|
+
mirrorLicenseKey,
|
|
376
|
+
assertVpsLicense,
|
|
377
|
+
startLicenseRevalidation,
|
|
378
|
+
stopLicenseRevalidation
|
|
379
|
+
};
|
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
import {
|
|
2
|
+
loadConfigSync
|
|
3
|
+
} from "./chunk-CEKZ77KA.js";
|
|
4
|
+
|
|
5
|
+
// src/lib/push-notifications.ts
|
|
6
|
+
import { execSync } from "child_process";
|
|
7
|
+
import nodemailer from "nodemailer";
|
|
8
|
+
var TELEGRAM_TIMEOUT_MS = 1e4;
|
|
9
|
+
var SMTP_TIMEOUT_MS = 15e3;
|
|
10
|
+
var DESKTOP_TIMEOUT_MS = 3e3;
|
|
11
|
+
var _cachedConfig = null;
|
|
12
|
+
var _notifyOn = null;
|
|
13
|
+
function loadPushConfig() {
|
|
14
|
+
if (_cachedConfig && _notifyOn) return { config: _cachedConfig, notifyOn: _notifyOn };
|
|
15
|
+
try {
|
|
16
|
+
const cfg = loadConfigSync();
|
|
17
|
+
_cachedConfig = cfg.notifications ?? {};
|
|
18
|
+
const events = cfg.notify_on;
|
|
19
|
+
_notifyOn = new Set(events ?? ["task_complete", "task_blocked", "context_full", "error_spike", "daemon_crash", "review_ready"]);
|
|
20
|
+
} catch {
|
|
21
|
+
_cachedConfig = {};
|
|
22
|
+
_notifyOn = /* @__PURE__ */ new Set(["task_complete", "task_blocked", "context_full", "error_spike", "daemon_crash", "review_ready"]);
|
|
23
|
+
}
|
|
24
|
+
return { config: _cachedConfig, notifyOn: _notifyOn };
|
|
25
|
+
}
|
|
26
|
+
function _resetPushConfig() {
|
|
27
|
+
_cachedConfig = null;
|
|
28
|
+
_notifyOn = null;
|
|
29
|
+
}
|
|
30
|
+
function sendDesktopNotification(title, body, sessionScope) {
|
|
31
|
+
try {
|
|
32
|
+
const safeTitle = title.replace(/["\\\n]/g, " ").slice(0, 100);
|
|
33
|
+
const safeBody = body.replace(/["\\\n]/g, " ").slice(0, 200);
|
|
34
|
+
try {
|
|
35
|
+
const osc9 = `\x1B]9;${safeTitle}: ${safeBody}\x07`;
|
|
36
|
+
let coordinatorSession;
|
|
37
|
+
if (sessionScope) {
|
|
38
|
+
coordinatorSession = sessionScope;
|
|
39
|
+
} else {
|
|
40
|
+
const sessions = execSync("tmux list-sessions -F '#{session_name}' 2>/dev/null", { encoding: "utf8", timeout: 2e3 }).trim().split("\n");
|
|
41
|
+
coordinatorSession = sessions.find((s) => /^exe\d*$/.test(s));
|
|
42
|
+
}
|
|
43
|
+
if (coordinatorSession) {
|
|
44
|
+
execSync(`printf '${osc9.replace(/'/g, "'\\''")}' > /dev/tty 2>/dev/null || tmux display-message -p -t ${coordinatorSession} '' 2>/dev/null`, { timeout: 2e3, stdio: "ignore" });
|
|
45
|
+
return true;
|
|
46
|
+
}
|
|
47
|
+
} catch {
|
|
48
|
+
}
|
|
49
|
+
if (process.platform === "linux") {
|
|
50
|
+
execSync(`notify-send "${safeTitle}" "${safeBody}"`, {
|
|
51
|
+
timeout: DESKTOP_TIMEOUT_MS,
|
|
52
|
+
stdio: "ignore"
|
|
53
|
+
});
|
|
54
|
+
return true;
|
|
55
|
+
}
|
|
56
|
+
return false;
|
|
57
|
+
} catch (err) {
|
|
58
|
+
process.stderr.write(`[push-notify] Desktop notification failed: ${err instanceof Error ? err.message : String(err)}
|
|
59
|
+
`);
|
|
60
|
+
return false;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
async function sendTelegram(botToken, chatId, message) {
|
|
64
|
+
try {
|
|
65
|
+
const url = `https://api.telegram.org/bot${botToken}/sendMessage`;
|
|
66
|
+
const res = await fetch(url, {
|
|
67
|
+
method: "POST",
|
|
68
|
+
headers: { "Content-Type": "application/json" },
|
|
69
|
+
body: JSON.stringify({
|
|
70
|
+
chat_id: chatId,
|
|
71
|
+
text: message,
|
|
72
|
+
parse_mode: "Markdown",
|
|
73
|
+
disable_web_page_preview: true
|
|
74
|
+
}),
|
|
75
|
+
signal: AbortSignal.timeout(TELEGRAM_TIMEOUT_MS)
|
|
76
|
+
});
|
|
77
|
+
return res.ok;
|
|
78
|
+
} catch (err) {
|
|
79
|
+
process.stderr.write(`[push-notify] Telegram send failed: ${err instanceof Error ? err.message : String(err)}
|
|
80
|
+
`);
|
|
81
|
+
return false;
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
async function sendEmail(smtpUrl, from, to, subject, body) {
|
|
85
|
+
if (smtpUrl.startsWith("http")) {
|
|
86
|
+
try {
|
|
87
|
+
const res = await fetch(smtpUrl, {
|
|
88
|
+
method: "POST",
|
|
89
|
+
headers: { "Content-Type": "application/json" },
|
|
90
|
+
body: JSON.stringify({ from, to, subject, text: body }),
|
|
91
|
+
signal: AbortSignal.timeout(SMTP_TIMEOUT_MS)
|
|
92
|
+
});
|
|
93
|
+
return res.ok;
|
|
94
|
+
} catch (err) {
|
|
95
|
+
process.stderr.write(`[push-notify] Email send failed: ${err instanceof Error ? err.message : String(err)}
|
|
96
|
+
`);
|
|
97
|
+
return false;
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
if (!smtpUrl.startsWith("smtp://") && !smtpUrl.startsWith("smtps://")) {
|
|
101
|
+
process.stderr.write(
|
|
102
|
+
`[push-notify] Unsupported email.smtp_url scheme "${smtpUrl}". Use an smtp:// / smtps:// connection URL or an https:// API endpoint.
|
|
103
|
+
`
|
|
104
|
+
);
|
|
105
|
+
return false;
|
|
106
|
+
}
|
|
107
|
+
try {
|
|
108
|
+
const transporter = nodemailer.createTransport(smtpUrl);
|
|
109
|
+
await transporter.sendMail({ from, to, subject, text: body });
|
|
110
|
+
return true;
|
|
111
|
+
} catch (err) {
|
|
112
|
+
process.stderr.write(
|
|
113
|
+
`[push-notify] SMTP send failed: ${err instanceof Error ? err.message : String(err)}
|
|
114
|
+
`
|
|
115
|
+
);
|
|
116
|
+
return false;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
function formatMessage(msg) {
|
|
120
|
+
const emoji = msg.priority === "urgent" ? "\u{1F534}" : msg.priority === "low" ? "\u26AA" : "\u{1F535}";
|
|
121
|
+
const lines = [
|
|
122
|
+
`${emoji} *${msg.title}*`,
|
|
123
|
+
msg.body
|
|
124
|
+
];
|
|
125
|
+
if (msg.agent) lines.push(`Agent: ${msg.agent}`);
|
|
126
|
+
if (msg.project) lines.push(`Project: ${msg.project}`);
|
|
127
|
+
return lines.join("\n");
|
|
128
|
+
}
|
|
129
|
+
async function pushNotify(msg) {
|
|
130
|
+
const { config, notifyOn } = loadPushConfig();
|
|
131
|
+
const result = { desktop: false, telegram: false, email: false };
|
|
132
|
+
if (!notifyOn.has(msg.event)) return result;
|
|
133
|
+
const formatted = formatMessage(msg);
|
|
134
|
+
if (config.desktop !== false) {
|
|
135
|
+
result.desktop = sendDesktopNotification(msg.title, msg.body, msg.sessionScope);
|
|
136
|
+
}
|
|
137
|
+
if (config.telegram?.bot_token && config.telegram?.chat_id) {
|
|
138
|
+
result.telegram = await sendTelegram(
|
|
139
|
+
config.telegram.bot_token,
|
|
140
|
+
config.telegram.chat_id,
|
|
141
|
+
formatted
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
if (config.email?.smtp_url && config.email?.from && config.email?.to) {
|
|
145
|
+
const url = config.email.smtp_url;
|
|
146
|
+
const recognized = url.startsWith("smtp://") || url.startsWith("smtps://") || url.startsWith("http");
|
|
147
|
+
if (!recognized) {
|
|
148
|
+
process.stderr.write(
|
|
149
|
+
`[push-notify] WARNING: email.smtp_url "${url}" has an unrecognized scheme. Use an smtp:// / smtps:// connection URL or an https:// email API endpoint.
|
|
150
|
+
`
|
|
151
|
+
);
|
|
152
|
+
} else {
|
|
153
|
+
result.email = await sendEmail(
|
|
154
|
+
url,
|
|
155
|
+
config.email.from,
|
|
156
|
+
config.email.to,
|
|
157
|
+
`[exe-os] ${msg.title}`,
|
|
158
|
+
msg.body
|
|
159
|
+
);
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
return result;
|
|
163
|
+
}
|
|
164
|
+
function pushNotifyAsync(msg) {
|
|
165
|
+
pushNotify(msg).catch((err) => {
|
|
166
|
+
process.stderr.write(`[push-notify] async send failed: ${err instanceof Error ? err.message : String(err)}
|
|
167
|
+
`);
|
|
168
|
+
});
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
export {
|
|
172
|
+
loadPushConfig,
|
|
173
|
+
_resetPushConfig,
|
|
174
|
+
pushNotify,
|
|
175
|
+
pushNotifyAsync
|
|
176
|
+
};
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getClient
|
|
3
|
+
} from "./chunk-WLDBPKRL.js";
|
|
4
|
+
|
|
5
|
+
// src/lib/session-kill-telemetry.ts
|
|
6
|
+
import crypto from "crypto";
|
|
7
|
+
import { appendFileSync } from "fs";
|
|
8
|
+
import path from "path";
|
|
9
|
+
import os from "os";
|
|
10
|
+
var KILL_FILE_LOG = path.join(os.homedir(), ".exe-os", "logs", "session-kills.jsonl");
|
|
11
|
+
var TOKENS_PER_IDLE_MINUTE = 50;
|
|
12
|
+
async function recordSessionKill(input) {
|
|
13
|
+
const killedAt = (/* @__PURE__ */ new Date()).toISOString();
|
|
14
|
+
const id = crypto.randomUUID();
|
|
15
|
+
try {
|
|
16
|
+
const logEntry = JSON.stringify({
|
|
17
|
+
id,
|
|
18
|
+
timestamp: killedAt,
|
|
19
|
+
session: input.sessionName,
|
|
20
|
+
agent: input.agentId,
|
|
21
|
+
reason: input.reason,
|
|
22
|
+
ticksIdle: input.ticksIdle ?? null,
|
|
23
|
+
tokensSaved: input.estimatedTokensSaved ?? null
|
|
24
|
+
});
|
|
25
|
+
appendFileSync(KILL_FILE_LOG, logEntry + "\n");
|
|
26
|
+
} catch {
|
|
27
|
+
process.stderr.write(
|
|
28
|
+
`[session-kill-telemetry] file log write failed for ${input.sessionName}
|
|
29
|
+
`
|
|
30
|
+
);
|
|
31
|
+
}
|
|
32
|
+
try {
|
|
33
|
+
const client = getClient();
|
|
34
|
+
await client.execute({
|
|
35
|
+
sql: `INSERT INTO session_kills
|
|
36
|
+
(id, session_name, agent_id, killed_at, reason,
|
|
37
|
+
ticks_idle, estimated_tokens_saved)
|
|
38
|
+
VALUES (?, ?, ?, ?, ?, ?, ?)`,
|
|
39
|
+
args: [
|
|
40
|
+
id,
|
|
41
|
+
input.sessionName,
|
|
42
|
+
input.agentId,
|
|
43
|
+
killedAt,
|
|
44
|
+
input.reason,
|
|
45
|
+
input.ticksIdle ?? null,
|
|
46
|
+
input.estimatedTokensSaved ?? null
|
|
47
|
+
]
|
|
48
|
+
});
|
|
49
|
+
} catch (err) {
|
|
50
|
+
process.stderr.write(
|
|
51
|
+
`[session-kill-telemetry] DB write failed: ${err instanceof Error ? err.message : String(err)}
|
|
52
|
+
`
|
|
53
|
+
);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
async function countKillsSince(sinceISO) {
|
|
57
|
+
try {
|
|
58
|
+
const client = getClient();
|
|
59
|
+
const result = await client.execute({
|
|
60
|
+
sql: `SELECT COUNT(*) AS n FROM session_kills WHERE killed_at >= ?`,
|
|
61
|
+
args: [sinceISO]
|
|
62
|
+
});
|
|
63
|
+
const row = result.rows[0];
|
|
64
|
+
return row ? Number(row.n) : 0;
|
|
65
|
+
} catch {
|
|
66
|
+
return 0;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
var IDLE_KILL_STREAK_META_KEY = "idle_kill_suspect_streak";
|
|
70
|
+
var IDLE_KILL_SUSPECT_DAY_THRESHOLD = 3;
|
|
71
|
+
var IDLE_KILL_MIN_LIVE_SESSIONS = 5;
|
|
72
|
+
function parseStreakState(raw) {
|
|
73
|
+
if (!raw) return { lastDate: null, streak: 0 };
|
|
74
|
+
try {
|
|
75
|
+
const parsed = JSON.parse(raw);
|
|
76
|
+
return {
|
|
77
|
+
lastDate: typeof parsed.lastDate === "string" ? parsed.lastDate : null,
|
|
78
|
+
streak: typeof parsed.streak === "number" ? parsed.streak : 0
|
|
79
|
+
};
|
|
80
|
+
} catch {
|
|
81
|
+
return { lastDate: null, streak: 0 };
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
function nextStreakState(prev, qualifiesToday, todayDate) {
|
|
85
|
+
if (!qualifiesToday) return { lastDate: todayDate, streak: 0 };
|
|
86
|
+
if (prev.lastDate === todayDate) return prev;
|
|
87
|
+
return { lastDate: todayDate, streak: prev.streak + 1 };
|
|
88
|
+
}
|
|
89
|
+
function computeIdleKillSuspectStreak(prev, killsToday, liveSessions, todayDate) {
|
|
90
|
+
const qualifies = killsToday === 0 && liveSessions >= IDLE_KILL_MIN_LIVE_SESSIONS;
|
|
91
|
+
const state = nextStreakState(prev, qualifies, todayDate);
|
|
92
|
+
return {
|
|
93
|
+
state,
|
|
94
|
+
suspect: state.streak >= IDLE_KILL_SUSPECT_DAY_THRESHOLD
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
async function sumTokensSavedSince(sinceISO) {
|
|
98
|
+
try {
|
|
99
|
+
const client = getClient();
|
|
100
|
+
const result = await client.execute({
|
|
101
|
+
sql: `SELECT COALESCE(SUM(estimated_tokens_saved), 0) AS total
|
|
102
|
+
FROM session_kills
|
|
103
|
+
WHERE killed_at >= ?`,
|
|
104
|
+
args: [sinceISO]
|
|
105
|
+
});
|
|
106
|
+
const row = result.rows[0];
|
|
107
|
+
return row ? Number(row.total) : 0;
|
|
108
|
+
} catch {
|
|
109
|
+
return 0;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
export {
|
|
114
|
+
TOKENS_PER_IDLE_MINUTE,
|
|
115
|
+
recordSessionKill,
|
|
116
|
+
countKillsSince,
|
|
117
|
+
IDLE_KILL_STREAK_META_KEY,
|
|
118
|
+
IDLE_KILL_SUSPECT_DAY_THRESHOLD,
|
|
119
|
+
IDLE_KILL_MIN_LIVE_SESSIONS,
|
|
120
|
+
parseStreakState,
|
|
121
|
+
computeIdleKillSuspectStreak,
|
|
122
|
+
sumTokensSavedSince
|
|
123
|
+
};
|