testcafe 1.14.1 → 1.14.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +6 -0
- package/LICENSE +21 -21
- package/bin/testcafe-with-v8-flag-filter.js +0 -0
- package/lib/api/exportable-lib/index.js +49 -49
- package/lib/api/request-hooks/assert-type.js +7 -7
- package/lib/api/request-hooks/hook-method-names.js +9 -9
- package/lib/api/request-hooks/hook.js +32 -32
- package/lib/api/request-hooks/interfaces.js +2 -2
- package/lib/api/request-hooks/request-logger.js +112 -112
- package/lib/api/request-hooks/request-mock/create-request-mock.js +10 -10
- package/lib/api/request-hooks/request-mock/index.js +44 -44
- package/lib/api/request-hooks/request-mock.js +46 -46
- package/lib/api/structure/base-unit.js +11 -11
- package/lib/api/structure/fixture.js +75 -75
- package/lib/api/structure/interfaces.js +2 -2
- package/lib/api/structure/test-file.js +31 -31
- package/lib/api/structure/test-timeout.js +9 -9
- package/lib/api/structure/test.js +85 -85
- package/lib/api/structure/testing-unit.js +89 -89
- package/lib/api/structure/unit-type.js +9 -9
- package/lib/api/test-controller/assertion.js +88 -88
- package/lib/api/test-controller/execution-context.js +82 -82
- package/lib/api/test-controller/index.js +358 -358
- package/lib/api/test-controller/proxy.js +28 -28
- package/lib/api/test-page-url.js +60 -60
- package/lib/api/test-run-tracker.js +68 -68
- package/lib/api/wrap-test-function.js +49 -49
- package/lib/assertions/executor.js +74 -74
- package/lib/assertions/get-fn.js +46 -46
- package/lib/assertions/type.js +20 -20
- package/lib/assets/content-types.js +9 -9
- package/lib/assets/injectables.js +18 -18
- package/lib/browser/connection/command.js +10 -10
- package/lib/browser/connection/error-hints.js +9 -9
- package/lib/browser/connection/gateway.js +159 -159
- package/lib/browser/connection/get-hints.js +33 -33
- package/lib/browser/connection/heartbeat-status.js +8 -8
- package/lib/browser/connection/index.js +328 -328
- package/lib/browser/connection/remotes-queue.js +46 -46
- package/lib/browser/connection/service-routes.js +12 -12
- package/lib/browser/connection/status.js +12 -12
- package/lib/browser/interfaces.js +2 -2
- package/lib/browser/provider/built-in/dedicated/base.js +80 -80
- package/lib/browser/provider/built-in/dedicated/chrome/browser-client.js +204 -204
- package/lib/browser/provider/built-in/dedicated/chrome/build-chrome-args.js +17 -17
- package/lib/browser/provider/built-in/dedicated/chrome/config.js +110 -110
- package/lib/browser/provider/built-in/dedicated/chrome/create-temp-profile.js +45 -45
- package/lib/browser/provider/built-in/dedicated/chrome/elapsed-upperbounds.js +15 -15
- package/lib/browser/provider/built-in/dedicated/chrome/index.js +102 -102
- package/lib/browser/provider/built-in/dedicated/chrome/interfaces.js +2 -2
- package/lib/browser/provider/built-in/dedicated/chrome/local-chrome.js +24 -24
- package/lib/browser/provider/built-in/dedicated/chrome/runtime-info.js +29 -29
- package/lib/browser/provider/built-in/dedicated/edge/index.js +10 -10
- package/lib/browser/provider/built-in/dedicated/edge/runtime-info.js +29 -29
- package/lib/browser/provider/built-in/dedicated/firefox/config.js +33 -33
- package/lib/browser/provider/built-in/dedicated/firefox/create-temp-profile.js +78 -78
- package/lib/browser/provider/built-in/dedicated/firefox/index.js +73 -73
- package/lib/browser/provider/built-in/dedicated/firefox/local-firefox.js +36 -36
- package/lib/browser/provider/built-in/dedicated/firefox/marionette-client/commands.js +13 -13
- package/lib/browser/provider/built-in/dedicated/firefox/marionette-client/index.js +200 -200
- package/lib/browser/provider/built-in/dedicated/firefox/runtime-info.js +17 -17
- package/lib/browser/provider/built-in/index.js +21 -21
- package/lib/browser/provider/built-in/locally-installed.js +30 -30
- package/lib/browser/provider/built-in/path.js +47 -47
- package/lib/browser/provider/built-in/remote.js +58 -58
- package/lib/browser/provider/index.js +303 -303
- package/lib/browser/provider/parse-provider-name.js +16 -16
- package/lib/browser/provider/plugin-host.js +121 -121
- package/lib/browser/provider/pool.js +115 -115
- package/lib/browser/provider/utils/argument-parsing.js +74 -74
- package/lib/browser/provider/utils/browser-starter.js +34 -34
- package/lib/browser/provider/utils/client-functions.js +24 -24
- package/lib/browser/provider/utils/get-maximized-headless-window-size.js +9 -9
- package/lib/cli/argument-parser.js +284 -284
- package/lib/cli/authentication-helper.js +35 -35
- package/lib/cli/cli.js +134 -134
- package/lib/cli/correct-browsers-and-sources.js +40 -40
- package/lib/cli/index.js +19 -19
- package/lib/cli/log.js +43 -43
- package/lib/cli/remotes-wizard.js +36 -36
- package/lib/cli/termination-handler.js +38 -38
- package/lib/client/automation/index.js +34 -16
- package/lib/client/automation/index.min.js +1 -1
- package/lib/client/browser/idle-page/index.html.mustache +35 -35
- package/lib/client/browser/idle-page/logo.svg +86 -86
- package/lib/client/core/index.js +14 -14
- package/lib/client/driver/index.js +46 -16
- package/lib/client/driver/index.min.js +1 -1
- package/lib/client/driver/internal-properties.js +9 -9
- package/lib/client/test-run/iframe.js.mustache +17 -17
- package/lib/client/test-run/index.js.mustache +51 -51
- package/lib/client/ui/index.js +14 -14
- package/lib/client/ui/sprite.svg +42 -42
- package/lib/client-functions/builder-symbol.js +4 -4
- package/lib/client-functions/client-function-builder.js +155 -155
- package/lib/client-functions/replicator.js +61 -61
- package/lib/client-functions/return-single-prop-mode.js +8 -8
- package/lib/client-functions/selector-api-execution-mode.js +20 -20
- package/lib/client-functions/selectors/add-api.js +645 -645
- package/lib/client-functions/selectors/create-snapshot-methods.js +13 -13
- package/lib/client-functions/selectors/prepare-api-args.js +20 -20
- package/lib/client-functions/selectors/selector-attribute-filter.js +22 -22
- package/lib/client-functions/selectors/selector-builder.js +153 -153
- package/lib/client-functions/selectors/selector-text-filter.js +43 -43
- package/lib/client-functions/selectors/snapshot-properties.js +48 -48
- package/lib/client-functions/types.js +18 -18
- package/lib/compiler/babel/format-babel-produced-code.js +10 -10
- package/lib/compiler/babel/get-base-babel-options.js +11 -11
- package/lib/compiler/babel/load-libs.js +63 -63
- package/lib/compiler/babel/preset-stage-2.js +19 -19
- package/lib/compiler/compile-client-function.js +73 -73
- package/lib/compiler/compilers.js +33 -33
- package/lib/compiler/index.js +92 -92
- package/lib/compiler/interfaces.js +2 -2
- package/lib/compiler/test-file/api-based.js +146 -146
- package/lib/compiler/test-file/base.js +36 -36
- package/lib/compiler/test-file/exportble-lib-path.js +5 -5
- package/lib/compiler/test-file/formats/coffeescript/compiler.js +38 -38
- package/lib/compiler/test-file/formats/coffeescript/get-test-list.js +29 -29
- package/lib/compiler/test-file/formats/es-next/compiler.js +42 -42
- package/lib/compiler/test-file/formats/es-next/get-test-list.js +166 -166
- package/lib/compiler/test-file/formats/es-next/is-flow-code.js +7 -7
- package/lib/compiler/test-file/formats/raw.js +85 -85
- package/lib/compiler/test-file/formats/typescript/compiler.js +135 -135
- package/lib/compiler/test-file/formats/typescript/get-test-list.js +185 -185
- package/lib/compiler/test-file/test-file-parser-base.js +214 -214
- package/lib/configuration/configuration-base.js +165 -165
- package/lib/configuration/constants.js +9 -9
- package/lib/configuration/customizable-compilers.js +7 -7
- package/lib/configuration/default-values.js +51 -51
- package/lib/configuration/interfaces.js +2 -2
- package/lib/configuration/option-names.js +53 -53
- package/lib/configuration/option-source.js +9 -9
- package/lib/configuration/option.js +14 -14
- package/lib/configuration/quarantine-option-names.js +8 -8
- package/lib/configuration/run-option-names.js +26 -26
- package/lib/configuration/screenshot-option-names.js +10 -10
- package/lib/configuration/testcafe-configuration.js +163 -163
- package/lib/configuration/types.js +2 -2
- package/lib/configuration/typescript-configuration.js +62 -62
- package/lib/custom-client-scripts/assert-type.js +7 -7
- package/lib/custom-client-scripts/client-script-init.js +2 -2
- package/lib/custom-client-scripts/client-script.js +106 -106
- package/lib/custom-client-scripts/get-code.js +11 -11
- package/lib/custom-client-scripts/get-url.js +6 -6
- package/lib/custom-client-scripts/load.js +15 -15
- package/lib/custom-client-scripts/problematic-scripts.js +2 -2
- package/lib/custom-client-scripts/routing.js +36 -36
- package/lib/custom-client-scripts/utils.js +60 -60
- package/lib/embedding-utils.js +83 -83
- package/lib/errors/create-stack-filter.js +18 -18
- package/lib/errors/error-list.js +26 -26
- package/lib/errors/get-callsite.js +31 -31
- package/lib/errors/internal-modules-prefix.js +8 -8
- package/lib/errors/is-internal-stack-frame.js +45 -45
- package/lib/errors/process-test-fn-error.js +37 -37
- package/lib/errors/runtime/index.js +123 -123
- package/lib/errors/runtime/templates.js +115 -115
- package/lib/errors/runtime/type-assertions.js +112 -112
- package/lib/errors/stack-cleaning-hook.js +64 -64
- package/lib/errors/test-run/formattable-adapter.js +59 -59
- package/lib/errors/test-run/index.js +301 -301
- package/lib/errors/test-run/render-error-template.js +31 -31
- package/lib/errors/test-run/templates.js +91 -91
- package/lib/errors/test-run/utils.js +89 -89
- package/lib/errors/types.js +156 -156
- package/lib/index.js +81 -81
- package/lib/live/bootstrapper.js +43 -43
- package/lib/live/controller.js +107 -107
- package/lib/live/file-watcher/index.js +67 -67
- package/lib/live/file-watcher/modules-graph.js +58 -58
- package/lib/live/keyboard-observer.js +76 -76
- package/lib/live/logger/index.js +64 -64
- package/lib/live/test-run-controller.js +96 -96
- package/lib/live/test-run-state.js +6 -6
- package/lib/live/test-run.js +56 -56
- package/lib/live/test-runner.js +167 -167
- package/lib/load-assets.js +29 -29
- package/lib/notifications/add-rendered-warning.js +16 -16
- package/lib/notifications/debug-logger.js +78 -78
- package/lib/notifications/deprecated.js +24 -24
- package/lib/notifications/information-message.js +9 -9
- package/lib/notifications/warning-log.js +31 -31
- package/lib/notifications/warning-message.js +47 -47
- package/lib/reporter/command/command-formatter.js +120 -109
- package/lib/reporter/command/format-command.js +8 -8
- package/lib/reporter/command/interfaces.js +2 -2
- package/lib/reporter/index.js +319 -314
- package/lib/reporter/interfaces.js +2 -2
- package/lib/reporter/plugin-host.js +135 -135
- package/lib/reporter/plugin-methods.js +12 -12
- package/lib/role/index.js +74 -74
- package/lib/role/marker-symbol.js +7 -7
- package/lib/role/phase.js +9 -9
- package/lib/runner/bootstrapper.js +271 -271
- package/lib/runner/browser-job-result.js +9 -9
- package/lib/runner/browser-job.js +152 -152
- package/lib/runner/browser-set.js +114 -114
- package/lib/runner/fixture-hook-controller.js +85 -85
- package/lib/runner/index.js +449 -449
- package/lib/runner/interfaces.js +2 -2
- package/lib/runner/reporter-stream-controller.js +27 -27
- package/lib/runner/task/index.js +151 -151
- package/lib/runner/task/phase.js +9 -9
- package/lib/runner/test-run-controller.js +165 -165
- package/lib/runner/tested-app.js +72 -72
- package/lib/screenshots/capturer.js +141 -141
- package/lib/screenshots/constants.js +11 -11
- package/lib/screenshots/crop.js +111 -111
- package/lib/screenshots/default-extension.js +4 -4
- package/lib/screenshots/index.js +67 -67
- package/lib/screenshots/utils.js +39 -39
- package/lib/services/compiler/host.js +190 -190
- package/lib/services/compiler/io.js +9 -9
- package/lib/services/compiler/protocol.js +17 -16
- package/lib/services/compiler/service.js +225 -212
- package/lib/services/compiler/test-run-proxy.js +111 -111
- package/lib/services/interfaces.js +2 -2
- package/lib/services/process-title.js +8 -8
- package/lib/services/serialization/prepare-options.js +17 -17
- package/lib/services/serialization/replicator/create-replicator.js +25 -25
- package/lib/services/serialization/replicator/custom-error-transform.js +26 -26
- package/lib/services/serialization/test-structure.js +92 -92
- package/lib/services/utils/ipc/interfaces.js +30 -30
- package/lib/services/utils/ipc/io.js +108 -108
- package/lib/services/utils/ipc/message.js +75 -75
- package/lib/services/utils/ipc/packet.js +55 -55
- package/lib/services/utils/ipc/proxy.js +109 -109
- package/lib/services/utils/ipc/transport.js +64 -64
- package/lib/shared/errors/index.js +382 -382
- package/lib/shared/node-modules-folder-name.js +4 -4
- package/lib/test-run/bookmark.js +90 -90
- package/lib/test-run/browser-console-messages.js +73 -73
- package/lib/test-run/browser-manipulation-queue.js +92 -92
- package/lib/test-run/client-messages.js +9 -9
- package/lib/test-run/commands/actions.js +486 -483
- package/lib/test-run/commands/assertion.js +45 -45
- package/lib/test-run/commands/base.js +14 -14
- package/lib/test-run/commands/browser-manipulation.js +95 -95
- package/lib/test-run/commands/from-object.js +82 -82
- package/lib/test-run/commands/observation.js +61 -61
- package/lib/test-run/commands/options.js +231 -215
- package/lib/test-run/commands/service.js +54 -48
- package/lib/test-run/commands/type.js +65 -64
- package/lib/test-run/commands/utils.js +87 -87
- package/lib/test-run/commands/validations/argument.js +90 -90
- package/lib/test-run/commands/validations/factories.js +47 -47
- package/lib/test-run/commands/validations/initializers.js +44 -44
- package/lib/test-run/debug-log.js +32 -32
- package/lib/test-run/execute-js-expression.js +74 -74
- package/lib/test-run/index.js +823 -799
- package/lib/test-run/marker-symbol.js +7 -7
- package/lib/test-run/observed-callsites-storage.js +17 -17
- package/lib/test-run/phase.js +16 -16
- package/lib/test-run/session-controller.js +104 -104
- package/lib/testcafe.js +118 -118
- package/lib/utils/assignable.js +39 -39
- package/lib/utils/async-event-emitter.js +28 -28
- package/lib/utils/async-queue.js +14 -14
- package/lib/utils/browser-connection-timeouts.js +19 -19
- package/lib/utils/callsite.js +17 -17
- package/lib/utils/check-file-path.js +31 -31
- package/lib/utils/check-url.js +51 -51
- package/lib/utils/convert-to-best-fit-type.js +16 -16
- package/lib/utils/correct-file-path.js +21 -21
- package/lib/utils/define-lazy-property.js +13 -13
- package/lib/utils/delay.js +6 -6
- package/lib/utils/delegated-api.js +44 -44
- package/lib/utils/detect-display.js +6 -6
- package/lib/utils/detect-ffmpeg.js +44 -44
- package/lib/utils/diff/colors.js +29 -29
- package/lib/utils/diff/index.js +52 -52
- package/lib/utils/diff/util.js +23 -23
- package/lib/utils/diff.js +29 -29
- package/lib/utils/escape-user-agent.js +10 -10
- package/lib/utils/flag-list.js +17 -17
- package/lib/utils/get-any-key.js +8 -8
- package/lib/utils/get-browser.js +8 -8
- package/lib/utils/get-common-path.js +34 -34
- package/lib/utils/get-filter-fn.js +40 -40
- package/lib/utils/get-options/base.js +36 -36
- package/lib/utils/get-options/compiler.js +33 -33
- package/lib/utils/get-options/grep.js +15 -15
- package/lib/utils/get-options/index.js +20 -20
- package/lib/utils/get-options/meta.js +22 -22
- package/lib/utils/get-options/quarantine.js +91 -91
- package/lib/utils/get-options/screenshot.js +17 -17
- package/lib/utils/get-options/ssl.js +45 -45
- package/lib/utils/get-options/video.js +10 -10
- package/lib/utils/get-viewport-width.js +17 -17
- package/lib/utils/guard-time-execution.js +10 -10
- package/lib/utils/handle-errors.js +74 -74
- package/lib/utils/handle-tag-args.js +8 -8
- package/lib/utils/http.js +30 -30
- package/lib/utils/is-localhost.js +11 -11
- package/lib/utils/is-password-input.js +11 -0
- package/lib/utils/is-repl.js +10 -10
- package/lib/utils/is-window-in-iframe.js +6 -6
- package/lib/utils/limit-number.js +10 -10
- package/lib/utils/make-reg-exp.js +7 -7
- package/lib/utils/moment-loader.js +20 -20
- package/lib/utils/parse-file-list.js +71 -71
- package/lib/utils/parse-user-agent.js +55 -55
- package/lib/utils/path-pattern.js +114 -114
- package/lib/utils/prepare-reporters.js +30 -30
- package/lib/utils/prerender-callsite.js +18 -18
- package/lib/utils/process.js +119 -119
- package/lib/utils/promisified-functions.js +46 -46
- package/lib/utils/re-executable-promise.js +39 -39
- package/lib/utils/render-callsite-sync.js +29 -29
- package/lib/utils/render-template.js +9 -9
- package/lib/utils/reporter.js +30 -30
- package/lib/utils/resolve-path-relatively-cwd.js +7 -7
- package/lib/utils/string.js +105 -105
- package/lib/utils/temp-directory/cleanup-process/commands.js +7 -7
- package/lib/utils/temp-directory/cleanup-process/index.js +143 -143
- package/lib/utils/temp-directory/cleanup-process/worker.js +58 -58
- package/lib/utils/temp-directory/index.js +87 -87
- package/lib/utils/temp-directory/lockfile.js +56 -56
- package/lib/utils/thennable.js +7 -7
- package/lib/utils/timer.js +15 -15
- package/lib/utils/to-posix-path.js +8 -8
- package/lib/utils/types.js +2 -2
- package/lib/video-recorder/interfaces.js +2 -2
- package/lib/video-recorder/process.js +126 -126
- package/lib/video-recorder/recorder.js +136 -136
- package/lib/video-recorder/test-run-video-recorder.js +69 -69
- package/lib/video-recorder/videos.js +37 -37
- package/package.json +2 -2
- package/ts-defs/index.d.ts +25 -14
- package/ts-defs/selectors.d.ts +16 -5
- package/ts-defs/testcafe-scripts.d.ts +17 -6
|
@@ -1,330 +1,330 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
-
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
-
};
|
|
5
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
-
const debug_1 = __importDefault(require("debug"));
|
|
7
|
-
const time_limit_promise_1 = __importDefault(require("time-limit-promise"));
|
|
8
|
-
const events_1 = require("events");
|
|
9
|
-
const mustache_1 = __importDefault(require("mustache"));
|
|
10
|
-
const lodash_1 = require("lodash");
|
|
11
|
-
const parse_user_agent_1 = __importDefault(require("../../utils/parse-user-agent"));
|
|
12
|
-
const read_file_relative_1 = require("read-file-relative");
|
|
13
|
-
const promisify_event_1 = __importDefault(require("promisify-event"));
|
|
14
|
-
const nanoid_1 = __importDefault(require("nanoid"));
|
|
15
|
-
const command_1 = __importDefault(require("./command"));
|
|
16
|
-
const status_1 = __importDefault(require("./status"));
|
|
17
|
-
const heartbeat_status_1 = __importDefault(require("./heartbeat-status"));
|
|
18
|
-
const runtime_1 = require("../../errors/runtime");
|
|
19
|
-
const types_1 = require("../../errors/types");
|
|
20
|
-
const warning_log_1 = __importDefault(require("../../notifications/warning-log"));
|
|
21
|
-
const browser_connection_timeouts_1 = require("../../utils/browser-connection-timeouts");
|
|
22
|
-
const getBrowserConnectionDebugScope = (id) => `testcafe:browser:connection:${id}`;
|
|
23
|
-
const IDLE_PAGE_TEMPLATE = read_file_relative_1.readSync('../../client/browser/idle-page/index.html.mustache');
|
|
24
|
-
const connections = {};
|
|
25
|
-
class BrowserConnection extends events_1.EventEmitter {
|
|
26
|
-
constructor(gateway, browserInfo, permanent, disableMultipleWindows = false) {
|
|
27
|
-
super();
|
|
28
|
-
this.HEARTBEAT_TIMEOUT = browser_connection_timeouts_1.HEARTBEAT_TIMEOUT;
|
|
29
|
-
this.BROWSER_CLOSE_TIMEOUT = browser_connection_timeouts_1.BROWSER_CLOSE_TIMEOUT;
|
|
30
|
-
this.BROWSER_RESTART_TIMEOUT = browser_connection_timeouts_1.BROWSER_RESTART_TIMEOUT;
|
|
31
|
-
this.id = BrowserConnection._generateId();
|
|
32
|
-
this.jobQueue = [];
|
|
33
|
-
this.initScriptsQueue = [];
|
|
34
|
-
this.browserConnectionGateway = gateway;
|
|
35
|
-
this.disconnectionPromise = null;
|
|
36
|
-
this.testRunAborted = false;
|
|
37
|
-
this.warningLog = new warning_log_1.default();
|
|
38
|
-
this.debugLogger = debug_1.default(getBrowserConnectionDebugScope(this.id));
|
|
39
|
-
this.browserInfo = browserInfo;
|
|
40
|
-
this.browserInfo.userAgentProviderMetaInfo = '';
|
|
41
|
-
this.provider = browserInfo.provider;
|
|
42
|
-
this.permanent = permanent;
|
|
43
|
-
this.status = status_1.default.uninitialized;
|
|
44
|
-
this.idle = true;
|
|
45
|
-
this.heartbeatTimeout = null;
|
|
46
|
-
this.pendingTestRunUrl = null;
|
|
47
|
-
this.disableMultipleWindows = disableMultipleWindows;
|
|
48
|
-
this.url = `${gateway.domain}/browser/connect/${this.id}`;
|
|
49
|
-
this.idleUrl = `${gateway.domain}/browser/idle/${this.id}`;
|
|
50
|
-
this.forcedIdleUrl = `${gateway.domain}/browser/idle-forced/${this.id}`;
|
|
51
|
-
this.initScriptUrl = `${gateway.domain}/browser/init-script/${this.id}`;
|
|
52
|
-
this.heartbeatRelativeUrl = `/browser/heartbeat/${this.id}`;
|
|
53
|
-
this.statusRelativeUrl = `/browser/status/${this.id}`;
|
|
54
|
-
this.statusDoneRelativeUrl = `/browser/status-done/${this.id}`;
|
|
55
|
-
this.activeWindowIdUrl = `/browser/active-window-id/${this.id}`;
|
|
56
|
-
this.heartbeatUrl = `${gateway.domain}${this.heartbeatRelativeUrl}`;
|
|
57
|
-
this.statusUrl = `${gateway.domain}${this.statusRelativeUrl}`;
|
|
58
|
-
this.statusDoneUrl = `${gateway.domain}${this.statusDoneRelativeUrl}`;
|
|
59
|
-
this._setEventHandlers();
|
|
60
|
-
connections[this.id] = this;
|
|
61
|
-
this.previousActiveWindowId = null;
|
|
62
|
-
this.browserConnectionGateway.startServingConnection(this);
|
|
63
|
-
// NOTE: Give a caller time to assign event listeners
|
|
64
|
-
process.nextTick(() => this._runBrowser());
|
|
65
|
-
}
|
|
66
|
-
_setEventHandlers() {
|
|
67
|
-
this.on('error', e => {
|
|
68
|
-
this.debugLogger(e);
|
|
69
|
-
this._forceIdle();
|
|
70
|
-
this.close();
|
|
71
|
-
});
|
|
72
|
-
for (const name in status_1.default) {
|
|
73
|
-
const status = status_1.default[name];
|
|
74
|
-
this.on(status, () => {
|
|
75
|
-
this.debugLogger(`status changed to '${status}'`);
|
|
76
|
-
});
|
|
77
|
-
}
|
|
78
|
-
}
|
|
79
|
-
static _generateId() {
|
|
80
|
-
return nanoid_1.default(7);
|
|
81
|
-
}
|
|
82
|
-
async _runBrowser() {
|
|
83
|
-
try {
|
|
84
|
-
await this.provider.openBrowser(this.id, this.url, this.browserInfo.browserName, this.disableMultipleWindows);
|
|
85
|
-
if (this.status !== status_1.default.ready)
|
|
86
|
-
await promisify_event_1.default(this, 'ready');
|
|
87
|
-
this.status = status_1.default.opened;
|
|
88
|
-
this.emit('opened');
|
|
89
|
-
}
|
|
90
|
-
catch (err) {
|
|
91
|
-
this.emit('error', new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.unableToOpenBrowser, this.browserInfo.providerName + ':' + this.browserInfo.browserName, err.stack));
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
async _closeBrowser() {
|
|
95
|
-
if (!this.idle)
|
|
96
|
-
await promisify_event_1.default(this, 'idle');
|
|
97
|
-
try {
|
|
98
|
-
await this.provider.closeBrowser(this.id);
|
|
99
|
-
}
|
|
100
|
-
catch (err) {
|
|
101
|
-
// NOTE: A warning would be really nice here, but it can't be done while log is stored in a task.
|
|
102
|
-
this.debugLogger(err);
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
_forceIdle() {
|
|
106
|
-
if (!this.idle) {
|
|
107
|
-
this.idle = true;
|
|
108
|
-
this.emit('idle');
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
_createBrowserDisconnectedError() {
|
|
112
|
-
return new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.browserDisconnected, this.userAgent);
|
|
113
|
-
}
|
|
114
|
-
_waitForHeartbeat() {
|
|
115
|
-
this.heartbeatTimeout = setTimeout(() => {
|
|
116
|
-
const err = this._createBrowserDisconnectedError();
|
|
117
|
-
this.status = status_1.default.disconnected;
|
|
118
|
-
this.testRunAborted = true;
|
|
119
|
-
this.emit('disconnected', err);
|
|
120
|
-
this._restartBrowserOnDisconnect(err);
|
|
121
|
-
}, this.HEARTBEAT_TIMEOUT);
|
|
122
|
-
}
|
|
123
|
-
async _getTestRunUrl(needPopNext) {
|
|
124
|
-
if (needPopNext || !this.pendingTestRunUrl)
|
|
125
|
-
this.pendingTestRunUrl = await this._popNextTestRunUrl();
|
|
126
|
-
return this.pendingTestRunUrl;
|
|
127
|
-
}
|
|
128
|
-
async _popNextTestRunUrl() {
|
|
129
|
-
while (this.hasQueuedJobs && !this.currentJob.hasQueuedTestRuns)
|
|
130
|
-
this.jobQueue.shift();
|
|
131
|
-
return this.hasQueuedJobs ? await this.currentJob.popNextTestRunUrl(this) : null;
|
|
132
|
-
}
|
|
133
|
-
static getById(id) {
|
|
134
|
-
return connections[id] || null;
|
|
135
|
-
}
|
|
136
|
-
async _restartBrowser() {
|
|
137
|
-
this.status = status_1.default.uninitialized;
|
|
138
|
-
this._forceIdle();
|
|
139
|
-
let resolveTimeout = null;
|
|
140
|
-
let isTimeoutExpired = false;
|
|
141
|
-
let timeout = null;
|
|
142
|
-
const restartPromise = time_limit_promise_1.default(this._closeBrowser(), this.BROWSER_CLOSE_TIMEOUT, { rejectWith: new runtime_1.TimeoutError() })
|
|
143
|
-
.catch(err => this.debugLogger(err))
|
|
144
|
-
.then(() => this._runBrowser());
|
|
145
|
-
const timeoutPromise = new Promise(resolve => {
|
|
146
|
-
resolveTimeout = resolve;
|
|
147
|
-
timeout = setTimeout(() => {
|
|
148
|
-
isTimeoutExpired = true;
|
|
149
|
-
resolve();
|
|
150
|
-
}, this.BROWSER_RESTART_TIMEOUT);
|
|
151
|
-
});
|
|
152
|
-
return Promise.race([restartPromise, timeoutPromise])
|
|
153
|
-
.then(() => {
|
|
154
|
-
clearTimeout(timeout);
|
|
155
|
-
if (isTimeoutExpired)
|
|
156
|
-
this.emit('error', this._createBrowserDisconnectedError());
|
|
157
|
-
else
|
|
158
|
-
resolveTimeout();
|
|
159
|
-
});
|
|
160
|
-
}
|
|
161
|
-
_restartBrowserOnDisconnect(err) {
|
|
162
|
-
let resolveFn = null;
|
|
163
|
-
let rejectFn = null;
|
|
164
|
-
this.disconnectionPromise = new Promise((resolve, reject) => {
|
|
165
|
-
resolveFn = resolve;
|
|
166
|
-
rejectFn = () => {
|
|
167
|
-
reject(err);
|
|
168
|
-
};
|
|
169
|
-
setTimeout(() => {
|
|
170
|
-
rejectFn();
|
|
171
|
-
});
|
|
172
|
-
})
|
|
173
|
-
.then(() => {
|
|
174
|
-
return this._restartBrowser();
|
|
175
|
-
})
|
|
176
|
-
.catch(e => {
|
|
177
|
-
this.emit('error', e);
|
|
178
|
-
});
|
|
179
|
-
this.disconnectionPromise.resolve = resolveFn;
|
|
180
|
-
this.disconnectionPromise.reject = rejectFn;
|
|
181
|
-
}
|
|
182
|
-
async getDefaultBrowserInitTimeout() {
|
|
183
|
-
const isLocalBrowser = await this.provider.isLocalBrowser(this.id, this.browserInfo.browserName);
|
|
184
|
-
return isLocalBrowser ? browser_connection_timeouts_1.LOCAL_BROWSER_INIT_TIMEOUT : browser_connection_timeouts_1.REMOTE_BROWSER_INIT_TIMEOUT;
|
|
185
|
-
}
|
|
186
|
-
async processDisconnection(disconnectionThresholdExceeded) {
|
|
187
|
-
const { resolve, reject } = this.disconnectionPromise;
|
|
188
|
-
if (disconnectionThresholdExceeded)
|
|
189
|
-
reject();
|
|
190
|
-
else
|
|
191
|
-
resolve();
|
|
192
|
-
}
|
|
193
|
-
addWarning(...args) {
|
|
194
|
-
if (this.currentJob)
|
|
195
|
-
this.currentJob.warningLog.addWarning(...args);
|
|
196
|
-
else
|
|
197
|
-
this.warningLog.addWarning(...args);
|
|
198
|
-
}
|
|
199
|
-
_appendToPrettyUserAgent(str) {
|
|
200
|
-
this.browserInfo.parsedUserAgent.prettyUserAgent += ` (${str})`;
|
|
201
|
-
}
|
|
202
|
-
_moveWarningLogToJob(job) {
|
|
203
|
-
this.warningLog.copyTo(job.warningLog);
|
|
204
|
-
this.warningLog.clear();
|
|
205
|
-
}
|
|
206
|
-
setProviderMetaInfo(str, options) {
|
|
207
|
-
const appendToUserAgent = options === null || options === void 0 ? void 0 : options.appendToUserAgent;
|
|
208
|
-
if (appendToUserAgent) {
|
|
209
|
-
// NOTE:
|
|
210
|
-
// change prettyUserAgent only when connection already was established
|
|
211
|
-
if (this.isReady())
|
|
212
|
-
this._appendToPrettyUserAgent(str);
|
|
213
|
-
else
|
|
214
|
-
this.on('ready', () => this._appendToPrettyUserAgent(str));
|
|
215
|
-
return;
|
|
216
|
-
}
|
|
217
|
-
this.browserInfo.userAgentProviderMetaInfo = str;
|
|
218
|
-
}
|
|
219
|
-
get userAgent() {
|
|
220
|
-
let userAgent = this.browserInfo.parsedUserAgent.prettyUserAgent;
|
|
221
|
-
if (this.browserInfo.userAgentProviderMetaInfo)
|
|
222
|
-
userAgent += ` (${this.browserInfo.userAgentProviderMetaInfo})`;
|
|
223
|
-
return userAgent;
|
|
224
|
-
}
|
|
225
|
-
get retryTestPages() {
|
|
226
|
-
return this.browserConnectionGateway.retryTestPages;
|
|
227
|
-
}
|
|
228
|
-
get hasQueuedJobs() {
|
|
229
|
-
return !!this.jobQueue.length;
|
|
230
|
-
}
|
|
231
|
-
get currentJob() {
|
|
232
|
-
return this.jobQueue[0];
|
|
233
|
-
}
|
|
234
|
-
// API
|
|
235
|
-
runInitScript(code) {
|
|
236
|
-
return new Promise(resolve => this.initScriptsQueue.push({ code, resolve }));
|
|
237
|
-
}
|
|
238
|
-
addJob(job) {
|
|
239
|
-
this.jobQueue.push(job);
|
|
240
|
-
this._moveWarningLogToJob(job);
|
|
241
|
-
}
|
|
242
|
-
removeJob(job) {
|
|
243
|
-
lodash_1.pull(this.jobQueue, job);
|
|
244
|
-
}
|
|
245
|
-
async close() {
|
|
246
|
-
if (this.status === status_1.default.closing || this.status === status_1.default.closed)
|
|
247
|
-
return;
|
|
248
|
-
this.status = status_1.default.closing;
|
|
249
|
-
this.emit(status_1.default.closing);
|
|
250
|
-
await this._closeBrowser();
|
|
251
|
-
this.browserConnectionGateway.stopServingConnection(this);
|
|
252
|
-
if (this.heartbeatTimeout)
|
|
253
|
-
clearTimeout(this.heartbeatTimeout);
|
|
254
|
-
delete connections[this.id];
|
|
255
|
-
this.status = status_1.default.closed;
|
|
256
|
-
this.emit(status_1.default.closed);
|
|
257
|
-
}
|
|
258
|
-
establish(userAgent) {
|
|
259
|
-
this.status = status_1.default.ready;
|
|
260
|
-
this.browserInfo.parsedUserAgent = parse_user_agent_1.default(userAgent);
|
|
261
|
-
this._waitForHeartbeat();
|
|
262
|
-
this.emit('ready');
|
|
263
|
-
}
|
|
264
|
-
heartbeat() {
|
|
265
|
-
if (this.heartbeatTimeout)
|
|
266
|
-
clearTimeout(this.heartbeatTimeout);
|
|
267
|
-
this._waitForHeartbeat();
|
|
268
|
-
return {
|
|
269
|
-
code: this.status === status_1.default.closing ? heartbeat_status_1.default.closing : heartbeat_status_1.default.ok,
|
|
270
|
-
url: this.status === status_1.default.closing ? this.idleUrl : ''
|
|
271
|
-
};
|
|
272
|
-
}
|
|
273
|
-
renderIdlePage() {
|
|
274
|
-
return mustache_1.default.render(IDLE_PAGE_TEMPLATE, {
|
|
275
|
-
userAgent: this.userAgent,
|
|
276
|
-
statusUrl: this.statusUrl,
|
|
277
|
-
heartbeatUrl: this.heartbeatUrl,
|
|
278
|
-
initScriptUrl: this.initScriptUrl,
|
|
279
|
-
retryTestPages: !!this.browserConnectionGateway.retryTestPages
|
|
280
|
-
});
|
|
281
|
-
}
|
|
282
|
-
getInitScript() {
|
|
283
|
-
const initScriptPromise = this.initScriptsQueue[0];
|
|
284
|
-
return { code: initScriptPromise ? initScriptPromise.code : null };
|
|
285
|
-
}
|
|
286
|
-
handleInitScriptResult(data) {
|
|
287
|
-
const initScriptPromise = this.initScriptsQueue.shift();
|
|
288
|
-
if (initScriptPromise)
|
|
289
|
-
initScriptPromise.resolve(JSON.parse(data));
|
|
290
|
-
}
|
|
291
|
-
isHeadlessBrowser() {
|
|
292
|
-
return this.provider.isHeadlessBrowser(this.id);
|
|
293
|
-
}
|
|
294
|
-
async reportJobResult(status, data) {
|
|
295
|
-
await this.provider.reportJobResult(this.id, status, data);
|
|
296
|
-
}
|
|
297
|
-
async getStatus(isTestDone) {
|
|
298
|
-
if (!this.idle && !isTestDone) {
|
|
299
|
-
this.idle = true;
|
|
300
|
-
this.emit('idle');
|
|
301
|
-
}
|
|
302
|
-
if (this.status === status_1.default.opened) {
|
|
303
|
-
const testRunUrl = await this._getTestRunUrl(isTestDone || this.testRunAborted);
|
|
304
|
-
this.testRunAborted = false;
|
|
305
|
-
if (testRunUrl) {
|
|
306
|
-
this.idle = false;
|
|
307
|
-
return { cmd: command_1.default.run, url: testRunUrl };
|
|
308
|
-
}
|
|
309
|
-
}
|
|
310
|
-
return { cmd: command_1.default.idle, url: this.idleUrl };
|
|
311
|
-
}
|
|
312
|
-
get activeWindowId() {
|
|
313
|
-
return this.provider.getActiveWindowId(this.id);
|
|
314
|
-
}
|
|
315
|
-
set activeWindowId(val) {
|
|
316
|
-
this.previousActiveWindowId = this.activeWindowId;
|
|
317
|
-
this.provider.setActiveWindowId(this.id, val);
|
|
318
|
-
}
|
|
319
|
-
async canUseDefaultWindowActions() {
|
|
320
|
-
return this.provider.canUseDefaultWindowActions(this.id);
|
|
321
|
-
}
|
|
322
|
-
isReady() {
|
|
323
|
-
return this.status === status_1.default.ready ||
|
|
324
|
-
this.status === status_1.default.opened ||
|
|
325
|
-
this.status === status_1.default.closing;
|
|
326
|
-
}
|
|
327
|
-
}
|
|
328
|
-
exports.default = BrowserConnection;
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
const debug_1 = __importDefault(require("debug"));
|
|
7
|
+
const time_limit_promise_1 = __importDefault(require("time-limit-promise"));
|
|
8
|
+
const events_1 = require("events");
|
|
9
|
+
const mustache_1 = __importDefault(require("mustache"));
|
|
10
|
+
const lodash_1 = require("lodash");
|
|
11
|
+
const parse_user_agent_1 = __importDefault(require("../../utils/parse-user-agent"));
|
|
12
|
+
const read_file_relative_1 = require("read-file-relative");
|
|
13
|
+
const promisify_event_1 = __importDefault(require("promisify-event"));
|
|
14
|
+
const nanoid_1 = __importDefault(require("nanoid"));
|
|
15
|
+
const command_1 = __importDefault(require("./command"));
|
|
16
|
+
const status_1 = __importDefault(require("./status"));
|
|
17
|
+
const heartbeat_status_1 = __importDefault(require("./heartbeat-status"));
|
|
18
|
+
const runtime_1 = require("../../errors/runtime");
|
|
19
|
+
const types_1 = require("../../errors/types");
|
|
20
|
+
const warning_log_1 = __importDefault(require("../../notifications/warning-log"));
|
|
21
|
+
const browser_connection_timeouts_1 = require("../../utils/browser-connection-timeouts");
|
|
22
|
+
const getBrowserConnectionDebugScope = (id) => `testcafe:browser:connection:${id}`;
|
|
23
|
+
const IDLE_PAGE_TEMPLATE = read_file_relative_1.readSync('../../client/browser/idle-page/index.html.mustache');
|
|
24
|
+
const connections = {};
|
|
25
|
+
class BrowserConnection extends events_1.EventEmitter {
|
|
26
|
+
constructor(gateway, browserInfo, permanent, disableMultipleWindows = false) {
|
|
27
|
+
super();
|
|
28
|
+
this.HEARTBEAT_TIMEOUT = browser_connection_timeouts_1.HEARTBEAT_TIMEOUT;
|
|
29
|
+
this.BROWSER_CLOSE_TIMEOUT = browser_connection_timeouts_1.BROWSER_CLOSE_TIMEOUT;
|
|
30
|
+
this.BROWSER_RESTART_TIMEOUT = browser_connection_timeouts_1.BROWSER_RESTART_TIMEOUT;
|
|
31
|
+
this.id = BrowserConnection._generateId();
|
|
32
|
+
this.jobQueue = [];
|
|
33
|
+
this.initScriptsQueue = [];
|
|
34
|
+
this.browserConnectionGateway = gateway;
|
|
35
|
+
this.disconnectionPromise = null;
|
|
36
|
+
this.testRunAborted = false;
|
|
37
|
+
this.warningLog = new warning_log_1.default();
|
|
38
|
+
this.debugLogger = debug_1.default(getBrowserConnectionDebugScope(this.id));
|
|
39
|
+
this.browserInfo = browserInfo;
|
|
40
|
+
this.browserInfo.userAgentProviderMetaInfo = '';
|
|
41
|
+
this.provider = browserInfo.provider;
|
|
42
|
+
this.permanent = permanent;
|
|
43
|
+
this.status = status_1.default.uninitialized;
|
|
44
|
+
this.idle = true;
|
|
45
|
+
this.heartbeatTimeout = null;
|
|
46
|
+
this.pendingTestRunUrl = null;
|
|
47
|
+
this.disableMultipleWindows = disableMultipleWindows;
|
|
48
|
+
this.url = `${gateway.domain}/browser/connect/${this.id}`;
|
|
49
|
+
this.idleUrl = `${gateway.domain}/browser/idle/${this.id}`;
|
|
50
|
+
this.forcedIdleUrl = `${gateway.domain}/browser/idle-forced/${this.id}`;
|
|
51
|
+
this.initScriptUrl = `${gateway.domain}/browser/init-script/${this.id}`;
|
|
52
|
+
this.heartbeatRelativeUrl = `/browser/heartbeat/${this.id}`;
|
|
53
|
+
this.statusRelativeUrl = `/browser/status/${this.id}`;
|
|
54
|
+
this.statusDoneRelativeUrl = `/browser/status-done/${this.id}`;
|
|
55
|
+
this.activeWindowIdUrl = `/browser/active-window-id/${this.id}`;
|
|
56
|
+
this.heartbeatUrl = `${gateway.domain}${this.heartbeatRelativeUrl}`;
|
|
57
|
+
this.statusUrl = `${gateway.domain}${this.statusRelativeUrl}`;
|
|
58
|
+
this.statusDoneUrl = `${gateway.domain}${this.statusDoneRelativeUrl}`;
|
|
59
|
+
this._setEventHandlers();
|
|
60
|
+
connections[this.id] = this;
|
|
61
|
+
this.previousActiveWindowId = null;
|
|
62
|
+
this.browserConnectionGateway.startServingConnection(this);
|
|
63
|
+
// NOTE: Give a caller time to assign event listeners
|
|
64
|
+
process.nextTick(() => this._runBrowser());
|
|
65
|
+
}
|
|
66
|
+
_setEventHandlers() {
|
|
67
|
+
this.on('error', e => {
|
|
68
|
+
this.debugLogger(e);
|
|
69
|
+
this._forceIdle();
|
|
70
|
+
this.close();
|
|
71
|
+
});
|
|
72
|
+
for (const name in status_1.default) {
|
|
73
|
+
const status = status_1.default[name];
|
|
74
|
+
this.on(status, () => {
|
|
75
|
+
this.debugLogger(`status changed to '${status}'`);
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
static _generateId() {
|
|
80
|
+
return nanoid_1.default(7);
|
|
81
|
+
}
|
|
82
|
+
async _runBrowser() {
|
|
83
|
+
try {
|
|
84
|
+
await this.provider.openBrowser(this.id, this.url, this.browserInfo.browserName, this.disableMultipleWindows);
|
|
85
|
+
if (this.status !== status_1.default.ready)
|
|
86
|
+
await promisify_event_1.default(this, 'ready');
|
|
87
|
+
this.status = status_1.default.opened;
|
|
88
|
+
this.emit('opened');
|
|
89
|
+
}
|
|
90
|
+
catch (err) {
|
|
91
|
+
this.emit('error', new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.unableToOpenBrowser, this.browserInfo.providerName + ':' + this.browserInfo.browserName, err.stack));
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
async _closeBrowser() {
|
|
95
|
+
if (!this.idle)
|
|
96
|
+
await promisify_event_1.default(this, 'idle');
|
|
97
|
+
try {
|
|
98
|
+
await this.provider.closeBrowser(this.id);
|
|
99
|
+
}
|
|
100
|
+
catch (err) {
|
|
101
|
+
// NOTE: A warning would be really nice here, but it can't be done while log is stored in a task.
|
|
102
|
+
this.debugLogger(err);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
_forceIdle() {
|
|
106
|
+
if (!this.idle) {
|
|
107
|
+
this.idle = true;
|
|
108
|
+
this.emit('idle');
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
_createBrowserDisconnectedError() {
|
|
112
|
+
return new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.browserDisconnected, this.userAgent);
|
|
113
|
+
}
|
|
114
|
+
_waitForHeartbeat() {
|
|
115
|
+
this.heartbeatTimeout = setTimeout(() => {
|
|
116
|
+
const err = this._createBrowserDisconnectedError();
|
|
117
|
+
this.status = status_1.default.disconnected;
|
|
118
|
+
this.testRunAborted = true;
|
|
119
|
+
this.emit('disconnected', err);
|
|
120
|
+
this._restartBrowserOnDisconnect(err);
|
|
121
|
+
}, this.HEARTBEAT_TIMEOUT);
|
|
122
|
+
}
|
|
123
|
+
async _getTestRunUrl(needPopNext) {
|
|
124
|
+
if (needPopNext || !this.pendingTestRunUrl)
|
|
125
|
+
this.pendingTestRunUrl = await this._popNextTestRunUrl();
|
|
126
|
+
return this.pendingTestRunUrl;
|
|
127
|
+
}
|
|
128
|
+
async _popNextTestRunUrl() {
|
|
129
|
+
while (this.hasQueuedJobs && !this.currentJob.hasQueuedTestRuns)
|
|
130
|
+
this.jobQueue.shift();
|
|
131
|
+
return this.hasQueuedJobs ? await this.currentJob.popNextTestRunUrl(this) : null;
|
|
132
|
+
}
|
|
133
|
+
static getById(id) {
|
|
134
|
+
return connections[id] || null;
|
|
135
|
+
}
|
|
136
|
+
async _restartBrowser() {
|
|
137
|
+
this.status = status_1.default.uninitialized;
|
|
138
|
+
this._forceIdle();
|
|
139
|
+
let resolveTimeout = null;
|
|
140
|
+
let isTimeoutExpired = false;
|
|
141
|
+
let timeout = null;
|
|
142
|
+
const restartPromise = time_limit_promise_1.default(this._closeBrowser(), this.BROWSER_CLOSE_TIMEOUT, { rejectWith: new runtime_1.TimeoutError() })
|
|
143
|
+
.catch(err => this.debugLogger(err))
|
|
144
|
+
.then(() => this._runBrowser());
|
|
145
|
+
const timeoutPromise = new Promise(resolve => {
|
|
146
|
+
resolveTimeout = resolve;
|
|
147
|
+
timeout = setTimeout(() => {
|
|
148
|
+
isTimeoutExpired = true;
|
|
149
|
+
resolve();
|
|
150
|
+
}, this.BROWSER_RESTART_TIMEOUT);
|
|
151
|
+
});
|
|
152
|
+
return Promise.race([restartPromise, timeoutPromise])
|
|
153
|
+
.then(() => {
|
|
154
|
+
clearTimeout(timeout);
|
|
155
|
+
if (isTimeoutExpired)
|
|
156
|
+
this.emit('error', this._createBrowserDisconnectedError());
|
|
157
|
+
else
|
|
158
|
+
resolveTimeout();
|
|
159
|
+
});
|
|
160
|
+
}
|
|
161
|
+
_restartBrowserOnDisconnect(err) {
|
|
162
|
+
let resolveFn = null;
|
|
163
|
+
let rejectFn = null;
|
|
164
|
+
this.disconnectionPromise = new Promise((resolve, reject) => {
|
|
165
|
+
resolveFn = resolve;
|
|
166
|
+
rejectFn = () => {
|
|
167
|
+
reject(err);
|
|
168
|
+
};
|
|
169
|
+
setTimeout(() => {
|
|
170
|
+
rejectFn();
|
|
171
|
+
});
|
|
172
|
+
})
|
|
173
|
+
.then(() => {
|
|
174
|
+
return this._restartBrowser();
|
|
175
|
+
})
|
|
176
|
+
.catch(e => {
|
|
177
|
+
this.emit('error', e);
|
|
178
|
+
});
|
|
179
|
+
this.disconnectionPromise.resolve = resolveFn;
|
|
180
|
+
this.disconnectionPromise.reject = rejectFn;
|
|
181
|
+
}
|
|
182
|
+
async getDefaultBrowserInitTimeout() {
|
|
183
|
+
const isLocalBrowser = await this.provider.isLocalBrowser(this.id, this.browserInfo.browserName);
|
|
184
|
+
return isLocalBrowser ? browser_connection_timeouts_1.LOCAL_BROWSER_INIT_TIMEOUT : browser_connection_timeouts_1.REMOTE_BROWSER_INIT_TIMEOUT;
|
|
185
|
+
}
|
|
186
|
+
async processDisconnection(disconnectionThresholdExceeded) {
|
|
187
|
+
const { resolve, reject } = this.disconnectionPromise;
|
|
188
|
+
if (disconnectionThresholdExceeded)
|
|
189
|
+
reject();
|
|
190
|
+
else
|
|
191
|
+
resolve();
|
|
192
|
+
}
|
|
193
|
+
addWarning(...args) {
|
|
194
|
+
if (this.currentJob)
|
|
195
|
+
this.currentJob.warningLog.addWarning(...args);
|
|
196
|
+
else
|
|
197
|
+
this.warningLog.addWarning(...args);
|
|
198
|
+
}
|
|
199
|
+
_appendToPrettyUserAgent(str) {
|
|
200
|
+
this.browserInfo.parsedUserAgent.prettyUserAgent += ` (${str})`;
|
|
201
|
+
}
|
|
202
|
+
_moveWarningLogToJob(job) {
|
|
203
|
+
this.warningLog.copyTo(job.warningLog);
|
|
204
|
+
this.warningLog.clear();
|
|
205
|
+
}
|
|
206
|
+
setProviderMetaInfo(str, options) {
|
|
207
|
+
const appendToUserAgent = options === null || options === void 0 ? void 0 : options.appendToUserAgent;
|
|
208
|
+
if (appendToUserAgent) {
|
|
209
|
+
// NOTE:
|
|
210
|
+
// change prettyUserAgent only when connection already was established
|
|
211
|
+
if (this.isReady())
|
|
212
|
+
this._appendToPrettyUserAgent(str);
|
|
213
|
+
else
|
|
214
|
+
this.on('ready', () => this._appendToPrettyUserAgent(str));
|
|
215
|
+
return;
|
|
216
|
+
}
|
|
217
|
+
this.browserInfo.userAgentProviderMetaInfo = str;
|
|
218
|
+
}
|
|
219
|
+
get userAgent() {
|
|
220
|
+
let userAgent = this.browserInfo.parsedUserAgent.prettyUserAgent;
|
|
221
|
+
if (this.browserInfo.userAgentProviderMetaInfo)
|
|
222
|
+
userAgent += ` (${this.browserInfo.userAgentProviderMetaInfo})`;
|
|
223
|
+
return userAgent;
|
|
224
|
+
}
|
|
225
|
+
get retryTestPages() {
|
|
226
|
+
return this.browserConnectionGateway.retryTestPages;
|
|
227
|
+
}
|
|
228
|
+
get hasQueuedJobs() {
|
|
229
|
+
return !!this.jobQueue.length;
|
|
230
|
+
}
|
|
231
|
+
get currentJob() {
|
|
232
|
+
return this.jobQueue[0];
|
|
233
|
+
}
|
|
234
|
+
// API
|
|
235
|
+
runInitScript(code) {
|
|
236
|
+
return new Promise(resolve => this.initScriptsQueue.push({ code, resolve }));
|
|
237
|
+
}
|
|
238
|
+
addJob(job) {
|
|
239
|
+
this.jobQueue.push(job);
|
|
240
|
+
this._moveWarningLogToJob(job);
|
|
241
|
+
}
|
|
242
|
+
removeJob(job) {
|
|
243
|
+
lodash_1.pull(this.jobQueue, job);
|
|
244
|
+
}
|
|
245
|
+
async close() {
|
|
246
|
+
if (this.status === status_1.default.closing || this.status === status_1.default.closed)
|
|
247
|
+
return;
|
|
248
|
+
this.status = status_1.default.closing;
|
|
249
|
+
this.emit(status_1.default.closing);
|
|
250
|
+
await this._closeBrowser();
|
|
251
|
+
this.browserConnectionGateway.stopServingConnection(this);
|
|
252
|
+
if (this.heartbeatTimeout)
|
|
253
|
+
clearTimeout(this.heartbeatTimeout);
|
|
254
|
+
delete connections[this.id];
|
|
255
|
+
this.status = status_1.default.closed;
|
|
256
|
+
this.emit(status_1.default.closed);
|
|
257
|
+
}
|
|
258
|
+
establish(userAgent) {
|
|
259
|
+
this.status = status_1.default.ready;
|
|
260
|
+
this.browserInfo.parsedUserAgent = parse_user_agent_1.default(userAgent);
|
|
261
|
+
this._waitForHeartbeat();
|
|
262
|
+
this.emit('ready');
|
|
263
|
+
}
|
|
264
|
+
heartbeat() {
|
|
265
|
+
if (this.heartbeatTimeout)
|
|
266
|
+
clearTimeout(this.heartbeatTimeout);
|
|
267
|
+
this._waitForHeartbeat();
|
|
268
|
+
return {
|
|
269
|
+
code: this.status === status_1.default.closing ? heartbeat_status_1.default.closing : heartbeat_status_1.default.ok,
|
|
270
|
+
url: this.status === status_1.default.closing ? this.idleUrl : ''
|
|
271
|
+
};
|
|
272
|
+
}
|
|
273
|
+
renderIdlePage() {
|
|
274
|
+
return mustache_1.default.render(IDLE_PAGE_TEMPLATE, {
|
|
275
|
+
userAgent: this.userAgent,
|
|
276
|
+
statusUrl: this.statusUrl,
|
|
277
|
+
heartbeatUrl: this.heartbeatUrl,
|
|
278
|
+
initScriptUrl: this.initScriptUrl,
|
|
279
|
+
retryTestPages: !!this.browserConnectionGateway.retryTestPages
|
|
280
|
+
});
|
|
281
|
+
}
|
|
282
|
+
getInitScript() {
|
|
283
|
+
const initScriptPromise = this.initScriptsQueue[0];
|
|
284
|
+
return { code: initScriptPromise ? initScriptPromise.code : null };
|
|
285
|
+
}
|
|
286
|
+
handleInitScriptResult(data) {
|
|
287
|
+
const initScriptPromise = this.initScriptsQueue.shift();
|
|
288
|
+
if (initScriptPromise)
|
|
289
|
+
initScriptPromise.resolve(JSON.parse(data));
|
|
290
|
+
}
|
|
291
|
+
isHeadlessBrowser() {
|
|
292
|
+
return this.provider.isHeadlessBrowser(this.id);
|
|
293
|
+
}
|
|
294
|
+
async reportJobResult(status, data) {
|
|
295
|
+
await this.provider.reportJobResult(this.id, status, data);
|
|
296
|
+
}
|
|
297
|
+
async getStatus(isTestDone) {
|
|
298
|
+
if (!this.idle && !isTestDone) {
|
|
299
|
+
this.idle = true;
|
|
300
|
+
this.emit('idle');
|
|
301
|
+
}
|
|
302
|
+
if (this.status === status_1.default.opened) {
|
|
303
|
+
const testRunUrl = await this._getTestRunUrl(isTestDone || this.testRunAborted);
|
|
304
|
+
this.testRunAborted = false;
|
|
305
|
+
if (testRunUrl) {
|
|
306
|
+
this.idle = false;
|
|
307
|
+
return { cmd: command_1.default.run, url: testRunUrl };
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
return { cmd: command_1.default.idle, url: this.idleUrl };
|
|
311
|
+
}
|
|
312
|
+
get activeWindowId() {
|
|
313
|
+
return this.provider.getActiveWindowId(this.id);
|
|
314
|
+
}
|
|
315
|
+
set activeWindowId(val) {
|
|
316
|
+
this.previousActiveWindowId = this.activeWindowId;
|
|
317
|
+
this.provider.setActiveWindowId(this.id, val);
|
|
318
|
+
}
|
|
319
|
+
async canUseDefaultWindowActions() {
|
|
320
|
+
return this.provider.canUseDefaultWindowActions(this.id);
|
|
321
|
+
}
|
|
322
|
+
isReady() {
|
|
323
|
+
return this.status === status_1.default.ready ||
|
|
324
|
+
this.status === status_1.default.opened ||
|
|
325
|
+
this.status === status_1.default.closing;
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
exports.default = BrowserConnection;
|
|
329
329
|
module.exports = exports.default;
|
|
330
330
|
//# 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 parseUserAgent, { ParsedUserAgent } 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 { Dictionary } from '../../configuration/interfaces';\nimport BrowserConnectionGateway from './gateway';\nimport BrowserJob from '../../runner/browser-job';\nimport WarningLog from '../../notifications/warning-log';\nimport BrowserProvider from '../provider';\n\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';\n\nconst getBrowserConnectionDebugScope = (id: string): string => `testcafe:browser:connection:${id}`;\n\nconst IDLE_PAGE_TEMPLATE                         = read('../../client/browser/idle-page/index.html.mustache');\nconst connections: Dictionary<BrowserConnection> = {};\n\ninterface DisconnectionPromise<T> extends Promise<T> {\n    resolve: Function;\n    reject: Function;\n}\n\ninterface BrowserConnectionStatusResult {\n    cmd: string;\n    url: string;\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 BrowserInfo {\n    alias: string;\n    browserName: string;\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    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 readonly jobQueue: BrowserJob[];\n    private readonly initScriptsQueue: InitScriptTask[];\n    private browserConnectionGateway: BrowserConnectionGateway;\n    private disconnectionPromise: DisconnectionPromise<void> | null;\n    private testRunAborted: boolean;\n    public status: BrowserConnectionStatus;\n    private heartbeatTimeout: NodeJS.Timeout | null;\n    private pendingTestRunUrl: string | null;\n    public readonly url: string;\n    public readonly idleUrl: string;\n    private forcedIdleUrl: string;\n    private readonly initScriptUrl: string;\n    private readonly heartbeatRelativeUrl: string;\n    private readonly statusRelativeUrl: string;\n    private readonly statusDoneRelativeUrl: string;\n    private readonly heartbeatUrl: string;\n    private readonly statusUrl: string;\n    private readonly activeWindowIdUrl: string;\n    private statusDoneUrl: string;\n    private readonly debugLogger: debug.Debugger;\n\n    public readonly warningLog: WarningLog;\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        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();\n        this.debugLogger              = debug(getBrowserConnectionDebugScope(this.id));\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.pendingTestRunUrl      = null;\n        this.disableMultipleWindows = disableMultipleWindows;\n\n        this.url           = `${gateway.domain}/browser/connect/${this.id}`;\n        this.idleUrl       = `${gateway.domain}/browser/idle/${this.id}`;\n        this.forcedIdleUrl = `${gateway.domain}/browser/idle-forced/${this.id}`;\n        this.initScriptUrl = `${gateway.domain}/browser/init-script/${this.id}`;\n\n        this.heartbeatRelativeUrl  = `/browser/heartbeat/${this.id}`;\n        this.statusRelativeUrl     = `/browser/status/${this.id}`;\n        this.statusDoneRelativeUrl = `/browser/status-done/${this.id}`;\n        this.activeWindowIdUrl     = `/browser/active-window-id/${this.id}`;\n\n        this.heartbeatUrl  = `${gateway.domain}${this.heartbeatRelativeUrl}`;\n        this.statusUrl     = `${gateway.domain}${this.statusRelativeUrl}`;\n        this.statusDoneUrl = `${gateway.domain}${this.statusDoneRelativeUrl}`;\n\n        this._setEventHandlers();\n\n        connections[this.id] = 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 _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 async _runBrowser (): Promise<void> {\n        try {\n            await this.provider.openBrowser(this.id, this.url, this.browserInfo.browserName, this.disableMultipleWindows);\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) {\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 (): Promise<void> {\n        if (!this.idle)\n            await promisifyEvent(this, 'idle');\n\n        try {\n            await this.provider.closeBrowser(this.id);\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 _getTestRunUrl (needPopNext: boolean): Promise<string> {\n        if (needPopNext || !this.pendingTestRunUrl)\n            this.pendingTestRunUrl = await this._popNextTestRunUrl();\n\n        return this.pendingTestRunUrl as string;\n    }\n\n    private async _popNextTestRunUrl (): Promise<string | null> {\n        while (this.hasQueuedJobs && !this.currentJob.hasQueuedTestRuns)\n            this.jobQueue.shift();\n\n        return this.hasQueuedJobs ? await this.currentJob.popNextTestRunUrl(this) : null;\n    }\n\n    public static getById (id: string): BrowserConnection | null {\n        return connections[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(), this.BROWSER_CLOSE_TIMEOUT, { rejectWith: new TimeoutError() })\n            .catch(err => this.debugLogger(err))\n            .then(() => this._runBrowser());\n\n        const timeoutPromise = new Promise(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 (...args: any[]): void {\n        if (this.currentJob)\n            this.currentJob.warningLog.addWarning(...args);\n        else\n            this.warningLog.addWarning(...args);\n    }\n\n    private _appendToPrettyUserAgent (str: string): void {\n        this.browserInfo.parsedUserAgent.prettyUserAgent += ` (${str})`;\n    }\n\n    private _moveWarningLogToJob (job: BrowserJob): void {\n        this.warningLog.copyTo(job.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 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        delete connections[this.id];\n\n        this.status = BrowserConnectionStatus.closed;\n        this.emit(BrowserConnectionStatus.closed);\n    }\n\n    public establish (userAgent: string): void {\n        this.status                      = BrowserConnectionStatus.ready;\n        this.browserInfo.parsedUserAgent = parseUserAgent(userAgent);\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.userAgent,\n            statusUrl:      this.statusUrl,\n            heartbeatUrl:   this.heartbeatUrl,\n            initScriptUrl:  this.initScriptUrl,\n            retryTestPages: !!this.browserConnectionGateway.retryTestPages\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 testRunUrl = await this._getTestRunUrl(isTestDone || this.testRunAborted);\n\n            this.testRunAborted = false;\n\n            if (testRunUrl) {\n                this.idle = false;\n\n                return { cmd: COMMAND.run, url: testRunUrl };\n            }\n        }\n\n        return { cmd: COMMAND.idle, url: this.idleUrl };\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 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"]}
|