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,205 +0,0 @@
1
- import {
2
- classifyText
3
- } from "./chunk-GQSLDZTS.js";
4
-
5
- // src/plugin/observe.ts
6
- function isObservable(text) {
7
- if (!text || text.length < 20 || text.length > 5e3) return false;
8
- if (text.includes("<relevant-memories>")) return false;
9
- if (text.startsWith("[System")) return false;
10
- if (text.includes("HEARTBEAT")) return false;
11
- if (text.startsWith("NO_REPLY")) return false;
12
- if (text.startsWith("{") && text.includes('"')) return false;
13
- const markdownDensity = (text.match(/[#*`\-|>]/g) || []).length / text.length;
14
- if (markdownDensity > 0.15) return false;
15
- return true;
16
- }
17
- var OBSERVATION_PATTERNS = [
18
- // Preferences
19
- { pattern: /\b(i prefer|i like|i hate|i love|i want|i need|i always|i never|don't like|dont like)\b/i, weight: 2 },
20
- // Decisions
21
- { pattern: /\b(we decided|let's go with|we're going|i chose|we'll use|ship it|do it|go with)\b/i, weight: 2 },
22
- // Facts about people/things
23
- { pattern: /\b(my .+ is|his .+ is|her .+ is|their .+ is|works at|lives in|born in)\b/i, weight: 1.5 },
24
- // Contact info
25
- { pattern: /[\w.-]+@[\w.-]+\.\w+|\+\d{10,}/i, weight: 2 },
26
- // Explicit memory request
27
- { pattern: /\b(remember|don't forget|keep in mind|note that|important:)\b/i, weight: 2.5 },
28
- // Deadlines/dates
29
- { pattern: /\b(by tonight|by tomorrow|deadline|due date|by end of|ship by|ready by)\b/i, weight: 1.5 },
30
- // Lessons learned
31
- { pattern: /\b(i learned|we learned|lesson|realized|discovered|found out)\b/i, weight: 1.5 },
32
- // Tasks
33
- { pattern: /\b(need to|should|must|have to|todo|task)\b/i, weight: 1 },
34
- // Projects
35
- { pattern: /\b(working on|building|developing|project|initiative)\b/i, weight: 1 }
36
- ];
37
- function extractObservations(text) {
38
- const observations = [];
39
- const sentences = splitIntoSentences(text);
40
- const now = /* @__PURE__ */ new Date();
41
- for (const sentence of sentences) {
42
- if (sentence.length < 15) continue;
43
- let totalWeight = 0;
44
- for (const { pattern, weight } of OBSERVATION_PATTERNS) {
45
- if (pattern.test(sentence)) {
46
- totalWeight += weight;
47
- }
48
- }
49
- if (totalWeight < 1) continue;
50
- const classification = classifyText(sentence);
51
- const category = deriveCategoryFromPrimitive(classification.primitiveType, sentence);
52
- const tags = generateTags(classification, sentence);
53
- observations.push({
54
- text: sentence.trim(),
55
- primitiveType: classification.primitiveType,
56
- confidence: classification.confidence,
57
- matchedKeywords: classification.matchedKeywords,
58
- category,
59
- tags,
60
- extractedAt: now
61
- });
62
- }
63
- return observations;
64
- }
65
- function splitIntoSentences(text) {
66
- const raw = text.split(/(?<=[.!?\n])\s+/);
67
- const sentences = [];
68
- for (const s of raw) {
69
- const trimmed = s.trim();
70
- if (trimmed.length > 0) {
71
- sentences.push(trimmed);
72
- }
73
- }
74
- return sentences;
75
- }
76
- function deriveCategoryFromPrimitive(primitiveType, text) {
77
- const lower = text.toLowerCase();
78
- if (primitiveType === "memory_event") {
79
- if (/prefer|like|love|hate|want|need|always|never/i.test(lower)) {
80
- return "preference";
81
- }
82
- if (/remember|don't forget|keep in mind|note that/i.test(lower)) {
83
- return "note";
84
- }
85
- return "fact";
86
- }
87
- const categoryMap = {
88
- person: "entity",
89
- decision: "decision",
90
- task: "task",
91
- project: "project",
92
- lesson: "lesson",
93
- trigger: "automation",
94
- run: "execution",
95
- checkpoint: "checkpoint",
96
- handoff: "handoff",
97
- "daily-note": "daily",
98
- daily: "daily",
99
- party: "entity",
100
- workspace: "workspace"
101
- };
102
- return categoryMap[primitiveType] ?? "fact";
103
- }
104
- function generateTags(classification, text) {
105
- const tags = [classification.primitiveType];
106
- const lower = text.toLowerCase();
107
- if (/prefer|like|love/i.test(lower)) tags.push("positive");
108
- if (/hate|dislike|don't like/i.test(lower)) tags.push("negative");
109
- if (/deadline|due|by tomorrow|by tonight/i.test(lower)) tags.push("time-sensitive");
110
- if (/important|critical|urgent/i.test(lower)) tags.push("high-priority");
111
- if (/email|phone|contact/i.test(lower)) tags.push("contact-info");
112
- if (/decided|chose|approved/i.test(lower)) tags.push("finalized");
113
- if (/proposed|considering|might/i.test(lower)) tags.push("tentative");
114
- return [...new Set(tags)];
115
- }
116
- function processMessageForObservations(content, options = {}) {
117
- if (!isObservable(content)) {
118
- return {
119
- observations: [],
120
- skipped: 1,
121
- reason: "Content not observable"
122
- };
123
- }
124
- const observations = extractObservations(content);
125
- const maxObservations = 5;
126
- const limited = observations.slice(0, maxObservations);
127
- const skipped = observations.length - limited.length;
128
- return {
129
- observations: limited,
130
- skipped,
131
- reason: skipped > 0 ? `Limited to ${maxObservations} observations` : void 0
132
- };
133
- }
134
- function detectCategory(text) {
135
- const classification = classifyText(text);
136
- return deriveCategoryFromPrimitive(classification.primitiveType, text);
137
- }
138
- function extractPreferences(text) {
139
- const preferences = [];
140
- const patterns = [
141
- { regex: /i (?:really )?(?:like|love|enjoy|prefer)\s+(.+?)(?:\.|,|$)/gi, sentiment: "positive" },
142
- { regex: /i (?:don't|do not|hate|dislike)\s+(.+?)(?:\.|,|$)/gi, sentiment: "negative" },
143
- { regex: /my favorite\s+(.+?)\s+is\s+(.+?)(?:\.|,|$)/gi, sentiment: "positive", hasCategory: true },
144
- { regex: /i prefer\s+(.+?)\s+over\s+(.+?)(?:\.|,|$)/gi, sentiment: "positive" }
145
- ];
146
- for (const { regex, sentiment, hasCategory } of patterns) {
147
- let match;
148
- regex.lastIndex = 0;
149
- while ((match = regex.exec(text)) !== null) {
150
- if (hasCategory && match[2]) {
151
- preferences.push({
152
- category: normalizeCategory(match[1]),
153
- item: match[2].trim(),
154
- sentiment,
155
- confidence: 0.8,
156
- source: text.slice(0, 100)
157
- });
158
- } else if (match[1]) {
159
- const item = match[1].trim();
160
- preferences.push({
161
- category: inferCategory(item),
162
- item,
163
- sentiment,
164
- confidence: 0.7,
165
- source: text.slice(0, 100)
166
- });
167
- }
168
- }
169
- }
170
- return preferences;
171
- }
172
- function normalizeCategory(category) {
173
- return category.toLowerCase().trim().replace(/\s+/g, "_");
174
- }
175
- function inferCategory(item) {
176
- const itemLower = item.toLowerCase();
177
- const categoryKeywords = {
178
- food: ["pizza", "coffee", "tea", "food", "restaurant", "cuisine", "dish", "meal", "drink"],
179
- technology: ["programming", "code", "software", "app", "technology", "framework", "language", "tool"],
180
- entertainment: ["movie", "music", "book", "game", "show", "series", "film", "song", "album"],
181
- work: ["meeting", "project", "task", "deadline", "team", "client"],
182
- communication: ["email", "slack", "chat", "call", "message"]
183
- };
184
- for (const [category, keywords] of Object.entries(categoryKeywords)) {
185
- if (keywords.some((k) => itemLower.includes(k))) {
186
- return category;
187
- }
188
- }
189
- return "general";
190
- }
191
- function extractSearchTerms(input) {
192
- const noise = /\b(hey|hi|hello|um|uh|like|just|so|well|you know|i mean|basically|actually|really|very|pretty|quite|how does it feel|how do you|can you|could you|would you|do you|what do you think|tell me about)\b/gi;
193
- let cleaned = input.replace(noise, " ").replace(/\s+/g, " ").trim();
194
- if (cleaned.length < 5) cleaned = input.trim();
195
- return cleaned;
196
- }
197
-
198
- export {
199
- isObservable,
200
- extractObservations,
201
- processMessageForObservations,
202
- detectCategory,
203
- extractPreferences,
204
- extractSearchTerms
205
- };
@@ -1,327 +0,0 @@
1
- // src/plugin/inject.ts
2
- import { existsSync, readdirSync, readFileSync, statSync } from "fs";
3
- import { join, relative } from "path";
4
- function parseYamlFrontmatter(content) {
5
- const match = content.match(/^---\n([\s\S]*?)\n---\n?([\s\S]*)$/);
6
- if (!match) return null;
7
- const yamlContent = match[1];
8
- const body = match[2];
9
- try {
10
- const frontmatter = parseSimpleYaml(yamlContent);
11
- return { frontmatter, body };
12
- } catch {
13
- return null;
14
- }
15
- }
16
- function parseSimpleYaml(yaml) {
17
- const result = {};
18
- const lines = yaml.split("\n");
19
- for (const line of lines) {
20
- if (!line.trim() || line.trim().startsWith("#")) continue;
21
- const colonIndex = line.indexOf(":");
22
- if (colonIndex === -1) continue;
23
- const key = line.slice(0, colonIndex).trim();
24
- const valueStr = line.slice(colonIndex + 1).trim();
25
- if (valueStr === "" || valueStr.startsWith("|") || valueStr.startsWith(">")) continue;
26
- result[key] = parseYamlValue(valueStr);
27
- }
28
- return result;
29
- }
30
- function parseYamlValue(value) {
31
- if (value === "" || value === "null" || value === "~") return null;
32
- if (value === "true") return true;
33
- if (value === "false") return false;
34
- if (/^-?\d+$/.test(value)) return parseInt(value, 10);
35
- if (/^-?\d+\.\d+$/.test(value)) return parseFloat(value);
36
- if (value.startsWith('"') && value.endsWith('"') || value.startsWith("'") && value.endsWith("'")) {
37
- return value.slice(1, -1);
38
- }
39
- return value;
40
- }
41
- function scanVaultFiles(vaultPath, options = {}) {
42
- const files = [];
43
- const maxAge = options.maxAge ?? 7 * 24 * 60 * 60 * 1e3;
44
- const limit = options.limit ?? 100;
45
- const now = Date.now();
46
- const cutoff = now - maxAge;
47
- const dirsToScan = findVaultDirectories(vaultPath);
48
- for (const dir of dirsToScan) {
49
- if (!existsSync(dir)) continue;
50
- try {
51
- scanDirectory(dir, vaultPath, files, cutoff, options.primitiveTypes);
52
- } catch {
53
- }
54
- }
55
- files.sort((a, b) => b.modifiedAt.getTime() - a.modifiedAt.getTime());
56
- return files.slice(0, limit);
57
- }
58
- function findVaultDirectories(vaultPath) {
59
- const dirs = [];
60
- dirs.push(vaultPath);
61
- const commonDirs = [
62
- "tasks",
63
- "projects",
64
- "decisions",
65
- "people",
66
- "persons",
67
- "notes",
68
- "daily",
69
- "journal",
70
- "ledger",
71
- "memory",
72
- "memories",
73
- "observations",
74
- "lessons",
75
- "triggers",
76
- "runs",
77
- "checkpoints",
78
- "handoffs",
79
- "workspaces",
80
- "parties"
81
- ];
82
- for (const subdir of commonDirs) {
83
- const fullPath = join(vaultPath, subdir);
84
- if (existsSync(fullPath)) {
85
- dirs.push(fullPath);
86
- }
87
- }
88
- try {
89
- const entries = readdirSync(vaultPath, { withFileTypes: true });
90
- for (const entry of entries) {
91
- if (entry.isDirectory() && !entry.name.startsWith(".") && !entry.name.startsWith("_")) {
92
- const fullPath = join(vaultPath, entry.name);
93
- if (!dirs.includes(fullPath)) {
94
- dirs.push(fullPath);
95
- }
96
- }
97
- }
98
- } catch {
99
- }
100
- return dirs;
101
- }
102
- function scanDirectory(dir, vaultPath, files, cutoff, primitiveTypes) {
103
- const entries = readdirSync(dir, { withFileTypes: true });
104
- for (const entry of entries) {
105
- if (entry.name.startsWith(".") || entry.name.startsWith("_")) continue;
106
- const fullPath = join(dir, entry.name);
107
- if (entry.isDirectory()) {
108
- const depth = fullPath.replace(vaultPath, "").split("/").length;
109
- if (depth <= 3) {
110
- scanDirectory(fullPath, vaultPath, files, cutoff, primitiveTypes);
111
- }
112
- } else if (entry.name.endsWith(".md")) {
113
- try {
114
- const stat = statSync(fullPath);
115
- if (stat.mtimeMs < cutoff) continue;
116
- const content = readFileSync(fullPath, "utf-8");
117
- const parsed = parseYamlFrontmatter(content);
118
- if (!parsed) continue;
119
- const primitiveType = detectPrimitiveType(parsed.frontmatter, fullPath);
120
- if (primitiveTypes && !primitiveTypes.includes(primitiveType)) continue;
121
- files.push({
122
- path: fullPath,
123
- relativePath: relative(vaultPath, fullPath),
124
- primitiveType,
125
- frontmatter: parsed.frontmatter,
126
- content: parsed.body,
127
- modifiedAt: stat.mtime,
128
- createdAt: stat.birthtime
129
- });
130
- } catch {
131
- }
132
- }
133
- }
134
- }
135
- function detectPrimitiveType(frontmatter, filePath) {
136
- if (frontmatter.primitive) return String(frontmatter.primitive);
137
- if (frontmatter.type) return String(frontmatter.type);
138
- const pathLower = filePath.toLowerCase();
139
- if (pathLower.includes("/tasks/")) return "task";
140
- if (pathLower.includes("/projects/")) return "project";
141
- if (pathLower.includes("/decisions/")) return "decision";
142
- if (pathLower.includes("/people/") || pathLower.includes("/persons/")) return "person";
143
- if (pathLower.includes("/daily/") || pathLower.includes("/journal/")) return "daily-note";
144
- if (pathLower.includes("/lessons/")) return "lesson";
145
- if (pathLower.includes("/triggers/")) return "trigger";
146
- if (pathLower.includes("/runs/")) return "run";
147
- if (pathLower.includes("/checkpoints/")) return "checkpoint";
148
- if (pathLower.includes("/handoffs/")) return "handoff";
149
- if (pathLower.includes("/ledger/")) return "memory_event";
150
- if (pathLower.includes("/memory/") || pathLower.includes("/memories/")) return "memory_event";
151
- return "unknown";
152
- }
153
- function buildSessionRecap(vaultPath, options = {}) {
154
- const maxAge = options.maxAge ?? 24 * 60 * 60 * 1e3;
155
- const limit = options.limit ?? 20;
156
- const includeContent = options.includeContent ?? false;
157
- const files = scanVaultFiles(vaultPath, { maxAge, limit });
158
- if (files.length === 0) {
159
- return {
160
- xml: "",
161
- fileCount: 0,
162
- primitiveGroups: {},
163
- timeRange: null
164
- };
165
- }
166
- const groups = {};
167
- for (const file of files) {
168
- const type = file.primitiveType;
169
- if (!groups[type]) groups[type] = [];
170
- groups[type].push(file);
171
- }
172
- const lines = ["<session-recap>"];
173
- lines.push(`<summary>Found ${files.length} recent items across ${Object.keys(groups).length} categories</summary>`);
174
- for (const [primitiveType, groupFiles] of Object.entries(groups)) {
175
- lines.push(`<${primitiveType}-items count="${groupFiles.length}">`);
176
- for (const file of groupFiles.slice(0, 5)) {
177
- const title = file.frontmatter.title || file.frontmatter.summary || file.relativePath;
178
- const status = file.frontmatter.status || "";
179
- const modified = file.modifiedAt.toISOString().slice(0, 16).replace("T", " ");
180
- lines.push(` <item path="${file.relativePath}" modified="${modified}"${status ? ` status="${status}"` : ""}>`);
181
- lines.push(` <title>${escapeXml(String(title))}</title>`);
182
- if (includeContent && file.content) {
183
- const snippet = file.content.slice(0, 200).replace(/\n/g, " ").trim();
184
- if (snippet) {
185
- lines.push(` <snippet>${escapeXml(snippet)}</snippet>`);
186
- }
187
- }
188
- lines.push(" </item>");
189
- }
190
- if (groupFiles.length > 5) {
191
- lines.push(` <more count="${groupFiles.length - 5}" />`);
192
- }
193
- lines.push(`</${primitiveType}-items>`);
194
- }
195
- lines.push("</session-recap>");
196
- const sortedByTime = [...files].sort((a, b) => a.modifiedAt.getTime() - b.modifiedAt.getTime());
197
- const timeRange = {
198
- oldest: sortedByTime[0].modifiedAt,
199
- newest: sortedByTime[sortedByTime.length - 1].modifiedAt
200
- };
201
- const primitiveGroups = {};
202
- for (const [type, groupFiles] of Object.entries(groups)) {
203
- primitiveGroups[type] = groupFiles.length;
204
- }
205
- return {
206
- xml: lines.join("\n"),
207
- fileCount: files.length,
208
- primitiveGroups,
209
- timeRange
210
- };
211
- }
212
- function buildPreferenceContext(vaultPath, options = {}) {
213
- const maxAge = options.maxAge ?? 30 * 24 * 60 * 60 * 1e3;
214
- const limit = options.limit ?? 50;
215
- const files = scanVaultFiles(vaultPath, { maxAge, limit: limit * 2 });
216
- const preferenceFiles = files.filter((file) => {
217
- if (file.frontmatter.type === "preference") return true;
218
- if (file.primitiveType === "memory_event" && file.frontmatter.type === "preference") return true;
219
- const content = (file.content || "").toLowerCase();
220
- if (/\b(prefer|like|love|hate|dislike|want|need|always|never)\b/.test(content)) {
221
- return true;
222
- }
223
- return false;
224
- }).slice(0, limit);
225
- if (preferenceFiles.length === 0) {
226
- return {
227
- xml: "",
228
- preferenceCount: 0,
229
- categories: []
230
- };
231
- }
232
- const categories = /* @__PURE__ */ new Set();
233
- for (const file of preferenceFiles) {
234
- if (file.frontmatter.category) {
235
- categories.add(String(file.frontmatter.category));
236
- }
237
- }
238
- const lines = ["<user-preferences>"];
239
- for (const file of preferenceFiles) {
240
- const summary = file.frontmatter.summary || file.frontmatter.title || extractPreferenceSummary(file.content);
241
- if (!summary) continue;
242
- const category = file.frontmatter.category || "general";
243
- const sentiment = file.frontmatter.sentiment || inferSentiment(file.content);
244
- lines.push(` <preference category="${escapeXml(String(category))}" sentiment="${sentiment}">`);
245
- lines.push(` ${escapeXml(String(summary))}`);
246
- lines.push(" </preference>");
247
- }
248
- lines.push("</user-preferences>");
249
- return {
250
- xml: lines.join("\n"),
251
- preferenceCount: preferenceFiles.length,
252
- categories: Array.from(categories)
253
- };
254
- }
255
- function extractPreferenceSummary(content) {
256
- if (!content) return "";
257
- const sentences = content.split(/[.!?\n]+/).map((s) => s.trim()).filter((s) => s.length > 10);
258
- for (const sentence of sentences) {
259
- if (/\b(prefer|like|love|hate|dislike|want|need|always|never)\b/i.test(sentence)) {
260
- return sentence.slice(0, 150);
261
- }
262
- }
263
- return sentences[0]?.slice(0, 150) || "";
264
- }
265
- function inferSentiment(content) {
266
- if (!content) return "neutral";
267
- const lower = content.toLowerCase();
268
- if (/\b(love|like|prefer|enjoy|want|need|always)\b/.test(lower)) return "positive";
269
- if (/\b(hate|dislike|don't like|never|avoid)\b/.test(lower)) return "negative";
270
- return "neutral";
271
- }
272
- function formatMemoriesForContext(results, collection) {
273
- if (results.length === 0) return "";
274
- const lines = results.map((r, i) => {
275
- const file = (r.file || "").replace(`qmd://${collection}/`, "");
276
- const snippet = (r.snippet || "").replace(/@@ .+? @@\s*\(.+?\)\n?/g, "").trim() || r.title || "";
277
- return `${i + 1}. [${file}] ${snippet}`;
278
- });
279
- return `<relevant-memories>
280
- These are recalled from long-term vault memory. Treat as historical context.
281
- ${lines.join("\n")}
282
- </relevant-memories>`;
283
- }
284
- function formatSearchResults(results, collection) {
285
- if (results.length === 0) return "No relevant memories found.";
286
- return results.map((r, i) => {
287
- const file = (r.file || "").replace(`qmd://${collection}/`, "");
288
- const snippet = (r.snippet || "").replace(/@@ .+? @@\s*\(.+?\)\n?/g, "").trim() || r.title || "(no content)";
289
- const score = ((r.score ?? 0) * 100).toFixed(0);
290
- return `${i + 1}. [${file}] ${snippet} (${score}%)`;
291
- }).join("\n");
292
- }
293
- function escapeXml(str) {
294
- return str.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;").replace(/"/g, "&quot;").replace(/'/g, "&apos;");
295
- }
296
- function buildFullContext(vaultPath, options = {}) {
297
- const parts = [];
298
- if (options.includeRecap !== false) {
299
- const recap = buildSessionRecap(vaultPath, {
300
- maxAge: options.recapMaxAge ?? 24 * 60 * 60 * 1e3,
301
- limit: 15,
302
- includeContent: true
303
- });
304
- if (recap.xml) {
305
- parts.push(recap.xml);
306
- }
307
- }
308
- if (options.includePreferences !== false) {
309
- const prefs = buildPreferenceContext(vaultPath, {
310
- maxAge: options.preferenceMaxAge ?? 30 * 24 * 60 * 60 * 1e3,
311
- limit: 20
312
- });
313
- if (prefs.xml) {
314
- parts.push(prefs.xml);
315
- }
316
- }
317
- return parts.join("\n\n");
318
- }
319
-
320
- export {
321
- scanVaultFiles,
322
- buildSessionRecap,
323
- buildPreferenceContext,
324
- formatMemoriesForContext,
325
- formatSearchResults,
326
- buildFullContext
327
- };