/** * Copyright 2018 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 {mkdtemp} from 'fs/promises'; import os from 'os'; import path from 'path'; import expect from 'expect'; import {PuppeteerLaunchOptions} from 'puppeteer-core/internal/node/PuppeteerNode.js'; import {rmSync} from 'puppeteer-core/internal/node/util/fs.js'; import {getTestState, launch} from './mocha-utils.js'; const TMP_FOLDER = path.join(os.tmpdir(), 'pptr_tmp_folder-'); const extensionPath = path.join(__dirname, '../assets', 'simple-extension'); const serviceWorkerExtensionPath = path.join( __dirname, '..', 'assets', 'serviceworkers', 'extension' ); describe('headful tests', function () { /* These tests fire up an actual browser so let's * allow a higher timeout */ this.timeout(20_000); let headfulOptions: PuppeteerLaunchOptions | undefined; let headlessOptions: PuppeteerLaunchOptions & {headless: boolean}; let extensionOptions: PuppeteerLaunchOptions & { headless: boolean; args: string[]; }; let forcedOopifOptions: PuppeteerLaunchOptions & { headless: boolean; devtools: boolean; args: string[]; }; let devtoolsOptions: PuppeteerLaunchOptions & { headless: boolean; devtools: boolean; }; const browsers: Array<() => Promise> = []; beforeEach(async () => { const {server, defaultBrowserOptions} = await getTestState({ skipLaunch: true, }); headfulOptions = Object.assign({}, defaultBrowserOptions, { headless: false, }); headlessOptions = Object.assign({}, defaultBrowserOptions, { headless: true, }); extensionOptions = Object.assign({}, defaultBrowserOptions, { headless: false, args: [ `--disable-extensions-except=${extensionPath}`, `--load-extension=${extensionPath}`, ], }); forcedOopifOptions = Object.assign({}, defaultBrowserOptions, { headless: false, devtools: true, args: [ `--host-rules=MAP oopifdomain 127.0.0.1`, `--isolate-origins=${server.PREFIX.replace( 'localhost', 'oopifdomain' )}`, ], }); devtoolsOptions = Object.assign({}, defaultBrowserOptions, { headless: false, devtools: true, }); }); async function launchBrowser(options: any) { const {browser, close} = await launch(options, {createContext: false}); browsers.push(close); return browser; } afterEach(async () => { await Promise.all( browsers.map((close, index) => { delete browsers[index]; return close(); }) ); }); describe('HEADFUL', function () { it('background_page target type should be available', async () => { const browserWithExtension = await launchBrowser(extensionOptions); const page = await browserWithExtension.newPage(); const backgroundPageTarget = await browserWithExtension.waitForTarget( target => { return target.type() === 'background_page'; } ); await page.close(); await browserWithExtension.close(); expect(backgroundPageTarget).toBeTruthy(); }); it('service_worker target type should be available', async () => { const browserWithExtension = await launchBrowser({ headless: false, args: [ `--disable-extensions-except=${serviceWorkerExtensionPath}`, `--load-extension=${serviceWorkerExtensionPath}`, ], }); const page = await browserWithExtension.newPage(); const serviceWorkerTarget = await browserWithExtension.waitForTarget( target => { return target.type() === 'service_worker'; } ); await page.close(); await browserWithExtension.close(); expect(serviceWorkerTarget).toBeTruthy(); }); it('target.page() should return a background_page', async function () { const browserWithExtension = await launchBrowser(extensionOptions); const backgroundPageTarget = await browserWithExtension.waitForTarget( target => { return target.type() === 'background_page'; } ); const page = (await backgroundPageTarget.page())!; expect( await page.evaluate(() => { return 2 * 3; }) ).toBe(6); expect( await page.evaluate(() => { return (globalThis as any).MAGIC; }) ).toBe(42); await browserWithExtension.close(); }); it('target.page() should return a DevTools page if custom isPageTarget is provided', async function () { const {puppeteer} = await getTestState({skipLaunch: true}); const originalBrowser = await launchBrowser(devtoolsOptions); const browserWSEndpoint = originalBrowser.wsEndpoint(); const browser = await puppeteer.connect({ browserWSEndpoint, _isPageTarget(target) { return ( target.type === 'other' && target.url.startsWith('devtools://') ); }, }); const devtoolsPageTarget = await browser.waitForTarget(target => { return target.type() === 'other'; }); const page = (await devtoolsPageTarget.page())!; expect( await page.evaluate(() => { return 2 * 3; }) ).toBe(6); expect(await browser.pages()).toContainEqual(page); }); it('should have default url when launching browser', async function () { const browser = await launchBrowser(extensionOptions); const pages = (await browser.pages()).map((page: {url: () => any}) => { return page.url(); }); expect(pages).toEqual(['about:blank']); }); it('headless should be able to read cookies written by headful', async () => { /* Needs investigation into why but this fails consistently on Windows CI. */ const {server} = await getTestState({skipLaunch: true}); const userDataDir = await mkdtemp(TMP_FOLDER); // Write a cookie in headful chrome const headfulBrowser = await launchBrowser( Object.assign({userDataDir}, headfulOptions) ); const headfulPage = await headfulBrowser.newPage(); await headfulPage.goto(server.EMPTY_PAGE); await headfulPage.evaluate(() => { return (document.cookie = 'foo=true; expires=Fri, 31 Dec 9999 23:59:59 GMT'); }); await headfulBrowser.close(); // Read the cookie from headless chrome const headlessBrowser = await launchBrowser( Object.assign({userDataDir}, headlessOptions) ); const headlessPage = await headlessBrowser.newPage(); await headlessPage.goto(server.EMPTY_PAGE); const cookie = await headlessPage.evaluate(() => { return document.cookie; }); await headlessBrowser.close(); // This might throw. See https://github.com/puppeteer/puppeteer/issues/2778 try { rmSync(userDataDir); } catch {} expect(cookie).toBe('foo=true'); }); // TODO: Support OOOPIF. @see https://github.com/puppeteer/puppeteer/issues/2548 it.skip('OOPIF: should report google.com frame', async () => { const {server} = await getTestState({skipLaunch: true}); // https://google.com is isolated by default in Chromium embedder. const browser = await launchBrowser(headfulOptions); const page = await browser.newPage(); await page.goto(server.EMPTY_PAGE); await page.setRequestInterception(true); page.on('request', (r: {respond: (arg0: {body: string}) => any}) => { return r.respond({body: 'YO, GOOGLE.COM'}); }); await page.evaluate(() => { const frame = document.createElement('iframe'); frame.setAttribute('src', 'https://google.com/'); document.body.appendChild(frame); return new Promise(x => { return (frame.onload = x); }); }); await page.waitForSelector('iframe[src="https://google.com/"]'); const urls = page .frames() .map((frame: {url: () => any}) => { return frame.url(); }) .sort(); expect(urls).toEqual([server.EMPTY_PAGE, 'https://google.com/']); }); it('OOPIF: should expose events within OOPIFs', async () => { const {server} = await getTestState({skipLaunch: true}); const browser = await launchBrowser(forcedOopifOptions); const page = await browser.newPage(); // Setup our session listeners to observe OOPIF activity. const session = await page.target().createCDPSession(); const networkEvents: any[] = []; const otherSessions: any[] = []; await session.send('Target.setAutoAttach', { autoAttach: true, flatten: true, waitForDebuggerOnStart: true, }); session.on( 'sessionattached', async (session: { on: (arg0: string, arg1: (params: any) => number) => void; send: (arg0: string) => any; }) => { otherSessions.push(session); session.on('Network.requestWillBeSent', (params: any) => { return networkEvents.push(params); }); await session.send('Network.enable'); await session.send('Runtime.runIfWaitingForDebugger'); } ); // Navigate to the empty page and add an OOPIF iframe with at least one request. await page.goto(server.EMPTY_PAGE); await page.evaluate((frameUrl: string) => { const frame = document.createElement('iframe'); frame.setAttribute('src', frameUrl); document.body.appendChild(frame); return new Promise((x, y) => { frame.onload = x; frame.onerror = y; }); }, server.PREFIX.replace('localhost', 'oopifdomain') + '/one-style.html'); await page.waitForSelector('iframe'); // Ensure we found the iframe session. expect(otherSessions).toHaveLength(1); // Resume the iframe and trigger another request. const iframeSession = otherSessions[0]!; await iframeSession.send('Runtime.evaluate', { expression: `fetch('/fetch')`, awaitPromise: true, }); await browser.close(); const requests = networkEvents.map(event => { return event.request.url; }); expect(requests).toContain(`http://oopifdomain:${server.PORT}/fetch`); }); it('should close browser with beforeunload page', async () => { const {server} = await getTestState({skipLaunch: true}); const browser = await launchBrowser(headfulOptions); const page = await browser.newPage(); await page.goto(server.PREFIX + '/beforeunload.html'); // We have to interact with a page so that 'beforeunload' handlers // fire. await page.click('body'); await browser.close(); }); it('should open devtools when "devtools: true" option is given', async () => { const browser = await launchBrowser( Object.assign({devtools: true}, headfulOptions) ); const context = await browser.createIncognitoBrowserContext(); await Promise.all([ context.newPage(), browser.waitForTarget((target: {url: () => string | string[]}) => { return target.url().includes('devtools://'); }), ]); await browser.close(); }); }); describe('Page.bringToFront', function () { it('should work', async () => { const browser = await launchBrowser(headfulOptions); const page1 = await browser.newPage(); const page2 = await browser.newPage(); await page1.bringToFront(); expect( await page1.evaluate(() => { return document.visibilityState; }) ).toBe('visible'); expect( await page2.evaluate(() => { return document.visibilityState; }) ).toBe('hidden'); await page2.bringToFront(); expect( await page1.evaluate(() => { return document.visibilityState; }) ).toBe('hidden'); expect( await page2.evaluate(() => { return document.visibilityState; }) ).toBe('visible'); await page1.close(); await page2.close(); await browser.close(); }); }); describe('Page.screenshot', function () { it('should run in parallel in multiple pages', async () => { const {server} = await getTestState({skipLaunch: true}); const browser = await launchBrowser(headfulOptions); const context = await browser.createIncognitoBrowserContext(); const N = 2; const pages = await Promise.all( Array(N) .fill(0) .map(async () => { const page = await context.newPage(); await page.goto(server.PREFIX + '/grid.html'); return page; }) ); const promises = []; for (let i = 0; i < N; ++i) { promises.push( pages[i]!.screenshot({ clip: {x: 50 * i, y: 0, width: 50, height: 50}, }) ); } const screenshots = await Promise.all(promises); for (let i = 0; i < N; ++i) { expect(screenshots[i]).toBeGolden(`grid-cell-${i}.png`); } await Promise.all( pages.map(page => { return page.close(); }) ); await browser.close(); }); }); });