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.
Files changed (332) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/LICENSE +21 -21
  3. package/bin/testcafe-with-v8-flag-filter.js +0 -0
  4. package/lib/api/exportable-lib/index.js +49 -49
  5. package/lib/api/request-hooks/assert-type.js +7 -7
  6. package/lib/api/request-hooks/hook-method-names.js +9 -9
  7. package/lib/api/request-hooks/hook.js +32 -32
  8. package/lib/api/request-hooks/interfaces.js +2 -2
  9. package/lib/api/request-hooks/request-logger.js +112 -112
  10. package/lib/api/request-hooks/request-mock/create-request-mock.js +10 -10
  11. package/lib/api/request-hooks/request-mock/index.js +44 -44
  12. package/lib/api/request-hooks/request-mock.js +46 -46
  13. package/lib/api/structure/base-unit.js +11 -11
  14. package/lib/api/structure/fixture.js +75 -75
  15. package/lib/api/structure/interfaces.js +2 -2
  16. package/lib/api/structure/test-file.js +31 -31
  17. package/lib/api/structure/test-timeout.js +9 -9
  18. package/lib/api/structure/test.js +85 -85
  19. package/lib/api/structure/testing-unit.js +89 -89
  20. package/lib/api/structure/unit-type.js +9 -9
  21. package/lib/api/test-controller/assertion.js +88 -88
  22. package/lib/api/test-controller/execution-context.js +82 -82
  23. package/lib/api/test-controller/index.js +358 -358
  24. package/lib/api/test-controller/proxy.js +28 -28
  25. package/lib/api/test-page-url.js +60 -60
  26. package/lib/api/test-run-tracker.js +68 -68
  27. package/lib/api/wrap-test-function.js +49 -49
  28. package/lib/assertions/executor.js +74 -74
  29. package/lib/assertions/get-fn.js +46 -46
  30. package/lib/assertions/type.js +20 -20
  31. package/lib/assets/content-types.js +9 -9
  32. package/lib/assets/injectables.js +18 -18
  33. package/lib/browser/connection/command.js +10 -10
  34. package/lib/browser/connection/error-hints.js +9 -9
  35. package/lib/browser/connection/gateway.js +159 -159
  36. package/lib/browser/connection/get-hints.js +33 -33
  37. package/lib/browser/connection/heartbeat-status.js +8 -8
  38. package/lib/browser/connection/index.js +328 -328
  39. package/lib/browser/connection/remotes-queue.js +46 -46
  40. package/lib/browser/connection/service-routes.js +12 -12
  41. package/lib/browser/connection/status.js +12 -12
  42. package/lib/browser/interfaces.js +2 -2
  43. package/lib/browser/provider/built-in/dedicated/base.js +80 -80
  44. package/lib/browser/provider/built-in/dedicated/chrome/browser-client.js +204 -204
  45. package/lib/browser/provider/built-in/dedicated/chrome/build-chrome-args.js +17 -17
  46. package/lib/browser/provider/built-in/dedicated/chrome/config.js +110 -110
  47. package/lib/browser/provider/built-in/dedicated/chrome/create-temp-profile.js +45 -45
  48. package/lib/browser/provider/built-in/dedicated/chrome/elapsed-upperbounds.js +15 -15
  49. package/lib/browser/provider/built-in/dedicated/chrome/index.js +102 -102
  50. package/lib/browser/provider/built-in/dedicated/chrome/interfaces.js +2 -2
  51. package/lib/browser/provider/built-in/dedicated/chrome/local-chrome.js +24 -24
  52. package/lib/browser/provider/built-in/dedicated/chrome/runtime-info.js +29 -29
  53. package/lib/browser/provider/built-in/dedicated/edge/index.js +10 -10
  54. package/lib/browser/provider/built-in/dedicated/edge/runtime-info.js +29 -29
  55. package/lib/browser/provider/built-in/dedicated/firefox/config.js +33 -33
  56. package/lib/browser/provider/built-in/dedicated/firefox/create-temp-profile.js +78 -78
  57. package/lib/browser/provider/built-in/dedicated/firefox/index.js +73 -73
  58. package/lib/browser/provider/built-in/dedicated/firefox/local-firefox.js +36 -36
  59. package/lib/browser/provider/built-in/dedicated/firefox/marionette-client/commands.js +13 -13
  60. package/lib/browser/provider/built-in/dedicated/firefox/marionette-client/index.js +200 -200
  61. package/lib/browser/provider/built-in/dedicated/firefox/runtime-info.js +17 -17
  62. package/lib/browser/provider/built-in/index.js +21 -21
  63. package/lib/browser/provider/built-in/locally-installed.js +30 -30
  64. package/lib/browser/provider/built-in/path.js +47 -47
  65. package/lib/browser/provider/built-in/remote.js +58 -58
  66. package/lib/browser/provider/index.js +303 -303
  67. package/lib/browser/provider/parse-provider-name.js +16 -16
  68. package/lib/browser/provider/plugin-host.js +121 -121
  69. package/lib/browser/provider/pool.js +115 -115
  70. package/lib/browser/provider/utils/argument-parsing.js +74 -74
  71. package/lib/browser/provider/utils/browser-starter.js +34 -34
  72. package/lib/browser/provider/utils/client-functions.js +24 -24
  73. package/lib/browser/provider/utils/get-maximized-headless-window-size.js +9 -9
  74. package/lib/cli/argument-parser.js +284 -284
  75. package/lib/cli/authentication-helper.js +35 -35
  76. package/lib/cli/cli.js +134 -134
  77. package/lib/cli/correct-browsers-and-sources.js +40 -40
  78. package/lib/cli/index.js +19 -19
  79. package/lib/cli/log.js +43 -43
  80. package/lib/cli/remotes-wizard.js +36 -36
  81. package/lib/cli/termination-handler.js +38 -38
  82. package/lib/client/automation/index.js +34 -16
  83. package/lib/client/automation/index.min.js +1 -1
  84. package/lib/client/browser/idle-page/index.html.mustache +35 -35
  85. package/lib/client/browser/idle-page/logo.svg +86 -86
  86. package/lib/client/core/index.js +14 -14
  87. package/lib/client/driver/index.js +46 -16
  88. package/lib/client/driver/index.min.js +1 -1
  89. package/lib/client/driver/internal-properties.js +9 -9
  90. package/lib/client/test-run/iframe.js.mustache +17 -17
  91. package/lib/client/test-run/index.js.mustache +51 -51
  92. package/lib/client/ui/index.js +14 -14
  93. package/lib/client/ui/sprite.svg +42 -42
  94. package/lib/client-functions/builder-symbol.js +4 -4
  95. package/lib/client-functions/client-function-builder.js +155 -155
  96. package/lib/client-functions/replicator.js +61 -61
  97. package/lib/client-functions/return-single-prop-mode.js +8 -8
  98. package/lib/client-functions/selector-api-execution-mode.js +20 -20
  99. package/lib/client-functions/selectors/add-api.js +645 -645
  100. package/lib/client-functions/selectors/create-snapshot-methods.js +13 -13
  101. package/lib/client-functions/selectors/prepare-api-args.js +20 -20
  102. package/lib/client-functions/selectors/selector-attribute-filter.js +22 -22
  103. package/lib/client-functions/selectors/selector-builder.js +153 -153
  104. package/lib/client-functions/selectors/selector-text-filter.js +43 -43
  105. package/lib/client-functions/selectors/snapshot-properties.js +48 -48
  106. package/lib/client-functions/types.js +18 -18
  107. package/lib/compiler/babel/format-babel-produced-code.js +10 -10
  108. package/lib/compiler/babel/get-base-babel-options.js +11 -11
  109. package/lib/compiler/babel/load-libs.js +63 -63
  110. package/lib/compiler/babel/preset-stage-2.js +19 -19
  111. package/lib/compiler/compile-client-function.js +73 -73
  112. package/lib/compiler/compilers.js +33 -33
  113. package/lib/compiler/index.js +92 -92
  114. package/lib/compiler/interfaces.js +2 -2
  115. package/lib/compiler/test-file/api-based.js +146 -146
  116. package/lib/compiler/test-file/base.js +36 -36
  117. package/lib/compiler/test-file/exportble-lib-path.js +5 -5
  118. package/lib/compiler/test-file/formats/coffeescript/compiler.js +38 -38
  119. package/lib/compiler/test-file/formats/coffeescript/get-test-list.js +29 -29
  120. package/lib/compiler/test-file/formats/es-next/compiler.js +42 -42
  121. package/lib/compiler/test-file/formats/es-next/get-test-list.js +166 -166
  122. package/lib/compiler/test-file/formats/es-next/is-flow-code.js +7 -7
  123. package/lib/compiler/test-file/formats/raw.js +85 -85
  124. package/lib/compiler/test-file/formats/typescript/compiler.js +135 -135
  125. package/lib/compiler/test-file/formats/typescript/get-test-list.js +185 -185
  126. package/lib/compiler/test-file/test-file-parser-base.js +214 -214
  127. package/lib/configuration/configuration-base.js +165 -165
  128. package/lib/configuration/constants.js +9 -9
  129. package/lib/configuration/customizable-compilers.js +7 -7
  130. package/lib/configuration/default-values.js +51 -51
  131. package/lib/configuration/interfaces.js +2 -2
  132. package/lib/configuration/option-names.js +53 -53
  133. package/lib/configuration/option-source.js +9 -9
  134. package/lib/configuration/option.js +14 -14
  135. package/lib/configuration/quarantine-option-names.js +8 -8
  136. package/lib/configuration/run-option-names.js +26 -26
  137. package/lib/configuration/screenshot-option-names.js +10 -10
  138. package/lib/configuration/testcafe-configuration.js +163 -163
  139. package/lib/configuration/types.js +2 -2
  140. package/lib/configuration/typescript-configuration.js +62 -62
  141. package/lib/custom-client-scripts/assert-type.js +7 -7
  142. package/lib/custom-client-scripts/client-script-init.js +2 -2
  143. package/lib/custom-client-scripts/client-script.js +106 -106
  144. package/lib/custom-client-scripts/get-code.js +11 -11
  145. package/lib/custom-client-scripts/get-url.js +6 -6
  146. package/lib/custom-client-scripts/load.js +15 -15
  147. package/lib/custom-client-scripts/problematic-scripts.js +2 -2
  148. package/lib/custom-client-scripts/routing.js +36 -36
  149. package/lib/custom-client-scripts/utils.js +60 -60
  150. package/lib/embedding-utils.js +83 -83
  151. package/lib/errors/create-stack-filter.js +18 -18
  152. package/lib/errors/error-list.js +26 -26
  153. package/lib/errors/get-callsite.js +31 -31
  154. package/lib/errors/internal-modules-prefix.js +8 -8
  155. package/lib/errors/is-internal-stack-frame.js +45 -45
  156. package/lib/errors/process-test-fn-error.js +37 -37
  157. package/lib/errors/runtime/index.js +123 -123
  158. package/lib/errors/runtime/templates.js +115 -115
  159. package/lib/errors/runtime/type-assertions.js +112 -112
  160. package/lib/errors/stack-cleaning-hook.js +64 -64
  161. package/lib/errors/test-run/formattable-adapter.js +59 -59
  162. package/lib/errors/test-run/index.js +301 -301
  163. package/lib/errors/test-run/render-error-template.js +31 -31
  164. package/lib/errors/test-run/templates.js +91 -91
  165. package/lib/errors/test-run/utils.js +89 -89
  166. package/lib/errors/types.js +156 -156
  167. package/lib/index.js +81 -81
  168. package/lib/live/bootstrapper.js +43 -43
  169. package/lib/live/controller.js +107 -107
  170. package/lib/live/file-watcher/index.js +67 -67
  171. package/lib/live/file-watcher/modules-graph.js +58 -58
  172. package/lib/live/keyboard-observer.js +76 -76
  173. package/lib/live/logger/index.js +64 -64
  174. package/lib/live/test-run-controller.js +96 -96
  175. package/lib/live/test-run-state.js +6 -6
  176. package/lib/live/test-run.js +56 -56
  177. package/lib/live/test-runner.js +167 -167
  178. package/lib/load-assets.js +29 -29
  179. package/lib/notifications/add-rendered-warning.js +16 -16
  180. package/lib/notifications/debug-logger.js +78 -78
  181. package/lib/notifications/deprecated.js +24 -24
  182. package/lib/notifications/information-message.js +9 -9
  183. package/lib/notifications/warning-log.js +31 -31
  184. package/lib/notifications/warning-message.js +47 -47
  185. package/lib/reporter/command/command-formatter.js +120 -109
  186. package/lib/reporter/command/format-command.js +8 -8
  187. package/lib/reporter/command/interfaces.js +2 -2
  188. package/lib/reporter/index.js +319 -314
  189. package/lib/reporter/interfaces.js +2 -2
  190. package/lib/reporter/plugin-host.js +135 -135
  191. package/lib/reporter/plugin-methods.js +12 -12
  192. package/lib/role/index.js +74 -74
  193. package/lib/role/marker-symbol.js +7 -7
  194. package/lib/role/phase.js +9 -9
  195. package/lib/runner/bootstrapper.js +271 -271
  196. package/lib/runner/browser-job-result.js +9 -9
  197. package/lib/runner/browser-job.js +152 -152
  198. package/lib/runner/browser-set.js +114 -114
  199. package/lib/runner/fixture-hook-controller.js +85 -85
  200. package/lib/runner/index.js +449 -449
  201. package/lib/runner/interfaces.js +2 -2
  202. package/lib/runner/reporter-stream-controller.js +27 -27
  203. package/lib/runner/task/index.js +151 -151
  204. package/lib/runner/task/phase.js +9 -9
  205. package/lib/runner/test-run-controller.js +165 -165
  206. package/lib/runner/tested-app.js +72 -72
  207. package/lib/screenshots/capturer.js +141 -141
  208. package/lib/screenshots/constants.js +11 -11
  209. package/lib/screenshots/crop.js +111 -111
  210. package/lib/screenshots/default-extension.js +4 -4
  211. package/lib/screenshots/index.js +67 -67
  212. package/lib/screenshots/utils.js +39 -39
  213. package/lib/services/compiler/host.js +190 -190
  214. package/lib/services/compiler/io.js +9 -9
  215. package/lib/services/compiler/protocol.js +17 -16
  216. package/lib/services/compiler/service.js +225 -212
  217. package/lib/services/compiler/test-run-proxy.js +111 -111
  218. package/lib/services/interfaces.js +2 -2
  219. package/lib/services/process-title.js +8 -8
  220. package/lib/services/serialization/prepare-options.js +17 -17
  221. package/lib/services/serialization/replicator/create-replicator.js +25 -25
  222. package/lib/services/serialization/replicator/custom-error-transform.js +26 -26
  223. package/lib/services/serialization/test-structure.js +92 -92
  224. package/lib/services/utils/ipc/interfaces.js +30 -30
  225. package/lib/services/utils/ipc/io.js +108 -108
  226. package/lib/services/utils/ipc/message.js +75 -75
  227. package/lib/services/utils/ipc/packet.js +55 -55
  228. package/lib/services/utils/ipc/proxy.js +109 -109
  229. package/lib/services/utils/ipc/transport.js +64 -64
  230. package/lib/shared/errors/index.js +382 -382
  231. package/lib/shared/node-modules-folder-name.js +4 -4
  232. package/lib/test-run/bookmark.js +90 -90
  233. package/lib/test-run/browser-console-messages.js +73 -73
  234. package/lib/test-run/browser-manipulation-queue.js +92 -92
  235. package/lib/test-run/client-messages.js +9 -9
  236. package/lib/test-run/commands/actions.js +486 -483
  237. package/lib/test-run/commands/assertion.js +45 -45
  238. package/lib/test-run/commands/base.js +14 -14
  239. package/lib/test-run/commands/browser-manipulation.js +95 -95
  240. package/lib/test-run/commands/from-object.js +82 -82
  241. package/lib/test-run/commands/observation.js +61 -61
  242. package/lib/test-run/commands/options.js +231 -215
  243. package/lib/test-run/commands/service.js +54 -48
  244. package/lib/test-run/commands/type.js +65 -64
  245. package/lib/test-run/commands/utils.js +87 -87
  246. package/lib/test-run/commands/validations/argument.js +90 -90
  247. package/lib/test-run/commands/validations/factories.js +47 -47
  248. package/lib/test-run/commands/validations/initializers.js +44 -44
  249. package/lib/test-run/debug-log.js +32 -32
  250. package/lib/test-run/execute-js-expression.js +74 -74
  251. package/lib/test-run/index.js +823 -799
  252. package/lib/test-run/marker-symbol.js +7 -7
  253. package/lib/test-run/observed-callsites-storage.js +17 -17
  254. package/lib/test-run/phase.js +16 -16
  255. package/lib/test-run/session-controller.js +104 -104
  256. package/lib/testcafe.js +118 -118
  257. package/lib/utils/assignable.js +39 -39
  258. package/lib/utils/async-event-emitter.js +28 -28
  259. package/lib/utils/async-queue.js +14 -14
  260. package/lib/utils/browser-connection-timeouts.js +19 -19
  261. package/lib/utils/callsite.js +17 -17
  262. package/lib/utils/check-file-path.js +31 -31
  263. package/lib/utils/check-url.js +51 -51
  264. package/lib/utils/convert-to-best-fit-type.js +16 -16
  265. package/lib/utils/correct-file-path.js +21 -21
  266. package/lib/utils/define-lazy-property.js +13 -13
  267. package/lib/utils/delay.js +6 -6
  268. package/lib/utils/delegated-api.js +44 -44
  269. package/lib/utils/detect-display.js +6 -6
  270. package/lib/utils/detect-ffmpeg.js +44 -44
  271. package/lib/utils/diff/colors.js +29 -29
  272. package/lib/utils/diff/index.js +52 -52
  273. package/lib/utils/diff/util.js +23 -23
  274. package/lib/utils/diff.js +29 -29
  275. package/lib/utils/escape-user-agent.js +10 -10
  276. package/lib/utils/flag-list.js +17 -17
  277. package/lib/utils/get-any-key.js +8 -8
  278. package/lib/utils/get-browser.js +8 -8
  279. package/lib/utils/get-common-path.js +34 -34
  280. package/lib/utils/get-filter-fn.js +40 -40
  281. package/lib/utils/get-options/base.js +36 -36
  282. package/lib/utils/get-options/compiler.js +33 -33
  283. package/lib/utils/get-options/grep.js +15 -15
  284. package/lib/utils/get-options/index.js +20 -20
  285. package/lib/utils/get-options/meta.js +22 -22
  286. package/lib/utils/get-options/quarantine.js +91 -91
  287. package/lib/utils/get-options/screenshot.js +17 -17
  288. package/lib/utils/get-options/ssl.js +45 -45
  289. package/lib/utils/get-options/video.js +10 -10
  290. package/lib/utils/get-viewport-width.js +17 -17
  291. package/lib/utils/guard-time-execution.js +10 -10
  292. package/lib/utils/handle-errors.js +74 -74
  293. package/lib/utils/handle-tag-args.js +8 -8
  294. package/lib/utils/http.js +30 -30
  295. package/lib/utils/is-localhost.js +11 -11
  296. package/lib/utils/is-password-input.js +11 -0
  297. package/lib/utils/is-repl.js +10 -10
  298. package/lib/utils/is-window-in-iframe.js +6 -6
  299. package/lib/utils/limit-number.js +10 -10
  300. package/lib/utils/make-reg-exp.js +7 -7
  301. package/lib/utils/moment-loader.js +20 -20
  302. package/lib/utils/parse-file-list.js +71 -71
  303. package/lib/utils/parse-user-agent.js +55 -55
  304. package/lib/utils/path-pattern.js +114 -114
  305. package/lib/utils/prepare-reporters.js +30 -30
  306. package/lib/utils/prerender-callsite.js +18 -18
  307. package/lib/utils/process.js +119 -119
  308. package/lib/utils/promisified-functions.js +46 -46
  309. package/lib/utils/re-executable-promise.js +39 -39
  310. package/lib/utils/render-callsite-sync.js +29 -29
  311. package/lib/utils/render-template.js +9 -9
  312. package/lib/utils/reporter.js +30 -30
  313. package/lib/utils/resolve-path-relatively-cwd.js +7 -7
  314. package/lib/utils/string.js +105 -105
  315. package/lib/utils/temp-directory/cleanup-process/commands.js +7 -7
  316. package/lib/utils/temp-directory/cleanup-process/index.js +143 -143
  317. package/lib/utils/temp-directory/cleanup-process/worker.js +58 -58
  318. package/lib/utils/temp-directory/index.js +87 -87
  319. package/lib/utils/temp-directory/lockfile.js +56 -56
  320. package/lib/utils/thennable.js +7 -7
  321. package/lib/utils/timer.js +15 -15
  322. package/lib/utils/to-posix-path.js +8 -8
  323. package/lib/utils/types.js +2 -2
  324. package/lib/video-recorder/interfaces.js +2 -2
  325. package/lib/video-recorder/process.js +126 -126
  326. package/lib/video-recorder/recorder.js +136 -136
  327. package/lib/video-recorder/test-run-video-recorder.js +69 -69
  328. package/lib/video-recorder/videos.js +37 -37
  329. package/package.json +2 -2
  330. package/ts-defs/index.d.ts +25 -14
  331. package/ts-defs/selectors.d.ts +16 -5
  332. package/ts-defs/testcafe-scripts.d.ts +17 -6
@@ -1,451 +1,451 @@
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 path_1 = require("path");
7
- const debug_1 = __importDefault(require("debug"));
8
- const promisify_event_1 = __importDefault(require("promisify-event"));
9
- const events_1 = require("events");
10
- const lodash_1 = require("lodash");
11
- const bootstrapper_1 = __importDefault(require("./bootstrapper"));
12
- const reporter_1 = __importDefault(require("../reporter"));
13
- const task_1 = __importDefault(require("./task"));
14
- const debug_logger_1 = __importDefault(require("../notifications/debug-logger"));
15
- const runtime_1 = require("../errors/runtime");
16
- const types_1 = require("../errors/types");
17
- const type_assertions_1 = require("../errors/runtime/type-assertions");
18
- const utils_1 = require("../errors/test-run/utils");
19
- const detect_ffmpeg_1 = __importDefault(require("../utils/detect-ffmpeg"));
20
- const check_file_path_1 = __importDefault(require("../utils/check-file-path"));
21
- const handle_errors_1 = require("../utils/handle-errors");
22
- const option_names_1 = __importDefault(require("../configuration/option-names"));
23
- const flag_list_1 = __importDefault(require("../utils/flag-list"));
24
- const prepare_reporters_1 = __importDefault(require("../utils/prepare-reporters"));
25
- const load_1 = __importDefault(require("../custom-client-scripts/load"));
26
- const utils_2 = require("../custom-client-scripts/utils");
27
- const reporter_stream_controller_1 = __importDefault(require("./reporter-stream-controller"));
28
- const customizable_compilers_1 = __importDefault(require("../configuration/customizable-compilers"));
29
- const string_1 = require("../utils/string");
30
- const is_localhost_1 = __importDefault(require("../utils/is-localhost"));
31
- const warning_log_1 = __importDefault(require("../notifications/warning-log"));
32
- const DEBUG_LOGGER = debug_1.default('testcafe:runner');
33
- class Runner extends events_1.EventEmitter {
34
- constructor({ proxy, browserConnectionGateway, configuration, compilerService }) {
35
- super();
36
- this.proxy = proxy;
37
- this.bootstrapper = this._createBootstrapper(browserConnectionGateway, compilerService);
38
- this.pendingTaskPromises = [];
39
- this.configuration = configuration;
40
- this.isCli = false;
41
- this.warningLog = new warning_log_1.default();
42
- this.compilerService = compilerService;
43
- this.apiMethodWasCalled = new flag_list_1.default([
44
- option_names_1.default.src,
45
- option_names_1.default.browsers,
46
- option_names_1.default.reporter,
47
- option_names_1.default.clientScripts
48
- ]);
49
- }
50
- _createBootstrapper(browserConnectionGateway, compilerService) {
51
- return new bootstrapper_1.default({ browserConnectionGateway, compilerService });
52
- }
53
- _disposeBrowserSet(browserSet) {
54
- return browserSet.dispose().catch(e => DEBUG_LOGGER(e));
55
- }
56
- _disposeReporters(reporters) {
57
- return Promise.all(reporters.map(reporter => reporter.dispose().catch(e => DEBUG_LOGGER(e))));
58
- }
59
- _disposeTestedApp(testedApp) {
60
- return testedApp ? testedApp.kill().catch(e => DEBUG_LOGGER(e)) : Promise.resolve();
61
- }
62
- async _disposeTaskAndRelatedAssets(task, browserSet, reporters, testedApp) {
63
- task.abort();
64
- task.unRegisterClientScriptRouting();
65
- task.clearListeners();
66
- await this._disposeAssets(browserSet, reporters, testedApp);
67
- }
68
- _disposeAssets(browserSet, reporters, testedApp) {
69
- return Promise.all([
70
- this._disposeBrowserSet(browserSet),
71
- this._disposeReporters(reporters),
72
- this._disposeTestedApp(testedApp)
73
- ]);
74
- }
75
- _prepareArrayParameter(array) {
76
- array = lodash_1.flattenDeep(array);
77
- if (this.isCli)
78
- return array.length === 0 ? void 0 : array;
79
- return array;
80
- }
81
- _createCancelablePromise(taskPromise) {
82
- const promise = taskPromise.then(({ completionPromise }) => completionPromise);
83
- const removeFromPending = () => lodash_1.pull(this.pendingTaskPromises, promise);
84
- promise
85
- .then(removeFromPending)
86
- .catch(removeFromPending);
87
- promise.cancel = () => taskPromise
88
- .then(({ cancelTask }) => cancelTask())
89
- .then(removeFromPending);
90
- this.pendingTaskPromises.push(promise);
91
- return promise;
92
- }
93
- // Run task
94
- _getFailedTestCount(task, reporter) {
95
- let failedTestCount = reporter.testCount - reporter.passed;
96
- if (task.opts.stopOnFirstFail && !!failedTestCount)
97
- failedTestCount = 1;
98
- return failedTestCount;
99
- }
100
- async _getTaskResult(task, browserSet, reporters, testedApp) {
101
- if (!task.opts.live) {
102
- task.on('browser-job-done', job => {
103
- job.browserConnections.forEach(bc => browserSet.releaseConnection(bc));
104
- });
105
- }
106
- const browserSetErrorPromise = promisify_event_1.default(browserSet, 'error');
107
- const taskErrorPromise = promisify_event_1.default(task, 'error');
108
- const streamController = new reporter_stream_controller_1.default(task, reporters);
109
- const taskDonePromise = task.once('done')
110
- .then(() => browserSetErrorPromise.cancel())
111
- .then(() => {
112
- return Promise.all(reporters.map(reporter => reporter.pendingTaskDonePromise));
113
- });
114
- const promises = [
115
- taskDonePromise,
116
- browserSetErrorPromise,
117
- taskErrorPromise
118
- ];
119
- if (testedApp)
120
- promises.push(testedApp.errorPromise);
121
- try {
122
- await Promise.race(promises);
123
- }
124
- catch (err) {
125
- await this._disposeTaskAndRelatedAssets(task, browserSet, reporters, testedApp);
126
- throw err;
127
- }
128
- await this._disposeAssets(browserSet, reporters, testedApp);
129
- if (streamController.multipleStreamError)
130
- throw streamController.multipleStreamError;
131
- return this._getFailedTestCount(task, reporters[0]);
132
- }
133
- _createTask(tests, browserConnectionGroups, proxy, opts, warningLog) {
134
- return new task_1.default({
135
- tests,
136
- browserConnectionGroups,
137
- proxy,
138
- opts,
139
- runnerWarningLog: warningLog,
140
- compilerService: this.compilerService
141
- });
142
- }
143
- _runTask({ reporterPlugins, browserSet, tests, testedApp, options }) {
144
- const task = this._createTask(tests, browserSet.browserConnectionGroups, this.proxy, options, this.warningLog);
145
- const reporters = reporterPlugins.map(reporter => new reporter_1.default(reporter.plugin, task, reporter.outStream, reporter.name));
146
- const completionPromise = this._getTaskResult(task, browserSet, reporters, testedApp);
147
- let completed = false;
148
- task.on('start', handle_errors_1.startHandlingTestErrors);
149
- if (!this.configuration.getOption(option_names_1.default.skipUncaughtErrors)) {
150
- task.on('test-run-start', handle_errors_1.addRunningTest);
151
- task.on('test-run-done', handle_errors_1.removeRunningTest);
152
- }
153
- task.on('done', handle_errors_1.stopHandlingTestErrors);
154
- task.on('error', handle_errors_1.stopHandlingTestErrors);
155
- const onTaskCompleted = () => {
156
- task.unRegisterClientScriptRouting();
157
- completed = true;
158
- };
159
- completionPromise
160
- .then(onTaskCompleted)
161
- .catch(onTaskCompleted);
162
- const cancelTask = async () => {
163
- if (!completed)
164
- await this._disposeTaskAndRelatedAssets(task, browserSet, reporters, testedApp);
165
- };
166
- return { completionPromise, cancelTask };
167
- }
168
- _registerAssets(assets) {
169
- assets.forEach(asset => this.proxy.GET(asset.path, asset.info));
170
- }
171
- _validateDebugLogger() {
172
- const debugLogger = this.configuration.getOption(option_names_1.default.debugLogger);
173
- const debugLoggerDefinedCorrectly = debugLogger === null || !!debugLogger &&
174
- ['showBreakpoint', 'hideBreakpoint'].every(method => method in debugLogger && lodash_1.isFunction(debugLogger[method]));
175
- if (!debugLoggerDefinedCorrectly) {
176
- this.configuration.mergeOptions({
177
- [option_names_1.default.debugLogger]: debug_logger_1.default
178
- });
179
- }
180
- }
181
- _validateSpeedOption() {
182
- const speed = this.configuration.getOption(option_names_1.default.speed);
183
- if (speed === void 0)
184
- return;
185
- if (typeof speed !== 'number' || isNaN(speed) || speed < 0.01 || speed > 1)
186
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.invalidSpeedValue);
187
- }
188
- _validateConcurrencyOption() {
189
- const concurrency = this.configuration.getOption(option_names_1.default.concurrency);
190
- if (concurrency === void 0)
191
- return;
192
- if (typeof concurrency !== 'number' || isNaN(concurrency) || concurrency < 1)
193
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.invalidConcurrencyFactor);
194
- }
195
- _validateRequestTimeoutOption(optionName) {
196
- const requestTimeout = this.configuration.getOption(optionName);
197
- if (requestTimeout === void 0)
198
- return;
199
- type_assertions_1.assertType(type_assertions_1.is.nonNegativeNumber, null, `"${optionName}" option`, requestTimeout);
200
- }
201
- _validateProxyBypassOption() {
202
- let proxyBypass = this.configuration.getOption(option_names_1.default.proxyBypass);
203
- if (proxyBypass === void 0)
204
- return;
205
- type_assertions_1.assertType([type_assertions_1.is.string, type_assertions_1.is.array], null, '"proxyBypass" argument', proxyBypass);
206
- if (typeof proxyBypass === 'string')
207
- proxyBypass = [proxyBypass];
208
- proxyBypass = proxyBypass.reduce((arr, rules) => {
209
- type_assertions_1.assertType(type_assertions_1.is.string, null, '"proxyBypass" argument', rules);
210
- return arr.concat(rules.split(','));
211
- }, []);
212
- this.configuration.mergeOptions({ proxyBypass });
213
- }
214
- _getScreenshotOptions() {
215
- let { path, pathPattern } = this.configuration.getOption(option_names_1.default.screenshots) || {};
216
- if (!path)
217
- path = this.configuration.getOption(option_names_1.default.screenshotPath);
218
- if (!pathPattern)
219
- pathPattern = this.configuration.getOption(option_names_1.default.screenshotPathPattern);
220
- return { path, pathPattern };
221
- }
222
- _validateScreenshotOptions() {
223
- const { path, pathPattern } = this._getScreenshotOptions();
224
- const disableScreenshots = this.configuration.getOption(option_names_1.default.disableScreenshots) || !path;
225
- this.configuration.mergeOptions({ [option_names_1.default.disableScreenshots]: disableScreenshots });
226
- if (disableScreenshots)
227
- return;
228
- if (path) {
229
- this._validateScreenshotPath(path, 'screenshots base directory path');
230
- this.configuration.mergeOptions({ [option_names_1.default.screenshots]: { path: path_1.resolve(path) } });
231
- }
232
- if (pathPattern) {
233
- this._validateScreenshotPath(pathPattern, 'screenshots path pattern');
234
- this.configuration.mergeOptions({ [option_names_1.default.screenshots]: { pathPattern } });
235
- }
236
- }
237
- async _validateVideoOptions() {
238
- const videoPath = this.configuration.getOption(option_names_1.default.videoPath);
239
- const videoEncodingOptions = this.configuration.getOption(option_names_1.default.videoEncodingOptions);
240
- let videoOptions = this.configuration.getOption(option_names_1.default.videoOptions);
241
- if (!videoPath) {
242
- if (videoOptions || videoEncodingOptions)
243
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotSetVideoOptionsWithoutBaseVideoPathSpecified);
244
- return;
245
- }
246
- this.configuration.mergeOptions({ [option_names_1.default.videoPath]: path_1.resolve(videoPath) });
247
- if (!videoOptions) {
248
- videoOptions = {};
249
- this.configuration.mergeOptions({ [option_names_1.default.videoOptions]: videoOptions });
250
- }
251
- if (videoOptions.ffmpegPath)
252
- videoOptions.ffmpegPath = path_1.resolve(videoOptions.ffmpegPath);
253
- else
254
- videoOptions.ffmpegPath = await detect_ffmpeg_1.default();
255
- if (!videoOptions.ffmpegPath)
256
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotFindFFMPEG);
257
- }
258
- _validateCompilerOptions() {
259
- const compilerOptions = this.configuration.getOption(option_names_1.default.compilerOptions);
260
- if (!compilerOptions)
261
- return;
262
- const specifiedCompilers = Object.keys(compilerOptions);
263
- const customizedCompilers = Object.keys(customizable_compilers_1.default);
264
- const wrongCompilers = specifiedCompilers.filter(compiler => !customizedCompilers.includes(compiler));
265
- if (!wrongCompilers.length)
266
- return;
267
- const compilerListStr = string_1.getConcatenatedValuesString(wrongCompilers, void 0, "'");
268
- const pluralSuffix = string_1.getPluralSuffix(wrongCompilers);
269
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotCustomizeSpecifiedCompilers, compilerListStr, pluralSuffix);
270
- }
271
- _validateRetryTestPagesOption() {
272
- const retryTestPagesOption = this.configuration.getOption(option_names_1.default.retryTestPages);
273
- if (!retryTestPagesOption)
274
- return;
275
- const ssl = this.configuration.getOption(option_names_1.default.ssl);
276
- if (ssl)
277
- return;
278
- const hostname = this.configuration.getOption(option_names_1.default.hostname);
279
- if (is_localhost_1.default(hostname))
280
- return;
281
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotEnableRetryTestPagesOption);
282
- }
283
- async _validateRunOptions() {
284
- this._validateDebugLogger();
285
- this._validateScreenshotOptions();
286
- await this._validateVideoOptions();
287
- this._validateSpeedOption();
288
- this._validateConcurrencyOption();
289
- this._validateProxyBypassOption();
290
- this._validateCompilerOptions();
291
- this._validateRetryTestPagesOption();
292
- this._validateRequestTimeoutOption(option_names_1.default.pageRequestTimeout);
293
- this._validateRequestTimeoutOption(option_names_1.default.ajaxRequestTimeout);
294
- }
295
- _createRunnableConfiguration() {
296
- return this.bootstrapper
297
- .createRunnableConfiguration()
298
- .then(runnableConfiguration => {
299
- this.emit('done-bootstrapping');
300
- return runnableConfiguration;
301
- });
302
- }
303
- _validateScreenshotPath(screenshotPath, pathType) {
304
- const forbiddenCharsList = check_file_path_1.default(screenshotPath);
305
- if (forbiddenCharsList.length)
306
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.forbiddenCharatersInScreenshotPath, screenshotPath, pathType, utils_1.renderForbiddenCharsList(forbiddenCharsList));
307
- }
308
- _setBootstrapperOptions() {
309
- this.configuration.prepare();
310
- this.configuration.notifyAboutOverriddenOptions();
311
- this.configuration.notifyAboutDeprecatedOptions(this.warningLog);
312
- this.bootstrapper.sources = this.configuration.getOption(option_names_1.default.src) || this.bootstrapper.sources;
313
- this.bootstrapper.browsers = this.configuration.getOption(option_names_1.default.browsers) || this.bootstrapper.browsers;
314
- this.bootstrapper.concurrency = this.configuration.getOption(option_names_1.default.concurrency);
315
- this.bootstrapper.appCommand = this.configuration.getOption(option_names_1.default.appCommand) || this.bootstrapper.appCommand;
316
- this.bootstrapper.appInitDelay = this.configuration.getOption(option_names_1.default.appInitDelay);
317
- this.bootstrapper.filter = this.configuration.getOption(option_names_1.default.filter) || this.bootstrapper.filter;
318
- this.bootstrapper.reporters = this.configuration.getOption(option_names_1.default.reporter) || this.bootstrapper.reporters;
319
- this.bootstrapper.tsConfigPath = this.configuration.getOption(option_names_1.default.tsConfigPath);
320
- this.bootstrapper.clientScripts = this.configuration.getOption(option_names_1.default.clientScripts) || this.bootstrapper.clientScripts;
321
- this.bootstrapper.disableMultipleWindows = this.configuration.getOption(option_names_1.default.disableMultipleWindows);
322
- this.bootstrapper.compilerOptions = this.configuration.getOption(option_names_1.default.compilerOptions);
323
- this.bootstrapper.browserInitTimeout = this.configuration.getOption(option_names_1.default.browserInitTimeout);
324
- }
325
- async _prepareClientScripts(tests, clientScripts) {
326
- return Promise.all(tests.map(async (test) => {
327
- if (test.isLegacy)
328
- return;
329
- let loadedTestClientScripts = await load_1.default(test.clientScripts, path_1.dirname(test.testFile.filename));
330
- loadedTestClientScripts = clientScripts.concat(loadedTestClientScripts);
331
- test.clientScripts = utils_2.setUniqueUrls(loadedTestClientScripts);
332
- }));
333
- }
334
- // API
335
- embeddingOptions(opts) {
336
- const { assets, TestRunCtor } = opts;
337
- this._registerAssets(assets);
338
- this.configuration.mergeOptions({ TestRunCtor });
339
- return this;
340
- }
341
- src(...sources) {
342
- if (this.apiMethodWasCalled.src)
343
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.multipleAPIMethodCallForbidden, option_names_1.default.src);
344
- sources = this._prepareArrayParameter(sources);
345
- this.configuration.mergeOptions({ [option_names_1.default.src]: sources });
346
- this.apiMethodWasCalled.src = true;
347
- return this;
348
- }
349
- browsers(...browsers) {
350
- if (this.apiMethodWasCalled.browsers)
351
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.multipleAPIMethodCallForbidden, option_names_1.default.browsers);
352
- browsers = this._prepareArrayParameter(browsers);
353
- this.configuration.mergeOptions({ browsers });
354
- this.apiMethodWasCalled.browsers = true;
355
- return this;
356
- }
357
- concurrency(concurrency) {
358
- this.configuration.mergeOptions({ concurrency });
359
- return this;
360
- }
361
- reporter(name, output) {
362
- if (this.apiMethodWasCalled.reporter)
363
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.multipleAPIMethodCallForbidden, option_names_1.default.reporter);
364
- let reporters = prepare_reporters_1.default(name, output);
365
- reporters = this._prepareArrayParameter(reporters);
366
- this.configuration.mergeOptions({ [option_names_1.default.reporter]: reporters });
367
- this.apiMethodWasCalled.reporter = true;
368
- return this;
369
- }
370
- filter(filter) {
371
- this.configuration.mergeOptions({ filter });
372
- return this;
373
- }
374
- useProxy(proxy, proxyBypass) {
375
- this.configuration.mergeOptions({ proxy, proxyBypass });
376
- return this;
377
- }
378
- screenshots(...options) {
379
- let fullPage;
380
- let [path, takeOnFails, pathPattern] = options;
381
- if (options.length === 1 && options[0] && typeof options[0] === 'object')
382
- ({ path, takeOnFails, pathPattern, fullPage } = options[0]);
383
- this.configuration.mergeOptions({ screenshots: { path, takeOnFails, pathPattern, fullPage } });
384
- return this;
385
- }
386
- video(path, options, encodingOptions) {
387
- this.configuration.mergeOptions({
388
- [option_names_1.default.videoPath]: path,
389
- [option_names_1.default.videoOptions]: options,
390
- [option_names_1.default.videoEncodingOptions]: encodingOptions
391
- });
392
- return this;
393
- }
394
- startApp(command, initDelay) {
395
- this.configuration.mergeOptions({
396
- [option_names_1.default.appCommand]: command,
397
- [option_names_1.default.appInitDelay]: initDelay
398
- });
399
- return this;
400
- }
401
- tsConfigPath(path) {
402
- this.configuration.mergeOptions({
403
- [option_names_1.default.tsConfigPath]: path
404
- });
405
- return this;
406
- }
407
- clientScripts(...scripts) {
408
- if (this.apiMethodWasCalled.clientScripts)
409
- throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.multipleAPIMethodCallForbidden, option_names_1.default.clientScripts);
410
- scripts = this._prepareArrayParameter(scripts);
411
- this.configuration.mergeOptions({ [option_names_1.default.clientScripts]: scripts });
412
- this.apiMethodWasCalled.clientScripts = true;
413
- return this;
414
- }
415
- compilerOptions(opts) {
416
- this.configuration.mergeOptions({
417
- [option_names_1.default.compilerOptions]: opts
418
- });
419
- return this;
420
- }
421
- run(options = {}) {
422
- this.apiMethodWasCalled.reset();
423
- this.configuration.mergeOptions(options);
424
- this._setBootstrapperOptions();
425
- const runTaskPromise = Promise.resolve()
426
- .then(() => this._validateRunOptions())
427
- .then(() => this._createRunnableConfiguration())
428
- .then(async ({ reporterPlugins, browserSet, tests, testedApp, commonClientScripts }) => {
429
- var _a;
430
- await this._prepareClientScripts(tests, commonClientScripts);
431
- const resultOptions = this.configuration.getOptions();
432
- await ((_a = this.bootstrapper.compilerService) === null || _a === void 0 ? void 0 : _a.setOptions({ value: resultOptions }));
433
- return this._runTask({ reporterPlugins, browserSet, tests, testedApp, options: resultOptions });
434
- });
435
- return this._createCancelablePromise(runTaskPromise);
436
- }
437
- async stop() {
438
- // NOTE: When taskPromise is cancelled, it is removed from
439
- // the pendingTaskPromises array, which leads to shifting indexes
440
- // towards the beginning. So, we must copy the array in order to iterate it,
441
- // or we can perform iteration from the end to the beginning.
442
- const cancellationPromises = this.pendingTaskPromises.reduceRight((result, taskPromise) => {
443
- result.push(taskPromise.cancel());
444
- return result;
445
- }, []);
446
- await Promise.all(cancellationPromises);
447
- }
448
- }
449
- exports.default = Runner;
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 path_1 = require("path");
7
+ const debug_1 = __importDefault(require("debug"));
8
+ const promisify_event_1 = __importDefault(require("promisify-event"));
9
+ const events_1 = require("events");
10
+ const lodash_1 = require("lodash");
11
+ const bootstrapper_1 = __importDefault(require("./bootstrapper"));
12
+ const reporter_1 = __importDefault(require("../reporter"));
13
+ const task_1 = __importDefault(require("./task"));
14
+ const debug_logger_1 = __importDefault(require("../notifications/debug-logger"));
15
+ const runtime_1 = require("../errors/runtime");
16
+ const types_1 = require("../errors/types");
17
+ const type_assertions_1 = require("../errors/runtime/type-assertions");
18
+ const utils_1 = require("../errors/test-run/utils");
19
+ const detect_ffmpeg_1 = __importDefault(require("../utils/detect-ffmpeg"));
20
+ const check_file_path_1 = __importDefault(require("../utils/check-file-path"));
21
+ const handle_errors_1 = require("../utils/handle-errors");
22
+ const option_names_1 = __importDefault(require("../configuration/option-names"));
23
+ const flag_list_1 = __importDefault(require("../utils/flag-list"));
24
+ const prepare_reporters_1 = __importDefault(require("../utils/prepare-reporters"));
25
+ const load_1 = __importDefault(require("../custom-client-scripts/load"));
26
+ const utils_2 = require("../custom-client-scripts/utils");
27
+ const reporter_stream_controller_1 = __importDefault(require("./reporter-stream-controller"));
28
+ const customizable_compilers_1 = __importDefault(require("../configuration/customizable-compilers"));
29
+ const string_1 = require("../utils/string");
30
+ const is_localhost_1 = __importDefault(require("../utils/is-localhost"));
31
+ const warning_log_1 = __importDefault(require("../notifications/warning-log"));
32
+ const DEBUG_LOGGER = debug_1.default('testcafe:runner');
33
+ class Runner extends events_1.EventEmitter {
34
+ constructor({ proxy, browserConnectionGateway, configuration, compilerService }) {
35
+ super();
36
+ this.proxy = proxy;
37
+ this.bootstrapper = this._createBootstrapper(browserConnectionGateway, compilerService);
38
+ this.pendingTaskPromises = [];
39
+ this.configuration = configuration;
40
+ this.isCli = false;
41
+ this.warningLog = new warning_log_1.default();
42
+ this.compilerService = compilerService;
43
+ this.apiMethodWasCalled = new flag_list_1.default([
44
+ option_names_1.default.src,
45
+ option_names_1.default.browsers,
46
+ option_names_1.default.reporter,
47
+ option_names_1.default.clientScripts
48
+ ]);
49
+ }
50
+ _createBootstrapper(browserConnectionGateway, compilerService) {
51
+ return new bootstrapper_1.default({ browserConnectionGateway, compilerService });
52
+ }
53
+ _disposeBrowserSet(browserSet) {
54
+ return browserSet.dispose().catch(e => DEBUG_LOGGER(e));
55
+ }
56
+ _disposeReporters(reporters) {
57
+ return Promise.all(reporters.map(reporter => reporter.dispose().catch(e => DEBUG_LOGGER(e))));
58
+ }
59
+ _disposeTestedApp(testedApp) {
60
+ return testedApp ? testedApp.kill().catch(e => DEBUG_LOGGER(e)) : Promise.resolve();
61
+ }
62
+ async _disposeTaskAndRelatedAssets(task, browserSet, reporters, testedApp) {
63
+ task.abort();
64
+ task.unRegisterClientScriptRouting();
65
+ task.clearListeners();
66
+ await this._disposeAssets(browserSet, reporters, testedApp);
67
+ }
68
+ _disposeAssets(browserSet, reporters, testedApp) {
69
+ return Promise.all([
70
+ this._disposeBrowserSet(browserSet),
71
+ this._disposeReporters(reporters),
72
+ this._disposeTestedApp(testedApp)
73
+ ]);
74
+ }
75
+ _prepareArrayParameter(array) {
76
+ array = lodash_1.flattenDeep(array);
77
+ if (this.isCli)
78
+ return array.length === 0 ? void 0 : array;
79
+ return array;
80
+ }
81
+ _createCancelablePromise(taskPromise) {
82
+ const promise = taskPromise.then(({ completionPromise }) => completionPromise);
83
+ const removeFromPending = () => lodash_1.pull(this.pendingTaskPromises, promise);
84
+ promise
85
+ .then(removeFromPending)
86
+ .catch(removeFromPending);
87
+ promise.cancel = () => taskPromise
88
+ .then(({ cancelTask }) => cancelTask())
89
+ .then(removeFromPending);
90
+ this.pendingTaskPromises.push(promise);
91
+ return promise;
92
+ }
93
+ // Run task
94
+ _getFailedTestCount(task, reporter) {
95
+ let failedTestCount = reporter.testCount - reporter.passed;
96
+ if (task.opts.stopOnFirstFail && !!failedTestCount)
97
+ failedTestCount = 1;
98
+ return failedTestCount;
99
+ }
100
+ async _getTaskResult(task, browserSet, reporters, testedApp) {
101
+ if (!task.opts.live) {
102
+ task.on('browser-job-done', job => {
103
+ job.browserConnections.forEach(bc => browserSet.releaseConnection(bc));
104
+ });
105
+ }
106
+ const browserSetErrorPromise = promisify_event_1.default(browserSet, 'error');
107
+ const taskErrorPromise = promisify_event_1.default(task, 'error');
108
+ const streamController = new reporter_stream_controller_1.default(task, reporters);
109
+ const taskDonePromise = task.once('done')
110
+ .then(() => browserSetErrorPromise.cancel())
111
+ .then(() => {
112
+ return Promise.all(reporters.map(reporter => reporter.pendingTaskDonePromise));
113
+ });
114
+ const promises = [
115
+ taskDonePromise,
116
+ browserSetErrorPromise,
117
+ taskErrorPromise
118
+ ];
119
+ if (testedApp)
120
+ promises.push(testedApp.errorPromise);
121
+ try {
122
+ await Promise.race(promises);
123
+ }
124
+ catch (err) {
125
+ await this._disposeTaskAndRelatedAssets(task, browserSet, reporters, testedApp);
126
+ throw err;
127
+ }
128
+ await this._disposeAssets(browserSet, reporters, testedApp);
129
+ if (streamController.multipleStreamError)
130
+ throw streamController.multipleStreamError;
131
+ return this._getFailedTestCount(task, reporters[0]);
132
+ }
133
+ _createTask(tests, browserConnectionGroups, proxy, opts, warningLog) {
134
+ return new task_1.default({
135
+ tests,
136
+ browserConnectionGroups,
137
+ proxy,
138
+ opts,
139
+ runnerWarningLog: warningLog,
140
+ compilerService: this.compilerService
141
+ });
142
+ }
143
+ _runTask({ reporterPlugins, browserSet, tests, testedApp, options }) {
144
+ const task = this._createTask(tests, browserSet.browserConnectionGroups, this.proxy, options, this.warningLog);
145
+ const reporters = reporterPlugins.map(reporter => new reporter_1.default(reporter.plugin, task, reporter.outStream, reporter.name));
146
+ const completionPromise = this._getTaskResult(task, browserSet, reporters, testedApp);
147
+ let completed = false;
148
+ task.on('start', handle_errors_1.startHandlingTestErrors);
149
+ if (!this.configuration.getOption(option_names_1.default.skipUncaughtErrors)) {
150
+ task.on('test-run-start', handle_errors_1.addRunningTest);
151
+ task.on('test-run-done', handle_errors_1.removeRunningTest);
152
+ }
153
+ task.on('done', handle_errors_1.stopHandlingTestErrors);
154
+ task.on('error', handle_errors_1.stopHandlingTestErrors);
155
+ const onTaskCompleted = () => {
156
+ task.unRegisterClientScriptRouting();
157
+ completed = true;
158
+ };
159
+ completionPromise
160
+ .then(onTaskCompleted)
161
+ .catch(onTaskCompleted);
162
+ const cancelTask = async () => {
163
+ if (!completed)
164
+ await this._disposeTaskAndRelatedAssets(task, browserSet, reporters, testedApp);
165
+ };
166
+ return { completionPromise, cancelTask };
167
+ }
168
+ _registerAssets(assets) {
169
+ assets.forEach(asset => this.proxy.GET(asset.path, asset.info));
170
+ }
171
+ _validateDebugLogger() {
172
+ const debugLogger = this.configuration.getOption(option_names_1.default.debugLogger);
173
+ const debugLoggerDefinedCorrectly = debugLogger === null || !!debugLogger &&
174
+ ['showBreakpoint', 'hideBreakpoint'].every(method => method in debugLogger && lodash_1.isFunction(debugLogger[method]));
175
+ if (!debugLoggerDefinedCorrectly) {
176
+ this.configuration.mergeOptions({
177
+ [option_names_1.default.debugLogger]: debug_logger_1.default
178
+ });
179
+ }
180
+ }
181
+ _validateSpeedOption() {
182
+ const speed = this.configuration.getOption(option_names_1.default.speed);
183
+ if (speed === void 0)
184
+ return;
185
+ if (typeof speed !== 'number' || isNaN(speed) || speed < 0.01 || speed > 1)
186
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.invalidSpeedValue);
187
+ }
188
+ _validateConcurrencyOption() {
189
+ const concurrency = this.configuration.getOption(option_names_1.default.concurrency);
190
+ if (concurrency === void 0)
191
+ return;
192
+ if (typeof concurrency !== 'number' || isNaN(concurrency) || concurrency < 1)
193
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.invalidConcurrencyFactor);
194
+ }
195
+ _validateRequestTimeoutOption(optionName) {
196
+ const requestTimeout = this.configuration.getOption(optionName);
197
+ if (requestTimeout === void 0)
198
+ return;
199
+ type_assertions_1.assertType(type_assertions_1.is.nonNegativeNumber, null, `"${optionName}" option`, requestTimeout);
200
+ }
201
+ _validateProxyBypassOption() {
202
+ let proxyBypass = this.configuration.getOption(option_names_1.default.proxyBypass);
203
+ if (proxyBypass === void 0)
204
+ return;
205
+ type_assertions_1.assertType([type_assertions_1.is.string, type_assertions_1.is.array], null, '"proxyBypass" argument', proxyBypass);
206
+ if (typeof proxyBypass === 'string')
207
+ proxyBypass = [proxyBypass];
208
+ proxyBypass = proxyBypass.reduce((arr, rules) => {
209
+ type_assertions_1.assertType(type_assertions_1.is.string, null, '"proxyBypass" argument', rules);
210
+ return arr.concat(rules.split(','));
211
+ }, []);
212
+ this.configuration.mergeOptions({ proxyBypass });
213
+ }
214
+ _getScreenshotOptions() {
215
+ let { path, pathPattern } = this.configuration.getOption(option_names_1.default.screenshots) || {};
216
+ if (!path)
217
+ path = this.configuration.getOption(option_names_1.default.screenshotPath);
218
+ if (!pathPattern)
219
+ pathPattern = this.configuration.getOption(option_names_1.default.screenshotPathPattern);
220
+ return { path, pathPattern };
221
+ }
222
+ _validateScreenshotOptions() {
223
+ const { path, pathPattern } = this._getScreenshotOptions();
224
+ const disableScreenshots = this.configuration.getOption(option_names_1.default.disableScreenshots) || !path;
225
+ this.configuration.mergeOptions({ [option_names_1.default.disableScreenshots]: disableScreenshots });
226
+ if (disableScreenshots)
227
+ return;
228
+ if (path) {
229
+ this._validateScreenshotPath(path, 'screenshots base directory path');
230
+ this.configuration.mergeOptions({ [option_names_1.default.screenshots]: { path: path_1.resolve(path) } });
231
+ }
232
+ if (pathPattern) {
233
+ this._validateScreenshotPath(pathPattern, 'screenshots path pattern');
234
+ this.configuration.mergeOptions({ [option_names_1.default.screenshots]: { pathPattern } });
235
+ }
236
+ }
237
+ async _validateVideoOptions() {
238
+ const videoPath = this.configuration.getOption(option_names_1.default.videoPath);
239
+ const videoEncodingOptions = this.configuration.getOption(option_names_1.default.videoEncodingOptions);
240
+ let videoOptions = this.configuration.getOption(option_names_1.default.videoOptions);
241
+ if (!videoPath) {
242
+ if (videoOptions || videoEncodingOptions)
243
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotSetVideoOptionsWithoutBaseVideoPathSpecified);
244
+ return;
245
+ }
246
+ this.configuration.mergeOptions({ [option_names_1.default.videoPath]: path_1.resolve(videoPath) });
247
+ if (!videoOptions) {
248
+ videoOptions = {};
249
+ this.configuration.mergeOptions({ [option_names_1.default.videoOptions]: videoOptions });
250
+ }
251
+ if (videoOptions.ffmpegPath)
252
+ videoOptions.ffmpegPath = path_1.resolve(videoOptions.ffmpegPath);
253
+ else
254
+ videoOptions.ffmpegPath = await detect_ffmpeg_1.default();
255
+ if (!videoOptions.ffmpegPath)
256
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotFindFFMPEG);
257
+ }
258
+ _validateCompilerOptions() {
259
+ const compilerOptions = this.configuration.getOption(option_names_1.default.compilerOptions);
260
+ if (!compilerOptions)
261
+ return;
262
+ const specifiedCompilers = Object.keys(compilerOptions);
263
+ const customizedCompilers = Object.keys(customizable_compilers_1.default);
264
+ const wrongCompilers = specifiedCompilers.filter(compiler => !customizedCompilers.includes(compiler));
265
+ if (!wrongCompilers.length)
266
+ return;
267
+ const compilerListStr = string_1.getConcatenatedValuesString(wrongCompilers, void 0, "'");
268
+ const pluralSuffix = string_1.getPluralSuffix(wrongCompilers);
269
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotCustomizeSpecifiedCompilers, compilerListStr, pluralSuffix);
270
+ }
271
+ _validateRetryTestPagesOption() {
272
+ const retryTestPagesOption = this.configuration.getOption(option_names_1.default.retryTestPages);
273
+ if (!retryTestPagesOption)
274
+ return;
275
+ const ssl = this.configuration.getOption(option_names_1.default.ssl);
276
+ if (ssl)
277
+ return;
278
+ const hostname = this.configuration.getOption(option_names_1.default.hostname);
279
+ if (is_localhost_1.default(hostname))
280
+ return;
281
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotEnableRetryTestPagesOption);
282
+ }
283
+ async _validateRunOptions() {
284
+ this._validateDebugLogger();
285
+ this._validateScreenshotOptions();
286
+ await this._validateVideoOptions();
287
+ this._validateSpeedOption();
288
+ this._validateConcurrencyOption();
289
+ this._validateProxyBypassOption();
290
+ this._validateCompilerOptions();
291
+ this._validateRetryTestPagesOption();
292
+ this._validateRequestTimeoutOption(option_names_1.default.pageRequestTimeout);
293
+ this._validateRequestTimeoutOption(option_names_1.default.ajaxRequestTimeout);
294
+ }
295
+ _createRunnableConfiguration() {
296
+ return this.bootstrapper
297
+ .createRunnableConfiguration()
298
+ .then(runnableConfiguration => {
299
+ this.emit('done-bootstrapping');
300
+ return runnableConfiguration;
301
+ });
302
+ }
303
+ _validateScreenshotPath(screenshotPath, pathType) {
304
+ const forbiddenCharsList = check_file_path_1.default(screenshotPath);
305
+ if (forbiddenCharsList.length)
306
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.forbiddenCharatersInScreenshotPath, screenshotPath, pathType, utils_1.renderForbiddenCharsList(forbiddenCharsList));
307
+ }
308
+ _setBootstrapperOptions() {
309
+ this.configuration.prepare();
310
+ this.configuration.notifyAboutOverriddenOptions();
311
+ this.configuration.notifyAboutDeprecatedOptions(this.warningLog);
312
+ this.bootstrapper.sources = this.configuration.getOption(option_names_1.default.src) || this.bootstrapper.sources;
313
+ this.bootstrapper.browsers = this.configuration.getOption(option_names_1.default.browsers) || this.bootstrapper.browsers;
314
+ this.bootstrapper.concurrency = this.configuration.getOption(option_names_1.default.concurrency);
315
+ this.bootstrapper.appCommand = this.configuration.getOption(option_names_1.default.appCommand) || this.bootstrapper.appCommand;
316
+ this.bootstrapper.appInitDelay = this.configuration.getOption(option_names_1.default.appInitDelay);
317
+ this.bootstrapper.filter = this.configuration.getOption(option_names_1.default.filter) || this.bootstrapper.filter;
318
+ this.bootstrapper.reporters = this.configuration.getOption(option_names_1.default.reporter) || this.bootstrapper.reporters;
319
+ this.bootstrapper.tsConfigPath = this.configuration.getOption(option_names_1.default.tsConfigPath);
320
+ this.bootstrapper.clientScripts = this.configuration.getOption(option_names_1.default.clientScripts) || this.bootstrapper.clientScripts;
321
+ this.bootstrapper.disableMultipleWindows = this.configuration.getOption(option_names_1.default.disableMultipleWindows);
322
+ this.bootstrapper.compilerOptions = this.configuration.getOption(option_names_1.default.compilerOptions);
323
+ this.bootstrapper.browserInitTimeout = this.configuration.getOption(option_names_1.default.browserInitTimeout);
324
+ }
325
+ async _prepareClientScripts(tests, clientScripts) {
326
+ return Promise.all(tests.map(async (test) => {
327
+ if (test.isLegacy)
328
+ return;
329
+ let loadedTestClientScripts = await load_1.default(test.clientScripts, path_1.dirname(test.testFile.filename));
330
+ loadedTestClientScripts = clientScripts.concat(loadedTestClientScripts);
331
+ test.clientScripts = utils_2.setUniqueUrls(loadedTestClientScripts);
332
+ }));
333
+ }
334
+ // API
335
+ embeddingOptions(opts) {
336
+ const { assets, TestRunCtor } = opts;
337
+ this._registerAssets(assets);
338
+ this.configuration.mergeOptions({ TestRunCtor });
339
+ return this;
340
+ }
341
+ src(...sources) {
342
+ if (this.apiMethodWasCalled.src)
343
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.multipleAPIMethodCallForbidden, option_names_1.default.src);
344
+ sources = this._prepareArrayParameter(sources);
345
+ this.configuration.mergeOptions({ [option_names_1.default.src]: sources });
346
+ this.apiMethodWasCalled.src = true;
347
+ return this;
348
+ }
349
+ browsers(...browsers) {
350
+ if (this.apiMethodWasCalled.browsers)
351
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.multipleAPIMethodCallForbidden, option_names_1.default.browsers);
352
+ browsers = this._prepareArrayParameter(browsers);
353
+ this.configuration.mergeOptions({ browsers });
354
+ this.apiMethodWasCalled.browsers = true;
355
+ return this;
356
+ }
357
+ concurrency(concurrency) {
358
+ this.configuration.mergeOptions({ concurrency });
359
+ return this;
360
+ }
361
+ reporter(name, output) {
362
+ if (this.apiMethodWasCalled.reporter)
363
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.multipleAPIMethodCallForbidden, option_names_1.default.reporter);
364
+ let reporters = prepare_reporters_1.default(name, output);
365
+ reporters = this._prepareArrayParameter(reporters);
366
+ this.configuration.mergeOptions({ [option_names_1.default.reporter]: reporters });
367
+ this.apiMethodWasCalled.reporter = true;
368
+ return this;
369
+ }
370
+ filter(filter) {
371
+ this.configuration.mergeOptions({ filter });
372
+ return this;
373
+ }
374
+ useProxy(proxy, proxyBypass) {
375
+ this.configuration.mergeOptions({ proxy, proxyBypass });
376
+ return this;
377
+ }
378
+ screenshots(...options) {
379
+ let fullPage;
380
+ let [path, takeOnFails, pathPattern] = options;
381
+ if (options.length === 1 && options[0] && typeof options[0] === 'object')
382
+ ({ path, takeOnFails, pathPattern, fullPage } = options[0]);
383
+ this.configuration.mergeOptions({ screenshots: { path, takeOnFails, pathPattern, fullPage } });
384
+ return this;
385
+ }
386
+ video(path, options, encodingOptions) {
387
+ this.configuration.mergeOptions({
388
+ [option_names_1.default.videoPath]: path,
389
+ [option_names_1.default.videoOptions]: options,
390
+ [option_names_1.default.videoEncodingOptions]: encodingOptions
391
+ });
392
+ return this;
393
+ }
394
+ startApp(command, initDelay) {
395
+ this.configuration.mergeOptions({
396
+ [option_names_1.default.appCommand]: command,
397
+ [option_names_1.default.appInitDelay]: initDelay
398
+ });
399
+ return this;
400
+ }
401
+ tsConfigPath(path) {
402
+ this.configuration.mergeOptions({
403
+ [option_names_1.default.tsConfigPath]: path
404
+ });
405
+ return this;
406
+ }
407
+ clientScripts(...scripts) {
408
+ if (this.apiMethodWasCalled.clientScripts)
409
+ throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.multipleAPIMethodCallForbidden, option_names_1.default.clientScripts);
410
+ scripts = this._prepareArrayParameter(scripts);
411
+ this.configuration.mergeOptions({ [option_names_1.default.clientScripts]: scripts });
412
+ this.apiMethodWasCalled.clientScripts = true;
413
+ return this;
414
+ }
415
+ compilerOptions(opts) {
416
+ this.configuration.mergeOptions({
417
+ [option_names_1.default.compilerOptions]: opts
418
+ });
419
+ return this;
420
+ }
421
+ run(options = {}) {
422
+ this.apiMethodWasCalled.reset();
423
+ this.configuration.mergeOptions(options);
424
+ this._setBootstrapperOptions();
425
+ const runTaskPromise = Promise.resolve()
426
+ .then(() => this._validateRunOptions())
427
+ .then(() => this._createRunnableConfiguration())
428
+ .then(async ({ reporterPlugins, browserSet, tests, testedApp, commonClientScripts }) => {
429
+ var _a;
430
+ await this._prepareClientScripts(tests, commonClientScripts);
431
+ const resultOptions = this.configuration.getOptions();
432
+ await ((_a = this.bootstrapper.compilerService) === null || _a === void 0 ? void 0 : _a.setOptions({ value: resultOptions }));
433
+ return this._runTask({ reporterPlugins, browserSet, tests, testedApp, options: resultOptions });
434
+ });
435
+ return this._createCancelablePromise(runTaskPromise);
436
+ }
437
+ async stop() {
438
+ // NOTE: When taskPromise is cancelled, it is removed from
439
+ // the pendingTaskPromises array, which leads to shifting indexes
440
+ // towards the beginning. So, we must copy the array in order to iterate it,
441
+ // or we can perform iteration from the end to the beginning.
442
+ const cancellationPromises = this.pendingTaskPromises.reduceRight((result, taskPromise) => {
443
+ result.push(taskPromise.cancel());
444
+ return result;
445
+ }, []);
446
+ await Promise.all(cancellationPromises);
447
+ }
448
+ }
449
+ exports.default = Runner;
450
450
  module.exports = exports.default;
451
451
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/runner/index.js"],"names":[],"mappings":"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","sourcesContent":["import { resolve as resolvePath, dirname } from 'path';\nimport debug from 'debug';\nimport promisifyEvent from 'promisify-event';\nimport { EventEmitter } from 'events';\nimport {\n    flattenDeep as flatten,\n    pull as remove,\n    isFunction\n} from 'lodash';\n\nimport Bootstrapper from './bootstrapper';\nimport Reporter from '../reporter';\nimport Task from './task';\nimport defaultDebugLogger from '../notifications/debug-logger';\nimport { GeneralError } from '../errors/runtime';\nimport { RUNTIME_ERRORS } from '../errors/types';\nimport { assertType, is } from '../errors/runtime/type-assertions';\nimport { renderForbiddenCharsList } from '../errors/test-run/utils';\nimport detectFFMPEG from '../utils/detect-ffmpeg';\nimport checkFilePath from '../utils/check-file-path';\nimport {\n    addRunningTest,\n    removeRunningTest,\n    startHandlingTestErrors,\n    stopHandlingTestErrors\n} from '../utils/handle-errors';\n\nimport OPTION_NAMES from '../configuration/option-names';\nimport FlagList from '../utils/flag-list';\nimport prepareReporters from '../utils/prepare-reporters';\nimport loadClientScripts from '../custom-client-scripts/load';\nimport { setUniqueUrls } from '../custom-client-scripts/utils';\nimport ReporterStreamController from './reporter-stream-controller';\nimport CustomizableCompilers from '../configuration/customizable-compilers';\nimport { getConcatenatedValuesString, getPluralSuffix } from '../utils/string';\nimport isLocalhost from '../utils/is-localhost';\nimport WarningLog from '../notifications/warning-log';\n\nconst DEBUG_LOGGER = debug('testcafe:runner');\n\nexport default class Runner extends EventEmitter {\n    constructor ({ proxy, browserConnectionGateway, configuration, compilerService }) {\n        super();\n\n        this.proxy               = proxy;\n        this.bootstrapper        = this._createBootstrapper(browserConnectionGateway, compilerService);\n        this.pendingTaskPromises = [];\n        this.configuration       = configuration;\n        this.isCli               = false;\n        this.warningLog          = new WarningLog();\n        this.compilerService     = compilerService;\n\n        this.apiMethodWasCalled = new FlagList([\n            OPTION_NAMES.src,\n            OPTION_NAMES.browsers,\n            OPTION_NAMES.reporter,\n            OPTION_NAMES.clientScripts\n        ]);\n    }\n\n    _createBootstrapper (browserConnectionGateway, compilerService) {\n        return new Bootstrapper({ browserConnectionGateway, compilerService });\n    }\n\n    _disposeBrowserSet (browserSet) {\n        return browserSet.dispose().catch(e => DEBUG_LOGGER(e));\n    }\n\n    _disposeReporters (reporters) {\n        return Promise.all(reporters.map(reporter => reporter.dispose().catch(e => DEBUG_LOGGER(e))));\n    }\n\n    _disposeTestedApp (testedApp) {\n        return testedApp ? testedApp.kill().catch(e => DEBUG_LOGGER(e)) : Promise.resolve();\n    }\n\n    async _disposeTaskAndRelatedAssets (task, browserSet, reporters, testedApp) {\n        task.abort();\n        task.unRegisterClientScriptRouting();\n        task.clearListeners();\n\n        await this._disposeAssets(browserSet, reporters, testedApp);\n    }\n\n    _disposeAssets (browserSet, reporters, testedApp) {\n        return Promise.all([\n            this._disposeBrowserSet(browserSet),\n            this._disposeReporters(reporters),\n            this._disposeTestedApp(testedApp)\n        ]);\n    }\n\n    _prepareArrayParameter (array) {\n        array = flatten(array);\n\n        if (this.isCli)\n            return array.length === 0 ? void 0 : array;\n\n        return array;\n    }\n\n    _createCancelablePromise (taskPromise) {\n        const promise           = taskPromise.then(({ completionPromise }) => completionPromise);\n        const removeFromPending = () => remove(this.pendingTaskPromises, promise);\n\n        promise\n            .then(removeFromPending)\n            .catch(removeFromPending);\n\n        promise.cancel = () => taskPromise\n            .then(({ cancelTask }) => cancelTask())\n            .then(removeFromPending);\n\n        this.pendingTaskPromises.push(promise);\n\n        return promise;\n    }\n\n    // Run task\n    _getFailedTestCount (task, reporter) {\n        let failedTestCount = reporter.testCount - reporter.passed;\n\n        if (task.opts.stopOnFirstFail && !!failedTestCount)\n            failedTestCount = 1;\n\n        return failedTestCount;\n    }\n\n    async _getTaskResult (task, browserSet, reporters, testedApp) {\n        if (!task.opts.live) {\n            task.on('browser-job-done', job => {\n                job.browserConnections.forEach(bc => browserSet.releaseConnection(bc));\n            });\n        }\n\n        const browserSetErrorPromise = promisifyEvent(browserSet, 'error');\n        const taskErrorPromise       = promisifyEvent(task, 'error');\n        const streamController       = new ReporterStreamController(task, reporters);\n\n        const taskDonePromise = task.once('done')\n            .then(() => browserSetErrorPromise.cancel())\n            .then(() => {\n                return Promise.all(reporters.map(reporter => reporter.pendingTaskDonePromise));\n            });\n\n        const promises = [\n            taskDonePromise,\n            browserSetErrorPromise,\n            taskErrorPromise\n        ];\n\n        if (testedApp)\n            promises.push(testedApp.errorPromise);\n\n        try {\n            await Promise.race(promises);\n        }\n        catch (err) {\n            await this._disposeTaskAndRelatedAssets(task, browserSet, reporters, testedApp);\n\n            throw err;\n        }\n\n        await this._disposeAssets(browserSet, reporters, testedApp);\n\n        if (streamController.multipleStreamError)\n            throw streamController.multipleStreamError;\n\n        return this._getFailedTestCount(task, reporters[0]);\n    }\n\n    _createTask (tests, browserConnectionGroups, proxy, opts, warningLog) {\n        return new Task({\n            tests,\n            browserConnectionGroups,\n            proxy,\n            opts,\n            runnerWarningLog: warningLog,\n            compilerService:  this.compilerService\n        });\n    }\n\n    _runTask ({ reporterPlugins, browserSet, tests, testedApp, options }) {\n        const task              = this._createTask(tests, browserSet.browserConnectionGroups, this.proxy, options, this.warningLog);\n        const reporters         = reporterPlugins.map(reporter => new Reporter(reporter.plugin, task, reporter.outStream, reporter.name));\n        const completionPromise = this._getTaskResult(task, browserSet, reporters, testedApp);\n        let completed           = false;\n\n        task.on('start', startHandlingTestErrors);\n\n        if (!this.configuration.getOption(OPTION_NAMES.skipUncaughtErrors)) {\n            task.on('test-run-start', addRunningTest);\n            task.on('test-run-done', removeRunningTest);\n        }\n\n        task.on('done', stopHandlingTestErrors);\n\n        task.on('error', stopHandlingTestErrors);\n\n        const onTaskCompleted = () => {\n            task.unRegisterClientScriptRouting();\n\n            completed = true;\n        };\n\n        completionPromise\n            .then(onTaskCompleted)\n            .catch(onTaskCompleted);\n\n        const cancelTask = async () => {\n            if (!completed)\n                await this._disposeTaskAndRelatedAssets(task, browserSet, reporters, testedApp);\n        };\n\n        return { completionPromise, cancelTask };\n    }\n\n    _registerAssets (assets) {\n        assets.forEach(asset => this.proxy.GET(asset.path, asset.info));\n    }\n\n    _validateDebugLogger () {\n        const debugLogger = this.configuration.getOption(OPTION_NAMES.debugLogger);\n\n        const debugLoggerDefinedCorrectly = debugLogger === null || !!debugLogger &&\n            ['showBreakpoint', 'hideBreakpoint'].every(method => method in debugLogger && isFunction(debugLogger[method]));\n\n        if (!debugLoggerDefinedCorrectly) {\n            this.configuration.mergeOptions({\n                [OPTION_NAMES.debugLogger]: defaultDebugLogger\n            });\n        }\n    }\n\n    _validateSpeedOption () {\n        const speed = this.configuration.getOption(OPTION_NAMES.speed);\n\n        if (speed === void 0)\n            return;\n\n        if (typeof speed !== 'number' || isNaN(speed) || speed < 0.01 || speed > 1)\n            throw new GeneralError(RUNTIME_ERRORS.invalidSpeedValue);\n    }\n\n    _validateConcurrencyOption () {\n        const concurrency = this.configuration.getOption(OPTION_NAMES.concurrency);\n\n        if (concurrency === void 0)\n            return;\n\n        if (typeof concurrency !== 'number' || isNaN(concurrency) || concurrency < 1)\n            throw new GeneralError(RUNTIME_ERRORS.invalidConcurrencyFactor);\n    }\n\n    _validateRequestTimeoutOption (optionName) {\n        const requestTimeout = this.configuration.getOption(optionName);\n\n        if (requestTimeout === void 0)\n            return;\n\n        assertType(is.nonNegativeNumber, null, `\"${optionName}\" option`, requestTimeout);\n    }\n\n    _validateProxyBypassOption () {\n        let proxyBypass = this.configuration.getOption(OPTION_NAMES.proxyBypass);\n\n        if (proxyBypass === void 0)\n            return;\n\n        assertType([ is.string, is.array ], null, '\"proxyBypass\" argument', proxyBypass);\n\n        if (typeof proxyBypass === 'string')\n            proxyBypass = [proxyBypass];\n\n        proxyBypass = proxyBypass.reduce((arr, rules) => {\n            assertType(is.string, null, '\"proxyBypass\" argument', rules);\n\n            return arr.concat(rules.split(','));\n        }, []);\n\n        this.configuration.mergeOptions({ proxyBypass });\n    }\n\n    _getScreenshotOptions () {\n        let { path, pathPattern } = this.configuration.getOption(OPTION_NAMES.screenshots) || {};\n\n        if (!path)\n            path = this.configuration.getOption(OPTION_NAMES.screenshotPath);\n\n        if (!pathPattern)\n            pathPattern = this.configuration.getOption(OPTION_NAMES.screenshotPathPattern);\n\n        return { path, pathPattern };\n    }\n\n    _validateScreenshotOptions () {\n        const { path, pathPattern } = this._getScreenshotOptions();\n\n        const disableScreenshots = this.configuration.getOption(OPTION_NAMES.disableScreenshots) || !path;\n\n        this.configuration.mergeOptions({ [OPTION_NAMES.disableScreenshots]: disableScreenshots });\n\n        if (disableScreenshots)\n            return;\n\n        if (path) {\n            this._validateScreenshotPath(path, 'screenshots base directory path');\n\n            this.configuration.mergeOptions({ [OPTION_NAMES.screenshots]: { path: resolvePath(path) } });\n        }\n\n        if (pathPattern) {\n            this._validateScreenshotPath(pathPattern, 'screenshots path pattern');\n\n            this.configuration.mergeOptions({ [OPTION_NAMES.screenshots]: { pathPattern } });\n        }\n    }\n\n    async _validateVideoOptions () {\n        const videoPath            = this.configuration.getOption(OPTION_NAMES.videoPath);\n        const videoEncodingOptions = this.configuration.getOption(OPTION_NAMES.videoEncodingOptions);\n\n        let videoOptions = this.configuration.getOption(OPTION_NAMES.videoOptions);\n\n        if (!videoPath) {\n            if (videoOptions || videoEncodingOptions)\n                throw new GeneralError(RUNTIME_ERRORS.cannotSetVideoOptionsWithoutBaseVideoPathSpecified);\n\n            return;\n        }\n\n        this.configuration.mergeOptions({ [OPTION_NAMES.videoPath]: resolvePath(videoPath) });\n\n        if (!videoOptions) {\n            videoOptions = {};\n\n            this.configuration.mergeOptions({ [OPTION_NAMES.videoOptions]: videoOptions });\n        }\n\n        if (videoOptions.ffmpegPath)\n            videoOptions.ffmpegPath = resolvePath(videoOptions.ffmpegPath);\n        else\n            videoOptions.ffmpegPath = await detectFFMPEG();\n\n        if (!videoOptions.ffmpegPath)\n            throw new GeneralError(RUNTIME_ERRORS.cannotFindFFMPEG);\n    }\n\n    _validateCompilerOptions () {\n        const compilerOptions = this.configuration.getOption(OPTION_NAMES.compilerOptions);\n\n        if (!compilerOptions)\n            return;\n\n        const specifiedCompilers  = Object.keys(compilerOptions);\n        const customizedCompilers = Object.keys(CustomizableCompilers);\n        const wrongCompilers      = specifiedCompilers.filter(compiler => !customizedCompilers.includes(compiler));\n\n        if (!wrongCompilers.length)\n            return;\n\n        const compilerListStr = getConcatenatedValuesString(wrongCompilers, void 0, \"'\");\n        const pluralSuffix    = getPluralSuffix(wrongCompilers);\n\n        throw new GeneralError(RUNTIME_ERRORS.cannotCustomizeSpecifiedCompilers, compilerListStr, pluralSuffix);\n    }\n\n    _validateRetryTestPagesOption () {\n        const retryTestPagesOption = this.configuration.getOption(OPTION_NAMES.retryTestPages);\n\n        if (!retryTestPagesOption)\n            return;\n\n        const ssl = this.configuration.getOption(OPTION_NAMES.ssl);\n\n        if (ssl)\n            return;\n\n        const hostname = this.configuration.getOption(OPTION_NAMES.hostname);\n\n        if (isLocalhost(hostname))\n            return;\n\n        throw new GeneralError(RUNTIME_ERRORS.cannotEnableRetryTestPagesOption);\n    }\n\n    async _validateRunOptions () {\n        this._validateDebugLogger();\n        this._validateScreenshotOptions();\n        await this._validateVideoOptions();\n        this._validateSpeedOption();\n        this._validateConcurrencyOption();\n        this._validateProxyBypassOption();\n        this._validateCompilerOptions();\n        this._validateRetryTestPagesOption();\n        this._validateRequestTimeoutOption(OPTION_NAMES.pageRequestTimeout);\n        this._validateRequestTimeoutOption(OPTION_NAMES.ajaxRequestTimeout);\n    }\n\n    _createRunnableConfiguration () {\n        return this.bootstrapper\n            .createRunnableConfiguration()\n            .then(runnableConfiguration => {\n                this.emit('done-bootstrapping');\n\n                return runnableConfiguration;\n            });\n    }\n\n    _validateScreenshotPath (screenshotPath, pathType) {\n        const forbiddenCharsList = checkFilePath(screenshotPath);\n\n        if (forbiddenCharsList.length)\n            throw new GeneralError(RUNTIME_ERRORS.forbiddenCharatersInScreenshotPath, screenshotPath, pathType, renderForbiddenCharsList(forbiddenCharsList));\n    }\n\n    _setBootstrapperOptions () {\n        this.configuration.prepare();\n        this.configuration.notifyAboutOverriddenOptions();\n        this.configuration.notifyAboutDeprecatedOptions(this.warningLog);\n\n        this.bootstrapper.sources                = this.configuration.getOption(OPTION_NAMES.src) || this.bootstrapper.sources;\n        this.bootstrapper.browsers               = this.configuration.getOption(OPTION_NAMES.browsers) || this.bootstrapper.browsers;\n        this.bootstrapper.concurrency            = this.configuration.getOption(OPTION_NAMES.concurrency);\n        this.bootstrapper.appCommand             = this.configuration.getOption(OPTION_NAMES.appCommand) || this.bootstrapper.appCommand;\n        this.bootstrapper.appInitDelay           = this.configuration.getOption(OPTION_NAMES.appInitDelay);\n        this.bootstrapper.filter                 = this.configuration.getOption(OPTION_NAMES.filter) || this.bootstrapper.filter;\n        this.bootstrapper.reporters              = this.configuration.getOption(OPTION_NAMES.reporter) || this.bootstrapper.reporters;\n        this.bootstrapper.tsConfigPath           = this.configuration.getOption(OPTION_NAMES.tsConfigPath);\n        this.bootstrapper.clientScripts          = this.configuration.getOption(OPTION_NAMES.clientScripts) || this.bootstrapper.clientScripts;\n        this.bootstrapper.disableMultipleWindows = this.configuration.getOption(OPTION_NAMES.disableMultipleWindows);\n        this.bootstrapper.compilerOptions        = this.configuration.getOption(OPTION_NAMES.compilerOptions);\n        this.bootstrapper.browserInitTimeout     = this.configuration.getOption(OPTION_NAMES.browserInitTimeout);\n    }\n\n    async _prepareClientScripts (tests, clientScripts) {\n        return Promise.all(tests.map(async test => {\n            if (test.isLegacy)\n                return;\n\n            let loadedTestClientScripts = await loadClientScripts(test.clientScripts, dirname(test.testFile.filename));\n\n            loadedTestClientScripts = clientScripts.concat(loadedTestClientScripts);\n\n            test.clientScripts = setUniqueUrls(loadedTestClientScripts);\n        }));\n    }\n\n    // API\n    embeddingOptions (opts) {\n        const { assets, TestRunCtor } = opts;\n\n        this._registerAssets(assets);\n        this.configuration.mergeOptions({ TestRunCtor });\n\n        return this;\n    }\n\n    src (...sources) {\n        if (this.apiMethodWasCalled.src)\n            throw new GeneralError(RUNTIME_ERRORS.multipleAPIMethodCallForbidden, OPTION_NAMES.src);\n\n        sources = this._prepareArrayParameter(sources);\n        this.configuration.mergeOptions({ [OPTION_NAMES.src]: sources });\n\n        this.apiMethodWasCalled.src = true;\n\n        return this;\n    }\n\n    browsers (...browsers) {\n        if (this.apiMethodWasCalled.browsers)\n            throw new GeneralError(RUNTIME_ERRORS.multipleAPIMethodCallForbidden, OPTION_NAMES.browsers);\n\n        browsers = this._prepareArrayParameter(browsers);\n        this.configuration.mergeOptions({ browsers });\n\n        this.apiMethodWasCalled.browsers = true;\n\n        return this;\n    }\n\n    concurrency (concurrency) {\n        this.configuration.mergeOptions({ concurrency });\n\n        return this;\n    }\n\n    reporter (name, output) {\n        if (this.apiMethodWasCalled.reporter)\n            throw new GeneralError(RUNTIME_ERRORS.multipleAPIMethodCallForbidden, OPTION_NAMES.reporter);\n\n        let reporters = prepareReporters(name, output);\n\n        reporters = this._prepareArrayParameter(reporters);\n\n        this.configuration.mergeOptions({ [OPTION_NAMES.reporter]: reporters });\n\n        this.apiMethodWasCalled.reporter = true;\n\n        return this;\n    }\n\n    filter (filter) {\n        this.configuration.mergeOptions({ filter });\n\n        return this;\n    }\n\n    useProxy (proxy, proxyBypass) {\n        this.configuration.mergeOptions({ proxy, proxyBypass });\n\n        return this;\n    }\n\n    screenshots (...options) {\n        let fullPage;\n        let [path, takeOnFails, pathPattern] = options;\n\n        if (options.length === 1 && options[0] && typeof options[0] === 'object')\n            ({ path, takeOnFails, pathPattern, fullPage } = options[0]);\n\n        this.configuration.mergeOptions({ screenshots: { path, takeOnFails, pathPattern, fullPage } });\n\n        return this;\n    }\n\n    video (path, options, encodingOptions) {\n        this.configuration.mergeOptions({\n            [OPTION_NAMES.videoPath]:            path,\n            [OPTION_NAMES.videoOptions]:         options,\n            [OPTION_NAMES.videoEncodingOptions]: encodingOptions\n        });\n\n        return this;\n    }\n\n    startApp (command, initDelay) {\n        this.configuration.mergeOptions({\n            [OPTION_NAMES.appCommand]:   command,\n            [OPTION_NAMES.appInitDelay]: initDelay\n        });\n\n        return this;\n    }\n\n    tsConfigPath (path) {\n        this.configuration.mergeOptions({\n            [OPTION_NAMES.tsConfigPath]: path\n        });\n\n        return this;\n    }\n\n    clientScripts (...scripts) {\n        if (this.apiMethodWasCalled.clientScripts)\n            throw new GeneralError(RUNTIME_ERRORS.multipleAPIMethodCallForbidden, OPTION_NAMES.clientScripts);\n\n        scripts = this._prepareArrayParameter(scripts);\n\n        this.configuration.mergeOptions({ [OPTION_NAMES.clientScripts]: scripts });\n\n        this.apiMethodWasCalled.clientScripts = true;\n\n        return this;\n    }\n\n    compilerOptions (opts) {\n        this.configuration.mergeOptions({\n            [OPTION_NAMES.compilerOptions]: opts\n        });\n\n        return this;\n    }\n\n    run (options = {}) {\n        this.apiMethodWasCalled.reset();\n        this.configuration.mergeOptions(options);\n        this._setBootstrapperOptions();\n\n        const runTaskPromise = Promise.resolve()\n            .then(() => this._validateRunOptions())\n            .then(() => this._createRunnableConfiguration())\n            .then(async ({ reporterPlugins, browserSet, tests, testedApp, commonClientScripts }) => {\n                await this._prepareClientScripts(tests, commonClientScripts);\n\n                const resultOptions = this.configuration.getOptions();\n\n                await this.bootstrapper.compilerService?.setOptions({ value: resultOptions });\n\n                return this._runTask({ reporterPlugins, browserSet, tests, testedApp, options: resultOptions });\n            });\n\n        return this._createCancelablePromise(runTaskPromise);\n    }\n\n    async stop () {\n        // NOTE: When taskPromise is cancelled, it is removed from\n        // the pendingTaskPromises array, which leads to shifting indexes\n        // towards the beginning. So, we must copy the array in order to iterate it,\n        // or we can perform iteration from the end to the beginning.\n        const cancellationPromises = this.pendingTaskPromises.reduceRight((result, taskPromise) => {\n            result.push(taskPromise.cancel());\n\n            return result;\n        }, []);\n\n        await Promise.all(cancellationPromises);\n    }\n}\n"]}