@appium/base-driver 8.3.0 → 8.3.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 (110) hide show
  1. package/build/lib/basedriver/capabilities.js +2 -4
  2. package/build/lib/basedriver/commands/event.js +2 -4
  3. package/build/lib/basedriver/commands/find.js +2 -4
  4. package/build/lib/basedriver/commands/index.js +2 -4
  5. package/build/lib/basedriver/commands/log.js +2 -4
  6. package/build/lib/basedriver/commands/session.js +2 -4
  7. package/build/lib/basedriver/commands/settings.js +2 -4
  8. package/build/lib/basedriver/commands/timeout.js +2 -4
  9. package/build/lib/basedriver/desired-caps.js +2 -4
  10. package/build/lib/basedriver/device-settings.js +2 -4
  11. package/build/lib/basedriver/driver.js +17 -6
  12. package/build/lib/basedriver/helpers.js +2 -4
  13. package/build/lib/basedriver/logger.js +2 -4
  14. package/build/lib/constants.js +2 -4
  15. package/build/lib/express/crash.js +2 -4
  16. package/build/lib/express/express-logging.js +2 -4
  17. package/build/lib/express/idempotency.js +2 -4
  18. package/build/lib/express/logger.js +2 -4
  19. package/build/lib/express/middleware.js +2 -4
  20. package/build/lib/express/server.js +2 -4
  21. package/build/lib/express/static.js +2 -4
  22. package/build/lib/express/websocket.js +2 -4
  23. package/build/lib/helpers/capabilities.js +2 -4
  24. package/build/lib/index.js +2 -4
  25. package/build/lib/jsonwp-proxy/protocol-converter.js +2 -4
  26. package/build/lib/jsonwp-proxy/proxy.js +2 -4
  27. package/build/lib/jsonwp-status/status.js +2 -4
  28. package/build/lib/protocol/errors.js +2 -4
  29. package/build/lib/protocol/helpers.js +2 -4
  30. package/build/lib/protocol/index.js +2 -4
  31. package/build/lib/protocol/protocol.js +2 -4
  32. package/build/lib/protocol/routes.js +2 -4
  33. package/build/lib/protocol/validators.js +2 -4
  34. package/build/test/basedriver/README.md +5 -0
  35. package/build/test/basedriver/driver-e2e-tests.js +2 -4
  36. package/build/test/basedriver/driver-tests.js +2 -4
  37. package/build/test/basedriver/index.js +2 -4
  38. package/build/test/e2e/basedriver/driver.e2e.spec.js +15 -0
  39. package/build/test/e2e/basedriver/helpers.e2e.spec.js +192 -0
  40. package/build/test/e2e/basedriver/websockets.e2e.spec.js +82 -0
  41. package/build/test/e2e/express/server.e2e.spec.js +159 -0
  42. package/build/test/e2e/jsonwp-proxy/proxy.e2e.spec.js +59 -0
  43. package/build/test/e2e/protocol/fake-driver.js +163 -0
  44. package/build/test/e2e/protocol/helpers.js +25 -0
  45. package/build/test/e2e/protocol/protocol.e2e.spec.js +1186 -0
  46. package/build/test/helpers.js +2 -4
  47. package/build/test/unit/basedriver/capabilities.spec.js +672 -0
  48. package/build/test/unit/basedriver/capability.spec.js +353 -0
  49. package/build/test/unit/basedriver/commands/event.spec.js +110 -0
  50. package/build/test/unit/basedriver/commands/log.spec.js +85 -0
  51. package/build/test/unit/basedriver/driver.spec.js +15 -0
  52. package/build/test/unit/basedriver/helpers.spec.js +151 -0
  53. package/build/test/unit/basedriver/timeout.spec.js +135 -0
  54. package/build/test/unit/express/server.spec.js +155 -0
  55. package/build/test/unit/express/static.spec.js +26 -0
  56. package/build/test/unit/jsonwp-proxy/mock-request.js +91 -0
  57. package/build/test/unit/jsonwp-proxy/protocol-converter.spec.js +171 -0
  58. package/build/test/unit/jsonwp-proxy/proxy.spec.js +292 -0
  59. package/build/test/unit/jsonwp-proxy/url.spec.js +165 -0
  60. package/build/test/unit/jsonwp-status/status.spec.js +34 -0
  61. package/build/test/unit/protocol/errors.spec.js +390 -0
  62. package/build/test/unit/protocol/routes.spec.js +80 -0
  63. package/build/test/unit/protocol/validator.spec.js +149 -0
  64. package/lib/basedriver/driver.js +15 -4
  65. package/package.json +14 -9
  66. package/test/basedriver/README.md +5 -0
  67. package/test/basedriver/driver-e2e-tests.js +1 -1
  68. package/build/test/basedriver/capabilities-specs.js +0 -674
  69. package/build/test/basedriver/capability-specs.js +0 -355
  70. package/build/test/basedriver/commands/event-specs.js +0 -112
  71. package/build/test/basedriver/commands/log-specs.js +0 -87
  72. package/build/test/basedriver/driver-e2e-specs.js +0 -17
  73. package/build/test/basedriver/driver-specs.js +0 -17
  74. package/build/test/basedriver/helpers-e2e-specs.js +0 -194
  75. package/build/test/basedriver/helpers-specs.js +0 -153
  76. package/build/test/basedriver/timeout-specs.js +0 -137
  77. package/build/test/basedriver/websockets-e2e-specs.js +0 -84
  78. package/build/test/express/server-e2e-specs.js +0 -161
  79. package/build/test/express/server-specs.js +0 -157
  80. package/build/test/express/static-specs.js +0 -28
  81. package/build/test/jsonwp-proxy/mock-request.js +0 -93
  82. package/build/test/jsonwp-proxy/protocol-converter-specs.js +0 -173
  83. package/build/test/jsonwp-proxy/proxy-e2e-specs.js +0 -61
  84. package/build/test/jsonwp-proxy/proxy-specs.js +0 -294
  85. package/build/test/jsonwp-proxy/url-specs.js +0 -167
  86. package/build/test/jsonwp-status/status-specs.js +0 -36
  87. package/build/test/protocol/errors-specs.js +0 -388
  88. package/build/test/protocol/fake-driver.js +0 -165
  89. package/build/test/protocol/helpers.js +0 -27
  90. package/build/test/protocol/protocol-e2e-specs.js +0 -1188
  91. package/build/test/protocol/routes-specs.js +0 -82
  92. package/build/test/protocol/validator-specs.js +0 -151
  93. package/test/basedriver/capabilities-specs.js +0 -537
  94. package/test/basedriver/capability-specs.js +0 -383
  95. package/test/basedriver/commands/event-specs.js +0 -74
  96. package/test/basedriver/commands/log-specs.js +0 -79
  97. package/test/basedriver/driver-e2e-specs.js +0 -8
  98. package/test/basedriver/driver-specs.js +0 -8
  99. package/test/basedriver/fixtures/BadZippedApp.zip +0 -1
  100. package/test/basedriver/fixtures/FakeAndroidApp.apk +0 -1
  101. package/test/basedriver/fixtures/FakeAndroidApp.asd +0 -0
  102. package/test/basedriver/fixtures/FakeIOSApp.app +0 -1
  103. package/test/basedriver/fixtures/FakeIOSApp.app.zip +0 -0
  104. package/test/basedriver/fixtures/FakeIOSApp.ipa +0 -0
  105. package/test/basedriver/fixtures/custom-element-finder-bad.js +0 -5
  106. package/test/basedriver/fixtures/custom-element-finder.js +0 -29
  107. package/test/basedriver/helpers-e2e-specs.js +0 -187
  108. package/test/basedriver/helpers-specs.js +0 -137
  109. package/test/basedriver/timeout-specs.js +0 -128
  110. package/test/basedriver/websockets-e2e-specs.js +0 -75
@@ -0,0 +1,1186 @@
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
+ });
1186
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Rlc3QvZTJlL3Byb3RvY29sL3Byb3RvY29sLmUyZS5zcGVjLmpzIl0sIm5hbWVzIjpbInBvcnQiLCJiYXNlVXJsIiwiZGVzY3JpYmUiLCJzYW5kYm94IiwiYmVmb3JlRWFjaCIsImFmdGVyRWFjaCIsInJlc3RvcmUiLCJiZWZvcmUiLCJURVNUX0hPU1QiLCJkIiwiRmFrZURyaXZlciIsIml0Iiwic2V0VXJsIiwic2hvdWxkIiwiY29udGFpbiIsIm1qc29ud3BTZXJ2ZXIiLCJkcml2ZXIiLCJzZXNzaW9uSWQiLCJyb3V0ZUNvbmZpZ3VyaW5nRnVuY3Rpb24iLCJhZnRlciIsImNsb3NlIiwiZGF0YSIsInVybCIsIm1ldGhvZCIsImVxbCIsInZhbHVlIiwicmVxRGF0YSIsIlVSTFNlYXJjaFBhcmFtcyIsInNldCIsImhlYWRlcnMiLCJ0b1N0cmluZyIsInN0YXR1cyIsInZhbGlkYXRlU3RhdHVzIiwiZXF1YWwiLCJKU09OIiwic3RyaW5naWZ5IiwiZXZlbnR1YWxseSIsImJlIiwicmVqZWN0ZWQiLCJyZWplY3RlZFdpdGgiLCJpbmNsdWRlIiwiZXJyb3IiLCJtZXNzYWdlIiwiaWQiLCJyZXNwb25zZSIsInRleHQiLCJlbGVtZW50IiwieG9mZnNldCIsInlvZmZzZXQiLCJhcHBJZCIsImJ1bmRsZUlkIiwiZGVlcCIsImFjdGlvbnMiLCJheGlvcyIsImRlbGV0ZSIsImRlc2lyZWRDYXBhYmlsaXRpZXMiLCJhIiwidzNjQ2FwYWJpbGl0aWVzIiwiYWx3YXlzTWF0Y2giLCJjYXBhYmlsaXRpZXMiLCJub3QiLCJleGlzdCIsImNyZWF0ZVNlc3Npb25TdHViIiwic3R1YiIsImNhbGxzRmFrZSIsIkJhc2VEcml2ZXIiLCJwcm90b3R5cGUiLCJjcmVhdGVTZXNzaW9uIiwiY2FsbCIsInBsYXRmb3JtTmFtZSIsImZpcnN0TWF0Y2giLCJzZXNzaW9uVXJsIiwiYmFkIiwidzNjRXJyb3IiLCJzdGFja3RyYWNlIiwibWF0Y2giLCJzdHJpbmciLCJlcnJvcnMiLCJJbnZhbGlkQXJndW1lbnRFcnJvciIsIk5vdFlldEltcGxlbWVudGVkRXJyb3IiLCJwZXJmb3JtQWN0aW9ucyIsIkVycm9yIiwiVW5rbm93bkVycm9yIiwicmV0VmFsdWUiLCJzb21ldGhpbmciLCJNSlNPTldQX0VMRU1FTlRfS0VZIiwib3RoZXIiLCJleHBlY3RlZFZhbHVlIiwiVzNDX0VMRU1FTlRfS0VZIiwiZmluZEVsZW1lbnRzQmFja3VwIiwiZmluZEVsZW1lbnRzIiwicG9zdCIsInVzaW5nIiwic2V0VXJsU3R1YiIsIlRpbWVvdXRFcnJvciIsImpvaW4iLCJzZXJ2ZXIiLCJqd3Byb3h5IiwiYXBwIiwicmVzIiwiSldQcm94eSIsImhvc3QiLCJjb21tYW5kIiwicmVxIiwianNvbiIsInBhcmFtcyIsImJvZHkiLCJIVFRQU3RhdHVzQ29kZXMiLCJOT1RfRk9VTkQiLCJ3M2NTdGF0dXMiLCJleGVjdXRlQ29tbWFuZFN0dWIiLCJyZXR1cm5zIiwicHJvdG9jb2wiLCJlcnJNZXNzYWdlIiwiSU5URVJOQUxfU0VSVkVSX0VSUk9SIiwiaW5kZXhPZiIsInByb3h5QWN0aXZlIiwiY2FuUHJveHkiLCJleHRyYU1ldGhvZE1hcCIsIm5ld01ldGhvZE1hcCIsInByb3h5UmVxUmVzIiwianNvbndwIiwiUHJveHlSZXF1ZXN0RXJyb3IiLCJjdXN0b20iLCJnZXRQcm94eUF2b2lkTGlzdCIsIlJlZ0V4cCIsImJhZFByb3h5QXZvaWRhbmNlTGlzdCIsImxpc3QiLCJsaXN0cyIsImp3cFByb3h5QWN0aXZlIiwiZmFsc2UiXSwibWFwcGluZ3MiOiI7Ozs7OztBQUVBOztBQUdBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUdBOztBQUVBLElBQUlBLElBQUo7QUFDQSxJQUFJQyxPQUFKO0FBRUFDLFFBQVEsQ0FBQyxVQUFELEVBQWEsWUFBWTtBQUMvQixNQUFJQyxPQUFKO0FBRUFDLEVBQUFBLFVBQVUsQ0FBQyxZQUFZO0FBQ3JCRCxJQUFBQSxPQUFPLEdBQUcsMkJBQVY7QUFDRCxHQUZTLENB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QVgsQ0FBaUI3QyxNQUFNLENBQUNDLFNBQXhCO0FBQ0FELE1BQUFBLE1BQU0sQ0FBQ2lILGNBQVAsQ0FBc0JwSCxNQUF0QixDQUE2QndCLEVBQTdCLENBQWdDNkYsS0FBaEM7QUFDRCxLQVRDLENBQUY7QUFXQXZILElBQUFBLEVBQUUsQ0FBQyw4REFBRCxFQUFpRSxrQkFBa0I7QUFDbkYsWUFBTTtBQUFDb0IsUUFBQUEsTUFBRDtBQUFTVixRQUFBQTtBQUFULFVBQWlCLE1BQU0sb0JBQU07QUFDakNDLFFBQUFBLEdBQUcsRUFBRyxHQUFFckIsT0FBUSxZQUFXZ0IsU0FBVSxVQURKO0FBRWpDTSxRQUFBQSxNQUFNLEVBQUU7QUFGeUIsT0FBTixDQUE3QjtBQUtBUSxNQUFBQSxNQUFNLENBQUNsQixNQUFQLENBQWNvQixLQUFkLENBQW9CLEdBQXBCO0FBQ0FaLE1BQUFBLElBQUksQ0FBQ1IsTUFBTCxDQUFZVyxHQUFaLENBQWdCO0FBQ2RDLFFBQUFBLEtBQUssRUFBRSxzQkFETztBQUVkUixRQUFBQTtBQUZjLE9BQWhCO0FBSUQsS0FYQyxDQUFGO0FBYUQsR0F0S08sQ0FBUjtBQXVLRCxDQTErQk8sQ0FBUiIsInNvdXJjZXNDb250ZW50IjpbIi8vIHRyYW5zcGlsZTptb2NoYVxuXG5pbXBvcnQge1xuICBzZXJ2ZXIsIHJvdXRlQ29uZmlndXJpbmdGdW5jdGlvbiwgZXJyb3JzLCBKV1Byb3h5LCBCYXNlRHJpdmVyXG59IGZyb20gJy4uLy4uLy4uL2xpYic7XG5pbXBvcnQgeyBGYWtlRHJpdmVyIH0gZnJvbSAnLi9mYWtlLWRyaXZlcic7XG5pbXBvcnQgYXhpb3MgZnJvbSAnYXhpb3MnO1xuaW1wb3J0IHsgY3JlYXRlU2FuZGJveCB9IGZyb20gJ3Npbm9uJztcbmltcG9ydCB7IFN0YXR1c0NvZGVzIGFzIEhUVFBTdGF0dXNDb2RlcyB9IGZyb20gJ2h0dHAtc3RhdHVzLWNvZGVzJztcbmltcG9ydCB7IGNyZWF0ZVByb3h5U2VydmVyIH0gZnJvbSAnLi9oZWxwZXJzJztcbmltcG9ydCB7XG4gIE1KU09OV1BfRUxFTUVOVF9LRVksIFczQ19FTEVNRU5UX0tFWVxufSBmcm9tICcuLi8uLi8uLi9saWIvY29uc3RhbnRzJztcbmltcG9ydCB7VEVTVF9IT1NULCBnZXRUZXN0UG9ydH0gZnJvbSAnLi4vLi4vaGVscGVycyc7XG5cbmxldCBwb3J0O1xubGV0IGJhc2VVcmw7XG5cbmRlc2NyaWJlKCdQcm90b2NvbCcsIGZ1bmN0aW9uICgpIHtcbiAgbGV0IHNhbmRib3g7XG5cbiAgYmVmb3JlRWFjaChmdW5jdGlvbiAoKSB7XG4gICAgc2FuZGJveCA9IGNyZWF0ZVNhbmRib3goKTtcbiAgfSk7XG5cbiAgYWZ0ZXJFYWNoKGZ1bmN0aW9uICgpIHtcbiAgICBzYW5kYm94LnJlc3RvcmUoKTtcbiAgfSk7XG5cbiAgYmVmb3JlKGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICBwb3J0ID0gYXdhaXQgZ2V0VGVzdFBvcnQoKTtcbiAgICBiYXNlVXJsID0gYGh0dHA6Ly8ke1RFU1RfSE9TVH06JHtwb3J0fWA7XG4gIH0pO1xuXG4gIC8vVE9ETzogbW9yZSB0ZXN0cyE6XG4gIC8vIFVua25vd24gY29tbWFuZHMgc2hvdWxkIHJldHVybiA0MDRcblxuICBkZXNjcmliZSgnZGlyZWN0IHRvIGRyaXZlcicsIGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgZCA9IG5ldyBGYWtlRHJpdmVyKCk7XG4gICAgaXQoJ3Nob3VsZCByZXR1cm4gcmVzcG9uc2UgdmFsdWVzIGRpcmVjdGx5IGZyb20gdGhlIGRyaXZlcicsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIChhd2FpdCBkLnNldFVybCgnaHR0cDovL2dvb2dsZS5jb20nKSkuc2hvdWxkLmNvbnRhaW4oJ2dvb2dsZScpO1xuICAgIH0pO1xuICB9KTtcblxuICBkZXNjcmliZSgndmlhIGV4cHJlc3Mgcm91dGVyJywgZnVuY3Rpb24gKCkge1xuICAgIGxldCBtanNvbndwU2VydmVyO1xuICAgIGxldCBkcml2ZXI7XG5cbiAgICBiZWZvcmUoYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgZHJpdmVyID0gbmV3IEZha2VEcml2ZXIoKTtcbiAgICAgIGRyaXZlci5zZXNzaW9uSWQgPSAnZm9vJztcbiAgICAgIG1qc29ud3BTZXJ2ZXIgPSBhd2FpdCBzZXJ2ZXIoe1xuICAgICAgICByb3V0ZUNvbmZpZ3VyaW5nRnVuY3Rpb246IHJvdXRlQ29uZmlndXJpbmdGdW5jdGlvbihkcml2ZXIpLFxuICAgICAgICBwb3J0LFxuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBhZnRlcihhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBhd2FpdCBtanNvbndwU2VydmVyLmNsb3NlKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHByb3h5IHRvIGRyaXZlciBhbmQgcmV0dXJuIHZhbGlkIGpzb253cCByZXNwb25zZScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby91cmxgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgZGF0YToge3VybDogJ2h0dHA6Ly9nb29nbGUuY29tJ31cbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdOYXZpZ2F0ZWQgdG86IGh0dHA6Ly9nb29nbGUuY29tJyxcbiAgICAgICAgc2Vzc2lvbklkOiAnZm9vJ1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGFzc3VtZSByZXF1ZXN0cyB3aXRob3V0IGEgQ29udGVudC1UeXBlIGFyZSBqc29uIHJlcXVlc3RzJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdOYXZpZ2F0ZWQgdG86IGh0dHA6Ly9nb29nbGUuY29tJyxcbiAgICAgICAgc2Vzc2lvbklkOiAnZm9vJ1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHJlc3BvbmQgdG8geC13d3ctZm9ybS11cmxlbmNvZGVkIGFzIHdlbGwgYXMganNvbiByZXF1ZXN0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHJlcURhdGEgPSBuZXcgVVJMU2VhcmNoUGFyYW1zKCk7XG4gICAgICByZXFEYXRhLnNldCgndXJsJywgJ2h0dHA6Ly9nb29nbGUuY29tJyk7XG4gICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vdXJsYCxcbiAgICAgICAgaGVhZGVyczoge1xuICAgICAgICAgICdDb250ZW50LVR5cGUnOiAnYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkJyxcbiAgICAgICAgfSxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHJlcURhdGEudG9TdHJpbmcoKSxcbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdOYXZpZ2F0ZWQgdG86IGh0dHA6Ly9nb29nbGUuY29tJyxcbiAgICAgICAgc2Vzc2lvbklkOiAnZm9vJ1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGluY2x1ZGUgdXJsIHJlcXVlc3QgcGFyYW1ldGVycyBmb3IgbWV0aG9kcyB0byB1c2UgLSBzZXNzaW9uaWQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vYmFja2AsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7fSxcbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdmb28nLFxuICAgICAgICBzZXNzaW9uSWQ6ICdmb28nXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgaW5jbHVkZSB1cmwgcmVxdWVzdCBwYXJhbWV0ZXJzIGZvciBtZXRob2RzIHRvIHVzZSAtIGVsZW1lbnRpZCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci9jbGlja2AsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7fSxcbiAgICAgIH0pO1xuICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFsnYmFyJywgJ2ZvbyddKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgaW5jbHVkZSB1cmwgcmVxIHBhcmFtcyBpbiB0aGUgb3JkZXI6IGN1c3RvbSwgZWxlbWVudCwgc2Vzc2lvbicsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci9hdHRyaWJ1dGUvYmF6YCxcbiAgICAgIH0pO1xuICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFsnYmF6JywgJ2JhcicsICdmb28nXSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHJlc3BvbmQgd2l0aCA0MDAgQmFkIFJlcXVlc3QgaWYgcGFyYW1ldGVycyBtaXNzaW5nJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge2RhdGEsIHN0YXR1c30gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHt9LFxuICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgIH0pO1xuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg0MDApO1xuICAgICAgSlNPTi5zdHJpbmdpZnkoZGF0YSkuc2hvdWxkLmNvbnRhaW4oJ3VybCcpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCByZWplY3QgcmVxdWVzdHMgd2l0aCBhIGJhZGx5IGZvcm1hdHRlZCBib2R5IGFuZCBub3QgY3Jhc2gnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6ICdvaCBoZWxsbydcbiAgICAgIH0pLnNob3VsZC5ldmVudHVhbGx5LmJlLnJlamVjdGVkO1xuXG4gICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHt1cmw6ICdodHRwOi8vZ29vZ2xlLmNvbSd9XG4gICAgICB9KTtcbiAgICAgIGRhdGEuc2hvdWxkLmVxbCh7XG4gICAgICAgIHZhbHVlOiAnTmF2aWdhdGVkIHRvOiBodHRwOi8vZ29vZ2xlLmNvbScsXG4gICAgICAgIHNlc3Npb25JZDogJ2ZvbydcbiAgICAgIH0pO1xuXG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGdldCA0MDQgZm9yIGJhZCByb3V0ZXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vYmxhcmdpbWFyZ2AsXG4gICAgICB9KS5zaG91bGQuZXZlbnR1YWxseS5iZS5yZWplY3RlZFdpdGgoLzQwNC8pO1xuICAgIH0pO1xuXG4gICAgaXQoJzR4eCByZXNwb25zZXMgc2hvdWxkIGhhdmUgY29udGVudC10eXBlIG9mIGFwcGxpY2F0aW9uL2pzb24nLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCB7aGVhZGVyc30gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vYmxhcmdpbWFyZ2FyaXRhYCxcbiAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICB9KTtcblxuICAgICAgaGVhZGVyc1snY29udGVudC10eXBlJ10uc2hvdWxkLmluY2x1ZGUoJ2FwcGxpY2F0aW9uL2pzb24nKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdGhyb3cgbm90IHlldCBpbXBsZW1lbnRlZCBmb3IgdW5maWxsZWRvdXQgY29tbWFuZHMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci9sb2NhdGlvbmAsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgfSk7XG5cbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNDA1KTtcbiAgICAgIGRhdGEudmFsdWUuZXJyb3Iuc2hvdWxkLmVxbCgndW5rbm93biBtZXRob2QnKTtcbiAgICAgIGRhdGEudmFsdWUubWVzc2FnZS5zaG91bGQuZXFsKCdNZXRob2QgaGFzIG5vdCB5ZXQgYmVlbiBpbXBsZW1lbnRlZCcpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHRocm93IG5vdCBpbXBsZW1lbnRlZCBmb3IgaWdub3JlZCBjb21tYW5kcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2J1dHRvbnVwYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICBkYXRhOiB7fSxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDQwNSk7XG4gICAgICBkYXRhLnZhbHVlLmVycm9yLnNob3VsZC5lcWwoJ3Vua25vd24gbWV0aG9kJyk7XG4gICAgICBkYXRhLnZhbHVlLm1lc3NhZ2Uuc2hvdWxkLmVxbCgnTWV0aG9kIGhhcyBub3QgeWV0IGJlZW4gaW1wbGVtZW50ZWQnKTtcbiAgICAgIGRhdGEuc2Vzc2lvbklkLnNob3VsZC5lcWwoJ2ZvbycpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBnZXQgNDAwIGZvciBiYWQgcGFyYW1ldGVycycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby91cmxgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgZGF0YToge31cbiAgICAgIH0pLnNob3VsZC5ldmVudHVhbGx5LmJlLnJlamVjdGVkV2l0aCgvNDAwLyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGlnbm9yZSBzcGVjaWFsIGV4dHJhIHBheWxvYWQgcGFyYW1zIGluIHRoZSByaWdodCBjb250ZXh0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci92YWx1ZWAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7aWQ6ICdiYXonLCBzZXNzaW9uSWQ6ICdsb2wnLCB2YWx1ZTogWydhJ119XG4gICAgICB9KTtcblxuICAgICAgYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2VsZW1lbnQvYmFyL3ZhbHVlYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHtpZDogJ2Jheid9XG4gICAgICB9KS5zaG91bGQuZXZlbnR1YWxseS5iZS5yZWplY3RlZFdpdGgoLzQwMC8pO1xuXG4gICAgICAvLyBtYWtlIHN1cmUgYWRkaW5nIHRoZSBvcHRpb25hbCAnaWQnIGRvZXNuJ3QgY2xvYmJlciBhIHJvdXRlIHdoZXJlIHdlXG4gICAgICAvLyBoYXZlIGFuIGFjdHVhbCByZXF1aXJlZCAnaWQnXG4gICAgICBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb28vZnJhbWVgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgZGF0YToge2lkOiAnYmF6J31cbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCByZXR1cm4gdGhlIGNvcnJlY3QgZXJyb3IgZXZlbiBpZiBkcml2ZXIgZG9lcyBub3QgdGhyb3cnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9hcHBpdW0vcmVjZWl2ZV9hc3luY19yZXNwb25zZWAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7cmVzcG9uc2U6ICdiYXonfSxcbiAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICB9KTtcbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNTAwKTtcbiAgICAgIGRhdGEudmFsdWUuZXJyb3Iuc2hvdWxkLmVxbCgndW5rbm93biBlcnJvcicpO1xuICAgICAgZGF0YS52YWx1ZS5tZXNzYWdlLnNob3VsZC5lcWwoJ0FuIHVua25vd24gc2VydmVyLXNpZGUgZXJyb3Igb2NjdXJyZWQgd2hpbGUgcHJvY2Vzc2luZyAnICtcbiAgICAgICAgJ3RoZSBjb21tYW5kLiBPcmlnaW5hbCBlcnJvcjogTWlzaGFuZGxlZCBEcml2ZXIgRXJyb3InKTtcbiAgICAgIGRhdGEuc2Vzc2lvbklkLnNob3VsZC5lcWwoJ2ZvbycpO1xuICAgIH0pO1xuXG4gICAgZGVzY3JpYmUoJ3czYyBzZW5ka2V5cyBtaWdyYXRpb24nLCBmdW5jdGlvbiAoKSB7XG4gICAgICBpdCgnc2hvdWxkIGFjY2VwdCB2YWx1ZSBmb3Igc2VuZGtleXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2VsZW1lbnQvYmFyL3ZhbHVlYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiB7dmFsdWU6ICd0ZXh0IHRvIHR5cGUnfVxuICAgICAgICB9KTtcbiAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFsndGV4dCB0byB0eXBlJywgJ2JhciddKTtcbiAgICAgIH0pO1xuICAgICAgaXQoJ3Nob3VsZCBhY2NlcHQgdGV4dCBmb3Igc2VuZGtleXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2VsZW1lbnQvYmFyL3ZhbHVlYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiB7dGV4dDogJ3RleHQgdG8gdHlwZSd9XG4gICAgICAgIH0pO1xuICAgICAgICBkYXRhLnZhbHVlLnNob3VsZC5lcWwoWyd0ZXh0IHRvIHR5cGUnLCAnYmFyJ10pO1xuICAgICAgfSk7XG4gICAgICBpdCgnc2hvdWxkIGFjY2VwdCB2YWx1ZSBhbmQgdGV4dCBmb3Igc2VuZGtleXMsIGFuZCB1c2UgdmFsdWUnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2VsZW1lbnQvYmFyL3ZhbHVlYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiB7dmFsdWU6ICd0ZXh0IHRvIHR5cGUnLCB0ZXh0OiAndGV4dCB0byBpZ25vcmUnfVxuICAgICAgICB9KTtcbiAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFsndGV4dCB0byB0eXBlJywgJ2JhciddKTtcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgZGVzY3JpYmUoJ211bHRpcGxlIHNldHMgb2YgYXJndW1lbnRzJywgZnVuY3Rpb24gKCkge1xuICAgICAgZGVzY3JpYmUoJ29wdGlvbmFsJywgZnVuY3Rpb24gKCkge1xuICAgICAgICBpdCgnc2hvdWxkIGFsbG93IG1vdmV0byB3aXRoIGVsZW1lbnQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9tb3ZldG9gLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICBkYXRhOiB7ZWxlbWVudDogJzMnfVxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGRhdGEudmFsdWUuc2hvdWxkLmVxbChbJzMnLCBudWxsLCBudWxsXSk7XG4gICAgICAgIH0pO1xuICAgICAgICBpdCgnc2hvdWxkIGFsbG93IG1vdmV0byB3aXRoIHhvZmZzZXQveW9mZnNldCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL21vdmV0b2AsXG4gICAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgIGRhdGE6IHt4b2Zmc2V0OiA0MiwgeW9mZnNldDogMTd9XG4gICAgICAgICAgfSk7XG4gICAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKFtudWxsLCA0MiwgMTddKTtcbiAgICAgICAgfSk7XG4gICAgICB9KTtcbiAgICAgIGRlc2NyaWJlKCdyZXF1aXJlZCcsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgaXQoJ3Nob3VsZCBhbGxvdyByZW1vdmVBcHAgd2l0aCBhcHBJZCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL2FwcGl1bS9kZXZpY2UvcmVtb3ZlX2FwcGAsXG4gICAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgIGRhdGE6IHthcHBJZDogNDJ9XG4gICAgICAgICAgfSk7XG4gICAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKDQyKTtcbiAgICAgICAgfSk7XG4gICAgICAgIGl0KCdzaG91bGQgYWxsb3cgcmVtb3ZlQXBwIHdpdGggYnVuZGxlSWQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9hcHBpdW0vZGV2aWNlL3JlbW92ZV9hcHBgLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICBkYXRhOiB7YnVuZGxlSWQ6IDQyfVxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGRhdGEudmFsdWUuc2hvdWxkLmVxbCg0Mik7XG4gICAgICAgIH0pO1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBkZXNjcmliZSgnZGVmYXVsdCBwYXJhbSB3cmFwJywgZnVuY3Rpb24gKCkge1xuICAgICAgaXQoJ3Nob3VsZCB3cmFwJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby90b3VjaC9wZXJmb3JtYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiBbeydhY3Rpb24nOiAndGFwJywgJ29wdGlvbnMnOiB7J2VsZW1lbnQnOiAnMyd9fV1cbiAgICAgICAgfSk7XG4gICAgICAgIGRhdGEudmFsdWUuc2hvdWxkLmRlZXAuZXF1YWwoW1t7J2FjdGlvbic6ICd0YXAnLCAnb3B0aW9ucyc6IHsnZWxlbWVudCc6ICczJ319XSwgJ2ZvbyddKTtcbiAgICAgIH0pO1xuXG4gICAgICBpdCgnc2hvdWxkIG5vdCB3cmFwIHR3aWNlJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby90b3VjaC9wZXJmb3JtYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICBkYXRhOiB7YWN0aW9uczogW3snYWN0aW9uJzogJ3RhcCcsICdvcHRpb25zJzogeydlbGVtZW50JzogJzMnfX1dfVxuICAgICAgICB9KTtcbiAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZGVlcC5lcXVhbChbW3snYWN0aW9uJzogJ3RhcCcsICdvcHRpb25zJzogeydlbGVtZW50JzogJzMnfX1dLCAnZm9vJ10pO1xuICAgICAgfSk7XG5cbiAgICB9KTtcblxuICAgIGRlc2NyaWJlKCdjcmVhdGUgc2Vzc2lvbnMgdmlhIEhUVFAgZW5kcG9pbnQnLCBmdW5jdGlvbiAoKSB7XG4gICAgICBsZXQgc2Vzc2lvbklkO1xuXG4gICAgICBiZWZvcmVFYWNoKGZ1bmN0aW9uICgpIHtcbiAgICAgICAgc2Vzc2lvbklkID0gbnVsbDtcbiAgICAgIH0pO1xuICAgICAgYWZ0ZXJFYWNoKGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgaWYgKHNlc3Npb25JZCkge1xuICAgICAgICAgIGF3YWl0IGF4aW9zLmRlbGV0ZShgJHtiYXNlVXJsfS9zZXNzaW9uLyR7c2Vzc2lvbklkfWApO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgaXQoJ3Nob3VsZCBub3QgYWxsb3cgY3JlYXRlIHNlc3Npb24gd2l0aCBkZXNpcmVkIGNhcHMgKE1KU09OV1ApJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCBkZXNpcmVkQ2FwYWJpbGl0aWVzID0ge2E6ICdiJ307XG4gICAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb25gLFxuICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgIGRhdGE6IHtkZXNpcmVkQ2FwYWJpbGl0aWVzfVxuICAgICAgICB9KTtcbiAgICAgICAgc2hvdWxkLmVxdWFsKGRhdGEudmFsdWUsIG51bGwpO1xuICAgICAgfSk7XG4gICAgICBpdCgnc2hvdWxkIGZhaWwgdG8gY3JlYXRlIHNlc3Npb24gd2l0aG91dCBjYXBhYmlsaXRpZXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb25gLFxuICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgIGRhdGE6IHt9LFxuICAgICAgICB9KS5zaG91bGQuZXZlbnR1YWxseS5iZS5yZWplY3RlZFdpdGgoLzQwMC8pO1xuICAgICAgfSk7XG4gICAgICBpdCgnc2hvdWxkIGFsbG93IGNyZWF0ZSBzZXNzaW9uIHdpdGggY2FwYWJpbGl0aWVzIChXM0MpJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCB3M2NDYXBhYmlsaXRpZXMgPSB7YWx3YXlzTWF0Y2g6IHsnYXBwaXVtOmUnOiAnZid9fTtcbiAgICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbmAsXG4gICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgZGF0YToge2NhcGFiaWxpdGllczogdzNjQ2FwYWJpbGl0aWVzfVxuICAgICAgICB9KTtcbiAgICAgICAgc2hvdWxkLm5vdC5leGlzdChkYXRhLnN0YXR1cyk7XG4gICAgICAgIHNob3VsZC5ub3QuZXhpc3QoZGF0YS5zZXNzaW9uSWQpO1xuICAgICAgICBkYXRhLnZhbHVlLmNhcGFiaWxpdGllcy5zaG91bGQuZXFsKHczY0NhcGFiaWxpdGllcyk7XG4gICAgICAgIGRhdGEudmFsdWUuc2Vzc2lvbklkLnNob3VsZC5leGlzdDtcbiAgICAgICAgc2Vzc2lvbklkID0gZGF0YS52YWx1ZS5zZXNzaW9uSWQ7XG4gICAgICB9KTtcbiAgICAgIGl0KCdzaG91bGQgcmFpc2UgYW4gZXJyb3IgaWYgdGhlIGRyaXZlciBkb2VzIG5vdCBzdXBwb3J0IFczQyB5ZXQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IGNyZWF0ZVNlc3Npb25TdHViID0gc2FuZGJveC5zdHViKGRyaXZlciwgJ2NyZWF0ZVNlc3Npb24nKS5jYWxsc0Zha2UoZnVuY3Rpb24gKGNhcGFiaWxpdGllcykge1xuICAgICAgICAgIGRyaXZlci5zZXNzaW9uSWQgPSBudWxsO1xuICAgICAgICAgIHJldHVybiBCYXNlRHJpdmVyLnByb3RvdHlwZS5jcmVhdGVTZXNzaW9uLmNhbGwoZHJpdmVyLCBjYXBhYmlsaXRpZXMpO1xuICAgICAgICB9KTtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAvLyBsZXQge3N0YXR1cywgdmFsdWUsIHNlc3Npb25JZH0gPSBhd2FpdCByZXF1ZXN0KHtcbiAgICAgICAgICBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb25gLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgICAgIGNhcGFiaWxpdGllczoge1xuICAgICAgICAgICAgICAgIGFsd2F5c01hdGNoOiB7XG4gICAgICAgICAgICAgICAgICBwbGF0Zm9ybU5hbWU6ICdGYWtlJyxcbiAgICAgICAgICAgICAgICAgICdhcHBpdW06ZGV2aWNlTmFtZSc6ICdGYWtlJyxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgIGZpcnN0TWF0Y2g6IFt7fV0sXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSkuc2hvdWxkLmV2ZW50dWFsbHkuYmUucmVqZWN0ZWRXaXRoKC81MDAvKTtcbiAgICAgICAgfSBmaW5hbGx5IHtcbiAgICAgICAgICBjcmVhdGVTZXNzaW9uU3R1Yi5yZXN0b3JlKCk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuXG4gICAgICBkZXNjcmliZSgndzNjIGVuZHBvaW50cycsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgbGV0IHNlc3Npb25Vcmw7XG5cbiAgICAgICAgYmVmb3JlRWFjaChhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgLy8gU3RhcnQgYSBXM0Mgc2Vzc2lvblxuICAgICAgICAgIGNvbnN0IHt2YWx1ZX0gPSAoYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uYCxcbiAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgICBjYXBhYmlsaXRpZXM6IHtcbiAgICAgICAgICAgICAgICBhbHdheXNNYXRjaDoge1xuICAgICAgICAgICAgICAgICAgcGxhdGZvcm1OYW1lOiAnRmFrZScsXG4gICAgICAgICAgICAgICAgICAnYXBwaXVtOmRldmljZU5hbWUnOiAnQ29tbW9kb3JlIDY0JyxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgIGZpcnN0TWF0Y2g6IFt7fV0sXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIH0pKS5kYXRhO1xuICAgICAgICAgIHNlc3Npb25JZCA9IHZhbHVlLnNlc3Npb25JZDtcbiAgICAgICAgICBzZXNzaW9uVXJsID0gYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH1gO1xuICAgICAgICB9KTtcblxuICAgICAgICBpdChgc2hvdWxkIHRocm93IDQwMCBCYWQgUGFyYW1ldGVycyBleGNlcHRpb24gaWYgdGhlIHBhcmFtZXRlcnMgYXJlIGJhZGAsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICAgIHVybDogYCR7c2Vzc2lvblVybH0vYWN0aW9uc2AsXG4gICAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgICBiYWQ6ICdwYXJhbXMnLFxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0pO1xuICAgICAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNDAwKTtcblxuICAgICAgICAgIGNvbnN0IHtlcnJvcjogdzNjRXJyb3IsIG1lc3NhZ2UsIHN0YWNrdHJhY2V9ID0gZGF0YS52YWx1ZTtcbiAgICAgICAgICBtZXNzYWdlLnNob3VsZC5tYXRjaCgvUGFyYW1ldGVycyB3ZXJlIGluY29ycmVjdC8pO1xuICAgICAgICAgIHN0YWNrdHJhY2Uuc2hvdWxkLm1hdGNoKC9wcm90b2NvbC5qcy8pO1xuICAgICAgICAgIHczY0Vycm9yLnNob3VsZC5iZS5hLnN0cmluZztcbiAgICAgICAgICB3M2NFcnJvci5zaG91bGQuZXF1YWwoZXJyb3JzLkludmFsaWRBcmd1bWVudEVycm9yLmVycm9yKCkpO1xuICAgICAgICB9KTtcblxuICAgICAgICBpdChgc2hvdWxkIHRocm93IDQwNSBleGNlcHRpb24gaWYgdGhlIGNvbW1hbmQgaGFzbid0IGJlZW4gaW1wbGVtZW50ZWQgeWV0YCwgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtzZXNzaW9uVXJsfS9hY3Rpb25zYCxcbiAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgICAgIGFjdGlvbnM6IFtdLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICB9KTtcbiAgICAgICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDQwNSk7XG5cbiAgICAgICAgICBjb25zdCB7ZXJyb3I6IHczY0Vycm9yLCBtZXNzYWdlLCBzdGFja3RyYWNlfSA9IGRhdGEudmFsdWU7XG4gICAgICAgICAgbWVzc2FnZS5zaG91bGQubWF0Y2goL01ldGhvZCBoYXMgbm90IHlldCBiZWVuIGltcGxlbWVudGVkLyk7XG4gICAgICAgICAgc3RhY2t0cmFjZS5zaG91bGQubWF0Y2goL3Byb3RvY29sLmpzLyk7XG4gICAgICAgICAgdzNjRXJyb3Iuc2hvdWxkLmJlLmEuc3RyaW5nO1xuICAgICAgICAgIHczY0Vycm9yLnNob3VsZC5lcXVhbChlcnJvcnMuTm90WWV0SW1wbGVtZW50ZWRFcnJvci5lcnJvcigpKTtcbiAgICAgICAgICBtZXNzYWdlLnNob3VsZC5tYXRjaCgvTWV0aG9kIGhhcyBub3QgeWV0IGJlZW4gaW1wbGVtZW50ZWQvKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgaXQoYHNob3VsZCB0aHJvdyA1MDAgVW5rbm93biBFcnJvciBpZiB0aGUgY29tbWFuZCB0aHJvd3MgYW4gdW5leHBlY3RlZCBleGNlcHRpb25gLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgZHJpdmVyLnBlcmZvcm1BY3Rpb25zID0gKCkgPT4geyB0aHJvdyBuZXcgRXJyb3IoYERpZG4ndCB3b3JrYCk7IH07XG4gICAgICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICB1cmw6IGAke3Nlc3Npb25Vcmx9L2FjdGlvbnNgLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICAgIGRhdGE6IHtcbiAgICAgICAgICAgICAgYWN0aW9uczogW10sXG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSk7XG4gICAgICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg1MDApO1xuXG4gICAgICAgICAgY29uc3Qge2Vycm9yOiB3M2NFcnJvciwgbWVzc2FnZSwgc3RhY2t0cmFjZX0gPSBkYXRhLnZhbHVlO1xuICAgICAgICAgIHN0YWNrdHJhY2Uuc2hvdWxkLm1hdGNoKC9wcm90b2NvbC5qcy8pO1xuICAgICAgICAgIHczY0Vycm9yLnNob3VsZC5iZS5hLnN0cmluZztcbiAgICAgICAgICB3M2NFcnJvci5zaG91bGQuZXF1YWwoZXJyb3JzLlVua25vd25FcnJvci5lcnJvcigpKTtcbiAgICAgICAgICBtZXNzYWdlLnNob3VsZC5tYXRjaCgvRGlkbid0IHdvcmsvKTtcblxuICAgICAgICAgIGRlbGV0ZSBkcml2ZXIucGVyZm9ybUFjdGlvbnM7XG4gICAgICAgIH0pO1xuXG4gICAgICAgIGl0KGBzaG91bGQgdHJhbnNsYXRlIGVsZW1lbnQgZm9ybWF0IGZyb20gTUpTT05XUCB0byBXM0NgLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgY29uc3QgcmV0VmFsdWUgPSBbXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHNvbWV0aGluZzoge1xuICAgICAgICAgICAgICAgIFtNSlNPTldQX0VMRU1FTlRfS0VZXTogJ2Zvb28nLFxuICAgICAgICAgICAgICAgIG90aGVyOiAnYmFyJ1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9LCB7XG4gICAgICAgICAgICAgIFtNSlNPTldQX0VMRU1FTlRfS0VZXTogJ2JhcidcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAnaWdub3JlJyxcbiAgICAgICAgICBdO1xuXG4gICAgICAgICAgY29uc3QgZXhwZWN0ZWRWYWx1ZSA9IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgc29tZXRoaW5nOiB7XG4gICAgICAgICAgICAgICAgW01KU09OV1BfRUxFTUVOVF9LRVldOiAnZm9vbycsXG4gICAgICAgICAgICAgICAgW1czQ19FTEVNRU5UX0tFWV06ICdmb29vJyxcbiAgICAgICAgICAgICAgICBvdGhlcjogJ2JhcidcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSwge1xuICAgICAgICAgICAgICBbTUpTT05XUF9FTEVNRU5UX0tFWV06ICdiYXInLFxuICAgICAgICAgICAgICBbVzNDX0VMRU1FTlRfS0VZXTogJ2JhcidcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAnaWdub3JlJyxcbiAgICAgICAgICBdO1xuXG4gICAgICAgICAgY29uc3QgZmluZEVsZW1lbnRzQmFja3VwID0gZHJpdmVyLmZpbmRFbGVtZW50cztcbiAgICAgICAgICBkcml2ZXIuZmluZEVsZW1lbnRzID0gKCkgPT4gcmV0VmFsdWU7XG4gICAgICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3MucG9zdChgJHtzZXNzaW9uVXJsfS9lbGVtZW50c2AsIHtcbiAgICAgICAgICAgIHVzaW5nOiAnd2hhdGV2ZXInLFxuICAgICAgICAgICAgdmFsdWU6ICd3aGF0ZXZlcicsXG4gICAgICAgICAgfSk7XG4gICAgICAgICAgZGF0YS52YWx1ZS5zaG91bGQuZXFsKGV4cGVjdGVkVmFsdWUpO1xuICAgICAgICAgIGRyaXZlci5maW5kRWxlbWVudHMgPSBmaW5kRWxlbWVudHNCYWNrdXA7XG4gICAgICAgIH0pO1xuXG4gICAgICAgIGl0KGBzaG91bGQgZmFpbCB3aXRoIGEgNDA4IGVycm9yIGlmIGl0IHRocm93cyBhIFRpbWVvdXRFcnJvciBleGNlcHRpb25gLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgbGV0IHNldFVybFN0dWIgPSBzYW5kYm94LnN0dWIoZHJpdmVyLCAnc2V0VXJsJykuY2FsbHNGYWtlKGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBlcnJvcnMuVGltZW91dEVycm9yO1xuICAgICAgICAgIH0pO1xuICAgICAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgdXJsOiBgJHtzZXNzaW9uVXJsfS91cmxgLFxuICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICAgIGRhdGE6IHtcbiAgICAgICAgICAgICAgdXJsOiAnaHR0cHM6Ly9leGFtcGxlLmNvbS8nLFxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0pO1xuICAgICAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNDA4KTtcblxuICAgICAgICAgIGNvbnN0IHtlcnJvcjogdzNjRXJyb3IsIG1lc3NhZ2UsIHN0YWNrdHJhY2V9ID0gZGF0YS52YWx1ZTtcbiAgICAgICAgICBzdGFja3RyYWNlLnNob3VsZC5tYXRjaCgvcHJvdG9jb2wuanMvKTtcbiAgICAgICAgICB3M2NFcnJvci5zaG91bGQuYmUuYS5zdHJpbmc7XG4gICAgICAgICAgdzNjRXJyb3Iuc2hvdWxkLmVxdWFsKGVycm9ycy5UaW1lb3V0RXJyb3IuZXJyb3IoKSk7XG4gICAgICAgICAgbWVzc2FnZS5zaG91bGQubWF0Y2goL0FuIG9wZXJhdGlvbiBkaWQgbm90IGNvbXBsZXRlIGJlZm9yZSBpdHMgdGltZW91dCBleHBpcmVkLyk7XG5cbiAgICAgICAgICBzZXRVcmxTdHViLnJlc3RvcmUoKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgaXQoYHNob3VsZCBwYXNzIHdpdGggMjAwIEhUVFAgc3RhdHVzIGNvZGUgaWYgdGhlIGNvbW1hbmQgcmV0dXJucyBhIHZhbHVlYCwgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICAgIGRyaXZlci5wZXJmb3JtQWN0aW9ucyA9IChhY3Rpb25zKSA9PiAnSXQgd29ya3MgJyArIGFjdGlvbnMuam9pbignJyk7XG4gICAgICAgICAgY29uc3Qge3N0YXR1cywgdmFsdWUsIHNlc3Npb25JZH0gPSAoYXdhaXQgYXhpb3MucG9zdChgJHtzZXNzaW9uVXJsfS9hY3Rpb25zYCwge1xuICAgICAgICAgICAgYWN0aW9uczogWydhJywgJ2InLCAnYyddLFxuICAgICAgICAgIH0pKS5kYXRhO1xuICAgICAgICAgIHNob3VsZC5ub3QuZXhpc3Qoc2Vzc2lvbklkKTtcbiAgICAgICAgICBzaG91bGQubm90LmV4aXN0KHN0YXR1cyk7XG4gICAgICAgICAgdmFsdWUuc2hvdWxkLmVxdWFsKCdJdCB3b3JrcyBhYmMnKTtcbiAgICAgICAgICBkZWxldGUgZHJpdmVyLnBlcmZvcm1BY3Rpb25zO1xuICAgICAgICB9KTtcblxuICAgICAgICBkZXNjcmliZSgnandwcm94eScsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBsZXQgcG9ydDtcbiAgICAgICAgICBsZXQgc2VydmVyLCBqd3Byb3h5LCBhcHA7XG5cbiAgICAgICAgICBiZWZvcmUoYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICAgICAgcG9ydCA9IGF3YWl0IGdldFRlc3RQb3J0KHRydWUpO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgYmVmb3JlRWFjaChmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBjb25zdCByZXMgPSBjcmVhdGVQcm94eVNlcnZlcihzZXNzaW9uSWQsIHBvcnQpO1xuICAgICAgICAgICAgc2VydmVyID0gcmVzLnNlcnZlcjtcbiAgICAgICAgICAgIGFwcCA9IHJlcy5hcHA7XG4gICAgICAgICAgICBqd3Byb3h5ID0gbmV3IEpXUHJveHkoe2hvc3Q6IFRFU1RfSE9TVCwgcG9ydH0pO1xuICAgICAgICAgICAgandwcm94eS5zZXNzaW9uSWQgPSBzZXNzaW9uSWQ7XG4gICAgICAgICAgICBkcml2ZXIucGVyZm9ybUFjdGlvbnMgPSBhc3luYyAoYWN0aW9ucykgPT4gYXdhaXQgandwcm94eS5jb21tYW5kKCcvcGVyZm9ybS1hY3Rpb25zJywgJ1BPU1QnLCBhY3Rpb25zKTtcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIGFmdGVyRWFjaChhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBkZWxldGUgZHJpdmVyLnBlcmZvcm1BY3Rpb25zO1xuICAgICAgICAgICAgYXdhaXQgc2VydmVyLmNsb3NlKCk7XG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICBpdCgnc2hvdWxkIHdvcmsgaWYgYSBwcm94aWVkIHJlcXVlc3QgcmV0dXJucyBhIHJlc3BvbnNlIHdpdGggc3RhdHVzIDIwMCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICAgIGFwcC5wb3N0KCcvc2Vzc2lvbi86c2Vzc2lvbklkL3BlcmZvcm0tYWN0aW9ucycsIChyZXEsIHJlcykgPT4ge1xuICAgICAgICAgICAgICByZXMuanNvbih7XG4gICAgICAgICAgICAgICAgc2Vzc2lvbklkOiByZXEucGFyYW1zLnNlc3Npb25JZCxcbiAgICAgICAgICAgICAgICB2YWx1ZTogcmVxLmJvZHksXG4gICAgICAgICAgICAgICAgc3RhdHVzOiAwLFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgICBjb25zdCB7c3RhdHVzLCB2YWx1ZSwgc2Vzc2lvbklkfSA9IChhd2FpdCBheGlvcy5wb3N0KGAke3Nlc3Npb25Vcmx9L2FjdGlvbnNgLCB7XG4gICAgICAgICAgICAgIGFjdGlvbnM6IFsxLCAyLCAzXSxcbiAgICAgICAgICAgIH0pKS5kYXRhO1xuICAgICAgICAgICAgdmFsdWUuc2hvdWxkLmVxbChbMSwgMiwgM10pO1xuICAgICAgICAgICAgc2hvdWxkLm5vdC5leGlzdChzdGF0dXMpO1xuICAgICAgICAgICAgc2hvdWxkLm5vdC5leGlzdChzZXNzaW9uSWQpO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgaXQoJ3Nob3VsZCByZXR1cm4gZXJyb3IgaWYgYSBwcm94aWVkIHJlcXVlc3QgcmV0dXJucyBhIE1KU09OV1AgZXJyb3IgcmVzcG9uc2UnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBhcHAucG9zdCgnL3Nlc3Npb24vOnNlc3Npb25JZC9wZXJmb3JtLWFjdGlvbnMnLCAocmVxLCByZXMpID0+IHtcbiAgICAgICAgICAgICAgcmVzLnN0YXR1cyg1MDApLmpzb24oe1xuICAgICAgICAgICAgICAgIHNlc3Npb25JZCxcbiAgICAgICAgICAgICAgICBzdGF0dXM6IDYsXG4gICAgICAgICAgICAgICAgdmFsdWU6ICdBIHByb2JsZW0gb2NjdXJyZWQnLFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICAgIHVybDogYCR7c2Vzc2lvblVybH0vYWN0aW9uc2AsXG4gICAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgICAgIGFjdGlvbnM6IFsxLCAyLCAzXSxcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKEhUVFBTdGF0dXNDb2Rlcy5OT1RfRk9VTkQpO1xuICAgICAgICAgICAgSlNPTi5zdHJpbmdpZnkoZGF0YSkuc2hvdWxkLm1hdGNoKC9BIHByb2JsZW0gb2NjdXJyZWQvKTtcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIGl0KCdzaG91bGQgcmV0dXJuIFczQyBlcnJvciBpZiBhIHByb3hpZWQgcmVxdWVzdCByZXR1cm5zIGEgVzNDIGVycm9yIHJlc3BvbnNlJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICAgICAgY29uc3QgZXJyb3IgPSBuZXcgRXJyb3IoYFNvbWUgZXJyb3Igb2NjdXJyZWRgKTtcbiAgICAgICAgICAgIGVycm9yLnczY1N0YXR1cyA9IDQxNDtcbiAgICAgICAgICAgIGNvbnN0IGV4ZWN1dGVDb21tYW5kU3R1YiA9IHNhbmRib3guc3R1Yihkcml2ZXIsICdleGVjdXRlQ29tbWFuZCcpLnJldHVybnMoe1xuICAgICAgICAgICAgICBwcm90b2NvbDogJ1czQycsXG4gICAgICAgICAgICAgIGVycm9yLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICAgICAgdXJsOiBgJHtzZXNzaW9uVXJsfS9hY3Rpb25zYCxcbiAgICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICAgICAgICBkYXRhOiB7YWN0aW9uczogWzEsIDIsIDNdfSxcbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg0MTQpO1xuICAgICAgICAgICAgY29uc3Qge2Vycm9yOiB3M2NFcnJvciwgbWVzc2FnZTogZXJyTWVzc2FnZSwgc3RhY2t0cmFjZX0gPSBkYXRhLnZhbHVlO1xuICAgICAgICAgICAgdzNjRXJyb3Iuc2hvdWxkLmVxdWFsKCd1bmtub3duIGVycm9yJyk7XG4gICAgICAgICAgICBzdGFja3RyYWNlLnNob3VsZC5tYXRjaCgvU29tZSBlcnJvciBvY2N1cnJlZC8pO1xuICAgICAgICAgICAgZXJyTWVzc2FnZS5zaG91bGQuZXF1YWwoJ1NvbWUgZXJyb3Igb2NjdXJyZWQnKTtcbiAgICAgICAgICAgIGV4ZWN1dGVDb21tYW5kU3R1Yi5yZXN0b3JlKCk7XG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICBpdCgnc2hvdWxkIHJldHVybiBlcnJvciBpZiBhIHByb3hpZWQgcmVxdWVzdCByZXR1cm5zIGEgTUpTT05XUCBlcnJvciByZXNwb25zZSBidXQgSFRUUCBzdGF0dXMgY29kZSBpcyAyMDAnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBhcHAucG9zdCgnL3Nlc3Npb24vOnNlc3Npb25JZC9wZXJmb3JtLWFjdGlvbnMnLCAocmVxLCByZXMpID0+IHtcbiAgICAgICAgICAgICAgcmVzLnN0YXR1cygyMDApLmpzb24oe1xuICAgICAgICAgICAgICAgIHNlc3Npb25JZDogJ0Zha2UgU2Vzc2lvbiBJZCcsXG4gICAgICAgICAgICAgICAgc3RhdHVzOiA3LFxuICAgICAgICAgICAgICAgIHZhbHVlOiAnQSBwcm9ibGVtIG9jY3VycmVkJyxcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgICAgICB1cmw6IGAke3Nlc3Npb25Vcmx9L2FjdGlvbnNgLFxuICAgICAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICAgICAgICAgIGRhdGE6IHtcbiAgICAgICAgICAgICAgICBhY3Rpb25zOiBbMSwgMiwgM10sXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbChIVFRQU3RhdHVzQ29kZXMuTk9UX0ZPVU5EKTtcbiAgICAgICAgICAgIGNvbnN0IHtlcnJvcjogdzNjRXJyb3IsIG1lc3NhZ2U6IGVyck1lc3NhZ2UsIHN0YWNrdHJhY2V9ID0gZGF0YS52YWx1ZTtcbiAgICAgICAgICAgIHczY0Vycm9yLnNob3VsZC5lcXVhbCgnbm8gc3VjaCBlbGVtZW50Jyk7XG4gICAgICAgICAgICBlcnJNZXNzYWdlLnNob3VsZC5tYXRjaCgvQSBwcm9ibGVtIG9jY3VycmVkLyk7XG4gICAgICAgICAgICBzdGFja3RyYWNlLnNob3VsZC5leGlzdDtcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIGl0KCdzaG91bGQgcmV0dXJuIGVycm9yIGlmIGEgcHJveGllZCByZXF1ZXN0IHJldHVybnMgYSBXM0MgZXJyb3IgcmVzcG9uc2UnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICBhcHAucG9zdCgnL3Nlc3Npb24vOnNlc3Npb25JZC9wZXJmb3JtLWFjdGlvbnMnLCAocmVxLCByZXMpID0+IHtcbiAgICAgICAgICAgICAgcmVzLnN0YXR1cyg0MDQpLmpzb24oe1xuICAgICAgICAgICAgICAgIHZhbHVlOiB7XG4gICAgICAgICAgICAgICAgICBlcnJvcjogJ25vIHN1Y2ggZWxlbWVudCcsXG4gICAgICAgICAgICAgICAgICBtZXNzYWdlOiAnZG9lcyBub3QgbWFrZSBhIGRpZmZlcmVuY2UnLFxuICAgICAgICAgICAgICAgICAgc3RhY2t0cmFjZTogJ2FyYml0cmFyeSBzdGFja3RyYWNlJyxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgICAgICAgIHVybDogYCR7c2Vzc2lvblVybH0vYWN0aW9uc2AsXG4gICAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgICAgIGFjdGlvbnM6IFsxLCAyLCAzXSxcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKEhUVFBTdGF0dXNDb2Rlcy5OT1RfRk9VTkQpO1xuICAgICAgICAgICAgY29uc3Qge2Vycm9yOiB3M2NFcnJvciwgc3RhY2t0cmFjZX0gPSBkYXRhLnZhbHVlO1xuICAgICAgICAgICAgdzNjRXJyb3Iuc2hvdWxkLmVxdWFsKCdubyBzdWNoIGVsZW1lbnQnKTtcbiAgICAgICAgICAgIHN0YWNrdHJhY2Uuc2hvdWxkLm1hdGNoKC9hcmJpdHJhcnkgc3RhY2t0cmFjZS8pO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgaXQoJ3Nob3VsZCByZXR1cm4gYW4gZXJyb3IgaWYgYSBwcm94aWVkIHJlcXVlc3QgcmV0dXJucyBhIFczQyBlcnJvciByZXNwb25zZScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICAgIGFwcC5wb3N0KCcvc2Vzc2lvbi86c2Vzc2lvbklkL3BlcmZvcm0tYWN0aW9ucycsIChyZXEsIHJlcykgPT4ge1xuICAgICAgICAgICAgICByZXMuc2V0KCdDb25uZWN0aW9uJywgJ2Nsb3NlJyk7XG4gICAgICAgICAgICAgIHJlcy5zdGF0dXMoNDQ0KS5qc29uKHtcbiAgICAgICAgICAgICAgICB2YWx1ZToge1xuICAgICAgICAgICAgICAgICAgZXJyb3I6ICdib2d1cyBlcnJvciBjb2RlJyxcbiAgICAgICAgICAgICAgICAgIG1lc3NhZ2U6ICdkb2VzIG5vdCBtYWtlIGEgZGlmZmVyZW5jZScsXG4gICAgICAgICAgICAgICAgICBzdGFja3RyYWNlOiAnYXJiaXRyYXJ5IHN0YWNrdHJhY2UnLFxuICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBjb25zdCB7c3RhdHVzLCBkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgICAgICAgdXJsOiBgJHtzZXNzaW9uVXJsfS9hY3Rpb25zYCxcbiAgICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgICAgICAgYWN0aW9uczogWzEsIDIsIDNdLFxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoSFRUUFN0YXR1c0NvZGVzLklOVEVSTkFMX1NFUlZFUl9FUlJPUik7XG4gICAgICAgICAgICBjb25zdCB7ZXJyb3I6IHczY0Vycm9yLCBzdGFja3RyYWNlfSA9IGRhdGEudmFsdWU7XG4gICAgICAgICAgICB3M2NFcnJvci5zaG91bGQuZXF1YWwoJ3Vua25vd24gZXJyb3InKTtcbiAgICAgICAgICAgIHN0YWNrdHJhY2Uuc2hvdWxkLm1hdGNoKC9hcmJpdHJhcnkgc3RhY2t0cmFjZS8pO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgIH0pO1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGhhbmRsZSBjb21tYW5kcyB3aXRoIG5vIHJlc3BvbnNlIHZhbHVlcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9mb3J3YXJkYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICB9KTtcbiAgICAgIGRhdGEuc2hvdWxkLmVxbCh7XG4gICAgICAgIHZhbHVlOiBudWxsLFxuICAgICAgICBzZXNzaW9uSWQ6ICdmb28nXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgYWxsb3cgZW1wdHkgc3RyaW5nIHJlc3BvbnNlIHZhbHVlcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uL2Zvby9lbGVtZW50L2Jhci90ZXh0YCxcbiAgICAgIH0pO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICcnLFxuICAgICAgICBzZXNzaW9uSWQ6ICdmb28nXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgc2VuZCA1MDAgcmVzcG9uc2UgYW5kIGFuIFVua25vd24gb2JqZWN0IGZvciByZWplY3RlZCBjb21tYW5kcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vZm9vL3JlZnJlc2hgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg1MDApO1xuICAgICAgZGF0YS52YWx1ZS5lcnJvci5zaG91bGQuZXFsKCd1bmtub3duIGVycm9yJyk7XG4gICAgICBkYXRhLnZhbHVlLm1lc3NhZ2Uuc2hvdWxkLmVxbCgnQW4gdW5rbm93biBzZXJ2ZXItc2lkZSBlcnJvciBvY2N1cnJlZCB3aGlsZSBwcm9jZXNzaW5nICcgK1xuICAgICAgICAndGhlIGNvbW1hbmQuIE9yaWdpbmFsIGVycm9yOiBUb28gRnJlc2ghJyk7XG4gICAgICBkYXRhLnNlc3Npb25JZC5zaG91bGQuZXFsKCdmb28nKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgbm90IHRocm93IFVua25vd25FcnJvciB3aGVuIGtub3duJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi9mb29gLFxuICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDQwNCk7XG4gICAgICBkYXRhLnZhbHVlLmVycm9yLnNob3VsZC5lcWwoJ2ludmFsaWQgc2Vzc2lvbiBpZCcpO1xuICAgICAgZGF0YS52YWx1ZS5tZXNzYWdlLnNob3VsZC5lcWwoJ0Egc2Vzc2lvbiBpcyBlaXRoZXIgdGVybWluYXRlZCBvciBub3Qgc3RhcnRlZCcpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG4gIH0pO1xuXG4gIGRlc2NyaWJlKCdzZXNzaW9uIElkcycsIGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgZHJpdmVyID0gbmV3IEZha2VEcml2ZXIoKTtcbiAgICBsZXQgbWpzb253cFNlcnZlcjtcblxuICAgIGJlZm9yZShhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBtanNvbndwU2VydmVyID0gYXdhaXQgc2VydmVyKHtcbiAgICAgICAgcm91dGVDb25maWd1cmluZ0Z1bmN0aW9uOiByb3V0ZUNvbmZpZ3VyaW5nRnVuY3Rpb24oZHJpdmVyKSxcbiAgICAgICAgcG9ydCxcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgYWZ0ZXIoYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgYXdhaXQgbWpzb253cFNlcnZlci5jbG9zZSgpO1xuICAgIH0pO1xuXG4gICAgYWZ0ZXJFYWNoKGZ1bmN0aW9uICgpIHtcbiAgICAgIGRyaXZlci5zZXNzaW9uSWQgPSBudWxsO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCByZXR1cm4gbnVsbCBTZXNzaW9uSWQgZm9yIGNvbW1hbmRzIHdpdGhvdXQgc2Vzc2lvbklkcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHtkYXRhfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zdGF0dXNgLFxuICAgICAgfSk7XG5cbiAgICAgIHNob3VsZC5lcXVhbChkYXRhLnNlc3Npb25JZCwgbnVsbCk7XG4gICAgfSk7XG5cbiAgICBpdCgncmVzcG9uZHMgd2l0aCB0aGUgc2FtZSBzZXNzaW9uIElEIGluIHRoZSByZXF1ZXN0JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgbGV0IHNlc3Npb25JZCA9ICdWYWRlciBTZXNzaW9ucyc7XG4gICAgICBkcml2ZXIuc2Vzc2lvbklkID0gc2Vzc2lvbklkO1xuXG4gICAgICBjb25zdCB7ZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGRhdGE6IHt1cmw6ICdodHRwOi8vZ29vZ2xlLmNvbSd9XG4gICAgICB9KTtcblxuICAgICAgc2hvdWxkLmV4aXN0KGRhdGEuc2Vzc2lvbklkKTtcbiAgICAgIGRhdGEuc2Vzc2lvbklkLnNob3VsZC5lcWwoc2Vzc2lvbklkKTtcbiAgICB9KTtcblxuICAgIGl0KCd5ZWxscyBpZiBubyBzZXNzaW9uIGV4aXN0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGxldCBzZXNzaW9uSWQgPSAnVmFkZXIgU2Vzc2lvbnMnO1xuXG4gICAgICBjb25zdCB7ZGF0YSwgc3RhdHVzfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uLyR7c2Vzc2lvbklkfS91cmxgLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgdmFsaWRhdGVTdGF0dXM6IG51bGwsXG4gICAgICAgIGRhdGE6IHt1cmw6ICdodHRwOi8vZ29vZ2xlLmNvbSd9LFxuICAgICAgfSk7XG5cbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNDA0KTtcbiAgICAgIGRhdGEudmFsdWUubWVzc2FnZS5zaG91bGQuY29udGFpbignc2Vzc2lvbicpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3llbGxzIGlmIGludmFsaWQgc2Vzc2lvbiBpcyBzZW50JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgbGV0IHNlc3Npb25JZCA9ICdWYWRlciBTZXNzaW9ucyc7XG4gICAgICBkcml2ZXIuc2Vzc2lvbklkID0gJ3JlY2Vzc2lvbic7XG5cbiAgICAgIGNvbnN0IHtkYXRhLCBzdGF0dXN9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vJHtzZXNzaW9uSWR9L3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgZGF0YToge3VybDogJ2h0dHA6Ly9nb29nbGUuY29tJ30sXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg0MDQpO1xuICAgICAgZGF0YS52YWx1ZS5tZXNzYWdlLnNob3VsZC5jb250YWluKCdzZXNzaW9uJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGhhdmUgc2Vzc2lvbiBJRHMgaW4gZXJyb3IgcmVzcG9uc2VzJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgbGV0IHNlc3Npb25JZCA9ICdWYWRlciBTZXNzaW9ucyc7XG4gICAgICBkcml2ZXIuc2Vzc2lvbklkID0gc2Vzc2lvbklkO1xuXG4gICAgICBjb25zdCB7ZGF0YSwgc3RhdHVzfSA9IGF3YWl0IGF4aW9zKHtcbiAgICAgICAgdXJsOiBgJHtiYXNlVXJsfS9zZXNzaW9uLyR7c2Vzc2lvbklkfS9yZWZyZXNoYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgfSk7XG5cbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoNTAwKTtcbiAgICAgIGRhdGEudmFsdWUuZXJyb3Iuc2hvdWxkLmVxbCgndW5rbm93biBlcnJvcicpO1xuICAgICAgZGF0YS52YWx1ZS5tZXNzYWdlLnNob3VsZC5lcWwoJ0FuIHVua25vd24gc2VydmVyLXNpZGUgZXJyb3Igb2NjdXJyZWQgd2hpbGUgcHJvY2Vzc2luZyAnICtcbiAgICAgICAgJ3RoZSBjb21tYW5kLiBPcmlnaW5hbCBlcnJvcjogVG9vIEZyZXNoIScpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnVmFkZXIgU2Vzc2lvbnMnKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgcmV0dXJuIGEgbmV3IHNlc3Npb24gSUQgb24gY3JlYXRlJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge2RhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb25gLFxuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgZGF0YToge2NhcGFiaWxpdGllczoge2Fsd2F5c01hdGNoOiB7J2FwcGl1bTpncmVldGluZyc6ICdoZWxsbyd9LCBmaXJzdE1hdGNoOiBbe31dfX1cbiAgICAgIH0pO1xuXG4gICAgICBzaG91bGQuZXhpc3QoZGF0YS52YWx1ZS5zZXNzaW9uSWQpO1xuICAgICAgZGF0YS52YWx1ZS5zZXNzaW9uSWQuaW5kZXhPZignZmFrZVNlc3Npb25fJykuc2hvdWxkLmVxdWFsKDApO1xuICAgICAgZGF0YS52YWx1ZS5jYXBhYmlsaXRpZXMuc2hvdWxkLmVxbCh7YWx3YXlzTWF0Y2g6IHsnYXBwaXVtOmdyZWV0aW5nJzogJ2hlbGxvJ30sIGZpcnN0TWF0Y2g6IFt7fV19KTtcbiAgICB9KTtcbiAgfSk7XG5cbiAgZGVzY3JpYmUoJ3ZpYSBkcml2ZXJzIGpzb253cCBwcm94eScsIGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgZHJpdmVyO1xuICAgIGxldCBzZXNzaW9uSWQgPSAnZm9vJztcbiAgICBsZXQgbWpzb253cFNlcnZlcjtcblxuICAgIGJlZm9yZUVhY2goYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgZHJpdmVyID0gbmV3IEZha2VEcml2ZXIoKTtcbiAgICAgIGRyaXZlci5zZXNzaW9uSWQgPSBzZXNzaW9uSWQ7XG4gICAgICBkcml2ZXIucHJveHlBY3RpdmUgPSAoKSA9PiB0cnVlO1xuICAgICAgZHJpdmVyLmNhblByb3h5ID0gKCkgPT4gdHJ1ZTtcblxuICAgICAgbWpzb253cFNlcnZlciA9IGF3YWl0IHNlcnZlcih7XG4gICAgICAgIHJvdXRlQ29uZmlndXJpbmdGdW5jdGlvbjogcm91dGVDb25maWd1cmluZ0Z1bmN0aW9uKGRyaXZlciksXG4gICAgICAgIHBvcnQsXG4gICAgICAgIGV4dHJhTWV0aG9kTWFwOiBGYWtlRHJpdmVyLm5ld01ldGhvZE1hcFxuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBhZnRlckVhY2goYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgYXdhaXQgbWpzb253cFNlcnZlci5jbG9zZSgpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBnaXZlIGEgbmljZSBlcnJvciBpZiBwcm94eWluZyBpcyBzZXQgYnV0IG5vIHByb3h5IGZ1bmN0aW9uIGV4aXN0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGRyaXZlci5jYW5Qcm94eSA9ICgpID0+IGZhbHNlO1xuICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDUwMCk7XG4gICAgICBkYXRhLnZhbHVlLmVycm9yLnNob3VsZC5lcWwoJ3Vua25vd24gZXJyb3InKTtcbiAgICAgIGRhdGEudmFsdWUubWVzc2FnZS5zaG91bGQuZXFsKCdBbiB1bmtub3duIHNlcnZlci1zaWRlIGVycm9yIG9jY3VycmVkIHdoaWxlIHByb2Nlc3NpbmcgJyArXG4gICAgICAgICd0aGUgY29tbWFuZC4gT3JpZ2luYWwgZXJyb3I6IFRyeWluZyB0byBwcm94eSB0byBhICcgK1xuICAgICAgICAnc2VydmVyIGJ1dCB0aGUgZHJpdmVyIGlzIHVuYWJsZSB0byBwcm94eScpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHBhc3Mgb24gYW55IGVycm9ycyBpbiBwcm94eWluZycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGRyaXZlci5wcm94eVJlcVJlcyA9IGFzeW5jIGZ1bmN0aW9uICgpIHsgLy8gZXNsaW50LWRpc2FibGUtbGluZSByZXF1aXJlLWF3YWl0XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignZm9vJyk7XG4gICAgICB9O1xuICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vdXJsYCxcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHZhbGlkYXRlU3RhdHVzOiBudWxsLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDUwMCk7XG4gICAgICBkYXRhLnZhbHVlLmVycm9yLnNob3VsZC5lcWwoJ3Vua25vd24gZXJyb3InKTtcbiAgICAgIGRhdGEudmFsdWUubWVzc2FnZS5zaG91bGQuZXFsKCdBbiB1bmtub3duIHNlcnZlci1zaWRlIGVycm9yIG9jY3VycmVkIHdoaWxlIHByb2Nlc3NpbmcgJyArXG4gICAgICAgICd0aGUgY29tbWFuZC4gT3JpZ2luYWwgZXJyb3I6IENvdWxkIG5vdCBwcm94eS4gUHJveHkgZXJyb3I6IGZvbycpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGFibGUgdG8gdGhyb3cgUHJveHlSZXF1ZXN0RXJyb3IgaW4gcHJveHlpbmcnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBkcml2ZXIucHJveHlSZXFSZXMgPSBhc3luYyBmdW5jdGlvbiAoKSB7IC8vIGVzbGludC1kaXNhYmxlLWxpbmUgcmVxdWlyZS1hd2FpdFxuICAgICAgICBsZXQganNvbndwID0ge3N0YXR1czogMzUsIHZhbHVlOiAnTm8gc3VjaCBjb250ZXh0IGZvdW5kLicsIHNlc3Npb25JZDogJ2Zvbyd9O1xuICAgICAgICB0aHJvdyBuZXcgZXJyb3JzLlByb3h5UmVxdWVzdEVycm9yKGBDb3VsZCBub3QgcHJveHkgY29tbWFuZCB0byByZW1vdGUgc2VydmVyLiBgLCBqc29ud3ApO1xuICAgICAgfTtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vJHtzZXNzaW9uSWR9L3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgZGF0YToge3VybDogJ2h0dHA6Ly9nb29nbGUuY29tJ30sXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg0MDApO1xuICAgICAgZGF0YS52YWx1ZS5lcnJvci5zaG91bGQuZXFsKCd1bmtub3duIGVycm9yJyk7XG4gICAgICBkYXRhLnZhbHVlLm1lc3NhZ2Uuc2hvdWxkLmVxbCgnTm8gc3VjaCBjb250ZXh0IGZvdW5kLicpO1xuICAgICAgZGF0YS5zZXNzaW9uSWQuc2hvdWxkLmVxbCgnZm9vJyk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGxldCB0aGUgcHJveHkgaGFuZGxlIHJlcS9yZXMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBkcml2ZXIucHJveHlSZXFSZXMgPSBhc3luYyBmdW5jdGlvbiAocmVxLCByZXMpIHsgLy8gZXNsaW50LWRpc2FibGUtbGluZSByZXF1aXJlLWF3YWl0XG4gICAgICAgIHJlcy5zdGF0dXMoMjAwKS5qc29uKHtjdXN0b206ICdkYXRhJ30pO1xuICAgICAgfTtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vJHtzZXNzaW9uSWR9L3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfVxuICAgICAgfSk7XG5cbiAgICAgIHN0YXR1cy5zaG91bGQuZXF1YWwoMjAwKTtcbiAgICAgIGRhdGEuc2hvdWxkLmVxbCh7Y3VzdG9tOiAnZGF0YSd9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgYXZvaWQganNvbndwIHByb3h5aW5nIHdoZW4gcGF0aCBtYXRjaGVzIGF2b2lkYW5jZSBsaXN0JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgZHJpdmVyLmdldFByb3h5QXZvaWRMaXN0ID0gKCkgPT4gW1snUE9TVCcsIG5ldyBSZWdFeHAoJ14vc2Vzc2lvbi9bXi9dKy91cmwkJyldXTtcbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3Nlc3Npb24vJHtzZXNzaW9uSWR9L3VybGAsXG4gICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgIH0pO1xuXG4gICAgICBzdGF0dXMuc2hvdWxkLmVxdWFsKDIwMCk7XG4gICAgICBkYXRhLnNob3VsZC5lcWwoe1xuICAgICAgICB2YWx1ZTogJ05hdmlnYXRlZCB0bzogaHR0cDovL2dvb2dsZS5jb20nLFxuICAgICAgICBzZXNzaW9uSWRcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBmYWlsIGlmIGF2b2lkIHByb3h5IGxpc3QgaXMgbWFsZm9ybWVkIGluIHNvbWUgd2F5JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgYXN5bmMgZnVuY3Rpb24gYmFkUHJveHlBdm9pZGFuY2VMaXN0IChsaXN0KSB7XG4gICAgICAgIGRyaXZlci5nZXRQcm94eUF2b2lkTGlzdCA9ICgpID0+IGxpc3Q7XG4gICAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vdXJsYCxcbiAgICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICB2YWxpZGF0ZVN0YXR1czogbnVsbCxcbiAgICAgICAgICBkYXRhOiB7dXJsOiAnaHR0cDovL2dvb2dsZS5jb20nfSxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCg1MDApO1xuICAgICAgICBkYXRhLnZhbHVlLm1lc3NhZ2Uuc2hvdWxkLmNvbnRhaW4oJ3JveHknKTtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGxpc3RzID0gW1xuICAgICAgICAnZm9vJyxcbiAgICAgICAgW1snZm9vJ11dLFxuICAgICAgICBbWydCQVInLCAvbG9sL11dLFxuICAgICAgICBbWydHRVQnLCAnZm9vJ11dXG4gICAgICBdO1xuICAgICAgZm9yIChsZXQgbGlzdCBvZiBsaXN0cykge1xuICAgICAgICBhd2FpdCBiYWRQcm94eUF2b2lkYW5jZUxpc3QobGlzdCk7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGF2b2lkIHByb3h5aW5nIG5vbi1zZXNzaW9uIGNvbW1hbmRzIGV2ZW4gaWYgbm90IGluIHRoZSBsaXN0JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgZHJpdmVyLmdldFByb3h5QXZvaWRMaXN0ID0gKCkgPT4gW1snUE9TVCcsIG5ldyBSZWdFeHAoJycpXV07XG5cbiAgICAgIGNvbnN0IHtzdGF0dXMsIGRhdGF9ID0gYXdhaXQgYXhpb3Moe1xuICAgICAgICB1cmw6IGAke2Jhc2VVcmx9L3N0YXR1c2AsXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCgyMDApO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6IFwiSSdtIGZpbmVcIixcbiAgICAgICAgc2Vzc2lvbklkOiBudWxsXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgYXZvaWQgcHJveHlpbmcgZGVsZXRlU2Vzc2lvbiBjb21tYW5kcycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGRyaXZlci5nZXRQcm94eUF2b2lkTGlzdCA9ICgpID0+IFtbJ1BPU1QnLCBuZXcgUmVnRXhwKCcnKV1dO1xuXG4gICAgICBkcml2ZXIuc2Vzc2lvbklkLnNob3VsZC5lcXVhbChzZXNzaW9uSWQpO1xuICAgICAgY29uc3Qge3N0YXR1c30gPSBhd2FpdCBheGlvcy5kZWxldGUoYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH1gKTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCgyMDApO1xuICAgICAgc2hvdWxkLm5vdC5leGlzdChkcml2ZXIuc2Vzc2lvbklkKTtcbiAgICAgIGRyaXZlci5qd3BQcm94eUFjdGl2ZS5zaG91bGQuYmUuZmFsc2U7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGF2b2lkIHByb3h5aW5nIHdoZW4gY29tbWFuZCBzcGVjIHNwZWNpZmllcyBuZXZlclByb3h5JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3Qge3N0YXR1cywgZGF0YX0gPSBhd2FpdCBheGlvcyh7XG4gICAgICAgIHVybDogYCR7YmFzZVVybH0vc2Vzc2lvbi8ke3Nlc3Npb25JZH0vbm9wcm94eWAsXG4gICAgICAgIG1ldGhvZDogJ0dFVCcsXG4gICAgICB9KTtcblxuICAgICAgc3RhdHVzLnNob3VsZC5lcXVhbCgyMDApO1xuICAgICAgZGF0YS5zaG91bGQuZXFsKHtcbiAgICAgICAgdmFsdWU6ICdUaGlzIHdhcyBub3QgcHJveGllZCcsXG4gICAgICAgIHNlc3Npb25JZFxuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgfSk7XG59KTtcbiJdfQ==