clawvault 2.6.0 → 3.0.0

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 (232) hide show
  1. package/bin/command-registration.test.js +1 -3
  2. package/bin/register-core-commands.js +10 -23
  3. package/bin/register-maintenance-commands.js +3 -20
  4. package/bin/register-query-commands.js +23 -0
  5. package/bin/register-task-commands.js +1 -18
  6. package/bin/register-task-commands.test.js +0 -16
  7. package/bin/register-vault-operations-commands.js +1 -29
  8. package/dist/{chunk-QVMXF7FY.js → chunk-3D6BCTP6.js} +39 -1
  9. package/dist/{chunk-R2MIW5G7.js → chunk-3DHXQHYG.js} +1 -1
  10. package/dist/{chunk-Q2J5YTUF.js → chunk-3NSBOUT3.js} +73 -36
  11. package/dist/chunk-3RG5ZIWI.js +10 -0
  12. package/dist/{chunk-AZYOKJYC.js → chunk-62YTUT6J.js} +2 -2
  13. package/dist/chunk-6U6MK36V.js +205 -0
  14. package/dist/{chunk-4QYGFWRM.js → chunk-7R7O6STJ.js} +4 -4
  15. package/dist/{chunk-VXEOHTSL.js → chunk-C7OK5WKP.js} +4 -4
  16. package/dist/chunk-CMB7UL7C.js +327 -0
  17. package/dist/chunk-DEFFDRVP.js +938 -0
  18. package/dist/{chunk-K3CDT7IH.js → chunk-E7MFQB6D.js} +61 -20
  19. package/dist/{chunk-ME37YNW3.js → chunk-F2JEUD4J.js} +6 -4
  20. package/dist/chunk-GAJV4IGR.js +82 -0
  21. package/dist/chunk-GQSLDZTS.js +560 -0
  22. package/dist/{chunk-4OXMU5S2.js → chunk-GUKMRGM7.js} +1 -1
  23. package/dist/{chunk-YOSEUUNB.js → chunk-H34S76MB.js} +6 -6
  24. package/dist/{chunk-4TE4JMLA.js → chunk-JY6FYXIT.js} +10 -5
  25. package/dist/chunk-K234IDRJ.js +1073 -0
  26. package/dist/{chunk-IEVLHNLU.js → chunk-LNJA2UGL.js} +86 -9
  27. package/dist/{chunk-MFAWT5O5.js → chunk-LYHGEHXG.js} +1 -0
  28. package/dist/chunk-MFM6K7PU.js +374 -0
  29. package/dist/{chunk-QWQ3TIKS.js → chunk-N2AXRYLC.js} +1 -1
  30. package/dist/chunk-PAH27GSN.js +108 -0
  31. package/dist/{chunk-OIWVQYQF.js → chunk-QBLMXKF2.js} +1 -1
  32. package/dist/{chunk-FHFUXL6G.js → chunk-QK3UCXWL.js} +2 -2
  33. package/dist/{chunk-2YDBJS7M.js → chunk-SJSFRIYS.js} +1 -1
  34. package/dist/{chunk-GSD4ALSI.js → chunk-U55BGUAU.js} +2 -2
  35. package/dist/{chunk-PBEE567J.js → chunk-VGLOTGAS.js} +1 -1
  36. package/dist/{chunk-F55HGNU4.js → chunk-WAZ3NLWL.js} +47 -0
  37. package/dist/{chunk-KL4NAOMO.js → chunk-WGRQ6HDV.js} +1 -1
  38. package/dist/{chunk-UEOUADMO.js → chunk-YKTA5JOJ.js} +13 -10
  39. package/dist/{chunk-XAVB4GB4.js → chunk-ZVVFWOLW.js} +4 -4
  40. package/dist/cli/index.cjs +10033 -0
  41. package/dist/cli/index.d.cts +5 -0
  42. package/dist/cli/index.js +20 -18
  43. package/dist/commands/archive.cjs +287 -0
  44. package/dist/commands/archive.d.cts +11 -0
  45. package/dist/commands/archive.js +1 -0
  46. package/dist/commands/backlog.cjs +721 -0
  47. package/dist/commands/backlog.d.cts +53 -0
  48. package/dist/commands/backlog.js +3 -2
  49. package/dist/commands/blocked.cjs +204 -0
  50. package/dist/commands/blocked.d.cts +26 -0
  51. package/dist/commands/blocked.js +3 -2
  52. package/dist/commands/checkpoint.cjs +244 -0
  53. package/dist/commands/checkpoint.d.cts +41 -0
  54. package/dist/commands/checkpoint.js +2 -1
  55. package/dist/commands/compat.cjs +369 -0
  56. package/dist/commands/compat.d.cts +28 -0
  57. package/dist/commands/compat.js +2 -1
  58. package/dist/commands/context.cjs +2989 -0
  59. package/dist/commands/context.d.cts +2 -0
  60. package/dist/commands/context.js +5 -4
  61. package/dist/commands/doctor.cjs +3062 -0
  62. package/dist/commands/doctor.d.cts +21 -0
  63. package/dist/commands/doctor.d.ts +6 -1
  64. package/dist/commands/doctor.js +13 -11
  65. package/dist/commands/embed.cjs +232 -0
  66. package/dist/commands/embed.d.cts +17 -0
  67. package/dist/commands/embed.js +5 -2
  68. package/dist/commands/entities.cjs +141 -0
  69. package/dist/commands/entities.d.cts +7 -0
  70. package/dist/commands/entities.js +1 -0
  71. package/dist/commands/graph.cjs +501 -0
  72. package/dist/commands/graph.d.cts +21 -0
  73. package/dist/commands/graph.js +1 -0
  74. package/dist/commands/inject.cjs +1636 -0
  75. package/dist/commands/inject.d.cts +2 -0
  76. package/dist/commands/inject.d.ts +1 -1
  77. package/dist/commands/inject.js +4 -2
  78. package/dist/commands/kanban.cjs +884 -0
  79. package/dist/commands/kanban.d.cts +63 -0
  80. package/dist/commands/kanban.js +4 -3
  81. package/dist/commands/link.cjs +965 -0
  82. package/dist/commands/link.d.cts +11 -0
  83. package/dist/commands/link.js +1 -0
  84. package/dist/commands/migrate-observations.cjs +362 -0
  85. package/dist/commands/migrate-observations.d.cts +19 -0
  86. package/dist/commands/migrate-observations.js +3 -2
  87. package/dist/commands/observe.cjs +4099 -0
  88. package/dist/commands/observe.d.cts +23 -0
  89. package/dist/commands/observe.d.ts +1 -0
  90. package/dist/commands/observe.js +11 -9
  91. package/dist/commands/project.cjs +1341 -0
  92. package/dist/commands/project.d.cts +85 -0
  93. package/dist/commands/project.js +5 -4
  94. package/dist/commands/rebuild.cjs +3136 -0
  95. package/dist/commands/rebuild.d.cts +11 -0
  96. package/dist/commands/rebuild.js +10 -8
  97. package/dist/commands/recover.cjs +361 -0
  98. package/dist/commands/recover.d.cts +38 -0
  99. package/dist/commands/recover.js +3 -2
  100. package/dist/commands/reflect.cjs +1008 -0
  101. package/dist/commands/reflect.d.cts +11 -0
  102. package/dist/commands/reflect.js +6 -4
  103. package/dist/commands/repair-session.cjs +457 -0
  104. package/dist/commands/repair-session.d.cts +38 -0
  105. package/dist/commands/repair-session.js +1 -0
  106. package/dist/commands/replay.cjs +4103 -0
  107. package/dist/commands/replay.d.cts +16 -0
  108. package/dist/commands/replay.js +12 -10
  109. package/dist/commands/session-recap.cjs +353 -0
  110. package/dist/commands/session-recap.d.cts +27 -0
  111. package/dist/commands/session-recap.js +1 -0
  112. package/dist/commands/setup.cjs +1345 -0
  113. package/dist/commands/setup.d.cts +100 -0
  114. package/dist/commands/setup.d.ts +90 -2
  115. package/dist/commands/setup.js +21 -2
  116. package/dist/commands/shell-init.cjs +75 -0
  117. package/dist/commands/shell-init.d.cts +7 -0
  118. package/dist/commands/shell-init.js +2 -0
  119. package/dist/commands/sleep.cjs +6028 -0
  120. package/dist/commands/sleep.d.cts +36 -0
  121. package/dist/commands/sleep.d.ts +1 -1
  122. package/dist/commands/sleep.js +17 -15
  123. package/dist/commands/status.cjs +2736 -0
  124. package/dist/commands/status.d.cts +52 -0
  125. package/dist/commands/status.js +12 -10
  126. package/dist/commands/tailscale.cjs +1532 -0
  127. package/dist/commands/tailscale.d.cts +52 -0
  128. package/dist/commands/tailscale.js +1 -0
  129. package/dist/commands/task.cjs +1236 -0
  130. package/dist/commands/task.d.cts +97 -0
  131. package/dist/commands/task.js +4 -3
  132. package/dist/commands/template.cjs +457 -0
  133. package/dist/commands/template.d.cts +36 -0
  134. package/dist/commands/template.js +2 -1
  135. package/dist/commands/wake.cjs +2626 -0
  136. package/dist/commands/wake.d.cts +22 -0
  137. package/dist/commands/wake.d.ts +1 -1
  138. package/dist/commands/wake.js +12 -11
  139. package/dist/context-BUGaWpyL.d.cts +46 -0
  140. package/dist/index.cjs +14526 -0
  141. package/dist/index.d.cts +858 -0
  142. package/dist/index.d.ts +192 -7
  143. package/dist/index.js +101 -75
  144. package/dist/{inject-x65KXWPk.d.ts → inject-Bzi5E-By.d.cts} +1 -1
  145. package/dist/inject-Bzi5E-By.d.ts +137 -0
  146. package/dist/lib/auto-linker.cjs +176 -0
  147. package/dist/lib/auto-linker.d.cts +26 -0
  148. package/dist/lib/auto-linker.js +1 -0
  149. package/dist/lib/canvas-layout.cjs +136 -0
  150. package/dist/lib/canvas-layout.d.cts +31 -0
  151. package/dist/lib/canvas-layout.d.ts +16 -100
  152. package/dist/lib/canvas-layout.js +78 -20
  153. package/dist/lib/config.cjs +78 -0
  154. package/dist/lib/config.d.cts +11 -0
  155. package/dist/lib/config.js +1 -0
  156. package/dist/lib/entity-index.cjs +84 -0
  157. package/dist/lib/entity-index.d.cts +26 -0
  158. package/dist/lib/entity-index.js +1 -0
  159. package/dist/lib/project-utils.cjs +864 -0
  160. package/dist/lib/project-utils.d.cts +97 -0
  161. package/dist/lib/project-utils.js +4 -3
  162. package/dist/lib/session-repair.cjs +239 -0
  163. package/dist/lib/session-repair.d.cts +110 -0
  164. package/dist/lib/session-repair.js +1 -0
  165. package/dist/lib/session-utils.cjs +209 -0
  166. package/dist/lib/session-utils.d.cts +63 -0
  167. package/dist/lib/session-utils.js +1 -0
  168. package/dist/lib/tailscale.cjs +1183 -0
  169. package/dist/lib/tailscale.d.cts +225 -0
  170. package/dist/lib/tailscale.js +1 -0
  171. package/dist/lib/task-utils.cjs +1137 -0
  172. package/dist/lib/task-utils.d.cts +208 -0
  173. package/dist/lib/task-utils.js +3 -2
  174. package/dist/lib/template-engine.cjs +47 -0
  175. package/dist/lib/template-engine.d.cts +11 -0
  176. package/dist/lib/template-engine.js +1 -0
  177. package/dist/lib/webdav.cjs +568 -0
  178. package/dist/lib/webdav.d.cts +109 -0
  179. package/dist/lib/webdav.js +1 -0
  180. package/dist/plugin/index.cjs +1907 -0
  181. package/dist/plugin/index.d.cts +36 -0
  182. package/dist/plugin/index.d.ts +36 -0
  183. package/dist/plugin/index.js +572 -0
  184. package/dist/plugin/inject.cjs +356 -0
  185. package/dist/plugin/inject.d.cts +54 -0
  186. package/dist/plugin/inject.d.ts +54 -0
  187. package/dist/plugin/inject.js +17 -0
  188. package/dist/plugin/observe.cjs +631 -0
  189. package/dist/plugin/observe.d.cts +39 -0
  190. package/dist/plugin/observe.d.ts +39 -0
  191. package/dist/plugin/observe.js +18 -0
  192. package/dist/plugin/templates.cjs +593 -0
  193. package/dist/plugin/templates.d.cts +52 -0
  194. package/dist/plugin/templates.d.ts +52 -0
  195. package/dist/plugin/templates.js +25 -0
  196. package/dist/plugin/types.cjs +18 -0
  197. package/dist/plugin/types.d.cts +209 -0
  198. package/dist/plugin/types.d.ts +209 -0
  199. package/dist/plugin/types.js +0 -0
  200. package/dist/plugin/vault.cjs +927 -0
  201. package/dist/plugin/vault.d.cts +68 -0
  202. package/dist/plugin/vault.d.ts +68 -0
  203. package/dist/plugin/vault.js +22 -0
  204. package/dist/{types-C74wgGL1.d.ts → types-Y2_Um2Ls.d.cts} +44 -1
  205. package/dist/types-Y2_Um2Ls.d.ts +205 -0
  206. package/hooks/clawvault/handler.js +70 -7
  207. package/hooks/clawvault/handler.test.js +91 -0
  208. package/openclaw.plugin.json +56 -0
  209. package/package.json +17 -7
  210. package/templates/memory-event.md +67 -0
  211. package/templates/party.md +63 -0
  212. package/templates/primitive-registry.yaml +551 -0
  213. package/templates/run.md +68 -0
  214. package/templates/trigger.md +68 -0
  215. package/templates/workspace.md +50 -0
  216. package/dashboard/lib/graph-diff.js +0 -104
  217. package/dashboard/lib/graph-diff.test.js +0 -75
  218. package/dashboard/lib/vault-parser.js +0 -556
  219. package/dashboard/lib/vault-parser.test.js +0 -254
  220. package/dashboard/public/app.js +0 -796
  221. package/dashboard/public/index.html +0 -52
  222. package/dashboard/public/styles.css +0 -221
  223. package/dashboard/server.js +0 -374
  224. package/dist/chunk-HA5M6KJB.js +0 -33
  225. package/dist/chunk-MAKNAHAW.js +0 -375
  226. package/dist/chunk-MDIH26GC.js +0 -183
  227. package/dist/chunk-MGDEINGP.js +0 -99
  228. package/dist/chunk-RVYA52PY.js +0 -363
  229. package/dist/commands/canvas.d.ts +0 -15
  230. package/dist/commands/canvas.js +0 -199
  231. package/dist/commands/sync-bd.d.ts +0 -10
  232. package/dist/commands/sync-bd.js +0 -9
@@ -0,0 +1,16 @@
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,20 +1,22 @@
1
1
  import {
2
2
  registerReplayCommand,
3
3
  replayCommand
4
- } from "../chunk-4TE4JMLA.js";
5
- import "../chunk-YOSEUUNB.js";
6
- import "../chunk-Q2J5YTUF.js";
7
- import "../chunk-AZYOKJYC.js";
8
- import "../chunk-FHFUXL6G.js";
9
- import "../chunk-K3CDT7IH.js";
4
+ } from "../chunk-JY6FYXIT.js";
5
+ import "../chunk-H34S76MB.js";
6
+ import "../chunk-3NSBOUT3.js";
7
+ import "../chunk-62YTUT6J.js";
8
+ import "../chunk-E7MFQB6D.js";
10
9
  import "../chunk-ITPEXLHA.js";
11
- import "../chunk-2CDEETQN.js";
10
+ import "../chunk-PAH27GSN.js";
12
11
  import "../chunk-MQUJNOHK.js";
13
12
  import "../chunk-MXSSG3QU.js";
14
- import "../chunk-Z2XBWN7A.js";
15
- import "../chunk-QWQ3TIKS.js";
16
- import "../chunk-MFAWT5O5.js";
13
+ import "../chunk-N2AXRYLC.js";
14
+ import "../chunk-LYHGEHXG.js";
17
15
  import "../chunk-7766SIJP.js";
16
+ import "../chunk-2CDEETQN.js";
17
+ import "../chunk-QK3UCXWL.js";
18
+ import "../chunk-Z2XBWN7A.js";
19
+ import "../chunk-3RG5ZIWI.js";
18
20
  export {
19
21
  registerReplayCommand,
20
22
  replayCommand
@@ -0,0 +1,353 @@
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
+ });
@@ -0,0 +1,27 @@
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 };
@@ -4,6 +4,7 @@ import {
4
4
  sessionRecapCommand
5
5
  } from "../chunk-ZKGY7WTT.js";
6
6
  import "../chunk-HRLWZGMA.js";
7
+ import "../chunk-3RG5ZIWI.js";
7
8
  export {
8
9
  buildSessionRecap,
9
10
  formatSessionRecapMarkdown,