@decaf-ts/for-angular 0.0.16 → 0.0.17

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 (61) hide show
  1. package/assets/i18n/en.json +10 -0
  2. package/assets/i18n/pt.json +149 -0
  3. package/assets/icons/icon-128.webp +0 -0
  4. package/assets/icons/icon-192.webp +0 -0
  5. package/assets/icons/icon-256.webp +0 -0
  6. package/assets/icons/icon-48.webp +0 -0
  7. package/assets/icons/icon-512.webp +0 -0
  8. package/assets/icons/icon-72.webp +0 -0
  9. package/assets/icons/icon-96.webp +0 -0
  10. package/assets/images/apple-touch-icon.png +0 -0
  11. package/assets/images/favicon.png +0 -0
  12. package/assets/images/favicon.svg +29 -0
  13. package/components/component-renderer/component-renderer.component.d.ts +5 -4
  14. package/components/crud-field/crud-field.component.d.ts +174 -19
  15. package/components/crud-form/crud-form.component.d.ts +170 -6
  16. package/components/fieldset/fieldset.component.d.ts +374 -36
  17. package/components/list/list.component.d.ts +1 -1
  18. package/components/list-item/list-item.component.d.ts +2 -2
  19. package/components/model-renderer/model-renderer.component.d.ts +1 -5
  20. package/directives/collapsable.directive.d.ts +1 -0
  21. package/engine/NgxBaseComponent.d.ts +6 -6
  22. package/engine/NgxCrudFormField.d.ts +5 -2
  23. package/engine/NgxFormService.d.ts +113 -12
  24. package/engine/NgxRenderingEngine.d.ts +150 -25
  25. package/engine/constants.d.ts +11 -6
  26. package/engine/decorators.d.ts +2 -2
  27. package/engine/index.d.ts +4 -2
  28. package/engine/interfaces.d.ts +261 -0
  29. package/engine/types.d.ts +3 -206
  30. package/esm2022/components/component-renderer/component-renderer.component.mjs +13 -11
  31. package/esm2022/components/crud-field/crud-field.component.mjs +193 -8
  32. package/esm2022/components/crud-form/crud-form.component.mjs +116 -11
  33. package/esm2022/components/empty-state/empty-state.component.mjs +3 -3
  34. package/esm2022/components/fieldset/fieldset.component.mjs +482 -43
  35. package/esm2022/components/filter/filter.component.mjs +3 -3
  36. package/esm2022/components/layout/layout.component.mjs +3 -3
  37. package/esm2022/components/list/list.component.mjs +4 -5
  38. package/esm2022/components/list-item/list-item.component.mjs +9 -9
  39. package/esm2022/components/model-renderer/model-renderer.component.mjs +9 -8
  40. package/esm2022/components/pagination/pagination.component.mjs +4 -4
  41. package/esm2022/components/searchbar/searchbar.component.mjs +3 -3
  42. package/esm2022/directives/collapsable.directive.mjs +3 -2
  43. package/esm2022/engine/NgxBaseComponent.mjs +28 -22
  44. package/esm2022/engine/NgxCrudFormField.mjs +14 -4
  45. package/esm2022/engine/NgxFormService.mjs +239 -27
  46. package/esm2022/engine/NgxRenderingEngine.mjs +202 -46
  47. package/esm2022/engine/ValidatorFactory.mjs +6 -4
  48. package/esm2022/engine/constants.mjs +14 -9
  49. package/esm2022/engine/decorators.mjs +6 -6
  50. package/esm2022/engine/index.mjs +5 -3
  51. package/esm2022/engine/interfaces.mjs +4 -0
  52. package/esm2022/engine/types.mjs +1 -3
  53. package/esm2022/helpers/utils.mjs +6 -2
  54. package/fesm2022/decaf-ts-for-angular.mjs +2909 -2130
  55. package/fesm2022/decaf-ts-for-angular.mjs.map +1 -1
  56. package/helpers/utils.d.ts +1 -0
  57. package/package.json +2 -1
  58. package/engine/NgxRenderingEngine2.d.ts +0 -250
  59. package/esm2022/engine/NgxRenderingEngine2.mjs +0 -332
  60. package/esm2022/interfaces.mjs +0 -2
  61. package/interfaces.d.ts +0 -28
@@ -1,11 +1,20 @@
1
- import { __decorate } from "tslib";
2
- import { ChangeDetectorRef, Component, CUSTOM_ELEMENTS_SCHEMA, ElementRef, inject, Input, ViewChild } from '@angular/core';
3
- import { Dynamic } from '../../engine';
1
+ import { __decorate, __metadata } from "tslib";
2
+ import { ChangeDetectorRef, Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, ViewChild, Renderer2 } from '@angular/core';
3
+ import { Dynamic, EventConstants } from '../../engine';
4
4
  import { OperationKeys } from '@decaf-ts/db-decorators';
5
5
  import { ForAngularModule } from '../../for-angular.module';
6
6
  import { CollapsableDirective } from '../../directives/collapsable.directive';
7
- import { IonAccordion, IonAccordionGroup, IonItem } from '@ionic/angular/standalone';
7
+ import { IonAccordion, IonAccordionGroup, IonButton, IonItem, IonLabel, IonList, IonReorderGroup, IonReorder } from '@ionic/angular/standalone';
8
+ import { cleanSpaces, generateRandomValue, itemMapper, windowEventEmitter } from '../../helpers';
9
+ import { FormArray } from '@angular/forms';
10
+ import { NgxBaseComponent } from '../../engine';
11
+ import { alertCircleOutline, createOutline } from 'ionicons/icons';
12
+ import { TranslateService } from '@ngx-translate/core';
8
13
  import * as i0 from "@angular/core";
14
+ import * as i1 from "@ionic/angular/standalone";
15
+ import * as i2 from "@angular/common";
16
+ import * as i3 from "@angular/forms";
17
+ import * as i4 from "@ngx-translate/core";
9
18
  /**
10
19
  * @description Dynamic fieldset component with collapsible accordion functionality.
11
20
  * @summary This component provides a sophisticated fieldset container that automatically
@@ -56,8 +65,17 @@ import * as i0 from "@angular/core";
56
65
  *
57
66
  * @memberOf ForAngularModule
58
67
  */
59
- let FieldsetComponent = class FieldsetComponent {
68
+ let FieldsetComponent = class FieldsetComponent extends NgxBaseComponent {
69
+ /**
70
+ * @description Component constructor that initializes the fieldset with icons and component name.
71
+ * @summary Calls the parent NgxBaseComponent constructor with the component name and
72
+ * required Ionic icons (alertCircleOutline for validation errors and createOutline for add actions).
73
+ * Sets up the foundational component structure and icon registry.
74
+ *
75
+ * @memberOf FieldsetComponent
76
+ */
60
77
  constructor() {
78
+ super('FieldsetComponent', { alertCircleOutline, createOutline });
61
79
  /**
62
80
  * @description The display name or title of the fieldset section.
63
81
  * @summary Sets the legend or header text that appears in the accordion header. This text
@@ -70,6 +88,28 @@ let FieldsetComponent = class FieldsetComponent {
70
88
  * @memberOf FieldsetComponent
71
89
  */
72
90
  this.name = 'Child';
91
+ /**
92
+ * @description The parent component identifier for hierarchical fieldset relationships.
93
+ * @summary Specifies the parent component name that this fieldset belongs to in a hierarchical
94
+ * form structure. This property is used for event bubbling and establishing parent-child
95
+ * relationships between fieldsets in complex forms with nested structures.
96
+ *
97
+ * @type {string}
98
+ * @default 'Child'
99
+ * @memberOf FieldsetComponent
100
+ */
101
+ this.childOf = 'Child';
102
+ /**
103
+ * @description The parent component identifier for hierarchical fieldset relationships.
104
+ * @summary Specifies the parent component name that this fieldset belongs to in a hierarchical
105
+ * form structure. This property is used for event bubbling and establishing parent-child
106
+ * relationships between fieldsets in complex forms with nested structures.
107
+ *
108
+ * @type {string}
109
+ * @default 'Child'
110
+ * @memberOf FieldsetComponent
111
+ */
112
+ this.uid = generateRandomValue(12);
73
113
  /**
74
114
  * @description The current CRUD operation context.
75
115
  * @summary Determines the component's initial behavior and state based on the current operation.
@@ -81,6 +121,17 @@ let FieldsetComponent = class FieldsetComponent {
81
121
  * @default OperationKeys.READ
82
122
  * @memberOf FieldsetComponent
83
123
  */
124
+ /**
125
+ * @description The CRUD operation type for the current fieldset context.
126
+ * @summary Determines the component's initial behavior and state based on the current operation.
127
+ * This input is crucial for auto-state management: READ and DELETE operations automatically
128
+ * open the fieldset to show content, while CREATE and UPDATE operations keep it closed
129
+ * initially. This provides an intuitive user experience aligned with operation semantics.
130
+ *
131
+ * @type {OperationKeys}
132
+ * @default OperationKeys.READ
133
+ * @memberOf FieldsetComponent
134
+ */
84
135
  this.operation = OperationKeys.READ;
85
136
  /**
86
137
  * @description Form target attribute for nested form submissions.
@@ -94,18 +145,83 @@ let FieldsetComponent = class FieldsetComponent {
94
145
  * @memberOf FieldsetComponent
95
146
  */
96
147
  this.target = '_self';
148
+ /**
149
+ * @description Enables multiple item management within the fieldset.
150
+ * @summary Boolean flag that determines if the fieldset supports adding multiple values.
151
+ * When true, displays a reorderable list of items with add/remove functionality.
152
+ *
153
+ * @type {boolean}
154
+ * @default false
155
+ * @memberOf FieldsetComponent
156
+ */
157
+ this.multiple = false;
158
+ /**
159
+ * @description Array of raw values stored in the fieldset.
160
+ * @summary Contains the actual data values that have been added to the fieldset.
161
+ * This is the source of truth for the fieldset's data state.
162
+ *
163
+ * @type {KeyValue[]}
164
+ * @default []
165
+ * @memberOf FieldsetComponent
166
+ */
167
+ this.value = [];
168
+ /**
169
+ * @description Array of formatted items for UI display.
170
+ * @summary Contains the processed items ready for display in the component template.
171
+ * These items are mapped from the raw values using the mapper configuration.
172
+ *
173
+ * @type {IFieldSetItem[]}
174
+ * @default []
175
+ * @memberOf FieldsetComponent
176
+ */
177
+ this.items = [];
97
178
  /**
98
179
  * @description Current state of the accordion (expanded or collapsed).
99
180
  * @summary Boolean flag that tracks whether the fieldset accordion is currently open or closed.
100
181
  * This property is automatically managed based on user interactions and initial operation state.
101
182
  * It serves as the single source of truth for the component's visibility state and is used
102
- * to coordinate between user actions and programmatic state changes.
183
+ * to coordinate between user actions and programmatic state changes. The value is automatically
184
+ * set based on CRUD operations during initialization and updated through user interactions.
103
185
  *
104
186
  * @type {boolean}
105
187
  * @default false
106
188
  * @memberOf FieldsetComponent
107
189
  */
108
190
  this.isOpen = false;
191
+ /**
192
+ * @description Indicates whether the fieldset contains required form fields.
193
+ * @summary Boolean flag that signals the presence of mandatory input fields within the fieldset.
194
+ * This property is automatically set by the CollapsableDirective when required fields are detected,
195
+ * and can be used to apply special styling, validation logic, or UI indicators to highlight
196
+ * fieldsets that contain mandatory information. It helps with form validation feedback and
197
+ * user experience by making required sections more prominent.
198
+ *
199
+ * @type {boolean}
200
+ * @default false
201
+ * @memberOf FieldsetComponent
202
+ */
203
+ this.isRequired = false;
204
+ /**
205
+ * @description Indicates whether the fieldset contains validation errors.
206
+ * @summary Boolean flag that tracks if any form fields within the fieldset have validation errors.
207
+ * This property is used to control accordion behavior when errors are present, preventing
208
+ * users from collapsing the accordion when they need to see and address validation issues.
209
+ * It's automatically updated when validation error events are received from child form fields.
210
+ *
211
+ * @type {boolean}
212
+ * @default false
213
+ * @memberOf FieldsetComponent
214
+ */
215
+ this.hasValidationErrors = false;
216
+ /**
217
+ * @description Validation error message for duplicate values.
218
+ * @summary Stores the error message when a user attempts to add a duplicate value
219
+ * to the fieldset. Used to display uniqueness validation feedback.
220
+ *
221
+ * @type {string | undefined}
222
+ * @memberOf FieldsetComponent
223
+ */
224
+ this.isUniqueError = undefined;
109
225
  /**
110
226
  * @description Reference to CRUD operation constants for template usage.
111
227
  * @summary Exposes the OperationKeys enum to the component template, enabling conditional
@@ -130,45 +246,261 @@ let FieldsetComponent = class FieldsetComponent {
130
246
  * @memberOf FieldsetComponent
131
247
  */
132
248
  this.changeDetectorRef = inject(ChangeDetectorRef);
249
+ /**
250
+ * @description Angular Renderer2 service for safe DOM manipulation.
251
+ * @summary Injected service that provides a safe, platform-agnostic way to manipulate DOM elements.
252
+ * This service ensures proper handling of DOM operations across different platforms and environments,
253
+ * including server-side rendering and web workers.
254
+ *
255
+ * @private
256
+ * @type {Renderer2}
257
+ * @memberOf FieldsetComponent
258
+ */
259
+ this.renderer = inject(Renderer2);
260
+ /**
261
+ * @description Translation service for internationalization.
262
+ * @summary Injected service that provides translation capabilities for UI text.
263
+ * Used to translate button labels and validation messages based on the current locale.
264
+ *
265
+ * @private
266
+ * @type {TranslateService}
267
+ * @memberOf FieldsetComponent
268
+ */
269
+ this.translateService = inject(TranslateService);
133
270
  }
134
271
  /**
135
- * @description Initializes the component state after view and child components are rendered.
136
- * @summary This lifecycle hook implements intelligent auto-state management based on the current
137
- * CRUD operation. For READ and DELETE operations, the fieldset automatically opens to provide
138
- * immediate access to information, while CREATE and UPDATE operations keep it closed to maintain
139
- * a clean initial interface. The method directly manipulates the DOM to ensure proper accordion
140
- * synchronization and triggers change detection to reflect the programmatic state changes.
141
- *
142
- * @mermaid
143
- * sequenceDiagram
144
- * participant A as Angular Lifecycle
145
- * participant F as FieldsetComponent
146
- * participant D as DOM
147
- * participant C as ChangeDetector
148
- *
149
- * A->>F: ngAfterViewInit()
150
- * alt operation is READ or DELETE
151
- * F->>F: Set isOpen = true
152
- * F->>D: Query ion-accordion-group element
153
- * alt accordion element exists
154
- * F->>D: Set value attribute to 'open'
155
- * end
156
- * end
157
- * F->>C: detectChanges()
158
- * C->>F: Update view with new state
272
+ * @description Component initialization lifecycle method.
273
+ * @summary Initializes the component by setting up repository relationships if a model exists,
274
+ * and configures the initial button label for the add action based on the current locale.
275
+ * This method ensures proper setup of translation services and component state.
159
276
  *
160
277
  * @returns {void}
161
278
  * @memberOf FieldsetComponent
162
279
  */
280
+ ngOnInit() {
281
+ if (this.model)
282
+ this._repository = this.repository;
283
+ this.buttonLabel = this.translateService.instant(this.locale + '.add');
284
+ }
285
+ /**
286
+ * @description Initializes the component state after view and child components are rendered.
287
+ * @summary This lifecycle hook implements intelligent auto-state management based on the current
288
+ * CRUD operation. For READ and DELETE operations, the fieldset automatically opens to provide
289
+ * immediate access to information, while CREATE and UPDATE operations keep it closed to maintain
290
+ * a clean initial interface. The method directly manipulates the DOM to ensure proper accordion
291
+ * synchronization and triggers change detection to reflect the programmatic state changes.
292
+ *
293
+ * @mermaid
294
+ * sequenceDiagram
295
+ * participant A as Angular Lifecycle
296
+ * participant F as FieldsetComponent
297
+ * participant D as DOM
298
+ * participant C as ChangeDetector
299
+ *
300
+ * A->>F: ngAfterViewInit()
301
+ * alt operation is READ or DELETE
302
+ * F->>F: Set isOpen = true
303
+ * F->>D: Query ion-accordion-group element
304
+ * alt accordion element exists
305
+ * F->>D: Set value attribute to 'open'
306
+ * end
307
+ * end
308
+ * F->>C: detectChanges()
309
+ * C->>F: Update view with new state
310
+ *
311
+ * @returns {void}
312
+ * @memberOf FieldsetComponent
313
+ */
163
314
  ngAfterViewInit() {
164
315
  if (this.operation === OperationKeys.READ || this.operation === OperationKeys.DELETE) {
165
316
  this.isOpen = true;
317
+ // hidden remove button
166
318
  const accordionElement = this.component?.nativeElement.querySelector('ion-accordion-group');
167
- if (accordionElement)
168
- accordionElement.setAttribute('value', 'open');
319
+ if (this.accordionComponent)
320
+ this.renderer.setAttribute(accordionElement, 'value', 'open');
321
+ }
322
+ else {
323
+ const inputs = this.component?.nativeElement.querySelectorAll('[required]');
324
+ this.isRequired = inputs.length > 0;
325
+ if (this.isRequired) {
326
+ this.accordionComponent.value = 'open';
327
+ this.handleAccordionToggle();
328
+ }
169
329
  }
170
330
  this.changeDetectorRef.detectChanges();
171
331
  }
332
+ /**
333
+ * @description Handles removal of the fieldset with slide animation.
334
+ * @summary Initiates the removal process for the fieldset with a smooth slide-up animation.
335
+ * The method applies CSS classes for the slide animation and then safely removes the
336
+ * element from the DOM using Renderer2. This provides a polished user experience
337
+ * when removing fieldset instances from dynamic forms.
338
+ *
339
+ * @param {Event} event - DOM event from the remove button click
340
+ * @returns {void}
341
+ * @memberOf FieldsetComponent
342
+ */
343
+ handleRemoveComponent(event) {
344
+ event.stopImmediatePropagation();
345
+ this.component.nativeElement.classList.add('dcf-animation', 'dcf-animation-slide-top-medium', 'dcf-animation-reverse', 'dcf-animation-fast');
346
+ setTimeout(() => {
347
+ // Use Renderer2 to safely remove the element
348
+ const parent = this.renderer.parentNode(this.component.nativeElement);
349
+ if (parent)
350
+ this.renderer.removeChild(parent, this.component.nativeElement);
351
+ }, 150);
352
+ }
353
+ /**
354
+ * @description Handles creating new items or triggering group addition events.
355
+ * @summary Processes form validation events for item creation or emits events to trigger
356
+ * the addition of new fieldset groups. When called with validation event data, it validates
357
+ * uniqueness and adds the item to the fieldset. When called without parameters, it triggers
358
+ * a group addition event for parent components to handle.
359
+ *
360
+ * @param {CustomEvent<IFieldSetValidationEvent>} [event] - Optional validation event containing form data
361
+ * @returns {Promise<void>}
362
+ * @memberOf FieldsetComponent
363
+ *
364
+ * @example
365
+ * ```typescript
366
+ * // Called from form validation
367
+ * handleCreateItem(validationEvent);
368
+ *
369
+ * // Called to trigger group addition
370
+ * handleCreateItem();
371
+ * ```
372
+ */
373
+ async handleCreateItem(event) {
374
+ if (event && event instanceof CustomEvent) {
375
+ event.stopImmediatePropagation();
376
+ const { formGroup, value, isValid } = event.detail;
377
+ this.formGroup = formGroup;
378
+ if (!this.mapper)
379
+ this.mapper = this.getMapper(value);
380
+ if (isValid) {
381
+ this.isUniqueError = undefined;
382
+ this.buttonLabel = this.translateService.instant(this.locale + '.add');
383
+ this.setValue();
384
+ }
385
+ else {
386
+ this.isUniqueError = value?.[this.pk] || undefined;
387
+ }
388
+ }
389
+ else {
390
+ windowEventEmitter(EventConstants.FIELDSET_ADD_GROUP, {
391
+ component: this.component.nativeElement,
392
+ index: this.value?.length,
393
+ parent: this.childOf,
394
+ operation: !this.updatingItem ? OperationKeys.CREATE : OperationKeys.UPDATE
395
+ });
396
+ }
397
+ }
398
+ /**
399
+ * @description Handles item update operations with form state management.
400
+ * @summary Locates an item in the form array for editing and prepares the component
401
+ * for update mode. Updates the button label to reflect the edit state and stores
402
+ * the item being updated. Triggers a window event to notify parent components.
403
+ *
404
+ * @param {string | number} value - The identifier value of the item to update
405
+ * @param {number} index - The array index position of the item
406
+ * @returns {void}
407
+ * @memberOf FieldsetComponent
408
+ */
409
+ handleUpdateItem(value, index) {
410
+ const item = this.formGroup.controls.find(control => `${control.get(this.pk)?.value}`.toLowerCase() === cleanSpaces(`${value}`, true));
411
+ if (item) {
412
+ this.buttonLabel = this.translateService.instant(this.locale + '.update');
413
+ this.updatingItem = Object.assign({}, item.value || {});
414
+ windowEventEmitter(EventConstants.FIELDSET_UPDATE_GROUP, {
415
+ parent: this.childOf,
416
+ component: this.component.nativeElement,
417
+ index: index
418
+ });
419
+ }
420
+ }
421
+ /**
422
+ * @description Cancels the update mode and resets the UI state.
423
+ * @summary Exits the update mode by resetting the button label and clearing the updating item,
424
+ * restoring the component to its default state for adding new items. Notifies parent components
425
+ * that the update operation has been cancelled.
426
+ *
427
+ * @returns {void}
428
+ * @memberOf FieldsetComponent
429
+ */
430
+ handleCancelUpdateItem() {
431
+ this.buttonLabel = this.translateService.instant(this.locale + '.add');
432
+ this.updatingItem = undefined;
433
+ windowEventEmitter(EventConstants.FIELDSET_UPDATE_GROUP, {
434
+ parent: this.childOf,
435
+ component: this.component.nativeElement,
436
+ index: this.value?.length
437
+ });
438
+ }
439
+ /**
440
+ * @description Handles item removal operations with form array management.
441
+ * @summary Processes item removal by either handling validation events or removing specific
442
+ * items from the form array. When called with a validation event, it triggers value updates.
443
+ * When called with an identifier, it locates and removes the matching item from the form array.
444
+ *
445
+ * @param {string | undefined} value - The identifier of the item to remove
446
+ * @param {CustomEvent} [event] - Optional validation event for form updates
447
+ * @returns {void}
448
+ * @memberOf FieldsetComponent
449
+ */
450
+ handleRemoveItem(value, event) {
451
+ if (event && event instanceof CustomEvent) {
452
+ event.stopImmediatePropagation();
453
+ return this.setValue();
454
+ }
455
+ const formArray = this.formGroup;
456
+ const arrayLength = formArray.length;
457
+ for (let index = arrayLength - 1; index >= 0; index--) {
458
+ const group = formArray.at(index);
459
+ if (cleanSpaces(group.get(this.pk)?.value) === cleanSpaces(value)) {
460
+ windowEventEmitter(EventConstants.FIELDSET_REMOVE_GROUP, {
461
+ parent: this.childOf,
462
+ component: this.component.nativeElement,
463
+ index,
464
+ formGroup: group
465
+ });
466
+ }
467
+ }
468
+ }
469
+ /**
470
+ * @description Handles reordering of items within the fieldset list.
471
+ * @summary Processes drag-and-drop reorder events from the ion-reorder-group component.
472
+ * Updates both the display items array and the underlying value array to maintain
473
+ * consistency between UI state and data state. Preserves item indices after reordering.
474
+ *
475
+ * @param {CustomEvent<ItemReorderEventDetail>} event - Ionic reorder event containing source and target indices
476
+ * @returns {void}
477
+ * @memberOf FieldsetComponent
478
+ *
479
+ * @example
480
+ * ```html
481
+ * <ion-reorder-group (ionItemReorder)="handleReorder($event)">
482
+ * <!-- Reorderable items -->
483
+ * </ion-reorder-group>
484
+ * ```
485
+ */
486
+ handleReorderItems(event) {
487
+ const fromIndex = event.detail.from;
488
+ const toIndex = event.detail.to;
489
+ const items = [...this.items]; // sua estrutura visual
490
+ const formArray = this.formGroup; // FormArray reativo
491
+ if (fromIndex !== toIndex) {
492
+ // Reordenar os dados visuais
493
+ const itemToMove = items.splice(fromIndex, 1)[0];
494
+ items.splice(toIndex, 0, itemToMove);
495
+ items.forEach((item, index) => item['index'] = index + 1);
496
+ // Reordenar os controles do FormArray
497
+ const controlToMove = formArray.at(fromIndex);
498
+ formArray.removeAt(fromIndex);
499
+ formArray.insert(toIndex, controlToMove);
500
+ }
501
+ // Finaliza a operação de reorder do Ionic
502
+ event.detail.complete();
503
+ }
172
504
  /**
173
505
  * @description Handles accordion state change events from user interactions.
174
506
  * @summary Processes CustomEvent objects triggered when users expand or collapse the accordion.
@@ -196,30 +528,137 @@ let FieldsetComponent = class FieldsetComponent {
196
528
  *
197
529
  * @memberOf FieldsetComponent
198
530
  */
199
- handleChange(event) {
200
- const { target, detail } = event;
201
- const { value } = detail;
202
- if (target.tagName === 'ION-ACCORDION-GROUP')
203
- this.isOpen = !!value;
531
+ /**
532
+ * @description Handles accordion toggle functionality with validation error consideration.
533
+ * @summary Manages the expand/collapse state of the accordion while respecting validation error states.
534
+ * When validation errors are present, the accordion cannot be collapsed to ensure users can see
535
+ * and address the errors. When no errors exist, users can freely toggle the accordion state.
536
+ * This method also stops event propagation to prevent unwanted side effects.
537
+ *
538
+ * @param {CustomEvent} [event] - Optional event object from user interaction
539
+ * @returns {void}
540
+ * @memberOf FieldsetComponent
541
+ */
542
+ handleAccordionToggle(event) {
543
+ if (event)
544
+ event.stopImmediatePropagation();
545
+ if (!this.hasValidationErrors) {
546
+ this.accordionComponent.value = this.isOpen ? undefined : 'open';
547
+ this.isOpen = !!this.accordionComponent.value;
548
+ }
549
+ }
550
+ /**
551
+ * @description Handles validation error events from child form fields.
552
+ * @summary Processes validation error events dispatched by form fields within the fieldset.
553
+ * When errors are detected, the accordion is forced open and prevented from collapsing
554
+ * to ensure users can see the validation messages. This method updates the component's
555
+ * error state and accordion visibility accordingly.
556
+ *
557
+ * @param {CustomEvent} event - Custom event containing validation error details
558
+ * @returns {void}
559
+ * @memberOf FieldsetComponent
560
+ */
561
+ handleValidationError(event) {
562
+ event.stopImmediatePropagation();
563
+ const { hasErrors } = event.detail;
564
+ this.isOpen = this.hasValidationErrors = hasErrors;
565
+ if (hasErrors)
566
+ this.accordionComponent.value = 'open';
567
+ }
568
+ /**
569
+ * @description Processes and stores a new or updated value in the fieldset.
570
+ * @summary Handles both create and update operations for fieldset items. Parses and cleans
571
+ * the input value, determines the operation type based on the updating state, and either
572
+ * adds a new item or updates an existing one. Maintains data integrity and UI consistency.
573
+ *
574
+ * @returns {void}
575
+ * @private
576
+ * @memberOf FieldsetComponent
577
+ */
578
+ setValue() {
579
+ this.value = this.formGroup.controls.map(({ value }) => value);
580
+ this.items = this.value
581
+ .filter(v => v[this.pk] !== undefined)
582
+ .map((v, index) => {
583
+ return {
584
+ ...itemMapper(v, this.mapper),
585
+ index: index + 1
586
+ };
587
+ });
588
+ const inputContainers = this.component.nativeElement.querySelectorAll('.dcf-input-item');
589
+ inputContainers.forEach((container) => {
590
+ const input = container.querySelector('input, ion-input, ion-textarea, textarea');
591
+ if (input)
592
+ input.value = '';
593
+ });
594
+ this.updatingItem = undefined;
595
+ }
596
+ /**
597
+ * @description Automatically configures the field mapping based on the value structure.
598
+ * @summary Analyzes the provided value object to automatically determine the primary key
599
+ * and create appropriate field mappings for display purposes. Sets up the mapper object
600
+ * with title, description, and index fields based on the available data structure.
601
+ *
602
+ * @param {KeyValue} value - Sample value object used to determine field mappings
603
+ * @returns {KeyValue} The configured mapper object
604
+ * @private
605
+ * @memberOf FieldsetComponent
606
+ */
607
+ getMapper(value) {
608
+ if (!this.pk)
609
+ this.pk = Object.keys(value)[0];
610
+ if (!Object.keys(this.mapper).length)
611
+ this.mapper['title'] = this.pk;
612
+ this.mapper['index'] = "index";
613
+ for (const key in value) {
614
+ if (Object.keys(this.mapper).length >= 2 || Object.keys(this.mapper).length === Object.keys(value).length)
615
+ break;
616
+ if (!this.mapper['title']) {
617
+ this.mapper['title'] = key;
618
+ }
619
+ else {
620
+ this.mapper['description'] = key;
621
+ }
622
+ }
623
+ return this.mapper;
204
624
  }
205
625
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FieldsetComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
206
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: FieldsetComponent, isStandalone: true, selector: "ngx-decaf-fieldset", inputs: { name: "name", operation: "operation", target: "target" }, viewQueries: [{ propertyName: "component", first: true, predicate: ["component"], descendants: true, read: ElementRef }], ngImport: i0, template: "<fieldset [class]=\"'dcf-fieldset ' + operation\" #component>\n <ion-accordion-group (ionChange)=\"handleChange($event)\">\n <ion-accordion value=\"open\">\n <ion-item slot=\"header\" >\n <legend>{{ name }}</legend>\n </ion-item>\n <div slot=\"content\" [attr.aria-hidden]=\"!isOpen\" decafCollapsable>\n <ng-content></ng-content>\n </div>\n </ion-accordion>\n </ion-accordion-group>\n</fieldset>\n", styles: [".dcf-fieldset{margin-bottom:1.8rem;padding-bottom:0;padding-top:1rem;border:1px solid var(--ion-color-gray-2);border-radius:8px}.dcf-fieldset ion-accordion-group{--border-color:red !important}.dcf-fieldset.read{margin-top:1.25rem}.dcf-fieldset ion-accordion.accordion-collapsing,.dcf-fieldset ion-accordion.accordion-collapsed{margin-bottom:1rem}.dcf-fieldset ion-accordion ion-item[slot=header]{--border-color: transparent;--border-radius: 6px;--inner-border-width: 0;--padding-start: 12px}.dcf-fieldset ion-accordion ion-item[slot=header] legend{font-weight:600;font-size:1rem;color:#333;margin:0}.dcf-fieldset ion-accordion [slot=content]{padding-top:1rem!important;padding-inline:.75rem;background-color:#fff}\n"], dependencies: [{ kind: "ngmodule", type: ForAngularModule }, { kind: "component", type: IonAccordionGroup, selector: "ion-accordion-group", inputs: ["animated", "disabled", "expand", "mode", "multiple", "readonly", "value"] }, { kind: "component", type: IonAccordion, selector: "ion-accordion", inputs: ["disabled", "mode", "readonly", "toggleIcon", "toggleIconSlot", "value"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "directive", type: CollapsableDirective, selector: "[decafCollapsable]" }] }); }
626
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: FieldsetComponent, isStandalone: true, selector: "ngx-decaf-fieldset", inputs: { name: "name", childOf: "childOf", uid: "uid", customTypes: "customTypes", operation: "operation", formGroup: "formGroup", title: "title", description: "description", target: "target", multiple: "multiple", value: "value", handlers: "handlers" }, host: { properties: { "attr.id": "overriode " } }, viewQueries: [{ propertyName: "accordionComponent", first: true, predicate: ["accordionComponent"], descendants: true }], usesInheritance: true, ngImport: i0, template: "\n\n<fieldset\n (fieldsetAddGroupEvent)=\"handleCreateItem($event)\"\n (fieldsetRemoveGroupEvent)=\"handleRemoveItem(undefined, $event)\"\n [class]=\"'dcf-fieldset ' + operation\"\n #component>\n <ion-accordion-group [ngClass]=\"{'open': isOpen, 'hasValidationErrors': hasValidationErrors}\" (validationErrorEvent)=\"handleValidationError($event)\" #accordionComponent>\n <ion-accordion value=\"open\">\n <ion-item slot=\"header\" (click)=\"handleAccordionToggle($event)\">\n <div class=\"dcf-grid dcf-grid-collapse dcf-flex dcf-flex-middle dcf-width-1-1\">\n <div class=\"dcf-width-expand\">\n <legend>{{ name | translate }}</legend>\n </div>\n @if(!isRequired && ['create', 'update'].includes(operation)) {\n <div class=\"dcf-width-auto dcf-delete\">\n <ion-button fill=\"clear\" size=\"small\" (click)=\"handleRemoveComponent($event)\">\n <ion-icon name=\"trash-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n </ion-button>\n </div>\n }\n </div>\n </ion-item>\n <div slot=\"content\" [attr.aria-hidden]=\"!isOpen\">\n @if(multiple && items.length) {\n <ion-list class=\"dcf-fields-list\">\n <ion-reorder-group [formGroup]=\"formGroup.parent\" [disabled]=\"updatingItem\" (ionItemReorder)=\"handleReorderItems($any($event))\" #accordionComponent>\n @for(item of items; track item.index) {\n <ion-item [ngClass]=\"{'not-unique': item.title === isUniqueError}\" lines=\"full\" [button]=\"false\" [ngClass]=\"{'updating': updatingItem?.[pk] === item.title}\">\n @if(items?.length > 1 && !updatingItem) {\n <ion-reorder slot=\"start\">\n <ion-icon name=\"swap-vertical-outline\"></ion-icon>\n </ion-reorder>\n } @else {\n <div slot=\"start\">\n <ion-icon class=\"dcf-reorder-disabled\" size=\"small\" name=\"swap-vertical-outline\" disabled></ion-icon>\n </div>\n }\n <ion-label [color]=\"item.title === isUniqueError ? 'danger' : ''\">{{ item.index }}. {{ item.title }}\n @if(item.description?.length > 0) {\n <br />\n <ion-text class=\"dcf-subtitle\">{{item.description}}</ion-text>\n }\n </ion-label>\n @if(!updatingItem || updatingItem?.[pk] !== item.title) {\n <ion-button fill=\"clear\" size=\"small\" (click)=\"handleUpdateItem(item.title, $index)\">\n <ion-icon name=\"create-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n </ion-button>\n }\n\n @if(!updatingItem) {\n <ion-button fill=\"clear\" size=\"small\" (click)=\"handleRemoveItem(item.title)\">\n <ion-icon name=\"trash-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n </ion-button>\n }\n </ion-item>\n }\n </ion-reorder-group>\n </ion-list>\n }\n\n <ng-content></ng-content>\n @if(multiple && ['create', 'update'].includes(operation)) {\n @if(isUniqueError) {\n <div class=\"dcf-not-unique-container dcf-animation dcf-animation-bottom-small dcf-animation-fast\">\n <div class=\" dcf-grid dcf-grid-collapse dcf-width-1-1 \">\n <div class=\"dcf-auto\" [attr.style]=\"'max-width: 50px'\">\n <ion-icon name=\"alert-circle-outline\"></ion-icon>\n </div>\n <div class=\"dcf-width-expand\">\n <ion-text color=\"danger\" class=\"dcf-text-small\">{{ locale + '.not_unique' | translate : { value: isUniqueError } }}</ion-text>\n </div>\n </div>\n </div>\n }\n <div class=\"dcf-margin-bottom dcf-grid dcf-grid-collapse dcf-flex\">\n @if(updatingItem) {\n <ion-button size=\"small\" fill=\"clear\" color=\"danger\" (click)=\"handleCancelUpdateItem()\">\n {{ locale + '.cancel' | translate }}\n </ion-button>\n }\n <ion-button size=\"small\" fill=\"clear\" color=\"dark\" (click)=\"handleCreateItem()\">\n <ion-icon name=\"add-outline\" slot=\"start\"></ion-icon>\n {{buttonLabel}}\n </ion-button>\n\n </div>\n }\n\n </div>\n </ion-accordion>\n </ion-accordion-group>\n</fieldset>\n\n", styles: ["ion-accordion-group ion-item[slot=header] .dcf-delete{width:30px}ion-accordion-group ion-item[slot=header] .dcf-delete ion-button{transform:translateY(-2px)}ion-accordion-group ion-item[slot=header] .dcf-delete ion-icon{font-size:1.15rem}::ng-deep ion-accordion ngx-decaf-crud-field:last-child ion-item{--inner-border-width: 0px !important;--border-width: 0px !important}.dcf-fieldset{margin-bottom:1.8rem;padding-bottom:0;padding-top:1rem;border:1px solid var(--dcf-color-gray-2);border-radius:8px;height:100%}.dcf-fieldset.read{margin-top:1.25rem;padding-bottom:1rem}.dcf-fieldset.read [slot=content]{padding-top:0!important;background:red}.dcf-fieldset ion-accordion.accordion-collapsing,.dcf-fieldset ion-accordion.accordion-collapsed{margin-bottom:1rem}.dcf-fieldset ion-accordion ion-item[slot=header]{--border-color: transparent;--border-radius: 6px;--inner-border-width: 0;--padding-start: 12px}.dcf-fieldset ion-accordion ion-item[slot=header] legend{font-weight:600;font-size:1rem;color:#333;margin:0}.dcf-fieldset ion-accordion [slot=content]{padding-top:1rem!important;padding-inline:.75rem;background-color:#fff}.dcf-not-unique-container{display:flex;justify-content:center;align-items:center;margin-bottom:1rem;flex:1 1 auto}.dcf-not-unique-container>div{display:flex;justify-content:center;align-items:center}.dcf-not-unique-container ion-icon{transform:translatey(2px);margin-right:5px}.dcf-fields-list{margin-top:-1rem;margin-bottom:1rem;padding:0!important}ion-item{--min-height: 50px;--padding-top: .25rem;--padding-bottom: .25rem;--padding-start: .75rem;--padding-end: .75rem;--inner-padding-start: 0px !important;--inner-padding-end: 0px !important;--border-color: var(--dcf-color-gray-2);border:1px solid transparent;box-sizing:border-box}ion-item ion-icon.dcf-reorder-disabled{width:1rem;transform:translatey(2px);color:var(--dcf-color-gray-4)}ion-item.updating{--background: rgba(var(--dcf-color-primary-rgb), .1) !important}ion-item.not-unique{--background: rgba(var(--dcf-color-danger-rgb), .05) !important}ion-item .dcf-subtitle{font-size:.8rem;color:var(--dcf-color-gray-7)}\n"], dependencies: [{ kind: "ngmodule", type: ForAngularModule }, { kind: "component", type: i1.IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: i1.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }, { kind: "component", type: IonAccordionGroup, selector: "ion-accordion-group", inputs: ["animated", "disabled", "expand", "mode", "multiple", "readonly", "value"] }, { kind: "component", type: IonAccordion, selector: "ion-accordion", inputs: ["disabled", "mode", "readonly", "toggleIcon", "toggleIconSlot", "value"] }, { kind: "component", type: IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonReorder, selector: "ion-reorder" }, { kind: "component", type: IonReorderGroup, selector: "ion-reorder-group", inputs: ["disabled"] }] }); }
207
627
  };
208
628
  FieldsetComponent = __decorate([
209
- Dynamic()
629
+ Dynamic(),
630
+ __metadata("design:paramtypes", [])
210
631
  ], FieldsetComponent);
211
632
  export { FieldsetComponent };
212
633
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FieldsetComponent, decorators: [{
213
634
  type: Component,
214
- args: [{ standalone: true, selector: 'ngx-decaf-fieldset', schemas: [CUSTOM_ELEMENTS_SCHEMA], imports: [ForAngularModule, IonAccordionGroup, IonAccordion, IonItem, CollapsableDirective], template: "<fieldset [class]=\"'dcf-fieldset ' + operation\" #component>\n <ion-accordion-group (ionChange)=\"handleChange($event)\">\n <ion-accordion value=\"open\">\n <ion-item slot=\"header\" >\n <legend>{{ name }}</legend>\n </ion-item>\n <div slot=\"content\" [attr.aria-hidden]=\"!isOpen\" decafCollapsable>\n <ng-content></ng-content>\n </div>\n </ion-accordion>\n </ion-accordion-group>\n</fieldset>\n", styles: [".dcf-fieldset{margin-bottom:1.8rem;padding-bottom:0;padding-top:1rem;border:1px solid var(--ion-color-gray-2);border-radius:8px}.dcf-fieldset ion-accordion-group{--border-color:red !important}.dcf-fieldset.read{margin-top:1.25rem}.dcf-fieldset ion-accordion.accordion-collapsing,.dcf-fieldset ion-accordion.accordion-collapsed{margin-bottom:1rem}.dcf-fieldset ion-accordion ion-item[slot=header]{--border-color: transparent;--border-radius: 6px;--inner-border-width: 0;--padding-start: 12px}.dcf-fieldset ion-accordion ion-item[slot=header] legend{font-weight:600;font-size:1rem;color:#333;margin:0}.dcf-fieldset ion-accordion [slot=content]{padding-top:1rem!important;padding-inline:.75rem;background-color:#fff}\n"] }]
215
- }], propDecorators: { component: [{
635
+ args: [{ standalone: true, selector: 'ngx-decaf-fieldset', schemas: [CUSTOM_ELEMENTS_SCHEMA], imports: [ForAngularModule, IonAccordionGroup, IonAccordion, IonList, IonItem, IonLabel, IonReorder, IonButton, IonReorderGroup, CollapsableDirective], host: { '[attr.id]': 'overriode ' }, template: "\n\n<fieldset\n (fieldsetAddGroupEvent)=\"handleCreateItem($event)\"\n (fieldsetRemoveGroupEvent)=\"handleRemoveItem(undefined, $event)\"\n [class]=\"'dcf-fieldset ' + operation\"\n #component>\n <ion-accordion-group [ngClass]=\"{'open': isOpen, 'hasValidationErrors': hasValidationErrors}\" (validationErrorEvent)=\"handleValidationError($event)\" #accordionComponent>\n <ion-accordion value=\"open\">\n <ion-item slot=\"header\" (click)=\"handleAccordionToggle($event)\">\n <div class=\"dcf-grid dcf-grid-collapse dcf-flex dcf-flex-middle dcf-width-1-1\">\n <div class=\"dcf-width-expand\">\n <legend>{{ name | translate }}</legend>\n </div>\n @if(!isRequired && ['create', 'update'].includes(operation)) {\n <div class=\"dcf-width-auto dcf-delete\">\n <ion-button fill=\"clear\" size=\"small\" (click)=\"handleRemoveComponent($event)\">\n <ion-icon name=\"trash-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n </ion-button>\n </div>\n }\n </div>\n </ion-item>\n <div slot=\"content\" [attr.aria-hidden]=\"!isOpen\">\n @if(multiple && items.length) {\n <ion-list class=\"dcf-fields-list\">\n <ion-reorder-group [formGroup]=\"formGroup.parent\" [disabled]=\"updatingItem\" (ionItemReorder)=\"handleReorderItems($any($event))\" #accordionComponent>\n @for(item of items; track item.index) {\n <ion-item [ngClass]=\"{'not-unique': item.title === isUniqueError}\" lines=\"full\" [button]=\"false\" [ngClass]=\"{'updating': updatingItem?.[pk] === item.title}\">\n @if(items?.length > 1 && !updatingItem) {\n <ion-reorder slot=\"start\">\n <ion-icon name=\"swap-vertical-outline\"></ion-icon>\n </ion-reorder>\n } @else {\n <div slot=\"start\">\n <ion-icon class=\"dcf-reorder-disabled\" size=\"small\" name=\"swap-vertical-outline\" disabled></ion-icon>\n </div>\n }\n <ion-label [color]=\"item.title === isUniqueError ? 'danger' : ''\">{{ item.index }}. {{ item.title }}\n @if(item.description?.length > 0) {\n <br />\n <ion-text class=\"dcf-subtitle\">{{item.description}}</ion-text>\n }\n </ion-label>\n @if(!updatingItem || updatingItem?.[pk] !== item.title) {\n <ion-button fill=\"clear\" size=\"small\" (click)=\"handleUpdateItem(item.title, $index)\">\n <ion-icon name=\"create-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n </ion-button>\n }\n\n @if(!updatingItem) {\n <ion-button fill=\"clear\" size=\"small\" (click)=\"handleRemoveItem(item.title)\">\n <ion-icon name=\"trash-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n </ion-button>\n }\n </ion-item>\n }\n </ion-reorder-group>\n </ion-list>\n }\n\n <ng-content></ng-content>\n @if(multiple && ['create', 'update'].includes(operation)) {\n @if(isUniqueError) {\n <div class=\"dcf-not-unique-container dcf-animation dcf-animation-bottom-small dcf-animation-fast\">\n <div class=\" dcf-grid dcf-grid-collapse dcf-width-1-1 \">\n <div class=\"dcf-auto\" [attr.style]=\"'max-width: 50px'\">\n <ion-icon name=\"alert-circle-outline\"></ion-icon>\n </div>\n <div class=\"dcf-width-expand\">\n <ion-text color=\"danger\" class=\"dcf-text-small\">{{ locale + '.not_unique' | translate : { value: isUniqueError } }}</ion-text>\n </div>\n </div>\n </div>\n }\n <div class=\"dcf-margin-bottom dcf-grid dcf-grid-collapse dcf-flex\">\n @if(updatingItem) {\n <ion-button size=\"small\" fill=\"clear\" color=\"danger\" (click)=\"handleCancelUpdateItem()\">\n {{ locale + '.cancel' | translate }}\n </ion-button>\n }\n <ion-button size=\"small\" fill=\"clear\" color=\"dark\" (click)=\"handleCreateItem()\">\n <ion-icon name=\"add-outline\" slot=\"start\"></ion-icon>\n {{buttonLabel}}\n </ion-button>\n\n </div>\n }\n\n </div>\n </ion-accordion>\n </ion-accordion-group>\n</fieldset>\n\n", styles: ["ion-accordion-group ion-item[slot=header] .dcf-delete{width:30px}ion-accordion-group ion-item[slot=header] .dcf-delete ion-button{transform:translateY(-2px)}ion-accordion-group ion-item[slot=header] .dcf-delete ion-icon{font-size:1.15rem}::ng-deep ion-accordion ngx-decaf-crud-field:last-child ion-item{--inner-border-width: 0px !important;--border-width: 0px !important}.dcf-fieldset{margin-bottom:1.8rem;padding-bottom:0;padding-top:1rem;border:1px solid var(--dcf-color-gray-2);border-radius:8px;height:100%}.dcf-fieldset.read{margin-top:1.25rem;padding-bottom:1rem}.dcf-fieldset.read [slot=content]{padding-top:0!important;background:red}.dcf-fieldset ion-accordion.accordion-collapsing,.dcf-fieldset ion-accordion.accordion-collapsed{margin-bottom:1rem}.dcf-fieldset ion-accordion ion-item[slot=header]{--border-color: transparent;--border-radius: 6px;--inner-border-width: 0;--padding-start: 12px}.dcf-fieldset ion-accordion ion-item[slot=header] legend{font-weight:600;font-size:1rem;color:#333;margin:0}.dcf-fieldset ion-accordion [slot=content]{padding-top:1rem!important;padding-inline:.75rem;background-color:#fff}.dcf-not-unique-container{display:flex;justify-content:center;align-items:center;margin-bottom:1rem;flex:1 1 auto}.dcf-not-unique-container>div{display:flex;justify-content:center;align-items:center}.dcf-not-unique-container ion-icon{transform:translatey(2px);margin-right:5px}.dcf-fields-list{margin-top:-1rem;margin-bottom:1rem;padding:0!important}ion-item{--min-height: 50px;--padding-top: .25rem;--padding-bottom: .25rem;--padding-start: .75rem;--padding-end: .75rem;--inner-padding-start: 0px !important;--inner-padding-end: 0px !important;--border-color: var(--dcf-color-gray-2);border:1px solid transparent;box-sizing:border-box}ion-item ion-icon.dcf-reorder-disabled{width:1rem;transform:translatey(2px);color:var(--dcf-color-gray-4)}ion-item.updating{--background: rgba(var(--dcf-color-primary-rgb), .1) !important}ion-item.not-unique{--background: rgba(var(--dcf-color-danger-rgb), .05) !important}ion-item .dcf-subtitle{font-size:.8rem;color:var(--dcf-color-gray-7)}\n"] }]
636
+ }], ctorParameters: () => [], propDecorators: { accordionComponent: [{
216
637
  type: ViewChild,
217
- args: ['component', { static: false, read: ElementRef }]
638
+ args: ['accordionComponent', { static: false }]
218
639
  }], name: [{
219
640
  type: Input
641
+ }], childOf: [{
642
+ type: Input
643
+ }], uid: [{
644
+ type: Input
645
+ }], customTypes: [{
646
+ type: Input
220
647
  }], operation: [{
221
648
  type: Input
649
+ }], formGroup: [{
650
+ type: Input
651
+ }], title: [{
652
+ type: Input
653
+ }], description: [{
654
+ type: Input
222
655
  }], target: [{
223
656
  type: Input
657
+ }], multiple: [{
658
+ type: Input
659
+ }], value: [{
660
+ type: Input
661
+ }], handlers: [{
662
+ type: Input
224
663
  }] } });
225
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmllbGRzZXQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL2ZpZWxkc2V0L2ZpZWxkc2V0LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9maWVsZHNldC9maWVsZHNldC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQ0EsT0FBTyxFQUFpQixpQkFBaUIsRUFBRSxTQUFTLEVBQUUsc0JBQXNCLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzFJLE9BQU8sRUFBRSxPQUFPLEVBQWtCLE1BQU0sY0FBYyxDQUFDO0FBQ3ZELE9BQU8sRUFBa0IsYUFBYSxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDeEUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDNUQsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sd0NBQXdDLENBQUM7QUFDOUUsT0FBTyxFQUFFLFlBQVksRUFBRSxpQkFBaUIsRUFBRSxPQUFPLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQzs7QUFFckY7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FpREc7QUFVSSxJQUFNLGlCQUFpQixHQUF2QixNQUFNLGlCQUFpQjtJQUF2QjtRQWVMOzs7Ozs7Ozs7O1dBVUc7UUFFSCxTQUFJLEdBQVcsT0FBTyxDQUFDO1FBRXZCOzs7Ozs7Ozs7O1dBVUc7UUFFSCxjQUFTLEdBQWtCLGFBQWEsQ0FBQyxJQUFJLENBQUM7UUFFOUM7Ozs7Ozs7Ozs7V0FVRztRQUVILFdBQU0sR0FBbUIsT0FBTyxDQUFDO1FBRWpDOzs7Ozs7Ozs7O1dBVUc7UUFDSCxXQUFNLEdBQVksS0FBSyxDQUFDO1FBRXhCOzs7Ozs7Ozs7O1dBVUc7UUFDZ0Isa0JBQWEsR0FBbUIsYUFBYSxDQUFDLE1BQU0sQ0FBQztRQUV4RTs7Ozs7Ozs7OztXQVVHO1FBQ0ssc0JBQWlCLEdBQXNCLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO0tBMEUxRTtJQXhFQzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztPQTRCRztJQUNILGVBQWU7UUFDYixJQUFJLElBQUksQ0FBQyxTQUFTLEtBQUssYUFBYSxDQUFDLElBQUksSUFBSSxJQUFJLENBQUMsU0FBUyxLQUFLLGFBQWEsQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNyRixJQUFJLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQztZQUNuQixNQUFNLGdCQUFnQixHQUFHLElBQUksQ0FBQyxTQUFTLEVBQUUsYUFBYSxDQUFDLGFBQWEsQ0FBQyxxQkFBcUIsQ0FBQyxDQUFDO1lBQzVGLElBQUcsZ0JBQWdCO2dCQUNqQixnQkFBZ0IsQ0FBQyxZQUFZLENBQUMsT0FBTyxFQUFFLE1BQU0sQ0FBQyxDQUFDO1FBQ25ELENBQUM7UUFDRCxJQUFJLENBQUMsaUJBQWlCLENBQUMsYUFBYSxFQUFFLENBQUM7SUFDekMsQ0FBQztJQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztPQTBCRztJQUNILFlBQVksQ0FBQyxLQUFrQjtRQUM3QixNQUFNLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxHQUFHLEtBQUssQ0FBQztRQUNqQyxNQUFNLEVBQUUsS0FBSyxFQUFFLEdBQUcsTUFBTSxDQUFDO1FBQ3pCLElBQUssTUFBdUMsQ0FBQyxPQUFPLEtBQUsscUJBQXFCO1lBQzVFLElBQUksQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQztJQUMxQixDQUFDOytHQXZLVSxpQkFBaUI7bUdBQWpCLGlCQUFpQixxT0FZbUIsVUFBVSw2QkMvRTNELDBiQVlBLG93QkRxRFksZ0JBQWdCLCtCQUFFLGlCQUFpQixxSkFBRSxZQUFZLDZJQUFFLE9BQU8sME5BQUUsb0JBQW9COztBQUUvRSxpQkFBaUI7SUFUN0IsT0FBTyxFQUFFO0dBU0csaUJBQWlCLENBd0s3Qjs7NEZBeEtZLGlCQUFpQjtrQkFSN0IsU0FBUztpQ0FDSSxJQUFJLFlBQ04sb0JBQW9CLFdBR3JCLENBQUMsc0JBQXNCLENBQUMsV0FDeEIsQ0FBQyxnQkFBZ0IsRUFBRSxpQkFBaUIsRUFBRSxZQUFZLEVBQUUsT0FBTyxFQUFFLG9CQUFvQixDQUFDOzhCQWUzRixTQUFTO3NCQURSLFNBQVM7dUJBQUMsV0FBVyxFQUFFLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFO2dCQWUzRCxJQUFJO3NCQURILEtBQUs7Z0JBZU4sU0FBUztzQkFEUixLQUFLO2dCQWVOLE1BQU07c0JBREwsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbIlxuaW1wb3J0IHsgQWZ0ZXJWaWV3SW5pdCwgQ2hhbmdlRGV0ZWN0b3JSZWYsIENvbXBvbmVudCwgQ1VTVE9NX0VMRU1FTlRTX1NDSEVNQSwgRWxlbWVudFJlZiwgaW5qZWN0LCBJbnB1dCwgVmlld0NoaWxkIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBEeW5hbWljLCBIVE1MRm9ybVRhcmdldCB9IGZyb20gJy4uLy4uL2VuZ2luZSc7XG5pbXBvcnQgeyBDcnVkT3BlcmF0aW9ucywgT3BlcmF0aW9uS2V5cyB9IGZyb20gJ0BkZWNhZi10cy9kYi1kZWNvcmF0b3JzJztcbmltcG9ydCB7IEZvckFuZ3VsYXJNb2R1bGUgfSBmcm9tICcuLi8uLi9mb3ItYW5ndWxhci5tb2R1bGUnO1xuaW1wb3J0IHsgQ29sbGFwc2FibGVEaXJlY3RpdmUgfSBmcm9tICcuLi8uLi9kaXJlY3RpdmVzL2NvbGxhcHNhYmxlLmRpcmVjdGl2ZSc7XG5pbXBvcnQgeyBJb25BY2NvcmRpb24sIElvbkFjY29yZGlvbkdyb3VwLCBJb25JdGVtIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXIvc3RhbmRhbG9uZSc7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIER5bmFtaWMgZmllbGRzZXQgY29tcG9uZW50IHdpdGggY29sbGFwc2libGUgYWNjb3JkaW9uIGZ1bmN0aW9uYWxpdHkuXG4gKiBAc3VtbWFyeSBUaGlzIGNvbXBvbmVudCBwcm92aWRlcyBhIHNvcGhpc3RpY2F0ZWQgZmllbGRzZXQgY29udGFpbmVyIHRoYXQgYXV0b21hdGljYWxseVxuICogYWRhcHRzIGl0cyBiZWhhdmlvciBiYXNlZCBvbiBDUlVEIG9wZXJhdGlvbnMuIEl0IGludGVncmF0ZXMgc2VhbWxlc3NseSB3aXRoIElvbmljJ3NcbiAqIGFjY29yZGlvbiBjb21wb25lbnRzIHRvIGNyZWF0ZSBleHBhbmRhYmxlL2NvbGxhcHNpYmxlIHNlY3Rpb25zIGZvciBvcmdhbml6aW5nIGZvcm1cbiAqIGNvbnRlbnQgYW5kIHJlbGF0ZWQgaW5mb3JtYXRpb24uIFRoZSBjb21wb25lbnQgaW50ZWxsaWdlbnRseSBkZXRlcm1pbmVzIGl0cyBpbml0aWFsXG4gKiBzdGF0ZSBiYXNlZCBvbiB0aGUgb3BlcmF0aW9uIHR5cGUsIG9wZW5pbmcgYXV0b21hdGljYWxseSBmb3IgUkVBRCBhbmQgREVMRVRFIG9wZXJhdGlvbnNcbiAqIHdoaWxlIHJlbWFpbmluZyBjbG9zZWQgZm9yIENSRUFURSBhbmQgVVBEQVRFIG9wZXJhdGlvbnMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYGh0bWxcbiAqIDwhLS0gQmFzaWMgdXNhZ2Ugd2l0aCBhdXRvbWF0aWMgc3RhdGUgbWFuYWdlbWVudCAtLT5cbiAqIDxuZ3gtZGVjYWYtZmllbGRzZXRcbiAqICAgbmFtZT1cIlBlcnNvbmFsIEluZm9ybWF0aW9uXCJcbiAqICAgW29wZXJhdGlvbl09XCJPcGVyYXRpb25LZXlzLlJFQURcIlxuICogICB0YXJnZXQ9XCJfc2VsZlwiPlxuICogICA8aW9uLWlucHV0IGxhYmVsPVwiTmFtZVwiIHBsYWNlaG9sZGVyPVwiRW50ZXIgbmFtZVwiPjwvaW9uLWlucHV0PlxuICogICA8aW9uLWlucHV0IGxhYmVsPVwiRW1haWxcIiB0eXBlPVwiZW1haWxcIiBwbGFjZWhvbGRlcj1cIkVudGVyIGVtYWlsXCI+PC9pb24taW5wdXQ+XG4gKiA8L25neC1kZWNhZi1maWVsZHNldD5cbiAqXG4gKiA8IS0tIEFkdmFuY2VkIHVzYWdlIHdpdGggY3VzdG9tIG9wZXJhdGlvbiAtLT5cbiAqIDxuZ3gtZGVjYWYtZmllbGRzZXRcbiAqICAgbmFtZT1cIkNvbnRhY3QgRGV0YWlsc1wiXG4gKiAgIFtvcGVyYXRpb25dPVwiY3VycmVudE9wZXJhdGlvblwiXG4gKiAgIHRhcmdldD1cIl9ibGFua1wiPlxuICogICA8IS0tIENvbXBsZXggZm9ybSBmaWVsZHMgLS0+XG4gKiA8L25neC1kZWNhZi1maWVsZHNldD5cbiAqIGBgYFxuICpcbiAqIEBtZXJtYWlkXG4gKiBzZXF1ZW5jZURpYWdyYW1cbiAqICAgcGFydGljaXBhbnQgVSBhcyBVc2VyXG4gKiAgIHBhcnRpY2lwYW50IEYgYXMgRmllbGRzZXRDb21wb25lbnRcbiAqICAgcGFydGljaXBhbnQgSSBhcyBJb25pYyBBY2NvcmRpb25cbiAqICAgcGFydGljaXBhbnQgRCBhcyBET01cbiAqXG4gKiAgIEYtPj5GOiBuZ0FmdGVyVmlld0luaXQoKVxuICogICBhbHQgb3BlcmF0aW9uIGlzIFJFQUQgb3IgREVMRVRFXG4gKiAgICAgRi0+PkY6IFNldCBpc09wZW4gPSB0cnVlXG4gKiAgICAgRi0+PkQ6IFF1ZXJ5IGFjY29yZGlvbiBlbGVtZW50XG4gKiAgICAgRi0+Pkk6IFNldCB2YWx1ZSBhdHRyaWJ1dGUgdG8gJ29wZW4nXG4gKiAgICAgRi0+PkY6IFRyaWdnZXIgY2hhbmdlIGRldGVjdGlvblxuICogICBlbmRcbiAqICAgVS0+Pkk6IENsaWNrIGFjY29yZGlvbiBoZWFkZXJcbiAqICAgSS0+PkY6IGhhbmRsZUNoYW5nZShldmVudClcbiAqICAgRi0+PkY6IFVwZGF0ZSBpc09wZW4gc3RhdGVcbiAqICAgRi0+Pkk6IFJlZmxlY3QgbmV3IHN0YXRlXG4gKlxuICogQG1lbWJlck9mIEZvckFuZ3VsYXJNb2R1bGVcbiAqL1xuQER5bmFtaWMoKVxuQENvbXBvbmVudCh7XG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIHNlbGVjdG9yOiAnbmd4LWRlY2FmLWZpZWxkc2V0JyxcbiAgdGVtcGxhdGVVcmw6ICcuL2ZpZWxkc2V0LmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vZmllbGRzZXQuY29tcG9uZW50LnNjc3MnXSxcbiAgc2NoZW1hczogW0NVU1RPTV9FTEVNRU5UU19TQ0hFTUFdLFxuICBpbXBvcnRzOiBbRm9yQW5ndWxhck1vZHVsZSwgSW9uQWNjb3JkaW9uR3JvdXAsIElvbkFjY29yZGlvbiwgSW9uSXRlbSwgQ29sbGFwc2FibGVEaXJlY3RpdmVdLFxufSlcbmV4cG9ydCBjbGFzcyBGaWVsZHNldENvbXBvbmVudCBpbXBsZW1lbnRzIEFmdGVyVmlld0luaXQge1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmVmZXJlbmNlIHRvIHRoZSBjb21wb25lbnQncyBuYXRpdmUgRE9NIGVsZW1lbnQuXG4gICAqIEBzdW1tYXJ5IFZpZXdDaGlsZCByZWZlcmVuY2UgdGhhdCBwcm92aWRlcyBkaXJlY3QgYWNjZXNzIHRvIHRoZSBjb21wb25lbnQncyByb290IERPTSBlbGVtZW50LlxuICAgKiBUaGlzIGlzIGVzc2VudGlhbCBmb3IgbWFuaXB1bGF0aW5nIHRoZSBJb25pYyBhY2NvcmRpb24gZ3JvdXAgYWZ0ZXIgdmlldyBpbml0aWFsaXphdGlvbixcbiAgICogcGFydGljdWxhcmx5IGZvciBzZXR0aW5nIHRoZSBpbml0aWFsIG9wZW4gc3RhdGUgcHJvZ3JhbW1hdGljYWxseS4gVGhlIHJlZmVyZW5jZSBpcyB1c2VkXG4gICAqIHRvIHF1ZXJ5IGFuZCBtb2RpZnkgYWNjb3JkaW9uIGF0dHJpYnV0ZXMgdGhhdCBjb250cm9sIHRoZSBjb21wb25lbnQncyBleHBhbmRlZCBzdGF0ZS5cbiAgICpcbiAgICogQHR5cGUge0VsZW1lbnRSZWZ9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgQFZpZXdDaGlsZCgnY29tcG9uZW50JywgeyBzdGF0aWM6IGZhbHNlLCByZWFkOiBFbGVtZW50UmVmIH0pXG4gIGNvbXBvbmVudCE6IEVsZW1lbnRSZWY7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBUaGUgZGlzcGxheSBuYW1lIG9yIHRpdGxlIG9mIHRoZSBmaWVsZHNldCBzZWN0aW9uLlxuICAgKiBAc3VtbWFyeSBTZXRzIHRoZSBsZWdlbmQgb3IgaGVhZGVyIHRleHQgdGhhdCBhcHBlYXJzIGluIHRoZSBhY2NvcmRpb24gaGVhZGVyLiBUaGlzIHRleHRcbiAgICogcHJvdmlkZXMgYSBjbGVhciBsYWJlbCBmb3IgdGhlIGNvbGxhcHNpYmxlIHNlY3Rpb24sIGhlbHBpbmcgdXNlcnMgdW5kZXJzdGFuZCB3aGF0IGNvbnRlbnRcbiAgICogaXMgY29udGFpbmVkIHdpdGhpbi4gVGhlIG5hbWUgaXMgZGlzcGxheWVkIHByb21pbmVudGx5IGFuZCBzZXJ2ZXMgYXMgdGhlIGNsaWNrYWJsZSBhcmVhXG4gICAqIGZvciBleHBhbmRpbmcvY29sbGFwc2luZyB0aGUgZmllbGRzZXQuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmd9XG4gICAqIEBkZWZhdWx0ICdDaGlsZCdcbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBuYW1lOiBzdHJpbmcgPSAnQ2hpbGQnO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVGhlIGN1cnJlbnQgQ1JVRCBvcGVyYXRpb24gY29udGV4dC5cbiAgICogQHN1bW1hcnkgRGV0ZXJtaW5lcyB0aGUgY29tcG9uZW50J3MgaW5pdGlhbCBiZWhhdmlvciBhbmQgc3RhdGUgYmFzZWQgb24gdGhlIGN1cnJlbnQgb3BlcmF0aW9uLlxuICAgKiBUaGlzIGlucHV0IGlzIGNydWNpYWwgZm9yIGF1dG8tc3RhdGUgbWFuYWdlbWVudDogUkVBRCBhbmQgREVMRVRFIG9wZXJhdGlvbnMgYXV0b21hdGljYWxseVxuICAgKiBvcGVuIHRoZSBmaWVsZHNldCB0byBzaG93IGNvbnRlbnQsIHdoaWxlIENSRUFURSBhbmQgVVBEQVRFIG9wZXJhdGlvbnMga2VlcCBpdCBjbG9zZWRcbiAgICogaW5pdGlhbGx5LiBUaGlzIHByb3ZpZGVzIGFuIGludHVpdGl2ZSB1c2VyIGV4cGVyaWVuY2UgYWxpZ25lZCB3aXRoIG9wZXJhdGlvbiBzZW1hbnRpY3MuXG4gICAqXG4gICAqIEB0eXBlIHtPcGVyYXRpb25LZXlzfVxuICAgKiBAZGVmYXVsdCBPcGVyYXRpb25LZXlzLlJFQURcbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvcGVyYXRpb246IE9wZXJhdGlvbktleXMgPSBPcGVyYXRpb25LZXlzLlJFQUQ7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBGb3JtIHRhcmdldCBhdHRyaWJ1dGUgZm9yIG5lc3RlZCBmb3JtIHN1Ym1pc3Npb25zLlxuICAgKiBAc3VtbWFyeSBTcGVjaWZpZXMgd2hlcmUgdG8gZGlzcGxheSB0aGUgcmVzcG9uc2UgYWZ0ZXIgc3VibWl0dGluZyBmb3JtcyBjb250YWluZWQgd2l0aGluXG4gICAqIHRoZSBmaWVsZHNldC4gVGhpcyBhdHRyaWJ1dGUgbWlycm9ycyB0aGUgSFRNTCBmb3JtIHRhcmdldCBiZWhhdmlvciwgYWxsb3dpbmcgY29udHJvbCBvdmVyXG4gICAqIHdoZXRoZXIgZm9ybSBzdWJtaXNzaW9ucyBvcGVuIGluIHRoZSBzYW1lIHdpbmRvdywgbmV3IHdpbmRvdywgb3Igc3BlY2lmaWMgZnJhbWUuIFVzZWZ1bFxuICAgKiBmb3IgY29tcGxleCBmb3JtIHdvcmtmbG93cyBhbmQgbXVsdGktc3RlcCBwcm9jZXNzZXMuXG4gICAqXG4gICAqIEB0eXBlIHtIVE1MRm9ybVRhcmdldH1cbiAgICogQGRlZmF1bHQgJ19zZWxmJ1xuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIHRhcmdldDogSFRNTEZvcm1UYXJnZXQgPSAnX3NlbGYnO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ3VycmVudCBzdGF0ZSBvZiB0aGUgYWNjb3JkaW9uIChleHBhbmRlZCBvciBjb2xsYXBzZWQpLlxuICAgKiBAc3VtbWFyeSBCb29sZWFuIGZsYWcgdGhhdCB0cmFja3Mgd2hldGhlciB0aGUgZmllbGRzZXQgYWNjb3JkaW9uIGlzIGN1cnJlbnRseSBvcGVuIG9yIGNsb3NlZC5cbiAgICogVGhpcyBwcm9wZXJ0eSBpcyBhdXRvbWF0aWNhbGx5IG1hbmFnZWQgYmFzZWQgb24gdXNlciBpbnRlcmFjdGlvbnMgYW5kIGluaXRpYWwgb3BlcmF0aW9uIHN0YXRlLlxuICAgKiBJdCBzZXJ2ZXMgYXMgdGhlIHNpbmdsZSBzb3VyY2Ugb2YgdHJ1dGggZm9yIHRoZSBjb21wb25lbnQncyB2aXNpYmlsaXR5IHN0YXRlIGFuZCBpcyB1c2VkXG4gICAqIHRvIGNvb3JkaW5hdGUgYmV0d2VlbiB1c2VyIGFjdGlvbnMgYW5kIHByb2dyYW1tYXRpYyBzdGF0ZSBjaGFuZ2VzLlxuICAgKlxuICAgKiBAdHlwZSB7Ym9vbGVhbn1cbiAgICogQGRlZmF1bHQgZmFsc2VcbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBpc09wZW46IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFJlZmVyZW5jZSB0byBDUlVEIG9wZXJhdGlvbiBjb25zdGFudHMgZm9yIHRlbXBsYXRlIHVzYWdlLlxuICAgKiBAc3VtbWFyeSBFeHBvc2VzIHRoZSBPcGVyYXRpb25LZXlzIGVudW0gdG8gdGhlIGNvbXBvbmVudCB0ZW1wbGF0ZSwgZW5hYmxpbmcgY29uZGl0aW9uYWxcbiAgICogcmVuZGVyaW5nIGFuZCBiZWhhdmlvciBiYXNlZCBvbiBvcGVyYXRpb24gdHlwZXMuIFRoaXMgcHJvdGVjdGVkIHJlYWRvbmx5IHByb3BlcnR5IGVuc3VyZXNcbiAgICogdGhhdCB0ZW1wbGF0ZSBsb2dpYyBjYW4gYWNjZXNzIG9wZXJhdGlvbiBjb25zdGFudHMgd2hpbGUgbWFpbnRhaW5pbmcgZW5jYXBzdWxhdGlvbiBhbmRcbiAgICogcHJldmVudGluZyBhY2NpZGVudGFsIG1vZGlmaWNhdGlvbiBvZiB0aGUgZW51bSB2YWx1ZXMuXG4gICAqXG4gICAqIEB0eXBlIHtDcnVkT3BlcmF0aW9uc31cbiAgICogQGRlZmF1bHQgT3BlcmF0aW9uS2V5cy5DUkVBVEVcbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBwcm90ZWN0ZWQgcmVhZG9ubHkgT3BlcmF0aW9uS2V5czogQ3J1ZE9wZXJhdGlvbnMgPSBPcGVyYXRpb25LZXlzLkNSRUFURTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEFuZ3VsYXIgY2hhbmdlIGRldGVjdGlvbiBzZXJ2aWNlLlxuICAgKiBAc3VtbWFyeSBJbmplY3RlZCBzZXJ2aWNlIHRoYXQgcHJvdmlkZXMgbWFudWFsIGNvbnRyb2wgb3ZlciBjaGFuZ2UgZGV0ZWN0aW9uIGN5Y2xlcy5cbiAgICogVGhpcyBpcyBlc3NlbnRpYWwgZm9yIGVuc3VyaW5nIHRoYXQgcHJvZ3JhbW1hdGljIERPTSBjaGFuZ2VzIChsaWtlIHNldHRpbmcgYWNjb3JkaW9uXG4gICAqIGF0dHJpYnV0ZXMpIGFyZSBwcm9wZXJseSByZWZsZWN0ZWQgaW4gdGhlIGNvbXBvbmVudCdzIHN0YXRlIGFuZCB0cmlnZ2VyIGFwcHJvcHJpYXRlXG4gICAqIHZpZXcgdXBkYXRlcyB3aGVuIG1vZGlmaWNhdGlvbnMgb2NjdXIgb3V0c2lkZSB0aGUgbm9ybWFsIEFuZ3VsYXIgY2hhbmdlIGRldGVjdGlvbiBmbG93LlxuICAgKlxuICAgKiBAcHJpdmF0ZVxuICAgKiBAdHlwZSB7Q2hhbmdlRGV0ZWN0b3JSZWZ9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgcHJpdmF0ZSBjaGFuZ2VEZXRlY3RvclJlZjogQ2hhbmdlRGV0ZWN0b3JSZWYgPSBpbmplY3QoQ2hhbmdlRGV0ZWN0b3JSZWYpO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSW5pdGlhbGl6ZXMgdGhlIGNvbXBvbmVudCBzdGF0ZSBhZnRlciB2aWV3IGFuZCBjaGlsZCBjb21wb25lbnRzIGFyZSByZW5kZXJlZC5cbiAgICogQHN1bW1hcnkgVGhpcyBsaWZlY3ljbGUgaG9vayBpbXBsZW1lbnRzIGludGVsbGlnZW50IGF1dG8tc3RhdGUgbWFuYWdlbWVudCBiYXNlZCBvbiB0aGUgY3VycmVudFxuICAgKiBDUlVEIG9wZXJhdGlvbi4gRm9yIFJFQUQgYW5kIERFTEVURSBvcGVyYXRpb25zLCB0aGUgZmllbGRzZXQgYXV0b21hdGljYWxseSBvcGVucyB0byBwcm92aWRlXG4gICAqIGltbWVkaWF0ZSBhY2Nlc3MgdG8gaW5mb3JtYXRpb24sIHdoaWxlIENSRUFURSBhbmQgVVBEQVRFIG9wZXJhdGlvbnMga2VlcCBpdCBjbG9zZWQgdG8gbWFpbnRhaW5cbiAgICogYSBjbGVhbiBpbml0aWFsIGludGVyZmFjZS4gVGhlIG1ldGhvZCBkaXJlY3RseSBtYW5pcHVsYXRlcyB0aGUgRE9NIHRvIGVuc3VyZSBwcm9wZXIgYWNjb3JkaW9uXG4gICAqIHN5bmNocm9uaXphdGlvbiBhbmQgdHJpZ2dlcnMgY2hhbmdlIGRldGVjdGlvbiB0byByZWZsZWN0IHRoZSBwcm9ncmFtbWF0aWMgc3RhdGUgY2hhbmdlcy5cbiAgICpcbiAgICogQG1lcm1haWRcbiAgICogc2VxdWVuY2VEaWFncmFtXG4gICAqICAgcGFydGljaXBhbnQgQSBhcyBBbmd1bGFyIExpZmVjeWNsZVxuICAgKiAgIHBhcnRpY2lwYW50IEYgYXMgRmllbGRzZXRDb21wb25lbnRcbiAgICogICBwYXJ0aWNpcGFudCBEIGFzIERPTVxuICAgKiAgIHBhcnRpY2lwYW50IEMgYXMgQ2hhbmdlRGV0ZWN0b3JcbiAgICpcbiAgICogICBBLT4+RjogbmdBZnRlclZpZXdJbml0KClcbiAgICogICBhbHQgb3BlcmF0aW9uIGlzIFJFQUQgb3IgREVMRVRFXG4gICAqICAgICBGLT4+RjogU2V0IGlzT3BlbiA9IHRydWVcbiAgICogICAgIEYtPj5EOiBRdWVyeSBpb24tYWNjb3JkaW9uLWdyb3VwIGVsZW1lbnRcbiAgICogICAgIGFsdCBhY2NvcmRpb24gZWxlbWVudCBleGlzdHNcbiAgICogICAgICAgRi0+PkQ6IFNldCB2YWx1ZSBhdHRyaWJ1dGUgdG8gJ29wZW4nXG4gICAqICAgICBlbmRcbiAgICogICBlbmRcbiAgICogICBGLT4+QzogZGV0ZWN0Q2hhbmdlcygpXG4gICAqICAgQy0+PkY6IFVwZGF0ZSB2aWV3IHdpdGggbmV3IHN0YXRlXG4gICAqXG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIG5nQWZ0ZXJWaWV3SW5pdCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5vcGVyYXRpb24gPT09IE9wZXJhdGlvbktleXMuUkVBRCB8fCB0aGlzLm9wZXJhdGlvbiA9PT0gT3BlcmF0aW9uS2V5cy5ERUxFVEUpIHtcbiAgICAgIHRoaXMuaXNPcGVuID0gdHJ1ZTtcbiAgICAgIGNvbnN0IGFjY29yZGlvbkVsZW1lbnQgPSB0aGlzLmNvbXBvbmVudD8ubmF0aXZlRWxlbWVudC5xdWVyeVNlbGVjdG9yKCdpb24tYWNjb3JkaW9uLWdyb3VwJyk7XG4gICAgICBpZihhY2NvcmRpb25FbGVtZW50KVxuICAgICAgICBhY2NvcmRpb25FbGVtZW50LnNldEF0dHJpYnV0ZSgndmFsdWUnLCAnb3BlbicpO1xuICAgIH1cbiAgICB0aGlzLmNoYW5nZURldGVjdG9yUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSGFuZGxlcyBhY2NvcmRpb24gc3RhdGUgY2hhbmdlIGV2ZW50cyBmcm9tIHVzZXIgaW50ZXJhY3Rpb25zLlxuICAgKiBAc3VtbWFyeSBQcm9jZXNzZXMgQ3VzdG9tRXZlbnQgb2JqZWN0cyB0cmlnZ2VyZWQgd2hlbiB1c2VycyBleHBhbmQgb3IgY29sbGFwc2UgdGhlIGFjY29yZGlvbi5cbiAgICogVGhpcyBtZXRob2QgZXh0cmFjdHMgdGhlIG5ldyBzdGF0ZSBmcm9tIHRoZSBldmVudCBkZXRhaWxzIGFuZCB1cGRhdGVzIHRoZSBjb21wb25lbnQnc1xuICAgKiBpbnRlcm5hbCBzdGF0ZSBhY2NvcmRpbmdseS4gSXQgc3BlY2lmaWNhbGx5IGxpc3RlbnMgZm9yIElPTi1BQ0NPUkRJT04tR1JPVVAgZXZlbnRzIHRvXG4gICAqIGVuc3VyZSBwcm9wZXIgZXZlbnQgc291cmNlIHZhbGlkYXRpb24gYW5kIHByZXZlbnQgaGFuZGxpbmcgb2YgdW5yZWxhdGVkIGV2ZW50cy5cbiAgICpcbiAgICogQHBhcmFtIHtDdXN0b21FdmVudH0gZXZlbnQgLSBUaGUgZXZlbnQgb2JqZWN0IGNvbnRhaW5pbmcgYWNjb3JkaW9uIHN0YXRlIGNoYW5nZSBkZXRhaWxzXG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKlxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBVIGFzIFVzZXJcbiAgICogICBwYXJ0aWNpcGFudCBJIGFzIElvbi1BY2NvcmRpb25cbiAgICogICBwYXJ0aWNpcGFudCBGIGFzIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqXG4gICAqICAgVS0+Pkk6IENsaWNrIGFjY29yZGlvbiBoZWFkZXJcbiAgICogICBJLT4+RjogaGFuZGxlQ2hhbmdlKEN1c3RvbUV2ZW50KVxuICAgKiAgIEYtPj5GOiBFeHRyYWN0IHRhcmdldCBhbmQgZGV0YWlsIGZyb20gZXZlbnRcbiAgICogICBGLT4+RjogVmFsaWRhdGUgdGFyZ2V0IGlzIElPTi1BQ0NPUkRJT04tR1JPVVBcbiAgICogICBhbHQgdmFsaWQgdGFyZ2V0XG4gICAqICAgICBGLT4+RjogVXBkYXRlIGlzT3BlbiA9ICEhdmFsdWVcbiAgICogICBlbmRcbiAgICogICBGLT4+STogUmVmbGVjdCB1cGRhdGVkIHN0YXRlXG4gICAqXG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgaGFuZGxlQ2hhbmdlKGV2ZW50OiBDdXN0b21FdmVudCk6IHZvaWQge1xuICAgIGNvbnN0IHsgdGFyZ2V0LCBkZXRhaWwgfSA9IGV2ZW50O1xuICAgIGNvbnN0IHsgdmFsdWUgfSA9IGRldGFpbDtcbiAgICBpZiAoKHRhcmdldCBhcyBIVE1MSW9uQWNjb3JkaW9uR3JvdXBFbGVtZW50KS50YWdOYW1lID09PSAnSU9OLUFDQ09SRElPTi1HUk9VUCcpXG4gICAgICB0aGlzLmlzT3BlbiA9ICEhdmFsdWU7XG4gIH1cbn1cbiIsIjxmaWVsZHNldCBbY2xhc3NdPVwiJ2RjZi1maWVsZHNldCAnICsgb3BlcmF0aW9uXCIgI2NvbXBvbmVudD5cbiAgPGlvbi1hY2NvcmRpb24tZ3JvdXAgKGlvbkNoYW5nZSk9XCJoYW5kbGVDaGFuZ2UoJGV2ZW50KVwiPlxuICAgIDxpb24tYWNjb3JkaW9uIHZhbHVlPVwib3BlblwiPlxuICAgICAgPGlvbi1pdGVtIHNsb3Q9XCJoZWFkZXJcIiA+XG4gICAgICAgIDxsZWdlbmQ+e3sgbmFtZSB9fTwvbGVnZW5kPlxuICAgICAgPC9pb24taXRlbT5cbiAgICAgIDxkaXYgc2xvdD1cImNvbnRlbnRcIiBbYXR0ci5hcmlhLWhpZGRlbl09XCIhaXNPcGVuXCIgZGVjYWZDb2xsYXBzYWJsZT5cbiAgICAgICAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuICAgICAgPC9kaXY+XG4gICAgPC9pb24tYWNjb3JkaW9uPlxuICA8L2lvbi1hY2NvcmRpb24tZ3JvdXA+XG48L2ZpZWxkc2V0PlxuIl19
664
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmllbGRzZXQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL2ZpZWxkc2V0L2ZpZWxkc2V0LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9maWVsZHNldC9maWVsZHNldC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQ0EsT0FBTyxFQUFpQixpQkFBaUIsRUFBRSxTQUFTLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBRSxTQUFTLEVBQUUsU0FBUyxFQUFVLE1BQU0sZUFBZSxDQUFDO0FBQ2pKLE9BQU8sRUFBRSxPQUFPLEVBQUUsY0FBYyxFQUF5QyxNQUFNLGNBQWMsQ0FBQztBQUM5RixPQUFPLEVBQWtCLGFBQWEsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQ3hFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQzVELE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHdDQUF3QyxDQUFDO0FBQzlFLE9BQU8sRUFBRSxZQUFZLEVBQUUsaUJBQWlCLEVBQUUsU0FBUyxFQUFFLE9BQU8sRUFBRSxRQUFRLEVBQUUsT0FBTyxFQUEwQixlQUFlLEVBQUUsVUFBVSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDeEssT0FBTyxFQUFFLFdBQVcsRUFBRSxtQkFBbUIsRUFBRSxVQUFVLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDakcsT0FBTyxFQUFFLFNBQVMsRUFBMEIsTUFBTSxnQkFBZ0IsQ0FBQztBQUNuRSxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDaEQsT0FBTyxFQUFFLGtCQUFrQixFQUFFLGFBQWEsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ25FLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHFCQUFxQixDQUFD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QUFFLE9BQU8sME5BQUUsUUFBUSw2RkFBRSxVQUFVLHdEQUFhLGVBQWU7O0FBR3BILGlCQUFpQjtJQVY3QixPQUFPLEVBQUU7O0dBVUcsaUJBQWlCLENBd3JCN0I7OzRGQXhyQlksaUJBQWlCO2tCQVQ3QixTQUFTO2lDQUNJLElBQUksWUFDTixvQkFBb0IsV0FHckIsQ0FBQyxzQkFBc0IsQ0FBQyxXQUN4QixDQUFDLGdCQUFnQixFQUFFLGlCQUFpQixFQUFFLFlBQVksRUFBRSxPQUFPLEVBQUUsT0FBTyxFQUFFLFFBQVEsRUFBRSxVQUFVLEVBQUUsU0FBUyxFQUFFLGVBQWUsRUFBRSxvQkFBb0IsQ0FBQyxRQUNoSixFQUFDLFdBQVcsRUFBRSxZQUFZLEVBQUM7d0RBaUJqQyxrQkFBa0I7c0JBRGpCLFNBQVM7dUJBQUMsb0JBQW9CLEVBQUUsRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFO2dCQWdCbEQsSUFBSTtzQkFESCxLQUFLO2dCQWVOLE9BQU87c0JBRE4sS0FBSztnQkFlRyxHQUFHO3NCQURYLEtBQUs7Z0JBY04sV0FBVztzQkFEVixLQUFLO2dCQTBCTixTQUFTO3NCQURSLEtBQUs7Z0JBWU4sU0FBUztzQkFEUixLQUFLO2dCQVlOLEtBQUs7c0JBREosS0FBSztnQkFZTixXQUFXO3NCQURWLEtBQUs7Z0JBZU4sTUFBTTtzQkFETCxLQUFLO2dCQWNOLFFBQVE7c0JBRFAsS0FBSztnQkFhTixLQUFLO3NCQURKLEtBQUs7Z0JBYU4sUUFBUTtzQkFEUCxLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiXG5pbXBvcnQgeyBBZnRlclZpZXdJbml0LCBDaGFuZ2VEZXRlY3RvclJlZiwgQ29tcG9uZW50LCBDVVNUT01fRUxFTUVOVFNfU0NIRU1BLCBpbmplY3QsIElucHV0LCBWaWV3Q2hpbGQsIFJlbmRlcmVyMiwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBEeW5hbWljLCBFdmVudENvbnN0YW50cywgSGFuZGxlckxpa2UsIEhUTUxGb3JtVGFyZ2V0LCBLZXlWYWx1ZSB9IGZyb20gJy4uLy4uL2VuZ2luZSc7XG5pbXBvcnQgeyBDcnVkT3BlcmF0aW9ucywgT3BlcmF0aW9uS2V5cyB9IGZyb20gJ0BkZWNhZi10cy9kYi1kZWNvcmF0b3JzJztcbmltcG9ydCB7IEZvckFuZ3VsYXJNb2R1bGUgfSBmcm9tICcuLi8uLi9mb3ItYW5ndWxhci5tb2R1bGUnO1xuaW1wb3J0IHsgQ29sbGFwc2FibGVEaXJlY3RpdmUgfSBmcm9tICcuLi8uLi9kaXJlY3RpdmVzL2NvbGxhcHNhYmxlLmRpcmVjdGl2ZSc7XG5pbXBvcnQgeyBJb25BY2NvcmRpb24sIElvbkFjY29yZGlvbkdyb3VwLCBJb25CdXR0b24sIElvbkl0ZW0sIElvbkxhYmVsLCBJb25MaXN0LCBJdGVtUmVvcmRlckV2ZW50RGV0YWlsLCBJb25SZW9yZGVyR3JvdXAsIElvblJlb3JkZXIgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IGNsZWFuU3BhY2VzLCBnZW5lcmF0ZVJhbmRvbVZhbHVlLCBpdGVtTWFwcGVyLCB3aW5kb3dFdmVudEVtaXR0ZXIgfSBmcm9tICcuLi8uLi9oZWxwZXJzJztcbmltcG9ydCB7IEZvcm1BcnJheSwgRm9ybUNvbnRyb2wsIEZvcm1Hcm91cCB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7IE5neEJhc2VDb21wb25lbnQgfSBmcm9tICcuLi8uLi9lbmdpbmUnO1xuaW1wb3J0IHsgYWxlcnRDaXJjbGVPdXRsaW5lLCBjcmVhdGVPdXRsaW5lIH0gZnJvbSAnaW9uaWNvbnMvaWNvbnMnO1xuaW1wb3J0IHsgVHJhbnNsYXRlU2VydmljZSB9IGZyb20gJ0BuZ3gtdHJhbnNsYXRlL2NvcmUnO1xuaW1wb3J0IHsgSUZpZWxkU2V0SXRlbSwgSUZpZWxkU2V0VmFsaWRhdGlvbkV2ZW50IH0gZnJvbSAnLi4vLi4vZW5naW5lL2ludGVyZmFjZXMnO1xuXG5cblxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBEeW5hbWljIGZpZWxkc2V0IGNvbXBvbmVudCB3aXRoIGNvbGxhcHNpYmxlIGFjY29yZGlvbiBmdW5jdGlvbmFsaXR5LlxuICogQHN1bW1hcnkgVGhpcyBjb21wb25lbnQgcHJvdmlkZXMgYSBzb3BoaXN0aWNhdGVkIGZpZWxkc2V0IGNvbnRhaW5lciB0aGF0IGF1dG9tYXRpY2FsbHlcbiAqIGFkYXB0cyBpdHMgYmVoYXZpb3IgYmFzZWQgb24gQ1JVRCBvcGVyYXRpb25zLiBJdCBpbnRlZ3JhdGVzIHNlYW1sZXNzbHkgd2l0aCBJb25pYydzXG4gKiBhY2NvcmRpb24gY29tcG9uZW50cyB0byBjcmVhdGUgZXhwYW5kYWJsZS9jb2xsYXBzaWJsZSBzZWN0aW9ucyBmb3Igb3JnYW5pemluZyBmb3JtXG4gKiBjb250ZW50IGFuZCByZWxhdGVkIGluZm9ybWF0aW9uLiBUaGUgY29tcG9uZW50IGludGVsbGlnZW50bHkgZGV0ZXJtaW5lcyBpdHMgaW5pdGlhbFxuICogc3RhdGUgYmFzZWQgb24gdGhlIG9wZXJhdGlvbiB0eXBlLCBvcGVuaW5nIGF1dG9tYXRpY2FsbHkgZm9yIFJFQUQgYW5kIERFTEVURSBvcGVyYXRpb25zXG4gKiB3aGlsZSByZW1haW5pbmcgY2xvc2VkIGZvciBDUkVBVEUgYW5kIFVQREFURSBvcGVyYXRpb25zLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGBodG1sXG4gKiA8IS0tIEJhc2ljIHVzYWdlIHdpdGggYXV0b21hdGljIHN0YXRlIG1hbmFnZW1lbnQgLS0+XG4gKiA8bmd4LWRlY2FmLWZpZWxkc2V0XG4gKiAgIG5hbWU9XCJQZXJzb25hbCBJbmZvcm1hdGlvblwiXG4gKiAgIFtvcGVyYXRpb25dPVwiT3BlcmF0aW9uS2V5cy5SRUFEXCJcbiAqICAgdGFyZ2V0PVwiX3NlbGZcIj5cbiAqICAgPGlvbi1pbnB1dCBsYWJlbD1cIk5hbWVcIiBwbGFjZWhvbGRlcj1cIkVudGVyIG5hbWVcIj48L2lvbi1pbnB1dD5cbiAqICAgPGlvbi1pbnB1dCBsYWJlbD1cIkVtYWlsXCIgdHlwZT1cImVtYWlsXCIgcGxhY2Vob2xkZXI9XCJFbnRlciBlbWFpbFwiPjwvaW9uLWlucHV0PlxuICogPC9uZ3gtZGVjYWYtZmllbGRzZXQ+XG4gKlxuICogPCEtLSBBZHZhbmNlZCB1c2FnZSB3aXRoIGN1c3RvbSBvcGVyYXRpb24gLS0+XG4gKiA8bmd4LWRlY2FmLWZpZWxkc2V0XG4gKiAgIG5hbWU9XCJDb250YWN0IERldGFpbHNcIlxuICogICBbb3BlcmF0aW9uXT1cImN1cnJlbnRPcGVyYXRpb25cIlxuICogICB0YXJnZXQ9XCJfYmxhbmtcIj5cbiAqICAgPCEtLSBDb21wbGV4IGZvcm0gZmllbGRzIC0tPlxuICogPC9uZ3gtZGVjYWYtZmllbGRzZXQ+XG4gKiBgYGBcbiAqXG4gKiBAbWVybWFpZFxuICogc2VxdWVuY2VEaWFncmFtXG4gKiAgIHBhcnRpY2lwYW50IFUgYXMgVXNlclxuICogICBwYXJ0aWNpcGFudCBGIGFzIEZpZWxkc2V0Q29tcG9uZW50XG4gKiAgIHBhcnRpY2lwYW50IEkgYXMgSW9uaWMgQWNjb3JkaW9uXG4gKiAgIHBhcnRpY2lwYW50IEQgYXMgRE9NXG4gKlxuICogICBGLT4+RjogbmdBZnRlclZpZXdJbml0KClcbiAqICAgYWx0IG9wZXJhdGlvbiBpcyBSRUFEIG9yIERFTEVURVxuICogICAgIEYtPj5GOiBTZXQgaXNPcGVuID0gdHJ1ZVxuICogICAgIEYtPj5EOiBRdWVyeSBhY2NvcmRpb24gZWxlbWVudFxuICogICAgIEYtPj5JOiBTZXQgdmFsdWUgYXR0cmlidXRlIHRvICdvcGVuJ1xuICogICAgIEYtPj5GOiBUcmlnZ2VyIGNoYW5nZSBkZXRlY3Rpb25cbiAqICAgZW5kXG4gKiAgIFUtPj5JOiBDbGljayBhY2NvcmRpb24gaGVhZGVyXG4gKiAgIEktPj5GOiBoYW5kbGVDaGFuZ2UoZXZlbnQpXG4gKiAgIEYtPj5GOiBVcGRhdGUgaXNPcGVuIHN0YXRlXG4gKiAgIEYtPj5JOiBSZWZsZWN0IG5ldyBzdGF0ZVxuICpcbiAqIEBtZW1iZXJPZiBGb3JBbmd1bGFyTW9kdWxlXG4gKi9cbkBEeW5hbWljKClcbkBDb21wb25lbnQoe1xuICBzdGFuZGFsb25lOiB0cnVlLFxuICBzZWxlY3RvcjogJ25neC1kZWNhZi1maWVsZHNldCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9maWVsZHNldC5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL2ZpZWxkc2V0LmNvbXBvbmVudC5zY3NzJ10sXG4gIHNjaGVtYXM6IFtDVVNUT01fRUxFTUVOVFNfU0NIRU1BXSxcbiAgaW1wb3J0czogW0ZvckFuZ3VsYXJNb2R1bGUsIElvbkFjY29yZGlvbkdyb3VwLCBJb25BY2NvcmRpb24sIElvbkxpc3QsIElvbkl0ZW0sIElvbkxhYmVsLCBJb25SZW9yZGVyLCBJb25CdXR0b24sIElvblJlb3JkZXJHcm91cCwgQ29sbGFwc2FibGVEaXJlY3RpdmVdLFxuICBob3N0OiB7J1thdHRyLmlkXSc6ICdvdmVycmlvZGUgJ30sXG59KVxuZXhwb3J0IGNsYXNzIEZpZWxkc2V0Q29tcG9uZW50IGV4dGVuZHMgTmd4QmFzZUNvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCwgQWZ0ZXJWaWV3SW5pdCB7XG5cblxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmVmZXJlbmNlIHRvIHRoZSBpb24tYWNjb3JkaW9uLWdyb3VwIGNvbXBvbmVudCBmb3IgcHJvZ3JhbW1hdGljIGNvbnRyb2wuXG4gICAqIEBzdW1tYXJ5IFZpZXdDaGlsZCByZWZlcmVuY2UgdGhhdCBwcm92aWRlcyBkaXJlY3QgYWNjZXNzIHRvIHRoZSBJb25pYyBhY2NvcmRpb24gZ3JvdXAgY29tcG9uZW50LlxuICAgKiBUaGlzIGVuYWJsZXMgcHJvZ3JhbW1hdGljIGNvbnRyb2wgb3ZlciB0aGUgYWNjb3JkaW9uJ3MgZXhwYW5kL2NvbGxhcHNlIHN0YXRlLCBhbGxvd2luZ1xuICAgKiB0aGUgY29tcG9uZW50IHRvIG9wZW4vY2xvc2UgdGhlIGFjY29yZGlvbiBiYXNlZCBvbiB2YWxpZGF0aW9uIGVycm9ycywgQ1JVRCBvcGVyYXRpb25zLFxuICAgKiBvciBvdGhlciBidXNpbmVzcyBsb2dpYyByZXF1aXJlbWVudHMuXG4gICAqXG4gICAqIEB0eXBlIHtJb25BY2NvcmRpb25Hcm91cH1cbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBAVmlld0NoaWxkKCdhY2NvcmRpb25Db21wb25lbnQnLCB7IHN0YXRpYzogZmFsc2UgfSlcbiAgYWNjb3JkaW9uQ29tcG9uZW50ITogSW9uQWNjb3JkaW9uR3JvdXA7XG5cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBkaXNwbGF5IG5hbWUgb3IgdGl0bGUgb2YgdGhlIGZpZWxkc2V0IHNlY3Rpb24uXG4gICAqIEBzdW1tYXJ5IFNldHMgdGhlIGxlZ2VuZCBvciBoZWFkZXIgdGV4dCB0aGF0IGFwcGVhcnMgaW4gdGhlIGFjY29yZGlvbiBoZWFkZXIuIFRoaXMgdGV4dFxuICAgKiBwcm92aWRlcyBhIGNsZWFyIGxhYmVsIGZvciB0aGUgY29sbGFwc2libGUgc2VjdGlvbiwgaGVscGluZyB1c2VycyB1bmRlcnN0YW5kIHdoYXQgY29udGVudFxuICAgKiBpcyBjb250YWluZWQgd2l0aGluLiBUaGUgbmFtZSBpcyBkaXNwbGF5ZWQgcHJvbWluZW50bHkgYW5kIHNlcnZlcyBhcyB0aGUgY2xpY2thYmxlIGFyZWFcbiAgICogZm9yIGV4cGFuZGluZy9jb2xsYXBzaW5nIHRoZSBmaWVsZHNldC5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZ31cbiAgICogQGRlZmF1bHQgJ0NoaWxkJ1xuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG5hbWU6IHN0cmluZyA9ICdDaGlsZCc7XG5cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBwYXJlbnQgY29tcG9uZW50IGlkZW50aWZpZXIgZm9yIGhpZXJhcmNoaWNhbCBmaWVsZHNldCByZWxhdGlvbnNoaXBzLlxuICAgKiBAc3VtbWFyeSBTcGVjaWZpZXMgdGhlIHBhcmVudCBjb21wb25lbnQgbmFtZSB0aGF0IHRoaXMgZmllbGRzZXQgYmVsb25ncyB0byBpbiBhIGhpZXJhcmNoaWNhbFxuICAgKiBmb3JtIHN0cnVjdHVyZS4gVGhpcyBwcm9wZXJ0eSBpcyB1c2VkIGZvciBldmVudCBidWJibGluZyBhbmQgZXN0YWJsaXNoaW5nIHBhcmVudC1jaGlsZFxuICAgKiByZWxhdGlvbnNoaXBzIGJldHdlZW4gZmllbGRzZXRzIGluIGNvbXBsZXggZm9ybXMgd2l0aCBuZXN0ZWQgc3RydWN0dXJlcy5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZ31cbiAgICogQGRlZmF1bHQgJ0NoaWxkJ1xuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIGNoaWxkT2Y6IHN0cmluZyA9ICdDaGlsZCc7XG5cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBwYXJlbnQgY29tcG9uZW50IGlkZW50aWZpZXIgZm9yIGhpZXJhcmNoaWNhbCBmaWVsZHNldCByZWxhdGlvbnNoaXBzLlxuICAgKiBAc3VtbWFyeSBTcGVjaWZpZXMgdGhlIHBhcmVudCBjb21wb25lbnQgbmFtZSB0aGF0IHRoaXMgZmllbGRzZXQgYmVsb25ncyB0byBpbiBhIGhpZXJhcmNoaWNhbFxuICAgKiBmb3JtIHN0cnVjdHVyZS4gVGhpcyBwcm9wZXJ0eSBpcyB1c2VkIGZvciBldmVudCBidWJibGluZyBhbmQgZXN0YWJsaXNoaW5nIHBhcmVudC1jaGlsZFxuICAgKiByZWxhdGlvbnNoaXBzIGJldHdlZW4gZmllbGRzZXRzIGluIGNvbXBsZXggZm9ybXMgd2l0aCBuZXN0ZWQgc3RydWN0dXJlcy5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZ31cbiAgICogQGRlZmF1bHQgJ0NoaWxkJ1xuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIHVpZDogc3RyaW5nID0gZ2VuZXJhdGVSYW5kb21WYWx1ZSgxMik7XG5cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEN1c3RvbSB0eXBlIGRlZmluaXRpb25zIGZvciBzcGVjaWFsaXplZCBmaWVsZHNldCBiZWhhdmlvci5cbiAgICogQHN1bW1hcnkgRGVmaW5lcyBjdXN0b20gZGF0YSB0eXBlcyBvciB2YWxpZGF0aW9uIHJ1bGVzIHRoYXQgc2hvdWxkIGJlIGFwcGxpZWQgdG8gdGhpcyBmaWVsZHNldC5cbiAgICogQ2FuIGJlIGEgc2luZ2xlIHR5cGUgc3RyaW5nIG9yIGFycmF5IG9mIHR5cGVzIHRoYXQgZGV0ZXJtaW5lIGhvdyB0aGUgZmllbGRzZXQgaGFuZGxlc1xuICAgKiBkYXRhIHZhbGlkYXRpb24sIGZvcm1hdHRpbmcsIGFuZCBkaXNwbGF5IGJlaGF2aW9yIGZvciBzcGVjaWFsaXplZCB1c2UgY2FzZXMuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmcgfCBzdHJpbmdbXX1cbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBjdXN0b21UeXBlcyE6IHN0cmluZyB8IHN0cmluZ1tdO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVGhlIGN1cnJlbnQgQ1JVRCBvcGVyYXRpb24gY29udGV4dC5cbiAgICogQHN1bW1hcnkgRGV0ZXJtaW5lcyB0aGUgY29tcG9uZW50J3MgaW5pdGlhbCBiZWhhdmlvciBhbmQgc3RhdGUgYmFzZWQgb24gdGhlIGN1cnJlbnQgb3BlcmF0aW9uLlxuICAgKiBUaGlzIGlucHV0IGlzIGNydWNpYWwgZm9yIGF1dG8tc3RhdGUgbWFuYWdlbWVudDogUkVBRCBhbmQgREVMRVRFIG9wZXJhdGlvbnMgYXV0b21hdGljYWxseVxuICAgKiBvcGVuIHRoZSBmaWVsZHNldCB0byBzaG93IGNvbnRlbnQsIHdoaWxlIENSRUFURSBhbmQgVVBEQVRFIG9wZXJhdGlvbnMga2VlcCBpdCBjbG9zZWRcbiAgICogaW5pdGlhbGx5LiBUaGlzIHByb3ZpZGVzIGFuIGludHVpdGl2ZSB1c2VyIGV4cGVyaWVuY2UgYWxpZ25lZCB3aXRoIG9wZXJhdGlvbiBzZW1hbnRpY3MuXG4gICAqXG4gICAqIEB0eXBlIHtPcGVyYXRpb25LZXlzfVxuICAgKiBAZGVmYXVsdCBPcGVyYXRpb25LZXlzLlJFQURcbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBDUlVEIG9wZXJhdGlvbiB0eXBlIGZvciB0aGUgY3VycmVudCBmaWVsZHNldCBjb250ZXh0LlxuICAgKiBAc3VtbWFyeSBEZXRlcm1pbmVzIHRoZSBjb21wb25lbnQncyBpbml0aWFsIGJlaGF2aW9yIGFuZCBzdGF0ZSBiYXNlZCBvbiB0aGUgY3VycmVudCBvcGVyYXRpb24uXG4gICAqIFRoaXMgaW5wdXQgaXMgY3J1Y2lhbCBmb3IgYXV0by1zdGF0ZSBtYW5hZ2VtZW50OiBSRUFEIGFuZCBERUxFVEUgb3BlcmF0aW9ucyBhdXRvbWF0aWNhbGx5XG4gICAqIG9wZW4gdGhlIGZpZWxkc2V0IHRvIHNob3cgY29udGVudCwgd2hpbGUgQ1JFQVRFIGFuZCBVUERBVEUgb3BlcmF0aW9ucyBrZWVwIGl0IGNsb3NlZFxuICAgKiBpbml0aWFsbHkuIFRoaXMgcHJvdmlkZXMgYW4gaW50dWl0aXZlIHVzZXIgZXhwZXJpZW5jZSBhbGlnbmVkIHdpdGggb3BlcmF0aW9uIHNlbWFudGljcy5cbiAgICpcbiAgICogQHR5cGUge09wZXJhdGlvbktleXN9XG4gICAqIEBkZWZhdWx0IE9wZXJhdGlvbktleXMuUkVBRFxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG9wZXJhdGlvbjogT3BlcmF0aW9uS2V5cyA9IE9wZXJhdGlvbktleXMuUkVBRDtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFJlYWN0aXZlIGZvcm0gZ3JvdXAgYXNzb2NpYXRlZCB3aXRoIHRoaXMgZmllbGRzZXQuXG4gICAqIEBzdW1tYXJ5IFRoZSBGb3JtR3JvdXAgaW5zdGFuY2UgdGhhdCBjb250YWlucyBhbGwgZm9ybSBjb250cm9scyB3aXRoaW4gdGhpcyBmaWVsZHNldC5cbiAgICogVXNlZCBmb3IgZm9ybSB2YWxpZGF0aW9uLCB2YWx1ZSBtYW5hZ2VtZW50LCBhbmQgaW50ZWdyYXRpb24gd2l0aCBBbmd1bGFyJ3MgcmVhY3RpdmUgZm9ybXMuXG4gICAqXG4gICAqIEB0eXBlIHtGb3JtR3JvdXB9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgZm9ybUdyb3VwITogIEZvcm1BcnJheTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFByaW1hcnkgdGl0bGUgdGV4dCBmb3IgdGhlIGZpZWxkc2V0IGNvbnRlbnQuXG4gICAqIEBzdW1tYXJ5IERpc3BsYXkgdGl0bGUgdXNlZCBmb3IgZmllbGRzZXQgaWRlbnRpZmljYXRpb24gYW5kIGNvbnRlbnQgb3JnYW5pemF0aW9uLlxuICAgKiBQcm92aWRlcyBzZW1hbnRpYyBtZWFuaW5nIHRvIHRoZSBncm91cGVkIGZvcm0gZmllbGRzLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIHRpdGxlITogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gU2Vjb25kYXJ5IGRlc2NyaXB0aXZlIHRleHQgZm9yIHRoZSBmaWVsZHNldC5cbiAgICogQHN1bW1hcnkgQWRkaXRpb25hbCBpbmZvcm1hdGlvbiB0aGF0IHByb3ZpZGVzIGNvbnRleHQgb3IgaW5zdHJ1Y3Rpb25zXG4gICAqIHJlbGF0ZWQgdG8gdGhlIGZpZWxkc2V0IGNvbnRlbnQgYW5kIHB1cnBvc2UuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmd9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgZGVzY3JpcHRpb24hOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBGb3JtIHRhcmdldCBhdHRyaWJ1dGUgZm9yIG5lc3RlZCBmb3JtIHN1Ym1pc3Npb25zLlxuICAgKiBAc3VtbWFyeSBTcGVjaWZpZXMgd2hlcmUgdG8gZGlzcGxheSB0aGUgcmVzcG9uc2UgYWZ0ZXIgc3VibWl0dGluZyBmb3JtcyBjb250YWluZWQgd2l0aGluXG4gICAqIHRoZSBmaWVsZHNldC4gVGhpcyBhdHRyaWJ1dGUgbWlycm9ycyB0aGUgSFRNTCBmb3JtIHRhcmdldCBiZWhhdmlvciwgYWxsb3dpbmcgY29udHJvbCBvdmVyXG4gICAqIHdoZXRoZXIgZm9ybSBzdWJtaXNzaW9ucyBvcGVuIGluIHRoZSBzYW1lIHdpbmRvdywgbmV3IHdpbmRvdywgb3Igc3BlY2lmaWMgZnJhbWUuIFVzZWZ1bFxuICAgKiBmb3IgY29tcGxleCBmb3JtIHdvcmtmbG93cyBhbmQgbXVsdGktc3RlcCBwcm9jZXNzZXMuXG4gICAqXG4gICAqIEB0eXBlIHtIVE1MRm9ybVRhcmdldH1cbiAgICogQGRlZmF1bHQgJ19zZWxmJ1xuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIHRhcmdldDogSFRNTEZvcm1UYXJnZXQgPSAnX3NlbGYnO1xuXG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBFbmFibGVzIG11bHRpcGxlIGl0ZW0gbWFuYWdlbWVudCB3aXRoaW4gdGhlIGZpZWxkc2V0LlxuICAgKiBAc3VtbWFyeSBCb29sZWFuIGZsYWcgdGhhdCBkZXRlcm1pbmVzIGlmIHRoZSBmaWVsZHNldCBzdXBwb3J0cyBhZGRpbmcgbXVsdGlwbGUgdmFsdWVzLlxuICAgKiBXaGVuIHRydWUsIGRpc3BsYXlzIGEgcmVvcmRlcmFibGUgbGlzdCBvZiBpdGVtcyB3aXRoIGFkZC9yZW1vdmUgZnVuY3Rpb25hbGl0eS5cbiAgICpcbiAgICogQHR5cGUge2Jvb2xlYW59XG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgbXVsdGlwbGU6IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEFycmF5IG9mIHJhdyB2YWx1ZXMgc3RvcmVkIGluIHRoZSBmaWVsZHNldC5cbiAgICogQHN1bW1hcnkgQ29udGFpbnMgdGhlIGFjdHVhbCBkYXRhIHZhbHVlcyB0aGF0IGhhdmUgYmVlbiBhZGRlZCB0byB0aGUgZmllbGRzZXQuXG4gICAqIFRoaXMgaXMgdGhlIHNvdXJjZSBvZiB0cnV0aCBmb3IgdGhlIGZpZWxkc2V0J3MgZGF0YSBzdGF0ZS5cbiAgICpcbiAgICogQHR5cGUge0tleVZhbHVlW119XG4gICAqIEBkZWZhdWx0IFtdXG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgdmFsdWU6IEtleVZhbHVlW10gPSBbXTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEV2ZW50IGhhbmRsZXIgZnVuY3Rpb25zIGZvciBjdXN0b20gZmllbGRzZXQgYWN0aW9ucy5cbiAgICogQHN1bW1hcnkgQSByZWNvcmQgb2YgZXZlbnQgaGFuZGxlciBmdW5jdGlvbnMga2V5ZWQgYnkgZXZlbnQgbmFtZXMgdGhhdCBjYW4gYmUgdHJpZ2dlcmVkXG4gICAqIHdpdGhpbiB0aGUgZmllbGRzZXQuIFRoZXNlIGhhbmRsZXJzIHByb3ZpZGUgZXh0ZW5zaWJpbGl0eSBmb3IgY3VzdG9tIGJ1c2luZXNzIGxvZ2ljXG4gICAqIGFuZCBjYW4gYmUgaW52b2tlZCBmb3IgdmFyaW91cyBmaWVsZHNldCBvcGVyYXRpb25zIGFuZCB1c2VyIGludGVyYWN0aW9ucy5cbiAgICpcbiAgICogQHR5cGUge0hhbmRsZXJMaWtlfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIGhhbmRsZXJzITogSGFuZGxlckxpa2U7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBBcnJheSBvZiBmb3JtYXR0ZWQgaXRlbXMgZm9yIFVJIGRpc3BsYXkuXG4gICAqIEBzdW1tYXJ5IENvbnRhaW5zIHRoZSBwcm9jZXNzZWQgaXRlbXMgcmVhZHkgZm9yIGRpc3BsYXkgaW4gdGhlIGNvbXBvbmVudCB0ZW1wbGF0ZS5cbiAgICogVGhlc2UgaXRlbXMgYXJlIG1hcHBlZCBmcm9tIHRoZSByYXcgdmFsdWVzIHVzaW5nIHRoZSBtYXBwZXIgY29uZmlndXJhdGlvbi5cbiAgICpcbiAgICogQHR5cGUge0lGaWVsZFNldEl0ZW1bXX1cbiAgICogQGRlZmF1bHQgW11cbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBpdGVtczogSUZpZWxkU2V0SXRlbVtdID0gW107XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBDdXJyZW50bHkgc2VsZWN0ZWQgaXRlbSBmb3IgdXBkYXRlIG9wZXJhdGlvbnMuXG4gICAqIEBzdW1tYXJ5IEhvbGRzIHRoZSBpdGVtIGJlaW5nIGVkaXRlZCB3aGVuIGluIHVwZGF0ZSBtb2RlLiBVc2VkIHRvIHRyYWNrXG4gICAqIHdoaWNoIGl0ZW0gaXMgYmVpbmcgbW9kaWZpZWQgYW5kIGFwcGx5IGNoYW5nZXMgdG8gdGhlIGNvcnJlY3QgaXRlbS5cbiAgICpcbiAgICogQHR5cGUge0lGaWVsZFNldEl0ZW0gfCB1bmRlZmluZWR9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgdXBkYXRpbmdJdGVtITogSUZpZWxkU2V0SXRlbSB8IHVuZGVmaW5lZDtcblxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ3VycmVudCBzdGF0ZSBvZiB0aGUgYWNjb3JkaW9uIChleHBhbmRlZCBvciBjb2xsYXBzZWQpLlxuICAgKiBAc3VtbWFyeSBCb29sZWFuIGZsYWcgdGhhdCB0cmFja3Mgd2hldGhlciB0aGUgZmllbGRzZXQgYWNjb3JkaW9uIGlzIGN1cnJlbnRseSBvcGVuIG9yIGNsb3NlZC5cbiAgICogVGhpcyBwcm9wZXJ0eSBpcyBhdXRvbWF0aWNhbGx5IG1hbmFnZWQgYmFzZWQgb24gdXNlciBpbnRlcmFjdGlvbnMgYW5kIGluaXRpYWwgb3BlcmF0aW9uIHN0YXRlLlxuICAgKiBJdCBzZXJ2ZXMgYXMgdGhlIHNpbmdsZSBzb3VyY2Ugb2YgdHJ1dGggZm9yIHRoZSBjb21wb25lbnQncyB2aXNpYmlsaXR5IHN0YXRlIGFuZCBpcyB1c2VkXG4gICAqIHRvIGNvb3JkaW5hdGUgYmV0d2VlbiB1c2VyIGFjdGlvbnMgYW5kIHByb2dyYW1tYXRpYyBzdGF0ZSBjaGFuZ2VzLiBUaGUgdmFsdWUgaXMgYXV0b21hdGljYWxseVxuICAgKiBzZXQgYmFzZWQgb24gQ1JVRCBvcGVyYXRpb25zIGR1cmluZyBpbml0aWFsaXphdGlvbiBhbmQgdXBkYXRlZCB0aHJvdWdoIHVzZXIgaW50ZXJhY3Rpb25zLlxuICAgKlxuICAgKiBAdHlwZSB7Ym9vbGVhbn1cbiAgICogQGRlZmF1bHQgZmFsc2VcbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBpc09wZW46IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEluZGljYXRlcyB3aGV0aGVyIHRoZSBmaWVsZHNldCBjb250YWlucyByZXF1aXJlZCBmb3JtIGZpZWxkcy5cbiAgICogQHN1bW1hcnkgQm9vbGVhbiBmbGFnIHRoYXQgc2lnbmFscyB0aGUgcHJlc2VuY2Ugb2YgbWFuZGF0b3J5IGlucHV0IGZpZWxkcyB3aXRoaW4gdGhlIGZpZWxkc2V0LlxuICAgKiBUaGlzIHByb3BlcnR5IGlzIGF1dG9tYXRpY2FsbHkgc2V0IGJ5IHRoZSBDb2xsYXBzYWJsZURpcmVjdGl2ZSB3aGVuIHJlcXVpcmVkIGZpZWxkcyBhcmUgZGV0ZWN0ZWQsXG4gICAqIGFuZCBjYW4gYmUgdXNlZCB0byBhcHBseSBzcGVjaWFsIHN0eWxpbmcsIHZhbGlkYXRpb24gbG9naWMsIG9yIFVJIGluZGljYXRvcnMgdG8gaGlnaGxpZ2h0XG4gICAqIGZpZWxkc2V0cyB0aGF0IGNvbnRhaW4gbWFuZGF0b3J5IGluZm9ybWF0aW9uLiBJdCBoZWxwcyB3aXRoIGZvcm0gdmFsaWRhdGlvbiBmZWVkYmFjayBhbmRcbiAgICogdXNlciBleHBlcmllbmNlIGJ5IG1ha2luZyByZXF1aXJlZCBzZWN0aW9ucyBtb3JlIHByb21pbmVudC5cbiAgICpcbiAgICogQHR5cGUge2Jvb2xlYW59XG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgaXNSZXF1aXJlZDogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSW5kaWNhdGVzIHdoZXRoZXIgdGhlIGZpZWxkc2V0IGNvbnRhaW5zIHZhbGlkYXRpb24gZXJyb3JzLlxuICAgKiBAc3VtbWFyeSBCb29sZWFuIGZsYWcgdGhhdCB0cmFja3MgaWYgYW55IGZvcm0gZmllbGRzIHdpdGhpbiB0aGUgZmllbGRzZXQgaGF2ZSB2YWxpZGF0aW9uIGVycm9ycy5cbiAgICogVGhpcyBwcm9wZXJ0eSBpcyB1c2VkIHRvIGNvbnRyb2wgYWNjb3JkaW9uIGJlaGF2aW9yIHdoZW4gZXJyb3JzIGFyZSBwcmVzZW50LCBwcmV2ZW50aW5nXG4gICAqIHVzZXJzIGZyb20gY29sbGFwc2luZyB0aGUgYWNjb3JkaW9uIHdoZW4gdGhleSBuZWVkIHRvIHNlZSBhbmQgYWRkcmVzcyB2YWxpZGF0aW9uIGlzc3Vlcy5cbiAgICogSXQncyBhdXRvbWF0aWNhbGx5IHVwZGF0ZWQgd2hlbiB2YWxpZGF0aW9uIGVycm9yIGV2ZW50cyBhcmUgcmVjZWl2ZWQgZnJvbSBjaGlsZCBmb3JtIGZpZWxkcy5cbiAgICpcbiAgICogQHR5cGUge2Jvb2xlYW59XG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgaGFzVmFsaWRhdGlvbkVycm9yczogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVmFsaWRhdGlvbiBlcnJvciBtZXNzYWdlIGZvciBkdXBsaWNhdGUgdmFsdWVzLlxuICAgKiBAc3VtbWFyeSBTdG9yZXMgdGhlIGVycm9yIG1lc3NhZ2Ugd2hlbiBhIHVzZXIgYXR0ZW1wdHMgdG8gYWRkIGEgZHVwbGljYXRlIHZhbHVlXG4gICAqIHRvIHRoZSBmaWVsZHNldC4gVXNlZCB0byBkaXNwbGF5IHVuaXF1ZW5lc3MgdmFsaWRhdGlvbiBmZWVkYmFjay5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZyB8IHVuZGVmaW5lZH1cbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBpc1VuaXF1ZUVycm9yOiBzdHJpbmcgfCB1bmRlZmluZWQgPSB1bmRlZmluZWQ7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBSZWZlcmVuY2UgdG8gQ1JVRCBvcGVyYXRpb24gY29uc3RhbnRzIGZvciB0ZW1wbGF0ZSB1c2FnZS5cbiAgICogQHN1bW1hcnkgRXhwb3NlcyB0aGUgT3BlcmF0aW9uS2V5cyBlbnVtIHRvIHRoZSBjb21wb25lbnQgdGVtcGxhdGUsIGVuYWJsaW5nIGNvbmRpdGlvbmFsXG4gICAqIHJlbmRlcmluZyBhbmQgYmVoYXZpb3IgYmFzZWQgb24gb3BlcmF0aW9uIHR5cGVzLiBUaGlzIHByb3RlY3RlZCByZWFkb25seSBwcm9wZXJ0eSBlbnN1cmVzXG4gICAqIHRoYXQgdGVtcGxhdGUgbG9naWMgY2FuIGFjY2VzcyBvcGVyYXRpb24gY29uc3RhbnRzIHdoaWxlIG1haW50YWluaW5nIGVuY2Fwc3VsYXRpb24gYW5kXG4gICAqIHByZXZlbnRpbmcgYWNjaWRlbnRhbCBtb2RpZmljYXRpb24gb2YgdGhlIGVudW0gdmFsdWVzLlxuICAgKlxuICAgKiBAdHlwZSB7Q3J1ZE9wZXJhdGlvbnN9XG4gICAqIEBkZWZhdWx0IE9wZXJhdGlvbktleXMuQ1JFQVRFXG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgcHJvdGVjdGVkIHJlYWRvbmx5IE9wZXJhdGlvbktleXM6IENydWRPcGVyYXRpb25zID0gT3BlcmF0aW9uS2V5cy5DUkVBVEU7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBBbmd1bGFyIGNoYW5nZSBkZXRlY3Rpb24gc2VydmljZS5cbiAgICogQHN1bW1hcnkgSW5qZWN0ZWQgc2VydmljZSB0aGF0IHByb3ZpZGVzIG1hbnVhbCBjb250cm9sIG92ZXIgY2hhbmdlIGRldGVjdGlvbiBjeWNsZXMuXG4gICAqIFRoaXMgaXMgZXNzZW50aWFsIGZvciBlbnN1cmluZyB0aGF0IHByb2dyYW1tYXRpYyBET00gY2hhbmdlcyAobGlrZSBzZXR0aW5nIGFjY29yZGlvblxuICAgKiBhdHRyaWJ1dGVzKSBhcmUgcHJvcGVybHkgcmVmbGVjdGVkIGluIHRoZSBjb21wb25lbnQncyBzdGF0ZSBhbmQgdHJpZ2dlciBhcHByb3ByaWF0ZVxuICAgKiB2aWV3IHVwZGF0ZXMgd2hlbiBtb2RpZmljYXRpb25zIG9jY3VyIG91dHNpZGUgdGhlIG5vcm1hbCBBbmd1bGFyIGNoYW5nZSBkZXRlY3Rpb24gZmxvdy5cbiAgICpcbiAgICogQHByaXZhdGVcbiAgICogQHR5cGUge0NoYW5nZURldGVjdG9yUmVmfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIHByaXZhdGUgY2hhbmdlRGV0ZWN0b3JSZWY6IENoYW5nZURldGVjdG9yUmVmID0gaW5qZWN0KENoYW5nZURldGVjdG9yUmVmKTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEFuZ3VsYXIgUmVuZGVyZXIyIHNlcnZpY2UgZm9yIHNhZmUgRE9NIG1hbmlwdWxhdGlvbi5cbiAgICogQHN1bW1hcnkgSW5qZWN0ZWQgc2VydmljZSB0aGF0IHByb3ZpZGVzIGEgc2FmZSwgcGxhdGZvcm0tYWdub3N0aWMgd2F5IHRvIG1hbmlwdWxhdGUgRE9NIGVsZW1lbnRzLlxuICAgKiBUaGlzIHNlcnZpY2UgZW5zdXJlcyBwcm9wZXIgaGFuZGxpbmcgb2YgRE9NIG9wZXJhdGlvbnMgYWNyb3NzIGRpZmZlcmVudCBwbGF0Zm9ybXMgYW5kIGVudmlyb25tZW50cyxcbiAgICogaW5jbHVkaW5nIHNlcnZlci1zaWRlIHJlbmRlcmluZyBhbmQgd2ViIHdvcmtlcnMuXG4gICAqXG4gICAqIEBwcml2YXRlXG4gICAqIEB0eXBlIHtSZW5kZXJlcjJ9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgcHJpdmF0ZSByZW5kZXJlcjogUmVuZGVyZXIyID0gaW5qZWN0KFJlbmRlcmVyMik7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBUcmFuc2xhdGlvbiBzZXJ2aWNlIGZvciBpbnRlcm5hdGlvbmFsaXphdGlvbi5cbiAgICogQHN1bW1hcnkgSW5qZWN0ZWQgc2VydmljZSB0aGF0IHByb3ZpZGVzIHRyYW5zbGF0aW9uIGNhcGFiaWxpdGllcyBmb3IgVUkgdGV4dC5cbiAgICogVXNlZCB0byB0cmFuc2xhdGUgYnV0dG9uIGxhYmVscyBhbmQgdmFsaWRhdGlvbiBtZXNzYWdlcyBiYXNlZCBvbiB0aGUgY3VycmVudCBsb2NhbGUuXG4gICAqXG4gICAqIEBwcml2YXRlXG4gICAqIEB0eXBlIHtUcmFuc2xhdGVTZXJ2aWNlfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIHByaXZhdGUgdHJhbnNsYXRlU2VydmljZTogVHJhbnNsYXRlU2VydmljZSA9IGluamVjdChUcmFuc2xhdGVTZXJ2aWNlKTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIExvY2FsaXplZCBsYWJlbCB0ZXh0IGZvciBhY3Rpb24gYnV0dG9ucy5cbiAgICogQHN1bW1hcnkgRHluYW1pYyBidXR0b24gbGFiZWwgdGhhdCBjaGFuZ2VzIGJhc2VkIG9uIHRoZSBjdXJyZW50IG9wZXJhdGlvbiBtb2RlLlxuICAgKiBTaG93cyBcIkFkZFwiIGZvciBjcmVhdGUgb3BlcmF0aW9ucyBhbmQgXCJVcGRhdGVcIiBmb3IgZWRpdCBvcGVyYXRpb25zLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIGJ1dHRvbkxhYmVsITogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ29tcG9uZW50IGNvbnN0cnVjdG9yIHRoYXQgaW5pdGlhbGl6ZXMgdGhlIGZpZWxkc2V0IHdpdGggaWNvbnMgYW5kIGNvbXBvbmVudCBuYW1lLlxuICAgKiBAc3VtbWFyeSBDYWxscyB0aGUgcGFyZW50IE5neEJhc2VDb21wb25lbnQgY29uc3RydWN0b3Igd2l0aCB0aGUgY29tcG9uZW50IG5hbWUgYW5kXG4gICAqIHJlcXVpcmVkIElvbmljIGljb25zIChhbGVydENpcmNsZU91dGxpbmUgZm9yIHZhbGlkYXRpb24gZXJyb3JzIGFuZCBjcmVhdGVPdXRsaW5lIGZvciBhZGQgYWN0aW9ucykuXG4gICAqIFNldHMgdXAgdGhlIGZvdW5kYXRpb25hbCBjb21wb25lbnQgc3RydWN0dXJlIGFuZCBpY29uIHJlZ2lzdHJ5LlxuICAgKlxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKCdGaWVsZHNldENvbXBvbmVudCcsIHthbGVydENpcmNsZU91dGxpbmUsIGNyZWF0ZU91dGxpbmV9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ29tcG9uZW50IGluaXRpYWxpemF0aW9uIGxpZmVjeWNsZSBtZXRob2QuXG4gICAqIEBzdW1tYXJ5IEluaXRpYWxpemVzIHRoZSBjb21wb25lbnQgYnkgc2V0dGluZyB1cCByZXBvc2l0b3J5IHJlbGF0aW9uc2hpcHMgaWYgYSBtb2RlbCBleGlzdHMsXG4gICAqIGFuZCBjb25maWd1cmVzIHRoZSBpbml0aWFsIGJ1dHRvbiBsYWJlbCBmb3IgdGhlIGFkZCBhY3Rpb24gYmFzZWQgb24gdGhlIGN1cnJlbnQgbG9jYWxlLlxuICAgKiBUaGlzIG1ldGhvZCBlbnN1cmVzIHByb3BlciBzZXR1cCBvZiB0cmFuc2xhdGlvbiBzZXJ2aWNlcyBhbmQgY29tcG9uZW50IHN0YXRlLlxuICAgKlxuICAgKiBAcmV0dXJucyB7dm9pZH1cbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICBpZih0aGlzLm1vZGVsKVxuICAgICAgdGhpcy5fcmVwb3NpdG9yeSA9IHRoaXMucmVwb3NpdG9yeTtcbiAgICB0aGlzLmJ1dHRvbkxhYmVsID0gdGhpcy50cmFuc2xhdGVTZXJ2aWNlLmluc3RhbnQodGhpcy5sb2NhbGUgKyAnLmFkZCcpO1xuICB9XG5cbiAgIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSW5pdGlhbGl6ZXMgdGhlIGNvbXBvbmVudCBzdGF0ZSBhZnRlciB2aWV3IGFuZCBjaGlsZCBjb21wb25lbnRzIGFyZSByZW5kZXJlZC5cbiAgICogQHN1bW1hcnkgVGhpcyBsaWZlY3ljbGUgaG9vayBpbXBsZW1lbnRzIGludGVsbGlnZW50IGF1dG8tc3RhdGUgbWFuYWdlbWVudCBiYXNlZCBvbiB0aGUgY3VycmVudFxuICAgKiBDUlVEIG9wZXJhdGlvbi4gRm9yIFJFQUQgYW5kIERFTEVURSBvcGVyYXRpb25zLCB0aGUgZmllbGRzZXQgYXV0b21hdGljYWxseSBvcGVucyB0byBwcm92aWRlXG4gICAqIGltbWVkaWF0ZSBhY2Nlc3MgdG8gaW5mb3JtYXRpb24sIHdoaWxlIENSRUFURSBhbmQgVVBEQVRFIG9wZXJhdGlvbnMga2VlcCBpdCBjbG9zZWQgdG8gbWFpbnRhaW5cbiAgICogYSBjbGVhbiBpbml0aWFsIGludGVyZmFjZS4gVGhlIG1ldGhvZCBkaXJlY3RseSBtYW5pcHVsYXRlcyB0aGUgRE9NIHRvIGVuc3VyZSBwcm9wZXIgYWNjb3JkaW9uXG4gICAqIHN5bmNocm9uaXphdGlvbiBhbmQgdHJpZ2dlcnMgY2hhbmdlIGRldGVjdGlvbiB0byByZWZsZWN0IHRoZSBwcm9ncmFtbWF0aWMgc3RhdGUgY2hhbmdlcy5cbiAgICpcbiAgICogQG1lcm1haWRcbiAgICogc2VxdWVuY2VEaWFncmFtXG4gICAqICAgcGFydGljaXBhbnQgQSBhcyBBbmd1bGFyIExpZmVjeWNsZVxuICAgKiAgIHBhcnRpY2lwYW50IEYgYXMgRmllbGRzZXRDb21wb25lbnRcbiAgICogICBwYXJ0aWNpcGFudCBEIGFzIERPTVxuICAgKiAgIHBhcnRpY2lwYW50IEMgYXMgQ2hhbmdlRGV0ZWN0b3JcbiAgICpcbiAgICogICBBLT4+RjogbmdBZnRlclZpZXdJbml0KClcbiAgICogICBhbHQgb3BlcmF0aW9uIGlzIFJFQUQgb3IgREVMRVRFXG4gICAqICAgICBGLT4+RjogU2V0IGlzT3BlbiA9IHRydWVcbiAgICogICAgIEYtPj5EOiBRdWVyeSBpb24tYWNjb3JkaW9uLWdyb3VwIGVsZW1lbnRcbiAgICogICAgIGFsdCBhY2NvcmRpb24gZWxlbWVudCBleGlzdHNcbiAgICogICAgICAgRi0+PkQ6IFNldCB2YWx1ZSBhdHRyaWJ1dGUgdG8gJ29wZW4nXG4gICAqICAgICBlbmRcbiAgICogICBlbmRcbiAgICogICBGLT4+QzogZGV0ZWN0Q2hhbmdlcygpXG4gICAqICAgQy0+PkY6IFVwZGF0ZSB2aWV3IHdpdGggbmV3IHN0YXRlXG4gICAqXG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIG5nQWZ0ZXJWaWV3SW5pdCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5vcGVyYXRpb24gPT09IE9wZXJhdGlvbktleXMuUkVBRCB8fCB0aGlzLm9wZXJhdGlvbiA9PT0gT3BlcmF0aW9uS2V5cy5ERUxFVEUpIHtcbiAgICAgIHRoaXMuaXNPcGVuID0gdHJ1ZTtcbiAgICAgIC8vIGhpZGRlbiByZW1vdmUgYnV0dG9uXG4gICAgICBjb25zdCBhY2NvcmRpb25FbGVtZW50ID0gdGhpcy5jb21wb25lbnQ/Lm5hdGl2ZUVsZW1lbnQucXVlcnlTZWxlY3RvcignaW9uLWFjY29yZGlvbi1ncm91cCcpO1xuICAgICAgaWYodGhpcy5hY2NvcmRpb25Db21wb25lbnQpXG4gICAgICAgIHRoaXMucmVuZGVyZXIuc2V0QXR0cmlidXRlKGFjY29yZGlvbkVsZW1lbnQsICd2YWx1ZScsICdvcGVuJyk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IGlucHV0cyA9IHRoaXMuY29tcG9uZW50Py5uYXRpdmVFbGVtZW50LnF1ZXJ5U2VsZWN0b3JBbGwoJ1tyZXF1aXJlZF0nKTtcbiAgICAgIHRoaXMuaXNSZXF1aXJlZCA9IGlucHV0cy5sZW5ndGggPiAwO1xuICAgICAgaWYodGhpcy5pc1JlcXVpcmVkKSB7XG4gICAgICAgIHRoaXMuYWNjb3JkaW9uQ29tcG9uZW50LnZhbHVlID0gJ29wZW4nO1xuICAgICAgICB0aGlzLmhhbmRsZUFjY29yZGlvblRvZ2dsZSgpO1xuICAgICAgfVxuICAgIH1cbiAgICB0aGlzLmNoYW5nZURldGVjdG9yUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSGFuZGxlcyByZW1vdmFsIG9mIHRoZSBmaWVsZHNldCB3aXRoIHNsaWRlIGFuaW1hdGlvbi5cbiAgICogQHN1bW1hcnkgSW5pdGlhdGVzIHRoZSByZW1vdmFsIHByb2Nlc3MgZm9yIHRoZSBmaWVsZHNldCB3aXRoIGEgc21vb3RoIHNsaWRlLXVwIGFuaW1hdGlvbi5cbiAgICogVGhlIG1ldGhvZCBhcHBsaWVzIENTUyBjbGFzc2VzIGZvciB0aGUgc2xpZGUgYW5pbWF0aW9uIGFuZCB0aGVuIHNhZmVseSByZW1vdmVzIHRoZVxuICAgKiBlbGVtZW50IGZyb20gdGhlIERPTSB1c2luZyBSZW5kZXJlcjIuIFRoaXMgcHJvdmlkZXMgYSBwb2xpc2hlZCB1c2VyIGV4cGVyaWVuY2VcbiAgICogd2hlbiByZW1vdmluZyBmaWVsZHNldCBpbnN0YW5jZXMgZnJvbSBkeW5hbWljIGZvcm1zLlxuICAgKlxuICAgKiBAcGFyYW0ge0V2ZW50fSBldmVudCAtIERPTSBldmVudCBmcm9tIHRoZSByZW1vdmUgYnV0dG9uIGNsaWNrXG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIGhhbmRsZVJlbW92ZUNvbXBvbmVudChldmVudDogRXZlbnQpOiB2b2lkIHtcbiAgICBldmVudC5zdG9wSW1tZWRpYXRlUHJvcGFnYXRpb24oKTtcbiAgICB0aGlzLmNvbXBvbmVudC5uYXRpdmVFbGVtZW50LmNsYXNzTGlzdC5hZGQoJ2RjZi1hbmltYXRpb24nLCAnZGNmLWFuaW1hdGlvbi1zbGlkZS10b3AtbWVkaXVtJywgJ2RjZi1hbmltYXRpb24tcmV2ZXJzZScsICdkY2YtYW5pbWF0aW9uLWZhc3QnKTtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIC8vIFVzZSBSZW5kZXJlcjIgdG8gc2FmZWx5IHJlbW92ZSB0aGUgZWxlbWVudFxuICAgICAgY29uc3QgcGFyZW50ID0gdGhpcy5yZW5kZXJlci5wYXJlbnROb2RlKHRoaXMuY29tcG9uZW50Lm5hdGl2ZUVsZW1lbnQpO1xuICAgICAgaWYgKHBhcmVudClcbiAgICAgICAgdGhpcy5yZW5kZXJlci5yZW1vdmVDaGlsZChwYXJlbnQsIHRoaXMuY29tcG9uZW50Lm5hdGl2ZUVsZW1lbnQpO1xuICAgIH0sIDE1MCk7XG4gIH1cblxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSGFuZGxlcyBjcmVhdGluZyBuZXcgaXRlbXMgb3IgdHJpZ2dlcmluZyBncm91cCBhZGRpdGlvbiBldmVudHMuXG4gICAqIEBzdW1tYXJ5IFByb2Nlc3NlcyBmb3JtIHZhbGlkYXRpb24gZXZlbnRzIGZvciBpdGVtIGNyZWF0aW9uIG9yIGVtaXRzIGV2ZW50cyB0byB0cmlnZ2VyXG4gICAqIHRoZSBhZGRpdGlvbiBvZiBuZXcgZmllbGRzZXQgZ3JvdXBzLiBXaGVuIGNhbGxlZCB3aXRoIHZhbGlkYXRpb24gZXZlbnQgZGF0YSwgaXQgdmFsaWRhdGVzXG4gICAqIHVuaXF1ZW5lc3MgYW5kIGFkZHMgdGhlIGl0ZW0gdG8gdGhlIGZpZWxkc2V0LiBXaGVuIGNhbGxlZCB3aXRob3V0IHBhcmFtZXRlcnMsIGl0IHRyaWdnZXJzXG4gICAqIGEgZ3JvdXAgYWRkaXRpb24gZXZlbnQgZm9yIHBhcmVudCBjb21wb25lbnRzIHRvIGhhbmRsZS5cbiAgICpcbiAgICogQHBhcmFtIHtDdXN0b21FdmVudDxJRmllbGRTZXRWYWxpZGF0aW9uRXZlbnQ+fSBbZXZlbnRdIC0gT3B0aW9uYWwgdmFsaWRhdGlvbiBldmVudCBjb250YWluaW5nIGZvcm0gZGF0YVxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn1cbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogLy8gQ2FsbGVkIGZyb20gZm9ybSB2YWxpZGF0aW9uXG4gICAqIGhhbmRsZUNyZWF0ZUl0ZW0odmFsaWRhdGlvbkV2ZW50KTtcbiAgICpcbiAgICogLy8gQ2FsbGVkIHRvIHRyaWdnZXIgZ3JvdXAgYWRkaXRpb25cbiAgICogaGFuZGxlQ3JlYXRlSXRlbSgpO1xuICAgKiBgYGBcbiAgICovXG4gIGFzeW5jIGhhbmRsZUNyZWF0ZUl0ZW0oZXZlbnQ/OiBDdXN0b21FdmVudDxJRmllbGRTZXRWYWxpZGF0aW9uRXZlbnQ+KTogUHJvbWlzZTx2b2lkPiB7XG4gICAgaWYoZXZlbnQgJiYgZXZlbnQgaW5zdGFuY2VvZiBDdXN0b21FdmVudCkge1xuICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKCk7XG4gICAgICBjb25zdCB7Zm9ybUdyb3VwLCB2YWx1ZSwgaXNWYWxpZH0gPSBldmVudC5kZXRhaWw7XG4gICAgICB0aGlzLmZvcm1Hcm91cCA9IGZvcm1Hcm91cCBhcyBGb3JtQXJyYXk7XG4gICAgICBpZighdGhpcy5tYXBwZXIpXG4gICAgICAgIHRoaXMubWFwcGVyID0gdGhpcy5nZXRNYXBwZXIodmFsdWUgYXMgS2V5VmFsdWUpO1xuICAgICAgaWYoaXNWYWxpZCApe1xuICAgICAgICAgIHRoaXMuaXNVbmlxdWVFcnJvciA9IHVuZGVmaW5lZDtcbiAgICAgICAgICB0aGlzLmJ1dHRvbkxhYmVsID0gdGhpcy50cmFuc2xhdGVTZXJ2aWNlLmluc3RhbnQodGhpcy5sb2NhbGUgKyAnLmFkZCcpO1xuICAgICAgICAgIHRoaXMuc2V0VmFsdWUoKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgdGhpcy5pc1VuaXF1ZUVycm9yID0gKHZhbHVlIGFzIEtleVZhbHVlKT8uW3RoaXMucGtdIHx8IHVuZGVmaW5lZDtcbiAgICAgIH1cbiAgICB9IGVsc2Uge1xuICAgICAgd2luZG93RXZlbnRFbWl0dGVyKEV2ZW50Q29uc3RhbnRzLkZJRUxEU0VUX0FERF9HUk9VUCwge1xuICAgICAgICBjb21wb25lbnQ6IHRoaXMuY29tcG9uZW50Lm5hdGl2ZUVsZW1lbnQsXG4gICAgICAgIGluZGV4OiB0aGlzLnZhbHVlPy5sZW5ndGgsXG4gICAgICAgIHBhcmVudDogdGhpcy5jaGlsZE9mLFxuICAgICAgICBvcGVyYXRpb246ICF0aGlzLnVwZGF0aW5nSXRlbSA/IE9wZXJhdGlvbktleXMuQ1JFQVRFIDogT3BlcmF0aW9uS2V5cy5VUERBVEVcbiAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBIYW5kbGVzIGl0ZW0gdXBkYXRlIG9wZXJhdGlvbnMgd2l0aCBmb3JtIHN0YXRlIG1hbmFnZW1lbnQuXG4gICAqIEBzdW1tYXJ5IExvY2F0ZXMgYW4gaXRlbSBpbiB0aGUgZm9ybSBhcnJheSBmb3IgZWRpdGluZyBhbmQgcHJlcGFyZXMgdGhlIGNvbXBvbmVudFxuICAgKiBmb3IgdXBkYXRlIG1vZGUuIFVwZGF0ZXMgdGhlIGJ1dHRvbiBsYWJlbCB0byByZWZsZWN0IHRoZSBlZGl0IHN0YXRlIGFuZCBzdG9yZXNcbiAgICogdGhlIGl0ZW0gYmVpbmcgdXBkYXRlZC4gVHJpZ2dlcnMgYSB3aW5kb3cgZXZlbnQgdG8gbm90aWZ5IHBhcmVudCBjb21wb25lbnRzLlxuICAgKlxuICAgKiBAcGFyYW0ge3N0cmluZyB8IG51bWJlcn0gdmFsdWUgLSBUaGUgaWRlbnRpZmllciB2YWx1ZSBvZiB0aGUgaXRlbSB0byB1cGRhdGVcbiAgICogQHBhcmFtIHtudW1iZXJ9IGluZGV4IC0gVGhlIGFycmF5IGluZGV4IHBvc2l0aW9uIG9mIHRoZSBpdGVtXG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIGhhbmRsZVVwZGF0ZUl0ZW0odmFsdWU6IHN0cmluZyB8IG51bWJlciwgaW5kZXg6IG51bWJlcik6IHZvaWQge1xuICAgIGNvbnN0IGl0ZW0gPSB0aGlzLmZvcm1Hcm91cC5jb250cm9scy5maW5kKGNvbnRyb2wgPT4gYCR7Y29udHJvbC5nZXQodGhpcy5wayk/LnZhbHVlfWAudG9Mb3dlckNhc2UoKSA9PT0gY2xlYW5TcGFjZXMoYCR7dmFsdWV9YCwgdHJ1ZSkpIGFzIEZvcm1Db250cm9sO1xuICAgIGlmKGl0ZW0pIHtcbiAgICAgIHRoaXMuYnV0dG9uTGFiZWwgPSB0aGlzLnRyYW5zbGF0ZVNlcnZpY2UuaW5zdGFudCh0aGlzLmxvY2FsZSArICcudXBkYXRlJyk7XG4gICAgICB0aGlzLnVwZGF0aW5nSXRlbSA9IE9iamVjdC5hc3NpZ24oe30sIGl0ZW0udmFsdWUgfHwge30pO1xuICAgICAgd2luZG93RXZlbnRFbWl0dGVyKEV2ZW50Q29uc3RhbnRzLkZJRUxEU0VUX1VQREFURV9HUk9VUCwge1xuICAgICAgICBwYXJlbnQ6IHRoaXMuY2hpbGRPZixcbiAgICAgICAgY29tcG9uZW50OiB0aGlzLmNvbXBvbmVudC5uYXRpdmVFbGVtZW50LFxuICAgICAgICBpbmRleDogaW5kZXhcbiAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ2FuY2VscyB0aGUgdXBkYXRlIG1vZGUgYW5kIHJlc2V0cyB0aGUgVUkgc3RhdGUuXG4gICAqIEBzdW1tYXJ5IEV4aXRzIHRoZSB1cGRhdGUgbW9kZSBieSByZXNldHRpbmcgdGhlIGJ1dHRvbiBsYWJlbCBhbmQgY2xlYXJpbmcgdGhlIHVwZGF0aW5nIGl0ZW0sXG4gICAqIHJlc3RvcmluZyB0aGUgY29tcG9uZW50IHRvIGl0cyBkZWZhdWx0IHN0YXRlIGZvciBhZGRpbmcgbmV3IGl0ZW1zLiBOb3RpZmllcyBwYXJlbnQgY29tcG9uZW50c1xuICAgKiB0aGF0IHRoZSB1cGRhdGUgb3BlcmF0aW9uIGhhcyBiZWVuIGNhbmNlbGxlZC5cbiAgICpcbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgaGFuZGxlQ2FuY2VsVXBkYXRlSXRlbSgpOiB2b2lkIHtcbiAgICB0aGlzLmJ1dHRvbkxhYmVsID0gdGhpcy50cmFuc2xhdGVTZXJ2aWNlLmluc3RhbnQodGhpcy5sb2NhbGUgKyAnLmFkZCcpO1xuICAgIHRoaXMudXBkYXRpbmdJdGVtID0gdW5kZWZpbmVkO1xuICAgIHdpbmRvd0V2ZW50RW1pdHRlcihFdmVudENvbnN0YW50cy5GSUVMRFNFVF9VUERBVEVfR1JPVVAsIHtcbiAgICAgIHBhcmVudDogdGhpcy5jaGlsZE9mLFxuICAgICAgY29tcG9uZW50OiB0aGlzLmNvbXBvbmVudC5uYXRpdmVFbGVtZW50LFxuICAgICAgaW5kZXg6IHRoaXMudmFsdWU/Lmxlbmd0aFxuICAgIH0pO1xuICB9XG5cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEhhbmRsZXMgaXRlbSByZW1vdmFsIG9wZXJhdGlvbnMgd2l0aCBmb3JtIGFycmF5IG1hbmFnZW1lbnQuXG4gICAqIEBzdW1tYXJ5IFByb2Nlc3NlcyBpdGVtIHJlbW92YWwgYnkgZWl0aGVyIGhhbmRsaW5nIHZhbGlkYXRpb24gZXZlbnRzIG9yIHJlbW92aW5nIHNwZWNpZmljXG4gICAqIGl0ZW1zIGZyb20gdGhlIGZvcm0gYXJyYXkuIFdoZW4gY2FsbGVkIHdpdGggYSB2YWxpZGF0aW9uIGV2ZW50LCBpdCB0cmlnZ2VycyB2YWx1ZSB1cGRhdGVzLlxuICAgKiBXaGVuIGNhbGxlZCB3aXRoIGFuIGlkZW50aWZpZXIsIGl0IGxvY2F0ZXMgYW5kIHJlbW92ZXMgdGhlIG1hdGNoaW5nIGl0ZW0gZnJvbSB0aGUgZm9ybSBhcnJheS5cbiAgICpcbiAgICogQHBhcmFtIHtzdHJpbmcgfCB1bmRlZmluZWR9IHZhbHVlIC0gVGhlIGlkZW50aWZpZXIgb2YgdGhlIGl0ZW0gdG8gcmVtb3ZlXG4gICAqIEBwYXJhbSB7Q3VzdG9tRXZlbnR9IFtldmVudF0gLSBPcHRpb25hbCB2YWxpZGF0aW9uIGV2ZW50IGZvciBmb3JtIHVwZGF0ZXNcbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgaGFuZGxlUmVtb3ZlSXRlbSh2YWx1ZTogc3RyaW5nIHwgdW5kZWZpbmVkLCBldmVudD86IEN1c3RvbUV2ZW50KTogdm9pZCB7XG4gICAgaWYoZXZlbnQgJiYgZXZlbnQgaW5zdGFuY2VvZiBDdXN0b21FdmVudCkge1xuICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKCk7XG4gICAgICByZXR1cm4gdGhpcy5zZXRWYWx1ZSgpO1xuICAgIH1cbiAgICBjb25zdCBmb3JtQXJyYXkgPSB0aGlzLmZvcm1Hcm91cCBhcyBGb3JtQXJyYXk7XG4gICAgY29uc3QgYXJyYXlMZW5ndGggPSBmb3JtQXJyYXkubGVuZ3RoO1xuICAgIGZvciAobGV0IGluZGV4ID0gYXJyYXlMZW5ndGggLSAxOyBpbmRleCA+PSAwOyBpbmRleC0tKSB7XG4gICAgICBjb25zdCBncm91cCA9IGZvcm1BcnJheS5hdChpbmRleCkgYXMgRm9ybUdyb3VwO1xuICAgICAgaWYgKGNsZWFuU3BhY2VzKGdyb3VwLmdldCh0aGlzLnBrKT8udmFsdWUpID09PSBjbGVhblNwYWNlcyh2YWx1ZSBhcyBzdHJpbmcpKSB7XG4gICAgICAgIHdpbmRvd0V2ZW50RW1pdHRlcihFdmVudENvbnN0YW50cy5GSUVMRFNFVF9SRU1PVkVfR1JPVVAsIHtcbiAgICAgICAgICBwYXJlbnQ6IHRoaXMuY2hpbGRPZixcbiAgICAgICAgICBjb21wb25lbnQ6IHRoaXMuY29tcG9uZW50Lm5hdGl2ZUVsZW1lbnQsXG4gICAgICAgICAgaW5kZXgsXG4gICAgICAgICAgZm9ybUdyb3VwOiBncm91cFxuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSGFuZGxlcyByZW9yZGVyaW5nIG9mIGl0ZW1zIHdpdGhpbiB0aGUgZmllbGRzZXQgbGlzdC5cbiAgICogQHN1bW1hcnkgUHJvY2Vzc2VzIGRyYWctYW5kLWRyb3AgcmVvcmRlciBldmVudHMgZnJvbSB0aGUgaW9uLXJlb3JkZXItZ3JvdXAgY29tcG9uZW50LlxuICAgKiBVcGRhdGVzIGJvdGggdGhlIGRpc3BsYXkgaXRlbXMgYXJyYXkgYW5kIHRoZSB1bmRlcmx5aW5nIHZhbHVlIGFycmF5IHRvIG1haW50YWluXG4gICAqIGNvbnNpc3RlbmN5IGJldHdlZW4gVUkgc3RhdGUgYW5kIGRhdGEgc3RhdGUuIFByZXNlcnZlcyBpdGVtIGluZGljZXMgYWZ0ZXIgcmVvcmRlcmluZy5cbiAgICpcbiAgICogQHBhcmFtIHtDdXN0b21FdmVudDxJdGVtUmVvcmRlckV2ZW50RGV0YWlsPn0gZXZlbnQgLSBJb25pYyByZW9yZGVyIGV2ZW50IGNvbnRhaW5pbmcgc291cmNlIGFuZCB0YXJnZXQgaW5kaWNlc1xuICAgKiBAcmV0dXJucyB7dm9pZH1cbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYGh0bWxcbiAgICogPGlvbi1yZW9yZGVyLWdyb3VwIChpb25JdGVtUmVvcmRlcik9XCJoYW5kbGVSZW9yZGVyKCRldmVudClcIj5cbiAgICogICA8IS0tIFJlb3JkZXJhYmxlIGl0ZW1zIC0tPlxuICAgKiA8L2lvbi1yZW9yZGVyLWdyb3VwPlxuICAgKiBgYGBcbiAgICovXG4gIGhhbmRsZVJlb3JkZXJJdGVtcyhldmVudDogQ3VzdG9tRXZlbnQ8SXRlbVJlb3JkZXJFdmVudERldGFpbD4pOiB2b2lkIHtcbiAgIGNvbnN0IGZyb21JbmRleCA9IGV2ZW50LmRldGFpbC5mcm9tO1xuICAgIGNvbnN0IHRvSW5kZXggPSBldmVudC5kZXRhaWwudG87XG5cbiAgICBjb25zdCBpdGVtcyA9IFsuLi50aGlzLml0ZW1zXTsgLy8gc3VhIGVzdHJ1dHVyYSB2aXN1YWxcbiAgICBjb25zdCBmb3JtQXJyYXkgPSB0aGlzLmZvcm1Hcm91cCBhcyBGb3JtQXJyYXk7IC8vIEZvcm1BcnJheSByZWF0aXZvXG5cbiAgICBpZiAoZnJvbUluZGV4ICE9PSB0b0luZGV4KSB7XG4gICAgICAvLyBSZW9yZGVuYXIgb3MgZGFkb3MgdmlzdWFpc1xuICAgICAgY29uc3QgaXRlbVRvTW92ZSA9IGl0ZW1zLnNwbGljZShmcm9tSW5kZXgsIDEpWzBdO1xuICAgICAgaXRlbXMuc3BsaWNlKHRvSW5kZXgsIDAsIGl0ZW1Ub01vdmUpO1xuICAgICAgaXRlbXMuZm9yRWFjaCgoaXRlbSwgaW5kZXgpID0+IGl0ZW1bJ2luZGV4J10gPSBpbmRleCArIDEpO1xuXG4gICAgICAvLyBSZW9yZGVuYXIgb3MgY29udHJvbGVzIGRvIEZvcm1BcnJheVxuICAgICAgY29uc3QgY29udHJvbFRvTW92ZSA9IGZvcm1BcnJheS5hdChmcm9tSW5kZXgpO1xuICAgICAgZm9ybUFycmF5LnJlbW92ZUF0KGZyb21JbmRleCk7XG4gICAgICBmb3JtQXJyYXkuaW5zZXJ0KHRvSW5kZXgsIGNvbnRyb2xUb01vdmUpO1xuICAgIH1cbiAgICAvLyBGaW5hbGl6YSBhIG9wZXJhw6fDo28gZGUgcmVvcmRlciBkbyBJb25pY1xuICAgIGV2ZW50LmRldGFpbC5jb21wbGV0ZSgpO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBIYW5kbGVzIGFjY29yZGlvbiBzdGF0ZSBjaGFuZ2UgZXZlbnRzIGZyb20gdXNlciBpbnRlcmFjdGlvbnMuXG4gICAqIEBzdW1tYXJ5IFByb2Nlc3NlcyBDdXN0b21FdmVudCBvYmplY3RzIHRyaWdnZXJlZCB3aGVuIHVzZXJzIGV4cGFuZCBvciBjb2xsYXBzZSB0aGUgYWNjb3JkaW9uLlxuICAgKiBUaGlzIG1ldGhvZCBleHRyYWN0cyB0aGUgbmV3IHN0YXRlIGZyb20gdGhlIGV2ZW50IGRldGFpbHMgYW5kIHVwZGF0ZXMgdGhlIGNvbXBvbmVudCdzXG4gICAqIGludGVybmFsIHN0YXRlIGFjY29yZGluZ2x5LiBJdCBzcGVjaWZpY2FsbHkgbGlzdGVucyBmb3IgSU9OLUFDQ09SRElPTi1HUk9VUCBldmVudHMgdG9cbiAgICogZW5zdXJlIHByb3BlciBldmVudCBzb3VyY2UgdmFsaWRhdGlvbiBhbmQgcHJldmVudCBoYW5kbGluZyBvZiB1bnJlbGF0ZWQgZXZlbnRzLlxuICAgKlxuICAgKiBAcGFyYW0ge0N1c3RvbUV2ZW50fSBldmVudCAtIFRoZSBldmVudCBvYmplY3QgY29udGFpbmluZyBhY2NvcmRpb24gc3RhdGUgY2hhbmdlIGRldGFpbHNcbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IFUgYXMgVXNlclxuICAgKiAgIHBhcnRpY2lwYW50IEkgYXMgSW9uLUFjY29yZGlvblxuICAgKiAgIHBhcnRpY2lwYW50IEYgYXMgRmllbGRzZXRDb21wb25lbnRcbiAgICpcbiAgICogICBVLT4+STogQ2xpY2sgYWNjb3JkaW9uIGhlYWRlclxuICAgKiAgIEktPj5GOiBoYW5kbGVDaGFuZ2UoQ3VzdG9tRXZlbnQpXG4gICAqICAgRi0+PkY6IEV4dHJhY3QgdGFyZ2V0IGFuZCBkZXRhaWwgZnJvbSBldmVudFxuICAgKiAgIEYtPj5GOiBWYWxpZGF0ZSB0YXJnZXQgaXMgSU9OLUFDQ09SRElPTi1HUk9VUFxuICAgKiAgIGFsdCB2YWxpZCB0YXJnZXRcbiAgICogICAgIEYtPj5GOiBVcGRhdGUgaXNPcGVuID0gISF2YWx1ZVxuICAgKiAgIGVuZFxuICAgKiAgIEYtPj5JOiBSZWZsZWN0IHVwZGF0ZWQgc3RhdGVcbiAgICpcbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSGFuZGxlcyBhY2NvcmRpb24gdG9nZ2xlIGZ1bmN0aW9uYWxpdHkgd2l0aCB2YWxpZGF0aW9uIGVycm9yIGNvbnNpZGVyYXRpb24uXG4gICAqIEBzdW1tYXJ5IE1hbmFnZXMgdGhlIGV4cGFuZC9jb2xsYXBzZSBzdGF0ZSBvZiB0aGUgYWNjb3JkaW9uIHdoaWxlIHJlc3BlY3RpbmcgdmFsaWRhdGlvbiBlcnJvciBzdGF0ZXMuXG4gICAqIFdoZW4gdmFsaWRhdGlvbiBlcnJvcnMgYXJlIHByZXNlbnQsIHRoZSBhY2NvcmRpb24gY2Fubm90IGJlIGNvbGxhcHNlZCB0byBlbnN1cmUgdXNlcnMgY2FuIHNlZVxuICAgKiBhbmQgYWRkcmVzcyB0aGUgZXJyb3JzLiBXaGVuIG5vIGVycm9ycyBleGlzdCwgdXNlcnMgY2FuIGZyZWVseSB0b2dnbGUgdGhlIGFjY29yZGlvbiBzdGF0ZS5cbiAgICogVGhpcyBtZXRob2QgYWxzbyBzdG9wcyBldmVudCBwcm9wYWdhdGlvbiB0byBwcmV2ZW50IHVud2FudGVkIHNpZGUgZWZmZWN0cy5cbiAgICpcbiAgICogQHBhcmFtIHtDdXN0b21FdmVudH0gW2V2ZW50XSAtIE9wdGlvbmFsIGV2ZW50IG9iamVjdCBmcm9tIHVzZXIgaW50ZXJhY3Rpb25cbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqIEBtZW1iZXJPZiBGaWVsZHNldENvbXBvbmVudFxuICAgKi9cbiAgaGFuZGxlQWNjb3JkaW9uVG9nZ2xlKGV2ZW50PzogQ3VzdG9tRXZlbnQpOiB2b2lkIHtcbiAgICBpZihldmVudClcbiAgICAgIGV2ZW50LnN0b3BJbW1lZGlhdGVQcm9wYWdhdGlvbigpO1xuICAgIGlmKCF0aGlzLmhhc1ZhbGlkYXRpb25FcnJvcnMpIHtcbiAgICAgIHRoaXMuYWNjb3JkaW9uQ29tcG9uZW50LnZhbHVlID0gdGhpcy5pc09wZW4gPyB1bmRlZmluZWQgOiAnb3Blbic7XG4gICAgICB0aGlzLmlzT3BlbiA9ICEhdGhpcy5hY2NvcmRpb25Db21wb25lbnQudmFsdWU7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBIYW5kbGVzIHZhbGlkYXRpb24gZXJyb3IgZXZlbnRzIGZyb20gY2hpbGQgZm9ybSBmaWVsZHMuXG4gICAqIEBzdW1tYXJ5IFByb2Nlc3NlcyB2YWxpZGF0aW9uIGVycm9yIGV2ZW50cyBkaXNwYXRjaGVkIGJ5IGZvcm0gZmllbGRzIHdpdGhpbiB0aGUgZmllbGRzZXQuXG4gICAqIFdoZW4gZXJyb3JzIGFyZSBkZXRlY3RlZCwgdGhlIGFjY29yZGlvbiBpcyBmb3JjZWQgb3BlbiBhbmQgcHJldmVudGVkIGZyb20gY29sbGFwc2luZ1xuICAgKiB0byBlbnN1cmUgdXNlcnMgY2FuIHNlZSB0aGUgdmFsaWRhdGlvbiBtZXNzYWdlcy4gVGhpcyBtZXRob2QgdXBkYXRlcyB0aGUgY29tcG9uZW50J3NcbiAgICogZXJyb3Igc3RhdGUgYW5kIGFjY29yZGlvbiB2aXNpYmlsaXR5IGFjY29yZGluZ2x5LlxuICAgKlxuICAgKiBAcGFyYW0ge0N1c3RvbUV2ZW50fSBldmVudCAtIEN1c3RvbSBldmVudCBjb250YWluaW5nIHZhbGlkYXRpb24gZXJyb3IgZGV0YWlsc1xuICAgKiBAcmV0dXJucyB7dm9pZH1cbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBoYW5kbGVWYWxpZGF0aW9uRXJyb3IoZXZlbnQ6IEN1c3RvbUV2ZW50KTogdm9pZCB7XG4gICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKCk7XG4gICAgY29uc3Qge2hhc0Vycm9yc30gPSBldmVudC5kZXRhaWw7XG4gICAgdGhpcy5pc09wZW4gPSB0aGlzLmhhc1ZhbGlkYXRpb25FcnJvcnMgPSBoYXNFcnJvcnM7XG4gICAgaWYoaGFzRXJyb3JzKVxuICAgICAgdGhpcy5hY2NvcmRpb25Db21wb25lbnQudmFsdWUgPSAnb3Blbic7XG4gIH1cblxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUHJvY2Vzc2VzIGFuZCBzdG9yZXMgYSBuZXcgb3IgdXBkYXRlZCB2YWx1ZSBpbiB0aGUgZmllbGRzZXQuXG4gICAqIEBzdW1tYXJ5IEhhbmRsZXMgYm90aCBjcmVhdGUgYW5kIHVwZGF0ZSBvcGVyYXRpb25zIGZvciBmaWVsZHNldCBpdGVtcy4gUGFyc2VzIGFuZCBjbGVhbnNcbiAgICogdGhlIGlucHV0IHZhbHVlLCBkZXRlcm1pbmVzIHRoZSBvcGVyYXRpb24gdHlwZSBiYXNlZCBvbiB0aGUgdXBkYXRpbmcgc3RhdGUsIGFuZCBlaXRoZXJcbiAgICogYWRkcyBhIG5ldyBpdGVtIG9yIHVwZGF0ZXMgYW4gZXhpc3Rpbmcgb25lLiBNYWludGFpbnMgZGF0YSBpbnRlZ3JpdHkgYW5kIFVJIGNvbnNpc3RlbmN5LlxuICAgKlxuICAgKiBAcmV0dXJucyB7dm9pZH1cbiAgICogQHByaXZhdGVcbiAgICogQG1lbWJlck9mIEZpZWxkc2V0Q29tcG9uZW50XG4gICAqL1xuICBwcml2YXRlIHNldFZhbHVlKCk6IHZvaWQge1xuICAgIHRoaXMudmFsdWUgPSAodGhpcy5mb3JtR3JvdXAgYXMgRm9ybUFycmF5KS5jb250cm9scy5tYXAoKHt2YWx1ZX0pID0+IHZhbHVlKTtcbiAgICB0aGlzLml0ZW1zID0gdGhpcy52YWx1ZVxuICAgIC5maWx0ZXIodiA9PiB2W3RoaXMucGtdICE9PSB1bmRlZmluZWQpXG4gICAgLm1hcCgodiwgaW5kZXgpID0+IHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIC4uLml0ZW1NYXBwZXIodiwgdGhpcy5tYXBwZXIpLFxuICAgICAgICBpbmRleDogaW5kZXggKyAxXG4gICAgICB9IGFzIElGaWVsZFNldEl0ZW07XG4gICAgfSk7XG4gICAgY29uc3QgaW5wdXRDb250YWluZXJzID0gdGhpcy5jb21wb25lbnQubmF0aXZlRWxlbWVudC5xdWVyeVNlbGVjdG9yQWxsKCcuZGNmLWlucHV0LWl0ZW0nKTtcbiAgICBpbnB1dENvbnRhaW5lcnMuZm9yRWFjaCgoY29udGFpbmVyOiBIVE1MRWxlbWVudCkgPT4ge1xuICAgICAgY29uc3QgaW5wdXQgPSBjb250YWluZXIucXVlcnlTZWxlY3RvcignaW5wdXQsIGlvbi1pbnB1dCwgaW9uLXRleHRhcmVhLCB0ZXh0YXJlYScpIGFzIEhUTUxJbnB1dEVsZW1lbnQgfCBudWxsO1xuICAgICAgaWYoaW5wdXQpXG4gICAgICAgIGlucHV0LnZhbHVlID0gJyc7XG4gICAgfSlcbiAgICB0aGlzLnVwZGF0aW5nSXRlbSA9IHVuZGVmaW5lZDtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQXV0b21hdGljYWxseSBjb25maWd1cmVzIHRoZSBmaWVsZCBtYXBwaW5nIGJhc2VkIG9uIHRoZSB2YWx1ZSBzdHJ1Y3R1cmUuXG4gICAqIEBzdW1tYXJ5IEFuYWx5emVzIHRoZSBwcm92aWRlZCB2YWx1ZSBvYmplY3QgdG8gYXV0b21hdGljYWxseSBkZXRlcm1pbmUgdGhlIHByaW1hcnkga2V5XG4gICAqIGFuZCBjcmVhdGUgYXBwcm9wcmlhdGUgZmllbGQgbWFwcGluZ3MgZm9yIGRpc3BsYXkgcHVycG9zZXMuIFNldHMgdXAgdGhlIG1hcHBlciBvYmplY3RcbiAgICogd2l0aCB0aXRsZSwgZGVzY3JpcHRpb24sIGFuZCBpbmRleCBmaWVsZHMgYmFzZWQgb24gdGhlIGF2YWlsYWJsZSBkYXRhIHN0cnVjdHVyZS5cbiAgICpcbiAgICogQHBhcmFtIHtLZXlWYWx1ZX0gdmFsdWUgLSBTYW1wbGUgdmFsdWUgb2JqZWN0IHVzZWQgdG8gZGV0ZXJtaW5lIGZpZWxkIG1hcHBpbmdzXG4gICAqIEByZXR1cm5zIHtLZXlWYWx1ZX0gVGhlIGNvbmZpZ3VyZWQgbWFwcGVyIG9iamVjdFxuICAgKiBAcHJpdmF0ZVxuICAgKiBAbWVtYmVyT2YgRmllbGRzZXRDb21wb25lbnRcbiAgICovXG4gIHByaXZhdGUgZ2V0TWFwcGVyKHZhbHVlOiBLZXlWYWx1ZSk6IEtleVZhbHVlIHtcbiAgICBpZighdGhpcy5waylcbiAgICAgIHRoaXMucGsgPSBPYmplY3Qua2V5cyh2YWx1ZSlbMF07XG4gICAgaWYoIU9iamVjdC5rZXlzKHRoaXMubWFwcGVyKS5sZW5ndGgpXG4gICAgICB0aGlzLm1hcHBlclsndGl0bGUnXSA9IHRoaXMucGs7XG4gICAgdGhpcy5tYXBwZXJbJ2luZGV4J10gPSBcImluZGV4XCI7XG4gICAgZm9yKGNvbnN0IGtleSBpbiB2YWx1ZSkge1xuICAgICAgaWYoT2JqZWN0LmtleXModGhpcy5tYXBwZXIpLmxlbmd0aCA+PSAyIHx8IE9iamVjdC5rZXlzKHRoaXMubWFwcGVyKS5sZW5ndGggPT09IE9iamVjdC5rZXlzKHZhbHVlKS5sZW5ndGgpXG4gICAgICAgIGJyZWFrO1xuICAgICAgaWYoIXRoaXMubWFwcGVyWyd0aXRsZSddKSB7XG4gICAgICAgIHRoaXMubWFwcGVyWyd0aXRsZSddID0ga2V5O1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy5tYXBwZXJbJ2Rlc2NyaXB0aW9uJ10gPSBrZXk7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiB0aGlzLm1hcHBlcjtcbiAgfVxufVxuIiwiXG5cbjxmaWVsZHNldFxuICAoZmllbGRzZXRBZGRHcm91cEV2ZW50KT1cImhhbmRsZUNyZWF0ZUl0ZW0oJGV2ZW50KVwiXG4gIChmaWVsZHNldFJlbW92ZUdyb3VwRXZlbnQpPVwiaGFuZGxlUmVtb3ZlSXRlbSh1bmRlZmluZWQsICRldmVudClcIlxuICBbY2xhc3NdPVwiJ2RjZi1maWVsZHNldCAnICsgb3BlcmF0aW9uXCJcbiAgI2NvbXBvbmVudD5cbiAgPGlvbi1hY2NvcmRpb24tZ3JvdXAgW25nQ2xhc3NdPVwieydvcGVuJzogaXNPcGVuLCAnaGFzVmFsaWRhdGlvbkVycm9ycyc6IGhhc1ZhbGlkYXRpb25FcnJvcnN9XCIgICh2YWxpZGF0aW9uRXJyb3JFdmVudCk9XCJoYW5kbGVWYWxpZGF0aW9uRXJyb3IoJGV2ZW50KVwiICNhY2NvcmRpb25Db21wb25lbnQ+XG4gICAgPGlvbi1hY2NvcmRpb24gdmFsdWU9XCJvcGVuXCI+XG4gICAgICA8aW9uLWl0ZW0gc2xvdD1cImhlYWRlclwiIChjbGljayk9XCJoYW5kbGVBY2NvcmRpb25Ub2dnbGUoJGV2ZW50KVwiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwiZGNmLWdyaWQgZGNmLWdyaWQtY29sbGFwc2UgZGNmLWZsZXggZGNmLWZsZXgtbWlkZGxlIGRjZi13aWR0aC0xLTFcIj5cbiAgICAgICAgICA8ZGl2IGNsYXNzPVwiZGNmLXdpZHRoLWV4cGFuZFwiPlxuICAgICAgICAgICAgPGxlZ2VuZD57eyBuYW1lIHwgdHJhbnNsYXRlIH19PC9sZWdlbmQ+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgQGlmKCFpc1JlcXVpcmVkICYmIFsnY3JlYXRlJywgJ3VwZGF0ZSddLmluY2x1ZGVzKG9wZXJhdGlvbikpIHtcbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJkY2Ytd2lkdGgtYXV0byBkY2YtZGVsZXRlXCI+XG4gICAgICAgICAgICAgIDxpb24tYnV0dG9uIGZpbGw9XCJjbGVhclwiIHNpemU9XCJzbWFsbFwiIChjbGljayk9XCJoYW5kbGVSZW1vdmVDb21wb25lbnQoJGV2ZW50KVwiPlxuICAgICAgICAgICAgICAgIDxpb24taWNvbiBuYW1lPVwidHJhc2gtb3V0bGluZVwiIGNvbG9yPVwiZGFya1wiIHNsb3Q9XCJpY29uLW9ubHlcIj48L2lvbi1pY29uPlxuICAgICAgICAgICAgICA8L2lvbi1idXR0b24+XG4gICAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICB9XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9pb24taXRlbT5cbiAgICAgIDxkaXYgc2xvdD1cImNvbnRlbnRcIiBbYXR0ci5hcmlhLWhpZGRlbl09XCIhaXNPcGVuXCI+XG4gICAgICAgIEBpZihtdWx0aXBsZSAmJiBpdGVtcy5sZW5ndGgpIHtcbiAgICAgICAgICA8aW9uLWxpc3QgY2xhc3M9XCJkY2YtZmllbGRzLWxpc3RcIj5cbiAgICAgICAgICAgIDxpb24tcmVvcmRlci1ncm91cCBbZm9ybUdyb3VwXT1cImZvcm1Hcm91cC5wYXJlbnRcIiBbZGlzYWJsZWRdPVwidXBkYXRpbmdJdGVtXCIgKGlvbkl0ZW1SZW9yZGVyKT1cImhhbmRsZVJlb3JkZXJJdGVtcygkYW55KCRldmVudCkpXCIgI2FjY29yZGlvbkNvbXBvbmVudD5cbiAgICAgICAgICAgICAgQGZvcihpdGVtIG9mIGl0ZW1zOyB0cmFjayBpdGVtLmluZGV4KSB7XG4gICAgICAgICAgICAgICAgPGlvbi1pdGVtIFtuZ0NsYXNzXT1cInsnbm90LXVuaXF1ZSc6IGl0ZW0udGl0bGUgPT09IGlzVW5pcXVlRXJyb3J9XCIgbGluZXM9XCJmdWxsXCIgW2J1dHRvbl09XCJmYWxzZVwiIFtuZ0NsYXNzXT1cInsndXBkYXRpbmcnOiB1cGRhdGluZ0l0ZW0/Lltwa10gPT09IGl0ZW0udGl0bGV9XCI+XG4gICAgICAgICAgICAgICAgICBAaWYoaXRlbXM/Lmxlbmd0aCA+IDEgJiYgIXVwZGF0aW5nSXRlbSkge1xuICAgICAgICAgICAgICAgICAgICA8aW9uLXJlb3JkZXIgc2xvdD1cInN0YXJ0XCI+XG4gICAgICAgICAgICAgICAgICAgICAgPGlvbi1pY29uIG5hbWU9XCJzd2FwLXZlcnRpY2FsLW91dGxpbmVcIj48L2lvbi1pY29uPlxuICAgICAgICAgICAgICAgICAgICA8L2lvbi1yZW9yZGVyPlxuICAgICAgICAgICAgICAgICAgfSBAZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIDxkaXYgc2xvdD1cInN0YXJ0XCI+XG4gICAgICAgICAgICAgICAgICAgICAgPGlvbi1pY29uIGNsYXNzPVwiZGNmLXJlb3JkZXItZGlzYWJsZWRcIiBzaXplPVwic21hbGxcIiBuYW1lPVwic3dhcC12ZXJ0aWNhbC1vdXRsaW5lXCIgZGlzYWJsZWQ+PC9pb24taWNvbj5cbiAgICAgICAgICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICA8aW9uLWxhYmVsIFtjb2xvcl09XCJpdGVtLnRpdGxlID09PSBpc1VuaXF1ZUVycm9yID8gJ2RhbmdlcicgOiAnJ1wiPnt7IGl0ZW0uaW5kZXggfX0uIHt7IGl0ZW0udGl0bGUgfX1cbiAgICAgICAgICAgICAgICAgICAgQGlmKGl0ZW0uZGVzY3JpcHRpb24/Lmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgICAgICAgICAgICA8YnIgLz5cbiAgICAgICAgICAgICAgICAgICAgICA8aW9uLXRleHQgY2xhc3M9XCJkY2Ytc3VidGl0bGVcIj57e2l0ZW0uZGVzY3JpcHRpb259fTwvaW9uLXRleHQ+XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgIDwvaW9uLWxhYmVsPlxuICAgICAgICAgICAgICAgICAgQGlmKCF1cGRhdGluZ0l0ZW0gfHwgdXBkYXRpbmdJdGVtPy5bcGtdICE9PSBpdGVtLnRpdGxlKSB7XG4gICAgICAgICAgICAgICAgICAgIDxpb24tYnV0dG9uIGZpbGw9XCJjbGVhclwiIHNpemU9XCJzbWFsbFwiIChjbGljayk9XCJoYW5kbGVVcGRhdGVJdGVtKGl0ZW0udGl0bGUsICRpbmRleClcIj5cbiAgICAgICAgICAgICAgICAgICAgICA8aW9uLWljb24gbmFtZT1cImNyZWF0ZS1vdXRsaW5lXCIgY29sb3I9XCJkYXJrXCIgc2xvdD1cImljb24tb25seVwiPjwvaW9uLWljb24+XG4gICAgICAgICAgICAgICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAgICAgQGlmKCF1cGRhdGluZ0l0ZW0pIHtcbiAgICAgICAgICAgICAgICAgICAgPGlvbi1idXR0b24gZmlsbD1cImNsZWFyXCIgc2l6ZT1cInNtYWxsXCIgKGNsaWNrKT1cImhhbmRsZVJlbW92ZUl0ZW0oaXRlbS50aXRsZSlcIj5cbiAgICAgICAgICAgICAgICAgICAgICA8aW9uLWljb24gbmFtZT1cInRyYXNoLW91dGxpbmVcIiBjb2xvcj1cImRhcmtcIiBzbG90PVwiaWNvbi1vbmx5XCI+PC9pb24taWNvbj5cbiAgICAgICAgICAgICAgICAgICAgPC9pb24tYnV0dG9uPlxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIDwvaW9uLWl0ZW0+XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIDwvaW9uLXJlb3JkZXItZ3JvdXA+XG4gICAgICAgICAgPC9pb24tbGlzdD5cbiAgICAgICAgfVxuXG4gICAgICAgIDxuZy1jb250ZW50PjwvbmctY29udGVudD5cbiAgICAgICAgQGlmKG11bHRpcGxlICYmIFsnY3JlYXRlJywgJ3VwZGF0ZSddLmluY2x1ZGVzKG9wZXJhdGlvbikpIHtcbiAgICAgICAgICBAaWYoaXNVbmlxdWVFcnJvcikge1xuICAgICAgICAgICAgPGRpdiBjbGFzcz1cImRjZi1ub3QtdW5pcXVlLWNvbnRhaW5lciBkY2YtYW5pbWF0aW9uIGRjZi1hbmltYXRpb24tYm90dG9tLXNtYWxsIGRjZi1hbmltYXRpb24tZmFzdFwiPlxuICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwiIGRjZi1ncmlkIGRjZi1ncmlkLWNvbGxhcHNlIGRjZi13aWR0aC0xLTEgXCI+XG4gICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cImRjZi1hdXRvXCIgW2F0dHIuc3R5bGVdPVwiJ21heC13aWR0aDogNTBweCdcIj5cbiAgICAgICAgICAgICAgICAgIDxpb24taWNvbiBuYW1lPVwiYWxlcnQtY2lyY2xlLW91dGxpbmVcIj48L2lvbi1pY29uPlxuICAgICAgICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJkY2Ytd2lkdGgtZXhwYW5kXCI+XG4gICAgICAgICAgICAgICAgICA8aW9uLXRleHQgY29sb3I9XCJkYW5nZXJcIiBjbGFzcz1cImRjZi10ZXh0LXNtYWxsXCI+e3sgbG9jYWxlICsgJy5ub3RfdW5pcXVlJyB8IHRyYW5zbGF0ZSA6IHsgdmFsdWU6IGlzVW5pcXVlRXJyb3IgfSB9fTwvaW9uLXRleHQ+XG4gICAgICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgfVxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJkY2YtbWFyZ2luLWJvdHRvbSBkY2YtZ3JpZCBkY2YtZ3JpZC1jb2xsYXBzZSBkY2YtZmxleFwiPlxuICAgICAgICAgICAgQGlmKHVwZGF0aW5nSXRlbSkge1xuICAgICAgICAgICAgICA8aW9uLWJ1dHRvbiBzaXplPVwic21hbGxcIiBmaWxsPVwiY2xlYXJcIiBjb2xvcj1cImRhbmdlclwiIChjbGljayk9XCJoYW5kbGVDYW5jZWxVcGRhdGVJdGVtKClcIj5cbiAgICAgICAgICAgICAgICB7eyBsb2NhbGUgKyAnLmNhbmNlbCcgfCB0cmFuc2xhdGUgfX1cbiAgICAgICAgICAgICAgPC9pb24tYnV0dG9uPlxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgPGlvbi1idXR0b24gc2l6ZT1cInNtYWxsXCIgZmlsbD1cImNsZWFyXCIgY29sb3I9XCJkYXJrXCIgKGNsaWNrKT1cImhhbmRsZUNyZWF0ZUl0ZW0oKVwiPlxuICAgICAgICAgICAgICA8aW9uLWljb24gbmFtZT1cImFkZC1vdXRsaW5lXCIgc2xvdD1cInN0YXJ0XCI+PC9pb24taWNvbj5cbiAgICAgICAgICAgICAge3tidXR0b25MYWJlbH19XG4gICAgICAgICAgICA8L2lvbi1idXR0b24+XG5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuXG4gICAgICA8L2Rpdj5cbiAgICA8L2lvbi1hY2NvcmRpb24+XG4gIDwvaW9uLWFjY29yZGlvbi1ncm91cD5cbjwvZmllbGRzZXQ+XG5cbiJdfQ==