Replace the removed enterprise EE MCP (private apps/server/src/ee submodule,
license-gated /mcp route) with our docmost-mcp, vendored as an isolated ESM
workspace package and served by the server over HTTP — no enterprise license.
Backend:
- Add packages/mcp (@docmost/mcp): vendored docmost-mcp refactored into a
side-effect-free createDocmostMcpServer() factory (38 tools preserved),
stdio entry kept in stdio.ts, Streamable-HTTP session manager in http.ts.
- Add apps/server McpModule: @Post/@Get/@Delete('mcp') (served at /mcp via the
existing global-prefix exclude), @SkipTransform + reply.hijack to bridge raw
Fastify req/res into the SDK transport. The module dynamically imports the
ESM-only package from CommonJS via a Function-indirected import resolved with
require.resolve + file:// URL. Gated by the workspace ai.mcp toggle, a
service-account (MCP_DOCMOST_EMAIL/PASSWORD/API_URL) and optional MCP_TOKEN;
per-session idle eviction (MCP_SESSION_IDLE_MS).
- Drop the enterprise license check on mcpEnabled in workspace.service.
- Dockerfile: copy packages/mcp into the production image.
- .env.example: document MCP_DOCMOST_*, MCP_TOKEN, MCP_SESSION_IDLE_MS.
Frontend:
- Recreate the community "AI & MCP" workspace-settings panel (mcp-settings.tsx):
admin-only toggle on settings.ai.mcp with optimistic update, copyable
${APP_URL}/mcp URL; wired into workspace-settings page. Reuses existing i18n.
Fixes:
- Pin packages/mcp tiptap deps to 3.20.4 (matching the client) and inline
getStyleProperty, preventing a duplicate @tiptap/core@3.26.1 from leaking into
the client editor via pnpm shamefully-hoist (was breaking apps/client tsc).
4.9 KiB
Docmost MCP — Test Plan (editing & image tools)
Manual/E2E test plan for every content-mutating tool, with special focus on
images and image replacement. Executed against a live Docmost instance
(docs.vvzvlad.xyz) and verified visually in Chrome (public share + authenticated
editor).
How to run the automated part
DOCMOST_API_URL=https://<host>/api \
DOCMOST_EMAIL=<email> \
DOCMOST_PASSWORD=<password> \
node test-e2e.mjs
test-e2e.mjs creates a throwaway page, exercises every code path (including the
image upload/insert/replace cycle) and deletes the page afterwards. Collab writes
are debounced server-side, so the script waits ~16 s before reading back via REST.
Test matrix
| # | Tool / path | What is checked | Expected |
|---|---|---|---|
| 1 | create_page |
title with spaces, slugId returned | page created, title intact |
| 2 | update_page (markdown) |
headings, bold/italic/code/link, nested bullet + ordered lists, blockquote, code block, :::callout:::, table |
all structures survive re-import |
| 3 | get_page_json |
lossless ProseMirror, block ids, callout/table nodes | present (note: reads the debounced REST snapshot — recent collab writes may lag a few seconds) |
| 4 | edit_page_text |
surgical replace; block ids + marks preserved; ambiguous match rejected; missing match reported | edits applied, ids stable, errors correct |
| 5 | update_page_json |
full lossless write; custom block ids preserved; existing content (text edits, images, callout, table) not lost | round-trips intact |
| 6 | upload_image |
uploads attachment, returns node | src is a clean /api/files/<id>/<file> URL, served 200 image/* |
| 7 | insert_image (append / replaceText / afterText) |
three placements | image lands in the right place, all other block ids preserved |
| 8 | replace_image |
swap an existing figure for new bytes; comments/align/alt preserved; the new URL must actually serve the image | new image renders (200), old node repointed |
Image-specific assertions (the recurring bug area)
For every uploaded/inserted/replaced image, assert at the HTTP level that the
src actually serves bytes — this is what catches "broken image" regressions:
GET <src>→200,Content-Type: image/*, body starts with the image magic (89 50 4E 47for PNG, etc.).srcdoes not contain a?v=query (see "Known pitfalls").- After
replace_image: the returnednewAttachmentIddiffers from the old one (replacement uses a fresh attachment → fresh URL), andGET <new src>→200. - The old image node on the page is repointed to the new attachmentId.
Browser verification (Chrome)
Open the page (public /share/<key>/p/<slug> URL, or the authenticated editor)
and check each <img>:
[...document.querySelectorAll('.ProseMirror img')].map(im => ({
src: im.getAttribute('src'),
loaded: im.naturalWidth > 0, // 0 ⇒ broken
}));
loaded === true (naturalWidth > 0) means the image really rendered; 0 means a
broken/empty figure.
Known pitfalls (root-caused during testing)
-
In-place attachment overwrite corrupts the file (HTTP 500). Uploading with an existing
attachmentId(POST /files/upload+attachmentId) overwrites the bytes in place. On this Docmost the attachment then returns 500 for every URL (clean,?v=, any filename) → broken image. Thereforereplace_imagemust upload a new attachment and repoint the nodes; the new id yields a new URL that both renders and busts the browser cache. The old attachment is left as an unreferenced orphan: Docmost exposes no HTTP API to delete a single content attachment (verified against the attachment controller/service and by probing ~20 route variants live — all 404; an attachment unlinked from a page stays reachable with no auto-GC). Attachments are removed only by cascade (page/space/user deletion). This matches Docmost's own editor, which also orphans attachments on image removal/replacement. -
?v=<hash>cache-buster is unnecessary and was a red herring. The file endpoint serves…/file.png?v=<hash>exactly like the clean URL (200 image/*) — verified at the HTTP layer, on the public share, and in the authenticated editor. The broken images people saw came from pitfall #1, not from?v=. Imagesrcis kept clean (/api/files/<id>/<file>); cache-busting on replace is achieved by the new attachment id. -
REST snapshot lag.
get_page_jsonreads the debounced DB snapshot, so a write made moments earlier may not be visible yet. Wait (~16 s) before reading back, and never feed a possibly-stale snapshot straight intoupdate_page_json. -
Callout type narrowing (minor, open). A
:::warningcallout is imported astype: "info"— the markdown→callout conversion does not carry non-infotypes through. Cosmetic; tracked separately.