feat(ppfox): implement browser contexts (#3872)

BrowserContexts are necessary to run Puppeteer tests against Puppeteer-Firefox
This commit is contained in:
Andrey Lushnikov 2019-02-01 14:09:18 -08:00 committed by GitHub
parent b0e8084650
commit bd347558bc
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 315 additions and 9 deletions

View File

@ -1,4 +1,4 @@
const {helper} = require('./helper'); const {helper, assert} = require('./helper');
const {Page} = require('./Page'); const {Page} = require('./Page');
const EventEmitter = require('events'); const EventEmitter = require('events');
@ -9,7 +9,21 @@ class Browser extends EventEmitter {
* @param {?Puppeteer.ChildProcess} process * @param {?Puppeteer.ChildProcess} process
* @param {function():void} closeCallback * @param {function():void} closeCallback
*/ */
constructor(connection, defaultViewport, process, closeCallback) { static async create(connection, defaultViewport, process, closeCallback) {
const {browserContextIds} = await connection.send('Browser.getBrowserContexts');
const browser = new Browser(connection, browserContextIds, defaultViewport, process, closeCallback);
await connection.send('Browser.enable');
return browser;
}
/**
* @param {!Puppeteer.Connection} connection
* @param {!Array<string>} browserContextIds
* @param {?Puppeteer.Viewport} defaultViewport
* @param {?Puppeteer.ChildProcess} process
* @param {function():void} closeCallback
*/
constructor(connection, browserContextIds, defaultViewport, process, closeCallback) {
super(); super();
this._connection = connection; this._connection = connection;
this._defaultViewport = defaultViewport; this._defaultViewport = defaultViewport;
@ -19,6 +33,12 @@ class Browser extends EventEmitter {
/** @type {!Map<string, ?Target>} */ /** @type {!Map<string, ?Target>} */
this._pageTargets = new Map(); this._pageTargets = new Map();
this._defaultContext = new BrowserContext(this._connection, this, null);
/** @type {!Map<string, !BrowserContext>} */
this._contexts = new Map();
for (const browserContextId of browserContextIds)
this._contexts.set(browserContextId, new BrowserContext(this._connection, this, browserContextId));
this._eventListeners = [ this._eventListeners = [
helper.addEventListener(this._connection, 'Browser.tabOpened', this._onTabOpened.bind(this)), helper.addEventListener(this._connection, 'Browser.tabOpened', this._onTabOpened.bind(this)),
helper.addEventListener(this._connection, 'Browser.tabClosed', this._onTabClosed.bind(this)), helper.addEventListener(this._connection, 'Browser.tabClosed', this._onTabClosed.bind(this)),
@ -26,6 +46,32 @@ class Browser extends EventEmitter {
]; ];
} }
/**
* @return {!BrowserContext}
*/
async createIncognitoBrowserContext() {
const {browserContextId} = await this._connection.send('Browser.createBrowserContext');
const context = new BrowserContext(this._connection, this, browserContextId);
this._contexts.set(browserContextId, context);
return context;
}
/**
* @return {!Array<!BrowserContext>}
*/
browserContexts() {
return [this._defaultContext, ...Array.from(this._contexts.values())];
}
defaultBrowserContext() {
return this._defaultContext;
}
async _disposeContext(browserContextId) {
await this._connection.send('Browser.removeBrowserContext', {browserContextId});
this._contexts.delete(browserContextId);
}
/** /**
* @return {!Promise<string>} * @return {!Promise<string>}
*/ */
@ -83,8 +129,21 @@ class Browser extends EventEmitter {
} }
} }
async newPage() { /**
const {pageId} = await this._connection.send('Browser.newPage'); * @return {Promise<Page>}
*/
newPage() {
return this._createPageInContext(this._defaultContext._browserContextId);
}
/**
* @param {?string} browserContextId
* @return {Promise<Page>}
*/
async _createPageInContext(browserContextId) {
const {pageId} = await this._connection.send('Browser.newPage', {
browserContextId: browserContextId || undefined
});
const target = this._pageTargets.get(pageId); const target = this._pageTargets.get(pageId);
return await target.page(); return await target.page();
} }
@ -98,22 +157,26 @@ class Browser extends EventEmitter {
return Array.from(this._pageTargets.values()); return Array.from(this._pageTargets.values());
} }
_onTabOpened({pageId, url}) { _onTabOpened({pageId, url, browserContextId}) {
const target = new Target(this._connection, this, pageId, url); const context = browserContextId ? this._contexts.get(browserContextId) : this._defaultContext;
const target = new Target(this._connection, this, context, pageId, url);
this._pageTargets.set(pageId, target); this._pageTargets.set(pageId, target);
this.emit(Browser.Events.TargetCreated, target); this.emit(Browser.Events.TargetCreated, target);
context.emit(BrowserContext.Events.TargetCreated, target);
} }
_onTabClosed({pageId}) { _onTabClosed({pageId}) {
const target = this._pageTargets.get(pageId); const target = this._pageTargets.get(pageId);
this._pageTargets.delete(pageId); this._pageTargets.delete(pageId);
this.emit(Browser.Events.TargetDestroyed, target); this.emit(Browser.Events.TargetDestroyed, target);
target.browserContext().emit(BrowserContext.Events.TargetDestroyed, target);
} }
_onTabNavigated({pageId, url}) { _onTabNavigated({pageId, url}) {
const target = this._pageTargets.get(pageId); const target = this._pageTargets.get(pageId);
target._url = url; target._url = url;
this.emit(Browser.Events.TargetChanged, target); this.emit(Browser.Events.TargetChanged, target);
target.browserContext().emit(BrowserContext.Events.TargetChanged, target);
} }
async close() { async close() {
@ -134,11 +197,13 @@ class Target {
* *
* @param {*} connection * @param {*} connection
* @param {!Browser} browser * @param {!Browser} browser
* @param {!BrowserContext} context
* @param {string} pageId * @param {string} pageId
* @param {string} url * @param {string} url
*/ */
constructor(connection, browser, pageId, url) { constructor(connection, browser, context, pageId, url) {
this._browser = browser; this._browser = browser;
this._context = context;
this._connection = connection; this._connection = connection;
this._pageId = pageId; this._pageId = pageId;
/** @type {?Promise<!Page>} */ /** @type {?Promise<!Page>} */
@ -157,6 +222,13 @@ class Target {
return this._url; return this._url;
} }
/**
* @return {!BrowserContext}
*/
browserContext() {
return this._context;
}
async page() { async page() {
if (!this._pagePromise) if (!this._pagePromise)
this._pagePromise = Page.create(this._connection, this, this._pageId, this._browser._defaultViewport); this._pagePromise = Page.create(this._connection, this, this._pageId, this._browser._defaultViewport);
@ -168,4 +240,76 @@ class Target {
} }
} }
class BrowserContext extends EventEmitter {
/**
* @param {!Puppeteer.Connection} connection
* @param {!Browser} browser
* @param {?string} browserContextId
*/
constructor(connection, browser, browserContextId) {
super();
this._connection = connection;
this._browser = browser;
this._browserContextId = browserContextId;
}
/**
* @return {Array<Target>}
*/
targets() {
return this._browser.targets().filter(target => target.browserContext() === this);
}
/**
* @return {Promise<Array<Puppeteer.Page>>}
*/
async pages() {
const pages = await Promise.all(
this.targets()
.filter(target => target.type() === 'page')
.map(target => target.page())
);
return pages.filter(page => !!page);
}
/**
* @param {function(Target):boolean} predicate
* @param {{timeout?: number}=} options
* @return {!Promise<Target>}
*/
waitForTarget(predicate, options) {
return this._browser.waitForTarget(target => target.browserContext() === this && predicate(target), options);
}
/**
* @return {boolean}
*/
isIncognito() {
return !!this._browserContextId;
}
newPage() {
return this._browser._createPageInContext(this._browserContextId);
}
/**
* @return {!Browser}
*/
browser() {
return this._browser;
}
async close() {
assert(this._browserContextId, 'Non-incognito contexts cannot be closed!');
await this._browser._disposeContext(this._browserContextId);
}
}
/** @enum {string} */
BrowserContext.Events = {
TargetCreated: 'targetcreated',
TargetChanged: 'targetchanged',
TargetDestroyed: 'targetdestroyed'
}
module.exports = {Browser, Target}; module.exports = {Browser, Target};

View File

@ -116,7 +116,7 @@ class Launcher {
const port = await waitForWSEndpoint(firefoxProcess, 30000); const port = await waitForWSEndpoint(firefoxProcess, 30000);
const transport = await FirefoxTransport.create(parseInt(port, 10)); const transport = await FirefoxTransport.create(parseInt(port, 10));
connection = new Connection(transport, slowMo); connection = new Connection(transport, slowMo);
const browser = new Browser(connection, defaultViewport, firefoxProcess, killFirefox); const browser = await Browser.create(connection, defaultViewport, firefoxProcess, killFirefox);
if (ignoreHTTPSErrors) if (ignoreHTTPSErrors)
await connection.send('Browser.setIgnoreHTTPSErrors', {enabled: true}); await connection.send('Browser.setIgnoreHTTPSErrors', {enabled: true});
if (!browser.targets().length) if (!browser.targets().length)

View File

@ -8,7 +8,7 @@
"node": ">=8.9.4" "node": ">=8.9.4"
}, },
"puppeteer": { "puppeteer": {
"firefox_revision": "e5fdeac984d4f966caafcdbc9b14da7a7f73fbed" "firefox_revision": "2a2b2cd2c5e5b7062e7ede93b235a5a062d4dc9a"
}, },
"scripts": { "scripts": {
"install": "node install.js", "install": "node install.js",

View File

@ -0,0 +1,160 @@
/**
* 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.
*/
const utils = require('./utils');
const {TimeoutError} = require('../Errors');
module.exports.addTests = function({testRunner, expect, puppeteer, product}) {
const {describe, xdescribe, fdescribe} = testRunner;
const {it, fit, xit} = testRunner;
const {beforeAll, beforeEach, afterAll, afterEach} = testRunner;
const FFOX = product === 'firefox';
const CHROME = product === 'chromium';
describe('BrowserContext', function() {
it('should have default context', async function({browser, server}) {
expect(browser.browserContexts().length).toBe(1);
const defaultContext = browser.browserContexts()[0];
expect(defaultContext.isIncognito()).toBe(false);
let error = null;
await defaultContext.close().catch(e => error = e);
expect(browser.defaultBrowserContext()).toBe(defaultContext);
expect(error.message).toContain('cannot be closed');
});
it('should create new incognito context', async function({browser, server}) {
expect(browser.browserContexts().length).toBe(1);
const context = await browser.createIncognitoBrowserContext();
expect(context.isIncognito()).toBe(true);
expect(browser.browserContexts().length).toBe(2);
expect(browser.browserContexts().indexOf(context) !== -1).toBe(true);
await context.close();
expect(browser.browserContexts().length).toBe(1);
});
it('should close all belonging targets once closing context', async function({browser, server}) {
expect((await browser.pages()).length).toBe(1);
const context = await browser.createIncognitoBrowserContext();
await context.newPage();
expect((await browser.pages()).length).toBe(2);
expect((await context.pages()).length).toBe(1);
await context.close();
expect((await browser.pages()).length).toBe(1);
});
it('window.open should use parent tab context', async function({browser, server}) {
const context = await browser.createIncognitoBrowserContext();
const page = await context.newPage();
await page.goto(server.EMPTY_PAGE);
const [popupTarget] = await Promise.all([
utils.waitEvent(browser, 'targetcreated'),
page.evaluate(url => window.open(url), server.EMPTY_PAGE)
]);
expect(popupTarget.browserContext()).toBe(context);
await context.close();
});
it('should fire target events', async function({browser, server}) {
const context = await browser.createIncognitoBrowserContext();
const events = [];
context.on('targetcreated', target => events.push('CREATED: ' + target.url()));
context.on('targetchanged', target => events.push('CHANGED: ' + target.url()));
context.on('targetdestroyed', target => events.push('DESTROYED: ' + target.url()));
const page = await context.newPage();
await page.goto(server.EMPTY_PAGE);
await page.close();
expect(events).toEqual([
'CREATED: about:blank',
`CHANGED: ${server.EMPTY_PAGE}`,
`DESTROYED: ${server.EMPTY_PAGE}`
]);
await context.close();
});
it('should wait for a target', async function({browser, server}) {
const context = await browser.createIncognitoBrowserContext();
let resolved = false;
const targetPromise = context.waitForTarget(target => target.url() === server.EMPTY_PAGE);
targetPromise.then(() => resolved = true);
const page = await context.newPage();
expect(resolved).toBe(false);
await page.goto(server.EMPTY_PAGE);
const target = await targetPromise;
expect(await target.page()).toBe(page);
await context.close();
});
it('should timeout waiting for a non-existent target', async function({browser, server}) {
const context = await browser.createIncognitoBrowserContext();
const error = await context.waitForTarget(target => target.url() === server.EMPTY_PAGE, {timeout: 1}).catch(e => e);
expect(error).toBeInstanceOf(TimeoutError);
await context.close();
});
it('should isolate localStorage and cookies', async function({browser, server}) {
// Create two incognito contexts.
const context1 = await browser.createIncognitoBrowserContext();
const context2 = await browser.createIncognitoBrowserContext();
expect(context1.targets().length).toBe(0);
expect(context2.targets().length).toBe(0);
// Create a page in first incognito context.
const page1 = await context1.newPage();
await page1.goto(server.EMPTY_PAGE);
await page1.evaluate(() => {
localStorage.setItem('name', 'page1');
document.cookie = 'name=page1';
});
expect(context1.targets().length).toBe(1);
expect(context2.targets().length).toBe(0);
// Create a page in second incognito context.
const page2 = await context2.newPage();
await page2.goto(server.EMPTY_PAGE);
await page2.evaluate(() => {
localStorage.setItem('name', 'page2');
document.cookie = 'name=page2';
});
expect(context1.targets().length).toBe(1);
expect(context1.targets()[0]).toBe(page1.target());
expect(context2.targets().length).toBe(1);
expect(context2.targets()[0]).toBe(page2.target());
// Make sure pages don't share localstorage or cookies.
expect(await page1.evaluate(() => localStorage.getItem('name'))).toBe('page1');
expect(await page1.evaluate(() => document.cookie)).toBe('name=page1');
expect(await page2.evaluate(() => localStorage.getItem('name'))).toBe('page2');
expect(await page2.evaluate(() => document.cookie)).toBe('name=page2');
// Cleanup contexts.
await Promise.all([
context1.close(),
context2.close()
]);
expect(browser.browserContexts().length).toBe(1);
});
(FFOX ? xit : it)('should work across sessions', async function({browser, server}) {
expect(browser.browserContexts().length).toBe(1);
const context = await browser.createIncognitoBrowserContext();
expect(browser.browserContexts().length).toBe(2);
const remoteBrowser = await puppeteer.connect({
browserWSEndpoint: browser.wsEndpoint()
});
const contexts = remoteBrowser.browserContexts();
expect(contexts.length).toBe(2);
await remoteBrowser.disconnect();
await context.close();
});
});
};

View File

@ -38,6 +38,7 @@ module.exports.addTests = ({testRunner, product, puppeteer}) => testRunner.descr
}; };
if (product === 'firefox' && state.defaultBrowserOptions.executablePath) { if (product === 'firefox' && state.defaultBrowserOptions.executablePath) {
await require('../misc/install-preferences')(state.defaultBrowserOptions.executablePath); await require('../misc/install-preferences')(state.defaultBrowserOptions.executablePath);
console.log('RUNNING CUSTOM FIREFOX: ' + state.defaultBrowserOptions.executablePath);
} }
}); });
afterAll(state => { afterAll(state => {
@ -58,6 +59,7 @@ module.exports.addTests = ({testRunner, product, puppeteer}) => testRunner.descr
}); });
require('./browser.spec.js').addTests({testRunner, expect, product, puppeteer}); require('./browser.spec.js').addTests({testRunner, expect, product, puppeteer});
require('./browsercontext.spec.js').addTests({testRunner, expect, product, puppeteer});
describe('Page', () => { describe('Page', () => {
beforeEach(async state => { beforeEach(async state => {