d6d7dd82f6
Create @docmost/prosemirror-markdown — the single framework-free ProseMirror<-> Markdown converter + schema mirror that git-sync and mcp will both consume, ending the three-hand-synced-copies drift (#293). This step only CREATES the package (no consumer yet; git-sync untouched); the switch of git-sync and mcp onto it, plus the canonical format decisions, come in later commits of this PR. - packages/prosemirror-markdown/src/lib/: the 8 converter-core files copied VERBATIM from packages/git-sync/src/lib (docmost-schema, markdown-converter, markdown-to-prosemirror, canonicalize, markdown-document, node-ops, page-file, index). Confirmed byte-identical — no behavioral drift introduced. - src/index.ts barrel; package.json (@tiptap/* + jsdom/marked/zod, editor-ext workspace devDep for the contract test); tsconfig/vitest configs. - 24 converter-core test files + fixtures copied (engine-coupled layout/ redteam-layout-title tests correctly excluded — they import ../src/engine). - pnpm-lock importer added; build/ gitignored (CI-built). Verified (clean checkout, no network): pnpm --frozen-lockfile EXIT 0; tsc EXIT 0; vitest 23 files, 443 passed | 1 expected-fail (the same image-diagrams known-limitation carried from git-sync) — faithful extraction. git-sync untouched. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
88 lines
4.3 KiB
TypeScript
88 lines
4.3 KiB
TypeScript
import { describe, it, expect } from "vitest";
|
|
import { getSchema } from "@tiptap/core";
|
|
|
|
import { docmostExtensions } from "../src/lib/docmost-schema.js";
|
|
import * as editorExt from "@docmost/editor-ext";
|
|
|
|
// CROSS-PACKAGE SCHEMA CONTRACT (data-loss-sensitive).
|
|
//
|
|
// `src/lib/docmost-schema.ts` is a hand-synced VENDORED MIRROR of the canonical
|
|
// Docmost schema in `@docmost/editor-ext`. The sibling `schema-surface-snapshot`
|
|
// test pins the mirror's FULL surface (names + attrs) against an inline
|
|
// reference, but that reference is hand-curated and does not mechanically tie to
|
|
// editor-ext. This test closes that gap from the other side: it reads the ACTUAL
|
|
// Tiptap node/mark definitions exported by `@docmost/editor-ext` and asserts the
|
|
// vendored mirror is a SUPERSET of their type NAMES — so a Docmost-specific node
|
|
// or mark added upstream that the mirror forgets to vendor fails CI loudly
|
|
// (otherwise it is silently dropped on the markdown <-> ProseMirror round-trip).
|
|
//
|
|
// LIMITATION (intentional, see schema-surface-snapshot.test.ts): this is a
|
|
// NAME-LEVEL contract only, not a full attribute-level structural compare.
|
|
// editor-ext's Tiptap representation (node views, commands, suggestion plugins,
|
|
// addGlobalAttributes spread across separate extensions) differs from this
|
|
// minimal mirror, so a mechanical attribute-by-attribute equality would be
|
|
// fragile and produce false drift. Attribute parity is guarded by the inline
|
|
// surface snapshot (reviewed in every diff); this test guards that no canonical
|
|
// node/mark TYPE goes unmirrored. StarterKit-provided types (paragraph, bold,
|
|
// heading, …) are contributed by @tiptap/starter-kit in the mirror rather than
|
|
// by editor-ext, so they are naturally covered by the mirror's superset.
|
|
//
|
|
// NOT COVERED here (deferred): (1) the THIRD copy in `packages/mcp` — a separate
|
|
// package guarded by its own surface snapshot; (2) attribute *behaviour* drift,
|
|
// e.g. the details `open` attr read via getAttribute vs hasAttribute (PR #119
|
|
// review #2) — a name-level compare cannot see parseHTML/renderHTML differences.
|
|
// Mechanically guarding behavioural parity across all THREE copies needs the
|
|
// single framework-free "schema core" refactor (deferred — see AGENTS.md); until
|
|
// then each copy's header carries the manual keep-in-sync requirement.
|
|
|
|
/** Tiptap Node/Mark instances expose a `.name` and a `.type` of 'node'|'mark'. */
|
|
function isTiptapNodeOrMark(
|
|
value: unknown,
|
|
): value is { name: string; type: "node" | "mark" } {
|
|
return (
|
|
typeof value === "object" &&
|
|
value !== null &&
|
|
"name" in value &&
|
|
typeof (value as { name: unknown }).name === "string" &&
|
|
"type" in value &&
|
|
((value as { type: unknown }).type === "node" ||
|
|
(value as { type: unknown }).type === "mark")
|
|
);
|
|
}
|
|
|
|
/** The set of node/mark type names the vendored mirror actually registers. */
|
|
function vendoredNames(): Set<string> {
|
|
const schema = getSchema(docmostExtensions as never);
|
|
return new Set([
|
|
...Object.keys(schema.nodes),
|
|
...Object.keys(schema.marks),
|
|
]);
|
|
}
|
|
|
|
/** The Docmost-specific node/mark type names exported by @docmost/editor-ext. */
|
|
function editorExtNames(): Set<string> {
|
|
const names = new Set<string>();
|
|
for (const value of Object.values(editorExt)) {
|
|
if (isTiptapNodeOrMark(value)) names.add(value.name);
|
|
}
|
|
return names;
|
|
}
|
|
|
|
describe("docmost schema vs @docmost/editor-ext (name-level contract)", () => {
|
|
it("exposes Tiptap node/mark definitions from editor-ext (guards against the import going dark)", () => {
|
|
// If editor-ext ever stops exporting concrete node/mark objects (e.g. a
|
|
// barrel refactor), this contract would vacuously pass — assert it found a
|
|
// meaningful set so the test cannot silently become a no-op.
|
|
expect(editorExtNames().size).toBeGreaterThan(5);
|
|
});
|
|
|
|
it("vendors every Docmost-specific node/mark type defined in editor-ext (no silently-dropped types)", () => {
|
|
const vendored = vendoredNames();
|
|
const missing = [...editorExtNames()].filter((n) => !vendored.has(n)).sort();
|
|
// missing must be empty: any name here exists in editor-ext but NOT in the
|
|
// vendored mirror, so documents using it would lose that node/mark on a
|
|
// git-sync round-trip. Re-sync src/lib/docmost-schema.ts before clearing.
|
|
expect(missing).toEqual([]);
|
|
});
|
|
});
|