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