diff --git a/packages/prosemirror-dev-toolkit/src/history-and-diff/__snapshots__/createHistoryEntry.test.ts.snap b/packages/prosemirror-dev-toolkit/src/history-and-diff/__snapshots__/createHistoryEntry.test.ts.snap new file mode 100644 index 0000000..4935266 --- /dev/null +++ b/packages/prosemirror-dev-toolkit/src/history-and-diff/__snapshots__/createHistoryEntry.test.ts.snap @@ -0,0 +1,310 @@ +// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html + +exports[`createHistoryEntry > should collapse large src attributes 1`] = ` +{ + "contentDiff": { + "content": { + "0": { + "content": [ + [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + ], + }, + "1": [ + { + "content": [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + "type": "paragraph", + }, + ], + "2": [ + { + "type": "paragraph", + }, + ], + "_t": "a", + }, + }, + "id": "00", + "selectionDiff": { + "anchor": [ + 1, + 0, + ], + "empty": [ + true, + false, + ], + "from": [ + 1, + 0, + ], + "head": [ + 1, + 8, + ], + "to": [ + 1, + 8, + ], + }, + "selectionHtml": "<p> + <img src="..."> +</p> +<p> + <img src="..."> +</p> +<p></p>", + "state": { + "doc": { + "content": [ + { + "content": [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + "type": "paragraph", + }, + { + "content": [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + "type": "paragraph", + }, + { + "type": "paragraph", + }, + ], + "type": "doc", + }, + "selection": { + "type": "all", + }, + }, + "timeStr": "01:00:00:000", + "timestamp": 1706742000000, + "trs": [ + { + "curSelection": { + "type": "all", + }, + "curSelectionFor": 2, + "doc": { + "content": [ + { + "content": [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + "type": "paragraph", + }, + { + "content": [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + "type": "paragraph", + }, + { + "type": "paragraph", + }, + ], + "type": "doc", + }, + "docChanged": true, + "docs": [ + { + "content": [ + { + "type": "paragraph", + }, + ], + "type": "doc", + }, + { + "content": [ + { + "content": [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + "type": "paragraph", + }, + { + "type": "paragraph", + }, + ], + "type": "doc", + }, + ], + "isGeneric": true, + "mapping": Mapping { + "from": 0, + "maps": [ + StepMap { + "inverted": false, + "ranges": [ + 0, + 0, + 3, + ], + }, + StepMap { + "inverted": false, + "ranges": [ + 0, + 0, + 3, + ], + }, + ], + "mirror": undefined, + "to": 2, + }, + "meta": {}, + "scrolledIntoView": false, + "selectionSet": true, + "steps": [ + { + "from": 0, + "slice": { + "content": [ + { + "content": [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + "type": "paragraph", + }, + ], + }, + "stepType": "replace", + "to": 0, + }, + { + "from": 0, + "slice": { + "content": [ + { + "content": [ + { + "attrs": { + "alt": null, + "src": "", + "title": null, + }, + "type": "image", + }, + ], + "type": "paragraph", + }, + ], + }, + "stepType": "replace", + "to": 0, + }, + ], + "storedMarks": null, + "storedMarksSet": false, + "time": 1706742000000, + "updated": 1, + }, + ], +} +`; + +exports[`createHistoryEntry > should collapse large src attributes 2`] = ` + +
+

+ + +
+

+

+ + +
+

+

+
+

+
+ +`; diff --git a/packages/prosemirror-dev-toolkit/src/history-and-diff/createHistoryEntry.test.ts b/packages/prosemirror-dev-toolkit/src/history-and-diff/createHistoryEntry.test.ts new file mode 100644 index 0000000..c7f063e --- /dev/null +++ b/packages/prosemirror-dev-toolkit/src/history-and-diff/createHistoryEntry.test.ts @@ -0,0 +1,55 @@ +import { EditorView } from 'prosemirror-view' +import { beforeAll, describe, expect, it, vi } from 'vitest' + +import { createHistoryEntry } from './createHistoryEntry' +import { setupEditor } from '$test-utils/setupEditor' +import { AllSelection } from 'prosemirror-state' + +let view: EditorView + +describe('createHistoryEntry', () => { + beforeAll(() => { + const el = document.createElement('div') + document.body.appendChild(el) + el.id = 'pm-editor' + view = setupEditor(el) + vi.useFakeTimers() + const date = new Date(2024, 1, 1, 1) + vi.setSystemTime(date) + const oldMath = global.Math + vi.stubGlobal('Math', { + floor: oldMath.floor, + max: oldMath.max, + min: oldMath.min, + random: vi.fn(() => 0) + }) + }) + + it('should collapse base64 src attributes', () => { + const logSpy = vi.spyOn(console, 'log') + const infoSpy = vi.spyOn(console, 'info') + const warnSpy = vi.spyOn(console, 'warn') + const errorSpy = vi.spyOn(console, 'error') + + const startingState = view.state + // This will cause an infinite loop in 'html' package + const img1 = view.state.schema.nodes.image.create({ + src: '' + }) + const img2 = view.state.schema.nodes.image.create({ + src: '' + }) + const tr = view.state.tr + tr.insert(0, img1) + tr.insert(0, img2) + tr.setSelection(new AllSelection(tr.doc)) + view.dispatch(tr) + const entry = createHistoryEntry([tr], view.state, startingState) + expect(entry).toMatchSnapshot() + expect(logSpy).toHaveBeenCalledTimes(0) + expect(infoSpy).toHaveBeenCalledTimes(0) + expect(warnSpy).toHaveBeenCalledTimes(0) + expect(errorSpy).toHaveBeenCalledTimes(0) + expect(document.body).toMatchSnapshot() + }) +}) diff --git a/packages/prosemirror-dev-toolkit/src/history-and-diff/createHistoryEntry.ts b/packages/prosemirror-dev-toolkit/src/history-and-diff/createHistoryEntry.ts index 09b99c6..79064a0 100644 --- a/packages/prosemirror-dev-toolkit/src/history-and-diff/createHistoryEntry.ts +++ b/packages/prosemirror-dev-toolkit/src/history-and-diff/createHistoryEntry.ts @@ -36,12 +36,16 @@ const formatTimestamp = (timestamp: number) => { ].join(':') } -const regexp = /(<\/?[\w\d\s="']+>)/gim +// Matches any src attribute containing base64 data due to bug in html package and legibility +// https://github.com/TeemuKoivisto/prosemirror-dev-toolkit/issues/81 +const srcAttr = /src=[\"|\']data:(.*);base64,(.*)[\"|\']/g + +const wrappingCarets = /(<\/?[\w\d\s="']+>)/gim const highlightHtmlString = (html: string) => html .replace(//g, '>') - .replace(regexp, "$&") + .replace(wrappingCarets, "$&") export function createHistoryEntry( trs: readonly Transaction[], @@ -57,7 +61,7 @@ export function createHistoryEntry( if (domFragment) { let child = domFragment.firstChild as HTMLElement | null while (child) { - selectedElementsAsHtml.push(child.outerHTML) + selectedElementsAsHtml.push(child.outerHTML.replaceAll(srcAttr, 'src="..."')) child = child.nextSibling as HTMLElement | null } } diff --git a/packages/prosemirror-dev-toolkit/src/test-utils/schema.ts b/packages/prosemirror-dev-toolkit/src/test-utils/schema.ts index 5f0a870..c1974ab 100644 --- a/packages/prosemirror-dev-toolkit/src/test-utils/schema.ts +++ b/packages/prosemirror-dev-toolkit/src/test-utils/schema.ts @@ -1,10 +1,39 @@ -import { Schema } from 'prosemirror-model' +import { NodeSpec, Schema } from 'prosemirror-model' export const schema = new Schema({ nodes: { doc: { content: 'block+' }, + /// An inline image (``) node. Supports `src`, + /// `alt`, and `href` attributes. The latter two default to the empty + /// string. + image: { + inline: true, + attrs: { + src: { validate: 'string' }, + alt: { default: null, validate: 'string|null' }, + title: { default: null, validate: 'string|null' } + }, + group: 'inline', + draggable: true, + parseDOM: [ + { + tag: 'img[src]', + getAttrs(dom: HTMLElement) { + return { + src: dom.getAttribute('src'), + title: dom.getAttribute('title'), + alt: dom.getAttribute('alt') + } + } + } + ], + toDOM(node) { + const { src, alt, title } = node.attrs + return ['img', { src, alt, title }] + } + } as NodeSpec, paragraph: { content: 'inline*', group: 'block',