clawvault 3.1.0 → 3.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +422 -141
- package/bin/clawvault.js +10 -2
- package/bin/command-registration.test.js +3 -1
- package/bin/command-runtime.js +9 -1
- package/bin/register-core-commands.js +23 -28
- package/bin/register-maintenance-commands.js +39 -3
- package/bin/register-query-commands.js +58 -29
- package/bin/register-tailscale-commands.js +106 -0
- package/bin/register-task-commands.js +18 -1
- package/bin/register-task-commands.test.js +16 -0
- package/bin/register-vault-operations-commands.js +29 -1
- package/bin/register-workgraph-commands.js +1368 -0
- package/dashboard/lib/graph-diff.js +104 -0
- package/dashboard/lib/graph-diff.test.js +75 -0
- package/dashboard/lib/vault-parser.js +556 -0
- package/dashboard/lib/vault-parser.test.js +254 -0
- package/dashboard/public/app.js +796 -0
- package/dashboard/public/index.html +52 -0
- package/dashboard/public/styles.css +221 -0
- package/dashboard/server.js +374 -0
- package/dist/{chunk-F2JEUD4J.js → chunk-23YDQ3QU.js} +6 -8
- package/dist/{chunk-C7OK5WKP.js → chunk-2JQ3O2YL.js} +4 -4
- package/dist/{chunk-VR5NE7PZ.js → chunk-2RAZ4ZFE.js} +1 -1
- package/dist/chunk-2ZDO52B4.js +52 -0
- package/dist/{chunk-ZZA73MFY.js → chunk-33DOSHTA.js} +176 -36
- package/dist/chunk-33VSQP4J.js +37 -0
- package/dist/chunk-4BQTQMJP.js +93 -0
- package/dist/{chunk-GUKMRGM7.js → chunk-4OXMU5S2.js} +1 -1
- package/dist/{chunk-62YTUT6J.js → chunk-4PY655YM.js} +15 -3
- package/dist/chunk-6FH3IULF.js +352 -0
- package/dist/{chunk-3NSBOUT3.js → chunk-77Q5CSPJ.js} +404 -80
- package/dist/{chunk-4VQTUVH7.js → chunk-7YZWHM36.js} +52 -26
- package/dist/chunk-BSJ6RIT7.js +447 -0
- package/dist/chunk-BUEW6IIK.js +364 -0
- package/dist/{chunk-LI4O6NVK.js → chunk-CLJTREDS.js} +74 -14
- package/dist/chunk-EK6S23ZB.js +469 -0
- package/dist/{chunk-LNJA2UGL.js → chunk-ESFLMDRB.js} +9 -86
- package/dist/{chunk-H34S76MB.js → chunk-ESVS6K2B.js} +6 -6
- package/dist/{chunk-WAZ3NLWL.js → chunk-F55HGNU4.js} +0 -47
- package/dist/{chunk-QK3UCXWL.js → chunk-FHFUXL6G.js} +2 -2
- package/dist/{chunk-H62BP7RI.js → chunk-GAOWA7GR.js} +212 -46
- package/dist/chunk-GGA32J2R.js +784 -0
- package/dist/chunk-GNJL4YGR.js +79 -0
- package/dist/chunk-IVRIKYFE.js +520 -0
- package/dist/chunk-MDIH26GC.js +183 -0
- package/dist/{chunk-LYHGEHXG.js → chunk-MFAWT5O5.js} +0 -1
- package/dist/chunk-MM6QGW3P.js +207 -0
- package/dist/{chunk-P5EPF6MB.js → chunk-MW5C6ZQA.js} +110 -13
- package/dist/chunk-NCKFNBHJ.js +257 -0
- package/dist/{chunk-QBLMXKF2.js → chunk-OIWVQYQF.js} +1 -1
- package/dist/{chunk-42MXU7A6.js → chunk-P62WHA27.js} +58 -47
- package/dist/chunk-PBACDKKP.js +66 -0
- package/dist/{chunk-VGLOTGAS.js → chunk-QSHD36LH.js} +2 -2
- package/dist/{chunk-OZ7RIXTO.js → chunk-QSRRMEYM.js} +2 -2
- package/dist/chunk-QVEERJSP.js +152 -0
- package/dist/{chunk-N2AXRYLC.js → chunk-QWQ3TIKS.js} +1 -1
- package/dist/{chunk-3DHXQHYG.js → chunk-R2MIW5G7.js} +1 -1
- package/dist/{chunk-SJSFRIYS.js → chunk-SLXOR3CC.js} +2 -2
- package/dist/chunk-SS4B7P7V.js +99 -0
- package/dist/{chunk-JY6FYXIT.js → chunk-STCQGCEQ.js} +6 -11
- package/dist/chunk-TIGW564L.js +628 -0
- package/dist/chunk-U4O6C46S.js +154 -0
- package/dist/{chunk-ITPEXLHA.js → chunk-URXDAUVH.js} +24 -5
- package/dist/chunk-VSL7KY3M.js +189 -0
- package/dist/{chunk-U55BGUAU.js → chunk-W4SPAEE7.js} +6 -6
- package/dist/chunk-WMGIIABP.js +15 -0
- package/dist/{chunk-33UGEQRT.js → chunk-X3SPPUFG.js} +151 -64
- package/dist/chunk-Y6VJKXGL.js +373 -0
- package/dist/{chunk-3WRJEKN4.js → chunk-ZN54U2OZ.js} +123 -10
- package/dist/cli/index.js +34 -24
- package/dist/commands/archive.js +3 -3
- package/dist/commands/backlog.js +3 -3
- package/dist/commands/blocked.js +3 -3
- package/dist/commands/canvas.d.ts +15 -0
- package/dist/commands/canvas.js +200 -0
- package/dist/commands/checkpoint.js +2 -2
- package/dist/commands/compat.js +2 -2
- package/dist/commands/context.js +8 -6
- package/dist/commands/doctor.d.ts +11 -7
- package/dist/commands/doctor.js +18 -16
- package/dist/commands/embed.js +5 -6
- package/dist/commands/entities.js +2 -2
- package/dist/commands/graph.js +4 -4
- package/dist/commands/inject.d.ts +1 -1
- package/dist/commands/inject.js +5 -6
- package/dist/commands/kanban.js +4 -4
- package/dist/commands/link.js +5 -5
- package/dist/commands/migrate-observations.js +4 -4
- package/dist/commands/observe.d.ts +0 -1
- package/dist/commands/observe.js +14 -13
- package/dist/commands/project.js +5 -5
- package/dist/commands/rebuild-embeddings.d.ts +21 -0
- package/dist/commands/rebuild-embeddings.js +91 -0
- package/dist/commands/rebuild.js +12 -11
- package/dist/commands/recover.js +3 -3
- package/dist/commands/reflect.js +6 -7
- package/dist/commands/repair-session.js +1 -1
- package/dist/commands/replay.js +14 -14
- package/dist/commands/session-recap.js +1 -1
- package/dist/commands/setup.d.ts +2 -89
- package/dist/commands/setup.js +3 -21
- package/dist/commands/shell-init.js +1 -1
- package/dist/commands/sleep.d.ts +1 -1
- package/dist/commands/sleep.js +20 -19
- package/dist/commands/status.d.ts +2 -0
- package/dist/commands/status.js +57 -35
- package/dist/commands/sync-bd.d.ts +10 -0
- package/dist/commands/sync-bd.js +10 -0
- package/dist/commands/tailscale.d.ts +52 -0
- package/dist/commands/tailscale.js +26 -0
- package/dist/commands/task.js +4 -4
- package/dist/commands/template.js +2 -2
- package/dist/commands/wake.d.ts +1 -1
- package/dist/commands/wake.js +11 -10
- package/dist/commands/workgraph.d.ts +124 -0
- package/dist/commands/workgraph.js +38 -0
- package/dist/index.d.ts +341 -191
- package/dist/index.js +446 -116
- package/dist/{inject-Bzi5E-By.d.ts → inject-DYUrDqQO.d.ts} +3 -3
- package/dist/ledger-B7g7jhqG.d.ts +44 -0
- package/dist/lib/auto-linker.js +2 -2
- package/dist/lib/canvas-layout.d.ts +115 -0
- package/dist/lib/canvas-layout.js +35 -0
- package/dist/lib/config.d.ts +27 -3
- package/dist/lib/config.js +4 -2
- package/dist/lib/entity-index.js +1 -1
- package/dist/lib/project-utils.js +4 -4
- package/dist/lib/session-repair.js +1 -1
- package/dist/lib/session-utils.js +1 -1
- package/dist/lib/tailscale.d.ts +225 -0
- package/dist/lib/tailscale.js +50 -0
- package/dist/lib/task-utils.js +3 -3
- package/dist/lib/template-engine.js +1 -1
- package/dist/lib/webdav.d.ts +109 -0
- package/dist/lib/webdav.js +35 -0
- package/dist/onnxruntime_binding-5QEF3SUC.node +0 -0
- package/dist/onnxruntime_binding-BKPKNEGC.node +0 -0
- package/dist/onnxruntime_binding-FMOXGIUT.node +0 -0
- package/dist/onnxruntime_binding-OI2KMXC5.node +0 -0
- package/dist/onnxruntime_binding-UX44MLAZ.node +0 -0
- package/dist/onnxruntime_binding-Y2W7N7WY.node +0 -0
- package/dist/openclaw-plugin.d.ts +8 -0
- package/dist/openclaw-plugin.js +14 -0
- package/dist/registry-BR4326o0.d.ts +30 -0
- package/dist/store-CA-6sKCJ.d.ts +34 -0
- package/dist/thread-B9LhXNU0.d.ts +41 -0
- package/dist/transformers.node-A2ZRORSQ.js +46775 -0
- package/dist/{types-Y2_Um2Ls.d.ts → types-BbWJoC1c.d.ts} +1 -44
- package/dist/workgraph/index.d.ts +5 -0
- package/dist/workgraph/index.js +23 -0
- package/dist/workgraph/ledger.d.ts +2 -0
- package/dist/workgraph/ledger.js +25 -0
- package/dist/workgraph/registry.d.ts +2 -0
- package/dist/workgraph/registry.js +19 -0
- package/dist/workgraph/store.d.ts +2 -0
- package/dist/workgraph/store.js +25 -0
- package/dist/workgraph/thread.d.ts +2 -0
- package/dist/workgraph/thread.js +25 -0
- package/dist/workgraph/types.d.ts +54 -0
- package/dist/workgraph/types.js +7 -0
- package/hooks/clawvault/HOOK.md +113 -0
- package/hooks/clawvault/handler.js +1561 -0
- package/hooks/clawvault/handler.test.js +510 -0
- package/hooks/clawvault/openclaw.plugin.json +72 -0
- package/openclaw.plugin.json +65 -38
- package/package.json +25 -22
- package/dist/chunk-3RG5ZIWI.js +0 -10
- package/dist/chunk-3ZIH425O.js +0 -871
- package/dist/chunk-6U6MK36V.js +0 -205
- package/dist/chunk-CMB7UL7C.js +0 -327
- package/dist/chunk-D2H45LON.js +0 -1074
- package/dist/chunk-E7MFQB6D.js +0 -163
- package/dist/chunk-GQSLDZTS.js +0 -560
- package/dist/chunk-MFM6K7PU.js +0 -374
- package/dist/chunk-MXSSG3QU.js +0 -42
- package/dist/chunk-OCGVIN3L.js +0 -88
- package/dist/chunk-PAH27GSN.js +0 -108
- package/dist/chunk-YCUNCH2I.js +0 -78
- package/dist/cli/index.cjs +0 -8584
- package/dist/cli/index.d.cts +0 -5
- package/dist/commands/archive.cjs +0 -287
- package/dist/commands/archive.d.cts +0 -11
- package/dist/commands/backlog.cjs +0 -721
- package/dist/commands/backlog.d.cts +0 -53
- package/dist/commands/blocked.cjs +0 -204
- package/dist/commands/blocked.d.cts +0 -26
- package/dist/commands/checkpoint.cjs +0 -244
- package/dist/commands/checkpoint.d.cts +0 -41
- package/dist/commands/compat.cjs +0 -294
- package/dist/commands/compat.d.cts +0 -28
- package/dist/commands/context.cjs +0 -2990
- package/dist/commands/context.d.cts +0 -2
- package/dist/commands/doctor.cjs +0 -2986
- package/dist/commands/doctor.d.cts +0 -21
- package/dist/commands/embed.cjs +0 -232
- package/dist/commands/embed.d.cts +0 -17
- package/dist/commands/entities.cjs +0 -141
- package/dist/commands/entities.d.cts +0 -7
- package/dist/commands/graph.cjs +0 -501
- package/dist/commands/graph.d.cts +0 -21
- package/dist/commands/inject.cjs +0 -1636
- package/dist/commands/inject.d.cts +0 -2
- package/dist/commands/kanban.cjs +0 -884
- package/dist/commands/kanban.d.cts +0 -63
- package/dist/commands/link.cjs +0 -965
- package/dist/commands/link.d.cts +0 -11
- package/dist/commands/migrate-observations.cjs +0 -362
- package/dist/commands/migrate-observations.d.cts +0 -19
- package/dist/commands/observe.cjs +0 -4099
- package/dist/commands/observe.d.cts +0 -23
- package/dist/commands/project.cjs +0 -1341
- package/dist/commands/project.d.cts +0 -85
- package/dist/commands/rebuild.cjs +0 -3136
- package/dist/commands/rebuild.d.cts +0 -11
- package/dist/commands/recover.cjs +0 -361
- package/dist/commands/recover.d.cts +0 -38
- package/dist/commands/reflect.cjs +0 -1008
- package/dist/commands/reflect.d.cts +0 -11
- package/dist/commands/repair-session.cjs +0 -457
- package/dist/commands/repair-session.d.cts +0 -38
- package/dist/commands/replay.cjs +0 -4103
- package/dist/commands/replay.d.cts +0 -16
- package/dist/commands/session-recap.cjs +0 -353
- package/dist/commands/session-recap.d.cts +0 -27
- package/dist/commands/setup.cjs +0 -1278
- package/dist/commands/setup.d.cts +0 -99
- package/dist/commands/shell-init.cjs +0 -75
- package/dist/commands/shell-init.d.cts +0 -7
- package/dist/commands/sleep.cjs +0 -6029
- package/dist/commands/sleep.d.cts +0 -36
- package/dist/commands/status.cjs +0 -2737
- package/dist/commands/status.d.cts +0 -52
- package/dist/commands/task.cjs +0 -1236
- package/dist/commands/task.d.cts +0 -97
- package/dist/commands/template.cjs +0 -457
- package/dist/commands/template.d.cts +0 -36
- package/dist/commands/wake.cjs +0 -2627
- package/dist/commands/wake.d.cts +0 -22
- package/dist/context-BUGaWpyL.d.cts +0 -46
- package/dist/index.cjs +0 -12373
- package/dist/index.d.cts +0 -854
- package/dist/inject-Bzi5E-By.d.cts +0 -137
- package/dist/lib/auto-linker.cjs +0 -176
- package/dist/lib/auto-linker.d.cts +0 -26
- package/dist/lib/config.cjs +0 -78
- package/dist/lib/config.d.cts +0 -11
- package/dist/lib/entity-index.cjs +0 -84
- package/dist/lib/entity-index.d.cts +0 -26
- package/dist/lib/project-utils.cjs +0 -864
- package/dist/lib/project-utils.d.cts +0 -97
- package/dist/lib/session-repair.cjs +0 -239
- package/dist/lib/session-repair.d.cts +0 -110
- package/dist/lib/session-utils.cjs +0 -209
- package/dist/lib/session-utils.d.cts +0 -63
- package/dist/lib/task-utils.cjs +0 -1137
- package/dist/lib/task-utils.d.cts +0 -208
- package/dist/lib/template-engine.cjs +0 -47
- package/dist/lib/template-engine.d.cts +0 -11
- package/dist/plugin/index.cjs +0 -1907
- package/dist/plugin/index.d.cts +0 -36
- package/dist/plugin/index.d.ts +0 -36
- package/dist/plugin/index.js +0 -572
- package/dist/plugin/inject.cjs +0 -356
- package/dist/plugin/inject.d.cts +0 -54
- package/dist/plugin/inject.d.ts +0 -54
- package/dist/plugin/inject.js +0 -17
- package/dist/plugin/observe.cjs +0 -631
- package/dist/plugin/observe.d.cts +0 -39
- package/dist/plugin/observe.d.ts +0 -39
- package/dist/plugin/observe.js +0 -18
- package/dist/plugin/templates.cjs +0 -593
- package/dist/plugin/templates.d.cts +0 -52
- package/dist/plugin/templates.d.ts +0 -52
- package/dist/plugin/templates.js +0 -25
- package/dist/plugin/types.cjs +0 -18
- package/dist/plugin/types.d.cts +0 -209
- package/dist/plugin/types.d.ts +0 -209
- package/dist/plugin/types.js +0 -0
- package/dist/plugin/vault.cjs +0 -927
- package/dist/plugin/vault.d.cts +0 -68
- package/dist/plugin/vault.d.ts +0 -68
- package/dist/plugin/vault.js +0 -22
- package/dist/types-Y2_Um2Ls.d.cts +0 -205
- package/templates/memory-event.md +0 -67
- package/templates/party.md +0 -63
- package/templates/primitive-registry.yaml +0 -551
- package/templates/run.md +0 -68
- package/templates/trigger.md +0 -68
- package/templates/workspace.md +0 -50
|
@@ -4,6 +4,8 @@ import * as path from "path";
|
|
|
4
4
|
import matter from "gray-matter";
|
|
5
5
|
import { spawnSync } from "child_process";
|
|
6
6
|
import { fileURLToPath } from "url";
|
|
7
|
+
var REQUIRED_HOOK_EVENTS = ["gateway:startup", "command:new", "session:start"];
|
|
8
|
+
var REQUIRED_HOOK_BIN = "clawvault";
|
|
7
9
|
function readOptionalFile(filePath) {
|
|
8
10
|
try {
|
|
9
11
|
if (!fs.existsSync(filePath)) return null;
|
|
@@ -39,7 +41,7 @@ function checkOpenClawCli() {
|
|
|
39
41
|
label: "openclaw CLI available",
|
|
40
42
|
status: "warn",
|
|
41
43
|
detail: "openclaw binary not found",
|
|
42
|
-
hint: "Install OpenClaw CLI to enable
|
|
44
|
+
hint: "Install OpenClaw CLI to enable hook runtime validation."
|
|
43
45
|
};
|
|
44
46
|
}
|
|
45
47
|
if (typeof result.status === "number" && result.status !== 0) {
|
|
@@ -60,106 +62,190 @@ function checkOpenClawCli() {
|
|
|
60
62
|
}
|
|
61
63
|
return { label: "openclaw CLI available", status: "ok" };
|
|
62
64
|
}
|
|
63
|
-
function
|
|
64
|
-
const
|
|
65
|
-
|
|
66
|
-
);
|
|
67
|
-
if (!manifestRaw) {
|
|
65
|
+
function checkPackageHookRegistration(options) {
|
|
66
|
+
const packageRaw = readOptionalFile(resolveProjectFile("package.json", options.baseDir));
|
|
67
|
+
if (!packageRaw) {
|
|
68
68
|
return {
|
|
69
|
-
label: "
|
|
69
|
+
label: "package hook registration",
|
|
70
70
|
status: "error",
|
|
71
|
-
detail: "
|
|
72
|
-
hint: "Create openclaw.plugin.json with id, kind, and configSchema fields."
|
|
71
|
+
detail: "package.json not found"
|
|
73
72
|
};
|
|
74
73
|
}
|
|
75
74
|
try {
|
|
76
|
-
const
|
|
77
|
-
const
|
|
78
|
-
if (
|
|
79
|
-
if (!manifest.kind) issues.push("missing kind");
|
|
80
|
-
if (!manifest.configSchema) issues.push("missing configSchema");
|
|
81
|
-
if (issues.length > 0) {
|
|
75
|
+
const parsed = JSON.parse(packageRaw);
|
|
76
|
+
const registeredHooks = parsed.openclaw?.hooks ?? [];
|
|
77
|
+
if (registeredHooks.includes("./hooks/clawvault")) {
|
|
82
78
|
return {
|
|
83
|
-
label: "
|
|
84
|
-
status: "
|
|
85
|
-
detail:
|
|
79
|
+
label: "package hook registration",
|
|
80
|
+
status: "ok",
|
|
81
|
+
detail: "./hooks/clawvault"
|
|
86
82
|
};
|
|
87
83
|
}
|
|
88
84
|
return {
|
|
89
|
-
label: "
|
|
90
|
-
status: "
|
|
91
|
-
detail:
|
|
85
|
+
label: "package hook registration",
|
|
86
|
+
status: "error",
|
|
87
|
+
detail: "Missing ./hooks/clawvault in package openclaw.hooks"
|
|
92
88
|
};
|
|
93
89
|
} catch (err) {
|
|
94
90
|
return {
|
|
95
|
-
label: "
|
|
91
|
+
label: "package hook registration",
|
|
96
92
|
status: "error",
|
|
97
|
-
detail: err?.message || "Unable to parse
|
|
93
|
+
detail: err?.message || "Unable to parse package.json"
|
|
98
94
|
};
|
|
99
95
|
}
|
|
100
96
|
}
|
|
101
|
-
function
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
97
|
+
function checkHookManifest(options) {
|
|
98
|
+
const hookRaw = readOptionalFile(resolveProjectFile("hooks/clawvault/HOOK.md", options.baseDir));
|
|
99
|
+
if (!hookRaw) {
|
|
100
|
+
return {
|
|
101
|
+
label: "hook manifest",
|
|
102
|
+
status: "error",
|
|
103
|
+
detail: "HOOK.md not found"
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
try {
|
|
107
|
+
const parsed = matter(hookRaw);
|
|
108
|
+
const openclaw = parsed.data?.metadata?.openclaw;
|
|
109
|
+
const events = Array.isArray(openclaw?.events) ? openclaw?.events ?? [] : [];
|
|
110
|
+
const missingEvents = REQUIRED_HOOK_EVENTS.filter((event) => !events.includes(event));
|
|
111
|
+
if (missingEvents.length === 0) {
|
|
112
|
+
return {
|
|
113
|
+
label: "hook manifest events",
|
|
114
|
+
status: "ok",
|
|
115
|
+
detail: events.join(", ")
|
|
116
|
+
};
|
|
114
117
|
}
|
|
118
|
+
return {
|
|
119
|
+
label: "hook manifest events",
|
|
120
|
+
status: "error",
|
|
121
|
+
detail: `Missing events: ${missingEvents.join(", ")}`
|
|
122
|
+
};
|
|
123
|
+
} catch (err) {
|
|
124
|
+
return {
|
|
125
|
+
label: "hook manifest events",
|
|
126
|
+
status: "error",
|
|
127
|
+
detail: err?.message || "Unable to parse HOOK.md frontmatter"
|
|
128
|
+
};
|
|
115
129
|
}
|
|
116
|
-
|
|
130
|
+
}
|
|
131
|
+
function checkHookManifestRequirements(options) {
|
|
132
|
+
const hookRaw = readOptionalFile(resolveProjectFile("hooks/clawvault/HOOK.md", options.baseDir));
|
|
133
|
+
if (!hookRaw) {
|
|
117
134
|
return {
|
|
118
|
-
label: "
|
|
135
|
+
label: "hook manifest requirements",
|
|
119
136
|
status: "error",
|
|
120
|
-
detail: "
|
|
137
|
+
detail: "HOOK.md not found"
|
|
121
138
|
};
|
|
122
139
|
}
|
|
123
140
|
try {
|
|
124
|
-
const parsed =
|
|
125
|
-
const
|
|
126
|
-
|
|
141
|
+
const parsed = matter(hookRaw);
|
|
142
|
+
const requiresBins = parsed.data?.metadata?.openclaw?.requires?.bins;
|
|
143
|
+
const bins = Array.isArray(requiresBins) ? requiresBins : [];
|
|
144
|
+
if (bins.includes(REQUIRED_HOOK_BIN)) {
|
|
145
|
+
return {
|
|
146
|
+
label: "hook manifest requirements",
|
|
147
|
+
status: "ok",
|
|
148
|
+
detail: `bins: ${bins.join(", ")}`
|
|
149
|
+
};
|
|
150
|
+
}
|
|
151
|
+
return {
|
|
152
|
+
label: "hook manifest requirements",
|
|
153
|
+
status: "warn",
|
|
154
|
+
detail: `Missing required hook bin "${REQUIRED_HOOK_BIN}"`,
|
|
155
|
+
hint: 'Add metadata.openclaw.requires.bins: ["clawvault"] to hooks/clawvault/HOOK.md.'
|
|
156
|
+
};
|
|
157
|
+
} catch (err) {
|
|
158
|
+
return {
|
|
159
|
+
label: "hook manifest requirements",
|
|
160
|
+
status: "error",
|
|
161
|
+
detail: err?.message || "Unable to parse HOOK.md frontmatter"
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
function checkHookHandlerSafety(options) {
|
|
166
|
+
const handlerRaw = readOptionalFile(resolveProjectFile("hooks/clawvault/handler.js", options.baseDir));
|
|
167
|
+
if (!handlerRaw) {
|
|
168
|
+
return {
|
|
169
|
+
label: "hook handler script",
|
|
170
|
+
status: "error",
|
|
171
|
+
detail: "handler.js not found"
|
|
172
|
+
};
|
|
173
|
+
}
|
|
174
|
+
const usesExecFileSync = handlerRaw.includes("execFileSync");
|
|
175
|
+
const usesExecSync = /\bexecSync\b/.test(handlerRaw);
|
|
176
|
+
const enablesShell = /\bshell\s*:\s*true\b/.test(handlerRaw);
|
|
177
|
+
const delegatesAutoProfile = /['"]--profile['"]\s*,\s*['"]auto['"]/.test(handlerRaw);
|
|
178
|
+
const violations = [];
|
|
179
|
+
if (!usesExecFileSync || usesExecSync) {
|
|
180
|
+
violations.push("execFileSync-only execution path");
|
|
181
|
+
}
|
|
182
|
+
if (enablesShell) {
|
|
183
|
+
violations.push("shell:false execution option");
|
|
184
|
+
}
|
|
185
|
+
if (!delegatesAutoProfile) {
|
|
186
|
+
violations.push("shared context profile delegation (--profile auto)");
|
|
187
|
+
}
|
|
188
|
+
if (violations.length > 0) {
|
|
189
|
+
return {
|
|
190
|
+
label: "hook handler safety",
|
|
191
|
+
status: "warn",
|
|
192
|
+
detail: `Missing conventions: ${violations.join(", ")}`,
|
|
193
|
+
hint: "Use execFileSync (no shell), avoid execSync, and delegate profile inference via --profile auto."
|
|
194
|
+
};
|
|
195
|
+
}
|
|
196
|
+
return { label: "hook handler safety", status: "ok" };
|
|
197
|
+
}
|
|
198
|
+
function checkPluginManifest(options) {
|
|
199
|
+
const manifestRaw = readOptionalFile(resolveProjectFile("hooks/clawvault/openclaw.plugin.json", options.baseDir));
|
|
200
|
+
if (!manifestRaw) {
|
|
201
|
+
return {
|
|
202
|
+
label: "plugin manifest",
|
|
203
|
+
status: "error",
|
|
204
|
+
detail: "hooks/clawvault/openclaw.plugin.json not found",
|
|
205
|
+
hint: "Add openclaw.plugin.json to hooks/clawvault/ for config schema registration."
|
|
206
|
+
};
|
|
207
|
+
}
|
|
208
|
+
try {
|
|
209
|
+
const parsed = JSON.parse(manifestRaw);
|
|
210
|
+
if (!parsed.id || parsed.id !== "clawvault") {
|
|
127
211
|
return {
|
|
128
|
-
label: "plugin
|
|
212
|
+
label: "plugin manifest",
|
|
129
213
|
status: "error",
|
|
130
|
-
detail: "
|
|
131
|
-
hint: 'Add openclaw.extensions: ["./dist/plugin/index.js"] to package.json.'
|
|
214
|
+
detail: `Invalid plugin id: expected "clawvault", got "${parsed.id || "(missing)"}"`
|
|
132
215
|
};
|
|
133
216
|
}
|
|
134
|
-
|
|
135
|
-
const missing = extensions.filter(
|
|
136
|
-
(ext) => !fs.existsSync(path.resolve(baseDir, ext))
|
|
137
|
-
);
|
|
138
|
-
if (missing.length > 0) {
|
|
217
|
+
if (!parsed.configSchema || typeof parsed.configSchema !== "object") {
|
|
139
218
|
return {
|
|
140
|
-
label: "plugin
|
|
219
|
+
label: "plugin manifest",
|
|
141
220
|
status: "error",
|
|
142
|
-
detail:
|
|
143
|
-
hint: "
|
|
221
|
+
detail: "Missing configSchema in plugin manifest",
|
|
222
|
+
hint: "Add configSchema to openclaw.plugin.json for config validation."
|
|
223
|
+
};
|
|
224
|
+
}
|
|
225
|
+
const hasVaultPath = Boolean(parsed.configSchema.properties?.vaultPath);
|
|
226
|
+
if (!hasVaultPath) {
|
|
227
|
+
return {
|
|
228
|
+
label: "plugin manifest",
|
|
229
|
+
status: "warn",
|
|
230
|
+
detail: "configSchema missing vaultPath property",
|
|
231
|
+
hint: "Add vaultPath to configSchema.properties for vault path configuration."
|
|
144
232
|
};
|
|
145
233
|
}
|
|
146
234
|
return {
|
|
147
|
-
label: "plugin
|
|
235
|
+
label: "plugin manifest",
|
|
148
236
|
status: "ok",
|
|
149
|
-
detail:
|
|
237
|
+
detail: `id: ${parsed.id}, configSchema defined`
|
|
150
238
|
};
|
|
151
239
|
} catch (err) {
|
|
152
240
|
return {
|
|
153
|
-
label: "plugin
|
|
241
|
+
label: "plugin manifest",
|
|
154
242
|
status: "error",
|
|
155
|
-
detail: err?.message || "Unable to parse
|
|
243
|
+
detail: err?.message || "Unable to parse openclaw.plugin.json"
|
|
156
244
|
};
|
|
157
245
|
}
|
|
158
246
|
}
|
|
159
247
|
function checkSkillMetadata(options) {
|
|
160
|
-
const skillRaw = readOptionalFile(
|
|
161
|
-
resolveProjectFile("SKILL.md", options.baseDir)
|
|
162
|
-
);
|
|
248
|
+
const skillRaw = readOptionalFile(resolveProjectFile("SKILL.md", options.baseDir));
|
|
163
249
|
if (!skillRaw) {
|
|
164
250
|
return {
|
|
165
251
|
label: "skill metadata",
|
|
@@ -198,8 +284,11 @@ function checkSkillMetadata(options) {
|
|
|
198
284
|
function checkOpenClawCompatibility(options = {}) {
|
|
199
285
|
const checks = [
|
|
200
286
|
checkOpenClawCli(),
|
|
287
|
+
checkPackageHookRegistration(options),
|
|
201
288
|
checkPluginManifest(options),
|
|
202
|
-
|
|
289
|
+
checkHookManifest(options),
|
|
290
|
+
checkHookManifestRequirements(options),
|
|
291
|
+
checkHookHandlerSafety(options),
|
|
203
292
|
checkSkillMetadata(options)
|
|
204
293
|
];
|
|
205
294
|
const warnings = checks.filter((check) => check.status === "warn").length;
|
|
@@ -219,9 +308,7 @@ function formatCompatibilityReport(report) {
|
|
|
219
308
|
lines.push("");
|
|
220
309
|
for (const check of report.checks) {
|
|
221
310
|
const prefix = check.status === "ok" ? "\u2713" : check.status === "warn" ? "\u26A0" : "\u2717";
|
|
222
|
-
lines.push(
|
|
223
|
-
`${prefix} ${check.label}${check.detail ? ` \u2014 ${check.detail}` : ""}`
|
|
224
|
-
);
|
|
311
|
+
lines.push(`${prefix} ${check.label}${check.detail ? ` \u2014 ${check.detail}` : ""}`);
|
|
225
312
|
if (check.hint) {
|
|
226
313
|
lines.push(` ${check.hint}`);
|
|
227
314
|
}
|
|
@@ -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
|
+
};
|