clawvault 2.6.0 → 3.0.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 (232) hide show
  1. package/bin/command-registration.test.js +1 -3
  2. package/bin/register-core-commands.js +10 -23
  3. package/bin/register-maintenance-commands.js +3 -20
  4. package/bin/register-query-commands.js +23 -0
  5. package/bin/register-task-commands.js +1 -18
  6. package/bin/register-task-commands.test.js +0 -16
  7. package/bin/register-vault-operations-commands.js +1 -29
  8. package/dist/{chunk-QVMXF7FY.js → chunk-3D6BCTP6.js} +39 -1
  9. package/dist/{chunk-R2MIW5G7.js → chunk-3DHXQHYG.js} +1 -1
  10. package/dist/{chunk-Q2J5YTUF.js → chunk-3NSBOUT3.js} +73 -36
  11. package/dist/chunk-3RG5ZIWI.js +10 -0
  12. package/dist/{chunk-AZYOKJYC.js → chunk-62YTUT6J.js} +2 -2
  13. package/dist/chunk-6U6MK36V.js +205 -0
  14. package/dist/{chunk-4QYGFWRM.js → chunk-7R7O6STJ.js} +4 -4
  15. package/dist/{chunk-VXEOHTSL.js → chunk-C7OK5WKP.js} +4 -4
  16. package/dist/chunk-CMB7UL7C.js +327 -0
  17. package/dist/chunk-DEFFDRVP.js +938 -0
  18. package/dist/{chunk-K3CDT7IH.js → chunk-E7MFQB6D.js} +61 -20
  19. package/dist/{chunk-ME37YNW3.js → chunk-F2JEUD4J.js} +6 -4
  20. package/dist/chunk-GAJV4IGR.js +82 -0
  21. package/dist/chunk-GQSLDZTS.js +560 -0
  22. package/dist/{chunk-4OXMU5S2.js → chunk-GUKMRGM7.js} +1 -1
  23. package/dist/{chunk-YOSEUUNB.js → chunk-H34S76MB.js} +6 -6
  24. package/dist/{chunk-4TE4JMLA.js → chunk-JY6FYXIT.js} +10 -5
  25. package/dist/chunk-K234IDRJ.js +1073 -0
  26. package/dist/{chunk-IEVLHNLU.js → chunk-LNJA2UGL.js} +86 -9
  27. package/dist/{chunk-MFAWT5O5.js → chunk-LYHGEHXG.js} +1 -0
  28. package/dist/chunk-MFM6K7PU.js +374 -0
  29. package/dist/{chunk-QWQ3TIKS.js → chunk-N2AXRYLC.js} +1 -1
  30. package/dist/chunk-PAH27GSN.js +108 -0
  31. package/dist/{chunk-OIWVQYQF.js → chunk-QBLMXKF2.js} +1 -1
  32. package/dist/{chunk-FHFUXL6G.js → chunk-QK3UCXWL.js} +2 -2
  33. package/dist/{chunk-2YDBJS7M.js → chunk-SJSFRIYS.js} +1 -1
  34. package/dist/{chunk-GSD4ALSI.js → chunk-U55BGUAU.js} +2 -2
  35. package/dist/{chunk-PBEE567J.js → chunk-VGLOTGAS.js} +1 -1
  36. package/dist/{chunk-F55HGNU4.js → chunk-WAZ3NLWL.js} +47 -0
  37. package/dist/{chunk-KL4NAOMO.js → chunk-WGRQ6HDV.js} +1 -1
  38. package/dist/{chunk-UEOUADMO.js → chunk-YKTA5JOJ.js} +13 -10
  39. package/dist/{chunk-XAVB4GB4.js → chunk-ZVVFWOLW.js} +4 -4
  40. package/dist/cli/index.cjs +10033 -0
  41. package/dist/cli/index.d.cts +5 -0
  42. package/dist/cli/index.js +20 -18
  43. package/dist/commands/archive.cjs +287 -0
  44. package/dist/commands/archive.d.cts +11 -0
  45. package/dist/commands/archive.js +1 -0
  46. package/dist/commands/backlog.cjs +721 -0
  47. package/dist/commands/backlog.d.cts +53 -0
  48. package/dist/commands/backlog.js +3 -2
  49. package/dist/commands/blocked.cjs +204 -0
  50. package/dist/commands/blocked.d.cts +26 -0
  51. package/dist/commands/blocked.js +3 -2
  52. package/dist/commands/checkpoint.cjs +244 -0
  53. package/dist/commands/checkpoint.d.cts +41 -0
  54. package/dist/commands/checkpoint.js +2 -1
  55. package/dist/commands/compat.cjs +369 -0
  56. package/dist/commands/compat.d.cts +28 -0
  57. package/dist/commands/compat.js +2 -1
  58. package/dist/commands/context.cjs +2989 -0
  59. package/dist/commands/context.d.cts +2 -0
  60. package/dist/commands/context.js +5 -4
  61. package/dist/commands/doctor.cjs +3062 -0
  62. package/dist/commands/doctor.d.cts +21 -0
  63. package/dist/commands/doctor.d.ts +6 -1
  64. package/dist/commands/doctor.js +13 -11
  65. package/dist/commands/embed.cjs +232 -0
  66. package/dist/commands/embed.d.cts +17 -0
  67. package/dist/commands/embed.js +5 -2
  68. package/dist/commands/entities.cjs +141 -0
  69. package/dist/commands/entities.d.cts +7 -0
  70. package/dist/commands/entities.js +1 -0
  71. package/dist/commands/graph.cjs +501 -0
  72. package/dist/commands/graph.d.cts +21 -0
  73. package/dist/commands/graph.js +1 -0
  74. package/dist/commands/inject.cjs +1636 -0
  75. package/dist/commands/inject.d.cts +2 -0
  76. package/dist/commands/inject.d.ts +1 -1
  77. package/dist/commands/inject.js +4 -2
  78. package/dist/commands/kanban.cjs +884 -0
  79. package/dist/commands/kanban.d.cts +63 -0
  80. package/dist/commands/kanban.js +4 -3
  81. package/dist/commands/link.cjs +965 -0
  82. package/dist/commands/link.d.cts +11 -0
  83. package/dist/commands/link.js +1 -0
  84. package/dist/commands/migrate-observations.cjs +362 -0
  85. package/dist/commands/migrate-observations.d.cts +19 -0
  86. package/dist/commands/migrate-observations.js +3 -2
  87. package/dist/commands/observe.cjs +4099 -0
  88. package/dist/commands/observe.d.cts +23 -0
  89. package/dist/commands/observe.d.ts +1 -0
  90. package/dist/commands/observe.js +11 -9
  91. package/dist/commands/project.cjs +1341 -0
  92. package/dist/commands/project.d.cts +85 -0
  93. package/dist/commands/project.js +5 -4
  94. package/dist/commands/rebuild.cjs +3136 -0
  95. package/dist/commands/rebuild.d.cts +11 -0
  96. package/dist/commands/rebuild.js +10 -8
  97. package/dist/commands/recover.cjs +361 -0
  98. package/dist/commands/recover.d.cts +38 -0
  99. package/dist/commands/recover.js +3 -2
  100. package/dist/commands/reflect.cjs +1008 -0
  101. package/dist/commands/reflect.d.cts +11 -0
  102. package/dist/commands/reflect.js +6 -4
  103. package/dist/commands/repair-session.cjs +457 -0
  104. package/dist/commands/repair-session.d.cts +38 -0
  105. package/dist/commands/repair-session.js +1 -0
  106. package/dist/commands/replay.cjs +4103 -0
  107. package/dist/commands/replay.d.cts +16 -0
  108. package/dist/commands/replay.js +12 -10
  109. package/dist/commands/session-recap.cjs +353 -0
  110. package/dist/commands/session-recap.d.cts +27 -0
  111. package/dist/commands/session-recap.js +1 -0
  112. package/dist/commands/setup.cjs +1345 -0
  113. package/dist/commands/setup.d.cts +100 -0
  114. package/dist/commands/setup.d.ts +90 -2
  115. package/dist/commands/setup.js +21 -2
  116. package/dist/commands/shell-init.cjs +75 -0
  117. package/dist/commands/shell-init.d.cts +7 -0
  118. package/dist/commands/shell-init.js +2 -0
  119. package/dist/commands/sleep.cjs +6028 -0
  120. package/dist/commands/sleep.d.cts +36 -0
  121. package/dist/commands/sleep.d.ts +1 -1
  122. package/dist/commands/sleep.js +17 -15
  123. package/dist/commands/status.cjs +2736 -0
  124. package/dist/commands/status.d.cts +52 -0
  125. package/dist/commands/status.js +12 -10
  126. package/dist/commands/tailscale.cjs +1532 -0
  127. package/dist/commands/tailscale.d.cts +52 -0
  128. package/dist/commands/tailscale.js +1 -0
  129. package/dist/commands/task.cjs +1236 -0
  130. package/dist/commands/task.d.cts +97 -0
  131. package/dist/commands/task.js +4 -3
  132. package/dist/commands/template.cjs +457 -0
  133. package/dist/commands/template.d.cts +36 -0
  134. package/dist/commands/template.js +2 -1
  135. package/dist/commands/wake.cjs +2626 -0
  136. package/dist/commands/wake.d.cts +22 -0
  137. package/dist/commands/wake.d.ts +1 -1
  138. package/dist/commands/wake.js +12 -11
  139. package/dist/context-BUGaWpyL.d.cts +46 -0
  140. package/dist/index.cjs +14526 -0
  141. package/dist/index.d.cts +858 -0
  142. package/dist/index.d.ts +192 -7
  143. package/dist/index.js +101 -75
  144. package/dist/{inject-x65KXWPk.d.ts → inject-Bzi5E-By.d.cts} +1 -1
  145. package/dist/inject-Bzi5E-By.d.ts +137 -0
  146. package/dist/lib/auto-linker.cjs +176 -0
  147. package/dist/lib/auto-linker.d.cts +26 -0
  148. package/dist/lib/auto-linker.js +1 -0
  149. package/dist/lib/canvas-layout.cjs +136 -0
  150. package/dist/lib/canvas-layout.d.cts +31 -0
  151. package/dist/lib/canvas-layout.d.ts +16 -100
  152. package/dist/lib/canvas-layout.js +78 -20
  153. package/dist/lib/config.cjs +78 -0
  154. package/dist/lib/config.d.cts +11 -0
  155. package/dist/lib/config.js +1 -0
  156. package/dist/lib/entity-index.cjs +84 -0
  157. package/dist/lib/entity-index.d.cts +26 -0
  158. package/dist/lib/entity-index.js +1 -0
  159. package/dist/lib/project-utils.cjs +864 -0
  160. package/dist/lib/project-utils.d.cts +97 -0
  161. package/dist/lib/project-utils.js +4 -3
  162. package/dist/lib/session-repair.cjs +239 -0
  163. package/dist/lib/session-repair.d.cts +110 -0
  164. package/dist/lib/session-repair.js +1 -0
  165. package/dist/lib/session-utils.cjs +209 -0
  166. package/dist/lib/session-utils.d.cts +63 -0
  167. package/dist/lib/session-utils.js +1 -0
  168. package/dist/lib/tailscale.cjs +1183 -0
  169. package/dist/lib/tailscale.d.cts +225 -0
  170. package/dist/lib/tailscale.js +1 -0
  171. package/dist/lib/task-utils.cjs +1137 -0
  172. package/dist/lib/task-utils.d.cts +208 -0
  173. package/dist/lib/task-utils.js +3 -2
  174. package/dist/lib/template-engine.cjs +47 -0
  175. package/dist/lib/template-engine.d.cts +11 -0
  176. package/dist/lib/template-engine.js +1 -0
  177. package/dist/lib/webdav.cjs +568 -0
  178. package/dist/lib/webdav.d.cts +109 -0
  179. package/dist/lib/webdav.js +1 -0
  180. package/dist/plugin/index.cjs +1907 -0
  181. package/dist/plugin/index.d.cts +36 -0
  182. package/dist/plugin/index.d.ts +36 -0
  183. package/dist/plugin/index.js +572 -0
  184. package/dist/plugin/inject.cjs +356 -0
  185. package/dist/plugin/inject.d.cts +54 -0
  186. package/dist/plugin/inject.d.ts +54 -0
  187. package/dist/plugin/inject.js +17 -0
  188. package/dist/plugin/observe.cjs +631 -0
  189. package/dist/plugin/observe.d.cts +39 -0
  190. package/dist/plugin/observe.d.ts +39 -0
  191. package/dist/plugin/observe.js +18 -0
  192. package/dist/plugin/templates.cjs +593 -0
  193. package/dist/plugin/templates.d.cts +52 -0
  194. package/dist/plugin/templates.d.ts +52 -0
  195. package/dist/plugin/templates.js +25 -0
  196. package/dist/plugin/types.cjs +18 -0
  197. package/dist/plugin/types.d.cts +209 -0
  198. package/dist/plugin/types.d.ts +209 -0
  199. package/dist/plugin/types.js +0 -0
  200. package/dist/plugin/vault.cjs +927 -0
  201. package/dist/plugin/vault.d.cts +68 -0
  202. package/dist/plugin/vault.d.ts +68 -0
  203. package/dist/plugin/vault.js +22 -0
  204. package/dist/{types-C74wgGL1.d.ts → types-Y2_Um2Ls.d.cts} +44 -1
  205. package/dist/types-Y2_Um2Ls.d.ts +205 -0
  206. package/hooks/clawvault/handler.js +70 -7
  207. package/hooks/clawvault/handler.test.js +91 -0
  208. package/openclaw.plugin.json +56 -0
  209. package/package.json +17 -7
  210. package/templates/memory-event.md +67 -0
  211. package/templates/party.md +63 -0
  212. package/templates/primitive-registry.yaml +551 -0
  213. package/templates/run.md +68 -0
  214. package/templates/trigger.md +68 -0
  215. package/templates/workspace.md +50 -0
  216. package/dashboard/lib/graph-diff.js +0 -104
  217. package/dashboard/lib/graph-diff.test.js +0 -75
  218. package/dashboard/lib/vault-parser.js +0 -556
  219. package/dashboard/lib/vault-parser.test.js +0 -254
  220. package/dashboard/public/app.js +0 -796
  221. package/dashboard/public/index.html +0 -52
  222. package/dashboard/public/styles.css +0 -221
  223. package/dashboard/server.js +0 -374
  224. package/dist/chunk-HA5M6KJB.js +0 -33
  225. package/dist/chunk-MAKNAHAW.js +0 -375
  226. package/dist/chunk-MDIH26GC.js +0 -183
  227. package/dist/chunk-MGDEINGP.js +0 -99
  228. package/dist/chunk-RVYA52PY.js +0 -363
  229. package/dist/commands/canvas.d.ts +0 -15
  230. package/dist/commands/canvas.js +0 -199
  231. package/dist/commands/sync-bd.d.ts +0 -10
  232. package/dist/commands/sync-bd.js +0 -9
@@ -0,0 +1,501 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __export = (target, all) => {
9
+ for (var name in all)
10
+ __defProp(target, name, { get: all[name], enumerable: true });
11
+ };
12
+ var __copyProps = (to, from, except, desc) => {
13
+ if (from && typeof from === "object" || typeof from === "function") {
14
+ for (let key of __getOwnPropNames(from))
15
+ if (!__hasOwnProp.call(to, key) && key !== except)
16
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
+ }
18
+ return to;
19
+ };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
28
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
+
30
+ // src/commands/graph.ts
31
+ var graph_exports = {};
32
+ __export(graph_exports, {
33
+ graphCommand: () => graphCommand,
34
+ graphSummary: () => graphSummary
35
+ });
36
+ module.exports = __toCommonJS(graph_exports);
37
+
38
+ // src/lib/memory-graph.ts
39
+ var fs = __toESM(require("fs"), 1);
40
+ var path = __toESM(require("path"), 1);
41
+ var import_gray_matter = __toESM(require("gray-matter"), 1);
42
+ var import_glob = require("glob");
43
+ var MEMORY_GRAPH_SCHEMA_VERSION = 1;
44
+ var GRAPH_INDEX_RELATIVE_PATH = path.join(".clawvault", "graph-index.json");
45
+ var WIKI_LINK_RE = /\[\[([^\]]+)\]\]/g;
46
+ var HASH_TAG_RE = /(^|\s)#([\w-]+)/g;
47
+ var FRONTMATTER_RELATION_FIELDS = [
48
+ "related",
49
+ "depends_on",
50
+ "dependsOn",
51
+ "blocked_by",
52
+ "blocks",
53
+ "owner",
54
+ "project",
55
+ "people",
56
+ "links"
57
+ ];
58
+ function normalizeRelativePath(value) {
59
+ return value.split(path.sep).join("/").replace(/^\.\//, "").replace(/^\/+/, "");
60
+ }
61
+ function toNoteKey(relativePath) {
62
+ const normalized = normalizeRelativePath(relativePath);
63
+ return normalized.toLowerCase().endsWith(".md") ? normalized.slice(0, -3) : normalized;
64
+ }
65
+ function toNoteNodeId(noteKey) {
66
+ return `note:${noteKey}`;
67
+ }
68
+ function toTagNodeId(tag) {
69
+ return `tag:${tag.toLowerCase()}`;
70
+ }
71
+ function normalizeUnresolvedKey(raw) {
72
+ const normalized = raw.trim().toLowerCase().replace(/\\/g, "/").replace(/^\.\//, "").replace(/^\/+/, "").replace(/\.md$/, "").replace(/[^a-z0-9/_-]+/g, "-").replace(/\/+/g, "/").replace(/-+/g, "-").replace(/^[-/]+|[-/]+$/g, "");
73
+ return normalized || "unknown";
74
+ }
75
+ function toUnresolvedNodeId(raw) {
76
+ return `unresolved:${normalizeUnresolvedKey(raw)}`;
77
+ }
78
+ function titleFromNoteKey(noteKey) {
79
+ const basename = noteKey.split("/").pop() ?? noteKey;
80
+ return basename.replace(/[-_]+/g, " ").replace(/\s+/g, " ").trim().replace(/\b\w/g, (char) => char.toUpperCase());
81
+ }
82
+ function inferNodeType(relativePath, frontmatter) {
83
+ const normalized = normalizeRelativePath(relativePath).toLowerCase();
84
+ const category = normalized.split("/")[0] ?? "note";
85
+ const explicitType = typeof frontmatter.type === "string" ? frontmatter.type.toLowerCase() : "";
86
+ if (category.includes("daily") || explicitType === "daily") return "daily";
87
+ if (category === "observations" || explicitType === "observation") return "observation";
88
+ if (category === "handoffs" || explicitType === "handoff") return "handoff";
89
+ if (category === "decisions" || explicitType === "decision") return "decision";
90
+ if (category === "lessons" || explicitType === "lesson") return "lesson";
91
+ if (category === "projects" || explicitType === "project") return "project";
92
+ if (category === "people" || explicitType === "person") return "person";
93
+ if (category === "commitments" || explicitType === "commitment") return "commitment";
94
+ return "note";
95
+ }
96
+ function ensureClawvaultDir(vaultPath) {
97
+ const dirPath = path.join(vaultPath, ".clawvault");
98
+ if (!fs.existsSync(dirPath)) {
99
+ fs.mkdirSync(dirPath, { recursive: true });
100
+ }
101
+ return dirPath;
102
+ }
103
+ function getGraphIndexPath(vaultPath) {
104
+ return path.join(vaultPath, GRAPH_INDEX_RELATIVE_PATH);
105
+ }
106
+ function normalizeWikiTarget(target) {
107
+ let value = target.trim();
108
+ if (!value) return "";
109
+ const pipeIndex = value.indexOf("|");
110
+ if (pipeIndex >= 0) {
111
+ value = value.slice(0, pipeIndex);
112
+ }
113
+ const hashIndex = value.indexOf("#");
114
+ if (hashIndex >= 0) {
115
+ value = value.slice(0, hashIndex);
116
+ }
117
+ value = value.trim().replace(/\\/g, "/").replace(/^\.\//, "").replace(/^\/+/, "");
118
+ if (value.toLowerCase().endsWith(".md")) {
119
+ value = value.slice(0, -3);
120
+ }
121
+ return value.trim();
122
+ }
123
+ function collectTags(frontmatter, markdownContent) {
124
+ const tags = /* @__PURE__ */ new Set();
125
+ const fmTags = frontmatter.tags;
126
+ if (Array.isArray(fmTags)) {
127
+ for (const tag of fmTags) {
128
+ if (typeof tag === "string" && tag.trim()) tags.add(tag.trim().toLowerCase());
129
+ }
130
+ } else if (typeof fmTags === "string") {
131
+ for (const token of fmTags.split(",")) {
132
+ const normalized = token.trim().toLowerCase();
133
+ if (normalized) tags.add(normalized);
134
+ }
135
+ }
136
+ const markdownMatches = markdownContent.matchAll(HASH_TAG_RE);
137
+ for (const match of markdownMatches) {
138
+ const tag = match[2]?.trim().toLowerCase();
139
+ if (tag) tags.add(tag);
140
+ }
141
+ return [...tags].sort((a, b) => a.localeCompare(b));
142
+ }
143
+ function extractWikiTargets(markdownContent) {
144
+ const targets = /* @__PURE__ */ new Set();
145
+ for (const match of markdownContent.matchAll(WIKI_LINK_RE)) {
146
+ const candidate = match[1];
147
+ if (!candidate) continue;
148
+ const normalized = normalizeWikiTarget(candidate);
149
+ if (normalized) targets.add(normalized);
150
+ }
151
+ return [...targets];
152
+ }
153
+ function toStringArray(value) {
154
+ if (typeof value === "string") {
155
+ return value.split(",").map((entry) => entry.trim()).filter(Boolean);
156
+ }
157
+ if (Array.isArray(value)) {
158
+ return value.flatMap((entry) => typeof entry === "string" ? entry.split(",") : []).map((entry) => entry.trim()).filter(Boolean);
159
+ }
160
+ return [];
161
+ }
162
+ function extractFrontmatterRelations(frontmatter) {
163
+ const relations = [];
164
+ for (const field of FRONTMATTER_RELATION_FIELDS) {
165
+ const raw = frontmatter[field];
166
+ for (const value of toStringArray(raw)) {
167
+ const normalized = normalizeWikiTarget(value);
168
+ if (normalized) relations.push({ field, target: normalized });
169
+ }
170
+ }
171
+ return relations;
172
+ }
173
+ function buildNoteRegistry(relativePaths) {
174
+ const byLowerPath = /* @__PURE__ */ new Map();
175
+ const byLowerBasename = /* @__PURE__ */ new Map();
176
+ for (const relativePath of relativePaths) {
177
+ const noteKey = toNoteKey(relativePath);
178
+ const lowerKey = noteKey.toLowerCase();
179
+ if (!byLowerPath.has(lowerKey)) {
180
+ byLowerPath.set(lowerKey, noteKey);
181
+ }
182
+ const base = noteKey.split("/").pop() ?? noteKey;
183
+ const lowerBase = base.toLowerCase();
184
+ const existing = byLowerBasename.get(lowerBase) ?? [];
185
+ existing.push(noteKey);
186
+ byLowerBasename.set(lowerBase, existing);
187
+ }
188
+ return { byLowerPath, byLowerBasename };
189
+ }
190
+ function resolveTargetNodeId(rawTarget, registry) {
191
+ const normalized = normalizeWikiTarget(rawTarget);
192
+ if (!normalized) {
193
+ return toUnresolvedNodeId(rawTarget);
194
+ }
195
+ const lowerTarget = normalized.toLowerCase();
196
+ const direct = registry.byLowerPath.get(lowerTarget);
197
+ if (direct) {
198
+ return toNoteNodeId(direct);
199
+ }
200
+ if (!normalized.includes("/")) {
201
+ const basenameMatches = registry.byLowerBasename.get(lowerTarget) ?? [];
202
+ if (basenameMatches.length === 1) {
203
+ return toNoteNodeId(basenameMatches[0]);
204
+ }
205
+ }
206
+ return toUnresolvedNodeId(normalized);
207
+ }
208
+ function createEdgeId(type, source, target, label) {
209
+ const suffix = label ? `:${label}` : "";
210
+ return `${type}:${source}->${target}${suffix}`;
211
+ }
212
+ function buildFragmentNode(id, title, type, category, pathValue, tags, missing, modifiedAt) {
213
+ return {
214
+ id,
215
+ title,
216
+ type,
217
+ category,
218
+ path: pathValue,
219
+ tags,
220
+ missing,
221
+ degree: 0,
222
+ modifiedAt
223
+ };
224
+ }
225
+ function parseFileFragment(vaultPath, relativePath, mtimeMs, registry) {
226
+ const absolutePath = path.join(vaultPath, relativePath);
227
+ const raw = fs.readFileSync(absolutePath, "utf-8");
228
+ const parsed = (0, import_gray_matter.default)(raw);
229
+ const frontmatter = parsed.data ?? {};
230
+ const noteKey = toNoteKey(relativePath);
231
+ const noteNodeId = toNoteNodeId(noteKey);
232
+ const noteType = inferNodeType(relativePath, frontmatter);
233
+ const tags = collectTags(frontmatter, parsed.content);
234
+ const modifiedAt = new Date(mtimeMs).toISOString();
235
+ const nodes = /* @__PURE__ */ new Map();
236
+ const edges = /* @__PURE__ */ new Map();
237
+ nodes.set(
238
+ noteNodeId,
239
+ buildFragmentNode(
240
+ noteNodeId,
241
+ typeof frontmatter.title === "string" && frontmatter.title.trim() ? frontmatter.title.trim() : titleFromNoteKey(noteKey),
242
+ noteType,
243
+ noteType,
244
+ normalizeRelativePath(relativePath),
245
+ tags,
246
+ false,
247
+ modifiedAt
248
+ )
249
+ );
250
+ for (const tag of tags) {
251
+ const tagNodeId = toTagNodeId(tag);
252
+ if (!nodes.has(tagNodeId)) {
253
+ nodes.set(tagNodeId, buildFragmentNode(tagNodeId, `#${tag}`, "tag", "tag", null, [], false, null));
254
+ }
255
+ const edgeId = createEdgeId("tag", noteNodeId, tagNodeId);
256
+ edges.set(edgeId, {
257
+ id: edgeId,
258
+ source: noteNodeId,
259
+ target: tagNodeId,
260
+ type: "tag"
261
+ });
262
+ }
263
+ const wikiTargets = extractWikiTargets(parsed.content);
264
+ for (const target of wikiTargets) {
265
+ const targetNodeId = resolveTargetNodeId(target, registry);
266
+ if (targetNodeId.startsWith("unresolved:") && !nodes.has(targetNodeId)) {
267
+ nodes.set(
268
+ targetNodeId,
269
+ buildFragmentNode(targetNodeId, titleFromNoteKey(normalizeUnresolvedKey(target)), "unresolved", "unresolved", null, [], true, null)
270
+ );
271
+ }
272
+ const edgeId = createEdgeId("wiki_link", noteNodeId, targetNodeId);
273
+ edges.set(edgeId, {
274
+ id: edgeId,
275
+ source: noteNodeId,
276
+ target: targetNodeId,
277
+ type: "wiki_link"
278
+ });
279
+ }
280
+ for (const relation of extractFrontmatterRelations(frontmatter)) {
281
+ const targetNodeId = resolveTargetNodeId(relation.target, registry);
282
+ if (targetNodeId.startsWith("unresolved:") && !nodes.has(targetNodeId)) {
283
+ nodes.set(
284
+ targetNodeId,
285
+ buildFragmentNode(
286
+ targetNodeId,
287
+ titleFromNoteKey(normalizeUnresolvedKey(relation.target)),
288
+ "unresolved",
289
+ "unresolved",
290
+ null,
291
+ [],
292
+ true,
293
+ null
294
+ )
295
+ );
296
+ }
297
+ const edgeId = createEdgeId("frontmatter_relation", noteNodeId, targetNodeId, relation.field);
298
+ edges.set(edgeId, {
299
+ id: edgeId,
300
+ source: noteNodeId,
301
+ target: targetNodeId,
302
+ type: "frontmatter_relation",
303
+ label: relation.field
304
+ });
305
+ }
306
+ return {
307
+ relativePath: normalizeRelativePath(relativePath),
308
+ mtimeMs,
309
+ nodes: [...nodes.values()],
310
+ edges: [...edges.values()]
311
+ };
312
+ }
313
+ function combineFragments(fragments, generatedAt) {
314
+ const nodes = /* @__PURE__ */ new Map();
315
+ const edges = /* @__PURE__ */ new Map();
316
+ for (const fragment of Object.values(fragments)) {
317
+ for (const node of fragment.nodes) {
318
+ const existing = nodes.get(node.id);
319
+ if (!existing) {
320
+ nodes.set(node.id, { ...node, degree: 0 });
321
+ } else if (node.modifiedAt && (!existing.modifiedAt || node.modifiedAt > existing.modifiedAt)) {
322
+ nodes.set(node.id, { ...existing, ...node, degree: 0 });
323
+ }
324
+ }
325
+ for (const edge of fragment.edges) {
326
+ edges.set(edge.id, edge);
327
+ }
328
+ }
329
+ const degreeByNode = /* @__PURE__ */ new Map();
330
+ for (const edge of edges.values()) {
331
+ degreeByNode.set(edge.source, (degreeByNode.get(edge.source) ?? 0) + 1);
332
+ degreeByNode.set(edge.target, (degreeByNode.get(edge.target) ?? 0) + 1);
333
+ }
334
+ for (const node of nodes.values()) {
335
+ node.degree = degreeByNode.get(node.id) ?? 0;
336
+ }
337
+ const nodeTypeCounts = {};
338
+ for (const node of nodes.values()) {
339
+ nodeTypeCounts[node.type] = (nodeTypeCounts[node.type] ?? 0) + 1;
340
+ }
341
+ const edgeTypeCounts = {};
342
+ for (const edge of edges.values()) {
343
+ edgeTypeCounts[edge.type] = (edgeTypeCounts[edge.type] ?? 0) + 1;
344
+ }
345
+ const sortedNodes = [...nodes.values()].sort((a, b) => a.id.localeCompare(b.id));
346
+ const sortedEdges = [...edges.values()].sort((a, b) => a.id.localeCompare(b.id));
347
+ return {
348
+ schemaVersion: MEMORY_GRAPH_SCHEMA_VERSION,
349
+ nodes: sortedNodes,
350
+ edges: sortedEdges,
351
+ stats: {
352
+ generatedAt,
353
+ nodeCount: sortedNodes.length,
354
+ edgeCount: sortedEdges.length,
355
+ nodeTypeCounts,
356
+ edgeTypeCounts
357
+ }
358
+ };
359
+ }
360
+ function isValidIndex(index) {
361
+ if (!index || typeof index !== "object") return false;
362
+ const typed = index;
363
+ return typed.schemaVersion === MEMORY_GRAPH_SCHEMA_VERSION && typeof typed.vaultPath === "string" && typeof typed.generatedAt === "string" && Boolean(typed.files && typeof typed.files === "object") && Boolean(typed.graph && typeof typed.graph === "object");
364
+ }
365
+ function loadMemoryGraphIndex(vaultPath) {
366
+ const indexPath = getGraphIndexPath(path.resolve(vaultPath));
367
+ if (!fs.existsSync(indexPath)) {
368
+ return null;
369
+ }
370
+ try {
371
+ const parsed = JSON.parse(fs.readFileSync(indexPath, "utf-8"));
372
+ if (!isValidIndex(parsed)) {
373
+ return null;
374
+ }
375
+ return parsed;
376
+ } catch {
377
+ return null;
378
+ }
379
+ }
380
+ async function buildOrUpdateMemoryGraphIndex(vaultPathInput, options = {}) {
381
+ const vaultPath = path.resolve(vaultPathInput);
382
+ ensureClawvaultDir(vaultPath);
383
+ const existing = options.forceFull ? null : loadMemoryGraphIndex(vaultPath);
384
+ const markdownFiles = await (0, import_glob.glob)("**/*.md", {
385
+ cwd: vaultPath,
386
+ ignore: ["**/node_modules/**", "**/.git/**", "**/.obsidian/**", "**/.trash/**", "**/ledger/archive/**"]
387
+ });
388
+ const normalizedFiles = markdownFiles.map(normalizeRelativePath).sort((a, b) => a.localeCompare(b));
389
+ const registry = buildNoteRegistry(normalizedFiles);
390
+ const nextFragments = {};
391
+ const existingFragments = existing?.files ?? {};
392
+ const currentFileSet = new Set(normalizedFiles);
393
+ for (const relativePath of normalizedFiles) {
394
+ const absolutePath = path.join(vaultPath, relativePath);
395
+ const stat = fs.statSync(absolutePath);
396
+ const existingFragment = existingFragments[relativePath];
397
+ if (!options.forceFull && existingFragment && existingFragment.mtimeMs === stat.mtimeMs) {
398
+ nextFragments[relativePath] = existingFragment;
399
+ continue;
400
+ }
401
+ nextFragments[relativePath] = parseFileFragment(vaultPath, relativePath, stat.mtimeMs, registry);
402
+ }
403
+ for (const [relativePath, fragment] of Object.entries(existingFragments)) {
404
+ if (!currentFileSet.has(relativePath)) {
405
+ continue;
406
+ }
407
+ if (!nextFragments[relativePath]) {
408
+ nextFragments[relativePath] = fragment;
409
+ }
410
+ }
411
+ const generatedAt = (/* @__PURE__ */ new Date()).toISOString();
412
+ const graph = combineFragments(nextFragments, generatedAt);
413
+ const nextIndex = {
414
+ schemaVersion: MEMORY_GRAPH_SCHEMA_VERSION,
415
+ vaultPath,
416
+ generatedAt,
417
+ files: nextFragments,
418
+ graph
419
+ };
420
+ fs.writeFileSync(getGraphIndexPath(vaultPath), JSON.stringify(nextIndex, null, 2));
421
+ return nextIndex;
422
+ }
423
+
424
+ // src/lib/config.ts
425
+ var fs2 = __toESM(require("fs"), 1);
426
+ var path2 = __toESM(require("path"), 1);
427
+ function findNearestVaultPath(startPath = process.cwd()) {
428
+ let current = path2.resolve(startPath);
429
+ while (true) {
430
+ if (fs2.existsSync(path2.join(current, ".clawvault.json"))) {
431
+ return current;
432
+ }
433
+ const parent = path2.dirname(current);
434
+ if (parent === current) {
435
+ return null;
436
+ }
437
+ current = parent;
438
+ }
439
+ }
440
+ function resolveVaultPath(options = {}) {
441
+ if (options.explicitPath) {
442
+ return path2.resolve(options.explicitPath);
443
+ }
444
+ if (process.env.CLAWVAULT_PATH) {
445
+ return path2.resolve(process.env.CLAWVAULT_PATH);
446
+ }
447
+ const discovered = findNearestVaultPath(options.cwd ?? process.cwd());
448
+ if (discovered) {
449
+ return discovered;
450
+ }
451
+ throw new Error("No vault path found. Set CLAWVAULT_PATH, use --vault, or run inside a vault.");
452
+ }
453
+
454
+ // src/commands/graph.ts
455
+ function formatGraphSummary(summary) {
456
+ const lines = [];
457
+ lines.push("Memory Graph Summary");
458
+ lines.push("-".repeat(34));
459
+ lines.push(`Schema version: ${summary.schemaVersion}`);
460
+ lines.push(`Generated at: ${summary.generatedAt}`);
461
+ lines.push(`Files indexed: ${summary.fileCount}`);
462
+ lines.push(`Nodes: ${summary.nodeCount}`);
463
+ lines.push(`Edges: ${summary.edgeCount}`);
464
+ lines.push("");
465
+ lines.push("Node types:");
466
+ for (const [type, count] of Object.entries(summary.nodeTypeCounts).sort((a, b) => a[0].localeCompare(b[0]))) {
467
+ lines.push(` - ${type}: ${count}`);
468
+ }
469
+ lines.push("");
470
+ lines.push("Edge types:");
471
+ for (const [type, count] of Object.entries(summary.edgeTypeCounts).sort((a, b) => a[0].localeCompare(b[0]))) {
472
+ lines.push(` - ${type}: ${count}`);
473
+ }
474
+ return lines.join("\n");
475
+ }
476
+ async function graphSummary(options = {}) {
477
+ const vaultPath = resolveVaultPath({ explicitPath: options.vaultPath });
478
+ const index = options.refresh ? await buildOrUpdateMemoryGraphIndex(vaultPath) : loadMemoryGraphIndex(vaultPath) ?? await buildOrUpdateMemoryGraphIndex(vaultPath);
479
+ return {
480
+ schemaVersion: index.schemaVersion,
481
+ generatedAt: index.generatedAt,
482
+ nodeCount: index.graph.stats.nodeCount,
483
+ edgeCount: index.graph.stats.edgeCount,
484
+ nodeTypeCounts: index.graph.stats.nodeTypeCounts,
485
+ edgeTypeCounts: index.graph.stats.edgeTypeCounts,
486
+ fileCount: Object.keys(index.files).length
487
+ };
488
+ }
489
+ async function graphCommand(options = {}) {
490
+ const summary = await graphSummary(options);
491
+ if (options.json) {
492
+ console.log(JSON.stringify(summary, null, 2));
493
+ return;
494
+ }
495
+ console.log(formatGraphSummary(summary));
496
+ }
497
+ // Annotate the CommonJS export names for ESM import in node:
498
+ 0 && (module.exports = {
499
+ graphCommand,
500
+ graphSummary
501
+ });
@@ -0,0 +1,21 @@
1
+ interface GraphSummary {
2
+ schemaVersion: number;
3
+ generatedAt: string;
4
+ nodeCount: number;
5
+ edgeCount: number;
6
+ nodeTypeCounts: Record<string, number>;
7
+ edgeTypeCounts: Record<string, number>;
8
+ fileCount: number;
9
+ }
10
+ declare function graphSummary(options?: {
11
+ vaultPath?: string;
12
+ refresh?: boolean;
13
+ json?: boolean;
14
+ }): Promise<GraphSummary>;
15
+ declare function graphCommand(options?: {
16
+ vaultPath?: string;
17
+ refresh?: boolean;
18
+ json?: boolean;
19
+ }): Promise<void>;
20
+
21
+ export { type GraphSummary, graphCommand, graphSummary };
@@ -4,6 +4,7 @@ import {
4
4
  } from "../chunk-OZ7RIXTO.js";
5
5
  import "../chunk-MXSSG3QU.js";
6
6
  import "../chunk-ZZA73MFY.js";
7
+ import "../chunk-3RG5ZIWI.js";
7
8
  export {
8
9
  graphCommand,
9
10
  graphSummary