ng-firebase-table-kxp 1.0.6 → 1.0.8

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