refactor: unify response tracking in page.goto and waitForNavigation (#3258)
This patch unifies logic in response trackign in page.goto and page.waitForNavigation. As a drive-by, we now make sure that we return the right response for the right frame. This will come handy for future frame navigation API. References #2918
This commit is contained in:
parent
a1a211d9e7
commit
b97bddf8e5
@ -16,6 +16,7 @@
|
|||||||
|
|
||||||
const {helper, assert} = require('./helper');
|
const {helper, assert} = require('./helper');
|
||||||
const {FrameManager} = require('./FrameManager');
|
const {FrameManager} = require('./FrameManager');
|
||||||
|
const {NetworkManager} = require('./NetworkManager');
|
||||||
const {TimeoutError} = require('./Errors');
|
const {TimeoutError} = require('./Errors');
|
||||||
const {Connection} = require('./Connection');
|
const {Connection} = require('./Connection');
|
||||||
|
|
||||||
@ -23,11 +24,12 @@ class NavigatorWatcher {
|
|||||||
/**
|
/**
|
||||||
* @param {!Puppeteer.CDPSession} client
|
* @param {!Puppeteer.CDPSession} client
|
||||||
* @param {!FrameManager} frameManager
|
* @param {!FrameManager} frameManager
|
||||||
|
* @param {!NetworkManager} networkManager
|
||||||
* @param {!Puppeteer.Frame} frame
|
* @param {!Puppeteer.Frame} frame
|
||||||
* @param {number} timeout
|
* @param {number} timeout
|
||||||
* @param {!Object=} options
|
* @param {!Object=} options
|
||||||
*/
|
*/
|
||||||
constructor(client, frameManager, frame, timeout, options = {}) {
|
constructor(client, frameManager, networkManager, frame, timeout, options = {}) {
|
||||||
assert(options.networkIdleTimeout === undefined, 'ERROR: networkIdleTimeout option is no longer supported.');
|
assert(options.networkIdleTimeout === undefined, 'ERROR: networkIdleTimeout option is no longer supported.');
|
||||||
assert(options.networkIdleInflight === undefined, 'ERROR: networkIdleInflight option is no longer supported.');
|
assert(options.networkIdleInflight === undefined, 'ERROR: networkIdleInflight option is no longer supported.');
|
||||||
assert(options.waitUntil !== 'networkidle', 'ERROR: "networkidle" option is no longer supported. Use "networkidle2" instead');
|
assert(options.waitUntil !== 'networkidle', 'ERROR: "networkidle" option is no longer supported. Use "networkidle2" instead');
|
||||||
@ -43,15 +45,19 @@ class NavigatorWatcher {
|
|||||||
});
|
});
|
||||||
|
|
||||||
this._frameManager = frameManager;
|
this._frameManager = frameManager;
|
||||||
|
this._networkManager = networkManager;
|
||||||
this._frame = frame;
|
this._frame = frame;
|
||||||
this._initialLoaderId = frame._loaderId;
|
this._initialLoaderId = frame._loaderId;
|
||||||
this._timeout = timeout;
|
this._timeout = timeout;
|
||||||
|
/** @type {?Puppeteer.Request} */
|
||||||
|
this._navigationRequest = null;
|
||||||
this._hasSameDocumentNavigation = false;
|
this._hasSameDocumentNavigation = false;
|
||||||
this._eventListeners = [
|
this._eventListeners = [
|
||||||
helper.addEventListener(Connection.fromSession(client), Connection.Events.Disconnected, () => this._terminate(new Error('Navigation failed because browser has disconnected!'))),
|
helper.addEventListener(Connection.fromSession(client), Connection.Events.Disconnected, () => this._terminate(new Error('Navigation failed because browser has disconnected!'))),
|
||||||
helper.addEventListener(this._frameManager, FrameManager.Events.LifecycleEvent, this._checkLifecycleComplete.bind(this)),
|
helper.addEventListener(this._frameManager, FrameManager.Events.LifecycleEvent, this._checkLifecycleComplete.bind(this)),
|
||||||
helper.addEventListener(this._frameManager, FrameManager.Events.FrameNavigatedWithinDocument, this._navigatedWithinDocument.bind(this)),
|
helper.addEventListener(this._frameManager, FrameManager.Events.FrameNavigatedWithinDocument, this._navigatedWithinDocument.bind(this)),
|
||||||
helper.addEventListener(this._frameManager, FrameManager.Events.FrameDetached, this._checkLifecycleComplete.bind(this))
|
helper.addEventListener(this._frameManager, FrameManager.Events.FrameDetached, this._checkLifecycleComplete.bind(this)),
|
||||||
|
helper.addEventListener(this._networkManager, NetworkManager.Events.Request, this._onRequest.bind(this)),
|
||||||
];
|
];
|
||||||
|
|
||||||
this._sameDocumentNavigationPromise = new Promise(fulfill => {
|
this._sameDocumentNavigationPromise = new Promise(fulfill => {
|
||||||
@ -68,6 +74,22 @@ class NavigatorWatcher {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {!Puppeteer.Request} request
|
||||||
|
*/
|
||||||
|
_onRequest(request) {
|
||||||
|
if (request.frame() !== this._frame || !request.isNavigationRequest())
|
||||||
|
return;
|
||||||
|
this._navigationRequest = request;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return {?Puppeteer.Response}
|
||||||
|
*/
|
||||||
|
navigationResponse() {
|
||||||
|
return this._navigationRequest ? this._navigationRequest.response() : null;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {!Error} error
|
* @param {!Error} error
|
||||||
*/
|
*/
|
||||||
|
23
lib/Page.js
23
lib/Page.js
@ -579,18 +579,9 @@ class Page extends EventEmitter {
|
|||||||
async goto(url, options = {}) {
|
async goto(url, options = {}) {
|
||||||
const referrer = typeof options.referer === 'string' ? options.referer : this._networkManager.extraHTTPHeaders()['referer'];
|
const referrer = typeof options.referer === 'string' ? options.referer : this._networkManager.extraHTTPHeaders()['referer'];
|
||||||
|
|
||||||
/** @type {Map<string, !Puppeteer.Request>} */
|
|
||||||
const requests = new Map();
|
|
||||||
const eventListeners = [
|
|
||||||
helper.addEventListener(this._networkManager, NetworkManager.Events.Request, request => {
|
|
||||||
if (!requests.get(request.url()))
|
|
||||||
requests.set(request.url(), request);
|
|
||||||
})
|
|
||||||
];
|
|
||||||
|
|
||||||
const mainFrame = this._frameManager.mainFrame();
|
const mainFrame = this._frameManager.mainFrame();
|
||||||
const timeout = typeof options.timeout === 'number' ? options.timeout : this._defaultNavigationTimeout;
|
const timeout = typeof options.timeout === 'number' ? options.timeout : this._defaultNavigationTimeout;
|
||||||
const watcher = new NavigatorWatcher(this._client, this._frameManager, mainFrame, timeout, options);
|
const watcher = new NavigatorWatcher(this._client, this._frameManager, this._networkManager, mainFrame, timeout, options);
|
||||||
let ensureNewDocumentNavigation = false;
|
let ensureNewDocumentNavigation = false;
|
||||||
let error = await Promise.race([
|
let error = await Promise.race([
|
||||||
navigate(this._client, url, referrer),
|
navigate(this._client, url, referrer),
|
||||||
@ -603,11 +594,9 @@ class Page extends EventEmitter {
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
watcher.dispose();
|
watcher.dispose();
|
||||||
helper.removeEventListeners(eventListeners);
|
|
||||||
if (error)
|
if (error)
|
||||||
throw error;
|
throw error;
|
||||||
const request = requests.get(mainFrame._navigationURL);
|
return watcher.navigationResponse();
|
||||||
return request ? request.response() : null;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {!Puppeteer.CDPSession} client
|
* @param {!Puppeteer.CDPSession} client
|
||||||
@ -645,20 +634,16 @@ class Page extends EventEmitter {
|
|||||||
async waitForNavigation(options = {}) {
|
async waitForNavigation(options = {}) {
|
||||||
const mainFrame = this._frameManager.mainFrame();
|
const mainFrame = this._frameManager.mainFrame();
|
||||||
const timeout = typeof options.timeout === 'number' ? options.timeout : this._defaultNavigationTimeout;
|
const timeout = typeof options.timeout === 'number' ? options.timeout : this._defaultNavigationTimeout;
|
||||||
const watcher = new NavigatorWatcher(this._client, this._frameManager, mainFrame, timeout, options);
|
const watcher = new NavigatorWatcher(this._client, this._frameManager, this._networkManager, mainFrame, timeout, options);
|
||||||
|
|
||||||
const responses = new Map();
|
|
||||||
const listener = helper.addEventListener(this._networkManager, NetworkManager.Events.Response, response => responses.set(response.url(), response));
|
|
||||||
const error = await Promise.race([
|
const error = await Promise.race([
|
||||||
watcher.timeoutOrTerminationPromise(),
|
watcher.timeoutOrTerminationPromise(),
|
||||||
watcher.sameDocumentNavigationPromise(),
|
watcher.sameDocumentNavigationPromise(),
|
||||||
watcher.newDocumentNavigationPromise()
|
watcher.newDocumentNavigationPromise()
|
||||||
]);
|
]);
|
||||||
watcher.dispose();
|
watcher.dispose();
|
||||||
helper.removeEventListeners([listener]);
|
|
||||||
if (error)
|
if (error)
|
||||||
throw error;
|
throw error;
|
||||||
return responses.get(this.mainFrame().url()) || null;
|
return watcher.navigationResponse();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Loading…
Reference in New Issue
Block a user