feat(cli): vault set / watch add / webhook create + prune dead MCP stubs
Some checks failed
CI / Lint (push) Has been cancelled
CI / Typecheck (push) Has been cancelled
CI / Broker tests (Postgres) (push) Has been cancelled
CI / Docker build (linux/amd64) (push) Has been cancelled

Closes the last functional gaps where the MCP tool registry exposed
write verbs the CLI didn't:

- vault set <k> <v> [--type env|file --mount <path> --description ...]
  Client-side crypto_secretbox_easy with a fresh symmetric key sealed
  to the member's own pubkey via crypto_box_seal — same pattern used
  for file shares. Pairs with the existing vault list/delete.
- watch add <url> [--label --interval --mode --extract --notify-on]
  Pairs with watch list/remove.
- webhook create <name> — pairs with webhook list/delete.

Cleanup: deletes 22 stub files under apps/cli/src/mcp/tools/* plus
router.ts, middleware/, handlers/ (~120 LoC). These were FAMILY/TOOLS
metadata-only re-exports left over from before the 1.5.0 tool-less
push-pipe flip; nothing imports them. The legitimate MCP surfaces
stay: the inbound <channel> push pipe, mesh skills as prompts and
skill:// resources, and the mesh-service proxy mode.

Released as 1.23.0 on npm.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Alejandro Gutiérrez
2026-05-03 20:53:25 +01:00
parent 4eff4f5a20
commit c56910bfcf
31 changed files with 160 additions and 120 deletions

View File

@@ -348,6 +348,52 @@ export async function runVaultDelete(key: string, opts: Flags): Promise<number>
});
}
export interface VaultSetOpts extends Flags {
entryType?: "env" | "file";
mountPath?: string;
description?: string;
}
export async function runVaultSet(key: string, value: string, opts: VaultSetOpts): Promise<number> {
if (!key || value == null) {
render.err("Usage: claudemesh vault set <key> <value> [--type env|file] [--mount /path] [--description ...]");
return EXIT.INVALID_ARGS;
}
const { encryptFile, sealKeyForPeer } = await import("~/services/crypto/file-crypto.js");
const { getMeshConfig } = await import("~/services/config/facade.js");
const { readConfig } = await import("~/services/config/facade.js");
const config = readConfig();
const slug = opts.mesh ?? (config.meshes.length === 1 ? config.meshes[0]!.slug : null);
if (!slug) {
render.err("multiple meshes joined; pass --mesh <slug>");
return EXIT.INVALID_ARGS;
}
const mesh = getMeshConfig(slug);
if (!mesh) { render.err(`not joined to mesh "${slug}"`); return EXIT.NOT_FOUND; }
const plaintext = new TextEncoder().encode(value);
const enc = await encryptFile(plaintext);
const ciphertextB64 = Buffer.from(enc.ciphertext).toString("base64");
const sealed = await sealKeyForPeer(enc.key, mesh.pubkey);
return await withMesh({ meshSlug: slug }, async (client) => {
const ok = await client.vaultSet(
key,
ciphertextB64,
enc.nonce,
sealed,
opts.entryType ?? "env",
opts.mountPath,
opts.description,
);
if (opts.json) emitJson({ key, stored: ok });
else if (ok) render.ok(`vault[${bold(key)}] stored`, dim(`(${ciphertextB64.length}b)`));
else render.err(`vault set failed for "${key}"`);
return ok ? EXIT.SUCCESS : EXIT.IO_ERROR;
});
}
// ════════════════════════════════════════════════════════════════════════
// watch — URL change watchers
// ════════════════════════════════════════════════════════════════════════
@@ -368,6 +414,39 @@ export async function runWatchList(opts: Flags): Promise<number> {
});
}
export interface WatchAddOpts extends Flags {
label?: string;
interval?: number;
mode?: string;
extract?: string;
notifyOn?: string;
}
export async function runWatchAdd(url: string, opts: WatchAddOpts): Promise<number> {
if (!url) {
render.err("Usage: claudemesh watch add <url> [--label ...] [--interval <sec>] [--extract <css>] [--notify-on changed|always]");
return EXIT.INVALID_ARGS;
}
return await withMesh({ meshSlug: opts.mesh ?? null }, async (client) => {
const result = await client.watch(url, {
label: opts.label,
interval: opts.interval,
mode: opts.mode,
extract: opts.extract,
notify_on: opts.notifyOn,
});
if (result?.error) {
if (opts.json) emitJson({ ok: false, error: result.error });
else render.err(`watch add failed: ${result.error}`);
return EXIT.IO_ERROR;
}
const id = String((result as any)?.id ?? (result as any)?.watch_id ?? "?");
if (opts.json) emitJson({ ok: true, id, url, ...(opts.label ? { label: opts.label } : {}) });
else render.ok(`watching ${clay(url)}`, dim(id.slice(0, 8)));
return EXIT.SUCCESS;
});
}
export async function runUnwatch(id: string, opts: Flags): Promise<number> {
if (!id) { render.err("Usage: claudemesh watch remove <id>"); return EXIT.INVALID_ARGS; }
return await withMesh({ meshSlug: opts.mesh ?? null }, async (client) => {
@@ -397,6 +476,28 @@ export async function runWebhookList(opts: Flags): Promise<number> {
});
}
export async function runWebhookCreate(name: string, opts: Flags): Promise<number> {
if (!name) {
render.err("Usage: claudemesh webhook create <name>");
return EXIT.INVALID_ARGS;
}
return await withMesh({ meshSlug: opts.mesh ?? null }, async (client) => {
const created = await client.createWebhook(name);
if (!created) {
if (opts.json) emitJson({ ok: false, error: "create failed (timeout or duplicate)" });
else render.err(`webhook create "${name}" failed`);
return EXIT.IO_ERROR;
}
if (opts.json) emitJson({ ok: true, ...created });
else {
render.ok(`created webhook ${bold(created.name)}`);
process.stdout.write(` url: ${clay(created.url)}\n`);
process.stdout.write(` secret: ${dim(created.secret)} ${dim("(shown once)")}\n`);
}
return EXIT.SUCCESS;
});
}
export async function runWebhookDelete(name: string, opts: Flags): Promise<number> {
if (!name) { render.err("Usage: claudemesh webhook delete <name>"); return EXIT.INVALID_ARGS; }
return await withMesh({ meshSlug: opts.mesh ?? null }, async (client) => {