feat(Browser): introduce Browser.pages() (#554)
This patch: - introduces Target class that represents any inspectable target, such as service worker or page - emits events when targets come and go - introduces target.page() to instantiate a page from a target Fixes #386, fixes #443.
This commit is contained in:
parent
273c733237
commit
32398d11bd
46
docs/api.md
46
docs/api.md
@ -13,9 +13,14 @@
|
|||||||
* [puppeteer.executablePath()](#puppeteerexecutablepath)
|
* [puppeteer.executablePath()](#puppeteerexecutablepath)
|
||||||
* [puppeteer.launch([options])](#puppeteerlaunchoptions)
|
* [puppeteer.launch([options])](#puppeteerlaunchoptions)
|
||||||
- [class: Browser](#class-browser)
|
- [class: Browser](#class-browser)
|
||||||
|
* [event: 'targetchanged'](#event-targetchanged)
|
||||||
|
* [event: 'targetcreated'](#event-targetcreated)
|
||||||
|
* [event: 'targetdestroyed'](#event-targetdestroyed)
|
||||||
* [browser.close()](#browserclose)
|
* [browser.close()](#browserclose)
|
||||||
* [browser.disconnect()](#browserdisconnect)
|
* [browser.disconnect()](#browserdisconnect)
|
||||||
* [browser.newPage()](#browsernewpage)
|
* [browser.newPage()](#browsernewpage)
|
||||||
|
* [browser.pages()](#browserpages)
|
||||||
|
* [browser.targets()](#browsertargets)
|
||||||
* [browser.version()](#browserversion)
|
* [browser.version()](#browserversion)
|
||||||
* [browser.wsEndpoint()](#browserwsendpoint)
|
* [browser.wsEndpoint()](#browserwsendpoint)
|
||||||
- [class: Page](#class-page)
|
- [class: Page](#class-page)
|
||||||
@ -175,6 +180,10 @@
|
|||||||
* [response.status](#responsestatus)
|
* [response.status](#responsestatus)
|
||||||
* [response.text()](#responsetext)
|
* [response.text()](#responsetext)
|
||||||
* [response.url](#responseurl)
|
* [response.url](#responseurl)
|
||||||
|
- [class: Target](#class-target)
|
||||||
|
* [target.page()](#targetpage)
|
||||||
|
* [target.type()](#targettype)
|
||||||
|
* [target.url()](#targeturl)
|
||||||
|
|
||||||
<!-- tocstop -->
|
<!-- tocstop -->
|
||||||
|
|
||||||
@ -278,6 +287,21 @@ puppeteer.launch().then(async browser => {
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
|
#### event: 'targetchanged'
|
||||||
|
- <[Target]>
|
||||||
|
|
||||||
|
Emitted when the url of a target changes.
|
||||||
|
|
||||||
|
#### event: 'targetcreated'
|
||||||
|
- <[Target]>
|
||||||
|
|
||||||
|
Emitted when a target is created, for example when a new page is opened by [`window.open`](https://developer.mozilla.org/en-US/docs/Web/API/Window/open) or [`browser.newPage`](#browsernewpage).
|
||||||
|
|
||||||
|
#### event: 'targetdestroyed'
|
||||||
|
- <[Target]>
|
||||||
|
|
||||||
|
Emitted when a target is destroyed, for example when a page is closed.
|
||||||
|
|
||||||
#### browser.close()
|
#### browser.close()
|
||||||
- returns: <[Promise]>
|
- returns: <[Promise]>
|
||||||
|
|
||||||
@ -290,6 +314,12 @@ Disconnects Puppeteer from the browser, but leaves the Chromium process running.
|
|||||||
#### browser.newPage()
|
#### browser.newPage()
|
||||||
- returns: <[Promise]<[Page]>> Promise which resolves to a new [Page] object.
|
- returns: <[Promise]<[Page]>> Promise which resolves to a new [Page] object.
|
||||||
|
|
||||||
|
#### browser.pages()
|
||||||
|
- returns: <[Promise]<[Array]<[Page]>>> Promise which resolves to an array of all open pages.
|
||||||
|
|
||||||
|
#### browser.targets()
|
||||||
|
- returns: <[Array]<[Target]>> An array of all active targets.
|
||||||
|
|
||||||
#### browser.version()
|
#### browser.version()
|
||||||
- returns: <[Promise]<[string]>> For headless Chromium, this is similar to `HeadlessChrome/61.0.3153.0`. For non-headless, this is similar to `Chrome/61.0.3153.0`.
|
- returns: <[Promise]<[string]>> For headless Chromium, this is similar to `HeadlessChrome/61.0.3153.0`. For non-headless, this is similar to `Chrome/61.0.3153.0`.
|
||||||
|
|
||||||
@ -1896,6 +1926,21 @@ Contains the status code of the response (e.g., 200 for a success).
|
|||||||
|
|
||||||
Contains the URL of the response.
|
Contains the URL of the response.
|
||||||
|
|
||||||
|
### class: Target
|
||||||
|
|
||||||
|
#### target.page()
|
||||||
|
- returns: <[Promise]<[Page]>>
|
||||||
|
|
||||||
|
If the target is not of type `"page"`, returns `null`.
|
||||||
|
|
||||||
|
#### target.type()
|
||||||
|
- returns: <[string]>
|
||||||
|
|
||||||
|
Identifies what kind of target this is. Can be `"page"`, `"service_worker"`, or `"other"`.
|
||||||
|
|
||||||
|
#### target.url()
|
||||||
|
- returns: <[string]>
|
||||||
|
|
||||||
[Array]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array "Array"
|
[Array]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array "Array"
|
||||||
[boolean]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Data_structures#Boolean_type "Boolean"
|
[boolean]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Data_structures#Boolean_type "Boolean"
|
||||||
[Buffer]: https://nodejs.org/api/buffer.html#buffer_class_buffer "Buffer"
|
[Buffer]: https://nodejs.org/api/buffer.html#buffer_class_buffer "Buffer"
|
||||||
@ -1927,3 +1972,4 @@ Contains the URL of the response.
|
|||||||
[UIEvent.detail]: https://developer.mozilla.org/en-US/docs/Web/API/UIEvent/detail "UIEvent.detail"
|
[UIEvent.detail]: https://developer.mozilla.org/en-US/docs/Web/API/UIEvent/detail "UIEvent.detail"
|
||||||
[Serializable]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/JSON/stringify#Description "Serializable"
|
[Serializable]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/JSON/stringify#Description "Serializable"
|
||||||
[Touchscreen]: #class-touchscreen "Touchscreen"
|
[Touchscreen]: #class-touchscreen "Touchscreen"
|
||||||
|
[Target]: #class-target "Target"
|
||||||
|
149
lib/Browser.js
149
lib/Browser.js
@ -31,6 +31,54 @@ class Browser extends EventEmitter {
|
|||||||
this._screenshotTaskQueue = new TaskQueue();
|
this._screenshotTaskQueue = new TaskQueue();
|
||||||
this._connection = connection;
|
this._connection = connection;
|
||||||
this._closeCallback = closeCallback || new Function();
|
this._closeCallback = closeCallback || new Function();
|
||||||
|
/** @type {Map<string, Target>} */
|
||||||
|
this._targets = new Map();
|
||||||
|
this._connection.on('Target.targetCreated', this._targetCreated.bind(this));
|
||||||
|
this._connection.on('Target.targetDestroyed', this._targetDestroyed.bind(this));
|
||||||
|
this._connection.on('Target.targetInfoChanged', this._targetInfoChanged.bind(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {!Puppeteer.Connection} connection
|
||||||
|
* @param {boolean} ignoreHTTPSErrors
|
||||||
|
* @param {function()=} closeCallback
|
||||||
|
*/
|
||||||
|
static async create(connection, ignoreHTTPSErrors, closeCallback) {
|
||||||
|
const browser = new Browser(connection, ignoreHTTPSErrors, closeCallback);
|
||||||
|
await connection.send('Target.setDiscoverTargets', {discover: true});
|
||||||
|
return browser;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {{targetInfo: !Target.TargetInfo}} event
|
||||||
|
*/
|
||||||
|
async _targetCreated(event) {
|
||||||
|
const target = new Target(this, event.targetInfo);
|
||||||
|
console.assert(!this._targets.has(event.targetInfo.targetId), 'Target should not exist before targetCreated');
|
||||||
|
this._targets.set(event.targetInfo.targetId, target);
|
||||||
|
|
||||||
|
if (await target._initializedPromise)
|
||||||
|
this.emit(Browser.Events.TargetCreated, target);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {{targetId: string}} event
|
||||||
|
*/
|
||||||
|
async _targetDestroyed(event) {
|
||||||
|
const target = this._targets.get(event.targetId);
|
||||||
|
target._initializedCallback(false);
|
||||||
|
this._targets.delete(event.targetId);
|
||||||
|
if (await target._initializedPromise)
|
||||||
|
this.emit(Browser.Events.TargetDestroyed, target);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {{targetInfo: !Target.TargetInfo}} event
|
||||||
|
*/
|
||||||
|
_targetInfoChanged(event) {
|
||||||
|
const target = this._targets.get(event.targetInfo.targetId);
|
||||||
|
console.assert(target, 'target should exist before targetInfoChanged');
|
||||||
|
target._targetInfoChanged(event.targetInfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -45,8 +93,25 @@ class Browser extends EventEmitter {
|
|||||||
*/
|
*/
|
||||||
async newPage() {
|
async newPage() {
|
||||||
const {targetId} = await this._connection.send('Target.createTarget', {url: 'about:blank'});
|
const {targetId} = await this._connection.send('Target.createTarget', {url: 'about:blank'});
|
||||||
const client = await this._connection.createSession(targetId);
|
const target = await this._targets.get(targetId);
|
||||||
return await Page.create(client, this._ignoreHTTPSErrors, this._appMode, this._screenshotTaskQueue);
|
console.assert(await target._initializedPromise, 'Failed to create target for page');
|
||||||
|
const page = await target.page();
|
||||||
|
return page;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return {!Array<!Target>}
|
||||||
|
*/
|
||||||
|
targets() {
|
||||||
|
return Array.from(this._targets.values()).filter(target => target._isInitialized);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return {!Promise<!Array<!Page>>}
|
||||||
|
*/
|
||||||
|
async pages() {
|
||||||
|
const pages = await Promise.all(this.targets().map(target => target.page()));
|
||||||
|
return pages.filter(page => !!page);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -67,6 +132,13 @@ class Browser extends EventEmitter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @enum {string} */
|
||||||
|
Browser.Events = {
|
||||||
|
TargetCreated: 'targetcreated',
|
||||||
|
TargetDestroyed: 'targetdestroyed',
|
||||||
|
TargetChanged: 'targetchanged'
|
||||||
|
};
|
||||||
|
|
||||||
helper.tracePublicAPI(Browser);
|
helper.tracePublicAPI(Browser);
|
||||||
|
|
||||||
class TaskQueue {
|
class TaskQueue {
|
||||||
@ -84,4 +156,77 @@ class TaskQueue {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class Target {
|
||||||
|
/**
|
||||||
|
* @param {!Browser} browser
|
||||||
|
* @param {!Target.TargetInfo} targetInfo
|
||||||
|
*/
|
||||||
|
constructor(browser, targetInfo) {
|
||||||
|
this._browser = browser;
|
||||||
|
this._targetInfo = targetInfo;
|
||||||
|
/** @type {?Promise<!Page>} */
|
||||||
|
this._pagePromise = null;
|
||||||
|
this._initializedPromise = new Promise(fulfill => this._initializedCallback = fulfill);
|
||||||
|
this._isInitialized = this._targetInfo.type !== 'page' || this._targetInfo.url !== '';
|
||||||
|
if (this._isInitialized)
|
||||||
|
this._initializedCallback(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return {!Promise<?Page>}
|
||||||
|
*/
|
||||||
|
async page() {
|
||||||
|
if (this._targetInfo.type === 'page' && !this._pagePromise) {
|
||||||
|
this._pagePromise = this._browser._connection.createSession(this._targetInfo.targetId)
|
||||||
|
.then(client => Page.create(client, this._browser._ignoreHTTPSErrors, this._browser._appMode, this._browser._screenshotTaskQueue));
|
||||||
|
}
|
||||||
|
return this._pagePromise;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return {string}
|
||||||
|
*/
|
||||||
|
url() {
|
||||||
|
return this._targetInfo.url;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return {"page"|"service_worker"|"other"}
|
||||||
|
*/
|
||||||
|
type() {
|
||||||
|
const type = this._targetInfo.type;
|
||||||
|
if (type === 'page' || type === 'service_worker')
|
||||||
|
return type;
|
||||||
|
return 'other';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {!Target.TargetInfo} targetInfo
|
||||||
|
*/
|
||||||
|
_targetInfoChanged(targetInfo) {
|
||||||
|
const previousURL = this._targetInfo.url;
|
||||||
|
this._targetInfo = targetInfo;
|
||||||
|
|
||||||
|
if (!this._isInitialized && (this._targetInfo.type !== 'page' || this._targetInfo.url !== '')) {
|
||||||
|
this._isInitialized = true;
|
||||||
|
this._initializedCallback(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (previousURL !== targetInfo.url)
|
||||||
|
this._browser.emit(Browser.Events.TargetChanged, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
helper.tracePublicAPI(Target);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} Target.TargetInfo
|
||||||
|
* @property {string} type
|
||||||
|
* @property {string} targetId
|
||||||
|
* @property {string} title
|
||||||
|
* @property {string} url
|
||||||
|
* @property {boolean} attached
|
||||||
|
*/
|
||||||
|
|
||||||
module.exports = { Browser, TaskQueue };
|
module.exports = { Browser, TaskQueue };
|
||||||
|
@ -25,9 +25,10 @@ const readFileAsync = helper.promisify(fs.readFile);
|
|||||||
class FrameManager extends EventEmitter {
|
class FrameManager extends EventEmitter {
|
||||||
/**
|
/**
|
||||||
* @param {!Puppeteer.Session} client
|
* @param {!Puppeteer.Session} client
|
||||||
|
* @param {{frame: Object, childFrames: ?Array}} frameTree
|
||||||
* @param {!Puppeteer.Page} page
|
* @param {!Puppeteer.Page} page
|
||||||
*/
|
*/
|
||||||
constructor(client, page) {
|
constructor(client, frameTree, page) {
|
||||||
super();
|
super();
|
||||||
this._client = client;
|
this._client = client;
|
||||||
this._page = page;
|
this._page = page;
|
||||||
@ -40,6 +41,22 @@ class FrameManager extends EventEmitter {
|
|||||||
this._client.on('Page.frameNavigated', event => this._onFrameNavigated(event.frame));
|
this._client.on('Page.frameNavigated', event => this._onFrameNavigated(event.frame));
|
||||||
this._client.on('Page.frameDetached', event => this._onFrameDetached(event.frameId));
|
this._client.on('Page.frameDetached', event => this._onFrameDetached(event.frameId));
|
||||||
this._client.on('Runtime.executionContextCreated', event => this._onExecutionContextCreated(event.context));
|
this._client.on('Runtime.executionContextCreated', event => this._onExecutionContextCreated(event.context));
|
||||||
|
|
||||||
|
this._handleFrameTree(frameTree);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {{frame: Object, childFrames: ?Array}} frameTree
|
||||||
|
*/
|
||||||
|
_handleFrameTree(frameTree) {
|
||||||
|
if (frameTree.frame.parentId)
|
||||||
|
this._onFrameAttached(frameTree.frame.id, frameTree.frame.parentId);
|
||||||
|
this._onFrameNavigated(frameTree.frame);
|
||||||
|
if (!frameTree.childFrames)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (const child of frameTree.childFrames)
|
||||||
|
this._handleFrameTree(child);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -134,7 +134,7 @@ class Launcher {
|
|||||||
const connectionDelay = options.slowMo || 0;
|
const connectionDelay = options.slowMo || 0;
|
||||||
const browserWSEndpoint = await waitForWSEndpoint(chromeProcess, options.timeout || 30 * 1000);
|
const browserWSEndpoint = await waitForWSEndpoint(chromeProcess, options.timeout || 30 * 1000);
|
||||||
connection = await Connection.create(browserWSEndpoint, connectionDelay);
|
connection = await Connection.create(browserWSEndpoint, connectionDelay);
|
||||||
return new Browser(connection, options, killChrome);
|
return Browser.create(connection, options, killChrome);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
killChrome();
|
killChrome();
|
||||||
throw e;
|
throw e;
|
||||||
@ -179,7 +179,7 @@ class Launcher {
|
|||||||
*/
|
*/
|
||||||
static async connect(options = {}) {
|
static async connect(options = {}) {
|
||||||
const connection = await Connection.create(options.browserWSEndpoint);
|
const connection = await Connection.create(options.browserWSEndpoint);
|
||||||
return new Browser(connection, options, () => connection.send('Browser.close'));
|
return Browser.create(connection, options, () => connection.send('Browser.close'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
14
lib/Page.js
14
lib/Page.js
@ -37,35 +37,39 @@ class Page extends EventEmitter {
|
|||||||
* @return {!Promise<!Page>}
|
* @return {!Promise<!Page>}
|
||||||
*/
|
*/
|
||||||
static async create(client, ignoreHTTPSErrors, appMode, screenshotTaskQueue) {
|
static async create(client, ignoreHTTPSErrors, appMode, screenshotTaskQueue) {
|
||||||
|
|
||||||
|
await client.send('Page.enable');
|
||||||
|
const {frameTree} = await client.send('Page.getResourceTree');
|
||||||
|
const page = new Page(client, frameTree, ignoreHTTPSErrors, screenshotTaskQueue);
|
||||||
|
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
client.send('Network.enable', {}),
|
client.send('Network.enable', {}),
|
||||||
client.send('Page.enable', {}),
|
|
||||||
client.send('Runtime.enable', {}),
|
client.send('Runtime.enable', {}),
|
||||||
client.send('Security.enable', {}),
|
client.send('Security.enable', {}),
|
||||||
client.send('Performance.enable', {})
|
client.send('Performance.enable', {})
|
||||||
]);
|
]);
|
||||||
if (ignoreHTTPSErrors)
|
if (ignoreHTTPSErrors)
|
||||||
await client.send('Security.setOverrideCertificateErrors', {override: true});
|
await client.send('Security.setOverrideCertificateErrors', {override: true});
|
||||||
const page = new Page(client, ignoreHTTPSErrors, screenshotTaskQueue);
|
|
||||||
await page.goto('about:blank');
|
|
||||||
// Initialize default page size.
|
// Initialize default page size.
|
||||||
if (!appMode)
|
if (!appMode)
|
||||||
await page.setViewport({width: 800, height: 600});
|
await page.setViewport({width: 800, height: 600});
|
||||||
|
|
||||||
return page;
|
return page;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {!Puppeteer.Session} client
|
* @param {!Puppeteer.Session} client
|
||||||
|
* @param {{frame: Object, childFrames: ?Array}} frameTree
|
||||||
* @param {boolean} ignoreHTTPSErrors
|
* @param {boolean} ignoreHTTPSErrors
|
||||||
* @param {!Puppeteer.TaskQueue} screenshotTaskQueue
|
* @param {!Puppeteer.TaskQueue} screenshotTaskQueue
|
||||||
*/
|
*/
|
||||||
constructor(client, ignoreHTTPSErrors, screenshotTaskQueue) {
|
constructor(client, frameTree, ignoreHTTPSErrors, screenshotTaskQueue) {
|
||||||
super();
|
super();
|
||||||
this._client = client;
|
this._client = client;
|
||||||
this._keyboard = new Keyboard(client);
|
this._keyboard = new Keyboard(client);
|
||||||
this._mouse = new Mouse(client, this._keyboard);
|
this._mouse = new Mouse(client, this._keyboard);
|
||||||
this._touchscreen = new Touchscreen(client, this._keyboard);
|
this._touchscreen = new Touchscreen(client, this._keyboard);
|
||||||
this._frameManager = new FrameManager(client, this);
|
this._frameManager = new FrameManager(client, frameTree, this);
|
||||||
this._networkManager = new NetworkManager(client);
|
this._networkManager = new NetworkManager(client);
|
||||||
this._emulationManager = new EmulationManager(client);
|
this._emulationManager = new EmulationManager(client);
|
||||||
this._tracing = new Tracing(client);
|
this._tracing = new Tracing(client);
|
||||||
|
0
test/assets/sw.js
Normal file
0
test/assets/sw.js
Normal file
5
test/golden/reconnect-nested-frames.txt
Normal file
5
test/golden/reconnect-nested-frames.txt
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
http://localhost:8907/frames/nested-frames.html
|
||||||
|
http://localhost:8907/frames/two-frames.html
|
||||||
|
http://localhost:8907/frames/frame.html
|
||||||
|
http://localhost:8907/frames/frame.html
|
||||||
|
http://localhost:8907/frames/frame.html
|
103
test/test.js
103
test/test.js
@ -74,6 +74,12 @@ beforeAll(SX(async function() {
|
|||||||
rm(OUTPUT_DIR);
|
rm(OUTPUT_DIR);
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
beforeEach(SX(async function() {
|
||||||
|
server.reset();
|
||||||
|
httpsServer.reset();
|
||||||
|
GoldenUtils.addMatchers(jasmine, GOLDEN_DIR, OUTPUT_DIR);
|
||||||
|
}));
|
||||||
|
|
||||||
afterAll(SX(async function() {
|
afterAll(SX(async function() {
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
server.stop(),
|
server.stop(),
|
||||||
@ -179,11 +185,16 @@ describe('Puppeteer', function() {
|
|||||||
it('should be able to reconnect to a disconnected browser', SX(async function() {
|
it('should be able to reconnect to a disconnected browser', SX(async function() {
|
||||||
const originalBrowser = await puppeteer.launch(defaultBrowserOptions);
|
const originalBrowser = await puppeteer.launch(defaultBrowserOptions);
|
||||||
const browserWSEndpoint = originalBrowser.wsEndpoint();
|
const browserWSEndpoint = originalBrowser.wsEndpoint();
|
||||||
|
const page = await originalBrowser.newPage();
|
||||||
|
await page.goto(PREFIX + '/frames/nested-frames.html');
|
||||||
originalBrowser.disconnect();
|
originalBrowser.disconnect();
|
||||||
|
|
||||||
|
const FrameUtils = require('./frame-utils');
|
||||||
const browser = await puppeteer.connect({browserWSEndpoint});
|
const browser = await puppeteer.connect({browserWSEndpoint});
|
||||||
const page = await browser.newPage();
|
const pages = await browser.pages();
|
||||||
expect(await page.evaluate(() => 7 * 8)).toBe(56);
|
const restoredPage = pages.find(page => page.url() === PREFIX + '/frames/nested-frames.html');
|
||||||
|
expect(FrameUtils.dumpFrames(restoredPage.mainFrame())).toBeGolden('reconnect-nested-frames.txt');
|
||||||
|
expect(await restoredPage.evaluate(() => 7 * 8)).toBe(56);
|
||||||
await browser.close();
|
await browser.close();
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
@ -212,9 +223,6 @@ describe('Page', function() {
|
|||||||
|
|
||||||
beforeEach(SX(async function() {
|
beforeEach(SX(async function() {
|
||||||
page = await browser.newPage();
|
page = await browser.newPage();
|
||||||
server.reset();
|
|
||||||
httpsServer.reset();
|
|
||||||
GoldenUtils.addMatchers(jasmine, GOLDEN_DIR, OUTPUT_DIR);
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
afterEach(SX(async function() {
|
afterEach(SX(async function() {
|
||||||
@ -2739,6 +2747,91 @@ describe('Page', function() {
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('Target', function() {
|
||||||
|
it('Browser.targets should return all of the targets', SX(async function() {
|
||||||
|
// The pages will be the testing page and the original newtab page
|
||||||
|
const targets = browser.targets();
|
||||||
|
expect(targets.some(target => target.type() === 'page' &&
|
||||||
|
target.url() === 'about:blank')).toBeTruthy('Missing blank page');
|
||||||
|
expect(targets.some(target => target.type() === 'other' &&
|
||||||
|
target.url() === '')).toBeTruthy('Missing browser target');
|
||||||
|
}));
|
||||||
|
it('Browser.pages should return all of the pages', SX(async function() {
|
||||||
|
// The pages will be the testing page and the original newtab page
|
||||||
|
const allPages = await browser.pages();
|
||||||
|
expect(allPages.length).toBe(2);
|
||||||
|
expect(allPages).toContain(page);
|
||||||
|
expect(allPages[0]).not.toBe(allPages[1]);
|
||||||
|
}));
|
||||||
|
it('should be able to use the default page in the browser', SX(async function() {
|
||||||
|
// The pages will be the testing page and the original newtab page
|
||||||
|
const allPages = await browser.pages();
|
||||||
|
const originalPage = allPages.find(p => p !== page);
|
||||||
|
expect(await originalPage.evaluate(() => ['Hello', 'world'].join(' '))).toBe('Hello world');
|
||||||
|
expect(await originalPage.$('body')).toBeTruthy();
|
||||||
|
}));
|
||||||
|
it('should report when a new page is created and closed', SX(async function(){
|
||||||
|
const otherPagePromise = new Promise(fulfill => browser.once('targetcreated', target => fulfill(target.page())));
|
||||||
|
await page.evaluate(url => window.open(url), CROSS_PROCESS_PREFIX);
|
||||||
|
const otherPage = await otherPagePromise;
|
||||||
|
expect(otherPage.url()).toContain(CROSS_PROCESS_PREFIX);
|
||||||
|
|
||||||
|
expect(await otherPage.evaluate(() => ['Hello', 'world'].join(' '))).toBe('Hello world');
|
||||||
|
expect(await otherPage.$('body')).toBeTruthy();
|
||||||
|
|
||||||
|
let allPages = await browser.pages();
|
||||||
|
expect(allPages).toContain(page);
|
||||||
|
expect(allPages).toContain(otherPage);
|
||||||
|
|
||||||
|
const closePagePromise = new Promise(fulfill => browser.once('targetdestroyed', target => fulfill(target.page())));
|
||||||
|
await otherPage.close();
|
||||||
|
expect(await closePagePromise).toBe(otherPage);
|
||||||
|
|
||||||
|
allPages = await Promise.all(browser.targets().map(target => target.page()));
|
||||||
|
expect(allPages).toContain(page);
|
||||||
|
expect(allPages).not.toContain(otherPage);
|
||||||
|
}));
|
||||||
|
it('should report when a service worker is created and destroyed', SX(async function() {
|
||||||
|
await page.goto(EMPTY_PAGE);
|
||||||
|
const createdTarget = new Promise(fulfill => browser.once('targetcreated', target => fulfill(target)));
|
||||||
|
const registration = await page.evaluateHandle(() => navigator.serviceWorker.register('sw.js'));
|
||||||
|
|
||||||
|
expect((await createdTarget).type()).toBe('service_worker');
|
||||||
|
expect((await createdTarget).url()).toBe(PREFIX + '/sw.js');
|
||||||
|
|
||||||
|
const destroyedTarget = new Promise(fulfill => browser.once('targetdestroyed', target => fulfill(target)));
|
||||||
|
await page.evaluate(registration => registration.unregister(), registration);
|
||||||
|
expect(await destroyedTarget).toBe(await createdTarget);
|
||||||
|
}));
|
||||||
|
it('should report when a target url changes', SX(async function(){
|
||||||
|
await page.goto(EMPTY_PAGE);
|
||||||
|
let changedTarget = new Promise(fulfill => browser.once('targetchanged', target => fulfill(target)));
|
||||||
|
await page.goto(CROSS_PROCESS_PREFIX + '/');
|
||||||
|
expect((await changedTarget).url()).toBe(CROSS_PROCESS_PREFIX + '/');
|
||||||
|
|
||||||
|
changedTarget = new Promise(fulfill => browser.once('targetchanged', target => fulfill(target)));
|
||||||
|
await page.goto(EMPTY_PAGE);
|
||||||
|
expect((await changedTarget).url()).toBe(EMPTY_PAGE);
|
||||||
|
}));
|
||||||
|
it('should not report uninitialized pages', SX(async function() {
|
||||||
|
browser.on('targetchanged', () => {
|
||||||
|
expect(false).toBe(true, 'target should not be reported as changed');
|
||||||
|
});
|
||||||
|
const targetPromise = new Promise(fulfill => browser.once('targetcreated', target => fulfill(target)));
|
||||||
|
const newPagePromise = browser.newPage();
|
||||||
|
const target = await targetPromise;
|
||||||
|
expect(target.url()).toBe('about:blank');
|
||||||
|
|
||||||
|
const newPage = await newPagePromise;
|
||||||
|
const targetPromise2 = new Promise(fulfill => browser.once('targetcreated', target => fulfill(target)));
|
||||||
|
const evaluatePromise = newPage.evaluate(() => window.open('about:blank'));
|
||||||
|
const target2 = await targetPromise2;
|
||||||
|
expect(target2.url()).toBe('about:blank');
|
||||||
|
await evaluatePromise;
|
||||||
|
await newPage.close();
|
||||||
|
}));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
if (process.env.COVERAGE) {
|
if (process.env.COVERAGE) {
|
||||||
|
@ -35,6 +35,7 @@ const EXCLUDE_CLASSES = new Set([
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
const EXCLUDE_METHODS = new Set([
|
const EXCLUDE_METHODS = new Set([
|
||||||
|
'Browser.create',
|
||||||
'Headers.fromPayload',
|
'Headers.fromPayload',
|
||||||
'Page.create',
|
'Page.create',
|
||||||
'JSHandle.toString',
|
'JSHandle.toString',
|
||||||
|
Loading…
Reference in New Issue
Block a user