/** * Copyright 2021 Google Inc. All rights reserved. * * 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. */ import http from 'http'; import type {Server, IncomingMessage, ServerResponse} from 'http'; import type {AddressInfo} from 'net'; import os from 'os'; import {TestServer} from '@pptr/testserver'; import expect from 'expect'; import {getTestState, launch} from './mocha-utils.js'; let HOSTNAME = os.hostname(); // Hostname might not be always accessible in environments other than GitHub // Actions. Therefore, we try to find an external IPv4 address to be used as a // hostname in these tests. const networkInterfaces = os.networkInterfaces(); for (const key of Object.keys(networkInterfaces)) { const interfaces = networkInterfaces[key]; for (const net of interfaces || []) { if (net.family === 'IPv4' && !net.internal) { HOSTNAME = net.address; break; } } } /** * Requests to localhost do not get proxied by default. Create a URL using the hostname * instead. */ function getEmptyPageUrl(server: TestServer): string { const emptyPagePath = new URL(server.EMPTY_PAGE).pathname; return `http://${HOSTNAME}:${server.PORT}${emptyPagePath}`; } describe('request proxy', () => { let proxiedRequestUrls: string[]; let proxyServer: Server; let proxyServerUrl: string; const defaultArgs = [ // We disable this in tests so that proxy-related tests // don't intercept queries from this service in headful. '--disable-features=NetworkTimeServiceQuerying', ]; beforeEach(() => { proxiedRequestUrls = []; proxyServer = http .createServer( ( originalRequest: IncomingMessage, originalResponse: ServerResponse ) => { proxiedRequestUrls.push(originalRequest.url as string); const proxyRequest = http.request( originalRequest.url as string, { method: originalRequest.method, headers: originalRequest.headers, }, proxyResponse => { originalResponse.writeHead( proxyResponse.statusCode as number, proxyResponse.headers ); proxyResponse.pipe(originalResponse, {end: true}); } ); originalRequest.pipe(proxyRequest, {end: true}); } ) .listen(); proxyServerUrl = `http://${HOSTNAME}:${ (proxyServer.address() as AddressInfo).port }`; }); afterEach(async () => { await new Promise((resolve, reject) => { proxyServer.close(error => { if (error) { reject(error); } else { resolve(undefined); } }); }); }); it('should proxy requests when configured', async () => { const {server} = await getTestState({ skipLaunch: true, }); const emptyPageUrl = getEmptyPageUrl(server); const {browser, close} = await launch({ args: [...defaultArgs, `--proxy-server=${proxyServerUrl}`], }); try { const page = await browser.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([emptyPageUrl]); } finally { await close(); } }); it('should respect proxy bypass list', async () => { const {server} = await getTestState({ skipLaunch: true, }); const emptyPageUrl = getEmptyPageUrl(server); const {browser, close} = await launch({ args: [ ...defaultArgs, `--proxy-server=${proxyServerUrl}`, `--proxy-bypass-list=${new URL(emptyPageUrl).host}`, ], }); try { const page = await browser.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([]); } finally { await close(); } }); describe('in incognito browser context', () => { it('should proxy requests when configured at browser level', async () => { const {server} = await getTestState({ skipLaunch: true, }); const emptyPageUrl = getEmptyPageUrl(server); const {browser, close} = await launch({ args: [...defaultArgs, `--proxy-server=${proxyServerUrl}`], }); try { const context = await browser.createIncognitoBrowserContext(); const page = await context.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([emptyPageUrl]); } finally { await close(); } }); it('should respect proxy bypass list when configured at browser level', async () => { const {server} = await getTestState({ skipLaunch: true, }); const emptyPageUrl = getEmptyPageUrl(server); const {browser, close} = await launch({ args: [ ...defaultArgs, `--proxy-server=${proxyServerUrl}`, `--proxy-bypass-list=${new URL(emptyPageUrl).host}`, ], }); try { const context = await browser.createIncognitoBrowserContext(); const page = await context.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([]); } finally { await close(); } }); /** * See issues #7873, #7719, and #7698. */ it('should proxy requests when configured at context level', async () => { const {server} = await getTestState({ skipLaunch: true, }); const emptyPageUrl = getEmptyPageUrl(server); const {browser, close} = await launch({ args: defaultArgs, }); try { const context = await browser.createIncognitoBrowserContext({ proxyServer: proxyServerUrl, }); const page = await context.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([emptyPageUrl]); } finally { await close(); } }); it('should respect proxy bypass list when configured at context level', async () => { const {server} = await getTestState({ skipLaunch: true, }); const emptyPageUrl = getEmptyPageUrl(server); const {browser, close} = await launch({ args: defaultArgs, }); try { const context = await browser.createIncognitoBrowserContext({ proxyServer: proxyServerUrl, proxyBypassList: [new URL(emptyPageUrl).host], }); const page = await context.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([]); } finally { await close(); } }); }); });