@appium/base-driver 8.3.0 → 8.4.1

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 (202) hide show
  1. package/build/lib/basedriver/capabilities.d.ts +80 -0
  2. package/build/lib/basedriver/capabilities.d.ts.map +1 -0
  3. package/build/lib/basedriver/capabilities.js +12 -13
  4. package/build/lib/basedriver/commands/event.d.ts +9 -0
  5. package/build/lib/basedriver/commands/event.d.ts.map +1 -0
  6. package/build/lib/basedriver/commands/event.js +21 -23
  7. package/build/lib/basedriver/commands/find.d.ts +11 -0
  8. package/build/lib/basedriver/commands/find.d.ts.map +1 -0
  9. package/build/lib/basedriver/commands/find.js +42 -37
  10. package/build/lib/basedriver/commands/index.d.ts +8 -0
  11. package/build/lib/basedriver/commands/index.d.ts.map +1 -0
  12. package/build/lib/basedriver/commands/index.js +17 -16
  13. package/build/lib/basedriver/commands/log.d.ts +12 -0
  14. package/build/lib/basedriver/commands/log.d.ts.map +1 -0
  15. package/build/lib/basedriver/commands/log.js +23 -27
  16. package/build/lib/basedriver/commands/session.d.ts +11 -0
  17. package/build/lib/basedriver/commands/session.d.ts.map +1 -0
  18. package/build/lib/basedriver/commands/session.js +21 -149
  19. package/build/lib/basedriver/commands/settings.d.ts +10 -0
  20. package/build/lib/basedriver/commands/settings.d.ts.map +1 -0
  21. package/build/lib/basedriver/commands/settings.js +19 -19
  22. package/build/lib/basedriver/commands/timeout.d.ts +8 -0
  23. package/build/lib/basedriver/commands/timeout.d.ts.map +1 -0
  24. package/build/lib/basedriver/commands/timeout.js +118 -139
  25. package/build/lib/basedriver/core.d.ts +235 -0
  26. package/build/lib/basedriver/core.d.ts.map +1 -0
  27. package/build/lib/basedriver/core.js +283 -0
  28. package/build/lib/basedriver/desired-caps.d.ts +5 -0
  29. package/build/lib/basedriver/desired-caps.d.ts.map +1 -0
  30. package/build/lib/basedriver/desired-caps.js +2 -4
  31. package/build/lib/basedriver/device-settings.d.ts +32 -0
  32. package/build/lib/basedriver/device-settings.d.ts.map +1 -0
  33. package/build/lib/basedriver/device-settings.js +20 -14
  34. package/build/lib/basedriver/driver.d.ts +83 -0
  35. package/build/lib/basedriver/driver.d.ts.map +1 -0
  36. package/build/lib/basedriver/driver.js +101 -256
  37. package/build/lib/basedriver/helpers.d.ts +132 -0
  38. package/build/lib/basedriver/helpers.d.ts.map +1 -0
  39. package/build/lib/basedriver/helpers.js +16 -2
  40. package/build/lib/basedriver/logger.d.ts +3 -0
  41. package/build/lib/basedriver/logger.d.ts.map +1 -0
  42. package/build/lib/basedriver/logger.js +2 -4
  43. package/build/lib/constants.d.ts +9 -0
  44. package/build/lib/constants.d.ts.map +1 -0
  45. package/build/lib/constants.js +2 -4
  46. package/build/lib/express/crash.d.ts +3 -0
  47. package/build/lib/express/crash.d.ts.map +1 -0
  48. package/build/lib/express/crash.js +2 -4
  49. package/build/lib/express/express-logging.d.ts +3 -0
  50. package/build/lib/express/express-logging.d.ts.map +1 -0
  51. package/build/lib/express/express-logging.js +2 -4
  52. package/build/lib/express/idempotency.d.ts +2 -0
  53. package/build/lib/express/idempotency.d.ts.map +1 -0
  54. package/build/lib/express/idempotency.js +2 -4
  55. package/build/lib/express/logger.d.ts +3 -0
  56. package/build/lib/express/logger.d.ts.map +1 -0
  57. package/build/lib/express/logger.js +2 -4
  58. package/build/lib/express/middleware.d.ts +9 -0
  59. package/build/lib/express/middleware.d.ts.map +1 -0
  60. package/build/lib/express/middleware.js +2 -4
  61. package/build/lib/express/server.d.ts +10 -0
  62. package/build/lib/express/server.d.ts.map +1 -0
  63. package/build/lib/express/server.js +2 -4
  64. package/build/lib/express/static.d.ts +6 -0
  65. package/build/lib/express/static.d.ts.map +1 -0
  66. package/build/lib/express/static.js +2 -4
  67. package/build/lib/express/websocket.d.ts +64 -0
  68. package/build/lib/express/websocket.d.ts.map +1 -0
  69. package/build/lib/express/websocket.js +40 -41
  70. package/build/lib/helpers/capabilities.d.ts +13 -0
  71. package/build/lib/helpers/capabilities.d.ts.map +1 -0
  72. package/build/lib/helpers/capabilities.js +40 -2
  73. package/build/lib/index.d.ts +183 -0
  74. package/build/lib/index.d.ts.map +1 -0
  75. package/build/lib/index.js +41 -23
  76. package/build/lib/jsonwp-proxy/protocol-converter.d.ts +48 -0
  77. package/build/lib/jsonwp-proxy/protocol-converter.d.ts.map +1 -0
  78. package/build/lib/jsonwp-proxy/protocol-converter.js +2 -4
  79. package/build/lib/jsonwp-proxy/proxy.d.ts +41 -0
  80. package/build/lib/jsonwp-proxy/proxy.d.ts.map +1 -0
  81. package/build/lib/jsonwp-proxy/proxy.js +25 -9
  82. package/build/lib/jsonwp-status/status.d.ts +159 -0
  83. package/build/lib/jsonwp-status/status.d.ts.map +1 -0
  84. package/build/lib/jsonwp-status/status.js +2 -4
  85. package/build/lib/protocol/errors.d.ts +310 -0
  86. package/build/lib/protocol/errors.d.ts.map +1 -0
  87. package/build/lib/protocol/errors.js +82 -5
  88. package/build/lib/protocol/helpers.d.ts +22 -0
  89. package/build/lib/protocol/helpers.d.ts.map +1 -0
  90. package/build/lib/protocol/helpers.js +2 -4
  91. package/build/lib/protocol/index.d.ts +16 -0
  92. package/build/lib/protocol/index.d.ts.map +1 -0
  93. package/build/lib/protocol/index.js +8 -10
  94. package/build/lib/protocol/protocol.d.ts +11 -0
  95. package/build/lib/protocol/protocol.d.ts.map +1 -0
  96. package/build/lib/protocol/protocol.js +3 -9
  97. package/build/lib/protocol/routes.d.ts +6 -0
  98. package/build/lib/protocol/routes.d.ts.map +1 -0
  99. package/build/lib/protocol/routes.js +18 -4
  100. package/build/lib/protocol/validators.d.ts +8 -0
  101. package/build/lib/protocol/validators.d.ts.map +1 -0
  102. package/build/lib/protocol/validators.js +2 -4
  103. package/build/test/basedriver/README.md +5 -0
  104. package/build/test/basedriver/driver-e2e-tests.js +2 -4
  105. package/build/test/basedriver/driver-tests.js +31 -6
  106. package/build/test/basedriver/index.js +2 -4
  107. package/build/test/e2e/basedriver/driver.e2e.spec.js +15 -0
  108. package/build/test/e2e/basedriver/helpers.e2e.spec.js +192 -0
  109. package/build/test/e2e/basedriver/websockets.e2e.spec.js +87 -0
  110. package/build/test/e2e/express/server.e2e.spec.js +159 -0
  111. package/build/test/e2e/jsonwp-proxy/proxy.e2e.spec.js +59 -0
  112. package/build/test/e2e/protocol/fake-driver.js +163 -0
  113. package/build/test/e2e/protocol/helpers.js +25 -0
  114. package/build/test/e2e/protocol/protocol.e2e.spec.js +1186 -0
  115. package/build/test/helpers.js +2 -4
  116. package/build/test/unit/basedriver/capabilities.spec.js +672 -0
  117. package/build/test/unit/basedriver/capability.spec.js +353 -0
  118. package/build/test/unit/basedriver/commands/event.spec.js +110 -0
  119. package/build/test/unit/basedriver/commands/log.spec.js +92 -0
  120. package/build/test/unit/basedriver/driver.spec.js +15 -0
  121. package/build/test/unit/basedriver/helpers.spec.js +151 -0
  122. package/build/test/unit/basedriver/timeout.spec.js +135 -0
  123. package/build/test/unit/express/server.spec.js +155 -0
  124. package/build/test/unit/express/static.spec.js +26 -0
  125. package/build/test/unit/jsonwp-proxy/mock-request.js +91 -0
  126. package/build/test/unit/jsonwp-proxy/protocol-converter.spec.js +171 -0
  127. package/build/test/unit/jsonwp-proxy/proxy.spec.js +292 -0
  128. package/build/test/unit/jsonwp-proxy/url.spec.js +165 -0
  129. package/build/test/unit/jsonwp-status/status.spec.js +34 -0
  130. package/build/test/unit/protocol/errors.spec.js +390 -0
  131. package/build/test/unit/protocol/routes.spec.js +80 -0
  132. package/build/test/unit/protocol/validator.spec.js +149 -0
  133. package/build/tsconfig.tsbuildinfo +1 -0
  134. package/lib/basedriver/capabilities.js +49 -10
  135. package/lib/basedriver/commands/event.js +49 -31
  136. package/lib/basedriver/commands/find.js +108 -43
  137. package/lib/basedriver/commands/index.js +25 -19
  138. package/lib/basedriver/commands/log.js +60 -33
  139. package/lib/basedriver/commands/session.js +39 -141
  140. package/lib/basedriver/commands/settings.js +33 -13
  141. package/lib/basedriver/commands/timeout.js +153 -153
  142. package/lib/basedriver/core.js +497 -0
  143. package/lib/basedriver/desired-caps.js +1 -1
  144. package/lib/basedriver/device-settings.js +47 -12
  145. package/lib/basedriver/driver.js +272 -383
  146. package/lib/basedriver/helpers.js +18 -2
  147. package/lib/express/websocket.js +35 -32
  148. package/lib/helpers/capabilities.js +60 -1
  149. package/lib/index.js +16 -12
  150. package/lib/jsonwp-proxy/proxy.js +26 -6
  151. package/lib/protocol/errors.js +42 -42
  152. package/lib/protocol/index.js +4 -4
  153. package/lib/protocol/protocol.js +1 -3
  154. package/lib/protocol/routes.js +9 -0
  155. package/package.json +22 -14
  156. package/test/basedriver/README.md +5 -0
  157. package/test/basedriver/driver-e2e-tests.js +1 -1
  158. package/test/basedriver/driver-tests.js +31 -2
  159. package/build/test/basedriver/capabilities-specs.js +0 -674
  160. package/build/test/basedriver/capability-specs.js +0 -355
  161. package/build/test/basedriver/commands/event-specs.js +0 -112
  162. package/build/test/basedriver/commands/log-specs.js +0 -87
  163. package/build/test/basedriver/driver-e2e-specs.js +0 -17
  164. package/build/test/basedriver/driver-specs.js +0 -17
  165. package/build/test/basedriver/helpers-e2e-specs.js +0 -194
  166. package/build/test/basedriver/helpers-specs.js +0 -153
  167. package/build/test/basedriver/timeout-specs.js +0 -137
  168. package/build/test/basedriver/websockets-e2e-specs.js +0 -84
  169. package/build/test/express/server-e2e-specs.js +0 -161
  170. package/build/test/express/server-specs.js +0 -157
  171. package/build/test/express/static-specs.js +0 -28
  172. package/build/test/jsonwp-proxy/mock-request.js +0 -93
  173. package/build/test/jsonwp-proxy/protocol-converter-specs.js +0 -173
  174. package/build/test/jsonwp-proxy/proxy-e2e-specs.js +0 -61
  175. package/build/test/jsonwp-proxy/proxy-specs.js +0 -294
  176. package/build/test/jsonwp-proxy/url-specs.js +0 -167
  177. package/build/test/jsonwp-status/status-specs.js +0 -36
  178. package/build/test/protocol/errors-specs.js +0 -388
  179. package/build/test/protocol/fake-driver.js +0 -165
  180. package/build/test/protocol/helpers.js +0 -27
  181. package/build/test/protocol/protocol-e2e-specs.js +0 -1188
  182. package/build/test/protocol/routes-specs.js +0 -82
  183. package/build/test/protocol/validator-specs.js +0 -151
  184. package/index.d.ts +0 -386
  185. package/test/basedriver/capabilities-specs.js +0 -537
  186. package/test/basedriver/capability-specs.js +0 -383
  187. package/test/basedriver/commands/event-specs.js +0 -74
  188. package/test/basedriver/commands/log-specs.js +0 -79
  189. package/test/basedriver/driver-e2e-specs.js +0 -8
  190. package/test/basedriver/driver-specs.js +0 -8
  191. package/test/basedriver/fixtures/BadZippedApp.zip +0 -1
  192. package/test/basedriver/fixtures/FakeAndroidApp.apk +0 -1
  193. package/test/basedriver/fixtures/FakeAndroidApp.asd +0 -0
  194. package/test/basedriver/fixtures/FakeIOSApp.app +0 -1
  195. package/test/basedriver/fixtures/FakeIOSApp.app.zip +0 -0
  196. package/test/basedriver/fixtures/FakeIOSApp.ipa +0 -0
  197. package/test/basedriver/fixtures/custom-element-finder-bad.js +0 -5
  198. package/test/basedriver/fixtures/custom-element-finder.js +0 -29
  199. package/test/basedriver/helpers-e2e-specs.js +0 -187
  200. package/test/basedriver/helpers-specs.js +0 -137
  201. package/test/basedriver/timeout-specs.js +0 -128
  202. package/test/basedriver/websockets-e2e-specs.js +0 -75
@@ -1,1188 +0,0 @@
1
- "use strict";
2
-
3
- var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
-
5
- require("source-map-support/register");
6
-
7
- var _lib = require("../../lib");
8
-
9
- var _fakeDriver = require("./fake-driver");
10
-
11
- var _axios = _interopRequireDefault(require("axios"));
12
-
13
- var _sinon = require("sinon");
14
-
15
- var _httpStatusCodes = require("http-status-codes");
16
-
17
- var _helpers = require("./helpers");
18
-
19
- var _constants = require("../../lib/constants");
20
-
21
- var _helpers2 = require("../helpers");
22
-
23
- let port;
24
- let baseUrl;
25
- describe('Protocol', function () {
26
- let sandbox;
27
- beforeEach(function () {
28
- sandbox = (0, _sinon.createSandbox)();
29
- });
30
- afterEach(function () {
31
- sandbox.restore();
32
- });
33
- before(async function () {
34
- port = await (0, _helpers2.getTestPort)();
35
- baseUrl = `http://${_helpers2.TEST_HOST}:${port}`;
36
- });
37
- describe('direct to driver', function () {
38
- let d = new _fakeDriver.FakeDriver();
39
- it('should return response values directly from the driver', async function () {
40
- (await d.setUrl('http://google.com')).should.contain('google');
41
- });
42
- });
43
- describe('via express router', function () {
44
- let mjsonwpServer;
45
- let driver;
46
- before(async function () {
47
- driver = new _fakeDriver.FakeDriver();
48
- driver.sessionId = 'foo';
49
- mjsonwpServer = await (0, _lib.server)({
50
- routeConfiguringFunction: (0, _lib.routeConfiguringFunction)(driver),
51
- port
52
- });
53
- });
54
- after(async function () {
55
- await mjsonwpServer.close();
56
- });
57
- it('should proxy to driver and return valid jsonwp response', async function () {
58
- const {
59
- data
60
- } = await (0, _axios.default)({
61
- url: `${baseUrl}/session/foo/url`,
62
- method: 'POST',
63
- data: {
64
- url: 'http://google.com'
65
- }
66
- });
67
- data.should.eql({
68
- value: 'Navigated to: http://google.com',
69
- sessionId: 'foo'
70
- });
71
- });
72
- it('should assume requests without a Content-Type are json requests', async function () {
73
- const {
74
- data
75
- } = await (0, _axios.default)({
76
- url: `${baseUrl}/session/foo/url`,
77
- method: 'POST',
78
- data: {
79
- url: 'http://google.com'
80
- }
81
- });
82
- data.should.eql({
83
- value: 'Navigated to: http://google.com',
84
- sessionId: 'foo'
85
- });
86
- });
87
- it('should respond to x-www-form-urlencoded as well as json requests', async function () {
88
- const reqData = new URLSearchParams();
89
- reqData.set('url', 'http://google.com');
90
- const {
91
- data
92
- } = await (0, _axios.default)({
93
- url: `${baseUrl}/session/foo/url`,
94
- headers: {
95
- 'Content-Type': 'application/x-www-form-urlencoded'
96
- },
97
- method: 'POST',
98
- data: reqData.toString()
99
- });
100
- data.should.eql({
101
- value: 'Navigated to: http://google.com',
102
- sessionId: 'foo'
103
- });
104
- });
105
- it('should include url request parameters for methods to use - sessionid', async function () {
106
- const {
107
- data
108
- } = await (0, _axios.default)({
109
- url: `${baseUrl}/session/foo/back`,
110
- method: 'POST',
111
- data: {}
112
- });
113
- data.should.eql({
114
- value: 'foo',
115
- sessionId: 'foo'
116
- });
117
- });
118
- it('should include url request parameters for methods to use - elementid', async function () {
119
- const {
120
- data
121
- } = await (0, _axios.default)({
122
- url: `${baseUrl}/session/foo/element/bar/click`,
123
- method: 'POST',
124
- data: {}
125
- });
126
- data.value.should.eql(['bar', 'foo']);
127
- });
128
- it('should include url req params in the order: custom, element, session', async function () {
129
- const {
130
- data
131
- } = await (0, _axios.default)({
132
- url: `${baseUrl}/session/foo/element/bar/attribute/baz`
133
- });
134
- data.value.should.eql(['baz', 'bar', 'foo']);
135
- });
136
- it('should respond with 400 Bad Request if parameters missing', async function () {
137
- const {
138
- data,
139
- status
140
- } = await (0, _axios.default)({
141
- url: `${baseUrl}/session/foo/url`,
142
- method: 'POST',
143
- data: {},
144
- validateStatus: null
145
- });
146
- status.should.equal(400);
147
- JSON.stringify(data).should.contain('url');
148
- });
149
- it('should reject requests with a badly formatted body and not crash', async function () {
150
- await (0, _axios.default)({
151
- url: `${baseUrl}/session/foo/url`,
152
- method: 'POST',
153
- data: 'oh hello'
154
- }).should.eventually.be.rejected;
155
- const {
156
- data
157
- } = await (0, _axios.default)({
158
- url: `${baseUrl}/session/foo/url`,
159
- method: 'POST',
160
- data: {
161
- url: 'http://google.com'
162
- }
163
- });
164
- data.should.eql({
165
- value: 'Navigated to: http://google.com',
166
- sessionId: 'foo'
167
- });
168
- });
169
- it('should get 404 for bad routes', async function () {
170
- await (0, _axios.default)({
171
- url: `${baseUrl}/blargimarg`
172
- }).should.eventually.be.rejectedWith(/404/);
173
- });
174
- it('4xx responses should have content-type of application/json', async function () {
175
- const {
176
- headers
177
- } = await (0, _axios.default)({
178
- url: `${baseUrl}/blargimargarita`,
179
- validateStatus: null
180
- });
181
- headers['content-type'].should.include('application/json');
182
- });
183
- it('should throw not yet implemented for unfilledout commands', async function () {
184
- const {
185
- status,
186
- data
187
- } = await (0, _axios.default)({
188
- url: `${baseUrl}/session/foo/element/bar/location`,
189
- validateStatus: null
190
- });
191
- status.should.equal(405);
192
- data.value.error.should.eql('unknown method');
193
- data.value.message.should.eql('Method has not yet been implemented');
194
- data.sessionId.should.eql('foo');
195
- });
196
- it('should throw not implemented for ignored commands', async function () {
197
- const {
198
- status,
199
- data
200
- } = await (0, _axios.default)({
201
- url: `${baseUrl}/session/foo/buttonup`,
202
- method: 'POST',
203
- validateStatus: null,
204
- data: {}
205
- });
206
- status.should.equal(405);
207
- data.value.error.should.eql('unknown method');
208
- data.value.message.should.eql('Method has not yet been implemented');
209
- data.sessionId.should.eql('foo');
210
- });
211
- it('should get 400 for bad parameters', async function () {
212
- await (0, _axios.default)({
213
- url: `${baseUrl}/session/foo/url`,
214
- method: 'POST',
215
- data: {}
216
- }).should.eventually.be.rejectedWith(/400/);
217
- });
218
- it('should ignore special extra payload params in the right contexts', async function () {
219
- await (0, _axios.default)({
220
- url: `${baseUrl}/session/foo/element/bar/value`,
221
- method: 'POST',
222
- data: {
223
- id: 'baz',
224
- sessionId: 'lol',
225
- value: ['a']
226
- }
227
- });
228
- await (0, _axios.default)({
229
- url: `${baseUrl}/session/foo/element/bar/value`,
230
- method: 'POST',
231
- data: {
232
- id: 'baz'
233
- }
234
- }).should.eventually.be.rejectedWith(/400/);
235
- await (0, _axios.default)({
236
- url: `${baseUrl}/session/foo/frame`,
237
- method: 'POST',
238
- data: {
239
- id: 'baz'
240
- }
241
- });
242
- });
243
- it('should return the correct error even if driver does not throw', async function () {
244
- const {
245
- status,
246
- data
247
- } = await (0, _axios.default)({
248
- url: `${baseUrl}/session/foo/appium/receive_async_response`,
249
- method: 'POST',
250
- data: {
251
- response: 'baz'
252
- },
253
- validateStatus: null
254
- });
255
- status.should.equal(500);
256
- data.value.error.should.eql('unknown error');
257
- data.value.message.should.eql('An unknown server-side error occurred while processing ' + 'the command. Original error: Mishandled Driver Error');
258
- data.sessionId.should.eql('foo');
259
- });
260
- describe('w3c sendkeys migration', function () {
261
- it('should accept value for sendkeys', async function () {
262
- const {
263
- data
264
- } = await (0, _axios.default)({
265
- url: `${baseUrl}/session/foo/element/bar/value`,
266
- method: 'POST',
267
- data: {
268
- value: 'text to type'
269
- }
270
- });
271
- data.value.should.eql(['text to type', 'bar']);
272
- });
273
- it('should accept text for sendkeys', async function () {
274
- const {
275
- data
276
- } = await (0, _axios.default)({
277
- url: `${baseUrl}/session/foo/element/bar/value`,
278
- method: 'POST',
279
- data: {
280
- text: 'text to type'
281
- }
282
- });
283
- data.value.should.eql(['text to type', 'bar']);
284
- });
285
- it('should accept value and text for sendkeys, and use value', async function () {
286
- const {
287
- data
288
- } = await (0, _axios.default)({
289
- url: `${baseUrl}/session/foo/element/bar/value`,
290
- method: 'POST',
291
- data: {
292
- value: 'text to type',
293
- text: 'text to ignore'
294
- }
295
- });
296
- data.value.should.eql(['text to type', 'bar']);
297
- });
298
- });
299
- describe('multiple sets of arguments', function () {
300
- describe('optional', function () {
301
- it('should allow moveto with element', async function () {
302
- const {
303
- data
304
- } = await (0, _axios.default)({
305
- url: `${baseUrl}/session/foo/moveto`,
306
- method: 'POST',
307
- data: {
308
- element: '3'
309
- }
310
- });
311
- data.value.should.eql(['3', null, null]);
312
- });
313
- it('should allow moveto with xoffset/yoffset', async function () {
314
- const {
315
- data
316
- } = await (0, _axios.default)({
317
- url: `${baseUrl}/session/foo/moveto`,
318
- method: 'POST',
319
- data: {
320
- xoffset: 42,
321
- yoffset: 17
322
- }
323
- });
324
- data.value.should.eql([null, 42, 17]);
325
- });
326
- });
327
- describe('required', function () {
328
- it('should allow removeApp with appId', async function () {
329
- const {
330
- data
331
- } = await (0, _axios.default)({
332
- url: `${baseUrl}/session/foo/appium/device/remove_app`,
333
- method: 'POST',
334
- data: {
335
- appId: 42
336
- }
337
- });
338
- data.value.should.eql(42);
339
- });
340
- it('should allow removeApp with bundleId', async function () {
341
- const {
342
- data
343
- } = await (0, _axios.default)({
344
- url: `${baseUrl}/session/foo/appium/device/remove_app`,
345
- method: 'POST',
346
- data: {
347
- bundleId: 42
348
- }
349
- });
350
- data.value.should.eql(42);
351
- });
352
- });
353
- });
354
- describe('default param wrap', function () {
355
- it('should wrap', async function () {
356
- const {
357
- data
358
- } = await (0, _axios.default)({
359
- url: `${baseUrl}/session/foo/touch/perform`,
360
- method: 'POST',
361
- data: [{
362
- 'action': 'tap',
363
- 'options': {
364
- 'element': '3'
365
- }
366
- }]
367
- });
368
- data.value.should.deep.equal([[{
369
- 'action': 'tap',
370
- 'options': {
371
- 'element': '3'
372
- }
373
- }], 'foo']);
374
- });
375
- it('should not wrap twice', async function () {
376
- const {
377
- data
378
- } = await (0, _axios.default)({
379
- url: `${baseUrl}/session/foo/touch/perform`,
380
- method: 'POST',
381
- data: {
382
- actions: [{
383
- 'action': 'tap',
384
- 'options': {
385
- 'element': '3'
386
- }
387
- }]
388
- }
389
- });
390
- data.value.should.deep.equal([[{
391
- 'action': 'tap',
392
- 'options': {
393
- 'element': '3'
394
- }
395
- }], 'foo']);
396
- });
397
- });
398
- describe('create sessions via HTTP endpoint', function () {
399
- let sessionId;
400
- beforeEach(function () {
401
- sessionId = null;
402
- });
403
- afterEach(async function () {
404
- if (sessionId) {
405
- await _axios.default.delete(`${baseUrl}/session/${sessionId}`);
406
- }
407
- });
408
- it('should not allow create session with desired caps (MJSONWP)', async function () {
409
- const desiredCapabilities = {
410
- a: 'b'
411
- };
412
- const {
413
- data
414
- } = await (0, _axios.default)({
415
- url: `${baseUrl}/session`,
416
- method: 'POST',
417
- data: {
418
- desiredCapabilities
419
- }
420
- });
421
- should.equal(data.value, null);
422
- });
423
- it('should fail to create session without capabilities', async function () {
424
- await (0, _axios.default)({
425
- url: `${baseUrl}/session`,
426
- method: 'POST',
427
- data: {}
428
- }).should.eventually.be.rejectedWith(/400/);
429
- });
430
- it('should allow create session with capabilities (W3C)', async function () {
431
- const w3cCapabilities = {
432
- alwaysMatch: {
433
- 'appium:e': 'f'
434
- }
435
- };
436
- const {
437
- data
438
- } = await (0, _axios.default)({
439
- url: `${baseUrl}/session`,
440
- method: 'POST',
441
- data: {
442
- capabilities: w3cCapabilities
443
- }
444
- });
445
- should.not.exist(data.status);
446
- should.not.exist(data.sessionId);
447
- data.value.capabilities.should.eql(w3cCapabilities);
448
- data.value.sessionId.should.exist;
449
- sessionId = data.value.sessionId;
450
- });
451
- it('should raise an error if the driver does not support W3C yet', async function () {
452
- const createSessionStub = sandbox.stub(driver, 'createSession').callsFake(function (capabilities) {
453
- driver.sessionId = null;
454
- return _lib.BaseDriver.prototype.createSession.call(driver, capabilities);
455
- });
456
-
457
- try {
458
- await (0, _axios.default)({
459
- url: `${baseUrl}/session`,
460
- method: 'POST',
461
- data: {
462
- capabilities: {
463
- alwaysMatch: {
464
- platformName: 'Fake',
465
- 'appium:deviceName': 'Fake'
466
- },
467
- firstMatch: [{}]
468
- }
469
- }
470
- }).should.eventually.be.rejectedWith(/500/);
471
- } finally {
472
- createSessionStub.restore();
473
- }
474
- });
475
- describe('w3c endpoints', function () {
476
- let sessionUrl;
477
- beforeEach(async function () {
478
- const {
479
- value
480
- } = (await (0, _axios.default)({
481
- url: `${baseUrl}/session`,
482
- method: 'POST',
483
- data: {
484
- capabilities: {
485
- alwaysMatch: {
486
- platformName: 'Fake',
487
- 'appium:deviceName': 'Commodore 64'
488
- },
489
- firstMatch: [{}]
490
- }
491
- }
492
- })).data;
493
- sessionId = value.sessionId;
494
- sessionUrl = `${baseUrl}/session/${sessionId}`;
495
- });
496
- it(`should throw 400 Bad Parameters exception if the parameters are bad`, async function () {
497
- const {
498
- status,
499
- data
500
- } = await (0, _axios.default)({
501
- url: `${sessionUrl}/actions`,
502
- method: 'POST',
503
- validateStatus: null,
504
- data: {
505
- bad: 'params'
506
- }
507
- });
508
- status.should.equal(400);
509
- const {
510
- error: w3cError,
511
- message,
512
- stacktrace
513
- } = data.value;
514
- message.should.match(/Parameters were incorrect/);
515
- stacktrace.should.match(/protocol.js/);
516
- w3cError.should.be.a.string;
517
- w3cError.should.equal(_lib.errors.InvalidArgumentError.error());
518
- });
519
- it(`should throw 405 exception if the command hasn't been implemented yet`, async function () {
520
- const {
521
- status,
522
- data
523
- } = await (0, _axios.default)({
524
- url: `${sessionUrl}/actions`,
525
- method: 'POST',
526
- validateStatus: null,
527
- data: {
528
- actions: []
529
- }
530
- });
531
- status.should.equal(405);
532
- const {
533
- error: w3cError,
534
- message,
535
- stacktrace
536
- } = data.value;
537
- message.should.match(/Method has not yet been implemented/);
538
- stacktrace.should.match(/protocol.js/);
539
- w3cError.should.be.a.string;
540
- w3cError.should.equal(_lib.errors.NotYetImplementedError.error());
541
- message.should.match(/Method has not yet been implemented/);
542
- });
543
- it(`should throw 500 Unknown Error if the command throws an unexpected exception`, async function () {
544
- driver.performActions = () => {
545
- throw new Error(`Didn't work`);
546
- };
547
-
548
- const {
549
- status,
550
- data
551
- } = await (0, _axios.default)({
552
- url: `${sessionUrl}/actions`,
553
- method: 'POST',
554
- validateStatus: null,
555
- data: {
556
- actions: []
557
- }
558
- });
559
- status.should.equal(500);
560
- const {
561
- error: w3cError,
562
- message,
563
- stacktrace
564
- } = data.value;
565
- stacktrace.should.match(/protocol.js/);
566
- w3cError.should.be.a.string;
567
- w3cError.should.equal(_lib.errors.UnknownError.error());
568
- message.should.match(/Didn't work/);
569
- delete driver.performActions;
570
- });
571
- it(`should translate element format from MJSONWP to W3C`, async function () {
572
- const retValue = [{
573
- something: {
574
- [_constants.MJSONWP_ELEMENT_KEY]: 'fooo',
575
- other: 'bar'
576
- }
577
- }, {
578
- [_constants.MJSONWP_ELEMENT_KEY]: 'bar'
579
- }, 'ignore'];
580
- const expectedValue = [{
581
- something: {
582
- [_constants.MJSONWP_ELEMENT_KEY]: 'fooo',
583
- [_constants.W3C_ELEMENT_KEY]: 'fooo',
584
- other: 'bar'
585
- }
586
- }, {
587
- [_constants.MJSONWP_ELEMENT_KEY]: 'bar',
588
- [_constants.W3C_ELEMENT_KEY]: 'bar'
589
- }, 'ignore'];
590
- const findElementsBackup = driver.findElements;
591
-
592
- driver.findElements = () => retValue;
593
-
594
- const {
595
- data
596
- } = await _axios.default.post(`${sessionUrl}/elements`, {
597
- using: 'whatever',
598
- value: 'whatever'
599
- });
600
- data.value.should.eql(expectedValue);
601
- driver.findElements = findElementsBackup;
602
- });
603
- it(`should fail with a 408 error if it throws a TimeoutError exception`, async function () {
604
- let setUrlStub = sandbox.stub(driver, 'setUrl').callsFake(function () {
605
- throw new _lib.errors.TimeoutError();
606
- });
607
- const {
608
- status,
609
- data
610
- } = await (0, _axios.default)({
611
- url: `${sessionUrl}/url`,
612
- method: 'POST',
613
- validateStatus: null,
614
- data: {
615
- url: 'https://example.com/'
616
- }
617
- });
618
- status.should.equal(408);
619
- const {
620
- error: w3cError,
621
- message,
622
- stacktrace
623
- } = data.value;
624
- stacktrace.should.match(/protocol.js/);
625
- w3cError.should.be.a.string;
626
- w3cError.should.equal(_lib.errors.TimeoutError.error());
627
- message.should.match(/An operation did not complete before its timeout expired/);
628
- setUrlStub.restore();
629
- });
630
- it(`should pass with 200 HTTP status code if the command returns a value`, async function () {
631
- driver.performActions = actions => 'It works ' + actions.join('');
632
-
633
- const {
634
- status,
635
- value,
636
- sessionId
637
- } = (await _axios.default.post(`${sessionUrl}/actions`, {
638
- actions: ['a', 'b', 'c']
639
- })).data;
640
- should.not.exist(sessionId);
641
- should.not.exist(status);
642
- value.should.equal('It works abc');
643
- delete driver.performActions;
644
- });
645
- describe('jwproxy', function () {
646
- let port;
647
- let server, jwproxy, app;
648
- before(async function () {
649
- port = await (0, _helpers2.getTestPort)(true);
650
- });
651
- beforeEach(function () {
652
- const res = (0, _helpers.createProxyServer)(sessionId, port);
653
- server = res.server;
654
- app = res.app;
655
- jwproxy = new _lib.JWProxy({
656
- host: _helpers2.TEST_HOST,
657
- port
658
- });
659
- jwproxy.sessionId = sessionId;
660
-
661
- driver.performActions = async actions => await jwproxy.command('/perform-actions', 'POST', actions);
662
- });
663
- afterEach(async function () {
664
- delete driver.performActions;
665
- await server.close();
666
- });
667
- it('should work if a proxied request returns a response with status 200', async function () {
668
- app.post('/session/:sessionId/perform-actions', (req, res) => {
669
- res.json({
670
- sessionId: req.params.sessionId,
671
- value: req.body,
672
- status: 0
673
- });
674
- });
675
- const {
676
- status,
677
- value,
678
- sessionId
679
- } = (await _axios.default.post(`${sessionUrl}/actions`, {
680
- actions: [1, 2, 3]
681
- })).data;
682
- value.should.eql([1, 2, 3]);
683
- should.not.exist(status);
684
- should.not.exist(sessionId);
685
- });
686
- it('should return error if a proxied request returns a MJSONWP error response', async function () {
687
- app.post('/session/:sessionId/perform-actions', (req, res) => {
688
- res.status(500).json({
689
- sessionId,
690
- status: 6,
691
- value: 'A problem occurred'
692
- });
693
- });
694
- const {
695
- status,
696
- data
697
- } = await (0, _axios.default)({
698
- url: `${sessionUrl}/actions`,
699
- method: 'POST',
700
- validateStatus: null,
701
- data: {
702
- actions: [1, 2, 3]
703
- }
704
- });
705
- status.should.equal(_httpStatusCodes.StatusCodes.NOT_FOUND);
706
- JSON.stringify(data).should.match(/A problem occurred/);
707
- });
708
- it('should return W3C error if a proxied request returns a W3C error response', async function () {
709
- const error = new Error(`Some error occurred`);
710
- error.w3cStatus = 414;
711
- const executeCommandStub = sandbox.stub(driver, 'executeCommand').returns({
712
- protocol: 'W3C',
713
- error
714
- });
715
- const {
716
- status,
717
- data
718
- } = await (0, _axios.default)({
719
- url: `${sessionUrl}/actions`,
720
- method: 'POST',
721
- validateStatus: null,
722
- data: {
723
- actions: [1, 2, 3]
724
- }
725
- });
726
- status.should.equal(414);
727
- const {
728
- error: w3cError,
729
- message: errMessage,
730
- stacktrace
731
- } = data.value;
732
- w3cError.should.equal('unknown error');
733
- stacktrace.should.match(/Some error occurred/);
734
- errMessage.should.equal('Some error occurred');
735
- executeCommandStub.restore();
736
- });
737
- it('should return error if a proxied request returns a MJSONWP error response but HTTP status code is 200', async function () {
738
- app.post('/session/:sessionId/perform-actions', (req, res) => {
739
- res.status(200).json({
740
- sessionId: 'Fake Session Id',
741
- status: 7,
742
- value: 'A problem occurred'
743
- });
744
- });
745
- const {
746
- status,
747
- data
748
- } = await (0, _axios.default)({
749
- url: `${sessionUrl}/actions`,
750
- method: 'POST',
751
- validateStatus: null,
752
- data: {
753
- actions: [1, 2, 3]
754
- }
755
- });
756
- status.should.equal(_httpStatusCodes.StatusCodes.NOT_FOUND);
757
- const {
758
- error: w3cError,
759
- message: errMessage,
760
- stacktrace
761
- } = data.value;
762
- w3cError.should.equal('no such element');
763
- errMessage.should.match(/A problem occurred/);
764
- stacktrace.should.exist;
765
- });
766
- it('should return error if a proxied request returns a W3C error response', async function () {
767
- app.post('/session/:sessionId/perform-actions', (req, res) => {
768
- res.status(404).json({
769
- value: {
770
- error: 'no such element',
771
- message: 'does not make a difference',
772
- stacktrace: 'arbitrary stacktrace'
773
- }
774
- });
775
- });
776
- const {
777
- status,
778
- data
779
- } = await (0, _axios.default)({
780
- url: `${sessionUrl}/actions`,
781
- method: 'POST',
782
- validateStatus: null,
783
- data: {
784
- actions: [1, 2, 3]
785
- }
786
- });
787
- status.should.equal(_httpStatusCodes.StatusCodes.NOT_FOUND);
788
- const {
789
- error: w3cError,
790
- stacktrace
791
- } = data.value;
792
- w3cError.should.equal('no such element');
793
- stacktrace.should.match(/arbitrary stacktrace/);
794
- });
795
- it('should return an error if a proxied request returns a W3C error response', async function () {
796
- app.post('/session/:sessionId/perform-actions', (req, res) => {
797
- res.set('Connection', 'close');
798
- res.status(444).json({
799
- value: {
800
- error: 'bogus error code',
801
- message: 'does not make a difference',
802
- stacktrace: 'arbitrary stacktrace'
803
- }
804
- });
805
- });
806
- const {
807
- status,
808
- data
809
- } = await (0, _axios.default)({
810
- url: `${sessionUrl}/actions`,
811
- method: 'POST',
812
- validateStatus: null,
813
- data: {
814
- actions: [1, 2, 3]
815
- }
816
- });
817
- status.should.equal(_httpStatusCodes.StatusCodes.INTERNAL_SERVER_ERROR);
818
- const {
819
- error: w3cError,
820
- stacktrace
821
- } = data.value;
822
- w3cError.should.equal('unknown error');
823
- stacktrace.should.match(/arbitrary stacktrace/);
824
- });
825
- });
826
- });
827
- });
828
- it('should handle commands with no response values', async function () {
829
- const {
830
- data
831
- } = await (0, _axios.default)({
832
- url: `${baseUrl}/session/foo/forward`,
833
- method: 'POST'
834
- });
835
- data.should.eql({
836
- value: null,
837
- sessionId: 'foo'
838
- });
839
- });
840
- it('should allow empty string response values', async function () {
841
- const {
842
- data
843
- } = await (0, _axios.default)({
844
- url: `${baseUrl}/session/foo/element/bar/text`
845
- });
846
- data.should.eql({
847
- value: '',
848
- sessionId: 'foo'
849
- });
850
- });
851
- it('should send 500 response and an Unknown object for rejected commands', async function () {
852
- const {
853
- status,
854
- data
855
- } = await (0, _axios.default)({
856
- url: `${baseUrl}/session/foo/refresh`,
857
- method: 'POST',
858
- validateStatus: null
859
- });
860
- status.should.equal(500);
861
- data.value.error.should.eql('unknown error');
862
- data.value.message.should.eql('An unknown server-side error occurred while processing ' + 'the command. Original error: Too Fresh!');
863
- data.sessionId.should.eql('foo');
864
- });
865
- it('should not throw UnknownError when known', async function () {
866
- const {
867
- status,
868
- data
869
- } = await (0, _axios.default)({
870
- url: `${baseUrl}/session/foo`,
871
- validateStatus: null
872
- });
873
- status.should.equal(404);
874
- data.value.error.should.eql('invalid session id');
875
- data.value.message.should.eql('A session is either terminated or not started');
876
- data.sessionId.should.eql('foo');
877
- });
878
- });
879
- describe('session Ids', function () {
880
- let driver = new _fakeDriver.FakeDriver();
881
- let mjsonwpServer;
882
- before(async function () {
883
- mjsonwpServer = await (0, _lib.server)({
884
- routeConfiguringFunction: (0, _lib.routeConfiguringFunction)(driver),
885
- port
886
- });
887
- });
888
- after(async function () {
889
- await mjsonwpServer.close();
890
- });
891
- afterEach(function () {
892
- driver.sessionId = null;
893
- });
894
- it('should return null SessionId for commands without sessionIds', async function () {
895
- const {
896
- data
897
- } = await (0, _axios.default)({
898
- url: `${baseUrl}/status`
899
- });
900
- should.equal(data.sessionId, null);
901
- });
902
- it('responds with the same session ID in the request', async function () {
903
- let sessionId = 'Vader Sessions';
904
- driver.sessionId = sessionId;
905
- const {
906
- data
907
- } = await (0, _axios.default)({
908
- url: `${baseUrl}/session/${sessionId}/url`,
909
- method: 'POST',
910
- data: {
911
- url: 'http://google.com'
912
- }
913
- });
914
- should.exist(data.sessionId);
915
- data.sessionId.should.eql(sessionId);
916
- });
917
- it('yells if no session exists', async function () {
918
- let sessionId = 'Vader Sessions';
919
- const {
920
- data,
921
- status
922
- } = await (0, _axios.default)({
923
- url: `${baseUrl}/session/${sessionId}/url`,
924
- method: 'POST',
925
- validateStatus: null,
926
- data: {
927
- url: 'http://google.com'
928
- }
929
- });
930
- status.should.equal(404);
931
- data.value.message.should.contain('session');
932
- });
933
- it('yells if invalid session is sent', async function () {
934
- let sessionId = 'Vader Sessions';
935
- driver.sessionId = 'recession';
936
- const {
937
- data,
938
- status
939
- } = await (0, _axios.default)({
940
- url: `${baseUrl}/session/${sessionId}/url`,
941
- method: 'POST',
942
- validateStatus: null,
943
- data: {
944
- url: 'http://google.com'
945
- }
946
- });
947
- status.should.equal(404);
948
- data.value.message.should.contain('session');
949
- });
950
- it('should have session IDs in error responses', async function () {
951
- let sessionId = 'Vader Sessions';
952
- driver.sessionId = sessionId;
953
- const {
954
- data,
955
- status
956
- } = await (0, _axios.default)({
957
- url: `${baseUrl}/session/${sessionId}/refresh`,
958
- method: 'POST',
959
- validateStatus: null
960
- });
961
- status.should.equal(500);
962
- data.value.error.should.eql('unknown error');
963
- data.value.message.should.eql('An unknown server-side error occurred while processing ' + 'the command. Original error: Too Fresh!');
964
- data.sessionId.should.eql('Vader Sessions');
965
- });
966
- it('should return a new session ID on create', async function () {
967
- const {
968
- data
969
- } = await (0, _axios.default)({
970
- url: `${baseUrl}/session`,
971
- method: 'POST',
972
- data: {
973
- capabilities: {
974
- alwaysMatch: {
975
- 'appium:greeting': 'hello'
976
- },
977
- firstMatch: [{}]
978
- }
979
- }
980
- });
981
- should.exist(data.value.sessionId);
982
- data.value.sessionId.indexOf('fakeSession_').should.equal(0);
983
- data.value.capabilities.should.eql({
984
- alwaysMatch: {
985
- 'appium:greeting': 'hello'
986
- },
987
- firstMatch: [{}]
988
- });
989
- });
990
- });
991
- describe('via drivers jsonwp proxy', function () {
992
- let driver;
993
- let sessionId = 'foo';
994
- let mjsonwpServer;
995
- beforeEach(async function () {
996
- driver = new _fakeDriver.FakeDriver();
997
- driver.sessionId = sessionId;
998
-
999
- driver.proxyActive = () => true;
1000
-
1001
- driver.canProxy = () => true;
1002
-
1003
- mjsonwpServer = await (0, _lib.server)({
1004
- routeConfiguringFunction: (0, _lib.routeConfiguringFunction)(driver),
1005
- port,
1006
- extraMethodMap: _fakeDriver.FakeDriver.newMethodMap
1007
- });
1008
- });
1009
- afterEach(async function () {
1010
- await mjsonwpServer.close();
1011
- });
1012
- it('should give a nice error if proxying is set but no proxy function exists', async function () {
1013
- driver.canProxy = () => false;
1014
-
1015
- const {
1016
- status,
1017
- data
1018
- } = await (0, _axios.default)({
1019
- url: `${baseUrl}/session/${sessionId}/url`,
1020
- method: 'POST',
1021
- validateStatus: null,
1022
- data: {
1023
- url: 'http://google.com'
1024
- }
1025
- });
1026
- status.should.equal(500);
1027
- data.value.error.should.eql('unknown error');
1028
- data.value.message.should.eql('An unknown server-side error occurred while processing ' + 'the command. Original error: Trying to proxy to a ' + 'server but the driver is unable to proxy');
1029
- data.sessionId.should.eql('foo');
1030
- });
1031
- it('should pass on any errors in proxying', async function () {
1032
- driver.proxyReqRes = async function () {
1033
- throw new Error('foo');
1034
- };
1035
-
1036
- const {
1037
- status,
1038
- data
1039
- } = await (0, _axios.default)({
1040
- url: `${baseUrl}/session/${sessionId}/url`,
1041
- method: 'POST',
1042
- validateStatus: null,
1043
- data: {
1044
- url: 'http://google.com'
1045
- }
1046
- });
1047
- status.should.equal(500);
1048
- data.value.error.should.eql('unknown error');
1049
- data.value.message.should.eql('An unknown server-side error occurred while processing ' + 'the command. Original error: Could not proxy. Proxy error: foo');
1050
- data.sessionId.should.eql('foo');
1051
- });
1052
- it('should able to throw ProxyRequestError in proxying', async function () {
1053
- driver.proxyReqRes = async function () {
1054
- let jsonwp = {
1055
- status: 35,
1056
- value: 'No such context found.',
1057
- sessionId: 'foo'
1058
- };
1059
- throw new _lib.errors.ProxyRequestError(`Could not proxy command to remote server. `, jsonwp);
1060
- };
1061
-
1062
- const {
1063
- status,
1064
- data
1065
- } = await (0, _axios.default)({
1066
- url: `${baseUrl}/session/${sessionId}/url`,
1067
- method: 'POST',
1068
- validateStatus: null,
1069
- data: {
1070
- url: 'http://google.com'
1071
- }
1072
- });
1073
- status.should.equal(400);
1074
- data.value.error.should.eql('unknown error');
1075
- data.value.message.should.eql('No such context found.');
1076
- data.sessionId.should.eql('foo');
1077
- });
1078
- it('should let the proxy handle req/res', async function () {
1079
- driver.proxyReqRes = async function (req, res) {
1080
- res.status(200).json({
1081
- custom: 'data'
1082
- });
1083
- };
1084
-
1085
- const {
1086
- status,
1087
- data
1088
- } = await (0, _axios.default)({
1089
- url: `${baseUrl}/session/${sessionId}/url`,
1090
- method: 'POST',
1091
- data: {
1092
- url: 'http://google.com'
1093
- }
1094
- });
1095
- status.should.equal(200);
1096
- data.should.eql({
1097
- custom: 'data'
1098
- });
1099
- });
1100
- it('should avoid jsonwp proxying when path matches avoidance list', async function () {
1101
- driver.getProxyAvoidList = () => [['POST', new RegExp('^/session/[^/]+/url$')]];
1102
-
1103
- const {
1104
- status,
1105
- data
1106
- } = await (0, _axios.default)({
1107
- url: `${baseUrl}/session/${sessionId}/url`,
1108
- method: 'POST',
1109
- data: {
1110
- url: 'http://google.com'
1111
- }
1112
- });
1113
- status.should.equal(200);
1114
- data.should.eql({
1115
- value: 'Navigated to: http://google.com',
1116
- sessionId
1117
- });
1118
- });
1119
- it('should fail if avoid proxy list is malformed in some way', async function () {
1120
- async function badProxyAvoidanceList(list) {
1121
- driver.getProxyAvoidList = () => list;
1122
-
1123
- const {
1124
- status,
1125
- data
1126
- } = await (0, _axios.default)({
1127
- url: `${baseUrl}/session/${sessionId}/url`,
1128
- method: 'POST',
1129
- validateStatus: null,
1130
- data: {
1131
- url: 'http://google.com'
1132
- }
1133
- });
1134
- status.should.equal(500);
1135
- data.value.message.should.contain('roxy');
1136
- }
1137
-
1138
- const lists = ['foo', [['foo']], [['BAR', /lol/]], [['GET', 'foo']]];
1139
-
1140
- for (let list of lists) {
1141
- await badProxyAvoidanceList(list);
1142
- }
1143
- });
1144
- it('should avoid proxying non-session commands even if not in the list', async function () {
1145
- driver.getProxyAvoidList = () => [['POST', new RegExp('')]];
1146
-
1147
- const {
1148
- status,
1149
- data
1150
- } = await (0, _axios.default)({
1151
- url: `${baseUrl}/status`
1152
- });
1153
- status.should.equal(200);
1154
- data.should.eql({
1155
- value: "I'm fine",
1156
- sessionId: null
1157
- });
1158
- });
1159
- it('should avoid proxying deleteSession commands', async function () {
1160
- driver.getProxyAvoidList = () => [['POST', new RegExp('')]];
1161
-
1162
- driver.sessionId.should.equal(sessionId);
1163
- const {
1164
- status
1165
- } = await _axios.default.delete(`${baseUrl}/session/${sessionId}`);
1166
- status.should.equal(200);
1167
- should.not.exist(driver.sessionId);
1168
- driver.jwpProxyActive.should.be.false;
1169
- });
1170
- it('should avoid proxying when command spec specifies neverProxy', async function () {
1171
- const {
1172
- status,
1173
- data
1174
- } = await (0, _axios.default)({
1175
- url: `${baseUrl}/session/${sessionId}/noproxy`,
1176
- method: 'GET'
1177
- });
1178
- status.should.equal(200);
1179
- data.should.eql({
1180
- value: 'This was not proxied',
1181
- sessionId
1182
- });
1183
- });
1184
- });
1185
- });require('source-map-support').install();
1186
-
1187
-
1188
- //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInRlc3QvcHJvdG9jb2wvcHJvdG9jb2wtZTJlLXNwZWNzLmpzIl0sIm5hbWVzIjpbInBvcnQiLCJiYXNlVXJsIiwiZGVzY3JpYmUiLCJzYW5kYm94IiwiYmVmb3JlRWFjaCIsImFmdGVyRWFjaCIsInJlc3RvcmUiLCJiZWZvcmUiLCJURVNUX0hPU1QiLCJkIiwiRmFrZURyaXZlciIsIml0Iiwic2V0VXJsIiwic2hvdWxkIiwiY29udGFpbiIsIm1qc29ud3BTZXJ2ZXIiLCJkcml2ZXIiLCJzZXNzaW9uSWQiLCJyb3V0ZUNvbmZpZ3VyaW5nRnVuY3Rpb24iLCJhZnRlciIsImNsb3NlIiwiZGF0YSIsInVybCIsIm1ldGhvZCIsImVxbCIsInZhbHVlIiwicmVxRGF0YSIsIlVSTFNlYXJjaFBhcmFtcyIsInNldCIsImhlYWRlcnMiLCJ0b1N0cmluZyIsInN0YXR1cyIsInZhbGlkYXRlU3RhdHVzIiwiZXF1YWwiLCJKU09OIiwic3RyaW5naWZ5IiwiZXZlbnR1YWxseSIsImJlIiwicmVqZWN0ZWQiLCJyZWplY3RlZFdpdGgiLCJpbmNsdWRlIiwiZXJyb3IiLCJtZXNzYWdlIiwiaWQiLCJyZXNwb25zZSIsInRleHQiLCJlbGVtZW50IiwieG9mZnNldCIsInlvZmZzZXQiLCJhcHBJZCIsImJ1bmRsZUlkIiwiZGVlcCIsImFjdGlvbnMiLCJheGlvcyIsImRlbGV0ZSIsImRlc2lyZWRDYXBhYmlsaXRpZXMiLCJhIiwidzNjQ2FwYWJpbGl0aWVzIiwiYWx3YXlzTWF0Y2giLCJjYXBhYmlsaXRpZXMiLCJub3QiLCJleGlzdCIsImNyZWF0ZVNlc3Npb25TdHViIiwic3R1YiIsImNhbGxzRmFrZSIsIkJhc2VEcml2ZXIiLCJwcm90b3R5cGUiLCJjcmVhdGVTZXNzaW9uIiwiY2FsbCIsInBsYXRmb3JtTmFtZSIsImZpcnN0TWF0Y2giLCJzZXNzaW9uVXJsIiwiYmFkIiwidzNjRXJyb3IiLCJzdGFja3RyYWNlIiwibWF0Y2giLCJzdHJpbmciLCJlcnJvcnMiLCJJbnZhbGlkQXJndW1lbnRFcnJvciIsIk5vdFlldEltcGxlbWVudGVkRXJyb3IiLCJwZXJmb3JtQWN0aW9ucyIsIkVycm9yIiwiVW5rbm93bkVycm9yIiwicmV0VmFsdWUiLCJzb21ldGhpbmciLCJNSlNPTldQX0VMRU1FTlRfS0VZIiwib3RoZXIiLCJleHBlY3RlZFZhbHVlIiwiVzNDX0VMRU1FTlRfS0VZIiwiZmluZEVsZW1lbnRzQmFja3VwIiwiZmluZEVsZW1lbnRzIiwicG9zdCIsInVzaW5nIiwic2V0VXJsU3R1YiIsIlRpbWVvdXRFcnJvciIsImpvaW4iLCJzZXJ2ZXIiLCJqd3Byb3h5IiwiYXBwIiwicmVzIiwiSldQcm94eSIsImhvc3QiLCJjb21tYW5kIiwicmVxIiwianNvbiIsInBhcmFtcyIsImJvZHkiLCJIVFRQU3RhdHVzQ29kZXMiLCJOT1RfRk9VTkQiLCJ3M2NTdGF0dXMiLCJleGVjdXRlQ29tbWFuZFN0dWIiLCJyZXR1cm5zIiwicHJvdG9jb2wiLCJlcnJNZXNzYWdlIiwiSU5URVJOQUxfU0VSVkVSX0VSUk9SIiwiaW5kZXhPZiIsInByb3h5QWN0aXZlIiwiY2FuUHJveHkiLCJleHRyYU1ldGhvZE1hcCIsIm5ld01ldGhvZE1hcCIsInByb3h5UmVxUmVzIiwianNvbndwIiwiUHJveHlSZXF1ZXN0RXJyb3IiLCJjdXN0b20iLCJnZXRQcm94eUF2b2lkTGlzdCIsIlJlZ0V4cCIsImJhZFByb3h5QXZvaWRhbmNlTGlzdCIsImxpc3QiLCJsaXN0cyIsImp3cFByb3h5QWN0aXZlIiwiZmFsc2UiXSwibWFwcGluZ3MiOiI7Ozs7OztBQUVBOztBQUdBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUdBOztBQUVBLElBQUlBLElBQUo7QUFDQSxJQUFJQyxPQUFKO0FBRUFDLFFBQVEsQ0FBQyxVQUFELEVBQWEsWUFBWTtBQUMvQixNQUFJQyxPQUFKO0FBRUFDLEVBQUFBLFVBQVUsQ0FBQyxZQUFZO0FBQ3JCRCxJQUFBQSxPQUFPLEdBQUcsMkJBQVY7QUFDRCxHQUZTLENBQVY7QUFJQUUsRUFBQUEs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LENBQUNDLFNBQXhCO0FBQ0FELE1BQUFBLE1BQU0sQ0FBQ2lILGNBQVAsQ0FBc0JwSCxNQUF0QixDQUE2QndCLEVBQTdCLENBQWdDNkYsS0FBaEM7QUFDRCxLQVRDLENBQUY7QUFXQXZILElBQUFBLEVBQUUsQ0FBQyw4REFBRCxFQUFpRSxrQkFBa0I7QUFDbkYsWUFBTTtBQUFDb0IsUUFBQUEsTUFBRDtBQUFTVixRQUFBQTtBQUFULFVBQWlCLE1BQU0sb0JBQU07QUFDakNDLFFBQUFBLEdBQUcsRUFBRyxHQUFFckIsT0FBUSxZQUFXZ0IsU0FBVSxVQURKO0FBRWpDTSxRQUFBQSxNQUFNLEVBQUU7QUFGeUIsT0FBTixDQUE3QjtBQUtBUSxNQUFBQSxNQUFNLENBQUNsQixNQUFQLENBQWNvQixLQUFkLENBQW9CLEdBQXBCO0FBQ0FaLE1BQUFBLElBQUksQ0FBQ1IsTUFBTCxDQUFZVyxHQUFaLENBQWdCO0FBQ2RDLFFBQUFBLEtBQUssRUFBRSxzQkFETztBQUVkUixRQUFBQTtBQUZjLE9BQWhCO0FBSUQsS0FYQyxDQUFGO0FBYUQsR0F0S08sQ0FBUjtBQXVLRCxDQTErQk8sQ0FBUiIsInNvdXJjZXNDb250ZW50IjpbIi8vIHRyYW5zcGlsZTptb2NoYVxuXG5pbXBvcnQge1xuICBzZXJ2ZXIsIHJvdXRlQ29uZmlndXJpbmdGdW5jdGlvbiwgZXJyb3JzLCBKV1Byb3h5LCBCYXNlRHJpdmVyXG59IGZyb20gJy4uLy4uL2xpYic7XG5pbXBvcnQgeyBGYWtlRHJpdmVyIH0gZnJvbSAnLi9mYWtlLWRyaXZlcic7XG5pbXBvcnQgYXhpb3MgZnJvbSAnYXhpb3MnO1xuaW1wb3J0IHsgY3JlYXRlU2FuZGJveCB9IGZyb20gJ3Npbm9uJztcbmltcG9ydCB7IFN0YXR1c0NvZGVzIGFzIEhUVFBTdGF0dXNDb2RlcyB9IGZyb20gJ2h0dHAtc3RhdHVzLWNvZGVzJztcbmltcG9ydCB7IGNyZWF0ZVByb3h5U2VydmVyIH0gZnJvbSAnLi9oZWxwZXJzJztcbmltcG9ydCB7XG4gIE1KU09OV1BfRUxFTUVOVF9LRVksIFczQ19FTEVNRU5UX0tFWVxufSBmcm9tICcuLi8uLi9saWIvY29uc3RhbnRzJztcbmltcG9ydCB7VEVTVF9IT1NULCBnZXRUZXN0UG9ydH0gZnJvbSAnLi4vaGVscGVycyc7XG5cbmxldCBwb3J0O1xubGV0IGJhc2VVcmw7XG5cbmRlc2NyaWJlKCdQcm90b2NvbCcsIGZ1bmN0aW9uICgpIHtcbiAgbGV0IHNhbmRib3g7XG5cbiAgYmVmb3JlRWFjaChmdW5jdGlvbiAoKSB7XG4gICAgc2FuZGJveCA9IGNyZWF0ZVNhbmRib3goKTtcbiAgfSk7XG5cbiAgYWZ0ZXJFYWNoKGZ1bmN0aW9uICgpIHtcbiAgICBzYW5kYm94LnJlc3RvcmUoKTtcbiAgfSk7XG5cbiAgYmVmb3JlKGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICBwb3J0ID0gYXdhaXQgZ2V0VGVzdFBvcnQoKTtcbiAgICBiYXNlVXJsID0gYGh0dHA6Ly8ke1RFU1RfSE9TVH06JHtwb3J0fWA7XG4gIH0pO1xuXG4gIC8vVE9ETzogbW9yZSB0ZXN0cyE6XG4gIC8vIFVua25vd24gY29tbWFuZHMgc2hvdWxkIHJldHVybiA0MDRcblxuICBkZXNjcmliZSgnZGlyZWN0IHRvIGRyaXZlcicsIGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgZCA9IG5ldyBGYWtlRHJpdmVyKCk7XG4gICAgaXQoJ3Nob3VsZCByZXR1cm4gcmVzcG9uc2UgdmFsdWVzIGRpcmVjdGx5IGZyb20gdGhlIGRyaXZlcicsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIChhd2FpdCBkLnNldFVybCgnaHR0cDovL2dvb2dsZS5jb20nKSkuc2hvdWxkLmNvbnRhaW4oJ2dvb2dsZScpO1xuICAgIH0pO1xuICB9KTtcblxuICBkZXNjcmliZSgndmlhIGV4cHJlc3Mgcm91dGVyJywgZnVuY3Rpb24gKCkge1xuICAgIGxldCBtanNvbndwU2VydmVyO1xuICAgIGxldCBkcml2ZXI7XG5cbiAgICBiZWZvcmUoYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgZHJpdmVyID0gbmV3IEZha2VEcml2ZXIoKTtcbiAgICAgIGRyaXZlci5zZXNzaW9uSWQgPSAnZm9vJztcbiAgICAgIG1qc29ud3BTZXJ2ZXIgPSBhd2FpdCBzZXJ2ZXIoe1xuICAgICAgICByb3V0ZUNvbmZpZ3VyaW5nRnVuY3Rpb246IHJvdXRlQ29uZmlndXJpbmdGdW5jdGlvbihkcml2ZXIpLFxuICAgICAgICBwb3J0LFxuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBhZnRlcihhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBhd2FpdCBtanNvbndwU2VydmVyLmNsb3NlKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHByb3h5IHRvIGRyaXZlciBhbmQgcmV0dXJuIHZhbGlkIGpzb253cCByZXNwb25zZScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby91cmxgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgZGF0YToge3VybDogJ2h0dHA6Ly9nb29nbGUuY29tJ31cbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdOYXZpZ2F0ZWQgdG86IGh0dHA6Ly9nb29nbGUuY29tJyxcbiAgICAgICAgc2Vzc2lvbklkOiAnZm9vJ1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGFzc3VtZSByZXF1ZXN0cyB3aXRob3V0IGEgQ29udGVudC1UeXBlIGFyZSBqc29uIHJlcXVlc3RzJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdOYXZpZ2F0ZWQgdG86IGh0dHA6Ly9nb29nbGUuY29tJyxcbiAgICAgICAgc2Vzc2lvbklkOiAnZm9vJ1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHJlc3BvbmQgdG8geC13d3ctZm9ybS11cmxlbmNvZGVkIGFzIHdlbGwgYXMganNvbiByZXF1ZXN0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHJlcURhdGEgPSBuZXcgVVJMU2VhcmNoUGFyYW1zKCk7XG4gICAgICByZXFEYXRhLnNldCgndXJsJywgJ2h0dHA6Ly9nb29nbGUuY29tJyk7XG4gICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vdXJsYCxcbiAgICAgICAgaGVhZGVyczoge1xuICAgICAgICAgICdDb250ZW50LVR5cGUnOiAnYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkJyxcbiAgICAgICAgfSxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHJlcURhdGEudG9TdHJpbmcoKSxcbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdOYXZpZ2F0ZWQgdG86IGh0dHA6Ly9nb29nbGUuY29tJyxcbiAgICAgICAgc2Vzc2lvbklkOiAnZm9vJ1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGluY2x1ZGUgdXJsIHJlcXVlc3QgcGFyYW1ldGVycyBmb3IgbWV0aG9kcyB0byB1c2UgLSBzZXNzaW9uaWQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vYmFja2AsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7fSxcbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdmb28nLFxuICAgICAgICBzZXNzaW9uSWQ6ICdmb28nXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgaW5jbHVkZSB1cmwgcmVxdWVzdCBwYXJhbWV0ZXJzIGZvciBtZXRob2RzIHRvIHVzZSAtIGVsZW1lbnRpZCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci9jbGlja2AsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7fSxcbiAgICAgIH0pO1xuICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFsnYmFyJywgJ2ZvbyddKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgaW5jbHVkZSB1cmwgcmVxIHBhcmFtcyBpbiB0aGUgb3JkZXI6IGN1c3RvbSwgZWxlbWVudCwgc2Vzc2lvbicsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci9hdHRyaWJ1dGUvYmF6YCxcbiAgICAgIH0pO1xuICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFsnYmF6JywgJ2JhcicsICdmb28nXSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHJlc3BvbmQgd2l0aCA0MDAgQmFkIFJlcXVlc3QgaWYgcGFyYW1ldGVycyBtaXNzaW5nJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge2RhdGEsIHN0YXR1c30gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHt9LFxuICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgIH0pO1xuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg0MDApO1xuICAgICAgSlNPTi5zdHJpbmdpZnkoZGF0YSkuc2hvdWxkLmNvbnRhaW4oJ3VybCcpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCByZWplY3QgcmVxdWVzdHMgd2l0aCBhIGJhZGx5IGZvcm1hdHRlZCBib2R5IGFuZCBub3QgY3Jhc2gnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6ICdvaCBoZWxsbydcbiAgICAgIH0pLnNob3VsZC5ldmVudHVhbGx5LmJlLnJlamVjdGVkO1xuXG4gICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHt1cmw6ICdodHRwOi8vZ29vZ2xlLmNvbSd9XG4gICAgICB9KTtcbiAgICAgIGRhdGEuc2hvdWxkLmVxbCh7XG4gICAgICAgIHZhbHVlOiAnTmF2aWdhdGVkIHRvOiBodHRwOi8vZ29vZ2xlLmNvbScsXG4gICAgICAgIHNlc3Npb25JZDogJ2ZvbydcbiAgICAgIH0pO1xuXG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGdldCA0MDQgZm9yIGJhZCByb3V0ZXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vYmxhcmdpbWFyZ2AsXG4gICAgICB9KS5zaG91bGQuZXZlbnR1YWxseS5iZS5yZWplY3RlZFdpdGgoLzQwNC8pO1xuICAgIH0pO1xuXG4gICAgaXQoJzR4eCByZXNwb25zZXMgc2hvdWxkIGhhdmUgY29udGVudC10eXBlIG9mIGFwcGxpY2F0aW9uL2pzb24nLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCB7aGVhZGVyc30gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vYmxhcmdpbWFyZ2FyaXRhYCxcbiAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICB9KTtcblxuICAgICAgaGVhZGVyc1snY29udGVudC10eXBlJ10uc2hvdWxkLmluY2x1ZGUoJ2FwcGxpY2F0aW9uL2pzb24nKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdGhyb3cgbm90IHlldCBpbXBsZW1lbnRlZCBmb3IgdW5maWxsZWRvdXQgY29tbWFuZHMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci9sb2NhdGlvbmAsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgfSk7XG5cbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNDA1KTtcbiAgICAgIGRhdGEudmFsdWUuZXJyb3Iuc2hvdWxkLmVxbCgndW5rbm93biBtZXRob2QnKTtcbiAgICAgIGRhdGEudmFsdWUubWVzc2FnZS5zaG91bGQuZXFsKCdNZXRob2QgaGFzIG5vdCB5ZXQgYmVlbiBpbXBsZW1lbnRlZCcpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHRocm93IG5vdCBpbXBsZW1lbnRlZCBmb3IgaWdub3JlZCBjb21tYW5kcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2J1dHRvbnVwYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICBkYXRhOiB7fSxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDQwNSk7XG4gICAgICBkYXRhLnZhbHVlLmVycm9yLnNob3VsZC5lcWwoJ3Vua25vd24gbWV0aG9kJyk7XG4gICAgICBkYXRhLnZhbHVlLm1lc3NhZ2Uuc2hvdWxkLmVxbCgnTWV0aG9kIGhhcyBub3QgeWV0IGJlZW4gaW1wbGVtZW50ZWQnKTtcbiAgICAgIGRhdGEuc2Vzc2lvbklkLnNob3VsZC5lcWwoJ2ZvbycpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBnZXQgNDAwIGZvciBiYWQgcGFyYW1ldGVycycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby91cmxgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgZGF0YToge31cbiAgICAgIH0pLnNob3VsZC5ldmVudHVhbGx5LmJlLnJlamVjdGVkV2l0aCgvNDAwLyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGlnbm9yZSBzcGVjaWFsIGV4dHJhIHBheWxvYWQgcGFyYW1zIGluIHRoZSByaWdodCBjb250ZXh0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci92YWx1ZWAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7aWQ6ICdiYXonLCBzZXNzaW9uSWQ6ICdsb2wnLCB2YWx1ZTogWydhJ119XG4gICAgICB9KTtcblxuICAgICAgYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2VsZW1lbnQvYmFyL3ZhbHVlYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHtpZDogJ2Jheid9XG4gICAgICB9KS5zaG91bGQuZXZlbnR1YWxseS5iZS5yZWplY3RlZFdpdGgoLzQwMC8pO1xuXG4gICAgICAvLyBtYWtlIHN1cmUgYWRkaW5nIHRoZSBvcHRpb25hbCAnaWQnIGRvZXNuJ3QgY2xvYmJlciBhIHJvdXRlIHdoZXJlIHdlXG4gICAgICAvLyBoYXZlIGFuIGFjdHVhbCByZXF1aXJlZCAnaWQnXG4gICAgICBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vZnJhbWVgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgZGF0YToge2lkOiAnYmF6J31cbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCByZXR1cm4gdGhlIGNvcnJlY3QgZXJyb3IgZXZlbiBpZiBkcml2ZXIgZG9lcyBub3QgdGhyb3cnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9hcHBpdW0vcmVjZWl2ZV9hc3luY19yZXNwb25zZWAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7cmVzcG9uc2U6ICdiYXonfSxcbiAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICB9KTtcbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNTAwKTtcbiAgICAgIGRhdGEudmFsdWUuZXJyb3Iuc2hvdWxkLmVxbCgndW5rbm93biBlcnJvcicpO1xuICAgICAgZGF0YS52YWx1ZS5tZXNzYWdlLnNob3VsZC5lcWwoJ0FuIHVua25vd24gc2VydmVyLXNpZGUgZXJyb3Igb2NjdXJyZWQgd2hpbGUgcHJvY2Vzc2luZyAnICtcbiAgICAgICAgJ3RoZSBjb21tYW5kLiBPcmlnaW5hbCBlcnJvcjogTWlzaGFuZGxlZCBEcml2ZXIgRXJyb3InKTtcbiAgICAgIGRhdGEuc2Vzc2lvbklkLnNob3VsZC5lcWwoJ2ZvbycpO1xuICAgIH0pO1xuXG4gICAgZGVzY3JpYmUoJ3czYyBzZW5ka2V5cyBtaWdyYXRpb24nLCBmdW5jdGlvbiAoKSB7XG4gICAgICBpdCgnc2hvdWxkIGFjY2VwdCB2YWx1ZSBmb3Igc2VuZGtleXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2VsZW1lbnQvYmFyL3ZhbHVlYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiB7dmFsdWU6ICd0ZXh0IHRvIHR5cGUnfVxuICAgICAgICB9KTtcbiAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFsndGV4dCB0byB0eXBlJywgJ2JhciddKTtcbiAgICAgIH0pO1xuICAgICAgaXQoJ3Nob3VsZCBhY2NlcHQgdGV4dCBmb3Igc2VuZGtleXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2VsZW1lbnQvYmFyL3ZhbHVlYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiB7dGV4dDogJ3RleHQgdG8gdHlwZSd9XG4gICAgICAgIH0pO1xuICAgICAgICBkYXRhLnZhbHVlLnNob3VsZC5lcWwoWyd0ZXh0IHRvIHR5cGUnLCAnYmFyJ10pO1xuICAgICAgfSk7XG4gICAgICBpdCgnc2hvdWxkIGFjY2VwdCB2YWx1ZSBhbmQgdGV4dCBmb3Igc2VuZGtleXMsIGFuZCB1c2UgdmFsdWUnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2VsZW1lbnQvYmFyL3ZhbHVlYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiB7dmFsdWU6ICd0ZXh0IHRvIHR5cGUnLCB0ZXh0OiAndGV4dCB0byBpZ25vcmUnfVxuICAgICAgICB9KTtcbiAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFsndGV4dCB0byB0eXBlJywgJ2JhciddKTtcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgZGVzY3JpYmUoJ211bHRpcGxlIHNldHMgb2YgYXJndW1lbnRzJywgZnVuY3Rpb24gKCkge1xuICAgICAgZGVzY3JpYmUoJ29wdGlvbmFsJywgZnVuY3Rpb24gKCkge1xuICAgICAgICBpdCgnc2hvdWxkIGFsbG93IG1vdmV0byB3aXRoIGVsZW1lbnQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9tb3ZldG9gLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICBkYXRhOiB7ZWxlbWVudDogJzMnfVxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGRhdGEudmFsdWUuc2hvdWxkLmVxbChbJzMnLCBudWxsLCBudWxsXSk7XG4gICAgICAgIH0pO1xuICAgICAgICBpdCgnc2hvdWxkIGFsbG93IG1vdmV0byB3aXRoIHhvZmZzZXQveW9mZnNldCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL21vdmV0b2AsXG4gICAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgIGRhdGE6IHt4b2Zmc2V0OiA0MiwgeW9mZnNldDogMTd9XG4gICAgICAgICAgfSk7XG4gICAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFtudWxsLCA0MiwgMTddKTtcbiAgICAgICAgfSk7XG4gICAgICB9KTtcbiAgICAgIGRlc2NyaWJlKCdyZXF1aXJlZCcsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgaXQoJ3Nob3VsZCBhbGxvdyByZW1vdmVBcHAgd2l0aCBhcHBJZCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2FwcGl1bS9kZXZpY2UvcmVtb3ZlX2FwcGAsXG4gICAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgIGRhdGE6IHthcHBJZDogNDJ9XG4gICAgICAgICAgfSk7XG4gICAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKDQyKTtcbiAgICAgICAgfSk7XG4gICAgICAgIGl0KCdzaG91bGQgYWxsb3cgcmVtb3ZlQXBwIHdpdGggYnVuZGxlSWQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9hcHBpdW0vZGV2aWNlL3JlbW92ZV9hcHBgLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICBkYXRhOiB7YnVuZGxlSWQ6IDQyfVxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGRhdGEudmFsdWUuc2hvdWxkLmVxbCg0Mik7XG4gICAgICAgIH0pO1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBkZXNjcmliZSgnZGVmYXVsdCBwYXJhbSB3cmFwJywgZnVuY3Rpb24gKCkge1xuICAgICAgaXQoJ3Nob3VsZCB3cmFwJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby90b3VjaC9wZXJmb3JtYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiBbeydhY3Rpb24nOiAndGFwJywgJ29wdGlvbnMnOiB7J2VsZW1lbnQnOiAnMyd9fV1cbiAgICAgICAgfSk7XG4gICAgICAgIGRhdGEudmFsdWUuc2hvdWxkLmRlZXAuZXF1YWwoW1t7J2FjdGlvbic6ICd0YXAnLCAnb3B0aW9ucyc6IHsnZWxlbWVudCc6ICczJ319XSwgJ2ZvbyddKTtcbiAgICAgIH0pO1xuXG4gICAgICBpdCgnc2hvdWxkIG5vdCB3cmFwIHR3aWNlJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby90b3VjaC9wZXJmb3JtYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiB7YWN0aW9uczogW3snYWN0aW9uJzogJ3RhcCcsICdvcHRpb25zJzogeydlbGVtZW50JzogJzMnfX1dfVxuICAgICAgICB9KTtcbiAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZGVlcC5lcXVhbChbW3snYWN0aW9uJzogJ3RhcCcsICdvcHRpb25zJzogeydlbGVtZW50JzogJzMnfX1dLCAnZm9vJ10pO1xuICAgICAgfSk7XG5cbiAgICB9KTtcblxuICAgIGRlc2NyaWJlKCdjcmVhdGUgc2Vzc2lvbnMgdmlhIEhUVFAgZW5kcG9pbnQnLCBmdW5jdGlvbiAoKSB7XG4gICAgICBsZXQgc2Vzc2lvbklkO1xuXG4gICAgICBiZWZvcmVFYWNoKGZ1bmN0aW9uICgpIHtcbiAgICAgICAgc2Vzc2lvbklkID0gbnVsbDtcbiAgICAgIH0pO1xuICAgICAgYWZ0ZXJFYWNoKGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgaWYgKHNlc3Npb25JZCkge1xuICAgICAgICAgIGF3YWl0IGF4aW9zLmRlbGV0ZShgJHtiYXNlVXJsfS9zZXNzaW9uLyR7c2Vzc2lvbklkfWApO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgaXQoJ3Nob3VsZCBub3QgYWxsb3cgY3JlYXRlIHNlc3Npb24gd2l0aCBkZXNpcmVkIGNhcHMgKE1KU09OV1ApJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCBkZXNpcmVkQ2FwYWJpbGl0aWVzID0ge2E6ICdiJ307XG4gICAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb25gLFxuICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgIGRhdGE6IHtkZXNpcmVkQ2FwYWJpbGl0aWVzfVxuICAgICAgICB9KTtcbiAgICAgICAgc2hvdWxkLmVxdWFsKGRhdGEudmFsdWUsIG51bGwpO1xuICAgICAgfSk7XG4gICAgICBpdCgnc2hvdWxkIGZhaWwgdG8gY3JlYXRlIHNlc3Npb24gd2l0aG91dCBjYXBhYmlsaXRpZXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb25gLFxuICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgIGRhdGE6IHt9LFxuICAgICAgICB9KS5zaG91bGQuZXZlbnR1YWxseS5iZS5yZWplY3RlZFdpdGgoLzQwMC8pO1xuICAgICAgfSk7XG4gICAgICBpdCgnc2hvdWxkIGFsbG93IGNyZWF0ZSBzZXNzaW9uIHdpdGggY2FwYWJpbGl0aWVzIChXM0MpJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCB3M2NDYXBhYmlsaXRpZXMgPSB7YWx3YXlzTWF0Y2g6IHsnYXBwaXVtOmUnOiAnZid9fTtcbiAgICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbmAsXG4gICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgZGF0YToge2NhcGFiaWxpdGllczogdzNjQ2FwYWJpbGl0aWVzfVxuICAgICAgICB9KTtcbiAgICAgICAgc2hvdWxkLm5vdC5leGlzdChkYXRhLnN0YXR1cyk7XG4gICAgICAgIHNob3VsZC5ub3QuZXhpc3QoZGF0YS5zZXNzaW9uSWQpO1xuICAgICAgICBkYXRhLnZhbHVlLmNhcGFiaWxpdGllcy5zaG91bGQuZXFsKHczY0NhcGFiaWxpdGllcyk7XG4gICAgICAgIGRhdGEudmFsdWUuc2Vzc2lvbklkLnNob3VsZC5leGlzdDtcbiAgICAgICAgc2Vzc2lvbklkID0gZGF0YS52YWx1ZS5zZXNzaW9uSWQ7XG4gICAgICB9KTtcbiAgICAgIGl0KCdzaG91bGQgcmFpc2UgYW4gZXJyb3IgaWYgdGhlIGRyaXZlciBkb2VzIG5vdCBzdXBwb3J0IFczQyB5ZXQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IGNyZWF0ZVNlc3Npb25TdHViID0gc2FuZGJveC5zdHViKGRyaXZlciwgJ2NyZWF0ZVNlc3Npb24nKS5jYWxsc0Zha2UoZnVuY3Rpb24gKGNhcGFiaWxpdGllcykge1xuICAgICAgICAgIGRyaXZlci5zZXNzaW9uSWQgPSBudWxsO1xuICAgICAgICAgIHJldHVybiBCYXNlRHJpdmVyLnByb3RvdHlwZS5jcmVhdGVTZXNzaW9uLmNhbGwoZHJpdmVyLCBjYXBhYmlsaXRpZXMpO1xuICAgICAgICB9KTtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAvLyBsZXQge3N0YXR1cywgdmFsdWUsIHNlc3Npb25JZH0gPSBhd2FpdCByZXF1ZXN0KHtcbiAgICAgICAgICBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb25gLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgICAgIGNhcGFiaWxpdGllczoge1xuICAgICAgICAgICAgICAgIGFsd2F5c01hdGNoOiB7XG4gICAgICAgICAgICAgICAgICBwbGF0Zm9ybU5hbWU6ICdGYWtlJyxcbiAgICAgICAgICAgICAgICAgICdhcHBpdW06ZGV2aWNlTmFtZSc6ICdGYWtlJyxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgIGZpcnN0TWF0Y2g6IFt7fV0sXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSkuc2hvdWxkLmV2ZW50dWFsbHkuYmUucmVqZWN0ZWRXaXRoKC81MDAvKTtcbiAgICAgICAgfSBmaW5hbGx5IHtcbiAgICAgICAgICBjcmVhdGVTZXNzaW9uU3R1Yi5yZXN0b3JlKCk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuXG4gICAgICBkZXNjcmliZSgndzNjIGVuZHBvaW50cycsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgbGV0IHNlc3Npb25Vcmw7XG5cbiAgICAgICAgYmVmb3JlRWFjaChhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgLy8gU3RhcnQgYSBXM0Mgc2Vzc2lvblxuICAgICAgICAgIGNvbnN0IHt2YWx1ZX0gPSAoYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uYCxcbiAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgICBjYXBhYmlsaXRpZXM6IHtcbiAgICAgICAgICAgICAgICBhbHdheXNNYXRjaDoge1xuICAgICAgICAgICAgICAgICAgcGxhdGZvcm1OYW1lOiAnRmFrZScsXG4gICAgICAgICAgICAgICAgICAnYXBwaXVtOmRldmljZU5hbWUnOiAnQ29tbW9kb3JlIDY0JyxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgIGZpcnN0TWF0Y2g6IFt7fV0sXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIH0pKS5kYXRhO1xuICAgICAgICAgIHNlc3Npb25JZCA9IHZhbHVlLnNlc3Npb25JZDtcbiAgICAgICAgICBzZXNzaW9uVXJsID0gYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH1gO1xuICAgICAgICB9KTtcblxuICAgICAgICBpdChgc2hvdWxkIHRocm93IDQwMCBCYWQgUGFyYW1ldGVycyBleGNlcHRpb24gaWYgdGhlIHBhcmFtZXRlcnMgYXJlIGJhZGAsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICAgIHVybDogYCR7c2Vzc2lvblVybH0vYWN0aW9uc2AsXG4gICAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgICBiYWQ6ICdwYXJhbXMnLFxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0pO1xuICAgICAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNDAwKTtcblxuICAgICAgICAgIGNvbnN0IHtlcnJvcjogdzNjRXJyb3IsIG1lc3NhZ2UsIHN0YWNrdHJhY2V9ID0gZGF0YS52YWx1ZTtcbiAgICAgICAgICBtZXNzYWdlLnNob3VsZC5tYXRjaCgvUGFyYW1ldGVycyB3ZXJlIGluY29ycmVjdC8pO1xuICAgICAgICAgIHN0YWNrdHJhY2Uuc2hvdWxkLm1hdGNoKC9wcm90b2NvbC5qcy8pO1xuICAgICAgICAgIHczY0Vycm9yLnNob3VsZC5iZS5hLnN0cmluZztcbiAgICAgICAgICB3M2NFcnJvci5zaG91bGQuZXF1YWwoZXJyb3JzLkludmFsaWRBcmd1bWVudEVycm9yLmVycm9yKCkpO1xuICAgICAgICB9KTtcblxuICAgICAgICBpdChgc2hvdWxkIHRocm93IDQwNSBleGNlcHRpb24gaWYgdGhlIGNvbW1hbmQgaGFzbid0IGJlZW4gaW1wbGVtZW50ZWQgeWV0YCwgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtzZXNzaW9uVXJsfS9hY3Rpb25zYCxcbiAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgICAgIGFjdGlvbnM6IFtdLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICB9KTtcbiAgICAgICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDQwNSk7XG5cbiAgICAgICAgICBjb25zdCB7ZXJyb3I6IHczY0Vycm9yLCBtZXNzYWdlLCBzdGFja3RyYWNlfSA9IGRhdGEudmFsdWU7XG4gICAgICAgICAgbWVzc2FnZS5zaG91bGQubWF0Y2goL01ldGhvZCBoYXMgbm90IHlldCBiZWVuIGltcGxlbWVudGVkLyk7XG4gICAgICAgICAgc3RhY2t0cmFjZS5zaG91bGQubWF0Y2goL3Byb3RvY29sLmpzLyk7XG4gICAgICAgICAgdzNjRXJyb3Iuc2hvdWxkLmJlLmEuc3RyaW5nO1xuICAgICAgICAgIHczY0Vycm9yLnNob3VsZC5lcXVhbChlcnJvcnMuTm90WWV0SW1wbGVtZW50ZWRFcnJvci5lcnJvcigpKTtcbiAgICAgICAgICBtZXNzYWdlLnNob3VsZC5tYXRjaCgvTWV0aG9kIGhhcyBub3QgeWV0IGJlZW4gaW1wbGVtZW50ZWQvKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgaXQoYHNob3VsZCB0aHJvdyA1MDAgVW5rbm93biBFcnJvciBpZiB0aGUgY29tbWFuZCB0aHJvd3MgYW4gdW5leHBlY3RlZCBleGNlcHRpb25gLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgZHJpdmVyLnBlcmZvcm1BY3Rpb25zID0gKCkgPT4geyB0aHJvdyBuZXcgRXJyb3IoYERpZG4ndCB3b3JrYCk7IH07XG4gICAgICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICB1cmw6IGAke3Nlc3Npb25Vcmx9L2FjdGlvbnNgLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICAgIGRhdGE6IHtcbiAgICAgICAgICAgICAgYWN0aW9uczogW10sXG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSk7XG4gICAgICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg1MDApO1xuXG4gICAgICAgICAgY29uc3Qge2Vycm9yOiB3M2NFcnJvciwgbWVzc2FnZSwgc3RhY2t0cmFjZX0gPSBkYXRhLnZhbHVlO1xuICAgICAgICAgIHN0YWNrdHJhY2Uuc2hvdWxkLm1hdGNoKC9wcm90b2NvbC5qcy8pO1xuICAgICAgICAgIHczY0Vycm9yLnNob3VsZC5iZS5hLnN0cmluZztcbiAgICAgICAgICB3M2NFcnJvci5zaG91bGQuZXF1YWwoZXJyb3JzLlVua25vd25FcnJvci5lcnJvcigpKTtcbiAgICAgICAgICBtZXNzYWdlLnNob3VsZC5tYXRjaCgvRGlkbid0IHdvcmsvKTtcblxuICAgICAgICAgIGRlbGV0ZSBkcml2ZXIucGVyZm9ybUFjdGlvbnM7XG4gICAgICAgIH0pO1xuXG4gICAgICAgIGl0KGBzaG91bGQgdHJhbnNsYXRlIGVsZW1lbnQgZm9ybWF0IGZyb20gTUpTT05XUCB0byBXM0NgLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgY29uc3QgcmV0VmFsdWUgPSBbXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHNvbWV0aGluZzoge1xuICAgICAgICAgICAgICAgIFtNSlNPTldQX0VMRU1FTlRfS0VZXTogJ2Zvb28nLFxuICAgICAgICAgICAgICAgIG90aGVyOiAnYmFyJ1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9LCB7XG4gICAgICAgICAgICAgIFtNSlNPTldQX0VMRU1FTlRfS0VZXTogJ2JhcidcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAnaWdub3JlJyxcbiAgICAgICAgICBdO1xuXG4gICAgICAgICAgY29uc3QgZXhwZWN0ZWRWYWx1ZSA9IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgc29tZXRoaW5nOiB7XG4gICAgICAgICAgICAgICAgW01KU09OV1BfRUxFTUVOVF9LRVldOiAnZm9vbycsXG4gICAgICAgICAgICAgICAgW1czQ19FTEVNRU5UX0tFWV06ICdmb29vJyxcbiAgICAgICAgICAgICAgICBvdGhlcjogJ2JhcidcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSwge1xuICAgICAgICAgICAgICBbTUpTT05XUF9FTEVNRU5UX0tFWV06ICdiYXInLFxuICAgICAgICAgICAgICBbVzNDX0VMRU1FTlRfS0VZXTogJ2JhcidcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAnaWdub3JlJyxcbiAgICAgICAgICBdO1xuXG4gICAgICAgICAgY29uc3QgZmluZEVsZW1lbnRzQmFja3VwID0gZHJpdmVyLmZpbmRFbGVtZW50cztcbiAgICAgICAgICBkcml2ZXIuZmluZEVsZW1lbnRzID0gKCkgPT4gcmV0VmFsdWU7XG4gICAgICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3MucG9zdChgJHtzZXNzaW9uVXJsfS9lbGVtZW50c2AsIHtcbiAgICAgICAgICAgIHVzaW5nOiAnd2hhdGV2ZXInLFxuICAgICAgICAgICAgdmFsdWU6ICd3aGF0ZXZlcicsXG4gICAgICAgICAgfSk7XG4gICAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKGV4cGVjdGVkVmFsdWUpO1xuICAgICAgICAgIGRyaXZlci5maW5kRWxlbWVudHMgPSBmaW5kRWxlbWVudHNCYWNrdXA7XG4gICAgICAgIH0pO1xuXG4gICAgICAgIGl0KGBzaG91bGQgZmFpbCB3aXRoIGEgNDA4IGVycm9yIGlmIGl0IHRocm93cyBhIFRpbWVvdXRFcnJvciBleGNlcHRpb25gLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgbGV0IHNldFVybFN0dWIgPSBzYW5kYm94LnN0dWIoZHJpdmVyLCAnc2V0VXJsJykuY2FsbHNGYWtlKGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBlcnJvcnMuVGltZW91dEVycm9yO1xuICAgICAgICAgIH0pO1xuICAgICAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtzZXNzaW9uVXJsfS91cmxgLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICAgIGRhdGE6IHtcbiAgICAgICAgICAgICAgdXJsOiAnaHR0cHM6Ly9leGFtcGxlLmNvbS8nLFxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0pO1xuICAgICAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNDA4KTtcblxuICAgICAgICAgIGNvbnN0IHtlcnJvcjogdzNjRXJyb3IsIG1lc3NhZ2UsIHN0YWNrdHJhY2V9ID0gZGF0YS52YWx1ZTtcbiAgICAgICAgICBzdGFja3RyYWNlLnNob3VsZC5tYXRjaCgvcHJvdG9jb2wuanMvKTtcbiAgICAgICAgICB3M2NFcnJvci5zaG91bGQuYmUuYS5zdHJpbmc7XG4gICAgICAgICAgdzNjRXJyb3Iuc2hvdWxkLmVxdWFsKGVycm9ycy5UaW1lb3V0RXJyb3IuZXJyb3IoKSk7XG4gICAgICAgICAgbWVzc2FnZS5zaG91bGQubWF0Y2goL0FuIG9wZXJhdGlvbiBkaWQgbm90IGNvbXBsZXRlIGJlZm9yZSBpdHMgdGltZW91dCBleHBpcmVkLyk7XG5cbiAgICAgICAgICBzZXRVcmxTdHViLnJlc3RvcmUoKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgaXQoYHNob3VsZCBwYXNzIHdpdGggMjAwIEhUVFAgc3RhdHVzIGNvZGUgaWYgdGhlIGNvbW1hbmQgcmV0dXJucyBhIHZhbHVlYCwgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICAgIGRyaXZlci5wZXJmb3JtQWN0aW9ucyA9IChhY3Rpb25zKSA9PiAnSXQgd29ya3MgJyArIGFjdGlvbnMuam9pbignJyk7XG4gICAgICAgICAgY29uc3Qge3N0YXR1cywgdmFsdWUsIHNlc3Npb25JZH0gPSAoYXdhaXQgYXhpb3MucG9zdChgJHtzZXNzaW9uVXJsfS9hY3Rpb25zYCwge1xuICAgICAgICAgICAgYWN0aW9uczogWydhJywgJ2InLCAnYyddLFxuICAgICAgICAgIH0pKS5kYXRhO1xuICAgICAgICAgIHNob3VsZC5ub3QuZXhpc3Qoc2Vzc2lvbklkKTtcbiAgICAgICAgICBzaG91bGQubm90LmV4aXN0KHN0YXR1cyk7XG4gICAgICAgICAgdmFsdWUuc2hvdWxkLmVxdWFsKCdJdCB3b3JrcyBhYmMnKTtcbiAgICAgICAgICBkZWxldGUgZHJpdmVyLnBlcmZvcm1BY3Rpb25zO1xuICAgICAgICB9KTtcblxuICAgICAgICBkZXNjcmliZSgnandwcm94eScsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBsZXQgcG9ydDtcbiAgICAgICAgICBsZXQgc2VydmVyLCBqd3Byb3h5LCBhcHA7XG5cbiAgICAgICAgICBiZWZvcmUoYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICAgICAgcG9ydCA9IGF3YWl0IGdldFRlc3RQb3J0KHRydWUpO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgYmVmb3JlRWFjaChmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBjb25zdCByZXMgPSBjcmVhdGVQcm94eVNlcnZlcihzZXNzaW9uSWQsIHBvcnQpO1xuICAgICAgICAgICAgc2VydmVyID0gcmVzLnNlcnZlcjtcbiAgICAgICAgICAgIGFwcCA9IHJlcy5hcHA7XG4gICAgICAgICAgICBqd3Byb3h5ID0gbmV3IEpXUHJveHkoe2hvc3Q6IFRFU1RfSE9TVCwgcG9ydH0pO1xuICAgICAgICAgICAgandwcm94eS5zZXNzaW9uSWQgPSBzZXNzaW9uSWQ7XG4gICAgICAgICAgICBkcml2ZXIucGVyZm9ybUFjdGlvbnMgPSBhc3luYyAoYWN0aW9ucykgPT4gYXdhaXQgandwcm94eS5jb21tYW5kKCcvcGVyZm9ybS1hY3Rpb25zJywgJ1BPU1QnLCBhY3Rpb25zKTtcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIGFmdGVyRWFjaChhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBkZWxldGUgZHJpdmVyLnBlcmZvcm1BY3Rpb25zO1xuICAgICAgICAgICAgYXdhaXQgc2VydmVyLmNsb3NlKCk7XG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICBpdCgnc2hvdWxkIHdvcmsgaWYgYSBwcm94aWVkIHJlcXVlc3QgcmV0dXJucyBhIHJlc3BvbnNlIHdpdGggc3RhdHVzIDIwMCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICAgIGFwcC5wb3N0KCcvc2Vzc2lvbi86c2Vzc2lvbklkL3BlcmZvcm0tYWN0aW9ucycsIChyZXEsIHJlcykgPT4ge1xuICAgICAgICAgICAgICByZXMuanNvbih7XG4gICAgICAgICAgICAgICAgc2Vzc2lvbklkOiByZXEucGFyYW1zLnNlc3Npb25JZCxcbiAgICAgICAgICAgICAgICB2YWx1ZTogcmVxLmJvZHksXG4gICAgICAgICAgICAgICAgc3RhdHVzOiAwLFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgICBjb25zdCB7c3RhdHVzLCB2YWx1ZSwgc2Vzc2lvbklkfSA9IChhd2FpdCBheGlvcy5wb3N0KGAke3Nlc3Npb25Vcmx9L2FjdGlvbnNgLCB7XG4gICAgICAgICAgICAgIGFjdGlvbnM6IFsxLCAyLCAzXSxcbiAgICAgICAgICAgIH0pKS5kYXRhO1xuICAgICAgICAgICAgdmFsdWUuc2hvdWxkLmVxbChbMSwgMiwgM10pO1xuICAgICAgICAgICAgc2hvdWxkLm5vdC5leGlzdChzdGF0dXMpO1xuICAgICAgICAgICAgc2hvdWxkLm5vdC5leGlzdChzZXNzaW9uSWQpO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgaXQoJ3Nob3VsZCByZXR1cm4gZXJyb3IgaWYgYSBwcm94aWVkIHJlcXVlc3QgcmV0dXJucyBhIE1KU09OV1AgZXJyb3IgcmVzcG9uc2UnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBhcHAucG9zdCgnL3Nlc3Npb24vOnNlc3Npb25JZC9wZXJmb3JtLWFjdGlvbnMnLCAocmVxLCByZXMpID0+IHtcbiAgICAgICAgICAgICAgcmVzLnN0YXR1cyg1MDApLmpzb24oe1xuICAgICAgICAgICAgICAgIHNlc3Npb25JZCxcbiAgICAgICAgICAgICAgICBzdGF0dXM6IDYsXG4gICAgICAgICAgICAgICAgdmFsdWU6ICdBIHByb2JsZW0gb2NjdXJyZWQnLFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICAgIHVybDogYCR7c2Vzc2lvblVybH0vYWN0aW9uc2AsXG4gICAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgICAgIGFjdGlvbnM6IFsxLCAyLCAzXSxcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKEhUVFBTdGF0dXNDb2Rlcy5OT1RfRk9VTkQpO1xuICAgICAgICAgICAgSlNPTi5zdHJpbmdpZnkoZGF0YSkuc2hvdWxkLm1hdGNoKC9BIHByb2JsZW0gb2NjdXJyZWQvKTtcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIGl0KCdzaG91bGQgcmV0dXJuIFczQyBlcnJvciBpZiBhIHByb3hpZWQgcmVxdWVzdCByZXR1cm5zIGEgVzNDIGVycm9yIHJlc3BvbnNlJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICAgICAgY29uc3QgZXJyb3IgPSBuZXcgRXJyb3IoYFNvbWUgZXJyb3Igb2NjdXJyZWRgKTtcbiAgICAgICAgICAgIGVycm9yLnczY1N0YXR1cyA9IDQxNDtcbiAgICAgICAgICAgIGNvbnN0IGV4ZWN1dGVDb21tYW5kU3R1YiA9IHNhbmRib3guc3R1Yihkcml2ZXIsICdleGVjdXRlQ29tbWFuZCcpLnJldHVybnMoe1xuICAgICAgICAgICAgICBwcm90b2NvbDogJ1czQycsXG4gICAgICAgICAgICAgIGVycm9yLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICAgICAgdXJsOiBgJHtzZXNzaW9uVXJsfS9hY3Rpb25zYCxcbiAgICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICAgICAgICBkYXRhOiB7YWN0aW9uczogWzEsIDIsIDNdfSxcbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg0MTQpO1xuICAgICAgICAgICAgY29uc3Qge2Vycm9yOiB3M2NFcnJvciwgbWVzc2FnZTogZXJyTWVzc2FnZSwgc3RhY2t0cmFjZX0gPSBkYXRhLnZhbHVlO1xuICAgICAgICAgICAgdzNjRXJyb3Iuc2hvdWxkLmVxdWFsKCd1bmtub3duIGVycm9yJyk7XG4gICAgICAgICAgICBzdGFja3RyYWNlLnNob3VsZC5tYXRjaCgvU29tZSBlcnJvciBvY2N1cnJlZC8pO1xuICAgICAgICAgICAgZXJyTWVzc2FnZS5zaG91bGQuZXF1YWwoJ1NvbWUgZXJyb3Igb2NjdXJyZWQnKTtcbiAgICAgICAgICAgIGV4ZWN1dGVDb21tYW5kU3R1Yi5yZXN0b3JlKCk7XG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICBpdCgnc2hvdWxkIHJldHVybiBlcnJvciBpZiBhIHByb3hpZWQgcmVxdWVzdCByZXR1cm5zIGEgTUpTT05XUCBlcnJvciByZXNwb25zZSBidXQgSFRUUCBzdGF0dXMgY29kZSBpcyAyMDAnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBhcHAucG9zdCgnL3Nlc3Npb24vOnNlc3Npb25JZC9wZXJmb3JtLWFjdGlvbnMnLCAocmVxLCByZXMpID0+IHtcbiAgICAgICAgICAgICAgcmVzLnN0YXR1cygyMDApLmpzb24oe1xuICAgICAgICAgICAgICAgIHNlc3Npb25JZDogJ0Zha2UgU2Vzc2lvbiBJZCcsXG4gICAgICAgICAgICAgICAgc3RhdHVzOiA3LFxuICAgICAgICAgICAgICAgIHZhbHVlOiAnQSBwcm9ibGVtIG9jY3VycmVkJyxcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgICB1cmw6IGAke3Nlc3Npb25Vcmx9L2FjdGlvbnNgLFxuICAgICAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICAgICAgICAgIGRhdGE6IHtcbiAgICAgICAgICAgICAgICBhY3Rpb25zOiBbMSwgMiwgM10sXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbChIVFRQU3RhdHVzQ29kZXMuTk9UX0ZPVU5EKTtcbiAgICAgICAgICAgIGNvbnN0IHtlcnJvcjogdzNjRXJyb3IsIG1lc3NhZ2U6IGVyck1lc3NhZ2UsIHN0YWNrdHJhY2V9ID0gZGF0YS52YWx1ZTtcbiAgICAgICAgICAgIHczY0Vycm9yLnNob3VsZC5lcXVhbCgnbm8gc3VjaCBlbGVtZW50Jyk7XG4gICAgICAgICAgICBlcnJNZXNzYWdlLnNob3VsZC5tYXRjaCgvQSBwcm9ibGVtIG9jY3VycmVkLyk7XG4gICAgICAgICAgICBzdGFja3RyYWNlLnNob3VsZC5leGlzdDtcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIGl0KCdzaG91bGQgcmV0dXJuIGVycm9yIGlmIGEgcHJveGllZCByZXF1ZXN0IHJldHVybnMgYSBXM0MgZXJyb3IgcmVzcG9uc2UnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBhcHAucG9zdCgnL3Nlc3Npb24vOnNlc3Npb25JZC9wZXJmb3JtLWFjdGlvbnMnLCAocmVxLCByZXMpID0+IHtcbiAgICAgICAgICAgICAgcmVzLnN0YXR1cyg0MDQpLmpzb24oe1xuICAgICAgICAgICAgICAgIHZhbHVlOiB7XG4gICAgICAgICAgICAgICAgICBlcnJvcjogJ25vIHN1Y2ggZWxlbWVudCcsXG4gICAgICAgICAgICAgICAgICBtZXNzYWdlOiAnZG9lcyBub3QgbWFrZSBhIGRpZmZlcmVuY2UnLFxuICAgICAgICAgICAgICAgICAgc3RhY2t0cmFjZTogJ2FyYml0cmFyeSBzdGFja3RyYWNlJyxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICAgIHVybDogYCR7c2Vzc2lvblVybH0vYWN0aW9uc2AsXG4gICAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgICAgIGFjdGlvbnM6IFsxLCAyLCAzXSxcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKEhUVFBTdGF0dXNDb2Rlcy5OT1RfRk9VTkQpO1xuICAgICAgICAgICAgY29uc3Qge2Vycm9yOiB3M2NFcnJvciwgc3RhY2t0cmFjZX0gPSBkYXRhLnZhbHVlO1xuICAgICAgICAgICAgdzNjRXJyb3Iuc2hvdWxkLmVxdWFsKCdubyBzdWNoIGVsZW1lbnQnKTtcbiAgICAgICAgICAgIHN0YWNrdHJhY2Uuc2hvdWxkLm1hdGNoKC9hcmJpdHJhcnkgc3RhY2t0cmFjZS8pO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgaXQoJ3Nob3VsZCByZXR1cm4gYW4gZXJyb3IgaWYgYSBwcm94aWVkIHJlcXVlc3QgcmV0dXJucyBhIFczQyBlcnJvciByZXNwb25zZScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICAgIGFwcC5wb3N0KCcvc2Vzc2lvbi86c2Vzc2lvbklkL3BlcmZvcm0tYWN0aW9ucycsIChyZXEsIHJlcykgPT4ge1xuICAgICAgICAgICAgICByZXMuc2V0KCdDb25uZWN0aW9uJywgJ2Nsb3NlJyk7XG4gICAgICAgICAgICAgIHJlcy5zdGF0dXMoNDQ0KS5qc29uKHtcbiAgICAgICAgICAgICAgICB2YWx1ZToge1xuICAgICAgICAgICAgICAgICAgZXJyb3I6ICdib2d1cyBlcnJvciBjb2RlJyxcbiAgICAgICAgICAgICAgICAgIG1lc3NhZ2U6ICdkb2VzIG5vdCBtYWtlIGEgZGlmZmVyZW5jZScsXG4gICAgICAgICAgICAgICAgICBzdGFja3RyYWNlOiAnYXJiaXRyYXJ5IHN0YWNrdHJhY2UnLFxuICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICAgICAgdXJsOiBgJHtzZXNzaW9uVXJsfS9hY3Rpb25zYCxcbiAgICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgICAgICAgYWN0aW9uczogWzEsIDIsIDNdLFxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoSFRUUFN0YXR1c0NvZGVzLklOVEVSTkFMX1NFUlZFUl9FUlJPUik7XG4gICAgICAgICAgICBjb25zdCB7ZXJyb3I6IHczY0Vycm9yLCBzdGFja3RyYWNlfSA9IGRhdGEudmFsdWU7XG4gICAgICAgICAgICB3M2NFcnJvci5zaG91bGQuZXF1YWwoJ3Vua25vd24gZXJyb3InKTtcbiAgICAgICAgICAgIHN0YWNrdHJhY2Uuc2hvdWxkLm1hdGNoKC9hcmJpdHJhcnkgc3RhY2t0cmFjZS8pO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgIH0pO1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGhhbmRsZSBjb21tYW5kcyB3aXRoIG5vIHJlc3BvbnNlIHZhbHVlcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9mb3J3YXJkYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICB9KTtcbiAgICAgIGRhdGEuc2hvdWxkLmVxbCh7XG4gICAgICAgIHZhbHVlOiBudWxsLFxuICAgICAgICBzZXNzaW9uSWQ6ICdmb28nXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgYWxsb3cgZW1wdHkgc3RyaW5nIHJlc3BvbnNlIHZhbHVlcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci90ZXh0YCxcbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICcnLFxuICAgICAgICBzZXNzaW9uSWQ6ICdmb28nXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgc2VuZCA1MDAgcmVzcG9uc2UgYW5kIGFuIFVua25vd24gb2JqZWN0IGZvciByZWplY3RlZCBjb21tYW5kcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL3JlZnJlc2hgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg1MDApO1xuICAgICAgZGF0YS52YWx1ZS5lcnJvci5zaG91bGQuZXFsKCd1bmtub3duIGVycm9yJyk7XG4gICAgICBkYXRhLnZhbHVlLm1lc3NhZ2Uuc2hvdWxkLmVxbCgnQW4gdW5rbm93biBzZXJ2ZXItc2lkZSBlcnJvciBvY2N1cnJlZCB3aGlsZSBwcm9jZXNzaW5nICcgK1xuICAgICAgICAndGhlIGNvbW1hbmQuIE9yaWdpbmFsIGVycm9yOiBUb28gRnJlc2ghJyk7XG4gICAgICBkYXRhLnNlc3Npb25JZC5zaG91bGQuZXFsKCdmb28nKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgbm90IHRocm93IFVua25vd25FcnJvciB3aGVuIGtub3duJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb29gLFxuICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDQwNCk7XG4gICAgICBkYXRhLnZhbHVlLmVycm9yLnNob3VsZC5lcWwoJ2ludmFsaWQgc2Vzc2lvbiBpZCcpO1xuICAgICAgZGF0YS52YWx1ZS5tZXNzYWdlLnNob3VsZC5lcWwoJ0Egc2Vzc2lvbiBpcyBlaXRoZXIgdGVybWluYXRlZCBvciBub3Qgc3RhcnRlZCcpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG4gIH0pO1xuXG4gIGRlc2NyaWJlKCdzZXNzaW9uIElkcycsIGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgZHJpdmVyID0gbmV3IEZha2VEcml2ZXIoKTtcbiAgICBsZXQgbWpzb253cFNlcnZlcjtcblxuICAgIGJlZm9yZShhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBtanNvbndwU2VydmVyID0gYXdhaXQgc2VydmVyKHtcbiAgICAgICAgcm91dGVDb25maWd1cmluZ0Z1bmN0aW9uOiByb3V0ZUNvbmZpZ3VyaW5nRnVuY3Rpb24oZHJpdmVyKSxcbiAgICAgICAgcG9ydCxcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgYWZ0ZXIoYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgYXdhaXQgbWpzb253cFNlcnZlci5jbG9zZSgpO1xuICAgIH0pO1xuXG4gICAgYWZ0ZXJFYWNoKGZ1bmN0aW9uICgpIHtcbiAgICAgIGRyaXZlci5zZXNzaW9uSWQgPSBudWxsO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCByZXR1cm4gbnVsbCBTZXNzaW9uSWQgZm9yIGNvbW1hbmRzIHdpdGhvdXQgc2Vzc2lvbklkcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zdGF0dXNgLFxuICAgICAgfSk7XG5cbiAgICAgIHNob3VsZC5lcXVhbChkYXRhLnNlc3Npb25JZCwgbnVsbCk7XG4gICAgfSk7XG5cbiAgICBpdCgncmVzcG9uZHMgd2l0aCB0aGUgc2FtZSBzZXNzaW9uIElEIGluIHRoZSByZXF1ZXN0JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgbGV0IHNlc3Npb25JZCA9ICdWYWRlciBTZXNzaW9ucyc7XG4gICAgICBkcml2ZXIuc2Vzc2lvbklkID0gc2Vzc2lvbklkO1xuXG4gICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHt1cmw6ICdodHRwOi8vZ29vZ2xlLmNvbSd9XG4gICAgICB9KTtcblxuICAgICAgc2hvdWxkLmV4aXN0KGRhdGEuc2Vzc2lvbklkKTtcbiAgICAgIGRhdGEuc2Vzc2lvbklkLnNob3VsZC5lcWwoc2Vzc2lvbklkKTtcbiAgICB9KTtcblxuICAgIGl0KCd5ZWxscyBpZiBubyBzZXNzaW9uIGV4aXN0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGxldCBzZXNzaW9uSWQgPSAnVmFkZXIgU2Vzc2lvbnMnO1xuXG4gICAgICBjb25zdCB7ZGF0YSwgc3RhdHVzfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uLyR7c2Vzc2lvbklkfS91cmxgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICAgIGRhdGE6IHt1cmw6ICdodHRwOi8vZ29vZ2xlLmNvbSd9LFxuICAgICAgfSk7XG5cbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNDA0KTtcbiAgICAgIGRhdGEudmFsdWUubWVzc2FnZS5zaG91bGQuY29udGFpbignc2Vzc2lvbicpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3llbGxzIGlmIGludmFsaWQgc2Vzc2lvbiBpcyBzZW50JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgbGV0IHNlc3Npb25JZCA9ICdWYWRlciBTZXNzaW9ucyc7XG4gICAgICBkcml2ZXIuc2Vzc2lvbklkID0gJ3JlY2Vzc2lvbic7XG5cbiAgICAgIGNvbnN0IHtkYXRhLCBzdGF0dXN9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vJHtzZXNzaW9uSWR9L3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgZGF0YToge3VybDogJ2h0dHA6Ly9nb29nbGUuY29tJ30sXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg0MDQpO1xuICAgICAgZGF0YS52YWx1ZS5tZXNzYWdlLnNob3VsZC5jb250YWluKCdzZXNzaW9uJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGhhdmUgc2Vzc2lvbiBJRHMgaW4gZXJyb3IgcmVzcG9uc2VzJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgbGV0IHNlc3Npb25JZCA9ICdWYWRlciBTZXNzaW9ucyc7XG4gICAgICBkcml2ZXIuc2Vzc2lvbklkID0gc2Vzc2lvbklkO1xuXG4gICAgICBjb25zdCB7ZGF0YSwgc3RhdHVzfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uLyR7c2Vzc2lvbklkfS9yZWZyZXNoYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgfSk7XG5cbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNTAwKTtcbiAgICAgIGRhdGEudmFsdWUuZXJyb3Iuc2hvdWxkLmVxbCgndW5rbm93biBlcnJvcicpO1xuICAgICAgZGF0YS52YWx1ZS5tZXNzYWdlLnNob3VsZC5lcWwoJ0FuIHVua25vd24gc2VydmVyLXNpZGUgZXJyb3Igb2NjdXJyZWQgd2hpbGUgcHJvY2Vzc2luZyAnICtcbiAgICAgICAgJ3RoZSBjb21tYW5kLiBPcmlnaW5hbCBlcnJvcjogVG9vIEZyZXNoIScpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnVmFkZXIgU2Vzc2lvbnMnKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgcmV0dXJuIGEgbmV3IHNlc3Npb24gSUQgb24gY3JlYXRlJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb25gLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgZGF0YToge2NhcGFiaWxpdGllczoge2Fsd2F5c01hdGNoOiB7J2FwcGl1bTpncmVldGluZyc6ICdoZWxsbyd9LCBmaXJzdE1hdGNoOiBbe31dfX1cbiAgICAgIH0pO1xuXG4gICAgICBzaG91bGQuZXhpc3QoZGF0YS52YWx1ZS5zZXNzaW9uSWQpO1xuICAgICAgZGF0YS52YWx1ZS5zZXNzaW9uSWQuaW5kZXhPZignZmFrZVNlc3Npb25fJykuc2hvdWxkLmVxdWFsKDApO1xuICAgICAgZGF0YS52YWx1ZS5jYXBhYmlsaXRpZXMuc2hvdWxkLmVxbCh7YWx3YXlzTWF0Y2g6IHsnYXBwaXVtOmdyZWV0aW5nJzogJ2hlbGxvJ30sIGZpcnN0TWF0Y2g6IFt7fV19KTtcbiAgICB9KTtcbiAgfSk7XG5cbiAgZGVzY3JpYmUoJ3ZpYSBkcml2ZXJzIGpzb253cCBwcm94eScsIGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgZHJpdmVyO1xuICAgIGxldCBzZXNzaW9uSWQgPSAnZm9vJztcbiAgICBsZXQgbWpzb253cFNlcnZlcjtcblxuICAgIGJlZm9yZUVhY2goYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgZHJpdmVyID0gbmV3IEZha2VEcml2ZXIoKTtcbiAgICAgIGRyaXZlci5zZXNzaW9uSWQgPSBzZXNzaW9uSWQ7XG4gICAgICBkcml2ZXIucHJveHlBY3RpdmUgPSAoKSA9PiB0cnVlO1xuICAgICAgZHJpdmVyLmNhblByb3h5ID0gKCkgPT4gdHJ1ZTtcblxuICAgICAgbWpzb253cFNlcnZlciA9IGF3YWl0IHNlcnZlcih7XG4gICAgICAgIHJvdXRlQ29uZmlndXJpbmdGdW5jdGlvbjogcm91dGVDb25maWd1cmluZ0Z1bmN0aW9uKGRyaXZlciksXG4gICAgICAgIHBvcnQsXG4gICAgICAgIGV4dHJhTWV0aG9kTWFwOiBGYWtlRHJpdmVyLm5ld01ldGhvZE1hcFxuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBhZnRlckVhY2goYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgYXdhaXQgbWpzb253cFNlcnZlci5jbG9zZSgpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBnaXZlIGEgbmljZSBlcnJvciBpZiBwcm94eWluZyBpcyBzZXQgYnV0IG5vIHByb3h5IGZ1bmN0aW9uIGV4aXN0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGRyaXZlci5jYW5Qcm94eSA9ICgpID0+IGZhbHNlO1xuICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDUwMCk7XG4gICAgICBkYXRhLnZhbHVlLmVycm9yLnNob3VsZC5lcWwoJ3Vua25vd24gZXJyb3InKTtcbiAgICAgIGRhdGEudmFsdWUubWVzc2FnZS5zaG91bGQuZXFsKCdBbiB1bmtub3duIHNlcnZlci1zaWRlIGVycm9yIG9jY3VycmVkIHdoaWxlIHByb2Nlc3NpbmcgJyArXG4gICAgICAgICd0aGUgY29tbWFuZC4gT3JpZ2luYWwgZXJyb3I6IFRyeWluZyB0byBwcm94eSB0byBhICcgK1xuICAgICAgICAnc2VydmVyIGJ1dCB0aGUgZHJpdmVyIGlzIHVuYWJsZSB0byBwcm94eScpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHBhc3Mgb24gYW55IGVycm9ycyBpbiBwcm94eWluZycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGRyaXZlci5wcm94eVJlcVJlcyA9IGFzeW5jIGZ1bmN0aW9uICgpIHsgLy8gZXNsaW50LWRpc2FibGUtbGluZSByZXF1aXJlLWF3YWl0XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignZm9vJyk7XG4gICAgICB9O1xuICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDUwMCk7XG4gICAgICBkYXRhLnZhbHVlLmVycm9yLnNob3VsZC5lcWwoJ3Vua25vd24gZXJyb3InKTtcbiAgICAgIGRhdGEudmFsdWUubWVzc2FnZS5zaG91bGQuZXFsKCdBbiB1bmtub3duIHNlcnZlci1zaWRlIGVycm9yIG9jY3VycmVkIHdoaWxlIHByb2Nlc3NpbmcgJyArXG4gICAgICAgICd0aGUgY29tbWFuZC4gT3JpZ2luYWwgZXJyb3I6IENvdWxkIG5vdCBwcm94eS4gUHJveHkgZXJyb3I6IGZvbycpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGFibGUgdG8gdGhyb3cgUHJveHlSZXF1ZXN0RXJyb3IgaW4gcHJveHlpbmcnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBkcml2ZXIucHJveHlSZXFSZXMgPSBhc3luYyBmdW5jdGlvbiAoKSB7IC8vIGVzbGludC1kaXNhYmxlLWxpbmUgcmVxdWlyZS1hd2FpdFxuICAgICAgICBsZXQganNvbndwID0ge3N0YXR1czogMzUsIHZhbHVlOiAnTm8gc3VjaCBjb250ZXh0IGZvdW5kLicsIHNlc3Npb25JZDogJ2Zvbyd9O1xuICAgICAgICB0aHJvdyBuZXcgZXJyb3JzLlByb3h5UmVxdWVzdEVycm9yKGBDb3VsZCBub3QgcHJveHkgY29tbWFuZCB0byByZW1vdGUgc2VydmVyLiBgLCBqc29ud3ApO1xuICAgICAgfTtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vJHtzZXNzaW9uSWR9L3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgZGF0YToge3VybDogJ2h0dHA6Ly9nb29nbGUuY29tJ30sXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg0MDApO1xuICAgICAgZGF0YS52YWx1ZS5lcnJvci5zaG91bGQuZXFsKCd1bmtub3duIGVycm9yJyk7XG4gICAgICBkYXRhLnZhbHVlLm1lc3NhZ2Uuc2hvdWxkLmVxbCgnTm8gc3VjaCBjb250ZXh0IGZvdW5kLicpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGxldCB0aGUgcHJveHkgaGFuZGxlIHJlcS9yZXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBkcml2ZXIucHJveHlSZXFSZXMgPSBhc3luYyBmdW5jdGlvbiAocmVxLCByZXMpIHsgLy8gZXNsaW50LWRpc2FibGUtbGluZSByZXF1aXJlLWF3YWl0XG4gICAgICAgIHJlcy5zdGF0dXMoMjAwKS5qc29uKHtjdXN0b206ICdkYXRhJ30pO1xuICAgICAgfTtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vJHtzZXNzaW9uSWR9L3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfVxuICAgICAgfSk7XG5cbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoMjAwKTtcbiAgICAgIGRhdGEuc2hvdWxkLmVxbCh7Y3VzdG9tOiAnZGF0YSd9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgYXZvaWQganNvbndwIHByb3h5aW5nIHdoZW4gcGF0aCBtYXRjaGVzIGF2b2lkYW5jZSBsaXN0JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgZHJpdmVyLmdldFByb3h5QXZvaWRMaXN0ID0gKCkgPT4gW1snUE9TVCcsIG5ldyBSZWdFeHAoJ14vc2Vzc2lvbi9bXi9dKy91cmwkJyldXTtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vJHtzZXNzaW9uSWR9L3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDIwMCk7XG4gICAgICBkYXRhLnNob3VsZC5lcWwoe1xuICAgICAgICB2YWx1ZTogJ05hdmlnYXRlZCB0bzogaHR0cDovL2dvb2dsZS5jb20nLFxuICAgICAgICBzZXNzaW9uSWRcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBmYWlsIGlmIGF2b2lkIHByb3h5IGxpc3QgaXMgbWFsZm9ybWVkIGluIHNvbWUgd2F5JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgYXN5bmMgZnVuY3Rpb24gYmFkUHJveHlBdm9pZGFuY2VMaXN0IChsaXN0KSB7XG4gICAgICAgIGRyaXZlci5nZXRQcm94eUF2b2lkTGlzdCA9ICgpID0+IGxpc3Q7XG4gICAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vdXJsYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg1MDApO1xuICAgICAgICBkYXRhLnZhbHVlLm1lc3NhZ2Uuc2hvdWxkLmNvbnRhaW4oJ3JveHknKTtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGxpc3RzID0gW1xuICAgICAgICAnZm9vJyxcbiAgICAgICAgW1snZm9vJ11dLFxuICAgICAgICBbWydCQVInLCAvbG9sL11dLFxuICAgICAgICBbWydHRVQnLCAnZm9vJ11dXG4gICAgICBdO1xuICAgICAgZm9yIChsZXQgbGlzdCBvZiBsaXN0cykge1xuICAgICAgICBhd2FpdCBiYWRQcm94eUF2b2lkYW5jZUxpc3QobGlzdCk7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGF2b2lkIHByb3h5aW5nIG5vbi1zZXNzaW9uIGNvbW1hbmRzIGV2ZW4gaWYgbm90IGluIHRoZSBsaXN0JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgZHJpdmVyLmdldFByb3h5QXZvaWRMaXN0ID0gKCkgPT4gW1snUE9TVCcsIG5ldyBSZWdFeHAoJycpXV07XG5cbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3N0YXR1c2AsXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCgyMDApO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6IFwiSSdtIGZpbmVcIixcbiAgICAgICAgc2Vzc2lvbklkOiBudWxsXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgYXZvaWQgcHJveHlpbmcgZGVsZXRlU2Vzc2lvbiBjb21tYW5kcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGRyaXZlci5nZXRQcm94eUF2b2lkTGlzdCA9ICgpID0+IFtbJ1BPU1QnLCBuZXcgUmVnRXhwKCcnKV1dO1xuXG4gICAgICBkcml2ZXIuc2Vzc2lvbklkLnNob3VsZC5lcXVhbChzZXNzaW9uSWQpO1xuICAgICAgY29uc3Qge3N0YXR1c30gPSBhd2FpdCBheGlvcy5kZWxldGUoYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH1gKTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCgyMDApO1xuICAgICAgc2hvdWxkLm5vdC5leGlzdChkcml2ZXIuc2Vzc2lvbklkKTtcbiAgICAgIGRyaXZlci5qd3BQcm94eUFjdGl2ZS5zaG91bGQuYmUuZmFsc2U7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGF2b2lkIHByb3h5aW5nIHdoZW4gY29tbWFuZCBzcGVjIHNwZWNpZmllcyBuZXZlclByb3h5JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vbm9wcm94eWAsXG4gICAgICAgIG1ldGhvZDogJ0dFVCcsXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCgyMDApO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdUaGlzIHdhcyBub3QgcHJveGllZCcsXG4gICAgICAgIHNlc3Npb25JZFxuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgfSk7XG59KTtcbiJdLCJmaWxlIjoidGVzdC9wcm90b2NvbC9wcm90b2NvbC1lMmUtc3BlY3MuanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4ifQ==