clawvault 3.0.0 → 3.1.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 (60) hide show
  1. package/README.md +156 -105
  2. package/bin/clawvault.js +0 -2
  3. package/bin/register-core-commands.js +20 -2
  4. package/dist/{chunk-3D6BCTP6.js → chunk-33UGEQRT.js} +70 -145
  5. package/dist/{chunk-ZVVFWOLW.js → chunk-3WRJEKN4.js} +1 -1
  6. package/dist/{chunk-DEFFDRVP.js → chunk-3ZIH425O.js} +3 -70
  7. package/dist/{chunk-K234IDRJ.js → chunk-D2H45LON.js} +1 -0
  8. package/dist/{chunk-YKTA5JOJ.js → chunk-H62BP7RI.js} +3 -3
  9. package/dist/{chunk-WGRQ6HDV.js → chunk-LI4O6NVK.js} +1 -1
  10. package/dist/{chunk-7R7O6STJ.js → chunk-OCGVIN3L.js} +1 -1
  11. package/dist/{chunk-GAJV4IGR.js → chunk-YCUNCH2I.js} +3 -7
  12. package/dist/cli/index.cjs +10 -1459
  13. package/dist/cli/index.js +5 -8
  14. package/dist/commands/compat.cjs +70 -145
  15. package/dist/commands/compat.js +1 -1
  16. package/dist/commands/context.cjs +1 -0
  17. package/dist/commands/context.js +3 -3
  18. package/dist/commands/doctor.cjs +68 -144
  19. package/dist/commands/doctor.js +4 -4
  20. package/dist/commands/embed.js +2 -2
  21. package/dist/commands/setup.cjs +2 -69
  22. package/dist/commands/setup.d.cts +0 -1
  23. package/dist/commands/setup.d.ts +0 -1
  24. package/dist/commands/setup.js +2 -2
  25. package/dist/commands/sleep.cjs +1 -0
  26. package/dist/commands/sleep.js +2 -2
  27. package/dist/commands/status.cjs +1 -0
  28. package/dist/commands/status.js +2 -2
  29. package/dist/commands/wake.cjs +1 -0
  30. package/dist/commands/wake.js +2 -2
  31. package/dist/index.cjs +447 -2600
  32. package/dist/index.d.cts +0 -4
  33. package/dist/index.d.ts +0 -4
  34. package/dist/index.js +8 -69
  35. package/dist/plugin/index.cjs +3 -3
  36. package/dist/plugin/index.js +10 -10
  37. package/package.json +11 -17
  38. package/bin/register-tailscale-commands.js +0 -106
  39. package/dist/chunk-IVRIKYFE.js +0 -520
  40. package/dist/chunk-THRJVD4L.js +0 -373
  41. package/dist/chunk-TIGW564L.js +0 -628
  42. package/dist/commands/tailscale.cjs +0 -1532
  43. package/dist/commands/tailscale.d.cts +0 -52
  44. package/dist/commands/tailscale.d.ts +0 -52
  45. package/dist/commands/tailscale.js +0 -26
  46. package/dist/lib/canvas-layout.cjs +0 -136
  47. package/dist/lib/canvas-layout.d.cts +0 -31
  48. package/dist/lib/canvas-layout.d.ts +0 -31
  49. package/dist/lib/canvas-layout.js +0 -92
  50. package/dist/lib/tailscale.cjs +0 -1183
  51. package/dist/lib/tailscale.d.cts +0 -225
  52. package/dist/lib/tailscale.d.ts +0 -225
  53. package/dist/lib/tailscale.js +0 -50
  54. package/dist/lib/webdav.cjs +0 -568
  55. package/dist/lib/webdav.d.cts +0 -109
  56. package/dist/lib/webdav.d.ts +0 -109
  57. package/dist/lib/webdav.js +0 -35
  58. package/hooks/clawvault/HOOK.md +0 -83
  59. package/hooks/clawvault/handler.js +0 -879
  60. package/hooks/clawvault/handler.test.js +0 -354
@@ -1,373 +0,0 @@
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-MXSSG3QU.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
- };