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.
Files changed (289) 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 +1368 -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-F2JEUD4J.js → chunk-23YDQ3QU.js} +6 -8
  22. package/dist/{chunk-C7OK5WKP.js → chunk-2JQ3O2YL.js} +4 -4
  23. package/dist/{chunk-VR5NE7PZ.js → chunk-2RAZ4ZFE.js} +1 -1
  24. package/dist/chunk-2ZDO52B4.js +52 -0
  25. package/dist/{chunk-ZZA73MFY.js → chunk-33DOSHTA.js} +176 -36
  26. package/dist/chunk-33VSQP4J.js +37 -0
  27. package/dist/chunk-4BQTQMJP.js +93 -0
  28. package/dist/{chunk-GUKMRGM7.js → chunk-4OXMU5S2.js} +1 -1
  29. package/dist/{chunk-62YTUT6J.js → chunk-4PY655YM.js} +15 -3
  30. package/dist/chunk-6FH3IULF.js +352 -0
  31. package/dist/{chunk-3NSBOUT3.js → chunk-77Q5CSPJ.js} +404 -80
  32. package/dist/{chunk-4VQTUVH7.js → chunk-7YZWHM36.js} +52 -26
  33. package/dist/chunk-BSJ6RIT7.js +447 -0
  34. package/dist/chunk-BUEW6IIK.js +364 -0
  35. package/dist/{chunk-LI4O6NVK.js → chunk-CLJTREDS.js} +74 -14
  36. package/dist/chunk-EK6S23ZB.js +469 -0
  37. package/dist/{chunk-LNJA2UGL.js → chunk-ESFLMDRB.js} +9 -86
  38. package/dist/{chunk-H34S76MB.js → chunk-ESVS6K2B.js} +6 -6
  39. package/dist/{chunk-WAZ3NLWL.js → chunk-F55HGNU4.js} +0 -47
  40. package/dist/{chunk-QK3UCXWL.js → chunk-FHFUXL6G.js} +2 -2
  41. package/dist/{chunk-H62BP7RI.js → chunk-GAOWA7GR.js} +212 -46
  42. package/dist/chunk-GGA32J2R.js +784 -0
  43. package/dist/chunk-GNJL4YGR.js +79 -0
  44. package/dist/chunk-IVRIKYFE.js +520 -0
  45. package/dist/chunk-MDIH26GC.js +183 -0
  46. package/dist/{chunk-LYHGEHXG.js → chunk-MFAWT5O5.js} +0 -1
  47. package/dist/chunk-MM6QGW3P.js +207 -0
  48. package/dist/{chunk-P5EPF6MB.js → chunk-MW5C6ZQA.js} +110 -13
  49. package/dist/chunk-NCKFNBHJ.js +257 -0
  50. package/dist/{chunk-QBLMXKF2.js → chunk-OIWVQYQF.js} +1 -1
  51. package/dist/{chunk-42MXU7A6.js → chunk-P62WHA27.js} +58 -47
  52. package/dist/chunk-PBACDKKP.js +66 -0
  53. package/dist/{chunk-VGLOTGAS.js → chunk-QSHD36LH.js} +2 -2
  54. package/dist/{chunk-OZ7RIXTO.js → chunk-QSRRMEYM.js} +2 -2
  55. package/dist/chunk-QVEERJSP.js +152 -0
  56. package/dist/{chunk-N2AXRYLC.js → chunk-QWQ3TIKS.js} +1 -1
  57. package/dist/{chunk-3DHXQHYG.js → chunk-R2MIW5G7.js} +1 -1
  58. package/dist/{chunk-SJSFRIYS.js → chunk-SLXOR3CC.js} +2 -2
  59. package/dist/chunk-SS4B7P7V.js +99 -0
  60. package/dist/{chunk-JY6FYXIT.js → chunk-STCQGCEQ.js} +6 -11
  61. package/dist/chunk-TIGW564L.js +628 -0
  62. package/dist/chunk-U4O6C46S.js +154 -0
  63. package/dist/{chunk-ITPEXLHA.js → chunk-URXDAUVH.js} +24 -5
  64. package/dist/chunk-VSL7KY3M.js +189 -0
  65. package/dist/{chunk-U55BGUAU.js → chunk-W4SPAEE7.js} +6 -6
  66. package/dist/chunk-WMGIIABP.js +15 -0
  67. package/dist/{chunk-33UGEQRT.js → chunk-X3SPPUFG.js} +151 -64
  68. package/dist/chunk-Y6VJKXGL.js +373 -0
  69. package/dist/{chunk-3WRJEKN4.js → chunk-ZN54U2OZ.js} +123 -10
  70. package/dist/cli/index.js +34 -24
  71. package/dist/commands/archive.js +3 -3
  72. package/dist/commands/backlog.js +3 -3
  73. package/dist/commands/blocked.js +3 -3
  74. package/dist/commands/canvas.d.ts +15 -0
  75. package/dist/commands/canvas.js +200 -0
  76. package/dist/commands/checkpoint.js +2 -2
  77. package/dist/commands/compat.js +2 -2
  78. package/dist/commands/context.js +8 -6
  79. package/dist/commands/doctor.d.ts +11 -7
  80. package/dist/commands/doctor.js +18 -16
  81. package/dist/commands/embed.js +5 -6
  82. package/dist/commands/entities.js +2 -2
  83. package/dist/commands/graph.js +4 -4
  84. package/dist/commands/inject.d.ts +1 -1
  85. package/dist/commands/inject.js +5 -6
  86. package/dist/commands/kanban.js +4 -4
  87. package/dist/commands/link.js +5 -5
  88. package/dist/commands/migrate-observations.js +4 -4
  89. package/dist/commands/observe.d.ts +0 -1
  90. package/dist/commands/observe.js +14 -13
  91. package/dist/commands/project.js +5 -5
  92. package/dist/commands/rebuild-embeddings.d.ts +21 -0
  93. package/dist/commands/rebuild-embeddings.js +91 -0
  94. package/dist/commands/rebuild.js +12 -11
  95. package/dist/commands/recover.js +3 -3
  96. package/dist/commands/reflect.js +6 -7
  97. package/dist/commands/repair-session.js +1 -1
  98. package/dist/commands/replay.js +14 -14
  99. package/dist/commands/session-recap.js +1 -1
  100. package/dist/commands/setup.d.ts +2 -89
  101. package/dist/commands/setup.js +3 -21
  102. package/dist/commands/shell-init.js +1 -1
  103. package/dist/commands/sleep.d.ts +1 -1
  104. package/dist/commands/sleep.js +20 -19
  105. package/dist/commands/status.d.ts +2 -0
  106. package/dist/commands/status.js +57 -35
  107. package/dist/commands/sync-bd.d.ts +10 -0
  108. package/dist/commands/sync-bd.js +10 -0
  109. package/dist/commands/tailscale.d.ts +52 -0
  110. package/dist/commands/tailscale.js +26 -0
  111. package/dist/commands/task.js +4 -4
  112. package/dist/commands/template.js +2 -2
  113. package/dist/commands/wake.d.ts +1 -1
  114. package/dist/commands/wake.js +11 -10
  115. package/dist/commands/workgraph.d.ts +124 -0
  116. package/dist/commands/workgraph.js +38 -0
  117. package/dist/index.d.ts +341 -191
  118. package/dist/index.js +446 -116
  119. package/dist/{inject-Bzi5E-By.d.ts → inject-DYUrDqQO.d.ts} +3 -3
  120. package/dist/ledger-B7g7jhqG.d.ts +44 -0
  121. package/dist/lib/auto-linker.js +2 -2
  122. package/dist/lib/canvas-layout.d.ts +115 -0
  123. package/dist/lib/canvas-layout.js +35 -0
  124. package/dist/lib/config.d.ts +27 -3
  125. package/dist/lib/config.js +4 -2
  126. package/dist/lib/entity-index.js +1 -1
  127. package/dist/lib/project-utils.js +4 -4
  128. package/dist/lib/session-repair.js +1 -1
  129. package/dist/lib/session-utils.js +1 -1
  130. package/dist/lib/tailscale.d.ts +225 -0
  131. package/dist/lib/tailscale.js +50 -0
  132. package/dist/lib/task-utils.js +3 -3
  133. package/dist/lib/template-engine.js +1 -1
  134. package/dist/lib/webdav.d.ts +109 -0
  135. package/dist/lib/webdav.js +35 -0
  136. package/dist/onnxruntime_binding-5QEF3SUC.node +0 -0
  137. package/dist/onnxruntime_binding-BKPKNEGC.node +0 -0
  138. package/dist/onnxruntime_binding-FMOXGIUT.node +0 -0
  139. package/dist/onnxruntime_binding-OI2KMXC5.node +0 -0
  140. package/dist/onnxruntime_binding-UX44MLAZ.node +0 -0
  141. package/dist/onnxruntime_binding-Y2W7N7WY.node +0 -0
  142. package/dist/openclaw-plugin.d.ts +8 -0
  143. package/dist/openclaw-plugin.js +14 -0
  144. package/dist/registry-BR4326o0.d.ts +30 -0
  145. package/dist/store-CA-6sKCJ.d.ts +34 -0
  146. package/dist/thread-B9LhXNU0.d.ts +41 -0
  147. package/dist/transformers.node-A2ZRORSQ.js +46775 -0
  148. package/dist/{types-Y2_Um2Ls.d.ts → types-BbWJoC1c.d.ts} +1 -44
  149. package/dist/workgraph/index.d.ts +5 -0
  150. package/dist/workgraph/index.js +23 -0
  151. package/dist/workgraph/ledger.d.ts +2 -0
  152. package/dist/workgraph/ledger.js +25 -0
  153. package/dist/workgraph/registry.d.ts +2 -0
  154. package/dist/workgraph/registry.js +19 -0
  155. package/dist/workgraph/store.d.ts +2 -0
  156. package/dist/workgraph/store.js +25 -0
  157. package/dist/workgraph/thread.d.ts +2 -0
  158. package/dist/workgraph/thread.js +25 -0
  159. package/dist/workgraph/types.d.ts +54 -0
  160. package/dist/workgraph/types.js +7 -0
  161. package/hooks/clawvault/HOOK.md +113 -0
  162. package/hooks/clawvault/handler.js +1561 -0
  163. package/hooks/clawvault/handler.test.js +510 -0
  164. package/hooks/clawvault/openclaw.plugin.json +72 -0
  165. package/openclaw.plugin.json +65 -38
  166. package/package.json +25 -22
  167. package/dist/chunk-3RG5ZIWI.js +0 -10
  168. package/dist/chunk-3ZIH425O.js +0 -871
  169. package/dist/chunk-6U6MK36V.js +0 -205
  170. package/dist/chunk-CMB7UL7C.js +0 -327
  171. package/dist/chunk-D2H45LON.js +0 -1074
  172. package/dist/chunk-E7MFQB6D.js +0 -163
  173. package/dist/chunk-GQSLDZTS.js +0 -560
  174. package/dist/chunk-MFM6K7PU.js +0 -374
  175. package/dist/chunk-MXSSG3QU.js +0 -42
  176. package/dist/chunk-OCGVIN3L.js +0 -88
  177. package/dist/chunk-PAH27GSN.js +0 -108
  178. package/dist/chunk-YCUNCH2I.js +0 -78
  179. package/dist/cli/index.cjs +0 -8584
  180. package/dist/cli/index.d.cts +0 -5
  181. package/dist/commands/archive.cjs +0 -287
  182. package/dist/commands/archive.d.cts +0 -11
  183. package/dist/commands/backlog.cjs +0 -721
  184. package/dist/commands/backlog.d.cts +0 -53
  185. package/dist/commands/blocked.cjs +0 -204
  186. package/dist/commands/blocked.d.cts +0 -26
  187. package/dist/commands/checkpoint.cjs +0 -244
  188. package/dist/commands/checkpoint.d.cts +0 -41
  189. package/dist/commands/compat.cjs +0 -294
  190. package/dist/commands/compat.d.cts +0 -28
  191. package/dist/commands/context.cjs +0 -2990
  192. package/dist/commands/context.d.cts +0 -2
  193. package/dist/commands/doctor.cjs +0 -2986
  194. package/dist/commands/doctor.d.cts +0 -21
  195. package/dist/commands/embed.cjs +0 -232
  196. package/dist/commands/embed.d.cts +0 -17
  197. package/dist/commands/entities.cjs +0 -141
  198. package/dist/commands/entities.d.cts +0 -7
  199. package/dist/commands/graph.cjs +0 -501
  200. package/dist/commands/graph.d.cts +0 -21
  201. package/dist/commands/inject.cjs +0 -1636
  202. package/dist/commands/inject.d.cts +0 -2
  203. package/dist/commands/kanban.cjs +0 -884
  204. package/dist/commands/kanban.d.cts +0 -63
  205. package/dist/commands/link.cjs +0 -965
  206. package/dist/commands/link.d.cts +0 -11
  207. package/dist/commands/migrate-observations.cjs +0 -362
  208. package/dist/commands/migrate-observations.d.cts +0 -19
  209. package/dist/commands/observe.cjs +0 -4099
  210. package/dist/commands/observe.d.cts +0 -23
  211. package/dist/commands/project.cjs +0 -1341
  212. package/dist/commands/project.d.cts +0 -85
  213. package/dist/commands/rebuild.cjs +0 -3136
  214. package/dist/commands/rebuild.d.cts +0 -11
  215. package/dist/commands/recover.cjs +0 -361
  216. package/dist/commands/recover.d.cts +0 -38
  217. package/dist/commands/reflect.cjs +0 -1008
  218. package/dist/commands/reflect.d.cts +0 -11
  219. package/dist/commands/repair-session.cjs +0 -457
  220. package/dist/commands/repair-session.d.cts +0 -38
  221. package/dist/commands/replay.cjs +0 -4103
  222. package/dist/commands/replay.d.cts +0 -16
  223. package/dist/commands/session-recap.cjs +0 -353
  224. package/dist/commands/session-recap.d.cts +0 -27
  225. package/dist/commands/setup.cjs +0 -1278
  226. package/dist/commands/setup.d.cts +0 -99
  227. package/dist/commands/shell-init.cjs +0 -75
  228. package/dist/commands/shell-init.d.cts +0 -7
  229. package/dist/commands/sleep.cjs +0 -6029
  230. package/dist/commands/sleep.d.cts +0 -36
  231. package/dist/commands/status.cjs +0 -2737
  232. package/dist/commands/status.d.cts +0 -52
  233. package/dist/commands/task.cjs +0 -1236
  234. package/dist/commands/task.d.cts +0 -97
  235. package/dist/commands/template.cjs +0 -457
  236. package/dist/commands/template.d.cts +0 -36
  237. package/dist/commands/wake.cjs +0 -2627
  238. package/dist/commands/wake.d.cts +0 -22
  239. package/dist/context-BUGaWpyL.d.cts +0 -46
  240. package/dist/index.cjs +0 -12373
  241. package/dist/index.d.cts +0 -854
  242. package/dist/inject-Bzi5E-By.d.cts +0 -137
  243. package/dist/lib/auto-linker.cjs +0 -176
  244. package/dist/lib/auto-linker.d.cts +0 -26
  245. package/dist/lib/config.cjs +0 -78
  246. package/dist/lib/config.d.cts +0 -11
  247. package/dist/lib/entity-index.cjs +0 -84
  248. package/dist/lib/entity-index.d.cts +0 -26
  249. package/dist/lib/project-utils.cjs +0 -864
  250. package/dist/lib/project-utils.d.cts +0 -97
  251. package/dist/lib/session-repair.cjs +0 -239
  252. package/dist/lib/session-repair.d.cts +0 -110
  253. package/dist/lib/session-utils.cjs +0 -209
  254. package/dist/lib/session-utils.d.cts +0 -63
  255. package/dist/lib/task-utils.cjs +0 -1137
  256. package/dist/lib/task-utils.d.cts +0 -208
  257. package/dist/lib/template-engine.cjs +0 -47
  258. package/dist/lib/template-engine.d.cts +0 -11
  259. package/dist/plugin/index.cjs +0 -1907
  260. package/dist/plugin/index.d.cts +0 -36
  261. package/dist/plugin/index.d.ts +0 -36
  262. package/dist/plugin/index.js +0 -572
  263. package/dist/plugin/inject.cjs +0 -356
  264. package/dist/plugin/inject.d.cts +0 -54
  265. package/dist/plugin/inject.d.ts +0 -54
  266. package/dist/plugin/inject.js +0 -17
  267. package/dist/plugin/observe.cjs +0 -631
  268. package/dist/plugin/observe.d.cts +0 -39
  269. package/dist/plugin/observe.d.ts +0 -39
  270. package/dist/plugin/observe.js +0 -18
  271. package/dist/plugin/templates.cjs +0 -593
  272. package/dist/plugin/templates.d.cts +0 -52
  273. package/dist/plugin/templates.d.ts +0 -52
  274. package/dist/plugin/templates.js +0 -25
  275. package/dist/plugin/types.cjs +0 -18
  276. package/dist/plugin/types.d.cts +0 -209
  277. package/dist/plugin/types.d.ts +0 -209
  278. package/dist/plugin/types.js +0 -0
  279. package/dist/plugin/vault.cjs +0 -927
  280. package/dist/plugin/vault.d.cts +0 -68
  281. package/dist/plugin/vault.d.ts +0 -68
  282. package/dist/plugin/vault.js +0 -22
  283. package/dist/types-Y2_Um2Ls.d.cts +0 -205
  284. package/templates/memory-event.md +0 -67
  285. package/templates/party.md +0 -63
  286. package/templates/primitive-registry.yaml +0 -551
  287. package/templates/run.md +0 -68
  288. package/templates/trigger.md +0 -68
  289. 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 plugin runtime validation."
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 checkPluginManifest(options) {
64
- const manifestRaw = readOptionalFile(
65
- resolveProjectFile("openclaw.plugin.json", options.baseDir)
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: "plugin manifest",
69
+ label: "package hook registration",
70
70
  status: "error",
71
- detail: "openclaw.plugin.json not found",
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 manifest = JSON.parse(manifestRaw);
77
- const issues = [];
78
- if (!manifest.id) issues.push("missing id");
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: "plugin manifest",
84
- status: "error",
85
- detail: issues.join(", ")
79
+ label: "package hook registration",
80
+ status: "ok",
81
+ detail: "./hooks/clawvault"
86
82
  };
87
83
  }
88
84
  return {
89
- label: "plugin manifest",
90
- status: "ok",
91
- detail: `id=${manifest.id} kind=${manifest.kind}`
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: "plugin manifest",
91
+ label: "package hook registration",
96
92
  status: "error",
97
- detail: err?.message || "Unable to parse openclaw.plugin.json"
93
+ detail: err?.message || "Unable to parse package.json"
98
94
  };
99
95
  }
100
96
  }
101
- function checkPluginExtensions(options) {
102
- let packageRaw = readOptionalFile(
103
- resolveProjectFile("package.json", options.baseDir)
104
- );
105
- if (packageRaw && !options.baseDir) {
106
- try {
107
- const parsed = JSON.parse(packageRaw);
108
- if (!parsed.openclaw?.extensions) {
109
- const fallbackPath = path.resolve(findPackageRoot(), "package.json");
110
- const fallbackRaw = readOptionalFile(fallbackPath);
111
- if (fallbackRaw) packageRaw = fallbackRaw;
112
- }
113
- } catch {
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
- if (!packageRaw) {
130
+ }
131
+ function checkHookManifestRequirements(options) {
132
+ const hookRaw = readOptionalFile(resolveProjectFile("hooks/clawvault/HOOK.md", options.baseDir));
133
+ if (!hookRaw) {
117
134
  return {
118
- label: "plugin extensions registration",
135
+ label: "hook manifest requirements",
119
136
  status: "error",
120
- detail: "package.json not found"
137
+ detail: "HOOK.md not found"
121
138
  };
122
139
  }
123
140
  try {
124
- const parsed = JSON.parse(packageRaw);
125
- const extensions = parsed.openclaw?.extensions ?? [];
126
- if (extensions.length === 0) {
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 extensions registration",
212
+ label: "plugin manifest",
129
213
  status: "error",
130
- detail: "Missing openclaw.extensions in package.json",
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
- const baseDir = options.baseDir || findPackageRoot();
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 extensions registration",
219
+ label: "plugin manifest",
141
220
  status: "error",
142
- detail: `Entry file(s) not found: ${missing.join(", ")}`,
143
- hint: "Run npm run build to generate dist files."
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 extensions registration",
235
+ label: "plugin manifest",
148
236
  status: "ok",
149
- detail: extensions.join(", ")
237
+ detail: `id: ${parsed.id}, configSchema defined`
150
238
  };
151
239
  } catch (err) {
152
240
  return {
153
- label: "plugin extensions registration",
241
+ label: "plugin manifest",
154
242
  status: "error",
155
- detail: err?.message || "Unable to parse package.json"
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
- checkPluginExtensions(options),
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
+ };