@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.
- package/fesm2022/ecodev-natural-vanilla.mjs +1193 -0
- package/fesm2022/ecodev-natural-vanilla.mjs.map +1 -0
- package/fesm2022/ecodev-natural.mjs +436 -425
- package/fesm2022/ecodev-natural.mjs.map +1 -1
- package/lib/classes/network-activity.service.d.ts +54 -0
- package/lib/classes/validators.d.ts +1 -1
- package/lib/modules/columns-picker/columns-picker.component.d.ts +2 -2
- package/lib/modules/common/services/seo.provider.d.ts +2 -2
- package/lib/modules/dropdown-components/type-select/type-select.component.d.ts +1 -1
- package/lib/modules/file/abstract-file.d.ts +6 -3
- package/lib/modules/file/component/file.component.d.ts +2 -2
- package/lib/modules/file/file-drop.directive.d.ts +2 -3
- package/lib/modules/fixed-button-detail/fixed-button-detail.component.d.ts +2 -3
- package/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.d.ts +3 -3
- package/lib/modules/icon/icon.module.d.ts +2 -2
- package/lib/modules/panels/panels.service.d.ts +1 -2
- package/lib/modules/relations/relations.component.d.ts +3 -3
- package/lib/modules/search/dropdown-container/dropdown-container.component.d.ts +2 -3
- package/lib/modules/search/group/group.component.d.ts +2 -3
- package/lib/modules/search/input/input.component.d.ts +5 -5
- package/lib/modules/search/search/search.component.d.ts +2 -2
- package/lib/modules/select/abstract-select.component.d.ts +3 -3
- package/lib/modules/select/select/select.component.d.ts +1 -1
- package/lib/modules/sidenav/sidenav-container/sidenav-container.component.d.ts +1 -1
- package/lib/modules/table-button/table-button.component.d.ts +4 -2
- package/package.json +16 -14
- package/public-api.d.ts +1 -0
- package/src/lib/_natural.theme.scss +1 -2
- package/vanilla/index.d.ts +5 -0
- package/vanilla/package.json +3 -0
- package/vanilla/public-api.d.ts +11 -0
- package/vanilla/src/lib/classes/crypto.d.ts +8 -0
- package/vanilla/src/lib/classes/data-source.d.ts +32 -0
- package/vanilla/src/lib/classes/query-variable-manager-utils.d.ts +2 -0
- package/vanilla/src/lib/classes/query-variable-manager.d.ts +91 -0
- package/vanilla/src/lib/classes/signing.d.ts +7 -0
- package/vanilla/src/lib/classes/utility.d.ts +77 -0
- package/vanilla/src/lib/modules/search/classes/graphql-doctrine.types.d.ts +83 -0
- package/vanilla/src/lib/modules/search/classes/utils.d.ts +17 -0
- package/vanilla/src/lib/modules/search/types/dropdown-component.d.ts +20 -0
- package/vanilla/src/lib/modules/search/types/facet.d.ts +75 -0
- package/vanilla/src/lib/modules/search/types/values.d.ts +32 -0
- package/vanilla/src/lib/services/abstract-model.service.d.ts +244 -0
- package/vanilla/src/lib/services/debounce.service.d.ts +52 -0
- package/vanilla/src/lib/types/types.d.ts +100 -0
- package/esm2022/ecodev-natural.mjs +0 -5
- package/esm2022/lib/classes/abstract-detail.mjs +0 -229
- package/esm2022/lib/classes/abstract-editable-list.mjs +0 -99
- package/esm2022/lib/classes/abstract-list.mjs +0 -461
- package/esm2022/lib/classes/abstract-navigable-list.mjs +0 -133
- package/esm2022/lib/classes/apollo-utils.mjs +0 -59
- package/esm2022/lib/classes/crypto.mjs +0 -23
- package/esm2022/lib/classes/cumulative-changes.mjs +0 -50
- package/esm2022/lib/classes/data-source.mjs +0 -71
- package/esm2022/lib/classes/providers.mjs +0 -13
- package/esm2022/lib/classes/query-variable-manager-utils.mjs +0 -14
- package/esm2022/lib/classes/query-variable-manager.mjs +0 -172
- package/esm2022/lib/classes/rxjs.mjs +0 -54
- package/esm2022/lib/classes/signing.mjs +0 -38
- package/esm2022/lib/classes/tld.mjs +0 -1476
- package/esm2022/lib/classes/utility.mjs +0 -234
- package/esm2022/lib/classes/validators.mjs +0 -179
- package/esm2022/lib/directives/http-prefix.directive.mjs +0 -47
- package/esm2022/lib/modules/alert/alert.service.mjs +0 -53
- package/esm2022/lib/modules/alert/confirm.component.mjs +0 -16
- package/esm2022/lib/modules/alert/public-api.mjs +0 -6
- package/esm2022/lib/modules/avatar/component/avatar.component.mjs +0 -203
- package/esm2022/lib/modules/avatar/public-api.mjs +0 -6
- package/esm2022/lib/modules/avatar/service/avatar.service.mjs +0 -63
- package/esm2022/lib/modules/avatar/sources/gravatar.mjs +0 -29
- package/esm2022/lib/modules/avatar/sources/image.mjs +0 -13
- package/esm2022/lib/modules/avatar/sources/initials.mjs +0 -39
- package/esm2022/lib/modules/avatar/sources/source.mjs +0 -16
- package/esm2022/lib/modules/columns-picker/columns-picker.component.mjs +0 -145
- package/esm2022/lib/modules/columns-picker/public-api.mjs +0 -5
- package/esm2022/lib/modules/columns-picker/types.mjs +0 -2
- package/esm2022/lib/modules/common/directives/background-density.directive.mjs +0 -63
- package/esm2022/lib/modules/common/directives/linkable-tab.directive.mjs +0 -93
- package/esm2022/lib/modules/common/directives/src-density.directive.mjs +0 -72
- package/esm2022/lib/modules/common/pipes/capitalize.pipe.mjs +0 -24
- package/esm2022/lib/modules/common/pipes/ellipsis.pipe.mjs +0 -17
- package/esm2022/lib/modules/common/pipes/enum.pipe.mjs +0 -24
- package/esm2022/lib/modules/common/pipes/time-ago.pipe.mjs +0 -140
- package/esm2022/lib/modules/common/public-api.mjs +0 -14
- package/esm2022/lib/modules/common/services/memory-storage.mjs +0 -110
- package/esm2022/lib/modules/common/services/seo.provider.mjs +0 -23
- package/esm2022/lib/modules/common/services/seo.service.mjs +0 -235
- package/esm2022/lib/modules/detail-header/detail-header.component.mjs +0 -84
- package/esm2022/lib/modules/detail-header/public-api.mjs +0 -5
- package/esm2022/lib/modules/dialog-trigger/dialog-trigger.component.mjs +0 -72
- package/esm2022/lib/modules/dialog-trigger/public-api.mjs +0 -5
- package/esm2022/lib/modules/dropdown-components/abstract-association-select-component.directive.mjs +0 -100
- package/esm2022/lib/modules/dropdown-components/public-api.mjs +0 -14
- package/esm2022/lib/modules/dropdown-components/type-boolean/type-boolean.component.mjs +0 -39
- package/esm2022/lib/modules/dropdown-components/type-date/type-date.component.mjs +0 -173
- package/esm2022/lib/modules/dropdown-components/type-date-range/type-date-range.component.mjs +0 -134
- package/esm2022/lib/modules/dropdown-components/type-hierarchic-selector/type-hierarchic-selector.component.mjs +0 -80
- package/esm2022/lib/modules/dropdown-components/type-natural-select/type-natural-select.component.mjs +0 -48
- package/esm2022/lib/modules/dropdown-components/type-number/type-number.component.mjs +0 -110
- package/esm2022/lib/modules/dropdown-components/type-options/type-options.component.mjs +0 -64
- package/esm2022/lib/modules/dropdown-components/type-select/type-select.component.mjs +0 -175
- package/esm2022/lib/modules/dropdown-components/type-text/type-text.component.mjs +0 -62
- package/esm2022/lib/modules/dropdown-components/types.mjs +0 -41
- package/esm2022/lib/modules/dropdown-components/utils.mjs +0 -35
- package/esm2022/lib/modules/file/abstract-file.mjs +0 -230
- package/esm2022/lib/modules/file/component/file.component.mjs +0 -172
- package/esm2022/lib/modules/file/file-drop.directive.mjs +0 -111
- package/esm2022/lib/modules/file/file-select.directive.mjs +0 -26
- package/esm2022/lib/modules/file/file.service.mjs +0 -43
- package/esm2022/lib/modules/file/public-api.mjs +0 -9
- package/esm2022/lib/modules/file/types.mjs +0 -2
- package/esm2022/lib/modules/file/utils.mjs +0 -129
- package/esm2022/lib/modules/fixed-button/fixed-button.component.mjs +0 -30
- package/esm2022/lib/modules/fixed-button/public-api.mjs +0 -5
- package/esm2022/lib/modules/fixed-button-detail/fixed-button-detail.component.mjs +0 -56
- package/esm2022/lib/modules/fixed-button-detail/public-api.mjs +0 -5
- package/esm2022/lib/modules/hierarchic-selector/classes/flat-node.mjs +0 -18
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-configuration.mjs +0 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-filters-configuration.mjs +0 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/model-node.mjs +0 -14
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.mjs +0 -398
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs +0 -243
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component.mjs +0 -38
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service.mjs +0 -22
- package/esm2022/lib/modules/hierarchic-selector/public-api.mjs +0 -10
- package/esm2022/lib/modules/icon/icon.directive.mjs +0 -96
- package/esm2022/lib/modules/icon/icon.module.mjs +0 -33
- package/esm2022/lib/modules/icon/public-api.mjs +0 -6
- package/esm2022/lib/modules/logger/error-handler.mjs +0 -87
- package/esm2022/lib/modules/logger/error.module.mjs +0 -22
- package/esm2022/lib/modules/logger/public-api.mjs +0 -6
- package/esm2022/lib/modules/matomo/matomo.service.mjs +0 -96
- package/esm2022/lib/modules/matomo/public-api.mjs +0 -5
- package/esm2022/lib/modules/panels/abstract-panel.mjs +0 -76
- package/esm2022/lib/modules/panels/fallback-if-no-opened-panels.urlmatcher.mjs +0 -12
- package/esm2022/lib/modules/panels/panels.component.mjs +0 -27
- package/esm2022/lib/modules/panels/panels.module.mjs +0 -10
- package/esm2022/lib/modules/panels/panels.service.mjs +0 -329
- package/esm2022/lib/modules/panels/panels.urlmatcher.mjs +0 -75
- package/esm2022/lib/modules/panels/public-api.mjs +0 -11
- package/esm2022/lib/modules/panels/types.mjs +0 -3
- package/esm2022/lib/modules/relations/public-api.mjs +0 -5
- package/esm2022/lib/modules/relations/relations.component.mjs +0 -254
- package/esm2022/lib/modules/search/classes/graphql-doctrine.mjs +0 -111
- package/esm2022/lib/modules/search/classes/graphql-doctrine.types.mjs +0 -14
- package/esm2022/lib/modules/search/classes/transformers.mjs +0 -142
- package/esm2022/lib/modules/search/classes/url.mjs +0 -53
- package/esm2022/lib/modules/search/classes/utils.mjs +0 -25
- package/esm2022/lib/modules/search/dropdown-container/dropdown-container-animations.mjs +0 -44
- package/esm2022/lib/modules/search/dropdown-container/dropdown-container.component.mjs +0 -87
- package/esm2022/lib/modules/search/dropdown-container/dropdown-ref.mjs +0 -24
- package/esm2022/lib/modules/search/dropdown-container/dropdown.service.mjs +0 -90
- package/esm2022/lib/modules/search/facet-selector/facet-selector.component.mjs +0 -45
- package/esm2022/lib/modules/search/group/group.component.mjs +0 -53
- package/esm2022/lib/modules/search/input/input.component.mjs +0 -365
- package/esm2022/lib/modules/search/public-api.mjs +0 -7
- package/esm2022/lib/modules/search/search/search.component.mjs +0 -102
- package/esm2022/lib/modules/search/types/dropdown-component.mjs +0 -2
- package/esm2022/lib/modules/search/types/facet.mjs +0 -2
- package/esm2022/lib/modules/search/types/values.mjs +0 -2
- package/esm2022/lib/modules/select/abstract-select.component.mjs +0 -232
- package/esm2022/lib/modules/select/public-api.mjs +0 -7
- package/esm2022/lib/modules/select/select/select.component.mjs +0 -310
- package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +0 -57
- package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +0 -155
- package/esm2022/lib/modules/sidenav/public-api.mjs +0 -9
- package/esm2022/lib/modules/sidenav/sidenav/sidenav.component.mjs +0 -15
- package/esm2022/lib/modules/sidenav/sidenav-container/sidenav-container.component.mjs +0 -90
- package/esm2022/lib/modules/sidenav/sidenav-content/sidenav-content.component.mjs +0 -11
- package/esm2022/lib/modules/sidenav/sidenav-stack.service.mjs +0 -50
- package/esm2022/lib/modules/sidenav/sidenav.service.mjs +0 -196
- package/esm2022/lib/modules/stamp/public-api.mjs +0 -5
- package/esm2022/lib/modules/stamp/stamp.component.mjs +0 -23
- package/esm2022/lib/modules/table-button/public-api.mjs +0 -5
- package/esm2022/lib/modules/table-button/table-button.component.mjs +0 -78
- package/esm2022/lib/services/abstract-model.service.mjs +0 -526
- package/esm2022/lib/services/debounce.service.mjs +0 -149
- package/esm2022/lib/services/enum.service.mjs +0 -64
- package/esm2022/lib/services/link-mutation.service.mjs +0 -154
- package/esm2022/lib/services/persistence.service.mjs +0 -115
- package/esm2022/lib/services/swiss-parsing-date-adapter.service.mjs +0 -63
- package/esm2022/lib/types/types.mjs +0 -2
- 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
|