@xopcai/xopc 0.0.89 → 0.0.91

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 (267) hide show
  1. package/README.md +36 -12
  2. package/README.zh-CN.md +36 -12
  3. package/dist/browser-ext/manifest.json +1 -1
  4. package/dist/extensions/telegram/xopc.extension.json +1 -1
  5. package/dist/gateway/static/root/assets/Combination-HAlzriaz.js +41 -0
  6. package/dist/gateway/static/root/assets/agents-bVWUlrlD.js +222 -0
  7. package/dist/gateway/static/root/assets/apps-page-CIC8bmvZ.js +1 -0
  8. package/dist/gateway/static/root/assets/{attachment-preview-renderer-CpyoFbs4.js → attachment-preview-renderer-DBAxQXb-.js} +2 -2
  9. package/dist/gateway/static/root/assets/{attachment-process-heavy-CqVriadb.js → attachment-process-heavy-Csq3TrrP.js} +4 -4
  10. package/dist/gateway/static/root/assets/channels-settings-C8G8RAAP.js +1 -0
  11. package/dist/gateway/static/root/assets/{channels-status-swr-DaHGkRF1.js → channels-status-swr-CYWL5DLD.js} +1 -1
  12. package/dist/gateway/static/root/assets/circle-check-C23XjkUj.js +1 -0
  13. package/dist/gateway/static/root/assets/copy-Dv6d4Dvw.js +1 -0
  14. package/dist/gateway/static/root/assets/cron-api-TVqLlGAC.js +1 -0
  15. package/dist/gateway/static/root/assets/cron-dreaming-jobs-Ip703-qM.js +2 -0
  16. package/dist/gateway/static/root/assets/cron-page-BtcFYlvv.js +1 -0
  17. package/dist/gateway/static/root/assets/dist-CUV1uY5f.js +1 -0
  18. package/dist/gateway/static/root/assets/{extension-debug-page-CtuKJ9tE.js → extension-debug-page-mTLHRDp1.js} +1 -1
  19. package/dist/gateway/static/root/assets/{extension-page-ykzjOkR5.js → extension-page-iI8BI7WK.js} +1 -1
  20. package/dist/gateway/static/root/assets/{extension-settings-page-Ce2qrdpO.js → extension-settings-page-ByXcdubM.js} +1 -1
  21. package/dist/gateway/static/root/assets/{fetch-C9FFJjuH.js → fetch-BWtQq_Ys.js} +1 -1
  22. package/dist/gateway/static/root/assets/{field-primitives-BFcrNeTU.js → field-primitives-BsZ-4VT5.js} +1 -1
  23. package/dist/gateway/static/root/assets/{heartbeat-config-api-CEg4Vr9R.js → heartbeat-config-api-WjTsRLCU.js} +1 -1
  24. package/dist/gateway/static/root/assets/{index-CZfy9oxs.js → index-CKkR-v9U.js} +101 -97
  25. package/dist/gateway/static/root/assets/index-VlELBY99.css +1 -0
  26. package/dist/gateway/static/root/assets/logs-page-ClnIpxfd.js +1 -0
  27. package/dist/gateway/static/root/assets/note-detail-page-B91pLkEI.css +1 -0
  28. package/dist/gateway/static/root/assets/note-detail-page-DJ2Mb4x7.js +179 -0
  29. package/dist/gateway/static/root/assets/note-time-JLBPSLzK.js +1 -0
  30. package/dist/gateway/static/root/assets/notes-page-BE-75qz9.js +1 -0
  31. package/dist/gateway/static/root/assets/{pdf-BnEvgIXZ.js → pdf-epILhEOn.js} +1 -1
  32. package/dist/gateway/static/root/assets/preload-helper-zJ_50EbN.js +1 -0
  33. package/dist/gateway/static/root/assets/sessions-page-bJJkWtTl.js +1 -0
  34. package/dist/gateway/static/root/assets/{settings-form-section-BqdzA28u.js → settings-form-section-DSYCknxM.js} +1 -1
  35. package/dist/gateway/static/root/assets/settings-page-WcMXLq2U.js +3 -0
  36. package/dist/gateway/static/root/assets/share-preview-page-awRqs4hV.js +2 -0
  37. package/dist/gateway/static/root/assets/skills-page-Lu-i1JG7.js +2 -0
  38. package/dist/gateway/static/root/assets/{theme-store-CNqbmTNV.js → theme-store-BC-42BoZ.js} +1 -1
  39. package/dist/gateway/static/root/assets/toast-z0toXu32.js +1 -0
  40. package/dist/gateway/static/root/assets/url-CY1RQKTU.js +3 -0
  41. package/dist/gateway/static/root/assets/{utils-BWm2tG2w.js → utils-DX3TQuap.js} +1 -1
  42. package/dist/gateway/static/root/assets/vendor-codemirror-DYoKfS8f.js +45 -0
  43. package/dist/gateway/static/root/assets/voice-api-key-field-B5uKlDqA.js +1 -0
  44. package/dist/gateway/static/root/assets/workflow-page.utils-ClC37yEp.js +1 -0
  45. package/dist/gateway/static/root/assets/workflows-page-C7VhIXtR.js +27 -0
  46. package/dist/gateway/static/root/index.html +11 -7
  47. package/dist/package.js +1 -1
  48. package/dist/src/agent/skills/marketplace/adapters/skillhub/adapter.js +20 -18
  49. package/dist/src/agent/skills/marketplace/adapters/skillhub/adapter.js.map +1 -1
  50. package/dist/src/agent/tools/cronjob-tool.d.ts +6 -0
  51. package/dist/src/agent/tools/cronjob-tool.js +74 -9
  52. package/dist/src/agent/tools/cronjob-tool.js.map +1 -1
  53. package/dist/src/agent/tools/edit.d.ts +5 -1
  54. package/dist/src/agent/tools/edit.js +7 -5
  55. package/dist/src/agent/tools/edit.js.map +1 -1
  56. package/dist/src/agent/tools/factory.js +2 -2
  57. package/dist/src/agent/tools/factory.js.map +1 -1
  58. package/dist/src/agent/tools/write.d.ts +5 -1
  59. package/dist/src/agent/tools/write.js +7 -5
  60. package/dist/src/agent/tools/write.js.map +1 -1
  61. package/dist/src/agent/workflow/agent-progress.js +2 -0
  62. package/dist/src/agent/workflow/agent-progress.js.map +1 -1
  63. package/dist/src/agent/workflow/builtins/client-proposal.d.ts +12 -0
  64. package/dist/src/agent/workflow/builtins/client-proposal.js +155 -0
  65. package/dist/src/agent/workflow/builtins/client-proposal.js.map +1 -0
  66. package/dist/src/agent/workflow/builtins/competitor-scan.d.ts +12 -0
  67. package/dist/src/agent/workflow/builtins/competitor-scan.js +150 -0
  68. package/dist/src/agent/workflow/builtins/competitor-scan.js.map +1 -0
  69. package/dist/src/agent/workflow/builtins/content-draft.d.ts +13 -0
  70. package/dist/src/agent/workflow/builtins/content-draft.js +146 -0
  71. package/dist/src/agent/workflow/builtins/content-draft.js.map +1 -0
  72. package/dist/src/agent/workflow/builtins/content-repurpose.d.ts +11 -0
  73. package/dist/src/agent/workflow/builtins/content-repurpose.js +137 -0
  74. package/dist/src/agent/workflow/builtins/content-repurpose.js.map +1 -0
  75. package/dist/src/agent/workflow/builtins/decision-compare.d.ts +13 -0
  76. package/dist/src/agent/workflow/builtins/decision-compare.js +173 -0
  77. package/dist/src/agent/workflow/builtins/decision-compare.js.map +1 -0
  78. package/dist/src/agent/workflow/builtins/inbox-triage.d.ts +11 -0
  79. package/dist/src/agent/workflow/builtins/inbox-triage.js +148 -0
  80. package/dist/src/agent/workflow/builtins/inbox-triage.js.map +1 -0
  81. package/dist/src/agent/workflow/builtins/index.d.ts +10 -1
  82. package/dist/src/agent/workflow/builtins/index.js +46 -1
  83. package/dist/src/agent/workflow/builtins/index.js.map +1 -1
  84. package/dist/src/agent/workflow/builtins/meeting-prep.d.ts +12 -0
  85. package/dist/src/agent/workflow/builtins/meeting-prep.js +144 -0
  86. package/dist/src/agent/workflow/builtins/meeting-prep.js.map +1 -0
  87. package/dist/src/agent/workflow/builtins/offer-design.d.ts +12 -0
  88. package/dist/src/agent/workflow/builtins/offer-design.js +161 -0
  89. package/dist/src/agent/workflow/builtins/offer-design.js.map +1 -0
  90. package/dist/src/agent/workflow/builtins/weekly-review.d.ts +12 -0
  91. package/dist/src/agent/workflow/builtins/weekly-review.js +131 -0
  92. package/dist/src/agent/workflow/builtins/weekly-review.js.map +1 -0
  93. package/dist/src/agent/workflow/step-labels.js +2 -2
  94. package/dist/src/agent/workflow/step-labels.js.map +1 -1
  95. package/dist/src/agent/workflow/subagent-runner.js +3 -1
  96. package/dist/src/agent/workflow/subagent-runner.js.map +1 -1
  97. package/dist/src/agent/workflow/types.d.ts +4 -0
  98. package/dist/src/chat-commands/agent-edit.d.ts +4 -0
  99. package/dist/src/chat-commands/agent-edit.js +136 -0
  100. package/dist/src/chat-commands/agent-edit.js.map +1 -0
  101. package/dist/src/chat-commands/index.d.ts +1 -0
  102. package/dist/src/chat-commands/index.js +3 -1
  103. package/dist/src/chat-commands/index.js.map +1 -1
  104. package/dist/src/cli/bin.js +2 -0
  105. package/dist/src/cli/bin.js.map +1 -1
  106. package/dist/src/cli/commands/cron.js +42 -3
  107. package/dist/src/cli/commands/cron.js.map +1 -1
  108. package/dist/src/cli/commands/doctor/checks/session-integrity.js +79 -56
  109. package/dist/src/cli/commands/doctor/checks/session-integrity.js.map +1 -1
  110. package/dist/src/cli/commands/gateway/lifecycle.js +1 -1
  111. package/dist/src/cli/commands/update.js +86 -79
  112. package/dist/src/cli/commands/update.js.map +1 -1
  113. package/dist/src/commands/agents.config.d.ts +3 -2
  114. package/dist/src/commands/agents.config.js +5 -2
  115. package/dist/src/commands/agents.config.js.map +1 -1
  116. package/dist/src/config/agent-typed-models.d.ts +2 -7
  117. package/dist/src/config/agent-typed-models.js +3 -14
  118. package/dist/src/config/agent-typed-models.js.map +1 -1
  119. package/dist/src/config/localized-text.d.ts +6 -0
  120. package/dist/src/config/localized-text.js +42 -0
  121. package/dist/src/config/localized-text.js.map +1 -0
  122. package/dist/src/config/models-json.d.ts +6 -6
  123. package/dist/src/config/schema.d.ts +6 -21
  124. package/dist/src/config/schema.js +4 -4
  125. package/dist/src/config/schema.js.map +1 -1
  126. package/dist/src/cron/executor.d.ts +2 -0
  127. package/dist/src/cron/executor.js +111 -1
  128. package/dist/src/cron/executor.js.map +1 -1
  129. package/dist/src/cron/types.d.ts +8 -1
  130. package/dist/src/cron/validation.d.ts +4 -0
  131. package/dist/src/cron/validation.js +4 -3
  132. package/dist/src/cron/validation.js.map +1 -1
  133. package/dist/src/cron/workflow-run-completion.d.ts +23 -0
  134. package/dist/src/cron/workflow-run-completion.js +72 -0
  135. package/dist/src/cron/workflow-run-completion.js.map +1 -0
  136. package/dist/src/extensions/update.d.ts +51 -0
  137. package/dist/src/extensions/update.js +260 -0
  138. package/dist/src/extensions/update.js.map +1 -0
  139. package/dist/src/gateway/agents-admin.d.ts +15 -8
  140. package/dist/src/gateway/agents-admin.js +77 -28
  141. package/dist/src/gateway/agents-admin.js.map +1 -1
  142. package/dist/src/gateway/heartbeat/service.js +1 -1
  143. package/dist/src/gateway/hono/lib/config-payload.d.ts +6 -0
  144. package/dist/src/gateway/hono/lib/config-payload.js +3 -1
  145. package/dist/src/gateway/hono/lib/config-payload.js.map +1 -1
  146. package/dist/src/gateway/hono/middleware/auth.d.ts +2 -0
  147. package/dist/src/gateway/hono/middleware/auth.js +11 -7
  148. package/dist/src/gateway/hono/middleware/auth.js.map +1 -1
  149. package/dist/src/gateway/hono/routes/agents.js +55 -12
  150. package/dist/src/gateway/hono/routes/agents.js.map +1 -1
  151. package/dist/src/gateway/hono/routes/config-patch/agents.js +1 -1
  152. package/dist/src/gateway/hono/routes/config-patch/gateway.d.ts +2 -2
  153. package/dist/src/gateway/hono/routes/config-patch/gateway.js +12 -0
  154. package/dist/src/gateway/hono/routes/config-patch/gateway.js.map +1 -1
  155. package/dist/src/gateway/hono/routes/lazy-bundles.js +8 -0
  156. package/dist/src/gateway/hono/routes/lazy-bundles.js.map +1 -1
  157. package/dist/src/gateway/hono/routes/notes.d.ts +3 -0
  158. package/dist/src/gateway/hono/routes/notes.js +274 -0
  159. package/dist/src/gateway/hono/routes/notes.js.map +1 -0
  160. package/dist/src/gateway/hono/routes/sessions.js.map +1 -1
  161. package/dist/src/gateway/hono/routes/update.js +55 -107
  162. package/dist/src/gateway/hono/routes/update.js.map +1 -1
  163. package/dist/src/gateway/hono/routes/workflows.js +3 -1
  164. package/dist/src/gateway/hono/routes/workflows.js.map +1 -1
  165. package/dist/src/gateway/server.js +2 -0
  166. package/dist/src/gateway/server.js.map +1 -1
  167. package/dist/src/gateway/service.d.ts +3 -0
  168. package/dist/src/gateway/service.js +12 -1
  169. package/dist/src/gateway/service.js.map +1 -1
  170. package/dist/src/gateway/workspace-ripgrep.d.ts +6 -0
  171. package/dist/src/gateway/workspace-ripgrep.js +62 -11
  172. package/dist/src/gateway/workspace-ripgrep.js.map +1 -1
  173. package/dist/src/heartbeat/index.js +1 -1
  174. package/dist/src/infra/brew.d.ts +4 -0
  175. package/dist/src/infra/brew.js +20 -0
  176. package/dist/src/infra/brew.js.map +1 -0
  177. package/dist/src/infra/package-json.d.ts +2 -0
  178. package/dist/src/infra/package-json.js +23 -0
  179. package/dist/src/infra/package-json.js.map +1 -0
  180. package/dist/src/infra/package-update-steps.d.ts +35 -0
  181. package/dist/src/infra/package-update-steps.js +304 -0
  182. package/dist/src/infra/package-update-steps.js.map +1 -0
  183. package/dist/src/infra/path-env.d.ts +11 -0
  184. package/dist/src/infra/path-env.js +90 -0
  185. package/dist/src/infra/path-env.js.map +1 -0
  186. package/dist/src/infra/path-prepend.d.ts +7 -0
  187. package/dist/src/infra/path-prepend.js +44 -0
  188. package/dist/src/infra/path-prepend.js.map +1 -0
  189. package/dist/src/infra/stable-node-path.d.ts +2 -0
  190. package/dist/src/infra/stable-node-path.js +28 -0
  191. package/dist/src/infra/stable-node-path.js.map +1 -0
  192. package/dist/src/infra/update-global.d.ts +30 -23
  193. package/dist/src/infra/update-global.js +113 -64
  194. package/dist/src/infra/update-global.js.map +1 -1
  195. package/dist/src/infra/update-log.d.ts +1 -0
  196. package/dist/src/infra/update-log.js +12 -0
  197. package/dist/src/infra/update-log.js.map +1 -0
  198. package/dist/src/infra/update-restart.d.ts +20 -0
  199. package/dist/src/infra/update-restart.js +165 -0
  200. package/dist/src/infra/update-restart.js.map +1 -0
  201. package/dist/src/infra/update-runner.d.ts +89 -1
  202. package/dist/src/infra/update-runner.js +604 -173
  203. package/dist/src/infra/update-runner.js.map +1 -1
  204. package/dist/src/infra/update-startup.d.ts +3 -0
  205. package/dist/src/infra/update-startup.js +8 -4
  206. package/dist/src/infra/update-startup.js.map +1 -1
  207. package/dist/src/notes/attachment-ref.d.ts +9 -0
  208. package/dist/src/notes/attachment-ref.js +27 -0
  209. package/dist/src/notes/attachment-ref.js.map +1 -0
  210. package/dist/src/notes/index.d.ts +4 -0
  211. package/dist/src/notes/index.js +4 -0
  212. package/dist/src/notes/note-attachment-sync.d.ts +7 -0
  213. package/dist/src/notes/note-attachment-sync.js +46 -0
  214. package/dist/src/notes/note-attachment-sync.js.map +1 -0
  215. package/dist/src/notes/note-index-meta.d.ts +14 -0
  216. package/dist/src/notes/note-index-meta.js +87 -0
  217. package/dist/src/notes/note-index-meta.js.map +1 -0
  218. package/dist/src/notes/paths.d.ts +5 -0
  219. package/dist/src/notes/paths.js +23 -0
  220. package/dist/src/notes/paths.js.map +1 -0
  221. package/dist/src/notes/service.d.ts +42 -0
  222. package/dist/src/notes/service.js +331 -0
  223. package/dist/src/notes/service.js.map +1 -0
  224. package/dist/src/notes/store.d.ts +33 -0
  225. package/dist/src/notes/store.js +317 -0
  226. package/dist/src/notes/store.js.map +1 -0
  227. package/dist/src/notes/types.d.ts +162 -0
  228. package/dist/src/notes/types.js +1 -0
  229. package/dist/src/routing/resolve-route.d.ts +3 -1
  230. package/dist/src/routing/resolve-route.js.map +1 -1
  231. package/dist/src/session/store.d.ts +5 -3
  232. package/dist/src/session/store.js +66 -20
  233. package/dist/src/session/store.js.map +1 -1
  234. package/dist/src/utils/logger/stats.d.ts +1 -1
  235. package/dist/src/workflows/domain/event.d.ts +3 -0
  236. package/dist/src/workflows/domain/run.d.ts +3 -0
  237. package/dist/src/workflows/domain/run.js.map +1 -1
  238. package/dist/src/workflows/engine/projector.js +17 -0
  239. package/dist/src/workflows/engine/projector.js.map +1 -1
  240. package/dist/src/workflows/engine/workflow-engine.js +127 -0
  241. package/dist/src/workflows/engine/workflow-engine.js.map +1 -1
  242. package/dist/src/workflows/index.js +1 -1
  243. package/dist/src/workflows/service/run-view-to-snapshot.js +3 -1
  244. package/dist/src/workflows/service/run-view-to-snapshot.js.map +1 -1
  245. package/dist/src/workflows/service/workflow-run-service.d.ts +1 -0
  246. package/dist/src/workflows/service/workflow-run-service.js +4 -1
  247. package/dist/src/workflows/service/workflow-run-service.js.map +1 -1
  248. package/dist/src/workflows/service/workflow-session-bridge.js +1 -1
  249. package/package.json +1 -1
  250. package/dist/gateway/static/root/assets/agents-B6PJB07W.js +0 -222
  251. package/dist/gateway/static/root/assets/apps-page-BOr0B1wv.js +0 -1
  252. package/dist/gateway/static/root/assets/channels-settings-BelUKggl.js +0 -1
  253. package/dist/gateway/static/root/assets/cron-api-CjOg-BIj.js +0 -1
  254. package/dist/gateway/static/root/assets/cron-dreaming-jobs-DueM3rBz.js +0 -2
  255. package/dist/gateway/static/root/assets/cron-page-DhoZmZXb.js +0 -1
  256. package/dist/gateway/static/root/assets/dist-6LecgDx5.js +0 -1
  257. package/dist/gateway/static/root/assets/dist-BTWC-BTN.js +0 -45
  258. package/dist/gateway/static/root/assets/index-CiN1cQiQ.css +0 -1
  259. package/dist/gateway/static/root/assets/logs-page-BwWLfqvd.js +0 -1
  260. package/dist/gateway/static/root/assets/sessions-page-DV5WN8uk.js +0 -1
  261. package/dist/gateway/static/root/assets/settings-page-CfOBRbPX.js +0 -3
  262. package/dist/gateway/static/root/assets/share-preview-page-Di5Bzh4g.js +0 -2
  263. package/dist/gateway/static/root/assets/skills-page-D0H5Kaxg.js +0 -2
  264. package/dist/gateway/static/root/assets/url-aYn-Rj1C.js +0 -7
  265. package/dist/gateway/static/root/assets/vendor-codemirror-D0yxdRpg.js +0 -58
  266. package/dist/gateway/static/root/assets/voice-api-key-field-X2UfnHeq.js +0 -1
  267. package/dist/gateway/static/root/assets/workflows-page-BOPpO3NG.js +0 -27
@@ -1,3 +1,4 @@
1
+ import { resolveStateDir } from "../config/paths-state.js";
1
2
  import { init_agent_scope, listAgentEntries, resolveDefaultAgentId } from "../agent/agent-scope.js";
2
3
  import { init_session_key, parseSessionKey } from "../routing/session-key.js";
3
4
  import { resolveEffectiveAgentProfileForSession } from "../config/agent-profile.js";
@@ -23,6 +24,7 @@ import { join } from "path";
23
24
  import { existsSync } from "fs";
24
25
  import { copyFile, mkdir, readdir, stat, unlink } from "fs/promises";
25
26
  import { randomUUID } from "node:crypto";
27
+ import { performance } from "node:perf_hooks";
26
28
  //#region src/session/store.ts
27
29
  init_paths();
28
30
  init_agent_scope();
@@ -34,6 +36,7 @@ init_session_id();
34
36
  const log = createLogger("SessionStore");
35
37
  const INDEX_VERSION = "1.0";
36
38
  const DELETED_MARKER = ".jsonl.deleted.";
39
+ const ALL_SESSIONS_MAP_CACHE_TTL_MS = 2e3;
37
40
  var SessionStore = class {
38
41
  sessionsDir;
39
42
  archiveDir;
@@ -42,6 +45,7 @@ var SessionStore = class {
42
45
  compactor;
43
46
  storeMutationDepth = 0;
44
47
  storeMutationChain = Promise.resolve();
48
+ allSessionsMapCache;
45
49
  /** Cache of per-agent sessions dirs to avoid re-resolution on every call. */
46
50
  agentSessionsDirCache = /* @__PURE__ */ new Map();
47
51
  constructor(options, windowConfig, compactionConfig) {
@@ -104,23 +108,54 @@ var SessionStore = class {
104
108
  async readMap() {
105
109
  return readSessionsJsonFile(this.storePath);
106
110
  }
111
+ invalidateAllSessionsMapCache() {
112
+ this.allSessionsMapCache = void 0;
113
+ }
114
+ async discoverSessionMapPaths() {
115
+ const agents = listAgentEntries(this.options.config);
116
+ const defaultId = resolveDefaultAgentId(this.options.config);
117
+ const agentIds = new Set([defaultId, ...agents.map((agent) => agent.id)]);
118
+ const agentsRoot = join(resolveStateDir(process.env), "agents");
119
+ if (existsSync(agentsRoot)) {
120
+ const entries = await readdir(agentsRoot, { withFileTypes: true }).catch(() => []);
121
+ for (const entry of entries) if (entry.isDirectory()) agentIds.add(entry.name);
122
+ }
123
+ return [...agentIds].map((agentId) => ({
124
+ agentId,
125
+ mapPath: join(resolveSessionsDir(this.options.config, agentId), FILENAMES.SESSIONS_MAP)
126
+ }));
127
+ }
107
128
  /**
108
- * OpenClaw-aligned: read sessions.json from ALL known agents and merge into a single map.
109
- * Used by aggregation queries (list, getByAgent, getByAccount, etc.) so the gateway UI
110
- * can display sessions across all agents.
129
+ * Unified cross-agent aggregation entry for global session views.
130
+ * Reads configured agents plus existing per-agent session maps under the state directory.
111
131
  */
112
132
  async readAllMaps() {
113
133
  if (this.options.sessionsDir) return this.readMap();
114
- const agents = listAgentEntries(this.options.config);
115
- const defaultId = resolveDefaultAgentId(this.options.config);
116
- const agentIds = new Set([defaultId, ...agents.map((a) => a.id)]);
134
+ const nowMs = Date.now();
135
+ if (this.allSessionsMapCache && this.allSessionsMapCache.expiresAtMs > nowMs) {
136
+ log.debug({ sessionCount: Object.keys(this.allSessionsMapCache.map).length }, "All session maps cache hit");
137
+ return this.allSessionsMapCache.map;
138
+ }
139
+ const startedAt = performance.now();
140
+ const paths = await this.discoverSessionMapPaths();
117
141
  const merged = {};
118
- for (const id of agentIds) {
119
- const path = join(resolveSessionsDir(this.options.config, id), FILENAMES.SESSIONS_MAP);
120
- if (!existsSync(path)) continue;
121
- const map = await readSessionsJsonFile(path);
142
+ let scannedMapCount = 0;
143
+ for (const { mapPath } of paths) {
144
+ if (!existsSync(mapPath)) continue;
145
+ const map = await readSessionsJsonFile(mapPath);
122
146
  Object.assign(merged, map);
147
+ scannedMapCount++;
123
148
  }
149
+ this.allSessionsMapCache = {
150
+ expiresAtMs: nowMs + ALL_SESSIONS_MAP_CACHE_TTL_MS,
151
+ map: merged
152
+ };
153
+ log.debug({
154
+ candidateAgentCount: paths.length,
155
+ scannedMapCount,
156
+ sessionCount: Object.keys(merged).length,
157
+ durationMs: Math.round(performance.now() - startedAt)
158
+ }, "All session maps scanned");
124
159
  return merged;
125
160
  }
126
161
  async getDiskEntry(sessionKey) {
@@ -214,39 +249,44 @@ var SessionStore = class {
214
249
  const keyStorePath = this.resolveStorePathForKey(sessionKey);
215
250
  const keySessionsDir = this.resolveSessionsDirForKey(sessionKey);
216
251
  await mkdir(keySessionsDir, { recursive: true });
217
- return withSessionsJsonLock(keyStorePath, async (map) => {
252
+ let changed = false;
253
+ const entry = await withSessionsJsonLock(keyStorePath, async (map) => {
218
254
  const existing = map[sessionKey];
219
255
  if (existing?.pluginExtensions?.xopc?.metadata) return existing;
220
- let entry = existing;
221
- if (!entry) {
256
+ let nextEntry = existing;
257
+ if (!nextEntry) {
222
258
  const sessionId = randomUUID();
223
259
  validateSessionId(sessionId);
224
260
  const sessionFile = `${sessionId}.jsonl`;
225
261
  const now = Date.now();
226
262
  const metadata = this.buildDefaultMetadata(sessionKey);
227
263
  metadata.transcriptId = sessionId;
228
- entry = {
264
+ nextEntry = {
229
265
  sessionId,
230
266
  updatedAt: now,
231
267
  sessionStartedAt: now,
232
268
  sessionFile,
233
269
  pluginExtensions: { xopc: { metadata } }
234
270
  };
235
- map[sessionKey] = entry;
271
+ map[sessionKey] = nextEntry;
236
272
  await writeTranscriptJsonl({
237
273
  absPath: resolveSessionTranscriptPathInDir(sessionId, keySessionsDir),
238
274
  sessionId,
239
275
  cwd: process.cwd(),
240
276
  rows: []
241
277
  });
242
- } else if (!entry.pluginExtensions?.xopc?.metadata) {
278
+ changed = true;
279
+ } else if (!nextEntry.pluginExtensions?.xopc?.metadata) {
243
280
  const metadata = this.buildDefaultMetadata(sessionKey);
244
- metadata.transcriptId = entry.sessionId;
245
- entry.pluginExtensions = { xopc: { metadata } };
246
- map[sessionKey] = entry;
281
+ metadata.transcriptId = nextEntry.sessionId;
282
+ nextEntry.pluginExtensions = { xopc: { metadata } };
283
+ map[sessionKey] = nextEntry;
284
+ changed = true;
247
285
  }
248
- return entry;
286
+ return nextEntry;
249
287
  });
288
+ if (changed) this.invalidateAllSessionsMapCache();
289
+ return entry;
250
290
  }
251
291
  metadataFromEntry(sessionKey, entry) {
252
292
  const base = entry.pluginExtensions?.xopc?.metadata ?? this.buildDefaultMetadata(sessionKey);
@@ -445,6 +485,7 @@ var SessionStore = class {
445
485
  entry.updatedAt = Date.now();
446
486
  map[key] = entry;
447
487
  });
488
+ this.invalidateAllSessionsMapCache();
448
489
  invalidateSessionSearchIndexCache();
449
490
  log.debug({
450
491
  key,
@@ -495,6 +536,7 @@ var SessionStore = class {
495
536
  meta.updatedAt = new Date(now).toISOString();
496
537
  map[key] = e;
497
538
  });
539
+ this.invalidateAllSessionsMapCache();
498
540
  invalidateSessionSearchIndexCache();
499
541
  log.info({
500
542
  key,
@@ -524,6 +566,7 @@ var SessionStore = class {
524
566
  key
525
567
  }, "Transcript archive on delete failed");
526
568
  }
569
+ this.invalidateAllSessionsMapCache();
527
570
  invalidateSessionSearchIndexCache();
528
571
  log.info({ key }, "Session deleted");
529
572
  return true;
@@ -689,6 +732,7 @@ var SessionStore = class {
689
732
  patchSessionsJsonEntryStats(e, buildSessionsJsonStatsPatch(llm.length, this.estimateTokens(llm)));
690
733
  map[key] = e;
691
734
  });
735
+ this.invalidateAllSessionsMapCache();
692
736
  invalidateSessionSearchIndexCache();
693
737
  }
694
738
  /** Incremental sessions.json stats after guard append (OpenClaw transcript-events). */
@@ -709,6 +753,7 @@ var SessionStore = class {
709
753
  patchSessionsJsonEntryStats(e, buildSessionsJsonStatsPatch(messageCount, this.estimateTokens(llm)));
710
754
  map[sessionKey] = e;
711
755
  });
756
+ this.invalidateAllSessionsMapCache();
712
757
  invalidateSessionSearchIndexCache();
713
758
  });
714
759
  }
@@ -739,6 +784,7 @@ var SessionStore = class {
739
784
  patchSessionsJsonEntryStats(e, buildSessionsJsonStatsPatch(llm.length, this.estimateTokens(llm)));
740
785
  map[key] = e;
741
786
  });
787
+ this.invalidateAllSessionsMapCache();
742
788
  invalidateSessionSearchIndexCache();
743
789
  });
744
790
  }
@@ -1 +1 @@
1
- {"version":3,"file":"store.js","names":["parseRoutingSessionKey"],"sources":["../../../src/session/store.ts"],"sourcesContent":["import { randomUUID } from 'node:crypto';\nimport { copyFile, mkdir, readdir, stat, unlink } from 'fs/promises';\nimport { join } from 'path';\nimport { existsSync } from 'fs';\n\nimport type { AgentMessage } from '@earendil-works/pi-agent-core';\nimport type { CompactionEntry } from '@earendil-works/pi-coding-agent';\n\nimport { loadEntriesFromFile } from './parity/load-jsonl-entries.js';\n\nimport type { Config } from '../config/schema.js';\nimport { resolveSessionsDir, FILENAMES } from '../config/paths.js';\nimport { resolveDefaultAgentId, listAgentEntries } from '../agent/agent-scope.js';\nimport { resolveEffectiveAgentProfileForSession } from '../config/agent-profile.js';\nimport { readPostCompactionContext } from '../agent/reply/post-compaction-context.js';\nimport { parseSessionKey as parseRoutingSessionKey } from '../routing/session-key.js';\nimport { createLogger } from '../utils/logger.js';\nimport { SessionCompactor, type CompactionConfig, type CompactionResult } from '../agent/memory/compaction.js';\nimport { SlidingWindow, type WindowConfig } from '../agent/memory/window.js';\nimport { invalidateSessionSearchIndexCache } from './search-index-cache.js';\nimport type { TranscriptCompactionRecord, XopcSessionTranscriptV1 } from './transcript-format.js';\nimport {\n mergeLlmMessagesPreservingContextRows,\n type TranscriptStoredRow,\n type XopcTranscriptContextEntry,\n} from './session-context-for-llm.js';\nimport { normalizeCompactionCheckpointId } from './compaction-checkpoints.js';\nimport type {\n SessionMetadata,\n SessionDetail,\n SessionListQuery,\n PaginatedResult,\n GlobalSessionStats,\n ExportFormat,\n SessionExport,\n SessionTranscriptSummary,\n CompactionCheckpointSummary,\n CompactionCheckpointDetail,\n} from './types.js';\nimport { SessionStatus } from './types.js';\nimport type { Message } from './types.js';\nimport { parseCompactionCheckpointTranscriptFileName } from './parity/artifacts.js';\nimport { archiveFileOnDisk, resolveSessionFilePath, resolveSessionTranscriptPathInDir } from './parity/transcript-paths.js';\nimport { validateSessionId } from './parity/session-id.js';\nimport { readSessionsJsonFile, withSessionsJsonLock } from './parity/sessions-json-file.js';\nimport {\n buildSessionsJsonStatsPatch,\n incrementSessionsJsonStatsForAppend,\n isAppendOnlyLlmTranscriptMessage,\n patchSessionsJsonEntryStats,\n} from './parity/sessions-json-patch.js';\nimport {\n countTranscriptMessageRows,\n readDisplayMessagePageFromTranscriptFile,\n} from './parity/transcript-pagination.js';\nimport type { XopcSessionDiskEntry } from './parity/xopc-session-disk-entry.js';\nimport {\n appendPiTranscriptContextEntry,\n persistMergedTranscriptRows,\n readTranscriptRowsFromFile,\n rowsToLlmMessages,\n writeTranscriptJsonl,\n} from './parity/jsonl-transcript-io.js';\nimport type { SessionTranscriptUpdate } from './transcript-events.js';\n\nconst log = createLogger('SessionStore');\n\nconst INDEX_VERSION = '1.0';\nconst DELETED_MARKER = '.jsonl.deleted.';\n\nexport interface SessionStoreOptions {\n config: Config;\n agentId?: string;\n sessionsDir?: string;\n}\n\nexport class SessionStore {\n private sessionsDir: string;\n private archiveDir: string;\n private storePath: string;\n private window: SlidingWindow;\n private compactor: SessionCompactor;\n private storeMutationDepth = 0;\n private storeMutationChain: Promise<void> = Promise.resolve();\n /** Cache of per-agent sessions dirs to avoid re-resolution on every call. */\n private agentSessionsDirCache = new Map<string, string>();\n\n constructor(\n private options: SessionStoreOptions,\n windowConfig?: Partial<WindowConfig>,\n compactionConfig?: Partial<CompactionConfig>,\n ) {\n const agentId = options.agentId ?? resolveDefaultAgentId(options.config);\n this.sessionsDir = options.sessionsDir ?? resolveSessionsDir(options.config, agentId);\n this.archiveDir = join(this.sessionsDir, 'archive');\n this.storePath = join(this.sessionsDir, FILENAMES.SESSIONS_MAP);\n this.window = new SlidingWindow(windowConfig);\n this.compactor = new SessionCompactor(compactionConfig);\n }\n\n getSessionsRoot(): string {\n return this.sessionsDir;\n }\n\n /**\n * OpenClaw-aligned: resolve the sessions directory for a given session key.\n * Extracts agentId from the session key and routes to `agents/<agentId>/sessions/`.\n * Falls back to the default sessions directory when agentId cannot be parsed\n * or when `sessionsDir` was explicitly provided in options.\n */\n private resolveSessionsDirForKey(sessionKey: string): string {\n if (this.options.sessionsDir) {\n return this.sessionsDir;\n }\n const parsed = parseRoutingSessionKey(sessionKey);\n if (!parsed) {\n return this.sessionsDir;\n }\n const agentId = parsed.agentId;\n const cached = this.agentSessionsDirCache.get(agentId);\n if (cached) {\n return cached;\n }\n const resolved = resolveSessionsDir(this.options.config, agentId);\n this.agentSessionsDirCache.set(agentId, resolved);\n return resolved;\n }\n\n private resolveStorePathForKey(sessionKey: string): string {\n return join(this.resolveSessionsDirForKey(sessionKey), FILENAMES.SESSIONS_MAP);\n }\n\n private async runStoreMutation<T>(fn: () => Promise<T>): Promise<T> {\n if (this.storeMutationDepth > 0) {\n return fn();\n }\n const run = this.storeMutationChain.then(async () => {\n this.storeMutationDepth++;\n try {\n return await fn();\n } finally {\n this.storeMutationDepth--;\n }\n });\n this.storeMutationChain = run.then(() => undefined).catch(() => undefined);\n return run as Promise<T>;\n }\n\n async initialize(): Promise<void> {\n await mkdir(this.sessionsDir, { recursive: true });\n await mkdir(this.archiveDir, { recursive: true });\n if (!existsSync(this.storePath)) {\n await withSessionsJsonLock(this.storePath, async () => undefined);\n }\n log.debug('Session store initialized (sessions.json + JSONL)');\n }\n\n private transcriptPathForEntry(entry: XopcSessionDiskEntry, sessionsDir?: string): string {\n return resolveSessionFilePath(entry.sessionId, entry, { sessionsDir: sessionsDir ?? this.sessionsDir });\n }\n\n private async readMapForKey(sessionKey: string): Promise<Record<string, XopcSessionDiskEntry>> {\n const storePath = this.resolveStorePathForKey(sessionKey);\n return readSessionsJsonFile<XopcSessionDiskEntry>(storePath);\n }\n\n private async readMap(): Promise<Record<string, XopcSessionDiskEntry>> {\n return readSessionsJsonFile<XopcSessionDiskEntry>(this.storePath);\n }\n\n /**\n * OpenClaw-aligned: read sessions.json from ALL known agents and merge into a single map.\n * Used by aggregation queries (list, getByAgent, getByAccount, etc.) so the gateway UI\n * can display sessions across all agents.\n */\n private async readAllMaps(): Promise<Record<string, XopcSessionDiskEntry>> {\n if (this.options.sessionsDir) {\n return this.readMap();\n }\n const agents = listAgentEntries(this.options.config);\n const defaultId = resolveDefaultAgentId(this.options.config);\n const agentIds = new Set<string>([defaultId, ...agents.map((a) => a.id)]);\n\n const merged: Record<string, XopcSessionDiskEntry> = {};\n for (const id of agentIds) {\n const dir = resolveSessionsDir(this.options.config, id);\n const path = join(dir, FILENAMES.SESSIONS_MAP);\n if (!existsSync(path)) {\n continue;\n }\n const map = await readSessionsJsonFile<XopcSessionDiskEntry>(path);\n Object.assign(merged, map);\n }\n return merged;\n }\n\n private async getDiskEntry(sessionKey: string): Promise<XopcSessionDiskEntry | undefined> {\n const map = await this.readMapForKey(sessionKey);\n return map[sessionKey];\n }\n\n private buildDefaultMetadata(key: string): SessionMetadata {\n const { channel, chatId } = this.parseSessionKey(key);\n const routing = this.extractRoutingFromKey(key);\n const isCronSession = channel === 'cron';\n const isHeartbeatSession = channel === 'heartbeat';\n const now = new Date().toISOString();\n return {\n key,\n status: SessionStatus.ACTIVE,\n tags: [],\n createdAt: now,\n updatedAt: now,\n lastAccessedAt: now,\n messageCount: 0,\n estimatedTokens: 0,\n compactedCount: 0,\n sourceChannel: channel,\n sourceChatId: chatId,\n routing,\n ...(isCronSession\n ? { sessionType: 'cron', customData: { cronJobId: chatId } }\n : {}),\n ...(isHeartbeatSession\n ? { sessionType: 'heartbeat', customData: { heartbeatTarget: chatId } }\n : {}),\n stats: { messageCount: 0, tokenCount: 0 },\n };\n }\n\n private parseSessionKey(key: string): { channel: string; chatId: string } {\n const parts = key.split(':');\n if (parts.length >= 2 && parts[0] === 'heartbeat') {\n return { channel: 'heartbeat', chatId: parts.slice(1).join(':') };\n }\n const parsed = parseRoutingSessionKey(key);\n if (parsed) {\n if (parsed.source === 'cron') {\n return { channel: 'cron', chatId: parsed.peerId };\n }\n return {\n channel: parsed.source,\n chatId: [parsed.accountId, parsed.peerKind, parsed.peerId].join(':'),\n };\n }\n return { channel: 'unknown', chatId: key };\n }\n\n private extractRoutingFromKey(key: string): SessionMetadata['routing'] {\n const parsed = parseRoutingSessionKey(key);\n if (!parsed) {\n return undefined;\n }\n return {\n agentId: parsed.agentId?.toLowerCase() || 'main',\n source: parsed.source?.toLowerCase() || 'unknown',\n accountId: parsed.accountId?.toLowerCase() || 'default',\n peerKind: parsed.peerKind?.toLowerCase() || 'dm',\n peerId: parsed.peerId?.toLowerCase() || 'unknown',\n threadId: parsed.threadId,\n scopeId: parsed.scopeId,\n };\n }\n\n /** Resolve on-disk transcript path; creates session row + empty JSONL when missing. */\n async resolveTranscriptPath(\n sessionKey: string,\n ): Promise<{ sessionId: string; absPath: string; sessionsDir: string }> {\n const entry = await this.ensureSession(sessionKey);\n const sessionsDir = this.resolveSessionsDirForKey(sessionKey);\n const absPath = this.transcriptPathForEntry(entry, sessionsDir);\n return { sessionId: entry.sessionId, absPath, sessionsDir };\n }\n\n /** Ensure sessions.json has an entry and transcript file exist for `sessionKey`. */\n private async ensureSession(sessionKey: string): Promise<XopcSessionDiskEntry> {\n const keyStorePath = this.resolveStorePathForKey(sessionKey);\n const keySessionsDir = this.resolveSessionsDirForKey(sessionKey);\n await mkdir(keySessionsDir, { recursive: true });\n return withSessionsJsonLock(keyStorePath, async (map) => {\n const existing = map[sessionKey] as XopcSessionDiskEntry | undefined;\n if (existing?.pluginExtensions?.xopc?.metadata) {\n return existing;\n }\n let entry = existing;\n if (!entry) {\n const sessionId = randomUUID();\n validateSessionId(sessionId);\n const sessionFile = `${sessionId}.jsonl`;\n const now = Date.now();\n const metadata = this.buildDefaultMetadata(sessionKey);\n metadata.transcriptId = sessionId;\n entry = {\n sessionId,\n updatedAt: now,\n sessionStartedAt: now,\n sessionFile,\n pluginExtensions: { xopc: { metadata } },\n };\n map[sessionKey] = entry as Record<string, unknown>;\n const abs = resolveSessionTranscriptPathInDir(sessionId, keySessionsDir);\n await writeTranscriptJsonl({\n absPath: abs,\n sessionId,\n cwd: process.cwd(),\n rows: [],\n });\n } else if (!entry.pluginExtensions?.xopc?.metadata) {\n const metadata = this.buildDefaultMetadata(sessionKey);\n metadata.transcriptId = entry.sessionId;\n entry.pluginExtensions = { xopc: { metadata } };\n map[sessionKey] = entry as Record<string, unknown>;\n }\n return entry!;\n });\n }\n\n private metadataFromEntry(sessionKey: string, entry: XopcSessionDiskEntry): SessionMetadata {\n const base = entry.pluginExtensions?.xopc?.metadata ?? this.buildDefaultMetadata(sessionKey);\n const { channel: keySource, chatId: keyChatId } = this.parseSessionKey(sessionKey);\n const diskSc = typeof base.sourceChannel === 'string' ? base.sourceChannel.trim() : '';\n const diskChat = typeof base.sourceChatId === 'string' ? base.sourceChatId.trim() : '';\n return {\n ...base,\n key: sessionKey,\n transcriptId: entry.sessionId,\n sourceChannel: diskSc || keySource,\n sourceChatId: diskChat || keyChatId,\n };\n }\n\n async getByAgent(agentId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.agentId?.toLowerCase() === agentId.toLowerCase()) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getByAccount(accountId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.accountId === accountId) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getByPeer(peerKind: string, peerId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.peerKind === peerKind && m.routing.peerId === peerId) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getMainSession(channel: string, accountId: string): Promise<SessionMetadata | null> {\n const map = await this.readAllMaps();\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (\n m.routing?.source === channel &&\n m.routing.accountId === accountId &&\n m.routing.peerKind === 'dm' &&\n m.routing.peerId === 'main'\n ) {\n return m;\n }\n }\n return null;\n }\n\n async refreshIndex(): Promise<void> {\n /* no-op: sessions.json is authoritative */\n }\n\n async list(query: SessionListQuery = {}): Promise<PaginatedResult<SessionMetadata>> {\n const map = await this.readAllMaps();\n let sessions = Object.entries(map).map(([k, e]) => this.metadataFromEntry(k, e));\n\n if (query.status) {\n const statuses = Array.isArray(query.status) ? query.status : [query.status];\n sessions = sessions.filter((s) => statuses.includes(s.status));\n }\n if (query.channel) {\n const rawChannels = query.channel\n .split(',')\n .map((c) => c.trim().toLowerCase())\n .filter(Boolean);\n /**\n * `ui` is a legacy console source; treat as webchat when filtering web sessions.\n * `webui` matches slash-command normalization to `gateway` (see `chat-commands/session-key.ts`).\n */\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 sessions = [];\n } else if (channels.length === 1) {\n const ch = channels[0]!;\n sessions = sessions.filter((s) => (s.sourceChannel ?? '').toLowerCase() === ch);\n } else {\n sessions = sessions.filter((s) => channels.includes((s.sourceChannel ?? '').toLowerCase()));\n }\n }\n if (query.tags?.length) {\n sessions = sessions.filter((s) => query.tags!.some((t) => s.tags.includes(t)));\n }\n if (query.search) {\n const q = query.search.toLowerCase();\n sessions = sessions.filter(\n (s) =>\n s.key.toLowerCase().includes(q) ||\n (s.name?.toLowerCase().includes(q) ?? false) ||\n s.tags.some((t) => t.toLowerCase().includes(q)),\n );\n }\n\n const sortBy = query.sortBy || 'updatedAt';\n const sortOrder = query.sortOrder || 'desc';\n sessions.sort((a, b) => {\n const av = a[sortBy];\n const bv = b[sortBy];\n const c = av < bv ? -1 : av > bv ? 1 : 0;\n return sortOrder === 'asc' ? c : -c;\n });\n\n const total = sessions.length;\n const limit = query.limit || 50;\n const offset = query.offset || 0;\n const items = sessions.slice(offset, offset + limit);\n return { items, total, limit, offset, hasMore: offset + limit < total };\n }\n\n async get(\n key: string,\n options?: { includeTranscriptSummary?: boolean; includeTranscriptRows?: boolean },\n ): Promise<SessionDetail | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return null;\n }\n const messages = await this.loadDisplayMessages(key);\n const detail = await this.buildSessionDetail(key, metadata, messages, options);\n return detail;\n }\n\n async getMessagePage(\n key: string,\n options: {\n offset?: number;\n limit?: number;\n before?: string;\n includeTranscriptSummary?: boolean;\n includeTranscriptRows?: boolean;\n } = {},\n ): Promise<{\n session: SessionDetail;\n pagination: {\n total: number;\n limit: number;\n offset: number;\n hasMore: boolean;\n before?: string;\n nextBeforeCursor?: string;\n };\n } | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return null;\n }\n\n const limit = Math.min(200, Math.max(1, Math.trunc(options.limit ?? 50)));\n const offset = Math.max(0, Math.trunc(options.offset ?? 0));\n const parsedBefore = options.before ? Number.parseInt(options.before, 10) : undefined;\n const hasBeforeCursor = parsedBefore !== undefined && Number.isFinite(parsedBefore);\n\n const checkpoints = await this.listCompactionCheckpoints(key);\n if (checkpoints.length === 0) {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n const page = await readDisplayMessagePageFromTranscriptFile(primary, {\n limit,\n offset: hasBeforeCursor ? undefined : offset,\n beforeIndex: hasBeforeCursor ? parsedBefore : undefined,\n });\n const session = await this.buildSessionDetail(key, metadata, page.messages, options);\n const nextBeforeCursor = page.startIndex > 0 ? String(page.startIndex) : undefined;\n\n return {\n session,\n pagination: {\n total: page.total,\n limit,\n offset,\n hasMore: page.startIndex > 0,\n ...(hasBeforeCursor ? { before: String(page.endIndex) } : {}),\n ...(nextBeforeCursor ? { nextBeforeCursor } : {}),\n },\n };\n }\n\n const messages = await this.loadDisplayMessages(key);\n const total = messages.length;\n const endExclusive = hasBeforeCursor\n ? Math.min(total, Math.max(0, Math.trunc(parsedBefore!)))\n : Math.max(0, total - offset);\n const startInclusive = Math.max(0, endExclusive - limit);\n const pageMessages = messages.slice(startInclusive, endExclusive);\n const session = await this.buildSessionDetail(key, metadata, pageMessages, options);\n const nextBeforeCursor = startInclusive > 0 ? String(startInclusive) : undefined;\n\n return {\n session,\n pagination: {\n total,\n limit,\n offset,\n hasMore: startInclusive > 0,\n ...(hasBeforeCursor ? { before: String(endExclusive) } : {}),\n ...(nextBeforeCursor ? { nextBeforeCursor } : {}),\n },\n };\n }\n\n private async buildSessionDetail(\n key: string,\n metadata: SessionMetadata,\n messages: AgentMessage[],\n options?: { includeTranscriptSummary?: boolean; includeTranscriptRows?: boolean },\n ): Promise<SessionDetail> {\n let transcriptSummary: SessionTranscriptSummary | undefined;\n if (options?.includeTranscriptSummary) {\n const env = await this.loadTranscriptDocument(key);\n if (env) {\n transcriptSummary = {\n id: env.id,\n version: env.version,\n createdAt: env.createdAt,\n updatedAt: env.updatedAt,\n compactionCount: env.compactions?.length ?? 0,\n };\n }\n }\n let transcriptRows: TranscriptStoredRow[] | undefined;\n if (options?.includeTranscriptRows) {\n transcriptRows = await this.loadTranscriptRows(key);\n }\n return {\n ...metadata,\n messages: this.convertMessages(messages),\n ...(transcriptSummary ? { transcriptSummary } : {}),\n ...(transcriptRows !== undefined ? { transcriptRows } : {}),\n };\n }\n\n async loadTranscriptRows(key: string): Promise<TranscriptStoredRow[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const path = this.transcriptPathForEntry(entry, this.resolveSessionsDirForKey(key));\n return readTranscriptRowsFromFile(path);\n }\n\n async getMetadata(key: string): Promise<SessionMetadata | null> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n return this.metadataFromEntry(key, entry);\n }\n\n async updateMetadata(key: string, updates: Partial<SessionMetadata>): Promise<void> {\n return this.runStoreMutation(async () => {\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const entry = map[key] as XopcSessionDiskEntry | undefined;\n if (!entry?.pluginExtensions?.xopc?.metadata) {\n throw new Error(`Session not found: ${key}`);\n }\n const meta = { ...entry.pluginExtensions.xopc.metadata, ...updates, updatedAt: new Date().toISOString() };\n entry.pluginExtensions.xopc.metadata = meta;\n entry.updatedAt = Date.now();\n map[key] = entry as Record<string, unknown>;\n });\n invalidateSessionSearchIndexCache();\n log.debug({ key, updates }, 'Session metadata updated');\n });\n }\n\n /**\n * Reset transcript for an existing session key: archive the current JSONL as\n * `*.reset.*`, assign a new `sessionId`, and preserve per-session overrides\n * on the disk entry (thinking/verbose) and in `sessions/config/*.json`.\n */\n async reset(key: string): Promise<{ sessionId: string; previousSessionId: string } | null> {\n return this.runStoreMutation(async () => {\n const existing = await this.getDiskEntry(key);\n if (!existing?.pluginExtensions?.xopc?.metadata) {\n return null;\n }\n\n const previousSessionId = existing.sessionId;\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(existing, keySessionsDir);\n if (existsSync(abs)) {\n try {\n archiveFileOnDisk(abs, 'reset');\n } catch (err) {\n log.warn({ err, key }, 'Transcript archive on reset failed');\n }\n }\n\n const sessionId = randomUUID();\n validateSessionId(sessionId);\n const now = Date.now();\n const nextAbs = resolveSessionTranscriptPathInDir(sessionId, keySessionsDir);\n await writeTranscriptJsonl({\n absPath: nextAbs,\n sessionId,\n cwd: process.cwd(),\n rows: [],\n });\n\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n e.sessionId = sessionId;\n e.sessionFile = `${sessionId}.jsonl`;\n e.updatedAt = now;\n e.sessionStartedAt = now;\n e.lastInteractionAt = undefined;\n patchSessionsJsonEntryStats(e, buildSessionsJsonStatsPatch(0, 0));\n const meta = e.pluginExtensions.xopc.metadata;\n meta.transcriptId = sessionId;\n meta.updatedAt = new Date(now).toISOString();\n map[key] = e as Record<string, unknown>;\n });\n\n invalidateSessionSearchIndexCache();\n log.info({ key, previousSessionId, sessionId }, 'Session reset');\n return { sessionId, previousSessionId };\n });\n }\n\n async delete(key: string): Promise<boolean> {\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return false;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n delete map[key];\n });\n try {\n if (existsSync(abs)) {\n archiveFileOnDisk(abs, 'deleted');\n }\n } catch (err) {\n log.warn({ err, key }, 'Transcript archive on delete failed');\n }\n invalidateSessionSearchIndexCache();\n log.info({ key }, 'Session deleted');\n return true;\n });\n }\n\n async deleteMany(keys: string[]): Promise<{ success: string[]; failed: string[] }> {\n const success: string[] = [];\n const failed: string[] = [];\n for (const key of keys) {\n try {\n if (await this.delete(key)) {\n success.push(key);\n } else {\n failed.push(key);\n }\n } catch {\n failed.push(key);\n }\n }\n return { success, failed };\n }\n\n async setStatus(key: string, status: SessionStatus): Promise<void> {\n await this.updateMetadata(key, { status });\n if (status === SessionStatus.ARCHIVED) {\n await this.moveToArchive(key);\n } else {\n await this.moveFromArchive(key);\n }\n }\n\n async archive(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ARCHIVED);\n }\n\n async unarchive(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ACTIVE);\n }\n\n async pin(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.PINNED);\n }\n\n async unpin(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ACTIVE);\n }\n\n private async moveToArchive(key: string): Promise<void> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n if (existsSync(abs)) {\n try {\n archiveFileOnDisk(abs, 'deleted');\n } catch (err) {\n log.warn({ err, key }, 'Archive transcript rename failed');\n }\n }\n }\n\n private async findMostRecentDeletedTranscript(sessionId: string, sessionsDir: string): Promise<string | null> {\n let names: string[];\n try {\n names = await readdir(sessionsDir);\n } catch {\n return null;\n }\n const prefix = `${sessionId}${DELETED_MARKER}`;\n const hits = names.filter((n) => n.startsWith(prefix) && n.endsWith('Z'));\n hits.sort().reverse();\n const first = hits[0];\n return first ? join(sessionsDir, first) : null;\n }\n\n private async moveFromArchive(key: string): Promise<void> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const target = resolveSessionTranscriptPathInDir(entry.sessionId, keySessionsDir);\n if (existsSync(target)) {\n return;\n }\n const src = await this.findMostRecentDeletedTranscript(entry.sessionId, keySessionsDir);\n if (!src) {\n return;\n }\n try {\n const { rename } = await import('fs/promises');\n await rename(src, target);\n } catch (err) {\n log.warn({ err, key, src, target }, 'Unarchive transcript rename failed');\n }\n }\n\n async loadMessages(key: string, options?: { fromArchive?: boolean }): Promise<AgentMessage[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n if (existsSync(primary)) {\n const rows = await readTranscriptRowsFromFile(primary);\n return rowsToLlmMessages(rows);\n }\n if (options?.fromArchive) {\n const archived = await this.findMostRecentDeletedTranscript(entry.sessionId, keySessionsDir);\n if (!archived) {\n return [];\n }\n const rows = await readTranscriptRowsFromFile(archived);\n return rowsToLlmMessages(rows);\n }\n return [];\n }\n\n private async loadDisplayMessages(key: string): Promise<AgentMessage[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n const transcriptPaths: string[] = [];\n const checkpoints = await this.listCompactionCheckpoints(key);\n for (const checkpoint of [...checkpoints].reverse()) {\n transcriptPaths.push(join(keySessionsDir, `${this.checkpointBasename(entry.sessionId)}${checkpoint.id}.jsonl`));\n }\n transcriptPaths.push(primary);\n\n const messages: AgentMessage[] = [];\n const seenMessages = new Set<string>();\n for (const transcriptPath of transcriptPaths) {\n if (!existsSync(transcriptPath)) {\n continue;\n }\n const rows = await readTranscriptRowsFromFile(transcriptPath);\n for (const message of rowsToLlmMessages(rows)) {\n if (this.isCompactionSummaryMessage(message)) {\n continue;\n }\n const key = this.displayMessageIdentity(message);\n if (seenMessages.has(key)) {\n continue;\n }\n seenMessages.add(key);\n messages.push(message);\n }\n }\n return messages;\n }\n\n async loadTranscriptDocument(key: string): Promise<XopcSessionTranscriptV1 | null> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const path = this.transcriptPathForEntry(entry, this.resolveSessionsDirForKey(key));\n if (!existsSync(path)) {\n return null;\n }\n const entries = loadEntriesFromFile(path);\n const header = entries.find((e) => e.type === 'session');\n if (!header || typeof (header as { id?: unknown }).id !== 'string') {\n return null;\n }\n const sessionHeader = header as { type: 'session'; id: string; timestamp?: string };\n const rows = await readTranscriptRowsFromFile(path);\n const compactions = entries\n .filter((e): e is CompactionEntry => e.type === 'compaction')\n .map((c) => ({\n at: c.timestamp,\n summary: c.summary,\n firstKeptIndex: Number.parseInt(String(c.firstKeptEntryId), 10) || 0,\n tokensBefore: c.tokensBefore,\n tokensAfter:\n typeof c.details === 'object' &&\n c.details &&\n 'tokensAfter' in c.details &&\n typeof (c.details as { tokensAfter?: unknown }).tokensAfter === 'number'\n ? (c.details as { tokensAfter: number }).tokensAfter\n : 0,\n }));\n return {\n type: 'xopc_session_transcript',\n version: 1,\n id: sessionHeader.id,\n createdAt: sessionHeader.timestamp ?? new Date().toISOString(),\n updatedAt: new Date().toISOString(),\n messages: rows,\n ...(compactions.length > 0 ? { compactions } : {}),\n };\n }\n\n private async writeTranscriptAndSyncIndex(\n key: string,\n rows: TranscriptStoredRow[],\n opts?: { appendCompaction?: TranscriptCompactionRecord },\n ): Promise<void> {\n const entry = await this.ensureSession(key);\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const llm = rowsToLlmMessages(rows);\n await persistMergedTranscriptRows({\n absPath: abs,\n sessionId: entry.sessionId,\n cwd: process.cwd(),\n rows,\n appendCompaction: opts?.appendCompaction,\n });\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(llm.length, this.estimateTokens(llm)),\n );\n map[key] = e as Record<string, unknown>;\n });\n invalidateSessionSearchIndexCache();\n }\n\n /** Incremental sessions.json stats after guard append (OpenClaw transcript-events). */\n async syncSessionsJsonFromTranscriptUpdate(update: SessionTranscriptUpdate): Promise<void> {\n const sessionKey = update.sessionKey?.trim();\n if (!sessionKey || !existsSync(update.sessionFile)) {\n return;\n }\n return this.runStoreMutation(async () => {\n const keyStorePath = this.resolveStorePathForKey(sessionKey);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[sessionKey] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n\n if (update.message && isAppendOnlyLlmTranscriptMessage(update.message)) {\n incrementSessionsJsonStatsForAppend(e);\n map[sessionKey] = e as Record<string, unknown>;\n return;\n }\n\n const messageCount = await countTranscriptMessageRows(update.sessionFile);\n const rows = await readTranscriptRowsFromFile(update.sessionFile);\n const llm = rowsToLlmMessages(rows);\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(messageCount, this.estimateTokens(llm)),\n );\n map[sessionKey] = e as Record<string, unknown>;\n });\n invalidateSessionSearchIndexCache();\n });\n }\n\n async appendTranscriptContextEntry(\n key: string,\n entry: Omit<XopcTranscriptContextEntry, 'kind'> & Partial<Pick<XopcTranscriptContextEntry, 'kind'>>,\n ): Promise<void> {\n return this.runStoreMutation(async () => {\n await this.ensureSession(key);\n const disk = await this.getDiskEntry(key);\n if (!disk) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const absPath = this.transcriptPathForEntry(disk, keySessionsDir);\n const row: XopcTranscriptContextEntry = {\n kind: 'context',\n id: typeof entry.id === 'string' ? entry.id : undefined,\n text: typeof entry.text === 'string' ? entry.text : undefined,\n data: entry.data,\n createdAt: entry.createdAt ?? new Date().toISOString(),\n };\n await appendPiTranscriptContextEntry({\n absPath,\n cwd: process.cwd(),\n entry: row,\n sessionKey: key,\n });\n const rows = existsSync(absPath) ? await readTranscriptRowsFromFile(absPath) : [];\n const llm = rowsToLlmMessages(rows);\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(llm.length, this.estimateTokens(llm)),\n );\n map[key] = e as Record<string, unknown>;\n });\n invalidateSessionSearchIndexCache();\n });\n }\n\n /**\n * Bulk write entry point used by compaction, tests, and admin tools.\n * Runtime agent turns must persist via {@link guardSessionManager} + appendMessage.\n */\n async saveMessages(key: string, messages: AgentMessage[]): Promise<void> {\n return this.runStoreMutation(async () => {\n await this.ensureSession(key);\n const prev = await this.loadTranscriptRows(key);\n const merged = mergeLlmMessagesPreservingContextRows(prev, messages);\n await this.writeTranscriptAndSyncIndex(key, merged);\n });\n }\n\n getWindowStats(messages: AgentMessage[]) {\n return this.window.getStats(messages);\n }\n\n needsCompaction(key: string, messages: AgentMessage[], contextWindow: number) {\n return this.compactor.needsCompaction(messages, contextWindow);\n }\n\n prepareCompaction(\n key: string,\n messages: AgentMessage[],\n contextWindow: number,\n ): { needsCompaction: boolean; messages: AgentMessage[]; stats?: ReturnType<typeof this.compactor.needsCompaction> } {\n const result = this.compactor.needsCompaction(messages, contextWindow);\n return { needsCompaction: result.needed, messages, stats: result };\n }\n\n private checkpointBasename(sessionId: string): string {\n return `${sessionId}.checkpoint.`;\n }\n\n private async pruneCompactionCheckpoints(sessionId: string, sessionsDir: string): Promise<void> {\n const MAX = 15;\n const prefix = this.checkpointBasename(sessionId);\n let names: string[];\n try {\n names = await readdir(sessionsDir);\n } catch {\n return;\n }\n const candidates = names.filter((n) => n.startsWith(prefix) && n.endsWith('.jsonl'));\n if (candidates.length <= MAX) {\n return;\n }\n const stats = await Promise.all(\n candidates.map(async (name) => {\n const p = join(sessionsDir, name);\n try {\n const s = await stat(p);\n return { p, mtimeMs: s.mtimeMs };\n } catch {\n return { p: join(sessionsDir, name), mtimeMs: 0 };\n }\n }),\n );\n stats.sort((a, b) => a.mtimeMs - b.mtimeMs);\n for (let i = 0; i < stats.length - MAX; i++) {\n try {\n await unlink(stats[i]!.p);\n } catch {\n /* ignore */\n }\n }\n }\n\n private async captureCompactionCheckpoint(sessionId: string, transcriptAbs: string, sessionsDir: string): Promise<void> {\n if (!existsSync(transcriptAbs)) {\n return;\n }\n const id = randomUUID();\n const dest = join(sessionsDir, `${sessionId}.checkpoint.${id}.jsonl`);\n try {\n await copyFile(transcriptAbs, dest);\n await this.pruneCompactionCheckpoints(sessionId, sessionsDir);\n } catch (err) {\n log.warn({ err, sessionId }, 'Compaction checkpoint copy failed');\n }\n }\n\n async applyCompaction(\n key: string,\n messages: AgentMessage[],\n result: CompactionResult,\n ): Promise<AgentMessage[]> {\n const compacted = this.compactor.applyCompaction(messages, result);\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return compacted;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n await this.captureCompactionCheckpoint(entry.sessionId, abs, keySessionsDir);\n const prev = await this.loadTranscriptRows(key);\n const merged = mergeLlmMessagesPreservingContextRows(prev, compacted);\n await this.writeTranscriptAndSyncIndex(key, merged, {\n appendCompaction: {\n at: new Date().toISOString(),\n summary: result.summary,\n firstKeptIndex: result.firstKeptIndex,\n tokensBefore: result.tokensBefore,\n tokensAfter: result.tokensAfter,\n },\n });\n const metadata = await this.getMetadata(key);\n if (metadata) {\n await this.updateMetadata(key, { compactedCount: metadata.compactedCount + 1 });\n }\n log.info(\n { key, tokensBefore: result.tokensBefore, tokensAfter: result.tokensAfter, keptMessages: compacted.length },\n 'Session compacted',\n );\n await this.injectPostCompactionContext(key);\n return compacted;\n });\n }\n\n async compact(\n key: string,\n messages: AgentMessage[],\n contextWindow: number,\n instructions?: string,\n force?: boolean,\n ): Promise<CompactionResult> {\n const result = await this.compactor.compact(messages, instructions, force);\n if (result.compacted) {\n await this.applyCompaction(key, messages, result);\n }\n return result;\n }\n\n async getCompactionStats(key: string) {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return undefined;\n }\n return {\n compactionCount: metadata.compactedCount,\n totalTokensBefore: 0,\n totalTokensAfter: 0,\n lastCompactionAt: undefined,\n };\n }\n\n async listCompactionCheckpoints(key: string): Promise<CompactionCheckpointSummary[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const sessionId = entry.sessionId;\n const prefix = this.checkpointBasename(sessionId);\n let names: string[];\n try {\n names = await readdir(keySessionsDir);\n } catch {\n return [];\n }\n const files = names.filter((n) => n.startsWith(prefix) && n.endsWith('.jsonl'));\n const rows = await Promise.all(\n files.map(async (name) => {\n const p = join(keySessionsDir, name);\n const parsed = parseCompactionCheckpointTranscriptFileName(name);\n const id = parsed?.checkpointId;\n if (!id || !normalizeCompactionCheckpointId(id)) {\n return null;\n }\n try {\n const s = await stat(p);\n return {\n id: normalizeCompactionCheckpointId(id)!,\n sizeBytes: s.size,\n modifiedAt: new Date(s.mtimeMs).toISOString(),\n } satisfies CompactionCheckpointSummary;\n } catch {\n return null;\n }\n }),\n );\n const valid = rows.filter((r): r is CompactionCheckpointSummary => r !== null);\n valid.sort((a, b) => b.modifiedAt.localeCompare(a.modifiedAt));\n return valid;\n }\n\n async getCompactionCheckpointDetail(\n key: string,\n checkpointId: string,\n ): Promise<CompactionCheckpointDetail | null> {\n const id = normalizeCompactionCheckpointId(checkpointId);\n if (!id) {\n return null;\n }\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const fname = `${this.checkpointBasename(entry.sessionId)}${id}.jsonl`;\n const cpPath = join(keySessionsDir, fname);\n if (!existsSync(cpPath)) {\n return null;\n }\n try {\n const rows = await readTranscriptRowsFromFile(cpPath);\n const llm = rowsToLlmMessages(rows);\n const s = await stat(cpPath);\n return {\n id,\n sizeBytes: s.size,\n modifiedAt: new Date(s.mtimeMs).toISOString(),\n messageCount: llm.length,\n };\n } catch {\n return null;\n }\n }\n\n async restoreCompactionCheckpoint(key: string, checkpointId: string): Promise<void> {\n const id = normalizeCompactionCheckpointId(checkpointId);\n if (!id) {\n throw new Error('Invalid checkpoint id');\n }\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n throw new Error(`Session not found: ${key}`);\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const cpPath = join(keySessionsDir, `${this.checkpointBasename(entry.sessionId)}${id}.jsonl`);\n if (!existsSync(cpPath)) {\n throw new Error(`Checkpoint not found: ${id}`);\n }\n const target = this.transcriptPathForEntry(entry, keySessionsDir);\n await copyFile(cpPath, target);\n const messages = await this.loadMessages(key);\n await this.saveMessages(key, messages);\n log.info({ key, checkpointId: id }, 'Session transcript restored from compaction checkpoint');\n });\n }\n\n async deleteSession(key: string): Promise<boolean> {\n return this.delete(key);\n }\n\n async load(key: string, options?: { fromArchive?: boolean }): Promise<AgentMessage[]> {\n return this.loadMessages(key, options);\n }\n\n async estimateTokenUsage(_key: string, messages: AgentMessage[]): Promise<number> {\n return this.estimateTokens(messages);\n }\n\n async searchInSession(key: string, keyword: string): Promise<Message[]> {\n const messages = await this.loadDisplayMessages(key);\n const keywordLower = keyword.toLowerCase();\n return this.convertMessages(\n messages.filter((m) => {\n const content = this.extractTextContent(this.messageContent(m));\n return content.toLowerCase().includes(keywordLower);\n }),\n );\n }\n\n async exportSession(key: string, format: ExportFormat): Promise<string> {\n const detail = await this.get(key);\n if (!detail) {\n throw new Error(`Session not found: ${key}`);\n }\n if (format === 'json') {\n const transcriptRows = await this.loadTranscriptRows(key);\n const exportData: SessionExport = {\n version: INDEX_VERSION,\n exportedAt: new Date().toISOString(),\n metadata: detail,\n messages: detail.messages,\n transcriptRows,\n };\n return JSON.stringify(exportData, null, 2);\n }\n const lines = [\n `# ${detail.name || detail.key}`,\n '',\n `- **Channel:** ${detail.sourceChannel}`,\n `- **Created:** ${detail.createdAt}`,\n `- **Messages:** ${detail.messageCount}`,\n `- **Tags:** ${detail.tags.join(', ') || 'none'}`,\n '',\n '---',\n '',\n ];\n for (const msg of detail.messages) {\n const role = msg.role === 'assistant' ? 'Assistant' : msg.role === 'user' ? 'User' : msg.role;\n lines.push(`## ${role}`, '');\n const body = typeof msg.content === 'string' ? msg.content : JSON.stringify(msg.content, null, 2);\n lines.push(body, '', '---', '');\n }\n return lines.join('\\n');\n }\n\n async getStats(): Promise<GlobalSessionStats> {\n const list = await this.list({ limit: 100000 });\n const sessions = list.items;\n const byChannel: Record<string, number> = {};\n for (const s of sessions) {\n byChannel[s.sourceChannel] = (byChannel[s.sourceChannel] || 0) + 1;\n }\n let oldestSession: string | undefined;\n let newestSession: string | undefined;\n if (sessions.length > 0) {\n const sorted = [...sessions].sort((a, b) => a.createdAt.localeCompare(b.createdAt));\n oldestSession = sorted[0]!.createdAt;\n newestSession = sorted[sorted.length - 1]!.createdAt;\n }\n return {\n totalSessions: sessions.length,\n activeSessions: sessions.filter((s) => s.status === SessionStatus.ACTIVE || s.status === SessionStatus.IDLE).length,\n archivedSessions: sessions.filter((s) => s.status === SessionStatus.ARCHIVED).length,\n pinnedSessions: sessions.filter((s) => s.status === SessionStatus.PINNED).length,\n totalMessages: sessions.reduce((sum, s) => sum + s.messageCount, 0),\n totalTokens: sessions.reduce((sum, s) => sum + s.estimatedTokens, 0),\n oldestSession,\n newestSession,\n byChannel,\n };\n }\n\n async archiveOld(olderThanDays: number): Promise<number> {\n const cutoff = new Date();\n cutoff.setDate(cutoff.getDate() - olderThanDays);\n const list = await this.list({ limit: 100000 });\n let archived = 0;\n for (const session of list.items) {\n if (session.status !== SessionStatus.ARCHIVED && session.status !== SessionStatus.PINNED) {\n const lastAccess = new Date(session.lastAccessedAt);\n if (lastAccess < cutoff) {\n await this.archive(session.key);\n archived++;\n }\n }\n }\n return archived;\n }\n\n estimateTokens(messages: AgentMessage[]): number {\n let total = 0;\n for (const msg of messages) {\n total += Math.ceil(this.extractTextContent(this.messageContent(msg)).length / 4);\n }\n return total;\n }\n\n private messageContent(msg: AgentMessage): unknown {\n return (msg as { content?: unknown }).content;\n }\n\n private isCompactionSummaryMessage(msg: AgentMessage): boolean {\n if (msg.role !== 'user') {\n return false;\n }\n const text = this.extractTextContent(this.messageContent(msg)).trim();\n return /^\\[Previous conversation summary\\]/i.test(text);\n }\n\n private displayMessageIdentity(message: AgentMessage): string {\n const record = message as unknown as Record<string, unknown>;\n return JSON.stringify({\n role: message.role,\n timestamp: record.timestamp,\n toolCallId: record.toolCallId ?? record.tool_call_id,\n toolName: record.toolName,\n content: this.messageContent(message),\n });\n }\n\n private extractTextContent(content: unknown): string {\n if (typeof content === 'string') {\n return content;\n }\n if (Array.isArray(content)) {\n const parts: string[] = [];\n for (const item of content) {\n if (typeof item !== 'object' || item === null || !('type' in item)) {\n continue;\n }\n const c = item as { type?: string; text?: string; name?: string };\n if (c.type === 'text' && typeof c.text === 'string') {\n parts.push(c.text);\n } else if (c.type === 'toolCall' || c.type === 'tool_use') {\n parts.push(c.name ? `[${c.name}]` : '');\n }\n }\n return parts.join('');\n }\n return '';\n }\n\n private async injectPostCompactionContext(key: string): Promise<void> {\n const contextText = readPostCompactionContext({\n cfg: this.options.config,\n sessionKey: key,\n });\n if (!contextText?.trim()) {\n return;\n }\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const workspaceDir = resolveEffectiveAgentProfileForSession(this.options.config, key).resolvedWorkspacePath;\n try {\n await appendPiTranscriptContextEntry({\n absPath: abs,\n cwd: workspaceDir,\n sessionKey: key,\n entry: {\n kind: 'context',\n id: `post-compaction-${Date.now()}`,\n text: contextText,\n createdAt: new Date().toISOString(),\n },\n });\n } catch (err) {\n log.warn({ err, key }, 'Post-compaction context injection failed');\n }\n }\n\n private convertMessages(messages: AgentMessage[]): Message[] {\n return messages.map((m: AgentMessage & Record<string, unknown>) => {\n const c = this.messageContent(m);\n const content: string | unknown[] =\n typeof c === 'string' ? c : Array.isArray(c) ? c : this.extractTextContent(c);\n const row: Message = {\n role: m.role as Message['role'],\n content,\n timestamp: m.timestamp ? new Date(m.timestamp as string | number).toISOString() : undefined,\n tool_call_id: (m.tool_call_id as string | undefined) || (m.toolCallId as string | undefined),\n tool_calls: m.tool_calls as Message['tool_calls'],\n name: m.name as string | undefined,\n };\n if (Array.isArray(m.attachments) && m.attachments.length > 0) {\n row.attachments = m.attachments as Message['attachments'];\n }\n const rawUsage = m.usage as { input?: number; output?: number; totalTokens?: number; total?: number } | undefined;\n if (rawUsage && typeof rawUsage === 'object') {\n const inputTokens = typeof rawUsage.input === 'number' ? rawUsage.input : undefined;\n const outputTokens = typeof rawUsage.output === 'number' ? rawUsage.output : undefined;\n const totalTokens = typeof rawUsage.totalTokens === 'number'\n ? rawUsage.totalTokens\n : typeof rawUsage.total === 'number'\n ? rawUsage.total\n : undefined;\n if (inputTokens != null || outputTokens != null || totalTokens != null) {\n row.usage = { inputTokens, outputTokens, totalTokens };\n }\n }\n return row;\n });\n }\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"store.js","names":["parseRoutingSessionKey"],"sources":["../../../src/session/store.ts"],"sourcesContent":["import { randomUUID } from 'node:crypto';\nimport { performance } from 'node:perf_hooks';\nimport { copyFile, mkdir, readdir, stat, unlink } from 'fs/promises';\nimport { join } from 'path';\nimport { existsSync } from 'fs';\n\nimport type { AgentMessage } from '@earendil-works/pi-agent-core';\nimport type { CompactionEntry } from '@earendil-works/pi-coding-agent';\n\nimport { loadEntriesFromFile } from './parity/load-jsonl-entries.js';\n\nimport type { Config } from '../config/schema.js';\nimport { resolveSessionsDir, resolveStateDir, FILENAMES } from '../config/paths.js';\nimport { resolveDefaultAgentId, listAgentEntries } from '../agent/agent-scope.js';\nimport { resolveEffectiveAgentProfileForSession } from '../config/agent-profile.js';\nimport { readPostCompactionContext } from '../agent/reply/post-compaction-context.js';\nimport { parseSessionKey as parseRoutingSessionKey } from '../routing/session-key.js';\nimport { createLogger } from '../utils/logger.js';\nimport { SessionCompactor, type CompactionConfig, type CompactionResult } from '../agent/memory/compaction.js';\nimport { SlidingWindow, type WindowConfig } from '../agent/memory/window.js';\nimport { invalidateSessionSearchIndexCache } from './search-index-cache.js';\nimport type { TranscriptCompactionRecord, XopcSessionTranscriptV1 } from './transcript-format.js';\nimport {\n mergeLlmMessagesPreservingContextRows,\n type TranscriptStoredRow,\n type XopcTranscriptContextEntry,\n} from './session-context-for-llm.js';\nimport { normalizeCompactionCheckpointId } from './compaction-checkpoints.js';\nimport type {\n SessionMetadata,\n SessionDetail,\n SessionListQuery,\n PaginatedResult,\n GlobalSessionStats,\n ExportFormat,\n SessionExport,\n SessionTranscriptSummary,\n CompactionCheckpointSummary,\n CompactionCheckpointDetail,\n} from './types.js';\nimport { SessionStatus } from './types.js';\nimport type { Message } from './types.js';\nimport { parseCompactionCheckpointTranscriptFileName } from './parity/artifacts.js';\nimport { archiveFileOnDisk, resolveSessionFilePath, resolveSessionTranscriptPathInDir } from './parity/transcript-paths.js';\nimport { validateSessionId } from './parity/session-id.js';\nimport { readSessionsJsonFile, withSessionsJsonLock } from './parity/sessions-json-file.js';\nimport {\n buildSessionsJsonStatsPatch,\n incrementSessionsJsonStatsForAppend,\n isAppendOnlyLlmTranscriptMessage,\n patchSessionsJsonEntryStats,\n} from './parity/sessions-json-patch.js';\nimport {\n countTranscriptMessageRows,\n readDisplayMessagePageFromTranscriptFile,\n} from './parity/transcript-pagination.js';\nimport type { XopcSessionDiskEntry } from './parity/xopc-session-disk-entry.js';\nimport {\n appendPiTranscriptContextEntry,\n persistMergedTranscriptRows,\n readTranscriptRowsFromFile,\n rowsToLlmMessages,\n writeTranscriptJsonl,\n} from './parity/jsonl-transcript-io.js';\nimport type { SessionTranscriptUpdate } from './transcript-events.js';\n\nconst log = createLogger('SessionStore');\n\nconst INDEX_VERSION = '1.0';\nconst DELETED_MARKER = '.jsonl.deleted.';\nconst ALL_SESSIONS_MAP_CACHE_TTL_MS = 2_000;\n\nexport interface SessionStoreOptions {\n config: Config;\n agentId?: string;\n sessionsDir?: string;\n}\n\nexport class SessionStore {\n private sessionsDir: string;\n private archiveDir: string;\n private storePath: string;\n private window: SlidingWindow;\n private compactor: SessionCompactor;\n private storeMutationDepth = 0;\n private storeMutationChain: Promise<void> = Promise.resolve();\n private allSessionsMapCache?: {\n expiresAtMs: number;\n map: Record<string, XopcSessionDiskEntry>;\n };\n /** Cache of per-agent sessions dirs to avoid re-resolution on every call. */\n private agentSessionsDirCache = new Map<string, string>();\n\n constructor(\n private options: SessionStoreOptions,\n windowConfig?: Partial<WindowConfig>,\n compactionConfig?: Partial<CompactionConfig>,\n ) {\n const agentId = options.agentId ?? resolveDefaultAgentId(options.config);\n this.sessionsDir = options.sessionsDir ?? resolveSessionsDir(options.config, agentId);\n this.archiveDir = join(this.sessionsDir, 'archive');\n this.storePath = join(this.sessionsDir, FILENAMES.SESSIONS_MAP);\n this.window = new SlidingWindow(windowConfig);\n this.compactor = new SessionCompactor(compactionConfig);\n }\n\n getSessionsRoot(): string {\n return this.sessionsDir;\n }\n\n /**\n * OpenClaw-aligned: resolve the sessions directory for a given session key.\n * Extracts agentId from the session key and routes to `agents/<agentId>/sessions/`.\n * Falls back to the default sessions directory when agentId cannot be parsed\n * or when `sessionsDir` was explicitly provided in options.\n */\n private resolveSessionsDirForKey(sessionKey: string): string {\n if (this.options.sessionsDir) {\n return this.sessionsDir;\n }\n const parsed = parseRoutingSessionKey(sessionKey);\n if (!parsed) {\n return this.sessionsDir;\n }\n const agentId = parsed.agentId;\n const cached = this.agentSessionsDirCache.get(agentId);\n if (cached) {\n return cached;\n }\n const resolved = resolveSessionsDir(this.options.config, agentId);\n this.agentSessionsDirCache.set(agentId, resolved);\n return resolved;\n }\n\n private resolveStorePathForKey(sessionKey: string): string {\n return join(this.resolveSessionsDirForKey(sessionKey), FILENAMES.SESSIONS_MAP);\n }\n\n private async runStoreMutation<T>(fn: () => Promise<T>): Promise<T> {\n if (this.storeMutationDepth > 0) {\n return fn();\n }\n const run = this.storeMutationChain.then(async () => {\n this.storeMutationDepth++;\n try {\n return await fn();\n } finally {\n this.storeMutationDepth--;\n }\n });\n this.storeMutationChain = run.then(() => undefined).catch(() => undefined);\n return run as Promise<T>;\n }\n\n async initialize(): Promise<void> {\n await mkdir(this.sessionsDir, { recursive: true });\n await mkdir(this.archiveDir, { recursive: true });\n if (!existsSync(this.storePath)) {\n await withSessionsJsonLock(this.storePath, async () => undefined);\n }\n log.debug('Session store initialized (sessions.json + JSONL)');\n }\n\n private transcriptPathForEntry(entry: XopcSessionDiskEntry, sessionsDir?: string): string {\n return resolveSessionFilePath(entry.sessionId, entry, { sessionsDir: sessionsDir ?? this.sessionsDir });\n }\n\n private async readMapForKey(sessionKey: string): Promise<Record<string, XopcSessionDiskEntry>> {\n const storePath = this.resolveStorePathForKey(sessionKey);\n return readSessionsJsonFile<XopcSessionDiskEntry>(storePath);\n }\n\n private async readMap(): Promise<Record<string, XopcSessionDiskEntry>> {\n return readSessionsJsonFile<XopcSessionDiskEntry>(this.storePath);\n }\n\n private invalidateAllSessionsMapCache(): void {\n this.allSessionsMapCache = undefined;\n }\n\n private async discoverSessionMapPaths(): Promise<Array<{ agentId: string; mapPath: string }>> {\n const agents = listAgentEntries(this.options.config);\n const defaultId = resolveDefaultAgentId(this.options.config);\n const agentIds = new Set<string>([defaultId, ...agents.map((agent) => agent.id)]);\n\n const agentsRoot = join(resolveStateDir(process.env), 'agents');\n if (existsSync(agentsRoot)) {\n const entries = await readdir(agentsRoot, { withFileTypes: true }).catch(() => []);\n for (const entry of entries) {\n if (entry.isDirectory()) {\n agentIds.add(entry.name);\n }\n }\n }\n\n return [...agentIds].map((agentId) => ({\n agentId,\n mapPath: join(resolveSessionsDir(this.options.config, agentId), FILENAMES.SESSIONS_MAP),\n }));\n }\n\n /**\n * Unified cross-agent aggregation entry for global session views.\n * Reads configured agents plus existing per-agent session maps under the state directory.\n */\n private async readAllMaps(): Promise<Record<string, XopcSessionDiskEntry>> {\n if (this.options.sessionsDir) {\n return this.readMap();\n }\n\n const nowMs = Date.now();\n if (this.allSessionsMapCache && this.allSessionsMapCache.expiresAtMs > nowMs) {\n log.debug(\n { sessionCount: Object.keys(this.allSessionsMapCache.map).length },\n 'All session maps cache hit',\n );\n return this.allSessionsMapCache.map;\n }\n\n const startedAt = performance.now();\n const paths = await this.discoverSessionMapPaths();\n const merged: Record<string, XopcSessionDiskEntry> = {};\n let scannedMapCount = 0;\n for (const { mapPath } of paths) {\n if (!existsSync(mapPath)) {\n continue;\n }\n const map = await readSessionsJsonFile<XopcSessionDiskEntry>(mapPath);\n Object.assign(merged, map);\n scannedMapCount++;\n }\n\n this.allSessionsMapCache = {\n expiresAtMs: nowMs + ALL_SESSIONS_MAP_CACHE_TTL_MS,\n map: merged,\n };\n log.debug(\n {\n candidateAgentCount: paths.length,\n scannedMapCount,\n sessionCount: Object.keys(merged).length,\n durationMs: Math.round(performance.now() - startedAt),\n },\n 'All session maps scanned',\n );\n return merged;\n }\n\n private async getDiskEntry(sessionKey: string): Promise<XopcSessionDiskEntry | undefined> {\n const map = await this.readMapForKey(sessionKey);\n return map[sessionKey];\n }\n\n private buildDefaultMetadata(key: string): SessionMetadata {\n const { channel, chatId } = this.parseSessionKey(key);\n const routing = this.extractRoutingFromKey(key);\n const isCronSession = channel === 'cron';\n const isHeartbeatSession = channel === 'heartbeat';\n const now = new Date().toISOString();\n return {\n key,\n status: SessionStatus.ACTIVE,\n tags: [],\n createdAt: now,\n updatedAt: now,\n lastAccessedAt: now,\n messageCount: 0,\n estimatedTokens: 0,\n compactedCount: 0,\n sourceChannel: channel,\n sourceChatId: chatId,\n routing,\n ...(isCronSession\n ? { sessionType: 'cron', customData: { cronJobId: chatId } }\n : {}),\n ...(isHeartbeatSession\n ? { sessionType: 'heartbeat', customData: { heartbeatTarget: chatId } }\n : {}),\n stats: { messageCount: 0, tokenCount: 0 },\n };\n }\n\n private parseSessionKey(key: string): { channel: string; chatId: string } {\n const parts = key.split(':');\n if (parts.length >= 2 && parts[0] === 'heartbeat') {\n return { channel: 'heartbeat', chatId: parts.slice(1).join(':') };\n }\n const parsed = parseRoutingSessionKey(key);\n if (parsed) {\n if (parsed.source === 'cron') {\n return { channel: 'cron', chatId: parsed.peerId };\n }\n return {\n channel: parsed.source,\n chatId: [parsed.accountId, parsed.peerKind, parsed.peerId].join(':'),\n };\n }\n return { channel: 'unknown', chatId: key };\n }\n\n private extractRoutingFromKey(key: string): SessionMetadata['routing'] {\n const parsed = parseRoutingSessionKey(key);\n if (!parsed) {\n return undefined;\n }\n return {\n agentId: parsed.agentId?.toLowerCase() || 'main',\n source: parsed.source?.toLowerCase() || 'unknown',\n accountId: parsed.accountId?.toLowerCase() || 'default',\n peerKind: parsed.peerKind?.toLowerCase() || 'dm',\n peerId: parsed.peerId?.toLowerCase() || 'unknown',\n threadId: parsed.threadId,\n scopeId: parsed.scopeId,\n };\n }\n\n /** Resolve on-disk transcript path; creates session row + empty JSONL when missing. */\n async resolveTranscriptPath(\n sessionKey: string,\n ): Promise<{ sessionId: string; absPath: string; sessionsDir: string }> {\n const entry = await this.ensureSession(sessionKey);\n const sessionsDir = this.resolveSessionsDirForKey(sessionKey);\n const absPath = this.transcriptPathForEntry(entry, sessionsDir);\n return { sessionId: entry.sessionId, absPath, sessionsDir };\n }\n\n /** Ensure sessions.json has an entry and transcript file exist for `sessionKey`. */\n private async ensureSession(sessionKey: string): Promise<XopcSessionDiskEntry> {\n const keyStorePath = this.resolveStorePathForKey(sessionKey);\n const keySessionsDir = this.resolveSessionsDirForKey(sessionKey);\n await mkdir(keySessionsDir, { recursive: true });\n let changed = false;\n const entry = await withSessionsJsonLock(keyStorePath, async (map) => {\n const existing = map[sessionKey] as XopcSessionDiskEntry | undefined;\n if (existing?.pluginExtensions?.xopc?.metadata) {\n return existing;\n }\n let nextEntry = existing;\n if (!nextEntry) {\n const sessionId = randomUUID();\n validateSessionId(sessionId);\n const sessionFile = `${sessionId}.jsonl`;\n const now = Date.now();\n const metadata = this.buildDefaultMetadata(sessionKey);\n metadata.transcriptId = sessionId;\n nextEntry = {\n sessionId,\n updatedAt: now,\n sessionStartedAt: now,\n sessionFile,\n pluginExtensions: { xopc: { metadata } },\n };\n map[sessionKey] = nextEntry as Record<string, unknown>;\n const abs = resolveSessionTranscriptPathInDir(sessionId, keySessionsDir);\n await writeTranscriptJsonl({\n absPath: abs,\n sessionId,\n cwd: process.cwd(),\n rows: [],\n });\n changed = true;\n } else if (!nextEntry.pluginExtensions?.xopc?.metadata) {\n const metadata = this.buildDefaultMetadata(sessionKey);\n metadata.transcriptId = nextEntry.sessionId;\n nextEntry.pluginExtensions = { xopc: { metadata } };\n map[sessionKey] = nextEntry as Record<string, unknown>;\n changed = true;\n }\n return nextEntry!;\n });\n if (changed) {\n this.invalidateAllSessionsMapCache();\n }\n return entry;\n }\n\n private metadataFromEntry(sessionKey: string, entry: XopcSessionDiskEntry): SessionMetadata {\n const base = entry.pluginExtensions?.xopc?.metadata ?? this.buildDefaultMetadata(sessionKey);\n const { channel: keySource, chatId: keyChatId } = this.parseSessionKey(sessionKey);\n const diskSc = typeof base.sourceChannel === 'string' ? base.sourceChannel.trim() : '';\n const diskChat = typeof base.sourceChatId === 'string' ? base.sourceChatId.trim() : '';\n return {\n ...base,\n key: sessionKey,\n transcriptId: entry.sessionId,\n sourceChannel: diskSc || keySource,\n sourceChatId: diskChat || keyChatId,\n };\n }\n\n async getByAgent(agentId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.agentId?.toLowerCase() === agentId.toLowerCase()) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getByAccount(accountId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.accountId === accountId) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getByPeer(peerKind: string, peerId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.peerKind === peerKind && m.routing.peerId === peerId) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getMainSession(channel: string, accountId: string): Promise<SessionMetadata | null> {\n const map = await this.readAllMaps();\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (\n m.routing?.source === channel &&\n m.routing.accountId === accountId &&\n m.routing.peerKind === 'dm' &&\n m.routing.peerId === 'main'\n ) {\n return m;\n }\n }\n return null;\n }\n\n async refreshIndex(): Promise<void> {\n /* no-op: sessions.json is authoritative */\n }\n\n async list(query: SessionListQuery = {}): Promise<PaginatedResult<SessionMetadata>> {\n const map = await this.readAllMaps();\n let sessions = Object.entries(map).map(([k, e]) => this.metadataFromEntry(k, e));\n\n if (query.status) {\n const statuses = Array.isArray(query.status) ? query.status : [query.status];\n sessions = sessions.filter((s) => statuses.includes(s.status));\n }\n if (query.channel) {\n const rawChannels = query.channel\n .split(',')\n .map((c) => c.trim().toLowerCase())\n .filter(Boolean);\n /**\n * `ui` is a legacy console source; treat as webchat when filtering web sessions.\n * `webui` matches slash-command normalization to `gateway` (see `chat-commands/session-key.ts`).\n */\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 sessions = [];\n } else if (channels.length === 1) {\n const ch = channels[0]!;\n sessions = sessions.filter((s) => (s.sourceChannel ?? '').toLowerCase() === ch);\n } else {\n sessions = sessions.filter((s) => channels.includes((s.sourceChannel ?? '').toLowerCase()));\n }\n }\n if (query.tags?.length) {\n sessions = sessions.filter((s) => query.tags!.some((t) => s.tags.includes(t)));\n }\n if (query.search) {\n const q = query.search.toLowerCase();\n sessions = sessions.filter(\n (s) =>\n s.key.toLowerCase().includes(q) ||\n (s.name?.toLowerCase().includes(q) ?? false) ||\n s.tags.some((t) => t.toLowerCase().includes(q)),\n );\n }\n\n const sortBy = query.sortBy || 'updatedAt';\n const sortOrder = query.sortOrder || 'desc';\n sessions.sort((a, b) => {\n const av = a[sortBy];\n const bv = b[sortBy];\n const c = av < bv ? -1 : av > bv ? 1 : 0;\n return sortOrder === 'asc' ? c : -c;\n });\n\n const total = sessions.length;\n const limit = query.limit || 50;\n const offset = query.offset || 0;\n const items = sessions.slice(offset, offset + limit);\n return { items, total, limit, offset, hasMore: offset + limit < total };\n }\n\n async get(\n key: string,\n options?: { includeTranscriptSummary?: boolean; includeTranscriptRows?: boolean },\n ): Promise<SessionDetail | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return null;\n }\n const messages = await this.loadDisplayMessages(key);\n const detail = await this.buildSessionDetail(key, metadata, messages, options);\n return detail;\n }\n\n async getMessagePage(\n key: string,\n options: {\n offset?: number;\n limit?: number;\n before?: string;\n includeTranscriptSummary?: boolean;\n includeTranscriptRows?: boolean;\n } = {},\n ): Promise<{\n session: SessionDetail;\n pagination: {\n total: number;\n limit: number;\n offset: number;\n hasMore: boolean;\n before?: string;\n nextBeforeCursor?: string;\n };\n } | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return null;\n }\n\n const limit = Math.min(200, Math.max(1, Math.trunc(options.limit ?? 50)));\n const offset = Math.max(0, Math.trunc(options.offset ?? 0));\n const parsedBefore = options.before ? Number.parseInt(options.before, 10) : undefined;\n const hasBeforeCursor = parsedBefore !== undefined && Number.isFinite(parsedBefore);\n\n const checkpoints = await this.listCompactionCheckpoints(key);\n if (checkpoints.length === 0) {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n const page = await readDisplayMessagePageFromTranscriptFile(primary, {\n limit,\n offset: hasBeforeCursor ? undefined : offset,\n beforeIndex: hasBeforeCursor ? parsedBefore : undefined,\n });\n const session = await this.buildSessionDetail(key, metadata, page.messages, options);\n const nextBeforeCursor = page.startIndex > 0 ? String(page.startIndex) : undefined;\n\n return {\n session,\n pagination: {\n total: page.total,\n limit,\n offset,\n hasMore: page.startIndex > 0,\n ...(hasBeforeCursor ? { before: String(page.endIndex) } : {}),\n ...(nextBeforeCursor ? { nextBeforeCursor } : {}),\n },\n };\n }\n\n const messages = await this.loadDisplayMessages(key);\n const total = messages.length;\n const endExclusive = hasBeforeCursor\n ? Math.min(total, Math.max(0, Math.trunc(parsedBefore!)))\n : Math.max(0, total - offset);\n const startInclusive = Math.max(0, endExclusive - limit);\n const pageMessages = messages.slice(startInclusive, endExclusive);\n const session = await this.buildSessionDetail(key, metadata, pageMessages, options);\n const nextBeforeCursor = startInclusive > 0 ? String(startInclusive) : undefined;\n\n return {\n session,\n pagination: {\n total,\n limit,\n offset,\n hasMore: startInclusive > 0,\n ...(hasBeforeCursor ? { before: String(endExclusive) } : {}),\n ...(nextBeforeCursor ? { nextBeforeCursor } : {}),\n },\n };\n }\n\n private async buildSessionDetail(\n key: string,\n metadata: SessionMetadata,\n messages: AgentMessage[],\n options?: { includeTranscriptSummary?: boolean; includeTranscriptRows?: boolean },\n ): Promise<SessionDetail> {\n let transcriptSummary: SessionTranscriptSummary | undefined;\n if (options?.includeTranscriptSummary) {\n const env = await this.loadTranscriptDocument(key);\n if (env) {\n transcriptSummary = {\n id: env.id,\n version: env.version,\n createdAt: env.createdAt,\n updatedAt: env.updatedAt,\n compactionCount: env.compactions?.length ?? 0,\n };\n }\n }\n let transcriptRows: TranscriptStoredRow[] | undefined;\n if (options?.includeTranscriptRows) {\n transcriptRows = await this.loadTranscriptRows(key);\n }\n return {\n ...metadata,\n messages: this.convertMessages(messages),\n ...(transcriptSummary ? { transcriptSummary } : {}),\n ...(transcriptRows !== undefined ? { transcriptRows } : {}),\n };\n }\n\n async loadTranscriptRows(key: string): Promise<TranscriptStoredRow[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const path = this.transcriptPathForEntry(entry, this.resolveSessionsDirForKey(key));\n return readTranscriptRowsFromFile(path);\n }\n\n async getMetadata(key: string): Promise<SessionMetadata | null> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n return this.metadataFromEntry(key, entry);\n }\n\n async updateMetadata(key: string, updates: Partial<SessionMetadata>): Promise<void> {\n return this.runStoreMutation(async () => {\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const entry = map[key] as XopcSessionDiskEntry | undefined;\n if (!entry?.pluginExtensions?.xopc?.metadata) {\n throw new Error(`Session not found: ${key}`);\n }\n const meta = { ...entry.pluginExtensions.xopc.metadata, ...updates, updatedAt: new Date().toISOString() };\n entry.pluginExtensions.xopc.metadata = meta;\n entry.updatedAt = Date.now();\n map[key] = entry as Record<string, unknown>;\n });\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n log.debug({ key, updates }, 'Session metadata updated');\n });\n }\n\n /**\n * Reset transcript for an existing session key: archive the current JSONL as\n * `*.reset.*`, assign a new `sessionId`, and preserve per-session overrides\n * on the disk entry (thinking/verbose) and in `sessions/config/*.json`.\n */\n async reset(key: string): Promise<{ sessionId: string; previousSessionId: string } | null> {\n return this.runStoreMutation(async () => {\n const existing = await this.getDiskEntry(key);\n if (!existing?.pluginExtensions?.xopc?.metadata) {\n return null;\n }\n\n const previousSessionId = existing.sessionId;\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(existing, keySessionsDir);\n if (existsSync(abs)) {\n try {\n archiveFileOnDisk(abs, 'reset');\n } catch (err) {\n log.warn({ err, key }, 'Transcript archive on reset failed');\n }\n }\n\n const sessionId = randomUUID();\n validateSessionId(sessionId);\n const now = Date.now();\n const nextAbs = resolveSessionTranscriptPathInDir(sessionId, keySessionsDir);\n await writeTranscriptJsonl({\n absPath: nextAbs,\n sessionId,\n cwd: process.cwd(),\n rows: [],\n });\n\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n e.sessionId = sessionId;\n e.sessionFile = `${sessionId}.jsonl`;\n e.updatedAt = now;\n e.sessionStartedAt = now;\n e.lastInteractionAt = undefined;\n patchSessionsJsonEntryStats(e, buildSessionsJsonStatsPatch(0, 0));\n const meta = e.pluginExtensions.xopc.metadata;\n meta.transcriptId = sessionId;\n meta.updatedAt = new Date(now).toISOString();\n map[key] = e as Record<string, unknown>;\n });\n\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n log.info({ key, previousSessionId, sessionId }, 'Session reset');\n return { sessionId, previousSessionId };\n });\n }\n\n async delete(key: string): Promise<boolean> {\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return false;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n delete map[key];\n });\n try {\n if (existsSync(abs)) {\n archiveFileOnDisk(abs, 'deleted');\n }\n } catch (err) {\n log.warn({ err, key }, 'Transcript archive on delete failed');\n }\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n log.info({ key }, 'Session deleted');\n return true;\n });\n }\n\n async deleteMany(keys: string[]): Promise<{ success: string[]; failed: string[] }> {\n const success: string[] = [];\n const failed: string[] = [];\n for (const key of keys) {\n try {\n if (await this.delete(key)) {\n success.push(key);\n } else {\n failed.push(key);\n }\n } catch {\n failed.push(key);\n }\n }\n return { success, failed };\n }\n\n async setStatus(key: string, status: SessionStatus): Promise<void> {\n await this.updateMetadata(key, { status });\n if (status === SessionStatus.ARCHIVED) {\n await this.moveToArchive(key);\n } else {\n await this.moveFromArchive(key);\n }\n }\n\n async archive(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ARCHIVED);\n }\n\n async unarchive(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ACTIVE);\n }\n\n async pin(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.PINNED);\n }\n\n async unpin(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ACTIVE);\n }\n\n private async moveToArchive(key: string): Promise<void> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n if (existsSync(abs)) {\n try {\n archiveFileOnDisk(abs, 'deleted');\n } catch (err) {\n log.warn({ err, key }, 'Archive transcript rename failed');\n }\n }\n }\n\n private async findMostRecentDeletedTranscript(sessionId: string, sessionsDir: string): Promise<string | null> {\n let names: string[];\n try {\n names = await readdir(sessionsDir);\n } catch {\n return null;\n }\n const prefix = `${sessionId}${DELETED_MARKER}`;\n const hits = names.filter((n) => n.startsWith(prefix) && n.endsWith('Z'));\n hits.sort().reverse();\n const first = hits[0];\n return first ? join(sessionsDir, first) : null;\n }\n\n private async moveFromArchive(key: string): Promise<void> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const target = resolveSessionTranscriptPathInDir(entry.sessionId, keySessionsDir);\n if (existsSync(target)) {\n return;\n }\n const src = await this.findMostRecentDeletedTranscript(entry.sessionId, keySessionsDir);\n if (!src) {\n return;\n }\n try {\n const { rename } = await import('fs/promises');\n await rename(src, target);\n } catch (err) {\n log.warn({ err, key, src, target }, 'Unarchive transcript rename failed');\n }\n }\n\n async loadMessages(key: string, options?: { fromArchive?: boolean }): Promise<AgentMessage[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n if (existsSync(primary)) {\n const rows = await readTranscriptRowsFromFile(primary);\n return rowsToLlmMessages(rows);\n }\n if (options?.fromArchive) {\n const archived = await this.findMostRecentDeletedTranscript(entry.sessionId, keySessionsDir);\n if (!archived) {\n return [];\n }\n const rows = await readTranscriptRowsFromFile(archived);\n return rowsToLlmMessages(rows);\n }\n return [];\n }\n\n private async loadDisplayMessages(key: string): Promise<AgentMessage[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n const transcriptPaths: string[] = [];\n const checkpoints = await this.listCompactionCheckpoints(key);\n for (const checkpoint of [...checkpoints].reverse()) {\n transcriptPaths.push(join(keySessionsDir, `${this.checkpointBasename(entry.sessionId)}${checkpoint.id}.jsonl`));\n }\n transcriptPaths.push(primary);\n\n const messages: AgentMessage[] = [];\n const seenMessages = new Set<string>();\n for (const transcriptPath of transcriptPaths) {\n if (!existsSync(transcriptPath)) {\n continue;\n }\n const rows = await readTranscriptRowsFromFile(transcriptPath);\n for (const message of rowsToLlmMessages(rows)) {\n if (this.isCompactionSummaryMessage(message)) {\n continue;\n }\n const key = this.displayMessageIdentity(message);\n if (seenMessages.has(key)) {\n continue;\n }\n seenMessages.add(key);\n messages.push(message);\n }\n }\n return messages;\n }\n\n async loadTranscriptDocument(key: string): Promise<XopcSessionTranscriptV1 | null> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const path = this.transcriptPathForEntry(entry, this.resolveSessionsDirForKey(key));\n if (!existsSync(path)) {\n return null;\n }\n const entries = loadEntriesFromFile(path);\n const header = entries.find((e) => e.type === 'session');\n if (!header || typeof (header as { id?: unknown }).id !== 'string') {\n return null;\n }\n const sessionHeader = header as { type: 'session'; id: string; timestamp?: string };\n const rows = await readTranscriptRowsFromFile(path);\n const compactions = entries\n .filter((e): e is CompactionEntry => e.type === 'compaction')\n .map((c) => ({\n at: c.timestamp,\n summary: c.summary,\n firstKeptIndex: Number.parseInt(String(c.firstKeptEntryId), 10) || 0,\n tokensBefore: c.tokensBefore,\n tokensAfter:\n typeof c.details === 'object' &&\n c.details &&\n 'tokensAfter' in c.details &&\n typeof (c.details as { tokensAfter?: unknown }).tokensAfter === 'number'\n ? (c.details as { tokensAfter: number }).tokensAfter\n : 0,\n }));\n return {\n type: 'xopc_session_transcript',\n version: 1,\n id: sessionHeader.id,\n createdAt: sessionHeader.timestamp ?? new Date().toISOString(),\n updatedAt: new Date().toISOString(),\n messages: rows,\n ...(compactions.length > 0 ? { compactions } : {}),\n };\n }\n\n private async writeTranscriptAndSyncIndex(\n key: string,\n rows: TranscriptStoredRow[],\n opts?: { appendCompaction?: TranscriptCompactionRecord },\n ): Promise<void> {\n const entry = await this.ensureSession(key);\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const llm = rowsToLlmMessages(rows);\n await persistMergedTranscriptRows({\n absPath: abs,\n sessionId: entry.sessionId,\n cwd: process.cwd(),\n rows,\n appendCompaction: opts?.appendCompaction,\n });\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(llm.length, this.estimateTokens(llm)),\n );\n map[key] = e as Record<string, unknown>;\n });\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n }\n\n /** Incremental sessions.json stats after guard append (OpenClaw transcript-events). */\n async syncSessionsJsonFromTranscriptUpdate(update: SessionTranscriptUpdate): Promise<void> {\n const sessionKey = update.sessionKey?.trim();\n if (!sessionKey || !existsSync(update.sessionFile)) {\n return;\n }\n return this.runStoreMutation(async () => {\n const keyStorePath = this.resolveStorePathForKey(sessionKey);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[sessionKey] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n\n if (update.message && isAppendOnlyLlmTranscriptMessage(update.message)) {\n incrementSessionsJsonStatsForAppend(e);\n map[sessionKey] = e as Record<string, unknown>;\n return;\n }\n\n const messageCount = await countTranscriptMessageRows(update.sessionFile);\n const rows = await readTranscriptRowsFromFile(update.sessionFile);\n const llm = rowsToLlmMessages(rows);\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(messageCount, this.estimateTokens(llm)),\n );\n map[sessionKey] = e as Record<string, unknown>;\n });\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n });\n }\n\n async appendTranscriptContextEntry(\n key: string,\n entry: Omit<XopcTranscriptContextEntry, 'kind'> & Partial<Pick<XopcTranscriptContextEntry, 'kind'>>,\n ): Promise<void> {\n return this.runStoreMutation(async () => {\n await this.ensureSession(key);\n const disk = await this.getDiskEntry(key);\n if (!disk) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const absPath = this.transcriptPathForEntry(disk, keySessionsDir);\n const row: XopcTranscriptContextEntry = {\n kind: 'context',\n id: typeof entry.id === 'string' ? entry.id : undefined,\n text: typeof entry.text === 'string' ? entry.text : undefined,\n data: entry.data,\n createdAt: entry.createdAt ?? new Date().toISOString(),\n };\n await appendPiTranscriptContextEntry({\n absPath,\n cwd: process.cwd(),\n entry: row,\n sessionKey: key,\n });\n const rows = existsSync(absPath) ? await readTranscriptRowsFromFile(absPath) : [];\n const llm = rowsToLlmMessages(rows);\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(llm.length, this.estimateTokens(llm)),\n );\n map[key] = e as Record<string, unknown>;\n });\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n });\n }\n\n /**\n * Bulk write entry point used by compaction, tests, and admin tools.\n * Runtime agent turns must persist via {@link guardSessionManager} + appendMessage.\n */\n async saveMessages(key: string, messages: AgentMessage[]): Promise<void> {\n return this.runStoreMutation(async () => {\n await this.ensureSession(key);\n const prev = await this.loadTranscriptRows(key);\n const merged = mergeLlmMessagesPreservingContextRows(prev, messages);\n await this.writeTranscriptAndSyncIndex(key, merged);\n });\n }\n\n getWindowStats(messages: AgentMessage[]) {\n return this.window.getStats(messages);\n }\n\n needsCompaction(key: string, messages: AgentMessage[], contextWindow: number) {\n return this.compactor.needsCompaction(messages, contextWindow);\n }\n\n prepareCompaction(\n key: string,\n messages: AgentMessage[],\n contextWindow: number,\n ): { needsCompaction: boolean; messages: AgentMessage[]; stats?: ReturnType<typeof this.compactor.needsCompaction> } {\n const result = this.compactor.needsCompaction(messages, contextWindow);\n return { needsCompaction: result.needed, messages, stats: result };\n }\n\n private checkpointBasename(sessionId: string): string {\n return `${sessionId}.checkpoint.`;\n }\n\n private async pruneCompactionCheckpoints(sessionId: string, sessionsDir: string): Promise<void> {\n const MAX = 15;\n const prefix = this.checkpointBasename(sessionId);\n let names: string[];\n try {\n names = await readdir(sessionsDir);\n } catch {\n return;\n }\n const candidates = names.filter((n) => n.startsWith(prefix) && n.endsWith('.jsonl'));\n if (candidates.length <= MAX) {\n return;\n }\n const stats = await Promise.all(\n candidates.map(async (name) => {\n const p = join(sessionsDir, name);\n try {\n const s = await stat(p);\n return { p, mtimeMs: s.mtimeMs };\n } catch {\n return { p: join(sessionsDir, name), mtimeMs: 0 };\n }\n }),\n );\n stats.sort((a, b) => a.mtimeMs - b.mtimeMs);\n for (let i = 0; i < stats.length - MAX; i++) {\n try {\n await unlink(stats[i]!.p);\n } catch {\n /* ignore */\n }\n }\n }\n\n private async captureCompactionCheckpoint(sessionId: string, transcriptAbs: string, sessionsDir: string): Promise<void> {\n if (!existsSync(transcriptAbs)) {\n return;\n }\n const id = randomUUID();\n const dest = join(sessionsDir, `${sessionId}.checkpoint.${id}.jsonl`);\n try {\n await copyFile(transcriptAbs, dest);\n await this.pruneCompactionCheckpoints(sessionId, sessionsDir);\n } catch (err) {\n log.warn({ err, sessionId }, 'Compaction checkpoint copy failed');\n }\n }\n\n async applyCompaction(\n key: string,\n messages: AgentMessage[],\n result: CompactionResult,\n ): Promise<AgentMessage[]> {\n const compacted = this.compactor.applyCompaction(messages, result);\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return compacted;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n await this.captureCompactionCheckpoint(entry.sessionId, abs, keySessionsDir);\n const prev = await this.loadTranscriptRows(key);\n const merged = mergeLlmMessagesPreservingContextRows(prev, compacted);\n await this.writeTranscriptAndSyncIndex(key, merged, {\n appendCompaction: {\n at: new Date().toISOString(),\n summary: result.summary,\n firstKeptIndex: result.firstKeptIndex,\n tokensBefore: result.tokensBefore,\n tokensAfter: result.tokensAfter,\n },\n });\n const metadata = await this.getMetadata(key);\n if (metadata) {\n await this.updateMetadata(key, { compactedCount: metadata.compactedCount + 1 });\n }\n log.info(\n { key, tokensBefore: result.tokensBefore, tokensAfter: result.tokensAfter, keptMessages: compacted.length },\n 'Session compacted',\n );\n await this.injectPostCompactionContext(key);\n return compacted;\n });\n }\n\n async compact(\n key: string,\n messages: AgentMessage[],\n contextWindow: number,\n instructions?: string,\n force?: boolean,\n ): Promise<CompactionResult> {\n const result = await this.compactor.compact(messages, instructions, force);\n if (result.compacted) {\n await this.applyCompaction(key, messages, result);\n }\n return result;\n }\n\n async getCompactionStats(key: string) {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return undefined;\n }\n return {\n compactionCount: metadata.compactedCount,\n totalTokensBefore: 0,\n totalTokensAfter: 0,\n lastCompactionAt: undefined,\n };\n }\n\n async listCompactionCheckpoints(key: string): Promise<CompactionCheckpointSummary[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const sessionId = entry.sessionId;\n const prefix = this.checkpointBasename(sessionId);\n let names: string[];\n try {\n names = await readdir(keySessionsDir);\n } catch {\n return [];\n }\n const files = names.filter((n) => n.startsWith(prefix) && n.endsWith('.jsonl'));\n const rows = await Promise.all(\n files.map(async (name) => {\n const p = join(keySessionsDir, name);\n const parsed = parseCompactionCheckpointTranscriptFileName(name);\n const id = parsed?.checkpointId;\n if (!id || !normalizeCompactionCheckpointId(id)) {\n return null;\n }\n try {\n const s = await stat(p);\n return {\n id: normalizeCompactionCheckpointId(id)!,\n sizeBytes: s.size,\n modifiedAt: new Date(s.mtimeMs).toISOString(),\n } satisfies CompactionCheckpointSummary;\n } catch {\n return null;\n }\n }),\n );\n const valid = rows.filter((r): r is CompactionCheckpointSummary => r !== null);\n valid.sort((a, b) => b.modifiedAt.localeCompare(a.modifiedAt));\n return valid;\n }\n\n async getCompactionCheckpointDetail(\n key: string,\n checkpointId: string,\n ): Promise<CompactionCheckpointDetail | null> {\n const id = normalizeCompactionCheckpointId(checkpointId);\n if (!id) {\n return null;\n }\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const fname = `${this.checkpointBasename(entry.sessionId)}${id}.jsonl`;\n const cpPath = join(keySessionsDir, fname);\n if (!existsSync(cpPath)) {\n return null;\n }\n try {\n const rows = await readTranscriptRowsFromFile(cpPath);\n const llm = rowsToLlmMessages(rows);\n const s = await stat(cpPath);\n return {\n id,\n sizeBytes: s.size,\n modifiedAt: new Date(s.mtimeMs).toISOString(),\n messageCount: llm.length,\n };\n } catch {\n return null;\n }\n }\n\n async restoreCompactionCheckpoint(key: string, checkpointId: string): Promise<void> {\n const id = normalizeCompactionCheckpointId(checkpointId);\n if (!id) {\n throw new Error('Invalid checkpoint id');\n }\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n throw new Error(`Session not found: ${key}`);\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const cpPath = join(keySessionsDir, `${this.checkpointBasename(entry.sessionId)}${id}.jsonl`);\n if (!existsSync(cpPath)) {\n throw new Error(`Checkpoint not found: ${id}`);\n }\n const target = this.transcriptPathForEntry(entry, keySessionsDir);\n await copyFile(cpPath, target);\n const messages = await this.loadMessages(key);\n await this.saveMessages(key, messages);\n log.info({ key, checkpointId: id }, 'Session transcript restored from compaction checkpoint');\n });\n }\n\n async deleteSession(key: string): Promise<boolean> {\n return this.delete(key);\n }\n\n async load(key: string, options?: { fromArchive?: boolean }): Promise<AgentMessage[]> {\n return this.loadMessages(key, options);\n }\n\n async estimateTokenUsage(_key: string, messages: AgentMessage[]): Promise<number> {\n return this.estimateTokens(messages);\n }\n\n async searchInSession(key: string, keyword: string): Promise<Message[]> {\n const messages = await this.loadDisplayMessages(key);\n const keywordLower = keyword.toLowerCase();\n return this.convertMessages(\n messages.filter((m) => {\n const content = this.extractTextContent(this.messageContent(m));\n return content.toLowerCase().includes(keywordLower);\n }),\n );\n }\n\n async exportSession(key: string, format: ExportFormat): Promise<string> {\n const detail = await this.get(key);\n if (!detail) {\n throw new Error(`Session not found: ${key}`);\n }\n if (format === 'json') {\n const transcriptRows = await this.loadTranscriptRows(key);\n const exportData: SessionExport = {\n version: INDEX_VERSION,\n exportedAt: new Date().toISOString(),\n metadata: detail,\n messages: detail.messages,\n transcriptRows,\n };\n return JSON.stringify(exportData, null, 2);\n }\n const lines = [\n `# ${detail.name || detail.key}`,\n '',\n `- **Channel:** ${detail.sourceChannel}`,\n `- **Created:** ${detail.createdAt}`,\n `- **Messages:** ${detail.messageCount}`,\n `- **Tags:** ${detail.tags.join(', ') || 'none'}`,\n '',\n '---',\n '',\n ];\n for (const msg of detail.messages) {\n const role = msg.role === 'assistant' ? 'Assistant' : msg.role === 'user' ? 'User' : msg.role;\n lines.push(`## ${role}`, '');\n const body = typeof msg.content === 'string' ? msg.content : JSON.stringify(msg.content, null, 2);\n lines.push(body, '', '---', '');\n }\n return lines.join('\\n');\n }\n\n async getStats(): Promise<GlobalSessionStats> {\n const list = await this.list({ limit: 100000 });\n const sessions = list.items;\n const byChannel: Record<string, number> = {};\n for (const s of sessions) {\n byChannel[s.sourceChannel] = (byChannel[s.sourceChannel] || 0) + 1;\n }\n let oldestSession: string | undefined;\n let newestSession: string | undefined;\n if (sessions.length > 0) {\n const sorted = [...sessions].sort((a, b) => a.createdAt.localeCompare(b.createdAt));\n oldestSession = sorted[0]!.createdAt;\n newestSession = sorted[sorted.length - 1]!.createdAt;\n }\n return {\n totalSessions: sessions.length,\n activeSessions: sessions.filter((s) => s.status === SessionStatus.ACTIVE || s.status === SessionStatus.IDLE).length,\n archivedSessions: sessions.filter((s) => s.status === SessionStatus.ARCHIVED).length,\n pinnedSessions: sessions.filter((s) => s.status === SessionStatus.PINNED).length,\n totalMessages: sessions.reduce((sum, s) => sum + s.messageCount, 0),\n totalTokens: sessions.reduce((sum, s) => sum + s.estimatedTokens, 0),\n oldestSession,\n newestSession,\n byChannel,\n };\n }\n\n async archiveOld(olderThanDays: number): Promise<number> {\n const cutoff = new Date();\n cutoff.setDate(cutoff.getDate() - olderThanDays);\n const list = await this.list({ limit: 100000 });\n let archived = 0;\n for (const session of list.items) {\n if (session.status !== SessionStatus.ARCHIVED && session.status !== SessionStatus.PINNED) {\n const lastAccess = new Date(session.lastAccessedAt);\n if (lastAccess < cutoff) {\n await this.archive(session.key);\n archived++;\n }\n }\n }\n return archived;\n }\n\n estimateTokens(messages: AgentMessage[]): number {\n let total = 0;\n for (const msg of messages) {\n total += Math.ceil(this.extractTextContent(this.messageContent(msg)).length / 4);\n }\n return total;\n }\n\n private messageContent(msg: AgentMessage): unknown {\n return (msg as { content?: unknown }).content;\n }\n\n private isCompactionSummaryMessage(msg: AgentMessage): boolean {\n if (msg.role !== 'user') {\n return false;\n }\n const text = this.extractTextContent(this.messageContent(msg)).trim();\n return /^\\[Previous conversation summary\\]/i.test(text);\n }\n\n private displayMessageIdentity(message: AgentMessage): string {\n const record = message as unknown as Record<string, unknown>;\n return JSON.stringify({\n role: message.role,\n timestamp: record.timestamp,\n toolCallId: record.toolCallId ?? record.tool_call_id,\n toolName: record.toolName,\n content: this.messageContent(message),\n });\n }\n\n private extractTextContent(content: unknown): string {\n if (typeof content === 'string') {\n return content;\n }\n if (Array.isArray(content)) {\n const parts: string[] = [];\n for (const item of content) {\n if (typeof item !== 'object' || item === null || !('type' in item)) {\n continue;\n }\n const c = item as { type?: string; text?: string; name?: string };\n if (c.type === 'text' && typeof c.text === 'string') {\n parts.push(c.text);\n } else if (c.type === 'toolCall' || c.type === 'tool_use') {\n parts.push(c.name ? `[${c.name}]` : '');\n }\n }\n return parts.join('');\n }\n return '';\n }\n\n private async injectPostCompactionContext(key: string): Promise<void> {\n const contextText = readPostCompactionContext({\n cfg: this.options.config,\n sessionKey: key,\n });\n if (!contextText?.trim()) {\n return;\n }\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const workspaceDir = resolveEffectiveAgentProfileForSession(this.options.config, key).resolvedWorkspacePath;\n try {\n await appendPiTranscriptContextEntry({\n absPath: abs,\n cwd: workspaceDir,\n sessionKey: key,\n entry: {\n kind: 'context',\n id: `post-compaction-${Date.now()}`,\n text: contextText,\n createdAt: new Date().toISOString(),\n },\n });\n } catch (err) {\n log.warn({ err, key }, 'Post-compaction context injection failed');\n }\n }\n\n private convertMessages(messages: AgentMessage[]): Message[] {\n return messages.map((m: AgentMessage & Record<string, unknown>) => {\n const c = this.messageContent(m);\n const content: string | unknown[] =\n typeof c === 'string' ? c : Array.isArray(c) ? c : this.extractTextContent(c);\n const row: Message = {\n role: m.role as Message['role'],\n content,\n timestamp: m.timestamp ? new Date(m.timestamp as string | number).toISOString() : undefined,\n tool_call_id: (m.tool_call_id as string | undefined) || (m.toolCallId as string | undefined),\n tool_calls: m.tool_calls as Message['tool_calls'],\n name: m.name as string | undefined,\n };\n if (Array.isArray(m.attachments) && m.attachments.length > 0) {\n row.attachments = m.attachments as Message['attachments'];\n }\n const rawUsage = m.usage as { input?: number; output?: number; totalTokens?: number; total?: number } | undefined;\n if (rawUsage && typeof rawUsage === 'object') {\n const inputTokens = typeof rawUsage.input === 'number' ? rawUsage.input : undefined;\n const outputTokens = typeof rawUsage.output === 'number' ? rawUsage.output : undefined;\n const totalTokens = typeof rawUsage.totalTokens === 'number'\n ? rawUsage.totalTokens\n : typeof rawUsage.total === 'number'\n ? rawUsage.total\n : undefined;\n if (inputTokens != null || outputTokens != null || totalTokens != null) {\n row.usage = { inputTokens, outputTokens, totalTokens };\n }\n }\n return row;\n });\n }\n}\n"],"mappings":"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"}
@@ -12,8 +12,8 @@ export declare function incrementStats(level: LogLevel, module?: string): void;
12
12
  */
13
13
  export declare function getLogStats(): {
14
14
  byLevel: {
15
- debug: number;
16
15
  error: number;
16
+ debug: number;
17
17
  trace: number;
18
18
  info: number;
19
19
  warn: number;
@@ -37,12 +37,15 @@ export interface AgentStepStartedPayload {
37
37
  stepId: string;
38
38
  label: string;
39
39
  kind: 'tool' | 'llm' | 'thinking';
40
+ toolName?: string;
40
41
  detail?: string;
41
42
  }
42
43
  export interface AgentStepCompletedPayload {
43
44
  agentId: string;
44
45
  stepId: string;
45
46
  status: 'done' | 'error';
47
+ resultPreview?: string;
48
+ error?: string;
46
49
  }
47
50
  export interface AgentCompletedPayload {
48
51
  agentId: string;
@@ -159,8 +159,11 @@ export interface WorkflowAgentStepView {
159
159
  id: string;
160
160
  label: string;
161
161
  kind: 'tool' | 'llm' | 'thinking';
162
+ toolName?: string;
162
163
  detail?: string;
163
164
  status: WorkflowAgentStepStatus;
165
+ resultPreview?: string;
166
+ error?: string;
164
167
  startedAtMs?: number;
165
168
  completedAtMs?: number;
166
169
  }