@decaf-ts/for-angular 0.0.12 → 0.0.14

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 (143) hide show
  1. package/{dist/lib/components → components}/component-renderer/component-renderer.component.d.ts +0 -1
  2. package/{dist/lib/components → components}/crud-field/crud-field.component.d.ts +0 -1
  3. package/{dist/lib/components → components}/crud-form/constants.d.ts +0 -1
  4. package/{dist/lib/components → components}/crud-form/crud-form.component.d.ts +0 -1
  5. package/{dist/lib/components → components}/crud-form/types.d.ts +0 -1
  6. package/{dist/lib/components → components}/empty-state/empty-state.component.d.ts +0 -1
  7. package/{dist/lib/components → components}/fieldset/fieldset.component.d.ts +0 -1
  8. package/{dist/lib/components → components}/filter/filter.component.d.ts +0 -1
  9. package/{dist/lib/components → components}/for-angular-components.module.d.ts +0 -1
  10. package/{dist/lib/components → components}/index.d.ts +0 -1
  11. package/{dist/lib/components → components}/layout/layout.component.d.ts +0 -1
  12. package/{dist/lib/components → components}/list/constants.d.ts +0 -1
  13. package/{dist/lib/components → components}/list/list.component.d.ts +0 -1
  14. package/{dist/lib/components → components}/list-item/list-item.component.d.ts +0 -1
  15. package/{dist/lib/components → components}/model-renderer/model-renderer.component.d.ts +0 -1
  16. package/{dist/lib/components → components}/pagination/constants.d.ts +0 -1
  17. package/{dist/lib/components → components}/pagination/pagination.component.d.ts +0 -1
  18. package/{dist/lib/components → components}/searchbar/searchbar.component.d.ts +0 -1
  19. package/{dist/lib/directives → directives}/collapsable.directive.d.ts +0 -1
  20. package/{dist/lib/directives → directives}/index.d.ts +0 -1
  21. package/{dist/lib/engine → engine}/DynamicModule.d.ts +0 -1
  22. package/{dist/lib/engine → engine}/NgxBaseComponent.d.ts +0 -1
  23. package/{dist/lib/engine → engine}/NgxCrudFormField.d.ts +0 -1
  24. package/{dist/lib/engine → engine}/NgxFormService.d.ts +0 -1
  25. package/{dist/lib/engine → engine}/NgxRenderingEngine.d.ts +0 -1
  26. package/{dist/lib/engine → engine}/NgxRenderingEngine2.d.ts +0 -1
  27. package/{dist/lib/engine → engine}/ValidatorFactory.d.ts +0 -1
  28. package/{dist/lib/engine → engine}/constants.d.ts +0 -1
  29. package/{dist/lib/engine → engine}/decorators.d.ts +0 -1
  30. package/{dist/lib/engine → engine}/index.d.ts +0 -1
  31. package/{dist/lib/engine → engine}/types.d.ts +0 -1
  32. package/esm2022/components/component-renderer/component-renderer.component.mjs +313 -0
  33. package/esm2022/components/crud-field/crud-field.component.mjs +301 -0
  34. package/esm2022/components/crud-form/crud-form.component.mjs +139 -0
  35. package/esm2022/components/empty-state/empty-state.component.mjs +348 -0
  36. package/esm2022/components/fieldset/fieldset.component.mjs +225 -0
  37. package/esm2022/components/filter/filter.component.mjs +689 -0
  38. package/esm2022/components/for-angular-components.module.mjs +71 -0
  39. package/esm2022/components/layout/layout.component.mjs +176 -0
  40. package/esm2022/components/list/list.component.mjs +1236 -0
  41. package/esm2022/components/list-item/list-item.component.mjs +408 -0
  42. package/esm2022/components/model-renderer/model-renderer.component.mjs +138 -0
  43. package/esm2022/components/pagination/pagination.component.mjs +323 -0
  44. package/esm2022/components/searchbar/searchbar.component.mjs +493 -0
  45. package/esm2022/directives/collapsable.directive.mjs +28 -0
  46. package/esm2022/engine/NgxBaseComponent.mjs +539 -0
  47. package/esm2022/for-angular.module.mjs +118 -0
  48. package/{dist/lib/fesm2022 → fesm2022}/decaf-ts-for-angular.mjs +621 -1805
  49. package/fesm2022/decaf-ts-for-angular.mjs.map +1 -0
  50. package/{dist/lib/for-angular.module.d.ts → for-angular.module.d.ts} +0 -1
  51. package/{dist/lib/helpers → helpers}/index.d.ts +0 -1
  52. package/{dist/lib/helpers → helpers}/utils.d.ts +0 -1
  53. package/{dist/lib/index.d.ts → index.d.ts} +0 -1
  54. package/{dist/lib/interfaces.d.ts → interfaces.d.ts} +0 -1
  55. package/package.json +19 -95
  56. package/{dist/lib/public-apis.d.ts → public-apis.d.ts} +0 -1
  57. package/LICENSE.md +0 -659
  58. package/dist/for-angular/cli/cli-module.d.ts +0 -22
  59. package/dist/for-angular/cli/cli-module.d.ts.map +0 -1
  60. package/dist/for-angular/cli/cli-module.js +0 -131
  61. package/dist/for-angular/cli/cli-module.js.map +0 -1
  62. package/dist/lib/README.md +0 -92
  63. package/dist/lib/components/component-renderer/component-renderer.component.d.ts.map +0 -1
  64. package/dist/lib/components/crud-field/crud-field.component.d.ts.map +0 -1
  65. package/dist/lib/components/crud-form/constants.d.ts.map +0 -1
  66. package/dist/lib/components/crud-form/crud-form.component.d.ts.map +0 -1
  67. package/dist/lib/components/crud-form/types.d.ts.map +0 -1
  68. package/dist/lib/components/empty-state/empty-state.component.d.ts.map +0 -1
  69. package/dist/lib/components/fieldset/fieldset.component.d.ts.map +0 -1
  70. package/dist/lib/components/filter/filter.component.d.ts.map +0 -1
  71. package/dist/lib/components/for-angular-components.module.d.ts.map +0 -1
  72. package/dist/lib/components/index.d.ts.map +0 -1
  73. package/dist/lib/components/layout/layout.component.d.ts.map +0 -1
  74. package/dist/lib/components/list/constants.d.ts.map +0 -1
  75. package/dist/lib/components/list/list.component.d.ts.map +0 -1
  76. package/dist/lib/components/list-item/list-item.component.d.ts.map +0 -1
  77. package/dist/lib/components/model-renderer/model-renderer.component.d.ts.map +0 -1
  78. package/dist/lib/components/pagination/constants.d.ts.map +0 -1
  79. package/dist/lib/components/pagination/pagination.component.d.ts.map +0 -1
  80. package/dist/lib/components/searchbar/searchbar.component.d.ts.map +0 -1
  81. package/dist/lib/decaf-ts-for-angular.d.ts.map +0 -1
  82. package/dist/lib/directives/collapsable.directive.d.ts.map +0 -1
  83. package/dist/lib/directives/index.d.ts.map +0 -1
  84. package/dist/lib/engine/DynamicModule.d.ts.map +0 -1
  85. package/dist/lib/engine/NgxBaseComponent.d.ts.map +0 -1
  86. package/dist/lib/engine/NgxCrudFormField.d.ts.map +0 -1
  87. package/dist/lib/engine/NgxFormService.d.ts.map +0 -1
  88. package/dist/lib/engine/NgxRenderingEngine.d.ts.map +0 -1
  89. package/dist/lib/engine/NgxRenderingEngine2.d.ts.map +0 -1
  90. package/dist/lib/engine/ValidatorFactory.d.ts.map +0 -1
  91. package/dist/lib/engine/constants.d.ts.map +0 -1
  92. package/dist/lib/engine/decorators.d.ts.map +0 -1
  93. package/dist/lib/engine/index.d.ts.map +0 -1
  94. package/dist/lib/engine/types.d.ts.map +0 -1
  95. package/dist/lib/esm2022/components/component-renderer/component-renderer.component.mjs +0 -359
  96. package/dist/lib/esm2022/components/crud-field/crud-field.component.mjs +0 -456
  97. package/dist/lib/esm2022/components/crud-form/crud-form.component.mjs +0 -272
  98. package/dist/lib/esm2022/components/empty-state/empty-state.component.mjs +0 -410
  99. package/dist/lib/esm2022/components/fieldset/fieldset.component.mjs +0 -250
  100. package/dist/lib/esm2022/components/filter/filter.component.mjs +0 -906
  101. package/dist/lib/esm2022/components/for-angular-components.module.mjs +0 -72
  102. package/dist/lib/esm2022/components/layout/layout.component.mjs +0 -245
  103. package/dist/lib/esm2022/components/list/list.component.mjs +0 -1405
  104. package/dist/lib/esm2022/components/list-item/list-item.component.mjs +0 -612
  105. package/dist/lib/esm2022/components/model-renderer/model-renderer.component.mjs +0 -182
  106. package/dist/lib/esm2022/components/pagination/pagination.component.mjs +0 -364
  107. package/dist/lib/esm2022/components/searchbar/searchbar.component.mjs +0 -503
  108. package/dist/lib/esm2022/directives/collapsable.directive.mjs +0 -28
  109. package/dist/lib/esm2022/engine/NgxBaseComponent.mjs +0 -546
  110. package/dist/lib/esm2022/for-angular.module.mjs +0 -119
  111. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs.map +0 -1
  112. package/dist/lib/for-angular.module.d.ts.map +0 -1
  113. package/dist/lib/helpers/index.d.ts.map +0 -1
  114. package/dist/lib/helpers/utils.d.ts.map +0 -1
  115. package/dist/lib/interfaces.d.ts.map +0 -1
  116. package/dist/lib/public-apis.d.ts.map +0 -1
  117. /package/{dist/lib/assets → assets}/i18n/en.json +0 -0
  118. /package/{dist/lib/assets → assets}/images/angular-logo.svg +0 -0
  119. /package/{dist/lib/assets → assets}/images/decaf-logo-black.svg +0 -0
  120. /package/{dist/lib/assets → assets}/images/decaf-logo-lw.svg +0 -0
  121. /package/{dist/lib/assets → assets}/images/decaf-logo-white.svg +0 -0
  122. /package/{dist/lib/assets → assets}/images/decaf-logo.svg +0 -0
  123. /package/{dist/lib/esm2022 → esm2022}/components/crud-form/constants.mjs +0 -0
  124. /package/{dist/lib/esm2022 → esm2022}/components/crud-form/types.mjs +0 -0
  125. /package/{dist/lib/esm2022 → esm2022}/components/index.mjs +0 -0
  126. /package/{dist/lib/esm2022 → esm2022}/components/list/constants.mjs +0 -0
  127. /package/{dist/lib/esm2022 → esm2022}/components/pagination/constants.mjs +0 -0
  128. /package/{dist/lib/esm2022 → esm2022}/decaf-ts-for-angular.mjs +0 -0
  129. /package/{dist/lib/esm2022 → esm2022}/directives/index.mjs +0 -0
  130. /package/{dist/lib/esm2022 → esm2022}/engine/DynamicModule.mjs +0 -0
  131. /package/{dist/lib/esm2022 → esm2022}/engine/NgxCrudFormField.mjs +0 -0
  132. /package/{dist/lib/esm2022 → esm2022}/engine/NgxFormService.mjs +0 -0
  133. /package/{dist/lib/esm2022 → esm2022}/engine/NgxRenderingEngine.mjs +0 -0
  134. /package/{dist/lib/esm2022 → esm2022}/engine/NgxRenderingEngine2.mjs +0 -0
  135. /package/{dist/lib/esm2022 → esm2022}/engine/ValidatorFactory.mjs +0 -0
  136. /package/{dist/lib/esm2022 → esm2022}/engine/constants.mjs +0 -0
  137. /package/{dist/lib/esm2022 → esm2022}/engine/decorators.mjs +0 -0
  138. /package/{dist/lib/esm2022 → esm2022}/engine/index.mjs +0 -0
  139. /package/{dist/lib/esm2022 → esm2022}/engine/types.mjs +0 -0
  140. /package/{dist/lib/esm2022 → esm2022}/helpers/index.mjs +0 -0
  141. /package/{dist/lib/esm2022 → esm2022}/helpers/utils.mjs +0 -0
  142. /package/{dist/lib/esm2022 → esm2022}/interfaces.mjs +0 -0
  143. /package/{dist/lib/esm2022 → esm2022}/public-apis.mjs +0 -0
@@ -1,1405 +0,0 @@
1
- import { __decorate, __metadata } from "tslib";
2
- import { Component, EventEmitter, Output, Input, HostListener } from '@angular/core';
3
- import { IonInfiniteScroll, IonInfiniteScrollContent, IonItem, IonLabel, IonList, IonRefresher, IonRefresherContent, IonSkeletonText, IonText, IonThumbnail, IonLoading } from '@ionic/angular/standalone';
4
- import { debounceTime, Subject } from 'rxjs';
5
- import { OperationKeys } from '@decaf-ts/db-decorators';
6
- import { Model, Primitives } from '@decaf-ts/decorator-validation';
7
- import { Condition, OrderDirection } from '@decaf-ts/core';
8
- import { Dynamic, EventConstants, ComponentsTagNames } from '../../engine';
9
- import { ForAngularModule } from '../../for-angular.module';
10
- import { NgxBaseComponent } from '../../engine/NgxBaseComponent';
11
- import { stringToBoolean, formatDate, isValidDate } from '../../helpers';
12
- import { SearchbarComponent } from '../searchbar/searchbar.component';
13
- import { EmptyStateComponent } from '../empty-state/empty-state.component';
14
- import { ListItemComponent } from '../list-item/list-item.component';
15
- import { ComponentRendererComponent } from '../component-renderer/component-renderer.component';
16
- import { PaginationComponent } from '../pagination/pagination.component';
17
- import { ListComponentsTypes } from './constants';
18
- import { FilterComponent } from '../filter/filter.component';
19
- import * as i0 from "@angular/core";
20
- import * as i1 from "@ionic/angular/standalone";
21
- import * as i2 from "@angular/common";
22
- import * as i3 from "@ngx-translate/core";
23
- const _c0 = ["*"];
24
- const _c1 = (a0, a1, a2) => ({ item: a0, mapper: a1, route: a2 });
25
- function ListComponent_Conditional_0_Template(rf, ctx) { if (rf & 1) {
26
- const _r1 = i0.ɵɵgetCurrentView();
27
- i0.ɵɵelementStart(0, "ion-refresher", 2);
28
- i0.ɵɵlistener("ionRefresh", function ListComponent_Conditional_0_Template_ion_refresher_ionRefresh_0_listener($event) { i0.ɵɵrestoreView(_r1); const ctx_r1 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r1.handleRefresh($event)); });
29
- i0.ɵɵelement(1, "ion-refresher-content");
30
- i0.ɵɵelementEnd();
31
- } if (rf & 2) {
32
- i0.ɵɵproperty("pullFactor", 1)("pullMin", 100)("pullMax", 200);
33
- } }
34
- function ListComponent_Conditional_1_Conditional_0_Template(rf, ctx) { if (rf & 1) {
35
- const _r3 = i0.ɵɵgetCurrentView();
36
- i0.ɵɵelementStart(0, "ngx-decaf-filter", 5);
37
- i0.ɵɵlistener("filterEvent", function ListComponent_Conditional_1_Conditional_0_Template_ngx_decaf_filter_filterEvent_0_listener($event) { i0.ɵɵrestoreView(_r3); const ctx_r1 = i0.ɵɵnextContext(2); return i0.ɵɵresetView(ctx_r1.handleFilter($event)); })("searchEvent", function ListComponent_Conditional_1_Conditional_0_Template_ngx_decaf_filter_searchEvent_0_listener($event) { i0.ɵɵrestoreView(_r3); const ctx_r1 = i0.ɵɵnextContext(2); return i0.ɵɵresetView(ctx_r1.handleSearch($event)); });
38
- i0.ɵɵelementEnd();
39
- } if (rf & 2) {
40
- const ctx_r1 = i0.ɵɵnextContext(2);
41
- i0.ɵɵproperty("model", ctx_r1.model)("sortDirection", ctx_r1.sortDirection)("disableSort", ctx_r1.disableSort);
42
- } }
43
- function ListComponent_Conditional_1_Conditional_1_Template(rf, ctx) { if (rf & 1) {
44
- const _r4 = i0.ɵɵgetCurrentView();
45
- i0.ɵɵelementStart(0, "ngx-decaf-searchbar", 6);
46
- i0.ɵɵlistener("searchEvent", function ListComponent_Conditional_1_Conditional_1_Template_ngx_decaf_searchbar_searchEvent_0_listener($event) { i0.ɵɵrestoreView(_r4); const ctx_r1 = i0.ɵɵnextContext(2); return i0.ɵɵresetView(ctx_r1.handleSearch($event)); });
47
- i0.ɵɵelementEnd();
48
- } if (rf & 2) {
49
- i0.ɵɵproperty("emitEventToWindow", false)("debounce", 500);
50
- } }
51
- function ListComponent_Conditional_1_Template(rf, ctx) { if (rf & 1) {
52
- i0.ɵɵtemplate(0, ListComponent_Conditional_1_Conditional_0_Template, 1, 3, "ngx-decaf-filter", 3)(1, ListComponent_Conditional_1_Conditional_1_Template, 1, 2, "ngx-decaf-searchbar", 4);
53
- } if (rf & 2) {
54
- const ctx_r1 = i0.ɵɵnextContext();
55
- i0.ɵɵconditional(ctx_r1.model && ctx_r1.enableFilter ? 0 : 1);
56
- } }
57
- function ListComponent_Conditional_2_Conditional_2_For_1_Template(rf, ctx) { if (rf & 1) {
58
- const _r5 = i0.ɵɵgetCurrentView();
59
- i0.ɵɵelementStart(0, "ngx-decaf-component-renderer", 9);
60
- i0.ɵɵlistener("listenEvent", function ListComponent_Conditional_2_Conditional_2_For_1_Template_ngx_decaf_component_renderer_listenEvent_0_listener($event) { i0.ɵɵrestoreView(_r5); const ctx_r1 = i0.ɵɵnextContext(3); return i0.ɵɵresetView(ctx_r1.handleEvent($event)); });
61
- i0.ɵɵelementEnd();
62
- } if (rf & 2) {
63
- const child_r6 = ctx.$implicit;
64
- const ctx_r1 = i0.ɵɵnextContext(3);
65
- i0.ɵɵproperty("tag", ctx_r1.item.tag)("globals", i0.ɵɵpureFunction3(2, _c1, child_r6, ctx_r1.mapper, ctx_r1.route));
66
- } }
67
- function ListComponent_Conditional_2_Conditional_2_Template(rf, ctx) { if (rf & 1) {
68
- i0.ɵɵrepeaterCreate(0, ListComponent_Conditional_2_Conditional_2_For_1_Template, 1, 6, "ngx-decaf-component-renderer", 8, i0.ɵɵcomponentInstance().trackItemFn, true);
69
- } if (rf & 2) {
70
- const ctx_r1 = i0.ɵɵnextContext(2);
71
- i0.ɵɵrepeater(ctx_r1.items);
72
- } }
73
- function ListComponent_Conditional_2_Conditional_3_Template(rf, ctx) { if (rf & 1) {
74
- i0.ɵɵprojection(0);
75
- } }
76
- function ListComponent_Conditional_2_Conditional_4_Conditional_0_Template(rf, ctx) { if (rf & 1) {
77
- const _r7 = i0.ɵɵgetCurrentView();
78
- i0.ɵɵelementStart(0, "ngx-decaf-pagination", 12);
79
- i0.ɵɵlistener("clickEvent", function ListComponent_Conditional_2_Conditional_4_Conditional_0_Template_ngx_decaf_pagination_clickEvent_0_listener($event) { i0.ɵɵrestoreView(_r7); const ctx_r1 = i0.ɵɵnextContext(3); return i0.ɵɵresetView(ctx_r1.handlePaginate($event)); });
80
- i0.ɵɵelementEnd();
81
- } if (rf & 2) {
82
- const ctx_r1 = i0.ɵɵnextContext(3);
83
- i0.ɵɵproperty("totalPages", ctx_r1.pages)("current", ctx_r1.page);
84
- } }
85
- function ListComponent_Conditional_2_Conditional_4_Conditional_1_Template(rf, ctx) { if (rf & 1) {
86
- const _r8 = i0.ɵɵgetCurrentView();
87
- i0.ɵɵelementStart(0, "ion-infinite-scroll", 13);
88
- i0.ɵɵlistener("ionInfinite", function ListComponent_Conditional_2_Conditional_4_Conditional_1_Template_ion_infinite_scroll_ionInfinite_0_listener($event) { i0.ɵɵrestoreView(_r8); const ctx_r1 = i0.ɵɵnextContext(3); return i0.ɵɵresetView(ctx_r1.handleRefresh($event)); });
89
- i0.ɵɵelement(1, "ion-infinite-scroll-content", 14);
90
- i0.ɵɵelementEnd();
91
- } if (rf & 2) {
92
- const ctx_r1 = i0.ɵɵnextContext(3);
93
- i0.ɵɵclassMap((ctx_r1.searchValue == null ? null : ctx_r1.searchValue.length) ? "dcf-hidden" : "");
94
- i0.ɵɵproperty("position", ctx_r1.scrollPosition)("threshold", ctx_r1.scrollThreshold);
95
- i0.ɵɵadvance();
96
- i0.ɵɵproperty("loadingSpinner", ctx_r1.loadingSpinner)("loadingText", ctx_r1.loadingText);
97
- } }
98
- function ListComponent_Conditional_2_Conditional_4_Template(rf, ctx) { if (rf & 1) {
99
- i0.ɵɵtemplate(0, ListComponent_Conditional_2_Conditional_4_Conditional_0_Template, 1, 2, "ngx-decaf-pagination", 10)(1, ListComponent_Conditional_2_Conditional_4_Conditional_1_Template, 2, 6, "ion-infinite-scroll", 11);
100
- } if (rf & 2) {
101
- const ctx_r1 = i0.ɵɵnextContext(2);
102
- i0.ɵɵconditional(ctx_r1.pages > 0 && ctx_r1.type === "paginated" && !(ctx_r1.searchValue == null ? null : ctx_r1.searchValue.length) ? 0 : 1);
103
- } }
104
- function ListComponent_Conditional_2_Template(rf, ctx) { if (rf & 1) {
105
- i0.ɵɵelementStart(0, "ion-list", 7, 0);
106
- i0.ɵɵtemplate(2, ListComponent_Conditional_2_Conditional_2_Template, 2, 0)(3, ListComponent_Conditional_2_Conditional_3_Template, 1, 0);
107
- i0.ɵɵelementEnd();
108
- i0.ɵɵtemplate(4, ListComponent_Conditional_2_Conditional_4_Template, 2, 1);
109
- } if (rf & 2) {
110
- const ctx_r1 = i0.ɵɵnextContext();
111
- i0.ɵɵproperty("inset", ctx_r1.inset)("lines", ctx_r1.lines);
112
- i0.ɵɵadvance(2);
113
- i0.ɵɵconditional((ctx_r1.item == null ? null : ctx_r1.item.tag) ? 2 : 3);
114
- i0.ɵɵadvance(2);
115
- i0.ɵɵconditional(ctx_r1.loadMoreData ? 4 : -1);
116
- } }
117
- function ListComponent_Conditional_3_Conditional_0_ion_item_0_Template(rf, ctx) { if (rf & 1) {
118
- i0.ɵɵelementStart(0, "ion-item")(1, "ion-thumbnail", 16);
119
- i0.ɵɵelement(2, "ion-skeleton-text", 17);
120
- i0.ɵɵelementEnd();
121
- i0.ɵɵelementStart(3, "ion-label");
122
- i0.ɵɵelement(4, "ion-skeleton-text", 17);
123
- i0.ɵɵelementStart(5, "ion-text", 18);
124
- i0.ɵɵelement(6, "ion-skeleton-text", 17);
125
- i0.ɵɵelementEnd()()();
126
- } if (rf & 2) {
127
- i0.ɵɵadvance(2);
128
- i0.ɵɵproperty("animated", true);
129
- i0.ɵɵadvance(2);
130
- i0.ɵɵproperty("animated", true);
131
- i0.ɵɵadvance(2);
132
- i0.ɵɵproperty("animated", true);
133
- } }
134
- function ListComponent_Conditional_3_Conditional_0_Template(rf, ctx) { if (rf & 1) {
135
- i0.ɵɵtemplate(0, ListComponent_Conditional_3_Conditional_0_ion_item_0_Template, 7, 3, "ion-item", 15);
136
- } if (rf & 2) {
137
- const ctx_r1 = i0.ɵɵnextContext(2);
138
- i0.ɵɵproperty("ngForOf", ctx_r1.skeletonData);
139
- } }
140
- function ListComponent_Conditional_3_Conditional_1_Conditional_0_Template(rf, ctx) { if (rf & 1) {
141
- i0.ɵɵelement(0, "ngx-decaf-empty-state", 19);
142
- i0.ɵɵpipe(1, "translate");
143
- i0.ɵɵpipe(2, "translate");
144
- i0.ɵɵpipe(3, "translate");
145
- } if (rf & 2) {
146
- const ctx_r1 = i0.ɵɵnextContext(3);
147
- i0.ɵɵproperty("title", i0.ɵɵpipeBind1(1, 4, ctx_r1.locale + "." + ctx_r1.empty.title))("subtitle", i0.ɵɵpipeBind1(2, 6, ctx_r1.locale + "." + ctx_r1.empty.subtitle))("buttonText", ctx_r1.empty.showButton ? i0.ɵɵpipeBind1(3, 8, ctx_r1.locale + "." + ctx_r1.empty.button) : "")("buttonLink", ctx_r1.empty.showButton ? ctx_r1.empty.route : "");
148
- } }
149
- function ListComponent_Conditional_3_Conditional_1_Conditional_1_Template(rf, ctx) { if (rf & 1) {
150
- i0.ɵɵelement(0, "ngx-decaf-empty-state", 20);
151
- } if (rf & 2) {
152
- const ctx_r1 = i0.ɵɵnextContext(3);
153
- i0.ɵɵproperty("translatable", true)("searchValue", ctx_r1.searchValue);
154
- } }
155
- function ListComponent_Conditional_3_Conditional_1_Template(rf, ctx) { if (rf & 1) {
156
- i0.ɵɵtemplate(0, ListComponent_Conditional_3_Conditional_1_Conditional_0_Template, 4, 10, "ngx-decaf-empty-state", 19)(1, ListComponent_Conditional_3_Conditional_1_Conditional_1_Template, 1, 2, "ngx-decaf-empty-state", 20);
157
- } if (rf & 2) {
158
- const ctx_r1 = i0.ɵɵnextContext(2);
159
- i0.ɵɵconditional(!(ctx_r1.searchValue == null ? null : ctx_r1.searchValue.length) ? 0 : 1);
160
- } }
161
- function ListComponent_Conditional_3_Template(rf, ctx) { if (rf & 1) {
162
- i0.ɵɵtemplate(0, ListComponent_Conditional_3_Conditional_0_Template, 1, 1, "ion-item")(1, ListComponent_Conditional_3_Conditional_1_Template, 2, 1);
163
- } if (rf & 2) {
164
- const ctx_r1 = i0.ɵɵnextContext();
165
- i0.ɵɵconditional(ctx_r1.refreshing ? 0 : 1);
166
- } }
167
- /**
168
- * @description A versatile list component that supports various data display modes.
169
- * @summary This component provides a flexible way to display lists of data with support
170
- * for infinite scrolling, pagination, searching, and custom item rendering. It can fetch
171
- * data from various sources including models, functions, or direct data input.
172
- *
173
- * The component supports two main display types:
174
- * 1. Infinite scrolling - Loads more data as the user scrolls
175
- * 2. Pagination - Displays data in pages with navigation controls
176
- *
177
- * Additional features include:
178
- * - Pull-to-refresh functionality
179
- * - Search filtering
180
- * - Empty state customization
181
- * - Custom item rendering
182
- * - Event emission for interactions
183
- *
184
- * @mermaid
185
- * sequenceDiagram
186
- * participant U as User
187
- * participant L as ListComponent
188
- * participant D as Data Source
189
- * participant E as External Components
190
- *
191
- * U->>L: Initialize component
192
- * L->>L: ngOnInit()
193
- * L->>D: Request initial data
194
- * D-->>L: Return data
195
- * L->>L: Process and display data
196
- *
197
- * alt User scrolls (Infinite mode)
198
- * U->>L: Scroll to bottom
199
- * L->>D: Request more data
200
- * D-->>L: Return additional data
201
- * L->>L: Append to existing data
202
- * else User changes page (Paginated mode)
203
- * U->>L: Click page number
204
- * L->>L: handlePaginate()
205
- * L->>D: Request data for page
206
- * D-->>L: Return page data
207
- * L->>L: Replace displayed data
208
- * end
209
- *
210
- * alt User searches
211
- * U->>L: Enter search term
212
- * L->>L: handleSearch()
213
- * L->>D: Filter data by search term
214
- * D-->>L: Return filtered data
215
- * L->>L: Update displayed data
216
- * end
217
- *
218
- * alt User clicks item
219
- * U->>L: Click list item
220
- * L->>L: handleClick()
221
- * L->>E: Emit clickEvent
222
- * end
223
- *
224
- * @example
225
- * <ngx-decaf-list
226
- * [source]="dataSource"
227
- * [limit]="10"
228
- * [type]="'infinite'"
229
- * [showSearchbar]="true"
230
- * (clickEvent)="handleItemClick($event)"
231
- * (refreshEvent)="handleRefresh($event)">
232
- * </ngx-decaf-list>
233
- *
234
- * @extends {NgxBaseComponent}
235
- * @implements {OnInit}
236
- */
237
- let ListComponent = class ListComponent extends NgxBaseComponent {
238
- /**
239
- * @description Initializes a new instance of the ListComponent.
240
- * @summary Creates a new ListComponent and sets up the base component with the appropriate
241
- * component name. This constructor is called when Angular instantiates the component and
242
- * before any input properties are set. It passes the component name to the parent class
243
- * constructor to enable proper localization and component identification.
244
- *
245
- * The constructor is intentionally minimal, with most initialization logic deferred to
246
- * the ngOnInit lifecycle hook. This follows Angular best practices by keeping the constructor
247
- * focused on dependency injection and basic setup, while complex initialization that depends
248
- * on input properties is handled in ngOnInit.
249
- *
250
- * @memberOf ListComponent
251
- */
252
- constructor() {
253
- super("ListComponent");
254
- /**
255
- * @description The display mode for the list component.
256
- * @summary Determines how the list data is loaded and displayed. Options include:
257
- * - INFINITE: Loads more data as the user scrolls (infinite scrolling)
258
- * - PAGINATED: Displays data in pages with navigation controls
259
- *
260
- * @type {ListComponentsTypes}
261
- * @default ListComponentsTypes.INFINITE
262
- * @memberOf ListComponent
263
- */
264
- this.type = ListComponentsTypes.INFINITE;
265
- /**
266
- * @description Controls whether the component uses translation services.
267
- * @summary When set to true, the component will attempt to use translation services
268
- * for any text content. This allows for internationalization of the list component.
269
- *
270
- * @type {StringOrBoolean}
271
- * @default true
272
- * @memberOf ListComponent
273
- */
274
- this.translatable = true;
275
- /**
276
- * @description Controls the visibility of the search bar.
277
- * @summary When set to true, displays a search bar at the top of the list that allows
278
- * users to filter the list items. The search functionality works by filtering the
279
- * existing data or by triggering a new data fetch with search parameters.
280
- *
281
- * @type {StringOrBoolean}
282
- * @default true
283
- * @memberOf ListComponent
284
- */
285
- this.showSearchbar = true;
286
- /**
287
- * @description Direct data input for the list component.
288
- * @summary Provides a way to directly pass data to the list component instead of
289
- * fetching it from a source. When both data and source are provided, the component
290
- * will use the source to fetch data only if the data array is empty.
291
- *
292
- * @type {KeyValue[] | undefined}
293
- * @default undefined
294
- * @memberOf ListComponent
295
- */
296
- this.data = undefined;
297
- /**
298
- * @description The starting index for data fetching.
299
- * @summary Specifies the index from which to start fetching data. This is used
300
- * for pagination and infinite scrolling to determine which subset of data to load.
301
- *
302
- * @type {number}
303
- * @default 0
304
- * @memberOf ListComponent
305
- */
306
- this.start = 0;
307
- /**
308
- * @description The number of items to fetch per page or load operation.
309
- * @summary Determines how many items are loaded at once during pagination or
310
- * infinite scrolling. This affects the size of data chunks requested from the source.
311
- *
312
- * @type {number}
313
- * @default 10
314
- * @memberOf ListComponent
315
- */
316
- this.limit = 10;
317
- /**
318
- * @description Controls whether more data can be loaded.
319
- * @summary When set to true, the component will allow loading additional data
320
- * through infinite scrolling or pagination. When false, the component will not
321
- * attempt to load more data beyond what is initially displayed.
322
- *
323
- * @type {StringOrBoolean}
324
- * @default true
325
- * @memberOf ListComponent
326
- */
327
- this.loadMoreData = true;
328
- /**
329
- * @description The style of dividing lines between list items.
330
- * @summary Determines how dividing lines appear between list items. Options include:
331
- * - "inset": Lines are inset from the edges
332
- * - "full": Lines extend the full width
333
- * - "none": No dividing lines
334
- *
335
- * @type {"inset" | "full" | "none"}
336
- * @default "full"
337
- * @memberOf ListComponent
338
- */
339
- this.lines = "full";
340
- /**
341
- * @description Controls whether the list has inset styling.
342
- * @summary When set to true, the list will have inset styling with rounded corners
343
- * and margin around the edges. This creates a card-like appearance for the list.
344
- *
345
- * @type {StringOrBoolean}
346
- * @default false
347
- * @memberOf ListComponent
348
- */
349
- this.inset = false;
350
- /**
351
- * @description The threshold for triggering infinite scroll loading.
352
- * @summary Specifies how close to the bottom of the list the user must scroll
353
- * before the component triggers loading of additional data. This is expressed
354
- * as a percentage of the list height.
355
- *
356
- * @type {string}
357
- * @default "15%"
358
- * @memberOf ListComponent
359
- */
360
- this.scrollThreshold = "15%";
361
- /**
362
- * @description The position where new items are added during infinite scrolling.
363
- * @summary Determines whether new items are added to the top or bottom of the list
364
- * when loading more data through infinite scrolling.
365
- *
366
- * @type {"bottom" | "top"}
367
- * @default "bottom"
368
- * @memberOf ListComponent
369
- */
370
- this.scrollPosition = "bottom";
371
- /**
372
- * @description Controls the visibility of the pull-to-refresh feature.
373
- * @summary When set to true, enables the pull-to-refresh functionality that allows
374
- * users to refresh the list data by pulling down from the top of the list.
375
- *
376
- * @type {StringOrBoolean}
377
- * @default true
378
- * @memberOf ListComponent
379
- */
380
- this.showRefresher = true;
381
- /**
382
- * @description The type of spinner to display during loading operations.
383
- * @summary Specifies the visual style of the loading spinner shown during data
384
- * fetching operations. Uses Ionic's predefined spinner types.
385
- *
386
- * @type {SpinnerTypes}
387
- * @default "circular"
388
- * @memberOf ListComponent
389
- */
390
- this.loadingSpinner = "circular";
391
- // /**
392
- // * @description Query parameters for data fetching.
393
- // * @summary Specifies additional query parameters to use when fetching data from
394
- // * the source. This can be provided as a string (JSON) or a direct object.
395
- // *
396
- // * @type {string | KeyValue | undefined}
397
- // * @memberOf ListComponent
398
- // */
399
- // @Input()
400
- // query?: string | KeyValue;
401
- /**
402
- * @description Controls whether the filtering functionality is enabled.
403
- * @summary When set to true, enables the filter component that allows users to create
404
- * complex search criteria with multiple field filters, conditions, and values.
405
- * When false, disables the filter interface entirely.
406
- *
407
- * @type {StringOrBoolean}
408
- * @default true
409
- * @memberOf ListComponent
410
- */
411
- this.enableFilter = true;
412
- /**
413
- * @description Sorting parameters for data fetching.
414
- * @summary Specifies how the fetched data should be sorted. This can be provided
415
- * as a string (field name with optional direction) or a direct object.
416
- *
417
- * @type {string | KeyValue | undefined}
418
- * @memberOf ListComponent
419
- */
420
- this.sortDirection = OrderDirection.DSC;
421
- /**
422
- * @description Controls whether sorting functionality is disabled.
423
- * @summary When set to true, disables the sort controls and prevents users from
424
- * changing the sort order or field. The list will maintain its default or
425
- * programmatically set sort configuration without user interaction.
426
- *
427
- * @type {StringOrBoolean}
428
- * @default false
429
- * @memberOf ListComponent
430
- */
431
- this.disableSort = false;
432
- /**
433
- * @description Icon to display when the list is empty.
434
- * @summary Specifies the icon shown in the empty state when no data is available.
435
- * This can be any icon name supported by the application's icon system.
436
- *
437
- * @type {string | undefined}
438
- * @default 'ti-database-exclamation'
439
- * @memberOf ListComponent
440
- */
441
- this.emptyIcon = 'ti-database-exclamation';
442
- /**
443
- * @description Configuration for the empty state display.
444
- * @summary Customizes how the empty state is displayed when no data is available.
445
- * This includes the title, subtitle, button text, icon, and navigation link.
446
- *
447
- * @type {Partial<IListEmptyResult>}
448
- * @default {
449
- * title: 'empty.title',
450
- * subtitle: 'empty.subtitle',
451
- * showButton: false,
452
- * icon: 'alert-circle-outline',
453
- * buttonText: 'locale.empty.button',
454
- * link: ''
455
- * }
456
- * @memberOf ListComponent
457
- */
458
- this.empty = {
459
- title: 'empty.title',
460
- subtitle: 'empty.subtitle',
461
- showButton: false,
462
- icon: 'alert-circle-outline',
463
- buttonText: 'locale.empty.button',
464
- link: ''
465
- };
466
- /**
467
- * @description The current page number in paginated mode.
468
- * @summary Tracks which page is currently being displayed when the component
469
- * is in paginated mode. This is used for pagination controls and data fetching.
470
- *
471
- * @type {number}
472
- * @default 1
473
- * @memberOf ListComponent
474
- */
475
- this.page = 1;
476
- /**
477
- * @description Indicates whether a refresh operation is in progress.
478
- * @summary When true, the component is currently fetching new data. This is used
479
- * to control loading indicators and prevent duplicate refresh operations from
480
- * being triggered simultaneously.
481
- *
482
- * @type {boolean}
483
- * @default false
484
- * @memberOf ListComponent
485
- */
486
- this.refreshing = false;
487
- /**
488
- * @description Array used for rendering skeleton loading placeholders.
489
- * @summary Contains placeholder items that are displayed during data loading.
490
- * The length of this array determines how many skeleton items are shown.
491
- *
492
- * @type {string[]}
493
- * @default new Array(2)
494
- * @memberOf ListComponent
495
- */
496
- this.skeletonData = new Array(2);
497
- /**
498
- * @description The last page number that was displayed.
499
- * @summary Keeps track of the previously displayed page number, which is useful
500
- * for handling navigation and search operations in paginated mode.
501
- *
502
- * @type {number}
503
- * @default 1
504
- * @memberOf ListComponent
505
- */
506
- this.lastPage = 1;
507
- /**
508
- * @description Event emitter for refresh operations.
509
- * @summary Emits an event when the list data is refreshed, either through pull-to-refresh
510
- * or programmatic refresh. The event includes the refreshed data and component information.
511
- *
512
- * @type {EventEmitter<BaseCustomEvent>}
513
- * @memberOf ListComponent
514
- */
515
- this.refreshEvent = new EventEmitter();
516
- /**
517
- * @description Event emitter for item click interactions.
518
- * @summary Emits an event when a list item is clicked. The event includes the data
519
- * of the clicked item, allowing parent components to respond to the interaction.
520
- *
521
- * @type {EventEmitter<KeyValue>}
522
- * @memberOf ListComponent
523
- */
524
- this.clickEvent = new EventEmitter();
525
- /**
526
- * @description Subject for debouncing click events.
527
- * @summary Uses RxJS Subject to collect click events and emit them after a debounce
528
- * period. This prevents multiple rapid clicks from triggering multiple events.
529
- *
530
- * @private
531
- * @type {Subject<CustomEvent | ListItemCustomEvent | RendererCustomEvent>}
532
- * @memberOf ListComponent
533
- */
534
- this.clickItemSubject = new Subject();
535
- /**
536
- * @description Subject for debouncing repository observation events.
537
- * @summary RxJS Subject that collects repository change events and emits them after
538
- * a debounce period. This prevents multiple rapid repository changes from triggering
539
- * multiple list refresh operations, improving performance and user experience.
540
- *
541
- * @private
542
- * @type {Subject<any>}
543
- * @memberOf ListComponent
544
- */
545
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
546
- this.observerSubjet = new Subject();
547
- /**
548
- * @description Observer object for repository change notifications.
549
- * @summary Implements the Observer interface to receive notifications when the
550
- * underlying data repository changes. This enables automatic list updates when
551
- * data is created, updated, or deleted through the repository.
552
- *
553
- * @private
554
- * @type {Observer}
555
- * @memberOf ListComponent
556
- */
557
- this.observer = { refresh: async (...args) => this.observeRepository(...args) };
558
- }
559
- /**
560
- * @description Initializes the component after Angular sets the input properties.
561
- * @summary Sets up the component by initializing event subscriptions, processing boolean
562
- * inputs, and loading the initial data. This method prepares the component for user
563
- * interaction by ensuring all properties are properly initialized and data is loaded.
564
- *
565
- * @returns {Promise<void>}
566
- *
567
- * @mermaid
568
- * sequenceDiagram
569
- * participant A as Angular Lifecycle
570
- * participant L as ListComponent
571
- * participant D as Data Source
572
- *
573
- * A->>L: ngOnInit()
574
- * L->>L: Set up click event debouncing
575
- * L->>L: Process boolean inputs
576
- * L->>L: Configure component based on inputs
577
- * L->>L: refresh()
578
- * L->>D: Request initial data
579
- * D-->>L: Return data
580
- * L->>L: Process and display data
581
- * L->>L: Configure empty state if needed
582
- * L->>L: initialize()
583
- *
584
- * @memberOf ListComponent
585
- */
586
- async ngOnInit() {
587
- this.clickItemSubject.pipe(debounceTime(100)).subscribe(event => this.clickEventEmit(event));
588
- this.observerSubjet.pipe(debounceTime(100)).subscribe(args => this.handleObserveEvent(args[0], args[1], args[2]));
589
- this.enableFilter = stringToBoolean(this.enableFilter);
590
- this.limit = Number(this.limit);
591
- this.start = Number(this.start);
592
- this.inset = stringToBoolean(this.inset);
593
- this.showRefresher = stringToBoolean(this.showRefresher);
594
- this.loadMoreData = stringToBoolean(this.loadMoreData);
595
- this.showSearchbar = stringToBoolean(this.showSearchbar);
596
- this.disableSort = stringToBoolean(this.disableSort);
597
- if (typeof this.item?.['tag'] === 'boolean' && this.item?.['tag'] === true)
598
- this.item['tag'] = ComponentsTagNames.LIST_ITEM;
599
- await this.refresh();
600
- if (this.operations.includes(OperationKeys.CREATE) && this.route)
601
- this.empty.link = `${this.route}/${OperationKeys.CREATE}`;
602
- this.initialize();
603
- if (this.model instanceof Model && this._repository)
604
- this._repository.observe(this.observer);
605
- }
606
- /**
607
- * @description Cleans up resources when the component is destroyed.
608
- * @summary Performs cleanup operations when the component is being removed from the DOM.
609
- * This includes clearing references to models and data to prevent memory leaks.
610
- *
611
- * @returns {void}
612
- * @memberOf ListComponent
613
- */
614
- ngOnDestroy() {
615
- if (this._repository)
616
- this._repository.unObserve(this.observer);
617
- this.data = this.model = this._repository = this.paginator = undefined;
618
- }
619
- /**
620
- * @description Handles repository observation events with debouncing.
621
- * @summary Processes repository change notifications and routes them appropriately.
622
- * For CREATE events with a UID, handles them immediately. For other events,
623
- * passes them to the debounced observer subject to prevent excessive updates.
624
- *
625
- * @param {...unknown[]} args - The repository event arguments including table, event type, and UID
626
- * @returns {Promise<void>}
627
- * @memberOf ListComponent
628
- */
629
- async observeRepository(...args) {
630
- const [table, event, uid] = args;
631
- if (event === OperationKeys.CREATE && !!uid)
632
- return this.handleObserveEvent(table, event, uid);
633
- return this.observerSubjet.next(args);
634
- }
635
- /**
636
- * @description Handles specific repository events and updates the list accordingly.
637
- * @summary Processes repository change events (CREATE, UPDATE, DELETE) and performs
638
- * the appropriate list operations. This includes adding new items, updating existing
639
- * ones, or removing deleted items from the list display.
640
- *
641
- * @param {string} table - The table/model name that changed
642
- * @param {OperationKeys} event - The type of operation (CREATE, UPDATE, DELETE)
643
- * @param {string | number} uid - The unique identifier of the affected item
644
- * @returns {Promise<void>}
645
- * @memberOf ListComponent
646
- */
647
- async handleObserveEvent(table, event, uid) {
648
- if (event === OperationKeys.CREATE) {
649
- if (uid) {
650
- await this.handleCreate(uid);
651
- }
652
- else {
653
- await this.refresh(true);
654
- }
655
- }
656
- else {
657
- if (event === OperationKeys.UPDATE)
658
- await this.handleUpdate(uid);
659
- if (event === OperationKeys.DELETE)
660
- this.handleDelete(uid);
661
- this.refreshEventEmit();
662
- }
663
- }
664
- /**
665
- * @description Function for tracking items in the list.
666
- * @summary Provides a tracking function for the `*ngFor` directive in the component template.
667
- * This function is used to identify and control the rendering of items in the list,
668
- * preventing duplicate or unnecessary rendering.
669
- *
670
- * The `trackItemFn` function takes two parameters: `index` (the index of the item in the list)
671
- * and `item` (the actual item from the list). It returns the tracking key, which in this case
672
- * is the union of the `uid` of the item with the model name.
673
- *
674
- * @param {number} index - The index of the item in the list.
675
-
676
- * @param {KeyValue | string | number} item - The actual item from the list.
677
- * @returns {string | number} The tracking key for the item.
678
- * @memberOf ListComponent
679
- */
680
- trackItemFn(index, item) {
681
- return `${item?.['uid'] || item?.[this.pk]}-${index}`;
682
- }
683
- /**
684
- * Handles the create event from the repository.
685
- *
686
- * @param {string | number} uid - The ID of the item to create.
687
- * @returns {Promise<void>} A promise that resolves when the item is created and added to the list.
688
- */
689
- async handleCreate(uid) {
690
- const result = await this._repository?.read(uid);
691
- const item = this.mapResults([result])[0];
692
- this.items = this.data = [item, ...this.items || []];
693
- }
694
- /**
695
- * @description Handles the update event from the repository.
696
- * @summary Updates the list item with the specified ID based on the new data.
697
- *
698
- * @param {string | number} uid - The ID of the item to update
699
- * @returns {Promise<void>}
700
- * @private
701
- * @memberOf ListComponent
702
- */
703
- async handleUpdate(uid) {
704
- const item = this.itemMapper(await this._repository?.read(uid) || {}, this.mapper);
705
- this.data = [];
706
- for (const key in this.items) {
707
- const child = this.items[key];
708
- if (child['uid'] === item['uid']) {
709
- this.items[key] = Object.assign({}, child, item);
710
- break;
711
- }
712
- }
713
- setTimeout(() => {
714
- this.data = [...this.items];
715
- }, 0);
716
- }
717
- /**
718
- * @description Removes an item from the list by ID.
719
- * @summary Filters out an item with the specified ID from the data array and
720
- * refreshes the list display. This is typically used after a delete operation.
721
- *
722
- * @param {string} uid - The ID of the item to delete
723
- * @param {string} pk - The primary key field name
724
- * @returns {Promise<void>}
725
- *
726
- * @memberOf ListComponent
727
- */
728
- handleDelete(uid, pk) {
729
- if (!pk)
730
- pk = this.pk;
731
- this.items = this.data?.filter((item) => item['uid'] !== uid) || [];
732
- }
733
- /**
734
- * @description Handles click events from list items.
735
- * @summary Listens for global ListItemClickEvent events and passes them to the
736
- * debounced click subject. This allows the component to respond to clicks on
737
- * list items regardless of where they originate from.
738
- *
739
- * @param {ListItemCustomEvent | RendererCustomEvent} event - The click event
740
- * @returns {void}
741
- *
742
- * @memberOf ListComponent
743
- */
744
- handleClick(event) {
745
- this.clickItemSubject.next(event);
746
- }
747
- /**
748
- * @description Handles search events from the search bar.
749
- * @summary Processes search queries from the search bar component, updating the
750
- * displayed data based on the search term. The behavior differs between infinite
751
- * and paginated modes to provide the best user experience for each mode.
752
- *
753
- * @param {string | undefined} value - The search term or undefined to clear search
754
- * @returns {Promise<void>}
755
- *
756
- * @mermaid
757
- * flowchart TD
758
- * A[Search Event] --> B{Type is Infinite?}
759
- * B -->|Yes| C[Disable loadMoreData]
760
- * B -->|No| D[Enable loadMoreData]
761
- * C --> E{Search value undefined?}
762
- * E -->|Yes| F[Enable loadMoreData]
763
- * E -->|No| G[Store search value]
764
- * D --> G
765
- * F --> H[Reset page to 1]
766
- * G --> I[Refresh data]
767
- * H --> I
768
- *
769
- * @memberOf ListComponent
770
- */
771
- async handleSearch(value) {
772
- if (this.type === ListComponentsTypes.INFINITE) {
773
- this.loadMoreData = false;
774
- if (value === undefined) {
775
- this.loadMoreData = true;
776
- this.page = 1;
777
- }
778
- this.searchValue = value;
779
- await this.refresh(true);
780
- }
781
- else {
782
- this.loadMoreData = true;
783
- this.searchValue = value;
784
- if (value === undefined)
785
- this.page = this.lastPage;
786
- await this.refresh(true);
787
- }
788
- }
789
- /**
790
- * @description Handles filter events from the filter component.
791
- * @summary Processes filter queries from the filter component and applies them
792
- * to the list data. This method acts as a bridge between the filter component
793
- * and the search functionality, converting filter queries into search operations.
794
- *
795
- * @param {IFilterQuery | undefined} value - The filter query object or undefined to clear filters
796
- * @returns {Promise<void>}
797
- * @memberOf ListComponent
798
- */
799
- async handleFilter(value) {
800
- await this.handleSearch(value);
801
- }
802
- /**
803
- * @description Clears the current search and resets the list.
804
- * @summary Convenience method that clears the search by calling handleSearch
805
- * with undefined. This resets the list to show all data without filtering.
806
- *
807
- * @returns {Promise<void>}
808
- * @memberOf ListComponent
809
- */
810
- async clearSearch() {
811
- await this.handleSearch(undefined);
812
- }
813
- /**
814
- * @description Emits a refresh event with the current data.
815
- * @summary Creates and emits a refresh event containing the current list data.
816
- * This notifies parent components that the list data has been refreshed.
817
- *
818
- * @param {KeyValue[]} [data] - Optional data to include in the event
819
- * @returns {void}
820
- *
821
- * @memberOf ListComponent
822
- */
823
- refreshEventEmit(data) {
824
- if (!data)
825
- data = this.items;
826
- this.skeletonData = new Array(data?.length || 2);
827
- this.refreshEvent.emit({
828
- name: EventConstants.REFRESH_EVENT,
829
- data: data || [],
830
- component: this.componentName
831
- });
832
- }
833
- /**
834
- * @description Emits a click event for a list item.
835
- * @summary Processes and emits a click event when a list item is clicked.
836
- * This extracts the relevant data from the event and passes it to parent components.
837
- *
838
- * @private
839
- * @param {ListItemCustomEvent | RendererCustomEvent} event - The click event
840
- * @returns {void}
841
- *
842
- * @memberOf ListComponent
843
- */
844
- clickEventEmit(event) {
845
- this.clickEvent.emit(event);
846
- }
847
- /**
848
- * @description Refreshes the list data from the configured source.
849
- * @summary This method handles both initial data loading and subsequent refresh operations,
850
- * including pull-to-refresh and infinite scrolling. It manages the data fetching process,
851
- * updates the component's state, and handles pagination or infinite scrolling logic based
852
- * on the component's configuration.
853
- *
854
- * The method performs the following steps:
855
- * 1. Sets the refreshing flag to indicate a data fetch is in progress
856
- * 2. Calculates the appropriate start and limit values based on pagination settings
857
- * 3. Fetches data from the appropriate source (model or request)
858
- * 4. Updates the component's data and emits a refresh event
859
- * 5. Handles pagination or infinite scrolling state updates
860
- * 6. Completes any provided event (like InfiniteScrollCustomEvent)
861
- *
862
- * @param {InfiniteScrollCustomEvent | RefresherCustomEvent | boolean} event - The event that triggered the refresh,
863
- * or a boolean flag indicating if this is a forced refresh
864
- * @returns {Promise<void>} A promise that resolves when the refresh operation is complete
865
- *
866
- * @mermaid
867
- * sequenceDiagram
868
- * participant L as ListComponent
869
- * participant D as Data Source
870
- * participant E as Event System
871
- *
872
- * L->>L: refresh(event)
873
- * L->>L: Set refreshing flag
874
- * L->>L: Calculate start and limit
875
- * alt Using model
876
- * L->>D: getFromModel(force, start, limit)
877
- * D-->>L: Return data
878
- * else Using request
879
- * L->>D: getFromRequest(force, start, limit)
880
- * D-->>L: Return data
881
- * end
882
- * L->>E: refreshEventEmit()
883
- * alt Infinite scrolling mode
884
- * L->>L: Check if reached last page
885
- * alt Last page reached
886
- * L->>L: Complete scroll event
887
- * L->>L: Disable loadMoreData
888
- * else More pages available
889
- * L->>L: Increment page number
890
- * L->>L: Complete scroll event after delay
891
- * end
892
- * else Paginated mode
893
- * L->>L: Clear refreshing flag after delay
894
- * end
895
- *
896
- * @memberOf ListComponent
897
- */
898
- async refresh(event = false) {
899
- // if(typeof force !== 'boolean' && force.type === EventConstants.BACK_BUTTON_NAVIGATION) {
900
- // const {refresh} = (force as CustomEvent).detail;
901
- // if(!refresh)
902
- // return false;
903
- // }
904
- this.refreshing = true;
905
- const start = this.page > 1 ? (this.page - 1) * this.limit : this.start;
906
- const limit = (this.page * (this.limit > 12 ? 12 : this.limit));
907
- this.data = !this.model ?
908
- await this.getFromRequest(!!event, start, limit)
909
- : await this.getFromModel(!!event);
910
- this.refreshEventEmit();
911
- if (this.type === ListComponentsTypes.INFINITE) {
912
- if (this.page === this.pages) {
913
- if (event?.target)
914
- event.target.complete();
915
- this.loadMoreData = false;
916
- }
917
- else {
918
- this.page += 1;
919
- this.refreshing = false;
920
- setTimeout(() => {
921
- if (event?.target && event?.type !== EventConstants.BACK_BUTTON_NAVIGATION)
922
- event.target.complete();
923
- }, 200);
924
- }
925
- }
926
- else {
927
- setTimeout(() => {
928
- this.refreshing = false;
929
- }, 200);
930
- }
931
- }
932
- /**
933
- * @description Handles pagination events from the pagination component.
934
- * @summary Processes pagination events by updating the current page number and
935
- * refreshing the list data to display the selected page. This method is called
936
- * when a user interacts with the pagination controls to navigate between pages.
937
- *
938
- * @param {PaginationCustomEvent} event - The pagination event containing page information
939
- * @returns {void}
940
- *
941
- * @memberOf ListComponent
942
- */
943
- handlePaginate(event) {
944
- const { page } = event.data;
945
- this.page = page;
946
- this.refresh(true);
947
- }
948
- /**
949
- * @description Handles pull-to-refresh events from the refresher component.
950
- * @summary Processes refresh events triggered by the user pulling down on the list
951
- * or by programmatic refresh requests. This method refreshes the list data and
952
- * completes the refresher animation when the data is loaded.
953
- *
954
- * @param {InfiniteScrollCustomEvent | CustomEvent} [event] - The refresh event
955
- * @returns {Promise<void>} A promise that resolves when the refresh operation is complete
956
- *
957
- * @memberOf ListComponent
958
- */
959
- async handleRefresh(event) {
960
- await this.refresh(event || true);
961
- if (event instanceof CustomEvent)
962
- setTimeout(() => {
963
- // Any calls to load data go here
964
- event.target.complete();
965
- }, 400);
966
- }
967
- /**
968
- * @description Filters data based on a search string.
969
- * @summary Processes the current data array to find items that match the provided
970
- * search string. This uses the arrayQueryByString utility to perform the filtering
971
- * across all properties of the items.
972
- *
973
- * @param {KeyValue[]} results - The array of items to search through
974
- * @param {string} search - The search string to filter by
975
- * @returns {KeyValue[]} A promise that resolves to the filtered array of items
976
- *
977
- * @memberOf ListComponent
978
- */
979
- parseSearchResults(results, search) {
980
- return results.filter((item) => Object.values(item).some(value => value.toString().toLowerCase().includes(search?.toLowerCase())));
981
- }
982
- /**
983
- * @description Fetches data from a request source.
984
- * @summary Retrieves data from the configured source function or URL, processes it,
985
- * and updates the component's data state. This method handles both initial data loading
986
- * and subsequent refresh operations when using an external data source rather than a model.
987
- *
988
- * @param {boolean} force - Whether to force a refresh even if data already exists
989
- * @param {number} start - The starting index for pagination
990
- * @param {number} limit - The maximum number of items to retrieve
991
- * @returns {Promise<KeyValue[]>} A promise that resolves to the fetched data
992
- *
993
- * @memberOf ListComponent
994
- */
995
- async getFromRequest(force = false, start, limit) {
996
- let request = [];
997
- if (!this.data?.length || force || this.searchValue?.length || !!this.searchValue) {
998
- // (self.data as ListItem[]) = [];
999
- if (!this.searchValue?.length && !this.searchValue) {
1000
- if (!this.source && !this.data?.length) {
1001
- this.logger.info('No data and source passed to infinite list');
1002
- return [];
1003
- }
1004
- if (this.source instanceof Function)
1005
- request = await this.source();
1006
- if (!Array.isArray(request))
1007
- request = request?.['response']?.['data'] || request?.['results'] || [];
1008
- this.data = [...await this.parseResult(request)];
1009
- if (this.data?.length)
1010
- this.items = this.type === ListComponentsTypes.INFINITE ?
1011
- (this.items || []).concat([...this.data.slice(start, limit)]) : [...request.slice(start, limit)];
1012
- }
1013
- else {
1014
- this.data = this.parseSearchResults(this.data, this.searchValue);
1015
- this.items = this.data;
1016
- }
1017
- }
1018
- if (this.loadMoreData && this.type === ListComponentsTypes.PAGINATED)
1019
- this.getMoreData(this.data?.length || 0);
1020
- return this.data || [];
1021
- }
1022
- /**
1023
- * @description Fetches data from a model source.
1024
- * @summary Retrieves data from the configured model using its pagination or find methods,
1025
- * processes it, and updates the component's data state. This method handles both initial
1026
- * data loading and subsequent refresh operations when using a model as the data source.
1027
- *
1028
- * @param {boolean} force - Whether to force a refresh even if data already exists
1029
- * @param {number} start - The starting index for pagination
1030
- * @param {number} limit - The maximum number of items to retrieve
1031
- * @returns {Promise<KeyValue[]>} A promise that resolves to the fetched data
1032
- *
1033
- * @memberOf ListComponent
1034
- */
1035
- async getFromModel(force = false) {
1036
- let data = [...this.data || []];
1037
- let request = [];
1038
- // getting model repository
1039
- if (!this._repository)
1040
- this._repository = this.repository;
1041
- const repo = this._repository;
1042
- if (!this.data?.length || force || this.searchValue?.length || !!this.searchValue) {
1043
- try {
1044
- if (!this.searchValue?.length && !this.searchValue) {
1045
- this.data = [];
1046
- // const rawQuery = this.parseQuery(self.model as Repository<Model>, start, limit);
1047
- // request = this.parseResult(await (this.model as any)?.paginate(start, limit));
1048
- if (!this.paginator) {
1049
- this.paginator = await repo
1050
- .select()
1051
- .orderBy([this.pk, this.sortDirection])
1052
- .paginate(this.limit);
1053
- }
1054
- request = await this.parseResult(this.paginator);
1055
- }
1056
- else {
1057
- if (!this.indexes)
1058
- this.indexes = (Object.values(this.mapper) || [this.pk]);
1059
- const condition = this.parseConditions(this.searchValue);
1060
- request = await this.parseResult(await repo.query(condition, (this.sortBy || this.pk), this.sortDirection));
1061
- data = [];
1062
- }
1063
- data = this.type === ListComponentsTypes.INFINITE ? [...(data).concat(request)] : [...request];
1064
- }
1065
- catch (error) {
1066
- this.logger.error(error?.message || `Unable to find ${this.model} on registry. Return empty array from component`);
1067
- }
1068
- }
1069
- if (data?.length) {
1070
- if (this.searchValue) {
1071
- this.items = [...data];
1072
- if (this.items?.length <= this.limit)
1073
- this.loadMoreData = false;
1074
- }
1075
- else {
1076
- this.items = [...data];
1077
- }
1078
- }
1079
- if (this.type === ListComponentsTypes.PAGINATED && this.paginator)
1080
- this.getMoreData(this.paginator.total);
1081
- return data || [];
1082
- }
1083
- /**
1084
- * @description Converts search values or filter queries into database conditions.
1085
- * @summary Transforms search input or complex filter queries into Condition objects
1086
- * that can be used for database querying. Handles both simple string/number searches
1087
- * across indexed fields and complex filter queries with multiple criteria.
1088
- *
1089
- * For simple searches (string/number):
1090
- * - Creates conditions that search across all indexed fields
1091
- * - Uses equality for numeric values and regex for string values
1092
- * - Combines conditions with OR logic to search multiple fields
1093
- *
1094
- * For complex filter queries:
1095
- * - Processes each filter item with its specific condition type
1096
- * - Supports Equal, Not Equal, Contains, Not Contains, Greater Than, Less Than
1097
- * - Updates sort configuration based on the filter query
1098
- * - Combines multiple filter conditions with OR logic
1099
- *
1100
- * @param {string | number | IFilterQuery} value - The search value or filter query object
1101
- * @returns {Condition<Model>} A Condition object for database querying
1102
- * @memberOf ListComponent
1103
- */
1104
- parseConditions(value) {
1105
- let _condition;
1106
- if (typeof value === Primitives.STRING || typeof value === Primitives.NUMBER) {
1107
- _condition = Condition.attribute(this.pk).eq(!isNaN(value) ? Number(value) : value);
1108
- for (const index of this.indexes) {
1109
- if (index === this.pk)
1110
- continue;
1111
- let orCondition;
1112
- if (!isNaN(value)) {
1113
- orCondition = Condition.attribute(index).eq(Number(value));
1114
- }
1115
- else {
1116
- orCondition = Condition.attribute(index).regexp(value);
1117
- }
1118
- _condition = _condition.or(orCondition);
1119
- }
1120
- }
1121
- else {
1122
- const { query, sort } = value;
1123
- _condition = Condition.attribute(this.pk).dif('null');
1124
- if (query?.length)
1125
- _condition = undefined;
1126
- (query || []).forEach((item) => {
1127
- const { value, condition, index } = item;
1128
- let val = value;
1129
- if (index === this.pk || !isNaN(val))
1130
- val = Number(val);
1131
- let orCondition;
1132
- switch (condition) {
1133
- case "Equal":
1134
- orCondition = Condition.attribute(index).eq(val);
1135
- break;
1136
- case "Not Equal":
1137
- orCondition = Condition.attribute(index).dif(val);
1138
- break;
1139
- case "Not Contains":
1140
- orCondition = !Condition.attribute(index).regexp(new RegExp(`^(?!.*${val}).*$`));
1141
- break;
1142
- case "Contains":
1143
- orCondition = Condition.attribute(index).regexp(val);
1144
- break;
1145
- case "Greater Than":
1146
- orCondition = Condition.attribute(index).gte(val);
1147
- break;
1148
- case "Less Than":
1149
- orCondition = Condition.attribute(index).lte(val);
1150
- break;
1151
- }
1152
- _condition = (!_condition ?
1153
- orCondition : _condition.and(orCondition));
1154
- });
1155
- this.sortBy = sort?.value || this.pk;
1156
- this.sortDirection = sort?.direction || this.sortDirection;
1157
- }
1158
- console.log(_condition);
1159
- return _condition;
1160
- }
1161
- /**
1162
- * @description Processes query results into a standardized format.
1163
- * @summary Handles different result formats from various data sources, extracting
1164
- * pagination information when available and applying any configured data mapping.
1165
- * This ensures consistent data structure regardless of the source.
1166
- *
1167
- * @protected
1168
- * @param {KeyValue[] | Paginator} result - The raw query result
1169
- * @returns {KeyValue[]} The processed array of items
1170
- *
1171
- * @memberOf ListComponent
1172
- */
1173
- async parseResult(result) {
1174
- if (!Array.isArray(result) && ('page' in result && 'total' in result)) {
1175
- const paginator = result;
1176
- result = await paginator.page(this.page);
1177
- // TODO: Chage for result.total;
1178
- this.getMoreData(paginator.total);
1179
- }
1180
- else {
1181
- this.getMoreData(result?.length || 0);
1182
- }
1183
- return (Object.keys(this.mapper || {}).length) ?
1184
- this.mapResults(result) : result;
1185
- }
1186
- /**
1187
- * @description Updates pagination state based on data length.
1188
- * @summary Calculates whether more data is available and how many pages exist
1189
- * based on the total number of items and the configured limit per page.
1190
- * This information is used to control pagination UI and infinite scrolling behavior.
1191
- *
1192
- * @param {number} length - The total number of items available
1193
- * @returns {void}
1194
- *
1195
- * @memberOf ListComponent
1196
- */
1197
- getMoreData(length) {
1198
- if (this.type === ListComponentsTypes.INFINITE) {
1199
- if (this.paginator)
1200
- length = length * this.limit;
1201
- if (length <= this.limit) {
1202
- this.loadMoreData = false;
1203
- }
1204
- else {
1205
- this.pages = Math.floor(length / this.limit);
1206
- if ((this.pages * this.limit) < length)
1207
- this.pages += 1;
1208
- if (this.pages === 1)
1209
- this.loadMoreData = false;
1210
- }
1211
- }
1212
- else {
1213
- this.pages = length;
1214
- if (this.pages === 1)
1215
- this.loadMoreData = false;
1216
- }
1217
- }
1218
- /**
1219
- * @description Maps a single item using the configured mapper.
1220
- * @summary Transforms a data item according to the mapping configuration,
1221
- * extracting nested properties and formatting values as needed. This allows
1222
- * the component to display data in a format different from how it's stored.
1223
- *
1224
- * @protected
1225
- * @param {KeyValue} item - The item to map
1226
- * @param {KeyValue} mapper - The mapping configuration
1227
- * @param {KeyValue} [props] - Additional properties to include
1228
- * @returns {KeyValue} The mapped item
1229
- *
1230
- * @memberOf ListComponent
1231
- */
1232
- itemMapper(item, mapper, props) {
1233
- return Object.entries(mapper).reduce((accum, [key, value]) => {
1234
- const arrayValue = value.split('.');
1235
- if (!value) {
1236
- accum[key] = value;
1237
- }
1238
- else {
1239
- if (arrayValue.length === 1) {
1240
- value = item?.[value] || value;
1241
- if (isValidDate(value))
1242
- value = `${formatDate(value)}`;
1243
- accum[key] = value;
1244
- }
1245
- else {
1246
- let val;
1247
- for (const _value of arrayValue)
1248
- val = !val
1249
- ? item[_value]
1250
- : (typeof val === 'string' ? JSON.parse(val) : val)[_value];
1251
- if (isValidDate(new Date(val)))
1252
- val = `${formatDate(val)}`;
1253
- accum[key] = val === null || val === undefined ? value : val;
1254
- }
1255
- }
1256
- return Object.assign({}, props || {}, accum);
1257
- }, {});
1258
- }
1259
- /**
1260
- * @description Maps all result items using the configured mapper.
1261
- * @summary Applies the itemMapper to each item in the result set, adding
1262
- * common properties like operations and route information. This transforms
1263
- * the raw data into the format expected by the list item components.
1264
- *
1265
- * @param {KeyValue[]} data - The array of items to map
1266
- * @returns {KeyValue[]} The array of mapped items
1267
- *
1268
- * @memberOf ListComponent
1269
- */
1270
- mapResults(data) {
1271
- if (!data || !data.length)
1272
- return [];
1273
- // passing uid as prop to mapper
1274
- this.mapper = { ...this.mapper, ...{ uid: this.pk } };
1275
- const props = Object.assign({
1276
- operations: this.operations,
1277
- route: this.route,
1278
- ...Object.keys(this.item).reduce((acc, key) => {
1279
- acc[key] = this.item[key];
1280
- return acc;
1281
- }, {}),
1282
- // ... (!this.item.render ? {} : Object.keys(this.item).reduce((acc: KeyValue, key: string) => {
1283
- // acc[key] = this.item[key as keyof IListItemProp];
1284
- // return acc;
1285
- // }, {}))
1286
- });
1287
- return data.reduce((accum, curr) => {
1288
- accum.push({ ...this.itemMapper(curr, this.mapper, props), ...{ pk: this.pk } });
1289
- return accum;
1290
- }, []);
1291
- }
1292
- static { this.ɵfac = function ListComponent_Factory(__ngFactoryType__) { return new (__ngFactoryType__ || ListComponent)(); }; }
1293
- static { this.ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: ListComponent, selectors: [["ngx-decaf-list"]], hostBindings: function ListComponent_HostBindings(rf, ctx) { if (rf & 1) {
1294
- i0.ɵɵlistener("ListItemClickEvent", function ListComponent_ListItemClickEvent_HostBindingHandler($event) { return ctx.handleClick($event); }, false, i0.ɵɵresolveWindow)("searchbarEvent", function ListComponent_searchbarEvent_HostBindingHandler($event) { return ctx.handleSearch($event); }, false, i0.ɵɵresolveWindow)("BackButtonNavigationEndEvent", function ListComponent_BackButtonNavigationEndEvent_HostBindingHandler($event) { return ctx.refresh($event); }, false, i0.ɵɵresolveWindow);
1295
- } }, inputs: { type: "type", translatable: "translatable", showSearchbar: "showSearchbar", data: "data", source: "source", start: "start", limit: "limit", loadMoreData: "loadMoreData", lines: "lines", inset: "inset", scrollThreshold: "scrollThreshold", scrollPosition: "scrollPosition", loadingText: "loadingText", showRefresher: "showRefresher", loadingSpinner: "loadingSpinner", enableFilter: "enableFilter", sortDirection: "sortDirection", sortBy: "sortBy", disableSort: "disableSort", emptyIcon: "emptyIcon", empty: "empty" }, outputs: { refreshEvent: "refreshEvent", clickEvent: "clickEvent" }, standalone: true, features: [i0.ɵɵInheritDefinitionFeature, i0.ɵɵStandaloneFeature], ngContentSelectors: _c0, decls: 4, vars: 3, consts: [["component", ""], ["slot", "fixed", 3, "pullFactor", "pullMin", "pullMax"], ["slot", "fixed", 3, "ionRefresh", "pullFactor", "pullMin", "pullMax"], [3, "model", "sortDirection", "disableSort"], [3, "emitEventToWindow", "debounce"], [3, "filterEvent", "searchEvent", "model", "sortDirection", "disableSort"], [3, "searchEvent", "emitEventToWindow", "debounce"], [3, "inset", "lines"], [3, "tag", "globals"], [3, "listenEvent", "tag", "globals"], [3, "totalPages", "current"], [3, "class", "position", "threshold"], [3, "clickEvent", "totalPages", "current"], [3, "ionInfinite", "position", "threshold"], [3, "loadingSpinner", "loadingText"], [4, "ngFor", "ngForOf"], ["slot", "start"], [3, "animated"], [1, "date", 2, "width", "20%"], [3, "title", "subtitle", "buttonText", "buttonLink"], ["icon", "search-outline", "ngClass", "empty-search", "title", "search.title", "subtitle", "search.subtitle", 3, "translatable", "searchValue"]], template: function ListComponent_Template(rf, ctx) { if (rf & 1) {
1296
- i0.ɵɵprojectionDef();
1297
- i0.ɵɵtemplate(0, ListComponent_Conditional_0_Template, 2, 3, "ion-refresher", 1)(1, ListComponent_Conditional_1_Template, 2, 1)(2, ListComponent_Conditional_2_Template, 5, 4)(3, ListComponent_Conditional_3_Template, 2, 1);
1298
- } if (rf & 2) {
1299
- i0.ɵɵconditional(ctx.showRefresher ? 0 : -1);
1300
- i0.ɵɵadvance();
1301
- i0.ɵɵconditional(ctx.showSearchbar ? 1 : -1);
1302
- i0.ɵɵadvance();
1303
- i0.ɵɵconditional((ctx.data == null ? null : ctx.data.length) ? 2 : 3);
1304
- } }, dependencies: [ForAngularModule, i1.IonText, i2.NgClass, i2.NgForOf, i3.TranslatePipe, IonRefresher,
1305
- PaginationComponent,
1306
- IonList,
1307
- IonItem,
1308
- IonThumbnail,
1309
- IonSkeletonText,
1310
- IonLabel,
1311
- IonRefresherContent,
1312
- IonInfiniteScroll,
1313
- IonInfiniteScrollContent,
1314
- SearchbarComponent,
1315
- EmptyStateComponent,
1316
- FilterComponent,
1317
- ComponentRendererComponent], styles: ["ion-infinite-scroll-content[_ngcontent-%COMP%] ion-spinner[_ngcontent-%COMP%]{--color: var(--ion-color-primary)}@media (max-width: 768px){#end[_ngcontent-%COMP%], [slot=end][_ngcontent-%COMP%]{display:none!important}}"] }); }
1318
- };
1319
- ListComponent = __decorate([
1320
- Dynamic(),
1321
- __metadata("design:paramtypes", [])
1322
- ], ListComponent);
1323
- export { ListComponent };
1324
- (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(ListComponent, [{
1325
- type: Component,
1326
- args: [{ selector: 'ngx-decaf-list', standalone: true, imports: [
1327
- ForAngularModule,
1328
- IonRefresher,
1329
- IonLoading,
1330
- PaginationComponent,
1331
- IonList,
1332
- IonItem,
1333
- IonThumbnail,
1334
- IonSkeletonText,
1335
- IonLabel,
1336
- IonText,
1337
- IonRefresherContent,
1338
- IonInfiniteScroll,
1339
- IonInfiniteScrollContent,
1340
- IonThumbnail,
1341
- IonSkeletonText,
1342
- SearchbarComponent,
1343
- EmptyStateComponent,
1344
- ListItemComponent,
1345
- FilterComponent,
1346
- ComponentRendererComponent
1347
- ], template: "\n@if(showRefresher) {\n <ion-refresher slot=\"fixed\" [pullFactor]=\"1\" [pullMin]=\"100\" [pullMax]=\"200\" (ionRefresh)=\"handleRefresh($event)\">\n <ion-refresher-content />\n </ion-refresher>\n}\n\n@if(showSearchbar) {\n @if(model && enableFilter) {\n <ngx-decaf-filter\n [model]=\"model\"\n [sortDirection]=\"sortDirection\"\n [disableSort]=\"disableSort\"\n (filterEvent)=\"handleFilter($event)\"\n (searchEvent)=\"handleSearch($event)\"\n />\n } @else {\n <ngx-decaf-searchbar [emitEventToWindow]=\"false\" [debounce]=\"500\" (searchEvent)=\"handleSearch($event)\" />\n }\n}\n\n@if(data?.length) {\n <ion-list [inset]=\"inset\" [lines]=\"lines\" #component>\n @if(item?.tag) {\n @for(child of items; track trackItemFn($index, child)) {\n <ngx-decaf-component-renderer\n [tag]=\"item.tag\"\n (listenEvent)=\"handleEvent($event)\"\n [globals]='{\n item: child,\n mapper: mapper,\n route: route\n }'>\n </ngx-decaf-component-renderer>\n }\n } @else {\n <ng-content></ng-content>\n }\n </ion-list>\n\n @if(loadMoreData) {\n @if(pages > 0 && type === 'paginated' && !searchValue?.length) {\n <ngx-decaf-pagination\n [totalPages]=\"pages\"\n [current]=\"page\"\n (clickEvent)=\"handlePaginate($event)\"\n />\n\n } @else {\n <ion-infinite-scroll\n [class]=\"searchValue?.length ? 'dcf-hidden' : ''\"\n [position]=\"scrollPosition\"\n [threshold]=\"scrollThreshold\"\n (ionInfinite)=\"handleRefresh($event)\">\n <ion-infinite-scroll-content [loadingSpinner]=\"loadingSpinner\" [loadingText]=\"loadingText\" />\n </ion-infinite-scroll>\n }\n }\n} @else {\n @if(refreshing) {\n <ion-item *ngFor=\"let skl of skeletonData\">\n <ion-thumbnail slot=\"start\">\n <ion-skeleton-text [animated]=\"true\"></ion-skeleton-text>\n </ion-thumbnail>\n <ion-label>\n <ion-skeleton-text [animated]=\"true\"></ion-skeleton-text>\n <ion-text class=\"date\" style=\"width: 20%;\"><ion-skeleton-text [animated]=\"true\"></ion-skeleton-text></ion-text>\n </ion-label>\n </ion-item>\n } @else {\n @if(!searchValue?.length) {\n <ngx-decaf-empty-state\n [title]=\"(locale + '.'+ empty.title) | translate\"\n [subtitle]=\"(locale + '.'+ empty.subtitle) | translate\"\n [buttonText]=\"empty.showButton ? (locale + '.'+ empty.button | translate) : ''\"\n [buttonLink]=\"empty.showButton ? empty.route : ''\"\n />\n } @else {\n <ngx-decaf-empty-state\n icon=\"search-outline\"\n ngClass=\"empty-search\"\n [translatable]=\"true\"\n title=\"search.title\"\n subtitle=\"search.subtitle\"\n [searchValue]=\"searchValue\"\n />\n }\n }\n}\n\n", styles: ["ion-infinite-scroll-content ion-spinner{--color: var(--ion-color-primary)}@media (max-width: 768px){#end,[slot=end]{display:none!important}}\n"] }]
1348
- }], () => [], { type: [{
1349
- type: Input
1350
- }], translatable: [{
1351
- type: Input
1352
- }], showSearchbar: [{
1353
- type: Input
1354
- }], data: [{
1355
- type: Input
1356
- }], source: [{
1357
- type: Input
1358
- }], start: [{
1359
- type: Input
1360
- }], limit: [{
1361
- type: Input
1362
- }], loadMoreData: [{
1363
- type: Input
1364
- }], lines: [{
1365
- type: Input
1366
- }], inset: [{
1367
- type: Input
1368
- }], scrollThreshold: [{
1369
- type: Input
1370
- }], scrollPosition: [{
1371
- type: Input
1372
- }], loadingText: [{
1373
- type: Input
1374
- }], showRefresher: [{
1375
- type: Input
1376
- }], loadingSpinner: [{
1377
- type: Input
1378
- }], enableFilter: [{
1379
- type: Input
1380
- }], sortDirection: [{
1381
- type: Input
1382
- }], sortBy: [{
1383
- type: Input
1384
- }], disableSort: [{
1385
- type: Input
1386
- }], emptyIcon: [{
1387
- type: Input
1388
- }], empty: [{
1389
- type: Input
1390
- }], refreshEvent: [{
1391
- type: Output
1392
- }], clickEvent: [{
1393
- type: Output
1394
- }], handleClick: [{
1395
- type: HostListener,
1396
- args: ['window:ListItemClickEvent', ['$event']]
1397
- }], handleSearch: [{
1398
- type: HostListener,
1399
- args: ['window:searchbarEvent', ['$event']]
1400
- }], refresh: [{
1401
- type: HostListener,
1402
- args: ['window:BackButtonNavigationEndEvent', ['$event']]
1403
- }] }); })();
1404
- (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassDebugInfo(ListComponent, { className: "ListComponent", filePath: "components/list/list.component.ts", lineNumber: 146 }); })();
1405
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlzdC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbGlzdC9saXN0LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9saXN0L2xpc3QuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQVUsWUFBWSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsWUFBWSxFQUFjLE1BQU0sZUFBZSxDQUFDO0FBRXpHLE9BQU8sRUFDTCxpQkFBaUIsRUFDakIsd0JBQXdCLEVBQ3hCLE9BQU8sRUFDUCxRQUFRLEVBQ1IsT0FBTyxFQUNQLFlBQVksRUFDWixtQkFBbUIsRUFDbkIsZUFBZSxFQUNmLE9BQU8sRUFDUCxZQUFZLEVBQ1osVUFBVSxFQUNYLE1BQU0sMkJBQTJCLENBQUM7QUFDbkMsT0FBTyxFQUFFLFlBQVksRUFBRSxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDN0MsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQ3hELE9BQU8sRUFBRSxLQUFLLEVBQUUsVUFBVSxFQUFFLE1BQU0sZ0NBQWdDLENBQUM7QUFDbkUsT0FBTyxFQUFFLFNBQVMsRUFBWSxjQUFjLEVBQWEsTUFBTSxnQkFBZ0IsQ0FBQztBQUNoRixPQUFPLEVBRUwsT0FBTyxFQUNQLGNBQWMsRUFDZCxrQkFBa0IsRUFLbkIsTUFBTSxjQUFjLENBQUM7QUFDdEIsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDNUQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDakUsT0FBTyxFQUNMLGVBQWUsRUFDZixVQUFVLEVBQ1YsV0FBVyxFQUNaLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGtDQUFrQyxD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O2tCQURKLEtBQUs7WUFhTixLQUFLO2tCQURKLEtBQUs7WUFjTixlQUFlO2tCQURkLEtBQUs7WUFhTixjQUFjO2tCQURiLEtBQUs7WUFZTixXQUFXO2tCQURWLEtBQUs7WUFhTixhQUFhO2tCQURaLEtBQUs7WUFhTixjQUFjO2tCQURiLEtBQUs7WUF5Qk4sWUFBWTtrQkFEWCxLQUFLO1lBWU4sYUFBYTtrQkFEWixLQUFLO1lBYU4sTUFBTTtrQkFETCxLQUFLO1lBZU4sV0FBVztrQkFEVixLQUFLO1lBY04sU0FBUztrQkFEUixLQUFLO1lBb0JOLEtBQUs7a0JBREosS0FBSztZQThHTixZQUFZO2tCQURYLE1BQU07WUFZUCxVQUFVO2tCQURULE1BQU07WUFpUlAsV0FBVztrQkFEVixZQUFZO21CQUFDLDJCQUEyQixFQUFFLENBQUMsUUFBUSxDQUFDO1lBOEIvQyxZQUFZO2tCQURqQixZQUFZO21CQUFDLHVCQUF1QixFQUFFLENBQUMsUUFBUSxDQUFDO1lBc0kzQyxPQUFPO2tCQURaLFlBQVk7bUJBQUMscUNBQXFDLEVBQUUsQ0FBQyxRQUFRLENBQUM7O2tGQWwwQnBELGFBQWEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIE9uSW5pdCwgRXZlbnRFbWl0dGVyLCBPdXRwdXQsIElucHV0LCBIb3N0TGlzdGVuZXIsIE9uRGVzdHJveSAgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEluZmluaXRlU2Nyb2xsQ3VzdG9tRXZlbnQsIFJlZnJlc2hlckN1c3RvbUV2ZW50LCBTcGlubmVyVHlwZXMgfSBmcm9tICdAaW9uaWMvYW5ndWxhcic7XG5pbXBvcnQge1xuICBJb25JbmZpbml0ZVNjcm9sbCxcbiAgSW9uSW5maW5pdGVTY3JvbGxDb250ZW50LFxuICBJb25JdGVtLFxuICBJb25MYWJlbCxcbiAgSW9uTGlzdCxcbiAgSW9uUmVmcmVzaGVyLFxuICBJb25SZWZyZXNoZXJDb250ZW50LFxuICBJb25Ta2VsZXRvblRleHQsXG4gIElvblRleHQsXG4gIElvblRodW1ibmFpbCxcbiAgSW9uTG9hZGluZ1xufSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IGRlYm91bmNlVGltZSwgU3ViamVjdCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgT3BlcmF0aW9uS2V5cyB9IGZyb20gJ0BkZWNhZi10cy9kYi1kZWNvcmF0b3JzJztcbmltcG9ydCB7IE1vZGVsLCBQcmltaXRpdmVzIH0gZnJvbSAnQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uJztcbmltcG9ydCB7IENvbmRpdGlvbiwgT2JzZXJ2ZXIsIE9yZGVyRGlyZWN0aW9uLCBQYWdpbmF0b3IgfSBmcm9tICdAZGVjYWYtdHMvY29yZSc7XG5pbXBvcnQge1xuICBCYXNlQ3VzdG9tRXZlbnQsXG4gIER5bmFtaWMsXG4gIEV2ZW50Q29uc3RhbnRzLFxuICBDb21wb25lbnRzVGFnTmFtZXMsXG4gIFJlbmRlcmVyQ3VzdG9tRXZlbnQsXG4gIFN0cmluZ09yQm9vbGVhbixcbiAgS2V5VmFsdWUsXG4gIExpc3RJdGVtQ3VzdG9tRXZlbnRcbn0gZnJvbSAnLi4vLi4vZW5naW5lJztcbmltcG9ydCB7IEZvckFuZ3VsYXJNb2R1bGUgfSBmcm9tICcuLi8uLi9mb3ItYW5ndWxhci5tb2R1bGUnO1xuaW1wb3J0IHsgTmd4QmFzZUNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2VuZ2luZS9OZ3hCYXNlQ29tcG9uZW50JztcbmltcG9ydCB7XG4gIHN0cmluZ1RvQm9vbGVhbixcbiAgZm9ybWF0RGF0ZSxcbiAgaXNWYWxpZERhdGVcbn0gZnJvbSAnLi4vLi4vaGVscGVycyc7XG5pbXBvcnQgeyBTZWFyY2hiYXJDb21wb25lbnQgfSBmcm9tICcuLi9zZWFyY2hiYXIvc2VhcmNoYmFyLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBFbXB0eVN0YXRlQ29tcG9uZW50IH0gZnJvbSAnLi4vZW1wdHktc3RhdGUvZW1wdHktc3RhdGUuY29tcG9uZW50JztcbmltcG9ydCB7IExpc3RJdGVtQ29tcG9uZW50IH0gZnJvbSAnLi4vbGlzdC1pdGVtL2xpc3QtaXRlbS5jb21wb25lbnQnO1xuaW1wb3J0IHsgQ29tcG9uZW50UmVuZGVyZXJDb21wb25lbnQgfSBmcm9tICcuLi9jb21wb25lbnQtcmVuZGVyZXIvY29tcG9uZW50LXJlbmRlcmVyLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBQYWdpbmF0aW9uQ29tcG9uZW50IH0gZnJvbSAnLi4vcGFnaW5hdGlvbi9wYWdpbmF0aW9uLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBQYWdpbmF0aW9uQ3VzdG9tRXZlbnQgfSBmcm9tICcuLi9wYWdpbmF0aW9uL2NvbnN0YW50cyc7XG5pbXBvcnQgeyBJTGlzdEVtcHR5UmVzdWx0LCBMaXN0Q29tcG9uZW50c1R5cGVzLCBEZWNhZlJlcG9zaXRvcnkgfSBmcm9tICcuL2NvbnN0YW50cyc7XG5pbXBvcnQgeyBGdW5jdGlvbkxpa2UsIElGaWx0ZXJRdWVyeSwgSUZpbHRlclF1ZXJ5SXRlbSB9IGZyb20gJy4uLy4uL2VuZ2luZS90eXBlcyc7XG5pbXBvcnQgeyBGaWx0ZXJDb21wb25lbnQgfSBmcm9tICcuLi9maWx0ZXIvZmlsdGVyLmNvbXBvbmVudCc7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEEgdmVyc2F0aWxlIGxpc3QgY29tcG9uZW50IHRoYXQgc3VwcG9ydHMgdmFyaW91cyBkYXRhIGRpc3BsYXkgbW9kZXMuXG4gKiBAc3VtbWFyeSBUaGlzIGNvbXBvbmVudCBwcm92aWRlcyBhIGZsZXhpYmxlIHdheSB0byBkaXNwbGF5IGxpc3RzIG9mIGRhdGEgd2l0aCBzdXBwb3J0XG4gKiBmb3IgaW5maW5pdGUgc2Nyb2xsaW5nLCBwYWdpbmF0aW9uLCBzZWFyY2hpbmcsIGFuZCBjdXN0b20gaXRlbSByZW5kZXJpbmcuIEl0IGNhbiBmZXRjaFxuICogZGF0YSBmcm9tIHZhcmlvdXMgc291cmNlcyBpbmNsdWRpbmcgbW9kZWxzLCBmdW5jdGlvbnMsIG9yIGRpcmVjdCBkYXRhIGlucHV0LlxuICpcbiAqIFRoZSBjb21wb25lbnQgc3VwcG9ydHMgdHdvIG1haW4gZGlzcGxheSB0eXBlczpcbiAqIDEuIEluZmluaXRlIHNjcm9sbGluZyAtIExvYWRzIG1vcmUgZGF0YSBhcyB0aGUgdXNlciBzY3JvbGxzXG4gKiAyLiBQYWdpbmF0aW9uIC0gRGlzcGxheXMgZGF0YSBpbiBwYWdlcyB3aXRoIG5hdmlnYXRpb24gY29udHJvbHNcbiAqXG4gKiBBZGRpdGlvbmFsIGZlYXR1cmVzIGluY2x1ZGU6XG4gKiAtIFB1bGwtdG8tcmVmcmVzaCBmdW5jdGlvbmFsaXR5XG4gKiAtIFNlYXJjaCBmaWx0ZXJpbmdcbiAqIC0gRW1wdHkgc3RhdGUgY3VzdG9taXphdGlvblxuICogLSBDdXN0b20gaXRlbSByZW5kZXJpbmdcbiAqIC0gRXZlbnQgZW1pc3Npb24gZm9yIGludGVyYWN0aW9uc1xuICpcbiAqIEBtZXJtYWlkXG4gKiBzZXF1ZW5jZURpYWdyYW1cbiAqICAgcGFydGljaXBhbnQgVSBhcyBVc2VyXG4gKiAgIHBhcnRpY2lwYW50IEwgYXMgTGlzdENvbXBvbmVudFxuICogICBwYXJ0aWNpcGFudCBEIGFzIERhdGEgU291cmNlXG4gKiAgIHBhcnRpY2lwYW50IEUgYXMgRXh0ZXJuYWwgQ29tcG9uZW50c1xuICpcbiAqICAgVS0+Pkw6IEluaXRpYWxpemUgY29tcG9uZW50XG4gKiAgIEwtPj5MOiBuZ09uSW5pdCgpXG4gKiAgIEwtPj5EOiBSZXF1ZXN0IGluaXRpYWwgZGF0YVxuICogICBELS0+Pkw6IFJldHVybiBkYXRhXG4gKiAgIEwtPj5MOiBQcm9jZXNzIGFuZCBkaXNwbGF5IGRhdGFcbiAqXG4gKiAgIGFsdCBVc2VyIHNjcm9sbHMgKEluZmluaXRlIG1vZGUpXG4gKiAgICAgVS0+Pkw6IFNjcm9sbCB0byBib3R0b21cbiAqICAgICBMLT4+RDogUmVxdWVzdCBtb3JlIGRhdGFcbiAqICAgICBELS0+Pkw6IFJldHVybiBhZGRpdGlvbmFsIGRhdGFcbiAqICAgICBMLT4+TDogQXBwZW5kIHRvIGV4aXN0aW5nIGRhdGFcbiAqICAgZWxzZSBVc2VyIGNoYW5nZXMgcGFnZSAoUGFnaW5hdGVkIG1vZGUpXG4gKiAgICAgVS0+Pkw6IENsaWNrIHBhZ2UgbnVtYmVyXG4gKiAgICAgTC0+Pkw6IGhhbmRsZVBhZ2luYXRlKClcbiAqICAgICBMLT4+RDogUmVxdWVzdCBkYXRhIGZvciBwYWdlXG4gKiAgICAgRC0tPj5MOiBSZXR1cm4gcGFnZSBkYXRhXG4gKiAgICAgTC0+Pkw6IFJlcGxhY2UgZGlzcGxheWVkIGRhdGFcbiAqICAgZW5kXG4gKlxuICogICBhbHQgVXNlciBzZWFyY2hlc1xuICogICAgIFUtPj5MOiBFbnRlciBzZWFyY2ggdGVybVxuICogICAgIEwtPj5MOiBoYW5kbGVTZWFyY2goKVxuICogICAgIEwtPj5EOiBGaWx0ZXIgZGF0YSBieSBzZWFyY2ggdGVybVxuICogICAgIEQtLT4+TDogUmV0dXJuIGZpbHRlcmVkIGRhdGFcbiAqICAgICBMLT4+TDogVXBkYXRlIGRpc3BsYXllZCBkYXRhXG4gKiAgIGVuZFxuICpcbiAqICAgYWx0IFVzZXIgY2xpY2tzIGl0ZW1cbiAqICAgICBVLT4+TDogQ2xpY2sgbGlzdCBpdGVtXG4gKiAgICAgTC0+Pkw6IGhhbmRsZUNsaWNrKClcbiAqICAgICBMLT4+RTogRW1pdCBjbGlja0V2ZW50XG4gKiAgIGVuZFxuICpcbiAqIEBleGFtcGxlXG4gKiA8bmd4LWRlY2FmLWxpc3RcbiAqICAgW3NvdXJjZV09XCJkYXRhU291cmNlXCJcbiAqICAgW2xpbWl0XT1cIjEwXCJcbiAqICAgW3R5cGVdPVwiJ2luZmluaXRlJ1wiXG4gKiAgIFtzaG93U2VhcmNoYmFyXT1cInRydWVcIlxuICogICAoY2xpY2tFdmVudCk9XCJoYW5kbGVJdGVtQ2xpY2soJGV2ZW50KVwiXG4gKiAgIChyZWZyZXNoRXZlbnQpPVwiaGFuZGxlUmVmcmVzaCgkZXZlbnQpXCI+XG4gKiA8L25neC1kZWNhZi1saXN0PlxuICpcbiAqIEBleHRlbmRzIHtOZ3hCYXNlQ29tcG9uZW50fVxuICogQGltcGxlbWVudHMge09uSW5pdH1cbiAqL1xuQER5bmFtaWMoKVxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnbmd4LWRlY2FmLWxpc3QnLFxuICB0ZW1wbGF0ZVVybDogJy4vbGlzdC5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL2xpc3QuY29tcG9uZW50LnNjc3MnXSxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW1xuICAgIEZvckFuZ3VsYXJNb2R1bGUsXG4gICAgSW9uUmVmcmVzaGVyLFxuICAgIElvbkxvYWRpbmcsXG4gICAgUGFnaW5hdGlvbkNvbXBvbmVudCxcbiAgICBJb25MaXN0LFxuICAgIElvbkl0ZW0sXG4gICAgSW9uVGh1bWJuYWlsLFxuICAgIElvblNrZWxldG9uVGV4dCxcbiAgICBJb25MYWJlbCxcbiAgICBJb25UZXh0LFxuICAgIElvblJlZnJlc2hlckNvbnRlbnQsXG4gICAgSW9uSW5maW5pdGVTY3JvbGwsXG4gICAgSW9uSW5maW5pdGVTY3JvbGxDb250ZW50LFxuICAgIElvblRodW1ibmFpbCxcbiAgICBJb25Ta2VsZXRvblRleHQsXG4gICAgU2VhcmNoYmFyQ29tcG9uZW50LFxuICAgIEVtcHR5U3RhdGVDb21wb25lbnQsXG4gICAgTGlzdEl0ZW1Db21wb25lbnQsXG4gICAgRmlsdGVyQ29tcG9uZW50LFxuICAgIENvbXBvbmVudFJlbmRlcmVyQ29tcG9uZW50XG4gIF1cbn0pXG5leHBvcnQgY2xhc3MgTGlzdENvbXBvbmVudCBleHRlbmRzIE5neEJhc2VDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQsIE9uRGVzdHJveSB7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBUaGUgZGlzcGxheSBtb2RlIGZvciB0aGUgbGlzdCBjb21wb25lbnQuXG4gICAqIEBzdW1tYXJ5IERldGVybWluZXMgaG93IHRoZSBsaXN0IGRhdGEgaXMgbG9hZGVkIGFuZCBkaXNwbGF5ZWQuIE9wdGlvbnMgaW5jbHVkZTpcbiAgICogLSBJTkZJTklURTogTG9hZHMgbW9yZSBkYXRhIGFzIHRoZSB1c2VyIHNjcm9sbHMgKGluZmluaXRlIHNjcm9sbGluZylcbiAgICogLSBQQUdJTkFURUQ6IERpc3BsYXlzIGRhdGEgaW4gcGFnZXMgd2l0aCBuYXZpZ2F0aW9uIGNvbnRyb2xzXG4gICAqXG4gICAqIEB0eXBlIHtMaXN0Q29tcG9uZW50c1R5cGVzfVxuICAgKiBAZGVmYXVsdCBMaXN0Q29tcG9uZW50c1R5cGVzLklORklOSVRFXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICB0eXBlOiBMaXN0Q29tcG9uZW50c1R5cGVzID0gTGlzdENvbXBvbmVudHNUeXBlcy5JTkZJTklURTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENvbnRyb2xzIHdoZXRoZXIgdGhlIGNvbXBvbmVudCB1c2VzIHRyYW5zbGF0aW9uIHNlcnZpY2VzLlxuICAgKiBAc3VtbWFyeSBXaGVuIHNldCB0byB0cnVlLCB0aGUgY29tcG9uZW50IHdpbGwgYXR0ZW1wdCB0byB1c2UgdHJhbnNsYXRpb24gc2VydmljZXNcbiAgICogZm9yIGFueSB0ZXh0IGNvbnRlbnQuIFRoaXMgYWxsb3dzIGZvciBpbnRlcm5hdGlvbmFsaXphdGlvbiBvZiB0aGUgbGlzdCBjb21wb25lbnQuXG4gICAqXG4gICAqIEB0eXBlIHtTdHJpbmdPckJvb2xlYW59XG4gICAqIEBkZWZhdWx0IHRydWVcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIHRyYW5zbGF0YWJsZTogU3RyaW5nT3JCb29sZWFuID0gdHJ1ZTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENvbnRyb2xzIHRoZSB2aXNpYmlsaXR5IG9mIHRoZSBzZWFyY2ggYmFyLlxuICAgKiBAc3VtbWFyeSBXaGVuIHNldCB0byB0cnVlLCBkaXNwbGF5cyBhIHNlYXJjaCBiYXIgYXQgdGhlIHRvcCBvZiB0aGUgbGlzdCB0aGF0IGFsbG93c1xuICAgKiB1c2VycyB0byBmaWx0ZXIgdGhlIGxpc3QgaXRlbXMuIFRoZSBzZWFyY2ggZnVuY3Rpb25hbGl0eSB3b3JrcyBieSBmaWx0ZXJpbmcgdGhlXG4gICAqIGV4aXN0aW5nIGRhdGEgb3IgYnkgdHJpZ2dlcmluZyBhIG5ldyBkYXRhIGZldGNoIHdpdGggc2VhcmNoIHBhcmFtZXRlcnMuXG4gICAqXG4gICAqIEB0eXBlIHtTdHJpbmdPckJvb2xlYW59XG4gICAqIEBkZWZhdWx0IHRydWVcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIHNob3dTZWFyY2hiYXI6IFN0cmluZ09yQm9vbGVhbiA9IHRydWU7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBEaXJlY3QgZGF0YSBpbnB1dCBmb3IgdGhlIGxpc3QgY29tcG9uZW50LlxuICAgKiBAc3VtbWFyeSBQcm92aWRlcyBhIHdheSB0byBkaXJlY3RseSBwYXNzIGRhdGEgdG8gdGhlIGxpc3QgY29tcG9uZW50IGluc3RlYWQgb2ZcbiAgICogZmV0Y2hpbmcgaXQgZnJvbSBhIHNvdXJjZS4gV2hlbiBib3RoIGRhdGEgYW5kIHNvdXJjZSBhcmUgcHJvdmlkZWQsIHRoZSBjb21wb25lbnRcbiAgICogd2lsbCB1c2UgdGhlIHNvdXJjZSB0byBmZXRjaCBkYXRhIG9ubHkgaWYgdGhlIGRhdGEgYXJyYXkgaXMgZW1wdHkuXG4gICAqXG4gICAqIEB0eXBlIHtLZXlWYWx1ZVtdIHwgdW5kZWZpbmVkfVxuICAgKiBAZGVmYXVsdCB1bmRlZmluZWRcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIGRhdGE/OiBLZXlWYWx1ZVtdIHwgdW5kZWZpbmVkID0gdW5kZWZpbmVkO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVGhlIGRhdGEgc291cmNlIGZvciB0aGUgbGlzdCBjb21wb25lbnQuXG4gICAqIEBzdW1tYXJ5IFNwZWNpZmllcyB3aGVyZSB0aGUgbGlzdCBzaG91bGQgZmV0Y2ggaXRzIGRhdGEgZnJvbS4gVGhpcyBjYW4gYmUgZWl0aGVyOlxuICAgKiAtIEEgc3RyaW5nIFVSTCBvciBlbmRwb2ludCBpZGVudGlmaWVyXG4gICAqIC0gQSBmdW5jdGlvbiB0aGF0IHJldHVybnMgZGF0YSB3aGVuIGNhbGxlZFxuICAgKiBUaGUgY29tcG9uZW50IHdpbGwgY2FsbCB0aGlzIHNvdXJjZSB3aGVuIGl0IG5lZWRzIHRvIGxvYWQgb3IgcmVmcmVzaCBkYXRhLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nIHwgRnVuY3Rpb25MaWtlfVxuICAgKiBAcmVxdWlyZWRcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIHNvdXJjZSE6IHN0cmluZyB8IEZ1bmN0aW9uTGlrZTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBzdGFydGluZyBpbmRleCBmb3IgZGF0YSBmZXRjaGluZy5cbiAgICogQHN1bW1hcnkgU3BlY2lmaWVzIHRoZSBpbmRleCBmcm9tIHdoaWNoIHRvIHN0YXJ0IGZldGNoaW5nIGRhdGEuIFRoaXMgaXMgdXNlZFxuICAgKiBmb3IgcGFnaW5hdGlvbiBhbmQgaW5maW5pdGUgc2Nyb2xsaW5nIHRvIGRldGVybWluZSB3aGljaCBzdWJzZXQgb2YgZGF0YSB0byBsb2FkLlxuICAgKlxuICAgKiBAdHlwZSB7bnVtYmVyfVxuICAgKiBAZGVmYXVsdCAwXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBzdGFydDogbnVtYmVyID0gMDtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBudW1iZXIgb2YgaXRlbXMgdG8gZmV0Y2ggcGVyIHBhZ2Ugb3IgbG9hZCBvcGVyYXRpb24uXG4gICAqIEBzdW1tYXJ5IERldGVybWluZXMgaG93IG1hbnkgaXRlbXMgYXJlIGxvYWRlZCBhdCBvbmNlIGR1cmluZyBwYWdpbmF0aW9uIG9yXG4gICAqIGluZmluaXRlIHNjcm9sbGluZy4gVGhpcyBhZmZlY3RzIHRoZSBzaXplIG9mIGRhdGEgY2h1bmtzIHJlcXVlc3RlZCBmcm9tIHRoZSBzb3VyY2UuXG4gICAqXG4gICAqIEB0eXBlIHtudW1iZXJ9XG4gICAqIEBkZWZhdWx0IDEwXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBsaW1pdDogbnVtYmVyID0gMTA7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBDb250cm9scyB3aGV0aGVyIG1vcmUgZGF0YSBjYW4gYmUgbG9hZGVkLlxuICAgKiBAc3VtbWFyeSBXaGVuIHNldCB0byB0cnVlLCB0aGUgY29tcG9uZW50IHdpbGwgYWxsb3cgbG9hZGluZyBhZGRpdGlvbmFsIGRhdGFcbiAgICogdGhyb3VnaCBpbmZpbml0ZSBzY3JvbGxpbmcgb3IgcGFnaW5hdGlvbi4gV2hlbiBmYWxzZSwgdGhlIGNvbXBvbmVudCB3aWxsIG5vdFxuICAgKiBhdHRlbXB0IHRvIGxvYWQgbW9yZSBkYXRhIGJleW9uZCB3aGF0IGlzIGluaXRpYWxseSBkaXNwbGF5ZWQuXG4gICAqXG4gICAqIEB0eXBlIHtTdHJpbmdPckJvb2xlYW59XG4gICAqIEBkZWZhdWx0IHRydWVcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIGxvYWRNb3JlRGF0YTogU3RyaW5nT3JCb29sZWFuID0gdHJ1ZVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVGhlIHN0eWxlIG9mIGRpdmlkaW5nIGxpbmVzIGJldHdlZW4gbGlzdCBpdGVtcy5cbiAgICogQHN1bW1hcnkgRGV0ZXJtaW5lcyBob3cgZGl2aWRpbmcgbGluZXMgYXBwZWFyIGJldHdlZW4gbGlzdCBpdGVtcy4gT3B0aW9ucyBpbmNsdWRlOlxuICAgKiAtIFwiaW5zZXRcIjogTGluZXMgYXJlIGluc2V0IGZyb20gdGhlIGVkZ2VzXG4gICAqIC0gXCJmdWxsXCI6IExpbmVzIGV4dGVuZCB0aGUgZnVsbCB3aWR0aFxuICAgKiAtIFwibm9uZVwiOiBObyBkaXZpZGluZyBsaW5lc1xuICAgKlxuICAgKiBAdHlwZSB7XCJpbnNldFwiIHwgXCJmdWxsXCIgfCBcIm5vbmVcIn1cbiAgICogQGRlZmF1bHQgXCJmdWxsXCJcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIGxpbmVzOiBcImluc2V0XCIgfCBcImZ1bGxcIiB8IFwibm9uZVwiID0gXCJmdWxsXCI7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBDb250cm9scyB3aGV0aGVyIHRoZSBsaXN0IGhhcyBpbnNldCBzdHlsaW5nLlxuICAgKiBAc3VtbWFyeSBXaGVuIHNldCB0byB0cnVlLCB0aGUgbGlzdCB3aWxsIGhhdmUgaW5zZXQgc3R5bGluZyB3aXRoIHJvdW5kZWQgY29ybmVyc1xuICAgKiBhbmQgbWFyZ2luIGFyb3VuZCB0aGUgZWRnZXMuIFRoaXMgY3JlYXRlcyBhIGNhcmQtbGlrZSBhcHBlYXJhbmNlIGZvciB0aGUgbGlzdC5cbiAgICpcbiAgICogQHR5cGUge1N0cmluZ09yQm9vbGVhbn1cbiAgICogQGRlZmF1bHQgZmFsc2VcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIGluc2V0OiBTdHJpbmdPckJvb2xlYW4gPSBmYWxzZTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSB0aHJlc2hvbGQgZm9yIHRyaWdnZXJpbmcgaW5maW5pdGUgc2Nyb2xsIGxvYWRpbmcuXG4gICAqIEBzdW1tYXJ5IFNwZWNpZmllcyBob3cgY2xvc2UgdG8gdGhlIGJvdHRvbSBvZiB0aGUgbGlzdCB0aGUgdXNlciBtdXN0IHNjcm9sbFxuICAgKiBiZWZvcmUgdGhlIGNvbXBvbmVudCB0cmlnZ2VycyBsb2FkaW5nIG9mIGFkZGl0aW9uYWwgZGF0YS4gVGhpcyBpcyBleHByZXNzZWRcbiAgICogYXMgYSBwZXJjZW50YWdlIG9mIHRoZSBsaXN0IGhlaWdodC5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZ31cbiAgICogQGRlZmF1bHQgXCIxNSVcIlxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgc2Nyb2xsVGhyZXNob2xkOiBzdHJpbmcgPSBcIjE1JVwiO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVGhlIHBvc2l0aW9uIHdoZXJlIG5ldyBpdGVtcyBhcmUgYWRkZWQgZHVyaW5nIGluZmluaXRlIHNjcm9sbGluZy5cbiAgICogQHN1bW1hcnkgRGV0ZXJtaW5lcyB3aGV0aGVyIG5ldyBpdGVtcyBhcmUgYWRkZWQgdG8gdGhlIHRvcCBvciBib3R0b20gb2YgdGhlIGxpc3RcbiAgICogd2hlbiBsb2FkaW5nIG1vcmUgZGF0YSB0aHJvdWdoIGluZmluaXRlIHNjcm9sbGluZy5cbiAgICpcbiAgICogQHR5cGUge1wiYm90dG9tXCIgfCBcInRvcFwifVxuICAgKiBAZGVmYXVsdCBcImJvdHRvbVwiXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBzY3JvbGxQb3NpdGlvbjogXCJib3R0b21cIiB8IFwidG9wXCIgPSBcImJvdHRvbVwiO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ3VzdG9tIHRleHQgdG8gZGlzcGxheSBkdXJpbmcgbG9hZGluZyBvcGVyYXRpb25zLlxuICAgKiBAc3VtbWFyeSBTcGVjaWZpZXMgdGhlIHRleHQgc2hvd24gaW4gdGhlIGxvYWRpbmcgaW5kaWNhdG9yIHdoZW4gdGhlIGNvbXBvbmVudFxuICAgKiBpcyBmZXRjaGluZyBkYXRhLiBJZiBub3QgcHJvdmlkZWQsIGEgZGVmYXVsdCBsb2FkaW5nIG1lc3NhZ2Ugd2lsbCBiZSB1c2VkLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nIHwgdW5kZWZpbmVkfVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgbG9hZGluZ1RleHQ/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBDb250cm9scyB0aGUgdmlzaWJpbGl0eSBvZiB0aGUgcHVsbC10by1yZWZyZXNoIGZlYXR1cmUuXG4gICAqIEBzdW1tYXJ5IFdoZW4gc2V0IHRvIHRydWUsIGVuYWJsZXMgdGhlIHB1bGwtdG8tcmVmcmVzaCBmdW5jdGlvbmFsaXR5IHRoYXQgYWxsb3dzXG4gICAqIHVzZXJzIHRvIHJlZnJlc2ggdGhlIGxpc3QgZGF0YSBieSBwdWxsaW5nIGRvd24gZnJvbSB0aGUgdG9wIG9mIHRoZSBsaXN0LlxuICAgKlxuICAgKiBAdHlwZSB7U3RyaW5nT3JCb29sZWFufVxuICAgKiBAZGVmYXVsdCB0cnVlXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBzaG93UmVmcmVzaGVyOiBTdHJpbmdPckJvb2xlYW4gPSB0cnVlO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVGhlIHR5cGUgb2Ygc3Bpbm5lciB0byBkaXNwbGF5IGR1cmluZyBsb2FkaW5nIG9wZXJhdGlvbnMuXG4gICAqIEBzdW1tYXJ5IFNwZWNpZmllcyB0aGUgdmlzdWFsIHN0eWxlIG9mIHRoZSBsb2FkaW5nIHNwaW5uZXIgc2hvd24gZHVyaW5nIGRhdGFcbiAgICogZmV0Y2hpbmcgb3BlcmF0aW9ucy4gVXNlcyBJb25pYydzIHByZWRlZmluZWQgc3Bpbm5lciB0eXBlcy5cbiAgICpcbiAgICogQHR5cGUge1NwaW5uZXJUeXBlc31cbiAgICogQGRlZmF1bHQgXCJjaXJjdWxhclwiXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBsb2FkaW5nU3Bpbm5lcjogU3Bpbm5lclR5cGVzID0gXCJjaXJjdWxhclwiO1xuXG4gIC8vIC8qKlxuICAvLyAgKiBAZGVzY3JpcHRpb24gUXVlcnkgcGFyYW1ldGVycyBmb3IgZGF0YSBmZXRjaGluZy5cbiAgLy8gICogQHN1bW1hcnkgU3BlY2lmaWVzIGFkZGl0aW9uYWwgcXVlcnkgcGFyYW1ldGVycyB0byB1c2Ugd2hlbiBmZXRjaGluZyBkYXRhIGZyb21cbiAgLy8gICogdGhlIHNvdXJjZS4gVGhpcyBjYW4gYmUgcHJvdmlkZWQgYXMgYSBzdHJpbmcgKEpTT04pIG9yIGEgZGlyZWN0IG9iamVjdC5cbiAgLy8gICpcbiAgLy8gICogQHR5cGUge3N0cmluZyB8IEtleVZhbHVlIHwgdW5kZWZpbmVkfVxuICAvLyAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAvLyAgKi9cbiAgLy8gQElucHV0KClcbiAgLy8gcXVlcnk/OiBzdHJpbmcgfCBLZXlWYWx1ZTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENvbnRyb2xzIHdoZXRoZXIgdGhlIGZpbHRlcmluZyBmdW5jdGlvbmFsaXR5IGlzIGVuYWJsZWQuXG4gICAqIEBzdW1tYXJ5IFdoZW4gc2V0IHRvIHRydWUsIGVuYWJsZXMgdGhlIGZpbHRlciBjb21wb25lbnQgdGhhdCBhbGxvd3MgdXNlcnMgdG8gY3JlYXRlXG4gICAqIGNvbXBsZXggc2VhcmNoIGNyaXRlcmlhIHdpdGggbXVsdGlwbGUgZmllbGQgZmlsdGVycywgY29uZGl0aW9ucywgYW5kIHZhbHVlcy5cbiAgICogV2hlbiBmYWxzZSwgZGlzYWJsZXMgdGhlIGZpbHRlciBpbnRlcmZhY2UgZW50aXJlbHkuXG4gICAqXG4gICAqIEB0eXBlIHtTdHJpbmdPckJvb2xlYW59XG4gICAqIEBkZWZhdWx0IHRydWVcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIGVuYWJsZUZpbHRlcjogU3RyaW5nT3JCb29sZWFuID0gdHJ1ZTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFNvcnRpbmcgcGFyYW1ldGVycyBmb3IgZGF0YSBmZXRjaGluZy5cbiAgICogQHN1bW1hcnkgU3BlY2lmaWVzIGhvdyB0aGUgZmV0Y2hlZCBkYXRhIHNob3VsZCBiZSBzb3J0ZWQuIFRoaXMgY2FuIGJlIHByb3ZpZGVkXG4gICAqIGFzIGEgc3RyaW5nIChmaWVsZCBuYW1lIHdpdGggb3B0aW9uYWwgZGlyZWN0aW9uKSBvciBhIGRpcmVjdCBvYmplY3QuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmcgfCBLZXlWYWx1ZSB8IHVuZGVmaW5lZH1cbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIHNvcnREaXJlY3Rpb246IE9yZGVyRGlyZWN0aW9uID0gT3JkZXJEaXJlY3Rpb24uRFNDO1xuXG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBTb3J0aW5nIHBhcmFtZXRlcnMgZm9yIGRhdGEgZmV0Y2hpbmcuXG4gICAqIEBzdW1tYXJ5IFNwZWNpZmllcyBob3cgdGhlIGZldGNoZWQgZGF0YSBzaG91bGQgYmUgc29ydGVkLiBUaGlzIGNhbiBiZSBwcm92aWRlZFxuICAgKiBhcyBhIHN0cmluZyAoZmllbGQgbmFtZSB3aXRoIG9wdGlvbmFsIGRpcmVjdGlvbikgb3IgYSBkaXJlY3Qgb2JqZWN0LlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nIHwgS2V5VmFsdWUgfCB1bmRlZmluZWR9XG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBzb3J0QnkhOiBzdHJpbmc7XG5cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENvbnRyb2xzIHdoZXRoZXIgc29ydGluZyBmdW5jdGlvbmFsaXR5IGlzIGRpc2FibGVkLlxuICAgKiBAc3VtbWFyeSBXaGVuIHNldCB0byB0cnVlLCBkaXNhYmxlcyB0aGUgc29ydCBjb250cm9scyBhbmQgcHJldmVudHMgdXNlcnMgZnJvbVxuICAgKiBjaGFuZ2luZyB0aGUgc29ydCBvcmRlciBvciBmaWVsZC4gVGhlIGxpc3Qgd2lsbCBtYWludGFpbiBpdHMgZGVmYXVsdCBvclxuICAgKiBwcm9ncmFtbWF0aWNhbGx5IHNldCBzb3J0IGNvbmZpZ3VyYXRpb24gd2l0aG91dCB1c2VyIGludGVyYWN0aW9uLlxuICAgKlxuICAgKiBAdHlwZSB7U3RyaW5nT3JCb29sZWFufVxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgZGlzYWJsZVNvcnQ6IFN0cmluZ09yQm9vbGVhbiA9IGZhbHNlO1xuXG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBJY29uIHRvIGRpc3BsYXkgd2hlbiB0aGUgbGlzdCBpcyBlbXB0eS5cbiAgICogQHN1bW1hcnkgU3BlY2lmaWVzIHRoZSBpY29uIHNob3duIGluIHRoZSBlbXB0eSBzdGF0ZSB3aGVuIG5vIGRhdGEgaXMgYXZhaWxhYmxlLlxuICAgKiBUaGlzIGNhbiBiZSBhbnkgaWNvbiBuYW1lIHN1cHBvcnRlZCBieSB0aGUgYXBwbGljYXRpb24ncyBpY29uIHN5c3RlbS5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZyB8IHVuZGVmaW5lZH1cbiAgICogQGRlZmF1bHQgJ3RpLWRhdGFiYXNlLWV4Y2xhbWF0aW9uJ1xuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgZW1wdHlJY29uPzogc3RyaW5nID0gJ3RpLWRhdGFiYXNlLWV4Y2xhbWF0aW9uJztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENvbmZpZ3VyYXRpb24gZm9yIHRoZSBlbXB0eSBzdGF0ZSBkaXNwbGF5LlxuICAgKiBAc3VtbWFyeSBDdXN0b21pemVzIGhvdyB0aGUgZW1wdHkgc3RhdGUgaXMgZGlzcGxheWVkIHdoZW4gbm8gZGF0YSBpcyBhdmFpbGFibGUuXG4gICAqIFRoaXMgaW5jbHVkZXMgdGhlIHRpdGxlLCBzdWJ0aXRsZSwgYnV0dG9uIHRleHQsIGljb24sIGFuZCBuYXZpZ2F0aW9uIGxpbmsuXG4gICAqXG4gICAqIEB0eXBlIHtQYXJ0aWFsPElMaXN0RW1wdHlSZXN1bHQ+fVxuICAgKiBAZGVmYXVsdCB7XG4gICAqICAgdGl0bGU6ICdlbXB0eS50aXRsZScsXG4gICAqICAgc3VidGl0bGU6ICdlbXB0eS5zdWJ0aXRsZScsXG4gICAqICAgc2hvd0J1dHRvbjogZmFsc2UsXG4gICAqICAgaWNvbjogJ2FsZXJ0LWNpcmNsZS1vdXRsaW5lJyxcbiAgICogICBidXR0b25UZXh0OiAnbG9jYWxlLmVtcHR5LmJ1dHRvbicsXG4gICAqICAgbGluazogJydcbiAgICogfVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgZW1wdHk6IFBhcnRpYWw8SUxpc3RFbXB0eVJlc3VsdD4gPSB7XG4gICAgdGl0bGU6ICdlbXB0eS50aXRsZScsXG4gICAgc3VidGl0bGU6ICdlbXB0eS5zdWJ0aXRsZScsXG4gICAgc2hvd0J1dHRvbjogZmFsc2UsXG4gICAgaWNvbjogJ2FsZXJ0LWNpcmNsZS1vdXRsaW5lJyxcbiAgICBidXR0b25UZXh0OiAnbG9jYWxlLmVtcHR5LmJ1dHRvbicsXG4gICAgbGluazogJydcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVGhlIGN1cnJlbnQgcGFnZSBudW1iZXIgaW4gcGFnaW5hdGVkIG1vZGUuXG4gICAqIEBzdW1tYXJ5IFRyYWNrcyB3aGljaCBwYWdlIGlzIGN1cnJlbnRseSBiZWluZyBkaXNwbGF5ZWQgd2hlbiB0aGUgY29tcG9uZW50XG4gICAqIGlzIGluIHBhZ2luYXRlZCBtb2RlLiBUaGlzIGlzIHVzZWQgZm9yIHBhZ2luYXRpb24gY29udHJvbHMgYW5kIGRhdGEgZmV0Y2hpbmcuXG4gICAqXG4gICAqIEB0eXBlIHtudW1iZXJ9XG4gICAqIEBkZWZhdWx0IDFcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIHBhZ2U6IG51bWJlciA9IDE7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBUaGUgdG90YWwgbnVtYmVyIG9mIHBhZ2VzIGF2YWlsYWJsZS5cbiAgICogQHN1bW1hcnkgU3RvcmVzIHRoZSBjYWxjdWxhdGVkIHRvdGFsIG51bWJlciBvZiBwYWdlcyBiYXNlZCBvbiB0aGUgZGF0YSBzaXplXG4gICAqIGFuZCBsaW1pdC4gVGhpcyBpcyB1c2VkIGZvciBwYWdpbmF0aW9uIGNvbnRyb2xzIGFuZCBib3VuZGFyeSBjaGVja2luZy5cbiAgICpcbiAgICogQHR5cGUge251bWJlcn1cbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIHBhZ2VzITogbnVtYmVyO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSW5kaWNhdGVzIHdoZXRoZXIgYSByZWZyZXNoIG9wZXJhdGlvbiBpcyBpbiBwcm9ncmVzcy5cbiAgICogQHN1bW1hcnkgV2hlbiB0cnVlLCB0aGUgY29tcG9uZW50IGlzIGN1cnJlbnRseSBmZXRjaGluZyBuZXcgZGF0YS4gVGhpcyBpcyB1c2VkXG4gICAqIHRvIGNvbnRyb2wgbG9hZGluZyBpbmRpY2F0b3JzIGFuZCBwcmV2ZW50IGR1cGxpY2F0ZSByZWZyZXNoIG9wZXJhdGlvbnMgZnJvbVxuICAgKiBiZWluZyB0cmlnZ2VyZWQgc2ltdWx0YW5lb3VzbHkuXG4gICAqXG4gICAqIEB0eXBlIHtib29sZWFufVxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgcmVmcmVzaGluZzogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQXJyYXkgdXNlZCBmb3IgcmVuZGVyaW5nIHNrZWxldG9uIGxvYWRpbmcgcGxhY2Vob2xkZXJzLlxuICAgKiBAc3VtbWFyeSBDb250YWlucyBwbGFjZWhvbGRlciBpdGVtcyB0aGF0IGFyZSBkaXNwbGF5ZWQgZHVyaW5nIGRhdGEgbG9hZGluZy5cbiAgICogVGhlIGxlbmd0aCBvZiB0aGlzIGFycmF5IGRldGVybWluZXMgaG93IG1hbnkgc2tlbGV0b24gaXRlbXMgYXJlIHNob3duLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nW119XG4gICAqIEBkZWZhdWx0IG5ldyBBcnJheSgyKVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgc2tlbGV0b25EYXRhOiBzdHJpbmdbXSA9IG5ldyBBcnJheSgyKTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBwcm9jZXNzZWQgbGlzdCBpdGVtcyByZWFkeSBmb3IgZGlzcGxheS5cbiAgICogQHN1bW1hcnkgU3RvcmVzIHRoZSBjdXJyZW50IHNldCBvZiBpdGVtcyBiZWluZyBkaXNwbGF5ZWQgaW4gdGhlIGxpc3QgYWZ0ZXJcbiAgICogcHJvY2Vzc2luZyBmcm9tIHRoZSByYXcgZGF0YSBzb3VyY2UuIFRoaXMgbWF5IGJlIGEgc3Vic2V0IG9mIHRoZSBmdWxsIGRhdGFcbiAgICogd2hlbiB1c2luZyBwYWdpbmF0aW9uIG9yIGluZmluaXRlIHNjcm9sbGluZy5cbiAgICpcbiAgICogQHR5cGUge0tleVZhbHVlW119XG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBpdGVtcyE6IEtleVZhbHVlW107XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBUaGUgY3VycmVudCBzZWFyY2ggcXVlcnkgdmFsdWUuXG4gICAqIEBzdW1tYXJ5IFN0b3JlcyB0aGUgdGV4dCBlbnRlcmVkIGluIHRoZSBzZWFyY2ggYmFyLiBUaGlzIGlzIHVzZWQgdG8gZmlsdGVyXG4gICAqIHRoZSBsaXN0IGRhdGEgb3IgdG8gc2VuZCBhcyBhIHNlYXJjaCBwYXJhbWV0ZXIgd2hlbiBmZXRjaGluZyBuZXcgZGF0YS5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZyB8IHVuZGVmaW5lZH1cbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIHNlYXJjaFZhbHVlPzogc3RyaW5nIHwgSUZpbHRlclF1ZXJ5IHwgdW5kZWZpbmVkO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQSBwYWdpbmF0b3Igb2JqZWN0IGZvciBoYW5kbGluZyBwYWdpbmF0aW9uIG9wZXJhdGlvbnMuXG4gICAqIEBzdW1tYXJ5IFByb3ZpZGVzIGEgcGFnaW5hdG9yIG9iamVjdCB0aGF0IGNhbiBiZSB1c2VkIHRvIHJldHJpZXZlIGFuZCBuYXZpZ2F0ZVxuICAgKiB0aHJvdWdoIGRhdGEgaW4gY2h1bmtzLCByZWR1Y2luZyBtZW1vcnkgdXNhZ2UgYW5kIGltcHJvdmluZyBwZXJmb3JtYW5jZS5cbiAgICpcbiAgICogVGhlIHBhZ2luYXRvciBvYmplY3QgaXMgaW5pdGlhbGl6ZWQgaW4gdGhlIGBuZ09uSW5pdGAgbGlmZWN5Y2xlIGhvb2sgYW5kIGlzXG4gICAqIHVzZWQgdG8gZmV0Y2ggYW5kIGRpc3BsYXkgZGF0YSBpbiB0aGUgcGFnaW5hdGlvbiBjb21wb25lbnQuIEl0IGlzIGFuIGluc3RhbmNlXG4gICAqIG9mIHRoZSBgUGFnaW5hdG9yYCBjbGFzcyBmcm9tIHRoZSBgQGRlY2FmLXRzL2NvcmVgIHBhY2thZ2UsIHdoaWNoIHByb3ZpZGVzXG4gICAqIG1ldGhvZHMgZm9yIHF1ZXJ5aW5nIGFuZCBtYW5pcHVsYXRpbmcgcGFnaW5hdGVkIGRhdGEuXG4gICAqXG4gICAqIEB0eXBlIHtQYWdpbmF0b3I8TW9kZWw+fVxuICAgKiBAbWVtYmVyT2YgUGFnaW5hdGlvbkNvbXBvbmVudFxuICAgKi9cbiAgcGFnaW5hdG9yITogUGFnaW5hdG9yPE1vZGVsPiB8IHVuZGVmaW5lZDtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBsYXN0IHBhZ2UgbnVtYmVyIHRoYXQgd2FzIGRpc3BsYXllZC5cbiAgICogQHN1bW1hcnkgS2VlcHMgdHJhY2sgb2YgdGhlIHByZXZpb3VzbHkgZGlzcGxheWVkIHBhZ2UgbnVtYmVyLCB3aGljaCBpcyB1c2VmdWxcbiAgICogZm9yIGhhbmRsaW5nIG5hdmlnYXRpb24gYW5kIHNlYXJjaCBvcGVyYXRpb25zIGluIHBhZ2luYXRlZCBtb2RlLlxuICAgKlxuICAgKiBAdHlwZSB7bnVtYmVyfVxuICAgKiBAZGVmYXVsdCAxXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBsYXN0UGFnZTogbnVtYmVyID0gMVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gRXZlbnQgZW1pdHRlciBmb3IgcmVmcmVzaCBvcGVyYXRpb25zLlxuICAgKiBAc3VtbWFyeSBFbWl0cyBhbiBldmVudCB3aGVuIHRoZSBsaXN0IGRhdGEgaXMgcmVmcmVzaGVkLCBlaXRoZXIgdGhyb3VnaCBwdWxsLXRvLXJlZnJlc2hcbiAgICogb3IgcHJvZ3JhbW1hdGljIHJlZnJlc2guIFRoZSBldmVudCBpbmNsdWRlcyB0aGUgcmVmcmVzaGVkIGRhdGEgYW5kIGNvbXBvbmVudCBpbmZvcm1hdGlvbi5cbiAgICpcbiAgICogQHR5cGUge0V2ZW50RW1pdHRlcjxCYXNlQ3VzdG9tRXZlbnQ+fVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgQE91dHB1dCgpXG4gIHJlZnJlc2hFdmVudDogRXZlbnRFbWl0dGVyPEJhc2VDdXN0b21FdmVudD4gPSBuZXcgRXZlbnRFbWl0dGVyPEJhc2VDdXN0b21FdmVudD4oKTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEV2ZW50IGVtaXR0ZXIgZm9yIGl0ZW0gY2xpY2sgaW50ZXJhY3Rpb25zLlxuICAgKiBAc3VtbWFyeSBFbWl0cyBhbiBldmVudCB3aGVuIGEgbGlzdCBpdGVtIGlzIGNsaWNrZWQuIFRoZSBldmVudCBpbmNsdWRlcyB0aGUgZGF0YVxuICAgKiBvZiB0aGUgY2xpY2tlZCBpdGVtLCBhbGxvd2luZyBwYXJlbnQgY29tcG9uZW50cyB0byByZXNwb25kIHRvIHRoZSBpbnRlcmFjdGlvbi5cbiAgICpcbiAgICogQHR5cGUge0V2ZW50RW1pdHRlcjxLZXlWYWx1ZT59XG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBAT3V0cHV0KClcbiAgY2xpY2tFdmVudDogIEV2ZW50RW1pdHRlcjxMaXN0SXRlbUN1c3RvbUV2ZW50fFJlbmRlcmVyQ3VzdG9tRXZlbnQ+ID0gbmV3IEV2ZW50RW1pdHRlcjxMaXN0SXRlbUN1c3RvbUV2ZW50fFJlbmRlcmVyQ3VzdG9tRXZlbnQ+KCk7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBTdWJqZWN0IGZvciBkZWJvdW5jaW5nIGNsaWNrIGV2ZW50cy5cbiAgICogQHN1bW1hcnkgVXNlcyBSeEpTIFN1YmplY3QgdG8gY29sbGVjdCBjbGljayBldmVudHMgYW5kIGVtaXQgdGhlbSBhZnRlciBhIGRlYm91bmNlXG4gICAqIHBlcmlvZC4gVGhpcyBwcmV2ZW50cyBtdWx0aXBsZSByYXBpZCBjbGlja3MgZnJvbSB0cmlnZ2VyaW5nIG11bHRpcGxlIGV2ZW50cy5cbiAgICpcbiAgICogQHByaXZhdGVcbiAgICogQHR5cGUge1N1YmplY3Q8Q3VzdG9tRXZlbnQgfCBMaXN0SXRlbUN1c3RvbUV2ZW50IHwgUmVuZGVyZXJDdXN0b21FdmVudD59XG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBwcml2YXRlIGNsaWNrSXRlbVN1YmplY3Q6IFN1YmplY3Q8Q3VzdG9tRXZlbnQgfCBMaXN0SXRlbUN1c3RvbUV2ZW50IHwgUmVuZGVyZXJDdXN0b21FdmVudD4gPSBuZXcgU3ViamVjdDxDdXN0b21FdmVudCB8IExpc3RJdGVtQ3VzdG9tRXZlbnQgfCBSZW5kZXJlckN1c3RvbUV2ZW50PigpO1xuXG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBTdWJqZWN0IGZvciBkZWJvdW5jaW5nIHJlcG9zaXRvcnkgb2JzZXJ2YXRpb24gZXZlbnRzLlxuICAgKiBAc3VtbWFyeSBSeEpTIFN1YmplY3QgdGhhdCBjb2xsZWN0cyByZXBvc2l0b3J5IGNoYW5nZSBldmVudHMgYW5kIGVtaXRzIHRoZW0gYWZ0ZXJcbiAgICogYSBkZWJvdW5jZSBwZXJpb2QuIFRoaXMgcHJldmVudHMgbXVsdGlwbGUgcmFwaWQgcmVwb3NpdG9yeSBjaGFuZ2VzIGZyb20gdHJpZ2dlcmluZ1xuICAgKiBtdWx0aXBsZSBsaXN0IHJlZnJlc2ggb3BlcmF0aW9ucywgaW1wcm92aW5nIHBlcmZvcm1hbmNlIGFuZCB1c2VyIGV4cGVyaWVuY2UuXG4gICAqXG4gICAqIEBwcml2YXRlXG4gICAqIEB0eXBlIHtTdWJqZWN0PGFueT59XG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLWV4cGxpY2l0LWFueVxuICBwcml2YXRlIG9ic2VydmVyU3ViamV0OiBTdWJqZWN0PGFueT4gPSBuZXcgU3ViamVjdDxhbnk+KCk7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBPYnNlcnZlciBvYmplY3QgZm9yIHJlcG9zaXRvcnkgY2hhbmdlIG5vdGlmaWNhdGlvbnMuXG4gICAqIEBzdW1tYXJ5IEltcGxlbWVudHMgdGhlIE9ic2VydmVyIGludGVyZmFjZSB0byByZWNlaXZlIG5vdGlmaWNhdGlvbnMgd2hlbiB0aGVcbiAgICogdW5kZXJseWluZyBkYXRhIHJlcG9zaXRvcnkgY2hhbmdlcy4gVGhpcyBlbmFibGVzIGF1dG9tYXRpYyBsaXN0IHVwZGF0ZXMgd2hlblxuICAgKiBkYXRhIGlzIGNyZWF0ZWQsIHVwZGF0ZWQsIG9yIGRlbGV0ZWQgdGhyb3VnaCB0aGUgcmVwb3NpdG9yeS5cbiAgICpcbiAgICogQHByaXZhdGVcbiAgICogQHR5cGUge09ic2VydmVyfVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgcHJpdmF0ZSBvYnNlcnZlcjogT2JzZXJ2ZXIgPSB7IHJlZnJlc2g6IGFzeW5jICguLi4gYXJnczogdW5rbm93bltdKTogUHJvbWlzZTx2b2lkPiA9PiB0aGlzLm9ic2VydmVSZXBvc2l0b3J5KC4uLmFyZ3MpfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gTGlzdCBvZiBhdmFpbGFibGUgaW5kZXhlcyBmb3IgZGF0YSBxdWVyeWluZyBhbmQgZmlsdGVyaW5nLlxuICAgKiBAc3VtbWFyeSBQcm92aWRlcyBhIGxpc3Qgb2YgaW5kZXggbmFtZXMgdGhhdCBjYW4gYmUgdXNlZCB0byBvcHRpbWl6ZSBkYXRhIHF1ZXJ5aW5nIGFuZCBmaWx0ZXJpbmdcbiAgICogb3BlcmF0aW9ucywgZXNwZWNpYWxseSBpbiBzY2VuYXJpb3Mgd2l0aCBsYXJnZSBkYXRhc2V0cy5cbiAgICpcbiAgICogSW5kZXhlcyBjYW4gc2lnbmlmaWNhbnRseSBpbXByb3ZlIHRoZSBwZXJmb3JtYW5jZSBvZiBkYXRhIHJldHJpZXZhbCBieSBhbGxvd2luZyB0aGUgZGF0YWJhc2VcbiAgICogdG8gcXVpY2tseSBsb2NhdGUgYW5kIHJldHJpZXZlIHJlbGV2YW50IGRhdGEgYmFzZWQgb24gaW5kZXhlZCBmaWVsZHMuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmdbXX1cbiAgICogQGRlZmF1bHQgW11cbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIGluZGV4ZXMhOiBzdHJpbmdbXTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEluaXRpYWxpemVzIGEgbmV3IGluc3RhbmNlIG9mIHRoZSBMaXN0Q29tcG9uZW50LlxuICAgKiBAc3VtbWFyeSBDcmVhdGVzIGEgbmV3IExpc3RDb21wb25lbnQgYW5kIHNldHMgdXAgdGhlIGJhc2UgY29tcG9uZW50IHdpdGggdGhlIGFwcHJvcHJpYXRlXG4gICAqIGNvbXBvbmVudCBuYW1lLiBUaGlzIGNvbnN0cnVjdG9yIGlzIGNhbGxlZCB3aGVuIEFuZ3VsYXIgaW5zdGFudGlhdGVzIHRoZSBjb21wb25lbnQgYW5kXG4gICAqIGJlZm9yZSBhbnkgaW5wdXQgcHJvcGVydGllcyBhcmUgc2V0LiBJdCBwYXNzZXMgdGhlIGNvbXBvbmVudCBuYW1lIHRvIHRoZSBwYXJlbnQgY2xhc3NcbiAgICogY29uc3RydWN0b3IgdG8gZW5hYmxlIHByb3BlciBsb2NhbGl6YXRpb24gYW5kIGNvbXBvbmVudCBpZGVudGlmaWNhdGlvbi5cbiAgICpcbiAgICogVGhlIGNvbnN0cnVjdG9yIGlzIGludGVudGlvbmFsbHkgbWluaW1hbCwgd2l0aCBtb3N0IGluaXRpYWxpemF0aW9uIGxvZ2ljIGRlZmVycmVkIHRvXG4gICAqIHRoZSBuZ09uSW5pdCBsaWZlY3ljbGUgaG9vay4gVGhpcyBmb2xsb3dzIEFuZ3VsYXIgYmVzdCBwcmFjdGljZXMgYnkga2VlcGluZyB0aGUgY29uc3RydWN0b3JcbiAgICogZm9jdXNlZCBvbiBkZXBlbmRlbmN5IGluamVjdGlvbiBhbmQgYmFzaWMgc2V0dXAsIHdoaWxlIGNvbXBsZXggaW5pdGlhbGl6YXRpb24gdGhhdCBkZXBlbmRzXG4gICAqIG9uIGlucHV0IHByb3BlcnRpZXMgaXMgaGFuZGxlZCBpbiBuZ09uSW5pdC5cbiAgICpcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKFwiTGlzdENvbXBvbmVudFwiKTtcbiAgfVxuXG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBJbml0aWFsaXplcyB0aGUgY29tcG9uZW50IGFmdGVyIEFuZ3VsYXIgc2V0cyB0aGUgaW5wdXQgcHJvcGVydGllcy5cbiAgICogQHN1bW1hcnkgU2V0cyB1cCB0aGUgY29tcG9uZW50IGJ5IGluaXRpYWxpemluZyBldmVudCBzdWJzY3JpcHRpb25zLCBwcm9jZXNzaW5nIGJvb2xlYW5cbiAgICogaW5wdXRzLCBhbmQgbG9hZGluZyB0aGUgaW5pdGlhbCBkYXRhLiBUaGlzIG1ldGhvZCBwcmVwYXJlcyB0aGUgY29tcG9uZW50IGZvciB1c2VyXG4gICAqIGludGVyYWN0aW9uIGJ5IGVuc3VyaW5nIGFsbCBwcm9wZXJ0aWVzIGFyZSBwcm9wZXJseSBpbml0aWFsaXplZCBhbmQgZGF0YSBpcyBsb2FkZWQuXG4gICAqXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fVxuICAgKlxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBBIGFzIEFuZ3VsYXIgTGlmZWN5Y2xlXG4gICAqICAgcGFydGljaXBhbnQgTCBhcyBMaXN0Q29tcG9uZW50XG4gICAqICAgcGFydGljaXBhbnQgRCBhcyBEYXRhIFNvdXJjZVxuICAgKlxuICAgKiAgIEEtPj5MOiBuZ09uSW5pdCgpXG4gICAqICAgTC0+Pkw6IFNldCB1cCBjbGljayBldmVudCBkZWJvdW5jaW5nXG4gICAqICAgTC0+Pkw6IFByb2Nlc3MgYm9vbGVhbiBpbnB1dHNcbiAgICogICBMLT4+TDogQ29uZmlndXJlIGNvbXBvbmVudCBiYXNlZCBvbiBpbnB1dHNcbiAgICogICBMLT4+TDogcmVmcmVzaCgpXG4gICAqICAgTC0+PkQ6IFJlcXVlc3QgaW5pdGlhbCBkYXRhXG4gICAqICAgRC0tPj5MOiBSZXR1cm4gZGF0YVxuICAgKiAgIEwtPj5MOiBQcm9jZXNzIGFuZCBkaXNwbGF5IGRhdGFcbiAgICogICBMLT4+TDogQ29uZmlndXJlIGVtcHR5IHN0YXRlIGlmIG5lZWRlZFxuICAgKiAgIEwtPj5MOiBpbml0aWFsaXplKClcbiAgICpcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIGFzeW5jIG5nT25Jbml0KCk6IFByb21pc2U8dm9pZD4ge1xuICAgIHRoaXMuY2xpY2tJdGVtU3ViamVjdC5waXBlKGRlYm91bmNlVGltZSgxMDApKS5zdWJzY3JpYmUoZXZlbnQgPT4gdGhpcy5jbGlja0V2ZW50RW1pdChldmVudCBhcyBMaXN0SXRlbUN1c3RvbUV2ZW50IHwgUmVuZGVyZXJDdXN0b21FdmVudCkpO1xuICAgIHRoaXMub2JzZXJ2ZXJTdWJqZXQucGlwZShkZWJvdW5jZVRpbWUoMTAwKSkuc3Vic2NyaWJlKGFyZ3MgPT4gdGhpcy5oYW5kbGVPYnNlcnZlRXZlbnQoYXJnc1swXSwgYXJnc1sxXSwgYXJnc1syXSkpO1xuICAgIHRoaXMuZW5hYmxlRmlsdGVyID0gc3RyaW5nVG9Cb29sZWFuKHRoaXMuZW5hYmxlRmlsdGVyKTtcbiAgICB0aGlzLmxpbWl0ID0gTnVtYmVyKHRoaXMubGltaXQpO1xuICAgIHRoaXMuc3RhcnQgPSBOdW1iZXIodGhpcy5zdGFydCk7XG4gICAgdGhpcy5pbnNldCA9IHN0cmluZ1RvQm9vbGVhbih0aGlzLmluc2V0KTtcbiAgICB0aGlzLnNob3dSZWZyZXNoZXIgPSBzdHJpbmdUb0Jvb2xlYW4odGhpcy5zaG93UmVmcmVzaGVyKTtcbiAgICB0aGlzLmxvYWRNb3JlRGF0YSA9IHN0cmluZ1RvQm9vbGVhbih0aGlzLmxvYWRNb3JlRGF0YSk7XG4gICAgdGhpcy5zaG93U2VhcmNoYmFyID0gc3RyaW5nVG9Cb29sZWFuKHRoaXMuc2hvd1NlYXJjaGJhcik7XG4gICAgdGhpcy5kaXNhYmxlU29ydCA9IHN0cmluZ1RvQm9vbGVhbih0aGlzLmRpc2FibGVTb3J0KTtcbiAgICBpZih0eXBlb2YgdGhpcy5pdGVtPy5bJ3RhZyddID09PSAnYm9vbGVhbicgJiYgdGhpcy5pdGVtPy5bJ3RhZyddID09PSB0cnVlKVxuICAgICAgdGhpcy5pdGVtWyd0YWcnXSA9IENvbXBvbmVudHNUYWdOYW1lcy5MSVNUX0lURU0gYXMgc3RyaW5nO1xuXG4gICAgYXdhaXQgdGhpcy5yZWZyZXNoKCk7XG5cbiAgICBpZih0aGlzLm9wZXJhdGlvbnMuaW5jbHVkZXMoT3BlcmF0aW9uS2V5cy5DUkVBVEUpICYmIHRoaXMucm91dGUpXG4gICAgICB0aGlzLmVtcHR5LmxpbmsgPSBgJHt0aGlzLnJvdXRlfS8ke09wZXJhdGlvbktleXMuQ1JFQVRFfWA7XG5cbiAgICB0aGlzLmluaXRpYWxpemUoKTtcblxuICAgIGlmKHRoaXMubW9kZWwgaW5zdGFuY2VvZiBNb2RlbCAmJiB0aGlzLl9yZXBvc2l0b3J5KVxuICAgICAgdGhpcy5fcmVwb3NpdG9yeS5vYnNlcnZlKHRoaXMub2JzZXJ2ZXIpO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBDbGVhbnMgdXAgcmVzb3VyY2VzIHdoZW4gdGhlIGNvbXBvbmVudCBpcyBkZXN0cm95ZWQuXG4gICAqIEBzdW1tYXJ5IFBlcmZvcm1zIGNsZWFudXAgb3BlcmF0aW9ucyB3aGVuIHRoZSBjb21wb25lbnQgaXMgYmVpbmcgcmVtb3ZlZCBmcm9tIHRoZSBET00uXG4gICAqIFRoaXMgaW5jbHVkZXMgY2xlYXJpbmcgcmVmZXJlbmNlcyB0byBtb2RlbHMgYW5kIGRhdGEgdG8gcHJldmVudCBtZW1vcnkgbGVha3MuXG4gICAqXG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgaWYodGhpcy5fcmVwb3NpdG9yeSlcbiAgICAgIHRoaXMuX3JlcG9zaXRvcnkudW5PYnNlcnZlKHRoaXMub2JzZXJ2ZXIpO1xuICAgIHRoaXMuZGF0YSA9ICB0aGlzLm1vZGVsID0gdGhpcy5fcmVwb3NpdG9yeSA9IHRoaXMucGFnaW5hdG9yID0gdW5kZWZpbmVkO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBIYW5kbGVzIHJlcG9zaXRvcnkgb2JzZXJ2YXRpb24gZXZlbnRzIHdpdGggZGVib3VuY2luZy5cbiAgICogQHN1bW1hcnkgUHJvY2Vzc2VzIHJlcG9zaXRvcnkgY2hhbmdlIG5vdGlmaWNhdGlvbnMgYW5kIHJvdXRlcyB0aGVtIGFwcHJvcHJpYXRlbHkuXG4gICAqIEZvciBDUkVBVEUgZXZlbnRzIHdpdGggYSBVSUQsIGhhbmRsZXMgdGhlbSBpbW1lZGlhdGVseS4gRm9yIG90aGVyIGV2ZW50cyxcbiAgICogcGFzc2VzIHRoZW0gdG8gdGhlIGRlYm91bmNlZCBvYnNlcnZlciBzdWJqZWN0IHRvIHByZXZlbnQgZXhjZXNzaXZlIHVwZGF0ZXMuXG4gICAqXG4gICAqIEBwYXJhbSB7Li4udW5rbm93bltdfSBhcmdzIC0gVGhlIHJlcG9zaXRvcnkgZXZlbnQgYXJndW1lbnRzIGluY2x1ZGluZyB0YWJsZSwgZXZlbnQgdHlwZSwgYW5kIFVJRFxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn1cbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIGFzeW5jIG9ic2VydmVSZXBvc2l0b3J5KC4uLmFyZ3M6IHVua25vd25bXSk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IFt0YWJsZSwgZXZlbnQsIHVpZF0gPSBhcmdzO1xuICAgIGlmKGV2ZW50ID09PSBPcGVyYXRpb25LZXlzLkNSRUFURSAmJiAhIXVpZClcbiAgICAgIHJldHVybiB0aGlzLmhhbmRsZU9ic2VydmVFdmVudCh0YWJsZSBhcyBzdHJpbmcsIGV2ZW50LCB1aWQgYXMgc3RyaW5nIHwgbnVtYmVyKTtcbiAgICByZXR1cm4gdGhpcy5vYnNlcnZlclN1YmpldC5uZXh0KGFyZ3MpO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBIYW5kbGVzIHNwZWNpZmljIHJlcG9zaXRvcnkgZXZlbnRzIGFuZCB1cGRhdGVzIHRoZSBsaXN0IGFjY29yZGluZ2x5LlxuICAgKiBAc3VtbWFyeSBQcm9jZXNzZXMgcmVwb3NpdG9yeSBjaGFuZ2UgZXZlbnRzIChDUkVBVEUsIFVQREFURSwgREVMRVRFKSBhbmQgcGVyZm9ybXNcbiAgICogdGhlIGFwcHJvcHJpYXRlIGxpc3Qgb3BlcmF0aW9ucy4gVGhpcyBpbmNsdWRlcyBhZGRpbmcgbmV3IGl0ZW1zLCB1cGRhdGluZyBleGlzdGluZ1xuICAgKiBvbmVzLCBvciByZW1vdmluZyBkZWxldGVkIGl0ZW1zIGZyb20gdGhlIGxpc3QgZGlzcGxheS5cbiAgICpcbiAgICogQHBhcmFtIHtzdHJpbmd9IHRhYmxlIC0gVGhlIHRhYmxlL21vZGVsIG5hbWUgdGhhdCBjaGFuZ2VkXG4gICAqIEBwYXJhbSB7T3BlcmF0aW9uS2V5c30gZXZlbnQgLSBUaGUgdHlwZSBvZiBvcGVyYXRpb24gKENSRUFURSwgVVBEQVRFLCBERUxFVEUpXG4gICAqIEBwYXJhbSB7c3RyaW5nIHwgbnVtYmVyfSB1aWQgLSBUaGUgdW5pcXVlIGlkZW50aWZpZXIgb2YgdGhlIGFmZmVjdGVkIGl0ZW1cbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59XG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBhc3luYyBoYW5kbGVPYnNlcnZlRXZlbnQodGFibGU6IHN0cmluZywgZXZlbnQ6IE9wZXJhdGlvbktleXMsIHVpZDogc3RyaW5nIHwgbnVtYmVyKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgaWYoZXZlbnQgPT09IE9wZXJhdGlvbktleXMuQ1JFQVRFKSB7XG4gICAgICBpZih1aWQpIHtcbiAgICAgICAgYXdhaXQgdGhpcy5oYW5kbGVDcmVhdGUodWlkKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGF3YWl0IHRoaXMucmVmcmVzaCh0cnVlKTtcbiAgICAgIH1cbiAgICB9IGVsc2Uge1xuICAgICAgaWYoZXZlbnQgPT09IE9wZXJhdGlvbktleXMuVVBEQVRFKVxuICAgICAgICBhd2FpdCB0aGlzLmhhbmRsZVVwZGF0ZSh1aWQpO1xuICAgICAgaWYoZXZlbnQgPT09IE9wZXJhdGlvbktleXMuREVMRVRFKVxuICAgICAgICB0aGlzLmhhbmRsZURlbGV0ZSh1aWQpO1xuICAgICAgdGhpcy5yZWZyZXNoRXZlbnRFbWl0KCk7XG4gICAgfVxuICB9XG5cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEZ1bmN0aW9uIGZvciB0cmFja2luZyBpdGVtcyBpbiB0aGUgbGlzdC5cbiAgICogQHN1bW1hcnkgUHJvdmlkZXMgYSB0cmFja2luZyBmdW5jdGlvbiBmb3IgdGhlIGAqbmdGb3JgIGRpcmVjdGl2ZSBpbiB0aGUgY29tcG9uZW50IHRlbXBsYXRlLlxuICAgKiBUaGlzIGZ1bmN0aW9uIGlzIHVzZWQgdG8gaWRlbnRpZnkgYW5kIGNvbnRyb2wgdGhlIHJlbmRlcmluZyBvZiBpdGVtcyBpbiB0aGUgbGlzdCxcbiAgICogcHJldmVudGluZyBkdXBsaWNhdGUgb3IgdW5uZWNlc3NhcnkgcmVuZGVyaW5nLlxuICAgKlxuICAgKiBUaGUgYHRyYWNrSXRlbUZuYCBmdW5jdGlvbiB0YWtlcyB0d28gcGFyYW1ldGVyczogYGluZGV4YCAodGhlIGluZGV4IG9mIHRoZSBpdGVtIGluIHRoZSBsaXN0KVxuICAgKiBhbmQgYGl0ZW1gICh0aGUgYWN0dWFsIGl0ZW0gZnJvbSB0aGUgbGlzdCkuIEl0IHJldHVybnMgdGhlIHRyYWNraW5nIGtleSwgd2hpY2ggaW4gdGhpcyBjYXNlXG4gICAqIGlzIHRoZSB1bmlvbiBvZiB0aGUgYHVpZGAgb2YgdGhlIGl0ZW0gd2l0aCB0aGUgbW9kZWwgbmFtZS5cbiAgICpcbiAgICogQHBhcmFtIHtudW1iZXJ9IGluZGV4IC0gVGhlIGluZGV4IG9mIHRoZSBpdGVtIGluIHRoZSBsaXN0LlxuXG4gICAqIEBwYXJhbSB7S2V5VmFsdWUgfCBzdHJpbmcgfCBudW1iZXJ9IGl0ZW0gLSBUaGUgYWN0dWFsIGl0ZW0gZnJvbSB0aGUgbGlzdC5cbiAgICogQHJldHVybnMge3N0cmluZyB8IG51bWJlcn0gVGhlIHRyYWNraW5nIGtleSBmb3IgdGhlIGl0ZW0uXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBvdmVycmlkZSB0cmFja0l0ZW1GbihpbmRleDogbnVtYmVyLCBpdGVtOiBLZXlWYWx1ZSB8IHN0cmluZyB8IG51bWJlcik6IHN0cmluZyB8IG51bWJlciB7XG4gICAgcmV0dXJuIGAkeyAoaXRlbSBhcyBLZXlWYWx1ZSk/LlsndWlkJ10gfHwgKGl0ZW0gYXMgS2V5VmFsdWUpPy5bdGhpcy5wa119LSR7aW5kZXh9YDtcbiAgfVxuXG5cbiAgLyoqXG4gICAqIEhhbmRsZXMgdGhlIGNyZWF0ZSBldmVudCBmcm9tIHRoZSByZXBvc2l0b3J5LlxuICAgKlxuICAgKiBAcGFyYW0ge3N0cmluZyB8IG51bWJlcn0gdWlkIC0gVGhlIElEIG9mIHRoZSBpdGVtIHRvIGNyZWF0ZS5cbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHdoZW4gdGhlIGl0ZW0gaXMgY3JlYXRlZCBhbmQgYWRkZWQgdG8gdGhlIGxpc3QuXG4gICAqL1xuICBhc3luYyBoYW5kbGVDcmVhdGUodWlkOiBzdHJpbmcgfCBudW1iZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0aGlzLl9yZXBvc2l0b3J5Py5yZWFkKHVpZCk7XG4gICAgY29uc3QgaXRlbSA9IHRoaXMubWFwUmVzdWx0cyhbcmVzdWx0IGFzIEtleVZhbHVlXSlbMF07XG4gICAgdGhpcy5pdGVtcyA9IHRoaXMuZGF0YSA9IFtpdGVtLCAuLi50aGlzLml0ZW1zIHx8IFtdXTtcbiAgfVxuXG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBIYW5kbGVzIHRoZSB1cGRhdGUgZXZlbnQgZnJvbSB0aGUgcmVwb3NpdG9yeS5cbiAgICogQHN1bW1hcnkgVXBkYXRlcyB0aGUgbGlzdCBpdGVtIHdpdGggdGhlIHNwZWNpZmllZCBJRCBiYXNlZCBvbiB0aGUgbmV3IGRhdGEuXG4gICAqXG4gICAqIEBwYXJhbSB7c3RyaW5nIHwgbnVtYmVyfSB1aWQgLSBUaGUgSUQgb2YgdGhlIGl0ZW0gdG8gdXBkYXRlXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fVxuICAgKiBAcHJpdmF0ZVxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgYXN5bmMgaGFuZGxlVXBkYXRlKHVpZDogc3RyaW5nIHwgbnVtYmVyKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgaXRlbTogS2V5VmFsdWUgPSB0aGlzLml0ZW1NYXBwZXIoYXdhaXQgdGhpcy5fcmVwb3NpdG9yeT8ucmVhZCh1aWQpIHx8IHt9LCB0aGlzLm1hcHBlcik7XG4gICAgdGhpcy5kYXRhID0gW107XG4gICAgZm9yKGNvbnN0IGtleSBpbiB0aGlzLml0ZW1zIGFzIEtleVZhbHVlW10pIHtcbiAgICAgICAgY29uc3QgY2hpbGQgPSB0aGlzLml0ZW1zW2tleV0gYXMgS2V5VmFsdWU7XG4gICAgICAgIGlmKGNoaWxkWyd1aWQnXSA9PT0gaXRlbVsndWlkJ10pIHtcbiAgICAgICAgICB0aGlzLml0ZW1zW2tleV0gPSBPYmplY3QuYXNzaWduKHt9LCBjaGlsZCwgaXRlbSk7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIH1cbiAgICB9XG4gICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICB0aGlzLmRhdGEgPSBbIC4uLnRoaXMuaXRlbXNdO1xuICAgIH0sIDApO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBSZW1vdmVzIGFuIGl0ZW0gZnJvbSB0aGUgbGlzdCBieSBJRC5cbiAgICogQHN1bW1hcnkgRmlsdGVycyBvdXQgYW4gaXRlbSB3aXRoIHRoZSBzcGVjaWZpZWQgSUQgZnJvbSB0aGUgZGF0YSBhcnJheSBhbmRcbiAgICogcmVmcmVzaGVzIHRoZSBsaXN0IGRpc3BsYXkuIFRoaXMgaXMgdHlwaWNhbGx5IHVzZWQgYWZ0ZXIgYSBkZWxldGUgb3BlcmF0aW9uLlxuICAgKlxuICAgKiBAcGFyYW0ge3N0cmluZ30gdWlkIC0gVGhlIElEIG9mIHRoZSBpdGVtIHRvIGRlbGV0ZVxuICAgKiBAcGFyYW0ge3N0cmluZ30gcGsgLSBUaGUgcHJpbWFyeSBrZXkgZmllbGQgbmFtZVxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn1cbiAgICpcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIGhhbmRsZURlbGV0ZSh1aWQ6IHN0cmluZyB8IG51bWJlciwgcGs/OiBzdHJpbmcpOiB2b2lkICB7XG4gICAgaWYoIXBrKVxuICAgICAgcGsgPSB0aGlzLnBrO1xuICAgIHRoaXMuaXRlbXMgPSB0aGlzLmRhdGE/LmZpbHRlcigoaXRlbTogS2V5VmFsdWUpID0+IGl0ZW1bJ3VpZCddICE9PSB1aWQpIHx8IFtdO1xuICB9XG5cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEhhbmRsZXMgY2xpY2sgZXZlbnRzIGZyb20gbGlzdCBpdGVtcy5cbiAgICogQHN1bW1hcnkgTGlzdGVucyBmb3IgZ2xvYmFsIExpc3RJdGVtQ2xpY2tFdmVudCBldmVudHMgYW5kIHBhc3NlcyB0aGVtIHRvIHRoZVxuICAgKiBkZWJvdW5jZWQgY2xpY2sgc3ViamVjdC4gVGhpcyBhbGxvd3MgdGhlIGNvbXBvbmVudCB0byByZXNwb25kIHRvIGNsaWNrcyBvblxuICAgKiBsaXN0IGl0ZW1zIHJlZ2FyZGxlc3Mgb2Ygd2hlcmUgdGhleSBvcmlnaW5hdGUgZnJvbS5cbiAgICpcbiAgICogQHBhcmFtIHtMaXN0SXRlbUN1c3RvbUV2ZW50IHwgUmVuZGVyZXJDdXN0b21FdmVudH0gZXZlbnQgLSBUaGUgY2xpY2sgZXZlbnRcbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqXG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBASG9zdExpc3RlbmVyKCd3aW5kb3c6TGlzdEl0ZW1DbGlja0V2ZW50JywgWyckZXZlbnQnXSlcbiAgaGFuZGxlQ2xpY2soZXZlbnQ6IExpc3RJdGVtQ3VzdG9tRXZlbnQgfCBSZW5kZXJlckN1c3RvbUV2ZW50KTogdm9pZCB7XG4gICAgdGhpcy5jbGlja0l0ZW1TdWJqZWN0Lm5leHQoZXZlbnQpO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBIYW5kbGVzIHNlYXJjaCBldmVudHMgZnJvbSB0aGUgc2VhcmNoIGJhci5cbiAgICogQHN1bW1hcnkgUHJvY2Vzc2VzIHNlYXJjaCBxdWVyaWVzIGZyb20gdGhlIHNlYXJjaCBiYXIgY29tcG9uZW50LCB1cGRhdGluZyB0aGVcbiAgICogZGlzcGxheWVkIGRhdGEgYmFzZWQgb24gdGhlIHNlYXJjaCB0ZXJtLiBUaGUgYmVoYXZpb3IgZGlmZmVycyBiZXR3ZWVuIGluZmluaXRlXG4gICAqIGFuZCBwYWdpbmF0ZWQgbW9kZXMgdG8gcHJvdmlkZSB0aGUgYmVzdCB1c2VyIGV4cGVyaWVuY2UgZm9yIGVhY2ggbW9kZS5cbiAgICpcbiAgICogQHBhcmFtIHtzdHJpbmcgfCB1bmRlZmluZWR9IHZhbHVlIC0gVGhlIHNlYXJjaCB0ZXJtIG9yIHVuZGVmaW5lZCB0byBjbGVhciBzZWFyY2hcbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59XG4gICAqXG4gICAqIEBtZXJtYWlkXG4gICAqIGZsb3djaGFydCBURFxuICAgKiAgIEFbU2VhcmNoIEV2ZW50XSAtLT4gQntUeXBlIGlzIEluZmluaXRlP31cbiAgICogICBCIC0tPnxZZXN8IENbRGlzYWJsZSBsb2FkTW9yZURhdGFdXG4gICAqICAgQiAtLT58Tm98IERbRW5hYmxlIGxvYWRNb3JlRGF0YV1cbiAgICogICBDIC0tPiBFe1NlYXJjaCB2YWx1ZSB1bmRlZmluZWQ/fVxuICAgKiAgIEUgLS0+fFllc3wgRltFbmFibGUgbG9hZE1vcmVEYXRhXVxuICAgKiAgIEUgLS0+fE5vfCBHW1N0b3JlIHNlYXJjaCB2YWx1ZV1cbiAgICogICBEIC0tPiBHXG4gICAqICAgRiAtLT4gSFtSZXNldCBwYWdlIHRvIDFdXG4gICAqICAgRyAtLT4gSVtSZWZyZXNoIGRhdGFdXG4gICAqICAgSCAtLT4gSVxuICAgKlxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgQEhvc3RMaXN0ZW5lcignd2luZG93OnNlYXJjaGJhckV2ZW50JywgWyckZXZlbnQnXSlcbiAgYXN5bmMgaGFuZGxlU2VhcmNoKHZhbHVlOiBzdHJpbmcgfCBJRmlsdGVyUXVlcnkgfCB1bmRlZmluZWQpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBpZih0aGlzLnR5cGUgPT09IExpc3RDb21wb25lbnRzVHlwZXMuSU5GSU5JVEUpIHtcbiAgICAgIHRoaXMubG9hZE1vcmVEYXRhID0gZmFsc2U7XG4gICAgICBpZih2YWx1ZSA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIHRoaXMubG9hZE1vcmVEYXRhID0gdHJ1ZTtcbiAgICAgICAgdGhpcy5wYWdlID0gMTtcbiAgICAgIH1cbiAgICAgIHRoaXMuc2VhcmNoVmFsdWUgPSB2YWx1ZTtcbiAgICAgIGF3YWl0IHRoaXMucmVmcmVzaCh0cnVlKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5sb2FkTW9yZURhdGEgPSB0cnVlO1xuICAgICAgdGhpcy5zZWFyY2hWYWx1ZSA9IHZhbHVlO1xuICAgICAgaWYodmFsdWUgPT09IHVuZGVmaW5lZClcbiAgICAgICAgdGhpcy5wYWdlID0gdGhpcy5sYXN0UGFnZTtcbiAgICAgIGF3YWl0IHRoaXMucmVmcmVzaCh0cnVlKTtcbiAgICB9XG4gIH1cblxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSGFuZGxlcyBmaWx0ZXIgZXZlbnRzIGZyb20gdGhlIGZpbHRlciBjb21wb25lbnQuXG4gICAqIEBzdW1tYXJ5IFByb2Nlc3NlcyBmaWx0ZXIgcXVlcmllcyBmcm9tIHRoZSBmaWx0ZXIgY29tcG9uZW50IGFuZCBhcHBsaWVzIHRoZW1cbiAgICogdG8gdGhlIGxpc3QgZGF0YS4gVGhpcyBtZXRob2QgYWN0cyBhcyBhIGJyaWRnZSBiZXR3ZWVuIHRoZSBmaWx0ZXIgY29tcG9uZW50XG4gICAqIGFuZCB0aGUgc2VhcmNoIGZ1bmN0aW9uYWxpdHksIGNvbnZlcnRpbmcgZmlsdGVyIHF1ZXJpZXMgaW50byBzZWFyY2ggb3BlcmF0aW9ucy5cbiAgICpcbiAgICogQHBhcmFtIHtJRmlsdGVyUXVlcnkgfCB1bmRlZmluZWR9IHZhbHVlIC0gVGhlIGZpbHRlciBxdWVyeSBvYmplY3Qgb3IgdW5kZWZpbmVkIHRvIGNsZWFyIGZpbHRlcnNcbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59XG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBhc3luYyBoYW5kbGVGaWx0ZXIodmFsdWU6IElGaWx0ZXJRdWVyeSB8IHVuZGVmaW5lZCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGF3YWl0IHRoaXMuaGFuZGxlU2VhcmNoKHZhbHVlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ2xlYXJzIHRoZSBjdXJyZW50IHNlYXJjaCBhbmQgcmVzZXRzIHRoZSBsaXN0LlxuICAgKiBAc3VtbWFyeSBDb252ZW5pZW5jZSBtZXRob2QgdGhhdCBjbGVhcnMgdGhlIHNlYXJjaCBieSBjYWxsaW5nIGhhbmRsZVNlYXJjaFxuICAgKiB3aXRoIHVuZGVmaW5lZC4gVGhpcyByZXNldHMgdGhlIGxpc3QgdG8gc2hvdyBhbGwgZGF0YSB3aXRob3V0IGZpbHRlcmluZy5cbiAgICpcbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59XG4gICAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gICAqL1xuICBhc3luYyBjbGVhclNlYXJjaCgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBhd2FpdCB0aGlzLmhhbmRsZVNlYXJjaCh1bmRlZmluZWQpO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBFbWl0cyBhIHJlZnJlc2ggZXZlbnQgd2l0aCB0aGUgY3VycmVudCBkYXRhLlxuICAgKiBAc3VtbWFyeSBDcmVhdGVzIGFuZCBlbWl0cyBhIHJlZnJlc2ggZXZlbnQgY29udGFpbmluZyB0aGUgY3VycmVudCBsaXN0IGRhdGEuXG4gICAqIFRoaXMgbm90aWZpZXMgcGFyZW50IGNvbXBvbmVudHMgdGhhdCB0aGUgbGlzdCBkYXRhIGhhcyBiZWVuIHJlZnJlc2hlZC5cbiAgICpcbiAgICogQHBhcmFtIHtLZXlWYWx1ZVtdfSBbZGF0YV0gLSBPcHRpb25hbCBkYXRhIHRvIGluY2x1ZGUgaW4gdGhlIGV2ZW50XG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKlxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgcmVmcmVzaEV2ZW50RW1pdChkYXRhPzogS2V5VmFsdWVbXSk6IHZvaWQge1xuICAgIGlmKCFkYXRhKVxuICAgICAgZGF0YSA9IHRoaXMuaXRlbXM7XG4gICAgdGhpcy5za2VsZXRvbkRhdGEgPSBuZXcgQXJyYXkoZGF0YT8ubGVuZ3RoIHx8IDIpO1xuICAgIHRoaXMucmVmcmVzaEV2ZW50LmVtaXQoe1xuICAgICAgbmFtZTogRXZlbnRDb25zdGFudHMuUkVGUkVTSF9FVkVOVCxcbiAgICAgIGRhdGE6IGRhdGEgfHwgW10sXG4gICAgICBjb21wb25lbnQ6IHRoaXMuY29tcG9uZW50TmFtZVxuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBFbWl0cyBhIGNsaWNrIGV2ZW50IGZvciBhIGxpc3QgaXRlbS5cbiAgICogQHN1bW1hcnkgUHJvY2Vzc2VzIGFuZCBlbWl0cyBhIGNsaWNrIGV2ZW50IHdoZW4gYSBsaXN0IGl0ZW0gaXMgY2xpY2tlZC5cbiAgICogVGhpcyBleHRyYWN0cyB0aGUgcmVsZXZhbnQgZGF0YSBmcm9tIHRoZSBldmVudCBhbmQgcGFzc2VzIGl0IHRvIHBhcmVudCBjb21wb25lbnRzLlxuICAgKlxuICAgKiBAcHJpdmF0ZVxuICAgKiBAcGFyYW0ge0xpc3RJdGVtQ3VzdG9tRXZlbnQgfCBSZW5kZXJlckN1c3RvbUV2ZW50fSBldmVudCAtIFRoZSBjbGljayBldmVudFxuICAgKiBAcmV0dXJucyB7dm9pZH1cbiAgICpcbiAgICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAgICovXG4gIHByaXZhdGUgY2xpY2tFdmVudEVtaXQoZXZlbnQ6IExpc3RJdGVtQ3VzdG9tRXZlbnQgfCBSZW5kZXJlckN1c3RvbUV2ZW50KTogdm9pZCB7XG4gICAgdGhpcy5jbGlja0V2ZW50LmVtaXQoZXZlbnQpO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBSZWZyZXNoZXMgdGhlIGxpc3QgZGF0YSBmcm9tIHRoZSBjb25maWd1cmVkIHNvdXJjZS5cbiAgICogQHN1bW1hcnkgVGhpcyBtZXRob2QgaGFuZGxlcyBib3RoIGluaXRpYWwgZGF0YSBsb2FkaW5nIGFuZCBzdWJzZXF1ZW50IHJlZnJlc2ggb3BlcmF0aW9ucyxcbiAgICogaW5jbHVkaW5nIHB1bGwtdG8tcmVmcmVzaCBhbmQgaW5maW5pdGUgc2Nyb2xsaW5nLiBJdCBtYW5hZ2VzIHRoZSBkYXRhIGZldGNoaW5nIHByb2Nlc3MsXG4gICAqIHVwZGF0ZXMgdGhlIGNvbXBvbmVudCdzIHN0YXRlLCBhbmQgaGFuZGxlcyBwYWdpbmF0aW9uIG9yIGluZmluaXRlIHNjcm9sbGluZyBsb2dpYyBiYXNlZFxuICAgKiBvbiB0aGUgY29tcG9uZW50J3MgY29uZmlndXJhdGlvbi5cbiAgICpcbiAgICogVGhlIG1ldGhvZCBwZXJmb3JtcyB0aGUgZm9sbG93aW5nIHN0ZXBzOlxuICAgKiAxLiBTZXRzIHRoZSByZWZyZXNoaW5nIGZsYWcgdG8gaW5kaWNhdGUgYSBkYXRhIGZldGNoIGlzIGluIHByb2dyZXNzXG4gICAqIDIuIENhbGN1bGF0ZXMgdGhlIGFwcHJvcHJpYXRlIHN0YXJ0IGFuZCBsaW1pdCB2YWx1ZXMgYmFzZWQgb24gcGFnaW5hdGlvbiBzZXR0aW5nc1xuICAgKiAzLiBGZXRjaGVzIGRhdGEgZnJvbSB0aGUgYXBwcm9wcmlhdGUgc291cmNlIChtb2RlbCBvciByZXF1ZXN0KVxuICAgKiA0LiBVcGRhdGVzIHRoZSBjb21wb25lbnQncyBkYXRhIGFuZCBlbWl0cyBhIHJlZnJlc2ggZXZlbnRcbiAgICogNS4gSGFuZGxlcyBwYWdpbmF0aW9uIG9yIGluZmluaXRlIHNjcm9sbGluZyBzdGF0ZSB1cGRhdGVzXG4gICAqIDYuIENvbXBsZXRlcyBhbnkgcHJvdmlkZWQgZXZlbnQgKGxpa2UgSW5maW5pdGVTY3JvbGxDdXN0b21FdmVudClcbiAgICpcbiAgICogQHBhcmFtIHtJbmZpbml0ZVNjcm9sbEN1c3RvbUV2ZW50IHwgUmVmcmVzaGVyQ3VzdG9tRXZlbnQgfCBib29sZWFufSBldmVudCAtIFRoZSBldmVudCB0aGF0IHRyaWdnZXJlZCB0aGUgcmVmcmVzaCxcbiAgICogb3IgYSBib29sZWFuIGZsYWcgaW5kaWNhdGluZyBpZiB0aGlzIGlzIGEgZm9yY2VkIHJlZnJlc2hcbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHdoZW4gdGhlIHJlZnJlc2ggb3BlcmF0aW9uIGlzIGNvbXBsZXRlXG4gICAqXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IEwgYXMgTGlzdENvbXBvbmVudFxuICAgKiAgIHBhcnRpY2lwYW50IEQgYXMgRGF0YSBTb3VyY2VcbiAgICogICBwYXJ0aWNpcGFudCBFIGFzIEV2ZW50IFN5c3RlbVxuICAgKlxuICAgKiAgIEwtPj5MOiByZWZyZXNoKGV2ZW50KVxuICAgKiAgIEwtPj5MOiBTZXQgcmVmcmVzaGluZyBmbGFnXG4gICAqICAgTC0+Pkw6IENhbGN1bGF0ZSBzdGFydCBhbmQgbGltaXRcbiAgICogICBhbHQgVXNpbmcgbW9kZWxcbiAgICogICAgIEwtPj5EOiBnZXRGcm9tTW9kZWwoZm9yY2UsIHN0YXJ0LCBsaW1pdClcbiAgICogICAgIEQtLT4+TDogUmV0dXJuIGRhdGFcbiAgICogICBlbHNlIFVzaW5nIHJlcXVlc3RcbiAgICogICAgIEwtPj5EOiBnZXRGcm9tUmVxdWVzdChmb3JjZSwgc3RhcnQsIGxpbWl0KVxuICAgKiAgICAgRC0tPj5MOiBSZXR1cm4gZGF0YVxuICAgKiAgIGVuZFxuICAgKiAgIEwtPj5FOiByZWZyZXNoRXZlbnRFbWl0KClcbiAgICogICBhbHQgSW5maW5pdGUgc2Nyb2xsaW5nIG1vZGVcbiAgICogICAgIEwtPj5MOiBDaGVjayBpZiByZWFjaGVkIGxhc3QgcGFnZVxuICAgKiAgICAgYWx0IExhc3QgcGFnZSByZWFjaGVkXG4gICAqICAgICAgIEwtPj5MOiBDb21wbGV0ZSBzY3JvbGwgZXZlbnRcbiAgICogICAgICAgTC0+Pkw6IERpc2FibGUgbG9hZE1vcmVEYXRhXG4gICAqICAgICBlbHNlIE1vcmUgcGFnZXMgYXZhaWxhYmxlXG4gICAqICAgICAgIEwtPj5MOiBJbmNyZW1lbnQgcGFnZSBudW1iZXJcbiAgICogICAgICAgTC0+Pkw6IENvbXBsZXRlIHNjcm9sbCBldmVudCBhZnRlciBkZWxheVxuICAgKiAgICAgZW5kXG4gICAqICAgZWxzZSBQYWdpbmF0ZWQgbW9kZVxuICAgKiAgICAgTC0+Pkw6IENsZWFyIHJlZnJlc2hpbmcgZmxhZyBhZnRlciBkZWxheVxuICAgKiAgIGVuZFxuICAgKlxuICAgKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICAgKi9cbiAgQEhvc3RMaXN0ZW5lcignd2luZG93OkJhY2tCdXR0b25OYXZpZ2F0aW9uRW5kRXZlbnQnLCBbJyRldmVudCddKVxuICBhc3luYyByZWZyZXNoKGV2ZW50OiBJbmZpbml0ZVNjcm9sbEN1c3RvbUV2ZW50IHwgUmVmcmVzaGVyQ3VzdG9tRXZlbnQgfCBib29sZWFuID0gZmFsc2UpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyAgaWYodHlwZW9mIGZvcmNlICE9PSAnYm9vbGVhbicgJiYgZm9yY2UudHlwZSA9PT0gRXZlbnRDb25zdGFudHMuQkFDS19CVVRUT05fTkFWSUdBVElPTikge1xuICAgIC8vICAgIGNvbnN0IHtyZWZyZXNofSA9IChmb3JjZSBhcyBDdXN0b21FdmVudCkuZGV0YWlsO1xuICAgIC8vICAgIGlmKCFyZWZyZXNoKVxuICAgIC8vICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIC8vICB9XG5cbiAgICB0aGlzLnJlZnJlc2hpbmcgPSB0cnVlO1xuICAgIGNvbnN0IHN0YXJ0OiBudW1iZXIgPSB0aGlzLnBhZ2UgPiAxID8gKHRoaXMucGFnZSAtIDEpICogdGhpcy5saW1pdCA6IHRoaXMuc3RhcnQ7XG4gICAgY29uc3QgbGltaXQ6IG51bWJlciA9ICh0aGlzLnBhZ2UgKiAodGhpcy5saW1pdCA+IDEyID8gMTIgOiB0aGlzLmxpbWl0KSk7XG5cbiAgICB0aGlzLmRhdGEgPSAhdGhpcy5tb2RlbCA/XG4gICAgICBhd2FpdCB0aGlzLmdldEZyb21SZXF1ZXN0KCEhZXZlbnQsIHN0YXJ0LCBsaW1pdClcbiAgICAgIDogYXdhaXQgdGhpcy5nZXRGcm9tTW9kZWwoISFldmVudCkgYXMgS2V5VmFsdWVbXTtcblxuICAgIHRoaXMucmVmcmVzaEV2ZW50RW1pdCgpO1xuXG4gICAgaWYodGhpcy50eXBlID09PSBMaXN0Q29tcG9uZW50c1R5cGVzLklORklOSVRFKSB7XG4gICAgICBpZih0aGlzLnBhZ2UgPT09IHRoaXMucGFnZXMpIHtcbiAgICAgICAgaWYoKGV2ZW50IGFzIEluZmluaXRlU2Nyb2xsQ3VzdG9tRXZlbnQpPy50YXJnZXQpXG4gICAgICAgICAgKGV2ZW50IGFzIEluZmluaXRlU2Nyb2xsQ3VzdG9tRXZlbnQpLnRhcmdldC5jb21wbGV0ZSgpO1xuICAgICAgICB0aGlzLmxvYWRNb3JlRGF0YSA9IGZhbHNlO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy5wYWdlICs9IDE7XG4gICAgICAgIHRoaXMucmVmcmVzaGluZyA9IGZhbHNlO1xuICAgICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICAgIGlmKChldmVudCBhcyBJbmZpbml0ZVNjcm9sbEN1c3RvbUV2ZW50KT8udGFyZ2V0ICYmIChldmVudCBhcyBDdXN0b21FdmVudCk/LnR5cGUgIT09IEV2ZW50Q29uc3RhbnRzLkJBQ0tfQlVUVE9OX05BVklHQVRJT04pXG4gICAgICAgICAgICAgIChldmVudCBhcyBJbmZpbml0ZVNjcm9sbEN1c3RvbUV2ZW50KS50YXJnZXQuY29tcGxldGUoKTtcbiAgICAgICAgfSwgMjAwKTtcbiAgICAgIH1cbiAgICB9IGVsc2Uge1xuICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgIHRoaXMucmVmcmVzaGluZyA9IGZhbHNlO1xuICAgICAgfSwgMjAwKVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICogQGRlc2NyaXB0aW9uIEhhbmRsZXMgcGFnaW5hdGlvbiBldmVudHMgZnJvbSB0aGUgcGFnaW5hdGlvbiBjb21wb25lbnQuXG4gKiBAc3VtbWFyeSBQcm9jZXNzZXMgcGFnaW5hdGlvbiBldmVudHMgYnkgdXBkYXRpbmcgdGhlIGN1cnJlbnQgcGFnZSBudW1iZXIgYW5kXG4gKiByZWZyZXNoaW5nIHRoZSBsaXN0IGRhdGEgdG8gZGlzcGxheSB0aGUgc2VsZWN0ZWQgcGFnZS4gVGhpcyBtZXRob2QgaXMgY2FsbGVkXG4gKiB3aGVuIGEgdXNlciBpbnRlcmFjdHMgd2l0aCB0aGUgcGFnaW5hdGlvbiBjb250cm9scyB0byBuYXZpZ2F0ZSBiZXR3ZWVuIHBhZ2VzLlxuICpcbiAqIEBwYXJhbSB7UGFnaW5hdGlvbkN1c3RvbUV2ZW50fSBldmVudCAtIFRoZSBwYWdpbmF0aW9uIGV2ZW50IGNvbnRhaW5pbmcgcGFnZSBpbmZvcm1hdGlvblxuICogQHJldHVybnMge3ZvaWR9XG4gKlxuICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAqL1xuaGFuZGxlUGFnaW5hdGUoZXZlbnQ6IFBhZ2luYXRpb25DdXN0b21FdmVudCk6IHZvaWQge1xuICBjb25zdCB7IHBhZ2V9ID0gZXZlbnQuZGF0YTtcbiAgdGhpcy5wYWdlID0gcGFnZTtcbiAgdGhpcy5yZWZyZXNoKHRydWUpO1xufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBIYW5kbGVzIHB1bGwtdG8tcmVmcmVzaCBldmVudHMgZnJvbSB0aGUgcmVmcmVzaGVyIGNvbXBvbmVudC5cbiAqIEBzdW1tYXJ5IFByb2Nlc3NlcyByZWZyZXNoIGV2ZW50cyB0cmlnZ2VyZWQgYnkgdGhlIHVzZXIgcHVsbGluZyBkb3duIG9uIHRoZSBsaXN0XG4gKiBvciBieSBwcm9ncmFtbWF0aWMgcmVmcmVzaCByZXF1ZXN0cy4gVGhpcyBtZXRob2QgcmVmcmVzaGVzIHRoZSBsaXN0IGRhdGEgYW5kXG4gKiBjb21wbGV0ZXMgdGhlIHJlZnJlc2hlciBhbmltYXRpb24gd2hlbiB0aGUgZGF0YSBpcyBsb2FkZWQuXG4gKlxuICogQHBhcmFtIHtJbmZpbml0ZVNjcm9sbEN1c3RvbUV2ZW50IHwgQ3VzdG9tRXZlbnR9IFtldmVudF0gLSBUaGUgcmVmcmVzaCBldmVudFxuICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHdoZW4gdGhlIHJlZnJlc2ggb3BlcmF0aW9uIGlzIGNvbXBsZXRlXG4gKlxuICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAqL1xuYXN5bmMgaGFuZGxlUmVmcmVzaChldmVudD86IEluZmluaXRlU2Nyb2xsQ3VzdG9tRXZlbnQgfCBDdXN0b21FdmVudCk6IFByb21pc2U8dm9pZD4ge1xuICBhd2FpdCB0aGlzLnJlZnJlc2goZXZlbnQgYXMgSW5maW5pdGVTY3JvbGxDdXN0b21FdmVudCB8fCB0cnVlKTtcbiAgaWYoZXZlbnQgaW5zdGFuY2VvZiBDdXN0b21FdmVudClcbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIC8vIEFueSBjYWxscyB0byBsb2FkIGRhdGEgZ28gaGVyZVxuICAgICAgKGV2ZW50LnRhcmdldCBhcyBIVE1MSW9uUmVmcmVzaGVyRWxlbWVudCkuY29tcGxldGUoKTtcbiAgICB9LCA0MDApO1xufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBGaWx0ZXJzIGRhdGEgYmFzZWQgb24gYSBzZWFyY2ggc3RyaW5nLlxuICogQHN1bW1hcnkgUHJvY2Vzc2VzIHRoZSBjdXJyZW50IGRhdGEgYXJyYXkgdG8gZmluZCBpdGVtcyB0aGF0IG1hdGNoIHRoZSBwcm92aWRlZFxuICogc2VhcmNoIHN0cmluZy4gVGhpcyB1c2VzIHRoZSBhcnJheVF1ZXJ5QnlTdHJpbmcgdXRpbGl0eSB0byBwZXJmb3JtIHRoZSBmaWx0ZXJpbmdcbiAqIGFjcm9zcyBhbGwgcHJvcGVydGllcyBvZiB0aGUgaXRlbXMuXG4gKlxuICogQHBhcmFtIHtLZXlWYWx1ZVtdfSByZXN1bHRzIC0gVGhlIGFycmF5IG9mIGl0ZW1zIHRvIHNlYXJjaCB0aHJvdWdoXG4gKiBAcGFyYW0ge3N0cmluZ30gc2VhcmNoIC0gVGhlIHNlYXJjaCBzdHJpbmcgdG8gZmlsdGVyIGJ5XG4gKiBAcmV0dXJucyB7S2V5VmFsdWVbXX0gQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdGhlIGZpbHRlcmVkIGFycmF5IG9mIGl0ZW1zXG4gKlxuICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAqL1xuICBwYXJzZVNlYXJjaFJlc3VsdHMocmVzdWx0czogS2V5VmFsdWVbXSwgc2VhcmNoOiBzdHJpbmcpOiBLZXlWYWx1ZVtdIHtcbiAgICByZXR1cm4gcmVzdWx0cy5maWx0ZXIoKGl0ZW06IEtleVZhbHVlKSA9PlxuICAgICAgT2JqZWN0LnZhbHVlcyhpdGVtKS5zb21lKHZhbHVlID0+XG4gICAgICAgICAgdmFsdWUudG9TdHJpbmcoKS50b0xvd2VyQ2FzZSgpLmluY2x1ZGVzKChzZWFyY2ggYXMgc3RyaW5nKT8udG9Mb3dlckNhc2UoKSlcbiAgICAgICAgKVxuICAgICk7XG4gIH1cblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gRmV0Y2hlcyBkYXRhIGZyb20gYSByZXF1ZXN0IHNvdXJjZS5cbiAqIEBzdW1tYXJ5IFJldHJpZXZlcyBkYXRhIGZyb20gdGhlIGNvbmZpZ3VyZWQgc291cmNlIGZ1bmN0aW9uIG9yIFVSTCwgcHJvY2Vzc2VzIGl0LFxuICogYW5kIHVwZGF0ZXMgdGhlIGNvbXBvbmVudCdzIGRhdGEgc3RhdGUuIFRoaXMgbWV0aG9kIGhhbmRsZXMgYm90aCBpbml0aWFsIGRhdGEgbG9hZGluZ1xuICogYW5kIHN1YnNlcXVlbnQgcmVmcmVzaCBvcGVyYXRpb25zIHdoZW4gdXNpbmcgYW4gZXh0ZXJuYWwgZGF0YSBzb3VyY2UgcmF0aGVyIHRoYW4gYSBtb2RlbC5cbiAqXG4gKiBAcGFyYW0ge2Jvb2xlYW59IGZvcmNlIC0gV2hldGhlciB0byBmb3JjZSBhIHJlZnJlc2ggZXZlbiBpZiBkYXRhIGFscmVhZHkgZXhpc3RzXG4gKiBAcGFyYW0ge251bWJlcn0gc3RhcnQgLSBUaGUgc3RhcnRpbmcgaW5kZXggZm9yIHBhZ2luYXRpb25cbiAqIEBwYXJhbSB7bnVtYmVyfSBsaW1pdCAtIFRoZSBtYXhpbXVtIG51bWJlciBvZiBpdGVtcyB0byByZXRyaWV2ZVxuICogQHJldHVybnMge1Byb21pc2U8S2V5VmFsdWVbXT59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSBmZXRjaGVkIGRhdGFcbiAqXG4gKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICovXG5hc3luYyBnZXRGcm9tUmVxdWVzdChmb3JjZTogYm9vbGVhbiA9IGZhbHNlLCBzdGFydDogbnVtYmVyLCBsaW1pdDogbnVtYmVyKTogUHJvbWlzZTxLZXlWYWx1ZVtdPiB7XG4gIGxldCByZXF1ZXN0OiBLZXlWYWx1ZVtdID0gW107XG4gIGlmKCF0aGlzLmRhdGE/Lmxlbmd0aCB8fCBmb3JjZSB8fCAodGhpcy5zZWFyY2hWYWx1ZSBhcyBzdHJpbmcpPy5sZW5ndGggfHwgISEodGhpcy5zZWFyY2hWYWx1ZSBhcyBJRmlsdGVyUXVlcnkpKSB7XG4gICAgLy8gKHNlbGYuZGF0YSBhcyBMaXN0SXRlbVtdKSA9IFtdO1xuICAgIGlmKCEodGhpcy5zZWFyY2hWYWx1ZSBhcyBzdHJpbmcpPy5sZW5ndGggJiYgISh0aGlzLnNlYXJjaFZhbHVlIGFzIElGaWx0ZXJRdWVyeSkpIHtcbiAgICAgIGlmKCF0aGlzLnNvdXJjZSAmJiAhdGhpcy5kYXRhPy5sZW5ndGgpIHtcbiAgICAgICAgdGhpcy5sb2dnZXIuaW5mbygnTm8gZGF0YSBhbmQgc291cmNlIHBhc3NlZCB0byBpbmZpbml0ZSBsaXN0Jyk7XG4gICAgICAgIHJldHVybiBbXTtcbiAgICAgIH1cblxuICAgICAgaWYodGhpcy5zb3VyY2UgaW5zdGFuY2VvZiBGdW5jdGlvbilcbiAgICAgICAgcmVxdWVzdCA9IGF3YWl0IHRoaXMuc291cmNlKCk7XG5cbiAgICAgIGlmKCFBcnJheS5pc0FycmF5KHJlcXVlc3QpKVxuICAgICAgICByZXF1ZXN0ID0gcmVxdWVzdD8uWydyZXNwb25zZSddPy5bJ2RhdGEnXSB8fCByZXF1ZXN0Py5bJ3Jlc3VsdHMnXSB8fCBbXTtcbiAgICAgIHRoaXMuZGF0YSA9IFsuLi4gYXdhaXQgdGhpcy5wYXJzZVJlc3VsdChyZXF1ZXN0KV07XG4gICAgICBpZih0aGlzLmRhdGE/Lmxlbmd0aClcbiAgICAgICAgdGhpcy5pdGVtcyA9IHRoaXMudHlwZSA9PT0gTGlzdENvbXBvbmVudHNUeXBlcy5JTkZJTklURSA/XG4gICAgICAgICAgKHRoaXMuaXRlbXMgfHwgW10pLmNvbmNhdChbLi4udGhpcy5kYXRhLnNsaWNlKHN0YXJ0LCBsaW1pdCldKSA6IFsuLi5yZXF1ZXN0LnNsaWNlKHN0YXJ0LCBsaW1pdCkgYXMgS2V5VmFsdWVbXV07XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMuZGF0YSA9IHRoaXMucGFyc2VTZWFyY2hSZXN1bHRzKHRoaXMuZGF0YSBhcyBbXSwgdGhpcy5zZWFyY2hWYWx1ZSBhcyBzdHJpbmcpO1xuICAgICAgdGhpcy5pdGVtcyA9IHRoaXMuZGF0YTtcbiAgICB9XG4gIH1cblxuICBpZih0aGlzLmxvYWRNb3JlRGF0YSAmJiB0aGlzLnR5cGUgPT09IExpc3RDb21wb25lbnRzVHlwZXMuUEFHSU5BVEVEKVxuICAgIHRoaXMuZ2V0TW9yZURhdGEodGhpcy5kYXRhPy5sZW5ndGggfHwgMCk7XG4gIHJldHVybiB0aGlzLmRhdGEgfHwgW10gYXMgS2V5VmFsdWVbXTtcbn1cblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gRmV0Y2hlcyBkYXRhIGZyb20gYSBtb2RlbCBzb3VyY2UuXG4gKiBAc3VtbWFyeSBSZXRyaWV2ZXMgZGF0YSBmcm9tIHRoZSBjb25maWd1cmVkIG1vZGVsIHVzaW5nIGl0cyBwYWdpbmF0aW9uIG9yIGZpbmQgbWV0aG9kcyxcbiAqIHByb2Nlc3NlcyBpdCwgYW5kIHVwZGF0ZXMgdGhlIGNvbXBvbmVudCdzIGRhdGEgc3RhdGUuIFRoaXMgbWV0aG9kIGhhbmRsZXMgYm90aCBpbml0aWFsXG4gKiBkYXRhIGxvYWRpbmcgYW5kIHN1YnNlcXVlbnQgcmVmcmVzaCBvcGVyYXRpb25zIHdoZW4gdXNpbmcgYSBtb2RlbCBhcyB0aGUgZGF0YSBzb3VyY2UuXG4gKlxuICogQHBhcmFtIHtib29sZWFufSBmb3JjZSAtIFdoZXRoZXIgdG8gZm9yY2UgYSByZWZyZXNoIGV2ZW4gaWYgZGF0YSBhbHJlYWR5IGV4aXN0c1xuICogQHBhcmFtIHtudW1iZXJ9IHN0YXJ0IC0gVGhlIHN0YXJ0aW5nIGluZGV4IGZvciBwYWdpbmF0aW9uXG4gKiBAcGFyYW0ge251bWJlcn0gbGltaXQgLSBUaGUgbWF4aW11bSBudW1iZXIgb2YgaXRlbXMgdG8gcmV0cmlldmVcbiAqIEByZXR1cm5zIHtQcm9taXNlPEtleVZhbHVlW10+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgZmV0Y2hlZCBkYXRhXG4gKlxuICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAqL1xuYXN5bmMgZ2V0RnJvbU1vZGVsKGZvcmNlOiBib29sZWFuID0gZmFsc2UpOiBQcm9taXNlPEtleVZhbHVlW10+IHtcbiAgbGV0IGRhdGEgPSBbIC4uLiB0aGlzLmRhdGEgfHwgW11dO1xuICBsZXQgcmVxdWVzdDogS2V5VmFsdWVbXSA9IFtdO1xuXG4gIC8vIGdldHRpbmcgbW9kZWwgcmVwb3NpdG9yeVxuICBpZighdGhpcy5fcmVwb3NpdG9yeSlcbiAgICB0aGlzLl9yZXBvc2l0b3J5ID0gdGhpcy5yZXBvc2l0b3J5O1xuICBjb25zdCByZXBvID0gdGhpcy5fcmVwb3NpdG9yeSBhcyBEZWNhZlJlcG9zaXRvcnk8TW9kZWw+O1xuICBpZighdGhpcy5kYXRhPy5sZW5ndGggfHwgZm9yY2UgfHwgKHRoaXMuc2VhcmNoVmFsdWUgYXMgc3RyaW5nKT8ubGVuZ3RoIHx8ICEhKHRoaXMuc2VhcmNoVmFsdWUgYXMgSUZpbHRlclF1ZXJ5KSkge1xuICAgIHRyeSB7XG4gICAgIGlmKCEodGhpcy5zZWFyY2hWYWx1ZSBhcyBzdHJpbmcpPy5sZW5ndGggJiYgISh0aGlzLnNlYXJjaFZhbHVlIGFzIElGaWx0ZXJRdWVyeSkpIHtcbiAgICAgICAgKHRoaXMuZGF0YSBhcyBLZXlWYWx1ZVtdKSA9IFtdO1xuICAgICAgICAvLyBjb25zdCByYXdRdWVyeSA9IHRoaXMucGFyc2VRdWVyeShzZWxmLm1vZGVsIGFzIFJlcG9zaXRvcnk8TW9kZWw+LCBzdGFydCwgbGltaXQpO1xuICAgICAgICAvLyByZXF1ZXN0ID0gdGhpcy5wYXJzZVJlc3VsdChhd2FpdCAodGhpcy5tb2RlbCBhcyBhbnkpPy5wYWdpbmF0ZShzdGFydCwgbGltaXQpKTtcbiAgICAgICAgICBpZighdGhpcy5wYWdpbmF0b3IpIHtcbiAgICAgICAgICAgIHRoaXMucGFnaW5hdG9yID0gYXdhaXQgcmVwb1xuICAgICAgICAgICAgICAuc2VsZWN0KClcbiAgICAgICAgICAgICAgLm9yZGVyQnkoW3RoaXMucGsgYXMga2V5b2YgTW9kZWwsIHRoaXMuc29ydERpcmVjdGlvbl0pXG4gICAgICAgICAgICAgIC5wYWdpbmF0ZSh0aGlzLmxpbWl0KTtcbiAgICAgICAgICB9XG4gICAgICAgICAgcmVxdWVzdCA9IGF3YWl0IHRoaXMucGFyc2VSZXN1bHQodGhpcy5wYWdpbmF0b3IpO1xuICAgICAgfSBlbHNlIHtcblxuICAgICAgICBpZighdGhpcy5pbmRleGVzKVxuICAgICAgICAgIHRoaXMuaW5kZXhlcyA9IChPYmplY3QudmFsdWVzKHRoaXMubWFwcGVyKSB8fCBbdGhpcy5wa10pO1xuXG4gICAgICAgIGNvbnN0IGNvbmRpdGlvbiA9IHRoaXMucGFyc2VDb25kaXRpb25zKHRoaXMuc2VhcmNoVmFsdWUgYXMgc3RyaW5nIHwgbnVtYmVyIHwgSUZpbHRlclF1ZXJ5KTtcbiAgICAgICAgcmVxdWVzdCA9IGF3YWl0IHRoaXMucGFyc2VSZXN1bHQoYXdhaXQgcmVwby5xdWVyeShjb25kaXRpb24sICh0aGlzLnNvcnRCeSB8fCB0aGlzLnBrKSBhcyBrZXlvZiBNb2RlbCwgdGhpcy5zb3J0RGlyZWN0aW9uKSk7XG4gICAgICAgIGRhdGEgPSBbXTtcbiAgICAgIH1cbiAgICAgIGRhdGEgPSB0aGlzLnR5cGUgPT09IExpc3RDb21wb25lbnRzVHlwZXMuSU5GSU5JVEUgPyBbLi4uIChkYXRhKS5jb25jYXQocmVxdWVzdCldIDogWy4uLnJlcXVlc3RdO1xuICAgIH0gY2F0Y2goZXJyb3I6IHVua25vd24pIHtcbiAgICAgIHRoaXMubG9nZ2VyLmVycm9yKChlcnJvciBhcyBFcnJvcik/Lm1lc3NhZ2UgfHwgYFVuYWJsZSB0byBmaW5kICR7dGhpcy5tb2RlbH0gb24gcmVnaXN0cnkuIFJldHVybiBlbXB0eSBhcnJheSBmcm9tIGNvbXBvbmVudGApO1xuICAgIH1cbiAgfVxuXG4gIGlmKGRhdGE/Lmxlbmd0aCkge1xuICAgIGlmKHRoaXMuc2VhcmNoVmFsdWUpIHtcbiAgICAgIHRoaXMuaXRlbXMgPSBbLi4uZGF0YV07XG4gICAgICBpZih0aGlzLml0ZW1zPy5sZW5ndGggPD0gdGhpcy5saW1pdClcbiAgICAgICAgdGhpcy5sb2FkTW9yZURhdGEgPSBmYWxzZTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5pdGVtcyA9IFsuLi5kYXRhXTtcbiAgICB9XG4gIH1cbiAgaWYodGhpcy50eXBlID09PSBMaXN0Q29tcG9uZW50c1R5cGVzLlBBR0lOQVRFRCAmJiB0aGlzLnBhZ2luYXRvcilcbiAgICAgIHRoaXMuZ2V0TW9yZURhdGEodGhpcy5wYWdpbmF0b3IudG90YWwpO1xuICByZXR1cm4gZGF0YSB8fCBbXSBhcyBLZXlWYWx1ZVtdO1xufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBDb252ZXJ0cyBzZWFyY2ggdmFsdWVzIG9yIGZpbHRlciBxdWVyaWVzIGludG8gZGF0YWJhc2UgY29uZGl0aW9ucy5cbiAqIEBzdW1tYXJ5IFRyYW5zZm9ybXMgc2VhcmNoIGlucHV0IG9yIGNvbXBsZXggZmlsdGVyIHF1ZXJpZXMgaW50byBDb25kaXRpb24gb2JqZWN0c1xuICogdGhhdCBjYW4gYmUgdXNlZCBmb3IgZGF0YWJhc2UgcXVlcnlpbmcuIEhhbmRsZXMgYm90aCBzaW1wbGUgc3RyaW5nL251bWJlciBzZWFyY2hlc1xuICogYWNyb3NzIGluZGV4ZWQgZmllbGRzIGFuZCBjb21wbGV4IGZpbHRlciBxdWVyaWVzIHdpdGggbXVsdGlwbGUgY3JpdGVyaWEuXG4gKlxuICogRm9yIHNpbXBsZSBzZWFyY2hlcyAoc3RyaW5nL251bWJlcik6XG4gKiAtIENyZWF0ZXMgY29uZGl0aW9ucyB0aGF0IHNlYXJjaCBhY3Jvc3MgYWxsIGluZGV4ZWQgZmllbGRzXG4gKiAtIFVzZXMgZXF1YWxpdHkgZm9yIG51bWVyaWMgdmFsdWVzIGFuZCByZWdleCBmb3Igc3RyaW5nIHZhbHVlc1xuICogLSBDb21iaW5lcyBjb25kaXRpb25zIHdpdGggT1IgbG9naWMgdG8gc2VhcmNoIG11bHRpcGxlIGZpZWxkc1xuICpcbiAqIEZvciBjb21wbGV4IGZpbHRlciBxdWVyaWVzOlxuICogLSBQcm9jZXNzZXMgZWFjaCBmaWx0ZXIgaXRlbSB3aXRoIGl0cyBzcGVjaWZpYyBjb25kaXRpb24gdHlwZVxuICogLSBTdXBwb3J0cyBFcXVhbCwgTm90IEVxdWFsLCBDb250YWlucywgTm90IENvbnRhaW5zLCBHcmVhdGVyIFRoYW4sIExlc3MgVGhhblxuICogLSBVcGRhdGVzIHNvcnQgY29uZmlndXJhdGlvbiBiYXNlZCBvbiB0aGUgZmlsdGVyIHF1ZXJ5XG4gKiAtIENvbWJpbmVzIG11bHRpcGxlIGZpbHRlciBjb25kaXRpb25zIHdpdGggT1IgbG9naWNcbiAqXG4gKiBAcGFyYW0ge3N0cmluZyB8IG51bWJlciB8IElGaWx0ZXJRdWVyeX0gdmFsdWUgLSBUaGUgc2VhcmNoIHZhbHVlIG9yIGZpbHRlciBxdWVyeSBvYmplY3RcbiAqIEByZXR1cm5zIHtDb25kaXRpb248TW9kZWw+fSBBIENvbmRpdGlvbiBvYmplY3QgZm9yIGRhdGFiYXNlIHF1ZXJ5aW5nXG4gKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICovXG5wYXJzZUNvbmRpdGlvbnModmFsdWU6IHN0cmluZyB8IG51bWJlciB8IElGaWx0ZXJRdWVyeSk6IENvbmRpdGlvbjxNb2RlbD4ge1xuICBsZXQgX2NvbmRpdGlvbjogQ29uZGl0aW9uPE1vZGVsPjtcbiAgaWYodHlwZW9mIHZhbHVlID09PSBQcmltaXRpdmVzLlNUUklORyB8fCB0eXBlb2YgdmFsdWUgPT09IFByaW1pdGl2ZXMuTlVNQkVSKSB7XG4gICAgX2NvbmRpdGlvbiA9IENvbmRpdGlvbi5hdHRyaWJ1dGU8TW9kZWw+KHRoaXMucGsgYXMga2V5b2YgTW9kZWwpLmVxKCFpc05hTih2YWx1ZSBhcyBudW1iZXIpID8gTnVtYmVyKHZhbHVlKSA6IHZhbHVlKTtcbiAgICBmb3IgKGNvbnN0IGluZGV4IG9mIHRoaXMuaW5kZXhlcykge1xuICAgICAgICBpZihpbmRleCA9PT0gdGhpcy5waylcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgbGV0IG9yQ29uZGl0aW9uO1xuICAgICAgICBpZighaXNOYU4odmFsdWUgYXMgbnVtYmVyKSkge1xuICAgICAgICAgIG9yQ29uZGl0aW9uID0gQ29uZGl0aW9uLmF0dHJpYnV0ZTxNb2RlbD4oaW5kZXggYXMga2V5b2YgTW9kZWwpLmVxKE51bWJlcih2YWx1ZSkpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIG9yQ29uZGl0aW9uID0gQ29uZGl0aW9uLmF0dHJpYnV0ZTxNb2RlbD4oaW5kZXggYXMga2V5b2YgTW9kZWwpLnJlZ2V4cCh2YWx1ZSBhcyBzdHJpbmcpO1xuICAgICAgICB9XG4gICAgICAgIF9jb25kaXRpb24gPSBfY29uZGl0aW9uLm9yKG9yQ29uZGl0aW9uKTtcbiAgICB9XG4gIH0gZWxzZSB7XG4gICAgY29uc3Qge3F1ZXJ5LCBzb3J0fSA9IHZhbHVlIGFzIElGaWx0ZXJRdWVyeTtcbiAgICBfY29uZGl0aW9uID0gQ29uZGl0aW9uLmF0dHJpYnV0ZTxNb2RlbD4odGhpcy5wayBhcyBrZXlvZiBNb2RlbCkuZGlmKCdudWxsJyk7XG5cbiAgICBpZihxdWVyeT8ubGVuZ3RoKVxuICAgICAgX2NvbmRpdGlvbiA9IHVuZGVmaW5lZCBhcyB1bmtub3duIGFzIENvbmRpdGlvbjxNb2RlbD47XG5cbiAgICAocXVlcnkgfHwgW10pLmZvckVhY2goKGl0ZW06IElGaWx0ZXJRdWVyeUl0ZW0pID0+IHtcbiAgICAgIGNvbnN0IHt2YWx1ZSwgY29uZGl0aW9uLCBpbmRleH0gPSBpdGVtO1xuICAgICAgbGV0IHZhbCA9IHZhbHVlIGFzIHN0cmluZyB8IG51bWJlcjtcbiAgICAgIGlmKGluZGV4ID09PSB0aGlzLnBrIHx8ICFpc05hTih2YWwgYXMgbnVtYmVyKSlcbiAgICAgICAgdmFsID0gTnVtYmVyKHZhbCk7XG4gICAgICBsZXQgb3JDb25kaXRpb247XG4gICAgICBzd2l0Y2ggKGNvbmRpdGlvbikge1xuICAgICAgICBjYXNlIFwiRXF1YWxcIjpcbiAgICAgICAgICBvckNvbmRpdGlvbiA9IENvbmRpdGlvbi5hdHRyaWJ1dGU8TW9kZWw+KGluZGV4IGFzIGtleW9mIE1vZGVsKS5lcSh2YWwpO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlIFwiTm90IEVxdWFsXCI6XG4gICAgICAgICAgb3JDb25kaXRpb24gPSBDb25kaXRpb24uYXR0cmlidXRlPE1vZGVsPihpbmRleCBhcyBrZXlvZiBNb2RlbCkuZGlmKHZhbCk7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgXCJOb3QgQ29udGFpbnNcIjpcbiAgICAgICAgICBvckNvbmRpdGlvbiA9ICFDb25kaXRpb24uYXR0cmlidXRlPE1vZGVsPihpbmRleCBhcyBrZXlvZiBNb2RlbCkucmVnZXhwKG5ldyBSZWdFeHAoYF4oPyEuKiR7dmFsfSkuKiRgKSk7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgXCJDb250YWluc1wiOlxuICAgICAgICAgIG9yQ29uZGl0aW9uID0gQ29uZGl0aW9uLmF0dHJpYnV0ZTxNb2RlbD4oaW5kZXggYXMga2V5b2YgTW9kZWwpLnJlZ2V4cCh2YWwgYXMgc3RyaW5nKTtcbiAgICAgICAgICBicmVhaztcbiAgICAgICAgY2FzZSBcIkdyZWF0ZXIgVGhhblwiOlxuICAgICAgICAgIG9yQ29uZGl0aW9uID0gQ29uZGl0aW9uLmF0dHJpYnV0ZTxNb2RlbD4oaW5kZXggYXMga2V5b2YgTW9kZWwpLmd0ZSh2YWwpO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlIFwiTGVzcyBUaGFuXCI6XG4gICAgICAgICAgb3JDb25kaXRpb24gPSBDb25kaXRpb24uYXR0cmlidXRlPE1vZGVsPihpbmRleCBhcyBrZXlvZiBNb2RlbCkubHRlKHZhbCk7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICB9XG4gICAgICBfY29uZGl0aW9uID0gKCFfY29uZGl0aW9uID9cbiAgICAgICAgb3JDb25kaXRpb24gOiBfY29uZGl0aW9uLmFuZChvckNvbmRpdGlvbiBhcyB1bmtub3duIGFzIENvbmRpdGlvbjxNb2RlbD4pKSBhcyBDb25kaXRpb248TW9kZWw+O1xuICAgIH0pO1xuXG4gICAgdGhpcy5zb3J0QnkgPSBzb3J0Py52YWx1ZSBhcyBrZXlvZiBNb2RlbCB8fCB0aGlzLnBrO1xuICAgIHRoaXMuc29ydERpcmVjdGlvbiA9IHNvcnQ/LmRpcmVjdGlvbiB8fCB0aGlzLnNvcnREaXJlY3Rpb247XG4gIH1cbiAgY29uc29sZS5sb2coX2NvbmRpdGlvbik7XG4gIHJldHVybiBfY29uZGl0aW9uIGFzIENvbmRpdGlvbjxNb2RlbD47XG5cbn1cblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gUHJvY2Vzc2VzIHF1ZXJ5IHJlc3VsdHMgaW50byBhIHN0YW5kYXJkaXplZCBmb3JtYXQuXG4gKiBAc3VtbWFyeSBIYW5kbGVzIGRpZmZlcmVudCByZXN1bHQgZm9ybWF0cyBmcm9tIHZhcmlvdXMgZGF0YSBzb3VyY2VzLCBleHRyYWN0aW5nXG4gKiBwYWdpbmF0aW9uIGluZm9ybWF0aW9uIHdoZW4gYXZhaWxhYmxlIGFuZCBhcHBseWluZyBhbnkgY29uZmlndXJlZCBkYXRhIG1hcHBpbmcuXG4gKiBUaGlzIGVuc3VyZXMgY29uc2lzdGVudCBkYXRhIHN0cnVjdHVyZSByZWdhcmRsZXNzIG9mIHRoZSBzb3VyY2UuXG4gKlxuICogQHByb3RlY3RlZFxuICogQHBhcmFtIHtLZXlWYWx1ZVtdIHwgUGFnaW5hdG9yfSByZXN1bHQgLSBUaGUgcmF3IHF1ZXJ5IHJlc3VsdFxuICogQHJldHVybnMge0tleVZhbHVlW119IFRoZSBwcm9jZXNzZWQgYXJyYXkgb2YgaXRlbXNcbiAqXG4gKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICovXG5wcm90ZWN0ZWQgYXN5bmMgcGFyc2VSZXN1bHQocmVzdWx0OiBLZXlWYWx1ZVtdIHwgUGFnaW5hdG9yPE1vZGVsPik6IFByb21pc2U8S2V5VmFsdWVbXT4ge1xuICBpZighQXJyYXkuaXNBcnJheShyZXN1bHQpICYmICgncGFnZScgaW4gcmVzdWx0ICYmICd0b3RhbCcgaW4gcmVzdWx0KSkge1xuICAgIGNvbnN0IHBhZ2luYXRvciA9IHJlc3VsdCBhcyBQYWdpbmF0b3I8TW9kZWw+O1xuICAgIHJlc3VsdCA9IGF3YWl0IHBhZ2luYXRvci5wYWdlKHRoaXMucGFnZSk7XG4gICAgLy8gVE9ETzogQ2hhZ2UgZm9yIHJlc3VsdC50b3RhbDtcbiAgICB0aGlzLmdldE1vcmVEYXRhKHBhZ2luYXRvci50b3RhbCk7XG4gIH0gZWxzZSB7XG4gICAgdGhpcy5nZXRNb3JlRGF0YSgocmVzdWx0IGFzIEtleVZhbHVlW10pPy5sZW5ndGggfHwgMCk7XG4gIH1cbiAgcmV0dXJuIChPYmplY3Qua2V5cyh0aGlzLm1hcHBlciB8fCB7fSkubGVuZ3RoKSA/XG4gICAgdGhpcy5tYXBSZXN1bHRzKHJlc3VsdCkgOiByZXN1bHQ7XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFVwZGF0ZXMgcGFnaW5hdGlvbiBzdGF0ZSBiYXNlZCBvbiBkYXRhIGxlbmd0aC5cbiAqIEBzdW1tYXJ5IENhbGN1bGF0ZXMgd2hldGhlciBtb3JlIGRhdGEgaXMgYXZhaWxhYmxlIGFuZCBob3cgbWFueSBwYWdlcyBleGlzdFxuICogYmFzZWQgb24gdGhlIHRvdGFsIG51bWJlciBvZiBpdGVtcyBhbmQgdGhlIGNvbmZpZ3VyZWQgbGltaXQgcGVyIHBhZ2UuXG4gKiBUaGlzIGluZm9ybWF0aW9uIGlzIHVzZWQgdG8gY29udHJvbCBwYWdpbmF0aW9uIFVJIGFuZCBpbmZpbml0ZSBzY3JvbGxpbmcgYmVoYXZpb3IuXG4gKlxuICogQHBhcmFtIHtudW1iZXJ9IGxlbmd0aCAtIFRoZSB0b3RhbCBudW1iZXIgb2YgaXRlbXMgYXZhaWxhYmxlXG4gKiBAcmV0dXJucyB7dm9pZH1cbiAqXG4gKiBAbWVtYmVyT2YgTGlzdENvbXBvbmVudFxuICovXG5nZXRNb3JlRGF0YShsZW5ndGg6IG51bWJlcik6IHZvaWQge1xuICBpZih0aGlzLnR5cGUgPT09IExpc3RDb21wb25lbnRzVHlwZXMuSU5GSU5JVEUpIHtcbiAgICBpZih0aGlzLnBhZ2luYXRvcilcbiAgICAgIGxlbmd0aCA9IGxlbmd0aCAqIHRoaXMubGltaXQ7XG4gICAgaWYobGVuZ3RoIDw9IHRoaXMubGltaXQpIHtcbiAgICAgIHRoaXMubG9hZE1vcmVEYXRhID0gZmFsc2U7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMucGFnZXMgPSBNYXRoLmZsb29yKGxlbmd0aCAvIHRoaXMubGltaXQpO1xuICAgICAgaWYoKHRoaXMucGFnZXMgKiB0aGlzLmxpbWl0KSA8IGxlbmd0aClcbiAgICAgICAgdGhpcy5wYWdlcyArPSAxO1xuICAgICAgaWYodGhpcy5wYWdlcyA9PT0gMSlcbiAgICAgICAgdGhpcy5sb2FkTW9yZURhdGEgPSBmYWxzZTtcbiAgICB9XG4gIH0gZWxzZSB7XG4gICAgdGhpcy5wYWdlcyA9IGxlbmd0aDtcbiAgICBpZih0aGlzLnBhZ2VzID09PSAxKVxuICAgICAgdGhpcy5sb2FkTW9yZURhdGEgPSBmYWxzZTtcbiAgfVxufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBNYXBzIGEgc2luZ2xlIGl0ZW0gdXNpbmcgdGhlIGNvbmZpZ3VyZWQgbWFwcGVyLlxuICogQHN1bW1hcnkgVHJhbnNmb3JtcyBhIGRhdGEgaXRlbSBhY2NvcmRpbmcgdG8gdGhlIG1hcHBpbmcgY29uZmlndXJhdGlvbixcbiAqIGV4dHJhY3RpbmcgbmVzdGVkIHByb3BlcnRpZXMgYW5kIGZvcm1hdHRpbmcgdmFsdWVzIGFzIG5lZWRlZC4gVGhpcyBhbGxvd3NcbiAqIHRoZSBjb21wb25lbnQgdG8gZGlzcGxheSBkYXRhIGluIGEgZm9ybWF0IGRpZmZlcmVudCBmcm9tIGhvdyBpdCdzIHN0b3JlZC5cbiAqXG4gKiBAcHJvdGVjdGVkXG4gKiBAcGFyYW0ge0tleVZhbHVlfSBpdGVtIC0gVGhlIGl0ZW0gdG8gbWFwXG4gKiBAcGFyYW0ge0tleVZhbHVlfSBtYXBwZXIgLSBUaGUgbWFwcGluZyBjb25maWd1cmF0aW9uXG4gKiBAcGFyYW0ge0tleVZhbHVlfSBbcHJvcHNdIC0gQWRkaXRpb25hbCBwcm9wZXJ0aWVzIHRvIGluY2x1ZGVcbiAqIEByZXR1cm5zIHtLZXlWYWx1ZX0gVGhlIG1hcHBlZCBpdGVtXG4gKlxuICogQG1lbWJlck9mIExpc3RDb21wb25lbnRcbiAqL1xucHJvdGVjdGVkIGl0ZW1NYXBwZXIoaXRlbTogS2V5VmFsdWUsIG1hcHBlcjogS2V5VmFsdWUsIHByb3BzPzogS2V5VmFsdWUpOiBLZXlWYWx1ZSB7XG4gIHJldHVybiBPYmplY3QuZW50cmllcyhtYXBwZXIpLnJlZHVjZSgoYWNjdW06IEtleVZhbHVlLCBba2V5LCB2YWx1ZV0pID0+IHtcbiAgICBjb25zdCBhcnJheVZhbHVlID0gdmFsdWUuc3BsaXQoJy4nKTtcbiAgICBpZiAoIXZhbHVlKSB7XG4gICAgICBhY2N1bVtrZXldID0gdmFsdWU7XG4gICAgfSBlbHNlIHtcbiAgICAgIGlmIChhcnJheVZhbHVlLmxlbmd0aCA9PT0gMSkge1xuICAgICAgICB2YWx1ZSA9IGl0ZW0/Llt2YWx1ZV0gfHwgdmFsdWU7XG4gICAgICAgIGlmKGlzVmFsaWREYXRlKHZhbHVlKSlcbiAgICAgICAgICB2YWx1ZSA9IGAke2Zvcm1hdERhdGUodmFsdWUpfWA7XG4gICAgICAgIGFjY3VtW2tleV0gPSB2YWx1ZTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGxldCB2YWw7XG5cbiAgICAgICAgZm9yIChjb25zdCBfdmFsdWUgb2YgYXJyYXlWYWx1ZSlcbiAgICAgICAgICB2YWwgPSAhdmFsXG4gICAgICAgICAgICA/IGl0ZW1bX3ZhbHVlXVxuICAgICAgICAgICAgOiAodHlwZW9mIHZhbCA9PT0gJ3N0cmluZycgPyBKU09OLnBhcnNlKHZhbCkgOiB2YWwpW192YWx1ZV07XG5cblxuICAgICAgICBpZiAoaXNWYWxpZERhdGUobmV3IERhdGUodmFsKSkpXG4gICAgICAgICAgdmFsID0gYCR7Zm9ybWF0RGF0ZSh2YWwpfWA7XG5cbiAgICAgICAgYWNjdW1ba2V5XSA9IHZhbCA9PT0gbnVsbCB8fCB2YWwgPT09IHVuZGVmaW5lZCA/IHZhbHVlIDogdmFsO1xuICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gT2JqZWN0LmFzc2lnbih7fSwgcHJvcHMgfHwge30sIGFjY3VtKTtcbiAgfSwge30pO1xufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBNYXBzIGFsbCByZXN1bHQgaXRlbXMgdXNpbmcgdGhlIGNvbmZpZ3VyZWQgbWFwcGVyLlxuICogQHN1bW1hcnkgQXBwbGllcyB0aGUgaXRlbU1hcHBlciB0byBlYWNoIGl0ZW0gaW4gdGhlIHJlc3VsdCBzZXQsIGFkZGluZ1xuICogY29tbW9uIHByb3BlcnRpZXMgbGlrZSBvcGVyYXRpb25zIGFuZCByb3V0ZSBpbmZvcm1hdGlvbi4gVGhpcyB0cmFuc2Zvcm1zXG4gKiB0aGUgcmF3IGRhdGEgaW50byB0aGUgZm9ybWF0IGV4cGVjdGVkIGJ5IHRoZSBsaXN0IGl0ZW0gY29tcG9uZW50cy5cbiAqXG4gKiBAcGFyYW0ge0tleVZhbHVlW119IGRhdGEgLSBUaGUgYXJyYXkgb2YgaXRlbXMgdG8gbWFwXG4gKiBAcmV0dXJucyB7S2V5VmFsdWVbXX0gVGhlIGFycmF5IG9mIG1hcHBlZCBpdGVtc1xuICpcbiAqIEBtZW1iZXJPZiBMaXN0Q29tcG9uZW50XG4gKi9cbiAgbWFwUmVzdWx0cyhkYXRhOiBLZXlWYWx1ZVtdKTogS2V5VmFsdWVbXSB7XG4gICAgaWYoIWRhdGEgfHwgIWRhdGEubGVuZ3RoKVxuICAgICAgcmV0dXJuIFtdO1xuICAgIC8vIHBhc3NpbmcgdWlkIGFzIHByb3AgdG8gbWFwcGVyXG4gICAgdGhpcy5tYXBwZXIgPSB7Li4uIHRoaXMubWFwcGVyLCAuLi4ge3VpZDogdGhpcy5wa319O1xuICAgIGNvbnN0IHByb3BzID0gT2JqZWN0LmFzc2lnbih7XG4gICAgICBvcGVyYXRpb25zOiB0aGlzLm9wZXJhdGlvbnMsXG4gICAgICByb3V0ZTogdGhpcy5yb3V0ZSxcbiAgICAgIC4uLiAgT2JqZWN0LmtleXModGhpcy5pdGVtKS5yZWR1Y2UoKGFjYzogS2V5VmFsdWUsIGtleTogc3RyaW5nKSA9PiB7XG4gICAgICAgIGFjY1trZXldID0gdGhpcy5pdGVtW2tleV07XG4gICAgICAgIHJldHVybiBhY2M7XG4gICAgICB9LCB7fSksXG4gICAgICAvLyAuLi4gKCF0aGlzLml0ZW0ucmVuZGVyID8ge30gOiAgT2JqZWN0LmtleXModGhpcy5pdGVtKS5yZWR1Y2UoKGFjYzogS2V5VmFsdWUsIGtleTogc3RyaW5nKSA9PiB7XG4gICAgICAvLyAgIGFjY1trZXldID0gdGhpcy5pdGVtW2tleSBhcyBrZXlvZiBJTGlzdEl0ZW1Qcm9wXTtcbiAgICAgIC8vICAgcmV0dXJuIGFjYztcbiAgICAgIC8vIH0sIHt9KSlcbiAgICB9KTtcbiAgICByZXR1cm4gZGF0YS5yZWR1Y2UoKGFjY3VtOiBLZXlWYWx1ZVtdLCBjdXJyKSA9PiB7XG4gICAgICAgIGFjY3VtLnB1c2goey4uLiB0aGlzLml0ZW1NYXBwZXIoY3VyciwgdGhpcy5tYXBwZXIgYXMgS2V5VmFsdWUsIHByb3BzKSwgLi4uIHtwazogdGhpcy5wa319KTtcbiAgICAgICAgcmV0dXJuIGFjY3VtO1xuICAgIH0sIFtdKTtcbiAgfVxufVxuXG4iLCJcbkBpZihzaG93UmVmcmVzaGVyKSB7XG4gIDxpb24tcmVmcmVzaGVyIHNsb3Q9XCJmaXhlZFwiIFtwdWxsRmFjdG9yXT1cIjFcIiBbcHVsbE1pbl09XCIxMDBcIiBbcHVsbE1heF09XCIyMDBcIiAoaW9uUmVmcmVzaCk9XCJoYW5kbGVSZWZyZXNoKCRldmVudClcIj5cbiAgICA8aW9uLXJlZnJlc2hlci1jb250ZW50IC8+XG4gIDwvaW9uLXJlZnJlc2hlcj5cbn1cblxuQGlmKHNob3dTZWFyY2hiYXIpIHtcbiAgQGlmKG1vZGVsICYmIGVuYWJsZUZpbHRlcikge1xuICAgIDxuZ3gtZGVjYWYtZmlsdGVyXG4gICAgICBbbW9kZWxdPVwibW9kZWxcIlxuICAgICAgW3NvcnREaXJlY3Rpb25dPVwic29ydERpcmVjdGlvblwiXG4gICAgICBbZGlzYWJsZVNvcnRdPVwiZGlzYWJsZVNvcnRcIlxuICAgICAgKGZpbHRlckV2ZW50KT1cImhhbmRsZUZpbHRlcigkZXZlbnQpXCJcbiAgICAgIChzZWFyY2hFdmVudCk9XCJoYW5kbGVTZWFyY2goJGV2ZW50KVwiXG4gICAgLz5cbiAgfSBAZWxzZSB7XG4gICAgPG5neC1kZWNhZi1zZWFyY2hiYXIgW2VtaXRFdmVudFRvV2luZG93XT1cImZhbHNlXCIgW2RlYm91bmNlXT1cIjUwMFwiIChzZWFyY2hFdmVudCk9XCJoYW5kbGVTZWFyY2goJGV2ZW50KVwiIC8+XG4gIH1cbn1cblxuQGlmKGRhdGE/Lmxlbmd0aCkge1xuICA8aW9uLWxpc3QgW2luc2V0XT1cImluc2V0XCIgW2xpbmVzXT1cImxpbmVzXCIgI2NvbXBvbmVudD5cbiAgICBAaWYoaXRlbT8udGFnKSB7XG4gICAgICBAZm9yKGNoaWxkIG9mIGl0ZW1zOyB0cmFjayB0cmFja0l0ZW1GbigkaW5kZXgsIGNoaWxkKSkge1xuICAgICAgICA8bmd4LWRlY2FmLWNvbXBvbmVudC1yZW5kZXJlclxuICAgICAgICAgIFt0YWddPVwiaXRlbS50YWdcIlxuICAgICAgICAgIChsaXN0ZW5FdmVudCk9XCJoYW5kbGVFdmVudCgkZXZlbnQpXCJcbiAgICAgICAgICBbZ2xvYmFsc109J3tcbiAgICAgICAgICAgIGl0ZW06IGNoaWxkLFxuICAgICAgICAgICAgbWFwcGVyOiBtYXBwZXIsXG4gICAgICAgICAgICByb3V0ZTogcm91dGVcbiAgICAgICAgICB9Jz5cbiAgICAgICAgICA8L25neC1kZWNhZi1jb21wb25lbnQtcmVuZGVyZXI+XG4gICAgICAgIH1cbiAgICB9IEBlbHNlIHtcbiAgICAgIDxuZy1jb250ZW50PjwvbmctY29udGVudD5cbiAgICB9XG4gIDwvaW9uLWxpc3Q+XG5cbiAgQGlmKGxvYWRNb3JlRGF0YSkge1xuICAgIEBpZihwYWdlcyA+IDAgJiYgdHlwZSA9PT0gJ3BhZ2luYXRlZCcgJiYgIXNlYXJjaFZhbHVlPy5sZW5ndGgpIHtcbiAgICAgIDxuZ3gtZGVjYWYtcGFnaW5hdGlvblxuICAgICAgICBbdG90YWxQYWdlc109XCJwYWdlc1wiXG4gICAgICAgIFtjdXJyZW50XT1cInBhZ2VcIlxuICAgICAgICAoY2xpY2tFdmVudCk9XCJoYW5kbGVQYWdpbmF0ZSgkZXZlbnQpXCJcbiAgICAgIC8+XG5cbiAgICB9IEBlbHNlIHtcbiAgICAgIDxpb24taW5maW5pdGUtc2Nyb2xsXG4gICAgICAgIFtjbGFzc109XCJzZWFyY2hWYWx1ZT8ubGVuZ3RoID8gJ2RjZi1oaWRkZW4nIDogJydcIlxuICAgICAgICBbcG9zaXRpb25dPVwic2Nyb2xsUG9zaXRpb25cIlxuICAgICAgICBbdGhyZXNob2xkXT1cInNjcm9sbFRocmVzaG9sZFwiXG4gICAgICAgIChpb25JbmZpbml0ZSk9XCJoYW5kbGVSZWZyZXNoKCRldmVudClcIj5cbiAgICAgICAgPGlvbi1pbmZpbml0ZS1zY3JvbGwtY29udGVudCBbbG9hZGluZ1NwaW5uZXJdPVwibG9hZGluZ1NwaW5uZXJcIiBbbG9hZGluZ1RleHRdPVwibG9hZGluZ1RleHRcIiAvPlxuICAgICAgPC9pb24taW5maW5pdGUtc2Nyb2xsPlxuICAgIH1cbiAgfVxufSBAZWxzZSB7XG4gIEBpZihyZWZyZXNoaW5nKSB7XG4gICAgPGlvbi1pdGVtICpuZ0Zvcj1cImxldCBza2wgb2Ygc2tlbGV0b25EYXRhXCI+XG4gICAgICA8aW9uLXRodW1ibmFpbCBzbG90PVwic3RhcnRcIj5cbiAgICAgICAgPGlvbi1za2VsZXRvbi10ZXh0IFthbmltYXRlZF09XCJ0cnVlXCI+PC9pb24tc2tlbGV0b24tdGV4dD5cbiAgICAgIDwvaW9uLXRodW1ibmFpbD5cbiAgICAgIDxpb24tbGFiZWw+XG4gICAgICAgIDxpb24tc2tlbGV0b24tdGV4dCBbYW5pbWF0ZWRdPVwidHJ1ZVwiPjwvaW9uLXNrZWxldG9uLXRleHQ+XG4gICAgICAgIDxpb24tdGV4dCBjbGFzcz1cImRhdGVcIiBzdHlsZT1cIndpZHRoOiAyMCU7XCI+PGlvbi1za2VsZXRvbi10ZXh0IFthbmltYXRlZF09XCJ0cnVlXCI+PC9pb24tc2tlbGV0b24tdGV4dD48L2lvbi10ZXh0PlxuICAgICAgPC9pb24tbGFiZWw+XG4gICAgPC9pb24taXRlbT5cbiAgfSBAZWxzZSB7XG4gICAgQGlmKCFzZWFyY2hWYWx1ZT8ubGVuZ3RoKSB7XG4gICAgICA8bmd4LWRlY2FmLWVtcHR5LXN0YXRlXG4gICAgICAgIFt0aXRsZV09XCIobG9jYWxlICsgJy4nKyBlbXB0eS50aXRsZSkgfCB0cmFuc2xhdGVcIlxuICAgICAgICBbc3VidGl0bGVdPVwiKGxvY2FsZSArICcuJysgZW1wdHkuc3VidGl0bGUpIHwgdHJhbnNsYXRlXCJcbiAgICAgICAgW2J1dHRvblRleHRdPVwiZW1wdHkuc2hvd0J1dHRvbiA/IChsb2NhbGUgKyAnLicrIGVtcHR5LmJ1dHRvbiB8IHRyYW5zbGF0ZSkgOiAnJ1wiXG4gICAgICAgIFtidXR0b25MaW5rXT1cImVtcHR5LnNob3dCdXR0b24gPyBlbXB0eS5yb3V0ZSA6ICcnXCJcbiAgICAgIC8+XG4gICAgfSBAZWxzZSB7XG4gICAgICA8bmd4LWRlY2FmLWVtcHR5LXN0YXRlXG4gICAgICAgIGljb249XCJzZWFyY2gtb3V0bGluZVwiXG4gICAgICAgIG5nQ2xhc3M9XCJlbXB0eS1zZWFyY2hcIlxuICAgICAgICBbdHJhbnNsYXRhYmxlXT1cInRydWVcIlxuICAgICAgICB0aXRsZT1cInNlYXJjaC50aXRsZVwiXG4gICAgICAgIHN1YnRpdGxlPVwic2VhcmNoLnN1YnRpdGxlXCJcbiAgICAgICAgW3NlYXJjaFZhbHVlXT1cInNlYXJjaFZhbHVlXCJcbiAgICAgIC8+XG4gICAgfVxuICB9XG59XG5cbiJdfQ==