"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const debug_1 = __importDefault(require("debug")); const time_limit_promise_1 = __importDefault(require("time-limit-promise")); const events_1 = require("events"); const mustache_1 = __importDefault(require("mustache")); const lodash_1 = require("lodash"); const parse_user_agent_1 = require("../../utils/parse-user-agent"); const read_file_relative_1 = require("read-file-relative"); const promisify_event_1 = __importDefault(require("promisify-event")); const nanoid_1 = require("nanoid"); const command_1 = __importDefault(require("./command")); const status_1 = __importDefault(require("./status")); const heartbeat_status_1 = __importDefault(require("./heartbeat-status")); const runtime_1 = require("../../errors/runtime"); const types_1 = require("../../errors/types"); const warning_log_1 = __importDefault(require("../../notifications/warning-log")); const service_routes_1 = __importDefault(require("./service-routes")); const browser_connection_timeouts_1 = require("../../utils/browser-connection-timeouts"); const tracker_1 = __importDefault(require("./tracker")); const getBrowserConnectionDebugScope = (id) => `testcafe:browser:connection:${id}`; const IDLE_PAGE_TEMPLATE = (0, read_file_relative_1.readSync)('../../client/browser/idle-page/index.html.mustache'); class BrowserConnection extends events_1.EventEmitter { constructor(gateway, browserInfo, permanent, disableMultipleWindows = false, proxyless = false, messageBus) { super(); this._currentTestRun = null; this.url = ''; this.idleUrl = ''; this.forcedIdleUrl = ''; this.initScriptUrl = ''; this.heartbeatUrl = ''; this.statusUrl = ''; this.activeWindowIdUrl = ''; this.closeWindowUrl = ''; this.statusDoneUrl = ''; this.heartbeatRelativeUrl = ''; this.statusRelativeUrl = ''; this.statusDoneRelativeUrl = ''; this.idleRelativeUrl = ''; this.openFileProtocolRelativeUrl = ''; this.openFileProtocolUrl = ''; this.dispatchProxylessEventRelativeUrl = ''; this.osInfo = null; this.HEARTBEAT_TIMEOUT = browser_connection_timeouts_1.HEARTBEAT_TIMEOUT; this.BROWSER_CLOSE_TIMEOUT = browser_connection_timeouts_1.BROWSER_CLOSE_TIMEOUT; this.BROWSER_RESTART_TIMEOUT = browser_connection_timeouts_1.BROWSER_RESTART_TIMEOUT; this.id = BrowserConnection._generateId(); this.jobQueue = []; this.initScriptsQueue = []; this.browserConnectionGateway = gateway; this.disconnectionPromise = null; this.testRunAborted = false; this.warningLog = new warning_log_1.default(null, warning_log_1.default.createAddWarningCallback(messageBus)); this.debugLogger = (0, debug_1.default)(getBrowserConnectionDebugScope(this.id)); if (messageBus) this.messageBus = messageBus; this.browserInfo = browserInfo; this.browserInfo.userAgentProviderMetaInfo = ''; this.provider = browserInfo.provider; this.permanent = permanent; this.status = status_1.default.uninitialized; this.idle = true; this.heartbeatTimeout = null; this.pendingTestRunInfo = null; this.disableMultipleWindows = disableMultipleWindows; this.proxyless = proxyless; this._buildCommunicationUrls(gateway.proxy); this._setEventHandlers(); tracker_1.default.add(this); this.previousActiveWindowId = null; this.browserConnectionGateway.startServingConnection(this); // NOTE: Give a caller time to assign event listeners process.nextTick(() => this._runBrowser()); } _buildCommunicationUrls(proxy) { this.url = proxy.resolveRelativeServiceUrl(`${service_routes_1.default.connect}/${this.id}`); this.forcedIdleUrl = proxy.resolveRelativeServiceUrl(`${service_routes_1.default.idleForced}/${this.id}`); this.initScriptUrl = proxy.resolveRelativeServiceUrl(`${service_routes_1.default.initScript}/${this.id}`); this.heartbeatRelativeUrl = `${service_routes_1.default.heartbeat}/${this.id}`; this.statusRelativeUrl = `${service_routes_1.default.status}/${this.id}`; this.statusDoneRelativeUrl = `${service_routes_1.default.statusDone}/${this.id}`; this.idleRelativeUrl = `${service_routes_1.default.idle}/${this.id}`; this.activeWindowIdUrl = `${service_routes_1.default.activeWindowId}/${this.id}`; this.closeWindowUrl = `${service_routes_1.default.closeWindow}/${this.id}`; this.openFileProtocolRelativeUrl = `${service_routes_1.default.openFileProtocol}/${this.id}`; this.dispatchProxylessEventRelativeUrl = `${service_routes_1.default.dispatchProxylessEvent}/${this.id}`; this.idleUrl = proxy.resolveRelativeServiceUrl(this.idleRelativeUrl); this.heartbeatUrl = proxy.resolveRelativeServiceUrl(this.heartbeatRelativeUrl); this.statusUrl = proxy.resolveRelativeServiceUrl(this.statusRelativeUrl); this.statusDoneUrl = proxy.resolveRelativeServiceUrl(this.statusDoneRelativeUrl); this.openFileProtocolUrl = proxy.resolveRelativeServiceUrl(this.openFileProtocolRelativeUrl); } set messageBus(messageBus) { this._messageBus = messageBus; this.warningLog.callback = warning_log_1.default.createAddWarningCallback(this._messageBus); if (messageBus) { messageBus.on('test-run-start', testRun => { if (testRun.browserConnection.id === this.id) this._currentTestRun = testRun; }); } this.emit('message-bus-initialized', messageBus); } get messageBus() { return this._messageBus; } _setEventHandlers() { this.on('error', e => { this.debugLogger(e); this._forceIdle(); this.close(); }); for (const name in status_1.default) { const status = status_1.default[name]; this.on(status, () => { this.debugLogger(`status changed to '${status}'`); }); } } static _generateId() { return (0, nanoid_1.nanoid)(7); } _getAdditionalBrowserOptions() { const options = { disableMultipleWindows: this.disableMultipleWindows, }; if (this.proxyless) { options.proxyless = { serviceDomains: [ this.browserConnectionGateway.proxy.server1Info.domain, this.browserConnectionGateway.proxy.server2Info.domain, ], developmentMode: this.browserConnectionGateway.proxy.options.developmentMode, }; } return options; } async _runBrowser() { try { const additionalOptions = this._getAdditionalBrowserOptions(); await this.provider.openBrowser(this.id, this.url, this.browserInfo.browserOption, additionalOptions); if (this.status !== status_1.default.ready) await (0, promisify_event_1.default)(this, 'ready'); this.status = status_1.default.opened; this.emit('opened'); } catch (err) { this.emit('error', new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.unableToOpenBrowser, this.browserInfo.providerName + ':' + this.browserInfo.browserName, err.stack)); } } async _closeBrowser(data = {}) { if (!this.idle) await (0, promisify_event_1.default)(this, 'idle'); try { await this.provider.closeBrowser(this.id, data); } catch (err) { // NOTE: A warning would be really nice here, but it can't be done while log is stored in a task. this.debugLogger(err); } } _forceIdle() { if (!this.idle) { this.idle = true; this.emit('idle'); } } _createBrowserDisconnectedError() { return new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.browserDisconnected, this.userAgent); } _waitForHeartbeat() { this.heartbeatTimeout = setTimeout(() => { const err = this._createBrowserDisconnectedError(); this.status = status_1.default.disconnected; this.testRunAborted = true; this.emit('disconnected', err); this._restartBrowserOnDisconnect(err); }, this.HEARTBEAT_TIMEOUT); } async _getTestRunInfo(needPopNext) { if (needPopNext || !this.pendingTestRunInfo) this.pendingTestRunInfo = await this._popNextTestRunInfo(); return this.pendingTestRunInfo; } async _popNextTestRunInfo() { while (this.hasQueuedJobs && !this.currentJob.hasQueuedTestRuns) this.jobQueue.shift(); return this.hasQueuedJobs ? await this.currentJob.popNextTestRunInfo(this) : null; } getCurrentTestRun() { return this._currentTestRun; } static getById(id) { return tracker_1.default.activeBrowserConnections[id] || null; } async _restartBrowser() { this.status = status_1.default.uninitialized; this._forceIdle(); let resolveTimeout = null; let isTimeoutExpired = false; let timeout = null; const restartPromise = (0, time_limit_promise_1.default)(this._closeBrowser({ isRestarting: true }), this.BROWSER_CLOSE_TIMEOUT, { rejectWith: new runtime_1.TimeoutError() }) .catch(err => this.debugLogger(err)) .then(() => this._runBrowser()); const timeoutPromise = new Promise(resolve => { resolveTimeout = resolve; timeout = setTimeout(() => { isTimeoutExpired = true; resolve(); }, this.BROWSER_RESTART_TIMEOUT); }); return Promise.race([restartPromise, timeoutPromise]) .then(() => { clearTimeout(timeout); if (isTimeoutExpired) this.emit('error', this._createBrowserDisconnectedError()); else resolveTimeout(); }); } _restartBrowserOnDisconnect(err) { let resolveFn = null; let rejectFn = null; this.disconnectionPromise = new Promise((resolve, reject) => { resolveFn = resolve; rejectFn = () => { reject(err); }; setTimeout(() => { rejectFn(); }); }) .then(() => { return this._restartBrowser(); }) .catch(e => { this.emit('error', e); }); this.disconnectionPromise.resolve = resolveFn; this.disconnectionPromise.reject = rejectFn; } async getDefaultBrowserInitTimeout() { const isLocalBrowser = await this.provider.isLocalBrowser(this.id, this.browserInfo.browserName); return isLocalBrowser ? browser_connection_timeouts_1.LOCAL_BROWSER_INIT_TIMEOUT : browser_connection_timeouts_1.REMOTE_BROWSER_INIT_TIMEOUT; } async processDisconnection(disconnectionThresholdExceeded) { const { resolve, reject } = this.disconnectionPromise; if (disconnectionThresholdExceeded) reject(); else resolve(); } addWarning(message, ...args) { if (this.currentJob) this.currentJob.warningLog.addWarning(message, ...args); else this.warningLog.addWarning(message, ...args); } _appendToPrettyUserAgent(str) { this.browserInfo.parsedUserAgent.prettyUserAgent += ` (${str})`; } _moveWarningLogToJob(job) { job.warningLog.copyFrom(this.warningLog); this.warningLog.clear(); } setProviderMetaInfo(str, options) { const appendToUserAgent = options === null || options === void 0 ? void 0 : options.appendToUserAgent; if (appendToUserAgent) { // NOTE: // change prettyUserAgent only when connection already was established if (this.isReady()) this._appendToPrettyUserAgent(str); else this.on('ready', () => this._appendToPrettyUserAgent(str)); return; } this.browserInfo.userAgentProviderMetaInfo = str; } get userAgent() { let userAgent = this.browserInfo.parsedUserAgent.prettyUserAgent; if (this.browserInfo.userAgentProviderMetaInfo) userAgent += ` (${this.browserInfo.userAgentProviderMetaInfo})`; return userAgent; } get connectionInfo() { if (!this.osInfo) return this.userAgent; const { name, version } = this.browserInfo.parsedUserAgent; let connectionInfo = (0, parse_user_agent_1.calculatePrettyUserAgent)({ name, version }, this.osInfo); const metaInfo = this.browserInfo.userAgentProviderMetaInfo || (0, parse_user_agent_1.extractMetaInfo)(this.browserInfo.parsedUserAgent.prettyUserAgent); if (metaInfo) connectionInfo += ` (${metaInfo})`; return connectionInfo; } get retryTestPages() { return this.browserConnectionGateway.retryTestPages; } get hasQueuedJobs() { return !!this.jobQueue.length; } get currentJob() { return this.jobQueue[0]; } // API runInitScript(code) { return new Promise(resolve => this.initScriptsQueue.push({ code, resolve })); } addJob(job) { this.jobQueue.push(job); this._moveWarningLogToJob(job); } removeJob(job) { (0, lodash_1.pull)(this.jobQueue, job); } async close() { if (this.status === status_1.default.closing || this.status === status_1.default.closed) return; this.status = status_1.default.closing; this.emit(status_1.default.closing); await this._closeBrowser(); this.browserConnectionGateway.stopServingConnection(this); if (this.heartbeatTimeout) clearTimeout(this.heartbeatTimeout); tracker_1.default.remove(this); this.status = status_1.default.closed; this.emit(status_1.default.closed); } async establish(userAgent) { this.status = status_1.default.ready; this.browserInfo.parsedUserAgent = (0, parse_user_agent_1.parseUserAgent)(userAgent); this.osInfo = await this.provider.getOSInfo(this.id); this._waitForHeartbeat(); this.emit('ready'); } heartbeat() { if (this.heartbeatTimeout) clearTimeout(this.heartbeatTimeout); this._waitForHeartbeat(); return { code: this.status === status_1.default.closing ? heartbeat_status_1.default.closing : heartbeat_status_1.default.ok, url: this.status === status_1.default.closing ? this.idleUrl : '', }; } renderIdlePage() { return mustache_1.default.render(IDLE_PAGE_TEMPLATE, { userAgent: this.connectionInfo, statusUrl: this.statusUrl, heartbeatUrl: this.heartbeatUrl, initScriptUrl: this.initScriptUrl, openFileProtocolUrl: this.openFileProtocolUrl, retryTestPages: !!this.browserConnectionGateway.retryTestPages, proxyless: this.proxyless, }); } getInitScript() { const initScriptPromise = this.initScriptsQueue[0]; return { code: initScriptPromise ? initScriptPromise.code : null }; } handleInitScriptResult(data) { const initScriptPromise = this.initScriptsQueue.shift(); if (initScriptPromise) initScriptPromise.resolve(JSON.parse(data)); } isHeadlessBrowser() { return this.provider.isHeadlessBrowser(this.id); } async reportJobResult(status, data) { await this.provider.reportJobResult(this.id, status, data); } async getStatus(isTestDone) { if (!this.idle && !isTestDone) { this.idle = true; this.emit('idle'); } if (this.status === status_1.default.opened) { const nextTestRunInfo = await this._getTestRunInfo(isTestDone || this.testRunAborted); this.testRunAborted = false; if (nextTestRunInfo) { this.idle = false; return { cmd: command_1.default.run, testRunId: nextTestRunInfo.testRunId, url: nextTestRunInfo.url, }; } } return { cmd: command_1.default.idle, url: this.idleUrl, testRunId: null, }; } get activeWindowId() { return this.provider.getActiveWindowId(this.id); } set activeWindowId(val) { this.previousActiveWindowId = this.activeWindowId; this.provider.setActiveWindowId(this.id, val); } async openFileProtocol(url) { return this.provider.openFileProtocol(this.id, url); } async dispatchProxylessEvent(type, options) { return this.provider.dispatchProxylessEvent(this.id, type, options); } async canUseDefaultWindowActions() { return this.provider.canUseDefaultWindowActions(this.id); } isReady() { return this.status === status_1.default.ready || this.status === status_1.default.opened || this.status === status_1.default.closing; } } exports.default = BrowserConnection; module.exports = exports.default; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/browser/connection/index.ts"],"names":[],"mappings":"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","sourcesContent":["import debug from 'debug';\nimport timeLimit from 'time-limit-promise';\nimport { EventEmitter } from 'events';\nimport Mustache from 'mustache';\nimport { pull as remove } from 'lodash';\nimport {\n    calculatePrettyUserAgent,\n    extractMetaInfo,\n    ParsedUserAgent,\n    parseUserAgent,\n} from '../../utils/parse-user-agent';\nimport { readSync as read } from 'read-file-relative';\nimport promisifyEvent from 'promisify-event';\nimport { nanoid } from 'nanoid';\nimport COMMAND from './command';\nimport BrowserConnectionStatus from './status';\nimport HeartbeatStatus from './heartbeat-status';\nimport { GeneralError, TimeoutError } from '../../errors/runtime';\nimport { RUNTIME_ERRORS } from '../../errors/types';\nimport BrowserConnectionGateway from './gateway';\nimport BrowserJob from '../../runner/browser-job';\nimport WarningLog from '../../notifications/warning-log';\nimport BrowserProvider from '../provider';\nimport { OSInfo } from 'get-os-info';\nimport SERVICE_ROUTES from './service-routes';\nimport {\n    BROWSER_RESTART_TIMEOUT,\n    BROWSER_CLOSE_TIMEOUT,\n    HEARTBEAT_TIMEOUT,\n    LOCAL_BROWSER_INIT_TIMEOUT,\n    REMOTE_BROWSER_INIT_TIMEOUT,\n} from '../../utils/browser-connection-timeouts';\nimport MessageBus from '../../utils/message-bus';\nimport BrowserConnectionTracker from './tracker';\nimport TestRun from '../../test-run';\n// @ts-ignore\nimport { TestRun as LegacyTestRun } from 'testcafe-legacy-api';\nimport { Proxy } from 'testcafe-hammerhead';\nimport { NextTestRunInfo, OpenBrowserAdditionalOptions } from '../../shared/types';\nimport { EventType } from '../../proxyless/types';\n\nconst getBrowserConnectionDebugScope = (id: string): string => `testcafe:browser:connection:${id}`;\n\nconst IDLE_PAGE_TEMPLATE = read('../../client/browser/idle-page/index.html.mustache');\n\n\ninterface DisconnectionPromise<T> extends Promise<T> {\n    resolve: Function;\n    reject: Function;\n}\n\ninterface BrowserConnectionStatusResult {\n    cmd: string;\n    url: string;\n    testRunId: string | null;\n}\n\ninterface HeartbeatStatusResult {\n    code: HeartbeatStatus;\n    url: string;\n}\n\ninterface InitScript {\n    code: string | null;\n}\n\ninterface InitScriptTask extends InitScript {\n    resolve: Function;\n}\n\ninterface ProviderMetaInfoOptions {\n    appendToUserAgent?: boolean;\n}\n\nexport interface BrowserClosingInfo {\n    isRestarting?: boolean;\n}\n\nexport interface BrowserInfo {\n    alias: string;\n    browserName: string;\n    browserOption: unknown;\n    providerName: string;\n    provider: BrowserProvider;\n    userAgentProviderMetaInfo: string;\n    parsedUserAgent: ParsedUserAgent;\n}\n\nexport default class BrowserConnection extends EventEmitter {\n    public permanent: boolean;\n    public previousActiveWindowId: string | null;\n    private readonly disableMultipleWindows: boolean;\n    public readonly proxyless: boolean;\n    private readonly HEARTBEAT_TIMEOUT: number;\n    private readonly BROWSER_CLOSE_TIMEOUT: number;\n    private readonly BROWSER_RESTART_TIMEOUT: number;\n    public readonly id: string;\n    private _currentTestRun: TestRun | null = null;\n    private readonly jobQueue: BrowserJob[];\n    private readonly initScriptsQueue: InitScriptTask[];\n    public browserConnectionGateway: BrowserConnectionGateway;\n    private disconnectionPromise: DisconnectionPromise<void> | null;\n    private testRunAborted: boolean;\n    public status: BrowserConnectionStatus;\n    private heartbeatTimeout: NodeJS.Timeout | null;\n    private pendingTestRunInfo: NextTestRunInfo | null;\n    public url = '';\n    public idleUrl = '';\n    private forcedIdleUrl = '';\n    private initScriptUrl = '';\n    public heartbeatUrl = '';\n    public statusUrl = '';\n    public activeWindowIdUrl = '';\n    public closeWindowUrl = '';\n    public statusDoneUrl = '';\n    public heartbeatRelativeUrl = '';\n    public statusRelativeUrl = '';\n    public statusDoneRelativeUrl = '';\n    public idleRelativeUrl = '';\n    public openFileProtocolRelativeUrl = '';\n    public openFileProtocolUrl = '';\n    public dispatchProxylessEventRelativeUrl = '';\n    private readonly debugLogger: debug.Debugger;\n    private osInfo: OSInfo | null = null;\n\n    public readonly warningLog: WarningLog;\n    private _messageBus?: MessageBus;\n\n    public idle: boolean;\n\n    public browserInfo: BrowserInfo;\n    public provider: any;\n\n    public constructor (\n        gateway: BrowserConnectionGateway,\n        browserInfo: BrowserInfo,\n        permanent: boolean,\n        disableMultipleWindows = false,\n        proxyless = false,\n        messageBus?: MessageBus) {\n        super();\n\n        this.HEARTBEAT_TIMEOUT       = HEARTBEAT_TIMEOUT;\n        this.BROWSER_CLOSE_TIMEOUT   = BROWSER_CLOSE_TIMEOUT;\n        this.BROWSER_RESTART_TIMEOUT = BROWSER_RESTART_TIMEOUT;\n\n        this.id                       = BrowserConnection._generateId();\n        this.jobQueue                 = [];\n        this.initScriptsQueue         = [];\n        this.browserConnectionGateway = gateway;\n        this.disconnectionPromise     = null;\n        this.testRunAborted           = false;\n        this.warningLog               = new WarningLog(null, WarningLog.createAddWarningCallback(messageBus));\n        this.debugLogger              = debug(getBrowserConnectionDebugScope(this.id));\n\n        if (messageBus)\n            this.messageBus = messageBus;\n\n        this.browserInfo                           = browserInfo;\n        this.browserInfo.userAgentProviderMetaInfo = '';\n\n        this.provider = browserInfo.provider;\n\n        this.permanent              = permanent;\n        this.status                 = BrowserConnectionStatus.uninitialized;\n        this.idle                   = true;\n        this.heartbeatTimeout       = null;\n        this.pendingTestRunInfo     = null;\n        this.disableMultipleWindows = disableMultipleWindows;\n        this.proxyless              = proxyless;\n\n        this._buildCommunicationUrls(gateway.proxy);\n        this._setEventHandlers();\n\n        BrowserConnectionTracker.add(this);\n\n        this.previousActiveWindowId = null;\n\n        this.browserConnectionGateway.startServingConnection(this);\n\n        // NOTE: Give a caller time to assign event listeners\n        process.nextTick(() => this._runBrowser());\n    }\n\n    private _buildCommunicationUrls (proxy: Proxy): void {\n        this.url               = proxy.resolveRelativeServiceUrl(`${SERVICE_ROUTES.connect}/${this.id}`);\n        this.forcedIdleUrl     = proxy.resolveRelativeServiceUrl(`${SERVICE_ROUTES.idleForced}/${this.id}`);\n        this.initScriptUrl     = proxy.resolveRelativeServiceUrl(`${SERVICE_ROUTES.initScript}/${this.id}`);\n\n        this.heartbeatRelativeUrl              = `${SERVICE_ROUTES.heartbeat}/${this.id}`;\n        this.statusRelativeUrl                 = `${SERVICE_ROUTES.status}/${this.id}`;\n        this.statusDoneRelativeUrl             = `${SERVICE_ROUTES.statusDone}/${this.id}`;\n        this.idleRelativeUrl                   = `${SERVICE_ROUTES.idle}/${this.id}`;\n        this.activeWindowIdUrl                 = `${SERVICE_ROUTES.activeWindowId}/${this.id}`;\n        this.closeWindowUrl                    = `${SERVICE_ROUTES.closeWindow}/${this.id}`;\n        this.openFileProtocolRelativeUrl       = `${SERVICE_ROUTES.openFileProtocol}/${this.id}`;\n        this.dispatchProxylessEventRelativeUrl = `${SERVICE_ROUTES.dispatchProxylessEvent}/${this.id}`;\n\n        this.idleUrl             = proxy.resolveRelativeServiceUrl(this.idleRelativeUrl);\n        this.heartbeatUrl        = proxy.resolveRelativeServiceUrl(this.heartbeatRelativeUrl);\n        this.statusUrl           = proxy.resolveRelativeServiceUrl(this.statusRelativeUrl);\n        this.statusDoneUrl       = proxy.resolveRelativeServiceUrl(this.statusDoneRelativeUrl);\n        this.openFileProtocolUrl = proxy.resolveRelativeServiceUrl(this.openFileProtocolRelativeUrl);\n    }\n\n    public set messageBus (messageBus: MessageBus | undefined) {\n        this._messageBus         = messageBus;\n        this.warningLog.callback = WarningLog.createAddWarningCallback(this._messageBus);\n\n        if (messageBus) {\n            messageBus.on('test-run-start', testRun => {\n                if (testRun.browserConnection.id === this.id)\n                    this._currentTestRun = testRun;\n            });\n        }\n\n        this.emit('message-bus-initialized', messageBus);\n    }\n\n    public get messageBus (): MessageBus | undefined {\n        return this._messageBus;\n    }\n\n    private _setEventHandlers (): void {\n        this.on('error', e => {\n            this.debugLogger(e);\n            this._forceIdle();\n            this.close();\n        });\n\n        for (const name in BrowserConnectionStatus) {\n            const status = BrowserConnectionStatus[name as keyof typeof BrowserConnectionStatus];\n\n            this.on(status, () => {\n                this.debugLogger(`status changed to '${status}'`);\n            });\n        }\n    }\n\n    private static _generateId (): string {\n        return nanoid(7);\n    }\n\n    private _getAdditionalBrowserOptions (): OpenBrowserAdditionalOptions {\n        const options = {\n            disableMultipleWindows: this.disableMultipleWindows,\n        } as OpenBrowserAdditionalOptions;\n\n        if (this.proxyless) {\n            options.proxyless = {\n                serviceDomains: [\n                    this.browserConnectionGateway.proxy.server1Info.domain,\n                    this.browserConnectionGateway.proxy.server2Info.domain,\n                ],\n\n                developmentMode: this.browserConnectionGateway.proxy.options.developmentMode,\n            };\n        }\n\n        return options;\n    }\n\n    private async _runBrowser (): Promise<void> {\n        try {\n            const additionalOptions = this._getAdditionalBrowserOptions();\n\n            await this.provider.openBrowser(this.id, this.url, this.browserInfo.browserOption, additionalOptions);\n\n            if (this.status !== BrowserConnectionStatus.ready)\n                await promisifyEvent(this, 'ready');\n\n            this.status = BrowserConnectionStatus.opened;\n            this.emit('opened');\n        }\n        catch (err: any) {\n            this.emit('error', new GeneralError(\n                RUNTIME_ERRORS.unableToOpenBrowser,\n                this.browserInfo.providerName + ':' + this.browserInfo.browserName,\n                err.stack\n            ));\n        }\n    }\n\n    private async _closeBrowser (data: BrowserClosingInfo = {}): Promise<void> {\n        if (!this.idle)\n            await promisifyEvent(this, 'idle');\n\n        try {\n            await this.provider.closeBrowser(this.id, data);\n        }\n        catch (err) {\n            // NOTE: A warning would be really nice here, but it can't be done while log is stored in a task.\n            this.debugLogger(err);\n        }\n    }\n\n    private _forceIdle (): void {\n        if (!this.idle) {\n            this.idle = true;\n\n            this.emit('idle');\n        }\n    }\n\n    private _createBrowserDisconnectedError (): GeneralError {\n        return new GeneralError(RUNTIME_ERRORS.browserDisconnected, this.userAgent);\n    }\n\n    private _waitForHeartbeat (): void {\n        this.heartbeatTimeout = setTimeout(() => {\n            const err = this._createBrowserDisconnectedError();\n\n            this.status         = BrowserConnectionStatus.disconnected;\n            this.testRunAborted = true;\n\n            this.emit('disconnected', err);\n\n            this._restartBrowserOnDisconnect(err);\n        }, this.HEARTBEAT_TIMEOUT);\n    }\n\n    private async _getTestRunInfo (needPopNext: boolean): Promise<NextTestRunInfo> {\n        if (needPopNext || !this.pendingTestRunInfo)\n            this.pendingTestRunInfo = await this._popNextTestRunInfo();\n\n        return this.pendingTestRunInfo as NextTestRunInfo;\n    }\n\n    private async _popNextTestRunInfo (): Promise<NextTestRunInfo | null> {\n        while (this.hasQueuedJobs && !this.currentJob.hasQueuedTestRuns)\n            this.jobQueue.shift();\n\n        return this.hasQueuedJobs ? await this.currentJob.popNextTestRunInfo(this) : null;\n    }\n\n    public getCurrentTestRun (): LegacyTestRun | TestRun | null {\n        return this._currentTestRun;\n    }\n\n    public static getById (id: string): BrowserConnection | null {\n        return BrowserConnectionTracker.activeBrowserConnections[id] || null;\n    }\n\n    private async _restartBrowser (): Promise<void> {\n        this.status = BrowserConnectionStatus.uninitialized;\n\n        this._forceIdle();\n\n        let resolveTimeout: Function | null = null;\n        let isTimeoutExpired                = false;\n        let timeout: NodeJS.Timeout | null  = null;\n\n        const restartPromise = timeLimit(this._closeBrowser({ isRestarting: true }), this.BROWSER_CLOSE_TIMEOUT, { rejectWith: new TimeoutError() })\n            .catch(err => this.debugLogger(err))\n            .then(() => this._runBrowser());\n\n        const timeoutPromise = new Promise<void>(resolve => {\n            resolveTimeout = resolve;\n\n            timeout = setTimeout(() => {\n                isTimeoutExpired = true;\n\n                resolve();\n            }, this.BROWSER_RESTART_TIMEOUT);\n        });\n\n        return Promise.race([restartPromise, timeoutPromise])\n            .then(() => {\n                clearTimeout(timeout as NodeJS.Timeout);\n\n                if (isTimeoutExpired)\n                    this.emit('error', this._createBrowserDisconnectedError());\n                else\n                    (resolveTimeout as Function)();\n            });\n    }\n\n    private _restartBrowserOnDisconnect (err: Error): void {\n        let resolveFn: Function | null = null;\n        let rejectFn: Function | null  = null;\n\n        this.disconnectionPromise = new Promise((resolve, reject) => {\n            resolveFn = resolve;\n\n            rejectFn = () => {\n                reject(err);\n            };\n\n            setTimeout(() => {\n                (rejectFn as Function)();\n            });\n        })\n            .then(() => {\n                return this._restartBrowser();\n            })\n            .catch(e => {\n                this.emit('error', e);\n            }) as DisconnectionPromise<void>;\n\n        this.disconnectionPromise.resolve = resolveFn as unknown as Function;\n        this.disconnectionPromise.reject  = rejectFn as unknown as Function;\n    }\n\n    public async getDefaultBrowserInitTimeout (): Promise<number> {\n        const isLocalBrowser = await this.provider.isLocalBrowser(this.id, this.browserInfo.browserName);\n\n        return isLocalBrowser ? LOCAL_BROWSER_INIT_TIMEOUT : REMOTE_BROWSER_INIT_TIMEOUT;\n    }\n\n    public async processDisconnection (disconnectionThresholdExceeded: boolean): Promise<void> {\n        const { resolve, reject } = this.disconnectionPromise as DisconnectionPromise<void>;\n\n        if (disconnectionThresholdExceeded)\n            reject();\n        else\n            resolve();\n    }\n\n    public addWarning (message: string, ...args: any[]): void {\n        if (this.currentJob)\n            this.currentJob.warningLog.addWarning(message, ...args);\n        else\n            this.warningLog.addWarning(message, ...args);\n    }\n\n    private _appendToPrettyUserAgent (str: string): void {\n        this.browserInfo.parsedUserAgent.prettyUserAgent += ` (${str})`;\n    }\n\n    private _moveWarningLogToJob (job: BrowserJob): void {\n        job.warningLog.copyFrom(this.warningLog);\n        this.warningLog.clear();\n    }\n\n    public setProviderMetaInfo (str: string, options?: ProviderMetaInfoOptions): void {\n        const appendToUserAgent = options?.appendToUserAgent as boolean;\n\n        if (appendToUserAgent) {\n            // NOTE:\n            // change prettyUserAgent only when connection already was established\n            if (this.isReady())\n                this._appendToPrettyUserAgent(str);\n            else\n                this.on('ready', () => this._appendToPrettyUserAgent(str));\n\n            return;\n        }\n\n        this.browserInfo.userAgentProviderMetaInfo = str;\n    }\n\n    public get userAgent (): string {\n        let userAgent = this.browserInfo.parsedUserAgent.prettyUserAgent;\n\n        if (this.browserInfo.userAgentProviderMetaInfo)\n            userAgent += ` (${this.browserInfo.userAgentProviderMetaInfo})`;\n\n        return userAgent;\n    }\n\n    public get connectionInfo (): string {\n        if (!this.osInfo)\n            return this.userAgent;\n\n        const { name, version } = this.browserInfo.parsedUserAgent;\n        let connectionInfo      = calculatePrettyUserAgent({ name, version }, this.osInfo);\n        const metaInfo          = this.browserInfo.userAgentProviderMetaInfo || extractMetaInfo(this.browserInfo.parsedUserAgent.prettyUserAgent);\n\n        if (metaInfo)\n            connectionInfo += ` (${ metaInfo })`;\n\n        return connectionInfo;\n    }\n\n    public get retryTestPages (): boolean {\n        return this.browserConnectionGateway.retryTestPages;\n    }\n\n    public get hasQueuedJobs (): boolean {\n        return !!this.jobQueue.length;\n    }\n\n    public get currentJob (): BrowserJob {\n        return this.jobQueue[0];\n    }\n\n    // API\n    public runInitScript (code: string): Promise<string | unknown> {\n        return new Promise(resolve => this.initScriptsQueue.push({ code, resolve }));\n    }\n\n    public addJob (job: BrowserJob): void {\n        this.jobQueue.push(job);\n\n        this._moveWarningLogToJob(job);\n    }\n\n    public removeJob (job: BrowserJob): void {\n        remove(this.jobQueue, job);\n    }\n\n    public async close (): Promise<void> {\n        if (this.status === BrowserConnectionStatus.closing || this.status === BrowserConnectionStatus.closed)\n            return;\n\n        this.status = BrowserConnectionStatus.closing;\n        this.emit(BrowserConnectionStatus.closing);\n\n        await this._closeBrowser();\n\n        this.browserConnectionGateway.stopServingConnection(this);\n\n        if (this.heartbeatTimeout)\n            clearTimeout(this.heartbeatTimeout);\n\n        BrowserConnectionTracker.remove(this);\n\n        this.status = BrowserConnectionStatus.closed;\n        this.emit(BrowserConnectionStatus.closed);\n    }\n\n    public async establish (userAgent: string): Promise<void> {\n        this.status                      = BrowserConnectionStatus.ready;\n        this.browserInfo.parsedUserAgent = parseUserAgent(userAgent);\n        this.osInfo                      = await this.provider.getOSInfo(this.id);\n\n        this._waitForHeartbeat();\n        this.emit('ready');\n    }\n\n    public heartbeat (): HeartbeatStatusResult {\n        if (this.heartbeatTimeout)\n            clearTimeout(this.heartbeatTimeout);\n\n        this._waitForHeartbeat();\n\n        return {\n            code: this.status === BrowserConnectionStatus.closing ? HeartbeatStatus.closing : HeartbeatStatus.ok,\n            url:  this.status === BrowserConnectionStatus.closing ? this.idleUrl : '',\n        };\n    }\n\n    public renderIdlePage (): string {\n        return Mustache.render(IDLE_PAGE_TEMPLATE as string, {\n            userAgent:           this.connectionInfo,\n            statusUrl:           this.statusUrl,\n            heartbeatUrl:        this.heartbeatUrl,\n            initScriptUrl:       this.initScriptUrl,\n            openFileProtocolUrl: this.openFileProtocolUrl,\n            retryTestPages:      !!this.browserConnectionGateway.retryTestPages,\n            proxyless:           this.proxyless,\n        });\n    }\n\n    public getInitScript (): InitScript {\n        const initScriptPromise = this.initScriptsQueue[0];\n\n        return { code: initScriptPromise ? initScriptPromise.code : null };\n    }\n\n    public handleInitScriptResult (data: string): void {\n        const initScriptPromise = this.initScriptsQueue.shift();\n\n        if (initScriptPromise)\n            initScriptPromise.resolve(JSON.parse(data));\n    }\n\n    public isHeadlessBrowser (): boolean {\n        return this.provider.isHeadlessBrowser(this.id);\n    }\n\n    public async reportJobResult (status: string, data: any): Promise<any> {\n        await this.provider.reportJobResult(this.id, status, data);\n    }\n\n    public async getStatus (isTestDone: boolean): Promise<BrowserConnectionStatusResult> {\n        if (!this.idle && !isTestDone) {\n            this.idle = true;\n            this.emit('idle');\n        }\n\n        if (this.status === BrowserConnectionStatus.opened) {\n            const nextTestRunInfo = await this._getTestRunInfo(isTestDone || this.testRunAborted);\n\n            this.testRunAborted = false;\n\n            if (nextTestRunInfo) {\n                this.idle = false;\n\n                return {\n                    cmd:       COMMAND.run,\n                    testRunId: nextTestRunInfo.testRunId,\n                    url:       nextTestRunInfo.url,\n                };\n            }\n        }\n\n        return {\n            cmd:       COMMAND.idle,\n            url:       this.idleUrl,\n            testRunId: null,\n        };\n    }\n\n    public get activeWindowId (): null | string {\n        return this.provider.getActiveWindowId(this.id);\n    }\n\n    public set activeWindowId (val) {\n        this.previousActiveWindowId = this.activeWindowId;\n\n        this.provider.setActiveWindowId(this.id, val);\n    }\n\n    public async openFileProtocol (url: string): Promise<void> {\n        return this.provider.openFileProtocol(this.id, url);\n    }\n\n    public async dispatchProxylessEvent (type: EventType, options: any): Promise<void> {\n        return this.provider.dispatchProxylessEvent(this.id, type, options);\n    }\n\n    public async canUseDefaultWindowActions (): Promise<boolean> {\n        return this.provider.canUseDefaultWindowActions(this.id);\n    }\n\n    public isReady (): boolean {\n        return this.status === BrowserConnectionStatus.ready ||\n            this.status === BrowserConnectionStatus.opened ||\n            this.status === BrowserConnectionStatus.closing;\n    }\n}\n"]}