| 
									
										
										
										
											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-04-05 18:48:46 -07:00
										 |  |  | import { test as it, expect } from './config/contextTest'; | 
					
						
							| 
									
										
										
										
											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-03-01 12:20:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 09:18:56 -07:00
										 |  |  | it.describe('snapshots', () => { | 
					
						
							| 
									
										
										
										
											2021-04-01 13:18:04 -07:00
										 |  |  |   let snapshotter: any; | 
					
						
							|  |  |  |   let httpServer: any; | 
					
						
							|  |  |  |   let snapshotPort: number; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 18:48:46 -07:00
										 |  |  |   it.beforeEach(async ({ mode, toImpl, context }, testInfo) => { | 
					
						
							| 
									
										
										
										
											2021-04-05 09:18:56 -07:00
										 |  |  |     it.skip(mode !== 'default'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 13:18:04 -07:00
										 |  |  |     snapshotter = new InMemorySnapshotter(toImpl(context)); | 
					
						
							|  |  |  |     await snapshotter.initialize(); | 
					
						
							|  |  |  |     httpServer = new HttpServer(); | 
					
						
							|  |  |  |     new SnapshotServer(httpServer, snapshotter); | 
					
						
							| 
									
										
										
										
											2021-04-05 18:48:46 -07:00
										 |  |  |     snapshotPort = 11000 + testInfo.workerIndex; | 
					
						
							| 
									
										
										
										
											2021-04-01 13:18:04 -07:00
										 |  |  |     httpServer.start(snapshotPort); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 09:18:56 -07:00
										 |  |  |   it.afterEach(async () => { | 
					
						
							| 
									
										
										
										
											2021-04-01 13:18:04 -07:00
										 |  |  |     await snapshotter.dispose(); | 
					
						
							|  |  |  |     httpServer.stop(); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 18:48:46 -07:00
										 |  |  |   it('should collect snapshot', async ({ page, toImpl }) => { | 
					
						
							| 
									
										
										
										
											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-04-05 18:48:46 -07:00
										 |  |  |   it('should capture resources', async ({ page, toImpl, server }) => { | 
					
						
							| 
									
										
										
										
											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'); | 
					
						
							|  |  |  |     const { resources } = snapshot.render(); | 
					
						
							|  |  |  |     const cssHref = `http://localhost:${server.PORT}/style.css`; | 
					
						
							|  |  |  |     expect(resources[cssHref]).toBeTruthy(); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 18:48:46 -07:00
										 |  |  |   it('should collect multiple', async ({ page, toImpl }) => { | 
					
						
							| 
									
										
										
										
											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-04-05 18:48:46 -07:00
										 |  |  |   it('should only collect on change', async ({ page }) => { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |     await page.setContent('<button>Hello</button>'); | 
					
						
							|  |  |  |     const snapshots = []; | 
					
						
							|  |  |  |     snapshotter.on('snapshot', snapshot => snapshots.push(snapshot)); | 
					
						
							|  |  |  |     await Promise.all([ | 
					
						
							|  |  |  |       new Promise(f => snapshotter.once('snapshot', f)), | 
					
						
							| 
									
										
										
										
											2021-04-08 05:32:12 +08:00
										 |  |  |       snapshotter.setAutoSnapshotIntervalForTest(25), | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |     ]); | 
					
						
							|  |  |  |     await Promise.all([ | 
					
						
							|  |  |  |       new Promise(f => snapshotter.once('snapshot', f)), | 
					
						
							|  |  |  |       page.setContent('<button>Hello 2</button>') | 
					
						
							|  |  |  |     ]); | 
					
						
							|  |  |  |     expect(snapshots.length).toBe(2); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 18:48:46 -07:00
										 |  |  |   it('should respect inline CSSOM change', async ({ page }) => { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |     await page.setContent('<style>button { color: red; }</style><button>Hello</button>'); | 
					
						
							|  |  |  |     const snapshots = []; | 
					
						
							|  |  |  |     snapshotter.on('snapshot', snapshot => snapshots.push(snapshot)); | 
					
						
							|  |  |  |     await Promise.all([ | 
					
						
							|  |  |  |       new Promise(f => snapshotter.once('snapshot', f)), | 
					
						
							| 
									
										
										
										
											2021-04-08 05:32:12 +08:00
										 |  |  |       snapshotter.setAutoSnapshotIntervalForTest(25), | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |     ]); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshots[0])).toBe('<style>button { color: red; }</style><BUTTON>Hello</BUTTON>'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await Promise.all([ | 
					
						
							|  |  |  |       new Promise(f => snapshotter.once('snapshot', f)), | 
					
						
							|  |  |  |       page.evaluate(() => { | 
					
						
							|  |  |  |         (document.styleSheets[0].cssRules[0] as any).style.color = 'blue'; | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     ]); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshots[1])).toBe('<style>button { color: blue; }</style><BUTTON>Hello</BUTTON>'); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 18:48:46 -07:00
										 |  |  |   it('should respect subresource CSSOM change', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											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 snapshots = []; | 
					
						
							|  |  |  |     snapshotter.on('snapshot', snapshot => snapshots.push(snapshot)); | 
					
						
							|  |  |  |     await Promise.all([ | 
					
						
							|  |  |  |       new Promise(f => snapshotter.once('snapshot', f)), | 
					
						
							| 
									
										
										
										
											2021-04-08 05:32:12 +08:00
										 |  |  |       snapshotter.setAutoSnapshotIntervalForTest(25), | 
					
						
							| 
									
										
										
										
											2021-03-01 12:20:04 -08:00
										 |  |  |     ]); | 
					
						
							|  |  |  |     expect(distillSnapshot(snapshots[0])).toBe('<LINK rel=\"stylesheet\" href=\"style.css\"><BUTTON>Hello</BUTTON>'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await Promise.all([ | 
					
						
							|  |  |  |       new Promise(f => snapshotter.once('snapshot', f)), | 
					
						
							|  |  |  |       page.evaluate(() => { | 
					
						
							|  |  |  |         (document.styleSheets[0].cssRules[0] as any).style.color = 'blue'; | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     ]); | 
					
						
							|  |  |  |     const { resources } = snapshots[1].render(); | 
					
						
							|  |  |  |     const cssHref = `http://localhost:${server.PORT}/style.css`; | 
					
						
							|  |  |  |     const { sha1 } = resources[cssHref]; | 
					
						
							|  |  |  |     expect(snapshotter.resourceContent(sha1).toString()).toBe('button { color: blue; }'); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-03-08 19:49:57 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 18:48:46 -07:00
										 |  |  |   it('should capture iframe', async ({ page, contextFactory, server, toImpl, browserName }) => { | 
					
						
							| 
									
										
										
										
											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.
 | 
					
						
							|  |  |  |     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}`); | 
					
						
							|  |  |  |     while (previewPage.frames().length < 4) | 
					
						
							|  |  |  |       await new Promise(f => previewPage.once('frameattached', f)); | 
					
						
							|  |  |  |     const button = await previewPage.frames()[3].waitForSelector('button'); | 
					
						
							|  |  |  |     expect(await button.textContent()).toBe('Hello iframe'); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-03-10 11:43:26 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 18:48:46 -07:00
										 |  |  |   it('should capture snapshot target', async ({ page, toImpl }) => { | 
					
						
							| 
									
										
										
										
											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-04-05 18:48:46 -07:00
										 |  |  |   it('should collect on attribute change', async ({ page, toImpl }) => { | 
					
						
							| 
									
										
										
										
											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-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
										 |  |  | } |