ng-firebase-table-kxp 1.2.0 → 1.2.2
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/{fesm2020 → fesm2022}/ng-firebase-table-kxp.mjs +2762 -2744
- package/fesm2022/ng-firebase-table-kxp.mjs.map +1 -0
- package/index.d.ts +665 -5
- package/package.json +3 -11
- package/esm2020/lib/components/table/table.component.mjs +0 -710
- package/esm2020/lib/components/table-tabs/table-tabs.component.mjs +0 -73
- package/esm2020/lib/components/table-tooltip/table-tooltip.component.mjs +0 -34
- package/esm2020/lib/ng-firebase-table-kxp.component.mjs +0 -11
- package/esm2020/lib/ng-firebase-table-kxp.module.mjs +0 -98
- package/esm2020/lib/ng-firebase-table-kxp.service.mjs +0 -14
- package/esm2020/lib/services/filter.service.mjs +0 -416
- package/esm2020/lib/services/pagination.service.mjs +0 -115
- package/esm2020/lib/services/table.service.mjs +0 -1140
- package/esm2020/lib/services/tooltip.service.mjs +0 -141
- package/esm2020/lib/types/Table.mjs +0 -9
- package/esm2020/lib/utils/table.utils.mjs +0 -75
- package/esm2020/ng-firebase-table-kxp.mjs +0 -5
- package/esm2020/public-api.mjs +0 -22
- package/fesm2015/ng-firebase-table-kxp.mjs +0 -2869
- package/fesm2015/ng-firebase-table-kxp.mjs.map +0 -1
- package/fesm2020/ng-firebase-table-kxp.mjs.map +0 -1
- package/lib/components/table/table.component.d.ts +0 -132
- package/lib/components/table/table.component.d.ts.map +0 -1
- package/lib/components/table-tabs/table-tabs.component.d.ts +0 -34
- package/lib/components/table-tabs/table-tabs.component.d.ts.map +0 -1
- package/lib/components/table-tooltip/table-tooltip.component.d.ts +0 -18
- package/lib/components/table-tooltip/table-tooltip.component.d.ts.map +0 -1
- package/lib/ng-firebase-table-kxp.component.d.ts +0 -5
- package/lib/ng-firebase-table-kxp.component.d.ts.map +0 -1
- package/lib/ng-firebase-table-kxp.module.d.ts +0 -23
- package/lib/ng-firebase-table-kxp.module.d.ts.map +0 -1
- package/lib/ng-firebase-table-kxp.service.d.ts +0 -6
- package/lib/ng-firebase-table-kxp.service.d.ts.map +0 -1
- package/lib/services/filter.service.d.ts +0 -88
- package/lib/services/filter.service.d.ts.map +0 -1
- package/lib/services/pagination.service.d.ts +0 -34
- package/lib/services/pagination.service.d.ts.map +0 -1
- package/lib/services/table.service.d.ts +0 -80
- package/lib/services/table.service.d.ts.map +0 -1
- package/lib/services/tooltip.service.d.ts +0 -73
- package/lib/services/tooltip.service.d.ts.map +0 -1
- package/lib/types/Table.d.ts +0 -162
- package/lib/types/Table.d.ts.map +0 -1
- package/lib/utils/table.utils.d.ts +0 -25
- package/lib/utils/table.utils.d.ts.map +0 -1
- package/ng-firebase-table-kxp.d.ts.map +0 -1
- package/public-api.d.ts +0 -12
- package/public-api.d.ts.map +0 -1
|
@@ -1,1140 +0,0 @@
|
|
|
1
|
-
import { Injectable, Optional } from '@angular/core';
|
|
2
|
-
import firebase from 'firebase/compat/app';
|
|
3
|
-
import { firstValueFrom } from 'rxjs';
|
|
4
|
-
import * as moment from 'moment';
|
|
5
|
-
import * as i0 from "@angular/core";
|
|
6
|
-
import * as i1 from "@angular/fire/compat/firestore";
|
|
7
|
-
import * as i2 from "@angular/material/dialog";
|
|
8
|
-
import * as i3 from "ngx-toastr";
|
|
9
|
-
export class TableService {
|
|
10
|
-
constructor(ngFire, dialog, toastr) {
|
|
11
|
-
this.ngFire = ngFire;
|
|
12
|
-
this.dialog = dialog;
|
|
13
|
-
this.toastr = toastr;
|
|
14
|
-
this.operators = {
|
|
15
|
-
'==': (a, b) => a === b,
|
|
16
|
-
'!=': (a, b) => a !== b,
|
|
17
|
-
'>': (a, b) => a > b,
|
|
18
|
-
'<': (a, b) => a < b,
|
|
19
|
-
'>=': (a, b) => a >= b,
|
|
20
|
-
'<=': (a, b) => a <= b,
|
|
21
|
-
in: (a, b) => Array.isArray(b) && b.includes(a),
|
|
22
|
-
'not-in': (a, b) => Array.isArray(b) && !b.includes(a),
|
|
23
|
-
'array-contains': (a, b) => Array.isArray(a) && a.includes(b),
|
|
24
|
-
'array-contains-any': (a, b) => Array.isArray(a) &&
|
|
25
|
-
Array.isArray(b) &&
|
|
26
|
-
b.some((item) => a.includes(item)),
|
|
27
|
-
includes: (a, b) => {
|
|
28
|
-
if (typeof a === 'string' && typeof b === 'string') {
|
|
29
|
-
return a.includes(b);
|
|
30
|
-
}
|
|
31
|
-
if (Array.isArray(a)) {
|
|
32
|
-
return a.includes(b);
|
|
33
|
-
}
|
|
34
|
-
return false;
|
|
35
|
-
},
|
|
36
|
-
};
|
|
37
|
-
}
|
|
38
|
-
async getItems(collection) {
|
|
39
|
-
try {
|
|
40
|
-
const querySnapshot = await collection.get();
|
|
41
|
-
return querySnapshot.docs.map((doc) => {
|
|
42
|
-
const data = doc.data();
|
|
43
|
-
return { ...(data ?? {}), id: doc.id };
|
|
44
|
-
});
|
|
45
|
-
}
|
|
46
|
-
catch (error) {
|
|
47
|
-
console.warn('Collection não encontrada:', error);
|
|
48
|
-
return [];
|
|
49
|
-
}
|
|
50
|
-
}
|
|
51
|
-
async executeQuery(params) {
|
|
52
|
-
if (params.filterFn) {
|
|
53
|
-
// Lógica com filtro no cliente (filterFn)
|
|
54
|
-
const BATCH_FETCH_SIZE = params.batchSize;
|
|
55
|
-
const GOAL_SIZE = params.batchSize + 1;
|
|
56
|
-
if (params.navigation === 'forward' || params.navigation === 'reload') {
|
|
57
|
-
if (params.navigation === 'reload' && params.doc) {
|
|
58
|
-
params.doc.lastDoc = null;
|
|
59
|
-
}
|
|
60
|
-
let lastDocCursor = params.doc ? params.doc.lastDoc : null;
|
|
61
|
-
let pageResults = [];
|
|
62
|
-
let allFetchedDocs = [];
|
|
63
|
-
let hasMoreDocsInDb = true;
|
|
64
|
-
while (pageResults.length < GOAL_SIZE && hasMoreDocsInDb) {
|
|
65
|
-
let query = this.ngFire.collection(params.collection).ref;
|
|
66
|
-
query = this.applyFilters(query, params.arrange, params.conditions);
|
|
67
|
-
if (lastDocCursor) {
|
|
68
|
-
query = query.startAfter(lastDocCursor);
|
|
69
|
-
}
|
|
70
|
-
query = query.limit(BATCH_FETCH_SIZE);
|
|
71
|
-
const snapshot = await query.get();
|
|
72
|
-
if (snapshot.empty) {
|
|
73
|
-
hasMoreDocsInDb = false;
|
|
74
|
-
break;
|
|
75
|
-
}
|
|
76
|
-
lastDocCursor = snapshot.docs[snapshot.docs.length - 1];
|
|
77
|
-
allFetchedDocs.push(...snapshot.docs);
|
|
78
|
-
const batchUsers = snapshot.docs
|
|
79
|
-
.map((doc) => {
|
|
80
|
-
const data = doc.data();
|
|
81
|
-
return { id: doc.id, ...(data ?? {}) };
|
|
82
|
-
})
|
|
83
|
-
.filter(params.filterFn);
|
|
84
|
-
pageResults.push(...batchUsers);
|
|
85
|
-
if (snapshot.size < BATCH_FETCH_SIZE) {
|
|
86
|
-
hasMoreDocsInDb = false;
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
const hasNextPage = pageResults.length > params.batchSize;
|
|
90
|
-
const finalItems = pageResults.slice(0, params.batchSize);
|
|
91
|
-
const firstDocOfPage = allFetchedDocs.find((doc) => doc.id === finalItems[0]?.id) || null;
|
|
92
|
-
const lastDocOfPage = allFetchedDocs.find((doc) => doc.id === finalItems[finalItems.length - 1]?.id) || null;
|
|
93
|
-
return {
|
|
94
|
-
items: finalItems,
|
|
95
|
-
filterLength: null,
|
|
96
|
-
firstDoc: firstDocOfPage,
|
|
97
|
-
lastDoc: lastDocOfPage,
|
|
98
|
-
hasNextPage: hasNextPage,
|
|
99
|
-
hasPreviousPage: !!(params.doc && params.doc.lastDoc) &&
|
|
100
|
-
params.navigation !== 'reload',
|
|
101
|
-
currentClientPageIndex: undefined,
|
|
102
|
-
};
|
|
103
|
-
}
|
|
104
|
-
// Lógica para trás (backward)
|
|
105
|
-
else if (params.navigation === 'backward') {
|
|
106
|
-
if (!params.doc || !params.doc.firstDoc) {
|
|
107
|
-
return {
|
|
108
|
-
items: [],
|
|
109
|
-
filterLength: null,
|
|
110
|
-
firstDoc: null,
|
|
111
|
-
lastDoc: null,
|
|
112
|
-
hasNextPage: true,
|
|
113
|
-
hasPreviousPage: false,
|
|
114
|
-
currentClientPageIndex: undefined,
|
|
115
|
-
};
|
|
116
|
-
}
|
|
117
|
-
let pageResults = [];
|
|
118
|
-
let allFetchedDocs = [];
|
|
119
|
-
let hasMoreDocsInDb = true;
|
|
120
|
-
let boundaryDoc = params.doc.firstDoc;
|
|
121
|
-
while (pageResults.length < GOAL_SIZE && hasMoreDocsInDb) {
|
|
122
|
-
if (!boundaryDoc)
|
|
123
|
-
break;
|
|
124
|
-
let query = this.ngFire.collection(params.collection).ref;
|
|
125
|
-
query = this.applyFilters(query, params.arrange, params.conditions);
|
|
126
|
-
query = query.endBefore(boundaryDoc);
|
|
127
|
-
query = query.limitToLast(BATCH_FETCH_SIZE);
|
|
128
|
-
const snapshot = await query.get();
|
|
129
|
-
if (snapshot.empty) {
|
|
130
|
-
hasMoreDocsInDb = false;
|
|
131
|
-
break;
|
|
132
|
-
}
|
|
133
|
-
boundaryDoc = snapshot
|
|
134
|
-
.docs[0];
|
|
135
|
-
allFetchedDocs = [...snapshot.docs, ...allFetchedDocs];
|
|
136
|
-
const batchUsers = snapshot.docs
|
|
137
|
-
.map((doc) => {
|
|
138
|
-
const data = doc.data();
|
|
139
|
-
return { id: doc.id, ...(data ?? {}) };
|
|
140
|
-
})
|
|
141
|
-
.filter(params.filterFn);
|
|
142
|
-
pageResults = [...batchUsers, ...pageResults];
|
|
143
|
-
if (snapshot.size < BATCH_FETCH_SIZE) {
|
|
144
|
-
hasMoreDocsInDb = false;
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
const finalItems = pageResults.slice(0, params.batchSize);
|
|
148
|
-
const firstDocOfPage = allFetchedDocs.find((doc) => doc.id === finalItems[0]?.id) || null;
|
|
149
|
-
const lastDocOfPage = allFetchedDocs.find((doc) => doc.id === finalItems[finalItems.length - 1]?.id) || null;
|
|
150
|
-
return {
|
|
151
|
-
items: finalItems,
|
|
152
|
-
filterLength: null,
|
|
153
|
-
firstDoc: firstDocOfPage,
|
|
154
|
-
lastDoc: lastDocOfPage,
|
|
155
|
-
hasNextPage: true,
|
|
156
|
-
currentClientPageIndex: undefined,
|
|
157
|
-
};
|
|
158
|
-
}
|
|
159
|
-
}
|
|
160
|
-
else {
|
|
161
|
-
let items = [];
|
|
162
|
-
let docs = [];
|
|
163
|
-
let hasNextPage = false;
|
|
164
|
-
let filterLength = null;
|
|
165
|
-
let query = this.ngFire.collection(params.collection).ref;
|
|
166
|
-
if (params.conditions) {
|
|
167
|
-
params.conditions.forEach((c) => {
|
|
168
|
-
if (c.operator === '!=') {
|
|
169
|
-
query = query.orderBy(c.firestoreProperty);
|
|
170
|
-
}
|
|
171
|
-
});
|
|
172
|
-
}
|
|
173
|
-
query = this.applyFilters(query, params.arrange, params.conditions);
|
|
174
|
-
if (params.navigation === 'reload') {
|
|
175
|
-
query = query.limit(params.batchSize + 1);
|
|
176
|
-
if (params.doc && params.doc.firstDoc) {
|
|
177
|
-
query = query.startAt(params.doc.firstDoc);
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
else if (params.navigation === 'forward') {
|
|
181
|
-
query = query.limit(params.batchSize + 1);
|
|
182
|
-
if (params.doc && params.doc.lastDoc) {
|
|
183
|
-
query = query.startAfter(params.doc.lastDoc);
|
|
184
|
-
}
|
|
185
|
-
}
|
|
186
|
-
else {
|
|
187
|
-
// backward
|
|
188
|
-
query = query.limitToLast(params.batchSize + 1);
|
|
189
|
-
if (params.doc && params.doc.firstDoc) {
|
|
190
|
-
query = query.endBefore(params.doc.firstDoc);
|
|
191
|
-
}
|
|
192
|
-
}
|
|
193
|
-
const itemCol = await query.get();
|
|
194
|
-
itemCol.docs.forEach((doc) => docs.push(doc));
|
|
195
|
-
const itemPromises = docs.map(async (item) => {
|
|
196
|
-
const itemData = item.data();
|
|
197
|
-
items.push({ id: item.id, ...(itemData ?? {}) });
|
|
198
|
-
});
|
|
199
|
-
let lastDoc = docs[docs.length - 1] || null;
|
|
200
|
-
let firstDoc = docs[0];
|
|
201
|
-
if ((items.length > params.batchSize && params.navigation === 'forward') ||
|
|
202
|
-
(params.navigation === 'reload' && items.length > params.batchSize)) {
|
|
203
|
-
lastDoc = docs[docs.length - 2] || null;
|
|
204
|
-
items.pop();
|
|
205
|
-
hasNextPage = true;
|
|
206
|
-
}
|
|
207
|
-
if (items.length > params.batchSize && params.navigation === 'backward') {
|
|
208
|
-
firstDoc = docs[1];
|
|
209
|
-
items.shift();
|
|
210
|
-
hasNextPage = true;
|
|
211
|
-
}
|
|
212
|
-
await Promise.all(itemPromises);
|
|
213
|
-
return {
|
|
214
|
-
items,
|
|
215
|
-
filterLength,
|
|
216
|
-
lastDoc,
|
|
217
|
-
firstDoc,
|
|
218
|
-
hasNextPage,
|
|
219
|
-
currentClientPageIndex: undefined,
|
|
220
|
-
};
|
|
221
|
-
}
|
|
222
|
-
// Fallback para garantir que sempre retornamos algo
|
|
223
|
-
return {
|
|
224
|
-
items: [],
|
|
225
|
-
filterLength: null,
|
|
226
|
-
firstDoc: null,
|
|
227
|
-
lastDoc: null,
|
|
228
|
-
hasNextPage: false,
|
|
229
|
-
currentClientPageIndex: undefined,
|
|
230
|
-
};
|
|
231
|
-
}
|
|
232
|
-
applyFilters(query, arrange, conditions) {
|
|
233
|
-
if (conditions) {
|
|
234
|
-
conditions.map((cond) => {
|
|
235
|
-
query = query.where(cond.firestoreProperty, cond.operator, cond.dashProperty);
|
|
236
|
-
});
|
|
237
|
-
}
|
|
238
|
-
let hasFilterSpecificOrderBy = false;
|
|
239
|
-
let appliedOrderByField = null;
|
|
240
|
-
const equalsFilters = arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
|
|
241
|
-
const otherFilters = arrange.filters.filter((f) => f.arrange !== 'equals');
|
|
242
|
-
const equalsGroupedByProperty = equalsFilters.reduce((acc, current) => {
|
|
243
|
-
const prop = current.filter.property;
|
|
244
|
-
if (!acc[prop]) {
|
|
245
|
-
acc[prop] = [];
|
|
246
|
-
}
|
|
247
|
-
acc[prop].push(current.filter.filtering);
|
|
248
|
-
return acc;
|
|
249
|
-
}, {});
|
|
250
|
-
for (const prop in equalsGroupedByProperty) {
|
|
251
|
-
const values = equalsGroupedByProperty[prop];
|
|
252
|
-
if (values.length > 0) {
|
|
253
|
-
query = query.where(prop, 'in', values);
|
|
254
|
-
}
|
|
255
|
-
}
|
|
256
|
-
otherFilters.forEach((filterItem) => {
|
|
257
|
-
// Aplicar filtragem por busca
|
|
258
|
-
if (filterItem.filter?.filtering &&
|
|
259
|
-
filterItem.filter?.property !== '' &&
|
|
260
|
-
filterItem.arrange === 'filter') {
|
|
261
|
-
query = query
|
|
262
|
-
.where(filterItem.filter.property, '>=', filterItem.filter.filtering.trim().toUpperCase())
|
|
263
|
-
.where(filterItem.filter.property, '<=', filterItem.filter.filtering.trim().toUpperCase() + '\uf8ff');
|
|
264
|
-
if (!hasFilterSpecificOrderBy) {
|
|
265
|
-
query = query.orderBy(filterItem.filter.property);
|
|
266
|
-
hasFilterSpecificOrderBy = true;
|
|
267
|
-
appliedOrderByField = filterItem.filter.property;
|
|
268
|
-
}
|
|
269
|
-
}
|
|
270
|
-
// Aplicar filtro do tipo "filterByDate"
|
|
271
|
-
if (filterItem.dateFilter && filterItem.arrange === 'filterByDate') {
|
|
272
|
-
query = query
|
|
273
|
-
.where(arrange.sortBy.field, '>=', filterItem.dateFilter.initial)
|
|
274
|
-
.where(arrange.sortBy.field, '<=', filterItem.dateFilter.final);
|
|
275
|
-
if (!hasFilterSpecificOrderBy) {
|
|
276
|
-
query = query.orderBy(arrange.sortBy.field);
|
|
277
|
-
hasFilterSpecificOrderBy = true;
|
|
278
|
-
appliedOrderByField = arrange.sortBy.field;
|
|
279
|
-
}
|
|
280
|
-
}
|
|
281
|
-
});
|
|
282
|
-
// Aplicar sortBy
|
|
283
|
-
if (arrange.sortBy && arrange.sortBy.field && arrange.sortBy.order) {
|
|
284
|
-
if (appliedOrderByField !== arrange.sortBy.field) {
|
|
285
|
-
query = query.orderBy(arrange.sortBy.field, arrange.sortBy.order);
|
|
286
|
-
}
|
|
287
|
-
}
|
|
288
|
-
return query;
|
|
289
|
-
}
|
|
290
|
-
getIdFilter(params) {
|
|
291
|
-
if (!params.arrange?.filters)
|
|
292
|
-
return null;
|
|
293
|
-
const idFilter = params.arrange.filters.find((f) => f.arrange === 'filter' &&
|
|
294
|
-
!!f.filter &&
|
|
295
|
-
f.filter.property === 'id' &&
|
|
296
|
-
!!f.filter.filtering);
|
|
297
|
-
return idFilter?.filter?.filtering?.trim() || null;
|
|
298
|
-
}
|
|
299
|
-
async getDocumentById(collection, docId) {
|
|
300
|
-
try {
|
|
301
|
-
const docRef = this.ngFire.collection(collection).doc(docId);
|
|
302
|
-
const docSnapshot = await docRef.get().toPromise();
|
|
303
|
-
if (docSnapshot && docSnapshot.exists) {
|
|
304
|
-
const data = docSnapshot.data();
|
|
305
|
-
return {
|
|
306
|
-
id: docSnapshot.id,
|
|
307
|
-
...(data ?? {}),
|
|
308
|
-
};
|
|
309
|
-
}
|
|
310
|
-
return null;
|
|
311
|
-
}
|
|
312
|
-
catch (error) {
|
|
313
|
-
console.warn('Erro ao buscar documento por ID:', error);
|
|
314
|
-
return null;
|
|
315
|
-
}
|
|
316
|
-
}
|
|
317
|
-
async searchByIdPartial(params, searchTerm) {
|
|
318
|
-
const exactMatch = await this.getDocumentById(params.collection, searchTerm);
|
|
319
|
-
if (exactMatch) {
|
|
320
|
-
if (params.conditions) {
|
|
321
|
-
const operators = this.operators;
|
|
322
|
-
const passesConditions = params.conditions.every((cond) => {
|
|
323
|
-
const operatorFn = operators[cond.operator];
|
|
324
|
-
return operatorFn
|
|
325
|
-
? operatorFn(exactMatch[cond.firestoreProperty], cond.dashProperty)
|
|
326
|
-
: false;
|
|
327
|
-
});
|
|
328
|
-
if (!passesConditions) {
|
|
329
|
-
return {
|
|
330
|
-
items: [],
|
|
331
|
-
filterLength: 0,
|
|
332
|
-
firstDoc: null,
|
|
333
|
-
lastDoc: null,
|
|
334
|
-
hasNextPage: false,
|
|
335
|
-
hasPreviousPage: false,
|
|
336
|
-
currentClientPageIndex: 0,
|
|
337
|
-
totalPages: 0,
|
|
338
|
-
};
|
|
339
|
-
}
|
|
340
|
-
}
|
|
341
|
-
if (params.filterFn && !params.filterFn(exactMatch)) {
|
|
342
|
-
return {
|
|
343
|
-
items: [],
|
|
344
|
-
filterLength: 0,
|
|
345
|
-
firstDoc: null,
|
|
346
|
-
lastDoc: null,
|
|
347
|
-
hasNextPage: false,
|
|
348
|
-
hasPreviousPage: false,
|
|
349
|
-
currentClientPageIndex: 0,
|
|
350
|
-
totalPages: 0,
|
|
351
|
-
};
|
|
352
|
-
}
|
|
353
|
-
return {
|
|
354
|
-
items: [exactMatch],
|
|
355
|
-
filterLength: 1,
|
|
356
|
-
firstDoc: null,
|
|
357
|
-
lastDoc: null,
|
|
358
|
-
hasNextPage: false,
|
|
359
|
-
hasPreviousPage: false,
|
|
360
|
-
currentClientPageIndex: 0,
|
|
361
|
-
totalPages: 1,
|
|
362
|
-
};
|
|
363
|
-
}
|
|
364
|
-
const searchTermLower = searchTerm.toLowerCase();
|
|
365
|
-
const paramsWithoutIdFilter = {
|
|
366
|
-
...params,
|
|
367
|
-
arrange: {
|
|
368
|
-
...params.arrange,
|
|
369
|
-
filters: params.arrange.filters.filter((f) => !(f.arrange === 'filter' && f.filter?.property === 'id')),
|
|
370
|
-
},
|
|
371
|
-
};
|
|
372
|
-
let query = this.ngFire.collection(params.collection).ref;
|
|
373
|
-
// Aplicar conditions
|
|
374
|
-
if (params.conditions) {
|
|
375
|
-
params.conditions.forEach((cond) => {
|
|
376
|
-
query = query.where(cond.firestoreProperty, cond.operator, cond.dashProperty);
|
|
377
|
-
});
|
|
378
|
-
}
|
|
379
|
-
// Aplicar sortBy
|
|
380
|
-
if (params.arrange?.sortBy?.field && params.arrange?.sortBy?.order) {
|
|
381
|
-
query = query.orderBy(params.arrange.sortBy.field, params.arrange.sortBy.order);
|
|
382
|
-
}
|
|
383
|
-
const snapshot = await query.get();
|
|
384
|
-
let items = snapshot.docs
|
|
385
|
-
.map((doc) => {
|
|
386
|
-
const data = doc.data();
|
|
387
|
-
return { id: doc.id, ...(data ?? {}) };
|
|
388
|
-
})
|
|
389
|
-
.filter((item) => item.id.toLowerCase().includes(searchTermLower));
|
|
390
|
-
// Separar equals filters e outros filtros
|
|
391
|
-
const equalsFilters = paramsWithoutIdFilter.arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
|
|
392
|
-
const otherFilters = paramsWithoutIdFilter.arrange.filters.filter((f) => f.arrange !== 'equals' &&
|
|
393
|
-
(f.arrange !== 'filter' || f.filter?.property !== 'id'));
|
|
394
|
-
// Aplicar equals filters com lógica OR dentro de cada propriedade
|
|
395
|
-
if (equalsFilters.length > 0) {
|
|
396
|
-
// Agrupar por propriedade para aplicar OR
|
|
397
|
-
const groupedByProperty = equalsFilters.reduce((acc, f) => {
|
|
398
|
-
const prop = f.filter.property;
|
|
399
|
-
if (!acc[prop]) {
|
|
400
|
-
acc[prop] = [];
|
|
401
|
-
}
|
|
402
|
-
acc[prop].push(f.filter.filtering);
|
|
403
|
-
return acc;
|
|
404
|
-
}, {});
|
|
405
|
-
// Filtrar: item deve ter pelo menos um valor de CADA propriedade
|
|
406
|
-
items = items.filter((item) => {
|
|
407
|
-
return Object.entries(groupedByProperty).every(([prop, values]) => {
|
|
408
|
-
const itemValue = item[prop];
|
|
409
|
-
return values.includes(itemValue);
|
|
410
|
-
});
|
|
411
|
-
});
|
|
412
|
-
}
|
|
413
|
-
// Aplicar outros filtros
|
|
414
|
-
otherFilters.forEach((filterItem) => {
|
|
415
|
-
if (filterItem.arrange === 'filter' &&
|
|
416
|
-
filterItem.filter?.filtering &&
|
|
417
|
-
filterItem.filter?.property) {
|
|
418
|
-
const filterValue = String(filterItem.filter.filtering)
|
|
419
|
-
.trim()
|
|
420
|
-
.toLowerCase();
|
|
421
|
-
items = items.filter((item) => {
|
|
422
|
-
const itemValue = String(item[filterItem.filter.property] ?? '').toLowerCase();
|
|
423
|
-
return itemValue.includes(filterValue);
|
|
424
|
-
});
|
|
425
|
-
}
|
|
426
|
-
});
|
|
427
|
-
// Aplicar filterFn se existir
|
|
428
|
-
if (params.filterFn) {
|
|
429
|
-
items = items.filter(params.filterFn);
|
|
430
|
-
}
|
|
431
|
-
// Paginação
|
|
432
|
-
const pageSize = params.batchSize;
|
|
433
|
-
let currentClientPageIndex = 0;
|
|
434
|
-
if (params.navigation === 'reload') {
|
|
435
|
-
currentClientPageIndex = 0;
|
|
436
|
-
}
|
|
437
|
-
else {
|
|
438
|
-
// Usar o índice passado pelo componente sem incrementar/decrementar
|
|
439
|
-
currentClientPageIndex = params.clientPageIndex || 0;
|
|
440
|
-
}
|
|
441
|
-
const startIndex = currentClientPageIndex * pageSize;
|
|
442
|
-
const endIndex = startIndex + pageSize;
|
|
443
|
-
const paginatedItems = items.slice(startIndex, endIndex);
|
|
444
|
-
const totalPages = Math.ceil(items.length / pageSize);
|
|
445
|
-
const hasNextPage = currentClientPageIndex < totalPages - 1;
|
|
446
|
-
const hasPreviousPage = currentClientPageIndex > 0;
|
|
447
|
-
return {
|
|
448
|
-
items: paginatedItems,
|
|
449
|
-
filterLength: items.length,
|
|
450
|
-
firstDoc: null,
|
|
451
|
-
lastDoc: null,
|
|
452
|
-
hasNextPage,
|
|
453
|
-
hasPreviousPage,
|
|
454
|
-
currentClientPageIndex,
|
|
455
|
-
totalPages,
|
|
456
|
-
};
|
|
457
|
-
}
|
|
458
|
-
shouldUseClientSideFallback(params) {
|
|
459
|
-
const hasConditions = params.conditions && params.conditions.length > 0;
|
|
460
|
-
const hasArrangeFilters = params.arrange?.filters && params.arrange.filters.length > 0;
|
|
461
|
-
const hasSortBy = params.arrange?.sortBy?.field;
|
|
462
|
-
if (params.filterFn) {
|
|
463
|
-
return false;
|
|
464
|
-
}
|
|
465
|
-
if (hasArrangeFilters) {
|
|
466
|
-
const equalsFilters = params.arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
|
|
467
|
-
if (equalsFilters.length > 0) {
|
|
468
|
-
const propertiesSet = new Set(equalsFilters.map((f) => f.filter.property));
|
|
469
|
-
if (propertiesSet.size > 1) {
|
|
470
|
-
return true;
|
|
471
|
-
}
|
|
472
|
-
}
|
|
473
|
-
}
|
|
474
|
-
if (hasConditions && hasArrangeFilters && hasSortBy) {
|
|
475
|
-
return true;
|
|
476
|
-
}
|
|
477
|
-
if (hasConditions && hasArrangeFilters) {
|
|
478
|
-
return true;
|
|
479
|
-
}
|
|
480
|
-
if (hasArrangeFilters && params.arrange.filters.length > 1 && hasSortBy) {
|
|
481
|
-
const equalsFilters = params.arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
|
|
482
|
-
if (equalsFilters.length > 0) {
|
|
483
|
-
const propertiesSet = new Set(equalsFilters.map((f) => f.filter.property));
|
|
484
|
-
if (propertiesSet.size === 1 &&
|
|
485
|
-
equalsFilters.length === params.arrange.filters.length) {
|
|
486
|
-
return false;
|
|
487
|
-
}
|
|
488
|
-
}
|
|
489
|
-
return true;
|
|
490
|
-
}
|
|
491
|
-
return false;
|
|
492
|
-
}
|
|
493
|
-
async getPaginated(params) {
|
|
494
|
-
const idFilterValue = this.getIdFilter(params);
|
|
495
|
-
if (idFilterValue) {
|
|
496
|
-
const result = await this.searchByIdPartial(params, idFilterValue);
|
|
497
|
-
return result;
|
|
498
|
-
}
|
|
499
|
-
// Detectar preventivamente se deve usar fallback
|
|
500
|
-
if (this.shouldUseClientSideFallback(params)) {
|
|
501
|
-
await this.trackMissingIndexPreventive(params.collection, params.arrange, params.conditions);
|
|
502
|
-
const result = await this.executeClientSideQuery(params);
|
|
503
|
-
return result;
|
|
504
|
-
}
|
|
505
|
-
try {
|
|
506
|
-
const result = await this.executeQuery(params);
|
|
507
|
-
return result;
|
|
508
|
-
}
|
|
509
|
-
catch (error) {
|
|
510
|
-
if (error &&
|
|
511
|
-
typeof error === 'object' &&
|
|
512
|
-
'code' in error &&
|
|
513
|
-
error.code === 'failed-precondition') {
|
|
514
|
-
await this.trackMissingIndex(error, params.collection, params.arrange, params.conditions);
|
|
515
|
-
const result = await this.executeClientSideQuery(params);
|
|
516
|
-
return result;
|
|
517
|
-
}
|
|
518
|
-
else if (error &&
|
|
519
|
-
typeof error === 'object' &&
|
|
520
|
-
'code' in error &&
|
|
521
|
-
error.code === 'invalid-argument') {
|
|
522
|
-
await this.trackMissingIndex(error, params.collection, params.arrange, params.conditions);
|
|
523
|
-
const result = await this.executeClientSideQuery(params);
|
|
524
|
-
return result;
|
|
525
|
-
}
|
|
526
|
-
else {
|
|
527
|
-
throw error;
|
|
528
|
-
}
|
|
529
|
-
}
|
|
530
|
-
}
|
|
531
|
-
async executeClientSideQuery(params) {
|
|
532
|
-
// Otimizar usando pelo menos uma cláusula .where() quando possível
|
|
533
|
-
let query = this.ngFire.collection(params.collection).ref;
|
|
534
|
-
let appliedCondition = null;
|
|
535
|
-
let hasAppliedWhereClause = false;
|
|
536
|
-
// Primeiro, tenta aplicar condições simples
|
|
537
|
-
if (params.conditions && params.conditions.length > 0) {
|
|
538
|
-
const simpleCondition = params.conditions.find((cond) => ['==', '>', '<', '>=', '<=', 'in', 'array-contains'].includes(cond.operator));
|
|
539
|
-
if (simpleCondition) {
|
|
540
|
-
query = query.where(simpleCondition.firestoreProperty, simpleCondition.operator, simpleCondition.dashProperty);
|
|
541
|
-
appliedCondition = simpleCondition;
|
|
542
|
-
hasAppliedWhereClause = true;
|
|
543
|
-
}
|
|
544
|
-
}
|
|
545
|
-
// Se não há condições disponíveis, tenta aplicar filtros do arrange
|
|
546
|
-
let appliedFirestoreFilter = null;
|
|
547
|
-
if (!hasAppliedWhereClause && params.arrange?.filters) {
|
|
548
|
-
// Agrupar equals filters por propriedade
|
|
549
|
-
const equalsFilters = params.arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
|
|
550
|
-
const equalsGroupedByProperty = equalsFilters.reduce((acc, current) => {
|
|
551
|
-
const prop = current.filter.property;
|
|
552
|
-
if (!acc[prop]) {
|
|
553
|
-
acc[prop] = [];
|
|
554
|
-
}
|
|
555
|
-
acc[prop].push(current.filter.filtering);
|
|
556
|
-
return acc;
|
|
557
|
-
}, {});
|
|
558
|
-
// Se há apenas UMA propriedade com múltiplos valores, aplicar no Firestore com 'in'
|
|
559
|
-
const properties = Object.keys(equalsGroupedByProperty);
|
|
560
|
-
if (properties.length === 1 && equalsFilters.length > 0) {
|
|
561
|
-
const prop = properties[0];
|
|
562
|
-
const values = equalsGroupedByProperty[prop];
|
|
563
|
-
query = query.where(prop, 'in', values);
|
|
564
|
-
hasAppliedWhereClause = true;
|
|
565
|
-
// Marcar TODOS os equals filters dessa propriedade como aplicados
|
|
566
|
-
appliedFirestoreFilter = 'all-equals';
|
|
567
|
-
}
|
|
568
|
-
else if (properties.length === 0) {
|
|
569
|
-
const otherFilter = params.arrange.filters.find((f) => (f.arrange === 'filter' &&
|
|
570
|
-
!!f.filter?.filtering &&
|
|
571
|
-
!!f.filter?.property) ||
|
|
572
|
-
(f.arrange === 'filterByDate' &&
|
|
573
|
-
!!f.dateFilter?.initial &&
|
|
574
|
-
!!f.dateFilter?.final));
|
|
575
|
-
if (otherFilter) {
|
|
576
|
-
if (otherFilter.arrange === 'filter' && otherFilter.filter) {
|
|
577
|
-
const filterValue = otherFilter.filter.filtering
|
|
578
|
-
.trim()
|
|
579
|
-
.toUpperCase();
|
|
580
|
-
query = query
|
|
581
|
-
.where(otherFilter.filter.property, '>=', filterValue)
|
|
582
|
-
.where(otherFilter.filter.property, '<=', filterValue + '\uf8ff');
|
|
583
|
-
hasAppliedWhereClause = true;
|
|
584
|
-
appliedFirestoreFilter = otherFilter;
|
|
585
|
-
}
|
|
586
|
-
else if (otherFilter.arrange === 'filterByDate' &&
|
|
587
|
-
otherFilter.dateFilter &&
|
|
588
|
-
params.arrange.sortBy?.field) {
|
|
589
|
-
query = query
|
|
590
|
-
.where(params.arrange.sortBy.field, '>=', otherFilter.dateFilter.initial)
|
|
591
|
-
.where(params.arrange.sortBy.field, '<=', otherFilter.dateFilter.final);
|
|
592
|
-
hasAppliedWhereClause = true;
|
|
593
|
-
appliedFirestoreFilter = otherFilter;
|
|
594
|
-
}
|
|
595
|
-
}
|
|
596
|
-
}
|
|
597
|
-
}
|
|
598
|
-
const allDocsSnapshot = await query.get();
|
|
599
|
-
let items = allDocsSnapshot.docs.map((doc) => {
|
|
600
|
-
const data = doc.data();
|
|
601
|
-
return { id: doc.id, ...(data ?? {}) };
|
|
602
|
-
});
|
|
603
|
-
// Aplicar condições restantes
|
|
604
|
-
if (params.conditions) {
|
|
605
|
-
const remainingConditions = params.conditions.filter((cond) => cond !== appliedCondition);
|
|
606
|
-
if (remainingConditions.length > 0) {
|
|
607
|
-
const operators = this.operators;
|
|
608
|
-
items = items.filter((item) => {
|
|
609
|
-
return remainingConditions.every((cond) => {
|
|
610
|
-
const operatorFn = operators[cond.operator];
|
|
611
|
-
return operatorFn
|
|
612
|
-
? operatorFn(item[cond.firestoreProperty], cond.dashProperty)
|
|
613
|
-
: false;
|
|
614
|
-
});
|
|
615
|
-
});
|
|
616
|
-
}
|
|
617
|
-
}
|
|
618
|
-
const { filters, sortBy } = params.arrange;
|
|
619
|
-
// Track which filter was already applied in Firestore to avoid double filtering
|
|
620
|
-
if (hasAppliedWhereClause && !appliedCondition && params.arrange?.filters) {
|
|
621
|
-
const equalsFilter = params.arrange.filters.find((f) => f.arrange === 'equals' && !!f.filter?.filtering);
|
|
622
|
-
if (equalsFilter) {
|
|
623
|
-
appliedFirestoreFilter = equalsFilter;
|
|
624
|
-
}
|
|
625
|
-
else {
|
|
626
|
-
const foundFilter = params.arrange.filters.find((f) => (f.arrange === 'filter' &&
|
|
627
|
-
!!f.filter?.filtering &&
|
|
628
|
-
!!f.filter?.property) ||
|
|
629
|
-
(f.arrange === 'filterByDate' &&
|
|
630
|
-
!!f.dateFilter?.initial &&
|
|
631
|
-
!!f.dateFilter?.final));
|
|
632
|
-
if (foundFilter) {
|
|
633
|
-
appliedFirestoreFilter = foundFilter;
|
|
634
|
-
}
|
|
635
|
-
}
|
|
636
|
-
}
|
|
637
|
-
const equalsFilters = filters.filter((f) => f.arrange === 'equals');
|
|
638
|
-
const otherFilters = filters.filter((f) => f.arrange !== 'equals');
|
|
639
|
-
// Aplicar equals filters no client-side apenas se não foram aplicados no Firestore
|
|
640
|
-
if (appliedFirestoreFilter !== 'all-equals' && equalsFilters.length > 0) {
|
|
641
|
-
// Agrupar por propriedade para aplicar OR dentro de cada propriedade
|
|
642
|
-
const groupedByProperty = equalsFilters.reduce((acc, f) => {
|
|
643
|
-
if (f.filter) {
|
|
644
|
-
const prop = f.filter.property;
|
|
645
|
-
if (!acc[prop]) {
|
|
646
|
-
acc[prop] = [];
|
|
647
|
-
}
|
|
648
|
-
acc[prop].push(f.filter.filtering);
|
|
649
|
-
}
|
|
650
|
-
return acc;
|
|
651
|
-
}, {});
|
|
652
|
-
// Filtrar: item deve ter pelo menos um valor de CADA propriedade
|
|
653
|
-
// (AND entre propriedades, OR dentro de cada propriedade)
|
|
654
|
-
items = items.filter((item) => {
|
|
655
|
-
return Object.entries(groupedByProperty).every(([prop, values]) => {
|
|
656
|
-
const itemValue = item[prop];
|
|
657
|
-
return values.includes(itemValue);
|
|
658
|
-
});
|
|
659
|
-
});
|
|
660
|
-
}
|
|
661
|
-
otherFilters.forEach((filterItem) => {
|
|
662
|
-
if (appliedFirestoreFilter === filterItem) {
|
|
663
|
-
return;
|
|
664
|
-
}
|
|
665
|
-
if (filterItem.arrange === 'filter' &&
|
|
666
|
-
filterItem.filter?.filtering &&
|
|
667
|
-
filterItem.filter?.property) {
|
|
668
|
-
const filterValue = String(filterItem.filter.filtering)
|
|
669
|
-
.trim()
|
|
670
|
-
.toLowerCase();
|
|
671
|
-
items = items.filter((item) => {
|
|
672
|
-
const itemValue = String(item[filterItem.filter.property] ?? '').toLowerCase();
|
|
673
|
-
return itemValue.includes(filterValue);
|
|
674
|
-
});
|
|
675
|
-
}
|
|
676
|
-
if (filterItem.arrange === 'filterByDate' &&
|
|
677
|
-
filterItem.dateFilter?.initial &&
|
|
678
|
-
filterItem.dateFilter?.final &&
|
|
679
|
-
sortBy.field) {
|
|
680
|
-
items = items.filter((item) => {
|
|
681
|
-
try {
|
|
682
|
-
const fieldValue = item[sortBy.field];
|
|
683
|
-
if (!fieldValue) {
|
|
684
|
-
return false;
|
|
685
|
-
}
|
|
686
|
-
let itemDate;
|
|
687
|
-
const fieldValueObj = fieldValue;
|
|
688
|
-
if (fieldValueObj &&
|
|
689
|
-
typeof fieldValueObj === 'object' &&
|
|
690
|
-
'toDate' in fieldValueObj &&
|
|
691
|
-
typeof fieldValueObj.toDate === 'function') {
|
|
692
|
-
itemDate = fieldValueObj.toDate();
|
|
693
|
-
}
|
|
694
|
-
else if (fieldValueObj instanceof Date) {
|
|
695
|
-
itemDate = fieldValueObj;
|
|
696
|
-
}
|
|
697
|
-
else if (typeof fieldValueObj === 'string') {
|
|
698
|
-
itemDate = new Date(fieldValueObj);
|
|
699
|
-
if (isNaN(itemDate.getTime())) {
|
|
700
|
-
return false;
|
|
701
|
-
}
|
|
702
|
-
}
|
|
703
|
-
else if (typeof fieldValueObj === 'number') {
|
|
704
|
-
itemDate = new Date(fieldValueObj);
|
|
705
|
-
}
|
|
706
|
-
else {
|
|
707
|
-
return false;
|
|
708
|
-
}
|
|
709
|
-
if (!filterItem.dateFilter) {
|
|
710
|
-
return false;
|
|
711
|
-
}
|
|
712
|
-
return (itemDate >= filterItem.dateFilter.initial &&
|
|
713
|
-
itemDate <= filterItem.dateFilter.final);
|
|
714
|
-
}
|
|
715
|
-
catch (error) {
|
|
716
|
-
console.warn('Erro ao processar filtro de data para o item:', item.id, error);
|
|
717
|
-
return false;
|
|
718
|
-
}
|
|
719
|
-
});
|
|
720
|
-
}
|
|
721
|
-
});
|
|
722
|
-
// Aplicar filterFn se existir
|
|
723
|
-
if (params.filterFn) {
|
|
724
|
-
items = items.filter(params.filterFn);
|
|
725
|
-
}
|
|
726
|
-
if (sortBy && sortBy.field && sortBy.order) {
|
|
727
|
-
items.sort((a, b) => {
|
|
728
|
-
const valA = a[sortBy.field];
|
|
729
|
-
const valB = b[sortBy.field];
|
|
730
|
-
if (valA === null || valA === undefined)
|
|
731
|
-
return 1;
|
|
732
|
-
if (valB === null || valB === undefined)
|
|
733
|
-
return -1;
|
|
734
|
-
if (valA < valB) {
|
|
735
|
-
return sortBy.order === 'asc' ? -1 : 1;
|
|
736
|
-
}
|
|
737
|
-
if (valA > valB) {
|
|
738
|
-
return sortBy.order === 'asc' ? 1 : -1;
|
|
739
|
-
}
|
|
740
|
-
return 0;
|
|
741
|
-
});
|
|
742
|
-
}
|
|
743
|
-
// Implementação adequada da paginação
|
|
744
|
-
let currentClientPageIndex = 0;
|
|
745
|
-
// Determinar a página atual baseada na navegação
|
|
746
|
-
if (params.navigation === 'reload') {
|
|
747
|
-
currentClientPageIndex = 0;
|
|
748
|
-
}
|
|
749
|
-
else {
|
|
750
|
-
currentClientPageIndex = params.clientPageIndex || 0;
|
|
751
|
-
}
|
|
752
|
-
const pageSize = params.batchSize;
|
|
753
|
-
const startIndex = currentClientPageIndex * pageSize;
|
|
754
|
-
const endIndex = startIndex + pageSize;
|
|
755
|
-
const paginatedItems = items.slice(startIndex, endIndex);
|
|
756
|
-
const totalPages = Math.ceil(items.length / pageSize);
|
|
757
|
-
const hasNextPage = currentClientPageIndex < totalPages - 1;
|
|
758
|
-
const hasPreviousPage = currentClientPageIndex > 0;
|
|
759
|
-
return {
|
|
760
|
-
items: paginatedItems,
|
|
761
|
-
filterLength: items.length,
|
|
762
|
-
lastDoc: null,
|
|
763
|
-
firstDoc: null,
|
|
764
|
-
hasNextPage: hasNextPage,
|
|
765
|
-
hasPreviousPage: hasPreviousPage,
|
|
766
|
-
currentClientPageIndex: currentClientPageIndex,
|
|
767
|
-
totalPages: totalPages,
|
|
768
|
-
};
|
|
769
|
-
}
|
|
770
|
-
async getItemsData(collection, arrange, conditions = undefined) {
|
|
771
|
-
try {
|
|
772
|
-
let query = this.ngFire.collection(collection).ref;
|
|
773
|
-
query = this.applyFilters(query, arrange, conditions);
|
|
774
|
-
const snapshot = await query.get();
|
|
775
|
-
return await Promise.all(snapshot.docs.map(async (doc) => {
|
|
776
|
-
const data = doc.data();
|
|
777
|
-
const id = doc.id;
|
|
778
|
-
return {
|
|
779
|
-
id,
|
|
780
|
-
...(data ?? {}),
|
|
781
|
-
};
|
|
782
|
-
}));
|
|
783
|
-
}
|
|
784
|
-
catch (e) {
|
|
785
|
-
throw e;
|
|
786
|
-
}
|
|
787
|
-
}
|
|
788
|
-
async deleteIndex(id, col) {
|
|
789
|
-
try {
|
|
790
|
-
const batch = this.ngFire.firestore.batch();
|
|
791
|
-
const docRef = this.ngFire.collection(col).doc(id);
|
|
792
|
-
const docSnapshot = (await firstValueFrom(docRef.get()));
|
|
793
|
-
const doc = docSnapshot.data();
|
|
794
|
-
batch.delete(docRef.ref);
|
|
795
|
-
if (doc && typeof doc['index'] === 'number') {
|
|
796
|
-
await this.reindex(doc['index'], col, batch);
|
|
797
|
-
}
|
|
798
|
-
await batch.commit();
|
|
799
|
-
this.toastr.success('Item excluído com sucesso!');
|
|
800
|
-
return true;
|
|
801
|
-
}
|
|
802
|
-
catch (e) {
|
|
803
|
-
const error = e;
|
|
804
|
-
console.error('Erro ao deletar item:', error);
|
|
805
|
-
this.toastr.error('Erro ao deletar item.');
|
|
806
|
-
return false;
|
|
807
|
-
}
|
|
808
|
-
}
|
|
809
|
-
async reindex(index, col, batch) {
|
|
810
|
-
try {
|
|
811
|
-
const snapshot = (await firstValueFrom(this.ngFire.collection(col).get()));
|
|
812
|
-
const docs = snapshot.docs;
|
|
813
|
-
for (let doc of docs) {
|
|
814
|
-
const data = doc.data();
|
|
815
|
-
if (data &&
|
|
816
|
-
typeof data['index'] === 'number' &&
|
|
817
|
-
data['index'] > index) {
|
|
818
|
-
data['index'] = (data['index'] - 1);
|
|
819
|
-
const docRef = this.ngFire.collection(col).doc(doc.id).ref;
|
|
820
|
-
batch.update(docRef, data);
|
|
821
|
-
}
|
|
822
|
-
}
|
|
823
|
-
}
|
|
824
|
-
catch (error) {
|
|
825
|
-
console.error('Erro ao reindexar:', error);
|
|
826
|
-
}
|
|
827
|
-
return;
|
|
828
|
-
}
|
|
829
|
-
dateFormatValidator() {
|
|
830
|
-
return (control) => {
|
|
831
|
-
if (!control.value) {
|
|
832
|
-
return null;
|
|
833
|
-
}
|
|
834
|
-
const dateStr = control.value.trim();
|
|
835
|
-
const datePattern = /^(\d{1,2})\/(\d{1,2})\/(\d{4})$/;
|
|
836
|
-
if (!datePattern.test(dateStr)) {
|
|
837
|
-
return { invalidFormat: true };
|
|
838
|
-
}
|
|
839
|
-
const parts = dateStr.split('/');
|
|
840
|
-
const day = parts[0].padStart(2, '0');
|
|
841
|
-
const month = parts[1].padStart(2, '0');
|
|
842
|
-
const year = parts[2];
|
|
843
|
-
const normalizedDate = `${day}/${month}/${year}`;
|
|
844
|
-
const date = moment(normalizedDate, 'DD/MM/YYYY', true);
|
|
845
|
-
if (!date.isValid()) {
|
|
846
|
-
return { invalidDate: true };
|
|
847
|
-
}
|
|
848
|
-
return null;
|
|
849
|
-
};
|
|
850
|
-
}
|
|
851
|
-
async updateIndex(index, id, col) {
|
|
852
|
-
await this.ngFire.collection(col).doc(id).update({ index });
|
|
853
|
-
}
|
|
854
|
-
/**
|
|
855
|
-
* Extrai o link de criação de índice da mensagem de erro do Firestore
|
|
856
|
-
*/
|
|
857
|
-
extractIndexLink(error) {
|
|
858
|
-
if (!error ||
|
|
859
|
-
typeof error !== 'object' ||
|
|
860
|
-
!('message' in error) ||
|
|
861
|
-
typeof error.message !== 'string')
|
|
862
|
-
return null;
|
|
863
|
-
const linkMatch = error.message.match(/(https:\/\/console\.firebase\.google\.com\/[^\s]+)/);
|
|
864
|
-
return linkMatch ? linkMatch[1] : null;
|
|
865
|
-
}
|
|
866
|
-
/**
|
|
867
|
-
* Rastreia índices ausentes ao usar fallback preventivo
|
|
868
|
-
*/
|
|
869
|
-
async trackMissingIndexPreventive(collection, arrange, conditions = undefined) {
|
|
870
|
-
try {
|
|
871
|
-
const querySignature = this.generateQuerySignature(collection, arrange, conditions);
|
|
872
|
-
const docId = `${collection}_${querySignature}`;
|
|
873
|
-
const indexLink = this.generateIndexLink(collection, arrange, conditions);
|
|
874
|
-
const indexInstructions = this.generateIndexInstructions(collection, arrange, conditions);
|
|
875
|
-
const trackingData = {
|
|
876
|
-
collection,
|
|
877
|
-
indexLink,
|
|
878
|
-
indexInstructions,
|
|
879
|
-
arrange: {
|
|
880
|
-
sortBy: arrange.sortBy,
|
|
881
|
-
filters: arrange.filters?.map((f) => ({
|
|
882
|
-
arrange: f.arrange,
|
|
883
|
-
property: f.filter?.property || null,
|
|
884
|
-
dateField: f.arrange === 'filterByDate' ? arrange.sortBy?.field : null,
|
|
885
|
-
})) || [],
|
|
886
|
-
},
|
|
887
|
-
conditions: conditions?.map((c) => ({
|
|
888
|
-
property: c.firestoreProperty,
|
|
889
|
-
operator: c.operator,
|
|
890
|
-
})) || [],
|
|
891
|
-
errorMessage: `Fallback preventivo usado para a collection ${collection}. A query exigiria índice composto.`,
|
|
892
|
-
updatedAt: firebase.firestore.FieldValue.serverTimestamp(),
|
|
893
|
-
};
|
|
894
|
-
const docRef = this.ngFire.collection('missingIndexes').doc(docId);
|
|
895
|
-
const doc = await docRef.get().toPromise();
|
|
896
|
-
if (doc && doc.exists) {
|
|
897
|
-
await docRef.update({
|
|
898
|
-
count: firebase.firestore.FieldValue.increment(1),
|
|
899
|
-
updatedAt: firebase.firestore.FieldValue.serverTimestamp(),
|
|
900
|
-
lastError: trackingData['errorMessage'],
|
|
901
|
-
});
|
|
902
|
-
}
|
|
903
|
-
else {
|
|
904
|
-
await docRef.set({
|
|
905
|
-
...trackingData,
|
|
906
|
-
count: 1,
|
|
907
|
-
createdAt: firebase.firestore.FieldValue.serverTimestamp(),
|
|
908
|
-
});
|
|
909
|
-
}
|
|
910
|
-
}
|
|
911
|
-
catch (trackingError) {
|
|
912
|
-
console.warn('Falha ao rastrear fallback preventivo:', trackingError);
|
|
913
|
-
}
|
|
914
|
-
}
|
|
915
|
-
/**
|
|
916
|
-
* Gera uma assinatura única para uma query
|
|
917
|
-
*/
|
|
918
|
-
generateQuerySignature(collection, arrange, conditions = undefined) {
|
|
919
|
-
const signature = {
|
|
920
|
-
collection,
|
|
921
|
-
sortBy: arrange.sortBy,
|
|
922
|
-
filters: arrange.filters?.map((f) => ({
|
|
923
|
-
arrange: f.arrange,
|
|
924
|
-
property: f.filter?.property || null,
|
|
925
|
-
})) || [],
|
|
926
|
-
conditions: conditions?.map((c) => ({
|
|
927
|
-
property: c.firestoreProperty,
|
|
928
|
-
operator: c.operator,
|
|
929
|
-
})) || [],
|
|
930
|
-
};
|
|
931
|
-
return btoa(JSON.stringify(signature))
|
|
932
|
-
.replace(/[^a-zA-Z0-9]/g, '')
|
|
933
|
-
.substring(0, 20);
|
|
934
|
-
}
|
|
935
|
-
/**
|
|
936
|
-
* Gera instruções claras para criar o índice manualmente
|
|
937
|
-
*/
|
|
938
|
-
generateIndexInstructions(collection, arrange, conditions = undefined) {
|
|
939
|
-
const instructions = {
|
|
940
|
-
summary: '',
|
|
941
|
-
collection: collection,
|
|
942
|
-
fields: [],
|
|
943
|
-
queryExample: '',
|
|
944
|
-
stepByStep: [],
|
|
945
|
-
notes: [],
|
|
946
|
-
};
|
|
947
|
-
const fields = [];
|
|
948
|
-
if (conditions && conditions.length > 0) {
|
|
949
|
-
conditions.forEach((condition) => {
|
|
950
|
-
if (condition.firestoreProperty) {
|
|
951
|
-
fields.push({
|
|
952
|
-
field: condition.firestoreProperty,
|
|
953
|
-
order: 'Ascending',
|
|
954
|
-
type: 'WHERE clause',
|
|
955
|
-
operator: condition.operator,
|
|
956
|
-
description: `Filtrar por ${condition.firestoreProperty} usando operador ${condition.operator}`,
|
|
957
|
-
});
|
|
958
|
-
}
|
|
959
|
-
});
|
|
960
|
-
}
|
|
961
|
-
if (arrange.filters && arrange.filters.length > 0) {
|
|
962
|
-
arrange.filters.forEach((filter) => {
|
|
963
|
-
if (filter.filter?.property) {
|
|
964
|
-
fields.push({
|
|
965
|
-
field: filter.filter.property,
|
|
966
|
-
order: 'Ascending',
|
|
967
|
-
type: 'WHERE clause (filter)',
|
|
968
|
-
operator: filter.arrange === 'filter' ? 'CONTAINS' : 'RANGE',
|
|
969
|
-
description: `Filtrar por ${filter.filter.property} usando filtro ${filter.arrange}`,
|
|
970
|
-
});
|
|
971
|
-
}
|
|
972
|
-
});
|
|
973
|
-
}
|
|
974
|
-
if (arrange.sortBy?.field) {
|
|
975
|
-
fields.push({
|
|
976
|
-
field: arrange.sortBy.field,
|
|
977
|
-
order: arrange.sortBy.order === 'desc' ? 'Descending' : 'Ascending',
|
|
978
|
-
type: 'ORDER BY clause',
|
|
979
|
-
operator: 'N/A',
|
|
980
|
-
description: `Ordenar resultados por ${arrange.sortBy.field} em ordem ${arrange.sortBy.order}`,
|
|
981
|
-
});
|
|
982
|
-
}
|
|
983
|
-
instructions.fields = fields;
|
|
984
|
-
const fieldNames = fields.map((f) => f.field).join(' + ');
|
|
985
|
-
instructions.summary = `Criar índice composto para ${collection}: ${fieldNames}`;
|
|
986
|
-
let queryExample = `db.collection('${collection}')`;
|
|
987
|
-
fields.forEach((field, index) => {
|
|
988
|
-
if (field.type.includes('WHERE')) {
|
|
989
|
-
if (field.operator === '==') {
|
|
990
|
-
queryExample += `\n .where('${field.field}', '==', 'value')`;
|
|
991
|
-
}
|
|
992
|
-
else if (field.operator === 'CONTAINS') {
|
|
993
|
-
queryExample += `\n .where('${field.field}', '>=', 'searchText')`;
|
|
994
|
-
}
|
|
995
|
-
else {
|
|
996
|
-
queryExample += `\n .where('${field.field}', '${field.operator}', 'value')`;
|
|
997
|
-
}
|
|
998
|
-
}
|
|
999
|
-
});
|
|
1000
|
-
const orderByField = fields.find((f) => f.type.includes('ORDER BY'));
|
|
1001
|
-
if (orderByField) {
|
|
1002
|
-
queryExample += `\n .orderBy('${orderByField.field}', '${orderByField.order.toLowerCase()}')`;
|
|
1003
|
-
}
|
|
1004
|
-
instructions.queryExample = queryExample;
|
|
1005
|
-
instructions.stepByStep = [
|
|
1006
|
-
'1. Ir para Firebase Console → Firestore → Indexes',
|
|
1007
|
-
'2. Clicar em "Create Index"',
|
|
1008
|
-
`3. Definir Collection ID: ${collection}`,
|
|
1009
|
-
'4. Configurar campos nesta ORDEM EXATA:',
|
|
1010
|
-
...fields.map((field, index) => ` ${index + 1}. Campo: ${field.field}, Order: ${field.order}, Array: No`),
|
|
1011
|
-
'5. Definir Query scopes: Collection',
|
|
1012
|
-
'6. Clicar em "Create" e aguardar conclusão',
|
|
1013
|
-
];
|
|
1014
|
-
instructions.notes = [
|
|
1015
|
-
'⚠️ A ordem dos campos é CRÍTICA - deve corresponder exatamente à ordem da query',
|
|
1016
|
-
'⚠️ As cláusulas WHERE devem vir ANTES do campo ORDER BY',
|
|
1017
|
-
'⚠️ Este índice só funcionará para queries com esta combinação EXATA de campos',
|
|
1018
|
-
'⚠️ A criação do índice pode levar vários minutos',
|
|
1019
|
-
];
|
|
1020
|
-
return instructions;
|
|
1021
|
-
}
|
|
1022
|
-
/**
|
|
1023
|
-
* Gera um link de índice baseado na estrutura da query
|
|
1024
|
-
*/
|
|
1025
|
-
generateIndexLink(collection, arrange, conditions = undefined) {
|
|
1026
|
-
try {
|
|
1027
|
-
const indexFields = [];
|
|
1028
|
-
if (conditions && conditions.length > 0) {
|
|
1029
|
-
conditions.forEach((condition) => {
|
|
1030
|
-
if (condition.firestoreProperty) {
|
|
1031
|
-
indexFields.push(condition.firestoreProperty);
|
|
1032
|
-
}
|
|
1033
|
-
});
|
|
1034
|
-
}
|
|
1035
|
-
if (arrange.filters && arrange.filters.length > 0) {
|
|
1036
|
-
arrange.filters.forEach((filter) => {
|
|
1037
|
-
if (filter.filter?.property) {
|
|
1038
|
-
indexFields.push(filter.filter.property);
|
|
1039
|
-
}
|
|
1040
|
-
});
|
|
1041
|
-
}
|
|
1042
|
-
if (arrange.sortBy?.field) {
|
|
1043
|
-
indexFields.push(arrange.sortBy.field);
|
|
1044
|
-
}
|
|
1045
|
-
if (indexFields.length > 1) {
|
|
1046
|
-
// Obter o projectId dinamicamente do Firebase
|
|
1047
|
-
const options = this.ngFire?.firestore?.app?.options;
|
|
1048
|
-
const projectId = options?.['projectId'];
|
|
1049
|
-
if (!projectId) {
|
|
1050
|
-
console.warn('Não foi possível obter o projectId do Firebase');
|
|
1051
|
-
return null;
|
|
1052
|
-
}
|
|
1053
|
-
const baseUrl = `https://console.firebase.google.com/project/${projectId}/firestore/indexes`;
|
|
1054
|
-
const queryParams = new URLSearchParams({
|
|
1055
|
-
create_composite: `collection=${collection}&fields=${indexFields.join(',')}`,
|
|
1056
|
-
});
|
|
1057
|
-
const finalLink = `${baseUrl}?${queryParams.toString()}`;
|
|
1058
|
-
return finalLink;
|
|
1059
|
-
}
|
|
1060
|
-
return null;
|
|
1061
|
-
}
|
|
1062
|
-
catch (error) {
|
|
1063
|
-
console.warn('Falha ao gerar link de índice:', error);
|
|
1064
|
-
return null;
|
|
1065
|
-
}
|
|
1066
|
-
}
|
|
1067
|
-
async trackMissingIndex(error, collection, arrange, conditions = undefined) {
|
|
1068
|
-
try {
|
|
1069
|
-
const indexLink = this.extractIndexLink(error);
|
|
1070
|
-
if (!indexLink)
|
|
1071
|
-
return;
|
|
1072
|
-
const linkHash = btoa(indexLink)
|
|
1073
|
-
.replace(/[^a-zA-Z0-9]/g, '')
|
|
1074
|
-
.substring(0, 20);
|
|
1075
|
-
const docId = `${collection}_${linkHash}`;
|
|
1076
|
-
const indexInstructions = this.generateIndexInstructions(collection, arrange, conditions);
|
|
1077
|
-
const trackingData = {
|
|
1078
|
-
collection,
|
|
1079
|
-
indexLink,
|
|
1080
|
-
indexInstructions,
|
|
1081
|
-
arrange: {
|
|
1082
|
-
sortBy: arrange.sortBy,
|
|
1083
|
-
filters: arrange.filters?.map((f) => ({
|
|
1084
|
-
arrange: f.arrange,
|
|
1085
|
-
property: f.filter?.property || null,
|
|
1086
|
-
dateField: f.arrange === 'filterByDate' ? arrange.sortBy?.field : null,
|
|
1087
|
-
})) || [],
|
|
1088
|
-
},
|
|
1089
|
-
conditions: conditions?.map((c) => ({
|
|
1090
|
-
property: c.firestoreProperty,
|
|
1091
|
-
operator: c.operator,
|
|
1092
|
-
})) || [],
|
|
1093
|
-
errorMessage: error &&
|
|
1094
|
-
typeof error === 'object' &&
|
|
1095
|
-
'message' in error &&
|
|
1096
|
-
typeof error.message === 'string'
|
|
1097
|
-
? error.message
|
|
1098
|
-
: String(error),
|
|
1099
|
-
updatedAt: firebase.firestore.FieldValue.serverTimestamp(),
|
|
1100
|
-
};
|
|
1101
|
-
const docRef = this.ngFire.collection('missingIndexes').doc(docId);
|
|
1102
|
-
const doc = await docRef.get().toPromise();
|
|
1103
|
-
const errorMessage = error &&
|
|
1104
|
-
typeof error === 'object' &&
|
|
1105
|
-
'message' in error &&
|
|
1106
|
-
typeof error.message === 'string'
|
|
1107
|
-
? error.message
|
|
1108
|
-
: String(error);
|
|
1109
|
-
if (doc && doc.exists) {
|
|
1110
|
-
await docRef.update({
|
|
1111
|
-
count: firebase.firestore.FieldValue.increment(1),
|
|
1112
|
-
updatedAt: firebase.firestore.FieldValue.serverTimestamp(),
|
|
1113
|
-
lastError: errorMessage,
|
|
1114
|
-
});
|
|
1115
|
-
}
|
|
1116
|
-
else {
|
|
1117
|
-
await docRef.set({
|
|
1118
|
-
...trackingData,
|
|
1119
|
-
count: 1,
|
|
1120
|
-
createdAt: firebase.firestore.FieldValue.serverTimestamp(),
|
|
1121
|
-
});
|
|
1122
|
-
}
|
|
1123
|
-
}
|
|
1124
|
-
catch (trackingError) {
|
|
1125
|
-
console.warn('Falha ao rastrear índice ausente:', trackingError);
|
|
1126
|
-
}
|
|
1127
|
-
}
|
|
1128
|
-
}
|
|
1129
|
-
TableService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableService, deps: [{ token: i1.AngularFirestore, optional: true }, { token: i2.MatDialog, optional: true }, { token: i3.ToastrService, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
1130
|
-
TableService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableService });
|
|
1131
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableService, decorators: [{
|
|
1132
|
-
type: Injectable
|
|
1133
|
-
}], ctorParameters: function () { return [{ type: i1.AngularFirestore, decorators: [{
|
|
1134
|
-
type: Optional
|
|
1135
|
-
}] }, { type: i2.MatDialog, decorators: [{
|
|
1136
|
-
type: Optional
|
|
1137
|
-
}] }, { type: i3.ToastrService, decorators: [{
|
|
1138
|
-
type: Optional
|
|
1139
|
-
}] }]; } });
|
|
1140
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25nLWZpcmViYXNlLXRhYmxlLWt4cC9zcmMvbGliL3NlcnZpY2VzL3RhYmxlLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFNckQsT0FBTyxRQUFRLE1BQU0scUJBQXFCLENBQUM7QUFFM0MsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUl0QyxPQUFPLEtBQUssTUFBTSxNQUFNLFFBQVEsQ0FBQzs7Ozs7QUFzQmpDLE1BQU0sT0FBTyxZQUFZO0lBQ3ZCLFlBQ3NCLE1BQXdCLEVBQ3hCLE1BQWlCLEVBQ2pCLE1BQXFCO1FBRnJCLFdBQU0sR0FBTixNQUFNLENBQWtCO1FBQ3hCLFdBQU0sR0FBTixNQUFNLENBQVc7UUFDakIsV0FBTSxHQUFOLE1BQU0sQ0FBZTtRQSttQ3BDLGNBQVMsR0FBRztZQUNqQixJQUFJLEVBQUUsQ0FBQyxDQUFVLEVBQUUsQ0FBVSxFQUFXLEVBQUUsQ0FBQyxDQUFDLEtBQUssQ0FBQztZQUNsRCxJQUFJLEVBQUUsQ0FBQyxDQUFVLEVBQUUsQ0FBVSxFQUFXLEVBQUUsQ0FBQyxDQUFDLEtBQUssQ0FBQztZQUNsRCxHQUFHLEVBQUUsQ0FBQyxDQUFVLEVBQUUsQ0FBVSxFQUFXLEVBQUUsQ0FBRSxDQUFZLEdBQUksQ0FBWTtZQUN2RSxHQUFHLEVBQUUsQ0FBQyxDQUFVLEVBQUUsQ0FBVSxFQUFXLEVBQUUsQ0FBRSxDQUFZLEdBQUksQ0FBWTtZQUN2RSxJQUFJLEVBQUUsQ0FBQyxDQUFVLEVBQUUsQ0FBVSxFQUFXLEVBQUUsQ0FBRSxDQUFZLElBQUssQ0FBWTtZQUN6RSxJQUFJLEVBQUUsQ0FBQyxDQUFVLEVBQUUsQ0FBVSxFQUFXLEVBQUUsQ0FBRSxDQUFZLElBQUssQ0FBWTtZQUN6RSxF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QUFFLEtBQTBCLENBQUMsT0FBTztnQkFDckMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUVwQixJQUFJLEdBQUcsSUFBSSxHQUFHLENBQUMsTUFBTSxFQUFFO2dCQUNyQixNQUFNLE1BQU0sQ0FBQyxNQUFNLENBQUM7b0JBQ2xCLEtBQUssRUFBRSxRQUFRLENBQUMsU0FBUyxDQUFDLFVBQVUsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDO29CQUNqRCxTQUFTLEVBQUUsUUFBUSxDQUFDLFNBQVMsQ0FBQyxVQUFVLENBQUMsZUFBZSxFQUFFO29CQUMxRCxTQUFTLEVBQUUsWUFBWTtpQkFDeEIsQ0FBQyxDQUFDO2FBQ0o7aUJBQU07Z0JBQ0wsTUFBTSxNQUFNLENBQUMsR0FBRyxDQUFDO29CQUNmLEdBQUcsWUFBWTtvQkFDZixLQUFLLEVBQUUsQ0FBQztvQkFDUixTQUFTLEVBQUUsUUFBUSxDQUFDLFNBQVMsQ0FBQyxVQUFVLENBQUMsZUFBZSxFQUFFO2lCQUMzRCxDQUFDLENBQUM7YUFDSjtTQUNGO1FBQUMsT0FBTyxhQUFhLEVBQUU7WUFDdEIsT0FBTyxDQUFDLElBQUksQ0FBQyxtQ0FBbUMsRUFBRSxhQUFhLENBQUMsQ0FBQztTQUNsRTtJQUNILENBQUM7OzBHQTltRFUsWUFBWTs4R0FBWixZQUFZOzRGQUFaLFlBQVk7a0JBRHhCLFVBQVU7OzBCQUdOLFFBQVE7OzBCQUNSLFFBQVE7OzBCQUNSLFFBQVEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlLCBPcHRpb25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQge1xyXG4gIEFuZ3VsYXJGaXJlc3RvcmUsXHJcbiAgQ29sbGVjdGlvblJlZmVyZW5jZSxcclxuICBRdWVyeURvY3VtZW50U25hcHNob3QsXHJcbn0gZnJvbSAnQGFuZ3VsYXIvZmlyZS9jb21wYXQvZmlyZXN0b3JlJztcclxuaW1wb3J0IGZpcmViYXNlIGZyb20gJ2ZpcmViYXNlL2NvbXBhdC9hcHAnO1xyXG5pbXBvcnQgeyBBcnJhbmdlLCBDb25kaXRpb24sIFBhZ2luYXRpb24gfSBmcm9tICcuLi90eXBlcy9UYWJsZSc7XHJcbmltcG9ydCB7IGZpcnN0VmFsdWVGcm9tIH0gZnJvbSAncnhqcyc7XHJcbmltcG9ydCB7IE1hdERpYWxvZyB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2RpYWxvZyc7XHJcbmltcG9ydCB7IFRvYXN0clNlcnZpY2UgfSBmcm9tICduZ3gtdG9hc3RyJztcclxuaW1wb3J0IHsgQWJzdHJhY3RDb250cm9sLCBWYWxpZGF0b3JGbiB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcclxuaW1wb3J0ICogYXMgbW9tZW50IGZyb20gJ21vbWVudCc7XHJcblxyXG5pbnRlcmZhY2UgRXJyb3JXaXRoQ29kZSBleHRlbmRzIEVycm9yIHtcclxuICBjb2RlOiBzdHJpbmc7XHJcbn1cclxuXHJcbmludGVyZmFjZSBFcnJvcldpdGhNZXNzYWdlIHtcclxuICBtZXNzYWdlOiBzdHJpbmc7XHJcbn1cclxuXHJcbmludGVyZmFjZSBQYWdpbmF0aW9uUmVzdWx0IHtcclxuICBpdGVtczogQXJyYXk8UmVjb3JkPHN0cmluZywgdW5rbm93bj4gJiB7IGlkOiBzdHJpbmcgfT47XHJcbiAgZmlsdGVyTGVuZ3RoOiBudW1iZXIgfCBudWxsO1xyXG4gIGZpcnN0RG9jOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+IHwgbnVsbDtcclxuICBsYXN0RG9jOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+IHwgbnVsbDtcclxuICBoYXNOZXh0UGFnZTogYm9vbGVhbjtcclxuICBoYXNQcmV2aW91c1BhZ2U/OiBib29sZWFuO1xyXG4gIGN1cnJlbnRDbGllbnRQYWdlSW5kZXg/OiBudW1iZXI7XHJcbiAgdG90YWxQYWdlcz86IG51bWJlcjtcclxufVxyXG5cclxuQEluamVjdGFibGUoKVxyXG5leHBvcnQgY2xhc3MgVGFibGVTZXJ2aWNlIHtcclxuICBjb25zdHJ1Y3RvcihcclxuICAgIEBPcHRpb25hbCgpIHByaXZhdGUgbmdGaXJlOiBBbmd1bGFyRmlyZXN0b3JlLFxyXG4gICAgQE9wdGlvbmFsKCkgcHJpdmF0ZSBkaWFsb2c6IE1hdERpYWxvZyxcclxuICAgIEBPcHRpb25hbCgpIHByaXZhdGUgdG9hc3RyOiBUb2FzdHJTZXJ2aWNlXHJcbiAgKSB7fVxyXG5cclxuICBhc3luYyBnZXRJdGVtcyhcclxuICAgIGNvbGxlY3Rpb246IENvbGxlY3Rpb25SZWZlcmVuY2U8dW5rbm93bj5cclxuICApOiBQcm9taXNlPEFycmF5PFJlY29yZDxzdHJpbmcsIHVua25vd24+ICYgeyBpZDogc3RyaW5nIH0+PiB7XHJcbiAgICB0cnkge1xyXG4gICAgICBjb25zdCBxdWVyeVNuYXBzaG90ID0gYXdhaXQgY29sbGVjdGlvbi5nZXQoKTtcclxuICAgICAgcmV0dXJuIHF1ZXJ5U25hcHNob3QuZG9jcy5tYXAoXHJcbiAgICAgICAgKGRvYzogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5RG9jdW1lbnRTbmFwc2hvdDx1bmtub3duPikgPT4ge1xyXG4gICAgICAgICAgY29uc3QgZGF0YSA9IGRvYy5kYXRhKCkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gfCB1bmRlZmluZWQ7XHJcbiAgICAgICAgICByZXR1cm4geyAuLi4oZGF0YSA/PyB7fSksIGlkOiBkb2MuaWQgfSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiAmIHtcclxuICAgICAgICAgICAgaWQ6IHN0cmluZztcclxuICAgICAgICAgIH07XHJcbiAgICAgICAgfVxyXG4gICAgICApO1xyXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgY29uc29sZS53YXJuKCdDb2xsZWN0aW9uIG7Do28gZW5jb250cmFkYTonLCBlcnJvcik7XHJcbiAgICAgIHJldHVybiBbXTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHByaXZhdGUgYXN5bmMgZXhlY3V0ZVF1ZXJ5KHBhcmFtczogUGFnaW5hdGlvbik6IFByb21pc2U8UGFnaW5hdGlvblJlc3VsdD4ge1xyXG4gICAgaWYgKHBhcmFtcy5maWx0ZXJGbikge1xyXG4gICAgICAvLyBMw7NnaWNhIGNvbSBmaWx0cm8gbm8gY2xpZW50ZSAoZmlsdGVyRm4pXHJcbiAgICAgIGNvbnN0IEJBVENIX0ZFVENIX1NJWkUgPSBwYXJhbXMuYmF0Y2hTaXplO1xyXG4gICAgICBjb25zdCBHT0FMX1NJWkUgPSBwYXJhbXMuYmF0Y2hTaXplICsgMTtcclxuXHJcbiAgICAgIGlmIChwYXJhbXMubmF2aWdhdGlvbiA9PT0gJ2ZvcndhcmQnIHx8IHBhcmFtcy5uYXZpZ2F0aW9uID09PSAncmVsb2FkJykge1xyXG4gICAgICAgIGlmIChwYXJhbXMubmF2aWdhdGlvbiA9PT0gJ3JlbG9hZCcgJiYgcGFyYW1zLmRvYykge1xyXG4gICAgICAgICAgcGFyYW1zLmRvYy5sYXN0RG9jID0gbnVsbDtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGxldCBsYXN0RG9jQ3Vyc29yOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+IHwgbnVsbCA9XHJcbiAgICAgICAgICBwYXJhbXMuZG9jID8gcGFyYW1zLmRvYy5sYXN0RG9jIDogbnVsbDtcclxuICAgICAgICBsZXQgcGFnZVJlc3VsdHM6IEFycmF5PFJlY29yZDxzdHJpbmcsIHVua25vd24+ICYgeyBpZDogc3RyaW5nIH0+ID0gW107XHJcbiAgICAgICAgbGV0IGFsbEZldGNoZWREb2NzOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+W10gPVxyXG4gICAgICAgICAgW107XHJcbiAgICAgICAgbGV0IGhhc01vcmVEb2NzSW5EYiA9IHRydWU7XHJcblxyXG4gICAgICAgIHdoaWxlIChwYWdlUmVzdWx0cy5sZW5ndGggPCBHT0FMX1NJWkUgJiYgaGFzTW9yZURvY3NJbkRiKSB7XHJcbiAgICAgICAgICBsZXQgcXVlcnk6IGZpcmViYXNlLmZpcmVzdG9yZS5RdWVyeTx1bmtub3duPiA9IHRoaXMubmdGaXJlLmNvbGxlY3Rpb24oXHJcbiAgICAgICAgICAgIHBhcmFtcy5jb2xsZWN0aW9uXHJcbiAgICAgICAgICApLnJlZjtcclxuICAgICAgICAgIHF1ZXJ5ID0gdGhpcy5hcHBseUZpbHRlcnMocXVlcnksIHBhcmFtcy5hcnJhbmdlLCBwYXJhbXMuY29uZGl0aW9ucyk7XHJcblxyXG4gICAgICAgICAgaWYgKGxhc3REb2NDdXJzb3IpIHtcclxuICAgICAgICAgICAgcXVlcnkgPSBxdWVyeS5zdGFydEFmdGVyKGxhc3REb2NDdXJzb3IpO1xyXG4gICAgICAgICAgfVxyXG4gICAgICAgICAgcXVlcnkgPSBxdWVyeS5saW1pdChCQVRDSF9GRVRDSF9TSVpFKTtcclxuXHJcbiAgICAgICAgICBjb25zdCBzbmFwc2hvdCA9IGF3YWl0IHF1ZXJ5LmdldCgpO1xyXG5cclxuICAgICAgICAgIGlmIChzbmFwc2hvdC5lbXB0eSkge1xyXG4gICAgICAgICAgICBoYXNNb3JlRG9jc0luRGIgPSBmYWxzZTtcclxuICAgICAgICAgICAgYnJlYWs7XHJcbiAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgbGFzdERvY0N1cnNvciA9IHNuYXBzaG90LmRvY3Nbc25hcHNob3QuZG9jcy5sZW5ndGggLSAxXTtcclxuICAgICAgICAgIGFsbEZldGNoZWREb2NzLnB1c2goLi4uc25hcHNob3QuZG9jcyk7XHJcblxyXG4gICAgICAgICAgY29uc3QgYmF0Y2hVc2VycyA9IHNuYXBzaG90LmRvY3NcclxuICAgICAgICAgICAgLm1hcCgoZG9jOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+KSA9PiB7XHJcbiAgICAgICAgICAgICAgY29uc3QgZGF0YSA9IGRvYy5kYXRhKCkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gfCB1bmRlZmluZWQ7XHJcbiAgICAgICAgICAgICAgcmV0dXJuIHsgaWQ6IGRvYy5pZCwgLi4uKGRhdGEgPz8ge30pIH0gYXMgUmVjb3JkPFxyXG4gICAgICAgICAgICAgICAgc3RyaW5nLFxyXG4gICAgICAgICAgICAgICAgdW5rbm93blxyXG4gICAgICAgICAgICAgID4gJiB7IGlkOiBzdHJpbmcgfTtcclxuICAgICAgICAgICAgfSlcclxuICAgICAgICAgICAgLmZpbHRlcihwYXJhbXMuZmlsdGVyRm4pO1xyXG5cclxuICAgICAgICAgIHBhZ2VSZXN1bHRzLnB1c2goLi4uYmF0Y2hVc2Vycyk7XHJcblxyXG4gICAgICAgICAgaWYgKHNuYXBzaG90LnNpemUgPCBCQVRDSF9GRVRDSF9TSVpFKSB7XHJcbiAgICAgICAgICAgIGhhc01vcmVEb2NzSW5EYiA9IGZhbHNlO1xyXG4gICAgICAgICAgfVxyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgY29uc3QgaGFzTmV4dFBhZ2UgPSBwYWdlUmVzdWx0cy5sZW5ndGggPiBwYXJhbXMuYmF0Y2hTaXplO1xyXG4gICAgICAgIGNvbnN0IGZpbmFsSXRlbXMgPSBwYWdlUmVzdWx0cy5zbGljZSgwLCBwYXJhbXMuYmF0Y2hTaXplKTtcclxuXHJcbiAgICAgICAgY29uc3QgZmlyc3REb2NPZlBhZ2UgPVxyXG4gICAgICAgICAgYWxsRmV0Y2hlZERvY3MuZmluZChcclxuICAgICAgICAgICAgKGRvYzogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5RG9jdW1lbnRTbmFwc2hvdDx1bmtub3duPikgPT5cclxuICAgICAgICAgICAgICBkb2MuaWQgPT09IGZpbmFsSXRlbXNbMF0/LmlkXHJcbiAgICAgICAgICApIHx8IG51bGw7XHJcbiAgICAgICAgY29uc3QgbGFzdERvY09mUGFnZSA9XHJcbiAgICAgICAgICBhbGxGZXRjaGVkRG9jcy5maW5kKFxyXG4gICAgICAgICAgICAoZG9jOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+KSA9PlxyXG4gICAgICAgICAgICAgIGRvYy5pZCA9PT0gZmluYWxJdGVtc1tmaW5hbEl0ZW1zLmxlbmd0aCAtIDFdPy5pZFxyXG4gICAgICAgICAgKSB8fCBudWxsO1xyXG5cclxuICAgICAgICByZXR1cm4ge1xyXG4gICAgICAgICAgaXRlbXM6IGZpbmFsSXRlbXMsXHJcbiAgICAgICAgICBmaWx0ZXJMZW5ndGg6IG51bGwsXHJcbiAgICAgICAgICBmaXJzdERvYzogZmlyc3REb2NPZlBhZ2UsXHJcbiAgICAgICAgICBsYXN0RG9jOiBsYXN0RG9jT2ZQYWdlLFxyXG4gICAgICAgICAgaGFzTmV4dFBhZ2U6IGhhc05leHRQYWdlLFxyXG4gICAgICAgICAgaGFzUHJldmlvdXNQYWdlOlxyXG4gICAgICAgICAgICAhIShwYXJhbXMuZG9jICYmIHBhcmFtcy5kb2MubGFzdERvYykgJiZcclxuICAgICAgICAgICAgcGFyYW1zLm5hdmlnYXRpb24gIT09ICdyZWxvYWQnLFxyXG4gICAgICAgICAgY3VycmVudENsaWVudFBhZ2VJbmRleDogdW5kZWZpbmVkLFxyXG4gICAgICAgIH07XHJcbiAgICAgIH1cclxuICAgICAgLy8gTMOzZ2ljYSBwYXJhIHRyw6FzIChiYWNrd2FyZClcclxuICAgICAgZWxzZSBpZiAocGFyYW1zLm5hdmlnYXRpb24gPT09ICdiYWNrd2FyZCcpIHtcclxuICAgICAgICBpZiAoIXBhcmFtcy5kb2MgfHwgIXBhcmFtcy5kb2MuZmlyc3REb2MpIHtcclxuICAgICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICAgIGl0ZW1zOiBbXSxcclxuICAgICAgICAgICAgZmlsdGVyTGVuZ3RoOiBudWxsLFxyXG4gICAgICAgICAgICBmaXJzdERvYzogbnVsbCxcclxuICAgICAgICAgICAgbGFzdERvYzogbnVsbCxcclxuICAgICAgICAgICAgaGFzTmV4dFBhZ2U6IHRydWUsXHJcbiAgICAgICAgICAgIGhhc1ByZXZpb3VzUGFnZTogZmFsc2UsXHJcbiAgICAgICAgICAgIGN1cnJlbnRDbGllbnRQYWdlSW5kZXg6IHVuZGVmaW5lZCxcclxuICAgICAgICAgIH07XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBsZXQgcGFnZVJlc3VsdHM6IEFycmF5PFJlY29yZDxzdHJpbmcsIHVua25vd24+ICYgeyBpZDogc3RyaW5nIH0+ID0gW107XHJcbiAgICAgICAgbGV0IGFsbEZldGNoZWREb2NzOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+W10gPVxyXG4gICAgICAgICAgW107XHJcbiAgICAgICAgbGV0IGhhc01vcmVEb2NzSW5EYiA9IHRydWU7XHJcblxyXG4gICAgICAgIGxldCBib3VuZGFyeURvYzogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5RG9jdW1lbnRTbmFwc2hvdDx1bmtub3duPiB8IG51bGwgPVxyXG4gICAgICAgICAgcGFyYW1zLmRvYy5maXJzdERvYztcclxuXHJcbiAgICAgICAgd2hpbGUgKHBhZ2VSZXN1bHRzLmxlbmd0aCA8IEdPQUxfU0laRSAmJiBoYXNNb3JlRG9jc0luRGIpIHtcclxuICAgICAgICAgIGlmICghYm91bmRhcnlEb2MpIGJyZWFrO1xyXG5cclxuICAgICAgICAgIGxldCBxdWVyeTogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5PHVua25vd24+ID0gdGhpcy5uZ0ZpcmUuY29sbGVjdGlvbihcclxuICAgICAgICAgICAgcGFyYW1zLmNvbGxlY3Rpb25cclxuICAgICAgICAgICkucmVmO1xyXG5cclxuICAgICAgICAgIHF1ZXJ5ID0gdGhpcy5hcHBseUZpbHRlcnMocXVlcnksIHBhcmFtcy5hcnJhbmdlLCBwYXJhbXMuY29uZGl0aW9ucyk7XHJcblxyXG4gICAgICAgICAgcXVlcnkgPSBxdWVyeS5lbmRCZWZvcmUoYm91bmRhcnlEb2MpO1xyXG4gICAgICAgICAgcXVlcnkgPSBxdWVyeS5saW1pdFRvTGFzdChCQVRDSF9GRVRDSF9TSVpFKTtcclxuXHJcbiAgICAgICAgICBjb25zdCBzbmFwc2hvdCA9IGF3YWl0IHF1ZXJ5LmdldCgpO1xyXG5cclxuICAgICAgICAgIGlmIChzbmFwc2hvdC5lbXB0eSkge1xyXG4gICAgICAgICAgICBoYXNNb3JlRG9jc0luRGIgPSBmYWxzZTtcclxuICAgICAgICAgICAgYnJlYWs7XHJcbiAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgYm91bmRhcnlEb2MgPSBzbmFwc2hvdFxyXG4gICAgICAgICAgICAuZG9jc1swXSBhcyBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+O1xyXG5cclxuICAgICAgICAgIGFsbEZldGNoZWREb2NzID0gWy4uLnNuYXBzaG90LmRvY3MsIC4uLmFsbEZldGNoZWREb2NzXTtcclxuXHJcbiAgICAgICAgICBjb25zdCBiYXRjaFVzZXJzID0gc25hcHNob3QuZG9jc1xyXG4gICAgICAgICAgICAubWFwKChkb2M6IGZpcmViYXNlLmZpcmVzdG9yZS5RdWVyeURvY3VtZW50U25hcHNob3Q8dW5rbm93bj4pID0+IHtcclxuICAgICAgICAgICAgICBjb25zdCBkYXRhID0gZG9jLmRhdGEoKSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiB8IHVuZGVmaW5lZDtcclxuICAgICAgICAgICAgICByZXR1cm4geyBpZDogZG9jLmlkLCAuLi4oZGF0YSA/PyB7fSkgfSBhcyBSZWNvcmQ8XHJcbiAgICAgICAgICAgICAgICBzdHJpbmcsXHJcbiAgICAgICAgICAgICAgICB1bmtub3duXHJcbiAgICAgICAgICAgICAgPiAmIHsgaWQ6IHN0cmluZyB9O1xyXG4gICAgICAgICAgICB9KVxyXG4gICAgICAgICAgICAuZmlsdGVyKHBhcmFtcy5maWx0ZXJGbik7XHJcblxyXG4gICAgICAgICAgcGFnZVJlc3VsdHMgPSBbLi4uYmF0Y2hVc2VycywgLi4ucGFnZVJlc3VsdHNdO1xyXG5cclxuICAgICAgICAgIGlmIChzbmFwc2hvdC5zaXplIDwgQkFUQ0hfRkVUQ0hfU0laRSkge1xyXG4gICAgICAgICAgICBoYXNNb3JlRG9jc0luRGIgPSBmYWxzZTtcclxuICAgICAgICAgIH1cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGNvbnN0IGZpbmFsSXRlbXMgPSBwYWdlUmVzdWx0cy5zbGljZSgwLCBwYXJhbXMuYmF0Y2hTaXplKTtcclxuXHJcbiAgICAgICAgY29uc3QgZmlyc3REb2NPZlBhZ2UgPVxyXG4gICAgICAgICAgYWxsRmV0Y2hlZERvY3MuZmluZChcclxuICAgICAgICAgICAgKGRvYzogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5RG9jdW1lbnRTbmFwc2hvdDx1bmtub3duPikgPT5cclxuICAgICAgICAgICAgICBkb2MuaWQgPT09IGZpbmFsSXRlbXNbMF0/LmlkXHJcbiAgICAgICAgICApIHx8IG51bGw7XHJcbiAgICAgICAgY29uc3QgbGFzdERvY09mUGFnZSA9XHJcbiAgICAgICAgICBhbGxGZXRjaGVkRG9jcy5maW5kKFxyXG4gICAgICAgICAgICAoZG9jOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+KSA9PlxyXG4gICAgICAgICAgICAgIGRvYy5pZCA9PT0gZmluYWxJdGVtc1tmaW5hbEl0ZW1zLmxlbmd0aCAtIDFdPy5pZFxyXG4gICAgICAgICAgKSB8fCBudWxsO1xyXG5cclxuICAgICAgICByZXR1cm4ge1xyXG4gICAgICAgICAgaXRlbXM6IGZpbmFsSXRlbXMsXHJcbiAgICAgICAgICBmaWx0ZXJMZW5ndGg6IG51bGwsXHJcbiAgICAgICAgICBmaXJzdERvYzogZmlyc3REb2NPZlBhZ2UsXHJcbiAgICAgICAgICBsYXN0RG9jOiBsYXN0RG9jT2ZQYWdlLFxyXG4gICAgICAgICAgaGFzTmV4dFBhZ2U6IHRydWUsXHJcbiAgICAgICAgICBjdXJyZW50Q2xpZW50UGFnZUluZGV4OiB1bmRlZmluZWQsXHJcbiAgICAgICAgfTtcclxuICAgICAgfVxyXG4gICAgfSBlbHNlIHtcclxuICAgICAgbGV0IGl0ZW1zOiBBcnJheTxSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiAmIHsgaWQ6IHN0cmluZyB9PiA9IFtdO1xyXG4gICAgICBsZXQgZG9jczogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5RG9jdW1lbnRTbmFwc2hvdDx1bmtub3duPltdID0gW107XHJcbiAgICAgIGxldCBoYXNOZXh0UGFnZSA9IGZhbHNlO1xyXG4gICAgICBsZXQgZmlsdGVyTGVuZ3RoOiBudWxsIHwgbnVtYmVyID0gbnVsbDtcclxuXHJcbiAgICAgIGxldCBxdWVyeTogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5PHVua25vd24+ID0gdGhpcy5uZ0ZpcmUuY29sbGVjdGlvbihcclxuICAgICAgICBwYXJhbXMuY29sbGVjdGlvblxyXG4gICAgICApLnJlZjtcclxuXHJcbiAgICAgIGlmIChwYXJhbXMuY29uZGl0aW9ucykge1xyXG4gICAgICAgIHBhcmFtcy5jb25kaXRpb25zLmZvckVhY2goKGM6IENvbmRpdGlvbikgPT4ge1xyXG4gICAgICAgICAgaWYgKGMub3BlcmF0b3IgPT09ICchPScpIHtcclxuICAgICAgICAgICAgcXVlcnkgPSBxdWVyeS5vcmRlckJ5KGMuZmlyZXN0b3JlUHJvcGVydHkpO1xyXG4gICAgICAgICAgfVxyXG4gICAgICAgIH0pO1xyXG4gICAgICB9XHJcblxyXG4gICAgICBxdWVyeSA9IHRoaXMuYXBwbHlGaWx0ZXJzKHF1ZXJ5LCBwYXJhbXMuYXJyYW5nZSwgcGFyYW1zLmNvbmRpdGlvbnMpO1xyXG5cclxuICAgICAgaWYgKHBhcmFtcy5uYXZpZ2F0aW9uID09PSAncmVsb2FkJykge1xyXG4gICAgICAgIHF1ZXJ5ID0gcXVlcnkubGltaXQocGFyYW1zLmJhdGNoU2l6ZSArIDEpO1xyXG4gICAgICAgIGlmIChwYXJhbXMuZG9jICYmIHBhcmFtcy5kb2MuZmlyc3REb2MpIHtcclxuICAgICAgICAgIHF1ZXJ5ID0gcXVlcnkuc3RhcnRBdChwYXJhbXMuZG9jLmZpcnN0RG9jKTtcclxuICAgICAgICB9XHJcbiAgICAgIH0gZWxzZSBpZiAocGFyYW1zLm5hdmlnYXRpb24gPT09ICdmb3J3YXJkJykge1xyXG4gICAgICAgIHF1ZXJ5ID0gcXVlcnkubGltaXQocGFyYW1zLmJhdGNoU2l6ZSArIDEpO1xyXG5cclxuICAgICAgICBpZiAocGFyYW1zLmRvYyAmJiBwYXJhbXMuZG9jLmxhc3REb2MpIHtcclxuICAgICAgICAgIHF1ZXJ5ID0gcXVlcnkuc3RhcnRBZnRlcihwYXJhbXMuZG9jLmxhc3REb2MpO1xyXG4gICAgICAgIH1cclxuICAgICAgfSBlbHNlIHtcclxuICAgICAgICAvLyBiYWNrd2FyZFxyXG4gICAgICAgIHF1ZXJ5ID0gcXVlcnkubGltaXRUb0xhc3QocGFyYW1zLmJhdGNoU2l6ZSArIDEpO1xyXG4gICAgICAgIGlmIChwYXJhbXMuZG9jICYmIHBhcmFtcy5kb2MuZmlyc3REb2MpIHtcclxuICAgICAgICAgIHF1ZXJ5ID0gcXVlcnkuZW5kQmVmb3JlKHBhcmFtcy5kb2MuZmlyc3REb2MpO1xyXG4gICAgICAgIH1cclxuICAgICAgfVxyXG5cclxuICAgICAgY29uc3QgaXRlbUNvbCA9IGF3YWl0IHF1ZXJ5LmdldCgpO1xyXG4gICAgICBpdGVtQ29sLmRvY3MuZm9yRWFjaChcclxuICAgICAgICAoZG9jOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+KSA9PlxyXG4gICAgICAgICAgZG9jcy5wdXNoKGRvYylcclxuICAgICAgKTtcclxuICAgICAgY29uc3QgaXRlbVByb21pc2VzID0gZG9jcy5tYXAoXHJcbiAgICAgICAgYXN5bmMgKGl0ZW06IGZpcmViYXNlLmZpcmVzdG9yZS5RdWVyeURvY3VtZW50U25hcHNob3Q8dW5rbm93bj4pID0+IHtcclxuICAgICAgICAgIGNvbnN0IGl0ZW1EYXRhID0gaXRlbS5kYXRhKCkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gfCB1bmRlZmluZWQ7XHJcbiAgICAgICAgICBpdGVtcy5wdXNoKHsgaWQ6IGl0ZW0uaWQsIC4uLihpdGVtRGF0YSA/PyB7fSkgfSBhcyBSZWNvcmQ8XHJcbiAgICAgICAgICAgIHN0cmluZyxcclxuICAgICAgICAgICAgdW5rbm93blxyXG4gICAgICAgICAgPiAmIHsgaWQ6IHN0cmluZyB9KTtcclxuICAgICAgICB9XHJcbiAgICAgICk7XHJcblxyXG4gICAgICBsZXQgbGFzdERvYzogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5RG9jdW1lbnRTbmFwc2hvdDx1bmtub3duPiB8IG51bGwgPVxyXG4gICAgICAgIGRvY3NbZG9jcy5sZW5ndGggLSAxXSB8fCBudWxsO1xyXG4gICAgICBsZXQgZmlyc3REb2M6IGZpcmViYXNlLmZpcmVzdG9yZS5RdWVyeURvY3VtZW50U25hcHNob3Q8dW5rbm93bj4gfCBudWxsID1cclxuICAgICAgICBkb2NzWzBdO1xyXG5cclxuICAgICAgaWYgKFxyXG4gICAgICAgIChpdGVtcy5sZW5ndGggPiBwYXJhbXMuYmF0Y2hTaXplICYmIHBhcmFtcy5uYXZpZ2F0aW9uID09PSAnZm9yd2FyZCcpIHx8XHJcbiAgICAgICAgKHBhcmFtcy5uYXZpZ2F0aW9uID09PSAncmVsb2FkJyAmJiBpdGVtcy5sZW5ndGggPiBwYXJhbXMuYmF0Y2hTaXplKVxyXG4gICAgICApIHtcclxuICAgICAgICBsYXN0RG9jID0gZG9jc1tkb2NzLmxlbmd0aCAtIDJdIHx8IG51bGw7XHJcbiAgICAgICAgaXRlbXMucG9wKCk7XHJcbiAgICAgICAgaGFzTmV4dFBhZ2UgPSB0cnVlO1xyXG4gICAgICB9XHJcbiAgICAgIGlmIChpdGVtcy5sZW5ndGggPiBwYXJhbXMuYmF0Y2hTaXplICYmIHBhcmFtcy5uYXZpZ2F0aW9uID09PSAnYmFja3dhcmQnKSB7XHJcbiAgICAgICAgZmlyc3REb2MgPSBkb2NzWzFdO1xyXG4gICAgICAgIGl0ZW1zLnNoaWZ0KCk7XHJcbiAgICAgICAgaGFzTmV4dFBhZ2UgPSB0cnVlO1xyXG4gICAgICB9XHJcblxyXG4gICAgICBhd2FpdCBQcm9taXNlLmFsbChpdGVtUHJvbWlzZXMpO1xyXG4gICAgICByZXR1cm4ge1xyXG4gICAgICAgIGl0ZW1zLFxyXG4gICAgICAgIGZpbHRlckxlbmd0aCxcclxuICAgICAgICBsYXN0RG9jLFxyXG4gICAgICAgIGZpcnN0RG9jLFxyXG4gICAgICAgIGhhc05leHRQYWdlLFxyXG4gICAgICAgIGN1cnJlbnRDbGllbnRQYWdlSW5kZXg6IHVuZGVmaW5lZCxcclxuICAgICAgfTtcclxuICAgIH1cclxuXHJcbiAgICAvLyBGYWxsYmFjayBwYXJhIGdhcmFudGlyIHF1ZSBzZW1wcmUgcmV0b3JuYW1vcyBhbGdvXHJcbiAgICByZXR1cm4ge1xyXG4gICAgICBpdGVtczogW10sXHJcbiAgICAgIGZpbHRlckxlbmd0aDogbnVsbCxcclxuICAgICAgZmlyc3REb2M6IG51bGwsXHJcbiAgICAgIGxhc3REb2M6IG51bGwsXHJcbiAgICAgIGhhc05leHRQYWdlOiBmYWxzZSxcclxuICAgICAgY3VycmVudENsaWVudFBhZ2VJbmRleDogdW5kZWZpbmVkLFxyXG4gICAgfTtcclxuICB9XHJcblxyXG4gIGFwcGx5RmlsdGVycyhcclxuICAgIHF1ZXJ5OiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnk8dW5rbm93bj4sXHJcbiAgICBhcnJhbmdlOiBBcnJhbmdlLFxyXG4gICAgY29uZGl0aW9uczogQ29uZGl0aW9uW10gfCB1bmRlZmluZWRcclxuICApOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnk8dW5rbm93bj4ge1xyXG4gICAgaWYgKGNvbmRpdGlvbnMpIHtcclxuICAgICAgY29uZGl0aW9ucy5tYXAoKGNvbmQ6IENvbmRpdGlvbikgPT4ge1xyXG4gICAgICAgIHF1ZXJ5ID0gcXVlcnkud2hlcmUoXHJcbiAgICAgICAgICBjb25kLmZpcmVzdG9yZVByb3BlcnR5LFxyXG4gICAgICAgICAgY29uZC5vcGVyYXRvcixcclxuICAgICAgICAgIGNvbmQuZGFzaFByb3BlcnR5XHJcbiAgICAgICAgKTtcclxuICAgICAgfSk7XHJcbiAgICB9XHJcblxyXG4gICAgbGV0IGhhc0ZpbHRlclNwZWNpZmljT3JkZXJCeSA9IGZhbHNlO1xyXG4gICAgbGV0IGFwcGxpZWRPcmRlckJ5RmllbGQ6IHN0cmluZyB8IG51bGwgPSBudWxsO1xyXG5cclxuICAgIGNvbnN0IGVxdWFsc0ZpbHRlcnMgPSBhcnJhbmdlLmZpbHRlcnMuZmlsdGVyKFxyXG4gICAgICAoXHJcbiAgICAgICAgZlxyXG4gICAgICApOiBmIGlzIEFycmFuZ2VbJ2ZpbHRlcnMnXVswXSAmIHtcclxuICAgICAgICBhcnJhbmdlOiAnZXF1YWxzJztcclxuICAgICAgICBmaWx0ZXI6IHsgcHJvcGVydHk6IHN0cmluZzsgZmlsdGVyaW5nOiBzdHJpbmcgfTtcclxuICAgICAgfSA9PiBmLmFycmFuZ2UgPT09ICdlcXVhbHMnICYmICEhZi5maWx0ZXJcclxuICAgICk7XHJcbiAgICBjb25zdCBvdGhlckZpbHRlcnMgPSBhcnJhbmdlLmZpbHRlcnMuZmlsdGVyKFxyXG4gICAgICAoZik6IGYgaXMgQXJyYW5nZVsnZmlsdGVycyddWzBdID0+IGYuYXJyYW5nZSAhPT0gJ2VxdWFscydcclxuICAgICk7XHJcblxyXG4gICAgY29uc3QgZXF1YWxzR3JvdXBlZEJ5UHJvcGVydHkgPSBlcXVhbHNGaWx0ZXJzLnJlZHVjZShcclxuICAgICAgKGFjYzogUmVjb3JkPHN0cmluZywgc3RyaW5nW10+LCBjdXJyZW50KSA9PiB7XHJcbiAgICAgICAgY29uc3QgcHJvcCA9IGN1cnJlbnQuZmlsdGVyIS5wcm9wZXJ0eTtcclxuICAgICAgICBpZiAoIWFjY1twcm9wXSkge1xyXG4gICAgICAgICAgYWNjW3Byb3BdID0gW107XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGFjY1twcm9wXS5wdXNoKGN1cnJlbnQuZmlsdGVyIS5maWx0ZXJpbmcpO1xyXG4gICAgICAgIHJldHVybiBhY2M7XHJcbiAgICAgIH0sXHJcbiAgICAgIHt9IGFzIFJlY29yZDxzdHJpbmcsIHN0cmluZ1tdPlxyXG4gICAgKTtcclxuXHJcbiAgICBmb3IgKGNvbnN0IHByb3AgaW4gZXF1YWxzR3JvdXBlZEJ5UHJvcGVydHkpIHtcclxuICAgICAgY29uc3QgdmFsdWVzID0gZXF1YWxzR3JvdXBlZEJ5UHJvcGVydHlbcHJvcF07XHJcbiAgICAgIGlmICh2YWx1ZXMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgIHF1ZXJ5ID0gcXVlcnkud2hlcmUocHJvcCwgJ2luJywgdmFsdWVzKTtcclxuICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIG90aGVyRmlsdGVycy5mb3JFYWNoKChmaWx0ZXJJdGVtOiBBcnJhbmdlWydmaWx0ZXJzJ11bMF0pID0+IHtcclxuICAgICAgLy8gQXBsaWNhciBmaWx0cmFnZW0gcG9yIGJ1c2NhXHJcbiAgICAgIGlmIChcclxuICAgICAgICBmaWx0ZXJJdGVtLmZpbHRlcj8uZmlsdGVyaW5nICYmXHJcbiAgICAgICAgZmlsdGVySXRlbS5maWx0ZXI/LnByb3BlcnR5ICE9PSAnJyAmJlxyXG4gICAgICAgIGZpbHRlckl0ZW0uYXJyYW5nZSA9PT0gJ2ZpbHRlcidcclxuICAgICAgKSB7XHJcbiAgICAgICAgcXVlcnkgPSBxdWVyeVxyXG4gICAgICAgICAgLndoZXJlKFxyXG4gICAgICAgICAgICBmaWx0ZXJJdGVtLmZpbHRlci5wcm9wZXJ0eSxcclxuICAgICAgICAgICAgJz49JyxcclxuICAgICAgICAgICAgZmlsdGVySXRlbS5maWx0ZXIuZmlsdGVyaW5nLnRyaW0oKS50b1VwcGVyQ2FzZSgpXHJcbiAgICAgICAgICApXHJcbiAgICAgICAgICAud2hlcmUoXHJcbiAgICAgICAgICAgIGZpbHRlckl0ZW0uZmlsdGVyLnByb3BlcnR5LFxyXG4gICAgICAgICAgICAnPD0nLFxyXG4gICAgICAgICAgICBmaWx0ZXJJdGVtLmZpbHRlci5maWx0ZXJpbmcudHJpbSgpLnRvVXBwZXJDYXNlKCkgKyAnXFx1ZjhmZidcclxuICAgICAgICAgICk7XHJcbiAgICAgICAgaWYgKCFoYXNGaWx0ZXJTcGVjaWZpY09yZGVyQnkpIHtcclxuICAgICAgICAgIHF1ZXJ5ID0gcXVlcnkub3JkZXJCeShmaWx0ZXJJdGVtLmZpbHRlci5wcm9wZXJ0eSk7XHJcbiAgICAgICAgICBoYXNGaWx0ZXJTcGVjaWZpY09yZGVyQnkgPSB0cnVlO1xyXG4gICAgICAgICAgYXBwbGllZE9yZGVyQnlGaWVsZCA9IGZpbHRlckl0ZW0uZmlsdGVyLnByb3BlcnR5O1xyXG4gICAgICAgIH1cclxuICAgICAgfVxyXG5cclxuICAgICAgLy8gQXBsaWNhciBmaWx0cm8gZG8gdGlwbyBcImZpbHRlckJ5RGF0ZVwiXHJcbiAgICAgIGlmIChmaWx0ZXJJdGVtLmRhdGVGaWx0ZXIgJiYgZmlsdGVySXRlbS5hcnJhbmdlID09PSAnZmlsdGVyQnlEYXRlJykge1xyXG4gICAgICAgIHF1ZXJ5ID0gcXVlcnlcclxuICAgICAgICAgIC53aGVyZShhcnJhbmdlLnNvcnRCeS5maWVsZCwgJz49JywgZmlsdGVySXRlbS5kYXRlRmlsdGVyLmluaXRpYWwpXHJcbiAgICAgICAgICAud2hlcmUoYXJyYW5nZS5zb3J0QnkuZmllbGQsICc8PScsIGZpbHRlckl0ZW0uZGF0ZUZpbHRlci5maW5hbCk7XHJcbiAgICAgICAgaWYgKCFoYXNGaWx0ZXJTcGVjaWZpY09yZGVyQnkpIHtcclxuICAgICAgICAgIHF1ZXJ5ID0gcXVlcnkub3JkZXJCeShhcnJhbmdlLnNvcnRCeS5maWVsZCk7XHJcbiAgICAgICAgICBoYXNGaWx0ZXJTcGVjaWZpY09yZGVyQnkgPSB0cnVlO1xyXG4gICAgICAgICAgYXBwbGllZE9yZGVyQnlGaWVsZCA9IGFycmFuZ2Uuc29ydEJ5LmZpZWxkO1xyXG4gICAgICAgIH1cclxuICAgICAgfVxyXG4gICAgfSk7XHJcblxyXG4gICAgLy8gQXBsaWNhciBzb3J0QnlcclxuICAgIGlmIChhcnJhbmdlLnNvcnRCeSAmJiBhcnJhbmdlLnNvcnRCeS5maWVsZCAmJiBhcnJhbmdlLnNvcnRCeS5vcmRlcikge1xyXG4gICAgICBpZiAoYXBwbGllZE9yZGVyQnlGaWVsZCAhPT0gYXJyYW5nZS5zb3J0QnkuZmllbGQpIHtcclxuICAgICAgICBxdWVyeSA9IHF1ZXJ5Lm9yZGVyQnkoYXJyYW5nZS5zb3J0QnkuZmllbGQsIGFycmFuZ2Uuc29ydEJ5Lm9yZGVyKTtcclxuICAgICAgfVxyXG4gICAgfVxyXG4gICAgcmV0dXJuIHF1ZXJ5O1xyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBnZXRJZEZpbHRlcihwYXJhbXM6IFBhZ2luYXRpb24pOiBzdHJpbmcgfCBudWxsIHtcclxuICAgIGlmICghcGFyYW1zLmFycmFuZ2U/LmZpbHRlcnMpIHJldHVybiBudWxsO1xyXG5cclxuICAgIGNvbnN0IGlkRmlsdGVyID0gcGFyYW1zLmFycmFuZ2UuZmlsdGVycy5maW5kKFxyXG4gICAgICAoXHJcbiAgICAgICAgZlxyXG4gICAgICApOiBmIGlzIEFycmFuZ2VbJ2ZpbHRlcnMnXVswXSAmIHtcclxuICAgICAgICBhcnJhbmdlOiAnZmlsdGVyJztcclxuICAgICAgICBmaWx0ZXI6IHsgcHJvcGVydHk6IHN0cmluZzsgZmlsdGVyaW5nOiBzdHJpbmcgfTtcclxuICAgICAgfSA9PlxyXG4gICAgICAgIGYuYXJyYW5nZSA9PT0gJ2ZpbHRlcicgJiZcclxuICAgICAgICAhIWYuZmlsdGVyICYmXHJcbiAgICAgICAgZi5maWx0ZXIucHJvcGVydHkgPT09ICdpZCcgJiZcclxuICAgICAgICAhIWYuZmlsdGVyLmZpbHRlcmluZ1xyXG4gICAgKTtcclxuXHJcbiAgICByZXR1cm4gaWRGaWx0ZXI/LmZpbHRlcj8uZmlsdGVyaW5nPy50cmltKCkgfHwgbnVsbDtcclxuICB9XHJcblxyXG4gIHByaXZhdGUgYXN5bmMgZ2V0RG9jdW1lbnRCeUlkKFxyXG4gICAgY29sbGVjdGlvbjogc3RyaW5nLFxyXG4gICAgZG9jSWQ6IHN0cmluZ1xyXG4gICk6IFByb21pc2U8KFJlY29yZDxzdHJpbmcsIHVua25vd24+ICYgeyBpZDogc3RyaW5nIH0pIHwgbnVsbD4ge1xyXG4gICAgdHJ5IHtcclxuICAgICAgY29uc3QgZG9jUmVmID0gdGhpcy5uZ0ZpcmUuY29sbGVjdGlvbihjb2xsZWN0aW9uKS5kb2MoZG9jSWQpO1xyXG4gICAgICBjb25zdCBkb2NTbmFwc2hvdCA9IGF3YWl0IGRvY1JlZi5nZXQoKS50b1Byb21pc2UoKTtcclxuXHJcbiAgICAgIGlmIChkb2NTbmFwc2hvdCAmJiBkb2NTbmFwc2hvdC5leGlzdHMpIHtcclxuICAgICAgICBjb25zdCBkYXRhID0gZG9jU25hcHNob3QuZGF0YSgpIGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+IHwgdW5kZWZpbmVkO1xyXG4gICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICBpZDogZG9jU25hcHNob3QuaWQsXHJcbiAgICAgICAgICAuLi4oZGF0YSA/PyB7fSksXHJcbiAgICAgICAgfSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiAmIHsgaWQ6IHN0cmluZyB9O1xyXG4gICAgICB9XHJcbiAgICAgIHJldHVybiBudWxsO1xyXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgY29uc29sZS53YXJuKCdFcnJvIGFvIGJ1c2NhciBkb2N1bWVudG8gcG9yIElEOicsIGVycm9yKTtcclxuICAgICAgcmV0dXJuIG51bGw7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGFzeW5jIHNlYXJjaEJ5SWRQYXJ0aWFsKFxyXG4gICAgcGFyYW1zOiBQYWdpbmF0aW9uLFxyXG4gICAgc2VhcmNoVGVybTogc3RyaW5nXHJcbiAgKTogUHJvbWlzZTxQYWdpbmF0aW9uUmVzdWx0PiB7XHJcbiAgICBjb25zdCBleGFjdE1hdGNoID0gYXdhaXQgdGhpcy5nZXREb2N1bWVudEJ5SWQoXHJcbiAgICAgIHBhcmFtcy5jb2xsZWN0aW9uLFxyXG4gICAgICBzZWFyY2hUZXJtXHJcbiAgICApO1xyXG5cclxuICAgIGlmIChleGFjdE1hdGNoKSB7XHJcbiAgICAgIGlmIChwYXJhbXMuY29uZGl0aW9ucykge1xyXG4gICAgICAgIGNvbnN0IG9wZXJhdG9ycyA9IHRoaXMub3BlcmF0b3JzO1xyXG4gICAgICAgIGNvbnN0IHBhc3Nlc0NvbmRpdGlvbnMgPSBwYXJhbXMuY29uZGl0aW9ucy5ldmVyeSgoY29uZCkgPT4ge1xyXG4gICAgICAgICAgY29uc3Qgb3BlcmF0b3JGbiA9IG9wZXJhdG9yc1tjb25kLm9wZXJhdG9yIGFzIGtleW9mIHR5cGVvZiBvcGVyYXRvcnNdO1xyXG4gICAgICAgICAgcmV0dXJuIG9wZXJhdG9yRm5cclxuICAgICAgICAgICAgPyBvcGVyYXRvckZuKGV4YWN0TWF0Y2hbY29uZC5maXJlc3RvcmVQcm9wZXJ0eV0sIGNvbmQuZGFzaFByb3BlcnR5KVxyXG4gICAgICAgICAgICA6IGZhbHNlO1xyXG4gICAgICAgIH0pO1xyXG5cclxuICAgICAgICBpZiAoIXBhc3Nlc0NvbmRpdGlvbnMpIHtcclxuICAgICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICAgIGl0ZW1zOiBbXSxcclxuICAgICAgICAgICAgZmlsdGVyTGVuZ3RoOiAwLFxyXG4gICAgICAgICAgICBmaXJzdERvYzogbnVsbCxcclxuICAgICAgICAgICAgbGFzdERvYzogbnVsbCxcclxuICAgICAgICAgICAgaGFzTmV4dFBhZ2U6IGZhbHNlLFxyXG4gICAgICAgICAgICBoYXNQcmV2aW91c1BhZ2U6IGZhbHNlLFxyXG4gICAgICAgICAgICBjdXJyZW50Q2xpZW50UGFnZUluZGV4OiAwLFxyXG4gICAgICAgICAgICB0b3RhbFBhZ2VzOiAwLFxyXG4gICAgICAgICAgfTtcclxuICAgICAgICB9XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIGlmIChwYXJhbXMuZmlsdGVyRm4gJiYgIXBhcmFtcy5maWx0ZXJGbihleGFjdE1hdGNoKSkge1xyXG4gICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICBpdGVtczogW10sXHJcbiAgICAgICAgICBmaWx0ZXJMZW5ndGg6IDAsXHJcbiAgICAgICAgICBmaXJzdERvYzogbnVsbCxcclxuICAgICAgICAgIGxhc3REb2M6IG51bGwsXHJcbiAgICAgICAgICBoYXNOZXh0UGFnZTogZmFsc2UsXHJcbiAgICAgICAgICBoYXNQcmV2aW91c1BhZ2U6IGZhbHNlLFxyXG4gICAgICAgICAgY3VycmVudENsaWVudFBhZ2VJbmRleDogMCxcclxuICAgICAgICAgIHRvdGFsUGFnZXM6IDAsXHJcbiAgICAgICAgfTtcclxuICAgICAgfVxyXG5cclxuICAgICAgcmV0dXJuIHtcclxuICAgICAgICBpdGVtczogW2V4YWN0TWF0Y2hdLFxyXG4gICAgICAgIGZpbHRlckxlbmd0aDogMSxcclxuICAgICAgICBmaXJzdERvYzogbnVsbCxcclxuICAgICAgICBsYXN0RG9jOiBudWxsLFxyXG4gICAgICAgIGhhc05leHRQYWdlOiBmYWxzZSxcclxuICAgICAgICBoYXNQcmV2aW91c1BhZ2U6IGZhbHNlLFxyXG4gICAgICAgIGN1cnJlbnRDbGllbnRQYWdlSW5kZXg6IDAsXHJcbiAgICAgICAgdG90YWxQYWdlczogMSxcclxuICAgICAgfTtcclxuICAgIH1cclxuXHJcbiAgICBjb25zdCBzZWFyY2hUZXJtTG93ZXIgPSBzZWFyY2hUZXJtLnRvTG93ZXJDYXNlKCk7XHJcblxyXG4gICAgY29uc3QgcGFyYW1zV2l0aG91dElkRmlsdGVyID0ge1xyXG4gICAgICAuLi5wYXJhbXMsXHJcbiAgICAgIGFycmFuZ2U6IHtcclxuICAgICAgICAuLi5wYXJhbXMuYXJyYW5nZSxcclxuICAgICAgICBmaWx0ZXJzOiBwYXJhbXMuYXJyYW5nZS5maWx0ZXJzLmZpbHRlcihcclxuICAgICAgICAgIChmKSA9PiAhKGYuYXJyYW5nZSA9PT0gJ2ZpbHRlcicgJiYgZi5maWx0ZXI/LnByb3BlcnR5ID09PSAnaWQnKVxyXG4gICAgICAgICksXHJcbiAgICAgIH0sXHJcbiAgICB9O1xyXG5cclxuICAgIGxldCBxdWVyeTogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5PHVua25vd24+ID0gdGhpcy5uZ0ZpcmUuY29sbGVjdGlvbihcclxuICAgICAgcGFyYW1zLmNvbGxlY3Rpb25cclxuICAgICkucmVmO1xyXG5cclxuICAgIC8vIEFwbGljYXIgY29uZGl0aW9uc1xyXG4gICAgaWYgKHBhcmFtcy5jb25kaXRpb25zKSB7XHJcbiAgICAgIHBhcmFtcy5jb25kaXRpb25zLmZvckVhY2goKGNvbmQpID0+IHtcclxuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKFxyXG4gICAgICAgICAgY29uZC5maXJlc3RvcmVQcm9wZXJ0eSxcclxuICAgICAgICAgIGNvbmQub3BlcmF0b3IsXHJcbiAgICAgICAgICBjb25kLmRhc2hQcm9wZXJ0eVxyXG4gICAgICAgICk7XHJcbiAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIEFwbGljYXIgc29ydEJ5XHJcbiAgICBpZiAocGFyYW1zLmFycmFuZ2U/LnNvcnRCeT8uZmllbGQgJiYgcGFyYW1zLmFycmFuZ2U/LnNvcnRCeT8ub3JkZXIpIHtcclxuICAgICAgcXVlcnkgPSBxdWVyeS5vcmRlckJ5KFxyXG4gICAgICAgIHBhcmFtcy5hcnJhbmdlLnNvcnRCeS5maWVsZCxcclxuICAgICAgICBwYXJhbXMuYXJyYW5nZS5zb3J0Qnkub3JkZXJcclxuICAgICAgKTtcclxuICAgIH1cclxuXHJcbiAgICBjb25zdCBzbmFwc2hvdCA9IGF3YWl0IHF1ZXJ5LmdldCgpO1xyXG4gICAgbGV0IGl0ZW1zOiBBcnJheTxSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiAmIHsgaWQ6IHN0cmluZyB9PiA9IHNuYXBzaG90LmRvY3NcclxuICAgICAgLm1hcCgoZG9jKSA9PiB7XHJcbiAgICAgICAgY29uc3QgZGF0YSA9IGRvYy5kYXRhKCkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gfCB1bmRlZmluZWQ7XHJcbiAgICAgICAgcmV0dXJuIHsgaWQ6IGRvYy5pZCwgLi4uKGRhdGEgPz8ge30pIH0gYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gJiB7XHJcbiAgICAgICAgICBpZDogc3RyaW5nO1xyXG4gICAgICAgIH07XHJcbiAgICAgIH0pXHJcbiAgICAgIC5maWx0ZXIoKGl0ZW0pID0+IGl0ZW0uaWQudG9Mb3dlckNhc2UoKS5pbmNsdWRlcyhzZWFyY2hUZXJtTG93ZXIpKTtcclxuXHJcbiAgICAvLyBTZXBhcmFyIGVxdWFscyBmaWx0ZXJzIGUgb3V0cm9zIGZpbHRyb3NcclxuICAgIGNvbnN0IGVxdWFsc0ZpbHRlcnMgPSBwYXJhbXNXaXRob3V0SWRGaWx0ZXIuYXJyYW5nZS5maWx0ZXJzLmZpbHRlcihcclxuICAgICAgKFxyXG4gICAgICAgIGZcclxuICAgICAgKTogZiBpcyBBcnJhbmdlWydmaWx0ZXJzJ11bMF0gJiB7XHJcbiAgICAgICAgYXJyYW5nZTogJ2VxdWFscyc7XHJcbiAgICAgICAgZmlsdGVyOiB7IHByb3BlcnR5OiBzdHJpbmc7IGZpbHRlcmluZzogc3RyaW5nIH07XHJcbiAgICAgIH0gPT4gZi5hcnJhbmdlID09PSAnZXF1YWxzJyAmJiAhIWYuZmlsdGVyXHJcbiAgICApO1xyXG4gICAgY29uc3Qgb3RoZXJGaWx0ZXJzID0gcGFyYW1zV2l0aG91dElkRmlsdGVyLmFycmFuZ2UuZmlsdGVycy5maWx0ZXIoXHJcbiAgICAgIChmKTogZiBpcyBBcnJhbmdlWydmaWx0ZXJzJ11bMF0gPT5cclxuICAgICAgICBmLmFycmFuZ2UgIT09ICdlcXVhbHMnICYmXHJcbiAgICAgICAgKGYuYXJyYW5nZSAhPT0gJ2ZpbHRlcicgfHwgZi5maWx0ZXI/LnByb3BlcnR5ICE9PSAnaWQnKVxyXG4gICAgKTtcclxuXHJcbiAgICAvLyBBcGxpY2FyIGVxdWFscyBmaWx0ZXJzIGNvbSBsw7NnaWNhIE9SIGRlbnRybyBkZSBjYWRhIHByb3ByaWVkYWRlXHJcbiAgICBpZiAoZXF1YWxzRmlsdGVycy5sZW5ndGggPiAwKSB7XHJcbiAgICAgIC8vIEFncnVwYXIgcG9yIHByb3ByaWVkYWRlIHBhcmEgYXBsaWNhciBPUlxyXG4gICAgICBjb25zdCBncm91cGVkQnlQcm9wZXJ0eSA9IGVxdWFsc0ZpbHRlcnMucmVkdWNlKFxyXG4gICAgICAgIChhY2M6IFJlY29yZDxzdHJpbmcsIChzdHJpbmcgfCBib29sZWFuKVtdPiwgZikgPT4ge1xyXG4gICAgICAgICAgY29uc3QgcHJvcCA9IGYuZmlsdGVyLnByb3BlcnR5O1xyXG4gICAgICAgICAgaWYgKCFhY2NbcHJvcF0pIHtcclxuICAgICAgICAgICAgYWNjW3Byb3BdID0gW107XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgICBhY2NbcHJvcF0ucHVzaChmLmZpbHRlci5maWx0ZXJpbmcpO1xyXG4gICAgICAgICAgcmV0dXJuIGFjYztcclxuICAgICAgICB9LFxyXG4gICAgICAgIHt9IGFzIFJlY29yZDxzdHJpbmcsIChzdHJpbmcgfCBib29sZWFuKVtdPlxyXG4gICAgICApO1xyXG5cclxuICAgICAgLy8gRmlsdHJhcjogaXRlbSBkZXZlIHRlciBwZWxvIG1lbm9zIHVtIHZhbG9yIGRlIENBREEgcHJvcHJpZWRhZGVcclxuICAgICAgaXRlbXMgPSBpdGVtcy5maWx0ZXIoKGl0ZW0pID0+IHtcclxuICAgICAgICByZXR1cm4gT2JqZWN0LmVudHJpZXMoZ3JvdXBlZEJ5UHJvcGVydHkpLmV2ZXJ5KChbcHJvcCwgdmFsdWVzXSkgPT4ge1xyXG4gICAgICAgICAgY29uc3QgaXRlbVZhbHVlID0gaXRlbVtwcm9wXTtcclxuICAgICAgICAgIHJldHVybiB2YWx1ZXMuaW5jbHVkZXMoaXRlbVZhbHVlIGFzIHN0cmluZyB8IGJvb2xlYW4pO1xyXG4gICAgICAgIH0pO1xyXG4gICAgICB9KTtcclxuICAgIH1cclxuXHJcbiAgICAvLyBBcGxpY2FyIG91dHJvcyBmaWx0cm9zXHJcbiAgICBvdGhlckZpbHRlcnMuZm9yRWFjaCgoZmlsdGVySXRlbSkgPT4ge1xyXG4gICAgICBpZiAoXHJcbiAgICAgICAgZmlsdGVySXRlbS5hcnJhbmdlID09PSAnZmlsdGVyJyAmJlxyXG4gICAgICAgIGZpbHRlckl0ZW0uZmlsdGVyPy5maWx0ZXJpbmcgJiZcclxuICAgICAgICBmaWx0ZXJJdGVtLmZpbHRlcj8ucHJvcGVydHlcclxuICAgICAgKSB7XHJcbiAgICAgICAgY29uc3QgZmlsdGVyVmFsdWUgPSBTdHJpbmcoZmlsdGVySXRlbS5maWx0ZXIuZmlsdGVyaW5nKVxyXG4gICAgICAgICAgLnRyaW0oKVxyXG4gICAgICAgICAgLnRvTG93ZXJDYXNlKCk7XHJcbiAgICAgICAgaXRlbXMgPSBpdGVtcy5maWx0ZXIoKGl0ZW0pID0+IHtcclxuICAgICAgICAgIGNvbnN0IGl0ZW1WYWx1ZSA9IFN0cmluZyhcclxuICAgICAgICAgICAgaXRlbVtmaWx0ZXJJdGVtLmZpbHRlciEucHJvcGVydHldID8/ICcnXHJcbiAgICAgICAgICApLnRvTG93ZXJDYXNlKCk7XHJcbiAgICAgICAgICByZXR1cm4gaXRlbVZhbHVlLmluY2x1ZGVzKGZpbHRlclZhbHVlKTtcclxuICAgICAgICB9KTtcclxuICAgICAgfVxyXG4gICAgfSk7XHJcblxyXG4gICAgLy8gQXBsaWNhciBmaWx0ZXJGbiBzZSBleGlzdGlyXHJcbiAgICBpZiAocGFyYW1zLmZpbHRlckZuKSB7XHJcbiAgICAgIGl0ZW1zID0gaXRlbXMuZmlsdGVyKHBhcmFtcy5maWx0ZXJGbik7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gUGFnaW5hw6fDo29cclxuICAgIGNvbnN0IHBhZ2VTaXplID0gcGFyYW1zLmJhdGNoU2l6ZTtcclxuICAgIGxldCBjdXJyZW50Q2xpZW50UGFnZUluZGV4ID0gMDtcclxuXHJcbiAgICBpZiAocGFyYW1zLm5hdmlnYXRpb24gPT09ICdyZWxvYWQnKSB7XHJcbiAgICAgIGN1cnJlbnRDbGllbnRQYWdlSW5kZXggPSAwO1xyXG4gICAgfSBlbHNlIHtcclxuICAgICAgLy8gVXNhciBvIMOtbmRpY2UgcGFzc2FkbyBwZWxvIGNvbXBvbmVudGUgc2VtIGluY3JlbWVudGFyL2RlY3JlbWVudGFyXHJcbiAgICAgIGN1cnJlbnRDbGllbnRQYWdlSW5kZXggPSBwYXJhbXMuY2xpZW50UGFnZUluZGV4IHx8IDA7XHJcbiAgICB9XHJcblxyXG4gICAgY29uc3Qgc3RhcnRJbmRleCA9IGN1cnJlbnRDbGllbnRQYWdlSW5kZXggKiBwYWdlU2l6ZTtcclxuICAgIGNvbnN0IGVuZEluZGV4ID0gc3RhcnRJbmRleCArIHBhZ2VTaXplO1xyXG4gICAgY29uc3QgcGFnaW5hdGVkSXRlbXMgPSBpdGVtcy5zbGljZShzdGFydEluZGV4LCBlbmRJbmRleCk7XHJcblxyXG4gICAgY29uc3QgdG90YWxQYWdlcyA9IE1hdGguY2VpbChpdGVtcy5sZW5ndGggLyBwYWdlU2l6ZSk7XHJcbiAgICBjb25zdCBoYXNOZXh0UGFnZSA9IGN1cnJlbnRDbGllbnRQYWdlSW5kZXggPCB0b3RhbFBhZ2VzIC0gMTtcclxuICAgIGNvbnN0IGhhc1ByZXZpb3VzUGFnZSA9IGN1cnJlbnRDbGllbnRQYWdlSW5kZXggPiAwO1xyXG5cclxuICAgIHJldHVybiB7XHJcbiAgICAgIGl0ZW1zOiBwYWdpbmF0ZWRJdGVtcyxcclxuICAgICAgZmlsdGVyTGVuZ3RoOiBpdGVtcy5sZW5ndGgsXHJcbiAgICAgIGZpcnN0RG9jOiBudWxsLFxyXG4gICAgICBsYXN0RG9jOiBudWxsLFxyXG4gICAgICBoYXNOZXh0UGFnZSxcclxuICAgICAgaGFzUHJldmlvdXNQYWdlLFxyXG4gICAgICBjdXJyZW50Q2xpZW50UGFnZUluZGV4LFxyXG4gICAgICB0b3RhbFBhZ2VzLFxyXG4gICAgfTtcclxuICB9XHJcblxyXG4gIHByaXZhdGUgc2hvdWxkVXNlQ2xpZW50U2lkZUZhbGxiYWNrKHBhcmFtczogUGFnaW5hdGlvbik6IGJvb2xlYW4ge1xyXG4gICAgY29uc3QgaGFzQ29uZGl0aW9ucyA9IHBhcmFtcy5jb25kaXRpb25zICYmIHBhcmFtcy5jb25kaXRpb25zLmxlbmd0aCA+IDA7XHJcbiAgICBjb25zdCBoYXNBcnJhbmdlRmlsdGVycyA9XHJcbiAgICAgIHBhcmFtcy5hcnJhbmdlPy5maWx0ZXJzICYmIHBhcmFtcy5hcnJhbmdlLmZpbHRlcnMubGVuZ3RoID4gMDtcclxuICAgIGNvbnN0IGhhc1NvcnRCeSA9IHBhcmFtcy5hcnJhbmdlPy5zb3J0Qnk/LmZpZWxkO1xyXG5cclxuICAgIGlmIChwYXJhbXMuZmlsdGVyRm4pIHtcclxuICAgICAgcmV0dXJuIGZhbHNlO1xyXG4gICAgfVxyXG5cclxuICAgIGlmIChoYXNBcnJhbmdlRmlsdGVycykge1xyXG4gICAgICBjb25zdCBlcXVhbHNGaWx0ZXJzID0gcGFyYW1zLmFycmFuZ2UuZmlsdGVycy5maWx0ZXIoXHJcbiAgICAgICAgKFxyXG4gICAgICAgICAgZlxyXG4gICAgICAgICk6IGYgaXMgQXJyYW5nZVsnZmlsdGVycyddWzBdICYge1xyXG4gICAgICAgICAgYXJyYW5nZTogJ2VxdWFscyc7XHJcbiAgICAgICAgICBmaWx0ZXI6IHsgcHJvcGVydHk6IHN0cmluZzsgZmlsdGVyaW5nOiBzdHJpbmcgfTtcclxuICAgICAgICB9ID0+IGYuYXJyYW5nZSA9PT0gJ2VxdWFscycgJiYgISFmLmZpbHRlclxyXG4gICAgICApO1xyXG5cclxuICAgICAgaWYgKGVxdWFsc0ZpbHRlcnMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgIGNvbnN0IHByb3BlcnRpZXNTZXQgPSBuZXcgU2V0KFxyXG4gICAgICAgICAgZXF1YWxzRmlsdGVycy5tYXAoKGYpID0+IGYuZmlsdGVyLnByb3BlcnR5KVxyXG4gICAgICAgICk7XHJcblxyXG4gICAgICAgIGlmIChwcm9wZXJ0aWVzU2V0LnNpemUgPiAxKSB7XHJcbiAgICAgICAgICByZXR1cm4gdHJ1ZTtcclxuICAgICAgICB9XHJcbiAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICBpZiAoaGFzQ29uZGl0aW9ucyAmJiBoYXNBcnJhbmdlRmlsdGVycyAmJiBoYXNTb3J0QnkpIHtcclxuICAgICAgcmV0dXJuIHRydWU7XHJcbiAgICB9XHJcblxyXG4gICAgaWYgKGhhc0NvbmRpdGlvbnMgJiYgaGFzQXJyYW5nZUZpbHRlcnMpIHtcclxuICAgICAgcmV0dXJuIHRydWU7XHJcbiAgICB9XHJcblxyXG4gICAgaWYgKGhhc0FycmFuZ2VGaWx0ZXJzICYmIHBhcmFtcy5hcnJhbmdlLmZpbHRlcnMubGVuZ3RoID4gMSAmJiBoYXNTb3J0QnkpIHtcclxuICAgICAgY29uc3QgZXF1YWxzRmlsdGVycyA9IHBhcmFtcy5hcnJhbmdlLmZpbHRlcnMuZmlsdGVyKFxyXG4gICAgICAgIChcclxuICAgICAgICAgIGZcclxuICAgICAgICApOiBmIGlzIEFycmFuZ2VbJ2ZpbHRlcnMnXVswXSAmIHtcclxuICAgICAgICAgIGFycmFuZ2U6ICdlcXVhbHMnO1xyXG4gICAgICAgICAgZmlsdGVyOiB7IHByb3BlcnR5OiBzdHJpbmc7IGZpbHRlcmluZzogc3RyaW5nIH07XHJcbiAgICAgICAgfSA9PiBmLmFycmFuZ2UgPT09ICdlcXVhbHMnICYmICEhZi5maWx0ZXJcclxuICAgICAgKTtcclxuXHJcbiAgICAgIGlmIChlcXVhbHNGaWx0ZXJzLmxlbmd0aCA+IDApIHtcclxuICAgICAgICBjb25zdCBwcm9wZXJ0aWVzU2V0ID0gbmV3IFNldChcclxuICAgICAgICAgIGVxdWFsc0ZpbHRlcnMubWFwKChmKSA9PiBmLmZpbHRlci5wcm9wZXJ0eSlcclxuICAgICAgICApO1xyXG4gICAgICAgIGlmIChcclxuICAgICAgICAgIHByb3BlcnRpZXNTZXQuc2l6ZSA9PT0gMSAmJlxyXG4gICAgICAgICAgZXF1YWxzRmlsdGVycy5sZW5ndGggPT09IHBhcmFtcy5hcnJhbmdlLmZpbHRlcnMubGVuZ3RoXHJcbiAgICAgICAgKSB7XHJcbiAgICAgICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICAgICAgfVxyXG4gICAgICB9XHJcblxyXG4gICAgICByZXR1cm4gdHJ1ZTtcclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4gZmFsc2U7XHJcbiAgfVxyXG5cclxuICBhc3luYyBnZXRQYWdpbmF0ZWQocGFyYW1zOiBQYWdpbmF0aW9uKTogUHJvbWlzZTxQYWdpbmF0aW9uUmVzdWx0PiB7XHJcbiAgICBjb25zdCBpZEZpbHRlclZhbHVlID0gdGhpcy5nZXRJZEZpbHRlcihwYXJhbXMpO1xyXG4gICAgaWYgKGlkRmlsdGVyVmFsdWUpIHtcclxuICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgdGhpcy5zZWFyY2hCeUlkUGFydGlhbChwYXJhbXMsIGlkRmlsdGVyVmFsdWUpO1xyXG4gICAgICByZXR1cm4gcmVzdWx0O1xyXG4gICAgfVxyXG5cclxuICAgIC8vIERldGVjdGFyIHByZXZlbnRpdmFtZW50ZSBzZSBkZXZlIHVzYXIgZmFsbGJhY2tcclxuICAgIGlmICh0aGlzLnNob3VsZFVzZUNsaWVudFNpZGVGYWxsYmFjayhwYXJhbXMpKSB7XHJcbiAgICAgIGF3YWl0IHRoaXMudHJhY2tNaXNzaW5nSW5kZXhQcmV2ZW50aXZlKFxyXG4gICAgICAgIHBhcmFtcy5jb2xsZWN0aW9uLFxyXG4gICAgICAgIHBhcmFtcy5hcnJhbmdlLFxyXG4gICAgICAgIHBhcmFtcy5jb25kaXRpb25zXHJcbiAgICAgICk7XHJcblxyXG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0aGlzLmV4ZWN1dGVDbGllbnRTaWRlUXVlcnkocGFyYW1zKTtcclxuICAgICAgcmV0dXJuIHJlc3VsdDtcclxuICAgIH1cclxuXHJcbiAgICB0cnkge1xyXG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0aGlzLmV4ZWN1dGVRdWVyeShwYXJhbXMpO1xyXG4gICAgICByZXR1cm4gcmVzdWx0O1xyXG4gICAgfSBjYXRjaCAoZXJyb3I6IHVua25vd24pIHtcclxuICAgICAgaWYgKFxyXG4gICAgICAgIGVycm9yICYmXHJcbiAgICAgICAgdHlwZW9mIGVycm9yID09PSAnb2JqZWN0JyAmJlxyXG4gICAgICAgICdjb2RlJyBpbiBlcnJvciAmJlxyXG4gICAgICAgIChlcnJvciBhcyBFcnJvcldpdGhDb2RlKS5jb2RlID09PSAnZmFpbGVkLXByZWNvbmRpdGlvbidcclxuICAgICAgKSB7XHJcbiAgICAgICAgYXdhaXQgdGhpcy50cmFja01pc3NpbmdJbmRleChcclxuICAgICAgICAgIGVycm9yLFxyXG4gICAgICAgICAgcGFyYW1zLmNvbGxlY3Rpb24sXHJcbiAgICAgICAgICBwYXJhbXMuYXJyYW5nZSxcclxuICAgICAgICAgIHBhcmFtcy5jb25kaXRpb25zXHJcbiAgICAgICAgKTtcclxuXHJcbiAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgdGhpcy5leGVjdXRlQ2xpZW50U2lkZVF1ZXJ5KHBhcmFtcyk7XHJcbiAgICAgICAgcmV0dXJuIHJlc3VsdDtcclxuICAgICAgfSBlbHNlIGlmIChcclxuICAgICAgICBlcnJvciAmJlxyXG4gICAgICAgIHR5cGVvZiBlcnJvciA9PT0gJ29iamVjdCcgJiZcclxuICAgICAgICAnY29kZScgaW4gZXJyb3IgJiZcclxuICAgICAgICAoZXJyb3IgYXMgRXJyb3JXaXRoQ29kZSkuY29kZSA9PT0gJ2ludmFsaWQtYXJndW1lbnQnXHJcbiAgICAgICkge1xyXG4gICAgICAgIGF3YWl0IHRoaXMudHJhY2tNaXNzaW5nSW5kZXgoXHJcbiAgICAgICAgICBlcnJvcixcclxuICAgICAgICAgIHBhcmFtcy5jb2xsZWN0aW9uLFxyXG4gICAgICAgICAgcGFyYW1zLmFycmFuZ2UsXHJcbiAgICAgICAgICBwYXJhbXMuY29uZGl0aW9uc1xyXG4gICAgICAgICk7XHJcblxyXG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHRoaXMuZXhlY3V0ZUNsaWVudFNpZGVRdWVyeShwYXJhbXMpO1xyXG4gICAgICAgIHJldHVybiByZXN1bHQ7XHJcbiAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgdGhyb3cgZXJyb3I7XHJcbiAgICAgIH1cclxuICAgIH1cclxuICB9XHJcblxyXG4gIGFzeW5jIGV4ZWN1dGVDbGllbnRTaWRlUXVlcnkocGFyYW1zOiBQYWdpbmF0aW9uKTogUHJvbWlzZTxQYWdpbmF0aW9uUmVzdWx0PiB7XHJcbiAgICAvLyBPdGltaXphciB1c2FuZG8gcGVsbyBtZW5vcyB1bWEgY2zDoXVzdWxhIC53aGVyZSgpIHF1YW5kbyBwb3Nzw612ZWxcclxuICAgIGxldCBxdWVyeTogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5PHVua25vd24+ID0gdGhpcy5uZ0ZpcmUuY29sbGVjdGlvbihcclxuICAgICAgcGFyYW1zLmNvbGxlY3Rpb25cclxuICAgICkucmVmO1xyXG5cclxuICAgIGxldCBhcHBsaWVkQ29uZGl0aW9uOiBDb25kaXRpb24gfCBudWxsID0gbnVsbDtcclxuICAgIGxldCBoYXNBcHBsaWVkV2hlcmVDbGF1c2UgPSBmYWxzZTtcclxuXHJcbiAgICAvLyBQcmltZWlybywgdGVudGEgYXBsaWNhciBjb25kacOnw7VlcyBzaW1wbGVzXHJcbiAgICBpZiAocGFyYW1zLmNvbmRpdGlvbnMgJiYgcGFyYW1zLmNvbmRpdGlvbnMubGVuZ3RoID4gMCkge1xyXG4gICAgICBjb25zdCBzaW1wbGVDb25kaXRpb24gPSBwYXJhbXMuY29uZGl0aW9ucy5maW5kKChjb25kOiBDb25kaXRpb24pID0+XHJcbiAgICAgICAgWyc9PScsICc+JywgJzwnLCAnPj0nLCAnPD0nLCAnaW4nLCAnYXJyYXktY29udGFpbnMnXS5pbmNsdWRlcyhcclxuICAgICAgICAgIGNvbmQub3BlcmF0b3JcclxuICAgICAgICApXHJcbiAgICAgICk7XHJcblxyXG4gICAgICBpZiAoc2ltcGxlQ29uZGl0aW9uKSB7XHJcbiAgICAgICAgcXVlcnkgPSBxdWVyeS53aGVyZShcclxuICAgICAgICAgIHNpbXBsZUNvbmRpdGlvbi5maXJlc3RvcmVQcm9wZXJ0eSxcclxuICAgICAgICAgIHNpbXBsZUNvbmRpdGlvbi5vcGVyYXRvcixcclxuICAgICAgICAgIHNpbXBsZUNvbmRpdGlvbi5kYXNoUHJvcGVydHlcclxuICAgICAgICApO1xyXG4gICAgICAgIGFwcGxpZWRDb25kaXRpb24gPSBzaW1wbGVDb25kaXRpb247XHJcbiAgICAgICAgaGFzQXBwbGllZFdoZXJlQ2xhdXNlID0gdHJ1ZTtcclxuICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8vIFNlIG7Do28gaMOhIGNvbmRpw6fDtWVzIGRpc3BvbsOtdmVpcywgdGVudGEgYXBsaWNhciBmaWx0cm9zIGRvIGFycmFuZ2VcclxuICAgIGxldCBhcHBsaWVkRmlyZXN0b3JlRmlsdGVyOiBBcnJhbmdlWydmaWx0ZXJzJ11bMF0gfCAnYWxsLWVxdWFscycgfCBudWxsID1cclxuICAgICAgbnVsbDtcclxuICAgIGlmICghaGFzQXBwbGllZFdoZXJlQ2xhdXNlICYmIHBhcmFtcy5hcnJhbmdlPy5maWx0ZXJzKSB7XHJcbiAgICAgIC8vIEFncnVwYXIgZXF1YWxzIGZpbHRlcnMgcG9yIHByb3ByaWVkYWRlXHJcbiAgICAgIGNvbnN0IGVxdWFsc0ZpbHRlcnMgPSBwYXJhbXMuYXJyYW5nZS5maWx0ZXJzLmZpbHRlcihcclxuICAgICAgICAoXHJcbiAgICAgICAgICBmXHJcbiAgICAgICAgKTogZiBpcyBBcnJhbmdlWydmaWx0ZXJzJ11bMF0gJiB7XHJcbiAgICAgICAgICBhcnJhbmdlOiAnZXF1YWxzJztcclxuICAgICAgICAgIGZpbHRlcjogeyBwcm9wZXJ0eTogc3RyaW5nOyBmaWx0ZXJpbmc6IHN0cmluZyB9O1xyXG4gICAgICAgIH0gPT4gZi5hcnJhbmdlID09PSAnZXF1YWxzJyAmJiAhIWYuZmlsdGVyXHJcbiAgICAgICk7XHJcblxyXG4gICAgICBjb25zdCBlcXVhbHNHcm91cGVkQnlQcm9wZXJ0eSA9IGVxdWFsc0ZpbHRlcnMucmVkdWNlKFxyXG4gICAgICAgIChhY2M6IFJlY29yZDxzdHJpbmcsIChzdHJpbmcgfCBib29sZWFuKVtdPiwgY3VycmVudCkgPT4ge1xyXG4gICAgICAgICAgY29uc3QgcHJvcCA9IGN1cnJlbnQuZmlsdGVyLnByb3BlcnR5O1xyXG4gICAgICAgICAgaWYgKCFhY2NbcHJvcF0pIHtcclxuICAgICAgICAgICAgYWNjW3Byb3BdID0gW107XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgICBhY2NbcHJvcF0ucHVzaChjdXJyZW50LmZpbHRlci5maWx0ZXJpbmcpO1xyXG4gICAgICAgICAgcmV0dXJuIGFjYztcclxuICAgICAgICB9LFxyXG4gICAgICAgIHt9IGFzIFJlY29yZDxzdHJpbmcsIChzdHJpbmcgfCBib29sZWFuKVtdPlxyXG4gICAgICApO1xyXG5cclxuICAgICAgLy8gU2UgaMOhIGFwZW5hcyBVTUEgcHJvcHJpZWRhZGUgY29tIG3Dumx0aXBsb3MgdmFsb3JlcywgYXBsaWNhciBubyBGaXJlc3RvcmUgY29tICdpbidcclxuICAgICAgY29uc3QgcHJvcGVydGllcyA9IE9iamVjdC5rZXlzKGVxdWFsc0dyb3VwZWRCeVByb3BlcnR5KTtcclxuICAgICAgaWYgKHByb3BlcnRpZXMubGVuZ3RoID09PSAxICYmIGVxdWFsc0ZpbHRlcnMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgIGNvbnN0IHByb3AgPSBwcm9wZXJ0aWVzWzBdO1xyXG4gICAgICAgIGNvbnN0IHZhbHVlcyA9IGVxdWFsc0dyb3VwZWRCeVByb3BlcnR5W3Byb3BdO1xyXG4gICAgICAgIHF1ZXJ5ID0gcXVlcnkud2hlcmUocHJvcCwgJ2luJywgdmFsdWVzKTtcclxuICAgICAgICBoYXNBcHBsaWVkV2hlcmVDbGF1c2UgPSB0cnVlO1xyXG4gICAgICAgIC8vIE1hcmNhciBUT0RPUyBvcyBlcXVhbHMgZmlsdGVycyBkZXNzYSBwcm9wcmllZGFkZSBjb21vIGFwbGljYWRvc1xyXG4gICAgICAgIGFwcGxpZWRGaXJlc3RvcmVGaWx0ZXIgPSAnYWxsLWVxdWFscyc7XHJcbiAgICAgIH0gZWxzZSBpZiAocHJvcGVydGllcy5sZW5ndGggPT09IDApIHtcclxuICAgICAgICBjb25zdCBvdGhlckZpbHRlciA9IHBhcmFtcy5hcnJhbmdlLmZpbHRlcnMuZmluZChcclxuICAgICAgICAgIChcclxuICAgICAgICAgICAgZlxyXG4gICAgICAgICAgKTogZiBpcyBBcnJhbmdlWydmaWx0ZXJzJ11bMF0gJlxyXG4gICAgICAgICAgICAoXHJcbiAgICAgICAgICAgICAgfCB7XHJcbiAgICAgICAgICAgICAgICAgIGFycmFuZ2U6ICdmaWx0ZXInO1xyXG4gICAgICAgICAgICAgICAgICBmaWx0ZXI6IHsgcHJvcGVydHk6IHN0cmluZzsgZmlsdGVyaW5nOiBzdHJpbmcgfTtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICB8IHtcclxuICAgICAgICAgICAgICAgICAgYXJyYW5nZTogJ2ZpbHRlckJ5RGF0ZSc7XHJcbiAgICAgICAgICAgICAgICAgIGRhdGVGaWx0ZXI6IHsgaW5pdGlhbDogRGF0ZTsgZmluYWw6IERhdGUgfTtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgKSA9PlxyXG4gICAgICAgICAgICAoZi5hcnJhbmdlID09PSAnZmlsdGVyJyAmJlxyXG4gICAgICAgICAgICAgICEhZi5maWx0ZXI/LmZpbHRlcmluZyAmJlxyXG4gICAgICAgICAgICAgICEhZi5maWx0ZXI/LnByb3BlcnR5KSB8fFxyXG4gICAgICAgICAgICAoZi5hcnJhbmdlID09PSAnZmlsdGVyQnlEYXRlJyAmJlxyXG4gICAgICAgICAgICAgICEhZi5kYXRlRmlsdGVyPy5pbml0aWFsICYmXHJcbiAgICAgICAgICAgICAgISFmLmRhdGVGaWx0ZXI/LmZpbmFsKVxyXG4gICAgICAgICk7XHJcblxyXG4gICAgICAgIGlmIChvdGhlckZpbHRlcikge1xyXG4gICAgICAgICAgaWYgKG90aGVyRmlsdGVyLmFycmFuZ2UgPT09ICdmaWx0ZXInICYmIG90aGVyRmlsdGVyLmZpbHRlcikge1xyXG4gICAgICAgICAgICBjb25zdCBmaWx0ZXJWYWx1ZSA9IG90aGVyRmlsdGVyLmZpbHRlci5maWx0ZXJpbmdcclxuICAgICAgICAgICAgICAudHJpbSgpXHJcbiAgICAgICAgICAgICAgLnRvVXBwZXJDYXNlKCk7XHJcbiAgICAgICAgICAgIHF1ZXJ5ID0gcXVlcnlcclxuICAgICAgICAgICAgICAud2hlcmUob3RoZXJGaWx0ZXIuZmlsdGVyLnByb3BlcnR5LCAnPj0nLCBmaWx0ZXJWYWx1ZSlcclxuICAgICAgICAgICAgICAud2hlcmUob3RoZXJGaWx0ZXIuZmlsdGVyLnByb3BlcnR5LCAnPD0nLCBmaWx0ZXJWYWx1ZSArICdcXHVmOGZmJyk7XHJcbiAgICAgICAgICAgIGhhc0FwcGxpZWRXaGVyZUNsYXVzZSA9IHRydWU7XHJcbiAgICAgICAgICAgIGFwcGxpZWRGaXJlc3RvcmVGaWx0ZXIgPSBvdGhlckZpbHRlcjtcclxuICAgICAgICAgIH0gZWxzZSBpZiAoXHJcbiAgICAgICAgICAgIG90aGVyRmlsdGVyLmFycmFuZ2UgPT09ICdmaWx0ZXJCeURhdGUnICYmXHJcbiAgICAgICAgICAgIG90aGVyRmlsdGVyLmRhdGVGaWx0ZXIgJiZcclxuICAgICAgICAgICAgcGFyYW1zLmFycmFuZ2Uuc29ydEJ5Py5maWVsZFxyXG4gICAgICAgICAgKSB7XHJcbiAgICAgICAgICAgIHF1ZXJ5ID0gcXVlcnlcclxuICAgICAgICAgICAgICAud2hlcmUoXHJcbiAgICAgICAgICAgICAgICBwYXJhbXMuYXJyYW5nZS5zb3J0QnkuZmllbGQsXHJcbiAgICAgICAgICAgICAgICAnPj0nLFxyXG4gICAgICAgICAgICAgICAgb3RoZXJGaWx0ZXIuZGF0ZUZpbHRlci5pbml0aWFsXHJcbiAgICAgICAgICAgICAgKVxyXG4gICAgICAgICAgICAgIC53aGVyZShcclxuICAgICAgICAgICAgICAgIHBhcmFtcy5hcnJhbmdlLnNvcnRCeS5maWVsZCxcclxuICAgICAgICAgICAgICAgICc8PScsXHJcbiAgICAgICAgICAgICAgICBvdGhlckZpbHRlci5kYXRlRmlsdGVyLmZpbmFsXHJcbiAgICAgICAgICAgICAgKTtcclxuICAgICAgICAgICAgaGFzQXBwbGllZFdoZXJlQ2xhdXNlID0gdHJ1ZTtcclxuICAgICAgICAgICAgYXBwbGllZEZpcmVzdG9yZUZpbHRlciA9IG90aGVyRmlsdGVyO1xyXG4gICAgICAgICAgfVxyXG4gICAgICAgIH1cclxuICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIGNvbnN0IGFsbERvY3NTbmFwc2hvdCA9IGF3YWl0IHF1ZXJ5LmdldCgpO1xyXG4gICAgbGV0IGl0ZW1zOiBBcnJheTxSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiAmIHsgaWQ6IHN0cmluZyB9PiA9XHJcbiAgICAgIGFsbERvY3NTbmFwc2hvdC5kb2NzLm1hcChcclxuICAgICAgICAoZG9jOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+KSA9PiB7XHJcbiAgICAgICAgICBjb25zdCBkYXRhID0gZG9jLmRhdGEoKSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiB8IHVuZGVmaW5lZDtcclxuICAgICAgICAgIHJldHVybiB7IGlkOiBkb2MuaWQsIC4uLihkYXRhID8/IHt9KSB9IGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+ICYge1xyXG4gICAgICAgICAgICBpZDogc3RyaW5nO1xyXG4gICAgICAgICAgfTtcclxuICAgICAgICB9XHJcbiAgICAgICk7XHJcblxyXG4gICAgLy8gQXBsaWNhciBjb25kacOnw7VlcyByZXN0YW50ZXNcclxuICAgIGlmIChwYXJhbXMuY29uZGl0aW9ucykge1xyXG4gICAgICBjb25zdCByZW1haW5pbmdDb25kaXRpb25zID0gcGFyYW1zLmNvbmRpdGlvbnMuZmlsdGVyKFxyXG4gICAgICAgIChjb25kOiBDb25kaXRpb24pID0+IGNvbmQgIT09IGFwcGxpZWRDb25kaXRpb25cclxuICAgICAgKTtcclxuICAgICAgaWYgKHJlbWFpbmluZ0NvbmRpdGlvbnMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgIGNvbnN0IG9wZXJhdG9ycyA9IHRoaXMub3BlcmF0b3JzO1xyXG4gICAgICAgIGl0ZW1zID0gaXRlbXMuZmlsdGVyKChpdGVtKSA9PiB7XHJcbiAgICAgICAgICByZXR1cm4gcmVtYWluaW5nQ29uZGl0aW9ucy5ldmVyeSgoY29uZDogQ29uZGl0aW9uKSA9PiB7XHJcbiAgICAgICAgICAgIGNvbnN0IG9wZXJhdG9yRm4gPVxyXG4gICAgICAgICAgICAgIG9wZXJhdG9yc1tjb25kLm9wZXJhdG9yIGFzIGtleW9mIHR5cGVvZiBvcGVyYXRvcnNdO1xyXG4gICAgICAgICAgICByZXR1cm4gb3BlcmF0b3JGblxyXG4gICAgICAgICAgICAgID8gb3BlcmF0b3JGbihpdGVtW2NvbmQuZmlyZXN0b3JlUHJvcGVydHldLCBjb25kLmRhc2hQcm9wZXJ0eSlcclxuICAgICAgICAgICAgICA6IGZhbHNlO1xyXG4gICAgICAgICAgfSk7XHJcbiAgICAgICAgfSk7XHJcbiAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICBjb25zdCB7IGZpbHRlcnMsIHNvcnRCeSB9ID0gcGFyYW1zLmFycmFuZ2U7XHJcblxyXG4gICAgLy8gVHJhY2sgd2hpY2ggZmlsdGVyIHdhcyBhbHJlYWR5IGFwcGxpZWQgaW4gRmlyZXN0b3JlIHRvIGF2b2lkIGRvdWJsZSBmaWx0ZXJpbmdcclxuICAgIGlmIChoYXNBcHBsaWVkV2hlcmVDbGF1c2UgJiYgIWFwcGxpZWRDb25kaXRpb24gJiYgcGFyYW1zLmFycmFuZ2U/LmZpbHRlcnMpIHtcclxuICAgICAgY29uc3QgZXF1YWxzRmlsdGVyID0gcGFyYW1zLmFycmFuZ2UuZmlsdGVycy5maW5kKFxyXG4gICAgICAgIChcclxuICAgICAgICAgIGZcclxuICAgICAgICApOiBmIGlzIEFycmFuZ2VbJ2ZpbHRlcnMnXVswXSAmIHtcclxuICAgICAgICAgIGFycmFuZ2U6ICdlcXVhbHMnO1xyXG4gICAgICAgICAgZmlsdGVyOiB7IHByb3BlcnR5OiBzdHJpbmc7IGZpbHRlcmluZzogc3RyaW5nIH07XHJcbiAgICAgICAgfSA9PiBmLmFycmFuZ2UgPT09ICdlcXVhbHMnICYmICEhZi5maWx0ZXI/LmZpbHRlcmluZ1xyXG4gICAgICApO1xyXG4gICAgICBpZiAoZXF1YWxzRmlsdGVyKSB7XHJcbiAgICAgICAgYXBwbGllZEZpcmVzdG9yZUZpbHRlciA9IGVxdWFsc0ZpbHRlcjtcclxuICAgICAgfSBlbHNlIHtcclxuICAgICAgICBjb25zdCBmb3VuZEZpbHRlciA9IHBhcmFtcy5hcnJhbmdlLmZpbHRlcnMuZmluZChcclxuICAgICAgICAgIChcclxuICAgICAgICAgICAgZlxyXG4gICAgICAgICAgKTogZiBpcyBBcnJhbmdlWydmaWx0ZXJzJ11bMF0gJlxyXG4gICAgICAgICAgICAoXHJcbiAgICAgICAgICAgICAgfCB7XHJcbiAgICAgICAgICAgICAgICAgIGFycmFuZ2U6ICdmaWx0ZXInO1xyXG4gICAgICAgICAgICAgICAgICBmaWx0ZXI6IHsgcHJvcGVydHk6IHN0cmluZzsgZmlsdGVyaW5nOiBzdHJpbmcgfTtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICB8IHtcclxuICAgICAgICAgICAgICAgICAgYXJyYW5nZTogJ2ZpbHRlckJ5RGF0ZSc7XHJcbiAgICAgICAgICAgICAgICAgIGRhdGVGaWx0ZXI6IHsgaW5pdGlhbDogRGF0ZTsgZmluYWw6IERhdGUgfTtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgKSA9PlxyXG4gICAgICAgICAgICAoZi5hcnJhbmdlID09PSAnZmlsdGVyJyAmJlxyXG4gICAgICAgICAgICAgICEhZi5maWx0ZXI/LmZpbHRlcmluZyAmJlxyXG4gICAgICAgICAgICAgICEhZi5maWx0ZXI/LnByb3BlcnR5KSB8fFxyXG4gICAgICAgICAgICAoZi5hcnJhbmdlID09PSAnZmlsdGVyQnlEYXRlJyAmJlxyXG4gICAgICAgICAgICAgICEhZi5kYXRlRmlsdGVyPy5pbml0aWFsICYmXHJcbiAgICAgICAgICAgICAgISFmLmRhdGVGaWx0ZXI/LmZpbmFsKVxyXG4gICAgICAgICk7XHJcbiAgICAgICAgaWYgKGZvdW5kRmlsdGVyKSB7XHJcbiAgICAgICAgICBhcHBsaWVkRmlyZXN0b3JlRmlsdGVyID0gZm91bmRGaWx0ZXI7XHJcbiAgICAgICAgfVxyXG4gICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgY29uc3QgZXF1YWxzRmlsdGVycyA9IGZpbHRlcnMuZmlsdGVyKFxyXG4gICAgICAoZik6IGYgaXMgQXJyYW5nZVsnZmlsdGVycyddWzBdICYgeyBhcnJhbmdlOiAnZXF1YWxzJyB9ID0+XHJcbiAgICAgICAgZi5hcnJhbmdlID09PSAnZXF1YWxzJ1xyXG4gICAgKTtcclxuICAgIGNvbnN0IG90aGVyRmlsdGVycyA9IGZpbHRlcnMuZmlsdGVyKFxyXG4gICAgICAoZik6IGYgaXMgQXJyYW5nZVsnZmlsdGVycyddWzBdID0+IGYuYXJyYW5nZSAhPT0gJ2VxdWFscydcclxuICAgICk7XHJcblxyXG4gICAgLy8gQXBsaWNhciBlcXVhbHMgZmlsdGVycyBubyBjbGllbnQtc2lkZSBhcGVuYXMgc2UgbsOjbyBmb3JhbSBhcGxpY2Fkb3Mgbm8gRmlyZXN0b3JlXHJcbiAgICBpZiAoYXBwbGllZEZpcmVzdG9yZUZpbHRlciAhPT0gJ2FsbC1lcXVhbHMnICYmIGVxdWFsc0ZpbHRlcnMubGVuZ3RoID4gMCkge1xyXG4gICAgICAvLyBBZ3J1cGFyIHBvciBwcm9wcmllZGFkZSBwYXJhIGFwbGljYXIgT1IgZGVudHJvIGRlIGNhZGEgcHJvcHJpZWRhZGVcclxuICAgICAgY29uc3QgZ3JvdXBlZEJ5UHJvcGVydHkgPSBlcXVhbHNGaWx0ZXJzLnJlZHVjZShcclxuICAgICAgICAoYWNjOiBSZWNvcmQ8c3RyaW5nLCAoc3RyaW5nIHwgYm9vbGVhbilbXT4sIGYpID0+IHtcclxuICAgICAgICAgIGlmIChmLmZpbHRlcikge1xyXG4gICAgICAgICAgICBjb25zdCBwcm9wID0gZi5maWx0ZXIucHJvcGVydHk7XHJcbiAgICAgICAgICAgIGlmICghYWNjW3Byb3BdKSB7XHJcbiAgICAgICAgICAgICAgYWNjW3Byb3BdID0gW107XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgYWNjW3Byb3BdLnB1c2goZi5maWx0ZXIuZmlsdGVyaW5nKTtcclxuICAgICAgICAgIH1cclxuICAgICAgICAgIHJldHVybiBhY2M7XHJcbiAgICAgICAgfSxcclxuICAgICAgICB7fSBhcyBSZWNvcmQ8c3RyaW5nLCAoc3RyaW5nIHwgYm9vbGVhbilbXT5cclxuICAgICAgKTtcclxuXHJcbiAgICAgIC8vIEZpbHRyYXI6IGl0ZW0gZGV2ZSB0ZXIgcGVsbyBtZW5vcyB1bSB2YWxvciBkZSBDQURBIHByb3ByaWVkYWRlXHJcbiAgICAgIC8vIChBTkQgZW50cmUgcHJvcHJpZWRhZGVzLCBPUiBkZW50cm8gZGUgY2FkYSBwcm9wcmllZGFkZSlcclxuICAgICAgaXRlbXMgPSBpdGVtcy5maWx0ZXIoKGl0ZW0pID0+IHtcclxuICAgICAgICByZXR1cm4gT2JqZWN0LmVudHJpZXMoZ3JvdXBlZEJ5UHJvcGVydHkpLmV2ZXJ5KChbcHJvcCwgdmFsdWVzXSkgPT4ge1xyXG4gICAgICAgICAgY29uc3QgaXRlbVZhbHVlID0gaXRlbVtwcm9wXTtcclxuICAgICAgICAgIHJldHVybiB2YWx1ZXMuaW5jbHVkZXMoaXRlbVZhbHVlIGFzIHN0cmluZyB8IGJvb2xlYW4pO1xyXG4gICAgICAgIH0pO1xyXG4gICAgICB9KTtcclxuICAgIH1cclxuXHJcbiAgICBvdGhlckZpbHRlcnMuZm9yRWFjaCgoZmlsdGVySXRlbSkgPT4ge1xyXG4gICAgICBpZiAoYXBwbGllZEZpcmVzdG9yZUZpbHRlciA9PT0gZmlsdGVySXRlbSkge1xyXG4gICAgICAgIHJldHVybjtcclxuICAgICAgfVxyXG5cclxuICAgICAgaWYgKFxyXG4gICAgICAgIGZpbHRlckl0ZW0uYXJyYW5nZSA9PT0gJ2ZpbHRlcicgJiZcclxuICAgICAgICBmaWx0ZXJJdGVtLmZpbHRlcj8uZmlsdGVyaW5nICYmXHJcbiAgICAgICAgZmlsdGVySXRlbS5maWx0ZXI/LnByb3BlcnR5XHJcbiAgICAgICkge1xyXG4gICAgICAgIGNvbnN0IGZpbHRlclZhbHVlID0gU3RyaW5nKGZpbHRlckl0ZW0uZmlsdGVyLmZpbHRlcmluZylcclxuICAgICAgICAgIC50cmltKClcclxuICAgICAgICAgIC50b0xvd2VyQ2FzZSgpO1xyXG4gICAgICAgIGl0ZW1zID0gaXRlbXMuZmlsdGVyKChpdGVtKSA9PiB7XHJcbiAgICAgICAgICBjb25zdCBpdGVtVmFsdWUgPSBTdHJpbmcoXHJcbiAgICAgICAgICAgIGl0ZW1bZmlsdGVySXRlbS5maWx0ZXIhLnByb3BlcnR5XSA/PyAnJ1xyXG4gICAgICAgICAgKS50b0xvd2VyQ2FzZSgpO1xyXG4gICAgICAgICAgcmV0dXJuIGl0ZW1WYWx1ZS5pbmNsdWRlcyhmaWx0ZXJWYWx1ZSk7XHJcbiAgICAgICAgfSk7XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIGlmIChcclxuICAgICAgICBmaWx0ZXJJdGVtLmFycmFuZ2UgPT09ICdmaWx0ZXJCeURhdGUnICYmXHJcbiAgICAgICAgZmlsdGVySXRlbS5kYXRlRmlsdGVyPy5pbml0aWFsICYmXHJcbiAgICAgICAgZmlsdGVySXRlbS5kYXRlRmlsdGVyPy5maW5hbCAmJlxyXG4gICAgICAgIHNvcnRCeS5maWVsZFxyXG4gICAgICApIHtcclxuICAgICAgICBpdGVtcyA9IGl0ZW1zLmZpbHRlcigoaXRlbSkgPT4ge1xyXG4gICAgICAgICAgdHJ5IHtcclxuICAgICAgICAgICAgY29uc3QgZmllbGRWYWx1ZSA9IGl0ZW1bc29ydEJ5LmZpZWxkXTtcclxuXHJcbiAgICAgICAgICAgIGlmICghZmllbGRWYWx1ZSkge1xyXG4gICAgICAgICAgICAgIHJldHVybiBmYWxzZTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgbGV0IGl0ZW1EYXRlOiBEYXRlO1xyXG4gICAgICAgICAgICBjb25zdCBmaWVsZFZhbHVlT2JqID0gZmllbGRWYWx1ZSBhc1xyXG4gICAgICAgICAgICAgIHwgeyB0b0RhdGU/OiAoKSA9PiBEYXRlIH1cclxuICAgICAgICAgICAgICB8IERhdGVcclxuICAgICAgICAgICAgICB8IHN0cmluZ1xyXG4gICAgICAgICAgICAgIHwgbnVtYmVyXHJcbiAgICAgICAgICAgICAgfCBudWxsXHJcbiAgICAgICAgICAgICAgfCB1bmRlZmluZWQ7XHJcbiAgICAgICAgICAgIGlmIChcclxuICAgICAgICAgICAgICBmaWVsZFZhbHVlT2JqICYmXHJcbiAgICAgICAgICAgICAgdHlwZW9mIGZpZWxkVmFsdWVPYmogPT09ICdvYmplY3QnICYmXHJcbiAgICAgICAgICAgICAgJ3RvRGF0ZScgaW4gZmllbGRWYWx1ZU9iaiAmJlxyXG4gICAgICAgICAgICAgIHR5cGVvZiBmaWVsZFZhbHVlT2JqLnRvRGF0ZSA9PT0gJ2Z1bmN0aW9uJ1xyXG4gICAgICAgICAgICApIHtcclxuICAgICAgICAgICAgICBpdGVtRGF0ZSA9IGZpZWxkVmFsdWVPYmoudG9EYXRlKCk7XHJcbiAgICAgICAgICAgIH0gZWxzZSBpZiAoZmllbGRWYWx1ZU9iaiBpbnN0YW5jZW9mIERhdGUpIHtcclxuICAgICAgICAgICAgICBpdGVtRGF0ZSA9IGZpZWxkVmFsdWVPYmo7XHJcbiAgICAgICAgICAgIH0gZWxzZSBpZiAodHlwZW9mIGZpZWxkVmFsdWVPYmogPT09ICdzdHJpbmcnKSB7XHJcbiAgICAgICAgICAgICAgaXRlbURhdGUgPSBuZXcgRGF0ZShmaWVsZFZhbHVlT2JqKTtcclxuICAgICAgICAgICAgICBpZiAoaXNOYU4oaXRlbURhdGUuZ2V0VGltZSgpKSkge1xyXG4gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlO1xyXG4gICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgfSBlbHNlIGlmICh0eXBlb2YgZmllbGRWYWx1ZU9iaiA9PT0gJ251bWJlcicpIHtcclxuICAgICAgICAgICAgICBpdGVtRGF0ZSA9IG5ldyBEYXRlKGZpZWxkVmFsdWVPYmopO1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgIHJldHVybiBmYWxzZTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgaWYgKCFmaWx0ZXJJdGVtLmRhdGVGaWx0ZXIpIHtcclxuICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICAgICAgICAgIH1cclxuXHJcbiAgICAgICAgICAgIHJldHVybiAoXHJcbiAgICAgICAgICAgICAgaXRlbURhdGUgPj0gZmlsdGVySXRlbS5kYXRlRmlsdGVyLmluaXRpYWwgJiZcclxuICAgICAgICAgICAgICBpdGVtRGF0ZSA8PSBmaWx0ZXJJdGVtLmRhdGVGaWx0ZXIuZmluYWxcclxuICAgICAgICAgICAgKTtcclxuICAgICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XHJcbiAgICAgICAgICAgIGNvbnNvbGUud2FybihcclxuICAgICAgICAgICAgICAnRXJybyBhbyBwcm9jZXNzYXIgZmlsdHJvIGRlIGRhdGEgcGFyYSBvIGl0ZW06JyxcclxuICAgICAgICAgICAgICBpdGVtLmlkLFxyXG4gICAgICAgICAgICAgIGVycm9yXHJcbiAgICAgICAgICAgICk7XHJcbiAgICAgICAgICAgIHJldHVybiBmYWxzZTtcclxuICAgICAgICAgIH1cclxuICAgICAgICB9KTtcclxuICAgICAgfVxyXG4gICAgfSk7XHJcblxyXG4gICAgLy8gQXBsaWNhciBmaWx0ZXJGbiBzZSBleGlzdGlyXHJcbiAgICBpZiAocGFyYW1zLmZpbHRlckZuKSB7XHJcbiAgICAgIGl0ZW1zID0gaXRlbXMuZmlsdGVyKHBhcmFtcy5maWx0ZXJGbik7XHJcbiAgICB9XHJcblxyXG4gICAgaWYgKHNvcnRCeSAmJiBzb3J0QnkuZmllbGQgJiYgc29ydEJ5Lm9yZGVyKSB7XHJcbiAgICAgIGl0ZW1zLnNvcnQoKGEsIGIpID0+IHtcclxuICAgICAgICBjb25zdCB2YWxBID0gYVtzb3J0QnkuZmllbGRdIGFzIHVua25vd247XHJcbiAgICAgICAgY29uc3QgdmFsQiA9IGJbc29ydEJ5LmZpZWxkXSBhcyB1bmtub3duO1xyXG5cclxuICAgICAgICBpZiAodmFsQSA9PT0gbnVsbCB8fCB2YWxBID09PSB1bmRlZmluZWQpIHJldHVybiAxO1xyXG4gICAgICAgIGlmICh2YWxCID09PSBudWxsIHx8IHZhbEIgPT09IHVuZGVmaW5lZCkgcmV0dXJuIC0xO1xyXG5cclxuICAgICAgICBpZiAodmFsQSA8IHZhbEIpIHtcclxuICAgICAgICAgIHJldHVybiBzb3J0Qnkub3JkZXIgPT09ICdhc2MnID8gLTEgOiAxO1xyXG4gICAgICAgIH1cclxuICAgICAgICBpZiAodmFsQSA+IHZhbEIpIHtcclxuICAgICAgICAgIHJldHVybiBzb3J0Qnkub3JkZXIgPT09ICdhc2MnID8gMSA6IC0xO1xyXG4gICAgICAgIH1cclxuICAgICAgICByZXR1cm4gMDtcclxuICAgICAgfSk7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gSW1wbGVtZW50YcOnw6NvIGFkZXF1YWRhIGRhIHBhZ2luYcOnw6NvXHJcbiAgICBsZXQgY3VycmVudENsaWVudFBhZ2VJbmRleCA9IDA7XHJcblxyXG4gICAgLy8gRGV0ZXJtaW5hciBhIHDDoWdpbmEgYXR1YWwgYmFzZWFkYSBuYSBuYXZlZ2HDp8Ojb1xyXG4gICAgaWYgKHBhcmFtcy5uYXZpZ2F0aW9uID09PSAncmVsb2FkJykge1xyXG4gICAgICBjdXJyZW50Q2xpZW50UGFnZUluZGV4ID0gMDtcclxuICAgIH0gZWxzZSB7XHJcbiAgICAgIGN1cnJlbnRDbGllbnRQYWdlSW5kZXggPSBwYXJhbXMuY2xpZW50UGFnZUluZGV4IHx8IDA7XHJcbiAgICB9XHJcblxyXG4gICAgY29uc3QgcGFnZVNpemUgPSBwYXJhbXMuYmF0Y2hTaXplO1xyXG4gICAgY29uc3Qgc3RhcnRJbmRleCA9IGN1cnJlbnRDbGllbnRQYWdlSW5kZXggKiBwYWdlU2l6ZTtcclxuICAgIGNvbnN0IGVuZEluZGV4ID0gc3RhcnRJbmRleCArIHBhZ2VTaXplO1xyXG4gICAgY29uc3QgcGFnaW5hdGVkSXRlbXMgPSBpdGVtcy5zbGljZShzdGFydEluZGV4LCBlbmRJbmRleCk7XHJcblxyXG4gICAgY29uc3QgdG90YWxQYWdlcyA9IE1hdGguY2VpbChpdGVtcy5sZW5ndGggLyBwYWdlU2l6ZSk7XHJcbiAgICBjb25zdCBoYXNOZXh0UGFnZSA9IGN1cnJlbnRDbGllbnRQYWdlSW5kZXggPCB0b3RhbFBhZ2VzIC0gMTtcclxuICAgIGNvbnN0IGhhc1ByZXZpb3VzUGFnZSA9IGN1cnJlbnRDbGllbnRQYWdlSW5kZXggPiAwO1xyXG5cclxuICAgIHJldHVybiB7XHJcbiAgICAgIGl0ZW1zOiBwYWdpbmF0ZWRJdGVtcyxcclxuICAgICAgZmlsdGVyTGVuZ3RoOiBpdGVtcy5sZW5ndGgsXHJcbiAgICAgIGxhc3REb2M6IG51bGwsXHJcbiAgICAgIGZpcnN0RG9jOiBudWxsLFxyXG4gICAgICBoYXNOZXh0UGFnZTogaGFzTmV4dFBhZ2UsXHJcbiAgICAgIGhhc1ByZXZpb3VzUGFnZTogaGFzUHJldmlvdXNQYWdlLFxyXG4gICAgICBjdXJyZW50Q2xpZW50UGFnZUluZGV4OiBjdXJyZW50Q2xpZW50UGFnZUluZGV4LFxyXG4gICAgICB0b3RhbFBhZ2VzOiB0b3RhbFBhZ2VzLFxyXG4gICAgfTtcclxuICB9XHJcblxyXG4gIGFzeW5jIGdldEl0ZW1zRGF0YShcclxuICAgIGNvbGxlY3Rpb246IHN0cmluZyxcclxuICAgIGFycmFuZ2U6IEFycmFuZ2UsXHJcbiAgICBjb25kaXRpb25zOiBDb25kaXRpb25bXSB8IHVuZGVmaW5lZCA9IHVuZGVmaW5lZFxyXG4gICk6IFByb21pc2U8QXJyYXk8UmVjb3JkPHN0cmluZywgdW5rbm93bj4gJiB7IGlkOiBzdHJpbmcgfT4+IHtcclxuICAgIHRyeSB7XHJcbiAgICAgIGxldCBxdWVyeTogZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5PHVua25vd24+ID1cclxuICAgICAgICB0aGlzLm5nRmlyZS5jb2xsZWN0aW9uKGNvbGxlY3Rpb24pLnJlZjtcclxuXHJcbiAgICAgIHF1ZXJ5ID0gdGhpcy5hcHBseUZpbHRlcnMocXVlcnksIGFycmFuZ2UsIGNvbmRpdGlvbnMpO1xyXG4gICAgICBjb25zdCBzbmFwc2hvdCA9IGF3YWl0IHF1ZXJ5LmdldCgpO1xyXG4gICAgICByZXR1cm4gYXdhaXQgUHJvbWlzZS5hbGwoXHJcbiAgICAgICAgc25hcHNob3QuZG9jcy5tYXAoXHJcbiAgICAgICAgICBhc3luYyAoZG9jOiBmaXJlYmFzZS5maXJlc3RvcmUuUXVlcnlEb2N1bWVudFNuYXBzaG90PHVua25vd24+KSA9PiB7XHJcbiAgICAgICAgICAgIGNvbnN0IGRhdGEgPSBkb2MuZGF0YSgpIGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+IHwgdW5kZWZpbmVkO1xyXG4gICAgICAgICAgICBjb25zdCBpZCA9IGRvYy5pZDtcclxuICAgICAgICAgICAgcmV0dXJuIHtcclxuICAgICAgICAgICAgICBpZCxcclxuICAgICAgICAgICAgICAuLi4oZGF0YSA/PyB7fSksXHJcbiAgICAgICAgICAgIH0gYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gJiB7IGlkOiBzdHJpbmcgfTtcclxuICAgICAgICAgIH1cclxuICAgICAgICApXHJcbiAgICAgICk7XHJcbiAgICB9IGNhdGNoIChlKSB7XHJcbiAgICAgIHRocm93IGU7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBwdWJsaWMgb3BlcmF0b3JzID0ge1xyXG4gICAgJz09JzogKGE6IHVua25vd24sIGI6IHVua25vd24pOiBib29sZWFuID0+IGEgPT09IGIsXHJcbiAgICAnIT0nOiAoYTogdW5rbm93biwgYjogdW5rbm93bik6IGJvb2xlYW4gPT4gYSAhPT0gYixcclxuICAgICc+JzogKGE6IHVua25vd24sIGI6IHVua25vd24pOiBib29sZWFuID0+IChhIGFzIG51bWJlcikgPiAoYiBhcyBudW1iZXIpLFxyXG4gICAgJzwnOiAoYTogdW5rbm93biwgYjogdW5rbm93bik6IGJvb2xlYW4gPT4gKGEgYXMgbnVtYmVyKSA8IChiIGFzIG51bWJlciksXHJcbiAgICAnPj0nOiAoYTogdW5rbm93biwgYjogdW5rbm93bik6IGJvb2xlYW4gPT4gKGEgYXMgbnVtYmVyKSA+PSAoYiBhcyBudW1iZXIpLFxyXG4gICAgJzw9JzogKGE6IHVua25vd24sIGI6IHVua25vd24pOiBib29sZWFuID0+IChhIGFzIG51bWJlcikgPD0gKGIgYXMgbnVtYmVyKSxcclxuICAgIGluOiAoYTogdW5rbm93biwgYjogdW5rbm93bik6IGJvb2xlYW4gPT4gQXJyYXkuaXNBcnJheShiKSAmJiBiLmluY2x1ZGVzKGEpLFxyXG4gICAgJ25vdC1pbic6IChhOiB1bmtub3duLCBiOiB1bmtub3duKTogYm9vbGVhbiA9PlxyXG4gICAgICBBcnJheS5pc0FycmF5KGIpICYmICFiLmluY2x1ZGVzKGEpLFxyXG4gICAgJ2FycmF5LWNvbnRhaW5zJzogKGE6IHVua25vd24sIGI6IHVua25vd24pOiBib29sZWFuID0+XHJcbiAgICAgIEFycmF5LmlzQXJyYXkoYSkgJiYgYS5pbmNsdWRlcyhiKSxcclxuICAgICdhcnJheS1jb250YWlucy1hbnknOiAoYTogdW5rbm93biwgYjogdW5rbm93bik6IGJvb2xlYW4gPT5cclxuICAgICAgQXJyYXkuaXNBcnJheShhKSAmJlxyXG4gICAgICBBcnJheS5pc0FycmF5KGIpICYmXHJcbiAgICAgIGIuc29tZSgoaXRlbTogdW5rbm93bikgPT4gYS5pbmNsdWRlcyhpdGVtKSksXHJcbiAgICBpbmNsdWRlczogKGE6IHVua25vd24sIGI6IHVua25vd24pOiBib29sZWFuID0+IHtcclxuICAgICAgaWYgKHR5cGVvZiBhID09PSAnc3RyaW5nJyAmJiB0eXBlb2YgYiA9PT0gJ3N0cmluZycpIHtcclxuICAgICAgICByZXR1cm4gYS5pbmNsdWRlcyhiKTtcclxuICAgICAgfVxyXG4gICAgICBpZiAoQXJyYXkuaXNBcnJheShhKSkge1xyXG4gICAgICAgIHJldHVybiBhLmluY2x1ZGVzKGIpO1xyXG4gICAgICB9XHJcbiAgICAgIHJldHVybiBmYWxzZTtcclxuICAgIH0sXHJcbiAgfTtcclxuXHJcbiAgYXN5bmMgZGVsZXRlSW5kZXgoaWQ6IHN0cmluZywgY29sOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+IHtcclxuICAgIHRyeSB7XHJcbiAgICAgIGNvbnN0IGJhdGNoID0gdGhpcy5uZ0ZpcmUuZmlyZXN0b3JlLmJhdGNoKCk7XHJcblxyXG4gICAgICBjb25zdCBkb2NSZWYgPSB0aGlzLm5nRmlyZS5jb2xsZWN0aW9uKGNvbCkuZG9jKGlkKTtcclxuICAgICAgY29uc3QgZG9jU25hcHNob3QgPSAoYXdhaXQgZmlyc3RWYWx1ZUZyb20oXHJcbiAgICAgICAgZG9jUmVmLmdldCgpXHJcbiAgICAgICkpIGFzIGZpcmViYXNlLmZpcmVzdG9yZS5Eb2N1bWVudFNuYXBzaG90PHVua25vd24+O1xyXG4gICAgICBjb25zdCBkb2MgPSBkb2NTbmFwc2hvdC5kYXRhKCkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gfCB1bmRlZmluZWQ7XHJcbiAgICAgIGJhdGNoLmRlbGV0ZShkb2NSZWYucmVmKTtcclxuICAgICAgaWYgKGRvYyAmJiB0eXBlb2YgZG9jWydpbmRleCddID09PSAnbnVtYmVyJykge1xyXG4gICAgICAgIGF3YWl0IHRoaXMucmVpbmRleChkb2NbJ2luZGV4J10gYXMgbnVtYmVyLCBjb2wsIGJhdGNoKTtcclxuICAgICAgfVxyXG4gICAgICBhd2FpdCBiYXRjaC5jb21taXQoKTtcclxuXHJcbiAgICAgIHRoaXMudG9hc3RyLnN1Y2Nlc3MoJ0l0ZW0gZXhjbHXDrWRvIGNvbSBzdWNlc3NvIScpO1xyXG4gICAgICByZXR1cm4gdHJ1ZTtcclxuICAgIH0gY2F0Y2ggKGUpIHtcclxuICAgICAgY29uc3QgZXJyb3IgPSBlIGFzIHVua25vd247XHJcbiAgICAgIGNvbnNvbGUuZXJyb3IoJ0Vycm8gYW8gZGVsZXRhciBpdGVtOicsIGVycm9yKTtcclxuICAgICAgdGhpcy50b2FzdHIuZXJyb3IoJ0Vycm8gYW8gZGVsZXRhciBpdGVtLicpO1xyXG4gICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBhc3luYyByZWluZGV4KFxyXG4gICAgaW5kZXg6IG51bWJlcixcclxuICAgIGNvbDogc3RyaW5nLFxyXG4gICAgYmF0Y2g6IGZpcmViYXNlLmZpcmVzdG9yZS5Xcml0ZUJhdGNoXHJcbiAgKTogUHJvbWlzZTx2b2lkPiB7XHJcbiAgICB0cnkge1xyXG4gICAgICBjb25zdCBzbmFwc2hvdCA9IChhd2FpdCBmaXJzdFZhbHVlRnJvbShcclxuICAgICAgICB0aGlzLm5nRmlyZS5jb2xsZWN0aW9uKGNvbCkuZ2V0KClcclxuICAgICAgKSkgYXMgZmlyZWJhc2UuZmlyZXN0b3JlLlF1ZXJ5U25hcHNob3Q8dW5rbm93bj47XHJcbiAgICAgIGNvbnN0IGRvY3MgPSBzbmFwc2hvdC5kb2NzO1xyXG4gICAgICBmb3IgKGxldCBkb2Mgb2YgZG9jcykge1xyXG4gICAgICAgIGNvbnN0IGRhdGEgPSBkb2MuZGF0YSgpIGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+IHwgdW5kZWZpbmVkO1xyXG4gICAgICAgIGlmIChcclxuICAgICAgICAgIGRhdGEgJiZcclxuICAgICAgICAgIHR5cGVvZiBkYXRhWydpbmRleCddID09PSAnbnVtYmVyJyAmJlxyXG4gICAgICAgICAgKGRhdGFbJ2luZGV4J10gYXMgbnVtYmVyKSA+IGluZGV4XHJcbiAgICAgICAgKSB7XHJcbiAgICAgICAgICBkYXRhWydpbmRleCddID0gKChkYXRhWydpbmRleCddIGFzIG51bWJlcikgLSAxKSBhcyB1bmtub3duO1xyXG4gICAgICAgICAgY29uc3QgZG9jUmVmID0gdGhpcy5uZ0ZpcmUuY29sbGVjdGlvbihjb2wpLmRvYyhkb2MuaWQpLnJlZjtcclxuICAgICAgICAgIGJhdGNoLnVwZGF0ZShkb2NSZWYsIGRhdGEpO1xyXG4gICAgICAgIH1cclxuICAgICAgfVxyXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgY29uc29sZS5lcnJvcignRXJybyBhbyByZWluZGV4YXI6JywgZXJyb3IpO1xyXG4gICAgfVxyXG4gICAgcmV0dXJuO1xyXG4gIH1cclxuXHJcbiAgZGF0ZUZvcm1hdFZhbGlkYXRvcigpOiBWYWxpZGF0b3JGbiB7XHJcbiAgICByZXR1cm4gKGNvbnRyb2w6IEFic3RyYWN0Q29udHJvbCk6IHsgW2tleTogc3RyaW5nXTogdW5rbm93biB9IHwgbnVsbCA9PiB7XHJcbiAgICAgIGlmICghY29udHJvbC52YWx1ZSkge1xyXG4gICAgICAgIHJldHVybiBudWxsO1xyXG4gICAgICB9XHJcblxyXG4gICAgICBjb25zdCBkYXRlU3RyID0gY29udHJvbC52YWx1ZS50cmltKCk7XHJcbiAgICAgIGNvbnN0IGRhdGVQYXR0ZXJuID0gL14oXFxkezEsMn0pXFwvKFxcZHsxLDJ9KVxcLyhcXGR7NH0pJC87XHJcblxyXG4gICAgICBpZiAoIWRhdGVQYXR0ZXJuLnRlc3QoZGF0ZVN0cikpIHtcclxuICAgICAgICByZXR1cm4geyBpbnZhbGlkRm9ybWF0OiB0cnVlIH07XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIGNvbnN0IHBhcnRzID0gZGF0ZVN0ci5zcGxpdCgnLycpO1xyXG4gICAgICBjb25zdCBkYXkgPSBwYXJ0c1swXS5wYWRTdGFydCgyLCAnMCcpO1xyXG4gICAgICBjb25zdCBtb250aCA9IHBhcnRzWzFdLnBhZFN0YXJ0KDIsICcwJyk7XHJcbiAgICAgIGNvbnN0IHllYXIgPSBwYXJ0c1syXTtcclxuICAgICAgY29uc3Qgbm9ybWFsaXplZERhdGUgPSBgJHtkYXl9LyR7bW9udGh9LyR7eWVhcn1gO1xyXG5cclxuICAgICAgY29uc3QgZGF0ZSA9IG1vbWVudChub3JtYWxpemVkRGF0ZSwgJ0REL01NL1lZWVknLCB0cnVlKTtcclxuXHJcbiAgICAgIGlmICghZGF0ZS5pc1ZhbGlkKCkpIHtcclxuICAgICAgICByZXR1cm4geyBpbnZhbGlkRGF0ZTogdHJ1ZSB9O1xyXG4gICAgICB9XHJcblxyXG4gICAgICByZXR1cm4gbnVsbDtcclxuICAgIH07XHJcbiAgfVxyXG5cclxuICBhc3luYyB1cGRhdGVJbmRleChpbmRleDogbnVtYmVyLCBpZDogc3RyaW5nLCBjb2w6IHN0cmluZyk6IFByb21pc2U8dm9pZD4ge1xyXG4gICAgYXdhaXQgdGhpcy5uZ0ZpcmUuY29sbGVjdGlvbihjb2wpLmRvYyhpZCkudXBkYXRlKHsgaW5kZXggfSk7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBFeHRyYWkgbyBsaW5rIGRlIGNyaWHDp8OjbyBkZSDDrW5kaWNlIGRhIG1lbnNhZ2VtIGRlIGVycm8gZG8gRmlyZXN0b3JlXHJcbiAgICovXHJcbiAgcHJpdmF0ZSBleHRyYWN0SW5kZXhMaW5rKGVycm9yOiB1bmtub3duKTogc3RyaW5nIHwgbnVsbCB7XHJcbiAgICBpZiAoXHJcbiAgICAgICFlcnJvciB8fFxyXG4gICAgICB0eXBlb2YgZXJyb3IgIT09ICdvYmplY3QnIHx8XHJcbiAgICAgICEoJ21lc3NhZ2UnIGluIGVycm9yKSB8fFxyXG4gICAgICB0eXBlb2YgKGVycm9yIGFzIEVycm9yV2l0aE1lc3NhZ2UpLm1lc3NhZ2UgIT09ICdzdHJpbmcnXHJcbiAgICApXHJcbiAgICAgIHJldHVybiBudWxsO1xyXG5cclxuICAgIGNvbnN0IGxpbmtNYXRjaCA9IChlcnJvciBhcyBFcnJvcldpdGhNZXNzYWdlKS5tZXNzYWdlLm1hdGNoKFxyXG4gICAgICAvKGh0dHBzOlxcL1xcL2NvbnNvbGVcXC5maXJlYmFzZVxcLmdvb2dsZVxcLmNvbVxcL1teXFxzXSspL1xyXG4gICAgKTtcclxuICAgIHJldHVybiBsaW5rTWF0Y2ggPyBsaW5rTWF0Y2hbMV0gOiBudWxsO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogUmFzdHJlaWEgw61uZGljZXMgYXVzZW50ZXMgYW8gdXNhciBmYWxsYmFjayBwcmV2ZW50aXZvXHJcbiAgICovXHJcbiAgcHJpdmF0ZSBhc3luYyB0cmFja01pc3NpbmdJbmRleFByZXZlbnRpdmUoXHJcbiAgICBjb2xsZWN0aW9uOiBzdHJpbmcsXHJcbiAgICBhcnJhbmdlOiBBcnJhbmdlLFxyXG4gICAgY29uZGl0aW9uczogQ29uZGl0aW9uW10gfCB1bmRlZmluZWQgPSB1bmRlZmluZWRcclxuICApOiBQcm9taXNlPHZvaWQ+IHtcclxuICAgIHRyeSB7XHJcbiAgICAgIGNvbnN0IHF1ZXJ5U2lnbmF0dXJlID0gdGhpcy5nZW5lcmF0ZVF1ZXJ5U2lnbmF0dXJlKFxyXG4gICAgICAgIGNvbGxlY3Rpb24sXHJcbiAgICAgICAgYXJyYW5nZSxcclxuICAgICAgICBjb25kaXRpb25zXHJcbiAgICAgICk7XHJcbiAgICAgIGNvbnN0IGRvY0lkID0gYCR7Y29sbGVjdGlvbn1fJHtxdWVyeVNpZ25hdHVyZX1gO1xyXG5cclxuICAgICAgY29uc3QgaW5kZXhMaW5rID0gdGhpcy5nZW5lcmF0ZUluZGV4TGluayhjb2xsZWN0aW9uLCBhcnJhbmdlLCBjb25kaXRpb25zKTtcclxuXHJcbiAgICAgIGNvbnN0IGluZGV4SW5zdHJ1Y3Rpb25zID0gdGhpcy5nZW5lcmF0ZUluZGV4SW5zdHJ1Y3Rpb25zKFxyXG4gICAgICAgIGNvbGxlY3Rpb24sXHJcbiAgICAgICAgYXJyYW5nZSxcclxuICAgICAgICBjb25kaXRpb25zXHJcbiAgICAgICk7XHJcblxyXG4gICAgICBjb25zdCB0cmFja2luZ0RhdGE6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge1xyXG4gICAgICAgIGNvbGxlY3Rpb24sXHJcbiAgICAgICAgaW5kZXhMaW5rLFxyXG4gICAgICAgIGluZGV4SW5zdHJ1Y3Rpb25zLFxyXG4gICAgICAgIGFycmFuZ2U6IHtcclxuICAgICAgICAgIHNvcnRCeTogYXJyYW5nZS5zb3J0QnksXHJcbiAgICAgICAgICBmaWx0ZXJzOlxyXG4gICAgICAgICAgICBhcnJhbmdlLmZpbHRlcnM/Lm1hcCgoZikgPT4gKHtcclxuICAgICAgICAgICAgICBhcnJhbmdlOiBmLmFycmFuZ2UsXHJcbiAgICAgICAgICAgICAgcHJvcGVydHk6IGYuZmlsdGVyPy5wcm9wZXJ0eSB8fCBudWxsLFxyXG4gICAgICAgICAgICAgIGRhdGVGaWVsZDpcclxuICAgICAgICAgICAgICAgIGYuYXJyYW5nZSA9PT0gJ2ZpbHRlckJ5RGF0ZScgPyBhcnJhbmdlLnNvcnRCeT8uZmllbGQgOiBudWxsLFxyXG4gICAgICAgICAgICB9KSkgfHwgW10sXHJcbiAgICAgICAgfSxcclxuICAgICAgICBjb25kaXRpb25zOlxyXG4gICAgICAgICAgY29uZGl0aW9ucz8ubWFwKChjOiBDb25kaXRpb24pID0+ICh7XHJcbiAgICAgICAgICAgIHByb3BlcnR5OiBjLmZpcmVzdG9yZVByb3BlcnR5LFxyXG4gICAgICAgICAgICBvcGVyYXRvcjogYy5vcGVyYXRvcixcclxuICAgICAgICAgIH0pKSB8fCBbXSxcclxuICAgICAgICBlcnJvck1lc3NhZ2U6IGBGYWxsYmFjayBwcmV2ZW50aXZvIHVzYWRvIHBhcmEgYSBjb2xsZWN0aW9uICR7Y29sbGVjdGlvbn0uIEEgcXVlcnkgZXhpZ2lyaWEgw61uZGljZSBjb21wb3N0by5gLFxyXG4gICAgICAgIHVwZGF0ZWRBdDogZmlyZWJhc2UuZmlyZXN0b3JlLkZpZWxkVmFsdWUuc2VydmVyVGltZXN0YW1wKCksXHJcbiAgICAgIH07XHJcblxyXG4gICAgICBjb25zdCBkb2NSZWYgPSB0aGlzLm5nRmlyZS5jb2xsZWN0aW9uKCdtaXNzaW5nSW5kZXhlcycpLmRvYyhkb2NJZCk7XHJcbiAgICAgIGNvbnN0IGRvYyA9IGF3YWl0IGRvY1JlZi5nZXQoKS50b1Byb21pc2UoKTtcclxuXHJcbiAgICAgIGlmIChkb2MgJiYgZG9jLmV4aXN0cykge1xyXG4gICAgICAgIGF3YWl0IGRvY1JlZi51cGRhdGUoe1xyXG4gICAgICAgICAgY291bnQ6IGZpcmViYXNlLmZpcmVzdG9yZS5GaWVsZFZhbHVlLmluY3JlbWVudCgxKSxcclxuICAgICAgICAgIHVwZGF0ZWRBdDogZmlyZWJhc2UuZmlyZXN0b3JlLkZpZWxkVmFsdWUuc2VydmVyVGltZXN0YW1wKCksXHJcbiAgICAgICAgICBsYXN0RXJyb3I6IHRyYWNraW5nRGF0YVsnZXJyb3JNZXNzYWdlJ10sXHJcbiAgICAgICAgfSk7XHJcbiAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgYXdhaXQgZG9jUmVmLnNldCh7XHJcbiAgICAgICAgICAuLi50cmFja2luZ0RhdGEsXHJcbiAgICAgICAgICBjb3VudDogMSxcclxuICAgICAgICAgIGNyZWF0ZWRBdDogZmlyZWJhc2UuZmlyZXN0b3JlLkZpZWxkVmFsdWUuc2VydmVyVGltZXN0YW1wKCksXHJcbiAgICAgICAgfSk7XHJcbiAgICAgIH1cclxuICAgIH0gY2F0Y2ggKHRyYWNraW5nRXJyb3IpIHtcclxuICAgICAgY29uc29sZS53YXJuKCdGYWxoYSBhbyByYXN0cmVhciBmYWxsYmFjayBwcmV2ZW50aXZvOicsIHRyYWNraW5nRXJyb3IpO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogR2VyYSB1bWEgYXNzaW5hdHVyYSDDum5pY2EgcGFyYSB1bWEgcXVlcnlcclxuICAgKi9cclxuICBwcml2YXRlIGdlbmVyYXRlUXVlcnlTaWduYXR1cmUoXHJcbiAgICBjb2xsZWN0aW9uOiBzdHJpbmcsXHJcbiAgICBhcnJhbmdlOiBBcnJhbmdlLFxyXG4gICAgY29uZGl0aW9uczogQ29uZGl0aW9uW10gfCB1bmRlZmluZWQgPSB1bmRlZmluZWRcclxuICApOiBzdHJpbmcge1xyXG4gICAgY29uc3Qgc2lnbmF0dXJlID0ge1xyXG4gICAgICBjb2xsZWN0aW9uLFxyXG4gICAgICBzb3J0Qnk6IGFycmFuZ2Uuc29ydEJ5LFxyXG4gICAgICBmaWx0ZXJzOlxyXG4gICAgICAgIGFycmFuZ2UuZmlsdGVycz8ubWFwKChmOiBBcnJhbmdlWydmaWx0ZXJzJ11bMF0pID0+ICh7XHJcbiAgICAgICAgICBhcnJhbmdlOiBmLmFycmFuZ2UsXHJcbiAgICAgICAgICBwcm9wZXJ0eTogZi5maWx0ZXI/LnByb3BlcnR5IHx8IG51bGwsXHJcbiAgICAgICAgfSkpIHx8IFtdLFxyXG4gICAgICBjb25kaXRpb25zOlxyXG4gICAgICAgIGNvbmRpdGlvbnM/Lm1hcCgoYzogQ29uZGl0aW9uKSA9PiAoe1xyXG4gICAgICAgICAgcHJvcGVydHk6IGMuZmlyZXN0b3JlUHJvcGVydHksXHJcbiAgICAgICAgICBvcGVyYXRvcjogYy5vcGVyYXRvcixcclxuICAgICAgICB9KSkgfHwgW10sXHJcbiAgICB9O1xyXG5cclxuICAgIHJldHVybiBidG9hKEpTT04uc3RyaW5naWZ5KHNpZ25hdHVyZSkpXHJcbiAgICAgIC5yZXBsYWNlKC9bXmEtekEtWjAtOV0vZywgJycpXHJcbiAgICAgIC5zdWJzdHJpbmcoMCwgMjApO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogR2VyYSBpbnN0cnXDp8O1ZXMgY2xhcmFzIHBhcmEgY3JpYXIgbyDDrW5kaWNlIG1hbnVhbG1lbnRlXHJcbiAgICovXHJcbiAgcHJpdmF0ZSBnZW5lcmF0ZUluZGV4SW5zdHJ1Y3Rpb25zKFxyXG4gICAgY29sbGVjdGlvbjogc3RyaW5nLFxyXG4gICAgYXJyYW5nZTogQXJyYW5nZSxcclxuICAgIGNvbmRpdGlvbnM6IENvbmRpdGlvbltdIHwgdW5kZWZpbmVkID0gdW5kZWZpbmVkXHJcbiAgKToge1xyXG4gICAgc3VtbWFyeTogc3RyaW5nO1xyXG4gICAgY29sbGVjdGlvbjogc3RyaW5nO1xyXG4gICAgZmllbGRzOiBBcnJheTx7XHJcbiAgICAgIGZpZWxkOiBzdHJpbmc7XHJcbiAgICAgIG9yZGVyOiBzdHJpbmc7XHJcbiAgICAgIHR5cGU6IHN0cmluZztcclxuICAgICAgb3BlcmF0b3I6IHN0cmluZztcclxuICAgICAgZGVzY3JpcHRpb246IHN0cmluZztcclxuICAgIH0+O1xyXG4gICAgcXVlcnlFeGFtcGxlOiBzdHJpbmc7XHJcbiAgICBzdGVwQnlTdGVwOiBzdHJpbmdbXTtcclxuICAgIG5vdGVzOiBzdHJpbmdbXTtcclxuICB9IHtcclxuICAgIGNvbnN0IGluc3RydWN0aW9uczoge1xyXG4gICAgICBzdW1tYXJ5OiBzdHJpbmc7XHJcbiAgICAgIGNvbGxlY3Rpb246IHN0cmluZztcclxuICAgICAgZmllbGRzOiBBcnJheTx7XHJcbiAgICAgICAgZmllbGQ6IHN0cmluZztcclxuICAgICAgICBvcmRlcjogc3RyaW5nO1xyXG4gICAgICAgIHR5cGU6IHN0cmluZztcclxuICAgICAgICBvcGVyYXRvcjogc3RyaW5nO1xyXG4gICAgICAgIGRlc2NyaXB0aW9uOiBzdHJpbmc7XHJcbiAgICAgIH0+O1xyXG4gICAgICBxdWVyeUV4YW1wbGU6IHN0cmluZztcclxuICAgICAgc3RlcEJ5U3RlcDogc3RyaW5nW107XHJcbiAgICAgIG5vdGVzOiBzdHJpbmdbXTtcclxuICAgIH0gPSB7XHJcbiAgICAgIHN1bW1hcnk6ICcnLFxyXG4gICAgICBjb2xsZWN0aW9uOiBjb2xsZWN0aW9uLFxyXG4gICAgICBmaWVsZHM6IFtdLFxyXG4gICAgICBxdWVyeUV4YW1wbGU6ICcnLFxyXG4gICAgICBzdGVwQnlTdGVwOiBbXSxcclxuICAgICAgbm90ZXM6IFtdLFxyXG4gICAgfTtcclxuXHJcbiAgICBjb25zdCBmaWVsZHM6IEFycmF5PHtcclxuICAgICAgZmllbGQ6IHN0cmluZztcclxuICAgICAgb3JkZXI6IHN0cmluZztcclxuICAgICAgdHlwZTogc3RyaW5nO1xyXG4gICAgICBvcGVyYXRvcjogc3RyaW5nO1xyXG4gICAgICBkZXNjcmlwdGlvbjogc3RyaW5nO1xyXG4gICAgfT4gPSBbXTtcclxuXHJcbiAgICBpZiAoY29uZGl0aW9ucyAmJiBjb25kaXRpb25zLmxlbmd0aCA+IDApIHtcclxuICAgICAgY29uZGl0aW9ucy5mb3JFYWNoKChjb25kaXRpb246IENvbmRpdGlvbikgPT4ge1xyXG4gICAgICAgIGlmIChjb25kaXRpb24uZmlyZXN0b3JlUHJvcGVydHkpIHtcclxuICAgICAgICAgIGZpZWxkcy5wdXNoKHtcclxuICAgICAgICAgICAgZmllbGQ6IGNvbmRpdGlvbi5maXJlc3RvcmVQcm9wZXJ0eSxcclxuICAgICAgICAgICAgb3JkZXI6ICdBc2NlbmRpbmcnLFxyXG4gICAgICAgICAgICB0eXBlOiAnV0hFUkUgY2xhdXNlJyxcclxuICAgICAgICAgICAgb3BlcmF0b3I6IGNvbmRpdGlvbi5vcGVyYXRvcixcclxuICAgICAgICAgICAgZGVzY3JpcHRpb246IGBGaWx0cmFyIHBvciAke2NvbmRpdGlvbi5maXJlc3RvcmVQcm9wZXJ0eX0gdXNhbmRvIG9wZXJhZG9yICR7Y29uZGl0aW9uLm9wZXJhdG9yfWAsXHJcbiAgICAgICAgICB9KTtcclxuICAgICAgICB9XHJcbiAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIGlmIChhcnJhbmdlLmZpbHRlcnMgJiYgYXJyYW5nZS5maWx0ZXJzLmxlbmd0aCA+IDApIHtcclxuICAgICAgYXJyYW5nZS5maWx0ZXJzLmZvckVhY2goKGZpbHRlcikgPT4ge1xyXG4gICAgICAgIGlmIChmaWx0ZXIuZmlsdGVyPy5wcm9wZXJ0eSkge1xyXG4gICAgICAgICAgZmllbGRzLnB1c2goe1xyXG4gICAgICAgICAgICBmaWVsZDogZmlsdGVyLmZpbHRlci5wcm9wZXJ0eSxcclxuICAgICAgICAgICAgb3JkZXI6ICdBc2NlbmRpbmcnLFxyXG4gICAgICAgICAgICB0eXBlOiAnV0hFUkUgY2xhdXNlIChmaWx0ZXIpJyxcclxuICAgICAgICAgICAgb3BlcmF0b3I6IGZpbHRlci5hcnJhbmdlID09PSAnZmlsdGVyJyA/ICdDT05UQUlOUycgOiAnUkFOR0UnLFxyXG4gICAgICAgICAgICBkZXNjcmlwdGlvbjogYEZpbHRyYXIgcG9yICR7ZmlsdGVyLmZpbHRlci5wcm9wZXJ0eX0gdXNhbmRvIGZpbHRybyAke2ZpbHRlci5hcnJhbmdlfWAsXHJcbiAgICAgICAgICB9KTtcclxuICAgICAgICB9XHJcbiAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIGlmIChhcnJhbmdlLnNvcnRCeT8uZmllbGQpIHtcclxuICAgICAgZmllbGRzLnB1c2goe1xyXG4gICAgICAgIGZpZWxkOiBhcnJhbmdlLnNvcnRCeS5maWVsZCxcclxuICAgICAgICBvcmRlcjogYXJyYW5nZS5zb3J0Qnkub3JkZXIgPT09ICdkZXNjJyA/ICdEZXNjZW5kaW5nJyA6ICdBc2NlbmRpbmcnLFxyXG4gICAgICAgIHR5cGU6ICdPUkRFUiBCWSBjbGF1c2UnLFxyXG4gICAgICAgIG9wZXJhdG9yOiAnTi9BJyxcclxuICAgICAgICBkZXNjcmlwdGlvbjogYE9yZGVuYXIgcmVzdWx0YWRvcyBwb3IgJHthcnJhbmdlLnNvcnRCeS5maWVsZH0gZW0gb3JkZW0gJHthcnJhbmdlLnNvcnRCeS5vcmRlcn1gLFxyXG4gICAgICB9KTtcclxuICAgIH1cclxuXHJcbiAgICBpbnN0cnVjdGlvbnMuZmllbGRzID0gZmllbGRzO1xyXG5cclxuICAgIGNvbnN0IGZpZWxkTmFtZXMgPSBmaWVsZHMubWFwKChmKSA9PiBmLmZpZWxkKS5qb2luKCcgKyAnKTtcclxuICAgIGluc3RydWN0aW9ucy5zdW1tYXJ5ID0gYENyaWFyIMOtbmRpY2UgY29tcG9zdG8gcGFyYSAke2NvbGxlY3Rpb259OiAke2ZpZWxkTmFtZXN9YDtcclxuXHJcbiAgICBsZXQgcXVlcnlFeGFtcGxlID0gYGRiLmNvbGxlY3Rpb24oJyR7Y29sbGVjdGlvbn0nKWA7XHJcblxyXG4gICAgZmllbGRzLmZvckVhY2goKGZpZWxkLCBpbmRleDogbnVtYmVyKSA9PiB7XHJcbiAgICAgIGlmIChmaWVsZC50eXBlLmluY2x1ZGVzKCdXSEVSRScpKSB7XHJcbiAgICAgICAgaWYgKGZpZWxkLm9wZXJhdG9yID09PSAnPT0nKSB7XHJcbiAgICAgICAgICBxdWVyeUV4YW1wbGUgKz0gYFxcbiAgLndoZXJlKCcke2ZpZWxkLmZpZWxkfScsICc9PScsICd2YWx1ZScpYDtcclxuICAgICAgICB9IGVsc2UgaWYgKGZpZWxkLm9wZXJhdG9yID09PSAnQ09OVEFJTlMnKSB7XHJcbiAgICAgICAgICBxdWVyeUV4YW1wbGUgKz0gYFxcbiAgLndoZXJlKCcke2ZpZWxkLmZpZWxkfScsICc+PScsICdzZWFyY2hUZXh0JylgO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICBxdWVyeUV4YW1wbGUgKz0gYFxcbiAgLndoZXJlKCcke2ZpZWxkLmZpZWxkfScsICcke2ZpZWxkLm9wZXJhdG9yfScsICd2YWx1ZScpYDtcclxuICAgICAgICB9XHJcbiAgICAgIH1cclxuICAgIH0pO1xyXG5cclxuICAgIGNvbnN0IG9yZGVyQnlGaWVsZCA9IGZpZWxkcy5maW5kKChmKSA9PiBmLnR5cGUuaW5jbHVkZXMoJ09SREVSIEJZJykpO1xyXG4gICAgaWYgKG9yZGVyQnlGaWVsZCkge1xyXG4gICAgICBxdWVyeUV4YW1wbGUgKz0gYFxcbiAgLm9yZGVyQnkoJyR7XHJcbiAgICAgICAgb3JkZXJCeUZpZWxkLmZpZWxkXHJcbiAgICAgIH0nLCAnJHtvcmRlckJ5RmllbGQub3JkZXIudG9Mb3dlckNhc2UoKX0nKWA7XHJcbiAgICB9XHJcblxyXG4gICAgaW5zdHJ1Y3Rpb25zLnF1ZXJ5RXhhbXBsZSA9IHF1ZXJ5RXhhbXBsZTtcclxuXHJcbiAgICBpbnN0cnVjdGlvbnMuc3RlcEJ5U3RlcCA9IFtcclxuICAgICAgJzEuIElyIHBhcmEgRmlyZWJhc2UgQ29uc29sZSDihpIgRmlyZXN0b3JlIOKGkiBJbmRleGVzJyxcclxuICAgICAgJzIuIENsaWNhciBlbSBcIkNyZWF0ZSBJbmRleFwiJyxcclxuICAgICAgYDMuIERlZmluaXIgQ29sbGVjdGlvbiBJRDogJHtjb2xsZWN0aW9ufWAsXHJcbiAgICAgICc0LiBDb25maWd1cmFyIGNhbXBvcyBuZXN0YSBPUkRFTSBFWEFUQTonLFxyXG4gICAgICAuLi5maWVsZHMubWFwKFxyXG4gICAgICAgIChmaWVsZCwgaW5kZXg6IG51bWJlcikgPT5cclxuICAgICAgICAgIGAgICAke2luZGV4ICsgMX0uIENhbXBvOiAke2ZpZWxkLmZpZWxkfSwgT3JkZXI6ICR7XHJcbiAgICAgICAgICAgIGZpZWxkLm9yZGVyXHJcbiAgICAgICAgICB9LCBBcnJheTogTm9gXHJcbiAgICAgICksXHJcbiAgICAgICc1LiBEZWZpbmlyIFF1ZXJ5IHNjb3BlczogQ29sbGVjdGlvbicsXHJcbiAgICAgICc2LiBDbGljYXIgZW0gXCJDcmVhdGVcIiBlIGFndWFyZGFyIGNvbmNsdXPDo28nLFxyXG4gICAgXTtcclxuXHJcbiAgICBpbnN0cnVjdGlvbnMubm90ZXMgPSBbXHJcbiAgICAgICfimqDvuI8gQSBvcmRlbSBkb3MgY2FtcG9zIMOpIENSw41USUNBIC0gZGV2ZSBjb3JyZXNwb25kZXIgZXhhdGFtZW50ZSDDoCBvcmRlbSBkYSBxdWVyeScsXHJcbiAgICAgICfimqDvuI8gQXMgY2zDoXVzdWxhcyBXSEVSRSBkZXZlbSB2aXIgQU5URVMgZG8gY2FtcG8gT1JERVIgQlknLFxyXG4gICAgICAn4pqg77iPIEVzdGUgw61uZGljZSBzw7MgZnVuY2lvbmFyw6EgcGFyYSBxdWVyaWVzIGNvbSBlc3RhIGNvbWJpbmHDp8OjbyBFWEFUQSBkZSBjYW1wb3MnLFxyXG4gICAgICAn4pqg77iPIEEgY3JpYcOnw6NvIGRvIMOtbmRpY2UgcG9kZSBsZXZhciB2w6FyaW9zIG1pbnV0b3MnLFxyXG4gICAgXTtcclxuXHJcbiAgICByZXR1cm4gaW5zdHJ1Y3Rpb25zO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogR2VyYSB1bSBsaW5rIGRlIMOtbmRpY2UgYmFzZWFkbyBuYSBlc3RydXR1cmEgZGEgcXVlcnlcclxuICAgKi9cclxuICBwcml2YXRlIGdlbmVyYXRlSW5kZXhMaW5rKFxyXG4gICAgY29sbGVjdGlvbjogc3RyaW5nLFxyXG4gICAgYXJyYW5nZTogQXJyYW5nZSxcclxuICAgIGNvbmRpdGlvbnM6IENvbmRpdGlvbltdIHwgdW5kZWZpbmVkID0gdW5kZWZpbmVkXHJcbiAgKTogc3RyaW5nIHwgbnVsbCB7XHJcbiAgICB0cnkge1xyXG4gICAgICBjb25zdCBpbmRleEZpZWxkczogc3RyaW5nW10gPSBbXTtcclxuXHJcbiAgICAgIGlmIChjb25kaXRpb25zICYmIGNvbmRpdGlvbnMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgIGNvbmRpdGlvbnMuZm9yRWFjaCgoY29uZGl0aW9uOiBDb25kaXRpb24pID0+IHtcclxuICAgICAgICAgIGlmIChjb25kaXRpb24uZmlyZXN0b3JlUHJvcGVydHkpIHtcclxuICAgICAgICAgICAgaW5kZXhGaWVsZHMucHVzaChjb25kaXRpb24uZmlyZXN0b3JlUHJvcGVydHkpO1xyXG4gICAgICAgICAgfVxyXG4gICAgICAgIH0pO1xyXG4gICAgICB9XHJcblxyXG4gICAgICBpZiAoYXJyYW5nZS5maWx0ZXJzICYmIGFycmFuZ2UuZmlsdGVycy5sZW5ndGggPiAwKSB7XHJcbiAgICAgICAgYXJyYW5nZS5maWx0ZXJzLmZvckVhY2goKGZpbHRlcjogQXJyYW5nZVsnZmlsdGVycyddWzBdKSA9PiB7XHJcbiAgICAgICAgICBpZiAoZmlsdGVyLmZpbHRlcj8ucHJvcGVydHkpIHtcclxuICAgICAgICAgICAgaW5kZXhGaWVsZHMucHVzaChmaWx0ZXIuZmlsdGVyLnByb3BlcnR5KTtcclxuICAgICAgICAgIH1cclxuICAgICAgICB9KTtcclxuICAgICAgfVxyXG5cclxuICAgICAgaWYgKGFycmFuZ2Uuc29ydEJ5Py5maWVsZCkge1xyXG4gICAgICAgIGluZGV4RmllbGRzLnB1c2goYXJyYW5nZS5zb3J0QnkuZmllbGQpO1xyXG4gICAgICB9XHJcblxyXG4gICAgICBpZiAoaW5kZXhGaWVsZHMubGVuZ3RoID4gMSkge1xyXG4gICAgICAgIC8vIE9idGVyIG8gcHJvamVjdElkIGRpbmFtaWNhbWVudGUgZG8gRmlyZWJhc2VcclxuICAgICAgICBjb25zdCBvcHRpb25zID0gdGhpcy5uZ0ZpcmU/LmZpcmVzdG9yZT8uYXBwPy5vcHRpb25zIGFzXHJcbiAgICAgICAgICB8IFJlY29yZDxzdHJpbmcsIHVua25vd24+XHJcbiAgICAgICAgICB8IHVuZGVmaW5lZDtcclxuICAgICAgICBjb25zdCBwcm9qZWN0SWQgPSBvcHRpb25zPy5bJ3Byb2plY3RJZCddIGFzIHN0cmluZyB8IHVuZGVmaW5lZDtcclxuXHJcbiAgICAgICAgaWYgKCFwcm9qZWN0SWQpIHtcclxuICAgICAgICAgIGNvbnNvbGUud2FybignTsOjbyBmb2kgcG9zc8OtdmVsIG9idGVyIG8gcHJvamVjdElkIGRvIEZpcmViYXNlJyk7XHJcbiAgICAgICAgICByZXR1cm4gbnVsbDtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGNvbnN0IGJhc2VVcmwgPSBgaHR0cHM6Ly9jb25zb2xlLmZpcmViYXNlLmdvb2dsZS5jb20vcHJvamVjdC8ke3Byb2plY3RJZH0vZmlyZXN0b3JlL2luZGV4ZXNgO1xyXG4gICAgICAgIGNvbnN0IHF1ZXJ5UGFyYW1zID0gbmV3IFVSTFNlYXJjaFBhcmFtcyh7XHJcbiAgICAgICAgICBjcmVhdGVfY29tcG9zaXRlOiBgY29sbGVjdGlvbj0ke2NvbGxlY3Rpb259JmZpZWxkcz0ke2luZGV4RmllbGRzLmpvaW4oXHJcbiAgICAgICAgICAgICcsJ1xyXG4gICAgICAgICAgKX1gLFxyXG4gICAgICAgIH0pO1xyXG5cclxuICAgICAgICBjb25zdCBmaW5hbExpbmsgPSBgJHtiYXNlVXJsfT8ke3F1ZXJ5UGFyYW1zLnRvU3RyaW5nKCl9YDtcclxuICAgICAgICByZXR1cm4gZmluYWxMaW5rO1xyXG4gICAgICB9XHJcblxyXG4gICAgICByZXR1cm4gbnVsbDtcclxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XHJcbiAgICAgIGNvbnNvbGUud2FybignRmFsaGEgYW8gZ2VyYXIgbGluayBkZSDDrW5kaWNlOicsIGVycm9yKTtcclxuICAgICAgcmV0dXJuIG51bGw7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGFzeW5jIHRyYWNrTWlzc2luZ0luZGV4KFxyXG4gICAgZXJyb3I6IHVua25vd24sXHJcbiAgICBjb2xsZWN0aW9uOiBzdHJpbmcsXHJcbiAgICBhcnJhbmdlOiBBcnJhbmdlLFxyXG4gICAgY29uZGl0aW9uczogQ29uZGl0aW9uW10gfCB1bmRlZmluZWQgPSB1bmRlZmluZWRcclxuICApOiBQcm9taXNlPHZvaWQ+IHtcclxuICAgIHRyeSB7XHJcbiAgICAgIGNvbnN0IGluZGV4TGluayA9IHRoaXMuZXh0cmFjdEluZGV4TGluayhlcnJvcik7XHJcbiAgICAgIGlmICghaW5kZXhMaW5rKSByZXR1cm47XHJcblxyXG4gICAgICBjb25zdCBsaW5rSGFzaCA9IGJ0b2EoaW5kZXhMaW5rKVxyXG4gICAgICAgIC5yZXBsYWNlKC9bXmEtekEtWjAtOV0vZywgJycpXHJcbiAgICAgICAgLnN1YnN0cmluZygwLCAyMCk7XHJcbiAgICAgIGNvbnN0IGRvY0lkID0gYCR7Y29sbGVjdGlvbn1fJHtsaW5rSGFzaH1gO1xyXG5cclxuICAgICAgY29uc3QgaW5kZXhJbnN0cnVjdGlvbnMgPSB0aGlzLmdlbmVyYXRlSW5kZXhJbnN0cnVjdGlvbnMoXHJcbiAgICAgICAgY29sbGVjdGlvbixcclxuICAgICAgICBhcnJhbmdlLFxyXG4gICAgICAgIGNvbmRpdGlvbnNcclxuICAgICAgKTtcclxuXHJcbiAgICAgIGNvbnN0IHRyYWNraW5nRGF0YTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSB7XHJcbiAgICAgICAgY29sbGVjdGlvbixcclxuICAgICAgICBpbmRleExpbmssXHJcbiAgICAgICAgaW5kZXhJbnN0cnVjdGlvbnMsXHJcbiAgICAgICAgYXJyYW5nZToge1xyXG4gICAgICAgICAgc29ydEJ5OiBhcnJhbmdlLnNvcnRCeSxcclxuICAgICAgICAgIGZpbHRlcnM6XHJcbiAgICAgICAgICAgIGFycmFuZ2UuZmlsdGVycz8ubWFwKChmKSA9PiAoe1xyXG4gICAgICAgICAgICAgIGFycmFuZ2U6IGYuYXJyYW5nZSxcclxuICAgICAgICAgICAgICBwcm9wZXJ0eTogZi5maWx0ZXI/LnByb3BlcnR5IHx8IG51bGwsXHJcbiAgICAgICAgICAgICAgZGF0ZUZpZWxkOlxyXG4gICAgICAgICAgICAgICAgZi5hcnJhbmdlID09PSAnZmlsdGVyQnlEYXRlJyA/IGFycmFuZ2Uuc29ydEJ5Py5maWVsZCA6IG51bGwsXHJcbiAgICAgICAgICAgIH0pKSB8fCBbXSxcclxuICAgICAgICB9LFxyXG4gICAgICAgIGNvbmRpdGlvbnM6XHJcbiAgICAgICAgICBjb25kaXRpb25zPy5tYXAoKGM6IENvbmRpdGlvbikgPT4gKHtcclxuICAgICAgICAgICAgcHJvcGVydHk6IGMuZmlyZXN0b3JlUHJvcGVydHksXHJcbiAgICAgICAgICAgIG9wZXJhdG9yOiBjLm9wZXJhdG9yLFxyXG4gICAgICAgICAgfSkpIHx8IFtdLFxyXG4gICAgICAgIGVycm9yTWVzc2FnZTpcclxuICAgICAgICAgIGVycm9yICYmXHJcbiAgICAgICAgICB0eXBlb2YgZXJyb3IgPT09ICdvYmplY3QnICYmXHJcbiAgICAgICAgICAnbWVzc2FnZScgaW4gZXJyb3IgJiZcclxuICAgICAgICAgIHR5cGVvZiAoZXJyb3IgYXMgRXJyb3JXaXRoTWVzc2FnZSkubWVzc2FnZSA9PT0gJ3N0cmluZydcclxuICAgICAgICAgICAgPyAoZXJyb3IgYXMgRXJyb3JXaXRoTWVzc2FnZSkubWVzc2FnZVxyXG4gICAgICAgICAgICA6IFN0cmluZyhlcnJvciksXHJcbiAgICAgICAgdXBkYXRlZEF0OiBmaXJlYmFzZS5maXJlc3RvcmUuRmllbGRWYWx1ZS5zZXJ2ZXJUaW1lc3RhbXAoKSxcclxuICAgICAgfTtcclxuXHJcbiAgICAgIGNvbnN0IGRvY1JlZiA9IHRoaXMubmdGaXJlLmNvbGxlY3Rpb24oJ21pc3NpbmdJbmRleGVzJykuZG9jKGRvY0lkKTtcclxuICAgICAgY29uc3QgZG9jID0gYXdhaXQgZG9jUmVmLmdldCgpLnRvUHJvbWlzZSgpO1xyXG5cclxuICAgICAgY29uc3QgZXJyb3JNZXNzYWdlID1cclxuICAgICAgICBlcnJvciAmJlxyXG4gICAgICAgIHR5cGVvZiBlcnJvciA9PT0gJ29iamVjdCcgJiZcclxuICAgICAgICAnbWVzc2FnZScgaW4gZXJyb3IgJiZcclxuICAgICAgICB0eXBlb2YgKGVycm9yIGFzIEVycm9yV2l0aE1lc3NhZ2UpLm1lc3NhZ2UgPT09ICdzdHJpbmcnXHJcbiAgICAgICAgICA/IChlcnJvciBhcyBFcnJvcldpdGhNZXNzYWdlKS5tZXNzYWdlXHJcbiAgICAgICAgICA6IFN0cmluZyhlcnJvcik7XHJcblxyXG4gICAgICBpZiAoZG9jICYmIGRvYy5leGlzdHMpIHtcclxuICAgICAgICBhd2FpdCBkb2NSZWYudXBkYXRlKHtcclxuICAgICAgICAgIGNvdW50OiBmaXJlYmFzZS5maXJlc3RvcmUuRmllbGRWYWx1ZS5pbmNyZW1lbnQoMSksXHJcbiAgICAgICAgICB1cGRhdGVkQXQ6IGZpcmViYXNlLmZpcmVzdG9yZS5GaWVsZFZhbHVlLnNlcnZlclRpbWVzdGFtcCgpLFxyXG4gICAgICAgICAgbGFzdEVycm9yOiBlcnJvck1lc3NhZ2UsXHJcbiAgICAgICAgfSk7XHJcbiAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgYXdhaXQgZG9jUmVmLnNldCh7XHJcbiAgICAgICAgICAuLi50cmFja2luZ0RhdGEsXHJcbiAgICAgICAgICBjb3VudDogMSxcclxuICAgICAgICAgIGNyZWF0ZWRBdDogZmlyZWJhc2UuZmlyZXN0b3JlLkZpZWxkVmFsdWUuc2VydmVyVGltZXN0YW1wKCksXHJcbiAgICAgICAgfSk7XHJcbiAgICAgIH1cclxuICAgIH0gY2F0Y2ggKHRyYWNraW5nRXJyb3IpIHtcclxuICAgICAgY29uc29sZS53YXJuKCdGYWxoYSBhbyByYXN0cmVhciDDrW5kaWNlIGF1c2VudGU6JywgdHJhY2tpbmdFcnJvcik7XHJcbiAgICB9XHJcbiAgfVxyXG59XHJcbiJdfQ==
|