| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Copyright 2018 Google Inc. All rights reserved. | 
					
						
							|  |  |  |  * Modifications 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. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-08-19 21:32:12 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-06 07:08:22 -07:00
										 |  |  | import { test as it, expect } from './pageTest'; | 
					
						
							| 
									
										
										
										
											2024-08-13 12:47:02 -07:00
										 |  |  | import { globToRegex } from '../../packages/playwright-core/lib/utils/isomorphic/urlMatch'; | 
					
						
							| 
									
										
										
										
											2020-08-11 15:50:53 -07:00
										 |  |  | import vm from 'vm'; | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-10 19:42:52 +01:00
										 |  |  | it('should work with navigation @smoke', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |   const requests = new Map(); | 
					
						
							|  |  |  |   await page.route('**/*', route => { | 
					
						
							|  |  |  |     requests.set(route.request().url().split('/').pop(), route.request()); | 
					
						
							| 
									
										
										
										
											2023-06-02 21:59:12 +02:00
										 |  |  |     void route.continue(); | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |   }); | 
					
						
							|  |  |  |   server.setRedirect('/rrredirect', '/frames/one-frame.html'); | 
					
						
							|  |  |  |   await page.goto(server.PREFIX + '/rrredirect'); | 
					
						
							|  |  |  |   expect(requests.get('rrredirect').isNavigationRequest()).toBe(true); | 
					
						
							|  |  |  |   expect(requests.get('frame.html').isNavigationRequest()).toBe(true); | 
					
						
							|  |  |  |   expect(requests.get('script.js').isNavigationRequest()).toBe(false); | 
					
						
							|  |  |  |   expect(requests.get('style.css').isNavigationRequest()).toBe(false); | 
					
						
							|  |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-21 00:43:26 +02:00
										 |  |  | it('should intercept after a service worker', async ({ page, server, browserName, isAndroid }) => { | 
					
						
							| 
									
										
										
										
											2021-04-09 07:59:09 -07:00
										 |  |  |   it.skip(isAndroid); | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |   await page.goto(server.PREFIX + '/serviceworkers/fetchdummy/sw.html'); | 
					
						
							| 
									
										
										
										
											2020-08-28 04:20:29 -07:00
										 |  |  |   await page.evaluate(() => window['activationPromise']); | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Sanity check.
 | 
					
						
							| 
									
										
										
										
											2020-08-28 04:20:29 -07:00
										 |  |  |   const swResponse = await page.evaluate(() => window['fetchDummy']('foo')); | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |   expect(swResponse).toBe('responseFromServiceWorker:foo'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   await page.route('**/foo', route => { | 
					
						
							|  |  |  |     const slash = route.request().url().lastIndexOf('/'); | 
					
						
							|  |  |  |     const name = route.request().url().substring(slash + 1); | 
					
						
							| 
									
										
										
										
											2023-06-02 21:59:12 +02:00
										 |  |  |     void route.fulfill({ | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |       status: 200, | 
					
						
							|  |  |  |       contentType: 'text/css', | 
					
						
							|  |  |  |       body: 'responseFromInterception:' + name | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Page route is applied after service worker fetch event.
 | 
					
						
							| 
									
										
										
										
											2020-08-28 04:20:29 -07:00
										 |  |  |   const swResponse2 = await page.evaluate(() => window['fetchDummy']('foo')); | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |   expect(swResponse2).toBe('responseFromServiceWorker:foo'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Page route is not applied to service worker initiated fetch.
 | 
					
						
							| 
									
										
										
										
											2020-08-28 04:20:29 -07:00
										 |  |  |   const nonInterceptedResponse = await page.evaluate(() => window['fetchDummy']('passthrough')); | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |   expect(nonInterceptedResponse).toBe('FAILURE: Not Found'); | 
					
						
							| 
									
										
										
										
											2022-06-02 17:16:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Firefox does not want to fetch the redirect for some reason.
 | 
					
						
							|  |  |  |   if (browserName !== 'firefox') { | 
					
						
							|  |  |  |     // Page route is not applied to service worker initiated fetch with redirect.
 | 
					
						
							|  |  |  |     server.setRedirect('/serviceworkers/fetchdummy/passthrough', '/simple.json'); | 
					
						
							|  |  |  |     const redirectedResponse = await page.evaluate(() => window['fetchDummy']('passthrough')); | 
					
						
							|  |  |  |     expect(redirectedResponse).toBe('{"foo": "bar"}\n'); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-28 04:20:29 -07:00
										 |  |  | it('should work with glob', async () => { | 
					
						
							| 
									
										
										
										
											2020-08-19 13:27:58 -07:00
										 |  |  |   expect(globToRegex('**/*.js').test('https://localhost:8080/foo.js')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('**/*.css').test('https://localhost:8080/foo.js')).toBeFalsy(); | 
					
						
							|  |  |  |   expect(globToRegex('*.js').test('https://localhost:8080/foo.js')).toBeFalsy(); | 
					
						
							|  |  |  |   expect(globToRegex('https://**/*.js').test('https://localhost:8080/foo.js')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('http://localhost:8080/simple/path.js').test('http://localhost:8080/simple/path.js')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('http://localhost:8080/?imple/path.js').test('http://localhost:8080/Simple/path.js')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('**/{a,b}.js').test('https://localhost:8080/a.js')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('**/{a,b}.js').test('https://localhost:8080/b.js')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('**/{a,b}.js').test('https://localhost:8080/c.js')).toBeFalsy(); | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 13:27:58 -07:00
										 |  |  |   expect(globToRegex('**/*.{png,jpg,jpeg}').test('https://localhost:8080/c.jpg')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('**/*.{png,jpg,jpeg}').test('https://localhost:8080/c.jpeg')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('**/*.{png,jpg,jpeg}').test('https://localhost:8080/c.png')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('**/*.{png,jpg,jpeg}').test('https://localhost:8080/c.css')).toBeFalsy(); | 
					
						
							| 
									
										
										
										
											2021-06-04 13:33:45 -07:00
										 |  |  |   expect(globToRegex('foo*').test('foo.js')).toBeTruthy(); | 
					
						
							|  |  |  |   expect(globToRegex('foo*').test('foo/bar.js')).toBeFalsy(); | 
					
						
							|  |  |  |   expect(globToRegex('http://localhost:3000/signin-oidc*').test('http://localhost:3000/signin-oidc/foo')).toBeFalsy(); | 
					
						
							|  |  |  |   expect(globToRegex('http://localhost:3000/signin-oidc*').test('http://localhost:3000/signin-oidcnice')).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2023-06-28 16:33:04 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-24 20:49:05 +04:00
										 |  |  |   expect(globToRegex('**/three-columns/settings.html?**id=[a-z]**').test('http://mydomain:8080/blah/blah/three-columns/settings.html?id=settings-e3c58efe-02e9-44b0-97ac-dd138100cf7c&blah')).toBeTruthy(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-28 16:33:04 -07:00
										 |  |  |   expect(globToRegex('\\?')).toEqual(/^\?$/); | 
					
						
							|  |  |  |   expect(globToRegex('\\')).toEqual(/^\\$/); | 
					
						
							|  |  |  |   expect(globToRegex('\\\\')).toEqual(/^\\$/); | 
					
						
							|  |  |  |   expect(globToRegex('\\[')).toEqual(/^\[$/); | 
					
						
							| 
									
										
										
										
											2023-07-24 20:49:05 +04:00
										 |  |  |   expect(globToRegex('[a-z]')).toEqual(/^[a-z]$/); | 
					
						
							|  |  |  |   expect(globToRegex('$^+.\\*()|\\?\\{\\}\\[\\]')).toEqual(/^\$\^\+\.\*\(\)\|\?\{\}\[\]$/); | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-29 10:47:31 -07:00
										 |  |  | it('should intercept network activity from worker', async function({ page, server, isAndroid, browserName, browserMajorVersion }) { | 
					
						
							| 
									
										
										
										
											2023-05-12 09:27:08 -07:00
										 |  |  |   it.skip(browserName === 'firefox' && browserMajorVersion < 114, 'https://github.com/microsoft/playwright/issues/21760'); | 
					
						
							| 
									
										
										
										
											2021-04-09 07:59:09 -07:00
										 |  |  |   it.skip(isAndroid); | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 15:03:34 -08:00
										 |  |  |   await page.goto(server.EMPTY_PAGE); | 
					
						
							|  |  |  |   server.setRoute('/data_for_worker', (req, res) => res.end('failed to intercept')); | 
					
						
							|  |  |  |   const url = server.PREFIX + '/data_for_worker'; | 
					
						
							| 
									
										
										
										
											2020-12-10 15:04:57 -08:00
										 |  |  |   await page.route(url, route => { | 
					
						
							|  |  |  |     route.fulfill({ | 
					
						
							| 
									
										
										
										
											2020-11-23 15:03:34 -08:00
										 |  |  |       status: 200, | 
					
						
							|  |  |  |       body: 'intercepted', | 
					
						
							| 
									
										
										
										
											2020-12-10 15:04:57 -08:00
										 |  |  |     }).catch(e => null); | 
					
						
							| 
									
										
										
										
											2020-11-23 15:03:34 -08:00
										 |  |  |   }); | 
					
						
							|  |  |  |   const [msg] = await Promise.all([ | 
					
						
							|  |  |  |     page.waitForEvent('console'), | 
					
						
							|  |  |  |     page.evaluate(url => new Worker(URL.createObjectURL(new Blob([`
 | 
					
						
							|  |  |  |       fetch("${url}").then(response => response.text()).then(console.log); | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |     `], { type: 'application/javascript' }))), url),
 | 
					
						
							| 
									
										
										
										
											2020-11-23 15:03:34 -08:00
										 |  |  |   ]); | 
					
						
							|  |  |  |   expect(msg.text()).toBe('intercepted'); | 
					
						
							|  |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-28 15:41:32 +02:00
										 |  |  | it('should intercept network activity from worker 2', async function({ page, server, isAndroid }) { | 
					
						
							| 
									
										
										
										
											2021-04-09 07:59:09 -07:00
										 |  |  |   it.skip(isAndroid); | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-10 15:04:57 -08:00
										 |  |  |   const url = server.PREFIX + '/worker/worker.js'; | 
					
						
							|  |  |  |   await page.route(url, route => { | 
					
						
							|  |  |  |     route.fulfill({ | 
					
						
							|  |  |  |       status: 200, | 
					
						
							|  |  |  |       body: 'console.log("intercepted");', | 
					
						
							|  |  |  |       contentType: 'application/javascript', | 
					
						
							|  |  |  |     }).catch(e => null); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   const [msg] = await Promise.all([ | 
					
						
							|  |  |  |     page.waitForEvent('console'), | 
					
						
							|  |  |  |     page.goto(server.PREFIX + '/worker/worker.html'), | 
					
						
							|  |  |  |   ]); | 
					
						
							|  |  |  |   expect(msg.text()).toBe('intercepted'); | 
					
						
							|  |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-28 15:41:32 +02:00
										 |  |  | it('should work with regular expression passed from a different context', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |   const ctx = vm.createContext(); | 
					
						
							|  |  |  |   const regexp = vm.runInContext('new RegExp("empty\\.html")', ctx); | 
					
						
							|  |  |  |   let intercepted = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   await page.route(regexp, (route, request) => { | 
					
						
							|  |  |  |     expect(route.request()).toBe(request); | 
					
						
							|  |  |  |     expect(request.url()).toContain('empty.html'); | 
					
						
							|  |  |  |     expect(request.headers()['user-agent']).toBeTruthy(); | 
					
						
							|  |  |  |     expect(request.method()).toBe('GET'); | 
					
						
							|  |  |  |     expect(request.postData()).toBe(null); | 
					
						
							|  |  |  |     expect(request.isNavigationRequest()).toBe(true); | 
					
						
							|  |  |  |     expect(request.resourceType()).toBe('document'); | 
					
						
							|  |  |  |     expect(request.frame() === page.mainFrame()).toBe(true); | 
					
						
							|  |  |  |     expect(request.frame().url()).toBe('about:blank'); | 
					
						
							| 
									
										
										
										
											2023-06-02 21:59:12 +02:00
										 |  |  |     void route.continue(); | 
					
						
							| 
									
										
										
										
											2020-08-04 16:32:10 -07:00
										 |  |  |     intercepted = true; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const response = await page.goto(server.EMPTY_PAGE); | 
					
						
							|  |  |  |   expect(response.ok()).toBe(true); | 
					
						
							|  |  |  |   expect(intercepted).toBe(true); | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
										
										
											2021-03-25 17:16:41 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 10:22:23 -07:00
										 |  |  | it('should not break remote worker importScripts', async ({ page, server, browserName, browserMajorVersion }) => { | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:32 -07:00
										 |  |  |   await page.route('**', async route => { | 
					
						
							|  |  |  |     await route.continue(); | 
					
						
							| 
									
										
										
										
											2021-03-25 17:16:41 -07:00
										 |  |  |   }); | 
					
						
							|  |  |  |   await page.goto(server.PREFIX + '/worker/worker-http-import.html'); | 
					
						
							|  |  |  |   await page.waitForSelector("#status:has-text('finished')"); | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
										
										
											2023-12-01 14:49:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | it('should disable memory cache when intercepting', async ({ page, server }) => { | 
					
						
							|  |  |  |   let interceted = 0; | 
					
						
							|  |  |  |   await page.route('**/page.html', route => { | 
					
						
							|  |  |  |     ++interceted; | 
					
						
							|  |  |  |     void route.fulfill({ | 
					
						
							|  |  |  |       body: 'success' | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   await page.goto(server.PREFIX + '/page.html'); | 
					
						
							|  |  |  |   expect(await page.locator('body').textContent()).toContain('success'); | 
					
						
							|  |  |  |   await page.goto(server.EMPTY_PAGE); | 
					
						
							|  |  |  |   await expect(page).toHaveURL(server.EMPTY_PAGE); | 
					
						
							|  |  |  |   expect(interceted).toBe(1); | 
					
						
							|  |  |  |   await page.goBack(); | 
					
						
							|  |  |  |   await expect(page).toHaveURL(server.PREFIX + '/page.html'); | 
					
						
							|  |  |  |   expect(interceted).toBe(2); | 
					
						
							|  |  |  | }); |