@rudderstack/integrations-lib 0.2.18 → 0.2.19

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.
@@ -1,520 +0,0 @@
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 (mod) {
19
- if (mod && mod.__esModule) return mod;
20
- var result = {};
21
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
- __setModuleDefault(result, mod);
23
- return result;
24
- };
25
- var __importDefault = (this && this.__importDefault) || function (mod) {
26
- return (mod && mod.__esModule) ? mod : { "default": mod };
27
- };
28
- Object.defineProperty(exports, "__esModule", { value: true });
29
- exports.JsonUtils = void 0;
30
- const handlebars_1 = __importDefault(require("handlebars"));
31
- const get_value_1 = __importDefault(require("get-value"));
32
- const set_value_1 = __importDefault(require("set-value"));
33
- const lodash_1 = require("lodash");
34
- const moment_timezone_1 = __importDefault(require("moment-timezone"));
35
- const logger = __importStar(require("../logger"));
36
- const platform_error_1 = require("../errors/platform_error");
37
- const sem_1 = require("./sem");
38
- const instrumentation_error_1 = require("../errors/instrumentation_error");
39
- const misc_1 = require("./misc");
40
- const transformation_error_1 = require("../errors/transformation_error");
41
- const constants_1 = require("../constants");
42
- class JsonUtils {
43
- /**
44
- * Description: This method is used to handle the source keys operation
45
- * @param message - the message object
46
- * @returns {any} - returns the value of the source key
47
- * if the source key is not present in the message
48
- * returns null
49
- *
50
- */
51
- static handleSourceKeysOperation({ message, operationObject, }) {
52
- const { operation, args } = operationObject;
53
- // populate the values from the arguments
54
- // in the same order it is populated
55
- const argValues = args.map((arg) => {
56
- const { sourceKeys, defaultVal } = arg;
57
- if (!Array.isArray(sourceKeys) && typeof sourceKeys !== 'string') {
58
- return defaultVal;
59
- }
60
- let val;
61
- let foundKey = sourceKeys;
62
- if (Array.isArray(sourceKeys)) {
63
- foundKey = sourceKeys.find((srcKey) => (0, get_value_1.default)(message, srcKey));
64
- }
65
- if (foundKey) {
66
- val = (0, get_value_1.default)(message, foundKey);
67
- }
68
- if (val || val === false || val === 0) {
69
- return val;
70
- }
71
- return defaultVal;
72
- });
73
- // quick sanity check for the undefined values in the list.
74
- // if there is any undefined values, return null
75
- // without going further for operations
76
- const isAllDefined = (0, lodash_1.every)(argValues, (v) => !(0, lodash_1.isNil)(v));
77
- if (!isAllDefined) {
78
- return null;
79
- }
80
- // start handling operations
81
- let result;
82
- switch (operation) {
83
- case 'multiplication':
84
- result = 1;
85
- // eslint-disable-next-line no-restricted-syntax
86
- for (const v of argValues) {
87
- if ((0, lodash_1.isNumber)(v)) {
88
- result *= v;
89
- }
90
- else {
91
- // if there is a non number argument simply return null
92
- // non numbers can't be operated arithmatically
93
- return null;
94
- }
95
- }
96
- return result;
97
- case 'addition':
98
- result = 0;
99
- // eslint-disable-next-line no-restricted-syntax
100
- for (const v of argValues) {
101
- if ((0, lodash_1.isNumber)(v)) {
102
- result += v;
103
- }
104
- else {
105
- // if there is a non number argument simply return null
106
- // non numbers can't be operated arithmatically
107
- return null;
108
- }
109
- }
110
- return result;
111
- default:
112
- return null;
113
- }
114
- }
115
- static checkTimestamp(validateTimestamp, formattedVal) {
116
- const { allowedPastTimeDifference, allowedPastTimeUnit, // seconds, minutes, hours
117
- allowedFutureTimeDifference, allowedFutureTimeUnit, // seconds, minutes, hours
118
- } = validateTimestamp;
119
- let pastTimeDifference;
120
- let futureTimeDifference;
121
- const currentTime = moment_timezone_1.default.unix(Number((0, moment_timezone_1.default)().format('X')));
122
- const time = moment_timezone_1.default.unix(Number((0, moment_timezone_1.default)(formattedVal).format('X')));
123
- switch (allowedPastTimeUnit) {
124
- case 'seconds':
125
- pastTimeDifference = Math.ceil(moment_timezone_1.default.duration(currentTime.diff(time)).asSeconds());
126
- break;
127
- case 'minutes':
128
- pastTimeDifference = Math.ceil(moment_timezone_1.default.duration(currentTime.diff(time)).asMinutes());
129
- break;
130
- case 'hours':
131
- pastTimeDifference = Math.ceil(moment_timezone_1.default.duration(currentTime.diff(time)).asHours());
132
- break;
133
- default:
134
- break;
135
- }
136
- if (pastTimeDifference > allowedPastTimeDifference) {
137
- throw new instrumentation_error_1.InstrumentationError(`Allowed timestamp is [${allowedPastTimeDifference} ${allowedPastTimeUnit}] into the past`);
138
- }
139
- if (!pastTimeDifference || pastTimeDifference <= 0) {
140
- switch (allowedFutureTimeUnit) {
141
- case 'seconds':
142
- futureTimeDifference = Math.ceil(moment_timezone_1.default.duration(time.diff(currentTime)).asSeconds());
143
- break;
144
- case 'minutes':
145
- futureTimeDifference = Math.ceil(moment_timezone_1.default.duration(time.diff(currentTime)).asMinutes());
146
- break;
147
- case 'hours':
148
- futureTimeDifference = Math.ceil(moment_timezone_1.default.duration(time.diff(currentTime)).asHours());
149
- break;
150
- default:
151
- break;
152
- }
153
- if (futureTimeDifference > allowedFutureTimeDifference) {
154
- throw new instrumentation_error_1.InstrumentationError(`Allowed timestamp is [${allowedFutureTimeDifference} ${allowedFutureTimeUnit}] into the future`);
155
- }
156
- }
157
- }
158
- static formatValues(formattedVal, formattingType, typeFormat, integrationsObj) {
159
- let curFormattedVal = formattedVal;
160
- const formattingFunctions = {
161
- timestamp: () => {
162
- curFormattedVal = (0, misc_1.formatTimeStamp)(String(formattedVal), typeFormat);
163
- },
164
- secondTimestamp: () => {
165
- if (!(0, moment_timezone_1.default)(String(formattedVal), 'x', true).isValid()) {
166
- curFormattedVal = Math.floor((0, misc_1.formatTimeStamp)(String(formattedVal), typeFormat) / 1000);
167
- }
168
- },
169
- microSecondTimestamp: () => {
170
- const temp_val = moment_timezone_1.default.unix(Number((0, moment_timezone_1.default)(String(formattedVal)).format('X')));
171
- curFormattedVal = temp_val.toDate().getTime() * 1000 + temp_val.toDate().getMilliseconds();
172
- },
173
- flatJson: () => {
174
- curFormattedVal = (0, misc_1.flattenJson)(formattedVal);
175
- },
176
- encodeURIComponent: () => {
177
- const t_val = formattedVal;
178
- curFormattedVal = encodeURIComponent(t_val);
179
- },
180
- jsonStringify: () => {
181
- curFormattedVal = JSON.stringify(formattedVal);
182
- },
183
- jsonStringifyOnFlatten: () => {
184
- curFormattedVal = JSON.stringify((0, misc_1.flattenJson)(formattedVal));
185
- },
186
- dobInMMDD: () => {
187
- curFormattedVal = String(formattedVal).slice(5);
188
- curFormattedVal = String(curFormattedVal).replace('-', '/');
189
- },
190
- jsonStringifyOnObject: () => {
191
- if (typeof formattedVal !== 'string') {
192
- curFormattedVal = JSON.stringify(formattedVal);
193
- }
194
- },
195
- numberForRevenue: () => {
196
- if ((typeof formattedVal === 'string' || formattedVal instanceof String) &&
197
- formattedVal.charAt(0) === '$') {
198
- curFormattedVal = formattedVal.substring(1);
199
- }
200
- curFormattedVal = Number.parseFloat(Number(curFormattedVal || 0).toFixed(2));
201
- if (Number.isNaN(curFormattedVal)) {
202
- throw new instrumentation_error_1.InstrumentationError('Revenue is not in the correct format');
203
- }
204
- },
205
- toString: () => {
206
- curFormattedVal = String(formattedVal);
207
- },
208
- toNumber: () => {
209
- curFormattedVal = Number(formattedVal);
210
- },
211
- toFloat: () => {
212
- curFormattedVal = parseFloat(String(formattedVal));
213
- },
214
- toInt: () => {
215
- curFormattedVal = parseInt(String(formattedVal), 10);
216
- },
217
- toLower: () => {
218
- curFormattedVal = String(formattedVal).toLowerCase();
219
- },
220
- hashToSha256: () => {
221
- curFormattedVal = integrationsObj?.hashed
222
- ? String(formattedVal)
223
- : (0, misc_1.hashToSha256)(String(formattedVal));
224
- },
225
- getOffsetInSec: () => {
226
- curFormattedVal = (0, misc_1.getOffsetInSec)(String(formattedVal));
227
- },
228
- domainUrl: () => {
229
- curFormattedVal = String(formattedVal).replace('https://', '').replace('http://', '');
230
- },
231
- domainUrlV2: () => {
232
- const url = (0, misc_1.isValidUrl)(formattedVal);
233
- if (!url) {
234
- throw new instrumentation_error_1.InstrumentationError(`Invalid URL: ${formattedVal}`);
235
- }
236
- curFormattedVal = url.hostname.replace('www.', '');
237
- },
238
- IsBoolean: () => {
239
- curFormattedVal = true;
240
- if (!(typeof formattedVal === 'boolean')) {
241
- logger.debug('Boolean value missing, so dropping it');
242
- curFormattedVal = false;
243
- }
244
- },
245
- trim: () => {
246
- if (typeof formattedVal === 'string') {
247
- curFormattedVal = formattedVal.trim();
248
- }
249
- },
250
- };
251
- if (formattingType in formattingFunctions) {
252
- const formattingFunction = formattingFunctions[formattingType];
253
- formattingFunction();
254
- }
255
- return curFormattedVal;
256
- }
257
- static handleExcludes(value, excludes, formattedVal) {
258
- if (typeof value === 'object') {
259
- // exclude the fields from the formattedVal
260
- excludes.forEach((key) => {
261
- // eslint-disable-next-line no-param-reassign
262
- delete formattedVal[key];
263
- });
264
- }
265
- else {
266
- logger.warn("excludes doesn't work with non-object data type. Ignoring excludes");
267
- }
268
- }
269
- // format the value as per the metadata values
270
- // Expected metadata keys are: (according to precedence)
271
- // - - type, typeFormat: expected data type and its format
272
- // - - template : need to have a handlebar expression {{value}}
273
- // - - excludes : fields you want to strip of from the final value (works only for object)
274
- // - - - - ex: "anonymousId", "userId" from traits
275
- static handleMetadataForValue(value, metadata, destKey, integrationsObj) {
276
- if (!metadata) {
277
- return value;
278
- }
279
- // get information from metadata
280
- const { type, typeFormat, template, defaultValue, excludes, multikeyMap, strictMultiMap, validateTimestamp, allowedKeyCheck, } = metadata;
281
- // if value is null and defaultValue is supplied - use that
282
- if (!(0, misc_1.isDefinedAndNotNull)(value)) {
283
- return defaultValue || value;
284
- }
285
- // we've got a correct value. start processing
286
- let formattedVal = value;
287
- /**
288
- * validate allowed time difference
289
- */
290
- if (validateTimestamp) {
291
- formattedVal = Number(formattedVal);
292
- if (Number.isNaN(formattedVal)) {
293
- throw new transformation_error_1.TransformationError('Timestamp is not a number, cannot validate');
294
- }
295
- JsonUtils.checkTimestamp(validateTimestamp, formattedVal);
296
- }
297
- if (type) {
298
- if (Array.isArray(type)) {
299
- type.forEach((eachType) => {
300
- formattedVal = JsonUtils.formatValues(formattedVal, eachType, typeFormat, integrationsObj);
301
- });
302
- }
303
- else {
304
- formattedVal = JsonUtils.formatValues(formattedVal, type, typeFormat, integrationsObj);
305
- }
306
- }
307
- // handle template
308
- if (template) {
309
- formattedVal = JsonUtils.handleTemplate(template, value);
310
- }
311
- // handle excludes
312
- if (excludes) {
313
- JsonUtils.handleExcludes(value, excludes, formattedVal);
314
- }
315
- // handle multikeyMap
316
- if (multikeyMap || strictMultiMap) {
317
- formattedVal = JsonUtils.handleMultikeyMap(multikeyMap, strictMultiMap, formattedVal, destKey);
318
- }
319
- if (allowedKeyCheck) {
320
- let foundVal = false;
321
- if (Array.isArray(allowedKeyCheck)) {
322
- allowedKeyCheck.some((key) => {
323
- switch (key.type) {
324
- case 'toLowerCase':
325
- formattedVal = String(formattedVal).toLowerCase();
326
- break;
327
- case 'toUpperCase':
328
- formattedVal = String(formattedVal).toUpperCase();
329
- break;
330
- default:
331
- break;
332
- }
333
- if (key.sourceVal.includes(formattedVal)) {
334
- foundVal = true;
335
- return true;
336
- }
337
- return false;
338
- });
339
- }
340
- if (!foundVal) {
341
- formattedVal = undefined;
342
- }
343
- }
344
- return formattedVal;
345
- }
346
- static handleTemplate(template, value) {
347
- const hTemplate = handlebars_1.default.compile(template.trim());
348
- const formattedVal = hTemplate({ value }).trim();
349
- return formattedVal;
350
- }
351
- static handleMultikeyMap(multikeyMap, strictMultiMap, formattedVal, destKey) {
352
- // sourceVal is expected to be an array
353
- // if value is present in sourceVal, returns the destVal
354
- // else returns the original value
355
- // Expected multikeyMap value:
356
- // "multikeyMap": [
357
- // {
358
- // "sourceVal": ["m", "M", "Male", "male"],
359
- // "destVal": "M"
360
- // },
361
- // {
362
- // "sourceVal": ["f", "F", "Female", "female"],
363
- // "destVal": "F"
364
- // }
365
- // ]
366
- let curFormattedVal = formattedVal;
367
- const finalKeyMap = multikeyMap || strictMultiMap;
368
- let foundVal = false;
369
- if (Array.isArray(finalKeyMap)) {
370
- finalKeyMap.some((map) => {
371
- if (!map.sourceVal || !(0, misc_1.isDefinedAndNotNull)(map.destVal) || !Array.isArray(map.sourceVal)) {
372
- logger.warn('multikeyMap skipped: sourceVal and destVal must be of valid type');
373
- foundVal = true;
374
- return true;
375
- }
376
- if (map.sourceVal.includes(String(formattedVal))) {
377
- curFormattedVal = map.destVal;
378
- foundVal = true;
379
- return true;
380
- }
381
- return false;
382
- });
383
- }
384
- else {
385
- logger.warn('multikeyMap skipped: multikeyMap must be an array');
386
- }
387
- if (!foundVal) {
388
- if (strictMultiMap) {
389
- throw new instrumentation_error_1.InstrumentationError(`Invalid entry for key ${destKey}`);
390
- }
391
- else {
392
- curFormattedVal = undefined;
393
- }
394
- }
395
- return curFormattedVal;
396
- }
397
- /**
398
- * Description: This method is used to get the value from the message
399
- * based on the source keys
400
- * @param message - the message object
401
- * @param sourceKeys - the source keys to be used to get the value from the message
402
- * @returns {any} - returns the value of the source key
403
- * if the source key is not present in the message
404
- * returns null
405
- */
406
- static getValueFromMessage(message, sourceKeys) {
407
- if (Array.isArray(sourceKeys) && sourceKeys.length > 0) {
408
- if (sourceKeys.length === 1) {
409
- logger.warn('List with single element is not ideal. Use it as string instead');
410
- }
411
- // got the possible sourceKeys
412
- // eslint-disable-next-line no-restricted-syntax
413
- for (const sourceKey of sourceKeys) {
414
- let val;
415
- // if the sourceKey is an object we expect it to be a operation
416
- if (typeof sourceKey === 'object') {
417
- val = JsonUtils.handleSourceKeysOperation({
418
- message,
419
- operationObject: sourceKey,
420
- });
421
- }
422
- else {
423
- val = (0, get_value_1.default)(message, sourceKey);
424
- }
425
- if (val || val === false || val === 0) {
426
- // return only if the value is valid.
427
- // else look for next possible source in precedence
428
- return val;
429
- }
430
- }
431
- }
432
- else if (typeof sourceKeys === 'string') {
433
- // got a single key
434
- // - we don't need to iterate over a loop for a single possible value
435
- return (0, get_value_1.default)(message, sourceKeys);
436
- }
437
- else if (typeof sourceKeys === 'object') {
438
- // if the sourceKey is an object we expect it to be a operation
439
- return JsonUtils.handleSourceKeysOperation({
440
- message,
441
- operationObject: sourceKeys,
442
- });
443
- }
444
- else {
445
- // wrong sourceKey type. abort
446
- // DEVELOPER ERROR
447
- throw new platform_error_1.PlatformError('Wrong sourceKey type or blank sourceKey array');
448
- }
449
- return null;
450
- }
451
- /**
452
- * Description: This method is used to get the value from the message
453
- * based on the source key
454
- * @param message - the message object
455
- * @param sourceKey - the source key to be used to get the value from the message
456
- * @returns {any} - returns the value of the source key
457
- * if the source key is not present in the message
458
- * returns null
459
- */
460
- static getFieldValueFromMessage(message, sourceKey) {
461
- const sourceKeyMap = JsonUtils.MESSAGE_MAPPING[sourceKey];
462
- if (sourceKeyMap) {
463
- return JsonUtils.getValueFromMessage(message, sourceKeyMap);
464
- }
465
- return null;
466
- }
467
- /**
468
- * Description: This method is used to construct the payload based on the mappingJson
469
- * @param message - the message object
470
- * @param mappingJson - the mappingJson to be used to construct the payload
471
- * @param destinationName - the destination name to be used to get the mappingJson
472
- * @returns {any} - returns the payload constructed based on the mappingJson
473
- * if the mappingJson is not present in the message or blank
474
- * returns null
475
- */
476
- static constructPayload(message, mappingJson, destinationName = '') {
477
- // Mapping JSON should be an array
478
- if (Array.isArray(mappingJson) && mappingJson.length > 0) {
479
- // - construct a blank payload and return at the end
480
- // - if you to need merge multiple constructPayload do it on the transformer code
481
- // - - will give a cleaner approach
482
- // - - you don't need to iterate over multiple loops to construct a payload for a single event
483
- const payload = {};
484
- mappingJson.forEach((mapping) => {
485
- const { sourceKeys, destKey, required, metadata, sourceFromGenericMap, } = mapping;
486
- // get the value from event, pass sourceFromGenericMap in the mapping to force this to take the
487
- // sourcekeys from GenericFieldMapping, else take the sourceKeys from specific destination mapping sourceKeys
488
- const integrationsObj = (0, lodash_1.isEmpty)(destinationName)
489
- ? sem_1.SemanticUtil.getIntegrationsObj(message, destinationName)
490
- : null;
491
- const value = JsonUtils.handleMetadataForValue(sourceFromGenericMap
492
- ? this.getFieldValueFromMessage(message, sourceKeys)
493
- : this.getValueFromMessage(message, sourceKeys), metadata, destKey, integrationsObj);
494
- if (value || value === 0 || value === false) {
495
- if (destKey) {
496
- // set the value only if correct
497
- (0, set_value_1.default)(payload, destKey, value);
498
- }
499
- else {
500
- // to set to root and flatten later
501
- payload[''] = value;
502
- }
503
- }
504
- else if (required) {
505
- // throw error if reqired value is missing
506
- throw new instrumentation_error_1.InstrumentationError(`Missing required value from ${JSON.stringify(sourceKeys)}`);
507
- }
508
- });
509
- return payload;
510
- }
511
- // invalid mappingJson
512
- return null;
513
- }
514
- static stringify(value) {
515
- return JSON.stringify(value);
516
- }
517
- }
518
- exports.JsonUtils = JsonUtils;
519
- JsonUtils.MESSAGE_MAPPING = constants_1.GENERIC_MAPPING_CONFIG;
520
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoianNvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy91dGlscy9qc29uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsNERBQW9DO0FBQ3BDLDBEQUE0QjtBQUM1QiwwREFBNEI7QUFDNUIsbUNBQXlEO0FBQ3pELHNFQUFxQztBQUVyQyxrREFBb0M7QUFDcEMsNkRBQXlEO0FBQ3pELCtCQUFxQztBQUNyQywyRUFBdUU7QUFDdkUsaUNBT2dCO0FBQ2hCLHlFQUFxRTtBQUNyRSw0Q0FBc0Q7QUFzQnRELE1BQWEsU0FBUztJQUdwQjs7Ozs7OztPQU9HO0lBQ0ssTUFBTSxDQUFDLHlCQUF5QixDQUFDLEVBQ3ZDLE9BQU8sRUFDUCxlQUFlLEdBSWhCO1FBQ0MsTUFBTSxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsR0FBRyxlQUFlLENBQUM7UUFFNUMseUNBQXlDO1FBQ3pDLG9DQUFvQztRQUNwQyxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsR0FBRyxFQUFFLEVBQUU7WUFDakMsTUFBTSxFQUFFLFVBQVUsRUFBRSxVQUFVLEVBQUUsR0FBRyxHQUFHLENBQUM7WUFDdkMsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDLElBQUksT0FBTyxVQUFVLEtBQUssUUFBUSxFQUFFO2dCQUNoRSxPQUFPLFVBQVUsQ0FBQzthQUNuQjtZQUNELElBQUksR0FBRyxDQUFDO1lBQ1IsSUFBSSxRQUFRLEdBQWtDLFVBQVUsQ0FBQztZQUN6RCxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDLEVBQUU7Z0JBQzdCLFFBQVEsR0FBRyxVQUFVLENBQUMsSUFBSSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxJQUFBLG1CQUFHLEVBQUMsT0FBTyxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQUM7YUFDOUQ7WUFDRCxJQUFJLFFBQVEsRUFB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USxFQUNSLE9BQU8sRUFDUCxlQUFlLENBQ2hCLENBQUM7Z0JBRUYsSUFBSSxLQUFLLElBQUksS0FBSyxLQUFLLENBQUMsSUFBSSxLQUFLLEtBQUssS0FBSyxFQUFFO29CQUMzQyxJQUFJLE9BQU8sRUFBRTt3QkFDWCxnQ0FBZ0M7d0JBQ2hDLElBQUEsbUJBQUcsRUFBQyxPQUFPLEVBQUUsT0FBTyxFQUFFLEtBQUssQ0FBQyxDQUFDO3FCQUM5Qjt5QkFBTTt3QkFDTCxtQ0FBbUM7d0JBQ25DLE9BQU8sQ0FBQyxFQUFFLENBQUMsR0FBRyxLQUFLLENBQUM7cUJBQ3JCO2lCQUNGO3FCQUFNLElBQUksUUFBUSxFQUFFO29CQUNuQiwwQ0FBMEM7b0JBQzFDLE1BQU0sSUFBSSw0Q0FBb0IsQ0FDNUIsK0JBQStCLElBQUksQ0FBQyxTQUFTLENBQUMsVUFBVSxDQUFDLEVBQUUsQ0FDNUQsQ0FBQztpQkFDSDtZQUNILENBQUMsQ0FBQyxDQUFDO1lBRUgsT0FBTyxPQUFPLENBQUM7U0FDaEI7UUFFRCxzQkFBc0I7UUFDdEIsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRU0sTUFBTSxDQUFDLFNBQVMsQ0FBQyxLQUFjO1FBQ3BDLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMvQixDQUFDOztBQTdsQkgsOEJBOGxCQztBQTdsQmdCLHlCQUFlLEdBQUcsa0NBQXNCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgSGFuZGxlYmFycyBmcm9tICdoYW5kbGViYXJzJztcbmltcG9ydCBnZXQgZnJvbSAnZ2V0LXZhbHVlJztcbmltcG9ydCBzZXQgZnJvbSAnc2V0LXZhbHVlJztcbmltcG9ydCB7IGV2ZXJ5LCBpc0VtcHR5LCBpc05pbCwgaXNOdW1iZXIgfSBmcm9tICdsb2Rhc2gnO1xuaW1wb3J0IG1vbWVudCBmcm9tICdtb21lbnQtdGltZXpvbmUnO1xuaW1wb3J0IHsgUnVkZGVyU3RhY2tFdmVudCB9IGZyb20gJy4uL3R5cGVzJztcbmltcG9ydCAqIGFzIGxvZ2dlciBmcm9tICcuLi9sb2dnZXInO1xuaW1wb3J0IHsgUGxhdGZvcm1FcnJvciB9IGZyb20gJy4uL2Vycm9ycy9wbGF0Zm9ybV9lcnJvcic7XG5pbXBvcnQgeyBTZW1hbnRpY1V0aWwgfSBmcm9tICcuL3NlbSc7XG5pbXBvcnQgeyBJbnN0cnVtZW50YXRpb25FcnJvciB9IGZyb20gJy4uL2Vycm9ycy9pbnN0cnVtZW50YXRpb25fZXJyb3InO1xuaW1wb3J0IHtcbiAgZmxhdHRlbkpzb24sXG4gIGZvcm1hdFRpbWVTdGFtcCxcbiAgZ2V0T2Zmc2V0SW5TZWMsXG4gIGhhc2hUb1NoYTI1NixcbiAgaXNEZWZpbmVkQW5kTm90TnVsbCxcbiAgaXNWYWxpZFVybCxcbn0gZnJvbSAnLi9taXNjJztcbmltcG9ydCB7IFRyYW5zZm9ybWF0aW9uRXJyb3IgfSBmcm9tICcuLi9lcnJvcnMvdHJhbnNmb3JtYXRpb25fZXJyb3InO1xuaW1wb3J0IHsgR0VORVJJQ19NQVBQSU5HX0NPTkZJRyB9IGZyb20gJy4uL2NvbnN0YW50cyc7XG5cbnR5cGUgT3BlcmF0aW9uID0ge1xuICBvcGVyYXRpb246IHN0cmluZztcbiAgYXJnczogeyBzb3VyY2VLZXlzOiBzdHJpbmdbXTsgZGVmYXVsdFZhbD86IGFueSB9W107XG59O1xudHlwZSBNYXBwaW5nTWV0YWRhdGEgPSB7XG4gIHR5cGU/OiBzdHJpbmcgfCBzdHJpbmdbXTtcbiAgdHlwZUZvcm1hdD86IHN0cmluZztcbiAgdGVtcGxhdGU/OiBzdHJpbmc7XG4gIGRlZmF1bHRWYWx1ZT86IHVua25vd247XG4gIGV4Y2x1ZGVzPzogc3RyaW5nW107XG4gIG11bHRpa2V5TWFwPzogeyBba2V5OiBzdHJpbmddOiBzdHJpbmdbXSB9O1xuICBzdHJpY3RNdWx0aU1hcD86IHsgc291cmNlVmFsOiBzdHJpbmdbXTsgZGVzdFZhbDogc3RyaW5nIH1bXTtcbiAgdmFsaWRhdGVUaW1lc3RhbXA/OiB7XG4gICAgYWxsb3dlZEZ1dHVyZVRpbWVEaWZmZXJlbmNlOiBudW1iZXI7XG4gICAgYWxsb3dlZEZ1dHVyZVRpbWVVbml0OiBzdHJpbmc7XG4gICAgYWxsb3dlZFBhc3RUaW1lRGlmZmVyZW5jZTogbnVtYmVyO1xuICAgIGFsbG93ZWRQYXN0VGltZVVuaXQ6IHN0cmluZztcbiAgfTtcbiAgYWxsb3dlZEtleUNoZWNrPzogeyB0eXBlOiBzdHJpbmc7IHNvdXJjZVZhbDogc3RyaW5nW10gfVtdO1xufTtcbmV4cG9ydCBjbGFzcyBKc29uVXRpbHMge1xuICBwcml2YXRlIHN0YXRpYyBNRVNTQUdFX01BUFBJTkcgPSBHRU5FUklDX01BUFBJTkdfQ09ORklHO1xuXG4gIC8qKlxuICAgKiBEZXNjcmlwdGlvbjogVGhpcyBtZXRob2QgaXMgdXNlZCB0byBoYW5kbGUgdGhlIHNvdXJjZSBrZXlzIG9wZXJhdGlvblxuICAgKiBAcGFyYW0gbWVzc2FnZSAtIHRoZSBtZXNzYWdlIG9iamVjdFxuICAgKiBAcmV0dXJuc1x0e2FueX1cdC0gcmV0dXJucyB0aGUgdmFsdWUgb2YgdGhlIHNvdXJjZSBrZXlcbiAgICogXHRcdFx0XHRcdFx0aWYgdGhlIHNvdXJjZSBrZXkgaXMgbm90IHByZXNlbnQgaW4gdGhlIG1lc3NhZ2VcbiAgICogXHRcdFx0XHRcdFx0cmV0dXJucyBudWxsXG4gICAqXG4gICAqL1xuICBwcml2YXRlIHN0YXRpYyBoYW5kbGVTb3VyY2VLZXlzT3BlcmF0aW9uKHtcbiAgICBtZXNzYWdlLFxuICAgIG9wZXJhdGlvbk9iamVjdCxcbiAgfToge1xuICAgIG1lc3NhZ2U6IFJ1ZGRlclN0YWNrRXZlbnQ7XG4gICAgb3BlcmF0aW9uT2JqZWN0OiBPcGVyYXRpb247XG4gIH0pOiBhbnkge1xuICAgIGNvbnN0IHsgb3BlcmF0aW9uLCBhcmdzIH0gPSBvcGVyYXRpb25PYmplY3Q7XG5cbiAgICAvLyBwb3B1bGF0ZSB0aGUgdmFsdWVzIGZyb20gdGhlIGFyZ3VtZW50c1xuICAgIC8vIGluIHRoZSBzYW1lIG9yZGVyIGl0IGlzIHBvcHVsYXRlZFxuICAgIGNvbnN0IGFyZ1ZhbHVlcyA9IGFyZ3MubWFwKChhcmcpID0+IHtcbiAgICAgIGNvbnN0IHsgc291cmNlS2V5cywgZGVmYXVsdFZhbCB9ID0gYXJnO1xuICAgICAgaWYgKCFBcnJheS5pc0FycmF5KHNvdXJjZUtleXMpICYmIHR5cGVvZiBzb3VyY2VLZXlzICE9PSAnc3RyaW5nJykge1xuICAgICAgICByZXR1cm4gZGVmYXVsdFZhbDtcbiAgICAgIH1cbiAgICAgIGxldCB2YWw7XG4gICAgICBsZXQgZm91bmRLZXk6IHN0cmluZyB8IHN0cmluZ1tdIHwgdW5kZWZpbmVkID0gc291cmNlS2V5cztcbiAgICAgIGlmIChBcnJheS5pc0FycmF5KHNvdXJjZUtleXMpKSB7XG4gICAgICAgIGZvdW5kS2V5ID0gc291cmNlS2V5cy5maW5kKChzcmNLZXkpID0+IGdldChtZXNzYWdlLCBzcmNLZXkpKTtcbiAgICAgIH1cbiAgICAgIGlmIChmb3VuZEtleSkge1xuICAgICAgICB2YWwgPSBnZXQobWVzc2FnZSwgZm91bmRLZXkpO1xuICAgICAgfVxuICAgICAgaWYgKHZhbCB8fCB2YWwgPT09IGZhbHNlIHx8IHZhbCA9PT0gMCkge1xuICAgICAgICByZXR1cm4gdmFsO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGRlZmF1bHRWYWw7XG4gICAgfSk7XG5cbiAgICAvLyBxdWljayBzYW5pdHkgY2hlY2sgZm9yIHRoZSB1bmRlZmluZWQgdmFsdWVzIGluIHRoZSBsaXN0LlxuICAgIC8vIGlmIHRoZXJlIGlzIGFueSB1bmRlZmluZWQgdmFsdWVzLCByZXR1cm4gbnVsbFxuICAgIC8vIHdpdGhvdXQgZ29pbmcgZnVydGhlciBmb3Igb3BlcmF0aW9uc1xuICAgIGNvbnN0IGlzQWxsRGVmaW5lZCA9IGV2ZXJ5KGFyZ1ZhbHVlcywgKHYpID0+ICFpc05pbCh2KSk7XG4gICAgaWYgKCFpc0FsbERlZmluZWQpIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIC8vIHN0YXJ0IGhhbmRsaW5nIG9wZXJhdGlvbnNcbiAgICBsZXQgcmVzdWx0OiBudW1iZXI7XG4gICAgc3dpdGNoIChvcGVyYXRpb24pIHtcbiAgICAgIGNhc2UgJ211bHRpcGxpY2F0aW9uJzpcbiAgICAgICAgcmVzdWx0ID0gMTtcbiAgICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIG5vLXJlc3RyaWN0ZWQtc3ludGF4XG4gICAgICAgIGZvciAoY29uc3QgdiBvZiBhcmdWYWx1ZXMpIHtcbiAgICAgICAgICBpZiAoaXNOdW1iZXIodikpIHtcbiAgICAgICAgICAgIHJlc3VsdCAqPSB2O1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyBpZiB0aGVyZSBpcyBhIG5vbiBudW1iZXIgYXJndW1lbnQgc2ltcGx5IHJldHVybiBudWxsXG4gICAgICAgICAgICAvLyBub24gbnVtYmVycyBjYW4ndCBiZSBvcGVyYXRlZCBhcml0aG1hdGljYWxseVxuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgICBjYXNlICdhZGRpdGlvbic6XG4gICAgICAgIHJlc3VsdCA9IDA7XG4gICAgICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby1yZXN0cmljdGVkLXN5bnRheFxuICAgICAgICBmb3IgKGNvbnN0IHYgb2YgYXJnVmFsdWVzKSB7XG4gICAgICAgICAgaWYgKGlzTnVtYmVyKHYpKSB7XG4gICAgICAgICAgICByZXN1bHQgKz0gdjtcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgLy8gaWYgdGhlcmUgaXMgYSBub24gbnVtYmVyIGFyZ3VtZW50IHNpbXBseSByZXR1cm4gbnVsbFxuICAgICAgICAgICAgLy8gbm9uIG51bWJlcnMgY2FuJ3QgYmUgb3BlcmF0ZWQgYXJpdGhtYXRpY2FsbHlcbiAgICAgICAgICAgIHJldHVybiBudWxsO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gcmVzdWx0O1xuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY2hlY2tUaW1lc3RhbXAoXG4gICAgdmFsaWRhdGVUaW1lc3RhbXA6IHtcbiAgICAgIGFsbG93ZWRGdXR1cmVUaW1lRGlmZmVyZW5jZTogbnVtYmVyO1xuICAgICAgYWxsb3dlZEZ1dHVyZVRpbWVVbml0OiBzdHJpbmc7XG4gICAgICBhbGxvd2VkUGFzdFRpbWVEaWZmZXJlbmNlOiBudW1iZXI7XG4gICAgICBhbGxvd2VkUGFzdFRpbWVVbml0OiBzdHJpbmc7XG4gICAgfSxcbiAgICBmb3JtYXR0ZWRWYWw6IG51bWJlcixcbiAgKSB7XG4gICAgY29uc3Qge1xuICAgICAgYWxsb3dlZFBhc3RUaW1lRGlmZmVyZW5jZSxcbiAgICAgIGFsbG93ZWRQYXN0VGltZVVuaXQsIC8vIHNlY29uZHMsIG1pbnV0ZXMsIGhvdXJzXG4gICAgICBhbGxvd2VkRnV0dXJlVGltZURpZmZlcmVuY2UsXG4gICAgICBhbGxvd2VkRnV0dXJlVGltZVVuaXQsIC8vIHNlY29uZHMsIG1pbnV0ZXMsIGhvdXJzXG4gICAgfSA9IHZhbGlkYXRlVGltZXN0YW1wO1xuXG4gICAgbGV0IHBhc3RUaW1lRGlmZmVyZW5jZTtcbiAgICBsZXQgZnV0dXJlVGltZURpZmZlcmVuY2U7XG5cbiAgICBjb25zdCBjdXJyZW50VGltZSA9IG1vbWVudC51bml4KE51bWJlcihtb21lbnQoKS5mb3JtYXQoJ1gnKSkpO1xuICAgIGNvbnN0IHRpbWUgPSBtb21lbnQudW5peChOdW1iZXIobW9tZW50KGZvcm1hdHRlZFZhbCkuZm9ybWF0KCdYJykpKTtcblxuICAgIHN3aXRjaCAoYWxsb3dlZFBhc3RUaW1lVW5pdCkge1xuICAgICAgY2FzZSAnc2Vjb25kcyc6XG4gICAgICAgIHBhc3RUaW1lRGlmZmVyZW5jZSA9IE1hdGguY2VpbChtb21lbnQuZHVyYXRpb24oY3VycmVudFRpbWUuZGlmZih0aW1lKSkuYXNTZWNvbmRzKCkpO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJ21pbnV0ZXMnOlxuICAgICAgICBwYXN0VGltZURpZmZlcmVuY2UgPSBNYXRoLmNlaWwobW9tZW50LmR1cmF0aW9uKGN1cnJlbnRUaW1lLmRpZmYodGltZSkpLmFzTWludXRlcygpKTtcbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlICdob3Vycyc6XG4gICAgICAgIHBhc3RUaW1lRGlmZmVyZW5jZSA9IE1hdGguY2VpbChtb21lbnQuZHVyYXRpb24oY3VycmVudFRpbWUuZGlmZih0aW1lKSkuYXNIb3VycygpKTtcbiAgICAgICAgYnJlYWs7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICBicmVhaztcbiAgICB9XG5cbiAgICBpZiAocGFzdFRpbWVEaWZmZXJlbmNlID4gYWxsb3dlZFBhc3RUaW1lRGlmZmVyZW5jZSkge1xuICAgICAgdGhyb3cgbmV3IEluc3RydW1lbnRhdGlvbkVycm9yKFxuICAgICAgICBgQWxsb3dlZCB0aW1lc3RhbXAgaXMgWyR7YWxsb3dlZFBhc3RUaW1lRGlmZmVyZW5jZX0gJHthbGxvd2VkUGFzdFRpbWVVbml0fV0gaW50byB0aGUgcGFzdGAsXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmICghcGFzdFRpbWVEaWZmZXJlbmNlIHx8IHBhc3RUaW1lRGlmZmVyZW5jZSA8PSAwKSB7XG4gICAgICBzd2l0Y2ggKGFsbG93ZWRGdXR1cmVUaW1lVW5pdCkge1xuICAgICAgICBjYXNlICdzZWNvbmRzJzpcbiAgICAgICAgICBmdXR1cmVUaW1lRGlmZmVyZW5jZSA9IE1hdGguY2VpbChtb21lbnQuZHVyYXRpb24odGltZS5kaWZmKGN1cnJlbnRUaW1lKSkuYXNTZWNvbmRzKCkpO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlICdtaW51dGVzJzpcbiAgICAgICAgICBmdXR1cmVUaW1lRGlmZmVyZW5jZSA9IE1hdGguY2VpbChtb21lbnQuZHVyYXRpb24odGltZS5kaWZmKGN1cnJlbnRUaW1lKSkuYXNNaW51dGVzKCkpO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlICdob3Vycyc6XG4gICAgICAgICAgZnV0dXJlVGltZURpZmZlcmVuY2UgPSBNYXRoLmNlaWwobW9tZW50LmR1cmF0aW9uKHRpbWUuZGlmZihjdXJyZW50VGltZSkpLmFzSG91cnMoKSk7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgYnJlYWs7XG4gICAgICB9XG5cbiAgICAgIGlmIChmdXR1cmVUaW1lRGlmZmVyZW5jZSA+IGFsbG93ZWRGdXR1cmVUaW1lRGlmZmVyZW5jZSkge1xuICAgICAgICB0aHJvdyBuZXcgSW5zdHJ1bWVudGF0aW9uRXJyb3IoXG4gICAgICAgICAgYEFsbG93ZWQgdGltZXN0YW1wIGlzIFske2FsbG93ZWRGdXR1cmVUaW1lRGlmZmVyZW5jZX0gJHthbGxvd2VkRnV0dXJlVGltZVVuaXR9XSBpbnRvIHRoZSBmdXR1cmVgLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGZvcm1hdFZhbHVlcyhcbiAgICBmb3JtYXR0ZWRWYWw6IHVua25vd24sXG4gICAgZm9ybWF0dGluZ1R5cGU6IHN0cmluZyxcbiAgICB0eXBlRm9ybWF0Pzogc3RyaW5nLFxuICAgIGludGVncmF0aW9uc09iaj86IHsgW2tleTogc3RyaW5nXTogYW55IH0sXG4gICkge1xuICAgIGxldCBjdXJGb3JtYXR0ZWRWYWwgPSBmb3JtYXR0ZWRWYWw7XG5cbiAgICBjb25zdCBmb3JtYXR0aW5nRnVuY3Rpb25zID0ge1xuICAgICAgdGltZXN0YW1wOiAoKSA9PiB7XG4gICAgICAgIGN1ckZvcm1hdHRlZFZhbCA9IGZvcm1hdFRpbWVTdGFtcChTdHJpbmcoZm9ybWF0dGVkVmFsKSwgdHlwZUZvcm1hdCk7XG4gICAgICB9LFxuICAgICAgc2Vjb25kVGltZXN0YW1wOiAoKSA9PiB7XG4gICAgICAgIGlmICghbW9tZW50KFN0cmluZyhmb3JtYXR0ZWRWYWwpLCAneCcsIHRydWUpLmlzVmFsaWQoKSkge1xuICAgICAgICAgIGN1ckZvcm1hdHRlZFZhbCA9IE1hdGguZmxvb3IoXG4gICAgICAgICAgICAoZm9ybWF0VGltZVN0YW1wKFN0cmluZyhmb3JtYXR0ZWRWYWwpLCB0eXBlRm9ybWF0KSBhcyBudW1iZXIpIC8gMTAwMCxcbiAgICAgICAgICApO1xuICAgICAgICB9XG4gICAgICB9LFxuICAgICAgbWljcm9TZWNvbmRUaW1lc3RhbXA6ICgpID0+IHtcbiAgICAgICAgY29uc3QgdGVtcF92YWwgPSBtb21lbnQudW5peChOdW1iZXIobW9tZW50KFN0cmluZyhmb3JtYXR0ZWRWYWwpKS5mb3JtYXQoJ1gnKSkpO1xuICAgICAgICBjdXJGb3JtYXR0ZWRWYWwgPSB0ZW1wX3ZhbC50b0RhdGUoKS5nZXRUaW1lKCkgKiAxMDAwICsgdGVtcF92YWwudG9EYXRlKCkuZ2V0TWlsbGlzZWNvbmRzKCk7XG4gICAgICB9LFxuICAgICAgZmxhdEpzb246ICgpID0+IHtcbiAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gZmxhdHRlbkpzb24oZm9ybWF0dGVkVmFsKTtcbiAgICAgIH0sXG4gICAgICBlbmNvZGVVUklDb21wb25lbnQ6ICgpID0+IHtcbiAgICAgICAgY29uc3QgdF92YWw6IHN0cmluZyB8IG51bWJlciB8IGJvb2xlYW4gPSBmb3JtYXR0ZWRWYWwgYXMgc3RyaW5nIHwgbnVtYmVyIHwgYm9vbGVhbjtcbiAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gZW5jb2RlVVJJQ29tcG9uZW50KHRfdmFsKTtcbiAgICAgIH0sXG4gICAgICBqc29uU3RyaW5naWZ5OiAoKSA9PiB7XG4gICAgICAgIGN1ckZvcm1hdHRlZFZhbCA9IEpTT04uc3RyaW5naWZ5KGZvcm1hdHRlZFZhbCk7XG4gICAgICB9LFxuICAgICAganNvblN0cmluZ2lmeU9uRmxhdHRlbjogKCkgPT4ge1xuICAgICAgICBjdXJGb3JtYXR0ZWRWYWwgPSBKU09OLnN0cmluZ2lmeShmbGF0dGVuSnNvbihmb3JtYXR0ZWRWYWwpKTtcbiAgICAgIH0sXG4gICAgICBkb2JJbk1NREQ6ICgpID0+IHtcbiAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gU3RyaW5nKGZvcm1hdHRlZFZhbCkuc2xpY2UoNSk7XG4gICAgICAgIGN1ckZvcm1hdHRlZFZhbCA9IFN0cmluZyhjdXJGb3JtYXR0ZWRWYWwpLnJlcGxhY2UoJy0nLCAnLycpO1xuICAgICAgfSxcbiAgICAgIGpzb25TdHJpbmdpZnlPbk9iamVjdDogKCkgPT4ge1xuICAgICAgICBpZiAodHlwZW9mIGZvcm1hdHRlZFZhbCAhPT0gJ3N0cmluZycpIHtcbiAgICAgICAgICBjdXJGb3JtYXR0ZWRWYWwgPSBKU09OLnN0cmluZ2lmeShmb3JtYXR0ZWRWYWwpO1xuICAgICAgICB9XG4gICAgICB9LFxuICAgICAgbnVtYmVyRm9yUmV2ZW51ZTogKCkgPT4ge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgKHR5cGVvZiBmb3JtYXR0ZWRWYWwgPT09ICdzdHJpbmcnIHx8IGZvcm1hdHRlZFZhbCBpbnN0YW5jZW9mIFN0cmluZykgJiZcbiAgICAgICAgICBmb3JtYXR0ZWRWYWwuY2hhckF0KDApID09PSAnJCdcbiAgICAgICAgKSB7XG4gICAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gZm9ybWF0dGVkVmFsLnN1YnN0cmluZygxKTtcbiAgICAgICAgfVxuICAgICAgICBjdXJGb3JtYXR0ZWRWYWwgPSBOdW1iZXIucGFyc2VGbG9hdChOdW1iZXIoY3VyRm9ybWF0dGVkVmFsIHx8IDApLnRvRml4ZWQoMikpO1xuICAgICAgICBpZiAoTnVtYmVyLmlzTmFOKGN1ckZvcm1hdHRlZFZhbCkpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgSW5zdHJ1bWVudGF0aW9uRXJyb3IoJ1JldmVudWUgaXMgbm90IGluIHRoZSBjb3JyZWN0IGZvcm1hdCcpO1xuICAgICAgICB9XG4gICAgICB9LFxuICAgICAgdG9TdHJpbmc6ICgpID0+IHtcbiAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gU3RyaW5nKGZvcm1hdHRlZFZhbCk7XG4gICAgICB9LFxuICAgICAgdG9OdW1iZXI6ICgpID0+IHtcbiAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gTnVtYmVyKGZvcm1hdHRlZFZhbCk7XG4gICAgICB9LFxuICAgICAgdG9GbG9hdDogKCkgPT4ge1xuICAgICAgICBjdXJGb3JtYXR0ZWRWYWwgPSBwYXJzZUZsb2F0KFN0cmluZyhmb3JtYXR0ZWRWYWwpKTtcbiAgICAgIH0sXG4gICAgICB0b0ludDogKCkgPT4ge1xuICAgICAgICBjdXJGb3JtYXR0ZWRWYWwgPSBwYXJzZUludChTdHJpbmcoZm9ybWF0dGVkVmFsKSwgMTApO1xuICAgICAgfSxcbiAgICAgIHRvTG93ZXI6ICgpID0+IHtcbiAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gU3RyaW5nKGZvcm1hdHRlZFZhbCkudG9Mb3dlckNhc2UoKTtcbiAgICAgIH0sXG4gICAgICBoYXNoVG9TaGEyNTY6ICgpID0+IHtcbiAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gaW50ZWdyYXRpb25zT2JqPy5oYXNoZWRcbiAgICAgICAgICA/IFN0cmluZyhmb3JtYXR0ZWRWYWwpXG4gICAgICAgICAgOiBoYXNoVG9TaGEyNTYoU3RyaW5nKGZvcm1hdHRlZFZhbCkpO1xuICAgICAgfSxcbiAgICAgIGdldE9mZnNldEluU2VjOiAoKSA9PiB7XG4gICAgICAgIGN1ckZvcm1hdHRlZFZhbCA9IGdldE9mZnNldEluU2VjKFN0cmluZyhmb3JtYXR0ZWRWYWwpKTtcbiAgICAgIH0sXG4gICAgICBkb21haW5Vcmw6ICgpID0+IHtcbiAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gU3RyaW5nKGZvcm1hdHRlZFZhbCkucmVwbGFjZSgnaHR0cHM6Ly8nLCAnJykucmVwbGFjZSgnaHR0cDovLycsICcnKTtcbiAgICAgIH0sXG4gICAgICBkb21haW5VcmxWMjogKCkgPT4ge1xuICAgICAgICBjb25zdCB1cmwgPSBpc1ZhbGlkVXJsKGZvcm1hdHRlZFZhbCBhcyBzdHJpbmcgfCBVUkwpO1xuICAgICAgICBpZiAoIXVybCkge1xuICAgICAgICAgIHRocm93IG5ldyBJbnN0cnVtZW50YXRpb25FcnJvcihgSW52YWxpZCBVUkw6ICR7Zm9ybWF0dGVkVmFsfWApO1xuICAgICAgICB9XG4gICAgICAgIGN1ckZvcm1hdHRlZFZhbCA9IHVybC5ob3N0bmFtZS5yZXBsYWNlKCd3d3cuJywgJycpO1xuICAgICAgfSxcbiAgICAgIElzQm9vbGVhbjogKCkgPT4ge1xuICAgICAgICBjdXJGb3JtYXR0ZWRWYWwgPSB0cnVlO1xuICAgICAgICBpZiAoISh0eXBlb2YgZm9ybWF0dGVkVmFsID09PSAnYm9vbGVhbicpKSB7XG4gICAgICAgICAgbG9nZ2VyLmRlYnVnKCdCb29sZWFuIHZhbHVlIG1pc3NpbmcsIHNvIGRyb3BwaW5nIGl0Jyk7XG4gICAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gZmFsc2U7XG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICB0cmltOiAoKSA9PiB7XG4gICAgICAgIGlmICh0eXBlb2YgZm9ybWF0dGVkVmFsID09PSAnc3RyaW5nJykge1xuICAgICAgICAgIGN1ckZvcm1hdHRlZFZhbCA9IGZvcm1hdHRlZFZhbC50cmltKCk7XG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgfTtcblxuICAgIGlmIChmb3JtYXR0aW5nVHlwZSBpbiBmb3JtYXR0aW5nRnVuY3Rpb25zKSB7XG4gICAgICBjb25zdCBmb3JtYXR0aW5nRnVuY3Rpb24gPSBmb3JtYXR0aW5nRnVuY3Rpb25zW2Zvcm1hdHRpbmdUeXBlXTtcbiAgICAgIGZvcm1hdHRpbmdGdW5jdGlvbigpO1xuICAgIH1cblxuICAgIHJldHVybiBjdXJGb3JtYXR0ZWRWYWw7XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBoYW5kbGVFeGNsdWRlcyhcbiAgICB2YWx1ZTogdW5rbm93bixcbiAgICBleGNsdWRlczogc3RyaW5nW10sXG4gICAgZm9ybWF0dGVkVmFsOiB7IFtrZXk6IHN0cmluZ106IGFueSB9LFxuICApIHtcbiAgICBpZiAodHlwZW9mIHZhbHVlID09PSAnb2JqZWN0Jykge1xuICAgICAgLy8gZXhjbHVkZSB0aGUgZmllbGRzIGZyb20gdGhlIGZvcm1hdHRlZFZhbFxuICAgICAgZXhjbHVkZXMuZm9yRWFjaCgoa2V5KSA9PiB7XG4gICAgICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby1wYXJhbS1yZWFzc2lnblxuICAgICAgICBkZWxldGUgZm9ybWF0dGVkVmFsW2tleV07XG4gICAgICB9KTtcbiAgICB9IGVsc2Uge1xuICAgICAgbG9nZ2VyLndhcm4oXCJleGNsdWRlcyBkb2Vzbid0IHdvcmsgd2l0aCBub24tb2JqZWN0IGRhdGEgdHlwZS4gSWdub3JpbmcgZXhjbHVkZXNcIik7XG4gICAgfVxuICB9XG5cbiAgLy8gZm9ybWF0IHRoZSB2YWx1ZSBhcyBwZXIgdGhlIG1ldGFkYXRhIHZhbHVlc1xuICAvLyBFeHBlY3RlZCBtZXRhZGF0YSBrZXlzIGFyZTogKGFjY29yZGluZyB0byBwcmVjZWRlbmNlKVxuICAvLyAtIC0gdHlwZSwgdHlwZUZvcm1hdDogZXhwZWN0ZWQgZGF0YSB0eXBlIGFuZCBpdHMgZm9ybWF0XG4gIC8vIC0gLSB0ZW1wbGF0ZSA6IG5lZWQgdG8gaGF2ZSBhIGhhbmRsZWJhciBleHByZXNzaW9uIHt7dmFsdWV9fVxuICAvLyAtIC0gZXhjbHVkZXMgOiBmaWVsZHMgeW91IHdhbnQgdG8gc3RyaXAgb2YgZnJvbSB0aGUgZmluYWwgdmFsdWUgKHdvcmtzIG9ubHkgZm9yIG9iamVjdClcbiAgLy8gLSAtIC0gLSBleDogXCJhbm9ueW1vdXNJZFwiLCBcInVzZXJJZFwiIGZyb20gdHJhaXRzXG4gIHByaXZhdGUgc3RhdGljIGhhbmRsZU1ldGFkYXRhRm9yVmFsdWUoXG4gICAgdmFsdWU6IHVua25vd24sXG4gICAgbWV0YWRhdGE6IE1hcHBpbmdNZXRhZGF0YSxcbiAgICBkZXN0S2V5OiBzdHJpbmcsXG4gICAgaW50ZWdyYXRpb25zT2JqOiB1bmtub3duLFxuICApIHtcbiAgICBpZiAoIW1ldGFkYXRhKSB7XG4gICAgICByZXR1cm4gdmFsdWU7XG4gICAgfVxuXG4gICAgLy8gZ2V0IGluZm9ybWF0aW9uIGZyb20gbWV0YWRhdGFcbiAgICBjb25zdCB7XG4gICAgICB0eXBlLFxuICAgICAgdHlwZUZvcm1hdCxcbiAgICAgIHRlbXBsYXRlLFxuICAgICAgZGVmYXVsdFZhbHVlLFxuICAgICAgZXhjbHVkZXMsXG4gICAgICBtdWx0aWtleU1hcCxcbiAgICAgIHN0cmljdE11bHRpTWFwLFxuICAgICAgdmFsaWRhdGVUaW1lc3RhbXAsXG4gICAgICBhbGxvd2VkS2V5Q2hlY2ssXG4gICAgfSA9IG1ldGFkYXRhO1xuXG4gICAgLy8gaWYgdmFsdWUgaXMgbnVsbCBhbmQgZGVmYXVsdFZhbHVlIGlzIHN1cHBsaWVkIC0gdXNlIHRoYXRcbiAgICBpZiAoIWlzRGVmaW5lZEFuZE5vdE51bGwodmFsdWUpKSB7XG4gICAgICByZXR1cm4gZGVmYXVsdFZhbHVlIHx8IHZhbHVlO1xuICAgIH1cbiAgICAvLyB3ZSd2ZSBnb3QgYSBjb3JyZWN0IHZhbHVlLiBzdGFydCBwcm9jZXNzaW5nXG4gICAgbGV0IGZvcm1hdHRlZFZhbCA9IHZhbHVlO1xuXG4gICAgLyoqXG4gICAgICogdmFsaWRhdGUgYWxsb3dlZCB0aW1lIGRpZmZlcmVuY2VcbiAgICAgKi9cbiAgICBpZiAodmFsaWRhdGVUaW1lc3RhbXApIHtcbiAgICAgIGZvcm1hdHRlZFZhbCA9IE51bWJlcihmb3JtYXR0ZWRWYWwpO1xuICAgICAgaWYgKE51bWJlci5pc05hTihmb3JtYXR0ZWRWYWwpKSB7XG4gICAgICAgIHRocm93IG5ldyBUcmFuc2Zvcm1hdGlvbkVycm9yKCdUaW1lc3RhbXAgaXMgbm90IGEgbnVtYmVyLCBjYW5ub3QgdmFsaWRhdGUnKTtcbiAgICAgIH1cbiAgICAgIEpzb25VdGlscy5jaGVja1RpbWVzdGFtcCh2YWxpZGF0ZVRpbWVzdGFtcCwgZm9ybWF0dGVkVmFsIGFzIG51bWJlcik7XG4gICAgfVxuXG4gICAgaWYgKHR5cGUpIHtcbiAgICAgIGlmIChBcnJheS5pc0FycmF5KHR5cGUpKSB7XG4gICAgICAgIHR5cGUuZm9yRWFjaCgoZWFjaFR5cGUpID0+IHtcbiAgICAgICAgICBmb3JtYXR0ZWRWYWwgPSBKc29uVXRpbHMuZm9ybWF0VmFsdWVzKFxuICAgICAgICAgICAgZm9ybWF0dGVkVmFsLFxuICAgICAgICAgICAgZWFjaFR5cGUsXG4gICAgICAgICAgICB0eXBlRm9ybWF0LFxuICAgICAgICAgICAgaW50ZWdyYXRpb25zT2JqIGFzIHsgW2tleTogc3RyaW5nXTogdW5rbm93biB9LFxuICAgICAgICAgICk7XG4gICAgICAgIH0pO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgZm9ybWF0dGVkVmFsID0gSnNvblV0aWxzLmZvcm1hdFZhbHVlcyhcbiAgICAgICAgICBmb3JtYXR0ZWRWYWwsXG4gICAgICAgICAgdHlwZSxcbiAgICAgICAgICB0eXBlRm9ybWF0LFxuICAgICAgICAgIGludGVncmF0aW9uc09iaiBhcyB7IFtrZXk6IHN0cmluZ106IHVua25vd24gfSxcbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBoYW5kbGUgdGVtcGxhdGVcbiAgICBpZiAodGVtcGxhdGUpIHtcbiAgICAgIGZvcm1hdHRlZFZhbCA9IEpzb25VdGlscy5oYW5kbGVUZW1wbGF0ZSh0ZW1wbGF0ZSwgdmFsdWUgYXMgc3RyaW5nKTtcbiAgICB9XG5cbiAgICAvLyBoYW5kbGUgZXhjbHVkZXNcbiAgICBpZiAoZXhjbHVkZXMpIHtcbiAgICAgIEpzb25VdGlscy5oYW5kbGVFeGNsdWRlcyh2YWx1ZSwgZXhjbHVkZXMsIGZvcm1hdHRlZFZhbCBhcyB7IFtrZXk6IHN0cmluZ106IGFueSB9KTtcbiAgICB9XG5cbiAgICAvLyBoYW5kbGUgbXVsdGlrZXlNYXBcbiAgICBpZiAobXVsdGlrZXlNYXAgfHwgc3RyaWN0TXVsdGlNYXApIHtcbiAgICAgIGZvcm1hdHRlZFZhbCA9IEpzb25VdGlscy5oYW5kbGVNdWx0aWtleU1hcChcbiAgICAgICAgbXVsdGlrZXlNYXAsXG4gICAgICAgIHN0cmljdE11bHRpTWFwLFxuICAgICAgICBmb3JtYXR0ZWRWYWwsXG4gICAgICAgIGRlc3RLZXksXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChhbGxvd2VkS2V5Q2hlY2spIHtcbiAgICAgIGxldCBmb3VuZFZhbCA9IGZhbHNlO1xuICAgICAgaWYgKEFycmF5LmlzQXJyYXkoYWxsb3dlZEtleUNoZWNrKSkge1xuICAgICAgICBhbGxvd2VkS2V5Q2hlY2suc29tZSgoa2V5KSA9PiB7XG4gICAgICAgICAgc3dpdGNoIChrZXkudHlwZSkge1xuICAgICAgICAgICAgY2FzZSAndG9Mb3dlckNhc2UnOlxuICAgICAgICAgICAgICBmb3JtYXR0ZWRWYWwgPSBTdHJpbmcoZm9ybWF0dGVkVmFsKS50b0xvd2VyQ2FzZSgpO1xuICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIGNhc2UgJ3RvVXBwZXJDYXNlJzpcbiAgICAgICAgICAgICAgZm9ybWF0dGVkVmFsID0gU3RyaW5nKGZvcm1hdHRlZFZhbCkudG9VcHBlckNhc2UoKTtcbiAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKGtleS5zb3VyY2VWYWwuaW5jbHVkZXMoZm9ybWF0dGVkVmFsIGFzIGFueSkpIHtcbiAgICAgICAgICAgIGZvdW5kVmFsID0gdHJ1ZTtcbiAgICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgICAgaWYgKCFmb3VuZFZhbCkge1xuICAgICAgICBmb3JtYXR0ZWRWYWwgPSB1bmRlZmluZWQ7XG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIGZvcm1hdHRlZFZhbDtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGhhbmRsZVRlbXBsYXRlKHRlbXBsYXRlOiBzdHJpbmcsIHZhbHVlOiBzdHJpbmcpIHtcbiAgICBjb25zdCBoVGVtcGxhdGUgPSBIYW5kbGViYXJzLmNvbXBpbGUodGVtcGxhdGUudHJpbSgpKTtcbiAgICBjb25zdCBmb3JtYXR0ZWRWYWwgPSBoVGVtcGxhdGUoeyB2YWx1ZSB9KS50cmltKCk7XG4gICAgcmV0dXJuIGZvcm1hdHRlZFZhbDtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGhhbmRsZU11bHRpa2V5TWFwKFxuICAgIG11bHRpa2V5TWFwOlxuICAgICAgfCB7XG4gICAgICAgICAgW2tleTogc3RyaW5nXTogc3RyaW5nW107XG4gICAgICAgIH1cbiAgICAgIHwgdW5kZWZpbmVkLFxuICAgIHN0cmljdE11bHRpTWFwOlxuICAgICAgfCB7XG4gICAgICAgICAgc291cmNlVmFsOiBzdHJpbmdbXTtcbiAgICAgICAgICBkZXN0VmFsOiBzdHJpbmc7XG4gICAgICAgIH1bXVxuICAgICAgfCB1bmRlZmluZWQsXG4gICAgZm9ybWF0dGVkVmFsOiB1bmtub3duLFxuICAgIGRlc3RLZXk6IHN0cmluZyxcbiAgKSB7XG4gICAgLy8gc291cmNlVmFsIGlzIGV4cGVjdGVkIHRvIGJlIGFuIGFycmF5XG4gICAgLy8gaWYgdmFsdWUgaXMgcHJlc2VudCBpbiBzb3VyY2VWYWwsIHJldHVybnMgdGhlIGRlc3RWYWxcbiAgICAvLyBlbHNlIHJldHVybnMgdGhlIG9yaWdpbmFsIHZhbHVlXG4gICAgLy8gRXhwZWN0ZWQgbXVsdGlrZXlNYXAgdmFsdWU6XG4gICAgLy8gXCJtdWx0aWtleU1hcFwiOiBbXG4gICAgLy8gICB7XG4gICAgLy8gICAgIFwic291cmNlVmFsXCI6IFtcIm1cIiwgXCJNXCIsIFwiTWFsZVwiLCBcIm1hbGVcIl0sXG4gICAgLy8gICAgIFwiZGVzdFZhbFwiOiBcIk1cIlxuICAgIC8vICAgfSxcbiAgICAvLyAgIHtcbiAgICAvLyAgICAgXCJzb3VyY2VWYWxcIjogW1wiZlwiLCBcIkZcIiwgXCJGZW1hbGVcIiwgXCJmZW1hbGVcIl0sXG4gICAgLy8gICAgIFwiZGVzdFZhbFwiOiBcIkZcIlxuICAgIC8vICAgfVxuICAgIC8vIF1cbiAgICBsZXQgY3VyRm9ybWF0dGVkVmFsID0gZm9ybWF0dGVkVmFsO1xuICAgIGNvbnN0IGZpbmFsS2V5TWFwID0gbXVsdGlrZXlNYXAgfHwgc3RyaWN0TXVsdGlNYXA7XG4gICAgbGV0IGZvdW5kVmFsID0gZmFsc2U7XG4gICAgaWYgKEFycmF5LmlzQXJyYXkoZmluYWxLZXlNYXApKSB7XG4gICAgICBmaW5hbEtleU1hcC5zb21lKChtYXApID0+IHtcbiAgICAgICAgaWYgKCFtYXAuc291cmNlVmFsIHx8ICFpc0RlZmluZWRBbmROb3ROdWxsKG1hcC5kZXN0VmFsKSB8fCAhQXJyYXkuaXNBcnJheShtYXAuc291cmNlVmFsKSkge1xuICAgICAgICAgIGxvZ2dlci53YXJuKCdtdWx0aWtleU1hcCBza2lwcGVkOiBzb3VyY2VWYWwgYW5kIGRlc3RWYWwgbXVzdCBiZSBvZiB2YWxpZCB0eXBlJyk7XG4gICAgICAgICAgZm91bmRWYWwgPSB0cnVlO1xuICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKG1hcC5zb3VyY2VWYWwuaW5jbHVkZXMoU3RyaW5nKGZvcm1hdHRlZFZhbCkpKSB7XG4gICAgICAgICAgY3VyRm9ybWF0dGVkVmFsID0gbWFwLmRlc3RWYWw7XG4gICAgICAgICAgZm91bmRWYWwgPSB0cnVlO1xuICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgfSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGxvZ2dlci53YXJuKCdtdWx0aWtleU1hcCBza2lwcGVkOiBtdWx0aWtleU1hcCBtdXN0IGJlIGFuIGFycmF5Jyk7XG4gICAgfVxuICAgIGlmICghZm91bmRWYWwpIHtcbiAgICAgIGlmIChzdHJpY3RNdWx0aU1hcCkge1xuICAgICAgICB0aHJvdyBuZXcgSW5zdHJ1bWVudGF0aW9uRXJyb3IoYEludmFsaWQgZW50cnkgZm9yIGtleSAke2Rlc3RLZXl9YCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjdXJGb3JtYXR0ZWRWYWwgPSB1bmRlZmluZWQ7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiBjdXJGb3JtYXR0ZWRWYWw7XG4gIH1cblxuICAvKipcbiAgICogRGVzY3JpcHRpb246IFRoaXMgbWV0aG9kIGlzIHVzZWQgdG8gZ2V0IHRoZSB2YWx1ZSBmcm9tIHRoZSBtZXNzYWdlXG4gICAqIFx0XHRcdFx0XHRcdCAgYmFzZWQgb24gdGhlIHNvdXJjZSBrZXlzXG4gICAqIEBwYXJhbSBtZXNzYWdlIC0gdGhlIG1lc3NhZ2Ugb2JqZWN0XG4gICAqIEBwYXJhbSBzb3VyY2VLZXlzIC0gdGhlIHNvdXJjZSBrZXlzIHRvIGJlIHVzZWQgdG8gZ2V0IHRoZSB2YWx1ZSBmcm9tIHRoZSBtZXNzYWdlXG4gICAqIEByZXR1cm5zICB7YW55fSAtIHJldHVybnMgdGhlIHZhbHVlIG9mIHRoZSBzb3VyY2Uga2V5XG4gICAqICAgICAgICAgICAgIGlmIHRoZSBzb3VyY2Uga2V5IGlzIG5vdCBwcmVzZW50IGluIHRoZSBtZXNzYWdlXG4gICAqICAgICAgICAgICAgcmV0dXJucyBudWxsXG4gICAqL1xuICBwdWJsaWMgc3RhdGljIGdldFZhbHVlRnJvbU1lc3NhZ2UoXG4gICAgbWVzc2FnZTogUnVkZGVyU3RhY2tFdmVudCxcbiAgICBzb3VyY2VLZXlzOiAoc3RyaW5nIHwgb2JqZWN0KVtdIHwgc3RyaW5nIHwgb2JqZWN0LFxuICApOiBhbnkge1xuICAgIGlmIChBcnJheS5pc0FycmF5KHNvdXJjZUtleXMpICYmIHNvdXJjZUtleXMubGVuZ3RoID4gMCkge1xuICAgICAgaWYgKHNvdXJjZUtleXMubGVuZ3RoID09PSAxKSB7XG4gICAgICAgIGxvZ2dlci53YXJuKCdMaXN0IHdpdGggc2luZ2xlIGVsZW1lbnQgaXMgbm90IGlkZWFsLiBVc2UgaXQgYXMgc3RyaW5nIGluc3RlYWQnKTtcbiAgICAgIH1cbiAgICAgIC8vIGdvdCB0aGUgcG9zc2libGUgc291cmNlS2V5c1xuICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIG5vLXJlc3RyaWN0ZWQtc3ludGF4XG4gICAgICBmb3IgKGNvbnN0IHNvdXJjZUtleSBvZiBzb3VyY2VLZXlzKSB7XG4gICAgICAgIGxldCB2YWw6IG51bWJlciB8IHN0cmluZyB8IGJvb2xlYW4gfCBBcnJheTxhbnk+IHwgT2JqZWN0IHwgbnVsbDtcbiAgICAgICAgLy8gaWYgdGhlIHNvdXJjZUtleSBpcyBhbiBvYmplY3Qgd2UgZXhwZWN0IGl0IHRvIGJlIGEgb3BlcmF0aW9uXG4gICAgICAgIGlmICh0eXBlb2Ygc291cmNlS2V5ID09PSAnb2JqZWN0Jykge1xuICAgICAgICAgIHZhbCA9IEpzb25VdGlscy5oYW5kbGVTb3VyY2VLZXlzT3BlcmF0aW9uKHtcbiAgICAgICAgICAgIG1lc3NhZ2UsXG4gICAgICAgICAgICBvcGVyYXRpb25PYmplY3Q6IHNvdXJjZUtleSBhcyBPcGVyYXRpb24sXG4gICAgICAgICAgfSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgdmFsID0gZ2V0KG1lc3NhZ2UsIHNvdXJjZUtleSk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHZhbCB8fCB2YWwgPT09IGZhbHNlIHx8IHZhbCA9PT0gMCkge1xuICAgICAgICAgIC8vIHJldHVybiBvbmx5IGlmIHRoZSB2YWx1ZSBpcyB2YWxpZC5cbiAgICAgICAgICAvLyBlbHNlIGxvb2sgZm9yIG5leHQgcG9zc2libGUgc291cmNlIGluIHByZWNlZGVuY2VcbiAgICAgICAgICByZXR1cm4gdmFsO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfSBlbHNlIGlmICh0eXBlb2Ygc291cmNlS2V5cyA9PT0gJ3N0cmluZycpIHtcbiAgICAgIC8vIGdvdCBhIHNpbmdsZSBrZXlcbiAgICAgIC8vIC0gd2UgZG9uJ3QgbmVlZCB0byBpdGVyYXRlIG92ZXIgYSBsb29wIGZvciBhIHNpbmdsZSBwb3NzaWJsZSB2YWx1ZVxuICAgICAgcmV0dXJuIGdldChtZXNzYWdlLCBzb3VyY2VLZXlzKTtcbiAgICB9IGVsc2UgaWYgKHR5cGVvZiBzb3VyY2VLZXlzID09PSAnb2JqZWN0Jykge1xuICAgICAgLy8gaWYgdGhlIHNvdXJjZUtleSBpcyBhbiBvYmplY3Qgd2UgZXhwZWN0IGl0IHRvIGJlIGEgb3BlcmF0aW9uXG4gICAgICByZXR1cm4gSnNvblV0aWxzLmhhbmRsZVNvdXJjZUtleXNPcGVyYXRpb24oe1xuICAgICAgICBtZXNzYWdlLFxuICAgICAgICBvcGVyYXRpb25PYmplY3Q6IHNvdXJjZUtleXMgYXMgeyBvcGVyYXRpb246IHN0cmluZzsgYXJnczogYW55W10gfSxcbiAgICAgIH0pO1xuICAgIH0gZWxzZSB7XG4gICAgICAvLyB3cm9uZyBzb3VyY2VLZXkgdHlwZS4gYWJvcnRcbiAgICAgIC8vIERFVkVMT1BFUiBFUlJPUlxuICAgICAgdGhyb3cgbmV3IFBsYXRmb3JtRXJyb3IoJ1dyb25nIHNvdXJjZUtleSB0eXBlIG9yIGJsYW5rIHNvdXJjZUtleSBhcnJheScpO1xuICAgIH1cbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZXNjcmlwdGlvbjogVGhpcyBtZXRob2QgaXMgdXNlZCB0byBnZXQgdGhlIHZhbHVlIGZyb20gdGhlIG1lc3NhZ2VcbiAgICogICAgICAgICAgICAgICBiYXNlZCBvbiB0aGUgc291cmNlIGtleVxuICAgKiBAcGFyYW0gbWVzc2FnZSAtIHRoZSBtZXNzYWdlIG9iamVjdFxuICAgKiBAcGFyYW0gc291cmNlS2V5IC0gdGhlIHNvdXJjZSBrZXkgdG8gYmUgdXNlZCB0byBnZXQgdGhlIHZhbHVlIGZyb20gdGhlIG1lc3NhZ2VcbiAgICogQHJldHVybnMgIHthbnl9IC0gcmV0dXJucyB0aGUgdmFsdWUgb2YgdGhlIHNvdXJjZSBrZXlcbiAgICogICAgICAgICAgICBpZiB0aGUgc291cmNlIGtleSBpcyBub3QgcHJlc2VudCBpbiB0aGUgbWVzc2FnZVxuICAgKiAgICAgICAgICAgcmV0dXJucyBudWxsXG4gICAqL1xuICBwdWJsaWMgc3RhdGljIGdldEZpZWxkVmFsdWVGcm9tTWVzc2FnZShtZXNzYWdlOiBSdWRkZXJTdGFja0V2ZW50LCBzb3VyY2VLZXk6IHN0cmluZyk6IGFueSB7XG4gICAgY29uc3Qgc291cmNlS2V5TWFwID0gSnNvblV0aWxzLk1FU1NBR0VfTUFQUElOR1tzb3VyY2VLZXldO1xuICAgIGlmIChzb3VyY2VLZXlNYXApIHtcbiAgICAgIHJldHVybiBKc29uVXRpbHMuZ2V0VmFsdWVGcm9tTWVzc2FnZShtZXNzYWdlLCBzb3VyY2VLZXlNYXApO1xuICAgIH1cbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZXNjcmlwdGlvbjogVGhpcyBtZXRob2QgaXMgdXNlZCB0byBjb25zdHJ1Y3QgdGhlIHBheWxvYWQgYmFzZWQgb24gdGhlIG1hcHBpbmdKc29uXG4gICAqIEBwYXJhbSBtZXNzYWdlIC0gdGhlIG1lc3NhZ2Ugb2JqZWN0XG4gICAqIEBwYXJhbSBtYXBwaW5nSnNvbiAtIHRoZSBtYXBwaW5nSnNvbiB0byBiZSB1c2VkIHRvIGNvbnN0cnVjdCB0aGUgcGF5bG9hZFxuICAgKiBAcGFyYW0gZGVzdGluYXRpb25OYW1lIC0gdGhlIGRlc3RpbmF0aW9uIG5hbWUgdG8gYmUgdXNlZCB0byBnZXQgdGhlIG1hcHBpbmdKc29uXG4gICAqIEByZXR1cm5zICB7YW55fSAtIHJldHVybnMgdGhlIHBheWxvYWQgY29uc3RydWN0ZWQgYmFzZWQgb24gdGhlIG1hcHBpbmdKc29uXG4gICAqICAgICAgICAgICBpZiB0aGUgbWFwcGluZ0pzb24gaXMgbm90IHByZXNlbnQgaW4gdGhlIG1lc3NhZ2Ugb3IgYmxhbmtcbiAgICogICAgICAgICAgIHJldHVybnMgbnVsbFxuICAgKi9cbiAgcHVibGljIHN0YXRpYyBjb25zdHJ1Y3RQYXlsb2FkKFxuICAgIG1lc3NhZ2U6IFJ1ZGRlclN0YWNrRXZlbnQsXG4gICAgbWFwcGluZ0pzb246IGFueVtdLFxuICAgIGRlc3RpbmF0aW9uTmFtZTogc3RyaW5nID0gJycsXG4gICk6IGFueSB7XG4gICAgLy8gTWFwcGluZyBKU09OIHNob3VsZCBiZSBhbiBhcnJheVxuICAgIGlmIChBcnJheS5pc0FycmF5KG1hcHBpbmdKc29uKSAmJiBtYXBwaW5nSnNvbi5sZW5ndGggPiAwKSB7XG4gICAgICAvLyAtIGNvbnN0cnVjdCBhIGJsYW5rIHBheWxvYWQgYW5kIHJldHVybiBhdCB0aGUgZW5kXG4gICAgICAvLyAtIGlmIHlvdSB0byBuZWVkIG1lcmdlIG11bHRpcGxlIGNvbnN0cnVjdFBheWxvYWQgZG8gaXQgb24gdGhlIHRyYW5zZm9ybWVyIGNvZGVcbiAgICAgIC8vIC0gLSB3aWxsIGdpdmUgYSBjbGVhbmVyIGFwcHJvYWNoXG4gICAgICAvLyAtIC0geW91IGRvbid0IG5lZWQgdG8gaXRlcmF0ZSBvdmVyIG11bHRpcGxlIGxvb3BzIHRvIGNvbnN0cnVjdCBhIHBheWxvYWQgZm9yIGEgc2luZ2xlIGV2ZW50XG4gICAgICBjb25zdCBwYXlsb2FkID0ge307XG5cbiAgICAgIG1hcHBpbmdKc29uLmZvckVhY2goKG1hcHBpbmcpID0+IHtcbiAgICAgICAgY29uc3Qge1xuICAgICAgICAgIHNvdXJjZUtleXMsXG4gICAgICAgICAgZGVzdEtleSxcbiAgICAgICAgICByZXF1aXJlZCxcbiAgICAgICAgICBtZXRhZGF0YSxcbiAgICAgICAgICBzb3VyY2VGcm9tR2VuZXJpY01hcCxcbiAgICAgICAgfToge1xuICAgICAgICAgIHNvdXJjZUtleXM6IChzdHJpbmcgfCBvYmplY3QpW10gfCBzdHJpbmcgfCBvYmplY3Q7XG4gICAgICAgICAgZGVzdEtleTogc3RyaW5nO1xuICAgICAgICAgIHJlcXVpcmVkOiBib29sZWFuO1xuICAgICAgICAgIG1ldGFkYXRhOiBNYXBwaW5nTWV0YWRhdGE7XG4gICAgICAgICAgc291cmNlRnJvbUdlbmVyaWNNYXA6IGJvb2xlYW47XG4gICAgICAgIH0gPSBtYXBwaW5nO1xuICAgICAgICAvLyBnZXQgdGhlIHZhbHVlIGZyb20gZXZlbnQsIHBhc3Mgc291cmNlRnJvbUdlbmVyaWNNYXAgaW4gdGhlIG1hcHBpbmcgdG8gZm9yY2UgdGhpcyB0byB0YWtlIHRoZVxuICAgICAgICAvLyBzb3VyY2VrZXlzIGZyb20gR2VuZXJpY0ZpZWxkTWFwcGluZywgZWxzZSB0YWtlIHRoZSBzb3VyY2VLZXlzIGZyb20gc3BlY2lmaWMgZGVzdGluYXRpb24gbWFwcGluZyBzb3VyY2VLZXlzXG4gICAgICAgIGNvbnN0IGludGVncmF0aW9uc09iaiA9IGlzRW1wdHkoZGVzdGluYXRpb25OYW1lKVxuICAgICAgICAgID8gU2VtYW50aWNVdGlsLmdldEludGVncmF0aW9uc09iaihtZXNzYWdlLCBkZXN0aW5hdGlvbk5hbWUpXG4gICAgICAgICAgOiBudWxsO1xuICAgICAgICBjb25zdCB2YWx1ZSA9IEpzb25VdGlscy5oYW5kbGVNZXRhZGF0YUZvclZhbHVlKFxuICAgICAgICAgIHNvdXJjZUZyb21HZW5lcmljTWFwXG4gICAgICAgICAgICA/IHRoaXMuZ2V0RmllbGRWYWx1ZUZyb21NZXNzYWdlKG1lc3NhZ2UsIHNvdXJjZUtleXMgYXMgc3RyaW5nKVxuICAgICAgICAgICAgOiB0aGlzLmdldFZhbHVlRnJvbU1lc3NhZ2UobWVzc2FnZSwgc291cmNlS2V5cyksXG4gICAgICAgICAgbWV0YWRhdGEsXG4gICAgICAgICAgZGVzdEtleSxcbiAgICAgICAgICBpbnRlZ3JhdGlvbnNPYmosXG4gICAgICAgICk7XG5cbiAgICAgICAgaWYgKHZhbHVlIHx8IHZhbHVlID09PSAwIHx8IHZhbHVlID09PSBmYWxzZSkge1xuICAgICAgICAgIGlmIChkZXN0S2V5KSB7XG4gICAgICAgICAgICAvLyBzZXQgdGhlIHZhbHVlIG9ubHkgaWYgY29ycmVjdFxuICAgICAgICAgICAgc2V0KHBheWxvYWQsIGRlc3RLZXksIHZhbHVlKTtcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgLy8gdG8gc2V0IHRvIHJvb3QgYW5kIGZsYXR0ZW4gbGF0ZXJcbiAgICAgICAgICAgIHBheWxvYWRbJyddID0gdmFsdWU7XG4gICAgICAgICAgfVxuICAgICAgICB9IGVsc2UgaWYgKHJlcXVpcmVkKSB7XG4gICAgICAgICAgLy8gdGhyb3cgZXJyb3IgaWYgcmVxaXJlZCB2YWx1ZSBpcyBtaXNzaW5nXG4gICAgICAgICAgdGhyb3cgbmV3IEluc3RydW1lbnRhdGlvbkVycm9yKFxuICAgICAgICAgICAgYE1pc3NpbmcgcmVxdWlyZWQgdmFsdWUgZnJvbSAke0pTT04uc3RyaW5naWZ5KHNvdXJjZUtleXMpfWAsXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfSk7XG5cbiAgICAgIHJldHVybiBwYXlsb2FkO1xuICAgIH1cblxuICAgIC8vIGludmFsaWQgbWFwcGluZ0pzb25cbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIHB1YmxpYyBzdGF0aWMgc3RyaW5naWZ5KHZhbHVlOiB1bmtub3duKTogc3RyaW5nIHtcbiAgICByZXR1cm4gSlNPTi5zdHJpbmdpZnkodmFsdWUpO1xuICB9XG59XG4iXX0=
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=json.test.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"json.test.d.ts","sourceRoot":"","sources":["../../src/utils/json.test.ts"],"names":[],"mappings":""}