@askexenow/exe-os 0.9.271 → 0.9.273
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/dist/active-agent-BDYXURXQ.js +26 -0
- package/dist/active-agent-YWBGAKGU.js +25 -0
- package/dist/agentic-ontology-56VHSVS3.js +25 -0
- package/dist/backfill-metadata-A3534S32.js +597 -0
- package/dist/backfill-metadata-B6F2KJJV.js +597 -0
- package/dist/backfill-metadata-BOM2MXLI.js +597 -0
- package/dist/backfill-metadata-G46ABBVR.js +597 -0
- package/dist/backfill-metadata-TAU33HJS.js +597 -0
- package/dist/backfill-metadata-VAV27KJK.js +597 -0
- package/dist/behaviors-USUTDXVA.js +25 -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 +6 -6
- package/dist/bin/cc-doctor.js +4 -4
- package/dist/bin/cleanup-stale-review-tasks.js +10 -10
- package/dist/bin/cli.js +16 -16
- package/dist/bin/exe-agent-config.js +3 -3
- 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 +4 -4
- 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 +37 -19
- package/dist/bin/exe-new-employee.js +6 -6
- package/dist/bin/exe-pending-messages.js +11 -11
- package/dist/bin/exe-pending-notifications.js +33 -18
- 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-team.js +3 -3
- 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 +6 -6
- 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/capacity-monitor-2GJOFXGB.js +49 -0
- package/dist/capacity-monitor-3Z7W4K25.js +49 -0
- package/dist/capacity-monitor-BENS3N7B.js +49 -0
- package/dist/capacity-monitor-IFVRCIM7.js +49 -0
- package/dist/capacity-monitor-MQUUEZKB.js +49 -0
- package/dist/capacity-monitor-Q47GBDSY.js +49 -0
- package/dist/catchup-brief-B4KGAIPU.js +151 -0
- package/dist/catchup-brief-NMOV3SSP.js +151 -0
- package/dist/catchup-brief-RP4QHXNT.js +151 -0
- package/dist/catchup-brief-TKA6TEK4.js +151 -0
- package/dist/catchup-brief-VMF3ESTZ.js +151 -0
- package/dist/catchup-brief-ZL7V3BXC.js +151 -0
- package/dist/chunk-23KJ2LXY.js +58 -0
- package/dist/chunk-2KWVJV6I.js +171 -0
- package/dist/chunk-2NQQP3FF.js +630 -0
- package/dist/chunk-3A4SOC66.js +551 -0
- package/dist/chunk-3FU5I3KV.js +526 -0
- package/dist/chunk-3GSGDPLK.js +171 -0
- package/dist/chunk-3IM3JNQV.js +377 -0
- package/dist/chunk-3OM3V545.js +448 -0
- package/dist/chunk-3T27ZQT6.js +495 -0
- package/dist/chunk-3VI3QIHU.js +214 -0
- package/dist/chunk-3WG3RRWA.js +1345 -0
- package/dist/chunk-42A3JV3A.js +128 -0
- package/dist/chunk-46IEEKPU.js +13696 -0
- package/dist/chunk-46WLFLGP.js +1073 -0
- package/dist/chunk-4JMPQB7K.js +1148 -0
- package/dist/chunk-4L25LLQM.js +382 -0
- package/dist/chunk-4Q7X3SAM.js +204 -0
- package/dist/chunk-4UAUCFHA.js +526 -0
- package/dist/chunk-4VRJX2SP.js +495 -0
- package/dist/chunk-57RBAR2A.js +214 -0
- package/dist/chunk-57UAFTO2.js +3958 -0
- package/dist/chunk-5AS622MM.js +3958 -0
- package/dist/chunk-5JF5OQQU.js +89 -0
- package/dist/chunk-5LTY4GLX.js +13745 -0
- package/dist/chunk-5M5RYJ22.js +3955 -0
- package/dist/chunk-5YO2FER3.js +76 -0
- package/dist/chunk-62DEE65H.js +371 -0
- package/dist/chunk-62YI2JOC.js +333 -0
- package/dist/chunk-64T6DFSS.js +447 -0
- package/dist/chunk-6AGPWYFC.js +447 -0
- package/dist/chunk-6BWDP63Z.js +197 -0
- package/dist/chunk-6CH7TYBG.js +58 -0
- package/dist/chunk-6CHHFVRQ.js +284 -0
- package/dist/chunk-6D64562N.js +330 -0
- package/dist/chunk-6F35WOSR.js +447 -0
- package/dist/chunk-6GPYL7TX.js +214 -0
- package/dist/chunk-6HQ22FC6.js +81 -0
- package/dist/chunk-6KWLUVFL.js +54 -0
- package/dist/chunk-6N5ISWBF.js +1148 -0
- package/dist/chunk-6OD7PVMC.js +333 -0
- package/dist/chunk-6ZSH2BZR.js +244 -0
- package/dist/chunk-74MF4T3T.js +3962 -0
- package/dist/chunk-77H7IO3O.js +382 -0
- package/dist/chunk-7BUWNG6M.js +159 -0
- package/dist/chunk-7ET5CYTD.js +382 -0
- package/dist/chunk-7IWLKR6N.js +76 -0
- package/dist/chunk-7OEUOJL5.js +1021 -0
- package/dist/chunk-7YEQI2WF.js +13745 -0
- package/dist/chunk-AIRJTKDK.js +204 -0
- package/dist/chunk-AJ63GPM7.js +54 -0
- package/dist/chunk-ATJ3NXDP.js +244 -0
- package/dist/chunk-B5IS7LE4.js +128 -0
- package/dist/chunk-BFJ45HQT.js +244 -0
- package/dist/chunk-BKINEQVI.js +244 -0
- package/dist/chunk-BMHE3UQU.js +495 -0
- package/dist/chunk-BNTUZVPS.js +1921 -0
- package/dist/chunk-BOJV6NI3.js +128 -0
- package/dist/chunk-BPHWI6N2.js +284 -0
- package/dist/chunk-BXCQWWJP.js +185 -0
- package/dist/chunk-BZ6K7AY3.js +50 -0
- package/dist/chunk-C54KIFLS.js +214 -0
- package/dist/chunk-C6ODVGTC.js +818 -0
- package/dist/chunk-C6OYEJJI.js +260 -0
- package/dist/chunk-CHBGCQXG.js +333 -0
- package/dist/chunk-CHBHR5W6.js +3556 -0
- package/dist/chunk-CHUOANKE.js +346 -0
- package/dist/chunk-CSF4RUCN.js +58 -0
- package/dist/chunk-CXKHWCNN.js +204 -0
- package/dist/chunk-CZR6Z5D7.js +330 -0
- package/dist/chunk-D24ANCWY.js +204 -0
- package/dist/chunk-D2T3272U.js +171 -0
- package/dist/chunk-DBJCWK6T.js +377 -0
- package/dist/chunk-DCHEIVGT.js +221 -0
- package/dist/chunk-DF4SM6ZX.js +128 -0
- package/dist/chunk-DHIBLMSP.js +30 -0
- package/dist/chunk-DOAC6CLC.js +127 -0
- package/dist/chunk-DOGNJ4VR.js +818 -0
- package/dist/chunk-DYXJFUCI.js +818 -0
- package/dist/chunk-E2AF2WYY.js +346 -0
- package/dist/chunk-E2KZEZZW.js +1090 -0
- package/dist/chunk-E4KWB4WM.js +348 -0
- package/dist/chunk-EGR2NYID.js +50 -0
- package/dist/chunk-ENU7URWK.js +1073 -0
- package/dist/chunk-EQ5UBJGX.js +81 -0
- package/dist/chunk-EZ7KAZMC.js +132 -0
- package/dist/chunk-F4FSSHR4.js +1073 -0
- package/dist/chunk-FBRQGHSU.js +377 -0
- package/dist/chunk-FPXU56FG.js +346 -0
- package/dist/chunk-FY7HHR5I.js +128 -0
- package/dist/chunk-FZ42OCSP.js +333 -0
- package/dist/chunk-G2S2UMU4.js +159 -0
- package/dist/chunk-G33BHQCO.js +70 -0
- package/dist/chunk-G5HWDSBH.js +50 -0
- package/dist/chunk-GCBG5TFS.js +1345 -0
- package/dist/chunk-GESN6IDC.js +127 -0
- package/dist/chunk-GHD7QG6P.js +58 -0
- package/dist/chunk-GJAILPCX.js +171 -0
- package/dist/chunk-GJQTL7RX.js +127 -0
- package/dist/chunk-GKUODJS7.js +214 -0
- package/dist/chunk-GLCKDEM2.js +97 -0
- package/dist/chunk-GLDM2FOM.js +76 -0
- package/dist/chunk-GMA34SXV.js +240 -0
- package/dist/chunk-GMM2BLFB.js +127 -0
- package/dist/chunk-GNM75IOI.js +159 -0
- package/dist/chunk-GVAVEBYR.js +2091 -0
- package/dist/chunk-GYIX2HLD.js +81 -0
- package/dist/chunk-H4LLEQ3F.js +551 -0
- package/dist/chunk-HBVCBBDA.js +127 -0
- package/dist/chunk-HBYRWOH5.js +171 -0
- package/dist/chunk-HFINM2JG.js +284 -0
- package/dist/chunk-HJGHALOG.js +1345 -0
- package/dist/chunk-HOSJTLBQ.js +513 -0
- package/dist/chunk-HRB5CP43.js +13745 -0
- package/dist/chunk-IC6HVAS3.js +56 -0
- package/dist/chunk-IDCLPPIM.js +3959 -0
- package/dist/chunk-IDFJNO44.js +1051 -0
- package/dist/chunk-II5SVNBN.js +551 -0
- package/dist/chunk-IIRLKWNZ.js +50 -0
- package/dist/chunk-ILFJMEY5.js +97 -0
- package/dist/chunk-IQXLUTWC.js +50 -0
- package/dist/chunk-ISQAOSL3.js +1921 -0
- package/dist/chunk-IWXTFDLS.js +244 -0
- package/dist/chunk-J2TGVCPE.js +1090 -0
- package/dist/chunk-J4Z5GAJ4.js +551 -0
- package/dist/chunk-J6V2DCZK.js +382 -0
- package/dist/chunk-JJSDZFKM.js +1148 -0
- package/dist/chunk-JMN2KOC4.js +128 -0
- package/dist/chunk-JP4CLFLR.js +1148 -0
- package/dist/chunk-JQVYPBR2.js +81 -0
- package/dist/chunk-JTIOZHWG.js +58 -0
- package/dist/chunk-KDICWAYV.js +1345 -0
- package/dist/chunk-KMU7PFO3.js +1148 -0
- package/dist/chunk-KOBIB6WG.js +159 -0
- package/dist/chunk-KQFDDQB6.js +13696 -0
- package/dist/chunk-KZNSOHCB.js +280 -0
- package/dist/chunk-LDDCAATQ.js +1090 -0
- package/dist/chunk-LJN2O5IG.js +197 -0
- package/dist/chunk-LSIYHKDS.js +54 -0
- package/dist/chunk-LVMBYP3C.js +171 -0
- package/dist/chunk-M2WQW5NC.js +227 -0
- package/dist/chunk-MREDKOS4.js +731 -0
- package/dist/chunk-MSF2Y5MS.js +346 -0
- package/dist/chunk-MY647ZHR.js +448 -0
- package/dist/chunk-MY6SP5NZ.js +551 -0
- package/dist/chunk-MZ5CEHPQ.js +89 -0
- package/dist/chunk-N2ACW2ZG.js +363 -0
- package/dist/chunk-NESTX6DR.js +76 -0
- package/dist/chunk-NQZORF6L.js +731 -0
- package/dist/chunk-NSMJDATI.js +495 -0
- package/dist/chunk-NSQ5JE23.js +1090 -0
- package/dist/chunk-NWEFAFJS.js +197 -0
- package/dist/chunk-NYF7GHC5.js +526 -0
- package/dist/chunk-NZGGRM4P.js +731 -0
- package/dist/chunk-NZL567WG.js +81 -0
- package/dist/chunk-NZM4E6Y3.js +89 -0
- package/dist/chunk-O5OMH6LI.js +244 -0
- package/dist/chunk-O6XF6NUN.js +1090 -0
- package/dist/chunk-OBUV3W7L.js +163 -0
- package/dist/chunk-OF4KG3L7.js +1090 -0
- package/dist/chunk-OLDS7LJN.js +495 -0
- package/dist/chunk-OO2I22RX.js +38 -0
- package/dist/chunk-OPUUT33V.js +447 -0
- package/dist/chunk-OQZPSWVN.js +526 -0
- package/dist/chunk-OR6KJ5HH.js +58 -0
- package/dist/chunk-OT3VMTKB.js +50 -0
- package/dist/chunk-OV6NT6QX.js +128 -0
- package/dist/chunk-P6RVIOVA.js +157 -0
- package/dist/chunk-PDTR3YUU.js +54 -0
- package/dist/chunk-PEGTV6EJ.js +1345 -0
- package/dist/chunk-PITVTSQW.js +333 -0
- package/dist/chunk-PSUAO4MZ.js +345 -0
- package/dist/chunk-PUA5564C.js +210 -0
- package/dist/chunk-PUQLKLQX.js +731 -0
- package/dist/chunk-PWQIS5E5.js +382 -0
- package/dist/chunk-PXXHKWDH.js +818 -0
- package/dist/chunk-QOC46BDY.js +346 -0
- package/dist/chunk-QROKS65G.js +76 -0
- package/dist/chunk-R54I2N2T.js +818 -0
- package/dist/chunk-RCFYQHUP.js +818 -0
- package/dist/chunk-RJTND4YS.js +284 -0
- package/dist/chunk-RTA6KSSK.js +89 -0
- package/dist/chunk-SBLHQMMZ.js +81 -0
- package/dist/chunk-SBX6HSEO.js +159 -0
- package/dist/chunk-SEUST6U5.js +284 -0
- package/dist/chunk-SG2ANG5C.js +123 -0
- package/dist/chunk-SUEQF3ZS.js +214 -0
- package/dist/chunk-SVFNKSZV.js +333 -0
- package/dist/chunk-SWNAM2NW.js +526 -0
- package/dist/chunk-TAQT2DC7.js +330 -0
- package/dist/chunk-TB7HFW7M.js +127 -0
- package/dist/chunk-TBJP46RP.js +1148 -0
- package/dist/chunk-TS7NGPU4.js +1073 -0
- package/dist/chunk-TUPDOPMG.js +731 -0
- package/dist/chunk-TYKUZVCA.js +1921 -0
- package/dist/chunk-TZMXJVZV.js +345 -0
- package/dist/chunk-U2DCN7M6.js +1073 -0
- package/dist/chunk-UJZPLZLU.js +197 -0
- package/dist/chunk-UKRKOJQZ.js +54 -0
- package/dist/chunk-UUKDAIH2.js +731 -0
- package/dist/chunk-V6VEFEEH.js +1345 -0
- package/dist/chunk-VCVGE7HK.js +1921 -0
- package/dist/chunk-VIO2ALGH.js +290 -0
- package/dist/chunk-VK6YZ6K7.js +1073 -0
- package/dist/chunk-VKCUSNJW.js +377 -0
- package/dist/chunk-VKT4N6WM.js +495 -0
- package/dist/chunk-VQUEP7UA.js +244 -0
- package/dist/chunk-VRPPJFIQ.js +1921 -0
- package/dist/chunk-VXODHQXB.js +377 -0
- package/dist/chunk-WHK7GXFR.js +13745 -0
- package/dist/chunk-WP3PVBBP.js +204 -0
- package/dist/chunk-WQEUY7DC.js +129 -0
- package/dist/chunk-WWPJTPPQ.js +197 -0
- package/dist/chunk-WXMXUKCA.js +262 -0
- package/dist/chunk-WYZSWV6A.js +346 -0
- package/dist/chunk-X2WBH2IO.js +297 -0
- package/dist/chunk-X33TSJNO.js +394 -0
- package/dist/chunk-X7MMI2UI.js +89 -0
- package/dist/chunk-XD6VOXK3.js +159 -0
- package/dist/chunk-XG3BQZIK.js +85 -0
- package/dist/chunk-XIKBIAOS.js +75 -0
- package/dist/chunk-XPEB545Q.js +54 -0
- package/dist/chunk-XWH2MLWS.js +330 -0
- package/dist/chunk-YH7V73XW.js +89 -0
- package/dist/chunk-YHSATGMH.js +3955 -0
- package/dist/chunk-YJBCGD46.js +13745 -0
- package/dist/chunk-YLOJPYCJ.js +284 -0
- package/dist/chunk-YMLM5D65.js +135 -0
- package/dist/chunk-YNJPRQ6J.js +377 -0
- package/dist/chunk-YSNEHBI6.js +551 -0
- package/dist/chunk-Z33XSFND.js +76 -0
- package/dist/chunk-ZA7N3ZTA.js +1921 -0
- package/dist/chunk-ZD6BMW2K.js +33 -0
- package/dist/chunk-ZFRG2MNB.js +382 -0
- package/dist/chunk-ZKG5IYCG.js +668 -0
- package/dist/chunk-ZRRRSVQF.js +204 -0
- package/dist/chunk-ZU4K7ZNX.js +197 -0
- package/dist/co-activation-HZMJC34P.js +72 -0
- package/dist/co-occurrence-AVYXRV4L.js +74 -0
- package/dist/core-memory-554Q3YN5.js +110 -0
- package/dist/core-memory-BC4YN5F4.js +110 -0
- package/dist/core-memory-NID6R3YR.js +110 -0
- package/dist/core-memory-NPJCVUMF.js +110 -0
- package/dist/core-memory-OKGXL33Z.js +110 -0
- package/dist/core-memory-XHIC5NAB.js +110 -0
- package/dist/crdt-sync-ZCH55JNR.js +33 -0
- package/dist/crm-webhook-6OMVUUGR.js +10 -0
- package/dist/crm-webhook-MHZTXU5N.js +10 -0
- package/dist/crm-webhook-TMWJT2Z5.js +10 -0
- package/dist/crm-webhook-UCWF3XDB.js +10 -0
- package/dist/crm-webhook-XISULXI7.js +10 -0
- package/dist/crm-webhook-YJ5A7F2E.js +10 -0
- package/dist/cto-delegation-gate-4PMJZL2T.js +206 -0
- package/dist/cto-delegation-gate-A7YKXTRO.js +206 -0
- package/dist/cto-delegation-gate-JFZFZGC2.js +206 -0
- package/dist/cto-delegation-gate-K32M4GVM.js +206 -0
- package/dist/cto-delegation-gate-OREBAHUM.js +206 -0
- package/dist/cto-delegation-gate-POHESML5.js +206 -0
- package/dist/daemon-orchestration-2Q7BYOHC.js +135 -0
- package/dist/daemon-orchestration-4RJ2CZJL.js +135 -0
- package/dist/daemon-orchestration-HXYPHSYU.js +135 -0
- package/dist/daemon-orchestration-I5BE46P3.js +135 -0
- package/dist/daemon-orchestration-NKE4FYQS.js +135 -0
- package/dist/daemon-orchestration-YWEXRAZA.js +135 -0
- package/dist/db-backup-5GA2YFDX.js +33 -0
- package/dist/dreaming-3F72ROTL.js +32 -0
- package/dist/dreaming-I6KXO6E2.js +32 -0
- package/dist/dreaming-JD7MNJGS.js +32 -0
- package/dist/dreaming-LCKPA3B4.js +32 -0
- package/dist/dreaming-NJBK5ILR.js +32 -0
- package/dist/dreaming-SDS5IQYC.js +32 -0
- package/dist/exe-drift-VSMIMHL4.js +68 -0
- package/dist/exe-export-DVHHIA6Y.js +73 -0
- package/dist/exe-export-GIVQDENS.js +73 -0
- package/dist/exe-export-IZ2OYMT4.js +73 -0
- package/dist/exe-export-JNWX6ZCQ.js +73 -0
- package/dist/exe-export-OQXCJLWB.js +73 -0
- package/dist/exe-export-YLVAZQAV.js +73 -0
- package/dist/exe-import-7N46LSMQ.js +76 -0
- package/dist/exe-import-AEJYBLA7.js +76 -0
- package/dist/exe-import-FINYUV5T.js +76 -0
- package/dist/exe-import-HWPYARCG.js +76 -0
- package/dist/exe-import-K4TWTG24.js +76 -0
- package/dist/exe-import-LZKZQ54C.js +76 -0
- package/dist/exe-key-6FPQHBW6.js +579 -0
- package/dist/exe-key-H45JY44F.js +579 -0
- package/dist/exe-key-MAEQGTB7.js +579 -0
- package/dist/exe-key-N3XYSEXP.js +579 -0
- package/dist/exe-key-Q3ZNYT6L.js +579 -0
- package/dist/exe-key-Q47RPB45.js +579 -0
- package/dist/exe-snapshot-2USE2HHM.js +164 -0
- package/dist/exe-snapshot-3TEM3BFD.js +164 -0
- package/dist/exe-snapshot-HECGUHL3.js +164 -0
- package/dist/exe-snapshot-HZU66HXX.js +164 -0
- package/dist/exe-snapshot-L7OQWZUH.js +164 -0
- package/dist/exe-snapshot-X5N5KIVJ.js +164 -0
- package/dist/fast-db-init-3CNTADVO.js +7 -0
- package/dist/fast-db-init-C6IPNVPU.js +7 -0
- package/dist/fast-db-init-HXCS2AP5.js +7 -0
- package/dist/fast-db-init-I7CMGBAN.js +7 -0
- package/dist/fast-db-init-P6YESOUL.js +7 -0
- package/dist/fast-db-init-VDNEFVQF.js +7 -0
- package/dist/gateway/index.js +8 -8
- package/dist/git-staleness-YCEBBIVK.js +110 -0
- package/dist/git-task-sweep-C4OV2CEY.js +40 -0
- package/dist/git-task-sweep-H34STRNT.js +40 -0
- package/dist/git-task-sweep-J66SYJMW.js +40 -0
- package/dist/git-task-sweep-JYCD3ZKQ.js +40 -0
- package/dist/git-task-sweep-O723DB7F.js +40 -0
- package/dist/git-task-sweep-YL7NLDCK.js +40 -0
- package/dist/global-procedures-IHZM6C2K.js +20 -0
- package/dist/graph-auto-extract-RZQ3MHP2.js +162 -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 +21 -21
- 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-4EW5ZDGD.js +296 -0
- package/dist/installer-B2JTQO55.js +38 -0
- package/dist/installer-MIL352T7.js +342 -0
- package/dist/lib/agent-config.js +9 -3
- package/dist/lib/cloud-sync.js +4 -4
- 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 +35 -34
- package/dist/lib/hybrid-search.js +5 -5
- package/dist/lib/identity.js +2 -2
- package/dist/lib/messaging.js +12 -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 +4 -4
- 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/mcp/register-tools.js +54 -54
- package/dist/mcp/server.js +55 -55
- 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 +5 -5
- 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-http-config-OJQR246S.js +27 -0
- package/dist/memory-cards-IPULSQFA.js +174 -0
- package/dist/memory-graph-extractor-3TZZOKHY.js +17 -0
- package/dist/memory-poisoning-defense-SGUGR5YJ.js +225 -0
- package/dist/memory-reflection-H3WGCEM6.js +238 -0
- package/dist/notifications-65STXW6N.js +45 -0
- package/dist/notifications-K3JDUPL5.js +45 -0
- package/dist/notifications-KQOD66ZK.js +45 -0
- package/dist/notifications-PFK5OQEF.js +45 -0
- package/dist/notifications-VWPO6NJF.js +45 -0
- package/dist/notifications-WCSRQN2V.js +45 -0
- package/dist/orchestration-events-O5PSDEIO.js +25 -0
- package/dist/orchestrator-3D7QEVGP.js +33 -0
- package/dist/orchestrator-CC32RZO5.js +33 -0
- package/dist/orchestrator-DWAYSAFR.js +33 -0
- package/dist/orchestrator-RAPEJUOI.js +33 -0
- package/dist/orchestrator-TL37EAWA.js +33 -0
- package/dist/orchestrator-XPG6LJAI.js +33 -0
- package/dist/pipeline-router-5NT6FUC3.js +13 -0
- package/dist/pipeline-router-ADLTS6DZ.js +13 -0
- package/dist/pipeline-router-KSUXONDT.js +13 -0
- package/dist/pipeline-router-O6ZLSM6U.js +13 -0
- package/dist/pipeline-router-QKLYUYU7.js +13 -0
- package/dist/pipeline-router-W2W5XDND.js +13 -0
- package/dist/plan-limits-53NXLNDQ.js +26 -0
- package/dist/project-boot-ITN3FZMM.js +299 -0
- package/dist/projection-worker-27XX5M2W.js +964 -0
- package/dist/reranker-3KLYAHO4.js +19 -0
- package/dist/reranker-64KDRYPP.js +19 -0
- package/dist/reranker-GU7L2PJX.js +19 -0
- package/dist/reranker-MGY5A7BQ.js +19 -0
- package/dist/reranker-TZEXIJAN.js +19 -0
- package/dist/reranker-ZBX6HSU2.js +19 -0
- package/dist/review-polling-3ZZ2T26N.js +124 -0
- package/dist/review-polling-BBQUF54Q.js +124 -0
- package/dist/review-polling-FA2J2Q5O.js +124 -0
- package/dist/review-polling-MLS4BQ3N.js +124 -0
- package/dist/review-polling-RXQZPGRY.js +124 -0
- package/dist/review-polling-YBB6DKA5.js +124 -0
- package/dist/runtime/index.js +12 -12
- package/dist/session-events-EAODNMNR.js +36 -0
- package/dist/session-events-MVO6JNUL.js +36 -0
- package/dist/session-events-PRVDH3QS.js +36 -0
- package/dist/session-events-PU5OQKMB.js +36 -0
- package/dist/session-events-WWGF3B2N.js +36 -0
- package/dist/session-events-ZHXXAH6B.js +36 -0
- package/dist/session-kill-telemetry-O4TJHHOZ.js +29 -0
- package/dist/session-scope-CQXB7VMH.js +86 -0
- package/dist/session-scope-HHUMJYF6.js +86 -0
- package/dist/session-scope-M47JR2SD.js +86 -0
- package/dist/session-scope-MRQYSD5S.js +86 -0
- package/dist/session-scope-TAH5BUYW.js +86 -0
- package/dist/session-scope-UXZ6RUWC.js +86 -0
- package/dist/setup-wizard-UM2RHSBJ.js +12 -0
- package/dist/skill-refinement-447DZWNK.js +157 -0
- package/dist/skill-refinement-567JSF7L.js +157 -0
- package/dist/skill-refinement-6JVQ3TMS.js +157 -0
- package/dist/skill-refinement-MJPOHYD5.js +157 -0
- package/dist/skill-refinement-NVUBRK22.js +157 -0
- package/dist/skill-refinement-XNGD3C62.js +157 -0
- package/dist/stack-release-BAPCXMXW.js +713 -0
- package/dist/stack-release-W4TWTEZP.js +731 -0
- package/dist/steward-gate-VLE7OCKO.js +13 -0
- package/dist/task-enforcement-AZEO67N6.js +391 -0
- package/dist/task-enforcement-EOYP6IO4.js +391 -0
- package/dist/task-enforcement-FUHDL6UR.js +391 -0
- package/dist/task-enforcement-L5XQKFOV.js +391 -0
- package/dist/task-enforcement-QL3K4N3F.js +391 -0
- package/dist/task-enforcement-RJPWWEAE.js +391 -0
- package/dist/task-scope-DRQRNWB7.js +35 -0
- package/dist/task-scope-GS7TS3UV.js +35 -0
- package/dist/task-scope-KQNCP42W.js +35 -0
- package/dist/task-scope-SM5F6RD3.js +35 -0
- package/dist/task-scope-TZYMB634.js +35 -0
- package/dist/task-scope-ZVLUBS4C.js +35 -0
- package/dist/tasks-crud-4MSLJWXE.js +77 -0
- package/dist/tasks-crud-6TWWETGB.js +77 -0
- package/dist/tasks-crud-DBHYO4MM.js +77 -0
- package/dist/tasks-crud-EFYWPPEI.js +77 -0
- package/dist/tasks-crud-HIPXKRKX.js +77 -0
- package/dist/tasks-crud-JIS5B4GZ.js +77 -0
- package/dist/tasks-notify-7JBUNE7R.js +38 -0
- package/dist/tasks-notify-GEJKT5TO.js +38 -0
- package/dist/tasks-notify-OW3JDPLK.js +38 -0
- package/dist/tasks-notify-UPIJ3L4O.js +38 -0
- package/dist/tasks-notify-W5WVP2FG.js +38 -0
- package/dist/tasks-notify-YKEOYOKN.js +38 -0
- package/dist/tasks-review-5SJSFTUB.js +47 -0
- package/dist/tasks-review-DRKN34HO.js +47 -0
- package/dist/tasks-review-IQSAXXXE.js +47 -0
- package/dist/tasks-review-JHSYBR5I.js +47 -0
- package/dist/tasks-review-KWELLLS3.js +47 -0
- package/dist/tasks-review-SUJ6AKAS.js +47 -0
- package/dist/telemetry-upload-BSGOXGUP.js +739 -0
- package/dist/telemetry-upload-FPQAB6ZU.js +739 -0
- package/dist/telemetry-upload-LTX3C5HZ.js +739 -0
- package/dist/telemetry-upload-MYVBVUGE.js +739 -0
- package/dist/telemetry-upload-UPAABLGK.js +739 -0
- package/dist/telemetry-upload-UYEHBFGO.js +739 -0
- package/dist/token-budget-2CDWQU3Q.js +84 -0
- package/dist/tool-telemetry-7YS7EN7B.js +17 -0
- package/dist/tui/App.js +17 -17
- package/dist/tui-data-GDGBOS6G.js +258 -0
- package/dist/tui-data-LYUZFNO4.js +258 -0
- package/dist/tui-data-QM5BOKRF.js +258 -0
- package/dist/tui-data-VAE43SM3.js +258 -0
- package/dist/tui-data-VXF2RBVM.js +258 -0
- package/dist/tui-data-X7HT3FXF.js +258 -0
- package/dist/wiki-acl-MJIMXRQV.js +111 -0
- package/dist/worker-gate-BZBWTMCY.js +21 -0
- package/dist/worker-gate-CHVL6UGT.js +21 -0
- package/dist/worker-gate-KQFS4RJE.js +21 -0
- package/dist/worker-gate-NRP7CMS7.js +21 -0
- package/dist/worker-gate-WQGTZOSM.js +21 -0
- package/dist/worker-gate-X2YDTKTL.js +21 -0
- package/dist/workflow-engine-AKKOMJJQ.js +28 -0
- package/dist/workflow-engine-CYXRZXBM.js +28 -0
- package/dist/workflow-engine-EHWQO3LX.js +28 -0
- package/dist/workflow-engine-I3OUMSF4.js +28 -0
- package/dist/workflow-engine-KMLAXVA4.js +28 -0
- package/dist/workflow-engine-PHTLEAXP.js +28 -0
- package/dist/worktree-NLSKVRNC.js +26 -0
- package/dist/worktree-sweep-44TMEPLE.js +19 -0
- package/package.json +1 -1
- package/release-notes.json +46 -35
|
@@ -0,0 +1,447 @@
|
|
|
1
|
+
import {
|
|
2
|
+
dispatchTaskToEmployee,
|
|
3
|
+
markTaskNotificationsRead,
|
|
4
|
+
notifyTaskDone
|
|
5
|
+
} from "./chunk-6ZSH2BZR.js";
|
|
6
|
+
import {
|
|
7
|
+
cleanupReviewFile,
|
|
8
|
+
createTaskCore,
|
|
9
|
+
deleteTaskCore,
|
|
10
|
+
markAsReadByTaskFile,
|
|
11
|
+
sessionScopeFilter,
|
|
12
|
+
updateTaskStatus,
|
|
13
|
+
writeNotification
|
|
14
|
+
} from "./chunk-74MF4T3T.js";
|
|
15
|
+
import {
|
|
16
|
+
orgBus
|
|
17
|
+
} from "./chunk-MP2AFCGL.js";
|
|
18
|
+
import {
|
|
19
|
+
getClient,
|
|
20
|
+
getCoordinatorName,
|
|
21
|
+
isCoordinatorName
|
|
22
|
+
} from "./chunk-CHBHR5W6.js";
|
|
23
|
+
import {
|
|
24
|
+
EXE_AI_DIR
|
|
25
|
+
} from "./chunk-VXIMSRTO.js";
|
|
26
|
+
|
|
27
|
+
// src/lib/tasks.ts
|
|
28
|
+
import path2 from "path";
|
|
29
|
+
import { writeFileSync, mkdirSync, unlinkSync } from "fs";
|
|
30
|
+
|
|
31
|
+
// src/lib/tasks-chain.ts
|
|
32
|
+
import path from "path";
|
|
33
|
+
import { readFile, writeFile } from "fs/promises";
|
|
34
|
+
async function cascadeUnblock(taskId, baseDir, now) {
|
|
35
|
+
const client = getClient();
|
|
36
|
+
const ubScope = sessionScopeFilter();
|
|
37
|
+
const blockedRows = await client.execute({
|
|
38
|
+
sql: `SELECT id, title, assigned_to, priority, task_file, project_name FROM tasks WHERE blocked_by = ? AND status = 'blocked'${ubScope.sql}`,
|
|
39
|
+
args: [taskId, ...ubScope.args]
|
|
40
|
+
});
|
|
41
|
+
if (blockedRows.rows.length === 0) return;
|
|
42
|
+
const ids = blockedRows.rows.map((r) => String(r.id));
|
|
43
|
+
const ph = ids.map(() => "?").join(",");
|
|
44
|
+
await client.execute({
|
|
45
|
+
sql: `UPDATE tasks SET status = 'open', blocked_by = NULL, updated_at = ?
|
|
46
|
+
WHERE id IN (${ph})`,
|
|
47
|
+
args: [now, ...ids]
|
|
48
|
+
});
|
|
49
|
+
const unblockedRows = blockedRows;
|
|
50
|
+
if (baseDir) {
|
|
51
|
+
for (const ur of unblockedRows.rows) {
|
|
52
|
+
try {
|
|
53
|
+
const ubFile = path.join(baseDir, String(ur.task_file));
|
|
54
|
+
let ubContent = await readFile(ubFile, "utf-8");
|
|
55
|
+
ubContent = ubContent.replace(/\*\*Status:\*\* blocked/, "**Status:** open");
|
|
56
|
+
ubContent = ubContent.replace(/\n\*\*Blocked by:\*\*.*\n/, "\n");
|
|
57
|
+
await writeFile(ubFile, ubContent, "utf-8");
|
|
58
|
+
} catch {
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
if (unblockedRows.rows.length > 0 && !process.env.VITEST) {
|
|
63
|
+
try {
|
|
64
|
+
const { dispatchTaskToEmployee: dispatchTaskToEmployee2 } = await import("./tasks-notify-GEJKT5TO.js");
|
|
65
|
+
const dispatched = /* @__PURE__ */ new Set();
|
|
66
|
+
for (const ur of unblockedRows.rows) {
|
|
67
|
+
const assignee = String(ur.assigned_to);
|
|
68
|
+
if (dispatched.has(assignee)) continue;
|
|
69
|
+
dispatched.add(assignee);
|
|
70
|
+
const unblockedId = String(ur.id);
|
|
71
|
+
const title = String(ur.title);
|
|
72
|
+
const priority = String(ur.priority ?? "P2");
|
|
73
|
+
const taskFile = String(ur.task_file ?? "");
|
|
74
|
+
const projectName = ur.project_name != null ? String(ur.project_name) : void 0;
|
|
75
|
+
dispatchTaskToEmployee2({
|
|
76
|
+
assignedTo: assignee,
|
|
77
|
+
taskId: unblockedId,
|
|
78
|
+
title,
|
|
79
|
+
priority,
|
|
80
|
+
taskFile,
|
|
81
|
+
initialStatus: "open",
|
|
82
|
+
projectDir: baseDir,
|
|
83
|
+
projectName
|
|
84
|
+
}).catch(() => {
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
} catch {
|
|
88
|
+
try {
|
|
89
|
+
const { queueIntercom } = await import("./intercom-queue-A6UJEFIF.js");
|
|
90
|
+
const dispatched = /* @__PURE__ */ new Set();
|
|
91
|
+
for (const ur of unblockedRows.rows) {
|
|
92
|
+
const assignee = String(ur.assigned_to);
|
|
93
|
+
if (dispatched.has(assignee)) continue;
|
|
94
|
+
dispatched.add(assignee);
|
|
95
|
+
queueIntercom(`${assignee}`, `unblocked: "${String(ur.title)}" is now ready`);
|
|
96
|
+
}
|
|
97
|
+
} catch {
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
async function findNextTask(assignedTo) {
|
|
103
|
+
const client = getClient();
|
|
104
|
+
const ntScope = sessionScopeFilter();
|
|
105
|
+
const nextResult = await client.execute({
|
|
106
|
+
sql: `SELECT title, task_file, priority FROM tasks
|
|
107
|
+
WHERE assigned_to = ? AND status = 'open'${ntScope.sql}
|
|
108
|
+
ORDER BY priority ASC, created_at ASC
|
|
109
|
+
LIMIT 1`,
|
|
110
|
+
args: [assignedTo, ...ntScope.args]
|
|
111
|
+
});
|
|
112
|
+
if (nextResult.rows.length === 1) {
|
|
113
|
+
const nr = nextResult.rows[0];
|
|
114
|
+
return {
|
|
115
|
+
title: String(nr.title),
|
|
116
|
+
priority: String(nr.priority),
|
|
117
|
+
taskFile: String(nr.task_file)
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
return void 0;
|
|
121
|
+
}
|
|
122
|
+
async function checkSubtaskCompletion(parentTaskId, projectName) {
|
|
123
|
+
const client = getClient();
|
|
124
|
+
const scScope = sessionScopeFilter();
|
|
125
|
+
const remaining = await client.execute({
|
|
126
|
+
sql: `SELECT COUNT(*) as cnt FROM tasks
|
|
127
|
+
WHERE parent_task_id = ? AND status NOT IN ('done', 'needs_review', 'cancelled', 'closed')${scScope.sql}`,
|
|
128
|
+
args: [parentTaskId, ...scScope.args]
|
|
129
|
+
});
|
|
130
|
+
const cnt = Number(remaining.rows[0]?.cnt ?? 1);
|
|
131
|
+
if (cnt === 0) {
|
|
132
|
+
const parentRow = await client.execute({
|
|
133
|
+
sql: `SELECT assigned_to, title, task_file, project_name, session_scope FROM tasks WHERE id = ?`,
|
|
134
|
+
args: [parentTaskId]
|
|
135
|
+
});
|
|
136
|
+
if (parentRow.rows.length === 1) {
|
|
137
|
+
const pr = parentRow.rows[0];
|
|
138
|
+
const parentProject = pr.project_name == null ? projectName : String(pr.project_name);
|
|
139
|
+
await writeNotification({
|
|
140
|
+
agentId: String(pr.assigned_to),
|
|
141
|
+
agentRole: "system",
|
|
142
|
+
event: "subtasks_complete",
|
|
143
|
+
project: parentProject,
|
|
144
|
+
summary: `All subtasks complete for "${String(pr.title)}" \u2014 ready for rollup review`,
|
|
145
|
+
taskFile: String(pr.task_file),
|
|
146
|
+
// Scope the notification to the parent task's coordinator session —
|
|
147
|
+
// prevents subtask-complete notifications from leaking to other coordinator lanes.
|
|
148
|
+
sessionScope: pr.session_scope ? String(pr.session_scope) : void 0
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// src/lib/tasks.ts
|
|
155
|
+
async function createTask(input) {
|
|
156
|
+
const result = await createTaskCore(input);
|
|
157
|
+
if (!input.skipDispatch && result.status !== "blocked" && !process.env.VITEST) {
|
|
158
|
+
dispatchTaskToEmployee({
|
|
159
|
+
assignedTo: input.assignedTo,
|
|
160
|
+
taskId: result.id,
|
|
161
|
+
title: input.title,
|
|
162
|
+
priority: input.priority,
|
|
163
|
+
taskFile: result.taskFile,
|
|
164
|
+
initialStatus: result.status,
|
|
165
|
+
projectName: input.projectName,
|
|
166
|
+
spawnRuntime: input.spawnRuntime ?? void 0,
|
|
167
|
+
spawnModel: input.spawnModel ?? void 0
|
|
168
|
+
});
|
|
169
|
+
}
|
|
170
|
+
return result;
|
|
171
|
+
}
|
|
172
|
+
async function updateTask(input) {
|
|
173
|
+
const { row, taskFile, now, taskId } = await updateTaskStatus(input);
|
|
174
|
+
try {
|
|
175
|
+
const agent = String(row.assigned_to);
|
|
176
|
+
const cacheDir = path2.join(EXE_AI_DIR, "session-cache");
|
|
177
|
+
const cachePath = path2.join(cacheDir, `current-task-${agent}.json`);
|
|
178
|
+
if (input.status === "in_progress") {
|
|
179
|
+
mkdirSync(cacheDir, { recursive: true });
|
|
180
|
+
writeFileSync(cachePath, JSON.stringify({ taskId, title: String(row.title) }));
|
|
181
|
+
} else if (input.status === "needs_review" || input.status === "done" || input.status === "blocked" || input.status === "cancelled" || input.status === "closed") {
|
|
182
|
+
try {
|
|
183
|
+
unlinkSync(cachePath);
|
|
184
|
+
} catch {
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
} catch (e) {
|
|
188
|
+
process.stderr.write(`[tasks] trajectory tagging failed: ${e instanceof Error ? e.message : String(e)}
|
|
189
|
+
`);
|
|
190
|
+
}
|
|
191
|
+
if (input.status === "needs_review" || input.status === "done" || input.status === "closed") {
|
|
192
|
+
await cleanupReviewFile(row, taskFile, input.baseDir);
|
|
193
|
+
}
|
|
194
|
+
if (input.status === "needs_review" || input.status === "done" || input.status === "cancelled" || input.status === "closed") {
|
|
195
|
+
try {
|
|
196
|
+
const client = getClient();
|
|
197
|
+
const taskTitle = String(row.title);
|
|
198
|
+
const escaped = taskTitle.replace(/%/g, "\\%").replace(/_/g, "\\_");
|
|
199
|
+
await client.execute({
|
|
200
|
+
sql: `UPDATE tasks SET status = 'cancelled', updated_at = ?
|
|
201
|
+
WHERE title LIKE ? ESCAPE '\\' AND status IN ('open', 'in_progress')`,
|
|
202
|
+
args: [now, `%left '${escaped}' as in\\_progress%`]
|
|
203
|
+
});
|
|
204
|
+
} catch (e) {
|
|
205
|
+
process.stderr.write(`[tasks] shadow task cleanup failed: ${e instanceof Error ? e.message : String(e)}
|
|
206
|
+
`);
|
|
207
|
+
}
|
|
208
|
+
const assignedAgent = String(row.assigned_to);
|
|
209
|
+
if (!isCoordinatorName(assignedAgent)) {
|
|
210
|
+
try {
|
|
211
|
+
const draftClient = getClient();
|
|
212
|
+
if (input.status === "needs_review" || input.status === "done" || input.status === "closed") {
|
|
213
|
+
await draftClient.execute({
|
|
214
|
+
sql: `UPDATE memories SET draft = 0 WHERE agent_id = ? AND draft = 1`,
|
|
215
|
+
args: [assignedAgent]
|
|
216
|
+
});
|
|
217
|
+
} else if (input.status === "cancelled") {
|
|
218
|
+
const now2 = (/* @__PURE__ */ new Date()).toISOString();
|
|
219
|
+
await draftClient.execute({
|
|
220
|
+
sql: `UPDATE memories SET status = 'deleted', deleted_at = ? WHERE agent_id = ? AND draft = 1`,
|
|
221
|
+
args: [now2, assignedAgent]
|
|
222
|
+
});
|
|
223
|
+
}
|
|
224
|
+
} catch (e) {
|
|
225
|
+
process.stderr.write(`[tasks] draft memory promotion/purge failed: ${e instanceof Error ? e.message : String(e)}
|
|
226
|
+
`);
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
try {
|
|
230
|
+
const client = getClient();
|
|
231
|
+
const cascaded = await client.execute({
|
|
232
|
+
sql: `UPDATE tasks SET status = 'closed', updated_at = ?
|
|
233
|
+
WHERE parent_task_id = ? AND status = 'needs_review'`,
|
|
234
|
+
args: [now, taskId]
|
|
235
|
+
});
|
|
236
|
+
if (cascaded.rowsAffected > 0) {
|
|
237
|
+
process.stderr.write(
|
|
238
|
+
`[cascade] Closed ${cascaded.rowsAffected} orphaned review task(s) for parent ${taskId}
|
|
239
|
+
`
|
|
240
|
+
);
|
|
241
|
+
}
|
|
242
|
+
} catch (e) {
|
|
243
|
+
process.stderr.write(`[tasks] review cascade close failed: ${e instanceof Error ? e.message : String(e)}
|
|
244
|
+
`);
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
const isCoordinator = isCoordinatorName(String(row.assigned_to));
|
|
248
|
+
if (input.status === "done" && row.reviewer && !isCoordinator) {
|
|
249
|
+
input.status = "needs_review";
|
|
250
|
+
}
|
|
251
|
+
const isTerminal = input.status === "done" || input.status === "needs_review" || input.status === "closed";
|
|
252
|
+
if (isTerminal) {
|
|
253
|
+
if (!isCoordinator) {
|
|
254
|
+
notifyTaskDone();
|
|
255
|
+
}
|
|
256
|
+
if ((input.status === "done" || input.status === "needs_review") && row.reviewer && !isCoordinator) {
|
|
257
|
+
const reviewer = String(row.reviewer);
|
|
258
|
+
const taskTitle = String(row.title);
|
|
259
|
+
let delivered = false;
|
|
260
|
+
try {
|
|
261
|
+
const { sendIntercom, resolveExeSession, employeeSessionName } = await import("./lib/tmux-routing.js");
|
|
262
|
+
const exeSession = row.session_scope ? String(row.session_scope) : resolveExeSession();
|
|
263
|
+
if (exeSession) {
|
|
264
|
+
if (isCoordinatorName(reviewer)) {
|
|
265
|
+
const cooResult = sendIntercom(exeSession, { force: true, reason: "completion" });
|
|
266
|
+
if (cooResult !== "failed") {
|
|
267
|
+
delivered = true;
|
|
268
|
+
process.stderr.write(`[tasks] EVENT: notified coordinator "${reviewer}" for "${taskTitle}" (force)
|
|
269
|
+
`);
|
|
270
|
+
}
|
|
271
|
+
} else {
|
|
272
|
+
const reviewerSession = employeeSessionName(reviewer, exeSession);
|
|
273
|
+
const result = sendIntercom(reviewerSession, { force: true, reason: "completion" });
|
|
274
|
+
if (result !== "failed") {
|
|
275
|
+
delivered = true;
|
|
276
|
+
process.stderr.write(`[tasks] EVENT: notified reviewer "${reviewer}" via tmux for "${taskTitle}"
|
|
277
|
+
`);
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
} catch (e) {
|
|
282
|
+
process.stderr.write(`[tasks] EVENT: tmux delivery failed: ${e instanceof Error ? e.message : String(e)}
|
|
283
|
+
`);
|
|
284
|
+
}
|
|
285
|
+
if (!delivered) {
|
|
286
|
+
try {
|
|
287
|
+
process.stderr.write(`[tasks] EVENT: tmux delivery failed for reviewer="${reviewer}". Queueing for daemon delivery.
|
|
288
|
+
`);
|
|
289
|
+
const { queueIntercom } = await import("./intercom-queue-A6UJEFIF.js");
|
|
290
|
+
const { resolveExeSession } = await import("./lib/tmux-routing.js");
|
|
291
|
+
const exeSession = row.session_scope ? String(row.session_scope) : resolveExeSession();
|
|
292
|
+
if (exeSession) {
|
|
293
|
+
const reviewerSession = isCoordinatorName(reviewer) ? exeSession : `${reviewer}-${exeSession}`;
|
|
294
|
+
queueIntercom(
|
|
295
|
+
reviewerSession,
|
|
296
|
+
`Review ready: "${taskTitle}" by ${String(row.assigned_to)}. Run /exe-review.`,
|
|
297
|
+
"completion"
|
|
298
|
+
);
|
|
299
|
+
}
|
|
300
|
+
} catch (e) {
|
|
301
|
+
process.stderr.write(`[tasks] EVENT: intercom queue failed: ${e instanceof Error ? e.message : String(e)}
|
|
302
|
+
`);
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
try {
|
|
306
|
+
const { pushNotifyAsync } = await import("./push-notifications-AMHVR6DF.js");
|
|
307
|
+
pushNotifyAsync({
|
|
308
|
+
event: "review_ready",
|
|
309
|
+
title: `Review ready: ${taskTitle.slice(0, 50)}`,
|
|
310
|
+
body: `${String(row.assigned_to)} completed "${taskTitle}". Reviewer: ${reviewer}`,
|
|
311
|
+
priority: "normal"
|
|
312
|
+
});
|
|
313
|
+
} catch {
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
if (!isCoordinator) {
|
|
317
|
+
try {
|
|
318
|
+
const { pushNotifyAsync } = await import("./push-notifications-AMHVR6DF.js");
|
|
319
|
+
const event = input.status === "needs_review" ? "review_ready" : "task_complete";
|
|
320
|
+
pushNotifyAsync({
|
|
321
|
+
event,
|
|
322
|
+
title: `${String(row.assigned_to)} completed: ${String(row.title).slice(0, 80)}`,
|
|
323
|
+
body: input.result ? String(input.result).slice(0, 500) : `Task "${String(row.title)}" is ${input.status}.`,
|
|
324
|
+
agent: String(row.assigned_to),
|
|
325
|
+
project: String(row.project_name)
|
|
326
|
+
});
|
|
327
|
+
} catch {
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
await markTaskNotificationsRead(taskFile);
|
|
331
|
+
if (input.status === "needs_review" && !isCoordinator && row.reviewer) {
|
|
332
|
+
try {
|
|
333
|
+
const { writeNotification: writeNotification2 } = await import("./notifications-65STXW6N.js");
|
|
334
|
+
const reviewer = String(row.reviewer);
|
|
335
|
+
await writeNotification2({
|
|
336
|
+
agentId: reviewer,
|
|
337
|
+
agentRole: isCoordinatorName(reviewer) ? "COO" : "manager",
|
|
338
|
+
event: "task_complete",
|
|
339
|
+
project: String(row.project_name),
|
|
340
|
+
summary: `"${String(row.title)}" is ready for review`,
|
|
341
|
+
taskFile,
|
|
342
|
+
sessionScope: row.session_scope ? String(row.session_scope) : void 0
|
|
343
|
+
});
|
|
344
|
+
} catch (e) {
|
|
345
|
+
process.stderr.write(`[tasks] needs_review notification failed: ${e instanceof Error ? e.message : String(e)}
|
|
346
|
+
`);
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
if (input.status === "needs_review" || input.status === "done" || input.status === "closed") {
|
|
350
|
+
try {
|
|
351
|
+
await cascadeUnblock(taskId, input.baseDir, now);
|
|
352
|
+
} catch (e) {
|
|
353
|
+
process.stderr.write(`[tasks] cascade unblock failed: ${e instanceof Error ? e.message : String(e)}
|
|
354
|
+
`);
|
|
355
|
+
}
|
|
356
|
+
orgBus.emit({
|
|
357
|
+
type: "task_completed",
|
|
358
|
+
taskId,
|
|
359
|
+
employee: String(row.assigned_to),
|
|
360
|
+
result: input.result ?? "",
|
|
361
|
+
timestamp: now
|
|
362
|
+
});
|
|
363
|
+
if (row.parent_task_id) {
|
|
364
|
+
try {
|
|
365
|
+
await checkSubtaskCompletion(String(row.parent_task_id), String(row.project_name));
|
|
366
|
+
} catch (e) {
|
|
367
|
+
process.stderr.write(`[tasks] subtask completion check failed: ${e instanceof Error ? e.message : String(e)}
|
|
368
|
+
`);
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
if (input.status === "cancelled") {
|
|
374
|
+
try {
|
|
375
|
+
await cascadeUnblock(taskId, input.baseDir, now);
|
|
376
|
+
} catch (e) {
|
|
377
|
+
process.stderr.write(`[tasks] cancelled cascade unblock failed: ${e instanceof Error ? e.message : String(e)}
|
|
378
|
+
`);
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
if ((input.status === "needs_review" || input.status === "done" || input.status === "closed") && !isCoordinatorName(String(row.assigned_to)) && !process.env.VITEST) {
|
|
382
|
+
import("./lib/skill-learning.js").then(
|
|
383
|
+
({ captureAndLearn }) => captureAndLearn({
|
|
384
|
+
taskId,
|
|
385
|
+
agentId: String(row.assigned_to),
|
|
386
|
+
projectName: String(row.project_name),
|
|
387
|
+
taskTitle: String(row.title)
|
|
388
|
+
})
|
|
389
|
+
).catch((err) => {
|
|
390
|
+
process.stderr.write(
|
|
391
|
+
`[updateTask] skill learning failed: ${err instanceof Error ? err.message : String(err)}
|
|
392
|
+
`
|
|
393
|
+
);
|
|
394
|
+
});
|
|
395
|
+
}
|
|
396
|
+
let nextTask;
|
|
397
|
+
if (isTerminal && !isCoordinatorName(String(row.assigned_to))) {
|
|
398
|
+
try {
|
|
399
|
+
nextTask = await findNextTask(String(row.assigned_to));
|
|
400
|
+
} catch (e) {
|
|
401
|
+
process.stderr.write(`[tasks] task chaining failed: ${e instanceof Error ? e.message : String(e)}
|
|
402
|
+
`);
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
return {
|
|
406
|
+
id: String(row.id),
|
|
407
|
+
title: String(row.title),
|
|
408
|
+
assignedTo: String(row.assigned_to),
|
|
409
|
+
assignedBy: String(row.assigned_by),
|
|
410
|
+
reviewer: row.reviewer !== void 0 && row.reviewer !== null ? String(row.reviewer) : null,
|
|
411
|
+
projectName: String(row.project_name),
|
|
412
|
+
priority: String(row.priority),
|
|
413
|
+
status: input.status,
|
|
414
|
+
taskFile,
|
|
415
|
+
createdAt: String(row.created_at),
|
|
416
|
+
updatedAt: now,
|
|
417
|
+
budgetTokens: row.budget_tokens !== void 0 && row.budget_tokens !== null ? Number(row.budget_tokens) : null,
|
|
418
|
+
budgetFallbackModel: row.budget_fallback_model !== void 0 && row.budget_fallback_model !== null ? String(row.budget_fallback_model) : null,
|
|
419
|
+
tokensUsed: Number(row.tokens_used ?? 0),
|
|
420
|
+
tokensWarnedAt: row.tokens_warned_at !== void 0 && row.tokens_warned_at !== null ? Number(row.tokens_warned_at) : null,
|
|
421
|
+
spawnRuntime: row.spawn_runtime !== void 0 && row.spawn_runtime !== null ? String(row.spawn_runtime) : null,
|
|
422
|
+
spawnModel: row.spawn_model !== void 0 && row.spawn_model !== null ? String(row.spawn_model) : null,
|
|
423
|
+
sessionScope: row.session_scope !== void 0 && row.session_scope !== null ? String(row.session_scope) : null,
|
|
424
|
+
nextTask
|
|
425
|
+
};
|
|
426
|
+
}
|
|
427
|
+
async function deleteTask(taskId, baseDir) {
|
|
428
|
+
const client = getClient();
|
|
429
|
+
const { taskFile, assignedTo, assignedBy, taskSlug } = await deleteTaskCore(taskId, baseDir);
|
|
430
|
+
const coordinatorName = getCoordinatorName();
|
|
431
|
+
const reviewer = assignedBy || coordinatorName;
|
|
432
|
+
const reviewSlug = `review-${assignedTo}-${taskSlug}`;
|
|
433
|
+
const reviewFile = `exe/${reviewer}/${reviewSlug}.md`;
|
|
434
|
+
const legacyReviewFile = `exe/${coordinatorName}/${reviewSlug}.md`;
|
|
435
|
+
await client.execute({
|
|
436
|
+
sql: "DELETE FROM tasks WHERE task_file = ? OR task_file = ? OR task_file = ?",
|
|
437
|
+
args: [reviewFile, legacyReviewFile, `exe/exe/${reviewSlug}.md`]
|
|
438
|
+
});
|
|
439
|
+
await markAsReadByTaskFile(taskFile);
|
|
440
|
+
await markAsReadByTaskFile(reviewFile);
|
|
441
|
+
}
|
|
442
|
+
|
|
443
|
+
export {
|
|
444
|
+
createTask,
|
|
445
|
+
updateTask,
|
|
446
|
+
deleteTask
|
|
447
|
+
};
|
|
@@ -0,0 +1,214 @@
|
|
|
1
|
+
import {
|
|
2
|
+
sessionScopeFilter
|
|
3
|
+
} from "./chunk-5AS622MM.js";
|
|
4
|
+
|
|
5
|
+
// src/lib/git-task-sweep.ts
|
|
6
|
+
import { execSync } from "child_process";
|
|
7
|
+
var DEFAULT_COMMIT_LIMIT = 50;
|
|
8
|
+
var DEFAULT_STALE_MINUTES = 10;
|
|
9
|
+
var AUTO_ESCALATE_THRESHOLD = 0.6;
|
|
10
|
+
var EXACT_UUID_SCORE = 1;
|
|
11
|
+
var TITLE_KEYWORD_SCORE = 0.5;
|
|
12
|
+
var FILE_PATH_SCORE = 0.3;
|
|
13
|
+
var MIN_KEYWORD_OVERLAP = 3;
|
|
14
|
+
var STOP_WORDS = /* @__PURE__ */ new Set([
|
|
15
|
+
"a",
|
|
16
|
+
"an",
|
|
17
|
+
"the",
|
|
18
|
+
"and",
|
|
19
|
+
"or",
|
|
20
|
+
"but",
|
|
21
|
+
"in",
|
|
22
|
+
"on",
|
|
23
|
+
"at",
|
|
24
|
+
"to",
|
|
25
|
+
"for",
|
|
26
|
+
"of",
|
|
27
|
+
"with",
|
|
28
|
+
"by",
|
|
29
|
+
"from",
|
|
30
|
+
"is",
|
|
31
|
+
"it",
|
|
32
|
+
"as",
|
|
33
|
+
"be",
|
|
34
|
+
"was",
|
|
35
|
+
"are",
|
|
36
|
+
"this",
|
|
37
|
+
"that",
|
|
38
|
+
"not",
|
|
39
|
+
"no",
|
|
40
|
+
"if",
|
|
41
|
+
"so",
|
|
42
|
+
"do",
|
|
43
|
+
"up"
|
|
44
|
+
]);
|
|
45
|
+
function extractKeywords(text) {
|
|
46
|
+
return text.toLowerCase().replace(/[^a-z0-9\s-]/g, " ").split(/\s+/).filter((w) => w.length >= 3 && !STOP_WORDS.has(w));
|
|
47
|
+
}
|
|
48
|
+
function matchScore(task, commitMessage, changedFiles) {
|
|
49
|
+
if (task.id.length >= 8 && commitMessage.includes(task.id)) {
|
|
50
|
+
return EXACT_UUID_SCORE;
|
|
51
|
+
}
|
|
52
|
+
let score = 0;
|
|
53
|
+
const titleWords = extractKeywords(task.title);
|
|
54
|
+
const commitWords = new Set(extractKeywords(commitMessage));
|
|
55
|
+
const overlap = titleWords.filter((w) => commitWords.has(w));
|
|
56
|
+
if (overlap.length >= MIN_KEYWORD_OVERLAP) {
|
|
57
|
+
score += TITLE_KEYWORD_SCORE;
|
|
58
|
+
}
|
|
59
|
+
if (task.context && changedFiles.length > 0) {
|
|
60
|
+
const contextLower = task.context.toLowerCase();
|
|
61
|
+
const hasFileMatch = changedFiles.some(
|
|
62
|
+
(f) => contextLower.includes(f.toLowerCase())
|
|
63
|
+
);
|
|
64
|
+
if (hasFileMatch) {
|
|
65
|
+
score += FILE_PATH_SCORE;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
return score;
|
|
69
|
+
}
|
|
70
|
+
function getRecentCommits(limit = DEFAULT_COMMIT_LIMIT) {
|
|
71
|
+
try {
|
|
72
|
+
const SEPARATOR = "<<SEP>>";
|
|
73
|
+
const output = execSync(
|
|
74
|
+
`git log --format="%h${SEPARATOR}%s${SEPARATOR}%aI" --name-only -n ${limit} -z`,
|
|
75
|
+
{ encoding: "utf8", timeout: 1e4 }
|
|
76
|
+
);
|
|
77
|
+
const entries = output.split("\0").filter(Boolean);
|
|
78
|
+
const commits = [];
|
|
79
|
+
let current = null;
|
|
80
|
+
for (const entry of entries) {
|
|
81
|
+
if (entry.includes(SEPARATOR)) {
|
|
82
|
+
const lines = entry.split("\n");
|
|
83
|
+
const headerLine = lines[0];
|
|
84
|
+
const parts = headerLine.split(SEPARATOR);
|
|
85
|
+
if (parts.length >= 3) {
|
|
86
|
+
if (current) commits.push(current);
|
|
87
|
+
current = {
|
|
88
|
+
hash: parts[0],
|
|
89
|
+
message: parts[1],
|
|
90
|
+
files: lines.slice(1).filter(Boolean),
|
|
91
|
+
date: new Date(parts[2])
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
} else if (current) {
|
|
95
|
+
const files = entry.split("\n").filter(Boolean);
|
|
96
|
+
current.files.push(...files);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
if (current) commits.push(current);
|
|
100
|
+
return commits;
|
|
101
|
+
} catch {
|
|
102
|
+
return [];
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
function isStale(updatedAt, staleMinutes) {
|
|
106
|
+
const updated = new Date(updatedAt).getTime();
|
|
107
|
+
const threshold = Date.now() - staleMinutes * 6e4;
|
|
108
|
+
return updated < threshold;
|
|
109
|
+
}
|
|
110
|
+
function findBestMatch(task, commits) {
|
|
111
|
+
let best = null;
|
|
112
|
+
for (const commit of commits) {
|
|
113
|
+
const score = matchScore(task, commit.message, commit.files);
|
|
114
|
+
if (score >= AUTO_ESCALATE_THRESHOLD && (!best || score > best.score)) {
|
|
115
|
+
best = { commit, score };
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
return best;
|
|
119
|
+
}
|
|
120
|
+
async function sweepTasks(projectName, options = {}) {
|
|
121
|
+
const commitLimit = options.commitLimit ?? DEFAULT_COMMIT_LIMIT;
|
|
122
|
+
const staleMinutes = options.staleMinutes ?? DEFAULT_STALE_MINUTES;
|
|
123
|
+
const dryRun = options.dryRun ?? false;
|
|
124
|
+
const result = { escalated: [], unchanged: 0, errors: [] };
|
|
125
|
+
const commits = getRecentCommits(commitLimit);
|
|
126
|
+
if (commits.length === 0) {
|
|
127
|
+
result.errors.push("No git commits found (not a git repo or empty history)");
|
|
128
|
+
return result;
|
|
129
|
+
}
|
|
130
|
+
let tasks;
|
|
131
|
+
try {
|
|
132
|
+
const { initStore } = await import("./lib/store.js");
|
|
133
|
+
await initStore();
|
|
134
|
+
const { getClient } = await import("./lib/database.js");
|
|
135
|
+
const client = getClient();
|
|
136
|
+
const conditions = ["status = 'in_progress'"];
|
|
137
|
+
const args = [];
|
|
138
|
+
if (projectName) {
|
|
139
|
+
conditions.push("project_name = ?");
|
|
140
|
+
args.push(projectName);
|
|
141
|
+
}
|
|
142
|
+
const swScope = sessionScopeFilter();
|
|
143
|
+
if (swScope.sql) {
|
|
144
|
+
conditions.push("(session_scope IS NULL OR session_scope = ?)");
|
|
145
|
+
args.push(...swScope.args);
|
|
146
|
+
}
|
|
147
|
+
const queryResult = await client.execute({
|
|
148
|
+
sql: `SELECT id, title, assigned_to, project_name, status, updated_at, context
|
|
149
|
+
FROM tasks WHERE ${conditions.join(" AND ")}
|
|
150
|
+
ORDER BY updated_at ASC`,
|
|
151
|
+
args
|
|
152
|
+
});
|
|
153
|
+
tasks = queryResult.rows.map((r) => ({
|
|
154
|
+
id: String(r.id),
|
|
155
|
+
title: String(r.title),
|
|
156
|
+
assignedTo: String(r.assigned_to),
|
|
157
|
+
projectName: String(r.project_name),
|
|
158
|
+
status: String(r.status),
|
|
159
|
+
updatedAt: String(r.updated_at),
|
|
160
|
+
context: r.context ? String(r.context) : void 0
|
|
161
|
+
}));
|
|
162
|
+
} catch (err) {
|
|
163
|
+
result.errors.push(`DB query failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
164
|
+
return result;
|
|
165
|
+
}
|
|
166
|
+
if (tasks.length === 0) {
|
|
167
|
+
return result;
|
|
168
|
+
}
|
|
169
|
+
for (const task of tasks) {
|
|
170
|
+
if (!isStale(task.updatedAt, staleMinutes)) {
|
|
171
|
+
result.unchanged++;
|
|
172
|
+
continue;
|
|
173
|
+
}
|
|
174
|
+
const match = findBestMatch(task, commits);
|
|
175
|
+
if (!match) {
|
|
176
|
+
result.unchanged++;
|
|
177
|
+
continue;
|
|
178
|
+
}
|
|
179
|
+
if (!dryRun) {
|
|
180
|
+
try {
|
|
181
|
+
const { updateTaskStatus } = await import("./tasks-crud-HIPXKRKX.js");
|
|
182
|
+
await updateTaskStatus({
|
|
183
|
+
taskId: task.id,
|
|
184
|
+
status: "needs_review",
|
|
185
|
+
result: `Auto-escalated by git-sweep: matching commit ${match.commit.hash} found (score: ${match.score.toFixed(2)})`
|
|
186
|
+
});
|
|
187
|
+
} catch (err) {
|
|
188
|
+
result.errors.push(
|
|
189
|
+
`Failed to escalate task ${task.id}: ${err instanceof Error ? err.message : String(err)}`
|
|
190
|
+
);
|
|
191
|
+
continue;
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
result.escalated.push({
|
|
195
|
+
taskId: task.id,
|
|
196
|
+
title: task.title,
|
|
197
|
+
matchedCommit: match.commit.hash,
|
|
198
|
+
score: match.score
|
|
199
|
+
});
|
|
200
|
+
process.stderr.write(
|
|
201
|
+
`[git-sweep] ${dryRun ? "WOULD escalate" : "Escalated"} task ${task.id} \u2192 commit ${match.commit.hash} (score: ${match.score.toFixed(2)})
|
|
202
|
+
`
|
|
203
|
+
);
|
|
204
|
+
}
|
|
205
|
+
return result;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
export {
|
|
209
|
+
extractKeywords,
|
|
210
|
+
matchScore,
|
|
211
|
+
getRecentCommits,
|
|
212
|
+
findBestMatch,
|
|
213
|
+
sweepTasks
|
|
214
|
+
};
|