@bitgo-beta/sdk-api 1.10.1-beta.167 → 1.10.1-beta.1670
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.
- package/dist/package.json +16 -25
- package/dist/src/api.d.ts +8 -5
- package/dist/src/api.d.ts.map +1 -1
- package/dist/src/api.js +128 -69
- package/dist/src/bitgoAPI.d.ts +68 -13
- package/dist/src/bitgoAPI.d.ts.map +1 -1
- package/dist/src/bitgoAPI.js +468 -180
- package/dist/src/encrypt.d.ts +4 -4
- package/dist/src/encrypt.d.ts.map +1 -1
- package/dist/src/encrypt.js +44 -7
- package/dist/src/index.js +6 -2
- package/dist/src/types.d.ts +58 -43
- package/dist/src/types.d.ts.map +1 -1
- package/dist/src/types.js +4 -2
- package/dist/src/util.d.ts +12 -0
- package/dist/src/util.d.ts.map +1 -1
- package/dist/src/util.js +57 -4
- package/dist/src/v1/blockchain.js +25 -12
- package/dist/src/v1/keychains.js +27 -22
- package/dist/src/v1/markets.js +10 -5
- package/dist/src/v1/pendingapproval.d.ts.map +1 -1
- package/dist/src/v1/pendingapproval.js +57 -18
- package/dist/src/v1/pendingapprovals.d.ts.map +1 -1
- package/dist/src/v1/pendingapprovals.js +41 -7
- package/dist/src/v1/signPsbt.d.ts +14 -0
- package/dist/src/v1/signPsbt.d.ts.map +1 -0
- package/dist/src/v1/signPsbt.js +69 -0
- package/dist/src/v1/transactionBuilder.js +214 -90
- package/dist/src/v1/travelRule.js +60 -23
- package/dist/src/v1/verifyAddress.d.ts +6 -0
- package/dist/src/v1/verifyAddress.d.ts.map +1 -0
- package/dist/src/v1/verifyAddress.js +50 -0
- package/dist/src/v1/wallet.d.ts.map +1 -1
- package/dist/src/v1/wallet.js +479 -224
- package/dist/src/v1/wallets.js +93 -37
- package/dist/test/unit/api.d.ts +2 -0
- package/dist/test/unit/api.d.ts.map +1 -0
- package/dist/test/unit/api.js +307 -0
- package/dist/test/unit/bitgoAPI.d.ts +2 -0
- package/dist/test/unit/bitgoAPI.d.ts.map +1 -0
- package/dist/test/unit/bitgoAPI.js +717 -0
- package/dist/test/unit/encrypt.d.ts +2 -0
- package/dist/test/unit/encrypt.d.ts.map +1 -0
- package/dist/test/unit/encrypt.js +62 -0
- package/dist/test/unit/hmacStrategy.d.ts +2 -0
- package/dist/test/unit/hmacStrategy.d.ts.map +1 -0
- package/dist/test/unit/hmacStrategy.js +136 -0
- package/dist/test/unit/utils.d.ts +2 -0
- package/dist/test/unit/utils.d.ts.map +1 -0
- package/dist/test/unit/utils.js +65 -0
- package/dist/test/unit/v1/fixtures/accelerate-tx.d.ts +4 -0
- package/dist/test/unit/v1/fixtures/accelerate-tx.d.ts.map +1 -0
- package/dist/test/unit/v1/fixtures/accelerate-tx.js +53 -0
- package/dist/test/unit/v1/fixtures/index.d.ts +3 -0
- package/dist/test/unit/v1/fixtures/index.d.ts.map +1 -0
- package/dist/test/unit/v1/fixtures/index.js +24 -0
- package/dist/test/unit/v1/signPsbt.d.ts +2 -0
- package/dist/test/unit/v1/signPsbt.d.ts.map +1 -0
- package/dist/test/unit/v1/signPsbt.js +53 -0
- package/dist/test/unit/v1/wallet.d.ts +2 -0
- package/dist/test/unit/v1/wallet.d.ts.map +1 -0
- package/dist/test/unit/v1/wallet.js +1603 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/package.json +16 -25
- package/.eslintignore +0 -5
- package/CHANGELOG.md +0 -478
- package/dist/web/main.js +0 -2
- package/dist/web/main.js.LICENSE.txt +0 -110
|
@@ -0,0 +1,717 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
require("should");
|
|
40
|
+
const bitgoAPI_1 = require("../../src/bitgoAPI");
|
|
41
|
+
const proxy_agent_1 = require("proxy-agent");
|
|
42
|
+
const sinon = __importStar(require("sinon"));
|
|
43
|
+
const nock_1 = __importDefault(require("nock"));
|
|
44
|
+
describe('Constructor', function () {
|
|
45
|
+
describe('cookiesPropagationEnabled argument', function () {
|
|
46
|
+
it('cookiesPropagationEnabled is enabled explicitly', function () {
|
|
47
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
48
|
+
env: 'custom',
|
|
49
|
+
customRootURI: 'https://app.example.local',
|
|
50
|
+
cookiesPropagationEnabled: true,
|
|
51
|
+
});
|
|
52
|
+
bitgo.should.have.property('cookiesPropagationEnabled');
|
|
53
|
+
bitgo.cookiesPropagationEnabled.should.equal(true);
|
|
54
|
+
});
|
|
55
|
+
it('cookiesPropagationEnabled is disabled explicitly', function () {
|
|
56
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
57
|
+
env: 'custom',
|
|
58
|
+
customRootURI: 'https://app.example.local',
|
|
59
|
+
cookiesPropagationEnabled: false,
|
|
60
|
+
});
|
|
61
|
+
bitgo.should.have.property('cookiesPropagationEnabled');
|
|
62
|
+
bitgo.cookiesPropagationEnabled.should.equal(false);
|
|
63
|
+
});
|
|
64
|
+
it('cookiesPropagationEnabled is disabled by default', function () {
|
|
65
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
66
|
+
env: 'custom',
|
|
67
|
+
customRootURI: 'https://app.example.local',
|
|
68
|
+
});
|
|
69
|
+
bitgo.should.have.property('cookiesPropagationEnabled');
|
|
70
|
+
bitgo.cookiesPropagationEnabled.should.equal(false);
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
describe('requestIdPrefix argument', function () {
|
|
74
|
+
afterEach(function () {
|
|
75
|
+
nock_1.default.cleanAll();
|
|
76
|
+
});
|
|
77
|
+
it('should prepend requestIdPrefix to Request-ID header when set', async function () {
|
|
78
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
79
|
+
env: 'custom',
|
|
80
|
+
customRootURI: 'https://app.example.local',
|
|
81
|
+
requestIdPrefix: 'test-prefix-',
|
|
82
|
+
});
|
|
83
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
84
|
+
.get('/api/v1/ping')
|
|
85
|
+
.matchHeader('Request-ID', /^test-prefix-/)
|
|
86
|
+
.reply(200, { status: 'ok' });
|
|
87
|
+
await bitgo.ping({
|
|
88
|
+
reqId: {
|
|
89
|
+
toString: () => '12345',
|
|
90
|
+
inc: () => {
|
|
91
|
+
/* mock */
|
|
92
|
+
},
|
|
93
|
+
},
|
|
94
|
+
});
|
|
95
|
+
scope.isDone().should.be.true();
|
|
96
|
+
});
|
|
97
|
+
it('should not modify Request-ID header when requestIdPrefix is not set', async function () {
|
|
98
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
99
|
+
env: 'custom',
|
|
100
|
+
customRootURI: 'https://app.example.local',
|
|
101
|
+
});
|
|
102
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
103
|
+
.get('/api/v1/ping')
|
|
104
|
+
.matchHeader('Request-ID', /^12345$/)
|
|
105
|
+
.reply(200, { status: 'ok' });
|
|
106
|
+
await bitgo.ping({
|
|
107
|
+
reqId: {
|
|
108
|
+
toString: () => '12345',
|
|
109
|
+
inc: () => {
|
|
110
|
+
/* mock */
|
|
111
|
+
},
|
|
112
|
+
},
|
|
113
|
+
});
|
|
114
|
+
scope.isDone().should.be.true();
|
|
115
|
+
});
|
|
116
|
+
it('should correctly format Request-ID with prefix and numeric sequence', async function () {
|
|
117
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
118
|
+
env: 'custom',
|
|
119
|
+
customRootURI: 'https://app.example.local',
|
|
120
|
+
requestIdPrefix: 'myapp-v1-',
|
|
121
|
+
});
|
|
122
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
123
|
+
.get('/api/v1/ping')
|
|
124
|
+
.matchHeader('Request-ID', 'myapp-v1-trace-123')
|
|
125
|
+
.reply(200, { status: 'ok' });
|
|
126
|
+
await bitgo.ping({
|
|
127
|
+
reqId: {
|
|
128
|
+
toString: () => 'trace-123',
|
|
129
|
+
inc: () => {
|
|
130
|
+
/* mock */
|
|
131
|
+
},
|
|
132
|
+
},
|
|
133
|
+
});
|
|
134
|
+
scope.isDone().should.be.true();
|
|
135
|
+
});
|
|
136
|
+
it('should work with empty string prefix', async function () {
|
|
137
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
138
|
+
env: 'custom',
|
|
139
|
+
customRootURI: 'https://app.example.local',
|
|
140
|
+
requestIdPrefix: '',
|
|
141
|
+
});
|
|
142
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
143
|
+
.get('/api/v1/ping')
|
|
144
|
+
.matchHeader('Request-ID', 'abc-123')
|
|
145
|
+
.reply(200, { status: 'ok' });
|
|
146
|
+
await bitgo.ping({
|
|
147
|
+
reqId: {
|
|
148
|
+
toString: () => 'abc-123',
|
|
149
|
+
inc: () => {
|
|
150
|
+
/* mock */
|
|
151
|
+
},
|
|
152
|
+
},
|
|
153
|
+
});
|
|
154
|
+
scope.isDone().should.be.true();
|
|
155
|
+
});
|
|
156
|
+
});
|
|
157
|
+
describe('http proxy agent', function () {
|
|
158
|
+
it('http proxy agent shall be created when proxy(customProxyagent) is set', function () {
|
|
159
|
+
const customProxyAgent = new proxy_agent_1.ProxyAgent({
|
|
160
|
+
getProxyForUrl: () => 'http://localhost:3000',
|
|
161
|
+
});
|
|
162
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
163
|
+
env: 'custom',
|
|
164
|
+
customRootURI: 'https://app.example.local',
|
|
165
|
+
customProxyAgent,
|
|
166
|
+
});
|
|
167
|
+
bitgo.should.have.property('_customProxyAgent', customProxyAgent);
|
|
168
|
+
});
|
|
169
|
+
it('bitgo api is still initiated when proxy(customProxyAgent) is not set', function () {
|
|
170
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
171
|
+
env: 'custom',
|
|
172
|
+
customRootURI: 'https://app.example.local',
|
|
173
|
+
});
|
|
174
|
+
bitgo.should.have.property('_customProxyAgent', undefined);
|
|
175
|
+
});
|
|
176
|
+
});
|
|
177
|
+
describe('verifyAddress', function () {
|
|
178
|
+
it('should successfully verify a base58 address', function () {
|
|
179
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
180
|
+
env: 'test',
|
|
181
|
+
});
|
|
182
|
+
bitgo.verifyAddress({ address: '2N6paT2TU4N1XpaZjJiApWJXoeyrL3UWpkZ' }).should.be.true();
|
|
183
|
+
});
|
|
184
|
+
it('should successfully verify a bech32 address', function () {
|
|
185
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
186
|
+
env: 'test',
|
|
187
|
+
});
|
|
188
|
+
bitgo
|
|
189
|
+
.verifyAddress({ address: 'tb1qguzyk4w6kaqtpsczs5aj0w8r7598jq36egm8e98wqph3rwmex68seslgsg' })
|
|
190
|
+
.should.be.true();
|
|
191
|
+
});
|
|
192
|
+
});
|
|
193
|
+
describe('url', function () {
|
|
194
|
+
it('should return the correct URL for version 1', function () {
|
|
195
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
196
|
+
env: 'test',
|
|
197
|
+
customRootURI: 'https://test.bitgo.com',
|
|
198
|
+
});
|
|
199
|
+
const path = '/test-path';
|
|
200
|
+
const expectedUrl = 'https://test.bitgo.com/api/v1/test-path';
|
|
201
|
+
const result = bitgo.url(path, 1);
|
|
202
|
+
result.should.equal(expectedUrl);
|
|
203
|
+
});
|
|
204
|
+
it('should return the correct URL for version 2', function () {
|
|
205
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
206
|
+
env: 'test',
|
|
207
|
+
customRootURI: 'https://test.bitgo.com',
|
|
208
|
+
});
|
|
209
|
+
const path = '/test-path';
|
|
210
|
+
const expectedUrl = 'https://test.bitgo.com/api/v2/test-path';
|
|
211
|
+
const result = bitgo.url(path, 2);
|
|
212
|
+
result.should.equal(expectedUrl);
|
|
213
|
+
});
|
|
214
|
+
it('should return the correct URL for version 3', function () {
|
|
215
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
216
|
+
env: 'test',
|
|
217
|
+
customRootURI: 'https://test.bitgo.com',
|
|
218
|
+
});
|
|
219
|
+
const path = '/test-path';
|
|
220
|
+
const expectedUrl = 'https://test.bitgo.com/api/v3/test-path';
|
|
221
|
+
const result = bitgo.url(path, 3);
|
|
222
|
+
result.should.equal(expectedUrl);
|
|
223
|
+
});
|
|
224
|
+
it('should default to version 1 if no version is provided', function () {
|
|
225
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
226
|
+
env: 'test',
|
|
227
|
+
customRootURI: 'https://test.bitgo.com',
|
|
228
|
+
});
|
|
229
|
+
const path = '/test-path';
|
|
230
|
+
const expectedUrl = 'https://test.bitgo.com/api/v1/test-path';
|
|
231
|
+
const result = bitgo.url(path);
|
|
232
|
+
result.should.equal(expectedUrl);
|
|
233
|
+
});
|
|
234
|
+
});
|
|
235
|
+
describe('decryptKeys', function () {
|
|
236
|
+
let bitgo;
|
|
237
|
+
beforeEach(function () {
|
|
238
|
+
bitgo = new bitgoAPI_1.BitGoAPI({
|
|
239
|
+
env: 'test',
|
|
240
|
+
});
|
|
241
|
+
});
|
|
242
|
+
afterEach(function () {
|
|
243
|
+
sinon.restore();
|
|
244
|
+
});
|
|
245
|
+
it('should throw if no params are provided', function () {
|
|
246
|
+
try {
|
|
247
|
+
// @ts-expect-error - intentionally calling with no params for test
|
|
248
|
+
bitgo.decryptKeys();
|
|
249
|
+
throw new Error('Expected error but got none');
|
|
250
|
+
}
|
|
251
|
+
catch (e) {
|
|
252
|
+
e.message.should.containEql('Missing parameter');
|
|
253
|
+
}
|
|
254
|
+
});
|
|
255
|
+
it('should throw if walletIdEncryptedKeyPairs is missing', function () {
|
|
256
|
+
try {
|
|
257
|
+
// @ts-expect-error - intentionally missing required param
|
|
258
|
+
bitgo.decryptKeys({ password: 'password123' });
|
|
259
|
+
throw new Error('Expected error but got none');
|
|
260
|
+
}
|
|
261
|
+
catch (e) {
|
|
262
|
+
e.message.should.containEql('Missing parameter: walletIdEncryptedKeyPairs');
|
|
263
|
+
}
|
|
264
|
+
});
|
|
265
|
+
it('should throw if password is missing', function () {
|
|
266
|
+
try {
|
|
267
|
+
// @ts-expect-error - intentionally missing required param
|
|
268
|
+
bitgo.decryptKeys({ walletIdEncryptedKeyPairs: [] });
|
|
269
|
+
throw new Error('Expected error but got none');
|
|
270
|
+
}
|
|
271
|
+
catch (e) {
|
|
272
|
+
e.message.should.containEql('Missing parameter: password');
|
|
273
|
+
}
|
|
274
|
+
});
|
|
275
|
+
it('should throw if walletIdEncryptedKeyPairs is not an array', function () {
|
|
276
|
+
try {
|
|
277
|
+
// @ts-expect-error - intentionally providing wrong type
|
|
278
|
+
bitgo.decryptKeys({ walletIdEncryptedKeyPairs: 'not an array', password: 'password123' });
|
|
279
|
+
throw new Error('Expected error but got none');
|
|
280
|
+
}
|
|
281
|
+
catch (e) {
|
|
282
|
+
e.message.should.equal('walletIdEncryptedKeyPairs must be an array');
|
|
283
|
+
}
|
|
284
|
+
});
|
|
285
|
+
it('should return empty array for empty walletIdEncryptedKeyPairs', function () {
|
|
286
|
+
const result = bitgo.decryptKeys({ walletIdEncryptedKeyPairs: [], password: 'password123' });
|
|
287
|
+
result.should.be.an.Array();
|
|
288
|
+
result.should.be.empty();
|
|
289
|
+
});
|
|
290
|
+
it('should throw if any walletId is missing or not a string', function () {
|
|
291
|
+
try {
|
|
292
|
+
bitgo.decryptKeys({
|
|
293
|
+
walletIdEncryptedKeyPairs: [
|
|
294
|
+
// @ts-expect-error - intentionally missing walletId
|
|
295
|
+
{
|
|
296
|
+
encryptedPrv: 'encrypted-data',
|
|
297
|
+
},
|
|
298
|
+
],
|
|
299
|
+
password: 'password123',
|
|
300
|
+
});
|
|
301
|
+
throw new Error('Expected error but got none');
|
|
302
|
+
}
|
|
303
|
+
catch (e) {
|
|
304
|
+
e.message.should.equal('each key pair must have a string walletId');
|
|
305
|
+
}
|
|
306
|
+
try {
|
|
307
|
+
bitgo.decryptKeys({
|
|
308
|
+
walletIdEncryptedKeyPairs: [
|
|
309
|
+
{
|
|
310
|
+
// @ts-expect-error - intentionally providing wrong type
|
|
311
|
+
walletId: 123,
|
|
312
|
+
encryptedPrv: 'encrypted-data',
|
|
313
|
+
},
|
|
314
|
+
],
|
|
315
|
+
password: 'password123',
|
|
316
|
+
});
|
|
317
|
+
throw new Error('Expected error but got none');
|
|
318
|
+
}
|
|
319
|
+
catch (e) {
|
|
320
|
+
e.message.should.equal('each key pair must have a string walletId');
|
|
321
|
+
}
|
|
322
|
+
});
|
|
323
|
+
it('should throw if any encryptedPrv is missing or not a string', function () {
|
|
324
|
+
try {
|
|
325
|
+
bitgo.decryptKeys({
|
|
326
|
+
walletIdEncryptedKeyPairs: [
|
|
327
|
+
// @ts-expect-error - intentionally missing encryptedPrv
|
|
328
|
+
{
|
|
329
|
+
walletId: 'wallet-id-1',
|
|
330
|
+
},
|
|
331
|
+
],
|
|
332
|
+
password: 'password123',
|
|
333
|
+
});
|
|
334
|
+
throw new Error('Expected error but got none');
|
|
335
|
+
}
|
|
336
|
+
catch (e) {
|
|
337
|
+
e.message.should.equal('each key pair must have a string encryptedPrv');
|
|
338
|
+
}
|
|
339
|
+
try {
|
|
340
|
+
bitgo.decryptKeys({
|
|
341
|
+
walletIdEncryptedKeyPairs: [
|
|
342
|
+
{
|
|
343
|
+
walletId: 'wallet-id-1',
|
|
344
|
+
// @ts-expect-error - intentionally providing wrong type
|
|
345
|
+
encryptedPrv: 123,
|
|
346
|
+
},
|
|
347
|
+
],
|
|
348
|
+
password: 'password123',
|
|
349
|
+
});
|
|
350
|
+
throw new Error('Expected error but got none');
|
|
351
|
+
}
|
|
352
|
+
catch (e) {
|
|
353
|
+
e.message.should.equal('each key pair must have a string encryptedPrv');
|
|
354
|
+
}
|
|
355
|
+
});
|
|
356
|
+
it('should return walletIds of keys that failed to decrypt', function () {
|
|
357
|
+
// Create a stub for the decrypt method
|
|
358
|
+
const decryptStub = sinon.stub(bitgo, 'decrypt');
|
|
359
|
+
// Make it succeed for first wallet and fail for second wallet
|
|
360
|
+
decryptStub.onFirstCall().returns('decrypted-key-1');
|
|
361
|
+
decryptStub.onSecondCall().throws(new Error('decryption failed'));
|
|
362
|
+
const result = bitgo.decryptKeys({
|
|
363
|
+
walletIdEncryptedKeyPairs: [
|
|
364
|
+
{ walletId: 'wallet-id-1', encryptedPrv: 'encrypted-data-1' },
|
|
365
|
+
{ walletId: 'wallet-id-2', encryptedPrv: 'encrypted-data-2' },
|
|
366
|
+
],
|
|
367
|
+
password: 'password123',
|
|
368
|
+
});
|
|
369
|
+
result.should.be.an.Array();
|
|
370
|
+
result.should.have.length(1);
|
|
371
|
+
result[0].should.equal('wallet-id-2');
|
|
372
|
+
});
|
|
373
|
+
it('should correctly process multiple wallet keys', function () {
|
|
374
|
+
// Create a spy on the decrypt method
|
|
375
|
+
const decryptStub = sinon.stub(bitgo, 'decrypt');
|
|
376
|
+
// Configure the stub to throw for specific wallets
|
|
377
|
+
decryptStub
|
|
378
|
+
.withArgs({ input: 'encrypted-data-2', password: 'password123' })
|
|
379
|
+
.throws(new Error('decryption failed'));
|
|
380
|
+
decryptStub
|
|
381
|
+
.withArgs({ input: 'encrypted-data-4', password: 'password123' })
|
|
382
|
+
.throws(new Error('decryption failed'));
|
|
383
|
+
decryptStub.returns('success'); // Default return for other calls
|
|
384
|
+
const result = bitgo.decryptKeys({
|
|
385
|
+
walletIdEncryptedKeyPairs: [
|
|
386
|
+
{ walletId: 'wallet-id-1', encryptedPrv: 'encrypted-data-1' },
|
|
387
|
+
{ walletId: 'wallet-id-2', encryptedPrv: 'encrypted-data-2' },
|
|
388
|
+
{ walletId: 'wallet-id-3', encryptedPrv: 'encrypted-data-3' },
|
|
389
|
+
{ walletId: 'wallet-id-4', encryptedPrv: 'encrypted-data-4' },
|
|
390
|
+
],
|
|
391
|
+
password: 'password123',
|
|
392
|
+
});
|
|
393
|
+
// Should be called once for each wallet
|
|
394
|
+
decryptStub.callCount.should.equal(4);
|
|
395
|
+
// Should include only the failed wallet IDs
|
|
396
|
+
result.should.be.an.Array();
|
|
397
|
+
result.should.have.length(2);
|
|
398
|
+
result.should.containDeep(['wallet-id-2', 'wallet-id-4']);
|
|
399
|
+
});
|
|
400
|
+
});
|
|
401
|
+
describe('User-Agent header based on environment', function () {
|
|
402
|
+
afterEach(function () {
|
|
403
|
+
nock_1.default.cleanAll();
|
|
404
|
+
sinon.restore();
|
|
405
|
+
});
|
|
406
|
+
it('should set User-Agent header when running in Node.js (typeof window === undefined)', async function () {
|
|
407
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
408
|
+
env: 'custom',
|
|
409
|
+
customRootURI: 'https://app.example.local',
|
|
410
|
+
userAgent: 'TestAgent/1.0',
|
|
411
|
+
});
|
|
412
|
+
// Ensure we're in a Node.js environment by verifying window is undefined
|
|
413
|
+
(typeof window).should.equal('undefined');
|
|
414
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
415
|
+
.get('/api/v1/ping')
|
|
416
|
+
.matchHeader('User-Agent', 'TestAgent/1.0')
|
|
417
|
+
.reply(200, { status: 'ok' });
|
|
418
|
+
await bitgo.ping({
|
|
419
|
+
reqId: {
|
|
420
|
+
toString: () => 'test-123',
|
|
421
|
+
inc: () => {
|
|
422
|
+
/* mock */
|
|
423
|
+
},
|
|
424
|
+
},
|
|
425
|
+
});
|
|
426
|
+
scope.isDone().should.be.true();
|
|
427
|
+
});
|
|
428
|
+
it('should not set User-Agent header when running in browser (typeof window !== undefined)', async function () {
|
|
429
|
+
// Mock the window object to simulate browser environment
|
|
430
|
+
const windowStub = { location: 'mock' };
|
|
431
|
+
global.window = windowStub;
|
|
432
|
+
try {
|
|
433
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
434
|
+
env: 'custom',
|
|
435
|
+
customRootURI: 'https://app.example.local',
|
|
436
|
+
userAgent: 'TestAgent/1.0',
|
|
437
|
+
});
|
|
438
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
439
|
+
.get('/api/v1/ping')
|
|
440
|
+
.reply(function () {
|
|
441
|
+
// Verify User-Agent header is NOT set to our custom value
|
|
442
|
+
const userAgent = this.req.headers['user-agent'];
|
|
443
|
+
if (userAgent && userAgent.includes('TestAgent/1.0')) {
|
|
444
|
+
throw new Error('User-Agent should not be set in browser environment');
|
|
445
|
+
}
|
|
446
|
+
return [200, { status: 'ok' }];
|
|
447
|
+
});
|
|
448
|
+
await bitgo.ping({
|
|
449
|
+
reqId: {
|
|
450
|
+
toString: () => 'test-123',
|
|
451
|
+
inc: () => {
|
|
452
|
+
/* mock */
|
|
453
|
+
},
|
|
454
|
+
},
|
|
455
|
+
});
|
|
456
|
+
scope.isDone().should.be.true();
|
|
457
|
+
}
|
|
458
|
+
finally {
|
|
459
|
+
// Clean up the global window mock
|
|
460
|
+
delete global.window;
|
|
461
|
+
}
|
|
462
|
+
});
|
|
463
|
+
});
|
|
464
|
+
describe('hmacAuthStrategy token lifecycle', function () {
|
|
465
|
+
const ROOT = 'https://app.example.local';
|
|
466
|
+
// Builds a mock strategy whose setToken / clearToken are sinon stubs.
|
|
467
|
+
function makeStrategy(overrides = {}) {
|
|
468
|
+
const setTokenStub = sinon.stub().resolves();
|
|
469
|
+
const clearTokenStub = sinon.stub().resolves();
|
|
470
|
+
const strategy = {
|
|
471
|
+
calculateRequestHeaders: sinon.stub().resolves({ hmac: 'hmac', timestamp: 1, tokenHash: 'hash' }),
|
|
472
|
+
verifyResponse: sinon.stub().resolves({
|
|
473
|
+
isValid: true,
|
|
474
|
+
expectedHmac: 'hmac',
|
|
475
|
+
signatureSubject: '',
|
|
476
|
+
isInResponseValidityWindow: true,
|
|
477
|
+
verificationTime: Date.now(),
|
|
478
|
+
}),
|
|
479
|
+
calculateHMAC: sinon.stub().resolves('hashed-pw'),
|
|
480
|
+
setToken: setTokenStub,
|
|
481
|
+
clearToken: clearTokenStub,
|
|
482
|
+
...overrides,
|
|
483
|
+
};
|
|
484
|
+
return { strategy, setTokenStub, clearTokenStub };
|
|
485
|
+
}
|
|
486
|
+
afterEach(function () {
|
|
487
|
+
nock_1.default.cleanAll();
|
|
488
|
+
sinon.restore();
|
|
489
|
+
});
|
|
490
|
+
describe('authenticate()', function () {
|
|
491
|
+
it('calls setToken with the access_token received from the server', async function () {
|
|
492
|
+
const { strategy, setTokenStub } = makeStrategy();
|
|
493
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({ env: 'custom', customRootURI: ROOT, hmacAuthStrategy: strategy });
|
|
494
|
+
(0, nock_1.default)(ROOT)
|
|
495
|
+
.post('/api/auth/v1/session')
|
|
496
|
+
.reply(200, {
|
|
497
|
+
user: { username: 'test@example.com' },
|
|
498
|
+
access_token: 'v2xmyaccesstoken',
|
|
499
|
+
});
|
|
500
|
+
await bitgo.authenticate({ username: 'test@example.com', password: 'hunter2' });
|
|
501
|
+
setTokenStub.calledOnce.should.be.true();
|
|
502
|
+
setTokenStub.firstCall.args[0].should.equal('v2xmyaccesstoken');
|
|
503
|
+
});
|
|
504
|
+
it('awaits setToken before making ensureEcdhKeychain requests', async function () {
|
|
505
|
+
// This is the core regression test: if setToken is not awaited, the
|
|
506
|
+
// strategy's key material won't be ready before calculateRequestHeaders
|
|
507
|
+
// is called for the GET /user/settings request, and it would throw.
|
|
508
|
+
let keyReady = false;
|
|
509
|
+
const { strategy } = makeStrategy({
|
|
510
|
+
setToken: sinon.stub().callsFake(async () => {
|
|
511
|
+
// Simulate non-trivial async key derivation (like crypto.subtle.importKey).
|
|
512
|
+
await new Promise((resolve) => setImmediate(resolve));
|
|
513
|
+
keyReady = true;
|
|
514
|
+
}),
|
|
515
|
+
calculateRequestHeaders: sinon.stub().callsFake(async () => {
|
|
516
|
+
if (!keyReady) {
|
|
517
|
+
throw new Error('No token available. Call setToken() or restoreToken() first.');
|
|
518
|
+
}
|
|
519
|
+
return { hmac: 'hmac', timestamp: Date.now(), tokenHash: 'hash' };
|
|
520
|
+
}),
|
|
521
|
+
});
|
|
522
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({ env: 'custom', customRootURI: ROOT, hmacAuthStrategy: strategy });
|
|
523
|
+
(0, nock_1.default)(ROOT)
|
|
524
|
+
.post('/api/auth/v1/session')
|
|
525
|
+
.reply(200, {
|
|
526
|
+
user: { username: 'test@example.com' },
|
|
527
|
+
access_token: 'v2xmytoken',
|
|
528
|
+
});
|
|
529
|
+
// The GET /user/settings request made by ensureUserEcdhKeychainIsCreated
|
|
530
|
+
// must succeed — it would throw if setToken wasn't awaited first.
|
|
531
|
+
(0, nock_1.default)(ROOT)
|
|
532
|
+
.get('/api/v1/user/settings')
|
|
533
|
+
.reply(200, {
|
|
534
|
+
settings: { ecdhKeychain: 'xpub123' },
|
|
535
|
+
});
|
|
536
|
+
await bitgo.authenticate({
|
|
537
|
+
username: 'test@example.com',
|
|
538
|
+
password: 'hunter2',
|
|
539
|
+
ensureEcdhKeychain: true,
|
|
540
|
+
});
|
|
541
|
+
keyReady.should.be.true();
|
|
542
|
+
});
|
|
543
|
+
});
|
|
544
|
+
describe('authenticateWithPasskey()', function () {
|
|
545
|
+
const validPasskey = JSON.stringify({
|
|
546
|
+
id: 'credential-id',
|
|
547
|
+
rawId: 'raw-id',
|
|
548
|
+
type: 'public-key',
|
|
549
|
+
response: {
|
|
550
|
+
authenticatorData: 'auth-data',
|
|
551
|
+
clientDataJSON: 'client-data',
|
|
552
|
+
signature: 'sig',
|
|
553
|
+
userHandle: 'user-handle-123',
|
|
554
|
+
},
|
|
555
|
+
});
|
|
556
|
+
it('calls setToken with the access_token received from the server', async function () {
|
|
557
|
+
const { strategy, setTokenStub } = makeStrategy();
|
|
558
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({ env: 'custom', customRootURI: ROOT, hmacAuthStrategy: strategy });
|
|
559
|
+
(0, nock_1.default)(ROOT)
|
|
560
|
+
.post('/api/auth/v1/session')
|
|
561
|
+
.reply(200, {
|
|
562
|
+
user: { username: 'test@example.com' },
|
|
563
|
+
access_token: 'v2xpasskeytoken',
|
|
564
|
+
});
|
|
565
|
+
await bitgo.authenticateWithPasskey(validPasskey);
|
|
566
|
+
setTokenStub.calledOnce.should.be.true();
|
|
567
|
+
setTokenStub.firstCall.args[0].should.equal('v2xpasskeytoken');
|
|
568
|
+
});
|
|
569
|
+
});
|
|
570
|
+
describe('clearAsync()', function () {
|
|
571
|
+
it('clears _token and calls clearToken on the strategy', async function () {
|
|
572
|
+
const { strategy, clearTokenStub } = makeStrategy();
|
|
573
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({ env: 'custom', customRootURI: ROOT, hmacAuthStrategy: strategy });
|
|
574
|
+
bitgo.authenticateWithAccessToken({ accessToken: 'v2xsometoken' });
|
|
575
|
+
bitgo._token.should.equal('v2xsometoken');
|
|
576
|
+
await bitgo.clearAsync();
|
|
577
|
+
(bitgo._token === undefined).should.be.true();
|
|
578
|
+
clearTokenStub.calledOnce.should.be.true();
|
|
579
|
+
});
|
|
580
|
+
});
|
|
581
|
+
describe('refreshToken()', function () {
|
|
582
|
+
it('calls setToken with the new access_token from the OAuth response', async function () {
|
|
583
|
+
const { strategy, setTokenStub } = makeStrategy();
|
|
584
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
585
|
+
env: 'custom',
|
|
586
|
+
customRootURI: ROOT,
|
|
587
|
+
hmacAuthStrategy: strategy,
|
|
588
|
+
clientId: 'client-id',
|
|
589
|
+
clientSecret: 'client-secret',
|
|
590
|
+
});
|
|
591
|
+
bitgo._refreshToken = 'old-refresh-token';
|
|
592
|
+
(0, nock_1.default)(ROOT).post('/oauth/token').reply(200, {
|
|
593
|
+
access_token: 'v2xnewtoken',
|
|
594
|
+
refresh_token: 'new-refresh-token',
|
|
595
|
+
});
|
|
596
|
+
await bitgo.refreshToken();
|
|
597
|
+
setTokenStub.calledOnce.should.be.true();
|
|
598
|
+
setTokenStub.firstCall.args[0].should.equal('v2xnewtoken');
|
|
599
|
+
});
|
|
600
|
+
});
|
|
601
|
+
describe('authenticateWithAuthCode()', function () {
|
|
602
|
+
it('calls setToken with the access_token from the OAuth response', async function () {
|
|
603
|
+
const { strategy, setTokenStub } = makeStrategy();
|
|
604
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
605
|
+
env: 'custom',
|
|
606
|
+
customRootURI: ROOT,
|
|
607
|
+
hmacAuthStrategy: strategy,
|
|
608
|
+
clientId: 'client-id',
|
|
609
|
+
clientSecret: 'client-secret',
|
|
610
|
+
});
|
|
611
|
+
(0, nock_1.default)(ROOT).post('/oauth/token').reply(200, {
|
|
612
|
+
access_token: 'v2xauthcodetoken',
|
|
613
|
+
refresh_token: 'refresh-token',
|
|
614
|
+
});
|
|
615
|
+
// authenticateWithAuthCode calls this.me() after setting the token
|
|
616
|
+
(0, nock_1.default)(ROOT)
|
|
617
|
+
.get('/api/v1/user/me')
|
|
618
|
+
.reply(200, {
|
|
619
|
+
user: { username: 'test@example.com' },
|
|
620
|
+
});
|
|
621
|
+
await bitgo.authenticateWithAuthCode({ authCode: 'my-auth-code' });
|
|
622
|
+
setTokenStub.calledOnce.should.be.true();
|
|
623
|
+
setTokenStub.firstCall.args[0].should.equal('v2xauthcodetoken');
|
|
624
|
+
});
|
|
625
|
+
});
|
|
626
|
+
describe('sync token-setting methods', function () {
|
|
627
|
+
it('authenticateWithAccessToken does not call setToken (synchronous — caller must invoke setToken on the strategy manually)', function () {
|
|
628
|
+
const { strategy, setTokenStub } = makeStrategy();
|
|
629
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({ env: 'custom', customRootURI: ROOT, hmacAuthStrategy: strategy });
|
|
630
|
+
bitgo.authenticateWithAccessToken({ accessToken: 'v2xsynctoken' });
|
|
631
|
+
setTokenStub.called.should.be.false();
|
|
632
|
+
});
|
|
633
|
+
it('fromJSON does not call setToken (synchronous — caller must invoke setToken on the strategy manually)', function () {
|
|
634
|
+
const { strategy, setTokenStub } = makeStrategy();
|
|
635
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({ env: 'custom', customRootURI: ROOT, hmacAuthStrategy: strategy });
|
|
636
|
+
bitgo.fromJSON({ user: { username: 'test@example.com' }, token: 'v2xjsontoken' });
|
|
637
|
+
setTokenStub.called.should.be.false();
|
|
638
|
+
});
|
|
639
|
+
});
|
|
640
|
+
});
|
|
641
|
+
describe('constants parameter', function () {
|
|
642
|
+
it('should allow passing constants via options and expose via fetchConstants', async function () {
|
|
643
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
644
|
+
env: 'custom',
|
|
645
|
+
customRootURI: 'https://app.example.local',
|
|
646
|
+
clientConstants: { maxFeeRate: '123123123123123' },
|
|
647
|
+
});
|
|
648
|
+
const constants = await bitgo.fetchConstants();
|
|
649
|
+
constants.should.have.property('maxFeeRate', '123123123123123');
|
|
650
|
+
});
|
|
651
|
+
it('should refresh constants when cache has expired', async function () {
|
|
652
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
653
|
+
env: 'custom',
|
|
654
|
+
customRootURI: 'https://app.example.local',
|
|
655
|
+
});
|
|
656
|
+
// Set up cached constants with an expired cache
|
|
657
|
+
bitgoAPI_1.BitGoAPI._constants = bitgoAPI_1.BitGoAPI._constants || {};
|
|
658
|
+
bitgoAPI_1.BitGoAPI._constantsExpire = bitgoAPI_1.BitGoAPI._constantsExpire || {};
|
|
659
|
+
bitgoAPI_1.BitGoAPI._constants['custom'] = { maxFeeRate: 'old-value' };
|
|
660
|
+
bitgoAPI_1.BitGoAPI._constantsExpire['custom'] = new Date(Date.now() - 1000); // Expired 1 second ago
|
|
661
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
662
|
+
.get('/api/v1/client/constants')
|
|
663
|
+
.reply(200, {
|
|
664
|
+
constants: { maxFeeRate: 'new-value', newConstant: 'added' },
|
|
665
|
+
});
|
|
666
|
+
const constants = await bitgo.fetchConstants();
|
|
667
|
+
// Should return the new constants from the server
|
|
668
|
+
constants.should.have.property('maxFeeRate', 'new-value');
|
|
669
|
+
constants.should.have.property('newConstant', 'added');
|
|
670
|
+
scope.isDone().should.be.true();
|
|
671
|
+
nock_1.default.cleanAll();
|
|
672
|
+
});
|
|
673
|
+
it('should use cached constants when cache is still valid', async function () {
|
|
674
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
675
|
+
env: 'custom',
|
|
676
|
+
customRootURI: 'https://app.example.local',
|
|
677
|
+
});
|
|
678
|
+
// Set up cached constants with a future expiry
|
|
679
|
+
const cachedConstants = { maxFeeRate: 'cached-value', anotherSetting: 'cached-setting' };
|
|
680
|
+
bitgoAPI_1.BitGoAPI._constants = bitgoAPI_1.BitGoAPI._constants || {};
|
|
681
|
+
bitgoAPI_1.BitGoAPI._constantsExpire = bitgoAPI_1.BitGoAPI._constantsExpire || {};
|
|
682
|
+
bitgoAPI_1.BitGoAPI._constants['custom'] = cachedConstants;
|
|
683
|
+
bitgoAPI_1.BitGoAPI._constantsExpire['custom'] = new Date(Date.now() + 5 * 60 * 1000); // Valid for 5 more minutes
|
|
684
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
685
|
+
.get('/api/v1/client/constants')
|
|
686
|
+
.reply(200, { constants: { shouldNotBeUsed: true } });
|
|
687
|
+
const constants = await bitgo.fetchConstants();
|
|
688
|
+
// Should return the cached constants
|
|
689
|
+
constants.should.deepEqual(cachedConstants);
|
|
690
|
+
// Verify that no HTTP request was made (since cache was valid)
|
|
691
|
+
scope.isDone().should.be.false();
|
|
692
|
+
nock_1.default.cleanAll();
|
|
693
|
+
});
|
|
694
|
+
it('should use cached constants when no cache expiry is set', async function () {
|
|
695
|
+
const bitgo = new bitgoAPI_1.BitGoAPI({
|
|
696
|
+
env: 'custom',
|
|
697
|
+
customRootURI: 'https://app.example.local',
|
|
698
|
+
});
|
|
699
|
+
// Set up cached constants with no expiry
|
|
700
|
+
const cachedConstants = { maxFeeRate: 'no-expiry-value' };
|
|
701
|
+
bitgoAPI_1.BitGoAPI._constants = bitgoAPI_1.BitGoAPI._constants || {};
|
|
702
|
+
bitgoAPI_1.BitGoAPI._constantsExpire = bitgoAPI_1.BitGoAPI._constantsExpire || {};
|
|
703
|
+
bitgoAPI_1.BitGoAPI._constants['custom'] = cachedConstants;
|
|
704
|
+
bitgoAPI_1.BitGoAPI._constantsExpire['custom'] = undefined;
|
|
705
|
+
const scope = (0, nock_1.default)('https://app.example.local')
|
|
706
|
+
.get('/api/v1/client/constants')
|
|
707
|
+
.reply(200, { constants: { shouldNotBeUsed: true } });
|
|
708
|
+
const constants = await bitgo.fetchConstants();
|
|
709
|
+
// Should return the cached constants
|
|
710
|
+
constants.should.deepEqual(cachedConstants);
|
|
711
|
+
// Verify that no HTTP request was made (since no expiry means cache is always valid)
|
|
712
|
+
scope.isDone().should.be.false();
|
|
713
|
+
nock_1.default.cleanAll();
|
|
714
|
+
});
|
|
715
|
+
});
|
|
716
|
+
});
|
|
717
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYml0Z29BUEkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90ZXN0L3VuaXQvYml0Z29BUEkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxrQkFBZ0I7QUFDaEIsaURBQThDO0FBQzlDLDZDQUF5QztBQUN6Qyw2Q0FBK0I7QUFDL0IsZ0RBQXdCO0FBR3hCLFFBQVEsQ0FBQyxhQUFhLEVBQUU7SUFDdEIsUUFBUSxDQUFDLG9DQUFvQyxFQUFFO1FBQzdDLEVBQUUsQ0FBQyxpREFBaUQsRUFBRTtZQUNwRCxNQUFNLEtBQUssR0FBRyxJQUFJLG1CQUFRLENBQUM7Z0JBQ3pCLEdBQUcsRUFBRSxRQUFRO2dCQUNiLGFBQWEsRUFBRSwyQkFBMkI7Z0JBQzFDLHlCQUF5QixFQUFFLElBQUk7YUFDaEMsQ0FBQyxDQUFDO1lBRUgsS0FBSyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLDJCQUEyQixDQUFDLENBQUM7WUFDeEQsS0FBSyxDQUFDLHlCQUF5QixDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDckQsQ0FBQyxDQUFDLENBQUM7UUFFSCxFQUFFLENBQUMsa0RBQWtELEVBQUU7WUFDckQsTUFBTSxLQUFLLEdBQUcsSUFBSSxtQkFBUSxDQUFDO2dCQUN6QixHQUFHLEVBQUUsUUFBUTtnQkFDYixhQUFhLEVBQUUsMkJBQTJCO2dCQUMxQyx5QkFBeUIsRUFBRSxLQUFLO2FBQ2pDLENBQUMsQ0FBQztZQUVILEtBQUssQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQywyQkFBMkIsQ0FBQyxDQUFDO1lBQ3hELEtBQUssQ0FBQyx5QkFBeUIsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RELENBQUMsQ0FBQyxDQUFDO1FBRUgsRUFBRSxDQUFDLGtEQUFrRCxFQUFFO1lBQ3JELE1B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N0UsbUJBQWdCLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxHQUFHLGVBQWUsQ0FBQztZQUN4RCxtQkFBZ0IsQ0FBQyxnQkFBZ0IsQ0FBQyxRQUFRLENBQUMsR0FBRyxTQUFTLENBQUM7WUFFekQsTUFBTSxLQUFLLEdBQUcsSUFBQSxjQUFJLEVBQUMsMkJBQTJCLENBQUM7aUJBQzVDLEdBQUcsQ0FBQywwQkFBMEIsQ0FBQztpQkFDL0IsS0FBSyxDQUFDLEdBQUcsRUFBRSxFQUFFLFNBQVMsRUFBRSxFQUFFLGVBQWUsRUFBRSxJQUFJLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFFeEQsTUFBTSxTQUFTLEdBQUcsTUFBTSxLQUFLLENBQUMsY0FBYyxFQUFFLENBQUM7WUFFL0MscUNBQXFDO1lBQ3JDLFNBQVMsQ0FBQyxNQUFNLENBQUMsU0FBUyxDQUFDLGVBQWUsQ0FBQyxDQUFDO1lBRTVDLHFGQUFxRjtZQUNyRixLQUFLLENBQUMsTUFBTSxFQUFFLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUVqQyxjQUFJLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDbEIsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDLENBQUMsQ0FBQztBQUNMLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICdzaG91bGQnO1xuaW1wb3J0IHsgQml0R29BUEkgfSBmcm9tICcuLi8uLi9zcmMvYml0Z29BUEknO1xuaW1wb3J0IHsgUHJveHlBZ2VudCB9IGZyb20gJ3Byb3h5LWFnZW50JztcbmltcG9ydCAqIGFzIHNpbm9uIGZyb20gJ3Npbm9uJztcbmltcG9ydCBub2NrIGZyb20gJ25vY2snO1xuaW1wb3J0IHR5cGUgeyBJSG1hY0F1dGhTdHJhdGVneSB9IGZyb20gJ0BiaXRnby1iZXRhL3Nkay1obWFjJztcblxuZGVzY3JpYmUoJ0NvbnN0cnVjdG9yJywgZnVuY3Rpb24gKCkge1xuICBkZXNjcmliZSgnY29va2llc1Byb3BhZ2F0aW9uRW5hYmxlZCBhcmd1bWVudCcsIGZ1bmN0aW9uICgpIHtcbiAgICBpdCgnY29va2llc1Byb3BhZ2F0aW9uRW5hYmxlZCBpcyBlbmFibGVkIGV4cGxpY2l0bHknLCBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCBiaXRnbyA9IG5ldyBCaXRHb0FQSSh7XG4gICAgICAgIGVudjogJ2N1c3RvbScsXG4gICAgICAgIGN1c3RvbVJvb3RVUkk6ICdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJyxcbiAgICAgICAgY29va2llc1Byb3BhZ2F0aW9uRW5hYmxlZDogdHJ1ZSxcbiAgICAgIH0pO1xuXG4gICAgICBiaXRnby5zaG91bGQuaGF2ZS5wcm9wZXJ0eSgnY29va2llc1Byb3BhZ2F0aW9uRW5hYmxlZCcpO1xuICAgICAgYml0Z28uY29va2llc1Byb3BhZ2F0aW9uRW5hYmxlZC5zaG91bGQuZXF1YWwodHJ1ZSk7XG4gICAgfSk7XG5cbiAgICBpdCgnY29va2llc1Byb3BhZ2F0aW9uRW5hYmxlZCBpcyBkaXNhYmxlZCBleHBsaWNpdGx5JywgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICdjdXN0b20nLFxuICAgICAgICBjdXN0b21Sb290VVJJOiAnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcsXG4gICAgICAgIGNvb2tpZXNQcm9wYWdhdGlvbkVuYWJsZWQ6IGZhbHNlLFxuICAgICAgfSk7XG5cbiAgICAgIGJpdGdvLnNob3VsZC5oYXZlLnByb3BlcnR5KCdjb29raWVzUHJvcGFnYXRpb25FbmFibGVkJyk7XG4gICAgICBiaXRnby5jb29raWVzUHJvcGFnYXRpb25FbmFibGVkLnNob3VsZC5lcXVhbChmYWxzZSk7XG4gICAgfSk7XG5cbiAgICBpdCgnY29va2llc1Byb3BhZ2F0aW9uRW5hYmxlZCBpcyBkaXNhYmxlZCBieSBkZWZhdWx0JywgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICdjdXN0b20nLFxuICAgICAgICBjdXN0b21Sb290VVJJOiAnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcsXG4gICAgICB9KTtcblxuICAgICAgYml0Z28uc2hvdWxkLmhhdmUucHJvcGVydHkoJ2Nvb2tpZXNQcm9wYWdhdGlvbkVuYWJsZWQnKTtcbiAgICAgIGJpdGdvLmNvb2tpZXNQcm9wYWdhdGlvbkVuYWJsZWQuc2hvdWxkLmVxdWFsKGZhbHNlKTtcbiAgICB9KTtcbiAgfSk7XG5cbiAgZGVzY3JpYmUoJ3JlcXVlc3RJZFByZWZpeCBhcmd1bWVudCcsIGZ1bmN0aW9uICgpIHtcbiAgICBhZnRlckVhY2goZnVuY3Rpb24gKCkge1xuICAgICAgbm9jay5jbGVhbkFsbCgpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBwcmVwZW5kIHJlcXVlc3RJZFByZWZpeCB0byBSZXF1ZXN0LUlEIGhlYWRlciB3aGVuIHNldCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgZW52OiAnY3VzdG9tJyxcbiAgICAgICAgY3VzdG9tUm9vdFVSSTogJ2h0dHBzOi8vYXBwLmV4YW1wbGUubG9jYWwnLFxuICAgICAgICByZXF1ZXN0SWRQcmVmaXg6ICd0ZXN0LXByZWZpeC0nLFxuICAgICAgfSk7XG5cbiAgICAgIGNvbnN0IHNjb3BlID0gbm9jaygnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcpXG4gICAgICAgIC5nZXQoJy9hcGkvdjEvcGluZycpXG4gICAgICAgIC5tYXRjaEhlYWRlcignUmVxdWVzdC1JRCcsIC9edGVzdC1wcmVmaXgtLylcbiAgICAgICAgLnJlcGx5KDIwMCwgeyBzdGF0dXM6ICdvaycgfSk7XG5cbiAgICAgIGF3YWl0IGJpdGdvLnBpbmcoe1xuICAgICAgICByZXFJZDoge1xuICAgICAgICAgIHRvU3RyaW5nOiAoKSA9PiAnMTIzNDUnLFxuICAgICAgICAgIGluYzogKCkgPT4ge1xuICAgICAgICAgICAgLyogbW9jayAqL1xuICAgICAgICAgIH0sXG4gICAgICAgIH0gYXMgYW55LFxuICAgICAgfSk7XG5cbiAgICAgIHNjb3BlLmlzRG9uZSgpLnNob3VsZC5iZS50cnVlKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIG5vdCBtb2RpZnkgUmVxdWVzdC1JRCBoZWFkZXIgd2hlbiByZXF1ZXN0SWRQcmVmaXggaXMgbm90IHNldCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgZW52OiAnY3VzdG9tJyxcbiAgICAgICAgY3VzdG9tUm9vdFVSSTogJ2h0dHBzOi8vYXBwLmV4YW1wbGUubG9jYWwnLFxuICAgICAgfSk7XG5cbiAgICAgIGNvbnN0IHNjb3BlID0gbm9jaygnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcpXG4gICAgICAgIC5nZXQoJy9hcGkvdjEvcGluZycpXG4gICAgICAgIC5tYXRjaEhlYWRlcignUmVxdWVzdC1JRCcsIC9eMTIzNDUkLylcbiAgICAgICAgLnJlcGx5KDIwMCwgeyBzdGF0dXM6ICdvaycgfSk7XG5cbiAgICAgIGF3YWl0IGJpdGdvLnBpbmcoe1xuICAgICAgICByZXFJZDoge1xuICAgICAgICAgIHRvU3RyaW5nOiAoKSA9PiAnMTIzNDUnLFxuICAgICAgICAgIGluYzogKCkgPT4ge1xuICAgICAgICAgICAgLyogbW9jayAqL1xuICAgICAgICAgIH0sXG4gICAgICAgIH0gYXMgYW55LFxuICAgICAgfSk7XG5cbiAgICAgIHNjb3BlLmlzRG9uZSgpLnNob3VsZC5iZS50cnVlKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGNvcnJlY3RseSBmb3JtYXQgUmVxdWVzdC1JRCB3aXRoIHByZWZpeCBhbmQgbnVtZXJpYyBzZXF1ZW5jZScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgZW52OiAnY3VzdG9tJyxcbiAgICAgICAgY3VzdG9tUm9vdFVSSTogJ2h0dHBzOi8vYXBwLmV4YW1wbGUubG9jYWwnLFxuICAgICAgICByZXF1ZXN0SWRQcmVmaXg6ICdteWFwcC12MS0nLFxuICAgICAgfSk7XG5cbiAgICAgIGNvbnN0IHNjb3BlID0gbm9jaygnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcpXG4gICAgICAgIC5nZXQoJy9hcGkvdjEvcGluZycpXG4gICAgICAgIC5tYXRjaEhlYWRlcignUmVxdWVzdC1JRCcsICdteWFwcC12MS10cmFjZS0xMjMnKVxuICAgICAgICAucmVwbHkoMjAwLCB7IHN0YXR1czogJ29rJyB9KTtcblxuICAgICAgYXdhaXQgYml0Z28ucGluZyh7XG4gICAgICAgIHJlcUlkOiB7XG4gICAgICAgICAgdG9TdHJpbmc6ICgpID0+ICd0cmFjZS0xMjMnLFxuICAgICAgICAgIGluYzogKCkgPT4ge1xuICAgICAgICAgICAgLyogbW9jayAqL1xuICAgICAgICAgIH0sXG4gICAgICAgIH0gYXMgYW55LFxuICAgICAgfSk7XG5cbiAgICAgIHNjb3BlLmlzRG9uZSgpLnNob3VsZC5iZS50cnVlKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHdvcmsgd2l0aCBlbXB0eSBzdHJpbmcgcHJlZml4JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICdjdXN0b20nLFxuICAgICAgICBjdXN0b21Sb290VVJJOiAnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcsXG4gICAgICAgIHJlcXVlc3RJZFByZWZpeDogJycsXG4gICAgICB9KTtcblxuICAgICAgY29uc3Qgc2NvcGUgPSBub2NrKCdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJylcbiAgICAgICAgLmdldCgnL2FwaS92MS9waW5nJylcbiAgICAgICAgLm1hdGNoSGVhZGVyKCdSZXF1ZXN0LUlEJywgJ2FiYy0xMjMnKVxuICAgICAgICAucmVwbHkoMjAwLCB7IHN0YXR1czogJ29rJyB9KTtcblxuICAgICAgYXdhaXQgYml0Z28ucGluZyh7XG4gICAgICAgIHJlcUlkOiB7XG4gICAgICAgICAgdG9TdHJpbmc6ICgpID0+ICdhYmMtMTIzJyxcbiAgICAgICAgICBpbmM6ICgpID0+IHtcbiAgICAgICAgICAgIC8qIG1vY2sgKi9cbiAgICAgICAgICB9LFxuICAgICAgICB9IGFzIGFueSxcbiAgICAgIH0pO1xuXG4gICAgICBzY29wZS5pc0RvbmUoKS5zaG91bGQuYmUudHJ1ZSgpO1xuICAgIH0pO1xuICB9KTtcbiAgZGVzY3JpYmUoJ2h0dHAgcHJveHkgYWdlbnQnLCBmdW5jdGlvbiAoKSB7XG4gICAgaXQoJ2h0dHAgcHJveHkgYWdlbnQgc2hhbGwgYmUgY3JlYXRlZCB3aGVuIHByb3h5KGN1c3RvbVByb3h5YWdlbnQpIGlzIHNldCcsIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IGN1c3RvbVByb3h5QWdlbnQgPSBuZXcgUHJveHlBZ2VudCh7XG4gICAgICAgIGdldFByb3h5Rm9yVXJsOiAoKSA9PiAnaHR0cDovL2xvY2FsaG9zdDozMDAwJyxcbiAgICAgIH0pO1xuICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICdjdXN0b20nLFxuICAgICAgICBjdXN0b21Sb290VVJJOiAnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcsXG4gICAgICAgIGN1c3RvbVByb3h5QWdlbnQsXG4gICAgICB9KTtcblxuICAgICAgYml0Z28uc2hvdWxkLmhhdmUucHJvcGVydHkoJ19jdXN0b21Qcm94eUFnZW50JywgY3VzdG9tUHJveHlBZ2VudCk7XG4gICAgfSk7XG5cbiAgICBpdCgnYml0Z28gYXBpIGlzIHN0aWxsIGluaXRpYXRlZCB3aGVuIHByb3h5KGN1c3RvbVByb3h5QWdlbnQpIGlzIG5vdCBzZXQnLCBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCBiaXRnbyA9IG5ldyBCaXRHb0FQSSh7XG4gICAgICAgIGVudjogJ2N1c3RvbScsXG4gICAgICAgIGN1c3RvbVJvb3RVUkk6ICdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJyxcbiAgICAgIH0pO1xuXG4gICAgICBiaXRnby5zaG91bGQuaGF2ZS5wcm9wZXJ0eSgnX2N1c3RvbVByb3h5QWdlbnQnLCB1bmRlZmluZWQpO1xuICAgIH0pO1xuICB9KTtcblxuICBkZXNjcmliZSgndmVyaWZ5QWRkcmVzcycsIGZ1bmN0aW9uICgpIHtcbiAgICBpdCgnc2hvdWxkIHN1Y2Nlc3NmdWxseSB2ZXJpZnkgYSBiYXNlNTggYWRkcmVzcycsIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgZW52OiAndGVzdCcsXG4gICAgICB9KTtcblxuICAgICAgYml0Z28udmVyaWZ5QWRkcmVzcyh7IGFkZHJlc3M6ICcyTjZwYVQyVFU0TjFYcGFaakppQXBXSlhvZXlyTDNVV3BrWicgfSkuc2hvdWxkLmJlLnRydWUoKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgc3VjY2Vzc2Z1bGx5IHZlcmlmeSBhIGJlY2gzMiBhZGRyZXNzJywgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICd0ZXN0JyxcbiAgICAgIH0pO1xuXG4gICAgICBiaXRnb1xuICAgICAgICAudmVyaWZ5QWRkcmVzcyh7IGFkZHJlc3M6ICd0YjFxZ3V6eWs0dzZrYXF0cHNjenM1YWowdzhyNzU5OGpxMzZlZ204ZTk4d3FwaDNyd21leDY4c2VzbGdzZycgfSlcbiAgICAgICAgLnNob3VsZC5iZS50cnVlKCk7XG4gICAgfSk7XG4gIH0pO1xuXG4gIGRlc2NyaWJlKCd1cmwnLCBmdW5jdGlvbiAoKSB7XG4gICAgaXQoJ3Nob3VsZCByZXR1cm4gdGhlIGNvcnJlY3QgVVJMIGZvciB2ZXJzaW9uIDEnLCBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCBiaXRnbyA9IG5ldyBCaXRHb0FQSSh7XG4gICAgICAgIGVudjogJ3Rlc3QnLFxuICAgICAgICBjdXN0b21Sb290VVJJOiAnaHR0cHM6Ly90ZXN0LmJpdGdvLmNvbScsXG4gICAgICB9KTtcbiAgICAgIGNvbnN0IHBhdGggPSAnL3Rlc3QtcGF0aCc7XG4gICAgICBjb25zdCBleHBlY3RlZFVybCA9ICdodHRwczovL3Rlc3QuYml0Z28uY29tL2FwaS92MS90ZXN0LXBhdGgnO1xuICAgICAgY29uc3QgcmVzdWx0ID0gYml0Z28udXJsKHBhdGgsIDEpO1xuICAgICAgcmVzdWx0LnNob3VsZC5lcXVhbChleHBlY3RlZFVybCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHJldHVybiB0aGUgY29ycmVjdCBVUkwgZm9yIHZlcnNpb24gMicsIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgZW52OiAndGVzdCcsXG4gICAgICAgIGN1c3RvbVJvb3RVUkk6ICdodHRwczovL3Rlc3QuYml0Z28uY29tJyxcbiAgICAgIH0pO1xuICAgICAgY29uc3QgcGF0aCA9ICcvdGVzdC1wYXRoJztcbiAgICAgIGNvbnN0IGV4cGVjdGVkVXJsID0gJ2h0dHBzOi8vdGVzdC5iaXRnby5jb20vYXBpL3YyL3Rlc3QtcGF0aCc7XG4gICAgICBjb25zdCByZXN1bHQgPSBiaXRnby51cmwocGF0aCwgMik7XG4gICAgICByZXN1bHQuc2hvdWxkLmVxdWFsKGV4cGVjdGVkVXJsKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgcmV0dXJuIHRoZSBjb3JyZWN0IFVSTCBmb3IgdmVyc2lvbiAzJywgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICd0ZXN0JyxcbiAgICAgICAgY3VzdG9tUm9vdFVSSTogJ2h0dHBzOi8vdGVzdC5iaXRnby5jb20nLFxuICAgICAgfSk7XG4gICAgICBjb25zdCBwYXRoID0gJy90ZXN0LXBhdGgnO1xuICAgICAgY29uc3QgZXhwZWN0ZWRVcmwgPSAnaHR0cHM6Ly90ZXN0LmJpdGdvLmNvbS9hcGkvdjMvdGVzdC1wYXRoJztcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGJpdGdvLnVybChwYXRoLCAzKTtcbiAgICAgIHJlc3VsdC5zaG91bGQuZXF1YWwoZXhwZWN0ZWRVcmwpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBkZWZhdWx0IHRvIHZlcnNpb24gMSBpZiBubyB2ZXJzaW9uIGlzIHByb3ZpZGVkJywgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICd0ZXN0JyxcbiAgICAgICAgY3VzdG9tUm9vdFVSSTogJ2h0dHBzOi8vdGVzdC5iaXRnby5jb20nLFxuICAgICAgfSk7XG4gICAgICBjb25zdCBwYXRoID0gJy90ZXN0LXBhdGgnO1xuICAgICAgY29uc3QgZXhwZWN0ZWRVcmwgPSAnaHR0cHM6Ly90ZXN0LmJpdGdvLmNvbS9hcGkvdjEvdGVzdC1wYXRoJztcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGJpdGdvLnVybChwYXRoKTtcbiAgICAgIHJlc3VsdC5zaG91bGQuZXF1YWwoZXhwZWN0ZWRVcmwpO1xuICAgIH0pO1xuICB9KTtcblxuICBkZXNjcmliZSgnZGVjcnlwdEtleXMnLCBmdW5jdGlvbiAoKSB7XG4gICAgbGV0IGJpdGdvOiBCaXRHb0FQSTtcblxuICAgIGJlZm9yZUVhY2goZnVuY3Rpb24gKCkge1xuICAgICAgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICd0ZXN0JyxcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgYWZ0ZXJFYWNoKGZ1bmN0aW9uICgpIHtcbiAgICAgIHNpbm9uLnJlc3RvcmUoKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdGhyb3cgaWYgbm8gcGFyYW1zIGFyZSBwcm92aWRlZCcsIGZ1bmN0aW9uICgpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIC8vIEB0cy1leHBlY3QtZXJyb3IgLSBpbnRlbnRpb25hbGx5IGNhbGxpbmcgd2l0aCBubyBwYXJhbXMgZm9yIHRlc3RcbiAgICAgICAgYml0Z28uZGVjcnlwdEtleXMoKTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdFeHBlY3RlZCBlcnJvciBidXQgZ290IG5vbmUnKTtcbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgZS5tZXNzYWdlLnNob3VsZC5jb250YWluRXFsKCdNaXNzaW5nIHBhcmFtZXRlcicpO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCB0aHJvdyBpZiB3YWxsZXRJZEVuY3J5cHRlZEtleVBhaXJzIGlzIG1pc3NpbmcnLCBmdW5jdGlvbiAoKSB7XG4gICAgICB0cnkge1xuICAgICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIC0gaW50ZW50aW9uYWxseSBtaXNzaW5nIHJlcXVpcmVkIHBhcmFtXG4gICAgICAgIGJpdGdvLmRlY3J5cHRLZXlzKHsgcGFzc3dvcmQ6ICdwYXNzd29yZDEyMycgfSk7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignRXhwZWN0ZWQgZXJyb3IgYnV0IGdvdCBub25lJyk7XG4gICAgICB9IGNhdGNoIChlKSB7XG4gICAgICAgIGUubWVzc2FnZS5zaG91bGQuY29udGFpbkVxbCgnTWlzc2luZyBwYXJhbWV0ZXI6IHdhbGxldElkRW5jcnlwdGVkS2V5UGFpcnMnKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdGhyb3cgaWYgcGFzc3dvcmQgaXMgbWlzc2luZycsIGZ1bmN0aW9uICgpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIC8vIEB0cy1leHBlY3QtZXJyb3IgLSBpbnRlbnRpb25hbGx5IG1pc3NpbmcgcmVxdWlyZWQgcGFyYW1cbiAgICAgICAgYml0Z28uZGVjcnlwdEtleXMoeyB3YWxsZXRJZEVuY3J5cHRlZEtleVBhaXJzOiBbXSB9KTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdFeHBlY3RlZCBlcnJvciBidXQgZ290IG5vbmUnKTtcbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgZS5tZXNzYWdlLnNob3VsZC5jb250YWluRXFsKCdNaXNzaW5nIHBhcmFtZXRlcjogcGFzc3dvcmQnKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdGhyb3cgaWYgd2FsbGV0SWRFbmNyeXB0ZWRLZXlQYWlycyBpcyBub3QgYW4gYXJyYXknLCBmdW5jdGlvbiAoKSB7XG4gICAgICB0cnkge1xuICAgICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIC0gaW50ZW50aW9uYWxseSBwcm92aWRpbmcgd3JvbmcgdHlwZVxuICAgICAgICBiaXRnby5kZWNyeXB0S2V5cyh7IHdhbGxldElkRW5jcnlwdGVkS2V5UGFpcnM6ICdub3QgYW4gYXJyYXknLCBwYXNzd29yZDogJ3Bhc3N3b3JkMTIzJyB9KTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdFeHBlY3RlZCBlcnJvciBidXQgZ290IG5vbmUnKTtcbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgZS5tZXNzYWdlLnNob3VsZC5lcXVhbCgnd2FsbGV0SWRFbmNyeXB0ZWRLZXlQYWlycyBtdXN0IGJlIGFuIGFycmF5Jyk7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHJldHVybiBlbXB0eSBhcnJheSBmb3IgZW1wdHkgd2FsbGV0SWRFbmNyeXB0ZWRLZXlQYWlycycsIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGJpdGdvLmRlY3J5cHRLZXlzKHsgd2FsbGV0SWRFbmNyeXB0ZWRLZXlQYWlyczogW10sIHBhc3N3b3JkOiAncGFzc3dvcmQxMjMnIH0pO1xuICAgICAgcmVzdWx0LnNob3VsZC5iZS5hbi5BcnJheSgpO1xuICAgICAgcmVzdWx0LnNob3VsZC5iZS5lbXB0eSgpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCB0aHJvdyBpZiBhbnkgd2FsbGV0SWQgaXMgbWlzc2luZyBvciBub3QgYSBzdHJpbmcnLCBmdW5jdGlvbiAoKSB7XG4gICAgICB0cnkge1xuICAgICAgICBiaXRnby5kZWNyeXB0S2V5cyh7XG4gICAgICAgICAgd2FsbGV0SWRFbmNyeXB0ZWRLZXlQYWlyczogW1xuICAgICAgICAgICAgLy8gQHRzLWV4cGVjdC1lcnJvciAtIGludGVudGlvbmFsbHkgbWlzc2luZyB3YWxsZXRJZFxuICAgICAgICAgICAge1xuICAgICAgICAgICAgICBlbmNyeXB0ZWRQcnY6ICdlbmNyeXB0ZWQtZGF0YScsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIF0sXG4gICAgICAgICAgcGFzc3dvcmQ6ICdwYXNzd29yZDEyMycsXG4gICAgICAgIH0pO1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ0V4cGVjdGVkIGVycm9yIGJ1dCBnb3Qgbm9uZScpO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICBlLm1lc3NhZ2Uuc2hvdWxkLmVxdWFsKCdlYWNoIGtleSBwYWlyIG11c3QgaGF2ZSBhIHN0cmluZyB3YWxsZXRJZCcpO1xuICAgICAgfVxuXG4gICAgICB0cnkge1xuICAgICAgICBiaXRnby5kZWNyeXB0S2V5cyh7XG4gICAgICAgICAgd2FsbGV0SWRFbmNyeXB0ZWRLZXlQYWlyczogW1xuICAgICAgICAgICAge1xuICAgICAgICAgICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIC0gaW50ZW50aW9uYWxseSBwcm92aWRpbmcgd3JvbmcgdHlwZVxuICAgICAgICAgICAgICB3YWxsZXRJZDogMTIzLFxuICAgICAgICAgICAgICBlbmNyeXB0ZWRQcnY6ICdlbmNyeXB0ZWQtZGF0YScsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIF0sXG4gICAgICAgICAgcGFzc3dvcmQ6ICdwYXNzd29yZDEyMycsXG4gICAgICAgIH0pO1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ0V4cGVjdGVkIGVycm9yIGJ1dCBnb3Qgbm9uZScpO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICBlLm1lc3NhZ2Uuc2hvdWxkLmVxdWFsKCdlYWNoIGtleSBwYWlyIG11c3QgaGF2ZSBhIHN0cmluZyB3YWxsZXRJZCcpO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCB0aHJvdyBpZiBhbnkgZW5jcnlwdGVkUHJ2IGlzIG1pc3Npbmcgb3Igbm90IGEgc3RyaW5nJywgZnVuY3Rpb24gKCkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgYml0Z28uZGVjcnlwdEtleXMoe1xuICAgICAgICAgIHdhbGxldElkRW5jcnlwdGVkS2V5UGFpcnM6IFtcbiAgICAgICAgICAgIC8vIEB0cy1leHBlY3QtZXJyb3IgLSBpbnRlbnRpb25hbGx5IG1pc3NpbmcgZW5jcnlwdGVkUHJ2XG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHdhbGxldElkOiAnd2FsbGV0LWlkLTEnLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICBdLFxuICAgICAgICAgIHBhc3N3b3JkOiAncGFzc3dvcmQxMjMnLFxuICAgICAgICB9KTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdFeHBlY3RlZCBlcnJvciBidXQgZ290IG5vbmUnKTtcbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgZS5tZXNzYWdlLnNob3VsZC5lcXVhbCgnZWFjaCBrZXkgcGFpciBtdXN0IGhhdmUgYSBzdHJpbmcgZW5jcnlwdGVkUHJ2Jyk7XG4gICAgICB9XG5cbiAgICAgIHRyeSB7XG4gICAgICAgIGJpdGdvLmRlY3J5cHRLZXlzKHtcbiAgICAgICAgICB3YWxsZXRJZEVuY3J5cHRlZEtleVBhaXJzOiBbXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHdhbGxldElkOiAnd2FsbGV0LWlkLTEnLFxuICAgICAgICAgICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIC0gaW50ZW50aW9uYWxseSBwcm92aWRpbmcgd3JvbmcgdHlwZVxuICAgICAgICAgICAgICBlbmNyeXB0ZWRQcnY6IDEyMyxcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgXSxcbiAgICAgICAgICBwYXNzd29yZDogJ3Bhc3N3b3JkMTIzJyxcbiAgICAgICAgfSk7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignRXhwZWN0ZWQgZXJyb3IgYnV0IGdvdCBub25lJyk7XG4gICAgICB9IGNhdGNoIChlKSB7XG4gICAgICAgIGUubWVzc2FnZS5zaG91bGQuZXF1YWwoJ2VhY2gga2V5IHBhaXIgbXVzdCBoYXZlIGEgc3RyaW5nIGVuY3J5cHRlZFBydicpO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCByZXR1cm4gd2FsbGV0SWRzIG9mIGtleXMgdGhhdCBmYWlsZWQgdG8gZGVjcnlwdCcsIGZ1bmN0aW9uICgpIHtcbiAgICAgIC8vIENyZWF0ZSBhIHN0dWIgZm9yIHRoZSBkZWNyeXB0IG1ldGhvZFxuICAgICAgY29uc3QgZGVjcnlwdFN0dWIgPSBzaW5vbi5zdHViKGJpdGdvLCAnZGVjcnlwdCcpO1xuXG4gICAgICAvLyBNYWtlIGl0IHN1Y2NlZWQgZm9yIGZpcnN0IHdhbGxldCBhbmQgZmFpbCBmb3Igc2Vjb25kIHdhbGxldFxuICAgICAgZGVjcnlwdFN0dWIub25GaXJzdENhbGwoKS5yZXR1cm5zKCdkZWNyeXB0ZWQta2V5LTEnKTtcbiAgICAgIGRlY3J5cHRTdHViLm9uU2Vjb25kQ2FsbCgpLnRocm93cyhuZXcgRXJyb3IoJ2RlY3J5cHRpb24gZmFpbGVkJykpO1xuXG4gICAgICBjb25zdCByZXN1bHQgPSBiaXRnby5kZWNyeXB0S2V5cyh7XG4gICAgICAgIHdhbGxldElkRW5jcnlwdGVkS2V5UGFpcnM6IFtcbiAgICAgICAgICB7IHdhbGxldElkOiAnd2FsbGV0LWlkLTEnLCBlbmNyeXB0ZWRQcnY6ICdlbmNyeXB0ZWQtZGF0YS0xJyB9LFxuICAgICAgICAgIHsgd2FsbGV0SWQ6ICd3YWxsZXQtaWQtMicsIGVuY3J5cHRlZFBydjogJ2VuY3J5cHRlZC1kYXRhLTInIH0sXG4gICAgICAgIF0sXG4gICAgICAgIHBhc3N3b3JkOiAncGFzc3dvcmQxMjMnLFxuICAgICAgfSk7XG5cbiAgICAgIHJlc3VsdC5zaG91bGQuYmUuYW4uQXJyYXkoKTtcbiAgICAgIHJlc3VsdC5zaG91bGQuaGF2ZS5sZW5ndGgoMSk7XG4gICAgICByZXN1bHRbMF0uc2hvdWxkLmVxdWFsKCd3YWxsZXQtaWQtMicpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBjb3JyZWN0bHkgcHJvY2VzcyBtdWx0aXBsZSB3YWxsZXQga2V5cycsIGZ1bmN0aW9uICgpIHtcbiAgICAgIC8vIENyZWF0ZSBhIHNweSBvbiB0aGUgZGVjcnlwdCBtZXRob2RcbiAgICAgIGNvbnN0IGRlY3J5cHRTdHViID0gc2lub24uc3R1YihiaXRnbywgJ2RlY3J5cHQnKTtcblxuICAgICAgLy8gQ29uZmlndXJlIHRoZSBzdHViIHRvIHRocm93IGZvciBzcGVjaWZpYyB3YWxsZXRzXG4gICAgICBkZWNyeXB0U3R1YlxuICAgICAgICAud2l0aEFyZ3MoeyBpbnB1dDogJ2VuY3J5cHRlZC1kYXRhLTInLCBwYXNzd29yZDogJ3Bhc3N3b3JkMTIzJyB9KVxuICAgICAgICAudGhyb3dzKG5ldyBFcnJvcignZGVjcnlwdGlvbiBmYWlsZWQnKSk7XG4gICAgICBkZWNyeXB0U3R1YlxuICAgICAgICAud2l0aEFyZ3MoeyBpbnB1dDogJ2VuY3J5cHRlZC1kYXRhLTQnLCBwYXNzd29yZDogJ3Bhc3N3b3JkMTIzJyB9KVxuICAgICAgICAudGhyb3dzKG5ldyBFcnJvcignZGVjcnlwdGlvbiBmYWlsZWQnKSk7XG4gICAgICBkZWNyeXB0U3R1Yi5yZXR1cm5zKCdzdWNjZXNzJyk7IC8vIERlZmF1bHQgcmV0dXJuIGZvciBvdGhlciBjYWxsc1xuXG4gICAgICBjb25zdCByZXN1bHQgPSBiaXRnby5kZWNyeXB0S2V5cyh7XG4gICAgICAgIHdhbGxldElkRW5jcnlwdGVkS2V5UGFpcnM6IFtcbiAgICAgICAgICB7IHdhbGxldElkOiAnd2FsbGV0LWlkLTEnLCBlbmNyeXB0ZWRQcnY6ICdlbmNyeXB0ZWQtZGF0YS0xJyB9LFxuICAgICAgICAgIHsgd2FsbGV0SWQ6ICd3YWxsZXQtaWQtMicsIGVuY3J5cHRlZFBydjogJ2VuY3J5cHRlZC1kYXRhLTInIH0sXG4gICAgICAgICAgeyB3YWxsZXRJZDogJ3dhbGxldC1pZC0zJywgZW5jcnlwdGVkUHJ2OiAnZW5jcnlwdGVkLWRhdGEtMycgfSxcbiAgICAgICAgICB7IHdhbGxldElkOiAnd2FsbGV0LWlkLTQnLCBlbmNyeXB0ZWRQcnY6ICdlbmNyeXB0ZWQtZGF0YS00JyB9LFxuICAgICAgICBdLFxuICAgICAgICBwYXNzd29yZDogJ3Bhc3N3b3JkMTIzJyxcbiAgICAgIH0pO1xuXG4gICAgICAvLyBTaG91bGQgYmUgY2FsbGVkIG9uY2UgZm9yIGVhY2ggd2FsbGV0XG4gICAgICBkZWNyeXB0U3R1Yi5jYWxsQ291bnQuc2hvdWxkLmVxdWFsKDQpO1xuXG4gICAgICAvLyBTaG91bGQgaW5jbHVkZSBvbmx5IHRoZSBmYWlsZWQgd2FsbGV0IElEc1xuICAgICAgcmVzdWx0LnNob3VsZC5iZS5hbi5BcnJheSgpO1xuICAgICAgcmVzdWx0LnNob3VsZC5oYXZlLmxlbmd0aCgyKTtcbiAgICAgIHJlc3VsdC5zaG91bGQuY29udGFpbkRlZXAoWyd3YWxsZXQtaWQtMicsICd3YWxsZXQtaWQtNCddKTtcbiAgICB9KTtcbiAgfSk7XG5cbiAgZGVzY3JpYmUoJ1VzZXItQWdlbnQgaGVhZGVyIGJhc2VkIG9uIGVudmlyb25tZW50JywgZnVuY3Rpb24gKCkge1xuICAgIGFmdGVyRWFjaChmdW5jdGlvbiAoKSB7XG4gICAgICBub2NrLmNsZWFuQWxsKCk7XG4gICAgICBzaW5vbi5yZXN0b3JlKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHNldCBVc2VyLUFnZW50IGhlYWRlciB3aGVuIHJ1bm5pbmcgaW4gTm9kZS5qcyAodHlwZW9mIHdpbmRvdyA9PT0gdW5kZWZpbmVkKScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgZW52OiAnY3VzdG9tJyxcbiAgICAgICAgY3VzdG9tUm9vdFVSSTogJ2h0dHBzOi8vYXBwLmV4YW1wbGUubG9jYWwnLFxuICAgICAgICB1c2VyQWdlbnQ6ICdUZXN0QWdlbnQvMS4wJyxcbiAgICAgIH0pO1xuXG4gICAgICAvLyBFbnN1cmUgd2UncmUgaW4gYSBOb2RlLmpzIGVudmlyb25tZW50IGJ5IHZlcmlmeWluZyB3aW5kb3cgaXMgdW5kZWZpbmVkXG4gICAgICAodHlwZW9mIHdpbmRvdykuc2hvdWxkLmVxdWFsKCd1bmRlZmluZWQnKTtcblxuICAgICAgY29uc3Qgc2NvcGUgPSBub2NrKCdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJylcbiAgICAgICAgLmdldCgnL2FwaS92MS9waW5nJylcbiAgICAgICAgLm1hdGNoSGVhZGVyKCdVc2VyLUFnZW50JywgJ1Rlc3RBZ2VudC8xLjAnKVxuICAgICAgICAucmVwbHkoMjAwLCB7IHN0YXR1czogJ29rJyB9KTtcblxuICAgICAgYXdhaXQgYml0Z28ucGluZyh7XG4gICAgICAgIHJlcUlkOiB7XG4gICAgICAgICAgdG9TdHJpbmc6ICgpID0+ICd0ZXN0LTEyMycsXG4gICAgICAgICAgaW5jOiAoKSA9PiB7XG4gICAgICAgICAgICAvKiBtb2NrICovXG4gICAgICAgICAgfSxcbiAgICAgICAgfSBhcyBhbnksXG4gICAgICB9KTtcblxuICAgICAgc2NvcGUuaXNEb25lKCkuc2hvdWxkLmJlLnRydWUoKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgbm90IHNldCBVc2VyLUFnZW50IGhlYWRlciB3aGVuIHJ1bm5pbmcgaW4gYnJvd3NlciAodHlwZW9mIHdpbmRvdyAhPT0gdW5kZWZpbmVkKScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIC8vIE1vY2sgdGhlIHdpbmRvdyBvYmplY3QgdG8gc2ltdWxhdGUgYnJvd3NlciBlbnZpcm9ubWVudFxuICAgICAgY29uc3Qgd2luZG93U3R1YiA9IHsgbG9jYXRpb246ICdtb2NrJyB9O1xuICAgICAgKGdsb2JhbCBhcyBhbnkpLndpbmRvdyA9IHdpbmRvd1N0dWI7XG5cbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgICBlbnY6ICdjdXN0b20nLFxuICAgICAgICAgIGN1c3RvbVJvb3RVUkk6ICdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJyxcbiAgICAgICAgICB1c2VyQWdlbnQ6ICdUZXN0QWdlbnQvMS4wJyxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgY29uc3Qgc2NvcGUgPSBub2NrKCdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJylcbiAgICAgICAgICAuZ2V0KCcvYXBpL3YxL3BpbmcnKVxuICAgICAgICAgIC5yZXBseShmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgICAvLyBWZXJpZnkgVXNlci1BZ2VudCBoZWFkZXIgaXMgTk9UIHNldCB0byBvdXIgY3VzdG9tIHZhbHVlXG4gICAgICAgICAgICBjb25zdCB1c2VyQWdlbnQgPSB0aGlzLnJlcS5oZWFkZXJzWyd1c2VyLWFnZW50J107XG4gICAgICAgICAgICBpZiAodXNlckFnZW50ICYmIHVzZXJBZ2VudC5pbmNsdWRlcygnVGVzdEFnZW50LzEuMCcpKSB7XG4gICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcignVXNlci1BZ2VudCBzaG91bGQgbm90IGJlIHNldCBpbiBicm93c2VyIGVudmlyb25tZW50Jyk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gWzIwMCwgeyBzdGF0dXM6ICdvaycgfV07XG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgYXdhaXQgYml0Z28ucGluZyh7XG4gICAgICAgICAgcmVxSWQ6IHtcbiAgICAgICAgICAgIHRvU3RyaW5nOiAoKSA9PiAndGVzdC0xMjMnLFxuICAgICAgICAgICAgaW5jOiAoKSA9PiB7XG4gICAgICAgICAgICAgIC8qIG1vY2sgKi9cbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgfSBhcyBhbnksXG4gICAgICAgIH0pO1xuXG4gICAgICAgIHNjb3BlLmlzRG9uZSgpLnNob3VsZC5iZS50cnVlKCk7XG4gICAgICB9IGZpbmFsbHkge1xuICAgICAgICAvLyBDbGVhbiB1cCB0aGUgZ2xvYmFsIHdpbmRvdyBtb2NrXG4gICAgICAgIGRlbGV0ZSAoZ2xvYmFsIGFzIGFueSkud2luZG93O1xuICAgICAgfVxuICAgIH0pO1xuICB9KTtcblxuICBkZXNjcmliZSgnaG1hY0F1dGhTdHJhdGVneSB0b2tlbiBsaWZlY3ljbGUnLCBmdW5jdGlvbiAoKSB7XG4gICAgY29uc3QgUk9PVCA9ICdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJztcblxuICAgIC8vIEJ1aWxkcyBhIG1vY2sgc3RyYXRlZ3kgd2hvc2Ugc2V0VG9rZW4gLyBjbGVhclRva2VuIGFyZSBzaW5vbiBzdHVicy5cbiAgICBmdW5jdGlvbiBtYWtlU3RyYXRlZ3kob3ZlcnJpZGVzOiBQYXJ0aWFsPElIbWFjQXV0aFN0cmF0ZWd5PiA9IHt9KToge1xuICAgICAgc3RyYXRlZ3k6IElIbWFjQXV0aFN0cmF0ZWd5O1xuICAgICAgc2V0VG9rZW5TdHViOiBzaW5vbi5TaW5vblN0dWI7XG4gICAgICBjbGVhclRva2VuU3R1Yjogc2lub24uU2lub25TdHViO1xuICAgIH0ge1xuICAgICAgY29uc3Qgc2V0VG9rZW5TdHViID0gc2lub24uc3R1YigpLnJlc29sdmVzKCk7XG4gICAgICBjb25zdCBjbGVhclRva2VuU3R1YiA9IHNpbm9uLnN0dWIoKS5yZXNvbHZlcygpO1xuICAgICAgY29uc3Qgc3RyYXRlZ3k6IElIbWFjQXV0aFN0cmF0ZWd5ID0ge1xuICAgICAgICBjYWxjdWxhdGVSZXF1ZXN0SGVhZGVyczogc2lub24uc3R1YigpLnJlc29sdmVzKHsgaG1hYzogJ2htYWMnLCB0aW1lc3RhbXA6IDEsIHRva2VuSGFzaDogJ2hhc2gnIH0pLFxuICAgICAgICB2ZXJpZnlSZXNwb25zZTogc2lub24uc3R1YigpLnJlc29sdmVzKHtcbiAgICAgICAgICBpc1ZhbGlkOiB0cnVlLFxuICAgICAgICAgIGV4cGVjdGVkSG1hYzogJ2htYWMnLFxuICAgICAgICAgIHNpZ25hdHVyZVN1YmplY3Q6ICcnLFxuICAgICAgICAgIGlzSW5SZXNwb25zZVZhbGlkaXR5V2luZG93OiB0cnVlLFxuICAgICAgICAgIHZlcmlmaWNhdGlvblRpbWU6IERhdGUubm93KCksXG4gICAgICAgIH0pLFxuICAgICAgICBjYWxjdWxhdGVITUFDOiBzaW5vbi5zdHViKCkucmVzb2x2ZXMoJ2hhc2hlZC1wdycpLFxuICAgICAgICBzZXRUb2tlbjogc2V0VG9rZW5TdHViLFxuICAgICAgICBjbGVhclRva2VuOiBjbGVhclRva2VuU3R1YixcbiAgICAgICAgLi4ub3ZlcnJpZGVzLFxuICAgICAgfTtcbiAgICAgIHJldHVybiB7IHN0cmF0ZWd5LCBzZXRUb2tlblN0dWIsIGNsZWFyVG9rZW5TdHViIH07XG4gICAgfVxuXG4gICAgYWZ0ZXJFYWNoKGZ1bmN0aW9uICgpIHtcbiAgICAgIG5vY2suY2xlYW5BbGwoKTtcbiAgICAgIHNpbm9uLnJlc3RvcmUoKTtcbiAgICB9KTtcblxuICAgIGRlc2NyaWJlKCdhdXRoZW50aWNhdGUoKScsIGZ1bmN0aW9uICgpIHtcbiAgICAgIGl0KCdjYWxscyBzZXRUb2tlbiB3aXRoIHRoZSBhY2Nlc3NfdG9rZW4gcmVjZWl2ZWQgZnJvbSB0aGUgc2VydmVyJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCB7IHN0cmF0ZWd5LCBzZXRUb2tlblN0dWIgfSA9IG1ha2VTdHJhdGVneSgpO1xuICAgICAgICBjb25zdCBiaXRnbyA9IG5ldyBCaXRHb0FQSSh7IGVudjogJ2N1c3RvbScsIGN1c3RvbVJvb3RVUkk6IFJPT1QsIGhtYWNBdXRoU3RyYXRlZ3k6IHN0cmF0ZWd5IH0pO1xuXG4gICAgICAgIG5vY2soUk9PVClcbiAgICAgICAgICAucG9zdCgnL2FwaS9hdXRoL3YxL3Nlc3Npb24nKVxuICAgICAgICAgIC5yZXBseSgyMDAsIHtcbiAgICAgICAgICAgIHVzZXI6IHsgdXNlcm5hbWU6ICd0ZXN0QGV4YW1wbGUuY29tJyB9LFxuICAgICAgICAgICAgYWNjZXNzX3Rva2VuOiAndjJ4bXlhY2Nlc3N0b2tlbicsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgYXdhaXQgYml0Z28uYXV0aGVudGljYXRlKHsgdXNlcm5hbWU6ICd0ZXN0QGV4YW1wbGUuY29tJywgcGFzc3dvcmQ6ICdodW50ZXIyJyB9KTtcblxuICAgICAgICBzZXRUb2tlblN0dWIuY2FsbGVkT25jZS5zaG91bGQuYmUudHJ1ZSgpO1xuICAgICAgICBzZXRUb2tlblN0dWIuZmlyc3RDYWxsLmFyZ3NbMF0uc2hvdWxkLmVxdWFsKCd2MnhteWFjY2Vzc3Rva2VuJyk7XG4gICAgICB9KTtcblxuICAgICAgaXQoJ2F3YWl0cyBzZXRUb2tlbiBiZWZvcmUgbWFraW5nIGVuc3VyZUVjZGhLZXljaGFpbiByZXF1ZXN0cycsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgLy8gVGhpcyBpcyB0aGUgY29yZSByZWdyZXNzaW9uIHRlc3Q6IGlmIHNldFRva2VuIGlzIG5vdCBhd2FpdGVkLCB0aGVcbiAgICAgICAgLy8gc3RyYXRlZ3kncyBrZXkgbWF0ZXJpYWwgd29uJ3QgYmUgcmVhZHkgYmVmb3JlIGNhbGN1bGF0ZVJlcXVlc3RIZWFkZXJzXG4gICAgICAgIC8vIGlzIGNhbGxlZCBmb3IgdGhlIEdFVCAvdXNlci9zZXR0aW5ncyByZXF1ZXN0LCBhbmQgaXQgd291bGQgdGhyb3cuXG4gICAgICAgIGxldCBrZXlSZWFkeSA9IGZhbHNlO1xuICAgICAgICBjb25zdCB7IHN0cmF0ZWd5IH0gPSBtYWtlU3RyYXRlZ3koe1xuICAgICAgICAgIHNldFRva2VuOiBzaW5vbi5zdHViKCkuY2FsbHNGYWtlKGFzeW5jICgpID0+IHtcbiAgICAgICAgICAgIC8vIFNpbXVsYXRlIG5vbi10cml2aWFsIGFzeW5jIGtleSBkZXJpdmF0aW9uIChsaWtlIGNyeXB0by5zdWJ0bGUuaW1wb3J0S2V5KS5cbiAgICAgICAgICAgIGF3YWl0IG5ldyBQcm9taXNlPHZvaWQ+KChyZXNvbHZlKSA9PiBzZXRJbW1lZGlhdGUocmVzb2x2ZSkpO1xuICAgICAgICAgICAga2V5UmVhZHkgPSB0cnVlO1xuICAgICAgICAgIH0pLFxuICAgICAgICAgIGNhbGN1bGF0ZVJlcXVlc3RIZWFkZXJzOiBzaW5vbi5zdHViKCkuY2FsbHNGYWtlKGFzeW5jICgpID0+IHtcbiAgICAgICAgICAgIGlmICgha2V5UmVhZHkpIHtcbiAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdObyB0b2tlbiBhdmFpbGFibGUuIENhbGwgc2V0VG9rZW4oKSBvciByZXN0b3JlVG9rZW4oKSBmaXJzdC4nKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJldHVybiB7IGhtYWM6ICdobWFjJywgdGltZXN0YW1wOiBEYXRlLm5vdygpLCB0b2tlbkhhc2g6ICdoYXNoJyB9O1xuICAgICAgICAgIH0pLFxuICAgICAgICB9KTtcbiAgICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoeyBlbnY6ICdjdXN0b20nLCBjdXN0b21Sb290VVJJOiBST09ULCBobWFjQXV0aFN0cmF0ZWd5OiBzdHJhdGVneSB9KTtcblxuICAgICAgICBub2NrKFJPT1QpXG4gICAgICAgICAgLnBvc3QoJy9hcGkvYXV0aC92MS9zZXNzaW9uJylcbiAgICAgICAgICAucmVwbHkoMjAwLCB7XG4gICAgICAgICAgICB1c2VyOiB7IHVzZXJuYW1lOiAndGVzdEBleGFtcGxlLmNvbScgfSxcbiAgICAgICAgICAgIGFjY2Vzc190b2tlbjogJ3YyeG15dG9rZW4nLFxuICAgICAgICAgIH0pO1xuICAgICAgICAvLyBUaGUgR0VUIC91c2VyL3NldHRpbmdzIHJlcXVlc3QgbWFkZSBieSBlbnN1cmVVc2VyRWNkaEtleWNoYWluSXNDcmVhdGVkXG4gICAgICAgIC8vIG11c3Qgc3VjY2VlZCDigJQgaXQgd291bGQgdGhyb3cgaWYgc2V0VG9rZW4gd2Fzbid0IGF3YWl0ZWQgZmlyc3QuXG4gICAgICAgIG5vY2soUk9PVClcbiAgICAgICAgICAuZ2V0KCcvYXBpL3YxL3VzZXIvc2V0dGluZ3MnKVxuICAgICAgICAgIC5yZXBseSgyMDAsIHtcbiAgICAgICAgICAgIHNldHRpbmdzOiB7IGVjZGhLZXljaGFpbjogJ3hwdWIxMjMnIH0sXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgYXdhaXQgYml0Z28uYXV0aGVudGljYXRlKHtcbiAgICAgICAgICB1c2VybmFtZTogJ3Rlc3RAZXhhbXBsZS5jb20nLFxuICAgICAgICAgIHBhc3N3b3JkOiAnaHVudGVyMicsXG4gICAgICAgICAgZW5zdXJlRWNkaEtleWNoYWluOiB0cnVlLFxuICAgICAgICB9KTtcblxuICAgICAgICBrZXlSZWFkeS5zaG91bGQuYmUudHJ1ZSgpO1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBkZXNjcmliZSgnYXV0aGVudGljYXRlV2l0aFBhc3NrZXkoKScsIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHZhbGlkUGFzc2tleSA9IEpTT04uc3RyaW5naWZ5KHtcbiAgICAgICAgaWQ6ICdjcmVkZW50aWFsLWlkJyxcbiAgICAgICAgcmF3SWQ6ICdyYXctaWQnLFxuICAgICAgICB0eXBlOiAncHVibGljLWtleScsXG4gICAgICAgIHJlc3BvbnNlOiB7XG4gICAgICAgICAgYXV0aGVudGljYXRvckRhdGE6ICdhdXRoLWRhdGEnLFxuICAgICAgICAgIGNsaWVudERhdGFKU09OOiAnY2xpZW50LWRhdGEnLFxuICAgICAgICAgIHNpZ25hdHVyZTogJ3NpZycsXG4gICAgICAgICAgdXNlckhhbmRsZTogJ3VzZXItaGFuZGxlLTEyMycsXG4gICAgICAgIH0sXG4gICAgICB9KTtcblxuICAgICAgaXQoJ2NhbGxzIHNldFRva2VuIHdpdGggdGhlIGFjY2Vzc190b2tlbiByZWNlaXZlZCBmcm9tIHRoZSBzZXJ2ZXInLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHsgc3RyYXRlZ3ksIHNldFRva2VuU3R1YiB9ID0gbWFrZVN0cmF0ZWd5KCk7XG4gICAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHsgZW52OiAnY3VzdG9tJywgY3VzdG9tUm9vdFVSSTogUk9PVCwgaG1hY0F1dGhTdHJhdGVneTogc3RyYXRlZ3kgfSk7XG5cbiAgICAgICAgbm9jayhST09UKVxuICAgICAgICAgIC5wb3N0KCcvYXBpL2F1dGgvdjEvc2Vzc2lvbicpXG4gICAgICAgICAgLnJlcGx5KDIwMCwge1xuICAgICAgICAgICAgdXNlcjogeyB1c2VybmFtZTogJ3Rlc3RAZXhhbXBsZS5jb20nIH0sXG4gICAgICAgICAgICBhY2Nlc3NfdG9rZW46ICd2MnhwYXNza2V5dG9rZW4nLFxuICAgICAgICAgIH0pO1xuXG4gICAgICAgIGF3YWl0IGJpdGdvLmF1dGhlbnRpY2F0ZVdpdGhQYXNza2V5KHZhbGlkUGFzc2tleSk7XG5cbiAgICAgICAgc2V0VG9rZW5TdHViLmNhbGxlZE9uY2Uuc2hvdWxkLmJlLnRydWUoKTtcbiAgICAgICAgc2V0VG9rZW5TdHViLmZpcnN0Q2FsbC5hcmdzWzBdLnNob3VsZC5lcXVhbCgndjJ4cGFzc2tleXRva2VuJyk7XG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGRlc2NyaWJlKCdjbGVhckFzeW5jKCknLCBmdW5jdGlvbiAoKSB7XG4gICAgICBpdCgnY2xlYXJzIF90b2tlbiBhbmQgY2FsbHMgY2xlYXJUb2tlbiBvbiB0aGUgc3RyYXRlZ3knLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHsgc3RyYXRlZ3ksIGNsZWFyVG9rZW5TdHViIH0gPSBtYWtlU3RyYXRlZ3koKTtcbiAgICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoeyBlbnY6ICdjdXN0b20nLCBjdXN0b21Sb290VVJJOiBST09ULCBobWFjQXV0aFN0cmF0ZWd5OiBzdHJhdGVneSB9KTtcblxuICAgICAgICBiaXRnby5hdXRoZW50aWNhdGVXaXRoQWNjZXNzVG9rZW4oeyBhY2Nlc3NUb2tlbjogJ3YyeHNvbWV0b2tlbicgfSk7XG4gICAgICAgIChiaXRnbyBhcyBhbnkpLl90b2tlbi5zaG91bGQuZXF1YWwoJ3YyeHNvbWV0b2tlbicpO1xuXG4gICAgICAgIGF3YWl0IGJpdGdvLmNsZWFyQXN5bmMoKTtcblxuICAgICAgICAoKGJpdGdvIGFzIGFueSkuX3Rva2VuID09PSB1bmRlZmluZWQpLnNob3VsZC5iZS50cnVlKCk7XG4gICAgICAgIGNsZWFyVG9rZW5TdHViLmNhbGxlZE9uY2Uuc2hvdWxkLmJlLnRydWUoKTtcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgZGVzY3JpYmUoJ3JlZnJlc2hUb2tlbigpJywgZnVuY3Rpb24gKCkge1xuICAgICAgaXQoJ2NhbGxzIHNldFRva2VuIHdpdGggdGhlIG5ldyBhY2Nlc3NfdG9rZW4gZnJvbSB0aGUgT0F1dGggcmVzcG9uc2UnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICAgIGNvbnN0IHsgc3RyYXRlZ3ksIHNldFRva2VuU3R1YiB9ID0gbWFrZVN0cmF0ZWd5KCk7XG4gICAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgICBlbnY6ICdjdXN0b20nLFxuICAgICAgICAgIGN1c3RvbVJvb3RVUkk6IFJPT1QsXG4gICAgICAgICAgaG1hY0F1dGhTdHJhdGVneTogc3RyYXRlZ3ksXG4gICAgICAgICAgY2xpZW50SWQ6ICdjbGllbnQtaWQnLFxuICAgICAgICAgIGNsaWVudFNlY3JldDogJ2NsaWVudC1zZWNyZXQnLFxuICAgICAgICB9KTtcbiAgICAgICAgKGJpdGdvIGFzIGFueSkuX3JlZnJlc2hUb2tlbiA9ICdvbGQtcmVmcmVzaC10b2tlbic7XG5cbiAgICAgICAgbm9jayhST09UKS5wb3N0KCcvb2F1dGgvdG9rZW4nKS5yZXBseSgyMDAsIHtcbiAgICAgICAgICBhY2Nlc3NfdG9rZW46ICd2MnhuZXd0b2tlbicsXG4gICAgICAgICAgcmVmcmVzaF90b2tlbjogJ25ldy1yZWZyZXNoLXRva2VuJyxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgYXdhaXQgYml0Z28ucmVmcmVzaFRva2VuKCk7XG5cbiAgICAgICAgc2V0VG9rZW5TdHViLmNhbGxlZE9uY2Uuc2hvdWxkLmJlLnRydWUoKTtcbiAgICAgICAgc2V0VG9rZW5TdHViLmZpcnN0Q2FsbC5hcmdzWzBdLnNob3VsZC5lcXVhbCgndjJ4bmV3dG9rZW4nKTtcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgZGVzY3JpYmUoJ2F1dGhlbnRpY2F0ZVdpdGhBdXRoQ29kZSgpJywgZnVuY3Rpb24gKCkge1xuICAgICAgaXQoJ2NhbGxzIHNldFRva2VuIHdpdGggdGhlIGFjY2Vzc190b2tlbiBmcm9tIHRoZSBPQXV0aCByZXNwb25zZScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgY29uc3QgeyBzdHJhdGVneSwgc2V0VG9rZW5TdHViIH0gPSBtYWtlU3RyYXRlZ3koKTtcbiAgICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICAgIGVudjogJ2N1c3RvbScsXG4gICAgICAgICAgY3VzdG9tUm9vdFVSSTogUk9PVCxcbiAgICAgICAgICBobWFjQXV0aFN0cmF0ZWd5OiBzdHJhdGVneSxcbiAgICAgICAgICBjbGllbnRJZDogJ2NsaWVudC1pZCcsXG4gICAgICAgICAgY2xpZW50U2VjcmV0OiAnY2xpZW50LXNlY3JldCcsXG4gICAgICAgIH0pO1xuXG4gICAgICAgIG5vY2soUk9PVCkucG9zdCgnL29hdXRoL3Rva2VuJykucmVwbHkoMjAwLCB7XG4gICAgICAgICAgYWNjZXNzX3Rva2VuOiAndjJ4YXV0aGNvZGV0b2tlbicsXG4gICAgICAgICAgcmVmcmVzaF90b2tlbjogJ3JlZnJlc2gtdG9rZW4nLFxuICAgICAgICB9KTtcbiAgICAgICAgLy8gYXV0aGVudGljYXRlV2l0aEF1dGhDb2RlIGNhbGxzIHRoaXMubWUoKSBhZnRlciBzZXR0aW5nIHRoZSB0b2tlblxuICAgICAgICBub2NrKFJPT1QpXG4gICAgICAgICAgLmdldCgnL2FwaS92MS91c2VyL21lJylcbiAgICAgICAgICAucmVwbHkoMjAwLCB7XG4gICAgICAgICAgICB1c2VyOiB7IHVzZXJuYW1lOiAndGVzdEBleGFtcGxlLmNvbScgfSxcbiAgICAgICAgICB9KTtcblxuICAgICAgICBhd2FpdCBiaXRnby5hdXRoZW50aWNhdGVXaXRoQXV0aENvZGUoeyBhdXRoQ29kZTogJ215LWF1dGgtY29kZScgfSk7XG5cbiAgICAgICAgc2V0VG9rZW5TdHViLmNhbGxlZE9uY2Uuc2hvdWxkLmJlLnRydWUoKTtcbiAgICAgICAgc2V0VG9rZW5TdHViLmZpcnN0Q2FsbC5hcmdzWzBdLnNob3VsZC5lcXVhbCgndjJ4YXV0aGNvZGV0b2tlbicpO1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBkZXNjcmliZSgnc3luYyB0b2tlbi1zZXR0aW5nIG1ldGhvZHMnLCBmdW5jdGlvbiAoKSB7XG4gICAgICBpdCgnYXV0aGVudGljYXRlV2l0aEFjY2Vzc1Rva2VuIGRvZXMgbm90IGNhbGwgc2V0VG9rZW4gKHN5bmNocm9ub3VzIOKAlCBjYWxsZXIgbXVzdCBpbnZva2Ugc2V0VG9rZW4gb24gdGhlIHN0cmF0ZWd5IG1hbnVhbGx5KScsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgY29uc3QgeyBzdHJhdGVneSwgc2V0VG9rZW5TdHViIH0gPSBtYWtlU3RyYXRlZ3koKTtcbiAgICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoeyBlbnY6ICdjdXN0b20nLCBjdXN0b21Sb290VVJJOiBST09ULCBobWFjQXV0aFN0cmF0ZWd5OiBzdHJhdGVneSB9KTtcblxuICAgICAgICBiaXRnby5hdXRoZW50aWNhdGVXaXRoQWNjZXNzVG9rZW4oeyBhY2Nlc3NUb2tlbjogJ3YyeHN5bmN0b2tlbicgfSk7XG5cbiAgICAgICAgc2V0VG9rZW5TdHViLmNhbGxlZC5zaG91bGQuYmUuZmFsc2UoKTtcbiAgICAgIH0pO1xuXG4gICAgICBpdCgnZnJvbUpTT04gZG9lcyBub3QgY2FsbCBzZXRUb2tlbiAoc3luY2hyb25vdXMg4oCUIGNhbGxlciBtdXN0IGludm9rZSBzZXRUb2tlbiBvbiB0aGUgc3RyYXRlZ3kgbWFudWFsbHkpJywgZnVuY3Rpb24gKCkge1xuICAgICAgICBjb25zdCB7IHN0cmF0ZWd5LCBzZXRUb2tlblN0dWIgfSA9IG1ha2VTdHJhdGVneSgpO1xuICAgICAgICBjb25zdCBiaXRnbyA9IG5ldyBCaXRHb0FQSSh7IGVudjogJ2N1c3RvbScsIGN1c3RvbVJvb3RVUkk6IFJPT1QsIGhtYWNBdXRoU3RyYXRlZ3k6IHN0cmF0ZWd5IH0pO1xuXG4gICAgICAgIChiaXRnbyBhcyBhbnkpLmZyb21KU09OKHsgdXNlcjogeyB1c2VybmFtZTogJ3Rlc3RAZXhhbXBsZS5jb20nIH0sIHRva2VuOiAndjJ4anNvbnRva2VuJyB9KTtcblxuICAgICAgICBzZXRUb2tlblN0dWIuY2FsbGVkLnNob3VsZC5iZS5mYWxzZSgpO1xuICAgICAgfSk7XG4gICAgfSk7XG4gIH0pO1xuXG4gIGRlc2NyaWJlKCdjb25zdGFudHMgcGFyYW1ldGVyJywgZnVuY3Rpb24gKCkge1xuICAgIGl0KCdzaG91bGQgYWxsb3cgcGFzc2luZyBjb25zdGFudHMgdmlhIG9wdGlvbnMgYW5kIGV4cG9zZSB2aWEgZmV0Y2hDb25zdGFudHMnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCBiaXRnbyA9IG5ldyBCaXRHb0FQSSh7XG4gICAgICAgIGVudjogJ2N1c3RvbScsXG4gICAgICAgIGN1c3RvbVJvb3RVUkk6ICdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJyxcbiAgICAgICAgY2xpZW50Q29uc3RhbnRzOiB7IG1heEZlZVJhdGU6ICcxMjMxMjMxMjMxMjMxMjMnIH0sXG4gICAgICB9KTtcblxuICAgICAgY29uc3QgY29uc3RhbnRzID0gYXdhaXQgYml0Z28uZmV0Y2hDb25zdGFudHMoKTtcbiAgICAgIGNvbnN0YW50cy5zaG91bGQuaGF2ZS5wcm9wZXJ0eSgnbWF4RmVlUmF0ZScsICcxMjMxMjMxMjMxMjMxMjMnKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgcmVmcmVzaCBjb25zdGFudHMgd2hlbiBjYWNoZSBoYXMgZXhwaXJlZCcsIGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IGJpdGdvID0gbmV3IEJpdEdvQVBJKHtcbiAgICAgICAgZW52OiAnY3VzdG9tJyxcbiAgICAgICAgY3VzdG9tUm9vdFVSSTogJ2h0dHBzOi8vYXBwLmV4YW1wbGUubG9jYWwnLFxuICAgICAgfSk7XG5cbiAgICAgIC8vIFNldCB1cCBjYWNoZWQgY29uc3RhbnRzIHdpdGggYW4gZXhwaXJlZCBjYWNoZVxuICAgICAgKEJpdEdvQVBJIGFzIGFueSkuX2NvbnN0YW50cyA9IChCaXRHb0FQSSBhcyBhbnkpLl9jb25zdGFudHMgfHwge307XG4gICAgICAoQml0R29BUEkgYXMgYW55KS5fY29uc3RhbnRzRXhwaXJlID0gKEJpdEdvQVBJIGFzIGFueSkuX2NvbnN0YW50c0V4cGlyZSB8fCB7fTtcbiAgICAgIChCaXRHb0FQSSBhcyBhbnkpLl9jb25zdGFudHNbJ2N1c3RvbSddID0geyBtYXhGZWVSYXRlOiAnb2xkLXZhbHVlJyB9O1xuICAgICAgKEJpdEdvQVBJIGFzIGFueSkuX2NvbnN0YW50c0V4cGlyZVsnY3VzdG9tJ10gPSBuZXcgRGF0ZShEYXRlLm5vdygpIC0gMTAwMCk7IC8vIEV4cGlyZWQgMSBzZWNvbmQgYWdvXG5cbiAgICAgIGNvbnN0IHNjb3BlID0gbm9jaygnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcpXG4gICAgICAgIC5nZXQoJy9hcGkvdjEvY2xpZW50L2NvbnN0YW50cycpXG4gICAgICAgIC5yZXBseSgyMDAsIHtcbiAgICAgICAgICBjb25zdGFudHM6IHsgbWF4RmVlUmF0ZTogJ25ldy12YWx1ZScsIG5ld0NvbnN0YW50OiAnYWRkZWQnIH0sXG4gICAgICAgIH0pO1xuXG4gICAgICBjb25zdCBjb25zdGFudHMgPSBhd2FpdCBiaXRnby5mZXRjaENvbnN0YW50cygpO1xuXG4gICAgICAvLyBTaG91bGQgcmV0dXJuIHRoZSBuZXcgY29uc3RhbnRzIGZyb20gdGhlIHNlcnZlclxuICAgICAgY29uc3RhbnRzLnNob3VsZC5oYXZlLnByb3BlcnR5KCdtYXhGZWVSYXRlJywgJ25ldy12YWx1ZScpO1xuICAgICAgY29uc3RhbnRzLnNob3VsZC5oYXZlLnByb3BlcnR5KCduZXdDb25zdGFudCcsICdhZGRlZCcpO1xuXG4gICAgICBzY29wZS5pc0RvbmUoKS5zaG91bGQuYmUudHJ1ZSgpO1xuXG4gICAgICBub2NrLmNsZWFuQWxsKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHVzZSBjYWNoZWQgY29uc3RhbnRzIHdoZW4gY2FjaGUgaXMgc3RpbGwgdmFsaWQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCBiaXRnbyA9IG5ldyBCaXRHb0FQSSh7XG4gICAgICAgIGVudjogJ2N1c3RvbScsXG4gICAgICAgIGN1c3RvbVJvb3RVUkk6ICdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJyxcbiAgICAgIH0pO1xuXG4gICAgICAvLyBTZXQgdXAgY2FjaGVkIGNvbnN0YW50cyB3aXRoIGEgZnV0dXJlIGV4cGlyeVxuICAgICAgY29uc3QgY2FjaGVkQ29uc3RhbnRzID0geyBtYXhGZWVSYXRlOiAnY2FjaGVkLXZhbHVlJywgYW5vdGhlclNldHRpbmc6ICdjYWNoZWQtc2V0dGluZycgfTtcbiAgICAgIChCaXRHb0FQSSBhcyBhbnkpLl9jb25zdGFudHMgPSAoQml0R29BUEkgYXMgYW55KS5fY29uc3RhbnRzIHx8IHt9O1xuICAgICAgKEJpdEdvQVBJIGFzIGFueSkuX2NvbnN0YW50c0V4cGlyZSA9IChCaXRHb0FQSSBhcyBhbnkpLl9jb25zdGFudHNFeHBpcmUgfHwge307XG4gICAgICAoQml0R29BUEkgYXMgYW55KS5fY29uc3RhbnRzWydjdXN0b20nXSA9IGNhY2hlZENvbnN0YW50cztcbiAgICAgIChCaXRHb0FQSSBhcyBhbnkpLl9jb25zdGFudHNFeHBpcmVbJ2N1c3RvbSddID0gbmV3IERhdGUoRGF0ZS5ub3coKSArIDUgKiA2MCAqIDEwMDApOyAvLyBWYWxpZCBmb3IgNSBtb3JlIG1pbnV0ZXNcblxuICAgICAgY29uc3Qgc2NvcGUgPSBub2NrKCdodHRwczovL2FwcC5leGFtcGxlLmxvY2FsJylcbiAgICAgICAgLmdldCgnL2FwaS92MS9jbGllbnQvY29uc3RhbnRzJylcbiAgICAgICAgLnJlcGx5KDIwMCwgeyBjb25zdGFudHM6IHsgc2hvdWxkTm90QmVVc2VkOiB0cnVlIH0gfSk7XG5cbiAgICAgIGNvbnN0IGNvbnN0YW50cyA9IGF3YWl0IGJpdGdvLmZldGNoQ29uc3RhbnRzKCk7XG5cbiAgICAgIC8vIFNob3VsZCByZXR1cm4gdGhlIGNhY2hlZCBjb25zdGFudHNcbiAgICAgIGNvbnN0YW50cy5zaG91bGQuZGVlcEVxdWFsKGNhY2hlZENvbnN0YW50cyk7XG5cbiAgICAgIC8vIFZlcmlmeSB0aGF0IG5vIEhUVFAgcmVxdWVzdCB3YXMgbWFkZSAoc2luY2UgY2FjaGUgd2FzIHZhbGlkKVxuICAgICAgc2NvcGUuaXNEb25lKCkuc2hvdWxkLmJlLmZhbHNlKCk7XG5cbiAgICAgIG5vY2suY2xlYW5BbGwoKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdXNlIGNhY2hlZCBjb25zdGFudHMgd2hlbiBubyBjYWNoZSBleHBpcnkgaXMgc2V0JywgYXN5bmMgZnVuY3Rpb24gKCkge1xuICAgICAgY29uc3QgYml0Z28gPSBuZXcgQml0R29BUEkoe1xuICAgICAgICBlbnY6ICdjdXN0b20nLFxuICAgICAgICBjdXN0b21Sb290VVJJOiAnaHR0cHM6Ly9hcHAuZXhhbXBsZS5sb2NhbCcsXG4gICAgICB9KTtcblxuICAgICAgLy8gU2V0IHVwIGNhY2hlZCBjb25zdGFudHMgd2l0aCBubyBleHBpcnlcbiAgICAgIGNvbnN0IGNhY2hlZENvbnN0YW50cyA9IHsgbWF4RmVlUmF0ZTogJ25vLWV4cGlyeS12YWx1ZScgfTtcbiAgICAgIChCaXRHb0FQSSBhcyBhbnkpLl9jb25zdGFudHMgPSAoQml0R29BUEkgYXMgYW55KS5fY29uc3RhbnRzIHx8IHt9O1xuICAgICAgKEJpdEdvQVBJIGFzIGFueSkuX2NvbnN0YW50c0V4cGlyZSA9IChCaXRHb0FQSSBhcyBhbnkpLl9jb25zdGFudHNFeHBpcmUgfHwge307XG4gICAgICAoQml0R29BUEkgYXMgYW55KS5fY29uc3RhbnRzWydjdXN0b20nXSA9IGNhY2hlZENvbnN0YW50cztcbiAgICAgIChCaXRHb0FQSSBhcyBhbnkpLl9jb25zdGFudHNFeHBpcmVbJ2N1c3RvbSddID0gdW5kZWZpbmVkO1xuXG4gICAgICBjb25zdCBzY29wZSA9IG5vY2soJ2h0dHBzOi8vYXBwLmV4YW1wbGUubG9jYWwnKVxuICAgICAgICAuZ2V0KCcvYXBpL3YxL2NsaWVudC9jb25zdGFudHMnKVxuICAgICAgICAucmVwbHkoMjAwLCB7IGNvbnN0YW50czogeyBzaG91bGROb3RCZVVzZWQ6IHRydWUgfSB9KTtcblxuICAgICAgY29uc3QgY29uc3RhbnRzID0gYXdhaXQgYml0Z28uZmV0Y2hDb25zdGFudHMoKTtcblxuICAgICAgLy8gU2hvdWxkIHJldHVybiB0aGUgY2FjaGVkIGNvbnN0YW50c1xuICAgICAgY29uc3RhbnRzLnNob3VsZC5kZWVwRXF1YWwoY2FjaGVkQ29uc3RhbnRzKTtcblxuICAgICAgLy8gVmVyaWZ5IHRoYXQgbm8gSFRUUCByZXF1ZXN0IHdhcyBtYWRlIChzaW5jZSBubyBleHBpcnkgbWVhbnMgY2FjaGUgaXMgYWx3YXlzIHZhbGlkKVxuICAgICAgc2NvcGUuaXNEb25lKCkuc2hvdWxkLmJlLmZhbHNlKCk7XG5cbiAgICAgIG5vY2suY2xlYW5BbGwoKTtcbiAgICB9KTtcbiAgfSk7XG59KTtcbiJdfQ==
|