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,373 @@
1
+ import {
2
+ DEFAULT_SERVE_PORT,
3
+ configureTailscaleServe,
4
+ discoverClawVaultPeers,
5
+ findPeer,
6
+ getOnlinePeers,
7
+ getTailscaleStatus,
8
+ getTailscaleVersion,
9
+ serveVault,
10
+ stopTailscaleServe,
11
+ syncWithPeer
12
+ } from "./chunk-TIGW564L.js";
13
+ import {
14
+ resolveVaultPath
15
+ } from "./chunk-GNJL4YGR.js";
16
+
17
+ // src/commands/tailscale.ts
18
+ import * as path from "path";
19
+ async function tailscaleStatusCommand(options = {}) {
20
+ const status = getTailscaleStatus();
21
+ if (options.json) {
22
+ console.log(JSON.stringify(status, null, 2));
23
+ return status;
24
+ }
25
+ if (!status.installed) {
26
+ console.log("Tailscale: Not installed");
27
+ console.log(" Install from: https://tailscale.com/download");
28
+ return status;
29
+ }
30
+ const version = getTailscaleVersion();
31
+ console.log(`Tailscale: ${version || "installed"}`);
32
+ if (!status.running) {
33
+ console.log(" Status: Daemon not running");
34
+ if (status.error) {
35
+ console.log(` Error: ${status.error}`);
36
+ }
37
+ return status;
38
+ }
39
+ console.log(` Status: ${status.backendState}`);
40
+ if (status.connected) {
41
+ console.log(` Tailnet: ${status.tailnetName || "unknown"}`);
42
+ console.log(` Self IP: ${status.selfIP || "unknown"}`);
43
+ console.log(` Hostname: ${status.selfHostname || "unknown"}`);
44
+ if (status.selfDNSName) {
45
+ console.log(` DNS Name: ${status.selfDNSName}`);
46
+ }
47
+ if (options.peers || status.peers.length > 0) {
48
+ const onlinePeers = status.peers.filter((p) => p.online);
49
+ const offlinePeers = status.peers.filter((p) => !p.online);
50
+ console.log(`
51
+ Peers (${onlinePeers.length} online, ${offlinePeers.length} offline):`);
52
+ for (const peer of onlinePeers) {
53
+ const ip = peer.tailscaleIPs[0] || "no-ip";
54
+ const os = peer.os ? ` (${peer.os})` : "";
55
+ const clawvault = peer.clawvaultServing ? " [ClawVault]" : "";
56
+ console.log(` \u25CF ${peer.hostname}${os} - ${ip}${clawvault}`);
57
+ }
58
+ if (options.peers) {
59
+ for (const peer of offlinePeers) {
60
+ const ip = peer.tailscaleIPs[0] || "no-ip";
61
+ const os = peer.os ? ` (${peer.os})` : "";
62
+ console.log(` \u25CB ${peer.hostname}${os} - ${ip} [offline]`);
63
+ }
64
+ }
65
+ }
66
+ } else {
67
+ console.log(" Status: Not connected to tailnet");
68
+ if (status.error) {
69
+ console.log(` Error: ${status.error}`);
70
+ }
71
+ }
72
+ return status;
73
+ }
74
+ function registerTailscaleStatusCommand(program) {
75
+ program.command("tailscale-status").alias("ts-status").description("Show Tailscale connection status and peers").option("--json", "Output as JSON").option("--peers", "Show all peers including offline").action(async (rawOptions) => {
76
+ await tailscaleStatusCommand({
77
+ json: rawOptions.json,
78
+ peers: rawOptions.peers
79
+ });
80
+ });
81
+ }
82
+ async function tailscaleSyncCommand(options) {
83
+ const vaultPath = resolveVaultPath({ explicitPath: options.vaultPath });
84
+ const status = getTailscaleStatus();
85
+ if (!status.installed) {
86
+ const error = {
87
+ pushed: [],
88
+ pulled: [],
89
+ deleted: [],
90
+ unchanged: [],
91
+ errors: ["Tailscale not installed. Install from https://tailscale.com/download"],
92
+ stats: { bytesTransferred: 0, filesProcessed: 0, duration: 0 }
93
+ };
94
+ if (options.json) {
95
+ console.log(JSON.stringify(error, null, 2));
96
+ } else {
97
+ console.error("Error: Tailscale not installed");
98
+ }
99
+ return error;
100
+ }
101
+ if (!status.connected) {
102
+ const error = {
103
+ pushed: [],
104
+ pulled: [],
105
+ deleted: [],
106
+ unchanged: [],
107
+ errors: ["Not connected to Tailscale. Run `tailscale up` to connect."],
108
+ stats: { bytesTransferred: 0, filesProcessed: 0, duration: 0 }
109
+ };
110
+ if (options.json) {
111
+ console.log(JSON.stringify(error, null, 2));
112
+ } else {
113
+ console.error("Error: Not connected to Tailscale");
114
+ }
115
+ return error;
116
+ }
117
+ const peer = findPeer(options.peer);
118
+ if (!peer) {
119
+ const error = {
120
+ pushed: [],
121
+ pulled: [],
122
+ deleted: [],
123
+ unchanged: [],
124
+ errors: [`Peer not found: ${options.peer}`],
125
+ stats: { bytesTransferred: 0, filesProcessed: 0, duration: 0 }
126
+ };
127
+ if (options.json) {
128
+ console.log(JSON.stringify(error, null, 2));
129
+ } else {
130
+ console.error(`Error: Peer not found: ${options.peer}`);
131
+ console.log("\nAvailable online peers:");
132
+ for (const p of getOnlinePeers()) {
133
+ console.log(` - ${p.hostname} (${p.tailscaleIPs[0]})`);
134
+ }
135
+ }
136
+ return error;
137
+ }
138
+ if (!peer.online) {
139
+ const error = {
140
+ pushed: [],
141
+ pulled: [],
142
+ deleted: [],
143
+ unchanged: [],
144
+ errors: [`Peer is offline: ${peer.hostname}`],
145
+ stats: { bytesTransferred: 0, filesProcessed: 0, duration: 0 }
146
+ };
147
+ if (options.json) {
148
+ console.log(JSON.stringify(error, null, 2));
149
+ } else {
150
+ console.error(`Error: Peer is offline: ${peer.hostname}`);
151
+ }
152
+ return error;
153
+ }
154
+ const syncOptions = {
155
+ peer: peer.tailscaleIPs[0],
156
+ port: options.port || DEFAULT_SERVE_PORT,
157
+ direction: options.direction || "bidirectional",
158
+ dryRun: options.dryRun,
159
+ deleteOrphans: options.deleteOrphans,
160
+ categories: options.categories,
161
+ https: options.https
162
+ };
163
+ if (!options.json && !options.dryRun) {
164
+ console.log(`Syncing with ${peer.hostname} (${peer.tailscaleIPs[0]})...`);
165
+ }
166
+ const result = await syncWithPeer(vaultPath, syncOptions);
167
+ if (options.json) {
168
+ console.log(JSON.stringify(result, null, 2));
169
+ } else {
170
+ const prefix = options.dryRun ? "[dry-run] " : "";
171
+ if (result.pushed.length > 0) {
172
+ console.log(`
173
+ ${prefix}Pushed ${result.pushed.length} file(s):`);
174
+ for (const file of result.pushed.slice(0, 10)) {
175
+ console.log(` \u2192 ${file}`);
176
+ }
177
+ if (result.pushed.length > 10) {
178
+ console.log(` ... and ${result.pushed.length - 10} more`);
179
+ }
180
+ }
181
+ if (result.pulled.length > 0) {
182
+ console.log(`
183
+ ${prefix}Pulled ${result.pulled.length} file(s):`);
184
+ for (const file of result.pulled.slice(0, 10)) {
185
+ console.log(` \u2190 ${file}`);
186
+ }
187
+ if (result.pulled.length > 10) {
188
+ console.log(` ... and ${result.pulled.length - 10} more`);
189
+ }
190
+ }
191
+ if (result.deleted.length > 0) {
192
+ console.log(`
193
+ ${prefix}Deleted ${result.deleted.length} file(s):`);
194
+ for (const file of result.deleted.slice(0, 10)) {
195
+ console.log(` \u2717 ${file}`);
196
+ }
197
+ if (result.deleted.length > 10) {
198
+ console.log(` ... and ${result.deleted.length - 10} more`);
199
+ }
200
+ }
201
+ if (result.errors.length > 0) {
202
+ console.log(`
203
+ Errors (${result.errors.length}):`);
204
+ for (const error of result.errors) {
205
+ console.log(` ! ${error}`);
206
+ }
207
+ }
208
+ console.log(`
209
+ Summary:`);
210
+ console.log(` Pushed: ${result.pushed.length}`);
211
+ console.log(` Pulled: ${result.pulled.length}`);
212
+ console.log(` Deleted: ${result.deleted.length}`);
213
+ console.log(` Unchanged: ${result.unchanged.length}`);
214
+ console.log(` Errors: ${result.errors.length}`);
215
+ console.log(` Duration: ${result.stats.duration}ms`);
216
+ console.log(` Transferred: ${formatBytes(result.stats.bytesTransferred)}`);
217
+ }
218
+ return result;
219
+ }
220
+ function formatBytes(bytes) {
221
+ if (bytes === 0) return "0 B";
222
+ const k = 1024;
223
+ const sizes = ["B", "KB", "MB", "GB"];
224
+ const i = Math.floor(Math.log(bytes) / Math.log(k));
225
+ return `${parseFloat((bytes / Math.pow(k, i)).toFixed(2))} ${sizes[i]}`;
226
+ }
227
+ function registerTailscaleSyncCommand(program) {
228
+ program.command("tailscale-sync").alias("ts-sync").description("Sync vault with a peer on the Tailscale network").requiredOption("--peer <hostname>", "Peer hostname or IP to sync with").option("-v, --vault <path>", "Vault path").option("--port <number>", "Port on the peer", parseInt).option("--direction <dir>", "Sync direction: push, pull, or bidirectional", "bidirectional").option("--dry-run", "Show what would be synced without making changes").option("--delete-orphans", "Delete files that exist locally but not on peer (pull only)").option("--categories <list>", "Comma-separated list of categories to sync").option("--https", "Use HTTPS for connection").option("--json", "Output as JSON").action(async (rawOptions) => {
229
+ await tailscaleSyncCommand({
230
+ peer: rawOptions.peer,
231
+ vaultPath: rawOptions.vault,
232
+ port: rawOptions.port,
233
+ direction: rawOptions.direction,
234
+ dryRun: rawOptions.dryRun,
235
+ deleteOrphans: rawOptions.deleteOrphans,
236
+ categories: rawOptions.categories?.split(",").map((c) => c.trim()),
237
+ https: rawOptions.https,
238
+ json: rawOptions.json
239
+ });
240
+ });
241
+ }
242
+ var activeServeInstance = null;
243
+ async function tailscaleServeCommand(options) {
244
+ if (options.stop) {
245
+ if (activeServeInstance) {
246
+ await activeServeInstance.stop();
247
+ activeServeInstance = null;
248
+ console.log("ClawVault serve stopped.");
249
+ }
250
+ stopTailscaleServe();
251
+ return;
252
+ }
253
+ const vaultPath = resolveVaultPath({ explicitPath: options.vaultPath });
254
+ const port = options.port || DEFAULT_SERVE_PORT;
255
+ const status = getTailscaleStatus();
256
+ console.log(`Starting ClawVault serve...`);
257
+ console.log(` Vault: ${path.basename(vaultPath)}`);
258
+ console.log(` Port: ${port}`);
259
+ activeServeInstance = serveVault(vaultPath, { port });
260
+ console.log(` Local URL: http://localhost:${port}/.clawvault`);
261
+ if (status.connected) {
262
+ console.log(` Tailscale URL: http://${status.selfIP}:${port}/.clawvault`);
263
+ if (status.selfDNSName) {
264
+ const dnsHost = status.selfDNSName.replace(/\.$/, "");
265
+ console.log(` MagicDNS URL: http://${dnsHost}:${port}/.clawvault`);
266
+ }
267
+ if (options.funnel || options.background) {
268
+ console.log("\nConfiguring Tailscale serve...");
269
+ configureTailscaleServe(port, {
270
+ funnel: options.funnel,
271
+ background: options.background
272
+ });
273
+ if (options.funnel) {
274
+ console.log(" Funnel enabled - vault is accessible from the public internet");
275
+ }
276
+ }
277
+ } else {
278
+ console.log("\n Note: Not connected to Tailscale. Only local access available.");
279
+ }
280
+ console.log("\nEndpoints:");
281
+ console.log(` Health: /.clawvault/health`);
282
+ console.log(` Manifest: /.clawvault/manifest`);
283
+ console.log(` Files: /.clawvault/files/<path>`);
284
+ if (!options.background) {
285
+ console.log("\nPress Ctrl+C to stop serving.");
286
+ process.on("SIGINT", async () => {
287
+ console.log("\nStopping ClawVault serve...");
288
+ if (activeServeInstance) {
289
+ await activeServeInstance.stop();
290
+ activeServeInstance = null;
291
+ }
292
+ stopTailscaleServe();
293
+ process.exit(0);
294
+ });
295
+ await new Promise(() => {
296
+ });
297
+ }
298
+ }
299
+ function registerTailscaleServeCommand(program) {
300
+ program.command("tailscale-serve").alias("ts-serve").description("Serve vault for sync over Tailscale").option("-v, --vault <path>", "Vault path").option("--port <number>", `Port to serve on (default: ${DEFAULT_SERVE_PORT})`, parseInt).option("--funnel", "Expose via Tailscale Funnel (public internet)").option("--background", "Run in background").option("--stop", "Stop serving").action(async (rawOptions) => {
301
+ await tailscaleServeCommand({
302
+ vaultPath: rawOptions.vault,
303
+ port: rawOptions.port,
304
+ funnel: rawOptions.funnel,
305
+ background: rawOptions.background,
306
+ stop: rawOptions.stop
307
+ });
308
+ });
309
+ }
310
+ async function tailscaleDiscoverCommand(options = {}) {
311
+ const port = options.port || DEFAULT_SERVE_PORT;
312
+ const status = getTailscaleStatus();
313
+ if (!status.connected) {
314
+ if (options.json) {
315
+ console.log(JSON.stringify({ error: "Not connected to Tailscale", peers: [] }));
316
+ } else {
317
+ console.error("Error: Not connected to Tailscale");
318
+ }
319
+ return [];
320
+ }
321
+ if (!options.json) {
322
+ console.log("Discovering ClawVault peers on tailnet...");
323
+ }
324
+ const peers = await discoverClawVaultPeers(port);
325
+ if (options.json) {
326
+ console.log(JSON.stringify({ peers }, null, 2));
327
+ } else {
328
+ if (peers.length === 0) {
329
+ console.log("\nNo ClawVault peers found.");
330
+ console.log(" Run `clawvault tailscale-serve` on other devices to enable sync.");
331
+ } else {
332
+ console.log(`
333
+ Found ${peers.length} ClawVault peer(s):`);
334
+ for (const peer of peers) {
335
+ const ip = peer.tailscaleIPs[0] || "no-ip";
336
+ const os = peer.os ? ` (${peer.os})` : "";
337
+ console.log(` \u25CF ${peer.hostname}${os}`);
338
+ console.log(` IP: ${ip}`);
339
+ console.log(` Port: ${peer.clawvaultPort}`);
340
+ if (peer.dnsName) {
341
+ console.log(` DNS: ${peer.dnsName.replace(/\.$/, "")}`);
342
+ }
343
+ }
344
+ }
345
+ }
346
+ return peers;
347
+ }
348
+ function registerTailscaleDiscoverCommand(program) {
349
+ program.command("tailscale-discover").alias("ts-discover").description("Discover ClawVault peers on the Tailscale network").option("--port <number>", `Port to check (default: ${DEFAULT_SERVE_PORT})`, parseInt).option("--json", "Output as JSON").action(async (rawOptions) => {
350
+ await tailscaleDiscoverCommand({
351
+ port: rawOptions.port,
352
+ json: rawOptions.json
353
+ });
354
+ });
355
+ }
356
+ function registerTailscaleCommands(program) {
357
+ registerTailscaleStatusCommand(program);
358
+ registerTailscaleSyncCommand(program);
359
+ registerTailscaleServeCommand(program);
360
+ registerTailscaleDiscoverCommand(program);
361
+ }
362
+
363
+ export {
364
+ tailscaleStatusCommand,
365
+ registerTailscaleStatusCommand,
366
+ tailscaleSyncCommand,
367
+ registerTailscaleSyncCommand,
368
+ tailscaleServeCommand,
369
+ registerTailscaleServeCommand,
370
+ tailscaleDiscoverCommand,
371
+ registerTailscaleDiscoverCommand,
372
+ registerTailscaleCommands
373
+ };
@@ -1,3 +1,6 @@
1
+ import {
2
+ loadVaultQmdConfig
3
+ } from "./chunk-WIOLLGAD.js";
1
4
  import {
2
5
  QmdUnavailableError,
3
6
  SearchEngine,
@@ -6,7 +9,7 @@ import {
6
9
  hasQmd,
7
10
  qmdEmbed,
8
11
  qmdUpdate
9
- } from "./chunk-D2H45LON.js";
12
+ } from "./chunk-5PJ4STIC.js";
10
13
  import {
11
14
  DEFAULT_CATEGORIES,
12
15
  TYPE_TO_CATEGORY
@@ -29,7 +32,9 @@ var ClawVault = class {
29
32
  initialized = false;
30
33
  constructor(vaultPath) {
31
34
  if (!hasQmd()) {
32
- throw new QmdUnavailableError();
35
+ const error = new QmdUnavailableError("NOT_INSTALLED");
36
+ console.error(error.toUserMessage());
37
+ throw error;
33
38
  }
34
39
  this.config = {
35
40
  path: path.resolve(vaultPath),
@@ -46,7 +51,9 @@ var ClawVault = class {
46
51
  */
47
52
  async init(options = {}, initFlags) {
48
53
  if (!hasQmd()) {
49
- throw new QmdUnavailableError();
54
+ const error = new QmdUnavailableError("NOT_INSTALLED");
55
+ console.error(error.toUserMessage());
56
+ throw error;
50
57
  }
51
58
  const vaultPath = this.config.path;
52
59
  const flags = initFlags || {};
@@ -200,7 +207,9 @@ var ClawVault = class {
200
207
  */
201
208
  async load() {
202
209
  if (!hasQmd()) {
203
- throw new QmdUnavailableError();
210
+ const error = new QmdUnavailableError("NOT_INSTALLED");
211
+ console.error(error.toUserMessage());
212
+ throw error;
204
213
  }
205
214
  const vaultPath = this.config.path;
206
215
  const configPath = path.join(vaultPath, CONFIG_FILE);
@@ -624,14 +633,15 @@ var ClawVault = class {
624
633
  md += `## Recent Sessions
625
634
  `;
626
635
  for (const h of recap.recentHandoffs) {
636
+ const datePart = this.extractDatePart(h.created);
627
637
  if (brief) {
628
- md += `- **${h.created.split("T")[0]}:** ${h.workingOn.slice(0, 2).join(", ")}`;
638
+ md += `- **${datePart}:** ${h.workingOn.slice(0, 2).join(", ")}`;
629
639
  if (h.nextSteps.length > 0) md += ` \u2192 ${h.nextSteps[0]}`;
630
640
  md += `
631
641
  `;
632
642
  } else {
633
643
  md += `
634
- ### ${h.created.split("T")[0]}
644
+ ### ${datePart}
635
645
  `;
636
646
  md += `**Working on:** ${h.workingOn.join(", ")}
637
647
  `;
@@ -689,7 +699,7 @@ var ClawVault = class {
689
699
  */
690
700
  parseHandoff(doc) {
691
701
  return {
692
- created: doc.frontmatter.date || doc.modified.toISOString(),
702
+ created: this.toDateString(doc.frontmatter.date, doc.modified.toISOString()),
693
703
  sessionKey: doc.frontmatter.sessionKey,
694
704
  workingOn: doc.frontmatter.workingOn || [],
695
705
  blocked: doc.frontmatter.blocked || [],
@@ -700,9 +710,39 @@ var ClawVault = class {
700
710
  };
701
711
  }
702
712
  // === Private helpers ===
713
+ /**
714
+ * Safely convert a date value to ISO string format.
715
+ * Handles Date objects, strings, and undefined values.
716
+ */
717
+ toDateString(value, fallback) {
718
+ if (value instanceof Date) {
719
+ return value.toISOString();
720
+ }
721
+ if (typeof value === "string" && value.length > 0) {
722
+ return value;
723
+ }
724
+ return fallback || (/* @__PURE__ */ new Date()).toISOString();
725
+ }
726
+ /**
727
+ * Extract the date portion (YYYY-MM-DD) from an ISO date string or Date object.
728
+ * Provides safe handling for various date formats.
729
+ */
730
+ extractDatePart(value) {
731
+ const dateStr = this.toDateString(value);
732
+ if (dateStr.includes("T")) {
733
+ return dateStr.split("T")[0];
734
+ }
735
+ return dateStr.slice(0, 10);
736
+ }
703
737
  applyQmdConfig(meta) {
704
- const collection = meta?.qmdCollection || this.config.qmdCollection || this.config.name;
705
- const root = meta?.qmdRoot || this.config.qmdRoot || this.config.path;
738
+ const explicitCollection = meta?.qmdCollection || this.config.qmdCollection;
739
+ const explicitRoot = meta?.qmdRoot || this.config.qmdRoot || this.config.path;
740
+ const qmdConfig = loadVaultQmdConfig(this.config.path);
741
+ const collection = explicitCollection || qmdConfig.qmdCollection || this.config.name;
742
+ const root = explicitRoot || qmdConfig.qmdRoot;
743
+ if (qmdConfig.autoDetected) {
744
+ console.warn(`[clawvault] Auto-detected qmd collection: ${collection}`);
745
+ }
706
746
  this.config.qmdCollection = collection;
707
747
  this.config.qmdRoot = root;
708
748
  this.search.setVaultPath(this.config.path);
@@ -1,10 +1,10 @@
1
+ import {
2
+ listConfig
3
+ } from "./chunk-URXDAUVH.js";
1
4
  import {
2
5
  requestLlmCompletion,
3
6
  resolveLlmProvider
4
- } from "./chunk-E7MFQB6D.js";
5
- import {
6
- listConfig
7
- } from "./chunk-ITPEXLHA.js";
7
+ } from "./chunk-YXQCA6B7.js";
8
8
  import {
9
9
  getMemoryGraph,
10
10
  loadMemoryGraphIndex
@@ -563,7 +563,7 @@ async function runPromptInjection(vaultPathInput, message, options = {}) {
563
563
  const deterministicMs = Date.now() - startDeterministic;
564
564
  let llmProvider = null;
565
565
  if (useLlm) {
566
- llmProvider = await resolveLlmProvider();
566
+ llmProvider = resolveLlmProvider();
567
567
  if (llmProvider) {
568
568
  try {
569
569
  matches = await addLlmIntentMatches({