diff --git a/.changeset/red-peaches-explode.md b/.changeset/red-peaches-explode.md new file mode 100644 index 0000000000..1792341031 --- /dev/null +++ b/.changeset/red-peaches-explode.md @@ -0,0 +1,5 @@ +--- +"rrweb": patch +--- + +This fixes an issue where inlined CSS from a remotely loaded `` does not get applied properly due to object reference mutation. diff --git a/.github/workflows/ci-cd.yml b/.github/workflows/ci-cd.yml index b65caf96cd..06d4169baf 100644 --- a/.github/workflows/ci-cd.yml +++ b/.github/workflows/ci-cd.yml @@ -7,7 +7,7 @@ concurrency: ${{ github.workflow }}-${{ github.ref }} jobs: release: name: Tests - runs-on: ubuntu-latest + runs-on: ubuntu-22.04 steps: - name: Checkout Repo uses: actions/checkout@v3 @@ -19,7 +19,7 @@ jobs: uses: actions/setup-node@v3 with: node-version: lts/* - + - name: Install Dependencies run: yarn install --frozen-lockfile diff --git a/packages/rrweb/src/replay/index.ts b/packages/rrweb/src/replay/index.ts index 5dacdc104d..a80e1de592 100644 --- a/packages/rrweb/src/replay/index.ts +++ b/packages/rrweb/src/replay/index.ts @@ -1792,17 +1792,22 @@ export class Replayer { const newSn = mirror.getMeta( target as Node & RRNode, ) as serializedElementNodeWithId; - Object.assign( - newSn.attributes, - mutation.attributes as attributes, + const newNode = buildNodeWithSN( + { + ...newSn, + attributes: { + ...newSn.attributes, + ...(mutation.attributes as attributes), + }, + }, + { + doc: target.ownerDocument as Document, // can be Document or RRDocument + mirror: mirror as Mirror, + skipChild: true, + hackCss: true, + cache: this.cache, + }, ); - const newNode = buildNodeWithSN(newSn, { - doc: target.ownerDocument as Document, // can be Document or RRDocument - mirror: mirror as Mirror, - skipChild: true, - hackCss: true, - cache: this.cache, - }); const siblingNode = target.nextSibling; const parentNode = target.parentNode; if (newNode && parentNode) { diff --git a/packages/rrweb/test/__snapshots__/replayer.test.ts.snap b/packages/rrweb/test/__snapshots__/replayer.test.ts.snap index c4edd6d540..1682d09ed5 100644 --- a/packages/rrweb/test/__snapshots__/replayer.test.ts.snap +++ b/packages/rrweb/test/__snapshots__/replayer.test.ts.snap @@ -158,6 +158,73 @@ file-cid-3 " `; +exports[`replayer > can handle remote stylesheets 1`] = ` +"file-frame-2 + + + + + +
+
+ +
+ + + + +file-frame-3 + + + + + + + + + + +file-cid-0 +@charset \\"utf-8\\"; + +.rr-block { background: currentcolor; } + +noscript { display: none !important; } + +html.rrweb-paused *, html.rrweb-paused ::before, html.rrweb-paused ::after { animation-play-state: paused !important; } + + +file-cid-1 +@charset \\"utf-8\\"; + +.OverlayDrawer-modal-187 { } + +.OverlayDrawer-paper-188 { width: 100%; } + +@media (min-width: 48em) { + .OverlayDrawer-paper-188 { width: 38rem; } +} + +@media (min-width: 48em) { +} + +@media (min-width: 48em) { +} +" +`; + exports[`replayer > can handle removing style elements 1`] = ` "file-frame-1 diff --git a/packages/rrweb/test/replayer.test.ts b/packages/rrweb/test/replayer.test.ts index 96e2012b15..c38ec356da 100644 --- a/packages/rrweb/test/replayer.test.ts +++ b/packages/rrweb/test/replayer.test.ts @@ -8,6 +8,7 @@ import { launchPuppeteer, sampleEvents as events, sampleStyleSheetRemoveEvents as stylesheetRemoveEvents, + sampleRemoteStyleSheetEvents as remoteStyleSheetEvents, waitForRAF, } from './utils'; import styleSheetRuleEvents from './events/style-sheet-rule-events'; @@ -209,6 +210,23 @@ describe('replayer', function () { await assertDomSnapshot(page); }); + it('can handle remote stylesheets', async () => { + await page.evaluate(`events = ${JSON.stringify(remoteStyleSheetEvents)}`); + const actionLength = await page.evaluate(` + const { Replayer } = rrweb; + const replayer = new Replayer(events); + replayer.play(2500); + replayer['timer']['actions'].length; + `); + expect(actionLength).toEqual( + remoteStyleSheetEvents.filter( + (e) => e.timestamp - remoteStyleSheetEvents[0].timestamp >= 2500, + ).length, + ); + + await assertDomSnapshot(page); + }); + it('can fast forward selection events', async () => { await page.evaluate(`events = ${JSON.stringify(selectionEvents)}`); diff --git a/packages/rrweb/test/utils.ts b/packages/rrweb/test/utils.ts index 50b39ccb99..9699543a4a 100644 --- a/packages/rrweb/test/utils.ts +++ b/packages/rrweb/test/utils.ts @@ -561,6 +561,98 @@ export const sampleStyleSheetRemoveEvents: eventWithTime[] = [ }, ]; +export const sampleRemoteStyleSheetEvents: eventWithTime[] = [ + { + type: EventType.DomContentLoaded, + data: {}, + timestamp: now, + }, + { + type: EventType.Load, + data: {}, + timestamp: now + 1000, + }, + { + type: EventType.Meta, + data: { + href: 'http://localhost', + width: 1000, + height: 800, + }, + timestamp: now + 1000, + }, + { + type: EventType.FullSnapshot, + data: { + node: { + type: 0, + childNodes: [ + { + type: 2, + tagName: 'html', + attributes: {}, + childNodes: [ + { + type: 2, + tagName: 'head', + attributes: {}, + childNodes: [ + { + type: 2, + tagName: 'link', + attributes: { + rel: 'stylesheet', + href: '', + }, + childNodes: [], + id: 4, + }, + ], + id: 3, + }, + { + type: 2, + tagName: 'body', + attributes: {}, + childNodes: [], + id: 6, + }, + ], + id: 2, + }, + ], + id: 1, + }, + initialOffset: { + top: 0, + left: 0, + }, + }, + timestamp: now + 1000, + }, + { + type: EventType.IncrementalSnapshot, + data: { + source: IncrementalSource.Mutation, + texts: [], + attributes: [ + { + id: 4, + attributes: { + href: null, + rel: null, + _cssText: + '.OverlayDrawer-modal-187 { }.OverlayDrawer-paper-188 { width: 100%; }@media (min-width: 48em) {\n .OverlayDrawer-paper-188 { width: 38rem; }\n}@media (min-width: 48em) {\n}@media (min-width: 48em) {\n}', + }, + }, + ], + removes: [], + adds: [], + }, + timestamp: now + 2000, + }, +]; + export const polyfillWebGLGlobals = () => { // polyfill as jsdom does not have support for these classes // consider replacing with https://www.npmjs.com/package/canvas