@xopcai/xopc 0.0.95 → 0.0.97

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 (442) hide show
  1. package/dist/browser-ext/manifest.json +1 -1
  2. package/dist/extensions/feishu/src/outbound/media-load.js +1 -1
  3. package/dist/extensions/feishu/src/workflow-progress.js +1 -1
  4. package/dist/extensions/telegram/src/plugin.js +1 -1
  5. package/dist/extensions/telegram/src/routing-integration.js +2 -2
  6. package/dist/extensions/telegram/src/workflow-progress.js +1 -1
  7. package/dist/extensions/telegram/xopc.extension.json +1 -1
  8. package/dist/extensions/weixin/src/api/api.js +2 -2
  9. package/dist/extensions/weixin/src/auth/accounts.js +1 -1
  10. package/dist/extensions/weixin/src/cdn/upload.js +1 -1
  11. package/dist/extensions/weixin/src/media/data-url.js +1 -1
  12. package/dist/extensions/weixin/src/messaging/debug-mode.js +1 -1
  13. package/dist/extensions/weixin/src/messaging/inbound.js +1 -1
  14. package/dist/extensions/weixin/src/messaging/process-message.js +1 -1
  15. package/dist/extensions/weixin/src/plugin.js +1 -1
  16. package/dist/extensions/weixin/src/storage/sync-buf.js +1 -1
  17. package/dist/extensions/weixin/src/workflow-progress.js +1 -1
  18. package/dist/gateway/static/root/assets/{agents-CKe2LMnz.js → agents-B_YUvNi6.js} +2 -2
  19. package/dist/gateway/static/root/assets/{apps-page-Mi9mMIZ1.js → apps-page-BmwG5aur.js} +1 -1
  20. package/dist/gateway/static/root/assets/{channels-settings-BrdyC101.js → channels-settings-BiwkeKPb.js} +1 -1
  21. package/dist/gateway/static/root/assets/{channels-status-swr-D55Bu0nn.js → channels-status-swr-ChyN473C.js} +1 -1
  22. package/dist/gateway/static/root/assets/{cron-api-CPpx2l-E.js → cron-api-CvSifIfJ.js} +1 -1
  23. package/dist/gateway/static/root/assets/{cron-page-Bx2jB0YN.js → cron-page-BDqTDFy6.js} +1 -1
  24. package/dist/gateway/static/root/assets/{dist-D_AiG_Kg.js → dist-DxsUrjpy.js} +1 -1
  25. package/dist/gateway/static/root/assets/{extension-debug-page-6ieHsxRE.js → extension-debug-page-DV_Av5Jq.js} +1 -1
  26. package/dist/gateway/static/root/assets/{extension-page-B8nywHRO.js → extension-page-CwZwRhWw.js} +1 -1
  27. package/dist/gateway/static/root/assets/{extension-settings-page-DrskdEIV.js → extension-settings-page-Bb7TR1Se.js} +1 -1
  28. package/dist/gateway/static/root/assets/{fetch-B0aeeY0q.js → fetch-BLLOP2CM.js} +1 -1
  29. package/dist/gateway/static/root/assets/{field-primitives--9ooY8Xl.js → field-primitives-CyqVu1QR.js} +1 -1
  30. package/dist/gateway/static/root/assets/{heartbeat-config-api-DUZ_W1w-.js → heartbeat-config-api-Cd4M1eHt.js} +1 -1
  31. package/dist/gateway/static/root/assets/{index-Dj9FuxCm.js → index-0tS9lV85.js} +74 -74
  32. package/dist/gateway/static/root/assets/index-BJDmBCSl.css +1 -0
  33. package/dist/gateway/static/root/assets/{logs-page-CaXqhpKf.js → logs-page-BsAOSowN.js} +1 -1
  34. package/dist/gateway/static/root/assets/{note-detail-page-B91pLkEI.css → note-detail-page-D4ZIVQbk.css} +1 -1
  35. package/dist/gateway/static/root/assets/{note-detail-page-DYzym2B0.js → note-detail-page-Dlxoy6Ap.js} +54 -53
  36. package/dist/gateway/static/root/assets/{note-time-B-vSi2dR.js → note-time-B-r8yTpQ.js} +1 -1
  37. package/dist/gateway/static/root/assets/{notes-page-BkhWdGiT.js → notes-page-CHFcyqYW.js} +1 -1
  38. package/dist/gateway/static/root/assets/{sessions-page-53YFokoe.js → sessions-page-Ctu0kgt7.js} +1 -1
  39. package/dist/gateway/static/root/assets/{settings-advanced-gate-BaZmaklx.js → settings-advanced-gate-Dh0TyOOg.js} +1 -1
  40. package/dist/gateway/static/root/assets/{settings-form-section-DIJPKpTR.js → settings-form-section-DXMCEW1d.js} +1 -1
  41. package/dist/gateway/static/root/assets/{settings-page-Dvb230FF.js → settings-page-CIkZ7233.js} +1 -1
  42. package/dist/gateway/static/root/assets/{share-preview-page-CRyjTAG6.js → share-preview-page-7RV65xhJ.js} +1 -1
  43. package/dist/gateway/static/root/assets/{skills-page-C5ZJbfAe.js → skills-page-D_Az1SlU.js} +1 -1
  44. package/dist/gateway/static/root/assets/{theme-store-Cg_SuBw0.js → theme-store-e2q2yjs4.js} +1 -1
  45. package/dist/gateway/static/root/assets/url-DpFBIyN9.js +3 -0
  46. package/dist/gateway/static/root/assets/{utils-lMYoWhqo.js → utils-OA_b1q0Q.js} +1 -1
  47. package/dist/gateway/static/root/assets/{voice-api-key-field-Dda2pcUU.js → voice-api-key-field-SJml1hAt.js} +1 -1
  48. package/dist/gateway/static/root/assets/{workflow-page.utils-KIladUrU.js → workflow-page.utils-D90VVCzC.js} +1 -1
  49. package/dist/gateway/static/root/assets/{workflows-page-BTis4Z7Y.js → workflows-page-y7Btji0J.js} +1 -1
  50. package/dist/gateway/static/root/index.html +5 -5
  51. package/dist/package.js +1 -1
  52. package/dist/src/agent/agent-manager.js +12 -8
  53. package/dist/src/agent/agent-manager.js.map +1 -1
  54. package/dist/src/agent/agent-scope.d.ts +0 -1
  55. package/dist/src/agent/agent-scope.js +2 -5
  56. package/dist/src/agent/agent-scope.js.map +1 -1
  57. package/dist/src/agent/bootstrap/bootstrap-cache.d.ts +2 -0
  58. package/dist/src/agent/bootstrap/bootstrap-cache.js +10 -1
  59. package/dist/src/agent/bootstrap/bootstrap-cache.js.map +1 -1
  60. package/dist/src/agent/bootstrap/bootstrap-files.d.ts +2 -1
  61. package/dist/src/agent/bootstrap/bootstrap-files.js +34 -12
  62. package/dist/src/agent/bootstrap/bootstrap-files.js.map +1 -1
  63. package/dist/src/agent/bootstrap/load-bootstrap-files.d.ts +1 -2
  64. package/dist/src/agent/bootstrap/load-bootstrap-files.js +6 -12
  65. package/dist/src/agent/bootstrap/load-bootstrap-files.js.map +1 -1
  66. package/dist/src/agent/bootstrap/types.d.ts +5 -5
  67. package/dist/src/agent/context/workspace-seed.js +6 -6
  68. package/dist/src/agent/context/workspace-seed.js.map +1 -1
  69. package/dist/src/agent/context/workspace-state.d.ts +20 -0
  70. package/dist/src/agent/context/workspace-state.js +57 -0
  71. package/dist/src/agent/context/workspace-state.js.map +1 -0
  72. package/dist/src/agent/context/workspace-templates/AGENTS.md +0 -4
  73. package/dist/src/agent/embedded/index.d.ts +2 -2
  74. package/dist/src/agent/embedded/index.js +3 -3
  75. package/dist/src/agent/embedded/run-turn.js +0 -3
  76. package/dist/src/agent/embedded/run-turn.js.map +1 -1
  77. package/dist/src/agent/embedded/session-manager-init.d.ts +0 -17
  78. package/dist/src/agent/embedded/session-manager-init.js +1 -36
  79. package/dist/src/agent/embedded/session-manager-init.js.map +1 -1
  80. package/dist/src/agent/embedded/session-runner.d.ts +3 -12
  81. package/dist/src/agent/embedded/session-runner.js +12 -26
  82. package/dist/src/agent/embedded/session-runner.js.map +1 -1
  83. package/dist/src/agent/embedded/session-tool-result-guard.js +2 -4
  84. package/dist/src/agent/embedded/session-tool-result-guard.js.map +1 -1
  85. package/dist/src/agent/embedded/sqlite-hydrating-session-manager.d.ts +10 -0
  86. package/dist/src/agent/embedded/sqlite-hydrating-session-manager.js +34 -0
  87. package/dist/src/agent/embedded/sqlite-hydrating-session-manager.js.map +1 -0
  88. package/dist/src/agent/goals/goal-run-store.js +4 -4
  89. package/dist/src/agent/goals/persistent-goal-service.js +8 -15
  90. package/dist/src/agent/goals/persistent-goal-service.js.map +1 -1
  91. package/dist/src/agent/goals/post-turn.js +2 -2
  92. package/dist/src/agent/image/load-image-media.js +2 -2
  93. package/dist/src/agent/ipc/bus.js +1 -1
  94. package/dist/src/agent/ipc/inbox.js +2 -2
  95. package/dist/src/agent/ipc/socket.js +1 -1
  96. package/dist/src/agent/mcp/bundle-mcp-materialize.js +1 -1
  97. package/dist/src/agent/mcp/bundle-mcp-runtime.js +2 -2
  98. package/dist/src/agent/mcp/mcp-transport-config.js +1 -1
  99. package/dist/src/agent/mcp/mcp-transport.js +1 -1
  100. package/dist/src/agent/memory/builtin-memory-store.js +1 -1
  101. package/dist/src/agent/memory/dreaming/deep-promotion.js +1 -1
  102. package/dist/src/agent/memory/dreaming/events.js +1 -1
  103. package/dist/src/agent/memory/dreaming/last-run.js +1 -1
  104. package/dist/src/agent/memory/dreaming/light-sweep.js +1 -1
  105. package/dist/src/agent/memory/dreaming/preview.js +1 -1
  106. package/dist/src/agent/memory/dreaming/rem-patterns.js +1 -1
  107. package/dist/src/agent/memory/dreaming/short-term-store.js +1 -1
  108. package/dist/src/agent/memory/dreaming/utils.js +1 -1
  109. package/dist/src/agent/memory/plugin-discovery.js +1 -1
  110. package/dist/src/agent/models/manager.js +1 -1
  111. package/dist/src/agent/prompt/memory/index.d.ts +1 -0
  112. package/dist/src/agent/prompt/memory/index.js +34 -80
  113. package/dist/src/agent/prompt/memory/index.js.map +1 -1
  114. package/dist/src/agent/prompt/service-prompt-builder.js +2 -2
  115. package/dist/src/agent/prompt/system-prompt.js +0 -1
  116. package/dist/src/agent/prompt/system-prompt.js.map +1 -1
  117. package/dist/src/agent/reply/post-compaction-context.js +1 -1
  118. package/dist/src/agent/reply/workspace-boundary-read.js +1 -1
  119. package/dist/src/agent/sandbox/path-policy.js +2 -2
  120. package/dist/src/agent/service/build-direct-message-content.js +1 -1
  121. package/dist/src/agent/service/process-direct-one-shot.js +8 -17
  122. package/dist/src/agent/service/process-direct-one-shot.js.map +1 -1
  123. package/dist/src/agent/service/process-direct-streaming.js +14 -23
  124. package/dist/src/agent/service/process-direct-streaming.js.map +1 -1
  125. package/dist/src/agent/service.js +7 -11
  126. package/dist/src/agent/service.js.map +1 -1
  127. package/dist/src/agent/session/session-inspector.js +1 -1
  128. package/dist/src/agent/skills/config.js +1 -1
  129. package/dist/src/agent/skills/hub-hash.js +2 -2
  130. package/dist/src/agent/skills/hub-lock.js +1 -1
  131. package/dist/src/agent/skills/hub-pull.js +3 -3
  132. package/dist/src/agent/skills/index.js +1 -1
  133. package/dist/src/agent/skills/managed-store.js +1 -1
  134. package/dist/src/agent/skills/scanner.js +1 -1
  135. package/dist/src/agent/skills/skill-manage-ops.js +1 -1
  136. package/dist/src/agent/skills/skill-manager.js +1 -1
  137. package/dist/src/agent/tools/dreaming-tool.js +1 -1
  138. package/dist/src/agent/tools/factory.js +1 -1
  139. package/dist/src/agent/tools/image-generate-tool.js +1 -1
  140. package/dist/src/agent/tools/index.d.ts +0 -1
  141. package/dist/src/agent/tools/index.js +1 -2
  142. package/dist/src/agent/tools/send-media.js +1 -1
  143. package/dist/src/agent/tools/session-search-tool.d.ts +0 -1
  144. package/dist/src/agent/tools/session-search-tool.js +11 -6
  145. package/dist/src/agent/tools/session-search-tool.js.map +1 -1
  146. package/dist/src/agent/tools/skill-manage-tool.js +1 -1
  147. package/dist/src/agent/tools/tool-paths.js +1 -3
  148. package/dist/src/agent/tools/tool-paths.js.map +1 -1
  149. package/dist/src/agent/tools/workflow-tool.js +1 -1
  150. package/dist/src/agent/tools/write.js +1 -1
  151. package/dist/src/agent/workflow/catalog.js +1 -1
  152. package/dist/src/auth/credentials.js +3 -3
  153. package/dist/src/auth/profiles/store.js +1 -1
  154. package/dist/src/auth/sync-provider-auth.js +1 -1
  155. package/dist/src/browser/cache-dir-policy.js +1 -1
  156. package/dist/src/browser/cdp-local-launcher.js +2 -2
  157. package/dist/src/browser/providers/browser-ext-install.js +4 -4
  158. package/dist/src/browser/providers/cloakbrowser.js +4 -4
  159. package/dist/src/browser/providers/playwright-doctor.js +1 -1
  160. package/dist/src/browser/stealth.js +1 -1
  161. package/dist/src/channels/attachments/inbound-persist.js +1 -1
  162. package/dist/src/channels/attachments/outbound-tts-persist.js +1 -1
  163. package/dist/src/channels/outbound/persist-store.js +1 -1
  164. package/dist/src/channels/pairing/allow-from-file.js +1 -1
  165. package/dist/src/channels/pairing/pairing-store.js +2 -2
  166. package/dist/src/chat-commands/agent-edit.js +3 -4
  167. package/dist/src/chat-commands/agent-edit.js.map +1 -1
  168. package/dist/src/chat-commands/builtins/config.js +2 -2
  169. package/dist/src/chat-commands/context.js +1 -1
  170. package/dist/src/cli/commands/config.js +1 -1
  171. package/dist/src/cli/commands/doctor/checks/config-health.js +1 -1
  172. package/dist/src/cli/commands/doctor/checks/provider-auth.js +1 -1
  173. package/dist/src/cli/commands/doctor/checks/session-integrity.js +32 -95
  174. package/dist/src/cli/commands/doctor/checks/session-integrity.js.map +1 -1
  175. package/dist/src/cli/commands/doctor/checks/state-integrity.js +1 -1
  176. package/dist/src/cli/commands/doctor/checks/workspace-status.js +1 -1
  177. package/dist/src/cli/commands/extension-dev.js +1 -1
  178. package/dist/src/cli/commands/extension-marketplace.js +1 -1
  179. package/dist/src/cli/commands/extension-pack.js +1 -1
  180. package/dist/src/cli/commands/gateway/logs.js +1 -1
  181. package/dist/src/cli/commands/image.js +1 -1
  182. package/dist/src/cli/commands/init.js +5 -7
  183. package/dist/src/cli/commands/init.js.map +1 -1
  184. package/dist/src/cli/commands/onboard.js +0 -8
  185. package/dist/src/cli/commands/onboard.js.map +1 -1
  186. package/dist/src/cli/templates.d.ts +3 -10
  187. package/dist/src/cli/templates.js +4 -32
  188. package/dist/src/cli/templates.js.map +1 -1
  189. package/dist/src/cli/utils/init-workspace-core.js +2 -2
  190. package/dist/src/commands/agents.config.js +1 -1
  191. package/dist/src/config/agent-profile.js +1 -1
  192. package/dist/src/config/gateway-bind.js +1 -1
  193. package/dist/src/config/index.js +7 -8
  194. package/dist/src/config/index.js.map +1 -1
  195. package/dist/src/config/loader.js +2 -2
  196. package/dist/src/config/models-json.js +2 -2
  197. package/dist/src/config/paths-state.d.ts +3 -0
  198. package/dist/src/config/paths-state.js +7 -3
  199. package/dist/src/config/paths-state.js.map +1 -1
  200. package/dist/src/config/paths.d.ts +5 -36
  201. package/dist/src/config/paths.js +7 -52
  202. package/dist/src/config/paths.js.map +1 -1
  203. package/dist/src/config/profile.js +2 -2
  204. package/dist/src/config/schema.d.ts +15 -0
  205. package/dist/src/config/schema.js +11 -0
  206. package/dist/src/config/schema.js.map +1 -1
  207. package/dist/src/config/workspace-path.js +1 -1
  208. package/dist/src/cron/execution-types.d.ts +42 -0
  209. package/dist/src/cron/executor.js +2 -2
  210. package/dist/src/cron/persistence.js +1 -1
  211. package/dist/src/cron/run-log-store.d.ts +4 -8
  212. package/dist/src/cron/run-log-store.js +26 -78
  213. package/dist/src/cron/run-log-store.js.map +1 -1
  214. package/dist/src/cron/service.d.ts +3 -3
  215. package/dist/src/cron/service.js +2 -2
  216. package/dist/src/cron/service.js.map +1 -1
  217. package/dist/src/cron/types.d.ts +1 -42
  218. package/dist/src/daemon/constants.js +1 -1
  219. package/dist/src/daemon/install-plan.js +2 -2
  220. package/dist/src/daemon/launchd.js +2 -2
  221. package/dist/src/daemon/schtasks.js +2 -2
  222. package/dist/src/daemon/systemd.js +2 -2
  223. package/dist/src/extensions/bundle-mcp.js +1 -1
  224. package/dist/src/extensions/discover-extensions.js +1 -1
  225. package/dist/src/extensions/health.js +1 -1
  226. package/dist/src/extensions/loader.js +1 -1
  227. package/dist/src/extensions/lockfile.js +2 -2
  228. package/dist/src/extensions/update.js +1 -1
  229. package/dist/src/gateway/agents-admin.js +4 -4
  230. package/dist/src/gateway/agents-admin.js.map +1 -1
  231. package/dist/src/gateway/file-path-classifier.d.ts +0 -1
  232. package/dist/src/gateway/file-path-classifier.js +2 -8
  233. package/dist/src/gateway/file-path-classifier.js.map +1 -1
  234. package/dist/src/gateway/heartbeat/service.js +1 -1
  235. package/dist/src/gateway/hono/lib/config-payload.js +1 -1
  236. package/dist/src/gateway/hono/lib/extension-store.js +2 -2
  237. package/dist/src/gateway/hono/lib/static-ui.js +2 -2
  238. package/dist/src/gateway/hono/oauth.js +1 -1
  239. package/dist/src/gateway/hono/routes/agents.js +1 -1
  240. package/dist/src/gateway/hono/routes/auth-registry-extensions.js +1 -1
  241. package/dist/src/gateway/hono/routes/config-patch/misc.js +1 -1
  242. package/dist/src/gateway/hono/routes/dreaming.js +1 -1
  243. package/dist/src/gateway/hono/routes/host-fs.js +2 -2
  244. package/dist/src/gateway/hono/routes/models.js +1 -1
  245. package/dist/src/gateway/hono/routes/shares.js +1 -1
  246. package/dist/src/gateway/hono/routes/workspace.js +2 -2
  247. package/dist/src/gateway/lock.js +3 -3
  248. package/dist/src/gateway/ports.js +1 -1
  249. package/dist/src/gateway/service/agent-runner.js +2 -2
  250. package/dist/src/gateway/service/marketplace-service.js +2 -2
  251. package/dist/src/gateway/service.js +5 -1
  252. package/dist/src/gateway/service.js.map +1 -1
  253. package/dist/src/gateway/session-reset-service.d.ts +1 -1
  254. package/dist/src/gateway/session-reset-service.js +1 -1
  255. package/dist/src/gateway/session-reset-service.js.map +1 -1
  256. package/dist/src/gateway/workspace-fs-file-list.js +1 -1
  257. package/dist/src/heartbeat/index.js +1 -1
  258. package/dist/src/infra/brew.js +1 -1
  259. package/dist/src/infra/node-sqlite.d.ts +1 -0
  260. package/dist/src/infra/node-sqlite.js +17 -0
  261. package/dist/src/infra/node-sqlite.js.map +1 -0
  262. package/dist/src/infra/package-json.js +1 -1
  263. package/dist/src/infra/package-update-steps.js +1 -1
  264. package/dist/src/infra/path-env.js +2 -2
  265. package/dist/src/infra/restart.js +2 -2
  266. package/dist/src/infra/sqlite-errors.d.ts +1 -0
  267. package/dist/src/infra/sqlite-errors.js +77 -0
  268. package/dist/src/infra/sqlite-errors.js.map +1 -0
  269. package/dist/src/infra/stable-node-path.js +1 -1
  270. package/dist/src/infra/unhandled-rejections.d.ts +1 -0
  271. package/dist/src/infra/unhandled-rejections.js +25 -0
  272. package/dist/src/infra/unhandled-rejections.js.map +1 -0
  273. package/dist/src/infra/update-check.js +1 -1
  274. package/dist/src/infra/update-global.js +1 -1
  275. package/dist/src/infra/update-lock.js +3 -3
  276. package/dist/src/infra/update-runner.js +1 -1
  277. package/dist/src/infra/update-startup.js +2 -2
  278. package/dist/src/infra/warning-filter.d.ts +7 -0
  279. package/dist/src/infra/warning-filter.js +59 -0
  280. package/dist/src/infra/warning-filter.js.map +1 -0
  281. package/dist/src/infra/write-file-atomic.js +2 -2
  282. package/dist/src/notes/store.d.ts +3 -9
  283. package/dist/src/notes/store.js +22 -196
  284. package/dist/src/notes/store.js.map +1 -1
  285. package/dist/src/providers/auth-runtime/auth-profile-store.js +1 -1
  286. package/dist/src/providers/index.js +2 -2
  287. package/dist/src/providers/model-registry.js +1 -1
  288. package/dist/src/session/config-store.d.ts +6 -75
  289. package/dist/src/session/config-store.js +38 -144
  290. package/dist/src/session/config-store.js.map +1 -1
  291. package/dist/src/session/config-types.d.ts +15 -0
  292. package/dist/src/session/config-types.js +1 -0
  293. package/dist/src/session/index.d.ts +1 -3
  294. package/dist/src/session/index.js +3 -5
  295. package/dist/src/session/init-session-turn.d.ts +0 -6
  296. package/dist/src/session/init-session-turn.js +18 -18
  297. package/dist/src/session/init-session-turn.js.map +1 -1
  298. package/dist/src/session/lifecycle-timestamps.d.ts +5 -2
  299. package/dist/src/session/lifecycle-timestamps.js.map +1 -1
  300. package/dist/src/session/{parity/load-jsonl-entries.js → load-jsonl-entries.js} +1 -1
  301. package/dist/src/session/load-jsonl-entries.js.map +1 -0
  302. package/dist/src/session/manager.d.ts +5 -3
  303. package/dist/src/session/manager.js +1 -5
  304. package/dist/src/session/manager.js.map +1 -1
  305. package/dist/src/session/resolve-session.d.ts +3 -6
  306. package/dist/src/session/resolve-session.js +26 -31
  307. package/dist/src/session/resolve-session.js.map +1 -1
  308. package/dist/src/session/session-context-for-llm.js +5 -1
  309. package/dist/src/session/session-context-for-llm.js.map +1 -1
  310. package/dist/src/session/session-id.js +12 -0
  311. package/dist/src/session/session-id.js.map +1 -0
  312. package/dist/src/session/session-title.js +2 -2
  313. package/dist/src/session/session-workspace.d.ts +1 -1
  314. package/dist/src/session/session-workspace.js.map +1 -1
  315. package/dist/src/session/store.d.ts +14 -63
  316. package/dist/src/session/store.js +172 -847
  317. package/dist/src/session/store.js.map +1 -1
  318. package/dist/src/session/stored-rows-to-file-entries.d.ts +11 -0
  319. package/dist/src/session/stored-rows-to-file-entries.js +95 -0
  320. package/dist/src/session/stored-rows-to-file-entries.js.map +1 -0
  321. package/dist/src/session/transcript-events.d.ts +1 -2
  322. package/dist/src/session/transcript-events.js +5 -12
  323. package/dist/src/session/transcript-events.js.map +1 -1
  324. package/dist/src/session/transcript-format.d.ts +1 -1
  325. package/dist/src/session/transcript-format.js.map +1 -1
  326. package/dist/src/session/transcript-stats.d.ts +1 -0
  327. package/dist/src/session/transcript-stats.js +10 -0
  328. package/dist/src/session/transcript-stats.js.map +1 -0
  329. package/dist/src/share/share-auto.js +2 -2
  330. package/dist/src/share/share-store.js +3 -3
  331. package/dist/src/share/share-thumbnail.js +2 -2
  332. package/dist/src/share/share-zip.js +1 -1
  333. package/dist/src/share/site-share-store.js +3 -3
  334. package/dist/src/share/site-static-serve.js +1 -1
  335. package/dist/src/storage/sqlite/config-repository.d.ts +6 -0
  336. package/dist/src/storage/sqlite/config-repository.js +56 -0
  337. package/dist/src/storage/sqlite/config-repository.js.map +1 -0
  338. package/dist/src/storage/sqlite/connection.d.ts +38 -0
  339. package/dist/src/storage/sqlite/connection.js +258 -0
  340. package/dist/src/storage/sqlite/connection.js.map +1 -0
  341. package/dist/src/storage/sqlite/cron-run-repository.d.ts +5 -0
  342. package/dist/src/storage/sqlite/cron-run-repository.js +97 -0
  343. package/dist/src/storage/sqlite/cron-run-repository.js.map +1 -0
  344. package/dist/src/storage/sqlite/fts.d.ts +2 -0
  345. package/dist/src/storage/sqlite/fts.js +11 -0
  346. package/dist/src/storage/sqlite/fts.js.map +1 -0
  347. package/dist/src/storage/sqlite/index.d.ts +12 -0
  348. package/dist/src/storage/sqlite/index.js +13 -0
  349. package/dist/src/storage/sqlite/memory-index-repository.d.ts +18 -0
  350. package/dist/src/storage/sqlite/memory-index-repository.js +132 -0
  351. package/dist/src/storage/sqlite/memory-index-repository.js.map +1 -0
  352. package/dist/src/storage/sqlite/notes-repository.d.ts +11 -0
  353. package/dist/src/storage/sqlite/notes-repository.js +191 -0
  354. package/dist/src/storage/sqlite/notes-repository.js.map +1 -0
  355. package/dist/src/storage/sqlite/paths.d.ts +1 -0
  356. package/dist/src/storage/sqlite/paths.js +7 -0
  357. package/dist/src/storage/sqlite/paths.js.map +1 -0
  358. package/dist/src/storage/sqlite/row-mappers.d.ts +82 -0
  359. package/dist/src/storage/sqlite/row-mappers.js +164 -0
  360. package/dist/src/storage/sqlite/row-mappers.js.map +1 -0
  361. package/dist/src/storage/sqlite/schema.d.ts +5 -0
  362. package/dist/src/storage/sqlite/schema.js +43 -0
  363. package/dist/src/storage/sqlite/schema.js.map +1 -0
  364. package/dist/src/storage/sqlite/schema.sql +195 -0
  365. package/dist/src/storage/sqlite/session-metadata.d.ts +8 -0
  366. package/dist/src/storage/sqlite/session-metadata.js +83 -0
  367. package/dist/src/storage/sqlite/session-metadata.js.map +1 -0
  368. package/dist/src/storage/sqlite/session-repository.d.ts +29 -0
  369. package/dist/src/storage/sqlite/session-repository.js +268 -0
  370. package/dist/src/storage/sqlite/session-repository.js.map +1 -0
  371. package/dist/src/storage/sqlite/transaction.d.ts +11 -0
  372. package/dist/src/storage/sqlite/transaction.js +115 -0
  373. package/dist/src/storage/sqlite/transaction.js.map +1 -0
  374. package/dist/src/storage/sqlite/transcript-repository.d.ts +34 -0
  375. package/dist/src/storage/sqlite/transcript-repository.js +241 -0
  376. package/dist/src/storage/sqlite/transcript-repository.js.map +1 -0
  377. package/dist/src/tui/clipboard-image.js +3 -3
  378. package/dist/src/tui/theme-manager.js +1 -1
  379. package/dist/src/tui/tui-keybindings-file.js +1 -1
  380. package/dist/src/tui/tui-scoped-models.js +2 -2
  381. package/dist/src/tui/tui-settings.js +1 -1
  382. package/dist/src/tui/tui.js +3 -3
  383. package/dist/src/tunnel/frpc-binary.js +3 -3
  384. package/dist/src/tunnel/frpc-config.js +1 -1
  385. package/dist/src/tunnel/frpc-extract.js +1 -1
  386. package/dist/src/tunnel/tunnel-state.js +1 -1
  387. package/dist/src/utils/logger/audit.js +1 -1
  388. package/dist/src/utils/logger/log-store.js +1 -1
  389. package/dist/src/utils/logger/rotation.js +1 -1
  390. package/dist/src/voice/tts/audio.js +1 -1
  391. package/dist/src/voice/tts/providers/edge-speech.js +2 -2
  392. package/dist/src/workflows/service/workflow-session-bridge.js +41 -64
  393. package/dist/src/workflows/service/workflow-session-bridge.js.map +1 -1
  394. package/dist/src/workflows/store/event-store.js +1 -1
  395. package/dist/src/workflows/store/run-store.js +1 -1
  396. package/package.json +2 -2
  397. package/dist/gateway/static/root/assets/index-Bj_l8QDp.css +0 -1
  398. package/dist/gateway/static/root/assets/url-BHHmdJYc.js +0 -3
  399. package/dist/src/agent/context/workspace-templates/BOOTSTRAP.md +0 -61
  400. package/dist/src/agent/embedded/session-manager-cache.d.ts +0 -19
  401. package/dist/src/agent/embedded/session-manager-cache.js +0 -48
  402. package/dist/src/agent/embedded/session-manager-cache.js.map +0 -1
  403. package/dist/src/session/parity/artifacts.d.ts +0 -16
  404. package/dist/src/session/parity/artifacts.js +0 -80
  405. package/dist/src/session/parity/artifacts.js.map +0 -1
  406. package/dist/src/session/parity/jsonl-transcript-io.d.ts +0 -54
  407. package/dist/src/session/parity/jsonl-transcript-io.js +0 -236
  408. package/dist/src/session/parity/jsonl-transcript-io.js.map +0 -1
  409. package/dist/src/session/parity/load-jsonl-entries.js.map +0 -1
  410. package/dist/src/session/parity/session-id.js +0 -18
  411. package/dist/src/session/parity/session-id.js.map +0 -1
  412. package/dist/src/session/parity/sessions-json-cache.d.ts +0 -14
  413. package/dist/src/session/parity/sessions-json-cache.js +0 -98
  414. package/dist/src/session/parity/sessions-json-cache.js.map +0 -1
  415. package/dist/src/session/parity/sessions-json-file-read.d.ts +0 -6
  416. package/dist/src/session/parity/sessions-json-file-read.js +0 -19
  417. package/dist/src/session/parity/sessions-json-file-read.js.map +0 -1
  418. package/dist/src/session/parity/sessions-json-file.d.ts +0 -11
  419. package/dist/src/session/parity/sessions-json-file.js +0 -52
  420. package/dist/src/session/parity/sessions-json-file.js.map +0 -1
  421. package/dist/src/session/parity/sessions-json-patch.d.ts +0 -14
  422. package/dist/src/session/parity/sessions-json-patch.js +0 -40
  423. package/dist/src/session/parity/sessions-json-patch.js.map +0 -1
  424. package/dist/src/session/parity/transcript-file-lock.d.ts +0 -22
  425. package/dist/src/session/parity/transcript-file-lock.js +0 -142
  426. package/dist/src/session/parity/transcript-file-lock.js.map +0 -1
  427. package/dist/src/session/parity/transcript-pagination.d.ts +0 -29
  428. package/dist/src/session/parity/transcript-pagination.js +0 -132
  429. package/dist/src/session/parity/transcript-pagination.js.map +0 -1
  430. package/dist/src/session/parity/transcript-paths.d.ts +0 -13
  431. package/dist/src/session/parity/transcript-paths.js +0 -64
  432. package/dist/src/session/parity/transcript-paths.js.map +0 -1
  433. package/dist/src/session/parity/xopc-session-disk-entry.d.ts +0 -22
  434. package/dist/src/session/search-index-cache.d.ts +0 -6
  435. package/dist/src/session/search-index-cache.js +0 -44
  436. package/dist/src/session/search-index-cache.js.map +0 -1
  437. package/dist/src/session/search-index.d.ts +0 -20
  438. package/dist/src/session/search-index.js +0 -124
  439. package/dist/src/session/search-index.js.map +0 -1
  440. /package/dist/src/{session/parity/xopc-session-disk-entry.js → cron/execution-types.js} +0 -0
  441. /package/dist/src/session/{parity/load-jsonl-entries.d.ts → load-jsonl-entries.d.ts} +0 -0
  442. /package/dist/src/session/{parity/session-id.d.ts → session-id.d.ts} +0 -0
@@ -0,0 +1,268 @@
1
+ import { getSqliteDatabase, runSqliteWriteTransaction } from "./transaction.js";
2
+ import { buildDefaultSessionMetadata, resolveAgentIdFromSessionKey } from "./session-metadata.js";
3
+ import { buildGlobalSessionStats, metadataToSessionInsert, sessionRowToMetadata } from "./row-mappers.js";
4
+ import { validateSessionId } from "../../session/session-id.js";
5
+ import { escapeFts5Query } from "./fts.js";
6
+ import { randomUUID } from "node:crypto";
7
+ //#region src/storage/sqlite/session-repository.ts
8
+ const SESSION_COLUMNS = `
9
+ session_key, agent_id, current_transcript_id, status, name, tags_json,
10
+ created_at, updated_at, last_accessed_at, session_started_at, last_interaction_at,
11
+ source_channel, source_chat_id, session_type, routing_json, custom_data_json,
12
+ abort_cutoff_timestamp, message_count, estimated_tokens, compacted_count,
13
+ thinking_level, verbose_level
14
+ `;
15
+ const SELECT_SESSION = `SELECT ${SESSION_COLUMNS} FROM sessions WHERE session_key = ?`;
16
+ function readSessionRow(db, sessionKey) {
17
+ return db.prepare(SELECT_SESSION).get(sessionKey);
18
+ }
19
+ function insertSessionAndTranscript(db, sessionKey, transcriptId, cwd, metadata) {
20
+ const row = metadataToSessionInsert(sessionKey, transcriptId, metadata);
21
+ db.prepare(`INSERT INTO sessions (
22
+ session_key, agent_id, current_transcript_id, status, name, tags_json,
23
+ created_at, updated_at, last_accessed_at, session_started_at, last_interaction_at,
24
+ source_channel, source_chat_id, session_type, routing_json, custom_data_json,
25
+ abort_cutoff_timestamp, message_count, estimated_tokens, compacted_count,
26
+ thinking_level, verbose_level
27
+ ) VALUES (
28
+ ?, ?, ?, ?, ?, ?,
29
+ ?, ?, ?, ?, ?,
30
+ ?, ?, ?, ?, ?,
31
+ ?, ?, ?, ?,
32
+ ?, ?
33
+ )`).run(row.sessionKey, row.agentId, row.transcriptId, row.status, row.name, row.tagsJson, row.createdAt, row.updatedAt, row.lastAccessedAt, row.sessionStartedAt, row.lastInteractionAt, row.sourceChannel, row.sourceChatId, row.sessionType, row.routingJson, row.customDataJson, row.abortCutoffTimestamp, row.messageCount, row.estimatedTokens, row.compactedCount, row.thinkingLevel, row.verboseLevel);
34
+ const now = Date.now();
35
+ db.prepare(`INSERT INTO transcripts (transcript_id, session_key, status, created_at, cwd)
36
+ VALUES (?, ?, 'active', ?, ?)`).run(transcriptId, sessionKey, now, cwd);
37
+ }
38
+ function ensureSessionInTransaction(db, sessionKey, cwd) {
39
+ const existing = readSessionRow(db, sessionKey);
40
+ if (existing) return sessionRowToMetadata(sessionKey, existing);
41
+ const transcriptId = validateSessionId(randomUUID());
42
+ const metadata = buildDefaultSessionMetadata(sessionKey);
43
+ metadata.transcriptId = transcriptId;
44
+ insertSessionAndTranscript(db, sessionKey, transcriptId, cwd, metadata);
45
+ const row = readSessionRow(db, sessionKey);
46
+ if (!row) throw new Error(`Failed to create session: ${sessionKey}`);
47
+ return sessionRowToMetadata(sessionKey, row);
48
+ }
49
+ function ensureSessionRecord(sessionKey, cwd) {
50
+ return runSqliteWriteTransaction((db) => ensureSessionInTransaction(db, sessionKey, cwd));
51
+ }
52
+ function readCurrentTranscriptId(db, sessionKey) {
53
+ return db.prepare(`SELECT current_transcript_id FROM sessions WHERE session_key = ?`).get(sessionKey)?.current_transcript_id ?? null;
54
+ }
55
+ function getSessionMetadata(sessionKey) {
56
+ const row = readSessionRow(getSqliteDatabase(), sessionKey);
57
+ if (!row) return null;
58
+ return sessionRowToMetadata(sessionKey, row);
59
+ }
60
+ function getCurrentTranscriptId(sessionKey) {
61
+ return readCurrentTranscriptId(getSqliteDatabase(), sessionKey);
62
+ }
63
+ function listSessionMetadata(query = {}) {
64
+ const db = getSqliteDatabase();
65
+ const conditions = [];
66
+ const params = [];
67
+ if (query.status) {
68
+ const statuses = Array.isArray(query.status) ? query.status : [query.status];
69
+ conditions.push(`status IN (${statuses.map(() => "?").join(", ")})`);
70
+ params.push(...statuses);
71
+ }
72
+ if (query.channel) {
73
+ const rawChannels = query.channel.split(",").map((c) => c.trim().toLowerCase()).filter(Boolean);
74
+ const channels = [...new Set(rawChannels.flatMap((c) => {
75
+ if (c === "webchat") return ["webchat", "ui"];
76
+ if (c === "gateway") return ["gateway", "webui"];
77
+ return [c];
78
+ }))];
79
+ if (channels.length === 0) return {
80
+ items: [],
81
+ total: 0,
82
+ limit: query.limit ?? 50,
83
+ offset: query.offset ?? 0,
84
+ hasMore: false
85
+ };
86
+ const directMatch = `LOWER(source_channel) IN (${channels.map(() => "?").join(", ")})`;
87
+ const keyPatterns = /* @__PURE__ */ new Set();
88
+ for (const c of channels) if (c === "webchat" || c === "ui") keyPatterns.add(`session_key LIKE 'agent:%:webchat:%'`);
89
+ else if (c === "gateway" || c === "webui") keyPatterns.add(`session_key LIKE 'agent:%:gateway:%'`);
90
+ else keyPatterns.add(`session_key LIKE 'agent:%:${c.replace(/'/g, "''")}:%'`);
91
+ const keyFallback = keyPatterns.size > 0 ? `(TRIM(COALESCE(source_channel, '')) = '' AND (${[...keyPatterns].join(" OR ")}))` : "";
92
+ conditions.push(keyFallback ? `(${directMatch} OR ${keyFallback})` : directMatch);
93
+ params.push(...channels);
94
+ }
95
+ if (query.tags?.length) for (const tag of query.tags) {
96
+ conditions.push(`tags_json LIKE ?`);
97
+ params.push(`%"${tag}"%`);
98
+ }
99
+ let searchKeys = null;
100
+ if (query.search?.trim()) {
101
+ const rawSearch = query.search.trim();
102
+ const ftsQuery = escapeFts5Query(rawSearch);
103
+ const ftsRows = db.prepare(`SELECT DISTINCT session_key FROM transcript_fts WHERE transcript_fts MATCH ? LIMIT 500`).all(ftsQuery);
104
+ const ftsKeys = new Set(ftsRows.map((r) => r.session_key));
105
+ const like = `%${rawSearch.toLowerCase()}%`;
106
+ const metaRows = db.prepare(`SELECT session_key FROM sessions
107
+ WHERE LOWER(session_key) LIKE ?
108
+ OR LOWER(COALESCE(name, '')) LIKE ?
109
+ OR LOWER(source_channel) LIKE ?
110
+ OR LOWER(source_chat_id) LIKE ?
111
+ OR LOWER(tags_json) LIKE ?`).all(like, like, like, like, like);
112
+ searchKeys = [...new Set([...ftsKeys, ...metaRows.map((r) => r.session_key)])];
113
+ if (searchKeys.length === 0) return {
114
+ items: [],
115
+ total: 0,
116
+ limit: query.limit ?? 50,
117
+ offset: query.offset ?? 0,
118
+ hasMore: false
119
+ };
120
+ conditions.push(`session_key IN (${searchKeys.map(() => "?").join(", ")})`);
121
+ params.push(...searchKeys);
122
+ }
123
+ const where = conditions.length > 0 ? `WHERE ${conditions.join(" AND ")}` : "";
124
+ const sortColumn = sessionSortColumn(query.sortBy);
125
+ const sortOrder = query.sortOrder === "asc" ? "ASC" : "DESC";
126
+ const total = db.prepare(`SELECT COUNT(*) AS total FROM sessions ${where}`).get(...params).total;
127
+ const limit = query.limit ?? 50;
128
+ const offset = query.offset ?? 0;
129
+ return {
130
+ items: db.prepare(`SELECT ${SESSION_COLUMNS} FROM sessions ${where}
131
+ ORDER BY ${sortColumn} ${sortOrder}
132
+ LIMIT ? OFFSET ?`).all(...params, limit, offset).map((row) => sessionRowToMetadata(row.session_key, row)),
133
+ total,
134
+ limit,
135
+ offset,
136
+ hasMore: offset + limit < total
137
+ };
138
+ }
139
+ function sessionSortColumn(sortBy) {
140
+ switch (sortBy) {
141
+ case "createdAt": return "created_at";
142
+ case "messageCount": return "message_count";
143
+ case "lastAccessedAt": return "last_accessed_at";
144
+ default: return "updated_at";
145
+ }
146
+ }
147
+ function patchSessionMetadata(sessionKey, updates) {
148
+ return runSqliteWriteTransaction((db) => {
149
+ const existing = readSessionRow(db, sessionKey);
150
+ if (!existing) throw new Error(`Session not found: ${sessionKey}`);
151
+ const merged = {
152
+ ...sessionRowToMetadata(sessionKey, existing),
153
+ ...updates,
154
+ key: sessionKey
155
+ };
156
+ const now = Date.now();
157
+ db.prepare(`UPDATE sessions SET
158
+ status = ?,
159
+ name = ?,
160
+ tags_json = ?,
161
+ updated_at = ?,
162
+ last_accessed_at = ?,
163
+ session_started_at = ?,
164
+ last_interaction_at = ?,
165
+ source_channel = ?,
166
+ source_chat_id = ?,
167
+ session_type = ?,
168
+ routing_json = ?,
169
+ custom_data_json = ?,
170
+ abort_cutoff_timestamp = ?,
171
+ message_count = ?,
172
+ estimated_tokens = ?,
173
+ compacted_count = ?,
174
+ thinking_level = ?,
175
+ verbose_level = ?
176
+ WHERE session_key = ?`).run(merged.status, merged.name ?? null, JSON.stringify(merged.tags ?? []), Date.parse(merged.updatedAt) || now, Date.parse(merged.lastAccessedAt) || now, merged.sessionStartedAt ? Date.parse(merged.sessionStartedAt) : existing.session_started_at, merged.lastInteractionAt ? Date.parse(merged.lastInteractionAt) : existing.last_interaction_at, merged.sourceChannel, merged.sourceChatId, merged.sessionType ?? null, merged.routing ? JSON.stringify(merged.routing) : null, merged.customData ? JSON.stringify(merged.customData) : null, merged.abortCutoffTimestamp ?? null, merged.messageCount, merged.estimatedTokens, merged.compactedCount, existing.thinking_level, existing.verbose_level, sessionKey);
177
+ const row = readSessionRow(db, sessionKey);
178
+ if (!row) throw new Error(`Session not found after patch: ${sessionKey}`);
179
+ return sessionRowToMetadata(sessionKey, row);
180
+ });
181
+ }
182
+ function updateSessionStats(sessionKey, stats) {
183
+ runSqliteWriteTransaction((db) => {
184
+ const now = stats.lastInteractionAt ?? Date.now();
185
+ db.prepare(`UPDATE sessions SET
186
+ message_count = ?,
187
+ estimated_tokens = ?,
188
+ updated_at = ?,
189
+ last_accessed_at = ?,
190
+ last_interaction_at = ?
191
+ WHERE session_key = ?`).run(stats.messageCount, stats.estimatedTokens, now, now, now, sessionKey);
192
+ });
193
+ }
194
+ function incrementSessionStatsOnAppend(sessionKey, tokenDelta = 0) {
195
+ runSqliteWriteTransaction((db) => {
196
+ const now = Date.now();
197
+ db.prepare(`UPDATE sessions SET
198
+ message_count = message_count + 1,
199
+ estimated_tokens = estimated_tokens + ?,
200
+ updated_at = ?,
201
+ last_accessed_at = ?,
202
+ last_interaction_at = ?
203
+ WHERE session_key = ?`).run(tokenDelta, now, now, now, sessionKey);
204
+ });
205
+ }
206
+ function resetSessionRecord(sessionKey, cwd) {
207
+ return runSqliteWriteTransaction((db) => {
208
+ const existing = readSessionRow(db, sessionKey);
209
+ if (!existing) return null;
210
+ const previousSessionId = existing.current_transcript_id;
211
+ const now = Date.now();
212
+ db.prepare(`UPDATE transcripts SET status = 'archived', archive_reason = 'reset', archived_at = ?
213
+ WHERE transcript_id = ?`).run(now, previousSessionId);
214
+ const newTranscriptId = validateSessionId(randomUUID());
215
+ db.prepare(`INSERT INTO transcripts (transcript_id, session_key, status, created_at, cwd)
216
+ VALUES (?, ?, 'active', ?, ?)`).run(newTranscriptId, sessionKey, now, cwd);
217
+ db.prepare(`UPDATE sessions SET
218
+ current_transcript_id = ?,
219
+ updated_at = ?,
220
+ session_started_at = ?,
221
+ last_interaction_at = NULL,
222
+ message_count = 0,
223
+ estimated_tokens = 0
224
+ WHERE session_key = ?`).run(newTranscriptId, now, now, sessionKey);
225
+ return {
226
+ sessionId: newTranscriptId,
227
+ previousSessionId
228
+ };
229
+ });
230
+ }
231
+ function deleteSessionRecord(sessionKey) {
232
+ return runSqliteWriteTransaction((db) => {
233
+ const existing = readSessionRow(db, sessionKey);
234
+ if (!existing) return false;
235
+ const now = Date.now();
236
+ db.prepare(`UPDATE transcripts SET status = 'archived', archive_reason = 'delete', archived_at = ?
237
+ WHERE transcript_id = ?`).run(now, existing.current_transcript_id);
238
+ db.prepare(`DELETE FROM sessions WHERE session_key = ?`).run(sessionKey);
239
+ return true;
240
+ });
241
+ }
242
+ function listSessionsByAgent(agentId) {
243
+ return getSqliteDatabase().prepare(`SELECT ${SESSION_COLUMNS} FROM sessions WHERE agent_id = ? ORDER BY updated_at DESC`).all(agentId.toLowerCase()).map((row) => sessionRowToMetadata(row.session_key, row));
244
+ }
245
+ function getSessionPersistedLevels(sessionKey) {
246
+ const row = getSqliteDatabase().prepare(`SELECT thinking_level, verbose_level FROM sessions WHERE session_key = ?`).get(sessionKey);
247
+ if (!row) return null;
248
+ return {
249
+ thinkingLevel: row.thinking_level,
250
+ verboseLevel: row.verbose_level
251
+ };
252
+ }
253
+ function findSessionKeyByTranscriptId(transcriptId) {
254
+ return getSqliteDatabase().prepare(`SELECT session_key FROM sessions WHERE current_transcript_id = ?`).get(transcriptId)?.session_key ?? null;
255
+ }
256
+ function getGlobalSessionStats() {
257
+ return buildGlobalSessionStats(listSessionMetadata({
258
+ limit: 1e5,
259
+ offset: 0
260
+ }).items);
261
+ }
262
+ function resolveSessionAgentId(sessionKey) {
263
+ return resolveAgentIdFromSessionKey(sessionKey);
264
+ }
265
+ //#endregion
266
+ export { deleteSessionRecord, ensureSessionInTransaction, ensureSessionRecord, findSessionKeyByTranscriptId, getCurrentTranscriptId, getGlobalSessionStats, getSessionMetadata, getSessionPersistedLevels, incrementSessionStatsOnAppend, listSessionMetadata, listSessionsByAgent, patchSessionMetadata, readCurrentTranscriptId, resetSessionRecord, resolveSessionAgentId, updateSessionStats };
267
+
268
+ //# sourceMappingURL=session-repository.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-repository.js","names":[],"sources":["../../../../src/storage/sqlite/session-repository.ts"],"sourcesContent":["import { randomUUID } from 'node:crypto';\nimport type { DatabaseSync } from 'node:sqlite';\n\nimport { validateSessionId } from '../../session/session-id.js';\nimport type { SessionListQuery, SessionMetadata, PaginatedResult } from '../../session/types.js';\nimport {\n buildDefaultSessionMetadata,\n resolveAgentIdFromSessionKey,\n} from './session-metadata.js';\nimport {\n buildGlobalSessionStats,\n metadataToSessionInsert,\n sessionRowToMetadata,\n type SessionRow,\n} from './row-mappers.js';\nimport { getSqliteDatabase, runSqliteWriteTransaction } from './transaction.js';\n\nconst SESSION_COLUMNS = `\n session_key, agent_id, current_transcript_id, status, name, tags_json,\n created_at, updated_at, last_accessed_at, session_started_at, last_interaction_at,\n source_channel, source_chat_id, session_type, routing_json, custom_data_json,\n abort_cutoff_timestamp, message_count, estimated_tokens, compacted_count,\n thinking_level, verbose_level\n`;\n\nimport { escapeFts5Query } from './fts.js';\n\nconst SELECT_SESSION = `SELECT ${SESSION_COLUMNS} FROM sessions WHERE session_key = ?`;\n\nfunction readSessionRow(db: DatabaseSync, sessionKey: string): SessionRow | undefined {\n return db.prepare(SELECT_SESSION).get(sessionKey) as SessionRow | undefined;\n}\n\nfunction insertSessionAndTranscript(\n db: DatabaseSync,\n sessionKey: string,\n transcriptId: string,\n cwd: string,\n metadata: SessionMetadata,\n): void {\n const row = metadataToSessionInsert(sessionKey, transcriptId, metadata);\n db.prepare(\n `INSERT INTO sessions (\n session_key, agent_id, current_transcript_id, status, name, tags_json,\n created_at, updated_at, last_accessed_at, session_started_at, last_interaction_at,\n source_channel, source_chat_id, session_type, routing_json, custom_data_json,\n abort_cutoff_timestamp, message_count, estimated_tokens, compacted_count,\n thinking_level, verbose_level\n ) VALUES (\n ?, ?, ?, ?, ?, ?,\n ?, ?, ?, ?, ?,\n ?, ?, ?, ?, ?,\n ?, ?, ?, ?,\n ?, ?\n )`,\n ).run(\n row.sessionKey,\n row.agentId,\n row.transcriptId,\n row.status,\n row.name,\n row.tagsJson,\n row.createdAt,\n row.updatedAt,\n row.lastAccessedAt,\n row.sessionStartedAt,\n row.lastInteractionAt,\n row.sourceChannel,\n row.sourceChatId,\n row.sessionType,\n row.routingJson,\n row.customDataJson,\n row.abortCutoffTimestamp,\n row.messageCount,\n row.estimatedTokens,\n row.compactedCount,\n row.thinkingLevel,\n row.verboseLevel,\n );\n\n const now = Date.now();\n db.prepare(\n `INSERT INTO transcripts (transcript_id, session_key, status, created_at, cwd)\n VALUES (?, ?, 'active', ?, ?)`,\n ).run(transcriptId, sessionKey, now, cwd);\n}\n\nexport function ensureSessionInTransaction(\n db: DatabaseSync,\n sessionKey: string,\n cwd: string,\n): SessionMetadata {\n const existing = readSessionRow(db, sessionKey);\n if (existing) {\n return sessionRowToMetadata(sessionKey, existing);\n }\n\n const transcriptId = validateSessionId(randomUUID());\n const metadata = buildDefaultSessionMetadata(sessionKey);\n metadata.transcriptId = transcriptId;\n insertSessionAndTranscript(db, sessionKey, transcriptId, cwd, metadata);\n const row = readSessionRow(db, sessionKey);\n if (!row) {\n throw new Error(`Failed to create session: ${sessionKey}`);\n }\n return sessionRowToMetadata(sessionKey, row);\n}\n\nexport function ensureSessionRecord(sessionKey: string, cwd: string): SessionMetadata {\n return runSqliteWriteTransaction((db) => ensureSessionInTransaction(db, sessionKey, cwd));\n}\n\nexport function readCurrentTranscriptId(db: DatabaseSync, sessionKey: string): string | null {\n const row = db\n .prepare(`SELECT current_transcript_id FROM sessions WHERE session_key = ?`)\n .get(sessionKey) as { current_transcript_id?: string } | undefined;\n return row?.current_transcript_id ?? null;\n}\n\nexport function getSessionMetadata(sessionKey: string): SessionMetadata | null {\n const db = getSqliteDatabase();\n const row = readSessionRow(db, sessionKey);\n if (!row) {\n return null;\n }\n return sessionRowToMetadata(sessionKey, row);\n}\n\nexport function getCurrentTranscriptId(sessionKey: string): string | null {\n return readCurrentTranscriptId(getSqliteDatabase(), sessionKey);\n}\n\nexport function listSessionMetadata(query: SessionListQuery = {}): PaginatedResult<SessionMetadata> {\n const db = getSqliteDatabase();\n const conditions: string[] = [];\n const params: Array<string | number> = [];\n\n if (query.status) {\n const statuses = Array.isArray(query.status) ? query.status : [query.status];\n conditions.push(`status IN (${statuses.map(() => '?').join(', ')})`);\n params.push(...statuses);\n }\n\n if (query.channel) {\n const rawChannels = query.channel\n .split(',')\n .map((c) => c.trim().toLowerCase())\n .filter(Boolean);\n const channels = [\n ...new Set(\n rawChannels.flatMap((c) => {\n if (c === 'webchat') return ['webchat', 'ui'];\n if (c === 'gateway') return ['gateway', 'webui'];\n return [c];\n }),\n ),\n ];\n if (channels.length === 0) {\n return { items: [], total: 0, limit: query.limit ?? 50, offset: query.offset ?? 0, hasMore: false };\n }\n const directMatch = `LOWER(source_channel) IN (${channels.map(() => '?').join(', ')})`;\n const keyPatterns = new Set<string>();\n for (const c of channels) {\n if (c === 'webchat' || c === 'ui') {\n keyPatterns.add(`session_key LIKE 'agent:%:webchat:%'`);\n } else if (c === 'gateway' || c === 'webui') {\n keyPatterns.add(`session_key LIKE 'agent:%:gateway:%'`);\n } else {\n keyPatterns.add(`session_key LIKE 'agent:%:${c.replace(/'/g, \"''\")}:%'`);\n }\n }\n const keyFallback =\n keyPatterns.size > 0\n ? `(TRIM(COALESCE(source_channel, '')) = '' AND (${[...keyPatterns].join(' OR ')}))`\n : '';\n conditions.push(keyFallback ? `(${directMatch} OR ${keyFallback})` : directMatch);\n params.push(...channels);\n }\n\n if (query.tags?.length) {\n for (const tag of query.tags) {\n conditions.push(`tags_json LIKE ?`);\n params.push(`%\"${tag}\"%`);\n }\n }\n\n let searchKeys: string[] | null = null;\n if (query.search?.trim()) {\n const rawSearch = query.search.trim();\n const ftsQuery = escapeFts5Query(rawSearch);\n const ftsRows = db\n .prepare(\n `SELECT DISTINCT session_key FROM transcript_fts WHERE transcript_fts MATCH ? LIMIT 500`,\n )\n .all(ftsQuery) as Array<{ session_key: string }>;\n const ftsKeys = new Set(ftsRows.map((r) => r.session_key));\n const like = `%${rawSearch.toLowerCase()}%`;\n const metaRows = db\n .prepare(\n `SELECT session_key FROM sessions\n WHERE LOWER(session_key) LIKE ?\n OR LOWER(COALESCE(name, '')) LIKE ?\n OR LOWER(source_channel) LIKE ?\n OR LOWER(source_chat_id) LIKE ?\n OR LOWER(tags_json) LIKE ?`,\n )\n .all(like, like, like, like, like) as Array<{ session_key: string }>;\n searchKeys = [...new Set([...ftsKeys, ...metaRows.map((r) => r.session_key)])];\n if (searchKeys.length === 0) {\n return { items: [], total: 0, limit: query.limit ?? 50, offset: query.offset ?? 0, hasMore: false };\n }\n conditions.push(`session_key IN (${searchKeys.map(() => '?').join(', ')})`);\n params.push(...searchKeys);\n }\n\n const where = conditions.length > 0 ? `WHERE ${conditions.join(' AND ')}` : '';\n const sortColumn = sessionSortColumn(query.sortBy);\n const sortOrder = query.sortOrder === 'asc' ? 'ASC' : 'DESC';\n\n const countRow = db\n .prepare(`SELECT COUNT(*) AS total FROM sessions ${where}`)\n .get(...params) as { total: number };\n const total = countRow.total;\n\n const limit = query.limit ?? 50;\n const offset = query.offset ?? 0;\n const rows = db\n .prepare(\n `SELECT ${SESSION_COLUMNS} FROM sessions ${where}\n ORDER BY ${sortColumn} ${sortOrder}\n LIMIT ? OFFSET ?`,\n )\n .all(...params, limit, offset) as SessionRow[];\n\n const items = rows.map((row) => sessionRowToMetadata(row.session_key, row));\n return { items, total, limit, offset, hasMore: offset + limit < total };\n}\n\nfunction sessionSortColumn(sortBy: SessionListQuery['sortBy']): string {\n switch (sortBy) {\n case 'createdAt':\n return 'created_at';\n case 'messageCount':\n return 'message_count';\n case 'lastAccessedAt':\n return 'last_accessed_at';\n case 'updatedAt':\n default:\n return 'updated_at';\n }\n}\n\nexport function patchSessionMetadata(\n sessionKey: string,\n updates: Partial<SessionMetadata>,\n): SessionMetadata {\n return runSqliteWriteTransaction((db) => {\n const existing = readSessionRow(db, sessionKey);\n if (!existing) {\n throw new Error(`Session not found: ${sessionKey}`);\n }\n\n const current = sessionRowToMetadata(sessionKey, existing);\n const merged = { ...current, ...updates, key: sessionKey };\n const now = Date.now();\n\n db.prepare(\n `UPDATE sessions SET\n status = ?,\n name = ?,\n tags_json = ?,\n updated_at = ?,\n last_accessed_at = ?,\n session_started_at = ?,\n last_interaction_at = ?,\n source_channel = ?,\n source_chat_id = ?,\n session_type = ?,\n routing_json = ?,\n custom_data_json = ?,\n abort_cutoff_timestamp = ?,\n message_count = ?,\n estimated_tokens = ?,\n compacted_count = ?,\n thinking_level = ?,\n verbose_level = ?\n WHERE session_key = ?`,\n ).run(\n merged.status,\n merged.name ?? null,\n JSON.stringify(merged.tags ?? []),\n Date.parse(merged.updatedAt) || now,\n Date.parse(merged.lastAccessedAt) || now,\n merged.sessionStartedAt ? Date.parse(merged.sessionStartedAt) : existing.session_started_at,\n merged.lastInteractionAt ? Date.parse(merged.lastInteractionAt) : existing.last_interaction_at,\n merged.sourceChannel,\n merged.sourceChatId,\n merged.sessionType ?? null,\n merged.routing ? JSON.stringify(merged.routing) : null,\n merged.customData ? JSON.stringify(merged.customData) : null,\n merged.abortCutoffTimestamp ?? null,\n merged.messageCount,\n merged.estimatedTokens,\n merged.compactedCount,\n existing.thinking_level,\n existing.verbose_level,\n sessionKey,\n );\n\n const row = readSessionRow(db, sessionKey);\n if (!row) {\n throw new Error(`Session not found after patch: ${sessionKey}`);\n }\n return sessionRowToMetadata(sessionKey, row);\n });\n}\n\nexport function updateSessionStats(\n sessionKey: string,\n stats: { messageCount: number; estimatedTokens: number; lastInteractionAt?: number },\n): void {\n runSqliteWriteTransaction((db) => {\n const now = stats.lastInteractionAt ?? Date.now();\n db.prepare(\n `UPDATE sessions SET\n message_count = ?,\n estimated_tokens = ?,\n updated_at = ?,\n last_accessed_at = ?,\n last_interaction_at = ?\n WHERE session_key = ?`,\n ).run(stats.messageCount, stats.estimatedTokens, now, now, now, sessionKey);\n });\n}\n\nexport function incrementSessionStatsOnAppend(sessionKey: string, tokenDelta = 0): void {\n runSqliteWriteTransaction((db) => {\n const now = Date.now();\n db.prepare(\n `UPDATE sessions SET\n message_count = message_count + 1,\n estimated_tokens = estimated_tokens + ?,\n updated_at = ?,\n last_accessed_at = ?,\n last_interaction_at = ?\n WHERE session_key = ?`,\n ).run(tokenDelta, now, now, now, sessionKey);\n });\n}\n\nexport function resetSessionRecord(\n sessionKey: string,\n cwd: string,\n): { sessionId: string; previousSessionId: string } | null {\n return runSqliteWriteTransaction((db) => {\n const existing = readSessionRow(db, sessionKey);\n if (!existing) {\n return null;\n }\n\n const previousSessionId = existing.current_transcript_id;\n const now = Date.now();\n db.prepare(\n `UPDATE transcripts SET status = 'archived', archive_reason = 'reset', archived_at = ?\n WHERE transcript_id = ?`,\n ).run(now, previousSessionId);\n\n const newTranscriptId = validateSessionId(randomUUID());\n db.prepare(\n `INSERT INTO transcripts (transcript_id, session_key, status, created_at, cwd)\n VALUES (?, ?, 'active', ?, ?)`,\n ).run(newTranscriptId, sessionKey, now, cwd);\n\n db.prepare(\n `UPDATE sessions SET\n current_transcript_id = ?,\n updated_at = ?,\n session_started_at = ?,\n last_interaction_at = NULL,\n message_count = 0,\n estimated_tokens = 0\n WHERE session_key = ?`,\n ).run(newTranscriptId, now, now, sessionKey);\n\n return { sessionId: newTranscriptId, previousSessionId };\n });\n}\n\nexport function deleteSessionRecord(sessionKey: string): boolean {\n return runSqliteWriteTransaction((db) => {\n const existing = readSessionRow(db, sessionKey);\n if (!existing) {\n return false;\n }\n\n const now = Date.now();\n db.prepare(\n `UPDATE transcripts SET status = 'archived', archive_reason = 'delete', archived_at = ?\n WHERE transcript_id = ?`,\n ).run(now, existing.current_transcript_id);\n\n db.prepare(`DELETE FROM sessions WHERE session_key = ?`).run(sessionKey);\n return true;\n });\n}\n\nexport function listSessionsByAgent(agentId: string): SessionMetadata[] {\n const db = getSqliteDatabase();\n const rows = db\n .prepare(\n `SELECT ${SESSION_COLUMNS} FROM sessions WHERE agent_id = ? ORDER BY updated_at DESC`,\n )\n .all(agentId.toLowerCase()) as SessionRow[];\n return rows.map((row) => sessionRowToMetadata(row.session_key, row));\n}\n\nexport function getSessionPersistedLevels(sessionKey: string): {\n thinkingLevel: string | null;\n verboseLevel: string | null;\n} | null {\n const db = getSqliteDatabase();\n const row = db\n .prepare(`SELECT thinking_level, verbose_level FROM sessions WHERE session_key = ?`)\n .get(sessionKey) as { thinking_level: string | null; verbose_level: string | null } | undefined;\n if (!row) {\n return null;\n }\n return { thinkingLevel: row.thinking_level, verboseLevel: row.verbose_level };\n}\n\nexport function findSessionKeyByTranscriptId(transcriptId: string): string | null {\n const db = getSqliteDatabase();\n const row = db\n .prepare(`SELECT session_key FROM sessions WHERE current_transcript_id = ?`)\n .get(transcriptId) as { session_key?: string } | undefined;\n return row?.session_key ?? null;\n}\n\nexport function getGlobalSessionStats(): ReturnType<typeof buildGlobalSessionStats> {\n const all = listSessionMetadata({ limit: 100_000, offset: 0 });\n return buildGlobalSessionStats(all.items);\n}\n\nexport function resolveSessionAgentId(sessionKey: string): string {\n return resolveAgentIdFromSessionKey(sessionKey);\n}\n"],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { DatabaseSync } from 'node:sqlite';
2
+ /**
3
+ * Run an operation inside a `BEGIN IMMEDIATE` transaction with nested SAVEPOINT
4
+ * support and COMMIT retry on SQLITE_BUSY / SQLITE_LOCKED.
5
+ *
6
+ * When called inside an already-active transaction, automatically uses SAVEPOINT
7
+ * so the outer transaction controls final COMMIT/ROLLBACK. When called at the
8
+ * top level, owns the full transaction lifecycle with retry on contention.
9
+ */
10
+ export declare function runSqliteWriteTransaction<T>(fn: (db: DatabaseSync) => T): T;
11
+ export declare function getSqliteDatabase(): DatabaseSync;
@@ -0,0 +1,115 @@
1
+ import { getXopcDatabase } from "./connection.js";
2
+ //#region src/storage/sqlite/transaction.ts
3
+ const RETRYABLE_COMMIT_CODES = new Set(["SQLITE_BUSY", "SQLITE_LOCKED"]);
4
+ const MAX_COMMIT_ATTEMPTS = 8;
5
+ let nextSavepointId = 0;
6
+ function nextSavepointName() {
7
+ nextSavepointId += 1;
8
+ return `xopc_tx_${nextSavepointId}`;
9
+ }
10
+ const transactionDepthByDatabase = /* @__PURE__ */ new WeakMap();
11
+ function getTransactionDepth(db) {
12
+ return transactionDepthByDatabase.get(db) ?? 0;
13
+ }
14
+ function setTransactionDepth(db, depth) {
15
+ if (depth <= 0) {
16
+ transactionDepthByDatabase.delete(db);
17
+ return;
18
+ }
19
+ transactionDepthByDatabase.set(db, depth);
20
+ }
21
+ function isRetryableCommitError(error) {
22
+ const code = error && typeof error === "object" ? error.code : void 0;
23
+ return typeof code === "string" && RETRYABLE_COMMIT_CODES.has(code);
24
+ }
25
+ function commitImmediateTransaction(db) {
26
+ for (const attempt of Array.from({ length: MAX_COMMIT_ATTEMPTS }, (_, i) => i + 1)) try {
27
+ db.exec("COMMIT");
28
+ return;
29
+ } catch (error) {
30
+ if (!isRetryableCommitError(error) || attempt >= MAX_COMMIT_ATTEMPTS) throw error;
31
+ }
32
+ }
33
+ function abortImmediateTransaction(db) {
34
+ try {
35
+ db.exec("ROLLBACK");
36
+ } catch {
37
+ try {
38
+ db.close();
39
+ } catch {}
40
+ }
41
+ }
42
+ function isPromiseLike(value) {
43
+ return Boolean(value && typeof value.then === "function");
44
+ }
45
+ function assertSyncTransactionResult(value) {
46
+ if (isPromiseLike(value)) throw new Error("SQLite write transactions must be synchronous; Promise returns are not supported.");
47
+ }
48
+ /**
49
+ * Run an operation inside a `BEGIN IMMEDIATE` transaction with nested SAVEPOINT
50
+ * support and COMMIT retry on SQLITE_BUSY / SQLITE_LOCKED.
51
+ *
52
+ * When called inside an already-active transaction, automatically uses SAVEPOINT
53
+ * so the outer transaction controls final COMMIT/ROLLBACK. When called at the
54
+ * top level, owns the full transaction lifecycle with retry on contention.
55
+ */
56
+ function runSqliteWriteTransaction(fn) {
57
+ const { db } = getXopcDatabase();
58
+ const depth = getTransactionDepth(db);
59
+ if (depth > 0) {
60
+ const savepoint = nextSavepointName();
61
+ db.exec(`SAVEPOINT ${savepoint}`);
62
+ setTransactionDepth(db, depth + 1);
63
+ try {
64
+ const result = fn(db);
65
+ assertSyncTransactionResult(result);
66
+ db.exec(`RELEASE SAVEPOINT ${savepoint}`);
67
+ return result;
68
+ } catch (error) {
69
+ try {
70
+ db.exec(`ROLLBACK TO SAVEPOINT ${savepoint}`);
71
+ } finally {
72
+ db.exec(`RELEASE SAVEPOINT ${savepoint}`);
73
+ }
74
+ throw error;
75
+ } finally {
76
+ setTransactionDepth(db, depth);
77
+ }
78
+ }
79
+ db.exec("BEGIN IMMEDIATE");
80
+ setTransactionDepth(db, 1);
81
+ let transactionActive = true;
82
+ let result;
83
+ try {
84
+ result = fn(db);
85
+ assertSyncTransactionResult(result);
86
+ } catch (error) {
87
+ try {
88
+ abortImmediateTransaction(db);
89
+ transactionActive = false;
90
+ } catch {}
91
+ throw error;
92
+ } finally {
93
+ if (!transactionActive) setTransactionDepth(db, 0);
94
+ }
95
+ try {
96
+ commitImmediateTransaction(db);
97
+ transactionActive = false;
98
+ return result;
99
+ } catch (error) {
100
+ try {
101
+ abortImmediateTransaction(db);
102
+ transactionActive = false;
103
+ } catch {}
104
+ throw error;
105
+ } finally {
106
+ if (!transactionActive) setTransactionDepth(db, 0);
107
+ }
108
+ }
109
+ function getSqliteDatabase() {
110
+ return getXopcDatabase().db;
111
+ }
112
+ //#endregion
113
+ export { getSqliteDatabase, runSqliteWriteTransaction };
114
+
115
+ //# sourceMappingURL=transaction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transaction.js","names":[],"sources":["../../../../src/storage/sqlite/transaction.ts"],"sourcesContent":["import type { DatabaseSync } from 'node:sqlite';\n\nimport { getXopcDatabase } from './connection.js';\n\nconst RETRYABLE_COMMIT_CODES = new Set(['SQLITE_BUSY', 'SQLITE_LOCKED']);\nconst MAX_COMMIT_ATTEMPTS = 8;\n\nlet nextSavepointId = 0;\n\nfunction nextSavepointName(): string {\n nextSavepointId += 1;\n return `xopc_tx_${nextSavepointId}`;\n}\n\nconst transactionDepthByDatabase = new WeakMap<DatabaseSync, number>();\n\nfunction getTransactionDepth(db: DatabaseSync): number {\n return transactionDepthByDatabase.get(db) ?? 0;\n}\n\nfunction setTransactionDepth(db: DatabaseSync, depth: number): void {\n if (depth <= 0) {\n transactionDepthByDatabase.delete(db);\n return;\n }\n transactionDepthByDatabase.set(db, depth);\n}\n\nfunction isRetryableCommitError(error: unknown): boolean {\n const code =\n error && typeof error === 'object'\n ? (error as { code?: unknown }).code\n : undefined;\n return typeof code === 'string' && RETRYABLE_COMMIT_CODES.has(code);\n}\n\nfunction commitImmediateTransaction(db: DatabaseSync): void {\n for (const attempt of Array.from({ length: MAX_COMMIT_ATTEMPTS }, (_, i) => i + 1)) {\n try {\n db.exec('COMMIT');\n return;\n } catch (error) {\n if (!isRetryableCommitError(error) || attempt >= MAX_COMMIT_ATTEMPTS) {\n throw error;\n }\n }\n }\n}\n\nfunction abortImmediateTransaction(db: DatabaseSync): void {\n try {\n db.exec('ROLLBACK');\n } catch {\n try {\n db.close();\n } catch {\n // Preserve the original error; close failure is secondary.\n }\n }\n}\n\nfunction isPromiseLike(value: unknown): value is PromiseLike<unknown> {\n return Boolean(value && typeof (value as { then?: unknown }).then === 'function');\n}\n\nfunction assertSyncTransactionResult(value: unknown): void {\n if (isPromiseLike(value)) {\n throw new Error(\n 'SQLite write transactions must be synchronous; Promise returns are not supported.',\n );\n }\n}\n\n/**\n * Run an operation inside a `BEGIN IMMEDIATE` transaction with nested SAVEPOINT\n * support and COMMIT retry on SQLITE_BUSY / SQLITE_LOCKED.\n *\n * When called inside an already-active transaction, automatically uses SAVEPOINT\n * so the outer transaction controls final COMMIT/ROLLBACK. When called at the\n * top level, owns the full transaction lifecycle with retry on contention.\n */\nexport function runSqliteWriteTransaction<T>(fn: (db: DatabaseSync) => T): T {\n const { db } = getXopcDatabase();\n const depth = getTransactionDepth(db);\n\n // Nested call: use SAVEPOINT so outer transaction stays in control.\n if (depth > 0) {\n const savepoint = nextSavepointName();\n db.exec(`SAVEPOINT ${savepoint}`);\n setTransactionDepth(db, depth + 1);\n try {\n const result = fn(db);\n assertSyncTransactionResult(result);\n db.exec(`RELEASE SAVEPOINT ${savepoint}`);\n return result;\n } catch (error) {\n try {\n db.exec(`ROLLBACK TO SAVEPOINT ${savepoint}`);\n } finally {\n db.exec(`RELEASE SAVEPOINT ${savepoint}`);\n }\n throw error;\n } finally {\n setTransactionDepth(db, depth);\n }\n }\n\n // Top-level: own the full BEGIN IMMEDIATE / COMMIT / ROLLBACK lifecycle.\n db.exec('BEGIN IMMEDIATE');\n setTransactionDepth(db, 1);\n let transactionActive = true;\n let result: T;\n try {\n result = fn(db);\n assertSyncTransactionResult(result);\n } catch (error) {\n try {\n abortImmediateTransaction(db);\n transactionActive = false;\n } catch {\n // Preserve original error; rollback failure is secondary.\n }\n throw error;\n } finally {\n if (!transactionActive) {\n setTransactionDepth(db, 0);\n }\n }\n\n try {\n commitImmediateTransaction(db);\n transactionActive = false;\n return result;\n } catch (error) {\n try {\n abortImmediateTransaction(db);\n transactionActive = false;\n } catch {\n // Preserve original error; rollback failure is secondary.\n }\n throw error;\n } finally {\n if (!transactionActive) {\n setTransactionDepth(db, 0);\n }\n }\n}\n\nexport function getSqliteDatabase(): DatabaseSync {\n return getXopcDatabase().db;\n}\n"],"mappings":";;AAIA,MAAM,yBAAyB,IAAI,IAAI,CAAC,eAAe,gBAAgB,CAAC;AACxE,MAAM,sBAAsB;AAE5B,IAAI,kBAAkB;AAEtB,SAAS,oBAA4B;AACnC,oBAAmB;AACnB,QAAO,WAAW;;AAGpB,MAAM,6CAA6B,IAAI,SAA+B;AAEtE,SAAS,oBAAoB,IAA0B;AACrD,QAAO,2BAA2B,IAAI,GAAG,IAAI;;AAG/C,SAAS,oBAAoB,IAAkB,OAAqB;AAClE,KAAI,SAAS,GAAG;AACd,6BAA2B,OAAO,GAAG;AACrC;;AAEF,4BAA2B,IAAI,IAAI,MAAM;;AAG3C,SAAS,uBAAuB,OAAyB;CACvD,MAAM,OACJ,SAAS,OAAO,UAAU,WACrB,MAA6B,OAC9B,KAAA;AACN,QAAO,OAAO,SAAS,YAAY,uBAAuB,IAAI,KAAK;;AAGrE,SAAS,2BAA2B,IAAwB;AAC1D,MAAK,MAAM,WAAW,MAAM,KAAK,EAAE,QAAQ,qBAAqB,GAAG,GAAG,MAAM,IAAI,EAAE,CAChF,KAAI;AACF,KAAG,KAAK,SAAS;AACjB;UACO,OAAO;AACd,MAAI,CAAC,uBAAuB,MAAM,IAAI,WAAW,oBAC/C,OAAM;;;AAMd,SAAS,0BAA0B,IAAwB;AACzD,KAAI;AACF,KAAG,KAAK,WAAW;SACb;AACN,MAAI;AACF,MAAG,OAAO;UACJ;;;AAMZ,SAAS,cAAc,OAA+C;AACpE,QAAO,QAAQ,SAAS,OAAQ,MAA6B,SAAS,WAAW;;AAGnF,SAAS,4BAA4B,OAAsB;AACzD,KAAI,cAAc,MAAM,CACtB,OAAM,IAAI,MACR,oFACD;;;;;;;;;;AAYL,SAAgB,0BAA6B,IAAgC;CAC3E,MAAM,EAAE,OAAO,iBAAiB;CAChC,MAAM,QAAQ,oBAAoB,GAAG;AAGrC,KAAI,QAAQ,GAAG;EACb,MAAM,YAAY,mBAAmB;AACrC,KAAG,KAAK,aAAa,YAAY;AACjC,sBAAoB,IAAI,QAAQ,EAAE;AAClC,MAAI;GACF,MAAM,SAAS,GAAG,GAAG;AACrB,+BAA4B,OAAO;AACnC,MAAG,KAAK,qBAAqB,YAAY;AACzC,UAAO;WACA,OAAO;AACd,OAAI;AACF,OAAG,KAAK,yBAAyB,YAAY;aACrC;AACR,OAAG,KAAK,qBAAqB,YAAY;;AAE3C,SAAM;YACE;AACR,uBAAoB,IAAI,MAAM;;;AAKlC,IAAG,KAAK,kBAAkB;AAC1B,qBAAoB,IAAI,EAAE;CAC1B,IAAI,oBAAoB;CACxB,IAAI;AACJ,KAAI;AACF,WAAS,GAAG,GAAG;AACf,8BAA4B,OAAO;UAC5B,OAAO;AACd,MAAI;AACF,6BAA0B,GAAG;AAC7B,uBAAoB;UACd;AAGR,QAAM;WACE;AACR,MAAI,CAAC,kBACH,qBAAoB,IAAI,EAAE;;AAI9B,KAAI;AACF,6BAA2B,GAAG;AAC9B,sBAAoB;AACpB,SAAO;UACA,OAAO;AACd,MAAI;AACF,6BAA0B,GAAG;AAC7B,uBAAoB;UACd;AAGR,QAAM;WACE;AACR,MAAI,CAAC,kBACH,qBAAoB,IAAI,EAAE;;;AAKhC,SAAgB,oBAAkC;AAChD,QAAO,iBAAiB,CAAC"}
@@ -0,0 +1,34 @@
1
+ import type { AgentMessage } from '@earendil-works/pi-agent-core';
2
+ import type { CompactionCheckpointDetail, CompactionCheckpointSummary } from '../../session/types.js';
3
+ import type { TranscriptCompactionRecord } from '../../session/transcript-format.js';
4
+ import { type TranscriptStoredRow } from '../../session/session-context-for-llm.js';
5
+ import { type TranscriptEntryRow } from './row-mappers.js';
6
+ export declare function appendTranscriptEntry(sessionKey: string, row: TranscriptStoredRow, opts?: {
7
+ transcriptId?: string;
8
+ tokenDelta?: number;
9
+ }): TranscriptEntryRow;
10
+ export declare function loadTranscriptRowsForSession(sessionKey: string): TranscriptStoredRow[];
11
+ export declare function loadTranscriptRows(transcriptId: string): TranscriptStoredRow[];
12
+ export declare function loadLlmMessagesForSession(sessionKey: string): AgentMessage[];
13
+ export declare function replaceTranscriptRows(sessionKey: string, rows: TranscriptStoredRow[], opts?: {
14
+ appendCompaction?: TranscriptCompactionRecord;
15
+ }): void;
16
+ export declare function loadCheckpointRows(sessionKey: string, checkpointId: string): TranscriptStoredRow[];
17
+ export declare function paginateTranscriptMessages(sessionKey: string, options?: {
18
+ offset?: number;
19
+ limit?: number;
20
+ beforeSeq?: number;
21
+ /** 0-based exclusive end index (legacy cursor from gateway UI). */
22
+ beforeEndIndex?: number;
23
+ includeContext?: boolean;
24
+ }): {
25
+ rows: TranscriptStoredRow[];
26
+ messages: AgentMessage[];
27
+ total: number;
28
+ startSeq: number;
29
+ endSeq: number;
30
+ };
31
+ export declare function captureCompactionCheckpoint(sessionKey: string): string | null;
32
+ export declare function listCompactionCheckpoints(sessionKey: string): CompactionCheckpointSummary[];
33
+ export declare function getCompactionCheckpointDetail(sessionKey: string, checkpointId: string): CompactionCheckpointDetail | null;
34
+ export declare function restoreCompactionCheckpoint(sessionKey: string, checkpointId: string): void;