From a7d59b587e6b8a180f24b8275ce98c63da58de91 Mon Sep 17 00:00:00 2001 From: Andrey Lushnikov Date: Tue, 10 Apr 2018 15:59:41 -0700 Subject: [PATCH] fix(Page): Page.waitForNavigation should correctly handle mixed content (#2339) This patch teaches Page.waitForNavigation to correctly handle navigation to pages that have frames that might never load. These frames include: - frames which main resource loading was aborted due to mixed-content error - frames that artificially called `window.stop()` to interrupt loading themselves Fixes #1936. --- lib/FrameManager.js | 17 +++++++++++++++++ test/assets/frames/one-frame.html | 1 + test/page.spec.js | 13 +++++++++++++ test/puppeteer.spec.js | 11 +++++++++++ 4 files changed, 42 insertions(+) create mode 100644 test/assets/frames/one-frame.html diff --git a/lib/FrameManager.js b/lib/FrameManager.js index 2bd496e1..b29dcf7a 100644 --- a/lib/FrameManager.js +++ b/lib/FrameManager.js @@ -41,6 +41,7 @@ class FrameManager extends EventEmitter { this._client.on('Page.frameNavigated', event => this._onFrameNavigated(event.frame)); this._client.on('Page.navigatedWithinDocument', event => this._onFrameNavigatedWithinDocument(event.frameId, event.url)); this._client.on('Page.frameDetached', event => this._onFrameDetached(event.frameId)); + this._client.on('Page.frameStoppedLoading', event => this._onFrameStoppedLoading(event.frameId)); this._client.on('Runtime.executionContextCreated', event => this._onExecutionContextCreated(event.context)); this._client.on('Runtime.executionContextDestroyed', event => this._onExecutionContextDestroyed(event.executionContextId)); this._client.on('Runtime.executionContextsCleared', event => this._onExecutionContextsCleared()); @@ -60,6 +61,17 @@ class FrameManager extends EventEmitter { this.emit(FrameManager.Events.LifecycleEvent, frame); } + /** + * @param {string} frameId + */ + _onFrameStoppedLoading(frameId) { + const frame = this._frames.get(frameId); + if (!frame) + return; + frame._onLoadingStopped(); + this.emit(FrameManager.Events.LifecycleEvent, frame); + } + /** * @param {!Protocol.Page.FrameTree} frameTree */ @@ -784,6 +796,11 @@ class Frame { this._lifecycleEvents.add(name); } + _onLoadingStopped() { + this._lifecycleEvents.add('DOMContentLoaded'); + this._lifecycleEvents.add('load'); + } + _detach() { for (const waitTask of this._waitTasks) waitTask.terminate(new Error('waitForFunction failed: frame got detached.')); diff --git a/test/assets/frames/one-frame.html b/test/assets/frames/one-frame.html new file mode 100644 index 00000000..e941d795 --- /dev/null +++ b/test/assets/frames/one-frame.html @@ -0,0 +1 @@ + diff --git a/test/page.spec.js b/test/page.spec.js index 2a641b91..111e47da 100644 --- a/test/page.spec.js +++ b/test/page.spec.js @@ -674,6 +674,19 @@ module.exports.addTests = function({testRunner, expect, puppeteer, DeviceDescrip ]); expect(page.url()).toBe(server.PREFIX + '/second.html'); }); + it('should work when subframe issues window.stop()', async({page, server}) => { + server.setRoute('/frames/style.css', (req, res) => {}); + const navigationPromise = page.goto(server.PREFIX + '/frames/one-frame.html'); + const frame = await utils.waitEvent(page, 'frameattached'); + await new Promise(fulfill => { + page.on('framenavigated', f => { + if (f === frame) + fulfill(); + }); + }); + frame.evaluate(() => window.stop()); + await navigationPromise; + }); }); describe('Page.goBack', function() { diff --git a/test/puppeteer.spec.js b/test/puppeteer.spec.js index 3af151fa..86dbf625 100644 --- a/test/puppeteer.spec.js +++ b/test/puppeteer.spec.js @@ -99,6 +99,17 @@ module.exports.addTests = function({testRunner, expect, PROJECT_ROOT, defaultBro await page.close(); await browser.close(); }); + it('should work with mixed content', async({server, httpsServer}) => { + httpsServer.setRoute('/mixedcontent.html', (req, res) => { + res.end(``); + }); + const options = Object.assign({ignoreHTTPSErrors: true}, defaultBrowserOptions); + const browser = await puppeteer.launch(options); + const page = await browser.newPage(); + await page.goto(httpsServer.PREFIX + '/mixedcontent.html', {waitUntil: 'load'}); + await page.close(); + await browser.close(); + }); it('should reject all promises when browser is closed', async() => { const browser = await puppeteer.launch(defaultBrowserOptions); const page = await browser.newPage();