2021-03-01 23:20:04 +03:00
|
|
|
/**
|
|
|
|
* Copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
import { contextTest, expect } from './config/browserTest';
|
2021-03-01 23:20:04 +03:00
|
|
|
import { InMemorySnapshotter } from '../lib/server/snapshot/inMemorySnapshotter';
|
2021-03-09 06:49:57 +03:00
|
|
|
import { HttpServer } from '../lib/utils/httpServer';
|
|
|
|
import { SnapshotServer } from '../lib/server/snapshot/snapshotServer';
|
2021-03-01 23:20:04 +03:00
|
|
|
|
2021-05-17 05:58:26 +03:00
|
|
|
const it = contextTest.extend<{ snapshotPort: number, snapshotter: InMemorySnapshotter }>({
|
|
|
|
snapshotPort: async ({}, run, testInfo) => {
|
|
|
|
await run(11000 + testInfo.workerIndex);
|
2021-05-10 03:47:20 +03:00
|
|
|
},
|
|
|
|
|
2021-05-17 05:58:26 +03:00
|
|
|
snapshotter: async ({ mode, toImpl, context, snapshotPort }, run, testInfo) => {
|
|
|
|
if (mode !== 'default')
|
|
|
|
testInfo.skip();
|
|
|
|
const snapshotter = new InMemorySnapshotter(toImpl(context));
|
|
|
|
await snapshotter.initialize();
|
|
|
|
const httpServer = new HttpServer();
|
|
|
|
new SnapshotServer(httpServer, snapshotter);
|
|
|
|
await httpServer.start(snapshotPort);
|
|
|
|
await run(snapshotter);
|
|
|
|
await snapshotter.dispose();
|
|
|
|
await httpServer.stop();
|
2021-05-10 03:47:20 +03:00
|
|
|
},
|
|
|
|
});
|
2021-04-01 23:18:04 +03:00
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
it.describe('snapshots', () => {
|
|
|
|
it('should collect snapshot', async ({ page, toImpl, snapshotter }) => {
|
2021-03-01 23:20:04 +03:00
|
|
|
await page.setContent('<button>Hello</button>');
|
|
|
|
const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot');
|
|
|
|
expect(distillSnapshot(snapshot)).toBe('<BUTTON>Hello</BUTTON>');
|
|
|
|
});
|
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
it('should capture resources', async ({ page, toImpl, server, snapshotter }) => {
|
2021-03-01 23:20:04 +03:00
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await page.route('**/style.css', route => {
|
|
|
|
route.fulfill({ body: 'button { color: red; }', }).catch(() => {});
|
|
|
|
});
|
|
|
|
await page.setContent('<link rel="stylesheet" href="style.css"><button>Hello</button>');
|
|
|
|
const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot');
|
|
|
|
const { resources } = snapshot.render();
|
|
|
|
const cssHref = `http://localhost:${server.PORT}/style.css`;
|
|
|
|
expect(resources[cssHref]).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
it('should collect multiple', async ({ page, toImpl, snapshotter }) => {
|
2021-03-01 23:20:04 +03:00
|
|
|
await page.setContent('<button>Hello</button>');
|
|
|
|
const snapshots = [];
|
|
|
|
snapshotter.on('snapshot', snapshot => snapshots.push(snapshot));
|
|
|
|
await snapshotter.captureSnapshot(toImpl(page), 'snapshot1');
|
|
|
|
await snapshotter.captureSnapshot(toImpl(page), 'snapshot2');
|
|
|
|
expect(snapshots.length).toBe(2);
|
|
|
|
});
|
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
it('should respect inline CSSOM change', async ({ page, toImpl, snapshotter }) => {
|
2021-03-01 23:20:04 +03:00
|
|
|
await page.setContent('<style>button { color: red; }</style><button>Hello</button>');
|
2021-04-27 21:07:07 +03:00
|
|
|
const snapshot1 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot1');
|
|
|
|
expect(distillSnapshot(snapshot1)).toBe('<style>button { color: red; }</style><BUTTON>Hello</BUTTON>');
|
|
|
|
|
|
|
|
await page.evaluate(() => { (document.styleSheets[0].cssRules[0] as any).style.color = 'blue'; });
|
|
|
|
const snapshot2 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot2');
|
|
|
|
expect(distillSnapshot(snapshot2)).toBe('<style>button { color: blue; }</style><BUTTON>Hello</BUTTON>');
|
2021-03-01 23:20:04 +03:00
|
|
|
});
|
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
it('should respect subresource CSSOM change', async ({ page, server, toImpl, snapshotter }) => {
|
2021-03-01 23:20:04 +03:00
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await page.route('**/style.css', route => {
|
|
|
|
route.fulfill({ body: 'button { color: red; }', }).catch(() => {});
|
|
|
|
});
|
|
|
|
await page.setContent('<link rel="stylesheet" href="style.css"><button>Hello</button>');
|
|
|
|
|
2021-04-27 21:07:07 +03:00
|
|
|
const snapshot1 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot1');
|
|
|
|
expect(distillSnapshot(snapshot1)).toBe('<LINK rel=\"stylesheet\" href=\"style.css\"><BUTTON>Hello</BUTTON>');
|
|
|
|
|
|
|
|
await page.evaluate(() => { (document.styleSheets[0].cssRules[0] as any).style.color = 'blue'; });
|
|
|
|
const snapshot2 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot1');
|
|
|
|
const { resources } = snapshot2.render();
|
2021-03-01 23:20:04 +03:00
|
|
|
const cssHref = `http://localhost:${server.PORT}/style.css`;
|
|
|
|
const { sha1 } = resources[cssHref];
|
|
|
|
expect(snapshotter.resourceContent(sha1).toString()).toBe('button { color: blue; }');
|
|
|
|
});
|
2021-03-09 06:49:57 +03:00
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
it('should capture iframe', async ({ page, contextFactory, server, toImpl, browserName, snapshotter, snapshotPort }) => {
|
2021-04-05 19:18:56 +03:00
|
|
|
it.skip(browserName === 'firefox');
|
|
|
|
|
2021-03-09 06:49:57 +03:00
|
|
|
await page.route('**/empty.html', route => {
|
|
|
|
route.fulfill({
|
|
|
|
body: '<iframe src="iframe.html"></iframe>',
|
|
|
|
contentType: 'text/html'
|
|
|
|
}).catch(() => {});
|
|
|
|
});
|
|
|
|
await page.route('**/iframe.html', route => {
|
|
|
|
route.fulfill({
|
|
|
|
body: '<html><button>Hello iframe</button></html>',
|
|
|
|
contentType: 'text/html'
|
|
|
|
}).catch(() => {});
|
|
|
|
});
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
|
|
|
|
// Marking iframe hierarchy is racy, do not expect snapshot, wait for it.
|
|
|
|
let counter = 0;
|
|
|
|
let snapshot: any;
|
|
|
|
for (; ; ++counter) {
|
|
|
|
snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot' + counter);
|
|
|
|
const text = distillSnapshot(snapshot).replace(/frame@[^"]+["]/, '<id>"');
|
|
|
|
if (text === '<IFRAME src=\"/snapshot/<id>\"></IFRAME>')
|
|
|
|
break;
|
|
|
|
await page.waitForTimeout(250);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Render snapshot, check expectations.
|
|
|
|
const previewContext = await contextFactory();
|
|
|
|
const previewPage = await previewContext.newPage();
|
|
|
|
await previewPage.goto(`http://localhost:${snapshotPort}/snapshot/`);
|
|
|
|
await previewPage.evaluate(snapshotId => {
|
|
|
|
(window as any).showSnapshot(snapshotId);
|
|
|
|
}, `${snapshot.snapshot().pageId}?name=snapshot${counter}`);
|
2021-05-12 22:21:54 +03:00
|
|
|
while (previewPage.frames().length < 3)
|
2021-03-09 06:49:57 +03:00
|
|
|
await new Promise(f => previewPage.once('frameattached', f));
|
2021-05-12 22:21:54 +03:00
|
|
|
const button = await previewPage.frames()[2].waitForSelector('button');
|
2021-03-09 06:49:57 +03:00
|
|
|
expect(await button.textContent()).toBe('Hello iframe');
|
|
|
|
});
|
2021-03-10 22:43:26 +03:00
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
it('should capture snapshot target', async ({ page, toImpl, snapshotter }) => {
|
2021-03-10 22:43:26 +03:00
|
|
|
await page.setContent('<button>Hello</button><button>World</button>');
|
|
|
|
{
|
|
|
|
const handle = await page.$('text=Hello');
|
|
|
|
const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot', toImpl(handle));
|
|
|
|
expect(distillSnapshot(snapshot)).toBe('<BUTTON __playwright_target__=\"snapshot\">Hello</BUTTON><BUTTON>World</BUTTON>');
|
|
|
|
}
|
|
|
|
{
|
|
|
|
const handle = await page.$('text=World');
|
|
|
|
const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot2', toImpl(handle));
|
|
|
|
expect(distillSnapshot(snapshot)).toBe('<BUTTON __playwright_target__=\"snapshot\">Hello</BUTTON><BUTTON __playwright_target__=\"snapshot2\">World</BUTTON>');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2021-05-10 03:47:20 +03:00
|
|
|
it('should collect on attribute change', async ({ page, toImpl, snapshotter }) => {
|
2021-03-10 22:43:26 +03:00
|
|
|
await page.setContent('<button>Hello</button>');
|
|
|
|
{
|
|
|
|
const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot');
|
|
|
|
expect(distillSnapshot(snapshot)).toBe('<BUTTON>Hello</BUTTON>');
|
|
|
|
}
|
|
|
|
const handle = await page.$('text=Hello')!;
|
|
|
|
await handle.evaluate(element => element.setAttribute('data', 'one'));
|
|
|
|
{
|
|
|
|
const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot2');
|
|
|
|
expect(distillSnapshot(snapshot)).toBe('<BUTTON data="one">Hello</BUTTON>');
|
|
|
|
}
|
|
|
|
await handle.evaluate(element => element.setAttribute('data', 'two'));
|
|
|
|
{
|
|
|
|
const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot2');
|
|
|
|
expect(distillSnapshot(snapshot)).toBe('<BUTTON data="two">Hello</BUTTON>');
|
|
|
|
}
|
|
|
|
});
|
2021-03-01 23:20:04 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
function distillSnapshot(snapshot) {
|
|
|
|
const { html } = snapshot.render();
|
|
|
|
return html
|
2021-03-09 06:49:57 +03:00
|
|
|
.replace(/<script>[.\s\S]+<\/script>/, '')
|
2021-03-10 22:43:26 +03:00
|
|
|
.replace(/<style>.*__playwright_target__.*<\/style>/, '')
|
2021-03-01 23:20:04 +03:00
|
|
|
.replace(/<BASE href="about:blank">/, '')
|
|
|
|
.replace(/<BASE href="http:\/\/localhost:[\d]+\/empty.html">/, '')
|
|
|
|
.replace(/<HTML>/, '')
|
|
|
|
.replace(/<\/HTML>/, '')
|
|
|
|
.replace(/<HEAD>/, '')
|
|
|
|
.replace(/<\/HEAD>/, '')
|
|
|
|
.replace(/<BODY>/, '')
|
2021-03-10 22:43:26 +03:00
|
|
|
.replace(/<\/BODY>/, '').trim();
|
2021-03-01 23:20:04 +03:00
|
|
|
}
|