@askexenow/exe-os 0.9.127 → 0.9.129
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-3FDJYREJ.js +22 -0
- package/dist/active-agent-WC7YK3L5.js +23 -0
- package/dist/agent-context-CDWUZEXY.js +11 -0
- package/dist/agent-loop-G4VUR5ME.js +10 -0
- package/dist/agentic-ontology-2VYNTMNO.js +25 -0
- package/dist/backfill-metadata-OJWOCLSZ.js +601 -0
- package/dist/background-jobs-LGRDXGXC.js +25 -0
- package/dist/bash-Y5PEW5UQ.js +8 -0
- package/dist/behaviors-CSB2YLDT.js +22 -0
- package/dist/bin/age-ontology-load.js +10 -258
- package/dist/bin/agentic-ontology-backfill.js +23 -5296
- package/dist/bin/agentic-reflection-backfill.js +19 -4739
- package/dist/bin/agentic-semantic-label.js +28 -4723
- package/dist/bin/backfill-conversations.js +42 -5543
- package/dist/bin/backfill-responses.js +41 -5542
- package/dist/bin/backfill-vectors.js +43 -4795
- package/dist/bin/bulk-sync-postgres.js +22 -5313
- package/dist/bin/cc-doctor.js +10 -602
- package/dist/bin/cleanup-stale-review-tasks.js +32 -6803
- package/dist/bin/cli.js +169 -38993
- package/dist/bin/exe-agent-config.js +21 -253
- package/dist/bin/exe-agent.js +63 -2429
- package/dist/bin/exe-assign.js +34 -5810
- package/dist/bin/exe-boot.js +180 -12726
- package/dist/bin/exe-call.js +48 -1409
- package/dist/bin/exe-cloud.js +64 -8516
- package/dist/bin/exe-dispatch.js +35 -10818
- package/dist/bin/exe-doctor.js +24 -7537
- package/dist/bin/exe-export-behaviors.js +18 -6122
- package/dist/bin/exe-forget.js +22 -6274
- package/dist/bin/exe-gateway.js +62 -15286
- package/dist/bin/exe-healthcheck.js +9 -602
- package/dist/bin/exe-heartbeat.js +51 -6898
- package/dist/bin/exe-kill.js +30 -5961
- package/dist/bin/exe-launch-agent.js +101 -6788
- package/dist/bin/exe-new-employee.js +42 -3774
- package/dist/bin/exe-pending-messages.js +28 -6833
- package/dist/bin/exe-pending-notifications.js +31 -6869
- package/dist/bin/exe-pending-reviews.js +38 -6910
- package/dist/bin/exe-rename.js +57 -5766
- package/dist/bin/exe-review.js +29 -5948
- package/dist/bin/exe-search.js +22 -7692
- package/dist/bin/exe-session-cleanup.js +52 -11147
- package/dist/bin/exe-settings.js +35 -524
- package/dist/bin/exe-start-codex.js +70 -6641
- package/dist/bin/exe-start-opencode.js +54 -6584
- package/dist/bin/exe-status.js +26 -7081
- package/dist/bin/exe-support.js +10 -546
- package/dist/bin/exe-team.js +18 -5942
- package/dist/bin/git-sweep.js +26 -10909
- package/dist/bin/graph-backfill.js +24 -5979
- package/dist/bin/graph-export.js +31 -6128
- package/dist/bin/graph-layer-benchmark.js +4 -8
- package/dist/bin/install.js +47 -2322
- package/dist/bin/intercom-check.js +25 -11501
- package/dist/bin/list-providers.js +4 -18
- package/dist/bin/postgres-agentic-reflection-backfill.js +14 -357
- package/dist/bin/postgres-agentic-semantic-backfill.js +14 -404
- package/dist/bin/pre-publish.js +4 -298
- package/dist/bin/registry-proxy.js +8 -172
- package/dist/bin/scan-tasks.js +47 -10881
- package/dist/bin/setup.js +6 -9492
- package/dist/bin/shard-migrate.js +30 -5149
- package/dist/bin/stack-update.js +75 -203
- package/dist/bin/update.js +20 -654
- package/dist/capacity-monitor-3GZIHLB4.js +46 -0
- package/dist/catchup-brief-67QDCINS.js +148 -0
- package/dist/chunk-2QD7KBJN.js +664 -0
- package/dist/chunk-2WGSIVI4.js +149 -0
- package/dist/chunk-35TVNLJI.js +57 -0
- package/dist/chunk-3SYV36U5.js +47 -0
- package/dist/chunk-46SQTBQW.js +207 -0
- package/dist/chunk-4CMVLKEI.js +171 -0
- package/dist/chunk-4FLUOZTR.js +116 -0
- package/dist/chunk-4NYQAS33.js +17 -0
- package/dist/chunk-4QNPADWL.js +902 -0
- package/dist/chunk-4Y6QZBEO.js +58 -0
- package/dist/chunk-557C2IGL.js +0 -0
- package/dist/chunk-55S56YDO.js +305 -0
- package/dist/chunk-5BS5ELMC.js +85 -0
- package/dist/chunk-5DHM77JE.js +256 -0
- package/dist/chunk-5QTWQKNF.js +137 -0
- package/dist/chunk-6A7P74QV.js +53 -0
- package/dist/chunk-6BFIFI5F.js +45 -0
- package/dist/chunk-6KFICEAT.js +20 -0
- package/dist/chunk-6LKDJ5WX.js +317 -0
- package/dist/chunk-6M5II4EV.js +148 -0
- package/dist/chunk-6QEFN3DV.js +302 -0
- package/dist/chunk-6WTMJAG3.js +171 -0
- package/dist/chunk-6Y4B3QF6.js +19 -0
- package/dist/chunk-7RK3AYJE.js +163 -0
- package/dist/chunk-7TCP2XTP.js +390 -0
- package/dist/chunk-A2NZP64U.js +2307 -0
- package/dist/chunk-AFYIYM2K.js +322 -0
- package/dist/chunk-AJWFPKQS.js +573 -0
- package/dist/chunk-AVE2B4DQ.js +140 -0
- package/dist/chunk-B4TT2QQA.js +87 -0
- package/dist/chunk-BCMOO4OQ.js +1066 -0
- package/dist/chunk-BU3ID2WT.js +432 -0
- package/dist/chunk-CFIM43JH.js +862 -0
- package/dist/chunk-CURJUF3W.js +66 -0
- package/dist/chunk-DC4ZD76Y.js +288 -0
- package/dist/chunk-DL52JPGF.js +706 -0
- package/dist/chunk-DRSC6ZRA.js +36 -0
- package/dist/chunk-E2Q4Y22A.js +86 -0
- package/dist/chunk-ECGTESAP.js +236 -0
- package/dist/chunk-EHOJUXYG.js +365 -0
- package/dist/chunk-EIT6J37V.js +119 -0
- package/dist/chunk-ELE2C7L4.js +31 -0
- package/dist/chunk-EQSQ4HBW.js +9 -0
- package/dist/chunk-FWFFZGSC.js +94 -0
- package/dist/chunk-GCNWCYJI.js +249 -0
- package/dist/chunk-GGTUUI7J.js +6 -0
- package/dist/chunk-GU56L7OY.js +223 -0
- package/dist/chunk-HF25XPZE.js +83 -0
- package/dist/chunk-HOKOOBDL.js +52 -0
- package/dist/chunk-HP43GLZ6.js +89 -0
- package/dist/chunk-HSRM6PNM.js +682 -0
- package/dist/chunk-HVRTW3OP.js +164 -0
- package/dist/chunk-IHDHL5ML.js +125 -0
- package/dist/chunk-IHF5RXJM.js +168 -0
- package/dist/chunk-IJ73YE73.js +409 -0
- package/dist/chunk-INO4ZD5Y.js +162 -0
- package/dist/chunk-IRQTM7DY.js +475 -0
- package/dist/chunk-JD4FAPYG.js +1208 -0
- package/dist/chunk-JN44LQ53.js +316 -0
- package/dist/chunk-JQ6TLNIV.js +133 -0
- package/dist/chunk-JT76KRZQ.js +35 -0
- package/dist/chunk-K6OEWZTB.js +262 -0
- package/dist/chunk-KCUYRZSX.js +72 -0
- package/dist/chunk-KFQGP6VL.js +33 -0
- package/dist/chunk-KRLOFIPI.js +83 -0
- package/dist/chunk-KVPG5UT6.js +305 -0
- package/dist/chunk-L3TB7CC3.js +170 -0
- package/dist/chunk-L4X5EUHR.js +283 -0
- package/dist/chunk-LAN26C3W.js +655 -0
- package/dist/chunk-LPGBKR4H.js +54 -0
- package/dist/chunk-LVNMOVGD.js +132 -0
- package/dist/chunk-LX5WEEZ7.js +157 -0
- package/dist/chunk-MBH42QNH.js +295 -0
- package/dist/chunk-MFO7MNSD.js +102 -0
- package/dist/chunk-MILTNDZH.js +736 -0
- package/dist/chunk-MJAE4NB7.js +1090 -0
- package/dist/chunk-MLCUPTP7.js +132 -0
- package/dist/chunk-MPURH6JI.js +445 -0
- package/dist/chunk-MVBWW2EO.js +89 -0
- package/dist/chunk-MZIGUUDQ.js +42 -0
- package/dist/chunk-N3DTSVKE.js +157 -0
- package/dist/chunk-N5RRQOAC.js +134 -0
- package/dist/chunk-N7L33TED.js +1664 -0
- package/dist/chunk-NDYSNVI5.js +122 -0
- package/dist/chunk-NEAHEC5G.js +155 -0
- package/dist/chunk-NGVOA6ZQ.js +84 -0
- package/dist/chunk-NHK5KW6Y.js +193 -0
- package/dist/chunk-NMQKGFEP.js +458 -0
- package/dist/chunk-NPT4EHZK.js +352 -0
- package/dist/chunk-NXFJHTED.js +55 -0
- package/dist/chunk-NZLCDI6Y.js +216 -0
- package/dist/chunk-O377P7GM.js +100 -0
- package/dist/chunk-O4VFZRA5.js +33 -0
- package/dist/chunk-O7YD7JYE.js +237 -0
- package/dist/chunk-OFK72LTZ.js +81 -0
- package/dist/chunk-ORCCI2VV.js +140 -0
- package/dist/chunk-OVQR4KMC.js +579 -0
- package/dist/chunk-P3CVUMEH.js +54 -0
- package/dist/chunk-P4HIF72O.js +30 -0
- package/dist/chunk-PEDLSJBL.js +1039 -0
- package/dist/chunk-PFICKZZ5.js +219 -0
- package/dist/chunk-PGW7TIV6.js +142 -0
- package/dist/chunk-PJHHFJAN.js +63 -0
- package/dist/chunk-PP4SNOJ5.js +471 -0
- package/dist/chunk-PRKVT4KN.js +90 -0
- package/dist/chunk-Q2O4VJHH.js +305 -0
- package/dist/chunk-Q36CMZIX.js +38 -0
- package/dist/chunk-Q3V7K4ME.js +149 -0
- package/dist/chunk-QGYPJGHB.js +81 -0
- package/dist/chunk-R23EMFJV.js +198 -0
- package/dist/chunk-R4M6XCMU.js +39 -0
- package/dist/chunk-R4WMF32C.js +81 -0
- package/dist/chunk-RJT7H2KR.js +349 -0
- package/dist/chunk-RJZMRZNV.js +280 -0
- package/dist/chunk-RMFDPCA2.js +72 -0
- package/dist/chunk-RUG3N6P4.js +292 -0
- package/dist/chunk-RWO327FJ.js +84 -0
- package/dist/chunk-SG7EFHQ7.js +108 -0
- package/dist/chunk-SH45SJQW.js +0 -0
- package/dist/chunk-SPKE5ANC.js +42 -0
- package/dist/chunk-SVMFVPBF.js +104 -0
- package/dist/chunk-SVXDCELZ.js +163 -0
- package/dist/chunk-TMEE7AT7.js +72 -0
- package/dist/chunk-TWYEE4FB.js +125 -0
- package/dist/chunk-TYLF4LSW.js +110 -0
- package/dist/chunk-V4TZI6EO.js +60 -0
- package/dist/chunk-VFZTXKQI.js +372 -0
- package/dist/chunk-VQ4K2MLM.js +199 -0
- package/dist/chunk-VU447UAO.js +173 -0
- package/dist/chunk-WMC3TXJF.js +50 -0
- package/dist/chunk-WPOA7XAE.js +84 -0
- package/dist/chunk-WZQ4CPRG.js +64 -0
- package/dist/chunk-X2IMCCM5.js +49 -0
- package/dist/chunk-X2Z5GT3V.js +118 -0
- package/dist/chunk-XEKO37NM.js +43 -0
- package/dist/chunk-XUHFQHGZ.js +505 -0
- package/dist/chunk-YGAAZN3E.js +47 -0
- package/dist/chunk-YKOW7KMK.js +558 -0
- package/dist/chunk-YOQKMB6N.js +145 -0
- package/dist/chunk-YZFZDJWZ.js +107 -0
- package/dist/chunk-ZBY4AUD4.js +10342 -0
- package/dist/chunk-ZOJYPNCM.js +127 -0
- package/dist/chunk-ZS7MGQSD.js +1494 -0
- package/dist/chunk-ZTZQ5A6B.js +191 -0
- package/dist/code-context-index-HXGC4IJL.js +28 -0
- package/dist/content-extractor-UVZXRIPC.js +8 -0
- package/dist/conversation-entity-extractor-HKSNDF4L.js +113 -0
- package/dist/conversation-wiki-populator-NKDAQEZ5.js +105 -0
- package/dist/crdt-sync-T7MHVQCS.js +33 -0
- package/dist/crm-bridge-Q5JBVJ6P.js +19 -0
- package/dist/crm-webhook-OAFTLJYA.js +10 -0
- package/dist/cto-delegation-gate-XWHZCUJH.js +208 -0
- package/dist/daemon-auth-GWUXTUYO.js +15 -0
- package/dist/daemon-orchestration-2C3TVOHO.js +91 -0
- package/dist/daemon-protocol-RLH2XM2K.js +19 -0
- package/dist/db-backup-CH2KG3JY.js +23 -0
- package/dist/devtools-PLJCAU53.js +8 -0
- package/dist/discord-G4BQJ2VX.js +7 -0
- package/dist/email-XMRNMFXR.js +137 -0
- package/dist/entity-boost-IOER4VRR.js +249 -0
- package/dist/exe-drift-2IJFQV43.js +68 -0
- package/dist/exe-export-Y2RFXGNB.js +71 -0
- package/dist/exe-import-LN6PE2YT.js +74 -0
- package/dist/exe-key-DHKBWLIT.js +571 -0
- package/dist/exe-org-F7LZRHFB.js +73 -0
- package/dist/factory-YTD23AW7.js +67 -0
- package/dist/fast-db-init-IIP6EGWT.js +7 -0
- package/dist/file-edit-FJVEO7XZ.js +8 -0
- package/dist/file-read-Q2NH3J35.js +8 -0
- package/dist/file-write-UNSFXNH4.js +8 -0
- package/dist/gateway/index.js +87 -16563
- package/dist/gateway-client-APTYJNCL.js +11 -0
- package/dist/git-staleness-I2BUEDE2.js +111 -0
- package/dist/git-task-sweep-UEOTXRW4.js +35 -0
- package/dist/glob-GRMXLDMB.js +7 -0
- package/dist/global-procedures-MFE5LGRF.js +19 -0
- package/dist/graph-query-657O5JYJ.js +25 -0
- package/dist/graph-rag-ZE5N7C65.js +28 -0
- package/dist/grep-NBV6SKYZ.js +7 -0
- package/dist/hooks/bug-report-worker.js +41 -10823
- package/dist/hooks/codex-stop-task-finalizer.js +38 -9055
- package/dist/hooks/commit-complete.js +32 -10914
- package/dist/hooks/error-recall.js +35 -7955
- package/dist/hooks/exe-heartbeat-hook.js +17 -327
- package/dist/hooks/ingest-worker.js +16 -565
- package/dist/hooks/ingest.js +98 -11436
- package/dist/hooks/instructions-loaded.js +25 -6209
- package/dist/hooks/notification.js +16 -6098
- package/dist/hooks/post-compact.js +34 -7041
- package/dist/hooks/post-tool-combined.js +27 -9751
- package/dist/hooks/pre-compact.js +38 -11130
- package/dist/hooks/pre-tool-use.js +61 -7233
- package/dist/hooks/prompt-submit.js +113 -13358
- package/dist/hooks/session-end.js +57 -11503
- package/dist/hooks/session-start.js +46 -9423
- package/dist/hooks/stop.js +70 -7210
- package/dist/hooks/subagent-stop.js +33 -6897
- package/dist/hooks/summary-worker.js +86 -8629
- package/dist/hooks-KD3N2D7P.js +9 -0
- package/dist/imessage-L7WKA2WD.js +7 -0
- package/dist/index.js +49 -19391
- package/dist/installer-5N24GXFB.js +32 -0
- package/dist/installer-QSD76YRY.js +291 -0
- package/dist/installer-ZPQUKHU5.js +331 -0
- package/dist/intercom-queue-PEEGAJ33.js +19 -0
- package/dist/key-backup-status-57UB74DU.js +41 -0
- package/dist/lib/agent-config.js +19 -224
- package/dist/lib/cloud-sync.js +44 -5398
- package/dist/lib/cloudflare-dns.js +7 -111
- package/dist/lib/config.js +18 -278
- package/dist/lib/consolidation.js +21 -1047
- package/dist/lib/crypto.js +7 -45
- package/dist/lib/database.js +20 -2932
- package/dist/lib/db-daemon-client.js +24 -460
- package/dist/lib/db.js +22 -2932
- package/dist/lib/device-registry.js +20 -2992
- package/dist/lib/embedder.js +14 -822
- package/dist/lib/employee-templates.js +21 -1102
- package/dist/lib/employees.js +30 -497
- package/dist/lib/error-detector.js +9 -149
- package/dist/lib/exe-daemon-client.js +16 -570
- package/dist/lib/exe-daemon.js +509 -18398
- package/dist/lib/file-grep.js +2 -0
- package/dist/lib/hybrid-search.js +18 -7674
- package/dist/lib/identity-templates.js +2 -0
- package/dist/lib/identity.js +13 -309
- package/dist/lib/keychain.js +9 -499
- package/dist/lib/license.js +21 -549
- package/dist/lib/messaging.js +35 -1631
- package/dist/lib/post-tool-memory.js +8 -370
- package/dist/lib/registry-proxy.js +8 -155
- package/dist/lib/reminders.js +11 -231
- package/dist/lib/runtime-table.js +7 -20
- package/dist/lib/schedules.js +16 -4614
- package/dist/lib/session-registry.js +9 -57
- package/dist/lib/session-wrappers.js +3 -127
- package/dist/lib/skill-learning.js +20 -2101
- package/dist/lib/status-brief.js +4 -368
- package/dist/lib/store.js +25 -5593
- package/dist/lib/task-router.js +12 -254
- package/dist/lib/tasks.js +44 -5801
- package/dist/lib/tmux-routing.js +45 -5804
- package/dist/lib/tmux-status.js +16 -248
- package/dist/lib/tmux-transport.js +6 -90
- package/dist/lib/token-spend.js +9 -337
- package/dist/lib/transport.js +8 -134
- package/dist/lib/ws-auth.js +6 -14
- package/dist/lib/ws-client.js +4 -22
- package/dist/license-gate-NXXDOOAQ.js +14 -0
- package/dist/mcp/register-tools.js +90 -32454
- package/dist/mcp/server.js +139 -32625
- package/dist/mcp/tools/complete-reminder.js +10 -237
- package/dist/mcp/tools/create-reminder.js +10 -222
- package/dist/mcp/tools/create-task.js +28 -6730
- package/dist/mcp/tools/deactivate-behavior.js +13 -476
- package/dist/mcp/tools/list-reminders.js +10 -232
- package/dist/mcp/tools/list-tasks.js +28 -1571
- package/dist/mcp/tools/send-message.js +25 -1672
- package/dist/mcp/tools/update-task.js +27 -6123
- package/dist/mcp-diagnostics-OOZ3TDPY.js +200 -0
- package/dist/memory-cards-RIL5BZHK.js +175 -0
- package/dist/memory-queue-JYIJ7TMQ.js +13 -0
- package/dist/memory-queue-client-PE7757HU.js +14 -0
- package/dist/notifications-CJFULBMN.js +42 -0
- package/dist/orchestration-phase-Y2TYNW5H.js +23 -0
- package/dist/orchestrator-RXIPT6PZ.js +29 -0
- package/dist/permission-presets-WI524OSE.js +25 -0
- package/dist/permissions-TYT6YMNZ.js +14 -0
- package/dist/pg-ssl-KFHNJPGO.js +7 -0
- package/dist/plan-limits-5WY5YK6X.js +27 -0
- package/dist/preferences-MW3BYTQ5.js +10 -0
- package/dist/project-name-PL2ZDWNU.js +11 -0
- package/dist/projection-worker-OY4WYZW6.js +661 -0
- package/dist/reranker-LFJ73RRS.js +19 -0
- package/dist/review-gate-E6K7QQE5.js +120 -0
- package/dist/runtime/index.js +82 -13408
- package/dist/self-query-router-IZANO632.js +83 -0
- package/dist/session-events-TQ6S5KQN.js +32 -0
- package/dist/session-kill-telemetry-46YIJWE2.js +30 -0
- package/dist/session-scope-3MYK3YRV.js +34 -0
- package/dist/setup-wizard-ZJ7UCRBP.js +12 -0
- package/dist/shard-manager-RCFZ6RXY.js +29 -0
- package/dist/signal-XN2C7PIG.js +7 -0
- package/dist/slack-Z77TFJCK.js +7 -0
- package/dist/task-enforcement-337DZI3I.js +252 -0
- package/dist/task-scanner-IMIR3T7T.js +11 -0
- package/dist/task-scope-MUU4KXE6.js +32 -0
- package/dist/tasks-crud-PMLK3MM4.js +53 -0
- package/dist/tasks-review-56LVKKNL.js +44 -0
- package/dist/telegram-CRPZO7JT.js +7 -0
- package/dist/telemetry-SPVGPKU7.js +9 -0
- package/dist/tool-capability-index-XNBSDCH3.js +10 -0
- package/dist/tool-gates-LGH6CSY4.js +14 -0
- package/dist/tool-gates-W7SZCYTX.js +13 -0
- package/dist/tool-registry-436LVVLY.js +11 -0
- package/dist/tool-telemetry-L77WNAYT.js +15 -0
- package/dist/tui/App.js +253 -14668
- package/dist/tui-data-SB45REYU.js +161 -0
- package/dist/update-check-LZGZISRJ.js +11 -0
- package/dist/webhook-VVP7KQVC.js +96 -0
- package/dist/whatsapp-J6QNDBGY.js +9 -0
- package/dist/whatsapp-accounts-RTJF3LM6.js +15 -0
- package/dist/wiki-client-EKCIKCVC.js +136 -0
- package/dist/worker-gate-UYNN62GJ.js +19 -0
- package/dist/worktree-4T4BPWPN.js +205 -0
- package/package.json +1 -1
|
@@ -0,0 +1,352 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getIdentity
|
|
3
|
+
} from "./chunk-PGW7TIV6.js";
|
|
4
|
+
import {
|
|
5
|
+
init_employees,
|
|
6
|
+
loadEmployeesSync
|
|
7
|
+
} from "./chunk-RUG3N6P4.js";
|
|
8
|
+
|
|
9
|
+
// src/lib/drift-probes.ts
|
|
10
|
+
init_employees();
|
|
11
|
+
var DRIFT_THRESHOLD = 80;
|
|
12
|
+
async function getRecentMemories(agentId, limit) {
|
|
13
|
+
try {
|
|
14
|
+
const { getClient, isInitialized } = await import("./lib/database.js");
|
|
15
|
+
if (!isInitialized()) return { count: 0, texts: [] };
|
|
16
|
+
const client = getClient();
|
|
17
|
+
const result = await client.execute({
|
|
18
|
+
sql: `SELECT text FROM memories WHERE agent_id = ? ORDER BY created_at DESC LIMIT ?`,
|
|
19
|
+
args: [agentId, limit]
|
|
20
|
+
});
|
|
21
|
+
const texts = result.rows.map((r) => String(r.text ?? ""));
|
|
22
|
+
return { count: texts.length, texts };
|
|
23
|
+
} catch {
|
|
24
|
+
return { count: 0, texts: [] };
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
async function getDecisionCount(agentId) {
|
|
28
|
+
try {
|
|
29
|
+
const { getClient, isInitialized } = await import("./lib/database.js");
|
|
30
|
+
if (!isInitialized()) return 0;
|
|
31
|
+
const client = getClient();
|
|
32
|
+
const result = await client.execute({
|
|
33
|
+
sql: `SELECT COUNT(*) as cnt FROM memories WHERE agent_id = ? AND memory_type = 'decision'`,
|
|
34
|
+
args: [agentId]
|
|
35
|
+
});
|
|
36
|
+
return Number(result.rows[0]?.cnt ?? 0);
|
|
37
|
+
} catch {
|
|
38
|
+
return 0;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
function probeContinuity(ctx) {
|
|
42
|
+
const memCount = ctx.recentMemoryCount;
|
|
43
|
+
let score;
|
|
44
|
+
let detail;
|
|
45
|
+
if (memCount >= 20) {
|
|
46
|
+
score = 95;
|
|
47
|
+
detail = `${memCount} recent memories \u2014 strong continuity`;
|
|
48
|
+
} else if (memCount >= 10) {
|
|
49
|
+
score = 85;
|
|
50
|
+
detail = `${memCount} recent memories \u2014 adequate continuity`;
|
|
51
|
+
} else if (memCount >= 3) {
|
|
52
|
+
score = 70;
|
|
53
|
+
detail = `${memCount} recent memories \u2014 limited continuity`;
|
|
54
|
+
} else if (memCount >= 1) {
|
|
55
|
+
score = 50;
|
|
56
|
+
detail = `${memCount} recent memories \u2014 weak continuity`;
|
|
57
|
+
} else {
|
|
58
|
+
score = 30;
|
|
59
|
+
detail = "No recent memories \u2014 agent may lack session context";
|
|
60
|
+
}
|
|
61
|
+
if (ctx.storedDecisionCount > 0) {
|
|
62
|
+
score = Math.min(100, score + 5);
|
|
63
|
+
detail += ` (+${ctx.storedDecisionCount} decisions)`;
|
|
64
|
+
}
|
|
65
|
+
return { axis: "continuity", score, detail };
|
|
66
|
+
}
|
|
67
|
+
function probeConsistency(ctx) {
|
|
68
|
+
if (!ctx.identityBody || ctx.recentMemoryTexts.length === 0) {
|
|
69
|
+
return {
|
|
70
|
+
axis: "consistency",
|
|
71
|
+
score: ctx.identityBody ? 80 : 50,
|
|
72
|
+
detail: ctx.identityBody ? "No memories to check against identity" : "No identity file \u2014 cannot assess consistency"
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
const identityTerms = extractKeyTerms(ctx.identityBody);
|
|
76
|
+
if (identityTerms.length === 0) {
|
|
77
|
+
return { axis: "consistency", score: 75, detail: "Identity has no extractable key terms" };
|
|
78
|
+
}
|
|
79
|
+
const memoryText = ctx.recentMemoryTexts.join(" ").toLowerCase();
|
|
80
|
+
let matched = 0;
|
|
81
|
+
for (const term of identityTerms) {
|
|
82
|
+
if (memoryText.includes(term.toLowerCase())) matched++;
|
|
83
|
+
}
|
|
84
|
+
const ratio = matched / identityTerms.length;
|
|
85
|
+
const score = Math.round(50 + ratio * 50);
|
|
86
|
+
const detail = `${matched}/${identityTerms.length} identity terms found in recent memories`;
|
|
87
|
+
return { axis: "consistency", score, detail };
|
|
88
|
+
}
|
|
89
|
+
function probeRoleFidelity(ctx) {
|
|
90
|
+
const flags = [];
|
|
91
|
+
if (!ctx.identityBody) {
|
|
92
|
+
return {
|
|
93
|
+
axis: "role-fidelity",
|
|
94
|
+
score: 40,
|
|
95
|
+
detail: `No identity file for ${ctx.agent.name}`
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
let score = 100;
|
|
99
|
+
const identityLower = ctx.identityBody.toLowerCase();
|
|
100
|
+
const rosterRole = ctx.agent.role.toLowerCase();
|
|
101
|
+
if (!identityLower.includes(rosterRole)) {
|
|
102
|
+
score -= 15;
|
|
103
|
+
flags.push(`Identity does not mention role "${ctx.agent.role}"`);
|
|
104
|
+
}
|
|
105
|
+
if (!identityLower.includes(ctx.agent.name.toLowerCase())) {
|
|
106
|
+
score -= 10;
|
|
107
|
+
flags.push(`Identity does not mention agent name "${ctx.agent.name}"`);
|
|
108
|
+
}
|
|
109
|
+
const hasScopeBoundary = identityLower.includes("do not") || identityLower.includes("don't") || identityLower.includes("not your") || identityLower.includes("outside your") || identityLower.includes("you do not");
|
|
110
|
+
if (!hasScopeBoundary) {
|
|
111
|
+
score -= 10;
|
|
112
|
+
flags.push("Identity lacks explicit scope boundaries (what NOT to do)");
|
|
113
|
+
}
|
|
114
|
+
if (ctx.agent.systemPrompt) {
|
|
115
|
+
const promptTerms = extractKeyTerms(ctx.agent.systemPrompt);
|
|
116
|
+
const identityText = ctx.identityBody.toLowerCase();
|
|
117
|
+
let promptMatched = 0;
|
|
118
|
+
for (const term of promptTerms.slice(0, 20)) {
|
|
119
|
+
if (identityText.includes(term.toLowerCase())) promptMatched++;
|
|
120
|
+
}
|
|
121
|
+
const promptRatio = promptTerms.length > 0 ? promptMatched / Math.min(promptTerms.length, 20) : 1;
|
|
122
|
+
if (promptRatio < 0.3) {
|
|
123
|
+
score -= 15;
|
|
124
|
+
flags.push("Identity content diverges significantly from roster systemPrompt");
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
if (ctx.recentMemoryTexts.length > 5) {
|
|
128
|
+
const roleKeywords = extractRoleKeywords(ctx.agent.role);
|
|
129
|
+
const memoryText = ctx.recentMemoryTexts.join(" ").toLowerCase();
|
|
130
|
+
let domainHits = 0;
|
|
131
|
+
for (const kw of roleKeywords) {
|
|
132
|
+
if (memoryText.includes(kw)) domainHits++;
|
|
133
|
+
}
|
|
134
|
+
const domainRatio = roleKeywords.length > 0 ? domainHits / roleKeywords.length : 1;
|
|
135
|
+
if (domainRatio < 0.2) {
|
|
136
|
+
score -= 10;
|
|
137
|
+
flags.push(`Recent work shows little overlap with ${ctx.agent.role} domain keywords`);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
score = Math.max(0, Math.min(100, score));
|
|
141
|
+
const detail = flags.length > 0 ? flags.join("; ") : "Identity aligns with roster definition";
|
|
142
|
+
return { axis: "role-fidelity", score, detail };
|
|
143
|
+
}
|
|
144
|
+
function probeWorldModel(ctx) {
|
|
145
|
+
if (!ctx.identityBody) {
|
|
146
|
+
return {
|
|
147
|
+
axis: "world-model",
|
|
148
|
+
score: 50,
|
|
149
|
+
detail: "No identity file \u2014 cannot assess org awareness"
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
const identityLower = ctx.identityBody.toLowerCase();
|
|
153
|
+
const flags = [];
|
|
154
|
+
let score = 100;
|
|
155
|
+
const otherAgents = ctx.allEmployees.filter((e) => e.name !== ctx.agent.name);
|
|
156
|
+
let mentionedAgents = 0;
|
|
157
|
+
for (const other of otherAgents) {
|
|
158
|
+
if (identityLower.includes(other.name.toLowerCase())) mentionedAgents++;
|
|
159
|
+
}
|
|
160
|
+
if (otherAgents.length > 0) {
|
|
161
|
+
const mentionRatio = mentionedAgents / otherAgents.length;
|
|
162
|
+
if (mentionRatio < 0.3) {
|
|
163
|
+
score -= 15;
|
|
164
|
+
flags.push(`Identity mentions ${mentionedAgents}/${otherAgents.length} team members`);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
const hasReportingChain = identityLower.includes("report") || identityLower.includes("manager") || identityLower.includes("coo") || identityLower.includes("coordinator");
|
|
168
|
+
if (!hasReportingChain) {
|
|
169
|
+
score -= 10;
|
|
170
|
+
flags.push("Identity does not reference reporting chain");
|
|
171
|
+
}
|
|
172
|
+
const orgTerms = ["exe-os", "exe-wiki", "exe-crm", "askexe"];
|
|
173
|
+
let orgMentions = 0;
|
|
174
|
+
for (const term of orgTerms) {
|
|
175
|
+
if (identityLower.includes(term)) orgMentions++;
|
|
176
|
+
}
|
|
177
|
+
if (orgMentions === 0) {
|
|
178
|
+
score -= 5;
|
|
179
|
+
flags.push("Identity does not reference any org products");
|
|
180
|
+
}
|
|
181
|
+
score = Math.max(0, Math.min(100, score));
|
|
182
|
+
const detail = flags.length > 0 ? flags.join("; ") : "Identity reflects org structure";
|
|
183
|
+
return { axis: "world-model", score, detail };
|
|
184
|
+
}
|
|
185
|
+
function extractKeyTerms(text) {
|
|
186
|
+
const stopwords = /* @__PURE__ */ new Set([
|
|
187
|
+
"the",
|
|
188
|
+
"and",
|
|
189
|
+
"for",
|
|
190
|
+
"are",
|
|
191
|
+
"but",
|
|
192
|
+
"not",
|
|
193
|
+
"you",
|
|
194
|
+
"all",
|
|
195
|
+
"any",
|
|
196
|
+
"can",
|
|
197
|
+
"had",
|
|
198
|
+
"her",
|
|
199
|
+
"was",
|
|
200
|
+
"one",
|
|
201
|
+
"our",
|
|
202
|
+
"out",
|
|
203
|
+
"has",
|
|
204
|
+
"his",
|
|
205
|
+
"how",
|
|
206
|
+
"its",
|
|
207
|
+
"may",
|
|
208
|
+
"new",
|
|
209
|
+
"now",
|
|
210
|
+
"old",
|
|
211
|
+
"see",
|
|
212
|
+
"way",
|
|
213
|
+
"who",
|
|
214
|
+
"did",
|
|
215
|
+
"get",
|
|
216
|
+
"got",
|
|
217
|
+
"let",
|
|
218
|
+
"say",
|
|
219
|
+
"she",
|
|
220
|
+
"too",
|
|
221
|
+
"use",
|
|
222
|
+
"with",
|
|
223
|
+
"this",
|
|
224
|
+
"that",
|
|
225
|
+
"from",
|
|
226
|
+
"they",
|
|
227
|
+
"been",
|
|
228
|
+
"have",
|
|
229
|
+
"will",
|
|
230
|
+
"your",
|
|
231
|
+
"what",
|
|
232
|
+
"when",
|
|
233
|
+
"make",
|
|
234
|
+
"like",
|
|
235
|
+
"just",
|
|
236
|
+
"over",
|
|
237
|
+
"such",
|
|
238
|
+
"take",
|
|
239
|
+
"than",
|
|
240
|
+
"them",
|
|
241
|
+
"very",
|
|
242
|
+
"some",
|
|
243
|
+
"could",
|
|
244
|
+
"into",
|
|
245
|
+
"then",
|
|
246
|
+
"more",
|
|
247
|
+
"also",
|
|
248
|
+
"after",
|
|
249
|
+
"should",
|
|
250
|
+
"would",
|
|
251
|
+
"about",
|
|
252
|
+
"their",
|
|
253
|
+
"which",
|
|
254
|
+
"these",
|
|
255
|
+
"other",
|
|
256
|
+
"every",
|
|
257
|
+
"does",
|
|
258
|
+
"being",
|
|
259
|
+
"those",
|
|
260
|
+
"never",
|
|
261
|
+
"before",
|
|
262
|
+
"through"
|
|
263
|
+
]);
|
|
264
|
+
const words = text.replace(/[^\w\s-]/g, " ").split(/\s+/).filter((w) => w.length > 3 && !stopwords.has(w.toLowerCase()));
|
|
265
|
+
return [...new Set(words.map((w) => w.toLowerCase()))].slice(0, 50);
|
|
266
|
+
}
|
|
267
|
+
function extractRoleKeywords(role) {
|
|
268
|
+
const roleKeywordMap = {
|
|
269
|
+
"coo": ["coordinate", "review", "status", "team", "task", "priority", "dispatch"],
|
|
270
|
+
"cto": ["architecture", "code", "technical", "system", "design", "review", "security"],
|
|
271
|
+
"cmo": ["marketing", "brand", "content", "design", "seo", "social", "campaign"],
|
|
272
|
+
"principal engineer": ["code", "implement", "test", "fix", "feature", "refactor", "build"],
|
|
273
|
+
"staff code reviewer": ["review", "code", "quality", "issue", "fix", "pattern"],
|
|
274
|
+
"content production specialist": ["video", "image", "render", "content", "produce", "media"],
|
|
275
|
+
"ai product lead": ["competitive", "analysis", "feature", "product", "research", "repo"]
|
|
276
|
+
};
|
|
277
|
+
const normalized = role.toLowerCase();
|
|
278
|
+
return roleKeywordMap[normalized] ?? normalized.split(/\s+/).filter((w) => w.length > 2);
|
|
279
|
+
}
|
|
280
|
+
async function runDriftProbes(options = {}) {
|
|
281
|
+
const employees = loadEmployeesSync();
|
|
282
|
+
const targetAgents = options.agentId ? employees.filter((e) => e.name === options.agentId) : employees;
|
|
283
|
+
if (targetAgents.length === 0) {
|
|
284
|
+
return [];
|
|
285
|
+
}
|
|
286
|
+
const axes = options.axes ?? ["continuity", "consistency", "role-fidelity", "world-model"];
|
|
287
|
+
const results = [];
|
|
288
|
+
for (const agent of targetAgents) {
|
|
289
|
+
const identity = getIdentity(agent.name);
|
|
290
|
+
const { count: memCount, texts: memTexts } = await getRecentMemories(agent.name, 50);
|
|
291
|
+
const decisionCount = await getDecisionCount(agent.name);
|
|
292
|
+
const ctx = {
|
|
293
|
+
agent,
|
|
294
|
+
identityBody: identity?.body ?? null,
|
|
295
|
+
identityRole: identity?.frontmatter.role ?? null,
|
|
296
|
+
recentMemoryCount: memCount,
|
|
297
|
+
recentMemoryTexts: memTexts,
|
|
298
|
+
storedDecisionCount: decisionCount,
|
|
299
|
+
allEmployees: employees
|
|
300
|
+
};
|
|
301
|
+
const probes = [];
|
|
302
|
+
const probeFns = {
|
|
303
|
+
continuity: probeContinuity,
|
|
304
|
+
consistency: probeConsistency,
|
|
305
|
+
"role-fidelity": probeRoleFidelity,
|
|
306
|
+
"world-model": probeWorldModel
|
|
307
|
+
};
|
|
308
|
+
for (const axis of axes) {
|
|
309
|
+
probes.push(probeFns[axis](ctx));
|
|
310
|
+
}
|
|
311
|
+
const scores = {
|
|
312
|
+
continuity: 0,
|
|
313
|
+
consistency: 0,
|
|
314
|
+
"role-fidelity": 0,
|
|
315
|
+
"world-model": 0
|
|
316
|
+
};
|
|
317
|
+
for (const probe of probes) {
|
|
318
|
+
scores[probe.axis] = probe.score;
|
|
319
|
+
}
|
|
320
|
+
const weights = {
|
|
321
|
+
continuity: 0.2,
|
|
322
|
+
consistency: 0.2,
|
|
323
|
+
"role-fidelity": 0.35,
|
|
324
|
+
"world-model": 0.25
|
|
325
|
+
};
|
|
326
|
+
let weightedSum = 0;
|
|
327
|
+
let weightTotal = 0;
|
|
328
|
+
for (const axis of axes) {
|
|
329
|
+
weightedSum += scores[axis] * weights[axis];
|
|
330
|
+
weightTotal += weights[axis];
|
|
331
|
+
}
|
|
332
|
+
const overall = Math.round(weightTotal > 0 ? weightedSum / weightTotal : 0);
|
|
333
|
+
const flags = [];
|
|
334
|
+
for (const probe of probes) {
|
|
335
|
+
if (probe.score < DRIFT_THRESHOLD) {
|
|
336
|
+
flags.push(`${probe.axis} at ${probe.score}% \u2014 ${probe.detail}`);
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
results.push({
|
|
340
|
+
agent: agent.name,
|
|
341
|
+
scores,
|
|
342
|
+
overall,
|
|
343
|
+
drifting: flags.length > 0,
|
|
344
|
+
flags
|
|
345
|
+
});
|
|
346
|
+
}
|
|
347
|
+
return results;
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
export {
|
|
351
|
+
runDriftProbes
|
|
352
|
+
};
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
import {
|
|
2
|
+
EXE_AI_DIR,
|
|
3
|
+
init_config
|
|
4
|
+
} from "./chunk-KVPG5UT6.js";
|
|
5
|
+
import {
|
|
6
|
+
enforcePrivateFileSync,
|
|
7
|
+
ensurePrivateDirSync,
|
|
8
|
+
init_secure_files
|
|
9
|
+
} from "./chunk-X2IMCCM5.js";
|
|
10
|
+
import {
|
|
11
|
+
__esm
|
|
12
|
+
} from "./chunk-KFQGP6VL.js";
|
|
13
|
+
|
|
14
|
+
// src/lib/daemon-auth.ts
|
|
15
|
+
import crypto from "crypto";
|
|
16
|
+
import path from "path";
|
|
17
|
+
import { existsSync, readFileSync, writeFileSync } from "fs";
|
|
18
|
+
function normalizeToken(token) {
|
|
19
|
+
if (!token) return null;
|
|
20
|
+
const trimmed = token.trim();
|
|
21
|
+
return trimmed.length > 0 ? trimmed : null;
|
|
22
|
+
}
|
|
23
|
+
function readDaemonToken() {
|
|
24
|
+
try {
|
|
25
|
+
if (!existsSync(DAEMON_TOKEN_PATH)) return null;
|
|
26
|
+
return normalizeToken(readFileSync(DAEMON_TOKEN_PATH, "utf8"));
|
|
27
|
+
} catch {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
function ensureDaemonToken(seed) {
|
|
32
|
+
const existing = readDaemonToken();
|
|
33
|
+
if (existing) return existing;
|
|
34
|
+
const token = normalizeToken(seed) ?? crypto.randomBytes(32).toString("hex");
|
|
35
|
+
ensurePrivateDirSync(EXE_AI_DIR);
|
|
36
|
+
writeFileSync(DAEMON_TOKEN_PATH, `${token}
|
|
37
|
+
`, "utf8");
|
|
38
|
+
enforcePrivateFileSync(DAEMON_TOKEN_PATH);
|
|
39
|
+
return token;
|
|
40
|
+
}
|
|
41
|
+
var DAEMON_TOKEN_PATH;
|
|
42
|
+
var init_daemon_auth = __esm({
|
|
43
|
+
"src/lib/daemon-auth.ts"() {
|
|
44
|
+
init_config();
|
|
45
|
+
init_secure_files();
|
|
46
|
+
DAEMON_TOKEN_PATH = path.join(EXE_AI_DIR, "exed.token");
|
|
47
|
+
}
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
export {
|
|
51
|
+
DAEMON_TOKEN_PATH,
|
|
52
|
+
readDaemonToken,
|
|
53
|
+
ensureDaemonToken,
|
|
54
|
+
init_daemon_auth
|
|
55
|
+
};
|
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
import {
|
|
2
|
+
init_task_scope,
|
|
3
|
+
sessionScopeFilter
|
|
4
|
+
} from "./chunk-N7L33TED.js";
|
|
5
|
+
|
|
6
|
+
// src/lib/git-task-sweep.ts
|
|
7
|
+
init_task_scope();
|
|
8
|
+
import { execSync } from "child_process";
|
|
9
|
+
var DEFAULT_COMMIT_LIMIT = 50;
|
|
10
|
+
var DEFAULT_STALE_MINUTES = 10;
|
|
11
|
+
var AUTO_ESCALATE_THRESHOLD = 0.6;
|
|
12
|
+
var EXACT_UUID_SCORE = 1;
|
|
13
|
+
var TITLE_KEYWORD_SCORE = 0.5;
|
|
14
|
+
var FILE_PATH_SCORE = 0.3;
|
|
15
|
+
var MIN_KEYWORD_OVERLAP = 3;
|
|
16
|
+
var STOP_WORDS = /* @__PURE__ */ new Set([
|
|
17
|
+
"a",
|
|
18
|
+
"an",
|
|
19
|
+
"the",
|
|
20
|
+
"and",
|
|
21
|
+
"or",
|
|
22
|
+
"but",
|
|
23
|
+
"in",
|
|
24
|
+
"on",
|
|
25
|
+
"at",
|
|
26
|
+
"to",
|
|
27
|
+
"for",
|
|
28
|
+
"of",
|
|
29
|
+
"with",
|
|
30
|
+
"by",
|
|
31
|
+
"from",
|
|
32
|
+
"is",
|
|
33
|
+
"it",
|
|
34
|
+
"as",
|
|
35
|
+
"be",
|
|
36
|
+
"was",
|
|
37
|
+
"are",
|
|
38
|
+
"this",
|
|
39
|
+
"that",
|
|
40
|
+
"not",
|
|
41
|
+
"no",
|
|
42
|
+
"if",
|
|
43
|
+
"so",
|
|
44
|
+
"do",
|
|
45
|
+
"up"
|
|
46
|
+
]);
|
|
47
|
+
function extractKeywords(text) {
|
|
48
|
+
return text.toLowerCase().replace(/[^a-z0-9\s-]/g, " ").split(/\s+/).filter((w) => w.length >= 3 && !STOP_WORDS.has(w));
|
|
49
|
+
}
|
|
50
|
+
function matchScore(task, commitMessage, changedFiles) {
|
|
51
|
+
if (task.id.length >= 8 && commitMessage.includes(task.id)) {
|
|
52
|
+
return EXACT_UUID_SCORE;
|
|
53
|
+
}
|
|
54
|
+
let score = 0;
|
|
55
|
+
const titleWords = extractKeywords(task.title);
|
|
56
|
+
const commitWords = new Set(extractKeywords(commitMessage));
|
|
57
|
+
const overlap = titleWords.filter((w) => commitWords.has(w));
|
|
58
|
+
if (overlap.length >= MIN_KEYWORD_OVERLAP) {
|
|
59
|
+
score += TITLE_KEYWORD_SCORE;
|
|
60
|
+
}
|
|
61
|
+
if (task.context && changedFiles.length > 0) {
|
|
62
|
+
const contextLower = task.context.toLowerCase();
|
|
63
|
+
const hasFileMatch = changedFiles.some(
|
|
64
|
+
(f) => contextLower.includes(f.toLowerCase())
|
|
65
|
+
);
|
|
66
|
+
if (hasFileMatch) {
|
|
67
|
+
score += FILE_PATH_SCORE;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
return score;
|
|
71
|
+
}
|
|
72
|
+
function getRecentCommits(limit = DEFAULT_COMMIT_LIMIT) {
|
|
73
|
+
try {
|
|
74
|
+
const SEPARATOR = "<<SEP>>";
|
|
75
|
+
const output = execSync(
|
|
76
|
+
`git log --format="%h${SEPARATOR}%s${SEPARATOR}%aI" --name-only -n ${limit} -z`,
|
|
77
|
+
{ encoding: "utf8", timeout: 1e4 }
|
|
78
|
+
);
|
|
79
|
+
const entries = output.split("\0").filter(Boolean);
|
|
80
|
+
const commits = [];
|
|
81
|
+
let current = null;
|
|
82
|
+
for (const entry of entries) {
|
|
83
|
+
if (entry.includes(SEPARATOR)) {
|
|
84
|
+
const lines = entry.split("\n");
|
|
85
|
+
const headerLine = lines[0];
|
|
86
|
+
const parts = headerLine.split(SEPARATOR);
|
|
87
|
+
if (parts.length >= 3) {
|
|
88
|
+
if (current) commits.push(current);
|
|
89
|
+
current = {
|
|
90
|
+
hash: parts[0],
|
|
91
|
+
message: parts[1],
|
|
92
|
+
files: lines.slice(1).filter(Boolean),
|
|
93
|
+
date: new Date(parts[2])
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
} else if (current) {
|
|
97
|
+
const files = entry.split("\n").filter(Boolean);
|
|
98
|
+
current.files.push(...files);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
if (current) commits.push(current);
|
|
102
|
+
return commits;
|
|
103
|
+
} catch {
|
|
104
|
+
return [];
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
function isStale(updatedAt, staleMinutes) {
|
|
108
|
+
const updated = new Date(updatedAt).getTime();
|
|
109
|
+
const threshold = Date.now() - staleMinutes * 6e4;
|
|
110
|
+
return updated < threshold;
|
|
111
|
+
}
|
|
112
|
+
function findBestMatch(task, commits) {
|
|
113
|
+
let best = null;
|
|
114
|
+
for (const commit of commits) {
|
|
115
|
+
const score = matchScore(task, commit.message, commit.files);
|
|
116
|
+
if (score >= AUTO_ESCALATE_THRESHOLD && (!best || score > best.score)) {
|
|
117
|
+
best = { commit, score };
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
return best;
|
|
121
|
+
}
|
|
122
|
+
async function sweepTasks(projectName, options = {}) {
|
|
123
|
+
const commitLimit = options.commitLimit ?? DEFAULT_COMMIT_LIMIT;
|
|
124
|
+
const staleMinutes = options.staleMinutes ?? DEFAULT_STALE_MINUTES;
|
|
125
|
+
const dryRun = options.dryRun ?? false;
|
|
126
|
+
const result = { escalated: [], unchanged: 0, errors: [] };
|
|
127
|
+
const commits = getRecentCommits(commitLimit);
|
|
128
|
+
if (commits.length === 0) {
|
|
129
|
+
result.errors.push("No git commits found (not a git repo or empty history)");
|
|
130
|
+
return result;
|
|
131
|
+
}
|
|
132
|
+
let tasks;
|
|
133
|
+
try {
|
|
134
|
+
const { initStore } = await import("./lib/store.js");
|
|
135
|
+
await initStore();
|
|
136
|
+
const { getClient } = await import("./lib/database.js");
|
|
137
|
+
const client = getClient();
|
|
138
|
+
const conditions = ["status = 'in_progress'"];
|
|
139
|
+
const args = [];
|
|
140
|
+
if (projectName) {
|
|
141
|
+
conditions.push("project_name = ?");
|
|
142
|
+
args.push(projectName);
|
|
143
|
+
}
|
|
144
|
+
const swScope = sessionScopeFilter();
|
|
145
|
+
if (swScope.sql) {
|
|
146
|
+
conditions.push("(session_scope IS NULL OR session_scope = ?)");
|
|
147
|
+
args.push(...swScope.args);
|
|
148
|
+
}
|
|
149
|
+
const queryResult = await client.execute({
|
|
150
|
+
sql: `SELECT id, title, assigned_to, project_name, status, updated_at, context
|
|
151
|
+
FROM tasks WHERE ${conditions.join(" AND ")}
|
|
152
|
+
ORDER BY updated_at ASC`,
|
|
153
|
+
args
|
|
154
|
+
});
|
|
155
|
+
tasks = queryResult.rows.map((r) => ({
|
|
156
|
+
id: String(r.id),
|
|
157
|
+
title: String(r.title),
|
|
158
|
+
assignedTo: String(r.assigned_to),
|
|
159
|
+
projectName: String(r.project_name),
|
|
160
|
+
status: String(r.status),
|
|
161
|
+
updatedAt: String(r.updated_at),
|
|
162
|
+
context: r.context ? String(r.context) : void 0
|
|
163
|
+
}));
|
|
164
|
+
} catch (err) {
|
|
165
|
+
result.errors.push(`DB query failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
166
|
+
return result;
|
|
167
|
+
}
|
|
168
|
+
if (tasks.length === 0) {
|
|
169
|
+
return result;
|
|
170
|
+
}
|
|
171
|
+
for (const task of tasks) {
|
|
172
|
+
if (!isStale(task.updatedAt, staleMinutes)) {
|
|
173
|
+
result.unchanged++;
|
|
174
|
+
continue;
|
|
175
|
+
}
|
|
176
|
+
const match = findBestMatch(task, commits);
|
|
177
|
+
if (!match) {
|
|
178
|
+
result.unchanged++;
|
|
179
|
+
continue;
|
|
180
|
+
}
|
|
181
|
+
if (!dryRun) {
|
|
182
|
+
try {
|
|
183
|
+
const { updateTaskStatus } = await import("./tasks-crud-PMLK3MM4.js");
|
|
184
|
+
await updateTaskStatus({
|
|
185
|
+
taskId: task.id,
|
|
186
|
+
status: "needs_review",
|
|
187
|
+
result: `Auto-escalated by git-sweep: matching commit ${match.commit.hash} found (score: ${match.score.toFixed(2)})`
|
|
188
|
+
});
|
|
189
|
+
} catch (err) {
|
|
190
|
+
result.errors.push(
|
|
191
|
+
`Failed to escalate task ${task.id}: ${err instanceof Error ? err.message : String(err)}`
|
|
192
|
+
);
|
|
193
|
+
continue;
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
result.escalated.push({
|
|
197
|
+
taskId: task.id,
|
|
198
|
+
title: task.title,
|
|
199
|
+
matchedCommit: match.commit.hash,
|
|
200
|
+
score: match.score
|
|
201
|
+
});
|
|
202
|
+
process.stderr.write(
|
|
203
|
+
`[git-sweep] ${dryRun ? "WOULD escalate" : "Escalated"} task ${task.id} \u2192 commit ${match.commit.hash} (score: ${match.score.toFixed(2)})
|
|
204
|
+
`
|
|
205
|
+
);
|
|
206
|
+
}
|
|
207
|
+
return result;
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
export {
|
|
211
|
+
extractKeywords,
|
|
212
|
+
matchScore,
|
|
213
|
+
getRecentCommits,
|
|
214
|
+
findBestMatch,
|
|
215
|
+
sweepTasks
|
|
216
|
+
};
|