@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.
Files changed (373) hide show
  1. package/dist/active-agent-3FDJYREJ.js +22 -0
  2. package/dist/active-agent-WC7YK3L5.js +23 -0
  3. package/dist/agent-context-CDWUZEXY.js +11 -0
  4. package/dist/agent-loop-G4VUR5ME.js +10 -0
  5. package/dist/agentic-ontology-2VYNTMNO.js +25 -0
  6. package/dist/backfill-metadata-OJWOCLSZ.js +601 -0
  7. package/dist/background-jobs-LGRDXGXC.js +25 -0
  8. package/dist/bash-Y5PEW5UQ.js +8 -0
  9. package/dist/behaviors-CSB2YLDT.js +22 -0
  10. package/dist/bin/age-ontology-load.js +10 -258
  11. package/dist/bin/agentic-ontology-backfill.js +23 -5296
  12. package/dist/bin/agentic-reflection-backfill.js +19 -4739
  13. package/dist/bin/agentic-semantic-label.js +28 -4723
  14. package/dist/bin/backfill-conversations.js +42 -5543
  15. package/dist/bin/backfill-responses.js +41 -5542
  16. package/dist/bin/backfill-vectors.js +43 -4795
  17. package/dist/bin/bulk-sync-postgres.js +22 -5313
  18. package/dist/bin/cc-doctor.js +10 -602
  19. package/dist/bin/cleanup-stale-review-tasks.js +32 -6803
  20. package/dist/bin/cli.js +169 -38993
  21. package/dist/bin/exe-agent-config.js +21 -253
  22. package/dist/bin/exe-agent.js +63 -2429
  23. package/dist/bin/exe-assign.js +34 -5810
  24. package/dist/bin/exe-boot.js +180 -12726
  25. package/dist/bin/exe-call.js +48 -1409
  26. package/dist/bin/exe-cloud.js +64 -8516
  27. package/dist/bin/exe-dispatch.js +35 -10818
  28. package/dist/bin/exe-doctor.js +24 -7537
  29. package/dist/bin/exe-export-behaviors.js +18 -6122
  30. package/dist/bin/exe-forget.js +22 -6274
  31. package/dist/bin/exe-gateway.js +62 -15286
  32. package/dist/bin/exe-healthcheck.js +9 -602
  33. package/dist/bin/exe-heartbeat.js +51 -6898
  34. package/dist/bin/exe-kill.js +30 -5961
  35. package/dist/bin/exe-launch-agent.js +101 -6788
  36. package/dist/bin/exe-new-employee.js +42 -3774
  37. package/dist/bin/exe-pending-messages.js +28 -6833
  38. package/dist/bin/exe-pending-notifications.js +31 -6869
  39. package/dist/bin/exe-pending-reviews.js +38 -6910
  40. package/dist/bin/exe-rename.js +57 -5766
  41. package/dist/bin/exe-review.js +29 -5948
  42. package/dist/bin/exe-search.js +22 -7692
  43. package/dist/bin/exe-session-cleanup.js +52 -11147
  44. package/dist/bin/exe-settings.js +35 -524
  45. package/dist/bin/exe-start-codex.js +70 -6641
  46. package/dist/bin/exe-start-opencode.js +54 -6584
  47. package/dist/bin/exe-status.js +26 -7081
  48. package/dist/bin/exe-support.js +10 -546
  49. package/dist/bin/exe-team.js +18 -5942
  50. package/dist/bin/git-sweep.js +26 -10909
  51. package/dist/bin/graph-backfill.js +24 -5979
  52. package/dist/bin/graph-export.js +31 -6128
  53. package/dist/bin/graph-layer-benchmark.js +4 -8
  54. package/dist/bin/install.js +47 -2322
  55. package/dist/bin/intercom-check.js +25 -11501
  56. package/dist/bin/list-providers.js +4 -18
  57. package/dist/bin/postgres-agentic-reflection-backfill.js +14 -357
  58. package/dist/bin/postgres-agentic-semantic-backfill.js +14 -404
  59. package/dist/bin/pre-publish.js +4 -298
  60. package/dist/bin/registry-proxy.js +8 -172
  61. package/dist/bin/scan-tasks.js +47 -10881
  62. package/dist/bin/setup.js +6 -9492
  63. package/dist/bin/shard-migrate.js +30 -5149
  64. package/dist/bin/stack-update.js +75 -203
  65. package/dist/bin/update.js +20 -654
  66. package/dist/capacity-monitor-3GZIHLB4.js +46 -0
  67. package/dist/catchup-brief-67QDCINS.js +148 -0
  68. package/dist/chunk-2QD7KBJN.js +664 -0
  69. package/dist/chunk-2WGSIVI4.js +149 -0
  70. package/dist/chunk-35TVNLJI.js +57 -0
  71. package/dist/chunk-3SYV36U5.js +47 -0
  72. package/dist/chunk-46SQTBQW.js +207 -0
  73. package/dist/chunk-4CMVLKEI.js +171 -0
  74. package/dist/chunk-4FLUOZTR.js +116 -0
  75. package/dist/chunk-4NYQAS33.js +17 -0
  76. package/dist/chunk-4QNPADWL.js +902 -0
  77. package/dist/chunk-4Y6QZBEO.js +58 -0
  78. package/dist/chunk-557C2IGL.js +0 -0
  79. package/dist/chunk-55S56YDO.js +305 -0
  80. package/dist/chunk-5BS5ELMC.js +85 -0
  81. package/dist/chunk-5DHM77JE.js +256 -0
  82. package/dist/chunk-5QTWQKNF.js +137 -0
  83. package/dist/chunk-6A7P74QV.js +53 -0
  84. package/dist/chunk-6BFIFI5F.js +45 -0
  85. package/dist/chunk-6KFICEAT.js +20 -0
  86. package/dist/chunk-6LKDJ5WX.js +317 -0
  87. package/dist/chunk-6M5II4EV.js +148 -0
  88. package/dist/chunk-6QEFN3DV.js +302 -0
  89. package/dist/chunk-6WTMJAG3.js +171 -0
  90. package/dist/chunk-6Y4B3QF6.js +19 -0
  91. package/dist/chunk-7RK3AYJE.js +163 -0
  92. package/dist/chunk-7TCP2XTP.js +390 -0
  93. package/dist/chunk-A2NZP64U.js +2307 -0
  94. package/dist/chunk-AFYIYM2K.js +322 -0
  95. package/dist/chunk-AJWFPKQS.js +573 -0
  96. package/dist/chunk-AVE2B4DQ.js +140 -0
  97. package/dist/chunk-B4TT2QQA.js +87 -0
  98. package/dist/chunk-BCMOO4OQ.js +1066 -0
  99. package/dist/chunk-BU3ID2WT.js +432 -0
  100. package/dist/chunk-CFIM43JH.js +862 -0
  101. package/dist/chunk-CURJUF3W.js +66 -0
  102. package/dist/chunk-DC4ZD76Y.js +288 -0
  103. package/dist/chunk-DL52JPGF.js +706 -0
  104. package/dist/chunk-DRSC6ZRA.js +36 -0
  105. package/dist/chunk-E2Q4Y22A.js +86 -0
  106. package/dist/chunk-ECGTESAP.js +236 -0
  107. package/dist/chunk-EHOJUXYG.js +365 -0
  108. package/dist/chunk-EIT6J37V.js +119 -0
  109. package/dist/chunk-ELE2C7L4.js +31 -0
  110. package/dist/chunk-EQSQ4HBW.js +9 -0
  111. package/dist/chunk-FWFFZGSC.js +94 -0
  112. package/dist/chunk-GCNWCYJI.js +249 -0
  113. package/dist/chunk-GGTUUI7J.js +6 -0
  114. package/dist/chunk-GU56L7OY.js +223 -0
  115. package/dist/chunk-HF25XPZE.js +83 -0
  116. package/dist/chunk-HOKOOBDL.js +52 -0
  117. package/dist/chunk-HP43GLZ6.js +89 -0
  118. package/dist/chunk-HSRM6PNM.js +682 -0
  119. package/dist/chunk-HVRTW3OP.js +164 -0
  120. package/dist/chunk-IHDHL5ML.js +125 -0
  121. package/dist/chunk-IHF5RXJM.js +168 -0
  122. package/dist/chunk-IJ73YE73.js +409 -0
  123. package/dist/chunk-INO4ZD5Y.js +162 -0
  124. package/dist/chunk-IRQTM7DY.js +475 -0
  125. package/dist/chunk-JD4FAPYG.js +1208 -0
  126. package/dist/chunk-JN44LQ53.js +316 -0
  127. package/dist/chunk-JQ6TLNIV.js +133 -0
  128. package/dist/chunk-JT76KRZQ.js +35 -0
  129. package/dist/chunk-K6OEWZTB.js +262 -0
  130. package/dist/chunk-KCUYRZSX.js +72 -0
  131. package/dist/chunk-KFQGP6VL.js +33 -0
  132. package/dist/chunk-KRLOFIPI.js +83 -0
  133. package/dist/chunk-KVPG5UT6.js +305 -0
  134. package/dist/chunk-L3TB7CC3.js +170 -0
  135. package/dist/chunk-L4X5EUHR.js +283 -0
  136. package/dist/chunk-LAN26C3W.js +655 -0
  137. package/dist/chunk-LPGBKR4H.js +54 -0
  138. package/dist/chunk-LVNMOVGD.js +132 -0
  139. package/dist/chunk-LX5WEEZ7.js +157 -0
  140. package/dist/chunk-MBH42QNH.js +295 -0
  141. package/dist/chunk-MFO7MNSD.js +102 -0
  142. package/dist/chunk-MILTNDZH.js +736 -0
  143. package/dist/chunk-MJAE4NB7.js +1090 -0
  144. package/dist/chunk-MLCUPTP7.js +132 -0
  145. package/dist/chunk-MPURH6JI.js +445 -0
  146. package/dist/chunk-MVBWW2EO.js +89 -0
  147. package/dist/chunk-MZIGUUDQ.js +42 -0
  148. package/dist/chunk-N3DTSVKE.js +157 -0
  149. package/dist/chunk-N5RRQOAC.js +134 -0
  150. package/dist/chunk-N7L33TED.js +1664 -0
  151. package/dist/chunk-NDYSNVI5.js +122 -0
  152. package/dist/chunk-NEAHEC5G.js +155 -0
  153. package/dist/chunk-NGVOA6ZQ.js +84 -0
  154. package/dist/chunk-NHK5KW6Y.js +193 -0
  155. package/dist/chunk-NMQKGFEP.js +458 -0
  156. package/dist/chunk-NPT4EHZK.js +352 -0
  157. package/dist/chunk-NXFJHTED.js +55 -0
  158. package/dist/chunk-NZLCDI6Y.js +216 -0
  159. package/dist/chunk-O377P7GM.js +100 -0
  160. package/dist/chunk-O4VFZRA5.js +33 -0
  161. package/dist/chunk-O7YD7JYE.js +237 -0
  162. package/dist/chunk-OFK72LTZ.js +81 -0
  163. package/dist/chunk-ORCCI2VV.js +140 -0
  164. package/dist/chunk-OVQR4KMC.js +579 -0
  165. package/dist/chunk-P3CVUMEH.js +54 -0
  166. package/dist/chunk-P4HIF72O.js +30 -0
  167. package/dist/chunk-PEDLSJBL.js +1039 -0
  168. package/dist/chunk-PFICKZZ5.js +219 -0
  169. package/dist/chunk-PGW7TIV6.js +142 -0
  170. package/dist/chunk-PJHHFJAN.js +63 -0
  171. package/dist/chunk-PP4SNOJ5.js +471 -0
  172. package/dist/chunk-PRKVT4KN.js +90 -0
  173. package/dist/chunk-Q2O4VJHH.js +305 -0
  174. package/dist/chunk-Q36CMZIX.js +38 -0
  175. package/dist/chunk-Q3V7K4ME.js +149 -0
  176. package/dist/chunk-QGYPJGHB.js +81 -0
  177. package/dist/chunk-R23EMFJV.js +198 -0
  178. package/dist/chunk-R4M6XCMU.js +39 -0
  179. package/dist/chunk-R4WMF32C.js +81 -0
  180. package/dist/chunk-RJT7H2KR.js +349 -0
  181. package/dist/chunk-RJZMRZNV.js +280 -0
  182. package/dist/chunk-RMFDPCA2.js +72 -0
  183. package/dist/chunk-RUG3N6P4.js +292 -0
  184. package/dist/chunk-RWO327FJ.js +84 -0
  185. package/dist/chunk-SG7EFHQ7.js +108 -0
  186. package/dist/chunk-SH45SJQW.js +0 -0
  187. package/dist/chunk-SPKE5ANC.js +42 -0
  188. package/dist/chunk-SVMFVPBF.js +104 -0
  189. package/dist/chunk-SVXDCELZ.js +163 -0
  190. package/dist/chunk-TMEE7AT7.js +72 -0
  191. package/dist/chunk-TWYEE4FB.js +125 -0
  192. package/dist/chunk-TYLF4LSW.js +110 -0
  193. package/dist/chunk-V4TZI6EO.js +60 -0
  194. package/dist/chunk-VFZTXKQI.js +372 -0
  195. package/dist/chunk-VQ4K2MLM.js +199 -0
  196. package/dist/chunk-VU447UAO.js +173 -0
  197. package/dist/chunk-WMC3TXJF.js +50 -0
  198. package/dist/chunk-WPOA7XAE.js +84 -0
  199. package/dist/chunk-WZQ4CPRG.js +64 -0
  200. package/dist/chunk-X2IMCCM5.js +49 -0
  201. package/dist/chunk-X2Z5GT3V.js +118 -0
  202. package/dist/chunk-XEKO37NM.js +43 -0
  203. package/dist/chunk-XUHFQHGZ.js +505 -0
  204. package/dist/chunk-YGAAZN3E.js +47 -0
  205. package/dist/chunk-YKOW7KMK.js +558 -0
  206. package/dist/chunk-YOQKMB6N.js +145 -0
  207. package/dist/chunk-YZFZDJWZ.js +107 -0
  208. package/dist/chunk-ZBY4AUD4.js +10342 -0
  209. package/dist/chunk-ZOJYPNCM.js +127 -0
  210. package/dist/chunk-ZS7MGQSD.js +1494 -0
  211. package/dist/chunk-ZTZQ5A6B.js +191 -0
  212. package/dist/code-context-index-HXGC4IJL.js +28 -0
  213. package/dist/content-extractor-UVZXRIPC.js +8 -0
  214. package/dist/conversation-entity-extractor-HKSNDF4L.js +113 -0
  215. package/dist/conversation-wiki-populator-NKDAQEZ5.js +105 -0
  216. package/dist/crdt-sync-T7MHVQCS.js +33 -0
  217. package/dist/crm-bridge-Q5JBVJ6P.js +19 -0
  218. package/dist/crm-webhook-OAFTLJYA.js +10 -0
  219. package/dist/cto-delegation-gate-XWHZCUJH.js +208 -0
  220. package/dist/daemon-auth-GWUXTUYO.js +15 -0
  221. package/dist/daemon-orchestration-2C3TVOHO.js +91 -0
  222. package/dist/daemon-protocol-RLH2XM2K.js +19 -0
  223. package/dist/db-backup-CH2KG3JY.js +23 -0
  224. package/dist/devtools-PLJCAU53.js +8 -0
  225. package/dist/discord-G4BQJ2VX.js +7 -0
  226. package/dist/email-XMRNMFXR.js +137 -0
  227. package/dist/entity-boost-IOER4VRR.js +249 -0
  228. package/dist/exe-drift-2IJFQV43.js +68 -0
  229. package/dist/exe-export-Y2RFXGNB.js +71 -0
  230. package/dist/exe-import-LN6PE2YT.js +74 -0
  231. package/dist/exe-key-DHKBWLIT.js +571 -0
  232. package/dist/exe-org-F7LZRHFB.js +73 -0
  233. package/dist/factory-YTD23AW7.js +67 -0
  234. package/dist/fast-db-init-IIP6EGWT.js +7 -0
  235. package/dist/file-edit-FJVEO7XZ.js +8 -0
  236. package/dist/file-read-Q2NH3J35.js +8 -0
  237. package/dist/file-write-UNSFXNH4.js +8 -0
  238. package/dist/gateway/index.js +87 -16563
  239. package/dist/gateway-client-APTYJNCL.js +11 -0
  240. package/dist/git-staleness-I2BUEDE2.js +111 -0
  241. package/dist/git-task-sweep-UEOTXRW4.js +35 -0
  242. package/dist/glob-GRMXLDMB.js +7 -0
  243. package/dist/global-procedures-MFE5LGRF.js +19 -0
  244. package/dist/graph-query-657O5JYJ.js +25 -0
  245. package/dist/graph-rag-ZE5N7C65.js +28 -0
  246. package/dist/grep-NBV6SKYZ.js +7 -0
  247. package/dist/hooks/bug-report-worker.js +41 -10823
  248. package/dist/hooks/codex-stop-task-finalizer.js +38 -9055
  249. package/dist/hooks/commit-complete.js +32 -10914
  250. package/dist/hooks/error-recall.js +35 -7955
  251. package/dist/hooks/exe-heartbeat-hook.js +17 -327
  252. package/dist/hooks/ingest-worker.js +16 -565
  253. package/dist/hooks/ingest.js +98 -11436
  254. package/dist/hooks/instructions-loaded.js +25 -6209
  255. package/dist/hooks/notification.js +16 -6098
  256. package/dist/hooks/post-compact.js +34 -7041
  257. package/dist/hooks/post-tool-combined.js +27 -9751
  258. package/dist/hooks/pre-compact.js +38 -11130
  259. package/dist/hooks/pre-tool-use.js +61 -7233
  260. package/dist/hooks/prompt-submit.js +113 -13358
  261. package/dist/hooks/session-end.js +57 -11503
  262. package/dist/hooks/session-start.js +46 -9423
  263. package/dist/hooks/stop.js +70 -7210
  264. package/dist/hooks/subagent-stop.js +33 -6897
  265. package/dist/hooks/summary-worker.js +86 -8629
  266. package/dist/hooks-KD3N2D7P.js +9 -0
  267. package/dist/imessage-L7WKA2WD.js +7 -0
  268. package/dist/index.js +49 -19391
  269. package/dist/installer-5N24GXFB.js +32 -0
  270. package/dist/installer-QSD76YRY.js +291 -0
  271. package/dist/installer-ZPQUKHU5.js +331 -0
  272. package/dist/intercom-queue-PEEGAJ33.js +19 -0
  273. package/dist/key-backup-status-57UB74DU.js +41 -0
  274. package/dist/lib/agent-config.js +19 -224
  275. package/dist/lib/cloud-sync.js +44 -5398
  276. package/dist/lib/cloudflare-dns.js +7 -111
  277. package/dist/lib/config.js +18 -278
  278. package/dist/lib/consolidation.js +21 -1047
  279. package/dist/lib/crypto.js +7 -45
  280. package/dist/lib/database.js +20 -2932
  281. package/dist/lib/db-daemon-client.js +24 -460
  282. package/dist/lib/db.js +22 -2932
  283. package/dist/lib/device-registry.js +20 -2992
  284. package/dist/lib/embedder.js +14 -822
  285. package/dist/lib/employee-templates.js +21 -1102
  286. package/dist/lib/employees.js +30 -497
  287. package/dist/lib/error-detector.js +9 -149
  288. package/dist/lib/exe-daemon-client.js +16 -570
  289. package/dist/lib/exe-daemon.js +509 -18398
  290. package/dist/lib/file-grep.js +2 -0
  291. package/dist/lib/hybrid-search.js +18 -7674
  292. package/dist/lib/identity-templates.js +2 -0
  293. package/dist/lib/identity.js +13 -309
  294. package/dist/lib/keychain.js +9 -499
  295. package/dist/lib/license.js +21 -549
  296. package/dist/lib/messaging.js +35 -1631
  297. package/dist/lib/post-tool-memory.js +8 -370
  298. package/dist/lib/registry-proxy.js +8 -155
  299. package/dist/lib/reminders.js +11 -231
  300. package/dist/lib/runtime-table.js +7 -20
  301. package/dist/lib/schedules.js +16 -4614
  302. package/dist/lib/session-registry.js +9 -57
  303. package/dist/lib/session-wrappers.js +3 -127
  304. package/dist/lib/skill-learning.js +20 -2101
  305. package/dist/lib/status-brief.js +4 -368
  306. package/dist/lib/store.js +25 -5593
  307. package/dist/lib/task-router.js +12 -254
  308. package/dist/lib/tasks.js +44 -5801
  309. package/dist/lib/tmux-routing.js +45 -5804
  310. package/dist/lib/tmux-status.js +16 -248
  311. package/dist/lib/tmux-transport.js +6 -90
  312. package/dist/lib/token-spend.js +9 -337
  313. package/dist/lib/transport.js +8 -134
  314. package/dist/lib/ws-auth.js +6 -14
  315. package/dist/lib/ws-client.js +4 -22
  316. package/dist/license-gate-NXXDOOAQ.js +14 -0
  317. package/dist/mcp/register-tools.js +90 -32454
  318. package/dist/mcp/server.js +139 -32625
  319. package/dist/mcp/tools/complete-reminder.js +10 -237
  320. package/dist/mcp/tools/create-reminder.js +10 -222
  321. package/dist/mcp/tools/create-task.js +28 -6730
  322. package/dist/mcp/tools/deactivate-behavior.js +13 -476
  323. package/dist/mcp/tools/list-reminders.js +10 -232
  324. package/dist/mcp/tools/list-tasks.js +28 -1571
  325. package/dist/mcp/tools/send-message.js +25 -1672
  326. package/dist/mcp/tools/update-task.js +27 -6123
  327. package/dist/mcp-diagnostics-OOZ3TDPY.js +200 -0
  328. package/dist/memory-cards-RIL5BZHK.js +175 -0
  329. package/dist/memory-queue-JYIJ7TMQ.js +13 -0
  330. package/dist/memory-queue-client-PE7757HU.js +14 -0
  331. package/dist/notifications-CJFULBMN.js +42 -0
  332. package/dist/orchestration-phase-Y2TYNW5H.js +23 -0
  333. package/dist/orchestrator-RXIPT6PZ.js +29 -0
  334. package/dist/permission-presets-WI524OSE.js +25 -0
  335. package/dist/permissions-TYT6YMNZ.js +14 -0
  336. package/dist/pg-ssl-KFHNJPGO.js +7 -0
  337. package/dist/plan-limits-5WY5YK6X.js +27 -0
  338. package/dist/preferences-MW3BYTQ5.js +10 -0
  339. package/dist/project-name-PL2ZDWNU.js +11 -0
  340. package/dist/projection-worker-OY4WYZW6.js +661 -0
  341. package/dist/reranker-LFJ73RRS.js +19 -0
  342. package/dist/review-gate-E6K7QQE5.js +120 -0
  343. package/dist/runtime/index.js +82 -13408
  344. package/dist/self-query-router-IZANO632.js +83 -0
  345. package/dist/session-events-TQ6S5KQN.js +32 -0
  346. package/dist/session-kill-telemetry-46YIJWE2.js +30 -0
  347. package/dist/session-scope-3MYK3YRV.js +34 -0
  348. package/dist/setup-wizard-ZJ7UCRBP.js +12 -0
  349. package/dist/shard-manager-RCFZ6RXY.js +29 -0
  350. package/dist/signal-XN2C7PIG.js +7 -0
  351. package/dist/slack-Z77TFJCK.js +7 -0
  352. package/dist/task-enforcement-337DZI3I.js +252 -0
  353. package/dist/task-scanner-IMIR3T7T.js +11 -0
  354. package/dist/task-scope-MUU4KXE6.js +32 -0
  355. package/dist/tasks-crud-PMLK3MM4.js +53 -0
  356. package/dist/tasks-review-56LVKKNL.js +44 -0
  357. package/dist/telegram-CRPZO7JT.js +7 -0
  358. package/dist/telemetry-SPVGPKU7.js +9 -0
  359. package/dist/tool-capability-index-XNBSDCH3.js +10 -0
  360. package/dist/tool-gates-LGH6CSY4.js +14 -0
  361. package/dist/tool-gates-W7SZCYTX.js +13 -0
  362. package/dist/tool-registry-436LVVLY.js +11 -0
  363. package/dist/tool-telemetry-L77WNAYT.js +15 -0
  364. package/dist/tui/App.js +253 -14668
  365. package/dist/tui-data-SB45REYU.js +161 -0
  366. package/dist/update-check-LZGZISRJ.js +11 -0
  367. package/dist/webhook-VVP7KQVC.js +96 -0
  368. package/dist/whatsapp-J6QNDBGY.js +9 -0
  369. package/dist/whatsapp-accounts-RTJF3LM6.js +15 -0
  370. package/dist/wiki-client-EKCIKCVC.js +136 -0
  371. package/dist/worker-gate-UYNN62GJ.js +19 -0
  372. package/dist/worktree-4T4BPWPN.js +205 -0
  373. package/package.json +1 -1
package/dist/lib/db.js CHANGED
@@ -1,2935 +1,25 @@
1
- var __defProp = Object.defineProperty;
2
- var __getOwnPropNames = Object.getOwnPropertyNames;
3
- var __esm = (fn, res) => function __init() {
4
- return fn && (res = (0, fn[__getOwnPropNames(fn)[0]])(fn = 0)), res;
5
- };
6
- var __export = (target, all) => {
7
- for (var name in all)
8
- __defProp(target, name, { get: all[name], enumerable: true });
9
- };
10
-
11
- // src/lib/secure-files.ts
12
- import { chmodSync, existsSync, mkdirSync } from "fs";
13
- import { chmod, mkdir } from "fs/promises";
14
- function ensurePrivateDirSync(dirPath) {
15
- mkdirSync(dirPath, { recursive: true, mode: PRIVATE_DIR_MODE });
16
- try {
17
- chmodSync(dirPath, PRIVATE_DIR_MODE);
18
- } catch {
19
- }
20
- }
21
- function enforcePrivateFileSync(filePath) {
22
- try {
23
- if (existsSync(filePath)) chmodSync(filePath, PRIVATE_FILE_MODE);
24
- } catch {
25
- }
26
- }
27
- var PRIVATE_DIR_MODE, PRIVATE_FILE_MODE;
28
- var init_secure_files = __esm({
29
- "src/lib/secure-files.ts"() {
30
- "use strict";
31
- PRIVATE_DIR_MODE = 448;
32
- PRIVATE_FILE_MODE = 384;
33
- }
34
- });
35
-
36
- // src/lib/config.ts
37
- import { readFile, writeFile } from "fs/promises";
38
- import { readFileSync, existsSync as existsSync2, renameSync } from "fs";
39
- import path from "path";
40
- import os from "os";
41
- function resolveDataDir() {
42
- if (process.env.EXE_OS_DIR) return process.env.EXE_OS_DIR;
43
- if (process.env.EXE_MEM_DIR) return process.env.EXE_MEM_DIR;
44
- const newDir = path.join(os.homedir(), ".exe-os");
45
- const legacyDir = path.join(os.homedir(), ".exe-mem");
46
- if (!existsSync2(newDir) && existsSync2(legacyDir)) {
47
- try {
48
- renameSync(legacyDir, newDir);
49
- process.stderr.write(`[exe-os] Migrated data directory: ~/.exe-mem \u2192 ~/.exe-os
50
- `);
51
- } catch {
52
- return legacyDir;
53
- }
54
- }
55
- return newDir;
56
- }
57
- var EXE_AI_DIR, DB_PATH, MODELS_DIR, CONFIG_PATH, LEGACY_LANCE_PATH, CURRENT_CONFIG_VERSION, DEFAULT_CONFIG;
58
- var init_config = __esm({
59
- "src/lib/config.ts"() {
60
- "use strict";
61
- init_secure_files();
62
- EXE_AI_DIR = resolveDataDir();
63
- DB_PATH = path.join(EXE_AI_DIR, "memories.db");
64
- MODELS_DIR = path.join(EXE_AI_DIR, "models");
65
- CONFIG_PATH = path.join(EXE_AI_DIR, "config.json");
66
- LEGACY_LANCE_PATH = path.join(EXE_AI_DIR, "local.lance");
67
- CURRENT_CONFIG_VERSION = 1;
68
- DEFAULT_CONFIG = {
69
- config_version: CURRENT_CONFIG_VERSION,
70
- dbPath: DB_PATH,
71
- modelFile: "jina-embeddings-v5-small-q4_k_m.gguf",
72
- embeddingDim: 1024,
73
- batchSize: 20,
74
- flushIntervalMs: 1e4,
75
- autoIngestion: true,
76
- autoRetrieval: true,
77
- searchMode: "hybrid",
78
- hookSearchMode: "hybrid",
79
- fileGrepEnabled: true,
80
- splashEffect: true,
81
- consolidationEnabled: true,
82
- consolidationIntervalMs: 6 * 60 * 60 * 1e3,
83
- consolidationModel: "claude-haiku-4-5-20251001",
84
- consolidationMaxCallsPerRun: 20,
85
- selfQueryRouter: true,
86
- selfQueryModel: "claude-haiku-4-5-20251001",
87
- rerankerEnabled: true,
88
- scalingRoadmap: {
89
- rerankerAutoTrigger: {
90
- enabled: true,
91
- broadQueryMinCardinality: 5e4,
92
- fetchTopK: 150,
93
- returnTopK: 5
94
- }
95
- },
96
- graphRagEnabled: true,
97
- wikiEnabled: false,
98
- wikiUrl: "",
99
- wikiApiKey: "",
100
- wikiSyncIntervalMs: 30 * 60 * 1e3,
101
- wikiWorkspaceMapping: {},
102
- wikiAutoUpdate: true,
103
- wikiAutoUpdateThreshold: 0.5,
104
- wikiAutoUpdateCreateNew: true,
105
- skillLearning: true,
106
- skillThreshold: 3,
107
- skillModel: "claude-haiku-4-5-20251001",
108
- exeHeartbeat: {
109
- enabled: true,
110
- intervalSeconds: 60,
111
- staleInProgressThresholdHours: 2
112
- },
113
- sessionLifecycle: {
114
- idleKillEnabled: true,
115
- idleKillTicksRequired: 3,
116
- idleKillIntercomAckWindowMs: 1e4,
117
- maxAutoInstances: 10
118
- },
119
- autoUpdate: {
120
- checkOnBoot: true,
121
- autoInstall: false,
122
- checkIntervalMs: 24 * 60 * 60 * 1e3
123
- },
124
- support: {
125
- bugReportEndpoint: "https://askexe.com/v1/support/bug-reports"
126
- },
127
- orchestration: {
128
- phase: "phase_1_coo",
129
- phaseSetBy: "default"
130
- }
131
- };
132
- }
133
- });
134
-
135
- // src/lib/daemon-auth.ts
136
- import crypto from "crypto";
137
- import path4 from "path";
138
- import { existsSync as existsSync4, readFileSync as readFileSync3, writeFileSync as writeFileSync2 } from "fs";
139
- function normalizeToken(token) {
140
- if (!token) return null;
141
- const trimmed = token.trim();
142
- return trimmed.length > 0 ? trimmed : null;
143
- }
144
- function readDaemonToken() {
145
- try {
146
- if (!existsSync4(DAEMON_TOKEN_PATH)) return null;
147
- return normalizeToken(readFileSync3(DAEMON_TOKEN_PATH, "utf8"));
148
- } catch {
149
- return null;
150
- }
151
- }
152
- function ensureDaemonToken(seed) {
153
- const existing = readDaemonToken();
154
- if (existing) return existing;
155
- const token = normalizeToken(seed) ?? crypto.randomBytes(32).toString("hex");
156
- ensurePrivateDirSync(EXE_AI_DIR);
157
- writeFileSync2(DAEMON_TOKEN_PATH, `${token}
158
- `, "utf8");
159
- enforcePrivateFileSync(DAEMON_TOKEN_PATH);
160
- return token;
161
- }
162
- var DAEMON_TOKEN_PATH;
163
- var init_daemon_auth = __esm({
164
- "src/lib/daemon-auth.ts"() {
165
- "use strict";
166
- init_config();
167
- init_secure_files();
168
- DAEMON_TOKEN_PATH = path4.join(EXE_AI_DIR, "exed.token");
169
- }
170
- });
171
-
172
- // src/lib/exe-daemon-client.ts
173
- import net from "net";
174
- import os4 from "os";
175
- import { spawn, execSync as execSync2 } from "child_process";
176
- import { randomUUID } from "crypto";
177
- import { existsSync as existsSync5, unlinkSync as unlinkSync2, readFileSync as readFileSync4, openSync, closeSync, statSync } from "fs";
178
- import path5 from "path";
179
- import { fileURLToPath } from "url";
180
- function handleData(chunk) {
181
- _buffer += chunk.toString();
182
- if (_buffer.length > MAX_BUFFER) {
183
- _buffer = "";
184
- return;
185
- }
186
- let newlineIdx;
187
- while ((newlineIdx = _buffer.indexOf("\n")) !== -1) {
188
- const line = _buffer.slice(0, newlineIdx).trim();
189
- _buffer = _buffer.slice(newlineIdx + 1);
190
- if (!line) continue;
191
- try {
192
- const response = JSON.parse(line);
193
- const id = response.id;
194
- if (!id) continue;
195
- const entry = _pending.get(id);
196
- if (entry) {
197
- clearTimeout(entry.timer);
198
- _pending.delete(id);
199
- entry.resolve(response);
200
- }
201
- } catch {
202
- }
203
- }
204
- }
205
- function isZombie(pid) {
206
- try {
207
- const state = execSync2(`ps -p ${pid} -o state=`, { encoding: "utf8", timeout: 2e3 }).trim();
208
- return state.startsWith("Z");
209
- } catch {
210
- return false;
211
- }
212
- }
213
- function cleanupStaleFiles() {
214
- if (existsSync5(PID_PATH)) {
215
- try {
216
- const pid = parseInt(readFileSync4(PID_PATH, "utf8").trim(), 10);
217
- if (pid > 0) {
218
- try {
219
- process.kill(pid, 0);
220
- if (!isZombie(pid)) {
221
- return;
222
- }
223
- process.stderr.write(`[exed-client] PID ${pid} is a zombie \u2014 cleaning up stale files
224
- `);
225
- } catch {
226
- }
227
- }
228
- } catch {
229
- }
230
- try {
231
- unlinkSync2(PID_PATH);
232
- } catch {
233
- }
234
- try {
235
- unlinkSync2(SOCKET_PATH);
236
- } catch {
237
- }
238
- }
239
- }
240
- function findPackageRoot() {
241
- let dir = path5.dirname(fileURLToPath(import.meta.url));
242
- const { root } = path5.parse(dir);
243
- while (dir !== root) {
244
- if (existsSync5(path5.join(dir, "package.json"))) return dir;
245
- dir = path5.dirname(dir);
246
- }
247
- return null;
248
- }
249
- function spawnDaemon() {
250
- const totalGB = os4.totalmem() / (1024 * 1024 * 1024);
251
- if (totalGB <= 8) {
252
- process.stderr.write(
253
- `[exed-client] SKIP: ${totalGB.toFixed(0)}GB system \u2014 embedding daemon disabled. Using keyword search only. Minimum 16GB recommended for vector search.
254
- `
255
- );
256
- return;
257
- }
258
- const pkgRoot = findPackageRoot();
259
- if (!pkgRoot) {
260
- process.stderr.write("[exed-client] WARN: cannot find package root\n");
261
- return;
262
- }
263
- const daemonPath = path5.join(pkgRoot, "dist", "lib", "exe-daemon.js");
264
- if (!existsSync5(daemonPath)) {
265
- process.stderr.write(`[exed-client] WARN: daemon script not found at ${daemonPath}
266
- `);
267
- return;
268
- }
269
- const resolvedPath = daemonPath;
270
- const daemonToken = ensureDaemonToken(process.env[DAEMON_TOKEN_ENV] ?? null);
271
- process.stderr.write(`[exed-client] Spawning daemon: ${resolvedPath}
272
- `);
273
- const logPath = path5.join(path5.dirname(SOCKET_PATH), "exed.log");
274
- let stderrFd = "ignore";
275
- try {
276
- stderrFd = openSync(logPath, "a");
277
- } catch {
278
- }
279
- const heapCapMB = totalGB <= 8 ? 256 : 512;
280
- const nodeArgs = [`--max-old-space-size=${heapCapMB}`, resolvedPath];
281
- const child = spawn(process.execPath, nodeArgs, {
282
- detached: true,
283
- stdio: ["ignore", "ignore", stderrFd],
284
- env: {
285
- ...process.env,
286
- TMUX: void 0,
287
- // Daemon is global — must not inherit session scope
288
- TMUX_PANE: void 0,
289
- // Prevents resolveExeSession() from scoping to one session
290
- EXE_DAEMON_SOCK: SOCKET_PATH,
291
- EXE_DAEMON_PID: PID_PATH,
292
- [DAEMON_TOKEN_ENV]: daemonToken
293
- }
294
- });
295
- child.unref();
296
- if (typeof stderrFd === "number") {
297
- try {
298
- closeSync(stderrFd);
299
- } catch {
300
- }
301
- }
302
- }
303
- function acquireSpawnLock() {
304
- try {
305
- const fd = openSync(SPAWN_LOCK_PATH, "wx");
306
- closeSync(fd);
307
- return true;
308
- } catch {
309
- try {
310
- const stat = statSync(SPAWN_LOCK_PATH);
311
- if (Date.now() - stat.mtimeMs > SPAWN_LOCK_STALE_MS) {
312
- try {
313
- unlinkSync2(SPAWN_LOCK_PATH);
314
- } catch {
315
- }
316
- try {
317
- const fd = openSync(SPAWN_LOCK_PATH, "wx");
318
- closeSync(fd);
319
- return true;
320
- } catch {
321
- }
322
- }
323
- } catch {
324
- }
325
- return false;
326
- }
327
- }
328
- function releaseSpawnLock() {
329
- try {
330
- unlinkSync2(SPAWN_LOCK_PATH);
331
- } catch {
332
- }
333
- }
334
- function connectToSocket() {
335
- return new Promise((resolve) => {
336
- if (_socket && _connected) {
337
- resolve(true);
338
- return;
339
- }
340
- const socket = net.createConnection({ path: SOCKET_PATH });
341
- const connectTimeout = setTimeout(() => {
342
- socket.destroy();
343
- resolve(false);
344
- }, 2e3);
345
- socket.on("connect", () => {
346
- clearTimeout(connectTimeout);
347
- _socket = socket;
348
- _connected = true;
349
- _buffer = "";
350
- socket.on("data", handleData);
351
- socket.on("close", () => {
352
- _connected = false;
353
- _socket = null;
354
- for (const [id, entry] of _pending) {
355
- clearTimeout(entry.timer);
356
- _pending.delete(id);
357
- entry.resolve({ error: "Connection closed" });
358
- }
359
- });
360
- socket.on("error", () => {
361
- _connected = false;
362
- _socket = null;
363
- });
364
- resolve(true);
365
- });
366
- socket.on("error", () => {
367
- clearTimeout(connectTimeout);
368
- resolve(false);
369
- });
370
- });
371
- }
372
- async function connectEmbedDaemon() {
373
- if (_socket && _connected) return true;
374
- if (await connectToSocket()) return true;
375
- if (acquireSpawnLock()) {
376
- try {
377
- cleanupStaleFiles();
378
- spawnDaemon();
379
- } finally {
380
- releaseSpawnLock();
381
- }
382
- }
383
- const start = Date.now();
384
- let delay2 = 100;
385
- while (Date.now() - start < CONNECT_TIMEOUT_MS) {
386
- await new Promise((r) => setTimeout(r, delay2));
387
- if (await connectToSocket()) return true;
388
- delay2 = Math.min(delay2 * 2, 3e3);
389
- }
390
- return false;
391
- }
392
- function sendDaemonRequest(payload, timeoutMs = REQUEST_TIMEOUT_MS) {
393
- return new Promise((resolve) => {
394
- if (!_socket || !_connected) {
395
- resolve({ error: "Not connected" });
396
- return;
397
- }
398
- const id = randomUUID();
399
- const token = process.env[DAEMON_TOKEN_ENV] ?? readDaemonToken();
400
- const timer = setTimeout(() => {
401
- _pending.delete(id);
402
- resolve({ error: "Request timeout" });
403
- }, timeoutMs);
404
- _pending.set(id, { resolve, timer });
405
- try {
406
- _socket.write(JSON.stringify({ id, token, ...payload }) + "\n");
407
- } catch {
408
- clearTimeout(timer);
409
- _pending.delete(id);
410
- resolve({ error: "Write failed" });
411
- }
412
- });
413
- }
414
- function isClientConnected() {
415
- return _connected;
416
- }
417
- var SOCKET_PATH, PID_PATH, SPAWN_LOCK_PATH, SPAWN_LOCK_STALE_MS, CONNECT_TIMEOUT_MS, REQUEST_TIMEOUT_MS, DAEMON_TOKEN_ENV, _socket, _connected, _buffer, _pending, MAX_BUFFER;
418
- var init_exe_daemon_client = __esm({
419
- "src/lib/exe-daemon-client.ts"() {
420
- "use strict";
421
- init_config();
422
- init_daemon_auth();
423
- SOCKET_PATH = process.env.EXE_DAEMON_SOCK ?? process.env.EXE_EMBED_SOCK ?? path5.join(EXE_AI_DIR, "exed.sock");
424
- PID_PATH = process.env.EXE_DAEMON_PID ?? process.env.EXE_EMBED_PID ?? path5.join(EXE_AI_DIR, "exed.pid");
425
- SPAWN_LOCK_PATH = path5.join(EXE_AI_DIR, "exed-spawn.lock");
426
- SPAWN_LOCK_STALE_MS = 3e4;
427
- CONNECT_TIMEOUT_MS = 15e3;
428
- REQUEST_TIMEOUT_MS = 3e4;
429
- DAEMON_TOKEN_ENV = "EXE_DAEMON_TOKEN";
430
- _socket = null;
431
- _connected = false;
432
- _buffer = "";
433
- _pending = /* @__PURE__ */ new Map();
434
- MAX_BUFFER = 1e7;
435
- }
436
- });
437
-
438
- // src/lib/daemon-protocol.ts
439
- function serializeValue(v) {
440
- if (v === null || v === void 0) return null;
441
- if (typeof v === "bigint") return Number(v);
442
- if (typeof v === "boolean") return v ? 1 : 0;
443
- if (v instanceof Uint8Array) {
444
- return { __blob: Buffer.from(v).toString("base64") };
445
- }
446
- if (ArrayBuffer.isView(v)) {
447
- return { __blob: Buffer.from(v.buffer, v.byteOffset, v.byteLength).toString("base64") };
448
- }
449
- if (v instanceof ArrayBuffer) {
450
- return { __blob: Buffer.from(v).toString("base64") };
451
- }
452
- if (typeof v === "string" || typeof v === "number") return v;
453
- return String(v);
454
- }
455
- function deserializeValue(v) {
456
- if (v === null) return null;
457
- if (typeof v === "object" && v !== null && "__blob" in v) {
458
- const buf = Buffer.from(v.__blob, "base64");
459
- return buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength);
460
- }
461
- return v;
462
- }
463
- function deserializeResultSet(srs) {
464
- const rows = srs.rows.map((obj) => {
465
- const values = srs.columns.map(
466
- (col) => deserializeValue(obj[col] ?? null)
467
- );
468
- const row = values;
469
- for (let i = 0; i < srs.columns.length; i++) {
470
- const col = srs.columns[i];
471
- if (col !== void 0) {
472
- row[col] = values[i] ?? null;
473
- }
474
- }
475
- Object.defineProperty(row, "length", {
476
- value: values.length,
477
- enumerable: false
478
- });
479
- return row;
480
- });
481
- return {
482
- columns: srs.columns,
483
- columnTypes: srs.columnTypes ?? [],
484
- rows,
485
- rowsAffected: srs.rowsAffected,
486
- lastInsertRowid: srs.lastInsertRowid != null ? BigInt(srs.lastInsertRowid) : void 0,
487
- toJSON: () => ({
488
- columns: srs.columns,
489
- columnTypes: srs.columnTypes ?? [],
490
- rows: srs.rows,
491
- rowsAffected: srs.rowsAffected,
492
- lastInsertRowid: srs.lastInsertRowid
493
- })
494
- };
495
- }
496
- var init_daemon_protocol = __esm({
497
- "src/lib/daemon-protocol.ts"() {
498
- "use strict";
499
- }
500
- });
501
-
502
- // src/lib/db-daemon-client.ts
503
- var db_daemon_client_exports = {};
504
- __export(db_daemon_client_exports, {
505
- createDaemonDbClient: () => createDaemonDbClient,
506
- initDaemonDbClient: () => initDaemonDbClient
507
- });
508
- function normalizeStatement2(stmt) {
509
- if (typeof stmt === "string") {
510
- return { sql: stmt, args: [] };
511
- }
512
- const sql = stmt.sql;
513
- let args = [];
514
- if (Array.isArray(stmt.args)) {
515
- args = stmt.args.map((v) => serializeValue(v));
516
- } else if (stmt.args && typeof stmt.args === "object") {
517
- const named = {};
518
- for (const [key, val] of Object.entries(stmt.args)) {
519
- named[key] = serializeValue(val);
520
- }
521
- return { sql, args: named };
522
- }
523
- return { sql, args };
524
- }
525
- function createDaemonDbClient(fallbackClient) {
526
- let _useDaemon = false;
527
- const client = {
528
- async execute(stmt) {
529
- if (!_useDaemon || !isClientConnected()) {
530
- return fallbackClient.execute(stmt);
531
- }
532
- const { sql, args } = normalizeStatement2(stmt);
533
- const response = await sendDaemonRequest({
534
- type: "db-execute",
535
- sql,
536
- args
537
- });
538
- if (response.error) {
539
- const errMsg = String(response.error);
540
- if (errMsg === "Not connected" || errMsg === "Request timeout" || errMsg === "Write failed" || errMsg === "DB not initialized") {
541
- process.stderr.write(`[db-daemon] Transport error (${errMsg}), falling back to direct
542
- `);
543
- return fallbackClient.execute(stmt);
544
- }
545
- throw new Error(errMsg);
546
- }
547
- if (response.db) {
548
- return deserializeResultSet(response.db);
549
- }
550
- process.stderr.write("[db-daemon] Unexpected response shape, falling back to direct\n");
551
- return fallbackClient.execute(stmt);
552
- },
553
- async batch(stmts, mode) {
554
- if (!_useDaemon || !isClientConnected()) {
555
- return fallbackClient.batch(stmts, mode);
556
- }
557
- const statements = stmts.map(normalizeStatement2);
558
- const response = await sendDaemonRequest({
559
- type: "db-batch",
560
- statements,
561
- mode: mode ?? "deferred"
562
- });
563
- if (response.error) {
564
- const errMsg = String(response.error);
565
- if (errMsg === "Not connected" || errMsg === "Request timeout" || errMsg === "Write failed" || errMsg === "DB not initialized") {
566
- process.stderr.write(`[db-daemon] Batch transport error (${errMsg}), falling back to direct
567
- `);
568
- return fallbackClient.batch(stmts, mode);
569
- }
570
- throw new Error(errMsg);
571
- }
572
- const batchResults = response["db-batch"];
573
- if (batchResults) {
574
- return batchResults.map(deserializeResultSet);
575
- }
576
- process.stderr.write("[db-daemon] Unexpected batch response shape, falling back to direct\n");
577
- return fallbackClient.batch(stmts, mode);
578
- },
579
- // Transaction support — delegate to fallback (transactions need direct connection)
580
- async transaction(mode) {
581
- return fallbackClient.transaction(mode);
582
- },
583
- // executeMultiple — delegate to fallback (used only for schema migrations)
584
- async executeMultiple(sql) {
585
- return fallbackClient.executeMultiple(sql);
586
- },
587
- // migrate — delegate to fallback
588
- async migrate(stmts) {
589
- return fallbackClient.migrate(stmts);
590
- },
591
- // Sync mode — delegate to fallback
592
- sync() {
593
- return fallbackClient.sync();
594
- },
595
- close() {
596
- _useDaemon = false;
597
- },
598
- get closed() {
599
- return fallbackClient.closed;
600
- },
601
- get protocol() {
602
- return fallbackClient.protocol;
603
- }
604
- };
605
- return {
606
- ...client,
607
- /** Enable daemon routing (call after confirming daemon is connected) */
608
- _enableDaemon() {
609
- _useDaemon = true;
610
- },
611
- /** Check if daemon routing is active */
612
- _isDaemonActive() {
613
- return _useDaemon && isClientConnected();
614
- }
615
- };
616
- }
617
- async function initDaemonDbClient(fallbackClient) {
618
- if (process.env.EXE_IS_DAEMON === "1") return null;
619
- const connected = await connectEmbedDaemon();
620
- if (!connected) {
621
- process.stderr.write("[db-daemon] Daemon unavailable \u2014 using direct SQLite\n");
622
- return null;
623
- }
624
- const client = createDaemonDbClient(fallbackClient);
625
- client._enableDaemon();
626
- process.stderr.write("[db-daemon] DB routing through daemon (single-writer)\n");
627
- return client;
628
- }
629
- var init_db_daemon_client = __esm({
630
- "src/lib/db-daemon-client.ts"() {
631
- "use strict";
632
- init_exe_daemon_client();
633
- init_daemon_protocol();
634
- }
635
- });
636
-
637
- // src/lib/database.ts
638
- import { chmodSync as chmodSync2, existsSync as existsSync6, statSync as statSync2, copyFileSync, unlinkSync as unlinkSync3, openSync as openSync2, closeSync as closeSync2, mkdirSync as mkdirSync2 } from "fs";
639
- import { createClient } from "@libsql/client";
640
- import { homedir } from "os";
641
- import { join } from "path";
642
-
643
- // src/lib/db-retry.ts
644
- var MAX_RETRIES = 5;
645
- var BASE_DELAY_MS = 250;
646
- var MAX_JITTER_MS = 400;
647
- function isBusyError(err) {
648
- if (err instanceof Error) {
649
- const msg = err.message.toLowerCase();
650
- return msg.includes("sqlite_busy") || msg.includes("database is locked");
651
- }
652
- return false;
653
- }
654
- function delay(ms) {
655
- return new Promise((resolve) => setTimeout(resolve, ms));
656
- }
657
- async function retryOnBusy(fn, label) {
658
- let lastError;
659
- for (let attempt = 0; attempt <= MAX_RETRIES; attempt++) {
660
- try {
661
- return await fn();
662
- } catch (err) {
663
- lastError = err;
664
- if (!isBusyError(err) || attempt === MAX_RETRIES) {
665
- throw err;
666
- }
667
- const backoff = BASE_DELAY_MS * Math.pow(2, attempt);
668
- const jitter = Math.floor(Math.random() * MAX_JITTER_MS);
669
- process.stderr.write(
670
- `[exe-os] SQLITE_BUSY ${label} retry ${attempt + 1}/${MAX_RETRIES} \u2014 waiting ${backoff + jitter}ms
671
- `
672
- );
673
- await delay(backoff + jitter);
674
- }
675
- }
676
- throw lastError;
677
- }
678
- function wrapWithRetry(client) {
679
- return new Proxy(client, {
680
- get(target, prop, receiver) {
681
- if (prop === "execute") {
682
- return (sql) => retryOnBusy(() => target.execute(sql), "execute");
683
- }
684
- if (prop === "batch") {
685
- return (stmts, mode) => retryOnBusy(() => target.batch(stmts, mode), "batch");
686
- }
687
- return Reflect.get(target, prop, receiver);
688
- }
689
- });
690
- }
691
-
692
- // src/lib/employees.ts
693
- init_config();
694
- import { readFile as readFile2, writeFile as writeFile2, mkdir as mkdir2 } from "fs/promises";
695
- import { existsSync as existsSync3, symlinkSync, readlinkSync, readFileSync as readFileSync2, renameSync as renameSync2, unlinkSync, writeFileSync } from "fs";
696
- import { execSync } from "child_process";
697
- import path2 from "path";
698
- import os2 from "os";
699
- var EMPLOYEES_PATH = path2.join(EXE_AI_DIR, "exe-employees.json");
700
- var DEFAULT_COORDINATOR_TEMPLATE_NAME = "exe";
701
- var COORDINATOR_ROLE = "COO";
702
- function normalizeRole(role) {
703
- return (role ?? "").trim().toLowerCase();
704
- }
705
- function isCoordinatorRole(role) {
706
- return normalizeRole(role) === normalizeRole(COORDINATOR_ROLE);
707
- }
708
- function getCoordinatorEmployee(employees) {
709
- return employees.find((e) => isCoordinatorRole(e.role));
710
- }
711
- function getCoordinatorName(employees = loadEmployeesSync()) {
712
- return getCoordinatorEmployee(employees)?.name ?? DEFAULT_COORDINATOR_TEMPLATE_NAME;
713
- }
714
- function loadEmployeesSync(employeesPath = EMPLOYEES_PATH) {
715
- if (!existsSync3(employeesPath)) return [];
716
- try {
717
- return JSON.parse(readFileSync2(employeesPath, "utf-8"));
718
- } catch {
719
- return [];
720
- }
721
- }
722
- var IDENTITY_DIR = path2.join(EXE_AI_DIR, "identity");
723
-
724
- // src/lib/database-adapter.ts
725
- import os3 from "os";
726
- import path3 from "path";
727
- import { createRequire } from "module";
728
- import { pathToFileURL } from "url";
729
- var VIEW_MAPPINGS = [
730
- { view: "memories", source: "memory.memory_records" },
731
- { view: "tasks", source: "memory.tasks" },
732
- { view: "behaviors", source: "memory.behaviors" },
733
- { view: "entities", source: "memory.entities" },
734
- { view: "relationships", source: "memory.relationships" },
735
- { view: "entity_memories", source: "memory.entity_memories" },
736
- { view: "entity_aliases", source: "memory.entity_aliases" },
737
- { view: "notifications", source: "memory.notifications" },
738
- { view: "messages", source: "memory.messages" },
739
- { view: "users", source: "wiki.users" },
740
- { view: "workspaces", source: "wiki.workspaces" },
741
- { view: "workspace_users", source: "wiki.workspace_users" },
742
- { view: "documents", source: "wiki.workspace_documents" },
743
- { view: "chats", source: "wiki.workspace_chats" }
744
- ];
745
- var UPSERT_KEYS = {
746
- memories: ["id"],
747
- tasks: ["id"],
748
- behaviors: ["id"],
749
- entities: ["id"],
750
- relationships: ["id"],
751
- entity_aliases: ["alias"],
752
- notifications: ["id"],
753
- messages: ["id"],
754
- users: ["id"],
755
- workspaces: ["id"],
756
- workspace_users: ["id"],
757
- documents: ["id"],
758
- chats: ["id"]
759
- };
760
- var BOOLEAN_COLUMNS_BY_TABLE = {
761
- memories: /* @__PURE__ */ new Set(["has_error", "draft"]),
762
- behaviors: /* @__PURE__ */ new Set(["active"]),
763
- notifications: /* @__PURE__ */ new Set(["read"]),
764
- users: /* @__PURE__ */ new Set(["has_personal_memory"])
765
- };
766
- var BOOLEAN_COLUMN_NAMES = new Set(
767
- Object.values(BOOLEAN_COLUMNS_BY_TABLE).flatMap((cols) => [...cols])
768
- );
769
- var IMMEDIATE_FALLBACK_PATTERNS = [
770
- /\bPRAGMA\b/i,
771
- /\bsqlite_master\b/i,
772
- /(?:^|[.\s])(?:memories|conversations|entities)_fts\b/i,
773
- /\bMATCH\b/i,
774
- /\bvector_distance_cos\s*\(/i,
775
- /\bjson_extract\s*\(/i,
776
- /\bjulianday\s*\(/i,
777
- /\bstrftime\s*\(/i,
778
- /\blast_insert_rowid\s*\(/i
779
- ];
780
- var prismaClientPromise = null;
781
- var compatibilityBootstrapPromise = null;
782
- function quotedIdentifier(identifier) {
783
- return `"${identifier.replace(/"/g, '""')}"`;
784
- }
785
- function unqualifiedTableName(name) {
786
- const raw = name.trim().replace(/^"|"$/g, "");
787
- const parts = raw.split(".");
788
- return parts[parts.length - 1].replace(/^"|"$/g, "").toLowerCase();
789
- }
790
- function stripTrailingSemicolon(sql) {
791
- return sql.trim().replace(/;+\s*$/u, "");
792
- }
793
- function appendClause(sql, clause) {
794
- const trimmed = stripTrailingSemicolon(sql);
795
- const returningMatch = /\sRETURNING\b[\s\S]*$/iu.exec(trimmed);
796
- if (!returningMatch) {
797
- return `${trimmed}${clause}`;
798
- }
799
- const idx = returningMatch.index;
800
- return `${trimmed.slice(0, idx)}${clause}${trimmed.slice(idx)}`;
801
- }
802
- function normalizeStatement(stmt) {
803
- if (typeof stmt === "string") {
804
- return { kind: "positional", sql: stmt, args: [] };
805
- }
806
- const sql = stmt.sql;
807
- if (Array.isArray(stmt.args) || stmt.args === void 0) {
808
- return { kind: "positional", sql, args: stmt.args ?? [] };
809
- }
810
- return { kind: "named", sql, args: stmt.args };
811
- }
812
- function rewriteBooleanLiterals(sql) {
813
- let out = sql;
814
- for (const column of BOOLEAN_COLUMN_NAMES) {
815
- const scoped = `((?:\\b[a-z_][a-z0-9_]*\\.)?${column})`;
816
- out = out.replace(new RegExp(`${scoped}\\s*=\\s*0\\b`, "giu"), "$1 = FALSE");
817
- out = out.replace(new RegExp(`${scoped}\\s*=\\s*1\\b`, "giu"), "$1 = TRUE");
818
- out = out.replace(new RegExp(`${scoped}\\s*!=\\s*0\\b`, "giu"), "$1 != FALSE");
819
- out = out.replace(new RegExp(`${scoped}\\s*!=\\s*1\\b`, "giu"), "$1 != TRUE");
820
- out = out.replace(new RegExp(`${scoped}\\s*<>\\s*0\\b`, "giu"), "$1 <> FALSE");
821
- out = out.replace(new RegExp(`${scoped}\\s*<>\\s*1\\b`, "giu"), "$1 <> TRUE");
822
- }
823
- return out;
824
- }
825
- function rewriteInsertOrIgnore(sql) {
826
- if (!/^\s*INSERT\s+OR\s+IGNORE\s+INTO\b/iu.test(sql)) {
827
- return sql;
828
- }
829
- const replaced = sql.replace(/^\s*INSERT\s+OR\s+IGNORE\s+INTO\b/iu, "INSERT INTO");
830
- return /\bON\s+CONFLICT\b/iu.test(replaced) ? replaced : appendClause(replaced, " ON CONFLICT DO NOTHING");
831
- }
832
- function rewriteInsertOrReplace(sql) {
833
- const match = /^\s*INSERT\s+OR\s+REPLACE\s+INTO\s+([A-Za-z0-9_."]+)\s*\(([^)]+)\)([\s\S]*)$/iu.exec(sql);
834
- if (!match) {
835
- return sql;
836
- }
837
- const rawTable = match[1];
838
- const rawColumns = match[2];
839
- const remainder = match[3];
840
- const tableName = unqualifiedTableName(rawTable);
841
- const conflictKeys = UPSERT_KEYS[tableName];
842
- if (!conflictKeys?.length) {
843
- return sql;
844
- }
845
- const columns = rawColumns.split(",").map((col) => col.trim().replace(/^"|"$/g, ""));
846
- const updateColumns = columns.filter((col) => !conflictKeys.includes(col));
847
- const conflictTarget = conflictKeys.map(quotedIdentifier).join(", ");
848
- const updateClause = updateColumns.length === 0 ? " DO NOTHING" : ` DO UPDATE SET ${updateColumns.map((col) => `${quotedIdentifier(col)} = EXCLUDED.${quotedIdentifier(col)}`).join(", ")}`;
849
- return `INSERT INTO ${rawTable} (${rawColumns})${appendClause(remainder, ` ON CONFLICT (${conflictTarget})${updateClause}`)}`;
850
- }
851
- function rewriteSql(sql) {
852
- let out = sql;
853
- out = out.replace(/\bdatetime\(\s*['"]now['"]\s*\)/giu, "CURRENT_TIMESTAMP");
854
- out = out.replace(/\bvector32\s*\(\s*\?\s*\)/giu, "?");
855
- out = rewriteBooleanLiterals(out);
856
- out = rewriteInsertOrReplace(out);
857
- out = rewriteInsertOrIgnore(out);
858
- return stripTrailingSemicolon(out);
859
- }
860
- function toBoolean(value) {
861
- if (value === null || value === void 0) return value;
862
- if (typeof value === "boolean") return value;
863
- if (typeof value === "number") return value !== 0;
864
- if (typeof value === "bigint") return value !== 0n;
865
- if (typeof value === "string") {
866
- const normalized = value.trim().toLowerCase();
867
- if (normalized === "0" || normalized === "false") return false;
868
- if (normalized === "1" || normalized === "true") return true;
869
- }
870
- return Boolean(value);
871
- }
872
- function countQuestionMarks(sql, end) {
873
- let count = 0;
874
- let inSingle = false;
875
- let inDouble = false;
876
- let inLineComment = false;
877
- let inBlockComment = false;
878
- for (let i = 0; i < end; i++) {
879
- const ch = sql[i];
880
- const next = sql[i + 1];
881
- if (inLineComment) {
882
- if (ch === "\n") inLineComment = false;
883
- continue;
884
- }
885
- if (inBlockComment) {
886
- if (ch === "*" && next === "/") {
887
- inBlockComment = false;
888
- i += 1;
889
- }
890
- continue;
891
- }
892
- if (!inSingle && !inDouble && ch === "-" && next === "-") {
893
- inLineComment = true;
894
- i += 1;
895
- continue;
896
- }
897
- if (!inSingle && !inDouble && ch === "/" && next === "*") {
898
- inBlockComment = true;
899
- i += 1;
900
- continue;
901
- }
902
- if (!inDouble && ch === "'" && sql[i - 1] !== "\\") {
903
- inSingle = !inSingle;
904
- continue;
905
- }
906
- if (!inSingle && ch === '"' && sql[i - 1] !== "\\") {
907
- inDouble = !inDouble;
908
- continue;
909
- }
910
- if (!inSingle && !inDouble && ch === "?") {
911
- count += 1;
912
- }
913
- }
914
- return count;
915
- }
916
- function findBooleanPlaceholderIndexes(sql) {
917
- const indexes = /* @__PURE__ */ new Set();
918
- for (const column of BOOLEAN_COLUMN_NAMES) {
919
- const pattern = new RegExp(`(?:\\b[a-z_][a-z0-9_]*\\.)?${column}\\s*=\\s*\\?`, "giu");
920
- for (const match of sql.matchAll(pattern)) {
921
- const matchText = match[0];
922
- const qIndex = match.index + matchText.lastIndexOf("?");
923
- indexes.add(countQuestionMarks(sql, qIndex + 1));
924
- }
925
- }
926
- return indexes;
927
- }
928
- function coerceInsertBooleanArgs(sql, args) {
929
- const match = /^\s*INSERT(?:\s+OR\s+(?:IGNORE|REPLACE))?\s+INTO\s+([A-Za-z0-9_."]+)\s*\(([^)]+)\)/iu.exec(sql);
930
- if (!match) return;
931
- const rawTable = match[1];
932
- const rawColumns = match[2];
933
- const boolColumns = BOOLEAN_COLUMNS_BY_TABLE[unqualifiedTableName(rawTable)];
934
- if (!boolColumns?.size) return;
935
- const columns = rawColumns.split(",").map((col) => col.trim().replace(/^"|"$/g, ""));
936
- for (const [index, column] of columns.entries()) {
937
- if (boolColumns.has(column) && index < args.length) {
938
- args[index] = toBoolean(args[index]);
939
- }
940
- }
941
- }
942
- function coerceUpdateBooleanArgs(sql, args) {
943
- const match = /^\s*UPDATE\s+([A-Za-z0-9_."]+)\s+SET\s+([\s\S]+?)(?:\s+WHERE\b|$)/iu.exec(sql);
944
- if (!match) return;
945
- const rawTable = match[1];
946
- const setClause = match[2];
947
- const boolColumns = BOOLEAN_COLUMNS_BY_TABLE[unqualifiedTableName(rawTable)];
948
- if (!boolColumns?.size) return;
949
- const assignments = setClause.split(",");
950
- let placeholderIndex = 0;
951
- for (const assignment of assignments) {
952
- if (!assignment.includes("?")) continue;
953
- placeholderIndex += 1;
954
- const colMatch = /^\s*(?:[A-Za-z_][A-Za-z0-9_]*\.)?([A-Za-z_][A-Za-z0-9_]*)\s*=\s*\?/iu.exec(assignment);
955
- if (colMatch && boolColumns.has(colMatch[1])) {
956
- args[placeholderIndex - 1] = toBoolean(args[placeholderIndex - 1]);
957
- }
958
- }
959
- }
960
- function coerceBooleanArgs(sql, args) {
961
- const nextArgs = [...args];
962
- coerceInsertBooleanArgs(sql, nextArgs);
963
- coerceUpdateBooleanArgs(sql, nextArgs);
964
- const placeholderIndexes = findBooleanPlaceholderIndexes(sql);
965
- for (const index of placeholderIndexes) {
966
- if (index > 0 && index <= nextArgs.length) {
967
- nextArgs[index - 1] = toBoolean(nextArgs[index - 1]);
968
- }
969
- }
970
- return nextArgs;
971
- }
972
- function convertQuestionMarksToDollarParams(sql) {
973
- let out = "";
974
- let placeholder = 0;
975
- let inSingle = false;
976
- let inDouble = false;
977
- let inLineComment = false;
978
- let inBlockComment = false;
979
- for (let i = 0; i < sql.length; i++) {
980
- const ch = sql[i];
981
- const next = sql[i + 1];
982
- if (inLineComment) {
983
- out += ch;
984
- if (ch === "\n") inLineComment = false;
985
- continue;
986
- }
987
- if (inBlockComment) {
988
- out += ch;
989
- if (ch === "*" && next === "/") {
990
- out += next;
991
- inBlockComment = false;
992
- i += 1;
993
- }
994
- continue;
995
- }
996
- if (!inSingle && !inDouble && ch === "-" && next === "-") {
997
- out += ch + next;
998
- inLineComment = true;
999
- i += 1;
1000
- continue;
1001
- }
1002
- if (!inSingle && !inDouble && ch === "/" && next === "*") {
1003
- out += ch + next;
1004
- inBlockComment = true;
1005
- i += 1;
1006
- continue;
1007
- }
1008
- if (!inDouble && ch === "'" && sql[i - 1] !== "\\") {
1009
- inSingle = !inSingle;
1010
- out += ch;
1011
- continue;
1012
- }
1013
- if (!inSingle && ch === '"' && sql[i - 1] !== "\\") {
1014
- inDouble = !inDouble;
1015
- out += ch;
1016
- continue;
1017
- }
1018
- if (!inSingle && !inDouble && ch === "?") {
1019
- placeholder += 1;
1020
- out += `$${placeholder}`;
1021
- continue;
1022
- }
1023
- out += ch;
1024
- }
1025
- return out;
1026
- }
1027
- function translateStatementForPostgres(stmt) {
1028
- const normalized = normalizeStatement(stmt);
1029
- if (normalized.kind === "named") {
1030
- throw new Error("Named SQL parameters are not supported by the Prisma adapter.");
1031
- }
1032
- const rewrittenSql = rewriteSql(normalized.sql);
1033
- const coercedArgs = coerceBooleanArgs(rewrittenSql, normalized.args);
1034
- return {
1035
- sql: convertQuestionMarksToDollarParams(rewrittenSql),
1036
- args: coercedArgs
1037
- };
1038
- }
1039
- function shouldBypassPostgres(stmt) {
1040
- const normalized = normalizeStatement(stmt);
1041
- if (normalized.kind === "named") {
1042
- return true;
1043
- }
1044
- return IMMEDIATE_FALLBACK_PATTERNS.some((pattern) => pattern.test(normalized.sql));
1045
- }
1046
- function shouldFallbackOnError(error) {
1047
- const message = error instanceof Error ? error.message : String(error);
1048
- return /42P01|42883|42601|does not exist|syntax error|not supported|Named SQL parameters are not supported/iu.test(message);
1049
- }
1050
- function isReadQuery(sql) {
1051
- const trimmed = sql.trimStart();
1052
- return /^(SELECT|WITH|SHOW|EXPLAIN|VALUES)\b/iu.test(trimmed) || /\bRETURNING\b/iu.test(trimmed);
1053
- }
1054
- function buildRow(row, columns) {
1055
- const values = columns.map((column) => row[column]);
1056
- return Object.assign(values, row);
1057
- }
1058
- function buildResultSet(rows, rowsAffected = 0) {
1059
- const columns = rows[0] ? Object.keys(rows[0]) : [];
1060
- const resultRows = rows.map((row) => buildRow(row, columns));
1061
- return {
1062
- columns,
1063
- columnTypes: columns.map(() => ""),
1064
- rows: resultRows,
1065
- rowsAffected,
1066
- lastInsertRowid: void 0,
1067
- toJSON() {
1068
- return {
1069
- columns,
1070
- columnTypes: columns.map(() => ""),
1071
- rows,
1072
- rowsAffected,
1073
- lastInsertRowid: void 0
1074
- };
1075
- }
1076
- };
1077
- }
1078
- async function loadPrismaClient() {
1079
- if (!prismaClientPromise) {
1080
- prismaClientPromise = (async () => {
1081
- const explicitPath = process.env.EXE_OS_PRISMA_CLIENT_PATH;
1082
- if (explicitPath) {
1083
- const module2 = await import(pathToFileURL(explicitPath).href);
1084
- const PrismaClient2 = module2.PrismaClient ?? module2.default?.PrismaClient;
1085
- if (!PrismaClient2) {
1086
- throw new Error(`No PrismaClient export found at ${explicitPath}`);
1087
- }
1088
- return new PrismaClient2();
1089
- }
1090
- const exeDbRoot = process.env.EXE_DB_ROOT ?? path3.join(os3.homedir(), "exe-db");
1091
- const requireFromExeDb = createRequire(path3.join(exeDbRoot, "package.json"));
1092
- const prismaEntry = requireFromExeDb.resolve("@prisma/client");
1093
- const module = await import(pathToFileURL(prismaEntry).href);
1094
- const PrismaClient = module.PrismaClient ?? module.default?.PrismaClient;
1095
- if (!PrismaClient) {
1096
- throw new Error(`No PrismaClient export found in ${prismaEntry}`);
1097
- }
1098
- return new PrismaClient();
1099
- })();
1100
- }
1101
- return prismaClientPromise;
1102
- }
1103
- async function ensureCompatibilityViews(prisma) {
1104
- if (!compatibilityBootstrapPromise) {
1105
- compatibilityBootstrapPromise = (async () => {
1106
- for (const mapping of VIEW_MAPPINGS) {
1107
- const relation = mapping.source.replace(/"/g, "");
1108
- const rows = await prisma.$queryRawUnsafe(
1109
- "SELECT to_regclass($1)::text AS regclass",
1110
- relation
1111
- );
1112
- if (!rows[0]?.regclass) {
1113
- continue;
1114
- }
1115
- await prisma.$executeRawUnsafe(
1116
- `CREATE OR REPLACE VIEW public.${quotedIdentifier(mapping.view)} AS SELECT * FROM ${mapping.source}`
1117
- );
1118
- }
1119
- })();
1120
- }
1121
- return compatibilityBootstrapPromise;
1122
- }
1123
- async function executeOnPrisma(executor, stmt) {
1124
- const translated = translateStatementForPostgres(stmt);
1125
- if (isReadQuery(translated.sql)) {
1126
- const rows = await executor.$queryRawUnsafe(
1127
- translated.sql,
1128
- ...translated.args
1129
- );
1130
- return buildResultSet(rows, /\bRETURNING\b/iu.test(translated.sql) ? rows.length : 0);
1131
- }
1132
- const rowsAffected = await executor.$executeRawUnsafe(translated.sql, ...translated.args);
1133
- return buildResultSet([], rowsAffected);
1134
- }
1135
- function splitSqlStatements(sql) {
1136
- const parts = [];
1137
- let current = "";
1138
- let inSingle = false;
1139
- let inDouble = false;
1140
- let inLineComment = false;
1141
- let inBlockComment = false;
1142
- for (let i = 0; i < sql.length; i++) {
1143
- const ch = sql[i];
1144
- const next = sql[i + 1];
1145
- if (inLineComment) {
1146
- current += ch;
1147
- if (ch === "\n") inLineComment = false;
1148
- continue;
1149
- }
1150
- if (inBlockComment) {
1151
- current += ch;
1152
- if (ch === "*" && next === "/") {
1153
- current += next;
1154
- inBlockComment = false;
1155
- i += 1;
1156
- }
1157
- continue;
1158
- }
1159
- if (!inSingle && !inDouble && ch === "-" && next === "-") {
1160
- current += ch + next;
1161
- inLineComment = true;
1162
- i += 1;
1163
- continue;
1164
- }
1165
- if (!inSingle && !inDouble && ch === "/" && next === "*") {
1166
- current += ch + next;
1167
- inBlockComment = true;
1168
- i += 1;
1169
- continue;
1170
- }
1171
- if (!inDouble && ch === "'" && sql[i - 1] !== "\\") {
1172
- inSingle = !inSingle;
1173
- current += ch;
1174
- continue;
1175
- }
1176
- if (!inSingle && ch === '"' && sql[i - 1] !== "\\") {
1177
- inDouble = !inDouble;
1178
- current += ch;
1179
- continue;
1180
- }
1181
- if (!inSingle && !inDouble && ch === ";") {
1182
- if (current.trim()) {
1183
- parts.push(current.trim());
1184
- }
1185
- current = "";
1186
- continue;
1187
- }
1188
- current += ch;
1189
- }
1190
- if (current.trim()) {
1191
- parts.push(current.trim());
1192
- }
1193
- return parts;
1194
- }
1195
- async function createPrismaDbAdapter(fallbackClient) {
1196
- const prisma = await loadPrismaClient();
1197
- await ensureCompatibilityViews(prisma);
1198
- let closed = false;
1199
- let adapter;
1200
- const fallbackExecute = async (stmt, error) => {
1201
- if (!fallbackClient) {
1202
- if (error) throw error;
1203
- throw new Error("No fallback SQLite client is available for this Prisma-routed query.");
1204
- }
1205
- if (error) {
1206
- process.stderr.write(
1207
- `[database-adapter] Falling back to SQLite: ${error instanceof Error ? error.message : String(error)}
1208
- `
1209
- );
1210
- }
1211
- return fallbackClient.execute(stmt);
1212
- };
1213
- adapter = {
1214
- async execute(stmt) {
1215
- if (shouldBypassPostgres(stmt)) {
1216
- return fallbackExecute(stmt);
1217
- }
1218
- try {
1219
- return await executeOnPrisma(prisma, stmt);
1220
- } catch (error) {
1221
- if (shouldFallbackOnError(error)) {
1222
- return fallbackExecute(stmt, error);
1223
- }
1224
- throw error;
1225
- }
1226
- },
1227
- async batch(stmts, mode) {
1228
- if (stmts.some((stmt) => shouldBypassPostgres(stmt))) {
1229
- if (!fallbackClient) {
1230
- throw new Error("Cannot batch unsupported SQLite-only statements without a fallback client.");
1231
- }
1232
- return fallbackClient.batch(stmts, mode);
1233
- }
1234
- try {
1235
- if (prisma.$transaction) {
1236
- return await prisma.$transaction(async (tx) => {
1237
- const results2 = [];
1238
- for (const stmt of stmts) {
1239
- results2.push(await executeOnPrisma(tx, stmt));
1240
- }
1241
- return results2;
1242
- });
1243
- }
1244
- const results = [];
1245
- for (const stmt of stmts) {
1246
- results.push(await executeOnPrisma(prisma, stmt));
1247
- }
1248
- return results;
1249
- } catch (error) {
1250
- if (fallbackClient && shouldFallbackOnError(error)) {
1251
- process.stderr.write(
1252
- `[database-adapter] Falling back batch to SQLite: ${error instanceof Error ? error.message : String(error)}
1253
- `
1254
- );
1255
- return fallbackClient.batch(stmts, mode);
1256
- }
1257
- throw error;
1258
- }
1259
- },
1260
- async migrate(stmts) {
1261
- if (fallbackClient) {
1262
- return fallbackClient.migrate(stmts);
1263
- }
1264
- return adapter.batch(stmts, "deferred");
1265
- },
1266
- async transaction(mode) {
1267
- if (!fallbackClient) {
1268
- throw new Error("Interactive transactions are only supported on the SQLite fallback client.");
1269
- }
1270
- return fallbackClient.transaction(mode);
1271
- },
1272
- async executeMultiple(sql) {
1273
- if (fallbackClient && shouldBypassPostgres(sql)) {
1274
- return fallbackClient.executeMultiple(sql);
1275
- }
1276
- for (const statement of splitSqlStatements(sql)) {
1277
- await adapter.execute(statement);
1278
- }
1279
- },
1280
- async sync() {
1281
- if (fallbackClient) {
1282
- return fallbackClient.sync();
1283
- }
1284
- return { frame_no: 0, frames_synced: 0 };
1285
- },
1286
- close() {
1287
- closed = true;
1288
- prismaClientPromise = null;
1289
- compatibilityBootstrapPromise = null;
1290
- void prisma.$disconnect?.();
1291
- },
1292
- get closed() {
1293
- return closed;
1294
- },
1295
- get protocol() {
1296
- return "prisma-postgres";
1297
- }
1298
- };
1299
- return adapter;
1300
- }
1301
-
1302
- // src/types/memory.ts
1303
- var EMBEDDING_DIM = 1024;
1304
-
1305
- // src/lib/database.ts
1306
- var _debugDb = process.env.EXE_DEBUG === "1";
1307
- function logCatchDebug(context, err) {
1308
- if (_debugDb) {
1309
- process.stderr.write(
1310
- `[database] ${context}: ${err instanceof Error ? err.message : String(err)}
1311
- `
1312
- );
1313
- }
1314
- }
1315
- var _client = null;
1316
- var _resilientClient = null;
1317
- var _walCheckpointTimer = null;
1318
- var _daemonClient = null;
1319
- var _adapterClient = null;
1320
- var _lockFd = null;
1321
- var DB_LOCK_PATH = join(homedir(), ".exe-os", "db.lock");
1322
- function acquireDbLock() {
1323
- mkdirSync2(join(homedir(), ".exe-os"), { recursive: true });
1324
- try {
1325
- _lockFd = openSync2(DB_LOCK_PATH, "wx");
1326
- } catch (err) {
1327
- if (err && typeof err === "object" && "code" in err && err.code === "EEXIST") {
1328
- try {
1329
- const lockStat = statSync2(DB_LOCK_PATH);
1330
- if (Date.now() - lockStat.mtimeMs > 6e4) {
1331
- unlinkSync3(DB_LOCK_PATH);
1332
- _lockFd = openSync2(DB_LOCK_PATH, "wx");
1333
- return;
1334
- }
1335
- } catch (e) {
1336
- logCatchDebug("stale lock check", e);
1337
- }
1338
- process.stderr.write(
1339
- "[database] WARN: Another process holds db.lock \u2014 waiting briefly then proceeding.\n"
1340
- );
1341
- return;
1342
- }
1343
- throw err;
1344
- }
1345
- }
1346
- function releaseDbLock() {
1347
- if (_lockFd !== null) {
1348
- try {
1349
- closeSync2(_lockFd);
1350
- } catch (e) {
1351
- logCatchDebug("lock close", e);
1352
- }
1353
- _lockFd = null;
1354
- }
1355
- try {
1356
- unlinkSync3(DB_LOCK_PATH);
1357
- } catch (e) {
1358
- logCatchDebug("lock unlink", e);
1359
- }
1360
- }
1361
- var initTurso = initDatabase;
1362
- async function initDatabase(config) {
1363
- acquireDbLock();
1364
- if (existsSync6(config.dbPath)) {
1365
- const dbStat = statSync2(config.dbPath);
1366
- if (dbStat.size === 0) {
1367
- const walPath = config.dbPath + "-wal";
1368
- if (existsSync6(walPath) && statSync2(walPath).size > 0) {
1369
- const backupPath = config.dbPath + ".zeroed-" + Date.now();
1370
- copyFileSync(config.dbPath, backupPath);
1371
- unlinkSync3(config.dbPath);
1372
- process.stderr.write(
1373
- `[database] CRITICAL: DB was 0 bytes. Moved to ${backupPath}, attempting WAL recovery.
1374
- `
1375
- );
1376
- } else {
1377
- process.stderr.write(
1378
- `[database] CRITICAL: DB is 0 bytes and no WAL available for recovery. Data may be lost. Check backups at ${config.dbPath}.bak
1379
- `
1380
- );
1381
- }
1382
- }
1383
- }
1384
- if (_walCheckpointTimer) {
1385
- clearInterval(_walCheckpointTimer);
1386
- _walCheckpointTimer = null;
1387
- }
1388
- if (_daemonClient) {
1389
- _daemonClient.close();
1390
- _daemonClient = null;
1391
- }
1392
- if (_adapterClient && _adapterClient !== _resilientClient) {
1393
- _adapterClient.close();
1394
- }
1395
- _adapterClient = null;
1396
- if (_client) {
1397
- _client.close();
1398
- _client = null;
1399
- _resilientClient = null;
1400
- }
1401
- const opts = {
1402
- url: `file:${config.dbPath}`
1403
- };
1404
- if (config.encryptionKey) {
1405
- opts.encryptionKey = config.encryptionKey;
1406
- }
1407
- _client = createClient(opts);
1408
- _resilientClient = wrapWithRetry(_client);
1409
- _adapterClient = _resilientClient;
1410
- await _client.execute("PRAGMA busy_timeout = 30000");
1411
- await _client.execute("PRAGMA journal_mode = WAL");
1412
- if (_walCheckpointTimer) clearInterval(_walCheckpointTimer);
1413
- _walCheckpointTimer = setInterval(() => {
1414
- _client?.execute("PRAGMA wal_checkpoint(PASSIVE)").catch(() => {
1415
- });
1416
- }, 3e4);
1417
- _walCheckpointTimer.unref();
1418
- if (process.env.DATABASE_URL && process.env.EXE_USE_POSTGRES === "1") {
1419
- _adapterClient = await createPrismaDbAdapter(_resilientClient);
1420
- }
1421
- try {
1422
- chmodSync2(config.dbPath, 384);
1423
- for (const suffix of ["-wal", "-shm"]) {
1424
- try {
1425
- chmodSync2(config.dbPath + suffix, 384);
1426
- } catch (chmodErr) {
1427
- process.stderr.write(`[database] chmod ${suffix} failed: ${chmodErr instanceof Error ? chmodErr.message : String(chmodErr)}
1428
- `);
1429
- }
1430
- }
1431
- } catch (chmodErr) {
1432
- process.stderr.write(`[database] chmod db failed: ${chmodErr instanceof Error ? chmodErr.message : String(chmodErr)}
1433
- `);
1434
- }
1435
- releaseDbLock();
1436
- }
1437
- function isInitialized() {
1438
- return _adapterClient !== null || _client !== null;
1439
- }
1440
- function setExternalClient(client) {
1441
- _adapterClient = client;
1442
- }
1443
- var SOFT_DELETE_RETENTION_DAYS = 7;
1444
- function getClient() {
1445
- if (!_adapterClient) {
1446
- throw new Error("Database client not initialized. Call initDatabase() first.");
1447
- }
1448
- if (process.env.DATABASE_URL && process.env.EXE_USE_POSTGRES === "1") {
1449
- return _adapterClient;
1450
- }
1451
- if (process.env.EXE_IS_DAEMON === "1") {
1452
- return _resilientClient;
1453
- }
1454
- if (_daemonClient && _daemonClient._isDaemonActive()) {
1455
- return _daemonClient;
1456
- }
1457
- if (!_resilientClient) {
1458
- return _adapterClient;
1459
- }
1460
- if (process.env.EXE_DB_READONLY === "1" || process.env.EXE_DB_DIRECT === "1") {
1461
- return _resilientClient;
1462
- }
1463
- if (!process.env.EXE_MCP_MODE) {
1464
- process.stderr.write(
1465
- "[database] WARN: Daemon unavailable \u2014 using direct SQLite (single-writer CLI mode).\n"
1466
- );
1467
- return _resilientClient;
1468
- }
1469
- process.stderr.write(
1470
- "[database] ERROR: Daemon is not running \u2014 refusing direct SQLite write access.\n[database] Direct writes from MCP processes bypass the single-writer gate and corrupt FTS5.\n[database] Restart the daemon: kill $(cat ~/.exe-os/exed.pid) && exe-os update\n"
1471
- );
1472
- throw new Error(
1473
- "Daemon not running. Direct SQLite writes from MCP processes are blocked to prevent FTS5 corruption. Restart daemon to restore service."
1474
- );
1475
- }
1476
- async function initDaemonClient() {
1477
- if (process.env.DATABASE_URL && process.env.EXE_USE_POSTGRES === "1") return;
1478
- if (process.env.EXE_IS_DAEMON === "1") return;
1479
- if (process.env.VITEST) return;
1480
- if (!_resilientClient) return;
1481
- if (_daemonClient) return;
1482
- try {
1483
- const { initDaemonDbClient: initDaemonDbClient2 } = await Promise.resolve().then(() => (init_db_daemon_client(), db_daemon_client_exports));
1484
- _daemonClient = await initDaemonDbClient2(_resilientClient);
1485
- } catch (err) {
1486
- process.stderr.write(
1487
- `[database] Daemon client init failed (non-fatal): ${err instanceof Error ? err.message : String(err)}
1488
- `
1489
- );
1490
- }
1491
- }
1492
- function getRawClient() {
1493
- if (!_client) {
1494
- throw new Error("Database client not initialized. Call initDatabase() first.");
1495
- }
1496
- return _client;
1497
- }
1498
- async function ensureSchema() {
1499
- const client = getRawClient();
1500
- await client.execute("PRAGMA journal_mode = WAL");
1501
- await client.execute("PRAGMA busy_timeout = 30000");
1502
- await client.execute("PRAGMA wal_autocheckpoint = 1000");
1503
- try {
1504
- await client.execute("PRAGMA libsql_vector_search_ef = 128");
1505
- } catch (e) {
1506
- logCatchDebug("migration", e);
1507
- }
1508
- await client.executeMultiple(`
1509
- CREATE TABLE IF NOT EXISTS memories (
1510
- id TEXT PRIMARY KEY,
1511
- agent_id TEXT NOT NULL,
1512
- agent_role TEXT NOT NULL,
1513
- session_id TEXT NOT NULL,
1514
- timestamp TEXT NOT NULL,
1515
- tool_name TEXT NOT NULL,
1516
- project_name TEXT NOT NULL,
1517
- has_error INTEGER NOT NULL DEFAULT 0,
1518
- raw_text TEXT NOT NULL,
1519
- vector F32_BLOB(1024),
1520
- version INTEGER NOT NULL DEFAULT 0
1521
- );
1522
-
1523
- CREATE INDEX IF NOT EXISTS idx_memories_agent
1524
- ON memories(agent_id);
1525
-
1526
- CREATE INDEX IF NOT EXISTS idx_memories_timestamp
1527
- ON memories(timestamp);
1528
-
1529
- CREATE INDEX IF NOT EXISTS idx_memories_session
1530
- ON memories(session_id);
1531
-
1532
- CREATE INDEX IF NOT EXISTS idx_memories_project
1533
- ON memories(project_name);
1534
-
1535
- CREATE INDEX IF NOT EXISTS idx_memories_tool
1536
- ON memories(tool_name);
1537
-
1538
- CREATE INDEX IF NOT EXISTS idx_memories_version
1539
- ON memories(version);
1540
-
1541
- CREATE INDEX IF NOT EXISTS idx_memories_agent_project
1542
- ON memories(agent_id, project_name);
1543
- `);
1544
- await client.executeMultiple(`
1545
- CREATE VIRTUAL TABLE IF NOT EXISTS memories_fts USING fts5(
1546
- raw_text,
1547
- content='memories',
1548
- content_rowid='rowid'
1549
- );
1550
-
1551
- CREATE TRIGGER IF NOT EXISTS memories_fts_ai AFTER INSERT ON memories BEGIN
1552
- INSERT INTO memories_fts(rowid, raw_text) VALUES (new.rowid, new.raw_text);
1553
- END;
1554
-
1555
- CREATE TRIGGER IF NOT EXISTS memories_fts_ad AFTER DELETE ON memories BEGIN
1556
- INSERT INTO memories_fts(memories_fts, rowid, raw_text) VALUES('delete', old.rowid, old.raw_text);
1557
- END;
1558
-
1559
- CREATE TRIGGER IF NOT EXISTS memories_fts_au AFTER UPDATE ON memories
1560
- WHEN new.status IS NULL OR new.status != 'deleted' BEGIN
1561
- INSERT INTO memories_fts(memories_fts, rowid, raw_text) VALUES('delete', old.rowid, old.raw_text);
1562
- INSERT INTO memories_fts(rowid, raw_text) VALUES (new.rowid, new.raw_text);
1563
- END;
1564
-
1565
- -- Soft-delete trigger: remove from FTS when status changes to 'deleted'
1566
- CREATE TRIGGER IF NOT EXISTS memories_fts_soft_delete AFTER UPDATE ON memories
1567
- WHEN new.status = 'deleted' AND (old.status IS NULL OR old.status != 'deleted') BEGIN
1568
- INSERT INTO memories_fts(memories_fts, rowid, raw_text) VALUES('delete', old.rowid, old.raw_text);
1569
- END;
1570
- `);
1571
- try {
1572
- await client.execute("SELECT COUNT(*) FROM memories_fts LIMIT 1");
1573
- } catch (ftsErr) {
1574
- process.stderr.write(
1575
- `[database] WARN: memories_fts corrupted (${ftsErr instanceof Error ? ftsErr.message : String(ftsErr)}) \u2014 rebuilding FTS index.
1576
- `
1577
- );
1578
- try {
1579
- await client.execute("INSERT INTO memories_fts(memories_fts) VALUES('rebuild')");
1580
- process.stderr.write("[database] FTS index rebuilt successfully.\n");
1581
- } catch (rebuildErr) {
1582
- process.stderr.write(
1583
- `[database] ERROR: FTS rebuild failed: ${rebuildErr instanceof Error ? rebuildErr.message : String(rebuildErr)}
1584
- `
1585
- );
1586
- }
1587
- }
1588
- await client.executeMultiple(`
1589
- CREATE TABLE IF NOT EXISTS sync_meta (
1590
- key TEXT PRIMARY KEY,
1591
- value TEXT NOT NULL
1592
- );
1593
- `);
1594
- await client.executeMultiple(`
1595
- CREATE TABLE IF NOT EXISTS tasks (
1596
- id TEXT PRIMARY KEY,
1597
- title TEXT NOT NULL,
1598
- assigned_to TEXT NOT NULL,
1599
- assigned_by TEXT NOT NULL,
1600
- project_name TEXT NOT NULL,
1601
- priority TEXT NOT NULL DEFAULT 'p1',
1602
- status TEXT NOT NULL DEFAULT 'open',
1603
- task_file TEXT,
1604
- created_at TEXT NOT NULL,
1605
- updated_at TEXT NOT NULL
1606
- );
1607
-
1608
- CREATE INDEX IF NOT EXISTS idx_tasks_assignee_status
1609
- ON tasks(assigned_to, status);
1610
- `);
1611
- await client.executeMultiple(`
1612
- CREATE TABLE IF NOT EXISTS behaviors (
1613
- id TEXT PRIMARY KEY,
1614
- agent_id TEXT NOT NULL,
1615
- project_name TEXT,
1616
- domain TEXT,
1617
- content TEXT NOT NULL,
1618
- active INTEGER NOT NULL DEFAULT 1,
1619
- created_at TEXT NOT NULL,
1620
- updated_at TEXT NOT NULL
1621
- );
1622
-
1623
- CREATE INDEX IF NOT EXISTS idx_behaviors_agent
1624
- ON behaviors(agent_id, active);
1625
- `);
1626
- try {
1627
- const coordinatorName = getCoordinatorName();
1628
- const existing = await client.execute({
1629
- sql: "SELECT COUNT(*) as cnt FROM behaviors WHERE agent_id = ?",
1630
- args: [coordinatorName]
1631
- });
1632
- if (Number(existing.rows[0]?.cnt) === 0) {
1633
- const seededAt = "2026-03-25T00:00:00Z";
1634
- for (const [domain, content] of [
1635
- ["workflow", `Don't ask "keep going?" \u2014 just keep executing phases/plans autonomously`],
1636
- ["tool-use", "Always use create_task MCP tool, never write .md files directly for task creation"],
1637
- ["workflow", "Auto-start reviewing when idle and reviews are pending \u2014 never ask founder for permission"]
1638
- ]) {
1639
- await client.execute({
1640
- sql: `INSERT INTO behaviors (id, agent_id, project_name, domain, content, active, created_at, updated_at)
1641
- VALUES (hex(randomblob(16)), ?, NULL, ?, ?, 1, ?, ?)`,
1642
- args: [coordinatorName, domain, content, seededAt, seededAt]
1643
- });
1644
- }
1645
- }
1646
- } catch (seedErr) {
1647
- logCatchDebug("behavior seed", seedErr);
1648
- }
1649
- try {
1650
- await client.execute({
1651
- sql: `ALTER TABLE behaviors ADD COLUMN priority TEXT DEFAULT 'p1'`,
1652
- args: []
1653
- });
1654
- } catch (e) {
1655
- logCatchDebug("migration", e);
1656
- }
1657
- try {
1658
- await client.execute({
1659
- sql: `ALTER TABLE behaviors ADD COLUMN vector F32_BLOB(${EMBEDDING_DIM})`,
1660
- args: []
1661
- });
1662
- } catch (e) {
1663
- logCatchDebug("migration", e);
1664
- }
1665
- for (const col of ["created_by_agent TEXT", "created_by_device TEXT", "source_session_id TEXT"]) {
1666
- try {
1667
- await client.execute({ sql: `ALTER TABLE behaviors ADD COLUMN ${col}`, args: [] });
1668
- } catch (e) {
1669
- logCatchDebug("migration", e);
1670
- }
1671
- }
1672
- try {
1673
- await client.execute({
1674
- sql: `ALTER TABLE tasks ADD COLUMN blocked_by TEXT`,
1675
- args: []
1676
- });
1677
- } catch (e) {
1678
- logCatchDebug("migration", e);
1679
- }
1680
- try {
1681
- await client.execute({
1682
- sql: `ALTER TABLE tasks ADD COLUMN parent_task_id TEXT`,
1683
- args: []
1684
- });
1685
- } catch (e) {
1686
- logCatchDebug("migration", e);
1687
- }
1688
- try {
1689
- await client.execute({
1690
- sql: `CREATE INDEX IF NOT EXISTS idx_tasks_parent_task_id
1691
- ON tasks(parent_task_id)
1692
- WHERE parent_task_id IS NOT NULL`,
1693
- args: []
1694
- });
1695
- } catch (e) {
1696
- logCatchDebug("migration", e);
1697
- }
1698
- try {
1699
- await client.execute({
1700
- sql: `UPDATE tasks SET status = 'done' WHERE status = 'completed'`,
1701
- args: []
1702
- });
1703
- } catch (e) {
1704
- logCatchDebug("migration", e);
1705
- }
1706
- try {
1707
- await client.execute({
1708
- sql: `ALTER TABLE tasks ADD COLUMN reviewer TEXT`,
1709
- args: []
1710
- });
1711
- } catch (e) {
1712
- logCatchDebug("migration", e);
1713
- }
1714
- try {
1715
- await client.execute({
1716
- sql: `ALTER TABLE tasks ADD COLUMN context TEXT`,
1717
- args: []
1718
- });
1719
- } catch (e) {
1720
- logCatchDebug("migration", e);
1721
- }
1722
- try {
1723
- await client.execute({
1724
- sql: `ALTER TABLE tasks ADD COLUMN result TEXT`,
1725
- args: []
1726
- });
1727
- } catch (e) {
1728
- logCatchDebug("migration", e);
1729
- }
1730
- try {
1731
- await client.execute({
1732
- sql: `ALTER TABLE tasks ADD COLUMN assigned_tmux TEXT`,
1733
- args: []
1734
- });
1735
- } catch (e) {
1736
- logCatchDebug("migration", e);
1737
- }
1738
- try {
1739
- await client.execute({
1740
- sql: `ALTER TABLE tasks ADD COLUMN checkpoint TEXT`,
1741
- args: []
1742
- });
1743
- } catch (e) {
1744
- logCatchDebug("migration", e);
1745
- }
1746
- try {
1747
- await client.execute({
1748
- sql: `ALTER TABLE tasks ADD COLUMN checkpoint_count INTEGER NOT NULL DEFAULT 0`,
1749
- args: []
1750
- });
1751
- } catch (e) {
1752
- logCatchDebug("migration", e);
1753
- }
1754
- try {
1755
- await client.execute({
1756
- sql: `ALTER TABLE tasks ADD COLUMN complexity TEXT NOT NULL DEFAULT 'standard'`,
1757
- args: []
1758
- });
1759
- } catch (e) {
1760
- logCatchDebug("migration", e);
1761
- }
1762
- try {
1763
- await client.execute({
1764
- sql: `ALTER TABLE tasks ADD COLUMN session_scope TEXT`,
1765
- args: []
1766
- });
1767
- } catch (e) {
1768
- logCatchDebug("migration", e);
1769
- }
1770
- try {
1771
- await client.execute({
1772
- sql: `ALTER TABLE memories ADD COLUMN task_id TEXT`,
1773
- args: []
1774
- });
1775
- } catch (e) {
1776
- logCatchDebug("migration", e);
1777
- }
1778
- try {
1779
- await client.execute({
1780
- sql: `ALTER TABLE memories ADD COLUMN consolidated INTEGER NOT NULL DEFAULT 0`,
1781
- args: []
1782
- });
1783
- } catch (e) {
1784
- logCatchDebug("migration", e);
1785
- }
1786
- try {
1787
- await client.execute({
1788
- sql: `ALTER TABLE memories ADD COLUMN author_device_id TEXT`,
1789
- args: []
1790
- });
1791
- } catch (e) {
1792
- logCatchDebug("migration", e);
1793
- }
1794
- try {
1795
- await client.execute({
1796
- sql: `ALTER TABLE memories ADD COLUMN scope TEXT NOT NULL DEFAULT 'business'`,
1797
- args: []
1798
- });
1799
- } catch (e) {
1800
- logCatchDebug("migration", e);
1801
- }
1802
- await client.executeMultiple(`
1803
- CREATE TABLE IF NOT EXISTS consolidations (
1804
- id TEXT PRIMARY KEY,
1805
- consolidated_memory_id TEXT NOT NULL,
1806
- source_memory_id TEXT NOT NULL,
1807
- created_at TEXT NOT NULL
1808
- );
1809
-
1810
- CREATE INDEX IF NOT EXISTS idx_consolidations_source
1811
- ON consolidations(source_memory_id);
1812
-
1813
- CREATE INDEX IF NOT EXISTS idx_consolidations_consolidated
1814
- ON consolidations(consolidated_memory_id);
1815
- `);
1816
- await client.executeMultiple(`
1817
- CREATE TABLE IF NOT EXISTS reminders (
1818
- id TEXT PRIMARY KEY,
1819
- text TEXT NOT NULL,
1820
- created_at TEXT NOT NULL,
1821
- due_date TEXT,
1822
- completed_at TEXT
1823
- );
1824
- `);
1825
- await client.executeMultiple(`
1826
- CREATE TABLE IF NOT EXISTS notifications (
1827
- id TEXT PRIMARY KEY,
1828
- agent_id TEXT NOT NULL,
1829
- agent_role TEXT NOT NULL,
1830
- event TEXT NOT NULL,
1831
- project TEXT NOT NULL,
1832
- summary TEXT NOT NULL,
1833
- task_file TEXT,
1834
- session_scope TEXT,
1835
- read INTEGER NOT NULL DEFAULT 0,
1836
- created_at TEXT NOT NULL
1837
- );
1838
-
1839
- CREATE INDEX IF NOT EXISTS idx_notifications_read
1840
- ON notifications(read);
1841
-
1842
- CREATE INDEX IF NOT EXISTS idx_notifications_agent
1843
- ON notifications(agent_id, session_scope);
1844
-
1845
- CREATE INDEX IF NOT EXISTS idx_notifications_task_file
1846
- ON notifications(task_file);
1847
- `);
1848
- await client.executeMultiple(`
1849
- CREATE TABLE IF NOT EXISTS schedules (
1850
- id TEXT PRIMARY KEY,
1851
- cron TEXT NOT NULL,
1852
- description TEXT NOT NULL,
1853
- job_type TEXT NOT NULL DEFAULT 'report',
1854
- prompt TEXT,
1855
- assigned_to TEXT,
1856
- project_name TEXT,
1857
- active INTEGER NOT NULL DEFAULT 1,
1858
- use_crontab INTEGER NOT NULL DEFAULT 0,
1859
- created_at TEXT NOT NULL
1860
- );
1861
- `);
1862
- await client.executeMultiple(`
1863
- CREATE TABLE IF NOT EXISTS device_registry (
1864
- device_id TEXT PRIMARY KEY,
1865
- friendly_name TEXT NOT NULL,
1866
- hostname TEXT NOT NULL,
1867
- projects TEXT NOT NULL DEFAULT '[]',
1868
- agents TEXT NOT NULL DEFAULT '[]',
1869
- connected INTEGER DEFAULT 0,
1870
- last_seen TEXT NOT NULL
1871
- );
1872
- `);
1873
- await client.executeMultiple(`
1874
- CREATE TABLE IF NOT EXISTS messages (
1875
- id TEXT PRIMARY KEY,
1876
- from_agent TEXT NOT NULL,
1877
- from_device TEXT NOT NULL DEFAULT 'local',
1878
- target_agent TEXT NOT NULL,
1879
- target_project TEXT,
1880
- target_device TEXT NOT NULL DEFAULT 'local',
1881
- session_scope TEXT,
1882
- content TEXT NOT NULL,
1883
- priority TEXT DEFAULT 'normal',
1884
- status TEXT DEFAULT 'pending',
1885
- server_seq INTEGER,
1886
- retry_count INTEGER DEFAULT 0,
1887
- created_at TEXT NOT NULL,
1888
- delivered_at TEXT,
1889
- processed_at TEXT,
1890
- failed_at TEXT,
1891
- failure_reason TEXT
1892
- );
1893
-
1894
- CREATE INDEX IF NOT EXISTS idx_messages_target
1895
- ON messages(target_agent, session_scope, status);
1896
-
1897
- CREATE INDEX IF NOT EXISTS idx_messages_conversation_order
1898
- ON messages(target_agent, session_scope, from_agent, server_seq);
1899
- `);
1900
- try {
1901
- await client.execute({
1902
- sql: `ALTER TABLE notifications ADD COLUMN session_scope TEXT`,
1903
- args: []
1904
- });
1905
- } catch (e) {
1906
- logCatchDebug("migration", e);
1907
- }
1908
- try {
1909
- await client.execute({
1910
- sql: `ALTER TABLE messages ADD COLUMN session_scope TEXT`,
1911
- args: []
1912
- });
1913
- } catch (e) {
1914
- logCatchDebug("migration", e);
1915
- }
1916
- await client.executeMultiple(`
1917
- CREATE INDEX IF NOT EXISTS idx_notifications_agent_scope_read
1918
- ON notifications(agent_id, session_scope, read, created_at);
1919
-
1920
- CREATE INDEX IF NOT EXISTS idx_messages_target_scope_status
1921
- ON messages(target_agent, session_scope, status, created_at);
1922
- `);
1923
- try {
1924
- await client.execute({
1925
- sql: `UPDATE memories SET project_name = 'exe-create' WHERE project_name = 'web'`,
1926
- args: []
1927
- });
1928
- await client.execute({
1929
- sql: `UPDATE memories SET project_name = 'exe-os' WHERE project_name = 'worker'`,
1930
- args: []
1931
- });
1932
- await client.execute({
1933
- sql: `UPDATE tasks SET project_name = 'exe-create' WHERE project_name = 'web'`,
1934
- args: []
1935
- });
1936
- await client.execute({
1937
- sql: `UPDATE tasks SET project_name = 'exe-os' WHERE project_name = 'worker'`,
1938
- args: []
1939
- });
1940
- } catch (e) {
1941
- logCatchDebug("migration", e);
1942
- }
1943
- await client.executeMultiple(`
1944
- CREATE TABLE IF NOT EXISTS trajectories (
1945
- id TEXT PRIMARY KEY,
1946
- task_id TEXT NOT NULL,
1947
- agent_id TEXT NOT NULL,
1948
- project_name TEXT NOT NULL,
1949
- task_title TEXT NOT NULL,
1950
- signature TEXT NOT NULL,
1951
- signature_hash TEXT NOT NULL,
1952
- tool_count INTEGER NOT NULL,
1953
- skill_id TEXT,
1954
- created_at TEXT NOT NULL
1955
- );
1956
-
1957
- CREATE INDEX IF NOT EXISTS idx_trajectories_hash
1958
- ON trajectories(signature_hash);
1959
-
1960
- CREATE INDEX IF NOT EXISTS idx_trajectories_agent
1961
- ON trajectories(agent_id);
1962
- `);
1963
- try {
1964
- await client.execute("ALTER TABLE trajectories ADD COLUMN skill_id TEXT");
1965
- } catch (e) {
1966
- logCatchDebug("migration", e);
1967
- }
1968
- await client.executeMultiple(`
1969
- CREATE TABLE IF NOT EXISTS consolidations (
1970
- id TEXT PRIMARY KEY,
1971
- consolidated_memory_id TEXT NOT NULL,
1972
- source_memory_id TEXT NOT NULL,
1973
- created_at TEXT NOT NULL
1974
- );
1975
-
1976
- CREATE INDEX IF NOT EXISTS idx_consolidations_source
1977
- ON consolidations(source_memory_id);
1978
- `);
1979
- await client.executeMultiple(`
1980
- CREATE TABLE IF NOT EXISTS audit_trail (
1981
- id INTEGER PRIMARY KEY AUTOINCREMENT,
1982
- timestamp TEXT NOT NULL,
1983
- session_id TEXT NOT NULL,
1984
- agent_id TEXT NOT NULL,
1985
- tool TEXT NOT NULL,
1986
- input TEXT,
1987
- decision TEXT NOT NULL,
1988
- reason TEXT,
1989
- is_customer_facing INTEGER NOT NULL DEFAULT 0
1990
- );
1991
-
1992
- CREATE INDEX IF NOT EXISTS idx_audit_trail_agent
1993
- ON audit_trail(agent_id, timestamp);
1994
-
1995
- CREATE INDEX IF NOT EXISTS idx_audit_trail_session
1996
- ON audit_trail(session_id);
1997
- `);
1998
- try {
1999
- await client.execute({
2000
- sql: `ALTER TABLE memories ADD COLUMN consolidated INTEGER NOT NULL DEFAULT 0`,
2001
- args: []
2002
- });
2003
- } catch (e) {
2004
- logCatchDebug("migration", e);
2005
- }
2006
- try {
2007
- await client.execute({
2008
- sql: `ALTER TABLE memories ADD COLUMN importance INTEGER DEFAULT 5`,
2009
- args: []
2010
- });
2011
- } catch (e) {
2012
- logCatchDebug("migration", e);
2013
- }
2014
- try {
2015
- await client.execute({
2016
- sql: `ALTER TABLE memories ADD COLUMN status TEXT DEFAULT 'active'`,
2017
- args: []
2018
- });
2019
- } catch (e) {
2020
- logCatchDebug("migration", e);
2021
- }
2022
- try {
2023
- await client.execute({
2024
- sql: `ALTER TABLE memories ADD COLUMN deleted_at TEXT`,
2025
- args: []
2026
- });
2027
- } catch (e) {
2028
- logCatchDebug("migration", e);
2029
- }
2030
- try {
2031
- await client.execute({
2032
- sql: `ALTER TABLE memories ADD COLUMN confidence REAL DEFAULT 0.7`,
2033
- args: []
2034
- });
2035
- } catch (e) {
2036
- logCatchDebug("migration", e);
2037
- }
2038
- try {
2039
- await client.execute({
2040
- sql: `ALTER TABLE memories ADD COLUMN last_accessed TEXT`,
2041
- args: []
2042
- });
2043
- } catch (e) {
2044
- logCatchDebug("migration", e);
2045
- }
2046
- try {
2047
- await client.execute({
2048
- sql: `UPDATE memories SET last_accessed = timestamp WHERE last_accessed IS NULL`,
2049
- args: []
2050
- });
2051
- } catch (e) {
2052
- logCatchDebug("migration", e);
2053
- }
2054
- try {
2055
- await client.execute({
2056
- sql: `ALTER TABLE memories ADD COLUMN wiki_synced INTEGER DEFAULT 0`,
2057
- args: []
2058
- });
2059
- } catch (e) {
2060
- logCatchDebug("migration", e);
2061
- }
2062
- try {
2063
- await client.execute({
2064
- sql: `ALTER TABLE memories ADD COLUMN graph_extracted INTEGER DEFAULT 0`,
2065
- args: []
2066
- });
2067
- } catch (e) {
2068
- logCatchDebug("migration", e);
2069
- }
2070
- for (const col of [
2071
- "ALTER TABLE memories ADD COLUMN content_hash TEXT",
2072
- "ALTER TABLE memories ADD COLUMN graph_extracted_hash TEXT"
2073
- ]) {
2074
- try {
2075
- await client.execute(col);
2076
- } catch (e) {
2077
- logCatchDebug("migration", e);
2078
- }
2079
- }
2080
- try {
2081
- await client.execute(
2082
- `CREATE INDEX IF NOT EXISTS idx_memories_content_hash ON memories(content_hash, agent_id)`
2083
- );
2084
- } catch (e) {
2085
- logCatchDebug("migration", e);
2086
- }
2087
- try {
2088
- await client.execute(
2089
- `CREATE INDEX IF NOT EXISTS idx_memories_scoped_content_hash
2090
- ON memories(content_hash, agent_id, project_name, memory_type)
2091
- WHERE content_hash IS NOT NULL`
2092
- );
2093
- } catch (e) {
2094
- logCatchDebug("migration", e);
2095
- }
2096
- await client.executeMultiple(`
2097
- CREATE TABLE IF NOT EXISTS entities (
2098
- id TEXT PRIMARY KEY,
2099
- name TEXT NOT NULL,
2100
- type TEXT NOT NULL,
2101
- first_seen TEXT NOT NULL,
2102
- last_seen TEXT NOT NULL,
2103
- properties TEXT DEFAULT '{}',
2104
- UNIQUE(name, type)
2105
- );
2106
-
2107
- CREATE TABLE IF NOT EXISTS relationships (
2108
- id TEXT PRIMARY KEY,
2109
- source_entity_id TEXT NOT NULL,
2110
- target_entity_id TEXT NOT NULL,
2111
- type TEXT NOT NULL,
2112
- weight REAL DEFAULT 1.0,
2113
- timestamp TEXT NOT NULL,
2114
- properties TEXT DEFAULT '{}',
2115
- UNIQUE(source_entity_id, target_entity_id, type)
2116
- );
2117
-
2118
- CREATE TABLE IF NOT EXISTS entity_memories (
2119
- entity_id TEXT NOT NULL,
2120
- memory_id TEXT NOT NULL,
2121
- PRIMARY KEY (entity_id, memory_id)
2122
- );
2123
-
2124
- CREATE TABLE IF NOT EXISTS relationship_memories (
2125
- relationship_id TEXT NOT NULL,
2126
- memory_id TEXT NOT NULL,
2127
- PRIMARY KEY (relationship_id, memory_id)
2128
- );
2129
-
2130
- CREATE INDEX IF NOT EXISTS idx_entities_name ON entities(name);
2131
- CREATE INDEX IF NOT EXISTS idx_entities_type ON entities(type);
2132
- CREATE INDEX IF NOT EXISTS idx_relationships_source ON relationships(source_entity_id);
2133
- CREATE INDEX IF NOT EXISTS idx_relationships_target ON relationships(target_entity_id);
2134
-
2135
- CREATE TABLE IF NOT EXISTS hyperedges (
2136
- id TEXT PRIMARY KEY,
2137
- label TEXT NOT NULL,
2138
- relation TEXT NOT NULL,
2139
- confidence REAL DEFAULT 1.0,
2140
- timestamp TEXT NOT NULL
2141
- );
2142
-
2143
- CREATE TABLE IF NOT EXISTS hyperedge_nodes (
2144
- hyperedge_id TEXT NOT NULL,
2145
- entity_id TEXT NOT NULL,
2146
- PRIMARY KEY (hyperedge_id, entity_id)
2147
- );
2148
-
2149
- CREATE VIRTUAL TABLE IF NOT EXISTS entities_fts USING fts5(
2150
- name,
2151
- content=entities,
2152
- content_rowid=rowid
2153
- );
2154
-
2155
- CREATE TRIGGER IF NOT EXISTS entities_fts_ai AFTER INSERT ON entities BEGIN
2156
- INSERT INTO entities_fts(rowid, name) VALUES (new.rowid, new.name);
2157
- END;
2158
-
2159
- CREATE TRIGGER IF NOT EXISTS entities_fts_ad AFTER DELETE ON entities BEGIN
2160
- INSERT INTO entities_fts(entities_fts, rowid, name) VALUES('delete', old.rowid, old.name);
2161
- END;
2162
-
2163
- CREATE TRIGGER IF NOT EXISTS entities_fts_au AFTER UPDATE ON entities BEGIN
2164
- INSERT INTO entities_fts(entities_fts, rowid, name) VALUES('delete', old.rowid, old.name);
2165
- INSERT INTO entities_fts(rowid, name) VALUES (new.rowid, new.name);
2166
- END;
2167
- `);
2168
- try {
2169
- await client.execute("INSERT INTO entities_fts(entities_fts) VALUES('rebuild')");
2170
- } catch (e) {
2171
- logCatchDebug("migration", e);
2172
- }
2173
- await client.executeMultiple(`
2174
- CREATE TABLE IF NOT EXISTS entity_aliases (
2175
- alias TEXT NOT NULL PRIMARY KEY,
2176
- canonical_entity_id TEXT NOT NULL
2177
- );
2178
- CREATE INDEX IF NOT EXISTS idx_entity_aliases_canonical ON entity_aliases(canonical_entity_id);
2179
- `);
2180
- for (const col of [
2181
- "ALTER TABLE relationships ADD COLUMN confidence REAL DEFAULT 1.0",
2182
- "ALTER TABLE relationships ADD COLUMN confidence_label TEXT DEFAULT 'extracted'"
2183
- ]) {
2184
- try {
2185
- await client.execute(col);
2186
- } catch (e) {
2187
- logCatchDebug("migration", e);
2188
- }
2189
- }
2190
- try {
2191
- await client.execute(
2192
- `CREATE INDEX IF NOT EXISTS idx_memories_status ON memories(status)`
2193
- );
2194
- } catch (e) {
2195
- logCatchDebug("migration", e);
2196
- }
2197
- await client.executeMultiple(`
2198
- CREATE TABLE IF NOT EXISTS identity (
2199
- id INTEGER PRIMARY KEY AUTOINCREMENT,
2200
- agent_id TEXT NOT NULL UNIQUE,
2201
- content_hash TEXT NOT NULL,
2202
- updated_at TEXT NOT NULL,
2203
- updated_by TEXT NOT NULL
2204
- );
2205
-
2206
- CREATE INDEX IF NOT EXISTS idx_identity_agent ON identity(agent_id);
2207
- `);
2208
- await client.executeMultiple(`
2209
- CREATE TABLE IF NOT EXISTS chat_history (
2210
- id INTEGER PRIMARY KEY AUTOINCREMENT,
2211
- session_id TEXT NOT NULL,
2212
- role TEXT NOT NULL,
2213
- content TEXT NOT NULL,
2214
- tool_name TEXT,
2215
- tool_id TEXT,
2216
- is_error INTEGER NOT NULL DEFAULT 0,
2217
- timestamp INTEGER NOT NULL
2218
- );
2219
-
2220
- CREATE INDEX IF NOT EXISTS idx_chat_history_session
2221
- ON chat_history(session_id, id);
2222
- `);
2223
- await client.executeMultiple(`
2224
- CREATE TABLE IF NOT EXISTS session_events (
2225
- id TEXT PRIMARY KEY,
2226
- agent_id TEXT NOT NULL,
2227
- agent_role TEXT NOT NULL,
2228
- session_id TEXT NOT NULL,
2229
- session_scope TEXT,
2230
- project_name TEXT NOT NULL,
2231
- event_index INTEGER NOT NULL,
2232
- event_type TEXT NOT NULL,
2233
- tool_name TEXT,
2234
- tool_use_id TEXT,
2235
- content TEXT NOT NULL,
2236
- payload_json TEXT,
2237
- has_error INTEGER NOT NULL DEFAULT 0,
2238
- created_at TEXT NOT NULL
2239
- );
2240
-
2241
- CREATE INDEX IF NOT EXISTS idx_session_events_agent_time
2242
- ON session_events(agent_id, created_at DESC);
2243
-
2244
- CREATE INDEX IF NOT EXISTS idx_session_events_session_index
2245
- ON session_events(session_id, event_index);
2246
-
2247
- CREATE INDEX IF NOT EXISTS idx_session_events_scope_agent_time
2248
- ON session_events(session_scope, agent_id, created_at DESC);
2249
- `);
2250
- await client.executeMultiple(`
2251
- CREATE TABLE IF NOT EXISTS workspaces (
2252
- id TEXT PRIMARY KEY,
2253
- slug TEXT NOT NULL UNIQUE,
2254
- name TEXT NOT NULL,
2255
- owner_agent_id TEXT,
2256
- created_at TEXT NOT NULL,
2257
- metadata TEXT
2258
- );
2259
-
2260
- CREATE INDEX IF NOT EXISTS idx_workspaces_slug
2261
- ON workspaces(slug);
2262
- `);
2263
- await client.executeMultiple(`
2264
- CREATE TABLE IF NOT EXISTS documents (
2265
- id TEXT PRIMARY KEY,
2266
- workspace_id TEXT NOT NULL,
2267
- filename TEXT NOT NULL,
2268
- mime TEXT,
2269
- source_type TEXT,
2270
- user_id TEXT,
2271
- uploaded_at TEXT NOT NULL,
2272
- metadata TEXT,
2273
- FOREIGN KEY (workspace_id) REFERENCES workspaces(id)
2274
- );
2275
-
2276
- CREATE INDEX IF NOT EXISTS idx_documents_workspace
2277
- ON documents(workspace_id);
2278
-
2279
- CREATE INDEX IF NOT EXISTS idx_documents_user
2280
- ON documents(user_id);
2281
- `);
2282
- for (const column of [
2283
- "workspace_id TEXT",
2284
- "document_id TEXT",
2285
- "user_id TEXT",
2286
- "char_offset INTEGER",
2287
- "page_number INTEGER"
2288
- ]) {
2289
- try {
2290
- await client.execute({
2291
- sql: `ALTER TABLE memories ADD COLUMN ${column}`,
2292
- args: []
2293
- });
2294
- } catch (e) {
2295
- logCatchDebug("migration", e);
2296
- }
2297
- }
2298
- for (const col of [
2299
- "ALTER TABLE memories ADD COLUMN source_path TEXT",
2300
- "ALTER TABLE memories ADD COLUMN source_type TEXT DEFAULT 'text'"
2301
- ]) {
2302
- try {
2303
- await client.execute(col);
2304
- } catch (e) {
2305
- logCatchDebug("migration", e);
2306
- }
2307
- }
2308
- await client.executeMultiple(`
2309
- CREATE INDEX IF NOT EXISTS idx_memories_workspace
2310
- ON memories(workspace_id);
2311
-
2312
- CREATE INDEX IF NOT EXISTS idx_memories_document
2313
- ON memories(document_id);
2314
-
2315
- CREATE INDEX IF NOT EXISTS idx_memories_user
2316
- ON memories(user_id);
2317
- `);
2318
- await client.executeMultiple(`
2319
- CREATE TABLE IF NOT EXISTS session_kills (
2320
- id TEXT PRIMARY KEY,
2321
- session_name TEXT NOT NULL,
2322
- agent_id TEXT NOT NULL,
2323
- killed_at TIMESTAMP NOT NULL,
2324
- reason TEXT NOT NULL,
2325
- ticks_idle INTEGER,
2326
- estimated_tokens_saved INTEGER
2327
- );
2328
-
2329
- CREATE INDEX IF NOT EXISTS idx_session_kills_killed_at
2330
- ON session_kills(killed_at);
2331
-
2332
- CREATE INDEX IF NOT EXISTS idx_session_kills_agent
2333
- ON session_kills(agent_id);
2334
- `);
2335
- await client.execute(`
2336
- CREATE TABLE IF NOT EXISTS company_procedures (
2337
- id TEXT PRIMARY KEY,
2338
- title TEXT NOT NULL,
2339
- content TEXT NOT NULL,
2340
- priority TEXT NOT NULL DEFAULT 'p0',
2341
- domain TEXT,
2342
- active INTEGER NOT NULL DEFAULT 1,
2343
- created_at TEXT NOT NULL,
2344
- updated_at TEXT NOT NULL
2345
- )
2346
- `);
2347
- const legacyProcedureObject = await client.execute({
2348
- sql: "SELECT type FROM sqlite_master WHERE name = 'global_procedures'",
2349
- args: []
2350
- });
2351
- const legacyProcedureType = legacyProcedureObject.rows[0]?.type == null ? null : String(legacyProcedureObject.rows[0].type);
2352
- if (legacyProcedureType === "table") {
2353
- await client.execute(`
2354
- INSERT OR IGNORE INTO company_procedures
2355
- (id, title, content, priority, domain, active, created_at, updated_at)
2356
- SELECT id, title, content, priority, domain, active, created_at, updated_at
2357
- FROM global_procedures
2358
- `);
2359
- await client.executeMultiple(`
2360
- CREATE TRIGGER IF NOT EXISTS global_procedures_mirror_insert
2361
- AFTER INSERT ON global_procedures
2362
- BEGIN
2363
- INSERT OR IGNORE INTO company_procedures
2364
- (id, title, content, priority, domain, active, created_at, updated_at)
2365
- VALUES
2366
- (NEW.id, NEW.title, NEW.content, NEW.priority, NEW.domain, NEW.active, NEW.created_at, NEW.updated_at);
2367
- END;
2368
-
2369
- CREATE TRIGGER IF NOT EXISTS global_procedures_mirror_update
2370
- AFTER UPDATE ON global_procedures
2371
- BEGIN
2372
- UPDATE company_procedures
2373
- SET title = NEW.title,
2374
- content = NEW.content,
2375
- priority = NEW.priority,
2376
- domain = NEW.domain,
2377
- active = NEW.active,
2378
- created_at = NEW.created_at,
2379
- updated_at = NEW.updated_at
2380
- WHERE id = OLD.id;
2381
- END;
2382
- `);
2383
- } else {
2384
- await client.execute(`
2385
- CREATE VIEW IF NOT EXISTS global_procedures AS
2386
- SELECT id, title, content, priority, domain, active, created_at, updated_at
2387
- FROM company_procedures
2388
- `);
2389
- await client.executeMultiple(`
2390
- CREATE TRIGGER IF NOT EXISTS global_procedures_insert
2391
- INSTEAD OF INSERT ON global_procedures
2392
- BEGIN
2393
- INSERT INTO company_procedures
2394
- (id, title, content, priority, domain, active, created_at, updated_at)
2395
- VALUES
2396
- (NEW.id, NEW.title, NEW.content, NEW.priority, NEW.domain, NEW.active, NEW.created_at, NEW.updated_at);
2397
- END;
2398
-
2399
- CREATE TRIGGER IF NOT EXISTS global_procedures_update
2400
- INSTEAD OF UPDATE ON global_procedures
2401
- BEGIN
2402
- UPDATE company_procedures
2403
- SET title = NEW.title,
2404
- content = NEW.content,
2405
- priority = NEW.priority,
2406
- domain = NEW.domain,
2407
- active = NEW.active,
2408
- created_at = NEW.created_at,
2409
- updated_at = NEW.updated_at
2410
- WHERE id = OLD.id;
2411
- END;
2412
- `);
2413
- }
2414
- await client.executeMultiple(`
2415
- CREATE TABLE IF NOT EXISTS conversations (
2416
- id TEXT PRIMARY KEY,
2417
- platform TEXT NOT NULL,
2418
- external_id TEXT,
2419
- sender_id TEXT NOT NULL,
2420
- sender_name TEXT,
2421
- sender_phone TEXT,
2422
- sender_email TEXT,
2423
- recipient_id TEXT,
2424
- channel_id TEXT NOT NULL,
2425
- thread_id TEXT,
2426
- reply_to_id TEXT,
2427
- content_text TEXT,
2428
- content_media TEXT,
2429
- content_metadata TEXT,
2430
- agent_response TEXT,
2431
- agent_name TEXT,
2432
- timestamp TEXT NOT NULL,
2433
- ingested_at TEXT NOT NULL
2434
- );
2435
-
2436
- CREATE INDEX IF NOT EXISTS idx_conversations_platform
2437
- ON conversations(platform);
2438
-
2439
- CREATE INDEX IF NOT EXISTS idx_conversations_sender
2440
- ON conversations(sender_id);
2441
-
2442
- CREATE INDEX IF NOT EXISTS idx_conversations_timestamp
2443
- ON conversations(timestamp);
2444
-
2445
- CREATE INDEX IF NOT EXISTS idx_conversations_thread
2446
- ON conversations(thread_id);
2447
-
2448
- CREATE INDEX IF NOT EXISTS idx_conversations_channel
2449
- ON conversations(channel_id);
2450
- `);
2451
- await client.executeMultiple(`
2452
- CREATE TABLE IF NOT EXISTS session_agent_map (
2453
- session_uuid TEXT PRIMARY KEY,
2454
- agent_id TEXT NOT NULL,
2455
- session_name TEXT,
2456
- task_id TEXT,
2457
- project_name TEXT,
2458
- started_at TEXT NOT NULL,
2459
- cache_cold_count INTEGER NOT NULL DEFAULT 0
2460
- );
2461
-
2462
- CREATE INDEX IF NOT EXISTS idx_session_agent_map_agent
2463
- ON session_agent_map(agent_id);
2464
- `);
2465
- await client.executeMultiple(`
2466
- CREATE TABLE IF NOT EXISTS agent_file_reads (
2467
- session_uuid TEXT NOT NULL,
2468
- agent_id TEXT NOT NULL,
2469
- file_path TEXT NOT NULL,
2470
- read_at TEXT NOT NULL,
2471
- commit_hash TEXT,
2472
- PRIMARY KEY (session_uuid, file_path)
2473
- );
2474
-
2475
- CREATE INDEX IF NOT EXISTS idx_agent_file_reads_agent_read_at
2476
- ON agent_file_reads(agent_id, read_at);
2477
- `);
2478
- try {
2479
- const mapCount = await client.execute({ sql: `SELECT COUNT(*) as cnt FROM session_agent_map`, args: [] });
2480
- if (Number(mapCount.rows[0]?.cnt ?? 0) === 0) {
2481
- await client.execute({
2482
- sql: `INSERT OR IGNORE INTO session_agent_map (session_uuid, agent_id, session_name, started_at)
2483
- SELECT session_id, agent_id, '', MIN(timestamp)
2484
- FROM memories
2485
- WHERE session_id IS NOT NULL AND session_id != '' AND agent_id IS NOT NULL AND agent_id != ''
2486
- GROUP BY session_id, agent_id`,
2487
- args: []
2488
- });
2489
- }
2490
- } catch (e) {
2491
- logCatchDebug("session_agent_map backfill", e);
2492
- }
2493
- try {
2494
- await client.execute({
2495
- sql: `ALTER TABLE session_agent_map ADD COLUMN cache_cold_count INTEGER NOT NULL DEFAULT 0`,
2496
- args: []
2497
- });
2498
- } catch (e) {
2499
- logCatchDebug("migration", e);
2500
- }
2501
- try {
2502
- await client.execute({
2503
- sql: `ALTER TABLE tasks ADD COLUMN budget_tokens INTEGER`,
2504
- args: []
2505
- });
2506
- } catch (e) {
2507
- logCatchDebug("migration", e);
2508
- }
2509
- try {
2510
- await client.execute({
2511
- sql: `ALTER TABLE tasks ADD COLUMN budget_fallback_model TEXT`,
2512
- args: []
2513
- });
2514
- } catch (e) {
2515
- logCatchDebug("migration", e);
2516
- }
2517
- try {
2518
- await client.execute({
2519
- sql: `ALTER TABLE tasks ADD COLUMN tokens_used INTEGER DEFAULT 0`,
2520
- args: []
2521
- });
2522
- } catch (e) {
2523
- logCatchDebug("migration", e);
2524
- }
2525
- try {
2526
- await client.execute({
2527
- sql: `ALTER TABLE tasks ADD COLUMN tokens_warned_at INTEGER`,
2528
- args: []
2529
- });
2530
- } catch (e) {
2531
- logCatchDebug("migration", e);
2532
- }
2533
- try {
2534
- await client.execute({
2535
- sql: `ALTER TABLE tasks ADD COLUMN spawn_runtime TEXT`,
2536
- args: []
2537
- });
2538
- } catch (e) {
2539
- logCatchDebug("migration", e);
2540
- }
2541
- try {
2542
- await client.execute({
2543
- sql: `ALTER TABLE tasks ADD COLUMN spawn_model TEXT`,
2544
- args: []
2545
- });
2546
- } catch (e) {
2547
- logCatchDebug("migration", e);
2548
- }
2549
- await client.executeMultiple(`
2550
- CREATE VIRTUAL TABLE IF NOT EXISTS conversations_fts USING fts5(
2551
- content_text,
2552
- sender_name,
2553
- agent_response,
2554
- content='conversations',
2555
- content_rowid='rowid'
2556
- );
2557
-
2558
- CREATE TRIGGER IF NOT EXISTS conversations_fts_ai AFTER INSERT ON conversations BEGIN
2559
- INSERT INTO conversations_fts(rowid, content_text, sender_name, agent_response)
2560
- VALUES (new.rowid, new.content_text, new.sender_name, new.agent_response);
2561
- END;
2562
-
2563
- CREATE TRIGGER IF NOT EXISTS conversations_fts_ad AFTER DELETE ON conversations BEGIN
2564
- INSERT INTO conversations_fts(conversations_fts, rowid, content_text, sender_name, agent_response)
2565
- VALUES('delete', old.rowid, old.content_text, old.sender_name, old.agent_response);
2566
- END;
2567
-
2568
- CREATE TRIGGER IF NOT EXISTS conversations_fts_au AFTER UPDATE ON conversations BEGIN
2569
- INSERT INTO conversations_fts(conversations_fts, rowid, content_text, sender_name, agent_response)
2570
- VALUES('delete', old.rowid, old.content_text, old.sender_name, old.agent_response);
2571
- INSERT INTO conversations_fts(rowid, content_text, sender_name, agent_response)
2572
- VALUES (new.rowid, new.content_text, new.sender_name, new.agent_response);
2573
- END;
2574
- `);
2575
- await client.executeMultiple(`
2576
- CREATE TABLE IF NOT EXISTS memory_cards (
2577
- id TEXT PRIMARY KEY,
2578
- memory_id TEXT NOT NULL,
2579
- agent_id TEXT NOT NULL,
2580
- session_id TEXT NOT NULL,
2581
- project_name TEXT,
2582
- timestamp TEXT NOT NULL,
2583
- card_type TEXT NOT NULL,
2584
- subject TEXT,
2585
- predicate TEXT,
2586
- object TEXT,
2587
- content TEXT NOT NULL,
2588
- source_ref TEXT,
2589
- confidence REAL DEFAULT 0.6,
2590
- active INTEGER DEFAULT 1,
2591
- created_at TEXT NOT NULL
2592
- );
2593
-
2594
- CREATE INDEX IF NOT EXISTS idx_memory_cards_agent
2595
- ON memory_cards(agent_id, active, timestamp);
2596
-
2597
- CREATE INDEX IF NOT EXISTS idx_memory_cards_memory
2598
- ON memory_cards(memory_id);
2599
-
2600
- CREATE VIRTUAL TABLE IF NOT EXISTS memory_cards_fts
2601
- USING fts5(content, subject, predicate, object, content='memory_cards', content_rowid='rowid');
2602
-
2603
- CREATE TRIGGER IF NOT EXISTS memory_cards_fts_ai AFTER INSERT ON memory_cards BEGIN
2604
- INSERT INTO memory_cards_fts(rowid, content, subject, predicate, object)
2605
- VALUES (new.rowid, new.content, new.subject, new.predicate, new.object);
2606
- END;
2607
-
2608
- CREATE TRIGGER IF NOT EXISTS memory_cards_fts_ad AFTER DELETE ON memory_cards BEGIN
2609
- INSERT INTO memory_cards_fts(memory_cards_fts, rowid, content, subject, predicate, object)
2610
- VALUES('delete', old.rowid, old.content, old.subject, old.predicate, old.object);
2611
- END;
2612
-
2613
- CREATE TRIGGER IF NOT EXISTS memory_cards_fts_au AFTER UPDATE ON memory_cards BEGIN
2614
- INSERT INTO memory_cards_fts(memory_cards_fts, rowid, content, subject, predicate, object)
2615
- VALUES('delete', old.rowid, old.content, old.subject, old.predicate, old.object);
2616
- INSERT INTO memory_cards_fts(rowid, content, subject, predicate, object)
2617
- VALUES (new.rowid, new.content, new.subject, new.predicate, new.object);
2618
- END;
2619
- `);
2620
- await client.executeMultiple(`
2621
- CREATE TABLE IF NOT EXISTS agent_sessions (
2622
- id TEXT PRIMARY KEY,
2623
- agent_id TEXT NOT NULL,
2624
- project_name TEXT,
2625
- started_at TEXT NOT NULL,
2626
- last_event_at TEXT NOT NULL,
2627
- event_count INTEGER NOT NULL DEFAULT 0,
2628
- properties TEXT DEFAULT '{}'
2629
- );
2630
-
2631
- CREATE INDEX IF NOT EXISTS idx_agent_sessions_agent_time
2632
- ON agent_sessions(agent_id, started_at);
2633
-
2634
- CREATE TABLE IF NOT EXISTS agent_goals (
2635
- id TEXT PRIMARY KEY,
2636
- statement TEXT NOT NULL,
2637
- owner_agent_id TEXT,
2638
- project_name TEXT,
2639
- status TEXT NOT NULL DEFAULT 'open',
2640
- priority INTEGER NOT NULL DEFAULT 5,
2641
- success_criteria TEXT,
2642
- parent_goal_id TEXT,
2643
- due_at TEXT,
2644
- achieved_at TEXT,
2645
- supersedes_id TEXT,
2646
- created_at TEXT NOT NULL,
2647
- updated_at TEXT NOT NULL,
2648
- source_memory_id TEXT
2649
- );
2650
-
2651
- CREATE INDEX IF NOT EXISTS idx_agent_goals_project_status
2652
- ON agent_goals(project_name, status, priority);
2653
-
2654
- CREATE TABLE IF NOT EXISTS agent_events (
2655
- id TEXT PRIMARY KEY,
2656
- event_type TEXT NOT NULL,
2657
- occurred_at TEXT NOT NULL,
2658
- sequence_index INTEGER NOT NULL,
2659
- actor_agent_id TEXT,
2660
- agent_role TEXT,
2661
- project_name TEXT,
2662
- session_id TEXT,
2663
- task_id TEXT,
2664
- goal_id TEXT,
2665
- parent_event_id TEXT,
2666
- intention TEXT,
2667
- outcome TEXT,
2668
- evidence_memory_id TEXT,
2669
- impact TEXT,
2670
- payload TEXT DEFAULT '{}',
2671
- created_at TEXT NOT NULL
2672
- );
2673
-
2674
- CREATE INDEX IF NOT EXISTS idx_agent_events_time
2675
- ON agent_events(occurred_at, sequence_index);
2676
-
2677
- CREATE INDEX IF NOT EXISTS idx_agent_events_session_seq
2678
- ON agent_events(session_id, sequence_index);
2679
-
2680
- CREATE INDEX IF NOT EXISTS idx_agent_events_goal_time
2681
- ON agent_events(goal_id, occurred_at);
2682
-
2683
- CREATE INDEX IF NOT EXISTS idx_agent_events_memory
2684
- ON agent_events(evidence_memory_id);
2685
-
2686
- CREATE TABLE IF NOT EXISTS agent_goal_links (
2687
- id TEXT PRIMARY KEY,
2688
- goal_id TEXT NOT NULL,
2689
- link_type TEXT NOT NULL,
2690
- target_id TEXT NOT NULL,
2691
- target_type TEXT NOT NULL,
2692
- created_at TEXT NOT NULL
2693
- );
2694
-
2695
- CREATE INDEX IF NOT EXISTS idx_agent_goal_links_goal
2696
- ON agent_goal_links(goal_id, target_type);
2697
-
2698
- CREATE TABLE IF NOT EXISTS agent_semantic_labels (
2699
- id TEXT PRIMARY KEY,
2700
- source_memory_id TEXT NOT NULL,
2701
- event_id TEXT,
2702
- labeler TEXT NOT NULL,
2703
- schema_version INTEGER NOT NULL DEFAULT 1,
2704
- confidence REAL NOT NULL DEFAULT 0,
2705
- labels TEXT NOT NULL,
2706
- created_at TEXT NOT NULL,
2707
- updated_at TEXT NOT NULL
2708
- );
2709
-
2710
- CREATE INDEX IF NOT EXISTS idx_agent_semantic_labels_memory
2711
- ON agent_semantic_labels(source_memory_id, labeler);
2712
-
2713
- CREATE INDEX IF NOT EXISTS idx_agent_semantic_labels_event
2714
- ON agent_semantic_labels(event_id);
2715
-
2716
- CREATE TABLE IF NOT EXISTS agent_reflection_checkpoints (
2717
- id TEXT PRIMARY KEY,
2718
- project_name TEXT,
2719
- session_id TEXT,
2720
- window_start_at TEXT NOT NULL,
2721
- window_end_at TEXT NOT NULL,
2722
- event_count INTEGER NOT NULL DEFAULT 0,
2723
- goal_count INTEGER NOT NULL DEFAULT 0,
2724
- success_count INTEGER NOT NULL DEFAULT 0,
2725
- failure_count INTEGER NOT NULL DEFAULT 0,
2726
- risk_count INTEGER NOT NULL DEFAULT 0,
2727
- summary TEXT NOT NULL,
2728
- learnings TEXT NOT NULL DEFAULT '[]',
2729
- next_actions TEXT NOT NULL DEFAULT '[]',
2730
- evidence_event_ids TEXT NOT NULL DEFAULT '[]',
2731
- confidence REAL NOT NULL DEFAULT 0,
2732
- created_at TEXT NOT NULL
2733
- );
2734
-
2735
- CREATE INDEX IF NOT EXISTS idx_agent_reflection_project_time
2736
- ON agent_reflection_checkpoints(project_name, window_end_at);
2737
-
2738
- CREATE INDEX IF NOT EXISTS idx_agent_reflection_session_time
2739
- ON agent_reflection_checkpoints(session_id, window_end_at);
2740
- `);
2741
- try {
2742
- await client.execute({
2743
- sql: `ALTER TABLE memories ADD COLUMN tier INTEGER DEFAULT 3`,
2744
- args: []
2745
- });
2746
- } catch (e) {
2747
- logCatchDebug("migration", e);
2748
- }
2749
- try {
2750
- await client.execute(
2751
- `CREATE INDEX IF NOT EXISTS idx_memories_tier ON memories(tier)`
2752
- );
2753
- } catch (e) {
2754
- logCatchDebug("migration", e);
2755
- }
2756
- try {
2757
- await client.execute({
2758
- sql: `UPDATE memories SET tier = 1 WHERE tool_name = 'commit_to_long_term_memory' AND importance >= 8 AND tier = 3`,
2759
- args: []
2760
- });
2761
- await client.execute({
2762
- sql: `UPDATE memories SET tier = 2 WHERE tool_name IN ('store_memory', 'manual') AND importance >= 5 AND tier = 3`,
2763
- args: []
2764
- });
2765
- } catch (e) {
2766
- logCatchDebug("migration", e);
2767
- }
2768
- try {
2769
- await client.execute({
2770
- sql: `ALTER TABLE memories ADD COLUMN supersedes_id TEXT`,
2771
- args: []
2772
- });
2773
- } catch (e) {
2774
- logCatchDebug("migration", e);
2775
- }
2776
- try {
2777
- await client.execute(
2778
- `CREATE INDEX IF NOT EXISTS idx_memories_supersedes ON memories(supersedes_id) WHERE supersedes_id IS NOT NULL`
2779
- );
2780
- } catch (e) {
2781
- logCatchDebug("migration", e);
2782
- }
2783
- for (const col of [
2784
- "ALTER TABLE tasks ADD COLUMN checkpoint TEXT",
2785
- "ALTER TABLE tasks ADD COLUMN checkpoint_count INTEGER DEFAULT 0"
2786
- ]) {
2787
- try {
2788
- await client.execute(col);
2789
- } catch (e) {
2790
- logCatchDebug("migration", e);
2791
- }
2792
- }
2793
- try {
2794
- await client.execute({
2795
- sql: `ALTER TABLE memories ADD COLUMN draft INTEGER DEFAULT 0`,
2796
- args: []
2797
- });
2798
- } catch (e) {
2799
- logCatchDebug("migration", e);
2800
- }
2801
- try {
2802
- await client.execute(
2803
- `CREATE INDEX IF NOT EXISTS idx_memories_draft ON memories(draft) WHERE draft = 1`
2804
- );
2805
- } catch (e) {
2806
- logCatchDebug("migration", e);
2807
- }
2808
- for (const col of [
2809
- "ALTER TABLE memories ADD COLUMN valid_from TEXT",
2810
- "ALTER TABLE memories ADD COLUMN invalid_at TEXT"
2811
- ]) {
2812
- try {
2813
- await client.execute(col);
2814
- } catch (e) {
2815
- logCatchDebug("migration", e);
2816
- }
2817
- }
2818
- try {
2819
- await client.execute({
2820
- sql: `UPDATE memories SET valid_from = timestamp WHERE valid_from IS NULL`,
2821
- args: []
2822
- });
2823
- } catch (e) {
2824
- logCatchDebug("migration", e);
2825
- }
2826
- try {
2827
- await client.execute({
2828
- sql: `ALTER TABLE memories ADD COLUMN memory_type TEXT DEFAULT 'raw'`,
2829
- args: []
2830
- });
2831
- } catch (e) {
2832
- logCatchDebug("migration", e);
2833
- }
2834
- try {
2835
- await client.execute(
2836
- `CREATE INDEX IF NOT EXISTS idx_memories_type ON memories(memory_type)`
2837
- );
2838
- } catch (e) {
2839
- logCatchDebug("migration", e);
2840
- }
2841
- try {
2842
- await client.execute({
2843
- sql: `ALTER TABLE memories ADD COLUMN trajectory TEXT`,
2844
- args: []
2845
- });
2846
- } catch (e) {
2847
- logCatchDebug("migration", e);
2848
- }
2849
- for (const col of [
2850
- "ALTER TABLE memories ADD COLUMN intent TEXT",
2851
- "ALTER TABLE memories ADD COLUMN outcome TEXT",
2852
- "ALTER TABLE memories ADD COLUMN domain TEXT",
2853
- "ALTER TABLE memories ADD COLUMN referenced_entities TEXT",
2854
- "ALTER TABLE memories ADD COLUMN retrieval_count INTEGER DEFAULT 0",
2855
- "ALTER TABLE memories ADD COLUMN chain_position TEXT",
2856
- "ALTER TABLE memories ADD COLUMN review_status TEXT",
2857
- "ALTER TABLE memories ADD COLUMN context_window_pct INTEGER",
2858
- "ALTER TABLE memories ADD COLUMN file_paths TEXT",
2859
- "ALTER TABLE memories ADD COLUMN commit_hash TEXT",
2860
- "ALTER TABLE memories ADD COLUMN duration_ms INTEGER",
2861
- "ALTER TABLE memories ADD COLUMN token_cost REAL",
2862
- "ALTER TABLE memories ADD COLUMN audience TEXT",
2863
- "ALTER TABLE memories ADD COLUMN language_type TEXT",
2864
- "ALTER TABLE memories ADD COLUMN parent_memory_id TEXT"
2865
- ]) {
2866
- try {
2867
- await client.execute(col);
2868
- } catch (e) {
2869
- logCatchDebug("migration", e);
2870
- }
2871
- }
2872
- try {
2873
- await client.execute({
2874
- sql: `ALTER TABLE memories ADD COLUMN procedure_for TEXT`,
2875
- args: []
2876
- });
2877
- } catch (e) {
2878
- logCatchDebug("migration", e);
2879
- }
2880
- try {
2881
- await client.execute({
2882
- sql: `UPDATE tasks SET status = 'closed' WHERE status = 'done' AND result IS NOT NULL`,
2883
- args: []
2884
- });
2885
- } catch (e) {
2886
- logCatchDebug("migration", e);
2887
- }
2888
- try {
2889
- await client.execute({
2890
- sql: `ALTER TABLE memories ADD COLUMN visibility TEXT DEFAULT 'private'`,
2891
- args: []
2892
- });
2893
- } catch (e) {
2894
- logCatchDebug("migration", e);
2895
- }
2896
- try {
2897
- await client.execute({
2898
- sql: `ALTER TABLE memories ADD COLUMN strength REAL DEFAULT 1.0`,
2899
- args: []
2900
- });
2901
- } catch (e) {
2902
- logCatchDebug("migration", e);
2903
- }
2904
- }
2905
- var disposeTurso = disposeDatabase;
2906
- async function disposeDatabase() {
2907
- if (_walCheckpointTimer) {
2908
- clearInterval(_walCheckpointTimer);
2909
- _walCheckpointTimer = null;
2910
- }
2911
- if (_client) {
2912
- try {
2913
- await _client.execute("PRAGMA wal_checkpoint(PASSIVE)");
2914
- } catch (e) {
2915
- logCatchDebug("WAL checkpoint", e);
2916
- }
2917
- }
2918
- if (_daemonClient) {
2919
- _daemonClient.close();
2920
- _daemonClient = null;
2921
- }
2922
- if (_adapterClient && _adapterClient !== _resilientClient) {
2923
- _adapterClient.close();
2924
- }
2925
- _adapterClient = null;
2926
- if (_client) {
2927
- _client.close();
2928
- _client = null;
2929
- _resilientClient = null;
2930
- }
2931
- releaseDbLock();
2932
- }
1
+ import {
2
+ SOFT_DELETE_RETENTION_DAYS,
3
+ disposeDatabase,
4
+ disposeTurso,
5
+ ensureSchema,
6
+ getClient,
7
+ getRawClient,
8
+ initDaemonClient,
9
+ initDatabase,
10
+ initTurso,
11
+ init_database,
12
+ isInitialized,
13
+ setExternalClient
14
+ } from "../chunk-A2NZP64U.js";
15
+ import "../chunk-4NYQAS33.js";
16
+ import "../chunk-RUG3N6P4.js";
17
+ import "../chunk-KVPG5UT6.js";
18
+ import "../chunk-X2IMCCM5.js";
19
+ import "../chunk-KFQGP6VL.js";
20
+
21
+ // src/lib/db.ts
22
+ init_database();
2933
23
  export {
2934
24
  SOFT_DELETE_RETENTION_DAYS,
2935
25
  disposeDatabase,