UNPKG

18.5 kBJavaScriptView Raw
1"use strict";
2var __importDefault = (this && this.__importDefault) || function (mod) {
3 return (mod && mod.__esModule) ? mod : { "default": mod };
4};
5Object.defineProperty(exports, "__esModule", { value: true });
6const events_1 = require("events");
7const time_limit_promise_1 = __importDefault(require("time-limit-promise"));
8const promisify_event_1 = __importDefault(require("promisify-event"));
9const lodash_1 = require("lodash");
10const map_reverse_1 = __importDefault(require("map-reverse"));
11const runtime_1 = require("../errors/runtime");
12const types_1 = require("../errors/types");
13const LOCAL_BROWSERS_READY_TIMEOUT = 2 * 60 * 1000;
14const REMOTE_BROWSERS_READY_TIMEOUT = 6 * 60 * 1000;
15class BrowserSet extends events_1.EventEmitter {
16 constructor(browserConnectionGroups) {
17 super();
18 this.RELEASE_TIMEOUT = 10000;
19 this.pendingReleases = [];
20 this.browserConnectionGroups = browserConnectionGroups;
21 this.browserConnections = lodash_1.flatten(browserConnectionGroups);
22 this.connectionsReadyTimeout = null;
23 this.browserErrorHandler = error => this.emit('error', error);
24 this.browserConnections.forEach(bc => bc.on('error', this.browserErrorHandler));
25 // NOTE: We're setting an empty error handler, because Node kills the process on an 'error' event
26 // if there is no handler. See: https://nodejs.org/api/events.html#events_class_events_eventemitter
27 this.on('error', lodash_1.noop);
28 }
29 static async _waitIdle(bc) {
30 if (bc.idle || !bc.ready)
31 return;
32 await promisify_event_1.default(bc, 'idle');
33 }
34 static async _closeConnection(bc) {
35 if (bc.closed || !bc.ready)
36 return;
37 bc.close();
38 await promisify_event_1.default(bc, 'closed');
39 }
40 async _getReadyTimeout() {
41 const isLocalBrowser = connection => connection.provider.isLocalBrowser(connection.id, connection.browserInfo.browserName);
42 const remoteBrowsersExist = (await Promise.all(this.browserConnections.map(isLocalBrowser))).indexOf(false) > -1;
43 return remoteBrowsersExist ? REMOTE_BROWSERS_READY_TIMEOUT : LOCAL_BROWSERS_READY_TIMEOUT;
44 }
45 _createPendingConnectionPromise(readyPromise, timeout, timeoutError) {
46 const timeoutPromise = new Promise((_, reject) => {
47 this.connectionsReadyTimeout = setTimeout(() => reject(timeoutError), timeout);
48 });
49 return Promise
50 .race([readyPromise, timeoutPromise])
51 .then(value => {
52 this.connectionsReadyTimeout.unref();
53 return value;
54 }, error => {
55 this.connectionsReadyTimeout.unref();
56 throw error;
57 });
58 }
59 async _waitConnectionsOpened() {
60 const connectionsReadyPromise = Promise.all(this.browserConnections
61 .filter(bc => !bc.opened)
62 .map(bc => promisify_event_1.default(bc, 'opened')));
63 const timeoutError = new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotEstablishBrowserConnection);
64 const readyTimeout = await this._getReadyTimeout();
65 await this._createPendingConnectionPromise(connectionsReadyPromise, readyTimeout, timeoutError);
66 }
67 _checkForDisconnections() {
68 const disconnectedUserAgents = this.browserConnections
69 .filter(bc => bc.closed)
70 .map(bc => bc.userAgent);
71 if (disconnectedUserAgents.length)
72 throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotRunAgainstDisconnectedBrowsers, disconnectedUserAgents.join(', '));
73 }
74 //API
75 static from(browserConnections) {
76 const browserSet = new BrowserSet(browserConnections);
77 const prepareConnection = Promise.resolve()
78 .then(() => {
79 browserSet._checkForDisconnections();
80 return browserSet._waitConnectionsOpened();
81 })
82 .then(() => browserSet);
83 return Promise
84 .race([
85 prepareConnection,
86 promisify_event_1.default(browserSet, 'error')
87 ])
88 .catch(async (error) => {
89 await browserSet.dispose();
90 throw error;
91 });
92 }
93 releaseConnection(bc) {
94 if (this.browserConnections.indexOf(bc) < 0)
95 return Promise.resolve();
96 lodash_1.pull(this.browserConnections, bc);
97 bc.removeListener('error', this.browserErrorHandler);
98 const appropriateStateSwitch = !bc.permanent ?
99 BrowserSet._closeConnection(bc) :
100 BrowserSet._waitIdle(bc);
101 const release = time_limit_promise_1.default(appropriateStateSwitch, this.RELEASE_TIMEOUT).then(() => lodash_1.pull(this.pendingReleases, release));
102 this.pendingReleases.push(release);
103 return release;
104 }
105 async dispose() {
106 // NOTE: When browserConnection is cancelled, it is removed from
107 // the this.connections array, which leads to shifting indexes
108 // towards the beginning. So, we must copy the array in order to iterate it,
109 // or we can perform iteration from the end to the beginning.
110 if (this.connectionsReadyTimeout)
111 this.connectionsReadyTimeout.unref();
112 map_reverse_1.default(this.browserConnections, bc => this.releaseConnection(bc));
113 await Promise.all(this.pendingReleases);
114 }
115}
116exports.default = BrowserSet;
117module.exports = exports.default;
118//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser-set.js","sourceRoot":"","sources":["../../src/runner/browser-set.js"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\nimport getTimeLimitedPromise from 'time-limit-promise';\nimport promisifyEvent from 'promisify-event';\nimport { noop, pull as remove, flatten } from 'lodash';\nimport mapReverse from 'map-reverse';\nimport { GeneralError } from '../errors/runtime';\nimport { RUNTIME_ERRORS } from '../errors/types';\n\nconst LOCAL_BROWSERS_READY_TIMEOUT  = 2 * 60 * 1000;\nconst REMOTE_BROWSERS_READY_TIMEOUT = 6 * 60 * 1000;\n\nexport default class BrowserSet extends EventEmitter {\n    constructor (browserConnectionGroups) {\n        super();\n\n        this.RELEASE_TIMEOUT = 10000;\n\n        this.pendingReleases = [];\n\n        this.browserConnectionGroups = browserConnectionGroups;\n        this.browserConnections      = flatten(browserConnectionGroups);\n\n        this.connectionsReadyTimeout = null;\n\n        this.browserErrorHandler = error => this.emit('error', error);\n\n        this.browserConnections.forEach(bc => bc.on('error', this.browserErrorHandler));\n\n        // NOTE: We're setting an empty error handler, because Node kills the process on an 'error' event\n        // if there is no handler. See: https://nodejs.org/api/events.html#events_class_events_eventemitter\n        this.on('error', noop);\n    }\n\n    static async _waitIdle (bc) {\n        if (bc.idle || !bc.ready)\n            return;\n\n        await promisifyEvent(bc, 'idle');\n    }\n\n    static async _closeConnection (bc) {\n        if (bc.closed || !bc.ready)\n            return;\n\n        bc.close();\n\n        await promisifyEvent(bc, 'closed');\n    }\n\n    async _getReadyTimeout () {\n        const isLocalBrowser      = connection => connection.provider.isLocalBrowser(connection.id, connection.browserInfo.browserName);\n        const remoteBrowsersExist = (await Promise.all(this.browserConnections.map(isLocalBrowser))).indexOf(false) > -1;\n\n        return remoteBrowsersExist ? REMOTE_BROWSERS_READY_TIMEOUT : LOCAL_BROWSERS_READY_TIMEOUT;\n    }\n\n    _createPendingConnectionPromise (readyPromise, timeout, timeoutError) {\n        const timeoutPromise = new Promise((_, reject) => {\n            this.connectionsReadyTimeout = setTimeout(() => reject(timeoutError), timeout);\n        });\n\n        return Promise\n            .race([readyPromise, timeoutPromise])\n            .then(\n                value => {\n                    this.connectionsReadyTimeout.unref();\n                    return value;\n                },\n                error => {\n                    this.connectionsReadyTimeout.unref();\n                    throw error;\n                }\n            );\n    }\n\n    async _waitConnectionsOpened () {\n        const connectionsReadyPromise = Promise.all(\n            this.browserConnections\n                .filter(bc => !bc.opened)\n                .map(bc => promisifyEvent(bc, 'opened'))\n        );\n\n        const timeoutError = new GeneralError(RUNTIME_ERRORS.cannotEstablishBrowserConnection);\n        const readyTimeout = await this._getReadyTimeout();\n\n        await this._createPendingConnectionPromise(connectionsReadyPromise, readyTimeout, timeoutError);\n    }\n\n    _checkForDisconnections () {\n        const disconnectedUserAgents = this.browserConnections\n            .filter(bc => bc.closed)\n            .map(bc => bc.userAgent);\n\n        if (disconnectedUserAgents.length)\n            throw new GeneralError(RUNTIME_ERRORS.cannotRunAgainstDisconnectedBrowsers, disconnectedUserAgents.join(', '));\n    }\n\n\n    //API\n    static from (browserConnections) {\n        const browserSet = new BrowserSet(browserConnections);\n\n        const prepareConnection = Promise.resolve()\n            .then(() => {\n                browserSet._checkForDisconnections();\n                return browserSet._waitConnectionsOpened();\n            })\n            .then(() => browserSet);\n\n        return Promise\n            .race([\n                prepareConnection,\n                promisifyEvent(browserSet, 'error')\n            ])\n            .catch(async error => {\n                await browserSet.dispose();\n\n                throw error;\n            });\n    }\n\n    releaseConnection (bc) {\n        if (this.browserConnections.indexOf(bc) < 0)\n            return Promise.resolve();\n\n        remove(this.browserConnections, bc);\n\n        bc.removeListener('error', this.browserErrorHandler);\n\n        const appropriateStateSwitch = !bc.permanent ?\n            BrowserSet._closeConnection(bc) :\n            BrowserSet._waitIdle(bc);\n\n        const release = getTimeLimitedPromise(appropriateStateSwitch, this.RELEASE_TIMEOUT).then(() => remove(this.pendingReleases, release));\n\n        this.pendingReleases.push(release);\n\n        return release;\n    }\n\n    async dispose () {\n        // NOTE: When browserConnection is cancelled, it is removed from\n        // the this.connections array, which leads to shifting indexes\n        // towards the beginning. So, we must copy the array in order to iterate it,\n        // or we can perform iteration from the end to the beginning.\n        if (this.connectionsReadyTimeout)\n            this.connectionsReadyTimeout.unref();\n\n        mapReverse(this.browserConnections, bc => this.releaseConnection(bc));\n\n        await Promise.all(this.pendingReleases);\n    }\n}\n"]}
\No newline at end of file