clawvault 3.1.0 → 3.2.1

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 (273) hide show
  1. package/README.md +422 -141
  2. package/bin/clawvault.js +10 -2
  3. package/bin/command-registration.test.js +3 -1
  4. package/bin/command-runtime.js +9 -1
  5. package/bin/register-core-commands.js +23 -28
  6. package/bin/register-maintenance-commands.js +39 -3
  7. package/bin/register-query-commands.js +58 -29
  8. package/bin/register-tailscale-commands.js +106 -0
  9. package/bin/register-task-commands.js +18 -1
  10. package/bin/register-task-commands.test.js +16 -0
  11. package/bin/register-vault-operations-commands.js +29 -1
  12. package/bin/register-workgraph-commands.js +451 -0
  13. package/dashboard/lib/graph-diff.js +104 -0
  14. package/dashboard/lib/graph-diff.test.js +75 -0
  15. package/dashboard/lib/vault-parser.js +556 -0
  16. package/dashboard/lib/vault-parser.test.js +254 -0
  17. package/dashboard/public/app.js +796 -0
  18. package/dashboard/public/index.html +52 -0
  19. package/dashboard/public/styles.css +221 -0
  20. package/dashboard/server.js +374 -0
  21. package/dist/{chunk-C7OK5WKP.js → chunk-2JQ3O2YL.js} +4 -4
  22. package/dist/{chunk-VR5NE7PZ.js → chunk-2RAZ4ZFE.js} +1 -1
  23. package/dist/{chunk-F2JEUD4J.js → chunk-4ITRXIVT.js} +5 -7
  24. package/dist/{chunk-GUKMRGM7.js → chunk-4OXMU5S2.js} +1 -1
  25. package/dist/chunk-5PJ4STIC.js +465 -0
  26. package/dist/{chunk-62YTUT6J.js → chunk-AZYOKJYC.js} +2 -2
  27. package/dist/chunk-BSJ6RIT7.js +447 -0
  28. package/dist/chunk-ECRZL5XR.js +50 -0
  29. package/dist/chunk-ERNE2FZ5.js +189 -0
  30. package/dist/{chunk-WAZ3NLWL.js → chunk-F55HGNU4.js} +0 -47
  31. package/dist/{chunk-VGLOTGAS.js → chunk-FAKNOB7Y.js} +2 -2
  32. package/dist/{chunk-QK3UCXWL.js → chunk-FHFUXL6G.js} +2 -2
  33. package/dist/chunk-GNJL4YGR.js +79 -0
  34. package/dist/chunk-HR4KN6S2.js +152 -0
  35. package/dist/{chunk-OZ7RIXTO.js → chunk-IIOU45CK.js} +1 -1
  36. package/dist/chunk-IJBFGPCS.js +33 -0
  37. package/dist/chunk-IVRIKYFE.js +520 -0
  38. package/dist/chunk-K7PNYS45.js +93 -0
  39. package/dist/chunk-MDIH26GC.js +183 -0
  40. package/dist/{chunk-LYHGEHXG.js → chunk-MFAWT5O5.js} +0 -1
  41. package/dist/{chunk-H34S76MB.js → chunk-MNPUYCHQ.js} +6 -6
  42. package/dist/chunk-NTOPJI7W.js +207 -0
  43. package/dist/{chunk-QBLMXKF2.js → chunk-OIWVQYQF.js} +1 -1
  44. package/dist/chunk-PG56HX5T.js +154 -0
  45. package/dist/{chunk-LNJA2UGL.js → chunk-PI4WMLMG.js} +7 -84
  46. package/dist/chunk-QMHPQYUV.js +363 -0
  47. package/dist/{chunk-H62BP7RI.js → chunk-QPDDIHXE.js} +209 -43
  48. package/dist/{chunk-N2AXRYLC.js → chunk-QWQ3TIKS.js} +1 -1
  49. package/dist/{chunk-3DHXQHYG.js → chunk-R2MIW5G7.js} +1 -1
  50. package/dist/{chunk-SJSFRIYS.js → chunk-S5OJEGFG.js} +2 -2
  51. package/dist/chunk-SS4B7P7V.js +99 -0
  52. package/dist/chunk-TIGW564L.js +628 -0
  53. package/dist/chunk-U67V476Y.js +35 -0
  54. package/dist/{chunk-JY6FYXIT.js → chunk-UCQAOZHW.js} +6 -11
  55. package/dist/{chunk-ITPEXLHA.js → chunk-URXDAUVH.js} +24 -5
  56. package/dist/chunk-WIOLLGAD.js +190 -0
  57. package/dist/{chunk-3WRJEKN4.js → chunk-WJVWINEM.js} +72 -8
  58. package/dist/chunk-WMGIIABP.js +15 -0
  59. package/dist/{chunk-33UGEQRT.js → chunk-X3SPPUFG.js} +151 -64
  60. package/dist/{chunk-3NSBOUT3.js → chunk-Y3TIJEBP.js} +314 -79
  61. package/dist/chunk-Y6VJKXGL.js +373 -0
  62. package/dist/{chunk-LI4O6NVK.js → chunk-YDWHS4LJ.js} +49 -9
  63. package/dist/{chunk-U55BGUAU.js → chunk-YNIPYN4F.js} +5 -5
  64. package/dist/chunk-YXQCA6B7.js +226 -0
  65. package/dist/cli/index.js +26 -22
  66. package/dist/commands/archive.js +3 -3
  67. package/dist/commands/backlog.js +3 -3
  68. package/dist/commands/blocked.js +3 -3
  69. package/dist/commands/canvas.d.ts +15 -0
  70. package/dist/commands/canvas.js +200 -0
  71. package/dist/commands/checkpoint.js +2 -2
  72. package/dist/commands/compat.js +2 -2
  73. package/dist/commands/context.js +7 -5
  74. package/dist/commands/doctor.d.ts +11 -7
  75. package/dist/commands/doctor.js +16 -14
  76. package/dist/commands/embed.js +5 -6
  77. package/dist/commands/entities.js +2 -2
  78. package/dist/commands/graph.js +3 -3
  79. package/dist/commands/inject.d.ts +1 -1
  80. package/dist/commands/inject.js +4 -5
  81. package/dist/commands/kanban.js +4 -4
  82. package/dist/commands/link.js +2 -2
  83. package/dist/commands/migrate-observations.js +4 -4
  84. package/dist/commands/observe.d.ts +0 -1
  85. package/dist/commands/observe.js +13 -12
  86. package/dist/commands/project.js +5 -5
  87. package/dist/commands/rebuild-embeddings.d.ts +21 -0
  88. package/dist/commands/rebuild-embeddings.js +91 -0
  89. package/dist/commands/rebuild.js +12 -11
  90. package/dist/commands/recover.js +3 -3
  91. package/dist/commands/reflect.js +6 -7
  92. package/dist/commands/repair-session.js +1 -1
  93. package/dist/commands/replay.js +14 -14
  94. package/dist/commands/session-recap.js +1 -1
  95. package/dist/commands/setup.d.ts +2 -89
  96. package/dist/commands/setup.js +3 -21
  97. package/dist/commands/shell-init.js +1 -1
  98. package/dist/commands/sleep.d.ts +1 -1
  99. package/dist/commands/sleep.js +18 -17
  100. package/dist/commands/status.d.ts +2 -0
  101. package/dist/commands/status.js +40 -30
  102. package/dist/commands/sync-bd.d.ts +10 -0
  103. package/dist/commands/sync-bd.js +10 -0
  104. package/dist/commands/tailscale.d.ts +52 -0
  105. package/dist/commands/tailscale.js +26 -0
  106. package/dist/commands/task.js +4 -4
  107. package/dist/commands/template.js +2 -2
  108. package/dist/commands/wake.d.ts +1 -1
  109. package/dist/commands/wake.js +11 -10
  110. package/dist/index.d.ts +334 -191
  111. package/dist/index.js +432 -108
  112. package/dist/{inject-Bzi5E-By.d.ts → inject-DYUrDqQO.d.ts} +3 -3
  113. package/dist/ledger-B7g7jhqG.d.ts +44 -0
  114. package/dist/lib/auto-linker.js +1 -1
  115. package/dist/lib/canvas-layout.d.ts +115 -0
  116. package/dist/lib/canvas-layout.js +35 -0
  117. package/dist/lib/config.d.ts +27 -3
  118. package/dist/lib/config.js +4 -2
  119. package/dist/lib/entity-index.js +1 -1
  120. package/dist/lib/project-utils.js +4 -4
  121. package/dist/lib/session-repair.js +1 -1
  122. package/dist/lib/session-utils.js +1 -1
  123. package/dist/lib/tailscale.d.ts +225 -0
  124. package/dist/lib/tailscale.js +50 -0
  125. package/dist/lib/task-utils.js +3 -3
  126. package/dist/lib/template-engine.js +1 -1
  127. package/dist/lib/webdav.d.ts +109 -0
  128. package/dist/lib/webdav.js +35 -0
  129. package/dist/plugin/index.d.ts +344 -28
  130. package/dist/plugin/index.js +3919 -227
  131. package/dist/registry-BR4326o0.d.ts +30 -0
  132. package/dist/store-CA-6sKCJ.d.ts +34 -0
  133. package/dist/thread-B9LhXNU0.d.ts +41 -0
  134. package/dist/{types-Y2_Um2Ls.d.ts → types-BbWJoC1c.d.ts} +1 -44
  135. package/dist/workgraph/index.d.ts +5 -0
  136. package/dist/workgraph/index.js +23 -0
  137. package/dist/workgraph/ledger.d.ts +2 -0
  138. package/dist/workgraph/ledger.js +25 -0
  139. package/dist/workgraph/registry.d.ts +2 -0
  140. package/dist/workgraph/registry.js +19 -0
  141. package/dist/workgraph/store.d.ts +2 -0
  142. package/dist/workgraph/store.js +25 -0
  143. package/dist/workgraph/thread.d.ts +2 -0
  144. package/dist/workgraph/thread.js +25 -0
  145. package/dist/workgraph/types.d.ts +54 -0
  146. package/dist/workgraph/types.js +7 -0
  147. package/hooks/clawvault/HOOK.md +113 -0
  148. package/hooks/clawvault/handler.js +1559 -0
  149. package/hooks/clawvault/handler.test.js +510 -0
  150. package/hooks/clawvault/openclaw.plugin.json +72 -0
  151. package/openclaw.plugin.json +235 -30
  152. package/package.json +20 -20
  153. package/dist/chunk-3RG5ZIWI.js +0 -10
  154. package/dist/chunk-3ZIH425O.js +0 -871
  155. package/dist/chunk-6U6MK36V.js +0 -205
  156. package/dist/chunk-CMB7UL7C.js +0 -327
  157. package/dist/chunk-D2H45LON.js +0 -1074
  158. package/dist/chunk-E7MFQB6D.js +0 -163
  159. package/dist/chunk-GQSLDZTS.js +0 -560
  160. package/dist/chunk-MFM6K7PU.js +0 -374
  161. package/dist/chunk-MXSSG3QU.js +0 -42
  162. package/dist/chunk-OCGVIN3L.js +0 -88
  163. package/dist/chunk-PAH27GSN.js +0 -108
  164. package/dist/chunk-YCUNCH2I.js +0 -78
  165. package/dist/cli/index.cjs +0 -8584
  166. package/dist/cli/index.d.cts +0 -5
  167. package/dist/commands/archive.cjs +0 -287
  168. package/dist/commands/archive.d.cts +0 -11
  169. package/dist/commands/backlog.cjs +0 -721
  170. package/dist/commands/backlog.d.cts +0 -53
  171. package/dist/commands/blocked.cjs +0 -204
  172. package/dist/commands/blocked.d.cts +0 -26
  173. package/dist/commands/checkpoint.cjs +0 -244
  174. package/dist/commands/checkpoint.d.cts +0 -41
  175. package/dist/commands/compat.cjs +0 -294
  176. package/dist/commands/compat.d.cts +0 -28
  177. package/dist/commands/context.cjs +0 -2990
  178. package/dist/commands/context.d.cts +0 -2
  179. package/dist/commands/doctor.cjs +0 -2986
  180. package/dist/commands/doctor.d.cts +0 -21
  181. package/dist/commands/embed.cjs +0 -232
  182. package/dist/commands/embed.d.cts +0 -17
  183. package/dist/commands/entities.cjs +0 -141
  184. package/dist/commands/entities.d.cts +0 -7
  185. package/dist/commands/graph.cjs +0 -501
  186. package/dist/commands/graph.d.cts +0 -21
  187. package/dist/commands/inject.cjs +0 -1636
  188. package/dist/commands/inject.d.cts +0 -2
  189. package/dist/commands/kanban.cjs +0 -884
  190. package/dist/commands/kanban.d.cts +0 -63
  191. package/dist/commands/link.cjs +0 -965
  192. package/dist/commands/link.d.cts +0 -11
  193. package/dist/commands/migrate-observations.cjs +0 -362
  194. package/dist/commands/migrate-observations.d.cts +0 -19
  195. package/dist/commands/observe.cjs +0 -4099
  196. package/dist/commands/observe.d.cts +0 -23
  197. package/dist/commands/project.cjs +0 -1341
  198. package/dist/commands/project.d.cts +0 -85
  199. package/dist/commands/rebuild.cjs +0 -3136
  200. package/dist/commands/rebuild.d.cts +0 -11
  201. package/dist/commands/recover.cjs +0 -361
  202. package/dist/commands/recover.d.cts +0 -38
  203. package/dist/commands/reflect.cjs +0 -1008
  204. package/dist/commands/reflect.d.cts +0 -11
  205. package/dist/commands/repair-session.cjs +0 -457
  206. package/dist/commands/repair-session.d.cts +0 -38
  207. package/dist/commands/replay.cjs +0 -4103
  208. package/dist/commands/replay.d.cts +0 -16
  209. package/dist/commands/session-recap.cjs +0 -353
  210. package/dist/commands/session-recap.d.cts +0 -27
  211. package/dist/commands/setup.cjs +0 -1278
  212. package/dist/commands/setup.d.cts +0 -99
  213. package/dist/commands/shell-init.cjs +0 -75
  214. package/dist/commands/shell-init.d.cts +0 -7
  215. package/dist/commands/sleep.cjs +0 -6029
  216. package/dist/commands/sleep.d.cts +0 -36
  217. package/dist/commands/status.cjs +0 -2737
  218. package/dist/commands/status.d.cts +0 -52
  219. package/dist/commands/task.cjs +0 -1236
  220. package/dist/commands/task.d.cts +0 -97
  221. package/dist/commands/template.cjs +0 -457
  222. package/dist/commands/template.d.cts +0 -36
  223. package/dist/commands/wake.cjs +0 -2627
  224. package/dist/commands/wake.d.cts +0 -22
  225. package/dist/context-BUGaWpyL.d.cts +0 -46
  226. package/dist/index.cjs +0 -12373
  227. package/dist/index.d.cts +0 -854
  228. package/dist/inject-Bzi5E-By.d.cts +0 -137
  229. package/dist/lib/auto-linker.cjs +0 -176
  230. package/dist/lib/auto-linker.d.cts +0 -26
  231. package/dist/lib/config.cjs +0 -78
  232. package/dist/lib/config.d.cts +0 -11
  233. package/dist/lib/entity-index.cjs +0 -84
  234. package/dist/lib/entity-index.d.cts +0 -26
  235. package/dist/lib/project-utils.cjs +0 -864
  236. package/dist/lib/project-utils.d.cts +0 -97
  237. package/dist/lib/session-repair.cjs +0 -239
  238. package/dist/lib/session-repair.d.cts +0 -110
  239. package/dist/lib/session-utils.cjs +0 -209
  240. package/dist/lib/session-utils.d.cts +0 -63
  241. package/dist/lib/task-utils.cjs +0 -1137
  242. package/dist/lib/task-utils.d.cts +0 -208
  243. package/dist/lib/template-engine.cjs +0 -47
  244. package/dist/lib/template-engine.d.cts +0 -11
  245. package/dist/plugin/index.cjs +0 -1907
  246. package/dist/plugin/index.d.cts +0 -36
  247. package/dist/plugin/inject.cjs +0 -356
  248. package/dist/plugin/inject.d.cts +0 -54
  249. package/dist/plugin/inject.d.ts +0 -54
  250. package/dist/plugin/inject.js +0 -17
  251. package/dist/plugin/observe.cjs +0 -631
  252. package/dist/plugin/observe.d.cts +0 -39
  253. package/dist/plugin/observe.d.ts +0 -39
  254. package/dist/plugin/observe.js +0 -18
  255. package/dist/plugin/templates.cjs +0 -593
  256. package/dist/plugin/templates.d.cts +0 -52
  257. package/dist/plugin/templates.d.ts +0 -52
  258. package/dist/plugin/templates.js +0 -25
  259. package/dist/plugin/types.cjs +0 -18
  260. package/dist/plugin/types.d.cts +0 -209
  261. package/dist/plugin/types.d.ts +0 -209
  262. package/dist/plugin/types.js +0 -0
  263. package/dist/plugin/vault.cjs +0 -927
  264. package/dist/plugin/vault.d.cts +0 -68
  265. package/dist/plugin/vault.d.ts +0 -68
  266. package/dist/plugin/vault.js +0 -22
  267. package/dist/types-Y2_Um2Ls.d.cts +0 -205
  268. package/templates/memory-event.md +0 -67
  269. package/templates/party.md +0 -63
  270. package/templates/primitive-registry.yaml +0 -551
  271. package/templates/run.md +0 -68
  272. package/templates/trigger.md +0 -68
  273. package/templates/workspace.md +0 -50
@@ -1,16 +0,0 @@
1
- import { Command } from 'commander';
2
-
3
- type ReplaySource = 'chatgpt' | 'claude' | 'opencode' | 'openclaw';
4
-
5
- interface ReplayCommandOptions {
6
- source: ReplaySource;
7
- inputPath: string;
8
- from?: string;
9
- to?: string;
10
- dryRun?: boolean;
11
- vaultPath?: string;
12
- }
13
- declare function replayCommand(options: ReplayCommandOptions): Promise<void>;
14
- declare function registerReplayCommand(program: Command): void;
15
-
16
- export { type ReplayCommandOptions, registerReplayCommand, replayCommand };
@@ -1,353 +0,0 @@
1
- "use strict";
2
- var __create = Object.create;
3
- var __defProp = Object.defineProperty;
4
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
- var __getOwnPropNames = Object.getOwnPropertyNames;
6
- var __getProtoOf = Object.getPrototypeOf;
7
- var __hasOwnProp = Object.prototype.hasOwnProperty;
8
- var __export = (target, all) => {
9
- for (var name in all)
10
- __defProp(target, name, { get: all[name], enumerable: true });
11
- };
12
- var __copyProps = (to, from, except, desc) => {
13
- if (from && typeof from === "object" || typeof from === "function") {
14
- for (let key of __getOwnPropNames(from))
15
- if (!__hasOwnProp.call(to, key) && key !== except)
16
- __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
- }
18
- return to;
19
- };
20
- var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
- // If the importer is in node compatibility mode or this is not an ESM
22
- // file that has been converted to a CommonJS file using a Babel-
23
- // compatible transform (i.e. "__esModule" has not been set), then set
24
- // "default" to the CommonJS "module.exports" for node compatibility.
25
- isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
- mod
27
- ));
28
- var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
-
30
- // src/commands/session-recap.ts
31
- var session_recap_exports = {};
32
- __export(session_recap_exports, {
33
- buildSessionRecap: () => buildSessionRecap,
34
- formatSessionRecapMarkdown: () => formatSessionRecapMarkdown,
35
- sessionRecapCommand: () => sessionRecapCommand
36
- });
37
- module.exports = __toCommonJS(session_recap_exports);
38
- var fs2 = __toESM(require("fs"), 1);
39
- var path2 = __toESM(require("path"), 1);
40
-
41
- // src/lib/session-utils.ts
42
- var fs = __toESM(require("fs"), 1);
43
- var path = __toESM(require("path"), 1);
44
- var os = __toESM(require("os"), 1);
45
- function validateEnvPath(envValue) {
46
- if (!envValue) return null;
47
- const trimmed = envValue.trim();
48
- if (!trimmed) return null;
49
- const resolved = path.resolve(trimmed);
50
- if (!path.isAbsolute(resolved)) return null;
51
- return resolved;
52
- }
53
- function getOpenClawDir() {
54
- const customHome = validateEnvPath(process.env.OPENCLAW_HOME);
55
- if (customHome) {
56
- return customHome;
57
- }
58
- return path.join(os.homedir(), ".openclaw");
59
- }
60
- function getOpenClawAgentsDir() {
61
- const stateDir = validateEnvPath(process.env.OPENCLAW_STATE_DIR);
62
- if (stateDir) {
63
- return path.join(stateDir, "agents");
64
- }
65
- return path.join(getOpenClawDir(), "agents");
66
- }
67
- function getSessionsDir(agentId) {
68
- return path.join(getOpenClawAgentsDir(), agentId, "sessions");
69
- }
70
- function getSessionsJsonPath(agentId) {
71
- return path.join(getSessionsDir(agentId), "sessions.json");
72
- }
73
- function getSessionFilePath(agentId, sessionId) {
74
- return path.join(getSessionsDir(agentId), `${sessionId}.jsonl`);
75
- }
76
- function loadSessionsStore(agentId) {
77
- const sessionsJsonPath = getSessionsJsonPath(agentId);
78
- if (!fs.existsSync(sessionsJsonPath)) {
79
- return null;
80
- }
81
- try {
82
- const content = fs.readFileSync(sessionsJsonPath, "utf-8");
83
- return JSON.parse(content);
84
- } catch {
85
- return null;
86
- }
87
- }
88
-
89
- // src/commands/session-recap.ts
90
- var DEFAULT_LIMIT = 15;
91
- var MAX_LIMIT = 50;
92
- var READ_CHUNK_SIZE = 64 * 1024;
93
- var MAX_TURN_TEXT_LENGTH = 420;
94
- var MAX_TOTAL_TEXT_LENGTH = 12e3;
95
- var SESSION_KEY_PATTERN = /^agent:[a-zA-Z0-9_-]+:[a-zA-Z0-9:_-]+$/;
96
- var AGENT_ID_PATTERN = /^[a-zA-Z0-9_-]{1,100}$/;
97
- function sanitizeSessionKey(input) {
98
- const sessionKey = input.trim();
99
- if (!SESSION_KEY_PATTERN.test(sessionKey)) {
100
- throw new Error("Invalid session key. Expected format: agent:<agentId>:<scope>");
101
- }
102
- return sessionKey;
103
- }
104
- function sanitizeAgentId(input) {
105
- const agentId = input.trim();
106
- if (!AGENT_ID_PATTERN.test(agentId)) {
107
- throw new Error('Invalid agent ID. Use letters, numbers, "_" or "-".');
108
- }
109
- return agentId;
110
- }
111
- function parseAgentIdFromSessionKey(sessionKey) {
112
- const match = /^agent:([^:]+):/.exec(sessionKey);
113
- if (!match?.[1]) return null;
114
- return sanitizeAgentId(match[1]);
115
- }
116
- function resolveAgentId(sessionKey, explicitAgentId) {
117
- if (explicitAgentId) {
118
- return sanitizeAgentId(explicitAgentId);
119
- }
120
- const fromSessionKey = parseAgentIdFromSessionKey(sessionKey);
121
- if (fromSessionKey) return fromSessionKey;
122
- const fromEnv = process.env.OPENCLAW_AGENT_ID;
123
- if (fromEnv) return sanitizeAgentId(fromEnv);
124
- return "clawdious";
125
- }
126
- function normalizeLimit(limit) {
127
- if (limit === void 0 || Number.isNaN(limit)) return DEFAULT_LIMIT;
128
- const parsed = Math.floor(limit);
129
- return Math.min(MAX_LIMIT, Math.max(1, parsed));
130
- }
131
- function isPathInside(parentPath, candidatePath) {
132
- const normalizedParent = parentPath.endsWith(path2.sep) ? parentPath : `${parentPath}${path2.sep}`;
133
- return candidatePath.startsWith(normalizedParent);
134
- }
135
- function resolveSafeTranscriptPath(agentId, sessionId, sessionFile) {
136
- const sessionsDir = getSessionsDir(agentId);
137
- if (!fs2.existsSync(sessionsDir)) {
138
- throw new Error(`Sessions directory not found for agent "${agentId}".`);
139
- }
140
- const sessionsDirRealPath = fs2.realpathSync(sessionsDir);
141
- const candidatePaths = [];
142
- if (typeof sessionFile === "string" && sessionFile.trim()) {
143
- candidatePaths.push(path2.resolve(sessionFile));
144
- }
145
- candidatePaths.push(getSessionFilePath(agentId, sessionId));
146
- for (const candidatePath of candidatePaths) {
147
- if (path2.extname(candidatePath).toLowerCase() !== ".jsonl") continue;
148
- if (!fs2.existsSync(candidatePath)) continue;
149
- let candidateRealPath = "";
150
- try {
151
- candidateRealPath = fs2.realpathSync(candidatePath);
152
- } catch {
153
- continue;
154
- }
155
- if (!isPathInside(sessionsDirRealPath, candidateRealPath)) {
156
- continue;
157
- }
158
- const stat = fs2.statSync(candidateRealPath);
159
- if (!stat.isFile()) continue;
160
- return candidateRealPath;
161
- }
162
- throw new Error(`No valid transcript found for session "${sessionId}".`);
163
- }
164
- function getSessionStoreEntry(agentId, sessionKey) {
165
- const store = loadSessionsStore(agentId);
166
- if (!store) {
167
- throw new Error(`Could not load sessions store for agent "${agentId}".`);
168
- }
169
- const entry = store[sessionKey];
170
- if (!entry) {
171
- throw new Error(`Session key not found: ${sessionKey}`);
172
- }
173
- if (typeof entry.sessionId !== "string" || !entry.sessionId.trim()) {
174
- throw new Error(`Invalid session mapping for "${sessionKey}" (missing sessionId).`);
175
- }
176
- return entry;
177
- }
178
- function sanitizeText(input) {
179
- return input.replace(/[\x00-\x1f\x7f]/g, " ").replace(/\s+/g, " ").trim();
180
- }
181
- function truncateText(input, maxLength) {
182
- if (input.length <= maxLength) return input;
183
- return `${input.slice(0, Math.max(0, maxLength - 3)).trimEnd()}...`;
184
- }
185
- function extractTextFromContent(content) {
186
- if (typeof content === "string") {
187
- const cleaned = sanitizeText(content);
188
- return truncateText(cleaned, MAX_TURN_TEXT_LENGTH);
189
- }
190
- if (Array.isArray(content)) {
191
- const parts = [];
192
- for (const part of content) {
193
- if (typeof part === "string") {
194
- const cleaned2 = sanitizeText(part);
195
- if (cleaned2) parts.push(cleaned2);
196
- continue;
197
- }
198
- if (!part || typeof part !== "object") continue;
199
- const block = part;
200
- const blockType = typeof block.type === "string" ? block.type.toLowerCase() : "";
201
- if (blockType.includes("tool") || blockType.includes("thinking") || blockType.includes("reason")) {
202
- continue;
203
- }
204
- const blockText = typeof block.text === "string" ? block.text : typeof block.content === "string" && blockType.includes("text") ? block.content : "";
205
- const cleaned = sanitizeText(blockText);
206
- if (cleaned) parts.push(cleaned);
207
- }
208
- return truncateText(parts.join(" "), MAX_TURN_TEXT_LENGTH);
209
- }
210
- return "";
211
- }
212
- function parseTurnFromLine(line) {
213
- const trimmed = line.trim();
214
- if (!trimmed) return null;
215
- let parsed;
216
- try {
217
- parsed = JSON.parse(trimmed);
218
- } catch {
219
- return null;
220
- }
221
- if (!parsed || typeof parsed !== "object") return null;
222
- const entry = parsed;
223
- if (entry.type !== "message" || !entry.message || typeof entry.message !== "object") {
224
- return null;
225
- }
226
- const message = entry.message;
227
- const role = typeof message.role === "string" ? message.role.toLowerCase() : "";
228
- if (role !== "user" && role !== "assistant") return null;
229
- const text = extractTextFromContent(message.content);
230
- if (!text) return null;
231
- return { role, text };
232
- }
233
- function applyOutputBudget(turns) {
234
- let remaining = MAX_TOTAL_TEXT_LENGTH;
235
- const selected = [];
236
- for (let i = turns.length - 1; i >= 0; i -= 1) {
237
- if (remaining <= 0) break;
238
- const current = turns[i];
239
- let text = current.text;
240
- if (text.length > remaining) {
241
- if (remaining < 16) break;
242
- text = truncateText(text, remaining);
243
- }
244
- selected.push({ role: current.role, text });
245
- remaining -= text.length;
246
- }
247
- return selected.reverse();
248
- }
249
- function readRecentTurnsFromTranscript(filePath, limit) {
250
- if (limit <= 0) return [];
251
- const fileHandle = fs2.openSync(filePath, "r");
252
- const collected = [];
253
- let remainder = "";
254
- try {
255
- let position = fs2.fstatSync(fileHandle).size;
256
- while (position > 0 && collected.length < limit) {
257
- const readSize = Math.min(READ_CHUNK_SIZE, position);
258
- position -= readSize;
259
- const buffer = Buffer.allocUnsafe(readSize);
260
- fs2.readSync(fileHandle, buffer, 0, readSize, position);
261
- const chunk = buffer.toString("utf-8");
262
- const text = chunk + remainder;
263
- const lines = text.split("\n");
264
- remainder = lines.shift() ?? "";
265
- for (let lineIndex = lines.length - 1; lineIndex >= 0; lineIndex -= 1) {
266
- if (collected.length >= limit) break;
267
- const turn = parseTurnFromLine(lines[lineIndex]);
268
- if (turn) collected.push(turn);
269
- }
270
- }
271
- if (position === 0 && remainder && collected.length < limit) {
272
- const turn = parseTurnFromLine(remainder);
273
- if (turn) collected.push(turn);
274
- }
275
- } finally {
276
- fs2.closeSync(fileHandle);
277
- }
278
- return applyOutputBudget(collected.reverse());
279
- }
280
- function toSessionLabel(sessionKey, agentId) {
281
- const normalizedPrefix = `agent:${agentId}:`;
282
- if (sessionKey.startsWith(normalizedPrefix)) {
283
- return sessionKey.slice(normalizedPrefix.length) || sessionKey;
284
- }
285
- const parts = sessionKey.split(":");
286
- if (parts[0] === "agent" && parts.length > 2) {
287
- return parts.slice(2).join(":");
288
- }
289
- return sessionKey;
290
- }
291
- function formatSessionRecapMarkdown(result) {
292
- let output = `## Session Recap: ${result.sessionLabel}
293
-
294
- `;
295
- output += `### Recent Conversation (last ${result.count} messages)
296
- `;
297
- if (result.messages.length === 0) {
298
- output += "_No recent user/assistant messages found._\n";
299
- return output.trimEnd();
300
- }
301
- for (const message of result.messages) {
302
- const label = message.role === "user" ? "User" : "Assistant";
303
- output += `**${label}:** ${message.text}
304
-
305
- `;
306
- }
307
- return output.trimEnd();
308
- }
309
- async function buildSessionRecap(sessionKeyInput, options = {}) {
310
- const sessionKey = sanitizeSessionKey(sessionKeyInput);
311
- const agentId = resolveAgentId(sessionKey, options.agentId);
312
- const limit = normalizeLimit(options.limit);
313
- const entry = getSessionStoreEntry(agentId, sessionKey);
314
- const sessionId = String(entry.sessionId).trim();
315
- const transcriptPath = resolveSafeTranscriptPath(agentId, sessionId, entry.sessionFile);
316
- const messages = readRecentTurnsFromTranscript(transcriptPath, limit);
317
- const result = {
318
- sessionKey,
319
- sessionLabel: toSessionLabel(sessionKey, agentId),
320
- agentId,
321
- sessionId,
322
- transcriptPath,
323
- generated: (/* @__PURE__ */ new Date()).toISOString(),
324
- count: messages.length,
325
- messages,
326
- markdown: ""
327
- };
328
- result.markdown = formatSessionRecapMarkdown(result);
329
- return result;
330
- }
331
- async function sessionRecapCommand(sessionKey, options = {}) {
332
- const result = await buildSessionRecap(sessionKey, options);
333
- const format = options.format ?? "markdown";
334
- if (format === "json") {
335
- console.log(JSON.stringify({
336
- sessionKey: result.sessionKey,
337
- sessionLabel: result.sessionLabel,
338
- agentId: result.agentId,
339
- sessionId: result.sessionId,
340
- generated: result.generated,
341
- count: result.count,
342
- messages: result.messages
343
- }, null, 2));
344
- return;
345
- }
346
- console.log(result.markdown);
347
- }
348
- // Annotate the CommonJS export names for ESM import in node:
349
- 0 && (module.exports = {
350
- buildSessionRecap,
351
- formatSessionRecapMarkdown,
352
- sessionRecapCommand
353
- });
@@ -1,27 +0,0 @@
1
- type SessionRecapFormat = 'markdown' | 'json';
2
- type SessionRole = 'user' | 'assistant';
3
- interface SessionRecapOptions {
4
- limit?: number;
5
- format?: SessionRecapFormat;
6
- agentId?: string;
7
- }
8
- interface SessionTurn {
9
- role: SessionRole;
10
- text: string;
11
- }
12
- interface SessionRecapResult {
13
- sessionKey: string;
14
- sessionLabel: string;
15
- agentId: string;
16
- sessionId: string;
17
- transcriptPath: string;
18
- generated: string;
19
- count: number;
20
- messages: SessionTurn[];
21
- markdown: string;
22
- }
23
- declare function formatSessionRecapMarkdown(result: SessionRecapResult): string;
24
- declare function buildSessionRecap(sessionKeyInput: string, options?: SessionRecapOptions): Promise<SessionRecapResult>;
25
- declare function sessionRecapCommand(sessionKey: string, options?: SessionRecapOptions): Promise<void>;
26
-
27
- export { type SessionRecapFormat, type SessionRecapOptions, type SessionRecapResult, type SessionRole, type SessionTurn, buildSessionRecap, formatSessionRecapMarkdown, sessionRecapCommand };