/** * 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 expect from 'expect'; import http from 'http'; import os from 'os'; import {getTestState} from './mocha-utils.js'; import type {Server, IncomingMessage, ServerResponse} from 'http'; import type {Browser} from '../../lib/cjs/puppeteer/common/Browser.js'; import type {AddressInfo} from 'net'; import {TestServer} from '../../utils/testserver/lib/index.js'; import {it} 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 browser: Browser; let proxiedRequestUrls: string[]; let proxyServer: Server; let proxyServerUrl: string; const defaultArgs = [ '--disable-features=NetworkTimeServiceQuerying', // We disable this in tests so that proxy-related tests don't intercept queries from this service in headful. ]; 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 browser.close(); await new Promise((resolve, reject) => { proxyServer.close(error => { if (error) { reject(error); } else { resolve(undefined); } }); }); }); it('should proxy requests when configured', async () => { const {puppeteer, defaultBrowserOptions, server} = getTestState(); const emptyPageUrl = getEmptyPageUrl(server); browser = await puppeteer.launch({ ...defaultBrowserOptions, args: [...defaultArgs, `--proxy-server=${proxyServerUrl}`], }); const page = await browser.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([emptyPageUrl]); }); it('should respect proxy bypass list', async () => { const {puppeteer, defaultBrowserOptions, server} = getTestState(); const emptyPageUrl = getEmptyPageUrl(server); browser = await puppeteer.launch({ ...defaultBrowserOptions, args: [ ...defaultArgs, `--proxy-server=${proxyServerUrl}`, `--proxy-bypass-list=${new URL(emptyPageUrl).host}`, ], }); const page = await browser.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([]); }); describe('in incognito browser context', () => { it('should proxy requests when configured at browser level', async () => { const {puppeteer, defaultBrowserOptions, server} = getTestState(); const emptyPageUrl = getEmptyPageUrl(server); browser = await puppeteer.launch({ ...defaultBrowserOptions, args: [...defaultArgs, `--proxy-server=${proxyServerUrl}`], }); 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]); }); it('should respect proxy bypass list when configured at browser level', async () => { const {puppeteer, defaultBrowserOptions, server} = getTestState(); const emptyPageUrl = getEmptyPageUrl(server); browser = await puppeteer.launch({ ...defaultBrowserOptions, args: [ ...defaultArgs, `--proxy-server=${proxyServerUrl}`, `--proxy-bypass-list=${new URL(emptyPageUrl).host}`, ], }); const context = await browser.createIncognitoBrowserContext(); const page = await context.newPage(); const response = (await page.goto(emptyPageUrl))!; expect(response.ok()).toBe(true); expect(proxiedRequestUrls).toEqual([]); }); /** * See issues #7873, #7719, and #7698. */ it('should proxy requests when configured at context level', async () => { const {puppeteer, defaultBrowserOptions, server} = getTestState(); const emptyPageUrl = getEmptyPageUrl(server); browser = await puppeteer.launch({ ...defaultBrowserOptions, args: defaultArgs, }); 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]); }); it('should respect proxy bypass list when configured at context level', async () => { const {puppeteer, defaultBrowserOptions, server} = getTestState(); const emptyPageUrl = getEmptyPageUrl(server); browser = await puppeteer.launch({ ...defaultBrowserOptions, args: defaultArgs, }); 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([]); }); }); });