Test the 'networkidle' navigation logic

This patch adds a test to verify that navigation properly waits for the
network to become idle.

References #10
This commit is contained in:
Andrey Lushnikov 2017-06-27 22:02:46 -07:00 committed by GitHub
parent d5a91650ae
commit 5ed71fcb8f
4 changed files with 106 additions and 0 deletions

View File

@ -70,6 +70,7 @@ class Page extends EventEmitter {
client.on('Network.responseReceived', event => this.emit(Page.Events.ResponseReceived, event.response)); client.on('Network.responseReceived', event => this.emit(Page.Events.ResponseReceived, event.response));
client.on('Network.loadingFailed', event => this.emit(Page.Events.ResourceLoadingFailed, event)); client.on('Network.loadingFailed', event => this.emit(Page.Events.ResourceLoadingFailed, event));
client.on('Page.loadEventFired', event => this.emit(Page.Events.Load));
client.on('Network.requestIntercepted', event => this._onRequestIntercepted(event)); client.on('Network.requestIntercepted', event => this._onRequestIntercepted(event));
client.on('Runtime.consoleAPICalled', event => this._onConsoleAPI(event)); client.on('Runtime.consoleAPICalled', event => this._onConsoleAPI(event));
@ -632,6 +633,7 @@ Page.Events = {
FrameAttached: 'frameattached', FrameAttached: 'frameattached',
FrameDetached: 'framedetached', FrameDetached: 'framedetached',
FrameNavigated: 'framenavigated', FrameNavigated: 'framenavigated',
Load: 'load',
}; };
module.exports = Page; module.exports = Page;

View File

@ -20,6 +20,9 @@ let fs = require('fs');
let path = require('path'); let path = require('path');
let mime = require('mime'); let mime = require('mime');
const fulfillSymbol = Symbol('fullfill callback');
const rejectSymbol = Symbol('reject callback');
class StaticServer { class StaticServer {
/** /**
* @param {string} dirPath * @param {string} dirPath
@ -29,13 +32,69 @@ class StaticServer {
this._server = http.createServer(this._onRequest.bind(this)); this._server = http.createServer(this._onRequest.bind(this));
this._server.listen(port); this._server.listen(port);
this._dirPath = dirPath; this._dirPath = dirPath;
/** @type {!Map<string, function(!IncomingMessage, !ServerResponse)>} */
this._routes = new Map();
/** @type {!Map<string, !Promise>} */
this._requestSubscribers = new Map();
} }
stop() { stop() {
this._server.close(); this._server.close();
} }
/**
* @param {string} path
* @param {function(!IncomingMessage, !ServerResponse)} handler
*/
setRoute(path, handler) {
this._routes.set(path, handler);
}
/**
* @param {string} path
* @return {!Promise<!IncomingMessage>}
*/
waitForRequest(path) {
let promise = this._requestSubscribers.get(path);
if (promise)
return promise;
let fulfill, reject;
promise = new Promise((f, r) => {
fulfill = f;
reject = r;
});
promise[fulfillSymbol] = fulfill;
promise[rejectSymbol] = reject;
this._requestSubscribers.set(path, promise);
return promise;
}
reset() {
this._routes.clear();
let error = new Error('Static Server has been reset');
for (let subscriber of this._requestSubscribers.values())
subscriber[rejectSymbol].call(null, error);
this._requestSubscribers.clear();
}
_onRequest(request, response) { _onRequest(request, response) {
let pathName = url.parse(request.url).path;
// Notify request subscriber.
if (this._requestSubscribers.has(pathName))
this._requestSubscribers.get(pathName)[fulfillSymbol].call(null, request);
let handler = this._routes.get(pathName);
if (handler)
handler.call(null, request, response);
else
this.defaultHandler(request, response);
}
/**
* @param {!IncomingMessage} request
* @param {!ServerResponse} response
*/
defaultHandler(request, response) {
let pathName = url.parse(request.url).path; let pathName = url.parse(request.url).path;
if (pathName === '/') if (pathName === '/')
pathName = '/index.html'; pathName = '/index.html';

View File

@ -0,0 +1,5 @@
<script>
fetch('fetch-request-a.js');
fetch('fetch-request-b.js');
fetch('fetch-request-c.js');
</script>

View File

@ -43,6 +43,7 @@ describe('Puppeteer', function() {
beforeEach(SX(async function() { beforeEach(SX(async function() {
page = await browser.newPage(); page = await browser.newPage();
staticServer.reset();
GoldenUtils.addMatchers(jasmine); GoldenUtils.addMatchers(jasmine);
})); }));
@ -112,6 +113,45 @@ describe('Puppeteer', function() {
let success = await page.navigate(EMPTY_PAGE); let success = await page.navigate(EMPTY_PAGE);
expect(success).toBe(true); expect(success).toBe(true);
})); }));
it('should wait for network idle to succeed navigation', SX(async function() {
let responses = [];
// Hold on a bunch of requests without answering.
staticServer.setRoute('/fetch-request-a.js', (req, res) => responses.push(res));
staticServer.setRoute('/fetch-request-b.js', (req, res) => responses.push(res));
staticServer.setRoute('/fetch-request-c.js', (req, res) => responses.push(res));
let fetchResourcesRequested = Promise.all([
staticServer.waitForRequest('/fetch-request-a.js'),
staticServer.waitForRequest('/fetch-request-b.js'),
staticServer.waitForRequest('/fetch-request-c.js'),
]);
// Navigate to a page which loads immediately and then does a bunch of
// requests via javascript's fetch method.
let navigationPromise = page.navigate(STATIC_PREFIX + '/networkidle.html', {
minTime: 50 // Give page time to request more resources dynamically.
});
// Track when the navigation gets completed.
let navigationFinished = false;
navigationPromise.then(() => navigationFinished = true);
// Wait for the page's 'load' event.
await new Promise(fulfill => page.once('load', fulfill));
expect(navigationFinished).toBe(false);
// Wait for all three resources to be requested.
await fetchResourcesRequested;
// Expect navigation still to be not finished.
expect(navigationFinished).toBe(false);
// Respond to all requests.
for (let response of responses) {
response.statusCode = 404;
response.end(`File not found`);
}
let success = await navigationPromise;
// Expect navigation to succeed.
expect(success).toBe(true);
}));
}); });
describe('Page.setInPageCallback', function() { describe('Page.setInPageCallback', function() {