clawvault 3.2.1 → 3.4.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 (161) hide show
  1. package/README.md +56 -16
  2. package/bin/clawvault.js +0 -2
  3. package/bin/command-registration.test.js +15 -2
  4. package/bin/help-contract.test.js +16 -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 +84 -7
  8. package/bin/register-query-commands.js +45 -28
  9. package/bin/register-query-commands.test.js +15 -0
  10. package/bin/test-helpers/cli-command-fixtures.js +1 -0
  11. package/dist/chunk-2PKBIKDH.js +130 -0
  12. package/dist/{chunk-U67V476Y.js → chunk-2ZDO52B4.js} +18 -1
  13. package/dist/{chunk-ZZA73MFY.js → chunk-33DOSHTA.js} +176 -36
  14. package/dist/chunk-35JCYSRR.js +158 -0
  15. package/dist/{chunk-AZYOKJYC.js → chunk-4PY655YM.js} +13 -1
  16. package/dist/{chunk-2JQ3O2YL.js → chunk-5EFSWZO6.js} +3 -3
  17. package/dist/{chunk-Y3TIJEBP.js → chunk-7SWP5FKU.js} +34 -613
  18. package/dist/{chunk-4VQTUVH7.js → chunk-7YZWHM36.js} +52 -26
  19. package/dist/{chunk-URXDAUVH.js → chunk-AXSJIFOJ.js} +174 -1
  20. package/dist/{chunk-4ITRXIVT.js → chunk-BLQXXX7Q.js} +6 -6
  21. package/dist/chunk-CSHO3PJB.js +684 -0
  22. package/dist/chunk-D5U3Q4N5.js +872 -0
  23. package/dist/chunk-DCF4KMFD.js +158 -0
  24. package/dist/{chunk-S5OJEGFG.js → chunk-DOIUYIXV.js} +2 -2
  25. package/dist/{chunk-YXQCA6B7.js → chunk-DVOUSOR3.js} +112 -7
  26. package/dist/{chunk-YDWHS4LJ.js → chunk-ECGJYWNA.js} +205 -33
  27. package/dist/{chunk-QMHPQYUV.js → chunk-EL6UBSX5.js} +7 -6
  28. package/dist/chunk-FZ5I2NF7.js +352 -0
  29. package/dist/{chunk-WJVWINEM.js → chunk-GFCHWMGD.js} +55 -6
  30. package/dist/{chunk-GNJL4YGR.js → chunk-GJO3CFUN.js} +30 -6
  31. package/dist/chunk-H3JZIB5O.js +322 -0
  32. package/dist/chunk-HEHO7SMV.js +51 -0
  33. package/dist/{chunk-UCQAOZHW.js → chunk-HGDDW24U.js} +3 -3
  34. package/dist/chunk-J3YUXVID.js +907 -0
  35. package/dist/{chunk-Y6VJKXGL.js → chunk-KCYWJDDW.js} +1 -1
  36. package/dist/{chunk-P5EPF6MB.js → chunk-MW5C6ZQA.js} +110 -13
  37. package/dist/chunk-NSXYM6EZ.js +255 -0
  38. package/dist/{chunk-YNIPYN4F.js → chunk-OFOCU2V4.js} +6 -5
  39. package/dist/{chunk-42MXU7A6.js → chunk-P62WHA27.js} +58 -47
  40. package/dist/chunk-PTWPPVC7.js +972 -0
  41. package/dist/{chunk-FAKNOB7Y.js → chunk-QFWERBDP.js} +2 -2
  42. package/dist/chunk-QYQAGBTM.js +2097 -0
  43. package/dist/chunk-RL2L6I6K.js +223 -0
  44. package/dist/{chunk-IIOU45CK.js → chunk-S7N7HI5E.js} +2 -2
  45. package/dist/{chunk-ECRZL5XR.js → chunk-T7E764W3.js} +23 -7
  46. package/dist/{chunk-MNPUYCHQ.js → chunk-TWMI3SNN.js} +6 -5
  47. package/dist/{chunk-2RAZ4ZFE.js → chunk-VBILES4B.js} +1 -1
  48. package/dist/{chunk-PI4WMLMG.js → chunk-VXAGOLDP.js} +1 -1
  49. package/dist/{chunk-SS4B7P7V.js → chunk-YIDV4VV2.js} +1 -1
  50. package/dist/chunk-YTRZNA64.js +37 -0
  51. package/dist/chunk-ZKWPCBYT.js +600 -0
  52. package/dist/cli/index.js +28 -21
  53. package/dist/commands/archive.js +3 -3
  54. package/dist/commands/backlog.js +1 -1
  55. package/dist/commands/benchmark.d.ts +12 -0
  56. package/dist/commands/benchmark.js +12 -0
  57. package/dist/commands/blocked.js +1 -1
  58. package/dist/commands/canvas.js +2 -2
  59. package/dist/commands/checkpoint.js +1 -1
  60. package/dist/commands/compat.js +1 -1
  61. package/dist/commands/context.js +8 -7
  62. package/dist/commands/doctor.d.ts +8 -3
  63. package/dist/commands/doctor.js +8 -22
  64. package/dist/commands/embed.js +6 -5
  65. package/dist/commands/entities.d.ts +8 -1
  66. package/dist/commands/entities.js +46 -3
  67. package/dist/commands/graph.js +4 -4
  68. package/dist/commands/inbox.d.ts +23 -0
  69. package/dist/commands/inbox.js +11 -0
  70. package/dist/commands/inject.d.ts +1 -1
  71. package/dist/commands/inject.js +5 -5
  72. package/dist/commands/kanban.js +1 -1
  73. package/dist/commands/link.js +5 -5
  74. package/dist/commands/maintain.d.ts +32 -0
  75. package/dist/commands/maintain.js +13 -0
  76. package/dist/commands/migrate-observations.js +3 -3
  77. package/dist/commands/observe.js +11 -10
  78. package/dist/commands/project.js +2 -2
  79. package/dist/commands/rebuild-embeddings.js +48 -17
  80. package/dist/commands/rebuild.js +9 -8
  81. package/dist/commands/recall.d.ts +14 -0
  82. package/dist/commands/recall.js +15 -0
  83. package/dist/commands/recover.js +1 -1
  84. package/dist/commands/reflect.js +6 -6
  85. package/dist/commands/repair-session.js +1 -1
  86. package/dist/commands/replay.js +10 -9
  87. package/dist/commands/session-recap.js +1 -1
  88. package/dist/commands/setup.js +4 -3
  89. package/dist/commands/shell-init.js +1 -1
  90. package/dist/commands/sleep.d.ts +1 -1
  91. package/dist/commands/sleep.js +20 -18
  92. package/dist/commands/status.js +40 -26
  93. package/dist/commands/sync-bd.js +3 -3
  94. package/dist/commands/tailscale.js +3 -3
  95. package/dist/commands/task.js +1 -1
  96. package/dist/commands/template.js +1 -1
  97. package/dist/commands/wake.d.ts +1 -1
  98. package/dist/commands/wake.js +10 -9
  99. package/dist/index.d.ts +233 -16
  100. package/dist/index.js +325 -111
  101. package/dist/{inject-DYUrDqQO.d.ts → inject-DEb_jpLi.d.ts} +3 -1
  102. package/dist/lib/auto-linker.js +2 -2
  103. package/dist/lib/canvas-layout.js +1 -1
  104. package/dist/lib/config.js +2 -2
  105. package/dist/lib/entity-index.js +1 -1
  106. package/dist/lib/project-utils.js +2 -2
  107. package/dist/lib/session-repair.js +1 -1
  108. package/dist/lib/session-utils.js +1 -1
  109. package/dist/lib/tailscale.js +1 -1
  110. package/dist/lib/task-utils.js +1 -1
  111. package/dist/lib/template-engine.js +1 -1
  112. package/dist/lib/webdav.js +1 -1
  113. package/dist/onnxruntime_binding-5QEF3SUC.node +0 -0
  114. package/dist/onnxruntime_binding-BKPKNEGC.node +0 -0
  115. package/dist/onnxruntime_binding-FMOXGIUT.node +0 -0
  116. package/dist/onnxruntime_binding-OI2KMXC5.node +0 -0
  117. package/dist/onnxruntime_binding-UX44MLAZ.node +0 -0
  118. package/dist/onnxruntime_binding-Y2W7N7WY.node +0 -0
  119. package/dist/openclaw-plugin--gqA2BZw.d.ts +267 -0
  120. package/dist/openclaw-plugin.d.ts +4 -0
  121. package/dist/openclaw-plugin.js +20 -0
  122. package/dist/transformers.node-A2ZRORSQ.js +46775 -0
  123. package/dist/types-CbL-wIKi.d.ts +36 -0
  124. package/dist/{types-BbWJoC1c.d.ts → types-DslKvCaj.d.ts} +51 -1
  125. package/hooks/clawvault/HOOK.md +25 -8
  126. package/hooks/clawvault/handler.js +215 -78
  127. package/hooks/clawvault/handler.test.js +109 -43
  128. package/hooks/clawvault/integrity.js +112 -0
  129. package/hooks/clawvault/integrity.test.js +32 -0
  130. package/hooks/clawvault/openclaw.plugin.json +133 -15
  131. package/openclaw.plugin.json +161 -194
  132. package/package.json +8 -5
  133. package/bin/register-workgraph-commands.js +0 -451
  134. package/dist/chunk-5PJ4STIC.js +0 -465
  135. package/dist/chunk-ERNE2FZ5.js +0 -189
  136. package/dist/chunk-HR4KN6S2.js +0 -152
  137. package/dist/chunk-IJBFGPCS.js +0 -33
  138. package/dist/chunk-K7PNYS45.js +0 -93
  139. package/dist/chunk-NTOPJI7W.js +0 -207
  140. package/dist/chunk-PG56HX5T.js +0 -154
  141. package/dist/chunk-QPDDIHXE.js +0 -501
  142. package/dist/chunk-WIOLLGAD.js +0 -190
  143. package/dist/chunk-WMGIIABP.js +0 -15
  144. package/dist/ledger-B7g7jhqG.d.ts +0 -44
  145. package/dist/plugin/index.d.ts +0 -352
  146. package/dist/plugin/index.js +0 -4264
  147. package/dist/registry-BR4326o0.d.ts +0 -30
  148. package/dist/store-CA-6sKCJ.d.ts +0 -34
  149. package/dist/thread-B9LhXNU0.d.ts +0 -41
  150. package/dist/workgraph/index.d.ts +0 -5
  151. package/dist/workgraph/index.js +0 -23
  152. package/dist/workgraph/ledger.d.ts +0 -2
  153. package/dist/workgraph/ledger.js +0 -25
  154. package/dist/workgraph/registry.d.ts +0 -2
  155. package/dist/workgraph/registry.js +0 -19
  156. package/dist/workgraph/store.d.ts +0 -2
  157. package/dist/workgraph/store.js +0 -25
  158. package/dist/workgraph/thread.d.ts +0 -2
  159. package/dist/workgraph/thread.js +0 -25
  160. package/dist/workgraph/types.d.ts +0 -54
  161. package/dist/workgraph/types.js +0 -7
@@ -1,10 +1,10 @@
1
- import {
2
- hasQmd,
3
- withQmdIndexArgs
4
- } from "./chunk-5PJ4STIC.js";
5
1
  import {
6
2
  DEFAULT_CATEGORIES
7
3
  } from "./chunk-2CDEETQN.js";
4
+ import {
5
+ hasQmd,
6
+ withQmdIndexArgs
7
+ } from "./chunk-PTWPPVC7.js";
8
8
 
9
9
  // src/commands/setup.ts
10
10
  import * as fs from "fs";
@@ -339,14 +339,15 @@ async function setupCommand(options = {}) {
339
339
  const { collection, root } = getQmdConfig(target.vaultPath);
340
340
  try {
341
341
  execFileSync("qmd", withQmdIndexArgs(["collection", "add", root, "--name", collection, "--mask", "**/*.md"], options.qmdIndexName), {
342
- stdio: "ignore"
342
+ stdio: "ignore",
343
+ shell: process.platform === "win32"
343
344
  });
344
345
  console.log(`\u2713 qmd collection ready: ${collection}`);
345
346
  } catch {
346
347
  console.log("\u2298 qmd collection already exists.");
347
348
  }
348
349
  } else {
349
- console.log("\u2298 qmd not found \u2014 skipping semantic search setup.");
350
+ console.log("\u2298 qmd not found \u2014 skipping optional qmd fallback setup.");
350
351
  }
351
352
  console.log("\nTip: add this to your shell config:");
352
353
  console.log(` export CLAWVAULT_PATH="${target.vaultPath}"`);
@@ -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) {