clawvault 3.1.0 → 3.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (273) hide show
  1. package/README.md +422 -141
  2. package/bin/clawvault.js +10 -2
  3. package/bin/command-registration.test.js +3 -1
  4. package/bin/command-runtime.js +9 -1
  5. package/bin/register-core-commands.js +23 -28
  6. package/bin/register-maintenance-commands.js +39 -3
  7. package/bin/register-query-commands.js +58 -29
  8. package/bin/register-tailscale-commands.js +106 -0
  9. package/bin/register-task-commands.js +18 -1
  10. package/bin/register-task-commands.test.js +16 -0
  11. package/bin/register-vault-operations-commands.js +29 -1
  12. package/bin/register-workgraph-commands.js +451 -0
  13. package/dashboard/lib/graph-diff.js +104 -0
  14. package/dashboard/lib/graph-diff.test.js +75 -0
  15. package/dashboard/lib/vault-parser.js +556 -0
  16. package/dashboard/lib/vault-parser.test.js +254 -0
  17. package/dashboard/public/app.js +796 -0
  18. package/dashboard/public/index.html +52 -0
  19. package/dashboard/public/styles.css +221 -0
  20. package/dashboard/server.js +374 -0
  21. package/dist/{chunk-C7OK5WKP.js → chunk-2JQ3O2YL.js} +4 -4
  22. package/dist/{chunk-VR5NE7PZ.js → chunk-2RAZ4ZFE.js} +1 -1
  23. package/dist/{chunk-F2JEUD4J.js → chunk-4ITRXIVT.js} +5 -7
  24. package/dist/{chunk-GUKMRGM7.js → chunk-4OXMU5S2.js} +1 -1
  25. package/dist/chunk-5PJ4STIC.js +465 -0
  26. package/dist/{chunk-62YTUT6J.js → chunk-AZYOKJYC.js} +2 -2
  27. package/dist/chunk-BSJ6RIT7.js +447 -0
  28. package/dist/chunk-ECRZL5XR.js +50 -0
  29. package/dist/chunk-ERNE2FZ5.js +189 -0
  30. package/dist/{chunk-WAZ3NLWL.js → chunk-F55HGNU4.js} +0 -47
  31. package/dist/{chunk-VGLOTGAS.js → chunk-FAKNOB7Y.js} +2 -2
  32. package/dist/{chunk-QK3UCXWL.js → chunk-FHFUXL6G.js} +2 -2
  33. package/dist/chunk-GNJL4YGR.js +79 -0
  34. package/dist/chunk-HR4KN6S2.js +152 -0
  35. package/dist/{chunk-OZ7RIXTO.js → chunk-IIOU45CK.js} +1 -1
  36. package/dist/chunk-IJBFGPCS.js +33 -0
  37. package/dist/chunk-IVRIKYFE.js +520 -0
  38. package/dist/chunk-K7PNYS45.js +93 -0
  39. package/dist/chunk-MDIH26GC.js +183 -0
  40. package/dist/{chunk-LYHGEHXG.js → chunk-MFAWT5O5.js} +0 -1
  41. package/dist/{chunk-H34S76MB.js → chunk-MNPUYCHQ.js} +6 -6
  42. package/dist/chunk-NTOPJI7W.js +207 -0
  43. package/dist/{chunk-QBLMXKF2.js → chunk-OIWVQYQF.js} +1 -1
  44. package/dist/chunk-PG56HX5T.js +154 -0
  45. package/dist/{chunk-LNJA2UGL.js → chunk-PI4WMLMG.js} +7 -84
  46. package/dist/chunk-QMHPQYUV.js +363 -0
  47. package/dist/{chunk-H62BP7RI.js → chunk-QPDDIHXE.js} +209 -43
  48. package/dist/{chunk-N2AXRYLC.js → chunk-QWQ3TIKS.js} +1 -1
  49. package/dist/{chunk-3DHXQHYG.js → chunk-R2MIW5G7.js} +1 -1
  50. package/dist/{chunk-SJSFRIYS.js → chunk-S5OJEGFG.js} +2 -2
  51. package/dist/chunk-SS4B7P7V.js +99 -0
  52. package/dist/chunk-TIGW564L.js +628 -0
  53. package/dist/chunk-U67V476Y.js +35 -0
  54. package/dist/{chunk-JY6FYXIT.js → chunk-UCQAOZHW.js} +6 -11
  55. package/dist/{chunk-ITPEXLHA.js → chunk-URXDAUVH.js} +24 -5
  56. package/dist/chunk-WIOLLGAD.js +190 -0
  57. package/dist/{chunk-3WRJEKN4.js → chunk-WJVWINEM.js} +72 -8
  58. package/dist/chunk-WMGIIABP.js +15 -0
  59. package/dist/{chunk-33UGEQRT.js → chunk-X3SPPUFG.js} +151 -64
  60. package/dist/{chunk-3NSBOUT3.js → chunk-Y3TIJEBP.js} +314 -79
  61. package/dist/chunk-Y6VJKXGL.js +373 -0
  62. package/dist/{chunk-LI4O6NVK.js → chunk-YDWHS4LJ.js} +49 -9
  63. package/dist/{chunk-U55BGUAU.js → chunk-YNIPYN4F.js} +5 -5
  64. package/dist/chunk-YXQCA6B7.js +226 -0
  65. package/dist/cli/index.js +26 -22
  66. package/dist/commands/archive.js +3 -3
  67. package/dist/commands/backlog.js +3 -3
  68. package/dist/commands/blocked.js +3 -3
  69. package/dist/commands/canvas.d.ts +15 -0
  70. package/dist/commands/canvas.js +200 -0
  71. package/dist/commands/checkpoint.js +2 -2
  72. package/dist/commands/compat.js +2 -2
  73. package/dist/commands/context.js +7 -5
  74. package/dist/commands/doctor.d.ts +11 -7
  75. package/dist/commands/doctor.js +16 -14
  76. package/dist/commands/embed.js +5 -6
  77. package/dist/commands/entities.js +2 -2
  78. package/dist/commands/graph.js +3 -3
  79. package/dist/commands/inject.d.ts +1 -1
  80. package/dist/commands/inject.js +4 -5
  81. package/dist/commands/kanban.js +4 -4
  82. package/dist/commands/link.js +2 -2
  83. package/dist/commands/migrate-observations.js +4 -4
  84. package/dist/commands/observe.d.ts +0 -1
  85. package/dist/commands/observe.js +13 -12
  86. package/dist/commands/project.js +5 -5
  87. package/dist/commands/rebuild-embeddings.d.ts +21 -0
  88. package/dist/commands/rebuild-embeddings.js +91 -0
  89. package/dist/commands/rebuild.js +12 -11
  90. package/dist/commands/recover.js +3 -3
  91. package/dist/commands/reflect.js +6 -7
  92. package/dist/commands/repair-session.js +1 -1
  93. package/dist/commands/replay.js +14 -14
  94. package/dist/commands/session-recap.js +1 -1
  95. package/dist/commands/setup.d.ts +2 -89
  96. package/dist/commands/setup.js +3 -21
  97. package/dist/commands/shell-init.js +1 -1
  98. package/dist/commands/sleep.d.ts +1 -1
  99. package/dist/commands/sleep.js +18 -17
  100. package/dist/commands/status.d.ts +2 -0
  101. package/dist/commands/status.js +40 -30
  102. package/dist/commands/sync-bd.d.ts +10 -0
  103. package/dist/commands/sync-bd.js +10 -0
  104. package/dist/commands/tailscale.d.ts +52 -0
  105. package/dist/commands/tailscale.js +26 -0
  106. package/dist/commands/task.js +4 -4
  107. package/dist/commands/template.js +2 -2
  108. package/dist/commands/wake.d.ts +1 -1
  109. package/dist/commands/wake.js +11 -10
  110. package/dist/index.d.ts +334 -191
  111. package/dist/index.js +432 -108
  112. package/dist/{inject-Bzi5E-By.d.ts → inject-DYUrDqQO.d.ts} +3 -3
  113. package/dist/ledger-B7g7jhqG.d.ts +44 -0
  114. package/dist/lib/auto-linker.js +1 -1
  115. package/dist/lib/canvas-layout.d.ts +115 -0
  116. package/dist/lib/canvas-layout.js +35 -0
  117. package/dist/lib/config.d.ts +27 -3
  118. package/dist/lib/config.js +4 -2
  119. package/dist/lib/entity-index.js +1 -1
  120. package/dist/lib/project-utils.js +4 -4
  121. package/dist/lib/session-repair.js +1 -1
  122. package/dist/lib/session-utils.js +1 -1
  123. package/dist/lib/tailscale.d.ts +225 -0
  124. package/dist/lib/tailscale.js +50 -0
  125. package/dist/lib/task-utils.js +3 -3
  126. package/dist/lib/template-engine.js +1 -1
  127. package/dist/lib/webdav.d.ts +109 -0
  128. package/dist/lib/webdav.js +35 -0
  129. package/dist/plugin/index.d.ts +344 -28
  130. package/dist/plugin/index.js +3919 -227
  131. package/dist/registry-BR4326o0.d.ts +30 -0
  132. package/dist/store-CA-6sKCJ.d.ts +34 -0
  133. package/dist/thread-B9LhXNU0.d.ts +41 -0
  134. package/dist/{types-Y2_Um2Ls.d.ts → types-BbWJoC1c.d.ts} +1 -44
  135. package/dist/workgraph/index.d.ts +5 -0
  136. package/dist/workgraph/index.js +23 -0
  137. package/dist/workgraph/ledger.d.ts +2 -0
  138. package/dist/workgraph/ledger.js +25 -0
  139. package/dist/workgraph/registry.d.ts +2 -0
  140. package/dist/workgraph/registry.js +19 -0
  141. package/dist/workgraph/store.d.ts +2 -0
  142. package/dist/workgraph/store.js +25 -0
  143. package/dist/workgraph/thread.d.ts +2 -0
  144. package/dist/workgraph/thread.js +25 -0
  145. package/dist/workgraph/types.d.ts +54 -0
  146. package/dist/workgraph/types.js +7 -0
  147. package/hooks/clawvault/HOOK.md +113 -0
  148. package/hooks/clawvault/handler.js +1559 -0
  149. package/hooks/clawvault/handler.test.js +510 -0
  150. package/hooks/clawvault/openclaw.plugin.json +72 -0
  151. package/openclaw.plugin.json +235 -30
  152. package/package.json +20 -20
  153. package/dist/chunk-3RG5ZIWI.js +0 -10
  154. package/dist/chunk-3ZIH425O.js +0 -871
  155. package/dist/chunk-6U6MK36V.js +0 -205
  156. package/dist/chunk-CMB7UL7C.js +0 -327
  157. package/dist/chunk-D2H45LON.js +0 -1074
  158. package/dist/chunk-E7MFQB6D.js +0 -163
  159. package/dist/chunk-GQSLDZTS.js +0 -560
  160. package/dist/chunk-MFM6K7PU.js +0 -374
  161. package/dist/chunk-MXSSG3QU.js +0 -42
  162. package/dist/chunk-OCGVIN3L.js +0 -88
  163. package/dist/chunk-PAH27GSN.js +0 -108
  164. package/dist/chunk-YCUNCH2I.js +0 -78
  165. package/dist/cli/index.cjs +0 -8584
  166. package/dist/cli/index.d.cts +0 -5
  167. package/dist/commands/archive.cjs +0 -287
  168. package/dist/commands/archive.d.cts +0 -11
  169. package/dist/commands/backlog.cjs +0 -721
  170. package/dist/commands/backlog.d.cts +0 -53
  171. package/dist/commands/blocked.cjs +0 -204
  172. package/dist/commands/blocked.d.cts +0 -26
  173. package/dist/commands/checkpoint.cjs +0 -244
  174. package/dist/commands/checkpoint.d.cts +0 -41
  175. package/dist/commands/compat.cjs +0 -294
  176. package/dist/commands/compat.d.cts +0 -28
  177. package/dist/commands/context.cjs +0 -2990
  178. package/dist/commands/context.d.cts +0 -2
  179. package/dist/commands/doctor.cjs +0 -2986
  180. package/dist/commands/doctor.d.cts +0 -21
  181. package/dist/commands/embed.cjs +0 -232
  182. package/dist/commands/embed.d.cts +0 -17
  183. package/dist/commands/entities.cjs +0 -141
  184. package/dist/commands/entities.d.cts +0 -7
  185. package/dist/commands/graph.cjs +0 -501
  186. package/dist/commands/graph.d.cts +0 -21
  187. package/dist/commands/inject.cjs +0 -1636
  188. package/dist/commands/inject.d.cts +0 -2
  189. package/dist/commands/kanban.cjs +0 -884
  190. package/dist/commands/kanban.d.cts +0 -63
  191. package/dist/commands/link.cjs +0 -965
  192. package/dist/commands/link.d.cts +0 -11
  193. package/dist/commands/migrate-observations.cjs +0 -362
  194. package/dist/commands/migrate-observations.d.cts +0 -19
  195. package/dist/commands/observe.cjs +0 -4099
  196. package/dist/commands/observe.d.cts +0 -23
  197. package/dist/commands/project.cjs +0 -1341
  198. package/dist/commands/project.d.cts +0 -85
  199. package/dist/commands/rebuild.cjs +0 -3136
  200. package/dist/commands/rebuild.d.cts +0 -11
  201. package/dist/commands/recover.cjs +0 -361
  202. package/dist/commands/recover.d.cts +0 -38
  203. package/dist/commands/reflect.cjs +0 -1008
  204. package/dist/commands/reflect.d.cts +0 -11
  205. package/dist/commands/repair-session.cjs +0 -457
  206. package/dist/commands/repair-session.d.cts +0 -38
  207. package/dist/commands/replay.cjs +0 -4103
  208. package/dist/commands/replay.d.cts +0 -16
  209. package/dist/commands/session-recap.cjs +0 -353
  210. package/dist/commands/session-recap.d.cts +0 -27
  211. package/dist/commands/setup.cjs +0 -1278
  212. package/dist/commands/setup.d.cts +0 -99
  213. package/dist/commands/shell-init.cjs +0 -75
  214. package/dist/commands/shell-init.d.cts +0 -7
  215. package/dist/commands/sleep.cjs +0 -6029
  216. package/dist/commands/sleep.d.cts +0 -36
  217. package/dist/commands/status.cjs +0 -2737
  218. package/dist/commands/status.d.cts +0 -52
  219. package/dist/commands/task.cjs +0 -1236
  220. package/dist/commands/task.d.cts +0 -97
  221. package/dist/commands/template.cjs +0 -457
  222. package/dist/commands/template.d.cts +0 -36
  223. package/dist/commands/wake.cjs +0 -2627
  224. package/dist/commands/wake.d.cts +0 -22
  225. package/dist/context-BUGaWpyL.d.cts +0 -46
  226. package/dist/index.cjs +0 -12373
  227. package/dist/index.d.cts +0 -854
  228. package/dist/inject-Bzi5E-By.d.cts +0 -137
  229. package/dist/lib/auto-linker.cjs +0 -176
  230. package/dist/lib/auto-linker.d.cts +0 -26
  231. package/dist/lib/config.cjs +0 -78
  232. package/dist/lib/config.d.cts +0 -11
  233. package/dist/lib/entity-index.cjs +0 -84
  234. package/dist/lib/entity-index.d.cts +0 -26
  235. package/dist/lib/project-utils.cjs +0 -864
  236. package/dist/lib/project-utils.d.cts +0 -97
  237. package/dist/lib/session-repair.cjs +0 -239
  238. package/dist/lib/session-repair.d.cts +0 -110
  239. package/dist/lib/session-utils.cjs +0 -209
  240. package/dist/lib/session-utils.d.cts +0 -63
  241. package/dist/lib/task-utils.cjs +0 -1137
  242. package/dist/lib/task-utils.d.cts +0 -208
  243. package/dist/lib/template-engine.cjs +0 -47
  244. package/dist/lib/template-engine.d.cts +0 -11
  245. package/dist/plugin/index.cjs +0 -1907
  246. package/dist/plugin/index.d.cts +0 -36
  247. package/dist/plugin/inject.cjs +0 -356
  248. package/dist/plugin/inject.d.cts +0 -54
  249. package/dist/plugin/inject.d.ts +0 -54
  250. package/dist/plugin/inject.js +0 -17
  251. package/dist/plugin/observe.cjs +0 -631
  252. package/dist/plugin/observe.d.cts +0 -39
  253. package/dist/plugin/observe.d.ts +0 -39
  254. package/dist/plugin/observe.js +0 -18
  255. package/dist/plugin/templates.cjs +0 -593
  256. package/dist/plugin/templates.d.cts +0 -52
  257. package/dist/plugin/templates.d.ts +0 -52
  258. package/dist/plugin/templates.js +0 -25
  259. package/dist/plugin/types.cjs +0 -18
  260. package/dist/plugin/types.d.cts +0 -209
  261. package/dist/plugin/types.d.ts +0 -209
  262. package/dist/plugin/types.js +0 -0
  263. package/dist/plugin/vault.cjs +0 -927
  264. package/dist/plugin/vault.d.cts +0 -68
  265. package/dist/plugin/vault.d.ts +0 -68
  266. package/dist/plugin/vault.js +0 -22
  267. package/dist/types-Y2_Um2Ls.d.cts +0 -205
  268. package/templates/memory-event.md +0 -67
  269. package/templates/party.md +0 -63
  270. package/templates/primitive-registry.yaml +0 -551
  271. package/templates/run.md +0 -68
  272. package/templates/trigger.md +0 -68
  273. package/templates/workspace.md +0 -50
@@ -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
+ };
@@ -0,0 +1,50 @@
1
+ import {
2
+ loadVaultQmdConfig
3
+ } from "./chunk-WIOLLGAD.js";
4
+ import {
5
+ QmdUnavailableError,
6
+ hasQmd,
7
+ qmdEmbed
8
+ } from "./chunk-5PJ4STIC.js";
9
+ import {
10
+ resolveVaultPath
11
+ } from "./chunk-GNJL4YGR.js";
12
+
13
+ // src/commands/embed.ts
14
+ async function embedCommand(options = {}) {
15
+ if (!hasQmd()) {
16
+ throw new QmdUnavailableError();
17
+ }
18
+ const vaultPath = resolveVaultPath({ explicitPath: options.vaultPath });
19
+ const qmdConfig = loadVaultQmdConfig(vaultPath);
20
+ const startedAt = (/* @__PURE__ */ new Date()).toISOString();
21
+ if (!options.quiet) {
22
+ console.log(
23
+ `Embedding pending documents for collection "${qmdConfig.qmdCollection}" (root: ${qmdConfig.qmdRoot})...`
24
+ );
25
+ }
26
+ qmdEmbed(qmdConfig.qmdCollection);
27
+ const finishedAt = (/* @__PURE__ */ new Date()).toISOString();
28
+ if (!options.quiet) {
29
+ console.log(`\u2713 Embedding complete for "${qmdConfig.qmdCollection}"`);
30
+ }
31
+ return {
32
+ vaultPath,
33
+ qmdCollection: qmdConfig.qmdCollection,
34
+ qmdRoot: qmdConfig.qmdRoot,
35
+ startedAt,
36
+ finishedAt
37
+ };
38
+ }
39
+ function registerEmbedCommand(program) {
40
+ program.command("embed").description("Run qmd embedding for pending vault documents").option("-v, --vault <path>", "Vault path").action(async (rawOptions) => {
41
+ await embedCommand({
42
+ vaultPath: rawOptions.vault
43
+ });
44
+ });
45
+ }
46
+
47
+ export {
48
+ embedCommand,
49
+ registerEmbedCommand
50
+ };
@@ -0,0 +1,189 @@
1
+ import {
2
+ create,
3
+ read,
4
+ update
5
+ } from "./chunk-HR4KN6S2.js";
6
+ import {
7
+ append,
8
+ currentOwner
9
+ } from "./chunk-K7PNYS45.js";
10
+ import {
11
+ THREAD_STATUS_TRANSITIONS
12
+ } from "./chunk-WMGIIABP.js";
13
+ import {
14
+ __export
15
+ } from "./chunk-U67V476Y.js";
16
+
17
+ // src/workgraph/thread.ts
18
+ var thread_exports = {};
19
+ __export(thread_exports, {
20
+ block: () => block,
21
+ cancel: () => cancel,
22
+ claim: () => claim,
23
+ createThread: () => createThread,
24
+ decompose: () => decompose,
25
+ done: () => done,
26
+ release: () => release,
27
+ unblock: () => unblock
28
+ });
29
+ function createThread(vaultPath, title, goal, actor, opts = {}) {
30
+ return create(vaultPath, "thread", {
31
+ title,
32
+ goal,
33
+ status: "open",
34
+ priority: opts.priority ?? "medium",
35
+ deps: opts.deps ?? [],
36
+ parent: opts.parent,
37
+ context_refs: opts.context_refs ?? [],
38
+ tags: opts.tags ?? []
39
+ }, `## Goal
40
+
41
+ ${goal}
42
+ `, actor);
43
+ }
44
+ function claim(vaultPath, threadPath, actor) {
45
+ const thread = read(vaultPath, threadPath);
46
+ if (!thread) throw new Error(`Thread not found: ${threadPath}`);
47
+ const status = thread.fields.status;
48
+ if (status !== "open") {
49
+ throw new Error(`Cannot claim thread in "${status}" state. Only "open" threads can be claimed.`);
50
+ }
51
+ const owner = currentOwner(vaultPath, threadPath);
52
+ if (owner) {
53
+ throw new Error(`Thread already claimed by "${owner}". Wait for release or use a different thread.`);
54
+ }
55
+ append(vaultPath, actor, "claim", threadPath, "thread");
56
+ return update(vaultPath, threadPath, {
57
+ status: "active",
58
+ owner: actor
59
+ }, void 0, actor);
60
+ }
61
+ function release(vaultPath, threadPath, actor, reason) {
62
+ const thread = read(vaultPath, threadPath);
63
+ if (!thread) throw new Error(`Thread not found: ${threadPath}`);
64
+ assertOwner(vaultPath, threadPath, actor);
65
+ append(
66
+ vaultPath,
67
+ actor,
68
+ "release",
69
+ threadPath,
70
+ "thread",
71
+ reason ? { reason } : void 0
72
+ );
73
+ return update(vaultPath, threadPath, {
74
+ status: "open",
75
+ owner: null
76
+ }, void 0, actor);
77
+ }
78
+ function block(vaultPath, threadPath, actor, blockedBy, reason) {
79
+ const thread = read(vaultPath, threadPath);
80
+ if (!thread) throw new Error(`Thread not found: ${threadPath}`);
81
+ assertTransition(thread.fields.status, "blocked");
82
+ append(vaultPath, actor, "block", threadPath, "thread", {
83
+ blocked_by: blockedBy,
84
+ ...reason ? { reason } : {}
85
+ });
86
+ const currentDeps = thread.fields.deps ?? [];
87
+ const updatedDeps = currentDeps.includes(blockedBy) ? currentDeps : [...currentDeps, blockedBy];
88
+ return update(vaultPath, threadPath, {
89
+ status: "blocked",
90
+ deps: updatedDeps
91
+ }, void 0, actor);
92
+ }
93
+ function unblock(vaultPath, threadPath, actor) {
94
+ const thread = read(vaultPath, threadPath);
95
+ if (!thread) throw new Error(`Thread not found: ${threadPath}`);
96
+ assertTransition(thread.fields.status, "active");
97
+ append(vaultPath, actor, "unblock", threadPath, "thread");
98
+ return update(vaultPath, threadPath, {
99
+ status: "active"
100
+ }, void 0, actor);
101
+ }
102
+ function done(vaultPath, threadPath, actor, output) {
103
+ const thread = read(vaultPath, threadPath);
104
+ if (!thread) throw new Error(`Thread not found: ${threadPath}`);
105
+ assertTransition(thread.fields.status, "done");
106
+ assertOwner(vaultPath, threadPath, actor);
107
+ append(
108
+ vaultPath,
109
+ actor,
110
+ "done",
111
+ threadPath,
112
+ "thread",
113
+ output ? { output } : void 0
114
+ );
115
+ const newBody = output ? `${thread.body}
116
+
117
+ ## Output
118
+
119
+ ${output}
120
+ ` : thread.body;
121
+ return update(vaultPath, threadPath, {
122
+ status: "done"
123
+ }, newBody, actor);
124
+ }
125
+ function cancel(vaultPath, threadPath, actor, reason) {
126
+ const thread = read(vaultPath, threadPath);
127
+ if (!thread) throw new Error(`Thread not found: ${threadPath}`);
128
+ assertTransition(thread.fields.status, "cancelled");
129
+ append(
130
+ vaultPath,
131
+ actor,
132
+ "cancel",
133
+ threadPath,
134
+ "thread",
135
+ reason ? { reason } : void 0
136
+ );
137
+ return update(vaultPath, threadPath, {
138
+ status: "cancelled",
139
+ owner: null
140
+ }, void 0, actor);
141
+ }
142
+ function decompose(vaultPath, parentPath, subthreads, actor) {
143
+ const parent = read(vaultPath, parentPath);
144
+ if (!parent) throw new Error(`Thread not found: ${parentPath}`);
145
+ const created = [];
146
+ for (const sub of subthreads) {
147
+ const inst = createThread(vaultPath, sub.title, sub.goal, actor, {
148
+ parent: parentPath,
149
+ deps: sub.deps
150
+ });
151
+ created.push(inst);
152
+ }
153
+ const childRefs = created.map((c) => `[[${c.path}]]`);
154
+ const decomposeNote = `
155
+
156
+ ## Sub-threads
157
+
158
+ ${childRefs.map((r) => `- ${r}`).join("\n")}
159
+ `;
160
+ update(vaultPath, parentPath, {}, parent.body + decomposeNote, actor);
161
+ append(vaultPath, actor, "decompose", parentPath, "thread", {
162
+ children: created.map((c) => c.path)
163
+ });
164
+ return created;
165
+ }
166
+ function assertTransition(from, to) {
167
+ const allowed = THREAD_STATUS_TRANSITIONS[from];
168
+ if (!allowed?.includes(to)) {
169
+ throw new Error(`Invalid transition: "${from}" \u2192 "${to}". Allowed: ${allowed?.join(", ") ?? "none"}`);
170
+ }
171
+ }
172
+ function assertOwner(vaultPath, threadPath, actor) {
173
+ const owner = currentOwner(vaultPath, threadPath);
174
+ if (owner && owner !== actor) {
175
+ throw new Error(`Thread is owned by "${owner}", not "${actor}". Only the owner can perform this action.`);
176
+ }
177
+ }
178
+
179
+ export {
180
+ createThread,
181
+ claim,
182
+ release,
183
+ block,
184
+ unblock,
185
+ done,
186
+ cancel,
187
+ decompose,
188
+ thread_exports
189
+ };