From 89d0f1e1e7dc97e54b9180fe624e7e4748fdc100 Mon Sep 17 00:00:00 2001 From: Andrey Lushnikov Date: Tue, 12 Feb 2019 20:10:53 -0800 Subject: [PATCH] feat(firefox): implement frame.goto / frame.waitForNavigation (#3992) Some corner cases regarding iframes being detached during navigation are not yet supported. --- .../puppeteer-firefox/lib/FrameManager.js | 107 ++++++++- .../lib/NavigationWatchdog.js | 115 ++++++++++ .../puppeteer-firefox/lib/NetworkManager.js | 8 +- experimental/puppeteer-firefox/lib/Page.js | 208 +----------------- experimental/puppeteer-firefox/package.json | 2 +- test/navigation.spec.js | 4 +- 6 files changed, 235 insertions(+), 209 deletions(-) create mode 100644 experimental/puppeteer-firefox/lib/NavigationWatchdog.js diff --git a/experimental/puppeteer-firefox/lib/FrameManager.js b/experimental/puppeteer-firefox/lib/FrameManager.js index 02dfdf23..c6557595 100644 --- a/experimental/puppeteer-firefox/lib/FrameManager.js +++ b/experimental/puppeteer-firefox/lib/FrameManager.js @@ -5,6 +5,7 @@ const util = require('util'); const EventEmitter = require('events'); const {Events} = require('./Events'); const {ExecutionContext} = require('./ExecutionContext'); +const {NavigationWatchdog, NextNavigationWatchdog} = require('./NavigationWatchdog'); const readFileAsync = util.promisify(fs.readFile); @@ -13,10 +14,11 @@ class FrameManager extends EventEmitter { * @param {PageSession} session * @param {Page} page */ - constructor(session, page, timeoutSettings) { + constructor(session, page, networkManager, timeoutSettings) { super(); this._session = session; this._page = page; + this._networkManager = networkManager; this._timeoutSettings = timeoutSettings; this._mainFrame = null; this._frames = new Map(); @@ -65,7 +67,7 @@ class FrameManager extends EventEmitter { } _onFrameAttached(params) { - const frame = new Frame(this._session, this, this._page, params.frameId, this._timeoutSettings); + const frame = new Frame(this._session, this, this._networkManager, this._page, params.frameId, this._timeoutSettings); const parentFrame = this._frames.get(params.parentFrameId) || null; if (parentFrame) { frame._parentFrame = parentFrame; @@ -107,11 +109,12 @@ class Frame { * @param {!Page} page * @param {string} frameId */ - constructor(session, frameManager, page, frameId, timeoutSettings) { + constructor(session, frameManager, networkManager, page, frameId, timeoutSettings) { this._session = session; - this._frameManager = frameManager; - this._timeoutSettings = timeoutSettings; this._page = page; + this._frameManager = frameManager; + this._networkManager = networkManager; + this._timeoutSettings = timeoutSettings; this._frameId = frameId; /** @type {?Frame} */ this._parentFrame = null; @@ -134,6 +137,88 @@ class Frame { return this._executionContext; } + /** + * @param {!{timeout?: number, waitUntil?: string|!Array}} options + */ + async waitForNavigation(options = {}) { + const { + timeout = this._timeoutSettings.navigationTimeout(), + waitUntil = ['load'], + } = options; + const normalizedWaitUntil = normalizeWaitUntil(waitUntil); + + const timeoutError = new TimeoutError('Navigation Timeout Exceeded: ' + timeout + 'ms'); + let timeoutCallback; + const timeoutPromise = new Promise(resolve => timeoutCallback = resolve.bind(null, timeoutError)); + const timeoutId = timeout ? setTimeout(timeoutCallback, timeout) : null; + + const nextNavigationDog = new NextNavigationWatchdog(this._session, this); + const error1 = await Promise.race([ + nextNavigationDog.promise(), + timeoutPromise, + ]); + nextNavigationDog.dispose(); + + // If timeout happened first - throw. + if (error1) { + clearTimeout(timeoutId); + throw error1; + } + + const {navigationId, url} = nextNavigationDog.navigation(); + + if (!navigationId) { + // Same document navigation happened. + clearTimeout(timeoutId); + return null; + } + + const watchDog = new NavigationWatchdog(this._session, this, this._networkManager, navigationId, url, normalizedWaitUntil); + const error = await Promise.race([ + timeoutPromise, + watchDog.promise(), + ]); + watchDog.dispose(); + clearTimeout(timeoutId); + if (error) + throw error; + return watchDog.navigationResponse(); + } + + /** + * @param {string} url + * @param {!{timeout?: number, waitUntil?: string|!Array}} options + */ + async goto(url, options = {}) { + const { + timeout = this._timeoutSettings.navigationTimeout(), + waitUntil = ['load'], + } = options; + const normalizedWaitUntil = normalizeWaitUntil(waitUntil); + const {navigationId} = await this._session.send('Page.navigate', { + frameId: this._frameId, + url, + }); + if (!navigationId) + return; + + const timeoutError = new TimeoutError('Navigation Timeout Exceeded: ' + timeout + 'ms'); + let timeoutCallback; + const timeoutPromise = new Promise(resolve => timeoutCallback = resolve.bind(null, timeoutError)); + const timeoutId = timeout ? setTimeout(timeoutCallback, timeout) : null; + + const watchDog = new NavigationWatchdog(this._session, this, this._networkManager, navigationId, url, normalizedWaitUntil); + const error = await Promise.race([ + timeoutPromise, + watchDog.promise(), + ]); + watchDog.dispose(); + clearTimeout(timeoutId); + if (error) + throw error; + return watchDog.navigationResponse(); + } + /** * @param {string} selector * @param {!{delay?: number, button?: string, clickCount?: number}=} options @@ -747,4 +832,14 @@ async function waitForPredicatePageFunction(predicateBody, polling, timeout, ... } } -module.exports = {FrameManager, Frame}; +function normalizeWaitUntil(waitUntil) { + if (!Array.isArray(waitUntil)) + waitUntil = [waitUntil]; + for (const condition of waitUntil) { + if (condition !== 'load' && condition !== 'domcontentloaded') + throw new Error('Unknown waitUntil condition: ' + condition); + } + return waitUntil; +} + +module.exports = {FrameManager, Frame, normalizeWaitUntil}; diff --git a/experimental/puppeteer-firefox/lib/NavigationWatchdog.js b/experimental/puppeteer-firefox/lib/NavigationWatchdog.js new file mode 100644 index 00000000..321fe6ce --- /dev/null +++ b/experimental/puppeteer-firefox/lib/NavigationWatchdog.js @@ -0,0 +1,115 @@ +const {helper} = require('./helper'); +const {Events} = require('./Events'); + +/** + * @internal + */ +class NextNavigationWatchdog { + constructor(session, navigatedFrame) { + this._navigatedFrame = navigatedFrame; + this._promise = new Promise(x => this._resolveCallback = x); + this._navigation = null; + this._eventListeners = [ + helper.addEventListener(session, 'Page.navigationStarted', this._onNavigationStarted.bind(this)), + helper.addEventListener(session, 'Page.sameDocumentNavigation', this._onSameDocumentNavigation.bind(this)), + ]; + } + + promise() { + return this._promise; + } + + navigation() { + return this._navigation; + } + + _onNavigationStarted(params) { + if (params.frameId === this._navigatedFrame._frameId) { + this._navigation = { + navigationId: params.navigationId, + url: params.url, + }; + this._resolveCallback(); + } + } + + _onSameDocumentNavigation(params) { + if (params.frameId === this._navigatedFrame._frameId) { + this._navigation = { + navigationId: null, + }; + this._resolveCallback(); + } + } + + dispose() { + helper.removeEventListeners(this._eventListeners); + } +} + +/** + * @internal + */ +class NavigationWatchdog { + constructor(session, navigatedFrame, networkManager, targetNavigationId, targetURL, firedEvents) { + this._navigatedFrame = navigatedFrame; + this._targetNavigationId = targetNavigationId; + this._firedEvents = firedEvents; + this._targetURL = targetURL; + + this._promise = new Promise(x => this._resolveCallback = x); + this._navigationRequest = null; + + const check = this._checkNavigationComplete.bind(this); + this._eventListeners = [ + helper.addEventListener(session, 'Page.eventFired', check), + helper.addEventListener(session, 'Page.frameAttached', check), + helper.addEventListener(session, 'Page.frameDetached', check), + helper.addEventListener(session, 'Page.navigationStarted', check), + helper.addEventListener(session, 'Page.navigationCommitted', check), + helper.addEventListener(session, 'Page.navigationAborted', this._onNavigationAborted.bind(this)), + helper.addEventListener(networkManager, Events.NetworkManager.Request, this._onRequest.bind(this)), + ]; + check(); + } + + _onRequest(request) { + if (request.frame() !== this._navigatedFrame || !request.isNavigationRequest()) + return; + this._navigationRequest = request; + } + + navigationResponse() { + return this._navigationRequest ? this._navigationRequest.response() : null; + } + + _checkNavigationComplete() { + if (this._navigatedFrame._lastCommittedNavigationId === this._targetNavigationId + && checkFiredEvents(this._navigatedFrame, this._firedEvents)) { + this._resolveCallback(null); + } + + function checkFiredEvents(frame, firedEvents) { + for (const subframe of frame._children) { + if (!checkFiredEvents(subframe, firedEvents)) + return false; + } + return firedEvents.every(event => frame._firedEvents.has(event)); + } + } + + _onNavigationAborted(params) { + if (params.frameId === this._navigatedFrame._frameId && params.navigationId === this._targetNavigationId) + this._resolveCallback(new Error('Navigation to ' + this._targetURL + ' failed: ' + params.errorText)); + } + + promise() { + return this._promise; + } + + dispose() { + helper.removeEventListeners(this._eventListeners); + } +} + +module.exports = {NavigationWatchdog, NextNavigationWatchdog}; diff --git a/experimental/puppeteer-firefox/lib/NetworkManager.js b/experimental/puppeteer-firefox/lib/NetworkManager.js index 66fa5952..b97081ac 100644 --- a/experimental/puppeteer-firefox/lib/NetworkManager.js +++ b/experimental/puppeteer-firefox/lib/NetworkManager.js @@ -4,12 +4,12 @@ const EventEmitter = require('events'); const {Events} = require('./Events'); class NetworkManager extends EventEmitter { - constructor(session, frameManager) { + constructor(session) { super(); this._session = session; this._requests = new Map(); - this._frameManager = frameManager; + this._frameManager = null; this._eventListeners = [ helper.addEventListener(session, 'Page.requestWillBeSent', this._onRequestWillBeSent.bind(this)), @@ -18,6 +18,10 @@ class NetworkManager extends EventEmitter { ]; } + setFrameManager(frameManager) { + this._frameManager = frameManager; + } + _onRequestWillBeSent(event) { const redirected = event.redirectedFrom ? this._requests.get(event.redirectedFrom) : null; const frame = redirected ? redirected.frame() : (this._frameManager && event.frameId ? this._frameManager.frame(event.frameId) : null); diff --git a/experimental/puppeteer-firefox/lib/Page.js b/experimental/puppeteer-firefox/lib/Page.js index 5c15e5b6..034777c7 100644 --- a/experimental/puppeteer-firefox/lib/Page.js +++ b/experimental/puppeteer-firefox/lib/Page.js @@ -8,9 +8,10 @@ const util = require('util'); const EventEmitter = require('events'); const {createHandle} = require('./JSHandle'); const {Events} = require('./Events'); -const {FrameManager} = require('./FrameManager'); +const {FrameManager, normalizeWaitUntil} = require('./FrameManager'); const {NetworkManager} = require('./NetworkManager'); const {TimeoutSettings} = require('./TimeoutSettings'); +const {NavigationWatchdog, NextNavigationWatchdog} = require('./NavigationWatchdog'); const writeFileAsync = util.promisify(fs.writeFile); @@ -74,8 +75,9 @@ class Page extends EventEmitter { this._keyboard = new Keyboard(session); this._mouse = new Mouse(session, this._keyboard); this._isClosed = false; - this._frameManager = new FrameManager(session, this, this._timeoutSettings); - this._networkManager = new NetworkManager(session, this._frameManager); + this._networkManager = new NetworkManager(session); + this._frameManager = new FrameManager(session, this, this._networkManager, this._timeoutSettings); + this._networkManager.setFrameManager(this._frameManager); this._eventListeners = [ helper.addEventListener(this._session, 'Page.uncaughtError', this._onUncaughtError.bind(this)), helper.addEventListener(this._session, 'Page.consoleAPICalled', this._onConsole.bind(this)), @@ -255,63 +257,11 @@ class Page extends EventEmitter { return this._mouse; } - _normalizeWaitUntil(waitUntil) { - if (!Array.isArray(waitUntil)) - waitUntil = [waitUntil]; - for (const condition of waitUntil) { - if (condition !== 'load' && condition !== 'domcontentloaded') - throw new Error('Unknown waitUntil condition: ' + condition); - } - return waitUntil; - } - /** * @param {!{timeout?: number, waitUntil?: string|!Array}} options */ async waitForNavigation(options = {}) { - const { - timeout = this._timeoutSettings.navigationTimeout(), - waitUntil = ['load'], - } = options; - const frame = this._frameManager.mainFrame(); - const normalizedWaitUntil = this._normalizeWaitUntil(waitUntil); - - const timeoutError = new TimeoutError('Navigation Timeout Exceeded: ' + timeout + 'ms'); - let timeoutCallback; - const timeoutPromise = new Promise(resolve => timeoutCallback = resolve.bind(null, timeoutError)); - const timeoutId = timeout ? setTimeout(timeoutCallback, timeout) : null; - - const nextNavigationDog = new NextNavigationWatchdog(this._session, frame); - const error1 = await Promise.race([ - nextNavigationDog.promise(), - timeoutPromise, - ]); - nextNavigationDog.dispose(); - - // If timeout happened first - throw. - if (error1) { - clearTimeout(timeoutId); - throw error1; - } - - const {navigationId, url} = nextNavigationDog.navigation(); - - if (!navigationId) { - // Same document navigation happened. - clearTimeout(timeoutId); - return null; - } - - const watchDog = new NavigationWatchdog(this._session, frame, this._networkManager, navigationId, url, normalizedWaitUntil); - const error = await Promise.race([ - timeoutPromise, - watchDog.promise(), - ]); - watchDog.dispose(); - clearTimeout(timeoutId); - if (error) - throw error; - return watchDog.navigationResponse(); + return this._frameManager.mainFrame().waitForNavigation(options); } /** @@ -319,34 +269,7 @@ class Page extends EventEmitter { * @param {!{timeout?: number, waitUntil?: string|!Array}} options */ async goto(url, options = {}) { - const { - timeout = this._timeoutSettings.navigationTimeout(), - waitUntil = ['load'], - } = options; - const frame = this._frameManager.mainFrame(); - const normalizedWaitUntil = this._normalizeWaitUntil(waitUntil); - const {navigationId} = await this._session.send('Page.navigate', { - frameId: frame._frameId, - url, - }); - if (!navigationId) - return; - - const timeoutError = new TimeoutError('Navigation Timeout Exceeded: ' + timeout + 'ms'); - let timeoutCallback; - const timeoutPromise = new Promise(resolve => timeoutCallback = resolve.bind(null, timeoutError)); - const timeoutId = timeout ? setTimeout(timeoutCallback, timeout) : null; - - const watchDog = new NavigationWatchdog(this._session, frame, this._networkManager, navigationId, url, normalizedWaitUntil); - const error = await Promise.race([ - timeoutPromise, - watchDog.promise(), - ]); - watchDog.dispose(); - clearTimeout(timeoutId); - if (error) - throw error; - return watchDog.navigationResponse(); + return this._frameManager.mainFrame().goto(url, options); } /** @@ -358,7 +281,7 @@ class Page extends EventEmitter { waitUntil = ['load'], } = options; const frame = this._frameManager.mainFrame(); - const normalizedWaitUntil = this._normalizeWaitUntil(waitUntil); + const normalizedWaitUntil = normalizeWaitUntil(waitUntil); const {navigationId, navigationURL} = await this._session.send('Page.goBack', { frameId: frame._frameId, }); @@ -391,7 +314,7 @@ class Page extends EventEmitter { waitUntil = ['load'], } = options; const frame = this._frameManager.mainFrame(); - const normalizedWaitUntil = this._normalizeWaitUntil(waitUntil); + const normalizedWaitUntil = normalizeWaitUntil(waitUntil); const {navigationId, navigationURL} = await this._session.send('Page.goForward', { frameId: frame._frameId, }); @@ -424,7 +347,7 @@ class Page extends EventEmitter { waitUntil = ['load'], } = options; const frame = this._frameManager.mainFrame(); - const normalizedWaitUntil = this._normalizeWaitUntil(waitUntil); + const normalizedWaitUntil = normalizeWaitUntil(waitUntil); const {navigationId, navigationURL} = await this._session.send('Page.reload', { frameId: frame._frameId, }); @@ -706,115 +629,4 @@ function getScreenshotMimeType(options) { return 'image/png'; } -/** - * @internal - */ -class NextNavigationWatchdog { - constructor(session, navigatedFrame) { - this._navigatedFrame = navigatedFrame; - this._promise = new Promise(x => this._resolveCallback = x); - this._navigation = null; - this._eventListeners = [ - helper.addEventListener(session, 'Page.navigationStarted', this._onNavigationStarted.bind(this)), - helper.addEventListener(session, 'Page.sameDocumentNavigation', this._onSameDocumentNavigation.bind(this)), - ]; - } - - promise() { - return this._promise; - } - - navigation() { - return this._navigation; - } - - _onNavigationStarted(params) { - if (params.frameId === this._navigatedFrame._frameId) { - this._navigation = { - navigationId: params.navigationId, - url: params.url, - }; - this._resolveCallback(); - } - } - - _onSameDocumentNavigation(params) { - if (params.frameId === this._navigatedFrame._frameId) { - this._navigation = { - navigationId: null, - }; - this._resolveCallback(); - } - } - - dispose() { - helper.removeEventListeners(this._eventListeners); - } -} - -/** - * @internal - */ -class NavigationWatchdog { - constructor(session, navigatedFrame, networkManager, targetNavigationId, targetURL, firedEvents) { - this._navigatedFrame = navigatedFrame; - this._targetNavigationId = targetNavigationId; - this._firedEvents = firedEvents; - this._targetURL = targetURL; - - this._promise = new Promise(x => this._resolveCallback = x); - this._navigationRequest = null; - - const check = this._checkNavigationComplete.bind(this); - this._eventListeners = [ - helper.addEventListener(session, 'Page.eventFired', check), - helper.addEventListener(session, 'Page.frameAttached', check), - helper.addEventListener(session, 'Page.frameDetached', check), - helper.addEventListener(session, 'Page.navigationStarted', check), - helper.addEventListener(session, 'Page.navigationCommitted', check), - helper.addEventListener(session, 'Page.navigationAborted', this._onNavigationAborted.bind(this)), - helper.addEventListener(networkManager, Events.NetworkManager.Request, this._onRequest.bind(this)), - ]; - check(); - } - - _onRequest(request) { - if (request.frame() !== this._navigatedFrame || !request.isNavigationRequest()) - return; - this._navigationRequest = request; - } - - navigationResponse() { - return this._navigationRequest ? this._navigationRequest.response() : null; - } - - _checkNavigationComplete() { - if (this._navigatedFrame._lastCommittedNavigationId === this._targetNavigationId - && checkFiredEvents(this._navigatedFrame, this._firedEvents)) { - this._resolveCallback(null); - } - - function checkFiredEvents(frame, firedEvents) { - for (const subframe of frame._children) { - if (!checkFiredEvents(subframe, firedEvents)) - return false; - } - return firedEvents.every(event => frame._firedEvents.has(event)); - } - } - - _onNavigationAborted(params) { - if (params.frameId === this._navigatedFrame._frameId && params.navigationId === this._targetNavigationId) - this._resolveCallback(new Error('Navigation to ' + this._targetURL + ' failed: ' + params.errorText)); - } - - promise() { - return this._promise; - } - - dispose() { - helper.removeEventListeners(this._eventListeners); - } -} - module.exports = {Page, ConsoleMessage}; diff --git a/experimental/puppeteer-firefox/package.json b/experimental/puppeteer-firefox/package.json index c4d7eb2f..f75306bd 100644 --- a/experimental/puppeteer-firefox/package.json +++ b/experimental/puppeteer-firefox/package.json @@ -9,7 +9,7 @@ "node": ">=8.9.4" }, "puppeteer": { - "firefox_revision": "ad27e01304952cb0ff0b2817016b0e9f31d7f8fa" + "firefox_revision": "309b4f8466e83360f2045be982d2c61522bcf466" }, "scripts": { "install": "node install.js", diff --git a/test/navigation.spec.js b/test/navigation.spec.js index 550ea393..8766cc61 100644 --- a/test/navigation.spec.js +++ b/test/navigation.spec.js @@ -472,7 +472,7 @@ module.exports.addTests = function({testRunner, expect, Errors, CHROME}) { }); describe('Frame.goto', function() { - it_fails_ffox('should navigate subframes', async({page, server}) => { + it('should navigate subframes', async({page, server}) => { await page.goto(server.PREFIX + '/frames/one-frame.html'); expect(page.frames()[0].url()).toContain('/frames/one-frame.html'); expect(page.frames()[1].url()).toContain('/frames/frame.html'); @@ -522,7 +522,7 @@ module.exports.addTests = function({testRunner, expect, Errors, CHROME}) { }); describe('Frame.waitForNavigation', function() { - it_fails_ffox('should work', async({page, server}) => { + it('should work', async({page, server}) => { await page.goto(server.PREFIX + '/frames/one-frame.html'); const frame = page.frames()[1]; const [response] = await Promise.all([