@ecodev/natural 62.1.2 → 63.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (183) hide show
  1. package/fesm2022/ecodev-natural-vanilla.mjs +1193 -0
  2. package/fesm2022/ecodev-natural-vanilla.mjs.map +1 -0
  3. package/fesm2022/ecodev-natural.mjs +436 -425
  4. package/fesm2022/ecodev-natural.mjs.map +1 -1
  5. package/lib/classes/network-activity.service.d.ts +54 -0
  6. package/lib/classes/validators.d.ts +1 -1
  7. package/lib/modules/columns-picker/columns-picker.component.d.ts +2 -2
  8. package/lib/modules/common/services/seo.provider.d.ts +2 -2
  9. package/lib/modules/dropdown-components/type-select/type-select.component.d.ts +1 -1
  10. package/lib/modules/file/abstract-file.d.ts +6 -3
  11. package/lib/modules/file/component/file.component.d.ts +2 -2
  12. package/lib/modules/file/file-drop.directive.d.ts +2 -3
  13. package/lib/modules/fixed-button-detail/fixed-button-detail.component.d.ts +2 -3
  14. package/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.d.ts +3 -3
  15. package/lib/modules/icon/icon.module.d.ts +2 -2
  16. package/lib/modules/panels/panels.service.d.ts +1 -2
  17. package/lib/modules/relations/relations.component.d.ts +3 -3
  18. package/lib/modules/search/dropdown-container/dropdown-container.component.d.ts +2 -3
  19. package/lib/modules/search/group/group.component.d.ts +2 -3
  20. package/lib/modules/search/input/input.component.d.ts +5 -5
  21. package/lib/modules/search/search/search.component.d.ts +2 -2
  22. package/lib/modules/select/abstract-select.component.d.ts +3 -3
  23. package/lib/modules/select/select/select.component.d.ts +1 -1
  24. package/lib/modules/sidenav/sidenav-container/sidenav-container.component.d.ts +1 -1
  25. package/lib/modules/table-button/table-button.component.d.ts +4 -2
  26. package/package.json +16 -14
  27. package/public-api.d.ts +1 -0
  28. package/src/lib/_natural.theme.scss +1 -2
  29. package/vanilla/index.d.ts +5 -0
  30. package/vanilla/package.json +3 -0
  31. package/vanilla/public-api.d.ts +11 -0
  32. package/vanilla/src/lib/classes/crypto.d.ts +8 -0
  33. package/vanilla/src/lib/classes/data-source.d.ts +32 -0
  34. package/vanilla/src/lib/classes/query-variable-manager-utils.d.ts +2 -0
  35. package/vanilla/src/lib/classes/query-variable-manager.d.ts +91 -0
  36. package/vanilla/src/lib/classes/signing.d.ts +7 -0
  37. package/vanilla/src/lib/classes/utility.d.ts +77 -0
  38. package/vanilla/src/lib/modules/search/classes/graphql-doctrine.types.d.ts +83 -0
  39. package/vanilla/src/lib/modules/search/classes/utils.d.ts +17 -0
  40. package/vanilla/src/lib/modules/search/types/dropdown-component.d.ts +20 -0
  41. package/vanilla/src/lib/modules/search/types/facet.d.ts +75 -0
  42. package/vanilla/src/lib/modules/search/types/values.d.ts +32 -0
  43. package/vanilla/src/lib/services/abstract-model.service.d.ts +244 -0
  44. package/vanilla/src/lib/services/debounce.service.d.ts +52 -0
  45. package/vanilla/src/lib/types/types.d.ts +100 -0
  46. package/esm2022/ecodev-natural.mjs +0 -5
  47. package/esm2022/lib/classes/abstract-detail.mjs +0 -229
  48. package/esm2022/lib/classes/abstract-editable-list.mjs +0 -99
  49. package/esm2022/lib/classes/abstract-list.mjs +0 -461
  50. package/esm2022/lib/classes/abstract-navigable-list.mjs +0 -133
  51. package/esm2022/lib/classes/apollo-utils.mjs +0 -59
  52. package/esm2022/lib/classes/crypto.mjs +0 -23
  53. package/esm2022/lib/classes/cumulative-changes.mjs +0 -50
  54. package/esm2022/lib/classes/data-source.mjs +0 -71
  55. package/esm2022/lib/classes/providers.mjs +0 -13
  56. package/esm2022/lib/classes/query-variable-manager-utils.mjs +0 -14
  57. package/esm2022/lib/classes/query-variable-manager.mjs +0 -172
  58. package/esm2022/lib/classes/rxjs.mjs +0 -54
  59. package/esm2022/lib/classes/signing.mjs +0 -38
  60. package/esm2022/lib/classes/tld.mjs +0 -1476
  61. package/esm2022/lib/classes/utility.mjs +0 -234
  62. package/esm2022/lib/classes/validators.mjs +0 -179
  63. package/esm2022/lib/directives/http-prefix.directive.mjs +0 -47
  64. package/esm2022/lib/modules/alert/alert.service.mjs +0 -53
  65. package/esm2022/lib/modules/alert/confirm.component.mjs +0 -16
  66. package/esm2022/lib/modules/alert/public-api.mjs +0 -6
  67. package/esm2022/lib/modules/avatar/component/avatar.component.mjs +0 -203
  68. package/esm2022/lib/modules/avatar/public-api.mjs +0 -6
  69. package/esm2022/lib/modules/avatar/service/avatar.service.mjs +0 -63
  70. package/esm2022/lib/modules/avatar/sources/gravatar.mjs +0 -29
  71. package/esm2022/lib/modules/avatar/sources/image.mjs +0 -13
  72. package/esm2022/lib/modules/avatar/sources/initials.mjs +0 -39
  73. package/esm2022/lib/modules/avatar/sources/source.mjs +0 -16
  74. package/esm2022/lib/modules/columns-picker/columns-picker.component.mjs +0 -145
  75. package/esm2022/lib/modules/columns-picker/public-api.mjs +0 -5
  76. package/esm2022/lib/modules/columns-picker/types.mjs +0 -2
  77. package/esm2022/lib/modules/common/directives/background-density.directive.mjs +0 -63
  78. package/esm2022/lib/modules/common/directives/linkable-tab.directive.mjs +0 -93
  79. package/esm2022/lib/modules/common/directives/src-density.directive.mjs +0 -72
  80. package/esm2022/lib/modules/common/pipes/capitalize.pipe.mjs +0 -24
  81. package/esm2022/lib/modules/common/pipes/ellipsis.pipe.mjs +0 -17
  82. package/esm2022/lib/modules/common/pipes/enum.pipe.mjs +0 -24
  83. package/esm2022/lib/modules/common/pipes/time-ago.pipe.mjs +0 -140
  84. package/esm2022/lib/modules/common/public-api.mjs +0 -14
  85. package/esm2022/lib/modules/common/services/memory-storage.mjs +0 -110
  86. package/esm2022/lib/modules/common/services/seo.provider.mjs +0 -23
  87. package/esm2022/lib/modules/common/services/seo.service.mjs +0 -235
  88. package/esm2022/lib/modules/detail-header/detail-header.component.mjs +0 -84
  89. package/esm2022/lib/modules/detail-header/public-api.mjs +0 -5
  90. package/esm2022/lib/modules/dialog-trigger/dialog-trigger.component.mjs +0 -72
  91. package/esm2022/lib/modules/dialog-trigger/public-api.mjs +0 -5
  92. package/esm2022/lib/modules/dropdown-components/abstract-association-select-component.directive.mjs +0 -100
  93. package/esm2022/lib/modules/dropdown-components/public-api.mjs +0 -14
  94. package/esm2022/lib/modules/dropdown-components/type-boolean/type-boolean.component.mjs +0 -39
  95. package/esm2022/lib/modules/dropdown-components/type-date/type-date.component.mjs +0 -173
  96. package/esm2022/lib/modules/dropdown-components/type-date-range/type-date-range.component.mjs +0 -134
  97. package/esm2022/lib/modules/dropdown-components/type-hierarchic-selector/type-hierarchic-selector.component.mjs +0 -80
  98. package/esm2022/lib/modules/dropdown-components/type-natural-select/type-natural-select.component.mjs +0 -48
  99. package/esm2022/lib/modules/dropdown-components/type-number/type-number.component.mjs +0 -110
  100. package/esm2022/lib/modules/dropdown-components/type-options/type-options.component.mjs +0 -64
  101. package/esm2022/lib/modules/dropdown-components/type-select/type-select.component.mjs +0 -175
  102. package/esm2022/lib/modules/dropdown-components/type-text/type-text.component.mjs +0 -62
  103. package/esm2022/lib/modules/dropdown-components/types.mjs +0 -41
  104. package/esm2022/lib/modules/dropdown-components/utils.mjs +0 -35
  105. package/esm2022/lib/modules/file/abstract-file.mjs +0 -230
  106. package/esm2022/lib/modules/file/component/file.component.mjs +0 -172
  107. package/esm2022/lib/modules/file/file-drop.directive.mjs +0 -111
  108. package/esm2022/lib/modules/file/file-select.directive.mjs +0 -26
  109. package/esm2022/lib/modules/file/file.service.mjs +0 -43
  110. package/esm2022/lib/modules/file/public-api.mjs +0 -9
  111. package/esm2022/lib/modules/file/types.mjs +0 -2
  112. package/esm2022/lib/modules/file/utils.mjs +0 -129
  113. package/esm2022/lib/modules/fixed-button/fixed-button.component.mjs +0 -30
  114. package/esm2022/lib/modules/fixed-button/public-api.mjs +0 -5
  115. package/esm2022/lib/modules/fixed-button-detail/fixed-button-detail.component.mjs +0 -56
  116. package/esm2022/lib/modules/fixed-button-detail/public-api.mjs +0 -5
  117. package/esm2022/lib/modules/hierarchic-selector/classes/flat-node.mjs +0 -18
  118. package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-configuration.mjs +0 -2
  119. package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-filters-configuration.mjs +0 -2
  120. package/esm2022/lib/modules/hierarchic-selector/classes/model-node.mjs +0 -14
  121. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.mjs +0 -398
  122. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs +0 -243
  123. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component.mjs +0 -38
  124. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service.mjs +0 -22
  125. package/esm2022/lib/modules/hierarchic-selector/public-api.mjs +0 -10
  126. package/esm2022/lib/modules/icon/icon.directive.mjs +0 -96
  127. package/esm2022/lib/modules/icon/icon.module.mjs +0 -33
  128. package/esm2022/lib/modules/icon/public-api.mjs +0 -6
  129. package/esm2022/lib/modules/logger/error-handler.mjs +0 -87
  130. package/esm2022/lib/modules/logger/error.module.mjs +0 -22
  131. package/esm2022/lib/modules/logger/public-api.mjs +0 -6
  132. package/esm2022/lib/modules/matomo/matomo.service.mjs +0 -96
  133. package/esm2022/lib/modules/matomo/public-api.mjs +0 -5
  134. package/esm2022/lib/modules/panels/abstract-panel.mjs +0 -76
  135. package/esm2022/lib/modules/panels/fallback-if-no-opened-panels.urlmatcher.mjs +0 -12
  136. package/esm2022/lib/modules/panels/panels.component.mjs +0 -27
  137. package/esm2022/lib/modules/panels/panels.module.mjs +0 -10
  138. package/esm2022/lib/modules/panels/panels.service.mjs +0 -329
  139. package/esm2022/lib/modules/panels/panels.urlmatcher.mjs +0 -75
  140. package/esm2022/lib/modules/panels/public-api.mjs +0 -11
  141. package/esm2022/lib/modules/panels/types.mjs +0 -3
  142. package/esm2022/lib/modules/relations/public-api.mjs +0 -5
  143. package/esm2022/lib/modules/relations/relations.component.mjs +0 -254
  144. package/esm2022/lib/modules/search/classes/graphql-doctrine.mjs +0 -111
  145. package/esm2022/lib/modules/search/classes/graphql-doctrine.types.mjs +0 -14
  146. package/esm2022/lib/modules/search/classes/transformers.mjs +0 -142
  147. package/esm2022/lib/modules/search/classes/url.mjs +0 -53
  148. package/esm2022/lib/modules/search/classes/utils.mjs +0 -25
  149. package/esm2022/lib/modules/search/dropdown-container/dropdown-container-animations.mjs +0 -44
  150. package/esm2022/lib/modules/search/dropdown-container/dropdown-container.component.mjs +0 -87
  151. package/esm2022/lib/modules/search/dropdown-container/dropdown-ref.mjs +0 -24
  152. package/esm2022/lib/modules/search/dropdown-container/dropdown.service.mjs +0 -90
  153. package/esm2022/lib/modules/search/facet-selector/facet-selector.component.mjs +0 -45
  154. package/esm2022/lib/modules/search/group/group.component.mjs +0 -53
  155. package/esm2022/lib/modules/search/input/input.component.mjs +0 -365
  156. package/esm2022/lib/modules/search/public-api.mjs +0 -7
  157. package/esm2022/lib/modules/search/search/search.component.mjs +0 -102
  158. package/esm2022/lib/modules/search/types/dropdown-component.mjs +0 -2
  159. package/esm2022/lib/modules/search/types/facet.mjs +0 -2
  160. package/esm2022/lib/modules/search/types/values.mjs +0 -2
  161. package/esm2022/lib/modules/select/abstract-select.component.mjs +0 -232
  162. package/esm2022/lib/modules/select/public-api.mjs +0 -7
  163. package/esm2022/lib/modules/select/select/select.component.mjs +0 -310
  164. package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +0 -57
  165. package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +0 -155
  166. package/esm2022/lib/modules/sidenav/public-api.mjs +0 -9
  167. package/esm2022/lib/modules/sidenav/sidenav/sidenav.component.mjs +0 -15
  168. package/esm2022/lib/modules/sidenav/sidenav-container/sidenav-container.component.mjs +0 -90
  169. package/esm2022/lib/modules/sidenav/sidenav-content/sidenav-content.component.mjs +0 -11
  170. package/esm2022/lib/modules/sidenav/sidenav-stack.service.mjs +0 -50
  171. package/esm2022/lib/modules/sidenav/sidenav.service.mjs +0 -196
  172. package/esm2022/lib/modules/stamp/public-api.mjs +0 -5
  173. package/esm2022/lib/modules/stamp/stamp.component.mjs +0 -23
  174. package/esm2022/lib/modules/table-button/public-api.mjs +0 -5
  175. package/esm2022/lib/modules/table-button/table-button.component.mjs +0 -78
  176. package/esm2022/lib/services/abstract-model.service.mjs +0 -526
  177. package/esm2022/lib/services/debounce.service.mjs +0 -149
  178. package/esm2022/lib/services/enum.service.mjs +0 -64
  179. package/esm2022/lib/services/link-mutation.service.mjs +0 -154
  180. package/esm2022/lib/services/persistence.service.mjs +0 -115
  181. package/esm2022/lib/services/swiss-parsing-date-adapter.service.mjs +0 -63
  182. package/esm2022/lib/types/types.mjs +0 -2
  183. package/esm2022/public-api.mjs +0 -46
@@ -1,526 +0,0 @@
1
- import { Apollo, gql } from 'apollo-angular';
2
- import { NetworkStatus } from '@apollo/client/core';
3
- import { UntypedFormControl, UntypedFormGroup } from '@angular/forms';
4
- import { defaults, merge, pick } from 'lodash-es';
5
- import { catchError, combineLatest, EMPTY, first, from, Observable, of } from 'rxjs';
6
- import { debounceTime, filter, map, shareReplay, startWith, switchMap, takeWhile, tap } from 'rxjs/operators';
7
- import { NaturalQueryVariablesManager } from '../classes/query-variable-manager';
8
- import { makePlural, relationsToIds, upperCaseFirstLetter } from '../classes/utility';
9
- import { NaturalDebounceService } from './debounce.service';
10
- import { deepClone } from '../modules/search/classes/utils';
11
- import { inject } from '@angular/core';
12
- export class NaturalAbstractModelService {
13
- name;
14
- oneQuery;
15
- allQuery;
16
- createMutation;
17
- updateMutation;
18
- deleteMutation;
19
- createName;
20
- updateName;
21
- deleteName;
22
- /**
23
- * Store the creation mutations that are pending
24
- */
25
- creatingCache = new Map();
26
- apollo = inject(Apollo);
27
- naturalDebounceService = inject(NaturalDebounceService);
28
- plural;
29
- /**
30
- *
31
- * @param name service and single object query name (eg. userForFront or user).
32
- * @param oneQuery GraphQL query to fetch a single object from ID (eg. userForCrudQuery).
33
- * @param allQuery GraphQL query to fetch a filtered list of objects (eg. usersForCrudQuery).
34
- * @param createMutation GraphQL mutation to create an object.
35
- * @param updateMutation GraphQL mutation to update an object.
36
- * @param deleteMutation GraphQL mutation to delete a list of objects.
37
- * @param plural list query name (eg. usersForFront or users).
38
- * @param createName create object mutation name (eg. createUser).
39
- * @param updateName update object mutation name (eg. updateUser).
40
- * @param deleteName delete object mutation name (eg. deleteUsers).
41
- */
42
- constructor(name, oneQuery, allQuery, createMutation, updateMutation, deleteMutation, plural = null, createName = null, updateName = null, deleteName = null) {
43
- this.name = name;
44
- this.oneQuery = oneQuery;
45
- this.allQuery = allQuery;
46
- this.createMutation = createMutation;
47
- this.updateMutation = updateMutation;
48
- this.deleteMutation = deleteMutation;
49
- this.createName = createName;
50
- this.updateName = updateName;
51
- this.deleteName = deleteName;
52
- this.plural = plural ?? makePlural(this.name);
53
- }
54
- /**
55
- * List of individual fields validators
56
- */
57
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
58
- getFormValidators(model) {
59
- return {};
60
- }
61
- /**
62
- * List of individual async fields validators
63
- */
64
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
65
- getFormAsyncValidators(model) {
66
- return {};
67
- }
68
- /**
69
- * List of grouped fields validators (like password + confirm password)
70
- */
71
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
72
- getFormGroupValidators(model) {
73
- return [];
74
- }
75
- /**
76
- * List of async group fields validators (like unique constraint on multiple columns)
77
- */
78
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
79
- getFormGroupAsyncValidators(model) {
80
- return [];
81
- }
82
- getFormConfig(model) {
83
- const values = { ...this.getDefaultForServer(), ...this.getFormExtraFieldDefaultValues() };
84
- const validators = this.getFormValidators(model);
85
- const asyncValidators = this.getFormAsyncValidators(model);
86
- const controls = {};
87
- const disabled = model.permissions ? !model.permissions.update : false;
88
- if (model.id) {
89
- controls.id = new UntypedFormControl({ value: model.id, disabled: true });
90
- }
91
- // Configure form for each field of model
92
- for (const key of Object.keys(values)) {
93
- const value = model[key] !== undefined ? model[key] : values[key];
94
- const formState = {
95
- value: value,
96
- disabled: disabled,
97
- };
98
- const validator = typeof validators[key] !== 'undefined' ? validators[key] : null;
99
- const asyncValidator = typeof asyncValidators[key] !== 'undefined' ? asyncValidators[key] : null;
100
- controls[key] = new UntypedFormControl(formState, validator, asyncValidator);
101
- }
102
- // Configure form for extra validators that are not on a specific field
103
- for (const key of Object.keys(validators)) {
104
- if (!controls[key]) {
105
- const formState = {
106
- value: model[key] ? model[key] : null,
107
- disabled: disabled,
108
- };
109
- controls[key] = new UntypedFormControl(formState, validators[key]);
110
- }
111
- }
112
- for (const key of Object.keys(asyncValidators)) {
113
- if (controls[key] && asyncValidators[key]) {
114
- controls[key].setAsyncValidators(asyncValidators[key]);
115
- }
116
- else {
117
- const formState = {
118
- value: model[key] ? model[key] : null,
119
- disabled: disabled,
120
- };
121
- controls[key] = new UntypedFormControl(formState, null, asyncValidators[key]);
122
- }
123
- }
124
- return controls;
125
- }
126
- /**
127
- * Create the final FormGroup for the object, including all validators
128
- *
129
- * This method should **not** be overridden, but instead `getFormConfig`,
130
- * `getFormGroupValidators`, `getFormGroupAsyncValidators` might be.
131
- */
132
- getFormGroup(model) {
133
- const formConfig = this.getFormConfig(deepClone(model));
134
- return new UntypedFormGroup(formConfig, {
135
- validators: this.getFormGroupValidators(model),
136
- asyncValidators: this.getFormGroupAsyncValidators(model),
137
- });
138
- }
139
- /**
140
- * Get a single object
141
- *
142
- * If available it will emit object from cache immediately, then it
143
- * will **always** fetch from network and then the observable will be completed.
144
- *
145
- * You must subscribe to start getting results (and fetch from network).
146
- */
147
- getOne(id) {
148
- return this.prepareOneQuery(id, 'cache-and-network').pipe(takeWhile(result => result.networkStatus !== NetworkStatus.ready, true), map(result => result.data[this.name]));
149
- }
150
- /**
151
- * Watch a single object
152
- *
153
- * If available it will emit object from cache immediately, then it
154
- * will **always** fetch from network, and then keep watching the cache forever.
155
- *
156
- * You must subscribe to start getting results (and fetch from network).
157
- *
158
- * You **MUST** unsubscribe.
159
- */
160
- watchOne(id, fetchPolicy = 'cache-and-network') {
161
- return this.prepareOneQuery(id, fetchPolicy).pipe(map(result => result.data[this.name]));
162
- }
163
- prepareOneQuery(id, fetchPolicy) {
164
- this.throwIfObservable(id);
165
- this.throwIfNotQuery(this.oneQuery);
166
- return this.getVariablesForOne(id).pipe(switchMap(variables => {
167
- this.throwIfNotQuery(this.oneQuery);
168
- return this.apollo.watchQuery({
169
- query: this.oneQuery,
170
- variables: variables,
171
- fetchPolicy: fetchPolicy,
172
- nextFetchPolicy: 'cache-only',
173
- }).valueChanges;
174
- }), filter(result => !!result.data));
175
- }
176
- /**
177
- * Get a collection of objects
178
- *
179
- * It will **always** fetch from network and then the observable will be completed.
180
- * No cache is ever used, so it's slow but correct.
181
- */
182
- getAll(queryVariablesManager) {
183
- this.throwIfNotQuery(this.allQuery);
184
- return this.getPartialVariablesForAll().pipe(first(), switchMap(partialVariables => {
185
- this.throwIfNotQuery(this.allQuery);
186
- // Copy manager to prevent to apply internal variables to external QueryVariablesManager
187
- const manager = new NaturalQueryVariablesManager(queryVariablesManager);
188
- manager.merge('partial-variables', partialVariables);
189
- return this.apollo.query({
190
- query: this.allQuery,
191
- variables: manager.variables.value,
192
- fetchPolicy: 'network-only',
193
- });
194
- }), this.mapAll());
195
- }
196
- /**
197
- * Get a collection of objects
198
- *
199
- * Every time the observable variables change, and they are not undefined,
200
- * it will return result from cache, then it will **always** fetch from network,
201
- * and then keep watching the cache forever.
202
- *
203
- * You must subscribe to start getting results (and fetch from network).
204
- *
205
- * You **MUST** unsubscribe.
206
- */
207
- watchAll(queryVariablesManager, fetchPolicy = 'cache-and-network') {
208
- this.throwIfNotQuery(this.allQuery);
209
- return combineLatest({
210
- variables: queryVariablesManager.variables.pipe(
211
- // Ignore very fast variable changes
212
- debounceTime(20),
213
- // Wait for variables to be defined to prevent duplicate query: with and without variables
214
- // Null is accepted value for "no variables"
215
- filter(variables => typeof variables !== 'undefined')),
216
- partialVariables: this.getPartialVariablesForAll(),
217
- }).pipe(switchMap(result => {
218
- // Apply partial variables from service
219
- // Copy manager to prevent to apply internal variables to external QueryVariablesManager
220
- const manager = new NaturalQueryVariablesManager(queryVariablesManager);
221
- manager.merge('partial-variables', result.partialVariables);
222
- this.throwIfNotQuery(this.allQuery);
223
- return this.apollo
224
- .watchQuery({
225
- query: this.allQuery,
226
- variables: manager.variables.value,
227
- fetchPolicy: fetchPolicy,
228
- })
229
- .valueChanges.pipe(catchError(() => EMPTY), filter(r => !!r.data), this.mapAll());
230
- }));
231
- }
232
- /**
233
- * This functions allow to quickly create or update objects.
234
- *
235
- * Manages a "creation is pending" status, and update when creation is ready.
236
- * Uses regular update/updateNow and create methods.
237
- * Used mainly when editing multiple objects in same controller (like in editable arrays)
238
- */
239
- createOrUpdate(object, now = false) {
240
- this.throwIfObservable(object);
241
- this.throwIfNotQuery(this.createMutation);
242
- this.throwIfNotQuery(this.updateMutation);
243
- // If creation is pending, listen to creation observable and when ready, fire update
244
- const pendingCreation = this.creatingCache.get(object);
245
- if (pendingCreation) {
246
- return pendingCreation.pipe(switchMap(created => {
247
- return this.update({
248
- id: created.id,
249
- ...object,
250
- });
251
- }));
252
- }
253
- // If object has Id, just save it
254
- if ('id' in object && object.id) {
255
- if (now) {
256
- // used mainly for tests, because lodash debounced used in update() does not work fine with fakeAsync and tick()
257
- return this.updateNow(object);
258
- }
259
- else {
260
- return this.update(object);
261
- }
262
- }
263
- // If object was not saving, and has no ID, create it
264
- const creation = this.create(object).pipe(tap(() => {
265
- this.creatingCache.delete(object); // remove from cache
266
- }));
267
- // stores creating observable in a cache replayable version of the observable,
268
- // so several update() can subscribe to the same creation
269
- this.creatingCache.set(object, creation.pipe(shareReplay()));
270
- return creation;
271
- }
272
- /**
273
- * Create an object in DB and then refetch the list of objects
274
- */
275
- create(object) {
276
- this.throwIfObservable(object);
277
- this.throwIfNotQuery(this.createMutation);
278
- const variables = merge({}, { input: this.getInput(object, true) }, this.getPartialVariablesForCreation(object));
279
- return this.apollo
280
- .mutate({
281
- mutation: this.createMutation,
282
- variables: variables,
283
- })
284
- .pipe(map(result => {
285
- this.apollo.client.reFetchObservableQueries();
286
- return this.mapCreation(result);
287
- }));
288
- }
289
- /**
290
- * Update an object, after a short debounce
291
- */
292
- update(object) {
293
- this.throwIfObservable(object);
294
- this.throwIfNotQuery(this.updateMutation);
295
- // Keep a single instance of the debounced update function
296
- const id = object.id;
297
- return this.naturalDebounceService.debounce(this, id, object);
298
- }
299
- /**
300
- * Update an object immediately when subscribing
301
- */
302
- updateNow(object) {
303
- this.throwIfObservable(object);
304
- this.throwIfNotQuery(this.updateMutation);
305
- const variables = merge({
306
- id: object.id,
307
- input: this.getInput(object, false),
308
- }, this.getPartialVariablesForUpdate(object));
309
- return this.apollo
310
- .mutate({
311
- mutation: this.updateMutation,
312
- variables: variables,
313
- })
314
- .pipe(map(result => {
315
- this.apollo.client.reFetchObservableQueries();
316
- return this.mapUpdate(result);
317
- }));
318
- }
319
- /**
320
- * Delete objects and then refetch the list of objects
321
- */
322
- delete(objects) {
323
- this.throwIfObservable(objects);
324
- this.throwIfNotQuery(this.deleteMutation);
325
- const ids = objects.map(o => {
326
- // Cancel pending update
327
- this.naturalDebounceService.cancelOne(this, o.id);
328
- return o.id;
329
- });
330
- const variables = merge({
331
- ids: ids,
332
- }, this.getPartialVariablesForDelete(objects));
333
- return this.apollo
334
- .mutate({
335
- mutation: this.deleteMutation,
336
- variables: variables,
337
- })
338
- .pipe(
339
- // Delay the observable until Apollo refetch is completed
340
- switchMap(result => {
341
- const mappedResult = this.mapDelete(result);
342
- return from(this.apollo.client.reFetchObservableQueries()).pipe(map(() => mappedResult));
343
- }));
344
- }
345
- /**
346
- * If the id is provided, resolves an observable model. The observable model will only be emitted after we are sure
347
- * that Apollo cache is fresh and warm. Then the component can subscribe to the observable model to get the model
348
- * immediately from Apollo cache and any subsequents future mutations that may happen to Apollo cache.
349
- *
350
- * Without id, returns default values, in order to show a creation form.
351
- */
352
- resolve(id) {
353
- if (id) {
354
- const onlyNetwork = this.watchOne(id, 'network-only').pipe(first());
355
- const onlyCache = this.watchOne(id, 'cache-first');
356
- // In theory, we can rely on Apollo Cache to return a result instantly. It is very fast indeed,
357
- // but it is still asynchronous, so there may be a very short time when we don't have the model
358
- // available. To fix that, we can rely on RxJS, which is able to emit synchronously the value we just
359
- // got from server. Once Apollo Client moves to RxJS (https://github.com/apollographql/apollo-feature-requests/issues/375),
360
- // we could try to remove `startWith()`.
361
- return onlyNetwork.pipe(map(firstValue => onlyCache.pipe(startWith(firstValue))));
362
- }
363
- else {
364
- return of(of(this.getDefaultForServer()));
365
- }
366
- }
367
- /**
368
- * Return an object that match the GraphQL input type.
369
- * It creates an object with manually filled data and add uncompleted data (like required attributes that can be empty strings)
370
- */
371
- getInput(object, forCreation) {
372
- // Convert relations to their IDs for mutation
373
- object = relationsToIds(object);
374
- // Pick only attributes that we can find in the empty object
375
- // In other words, prevent to select data that has unwanted attributes
376
- const emptyObject = this.getDefaultForServer();
377
- let input = pick(object, Object.keys(emptyObject));
378
- // Complete a potentially uncompleted object with default values
379
- if (forCreation) {
380
- input = defaults(input, emptyObject);
381
- }
382
- return input;
383
- }
384
- /**
385
- * Return the number of objects matching the query. It may never complete.
386
- *
387
- * This is used for the unique validator
388
- */
389
- count(queryVariablesManager) {
390
- const queryName = 'Count' + upperCaseFirstLetter(this.plural);
391
- const filterType = upperCaseFirstLetter(this.name) + 'Filter';
392
- const query = gql `
393
- query ${queryName} ($filter: ${filterType}) {
394
- count: ${this.plural} (filter: $filter, pagination: {pageSize: 0, pageIndex: 0}) {
395
- length
396
- }
397
- }`;
398
- return this.getPartialVariablesForAll().pipe(switchMap(partialVariables => {
399
- // Copy manager to prevent to apply internal variables to external QueryVariablesManager
400
- const manager = new NaturalQueryVariablesManager(queryVariablesManager);
401
- manager.merge('partial-variables', partialVariables);
402
- return this.apollo.query({
403
- query: query,
404
- variables: manager.variables.value,
405
- fetchPolicy: 'network-only',
406
- });
407
- }), map(result => result.data.count.length));
408
- }
409
- /**
410
- * Return empty object with some default values from server perspective
411
- *
412
- * This is typically useful when showing a form for creation
413
- */
414
- getDefaultForServer() {
415
- return {};
416
- }
417
- /**
418
- * You probably **should not** use this.
419
- *
420
- * If you are trying to *call* this method, instead you probably want to call `getDefaultForServer()` to get default
421
- * values for a model, or `getFormConfig()` to get a configured form that includes extra form fields.
422
- *
423
- * If you are trying to *override* this method, instead you probably want to override `getDefaultForServer()`.
424
- *
425
- * The only and **very rare** reason to override this method is if the client needs extra form fields that cannot be
426
- * accepted by the server (not part of `XXXInput` type) and that are strictly for the client form needs. In that case,
427
- * then you can return default values for those extra form fields, and the form returned by `getFormConfig()` will
428
- * include those extra fields.
429
- */
430
- getFormExtraFieldDefaultValues() {
431
- return {};
432
- }
433
- /**
434
- * This is used to extract only the array of fetched objects out of the entire fetched data
435
- */
436
- mapAll() {
437
- return map(result => result.data[this.plural]); // See https://github.com/apollographql/apollo-client/issues/5662
438
- }
439
- /**
440
- * This is used to extract only the created object out of the entire fetched data
441
- */
442
- mapCreation(result) {
443
- const name = this.createName ?? 'create' + upperCaseFirstLetter(this.name);
444
- return result.data[name]; // See https://github.com/apollographql/apollo-client/issues/5662
445
- }
446
- /**
447
- * This is used to extract only the updated object out of the entire fetched data
448
- */
449
- mapUpdate(result) {
450
- const name = this.updateName ?? 'update' + upperCaseFirstLetter(this.name);
451
- return result.data[name]; // See https://github.com/apollographql/apollo-client/issues/5662
452
- }
453
- /**
454
- * This is used to extract only flag when deleting an object
455
- */
456
- mapDelete(result) {
457
- const name = this.deleteName ?? 'delete' + upperCaseFirstLetter(this.plural);
458
- return result.data[name]; // See https://github.com/apollographql/apollo-client/issues/5662
459
- }
460
- /**
461
- * Returns additional variables to be used when getting a single object
462
- *
463
- * This is typically a site or state ID, and is needed to get appropriate access rights
464
- */
465
- getPartialVariablesForOne() {
466
- return of({});
467
- }
468
- /**
469
- * Returns additional variables to be used when getting multiple objects
470
- *
471
- * This is typically a site or state ID, but it could be something else to further filter the query
472
- */
473
- getPartialVariablesForAll() {
474
- return of({});
475
- }
476
- /**
477
- * Returns additional variables to be used when creating an object
478
- *
479
- * This is typically a site or state ID
480
- */
481
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
482
- getPartialVariablesForCreation(object) {
483
- return {};
484
- }
485
- /**
486
- * Returns additional variables to be used when updating an object
487
- *
488
- * This is typically a site or state ID
489
- */
490
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
491
- getPartialVariablesForUpdate(object) {
492
- return {};
493
- }
494
- /**
495
- * Return additional variables to be used when deleting an object
496
- *
497
- * This is typically a site or state ID
498
- */
499
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
500
- getPartialVariablesForDelete(objects) {
501
- return {};
502
- }
503
- /**
504
- * Throw exception to prevent executing queries with invalid variables
505
- */
506
- throwIfObservable(value) {
507
- if (value instanceof Observable) {
508
- throw new Error('Cannot use Observable as variables. Instead you should use .subscribe() to call the method with a real value');
509
- }
510
- }
511
- /**
512
- * Merge given ID with additional partial variables if there is any
513
- */
514
- getVariablesForOne(id) {
515
- return this.getPartialVariablesForOne().pipe(map(partialVariables => merge({}, { id: id }, partialVariables)));
516
- }
517
- /**
518
- * Throw exception to prevent executing null queries
519
- */
520
- throwIfNotQuery(query) {
521
- if (!query) {
522
- throw new Error('GraphQL query for this method was not configured in this service constructor');
523
- }
524
- }
525
- }
526
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWJzdHJhY3QtbW9kZWwuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9zZXJ2aWNlcy9hYnN0cmFjdC1tb2RlbC5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxNQUFNLEVBQUUsR0FBRyxFQUFpQixNQUFNLGdCQUFnQixDQUFDO0FBQzNELE9BQU8sRUFBYyxhQUFhLEVBQXdCLE1BQU0scUJBQXFCLENBQUM7QUFDdEYsT0FBTyxFQUFvQyxrQkFBa0IsRUFBRSxnQkFBZ0IsRUFBYyxNQUFNLGdCQUFnQixDQUFDO0FBRXBILE9BQU8sRUFBQyxRQUFRLEVBQUUsS0FBSyxFQUFFLElBQUksRUFBQyxNQUFNLFdBQVcsQ0FBQztBQUNoRCxPQUFPLEVBQUMsVUFBVSxFQUFFLGFBQWEsRUFBRSxLQUFLLEVBQUUsS0FBSyxFQUFFLElBQUksRUFBRSxVQUFVLEVBQUUsRUFBRSxFQUFtQixNQUFNLE1BQU0sQ0FBQztBQUNyRyxPQUFPLEVBQUMsWUFBWSxFQUFFLE1BQU0sRUFBRSxHQUFHLEVBQUUsV0FBVyxFQUFFLFNBQVMsRUFBRSxTQUFTLEVBQUUsU0FBUyxFQUFFLEdBQUcsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBQzVHLE9BQU8sRUFBQyw0QkFBNEIsRUFBaUIsTUFBTSxtQ0FBbUMsQ0FBQztBQUUvRixPQUFPLEVBQUMsVUFBVSxFQUFFLGNBQWMsRUFBRSxvQkFBb0IsRUFBQyxNQUFNLG9CQUFvQixDQUFDO0FBRXBGLE9BQU8sRUFBQyxzQkFBc0IsRUFBQyxNQUFNLG9CQUFvQixDQUFDO0FBRTFELE9BQU8sRUFBQyxTQUFTLEVBQUMsTUFBTSxpQ0FBaUMsQ0FBQztBQUMxRCxPQUFPLEVBQUMsTUFBTSxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBY3JDLE1BQU0sT0FBZ0IsMkJBQTJCO0lBa0N0QjtJQUNBO0lBQ0E7SUFDQTtJQUNBO0lB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LE1BQWU7UUFDcEQsT0FBTyxFQUFFLENBQUM7SUFDZCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILDZEQUE2RDtJQUNuRCw0QkFBNEIsQ0FBQyxNQUFlO1FBQ2xELE9BQU8sRUFBRSxDQUFDO0lBQ2QsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCw2REFBNkQ7SUFDbkQsNEJBQTRCLENBQUMsT0FBa0I7UUFDckQsT0FBTyxFQUFFLENBQUM7SUFDZCxDQUFDO0lBRUQ7O09BRUc7SUFDTyxpQkFBaUIsQ0FBQyxLQUFjO1FBQ3RDLElBQUksS0FBSyxZQUFZLFVBQVUsRUFBRSxDQUFDO1lBQzlCLE1BQU0sSUFBSSxLQUFLLENBQ1gsOEdBQThHLENBQ2pILENBQUM7UUFDTixDQUFDO0lBQ0wsQ0FBQztJQUVEOztPQUVHO0lBQ0ssa0JBQWtCLENBQUMsRUFBVTtRQUNqQyxPQUFPLElBQUksQ0FBQyx5QkFBeUIsRUFBRSxDQUFDLElBQUksQ0FDeEMsR0FBRyxDQUFDLGdCQUFnQixDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsRUFBRSxFQUFFLEVBQUMsRUFBRSxFQUFFLEVBQUUsRUFBUyxFQUFFLGdCQUFnQixDQUFDLENBQUMsQ0FDekUsQ0FBQztJQUNOLENBQUM7SUFFRDs7T0FFRztJQUNLLGVBQWUsQ0FBQyxLQUEwQjtRQUM5QyxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDVCxNQUFNLElBQUksS0FBSyxDQUFDLDhFQUE4RSxDQUFDLENBQUM7UUFDcEcsQ0FBQztJQUNMLENBQUM7Q0FDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7QXBvbGxvLCBncWwsIE11dGF0aW9uUmVzdWx0fSBmcm9tICdhcG9sbG8tYW5ndWxhcic7XG5pbXBvcnQge0ZldGNoUmVzdWx0LCBOZXR3b3JrU3RhdHVzLCBXYXRjaFF1ZXJ5RmV0Y2hQb2xpY3l9IGZyb20gJ0BhcG9sbG8vY2xpZW50L2NvcmUnO1xuaW1wb3J0IHtBYnN0cmFjdENvbnRyb2wsIEFzeW5jVmFsaWRhdG9yRm4sIFVudHlwZWRGb3JtQ29udHJvbCwgVW50eXBlZEZvcm1Hcm91cCwgVmFsaWRhdG9yRm59IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7RG9jdW1lbnROb2RlfSBmcm9tICdncmFwaHFsJztcbmltcG9ydCB7ZGVmYXVsdHMsIG1lcmdlLCBwaWNrfSBmcm9tICdsb2Rhc2gtZXMnO1xuaW1wb3J0IHtjYXRjaEVycm9yLCBjb21iaW5lTGF0ZXN0LCBFTVBUWSwgZmlyc3QsIGZyb20sIE9ic2VydmFibGUsIG9mLCBPcGVyYXRvckZ1bmN0aW9ufSBmcm9tICdyeGpzJztcbmltcG9ydCB7ZGVib3VuY2VUaW1lLCBmaWx0ZXIsIG1hcCwgc2hhcmVSZXBsYXksIHN0YXJ0V2l0aCwgc3dpdGNoTWFwLCB0YWtlV2hpbGUsIHRhcH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHtOYXR1cmFsUXVlcnlWYXJpYWJsZXNNYW5hZ2VyLCBRdWVyeVZhcmlhYmxlc30gZnJvbSAnLi4vY2xhc3Nlcy9xdWVyeS12YXJpYWJsZS1tYW5hZ2VyJztcbmltcG9ydCB7TGl0ZXJhbH0gZnJvbSAnLi4vdHlwZXMvdHlwZXMnO1xuaW1wb3J0IHttYWtlUGx1cmFsLCByZWxhdGlvbnNUb0lkcywgdXBwZXJDYXNlRmlyc3RMZXR0ZXJ9IGZyb20gJy4uL2NsYXNzZXMvdXRpbGl0eSc7XG5pbXBvcnQge1BhZ2luYXRlZERhdGF9IGZyb20gJy4uL2NsYXNzZXMvZGF0YS1zb3VyY2UnO1xuaW1wb3J0IHtOYXR1cmFsRGVib3VuY2VTZXJ2aWNlfSBmcm9tICcuL2RlYm91bmNlLnNlcnZpY2UnO1xuaW1wb3J0IHtBcG9sbG9RdWVyeVJlc3VsdH0gZnJvbSAnQGFwb2xsby9jbGllbnQvY29yZS90eXBlcyc7XG5pbXBvcnQge2RlZXBDbG9uZX0gZnJvbSAnLi4vbW9kdWxlcy9zZWFyY2gvY2xhc3Nlcy91dGlscyc7XG5pbXBvcnQge2luamVjdH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbmV4cG9ydCB0eXBlIEZvcm1WYWxpZGF0b3JzID0gUmVjb3JkPHN0cmluZywgVmFsaWRhdG9yRm5bXT47XG5cbmV4cG9ydCB0eXBlIEZvcm1Bc3luY1ZhbGlkYXRvcnMgPSBSZWNvcmQ8c3RyaW5nLCBBc3luY1ZhbGlkYXRvckZuW10+O1xuXG5leHBvcnQgdHlwZSBWYXJpYWJsZXNXaXRoSW5wdXQgPSB7XG4gICAgaW5wdXQ6IExpdGVyYWw7XG59O1xuXG5leHBvcnQgdHlwZSBGb3JtQ29udHJvbHMgPSBSZWNvcmQ8c3RyaW5nLCBBYnN0cmFjdENvbnRyb2w+O1xuXG5leHBvcnQgdHlwZSBXaXRoSWQ8VD4gPSB7aWQ6IHN0cmluZ30gJiBUO1xuXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgTmF0dXJhbEFic3RyYWN0TW9kZWxTZXJ2aWNlPFxuICAgIFRvbmUsXG4gICAgVm9uZSBleHRlbmRzIHtpZDogc3RyaW5nfSxcbiAgICBUYWxsIGV4dGVuZHMgUGFnaW5hdGVkRGF0YTxMaXRlcmFsPixcbiAgICBWYWxsIGV4dGVuZHMgUXVlcnlWYXJpYWJsZXMsXG4gICAgVGNyZWF0ZSxcbiAgICBWY3JlYXRlIGV4dGVuZHMgVmFyaWFibGVzV2l0aElucHV0LFxuICAgIFR1cGRhdGUsXG4gICAgVnVwZGF0ZSBleHRlbmRzIHtpZDogc3RyaW5nOyBpbnB1dDogTGl0ZXJhbH0sXG4gICAgVGRlbGV0ZSxcbiAgICBWZGVsZXRlIGV4dGVuZHMge2lkczogc3RyaW5nW119LFxuPiB7XG4gICAgLyoqXG4gICAgICogU3RvcmUgdGhlIGNyZWF0aW9uIG11dGF0aW9ucyB0aGF0IGFyZSBwZW5kaW5nXG4gICAgICovXG4gICAgcHJpdmF0ZSByZWFkb25seSBjcmVhdGluZ0NhY2hlID0gbmV3IE1hcDxWY3JlYXRlWydpbnB1dCddIHwgV2l0aElkPFZ1cGRhdGVbJ2lucHV0J10+LCBPYnNlcnZhYmxlPFRjcmVhdGU+PigpO1xuICAgIHByb3RlY3RlZCByZWFkb25seSBhcG9sbG8gPSBpbmplY3QoQXBvbGxvKTtcbiAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgbmF0dXJhbERlYm91bmNlU2VydmljZSA9IGluamVjdChOYXR1cmFsRGVib3VuY2VTZXJ2aWNlKTtcbiAgICBwcml2YXRlIHJlYWRvbmx5IHBsdXJhbDogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICpcbiAgICAgKiBAcGFyYW0gbmFtZSBzZXJ2aWNlIGFuZCBzaW5nbGUgb2JqZWN0IHF1ZXJ5IG5hbWUgKGVnLiB1c2VyRm9yRnJvbnQgb3IgdXNlcikuXG4gICAgICogQHBhcmFtIG9uZVF1ZXJ5IEdyYXBoUUwgcXVlcnkgdG8gZmV0Y2ggYSBzaW5nbGUgb2JqZWN0IGZyb20gSUQgKGVnLiB1c2VyRm9yQ3J1ZFF1ZXJ5KS5cbiAgICAgKiBAcGFyYW0gYWxsUXVlcnkgR3JhcGhRTCBxdWVyeSB0byBmZXRjaCBhIGZpbHRlcmVkIGxpc3Qgb2Ygb2JqZWN0cyAoZWcuIHVzZXJzRm9yQ3J1ZFF1ZXJ5KS5cbiAgICAgKiBAcGFyYW0gY3JlYXRlTXV0YXRpb24gR3JhcGhRTCBtdXRhdGlvbiB0byBjcmVhdGUgYW4gb2JqZWN0LlxuICAgICAqIEBwYXJhbSB1cGRhdGVNdXRhdGlvbiBHcmFwaFFMIG11dGF0aW9uIHRvIHVwZGF0ZSBhbiBvYmplY3QuXG4gICAgICogQHBhcmFtIGRlbGV0ZU11dGF0aW9uIEdyYXBoUUwgbXV0YXRpb24gdG8gZGVsZXRlIGEgbGlzdCBvZiBvYmplY3RzLlxuICAgICAqIEBwYXJhbSBwbHVyYWwgbGlzdCBxdWVyeSBuYW1lIChlZy4gdXNlcnNGb3JGcm9udCBvciB1c2VycykuXG4gICAgICogQHBhcmFtIGNyZWF0ZU5hbWUgY3JlYXRlIG9iamVjdCBtdXRhdGlvbiBuYW1lIChlZy4gY3JlYXRlVXNlcikuXG4gICAgICogQHBhcmFtIHVwZGF0ZU5hbWUgdXBkYXRlIG9iamVjdCBtdXRhdGlvbiBuYW1lIChlZy4gdXBkYXRlVXNlcikuXG4gICAgICogQHBhcmFtIGRlbGV0ZU5hbWUgZGVsZXRlIG9iamVjdCBtdXRhdGlvbiBuYW1lIChlZy4gZGVsZXRlVXNlcnMpLlxuICAgICAqL1xuICAgIHB1YmxpYyBjb25zdHJ1Y3RvcihcbiAgICAgICAgcHJvdGVjdGVkIHJlYWRvbmx5IG5hbWU6IHN0cmluZyxcbiAgICAgICAgcHJvdGVjdGVkIHJlYWRvbmx5IG9uZVF1ZXJ5OiBEb2N1bWVudE5vZGUgfCBudWxsLFxuICAgICAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgYWxsUXVlcnk6IERvY3VtZW50Tm9kZSB8IG51bGwsXG4gICAgICAgIHByb3RlY3RlZCByZWFkb25seSBjcmVhdGVNdXRhdGlvbjogRG9jdW1lbnROb2RlIHwgbnVsbCxcbiAgICAgICAgcHJvdGVjdGVkIHJlYWRvbmx5IHVwZGF0ZU11dGF0aW9uOiBEb2N1bWVudE5vZGUgfCBudWxsLFxuICAgICAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgZGVsZXRlTXV0YXRpb246IERvY3VtZW50Tm9kZSB8IG51bGwsXG4gICAgICAgIHBsdXJhbDogc3RyaW5nIHwgbnVsbCA9IG51bGwsXG4gICAgICAgIHByaXZhdGUgcmVhZG9ubHkgY3JlYXRlTmFtZTogc3RyaW5nIHwgbnVsbCA9IG51bGwsXG4gICAgICAgIHByaXZhdGUgcmVhZG9ubHkgdXBkYXRlTmFtZTogc3RyaW5nIHwgbnVsbCA9IG51bGwsXG4gICAgICAgIHByaXZhdGUgcmVhZG9ubHkgZGVsZXRlTmFtZTogc3RyaW5nIHwgbnVsbCA9IG51bGwsXG4gICAgKSB7XG4gICAgICAgIHRoaXMucGx1cmFsID0gcGx1cmFsID8/IG1ha2VQbHVyYWwodGhpcy5uYW1lKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBMaXN0IG9mIGluZGl2aWR1YWwgZmllbGRzIHZhbGlkYXRvcnNcbiAgICAgKi9cbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gICAgcHVibGljIGdldEZvcm1WYWxpZGF0b3JzKG1vZGVsPzogTGl0ZXJhbCk6IEZvcm1WYWxpZGF0b3JzIHtcbiAgICAgICAgcmV0dXJuIHt9O1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIExpc3Qgb2YgaW5kaXZpZHVhbCBhc3luYyBmaWVsZHMgdmFsaWRhdG9yc1xuICAgICAqL1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW51c2VkLXZhcnNcbiAgICBwdWJsaWMgZ2V0Rm9ybUFzeW5jVmFsaWRhdG9ycyhtb2RlbD86IExpdGVyYWwpOiBGb3JtQXN5bmNWYWxpZGF0b3JzIHtcbiAgICAgICAgcmV0dXJuIHt9O1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIExpc3Qgb2YgZ3JvdXBlZCBmaWVsZHMgdmFsaWRhdG9ycyAobGlrZSBwYXNzd29yZCArIGNvbmZpcm0gcGFzc3dvcmQpXG4gICAgICovXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICAgIHB1YmxpYyBnZXRGb3JtR3JvdXBWYWxpZGF0b3JzKG1vZGVsPzogTGl0ZXJhbCk6IFZhbGlkYXRvckZuW10ge1xuICAgICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogTGlzdCBvZiBhc3luYyBncm91cCBmaWVsZHMgdmFsaWRhdG9ycyAobGlrZSB1bmlxdWUgY29uc3RyYWludCBvbiBtdWx0aXBsZSBjb2x1bW5zKVxuICAgICAqL1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW51c2VkLXZhcnNcbiAgICBwdWJsaWMgZ2V0Rm9ybUdyb3VwQXN5bmNWYWxpZGF0b3JzKG1vZGVsPzogTGl0ZXJhbCk6IEFzeW5jVmFsaWRhdG9yRm5bXSB7XG4gICAgICAgIHJldHVybiBbXTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0Rm9ybUNvbmZpZyhtb2RlbDogTGl0ZXJhbCk6IEZvcm1Db250cm9scyB7XG4gICAgICAgIGNvbnN0IHZhbHVlcyA9IHsuLi50aGlzLmdldERlZmF1bHRGb3JTZXJ2ZXIoKSwgLi4udGhpcy5nZXRGb3JtRXh0cmFGaWVsZERlZmF1bHRWYWx1ZXMoKX07XG4gICAgICAgIGNvbnN0IHZhbGlkYXRvcnMgPSB0aGlzLmdldEZvcm1WYWxpZGF0b3JzKG1vZGVsKTtcbiAgICAgICAgY29uc3QgYXN5bmNWYWxpZGF0b3JzID0gdGhpcy5nZXRGb3JtQXN5bmNWYWxpZGF0b3JzKG1vZGVsKTtcbiAgICAgICAgY29uc3QgY29udHJvbHM6IEZvcm1Db250cm9scyA9IHt9O1xuICAgICAgICBjb25zdCBkaXNhYmxlZCA9IG1vZGVsLnBlcm1pc3Npb25zID8gIW1vZGVsLnBlcm1pc3Npb25zLnVwZGF0ZSA6IGZhbHNlO1xuXG4gICAgICAgIGlmIChtb2RlbC5pZCkge1xuICAgICAgICAgICAgY29udHJvbHMuaWQgPSBuZXcgVW50eXBlZEZvcm1Db250cm9sKHt2YWx1ZTogbW9kZWwuaWQsIGRpc2FibGVkOiB0cnVlfSk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBDb25maWd1cmUgZm9ybSBmb3IgZWFjaCBmaWVsZCBvZiBtb2RlbFxuICAgICAgICBmb3IgKGNvbnN0IGtleSBvZiBPYmplY3Qua2V5cyh2YWx1ZXMpKSB7XG4gICAgICAgICAgICBjb25zdCB2YWx1ZSA9IG1vZGVsW2tleV0gIT09IHVuZGVmaW5lZCA/IG1vZGVsW2tleV0gOiB2YWx1ZXNba2V5XTtcbiAgICAgICAgICAgIGNvbnN0IGZvcm1TdGF0ZSA9IHtcbiAgICAgICAgICAgICAgICB2YWx1ZTogdmFsdWUsXG4gICAgICAgICAgICAgICAgZGlzYWJsZWQ6IGRpc2FibGVkLFxuICAgICAgICAgICAgfTtcbiAgICAgICAgICAgIGNvbnN0IHZhbGlkYXRvciA9IHR5cGVvZiB2YWxpZGF0b3JzW2tleV0gIT09ICd1bmRlZmluZWQnID8gdmFsaWRhdG9yc1trZXldIDogbnVsbDtcbiAgICAgICAgICAgIGNvbnN0IGFzeW5jVmFsaWRhdG9yID0gdHlwZW9mIGFzeW5jVmFsaWRhdG9yc1trZXldICE9PSAndW5kZWZpbmVkJyA/IGFzeW5jVmFsaWRhdG9yc1trZXldIDogbnVsbDtcblxuICAgICAgICAgICAgY29udHJvbHNba2V5XSA9IG5ldyBVbnR5cGVkRm9ybUNvbnRyb2woZm9ybVN0YXRlLCB2YWxpZGF0b3IsIGFzeW5jVmFsaWRhdG9yKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIENvbmZpZ3VyZSBmb3JtIGZvciBleHRyYSB2YWxpZGF0b3JzIHRoYXQgYXJlIG5vdCBvbiBhIHNwZWNpZmljIGZpZWxkXG4gICAgICAgIGZvciAoY29uc3Qga2V5IG9mIE9iamVjdC5rZXlzKHZhbGlkYXRvcnMpKSB7XG4gICAgICAgICAgICBpZiAoIWNvbnRyb2xzW2tleV0pIHtcbiAgICAgICAgICAgICAgICBjb25zdCBmb3JtU3RhdGUgPSB7XG4gICAgICAgICAgICAgICAgICAgIHZhbHVlOiBtb2RlbFtrZXldID8gbW9kZWxba2V5XSA6IG51bGwsXG4gICAgICAgICAgICAgICAgICAgIGRpc2FibGVkOiBkaXNhYmxlZCxcbiAgICAgICAgICAgICAgICB9O1xuXG4gICAgICAgICAgICAgICAgY29udHJvbHNba2V5XSA9IG5ldyBVbnR5cGVkRm9ybUNvbnRyb2woZm9ybVN0YXRlLCB2YWxpZGF0b3JzW2tleV0pO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgZm9yIChjb25zdCBrZXkgb2YgT2JqZWN0LmtleXMoYXN5bmNWYWxpZGF0b3JzKSkge1xuICAgICAgICAgICAgaWYgKGNvbnRyb2xzW2tleV0gJiYgYXN5bmNWYWxpZGF0b3JzW2tleV0pIHtcbiAgICAgICAgICAgICAgICBjb250cm9sc1trZXldLnNldEFzeW5jVmFsaWRhdG9ycyhhc3luY1ZhbGlkYXRvcnNba2V5XSk7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIGNvbnN0IGZvcm1TdGF0ZSA9IHtcbiAgICAgICAgICAgICAgICAgICAgdmFsdWU6IG1vZGVsW2tleV0gPyBtb2RlbFtrZXldIDogbnVsbCxcbiAgICAgICAgICAgICAgICAgICAgZGlzYWJsZWQ6IGRpc2FibGVkLFxuICAgICAgICAgICAgICAgIH07XG5cbiAgICAgICAgICAgICAgICBjb250cm9sc1trZXldID0gbmV3IFVudHlwZWRGb3JtQ29udHJvbChmb3JtU3RhdGUsIG51bGwsIGFzeW5jVmFsaWRhdG9yc1trZXldKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBjb250cm9scztcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGUgdGhlIGZpbmFsIEZvcm1Hcm91cCBmb3IgdGhlIG9iamVjdCwgaW5jbHVkaW5nIGFsbCB2YWxpZGF0b3JzXG4gICAgICpcbiAgICAgKiBUaGlzIG1ldGhvZCBzaG91bGQgKipub3QqKiBiZSBvdmVycmlkZGVuLCBidXQgaW5zdGVhZCBgZ2V0Rm9ybUNvbmZpZ2AsXG4gICAgICogYGdldEZvcm1Hcm91cFZhbGlkYXRvcnNgLCBgZ2V0Rm9ybUdyb3VwQXN5bmNWYWxpZGF0b3JzYCBtaWdodCBiZS5cbiAgICAgKi9cbiAgICBwdWJsaWMgZ2V0Rm9ybUdyb3VwKG1vZGVsOiBMaXRlcmFsKTogVW50eXBlZEZvcm1Hcm91cCB7XG4gICAgICAgIGNvbnN0IGZvcm1Db25maWcgPSB0aGlzLmdldEZvcm1Db25maWcoZGVlcENsb25lKG1vZGVsKSk7XG4gICAgICAgIHJldHVybiBuZXcgVW50eXBlZEZvcm1Hcm91cChmb3JtQ29uZmlnLCB7XG4gICAgICAgICAgICB2YWxpZGF0b3JzOiB0aGlzLmdldEZvcm1Hcm91cFZhbGlkYXRvcnMobW9kZWwpLFxuICAgICAgICAgICAgYXN5bmNWYWxpZGF0b3JzOiB0aGlzLmdldEZvcm1Hcm91cEFzeW5jVmFsaWRhdG9ycyhtb2RlbCksXG4gICAgICAgIH0pO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIEdldCBhIHNpbmdsZSBvYmplY3RcbiAgICAgKlxuICAgICAqIElmIGF2YWlsYWJsZSBpdCB3aWxsIGVtaXQgb2JqZWN0IGZyb20gY2FjaGUgaW1tZWRpYXRlbHksIHRoZW4gaXRcbiAgICAgKiB3aWxsICoqYWx3YXlzKiogZmV0Y2ggZnJvbSBuZXR3b3JrIGFuZCB0aGVuIHRoZSBvYnNlcnZhYmxlIHdpbGwgYmUgY29tcGxldGVkLlxuICAgICAqXG4gICAgICogWW91IG11c3Qgc3Vic2NyaWJlIHRvIHN0YXJ0IGdldHRpbmcgcmVzdWx0cyAoYW5kIGZldGNoIGZyb20gbmV0d29yaykuXG4gICAgICovXG4gICAgcHVibGljIGdldE9uZShpZDogc3RyaW5nKTogT2JzZXJ2YWJsZTxUb25lPiB7XG4gICAgICAgIHJldHVybiB0aGlzLnByZXBhcmVPbmVRdWVyeShpZCwgJ2NhY2hlLWFuZC1uZXR3b3JrJykucGlwZShcbiAgICAgICAgICAgIHRha2VXaGlsZShyZXN1bHQgPT4gcmVzdWx0Lm5ldHdvcmtTdGF0dXMgIT09IE5ldHdvcmtTdGF0dXMucmVhZHksIHRydWUpLFxuICAgICAgICAgICAgbWFwKHJlc3VsdCA9PiAocmVzdWx0LmRhdGEgYXMgTGl0ZXJhbClbdGhpcy5uYW1lXSksXG4gICAgICAgICk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogV2F0Y2ggYSBzaW5nbGUgb2JqZWN0XG4gICAgICpcbiAgICAgKiBJZiBhdmFpbGFibGUgaXQgd2lsbCBlbWl0IG9iamVjdCBmcm9tIGNhY2hlIGltbWVkaWF0ZWx5LCB0aGVuIGl0XG4gICAgICogd2lsbCAqKmFsd2F5cyoqIGZldGNoIGZyb20gbmV0d29yaywgYW5kIHRoZW4ga2VlcCB3YXRjaGluZyB0aGUgY2FjaGUgZm9yZXZlci5cbiAgICAgKlxuICAgICAqIFlvdSBtdXN0IHN1YnNjcmliZSB0byBzdGFydCBnZXR0aW5nIHJlc3VsdHMgKGFuZCBmZXRjaCBmcm9tIG5ldHdvcmspLlxuICAgICAqXG4gICAgICogWW91ICoqTVVTVCoqIHVuc3Vic2NyaWJlLlxuICAgICAqL1xuICAgIHB1YmxpYyB3YXRjaE9uZShpZDogc3RyaW5nLCBmZXRjaFBvbGljeTogV2F0Y2hRdWVyeUZldGNoUG9saWN5ID0gJ2NhY2hlLWFuZC1uZXR3b3JrJyk6IE9ic2VydmFibGU8VG9uZT4ge1xuICAgICAgICByZXR1cm4gdGhpcy5wcmVwYXJlT25lUXVlcnkoaWQsIGZldGNoUG9saWN5KS5waXBlKG1hcChyZXN1bHQgPT4gKHJlc3VsdC5kYXRhIGFzIExpdGVyYWwpW3RoaXMubmFtZV0pKTtcbiAgICB9XG5cbiAgICBwcml2YXRlIHByZXBhcmVPbmVRdWVyeShpZDogc3RyaW5nLCBmZXRjaFBvbGljeTogV2F0Y2hRdWVyeUZldGNoUG9saWN5KTogT2JzZXJ2YWJsZTxBcG9sbG9RdWVyeVJlc3VsdDx1bmtub3duPj4ge1xuICAgICAgICB0aGlzLnRocm93SWZPYnNlcnZhYmxlKGlkKTtcbiAgICAgICAgdGhpcy50aHJvd0lmTm90UXVlcnkodGhpcy5vbmVRdWVyeSk7XG5cbiAgICAgICAgcmV0dXJuIHRoaXMuZ2V0VmFyaWFibGVzRm9yT25lKGlkKS5waXBlKFxuICAgICAgICAgICAgc3dpdGNoTWFwKHZhcmlhYmxlcyA9PiB7XG4gICAgICAgICAgICAgICAgdGhpcy50aHJvd0lmTm90UXVlcnkodGhpcy5vbmVRdWVyeSk7XG5cbiAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5hcG9sbG8ud2F0Y2hRdWVyeTx1bmtub3duLCBWb25lPih7XG4gICAgICAgICAgICAgICAgICAgIHF1ZXJ5OiB0aGlzLm9uZVF1ZXJ5LFxuICAgICAgICAgICAgICAgICAgICB2YXJpYWJsZXM6IHZhcmlhYmxlcyxcbiAgICAgICAgICAgICAgICAgICAgZmV0Y2hQb2xpY3k6IGZldGNoUG9saWN5LFxuICAgICAgICAgICAgICAgICAgICBuZXh0RmV0Y2hQb2xpY3k6ICdjYWNoZS1vbmx5JyxcbiAgICAgICAgICAgICAgICB9KS52YWx1ZUNoYW5nZXM7XG4gICAgICAgICAgICB9KSxcbiAgICAgICAgICAgIGZpbHRlcihyZXN1bHQgPT4gISFyZXN1bHQuZGF0YSksXG4gICAgICAgICk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogR2V0IGEgY29sbGVjdGlvbiBvZiBvYmplY3RzXG4gICAgICpcbiAgICAgKiBJdCB3aWxsICoqYWx3YXlzKiogZmV0Y2ggZnJvbSBuZXR3b3JrIGFuZCB0aGVuIHRoZSBvYnNlcnZhYmxlIHdpbGwgYmUgY29tcGxldGVkLlxuICAgICAqIE5vIGNhY2hlIGlzIGV2ZXIgdXNlZCwgc28gaXQncyBzbG93IGJ1dCBjb3JyZWN0LlxuICAgICAqL1xuICAgIHB1YmxpYyBnZXRBbGwocXVlcnlWYXJpYWJsZXNNYW5hZ2VyOiBOYXR1cmFsUXVlcnlWYXJpYWJsZXNNYW5hZ2VyPFZhbGw+KTogT2JzZXJ2YWJsZTxUYWxsPiB7XG4gICAgICAgIHRoaXMudGhyb3dJZk5vdFF1ZXJ5KHRoaXMuYWxsUXVlcnkpO1xuXG4gICAgICAgIHJldHVybiB0aGlzLmdldFBhcnRpYWxWYXJpYWJsZXNGb3JBbGwoKS5waXBlKFxuICAgICAgICAgICAgZmlyc3QoKSxcbiAgICAgICAgICAgIHN3aXRjaE1hcChwYXJ0aWFsVmFyaWFibGVzID0+IHtcbiAgICAgICAgICAgICAgICB0aGlzLnRocm93SWZOb3RRdWVyeSh0aGlzLmFsbFF1ZXJ5KTtcblxuICAgICAgICAgICAgICAgIC8vIENvcHkgbWFuYWdlciB0byBwcmV2ZW50IHRvIGFwcGx5IGludGVybmFsIHZhcmlhYmxlcyB0byBleHRlcm5hbCBRdWVyeVZhcmlhYmxlc01hbmFnZXJcbiAgICAgICAgICAgICAgICBjb25zdCBtYW5hZ2VyID0gbmV3IE5hdHVyYWxRdWVyeVZhcmlhYmxlc01hbmFnZXI8VmFsbD4ocXVlcnlWYXJpYWJsZXNNYW5hZ2VyKTtcbiAgICAgICAgICAgICAgICBtYW5hZ2VyLm1lcmdlKCdwYXJ0aWFsLXZhcmlhYmxlcycsIHBhcnRpYWxWYXJpYWJsZXMpO1xuXG4gICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMuYXBvbGxvLnF1ZXJ5PHVua25vd24sIFZhbGw+KHtcbiAgICAgICAgICAgICAgICAgICAgcXVlcnk6IHRoaXMuYWxsUXVlcnksXG4gICAgICAgICAgICAgICAgICAgIHZhcmlhYmxlczogbWFuYWdlci52YXJpYWJsZXMudmFsdWUsXG4gICAgICAgICAgICAgICAgICAgIGZldGNoUG9saWN5OiAnbmV0d29yay1vbmx5JyxcbiAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH0pLFxuICAgICAgICAgICAgdGhpcy5tYXBBbGwoKSxcbiAgICAgICAgKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBHZXQgYSBjb2xsZWN0aW9uIG9mIG9iamVjdHNcbiAgICAgKlxuICAgICAqIEV2ZXJ5IHRpbWUgdGhlIG9ic2VydmFibGUgdmFyaWFibGVzIGNoYW5nZSwgYW5kIHRoZXkgYXJlIG5vdCB1bmRlZmluZWQsXG4gICAgICogaXQgd2lsbCByZXR1cm4gcmVzdWx0IGZyb20gY2FjaGUsIHRoZW4gaXQgd2lsbCAqKmFsd2F5cyoqIGZldGNoIGZyb20gbmV0d29yayxcbiAgICAgKiBhbmQgdGhlbiBrZWVwIHdhdGNoaW5nIHRoZSBjYWNoZSBmb3JldmVyLlxuICAgICAqXG4gICAgICogWW91IG11c3Qgc3Vic2NyaWJlIHRvIHN0YXJ0IGdldHRpbmcgcmVzdWx0cyAoYW5kIGZldGNoIGZyb20gbmV0d29yaykuXG4gICAgICpcbiAgICAgKiBZb3UgKipNVVNUKiogdW5zdWJzY3JpYmUuXG4gICAgICovXG4gICAgcHVibGljIHdhdGNoQWxsKFxuICAgICAgICBxdWVyeVZhcmlhYmxlc01hbmFnZXI6IE5hdHVyYWxRdWVyeVZhcmlhYmxlc01hbmFnZXI8VmFsbD4sXG4gICAgICAgIGZldGNoUG9saWN5OiBXYXRjaFF1ZXJ5RmV0Y2hQb2xpY3kgPSAnY2FjaGUtYW5kLW5ldHdvcmsnLFxuICAgICk6IE9ic2VydmFibGU8VGFsbD4ge1xuICAgICAgICB0aGlzLnRocm93SWZOb3RRdWVyeSh0aGlzLmFsbFF1ZXJ5KTtcblxuICAgICAgICByZXR1cm4gY29tYmluZUxhdGVzdCh7XG4gICAgICAgICAgICB2YXJpYWJsZXM6IHF1ZXJ5VmFyaWFibGVzTWFuYWdlci52YXJpYWJsZXMucGlwZShcbiAgICAgICAgICAgICAgICAvLyBJZ25vcmUgdmVyeSBmYXN0IHZhcmlhYmxlIGNoYW5nZXNcbiAgICAgICAgICAgICAgICBkZWJvdW5jZVRpbWUoMjApLFxuICAgICAgICAgICAgICAgIC8vIFdhaXQgZm9yIHZhcmlhYmxlcyB0byBiZSBkZWZpbmVkIHRvIHByZXZlbnQgZHVwbGljYXRlIHF1ZXJ5OiB3aXRoIGFuZCB3aXRob3V0IHZhcmlhYmxlc1xuICAgICAgICAgICAgICAgIC8vIE51bGwgaXMgYWNjZXB0ZWQgdmFsdWUgZm9yIFwibm8gdmFyaWFibGVzXCJcbiAgICAgICAgICAgICAgICBmaWx0ZXIodmFyaWFibGVzID0+IHR5cGVvZiB2YXJpYWJsZXMgIT09ICd1bmRlZmluZWQnKSxcbiAgICAgICAgICAgICksXG4gICAgICAgICAgICBwYXJ0aWFsVmFyaWFibGVzOiB0aGlzLmdldFBhcnRpYWxWYXJpYWJsZXNGb3JBbGwoKSxcbiAgICAgICAgfSkucGlwZShcbiAgICAgICAgICAgIHN3aXRjaE1hcChyZXN1bHQgPT4ge1xuICAgICAgICAgICAgICAgIC8vIEFwcGx5IHBhcnRpYWwgdmFyaWFibGVzIGZyb20gc2VydmljZVxuICAgICAgICAgICAgICAgIC8vIENvcHkgbWFuYWdlciB0byBwcmV2ZW50IHRvIGFwcGx5IGludGVybmFsIHZhcmlhYmxlcyB0byBleHRlcm5hbCBRdWVyeVZhcmlhYmxlc01hbmFnZXJcbiAgICAgICAgICAgICAgICBjb25zdCBtYW5hZ2VyID0gbmV3IE5hdHVyYWxRdWVyeVZhcmlhYmxlc01hbmFnZXI8VmFsbD4ocXVlcnlWYXJpYWJsZXNNYW5hZ2VyKTtcbiAgICAgICAgICAgICAgICBtYW5hZ2VyLm1lcmdlKCdwYXJ0aWFsLXZhcmlhYmxlcycsIHJlc3VsdC5wYXJ0aWFsVmFyaWFibGVzKTtcblxuICAgICAgICAgICAgICAgIHRoaXMudGhyb3dJZk5vdFF1ZXJ5KHRoaXMuYWxsUXVlcnkpO1xuXG4gICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMuYXBvbGxvXG4gICAgICAgICAgICAgICAgICAgIC53YXRjaFF1ZXJ5PHVua25vd24sIFZhbGw+KHtcbiAgICAgICAgICAgICAgICAgICAgICAgIHF1ZXJ5OiB0aGlzLmFsbFF1ZXJ5LFxuICAgICAgICAgICAgICAgICAgICAgICAgdmFyaWFibGVzOiBtYW5hZ2VyLnZhcmlhYmxlcy52YWx1ZSxcbiAgICAgICAgICAgICAgICAgICAgICAgIGZldGNoUG9saWN5OiBmZXRjaFBvbGljeSxcbiAgICAgICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAgICAgLnZhbHVlQ2hhbmdlcy5waXBlKFxuICAgICAgICAgICAgICAgICAgICAgICAgY2F0Y2hFcnJvcigoKSA9PiBFTVBUWSksXG4gICAgICAgICAgICAgICAgICAgICAgICBmaWx0ZXIociA9PiAhIXIuZGF0YSksXG4gICAgICAgICAgICAgICAgICAgICAgICB0aGlzLm1hcEFsbCgpLFxuICAgICAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfSksXG4gICAgICAgICk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogVGhpcyBmdW5jdGlvbnMgYWxsb3cgdG8gcXVpY2tseSBjcmVhdGUgb3IgdXBkYXRlIG9iamVjdHMuXG4gICAgICpcbiAgICAgKiBNYW5hZ2VzIGEgXCJjcmVhdGlvbiBpcyBwZW5kaW5nXCIgc3RhdHVzLCBhbmQgdXBkYXRlIHdoZW4gY3JlYXRpb24gaXMgcmVhZHkuXG4gICAgICogVXNlcyByZWd1bGFyIHVwZGF0ZS91cGRhdGVOb3cgYW5kIGNyZWF0ZSBtZXRob2RzLlxuICAgICAqIFVzZWQgbWFpbmx5IHdoZW4gZWRpdGluZyBtdWx0aXBsZSBvYmplY3RzIGluIHNhbWUgY29udHJvbGxlciAobGlrZSBpbiBlZGl0YWJsZSBhcnJheXMpXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yVXBkYXRlKFxuICAgICAgICBvYmplY3Q6IFZjcmVhdGVbJ2lucHV0J10gfCBXaXRoSWQ8VnVwZGF0ZVsnaW5wdXQnXT4sXG4gICAgICAgIG5vdyA9IGZhbHNlLFxuICAgICk6IE9ic2VydmFibGU8VGNyZWF0ZSB8IFR1cGRhdGU+IHtcbiAgICAgICAgdGhpcy50aHJvd0lmT2JzZXJ2YWJsZShvYmplY3QpO1xuICAgICAgICB0aGlzLnRocm93SWZOb3RRdWVyeSh0aGlzLmNyZWF0ZU11dGF0aW9uKTtcbiAgICAgICAgdGhpcy50aHJvd0lmTm90UXVlcnkodGhpcy51cGRhdGVNdXRhdGlvbik7XG5cbiAgICAgICAgLy8gSWYgY3JlYXRpb24gaXMgcGVuZGluZywgbGlzdGVuIHRvIGNyZWF0aW9uIG9ic2VydmFibGUgYW5kIHdoZW4gcmVhZHksIGZpcmUgdXBkYXRlXG4gICAgICAgIGNvbnN0IHBlbmRpbmdDcmVhdGlvbiA9IHRoaXMuY3JlYXRpbmdDYWNoZS5nZXQob2JqZWN0KTtcbiAgICAgICAgaWYgKHBlbmRpbmdDcmVhdGlvbikge1xuICAgICAgICAgICAgcmV0dXJuIHBlbmRpbmdDcmVhdGlvbi5waXBlKFxuICAgICAgICAgICAgICAgIHN3aXRjaE1hcChjcmVhdGVkID0+IHtcbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMudXBkYXRlKHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlkOiAoY3JlYXRlZCBhcyBXaXRoSWQ8VGNyZWF0ZT4pLmlkLFxuICAgICAgICAgICAgICAgICAgICAgICAgLi4uKG9iamVjdCBhcyBWY3JlYXRlWydpbnB1dCddKSxcbiAgICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgICAgfSksXG4gICAgICAgICAgICApO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gSWYgb2JqZWN0IGhhcyBJZCwganVzdCBzYXZlIGl0XG4gICAgICAgIGlmICgnaWQnIGluIG9iamVjdCAmJiBvYmplY3QuaWQpIHtcbiAgICAgICAgICAgIGlmIChub3cpIHtcbiAgICAgICAgICAgICAgICAvLyB1c2VkIG1haW5seSBmb3IgdGVzdHMsIGJlY2F1c2UgbG9kYXNoIGRlYm91bmNlZCB1c2VkIGluIHVwZGF0ZSgpIGRvZXMgbm90IHdvcmsgZmluZSB3aXRoIGZha2VBc3luYyBhbmQgdGljaygpXG4gICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMudXBkYXRlTm93KG9iamVjdCBhcyBXaXRoSWQ8VnVwZGF0ZVsnaW5wdXQnXT4pO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy51cGRhdGUob2JqZWN0IGFzIFdpdGhJZDxWdXBkYXRlWydpbnB1dCddPik7XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyBJZiBvYmplY3Qgd2FzIG5vdCBzYXZpbmcsIGFuZCBoYXMgbm8gSUQsIGNyZWF0ZSBpdFxuICAgICAgICBjb25zdCBjcmVhdGlvbiA9IHRoaXMuY3JlYXRlKG9iamVjdCkucGlwZShcbiAgICAgICAgICAgIHRhcCgoKSA9PiB7XG4gICAgICAgICAgICAgICAgdGhpcy5jcmVhdGluZ0NhY2hlLmRlbGV0ZShvYmplY3QpOyAvLyByZW1vdmUgZnJvbSBjYWNoZVxuICAgICAgICAgICAgfSksXG4gICAgICAgICk7XG5cbiAgICAgICAgLy8gc3RvcmVzIGNyZWF0aW5nIG9ic2VydmFibGUgaW4gYSBjYWNoZSByZXBsYXlhYmxlIHZlcnNpb24gb2YgdGhlIG9ic2VydmFibGUsXG4gICAgICAgIC8vIHNvIHNldmVyYWwgdXBkYXRlKCkgY2FuIHN1YnNjcmliZSB0byB0aGUgc2FtZSBjcmVhdGlvblxuICAgICAgICB0aGlzLmNyZWF0aW5nQ2FjaGUuc2V0KG9iamVjdCwgY3JlYXRpb24ucGlwZShzaGFyZVJlcGxheSgpKSk7XG5cbiAgICAgICAgcmV0dXJuIGNyZWF0aW9uO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZSBhbiBvYmplY3QgaW4gREIgYW5kIHRoZW4gcmVmZXRjaCB0aGUgbGlzdCBvZiBvYmplY3RzXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZShvYmplY3Q6IFZjcmVhdGVbJ2lucHV0J10pOiBPYnNlcnZhYmxlPFRjcmVhdGU+IHtcbiAgICAgICAgdGhpcy50aHJvd0lmT2JzZXJ2YWJsZShvYmplY3QpO1xuICAgICAgICB0aGlzLnRocm93SWZOb3RRdWVyeSh0aGlzLmNyZWF0ZU11dGF0aW9uKTtcblxuICAgICAgICBjb25zdCB2YXJpYWJsZXMgPSBtZXJnZShcbiAgICAgICAgICAgIHt9LFxuICAgICAgICAgICAge2lucHV0OiB0aGlzLmdldElucHV0KG9iamVjdCwgdHJ1ZSl9LFxuICAgICAgICAgICAgdGhpcy5nZXRQYXJ0aWFsVmFyaWFibGVzRm9yQ3JlYXRpb24ob2JqZWN0KSxcbiAgICAgICAgKSBhcyBWY3JlYXRlO1xuXG4gICAgICAgIHJldHVybiB0aGlzLmFwb2xsb1xuICAgICAgICAgICAgLm11dGF0ZTxUY3JlYXRlLCBWY3JlYXRlPih7XG4gICAgICAgICAgICAgICAgbXV0YXRpb246IHRoaXMuY3JlYXRlTXV0YXRpb24sXG4gICAgICAgICAgICAgICAgdmFyaWFibGVzOiB2YXJpYWJsZXMsXG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLnBpcGUoXG4gICAgICAgICAgICAgICAgbWFwKHJlc3VsdCA9PiB7XG4gICAgICAgICAgICAgICAgICAgIHRoaXMuYXBvbGxvLmNsaWVudC5yZUZldGNoT2JzZXJ2YWJsZVF1ZXJpZXMoKTtcbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMubWFwQ3JlYXRpb24ocmVzdWx0KTtcbiAgICAgICAgICAgICAgICB9KSxcbiAgICAgICAgICAgICk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogVXBkYXRlIGFuIG9iamVjdCwgYWZ0ZXIgYSBzaG9ydCBkZWJvdW5jZVxuICAgICAqL1xuICAgIHB1YmxpYyB1cGRhdGUob2JqZWN0OiBXaXRoSWQ8VnVwZGF0ZVsnaW5wdXQnXT4pOiBPYnNlcnZhYmxlPFR1cGRhdGU+IHtcbiAgICAgICAgdGhpcy50aHJvd0lmT2JzZXJ2YWJsZShvYmplY3QpO1xuICAgICAgICB0aGlzLnRocm93SWZOb3RRdWVyeSh0aGlzLnVwZGF0ZU11dGF0aW9uKTtcblxuICAgICAgICAvLyBLZWVwIGEgc2luZ2xlIGluc3RhbmNlIG9mIHRoZSBkZWJvdW5jZWQgdXBkYXRlIGZ1bmN0aW9uXG4gICAgICAgIGNvbnN0IGlkID0gb2JqZWN0LmlkO1xuXG4gICAgICAgIHJldHVybiB0aGlzLm5hdHVyYWxEZWJvdW5jZVNlcnZpY2UuZGVib3VuY2UodGhpcywgaWQsIG9iamVjdCk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogVXBkYXRlIGFuIG9iamVjdCBpbW1lZGlhdGVseSB3aGVuIHN1YnNjcmliaW5nXG4gICAgICovXG4gICAgcHVibGljIHVwZGF0ZU5vdyhvYmplY3Q6IFdpdGhJZDxWdXBkYXRlWydpbnB1dCddPik6IE9ic2VydmFibGU8VHVwZGF0ZT4ge1xuICAgICAgICB0aGlzLnRocm93SWZPYnNlcnZhYmxlKG9iamVjdCk7XG4gICAgICAgIHRoaXMudGhyb3dJZk5vdFF1ZXJ5KHRoaXMudXBkYXRlTXV0YXRpb24pO1xuXG4gICAgICAgIGNvbnN0IHZhcmlhYmxlcyA9IG1lcmdlKFxuICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgIGlkOiBvYmplY3QuaWQsXG4gICAgICAgICAgICAgICAgaW5wdXQ6IHRoaXMuZ2V0SW5wdXQob2JqZWN0LCBmYWxzZSksXG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAgdGhpcy5nZXRQYXJ0aWFsVmFyaWFibGVzRm9yVXBkYXRlKG9iamVjdCksXG4gICAgICAgICkgYXMgVnVwZGF0ZTtcblxuICAgICAgICByZXR1cm4gdGhpcy5hcG9sbG9cbiAgICAgICAgICAgIC5tdXRhdGU8VHVwZGF0ZSwgVnVwZGF0ZT4oe1xuICAgICAgICAgICAgICAgIG11dGF0aW9uOiB0aGlzLnVwZGF0ZU11dGF0aW9uLFxuICAgICAgICAgICAgICAgIHZhcmlhYmxlczogdmFyaWFibGVzLFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIC5waXBlKFxuICAgICAgICAgICAgICAgIG1hcChyZXN1bHQgPT4ge1xuICAgICAgICAgICAgICAgICAgICB0aGlzLmFwb2xsby5jbGllbnQucmVGZXRjaE9ic2VydmFibGVRdWVyaWVzKCk7XG4gICAgICAgICAgICAgICAgICAgIHJldHVybiB0aGlzLm1hcFVwZGF0ZShyZXN1bHQpO1xuICAgICAgICAgICAgICAgIH0pLFxuICAgICAgICAgICAgKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBEZWxldGUgb2JqZWN0cyBhbmQgdGhlbiByZWZldGNoIHRoZSBsaXN0IG9mIG9iamVjdHNcbiAgICAgKi9cbiAgICBwdWJsaWMgZGVsZXRlKG9iamVjdHM6IHtpZDogc3RyaW5nfVtdKTogT2JzZXJ2YWJsZTxUZGVsZXRlPiB7XG4gICAgICAgIHRoaXMudGhyb3dJZk9ic2VydmFibGUob2JqZWN0cyk7XG4gICAgICAgIHRoaXMudGhyb3dJZk5vdFF1ZXJ5KHRoaXMuZGVsZXRlTXV0YXRpb24pO1xuXG4gICAgICAgIGNvbnN0IGlkcyA9IG9iamVjdHMubWFwKG8gPT4ge1xuICAgICAgICAgICAgLy8gQ2FuY2VsIHBlbmRpbmcgdXBkYXRlXG4gICAgICAgICAgICB0aGlzLm5hdHVyYWxEZWJvdW5jZVNlcnZpY2UuY2FuY2VsT25lKHRoaXMsIG8uaWQpO1xuXG4gICAgICAgICAgICByZXR1cm4gby5pZDtcbiAgICAgICAgfSk7XG4gICAgICAgIGNvbnN0IHZhcmlhYmxlcyA9IG1lcmdlKFxuICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgIGlkczogaWRzLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIHRoaXMuZ2V0UGFydGlhbFZhcmlhYmxlc0ZvckRlbGV0ZShvYmplY3RzKSxcbiAgICAgICAgKSBhcyBWZGVsZXRlO1xuXG4gICAgICAgIHJldHVybiB0aGlzLmFwb2xsb1xuICAgICAgICAgICAgLm11dGF0ZTxUZGVsZXRlLCBWZGVsZXRlPih7XG4gICAgICAgICAgICAgICAgbXV0YXRpb246IHRoaXMuZGVsZXRlTXV0YXRpb24sXG4gICAgICAgICAgICAgICAgdmFyaWFibGVzOiB2YXJpYWJsZXMsXG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLnBpcGUoXG4gICAgICAgICAgICAgICAgLy8gRGVsYXkgdGhlIG9ic2VydmFibGUgdW50aWwgQXBvbGxvIHJlZmV0Y2ggaXMgY29tcGxldGVkXG4gICAgICAgICAgICAgICAgc3dpdGNoTWFwKHJlc3VsdCA9PiB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IG1hcHBlZFJlc3VsdCA9IHRoaXMubWFwRGVsZXRlKHJlc3VsdCk7XG5cbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZyb20odGhpcy5hcG9sbG8uY2xpZW50LnJlRmV0Y2hPYnNlcnZhYmxlUXVlcmllcygpKS5waXBlKG1hcCgoKSA9PiBtYXBwZWRSZXN1bHQpKTtcbiAgICAgICAgICAgICAgICB9KSxcbiAgICAgICAgICAgICk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogSWYgdGhlIGlkIGlzIHByb3ZpZGVkLCByZXNvbHZlcyBhbiBvYnNlcnZhYmxlIG1vZGVsLiBUaGUgb2JzZXJ2YWJsZSBtb2RlbCB3aWxsIG9ubHkgYmUgZW1pdHRlZCBhZnRlciB3ZSBhcmUgc3VyZVxuICAgICAqIHRoYXQgQXBvbGxvIGNhY2hlIGlzIGZyZXNoIGFuZCB3YXJtLiBUaGVuIHRoZSBjb21wb25lbnQgY2FuIHN1YnNjcmliZSB0byB0aGUgb2JzZXJ2YWJsZSBtb2RlbCB0byBnZXQgdGhlIG1vZGVsXG4gICAgICogaW1tZWRpYXRlbHkgZnJvbSBBcG9sbG8gY2FjaGUgYW5kIGFueSBzdWJzZXF1ZW50cyBmdXR1cmUgbXV0YXRpb25zIHRoYXQgbWF5IGhhcHBlbiB0byBBcG9sbG8gY2FjaGUuXG4gICAgICpcbiAgICAgKiBXaXRob3V0IGlkLCByZXR1cm5zIGRlZmF1bHQgdmFsdWVzLCBpbiBvcmRlciB0byBzaG93IGEgY3JlYXRpb24gZm9ybS5cbiAgICAgKi9cbiAgICBwdWJsaWMgcmVzb2x2ZShpZDogc3RyaW5nIHwgdW5kZWZpbmVkKTogT2JzZXJ2YWJsZTxPYnNlcnZhYmxlPFRvbmUgfCBWY3JlYXRlWydpbnB1dCddPj4ge1xuICAgICAgICBpZiAoaWQpIHtcbiAgICAgICAgICAgIGNvbnN0IG9ubHlOZXR3b3JrID0gdGhpcy53YXRjaE9uZShpZCwgJ25ldHdvcmstb25seScpLnBpcGUoZmlyc3QoKSk7XG4gICAgICAgICAgICBjb25zdCBvbmx5Q2FjaGUgPSB0aGlzLndhdGNoT25lKGlkLCAnY2FjaGUtZmlyc3QnKTtcblxuICAgICAgICAgICAgLy8gSW4gdGhlb3J5LCB3ZSBjYW4gcmVseSBvbiBBcG9sbG8gQ2FjaGUgdG8gcmV0dXJuIGEgcmVzdWx0IGluc3RhbnRseS4gSXQgaXMgdmVyeSBmYXN0IGluZGVlZCxcbiAgICAgICAgICAgIC8vIGJ1dCBpdCBpcyBzdGlsbCBhc3luY2hyb25vdXMsIHNvIHRoZXJlIG1heSBiZSBhIHZlcnkgc2hvcnQgdGltZSB3aGVuIHdlIGRvbid0IGhhdmUgdGhlIG1vZGVsXG4gICAgICAgICAgICAvLyBhdmFpbGFibGUuIFRvIGZpeCB0aGF0LCB3ZSBjYW4gcmVseSBvbiBSeEpTLCB3aGljaCBpcyBhYmxlIHRvIGVtaXQgc3luY2hyb25vdXNseSB0aGUgdmFsdWUgd2UganVzdFxuICAgICAgICAgICAgLy8gZ290IGZyb20gc2VydmVyLiBPbmNlIEFwb2xsbyBDbGllbnQgbW92ZXMgdG8gUnhKUyAoaHR0cHM6Ly9naXRodWIuY29tL2Fwb2xsb2dyYXBocWwvYXBvbGxvLWZlYXR1cmUtcmVxdWVzdHMvaXNzdWVzLzM3NSksXG4gICAgICAgICAgICAvLyB3ZSBjb3VsZCB0cnkgdG8gcmVtb3ZlIGBzdGFydFdpdGgoKWAuXG4gICAgICAgICAgICByZXR1cm4gb25seU5ldHdvcmsucGlwZShtYXAoZmlyc3RWYWx1ZSA9PiBvbmx5Q2FjaGUucGlwZShzdGFydFdpdGgoZmlyc3RWYWx1ZSkpKSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZXR1cm4gb2Yob2YodGhpcy5nZXREZWZhdWx0Rm9yU2VydmVyKCkpKTtcbiAgICAgICAgfVxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFJldHVybiBhbiBvYmplY3QgdGhhdCBtYXRjaCB0aGUgR3JhcGhRTCBpbnB1dCB0eXBlLlxuICAgICAqIEl0IGNyZWF0ZXMgYW4gb2JqZWN0IHdpdGggbWFudWFsbHkgZmlsbGVkIGRhdGEgYW5kIGFkZCB1bmNvbXBsZXRlZCBkYXRhIChsaWtlIHJlcXVpcmVkIGF0dHJpYnV0ZXMgdGhhdCBjYW4gYmUgZW1wdHkgc3RyaW5ncylcbiAgICAgKi9cbiAgICBwdWJsaWMgZ2V0SW5wdXQob2JqZWN0OiBMaXRlcmFsLCBmb3JDcmVhdGlvbjogYm9vbGVhbik6IFZjcmVhdGVbJ2lucHV0J10gfCBWdXBkYXRlWydpbnB1dCddIHtcbiAgICAgICAgLy8gQ29udmVydCByZWxhdGlvbnMgdG8gdGhlaXIgSURzIGZvciBtdXRhdGlvblxuICAgICAgICBvYmplY3QgPSByZWxhdGlvbnNUb0lkcyhvYmplY3QpO1xuXG4gICAgICAgIC8vIFBpY2sgb25seSBhdHRyaWJ1dGVzIHRoYXQgd2UgY2FuIGZpbmQgaW4gdGhlIGVtcHR5IG9iamVjdFxuICAgICAgICAvLyBJbiBvdGhlciB3b3JkcywgcHJldmVudCB0byBzZWxlY3QgZGF0YSB0aGF0IGhhcyB1bndhbnRlZCBhdHRyaWJ1dGVzXG4gICAgICAgIGNvbnN0IGVtcHR5T2JqZWN0ID0gdGhpcy5nZXREZWZhdWx0Rm9yU2VydmVyKCk7XG4gICAgICAgIGxldCBpbnB1dCA9IHBpY2sob2JqZWN0LCBPYmplY3Qua2V5cyhlbXB0eU9iamVjdCkpO1xuXG4gICAgICAgIC8vIENvbXBsZXRlIGEgcG90ZW50aWFsbHkgdW5jb21wbGV0ZWQgb2JqZWN0IHdpdGggZGVmYXVsdCB2YWx1ZXNcbiAgICAgICAgaWYgKGZvckNyZWF0aW9uKSB7XG4gICAgICAgICAgICBpbnB1dCA9IGRlZmF1bHRzKGlucHV0LCBlbXB0eU9iamVjdCk7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gaW5wdXQ7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogUmV0dXJuIHRoZSBudW1iZXIgb2Ygb2JqZWN0cyBtYXRjaGluZyB0aGUgcXVlcnkuIEl0IG1heSBuZXZlciBjb21wbGV0ZS5cbiAgICAgKlxuICAgICAqIFRoaXMgaXMgdXNlZCBmb3IgdGhlIHVuaXF1ZSB2YWxpZGF0b3JcbiAgICAgKi9cbiAgICBwdWJsaWMgY291bnQocXVlcnlWYXJpYWJsZXNNYW5hZ2VyOiBOYXR1cmFsUXVlcnlWYXJpYWJsZXNNYW5hZ2VyPFZhbGw+KTogT2JzZXJ2YWJsZTxudW1iZXI+IHtcbiAgICAgICAgY29uc3QgcXVlcnlOYW1lID0gJ0NvdW50JyArIHVwcGVyQ2FzZUZpcnN0TGV0dGVyKHRoaXMucGx1cmFsKTtcbiAgICAgICAgY29uc3QgZmlsdGVyVHlwZSA9IHVwcGVyQ2FzZUZpcnN0TGV0dGVyKHRoaXMubmFtZSkgKyAnRmlsdGVyJztcblxuICAgICAgICBjb25zdCBxdWVyeSA9IGdxbGBcbiAgICAgICAgICAgIHF1ZXJ5ICR7cXVlcnlOYW1lfSAoJGZpbHRlcjogJHtmaWx0ZXJUeXBlfSkge1xuICAgICAgICAgICAgY291bnQ6ICR7dGhpcy5wbHVyYWx9IChmaWx0ZXI6ICRmaWx0ZXIsIHBhZ2luYXRpb246IHtwYWdlU2l6ZTogMCwgcGFnZUluZGV4OiAwfSkge1xuICAgICAgICAgICAgbGVuZ3RoXG4gICAgICAgICAgICB9XG4gICAgICAgICAgICB9YDtcblxuICAgICAgICByZXR1cm4gdGhpcy5nZXRQYXJ0aWFsVmFyaWFibGVzRm9yQWxsKCkucGlwZShcbiAgICAgICAgICAgIHN3aXRjaE1hcChwYXJ0aWFsVmFyaWFibGVzID0+IHtcbiAgICAgICAgICAgICAgICAvLyBDb3B5IG1hbmFnZXIgdG8gcHJldmVudCB0byBhcHBseSBpbnRlcm5hbCB2YXJpYWJsZXMgdG8gZXh0ZXJuYWwgUXVlcnlWYXJpYWJsZXNNYW5hZ2VyXG4gICAgICAgICAgICAgICAgY29uc3QgbWFuYWdlciA9IG5ldyBOYXR1cmFsUXVlcnlWYXJpYWJsZXNNYW5hZ2VyPFZhbGw+KHF1ZXJ5VmFyaWFibGVzTWFuYWdlcik7XG4gICAgICAgICAgICAgICAgbWFuYWdlci5tZXJnZSgncGFydGlhbC12YXJpYWJsZXMnLCBwYXJ0aWFsVmFyaWFibGVzKTtcblxuICAgICAgICAgICAgICAgIHJldHVybiB0aGlzLmFwb2xsby5xdWVyeTx7Y291bnQ6IHtsZW5ndGg6IG51bWJlcn19LCBWYWxsPih7XG4gICAgICAgICAgICAgICAgICAgIHF1ZXJ5OiBxdWVyeSxcbiAgICAgICAgICAgICAgICAgICAgdmFyaWFibGVzOiBtYW5hZ2VyLnZhcmlhYmxlcy52YWx1ZSxcbiAgICAgICAgICAgICAgICAgICAgZmV0Y2hQb2xpY3k6ICduZXR3b3JrLW9ubHknLFxuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfSksXG4gICAgICAgICAgICBtYXAocmVzdWx0ID0+IHJlc3VsdC5kYXRhLmNvdW50Lmxlbmd0aCksXG4gICAgICAgICk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogUmV0dXJuIGVtcHR5IG9iamVjdCB3aXRoIHNvbWUgZGVmYXVsdCB2YWx1ZXMgZnJvbSBzZXJ2ZXIgcGVyc3BlY3RpdmVcbiAgICAgKlxuICAgICAqIFRoaXMgaXMgdHlwaWNhbGx5IHVzZWZ1bCB3aGVuIHNob3dpbmcgYSBmb3JtIGZvciBjcmVhdGlvblxuICAgICAqL1xuICAgIHB1YmxpYyBnZXREZWZhdWx0Rm9yU2VydmVyKCk6IFZjcmVhdGVbJ2lucHV0J10ge1xuICAgICAgICByZXR1cm4ge307XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogWW91IHByb2JhYmx5ICoqc2hvdWxkIG5vdCoqIHVzZSB0aGlzLlxuICAgICAqXG4gICAgICogSWYgeW91IGFyZSB0cnlpbmcgdG8gKmNhbGwqIHRoaXMgbWV0aG9kLCBpbnN0ZWFkIHlvdSBwcm9iYWJseSB3YW50IHRvIGNhbGwgYGdldERlZmF1bHRGb3JTZXJ2ZXIoKWAgdG8gZ2V0IGRlZmF1bHRcbiAgICAgKiB2YWx1ZXMgZm9yIGEgbW9kZWwsIG9yIGBnZXRGb3JtQ29uZmlnKClgIHRvIGdldCBhIGNvbmZpZ3VyZWQgZm9ybSB0aGF0IGluY2x1ZGVzIGV4dHJhIGZvcm0gZmllbGRzLlxuICAgICAqXG4gICAgICogSWYgeW91IGFyZSB0cnlpbmcgdG8gKm92ZXJyaWRlKiB0aGlzIG1ldGhvZCwgaW5zdGVhZCB5b3UgcHJvYmFibHkgd2FudCB0byBvdmVycmlkZSBgZ2V0RGVmYXVsdEZvclNlcnZlcigpYC5cbiAgICAgKlxuICAgICAqIFRoZSBvbmx5IGFuZCAqKnZlcnkgcmFyZSoqIHJlYXNvbiB0byBvdmVycmlkZSB0aGlzIG1ldGhvZCBpcyBpZiB0aGUgY2xpZW50IG5lZWRzIGV4dHJhIGZvcm0gZmllbGRzIHRoYXQgY2Fubm90IGJlXG4gICAgICogYWNjZXB0ZWQgYnkgdGhlIHNlcnZlciAobm90IHBhcnQgb2YgYFhYWElucHV0YCB0eXBlKSBhbmQgdGhhdCBhcmUgc3RyaWN0bHkgZm9yIHRoZSBjbGllbnQgZm9ybSBuZWVkcy4gSW4gdGhhdCBjYXNlLFxuICAgICAqIHRoZW4geW91IGNhbiByZXR1cm4gZGVmYXVsdCB2YWx1ZXMgZm9yIHRob3NlIGV4dHJhIGZvcm0gZmllbGRzLCBhbmQgdGhlIGZvcm0gcmV0dXJuZWQgYnkgYGdldEZvcm1Db25maWcoKWAgd2lsbFxuICAgICAqIGluY2x1ZGUgdGhvc2UgZXh0cmEgZmllbGRzLlxuICAgICAqL1xuICAgIHByb3RlY3RlZCBnZXRGb3JtRXh0cmFGaWVsZERlZmF1bHRWYWx1ZXMoKTogTGl0ZXJhbCB7XG4gICAgICAgIHJldHVybiB7fTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBUaGlzIGlzIHVzZWQgdG8gZXh0cmFjdCBvbmx5IHRoZSBhcnJheSBvZiBmZXRjaGVkIG9iamVjdHMgb3V0IG9mIHRoZSBlbnRpcmUgZmV0Y2hlZCBkYXRhXG4gICAgICovXG4gICAgcHJvdGVjdGVkIG1hcEFsbCgpOiBPcGVyYXRvckZ1bmN0aW9uPEZldGNoUmVzdWx0PHVua25vd24+LCBUYWxsPiB7XG4gICAgICAgIHJldHVybiBtYXAocmVzdWx0ID0+IChyZXN1bHQuZGF0YSBhcyBhbnkpW3RoaXMucGx1cmFsXSk7IC8vIFNlZSBodHRwczovL2dpdGh1Yi5jb20vYXBvbGxvZ3JhcGhxbC9hcG9sbG8tY2xpZW50L2lzc3Vlcy81NjYyXG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogVGhpcyBpcyB1c2VkIHRvIGV4dHJhY3Qgb25seSB0aGUgY3JlYXRlZCBvYmplY3Qgb3V0IG9mIHRoZSBlbnRpcmUgZmV0Y2hlZCBkYXRhXG4gICAgICovXG4gICAgcHJvdGVjdGVkIG1hcENyZWF0aW9uKHJlc3VsdDogTXV0YXRpb25SZXN1bHQ8dW5rbm93bj4pOiBUY3JlYXRlIHtcbiAgICAgICAgY29uc3QgbmFtZSA9IHRoaXMuY3JlYXRlTmFtZSA/PyAnY3JlYXRlJyArIHVwcGVyQ2FzZUZpcnN0TGV0dGVyKHRoaXMubmFtZSk7XG4gICAgICAgIHJldHVybiAocmVzdWx0LmRhdGEgYXMgYW55KVtuYW1lXTsgLy8gU2VlIGh0dHBzOi8vZ2l0aHViLmNvbS9hcG9sbG9ncmFwaHFsL2Fwb2xsby1jbGllbnQvaXNzdWVzLzU2NjJcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBUaGlzIGlzIHVzZWQgdG8gZXh0cmFjdCBvbmx5IHRoZSB1cGRhdGVkIG9iamVjdCBvdXQgb2YgdGhlIGVudGlyZSBmZXRjaGVkIGRhdGFcbiAgICAgKi9cbiAgICBwcm90ZWN0ZWQgbWFwVXBkYXRlKHJlc3VsdDogTXV0YXRpb25SZXN1bHQ8dW5rbm93bj4pOiBUdXBkYXRlIHtcbiAgICAgICAgY29uc3QgbmFtZSA9IHRoaXMudXBkYXRlTmFtZSA/PyAndXBkYXRlJyArIHVwcGVyQ2FzZUZpcnN0TGV0dGVyKHRoaXMubmFtZSk7XG4gICAgICAgIHJldHVybiAocmVzdWx0LmRhdGEgYXMgYW55KVtuYW1lXTsgLy8gU2VlIGh0dHBzOi8vZ2l0aHViLmNvbS9hcG9sbG9ncmFwaHFsL2Fwb2xsby1jbGllbnQvaXNzdWVzLzU2NjJcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBUaGlzIGlzIHVzZWQgdG8gZXh0cmFjdCBvbmx5IGZsYWcgd2hlbiBkZWxldGluZyBhbiBvYmplY3RcbiAgICAgKi9cbiAgICBwcm90ZWN0ZWQgbWFwRGVsZXRlKHJlc3VsdDogTXV0YXRpb25SZXN1bHQ8dW5rbm93bj4pOiBUZGVsZXRlIHtcbiAgICAgICAgY29uc3QgbmFtZSA9IHRoaXMuZGVsZXRlTmFtZSA/PyAnZGVsZXRlJyArIHVwcGVyQ2FzZUZpcnN0TGV0dGVyKHRoaXMucGx1cmFsKTtcbiAgICAgICAgcmV0dXJuIChyZXN1bHQuZGF0YSBhcyBhbnkpW25hbWVdOyAvLyBTZWUgaHR0cHM6Ly9naXRodWIuY29tL2Fwb2xsb2dyYXBocWwvYXBvbGxvLWNsaWVudC9pc3N1ZXMvNTY2MlxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFJldHVybnMgYWRkaXRpb25hbCB2YXJpYWJsZXMgdG8gYmUgdXNlZCB3aGVuIGdldHRpbmcgYSBzaW5nbGUgb2JqZWN0XG4gICAgICpcbiAgICAgKiBUaGlzIGlzIHR5cGljYWxseSBhIHNpdGUgb3Igc3RhdGUgSUQsIGFuZCBpcyBuZWVkZWQgdG8gZ2V0IGFwcHJvcHJpYXRlIGFjY2VzcyByaWdodHNcbiAgICAgKi9cbiAgICBwcm90ZWN0ZWQgZ2V0UGFydGlhbFZhcmlhYmxlc0Zvck9uZSgpOiBPYnNlcnZhYmxlPFBhcnRpYWw8Vm9uZT4+IHtcbiAgICAgICAgcmV0dXJuIG9mKHt9KTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBSZXR1cm5zIGFkZGl0aW9uYWwgdmFyaWFibGVzIHRvIGJlIHVzZWQgd2hlbiBnZXR0aW5nIG11bHRpcGxlIG9iamVjdHNcbiAgICAgKlxuICAgICAqIFRoaXMgaXMgdHlwaWNhbGx5IGEgc2l0ZSBvciBzdGF0ZSBJRCwgYnV0IGl0IGNvdWxkIGJlIHNvbWV0aGluZyBlbHNlIHRvIGZ1cnRoZXIgZmlsdGVyIHRoZSBxdWVyeVxuICAgICAqL1xuICAgIHB1YmxpYyBnZXRQYXJ0aWFsVmFyaWFibGVzRm9yQWxsKCk6IE9ic2VydmFibGU8UGFydGlhbDxWYWxsPj4ge1xuICAgICAgICByZXR1cm4gb2Yoe30pO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFJldHVybnMgYWRkaXRpb25hbCB2YXJpYWJsZXMgdG8gYmUgdXNlZCB3aGVuIGNyZWF0aW5nIGFuIG9iamVjdFxuICAgICAqXG4gICAgICogVGhpcyBpcyB0eXBpY2FsbHkgYSBzaXRlIG9yIHN0YXRlIElEXG4gICAgICovXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICAgIHByb3RlY3RlZCBnZXRQYXJ0aWFsVmFyaWFibGVzRm9yQ3JlYXRpb24ob2JqZWN0OiBMaXRlcmFsKTogUGFydGlhbDxWY3JlYXRlPiB7XG4gICAgICAgIHJldHVybiB7fTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBSZXR1cm5zIGFkZGl0aW9uYWwgdmFyaWFibGVzIHRvIGJlIHVzZWQgd2hlbiB1cGRhdGluZyBhbiBvYmplY3RcbiAgICAgKlxuICAgICAqIFRoaXMgaXMgdHlwaWNhbGx5IGEgc2l0ZSBvciBzdGF0ZSBJRFxuICAgICAqL1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW51c2VkLXZhcnNcbiAgICBwcm90ZWN0ZWQgZ2V0UGFydGlhbFZhcmlhYmxlc0ZvclVwZGF0ZShvYmplY3Q6IExpdGVyYWwpOiBQYXJ0aWFsPFZ1cGRhdGU+IHtcbiAgICAgICAgcmV0dXJuIHt9O1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFJldHVybiBhZGRpdGlvbmFsIHZhcmlhYmxlcyB0byBiZSB1c2VkIHdoZW4gZGVsZXRpbmcgYW4gb2JqZWN0XG4gICAgICpcbiAgICAgKiBUaGlzIGlzIHR5cGljYWxseSBhIHNpdGUgb3Igc3RhdGUgSURcbiAgICAgKi9cbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gICAgcHJvdGVjdGVkIGdldFBhcnRpYWxWYXJpYWJsZXNGb3JEZWxldGUob2JqZWN0czogTGl0ZXJhbFtdKTogUGFydGlhbDxWZGVsZXRlPiB7XG4gICAgICAgIHJldHVybiB7fTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBUaHJvdyBleGNlcHRpb24gdG8gcHJldmVudCBleGVjdXRpbmcgcXVlcmllcyB3aXRoIGludmFsaWQgdmFyaWFibGVzXG4gICAgICovXG4gICAgcHJvdGVjdGVkIHRocm93SWZPYnNlcnZhYmxlKHZhbHVlOiB1bmtub3duKTogdm9pZCB7XG4gICAgICAgIGlmICh2YWx1ZSBpbnN0YW5jZW9mIE9ic2VydmFibGUpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICAgICAgICAnQ2Fubm90IHVzZSBPYnNlcnZhYmxlIGFzIHZhcmlhYmxlcy4gSW5zdGVhZCB5b3Ugc2hvdWxkIHVzZSAuc3Vic2NyaWJlKCkgdG8gY2FsbCB0aGUgbWV0aG9kIHdpdGggYSByZWFsIHZhbHVlJyxcbiAgICAgICAgICAgICk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBNZXJnZSBnaXZlbiBJRCB3aXRoIGFkZGl0aW9uYWwgcGFydGlhbCB2YXJpYWJsZXMgaWYgdGhlcmUgaXMgYW55XG4gICAgICovXG4gICAgcHJpdmF0ZSBnZXRWYXJpYWJsZXNGb3JPbmUoaWQ6IHN0cmluZyk6IE9ic2VydmFibGU8Vm9uZT4ge1xuICAgICAgICByZXR1cm4gdGhpcy5nZXRQYXJ0aWFsVmFyaWFibGVzRm9yT25lKCkucGlwZShcbiAgICAgICAgICAgIG1hcChwYXJ0aWFsVmFyaWFibGVzID0+IG1lcmdlKHt9LCB7aWQ6IGlkfSBhcyBWb25lLCBwYXJ0aWFsVmFyaWFibGVzKSksXG4gICAgICAgICk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogVGhyb3cgZXhjZXB0aW9uIHRvIHByZXZlbnQgZXhlY3V0aW5nIG51bGwgcXVlcmllc1xuICAgICAqL1xuICAgIHByaXZhdGUgdGhyb3dJZk5vdFF1ZXJ5KHF1ZXJ5OiBEb2N1bWVudE5vZGUgfCBudWxsKTogYXNzZXJ0cyBxdWVyeSB7XG4gICAgICAgIGlmICghcXVlcnkpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcignR3JhcGhRTCBxdWVyeSBmb3IgdGhpcyBtZXRob2Qgd2FzIG5vdCBjb25maWd1cmVkIGluIHRoaXMgc2VydmljZSBjb25zdHJ1Y3RvcicpO1xuICAgICAgICB9XG4gICAgfVxufVxuIl19