@the-trybe/formula-engine 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/.claude/settings.local.json +6 -0
  2. package/PRD_FORMULA_ENGINE.md +1863 -0
  3. package/README.md +382 -0
  4. package/dist/decimal-utils.d.ts +180 -0
  5. package/dist/decimal-utils.js +355 -0
  6. package/dist/dependency-extractor.d.ts +20 -0
  7. package/dist/dependency-extractor.js +103 -0
  8. package/dist/dependency-graph.d.ts +60 -0
  9. package/dist/dependency-graph.js +252 -0
  10. package/dist/errors.d.ts +161 -0
  11. package/dist/errors.js +260 -0
  12. package/dist/evaluator.d.ts +51 -0
  13. package/dist/evaluator.js +494 -0
  14. package/dist/formula-engine.d.ts +79 -0
  15. package/dist/formula-engine.js +355 -0
  16. package/dist/functions.d.ts +3 -0
  17. package/dist/functions.js +720 -0
  18. package/dist/index.d.ts +10 -0
  19. package/dist/index.js +61 -0
  20. package/dist/lexer.d.ts +25 -0
  21. package/dist/lexer.js +357 -0
  22. package/dist/parser.d.ts +32 -0
  23. package/dist/parser.js +372 -0
  24. package/dist/types.d.ts +228 -0
  25. package/dist/types.js +62 -0
  26. package/jest.config.js +23 -0
  27. package/package.json +35 -0
  28. package/src/decimal-utils.ts +408 -0
  29. package/src/dependency-extractor.ts +117 -0
  30. package/src/dependency-graph.test.ts +238 -0
  31. package/src/dependency-graph.ts +288 -0
  32. package/src/errors.ts +296 -0
  33. package/src/evaluator.ts +604 -0
  34. package/src/formula-engine.test.ts +660 -0
  35. package/src/formula-engine.ts +430 -0
  36. package/src/functions.ts +770 -0
  37. package/src/index.ts +103 -0
  38. package/src/lexer.test.ts +288 -0
  39. package/src/lexer.ts +394 -0
  40. package/src/parser.test.ts +349 -0
  41. package/src/parser.ts +449 -0
  42. package/src/types.ts +347 -0
  43. package/tsconfig.json +29 -0
@@ -0,0 +1,494 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Evaluator = void 0;
4
+ const decimal_utils_1 = require("./decimal-utils");
5
+ const errors_1 = require("./errors");
6
+ const parser_1 = require("./parser");
7
+ class Evaluator {
8
+ constructor(decimalUtils, functions, config) {
9
+ this.recursionDepth = 0;
10
+ this.iterationCount = 0;
11
+ this.accessedVariables = new Set();
12
+ this.parser = new parser_1.Parser();
13
+ this.decimalUtils = decimalUtils;
14
+ this.functions = functions;
15
+ this.strictMode = config?.strictMode ?? true;
16
+ this.securityConfig = config?.security ?? {};
17
+ }
18
+ /**
19
+ * Evaluate an expression string
20
+ */
21
+ evaluate(expression, context) {
22
+ const startTime = Date.now();
23
+ this.accessedVariables = new Set();
24
+ this.recursionDepth = 0;
25
+ this.iterationCount = 0;
26
+ try {
27
+ const ast = this.parser.parse(expression);
28
+ const value = this.evaluateNode(ast, context);
29
+ return {
30
+ value,
31
+ success: true,
32
+ executionTimeMs: Date.now() - startTime,
33
+ accessedVariables: new Set(this.accessedVariables),
34
+ };
35
+ }
36
+ catch (error) {
37
+ return {
38
+ value: null,
39
+ success: false,
40
+ error: error,
41
+ executionTimeMs: Date.now() - startTime,
42
+ accessedVariables: new Set(this.accessedVariables),
43
+ };
44
+ }
45
+ }
46
+ /**
47
+ * Evaluate an AST node
48
+ */
49
+ evaluateNode(node, context) {
50
+ this.checkRecursionLimit();
51
+ switch (node.type) {
52
+ case 'DecimalLiteral':
53
+ return this.decimalUtils.from(node.value);
54
+ case 'NumberLiteral':
55
+ return node.value;
56
+ case 'StringLiteral':
57
+ return node.value;
58
+ case 'BooleanLiteral':
59
+ return node.value;
60
+ case 'NullLiteral':
61
+ return null;
62
+ case 'ArrayLiteral':
63
+ return node.elements.map(el => this.evaluateNode(el, context));
64
+ case 'VariableReference':
65
+ return this.evaluateVariable(node.prefix, node.name, context);
66
+ case 'BinaryOperation':
67
+ return this.evaluateBinaryOperation(node, context);
68
+ case 'UnaryOperation':
69
+ return this.evaluateUnaryOperation(node, context);
70
+ case 'ConditionalExpression':
71
+ return this.evaluateConditional(node, context);
72
+ case 'FunctionCall':
73
+ return this.evaluateFunctionCall(node, context);
74
+ case 'MemberAccess':
75
+ return this.evaluateMemberAccess(node, context);
76
+ case 'IndexAccess':
77
+ return this.evaluateIndexAccess(node, context);
78
+ default:
79
+ throw new Error(`Unknown node type: ${node.type}`);
80
+ }
81
+ }
82
+ checkRecursionLimit() {
83
+ this.recursionDepth++;
84
+ const limit = this.securityConfig.maxRecursionDepth ?? 100;
85
+ if (this.recursionDepth > limit) {
86
+ throw new errors_1.MaxRecursionError(limit);
87
+ }
88
+ }
89
+ checkIterationLimit() {
90
+ this.iterationCount++;
91
+ const limit = this.securityConfig.maxIterations ?? 10000;
92
+ if (this.iterationCount > limit) {
93
+ throw new errors_1.MaxIterationsError(limit);
94
+ }
95
+ }
96
+ evaluateVariable(prefix, name, context) {
97
+ this.accessedVariables.add(name);
98
+ if (prefix === '$') {
99
+ // Local variable
100
+ if (name in context.variables) {
101
+ const value = context.variables[name];
102
+ return this.maybeConvertToDecimal(value);
103
+ }
104
+ // Check if it's a special iteration variable ($it)
105
+ if (name === 'it' && context.extra && '_currentItem' in context.extra) {
106
+ return context.extra._currentItem;
107
+ }
108
+ if (this.strictMode) {
109
+ throw new errors_1.UndefinedVariableError(name, '');
110
+ }
111
+ return null;
112
+ }
113
+ else {
114
+ // Context variable (@)
115
+ if (context.extra && name in context.extra) {
116
+ return context.extra[name];
117
+ }
118
+ if (this.strictMode) {
119
+ throw new errors_1.UndefinedVariableError(`@${name}`, '');
120
+ }
121
+ return null;
122
+ }
123
+ }
124
+ maybeConvertToDecimal(value) {
125
+ if (typeof value === 'number' && !isNaN(value) && isFinite(value)) {
126
+ return this.decimalUtils.from(value);
127
+ }
128
+ return value;
129
+ }
130
+ evaluateBinaryOperation(node, context) {
131
+ const { operator } = node;
132
+ // Short-circuit evaluation for logical operators
133
+ if (operator === '&&') {
134
+ const left = this.evaluateNode(node.left, context);
135
+ if (!this.toBool(left))
136
+ return false;
137
+ return this.toBool(this.evaluateNode(node.right, context));
138
+ }
139
+ if (operator === '||') {
140
+ const left = this.evaluateNode(node.left, context);
141
+ if (this.toBool(left))
142
+ return true;
143
+ return this.toBool(this.evaluateNode(node.right, context));
144
+ }
145
+ const left = this.evaluateNode(node.left, context);
146
+ const right = this.evaluateNode(node.right, context);
147
+ switch (operator) {
148
+ // Arithmetic
149
+ case '+':
150
+ return this.add(left, right);
151
+ case '-':
152
+ return this.subtract(left, right);
153
+ case '*':
154
+ return this.multiply(left, right);
155
+ case '/':
156
+ return this.divide(left, right);
157
+ case '%':
158
+ return this.modulo(left, right);
159
+ case '^':
160
+ return this.power(left, right);
161
+ // Comparison
162
+ case '==':
163
+ return this.equals(left, right);
164
+ case '!=':
165
+ return !this.equals(left, right);
166
+ case '<':
167
+ return this.lessThan(left, right);
168
+ case '>':
169
+ return this.greaterThan(left, right);
170
+ case '<=':
171
+ return this.lessThanOrEqual(left, right);
172
+ case '>=':
173
+ return this.greaterThanOrEqual(left, right);
174
+ default:
175
+ throw new errors_1.InvalidOperationError(operator, [this.typeOf(left), this.typeOf(right)]);
176
+ }
177
+ }
178
+ evaluateUnaryOperation(node, context) {
179
+ const operand = this.evaluateNode(node.operand, context);
180
+ switch (node.operator) {
181
+ case '-':
182
+ return this.negate(operand);
183
+ case '!':
184
+ return !this.toBool(operand);
185
+ default:
186
+ throw new errors_1.InvalidOperationError(node.operator, [this.typeOf(operand)]);
187
+ }
188
+ }
189
+ evaluateConditional(node, context) {
190
+ const condition = this.evaluateNode(node.condition, context);
191
+ if (this.toBool(condition)) {
192
+ return this.evaluateNode(node.consequent, context);
193
+ }
194
+ return this.evaluateNode(node.alternate, context);
195
+ }
196
+ evaluateFunctionCall(node, context) {
197
+ const fnName = node.name.toUpperCase();
198
+ const fn = this.functions.get(fnName);
199
+ if (!fn) {
200
+ throw new errors_1.UndefinedFunctionError(fnName);
201
+ }
202
+ // Validate argument count
203
+ if (node.arguments.length < fn.minArgs) {
204
+ throw new errors_1.ArgumentCountError(fnName, { min: fn.minArgs, max: fn.maxArgs }, node.arguments.length);
205
+ }
206
+ if (fn.maxArgs !== -1 && node.arguments.length > fn.maxArgs) {
207
+ throw new errors_1.ArgumentCountError(fnName, { min: fn.minArgs, max: fn.maxArgs }, node.arguments.length);
208
+ }
209
+ // Handle special functions that need AST nodes (for iteration)
210
+ if (fnName === 'SUM' && node.arguments.length === 2) {
211
+ return this.evaluateSumWithExpression(node.arguments, context);
212
+ }
213
+ if (fnName === 'FILTER') {
214
+ return this.evaluateFilter(node.arguments, context);
215
+ }
216
+ if (fnName === 'MAP') {
217
+ return this.evaluateMap(node.arguments, context);
218
+ }
219
+ // Evaluate arguments
220
+ const args = node.arguments.map(arg => this.evaluateNode(arg, context));
221
+ // Call the function
222
+ return fn.implementation(args, context, this);
223
+ }
224
+ evaluateSumWithExpression(args, context) {
225
+ const array = this.evaluateNode(args[0], context);
226
+ if (!Array.isArray(array)) {
227
+ throw new Error('SUM first argument must be an array');
228
+ }
229
+ const expression = args[1];
230
+ let sum = this.decimalUtils.zero();
231
+ for (const item of array) {
232
+ this.checkIterationLimit();
233
+ const itemContext = {
234
+ ...context,
235
+ extra: {
236
+ ...context.extra,
237
+ _currentItem: item,
238
+ },
239
+ variables: {
240
+ ...context.variables,
241
+ it: item,
242
+ },
243
+ };
244
+ const value = this.evaluateNode(expression, itemContext);
245
+ if (this.isNumeric(value)) {
246
+ sum = this.decimalUtils.add(sum, this.toDecimal(value));
247
+ }
248
+ }
249
+ return sum;
250
+ }
251
+ evaluateFilter(args, context) {
252
+ const array = this.evaluateNode(args[0], context);
253
+ if (!Array.isArray(array)) {
254
+ throw new Error('FILTER first argument must be an array');
255
+ }
256
+ const condition = args[1];
257
+ const result = [];
258
+ for (const item of array) {
259
+ this.checkIterationLimit();
260
+ const itemContext = {
261
+ ...context,
262
+ extra: {
263
+ ...context.extra,
264
+ _currentItem: item,
265
+ },
266
+ variables: {
267
+ ...context.variables,
268
+ it: item,
269
+ },
270
+ };
271
+ const keep = this.evaluateNode(condition, itemContext);
272
+ if (this.toBool(keep)) {
273
+ result.push(item);
274
+ }
275
+ }
276
+ return result;
277
+ }
278
+ evaluateMap(args, context) {
279
+ const array = this.evaluateNode(args[0], context);
280
+ if (!Array.isArray(array)) {
281
+ throw new Error('MAP first argument must be an array');
282
+ }
283
+ const expression = args[1];
284
+ const result = [];
285
+ for (const item of array) {
286
+ this.checkIterationLimit();
287
+ const itemContext = {
288
+ ...context,
289
+ extra: {
290
+ ...context.extra,
291
+ _currentItem: item,
292
+ },
293
+ variables: {
294
+ ...context.variables,
295
+ it: item,
296
+ },
297
+ };
298
+ result.push(this.evaluateNode(expression, itemContext));
299
+ }
300
+ return result;
301
+ }
302
+ evaluateMemberAccess(node, context) {
303
+ const object = this.evaluateNode(node.object, context);
304
+ if (object === null || object === undefined) {
305
+ if (this.strictMode) {
306
+ throw new errors_1.PropertyAccessError(node.property, 'null');
307
+ }
308
+ return null;
309
+ }
310
+ if (typeof object !== 'object') {
311
+ throw new errors_1.PropertyAccessError(node.property, typeof object);
312
+ }
313
+ const value = object[node.property];
314
+ return this.maybeConvertToDecimal(value);
315
+ }
316
+ evaluateIndexAccess(node, context) {
317
+ const object = this.evaluateNode(node.object, context);
318
+ const index = this.evaluateNode(node.index, context);
319
+ if (object === null || object === undefined) {
320
+ if (this.strictMode) {
321
+ throw new errors_1.IndexAccessError(index, 'null');
322
+ }
323
+ return null;
324
+ }
325
+ if (Array.isArray(object)) {
326
+ const idx = this.toNumber(index);
327
+ if (idx < 0 || idx >= object.length) {
328
+ return null;
329
+ }
330
+ return this.maybeConvertToDecimal(object[idx]);
331
+ }
332
+ if (typeof object === 'object') {
333
+ const key = String(index);
334
+ const value = object[key];
335
+ return this.maybeConvertToDecimal(value);
336
+ }
337
+ throw new errors_1.IndexAccessError(index, typeof object);
338
+ }
339
+ // ============================================================================
340
+ // Helper methods
341
+ // ============================================================================
342
+ isNumeric(value) {
343
+ return value instanceof decimal_utils_1.Decimal || typeof value === 'number';
344
+ }
345
+ toDecimal(value) {
346
+ if (value instanceof decimal_utils_1.Decimal)
347
+ return value;
348
+ if (typeof value === 'number')
349
+ return this.decimalUtils.from(value);
350
+ if (typeof value === 'string')
351
+ return this.decimalUtils.from(value);
352
+ throw new errors_1.InvalidOperationError('toDecimal', [this.typeOf(value)]);
353
+ }
354
+ toNumber(value) {
355
+ if (value instanceof decimal_utils_1.Decimal)
356
+ return value.toNumber();
357
+ if (typeof value === 'number')
358
+ return value;
359
+ if (typeof value === 'string')
360
+ return parseFloat(value);
361
+ throw new errors_1.InvalidOperationError('toNumber', [this.typeOf(value)]);
362
+ }
363
+ toBool(value) {
364
+ if (value instanceof decimal_utils_1.Decimal)
365
+ return !value.isZero();
366
+ if (typeof value === 'boolean')
367
+ return value;
368
+ if (value === null || value === undefined)
369
+ return false;
370
+ if (typeof value === 'number')
371
+ return value !== 0;
372
+ if (typeof value === 'string')
373
+ return value.length > 0;
374
+ if (Array.isArray(value))
375
+ return value.length > 0;
376
+ return true;
377
+ }
378
+ typeOf(value) {
379
+ if (value === null)
380
+ return 'null';
381
+ if (value instanceof decimal_utils_1.Decimal)
382
+ return 'decimal';
383
+ if (Array.isArray(value))
384
+ return 'array';
385
+ return typeof value;
386
+ }
387
+ add(left, right) {
388
+ // String concatenation
389
+ if (typeof left === 'string' || typeof right === 'string') {
390
+ const leftStr = left instanceof decimal_utils_1.Decimal ? left.toString() : String(left);
391
+ const rightStr = right instanceof decimal_utils_1.Decimal ? right.toString() : String(right);
392
+ return leftStr + rightStr;
393
+ }
394
+ // Numeric addition
395
+ if (this.isNumeric(left) && this.isNumeric(right)) {
396
+ return this.decimalUtils.add(this.toDecimal(left), this.toDecimal(right));
397
+ }
398
+ throw new errors_1.InvalidOperationError('+', [this.typeOf(left), this.typeOf(right)]);
399
+ }
400
+ subtract(left, right) {
401
+ if (!this.isNumeric(left) || !this.isNumeric(right)) {
402
+ throw new errors_1.InvalidOperationError('-', [this.typeOf(left), this.typeOf(right)]);
403
+ }
404
+ return this.decimalUtils.subtract(this.toDecimal(left), this.toDecimal(right));
405
+ }
406
+ multiply(left, right) {
407
+ if (!this.isNumeric(left) || !this.isNumeric(right)) {
408
+ throw new errors_1.InvalidOperationError('*', [this.typeOf(left), this.typeOf(right)]);
409
+ }
410
+ return this.decimalUtils.multiply(this.toDecimal(left), this.toDecimal(right));
411
+ }
412
+ divide(left, right) {
413
+ if (!this.isNumeric(left) || !this.isNumeric(right)) {
414
+ throw new errors_1.InvalidOperationError('/', [this.typeOf(left), this.typeOf(right)]);
415
+ }
416
+ const divisor = this.toDecimal(right);
417
+ if (divisor.isZero()) {
418
+ throw new errors_1.DivisionByZeroError();
419
+ }
420
+ return this.decimalUtils.divide(this.toDecimal(left), divisor);
421
+ }
422
+ modulo(left, right) {
423
+ if (!this.isNumeric(left) || !this.isNumeric(right)) {
424
+ throw new errors_1.InvalidOperationError('%', [this.typeOf(left), this.typeOf(right)]);
425
+ }
426
+ const divisor = this.toDecimal(right);
427
+ if (divisor.isZero()) {
428
+ throw new errors_1.DivisionByZeroError();
429
+ }
430
+ return this.decimalUtils.modulo(this.toDecimal(left), divisor);
431
+ }
432
+ power(left, right) {
433
+ if (!this.isNumeric(left) || !this.isNumeric(right)) {
434
+ throw new errors_1.InvalidOperationError('^', [this.typeOf(left), this.typeOf(right)]);
435
+ }
436
+ return this.decimalUtils.power(this.toDecimal(left), this.toNumber(right));
437
+ }
438
+ negate(value) {
439
+ if (!this.isNumeric(value)) {
440
+ throw new errors_1.InvalidOperationError('-', [this.typeOf(value)]);
441
+ }
442
+ return this.decimalUtils.negate(this.toDecimal(value));
443
+ }
444
+ equals(left, right) {
445
+ if (left instanceof decimal_utils_1.Decimal && right instanceof decimal_utils_1.Decimal) {
446
+ return left.equals(right);
447
+ }
448
+ if (left instanceof decimal_utils_1.Decimal && typeof right === 'number') {
449
+ return left.equals(this.decimalUtils.from(right));
450
+ }
451
+ if (typeof left === 'number' && right instanceof decimal_utils_1.Decimal) {
452
+ return this.decimalUtils.from(left).equals(right);
453
+ }
454
+ return left === right;
455
+ }
456
+ lessThan(left, right) {
457
+ if (this.isNumeric(left) && this.isNumeric(right)) {
458
+ return this.decimalUtils.lessThan(this.toDecimal(left), this.toDecimal(right));
459
+ }
460
+ if (typeof left === 'string' && typeof right === 'string') {
461
+ return left < right;
462
+ }
463
+ throw new errors_1.InvalidOperationError('<', [this.typeOf(left), this.typeOf(right)]);
464
+ }
465
+ greaterThan(left, right) {
466
+ if (this.isNumeric(left) && this.isNumeric(right)) {
467
+ return this.decimalUtils.greaterThan(this.toDecimal(left), this.toDecimal(right));
468
+ }
469
+ if (typeof left === 'string' && typeof right === 'string') {
470
+ return left > right;
471
+ }
472
+ throw new errors_1.InvalidOperationError('>', [this.typeOf(left), this.typeOf(right)]);
473
+ }
474
+ lessThanOrEqual(left, right) {
475
+ if (this.isNumeric(left) && this.isNumeric(right)) {
476
+ return this.decimalUtils.lessThanOrEqual(this.toDecimal(left), this.toDecimal(right));
477
+ }
478
+ if (typeof left === 'string' && typeof right === 'string') {
479
+ return left <= right;
480
+ }
481
+ throw new errors_1.InvalidOperationError('<=', [this.typeOf(left), this.typeOf(right)]);
482
+ }
483
+ greaterThanOrEqual(left, right) {
484
+ if (this.isNumeric(left) && this.isNumeric(right)) {
485
+ return this.decimalUtils.greaterThanOrEqual(this.toDecimal(left), this.toDecimal(right));
486
+ }
487
+ if (typeof left === 'string' && typeof right === 'string') {
488
+ return left >= right;
489
+ }
490
+ throw new errors_1.InvalidOperationError('>=', [this.typeOf(left), this.typeOf(right)]);
491
+ }
492
+ }
493
+ exports.Evaluator = Evaluator;
494
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZhbHVhdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2V2YWx1YXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFRQSxtREFBd0Q7QUFDeEQscUNBVWtCO0FBQ2xCLHFDQUFrQztBQUVsQyxNQUFhLFNBQVM7SUFVcEIsWUFDRSxZQUEwQixFQUMxQixTQUEwQyxFQUMxQyxNQUE0QjtRQVB0QixtQkFBYyxHQUFXLENBQUMsQ0FBQztRQUMzQixtQkFBYyxHQUFXLENBQUMsQ0FBQztRQUMzQixzQkFBaUIsR0FBZ0IsSUFBSSxHQUFHLEVBQUUsQ0FBQztRQU9qRCxJQUFJLENBQUMsTUFBTSxHQUFHLElBQUksZUFBTSxFQUFFLENBQUM7UUFDM0IsSUFBSSxDQUFDLFlBQVksR0FBRyxZQUFZLENBQUM7UUFDakMsSUFBSSxDQUFDLFNBQVMsR0FBRyxTQUFTLENBQUM7UUFDM0IsSUFBSSxDQUFDLFVBQVUsR0FBRyxNQUFNLEVBQUUsVUFBVSxJQUFJLElBQUksQ0FBQztRQUM3QyxJQUFJLENBQUMsY0FBYyxHQUFHLE1BQU0sRUFBRSxRQUFRLElBQUksRUFBRSxDQUFDO0lBQy9DLENBQUM7SUFFRDs7T0FFRztJQUNILFFBQVEsQ0FBQyxVQUFrQixFQUFFLE9BQTBCO1FBQ3JELE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxHQUFHLEVBQUUsQ0FBQztRQUM3QixJQUFJLENBQUMsaUJBQWlCLEdBQUcsSUFBSSxHQUFHLEVBQUUsQ0FBQztRQUNuQyxJQUFJLENBQUMsY0FBYyxHQUFHLENBQUMsQ0FBQztRQUN4QixJQUFJLENBQUMsY0FBYyxHQUFHLENBQUMsQ0FBQztRQUV4QixJQUFJLENBQUM7WUFDSCxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsQ0FBQztZQUMxQyxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQztZQUU5QyxPQUFPO2dCQUNMLEtBQUs7Z0JBQ0wsT0FBTyxFQUFFLElBQUk7Z0JBQ2IsZUFBZSxFQUFFLElBQUksQ0FBQyxHQUFHLEVBQUUsR0FBRyxTQUFTO2dCQUN2QyxpQkFBaUIsRUFBRSxJQUFJLEdBQUcsQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUM7YUFDbkQsQ0FBQztRQUNKLENBQUM7UUFBQyxPQUFPLEtBQUssRUFBRSxDQUFDO1lBQ2YsT0FBTztnQkFDTCxLQUFLLEVBQUUsSUFBSTtnQkFDWCxPQUFPLEVBQUUsS0FBSztnQkFDZCxLQUFLLEVBQUUsS0FBYztnQkFDckIsZUFBZSxFQUFFLElBQUksQ0FBQyxHQUFHLEVBQUUsR0FBRyxTQUFTO2dCQUN2QyxpQkFBaUIsRUFBRSxJQUFJLEdBQUcsQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUM7YUFDbkQsQ0FBQztRQUNKLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxZQUFZLENBQUMsSUFBYSxFQUFFLE9BQTBCO1FBQ3BELElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDO1FBRTNCLFFBQVEsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ2xCLEtBQUssZ0JBQWdCO2dCQUNuQixPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUU1QyxLQUFLLGVBQWU7Z0JBQ2xCLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQztZQUVwQixLQUFLLGVBQWU7Z0JBQ2xCLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQztZQUVwQixLQUFLLGdCQUFnQjtnQkFDbkIsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDO1lBRXBCLEtBQUssYUFBYTtnQkFDaEIsT0FBTyxJQUFJLENBQUM7WUFFZCxLQUFLLGNBQWM7Z0JBQ2pCLE9BQU8sSUFBSSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEVBQUUsRUFBRSxPQUFPLENBQUMsQ0FBQyxDQUFDO1lBRWpFLEtBQUssbUJBQW1CO2dCQUN0QixPQUFPLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFFaEUsS0FBSyxpQkFBaUI7Z0JBQ3BCLE9BQU8sSUFBSSxDQUFDLHVCQUF1QixDQUFDLElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQztZQUVyRCxLQUFLLGdCQUFnQjtnQkFDbkIsT0FBTyxJQUFJLENBQUMsc0JBQXNCLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFDO1lBRXBELEtBQUssdUJBQXVCO2dCQUMxQixPQUFPLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFFakQsS0FBSyxjQUFjO2dCQUNqQixPQUFPLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFFbEQsS0FBSyxjQUFjO2dCQUNqQixPQUFPLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFFbEQsS0FBSyxhQUFhO2dCQUNoQixPQUFPLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFFakQ7Z0JBQ0UsTUFBTSxJQUFJLEtBQUssQ0FBQyxzQkFBdUIsSUFBWSxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7UUFDaEUsQ0FBQztJQUNILENBQUM7SUFFTyxtQkFBbUI7UUFDekIsSUFBSSxDQUFDLGNBQWMsRUFBRSxDQUFDO1FBQ3RCLE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxjQUFjLENBQUMsaUJBQWlCLElBQUksR0FBRyxDQUFDO1FBQzNELElBQUksSUFBSSxDQUFDLGNBQWMsR0FBRyxLQUFLLEVBQUUsQ0FBQztZQUNoQyxNQUFNLElBQUksMEJBQWlCLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDckMsQ0FBQztJQUNILENBQUM7SUFFTyxtQkFBbUI7UUFDekIsSUFBSSxDQUFDLGNBQWMsRUFBRSxDQUFDO1FBQ3RCLE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxjQUFjLENBQUMsYUFBYSxJQUFJLEtBQUssQ0FBQztRQUN6RCxJQUFJLElBQUksQ0FBQyxjQUFjLEdBQUcsS0FBSyxFQUFFLENBQUM7WUFDaEMsTUFBTSxJQUFJLDJCQUFrQixDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RDLENBQUM7SUFDSCxDQUFDO0lBRU8sZ0JBQWdCLENBQ3RCLE1BQWlCLEVBQ2pCLElBQVksRUFDWixPQUEwQjtRQUUxQixJQUFJLENBQUMsaUJBQWlCLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRWpDLElBQUksTUFBTSxLQUFLLEdBQUcsRUFBRSxDQUFDO1lBQ25CLGlCQUFpQjtZQUNqQixJQUFJLElBQUksSUFBSSxPQUFPLENBQUMsU0FBUyxFQUFFLENBQUM7Z0JBQzlCLE1BQU0sS0FBSyxHQUFHLE9BQU8sQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLENBQUM7Z0JBQ3RDLE9BQU8sSUFBSSxDQUFDLHFCQUFxQixDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQzNDLENBQUM7WUFFRCxtREFBbUQ7WUFDbkQsSUFBSSxJQUFJLEtBQUssSUFBSSxJQUFJLE9BQU8sQ0FBQyxLQUFLLElBQUksY0FBYyxJQUFJLE9BQU8sQ0FBQyxLQUFLLEVBQUUsQ0FBQztnQkFDdEUsT0FBTyxPQUFPLENBQUMsS0FBSyxDQUFDLFlBQVksQ0FBQztZQUNwQyxDQUFDO1lBRUQsSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7Z0JBQ3BCLE1BQU0sSUFBSSwrQkFBc0IsQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDN0MsQ0FBQztZQUNELE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQzthQUFNLENBQUM7WUFDTix1QkFBdUI7WUFDdkIsSUFBSSxPQUFPLENBQUMsS0FBSyxJQUFJLElBQUksSUFBSSxPQUFPLENBQUMsS0FBSyxFQUFFLENBQUM7Z0JBQzNDLE9BQU8sT0FBTyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUM3QixDQUFDO1lBRUQsSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7Z0JBQ3BCLE1BQU0sSUFBSSwrQkFBc0IsQ0FBQyxJQUFJLElBQUksRUFBRSxFQUFFLEVBQUUsQ0FBQyxDQUFDO1lBQ25ELENBQUM7WUFDRCxPQUFPLElBQUksQ0FBQztRQUNkLENBQUM7SUFDSCxDQUFDO0lBRU8scUJBQXFCLENBQUMsS0FBYztRQUMxQyxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsSUFBSSxRQUFRLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNsRSxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3ZDLENBQUM7UUFDRCxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7SUFFTyx1QkFBdUIsQ0FDN0IsSUFBeUQsRUFDekQsT0FBMEI7UUFFMUIsTUFBTSxFQUFFLFFBQVEsRUFBRSxHQUFHLElBQUksQ0FBQztRQUUxQixpREFBaUQ7UUFDakQsSUFBSSxRQUFRLEtBQUssSUFBSSxFQUFFLENBQUM7WUFDdEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFDO1lBQ25ELElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQztnQkFBRSxPQUFPLEtBQUssQ0FBQztZQUNyQyxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUM7UUFDN0QsQ0FBQztRQUVELElBQUksUUFBUSxLQUFLLElBQUksRUFBRSxDQUFDO1lBQ3RCLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQztZQUNuRCxJQUFJLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDO2dCQUFFLE9BQU8sSUFBSSxDQUFDO1lBQ25DLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsT0FBTyxDQUFDLENBQUMsQ0FBQztRQUM3RCxDQUFDO1FBRUQsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFDO1FBQ25ELE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxPQUFPLENBQUMsQ0FBQztRQUVyRCxRQUFRLFFBQVEsRUFBRSxDQUFDO1lBQ2pCLGFBQWE7WUFDYixLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUMvQixLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUNwQyxLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUNwQyxLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUNsQyxLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUNsQyxLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUVqQyxhQUFhO1lBQ2IsS0FBSyxJQUFJO2dCQUNQLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsS0FBSyxDQUFDLENBQUM7WUFDbEMsS0FBSyxJQUFJO2dCQUNQLE9BQU8sQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUNuQyxLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUNwQyxLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUN2QyxLQUFLLElBQUk7Z0JBQ1AsT0FBTyxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQztZQUMzQyxLQUFLLElBQUk7Z0JBQ1AsT0FBTyxJQUFJLENBQUMsa0JBQWtCLENBQUMsSUFBSSxFQUFFLEtBQUssQ0FBQyxDQUFDO1lBRTlDO2dCQUNFLE1BQU0sSUFBSSw4QkFBcUIsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ3ZGLENBQUM7SUFDSCxDQUFDO0lBRU8sc0JBQXNCLENBQzVCLElBQTRDLEVBQzVDLE9BQTBCO1FBRTFCLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxPQUFPLENBQUMsQ0FBQztRQUV6RCxRQUFRLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztZQUN0QixLQUFLLEdBQUc7Z0JBQ04sT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1lBQzlCLEtBQUssR0FBRztnQkFDTixPQUFPLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQztZQUMvQjtnQkFDRSxNQUFNLElBQUksOEJBQXFCLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzNFLENBQUM7SUFDSCxDQUFDO0lBRU8sbUJBQW1CLENBQ3pCLElBQXFFLEVBQ3JFLE9BQTBCO1FBRTFCLE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUM3RCxJQUFJLElBQUksQ0FBQyxNQUFNLENBQUMsU0FBUyxDQUFDLEVBQUUsQ0FBQztZQUMzQixPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLFVBQVUsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUNyRCxDQUFDO1FBQ0QsT0FBTyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDcEQsQ0FBQztJQUVPLG9CQUFvQixDQUMxQixJQUE0QyxFQUM1QyxPQUEwQjtRQUUxQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQ3ZDLE1BQU0sRUFBRSxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBRXRDLElBQUksQ0FBQyxFQUFFLEVBQUUsQ0FBQztZQUNSLE1BQU0sSUFBSSwrQkFBc0IsQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUMzQyxDQUFDO1FBRUQsMEJBQTBCO1FBQzFCLElBQUksSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLEdBQUcsRUFBRSxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ3ZDLE1BQU0sSUFBSSwyQkFBa0IsQ0FBQyxNQUFNLEVBQUUsRUFBRSxHQUFHLEVBQUUsRUFBRSxDQUFDLE9BQU8sRUFBRSxHQUFHLEVBQUUsRUFBRSxDQUFDLE9BQU8sRUFBRSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDcEcsQ0FBQztRQUNELElBQUksRUFBRSxDQUFDLE9BQU8sS0FBSyxDQUFDLENBQUMsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLE1BQU0sR0FBRyxFQUFFLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDNUQsTUFBTSxJQUFJLDJCQUFrQixDQUFDLE1BQU0sRUFBRSxFQUFFLEdBQUcsRUFBRSxFQUFFLENBQUMsT0FBTyxFQUFFLEdBQUcsRUFBRSxFQUFFLENBQUMsT0FBTyxFQUFFLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUNwRyxDQUFDO1FBRUQsK0RBQStEO1FBQy9ELElBQUksTUFBTSxLQUFLLEtBQUssSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNwRCxPQUFPLElBQUksQ0FBQyx5QkFBeUIsQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLE9BQU8sQ0FBQyxDQUFDO1FBQ2pFLENBQUM7UUFDRCxJQUFJLE1BQU0sS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUN4QixPQUFPLElBQUksQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUN0RCxDQUFDO1FBQ0QsSUFBSSxNQUFNLEtBQUssS0FBSyxFQUFFLENBQUM7WUFDckIsT0FBTyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUsT0FBTyxDQUFDLENBQUM7UUFDbkQsQ0FBQztRQUVELHFCQUFxQjtRQUNyQixNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUM7UUFFeEUsb0JBQW9CO1FBQ3BCLE9BQU8sRUFBRSxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsT0FBTyxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQ2hELENBQUM7SUFFTyx5QkFBeUIsQ0FBQyxJQUFlLEVBQUUsT0FBMEI7UUFDM0UsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsT0FBTyxDQUFDLENBQUM7UUFDbEQsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUMxQixNQUFNLElBQUksS0FBSyxDQUFDLHFDQUFxQyxDQUFDLENBQUM7UUFDekQsQ0FBQztRQUVELE1BQU0sVUFBVSxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUMzQixJQUFJLEdBQUcsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksRUFBRSxDQUFDO1FBRW5DLEtBQUssTUFBTSxJQUFJLElBQUksS0FBSyxFQUFFLENBQUM7WUFDekIsSUFBSSxDQUFDLG1CQUFtQixFQUFFLENBQUM7WUFDM0IsTUFBTSxXQUFXLEdBQXNCO2dCQUNyQyxHQUFHLE9BQU87Z0JBQ1YsS0FBSyxFQUFFO29CQUNMLEdBQUcsT0FBTyxDQUFDLEtBQUs7b0JBQ2hCLFlBQVksRUFBRSxJQUFJO2lCQUNuQjtnQkFDRCxTQUFTLEVBQUU7b0JBQ1QsR0FBRyxPQUFPLENBQUMsU0FBUztvQkFDcEIsRUFBRSxFQUFFLElBQUk7aUJBQ1Q7YUFDRixDQUFDO1lBQ0YsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxVQUFVLEVBQUUsV0FBVyxDQUFDLENBQUM7WUFDekQsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUM7Z0JBQzFCLEdBQUcsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1lBQzFELENBQUM7UUFDSCxDQUFDO1FBRUQsT0FBTyxHQUFHLENBQUM7SUFDYixDQUFDO0lBRU8sY0FBYyxDQUFDLElBQWUsRUFBRSxPQUEwQjtRQUNoRSxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUNsRCxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQzFCLE1BQU0sSUFBSSxLQUFLLENBQUMsd0NBQXdDLENBQUMsQ0FBQztRQUM1RCxDQUFDO1FBRUQsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzFCLE1BQU0sTUFBTSxHQUFjLEVBQUUsQ0FBQztRQUU3QixLQUFLLE1BQU0sSUFBSSxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ3pCLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDO1lBQzNCLE1BQU0sV0FBVyxHQUFzQjtnQkFDckMsR0FBRyxPQUFPO2dCQUNWLEtBQUssRUFBRTtvQkFDTCxHQUFHLE9BQU8sQ0FBQyxLQUFLO29CQUNoQixZQUFZLEVBQUUsSUFBSTtpQkFDbkI7Z0JBQ0QsU0FBUyxFQUFFO29CQUNULEdBQUcsT0FBTyxDQUFDLFNBQVM7b0JBQ3BCLEVBQUUsRUFBRSxJQUFJO2lCQUNUO2FBQ0YsQ0FBQztZQUNGLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsU0FBUyxFQUFFLFdBQVcsQ0FBQyxDQUFDO1lBQ3ZELElBQUksSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO2dCQUN0QixNQUFNLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO1lBQ3BCLENBQUM7UUFDSCxDQUFDO1FBRUQsT0FBTyxNQUFNLENBQUM7SUFDaEIsQ0FBQztJQUVPLFdBQVcsQ0FBQyxJQUFlLEVBQUUsT0FBMEI7UUFDN0QsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsT0FBTyxDQUFDLENBQUM7UUFDbEQsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUMxQixNQUFNLElBQUksS0FBSyxDQUFDLHFDQUFxQyxDQUFDLENBQUM7UUFDekQsQ0FBQztRQUVELE1BQU0sVUFBVSxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUMzQixNQUFNLE1BQU0sR0FBYyxFQUFFLENBQUM7UUFFN0IsS0FBSyxNQUFNLElBQUksSUFBSSxLQUFLLEVBQUUsQ0FBQztZQUN6QixJQUFJLENBQUMsbUJBQW1CLEVBQUUsQ0FBQztZQUMzQixNQUFNLFdBQVcsR0FBc0I7Z0JBQ3JDLEdBQUcsT0FBTztnQkFDVixLQUFLLEVBQUU7b0JBQ0wsR0FBRyxPQUFPLENBQUMsS0FBSztvQkFDaEIsWUFBWSxFQUFFLElBQUk7aUJBQ25CO2dCQUNELFNBQVMsRUFBRTtvQkFDVCxHQUFHLE9BQU8sQ0FBQyxTQUFTO29CQUNwQixFQUFFLEVBQUUsSUFBSTtpQkFDVDthQUNGLENBQUM7WUFDRixNQUFNLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsVUFBVSxFQUFFLFdBQVcsQ0FBQyxDQUFDLENBQUM7UUFDMUQsQ0FBQztRQUVELE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUM7SUFFTyxvQkFBb0IsQ0FDMUIsSUFBMkMsRUFDM0MsT0FBMEI7UUFFMUIsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLE9BQU8sQ0FBQyxDQUFDO1FBRXZELElBQUksTUFBTSxLQUFLLElBQUksSUFBSSxNQUFNLEtBQUssU0FBUyxFQUFFLENBQUM7WUFDNUMsSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7Z0JBQ3BCLE1BQU0sSUFBSSw0QkFBbUIsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLE1BQU0sQ0FBQyxDQUFDO1lBQ3ZELENBQUM7WUFDRCxPQUFPLElBQUksQ0FBQztRQUNkLENBQUM7UUFFRCxJQUFJLE9BQU8sTUFBTSxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQy9CLE1BQU0sSUFBSSw0QkFBbUIsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLE9BQU8sTUFBTSxDQUFDLENBQUM7UUFDOUQsQ0FBQztRQUVELE1BQU0sS0FBSyxHQUFJLE1BQWtDLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQ2pFLE9BQU8sSUFBSSxDQUFDLHFCQUFxQixDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzNDLENBQUM7SUFFTyxtQkFBbUIsQ0FDekIsSUFBeUMsRUFDekMsT0FBMEI7UUFFMUIsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLE9BQU8sQ0FBQyxDQUFDO1FBQ3ZELE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxPQUFPLENBQUMsQ0FBQztRQUVyRCxJQUFJLE1BQU0sS0FBSyxJQUFJLElBQUksTUFBTSxLQUFLLFNBQVMsRUFBRSxDQUFDO1lBQzVDLElBQUksSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDO2dCQUNwQixNQUFNLElBQUkseUJBQWdCLENBQUMsS0FBSyxFQUFFLE1BQU0sQ0FBQyxDQUFDO1lBQzVDLENBQUM7WUFDRCxPQUFPLElBQUksQ0FBQztRQUNkLENBQUM7UUFFRCxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztZQUMxQixNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQ2pDLElBQUksR0FBRyxHQUFHLENBQUMsSUFBSSxHQUFHLElBQUksTUFBTSxDQUFDLE1BQU0sRUFBRSxDQUFDO2dCQUNwQyxPQUFPLElBQUksQ0FBQztZQUNkLENBQUM7WUFDRCxPQUFPLElBQUksQ0FBQyxxQkFBcUIsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztRQUNqRCxDQUFDO1FBRUQsSUFBSSxPQUFPLE1BQU0sS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUMvQixNQUFNLEdBQUcsR0FBRyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDMUIsTUFBTSxLQUFLLEdBQUksTUFBa0MsQ0FBQyxHQUFHLENBQUMsQ0FBQztZQUN2RCxPQUFPLElBQUksQ0FBQyxxQkFBcUIsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMzQyxDQUFDO1FBRUQsTUFBTSxJQUFJLHlCQUFnQixDQUFDLEtBQUssRUFBRSxPQUFPLE1BQU0sQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFFRCwrRUFBK0U7SUFDL0UsaUJBQWlCO0lBQ2pCLCtFQUErRTtJQUV2RSxTQUFTLENBQUMsS0FBYztRQUM5QixPQUFPLEtBQUssWUFBWSx1QkFBTyxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsQ0FBQztJQUMvRCxDQUFDO0lBRU8sU0FBUyxDQUFDLEtBQWM7UUFDOUIsSUFBSSxLQUFLLFlBQVksdUJBQU87WUFBRSxPQUFPLEtBQUssQ0FBQztRQUMzQyxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVE7WUFBRSxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3BFLElBQUksT0FBTyxLQUFLLEtBQUssUUFBUTtZQUFFLE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDcEUsTUFBTSxJQUFJLDhCQUFxQixDQUFDLFdBQVcsRUFBRSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3JFLENBQUM7SUFFTyxRQUFRLENBQUMsS0FBYztRQUM3QixJQUFJLEtBQUssWUFBWSx1QkFBTztZQUFFLE9BQU8sS0FBSyxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ3RELElBQUksT0FBTyxLQUFLLEtBQUssUUFBUTtZQUFFLE9BQU8sS0FBSyxDQUFDO1FBQzVDLElBQUksT0FBTyxLQUFLLEtBQUssUUFBUTtZQUFFLE9BQU8sVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3hELE1BQU0sSUFBSSw4QkFBcUIsQ0FBQyxVQUFVLEVBQUUsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNwRSxDQUFDO0lBRU8sTUFBTSxDQUFDLEtBQWM7UUFDM0IsSUFBSSxLQUFLLFlBQVksdUJBQU87WUFBRSxPQUFPLENBQUMsS0FBSyxDQUFDLE1BQU0sRUFBRSxDQUFDO1FBQ3JELElBQUksT0FBTyxLQUFLLEtBQUssU0FBUztZQUFFLE9BQU8sS0FBSyxDQUFDO1FBQzdDLElBQUksS0FBSyxLQUFLLElBQUksSUFBSSxLQUFLLEtBQUssU0FBUztZQUFFLE9BQU8sS0FBSyxDQUFDO1FBQ3hELElBQUksT0FBTyxLQUFLLEtBQUssUUFBUTtZQUFFLE9BQU8sS0FBSyxLQUFLLENBQUMsQ0FBQztRQUNsRCxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVE7WUFBRSxPQUFPLEtBQUssQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBQ3ZELElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUM7WUFBRSxPQUFPLEtBQUssQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBQ2xELE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztJQUVPLE1BQU0sQ0FBQyxLQUFjO1FBQzNCLElBQUksS0FBSyxLQUFLLElBQUk7WUFBRSxPQUFPLE1BQU0sQ0FBQztRQUNsQyxJQUFJLEtBQUssWUFBWSx1QkFBTztZQUFFLE9BQU8sU0FBUyxDQUFDO1FBQy9DLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUM7WUFBRSxPQUFPLE9BQU8sQ0FBQztRQUN6QyxPQUFPLE9BQU8sS0FBSyxDQUFDO0lBQ3RCLENBQUM7SUFFTyxHQUFHLENBQUMsSUFBYSxFQUFFLEtBQWM7UUFDdkMsdUJBQXVCO1FBQ3ZCLElBQUksT0FBTyxJQUFJLEtBQUssUUFBUSxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzFELE1BQU0sT0FBTyxHQUFHLElBQUksWUFBWSx1QkFBTyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUN6RSxNQUFNLFFBQVEsR0FBRyxLQUFLLFlBQVksdUJBQU8sQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDN0UsT0FBTyxPQUFPLEdBQUcsUUFBUSxDQUFDO1FBQzVCLENBQUM7UUFFRCxtQkFBbUI7UUFDbkIsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNsRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQzVFLENBQUM7UUFFRCxNQUFNLElBQUksOEJBQXFCLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNoRixDQUFDO0lBRU8sUUFBUSxDQUFDLElBQWEsRUFBRSxLQUFjO1FBQzVDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQ3BELE1BQU0sSUFBSSw4QkFBcUIsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2hGLENBQUM7UUFDRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO0lBQ2pGLENBQUM7SUFFTyxRQUFRLENBQUMsSUFBYSxFQUFFLEtBQWM7UUFDNUMsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDcEQsTUFBTSxJQUFJLDhCQUFxQixDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDaEYsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7SUFDakYsQ0FBQztJQUVPLE1BQU0sQ0FBQyxJQUFhLEVBQUUsS0FBYztRQUMxQyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNwRCxNQUFNLElBQUksOEJBQXFCLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNoRixDQUFDO1FBQ0QsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUN0QyxJQUFJLE9BQU8sQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDO1lBQ3JCLE1BQU0sSUFBSSw0QkFBbUIsRUFBRSxDQUFDO1FBQ2xDLENBQUM7UUFDRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDakUsQ0FBQztJQUVPLE1BQU0sQ0FBQyxJQUFhLEVBQUUsS0FBYztRQUMxQyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNwRCxNQUFNLElBQUksOEJBQXFCLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNoRixDQUFDO1FBQ0QsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUN0QyxJQUFJLE9BQU8sQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDO1lBQ3JCLE1BQU0sSUFBSSw0QkFBbUIsRUFBRSxDQUFDO1FBQ2xDLENBQUM7UUFDRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDakUsQ0FBQztJQUVPLEtBQUssQ0FBQyxJQUFhLEVBQUUsS0FBYztRQUN6QyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNwRCxNQUFNLElBQUksOEJBQXFCLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNoRixDQUFDO1FBQ0QsT0FBTyxJQUFJLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxFQUFFLElBQUksQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztJQUM3RSxDQUFDO0lBRU8sTUFBTSxDQUFDLEtBQWM7UUFDM0IsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUMzQixNQUFNLElBQUksOEJBQXFCLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDN0QsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO0lBQ3pELENBQUM7SUFFTyxNQUFNLENBQUMsSUFBYSxFQUFFLEtBQWM7UUFDMUMsSUFBSSxJQUFJLFlBQVksdUJBQU8sSUFBSSxLQUFLLFlBQVksdUJBQU8sRUFBRSxDQUFDO1lBQ3hELE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM1QixDQUFDO1FBQ0QsSUFBSSxJQUFJLFlBQVksdUJBQU8sSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUN6RCxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztRQUNwRCxDQUFDO1FBQ0QsSUFBSSxPQUFPLElBQUksS0FBSyxRQUFRLElBQUksS0FBSyxZQUFZLHVCQUFPLEVBQUUsQ0FBQztZQUN6RCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNwRCxDQUFDO1FBQ0QsT0FBTyxJQUFJLEtBQUssS0FBSyxDQUFDO0lBQ3hCLENBQUM7SUFFTyxRQUFRLENBQUMsSUFBYSxFQUFFLEtBQWM7UUFDNUMsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNsRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQ2pGLENBQUM7UUFDRCxJQUFJLE9BQU8sSUFBSSxLQUFLLFFBQVEsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUMxRCxPQUFPLElBQUksR0FBRyxLQUFLLENBQUM7UUFDdEIsQ0FBQztRQUNELE1BQU0sSUFBSSw4QkFBcUIsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2hGLENBQUM7SUFFTyxXQUFXLENBQUMsSUFBYSxFQUFFLEtBQWM7UUFDL0MsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNsRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQ3BGLENBQUM7UUFDRCxJQUFJLE9BQU8sSUFBSSxLQUFLLFFBQVEsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUMxRCxPQUFPLElBQUksR0FBRyxLQUFLLENBQUM7UUFDdEIsQ0FBQztRQUNELE1BQU0sSUFBSSw4QkFBcUIsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2hGLENBQUM7SUFFTyxlQUFlLENBQUMsSUFBYSxFQUFFLEtBQWM7UUFDbkQsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUNsRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQ3hGLENBQUM7UUFDRCxJQUFJLE9BQU8sSUFBSSxLQUFLLFFBQVEsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUMxRCxPQUFPLElBQUksSUFBSSxLQUFLLENBQUM7UUFDdkIsQ0FBQztRQUNELE1BQU0sSUFBSSw4QkFBcUIsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2pGLENBQUM7SUFFTyxrQkFBa0IsQ0FBQyxJQUFhLEVBQUUsS0FBYztRQUN0RCxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQ2xELE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztRQUMzRixDQUFDO1FBQ0QsSUFBSSxPQUFPLElBQUksS0FBSyxRQUFRLElBQUksT0FBTyxLQUFLLEtBQUssUUFBUSxFQUFFLENBQUM7WUFDMUQsT0FBTyxJQUFJLElBQUksS0FBSyxDQUFDO1FBQ3ZCLENBQUM7UUFDRCxNQUFNLElBQUksOEJBQXFCLENBQUMsSUFBSSxFQUFFLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNqRixDQUFDO0NBQ0Y7QUFya0JELDhCQXFrQkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBU1ROb2RlLFxuICBFdmFsdWF0aW9uQ29udGV4dCxcbiAgRXZhbHVhdGlvblJlc3VsdCxcbiAgRnVuY3Rpb25EZWZpbml0aW9uLFxuICBGb3JtdWxhRW5naW5lQ29uZmlnLFxuICBTZWN1cml0eUNvbmZpZyxcbn0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBEZWNpbWFsVXRpbHMsIERlY2ltYWwgfSBmcm9tICcuL2RlY2ltYWwtdXRpbHMnO1xuaW1wb3J0IHtcbiAgVW5kZWZpbmVkVmFyaWFibGVFcnJvcixcbiAgVW5kZWZpbmVkRnVuY3Rpb25FcnJvcixcbiAgRGl2aXNpb25CeVplcm9FcnJvcixcbiAgSW52YWxpZE9wZXJhdGlvbkVycm9yLFxuICBQcm9wZXJ0eUFjY2Vzc0Vycm9yLFxuICBJbmRleEFjY2Vzc0Vycm9yLFxuICBBcmd1bWVudENvdW50RXJyb3IsXG4gIE1heEl0ZXJhdGlvbnNFcnJvcixcbiAgTWF4UmVjdXJzaW9uRXJyb3IsXG59IGZyb20gJy4vZXJyb3JzJztcbmltcG9ydCB7IFBhcnNlciB9IGZyb20gJy4vcGFyc2VyJztcblxuZXhwb3J0IGNsYXNzIEV2YWx1YXRvciB7XG4gIHByaXZhdGUgcGFyc2VyOiBQYXJzZXI7XG4gIHByaXZhdGUgZGVjaW1hbFV0aWxzOiBEZWNpbWFsVXRpbHM7XG4gIHByaXZhdGUgZnVuY3Rpb25zOiBNYXA8c3RyaW5nLCBGdW5jdGlvbkRlZmluaXRpb24+O1xuICBwcml2YXRlIHN0cmljdE1vZGU6IGJvb2xlYW47XG4gIHByaXZhdGUgc2VjdXJpdHlDb25maWc6IFNlY3VyaXR5Q29uZmlnO1xuICBwcml2YXRlIHJlY3Vyc2lvbkRlcHRoOiBudW1iZXIgPSAwO1xuICBwcml2YXRlIGl0ZXJhdGlvbkNvdW50OiBudW1iZXIgPSAwO1xuICBwcml2YXRlIGFjY2Vzc2VkVmFyaWFibGVzOiBTZXQ8c3RyaW5nPiA9IG5ldyBTZXQoKTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBkZWNpbWFsVXRpbHM6IERlY2ltYWxVdGlscyxcbiAgICBmdW5jdGlvbnM6IE1hcDxzdHJpbmcsIEZ1bmN0aW9uRGVmaW5pdGlvbj4sXG4gICAgY29uZmlnPzogRm9ybXVsYUVuZ2luZUNvbmZpZ1xuICApIHtcbiAgICB0aGlzLnBhcnNlciA9IG5ldyBQYXJzZXIoKTtcbiAgICB0aGlzLmRlY2ltYWxVdGlscyA9IGRlY2ltYWxVdGlscztcbiAgICB0aGlzLmZ1bmN0aW9ucyA9IGZ1bmN0aW9ucztcbiAgICB0aGlzLnN0cmljdE1vZGUgPSBjb25maWc/LnN0cmljdE1vZGUgPz8gdHJ1ZTtcbiAgICB0aGlzLnNlY3VyaXR5Q29uZmlnID0gY29uZmlnPy5zZWN1cml0eSA/PyB7fTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFdmFsdWF0ZSBhbiBleHByZXNzaW9uIHN0cmluZ1xuICAgKi9cbiAgZXZhbHVhdGUoZXhwcmVzc2lvbjogc3RyaW5nLCBjb250ZXh0OiBFdmFsdWF0aW9uQ29udGV4dCk6IEV2YWx1YXRpb25SZXN1bHQge1xuICAgIGNvbnN0IHN0YXJ0VGltZSA9IERhdGUubm93KCk7XG4gICAgdGhpcy5hY2Nlc3NlZFZhcmlhYmxlcyA9IG5ldyBTZXQoKTtcbiAgICB0aGlzLnJlY3Vyc2lvbkRlcHRoID0gMDtcbiAgICB0aGlzLml0ZXJhdGlvbkNvdW50ID0gMDtcblxuICAgIHRyeSB7XG4gICAgICBjb25zdCBhc3QgPSB0aGlzLnBhcnNlci5wYXJzZShleHByZXNzaW9uKTtcbiAgICAgIGNvbnN0IHZhbHVlID0gdGhpcy5ldmFsdWF0ZU5vZGUoYXN0LCBjb250ZXh0KTtcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgdmFsdWUsXG4gICAgICAgIHN1Y2Nlc3M6IHRydWUsXG4gICAgICAgIGV4ZWN1dGlvblRpbWVNczogRGF0ZS5ub3coKSAtIHN0YXJ0VGltZSxcbiAgICAgICAgYWNjZXNzZWRWYXJpYWJsZXM6IG5ldyBTZXQodGhpcy5hY2Nlc3NlZFZhcmlhYmxlcyksXG4gICAgICB9O1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICB2YWx1ZTogbnVsbCxcbiAgICAgICAgc3VjY2VzczogZmFsc2UsXG4gICAgICAgIGVycm9yOiBlcnJvciBhcyBFcnJvcixcbiAgICAgICAgZXhlY3V0aW9uVGltZU1zOiBEYXRlLm5vdygpIC0gc3RhcnRUaW1lLFxuICAgICAgICBhY2Nlc3NlZFZhcmlhYmxlczogbmV3IFNldCh0aGlzLmFjY2Vzc2VkVmFyaWFibGVzKSxcbiAgICAgIH07XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEV2YWx1YXRlIGFuIEFTVCBub2RlXG4gICAqL1xuICBldmFsdWF0ZU5vZGUobm9kZTogQVNUTm9kZSwgY29udGV4dDogRXZhbHVhdGlvbkNvbnRleHQpOiB1bmtub3duIHtcbiAgICB0aGlzLmNoZWNrUmVjdXJzaW9uTGltaXQoKTtcblxuICAgIHN3aXRjaCAobm9kZS50eXBlKSB7XG4gICAgICBjYXNlICdEZWNpbWFsTGl0ZXJhbCc6XG4gICAgICAgIHJldHVybiB0aGlzLmRlY2ltYWxVdGlscy5mcm9tKG5vZGUudmFsdWUpO1xuXG4gICAgICBjYXNlICdOdW1iZXJMaXRlcmFsJzpcbiAgICAgICAgcmV0dXJuIG5vZGUudmFsdWU7XG5cbiAgICAgIGNhc2UgJ1N0cmluZ0xpdGVyYWwnOlxuICAgICAgICByZXR1cm4gbm9kZS52YWx1ZTtcblxuICAgICAgY2FzZSAnQm9vbGVhbkxpdGVyYWwnOlxuICAgICAgICByZXR1cm4gbm9kZS52YWx1ZTtcblxuICAgICAgY2FzZSAnTnVsbExpdGVyYWwnOlxuICAgICAgICByZXR1cm4gbnVsbDtcblxuICAgICAgY2FzZSAnQXJyYXlMaXRlcmFsJzpcbiAgICAgICAgcmV0dXJuIG5vZGUuZWxlbWVudHMubWFwKGVsID0+IHRoaXMuZXZhbHVhdGVOb2RlKGVsLCBjb250ZXh0KSk7XG5cbiAgICAgIGNhc2UgJ1ZhcmlhYmxlUmVmZXJlbmNlJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuZXZhbHVhdGVWYXJpYWJsZShub2RlLnByZWZpeCwgbm9kZS5uYW1lLCBjb250ZXh0KTtcblxuICAgICAgY2FzZSAnQmluYXJ5T3BlcmF0aW9uJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuZXZhbHVhdGVCaW5hcnlPcGVyYXRpb24obm9kZSwgY29udGV4dCk7XG5cbiAgICAgIGNhc2UgJ1VuYXJ5T3BlcmF0aW9uJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuZXZhbHVhdGVVbmFyeU9wZXJhdGlvbihub2RlLCBjb250ZXh0KTtcblxuICAgICAgY2FzZSAnQ29uZGl0aW9uYWxFeHByZXNzaW9uJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuZXZhbHVhdGVDb25kaXRpb25hbChub2RlLCBjb250ZXh0KTtcblxuICAgICAgY2FzZSAnRnVuY3Rpb25DYWxsJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuZXZhbHVhdGVGdW5jdGlvbkNhbGwobm9kZSwgY29udGV4dCk7XG5cbiAgICAgIGNhc2UgJ01lbWJlckFjY2Vzcyc6XG4gICAgICAgIHJldHVybiB0aGlzLmV2YWx1YXRlTWVtYmVyQWNjZXNzKG5vZGUsIGNvbnRleHQpO1xuXG4gICAgICBjYXNlICdJbmRleEFjY2Vzcyc6XG4gICAgICAgIHJldHVybiB0aGlzLmV2YWx1YXRlSW5kZXhBY2Nlc3Mobm9kZSwgY29udGV4dCk7XG5cbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgVW5rbm93biBub2RlIHR5cGU6ICR7KG5vZGUgYXMgYW55KS50eXBlfWApO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgY2hlY2tSZWN1cnNpb25MaW1pdCgpOiB2b2lkIHtcbiAgICB0aGlzLnJlY3Vyc2lvbkRlcHRoKys7XG4gICAgY29uc3QgbGltaXQgPSB0aGlzLnNlY3VyaXR5Q29uZmlnLm1heFJlY3Vyc2lvbkRlcHRoID8/IDEwMDtcbiAgICBpZiAodGhpcy5yZWN1cnNpb25EZXB0aCA+IGxpbWl0KSB7XG4gICAgICB0aHJvdyBuZXcgTWF4UmVjdXJzaW9uRXJyb3IobGltaXQpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgY2hlY2tJdGVyYXRpb25MaW1pdCgpOiB2b2lkIHtcbiAgICB0aGlzLml0ZXJhdGlvbkNvdW50Kys7XG4gICAgY29uc3QgbGltaXQgPSB0aGlzLnNlY3VyaXR5Q29uZmlnLm1heEl0ZXJhdGlvbnMgPz8gMTAwMDA7XG4gICAgaWYgKHRoaXMuaXRlcmF0aW9uQ291bnQgPiBsaW1pdCkge1xuICAgICAgdGhyb3cgbmV3IE1heEl0ZXJhdGlvbnNFcnJvcihsaW1pdCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBldmFsdWF0ZVZhcmlhYmxlKFxuICAgIHByZWZpeDogJyQnIHwgJ0AnLFxuICAgIG5hbWU6IHN0cmluZyxcbiAgICBjb250ZXh0OiBFdmFsdWF0aW9uQ29udGV4dFxuICApOiB1bmtub3duIHtcbiAgICB0aGlzLmFjY2Vzc2VkVmFyaWFibGVzLmFkZChuYW1lKTtcblxuICAgIGlmIChwcmVmaXggPT09ICckJykge1xuICAgICAgLy8gTG9jYWwgdmFyaWFibGVcbiAgICAgIGlmIChuYW1lIGluIGNvbnRleHQudmFyaWFibGVzKSB7XG4gICAgICAgIGNvbnN0IHZhbHVlID0gY29udGV4dC52YXJpYWJsZXNbbmFtZV07XG4gICAgICAgIHJldHVybiB0aGlzLm1heWJlQ29udmVydFRvRGVjaW1hbCh2YWx1ZSk7XG4gICAgICB9XG5cbiAgICAgIC8vIENoZWNrIGlmIGl0J3MgYSBzcGVjaWFsIGl0ZXJhdGlvbiB2YXJpYWJsZSAoJGl0KVxuICAgICAgaWYgKG5hbWUgPT09ICdpdCcgJiYgY29udGV4dC5leHRyYSAmJiAnX2N1cnJlbnRJdGVtJyBpbiBjb250ZXh0LmV4dHJhKSB7XG4gICAgICAgIHJldHVybiBjb250ZXh0LmV4dHJhLl9jdXJyZW50SXRlbTtcbiAgICAgIH1cblxuICAgICAgaWYgKHRoaXMuc3RyaWN0TW9kZSkge1xuICAgICAgICB0aHJvdyBuZXcgVW5kZWZpbmVkVmFyaWFibGVFcnJvcihuYW1lLCAnJyk7XG4gICAgICB9XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9IGVsc2Uge1xuICAgICAgLy8gQ29udGV4dCB2YXJpYWJsZSAoQClcbiAgICAgIGlmIChjb250ZXh0LmV4dHJhICYmIG5hbWUgaW4gY29udGV4dC5leHRyYSkge1xuICAgICAgICByZXR1cm4gY29udGV4dC5leHRyYVtuYW1lXTtcbiAgICAgIH1cblxuICAgICAgaWYgKHRoaXMuc3RyaWN0TW9kZSkge1xuICAgICAgICB0aHJvdyBuZXcgVW5kZWZpbmVkVmFyaWFibGVFcnJvcihgQCR7bmFtZX1gLCAnJyk7XG4gICAgICB9XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIG1heWJlQ29udmVydFRvRGVjaW1hbCh2YWx1ZTogdW5rbm93bik6IHVua25vd24ge1xuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09ICdudW1iZXInICYmICFpc05hTih2YWx1ZSkgJiYgaXNGaW5pdGUodmFsdWUpKSB7XG4gICAgICByZXR1cm4gdGhpcy5kZWNpbWFsVXRpbHMuZnJvbSh2YWx1ZSk7XG4gICAgfVxuICAgIHJldHVybiB2YWx1ZTtcbiAgfVxuXG4gIHByaXZhdGUgZXZhbHVhdGVCaW5hcnlPcGVyYXRpb24oXG4gICAgbm9kZTogeyBvcGVyYXRvcjogc3RyaW5nOyBsZWZ0OiBBU1ROb2RlOyByaWdodDogQVNUTm9kZSB9LFxuICAgIGNvbnRleHQ6IEV2YWx1YXRpb25Db250ZXh0XG4gICk6IHVua25vd24ge1xuICAgIGNvbnN0IHsgb3BlcmF0b3IgfSA9IG5vZGU7XG5cbiAgICAvLyBTaG9ydC1jaXJjdWl0IGV2YWx1YXRpb24gZm9yIGxvZ2ljYWwgb3BlcmF0b3JzXG4gICAgaWYgKG9wZXJhdG9yID09PSAnJiYnKSB7XG4gICAgICBjb25zdCBsZWZ0ID0gdGhpcy5ldmFsdWF0ZU5vZGUobm9kZS5sZWZ0LCBjb250ZXh0KTtcbiAgICAgIGlmICghdGhpcy50b0Jvb2wobGVmdCkpIHJldHVybiBmYWxzZTtcbiAgICAgIHJldHVybiB0aGlzLnRvQm9vbCh0aGlzLmV2YWx1YXRlTm9kZShub2RlLnJpZ2h0LCBjb250ZXh0KSk7XG4gICAgfVxuXG4gICAgaWYgKG9wZXJhdG9yID09PSAnfHwnKSB7XG4gICAgICBjb25zdCBsZWZ0ID0gdGhpcy5ldmFsdWF0ZU5vZGUobm9kZS5sZWZ0LCBjb250ZXh0KTtcbiAgICAgIGlmICh0aGlzLnRvQm9vbChsZWZ0KSkgcmV0dXJuIHRydWU7XG4gICAgICByZXR1cm4gdGhpcy50b0Jvb2wodGhpcy5ldmFsdWF0ZU5vZGUobm9kZS5yaWdodCwgY29udGV4dCkpO1xuICAgIH1cblxuICAgIGNvbnN0IGxlZnQgPSB0aGlzLmV2YWx1YXRlTm9kZShub2RlLmxlZnQsIGNvbnRleHQpO1xuICAgIGNvbnN0IHJpZ2h0ID0gdGhpcy5ldmFsdWF0ZU5vZGUobm9kZS5yaWdodCwgY29udGV4dCk7XG5cbiAgICBzd2l0Y2ggKG9wZXJhdG9yKSB7XG4gICAgICAvLyBBcml0aG1ldGljXG4gICAgICBjYXNlICcrJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuYWRkKGxlZnQsIHJpZ2h0KTtcbiAgICAgIGNhc2UgJy0nOlxuICAgICAgICByZXR1cm4gdGhpcy5zdWJ0cmFjdChsZWZ0LCByaWdodCk7XG4gICAgICBjYXNlICcqJzpcbiAgICAgICAgcmV0dXJuIHRoaXMubXVsdGlwbHkobGVmdCwgcmlnaHQpO1xuICAgICAgY2FzZSAnLyc6XG4gICAgICAgIHJldHVybiB0aGlzLmRpdmlkZShsZWZ0LCByaWdodCk7XG4gICAgICBjYXNlICclJzpcbiAgICAgICAgcmV0dXJuIHRoaXMubW9kdWxvKGxlZnQsIHJpZ2h0KTtcbiAgICAgIGNhc2UgJ14nOlxuICAgICAgICByZXR1cm4gdGhpcy5wb3dlcihsZWZ0LCByaWdodCk7XG5cbiAgICAgIC8vIENvbXBhcmlzb25cbiAgICAgIGNhc2UgJz09JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuZXF1YWxzKGxlZnQsIHJpZ2h0KTtcbiAgICAgIGNhc2UgJyE9JzpcbiAgICAgICAgcmV0dXJuICF0aGlzLmVxdWFscyhsZWZ0LCByaWdodCk7XG4gICAgICBjYXNlICc8JzpcbiAgICAgICAgcmV0dXJuIHRoaXMubGVzc1RoYW4obGVmdCwgcmlnaHQpO1xuICAgICAgY2FzZSAnPic6XG4gICAgICAgIHJldHVybiB0aGlzLmdyZWF0ZXJUaGFuKGxlZnQsIHJpZ2h0KTtcbiAgICAgIGNhc2UgJzw9JzpcbiAgICAgICAgcmV0dXJuIHRoaXMubGVzc1RoYW5PckVxdWFsKGxlZnQsIHJpZ2h0KTtcbiAgICAgIGNhc2UgJz49JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuZ3JlYXRlclRoYW5PckVxdWFsKGxlZnQsIHJpZ2h0KTtcblxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgdGhyb3cgbmV3IEludmFsaWRPcGVyYXRpb25FcnJvcihvcGVyYXRvciwgW3RoaXMudHlwZU9mKGxlZnQpLCB0aGlzLnR5cGVPZihyaWdodCldKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGV2YWx1YXRlVW5hcnlPcGVyYXRpb24oXG4gICAgbm9kZTogeyBvcGVyYXRvcjogc3RyaW5nOyBvcGVyYW5kOiBBU1ROb2RlIH0sXG4gICAgY29udGV4dDogRXZhbHVhdGlvbkNvbnRleHRcbiAgKTogdW5rbm93biB7XG4gICAgY29uc3Qgb3BlcmFuZCA9IHRoaXMuZXZhbHVhdGVOb2RlKG5vZGUub3BlcmFuZCwgY29udGV4dCk7XG5cbiAgICBzd2l0Y2ggKG5vZGUub3BlcmF0b3IpIHtcbiAgICAgIGNhc2UgJy0nOlxuICAgICAgICByZXR1cm4gdGhpcy5uZWdhdGUob3BlcmFuZCk7XG4gICAgICBjYXNlICchJzpcbiAgICAgICAgcmV0dXJuICF0aGlzLnRvQm9vbChvcGVyYW5kKTtcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHRocm93IG5ldyBJbnZhbGlkT3BlcmF0aW9uRXJyb3Iobm9kZS5vcGVyYXRvciwgW3RoaXMudHlwZU9mKG9wZXJhbmQpXSk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBldmFsdWF0ZUNvbmRpdGlvbmFsKFxuICAgIG5vZGU6IHsgY29uZGl0aW9uOiBBU1ROb2RlOyBjb25zZXF1ZW50OiBBU1ROb2RlOyBhbHRlcm5hdGU6IEFTVE5vZGUgfSxcbiAgICBjb250ZXh0OiBFdmFsdWF0aW9uQ29udGV4dFxuICApOiB1bmtub3duIHtcbiAgICBjb25zdCBjb25kaXRpb24gPSB0aGlzLmV2YWx1YXRlTm9kZShub2RlLmNvbmRpdGlvbiwgY29udGV4dCk7XG4gICAgaWYgKHRoaXMudG9Cb29sKGNvbmRpdGlvbikpIHtcbiAgICAgIHJldHVybiB0aGlzLmV2YWx1YXRlTm9kZShub2RlLmNvbnNlcXVlbnQsIGNvbnRleHQpO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5ldmFsdWF0ZU5vZGUobm9kZS5hbHRlcm5hdGUsIGNvbnRleHQpO1xuICB9XG5cbiAgcHJpdmF0ZSBldmFsdWF0ZUZ1bmN0aW9uQ2FsbChcbiAgICBub2RlOiB7IG5hbWU6IHN0cmluZzsgYXJndW1lbnRzOiBBU1ROb2RlW10gfSxcbiAgICBjb250ZXh0OiBFdmFsdWF0aW9uQ29udGV4dFxuICApOiB1bmtub3duIHtcbiAgICBjb25zdCBmbk5hbWUgPSBub2RlLm5hbWUudG9VcHBlckNhc2UoKTtcbiAgICBjb25zdCBmbiA9IHRoaXMuZnVuY3Rpb25zLmdldChmbk5hbWUpO1xuXG4gICAgaWYgKCFmbikge1xuICAgICAgdGhyb3cgbmV3IFVuZGVmaW5lZEZ1bmN0aW9uRXJyb3IoZm5OYW1lKTtcbiAgICB9XG5cbiAgICAvLyBWYWxpZGF0ZSBhcmd1bWVudCBjb3VudFxuICAgIGlmIChub2RlLmFyZ3VtZW50cy5sZW5ndGggPCBmbi5taW5BcmdzKSB7XG4gICAgICB0aHJvdyBuZXcgQXJndW1lbnRDb3VudEVycm9yKGZuTmFtZSwgeyBtaW46IGZuLm1pbkFyZ3MsIG1heDogZm4ubWF4QXJncyB9LCBub2RlLmFyZ3VtZW50cy5sZW5ndGgpO1xuICAgIH1cbiAgICBpZiAoZm4ubWF4QXJncyAhPT0gLTEgJiYgbm9kZS5hcmd1bWVudHMubGVuZ3RoID4gZm4ubWF4QXJncykge1xuICAgICAgdGhyb3cgbmV3IEFyZ3VtZW50Q291bnRFcnJvcihmbk5hbWUsIHsgbWluOiBmbi5taW5BcmdzLCBtYXg6IGZuLm1heEFyZ3MgfSwgbm9kZS5hcmd1bWVudHMubGVuZ3RoKTtcbiAgICB9XG5cbiAgICAvLyBIYW5kbGUgc3BlY2lhbCBmdW5jdGlvbnMgdGhhdCBuZWVkIEFTVCBub2RlcyAoZm9yIGl0ZXJhdGlvbilcbiAgICBpZiAoZm5OYW1lID09PSAnU1VNJyAmJiBub2RlLmFyZ3VtZW50cy5sZW5ndGggPT09IDIpIHtcbiAgICAgIHJldHVybiB0aGlzLmV2YWx1YXRlU3VtV2l0aEV4cHJlc3Npb24obm9kZS5hcmd1bWVudHMsIGNvbnRleHQpO1xuICAgIH1cbiAgICBpZiAoZm5OYW1lID09PSAnRklMVEVSJykge1xuICAgICAgcmV0dXJuIHRoaXMuZXZhbHVhdGVGaWx0ZXIobm9kZS5hcmd1bWVudHMsIGNvbnRleHQpO1xuICAgIH1cbiAgICBpZiAoZm5OYW1lID09PSAnTUFQJykge1xuICAgICAgcmV0dXJuIHRoaXMuZXZhbHVhdGVNYXAobm9kZS5hcmd1bWVudHMsIGNvbnRleHQpO1xuICAgIH1cblxuICAgIC8vIEV2YWx1YXRlIGFyZ3VtZW50c1xuICAgIGNvbnN0IGFyZ3MgPSBub2RlLmFyZ3VtZW50cy5tYXAoYXJnID0+IHRoaXMuZXZhbHVhdGVOb2RlKGFyZywgY29udGV4dCkpO1xuXG4gICAgLy8gQ2FsbCB0aGUgZnVuY3Rpb25cbiAgICByZXR1cm4gZm4uaW1wbGVtZW50YXRpb24oYXJncywgY29udGV4dCwgdGhpcyk7XG4gIH1cblxuICBwcml2YXRlIGV2YWx1YXRlU3VtV2l0aEV4cHJlc3Npb24oYXJnczogQVNUTm9kZVtdLCBjb250ZXh0OiBFdmFsdWF0aW9uQ29udGV4dCk6IERlY2ltYWwge1xuICAgIGNvbnN0IGFycmF5ID0gdGhpcy5ldmFsdWF0ZU5vZGUoYXJnc1swXSwgY29udGV4dCk7XG4gICAgaWYgKCFBcnJheS5pc0FycmF5KGFycmF5KSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdTVU0gZmlyc3QgYXJndW1lbnQgbXVzdCBiZSBhbiBhcnJheScpO1xuICAgIH1cblxuICAgIGNvbnN0IGV4cHJlc3Npb24gPSBhcmdzWzFdO1xuICAgIGxldCBzdW0gPSB0aGlzLmRlY2ltYWxVdGlscy56ZXJvKCk7XG5cbiAgICBmb3IgKGNvbnN0IGl0ZW0gb2YgYXJyYXkpIHtcbiAgICAgIHRoaXMuY2hlY2tJdGVyYXRpb25MaW1pdCgpO1xuICAgICAgY29uc3QgaXRlbUNvbnRleHQ6IEV2YWx1YXRpb25Db250ZXh0ID0ge1xuICAgICAgICAuLi5jb250ZXh0LFxuICAgICAgICBleHRyYToge1xuICAgICAgICAgIC4uLmNvbnRleHQuZXh0cmEsXG4gICAgICAgICAgX2N1cnJlbnRJdGVtOiBpdGVtLFxuICAgICAgICB9LFxuICAgICAgICB2YXJpYWJsZXM6IHtcbiAgICAgICAgICAuLi5jb250ZXh0LnZhcmlhYmxlcyxcbiAgICAgICAgICBpdDogaXRlbSxcbiAgICAgICAgfSxcbiAgICAgIH07XG4gICAgICBjb25zdCB2YWx1ZSA9IHRoaXMuZXZhbHVhdGVOb2RlKGV4cHJlc3Npb24sIGl0ZW1Db250ZXh0KTtcbiAgICAgIGlmICh0aGlzLmlzTnVtZXJpYyh2YWx1ZSkpIHtcbiAgICAgICAgc3VtID0gdGhpcy5kZWNpbWFsVXRpbHMuYWRkKHN1bSwgdGhpcy50b0RlY2ltYWwodmFsdWUpKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICByZXR1cm4gc3VtO1xuICB9XG5cbiAgcHJpdmF0ZSBldmFsdWF0ZUZpbHRlcihhcmdzOiBBU1ROb2RlW10sIGNvbnRleHQ6IEV2YWx1YXRpb25Db250ZXh0KTogdW5rbm93bltdIHtcbiAgICBjb25zdCBhcnJheSA9IHRoaXMuZXZhbHVhdGVOb2RlKGFyZ3NbMF0sIGNvbnRleHQpO1xuICAgIGlmICghQXJyYXkuaXNBcnJheShhcnJheSkpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignRklMVEVSIGZpcnN0IGFyZ3VtZW50IG11c3QgYmUgYW4gYXJyYXknKTtcbiAgICB9XG5cbiAgICBjb25zdCBjb25kaXRpb24gPSBhcmdzWzFdO1xuICAgIGNvbnN0IHJlc3VsdDogdW5rbm93bltdID0gW107XG5cbiAgICBmb3IgKGNvbnN0IGl0ZW0gb2YgYXJyYXkpIHtcbiAgICAgIHRoaXMuY2hlY2tJdGVyYXRpb25MaW1pdCgpO1xuICAgICAgY29uc3QgaXRlbUNvbnRleHQ6IEV2YWx1YXRpb25Db250ZXh0ID0ge1xuICAgICAgICAuLi5jb250ZXh0LFxuICAgICAgICBleHRyYToge1xuICAgICAgICAgIC4uLmNvbnRleHQuZXh0cmEsXG4gICAgICAgICAgX2N1cnJlbnRJdGVtOiBpdGVtLFxuICAgICAgICB9LFxuICAgICAgICB2YXJpYWJsZXM6IHtcbiAgICAgICAgICAuLi5jb250ZXh0LnZhcmlhYmxlcyxcbiAgICAgICAgICBpdDogaXRlbSxcbiAgICAgICAgfSxcbiAgICAgIH07XG4gICAgICBjb25zdCBrZWVwID0gdGhpcy5ldmFsdWF0ZU5vZGUoY29uZGl0aW9uLCBpdGVtQ29udGV4dCk7XG4gICAgICBpZiAodGhpcy50b0Jvb2woa2VlcCkpIHtcbiAgICAgICAgcmVzdWx0LnB1c2goaXRlbSk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIHByaXZhdGUgZXZhbHVhdGVNYXAoYXJnczogQVNUTm9kZVtdLCBjb250ZXh0OiBFdmFsdWF0aW9uQ29udGV4dCk6IHVua25vd25bXSB7XG4gICAgY29uc3QgYXJyYXkgPSB0aGlzLmV2YWx1YXRlTm9kZShhcmdzWzBdLCBjb250ZXh0KTtcbiAgICBpZiAoIUFycmF5LmlzQXJyYXkoYXJyYXkpKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ01BUCBmaXJzdCBhcmd1bWVudCBtdXN0IGJlIGFuIGFycmF5Jyk7XG4gICAgfVxuXG4gICAgY29uc3QgZXhwcmVzc2lvbiA9IGFyZ3NbMV07XG4gICAgY29uc3QgcmVzdWx0OiB1bmtub3duW10gPSBbXTtcblxuICAgIGZvciAoY29uc3QgaXRlbSBvZiBhcnJheSkge1xuICAgICAgdGhpcy5jaGVja0l0ZXJhdGlvbkxpbWl0KCk7XG4gICAgICBjb25zdCBpdGVtQ29udGV4dDogRXZhbHVhdGlvbkNvbnRleHQgPSB7XG4gICAgICAgIC4uLmNvbnRleHQsXG4gICAgICAgIGV4dHJhOiB7XG4gICAgICAgICAgLi4uY29udGV4dC5leHRyYSxcbiAgICAgICAgICBfY3VycmVudEl0ZW06IGl0ZW0sXG4gICAgICAgIH0sXG4gICAgICAgIHZhcmlhYmxlczoge1xuICAgICAgICAgIC4uLmNvbnRleHQudmFyaWFibGVzLFxuICAgICAgICAgIGl0OiBpdGVtLFxuICAgICAgICB9LFxuICAgICAgfTtcbiAgICAgIHJlc3VsdC5wdXNoKHRoaXMuZXZhbHVhdGVOb2RlKGV4cHJlc3Npb24sIGl0ZW1Db250ZXh0KSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIHByaXZhdGUgZXZhbHVhdGVNZW1iZXJBY2Nlc3MoXG4gICAgbm9kZTogeyBvYmplY3Q6IEFTVE5vZGU7IHByb3BlcnR5OiBzdHJpbmcgfSxcbiAgICBjb250ZXh0OiBFdmFsdWF0aW9uQ29udGV4dFxuICApOiB1bmtub3duIHtcbiAgICBjb25zdCBvYmplY3QgPSB0aGlzLmV2YWx1YXRlTm9kZShub2RlLm9iamVjdCwgY29udGV4dCk7XG5cbiAgICBpZiAob2JqZWN0ID09PSBudWxsIHx8IG9iamVjdCA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICBpZiAodGhpcy5zdHJpY3RNb2RlKSB7XG4gICAgICAgIHRocm93IG5ldyBQcm9wZXJ0eUFjY2Vzc0Vycm9yKG5vZGUucHJvcGVydHksICdudWxsJyk7XG4gICAgICB9XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBpZiAodHlwZW9mIG9iamVjdCAhPT0gJ29iamVjdCcpIHtcbiAgICAgIHRocm93IG5ldyBQcm9wZXJ0eUFjY2Vzc0Vycm9yKG5vZGUucHJvcGVydHksIHR5cGVvZiBvYmplY3QpO1xuICAgIH1cblxuICAgIGNvbnN0IHZhbHVlID0gKG9iamVjdCBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPilbbm9kZS5wcm9wZXJ0eV07XG4gICAgcmV0dXJuIHRoaXMubWF5YmVDb252ZXJ0VG9EZWNpbWFsKHZhbHVlKTtcbiAgfVxuXG4gIHByaXZhdGUgZXZhbHVhdGVJbmRleEFjY2VzcyhcbiAgICBub2RlOiB7IG9iamVjdDogQVNUTm9kZTsgaW5kZXg6IEFTVE5vZGUgfSxcbiAgICBjb250ZXh0OiBFdmFsdWF0aW9uQ29udGV4dFxuICApOiB1bmtub3duIHtcbiAgICBjb25zdCBvYmplY3QgPSB0aGlzLmV2YWx1YXRlTm9kZShub2RlLm9iamVjdCwgY29udGV4dCk7XG4gICAgY29uc3QgaW5kZXggPSB0aGlzLmV2YWx1YXRlTm9kZShub2RlLmluZGV4LCBjb250ZXh0KTtcblxuICAgIGlmIChvYmplY3QgPT09IG51bGwgfHwgb2JqZWN0ID09PSB1bmRlZmluZWQpIHtcbiAgICAgIGlmICh0aGlzLnN0cmljdE1vZGUpIHtcbiAgICAgICAgdGhyb3cgbmV3IEluZGV4QWNjZXNzRXJyb3IoaW5kZXgsICdudWxsJyk7XG4gICAgICB9XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBpZiAoQXJyYXkuaXNBcnJheShvYmplY3QpKSB7XG4gICAgICBjb25zdCBpZHggPSB0aGlzLnRvTnVtYmVyKGluZGV4KTtcbiAgICAgIGlmIChpZHggPCAwIHx8IGlkeCA+PSBvYmplY3QubGVuZ3RoKSB7XG4gICAgICAgIHJldHVybiBudWxsO1xuICAgICAgfVxuICAgICAgcmV0dXJuIHRoaXMubWF5YmVDb252ZXJ0VG9EZWNpbWFsKG9iamVjdFtpZHhdKTtcbiAgICB9XG5cbiAgICBpZiAodHlwZW9mIG9iamVjdCA9PT0gJ29iamVjdCcpIHtcbiAgICAgIGNvbnN0IGtleSA9IFN0cmluZyhpbmRleCk7XG4gICAgICBjb25zdCB2YWx1ZSA9IChvYmplY3QgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pW2tleV07XG4gICAgICByZXR1cm4gdGhpcy5tYXliZUNvbnZlcnRUb0RlY2ltYWwodmFsdWUpO1xuICAgIH1cblxuICAgIHRocm93IG5ldyBJbmRleEFjY2Vzc0Vycm9yKGluZGV4LCB0eXBlb2Ygb2JqZWN0KTtcbiAgfVxuXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiAgLy8gSGVscGVyIG1ldGhvZHNcbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4gIHByaXZhdGUgaXNOdW1lcmljKHZhbHVlOiB1bmtub3duKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHZhbHVlIGluc3RhbmNlb2YgRGVjaW1hbCB8fCB0eXBlb2YgdmFsdWUgPT09ICdudW1iZXInO1xuICB9XG5cbiAgcHJpdmF0ZSB0b0RlY2ltYWwodmFsdWU6IHVua25vd24pOiBEZWNpbWFsIHtcbiAgICBpZiAodmFsdWUgaW5zdGFuY2VvZiBEZWNpbWFsKSByZXR1cm4gdmFsdWU7XG4gICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ251bWJlcicpIHJldHVybiB0aGlzLmRlY2ltYWxVdGlscy5mcm9tKHZhbHVlKTtcbiAgICBpZiAodHlwZW9mIHZhbHVlID09PSAnc3RyaW5nJykgcmV0dXJuIHRoaXMuZGVjaW1hbFV0aWxzLmZyb20odmFsdWUpO1xuICAgIHRocm93IG5ldyBJbnZhbGlkT3BlcmF0aW9uRXJyb3IoJ3RvRGVjaW1hbCcsIFt0aGlzLnR5cGVPZih2YWx1ZSldKTtcbiAgfVxuXG4gIHByaXZhdGUgdG9OdW1iZXIodmFsdWU6IHVua25vd24pOiBudW1iZXIge1xuICAgIGlmICh2YWx1ZSBpbnN0YW5jZW9mIERlY2ltYWwpIHJldHVybiB2YWx1ZS50b051bWJlcigpO1xuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09ICdudW1iZXInKSByZXR1cm4gdmFsdWU7XG4gICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ3N0cmluZycpIHJldHVybiBwYXJzZUZsb2F0KHZhbHVlKTtcbiAgICB0aHJvdyBuZXcgSW52YWxpZE9wZXJhdGlvbkVycm9yKCd0b051bWJlcicsIFt0aGlzLnR5cGVPZih2YWx1ZSldKTtcbiAgfVxuXG4gIHByaXZhdGUgdG9Cb29sKHZhbHVlOiB1bmtub3duKTogYm9vbGVhbiB7XG4gICAgaWYgKHZhbHVlIGluc3RhbmNlb2YgRGVjaW1hbCkgcmV0dXJuICF2YWx1ZS5pc1plcm8oKTtcbiAgICBpZiAodHlwZW9mIHZhbHVlID09PSAnYm9vbGVhbicpIHJldHVybiB2YWx1ZTtcbiAgICBpZiAodmFsdWUgPT09IG51bGwgfHwgdmFsdWUgPT09IHVuZGVmaW5lZCkgcmV0dXJuIGZhbHNlO1xuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09ICdudW1iZXInKSByZXR1cm4gdmFsdWUgIT09IDA7XG4gICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ3N0cmluZycpIHJldHVybiB2YWx1ZS5sZW5ndGggPiAwO1xuICAgIGlmIChBcnJheS5pc0FycmF5KHZhbHVlKSkgcmV0dXJuIHZhbHVlLmxlbmd0aCA+IDA7XG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICBwcml2YXRlIHR5cGVPZih2YWx1ZTogdW5rbm93bik6IHN0cmluZyB7XG4gICAgaWYgKHZhbHVlID09PSBudWxsKSByZXR1cm4gJ251bGwnO1xuICAgIGlmICh2YWx1ZSBpbnN0YW5jZW9mIERlY2ltYWwpIHJldHVybiAnZGVjaW1hbCc7XG4gICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSByZXR1cm4gJ2FycmF5JztcbiAgICByZXR1cm4gdHlwZW9mIHZhbHVlO1xuICB9XG5cbiAgcHJpdmF0ZSBhZGQobGVmdDogdW5rbm93biwgcmlnaHQ6IHVua25vd24pOiB1bmtub3duIHtcbiAgICAvLyBTdHJpbmcgY29uY2F0ZW5hdGlvblxuICAgIGlmICh0eXBlb2YgbGVmdCA9PT0gJ3N0cmluZycgfHwgdHlwZW9mIHJpZ2h0ID09PSAnc3RyaW5nJykge1xuICAgICAgY29uc3QgbGVmdFN0ciA9IGxlZnQgaW5zdGFuY2VvZiBEZWNpbWFsID8gbGVmdC50b1N0cmluZygpIDogU3RyaW5nKGxlZnQpO1xuICAgICAgY29uc3QgcmlnaHRTdHIgPSByaWdodCBpbnN0YW5jZW9mIERlY2ltYWwgPyByaWdodC50b1N0cmluZygpIDogU3RyaW5nKHJpZ2h0KTtcbiAgICAgIHJldHVybiBsZWZ0U3RyICsgcmlnaHRTdHI7XG4gICAgfVxuXG4gICAgLy8gTnVtZXJpYyBhZGRpdGlvblxuICAgIGlmICh0aGlzLmlzTnVtZXJpYyhsZWZ0KSAmJiB0aGlzLmlzTnVtZXJpYyhyaWdodCkpIHtcbiAgICAgIHJldHVybiB0aGlzLmRlY2ltYWxVdGlscy5hZGQodGhpcy50b0RlY2ltYWwobGVmdCksIHRoaXMudG9EZWNpbWFsKHJpZ2h0KSk7XG4gICAgfVxuXG4gICAgdGhyb3cgbmV3IEludmFsaWRPcGVyYXRpb25FcnJvcignKycsIFt0aGlzLnR5cGVPZihsZWZ0KSwgdGhpcy50eXBlT2YocmlnaHQpXSk7XG4gIH1cblxuICBwcml2YXRlIHN1YnRyYWN0KGxlZnQ6IHVua25vd24sIHJpZ2h0OiB1bmtub3duKTogRGVjaW1hbCB7XG4gICAgaWYgKCF0aGlzLmlzTnVtZXJpYyhsZWZ0KSB8fCAhdGhpcy5pc051bWVyaWMocmlnaHQpKSB7XG4gICAgICB0aHJvdyBuZXcgSW52YWxpZE9wZXJhdGlvbkVycm9yKCctJywgW3RoaXMudHlwZU9mKGxlZnQpLCB0aGlzLnR5cGVPZihyaWdodCldKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuZGVjaW1hbFV0aWxzLnN1YnRyYWN0KHRoaXMudG9EZWNpbWFsKGxlZnQpLCB0aGlzLnRvRGVjaW1hbChyaWdodCkpO1xuICB9XG5cbiAgcHJpdmF0ZSBtdWx0aXBseShsZWZ0OiB1bmtub3duLCByaWdodDogdW5rbm93bik6IERlY2ltYWwge1xuICAgIGlmICghdGhpcy5pc051bWVyaWMobGVmdCkgfHwgIXRoaXMuaXNOdW1lcmljKHJpZ2h0KSkge1xuICAgICAgdGhyb3cgbmV3IEludmFsaWRPcGVyYXRpb25FcnJvcignKicsIFt0aGlzLnR5cGVPZihsZWZ0KSwgdGhpcy50eXBlT2YocmlnaHQpXSk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLmRlY2ltYWxVdGlscy5tdWx0aXBseSh0aGlzLnRvRGVjaW1hbChsZWZ0KSwgdGhpcy50b0RlY2ltYWwocmlnaHQpKTtcbiAgfVxuXG4gIHByaXZhdGUgZGl2aWRlKGxlZnQ6IHVua25vd24sIHJpZ2h0OiB1bmtub3duKTogRGVjaW1hbCB7XG4gICAgaWYgKCF0aGlzLmlzTnVtZXJpYyhsZWZ0KSB8fCAhdGhpcy5pc051bWVyaWMocmlnaHQpKSB7XG4gICAgICB0aHJvdyBuZXcgSW52YWxpZE9wZXJhdGlvbkVycm9yKCcvJywgW3RoaXMudHlwZU9mKGxlZnQpLCB0aGlzLnR5cGVPZihyaWdodCldKTtcbiAgICB9XG4gICAgY29uc3QgZGl2aXNvciA9IHRoaXMudG9EZWNpbWFsKHJpZ2h0KTtcbiAgICBpZiAoZGl2aXNvci5pc1plcm8oKSkge1xuICAgICAgdGhyb3cgbmV3IERpdmlzaW9uQnlaZXJvRXJyb3IoKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuZGVjaW1hbFV0aWxzLmRpdmlkZSh0aGlzLnRvRGVjaW1hbChsZWZ0KSwgZGl2aXNvcik7XG4gIH1cblxuICBwcml2YXRlIG1vZHVsbyhsZWZ0OiB1bmtub3duLCByaWdodDogdW5rbm93bik6IERlY2ltYWwge1xuICAgIGlmICghdGhpcy5pc051bWVyaWMobGVmdCkgfHwgIXRoaXMuaXNOdW1lcmljKHJpZ2h0KSkge1xuICAgICAgdGhyb3cgbmV3IEludmFsaWRPcGVyYXRpb25FcnJvcignJScsIFt0aGlzLnR5cGVPZihsZWZ0KSwgdGhpcy50eXBlT2YocmlnaHQpXSk7XG4gICAgfVxuICAgIGNvbnN0IGRpdmlzb3IgPSB0aGlzLnRvRGVjaW1hbChyaWdodCk7XG4gICAgaWYgKGRpdmlzb3IuaXNaZXJvKCkpIHtcbiAgICAgIHRocm93IG5ldyBEaXZpc2lvbkJ5WmVyb0Vycm9yKCk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLmRlY2ltYWxVdGlscy5tb2R1bG8odGhpcy50b0RlY2ltYWwobGVmdCksIGRpdmlzb3IpO1xuICB9XG5cbiAgcHJpdmF0ZSBwb3dlcihsZWZ0OiB1bmtub3duLCByaWdodDogdW5rbm93bik6IERlY2ltYWwge1xuICAgIGlmICghdGhpcy5pc051bWVyaWMobGVmdCkgfHwgIXRoaXMuaXNOdW1lcmljKHJpZ2h0KSkge1xuICAgICAgdGhyb3cgbmV3IEludmFsaWRPcGVyYXRpb25FcnJvcignXicsIFt0aGlzLnR5cGVPZihsZWZ0KSwgdGhpcy50eXBlT2YocmlnaHQpXSk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLmRlY2ltYWxVdGlscy5wb3dlcih0aGlzLnRvRGVjaW1hbChsZWZ0KSwgdGhpcy50b051bWJlcihyaWdodCkpO1xuICB9XG5cbiAgcHJpdmF0ZSBuZWdhdGUodmFsdWU6IHVua25vd24pOiBEZWNpbWFsIHtcbiAgICBpZiAoIXRoaXMuaXNOdW1lcmljKHZhbHVlKSkge1xuICAgICAgdGhyb3cgbmV3IEludmFsaWRPcGVyYXRpb25FcnJvcignLScsIFt0aGlzLnR5cGVPZih2YWx1ZSldKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuZGVjaW1hbFV0aWxzLm5lZ2F0ZSh0aGlzLnRvRGVjaW1hbCh2YWx1ZSkpO1xuICB9XG5cbiAgcHJpdmF0ZSBlcXVhbHMobGVmdDogdW5rbm93biwgcmlnaHQ6IHVua25vd24pOiBib29sZWFuIHtcbiAgICBpZiAobGVmdCBpbnN0YW5jZW9mIERlY2ltYWwgJiYgcmlnaHQgaW5zdGFuY2VvZiBEZWNpbWFsKSB7XG4gICAgICByZXR1cm4gbGVmdC5lcXVhbHMocmlnaHQpO1xuICAgIH1cbiAgICBpZiAobGVmdCBpbnN0YW5jZW9mIERlY2ltYWwgJiYgdHlwZW9mIHJpZ2h0ID09PSAnbnVtYmVyJykge1xuICAgICAgcmV0dXJuIGxlZnQuZXF1YWxzKHRoaXMuZGVjaW1hbFV0aWxzLmZyb20ocmlnaHQpKTtcbiAgICB9XG4gICAgaWYgKHR5cGVvZiBsZWZ0ID09PSAnbnVtYmVyJyAmJiByaWdodCBpbnN0YW5jZW9mIERlY2ltYWwpIHtcbiAgICAgIHJldHVybiB0aGlzLmRlY2ltYWxVdGlscy5mcm9tKGxlZnQpLmVxdWFscyhyaWdodCk7XG4gICAgfVxuICAgIHJldHVybiBsZWZ0ID09PSByaWdodDtcbiAgfVxuXG4gIHByaXZhdGUgbGVzc1RoYW4obGVmdDogdW5rbm93biwgcmlnaHQ6IHVua25vd24pOiBib29sZWFuIHtcbiAgICBpZiAodGhpcy5pc051bWVyaWMobGVmdCkgJiYgdGhpcy5pc051bWVyaWMocmlnaHQpKSB7XG4gICAgICByZXR1cm4gdGhpcy5kZWNpbWFsVXRpbHMubGVzc1RoYW4odGhpcy50b0RlY2ltYWwobGVmdCksIHRoaXMudG9EZWNpbWFsKHJpZ2h0KSk7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgbGVmdCA9PT0gJ3N0cmluZycgJiYgdHlwZW9mIHJpZ2h0ID09PSAnc3RyaW5nJykge1xuICAgICAgcmV0dXJuIGxlZnQgPCByaWdodDtcbiAgICB9XG4gICAgdGhyb3cgbmV3IEludmFsaWRPcGVyYXRpb25FcnJvcignPCcsIFt0aGlzLnR5cGVPZihsZWZ0KSwgdGhpcy50eXBlT2YocmlnaHQpXSk7XG4gIH1cblxuICBwcml2YXRlIGdyZWF0ZXJUaGFuKGxlZnQ6IHVua25vd24sIHJpZ2h0OiB1bmtub3duKTogYm9vbGVhbiB7XG4gICAgaWYgKHRoaXMuaXNOdW1lcmljKGxlZnQpICYmIHRoaXMuaXNOdW1lcmljKHJpZ2h0KSkge1xuICAgICAgcmV0dXJuIHRoaXMuZGVjaW1hbFV0aWxzLmdyZWF0ZXJUaGFuKHRoaXMudG9EZWNpbWFsKGxlZnQpLCB0aGlzLnRvRGVjaW1hbChyaWdodCkpO1xuICAgIH1cbiAgICBpZiAodHlwZW9mIGxlZnQgPT09ICdzdHJpbmcnICYmIHR5cGVvZiByaWdodCA9PT0gJ3N0cmluZycpIHtcbiAgICAgIHJldHVybiBsZWZ0ID4gcmlnaHQ7XG4gICAgfVxuICAgIHRocm93IG5ldyBJbnZhbGlkT3BlcmF0aW9uRXJyb3IoJz4nLCBbdGhpcy50eXBlT2YobGVmdCksIHRoaXMudHlwZU9mKHJpZ2h0KV0pO1xuICB9XG5cbiAgcHJpdmF0ZSBsZXNzVGhhbk9yRXF1YWwobGVmdDogdW5rbm93biwgcmlnaHQ6IHVua25vd24pOiBib29sZWFuIHtcbiAgICBpZiAodGhpcy5pc051bWVyaWMobGVmdCkgJiYgdGhpcy5pc051bWVyaWMocmlnaHQpKSB7XG4gICAgICByZXR1cm4gdGhpcy5kZWNpbWFsVXRpbHMubGVzc1RoYW5PckVxdWFsKHRoaXMudG9EZWNpbWFsKGxlZnQpLCB0aGlzLnRvRGVjaW1hbChyaWdodCkpO1xuICAgIH1cbiAgICBpZiAodHlwZW9mIGxlZnQgPT09ICdzdHJpbmcnICYmIHR5cGVvZiByaWdodCA9PT0gJ3N0cmluZycpIHtcbiAgICAgIHJldHVybiBsZWZ0IDw9IHJpZ2h0O1xuICAgIH1cbiAgICB0aHJvdyBuZXcgSW52YWxpZE9wZXJhdGlvbkVycm9yKCc8PScsIFt0aGlzLnR5cGVPZihsZWZ0KSwgdGhpcy50eXBlT2YocmlnaHQpXSk7XG4gIH1cblxuICBwcml2YXRlIGdyZWF0ZXJUaGFuT3JFcXVhbChsZWZ0OiB1bmtub3duLCByaWdodDogdW5rbm93bik6IGJvb2xlYW4ge1xuICAgIGlmICh0aGlzLmlzTnVtZXJpYyhsZWZ0KSAmJiB0aGlzLmlzTnVtZXJpYyhyaWdodCkpIHtcbiAgICAgIHJldHVybiB0aGlzLmRlY2ltYWxVdGlscy5ncmVhdGVyVGhhbk9yRXF1YWwodGhpcy50b0RlY2ltYWwobGVmdCksIHRoaXMudG9EZWNpbWFsKHJpZ2h0KSk7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgbGVmdCA9PT0gJ3N0cmluZycgJiYgdHlwZW9mIHJpZ2h0ID09PSAnc3RyaW5nJykge1xuICAgICAgcmV0dXJuIGxlZnQgPj0gcmlnaHQ7XG4gICAgfVxuICAgIHRocm93IG5ldyBJbnZhbGlkT3BlcmF0aW9uRXJyb3IoJz49JywgW3RoaXMudHlwZU9mKGxlZnQpLCB0aGlzLnR5cGVPZihyaWdodCldKTtcbiAgfVxufVxuIl19