f720151c63
The same tool metadata (zod schema + model-facing description) was hand-duplicated between the standalone MCP server and the in-app AI-chat agent, so every tweak had to land in two places and copies drifted (a materialized parity bug). The shared transport-agnostic registry (packages/mcp/src/tool-specs.ts) already de-duplicates 14 tools; this migrates two more genuinely-identical ones — patch_node/patchNode and insert_node/insertNode. The canonical description is a strict SUPERSET of both originals (keeps MCP's "without resending the whole document" + table-structure/anchor guidance AND the in-app "reversible via page history" / "exactly one of anchorNodeId or anchorText" framing — no model-facing guidance dropped); the schema is identical (the in-app side just gains MCP's .min(1) on ids, a safe tightening). Each transport keeps its own execute/auth wrapper, and the in-app parseNodeArg node-arg normalization is unchanged. The three table tools are intentionally NOT merged (a real param-name divergence: table vs tableRef) — documented on both sides. Other per-transport divergences (search/share/create_comment/transform/list_pages) are left separate with a short comment explaining why (the issue asked to flag these as intentional). DocmostClientLike stays a hand-mirror (the ESM/CJS boundary blocks a compile-time type import; a runtime drift-guard already pins it). Also fixes a latent contract-spec bug: derive `required` from `instanceof z.ZodOptional` (matches the emitted JSON schema) instead of `isOptional()`, which wrongly reported z.any() fields as optional. Partially addresses #294. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
148 lines
6.1 KiB
JavaScript
148 lines
6.1 KiB
JavaScript
import { test } from "node:test";
|
|
import assert from "node:assert/strict";
|
|
import { z } from "zod";
|
|
|
|
import { SHARED_TOOL_SPECS } from "../../build/tool-specs.js";
|
|
|
|
// The shared registry is consumed by BOTH the zod-v3 MCP server and the zod-v4
|
|
// in-app AI-SDK service, so every spec must carry the cross-layer wiring
|
|
// (mcpName + inAppKey) and its builders must produce the right field set when
|
|
// called with a real zod namespace.
|
|
|
|
test("every spec exposes mcpName + inAppKey, and the key matches inAppKey", () => {
|
|
for (const [key, spec] of Object.entries(SHARED_TOOL_SPECS)) {
|
|
assert.equal(typeof spec.mcpName, "string");
|
|
assert.ok(spec.mcpName.length > 0, `${key}: empty mcpName`);
|
|
assert.equal(typeof spec.inAppKey, "string");
|
|
assert.ok(spec.inAppKey.length > 0, `${key}: empty inAppKey`);
|
|
assert.equal(typeof spec.description, "string");
|
|
assert.ok(spec.description.length > 0, `${key}: empty description`);
|
|
// The registry is keyed by inAppKey — keep the two in sync.
|
|
assert.equal(spec.inAppKey, key, `${key}: registry key must equal inAppKey`);
|
|
}
|
|
});
|
|
|
|
test("mcpName uses snake_case and inAppKey uses camelCase", () => {
|
|
for (const [key, spec] of Object.entries(SHARED_TOOL_SPECS)) {
|
|
assert.match(spec.mcpName, /^[a-z0-9]+(_[a-z0-9]+)*$/, `${key}: mcpName not snake_case`);
|
|
assert.match(spec.inAppKey, /^[a-z][a-zA-Z0-9]*$/, `${key}: inAppKey not camelCase`);
|
|
}
|
|
});
|
|
|
|
test("mcpName and inAppKey are each unique across the registry", () => {
|
|
const mcpNames = new Set();
|
|
const inAppKeys = new Set();
|
|
for (const spec of Object.values(SHARED_TOOL_SPECS)) {
|
|
assert.ok(!mcpNames.has(spec.mcpName), `duplicate mcpName: ${spec.mcpName}`);
|
|
assert.ok(!inAppKeys.has(spec.inAppKey), `duplicate inAppKey: ${spec.inAppKey}`);
|
|
mcpNames.add(spec.mcpName);
|
|
inAppKeys.add(spec.inAppKey);
|
|
}
|
|
});
|
|
|
|
test("buildShape (when present) returns a usable ZodRawShape with a real zod", () => {
|
|
for (const [key, spec] of Object.entries(SHARED_TOOL_SPECS)) {
|
|
if (!spec.buildShape) continue;
|
|
const shape = spec.buildShape(z);
|
|
assert.equal(typeof shape, "object");
|
|
// Each field must be a real zod type so z.object(shape) compiles a schema.
|
|
for (const [field, zt] of Object.entries(shape)) {
|
|
assert.ok(
|
|
zt && typeof zt.parse === "function",
|
|
`${key}.${field}: not a zod type`,
|
|
);
|
|
}
|
|
// The compiled object schema must parse a minimal valid input.
|
|
assert.doesNotThrow(() => z.object(shape));
|
|
}
|
|
});
|
|
|
|
test("editPageText builder produces { pageId, edits } and drops the stale strip-and-retry claim", () => {
|
|
const spec = SHARED_TOOL_SPECS.editPageText;
|
|
assert.equal(spec.mcpName, "edit_page_text");
|
|
const shape = spec.buildShape(z);
|
|
assert.deepEqual(Object.keys(shape).sort(), ["edits", "pageId"]);
|
|
// A valid edits batch parses.
|
|
const schema = z.object(shape);
|
|
const parsed = schema.parse({
|
|
pageId: "p1",
|
|
edits: [{ find: "teh", replace: "the" }],
|
|
});
|
|
assert.equal(parsed.pageId, "p1");
|
|
assert.equal(parsed.edits.length, 1);
|
|
// The canonical description must NOT carry the stale MCP strip-and-retry claim.
|
|
assert.ok(
|
|
!/strip-and-retry/i.test(spec.description),
|
|
"editPageText description still claims strip-and-retry",
|
|
);
|
|
assert.match(spec.description, /REFUSED into\s+failed\[\]/);
|
|
});
|
|
|
|
test("getNode builder produces exactly { pageId, nodeId }", () => {
|
|
const shape = SHARED_TOOL_SPECS.getNode.buildShape(z);
|
|
assert.deepEqual(Object.keys(shape).sort(), ["nodeId", "pageId"]);
|
|
});
|
|
|
|
test("patchNode spec exists, merges BOTH descriptions, builds { pageId, nodeId, node }", () => {
|
|
const spec = SHARED_TOOL_SPECS.patchNode;
|
|
assert.ok(spec, "patchNode spec missing");
|
|
assert.equal(spec.mcpName, "patch_node");
|
|
assert.equal(spec.inAppKey, "patchNode");
|
|
|
|
// The canonical description must carry the key guidance from BOTH originals:
|
|
// - MCP-only: "WITHOUT resending the whole document" + the cheaper/safer note.
|
|
// - in-app-only: "keeps the same node id" + the "Reversible ... page history"
|
|
// framing the MCP copy lacked.
|
|
assert.match(spec.description, /WITHOUT resending the whole document/);
|
|
assert.match(spec.description, /Cheaper and safer/);
|
|
assert.match(spec.description, /keeps the same node id/i);
|
|
assert.match(spec.description, /Reversible/i);
|
|
assert.match(spec.description, /page history/i);
|
|
|
|
const shape = spec.buildShape(z);
|
|
assert.deepEqual(Object.keys(shape).sort(), ["node", "nodeId", "pageId"]);
|
|
// A minimal valid input parses (node accepts an arbitrary object via z.any()).
|
|
const parsed = z.object(shape).parse({
|
|
pageId: "p1",
|
|
nodeId: "n1",
|
|
node: { type: "paragraph" },
|
|
});
|
|
assert.equal(parsed.pageId, "p1");
|
|
assert.equal(parsed.nodeId, "n1");
|
|
});
|
|
|
|
test("insertNode spec exists, merges BOTH descriptions, builds the full anchor shape", () => {
|
|
const spec = SHARED_TOOL_SPECS.insertNode;
|
|
assert.ok(spec, "insertNode spec missing");
|
|
assert.equal(spec.mcpName, "insert_node");
|
|
assert.equal(spec.inAppKey, "insertNode");
|
|
|
|
// Canonical description must keep BOTH sides' nuance:
|
|
// - in-app-only: "EXACTLY ONE of anchorNodeId or anchorText" + "Reversible".
|
|
// - MCP-only: the table-structure (tableRow/tableCell) insertion guidance.
|
|
assert.match(spec.description, /EXACTLY ONE of anchorNodeId or anchorText/);
|
|
assert.match(spec.description, /tableRow/);
|
|
assert.match(spec.description, /append is top-level only/);
|
|
assert.match(spec.description, /Reversible via page history/);
|
|
|
|
const shape = spec.buildShape(z);
|
|
assert.deepEqual(
|
|
Object.keys(shape).sort(),
|
|
["anchorNodeId", "anchorText", "node", "pageId", "position"],
|
|
);
|
|
// before/after/append are the only accepted positions; anchors are optional.
|
|
const schema = z.object(shape);
|
|
assert.doesNotThrow(() =>
|
|
schema.parse({ pageId: "p1", node: { type: "paragraph" }, position: "append" }),
|
|
);
|
|
assert.throws(() =>
|
|
schema.parse({ pageId: "p1", node: {}, position: "sideways" }),
|
|
);
|
|
});
|
|
|
|
test("no-arg specs (getWorkspace/listSpaces/listShares) omit buildShape", () => {
|
|
for (const key of ["getWorkspace", "listSpaces", "listShares"]) {
|
|
assert.equal(SHARED_TOOL_SPECS[key].buildShape, undefined, `${key} should be no-arg`);
|
|
}
|
|
});
|