testcafe 2.3.1 → 2.4.0

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 (57) hide show
  1. package/CHANGELOG.md +25 -0
  2. package/LICENSE +1 -1
  3. package/README.md +40 -46
  4. package/lib/api/test-controller/execution-context.js +4 -2
  5. package/lib/browser/connection/gateway.js +43 -1
  6. package/lib/browser/connection/index.js +24 -10
  7. package/lib/browser/connection/service-routes.js +3 -1
  8. package/lib/browser/provider/built-in/dedicated/chrome/index.js +19 -5
  9. package/lib/browser/provider/index.js +4 -1
  10. package/lib/client/automation/deps/hammerhead.js +24 -0
  11. package/lib/client/automation/deps/testcafe-core.js +24 -0
  12. package/lib/client/automation/index.js +251 -54
  13. package/lib/client/automation/index.min.js +1 -1
  14. package/lib/client/automation/playback/press/get-key-info.js +44 -0
  15. package/lib/client/automation/playback/press/utils.js +142 -0
  16. package/lib/client/automation/utils/get-key-code.js +15 -0
  17. package/lib/client/automation/utils/get-key-identifier.js +14 -0
  18. package/lib/client/automation/utils/is-letter.js +7 -0
  19. package/lib/client/automation/utils/key-identifier-maps.js +93 -0
  20. package/lib/client/browser/idle-page/index.js +7 -2
  21. package/lib/client/core/index.js +46 -3
  22. package/lib/client/core/index.min.js +1 -1
  23. package/lib/client/driver/index.js +45 -17
  24. package/lib/client/driver/index.min.js +1 -1
  25. package/lib/client/test-run/index.js.mustache +34 -30
  26. package/lib/client/ui/index.js +3230 -48
  27. package/lib/client/ui/index.min.js +1 -1
  28. package/lib/client/ui/sprite.svg +15 -0
  29. package/lib/client/ui/styles.css +183 -41
  30. package/lib/live/test-runner.js +4 -2
  31. package/lib/proxyless/api-base.js +3 -2
  32. package/lib/proxyless/client/event-descriptor.js +67 -0
  33. package/lib/proxyless/client/input.js +36 -0
  34. package/lib/proxyless/client/key-press/utils.js +31 -0
  35. package/lib/proxyless/client/types.js +1 -1
  36. package/lib/proxyless/client/utils.js +32 -2
  37. package/lib/proxyless/cookie-provider.js +31 -3
  38. package/lib/proxyless/errors.js +20 -0
  39. package/lib/proxyless/index.js +2 -10
  40. package/lib/proxyless/request-pipeline/index.js +32 -3
  41. package/lib/proxyless/resource-injector.js +5 -4
  42. package/lib/proxyless/types.js +3 -2
  43. package/lib/proxyless/utils/cdp.js +4 -3
  44. package/lib/reporter/index.js +2 -1
  45. package/lib/runner/bootstrapper.js +8 -1
  46. package/lib/runner/index.js +57 -44
  47. package/lib/shared/utils/is-file-protocol.js +2 -2
  48. package/lib/test-run/commands/actions.js +2 -2
  49. package/lib/test-run/cookies/base.js +4 -1
  50. package/lib/test-run/cookies/provider.js +9 -2
  51. package/lib/test-run/execute-js-expression/index.js +4 -2
  52. package/lib/test-run/index.js +17 -12
  53. package/lib/test-run/request/create-request-options.js +23 -12
  54. package/lib/test-run/request/send.js +2 -3
  55. package/lib/utils/check-is-vm.js +58 -0
  56. package/package.json +9 -6
  57. package/lib/proxyless/client/event-simulator.js +0 -40
package/CHANGELOG.md CHANGED
@@ -1,5 +1,30 @@
1
1
  # Changelog
2
2
 
3
+ ## v2.4.0 (2023-03-06)
4
+ TestCafe v2.4.0 introduces the Visual Selector Debugger. You can now create and debug Selector queries in the browser window.
5
+
6
+ ### Visual Selector Debugger
7
+
8
+ TestCafe v2.4.0 displays the Visual Selector Debugger panel when you activate [Debug Mode](https://testcafe.io/documentation/402835/guides/basic-guides/debug-tests). Use the panel to debug Selector queries from your test, or generate new Selector queries.
9
+
10
+ ![](https://testcafe.io/images/inspector/enter-query.gif)
11
+
12
+ If a Selector query causes your test to fail, add the [t.debug()](https://testcafe.io/documentation/402707/reference/test-api/testcontroller/debug) command after the last successful action, and launch the test.
13
+
14
+ When the test reaches the breakpoint, the window that runs the test displays the Selector Debugger panel. Copy the failing Selector query from test code to the Selector Debugger input field.
15
+
16
+ * TestCafe highlights page elements that match the Selector query.
17
+ * If no elements match the Selector query, the panel displays the **No Matching Elements** warning.
18
+ * If your Selector query contians a syntax error, the panel displays the **Invalid Selector** warning.
19
+
20
+ To interactively generate a Selector query, click the **Pick** button, and select the target element on the page.
21
+
22
+ For more information on the panel, its capabilities, and limitations, read the [Visual Selector Debugger Guide](https://testcafe.io/documentation/404288/guides/intermediate-guides/visual-selector-debugger).
23
+
24
+ ### Bug Fixes
25
+
26
+ * TestCafe cannot execute the [t.request](https://testcafe.io/documentation/403981/reference/test-api/testcontroller/request) action in [proxyless mode](https://testcafe.io/documentation/404237/guides/experimental-capabilities/proxyless-mode) ([#7523](https://github.com/DevExpress/testcafe/issues/7523)).
27
+
3
28
  ## v2.3.1 (2023-02-09)
4
29
  TestCafe v2.3.1 introduces a number of bug fixes.
5
30
 
package/LICENSE CHANGED
@@ -1,6 +1,6 @@
1
1
  The MIT License
2
2
 
3
- Copyright (C) 2012-2022 Developer Express Inc.
3
+ Copyright (C) 2012-2023 Developer Express Inc.
4
4
 
5
5
  Permission is hereby granted, free of charge, to any person obtaining a copy
6
6
  of this software and associated documentation files (the "Software"), to deal
package/README.md CHANGED
@@ -1,7 +1,7 @@
1
1
  <p align="center">
2
- <a href="https://testcafe.io/surveys/open-source-feedback-2022">
3
- <img src="https://raw.github.com/DevExpress/testcafe/master/media/2022-feedback-survey-banner.png" alt="2022 TestCafe Feedback Survey" />
4
- </a>
2
+ <a href="https://www.devexpress.com/products/testcafestudio/?utm_source=github.com&utm_medium=referral&utm_campaign=tc-gh-banner">
3
+ <img src="https://raw.github.com/DevExpress/testcafe/master/media/testcafe-studio-banner.png" alt="Try TestCafe Studio IDE" />
4
+ </a>
5
5
  </p>
6
6
 
7
7
  <p align="center">
@@ -17,7 +17,9 @@
17
17
  </p>
18
18
 
19
19
  <p align="center">
20
- <i>A Node.js tool to automate end-to-end web testing.<br/>Write tests in JS or TypeScript, run them and view results.</i>
20
+ <i> Automate end-to-end web testing with TestCafe, a Node.js-based testing framework.</i><br><br>
21
+ TestCafe is <strong>free</strong> and as easy to use as <strong>1-2-3</strong>:<br>
22
+ <strong>1.</strong> Write your tests in JS or TypeScript.<br><strong>2.</strong> Execute your tests.<br><strong>3.</strong> View test results.
21
23
  </p>
22
24
 
23
25
  <p align="center">
@@ -29,7 +31,7 @@
29
31
 
30
32
  * **Works on all popular environments**: TestCafe runs on Windows, MacOS, and Linux. It supports desktop, mobile, remote and cloud [browsers](https://testcafe.io/documentation/402828/guides/concepts/browsers#browser-support) (UI or headless).
31
33
  * **1 minute to set up**: You [do not need WebDriver](https://testcafe.io/402636/faq#i-have-heard-that-testcafe-does-not-use-selenium-how-does-it-operate) or any other testing software. Install TestCafe with one command, and you are ready to test: `npm install -g testcafe`
32
- * **Free and open source**: TestCafe is free to use under the [MIT license](https://github.com/DevExpress/testcafe/blob/master/LICENSE). [Plugins](#plugins) provide custom reports, integration with other tools, launching tests from IDE, etc. You can use the plugins made by the GitHub community or make your own.
34
+ * **Free and open source**: TestCafe is free to use under the [MIT license](https://github.com/DevExpress/testcafe/blob/master/LICENSE). [Plugins](#plugins) provide custom reports, integration with other tools, launching tests from IDE, etc. You can use the plugins made by the GitHub community or create your own.
33
35
 
34
36
  ![Install TestCafe and Run a Test](https://raw.githubusercontent.com/DevExpress/testcafe/master/media/install-and-run-test.gif)
35
37
 
@@ -62,26 +64,20 @@ You can change the maximum wait time.
62
64
  If elements load faster, tests skip the timeout and continue.
63
65
 
64
66
  **Rapid test development tool**<br/>
65
- Changes in test code immediately restart the test, and you see the results instantly.<br/>
66
- See how it works in the [TestCafe Live repository](https://github.com/DevExpress/testcafe-live).
67
+ When you enable [live mode](https://testcafe.io/documentation/403842/guides/intermediate-guides/live-mode), changes to test code immediately restart the test, and you instantly see the results.
67
68
 
68
69
  **Latest JS and TypeScript support**<br/>
69
- TestCafe supports the latest JavaScript features, including ES2017 (for example, async/await).
70
- You can also [use TypeScript](https://testcafe.io/documentation/402824/guides/concepts/typescript-and-coffeescript#typescript-support)
71
- if you prefer a strongly typed language.
70
+ TestCafe supports the most recent JavaScript-related features, including ES2017 (async/await). You can also [use TypeScript](https://testcafe.io/documentation/402824/guides/concepts/typescript-and-coffeescript#typescript-support) if you prefer a strongly typed language instead.
72
71
 
73
72
  **Detects JS errors in your code**<br/>
74
- TestCafe reports JS errors that it finds on the webpage.
75
- Tests automatically fail because of that.
76
- However, you can disable this.
73
+ TestCafe reports JS errors that it locates on a given webpage. Tests automatically fail if TestCafe encounters such errors.<br />
74
+ You can, however, disable this option.<br />
77
75
 
78
76
  **Concurrent test launch**<br/>
79
- TestCafe can open multiple instances of the same browser to run parallel
80
- tests which decreases test execution time.
77
+ TestCafe can open multiple instances of the same browser and run parallel tests (to help decrease test execution time).
81
78
 
82
79
  **PageObject pattern support**<br/>
83
- The TestCafe's [Test API](https://testcafe.io/documentation/402632/reference)
84
- includes a high-level selector library, assertions, etc.
80
+ The TestCafe's [Test API](https://testcafe.io/documentation/402632/reference) includes a high-level selector library, assertions, etc.
85
81
  You can combine them to implement readable tests with the [PageObject pattern](https://testcafe.io/documentation/402826/guides/concepts/page-model).
86
82
 
87
83
  ```js
@@ -89,14 +85,15 @@ const macOSInput = Selector('.column').find('label').withText('MacOS').child('in
89
85
  ```
90
86
 
91
87
  **Easy to include in a continuous integration system**<br/>
92
- You can run TestCafe from a console, and its reports can be viewed in a CI system's interface
93
- (TeamCity, Jenkins, Travis & etc.)
88
+ You can run TestCafe from a console, and its reports can be viewed within CI systems (TeamCity, Jenkins, Travis & etc.)
89
+
90
+ # Love TestCafe Open-source Edition? Want to Record Tests without Writing JavaScript or TypeScript Code?
94
91
 
95
92
  ## TestCafe Studio: IDE for End-to-End Web Testing
96
93
 
97
- TestCafe works great for JavaScript developers, but at some point you will need to delegate testing tasks to your Q&A department. If that's the case and you are looking for a codeless way to record and maintain tests compatible with your existing infrastructure, check out [TestCafe Studio](https://www.devexpress.com/products/testcafestudio/?utm_source=github.com&utm_medium=referral&utm_campaign=tc-gh-ide) - a testing IDE built on top of the open-source TestCafe.
94
+ TestCafe is the perfect choice for JavaScript developers and experienced Q&A teams. If you’d like to delegate testing to QA engineers and are looking for a code-free way to record and maintain tests compatible with your existing infrastructure, check out [TestCafe Studio](https://www.devexpress.com/products/testcafestudio/?utm_source=github.com&utm_medium=referral&utm_campaign=tc-gh-ide) - a testing IDE built atop the open-source version of TestCafe.
98
95
 
99
- Read the following article to learn how TestCafe Studio could fit into your workflow: [What's Better than TestCafe? TestCafe Studio](https://www.devexpress.com/products/testcafestudio/qa-end-to-end-web-testing.xml).
96
+ Review the following article to learn how TestCafe Studio can fit into any workflow: [What's Better than TestCafe? TestCafe Studio](https://www.devexpress.com/products/testcafestudio/qa-end-to-end-web-testing.xml).
100
97
 
101
98
  ![Get Started with TestCafe Studio](https://raw.githubusercontent.com/DevExpress/testcafe/master/media/testcafe-studio-get-started.gif)
102
99
 
@@ -104,11 +101,11 @@ Read the following article to learn how TestCafe Studio could fit into your work
104
101
  <i>Record and Run a Test in TestCafe Studio</i>
105
102
  </p>
106
103
 
107
- ## Getting Started
104
+ # Getting Started
108
105
 
109
106
  ### Installation
110
107
 
111
- Ensure that [Node.js](https://nodejs.org/) ([Current or Active LTS](https://github.com/nodejs/Release#release-phases) is recommended, version 12 at minimum) and [npm](https://www.npmjs.com/) are installed on your computer before running it:
108
+ Ensure that [Node.js](https://nodejs.org/) ([Current or Active LTS](https://github.com/nodejs/Release#release-phases) is recommended, version 18 at minimum) and [npm](https://www.npmjs.com/) are installed on your computer before running it:
112
109
 
113
110
  ```sh
114
111
  npm install -g testcafe
@@ -116,11 +113,9 @@ npm install -g testcafe
116
113
 
117
114
  ### Creating the Test
118
115
 
119
- As an example, we are going to test the [https://devexpress.github.io/testcafe/example](https://devexpress.github.io/testcafe/example) page.
116
+ For this simple example, we will test the following page: [https://devexpress.github.io/testcafe/example](https://devexpress.github.io/testcafe/example)
120
117
 
121
- Create a `.js` or `.ts` file on your computer.
122
- Note that it needs to have a specific structure: tests must be organized into fixtures.
123
- You can paste the following code to see the test in action:
118
+ Create a .js or .ts file on your computer. Remember that a .js or .ts file must maintain a specific structure: tests must be organized into fixtures. You can paste the following code to see the test in action:
124
119
 
125
120
  ```js
126
121
  import { Selector } from 'testcafe'; // first import testcafe selectors
@@ -129,57 +124,57 @@ fixture `Getting Started`// declare the fixture
129
124
  .page `https://devexpress.github.io/testcafe/example`; // specify the start page
130
125
 
131
126
 
132
- //then create a test and place your code there
127
+ //then create a test and place your code within it
133
128
  test('My first test', async t => {
134
129
  await t
135
130
  .typeText('#developer-name', 'John Smith')
136
131
  .click('#submit-button')
137
132
 
138
- // Use the assertion to check if the actual header text is equal to the expected one
133
+ // Use the assertion to check if actual header text equals expected text
139
134
  .expect(Selector('#article-header').innerText).eql('Thank you, John Smith!');
140
135
  });
141
136
  ```
142
137
 
143
138
  ### Running the Test
144
139
 
145
- Call the following command in a command shell.
146
- Specify the [target browser](https://testcafe.io/documentation/402639/reference/command-line-interface#browser-list)
147
- and [file path](https://testcafe.io/documentation/402639/reference/command-line-interface#file-pathglob-pattern).
140
+ Call the following command in a command shell. Specify the [target browser](https://testcafe.io/documentation/402639/reference/command-line-interface#browser-list) and [file path](https://testcafe.io/documentation/402639/reference/command-line-interface#file-pathglob-pattern).
148
141
 
149
142
  ```sh
150
143
  testcafe chrome test1.js
151
144
  ```
152
145
 
153
- TestCafe opens the browser and starts executing the test.
146
+ TestCafe opens the browser and begins test execution.
147
+
148
+ > Important: Make certain the browser tab that runs tests stays active.
149
+ > Do not minimize the browser window.
150
+ > Inactive tabs and minimized browser windows switch to lower resource consumption mode.
151
+ > In low consumption mode, tests may not execute correctly.
154
152
 
155
- > Important! Make sure the browser tab that runs tests stays active.
156
- > Do not minimize the browser window. Inactive tabs and minimized browser windows switch
157
- > to a lower resource consumption mode where tests are not guaranteed to execute correctly.
158
153
 
159
154
  ### Viewing the Results
160
155
 
161
- TestCafe outputs the results into a command shell by default. See [Reporters](https://testcafe.io/documentation/402825/guides/concepts/reporters)
162
- for more information. You can also use [plugins](#plugins) to customize the reports.
156
+ TestCafe outputs results into a command shell by default. See [Reporters](https://testcafe.io/documentation/402825/guides/concepts/reporters)
157
+ for more information. You can also use [plugins](#plugins) to customize reports.
163
158
 
164
159
  ![Test Report](https://testcafe.io/images/report.png)
165
160
 
166
- Read the [Getting Started](https://testcafe.io/documentation/402635/getting-started) page for a more detailed guide.
161
+ Read the [Getting Started](https://testcafe.io/documentation/402635/getting-started) page for additional assistance.
167
162
 
168
163
  ## Documentation
169
164
 
170
- Go to our website for full [documentation](https://testcafe.io/documentation/402635/getting-started) on TestCafe.
165
+ Visit the following webpage to review our online help system: [Documentation](https://testcafe.io/documentation/402635/getting-started).
171
166
 
172
167
  ## Get Help
173
168
 
174
- Join the TestCafe community on Stack Overflow to get help. Ask and answer [questions with the TestCafe tag](https://stackoverflow.com/questions/tagged/testcafe).
169
+ Join the TestCafe community on Stack Overflow. Ask and answer questions [using the TestCafe tag](https://stackoverflow.com/questions/tagged/testcafe).
175
170
 
176
171
  ## Issue Tracker
177
172
 
178
- Use our GitHub issues page to [report bugs](https://github.com/DevExpress/testcafe/issues/new?template=bug-report.md) and [suggest improvements](https://github.com/DevExpress/testcafe/issues/new?template=feature_request.md).
173
+ Use our GitHub issues page to [report bugs](https://github.com/DevExpress/testcafe/issues/new?template=bug-report.md) and [suggest enhancements](https://github.com/DevExpress/testcafe/issues/new?template=feature_request.md).
179
174
 
180
175
  ## Stay in Touch
181
176
 
182
- Follow us on [Twitter](https://twitter.com/DXTestCafe). We post TestCafe news and updates, several times a week.
177
+ Follow us on [Twitter](https://twitter.com/DXTestCafe). We post TestCafe news and updates.
183
178
 
184
179
  ## Contributing
185
180
 
@@ -187,14 +182,13 @@ Read our [Contributing Guide](https://github.com/DevExpress/testcafe/blob/master
187
182
 
188
183
  To create your own plugin for TestCafe, you can use these plugin generators:
189
184
 
190
- * [Build a browser provider](https://testcafe.io/documentation/402812/guides/extend-testcafe/browser-provider-plugin)
191
- to set up tests on your on-premises server farm, to use a cloud testing platform, or to start your local browsers in a special way. Use this [Yeoman generator](https://www.npmjs.com/package/generator-testcafe-browser-provider) to write only a few lines of code.
185
+ * [Build a browser provider](https://testcafe.io/documentation/402812/guides/extend-testcafe/browser-provider-plugin) to set up tests on your on-premises server farm, to use a cloud testing platform, or to start your local browsers in a special way. Use this [Yeoman generator](https://www.npmjs.com/package/generator-testcafe-browser-provider) to write only a few lines of code.
192
186
  * To [build a custom reporter](https://testcafe.io/documentation/402810/guides/extend-testcafe/reporter-plugin)
193
187
  with your formatting and style, check out this [generator](https://www.npmjs.com/package/generator-testcafe-reporter).
194
188
 
195
189
  If you want your plugin to be listed below, [send us a note in a Github issue](https://github.com/DevExpress/testcafe/issues/new).
196
190
 
197
- Thank you to all the people who already contributed to TestCafe!
191
+ Thanks to all of our contributors We appreciate your commitment to the TestCafe community.
198
192
 
199
193
  [<img alt="aha-oretama" src="https://avatars.githubusercontent.com/u/7259161?v=4&s=117" width="117">](https://github.com/aha-oretama) |[<img alt="ai" src="https://avatars.githubusercontent.com/u/19343?v=4&s=117" width="117">](https://github.com/ai) |[<img alt="aleks-pro" src="https://avatars.githubusercontent.com/u/14822473?v=4&s=117" width="117">](https://github.com/aleks-pro) |[<img alt="Aleksey28" src="https://avatars.githubusercontent.com/u/32869530?v=4&s=117" width="117">](https://github.com/Aleksey28) |[<img alt="AlexanderMoiseev" src="https://avatars.githubusercontent.com/u/1498953?v=4&s=117" width="117">](https://github.com/AlexanderMoiseev) |[<img alt="AlexanderMoskovkin" src="https://avatars.githubusercontent.com/u/12047804?v=4&s=117" width="117">](https://github.com/AlexanderMoskovkin) |
200
194
  :---: |:---: |:---: |:---: |:---: |:---: |
@@ -38,7 +38,9 @@ function createRequire(filename) {
38
38
  }
39
39
  function createSelectorDefinition(testRun) {
40
40
  return (fn, options = {}) => {
41
- const { skipVisibilityCheck, collectionMode } = testRun.controller.getExecutionContext()[OPTIONS_KEY];
41
+ const { skipVisibilityCheck, collectionMode } = testRun.controller ?
42
+ testRun.controller.getExecutionContext()[OPTIONS_KEY] :
43
+ createExecutionContext(testRun)[OPTIONS_KEY];
42
44
  if (skipVisibilityCheck)
43
45
  options.visibilityCheck = false;
44
46
  if (testRun && testRun.id)
@@ -86,4 +88,4 @@ function createExecutionContext(testRun) {
86
88
  }));
87
89
  }
88
90
  exports.createExecutionContext = createExecutionContext;
89
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execution-context.js","sourceRoot":"","sources":["../../../src/api/test-controller/execution-context.js"],"names":[],"mappings":"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","sourcesContent":["import { createContext } from 'vm';\nimport Module from 'module';\nimport path from 'path';\nimport exportableLib from '../exportable-lib';\nimport NODE_MODULES from '../../utils/node-modules-folder-name';\n\nconst OPTIONS_KEY = Symbol('options');\n\nexport const DEFAULT_CONTEXT_OPTIONS = {\n    skipVisibilityCheck: false,\n    collectionMode:      false,\n};\n\nfunction getModuleBasePaths (currentPath) {\n    const nodePaths = [];\n    let parentDir   = path.dirname(currentPath);\n\n    while (currentPath !== parentDir) {\n        currentPath = parentDir;\n        parentDir   = path.dirname(currentPath);\n\n        nodePaths.push(path.join(currentPath, NODE_MODULES));\n    }\n\n    return nodePaths;\n}\n\nfunction createRequire (filename) {\n    //Deprecated since: Node v12.2.0\n    if (Module.createRequireFromPath)\n        return Module.createRequireFromPath(filename);\n\n    if (Module.createRequire)\n        return Module.createRequire(filename);\n\n    const dummyModule          = new Module(filename, module);\n    const localModulesPaths    = getModuleBasePaths(filename);\n\n    dummyModule.filename = filename;\n    dummyModule.paths    = localModulesPaths.concat(module.paths);\n\n    return id => dummyModule.require(id);\n}\n\nfunction createSelectorDefinition (testRun) {\n    return (fn, options = {}) => {\n        const { skipVisibilityCheck, collectionMode } = testRun.controller.getExecutionContext()[OPTIONS_KEY];\n\n        if (skipVisibilityCheck)\n            options.visibilityCheck = false;\n\n        if (testRun && testRun.id)\n            options.boundTestRun = testRun;\n\n        if (collectionMode)\n            options.collectionMode = collectionMode;\n\n        return exportableLib.Selector(fn, options);\n    };\n}\n\nfunction createClientFunctionDefinition (testRun) {\n    return (fn, options = {}) => {\n        if (testRun && testRun.id)\n            options.boundTestRun = testRun;\n\n        return exportableLib.ClientFunction(fn, options);\n    };\n}\n\nexport function setContextOptions (context, options) {\n    context[OPTIONS_KEY] = options;\n}\n\nexport function createExecutionContext (testRun) {\n    const filename = testRun.test.testFile.filename;\n\n    const replacers = {\n        require:        createRequire(filename),\n        __filename:     filename,\n        __dirname:      path.dirname(filename),\n        t:              testRun.controller,\n        Selector:       createSelectorDefinition(testRun),\n        ClientFunction: createClientFunctionDefinition(testRun),\n        Role:           exportableLib.Role,\n        RequestLogger:  exportableLib.RequestLogger,\n        RequestMock:    exportableLib.RequestMock,\n        RequestHook:    exportableLib.RequestHook,\n        [OPTIONS_KEY]:  DEFAULT_CONTEXT_OPTIONS,\n        userVariables:  exportableLib.userVariables,\n    };\n\n    return createContext(new Proxy(replacers, {\n        get: (target, property) => {\n            if (replacers.hasOwnProperty(property))\n                return replacers[property];\n\n            if (global.hasOwnProperty(property))\n                return global[property];\n\n            throw new Error(`${property} is not defined`);\n        },\n    }));\n}\n"]}
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execution-context.js","sourceRoot":"","sources":["../../../src/api/test-controller/execution-context.js"],"names":[],"mappings":"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","sourcesContent":["import { createContext } from 'vm';\nimport Module from 'module';\nimport path from 'path';\nimport exportableLib from '../exportable-lib';\nimport NODE_MODULES from '../../utils/node-modules-folder-name';\n\nconst OPTIONS_KEY = Symbol('options');\n\nexport const DEFAULT_CONTEXT_OPTIONS = {\n    skipVisibilityCheck: false,\n    collectionMode:      false,\n};\n\nfunction getModuleBasePaths (currentPath) {\n    const nodePaths = [];\n    let parentDir   = path.dirname(currentPath);\n\n    while (currentPath !== parentDir) {\n        currentPath = parentDir;\n        parentDir   = path.dirname(currentPath);\n\n        nodePaths.push(path.join(currentPath, NODE_MODULES));\n    }\n\n    return nodePaths;\n}\n\nfunction createRequire (filename) {\n    //Deprecated since: Node v12.2.0\n    if (Module.createRequireFromPath)\n        return Module.createRequireFromPath(filename);\n\n    if (Module.createRequire)\n        return Module.createRequire(filename);\n\n    const dummyModule          = new Module(filename, module);\n    const localModulesPaths    = getModuleBasePaths(filename);\n\n    dummyModule.filename = filename;\n    dummyModule.paths    = localModulesPaths.concat(module.paths);\n\n    return id => dummyModule.require(id);\n}\n\nfunction createSelectorDefinition (testRun) {\n    return (fn, options = {}) => {\n        const { skipVisibilityCheck, collectionMode } = testRun.controller ?\n            testRun.controller.getExecutionContext()[OPTIONS_KEY] :\n            createExecutionContext(testRun)[OPTIONS_KEY];\n\n        if (skipVisibilityCheck)\n            options.visibilityCheck = false;\n\n        if (testRun && testRun.id)\n            options.boundTestRun = testRun;\n\n        if (collectionMode)\n            options.collectionMode = collectionMode;\n\n        return exportableLib.Selector(fn, options);\n    };\n}\n\nfunction createClientFunctionDefinition (testRun) {\n    return (fn, options = {}) => {\n        if (testRun && testRun.id)\n            options.boundTestRun = testRun;\n\n        return exportableLib.ClientFunction(fn, options);\n    };\n}\n\nexport function setContextOptions (context, options) {\n    context[OPTIONS_KEY] = options;\n}\n\nexport function createExecutionContext (testRun) {\n    const filename = testRun.test.testFile.filename;\n\n    const replacers = {\n        require:        createRequire(filename),\n        __filename:     filename,\n        __dirname:      path.dirname(filename),\n        t:              testRun.controller,\n        Selector:       createSelectorDefinition(testRun),\n        ClientFunction: createClientFunctionDefinition(testRun),\n        Role:           exportableLib.Role,\n        RequestLogger:  exportableLib.RequestLogger,\n        RequestMock:    exportableLib.RequestMock,\n        RequestHook:    exportableLib.RequestHook,\n        [OPTIONS_KEY]:  DEFAULT_CONTEXT_OPTIONS,\n        userVariables:  exportableLib.userVariables,\n    };\n\n    return createContext(new Proxy(replacers, {\n        get: (target, property) => {\n            if (replacers.hasOwnProperty(property))\n                return replacers[property];\n\n            if (global.hasOwnProperty(property))\n                return global[property];\n\n            throw new Error(`${property} is not defined`);\n        },\n    }));\n}\n"]}
@@ -10,6 +10,7 @@ const testcafe_hammerhead_1 = require("testcafe-hammerhead");
10
10
  const service_routes_1 = __importDefault(require("./service-routes"));
11
11
  const empty_page_markup_1 = __importDefault(require("../../proxyless/empty-page-markup"));
12
12
  const error_route_1 = __importDefault(require("../../proxyless/error-route"));
13
+ const initializers_1 = require("../../test-run/commands/validations/initializers");
13
14
  class BrowserConnectionGateway {
14
15
  constructor(proxy, options) {
15
16
  this._connections = {};
@@ -48,6 +49,8 @@ class BrowserConnectionGateway {
48
49
  this._dispatch(`${service_routes_1.default.closeWindow}/{id}`, proxy, BrowserConnectionGateway._onCloseWindowRequest, 'POST');
49
50
  this._dispatch(`${service_routes_1.default.openFileProtocol}/{id}`, proxy, BrowserConnectionGateway._onOpenFileProtocolRequest, 'POST');
50
51
  this._dispatch(`${service_routes_1.default.dispatchProxylessEvent}/{id}`, proxy, BrowserConnectionGateway._onDispatchProxylessEvent, 'POST', this.proxyless);
52
+ this._dispatch(`${service_routes_1.default.parseSelector}/{id}`, proxy, BrowserConnectionGateway._parseSelector, 'POST', this.proxyless);
53
+ this._dispatch(`${service_routes_1.default.dispatchProxylessEventSequence}/{id}`, proxy, BrowserConnectionGateway._onDispatchProxylessEventSequence, 'POST', this.proxyless);
51
54
  proxy.GET(service_routes_1.default.connect, (req, res) => this._connectNextRemoteBrowser(req, res));
52
55
  proxy.GET(service_routes_1.default.connectWithTrailingSlash, (req, res) => this._connectNextRemoteBrowser(req, res));
53
56
  proxy.GET(service_routes_1.default.serviceWorker, { content: serviceWorkerScript, contentType: 'application/x-javascript' });
@@ -176,6 +179,17 @@ class BrowserConnectionGateway {
176
179
  });
177
180
  }
178
181
  }
182
+ static _onDispatchProxylessEventSequence(req, res, connection) {
183
+ if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
184
+ BrowserConnectionGateway._fetchRequestData(req, data => {
185
+ const eventSequence = JSON.parse(data);
186
+ connection.dispatchProxylessEventSequence(eventSequence)
187
+ .then(() => {
188
+ (0, http_1.respondWithJSON)(res);
189
+ });
190
+ });
191
+ }
192
+ }
179
193
  async _connectNextRemoteBrowser(req, res) {
180
194
  (0, http_1.preventCaching)(res);
181
195
  const remoteConnection = await this._remotesQueue.shift();
@@ -184,6 +198,31 @@ class BrowserConnectionGateway {
184
198
  else
185
199
  (0, http_1.respond500)(res, 'There are no available _connections to establish.');
186
200
  }
201
+ static _getParsedSelector(testRun, rawSelector) {
202
+ const options = {
203
+ testRun,
204
+ skipVisibilityCheck: true,
205
+ collectionMode: true,
206
+ };
207
+ const value = rawSelector.trim().startsWith('Selector(') ? rawSelector : `'${rawSelector}'`;
208
+ const selector = { type: 'js-expr', value };
209
+ return (0, initializers_1.initSelector)('selector', selector, options);
210
+ }
211
+ static _parseSelector(req, res, connection) {
212
+ if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
213
+ BrowserConnectionGateway._fetchRequestData(req, data => {
214
+ try {
215
+ const testRun = connection.getCurrentTestRun();
216
+ const rawSelector = JSON.parse(data).selector;
217
+ const parsedSelector = BrowserConnectionGateway._getParsedSelector(testRun, rawSelector);
218
+ (0, http_1.respondWithJSON)(res, parsedSelector);
219
+ }
220
+ catch (error) {
221
+ (0, http_1.respondWithJSON)(res);
222
+ }
223
+ });
224
+ }
225
+ }
187
226
  // API
188
227
  startServingConnection(connection) {
189
228
  this._connections[connection.id] = connection;
@@ -199,7 +238,10 @@ class BrowserConnectionGateway {
199
238
  for (const id in this._connections)
200
239
  await this._connections[id].close();
201
240
  }
241
+ getConnections() {
242
+ return this._connections;
243
+ }
202
244
  }
203
245
  exports.default = BrowserConnectionGateway;
204
246
  module.exports = exports.default;
205
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gateway.js","sourceRoot":"","sources":["../../../src/browser/connection/gateway.ts"],"names":[],"mappings":"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","sourcesContent":["import loadAssets from '../../load-assets';\nimport {\n    respond404,\n    respond500,\n    respondWithJSON,\n    redirect,\n    preventCaching,\n} from '../../utils/http';\n\nimport RemotesQueue from './remotes-queue';\nimport { Proxy, acceptCrossOrigin } from 'testcafe-hammerhead';\nimport { Dictionary } from '../../configuration/interfaces';\nimport BrowserConnection from './index';\nimport { IncomingMessage, ServerResponse } from 'http';\nimport SERVICE_ROUTES from './service-routes';\nimport EMPTY_PAGE_MARKUP from '../../proxyless/empty-page-markup';\nimport PROXYLESS_ERROR_ROUTE from '../../proxyless/error-route';\n\nexport default class BrowserConnectionGateway {\n    private _connections: Dictionary<BrowserConnection> = {};\n    private _remotesQueue: RemotesQueue;\n    public readonly connectUrl: string;\n    public retryTestPages: boolean;\n    private readonly proxyless: boolean;\n    public readonly proxy: Proxy;\n\n    public constructor (proxy: Proxy, options: { retryTestPages: boolean; proxyless: boolean }) {\n        this._remotesQueue   = new RemotesQueue();\n        this.connectUrl      = proxy.resolveRelativeServiceUrl(SERVICE_ROUTES.connect);\n        this.retryTestPages  = options.retryTestPages;\n        this.proxyless       = options.proxyless;\n        this.proxy           = proxy;\n\n        this._registerRoutes(proxy);\n    }\n\n    private _dispatch (url: string, proxy: Proxy, handler: Function, method = 'GET', shouldAcceptCrossOrigin?: boolean): void {\n        // @ts-ignore Need to improve typings of the 'testcafe-hammerhead' module\n        proxy[method](url, (req: IncomingMessage, res: ServerResponse, serverInfo, params: Dictionary<string>) => {\n            const connection = this._connections[params.id];\n\n            preventCaching(res);\n\n            if (shouldAcceptCrossOrigin)\n                acceptCrossOrigin(res);\n\n            if (connection)\n                handler(req, res, connection);\n            else\n                respond404(res);\n        });\n    }\n\n    private _registerRoutes (proxy: Proxy): void {\n        const {\n            idlePageScript,\n            idlePageStyle,\n            idlePageLogo,\n            serviceWorkerScript,\n        } = loadAssets();\n\n        this._dispatch(`${SERVICE_ROUTES.connect}/{id}`, proxy, BrowserConnectionGateway._onConnection);\n        this._dispatch(`${SERVICE_ROUTES.heartbeat}/{id}`, proxy, BrowserConnectionGateway._onHeartbeat, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.idle}/{id}`, proxy, BrowserConnectionGateway._onIdle);\n        this._dispatch(`${SERVICE_ROUTES.idleForced}/{id}`, proxy, BrowserConnectionGateway._onIdleForced);\n        this._dispatch(`${SERVICE_ROUTES.status}/{id}`, proxy, BrowserConnectionGateway._onStatusRequest);\n        this._dispatch(`${SERVICE_ROUTES.statusDone}/{id}`, proxy, BrowserConnectionGateway._onStatusRequestOnTestDone, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.initScript}/{id}`, proxy, BrowserConnectionGateway._onInitScriptRequest);\n        this._dispatch(`${SERVICE_ROUTES.initScript}/{id}`, proxy, BrowserConnectionGateway._onInitScriptResponse, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.activeWindowId}/{id}`, proxy, BrowserConnectionGateway._onGetActiveWindowIdRequest, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.activeWindowId}/{id}`, proxy, BrowserConnectionGateway._onSetActiveWindowIdRequest, 'POST', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.closeWindow}/{id}`, proxy, BrowserConnectionGateway._onCloseWindowRequest, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.openFileProtocol}/{id}`, proxy, BrowserConnectionGateway._onOpenFileProtocolRequest, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.dispatchProxylessEvent}/{id}`, proxy, BrowserConnectionGateway._onDispatchProxylessEvent, 'POST', this.proxyless);\n\n        proxy.GET(SERVICE_ROUTES.connect, (req: IncomingMessage, res: ServerResponse) => this._connectNextRemoteBrowser(req, res));\n        proxy.GET(SERVICE_ROUTES.connectWithTrailingSlash, (req: IncomingMessage, res: ServerResponse) => this._connectNextRemoteBrowser(req, res));\n\n        proxy.GET(SERVICE_ROUTES.serviceWorker, { content: serviceWorkerScript, contentType: 'application/x-javascript' });\n        proxy.GET(SERVICE_ROUTES.assets.index, { content: idlePageScript, contentType: 'application/x-javascript' });\n        proxy.GET(SERVICE_ROUTES.assets.styles, { content: idlePageStyle, contentType: 'text/css' });\n        proxy.GET(SERVICE_ROUTES.assets.logo, { content: idlePageLogo, contentType: 'image/svg+xml' });\n\n        if (this.proxyless) {\n            proxy.GET(PROXYLESS_ERROR_ROUTE, (req: IncomingMessage, res: ServerResponse) => {\n                res.writeHead(200, { 'Content-Type': 'text/html' });\n                res.end(EMPTY_PAGE_MARKUP);\n            });\n        }\n    }\n\n    // Helpers\n    private static _ensureConnectionReady (res: ServerResponse, connection: BrowserConnection): boolean {\n        if (!connection.isReady()) {\n            respond500(res, 'The connection is not ready yet.');\n            return false;\n        }\n\n        return true;\n    }\n\n    private static _fetchRequestData (req: IncomingMessage, callback: (data: string) => void): void {\n        let data = '';\n\n        req.on('data', chunk => {\n            data += chunk;\n        });\n\n        req.on('end', () => {\n            callback(data.toString());\n        });\n    }\n\n    // Route handlers\n    private static async _onConnection (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        if (connection.isReady())\n            respond500(res, 'The connection is already established.');\n\n        else {\n            const userAgent = req.headers['user-agent'] as string;\n\n            await connection.establish(userAgent);\n            redirect(res, connection.idleUrl);\n        }\n    }\n\n    private static _onHeartbeat (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = connection.heartbeat();\n\n            respondWithJSON(res, status);\n        }\n    }\n\n    private static _onIdle (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection))\n            res.end(connection.renderIdlePage());\n    }\n\n    private static async _onIdleForced (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = await connection.getStatus(true);\n\n            redirect(res, status.url);\n        }\n    }\n\n    private static async _onStatusRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        return BrowserConnectionGateway._onStatusRequestCore(req, res, connection, false);\n    }\n\n    private static async _onStatusRequestOnTestDone (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        return BrowserConnectionGateway._onStatusRequestCore(req, res, connection, true);\n    }\n\n    private static async _onStatusRequestCore (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection, isTestDone: boolean): Promise<void> {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = await connection.getStatus(isTestDone);\n\n            respondWithJSON(res, status);\n        }\n    }\n\n    private static _onInitScriptRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const script = connection.getInitScript();\n\n            respondWithJSON(res, script);\n        }\n    }\n\n    private static _onInitScriptResponse (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                connection.handleInitScriptResult(data);\n\n                res.end();\n            });\n        }\n    }\n\n    private static _onGetActiveWindowIdRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            respondWithJSON(res, {\n                activeWindowId: connection.activeWindowId,\n            });\n        }\n    }\n\n    private static _onSetActiveWindowIdRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const parsedData = JSON.parse(data);\n\n                connection.activeWindowId = parsedData.windowId;\n\n                respondWithJSON(res);\n            });\n        }\n    }\n\n    private static _onCloseWindowRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            connection.provider.closeBrowserChildWindow(connection.id)\n                .then(() => {\n                    respondWithJSON(res);\n                });\n        }\n    }\n\n    private static _onOpenFileProtocolRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const parsedData = JSON.parse(data);\n\n                connection.openFileProtocol(parsedData.url)\n                    .then(() => {\n                        respondWithJSON(res);\n                    });\n            });\n        }\n    }\n\n    private static _onDispatchProxylessEvent (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const { type, options } = JSON.parse(data);\n\n                connection.dispatchProxylessEvent(type, options)\n                    .then(() => {\n                        respondWithJSON(res);\n                    });\n            });\n        }\n    }\n\n    private async _connectNextRemoteBrowser (req: IncomingMessage, res: ServerResponse): Promise<void> {\n        preventCaching(res);\n\n        const remoteConnection = await this._remotesQueue.shift();\n\n        if (remoteConnection)\n            redirect(res, remoteConnection.url);\n        else\n            respond500(res, 'There are no available _connections to establish.');\n    }\n\n    // API\n    public startServingConnection (connection: BrowserConnection): void {\n        this._connections[connection.id] = connection;\n\n        if (connection.browserInfo.providerName === 'remote')\n            this._remotesQueue.add(connection);\n    }\n\n    public stopServingConnection (connection: BrowserConnection): void {\n        delete this._connections[connection.id];\n\n        if (connection.browserInfo.providerName === 'remote')\n            this._remotesQueue.remove(connection);\n    }\n\n    public async close (): Promise<void> {\n        for (const id in this._connections)\n            await this._connections[id].close();\n    }\n}\n\n"]}
247
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gateway.js","sourceRoot":"","sources":["../../../src/browser/connection/gateway.ts"],"names":[],"mappings":"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","sourcesContent":["import loadAssets from '../../load-assets';\nimport {\n    respond404,\n    respond500,\n    respondWithJSON,\n    redirect,\n    preventCaching,\n} from '../../utils/http';\n\nimport RemotesQueue from './remotes-queue';\nimport { Proxy, acceptCrossOrigin } from 'testcafe-hammerhead';\nimport { Dictionary } from '../../configuration/interfaces';\nimport BrowserConnection from './index';\nimport { IncomingMessage, ServerResponse } from 'http';\nimport SERVICE_ROUTES from './service-routes';\nimport EMPTY_PAGE_MARKUP from '../../proxyless/empty-page-markup';\nimport PROXYLESS_ERROR_ROUTE from '../../proxyless/error-route';\nimport { initSelector } from '../../test-run/commands/validations/initializers';\nimport TestRun from '../../test-run';\n\nexport default class BrowserConnectionGateway {\n    private _connections: Dictionary<BrowserConnection> = {};\n    private _remotesQueue: RemotesQueue;\n    public readonly connectUrl: string;\n    public retryTestPages: boolean;\n    private readonly proxyless: boolean;\n    public readonly proxy: Proxy;\n\n    public constructor (proxy: Proxy, options: { retryTestPages: boolean; proxyless: boolean }) {\n        this._remotesQueue  = new RemotesQueue();\n        this.connectUrl     = proxy.resolveRelativeServiceUrl(SERVICE_ROUTES.connect);\n        this.retryTestPages = options.retryTestPages;\n        this.proxyless      = options.proxyless;\n        this.proxy          = proxy;\n\n        this._registerRoutes(proxy);\n    }\n\n    private _dispatch (url: string, proxy: Proxy, handler: Function, method = 'GET', shouldAcceptCrossOrigin?: boolean): void {\n        // @ts-ignore Need to improve typings of the 'testcafe-hammerhead' module\n        proxy[method](url, (req: IncomingMessage, res: ServerResponse, serverInfo, params: Dictionary<string>) => {\n            const connection = this._connections[params.id];\n\n            preventCaching(res);\n\n            if (shouldAcceptCrossOrigin)\n                acceptCrossOrigin(res);\n\n            if (connection)\n                handler(req, res, connection);\n            else\n                respond404(res);\n        });\n    }\n\n    private _registerRoutes (proxy: Proxy): void {\n        const {\n            idlePageScript,\n            idlePageStyle,\n            idlePageLogo,\n            serviceWorkerScript,\n        } = loadAssets();\n\n        this._dispatch(`${SERVICE_ROUTES.connect}/{id}`, proxy, BrowserConnectionGateway._onConnection);\n        this._dispatch(`${SERVICE_ROUTES.heartbeat}/{id}`, proxy, BrowserConnectionGateway._onHeartbeat, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.idle}/{id}`, proxy, BrowserConnectionGateway._onIdle);\n        this._dispatch(`${SERVICE_ROUTES.idleForced}/{id}`, proxy, BrowserConnectionGateway._onIdleForced);\n        this._dispatch(`${SERVICE_ROUTES.status}/{id}`, proxy, BrowserConnectionGateway._onStatusRequest);\n        this._dispatch(`${SERVICE_ROUTES.statusDone}/{id}`, proxy, BrowserConnectionGateway._onStatusRequestOnTestDone, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.initScript}/{id}`, proxy, BrowserConnectionGateway._onInitScriptRequest);\n        this._dispatch(`${SERVICE_ROUTES.initScript}/{id}`, proxy, BrowserConnectionGateway._onInitScriptResponse, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.activeWindowId}/{id}`, proxy, BrowserConnectionGateway._onGetActiveWindowIdRequest, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.activeWindowId}/{id}`, proxy, BrowserConnectionGateway._onSetActiveWindowIdRequest, 'POST', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.closeWindow}/{id}`, proxy, BrowserConnectionGateway._onCloseWindowRequest, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.openFileProtocol}/{id}`, proxy, BrowserConnectionGateway._onOpenFileProtocolRequest, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.dispatchProxylessEvent}/{id}`, proxy, BrowserConnectionGateway._onDispatchProxylessEvent, 'POST', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.parseSelector}/{id}`, proxy, BrowserConnectionGateway._parseSelector, 'POST', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.dispatchProxylessEventSequence}/{id}`, proxy, BrowserConnectionGateway._onDispatchProxylessEventSequence, 'POST', this.proxyless);\n\n        proxy.GET(SERVICE_ROUTES.connect, (req: IncomingMessage, res: ServerResponse) => this._connectNextRemoteBrowser(req, res));\n        proxy.GET(SERVICE_ROUTES.connectWithTrailingSlash, (req: IncomingMessage, res: ServerResponse) => this._connectNextRemoteBrowser(req, res));\n\n        proxy.GET(SERVICE_ROUTES.serviceWorker, { content: serviceWorkerScript, contentType: 'application/x-javascript' });\n        proxy.GET(SERVICE_ROUTES.assets.index, { content: idlePageScript, contentType: 'application/x-javascript' });\n        proxy.GET(SERVICE_ROUTES.assets.styles, { content: idlePageStyle, contentType: 'text/css' });\n        proxy.GET(SERVICE_ROUTES.assets.logo, { content: idlePageLogo, contentType: 'image/svg+xml' });\n\n        if (this.proxyless) {\n            proxy.GET(PROXYLESS_ERROR_ROUTE, (req: IncomingMessage, res: ServerResponse) => {\n                res.writeHead(200, { 'Content-Type': 'text/html' });\n                res.end(EMPTY_PAGE_MARKUP);\n            });\n        }\n    }\n\n    // Helpers\n    private static _ensureConnectionReady (res: ServerResponse, connection: BrowserConnection): boolean {\n        if (!connection.isReady()) {\n            respond500(res, 'The connection is not ready yet.');\n            return false;\n        }\n\n        return true;\n    }\n\n    private static _fetchRequestData (req: IncomingMessage, callback: (data: string) => void): void {\n        let data = '';\n\n        req.on('data', chunk => {\n            data += chunk;\n        });\n\n        req.on('end', () => {\n            callback(data.toString());\n        });\n    }\n\n    // Route handlers\n    private static async _onConnection (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        if (connection.isReady())\n            respond500(res, 'The connection is already established.');\n\n        else {\n            const userAgent = req.headers['user-agent'] as string;\n\n            await connection.establish(userAgent);\n            redirect(res, connection.idleUrl);\n        }\n    }\n\n    private static _onHeartbeat (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = connection.heartbeat();\n\n            respondWithJSON(res, status);\n        }\n    }\n\n    private static _onIdle (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection))\n            res.end(connection.renderIdlePage());\n    }\n\n    private static async _onIdleForced (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = await connection.getStatus(true);\n\n            redirect(res, status.url);\n        }\n    }\n\n    private static async _onStatusRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        return BrowserConnectionGateway._onStatusRequestCore(req, res, connection, false);\n    }\n\n    private static async _onStatusRequestOnTestDone (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        return BrowserConnectionGateway._onStatusRequestCore(req, res, connection, true);\n    }\n\n    private static async _onStatusRequestCore (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection, isTestDone: boolean): Promise<void> {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = await connection.getStatus(isTestDone);\n\n            respondWithJSON(res, status);\n        }\n    }\n\n    private static _onInitScriptRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const script = connection.getInitScript();\n\n            respondWithJSON(res, script);\n        }\n    }\n\n    private static _onInitScriptResponse (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                connection.handleInitScriptResult(data);\n\n                res.end();\n            });\n        }\n    }\n\n    private static _onGetActiveWindowIdRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            respondWithJSON(res, {\n                activeWindowId: connection.activeWindowId,\n            });\n        }\n    }\n\n    private static _onSetActiveWindowIdRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const parsedData = JSON.parse(data);\n\n                connection.activeWindowId = parsedData.windowId;\n\n                respondWithJSON(res);\n            });\n        }\n    }\n\n    private static _onCloseWindowRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            connection.provider.closeBrowserChildWindow(connection.id)\n                .then(() => {\n                    respondWithJSON(res);\n                });\n        }\n    }\n\n    private static _onOpenFileProtocolRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const parsedData = JSON.parse(data);\n\n                connection.openFileProtocol(parsedData.url)\n                    .then(() => {\n                        respondWithJSON(res);\n                    });\n            });\n        }\n    }\n\n    private static _onDispatchProxylessEvent (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const { type, options } = JSON.parse(data);\n\n                connection.dispatchProxylessEvent(type, options)\n                    .then(() => {\n                        respondWithJSON(res);\n                    });\n            });\n        }\n    }\n\n    private static _onDispatchProxylessEventSequence (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const eventSequence = JSON.parse(data);\n\n                connection.dispatchProxylessEventSequence(eventSequence)\n                    .then(() => {\n                        respondWithJSON(res);\n                    });\n            });\n        }\n    }\n\n    private async _connectNextRemoteBrowser (req: IncomingMessage, res: ServerResponse): Promise<void> {\n        preventCaching(res);\n\n        const remoteConnection = await this._remotesQueue.shift();\n\n        if (remoteConnection)\n            redirect(res, remoteConnection.url);\n        else\n            respond500(res, 'There are no available _connections to establish.');\n    }\n\n    private static _getParsedSelector (testRun: TestRun, rawSelector: string): any {\n        const options = {\n            testRun,\n\n            skipVisibilityCheck: true,\n            collectionMode:      true,\n        };\n\n        const value    = rawSelector.trim().startsWith('Selector(') ? rawSelector : `'${rawSelector}'`;\n        const selector = { type: 'js-expr', value };\n\n        return initSelector('selector', selector, options);\n    }\n\n    private static _parseSelector (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                try {\n                    const testRun        = connection.getCurrentTestRun();\n                    const rawSelector    = JSON.parse(data).selector;\n                    const parsedSelector = BrowserConnectionGateway._getParsedSelector(testRun, rawSelector);\n\n                    respondWithJSON(res, parsedSelector);\n                }\n                catch (error) {\n                    respondWithJSON(res);\n                }\n            });\n        }\n    }\n\n    // API\n    public startServingConnection (connection: BrowserConnection): void {\n        this._connections[connection.id] = connection;\n\n        if (connection.browserInfo.providerName === 'remote')\n            this._remotesQueue.add(connection);\n    }\n\n    public stopServingConnection (connection: BrowserConnection): void {\n        delete this._connections[connection.id];\n\n        if (connection.browserInfo.providerName === 'remote')\n            this._remotesQueue.remove(connection);\n    }\n\n    public async close (): Promise<void> {\n        for (const id in this._connections)\n            await this._connections[id].close();\n    }\n\n    public getConnections (): Dictionary<BrowserConnection> {\n        return this._connections;\n    }\n}\n\n"]}