feat(firefox): switch over to WebSocket and support multiclient (#4022)
- switch transport from TCP to WS (yay!) - implemenet `puppeter.connect()`, `browser.disconnect()`, `'disconnected'` event and `browser.wsEndpoint()`
This commit is contained in:
parent
e0d4a5d2ec
commit
c35821a1a1
@ -40,6 +40,8 @@ class Browser extends EventEmitter {
|
|||||||
for (const browserContextId of browserContextIds)
|
for (const browserContextId of browserContextIds)
|
||||||
this._contexts.set(browserContextId, new BrowserContext(this._connection, this, browserContextId));
|
this._contexts.set(browserContextId, new BrowserContext(this._connection, this, browserContextId));
|
||||||
|
|
||||||
|
this._connection.on(Events.Connection.Disconnected, () => this.emit(Events.Browser.Disconnected));
|
||||||
|
|
||||||
this._eventListeners = [
|
this._eventListeners = [
|
||||||
helper.addEventListener(this._connection, 'Browser.tabOpened', this._onTabOpened.bind(this)),
|
helper.addEventListener(this._connection, 'Browser.tabOpened', this._onTabOpened.bind(this)),
|
||||||
helper.addEventListener(this._connection, 'Browser.tabClosed', this._onTabClosed.bind(this)),
|
helper.addEventListener(this._connection, 'Browser.tabClosed', this._onTabClosed.bind(this)),
|
||||||
@ -47,6 +49,14 @@ class Browser extends EventEmitter {
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
wsEndpoint() {
|
||||||
|
return this._connection.url();
|
||||||
|
}
|
||||||
|
|
||||||
|
disconnect() {
|
||||||
|
this._connection.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return {!BrowserContext}
|
* @return {!BrowserContext}
|
||||||
*/
|
*/
|
||||||
|
@ -25,8 +25,9 @@ class Connection extends EventEmitter {
|
|||||||
* @param {!Puppeteer.ConnectionTransport} transport
|
* @param {!Puppeteer.ConnectionTransport} transport
|
||||||
* @param {number=} delay
|
* @param {number=} delay
|
||||||
*/
|
*/
|
||||||
constructor(transport, delay = 0) {
|
constructor(url, transport, delay = 0) {
|
||||||
super();
|
super();
|
||||||
|
this._url = url;
|
||||||
this._lastId = 0;
|
this._lastId = 0;
|
||||||
/** @type {!Map<number, {resolve: function, reject: function, error: !Error, method: string}>}*/
|
/** @type {!Map<number, {resolve: function, reject: function, error: !Error, method: string}>}*/
|
||||||
this._callbacks = new Map();
|
this._callbacks = new Map();
|
||||||
@ -38,6 +39,10 @@ class Connection extends EventEmitter {
|
|||||||
this._closed = false;
|
this._closed = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
url() {
|
||||||
|
return this._url;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {string} method
|
* @param {string} method
|
||||||
* @param {!Object=} params
|
* @param {!Object=} params
|
||||||
|
@ -16,6 +16,7 @@ const Events = {
|
|||||||
RequestFailed: 'requestfailed',
|
RequestFailed: 'requestfailed',
|
||||||
},
|
},
|
||||||
Browser: {
|
Browser: {
|
||||||
|
Disconnected: 'disconnected',
|
||||||
TargetCreated: 'targetcreated',
|
TargetCreated: 'targetcreated',
|
||||||
TargetChanged: 'targetchanged',
|
TargetChanged: 'targetchanged',
|
||||||
TargetDestroyed: 'targetdestroyed',
|
TargetDestroyed: 'targetdestroyed',
|
||||||
|
@ -23,9 +23,9 @@ const {BrowserFetcher} = require('./BrowserFetcher');
|
|||||||
const readline = require('readline');
|
const readline = require('readline');
|
||||||
const fs = require('fs');
|
const fs = require('fs');
|
||||||
const util = require('util');
|
const util = require('util');
|
||||||
const {helper} = require('./helper');
|
const {helper, debugError} = require('./helper');
|
||||||
const {TimeoutError} = require('./Errors')
|
const {TimeoutError} = require('./Errors')
|
||||||
const FirefoxTransport = require('./FirefoxTransport');
|
const WebSocketTransport = require('./WebSocketTransport');
|
||||||
|
|
||||||
const mkdtempAsync = util.promisify(fs.mkdtemp);
|
const mkdtempAsync = util.promisify(fs.mkdtemp);
|
||||||
const removeFolderAsync = util.promisify(removeFolder);
|
const removeFolderAsync = util.promisify(removeFolder);
|
||||||
@ -121,14 +121,13 @@ class Launcher {
|
|||||||
/** @type {?Connection} */
|
/** @type {?Connection} */
|
||||||
let connection = null;
|
let connection = null;
|
||||||
try {
|
try {
|
||||||
const port = await waitForWSEndpoint(firefoxProcess, 30000);
|
const url = await waitForWSEndpoint(firefoxProcess, 30000);
|
||||||
const transport = await FirefoxTransport.create(parseInt(port, 10));
|
const transport = await WebSocketTransport.create(url);
|
||||||
connection = new Connection(transport, slowMo);
|
connection = new Connection(url, transport, slowMo);
|
||||||
const browser = await Browser.create(connection, defaultViewport, firefoxProcess, killFirefox);
|
const browser = await Browser.create(connection, defaultViewport, firefoxProcess, killFirefox);
|
||||||
if (ignoreHTTPSErrors)
|
if (ignoreHTTPSErrors)
|
||||||
await connection.send('Browser.setIgnoreHTTPSErrors', {enabled: true});
|
await connection.send('Browser.setIgnoreHTTPSErrors', {enabled: true});
|
||||||
if (!browser.targets().length)
|
await browser.waitForTarget(t => t.type() === 'page');
|
||||||
await new Promise(x => browser.once('targetcreated', x));
|
|
||||||
return browser;
|
return browser;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
killFirefox();
|
killFirefox();
|
||||||
@ -156,6 +155,26 @@ class Launcher {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Object} options
|
||||||
|
* @return {!Promise<!Browser>}
|
||||||
|
*/
|
||||||
|
async connect(options = {}) {
|
||||||
|
const {
|
||||||
|
browserWSEndpoint,
|
||||||
|
slowMo = 0,
|
||||||
|
defaultViewport = {width: 800, height: 600},
|
||||||
|
ignoreHTTPSErrors = false,
|
||||||
|
} = options;
|
||||||
|
let connection = null;
|
||||||
|
const transport = await WebSocketTransport.create(browserWSEndpoint);
|
||||||
|
connection = new Connection(browserWSEndpoint, transport, slowMo);
|
||||||
|
const browser = await Browser.create(connection, defaultViewport, null, () => connection.send('Browser.close').catch(debugError));
|
||||||
|
if (ignoreHTTPSErrors)
|
||||||
|
await connection.send('Browser.setIgnoreHTTPSErrors', {enabled: true});
|
||||||
|
return browser;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return {string}
|
* @return {string}
|
||||||
*/
|
*/
|
||||||
@ -205,7 +224,7 @@ function waitForWSEndpoint(firefoxProcess, timeout) {
|
|||||||
*/
|
*/
|
||||||
function onLine(line) {
|
function onLine(line) {
|
||||||
stderr += line + '\n';
|
stderr += line + '\n';
|
||||||
const match = line.match(/^Juggler listening on (\d+)$/);
|
const match = line.match(/^Juggler listening on (ws:\/\/.*)$/);
|
||||||
if (!match)
|
if (!match)
|
||||||
return;
|
return;
|
||||||
cleanup();
|
cleanup();
|
||||||
|
@ -12,9 +12,9 @@ class NetworkManager extends EventEmitter {
|
|||||||
this._frameManager = null;
|
this._frameManager = null;
|
||||||
|
|
||||||
this._eventListeners = [
|
this._eventListeners = [
|
||||||
helper.addEventListener(session, 'Page.requestWillBeSent', this._onRequestWillBeSent.bind(this)),
|
helper.addEventListener(session, 'Network.requestWillBeSent', this._onRequestWillBeSent.bind(this)),
|
||||||
helper.addEventListener(session, 'Page.responseReceived', this._onResponseReceived.bind(this)),
|
helper.addEventListener(session, 'Network.responseReceived', this._onResponseReceived.bind(this)),
|
||||||
helper.addEventListener(session, 'Page.requestFinished', this._onRequestFinished.bind(this)),
|
helper.addEventListener(session, 'Network.requestFinished', this._onRequestFinished.bind(this)),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -15,6 +15,10 @@ class Puppeteer {
|
|||||||
return this._launcher.launch(options);
|
return this._launcher.launch(options);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async connect(options) {
|
||||||
|
return this._launcher.connect(options);
|
||||||
|
}
|
||||||
|
|
||||||
createBrowserFetcher(options) {
|
createBrowserFetcher(options) {
|
||||||
return new BrowserFetcher(this._projectRoot, options);
|
return new BrowserFetcher(this._projectRoot, options);
|
||||||
}
|
}
|
||||||
|
@ -13,63 +13,39 @@
|
|||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
const {Socket} = require('net');
|
const WebSocket = require('ws');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @implements {!Puppeteer.ConnectionTransport}
|
* @implements {!Puppeteer.ConnectionTransport}
|
||||||
* @internal
|
|
||||||
*/
|
*/
|
||||||
class FirefoxTransport {
|
class WebSocketTransport {
|
||||||
/**
|
/**
|
||||||
* @param {number} port
|
* @param {string} url
|
||||||
* @return {!Promise<!FirefoxTransport>}
|
* @return {!Promise<!WebSocketTransport>}
|
||||||
*/
|
*/
|
||||||
static async create(port) {
|
static create(url) {
|
||||||
const socket = new Socket();
|
return new Promise((resolve, reject) => {
|
||||||
try {
|
const ws = new WebSocket(url, [], { perMessageDeflate: false });
|
||||||
await new Promise((resolve, reject) => {
|
ws.addEventListener('open', () => resolve(new WebSocketTransport(ws)));
|
||||||
socket.once('connect', resolve);
|
ws.addEventListener('error', reject);
|
||||||
socket.once('error', reject);
|
});
|
||||||
socket.connect({
|
|
||||||
port,
|
|
||||||
host: 'localhost'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
} catch (e) {
|
|
||||||
socket.destroy();
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
return new FirefoxTransport(socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {!Socket} socket
|
* @param {!WebSocket} ws
|
||||||
*/
|
*/
|
||||||
constructor(socket) {
|
constructor(ws) {
|
||||||
this._socket = socket;
|
this._ws = ws;
|
||||||
this._socket.once('close', had_error => {
|
this._dispatchQueue = new DispatchQueue(this);
|
||||||
|
this._ws.addEventListener('message', event => {
|
||||||
|
this._dispatchQueue.enqueue(event.data);
|
||||||
|
});
|
||||||
|
this._ws.addEventListener('close', event => {
|
||||||
if (this.onclose)
|
if (this.onclose)
|
||||||
this.onclose.call(null);
|
this.onclose.call(null);
|
||||||
});
|
});
|
||||||
this._dispatchQueue = new DispatchQueue(this);
|
|
||||||
let buffer = Buffer.from('');
|
|
||||||
socket.on('data', async data => {
|
|
||||||
buffer = Buffer.concat([buffer, data]);
|
|
||||||
while (true) {
|
|
||||||
const bufferString = buffer.toString();
|
|
||||||
const seperatorIndex = bufferString.indexOf(':');
|
|
||||||
if (seperatorIndex === -1)
|
|
||||||
return;
|
|
||||||
const length = parseInt(bufferString.substring(0, seperatorIndex), 10);
|
|
||||||
if (buffer.length < length + seperatorIndex)
|
|
||||||
return;
|
|
||||||
const message = buffer.slice(seperatorIndex + 1, seperatorIndex + 1 + length).toString();
|
|
||||||
buffer = buffer.slice(seperatorIndex + 1 + length);
|
|
||||||
this._dispatchQueue.enqueue(message);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
// Silently ignore all errors - we don't know what to do with them.
|
// Silently ignore all errors - we don't know what to do with them.
|
||||||
this._socket.on('error', () => {});
|
this._ws.addEventListener('error', () => {});
|
||||||
this.onmessage = null;
|
this.onmessage = null;
|
||||||
this.onclose = null;
|
this.onclose = null;
|
||||||
}
|
}
|
||||||
@ -78,11 +54,11 @@ class FirefoxTransport {
|
|||||||
* @param {string} message
|
* @param {string} message
|
||||||
*/
|
*/
|
||||||
send(message) {
|
send(message) {
|
||||||
this._socket.write(Buffer.byteLength(message) + ':' + message);
|
this._ws.send(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
close() {
|
close() {
|
||||||
this._socket.destroy();
|
this._ws.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -123,4 +99,4 @@ class DispatchQueue {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = FirefoxTransport;
|
module.exports = WebSocketTransport;
|
@ -9,7 +9,7 @@
|
|||||||
"node": ">=8.9.4"
|
"node": ">=8.9.4"
|
||||||
},
|
},
|
||||||
"puppeteer": {
|
"puppeteer": {
|
||||||
"firefox_revision": "fd017c27c17d0b4fa8bdea3ad40b88ca2addaeda"
|
"firefox_revision": "c74102def6c16584c155a98741e8143ab5d615b9"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"install": "node install.js",
|
"install": "node install.js",
|
||||||
|
@ -31,7 +31,7 @@ module.exports.addTests = function({testRunner, expect, headless, puppeteer}) {
|
|||||||
const process = await browser.process();
|
const process = await browser.process();
|
||||||
expect(process.pid).toBeGreaterThan(0);
|
expect(process.pid).toBeGreaterThan(0);
|
||||||
});
|
});
|
||||||
it_fails_ffox('should not return child_process for remote browser', async function({browser}) {
|
it('should not return child_process for remote browser', async function({browser}) {
|
||||||
const browserWSEndpoint = browser.wsEndpoint();
|
const browserWSEndpoint = browser.wsEndpoint();
|
||||||
const remoteBrowser = await puppeteer.connect({browserWSEndpoint});
|
const remoteBrowser = await puppeteer.connect({browserWSEndpoint});
|
||||||
expect(remoteBrowser.process()).toBe(null);
|
expect(remoteBrowser.process()).toBe(null);
|
||||||
|
@ -141,7 +141,7 @@ module.exports.addTests = function({testRunner, expect, puppeteer, Errors}) {
|
|||||||
]);
|
]);
|
||||||
expect(browser.browserContexts().length).toBe(1);
|
expect(browser.browserContexts().length).toBe(1);
|
||||||
});
|
});
|
||||||
it_fails_ffox('should work across sessions', async function({browser, server}) {
|
it('should work across sessions', async function({browser, server}) {
|
||||||
expect(browser.browserContexts().length).toBe(1);
|
expect(browser.browserContexts().length).toBe(1);
|
||||||
const context = await browser.createIncognitoBrowserContext();
|
const context = await browser.createIncognitoBrowserContext();
|
||||||
expect(browser.browserContexts().length).toBe(2);
|
expect(browser.browserContexts().length).toBe(2);
|
||||||
|
@ -300,7 +300,7 @@ module.exports.addTests = function({testRunner, expect, defaultBrowserOptions, p
|
|||||||
await browser.close();
|
await browser.close();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
describe_fails_ffox('Puppeteer.connect', function() {
|
describe('Puppeteer.connect', function() {
|
||||||
it('should be able to connect multiple times to the same browser', async({server}) => {
|
it('should be able to connect multiple times to the same browser', async({server}) => {
|
||||||
const originalBrowser = await puppeteer.launch(defaultBrowserOptions);
|
const originalBrowser = await puppeteer.launch(defaultBrowserOptions);
|
||||||
const browser = await puppeteer.connect({
|
const browser = await puppeteer.connect({
|
||||||
@ -349,7 +349,7 @@ module.exports.addTests = function({testRunner, expect, defaultBrowserOptions, p
|
|||||||
expect(await restoredPage.evaluate(() => 7 * 8)).toBe(56);
|
expect(await restoredPage.evaluate(() => 7 * 8)).toBe(56);
|
||||||
await browser.close();
|
await browser.close();
|
||||||
});
|
});
|
||||||
it('should be able to connect using browserUrl, with and without trailing slash', async({server}) => {
|
it_fails_ffox('should be able to connect using browserUrl, with and without trailing slash', async({server}) => {
|
||||||
const originalBrowser = await puppeteer.launch(Object.assign({}, defaultBrowserOptions, {
|
const originalBrowser = await puppeteer.launch(Object.assign({}, defaultBrowserOptions, {
|
||||||
args: ['--remote-debugging-port=21222']
|
args: ['--remote-debugging-port=21222']
|
||||||
}));
|
}));
|
||||||
@ -366,7 +366,7 @@ module.exports.addTests = function({testRunner, expect, defaultBrowserOptions, p
|
|||||||
browser2.disconnect();
|
browser2.disconnect();
|
||||||
originalBrowser.close();
|
originalBrowser.close();
|
||||||
});
|
});
|
||||||
it('should throw when using both browserWSEndpoint and browserURL', async({server}) => {
|
it_fails_ffox('should throw when using both browserWSEndpoint and browserURL', async({server}) => {
|
||||||
const originalBrowser = await puppeteer.launch(Object.assign({}, defaultBrowserOptions, {
|
const originalBrowser = await puppeteer.launch(Object.assign({}, defaultBrowserOptions, {
|
||||||
args: ['--remote-debugging-port=21222']
|
args: ['--remote-debugging-port=21222']
|
||||||
}));
|
}));
|
||||||
@ -378,7 +378,7 @@ module.exports.addTests = function({testRunner, expect, defaultBrowserOptions, p
|
|||||||
|
|
||||||
originalBrowser.close();
|
originalBrowser.close();
|
||||||
});
|
});
|
||||||
it('should throw when trying to connect to non-existing browser', async({server}) => {
|
it_fails_ffox('should throw when trying to connect to non-existing browser', async({server}) => {
|
||||||
const originalBrowser = await puppeteer.launch(Object.assign({}, defaultBrowserOptions, {
|
const originalBrowser = await puppeteer.launch(Object.assign({}, defaultBrowserOptions, {
|
||||||
args: ['--remote-debugging-port=21222']
|
args: ['--remote-debugging-port=21222']
|
||||||
}));
|
}));
|
||||||
@ -414,7 +414,7 @@ module.exports.addTests = function({testRunner, expect, defaultBrowserOptions, p
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe_fails_ffox('Browser.Events.disconnected', function() {
|
describe('Browser.Events.disconnected', function() {
|
||||||
it('should be emitted when: browser gets closed, disconnected or underlying websocket gets closed', async() => {
|
it('should be emitted when: browser gets closed, disconnected or underlying websocket gets closed', async() => {
|
||||||
const originalBrowser = await puppeteer.launch(defaultBrowserOptions);
|
const originalBrowser = await puppeteer.launch(defaultBrowserOptions);
|
||||||
const browserWSEndpoint = originalBrowser.wsEndpoint();
|
const browserWSEndpoint = originalBrowser.wsEndpoint();
|
||||||
|
Loading…
Reference in New Issue
Block a user