clawvault 3.1.0 → 3.2.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 (289) 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 +1368 -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-F2JEUD4J.js → chunk-23YDQ3QU.js} +6 -8
  22. package/dist/{chunk-C7OK5WKP.js → chunk-2JQ3O2YL.js} +4 -4
  23. package/dist/{chunk-VR5NE7PZ.js → chunk-2RAZ4ZFE.js} +1 -1
  24. package/dist/chunk-2ZDO52B4.js +52 -0
  25. package/dist/{chunk-ZZA73MFY.js → chunk-33DOSHTA.js} +176 -36
  26. package/dist/chunk-33VSQP4J.js +37 -0
  27. package/dist/chunk-4BQTQMJP.js +93 -0
  28. package/dist/{chunk-GUKMRGM7.js → chunk-4OXMU5S2.js} +1 -1
  29. package/dist/{chunk-62YTUT6J.js → chunk-4PY655YM.js} +15 -3
  30. package/dist/chunk-6FH3IULF.js +352 -0
  31. package/dist/{chunk-3NSBOUT3.js → chunk-77Q5CSPJ.js} +404 -80
  32. package/dist/{chunk-4VQTUVH7.js → chunk-7YZWHM36.js} +52 -26
  33. package/dist/chunk-BSJ6RIT7.js +447 -0
  34. package/dist/chunk-BUEW6IIK.js +364 -0
  35. package/dist/{chunk-LI4O6NVK.js → chunk-CLJTREDS.js} +74 -14
  36. package/dist/chunk-EK6S23ZB.js +469 -0
  37. package/dist/{chunk-LNJA2UGL.js → chunk-ESFLMDRB.js} +9 -86
  38. package/dist/{chunk-H34S76MB.js → chunk-ESVS6K2B.js} +6 -6
  39. package/dist/{chunk-WAZ3NLWL.js → chunk-F55HGNU4.js} +0 -47
  40. package/dist/{chunk-QK3UCXWL.js → chunk-FHFUXL6G.js} +2 -2
  41. package/dist/{chunk-H62BP7RI.js → chunk-GAOWA7GR.js} +212 -46
  42. package/dist/chunk-GGA32J2R.js +784 -0
  43. package/dist/chunk-GNJL4YGR.js +79 -0
  44. package/dist/chunk-IVRIKYFE.js +520 -0
  45. package/dist/chunk-MDIH26GC.js +183 -0
  46. package/dist/{chunk-LYHGEHXG.js → chunk-MFAWT5O5.js} +0 -1
  47. package/dist/chunk-MM6QGW3P.js +207 -0
  48. package/dist/{chunk-P5EPF6MB.js → chunk-MW5C6ZQA.js} +110 -13
  49. package/dist/chunk-NCKFNBHJ.js +257 -0
  50. package/dist/{chunk-QBLMXKF2.js → chunk-OIWVQYQF.js} +1 -1
  51. package/dist/{chunk-42MXU7A6.js → chunk-P62WHA27.js} +58 -47
  52. package/dist/chunk-PBACDKKP.js +66 -0
  53. package/dist/{chunk-VGLOTGAS.js → chunk-QSHD36LH.js} +2 -2
  54. package/dist/{chunk-OZ7RIXTO.js → chunk-QSRRMEYM.js} +2 -2
  55. package/dist/chunk-QVEERJSP.js +152 -0
  56. package/dist/{chunk-N2AXRYLC.js → chunk-QWQ3TIKS.js} +1 -1
  57. package/dist/{chunk-3DHXQHYG.js → chunk-R2MIW5G7.js} +1 -1
  58. package/dist/{chunk-SJSFRIYS.js → chunk-SLXOR3CC.js} +2 -2
  59. package/dist/chunk-SS4B7P7V.js +99 -0
  60. package/dist/{chunk-JY6FYXIT.js → chunk-STCQGCEQ.js} +6 -11
  61. package/dist/chunk-TIGW564L.js +628 -0
  62. package/dist/chunk-U4O6C46S.js +154 -0
  63. package/dist/{chunk-ITPEXLHA.js → chunk-URXDAUVH.js} +24 -5
  64. package/dist/chunk-VSL7KY3M.js +189 -0
  65. package/dist/{chunk-U55BGUAU.js → chunk-W4SPAEE7.js} +6 -6
  66. package/dist/chunk-WMGIIABP.js +15 -0
  67. package/dist/{chunk-33UGEQRT.js → chunk-X3SPPUFG.js} +151 -64
  68. package/dist/chunk-Y6VJKXGL.js +373 -0
  69. package/dist/{chunk-3WRJEKN4.js → chunk-ZN54U2OZ.js} +123 -10
  70. package/dist/cli/index.js +34 -24
  71. package/dist/commands/archive.js +3 -3
  72. package/dist/commands/backlog.js +3 -3
  73. package/dist/commands/blocked.js +3 -3
  74. package/dist/commands/canvas.d.ts +15 -0
  75. package/dist/commands/canvas.js +200 -0
  76. package/dist/commands/checkpoint.js +2 -2
  77. package/dist/commands/compat.js +2 -2
  78. package/dist/commands/context.js +8 -6
  79. package/dist/commands/doctor.d.ts +11 -7
  80. package/dist/commands/doctor.js +18 -16
  81. package/dist/commands/embed.js +5 -6
  82. package/dist/commands/entities.js +2 -2
  83. package/dist/commands/graph.js +4 -4
  84. package/dist/commands/inject.d.ts +1 -1
  85. package/dist/commands/inject.js +5 -6
  86. package/dist/commands/kanban.js +4 -4
  87. package/dist/commands/link.js +5 -5
  88. package/dist/commands/migrate-observations.js +4 -4
  89. package/dist/commands/observe.d.ts +0 -1
  90. package/dist/commands/observe.js +14 -13
  91. package/dist/commands/project.js +5 -5
  92. package/dist/commands/rebuild-embeddings.d.ts +21 -0
  93. package/dist/commands/rebuild-embeddings.js +91 -0
  94. package/dist/commands/rebuild.js +12 -11
  95. package/dist/commands/recover.js +3 -3
  96. package/dist/commands/reflect.js +6 -7
  97. package/dist/commands/repair-session.js +1 -1
  98. package/dist/commands/replay.js +14 -14
  99. package/dist/commands/session-recap.js +1 -1
  100. package/dist/commands/setup.d.ts +2 -89
  101. package/dist/commands/setup.js +3 -21
  102. package/dist/commands/shell-init.js +1 -1
  103. package/dist/commands/sleep.d.ts +1 -1
  104. package/dist/commands/sleep.js +20 -19
  105. package/dist/commands/status.d.ts +2 -0
  106. package/dist/commands/status.js +57 -35
  107. package/dist/commands/sync-bd.d.ts +10 -0
  108. package/dist/commands/sync-bd.js +10 -0
  109. package/dist/commands/tailscale.d.ts +52 -0
  110. package/dist/commands/tailscale.js +26 -0
  111. package/dist/commands/task.js +4 -4
  112. package/dist/commands/template.js +2 -2
  113. package/dist/commands/wake.d.ts +1 -1
  114. package/dist/commands/wake.js +11 -10
  115. package/dist/commands/workgraph.d.ts +124 -0
  116. package/dist/commands/workgraph.js +38 -0
  117. package/dist/index.d.ts +341 -191
  118. package/dist/index.js +446 -116
  119. package/dist/{inject-Bzi5E-By.d.ts → inject-DYUrDqQO.d.ts} +3 -3
  120. package/dist/ledger-B7g7jhqG.d.ts +44 -0
  121. package/dist/lib/auto-linker.js +2 -2
  122. package/dist/lib/canvas-layout.d.ts +115 -0
  123. package/dist/lib/canvas-layout.js +35 -0
  124. package/dist/lib/config.d.ts +27 -3
  125. package/dist/lib/config.js +4 -2
  126. package/dist/lib/entity-index.js +1 -1
  127. package/dist/lib/project-utils.js +4 -4
  128. package/dist/lib/session-repair.js +1 -1
  129. package/dist/lib/session-utils.js +1 -1
  130. package/dist/lib/tailscale.d.ts +225 -0
  131. package/dist/lib/tailscale.js +50 -0
  132. package/dist/lib/task-utils.js +3 -3
  133. package/dist/lib/template-engine.js +1 -1
  134. package/dist/lib/webdav.d.ts +109 -0
  135. package/dist/lib/webdav.js +35 -0
  136. package/dist/onnxruntime_binding-5QEF3SUC.node +0 -0
  137. package/dist/onnxruntime_binding-BKPKNEGC.node +0 -0
  138. package/dist/onnxruntime_binding-FMOXGIUT.node +0 -0
  139. package/dist/onnxruntime_binding-OI2KMXC5.node +0 -0
  140. package/dist/onnxruntime_binding-UX44MLAZ.node +0 -0
  141. package/dist/onnxruntime_binding-Y2W7N7WY.node +0 -0
  142. package/dist/openclaw-plugin.d.ts +8 -0
  143. package/dist/openclaw-plugin.js +14 -0
  144. package/dist/registry-BR4326o0.d.ts +30 -0
  145. package/dist/store-CA-6sKCJ.d.ts +34 -0
  146. package/dist/thread-B9LhXNU0.d.ts +41 -0
  147. package/dist/transformers.node-A2ZRORSQ.js +46775 -0
  148. package/dist/{types-Y2_Um2Ls.d.ts → types-BbWJoC1c.d.ts} +1 -44
  149. package/dist/workgraph/index.d.ts +5 -0
  150. package/dist/workgraph/index.js +23 -0
  151. package/dist/workgraph/ledger.d.ts +2 -0
  152. package/dist/workgraph/ledger.js +25 -0
  153. package/dist/workgraph/registry.d.ts +2 -0
  154. package/dist/workgraph/registry.js +19 -0
  155. package/dist/workgraph/store.d.ts +2 -0
  156. package/dist/workgraph/store.js +25 -0
  157. package/dist/workgraph/thread.d.ts +2 -0
  158. package/dist/workgraph/thread.js +25 -0
  159. package/dist/workgraph/types.d.ts +54 -0
  160. package/dist/workgraph/types.js +7 -0
  161. package/hooks/clawvault/HOOK.md +113 -0
  162. package/hooks/clawvault/handler.js +1561 -0
  163. package/hooks/clawvault/handler.test.js +510 -0
  164. package/hooks/clawvault/openclaw.plugin.json +72 -0
  165. package/openclaw.plugin.json +65 -38
  166. package/package.json +25 -22
  167. package/dist/chunk-3RG5ZIWI.js +0 -10
  168. package/dist/chunk-3ZIH425O.js +0 -871
  169. package/dist/chunk-6U6MK36V.js +0 -205
  170. package/dist/chunk-CMB7UL7C.js +0 -327
  171. package/dist/chunk-D2H45LON.js +0 -1074
  172. package/dist/chunk-E7MFQB6D.js +0 -163
  173. package/dist/chunk-GQSLDZTS.js +0 -560
  174. package/dist/chunk-MFM6K7PU.js +0 -374
  175. package/dist/chunk-MXSSG3QU.js +0 -42
  176. package/dist/chunk-OCGVIN3L.js +0 -88
  177. package/dist/chunk-PAH27GSN.js +0 -108
  178. package/dist/chunk-YCUNCH2I.js +0 -78
  179. package/dist/cli/index.cjs +0 -8584
  180. package/dist/cli/index.d.cts +0 -5
  181. package/dist/commands/archive.cjs +0 -287
  182. package/dist/commands/archive.d.cts +0 -11
  183. package/dist/commands/backlog.cjs +0 -721
  184. package/dist/commands/backlog.d.cts +0 -53
  185. package/dist/commands/blocked.cjs +0 -204
  186. package/dist/commands/blocked.d.cts +0 -26
  187. package/dist/commands/checkpoint.cjs +0 -244
  188. package/dist/commands/checkpoint.d.cts +0 -41
  189. package/dist/commands/compat.cjs +0 -294
  190. package/dist/commands/compat.d.cts +0 -28
  191. package/dist/commands/context.cjs +0 -2990
  192. package/dist/commands/context.d.cts +0 -2
  193. package/dist/commands/doctor.cjs +0 -2986
  194. package/dist/commands/doctor.d.cts +0 -21
  195. package/dist/commands/embed.cjs +0 -232
  196. package/dist/commands/embed.d.cts +0 -17
  197. package/dist/commands/entities.cjs +0 -141
  198. package/dist/commands/entities.d.cts +0 -7
  199. package/dist/commands/graph.cjs +0 -501
  200. package/dist/commands/graph.d.cts +0 -21
  201. package/dist/commands/inject.cjs +0 -1636
  202. package/dist/commands/inject.d.cts +0 -2
  203. package/dist/commands/kanban.cjs +0 -884
  204. package/dist/commands/kanban.d.cts +0 -63
  205. package/dist/commands/link.cjs +0 -965
  206. package/dist/commands/link.d.cts +0 -11
  207. package/dist/commands/migrate-observations.cjs +0 -362
  208. package/dist/commands/migrate-observations.d.cts +0 -19
  209. package/dist/commands/observe.cjs +0 -4099
  210. package/dist/commands/observe.d.cts +0 -23
  211. package/dist/commands/project.cjs +0 -1341
  212. package/dist/commands/project.d.cts +0 -85
  213. package/dist/commands/rebuild.cjs +0 -3136
  214. package/dist/commands/rebuild.d.cts +0 -11
  215. package/dist/commands/recover.cjs +0 -361
  216. package/dist/commands/recover.d.cts +0 -38
  217. package/dist/commands/reflect.cjs +0 -1008
  218. package/dist/commands/reflect.d.cts +0 -11
  219. package/dist/commands/repair-session.cjs +0 -457
  220. package/dist/commands/repair-session.d.cts +0 -38
  221. package/dist/commands/replay.cjs +0 -4103
  222. package/dist/commands/replay.d.cts +0 -16
  223. package/dist/commands/session-recap.cjs +0 -353
  224. package/dist/commands/session-recap.d.cts +0 -27
  225. package/dist/commands/setup.cjs +0 -1278
  226. package/dist/commands/setup.d.cts +0 -99
  227. package/dist/commands/shell-init.cjs +0 -75
  228. package/dist/commands/shell-init.d.cts +0 -7
  229. package/dist/commands/sleep.cjs +0 -6029
  230. package/dist/commands/sleep.d.cts +0 -36
  231. package/dist/commands/status.cjs +0 -2737
  232. package/dist/commands/status.d.cts +0 -52
  233. package/dist/commands/task.cjs +0 -1236
  234. package/dist/commands/task.d.cts +0 -97
  235. package/dist/commands/template.cjs +0 -457
  236. package/dist/commands/template.d.cts +0 -36
  237. package/dist/commands/wake.cjs +0 -2627
  238. package/dist/commands/wake.d.cts +0 -22
  239. package/dist/context-BUGaWpyL.d.cts +0 -46
  240. package/dist/index.cjs +0 -12373
  241. package/dist/index.d.cts +0 -854
  242. package/dist/inject-Bzi5E-By.d.cts +0 -137
  243. package/dist/lib/auto-linker.cjs +0 -176
  244. package/dist/lib/auto-linker.d.cts +0 -26
  245. package/dist/lib/config.cjs +0 -78
  246. package/dist/lib/config.d.cts +0 -11
  247. package/dist/lib/entity-index.cjs +0 -84
  248. package/dist/lib/entity-index.d.cts +0 -26
  249. package/dist/lib/project-utils.cjs +0 -864
  250. package/dist/lib/project-utils.d.cts +0 -97
  251. package/dist/lib/session-repair.cjs +0 -239
  252. package/dist/lib/session-repair.d.cts +0 -110
  253. package/dist/lib/session-utils.cjs +0 -209
  254. package/dist/lib/session-utils.d.cts +0 -63
  255. package/dist/lib/task-utils.cjs +0 -1137
  256. package/dist/lib/task-utils.d.cts +0 -208
  257. package/dist/lib/template-engine.cjs +0 -47
  258. package/dist/lib/template-engine.d.cts +0 -11
  259. package/dist/plugin/index.cjs +0 -1907
  260. package/dist/plugin/index.d.cts +0 -36
  261. package/dist/plugin/index.d.ts +0 -36
  262. package/dist/plugin/index.js +0 -572
  263. package/dist/plugin/inject.cjs +0 -356
  264. package/dist/plugin/inject.d.cts +0 -54
  265. package/dist/plugin/inject.d.ts +0 -54
  266. package/dist/plugin/inject.js +0 -17
  267. package/dist/plugin/observe.cjs +0 -631
  268. package/dist/plugin/observe.d.cts +0 -39
  269. package/dist/plugin/observe.d.ts +0 -39
  270. package/dist/plugin/observe.js +0 -18
  271. package/dist/plugin/templates.cjs +0 -593
  272. package/dist/plugin/templates.d.cts +0 -52
  273. package/dist/plugin/templates.d.ts +0 -52
  274. package/dist/plugin/templates.js +0 -25
  275. package/dist/plugin/types.cjs +0 -18
  276. package/dist/plugin/types.d.cts +0 -209
  277. package/dist/plugin/types.d.ts +0 -209
  278. package/dist/plugin/types.js +0 -0
  279. package/dist/plugin/vault.cjs +0 -927
  280. package/dist/plugin/vault.d.cts +0 -68
  281. package/dist/plugin/vault.d.ts +0 -68
  282. package/dist/plugin/vault.js +0 -22
  283. package/dist/types-Y2_Um2Ls.d.cts +0 -205
  284. package/templates/memory-event.md +0 -67
  285. package/templates/party.md +0 -63
  286. package/templates/primitive-registry.yaml +0 -551
  287. package/templates/run.md +0 -68
  288. package/templates/trigger.md +0 -68
  289. package/templates/workspace.md +0 -50
@@ -1,13 +1,16 @@
1
1
  import {
2
2
  buildEntityIndex
3
3
  } from "./chunk-J7ZWCI2C.js";
4
+ import {
5
+ extractRawWikiLinks,
6
+ normalizeWikiLinkTarget
7
+ } from "./chunk-33DOSHTA.js";
4
8
 
5
9
  // src/lib/backlinks.ts
6
10
  import * as fs from "fs";
7
11
  import * as path from "path";
8
12
  var CLAWVAULT_DIR = ".clawvault";
9
13
  var BACKLINKS_FILE = "backlinks.json";
10
- var WIKI_LINK_REGEX = /\[\[([^\]]+)\]\]/g;
11
14
  function ensureClawvaultDir(vaultPath) {
12
15
  const dir = path.join(vaultPath, CLAWVAULT_DIR);
13
16
  if (!fs.existsSync(dir)) {
@@ -20,29 +23,47 @@ function toVaultId(vaultPath, filePath) {
20
23
  return relative2.split(path.sep).join("/");
21
24
  }
22
25
  function normalizeLinkTarget(raw) {
23
- let target = raw.trim();
24
- if (!target) return "";
25
- if (target.startsWith("[[") && target.endsWith("]]")) {
26
- target = target.slice(2, -2);
27
- }
28
- const pipeIndex = target.indexOf("|");
29
- if (pipeIndex !== -1) {
30
- target = target.slice(0, pipeIndex);
31
- }
32
- if (target.startsWith("#")) return "";
33
- const hashIndex = target.indexOf("#");
34
- if (hashIndex !== -1) {
35
- target = target.slice(0, hashIndex);
26
+ return normalizeWikiLinkTarget(raw);
27
+ }
28
+ function normalizeLookupCandidate(value) {
29
+ const normalized = normalizeLinkTarget(value);
30
+ if (!normalized) return "";
31
+ const resolved = path.posix.normalize(normalized).replace(/^\/+/, "");
32
+ if (!resolved || resolved === "." || resolved.startsWith("../")) {
33
+ return "";
36
34
  }
37
- target = target.trim();
38
- if (!target) return "";
39
- if (target.endsWith(".md")) {
40
- target = target.slice(0, -3);
35
+ return resolved;
36
+ }
37
+ function buildLookupCandidates(target, sourceId) {
38
+ const candidates = [];
39
+ const sourceDir = path.posix.dirname(sourceId);
40
+ const hasSourceDir = sourceDir !== ".";
41
+ const isRelativeTarget = target.startsWith("./") || target.startsWith("../");
42
+ const addCandidate = (candidate) => {
43
+ const normalized = normalizeLookupCandidate(candidate);
44
+ if (!normalized || candidates.includes(normalized)) return;
45
+ candidates.push(normalized);
46
+ };
47
+ if (isRelativeTarget) {
48
+ if (hasSourceDir) {
49
+ addCandidate(path.posix.join(sourceDir, target));
50
+ } else {
51
+ addCandidate(target);
52
+ }
53
+ if (target.startsWith("./")) {
54
+ addCandidate(target.slice(2));
55
+ }
56
+ return candidates;
41
57
  }
42
- if (target.startsWith("/")) {
43
- target = target.slice(1);
58
+ if (!target.includes("/")) {
59
+ if (hasSourceDir) {
60
+ addCandidate(`${sourceDir}/${target}`);
61
+ }
62
+ addCandidate(target);
63
+ return candidates;
44
64
  }
45
- return target.replace(/\\/g, "/");
65
+ addCandidate(target);
66
+ return candidates;
46
67
  }
47
68
  function listMarkdownFiles(vaultPath) {
48
69
  const files = [];
@@ -75,11 +96,16 @@ function buildKnownIds(vaultPath, files) {
75
96
  }
76
97
  return { ids, idsLower };
77
98
  }
78
- function resolveTarget(target, known, entityIndex) {
99
+ function resolveTarget(target, sourceId, known, entityIndex) {
79
100
  if (!target) return null;
80
- if (known.ids.has(target)) return target;
101
+ for (const candidate of buildLookupCandidates(target, sourceId)) {
102
+ if (known.ids.has(candidate)) return candidate;
103
+ const lowerCandidate = candidate.toLowerCase();
104
+ if (known.idsLower.has(lowerCandidate)) {
105
+ return known.idsLower.get(lowerCandidate);
106
+ }
107
+ }
81
108
  const lower = target.toLowerCase();
82
- if (known.idsLower.has(lower)) return known.idsLower.get(lower);
83
109
  if (entityIndex?.entries.has(lower)) return entityIndex.entries.get(lower);
84
110
  return null;
85
111
  }
@@ -93,12 +119,12 @@ function scanVaultLinks(vaultPath, options = {}) {
93
119
  for (const file of files) {
94
120
  const sourceId = toVaultId(vaultPath, file);
95
121
  const content = fs.readFileSync(file, "utf-8");
96
- const matches = content.match(WIKI_LINK_REGEX) || [];
122
+ const matches = extractRawWikiLinks(content);
97
123
  linkCount += matches.length;
98
124
  for (const match of matches) {
99
125
  const target = normalizeLinkTarget(match);
100
126
  if (!target) continue;
101
- const resolved = resolveTarget(target, known, entityIndex);
127
+ const resolved = resolveTarget(target, sourceId, known, entityIndex);
102
128
  if (!resolved) {
103
129
  orphans.push({ source: sourceId, target });
104
130
  continue;
@@ -0,0 +1,447 @@
1
+ // src/lib/fact-extractor.ts
2
+ function normalizeEntity(name) {
3
+ return name.toLowerCase().replace(/[^a-z0-9\s]/g, "").replace(/\s+/g, " ").trim();
4
+ }
5
+ function factId(entity, relation, value) {
6
+ const key = `${normalizeEntity(entity)}::${relation.toLowerCase()}::${value.toLowerCase().trim()}`;
7
+ let hash = 0;
8
+ for (let i = 0; i < key.length; i++) {
9
+ const char = key.charCodeAt(i);
10
+ hash = (hash << 5) - hash + char;
11
+ hash = hash & hash;
12
+ }
13
+ return Math.abs(hash).toString(36);
14
+ }
15
+ var PREFERENCE_PATTERNS = [
16
+ {
17
+ // "I prefer X" / "I like X" / "I love X" / "I enjoy X"
18
+ pattern: /\b(?:i|user|they)\s+(?:prefer|like|love|enjoy|want|favor)s?\s+(.+?)(?:\.|,|$)/i,
19
+ extract: (m) => ({
20
+ entity: "user",
21
+ relation: "prefers",
22
+ value: m[1].trim(),
23
+ category: "preference"
24
+ })
25
+ },
26
+ {
27
+ // "my favorite X is Y"
28
+ pattern: /\bmy\s+(?:favorite|favourite|preferred)\s+(\w+)\s+(?:is|are)\s+(.+?)(?:\.|,|$)/i,
29
+ extract: (m) => ({
30
+ entity: "user",
31
+ relation: `favorite_${m[1].toLowerCase()}`,
32
+ value: m[2].trim(),
33
+ category: "preference"
34
+ })
35
+ },
36
+ {
37
+ // "I don't like X" / "I hate X" / "I dislike X"
38
+ pattern: /\b(?:i|user)\s+(?:don'?t\s+like|hate|dislike|avoid)s?\s+(.+?)(?:\.|,|$)/i,
39
+ extract: (m) => ({
40
+ entity: "user",
41
+ relation: "dislikes",
42
+ value: m[1].trim(),
43
+ category: "preference"
44
+ })
45
+ },
46
+ {
47
+ // "I'm allergic to X" / "I have an allergy to X"
48
+ pattern: /\b(?:i'?m|i\s+am|i\s+have)\s+(?:an?\s+)?allerg(?:ic|y)\s+(?:to\s+)?(.+?)(?:\.|,|$)/i,
49
+ extract: (m) => ({
50
+ entity: "user",
51
+ relation: "allergic_to",
52
+ value: m[1].trim(),
53
+ category: "preference"
54
+ })
55
+ }
56
+ ];
57
+ var FACT_PATTERNS = [
58
+ {
59
+ // "X works at Y" / "X is employed at Y"
60
+ pattern: /\b(\w+(?:\s+\w+)?)\s+(?:works?\s+(?:at|for)|is\s+employed\s+(?:at|by))\s+(.+?)(?:\.|,|$)/i,
61
+ extract: (m) => ({
62
+ entity: m[1].trim(),
63
+ relation: "works_at",
64
+ value: m[2].trim(),
65
+ category: "fact"
66
+ })
67
+ },
68
+ {
69
+ // "X lives in Y" / "X moved to Y"
70
+ pattern: /\b(\w+(?:\s+\w+)?)\s+(?:live[sd]?\s+in|moved?\s+to|relocated?\s+to)\s+(.+?)(?:\.|,|$)/i,
71
+ extract: (m) => ({
72
+ entity: m[1].trim(),
73
+ relation: "lives_in",
74
+ value: m[2].trim(),
75
+ category: "fact"
76
+ })
77
+ },
78
+ {
79
+ // "X is Y years old" / "X's age is Y"
80
+ pattern: /\b(\w+(?:\s+\w+)?)\s+(?:is|turned)\s+(\d+)\s+years?\s+old/i,
81
+ extract: (m) => ({
82
+ entity: m[1].trim(),
83
+ relation: "age",
84
+ value: m[2],
85
+ category: "fact"
86
+ })
87
+ },
88
+ {
89
+ // "X bought Y" / "X purchased Y"
90
+ pattern: /\b(\w+(?:\s+\w+)?)\s+(?:bought|purchased|got|acquired)\s+(?:a\s+|an\s+|the\s+)?(.+?)(?:\s+for\s+\$?([\d,.]+))?(?:\.|,|$)/i,
91
+ extract: (m) => ({
92
+ entity: m[1].trim(),
93
+ relation: "bought",
94
+ value: m[3] ? `${m[2].trim()} ($${m[3]})` : m[2].trim(),
95
+ category: "event"
96
+ })
97
+ },
98
+ {
99
+ // "X spent $Y on Z"
100
+ pattern: /\b(\w+(?:\s+\w+)?)\s+spent\s+\$?([\d,.]+)\s+on\s+(.+?)(?:\.|,|$)/i,
101
+ extract: (m) => ({
102
+ entity: m[1].trim(),
103
+ relation: "spent_on",
104
+ value: `$${m[2]} on ${m[3].trim()}`,
105
+ category: "event"
106
+ })
107
+ }
108
+ ];
109
+ var DECISION_PATTERNS = [
110
+ {
111
+ // "decided to X" / "we decided X"
112
+ pattern: /\b(?:i|we|user)\s+decided\s+(?:to\s+)?(.+?)(?:\.|,|$)/i,
113
+ extract: (m) => ({
114
+ entity: "user",
115
+ relation: "decided",
116
+ value: m[1].trim(),
117
+ category: "decision"
118
+ })
119
+ },
120
+ {
121
+ // "chose X over Y"
122
+ pattern: /\b(?:i|we|user)\s+chose\s+(.+?)\s+over\s+(.+?)(?:\.|,|$)/i,
123
+ extract: (m) => ({
124
+ entity: "user",
125
+ relation: "chose",
126
+ value: `${m[1].trim()} (over ${m[2].trim()})`,
127
+ category: "decision"
128
+ })
129
+ }
130
+ ];
131
+ var ALL_PATTERNS = [...PREFERENCE_PATTERNS, ...FACT_PATTERNS, ...DECISION_PATTERNS];
132
+ function extractFactsRuleBased(text, source, timestamp) {
133
+ const facts = [];
134
+ const now = timestamp || (/* @__PURE__ */ new Date()).toISOString();
135
+ const sentences = text.split(/[.!?\n]+/).filter((s) => s.trim().length > 5);
136
+ for (const sentence of sentences) {
137
+ const trimmed = sentence.trim();
138
+ for (const rule of ALL_PATTERNS) {
139
+ const match = trimmed.match(rule.pattern);
140
+ if (match) {
141
+ const extracted = rule.extract(match);
142
+ if (extracted && extracted.value.length > 1 && extracted.value.length < 200) {
143
+ facts.push({
144
+ id: factId(extracted.entity, extracted.relation, extracted.value),
145
+ entity: extracted.entity,
146
+ entityNorm: normalizeEntity(extracted.entity),
147
+ relation: extracted.relation,
148
+ value: extracted.value,
149
+ validFrom: now,
150
+ validUntil: null,
151
+ confidence: 0.7,
152
+ // Rule-based gets moderate confidence
153
+ category: extracted.category,
154
+ source,
155
+ rawText: trimmed
156
+ });
157
+ }
158
+ }
159
+ }
160
+ }
161
+ return facts;
162
+ }
163
+ var EXTRACTION_PROMPT = `Extract structured facts from the following text. Return ONLY a JSON array of objects with these fields:
164
+ - entity: the subject (person, place, thing, or "user" for the speaker/first person)
165
+ - relation: the relationship type (see examples below)
166
+ - value: the object of the relation
167
+ - category: one of "preference", "fact", "decision", "entity", "event"
168
+ - confidence: 0.0 to 1.0
169
+
170
+ PREFERENCE EXTRACTION (critical \u2014 extract ALL of these):
171
+ - Likes, dislikes, preferences, favorites: "prefers", "likes", "dislikes", "favorite"
172
+ - Food/dietary: "allergic_to", "dietary_restriction", "favorite_food", "dislikes_food"
173
+ - Habits/routines: "habit", "routine", "schedule"
174
+ - Communication style: "prefers_communication", "timezone", "language"
175
+ - Tools/tech: "uses_tool", "prefers_editor", "prefers_language"
176
+
177
+ TEMPORAL FACTS (include dates when present):
178
+ - Include specific dates, times, relative references ("last Tuesday" = resolve if possible)
179
+ - Events: "happened_on", "started_on", "ended_on", "deadline"
180
+ - Use ISO format for dates when possible
181
+
182
+ OTHER RELATIONS:
183
+ - Identity: "works_at", "lives_in", "age", "role", "email", "phone"
184
+ - Actions: "bought", "spent_on", "created", "visited", "completed"
185
+ - Decisions: "decided", "chose", "rejected", "approved"
186
+ - Knowledge: "knows_about", "studied", "expertise"
187
+
188
+ Examples:
189
+
190
+ Input: "I really love Thai food, especially pad thai. I'm allergic to shellfish though."
191
+ Output: [
192
+ {"entity": "user", "relation": "favorite_food", "value": "Thai food, especially pad thai", "category": "preference", "confidence": 0.95},
193
+ {"entity": "user", "relation": "allergic_to", "value": "shellfish", "category": "preference", "confidence": 0.99}
194
+ ]
195
+
196
+ Input: "We decided on Tuesday to use PostgreSQL for the new project. John will lead the backend team."
197
+ Output: [
198
+ {"entity": "team", "relation": "decided", "value": "use PostgreSQL for the new project", "category": "decision", "confidence": 0.95},
199
+ {"entity": "John", "relation": "role", "value": "backend team lead", "category": "fact", "confidence": 0.9}
200
+ ]
201
+
202
+ Input: "My morning routine is: wake up at 6am, coffee, then gym. I prefer working out before work."
203
+ Output: [
204
+ {"entity": "user", "relation": "routine", "value": "wake up at 6am, coffee, then gym", "category": "preference", "confidence": 0.9},
205
+ {"entity": "user", "relation": "prefers", "value": "working out before work", "category": "preference", "confidence": 0.9}
206
+ ]
207
+
208
+ Rules:
209
+ - Extract ALL facts, preferences, decisions, and events \u2014 err on the side of extracting more
210
+ - For preferences, use "user" as entity unless a specific person is named
211
+ - For monetary amounts, include the currency symbol
212
+ - Be precise \u2014 only extract what is explicitly stated or strongly implied
213
+ - Return empty array [] if no extractable facts found
214
+
215
+ Text:
216
+ `;
217
+ async function extractFactsLlm(text, source, timestamp, llmFn) {
218
+ if (!llmFn) {
219
+ return extractFactsRuleBased(text, source, timestamp);
220
+ }
221
+ const now = timestamp || (/* @__PURE__ */ new Date()).toISOString();
222
+ try {
223
+ const response = await llmFn(EXTRACTION_PROMPT + text);
224
+ const jsonMatch = response.match(/\[[\s\S]*?\]/);
225
+ if (!jsonMatch) {
226
+ return extractFactsRuleBased(text, source, timestamp);
227
+ }
228
+ const parsed = JSON.parse(jsonMatch[0]);
229
+ return parsed.map((f) => ({
230
+ id: factId(f.entity, f.relation, f.value),
231
+ entity: f.entity,
232
+ entityNorm: normalizeEntity(f.entity),
233
+ relation: f.relation,
234
+ value: f.value,
235
+ validFrom: now,
236
+ validUntil: null,
237
+ confidence: Math.min(1, Math.max(0, f.confidence || 0.8)),
238
+ category: f.category || "fact",
239
+ source,
240
+ rawText: text.substring(0, 500)
241
+ }));
242
+ } catch {
243
+ return extractFactsRuleBased(text, source, timestamp);
244
+ }
245
+ }
246
+
247
+ // src/lib/fact-store.ts
248
+ import * as fs from "fs";
249
+ import * as path from "path";
250
+ var FactStore = class {
251
+ facts = /* @__PURE__ */ new Map();
252
+ byEntity = /* @__PURE__ */ new Map();
253
+ byRelation = /* @__PURE__ */ new Map();
254
+ byCategory = /* @__PURE__ */ new Map();
255
+ factsPath;
256
+ dirty = false;
257
+ constructor(vaultPath) {
258
+ this.factsPath = path.join(vaultPath, ".clawvault", "facts.jsonl");
259
+ }
260
+ /** Load facts from disk */
261
+ load() {
262
+ this.facts.clear();
263
+ this.byEntity.clear();
264
+ this.byRelation.clear();
265
+ this.byCategory.clear();
266
+ if (!fs.existsSync(this.factsPath)) return;
267
+ const lines = fs.readFileSync(this.factsPath, "utf-8").split("\n").filter((l) => l.trim());
268
+ for (const line of lines) {
269
+ try {
270
+ const fact = JSON.parse(line);
271
+ this.indexFact(fact);
272
+ } catch {
273
+ }
274
+ }
275
+ }
276
+ /** Add facts with conflict resolution. Returns number of conflicts resolved. */
277
+ addFacts(newFacts) {
278
+ let conflicts = 0;
279
+ for (const fact of newFacts) {
280
+ const existing = this.findConflict(fact);
281
+ if (existing) {
282
+ existing.validUntil = fact.validFrom;
283
+ conflicts++;
284
+ }
285
+ this.indexFact(fact);
286
+ this.dirty = true;
287
+ }
288
+ return conflicts;
289
+ }
290
+ /** Find an existing fact that conflicts with the new one */
291
+ findConflict(newFact) {
292
+ const entityFacts = this.byEntity.get(newFact.entityNorm);
293
+ if (!entityFacts) return null;
294
+ for (const id of entityFacts) {
295
+ const existing = this.facts.get(id);
296
+ if (!existing || existing.validUntil) continue;
297
+ if (existing.relation === newFact.relation) {
298
+ if (this.isSimilarValue(existing.value, newFact.value)) {
299
+ return existing;
300
+ }
301
+ if (this.isExclusiveRelation(newFact.relation)) {
302
+ return existing;
303
+ }
304
+ }
305
+ }
306
+ return null;
307
+ }
308
+ /** Check if two values are similar enough to be considered the same fact */
309
+ isSimilarValue(a, b) {
310
+ const na = a.toLowerCase().trim();
311
+ const nb = b.toLowerCase().trim();
312
+ if (na === nb) return true;
313
+ if (na.includes(nb) || nb.includes(na)) return true;
314
+ return false;
315
+ }
316
+ /** Relations where only one value can be active at a time */
317
+ isExclusiveRelation(relation) {
318
+ const exclusive = /* @__PURE__ */ new Set([
319
+ "lives_in",
320
+ "works_at",
321
+ "age",
322
+ "favorite_color",
323
+ "favorite_food",
324
+ "favorite_restaurant",
325
+ "favorite_movie",
326
+ "favorite_book",
327
+ "favorite_music",
328
+ "favorite_sport",
329
+ "job_title",
330
+ "employer",
331
+ "marital_status",
332
+ "city",
333
+ "country"
334
+ ]);
335
+ return exclusive.has(relation);
336
+ }
337
+ /** Index a fact in all lookup maps */
338
+ indexFact(fact) {
339
+ this.facts.set(fact.id, fact);
340
+ if (!this.byEntity.has(fact.entityNorm)) {
341
+ this.byEntity.set(fact.entityNorm, /* @__PURE__ */ new Set());
342
+ }
343
+ this.byEntity.get(fact.entityNorm).add(fact.id);
344
+ if (!this.byRelation.has(fact.relation)) {
345
+ this.byRelation.set(fact.relation, /* @__PURE__ */ new Set());
346
+ }
347
+ this.byRelation.get(fact.relation).add(fact.id);
348
+ if (!this.byCategory.has(fact.category)) {
349
+ this.byCategory.set(fact.category, /* @__PURE__ */ new Set());
350
+ }
351
+ this.byCategory.get(fact.category).add(fact.id);
352
+ }
353
+ /** Save facts to disk (full rewrite for consistency) */
354
+ save() {
355
+ if (!this.dirty && fs.existsSync(this.factsPath)) return;
356
+ const dir = path.dirname(this.factsPath);
357
+ if (!fs.existsSync(dir)) {
358
+ fs.mkdirSync(dir, { recursive: true });
359
+ }
360
+ const lines = Array.from(this.facts.values()).map((f) => JSON.stringify(f)).join("\n");
361
+ fs.writeFileSync(this.factsPath, lines + "\n", "utf-8");
362
+ this.dirty = false;
363
+ }
364
+ /** Append new facts to disk (faster than full rewrite) */
365
+ append(facts) {
366
+ const dir = path.dirname(this.factsPath);
367
+ if (!fs.existsSync(dir)) {
368
+ fs.mkdirSync(dir, { recursive: true });
369
+ }
370
+ const lines = facts.map((f) => JSON.stringify(f)).join("\n");
371
+ fs.appendFileSync(this.factsPath, lines + "\n", "utf-8");
372
+ }
373
+ // ─── Query methods ──────────────────────────────────────────────────────
374
+ /** Get all active facts for an entity */
375
+ getEntityFacts(entity) {
376
+ const norm = normalizeEntity(entity);
377
+ const ids = this.byEntity.get(norm);
378
+ if (!ids) return [];
379
+ return Array.from(ids).map((id) => this.facts.get(id)).filter((f) => f && !f.validUntil);
380
+ }
381
+ /** Get all active facts for a relation */
382
+ getRelationFacts(relation) {
383
+ const ids = this.byRelation.get(relation);
384
+ if (!ids) return [];
385
+ return Array.from(ids).map((id) => this.facts.get(id)).filter((f) => f && !f.validUntil);
386
+ }
387
+ /** Get all active facts in a category */
388
+ getCategoryFacts(category) {
389
+ const ids = this.byCategory.get(category);
390
+ if (!ids) return [];
391
+ return Array.from(ids).map((id) => this.facts.get(id)).filter((f) => f && !f.validUntil);
392
+ }
393
+ /** Get all active preferences */
394
+ getPreferences() {
395
+ return this.getCategoryFacts("preference");
396
+ }
397
+ /** Search facts by text query (simple keyword match) */
398
+ searchFacts(query) {
399
+ const terms = query.toLowerCase().split(/\s+/);
400
+ const results = [];
401
+ for (const fact of this.facts.values()) {
402
+ if (fact.validUntil) continue;
403
+ const text = `${fact.entity} ${fact.relation} ${fact.value} ${fact.rawText}`.toLowerCase();
404
+ const matches = terms.filter((t) => text.includes(t)).length;
405
+ if (matches >= Math.ceil(terms.length * 0.5)) {
406
+ results.push(fact);
407
+ }
408
+ }
409
+ return results;
410
+ }
411
+ /** Get facts valid at a specific time */
412
+ getFactsAt(timestamp) {
413
+ const t = new Date(timestamp).getTime();
414
+ const results = [];
415
+ for (const fact of this.facts.values()) {
416
+ const from = new Date(fact.validFrom).getTime();
417
+ const until = fact.validUntil ? new Date(fact.validUntil).getTime() : Infinity;
418
+ if (t >= from && t < until) {
419
+ results.push(fact);
420
+ }
421
+ }
422
+ return results;
423
+ }
424
+ /** Get stats */
425
+ stats() {
426
+ const active = Array.from(this.facts.values()).filter((f) => !f.validUntil);
427
+ return {
428
+ totalFacts: this.facts.size,
429
+ activeFacts: active.length,
430
+ supersededFacts: this.facts.size - active.length,
431
+ entities: this.byEntity.size,
432
+ relations: this.byRelation.size
433
+ };
434
+ }
435
+ /** Get all facts (for testing/debugging) */
436
+ getAllFacts() {
437
+ return Array.from(this.facts.values());
438
+ }
439
+ };
440
+
441
+ export {
442
+ normalizeEntity,
443
+ factId,
444
+ extractFactsRuleBased,
445
+ extractFactsLlm,
446
+ FactStore
447
+ };