clawvault 2.3.0 → 2.4.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.
- package/bin/clawvault.js +6 -0
- package/bin/register-core-commands.js +25 -2
- package/bin/register-tailscale-commands.js +106 -0
- package/bin/register-task-commands.js +18 -2
- package/bin/register-task-commands.test.js +24 -0
- package/dist/{chunk-SOTWYGH7.js → chunk-33GW63WK.js} +4 -35
- package/dist/chunk-4GBPTBFJ.js +628 -0
- package/dist/{chunk-GQVYQCY5.js → chunk-AHGUJG76.js} +3 -3
- package/dist/{chunk-GJEGPO7U.js → chunk-BI6SGGZP.js} +1 -1
- package/dist/chunk-CLE2HHNT.js +513 -0
- package/dist/{chunk-NGVAEFT2.js → chunk-DEFBIVQ3.js} +21 -0
- package/dist/{chunk-DPS7NYIU.js → chunk-DHJPXGC7.js} +2 -2
- package/dist/{chunk-2HM7ZI4X.js → chunk-FEFPBHH4.js} +287 -12
- package/dist/{chunk-K6XHCUFL.js → chunk-FHFUXL6G.js} +8 -1
- package/dist/{chunk-6BBTI7NV.js → chunk-GBIDDDSL.js} +2 -2
- package/dist/{chunk-VR5NE7PZ.js → chunk-HVTTYDCJ.js} +1 -1
- package/dist/chunk-IFTEGE4D.js +361 -0
- package/dist/{chunk-5WR6RRPX.js → chunk-JXY6T5R7.js} +2 -2
- package/dist/chunk-L3DJ36BZ.js +40 -0
- package/dist/{chunk-Z2XBWN7A.js → chunk-NAMFB7ZA.js} +2 -0
- package/dist/chunk-NZ4ZZNSR.js +373 -0
- package/dist/{chunk-OTQW3OMC.js → chunk-Q3WBH4P4.js} +97 -0
- package/dist/{chunk-MQUJNOHK.js → chunk-QALB2V3E.js} +1 -1
- package/dist/{chunk-PTSEIWXZ.js → chunk-SNEMCQP7.js} +13 -6
- package/dist/commands/archive.js +3 -3
- package/dist/commands/backlog.js +9 -2
- package/dist/commands/blocked.js +9 -2
- package/dist/commands/canvas.d.ts +11 -3
- package/dist/commands/canvas.js +1333 -25
- package/dist/commands/context.js +5 -4
- package/dist/commands/doctor.js +1 -1
- package/dist/commands/migrate-observations.js +3 -3
- package/dist/commands/observe.d.ts +1 -0
- package/dist/commands/observe.js +5 -4
- package/dist/commands/rebuild.js +5 -4
- package/dist/commands/reflect.js +5 -5
- package/dist/commands/replay.js +7 -6
- package/dist/commands/setup.d.ts +10 -2
- package/dist/commands/setup.js +1 -1
- package/dist/commands/sleep.js +7 -6
- package/dist/commands/status.js +1 -1
- package/dist/commands/tailscale.d.ts +52 -0
- package/dist/commands/tailscale.js +25 -0
- package/dist/commands/task.js +1 -1
- package/dist/commands/wake.js +4 -4
- package/dist/index.d.ts +9 -0
- package/dist/index.js +79 -15
- package/dist/lib/tailscale.d.ts +225 -0
- package/dist/lib/tailscale.js +49 -0
- package/dist/lib/task-utils.d.ts +13 -1
- package/dist/lib/task-utils.js +3 -1
- package/dist/lib/webdav.d.ts +109 -0
- package/dist/lib/webdav.js +34 -0
- package/package.json +2 -2
- package/dist/chunk-W463YRED.js +0 -97
|
@@ -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-4GBPTBFJ.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
|
+
};
|
|
@@ -83,9 +83,106 @@ var ClawVault = class {
|
|
|
83
83
|
qmdRoot: this.getQmdRoot()
|
|
84
84
|
};
|
|
85
85
|
fs.writeFileSync(configPath, JSON.stringify(meta, null, 2));
|
|
86
|
+
this.createBasesFiles();
|
|
86
87
|
await this.syncMemoryGraphIndex({ forceFull: true });
|
|
87
88
|
this.initialized = true;
|
|
88
89
|
}
|
|
90
|
+
createBasesFiles() {
|
|
91
|
+
const vaultPath = this.config.path;
|
|
92
|
+
const basesFiles = {
|
|
93
|
+
"all-tasks.base": [
|
|
94
|
+
"filters:",
|
|
95
|
+
" and:",
|
|
96
|
+
' - file.inFolder("tasks")',
|
|
97
|
+
' - status != "done"',
|
|
98
|
+
"formulas:",
|
|
99
|
+
" age: (now() - file.ctime).days",
|
|
100
|
+
' status_icon: if(status == "blocked", "\u{1F534}", if(status == "in-progress", "\u{1F528}", if(status == "open", "\u26AA", "\u2705")))',
|
|
101
|
+
"views:",
|
|
102
|
+
" - type: table",
|
|
103
|
+
" name: All Active Tasks",
|
|
104
|
+
" groupBy:",
|
|
105
|
+
" property: status",
|
|
106
|
+
" direction: ASC",
|
|
107
|
+
" order:",
|
|
108
|
+
" - formula.status_icon",
|
|
109
|
+
" - file.name",
|
|
110
|
+
" - status",
|
|
111
|
+
" - owner",
|
|
112
|
+
" - project",
|
|
113
|
+
" - priority",
|
|
114
|
+
" - blocked_by",
|
|
115
|
+
" - formula.age",
|
|
116
|
+
" - type: cards",
|
|
117
|
+
" name: Task Board",
|
|
118
|
+
" groupBy:",
|
|
119
|
+
" property: status",
|
|
120
|
+
" direction: ASC",
|
|
121
|
+
" order:",
|
|
122
|
+
" - file.name",
|
|
123
|
+
" - owner",
|
|
124
|
+
" - project",
|
|
125
|
+
" - priority"
|
|
126
|
+
].join("\n"),
|
|
127
|
+
"blocked.base": [
|
|
128
|
+
"filters:",
|
|
129
|
+
" and:",
|
|
130
|
+
' - file.inFolder("tasks")',
|
|
131
|
+
' - status == "blocked"',
|
|
132
|
+
"formulas:",
|
|
133
|
+
" days_blocked: (now() - file.ctime).days",
|
|
134
|
+
"views:",
|
|
135
|
+
" - type: table",
|
|
136
|
+
" name: Blocked Tasks",
|
|
137
|
+
" order:",
|
|
138
|
+
" - file.name",
|
|
139
|
+
" - owner",
|
|
140
|
+
" - project",
|
|
141
|
+
" - blocked_by",
|
|
142
|
+
" - formula.days_blocked",
|
|
143
|
+
" - priority"
|
|
144
|
+
].join("\n"),
|
|
145
|
+
"by-project.base": [
|
|
146
|
+
"filters:",
|
|
147
|
+
" and:",
|
|
148
|
+
' - file.inFolder("tasks")',
|
|
149
|
+
' - status != "done"',
|
|
150
|
+
"formulas:",
|
|
151
|
+
' status_icon: if(status == "blocked", "\u{1F534}", if(status == "in-progress", "\u{1F528}", "\u26AA"))',
|
|
152
|
+
"views:",
|
|
153
|
+
" - type: table",
|
|
154
|
+
" name: By Project",
|
|
155
|
+
" groupBy:",
|
|
156
|
+
" property: project",
|
|
157
|
+
" direction: ASC",
|
|
158
|
+
" order:",
|
|
159
|
+
" - formula.status_icon",
|
|
160
|
+
" - file.name",
|
|
161
|
+
" - status",
|
|
162
|
+
" - owner",
|
|
163
|
+
" - priority"
|
|
164
|
+
].join("\n"),
|
|
165
|
+
"backlog.base": [
|
|
166
|
+
"filters:",
|
|
167
|
+
" and:",
|
|
168
|
+
' - file.inFolder("backlog")',
|
|
169
|
+
"views:",
|
|
170
|
+
" - type: table",
|
|
171
|
+
" name: Backlog",
|
|
172
|
+
" order:",
|
|
173
|
+
" - file.name",
|
|
174
|
+
" - source",
|
|
175
|
+
" - project",
|
|
176
|
+
" - file.ctime"
|
|
177
|
+
].join("\n")
|
|
178
|
+
};
|
|
179
|
+
for (const [filename, content] of Object.entries(basesFiles)) {
|
|
180
|
+
const filePath = path.join(vaultPath, filename);
|
|
181
|
+
if (!fs.existsSync(filePath)) {
|
|
182
|
+
fs.writeFileSync(filePath, content);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
}
|
|
89
186
|
/**
|
|
90
187
|
* Load an existing vault
|
|
91
188
|
*/
|
|
@@ -6,13 +6,13 @@ import {
|
|
|
6
6
|
} from "./chunk-P5EPF6MB.js";
|
|
7
7
|
import {
|
|
8
8
|
Observer
|
|
9
|
-
} from "./chunk-
|
|
9
|
+
} from "./chunk-FEFPBHH4.js";
|
|
10
10
|
import {
|
|
11
11
|
resolveVaultPath
|
|
12
12
|
} from "./chunk-MXSSG3QU.js";
|
|
13
13
|
import {
|
|
14
14
|
getObservationPath
|
|
15
|
-
} from "./chunk-
|
|
15
|
+
} from "./chunk-NAMFB7ZA.js";
|
|
16
16
|
|
|
17
17
|
// src/commands/observe.ts
|
|
18
18
|
import * as fs3 from "fs";
|
|
@@ -523,7 +523,8 @@ async function observeActiveSessions(options, dependencies = {}) {
|
|
|
523
523
|
const observer = observerFactory(vaultPath, {
|
|
524
524
|
tokenThreshold: options.threshold,
|
|
525
525
|
reflectThreshold: options.reflectThreshold,
|
|
526
|
-
model: options.model
|
|
526
|
+
model: options.model,
|
|
527
|
+
extractTasks: options.extractTasks
|
|
527
528
|
});
|
|
528
529
|
let observedSessions = 0;
|
|
529
530
|
let cursorUpdates = 0;
|
|
@@ -591,6 +592,9 @@ function buildDaemonArgs(options) {
|
|
|
591
592
|
if (options.model) {
|
|
592
593
|
args.push("--model", options.model);
|
|
593
594
|
}
|
|
595
|
+
if (options.extractTasks === false) {
|
|
596
|
+
args.push("--no-extract-tasks");
|
|
597
|
+
}
|
|
594
598
|
if (options.vaultPath) {
|
|
595
599
|
args.push("--vault", options.vaultPath);
|
|
596
600
|
}
|
|
@@ -653,7 +657,8 @@ async function observeCommand(options) {
|
|
|
653
657
|
dryRun: options.dryRun,
|
|
654
658
|
threshold: options.threshold,
|
|
655
659
|
reflectThreshold: options.reflectThreshold,
|
|
656
|
-
model: options.model
|
|
660
|
+
model: options.model,
|
|
661
|
+
extractTasks: options.extractTasks
|
|
657
662
|
});
|
|
658
663
|
if (result.candidateSessions === 0) {
|
|
659
664
|
console.log(`No active sessions crossed threshold (${result.checkedSessions} checked).`);
|
|
@@ -678,7 +683,8 @@ async function observeCommand(options) {
|
|
|
678
683
|
const observer = new Observer(vaultPath, {
|
|
679
684
|
tokenThreshold: options.threshold,
|
|
680
685
|
reflectThreshold: options.reflectThreshold,
|
|
681
|
-
model: options.model
|
|
686
|
+
model: options.model,
|
|
687
|
+
extractTasks: options.extractTasks
|
|
682
688
|
});
|
|
683
689
|
if (options.compress) {
|
|
684
690
|
await runOneShotCompression(observer, options.compress, vaultPath);
|
|
@@ -711,7 +717,7 @@ async function observeCommand(options) {
|
|
|
711
717
|
await watchSessions(observer, watchPath);
|
|
712
718
|
}
|
|
713
719
|
function registerObserveCommand(program) {
|
|
714
|
-
program.command("observe").description("Observe session files and build observational memory").option("--watch <path>", "Watch session file or directory").option("--active", "Observe active OpenClaw sessions incrementally").option("--agent <id>", "OpenClaw agent ID (default: OPENCLAW_AGENT_ID or clawdious)").option("--min-new <bytes>", "Override minimum new-content threshold in bytes").option("--sessions-dir <path>", "Override OpenClaw sessions directory").option("--dry-run", "Show active observation candidates without compressing").option("--threshold <n>", "Compression token threshold", "30000").option("--reflect-threshold <n>", "Reflection token threshold", "40000").option("--model <model>", "LLM model override").option("--compress <file>", "One-shot compression for a conversation file").option("--daemon", "Run in detached background mode").option("-v, --vault <path>", "Vault path").action(async (rawOptions) => {
|
|
720
|
+
program.command("observe").description("Observe session files and build observational memory").option("--watch <path>", "Watch session file or directory").option("--active", "Observe active OpenClaw sessions incrementally").option("--agent <id>", "OpenClaw agent ID (default: OPENCLAW_AGENT_ID or clawdious)").option("--min-new <bytes>", "Override minimum new-content threshold in bytes").option("--sessions-dir <path>", "Override OpenClaw sessions directory").option("--dry-run", "Show active observation candidates without compressing").option("--threshold <n>", "Compression token threshold", "30000").option("--reflect-threshold <n>", "Reflection token threshold", "40000").option("--model <model>", "LLM model override").option("--extract-tasks", "Extract task-like observations into backlog", true).option("--no-extract-tasks", "Disable task extraction from observations").option("--compress <file>", "One-shot compression for a conversation file").option("--daemon", "Run in detached background mode").option("-v, --vault <path>", "Vault path").action(async (rawOptions) => {
|
|
715
721
|
await observeCommand({
|
|
716
722
|
watch: rawOptions.watch,
|
|
717
723
|
active: rawOptions.active,
|
|
@@ -722,6 +728,7 @@ function registerObserveCommand(program) {
|
|
|
722
728
|
threshold: parsePositiveInteger(rawOptions.threshold, "threshold"),
|
|
723
729
|
reflectThreshold: parsePositiveInteger(rawOptions.reflectThreshold, "reflect-threshold"),
|
|
724
730
|
model: rawOptions.model,
|
|
731
|
+
extractTasks: rawOptions.extractTasks,
|
|
725
732
|
compress: rawOptions.compress,
|
|
726
733
|
daemon: rawOptions.daemon,
|
|
727
734
|
vaultPath: rawOptions.vault
|
package/dist/commands/archive.js
CHANGED
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
import {
|
|
2
2
|
archiveCommand,
|
|
3
3
|
registerArchiveCommand
|
|
4
|
-
} from "../chunk-
|
|
5
|
-
import "../chunk-
|
|
4
|
+
} from "../chunk-HVTTYDCJ.js";
|
|
5
|
+
import "../chunk-QALB2V3E.js";
|
|
6
6
|
import "../chunk-MXSSG3QU.js";
|
|
7
|
-
import "../chunk-
|
|
7
|
+
import "../chunk-NAMFB7ZA.js";
|
|
8
8
|
export {
|
|
9
9
|
archiveCommand,
|
|
10
10
|
registerArchiveCommand
|
package/dist/commands/backlog.js
CHANGED
|
@@ -2,9 +2,16 @@ import {
|
|
|
2
2
|
createBacklogItem,
|
|
3
3
|
listBacklogItems,
|
|
4
4
|
promoteBacklogItem
|
|
5
|
-
} from "../chunk-
|
|
5
|
+
} from "../chunk-DEFBIVQ3.js";
|
|
6
6
|
|
|
7
7
|
// src/commands/backlog.ts
|
|
8
|
+
function toDateStr(val) {
|
|
9
|
+
if (!val) return "unknown";
|
|
10
|
+
if (val instanceof Date) return val.toISOString().split("T")[0];
|
|
11
|
+
const s = String(val);
|
|
12
|
+
if (s.includes("T")) return s.split("T")[0];
|
|
13
|
+
return s;
|
|
14
|
+
}
|
|
8
15
|
function backlogAdd(vaultPath, title, options = {}) {
|
|
9
16
|
return createBacklogItem(vaultPath, title, {
|
|
10
17
|
source: options.source,
|
|
@@ -35,7 +42,7 @@ function formatBacklogList(items) {
|
|
|
35
42
|
for (const item of items) {
|
|
36
43
|
const source = item.frontmatter.source || "-";
|
|
37
44
|
const project = item.frontmatter.project || "-";
|
|
38
|
-
const created = item.frontmatter.created
|
|
45
|
+
const created = toDateStr(item.frontmatter.created);
|
|
39
46
|
const title = item.title.length > widths[3] ? item.title.slice(0, widths[3] - 3) + "..." : item.title;
|
|
40
47
|
const row = [
|
|
41
48
|
source.padEnd(widths[0]),
|
package/dist/commands/blocked.js
CHANGED
|
@@ -1,8 +1,15 @@
|
|
|
1
1
|
import {
|
|
2
2
|
getBlockedTasks
|
|
3
|
-
} from "../chunk-
|
|
3
|
+
} from "../chunk-DEFBIVQ3.js";
|
|
4
4
|
|
|
5
5
|
// src/commands/blocked.ts
|
|
6
|
+
function toDateStr(val) {
|
|
7
|
+
if (!val) return "unknown";
|
|
8
|
+
if (val instanceof Date) return val.toISOString().split("T")[0];
|
|
9
|
+
const s = String(val);
|
|
10
|
+
if (s.includes("T")) return s.split("T")[0];
|
|
11
|
+
return s;
|
|
12
|
+
}
|
|
6
13
|
function blockedList(vaultPath, options = {}) {
|
|
7
14
|
return getBlockedTasks(vaultPath, options.project);
|
|
8
15
|
}
|
|
@@ -17,7 +24,7 @@ function formatBlockedList(tasks) {
|
|
|
17
24
|
const owner = task.frontmatter.owner || "unassigned";
|
|
18
25
|
const project = task.frontmatter.project || "no project";
|
|
19
26
|
const blockedBy = task.frontmatter.blocked_by || "unknown";
|
|
20
|
-
const updatedDate = task.frontmatter.updated
|
|
27
|
+
const updatedDate = toDateStr(task.frontmatter.updated);
|
|
21
28
|
output += `\u25A0 ${task.title} (${owner}, ${project})
|
|
22
29
|
`;
|
|
23
30
|
output += ` Blocked by: ${blockedBy}
|
|
@@ -2,18 +2,26 @@ import { Canvas } from '../lib/canvas-layout.js';
|
|
|
2
2
|
|
|
3
3
|
/**
|
|
4
4
|
* Canvas command for ClawVault
|
|
5
|
-
* Generates
|
|
5
|
+
* Generates Obsidian JSON Canvas dashboards using registered templates.
|
|
6
6
|
*/
|
|
7
7
|
|
|
8
8
|
interface CanvasOptions {
|
|
9
9
|
output?: string;
|
|
10
|
+
template?: string;
|
|
11
|
+
listTemplates?: boolean;
|
|
12
|
+
project?: string;
|
|
13
|
+
owner?: string;
|
|
14
|
+
width?: number;
|
|
15
|
+
height?: number;
|
|
16
|
+
includeDone?: boolean;
|
|
10
17
|
}
|
|
11
18
|
/**
|
|
12
|
-
* Generate the
|
|
19
|
+
* Generate the default dashboard canvas.
|
|
20
|
+
* Kept for compatibility with existing tests and callers.
|
|
13
21
|
*/
|
|
14
22
|
declare function generateCanvas(vaultPath: string): Canvas;
|
|
15
23
|
/**
|
|
16
|
-
* Canvas command handler for CLI
|
|
24
|
+
* Canvas command handler for CLI.
|
|
17
25
|
*/
|
|
18
26
|
declare function canvasCommand(vaultPath: string, options?: CanvasOptions): Promise<void>;
|
|
19
27
|
|