clawvault 3.2.1 → 3.3.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 (150) hide show
  1. package/README.md +56 -16
  2. package/bin/clawvault.js +0 -2
  3. package/bin/command-registration.test.js +13 -1
  4. package/bin/help-contract.test.js +14 -0
  5. package/bin/register-core-commands.js +88 -0
  6. package/bin/register-core-commands.test.js +80 -0
  7. package/bin/register-maintenance-commands.js +57 -6
  8. package/bin/register-query-commands.js +10 -28
  9. package/bin/test-helpers/cli-command-fixtures.js +1 -0
  10. package/dist/chunk-2PKBIKDH.js +130 -0
  11. package/dist/{chunk-U67V476Y.js → chunk-2ZDO52B4.js} +18 -1
  12. package/dist/{chunk-ZZA73MFY.js → chunk-33DOSHTA.js} +176 -36
  13. package/dist/{chunk-AZYOKJYC.js → chunk-4PY655YM.js} +13 -1
  14. package/dist/{chunk-2JQ3O2YL.js → chunk-5EFSWZO6.js} +3 -3
  15. package/dist/{chunk-Y3TIJEBP.js → chunk-7SWP5FKU.js} +34 -613
  16. package/dist/{chunk-4VQTUVH7.js → chunk-7YZWHM36.js} +52 -26
  17. package/dist/{chunk-URXDAUVH.js → chunk-AXSJIFOJ.js} +174 -1
  18. package/dist/{chunk-4ITRXIVT.js → chunk-BLQXXX7Q.js} +6 -6
  19. package/dist/chunk-CSHO3PJB.js +684 -0
  20. package/dist/{chunk-S5OJEGFG.js → chunk-DOIUYIXV.js} +2 -2
  21. package/dist/{chunk-YXQCA6B7.js → chunk-DVOUSOR3.js} +112 -7
  22. package/dist/{chunk-YDWHS4LJ.js → chunk-ECGJYWNA.js} +205 -33
  23. package/dist/{chunk-QMHPQYUV.js → chunk-EL6UBSX5.js} +7 -6
  24. package/dist/chunk-FZ5I2NF7.js +352 -0
  25. package/dist/{chunk-WJVWINEM.js → chunk-GFCHWMGD.js} +55 -6
  26. package/dist/{chunk-GNJL4YGR.js → chunk-GJO3CFUN.js} +30 -6
  27. package/dist/chunk-H3JZIB5O.js +322 -0
  28. package/dist/chunk-HEHO7SMV.js +51 -0
  29. package/dist/{chunk-UCQAOZHW.js → chunk-HGDDW24U.js} +3 -3
  30. package/dist/chunk-J3YUXVID.js +907 -0
  31. package/dist/{chunk-Y6VJKXGL.js → chunk-KCYWJDDW.js} +1 -1
  32. package/dist/{chunk-P5EPF6MB.js → chunk-MW5C6ZQA.js} +110 -13
  33. package/dist/{chunk-YNIPYN4F.js → chunk-OFOCU2V4.js} +6 -5
  34. package/dist/{chunk-42MXU7A6.js → chunk-P62WHA27.js} +58 -47
  35. package/dist/chunk-PTWPPVC7.js +972 -0
  36. package/dist/{chunk-FAKNOB7Y.js → chunk-QFWERBDP.js} +2 -2
  37. package/dist/{chunk-IIOU45CK.js → chunk-S7N7HI5E.js} +2 -2
  38. package/dist/{chunk-ECRZL5XR.js → chunk-T7E764W3.js} +23 -7
  39. package/dist/chunk-TDWFBDAQ.js +1016 -0
  40. package/dist/{chunk-MNPUYCHQ.js → chunk-TWMI3SNN.js} +6 -5
  41. package/dist/{chunk-2RAZ4ZFE.js → chunk-VBILES4B.js} +1 -1
  42. package/dist/{chunk-PI4WMLMG.js → chunk-VXAGOLDP.js} +1 -1
  43. package/dist/chunk-YCUVAOFC.js +158 -0
  44. package/dist/{chunk-SS4B7P7V.js → chunk-YIDV4VV2.js} +1 -1
  45. package/dist/chunk-ZKWPCBYT.js +600 -0
  46. package/dist/cli/index.js +27 -21
  47. package/dist/commands/archive.js +3 -3
  48. package/dist/commands/backlog.js +1 -1
  49. package/dist/commands/benchmark.d.ts +12 -0
  50. package/dist/commands/benchmark.js +12 -0
  51. package/dist/commands/blocked.js +1 -1
  52. package/dist/commands/canvas.js +2 -2
  53. package/dist/commands/checkpoint.js +1 -1
  54. package/dist/commands/compat.js +1 -1
  55. package/dist/commands/context.js +8 -7
  56. package/dist/commands/doctor.d.ts +8 -3
  57. package/dist/commands/doctor.js +8 -22
  58. package/dist/commands/embed.js +6 -5
  59. package/dist/commands/entities.js +2 -2
  60. package/dist/commands/graph.js +4 -4
  61. package/dist/commands/inbox.d.ts +23 -0
  62. package/dist/commands/inbox.js +11 -0
  63. package/dist/commands/inject.d.ts +1 -1
  64. package/dist/commands/inject.js +5 -5
  65. package/dist/commands/kanban.js +1 -1
  66. package/dist/commands/link.js +9 -9
  67. package/dist/commands/maintain.d.ts +32 -0
  68. package/dist/commands/maintain.js +12 -0
  69. package/dist/commands/migrate-observations.js +3 -3
  70. package/dist/commands/observe.js +11 -10
  71. package/dist/commands/project.js +2 -2
  72. package/dist/commands/rebuild-embeddings.js +48 -17
  73. package/dist/commands/rebuild.js +9 -8
  74. package/dist/commands/recover.js +1 -1
  75. package/dist/commands/reflect.js +6 -6
  76. package/dist/commands/repair-session.js +1 -1
  77. package/dist/commands/replay.js +10 -9
  78. package/dist/commands/session-recap.js +1 -1
  79. package/dist/commands/setup.js +4 -3
  80. package/dist/commands/shell-init.js +1 -1
  81. package/dist/commands/sleep.d.ts +1 -1
  82. package/dist/commands/sleep.js +20 -18
  83. package/dist/commands/status.js +40 -26
  84. package/dist/commands/sync-bd.js +3 -3
  85. package/dist/commands/tailscale.js +3 -3
  86. package/dist/commands/task.js +1 -1
  87. package/dist/commands/template.js +1 -1
  88. package/dist/commands/wake.d.ts +1 -1
  89. package/dist/commands/wake.js +10 -9
  90. package/dist/index.d.ts +175 -16
  91. package/dist/index.js +277 -108
  92. package/dist/{inject-DYUrDqQO.d.ts → inject-DEb_jpLi.d.ts} +3 -1
  93. package/dist/lib/auto-linker.js +2 -2
  94. package/dist/lib/canvas-layout.js +1 -1
  95. package/dist/lib/config.js +2 -2
  96. package/dist/lib/entity-index.js +1 -1
  97. package/dist/lib/project-utils.js +2 -2
  98. package/dist/lib/session-repair.js +1 -1
  99. package/dist/lib/session-utils.js +1 -1
  100. package/dist/lib/tailscale.js +1 -1
  101. package/dist/lib/task-utils.js +1 -1
  102. package/dist/lib/template-engine.js +1 -1
  103. package/dist/lib/webdav.js +1 -1
  104. package/dist/onnxruntime_binding-5QEF3SUC.node +0 -0
  105. package/dist/onnxruntime_binding-BKPKNEGC.node +0 -0
  106. package/dist/onnxruntime_binding-FMOXGIUT.node +0 -0
  107. package/dist/onnxruntime_binding-OI2KMXC5.node +0 -0
  108. package/dist/onnxruntime_binding-UX44MLAZ.node +0 -0
  109. package/dist/onnxruntime_binding-Y2W7N7WY.node +0 -0
  110. package/dist/openclaw-plugin.d.ts +8 -0
  111. package/dist/openclaw-plugin.js +14 -0
  112. package/dist/transformers.node-A2ZRORSQ.js +46775 -0
  113. package/dist/{types-BbWJoC1c.d.ts → types-DslKvCaj.d.ts} +51 -1
  114. package/hooks/clawvault/HOOK.md +25 -8
  115. package/hooks/clawvault/handler.js +215 -78
  116. package/hooks/clawvault/handler.test.js +109 -43
  117. package/hooks/clawvault/integrity.js +112 -0
  118. package/hooks/clawvault/integrity.test.js +32 -0
  119. package/hooks/clawvault/openclaw.plugin.json +133 -15
  120. package/openclaw.plugin.json +131 -203
  121. package/package.json +10 -7
  122. package/bin/register-workgraph-commands.js +0 -451
  123. package/dist/chunk-5PJ4STIC.js +0 -465
  124. package/dist/chunk-ERNE2FZ5.js +0 -189
  125. package/dist/chunk-HR4KN6S2.js +0 -152
  126. package/dist/chunk-IJBFGPCS.js +0 -33
  127. package/dist/chunk-K7PNYS45.js +0 -93
  128. package/dist/chunk-NTOPJI7W.js +0 -207
  129. package/dist/chunk-PG56HX5T.js +0 -154
  130. package/dist/chunk-QPDDIHXE.js +0 -501
  131. package/dist/chunk-WIOLLGAD.js +0 -190
  132. package/dist/chunk-WMGIIABP.js +0 -15
  133. package/dist/ledger-B7g7jhqG.d.ts +0 -44
  134. package/dist/plugin/index.d.ts +0 -352
  135. package/dist/plugin/index.js +0 -4264
  136. package/dist/registry-BR4326o0.d.ts +0 -30
  137. package/dist/store-CA-6sKCJ.d.ts +0 -34
  138. package/dist/thread-B9LhXNU0.d.ts +0 -41
  139. package/dist/workgraph/index.d.ts +0 -5
  140. package/dist/workgraph/index.js +0 -23
  141. package/dist/workgraph/ledger.d.ts +0 -2
  142. package/dist/workgraph/ledger.js +0 -25
  143. package/dist/workgraph/registry.d.ts +0 -2
  144. package/dist/workgraph/registry.js +0 -19
  145. package/dist/workgraph/store.d.ts +0 -2
  146. package/dist/workgraph/store.js +0 -25
  147. package/dist/workgraph/thread.d.ts +0 -2
  148. package/dist/workgraph/thread.js +0 -25
  149. package/dist/workgraph/types.d.ts +0 -54
  150. package/dist/workgraph/types.js +0 -7
@@ -0,0 +1,352 @@
1
+ import {
2
+ withQmdIndexArgs
3
+ } from "./chunk-PTWPPVC7.js";
4
+
5
+ // src/lib/qmd-collections.ts
6
+ import { execFileSync } from "child_process";
7
+ var COLLECTION_HEADER_RE = /^(\S+)\s+\(qmd:\/\/([^)]+)\)\s*$/;
8
+ var DETAIL_LINE_RE = /^\s+([A-Za-z][A-Za-z0-9 _-]*):\s*(.+)\s*$/;
9
+ var QMD_INDEX_ENV_VAR = "CLAWVAULT_QMD_INDEX";
10
+ function normalizeDetailKey(value) {
11
+ return value.trim().toLowerCase().replace(/[ -]+/g, "_");
12
+ }
13
+ function parseCount(raw) {
14
+ if (!raw) return void 0;
15
+ const match = raw.match(/-?\d[\d,]*/);
16
+ if (!match) return void 0;
17
+ const parsed = Number.parseInt(match[0].replace(/,/g, ""), 10);
18
+ return Number.isFinite(parsed) ? parsed : void 0;
19
+ }
20
+ function pickDetail(details, keys) {
21
+ for (const key of keys) {
22
+ const value = details[key];
23
+ if (typeof value === "string" && value.trim()) {
24
+ return value.trim();
25
+ }
26
+ }
27
+ return void 0;
28
+ }
29
+ function pickCount(details, keys) {
30
+ for (const key of keys) {
31
+ const parsed = parseCount(details[key]);
32
+ if (parsed !== void 0) {
33
+ return parsed;
34
+ }
35
+ }
36
+ return void 0;
37
+ }
38
+ function resolveQmdIndexName(indexName) {
39
+ const explicit = indexName?.trim();
40
+ if (explicit) {
41
+ return explicit;
42
+ }
43
+ const fromEnv = process.env[QMD_INDEX_ENV_VAR]?.trim();
44
+ return fromEnv || void 0;
45
+ }
46
+ function withQmdIndexArgs2(args, indexName) {
47
+ if (args.includes("--index")) {
48
+ return [...args];
49
+ }
50
+ const resolvedIndexName = resolveQmdIndexName(indexName);
51
+ if (!resolvedIndexName) {
52
+ return [...args];
53
+ }
54
+ return ["--index", resolvedIndexName, ...args];
55
+ }
56
+ function parseQmdCollectionList(raw) {
57
+ const collections = [];
58
+ let current = null;
59
+ for (const line of raw.split(/\r?\n/)) {
60
+ const headerMatch = line.match(COLLECTION_HEADER_RE);
61
+ if (headerMatch) {
62
+ current = {
63
+ name: headerMatch[1],
64
+ uri: headerMatch[2],
65
+ details: {}
66
+ };
67
+ collections.push(current);
68
+ continue;
69
+ }
70
+ if (!current) continue;
71
+ const detailMatch = line.match(DETAIL_LINE_RE);
72
+ if (!detailMatch) continue;
73
+ const key = normalizeDetailKey(detailMatch[1]);
74
+ current.details[key] = detailMatch[2].trim();
75
+ }
76
+ for (const collection of collections) {
77
+ const root = pickDetail(collection.details, ["root", "path", "directory"]);
78
+ if (root) {
79
+ collection.root = root;
80
+ }
81
+ collection.files = pickCount(collection.details, ["files", "documents", "docs"]);
82
+ collection.vectors = pickCount(collection.details, ["vectors", "embeddings", "vector_embeddings"]);
83
+ collection.pendingEmbeddings = pickCount(collection.details, [
84
+ "pending",
85
+ "pending_vectors",
86
+ "pending_embeddings",
87
+ "unembedded",
88
+ "without_embeddings"
89
+ ]);
90
+ if (collection.pendingEmbeddings === void 0 && collection.files !== void 0 && collection.vectors !== void 0) {
91
+ collection.pendingEmbeddings = Math.max(collection.files - collection.vectors, 0);
92
+ }
93
+ }
94
+ return collections;
95
+ }
96
+ function listQmdCollections(indexName) {
97
+ const output = execFileSync("qmd", withQmdIndexArgs2(["collection", "list"], indexName), {
98
+ encoding: "utf-8",
99
+ shell: process.platform === "win32"
100
+ });
101
+ return parseQmdCollectionList(output);
102
+ }
103
+ function removeQmdCollection(name, indexName) {
104
+ try {
105
+ execFileSync("qmd", withQmdIndexArgs2(["collection", "remove", name], indexName), { stdio: "ignore", shell: process.platform === "win32" });
106
+ return;
107
+ } catch {
108
+ execFileSync("qmd", withQmdIndexArgs2(["collection", "rm", name], indexName), { stdio: "ignore", shell: process.platform === "win32" });
109
+ }
110
+ }
111
+ function collectionExists(name, indexName) {
112
+ try {
113
+ const collections = listQmdCollections(indexName);
114
+ return collections.some((c) => c.name === name);
115
+ } catch {
116
+ return false;
117
+ }
118
+ }
119
+ function getCollectionByName(name, indexName) {
120
+ try {
121
+ const collections = listQmdCollections(indexName);
122
+ return collections.find((collection) => collection.name === name);
123
+ } catch {
124
+ return void 0;
125
+ }
126
+ }
127
+ function findCollectionByRoot(rootPath, indexName) {
128
+ try {
129
+ const collections = listQmdCollections(indexName);
130
+ const normalizedRoot = rootPath.replace(/\/$/, "");
131
+ return collections.find((c) => {
132
+ if (!c.root) return false;
133
+ const normalizedCollectionRoot = c.root.replace(/\/$/, "");
134
+ return normalizedCollectionRoot === normalizedRoot;
135
+ });
136
+ } catch {
137
+ return void 0;
138
+ }
139
+ }
140
+ function getFirstCollection(indexName) {
141
+ try {
142
+ const collections = listQmdCollections(indexName);
143
+ return collections[0];
144
+ } catch {
145
+ return void 0;
146
+ }
147
+ }
148
+
149
+ // src/lib/vault-qmd-config.ts
150
+ import * as fs from "fs";
151
+ import * as path from "path";
152
+ var CONFIG_FILE = ".clawvault.json";
153
+ function readTrimmedString(value) {
154
+ if (typeof value !== "string") return void 0;
155
+ const trimmed = value.trim();
156
+ return trimmed.length > 0 ? trimmed : void 0;
157
+ }
158
+ function autoDetectCollection(vaultPath, fallbackName) {
159
+ const byRoot = findCollectionByRoot(vaultPath);
160
+ if (byRoot) {
161
+ return { collection: byRoot.name, autoDetected: true };
162
+ }
163
+ if (collectionExists(fallbackName)) {
164
+ return { collection: fallbackName, autoDetected: false };
165
+ }
166
+ const first = getFirstCollection();
167
+ if (first) {
168
+ return { collection: first.name, autoDetected: true };
169
+ }
170
+ return { collection: fallbackName, autoDetected: false };
171
+ }
172
+ function loadVaultQmdConfig(vaultPath) {
173
+ const resolvedVaultPath = path.resolve(vaultPath);
174
+ const fallbackName = path.basename(resolvedVaultPath);
175
+ const fallbackRoot = resolvedVaultPath;
176
+ const configPath = path.join(resolvedVaultPath, CONFIG_FILE);
177
+ if (!fs.existsSync(configPath)) {
178
+ const { collection, autoDetected } = autoDetectCollection(resolvedVaultPath, fallbackName);
179
+ return {
180
+ vaultPath: resolvedVaultPath,
181
+ qmdCollection: collection,
182
+ qmdRoot: fallbackRoot,
183
+ autoDetected
184
+ };
185
+ }
186
+ try {
187
+ const raw = JSON.parse(fs.readFileSync(configPath, "utf-8"));
188
+ const configuredName = readTrimmedString(raw.name) ?? fallbackName;
189
+ const configuredCollection = readTrimmedString(raw.qmdCollection);
190
+ const rawRoot = readTrimmedString(raw.qmdRoot) ?? fallbackRoot;
191
+ const qmdRoot = path.isAbsolute(rawRoot) ? path.resolve(rawRoot) : path.resolve(resolvedVaultPath, rawRoot);
192
+ if (configuredCollection && collectionExists(configuredCollection)) {
193
+ return {
194
+ vaultPath: resolvedVaultPath,
195
+ qmdCollection: configuredCollection,
196
+ qmdRoot
197
+ };
198
+ }
199
+ const { collection, autoDetected } = autoDetectCollection(qmdRoot, configuredCollection ?? configuredName);
200
+ return {
201
+ vaultPath: resolvedVaultPath,
202
+ qmdCollection: collection,
203
+ qmdRoot,
204
+ autoDetected
205
+ };
206
+ } catch {
207
+ const { collection, autoDetected } = autoDetectCollection(resolvedVaultPath, fallbackName);
208
+ return {
209
+ vaultPath: resolvedVaultPath,
210
+ qmdCollection: collection,
211
+ qmdRoot: fallbackRoot,
212
+ autoDetected
213
+ };
214
+ }
215
+ }
216
+
217
+ // src/lib/qmd-embedding-recovery.ts
218
+ import { execFileSync as execFileSync2 } from "child_process";
219
+ import * as fs2 from "fs";
220
+ import * as path2 from "path";
221
+ var CLAWVAULT_DIR = ".clawvault";
222
+ var QMD_EMBED_WAL_FILE = "qmd-embed.wal.json";
223
+ var WALK_SKIP_DIRS = /* @__PURE__ */ new Set([".git", ".obsidian", ".trash", ".clawvault", "node_modules"]);
224
+ function getQmdEmbedWalPath(vaultPath) {
225
+ return path2.join(vaultPath, CLAWVAULT_DIR, QMD_EMBED_WAL_FILE);
226
+ }
227
+ function writeJsonAtomic(filePath, value) {
228
+ const dir = path2.dirname(filePath);
229
+ fs2.mkdirSync(dir, { recursive: true });
230
+ const tmpPath = path2.join(
231
+ dir,
232
+ `${path2.basename(filePath)}.${process.pid}.${Date.now()}.tmp`
233
+ );
234
+ try {
235
+ fs2.writeFileSync(tmpPath, `${JSON.stringify(value, null, 2)}
236
+ `, "utf-8");
237
+ fs2.renameSync(tmpPath, filePath);
238
+ } finally {
239
+ if (fs2.existsSync(tmpPath)) {
240
+ fs2.rmSync(tmpPath, { force: true });
241
+ }
242
+ }
243
+ }
244
+ function runQmdCommand(args, indexName) {
245
+ execFileSync2("qmd", withQmdIndexArgs(args, indexName), { stdio: "inherit", shell: process.platform === "win32" });
246
+ }
247
+ function runQmdEmbedForce(collection, indexName) {
248
+ try {
249
+ runQmdCommand(["embed", "-f", "-c", collection], indexName);
250
+ } catch {
251
+ runQmdCommand(["embed", "-c", collection], indexName);
252
+ }
253
+ }
254
+ function countMarkdownFiles(rootPath) {
255
+ let total = 0;
256
+ const stack = [rootPath];
257
+ while (stack.length > 0) {
258
+ const current = stack.pop();
259
+ if (!current) continue;
260
+ let entries;
261
+ try {
262
+ entries = fs2.readdirSync(current, { withFileTypes: true });
263
+ } catch {
264
+ continue;
265
+ }
266
+ for (const entry of entries) {
267
+ const entryPath = path2.join(current, entry.name);
268
+ if (entry.isDirectory()) {
269
+ if (entry.name.startsWith(".") || WALK_SKIP_DIRS.has(entry.name)) {
270
+ continue;
271
+ }
272
+ stack.push(entryPath);
273
+ continue;
274
+ }
275
+ if (entry.isFile() && entry.name.endsWith(".md") && !entry.name.startsWith(".")) {
276
+ total += 1;
277
+ }
278
+ }
279
+ }
280
+ return total;
281
+ }
282
+ function shouldRecoverEmptyVectors(collectionInfo, markdownCount) {
283
+ if (!collectionInfo) return false;
284
+ if (markdownCount === 0) return false;
285
+ if (collectionInfo.vectors === void 0) return false;
286
+ if (collectionInfo.vectors > 0) return false;
287
+ if (collectionInfo.pendingEmbeddings !== void 0 && collectionInfo.pendingEmbeddings > 0) {
288
+ return false;
289
+ }
290
+ return true;
291
+ }
292
+ function hasWalMarker(vaultPath) {
293
+ return fs2.existsSync(getQmdEmbedWalPath(vaultPath));
294
+ }
295
+ function clearQmdEmbedWalRecord(vaultPath) {
296
+ const walPath = getQmdEmbedWalPath(vaultPath);
297
+ if (fs2.existsSync(walPath)) {
298
+ fs2.rmSync(walPath, { force: true });
299
+ }
300
+ }
301
+ function writeQmdEmbedWalRecord(options) {
302
+ const wal = {
303
+ version: 1,
304
+ status: "running",
305
+ startedAt: (/* @__PURE__ */ new Date()).toISOString(),
306
+ collection: options.collection,
307
+ rootPath: options.rootPath,
308
+ indexName: options.indexName
309
+ };
310
+ writeJsonAtomic(getQmdEmbedWalPath(options.vaultPath), wal);
311
+ }
312
+ function runCrashSafeQmdEmbed(options) {
313
+ writeQmdEmbedWalRecord(options);
314
+ try {
315
+ runQmdCommand(["embed", "-c", options.collection], options.indexName);
316
+ clearQmdEmbedWalRecord(options.vaultPath);
317
+ } catch (err) {
318
+ throw err;
319
+ }
320
+ }
321
+ function recoverQmdEmbeddingIfNeeded(options) {
322
+ const mode = options.mode ?? "marker-or-empty";
323
+ let reason;
324
+ if (hasWalMarker(options.vaultPath)) {
325
+ reason = "interrupted_wal";
326
+ } else if (mode === "marker-or-empty") {
327
+ const collectionInfo = getCollectionByName(options.collection, options.indexName);
328
+ const markdownCount = countMarkdownFiles(options.rootPath);
329
+ if (shouldRecoverEmptyVectors(collectionInfo, markdownCount)) {
330
+ reason = "empty_vectors";
331
+ }
332
+ }
333
+ if (!reason) {
334
+ return { recovered: false };
335
+ }
336
+ options.onLog?.(
337
+ reason === "interrupted_wal" ? `Detected interrupted qmd embedding run for "${options.collection}". Rebuilding...` : `Detected empty qmd vector state for "${options.collection}". Rebuilding...`
338
+ );
339
+ runQmdCommand(["update", "-c", options.collection], options.indexName);
340
+ runQmdEmbedForce(options.collection, options.indexName);
341
+ clearQmdEmbedWalRecord(options.vaultPath);
342
+ return { recovered: true, reason };
343
+ }
344
+
345
+ export {
346
+ parseQmdCollectionList,
347
+ listQmdCollections,
348
+ removeQmdCollection,
349
+ loadVaultQmdConfig,
350
+ runCrashSafeQmdEmbed,
351
+ recoverQmdEmbeddingIfNeeded
352
+ };
@@ -1,15 +1,15 @@
1
- import {
2
- ClawVault
3
- } from "./chunk-YDWHS4LJ.js";
4
1
  import {
5
2
  FactStore
6
3
  } from "./chunk-BSJ6RIT7.js";
4
+ import {
5
+ ClawVault
6
+ } from "./chunk-ECGJYWNA.js";
7
7
  import {
8
8
  parseObservationMarkdown
9
9
  } from "./chunk-FHFUXL6G.js";
10
10
  import {
11
11
  getMemoryGraph
12
- } from "./chunk-ZZA73MFY.js";
12
+ } from "./chunk-33DOSHTA.js";
13
13
  import {
14
14
  listObservationFiles
15
15
  } from "./chunk-Z2XBWN7A.js";
@@ -144,7 +144,56 @@ var STOP_WORDS = /* @__PURE__ */ new Set([
144
144
  "why",
145
145
  "with",
146
146
  "you",
147
- "your"
147
+ "your",
148
+ "\u0438",
149
+ "\u0432",
150
+ "\u0432\u043E",
151
+ "\u043D\u0430",
152
+ "\u043F\u043E",
153
+ "\u0441",
154
+ "\u0441\u043E",
155
+ "\u043A",
156
+ "\u043A\u043E",
157
+ "\u0443",
158
+ "\u043E",
159
+ "\u043E\u0431",
160
+ "\u043E\u0442",
161
+ "\u0434\u043E",
162
+ "\u0437\u0430",
163
+ "\u0438\u0437",
164
+ "\u043F\u043E\u0434",
165
+ "\u043D\u0430\u0434",
166
+ "\u043D\u0435",
167
+ "\u043D\u043E",
168
+ "\u0430",
169
+ "\u0438\u043B\u0438",
170
+ "\u0447\u0442\u043E",
171
+ "\u043A\u0430\u043A",
172
+ "\u044D\u0442\u043E",
173
+ "\u0442\u043E",
174
+ "\u0436\u0435",
175
+ "\u043B\u0438",
176
+ "\u0434\u043B\u044F",
177
+ "\u043F\u0440\u0438",
178
+ "\u043F\u0440\u043E",
179
+ "\u043C\u044B",
180
+ "\u0432\u044B",
181
+ "\u043E\u043D\u0438",
182
+ "\u043E\u043D",
183
+ "\u043E\u043D\u0430",
184
+ "\u043E\u043D\u043E",
185
+ "\u044F",
186
+ "\u0442\u044B",
187
+ "\u043C\u043E\u0439",
188
+ "\u043C\u043E\u044F",
189
+ "\u043C\u043E\u0451",
190
+ "\u043C\u043E\u0438",
191
+ "\u0442\u0432\u043E\u0439",
192
+ "\u0442\u0432\u043E\u044F",
193
+ "\u0442\u0432\u043E\u0438",
194
+ "\u043D\u0430\u0448",
195
+ "\u043D\u0430\u0448\u0430",
196
+ "\u0438\u0445"
148
197
  ]);
149
198
  function formatRelativeAge(date, now = Date.now()) {
150
199
  const ageMs = Math.max(0, now - date.getTime());
@@ -199,7 +248,7 @@ var PROFILE_ORDERING = {
199
248
  }
200
249
  };
201
250
  function extractKeywords(text) {
202
- const raw = text.toLowerCase().match(/[a-z0-9]+/g) ?? [];
251
+ const raw = text.toLowerCase().match(/[\p{L}\p{N}]+/gu) ?? [];
203
252
  const seen = /* @__PURE__ */ new Set();
204
253
  const keywords = [];
205
254
  for (const token of raw) {
@@ -44,9 +44,31 @@ function resolveAgentVaultPath(agentVaults, agentId) {
44
44
  if (!agentPath || typeof agentPath !== "string") return null;
45
45
  return validateVaultPath(agentPath);
46
46
  }
47
+ function toNonEmptyPathString(value) {
48
+ if (typeof value !== "string") return null;
49
+ const trimmed = value.trim();
50
+ return trimmed ? trimmed : null;
51
+ }
52
+ function coercePathCandidate(value) {
53
+ const direct = toNonEmptyPathString(value);
54
+ if (direct) return direct;
55
+ if (!value || typeof value !== "object" || Array.isArray(value)) {
56
+ return null;
57
+ }
58
+ const record = value;
59
+ const candidates = [record.explicitPath, record.vaultPath, record.path, record.vault];
60
+ for (const candidate of candidates) {
61
+ const normalized = toNonEmptyPathString(candidate);
62
+ if (normalized) {
63
+ return normalized;
64
+ }
65
+ }
66
+ return null;
67
+ }
47
68
  function resolveVaultPath(options = {}) {
48
- if (options.explicitPath) {
49
- return path.resolve(options.explicitPath);
69
+ const explicitPath = coercePathCandidate(options.explicitPath);
70
+ if (explicitPath) {
71
+ return path.resolve(explicitPath);
50
72
  }
51
73
  if (options.agentId && options.pluginConfig?.agentVaults) {
52
74
  const agentVaultPath = resolveAgentVaultPath(
@@ -57,12 +79,14 @@ function resolveVaultPath(options = {}) {
57
79
  return agentVaultPath;
58
80
  }
59
81
  }
60
- if (options.pluginConfig?.vaultPath) {
61
- const validated = validateVaultPath(options.pluginConfig.vaultPath);
82
+ const configuredVaultPath = coercePathCandidate(options.pluginConfig?.vaultPath);
83
+ if (configuredVaultPath) {
84
+ const validated = validateVaultPath(configuredVaultPath);
62
85
  if (validated) return validated;
63
86
  }
64
- if (process.env.CLAWVAULT_PATH) {
65
- return path.resolve(process.env.CLAWVAULT_PATH);
87
+ const envVaultPath = toNonEmptyPathString(process.env.CLAWVAULT_PATH);
88
+ if (envVaultPath) {
89
+ return path.resolve(envVaultPath);
66
90
  }
67
91
  const discovered = findNearestVaultPath(options.cwd ?? process.cwd());
68
92
  if (discovered) {