@firestitch/app-acl 12.4.12 → 12.5.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 (61) hide show
  1. package/app/components/acl-entries/acl-entries.component.d.ts +33 -33
  2. package/app/components/acl-entry/acl-entry.component.d.ts +31 -31
  3. package/app/components/acl-object-roles/acl-object-roles.component.d.ts +18 -18
  4. package/app/components/acl-permission-popover/acl-permission-popover.component.d.ts +12 -12
  5. package/app/components/acl-role/acl-role.component.d.ts +53 -53
  6. package/app/components/acl-role-popover/acl-role-popover.component.d.ts +13 -13
  7. package/app/components/acl-roles/acl-roles.component.d.ts +38 -38
  8. package/app/consts/acl-role-accesses.d.ts +5 -5
  9. package/app/enums/acl-role-access.d.ts +6 -6
  10. package/app/fs-app-acl.module.d.ts +32 -33
  11. package/app/injectors/app-acl-config.injector.d.ts +2 -2
  12. package/app/interfaces/acl-entry-data.d.ts +11 -11
  13. package/app/interfaces/acl-entry.d.ts +18 -18
  14. package/app/interfaces/acl-level.d.ts +3 -3
  15. package/app/interfaces/acl-object-entry.d.ts +11 -11
  16. package/app/interfaces/acl-object-role.d.ts +6 -6
  17. package/app/interfaces/acl-object.d.ts +4 -4
  18. package/app/interfaces/acl-permission.d.ts +7 -7
  19. package/app/interfaces/acl-role-config.d.ts +5 -5
  20. package/app/interfaces/acl-role.d.ts +16 -16
  21. package/app/interfaces/app-acl-config.d.ts +7 -7
  22. package/app/interfaces/index.d.ts +2 -2
  23. package/app/interfaces/name-value.d.ts +4 -4
  24. package/app/interfaces/role-config.d.ts +12 -12
  25. package/app/pipes/bulk-options-filter.pipe.d.ts +13 -13
  26. package/app/services/app-acl.service.d.ts +16 -16
  27. package/bundles/firestitch-app-acl.umd.js +1377 -1381
  28. package/bundles/firestitch-app-acl.umd.js.map +1 -1
  29. package/esm2015/app/components/acl-entries/acl-entries.component.js +173 -174
  30. package/esm2015/app/components/acl-entry/acl-entry.component.js +93 -93
  31. package/esm2015/app/components/acl-object-roles/acl-object-roles.component.js +56 -56
  32. package/esm2015/app/components/acl-permission-popover/acl-permission-popover.component.js +32 -32
  33. package/esm2015/app/components/acl-role/acl-role.component.js +232 -233
  34. package/esm2015/app/components/acl-role-popover/acl-role-popover.component.js +37 -37
  35. package/esm2015/app/components/acl-roles/acl-roles.component.js +174 -174
  36. package/esm2015/app/consts/acl-role-accesses.js +7 -7
  37. package/esm2015/app/enums/acl-role-access.js +7 -7
  38. package/esm2015/app/fs-app-acl.module.js +127 -131
  39. package/esm2015/app/injectors/app-acl-config.injector.js +2 -2
  40. package/esm2015/app/interfaces/acl-entry-data.js +1 -1
  41. package/esm2015/app/interfaces/acl-entry.js +1 -1
  42. package/esm2015/app/interfaces/acl-level.js +1 -1
  43. package/esm2015/app/interfaces/acl-object-entry.js +1 -1
  44. package/esm2015/app/interfaces/acl-object-role.js +1 -1
  45. package/esm2015/app/interfaces/acl-object.js +1 -1
  46. package/esm2015/app/interfaces/acl-permission.js +1 -1
  47. package/esm2015/app/interfaces/acl-role-config.js +1 -1
  48. package/esm2015/app/interfaces/acl-role.js +1 -1
  49. package/esm2015/app/interfaces/app-acl-config.js +1 -1
  50. package/esm2015/app/interfaces/index.js +2 -2
  51. package/esm2015/app/interfaces/name-value.js +1 -1
  52. package/esm2015/app/interfaces/role-config.js +1 -1
  53. package/esm2015/app/pipes/bulk-options-filter.pipe.js +26 -26
  54. package/esm2015/app/services/app-acl.service.js +50 -50
  55. package/esm2015/firestitch-app-acl.js +4 -4
  56. package/esm2015/public_api.js +10 -10
  57. package/fesm2015/firestitch-app-acl.js +864 -869
  58. package/fesm2015/firestitch-app-acl.js.map +1 -1
  59. package/firestitch-app-acl.d.ts +5 -5
  60. package/package.json +1 -1
  61. package/public_api.d.ts +20 -20
@@ -1,176 +1,175 @@
1
- import { takeUntil } from 'rxjs/operators';
2
- import { Component, ViewChild, Input } from '@angular/core';
3
- import { MatDialog } from '@angular/material/dialog';
4
- import { Subject, Observable } from 'rxjs';
5
- import { sortBy, groupBy } from 'lodash-es';
6
- import { FsListComponent } from '@firestitch/list';
7
- import { FsPrompt } from '@firestitch/prompt';
8
- import { FsAppAclService } from '../../services/app-acl.service';
9
- import { FsAclEntryComponent } from '../acl-entry/acl-entry.component';
10
- import * as i0 from "@angular/core";
11
- import * as i1 from "../../services/app-acl.service";
12
- import * as i2 from "@angular/material/dialog";
13
- import * as i3 from "@firestitch/prompt";
14
- import * as i4 from "@firestitch/list";
15
- import * as i5 from "@firestitch/badge";
16
- import * as i6 from "../acl-role-popover/acl-role-popover.component";
17
- import * as i7 from "@angular/common";
18
- import * as i8 from "@angular/flex-layout/flex";
19
- export class FsAclEntriesComponent {
20
- constructor(_appAclService, _dialog, _confirm) {
21
- this._appAclService = _appAclService;
22
- this._dialog = _dialog;
23
- this._confirm = _confirm;
24
- this.environmentShow = true;
25
- this.environmentLabel = 'Environment';
26
- this.environmentKey = 'environment';
27
- this.actions = [];
28
- this.aclEntriesList = null;
29
- this.aclEntriesConfig = null;
30
- this.permissions = [];
31
- this._destroy$ = new Subject();
32
- }
33
- ngOnInit() {
34
- this._appAclService.getPermissions()
35
- .subscribe(response => {
36
- this.permissions = response;
37
- });
38
- this.aclEntriesConfig = {
39
- status: false,
40
- paging: false,
41
- actions: this.actions,
42
- rowActions: [
43
- {
44
- label: 'Remove All Roles',
45
- click: (aclObjectEntry) => {
46
- this._confirm
47
- .confirm({
48
- title: 'Remove All Roles',
49
- commitLabel: 'Save',
1
+ import { takeUntil } from 'rxjs/operators';
2
+ import { Component, ViewChild, Input } from '@angular/core';
3
+ import { MatDialog } from '@angular/material/dialog';
4
+ import { Subject, Observable } from 'rxjs';
5
+ import { sortBy, groupBy } from 'lodash-es';
6
+ import { FsListComponent } from '@firestitch/list';
7
+ import { FsPrompt } from '@firestitch/prompt';
8
+ import { FsAppAclService } from '../../services/app-acl.service';
9
+ import { FsAclEntryComponent } from '../acl-entry/acl-entry.component';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "../../services/app-acl.service";
12
+ import * as i2 from "@angular/material/dialog";
13
+ import * as i3 from "@firestitch/prompt";
14
+ import * as i4 from "@firestitch/list";
15
+ import * as i5 from "@firestitch/badge";
16
+ import * as i6 from "../acl-role-popover/acl-role-popover.component";
17
+ import * as i7 from "@angular/common";
18
+ export class FsAclEntriesComponent {
19
+ constructor(_appAclService, _dialog, _confirm) {
20
+ this._appAclService = _appAclService;
21
+ this._dialog = _dialog;
22
+ this._confirm = _confirm;
23
+ this.environmentShow = true;
24
+ this.environmentLabel = 'Environment';
25
+ this.environmentKey = 'environment';
26
+ this.actions = [];
27
+ this.aclEntriesList = null;
28
+ this.aclEntriesConfig = null;
29
+ this.permissions = [];
30
+ this._destroy$ = new Subject();
31
+ }
32
+ ngOnInit() {
33
+ this._appAclService.getPermissions()
34
+ .subscribe(response => {
35
+ this.permissions = response;
36
+ });
37
+ this.aclEntriesConfig = {
38
+ status: false,
39
+ paging: false,
40
+ actions: this.actions,
41
+ rowActions: [
42
+ {
43
+ label: 'Remove All Roles',
44
+ click: (aclObjectEntry) => {
45
+ this._confirm
46
+ .confirm({
47
+ title: 'Remove All Roles',
48
+ commitLabel: 'Save',
50
49
  template: `Please note that removing roles may prevent users from being able to successfully login.<br>
51
50
  These changes are effective immediately.<br>
52
- Are you sure you would like to continue?`,
53
- }).subscribe(() => {
54
- const data = Object.assign(Object.assign({}, aclObjectEntry), { aclEntries: [] });
55
- this.saveAclObjectEntry(data)
56
- .subscribe(() => {
57
- this.aclEntriesList.reload();
58
- });
59
- });
60
- }
61
- }
62
- ],
63
- fetch: () => {
64
- return new Observable((observer) => {
65
- this.loadAclEntries({
66
- aclRoles: true,
67
- aclRolePermissions: true,
68
- objects: true,
69
- aclRoleState: 'active',
70
- })
71
- .subscribe((aclEntries) => {
72
- const objects = aclEntries
73
- .filter((aclEntry) => (!!aclEntry.object))
74
- .reduce((items, item) => {
75
- return Object.assign(Object.assign({}, items), { [item.object.id]: item.object });
76
- }, {});
77
- const environments = aclEntries
78
- .filter((aclEntry) => (!!aclEntry[this.environmentKey]))
79
- .reduce((items, item) => {
80
- const environment = item[this.environmentKey];
81
- return Object.assign(Object.assign({}, items), { [environment.id]: environment });
82
- }, {});
83
- const groupedAclEntries = groupBy(aclEntries, (item) => {
84
- var _a;
85
- const environmentId = (_a = (item[this.environmentKey])) === null || _a === void 0 ? void 0 : _a.id;
86
- return [item.aclRole.level, environmentId, item.objectId];
87
- });
88
- let aclObjectEntries = Object.keys(groupedAclEntries)
89
- .reduce((accum, key) => {
90
- const parts = key.split(',');
91
- return [
92
- ...accum,
93
- {
94
- object: objects[parts[2]],
95
- level: parts[0],
96
- [`${this.environmentKey}Id`]: parts[1] ? parseInt(parts[1]) : null,
97
- [this.environmentKey]: environments[parts[1]],
98
- aclEntries: groupedAclEntries[key],
99
- }
100
- ];
101
- }, []);
102
- const hasApp = aclObjectEntries.some((item) => {
103
- return item.aclEntries.some((entry) => {
104
- return !entry.objectId;
105
- });
106
- });
107
- if (!hasApp) {
108
- aclObjectEntries.unshift({
109
- object: null,
110
- aclEntries: [],
111
- level: 'app',
112
- environmentId: null,
113
- });
114
- }
115
- aclObjectEntries = sortBy(aclObjectEntries, (item) => {
116
- return item.object ? item.level : '';
117
- });
118
- observer.next({ data: aclObjectEntries });
119
- observer.complete();
120
- });
121
- });
122
- },
123
- };
124
- }
125
- update(aclObjectEntry) {
126
- const data = {
127
- aclObjectEntry,
128
- required: false,
129
- loadAclRoles: this.loadAclRoles,
130
- saveAclObjectEntry: this.saveAclObjectEntry
131
- };
132
- this._dialog.open(FsAclEntryComponent, {
133
- data: data
134
- })
135
- .afterClosed()
136
- .pipe(takeUntil(this._destroy$))
137
- .subscribe(() => {
138
- this.aclEntriesList.reload();
139
- });
140
- }
141
- ngOnDestroy() {
142
- this._destroy$.next();
143
- this._destroy$.complete();
144
- }
145
- reload() {
146
- this.aclEntriesList.reload();
147
- }
148
- }
149
- FsAclEntriesComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: FsAclEntriesComponent, deps: [{ token: i1.FsAppAclService }, { token: i2.MatDialog }, { token: i3.FsPrompt }], target: i0.ɵɵFactoryTarget.Component });
150
- FsAclEntriesComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.17", type: FsAclEntriesComponent, selector: "fs-acl-entries", inputs: { loadAclEntries: "loadAclEntries", loadAclRoles: "loadAclRoles", saveAclObjectEntry: "saveAclObjectEntry", environmentShow: "environmentShow", environmentLabel: "environmentLabel", environmentKey: "environmentKey", actions: "actions" }, viewQueries: [{ propertyName: "aclEntriesList", first: true, predicate: FsListComponent, descendants: true }], ngImport: i0, template: "\n<fs-list [config]=\"aclEntriesConfig\">\n <fs-list-column>\n <ng-template fs-list-header>Context</ng-template>\n <ng-template fs-list-cell let-row=\"row\">\n <div\n *ngIf=\"row.object; else elseObject\"\n fxLayout=\"row\"\n fxLayoutAlign=\"start center\"\n fxLayoutGap=\"10px\">\n <fs-badge *ngIf=\"row.object.imageUrl\" shape=\"circle\" image=\"{{ row.object.imageUrl }}\"></fs-badge>\n <span>\n <div><small>{{ row.object.className }}</small></div>\n <a (click)=\"update(row)\">{{ row.object.name }}</a>\n </span>\n </div>\n\n <ng-template #elseObject>\n <a (click)=\"update(row)\">App</a>\n </ng-template>\n </ng-template>\n </fs-list-column>\n\n <fs-list-column [show]=\"environmentShow\">\n <ng-template fs-list-header>{{environmentLabel}}</ng-template>\n <ng-template fs-list-cell let-row=\"row\">\n {{row.environment?.name}}\n </ng-template>\n </fs-list-column>\n\n <fs-list-column>\n <ng-template fs-list-header>Roles</ng-template>\n <ng-template fs-list-cell let-row=\"row\">\n <div fxLayout=\"column\" fxLayoutGap=\"10px\">\n <div div *ngFor=\"let aclEntry of row.aclEntries\">\n <ng-container [ngSwitch]=\"row.level\">\n <ng-container *ngSwitchCase=\"'app'\">\n <fs-acl-role-popover [aclRole]=\"aclEntry.aclRole\" objectName=\"App\">{{aclEntry.aclRole.name}}</fs-acl-role-popover>\n </ng-container>\n <ng-container *ngSwitchDefault>\n <ng-container *ngIf=\"row.object\">\n <fs-acl-role-popover [aclRole]=\"aclEntry.aclRole\">{{aclEntry.aclRole.name}}</fs-acl-role-popover>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </div>\n </ng-template>\n </fs-list-column>\n</fs-list>\n", styles: [""], components: [{ type: i4.FsListComponent, selector: "fs-list", inputs: ["config", "loaderLines"], outputs: ["filtersReady"] }, { type: i5.FsBadgeComponent, selector: "fs-badge", inputs: ["color", "text", "tooltip", "size", "shape", "image", "icon", "iconSize", "iconColor", "backgroundSize"] }, { type: i6.FsAclRolePopoverComponent, selector: "fs-acl-role-popover", inputs: ["aclRole", "objectName"] }], directives: [{ type: i4.FsListColumnDirective, selector: "fs-list-column", inputs: ["show", "title", "name", "customize", "sortable", "sortableDefault", "direction", "align", "width", "class"] }, { type: i4.FsListHeaderDirective, selector: "[fs-list-header]", inputs: ["colspan", "align", "class"] }, { type: i4.FsListCellDirective, selector: "[fs-list-cell]", inputs: ["colspan", "align", "class"] }, { type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { type: i8.DefaultLayoutAlignDirective, selector: " [fxLayoutAlign], [fxLayoutAlign.xs], [fxLayoutAlign.sm], [fxLayoutAlign.md], [fxLayoutAlign.lg], [fxLayoutAlign.xl], [fxLayoutAlign.lt-sm], [fxLayoutAlign.lt-md], [fxLayoutAlign.lt-lg], [fxLayoutAlign.lt-xl], [fxLayoutAlign.gt-xs], [fxLayoutAlign.gt-sm], [fxLayoutAlign.gt-md], [fxLayoutAlign.gt-lg]", inputs: ["fxLayoutAlign", "fxLayoutAlign.xs", "fxLayoutAlign.sm", "fxLayoutAlign.md", "fxLayoutAlign.lg", "fxLayoutAlign.xl", "fxLayoutAlign.lt-sm", "fxLayoutAlign.lt-md", "fxLayoutAlign.lt-lg", "fxLayoutAlign.lt-xl", "fxLayoutAlign.gt-xs", "fxLayoutAlign.gt-sm", "fxLayoutAlign.gt-md", "fxLayoutAlign.gt-lg"] }, { type: i8.DefaultLayoutGapDirective, selector: " [fxLayoutGap], [fxLayoutGap.xs], [fxLayoutGap.sm], [fxLayoutGap.md], [fxLayoutGap.lg], [fxLayoutGap.xl], [fxLayoutGap.lt-sm], [fxLayoutGap.lt-md], [fxLayoutGap.lt-lg], [fxLayoutGap.lt-xl], [fxLayoutGap.gt-xs], [fxLayoutGap.gt-sm], [fxLayoutGap.gt-md], [fxLayoutGap.gt-lg]", inputs: ["fxLayoutGap", "fxLayoutGap.xs", "fxLayoutGap.sm", "fxLayoutGap.md", "fxLayoutGap.lg", "fxLayoutGap.xl", "fxLayoutGap.lt-sm", "fxLayoutGap.lt-md", "fxLayoutGap.lt-lg", "fxLayoutGap.lt-xl", "fxLayoutGap.gt-xs", "fxLayoutGap.gt-sm", "fxLayoutGap.gt-md", "fxLayoutGap.gt-lg"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i7.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { type: i7.NgSwitchDefault, selector: "[ngSwitchDefault]" }] });
151
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: FsAclEntriesComponent, decorators: [{
152
- type: Component,
153
- args: [{
154
- selector: 'fs-acl-entries',
155
- templateUrl: './acl-entries.component.html',
156
- styleUrls: ['./acl-entries.component.scss']
157
- }]
158
- }], ctorParameters: function () { return [{ type: i1.FsAppAclService }, { type: i2.MatDialog }, { type: i3.FsPrompt }]; }, propDecorators: { loadAclEntries: [{
159
- type: Input
160
- }], loadAclRoles: [{
161
- type: Input
162
- }], saveAclObjectEntry: [{
163
- type: Input
164
- }], environmentShow: [{
165
- type: Input
166
- }], environmentLabel: [{
167
- type: Input
168
- }], environmentKey: [{
169
- type: Input
170
- }], actions: [{
171
- type: Input
172
- }], aclEntriesList: [{
173
- type: ViewChild,
174
- args: [FsListComponent]
175
- }] } });
176
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWNsLWVudHJpZXMuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2FwcC9jb21wb25lbnRzL2FjbC1lbnRyaWVzL2FjbC1lbnRyaWVzLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9hcHAvY29tcG9uZW50cy9hY2wtZW50cmllcy9hY2wtZW50cmllcy5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFPLE1BQU0sZ0JBQWdCLENBQUM7QUFDaEQsT0FBTyxFQUFFLFNBQVMsRUFBVSxTQUFTLEVBQWEsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQy9FLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUVyRCxPQUFPLEVBQUUsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUUzQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE9BQU8sRUFBRSxNQUFNLFdBQVcsQ0FBQztBQUU1QyxPQUFPLEVBQWdCLGVBQWUsRUFBZ0IsTUFBTSxrQkFBa0IsQ0FBQztBQUMvRSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFOUMsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBS2pFLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGtDQUFrQyxDQUFDOzs7Ozs7Ozs7O0FBUXZFLE1BQU0sT0FBTyxxQkFBcUI7SUFrQmhDLFlBQ21CLGNBQStCLEVBQy9CLE9BQWtCLEVBQzNCLFFBQWtCO1FBRlQsbUJBQWMsR0FBZCxjQUFjLENBQWlCO1FBQy9CLFlBQU8sR0FBUCxPQUFPLENBQVc7UUFDM0IsYUFBUSxHQUFSLFFBQVEsQ0FBVTtRQWhCWixvQkFBZSxHQUFHLElBQUksQ0FBQztRQUN2QixxQkFBZ0IsR0FBRyxhQUFhLENBQUM7UUFDakMsbUJBQWMsR0FBRyxhQUFh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SUFFTSxXQUFXO1FBQ2hCLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDdEIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxRQUFRLEVBQUUsQ0FBQztJQUM1QixDQUFDO0lBRU0sTUFBTTtRQUNYLElBQUksQ0FBQyxjQUFjLENBQUMsTUFBTSxFQUFFLENBQUM7SUFDL0IsQ0FBQzs7bUhBN0pVLHFCQUFxQjt1R0FBckIscUJBQXFCLDRWQVVyQixlQUFlLGdEQ2xDNUIsaTJEQWtEQTs0RkQxQmEscUJBQXFCO2tCQUxqQyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSxnQkFBZ0I7b0JBQzFCLFdBQVcsRUFBRSw4QkFBOEI7b0JBQzNDLFNBQVMsRUFBRSxDQUFDLDhCQUE4QixDQUFDO2lCQUM1QztxSkFHaUIsY0FBYztzQkFBN0IsS0FBSztnQkFDVSxZQUFZO3NCQUEzQixLQUFLO2dCQUNVLGtCQUFrQjtzQkFBakMsS0FBSztnQkFDVSxlQUFlO3NCQUE5QixLQUFLO2dCQUNVLGdCQUFnQjtzQkFBL0IsS0FBSztnQkFDVSxjQUFjO3NCQUE3QixLQUFLO2dCQUNVLE9BQU87c0JBQXRCLEtBQUs7Z0JBR0MsY0FBYztzQkFEcEIsU0FBUzt1QkFBQyxlQUFlIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgdGFrZVVudGlsLCBtYXAgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5pbXBvcnQgeyBDb21wb25lbnQsIE9uSW5pdCwgVmlld0NoaWxkLCBPbkRlc3Ryb3ksIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBNYXREaWFsb2cgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9kaWFsb2cnO1xuXG5pbXBvcnQgeyBTdWJqZWN0LCBPYnNlcnZhYmxlIH0gZnJvbSAncnhqcyc7XG5cbmltcG9ydCB7IHNvcnRCeSwgZ3JvdXBCeSB9IGZyb20gJ2xvZGFzaC1lcyc7XG5cbmltcG9ydCB7IEZzTGlzdEFjdGlvbiwgRnNMaXN0Q29tcG9uZW50LCBGc0xpc3RDb25maWcgfSBmcm9tICdAZmlyZXN0aXRjaC9saXN0JztcbmltcG9ydCB7IEZzUHJvbXB0IH0gZnJvbSAnQGZpcmVzdGl0Y2gvcHJvbXB0JztcblxuaW1wb3J0IHsgRnNBcHBBY2xTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvYXBwLWFjbC5zZXJ2aWNlJztcbmltcG9ydCB7IEFjbEVudHJ5RGF0YSB9IGZyb20gJy4uLy4uL2ludGVyZmFjZXMvYWNsLWVudHJ5LWRhdGEnO1xuaW1wb3J0IHsgQWNsRW50cnkgfSBmcm9tICcuLi8uLi9pbnRlcmZhY2VzL2FjbC1lbnRyeSc7XG5pbXBvcnQgeyBBY2xSb2xlIH0gZnJvbSAnLi4vLi4vaW50ZXJmYWNlcy9hY2wtcm9sZSc7XG5pbXBvcnQgeyBBY2xPYmplY3RFbnRyeSB9IGZyb20gJy4uLy4uL2ludGVyZmFjZXMvYWNsLW9iamVjdC1lbnRyeSc7XG5pbXBvcnQgeyBGc0FjbEVudHJ5Q29tcG9uZW50IH0gZnJvbSAnLi4vYWNsLWVudHJ5L2FjbC1lbnRyeS5jb21wb25lbnQnO1xuXG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2ZzLWFjbC1lbnRyaWVzJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2FjbC1lbnRyaWVzLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vYWNsLWVudHJpZXMuY29tcG9uZW50LnNjc3MnXVxufSlcbmV4cG9ydCBjbGFzcyBGc0FjbEVudHJpZXNDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQsIE9uRGVzdHJveSB7XG5cbiAgQElucHV0KCkgcHVibGljIGxvYWRBY2xFbnRyaWVzOiAocXVlcnk6IGFueSkgPT4gT2JzZXJ2YWJsZTxBY2xFbnRyeVtdPjtcbiAgQElucHV0KCkgcHVibGljIGxvYWRBY2xSb2xlczogKHF1ZXJ5OiBhbnkpID0+IE9ic2VydmFibGU8QWNsUm9sZVtdPjtcbiAgQElucHV0KCkgcHVibGljIHNhdmVBY2xPYmplY3RFbnRyeTogKGFjbE9iamVjdEVudHJ5OiBBY2xPYmplY3RFbnRyeSkgPT4gT2JzZXJ2YWJsZTxhbnk+O1xuICBASW5wdXQoKSBwdWJsaWMgZW52aXJvbm1lbnRTaG93ID0gdHJ1ZTtcbiAgQElucHV0KCkgcHVibGljIGVudmlyb25tZW50TGFiZWwgPSAnRW52aXJvbm1lbnQnO1xuICBASW5wdXQoKSBwdWJsaWMgZW52aXJvbm1lbnRLZXkgPSAnZW52aXJvbm1lbnQnO1xuICBASW5wdXQoKSBwdWJsaWMgYWN0aW9uczogRnNMaXN0QWN0aW9uW10gPSBbXTtcblxuICBAVmlld0NoaWxkKEZzTGlzdENvbXBvbmVudClcbiAgcHVibGljIGFjbEVudHJpZXNMaXN0OiBGc0xpc3RDb21wb25lbnQgPSBudWxsO1xuXG4gIHB1YmxpYyBhY2xFbnRyaWVzQ29uZmlnOiBGc0xpc3RDb25maWcgPSBudWxsO1xuICBwdWJsaWMgcGVybWlzc2lvbnM6IGFueVtdID0gW107XG5cbiAgcHJpdmF0ZSBfZGVzdHJveSQgPSBuZXcgU3ViamVjdCgpO1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgcmVhZG9ubHkgX2FwcEFjbFNlcnZpY2U6IEZzQXBwQWNsU2VydmljZSxcbiAgICBwcml2YXRlIHJlYWRvbmx5IF9kaWFsb2c6IE1hdERpYWxvZyxcbiAgICBwcml2YXRlIF9jb25maXJtOiBGc1Byb21wdCxcbiAgKSB7IH1cblxuICBwdWJsaWMgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgdGhpcy5fYXBwQWNsU2VydmljZS5nZXRQZXJtaXNzaW9ucygpXG4gICAgICAuc3Vic2NyaWJlKHJlc3BvbnNlID0+IHtcbiAgICAgICAgdGhpcy5wZXJtaXNzaW9ucyA9IHJlc3BvbnNlO1xuICAgICAgfSk7XG5cbiAgICB0aGlzLmFjbEVudHJpZXNDb25maWcgPSB7XG4gICAgICBzdGF0dXM6IGZhbHNlLFxuICAgICAgcGFnaW5nOiBmYWxzZSxcbiAgICAgIGFjdGlvbnM6IHRoaXMuYWN0aW9ucyxcbiAgICAgIHJvd0FjdGlvbnM6IFtcbiAgICAgICAge1xuICAgICAgICAgIGxhYmVsOiAnUmVtb3ZlIEFsbCBSb2xlcycsXG4gICAgICAgICAgY2xpY2s6IChhY2xPYmplY3RFbnRyeTogQWNsT2JqZWN0RW50cnkpID0+IHtcbiAgICAgICAgICAgIHRoaXMuX2NvbmZpcm1cbiAgICAgICAgICAgICAgLmNvbmZpcm0oe1xuICAgICAgICAgICAgICAgIHRpdGxlOiAnUmVtb3ZlIEFsbCBSb2xlcycsXG4gICAgICAgICAgICAgICAgY29tbWl0TGFiZWw6ICdTYXZlJyxcbiAgICAgICAgICAgICAgICB0ZW1wbGF0ZTogYFBsZWFzZSBub3RlIHRoYXQgcmVtb3Zpbmcgcm9sZXMgbWF5IHByZXZlbnQgdXNlcnMgZnJvbSBiZWluZyBhYmxlIHRvIHN1Y2Nlc3NmdWxseSBsb2dpbi48YnI+XG4gICAgICAgICAgICAgICAgICBUaGVzZSBjaGFuZ2VzIGFyZSBlZmZlY3RpdmUgaW1tZWRpYXRlbHkuPGJyPlxuICAgICAgICAgICAgICAgICAgQXJlIHlvdSBzdXJlIHlvdSB3b3VsZCBsaWtlIHRvIGNvbnRpbnVlP2AsXG4gICAgICAgICAgICAgIH0pLnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgICAgICAgICAgY29uc3QgZGF0YSA9IHsgLi4uYWNsT2JqZWN0RW50cnksIGFjbEVudHJpZXM6IFtdIH07XG4gICAgICAgICAgICAgICAgdGhpcy5zYXZlQWNsT2JqZWN0RW50cnkoZGF0YSlcbiAgICAgICAgICAgICAgICAgIC5zdWJzY3JpYmUoKCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICB0aGlzLmFjbEVudHJpZXNMaXN0LnJlbG9hZCgpO1xuICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgXSxcbiAgICAgIGZldGNoOiAoKSA9PiB7XG4gICAgICAgIHJldHVybiBuZXcgT2JzZXJ2YWJsZSgob2JzZXJ2ZXIpID0+IHtcbiAgICAgICAgICB0aGlzLmxvYWRBY2xFbnRyaWVzKHtcbiAgICAgICAgICAgIGFjbFJvbGVzOiB0cnVlLFxuICAgICAgICAgICAgYWNsUm9sZVBlcm1pc3Npb25zOiB0cnVlLFxuICAgICAgICAgICAgb2JqZWN0czogdHJ1ZSxcbiAgICAgICAgICAgIGFjbFJvbGVTdGF0ZTogJ2FjdGl2ZScsXG4gICAgICAgICAgfSlcbiAgICAgICAgICAgIC5zdWJzY3JpYmUoKGFjbEVudHJpZXM6IEFjbEVudHJ5W10pID0+IHtcbiAgICAgICAgICAgICAgY29uc3Qgb2JqZWN0cyA9IGFjbEVudHJpZXNcbiAgICAgICAgICAgICAgICAuZmlsdGVyKChhY2xFbnRyeSkgPT4gKCEhYWNsRW50cnkub2JqZWN0KSlcbiAgICAgICAgICAgICAgICAucmVkdWNlKChpdGVtcywgaXRlbSkgPT4ge1xuICAgICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgICAgLi4uaXRlbXMsXG4gICAgICAgICAgICAgICAgICAgIFtpdGVtLm9iamVjdC5pZF06IGl0ZW0ub2JqZWN0LFxuICAgICAgICAgICAgICAgICAgfTtcbiAgICAgICAgICAgICAgICB9LCB7fSk7XG5cbiAgICAgICAgICAgICAgY29uc3QgZW52aXJvbm1lbnRzID0gYWNsRW50cmllc1xuICAgICAgICAgICAgICAgIC5maWx0ZXIoKGFjbEVudHJ5KSA9PiAoISFhY2xFbnRyeVt0aGlzLmVudmlyb25tZW50S2V5XSkpXG4gICAgICAgICAgICAgICAgLnJlZHVjZSgoaXRlbXMsIGl0ZW0pID0+IHtcbiAgICAgICAgICAgICAgICAgIGNvbnN0IGVudmlyb25tZW50ID0gaXRlbVt0aGlzLmVudmlyb25tZW50S2V5XTtcbiAgICAgICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgICAgIC4uLml0ZW1zLFxuICAgICAgICAgICAgICAgICAgICBbZW52aXJvbm1lbnQuaWRdOiBlbnZpcm9ubWVudCxcbiAgICAgICAgICAgICAgICAgIH07XG4gICAgICAgICAgICAgICAgfSwge30pO1xuXG4gICAgICAgICAgICAgIGNvbnN0IGdyb3VwZWRBY2xFbnRyaWVzID0gZ3JvdXBCeShhY2xFbnRyaWVzLCAoaXRlbSkgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IGVudmlyb25tZW50SWQgPSAoaXRlbVt0aGlzLmVudmlyb25tZW50S2V5XSk/LmlkO1xuICAgICAgICAgICAgICAgIHJldHVybiBbaXRlbS5hY2xSb2xlLmxldmVsLCBlbnZpcm9ubWVudElkLCBpdGVtLm9iamVjdElkXTtcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgbGV0IGFjbE9iamVjdEVudHJpZXM6IEFjbE9iamVjdEVudHJ5W10gPSBPYmplY3Qua2V5cyhncm91cGVkQWNsRW50cmllcylcbiAgICAgICAgICAgICAgICAucmVkdWNlKChhY2N1bSwga2V5OiBhbnkpID0+IHtcbiAgICAgICAgICAgICAgICAgIGNvbnN0IHBhcnRzID0ga2V5LnNwbGl0KCcsJyk7XG4gICAgICAgICAgICAgICAgICByZXR1cm4gW1xuICAgICAgICAgICAgICAgICAgICAuLi5hY2N1bSxcbiAgICAgICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICAgIG9iamVjdDogb2JqZWN0c1twYXJ0c1syXV0sXG4gICAgICAgICAgICAgICAgICAgICAgbGV2ZWw6IHBhcnRzWzBdLFxuICAgICAgICAgICAgICAgICAgICAgIFtgJHt0aGlzLmVudmlyb25tZW50S2V5fUlkYF06IHBhcnRzWzFdID8gcGFyc2VJbnQocGFydHNbMV0pIDogbnVsbCxcbiAgICAgICAgICAgICAgICAgICAgICBbdGhpcy5lbnZpcm9ubWVudEtleV06IGVudmlyb25tZW50c1twYXJ0c1sxXV0sXG4gICAgICAgICAgICAgICAgICAgICAgYWNsRW50cmllczogZ3JvdXBlZEFjbEVudHJpZXNba2V5XSxcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgXTtcbiAgICAgICAgICAgICAgICB9LCBbXSk7XG5cbiAgICAgICAgICAgICAgY29uc3QgaGFzQXBwID0gYWNsT2JqZWN0RW50cmllcy5zb21lKChpdGVtKSA9PiB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIGl0ZW0uYWNsRW50cmllcy5zb21lKChlbnRyeSkgPT4ge1xuICAgICAgICAgICAgICAgICAgcmV0dXJuICFlbnRyeS5vYmplY3RJZDtcbiAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgaWYgKCFoYXNBcHApIHtcbiAgICAgICAgICAgICAgICBhY2xPYmplY3RFbnRyaWVzLnVuc2hpZnQoe1xuICAgICAgICAgICAgICAgICAgb2JqZWN0OiBudWxsLFxuICAgICAgICAgICAgICAgICAgYWNsRW50cmllczogW10sXG4gICAgICAgICAgICAgICAgICBsZXZlbDogJ2FwcCcsXG4gICAgICAgICAgICAgICAgICBlbnZpcm9ubWVudElkOiBudWxsLFxuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgYWNsT2JqZWN0RW50cmllcyA9IHNvcnRCeShhY2xPYmplY3RFbnRyaWVzLCAoaXRlbTogQWNsT2JqZWN0RW50cnkpID0+IHtcbiAgICAgICAgICAgICAgICByZXR1cm4gaXRlbS5vYmplY3QgPyBpdGVtLmxldmVsIDogJyc7XG4gICAgICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgICAgIG9ic2VydmVyLm5leHQoeyBkYXRhOiBhY2xPYmplY3RFbnRyaWVzIH0pO1xuICAgICAgICAgICAgICBvYnNlcnZlci5jb21wbGV0ZSgpO1xuICAgICAgICAgICAgfSk7XG4gICAgICAgIH0pO1xuICAgICAgfSxcbiAgICB9O1xuICB9XG5cbiAgcHVibGljIHVwZGF0ZShhY2xPYmplY3RFbnRyeTogQWNsT2JqZWN0RW50cnkpIHtcbiAgICBjb25zdCBkYXRhOiBBY2xFbnRyeURhdGEgPSB7XG4gICAgICBhY2xPYmplY3RFbnRyeSxcbiAgICAgIHJlcXVpcmVkOiBmYWxzZSxcbiAgICAgIGxvYWRBY2xSb2xlczogdGhpcy5sb2FkQWNsUm9sZXMsXG4gICAgICBzYXZlQWNsT2JqZWN0RW50cnk6IHRoaXMuc2F2ZUFjbE9iamVjdEVudHJ5XG4gICAgfVxuXG4gICAgdGhpcy5fZGlhbG9nLm9wZW4oRnNBY2xFbnRyeUNvbXBvbmVudCwge1xuICAgICAgZGF0YTogZGF0YVxuICAgIH0pXG4gICAgICAuYWZ0ZXJDbG9zZWQoKVxuICAgICAgLnBpcGUoXG4gICAgICAgIHRha2VVbnRpbCh0aGlzLl9kZXN0cm95JCksXG4gICAgICApXG4gICAgICAuc3Vic2NyaWJlKCgpID0+IHtcbiAgICAgICAgdGhpcy5hY2xFbnRyaWVzTGlzdC5yZWxvYWQoKTtcbiAgICAgIH0pO1xuICB9XG5cbiAgcHVibGljIG5nT25EZXN0cm95KCkge1xuICAgIHRoaXMuX2Rlc3Ryb3kkLm5leHQoKTtcbiAgICB0aGlzLl9kZXN0cm95JC5jb21wbGV0ZSgpO1xuICB9XG5cbiAgcHVibGljIHJlbG9hZCgpOiB2b2lkIHtcbiAgICB0aGlzLmFjbEVudHJpZXNMaXN0LnJlbG9hZCgpO1xuICB9XG5cbn1cbiIsIlxuPGZzLWxpc3QgW2NvbmZpZ109XCJhY2xFbnRyaWVzQ29uZmlnXCI+XG4gIDxmcy1saXN0LWNvbHVtbj5cbiAgICA8bmctdGVtcGxhdGUgZnMtbGlzdC1oZWFkZXI+Q29udGV4dDwvbmctdGVtcGxhdGU+XG4gICAgPG5nLXRlbXBsYXRlIGZzLWxpc3QtY2VsbCBsZXQtcm93PVwicm93XCI+XG4gICAgICA8ZGl2XG4gICAgICAgICAgKm5nSWY9XCJyb3cub2JqZWN0OyBlbHNlIGVsc2VPYmplY3RcIlxuICAgICAgICAgIGZ4TGF5b3V0PVwicm93XCJcbiAgICAgICAgICBmeExheW91dEFsaWduPVwic3RhcnQgY2VudGVyXCJcbiAgICAgICAgICBmeExheW91dEdhcD1cIjEwcHhcIj5cbiAgICAgICAgPGZzLWJhZGdlICpuZ0lmPVwicm93Lm9iamVjdC5pbWFnZVVybFwiIHNoYXBlPVwiY2lyY2xlXCIgaW1hZ2U9XCJ7eyByb3cub2JqZWN0LmltYWdlVXJsIH19XCI+PC9mcy1iYWRnZT5cbiAgICAgICAgPHNwYW4+XG4gICAgICAgICAgPGRpdj48c21hbGw+e3sgcm93Lm9iamVjdC5jbGFzc05hbWUgfX08L3NtYWxsPjwvZGl2PlxuICAgICAgICAgIDxhIChjbGljayk9XCJ1cGRhdGUocm93KVwiPnt7IHJvdy5vYmplY3QubmFtZSB9fTwvYT5cbiAgICAgICAgPC9zcGFuPlxuICAgICAgPC9kaXY+XG5cbiAgICAgIDxuZy10ZW1wbGF0ZSAjZWxzZU9iamVjdD5cbiAgICAgICAgPGEgKGNsaWNrKT1cInVwZGF0ZShyb3cpXCI+QXBwPC9hPlxuICAgICAgPC9uZy10ZW1wbGF0ZT5cbiAgICA8L25nLXRlbXBsYXRlPlxuICA8L2ZzLWxpc3QtY29sdW1uPlxuXG4gIDxmcy1saXN0LWNvbHVtbiBbc2hvd109XCJlbnZpcm9ubWVudFNob3dcIj5cbiAgICA8bmctdGVtcGxhdGUgZnMtbGlzdC1oZWFkZXI+e3tlbnZpcm9ubWVudExhYmVsfX08L25nLXRlbXBsYXRlPlxuICAgIDxuZy10ZW1wbGF0ZSBmcy1saXN0LWNlbGwgbGV0LXJvdz1cInJvd1wiPlxuICAgICAge3tyb3cuZW52aXJvbm1lbnQ/Lm5hbWV9fVxuICAgIDwvbmctdGVtcGxhdGU+XG4gIDwvZnMtbGlzdC1jb2x1bW4+XG5cbiAgPGZzLWxpc3QtY29sdW1uPlxuICAgIDxuZy10ZW1wbGF0ZSBmcy1saXN0LWhlYWRlcj5Sb2xlczwvbmctdGVtcGxhdGU+XG4gICAgPG5nLXRlbXBsYXRlIGZzLWxpc3QtY2VsbCBsZXQtcm93PVwicm93XCI+XG4gICAgICA8ZGl2IGZ4TGF5b3V0PVwiY29sdW1uXCIgZnhMYXlvdXRHYXA9XCIxMHB4XCI+XG4gICAgICAgIDxkaXYgZGl2ICpuZ0Zvcj1cImxldCBhY2xFbnRyeSBvZiByb3cuYWNsRW50cmllc1wiPlxuICAgICAgICAgIDxuZy1jb250YWluZXIgW25nU3dpdGNoXT1cInJvdy5sZXZlbFwiPlxuICAgICAgICAgICAgPG5nLWNvbnRhaW5lciAqbmdTd2l0Y2hDYXNlPVwiJ2FwcCdcIj5cbiAgICAgICAgICAgICAgPGZzLWFjbC1yb2xlLXBvcG92ZXIgW2FjbFJvbGVdPVwiYWNsRW50cnkuYWNsUm9sZVwiIG9iamVjdE5hbWU9XCJBcHBcIj57e2FjbEVudHJ5LmFjbFJvbGUubmFtZX19PC9mcy1hY2wtcm9sZS1wb3BvdmVyPlxuICAgICAgICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ1N3aXRjaERlZmF1bHQ+XG4gICAgICAgICAgICAgIDxuZy1jb250YWluZXIgKm5nSWY9XCJyb3cub2JqZWN0XCI+XG4gICAgICAgICAgICAgICAgPGZzLWFjbC1yb2xlLXBvcG92ZXIgW2FjbFJvbGVdPVwiYWNsRW50cnkuYWNsUm9sZVwiPnt7YWNsRW50cnkuYWNsUm9sZS5uYW1lfX08L2ZzLWFjbC1yb2xlLXBvcG92ZXI+XG4gICAgICAgICAgICAgIDwvbmctY29udGFpbmVyPlxuICAgICAgICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgPC9uZy10ZW1wbGF0ZT5cbiAgPC9mcy1saXN0LWNvbHVtbj5cbjwvZnMtbGlzdD5cbiJdfQ==
51
+ Are you sure you would like to continue?`,
52
+ }).subscribe(() => {
53
+ const data = Object.assign(Object.assign({}, aclObjectEntry), { aclEntries: [] });
54
+ this.saveAclObjectEntry(data)
55
+ .subscribe(() => {
56
+ this.aclEntriesList.reload();
57
+ });
58
+ });
59
+ }
60
+ }
61
+ ],
62
+ fetch: () => {
63
+ return new Observable((observer) => {
64
+ this.loadAclEntries({
65
+ aclRoles: true,
66
+ aclRolePermissions: true,
67
+ objects: true,
68
+ aclRoleState: 'active',
69
+ })
70
+ .subscribe((aclEntries) => {
71
+ const objects = aclEntries
72
+ .filter((aclEntry) => (!!aclEntry.object))
73
+ .reduce((items, item) => {
74
+ return Object.assign(Object.assign({}, items), { [item.object.id]: item.object });
75
+ }, {});
76
+ const environments = aclEntries
77
+ .filter((aclEntry) => (!!aclEntry[this.environmentKey]))
78
+ .reduce((items, item) => {
79
+ const environment = item[this.environmentKey];
80
+ return Object.assign(Object.assign({}, items), { [environment.id]: environment });
81
+ }, {});
82
+ const groupedAclEntries = groupBy(aclEntries, (item) => {
83
+ var _a;
84
+ const environmentId = (_a = (item[this.environmentKey])) === null || _a === void 0 ? void 0 : _a.id;
85
+ return [item.aclRole.level, environmentId, item.objectId];
86
+ });
87
+ let aclObjectEntries = Object.keys(groupedAclEntries)
88
+ .reduce((accum, key) => {
89
+ const parts = key.split(',');
90
+ return [
91
+ ...accum,
92
+ {
93
+ object: objects[parts[2]],
94
+ level: parts[0],
95
+ [`${this.environmentKey}Id`]: parts[1] ? parseInt(parts[1]) : null,
96
+ [this.environmentKey]: environments[parts[1]],
97
+ aclEntries: groupedAclEntries[key],
98
+ }
99
+ ];
100
+ }, []);
101
+ const hasApp = aclObjectEntries.some((item) => {
102
+ return item.aclEntries.some((entry) => {
103
+ return !entry.objectId;
104
+ });
105
+ });
106
+ if (!hasApp) {
107
+ aclObjectEntries.unshift({
108
+ object: null,
109
+ aclEntries: [],
110
+ level: 'app',
111
+ environmentId: null,
112
+ });
113
+ }
114
+ aclObjectEntries = sortBy(aclObjectEntries, (item) => {
115
+ return item.object ? item.level : '';
116
+ });
117
+ observer.next({ data: aclObjectEntries });
118
+ observer.complete();
119
+ });
120
+ });
121
+ },
122
+ };
123
+ }
124
+ update(aclObjectEntry) {
125
+ const data = {
126
+ aclObjectEntry,
127
+ required: false,
128
+ loadAclRoles: this.loadAclRoles,
129
+ saveAclObjectEntry: this.saveAclObjectEntry
130
+ };
131
+ this._dialog.open(FsAclEntryComponent, {
132
+ data: data
133
+ })
134
+ .afterClosed()
135
+ .pipe(takeUntil(this._destroy$))
136
+ .subscribe(() => {
137
+ this.aclEntriesList.reload();
138
+ });
139
+ }
140
+ ngOnDestroy() {
141
+ this._destroy$.next();
142
+ this._destroy$.complete();
143
+ }
144
+ reload() {
145
+ this.aclEntriesList.reload();
146
+ }
147
+ }
148
+ FsAclEntriesComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: FsAclEntriesComponent, deps: [{ token: i1.FsAppAclService }, { token: i2.MatDialog }, { token: i3.FsPrompt }], target: i0.ɵɵFactoryTarget.Component });
149
+ FsAclEntriesComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.17", type: FsAclEntriesComponent, selector: "fs-acl-entries", inputs: { loadAclEntries: "loadAclEntries", loadAclRoles: "loadAclRoles", saveAclObjectEntry: "saveAclObjectEntry", environmentShow: "environmentShow", environmentLabel: "environmentLabel", environmentKey: "environmentKey", actions: "actions" }, viewQueries: [{ propertyName: "aclEntriesList", first: true, predicate: FsListComponent, descendants: true }], ngImport: i0, template: "\n<fs-list [config]=\"aclEntriesConfig\">\n <fs-list-column>\n <ng-template fs-list-header>Context</ng-template>\n <ng-template fs-list-cell let-row=\"row\">\n <div\n *ngIf=\"row.object; else elseObject\"\n class=\"fs-row.gap-small\">\n <fs-badge *ngIf=\"row.object.imageUrl\" shape=\"circle\" image=\"{{ row.object.imageUrl }}\"></fs-badge>\n <span>\n <div><small>{{ row.object.className }}</small></div>\n <a (click)=\"update(row)\">{{ row.object.name }}</a>\n </span>\n </div>\n\n <ng-template #elseObject>\n <a (click)=\"update(row)\">App</a>\n </ng-template>\n </ng-template>\n </fs-list-column>\n\n <fs-list-column [show]=\"environmentShow\">\n <ng-template fs-list-header>{{environmentLabel}}</ng-template>\n <ng-template fs-list-cell let-row=\"row\">\n {{row.environment?.name}}\n </ng-template>\n </fs-list-column>\n\n <fs-list-column>\n <ng-template fs-list-header>Roles</ng-template>\n <ng-template fs-list-cell let-row=\"row\">\n <div class=\"fs-column\">\n <div div *ngFor=\"let aclEntry of row.aclEntries\">\n <ng-container [ngSwitch]=\"row.level\">\n <ng-container *ngSwitchCase=\"'app'\">\n <fs-acl-role-popover [aclRole]=\"aclEntry.aclRole\" objectName=\"App\">{{aclEntry.aclRole.name}}</fs-acl-role-popover>\n </ng-container>\n <ng-container *ngSwitchDefault>\n <ng-container *ngIf=\"row.object\">\n <fs-acl-role-popover [aclRole]=\"aclEntry.aclRole\">{{aclEntry.aclRole.name}}</fs-acl-role-popover>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </div>\n </ng-template>\n </fs-list-column>\n</fs-list>\n", styles: [""], components: [{ type: i4.FsListComponent, selector: "fs-list", inputs: ["config", "loaderLines"], outputs: ["filtersReady"] }, { type: i5.FsBadgeComponent, selector: "fs-badge", inputs: ["color", "text", "tooltip", "size", "shape", "image", "icon", "iconSize", "iconColor", "backgroundSize"] }, { type: i6.FsAclRolePopoverComponent, selector: "fs-acl-role-popover", inputs: ["aclRole", "objectName"] }], directives: [{ type: i4.FsListColumnDirective, selector: "fs-list-column", inputs: ["show", "title", "name", "customize", "sortable", "sortableDefault", "direction", "align", "width", "class"] }, { type: i4.FsListHeaderDirective, selector: "[fs-list-header]", inputs: ["colspan", "align", "class"] }, { type: i4.FsListCellDirective, selector: "[fs-list-cell]", inputs: ["colspan", "align", "class"] }, { type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i7.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { type: i7.NgSwitchDefault, selector: "[ngSwitchDefault]" }] });
150
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: FsAclEntriesComponent, decorators: [{
151
+ type: Component,
152
+ args: [{
153
+ selector: 'fs-acl-entries',
154
+ templateUrl: './acl-entries.component.html',
155
+ styleUrls: ['./acl-entries.component.scss']
156
+ }]
157
+ }], ctorParameters: function () { return [{ type: i1.FsAppAclService }, { type: i2.MatDialog }, { type: i3.FsPrompt }]; }, propDecorators: { loadAclEntries: [{
158
+ type: Input
159
+ }], loadAclRoles: [{
160
+ type: Input
161
+ }], saveAclObjectEntry: [{
162
+ type: Input
163
+ }], environmentShow: [{
164
+ type: Input
165
+ }], environmentLabel: [{
166
+ type: Input
167
+ }], environmentKey: [{
168
+ type: Input
169
+ }], actions: [{
170
+ type: Input
171
+ }], aclEntriesList: [{
172
+ type: ViewChild,
173
+ args: [FsListComponent]
174
+ }] } });
175
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWNsLWVudHJpZXMuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2FwcC9jb21wb25lbnRzL2FjbC1lbnRyaWVzL2FjbC1lbnRyaWVzLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9hcHAvY29tcG9uZW50cy9hY2wtZW50cmllcy9hY2wtZW50cmllcy5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFPLE1BQU0sZ0JBQWdCLENBQUM7QUFDaEQsT0FBTyxFQUFFLFNBQVMsRUFBVSxTQUFTLEVBQWEsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQy9FLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUVyRCxPQUFPLEVBQUUsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUUzQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE9BQU8sRUFBRSxNQUFNLFdBQVcsQ0FBQztBQUU1QyxPQUFPLEVBQWdCLGVBQWUsRUFBZ0IsTUFBTSxrQkFBa0IsQ0FBQztBQUMvRSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFOUMsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBS2pFLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGtDQUFrQyxDQUFDOzs7Ozs7Ozs7QUFRdkUsTUFBTSxPQUFPLHFCQUFxQjtJQWtCaEMsWUFDbUIsY0FBK0IsRUFDL0IsT0FBa0IsRUFDM0IsUUFBa0I7UUFGVCxtQkFBYyxHQUFkLGNBQWMsQ0FBaUI7UUFDL0IsWUFBTyxHQUFQLE9BQU8sQ0FBVztRQUMzQixhQUFRLEdBQVIsUUFBUSxDQUFVO1FBaEJaLG9CQUFlLEdBQUcsSUFBSSxDQUFDO1FBQ3ZCLHFCQUFnQixHQUFHLGFBQWEsQ0FBQztRQUNqQyxtQkFBYyxHQUFHLGFBQWEs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QUVNLFdBQVc7UUFDaEIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUN0QixJQUFJLENBQUMsU0FBUyxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBQzVCLENBQUM7SUFFTSxNQUFNO1FBQ1gsSUFBSSxDQUFDLGNBQWMsQ0FBQyxNQUFNLEVBQUUsQ0FBQztJQUMvQixDQUFDOzttSEE3SlUscUJBQXFCO3VHQUFyQixxQkFBcUIsNFZBVXJCLGVBQWUsZ0RDbEM1Qiw0d0RBZ0RBOzRGRHhCYSxxQkFBcUI7a0JBTGpDLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLGdCQUFnQjtvQkFDMUIsV0FBVyxFQUFFLDhCQUE4QjtvQkFDM0MsU0FBUyxFQUFFLENBQUMsOEJBQThCLENBQUM7aUJBQzVDO3FKQUdpQixjQUFjO3NCQUE3QixLQUFLO2dCQUNVLFlBQVk7c0JBQTNCLEtBQUs7Z0JBQ1Usa0JBQWtCO3NCQUFqQyxLQUFLO2dCQUNVLGVBQWU7c0JBQTlCLEtBQUs7Z0JBQ1UsZ0JBQWdCO3NCQUEvQixLQUFLO2dCQUNVLGNBQWM7c0JBQTdCLEtBQUs7Z0JBQ1UsT0FBTztzQkFBdEIsS0FBSztnQkFHQyxjQUFjO3NCQURwQixTQUFTO3VCQUFDLGVBQWUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB0YWtlVW50aWwsIG1hcCB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcbmltcG9ydCB7IENvbXBvbmVudCwgT25Jbml0LCBWaWV3Q2hpbGQsIE9uRGVzdHJveSwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE1hdERpYWxvZyB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2RpYWxvZyc7XG5cbmltcG9ydCB7IFN1YmplY3QsIE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcblxuaW1wb3J0IHsgc29ydEJ5LCBncm91cEJ5IH0gZnJvbSAnbG9kYXNoLWVzJztcblxuaW1wb3J0IHsgRnNMaXN0QWN0aW9uLCBGc0xpc3RDb21wb25lbnQsIEZzTGlzdENvbmZpZyB9IGZyb20gJ0BmaXJlc3RpdGNoL2xpc3QnO1xuaW1wb3J0IHsgRnNQcm9tcHQgfSBmcm9tICdAZmlyZXN0aXRjaC9wcm9tcHQnO1xuXG5pbXBvcnQgeyBGc0FwcEFjbFNlcnZpY2UgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9hcHAtYWNsLnNlcnZpY2UnO1xuaW1wb3J0IHsgQWNsRW50cnlEYXRhIH0gZnJvbSAnLi4vLi4vaW50ZXJmYWNlcy9hY2wtZW50cnktZGF0YSc7XG5pbXBvcnQgeyBBY2xFbnRyeSB9IGZyb20gJy4uLy4uL2ludGVyZmFjZXMvYWNsLWVudHJ5JztcbmltcG9ydCB7IEFjbFJvbGUgfSBmcm9tICcuLi8uLi9pbnRlcmZhY2VzL2FjbC1yb2xlJztcbmltcG9ydCB7IEFjbE9iamVjdEVudHJ5IH0gZnJvbSAnLi4vLi4vaW50ZXJmYWNlcy9hY2wtb2JqZWN0LWVudHJ5JztcbmltcG9ydCB7IEZzQWNsRW50cnlDb21wb25lbnQgfSBmcm9tICcuLi9hY2wtZW50cnkvYWNsLWVudHJ5LmNvbXBvbmVudCc7XG5cblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnZnMtYWNsLWVudHJpZXMnLFxuICB0ZW1wbGF0ZVVybDogJy4vYWNsLWVudHJpZXMuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9hY2wtZW50cmllcy5jb21wb25lbnQuc2NzcyddXG59KVxuZXhwb3J0IGNsYXNzIEZzQWNsRW50cmllc0NvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCwgT25EZXN0cm95IHtcblxuICBASW5wdXQoKSBwdWJsaWMgbG9hZEFjbEVudHJpZXM6IChxdWVyeTogYW55KSA9PiBPYnNlcnZhYmxlPEFjbEVudHJ5W10+O1xuICBASW5wdXQoKSBwdWJsaWMgbG9hZEFjbFJvbGVzOiAocXVlcnk6IGFueSkgPT4gT2JzZXJ2YWJsZTxBY2xSb2xlW10+O1xuICBASW5wdXQoKSBwdWJsaWMgc2F2ZUFjbE9iamVjdEVudHJ5OiAoYWNsT2JqZWN0RW50cnk6IEFjbE9iamVjdEVudHJ5KSA9PiBPYnNlcnZhYmxlPGFueT47XG4gIEBJbnB1dCgpIHB1YmxpYyBlbnZpcm9ubWVudFNob3cgPSB0cnVlO1xuICBASW5wdXQoKSBwdWJsaWMgZW52aXJvbm1lbnRMYWJlbCA9ICdFbnZpcm9ubWVudCc7XG4gIEBJbnB1dCgpIHB1YmxpYyBlbnZpcm9ubWVudEtleSA9ICdlbnZpcm9ubWVudCc7XG4gIEBJbnB1dCgpIHB1YmxpYyBhY3Rpb25zOiBGc0xpc3RBY3Rpb25bXSA9IFtdO1xuXG4gIEBWaWV3Q2hpbGQoRnNMaXN0Q29tcG9uZW50KVxuICBwdWJsaWMgYWNsRW50cmllc0xpc3Q6IEZzTGlzdENvbXBvbmVudCA9IG51bGw7XG5cbiAgcHVibGljIGFjbEVudHJpZXNDb25maWc6IEZzTGlzdENvbmZpZyA9IG51bGw7XG4gIHB1YmxpYyBwZXJtaXNzaW9uczogYW55W10gPSBbXTtcblxuICBwcml2YXRlIF9kZXN0cm95JCA9IG5ldyBTdWJqZWN0KCk7XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSByZWFkb25seSBfYXBwQWNsU2VydmljZTogRnNBcHBBY2xTZXJ2aWNlLFxuICAgIHByaXZhdGUgcmVhZG9ubHkgX2RpYWxvZzogTWF0RGlhbG9nLFxuICAgIHByaXZhdGUgX2NvbmZpcm06IEZzUHJvbXB0LFxuICApIHsgfVxuXG4gIHB1YmxpYyBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICB0aGlzLl9hcHBBY2xTZXJ2aWNlLmdldFBlcm1pc3Npb25zKClcbiAgICAgIC5zdWJzY3JpYmUocmVzcG9uc2UgPT4ge1xuICAgICAgICB0aGlzLnBlcm1pc3Npb25zID0gcmVzcG9uc2U7XG4gICAgICB9KTtcblxuICAgIHRoaXMuYWNsRW50cmllc0NvbmZpZyA9IHtcbiAgICAgIHN0YXR1czogZmFsc2UsXG4gICAgICBwYWdpbmc6IGZhbHNlLFxuICAgICAgYWN0aW9uczogdGhpcy5hY3Rpb25zLFxuICAgICAgcm93QWN0aW9uczogW1xuICAgICAgICB7XG4gICAgICAgICAgbGFiZWw6ICdSZW1vdmUgQWxsIFJvbGVzJyxcbiAgICAgICAgICBjbGljazogKGFjbE9iamVjdEVudHJ5OiBBY2xPYmplY3RFbnRyeSkgPT4ge1xuICAgICAgICAgICAgdGhpcy5fY29uZmlybVxuICAgICAgICAgICAgICAuY29uZmlybSh7XG4gICAgICAgICAgICAgICAgdGl0bGU6ICdSZW1vdmUgQWxsIFJvbGVzJyxcbiAgICAgICAgICAgICAgICBjb21taXRMYWJlbDogJ1NhdmUnLFxuICAgICAgICAgICAgICAgIHRlbXBsYXRlOiBgUGxlYXNlIG5vdGUgdGhhdCByZW1vdmluZyByb2xlcyBtYXkgcHJldmVudCB1c2VycyBmcm9tIGJlaW5nIGFibGUgdG8gc3VjY2Vzc2Z1bGx5IGxvZ2luLjxicj5cbiAgICAgICAgICAgICAgICAgIFRoZXNlIGNoYW5nZXMgYXJlIGVmZmVjdGl2ZSBpbW1lZGlhdGVseS48YnI+XG4gICAgICAgICAgICAgICAgICBBcmUgeW91IHN1cmUgeW91IHdvdWxkIGxpa2UgdG8gY29udGludWU/YCxcbiAgICAgICAgICAgICAgfSkuc3Vic2NyaWJlKCgpID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCBkYXRhID0geyAuLi5hY2xPYmplY3RFbnRyeSwgYWNsRW50cmllczogW10gfTtcbiAgICAgICAgICAgICAgICB0aGlzLnNhdmVBY2xPYmplY3RFbnRyeShkYXRhKVxuICAgICAgICAgICAgICAgICAgLnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgIHRoaXMuYWNsRW50cmllc0xpc3QucmVsb2FkKCk7XG4gICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICBdLFxuICAgICAgZmV0Y2g6ICgpID0+IHtcbiAgICAgICAgcmV0dXJuIG5ldyBPYnNlcnZhYmxlKChvYnNlcnZlcikgPT4ge1xuICAgICAgICAgIHRoaXMubG9hZEFjbEVudHJpZXMoe1xuICAgICAgICAgICAgYWNsUm9sZXM6IHRydWUsXG4gICAgICAgICAgICBhY2xSb2xlUGVybWlzc2lvbnM6IHRydWUsXG4gICAgICAgICAgICBvYmplY3RzOiB0cnVlLFxuICAgICAgICAgICAgYWNsUm9sZVN0YXRlOiAnYWN0aXZlJyxcbiAgICAgICAgICB9KVxuICAgICAgICAgICAgLnN1YnNjcmliZSgoYWNsRW50cmllczogQWNsRW50cnlbXSkgPT4ge1xuICAgICAgICAgICAgICBjb25zdCBvYmplY3RzID0gYWNsRW50cmllc1xuICAgICAgICAgICAgICAgIC5maWx0ZXIoKGFjbEVudHJ5KSA9PiAoISFhY2xFbnRyeS5vYmplY3QpKVxuICAgICAgICAgICAgICAgIC5yZWR1Y2UoKGl0ZW1zLCBpdGVtKSA9PiB7XG4gICAgICAgICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgICAgICAuLi5pdGVtcyxcbiAgICAgICAgICAgICAgICAgICAgW2l0ZW0ub2JqZWN0LmlkXTogaXRlbS5vYmplY3QsXG4gICAgICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgICAgIH0sIHt9KTtcblxuICAgICAgICAgICAgICBjb25zdCBlbnZpcm9ubWVudHMgPSBhY2xFbnRyaWVzXG4gICAgICAgICAgICAgICAgLmZpbHRlcigoYWNsRW50cnkpID0+ICghIWFjbEVudHJ5W3RoaXMuZW52aXJvbm1lbnRLZXldKSlcbiAgICAgICAgICAgICAgICAucmVkdWNlKChpdGVtcywgaXRlbSkgPT4ge1xuICAgICAgICAgICAgICAgICAgY29uc3QgZW52aXJvbm1lbnQgPSBpdGVtW3RoaXMuZW52aXJvbm1lbnRLZXldO1xuICAgICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgICAgLi4uaXRlbXMsXG4gICAgICAgICAgICAgICAgICAgIFtlbnZpcm9ubWVudC5pZF06IGVudmlyb25tZW50LFxuICAgICAgICAgICAgICAgICAgfTtcbiAgICAgICAgICAgICAgICB9LCB7fSk7XG5cbiAgICAgICAgICAgICAgY29uc3QgZ3JvdXBlZEFjbEVudHJpZXMgPSBncm91cEJ5KGFjbEVudHJpZXMsIChpdGVtKSA9PiB7XG4gICAgICAgICAgICAgICAgY29uc3QgZW52aXJvbm1lbnRJZCA9IChpdGVtW3RoaXMuZW52aXJvbm1lbnRLZXldKT8uaWQ7XG4gICAgICAgICAgICAgICAgcmV0dXJuIFtpdGVtLmFjbFJvbGUubGV2ZWwsIGVudmlyb25tZW50SWQsIGl0ZW0ub2JqZWN0SWRdO1xuICAgICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgICBsZXQgYWNsT2JqZWN0RW50cmllczogQWNsT2JqZWN0RW50cnlbXSA9IE9iamVjdC5rZXlzKGdyb3VwZWRBY2xFbnRyaWVzKVxuICAgICAgICAgICAgICAgIC5yZWR1Y2UoKGFjY3VtLCBrZXk6IGFueSkgPT4ge1xuICAgICAgICAgICAgICAgICAgY29uc3QgcGFydHMgPSBrZXkuc3BsaXQoJywnKTtcbiAgICAgICAgICAgICAgICAgIHJldHVybiBbXG4gICAgICAgICAgICAgICAgICAgIC4uLmFjY3VtLFxuICAgICAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgICAgb2JqZWN0OiBvYmplY3RzW3BhcnRzWzJdXSxcbiAgICAgICAgICAgICAgICAgICAgICBsZXZlbDogcGFydHNbMF0sXG4gICAgICAgICAgICAgICAgICAgICAgW2Ake3RoaXMuZW52aXJvbm1lbnRLZXl9SWRgXTogcGFydHNbMV0gPyBwYXJzZUludChwYXJ0c1sxXSkgOiBudWxsLFxuICAgICAgICAgICAgICAgICAgICAgIFt0aGlzLmVudmlyb25tZW50S2V5XTogZW52aXJvbm1lbnRzW3BhcnRzWzFdXSxcbiAgICAgICAgICAgICAgICAgICAgICBhY2xFbnRyaWVzOiBncm91cGVkQWNsRW50cmllc1trZXldLFxuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICBdO1xuICAgICAgICAgICAgICAgIH0sIFtdKTtcblxuICAgICAgICAgICAgICBjb25zdCBoYXNBcHAgPSBhY2xPYmplY3RFbnRyaWVzLnNvbWUoKGl0ZW0pID0+IHtcbiAgICAgICAgICAgICAgICByZXR1cm4gaXRlbS5hY2xFbnRyaWVzLnNvbWUoKGVudHJ5KSA9PiB7XG4gICAgICAgICAgICAgICAgICByZXR1cm4gIWVudHJ5Lm9iamVjdElkO1xuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgICBpZiAoIWhhc0FwcCkge1xuICAgICAgICAgICAgICAgIGFjbE9iamVjdEVudHJpZXMudW5zaGlmdCh7XG4gICAgICAgICAgICAgICAgICBvYmplY3Q6IG51bGwsXG4gICAgICAgICAgICAgICAgICBhY2xFbnRyaWVzOiBbXSxcbiAgICAgICAgICAgICAgICAgIGxldmVsOiAnYXBwJyxcbiAgICAgICAgICAgICAgICAgIGVudmlyb25tZW50SWQ6IG51bGwsXG4gICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICBhY2xPYmplY3RFbnRyaWVzID0gc29ydEJ5KGFjbE9iamVjdEVudHJpZXMsIChpdGVtOiBBY2xPYmplY3RFbnRyeSkgPT4ge1xuICAgICAgICAgICAgICAgIHJldHVybiBpdGVtLm9iamVjdCA/IGl0ZW0ubGV2ZWwgOiAnJztcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgb2JzZXJ2ZXIubmV4dCh7IGRhdGE6IGFjbE9iamVjdEVudHJpZXMgfSk7XG4gICAgICAgICAgICAgIG9ic2VydmVyLmNvbXBsZXRlKCk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfSk7XG4gICAgICB9LFxuICAgIH07XG4gIH1cblxuICBwdWJsaWMgdXBkYXRlKGFjbE9iamVjdEVudHJ5OiBBY2xPYmplY3RFbnRyeSkge1xuICAgIGNvbnN0IGRhdGE6IEFjbEVudHJ5RGF0YSA9IHtcbiAgICAgIGFjbE9iamVjdEVudHJ5LFxuICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgbG9hZEFjbFJvbGVzOiB0aGlzLmxvYWRBY2xSb2xlcyxcbiAgICAgIHNhdmVBY2xPYmplY3RFbnRyeTogdGhpcy5zYXZlQWNsT2JqZWN0RW50cnlcbiAgICB9XG5cbiAgICB0aGlzLl9kaWFsb2cub3BlbihGc0FjbEVudHJ5Q29tcG9uZW50LCB7XG4gICAgICBkYXRhOiBkYXRhXG4gICAgfSlcbiAgICAgIC5hZnRlckNsb3NlZCgpXG4gICAgICAucGlwZShcbiAgICAgICAgdGFrZVVudGlsKHRoaXMuX2Rlc3Ryb3kkKSxcbiAgICAgIClcbiAgICAgIC5zdWJzY3JpYmUoKCkgPT4ge1xuICAgICAgICB0aGlzLmFjbEVudHJpZXNMaXN0LnJlbG9hZCgpO1xuICAgICAgfSk7XG4gIH1cblxuICBwdWJsaWMgbmdPbkRlc3Ryb3koKSB7XG4gICAgdGhpcy5fZGVzdHJveSQubmV4dCgpO1xuICAgIHRoaXMuX2Rlc3Ryb3kkLmNvbXBsZXRlKCk7XG4gIH1cblxuICBwdWJsaWMgcmVsb2FkKCk6IHZvaWQge1xuICAgIHRoaXMuYWNsRW50cmllc0xpc3QucmVsb2FkKCk7XG4gIH1cblxufVxuIiwiXG48ZnMtbGlzdCBbY29uZmlnXT1cImFjbEVudHJpZXNDb25maWdcIj5cbiAgPGZzLWxpc3QtY29sdW1uPlxuICAgIDxuZy10ZW1wbGF0ZSBmcy1saXN0LWhlYWRlcj5Db250ZXh0PC9uZy10ZW1wbGF0ZT5cbiAgICA8bmctdGVtcGxhdGUgZnMtbGlzdC1jZWxsIGxldC1yb3c9XCJyb3dcIj5cbiAgICAgIDxkaXZcbiAgICAgICAgICAqbmdJZj1cInJvdy5vYmplY3Q7IGVsc2UgZWxzZU9iamVjdFwiXG4gICAgICAgICAgY2xhc3M9XCJmcy1yb3cuZ2FwLXNtYWxsXCI+XG4gICAgICAgIDxmcy1iYWRnZSAqbmdJZj1cInJvdy5vYmplY3QuaW1hZ2VVcmxcIiBzaGFwZT1cImNpcmNsZVwiIGltYWdlPVwie3sgcm93Lm9iamVjdC5pbWFnZVVybCB9fVwiPjwvZnMtYmFkZ2U+XG4gICAgICAgIDxzcGFuPlxuICAgICAgICAgIDxkaXY+PHNtYWxsPnt7IHJvdy5vYmplY3QuY2xhc3NOYW1lIH19PC9zbWFsbD48L2Rpdj5cbiAgICAgICAgICA8YSAoY2xpY2spPVwidXBkYXRlKHJvdylcIj57eyByb3cub2JqZWN0Lm5hbWUgfX08L2E+XG4gICAgICAgIDwvc3Bhbj5cbiAgICAgIDwvZGl2PlxuXG4gICAgICA8bmctdGVtcGxhdGUgI2Vsc2VPYmplY3Q+XG4gICAgICAgIDxhIChjbGljayk9XCJ1cGRhdGUocm93KVwiPkFwcDwvYT5cbiAgICAgIDwvbmctdGVtcGxhdGU+XG4gICAgPC9uZy10ZW1wbGF0ZT5cbiAgPC9mcy1saXN0LWNvbHVtbj5cblxuICA8ZnMtbGlzdC1jb2x1bW4gW3Nob3ddPVwiZW52aXJvbm1lbnRTaG93XCI+XG4gICAgPG5nLXRlbXBsYXRlIGZzLWxpc3QtaGVhZGVyPnt7ZW52aXJvbm1lbnRMYWJlbH19PC9uZy10ZW1wbGF0ZT5cbiAgICA8bmctdGVtcGxhdGUgZnMtbGlzdC1jZWxsIGxldC1yb3c9XCJyb3dcIj5cbiAgICAgIHt7cm93LmVudmlyb25tZW50Py5uYW1lfX1cbiAgICA8L25nLXRlbXBsYXRlPlxuICA8L2ZzLWxpc3QtY29sdW1uPlxuXG4gIDxmcy1saXN0LWNvbHVtbj5cbiAgICA8bmctdGVtcGxhdGUgZnMtbGlzdC1oZWFkZXI+Um9sZXM8L25nLXRlbXBsYXRlPlxuICAgIDxuZy10ZW1wbGF0ZSBmcy1saXN0LWNlbGwgbGV0LXJvdz1cInJvd1wiPlxuICAgICAgPGRpdiBjbGFzcz1cImZzLWNvbHVtblwiPlxuICAgICAgICA8ZGl2IGRpdiAqbmdGb3I9XCJsZXQgYWNsRW50cnkgb2Ygcm93LmFjbEVudHJpZXNcIj5cbiAgICAgICAgICA8bmctY29udGFpbmVyIFtuZ1N3aXRjaF09XCJyb3cubGV2ZWxcIj5cbiAgICAgICAgICAgIDxuZy1jb250YWluZXIgKm5nU3dpdGNoQ2FzZT1cIidhcHAnXCI+XG4gICAgICAgICAgICAgIDxmcy1hY2wtcm9sZS1wb3BvdmVyIFthY2xSb2xlXT1cImFjbEVudHJ5LmFjbFJvbGVcIiBvYmplY3ROYW1lPVwiQXBwXCI+e3thY2xFbnRyeS5hY2xSb2xlLm5hbWV9fTwvZnMtYWNsLXJvbGUtcG9wb3Zlcj5cbiAgICAgICAgICAgIDwvbmctY29udGFpbmVyPlxuICAgICAgICAgICAgPG5nLWNvbnRhaW5lciAqbmdTd2l0Y2hEZWZhdWx0PlxuICAgICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ0lmPVwicm93Lm9iamVjdFwiPlxuICAgICAgICAgICAgICAgIDxmcy1hY2wtcm9sZS1wb3BvdmVyIFthY2xSb2xlXT1cImFjbEVudHJ5LmFjbFJvbGVcIj57e2FjbEVudHJ5LmFjbFJvbGUubmFtZX19PC9mcy1hY2wtcm9sZS1wb3BvdmVyPlxuICAgICAgICAgICAgICA8L25nLWNvbnRhaW5lcj5cbiAgICAgICAgICAgIDwvbmctY29udGFpbmVyPlxuICAgICAgICAgIDwvbmctY29udGFpbmVyPlxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuICAgIDwvbmctdGVtcGxhdGU+XG4gIDwvZnMtbGlzdC1jb2x1bW4+XG48L2ZzLWxpc3Q+XG4iXX0=