import { test } from "node:test"; import assert from "node:assert/strict"; import { docmostExtensions, clampCalloutType, } from "../../build/lib/docmost-schema.js"; import { TiptapTransformer } from "@hocuspocus/transformer"; test("clampCalloutType: a known type passes through", () => { assert.equal(clampCalloutType("warning"), "warning"); }); test("clampCalloutType: an uppercase known type folds to lower case", () => { assert.equal(clampCalloutType("WARNING"), "warning"); assert.equal(clampCalloutType("Info"), "info"); }); test("clampCalloutType: an unknown type falls back to info", () => { assert.equal(clampCalloutType("bogus"), "info"); }); test("clampCalloutType: null and undefined fall back to info", () => { assert.equal(clampCalloutType(null), "info"); assert.equal(clampCalloutType(undefined), "info"); }); // Minimal-doc builders for the toYdoc acceptance loop. const text = (t) => ({ type: "text", text: t }); const paragraph = (inline) => ({ type: "paragraph", content: inline }); const docOf = (...content) => ({ type: "doc", content }); // Each entry is a minimal valid doc for one Docmost node type. Inline atoms // (mention, mathInline) and inline-capable nodes go inside a paragraph; block // atoms and block containers go at the top level. const cases = { mention: docOf( paragraph([{ type: "mention", attrs: { id: "u1", label: "Bob" } }]), ), mathInline: docOf(paragraph([{ type: "mathInline", attrs: { text: "x^2" } }])), mathBlock: docOf({ type: "mathBlock", attrs: { text: "x^2" } }), details: docOf({ type: "details", content: [ { type: "detailsSummary", content: [text("Summary")] }, { type: "detailsContent", content: [paragraph([text("body")])] }, ], }), attachment: docOf({ type: "attachment", attrs: { url: "http://x/f.zip", name: "f.zip" }, }), video: docOf({ type: "video", attrs: { src: "http://x/v.mp4" } }), youtube: docOf({ type: "youtube", attrs: { src: "http://y/watch" } }), embed: docOf({ type: "embed", attrs: { src: "http://e", provider: "iframe" } }), htmlEmbed: docOf({ type: "htmlEmbed", attrs: { source: "", height: 320 }, }), drawio: docOf({ type: "drawio", attrs: { src: "http://d" } }), excalidraw: docOf({ type: "excalidraw", attrs: { src: "http://e" } }), columns: docOf({ type: "columns", content: [ { type: "column", content: [paragraph([text("c1")])] }, { type: "column", content: [paragraph([text("c2")])] }, ], }), subpages: docOf({ type: "subpages" }), audio: docOf({ type: "audio", attrs: { src: "http://a.mp3" } }), pdf: docOf({ type: "pdf", attrs: { src: "http://p.pdf" } }), pageBreak: docOf({ type: "pageBreak" }), }; for (const [name, doc] of Object.entries(cases)) { test(`toYdoc accepts a ${name} node without throwing`, () => { assert.doesNotThrow(() => { TiptapTransformer.toYdoc(doc, "default", docmostExtensions); }); }); } // htmlEmbed is the sandboxed raw-HTML block. The MCP write path carries it // through Yjs (toYdoc -> fromYdoc) without rendering, so a full round-trip must // preserve both the `source` snippet and the numeric `height`. test("htmlEmbed round-trips source and height through Yjs", () => { const doc = docOf({ type: "htmlEmbed", attrs: { source: "", height: 480 }, }); const ydoc = TiptapTransformer.toYdoc(doc, "default", docmostExtensions); const back = TiptapTransformer.fromYdoc(ydoc, "default"); const node = back.content.find((n) => n.type === "htmlEmbed"); assert.ok(node, "htmlEmbed node survives the round-trip"); assert.equal(node.attrs.source, ""); assert.equal(node.attrs.height, 480); });