@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.
- package/deploy/compose/.env.customer.example +1 -1
- package/deploy/compose/.env.default +2 -1
- package/deploy/compose/.env.example +1 -1
- package/deploy/compose/docker-compose.yml +74 -13
- package/deploy/compose/erp-nginx/nginx.conf +52 -0
- package/deploy/compose/generate-env.ts +1 -1
- package/deploy/compose/init-db.sql +13 -2
- package/deploy/stack-manifests/v0.9.json +11 -31
- package/dist/active-agent-6ZBHGHXF.js +26 -0
- package/dist/active-agent-E23SCYER.js +27 -0
- package/dist/active-agent-KMZT44S4.js +26 -0
- package/dist/active-agent-LFFTVROM.js +27 -0
- package/dist/agentic-ontology-PCZB5HV5.js +25 -0
- package/dist/agentic-ontology-PGGJN2ES.js +25 -0
- package/dist/assets/tmux.conf +2 -0
- package/dist/backfill-metadata-KQ4FEVUR.js +599 -0
- package/dist/backfill-metadata-Y3YWCHKJ.js +599 -0
- package/dist/behaviors-H4DZECKL.js +39 -0
- package/dist/behaviors-WIUTIJF6.js +39 -0
- package/dist/bin/agentic-ontology-backfill.js +5 -5
- package/dist/bin/agentic-reflection-backfill.js +6 -6
- package/dist/bin/agentic-semantic-label.js +5 -5
- package/dist/bin/backfill-conversations.js +4 -4
- package/dist/bin/backfill-responses.js +4 -4
- package/dist/bin/backfill-vectors.js +5 -5
- package/dist/bin/bulk-sync-postgres.js +7 -7
- package/dist/bin/cc-doctor.js +4 -4
- package/dist/bin/cleanup-stale-review-tasks.js +10 -10
- package/dist/bin/cli.js +15 -15
- package/dist/bin/exe-agent-config.js +2 -2
- package/dist/bin/exe-agent.js +4 -4
- package/dist/bin/exe-assign.js +5 -5
- package/dist/bin/exe-boot.js +17 -17
- package/dist/bin/exe-call.js +4 -4
- package/dist/bin/exe-cloud.js +5 -5
- package/dist/bin/exe-dispatch.js +10 -10
- package/dist/bin/exe-doctor.js +1 -1
- package/dist/bin/exe-export-behaviors.js +7 -7
- package/dist/bin/exe-forget.js +6 -6
- package/dist/bin/exe-gateway.js +7 -7
- package/dist/bin/exe-healthcheck.js +4 -4
- package/dist/bin/exe-heartbeat.js +10 -10
- package/dist/bin/exe-kill.js +13 -13
- package/dist/bin/exe-launch-agent.js +17 -17
- package/dist/bin/exe-new-employee.js +7 -7
- package/dist/bin/exe-pending-messages.js +11 -11
- package/dist/bin/exe-pending-notifications.js +10 -10
- package/dist/bin/exe-pending-reviews.js +10 -10
- package/dist/bin/exe-rename.js +4 -4
- package/dist/bin/exe-review.js +12 -12
- package/dist/bin/exe-search.js +5 -5
- package/dist/bin/exe-session-cleanup.js +15 -15
- package/dist/bin/exe-settings.js +5 -5
- package/dist/bin/exe-start-codex.js +11 -11
- package/dist/bin/exe-start-opencode.js +8 -8
- package/dist/bin/exe-status.js +11 -11
- package/dist/bin/exe-support.js +2 -2
- package/dist/bin/exe-team.js +3 -3
- package/dist/bin/exe-watchdog.js +17 -2
- package/dist/bin/git-sweep.js +11 -11
- package/dist/bin/graph-backfill.js +4 -4
- package/dist/bin/graph-export.js +5 -5
- package/dist/bin/import-history.js +7 -7
- package/dist/bin/install.js +8 -7
- package/dist/bin/intercom-check.js +4 -4
- package/dist/bin/mcp-sessions.js +2 -2
- package/dist/bin/orchestration-metrics.js +4 -4
- package/dist/bin/postgres-agentic-reflection-backfill.js +2 -2
- package/dist/bin/postgres-agentic-semantic-backfill.js +1 -1
- package/dist/bin/scan-tasks.js +10 -10
- package/dist/bin/setup.js +1 -1
- package/dist/bin/shard-migrate.js +4 -4
- package/dist/bin/stack-update.js +3 -3
- package/dist/bin/vps-health-gate.js +1 -1
- package/dist/capability-cards-F22XWGLB.js +88 -0
- package/dist/capability-cards-KCKITXXZ.js +88 -0
- package/dist/capacity-monitor-CHXGKVEO.js +50 -0
- package/dist/capacity-monitor-L3M5KB6T.js +50 -0
- package/dist/catchup-brief-BVZANLRZ.js +154 -0
- package/dist/catchup-brief-LDUGMC7S.js +154 -0
- package/dist/catchup-brief-NZDLL7SA.js +154 -0
- package/dist/catchup-brief-OOIXVGBA.js +154 -0
- package/dist/catchup-brief-UPXHDYTB.js +154 -0
- package/dist/chunk-2BI3FQKL.js +1876 -0
- package/dist/chunk-2CJUQGHH.js +362 -0
- package/dist/chunk-2EVYBMBJ.js +128 -0
- package/dist/chunk-2NQOZOXG.js +2113 -0
- package/dist/chunk-2QK5E3LB.js +128 -0
- package/dist/chunk-2VTCG4ZU.js +1352 -0
- package/dist/chunk-2YGI36DV.js +1119 -0
- package/dist/chunk-32BQN2QN.js +185 -0
- package/dist/chunk-33JMO4UV.js +157 -0
- package/dist/chunk-3DZAYLXY.js +377 -0
- package/dist/chunk-3MGBE7GR.js +76 -0
- package/dist/chunk-3MTV4FJL.js +271 -0
- package/dist/chunk-3NN7VQ27.js +1352 -0
- package/dist/chunk-4533HNOG.js +70 -0
- package/dist/chunk-456UW3MT.js +731 -0
- package/dist/chunk-4AU56XV2.js +58 -0
- package/dist/chunk-4MRP6EBR.js +280 -0
- package/dist/chunk-4MRWW52U.js +14219 -0
- package/dist/chunk-4WWECNAY.js +50 -0
- package/dist/chunk-4ZJDDR6L.js +171 -0
- package/dist/chunk-522EOPM6.js +382 -0
- package/dist/chunk-52UDAVZE.js +3208 -0
- package/dist/chunk-5FAMUB4X.js +204 -0
- package/dist/chunk-5PGZJQUI.js +58 -0
- package/dist/chunk-67TZJXNZ.js +262 -0
- package/dist/chunk-6FEZ7GN2.js +123 -0
- package/dist/chunk-6MOGND7S.js +14219 -0
- package/dist/chunk-6TRSVY7L.js +181 -0
- package/dist/chunk-6WCS7ZNK.js +85 -0
- package/dist/chunk-6WRYDREW.js +539 -0
- package/dist/chunk-72UA2FB3.js +181 -0
- package/dist/chunk-77DMFEOL.js +30 -0
- package/dist/chunk-7COXVQ5W.js +214 -0
- package/dist/chunk-7HLWBYH7.js +60 -0
- package/dist/chunk-ALSTZCWT.js +204 -0
- package/dist/chunk-ARUTDXZX.js +280 -0
- package/dist/chunk-AU47B4QY.js +129 -0
- package/dist/chunk-AXOREWVL.js +836 -0
- package/dist/chunk-B5MNC54V.js +127 -0
- package/dist/chunk-B74VSMKX.js +1350 -0
- package/dist/chunk-BHWLH44J.js +362 -0
- package/dist/chunk-BUPZ3HD2.js +85 -0
- package/dist/chunk-BWVLMA53.js +2113 -0
- package/dist/chunk-C5WLBKMJ.js +50 -0
- package/dist/chunk-C62T6R2A.js +97 -0
- package/dist/chunk-CV5KBAIK.js +33 -0
- package/dist/chunk-DAYSDWXA.js +1068 -0
- package/dist/chunk-DI4URIUB.js +227 -0
- package/dist/chunk-DNTCYFJ6.js +76 -0
- package/dist/chunk-DPDRRS7T.js +103 -0
- package/dist/chunk-DT3EV6CW.js +103 -0
- package/dist/chunk-DX45HDWY.js +1076 -0
- package/dist/chunk-E72BD6MG.js +284 -0
- package/dist/chunk-EAPUSVKS.js +375 -0
- package/dist/chunk-EFUANRRT.js +85 -0
- package/dist/chunk-EG2SCT5R.js +1352 -0
- package/dist/chunk-EJD2JU77.js +58 -0
- package/dist/chunk-EMXYUAVP.js +81 -0
- package/dist/chunk-ENM2TAAM.js +14219 -0
- package/dist/chunk-EPDRTPVP.js +1876 -0
- package/dist/chunk-EW6XDHID.js +221 -0
- package/dist/chunk-EYEGSAWZ.js +1094 -0
- package/dist/chunk-F6L33PAQ.js +231 -0
- package/dist/chunk-FC2SCTVE.js +38 -0
- package/dist/chunk-FSRKIZGZ.js +630 -0
- package/dist/chunk-FTG7I5CB.js +81 -0
- package/dist/chunk-GVFRLWX7.js +30 -0
- package/dist/chunk-H3XMZOWW.js +1119 -0
- package/dist/chunk-HAKXE6LN.js +123 -0
- package/dist/chunk-HLP3ZDTW.js +448 -0
- package/dist/chunk-HOYWKQAA.js +510 -0
- package/dist/chunk-HWDD64IW.js +712 -0
- package/dist/chunk-HWMCULHY.js +127 -0
- package/dist/chunk-HZZHRZPK.js +210 -0
- package/dist/chunk-ICRWTYNW.js +103 -0
- package/dist/chunk-J5MWPC33.js +167 -0
- package/dist/chunk-JBXANNNB.js +70 -0
- package/dist/chunk-JPBMIYWF.js +1352 -0
- package/dist/chunk-JY6EXBFI.js +373 -0
- package/dist/chunk-JZLIBXI7.js +14219 -0
- package/dist/chunk-JZPTKXJ6.js +668 -0
- package/dist/chunk-KEYMA4ZP.js +510 -0
- package/dist/chunk-KHGNN6GL.js +2078 -0
- package/dist/chunk-KVLB2PD6.js +97 -0
- package/dist/chunk-L3YBRBKL.js +1076 -0
- package/dist/chunk-L7VZ32NA.js +89 -0
- package/dist/chunk-LEHLADW4.js +221 -0
- package/dist/chunk-LPD4HILQ.js +262 -0
- package/dist/chunk-LSJ3ADDI.js +51 -0
- package/dist/chunk-LUMS2MAS.js +58 -0
- package/dist/chunk-LV4SEC6C.js +394 -0
- package/dist/chunk-LXXBEI4A.js +284 -0
- package/dist/chunk-M4NPCAIH.js +456 -0
- package/dist/chunk-MFJ62LQ5.js +157 -0
- package/dist/chunk-MGEZNKOD.js +836 -0
- package/dist/chunk-MJ7X5IBW.js +227 -0
- package/dist/chunk-MLGRWCY4.js +54 -0
- package/dist/chunk-MQZX57IY.js +348 -0
- package/dist/chunk-MY7MFF6J.js +103 -0
- package/dist/chunk-MYA2X5OY.js +185 -0
- package/dist/chunk-MYEABW5Z.js +630 -0
- package/dist/chunk-MZ2TDCAL.js +402 -0
- package/dist/chunk-NF3FRB7Z.js +271 -0
- package/dist/chunk-NXYIFEPV.js +539 -0
- package/dist/chunk-O2GVE5B5.js +58 -0
- package/dist/chunk-O5GUCDR2.js +456 -0
- package/dist/chunk-OBZNRECA.js +128 -0
- package/dist/chunk-OMZ2RLJG.js +214 -0
- package/dist/chunk-ORDHJRWN.js +299 -0
- package/dist/chunk-OSPIJMCD.js +210 -0
- package/dist/chunk-OUGWEH4J.js +240 -0
- package/dist/chunk-OV5MJQGC.js +1876 -0
- package/dist/chunk-PC635OAG.js +4318 -0
- package/dist/chunk-PH46R4J6.js +348 -0
- package/dist/chunk-PJP2EP7P.js +394 -0
- package/dist/chunk-PODFWH3V.js +333 -0
- package/dist/chunk-PPWH3SHR.js +1068 -0
- package/dist/chunk-PWPJK7KB.js +4318 -0
- package/dist/chunk-PXONZVG4.js +377 -0
- package/dist/chunk-QAOGJRZD.js +369 -0
- package/dist/chunk-QC3LAEI7.js +197 -0
- package/dist/chunk-QDWQDUWI.js +668 -0
- package/dist/chunk-QKJFD6BH.js +1350 -0
- package/dist/chunk-QNYVJGFM.js +345 -0
- package/dist/chunk-QQF3XGQ5.js +14219 -0
- package/dist/chunk-QRPFQNI3.js +150 -0
- package/dist/chunk-QSSU5XWD.js +731 -0
- package/dist/chunk-QXZAGVAV.js +2078 -0
- package/dist/chunk-RA54MW64.js +244 -0
- package/dist/chunk-RF7PUWXI.js +197 -0
- package/dist/chunk-S3INDYSO.js +244 -0
- package/dist/chunk-SLQVTHH5.js +369 -0
- package/dist/chunk-SY2B74KL.js +345 -0
- package/dist/chunk-T6QPXXXW.js +712 -0
- package/dist/chunk-TO5M5YCT.js +41 -0
- package/dist/chunk-TQ4VXUAF.js +129 -0
- package/dist/chunk-UAB7RQC4.js +41 -0
- package/dist/chunk-UMEIBDYW.js +97 -0
- package/dist/chunk-UXW5TB7Y.js +240 -0
- package/dist/chunk-VB2N5WOX.js +150 -0
- package/dist/chunk-VLZEMRG3.js +167 -0
- package/dist/chunk-VM3V6VK7.js +230 -0
- package/dist/chunk-VMCGKBHB.js +1352 -0
- package/dist/chunk-VNIYZAR5.js +128 -0
- package/dist/chunk-VYV4KOD2.js +85 -0
- package/dist/chunk-W4SRJBAT.js +171 -0
- package/dist/chunk-W5W3LZ3Q.js +54 -0
- package/dist/chunk-WDNZEOM3.js +38 -0
- package/dist/chunk-WUKEXVOR.js +3208 -0
- package/dist/chunk-X3Z35Q6L.js +373 -0
- package/dist/chunk-X4VOU6BQ.js +382 -0
- package/dist/chunk-X6EEVSVG.js +290 -0
- package/dist/chunk-XFHGWGNB.js +1094 -0
- package/dist/chunk-XKOLRWYA.js +33 -0
- package/dist/chunk-XMMIL3UD.js +402 -0
- package/dist/chunk-XWILC6VA.js +290 -0
- package/dist/chunk-Y4OQCX4C.js +97 -0
- package/dist/chunk-Y67VYYOA.js +231 -0
- package/dist/chunk-YGQCQTQH.js +230 -0
- package/dist/chunk-YGWFBN5A.js +299 -0
- package/dist/chunk-YMKUXZIG.js +379 -0
- package/dist/chunk-YOMLMT7E.js +230 -0
- package/dist/chunk-YPESIZOB.js +14219 -0
- package/dist/chunk-Z2CGCIU2.js +89 -0
- package/dist/chunk-ZLAWNHQR.js +448 -0
- package/dist/chunk-ZME5UQSN.js +333 -0
- package/dist/co-activation-NUEQYXE5.js +73 -0
- package/dist/co-activation-ZG5HLBCZ.js +73 -0
- package/dist/co-occurrence-7S5KWQB2.js +94 -0
- package/dist/co-occurrence-X5SWDXT2.js +94 -0
- package/dist/core-memory-GOPBRGGZ.js +110 -0
- package/dist/core-memory-XLCU6L5M.js +110 -0
- package/dist/crdt-sync-EPKHPGRZ.js +33 -0
- package/dist/crdt-sync-UIQJ5U7T.js +33 -0
- package/dist/crm-webhook-MKN23JNU.js +10 -0
- package/dist/crm-webhook-SM63BPXO.js +10 -0
- package/dist/cto-delegation-gate-PQY5TOVZ.js +279 -0
- package/dist/cto-delegation-gate-V5VVUR3G.js +279 -0
- package/dist/daemon-orchestration-C7AAS67Q.js +138 -0
- package/dist/daemon-orchestration-OBCAJB2H.js +138 -0
- package/dist/db-backup-F7VP4QRH.js +33 -0
- package/dist/db-backup-KVYC57W7.js +33 -0
- package/dist/doc-graph-extractor-H2ETEINP.js +132 -0
- package/dist/doc-graph-extractor-PCUZEYCH.js +132 -0
- package/dist/dreaming-4OZXSLE3.js +33 -0
- package/dist/dreaming-Z2RYEYNT.js +33 -0
- package/dist/exe-drift-GEWNIK7A.js +69 -0
- package/dist/exe-drift-XCGH7AFO.js +69 -0
- package/dist/exe-export-7DKAU5IP.js +75 -0
- package/dist/exe-export-BCHH6OE6.js +75 -0
- package/dist/exe-import-PDRIZVYF.js +78 -0
- package/dist/exe-import-ZCKUDFKL.js +78 -0
- package/dist/exe-key-FUWLLI3U.js +580 -0
- package/dist/exe-key-RKKNVUMP.js +580 -0
- package/dist/exe-snapshot-G4I5FQMK.js +337 -0
- package/dist/exe-snapshot-GWU7QTZK.js +337 -0
- package/dist/fast-db-init-6QG6YQNT.js +7 -0
- package/dist/fast-db-init-X2QDQUA4.js +7 -0
- package/dist/founder-context-TOMNUBGJ.js +96 -0
- package/dist/founder-context-UU3V6MAS.js +96 -0
- package/dist/gateway/index.js +8 -8
- package/dist/git-staleness-FEPFMZKF.js +111 -0
- package/dist/git-staleness-HYVYLCW3.js +111 -0
- package/dist/git-task-sweep-IRV52JIM.js +41 -0
- package/dist/git-task-sweep-T6BSM3GS.js +41 -0
- package/dist/global-procedures-3AURRMKO.js +21 -0
- package/dist/global-procedures-JPCYBZYC.js +21 -0
- package/dist/graph-auto-extract-OC3AOSMW.js +182 -0
- package/dist/graph-auto-extract-PVDYEJBY.js +182 -0
- package/dist/hooks/bug-report-worker.js +12 -12
- package/dist/hooks/codex-stop-task-finalizer.js +12 -12
- package/dist/hooks/commit-complete.js +12 -12
- package/dist/hooks/error-recall.js +6 -6
- package/dist/hooks/exe-heartbeat-hook.js +3 -3
- package/dist/hooks/ingest.js +6 -6
- package/dist/hooks/instructions-loaded.js +4 -4
- package/dist/hooks/manifest.json +19 -19
- package/dist/hooks/notification.js +4 -4
- package/dist/hooks/post-compact.js +11 -11
- package/dist/hooks/post-tool-combined.js +5 -5
- package/dist/hooks/pre-compact.js +12 -12
- package/dist/hooks/pre-tool-use.js +15 -15
- package/dist/hooks/prompt-submit.js +22 -22
- package/dist/hooks/session-end.js +16 -16
- package/dist/hooks/session-start.js +10 -10
- package/dist/hooks/stop.js +15 -15
- package/dist/hooks/subagent-stop.js +11 -11
- package/dist/hooks/summary-worker.js +15 -15
- package/dist/index.js +18 -18
- package/dist/installer-72XXLBRP.js +39 -0
- package/dist/installer-HDXG2BZN.js +343 -0
- package/dist/installer-JALMKPCS.js +297 -0
- package/dist/installer-Q46SNNLU.js +39 -0
- package/dist/installer-W7PIPRCX.js +343 -0
- package/dist/installer-Z7WQEOS7.js +297 -0
- package/dist/lib/cloud-sync.js +5 -5
- package/dist/lib/consolidation.js +5 -5
- package/dist/lib/database.js +2 -2
- package/dist/lib/db.js +2 -2
- package/dist/lib/employee-templates.js +4 -4
- package/dist/lib/employees.js +2 -2
- package/dist/lib/exe-daemon.js +45 -41
- package/dist/lib/hybrid-search.js +5 -5
- package/dist/lib/identity.js +2 -2
- package/dist/lib/license.js +1 -1
- package/dist/lib/messaging.js +10 -10
- package/dist/lib/reminders.js +3 -3
- package/dist/lib/schedules.js +5 -5
- package/dist/lib/session-registry.js +4 -4
- package/dist/lib/skill-learning.js +6 -6
- package/dist/lib/store.js +4 -4
- package/dist/lib/task-router.js +3 -3
- package/dist/lib/tasks.js +11 -11
- package/dist/lib/tmux-routing.js +9 -9
- package/dist/lib/token-spend.js +3 -3
- package/dist/license-gate-7JZCHOAG.js +14 -0
- package/dist/license-gate-OP4SKL4P.js +14 -0
- package/dist/mcp/register-tools.js +58 -58
- package/dist/mcp/server.js +59 -59
- package/dist/mcp/tools/complete-reminder.js +4 -4
- package/dist/mcp/tools/create-reminder.js +4 -4
- package/dist/mcp/tools/create-task.js +13 -13
- package/dist/mcp/tools/deactivate-behavior.js +7 -7
- package/dist/mcp/tools/list-reminders.js +4 -4
- package/dist/mcp/tools/list-tasks.js +13 -13
- package/dist/mcp/tools/send-message.js +12 -12
- package/dist/mcp/tools/update-task.js +12 -12
- package/dist/mcp-health-VULNT722.js +17 -0
- package/dist/mcp-health-WDOB6XUB.js +19 -0
- package/dist/mcp-http-config-2OZ7N74D.js +28 -0
- package/dist/mcp-http-config-4VXA5K73.js +28 -0
- package/dist/memory-cards-2K6QRZU6.js +179 -0
- package/dist/memory-cards-KSJF5OH2.js +179 -0
- package/dist/memory-graph-extractor-IJD5HWYT.js +21 -0
- package/dist/memory-graph-extractor-O4GAXOK5.js +21 -0
- package/dist/memory-poisoning-defense-2JRPWT5V.js +223 -0
- package/dist/memory-poisoning-defense-DH4A25NU.js +223 -0
- package/dist/memory-reflection-ISY2BBDB.js +243 -0
- package/dist/memory-reflection-Z5AQRR6H.js +243 -0
- package/dist/notifications-2VSWK2UJ.js +46 -0
- package/dist/notifications-4S253VQM.js +46 -0
- package/dist/oauth-server-D7D4574D.js +437 -0
- package/dist/oauth-server-MACN54SJ.js +437 -0
- package/dist/orchestration-events-BGP5RYQI.js +26 -0
- package/dist/orchestration-events-MDXUEVRZ.js +26 -0
- package/dist/orchestrator-DHK7RSSH.js +34 -0
- package/dist/orchestrator-R75WHQVA.js +34 -0
- package/dist/pipeline-router-4WUKQQEC.js +14 -0
- package/dist/pipeline-router-GWB2XK2Q.js +14 -0
- package/dist/plan-limits-NNJRAESF.js +27 -0
- package/dist/plan-limits-YTQW4UR4.js +27 -0
- package/dist/project-boot-46GZJTEX.js +299 -0
- package/dist/project-boot-PPHBBGIF.js +299 -0
- package/dist/projection-worker-UPAWXI7P.js +1034 -0
- package/dist/projection-worker-ZIKDYBW5.js +1034 -0
- package/dist/reranker-5ZBP2RRN.js +19 -0
- package/dist/reranker-E2MQIMJL.js +19 -0
- package/dist/reranker-GLSDJT3V.js +19 -0
- package/dist/reranker-LBBXWNOD.js +19 -0
- package/dist/reranker-XZ2EF4OH.js +19 -0
- package/dist/retrieval-health-JYRKPSII.js +7 -0
- package/dist/retrieval-health-OUV25J6S.js +7 -0
- package/dist/retrieval-health-U73JUAZL.js +7 -0
- package/dist/retrieval-health-WSZ7TYFF.js +7 -0
- package/dist/review-polling-62JV55ZT.js +125 -0
- package/dist/review-polling-CJXLWFWK.js +125 -0
- package/dist/runtime/index.js +12 -12
- package/dist/session-events-2ADD54VI.js +37 -0
- package/dist/session-events-QIJVBSKS.js +37 -0
- package/dist/session-kill-telemetry-HS6HD2YE.js +30 -0
- package/dist/session-kill-telemetry-MRT5FVSM.js +30 -0
- package/dist/session-scope-7ICYPC33.js +87 -0
- package/dist/session-scope-KMXD6EE6.js +87 -0
- package/dist/setup-wizard-B6GIT7YC.js +12 -0
- package/dist/setup-wizard-JUIJ4UZO.js +12 -0
- package/dist/skill-refinement-HIOX4VMC.js +158 -0
- package/dist/skill-refinement-T7JXRYUW.js +158 -0
- package/dist/stack-update-5KE6BZKQ.js +74 -0
- package/dist/stack-update-OP2RHP7N.js +74 -0
- package/dist/stack-update-VGCWDJEE.js +74 -0
- package/dist/steward-gate-L22WE3SY.js +14 -0
- package/dist/steward-gate-YKD2LUWN.js +14 -0
- package/dist/task-enforcement-5AOKXTY4.js +439 -0
- package/dist/task-enforcement-VO3YEGIO.js +439 -0
- package/dist/task-scope-YV2WPKRD.js +36 -0
- package/dist/task-scope-ZSXDZBRE.js +36 -0
- package/dist/tasks-crud-C6KADACT.js +78 -0
- package/dist/tasks-crud-NV6JEWGL.js +78 -0
- package/dist/tasks-notify-E22HSN6O.js +39 -0
- package/dist/tasks-notify-RPSEQ4WV.js +39 -0
- package/dist/tasks-review-V4ZLXOAZ.js +48 -0
- package/dist/tasks-review-ZVRI73JE.js +48 -0
- package/dist/telemetry-upload-LXUH7SKI.js +740 -0
- package/dist/telemetry-upload-TCDAZTUQ.js +740 -0
- package/dist/token-budget-OFBEZJTA.js +85 -0
- package/dist/token-budget-WAN57V6S.js +85 -0
- package/dist/tool-telemetry-UA3N32PK.js +17 -0
- package/dist/tool-telemetry-XXZJ35RR.js +17 -0
- package/dist/tui/App.js +17 -17
- package/dist/tui-data-46QLCJUE.js +259 -0
- package/dist/tui-data-ZDB7BLP2.js +259 -0
- package/dist/wiki-acl-HHSIBPF3.js +111 -0
- package/dist/wiki-acl-O65GZ2ZF.js +111 -0
- package/dist/worker-gate-27I4GAEZ.js +21 -0
- package/dist/worker-gate-DXU4HEPY.js +21 -0
- package/dist/workflow-engine-63EOEJ5Q.js +28 -0
- package/dist/workflow-engine-C6F2RMPN.js +28 -0
- package/dist/worktree-SFKKOMFD.js +27 -0
- package/dist/worktree-SVCE3S7X.js +27 -0
- package/dist/worktree-sweep-S3JHJTVP.js +20 -0
- package/dist/worktree-sweep-U3TIQ7WL.js +20 -0
- package/package.json +1 -1
- package/release-notes.json +88 -26
|
@@ -0,0 +1,394 @@
|
|
|
1
|
+
import {
|
|
2
|
+
readMcpHttpEvents,
|
|
3
|
+
summarizeMcpTransport
|
|
4
|
+
} from "./chunk-3CQQFYA3.js";
|
|
5
|
+
import {
|
|
6
|
+
mcpHttpPort
|
|
7
|
+
} from "./chunk-UMEIBDYW.js";
|
|
8
|
+
import {
|
|
9
|
+
EXE_AI_DIR
|
|
10
|
+
} from "./chunk-T3B5RK4H.js";
|
|
11
|
+
import {
|
|
12
|
+
isMainModule
|
|
13
|
+
} from "./chunk-6Y4B3QF6.js";
|
|
14
|
+
|
|
15
|
+
// src/bin/exe-healthcheck.ts
|
|
16
|
+
import { existsSync, readFileSync, readdirSync } from "fs";
|
|
17
|
+
import path from "path";
|
|
18
|
+
import { execSync } from "child_process";
|
|
19
|
+
import { fileURLToPath } from "url";
|
|
20
|
+
function findPackageRoot() {
|
|
21
|
+
let dir = path.dirname(fileURLToPath(import.meta.url));
|
|
22
|
+
const { root } = path.parse(dir);
|
|
23
|
+
while (dir !== root) {
|
|
24
|
+
if (existsSync(path.join(dir, "package.json"))) return dir;
|
|
25
|
+
dir = path.dirname(dir);
|
|
26
|
+
}
|
|
27
|
+
throw new Error("Cannot find package root");
|
|
28
|
+
}
|
|
29
|
+
function checkBuildIntegrity(pkgRoot) {
|
|
30
|
+
const results = [];
|
|
31
|
+
const tsupConfig = path.join(pkgRoot, "tsup.config.ts");
|
|
32
|
+
if (!existsSync(tsupConfig)) {
|
|
33
|
+
return [{ name: "build/tsup-config", pass: false, detail: "tsup.config.ts not found" }];
|
|
34
|
+
}
|
|
35
|
+
const configContent = readFileSync(tsupConfig, "utf-8");
|
|
36
|
+
const entryMatches = configContent.matchAll(/"([^"]+)":\s*"src\//g);
|
|
37
|
+
const missing = [];
|
|
38
|
+
let total = 0;
|
|
39
|
+
for (const match of entryMatches) {
|
|
40
|
+
const outputKey = match[1];
|
|
41
|
+
const expectedPath = path.join(pkgRoot, "dist", `${outputKey}.js`);
|
|
42
|
+
total++;
|
|
43
|
+
if (!existsSync(expectedPath)) {
|
|
44
|
+
missing.push(`dist/${outputKey}.js`);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
if (missing.length > 0) {
|
|
48
|
+
results.push({
|
|
49
|
+
name: "build/entry-points",
|
|
50
|
+
pass: false,
|
|
51
|
+
detail: `${missing.length}/${total} entry points missing:
|
|
52
|
+
${missing.join("\n ")}`
|
|
53
|
+
});
|
|
54
|
+
} else {
|
|
55
|
+
results.push({
|
|
56
|
+
name: "build/entry-points",
|
|
57
|
+
pass: true,
|
|
58
|
+
detail: `${total} entry points verified`
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
return results;
|
|
62
|
+
}
|
|
63
|
+
function checkEmbedPipeline(pkgRoot) {
|
|
64
|
+
const results = [];
|
|
65
|
+
const daemonPath = path.join(pkgRoot, "dist", "lib", "exe-daemon.js");
|
|
66
|
+
if (!existsSync(daemonPath)) {
|
|
67
|
+
results.push({
|
|
68
|
+
name: "exed/daemon-exists",
|
|
69
|
+
pass: false,
|
|
70
|
+
detail: `exe-daemon.js not found at ${daemonPath}`
|
|
71
|
+
});
|
|
72
|
+
return results;
|
|
73
|
+
}
|
|
74
|
+
results.push({ name: "exed/daemon-exists", pass: true, detail: "dist/lib/exe-daemon.js exists" });
|
|
75
|
+
const entryDirs = ["dist/hooks", "dist/bin", "dist/mcp"];
|
|
76
|
+
for (const dir of entryDirs) {
|
|
77
|
+
const fullDir = path.join(pkgRoot, dir);
|
|
78
|
+
if (!existsSync(fullDir)) continue;
|
|
79
|
+
let walkDir = fullDir;
|
|
80
|
+
const { root } = path.parse(walkDir);
|
|
81
|
+
let foundRoot = null;
|
|
82
|
+
while (walkDir !== root) {
|
|
83
|
+
if (existsSync(path.join(walkDir, "package.json"))) {
|
|
84
|
+
foundRoot = walkDir;
|
|
85
|
+
break;
|
|
86
|
+
}
|
|
87
|
+
walkDir = path.dirname(walkDir);
|
|
88
|
+
}
|
|
89
|
+
if (!foundRoot) {
|
|
90
|
+
results.push({
|
|
91
|
+
name: `exed/reachable-from-${dir}`,
|
|
92
|
+
pass: false,
|
|
93
|
+
detail: `Cannot find package root from ${dir}`
|
|
94
|
+
});
|
|
95
|
+
continue;
|
|
96
|
+
}
|
|
97
|
+
const resolvedDaemon = path.join(foundRoot, "dist", "lib", "exe-daemon.js");
|
|
98
|
+
const reachable = existsSync(resolvedDaemon);
|
|
99
|
+
results.push({
|
|
100
|
+
name: `exed/reachable-from-${dir}`,
|
|
101
|
+
pass: reachable,
|
|
102
|
+
detail: reachable ? `Resolves to ${resolvedDaemon}` : `NOT FOUND: ${resolvedDaemon}`
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
return results;
|
|
106
|
+
}
|
|
107
|
+
function checkTaskSystem(pkgRoot) {
|
|
108
|
+
const results = [];
|
|
109
|
+
const scannerPath = path.join(pkgRoot, "dist", "bin", "scan-tasks.js");
|
|
110
|
+
if (!existsSync(scannerPath)) {
|
|
111
|
+
results.push({ name: "tasks/scanner", pass: false, detail: "scan-tasks.js not found" });
|
|
112
|
+
return results;
|
|
113
|
+
}
|
|
114
|
+
try {
|
|
115
|
+
execSync(`node "${scannerPath}" /tmp/nonexistent-healthcheck-test --format=json 2>/dev/null`, {
|
|
116
|
+
timeout: 1e4,
|
|
117
|
+
encoding: "utf-8"
|
|
118
|
+
});
|
|
119
|
+
results.push({ name: "tasks/scanner", pass: true, detail: "scan-tasks.js runs without import errors" });
|
|
120
|
+
} catch (err) {
|
|
121
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
122
|
+
if (msg.includes("Cannot find module") || msg.includes("ERR_MODULE_NOT_FOUND") || msg.includes("SyntaxError")) {
|
|
123
|
+
results.push({ name: "tasks/scanner", pass: false, detail: `Import error: ${msg.slice(0, 200)}` });
|
|
124
|
+
} else {
|
|
125
|
+
results.push({ name: "tasks/scanner", pass: true, detail: "scan-tasks.js runs (no import errors)" });
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
return results;
|
|
129
|
+
}
|
|
130
|
+
function checkWorkerSpawning(pkgRoot) {
|
|
131
|
+
const results = [];
|
|
132
|
+
const workerPath = path.join(pkgRoot, "dist", "hooks", "ingest-worker.js");
|
|
133
|
+
if (!existsSync(workerPath)) {
|
|
134
|
+
results.push({ name: "workers/ingest-worker", pass: false, detail: "ingest-worker.js not found" });
|
|
135
|
+
return results;
|
|
136
|
+
}
|
|
137
|
+
try {
|
|
138
|
+
execSync(`node --check "${workerPath}" 2>&1`, { timeout: 1e4, encoding: "utf-8" });
|
|
139
|
+
results.push({ name: "workers/ingest-worker", pass: true, detail: "ingest-worker.js parses OK" });
|
|
140
|
+
} catch (err) {
|
|
141
|
+
results.push({
|
|
142
|
+
name: "workers/ingest-worker",
|
|
143
|
+
pass: false,
|
|
144
|
+
detail: `Parse error: ${err instanceof Error ? err.message.slice(0, 200) : String(err)}`
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
const hooksDir = path.join(pkgRoot, "dist", "hooks");
|
|
148
|
+
if (existsSync(hooksDir)) {
|
|
149
|
+
const hookFiles = readdirSync(hooksDir).filter((f) => f.endsWith(".js") && !f.endsWith(".js.map"));
|
|
150
|
+
let hooksPassed = 0;
|
|
151
|
+
const hooksFailed = [];
|
|
152
|
+
for (const hook of hookFiles) {
|
|
153
|
+
try {
|
|
154
|
+
execSync(`node --check "${path.join(hooksDir, hook)}" 2>&1`, { timeout: 1e4, encoding: "utf-8" });
|
|
155
|
+
hooksPassed++;
|
|
156
|
+
} catch {
|
|
157
|
+
hooksFailed.push(hook);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
if (hooksFailed.length > 0) {
|
|
161
|
+
results.push({
|
|
162
|
+
name: "workers/hooks-parse",
|
|
163
|
+
pass: false,
|
|
164
|
+
detail: `${hooksFailed.length}/${hookFiles.length} hooks fail to parse: ${hooksFailed.join(", ")}`
|
|
165
|
+
});
|
|
166
|
+
} else {
|
|
167
|
+
results.push({
|
|
168
|
+
name: "workers/hooks-parse",
|
|
169
|
+
pass: true,
|
|
170
|
+
detail: `${hooksPassed} hook entry points parse OK`
|
|
171
|
+
});
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
return results;
|
|
175
|
+
}
|
|
176
|
+
function checkMcpTransport() {
|
|
177
|
+
const results = [];
|
|
178
|
+
const pidPath = path.join(EXE_AI_DIR, "exed.pid");
|
|
179
|
+
const tokenPath = path.join(EXE_AI_DIR, "exed.token");
|
|
180
|
+
let daemonAlive = false;
|
|
181
|
+
if (existsSync(pidPath)) {
|
|
182
|
+
try {
|
|
183
|
+
const pid = parseInt(readFileSync(pidPath, "utf8").trim(), 10);
|
|
184
|
+
process.kill(pid, 0);
|
|
185
|
+
daemonAlive = true;
|
|
186
|
+
results.push({ name: "mcp/daemon-process", pass: true, detail: `exed process alive (pid ${pid})` });
|
|
187
|
+
} catch {
|
|
188
|
+
results.push({ name: "mcp/daemon-process", pass: false, detail: "exed.pid exists but process is not alive; restart exe-os/exed" });
|
|
189
|
+
}
|
|
190
|
+
} else {
|
|
191
|
+
results.push({ name: "mcp/daemon-process", pass: false, detail: "No exed.pid found; daemon is not running" });
|
|
192
|
+
}
|
|
193
|
+
if (daemonAlive && existsSync(tokenPath)) {
|
|
194
|
+
try {
|
|
195
|
+
const token = readFileSync(tokenPath, "utf8").trim();
|
|
196
|
+
const port = mcpHttpPort();
|
|
197
|
+
const response = execSync(
|
|
198
|
+
`curl -sS -i -m 1 -H "Authorization: Bearer ${token}" -H 'Accept: application/json, text/event-stream' http://127.0.0.1:${port}/mcp`,
|
|
199
|
+
{ encoding: "utf8", timeout: 1500 }
|
|
200
|
+
);
|
|
201
|
+
const jsonRpcError = response.includes("Content-Type: application/json") && response.includes("missing MCP session");
|
|
202
|
+
results.push({
|
|
203
|
+
name: "mcp/http-endpoint",
|
|
204
|
+
pass: jsonRpcError,
|
|
205
|
+
detail: jsonRpcError ? "MCP HTTP endpoint reachable; missing-session probe returns JSON-RPC (expected)" : "MCP HTTP endpoint reachable but did not return expected JSON-RPC missing-session response"
|
|
206
|
+
});
|
|
207
|
+
} catch (err) {
|
|
208
|
+
const msg = err instanceof Error ? err.message.slice(0, 180) : String(err);
|
|
209
|
+
const timedOut = msg.includes("ETIMEDOUT") || msg.includes("timed out") || msg.includes("killed");
|
|
210
|
+
results.push({
|
|
211
|
+
name: "mcp/http-endpoint",
|
|
212
|
+
pass: false,
|
|
213
|
+
detail: timedOut ? "Daemon alive but MCP unresponsive (timed out after 1.5s)" : `MCP HTTP endpoint not reachable: ${msg}`
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
} else if (daemonAlive) {
|
|
217
|
+
results.push({ name: "mcp/http-endpoint", pass: false, detail: "Daemon is alive but exed.token is missing; cannot probe MCP HTTP" });
|
|
218
|
+
}
|
|
219
|
+
const events = readMcpHttpEvents(200);
|
|
220
|
+
const summary = summarizeMcpTransport(events);
|
|
221
|
+
results.push({
|
|
222
|
+
name: "mcp/local-event-log",
|
|
223
|
+
pass: true,
|
|
224
|
+
detail: `${events.length} recent event(s); activeSessions=${summary.activeSessions ?? "unknown"}; lastHandshake=${summary.lastSuccessfulHandshake ?? "none"}; lastTool=${summary.lastSuccessfulToolCall ?? "none"}`
|
|
225
|
+
});
|
|
226
|
+
results.push({
|
|
227
|
+
name: "mcp/monitor-summary",
|
|
228
|
+
pass: true,
|
|
229
|
+
detail: `Privacy-safe local monitor summary: ${path.join(EXE_AI_DIR, "monitor", "mcp-transport-summary.json")}`
|
|
230
|
+
});
|
|
231
|
+
return results;
|
|
232
|
+
}
|
|
233
|
+
function checkClaudeCodeInstall() {
|
|
234
|
+
const results = [];
|
|
235
|
+
const execPath = process.env.CLAUDE_CODE_EXECPATH ?? "";
|
|
236
|
+
if (execPath.length > 0 && (execPath.includes("claude/versions/") || execPath.includes("claude.exe") || execPath.includes("claude-native"))) {
|
|
237
|
+
results.push({
|
|
238
|
+
name: "cc/execpath-clean",
|
|
239
|
+
pass: false,
|
|
240
|
+
detail: `CLAUDE_CODE_EXECPATH points to native binary: "${execPath}" \u2014 20K phantom billing risk. Unset it: unset CLAUDE_CODE_EXECPATH`
|
|
241
|
+
});
|
|
242
|
+
} else {
|
|
243
|
+
results.push({
|
|
244
|
+
name: "cc/execpath-clean",
|
|
245
|
+
pass: true,
|
|
246
|
+
detail: execPath ? `CLAUDE_CODE_EXECPATH=${execPath} (node runtime, OK)` : "CLAUDE_CODE_EXECPATH is unset"
|
|
247
|
+
});
|
|
248
|
+
}
|
|
249
|
+
try {
|
|
250
|
+
const claudePath = execSync("which claude 2>/dev/null || true", { encoding: "utf8", timeout: 5e3 }).trim();
|
|
251
|
+
if (!claudePath) {
|
|
252
|
+
results.push({
|
|
253
|
+
name: "cc/cli-binary",
|
|
254
|
+
pass: false,
|
|
255
|
+
detail: "claude not found in PATH"
|
|
256
|
+
});
|
|
257
|
+
} else {
|
|
258
|
+
let resolved = claudePath;
|
|
259
|
+
try {
|
|
260
|
+
resolved = execSync(`readlink -f "${claudePath}" 2>/dev/null || readlink "${claudePath}" 2>/dev/null || echo "${claudePath}"`, {
|
|
261
|
+
encoding: "utf8",
|
|
262
|
+
timeout: 5e3
|
|
263
|
+
}).trim();
|
|
264
|
+
} catch {
|
|
265
|
+
}
|
|
266
|
+
if (resolved.includes("bin/claude.exe") || resolved.includes("bin/claude-native")) {
|
|
267
|
+
results.push({
|
|
268
|
+
name: "cc/cli-binary",
|
|
269
|
+
pass: false,
|
|
270
|
+
detail: `claude resolves to native binary: ${resolved}. Should be cli.js. Run: npm install -g @anthropic-ai/claude-code@2.1.98`
|
|
271
|
+
});
|
|
272
|
+
} else {
|
|
273
|
+
results.push({
|
|
274
|
+
name: "cc/cli-binary",
|
|
275
|
+
pass: true,
|
|
276
|
+
detail: `claude resolves to: ${resolved}`
|
|
277
|
+
});
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
} catch {
|
|
281
|
+
results.push({
|
|
282
|
+
name: "cc/cli-binary",
|
|
283
|
+
pass: false,
|
|
284
|
+
detail: "Failed to check claude binary path"
|
|
285
|
+
});
|
|
286
|
+
}
|
|
287
|
+
const versionsDir = path.join(
|
|
288
|
+
process.env.HOME ?? process.env.USERPROFILE ?? "",
|
|
289
|
+
".local",
|
|
290
|
+
"share",
|
|
291
|
+
"claude",
|
|
292
|
+
"versions"
|
|
293
|
+
);
|
|
294
|
+
if (existsSync(versionsDir)) {
|
|
295
|
+
try {
|
|
296
|
+
const entries = readdirSync(versionsDir);
|
|
297
|
+
if (entries.length > 0) {
|
|
298
|
+
results.push({
|
|
299
|
+
name: "cc/native-cache-clean",
|
|
300
|
+
pass: false,
|
|
301
|
+
detail: `${entries.length} cached native version(s) found in ${versionsDir}: ${entries.slice(0, 3).join(", ")}${entries.length > 3 ? "..." : ""}. Remove: rm -rf "${versionsDir}"`
|
|
302
|
+
});
|
|
303
|
+
} else {
|
|
304
|
+
results.push({
|
|
305
|
+
name: "cc/native-cache-clean",
|
|
306
|
+
pass: true,
|
|
307
|
+
detail: `${versionsDir} is empty`
|
|
308
|
+
});
|
|
309
|
+
}
|
|
310
|
+
} catch {
|
|
311
|
+
results.push({
|
|
312
|
+
name: "cc/native-cache-clean",
|
|
313
|
+
pass: true,
|
|
314
|
+
detail: `${versionsDir} not readable (OK)`
|
|
315
|
+
});
|
|
316
|
+
}
|
|
317
|
+
} else {
|
|
318
|
+
results.push({
|
|
319
|
+
name: "cc/native-cache-clean",
|
|
320
|
+
pass: true,
|
|
321
|
+
detail: `${versionsDir} does not exist`
|
|
322
|
+
});
|
|
323
|
+
}
|
|
324
|
+
const disableOld = process.env.DISABLE_AUTOUPDATER === "1";
|
|
325
|
+
const disableNew = process.env.CLAUDE_CODE_AUTOUPDATER_DISABLED === "1";
|
|
326
|
+
if (!disableOld && !disableNew) {
|
|
327
|
+
results.push({
|
|
328
|
+
name: "cc/autoupdater-disabled",
|
|
329
|
+
pass: false,
|
|
330
|
+
detail: "Neither DISABLE_AUTOUPDATER=1 nor CLAUDE_CODE_AUTOUPDATER_DISABLED=1 is set. Auto-updater may install infected native binary. Export both in your shell profile."
|
|
331
|
+
});
|
|
332
|
+
} else {
|
|
333
|
+
const which = [
|
|
334
|
+
disableOld ? "DISABLE_AUTOUPDATER=1" : null,
|
|
335
|
+
disableNew ? "CLAUDE_CODE_AUTOUPDATER_DISABLED=1" : null
|
|
336
|
+
].filter(Boolean).join(" + ");
|
|
337
|
+
results.push({
|
|
338
|
+
name: "cc/autoupdater-disabled",
|
|
339
|
+
pass: true,
|
|
340
|
+
detail: `Auto-updater disabled via ${which}`
|
|
341
|
+
});
|
|
342
|
+
}
|
|
343
|
+
try {
|
|
344
|
+
const ccVersion = execSync("claude --version 2>/dev/null || echo unknown", {
|
|
345
|
+
encoding: "utf8",
|
|
346
|
+
timeout: 5e3
|
|
347
|
+
}).trim();
|
|
348
|
+
const versionMatch = ccVersion.match(/(\d+\.\d+\.\d+)/);
|
|
349
|
+
if (versionMatch) {
|
|
350
|
+
const ver = versionMatch[1];
|
|
351
|
+
const [, minor] = ver.split(".").map(Number);
|
|
352
|
+
const isRisky = minor !== void 0 && minor >= 1 && parseInt(ver.split(".")[2] ?? "0") >= 119;
|
|
353
|
+
results.push({
|
|
354
|
+
name: "cc/version",
|
|
355
|
+
pass: !isRisky,
|
|
356
|
+
detail: isRisky ? `CC version ${ver} \u2014 \u22652.1.119 ships native binary only. Pin: npm install -g @anthropic-ai/claude-code@2.1.98` : `CC version ${ver}`
|
|
357
|
+
});
|
|
358
|
+
}
|
|
359
|
+
} catch {
|
|
360
|
+
}
|
|
361
|
+
return results;
|
|
362
|
+
}
|
|
363
|
+
function runHealthCheck() {
|
|
364
|
+
const pkgRoot = findPackageRoot();
|
|
365
|
+
const results = [
|
|
366
|
+
...checkBuildIntegrity(pkgRoot),
|
|
367
|
+
...checkEmbedPipeline(pkgRoot),
|
|
368
|
+
...checkTaskSystem(pkgRoot),
|
|
369
|
+
...checkWorkerSpawning(pkgRoot),
|
|
370
|
+
...checkMcpTransport(),
|
|
371
|
+
...checkClaudeCodeInstall()
|
|
372
|
+
];
|
|
373
|
+
const passed = results.filter((r) => r.pass).length;
|
|
374
|
+
const failed = results.filter((r) => !r.pass).length;
|
|
375
|
+
return { results, passed, failed };
|
|
376
|
+
}
|
|
377
|
+
if (isMainModule(import.meta.url) && (process.argv[1] ?? "").includes("exe-healthcheck")) {
|
|
378
|
+
const { results, passed, failed } = runHealthCheck();
|
|
379
|
+
console.log("\n exe-os Health Check\n");
|
|
380
|
+
for (const r of results) {
|
|
381
|
+
const icon = r.pass ? "\x1B[32m\u2713\x1B[0m" : "\x1B[31m\u2717\x1B[0m";
|
|
382
|
+
console.log(` ${icon} ${r.name}`);
|
|
383
|
+
console.log(` ${r.detail}`);
|
|
384
|
+
}
|
|
385
|
+
console.log(`
|
|
386
|
+
${passed} passed, ${failed} failed
|
|
387
|
+
`);
|
|
388
|
+
process.exit(failed > 0 ? 1 : 0);
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
export {
|
|
392
|
+
checkClaudeCodeInstall,
|
|
393
|
+
runHealthCheck
|
|
394
|
+
};
|
|
@@ -0,0 +1,284 @@
|
|
|
1
|
+
import {
|
|
2
|
+
routeTask
|
|
3
|
+
} from "./chunk-QRPFQNI3.js";
|
|
4
|
+
import {
|
|
5
|
+
createTaskCore,
|
|
6
|
+
employeeSessionName,
|
|
7
|
+
ensureEmployee,
|
|
8
|
+
getSessionState,
|
|
9
|
+
isEmployeeAlive,
|
|
10
|
+
sessionScopeFilter,
|
|
11
|
+
updateTaskStatus
|
|
12
|
+
} from "./chunk-PC635OAG.js";
|
|
13
|
+
import {
|
|
14
|
+
DEFAULT_COORDINATOR_TEMPLATE_NAME,
|
|
15
|
+
getCoordinatorName
|
|
16
|
+
} from "./chunk-52UDAVZE.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
|
+
};
|