| 
									
										
										
										
											2021-03-01 12:20:04 -08: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-09 17:47:20 -07:00
										 |  |  | import { contextTest, expect } from './config/browserTest'; | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  | import { InMemorySnapshotter } from '../lib/server/snapshot/inMemorySnapshotter'; | 
					
						
							| 
									
										
										
										
											2021-03-08 19:49:57 -08:00
										 |  |  | import { HttpServer } from '../lib/utils/httpServer'; | 
					
						
							|  |  |  | import { SnapshotServer } from '../lib/server/snapshot/snapshotServer'; | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  | import type { Frame } from '..'; | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  | const it = contextTest.extend<{ snapshotPort: number, snapshotter: InMemorySnapshotter, showSnapshot: (snapshot: any) => Promise<Frame> }>({ | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |   snapshotPort: async ({}, run, testInfo) => { | 
					
						
							|  |  |  |     await run(11000 + testInfo.workerIndex); | 
					
						
							| 
									
										
										
										
											2021-05-09 17:47:20 -07:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |   snapshotter: async ({ mode, toImpl, context, snapshotPort }, run, testInfo) => { | 
					
						
							| 
									
										
										
										
											2021-05-23 16:21:18 -07:00
										 |  |  |     testInfo.skip(mode !== 'default'); | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |     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-09 17:47:20 -07:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   showSnapshot: async ({ contextFactory, snapshotPort }, use) => { | 
					
						
							|  |  |  |     await use(async (snapshot: any) => { | 
					
						
							|  |  |  |       const previewContext = await contextFactory(); | 
					
						
							|  |  |  |       const previewPage = await previewContext.newPage(); | 
					
						
							|  |  |  |       previewPage.on('console', console.log); | 
					
						
							|  |  |  |       await previewPage.goto(`http://localhost:${snapshotPort}/snapshot/`); | 
					
						
							|  |  |  |       const frameSnapshot = snapshot.snapshot(); | 
					
						
							|  |  |  |       await previewPage.evaluate(snapshotId => { | 
					
						
							|  |  |  |         (window as any).showSnapshot(snapshotId); | 
					
						
							|  |  |  |       }, `${frameSnapshot.pageId}?name=${frameSnapshot.snapshotName}`); | 
					
						
							|  |  |  |       // wait for the render frame to load
 | 
					
						
							|  |  |  |       while (previewPage.frames().length < 2) | 
					
						
							|  |  |  |         await new Promise(f => previewPage.once('frameattached', f)); | 
					
						
							|  |  |  |       const frame = previewPage.frames()[1]; | 
					
						
							|  |  |  |       await frame.waitForLoadState(); | 
					
						
							|  |  |  |       return frame; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   }, | 
					
						
							| 
									
										
										
										
											2021-05-09 17:47:20 -07:00
										 |  |  | }); | 
					
						
							| 
									
										
										
										
											2021-04-01 13:18:04 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 17:47:20 -07:00
										 |  |  | it.describe('snapshots', () => { | 
					
						
							|  |  |  |   it('should collect snapshot', async ({ page, toImpl, snapshotter }) => { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |     await page.setContent('<button>Hello</button>'); | 
					
						
							|  |  |  |     const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot'); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshot)).toBe('<BUTTON>Hello</BUTTON>'); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-04 21:11:35 -07:00
										 |  |  |   it('should preserve BASE and other content on reset', async ({ page, toImpl, snapshotter, server }) => { | 
					
						
							|  |  |  |     await page.goto(server.EMPTY_PAGE); | 
					
						
							|  |  |  |     const snapshot1 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot1'); | 
					
						
							|  |  |  |     const html1 = snapshot1.render().html; | 
					
						
							|  |  |  |     expect(html1).toContain(`<BASE href="${server.EMPTY_PAGE}"`); | 
					
						
							|  |  |  |     await snapshotter.reset(); | 
					
						
							|  |  |  |     const snapshot2 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot2'); | 
					
						
							|  |  |  |     const html2 = snapshot2.render().html; | 
					
						
							|  |  |  |     expect(html2.replace(`"snapshot2"`, `"snapshot1"`)).toEqual(html1); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 17:47:20 -07:00
										 |  |  |   it('should capture resources', async ({ page, toImpl, server, snapshotter }) => { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08: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'); | 
					
						
							| 
									
										
										
										
											2021-08-10 21:23:31 -07:00
										 |  |  |     const resource = snapshot.resourceByUrl(`http://localhost:${server.PORT}/style.css`); | 
					
						
							|  |  |  |     expect(resource).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 17:47:20 -07:00
										 |  |  |   it('should collect multiple', async ({ page, toImpl, snapshotter }) => { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08: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-09 17:47:20 -07:00
										 |  |  |   it('should respect inline CSSOM change', async ({ page, toImpl, snapshotter }) => { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |     await page.setContent('<style>button { color: red; }</style><button>Hello</button>'); | 
					
						
							| 
									
										
										
										
											2021-04-27 11:07:07 -07: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 12:20:04 -08:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-25 18:52:36 -07:00
										 |  |  |   it('should respect node removal', async ({ page, toImpl, snapshotter }) => { | 
					
						
							|  |  |  |     page.on('console', console.log); | 
					
						
							|  |  |  |     await page.setContent('<div><button id="button1"></button><button id="button2"></button></div>'); | 
					
						
							|  |  |  |     const snapshot1 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot1'); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshot1)).toBe('<DIV><BUTTON id=\"button1\"></BUTTON><BUTTON id=\"button2\"></BUTTON></DIV>'); | 
					
						
							|  |  |  |     await page.evaluate(() => document.getElementById('button2').remove()); | 
					
						
							|  |  |  |     const snapshot2 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot2'); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshot2)).toBe('<DIV><BUTTON id=\"button1\"></BUTTON></DIV>'); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should respect attr removal', async ({ page, toImpl, snapshotter }) => { | 
					
						
							|  |  |  |     page.on('console', console.log); | 
					
						
							|  |  |  |     await page.setContent('<div id="div" attr1="1" attr2="2"></div>'); | 
					
						
							|  |  |  |     const snapshot1 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot1'); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshot1)).toBe('<DIV id=\"div\" attr1=\"1\" attr2=\"2\"></DIV>'); | 
					
						
							|  |  |  |     await page.evaluate(() => document.getElementById('div').removeAttribute('attr2')); | 
					
						
							|  |  |  |     const snapshot2 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot2'); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshot2)).toBe('<DIV id=\"div\" attr1=\"1\"></DIV>'); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 09:41:29 -07:00
										 |  |  |   it('should have a custom doctype', async ({page, server, toImpl, snapshotter}) => { | 
					
						
							|  |  |  |     await page.goto(server.EMPTY_PAGE); | 
					
						
							|  |  |  |     await page.setContent('<!DOCTYPE foo><body>hi</body>'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot'); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshot)).toBe('<!DOCTYPE foo>hi'); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 17:47:20 -07:00
										 |  |  |   it('should respect subresource CSSOM change', async ({ page, server, toImpl, snapshotter }) => { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08: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 11:07:07 -07: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'); | 
					
						
							| 
									
										
										
										
											2021-08-10 21:23:31 -07:00
										 |  |  |     const resource = snapshot2.resourceByUrl(`http://localhost:${server.PORT}/style.css`); | 
					
						
							| 
									
										
										
										
											2021-08-24 13:17:58 -07:00
										 |  |  |     expect(snapshotter.resourceContent(resource.response.content._sha1).toString()).toBe('button { color: blue; }'); | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-03-08 19:49:57 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  |   it('should capture iframe', async ({ page, server, toImpl, browserName, snapshotter, showSnapshot }) => { | 
					
						
							| 
									
										
										
										
											2021-04-05 09:18:56 -07:00
										 |  |  |     it.skip(browserName === 'firefox'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 19:49:57 -08: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.
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  |     const frame = await showSnapshot(snapshot); | 
					
						
							|  |  |  |     while (frame.childFrames().length < 1) | 
					
						
							|  |  |  |       await new Promise(f => frame.page().once('frameattached', f)); | 
					
						
							|  |  |  |     const button = await frame.childFrames()[0].waitForSelector('button'); | 
					
						
							| 
									
										
										
										
											2021-03-08 19:49:57 -08:00
										 |  |  |     expect(await button.textContent()).toBe('Hello iframe'); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-03-10 11:43:26 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 17:47:20 -07:00
										 |  |  |   it('should capture snapshot target', async ({ page, toImpl, snapshotter }) => { | 
					
						
							| 
									
										
										
										
											2021-03-10 11:43:26 -08: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-09 17:47:20 -07:00
										 |  |  |   it('should collect on attribute change', async ({ page, toImpl, snapshotter }) => { | 
					
						
							| 
									
										
										
										
											2021-03-10 11:43:26 -08: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-06-17 09:41:29 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  |   it('should contain adopted style sheets', async ({ page, toImpl, showSnapshot, snapshotter, browserName }) => { | 
					
						
							| 
									
										
										
										
											2021-06-17 09:41:29 -07:00
										 |  |  |     it.skip(browserName !== 'chromium', 'Constructed stylesheets are only in Chromium.'); | 
					
						
							|  |  |  |     await page.setContent('<button>Hello</button>'); | 
					
						
							|  |  |  |     await page.evaluate(() => { | 
					
						
							|  |  |  |       const sheet = new CSSStyleSheet(); | 
					
						
							|  |  |  |       sheet.addRule('button', 'color: red'); | 
					
						
							|  |  |  |       (document as any).adoptedStyleSheets = [sheet]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const div = document.createElement('div'); | 
					
						
							|  |  |  |       const root = div.attachShadow({ | 
					
						
							|  |  |  |         mode: 'open' | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |       root.append('foo'); | 
					
						
							|  |  |  |       const sheet2 = new CSSStyleSheet(); | 
					
						
							|  |  |  |       sheet2.addRule(':host', 'color: blue'); | 
					
						
							|  |  |  |       (root as any).adoptedStyleSheets = [sheet2]; | 
					
						
							|  |  |  |       document.body.appendChild(div); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     const snapshot1 = await snapshotter.captureSnapshot(toImpl(page), 'snapshot1'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  |     const frame = await showSnapshot(snapshot1); | 
					
						
							|  |  |  |     await frame.waitForSelector('button'); | 
					
						
							|  |  |  |     const buttonColor = await frame.$eval('button', button => { | 
					
						
							| 
									
										
										
										
											2021-06-17 09:41:29 -07:00
										 |  |  |       return window.getComputedStyle(button).color; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     expect(buttonColor).toBe('rgb(255, 0, 0)'); | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  |     const divColor = await frame.$eval('div', div => { | 
					
						
							| 
									
										
										
										
											2021-06-17 09:41:29 -07:00
										 |  |  |       return window.getComputedStyle(div).color; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     expect(divColor).toBe('rgb(0, 0, 255)'); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-06-29 16:20:15 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  |   it('should restore scroll positions', async ({ page, showSnapshot, toImpl, snapshotter, browserName }) => { | 
					
						
							| 
									
										
										
										
											2021-06-29 16:20:15 -07:00
										 |  |  |     it.skip(browserName === 'firefox'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await page.setContent(`
 | 
					
						
							|  |  |  |       <style> | 
					
						
							|  |  |  |         li { height: 20px; margin: 0; padding: 0; } | 
					
						
							|  |  |  |         div { height: 60px; overflow-x: hidden; overflow-y: scroll; background: green; padding: 0; margin: 0; } | 
					
						
							|  |  |  |       </style> | 
					
						
							|  |  |  |       <div> | 
					
						
							|  |  |  |         <ul> | 
					
						
							|  |  |  |           <li>Item 1</li> | 
					
						
							|  |  |  |           <li>Item 2</li> | 
					
						
							|  |  |  |           <li>Item 3</li> | 
					
						
							|  |  |  |           <li>Item 4</li> | 
					
						
							|  |  |  |           <li>Item 5</li> | 
					
						
							|  |  |  |           <li>Item 6</li> | 
					
						
							|  |  |  |           <li>Item 7</li> | 
					
						
							|  |  |  |           <li>Item 8</li> | 
					
						
							|  |  |  |           <li>Item 9</li> | 
					
						
							|  |  |  |           <li>Item 10</li> | 
					
						
							|  |  |  |         </ul> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     `);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await (await page.$('text=Item 8')).scrollIntoViewIfNeeded(); | 
					
						
							|  |  |  |     const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'scrolled'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Render snapshot, check expectations.
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  |     const frame = await showSnapshot(snapshot); | 
					
						
							|  |  |  |     const div = await frame.waitForSelector('div'); | 
					
						
							| 
									
										
										
										
											2021-06-29 16:20:15 -07:00
										 |  |  |     expect(await div.evaluate(div => div.scrollTop)).toBe(136); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-03 13:44:02 -07:00
										 |  |  |   it('should work with meta CSP', async ({ page, showSnapshot, toImpl, snapshotter, browserName }) => { | 
					
						
							|  |  |  |     it.skip(browserName === 'firefox'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await page.setContent(`
 | 
					
						
							|  |  |  |       <head> | 
					
						
							|  |  |  |         <meta http-equiv="Content-Security-Policy" content="script-src 'none'"> | 
					
						
							|  |  |  |       </head> | 
					
						
							|  |  |  |       <body> | 
					
						
							|  |  |  |         <div>Hello</div> | 
					
						
							|  |  |  |       </body> | 
					
						
							|  |  |  |     `);
 | 
					
						
							|  |  |  |     await page.$eval('div', div => { | 
					
						
							|  |  |  |       const shadow = div.attachShadow({ mode: 'open' }); | 
					
						
							|  |  |  |       const span = document.createElement('span'); | 
					
						
							|  |  |  |       span.textContent = 'World'; | 
					
						
							|  |  |  |       shadow.appendChild(span); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'meta'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Render snapshot, check expectations.
 | 
					
						
							|  |  |  |     const frame = await showSnapshot(snapshot); | 
					
						
							|  |  |  |     await frame.waitForSelector('div'); | 
					
						
							|  |  |  |     // Should render shadow dom with post-processing script.
 | 
					
						
							|  |  |  |     expect(await frame.textContent('span')).toBe('World'); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:40:46 -07:00
										 |  |  |   it('should handle multiple headers', async ({ page, server, showSnapshot, toImpl, snapshotter, browserName }) => { | 
					
						
							|  |  |  |     it.skip(browserName === 'firefox'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     server.setRoute('/foo.css', (req, res) => { | 
					
						
							|  |  |  |       res.statusCode = 200; | 
					
						
							|  |  |  |       res.setHeader('vary', ['accepts-encoding', 'accepts-encoding']); | 
					
						
							|  |  |  |       res.end('body { padding: 42px }'); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await page.goto(server.EMPTY_PAGE); | 
					
						
							|  |  |  |     await page.setContent(`<head><link rel=stylesheet href="/foo.css"></head><body><div>Hello</div></body>`); | 
					
						
							|  |  |  |     const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot'); | 
					
						
							|  |  |  |     const frame = await showSnapshot(snapshot); | 
					
						
							|  |  |  |     await frame.waitForSelector('div'); | 
					
						
							|  |  |  |     const padding = await frame.$eval('body', body => window.getComputedStyle(body).paddingLeft); | 
					
						
							|  |  |  |     expect(padding).toBe('42px'); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-08-24 17:05:26 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   it('should handle src=blob', async ({ page, server, showSnapshot, toImpl, snapshotter, browserName }) => { | 
					
						
							|  |  |  |     it.skip(browserName === 'firefox'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await page.goto(server.EMPTY_PAGE); | 
					
						
							|  |  |  |     await page.evaluate(async () => { | 
					
						
							|  |  |  |       const dataUrl = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAoAAAASCAQAAADIvofAAAAABGdBTUEAALGPC/xhBQAAACBjSFJNAAB6JgAAgIQAAPoAAACA6AAAdTAAAOpgAAA6mAAAF3CculE8AAAAAmJLR0QA/4ePzL8AAAAHdElNRQfhBhAPKSstM+EuAAAAvUlEQVQY05WQIW4CYRgF599gEZgeoAKBWIfCNSmVvQMe3wv0ChhIViKwtTQEAYJwhgpISBA0JSxNIdlB7LIGTJ/8kpeZ7wW5TcT9o/QNBtvOrrWMrtg0sSGOFeELbHlCDsQ+ukeYiHNFJPHBDRKlQKVEbFkLUT3AiAxI6VGCXsWXAoQLBUl5E7HjUFwiyI4zf/wWoB3CFnxX5IeGdY8IGU/iwE9jcZrLy4pnEat+FL4hf/cbqREKo/Cf6W5zASVMeh234UtGAAAAJXRFWHRkYXRlOmNyZWF0ZQAyMDE3LTA2LTE2VDE1OjQxOjQzLTA3OjAwd1xNIQAAACV0RVh0ZGF0ZTptb2RpZnkAMjAxNy0wNi0xNlQxNTo0MTo0My0wNzowMAYB9Z0AAAAASUVORK5CYII='; | 
					
						
							|  |  |  |       const blob = await fetch(dataUrl).then(res => res.blob()); | 
					
						
							|  |  |  |       const url = window.URL.createObjectURL(blob); | 
					
						
							|  |  |  |       const img = document.createElement('img'); | 
					
						
							|  |  |  |       img.src = url; | 
					
						
							|  |  |  |       const loaded = new Promise(f => img.onload = f); | 
					
						
							|  |  |  |       document.body.appendChild(img); | 
					
						
							|  |  |  |       await loaded; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const snapshot = await snapshotter.captureSnapshot(toImpl(page), 'snapshot'); | 
					
						
							|  |  |  |     const frame = await showSnapshot(snapshot); | 
					
						
							|  |  |  |     const img = await frame.waitForSelector('img'); | 
					
						
							|  |  |  |     expect(await img.screenshot()).toMatchSnapshot('blob-src.png'); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function distillSnapshot(snapshot) { | 
					
						
							|  |  |  |   const { html } = snapshot.render(); | 
					
						
							|  |  |  |   return html | 
					
						
							| 
									
										
										
										
											2021-03-08 19:49:57 -08:00
										 |  |  |       .replace(/<script>[.\s\S]+<\/script>/, '') | 
					
						
							| 
									
										
										
										
											2021-03-10 11:43:26 -08:00
										 |  |  |       .replace(/<style>.*__playwright_target__.*<\/style>/, '') | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08: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 11:43:26 -08:00
										 |  |  |       .replace(/<\/BODY>/, '').trim(); | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  | } |