@decaf-ts/for-nest 0.0.2 → 0.0.3

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 (47) hide show
  1. package/dist/for-nest.cjs +468 -15
  2. package/dist/for-nest.esm.cjs +468 -14
  3. package/lib/RepoFactory.cjs +27 -0
  4. package/lib/RepoFactory.d.ts +6 -0
  5. package/lib/core-module.cjs +27 -7
  6. package/lib/core-module.d.ts +7 -2
  7. package/lib/decoration.cjs +123 -0
  8. package/lib/decoration.d.ts +1 -0
  9. package/lib/decorators/ApiProperty.cjs +63 -0
  10. package/lib/decorators/ApiProperty.d.ts +27 -0
  11. package/lib/decorators/helpers.cjs +140 -0
  12. package/lib/decorators/helpers.d.ts +8 -0
  13. package/lib/decorators/index.cjs +19 -0
  14. package/lib/decorators/index.d.ts +2 -0
  15. package/lib/esm/RepoFactory.d.ts +6 -0
  16. package/lib/esm/RepoFactory.js +24 -0
  17. package/lib/esm/core-module.d.ts +7 -2
  18. package/lib/esm/core-module.js +27 -7
  19. package/lib/esm/decoration.d.ts +1 -0
  20. package/lib/esm/decoration.js +121 -0
  21. package/lib/esm/decorators/ApiProperty.d.ts +27 -0
  22. package/lib/esm/decorators/ApiProperty.js +57 -0
  23. package/lib/esm/decorators/helpers.d.ts +8 -0
  24. package/lib/esm/decorators/helpers.js +132 -0
  25. package/lib/esm/decorators/index.d.ts +2 -0
  26. package/lib/esm/decorators/index.js +3 -0
  27. package/lib/esm/index.d.ts +7 -2
  28. package/lib/esm/index.js +8 -3
  29. package/lib/esm/model-module.d.ts +8 -0
  30. package/lib/esm/model-module.js +119 -0
  31. package/lib/esm/module.d.ts +1 -1
  32. package/lib/esm/module.js +11 -3
  33. package/lib/esm/types.d.ts +1 -0
  34. package/lib/esm/types.js +1 -1
  35. package/lib/esm/utils.d.ts +2 -0
  36. package/lib/esm/utils.js +11 -0
  37. package/lib/index.cjs +8 -3
  38. package/lib/index.d.ts +7 -2
  39. package/lib/model-module.cjs +122 -0
  40. package/lib/model-module.d.ts +8 -0
  41. package/lib/module.cjs +11 -3
  42. package/lib/module.d.ts +1 -1
  43. package/lib/types.cjs +1 -1
  44. package/lib/types.d.ts +1 -0
  45. package/lib/utils.cjs +14 -0
  46. package/lib/utils.d.ts +2 -0
  47. package/package.json +4 -2
package/dist/for-nest.cjs CHANGED
@@ -1,19 +1,332 @@
1
1
  (function (global, factory) {
2
- typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('@decaf-ts/decoration'), require('tslib'), require('@nestjs/common'), require('@nestjs/core'), require('@decaf-ts/logging/lib/LoggedClass')) :
3
- typeof define === 'function' && define.amd ? define(['exports', '@decaf-ts/decoration', 'tslib', '@nestjs/common', '@nestjs/core', '@decaf-ts/logging/lib/LoggedClass'], factory) :
4
- (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global["for-nest"] = {}, global.decoration, global.tslib, global.common, global.core, global.LoggedClass));
5
- })(this, (function (exports, decoration, tslib, common, core, LoggedClass) { 'use strict';
2
+ typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('@decaf-ts/decoration'), require('@decaf-ts/injectable-decorators'), require('@nestjs/common'), require('@decaf-ts/decorator-validation'), require('@nestjs/swagger/dist/decorators/helpers'), require('@nestjs/swagger/dist/utils/enum.utils'), require('@nestjs/swagger/dist/constants'), require('@nestjs/common/constants'), require('@nestjs/common/utils/shared.utils'), require('lodash'), require('@nestjs/swagger/dist/plugin/plugin-constants'), require('@decaf-ts/core'), require('tslib'), require('@nestjs/core'), require('@decaf-ts/logging'), require('@nestjs/swagger'), require('@decaf-ts/db-decorators')) :
3
+ typeof define === 'function' && define.amd ? define(['exports', '@decaf-ts/decoration', '@decaf-ts/injectable-decorators', '@nestjs/common', '@decaf-ts/decorator-validation', '@nestjs/swagger/dist/decorators/helpers', '@nestjs/swagger/dist/utils/enum.utils', '@nestjs/swagger/dist/constants', '@nestjs/common/constants', '@nestjs/common/utils/shared.utils', 'lodash', '@nestjs/swagger/dist/plugin/plugin-constants', '@decaf-ts/core', 'tslib', '@nestjs/core', '@decaf-ts/logging', '@nestjs/swagger', '@decaf-ts/db-decorators'], factory) :
4
+ (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global["for-nest"] = {}, global.decoration, global.injectableDecorators, global.common, global.decoratorValidation, global.helpers, global.enum_utils, global.constants, global.constants$1, global.shared_utils, global.lodash, global.pluginConstants, global.core, global.tslib, global.core$1, global.logging, global.swagger, global.dbDecorators));
5
+ })(this, (function (exports, decoration, injectableDecorators, common, decoratorValidation, helpers, enum_utils, constants, constants$1, shared_utils, lodash, pluginConstants, core, tslib, core$1, logging, swagger, dbDecorators) { 'use strict';
6
+
7
+ /* eslint-disable @typescript-eslint/no-unsafe-function-type */
8
+ function createMethodDecorator(metakey, metadata, { overrideExisting } = { overrideExisting: true }) {
9
+ return (target, key, descriptor) => {
10
+ if (typeof metadata === "object") {
11
+ const prevValue = Reflect.getMetadata(metakey, descriptor.value);
12
+ if (prevValue && !overrideExisting) {
13
+ return descriptor;
14
+ }
15
+ Reflect.defineMetadata(metakey, { ...prevValue, ...metadata }, descriptor.value);
16
+ return descriptor;
17
+ }
18
+ Reflect.defineMetadata(metakey, metadata, descriptor.value);
19
+ return descriptor;
20
+ };
21
+ }
22
+ function createClassDecorator(metakey, metadata = []) {
23
+ return (target) => {
24
+ const prevValue = Reflect.getMetadata(metakey, target) || [];
25
+ Reflect.defineMetadata(metakey, [...prevValue, ...metadata], target);
26
+ return target;
27
+ };
28
+ }
29
+ function createPropertyDecorator(metakey, metadata, overrideExisting = true) {
30
+ return (target, propertyKey) => {
31
+ const properties = Reflect.getMetadata(constants.DECORATORS.API_MODEL_PROPERTIES_ARRAY, target) || [];
32
+ const key = `:${propertyKey}`;
33
+ if (!properties.includes(key)) {
34
+ Reflect.defineMetadata(constants.DECORATORS.API_MODEL_PROPERTIES_ARRAY, [...properties, `:${propertyKey}`], target);
35
+ }
36
+ const existingMetadata = Reflect.getMetadata(metakey, target, propertyKey);
37
+ if (existingMetadata) {
38
+ const newMetadata = lodash.pickBy(metadata, lodash.negate(lodash.isUndefined));
39
+ const metadataToSave = overrideExisting
40
+ ? {
41
+ ...existingMetadata,
42
+ ...newMetadata,
43
+ }
44
+ : {
45
+ ...newMetadata,
46
+ ...existingMetadata,
47
+ };
48
+ Reflect.defineMetadata(metakey, metadataToSave, target, propertyKey);
49
+ }
50
+ else {
51
+ const type =
52
+ // @ts-expect-error nest js override
53
+ target?.constructor?.[pluginConstants.METADATA_FACTORY_NAME]?.()[propertyKey]?.type ??
54
+ Reflect.getMetadata("design:type", target, propertyKey);
55
+ Reflect.defineMetadata(metakey, {
56
+ type,
57
+ ...lodash.pickBy(metadata, lodash.negate(lodash.isUndefined)),
58
+ }, target, propertyKey);
59
+ }
60
+ };
61
+ }
62
+ function createMixedDecorator(metakey, metadata) {
63
+ return (target, key, descriptor) => {
64
+ if (descriptor) {
65
+ let metadatas;
66
+ if (Array.isArray(metadata)) {
67
+ const previousMetadata = Reflect.getMetadata(metakey, descriptor.value) || [];
68
+ metadatas = [...previousMetadata, ...metadata];
69
+ }
70
+ else {
71
+ const previousMetadata = Reflect.getMetadata(metakey, descriptor.value) || {};
72
+ metadatas = { ...previousMetadata, ...metadata };
73
+ }
74
+ Reflect.defineMetadata(metakey, metadatas, descriptor.value);
75
+ return descriptor;
76
+ }
77
+ let metadatas;
78
+ if (Array.isArray(metadata)) {
79
+ const previousMetadata = Reflect.getMetadata(metakey, target) || [];
80
+ metadatas = [...previousMetadata, ...metadata];
81
+ }
82
+ else {
83
+ const previousMetadata = Reflect.getMetadata(metakey, target) || {};
84
+ metadatas = Object.assign(Object.assign({}, previousMetadata), metadata);
85
+ }
86
+ Reflect.defineMetadata(metakey, metadatas, target);
87
+ return target;
88
+ };
89
+ }
90
+ function createParamDecorator(metadata, initial) {
91
+ return (target, key, descriptor) => {
92
+ const paramOptions = {
93
+ ...initial,
94
+ ...lodash.pickBy(metadata, lodash.negate(lodash.isUndefined)),
95
+ };
96
+ if (descriptor) {
97
+ const parameters = Reflect.getMetadata(constants.DECORATORS.API_PARAMETERS, descriptor.value) || [];
98
+ Reflect.defineMetadata(constants.DECORATORS.API_PARAMETERS, [...parameters, paramOptions], descriptor.value);
99
+ return descriptor;
100
+ }
101
+ if (typeof target === "object") {
102
+ return target;
103
+ }
104
+ const propertyKeys = Object.getOwnPropertyNames(target.prototype);
105
+ for (const propertyKey of propertyKeys) {
106
+ if (shared_utils.isConstructor(propertyKey)) {
107
+ continue;
108
+ }
109
+ const methodDescriptor = Object.getOwnPropertyDescriptor(target.prototype, propertyKey);
110
+ if (!methodDescriptor) {
111
+ continue;
112
+ }
113
+ const isApiMethod = Reflect.hasMetadata(constants$1.METHOD_METADATA, methodDescriptor.value);
114
+ if (!isApiMethod) {
115
+ continue;
116
+ }
117
+ const parameters = Reflect.getMetadata(constants.DECORATORS.API_PARAMETERS, methodDescriptor.value) || [];
118
+ Reflect.defineMetadata(constants.DECORATORS.API_PARAMETERS, [...parameters, paramOptions], methodDescriptor.value);
119
+ }
120
+ };
121
+ }
122
+ function getTypeIsArrayTuple(input, isArrayFlag) {
123
+ if (!input) {
124
+ return [input, isArrayFlag];
125
+ }
126
+ if (isArrayFlag) {
127
+ return [input, isArrayFlag];
128
+ }
129
+ const isInputArray = lodash.isArray(input);
130
+ const type = isInputArray ? input[0] : input;
131
+ return [type, isInputArray];
132
+ }
133
+
134
+ const isEnumArray = (opts) => (opts.isArray && "enum" in opts && opts.enum !== undefined);
135
+ /**
136
+ * @publicApi
137
+ */
138
+ function ApiProperty(options = {}) {
139
+ return createApiPropertyDecorator(options);
140
+ }
141
+ function createApiPropertyDecorator(options = {}, overrideExisting = true) {
142
+ const [type, isArray] = helpers.getTypeIsArrayTuple(options.type, options.isArray);
143
+ options = {
144
+ ...options,
145
+ type,
146
+ isArray,
147
+ };
148
+ if (isEnumArray(options)) {
149
+ options.type = "array";
150
+ const enumValues = enum_utils.getEnumValues(options.enum);
151
+ options.items = {
152
+ type: enum_utils.getEnumType(enumValues),
153
+ enum: enumValues,
154
+ };
155
+ // @ts-expect-error nest swagger override
156
+ delete options.enum;
157
+ }
158
+ else if ("enum" in options && options.enum !== undefined) {
159
+ const enumValues = enum_utils.getEnumValues(options.enum);
160
+ options.enum = enumValues;
161
+ options.type = enum_utils.getEnumType(enumValues);
162
+ }
163
+ if (Array.isArray(options.type)) {
164
+ options.type = "array";
165
+ options.items = {
166
+ type: "array",
167
+ items: {
168
+ type: options.type[0],
169
+ },
170
+ };
171
+ }
172
+ return createPropertyDecorator(constants.DECORATORS.API_MODEL_PROPERTIES, options, overrideExisting);
173
+ }
174
+ function ApiPropertyOptional(options = {}) {
175
+ return ApiProperty({
176
+ ...options,
177
+ required: false,
178
+ });
179
+ }
180
+ function ApiResponseProperty(options = {}) {
181
+ return ApiProperty({
182
+ readOnly: true,
183
+ ...options,
184
+ });
185
+ }
186
+
187
+ decoration.Decoration.for(injectableDecorators.InjectablesKeys.INJECTABLE)
188
+ .extend({
189
+ decorator: function nestInjectable(category, cfg) {
190
+ return common.Injectable({
191
+ scope: cfg.singleton ? common.Scope.DEFAULT : common.Scope.REQUEST,
192
+ durable: cfg.singleton ? undefined : true,
193
+ });
194
+ },
195
+ })
196
+ .apply();
197
+ decoration.Decoration.for(injectableDecorators.InjectablesKeys.INJECT)
198
+ .extend({
199
+ decorator: function nestInject(category,
200
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
201
+ cfg) {
202
+ return function innerNestInject(target, propertyKey, descriptor) {
203
+ return common.Inject(category || target)(target, propertyKey, descriptor);
204
+ };
205
+ },
206
+ })
207
+ .apply();
208
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.REQUIRED)
209
+ .extend(ApiProperty({
210
+ required: true,
211
+ }))
212
+ .apply();
213
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.MAX)
214
+ .extend({
215
+ decorator: function maxDec(max) {
216
+ return ApiProperty({ maximum: max });
217
+ },
218
+ })
219
+ .apply();
220
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.MIN)
221
+ .extend({
222
+ decorator: function minDec(min) {
223
+ return ApiProperty({ minimum: min });
224
+ },
225
+ })
226
+ .apply();
227
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.MAX_LENGTH)
228
+ .extend({
229
+ decorator: function maxLengthDec(max) {
230
+ return ApiProperty({ maxLength: max });
231
+ },
232
+ })
233
+ .apply();
234
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.MIN_LENGTH)
235
+ .extend({
236
+ decorator: function minLengthDec(min) {
237
+ return ApiProperty({ minLength: min });
238
+ },
239
+ })
240
+ .apply();
241
+ //
242
+ // Decoration.for(ValidationKeys.TYPE)
243
+ // .extend({
244
+ // decorator: function typeDec(type: (string | (() => string))[] | string | (() => string)) {
245
+ // return ApiProperty({ type: type as any });
246
+ // },
247
+ // })
248
+ // .apply();
249
+ //
250
+ // Decoration.for(ValidationKeys.DATE)
251
+ // .extend({
252
+ // decorator: function dateDec() {
253
+ // return ApiProperty({ type: Date });
254
+ // },
255
+ // })
256
+ // .apply();
257
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.LIST)
258
+ .extend({
259
+ decorator: function listDec(clazz) {
260
+ const c = Array.isArray(clazz) ? clazz[0] : clazz;
261
+ return ApiProperty({ type: [c] });
262
+ },
263
+ })
264
+ .apply();
265
+ //
266
+ // Decoration.for(ValidationKeys.OPTION)
267
+ // .extend({
268
+ // decorator: function optionDec(options: string[] | Record<string, any>) {
269
+ // const opts = Array.isArray(options) ? options : Object.values(options);
270
+ // return ApiProperty({ enum: opts });
271
+ // },
272
+ // })
273
+ // .apply();
274
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.PATTERN)
275
+ .extend({
276
+ decorator: function patternDec(pat) {
277
+ return ApiProperty({
278
+ pattern: typeof pat === "string" ? pat : pat.source,
279
+ });
280
+ },
281
+ })
282
+ .apply();
283
+ decoration.Decoration.for(core.PersistenceKeys.COLUMN)
284
+ .extend({
285
+ decorator: function columnDec(name) {
286
+ return ApiProperty({
287
+ name: name,
288
+ });
289
+ },
290
+ })
291
+ .apply();
292
+ decoration.Decoration.for(decoration.DecorationKeys.DESCRIPTION)
293
+ .extend({
294
+ decorator: function descriptionDec(description) {
295
+ return ApiProperty({
296
+ description: description,
297
+ });
298
+ },
299
+ })
300
+ .apply();
6
301
 
7
302
  const DECAF_MODULE_OPTIONS = "DecafModuleOptions";
8
303
  const DECAF_ADAPTER_ID = "DecafAdapter";
9
304
 
10
305
  var DecafCoreModule_1;
11
- let DecafCoreModule = DecafCoreModule_1 = class DecafCoreModule extends LoggedClass.LoggedClass {
306
+ let DecafCoreModule = class DecafCoreModule {
307
+ static { DecafCoreModule_1 = this; }
308
+ static { this._adapterInstance = null; }
309
+ static get log() {
310
+ if (!this._logger)
311
+ this._logger = logging.Logging.for(DecafCoreModule_1);
312
+ return this._logger;
313
+ }
12
314
  constructor(options, moduleRef) {
13
- super();
14
315
  this.options = options;
15
316
  this.moduleRef = moduleRef;
16
317
  }
318
+ static async createAdapter(options) {
319
+ if (!this._adapterInstance) {
320
+ const log = this.log.for(this.createAdapter);
321
+ log.info("Creating adapter instance...");
322
+ this._adapterInstance = new options.adapter(options.conf, options.alias);
323
+ log.info("Adapter instance created successfully!");
324
+ }
325
+ return this._adapterInstance;
326
+ }
327
+ static getAdapterInstance() {
328
+ return this._adapterInstance;
329
+ }
17
330
  static forRoot(options) {
18
331
  const typeOrmModuleOptions = {
19
332
  provide: DECAF_MODULE_OPTIONS,
@@ -21,7 +334,7 @@
21
334
  };
22
335
  const adapter = {
23
336
  useFactory: async (opts) => {
24
- return new opts.adapter(opts.conf, opts.alias);
337
+ return DecafCoreModule_1.createAdapter(opts);
25
338
  },
26
339
  provide: DECAF_ADAPTER_ID,
27
340
  durable: true,
@@ -37,15 +350,17 @@
37
350
  };
38
351
  }
39
352
  async onApplicationShutdown() {
353
+ const log = DecafCoreModule_1.log.for(this.onApplicationShutdown);
40
354
  const adapter = this.moduleRef.get(DECAF_ADAPTER_ID);
41
355
  try {
42
356
  if (adapter) {
43
- this.log.info("Shutting down");
357
+ log.info("Shutting down");
44
358
  await adapter.shutdown();
359
+ DecafCoreModule_1._adapterInstance = null;
45
360
  }
46
361
  }
47
362
  catch (e) {
48
- this.log.error(`Failed to shutdown application`, e);
363
+ log.error(`Failed to shutdown application`, e);
49
364
  }
50
365
  }
51
366
  };
@@ -53,18 +368,145 @@
53
368
  common.Global(),
54
369
  common.Module({}),
55
370
  tslib.__param(0, common.Inject(DECAF_MODULE_OPTIONS)),
56
- tslib.__metadata("design:paramtypes", [Object, core.ModuleRef])
371
+ tslib.__metadata("design:paramtypes", [Object, core$1.ModuleRef])
57
372
  ], DecafCoreModule);
58
373
 
374
+ function repoForModel(model) {
375
+ const m = decoratorValidation.Model.get(model);
376
+ if (!m)
377
+ throw new dbDecorators.InternalError(`Failed to find repository for ${model}`);
378
+ const repo = core.Repository.forModel(m);
379
+ return repo;
380
+ }
381
+
382
+ exports.RepoFactory = class RepoFactory {
383
+ for(model) {
384
+ model = typeof model === "string" ? decoratorValidation.Model.get(model) : model;
385
+ if (!model) {
386
+ throw new dbDecorators.InternalError(`Failed to find repository for ${model}`);
387
+ }
388
+ return core.Repository.forModel(model);
389
+ }
390
+ };
391
+ exports.RepoFactory = tslib.__decorate([
392
+ common.Injectable()
393
+ ], exports.RepoFactory);
394
+
395
+ var DecafModelModule_1;
396
+ let DecafModelModule = class DecafModelModule {
397
+ static { DecafModelModule_1 = this; }
398
+ static get log() {
399
+ if (!this._logger)
400
+ this._logger = logging.Logging.for(DecafModelModule_1);
401
+ return this._logger;
402
+ }
403
+ static toModelController(modelClass) {
404
+ const log = this.log.for(this.toModelController);
405
+ log.debug(`Creating model controller... ${modelClass.name}`);
406
+ const modelName = logging.toKebabCase(core.Repository.table(modelClass));
407
+ const route = modelName;
408
+ let DynamicModelController = class DynamicModelController extends logging.LoggedClass {
409
+ constructor(repoFactory) {
410
+ super();
411
+ this.repoFactory = repoFactory;
412
+ try {
413
+ this.repo = this.repoFactory.for(modelClass.name);
414
+ }
415
+ catch (e) {
416
+ log.error(`Failed to get repository for ${modelClass.name}`, e);
417
+ }
418
+ }
419
+ async create(data) {
420
+ const log = this.log.for(this.create);
421
+ log.verbose(`creating new ${modelName}`);
422
+ const r = repoForModel("Account");
423
+ const created = await r.create(data);
424
+ log.info(`created new ${modelName} with id ${created[r.pk]}`);
425
+ return created;
426
+ }
427
+ async read(id) {
428
+ const log = this.log.for(this.read);
429
+ log.debug(`reading ${modelName} with ${this.repo.pk} ${id}`);
430
+ const read = await this.repo.read(id);
431
+ log.info(`read ${modelName} with id ${read[this.repo.pk]}`);
432
+ return read;
433
+ }
434
+ };
435
+ tslib.__decorate([
436
+ common.Post(),
437
+ swagger.ApiOperation({ summary: `Create a new ${modelName}.` }),
438
+ swagger.ApiBody({
439
+ description: `Payload for ${modelName}`,
440
+ schema: { $ref: swagger.getSchemaPath(modelClass) },
441
+ }),
442
+ swagger.ApiCreatedResponse({ description: `${modelName} created successfully.` }),
443
+ swagger.ApiBadRequestResponse({ description: "Payload validation failed." }),
444
+ swagger.ApiUnprocessableEntityResponse({
445
+ description: "Repository rejected the provided payload.",
446
+ }),
447
+ tslib.__param(0, common.Body()),
448
+ tslib.__metadata("design:type", Function),
449
+ tslib.__metadata("design:paramtypes", [Object]),
450
+ tslib.__metadata("design:returntype", Promise)
451
+ ], DynamicModelController.prototype, "create", null);
452
+ tslib.__decorate([
453
+ common.Get(":id"),
454
+ swagger.ApiOperation({ summary: `Retrieve a ${modelName} by id.` }),
455
+ swagger.ApiParam({
456
+ name: "id",
457
+ description: "Primary key",
458
+ example: "1234-5678",
459
+ }),
460
+ swagger.ApiOkResponse({ description: `${modelName} retrieved successfully.` }),
461
+ swagger.ApiNotFoundResponse({
462
+ description: "No record matches the provided identifier.",
463
+ }),
464
+ tslib.__param(0, common.Param("id")),
465
+ tslib.__metadata("design:type", Function),
466
+ tslib.__metadata("design:paramtypes", [String]),
467
+ tslib.__metadata("design:returntype", Promise)
468
+ ], DynamicModelController.prototype, "read", null);
469
+ DynamicModelController = tslib.__decorate([
470
+ common.Controller(route),
471
+ swagger.ApiTags(modelName),
472
+ swagger.ApiExtraModels(modelClass),
473
+ tslib.__metadata("design:paramtypes", [exports.RepoFactory])
474
+ ], DynamicModelController);
475
+ return DynamicModelController;
476
+ }
477
+ static forRoot(flavour) {
478
+ const log = this.log.for(this.forRoot);
479
+ log.info(`Generating controllers for flavour...`);
480
+ const trackedModels = core.Adapter.models(flavour);
481
+ const controllers = trackedModels.map(this.toModelController.bind(this));
482
+ log.info(`Generated ${controllers.length} controllers`);
483
+ return {
484
+ module: DecafModelModule_1,
485
+ controllers,
486
+ providers: [exports.RepoFactory],
487
+ };
488
+ }
489
+ };
490
+ DecafModelModule = DecafModelModule_1 = tslib.__decorate([
491
+ common.Module({})
492
+ ], DecafModelModule);
493
+
59
494
  var DecafModule_1;
60
495
  /**
61
496
  * @publicApi
62
497
  */
63
498
  exports.DecafModule = DecafModule_1 = class DecafModule {
64
- static forRoot(options) {
499
+ static async forRootAsync(options) {
500
+ const { autoControllers } = options;
501
+ const adapter = await DecafCoreModule.createAdapter(options);
502
+ const flavour = adapter.flavour;
503
+ const imports = [DecafCoreModule.forRoot(options)];
504
+ if (autoControllers) {
505
+ imports.push(DecafModelModule.forRoot(flavour));
506
+ }
65
507
  return {
66
508
  module: DecafModule_1,
67
- imports: [DecafCoreModule.forRoot(options)],
509
+ imports: imports,
68
510
  };
69
511
  }
70
512
  };
@@ -73,7 +515,7 @@
73
515
  ], exports.DecafModule);
74
516
 
75
517
  /**
76
- * @module ts-workspace
518
+ * @module for-nest
77
519
  * @description This module serves as the main entry point for the ts-workspace library. It aggregates and exports
78
520
  * functionality from various submodules and utilities within the project.
79
521
  *
@@ -99,14 +541,25 @@
99
541
  * @constant
100
542
  * @type {string}
101
543
  */
102
- const VERSION = "0.0.1";
544
+ const VERSION = "0.0.2";
103
545
  const PACKAGE_NAME = "##PACKAGE_NAME##";
104
546
  decoration.Metadata.registerLibrary(PACKAGE_NAME, VERSION);
105
547
 
548
+ exports.ApiProperty = ApiProperty;
549
+ exports.ApiPropertyOptional = ApiPropertyOptional;
550
+ exports.ApiResponseProperty = ApiResponseProperty;
106
551
  exports.DECAF_ADAPTER_ID = DECAF_ADAPTER_ID;
107
552
  exports.DECAF_MODULE_OPTIONS = DECAF_MODULE_OPTIONS;
108
553
  exports.PACKAGE_NAME = PACKAGE_NAME;
109
554
  exports.VERSION = VERSION;
555
+ exports.createApiPropertyDecorator = createApiPropertyDecorator;
556
+ exports.createClassDecorator = createClassDecorator;
557
+ exports.createMethodDecorator = createMethodDecorator;
558
+ exports.createMixedDecorator = createMixedDecorator;
559
+ exports.createParamDecorator = createParamDecorator;
560
+ exports.createPropertyDecorator = createPropertyDecorator;
561
+ exports.getTypeIsArrayTuple = getTypeIsArrayTuple;
562
+ exports.repoForModel = repoForModel;
110
563
 
111
564
  }));
112
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9yLW5lc3QuY2pzIiwic291cmNlcyI6WyIuLi9zcmMvY29uc3RhbnRzLnRzIiwiLi4vc3JjL2NvcmUtbW9kdWxlLnRzIiwiLi4vc3JjL21vZHVsZS50cyIsIi4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgREVDQUZfTU9EVUxFX09QVElPTlMgPSBcIkRlY2FmTW9kdWxlT3B0aW9uc1wiO1xuZXhwb3J0IGNvbnN0IERFQ0FGX0FEQVBURVJfSUQgPSBcIkRlY2FmQWRhcHRlclwiO1xuIiwiaW1wb3J0IHtcbiAgRHluYW1pY01vZHVsZSxcbiAgR2xvYmFsLFxuICBJbmplY3QsXG4gIE1vZHVsZSxcbiAgT25BcHBsaWNhdGlvblNodXRkb3duLFxuICBTY29wZSxcbn0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBNb2R1bGVSZWYgfSBmcm9tIFwiQG5lc3Rqcy9jb3JlXCI7XG5pbXBvcnQgdHlwZSB7IERlY2FmTW9kdWxlT3B0aW9ucyB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQgeyBERUNBRl9BREFQVEVSX0lELCBERUNBRl9NT0RVTEVfT1BUSU9OUyB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgRmFjdG9yeVByb3ZpZGVyIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uL2ludGVyZmFjZXMvbW9kdWxlcy9wcm92aWRlci5pbnRlcmZhY2VcIjtcbmltcG9ydCB7IEFkYXB0ZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IExvZ2dlZENsYXNzIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nL2xpYi9Mb2dnZWRDbGFzc1wiO1xuXG5AR2xvYmFsKClcbkBNb2R1bGUoe30pXG5leHBvcnQgY2xhc3MgRGVjYWZDb3JlTW9kdWxlPFxuICAgIENPTkYsXG4gICAgQURBUFRFUiBleHRlbmRzIEFkYXB0ZXI8Q09ORiwgYW55LCBhbnksIGFueSwgYW55PixcbiAgPlxuICBleHRlbmRzIExvZ2dlZENsYXNzXG4gIGltcGxlbWVudHMgT25BcHBsaWNhdGlvblNodXRkb3duXG57XG4gIGNvbnN0cnVjdG9yKFxuICAgIEBJbmplY3QoREVDQUZfTU9EVUxFX09QVElPTlMpXG4gICAgcHJpdmF0ZSByZWFkb25seSBvcHRpb25zOiBEZWNhZk1vZHVsZU9wdGlvbnM8Q09ORiwgQURBUFRFUj4sXG4gICAgcHJpdmF0ZSByZWFkb25seSBtb2R1bGVSZWY6IE1vZHVsZVJlZlxuICApIHtcbiAgICBzdXBlcigpO1xuICB9XG5cbiAgc3RhdGljIGZvclJvb3Qob3B0aW9uczogRGVjYWZNb2R1bGVPcHRpb25zKTogRHluYW1pY01vZHVsZSB7XG4gICAgY29uc3QgdHlwZU9ybU1vZHVsZU9wdGlvbnMgPSB7XG4gICAgICBwcm92aWRlOiBERUNBRl9NT0RVTEVfT1BUSU9OUyxcbiAgICAgIHVzZVZhbHVlOiBvcHRpb25zLFxuICAgIH07XG5cbiAgICBjb25zdCBhZGFwdGVyOiBGYWN0b3J5UHJvdmlkZXI8YW55PiA9IHtcbiAgICAgIHVzZUZhY3Rvcnk6IGFzeW5jIChvcHRzOiBEZWNhZk1vZHVsZU9wdGlvbnMpID0+IHtcbiAgICAgICAgcmV0dXJuIG5ldyBvcHRzLmFkYXB0ZXIob3B0cy5jb25mLCBvcHRzLmFsaWFzKTtcbiAgICAgIH0sXG4gICAgICBwcm92aWRlOiBERUNBRl9BREFQVEVSX0lELFxuICAgICAgZHVyYWJsZTogdHJ1ZSxcbiAgICAgIHNjb3BlOiBTY29wZS5ERUZBVUxULFxuICAgICAgaW5qZWN0OiBbREVDQUZfTU9EVUxFX09QVElPTlNdLFxuICAgIH07XG5cbiAgICBjb25zdCBwcm92aWRlcnMgPSBbYWRhcHRlciwgdHlwZU9ybU1vZHVsZU9wdGlvbnNdO1xuICAgIGNvbnN0IGV4cG9ydHMgPSBbYWRhcHRlcl07XG5cbiAgICByZXR1cm4ge1xuICAgICAgbW9kdWxlOiBEZWNhZkNvcmVNb2R1bGUsXG4gICAgICBwcm92aWRlcnMsXG4gICAgICBleHBvcnRzLFxuICAgIH07XG4gIH1cblxuICBhc3luYyBvbkFwcGxpY2F0aW9uU2h1dGRvd24oKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgYWRhcHRlciA9IHRoaXMubW9kdWxlUmVmLmdldDxBREFQVEVSPihERUNBRl9BREFQVEVSX0lEKTtcbiAgICB0cnkge1xuICAgICAgaWYgKGFkYXB0ZXIpIHtcbiAgICAgICAgdGhpcy5sb2cuaW5mbyhcIlNodXR0aW5nIGRvd25cIik7XG4gICAgICAgIGF3YWl0IGFkYXB0ZXIuc2h1dGRvd24oKTtcbiAgICAgIH1cbiAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICB0aGlzLmxvZy5lcnJvcihgRmFpbGVkIHRvIHNodXRkb3duIGFwcGxpY2F0aW9uYCwgZSBhcyBFcnJvcik7XG4gICAgfVxuICB9XG59XG4iLCJpbXBvcnQgeyBEeW5hbWljTW9kdWxlLCBNb2R1bGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IERlY2FmTW9kdWxlT3B0aW9ucyB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQgeyBEZWNhZkNvcmVNb2R1bGUgfSBmcm9tIFwiLi9jb3JlLW1vZHVsZVwiO1xuXG4vKipcbiAqIEBwdWJsaWNBcGlcbiAqL1xuQE1vZHVsZSh7fSlcbmV4cG9ydCBjbGFzcyBEZWNhZk1vZHVsZSB7XG4gIHN0YXRpYyBmb3JSb290KG9wdGlvbnM6IERlY2FmTW9kdWxlT3B0aW9ucyk6IER5bmFtaWNNb2R1bGUge1xuICAgIHJldHVybiB7XG4gICAgICBtb2R1bGU6IERlY2FmTW9kdWxlLFxuICAgICAgaW1wb3J0czogW0RlY2FmQ29yZU1vZHVsZS5mb3JSb290KG9wdGlvbnMpXSxcbiAgICB9O1xuICB9XG59XG4iLCIvKipcbiAqIEBtb2R1bGUgdHMtd29ya3NwYWNlXG4gKiBAZGVzY3JpcHRpb24gVGhpcyBtb2R1bGUgc2VydmVzIGFzIHRoZSBtYWluIGVudHJ5IHBvaW50IGZvciB0aGUgdHMtd29ya3NwYWNlIGxpYnJhcnkuIEl0IGFnZ3JlZ2F0ZXMgYW5kIGV4cG9ydHNcbiAqIGZ1bmN0aW9uYWxpdHkgZnJvbSB2YXJpb3VzIHN1Ym1vZHVsZXMgYW5kIHV0aWxpdGllcyB3aXRoaW4gdGhlIHByb2plY3QuXG4gKlxuICogVGhlIG1vZHVsZSBpbmNsdWRlczpcbiAqIDEuIFV0aWxpdHkgZnVuY3Rpb25zIGFuZCB0eXBlcyBmcm9tIHRoZSBcIi4vdXRpbHNcIiBkaXJlY3Rvcnk6XG4gKiAgICAtIFRoZXNlIGxpa2VseSBjb250YWluIGhlbHBlciBmdW5jdGlvbnMsIGNvbW1vbiB0eXBlcywgYW5kIHNoYXJlZCBmdW5jdGlvbmFsaXR5IHVzZWQgdGhyb3VnaG91dCB0aGUgcHJvamVjdC5cbiAqICAgIC0gTWF5IGluY2x1ZGUgb3BlcmF0aW9ucyBmb3IgZGF0YSBtYW5pcHVsYXRpb24sIHR5cGUgY2hlY2tpbmcsIG9yIG90aGVyIGdlbmVyYWwtcHVycG9zZSB1dGlsaXRpZXMuXG4gKlxuICogMi4gQSBuYW1lc3BhY2UgYW5kIHJlbGF0ZWQgdHlwZXMgZnJvbSB0aGUgXCIuL25hbWVzcGFjZVwiIGRpcmVjdG9yeTpcbiAqICAgIC0gVGhpcyBjb3VsZCBjb250YWluIGRvbWFpbi1zcGVjaWZpYyBjb2RlIG9yIGEgY29sbGVjdGlvbiBvZiByZWxhdGVkIGZ1bmN0aW9uYWxpdHkuXG4gKiAgICAtIE1pZ2h0IGluY2x1ZGUgaW50ZXJmYWNlcywgdHlwZXMsIG9yIGNsYXNzZXMgdGhhdCByZXByZXNlbnQgY29yZSBjb25jZXB0cyBpbiB0aGUgbGlicmFyeS5cbiAqXG4gKiAzLiBBIFZFUlNJT04gY29uc3RhbnQ6XG4gKiAgICAtIFJlcHJlc2VudHMgdGhlIGN1cnJlbnQgdmVyc2lvbiBvZiB0aGUgbW9kdWxlLlxuICogICAgLSBVc2VmdWwgZm9yIHZlcnNpb24gY2hlY2tpbmcgYW5kIGNvbXBhdGliaWxpdHkgcHVycG9zZXMuXG4gKlxuICogVGhpcyBzdHJ1Y3R1cmUgcHJvdmlkZXMgYSBjbGVhbiBhbmQgb3JnYW5pemVkIGV4cG9ydCBvZiB0aGUgbW9kdWxlJ3MgZnVuY3Rpb25hbGl0eSwgYWxsb3dpbmcgY29uc3VtZXJzXG4gKiB0byBlYXNpbHkgaW1wb3J0IGFuZCB1c2Ugc3BlY2lmaWMgcGFydHMgb2YgdGhlIGxpYnJhcnkgYXMgbmVlZGVkLlxuICovXG5cbmltcG9ydCB7IE1ldGFkYXRhIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5cbmV4cG9ydCAqIGZyb20gXCIuL2NvbnN0YW50c1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vbW9kdWxlXCI7XG5leHBvcnQgKiBmcm9tIFwiLi90eXBlc1wiO1xuXG4vKipcbiAqIFJlcHJlc2VudHMgdGhlIGN1cnJlbnQgdmVyc2lvbiBvZiB0aGUgdHMtd29ya3NwYWNlIG1vZHVsZS5cbiAqIFRoZSBhY3R1YWwgdmVyc2lvbiBudW1iZXIgaXMgcmVwbGFjZWQgZHVyaW5nIHRoZSBidWlsZCBwcm9jZXNzLlxuICogQGNvbnN0YW50XG4gKiBAdHlwZSB7c3RyaW5nfVxuICovXG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IFwiIyNWRVJTSU9OIyNcIjtcbmV4cG9ydCBjb25zdCBQQUNLQUdFX05BTUUgPSBcIiMjUEFDS0FHRV9OQU1FIyNcIjtcblxuTWV0YWRhdGEucmVnaXN0ZXJMaWJyYXJ5KFBBQ0tBR0VfTkFNRSwgVkVSU0lPTik7XG4iXSwibmFtZXMiOlsiTG9nZ2VkQ2xhc3MiLCJTY29wZSIsIl9fZGVjb3JhdGUiLCJHbG9iYWwiLCJNb2R1bGUiLCJfX3BhcmFtIiwiSW5qZWN0IiwiTW9kdWxlUmVmIiwiRGVjYWZNb2R1bGUiLCJNZXRhZGF0YSJdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQU8sVUFBTSxvQkFBb0IsR0FBRztBQUM3QixVQUFNLGdCQUFnQixHQUFHOzs7SUNnQnpCLElBQU0sZUFBZSxHQUFBLGlCQUFBLEdBQXJCLE1BQU0sZUFJWCxTQUFRQSx1QkFBVyxDQUFBO1FBR25CLFdBRW1CLENBQUEsT0FBMEMsRUFDMUMsU0FBb0IsRUFBQTtJQUVyQyxRQUFBLEtBQUssRUFBRTtZQUhVLElBQU8sQ0FBQSxPQUFB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
565
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9yLW5lc3QuY2pzIiwic291cmNlcyI6WyIuLi9zcmMvZGVjb3JhdG9ycy9oZWxwZXJzLnRzIiwiLi4vc3JjL2RlY29yYXRvcnMvQXBpUHJvcGVydHkudHMiLCIuLi9zcmMvZGVjb3JhdGlvbi50cyIsIi4uL3NyYy9jb25zdGFudHMudHMiLCIuLi9zcmMvY29yZS1tb2R1bGUudHMiLCIuLi9zcmMvdXRpbHMudHMiLCIuLi9zcmMvUmVwb0ZhY3RvcnkudHMiLCIuLi9zcmMvbW9kZWwtbW9kdWxlLnRzIiwiLi4vc3JjL21vZHVsZS50cyIsIi4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWZ1bmN0aW9uLXR5cGUgKi9cblxuaW1wb3J0IHsgTUVUSE9EX01FVEFEQVRBIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgaXNDb25zdHJ1Y3RvciB9IGZyb20gXCJAbmVzdGpzL2NvbW1vbi91dGlscy9zaGFyZWQudXRpbHNcIjtcbmltcG9ydCB7IGlzQXJyYXksIGlzVW5kZWZpbmVkLCBuZWdhdGUsIHBpY2tCeSB9IGZyb20gXCJsb2Rhc2hcIjtcbmltcG9ydCB7IE1FVEFEQVRBX0ZBQ1RPUllfTkFNRSB9IGZyb20gXCJAbmVzdGpzL3N3YWdnZXIvZGlzdC9wbHVnaW4vcGx1Z2luLWNvbnN0YW50c1wiO1xuaW1wb3J0IHsgREVDT1JBVE9SUyB9IGZyb20gXCJAbmVzdGpzL3N3YWdnZXIvZGlzdC9jb25zdGFudHNcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZU1ldGhvZERlY29yYXRvcjxUID0gYW55PihcbiAgbWV0YWtleTogc3RyaW5nLFxuICBtZXRhZGF0YTogVCxcbiAgeyBvdmVycmlkZUV4aXN0aW5nIH0gPSB7IG92ZXJyaWRlRXhpc3Rpbmc6IHRydWUgfVxuKTogTWV0aG9kRGVjb3JhdG9yIHtcbiAgcmV0dXJuIChcbiAgICB0YXJnZXQ6IG9iamVjdCxcbiAgICBrZXk6IHN0cmluZyB8IHN5bWJvbCxcbiAgICBkZXNjcmlwdG9yOiBQcm9wZXJ0eURlc2NyaXB0b3JcbiAgKSA9PiB7XG4gICAgaWYgKHR5cGVvZiBtZXRhZGF0YSA9PT0gXCJvYmplY3RcIikge1xuICAgICAgY29uc3QgcHJldlZhbHVlID0gUmVmbGVjdC5nZXRNZXRhZGF0YShtZXRha2V5LCBkZXNjcmlwdG9yLnZhbHVlKTtcbiAgICAgIGlmIChwcmV2VmFsdWUgJiYgIW92ZXJyaWRlRXhpc3RpbmcpIHtcbiAgICAgICAgcmV0dXJuIGRlc2NyaXB0b3I7XG4gICAgICB9XG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBtZXRha2V5LFxuICAgICAgICB7IC4uLnByZXZWYWx1ZSwgLi4ubWV0YWRhdGEgfSxcbiAgICAgICAgZGVzY3JpcHRvci52YWx1ZVxuICAgICAgKTtcbiAgICAgIHJldHVybiBkZXNjcmlwdG9yO1xuICAgIH1cbiAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKG1ldGFrZXksIG1ldGFkYXRhLCBkZXNjcmlwdG9yLnZhbHVlKTtcbiAgICByZXR1cm4gZGVzY3JpcHRvcjtcbiAgfTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUNsYXNzRGVjb3JhdG9yPFQgZXh0ZW5kcyBBcnJheTxhbnk+ID0gYW55PihcbiAgbWV0YWtleTogc3RyaW5nLFxuICBtZXRhZGF0YTogVCA9IFtdIGFzIHVua25vd24gYXMgVFxuKTogQ2xhc3NEZWNvcmF0b3Ige1xuICByZXR1cm4gKHRhcmdldCkgPT4ge1xuICAgIGNvbnN0IHByZXZWYWx1ZSA9IFJlZmxlY3QuZ2V0TWV0YWRhdGEobWV0YWtleSwgdGFyZ2V0KSB8fCBbXTtcbiAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKG1ldGFrZXksIFsuLi5wcmV2VmFsdWUsIC4uLm1ldGFkYXRhXSwgdGFyZ2V0KTtcbiAgICByZXR1cm4gdGFyZ2V0O1xuICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlUHJvcGVydHlEZWNvcmF0b3I8VCBleHRlbmRzIFJlY29yZDxzdHJpbmcsIGFueT4gPSBhbnk+KFxuICBtZXRha2V5OiBzdHJpbmcsXG4gIG1ldGFkYXRhOiBULFxuICBvdmVycmlkZUV4aXN0aW5nID0gdHJ1ZVxuKTogUHJvcGVydHlEZWNvcmF0b3Ige1xuICByZXR1cm4gKHRhcmdldDogb2JqZWN0LCBwcm9wZXJ0eUtleT86IGFueSkgPT4ge1xuICAgIGNvbnN0IHByb3BlcnRpZXMgPVxuICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShERUNPUkFUT1JTLkFQSV9NT0RFTF9QUk9QRVJUSUVTX0FSUkFZLCB0YXJnZXQpIHx8IFtdO1xuXG4gICAgY29uc3Qga2V5ID0gYDoke3Byb3BlcnR5S2V5fWA7XG4gICAgaWYgKCFwcm9wZXJ0aWVzLmluY2x1ZGVzKGtleSkpIHtcbiAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoXG4gICAgICAgIERFQ09SQVRPUlMuQVBJX01PREVMX1BST1BFUlRJRVNfQVJSQVksXG4gICAgICAgIFsuLi5wcm9wZXJ0aWVzLCBgOiR7cHJvcGVydHlLZXl9YF0sXG4gICAgICAgIHRhcmdldFxuICAgICAgKTtcbiAgICB9XG4gICAgY29uc3QgZXhpc3RpbmdNZXRhZGF0YSA9IFJlZmxlY3QuZ2V0TWV0YWRhdGEobWV0YWtleSwgdGFyZ2V0LCBwcm9wZXJ0eUtleSk7XG4gICAgaWYgKGV4aXN0aW5nTWV0YWRhdGEpIHtcbiAgICAgIGNvbnN0IG5ld01ldGFkYXRhID0gcGlja0J5KG1ldGFkYXRhLCBuZWdhdGUoaXNVbmRlZmluZWQpKTtcbiAgICAgIGNvbnN0IG1ldGFkYXRhVG9TYXZlID0gb3ZlcnJpZGVFeGlzdGluZ1xuICAgICAgICA/IHtcbiAgICAgICAgICAgIC4uLmV4aXN0aW5nTWV0YWRhdGEsXG4gICAgICAgICAgICAuLi5uZXdNZXRhZGF0YSxcbiAgICAgICAgICB9XG4gICAgICAgIDoge1xuICAgICAgICAgICAgLi4ubmV3TWV0YWRhdGEsXG4gICAgICAgICAgICAuLi5leGlzdGluZ01ldGFkYXRhLFxuICAgICAgICAgIH07XG5cbiAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEobWV0YWtleSwgbWV0YWRhdGFUb1NhdmUsIHRhcmdldCwgcHJvcGVydHlLZXkpO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCB0eXBlID1cbiAgICAgICAgLy8gQHRzLWV4cGVjdC1lcnJvciBuZXN0IGpzIG92ZXJyaWRlXG4gICAgICAgIHRhcmdldD8uY29uc3RydWN0b3I/LltNRVRBREFUQV9GQUNUT1JZX05BTUVdPy4oKVtwcm9wZXJ0eUtleV0/LnR5cGUgPz9cbiAgICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShcImRlc2lnbjp0eXBlXCIsIHRhcmdldCwgcHJvcGVydHlLZXkpO1xuXG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBtZXRha2V5LFxuICAgICAgICB7XG4gICAgICAgICAgdHlwZSxcbiAgICAgICAgICAuLi5waWNrQnkobWV0YWRhdGEsIG5lZ2F0ZShpc1VuZGVmaW5lZCkpLFxuICAgICAgICB9LFxuICAgICAgICB0YXJnZXQsXG4gICAgICAgIHByb3BlcnR5S2V5XG4gICAgICApO1xuICAgIH1cbiAgfTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZU1peGVkRGVjb3JhdG9yPFQgPSBhbnk+KFxuICBtZXRha2V5OiBzdHJpbmcsXG4gIG1ldGFkYXRhOiBUXG4pOiBNZXRob2REZWNvcmF0b3IgJiBDbGFzc0RlY29yYXRvciB7XG4gIHJldHVybiAoXG4gICAgdGFyZ2V0OiBvYmplY3QsXG4gICAga2V5Pzogc3RyaW5nIHwgc3ltYm9sLFxuICAgIGRlc2NyaXB0b3I/OiBUeXBlZFByb3BlcnR5RGVzY3JpcHRvcjxhbnk+XG4gICk6IGFueSA9PiB7XG4gICAgaWYgKGRlc2NyaXB0b3IpIHtcbiAgICAgIGxldCBtZXRhZGF0YXM6IGFueTtcbiAgICAgIGlmIChBcnJheS5pc0FycmF5KG1ldGFkYXRhKSkge1xuICAgICAgICBjb25zdCBwcmV2aW91c01ldGFkYXRhID1cbiAgICAgICAgICBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIGRlc2NyaXB0b3IudmFsdWUpIHx8IFtdO1xuICAgICAgICBtZXRhZGF0YXMgPSBbLi4ucHJldmlvdXNNZXRhZGF0YSwgLi4ubWV0YWRhdGFdO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgY29uc3QgcHJldmlvdXNNZXRhZGF0YSA9XG4gICAgICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShtZXRha2V5LCBkZXNjcmlwdG9yLnZhbHVlKSB8fCB7fTtcbiAgICAgICAgbWV0YWRhdGFzID0geyAuLi5wcmV2aW91c01ldGFkYXRhLCAuLi5tZXRhZGF0YSB9O1xuICAgICAgfVxuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShtZXRha2V5LCBtZXRhZGF0YXMsIGRlc2NyaXB0b3IudmFsdWUpO1xuICAgICAgcmV0dXJuIGRlc2NyaXB0b3I7XG4gICAgfVxuXG4gICAgbGV0IG1ldGFkYXRhczogYW55O1xuICAgIGlmIChBcnJheS5pc0FycmF5KG1ldGFkYXRhKSkge1xuICAgICAgY29uc3QgcHJldmlvdXNNZXRhZGF0YSA9IFJlZmxlY3QuZ2V0TWV0YWRhdGEobWV0YWtleSwgdGFyZ2V0KSB8fCBbXTtcbiAgICAgIG1ldGFkYXRhcyA9IFsuLi5wcmV2aW91c01ldGFkYXRhLCAuLi5tZXRhZGF0YV07XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHByZXZpb3VzTWV0YWRhdGEgPSBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIHRhcmdldCkgfHwge307XG4gICAgICBtZXRhZGF0YXMgPSBPYmplY3QuYXNzaWduKE9iamVjdC5hc3NpZ24oe30sIHByZXZpb3VzTWV0YWRhdGEpLCBtZXRhZGF0YSk7XG4gICAgfVxuICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEobWV0YWtleSwgbWV0YWRhdGFzLCB0YXJnZXQpO1xuICAgIHJldHVybiB0YXJnZXQ7XG4gIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVQYXJhbURlY29yYXRvcjxUIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55PiA9IGFueT4oXG4gIG1ldGFkYXRhOiBULFxuICBpbml0aWFsOiBQYXJ0aWFsPFQ+XG4pOiBNZXRob2REZWNvcmF0b3IgJiBDbGFzc0RlY29yYXRvciB7XG4gIHJldHVybiAoXG4gICAgdGFyZ2V0OiBvYmplY3QgfCBGdW5jdGlvbixcbiAgICBrZXk/OiBzdHJpbmcgfCBzeW1ib2wsXG4gICAgZGVzY3JpcHRvcj86IFR5cGVkUHJvcGVydHlEZXNjcmlwdG9yPGFueT5cbiAgKTogYW55ID0+IHtcbiAgICBjb25zdCBwYXJhbU9wdGlvbnMgPSB7XG4gICAgICAuLi5pbml0aWFsLFxuICAgICAgLi4ucGlja0J5KG1ldGFkYXRhLCBuZWdhdGUoaXNVbmRlZmluZWQpKSxcbiAgICB9O1xuXG4gICAgaWYgKGRlc2NyaXB0b3IpIHtcbiAgICAgIGNvbnN0IHBhcmFtZXRlcnMgPVxuICAgICAgICBSZWZsZWN0LmdldE1ldGFkYXRhKERFQ09SQVRPUlMuQVBJX1BBUkFNRVRFUlMsIGRlc2NyaXB0b3IudmFsdWUpIHx8IFtdO1xuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICAgICAgREVDT1JBVE9SUy5BUElfUEFSQU1FVEVSUyxcbiAgICAgICAgWy4uLnBhcmFtZXRlcnMsIHBhcmFtT3B0aW9uc10sXG4gICAgICAgIGRlc2NyaXB0b3IudmFsdWVcbiAgICAgICk7XG4gICAgICByZXR1cm4gZGVzY3JpcHRvcjtcbiAgICB9XG5cbiAgICBpZiAodHlwZW9mIHRhcmdldCA9PT0gXCJvYmplY3RcIikge1xuICAgICAgcmV0dXJuIHRhcmdldDtcbiAgICB9XG5cbiAgICBjb25zdCBwcm9wZXJ0eUtleXMgPSBPYmplY3QuZ2V0T3duUHJvcGVydHlOYW1lcyh0YXJnZXQucHJvdG90eXBlKTtcblxuICAgIGZvciAoY29uc3QgcHJvcGVydHlLZXkgb2YgcHJvcGVydHlLZXlzKSB7XG4gICAgICBpZiAoaXNDb25zdHJ1Y3Rvcihwcm9wZXJ0eUtleSkpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IG1ldGhvZERlc2NyaXB0b3IgPSBPYmplY3QuZ2V0T3duUHJvcGVydHlEZXNjcmlwdG9yKFxuICAgICAgICB0YXJnZXQucHJvdG90eXBlLFxuICAgICAgICBwcm9wZXJ0eUtleVxuICAgICAgKTtcblxuICAgICAgaWYgKCFtZXRob2REZXNjcmlwdG9yKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBpc0FwaU1ldGhvZCA9IFJlZmxlY3QuaGFzTWV0YWRhdGEoXG4gICAgICAgIE1FVEhPRF9NRVRBREFUQSxcbiAgICAgICAgbWV0aG9kRGVzY3JpcHRvci52YWx1ZVxuICAgICAgKTtcblxuICAgICAgaWYgKCFpc0FwaU1ldGhvZCkge1xuICAgICAgICBjb250aW51ZTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgcGFyYW1ldGVycyA9XG4gICAgICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEoXG4gICAgICAgICAgREVDT1JBVE9SUy5BUElfUEFSQU1FVEVSUyxcbiAgICAgICAgICBtZXRob2REZXNjcmlwdG9yLnZhbHVlXG4gICAgICAgICkgfHwgW107XG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBERUNPUkFUT1JTLkFQSV9QQVJBTUVURVJTLFxuICAgICAgICBbLi4ucGFyYW1ldGVycywgcGFyYW1PcHRpb25zXSxcbiAgICAgICAgbWV0aG9kRGVzY3JpcHRvci52YWx1ZVxuICAgICAgKTtcbiAgICB9XG4gIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRUeXBlSXNBcnJheVR1cGxlKFxuICBpbnB1dDogRnVuY3Rpb24gfCBbRnVuY3Rpb25dIHwgdW5kZWZpbmVkIHwgc3RyaW5nIHwgUmVjb3JkPHN0cmluZywgYW55PixcbiAgaXNBcnJheUZsYWc6IGJvb2xlYW5cbik6IFtGdW5jdGlvbiB8IHVuZGVmaW5lZCwgYm9vbGVhbl0ge1xuICBpZiAoIWlucHV0KSB7XG4gICAgcmV0dXJuIFtpbnB1dCBhcyB1bmRlZmluZWQsIGlzQXJyYXlGbGFnXTtcbiAgfVxuICBpZiAoaXNBcnJheUZsYWcpIHtcbiAgICByZXR1cm4gW2lucHV0IGFzIEZ1bmN0aW9uLCBpc0FycmF5RmxhZ107XG4gIH1cbiAgY29uc3QgaXNJbnB1dEFycmF5ID0gaXNBcnJheShpbnB1dCk7XG4gIGNvbnN0IHR5cGUgPSBpc0lucHV0QXJyYXkgPyBpbnB1dFswXSA6IGlucHV0O1xuICByZXR1cm4gW3R5cGUgYXMgRnVuY3Rpb24sIGlzSW5wdXRBcnJheV07XG59XG4iLCJpbXBvcnQgeyBUeXBlIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQge1xuICBFbnVtQWxsb3dlZFR5cGVzLFxuICBTY2hlbWFPYmplY3RNZXRhZGF0YSxcbn0gZnJvbSBcIkBuZXN0anMvc3dhZ2dlci9kaXN0L2ludGVyZmFjZXMvc2NoZW1hLW9iamVjdC1tZXRhZGF0YS5pbnRlcmZhY2VcIjtcbmltcG9ydCB7IEVudW1TY2hlbWFBdHRyaWJ1dGVzIH0gZnJvbSBcIkBuZXN0anMvc3dhZ2dlci9kaXN0L2ludGVyZmFjZXMvZW51bS1zY2hlbWEtYXR0cmlidXRlcy5pbnRlcmZhY2VcIjtcbmltcG9ydCB7IGdldFR5cGVJc0FycmF5VHVwbGUgfSBmcm9tIFwiQG5lc3Rqcy9zd2FnZ2VyL2Rpc3QvZGVjb3JhdG9ycy9oZWxwZXJzXCI7XG5pbXBvcnQge1xuICBnZXRFbnVtVHlwZSxcbiAgZ2V0RW51bVZhbHVlcyxcbn0gZnJvbSBcIkBuZXN0anMvc3dhZ2dlci9kaXN0L3V0aWxzL2VudW0udXRpbHNcIjtcbmltcG9ydCB7IERFQ09SQVRPUlMgfSBmcm9tIFwiQG5lc3Rqcy9zd2FnZ2VyL2Rpc3QvY29uc3RhbnRzXCI7XG5pbXBvcnQgeyBjcmVhdGVQcm9wZXJ0eURlY29yYXRvciB9IGZyb20gXCIuL2hlbHBlcnNcIjtcblxuZXhwb3J0IHR5cGUgQXBpUHJvcGVydHlDb21tb25PcHRpb25zID0gU2NoZW1hT2JqZWN0TWV0YWRhdGEgJiB7XG4gIFwieC1lbnVtTmFtZXNcIj86IHN0cmluZ1tdO1xuICAvKipcbiAgICogTGF6eSBmdW5jdGlvbiByZXR1cm5pbmcgdGhlIHR5cGUgZm9yIHdoaWNoIHRoZSBkZWNvcmF0ZWQgcHJvcGVydHlcbiAgICogY2FuIGJlIHVzZWQgYXMgYW4gaWRcbiAgICpcbiAgICogVXNlIHRvZ2V0aGVyIHdpdGggQEFwaURlZmF1bHRHZXR0ZXIgb24gdGhlIGdldHRlciByb3V0ZSBvZiB0aGUgdHlwZVxuICAgKiB0byBnZW5lcmF0ZSBPcGVuQVBJIGxpbmsgb2JqZWN0c1xuICAgKlxuICAgKiBAc2VlIFtTd2FnZ2VyIGxpbmsgb2JqZWN0c10oaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9saW5rcy8pXG4gICAqL1xuICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1mdW5jdGlvbi10eXBlXG4gIGxpbms/OiAoKSA9PiBUeXBlPHVua25vd24+IHwgRnVuY3Rpb247XG59O1xuXG5leHBvcnQgdHlwZSBBcGlQcm9wZXJ0eU9wdGlvbnMgPVxuICB8IEFwaVByb3BlcnR5Q29tbW9uT3B0aW9uc1xuICB8IChBcGlQcm9wZXJ0eUNvbW1vbk9wdGlvbnMgJiB7XG4gICAgICBlbnVtTmFtZTogc3RyaW5nO1xuICAgICAgZW51bVNjaGVtYT86IEVudW1TY2hlbWFBdHRyaWJ1dGVzO1xuICAgIH0pO1xuXG5jb25zdCBpc0VudW1BcnJheSA9IChcbiAgb3B0czogQXBpUHJvcGVydHlPcHRpb25zXG4pOiBvcHRzIGlzIHtcbiAgaXNBcnJheTogdHJ1ZTtcbiAgZW51bTogRW51bUFsbG93ZWRUeXBlcztcbiAgdHlwZTogYW55O1xuICBpdGVtczogYW55O1xufSA9PiAob3B0cy5pc0FycmF5ICYmIFwiZW51bVwiIGluIG9wdHMgJiYgb3B0cy5lbnVtICE9PSB1bmRlZmluZWQpIGFzIGJvb2xlYW47XG5cbi8qKlxuICogQHB1YmxpY0FwaVxuICovXG5leHBvcnQgZnVuY3Rpb24gQXBpUHJvcGVydHkoXG4gIG9wdGlvbnM6IEFwaVByb3BlcnR5T3B0aW9ucyA9IHt9XG4pOiBQcm9wZXJ0eURlY29yYXRvciB7XG4gIHJldHVybiBjcmVhdGVBcGlQcm9wZXJ0eURlY29yYXRvcihvcHRpb25zKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUFwaVByb3BlcnR5RGVjb3JhdG9yKFxuICBvcHRpb25zOiBBcGlQcm9wZXJ0eU9wdGlvbnMgPSB7fSxcbiAgb3ZlcnJpZGVFeGlzdGluZyA9IHRydWVcbik6IFByb3BlcnR5RGVjb3JhdG9yIHtcbiAgY29uc3QgW3R5cGUsIGlzQXJyYXldID0gZ2V0VHlwZUlzQXJyYXlUdXBsZShcbiAgICBvcHRpb25zLnR5cGUsXG4gICAgb3B0aW9ucy5pc0FycmF5IGFzIGJvb2xlYW5cbiAgKTtcbiAgb3B0aW9ucyA9IHtcbiAgICAuLi5vcHRpb25zLFxuICAgIHR5cGUsXG4gICAgaXNBcnJheSxcbiAgfSBhcyBBcGlQcm9wZXJ0eU9wdGlvbnM7XG5cbiAgaWYgKGlzRW51bUFycmF5KG9wdGlvbnMpKSB7XG4gICAgb3B0aW9ucy50eXBlID0gXCJhcnJheVwiO1xuXG4gICAgY29uc3QgZW51bVZhbHVlcyA9IGdldEVudW1WYWx1ZXMob3B0aW9ucy5lbnVtKTtcbiAgICBvcHRpb25zLml0ZW1zID0ge1xuICAgICAgdHlwZTogZ2V0RW51bVR5cGUoZW51bVZhbHVlcyksXG4gICAgICBlbnVtOiBlbnVtVmFsdWVzLFxuICAgIH07XG4gICAgLy8gQHRzLWV4cGVjdC1lcnJvciBuZXN0IHN3YWdnZXIgb3ZlcnJpZGVcbiAgICBkZWxldGUgb3B0aW9ucy5lbnVtO1xuICB9IGVsc2UgaWYgKFwiZW51bVwiIGluIG9wdGlvbnMgJiYgb3B0aW9ucy5lbnVtICE9PSB1bmRlZmluZWQpIHtcbiAgICBjb25zdCBlbnVtVmFsdWVzID0gZ2V0RW51bVZhbHVlcyhvcHRpb25zLmVudW0pO1xuXG4gICAgb3B0aW9ucy5lbnVtID0gZW51bVZhbHVlcztcbiAgICBvcHRpb25zLnR5cGUgPSBnZXRFbnVtVHlwZShlbnVtVmFsdWVzKTtcbiAgfVxuXG4gIGlmIChBcnJheS5pc0FycmF5KG9wdGlvbnMudHlwZSkpIHtcbiAgICBvcHRpb25zLnR5cGUgPSBcImFycmF5XCI7XG4gICAgb3B0aW9ucy5pdGVtcyA9IHtcbiAgICAgIHR5cGU6IFwiYXJyYXlcIixcbiAgICAgIGl0ZW1zOiB7XG4gICAgICAgIHR5cGU6IG9wdGlvbnMudHlwZVswXSxcbiAgICAgIH0sXG4gICAgfTtcbiAgfVxuXG4gIHJldHVybiBjcmVhdGVQcm9wZXJ0eURlY29yYXRvcihcbiAgICBERUNPUkFUT1JTLkFQSV9NT0RFTF9QUk9QRVJUSUVTLFxuICAgIG9wdGlvbnMsXG4gICAgb3ZlcnJpZGVFeGlzdGluZ1xuICApO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gQXBpUHJvcGVydHlPcHRpb25hbChcbiAgb3B0aW9uczogQXBpUHJvcGVydHlPcHRpb25zID0ge31cbik6IFByb3BlcnR5RGVjb3JhdG9yIHtcbiAgcmV0dXJuIEFwaVByb3BlcnR5KHtcbiAgICAuLi5vcHRpb25zLFxuICAgIHJlcXVpcmVkOiBmYWxzZSxcbiAgfSBhcyBBcGlQcm9wZXJ0eU9wdGlvbnMpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gQXBpUmVzcG9uc2VQcm9wZXJ0eShcbiAgb3B0aW9uczogUGljazxcbiAgICBBcGlQcm9wZXJ0eU9wdGlvbnMsXG4gICAgXCJ0eXBlXCIgfCBcImV4YW1wbGVcIiB8IFwiZm9ybWF0XCIgfCBcImRlcHJlY2F0ZWRcIiB8IFwiZW51bVwiXG4gID4gPSB7fVxuKTogUHJvcGVydHlEZWNvcmF0b3Ige1xuICByZXR1cm4gQXBpUHJvcGVydHkoe1xuICAgIHJlYWRPbmx5OiB0cnVlLFxuICAgIC4uLm9wdGlvbnMsXG4gIH0gYXMgQXBpUHJvcGVydHlPcHRpb25zKTtcbn1cbiIsImltcG9ydCB7XG4gIEluamVjdGFibGVDb25maWcsXG4gIEluamVjdGFibGVzS2V5cyxcbiAgSW5qZWN0T3B0aW9ucyxcbn0gZnJvbSBcIkBkZWNhZi10cy9pbmplY3RhYmxlLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7IEluamVjdCwgSW5qZWN0YWJsZSwgU2NvcGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IENvbnN0cnVjdG9yLCBEZWNvcmF0aW9uLCBEZWNvcmF0aW9uS2V5cyB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHsgVmFsaWRhdGlvbktleXMgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgeyBBcGlQcm9wZXJ0eSB9IGZyb20gXCIuL2RlY29yYXRvcnMvQXBpUHJvcGVydHlcIjtcbmltcG9ydCB7IFBlcnNpc3RlbmNlS2V5cyB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuXG5EZWNvcmF0aW9uLmZvcihJbmplY3RhYmxlc0tleXMuSU5KRUNUQUJMRSlcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBuZXN0SW5qZWN0YWJsZShcbiAgICAgIGNhdGVnb3J5OiBzdHJpbmcgfCBDb25zdHJ1Y3RvcixcbiAgICAgIGNmZzogSW5qZWN0YWJsZUNvbmZpZ1xuICAgICkge1xuICAgICAgcmV0dXJuIEluamVjdGFibGUoe1xuICAgICAgICBzY29wZTogY2ZnLnNpbmdsZXRvbiA/IFNjb3BlLkRFRkFVTFQgOiBTY29wZS5SRVFVRVNULFxuICAgICAgICBkdXJhYmxlOiBjZmcuc2luZ2xldG9uID8gdW5kZWZpbmVkIDogdHJ1ZSxcbiAgICAgIH0pO1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihJbmplY3RhYmxlc0tleXMuSU5KRUNUKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG5lc3RJbmplY3QoXG4gICAgICBjYXRlZ29yeTogc3ltYm9sIHwgc3RyaW5nIHwgQ29uc3RydWN0b3IsXG4gICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gICAgICBjZmc6IEluamVjdE9wdGlvbnNcbiAgICApIHtcbiAgICAgIHJldHVybiBmdW5jdGlvbiBpbm5lck5lc3RJbmplY3QoXG4gICAgICAgIHRhcmdldDogb2JqZWN0LFxuICAgICAgICBwcm9wZXJ0eUtleT86IGFueSxcbiAgICAgICAgZGVzY3JpcHRvcj86IGFueVxuICAgICAgKSB7XG4gICAgICAgIHJldHVybiBJbmplY3QoY2F0ZWdvcnkgfHwgKHRhcmdldCBhcyBDb25zdHJ1Y3RvcikpKFxuICAgICAgICAgIHRhcmdldCxcbiAgICAgICAgICBwcm9wZXJ0eUtleSxcbiAgICAgICAgICBkZXNjcmlwdG9yXG4gICAgICAgICk7XG4gICAgICB9O1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5SRVFVSVJFRClcbiAgLmV4dGVuZChcbiAgICBBcGlQcm9wZXJ0eSh7XG4gICAgICByZXF1aXJlZDogdHJ1ZSxcbiAgICB9KVxuICApXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5NQVgpXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gbWF4RGVjKG1heDogbnVtYmVyKSB7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoeyBtYXhpbXVtOiBtYXggfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKFZhbGlkYXRpb25LZXlzLk1JTilcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBtaW5EZWMobWluOiBudW1iZXIpIHtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7IG1pbmltdW06IG1pbiB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuTUFYX0xFTkdUSClcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBtYXhMZW5ndGhEZWMobWF4OiBudW1iZXIpIHtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7IG1heExlbmd0aDogbWF4IH0pO1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5NSU5fTEVOR1RIKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG1pbkxlbmd0aERlYyhtaW46IG51bWJlcikge1xuICAgICAgcmV0dXJuIEFwaVByb3BlcnR5KHsgbWluTGVuZ3RoOiBtaW4gfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG4vL1xuLy8gRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuVFlQRSlcbi8vICAgLmV4dGVuZCh7XG4vLyAgICAgZGVjb3JhdG9yOiBmdW5jdGlvbiB0eXBlRGVjKHR5cGU6IChzdHJpbmcgfCAoKCkgPT4gc3RyaW5nKSlbXSB8IHN0cmluZyB8ICgoKSA9PiBzdHJpbmcpKSB7XG4vLyAgICAgICByZXR1cm4gQXBpUHJvcGVydHkoeyB0eXBlOiB0eXBlIGFzIGFueSB9KTtcbi8vICAgICB9LFxuLy8gICB9KVxuLy8gICAuYXBwbHkoKTtcbi8vXG4vLyBEZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5EQVRFKVxuLy8gICAuZXh0ZW5kKHtcbi8vICAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIGRhdGVEZWMoKSB7XG4vLyAgICAgICByZXR1cm4gQXBpUHJvcGVydHkoeyB0eXBlOiBEYXRlIH0pO1xuLy8gICAgIH0sXG4vLyAgIH0pXG4vLyAgIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5MSVNUKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIGxpc3REZWMoXG4gICAgICBjbGF6ejpcbiAgICAgICAgfCBDb25zdHJ1Y3Rvcjxhbnk+XG4gICAgICAgIHwgKCgpID0+IENvbnN0cnVjdG9yPGFueT4pXG4gICAgICAgIHwgKENvbnN0cnVjdG9yPGFueT4gfCAoKCkgPT4gQ29uc3RydWN0b3I8YW55PikpW11cbiAgICApIHtcbiAgICAgIGNvbnN0IGMgPSBBcnJheS5pc0FycmF5KGNsYXp6KSA/IGNsYXp6WzBdIDogY2xheno7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoeyB0eXBlOiBbY10gfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbi8vXG4vLyBEZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5PUFRJT04pXG4vLyAgIC5leHRlbmQoe1xuLy8gICAgIGRlY29yYXRvcjogZnVuY3Rpb24gb3B0aW9uRGVjKG9wdGlvbnM6IHN0cmluZ1tdIHwgUmVjb3JkPHN0cmluZywgYW55Pikge1xuLy8gICAgICAgY29uc3Qgb3B0cyA9IEFycmF5LmlzQXJyYXkob3B0aW9ucykgPyBvcHRpb25zIDogT2JqZWN0LnZhbHVlcyhvcHRpb25zKTtcbi8vICAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7IGVudW06IG9wdHMgfSk7XG4vLyAgICAgfSxcbi8vICAgfSlcbi8vICAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKFZhbGlkYXRpb25LZXlzLlBBVFRFUk4pXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gcGF0dGVybkRlYyhwYXQ6IFJlZ0V4cCB8IHN0cmluZykge1xuICAgICAgcmV0dXJuIEFwaVByb3BlcnR5KHtcbiAgICAgICAgcGF0dGVybjogdHlwZW9mIHBhdCA9PT0gXCJzdHJpbmdcIiA/IHBhdCA6IHBhdC5zb3VyY2UsXG4gICAgICB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoUGVyc2lzdGVuY2VLZXlzLkNPTFVNTilcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBjb2x1bW5EZWMobmFtZTogc3RyaW5nKSB7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoe1xuICAgICAgICBuYW1lOiBuYW1lLFxuICAgICAgfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKERlY29yYXRpb25LZXlzLkRFU0NSSVBUSU9OKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIGRlc2NyaXB0aW9uRGVjKGRlc2NyaXB0aW9uOiBzdHJpbmcpIHtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBkZXNjcmlwdGlvbixcbiAgICAgIH0pO1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuIiwiZXhwb3J0IGNvbnN0IERFQ0FGX01PRFVMRV9PUFRJT05TID0gXCJEZWNhZk1vZHVsZU9wdGlvbnNcIjtcbmV4cG9ydCBjb25zdCBERUNBRl9BREFQVEVSX0lEID0gXCJEZWNhZkFkYXB0ZXJcIjtcbiIsImltcG9ydCB7XG4gIER5bmFtaWNNb2R1bGUsXG4gIEdsb2JhbCxcbiAgSW5qZWN0LFxuICBNb2R1bGUsXG4gIE9uQXBwbGljYXRpb25TaHV0ZG93bixcbiAgU2NvcGUsXG59IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgTW9kdWxlUmVmIH0gZnJvbSBcIkBuZXN0anMvY29yZVwiO1xuaW1wb3J0IHR5cGUgeyBEZWNhZk1vZHVsZU9wdGlvbnMgfSBmcm9tIFwiLi90eXBlc1wiO1xuaW1wb3J0IHsgREVDQUZfQURBUFRFUl9JRCwgREVDQUZfTU9EVUxFX09QVElPTlMgfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmltcG9ydCB7IEZhY3RvcnlQcm92aWRlciB9IGZyb20gXCJAbmVzdGpzL2NvbW1vbi9pbnRlcmZhY2VzL21vZHVsZXMvcHJvdmlkZXIuaW50ZXJmYWNlXCI7XG5pbXBvcnQgeyBBZGFwdGVyIH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBMb2dnZXIsIExvZ2dpbmcgfSBmcm9tIFwiQGRlY2FmLXRzL2xvZ2dpbmdcIjtcblxuQEdsb2JhbCgpXG5ATW9kdWxlKHt9KVxuZXhwb3J0IGNsYXNzIERlY2FmQ29yZU1vZHVsZTxcbiAgQ09ORixcbiAgQURBUFRFUiBleHRlbmRzIEFkYXB0ZXI8Q09ORiwgYW55LCBhbnksIGFueSwgYW55Pixcbj4gaW1wbGVtZW50cyBPbkFwcGxpY2F0aW9uU2h1dGRvd25cbntcbiAgcHJpdmF0ZSBzdGF0aWMgX2xvZ2dlcjogTG9nZ2VyO1xuICBwcml2YXRlIHN0YXRpYyBfYWRhcHRlckluc3RhbmNlOiBBZGFwdGVyPGFueSwgYW55LCBhbnksIGFueSwgYW55PiB8IG51bGwgPVxuICAgIG51bGw7XG5cbiAgcHJvdGVjdGVkIHN0YXRpYyBnZXQgbG9nKCk6IExvZ2dlciB7XG4gICAgaWYgKCF0aGlzLl9sb2dnZXIpIHRoaXMuX2xvZ2dlciA9IExvZ2dpbmcuZm9yKERlY2FmQ29yZU1vZHVsZSk7XG4gICAgcmV0dXJuIHRoaXMuX2xvZ2dlcjtcbiAgfVxuXG4gIGNvbnN0cnVjdG9yKFxuICAgIEBJbmplY3QoREVDQUZfTU9EVUxFX09QVElPTlMpXG4gICAgcHJpdmF0ZSByZWFkb25seSBvcHRpb25zOiBEZWNhZk1vZHVsZU9wdGlvbnM8Q09ORiwgQURBUFRFUj4sXG4gICAgcHJpdmF0ZSByZWFkb25seSBtb2R1bGVSZWY6IE1vZHVsZVJlZlxuICApIHt9XG5cbiAgc3RhdGljIGFzeW5jIGNyZWF0ZUFkYXB0ZXIoXG4gICAgb3B0aW9uczogRGVjYWZNb2R1bGVPcHRpb25zXG4gICk6IFByb21pc2U8QWRhcHRlcjxhbnksIGFueSwgYW55LCBhbnksIGFueT4+IHtcbiAgICBpZiAoIXRoaXMuX2FkYXB0ZXJJbnN0YW5jZSkge1xuICAgICAgY29uc3QgbG9nID0gdGhpcy5sb2cuZm9yKHRoaXMuY3JlYXRlQWRhcHRlcik7XG4gICAgICBsb2cuaW5mbyhcIkNyZWF0aW5nIGFkYXB0ZXIgaW5zdGFuY2UuLi5cIik7XG4gICAgICB0aGlzLl9hZGFwdGVySW5zdGFuY2UgPSBuZXcgb3B0aW9ucy5hZGFwdGVyKG9wdGlvbnMuY29uZiwgb3B0aW9ucy5hbGlhcyk7XG4gICAgICBsb2cuaW5mbyhcIkFkYXB0ZXIgaW5zdGFuY2UgY3JlYXRlZCBzdWNjZXNzZnVsbHkhXCIpO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5fYWRhcHRlckluc3RhbmNlO1xuICB9XG5cbiAgc3RhdGljIGdldEFkYXB0ZXJJbnN0YW5jZSgpOiBBZGFwdGVyPGFueSwgYW55LCBhbnksIGFueSwgYW55PiB8IG51bGwge1xuICAgIHJldHVybiB0aGlzLl9hZGFwdGVySW5zdGFuY2U7XG4gIH1cblxuICBzdGF0aWMgZm9yUm9vdChvcHRpb25zOiBEZWNhZk1vZHVsZU9wdGlvbnMpOiBEeW5hbWljTW9kdWxlIHtcbiAgICBjb25zdCB0eXBlT3JtTW9kdWxlT3B0aW9ucyA9IHtcbiAgICAgIHByb3ZpZGU6IERFQ0FGX01PRFVMRV9PUFRJT05TLFxuICAgICAgdXNlVmFsdWU6IG9wdGlvbnMsXG4gICAgfTtcblxuICAgIGNvbnN0IGFkYXB0ZXI6IEZhY3RvcnlQcm92aWRlcjxhbnk+ID0ge1xuICAgICAgdXNlRmFjdG9yeTogYXN5bmMgKG9wdHM6IERlY2FmTW9kdWxlT3B0aW9ucykgPT4ge1xuICAgICAgICByZXR1cm4gRGVjYWZDb3JlTW9kdWxlLmNyZWF0ZUFkYXB0ZXIob3B0cyk7XG4gICAgICB9LFxuICAgICAgcHJvdmlkZTogREVDQUZfQURBUFRFUl9JRCxcbiAgICAgIGR1cmFibGU6IHRydWUsXG4gICAgICBzY29wZTogU2NvcGUuREVGQVVMVCxcbiAgICAgIGluamVjdDogW0RFQ0FGX01PRFVMRV9PUFRJT05TXSxcbiAgICB9O1xuXG4gICAgY29uc3QgcHJvdmlkZXJzID0gW2FkYXB0ZXIsIHR5cGVPcm1Nb2R1bGVPcHRpb25zXTtcbiAgICBjb25zdCBleHBvcnRzID0gW2FkYXB0ZXJdO1xuXG4gICAgcmV0dXJuIHtcbiAgICAgIG1vZHVsZTogRGVjYWZDb3JlTW9kdWxlLFxuICAgICAgcHJvdmlkZXJzLFxuICAgICAgZXhwb3J0cyxcbiAgICB9O1xuICB9XG5cbiAgYXN5bmMgb25BcHBsaWNhdGlvblNodXRkb3duKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IGxvZyA9IERlY2FmQ29yZU1vZHVsZS5sb2cuZm9yKHRoaXMub25BcHBsaWNhdGlvblNodXRkb3duKTtcbiAgICBjb25zdCBhZGFwdGVyID0gdGhpcy5tb2R1bGVSZWYuZ2V0PEFEQVBURVI+KERFQ0FGX0FEQVBURVJfSUQpO1xuICAgIHRyeSB7XG4gICAgICBpZiAoYWRhcHRlcikge1xuICAgICAgICBsb2cuaW5mbyhcIlNodXR0aW5nIGRvd25cIik7XG4gICAgICAgIGF3YWl0IGFkYXB0ZXIuc2h1dGRvd24oKTtcbiAgICAgICAgRGVjYWZDb3JlTW9kdWxlLl9hZGFwdGVySW5zdGFuY2UgPSBudWxsO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgIGxvZy5lcnJvcihgRmFpbGVkIHRvIHNodXRkb3duIGFwcGxpY2F0aW9uYCwgZSBhcyBFcnJvcik7XG4gICAgfVxuICB9XG59XG4iLCJpbXBvcnQge01vZGVsfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQge0ludGVybmFsRXJyb3J9IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHtSZXBvc2l0b3J5fSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIHJlcG9Gb3JNb2RlbChtb2RlbDogc3RyaW5nKSB7XG5cdGNvbnN0IG0gPSBNb2RlbC5nZXQobW9kZWwpO1xuXHRpZiAoIW0pXG5cdFx0dGhyb3cgbmV3IEludGVybmFsRXJyb3IoYEZhaWxlZCB0byBmaW5kIHJlcG9zaXRvcnkgZm9yICR7bW9kZWx9YClcblx0Y29uc3QgcmVwbyA9IFJlcG9zaXRvcnkuZm9yTW9kZWwobSk7XG5cdHJldHVybiByZXBvO1xufSIsImltcG9ydCB7IE1vZGVsIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgSW50ZXJuYWxFcnJvciB9IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgUmVwbywgUmVwb3NpdG9yeSB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgQ29uc3RydWN0b3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcblxuQEluamVjdGFibGUoKVxuZXhwb3J0IGNsYXNzIFJlcG9GYWN0b3J5IHtcbiAgZm9yPE0gZXh0ZW5kcyBNb2RlbD4obW9kZWw6IHN0cmluZyB8IENvbnN0cnVjdG9yPE0+KTogUmVwbzxNPiB7XG4gICAgbW9kZWwgPSB0eXBlb2YgbW9kZWwgPT09IFwic3RyaW5nXCIgPyAoTW9kZWwuZ2V0KG1vZGVsKSBhcyBhbnkpIDogbW9kZWw7XG4gICAgaWYgKCFtb2RlbCkge1xuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoYEZhaWxlZCB0byBmaW5kIHJlcG9zaXRvcnkgZm9yICR7bW9kZWx9YCk7XG4gICAgfVxuICAgIHJldHVybiBSZXBvc2l0b3J5LmZvck1vZGVsKG1vZGVsIGFzIENvbnN0cnVjdG9yPE0+KTtcbiAgfVxufVxuIiwiaW1wb3J0IHsgQm9keSwgQ29udHJvbGxlciwgR2V0LCBQYXJhbSwgUG9zdCB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHtcbiAgQXBpQmFkUmVxdWVzdFJlc3BvbnNlLFxuICBBcGlCb2R5LFxuICBBcGlDcmVhdGVkUmVzcG9uc2UsXG4gIEFwaUV4dHJhTW9kZWxzLFxuICBBcGlOb3RGb3VuZFJlc3BvbnNlLFxuICBBcGlPa1Jlc3BvbnNlLFxuICBBcGlPcGVyYXRpb24sXG4gIEFwaVBhcmFtLFxuICBBcGlUYWdzLFxuICBBcGlVbnByb2Nlc3NhYmxlRW50aXR5UmVzcG9uc2UsXG4gIGdldFNjaGVtYVBhdGgsXG59IGZyb20gXCJAbmVzdGpzL3N3YWdnZXJcIjtcbmltcG9ydCB7IEFkYXB0ZXIsIFJlcG9zaXRvcnkgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IE1vZGVsLCBNb2RlbENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgTG9nZ2VkQ2xhc3MsIExvZ2dlciwgTG9nZ2luZywgdG9LZWJhYkNhc2UgfSBmcm9tIFwiQGRlY2FmLXRzL2xvZ2dpbmdcIjtcbmltcG9ydCB7IHJlcG9Gb3JNb2RlbCB9IGZyb20gXCIuL3V0aWxzXCI7XG5pbXBvcnQgeyBSZXBvRmFjdG9yeSB9IGZyb20gXCIuL1JlcG9GYWN0b3J5XCI7XG5pbXBvcnQgeyBEeW5hbWljTW9kdWxlLCBNb2R1bGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcblxuQE1vZHVsZSh7fSlcbmV4cG9ydCBjbGFzcyBEZWNhZk1vZGVsTW9kdWxlIHtcbiAgcHJpdmF0ZSBzdGF0aWMgX2xvZ2dlcjogTG9nZ2VyO1xuXG4gIHByb3RlY3RlZCBzdGF0aWMgZ2V0IGxvZygpOiBMb2dnZXIge1xuICAgIGlmICghdGhpcy5fbG9nZ2VyKSB0aGlzLl9sb2dnZXIgPSBMb2dnaW5nLmZvcihEZWNhZk1vZGVsTW9kdWxlKTtcbiAgICByZXR1cm4gdGhpcy5fbG9nZ2VyO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgdG9Nb2RlbENvbnRyb2xsZXI8VCBleHRlbmRzIE1vZGVsPGFueT4+KFxuICAgIG1vZGVsQ2xhc3M6IE1vZGVsQ29uc3RydWN0b3I8YW55PlxuICApIHtcbiAgICBjb25zdCBsb2cgPSB0aGlzLmxvZy5mb3IodGhpcy50b01vZGVsQ29udHJvbGxlcik7XG4gICAgbG9nLmRlYnVnKGBDcmVhdGluZyBtb2RlbCBjb250cm9sbGVyLi4uICR7bW9kZWxDbGFzcy5uYW1lfWApO1xuICAgIGNvbnN0IG1vZGVsTmFtZSA9IHRvS2ViYWJDYXNlKFJlcG9zaXRvcnkudGFibGUobW9kZWxDbGFzcykpO1xuICAgIGNvbnN0IHJvdXRlID0gbW9kZWxOYW1lO1xuXG4gICAgQENvbnRyb2xsZXIocm91dGUpXG4gICAgQEFwaVRhZ3MobW9kZWxOYW1lKVxuICAgIEBBcGlFeHRyYU1vZGVscyhtb2RlbENsYXNzKVxuICAgIGNsYXNzIER5bmFtaWNNb2RlbENvbnRyb2xsZXIgZXh0ZW5kcyBMb2dnZWRDbGFzcyB7XG4gICAgICBwcml2YXRlIHJlYWRvbmx5IHJlcG86IGFueTsgLy89IHRoaXMucmVwb0ZhY3RvcnkuZm9yKG1vZGVsQ2xhc3MubmFtZSk7XG5cbiAgICAgIGNvbnN0cnVjdG9yKHByaXZhdGUgcmVhZG9ubHkgcmVwb0ZhY3Rvcnk6IFJlcG9GYWN0b3J5KSB7XG4gICAgICAgIHN1cGVyKCk7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgdGhpcy5yZXBvID0gdGhpcy5yZXBvRmFjdG9yeS5mb3IobW9kZWxDbGFzcy5uYW1lKTtcbiAgICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICAgIGxvZy5lcnJvcihcbiAgICAgICAgICAgIGBGYWlsZWQgdG8gZ2V0IHJlcG9zaXRvcnkgZm9yICR7bW9kZWxDbGFzcy5uYW1lfWAsXG4gICAgICAgICAgICBlIGFzIEVycm9yXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBAUG9zdCgpXG4gICAgICBAQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYENyZWF0ZSBhIG5ldyAke21vZGVsTmFtZX0uYCB9KVxuICAgICAgQEFwaUJvZHkoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYFBheWxvYWQgZm9yICR7bW9kZWxOYW1lfWAsXG4gICAgICAgIHNjaGVtYTogeyAkcmVmOiBnZXRTY2hlbWFQYXRoKG1vZGVsQ2xhc3MpIH0sXG4gICAgICB9KVxuICAgICAgQEFwaUNyZWF0ZWRSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBgJHttb2RlbE5hbWV9IGNyZWF0ZWQgc3VjY2Vzc2Z1bGx5LmAgfSlcbiAgICAgIEBBcGlCYWRSZXF1ZXN0UmVzcG9uc2UoeyBkZXNjcmlwdGlvbjogXCJQYXlsb2FkIHZhbGlkYXRpb24gZmFpbGVkLlwiIH0pXG4gICAgICBAQXBpVW5wcm9jZXNzYWJsZUVudGl0eVJlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IFwiUmVwb3NpdG9yeSByZWplY3RlZCB0aGUgcHJvdmlkZWQgcGF5bG9hZC5cIixcbiAgICAgIH0pXG4gICAgICBhc3luYyBjcmVhdGUoQEJvZHkoKSBkYXRhOiBUKTogUHJvbWlzZTxNb2RlbDxhbnk+PiB7XG4gICAgICAgIGNvbnN0IGxvZyA9IHRoaXMubG9nLmZvcih0aGlzLmNyZWF0ZSk7XG4gICAgICAgIGxvZy52ZXJib3NlKGBjcmVhdGluZyBuZXcgJHttb2RlbE5hbWV9YCk7XG4gICAgICAgIGNvbnN0IHIgPSByZXBvRm9yTW9kZWwoXCJBY2NvdW50XCIpO1xuICAgICAgICBjb25zdCBjcmVhdGVkID0gYXdhaXQgci5jcmVhdGUoZGF0YSk7XG4gICAgICAgIGxvZy5pbmZvKGBjcmVhdGVkIG5ldyAke21vZGVsTmFtZX0gd2l0aCBpZCAke2NyZWF0ZWRbci5wa119YCk7XG4gICAgICAgIHJldHVybiBjcmVhdGVkO1xuICAgICAgfVxuXG4gICAgICBAR2V0KFwiOmlkXCIpXG4gICAgICBAQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYFJldHJpZXZlIGEgJHttb2RlbE5hbWV9IGJ5IGlkLmAgfSlcbiAgICAgIEBBcGlQYXJhbSh7XG4gICAgICAgIG5hbWU6IFwiaWRcIixcbiAgICAgICAgZGVzY3JpcHRpb246IFwiUHJpbWFyeSBrZXlcIixcbiAgICAgICAgZXhhbXBsZTogXCIxMjM0LTU2NzhcIixcbiAgICAgIH0pXG4gICAgICBAQXBpT2tSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBgJHttb2RlbE5hbWV9IHJldHJpZXZlZCBzdWNjZXNzZnVsbHkuYCB9KVxuICAgICAgQEFwaU5vdEZvdW5kUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogXCJObyByZWNvcmQgbWF0Y2hlcyB0aGUgcHJvdmlkZWQgaWRlbnRpZmllci5cIixcbiAgICAgIH0pXG4gICAgICBhc3luYyByZWFkKEBQYXJhbShcImlkXCIpIGlkOiBzdHJpbmcpIHtcbiAgICAgICAgY29uc3QgbG9nID0gdGhpcy5sb2cuZm9yKHRoaXMucmVhZCk7XG4gICAgICAgIGxvZy5kZWJ1ZyhgcmVhZGluZyAke21vZGVsTmFtZX0gd2l0aCAke3RoaXMucmVwby5wayBhcyBzdHJpbmd9ICR7aWR9YCk7XG4gICAgICAgIGNvbnN0IHJlYWQgPSBhd2FpdCB0aGlzLnJlcG8ucmVhZChpZCk7XG4gICAgICAgIGxvZy5pbmZvKGByZWFkICR7bW9kZWxOYW1lfSB3aXRoIGlkICR7cmVhZFt0aGlzLnJlcG8ucGtdfWApO1xuICAgICAgICByZXR1cm4gcmVhZDtcbiAgICAgIH1cblxuICAgICAgLy8gQFBvc3QoKVxuICAgICAgLy8gQEFwaU9wZXJhdGlvbih7c3VtbWFyeTogXCJDcmVhdGUgYSBuZXcgcmVjb3JkIGZvciB0aGUgZ2l2ZW4gbW9kZWwuXCJ9KVxuICAgICAgLy8gQEFwaUNyZWF0ZWRSZXNwb25zZSh7ZGVzY3JpcHRpb246IFwiUmVjb3JkIGNyZWF0ZWQgc3VjY2Vzc2Z1bGx5LlwifSlcbiAgICAgIC8vIEBBcGlCYWRSZXF1ZXN0UmVzcG9uc2Uoe2Rlc2NyaXB0aW9uOiBcIlBheWxvYWQgdmFsaWRhdGlvbiBmYWlsZWQuXCJ9KVxuICAgICAgLy8gQEFwaVVucHJvY2Vzc2FibGVFbnRpdHlSZXNwb25zZSh7ZGVzY3JpcHRpb246IFwiUmVwb3NpdG9yeSByZWplY3RlZCB0aGUgcHJvdmlkZWQgcGF5bG9hZC5cIn0pXG4gICAgICAvLyBhc3luYyBjcmVhdGUoQFBhcmFtKFwibW9kZWxcIikgbW9kZWw6IHN0cmluZywgQEJvZHkoKSBkYXRhOiBhbnkpIHtcbiAgICAgIC8vICAgICBjb25zdCBsb2cgPSB0aGlzLmxvZy5mb3IodGhpcy5jcmVhdGUpO1xuICAgICAgLy8gICAgIGxvZy52ZXJib3NlKGBjcmVhdGluZyBuZXcgJHttb2RlbH1gKTtcbiAgICAgIC8vICAgICBsZXQgcmVwbzogUmVwbzxNb2RlbD47XG4gICAgICAvLyAgICAgbGV0IGNyZWF0ZWQ6IE1vZGVsO1xuICAgICAgLy8gICAgIHRyeSB7XG4gICAgICAvLyAgICAgICAgIHJlcG8gPSByZXBvRm9yTW9kZWwobW9kZWwpO1xuICAgICAgLy8gICAgICAgICBjcmVhdGVkID0gYXdhaXQgcmVwby5jcmVhdGUoZGF0YSk7XG4gICAgICAvLyAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgLy8gICAgICAgICBsb2cuZXJyb3IoYEZhaWxlZCB0byBjcmVhdGUgbmV3ICR7bW9kZWx9YCwgZSBhcyBFcnJvcik7XG4gICAgICAvLyAgICAgICAgIHRocm93IGU7XG4gICAgICAvLyAgICAgfVxuICAgICAgLy8gICAgIGxvZy5pbmZvKGBjcmVhdGVkIG5ldyAke21vZGVsfSB3aXRoIGlkICR7Y3JlYXRlZFtyZXBvLnBrXX1gKTtcbiAgICAgIC8vICAgICByZXR1cm4gY3JlYXRlZDtcbiAgICAgIC8vIH1cblxuICAgICAgLy8gQEdldChcIjppZFwiKVxuICAgICAgLy8gQEFwaU9wZXJhdGlvbih7c3VtbWFyeTogXCJSZXRyaWV2ZSBhIHNpbmdsZSByZWNvcmQgYnkgaWQuXCJ9KVxuICAgICAgLy8gQEFwaVBhcmFtKHtcbiAgICAgIC8vICAgICBuYW1lOiBcIm1vZGVsXCIsXG4gICAgICAvLyAgICAgLy8gZGVzY3JpcHRpb246ICdOYW1lIG9mIHRoZSBtb2RlbCByZXBvc2l0b3J5IHRvIHRhcmdldC4nICsgYFxcbiR7bW9kZWxMaXN0Lm1hcCgobTogc3RyaW5nLCBpOiBudW1iZXIpID0+IGAke219IC0gJHtNZXRhZGF0YS5kZXNjcmlwdGlvbih0cmFja2VkTW9kZWxzW2ldKX1gKS5qb2luKCdcXG4nKX1gLFxuICAgICAgLy8gICAgIGV4YW1wbGU6IFwiYWdlbnRcIixcbiAgICAgIC8vIH0pXG4gICAgICAvLyBAQXBpUGFyYW0oe1xuICAgICAgLy8gICAgIG5hbWU6IFwiaWRcIixcbiAgICAgIC8vICAgICBkZXNjcmlwdGlvbjogXCJQcmltYXJ5IGtleSB2YWx1ZSB1c2VkIHRvIGxvYWQgdGhlIHJlY29yZC5cIixcbiAgICAgIC8vICAgICBleGFtcGxlOiBcIjEyMzQtNTY3OFwiLFxuICAgICAgLy8gfSlcbiAgICAgIC8vIEBBcGlPa1Jlc3BvbnNlKHtkZXNjcmlwdGlvbjogXCJSZWNvcmQgcmV0cmlldmVkIHN1Y2Nlc3NmdWxseS5cIn0pXG4gICAgICAvLyBAQXBpTm90Rm91bmRSZXNwb25zZSh7ZGVzY3JpcHRpb246IFwiTm8gcmVjb3JkIG1hdGNoZXMgdGhlIHByb3ZpZGVkIGlkZW50aWZpZXIuXCJ9KVxuICAgICAgLy8gYXN5bmMgcmVhZChAUGFyYW0oXCJtb2RlbFwiKSBtb2RlbDogc3RyaW5nLCBAUGFyYW0oXCJpZFwiKSBpZDogc3RyaW5nKSB7XG4gICAgICAvLyAgICAgY29uc3QgbG9nID0gdGhpcy5sb2cuZm9yKHRoaXMucmVhZCk7XG4gICAgICAvLyAgICAgbGV0IHJlcG86IFJlcG88TW9kZWw+O1xuICAgICAgLy8gICAgIGxldCByZWFkOiBNb2RlbDtcbiAgICAgIC8vICAgICB0cnkge1xuICAgICAgLy8gICAgICAgICByZXBvID0gcmVwb0Zvck1vZGVsKG1vZGVsKTtcbiAgICAgIC8vICAgICAgICAgbG9nLmRlYnVnKGByZWFkaW5nICR7bW9kZWx9IHdpdGggJHtyZXBvLnBrIGFzIHN0cmluZ30gJHtpZH1gKTtcbiAgICAgIC8vICAgICAgICAgcmVhZCA9IGF3YWl0IHJlcG8ucmVhZChpZCk7XG4gICAgICAvLyAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgLy8gICAgICAgICBsb2cuZXJyb3IoYEZhaWxlZCB0byByZWFkICR7bW9kZWx9IHdpdGggaWQgJHtpZH1gLCBlIGFzIEVycm9yKTtcbiAgICAgIC8vICAgICAgICAgdGhyb3cgZTtcbiAgICAgIC8vICAgICB9XG4gICAgICAvLyAgICAgbG9nLmluZm8oYHJlYWQgJHttb2RlbH0gd2l0aCBpZCAke3JlYWRbcmVwby5wa119YCk7XG4gICAgICAvLyAgICAgcmV0dXJuIHJlYWQ7XG4gICAgICAvLyB9XG4gICAgfVxuXG4gICAgcmV0dXJuIER5bmFtaWNNb2RlbENvbnRyb2xsZXI7XG4gIH1cblxuICBzdGF0aWMgZm9yUm9vdChmbGF2b3VyOiBzdHJpbmcpOiBEeW5hbWljTW9kdWxlIHtcbiAgICBjb25zdCBsb2cgPSB0aGlzLmxvZy5mb3IodGhpcy5mb3JSb290KTtcbiAgICBsb2cuaW5mbyhgR2VuZXJhdGluZyBjb250cm9sbGVycyBmb3IgZmxhdm91ci4uLmApO1xuXG4gICAgY29uc3QgdHJhY2tlZE1vZGVscyA9IEFkYXB0ZXIubW9kZWxzKGZsYXZvdXIpO1xuICAgIGNvbnN0IGNvbnRyb2xsZXJzID0gdHJhY2tlZE1vZGVscy5tYXAodGhpcy50b01vZGVsQ29udHJvbGxlci5iaW5kKHRoaXMpKTtcblxuICAgIGxvZy5pbmZvKGBHZW5lcmF0ZWQgJHtjb250cm9sbGVycy5sZW5ndGh9IGNvbnRyb2xsZXJzYCk7XG5cbiAgICByZXR1cm4ge1xuICAgICAgbW9kdWxlOiBEZWNhZk1vZGVsTW9kdWxlLFxuICAgICAgY29udHJvbGxlcnMsXG4gICAgICBwcm92aWRlcnM6IFtSZXBvRmFjdG9yeV0sXG4gICAgfTtcbiAgfVxufVxuIiwiaW1wb3J0IHsgRHluYW1pY01vZHVsZSwgRm9yd2FyZFJlZmVyZW5jZSwgTW9kdWxlLCBUeXBlIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBEZWNhZk1vZHVsZU9wdGlvbnMgfSBmcm9tIFwiLi90eXBlc1wiO1xuaW1wb3J0IHsgRGVjYWZDb3JlTW9kdWxlIH0gZnJvbSBcIi4vY29yZS1tb2R1bGVcIjtcbmltcG9ydCB7IERlY2FmTW9kZWxNb2R1bGUgfSBmcm9tIFwiLi9tb2RlbC1tb2R1bGVcIjtcblxuLyoqXG4gKiBAcHVibGljQXBpXG4gKi9cbkBNb2R1bGUoe30pXG5leHBvcnQgY2xhc3MgRGVjYWZNb2R1bGUge1xuICBzdGF0aWMgYXN5bmMgZm9yUm9vdEFzeW5jKFxuICAgIG9wdGlvbnM6IERlY2FmTW9kdWxlT3B0aW9uc1xuICApOiBQcm9taXNlPER5bmFtaWNNb2R1bGU+IHtcbiAgICBjb25zdCB7IGF1dG9Db250cm9sbGVycyB9ID0gb3B0aW9ucztcblxuICAgIGNvbnN0IGFkYXB0ZXIgPSBhd2FpdCBEZWNhZkNvcmVNb2R1bGUuY3JlYXRlQWRhcHRlcihvcHRpb25zKTtcbiAgICBjb25zdCBmbGF2b3VyID0gYWRhcHRlci5mbGF2b3VyO1xuXG4gICAgY29uc3QgaW1wb3J0czpcbiAgICAgIHwgKFxuICAgICAgICAgIHwgRHluYW1pY01vZHVsZVxuICAgICAgICAgIHwgVHlwZTxhbnk+XG4gICAgICAgICAgfCBQcm9taXNlPER5bmFtaWNNb2R1bGU+XG4gICAgICAgICAgfCBGb3J3YXJkUmVmZXJlbmNlPGFueT5cbiAgICAgICAgKVtdXG4gICAgICB8IHVuZGVmaW5lZCA9IFtEZWNhZkNvcmVNb2R1bGUuZm9yUm9vdChvcHRpb25zKV07XG5cbiAgICBpZiAoYXV0b0NvbnRyb2xsZXJzKSB7XG4gICAgICBpbXBvcnRzLnB1c2goRGVjYWZNb2RlbE1vZHVsZS5mb3JSb290KGZsYXZvdXIpKTtcbiAgICB9XG5cbiAgICByZXR1cm4ge1xuICAgICAgbW9kdWxlOiBEZWNhZk1vZHVsZSxcbiAgICAgIGltcG9ydHM6IGltcG9ydHMsXG4gICAgfTtcbiAgfVxufVxuIiwiLyoqXG4gKiBAbW9kdWxlIGZvci1uZXN0XG4gKiBAZGVzY3JpcHRpb24gVGhpcyBtb2R1bGUgc2VydmVzIGFzIHRoZSBtYWluIGVudHJ5IHBvaW50IGZvciB0aGUgdHMtd29ya3NwYWNlIGxpYnJhcnkuIEl0IGFnZ3JlZ2F0ZXMgYW5kIGV4cG9ydHNcbiAqIGZ1bmN0aW9uYWxpdHkgZnJvbSB2YXJpb3VzIHN1Ym1vZHVsZXMgYW5kIHV0aWxpdGllcyB3aXRoaW4gdGhlIHByb2plY3QuXG4gKlxuICogVGhlIG1vZHVsZSBpbmNsdWRlczpcbiAqIDEuIFV0aWxpdHkgZnVuY3Rpb25zIGFuZCB0eXBlcyBmcm9tIHRoZSBcIi4vdXRpbHNcIiBkaXJlY3Rvcnk6XG4gKiAgICAtIFRoZXNlIGxpa2VseSBjb250YWluIGhlbHBlciBmdW5jdGlvbnMsIGNvbW1vbiB0eXBlcywgYW5kIHNoYXJlZCBmdW5jdGlvbmFsaXR5IHVzZWQgdGhyb3VnaG91dCB0aGUgcHJvamVjdC5cbiAqICAgIC0gTWF5IGluY2x1ZGUgb3BlcmF0aW9ucyBmb3IgZGF0YSBtYW5pcHVsYXRpb24sIHR5cGUgY2hlY2tpbmcsIG9yIG90aGVyIGdlbmVyYWwtcHVycG9zZSB1dGlsaXRpZXMuXG4gKlxuICogMi4gQSBuYW1lc3BhY2UgYW5kIHJlbGF0ZWQgdHlwZXMgZnJvbSB0aGUgXCIuL25hbWVzcGFjZVwiIGRpcmVjdG9yeTpcbiAqICAgIC0gVGhpcyBjb3VsZCBjb250YWluIGRvbWFpbi1zcGVjaWZpYyBjb2RlIG9yIGEgY29sbGVjdGlvbiBvZiByZWxhdGVkIGZ1bmN0aW9uYWxpdHkuXG4gKiAgICAtIE1pZ2h0IGluY2x1ZGUgaW50ZXJmYWNlcywgdHlwZXMsIG9yIGNsYXNzZXMgdGhhdCByZXByZXNlbnQgY29yZSBjb25jZXB0cyBpbiB0aGUgbGlicmFyeS5cbiAqXG4gKiAzLiBBIFZFUlNJT04gY29uc3RhbnQ6XG4gKiAgICAtIFJlcHJlc2VudHMgdGhlIGN1cnJlbnQgdmVyc2lvbiBvZiB0aGUgbW9kdWxlLlxuICogICAgLSBVc2VmdWwgZm9yIHZlcnNpb24gY2hlY2tpbmcgYW5kIGNvbXBhdGliaWxpdHkgcHVycG9zZXMuXG4gKlxuICogVGhpcyBzdHJ1Y3R1cmUgcHJvdmlkZXMgYSBjbGVhbiBhbmQgb3JnYW5pemVkIGV4cG9ydCBvZiB0aGUgbW9kdWxlJ3MgZnVuY3Rpb25hbGl0eSwgYWxsb3dpbmcgY29uc3VtZXJzXG4gKiB0byBlYXNpbHkgaW1wb3J0IGFuZCB1c2Ugc3BlY2lmaWMgcGFydHMgb2YgdGhlIGxpYnJhcnkgYXMgbmVlZGVkLlxuICovXG5cbmltcG9ydCB7IE1ldGFkYXRhIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgXCIuL2RlY29yYXRpb25cIjtcblxuZXhwb3J0ICogZnJvbSBcIi4vZGVjb3JhdG9yc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kZWNvcmF0aW9uXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9tb2R1bGVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL1JlcG9GYWN0b3J5XCI7XG5leHBvcnQgKiBmcm9tIFwiLi90eXBlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHNcIjtcblxuLyoqXG4gKiBSZXByZXNlbnRzIHRoZSBjdXJyZW50IHZlcnNpb24gb2YgdGhlIHRzLXdvcmtzcGFjZSBtb2R1bGUuXG4gKiBUaGUgYWN0dWFsIHZlcnNpb24gbnVtYmVyIGlzIHJlcGxhY2VkIGR1cmluZyB0aGUgYnVpbGQgcHJvY2Vzcy5cbiAqIEBjb25zdGFudFxuICogQHR5cGUge3N0cmluZ31cbiAqL1xuZXhwb3J0IGNvbnN0IFZFUlNJT04gPSBcIiMjVkVSU0lPTiMjXCI7XG5leHBvcnQgY29uc3QgUEFDS0FHRV9OQU1FID0gXCIjI1BBQ0tBR0VfTkFNRSMjXCI7XG5cbk1ldGFkYXRhLnJlZ2lzdGVyTGlicmFyeShQQUNLQUdFX05BTUUsIFZFUlNJT04pO1xuIl0sIm5hbWVzIjpbIkRFQ09SQVRPUlMiLCJwaWNrQnkiLCJuZWdhdGUiLCJpc1VuZGVmaW5lZCIsIk1FVEFEQVRBX0ZBQ1RPUllfTkFNRSIsImlzQ29uc3RydWN0b3IiLCJNRVRIT0RfTUVUQURBVEEiLCJpc0FycmF5IiwiZ2V0VHlwZUlzQXJyYXlUdXBsZSIsImdldEVudW1WYWx1ZXMiLCJnZXRFbnVtVHlwZSIsIkRlY29yYXRpb24iLCJJbmplY3RhYmxlc0tleXMiLCJJbmplY3RhYmxlIiwiU2NvcGUiLCJJbmplY3QiLCJWYWxpZGF0aW9uS2V5cyIsIlBlcnNpc3RlbmNlS2V5cyIsIkRlY29yYXRpb25LZXlzIiwiTG9nZ2luZyIsIl9fZGVjb3JhdGUiLCJHbG9iYWwiLCJNb2R1bGUiLCJfX3BhcmFtIiwiTW9kdWxlUmVmIiwiTW9kZWwiLCJJbnRlcm5hbEVycm9yIiwiUmVwb3NpdG9yeSIsIlJlcG9GYWN0b3J5IiwidG9LZWJhYkNhc2UiLCJMb2dnZWRDbGFzcyIsIlBvc3QiLCJBcGlPcGVyYXRpb24iLCJBcGlCb2R5IiwiZ2V0U2NoZW1hUGF0aCIsIkFwaUNyZWF0ZWRSZXNwb25zZSIsIkFwaUJhZFJlcXVlc3RSZXNwb25zZSIsIkFwaVVucHJvY2Vzc2FibGVFbnRpdHlSZXNwb25zZSIsIkJvZHkiLCJHZXQiLCJBcGlQYXJhbSIsIkFwaU9rUmVzcG9uc2UiLCJBcGlOb3RGb3VuZFJlc3BvbnNlIiwiUGFyYW0iLCJDb250cm9sbGVyIiwiQXBpVGFncyIsIkFwaUV4dHJhTW9kZWxzIiwiQWRhcHRlciIsIkRlY2FmTW9kdWxlIiwiTWV0YWRhdGEiXSwibWFwcGluZ3MiOiI7Ozs7OztJQUFBO0lBUWdCLFNBQUEscUJBQXFCLENBQ25DLE9BQWUsRUFDZixRQUFXLEVBQ1gsRUFBRSxnQkFBZ0IsRUFBRSxHQUFHLEVBQUUsZ0JBQWdCLEVBQUUsSUFBSSxFQUFFLEVBQUE7SUFFakQsSUFBQSxPQUFPLENBQ0wsTUFBYyxFQUNkLEdBQW9CLEVBQ3BCLFVBQThCLEtBQzVCO0lBQ0YsUUFBQSxJQUFJLE9BQU8sUUFBUSxLQUFLLFFBQVEsRUFBRTtJQUNoQyxZQUFBLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxXQUFXLENBQUMsT0FBTyxFQUFFLFVBQVUsQ0FBQyxLQUFLLENBQUM7SUFDaEUsWUFBQSxJQUFJLFNBQVMsSUFBSSxDQUFDLGdCQUFnQixFQUFFO0lBQ2xDLGdCQUFBLE9BQU8sVUFBVTs7SUFFbkIsWUFBQSxPQUFPLENBQUMsY0FBYyxDQUNwQixPQUFPLEVBQ1AsRUFBRSxHQUFHLFNBQVMsRUFBRSxHQUFHLFFBQVEsRUFBRSxFQUM3QixVQUFVLENBQUMsS0FBSyxDQUNq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