testcafe 1.14.1 → 1.14.2

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