@gnwebsoft/ui 4.0.14 → 4.0.16

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (46) hide show
  1. package/dist/{chunk-QATCBGCZ.cjs → chunk-3CHF3PN3.cjs} +10 -4
  2. package/dist/chunk-6PMJWQ4R.cjs +1 -1
  3. package/dist/{chunk-KSYRU4D5.cjs → chunk-F3RJ3NFK.cjs} +421 -290
  4. package/dist/chunk-LUW7V5GI.cjs +1 -1
  5. package/dist/chunk-ML5UQCRH.js +2535 -0
  6. package/dist/chunk-MVPLBJRK.cjs +1 -1
  7. package/dist/chunk-XTPFOURJ.cjs +1 -1
  8. package/dist/chunk-Y3QTSDLJ.cjs +1 -1
  9. package/dist/chunk-YTZULD34.js +2384 -0
  10. package/dist/core/components/LabelText/LabelText.d.ts.map +1 -1
  11. package/dist/core/index.cjs +3 -3
  12. package/dist/core/index.d.ts +0 -1
  13. package/dist/core/index.d.ts.map +1 -1
  14. package/dist/core/index.js +1 -1
  15. package/dist/hooks/index.cjs +1 -1
  16. package/dist/index.cjs +4 -4
  17. package/dist/index.js +4 -4
  18. package/dist/types/AsyncSelectPayload.d.ts +4 -4
  19. package/dist/types/AsyncSelectPayload.d.ts.map +1 -1
  20. package/dist/types/OptionItem.d.ts +0 -9
  21. package/dist/types/OptionItem.d.ts.map +1 -1
  22. package/dist/types/index.cjs +1 -1
  23. package/dist/utils/index.cjs +1 -1
  24. package/dist/wrappers/AsyncMultiSelect/AsyncMultiSelect.d.ts.map +1 -1
  25. package/dist/wrappers/AsyncMultiSelect/types.d.ts +5 -5
  26. package/dist/wrappers/AsyncMultiSelect/types.d.ts.map +1 -1
  27. package/dist/wrappers/AsyncSelect/index.d.ts.map +1 -1
  28. package/dist/wrappers/AsyncSelect/types.d.ts +2 -1
  29. package/dist/wrappers/AsyncSelect/types.d.ts.map +1 -1
  30. package/dist/wrappers/CheckboxGroup/CheckboxGroup.d.ts +2 -6
  31. package/dist/wrappers/CheckboxGroup/CheckboxGroup.d.ts.map +1 -1
  32. package/dist/wrappers/CheckboxGroup/index.d.ts +1 -1
  33. package/dist/wrappers/CheckboxGroup/index.d.ts.map +1 -1
  34. package/dist/wrappers/Field/index.d.ts +1 -1
  35. package/dist/wrappers/Field/index.d.ts.map +1 -1
  36. package/dist/wrappers/SelectCascadeElement/SelectCascadeElement.d.ts +1 -1
  37. package/dist/wrappers/SelectCascadeElement/SelectCascadeElement.d.ts.map +1 -1
  38. package/dist/wrappers/SelectElement/SelectElement.d.ts +5 -64
  39. package/dist/wrappers/SelectElement/SelectElement.d.ts.map +1 -1
  40. package/dist/wrappers/SelectMultiElement/SelectMultiElement.d.ts +2 -1
  41. package/dist/wrappers/SelectMultiElement/SelectMultiElement.d.ts.map +1 -1
  42. package/dist/wrappers/index.cjs +3 -3
  43. package/dist/wrappers/index.js +3 -3
  44. package/package.json +1 -1
  45. package/dist/chunk-6SIBDHHA.js +0 -2529
  46. package/dist/chunk-WNDGEBEU.js +0 -2253
@@ -0,0 +1,2535 @@
1
+ // src/core/api/CorrelationIdGenerator.ts
2
+ function generateUUID() {
3
+ if (typeof crypto !== "undefined" && crypto.randomUUID) {
4
+ return crypto.randomUUID();
5
+ }
6
+ return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, (c) => {
7
+ const r = Math.random() * 16 | 0;
8
+ const v = c === "x" ? r : r & 3 | 8;
9
+ return v.toString(16);
10
+ });
11
+ }
12
+ function generateCorrelationId(prefix) {
13
+ const uuid = generateUUID();
14
+ return prefix ? `${prefix}-${uuid}` : uuid;
15
+ }
16
+
17
+ // src/core/api/Errors/ErrorNormalizer.ts
18
+ var ErrorNormalizer = class {
19
+ /**
20
+ * Maps an HTTP status code to a standardized error type category.
21
+ *
22
+ * This categorization helps consumers handle different error classes appropriately:
23
+ * - `validation_error` (400): Client sent invalid data
24
+ * - `client_error` (401-499): Client-side issues (auth, permissions, not found, etc.)
25
+ * - `server_error` (500-599): Server-side failures
26
+ * - `unknown_error`: Unrecognized status codes
27
+ *
28
+ * @param status - HTTP status code from the response
29
+ * @returns The error type category as a string
30
+ *
31
+ * @example
32
+ * ```typescript
33
+ * normalizer.getErrorType(400); // => 'validation_error'
34
+ * normalizer.getErrorType(404); // => 'client_error'
35
+ * normalizer.getErrorType(500); // => 'server_error'
36
+ * normalizer.getErrorType(0); // => 'unknown_error'
37
+ * ```
38
+ */
39
+ getErrorType(status) {
40
+ if (status >= 400 && status < 500) {
41
+ return status === 400 ? "validation_error" : "client_error";
42
+ } else if (status >= 500) {
43
+ return "server_error";
44
+ }
45
+ return "unknown_error";
46
+ }
47
+ /**
48
+ * Maps an HTTP status code to a human-readable error title.
49
+ *
50
+ * Provides user-friendly error messages for common HTTP status codes.
51
+ * Falls back to a generic "HTTP Error {status}" format for unmapped codes.
52
+ *
53
+ * @param status - HTTP status code from the response
54
+ * @returns A human-readable error title
55
+ *
56
+ * @example
57
+ * ```typescript
58
+ * normalizer.getErrorTitle(404); // => 'Not Found'
59
+ * normalizer.getErrorTitle(500); // => 'Internal Server Error'
60
+ * normalizer.getErrorTitle(999); // => 'HTTP Error 999'
61
+ * ```
62
+ */
63
+ getErrorTitle(status) {
64
+ const titles = {
65
+ 400: "Bad Request",
66
+ 401: "Unauthorized",
67
+ 403: "Forbidden",
68
+ 404: "Not Found",
69
+ 405: "Method Not Allowed",
70
+ 408: "Request Timeout",
71
+ 409: "Conflict",
72
+ 422: "Unprocessable Entity",
73
+ 429: "Too Many Requests",
74
+ 500: "Internal Server Error",
75
+ 502: "Bad Gateway",
76
+ 503: "Service Unavailable",
77
+ 504: "Gateway Timeout"
78
+ };
79
+ return titles[status] || `HTTP Error ${status}`;
80
+ }
81
+ /**
82
+ * Normalizes any error into a consistent, structured ApiError format.
83
+ *
84
+ * This method handles various error scenarios and ensures they all conform to
85
+ * the {@link ApiError} interface with appropriate categorization and metadata:
86
+ *
87
+ * - **Existing ApiErrors**: Enhances with missing fields (traceId, config)
88
+ * - **AbortErrors**: Marks as `request_cancelled` with isAborted flag
89
+ * - **Timeout Errors**: Categorizes as `timeout_error` with 408 status
90
+ * - **Network Errors**: Categorizes as `network_error` with 0 status
91
+ * - **Unknown Errors**: Fallback category for unexpected error types
92
+ *
93
+ * All normalized errors include:
94
+ * - `type`: Error category for programmatic handling
95
+ * - `title`: Human-readable error title
96
+ * - `status`: HTTP status code (or 0 for non-HTTP errors)
97
+ * - `traceId`: Correlation ID for distributed tracing
98
+ * - `isAborted`: Boolean flag indicating if request was cancelled
99
+ * - `config`: Original request configuration for debugging
100
+ *
101
+ * @param error - The error to normalize (can be any type)
102
+ * @param config - The request configuration that led to this error
103
+ * @param correlationId - Optional correlation ID for tracing
104
+ * @returns A fully structured ApiError instance
105
+ *
106
+ * @example
107
+ * Normalizing a fetch AbortError:
108
+ * ```typescript
109
+ * try {
110
+ * await fetch(url, { signal });
111
+ * } catch (error) {
112
+ * const apiError = normalizer.normalizeError(error, config, 'req-123');
113
+ * // apiError.type === 'request_cancelled'
114
+ * // apiError.isAborted === true
115
+ * }
116
+ * ```
117
+ *
118
+ * @example
119
+ * Normalizing a timeout:
120
+ * ```typescript
121
+ * const timeoutError = new Error('Request timeout after 30000ms');
122
+ * const apiError = normalizer.normalizeError(timeoutError, config);
123
+ * // apiError.type === 'timeout_error'
124
+ * // apiError.status === 408
125
+ * ```
126
+ */
127
+ normalizeError(error, config, correlationId) {
128
+ if (error === null || error === void 0) {
129
+ return Object.assign(new Error("An unknown error occurred"), {
130
+ type: "unknown_error",
131
+ title: "Unknown Error",
132
+ status: 0,
133
+ traceId: correlationId,
134
+ isAborted: false,
135
+ config
136
+ });
137
+ }
138
+ if (typeof error === "string") {
139
+ return Object.assign(new Error(error), {
140
+ type: "unknown_error",
141
+ title: "Unknown Error",
142
+ status: 0,
143
+ traceId: correlationId,
144
+ isAborted: false,
145
+ config
146
+ });
147
+ }
148
+ const err = error;
149
+ if (err.type || err.title || err.errors) {
150
+ return Object.assign(
151
+ error instanceof Error ? error : new Error(err.message ?? "Unknown error"),
152
+ {
153
+ type: err.type,
154
+ title: err.title,
155
+ status: err.status,
156
+ traceId: err.traceId || correlationId,
157
+ errors: err.errors,
158
+ isAborted: err.isAborted || false,
159
+ config
160
+ }
161
+ );
162
+ }
163
+ if (err.name === "AbortError" || err.isAborted) {
164
+ return Object.assign(new Error(err.message ?? "Request was aborted"), {
165
+ type: "request_cancelled",
166
+ title: "Request was cancelled",
167
+ status: 0,
168
+ traceId: correlationId,
169
+ isAborted: true,
170
+ config
171
+ });
172
+ }
173
+ if (err.message?.includes("timeout")) {
174
+ return Object.assign(new Error(err.message), {
175
+ type: "timeout_error",
176
+ title: "Request Timeout",
177
+ status: 408,
178
+ traceId: correlationId,
179
+ isAborted: true,
180
+ config
181
+ });
182
+ }
183
+ if (err.message?.includes("network")) {
184
+ return Object.assign(new Error(err.message ?? "Network request failed"), {
185
+ type: "network_error",
186
+ title: "Network Error",
187
+ status: 0,
188
+ traceId: correlationId,
189
+ isAborted: false,
190
+ config
191
+ });
192
+ }
193
+ return Object.assign(
194
+ new Error(err.message ?? "An unknown error occurred"),
195
+ {
196
+ type: "unknown_error",
197
+ title: "Unknown Error",
198
+ status: 0,
199
+ traceId: correlationId,
200
+ isAborted: false,
201
+ config
202
+ }
203
+ );
204
+ }
205
+ };
206
+
207
+ // src/core/api/Interceptors/InterceptorManager.ts
208
+ var InterceptorManager = class {
209
+ /**
210
+ * Array of registered request interceptors
211
+ * @private
212
+ */
213
+ requestInterceptors = [];
214
+ /**
215
+ * Array of registered response interceptors
216
+ * @private
217
+ */
218
+ responseInterceptors = [];
219
+ /**
220
+ * Array of registered error interceptors
221
+ * @private
222
+ */
223
+ errorInterceptors = [];
224
+ /**
225
+ * Registers a request interceptor to modify requests before they are sent.
226
+ *
227
+ * Request interceptors can:
228
+ * - Add or modify headers
229
+ * - Transform request bodies
230
+ * - Add query parameters
231
+ * - Implement request signing
232
+ * - Log outgoing requests
233
+ *
234
+ * @param interceptor - Async function that receives and returns RequestConfig
235
+ * @returns Cleanup function to unregister this interceptor
236
+ *
237
+ * @example
238
+ * ```typescript
239
+ * // Add authentication header
240
+ * const unregister = manager.addRequestInterceptor(async (config) => {
241
+ * const token = await getAuthToken();
242
+ * config.headers = config.headers || new Headers();
243
+ * config.headers.set('Authorization', `Bearer ${token}`);
244
+ * return config;
245
+ * });
246
+ *
247
+ * // Later, remove the interceptor
248
+ * unregister();
249
+ * ```
250
+ */
251
+ addRequestInterceptor(interceptor) {
252
+ this.requestInterceptors.push(interceptor);
253
+ return () => {
254
+ const index = this.requestInterceptors.indexOf(interceptor);
255
+ if (index > -1) this.requestInterceptors.splice(index, 1);
256
+ };
257
+ }
258
+ /**
259
+ * Registers a response interceptor to transform responses before they are returned.
260
+ *
261
+ * Response interceptors can:
262
+ * - Transform response data format
263
+ * - Extract nested data structures
264
+ * - Add computed properties
265
+ * - Cache responses
266
+ * - Log successful responses
267
+ *
268
+ * @param interceptor - Async function that receives and returns ApiResponse
269
+ * @returns Cleanup function to unregister this interceptor
270
+ *
271
+ * @example
272
+ * ```typescript
273
+ * // Extract data from envelope
274
+ * manager.addResponseInterceptor(async (response) => {
275
+ * if (response.apiData?.result) {
276
+ * response.apiData = response.apiData.result;
277
+ * }
278
+ * return response;
279
+ * });
280
+ *
281
+ * // Add timestamps
282
+ * manager.addResponseInterceptor(async (response) => {
283
+ * return {
284
+ * ...response,
285
+ * receivedAt: new Date().toISOString()
286
+ * };
287
+ * });
288
+ * ```
289
+ */
290
+ addResponseInterceptor(interceptor) {
291
+ this.responseInterceptors.push(interceptor);
292
+ return () => {
293
+ const index = this.responseInterceptors.indexOf(interceptor);
294
+ if (index > -1) this.responseInterceptors.splice(index, 1);
295
+ };
296
+ }
297
+ /**
298
+ * Registers an error interceptor to handle or transform errors before they are thrown.
299
+ *
300
+ * Error interceptors can:
301
+ * - Log errors to monitoring services
302
+ * - Transform error formats
303
+ * - Implement retry logic
304
+ * - Show user notifications
305
+ * - Extract validation errors
306
+ *
307
+ * **Note:** Error interceptors should re-throw the error (or a transformed version)
308
+ * to maintain the error flow. The final error is always thrown.
309
+ *
310
+ * @param interceptor - Async function that receives and returns (or throws) ApiError
311
+ * @returns Cleanup function to unregister this interceptor
312
+ *
313
+ * @example
314
+ * ```typescript
315
+ * // Log to monitoring service
316
+ * manager.addErrorInterceptor(async (error) => {
317
+ * if (error.status >= 500) {
318
+ * await Sentry.captureException(error, {
319
+ * extra: { traceId: error.traceId }
320
+ * });
321
+ * }
322
+ * throw error; // Re-throw to continue error flow
323
+ * });
324
+ *
325
+ * // Transform error messages
326
+ * manager.addErrorInterceptor(async (error) => {
327
+ * if (error.status === 404) {
328
+ * error.title = 'Resource not found';
329
+ * }
330
+ * throw error;
331
+ * });
332
+ * ```
333
+ */
334
+ addErrorInterceptor(interceptor) {
335
+ this.errorInterceptors.push(interceptor);
336
+ return () => {
337
+ const index = this.errorInterceptors.indexOf(interceptor);
338
+ if (index > -1) this.errorInterceptors.splice(index, 1);
339
+ };
340
+ }
341
+ /**
342
+ * Applies all registered request interceptors in sequential order.
343
+ *
344
+ * Each interceptor receives the config modified by the previous interceptor,
345
+ * forming a processing pipeline. If any interceptor throws an error,
346
+ * the pipeline stops and the error propagates.
347
+ *
348
+ * @param config - The initial request configuration
349
+ * @returns The modified request configuration after all interceptors
350
+ *
351
+ * @example
352
+ * ```typescript
353
+ * const config = { method: 'GET', url: '/users' };
354
+ * const finalConfig = await manager.applyRequestInterceptors(config);
355
+ * // finalConfig has been processed by all registered interceptors
356
+ * ```
357
+ */
358
+ async applyRequestInterceptors(config) {
359
+ let modifiedConfig = { ...config };
360
+ for (const interceptor of this.requestInterceptors) {
361
+ modifiedConfig = await interceptor(modifiedConfig);
362
+ }
363
+ return modifiedConfig;
364
+ }
365
+ /**
366
+ * Applies all registered response interceptors in sequential order.
367
+ *
368
+ * Each interceptor receives the response modified by the previous interceptor,
369
+ * forming a processing pipeline. If any interceptor throws an error,
370
+ * the pipeline stops and the error propagates.
371
+ *
372
+ * @template T - The type of the response data
373
+ * @param response - The initial API response
374
+ * @returns The modified response after all interceptors
375
+ *
376
+ * @example
377
+ * ```typescript
378
+ * const response = { data: { id: 1, name: 'John' } };
379
+ * const finalResponse = await manager.applyResponseInterceptors(response);
380
+ * // finalResponse has been processed by all registered interceptors
381
+ * ```
382
+ */
383
+ async applyResponseInterceptors(response) {
384
+ let modifiedResponse = response;
385
+ for (const interceptor of this.responseInterceptors) {
386
+ modifiedResponse = await interceptor(modifiedResponse);
387
+ }
388
+ return modifiedResponse;
389
+ }
390
+ /**
391
+ * Applies all registered error interceptors in sequential order and re-throws.
392
+ *
393
+ * Each interceptor receives the error (potentially modified by previous interceptors).
394
+ * Interceptors can transform the error before re-throwing it. The final error
395
+ * is always thrown to maintain error flow.
396
+ *
397
+ * If an interceptor itself throws an error, that becomes the new error to process
398
+ * by subsequent interceptors.
399
+ *
400
+ * @param error - The initial API error
401
+ * @returns Never returns (always throws)
402
+ * @throws The final error after all interceptors have processed it
403
+ *
404
+ * @example
405
+ * ```typescript
406
+ * try {
407
+ * await manager.applyErrorInterceptors(error);
408
+ * } catch (finalError) {
409
+ * // finalError has been processed by all registered error interceptors
410
+ * }
411
+ * ```
412
+ */
413
+ async applyErrorInterceptors(error) {
414
+ let modifiedError = error;
415
+ for (const interceptor of this.errorInterceptors) {
416
+ try {
417
+ modifiedError = await interceptor(modifiedError);
418
+ } catch (e) {
419
+ modifiedError = e;
420
+ }
421
+ }
422
+ throw modifiedError;
423
+ }
424
+ };
425
+
426
+ // src/core/api/RequestManager.ts
427
+ var RequestManager = class {
428
+ /**
429
+ * Map of active request keys to their abort controllers
430
+ * @private
431
+ */
432
+ activeRequests = /* @__PURE__ */ new Map();
433
+ /**
434
+ * Map of request keys to their correlation IDs for tracing
435
+ * @private
436
+ */
437
+ correlationMap = /* @__PURE__ */ new Map();
438
+ /**
439
+ * Registers a new request for tracking and cancellation management.
440
+ *
441
+ * If a request with the same key already exists, it will be automatically
442
+ * cancelled before the new one is registered (request deduplication).
443
+ *
444
+ * @param key - Unique identifier for the request (typically method + URL + timestamp)
445
+ * @param controller - AbortController for cancelling the request
446
+ * @param correlationId - Correlation ID for distributed tracing
447
+ *
448
+ * @example
449
+ * ```typescript
450
+ * const controller = new AbortController();
451
+ * manager.add('GET_/api/users_1699999999', controller, 'api-abc123');
452
+ * ```
453
+ */
454
+ add(key, controller, correlationId) {
455
+ this.cancel(key);
456
+ this.activeRequests.set(key, controller);
457
+ this.correlationMap.set(key, correlationId);
458
+ }
459
+ /**
460
+ * Removes a request from tracking without cancelling it.
461
+ *
462
+ * This is typically called when a request completes successfully or fails.
463
+ * Use {@link cancel} instead if you need to abort the request.
464
+ *
465
+ * @param key - Unique identifier for the request to remove
466
+ *
467
+ * @example
468
+ * ```typescript
469
+ * // Called automatically after request completes
470
+ * manager.remove('GET_/api/users_1699999999');
471
+ * ```
472
+ */
473
+ remove(key) {
474
+ this.activeRequests.delete(key);
475
+ this.correlationMap.delete(key);
476
+ }
477
+ /**
478
+ * Cancels a specific request and removes it from tracking.
479
+ *
480
+ * If the request doesn't exist or was already cancelled, this operation is a no-op.
481
+ * The associated AbortController's signal will be triggered, causing any active
482
+ * fetch operations to abort.
483
+ *
484
+ * @param key - Unique identifier for the request to cancel
485
+ *
486
+ * @example
487
+ * ```typescript
488
+ * // User navigates away, cancel the pending request
489
+ * manager.cancel('GET_/api/users_1699999999');
490
+ * ```
491
+ */
492
+ cancel(key) {
493
+ const controller = this.activeRequests.get(key);
494
+ if (controller) {
495
+ controller.abort();
496
+ this.activeRequests.delete(key);
497
+ this.correlationMap.delete(key);
498
+ }
499
+ }
500
+ /**
501
+ * Cancels all active requests and clears all tracking data.
502
+ *
503
+ * This is useful for cleanup scenarios such as:
504
+ * - User logout
505
+ * - Component unmount
506
+ * - Navigation to a different part of the application
507
+ * - Error recovery that requires a clean slate
508
+ *
509
+ * @example
510
+ * ```typescript
511
+ * // Cancel all pending requests on logout
512
+ * function handleLogout() {
513
+ * apiClient.cancelAllRequests();
514
+ * // ... rest of logout logic
515
+ * }
516
+ * ```
517
+ */
518
+ cancelAll() {
519
+ this.activeRequests.forEach((controller) => controller.abort());
520
+ this.activeRequests.clear();
521
+ this.correlationMap.clear();
522
+ }
523
+ /**
524
+ * Checks if a request with the given key is currently being tracked.
525
+ *
526
+ * @param key - Unique identifier for the request
527
+ * @returns `true` if the request is active, `false` otherwise
528
+ *
529
+ * @example
530
+ * ```typescript
531
+ * if (manager.has('GET_/api/users_1699999999')) {
532
+ * console.log('Request is still pending');
533
+ * }
534
+ * ```
535
+ */
536
+ has(key) {
537
+ return this.activeRequests.has(key);
538
+ }
539
+ /**
540
+ * Retrieves the correlation ID for a given request key.
541
+ *
542
+ * Correlation IDs are used for distributed tracing and request tracking
543
+ * across services and logs.
544
+ *
545
+ * @param key - Unique identifier for the request
546
+ * @returns The correlation ID if found, `undefined` otherwise
547
+ *
548
+ * @example
549
+ * ```typescript
550
+ * const correlationId = manager.getCorrelationId('GET_/api/users_1699999999');
551
+ * if (correlationId) {
552
+ * console.log('Trace request with ID:', correlationId);
553
+ * }
554
+ * ```
555
+ */
556
+ getCorrelationId(key) {
557
+ return this.correlationMap.get(key);
558
+ }
559
+ };
560
+
561
+ // src/core/api/Retry/RetryHandler.ts
562
+ var RetryHandler = class {
563
+ /**
564
+ * Retries a failed request with exponential backoff strategy.
565
+ *
566
+ * The retry logic works as follows:
567
+ * 1. Attempts the request immediately
568
+ * 2. On failure, checks if the error is retryable
569
+ * 3. If retryable and retries remain, waits for the current delay
570
+ * 4. Doubles the delay for the next attempt
571
+ * 5. Repeats until success or retries exhausted
572
+ *
573
+ * **Non-Retryable Errors:**
574
+ * - Validation errors (400) - Client sent bad data
575
+ * - AbortErrors - Request was explicitly cancelled
576
+ * - Requests with aborted signals
577
+ *
578
+ * **Abort Handling:**
579
+ * If the signal is aborted during a retry delay, the retry is immediately
580
+ * cancelled and an AbortError is thrown.
581
+ *
582
+ * @template T - The return type of the function being retried
583
+ * @param fn - Async function to retry on failure
584
+ * @param retries - Number of retry attempts remaining (decrements each retry)
585
+ * @param delay - Current delay in milliseconds before next retry
586
+ * @param signal - Optional AbortSignal to cancel retries
587
+ * @returns Promise resolving to the function's result on success
588
+ * @throws The last error encountered if all retries are exhausted
589
+ * @throws AbortError if the signal is aborted during execution or delay
590
+ *
591
+ * @example
592
+ * Basic retry usage:
593
+ * ```typescript
594
+ * const handler = new RetryHandler();
595
+ * const fetchUser = () => fetch('/api/users/123').then(r => r.json());
596
+ *
597
+ * try {
598
+ * const user = await handler.retryRequest(
599
+ * fetchUser,
600
+ * 3, // 3 retries
601
+ * 1000 // Start with 1s delay
602
+ * );
603
+ * console.log('User:', user);
604
+ * } catch (error) {
605
+ * console.error('Failed after all retries:', error);
606
+ * }
607
+ * ```
608
+ *
609
+ * @example
610
+ * With cancellation support:
611
+ * ```typescript
612
+ * const controller = new AbortController();
613
+ * const signal = controller.signal;
614
+ *
615
+ * // Cancel after 5 seconds
616
+ * setTimeout(() => controller.abort(), 5000);
617
+ *
618
+ * try {
619
+ * await handler.retryRequest(fetchUser, 5, 1000, signal);
620
+ * } catch (error) {
621
+ * if (error.name === 'AbortError') {
622
+ * console.log('Retry cancelled');
623
+ * }
624
+ * }
625
+ * ```
626
+ */
627
+ async retryRequest(fn, retries, delay, signal) {
628
+ try {
629
+ if (signal?.aborted) {
630
+ throw new Error(signal.reason || "Request aborted");
631
+ }
632
+ return await fn();
633
+ } catch (error) {
634
+ const err = error;
635
+ if (err.name === "AbortError" || signal?.aborted) {
636
+ throw error;
637
+ }
638
+ if (err.type === "validation_error" || err.status === 400) {
639
+ throw error;
640
+ }
641
+ if (retries === 0) throw error;
642
+ await new Promise((resolve, reject) => {
643
+ const timeoutId = setTimeout(resolve, delay);
644
+ if (signal) {
645
+ signal.addEventListener(
646
+ "abort",
647
+ () => {
648
+ clearTimeout(timeoutId);
649
+ reject(new Error(signal.reason || "Request aborted"));
650
+ },
651
+ { once: true }
652
+ );
653
+ }
654
+ });
655
+ return this.retryRequest(fn, retries - 1, delay * 2, signal);
656
+ }
657
+ }
658
+ };
659
+
660
+ // src/core/api/Signals/SignalManager.ts
661
+ var SignalManager = class {
662
+ /**
663
+ * Creates a combined AbortController that aborts when any source signal aborts.
664
+ *
665
+ * This method implements the "any" pattern for cancellation: the combined signal
666
+ * will abort as soon as ANY of the source signals abort. This is useful for
667
+ * coordinating multiple cancellation conditions:
668
+ * - User clicks cancel button
669
+ * - Request timeout expires
670
+ * - Component unmounts
671
+ * - Parent request is cancelled
672
+ *
673
+ * **Early Abort Optimization:**
674
+ * If any source signal is already aborted when this method is called,
675
+ * the returned controller is immediately aborted without setting up listeners.
676
+ *
677
+ * **Memory Management:**
678
+ * Event listeners are registered with `{ once: true }` to prevent memory leaks,
679
+ * as they automatically clean up after firing.
680
+ *
681
+ * @param signals - Array of AbortSignals to combine (undefined values are ignored)
682
+ * @returns A new AbortController that aborts when any source signal aborts
683
+ *
684
+ * @example
685
+ * User cancellation + timeout:
686
+ * ```typescript
687
+ * const userController = new AbortController();
688
+ * const timeout = manager.createTimeoutSignal(30000);
689
+ *
690
+ * const combined = manager.createCombinedSignal([
691
+ * userController.signal,
692
+ * timeout.signal
693
+ * ]);
694
+ *
695
+ * // Request will be cancelled after 30s OR when user clicks cancel
696
+ * fetch('/api/data', { signal: combined.signal });
697
+ * ```
698
+ *
699
+ * @example
700
+ * React component with cleanup:
701
+ * ```typescript
702
+ * useEffect(() => {
703
+ * const controller = new AbortController();
704
+ *
705
+ * const combined = manager.createCombinedSignal([
706
+ * controller.signal,
707
+ * unmountSignal // From component lifecycle
708
+ * ]);
709
+ *
710
+ * fetchData(combined.signal);
711
+ *
712
+ * return () => controller.abort(); // Cleanup
713
+ * }, []);
714
+ * ```
715
+ */
716
+ createCombinedSignal(signals) {
717
+ const controller = new AbortController();
718
+ for (const signal of signals) {
719
+ if (signal) {
720
+ if (signal.aborted) {
721
+ controller.abort(signal.reason);
722
+ break;
723
+ }
724
+ signal.addEventListener(
725
+ "abort",
726
+ () => {
727
+ controller.abort(signal.reason);
728
+ },
729
+ { once: true }
730
+ );
731
+ }
732
+ }
733
+ return controller;
734
+ }
735
+ /**
736
+ * Creates an AbortController that automatically aborts after a specified timeout.
737
+ *
738
+ * This method creates a time-based cancellation mechanism useful for implementing
739
+ * request timeouts and deadlines. The signal will automatically abort after the
740
+ * specified duration, providing a consistent timeout experience.
741
+ *
742
+ * **Automatic Cleanup:**
743
+ * If the signal is aborted by other means before the timeout expires, the internal
744
+ * setTimeout is automatically cleared to prevent memory leaks.
745
+ *
746
+ * **Abort Reason:**
747
+ * The abort reason includes the timeout duration for debugging purposes:
748
+ * `"Request timeout after {timeout}ms"`
749
+ *
750
+ * @param timeout - Timeout duration in milliseconds
751
+ * @returns An AbortController that will abort after the timeout
752
+ *
753
+ * @example
754
+ * Simple request timeout:
755
+ * ```typescript
756
+ * const manager = new SignalManager();
757
+ * const timeout = manager.createTimeoutSignal(5000); // 5 seconds
758
+ *
759
+ * try {
760
+ * const response = await fetch('/api/slow-endpoint', {
761
+ * signal: timeout.signal
762
+ * });
763
+ * const data = await response.json();
764
+ * } catch (error) {
765
+ * if (error.name === 'AbortError') {
766
+ * console.error('Request timed out after 5 seconds');
767
+ * }
768
+ * }
769
+ * ```
770
+ *
771
+ * @example
772
+ * Different timeouts for different operations:
773
+ * ```typescript
774
+ * // Short timeout for quick operations
775
+ * const quickTimeout = manager.createTimeoutSignal(2000);
776
+ * await fetch('/api/health', { signal: quickTimeout.signal });
777
+ *
778
+ * // Long timeout for heavy operations
779
+ * const longTimeout = manager.createTimeoutSignal(60000);
780
+ * await fetch('/api/export', { signal: longTimeout.signal });
781
+ * ```
782
+ *
783
+ * @example
784
+ * Manual cancellation before timeout:
785
+ * ```typescript
786
+ * const timeout = manager.createTimeoutSignal(30000);
787
+ *
788
+ * // If user cancels, timeout is automatically cleaned up
789
+ * timeout.abort('User cancelled');
790
+ * // Internal setTimeout is cleared, no memory leak
791
+ * ```
792
+ */
793
+ createTimeoutSignal(timeout) {
794
+ const controller = new AbortController();
795
+ const timeoutId = setTimeout(() => {
796
+ controller.abort(`Request timeout after ${timeout}ms`);
797
+ }, timeout);
798
+ controller.signal.addEventListener(
799
+ "abort",
800
+ () => {
801
+ clearTimeout(timeoutId);
802
+ },
803
+ { once: true }
804
+ );
805
+ return controller;
806
+ }
807
+ };
808
+
809
+ // src/core/api/Utils/ResponseParser.ts
810
+ var ResponseParser = class {
811
+ /**
812
+ * Parses the HTTP response body into an appropriate JavaScript type.
813
+ *
814
+ * The parsing strategy is determined by the Content-Type header:
815
+ * 1. **JSON** (application/json): Calls `response.json()`
816
+ * 2. **Text** (text/*): Calls `response.text()`
817
+ * 3. **Binary** (application/octet-stream): Calls `response.blob()`
818
+ * 4. **Unknown**: Reads as text, attempts JSON parse, falls back to raw text
819
+ *
820
+ * **Fallback Behavior:**
821
+ * For responses without a Content-Type header or with unknown types, the parser
822
+ * attempts to parse as JSON first (common for APIs that don't set proper headers).
823
+ * If JSON parsing fails, it returns the raw text.
824
+ *
825
+ * @param response - The Fetch API Response object to parse
826
+ * @returns Promise resolving to the parsed response data
827
+ * @returns Can be: JSON object/array, string, or Blob depending on Content-Type
828
+ *
829
+ * @example
830
+ * API response parsing:
831
+ * ```typescript
832
+ * const response = await fetch('/api/users');
833
+ * const data = await parser.parseResponse(response);
834
+ *
835
+ * if (typeof data === 'string') {
836
+ * console.log('Text response:', data);
837
+ * } else if (data instanceof Blob) {
838
+ * console.log('Binary response:', data.size, 'bytes');
839
+ * } else {
840
+ * console.log('JSON response:', data);
841
+ * }
842
+ * ```
843
+ *
844
+ * @example
845
+ * Handling different content types:
846
+ * ```typescript
847
+ * // CSV file download
848
+ * const csvResponse = await fetch('/api/export.csv');
849
+ * const blob = await parser.parseResponse(csvResponse);
850
+ * // Returns Blob for download
851
+ *
852
+ * // JSON API
853
+ * const jsonResponse = await fetch('/api/users');
854
+ * const users = await parser.parseResponse(jsonResponse);
855
+ * // Returns parsed JSON array
856
+ *
857
+ * // Plain text logs
858
+ * const logResponse = await fetch('/api/logs');
859
+ * const logs = await parser.parseResponse(logResponse);
860
+ * // Returns string
861
+ * ```
862
+ */
863
+ async parseResponse(response) {
864
+ const contentType = response.headers.get("content-type");
865
+ if (contentType?.includes("application/json")) {
866
+ return response.json();
867
+ } else if (contentType?.includes("text/")) {
868
+ return response.text();
869
+ } else if (contentType?.includes("application/octet-stream")) {
870
+ return response.blob();
871
+ } else {
872
+ const text = await response.text();
873
+ try {
874
+ return JSON.parse(text);
875
+ } catch {
876
+ return text;
877
+ }
878
+ }
879
+ }
880
+ };
881
+
882
+ // src/core/api/Utils/UrlBuilder.ts
883
+ var UrlBuilder = class {
884
+ /**
885
+ * Builds a complete URL by combining base URL, endpoint, and query parameters.
886
+ *
887
+ * The URL construction process:
888
+ * 1. Combines `baseURL` and `endpoint` using URL API
889
+ * 2. Iterates through query parameters
890
+ * 3. Skips null/undefined values
891
+ * 4. Handles arrays by appending multiple values with same key
892
+ * 5. Converts all values to strings
893
+ * 6. Returns fully-qualified URL string
894
+ *
895
+ * **Path Handling:**
896
+ * The endpoint can be either relative or absolute:
897
+ * - Relative: `/users` → Combined with baseURL
898
+ * - Absolute: `https://other-api.com/users` → Uses absolute URL
899
+ *
900
+ * **Encoding:**
901
+ * All parameter values are automatically URL-encoded by the URL API,
902
+ * so special characters (spaces, &, =, etc.) are safely handled.
903
+ *
904
+ * @param baseURL - Base URL for the API (e.g., 'https://api.example.com')
905
+ * @param endpoint - API endpoint path relative to baseURL (e.g., '/users/123')
906
+ * @param params - Optional query parameters as key-value pairs
907
+ * @returns The fully-qualified URL string with encoded query parameters
908
+ *
909
+ * @example
910
+ * Basic URL construction:
911
+ * ```typescript
912
+ * const url = builder.buildURL(
913
+ * 'https://api.example.com',
914
+ * '/search',
915
+ * { q: 'hello world', limit: 10 }
916
+ * );
917
+ * // => "https://api.example.com/search?q=hello+world&limit=10"
918
+ * ```
919
+ *
920
+ * @example
921
+ * Array parameters:
922
+ * ```typescript
923
+ * const url = builder.buildURL(
924
+ * 'https://api.example.com',
925
+ * '/posts',
926
+ * { tags: ['javascript', 'typescript', 'react'] }
927
+ * );
928
+ * // => "https://api.example.com/posts?tags=javascript&tags=typescript&tags=react"
929
+ * ```
930
+ *
931
+ * @example
932
+ * Null/undefined handling:
933
+ * ```typescript
934
+ * const url = builder.buildURL(
935
+ * 'https://api.example.com',
936
+ * '/users',
937
+ * {
938
+ * name: 'John',
939
+ * age: null, // Skipped
940
+ * email: undefined // Skipped
941
+ * }
942
+ * );
943
+ * // => "https://api.example.com/users?name=John"
944
+ * ```
945
+ *
946
+ * @example
947
+ * Special characters encoding:
948
+ * ```typescript
949
+ * const url = builder.buildURL(
950
+ * 'https://api.example.com',
951
+ * '/search',
952
+ * { q: 'foo & bar', category: 'code/examples' }
953
+ * );
954
+ * // => "https://api.example.com/search?q=foo+%26+bar&category=code%2Fexamples"
955
+ * ```
956
+ */
957
+ buildURL(baseURL, endpoint, params) {
958
+ const normalizedEndpoint = endpoint.startsWith("/") ? endpoint : `/${endpoint}`;
959
+ const url = new URL(normalizedEndpoint, baseURL);
960
+ if (params) {
961
+ Object.keys(params).forEach((key) => {
962
+ const value = params[key];
963
+ if (value !== void 0 && value !== null) {
964
+ if (Array.isArray(value)) {
965
+ value.forEach((v) => url.searchParams.append(key, String(v)));
966
+ } else {
967
+ url.searchParams.append(key, String(value));
968
+ }
969
+ }
970
+ });
971
+ }
972
+ return url.toString();
973
+ }
974
+ };
975
+
976
+ // src/core/api/ApiClient.ts
977
+ var ApiClient = class {
978
+ baseURL;
979
+ defaultTimeout;
980
+ interceptorManager = new InterceptorManager();
981
+ signalManager = new SignalManager();
982
+ errorNormalizer = new ErrorNormalizer();
983
+ responseParser = new ResponseParser();
984
+ urlBuilder = new UrlBuilder();
985
+ retryHandler = new RetryHandler();
986
+ requestManager = new RequestManager();
987
+ authToken = null;
988
+ correlationIdPrefix = "api";
989
+ includeCorrelationId = true;
990
+ /**
991
+ * Creates a new API client instance
992
+ * @param baseURL - Base URL for all API requests (default: empty string for relative URLs)
993
+ * @param defaultTimeout - Default request timeout in milliseconds (default: 30000)
994
+ */
995
+ constructor(baseURL = "", defaultTimeout = 3e4) {
996
+ this.baseURL = baseURL;
997
+ this.defaultTimeout = defaultTimeout;
998
+ }
999
+ /**
1000
+ * Sets the prefix for auto-generated correlation IDs
1001
+ * @param prefix - The prefix to use for correlation IDs (e.g., 'api', 'web', 'mobile')
1002
+ */
1003
+ setCorrelationIdPrefix(prefix) {
1004
+ this.correlationIdPrefix = prefix;
1005
+ }
1006
+ /**
1007
+ * Enables or disables automatic correlation ID generation
1008
+ * @param include - Whether to include correlation IDs in requests
1009
+ */
1010
+ setIncludeCorrelationId(include) {
1011
+ this.includeCorrelationId = include;
1012
+ }
1013
+ /**
1014
+ * Registers a request interceptor to modify requests before they're sent
1015
+ * @param interceptor - Function to intercept and potentially modify request config
1016
+ * @returns Function to unregister this interceptor
1017
+ *
1018
+ * @example
1019
+ * ```typescript
1020
+ * const unregister = client.addRequestInterceptor(async (config) => {
1021
+ * config.headers = config.headers || new Headers();
1022
+ * config.headers.set('X-Client-Version', '1.0.0');
1023
+ * return config;
1024
+ * });
1025
+ *
1026
+ * // Later, to remove the interceptor:
1027
+ * unregister();
1028
+ * ```
1029
+ */
1030
+ addRequestInterceptor(interceptor) {
1031
+ return this.interceptorManager.addRequestInterceptor(interceptor);
1032
+ }
1033
+ /**
1034
+ * Registers a response interceptor to modify responses before they're returned
1035
+ * @param interceptor - Function to intercept and potentially modify responses
1036
+ * @returns Function to unregister this interceptor
1037
+ *
1038
+ * @example
1039
+ * ```typescript
1040
+ * client.addResponseInterceptor(async (response) => {
1041
+ * // Transform data format
1042
+ * if (response.apiData) {
1043
+ * response.apiData = camelCaseKeys(response.apiData);
1044
+ * }
1045
+ * return response;
1046
+ * });
1047
+ * ```
1048
+ */
1049
+ addResponseInterceptor(interceptor) {
1050
+ return this.interceptorManager.addResponseInterceptor(interceptor);
1051
+ }
1052
+ /**
1053
+ * Registers an error interceptor to handle or transform errors
1054
+ * @param interceptor - Function to intercept and potentially modify errors
1055
+ * @returns Function to unregister this interceptor
1056
+ *
1057
+ * @example
1058
+ * ```typescript
1059
+ * client.addErrorInterceptor(async (error) => {
1060
+ * // Log errors to monitoring service
1061
+ * if (error.status >= 500) {
1062
+ * await monitoringService.logError(error);
1063
+ * }
1064
+ * return error; // Re-throw the error
1065
+ * });
1066
+ * ```
1067
+ */
1068
+ addErrorInterceptor(interceptor) {
1069
+ return this.interceptorManager.addErrorInterceptor(interceptor);
1070
+ }
1071
+ /**
1072
+ * Sets the authentication token for subsequent requests
1073
+ * @param token - JWT token or null to clear authentication
1074
+ *
1075
+ * @example
1076
+ * ```typescript
1077
+ * // Set token after login
1078
+ * client.setAuthToken(loginResponse.accessToken);
1079
+ *
1080
+ * // Clear token on logout
1081
+ * client.setAuthToken(null);
1082
+ * ```
1083
+ */
1084
+ setAuthToken(token) {
1085
+ this.authToken = token;
1086
+ }
1087
+ /**
1088
+ * Retrieves the current authentication token
1089
+ * @returns The current auth token or null if not set
1090
+ */
1091
+ getAuthToken() {
1092
+ return this.authToken;
1093
+ }
1094
+ /**
1095
+ * Cancels a specific request by its key
1096
+ * @param key - The unique key identifying the request to cancel
1097
+ */
1098
+ cancelRequest(key) {
1099
+ this.requestManager.cancel(key);
1100
+ }
1101
+ /**
1102
+ * Cancels all pending requests
1103
+ * Useful for cleanup on navigation or component unmount
1104
+ */
1105
+ cancelAllRequests() {
1106
+ this.requestManager.cancelAll();
1107
+ }
1108
+ /**
1109
+ * Core request method that handles all HTTP operations
1110
+ * @template T - The expected response data type
1111
+ * @param endpoint - API endpoint relative to baseURL
1112
+ * @param config - Request configuration options
1113
+ * @returns Promise resolving to ApiResponse with data or error
1114
+ *
1115
+ * @example
1116
+ * ```typescript
1117
+ * const response = await client.request<User>('/users/123', {
1118
+ * method: 'GET',
1119
+ * timeout: 5000,
1120
+ * throwErrors: false
1121
+ * });
1122
+ * ```
1123
+ */
1124
+ async request(endpoint, config = {}) {
1125
+ const correlationId = config.correlationId || (!config.skipCorrelationId && this.includeCorrelationId ? generateCorrelationId(this.correlationIdPrefix) : void 0);
1126
+ const requestKey = `${config.method || "GET"}_${endpoint}_${Date.now()}`;
1127
+ const masterController = new AbortController();
1128
+ try {
1129
+ const signals = [
1130
+ config.signal,
1131
+ config.cancelToken?.signal,
1132
+ masterController.signal
1133
+ ];
1134
+ const timeout = config.timeout || this.defaultTimeout;
1135
+ const timeoutController = this.signalManager.createTimeoutSignal(timeout);
1136
+ signals.push(timeoutController.signal);
1137
+ const combinedController = this.signalManager.createCombinedSignal(signals);
1138
+ if (correlationId) {
1139
+ this.requestManager.add(requestKey, masterController, correlationId);
1140
+ }
1141
+ const finalConfig = await this.interceptorManager.applyRequestInterceptors({
1142
+ ...config,
1143
+ signal: combinedController.signal,
1144
+ correlationId
1145
+ });
1146
+ const url = this.urlBuilder.buildURL(
1147
+ this.baseURL,
1148
+ endpoint,
1149
+ finalConfig.params
1150
+ );
1151
+ const headers = new Headers(finalConfig.headers);
1152
+ if (correlationId) {
1153
+ headers.set("X-Correlation-Id", correlationId);
1154
+ headers.set("X-Request-Id", correlationId);
1155
+ }
1156
+ if (this.authToken && !finalConfig.skipAuthRefresh) {
1157
+ headers.set("Authorization", `Bearer ${this.authToken}`);
1158
+ }
1159
+ let fetchBody = finalConfig.body;
1160
+ if (finalConfig.body && typeof finalConfig.body === "object" && !(finalConfig.body instanceof FormData) && !(finalConfig.body instanceof Blob) && !(finalConfig.body instanceof ArrayBuffer) && !(finalConfig.body instanceof URLSearchParams) && !(finalConfig.body instanceof ReadableStream)) {
1161
+ headers.set("Content-Type", "application/json");
1162
+ fetchBody = JSON.stringify(finalConfig.body);
1163
+ } else if (finalConfig.body instanceof FormData) {
1164
+ headers.delete("Content-Type");
1165
+ }
1166
+ finalConfig.headers = headers;
1167
+ const fetchPromise = async () => {
1168
+ try {
1169
+ const response = await fetch(url, {
1170
+ ...finalConfig,
1171
+ body: fetchBody,
1172
+ signal: combinedController.signal
1173
+ });
1174
+ const responseData = await this.responseParser.parseResponse(response);
1175
+ if (!response.ok) {
1176
+ const errorData = responseData;
1177
+ const error = Object.assign(
1178
+ new Error(
1179
+ errorData.title || `HTTP ${response.status}: ${response.statusText}`
1180
+ ),
1181
+ {
1182
+ type: errorData.type || this.errorNormalizer.getErrorType(response.status),
1183
+ title: errorData.title || this.errorNormalizer.getErrorTitle(response.status),
1184
+ status: response.status,
1185
+ traceId: errorData.traceId || correlationId,
1186
+ errors: errorData.errors,
1187
+ isAborted: false,
1188
+ config: finalConfig
1189
+ }
1190
+ );
1191
+ if (finalConfig.throwErrors !== false) {
1192
+ throw error;
1193
+ } else {
1194
+ return await this.interceptorManager.applyResponseInterceptors({
1195
+ error
1196
+ });
1197
+ }
1198
+ }
1199
+ const apiResponse = {
1200
+ data: responseData
1201
+ };
1202
+ return await this.interceptorManager.applyResponseInterceptors(
1203
+ apiResponse
1204
+ );
1205
+ } catch (error) {
1206
+ if (error.name === "AbortError") {
1207
+ const abortError = Object.assign(
1208
+ new Error(error.message || "Request aborted"),
1209
+ {
1210
+ type: "request_cancelled",
1211
+ title: "Request was cancelled",
1212
+ status: 0,
1213
+ traceId: correlationId,
1214
+ isAborted: true,
1215
+ config: finalConfig
1216
+ }
1217
+ );
1218
+ if (finalConfig.throwErrors !== false) {
1219
+ throw abortError;
1220
+ } else {
1221
+ return await this.interceptorManager.applyResponseInterceptors({
1222
+ error: abortError
1223
+ });
1224
+ }
1225
+ }
1226
+ throw error;
1227
+ }
1228
+ };
1229
+ if (finalConfig.retries && finalConfig.retries > 0) {
1230
+ return await this.retryHandler.retryRequest(
1231
+ fetchPromise,
1232
+ finalConfig.retries,
1233
+ finalConfig.retryDelay || 1e3,
1234
+ combinedController.signal
1235
+ );
1236
+ }
1237
+ return await fetchPromise();
1238
+ } catch (error) {
1239
+ const apiError = this.errorNormalizer.normalizeError(
1240
+ error,
1241
+ config,
1242
+ correlationId
1243
+ );
1244
+ if (config.throwErrors !== false) {
1245
+ await this.interceptorManager.applyErrorInterceptors(apiError);
1246
+ throw apiError;
1247
+ } else {
1248
+ return {
1249
+ error: apiError
1250
+ };
1251
+ }
1252
+ } finally {
1253
+ this.requestManager.remove(requestKey);
1254
+ }
1255
+ }
1256
+ /**
1257
+ * Performs a GET request
1258
+ * @template T - The expected response data type
1259
+ * @param endpoint - API endpoint
1260
+ * @param config - Optional request configuration
1261
+ * @returns Promise resolving to ApiResponse
1262
+ *
1263
+ * @example
1264
+ * ```typescript
1265
+ * const { apiData, error } = await client.get<User[]>('/users', {
1266
+ * params: { active: true },
1267
+ * timeout: 5000
1268
+ * });
1269
+ * ```
1270
+ */
1271
+ get(endpoint, config) {
1272
+ return this.request(endpoint, { ...config, method: "GET" });
1273
+ }
1274
+ /**
1275
+ * Performs a POST request
1276
+ * @template T - The expected response data type
1277
+ * @template TData - The request body data type
1278
+ * @param endpoint - API endpoint
1279
+ * @param data - Request body data
1280
+ * @param config - Optional request configuration
1281
+ * @returns Promise resolving to ApiResponse
1282
+ *
1283
+ * @example
1284
+ * ```typescript
1285
+ * const { apiData, error } = await client.post<User, CreateUserDto>('/users', {
1286
+ * name: 'John Doe',
1287
+ * email: 'john@example.com'
1288
+ * });
1289
+ * ```
1290
+ */
1291
+ post(endpoint, data, config) {
1292
+ return this.request(endpoint, { ...config, method: "POST", body: data });
1293
+ }
1294
+ /**
1295
+ * Performs a PUT request
1296
+ * @template T - The expected response data type
1297
+ * @template TData - The request body data type
1298
+ * @param endpoint - API endpoint
1299
+ * @param data - Request body data
1300
+ * @param config - Optional request configuration
1301
+ * @returns Promise resolving to ApiResponse
1302
+ *
1303
+ * @example
1304
+ * ```typescript
1305
+ * const { apiData, error } = await client.put<User, UpdateUserDto>(
1306
+ * '/users/123',
1307
+ * { name: 'Jane Doe' }
1308
+ * );
1309
+ * ```
1310
+ */
1311
+ put(endpoint, data, config) {
1312
+ return this.request(endpoint, { ...config, method: "PUT", body: data });
1313
+ }
1314
+ /**
1315
+ * Performs a PATCH request
1316
+ * @template T - The expected response data type
1317
+ * @template TData - The request body data type
1318
+ * @param endpoint - API endpoint
1319
+ * @param data - Request body data
1320
+ * @param config - Optional request configuration
1321
+ * @returns Promise resolving to ApiResponse
1322
+ *
1323
+ * @example
1324
+ * ```typescript
1325
+ * const { apiData, error } = await client.patch<User>(
1326
+ * '/users/123',
1327
+ * { status: 'active' }
1328
+ * );
1329
+ * ```
1330
+ */
1331
+ patch(endpoint, data, config) {
1332
+ return this.request(endpoint, {
1333
+ ...config,
1334
+ method: "PATCH",
1335
+ body: data
1336
+ });
1337
+ }
1338
+ /**
1339
+ * Performs a DELETE request
1340
+ * @template T - The expected response data type
1341
+ * @param endpoint - API endpoint
1342
+ * @param config - Optional request configuration
1343
+ * @returns Promise resolving to ApiResponse
1344
+ *
1345
+ * @example
1346
+ * ```typescript
1347
+ * const { error } = await client.delete('/users/123');
1348
+ * if (!error) {
1349
+ * console.log('User deleted successfully');
1350
+ * }
1351
+ * ```
1352
+ */
1353
+ delete(endpoint, config) {
1354
+ return this.request(endpoint, { ...config, method: "DELETE" });
1355
+ }
1356
+ /**
1357
+ * Performs a filtered list request with pagination and sorting
1358
+ * @template TListModel - The type of individual list items
1359
+ * @template TFilter - The filter criteria type
1360
+ * @param url - API endpoint
1361
+ * @param data - Pagination and filter data
1362
+ * @param config - Optional request configuration
1363
+ * @returns Promise resolving to paginated list response
1364
+ *
1365
+ * @example
1366
+ * ```typescript
1367
+ * const { apiData, error } = await client.filter<User, UserFilter>(
1368
+ * '/users/filter',
1369
+ * {
1370
+ * pageOffset: 0,
1371
+ * pageSize: 20,
1372
+ * sortField: 'createdAt',
1373
+ * sortOrder: 'desc',
1374
+ * filterModel: { status: 'active' }
1375
+ * }
1376
+ * );
1377
+ *
1378
+ * if (apiData) {
1379
+ * console.log(`Found ${apiData.Total} users`);
1380
+ * console.log('Users:', apiData.Data);
1381
+ * }
1382
+ * ```
1383
+ */
1384
+ filter(url, data, config) {
1385
+ const mergedData = { ...data, ...data.filterModel };
1386
+ return this.request(url, {
1387
+ ...config,
1388
+ method: "POST",
1389
+ body: mergedData
1390
+ });
1391
+ }
1392
+ };
1393
+
1394
+ // src/core/api/createApiClient.ts
1395
+ var globalApiClient = null;
1396
+ function createApiClient(config) {
1397
+ const {
1398
+ baseURL,
1399
+ timeout = 3e4,
1400
+ correlationIdPrefix,
1401
+ includeCorrelationId = true,
1402
+ tokenStorageKey,
1403
+ requestInterceptors = [],
1404
+ responseInterceptors = [],
1405
+ errorInterceptors = []
1406
+ } = config;
1407
+ const client = new ApiClient(baseURL, timeout);
1408
+ client.addRequestInterceptor((config2) => {
1409
+ const token = localStorage.getItem(tokenStorageKey);
1410
+ if (token && !config2.skipAuthRefresh) {
1411
+ config2.headers = {
1412
+ ...config2.headers,
1413
+ Authorization: `Bearer ${token}`
1414
+ };
1415
+ }
1416
+ return config2;
1417
+ });
1418
+ client.setCorrelationIdPrefix(correlationIdPrefix || "web");
1419
+ client.setIncludeCorrelationId(includeCorrelationId || true);
1420
+ requestInterceptors.forEach((interceptor) => {
1421
+ client.addRequestInterceptor(interceptor);
1422
+ });
1423
+ responseInterceptors.forEach((interceptor) => {
1424
+ client.addResponseInterceptor(interceptor);
1425
+ });
1426
+ errorInterceptors.forEach((interceptor) => {
1427
+ client.addErrorInterceptor(interceptor);
1428
+ });
1429
+ return client;
1430
+ }
1431
+ function getGlobalApiClient() {
1432
+ if (!globalApiClient) {
1433
+ throw new Error(
1434
+ "getGlobalApiClient: No global client exists. Call initializeGlobalApiClient() first to configure the client."
1435
+ );
1436
+ }
1437
+ return globalApiClient;
1438
+ }
1439
+ function initializeGlobalApiClient(config) {
1440
+ if (globalApiClient) {
1441
+ throw new Error(
1442
+ "initializeGlobalApiClient: Global client already initialized. Use resetGlobalApiClient() first if you need to reinitialize."
1443
+ );
1444
+ }
1445
+ globalApiClient = createApiClient(config);
1446
+ return globalApiClient;
1447
+ }
1448
+ function setGlobalApiClient(client) {
1449
+ globalApiClient = client;
1450
+ }
1451
+ function resetGlobalApiClient() {
1452
+ globalApiClient = null;
1453
+ }
1454
+
1455
+ // src/core/api/types/CancelToken.ts
1456
+ var CancelToken = class _CancelToken {
1457
+ abortController;
1458
+ cancelPromise;
1459
+ cancelResolve;
1460
+ constructor() {
1461
+ this.abortController = new AbortController();
1462
+ this.cancelPromise = new Promise((resolve) => {
1463
+ this.cancelResolve = resolve;
1464
+ });
1465
+ }
1466
+ get signal() {
1467
+ return this.abortController.signal;
1468
+ }
1469
+ cancel(reason) {
1470
+ this.abortController.abort(reason);
1471
+ this.cancelResolve?.();
1472
+ }
1473
+ get isCancelled() {
1474
+ return this.abortController.signal.aborted;
1475
+ }
1476
+ throwIfCancelled() {
1477
+ if (this.isCancelled) {
1478
+ throw new Error("Request cancelled");
1479
+ }
1480
+ }
1481
+ static source() {
1482
+ const token = new _CancelToken();
1483
+ return {
1484
+ token,
1485
+ cancel: (reason) => token.cancel(reason)
1486
+ };
1487
+ }
1488
+ };
1489
+
1490
+ // src/core/api/useValidationErrors.ts
1491
+ import { useCallback } from "react";
1492
+ function useValidationErrors(error) {
1493
+ const getFieldError = useCallback(
1494
+ (field) => {
1495
+ if (!error?.errors || !error.errors[field]) return null;
1496
+ const fieldError = error.errors[field];
1497
+ if (typeof fieldError === "string") return fieldError;
1498
+ if (Array.isArray(fieldError)) return fieldError[0];
1499
+ if (typeof fieldError === "object" && "message" in fieldError) {
1500
+ return fieldError.message;
1501
+ }
1502
+ return null;
1503
+ },
1504
+ [error]
1505
+ );
1506
+ const hasFieldError = useCallback(
1507
+ (field) => {
1508
+ return !!getFieldError(field);
1509
+ },
1510
+ [getFieldError]
1511
+ );
1512
+ const getAllErrors = useCallback(() => {
1513
+ if (!error?.errors) return {};
1514
+ const result = {};
1515
+ Object.entries(error.errors).forEach(([key, value]) => {
1516
+ if (typeof value === "string") {
1517
+ result[key] = value;
1518
+ } else if (Array.isArray(value)) {
1519
+ result[key] = value.join(", ");
1520
+ } else if (typeof value === "object" && value && "message" in value) {
1521
+ result[key] = value.message;
1522
+ }
1523
+ });
1524
+ return result;
1525
+ }, [error]);
1526
+ return {
1527
+ getFieldError,
1528
+ hasFieldError,
1529
+ getAllErrors,
1530
+ hasErrors: error?.errors
1531
+ };
1532
+ }
1533
+
1534
+ // src/core/components/AuthorizedView/AuthorizedView.tsx
1535
+ import { Fragment, jsx } from "react/jsx-runtime";
1536
+ var AuthorizedView = ({ children, show }) => {
1537
+ if (!show) return /* @__PURE__ */ jsx(Fragment, {});
1538
+ return /* @__PURE__ */ jsx(Fragment, { children });
1539
+ };
1540
+
1541
+ // src/core/components/CancelButton/CancelButton.tsx
1542
+ import { Button } from "@mui/material";
1543
+ import { jsx as jsx2 } from "react/jsx-runtime";
1544
+ var CancelButton = ({
1545
+ children = "Cancel",
1546
+ variant = "outlined",
1547
+ sx,
1548
+ ...rest
1549
+ }) => /* @__PURE__ */ jsx2(Button, { variant, sx: { width: "6rem", ...sx }, ...rest, children });
1550
+
1551
+ // src/core/components/ClearButton/ClearButton.tsx
1552
+ import { Button as Button2 } from "@mui/material";
1553
+ import { jsx as jsx3 } from "react/jsx-runtime";
1554
+ var ClearButton = ({
1555
+ isSubmitting,
1556
+ handleClear,
1557
+ sx,
1558
+ storeKey
1559
+ }) => {
1560
+ const onClick = () => {
1561
+ handleClear();
1562
+ if (storeKey != null) {
1563
+ localStorage.removeItem(storeKey);
1564
+ }
1565
+ };
1566
+ return /* @__PURE__ */ jsx3(
1567
+ Button2,
1568
+ {
1569
+ variant: "outlined",
1570
+ onClick,
1571
+ disabled: isSubmitting,
1572
+ sx,
1573
+ children: "Clear"
1574
+ }
1575
+ );
1576
+ };
1577
+
1578
+ // src/core/components/Containers/SimpleContainer.tsx
1579
+ import { Container } from "@mui/material";
1580
+ import { jsx as jsx4 } from "react/jsx-runtime";
1581
+ var SimpleContainer = ({
1582
+ children,
1583
+ className,
1584
+ sx
1585
+ }) => /* @__PURE__ */ jsx4(Container, { className, sx: { ...sx }, children });
1586
+
1587
+ // src/core/components/FilterButton/FilterButton.tsx
1588
+ import FilterAltIcon from "@mui/icons-material/FilterAlt";
1589
+ import { LoadingButton } from "@mui/lab";
1590
+ import { Badge } from "@mui/material";
1591
+ import { jsx as jsx5 } from "react/jsx-runtime";
1592
+ var FilterButton = ({
1593
+ isSubmitting,
1594
+ show,
1595
+ title,
1596
+ icon,
1597
+ sx,
1598
+ iconSx
1599
+ }) => {
1600
+ return /* @__PURE__ */ jsx5(
1601
+ LoadingButton,
1602
+ {
1603
+ type: "submit",
1604
+ variant: "contained",
1605
+ loading: isSubmitting,
1606
+ disabled: !show,
1607
+ disableRipple: true,
1608
+ color: "primary",
1609
+ sx: {
1610
+ display: "flex",
1611
+ alignItems: "center",
1612
+ ...sx
1613
+ },
1614
+ startIcon: /* @__PURE__ */ jsx5(Badge, { color: "error", variant: "standard", children: icon ? icon : /* @__PURE__ */ jsx5(FilterAltIcon, { width: "20", height: "20", sx: iconSx }) }),
1615
+ children: title?.trim() === "" || !title ? "Filter" : title
1616
+ }
1617
+ );
1618
+ };
1619
+
1620
+ // src/core/components/FilterDisplay/FilterChip.tsx
1621
+ import Chip from "@mui/material/Chip";
1622
+ import { memo } from "react";
1623
+ import { jsx as jsx6 } from "react/jsx-runtime";
1624
+ var FilterChip = memo(
1625
+ ({
1626
+ fieldKey,
1627
+ filter,
1628
+ onDelete
1629
+ }) => {
1630
+ const hasValue = filter.Value !== null && filter.Value !== void 0 && filter.Value !== "";
1631
+ const label = `${fieldKey.replace("PK", "")}: ${filter.Label}`;
1632
+ return /* @__PURE__ */ jsx6(
1633
+ Chip,
1634
+ {
1635
+ label,
1636
+ variant: hasValue ? "filled" : "outlined",
1637
+ size: "small",
1638
+ onDelete: hasValue ? onDelete : void 0
1639
+ },
1640
+ fieldKey
1641
+ );
1642
+ }
1643
+ );
1644
+ FilterChip.displayName = "FilterChip";
1645
+
1646
+ // src/core/components/FilterDisplay/FilterDisplay.tsx
1647
+ import { Card, CardContent, Typography, Box } from "@mui/material";
1648
+ import { memo as memo2, useMemo } from "react";
1649
+ import { jsx as jsx7, jsxs } from "react/jsx-runtime";
1650
+ var ProgramsFilterDisplay = memo2(
1651
+ (props) => {
1652
+ const { friendlyFilter, onFriendlyFilterChange } = props;
1653
+ const deleteHandlers = useMemo(() => {
1654
+ if (!onFriendlyFilterChange) return {};
1655
+ const handlers = {};
1656
+ for (const key of Object.keys(friendlyFilter)) {
1657
+ handlers[key] = () => onFriendlyFilterChange(key);
1658
+ }
1659
+ return handlers;
1660
+ }, [onFriendlyFilterChange, friendlyFilter]);
1661
+ const chipList = useMemo(() => {
1662
+ return Object.entries(friendlyFilter).map(([key, filter]) => /* @__PURE__ */ jsx7(
1663
+ FilterChip,
1664
+ {
1665
+ fieldKey: key,
1666
+ filter,
1667
+ onDelete: deleteHandlers[key]
1668
+ },
1669
+ key
1670
+ ));
1671
+ }, [friendlyFilter, deleteHandlers]);
1672
+ return /* @__PURE__ */ jsx7(Card, { sx: { mb: 2 }, children: /* @__PURE__ */ jsxs(CardContent, { children: [
1673
+ /* @__PURE__ */ jsx7(Typography, { variant: "h6", gutterBottom: true, children: "Active Filters" }),
1674
+ /* @__PURE__ */ jsx7(Box, { display: "flex", gap: 1, flexWrap: "wrap", children: chipList })
1675
+ ] }) });
1676
+ }
1677
+ );
1678
+ ProgramsFilterDisplay.displayName = "FilterDisplay";
1679
+
1680
+ // src/core/components/FilterWrapper/FilterWrapper.tsx
1681
+ import ManageSearchIcon from "@mui/icons-material/ManageSearch";
1682
+ import {
1683
+ Box as Box2,
1684
+ Card as Card2,
1685
+ CardContent as CardContent2,
1686
+ CardHeader,
1687
+ Divider,
1688
+ Grid,
1689
+ Typography as Typography2,
1690
+ useTheme
1691
+ } from "@mui/material";
1692
+ import { Fragment as Fragment2, jsx as jsx8, jsxs as jsxs2 } from "react/jsx-runtime";
1693
+ var FilterWrapper = ({
1694
+ children,
1695
+ title,
1696
+ filterCount,
1697
+ cardSx,
1698
+ textSx,
1699
+ icon,
1700
+ iconSx,
1701
+ showCount
1702
+ }) => {
1703
+ const theme = useTheme();
1704
+ return /* @__PURE__ */ jsxs2(
1705
+ Card2,
1706
+ {
1707
+ sx: {
1708
+ position: "relative",
1709
+ borderRadius: "0px",
1710
+ mb: 2,
1711
+ ...cardSx
1712
+ },
1713
+ children: [
1714
+ /* @__PURE__ */ jsx8(
1715
+ CardHeader,
1716
+ {
1717
+ sx: {
1718
+ display: "flex",
1719
+ flexWrap: "wrap",
1720
+ p: "1rem",
1721
+ ".MuiCardHeader-action": {
1722
+ margin: 0,
1723
+ alignSelf: "center"
1724
+ },
1725
+ alignItems: "center"
1726
+ },
1727
+ title: /* @__PURE__ */ jsxs2(Box2, { sx: { display: "flex", alignItems: "center", gap: 0.5 }, children: [
1728
+ icon ? icon : /* @__PURE__ */ jsx8(
1729
+ ManageSearchIcon,
1730
+ {
1731
+ sx: {
1732
+ height: "2.5rem",
1733
+ color: theme.palette.primary.main,
1734
+ ...iconSx
1735
+ }
1736
+ }
1737
+ ),
1738
+ /* @__PURE__ */ jsxs2(
1739
+ Typography2,
1740
+ {
1741
+ variant: "h5",
1742
+ sx: {
1743
+ fontWeight: "bold",
1744
+ color: theme.palette.primary.main,
1745
+ ...textSx
1746
+ },
1747
+ children: [
1748
+ title ? title : "Filter",
1749
+ " ",
1750
+ showCount ? `(${filterCount ? filterCount : 0})` : /* @__PURE__ */ jsx8(Fragment2, {})
1751
+ ]
1752
+ }
1753
+ )
1754
+ ] })
1755
+ }
1756
+ ),
1757
+ /* @__PURE__ */ jsx8(Divider, {}),
1758
+ /* @__PURE__ */ jsx8(CardContent2, { sx: { py: 2 }, children: /* @__PURE__ */ jsx8(Grid, { container: true, spacing: 2, children }) })
1759
+ ]
1760
+ }
1761
+ );
1762
+ };
1763
+
1764
+ // src/core/components/Footer/Footer.tsx
1765
+ import { Box as Box3, Typography as Typography3 } from "@mui/material";
1766
+ import { jsx as jsx9 } from "react/jsx-runtime";
1767
+ var Footer = () => {
1768
+ const currentYear = (/* @__PURE__ */ new Date()).getFullYear();
1769
+ return /* @__PURE__ */ jsx9(
1770
+ Box3,
1771
+ {
1772
+ component: "footer",
1773
+ sx: {
1774
+ py: 2,
1775
+ px: 4,
1776
+ mt: "auto",
1777
+ backgroundColor: (theme) => theme.palette.mode === "light" ? theme.palette.grey[200] : theme.palette.grey[800]
1778
+ },
1779
+ children: /* @__PURE__ */ jsx9(Typography3, { variant: "body2", color: "text.secondary", align: "center", children: `\xA9 Copyright ${currentYear} GN. All rights reserved by Parul University.` })
1780
+ }
1781
+ );
1782
+ };
1783
+
1784
+ // src/core/components/LabelText/LabelText.tsx
1785
+ import { Grid as Grid2, Tooltip, Typography as Typography4, useTheme as useTheme2 } from "@mui/material";
1786
+ import { jsx as jsx10, jsxs as jsxs3 } from "react/jsx-runtime";
1787
+ var LabelText = ({
1788
+ label,
1789
+ value,
1790
+ gridSize,
1791
+ containerSize,
1792
+ labelSx,
1793
+ valueSx
1794
+ }) => {
1795
+ const theme = useTheme2();
1796
+ const defaultGridSize = {
1797
+ labelSize: { xs: 6, sm: 6, md: 6 },
1798
+ valueSize: { xs: 12, sm: 6, md: 6 }
1799
+ };
1800
+ const defaultContainerSize = { xs: 12, sm: 6, md: 6 };
1801
+ const size = gridSize || defaultGridSize;
1802
+ const container = containerSize || defaultContainerSize;
1803
+ return /* @__PURE__ */ jsxs3(
1804
+ Grid2,
1805
+ {
1806
+ size: container,
1807
+ sx: {
1808
+ display: "flex",
1809
+ flexDirection: { xs: "column", sm: "row", md: "row" },
1810
+ "&:hover": {
1811
+ "&:hover": {
1812
+ backgroundColor: theme.vars?.palette.action.hover
1813
+ },
1814
+ overflow: "hidden"
1815
+ }
1816
+ },
1817
+ children: [
1818
+ /* @__PURE__ */ jsxs3(
1819
+ Grid2,
1820
+ {
1821
+ size: size.labelSize,
1822
+ sx: {
1823
+ padding: "5px",
1824
+ fontSize: "14px",
1825
+ textAlign: { xs: "left", sm: "right", md: "right" },
1826
+ ...labelSx
1827
+ },
1828
+ children: [
1829
+ label,
1830
+ " :"
1831
+ ]
1832
+ }
1833
+ ),
1834
+ /* @__PURE__ */ jsx10(
1835
+ Grid2,
1836
+ {
1837
+ size: size.valueSize,
1838
+ sx: { padding: "5px", display: "flex", flexWrap: "wrap" },
1839
+ children: /* @__PURE__ */ jsx10(Tooltip, { title: value, arrow: true, children: /* @__PURE__ */ jsx10(
1840
+ Typography4,
1841
+ {
1842
+ sx: {
1843
+ fontSize: "14px",
1844
+ wordBreak: "break-word",
1845
+ overflow: "hidden",
1846
+ display: "-webkit-box",
1847
+ textOverflow: "ellipsis",
1848
+ WebkitLineClamp: 2,
1849
+ WebkitBoxOrient: "vertical",
1850
+ ...valueSx,
1851
+ color: "#078dee"
1852
+ },
1853
+ children: value ? value : "-"
1854
+ }
1855
+ ) })
1856
+ }
1857
+ )
1858
+ ]
1859
+ }
1860
+ );
1861
+ };
1862
+
1863
+ // src/core/components/RenderIf/RenderIf.tsx
1864
+ import { Fragment as Fragment3, jsx as jsx11 } from "react/jsx-runtime";
1865
+ var RenderIf = ({
1866
+ show,
1867
+ children
1868
+ }) => {
1869
+ return show ? /* @__PURE__ */ jsx11(Fragment3, { children }) : null;
1870
+ };
1871
+
1872
+ // src/core/components/SectionBox/SectionBox.tsx
1873
+ import { Box as Box4, Divider as Divider2, Grid as Grid3, Stack, Typography as Typography5 } from "@mui/material";
1874
+ import { memo as memo3, useMemo as useMemo2 } from "react";
1875
+ import { Fragment as Fragment4, jsx as jsx12, jsxs as jsxs4 } from "react/jsx-runtime";
1876
+ var getSectionTheme = (variant = "default") => {
1877
+ const themes = {
1878
+ default: {
1879
+ bgcolor: "#faebd7",
1880
+ color: "#925d21"
1881
+ },
1882
+ form: {
1883
+ bgcolor: "#cdced1",
1884
+ color: "black"
1885
+ },
1886
+ info: {
1887
+ bgcolor: "#e3f2fd",
1888
+ color: "#1976d2"
1889
+ },
1890
+ warning: {
1891
+ bgcolor: "#fff3e0",
1892
+ color: "#f57c00"
1893
+ },
1894
+ error: {
1895
+ bgcolor: "#ffebee",
1896
+ color: "#d32f2f"
1897
+ }
1898
+ };
1899
+ return themes[variant];
1900
+ };
1901
+ var SectionBox = memo3(
1902
+ ({
1903
+ title,
1904
+ children,
1905
+ spacing = 0,
1906
+ containerSx,
1907
+ titleSx,
1908
+ variant = "default",
1909
+ icon,
1910
+ actions
1911
+ }) => {
1912
+ const themeColors = useMemo2(() => getSectionTheme(variant), [variant]);
1913
+ const headerSx = useMemo2(
1914
+ () => ({
1915
+ px: 1.5,
1916
+ py: 0.1,
1917
+ width: "fit-content",
1918
+ ...themeColors,
1919
+ ...titleSx
1920
+ }),
1921
+ [themeColors, titleSx]
1922
+ );
1923
+ const contentSx = useMemo2(
1924
+ () => ({
1925
+ padding: "16px",
1926
+ ...containerSx
1927
+ }),
1928
+ [containerSx]
1929
+ );
1930
+ return /* @__PURE__ */ jsxs4(Fragment4, { children: [
1931
+ /* @__PURE__ */ jsxs4(Box4, { sx: { display: "flex", flexDirection: "column", width: "100%" }, children: [
1932
+ /* @__PURE__ */ jsxs4(
1933
+ Stack,
1934
+ {
1935
+ direction: "row",
1936
+ justifyContent: "space-between",
1937
+ alignItems: "center",
1938
+ sx: headerSx,
1939
+ children: [
1940
+ /* @__PURE__ */ jsxs4(Stack, { direction: "row", alignItems: "center", spacing: 1, children: [
1941
+ icon,
1942
+ /* @__PURE__ */ jsx12(Typography5, { sx: { fontSize: "15px", fontWeight: 400 }, children: title })
1943
+ ] }),
1944
+ actions
1945
+ ]
1946
+ }
1947
+ ),
1948
+ /* @__PURE__ */ jsx12(Divider2, {})
1949
+ ] }),
1950
+ /* @__PURE__ */ jsx12(Grid3, { container: true, spacing, sx: contentSx, children })
1951
+ ] });
1952
+ }
1953
+ );
1954
+
1955
+ // src/core/components/SimpleTabs/SimpleTabs.tsx
1956
+ import { TabContext } from "@mui/lab";
1957
+ import { Box as Box5, Tab, Tabs } from "@mui/material";
1958
+ import { useState } from "react";
1959
+ import { jsx as jsx13, jsxs as jsxs5 } from "react/jsx-runtime";
1960
+ var SimpleTabs = ({
1961
+ tabs,
1962
+ defaultValue = 1,
1963
+ onTabChange,
1964
+ children,
1965
+ tabSx,
1966
+ tabsSx
1967
+ }) => {
1968
+ const [value, setValue] = useState(defaultValue);
1969
+ const handleChange = (event, newValue) => {
1970
+ setValue(newValue);
1971
+ if (onTabChange) onTabChange(newValue);
1972
+ };
1973
+ return /* @__PURE__ */ jsxs5(TabContext, { value, children: [
1974
+ /* @__PURE__ */ jsx13(Box5, { sx: { borderBottom: 1, borderColor: "divider", width: "100%" }, children: /* @__PURE__ */ jsx13(
1975
+ Tabs,
1976
+ {
1977
+ value,
1978
+ onChange: handleChange,
1979
+ sx: { px: 2, py: 0, ...tabsSx },
1980
+ children: tabs.map((tab) => /* @__PURE__ */ jsx13(
1981
+ Tab,
1982
+ {
1983
+ label: tab.label,
1984
+ value: tab.value,
1985
+ disabled: tab.permission === false,
1986
+ sx: { fontSize: "1rem", ...tabSx }
1987
+ },
1988
+ tab.value
1989
+ ))
1990
+ }
1991
+ ) }),
1992
+ children
1993
+ ] });
1994
+ };
1995
+
1996
+ // src/core/components/SubmitButton/SubmitButton.tsx
1997
+ import { LoadingButton as LoadingButton2 } from "@mui/lab";
1998
+ import { jsx as jsx14 } from "react/jsx-runtime";
1999
+ var SubmitButton = ({
2000
+ loading = false,
2001
+ ...rest
2002
+ }) => /* @__PURE__ */ jsx14(
2003
+ LoadingButton2,
2004
+ {
2005
+ loading,
2006
+ variant: "contained",
2007
+ color: "primary",
2008
+ type: "submit",
2009
+ ...rest,
2010
+ sx: { fontWeight: 400 },
2011
+ children: "Submit"
2012
+ }
2013
+ );
2014
+
2015
+ // src/core/components/WithRef/WithRef.tsx
2016
+ import { forwardRef } from "react";
2017
+ function withDataModal(component) {
2018
+ return forwardRef(
2019
+ (props, ref) => component({ ...props, ref })
2020
+ );
2021
+ }
2022
+
2023
+ // src/core/config.ts
2024
+ var Config = {
2025
+ defaultPageSize: 20,
2026
+ apiBaseUrl: "http://localhost:5143"
2027
+ // apiBaseUrl: 'http://192.168.1.246:5143',
2028
+ };
2029
+ var dateTimePatterns = {
2030
+ dateTime: "DD MMM YYYY h:mm A",
2031
+ // 17 Apr 2022 12:00 am
2032
+ date: "DD MMM YYYY",
2033
+ // 17 Apr 2022
2034
+ month_year_short_format: "MMM YYYY",
2035
+ month_year_full_format: "MMMM YYYY",
2036
+ year: "YYYY",
2037
+ time: "h:mm a",
2038
+ // 12:00 am
2039
+ split: {
2040
+ dateTime: "DD/MM/YYYY h:mm A",
2041
+ // 17/04/2022 12:00 am
2042
+ date: "DD/MM/YYYY"
2043
+ // 17/04/2022
2044
+ },
2045
+ paramCase: {
2046
+ dateTime: "DD-MM-YYYY h:mm A",
2047
+ // 17-04-2022 12:00 am
2048
+ date: "DD-MM-YYYY",
2049
+ // 17-04-2022
2050
+ dateReverse: "YYYY-MM-DD",
2051
+ // 2022-04-17 for compare date
2052
+ MonthYear: "MMM-YYYY"
2053
+ }
2054
+ };
2055
+
2056
+ // src/core/hooks/useApiClient.ts
2057
+ import { useMemo as useMemo3 } from "react";
2058
+ function useApiClient(config) {
2059
+ return useMemo3(
2060
+ () => createApiClient(config),
2061
+ // eslint-disable-next-line react-hooks/exhaustive-deps
2062
+ [
2063
+ config.baseURL,
2064
+ config.timeout,
2065
+ config.correlationIdPrefix,
2066
+ config.includeCorrelationId,
2067
+ config.tokenStorageKey,
2068
+ config.authToken,
2069
+ config.requestInterceptors,
2070
+ config.responseInterceptors,
2071
+ config.errorInterceptors
2072
+ ]
2073
+ );
2074
+ }
2075
+
2076
+ // src/core/hooks/useFormErrorHandler.ts
2077
+ import { useCallback as useCallback2 } from "react";
2078
+ import { toast } from "sonner";
2079
+ var useFormErrorHandler = ({
2080
+ setError,
2081
+ successMessage = {
2082
+ create: "Created successfully",
2083
+ update: "Updated successfully"
2084
+ },
2085
+ errorMessage = {
2086
+ noChanges: "No changes were made",
2087
+ general: "Failed to save. Please try again."
2088
+ }
2089
+ }) => {
2090
+ const getFieldError = useCallback2(
2091
+ (fields, fieldName) => {
2092
+ if (!fields || !fields[fieldName]) return void 0;
2093
+ const fieldError = fields[fieldName];
2094
+ if (typeof fieldError === "string") {
2095
+ return fieldError;
2096
+ }
2097
+ if (Array.isArray(fieldError)) {
2098
+ return fieldError.join(", ");
2099
+ }
2100
+ if (typeof fieldError === "object" && "message" in fieldError) {
2101
+ return fieldError.message;
2102
+ }
2103
+ return void 0;
2104
+ },
2105
+ []
2106
+ );
2107
+ const handleSuccess = useCallback2(
2108
+ (isEditing, rowsAffected) => {
2109
+ if (rowsAffected !== void 0 && rowsAffected > 0) {
2110
+ toast.success(
2111
+ isEditing ? successMessage.update : successMessage.create
2112
+ );
2113
+ return true;
2114
+ } else if (rowsAffected === 0) {
2115
+ toast.error(errorMessage.noChanges);
2116
+ return false;
2117
+ }
2118
+ toast.success(isEditing ? successMessage.update : successMessage.create);
2119
+ return true;
2120
+ },
2121
+ [successMessage, errorMessage]
2122
+ );
2123
+ const handleError = useCallback2(
2124
+ (processedError) => {
2125
+ if (processedError.type === "validation_error" && processedError.errors && setError) {
2126
+ Object.keys(processedError.errors).forEach((fieldName) => {
2127
+ const fieldError = getFieldError(processedError.errors, fieldName);
2128
+ if (fieldError) {
2129
+ setError(fieldName, {
2130
+ type: "server",
2131
+ message: fieldError
2132
+ });
2133
+ }
2134
+ });
2135
+ toast.error(
2136
+ processedError.title || "Please check the form for validation errors"
2137
+ );
2138
+ } else {
2139
+ toast.error(processedError.title || errorMessage.general);
2140
+ }
2141
+ },
2142
+ [errorMessage.general, getFieldError, setError]
2143
+ );
2144
+ return {
2145
+ handleSuccess,
2146
+ handleError
2147
+ };
2148
+ };
2149
+ var useDeleteHandler = ({
2150
+ successMessage = "Deleted successfully",
2151
+ errorMessage = "Failed to delete. Please try again."
2152
+ } = {}) => {
2153
+ return useFormErrorHandler({
2154
+ successMessage: {
2155
+ create: successMessage,
2156
+ // Not used for delete, but required for type
2157
+ update: successMessage
2158
+ },
2159
+ errorMessage: {
2160
+ noChanges: "No changes were made",
2161
+ // Not typically used for delete
2162
+ general: errorMessage
2163
+ }
2164
+ // setError is omitted (undefined) for delete operations
2165
+ });
2166
+ };
2167
+
2168
+ // src/core/utils/CacheUtility/index.ts
2169
+ import { useQueryClient } from "@tanstack/react-query";
2170
+ import { useMemo as useMemo4 } from "react";
2171
+ var CacheUtility = class {
2172
+ constructor(queryClient) {
2173
+ this.queryClient = queryClient;
2174
+ }
2175
+ /**
2176
+ * Get cached data using only the queryKey from query factory
2177
+ */
2178
+ getCachedData(queryKey) {
2179
+ return this.queryClient.getQueryData(queryKey);
2180
+ }
2181
+ /**
2182
+ * Get cached data with transformation using select function
2183
+ */
2184
+ getCachedDataWithSelect(queryKey, select) {
2185
+ const cachedData = this.queryClient.getQueryData(queryKey);
2186
+ if (cachedData === void 0) {
2187
+ return void 0;
2188
+ }
2189
+ return select(cachedData);
2190
+ }
2191
+ };
2192
+ function useCacheUtility() {
2193
+ const queryClient = useQueryClient();
2194
+ return useMemo4(() => new CacheUtility(queryClient), [queryClient]);
2195
+ }
2196
+
2197
+ // src/core/utils/watch/core.ts
2198
+ import { useWatch } from "react-hook-form";
2199
+ var useWatchForm = (control) => useWatch({ control });
2200
+ var useWatchField = (control, name) => useWatch({ control, name });
2201
+ var useWatchFields = (control, names) => useWatch({ control, name: names });
2202
+
2203
+ // src/core/utils/watch/utilities.ts
2204
+ import { useEffect, useMemo as useMemo5, useState as useState2 } from "react";
2205
+ import { useWatch as useWatch2 } from "react-hook-form";
2206
+ var useWatchTransform = (control, name, transform) => {
2207
+ const value = useWatch2({ control, name });
2208
+ return useMemo5(() => transform(value), [value, transform]);
2209
+ };
2210
+ var useWatchDefault = (control, name, defaultValue) => {
2211
+ const value = useWatch2({ control, name });
2212
+ return value ?? defaultValue;
2213
+ };
2214
+ var useWatchBoolean = (control, name, defaultValue = false) => {
2215
+ const value = useWatch2({ control, name });
2216
+ return Boolean(value ?? defaultValue);
2217
+ };
2218
+ var useWatchBatch = (control, fields) => {
2219
+ const values = useWatch2({ control, name: fields });
2220
+ return useMemo5(() => {
2221
+ const result = {};
2222
+ fields.forEach((field, index) => {
2223
+ result[field] = values[index];
2224
+ });
2225
+ return result;
2226
+ }, [values, fields]);
2227
+ };
2228
+ var useWatchConditional = (control, name, shouldWatch, fallback) => {
2229
+ const activeValue = useWatch2({
2230
+ control,
2231
+ name,
2232
+ disabled: !shouldWatch
2233
+ });
2234
+ return shouldWatch ? activeValue : fallback;
2235
+ };
2236
+ var useWatchDebounced = (control, name, delay = 300) => {
2237
+ const value = useWatch2({ control, name });
2238
+ const [debouncedValue, setDebouncedValue] = useState2(value);
2239
+ useEffect(() => {
2240
+ const timer = setTimeout(() => {
2241
+ setDebouncedValue(value);
2242
+ }, delay);
2243
+ return () => clearTimeout(timer);
2244
+ }, [value, delay]);
2245
+ return debouncedValue;
2246
+ };
2247
+ var useWatchSelector = (control, name, selector, deps = []) => {
2248
+ const value = useWatch2({ control, name });
2249
+ return useMemo5(
2250
+ () => selector(value),
2251
+ [value, selector, ...deps]
2252
+ // eslint-disable-line react-hooks/exhaustive-deps
2253
+ );
2254
+ };
2255
+
2256
+ // src/core/utils/watch/index.ts
2257
+ var typedWatch = {
2258
+ // === CORE FUNCTIONS ===
2259
+ /** Watch entire form */
2260
+ form: useWatchForm,
2261
+ /** Watch single field */
2262
+ field: useWatchField,
2263
+ /** Watch multiple fields */
2264
+ fields: useWatchFields,
2265
+ // === UTILITY FUNCTIONS ===
2266
+ /** Watch with transformation */
2267
+ transform: useWatchTransform,
2268
+ /** Watch with default value */
2269
+ withDefault: useWatchDefault,
2270
+ /** Watch as boolean */
2271
+ boolean: useWatchBoolean,
2272
+ /** Watch multiple with custom keys */
2273
+ batch: useWatchBatch,
2274
+ /** Watch conditionally */
2275
+ conditional: useWatchConditional,
2276
+ /** Watch with debouncing */
2277
+ debounced: useWatchDebounced,
2278
+ /** Watch with selector */
2279
+ selector: useWatchSelector
2280
+ };
2281
+
2282
+ // src/core/utils/calculateFilterCount.ts
2283
+ var calculateFilterCount = (model) => Object.values(model).filter(
2284
+ (v) => v !== null && v !== void 0 && String(v).trim() !== ""
2285
+ ).length;
2286
+
2287
+ // src/core/utils/format-time.ts
2288
+ import dayjs from "dayjs";
2289
+ import duration from "dayjs/plugin/duration";
2290
+ import relativeTime from "dayjs/plugin/relativeTime";
2291
+ dayjs.extend(duration);
2292
+ dayjs.extend(relativeTime);
2293
+ var formatPatterns = {
2294
+ dateTime: "DD MMM YYYY h:mm A",
2295
+ // 17 Apr 2022 12:00 am
2296
+ date: "DD MMM YYYY",
2297
+ // 17 Apr 2022
2298
+ month_year_short_format: "MMM YYYY",
2299
+ month_year_full_format: "MMMM YYYY",
2300
+ year: "YYYY",
2301
+ time: "h:mm a",
2302
+ // 12:00 am
2303
+ split: {
2304
+ dateTime: "DD/MM/YYYY h:mm A",
2305
+ // 17/04/2022 12:00 am
2306
+ date: "DD/MM/YYYY"
2307
+ // 17/04/2022
2308
+ },
2309
+ paramCase: {
2310
+ dateTime: "DD-MM-YYYY h:mm A",
2311
+ // 17-04-2022 12:00 am
2312
+ date: "DD-MM-YYYY",
2313
+ // 17-04-2022
2314
+ dateReverse: "YYYY-MM-DD",
2315
+ // 2022-04-17 for compare date
2316
+ MonthYear: "MMM-YYYY"
2317
+ }
2318
+ };
2319
+ var isValidDate = (date) => date !== null && date !== void 0 && dayjs(date).isValid();
2320
+ function today(template) {
2321
+ return dayjs(/* @__PURE__ */ new Date()).startOf("day").format(template);
2322
+ }
2323
+ function fDateTime(date, template) {
2324
+ if (!isValidDate(date)) {
2325
+ return "Invalid date";
2326
+ }
2327
+ return dayjs(date).format(template ?? formatPatterns.dateTime);
2328
+ }
2329
+ function fDate(date, template) {
2330
+ if (!isValidDate(date)) {
2331
+ return "Invalid date";
2332
+ }
2333
+ return dayjs(date).format(template ?? formatPatterns.date);
2334
+ }
2335
+ function fTime(date, template) {
2336
+ if (!isValidDate(date)) {
2337
+ return "Invalid date";
2338
+ }
2339
+ return dayjs(date).format(template ?? formatPatterns.time);
2340
+ }
2341
+ function fTimestamp(date) {
2342
+ if (!isValidDate(date)) {
2343
+ return "Invalid date";
2344
+ }
2345
+ return dayjs(date).valueOf();
2346
+ }
2347
+ function fToNow(date) {
2348
+ if (!isValidDate(date)) {
2349
+ return "Invalid date";
2350
+ }
2351
+ return dayjs(date).toNow(true);
2352
+ }
2353
+ function fIsBetween(inputDate, startDate, endDate) {
2354
+ if (!isValidDate(inputDate) || !isValidDate(startDate) || !isValidDate(endDate)) {
2355
+ return false;
2356
+ }
2357
+ const formattedInputDate = fTimestamp(inputDate);
2358
+ const formattedStartDate = fTimestamp(startDate);
2359
+ const formattedEndDate = fTimestamp(endDate);
2360
+ if (formattedInputDate === "Invalid date" || formattedStartDate === "Invalid date" || formattedEndDate === "Invalid date") {
2361
+ return false;
2362
+ }
2363
+ return formattedInputDate >= formattedStartDate && formattedInputDate <= formattedEndDate;
2364
+ }
2365
+ function fIsAfter(startDate, endDate) {
2366
+ if (!isValidDate(startDate) || !isValidDate(endDate)) {
2367
+ return false;
2368
+ }
2369
+ return dayjs(startDate).isAfter(endDate);
2370
+ }
2371
+ function fIsSame(startDate, endDate, unitToCompare) {
2372
+ if (!isValidDate(startDate) || !isValidDate(endDate)) {
2373
+ return false;
2374
+ }
2375
+ return dayjs(startDate).isSame(endDate, unitToCompare ?? "year");
2376
+ }
2377
+ function fDateRangeShortLabel(startDate, endDate, initial) {
2378
+ if (!isValidDate(startDate) || !isValidDate(endDate) || fIsAfter(startDate, endDate)) {
2379
+ return "Invalid date";
2380
+ }
2381
+ let label = `${fDate(startDate)} - ${fDate(endDate)}`;
2382
+ if (initial) {
2383
+ return label;
2384
+ }
2385
+ const isSameYear = fIsSame(startDate, endDate, "year");
2386
+ const isSameMonth = fIsSame(startDate, endDate, "month");
2387
+ const isSameDay = fIsSame(startDate, endDate, "day");
2388
+ if (isSameYear && !isSameMonth) {
2389
+ label = `${fDate(startDate, "DD MMM")} - ${fDate(endDate)}`;
2390
+ } else if (isSameYear && isSameMonth && !isSameDay) {
2391
+ label = `${fDate(startDate, "DD")} - ${fDate(endDate)}`;
2392
+ } else if (isSameYear && isSameMonth && isSameDay) {
2393
+ label = `${fDate(endDate)}`;
2394
+ }
2395
+ return label;
2396
+ }
2397
+ function fAdd({
2398
+ years = 0,
2399
+ months = 0,
2400
+ days = 0,
2401
+ hours = 0,
2402
+ minutes = 0,
2403
+ seconds = 0,
2404
+ milliseconds = 0
2405
+ }) {
2406
+ const result = dayjs().add(
2407
+ dayjs.duration({
2408
+ years,
2409
+ months,
2410
+ days,
2411
+ hours,
2412
+ minutes,
2413
+ seconds,
2414
+ milliseconds
2415
+ })
2416
+ ).format();
2417
+ return result;
2418
+ }
2419
+ function fSub({
2420
+ years = 0,
2421
+ months = 0,
2422
+ days = 0,
2423
+ hours = 0,
2424
+ minutes = 0,
2425
+ seconds = 0,
2426
+ milliseconds = 0
2427
+ }) {
2428
+ const result = dayjs().subtract(
2429
+ dayjs.duration({
2430
+ years,
2431
+ months,
2432
+ days,
2433
+ hours,
2434
+ minutes,
2435
+ seconds,
2436
+ milliseconds
2437
+ })
2438
+ ).format();
2439
+ return result;
2440
+ }
2441
+
2442
+ // src/core/utils/getEmptyObject.ts
2443
+ function getEmptyObject(data, defaultValues = {}) {
2444
+ const obj = {};
2445
+ for (const key of Object.keys(data)) {
2446
+ const value = data[key];
2447
+ const type = typeof value;
2448
+ if (type === "number") {
2449
+ obj[key] = 0;
2450
+ } else if (type === "string" || type === "boolean") {
2451
+ obj[key] = null;
2452
+ } else if (value instanceof Date) {
2453
+ obj[key] = null;
2454
+ } else {
2455
+ obj[key] = null;
2456
+ }
2457
+ }
2458
+ return { ...obj, ...defaultValues };
2459
+ }
2460
+
2461
+ // src/core/utils/useStableRowCount.ts
2462
+ import { useRef, useMemo as useMemo6 } from "react";
2463
+ function useStableRowCount(currentTotal) {
2464
+ const rowCountRef = useRef(currentTotal || 0);
2465
+ const stableRowCount = useMemo6(() => {
2466
+ if (currentTotal !== void 0) {
2467
+ rowCountRef.current = currentTotal;
2468
+ }
2469
+ return rowCountRef.current;
2470
+ }, [currentTotal]);
2471
+ return stableRowCount;
2472
+ }
2473
+
2474
+ export {
2475
+ generateCorrelationId,
2476
+ RequestManager,
2477
+ ApiClient,
2478
+ createApiClient,
2479
+ getGlobalApiClient,
2480
+ initializeGlobalApiClient,
2481
+ setGlobalApiClient,
2482
+ resetGlobalApiClient,
2483
+ CancelToken,
2484
+ useValidationErrors,
2485
+ AuthorizedView,
2486
+ CancelButton,
2487
+ ClearButton,
2488
+ SimpleContainer,
2489
+ FilterButton,
2490
+ FilterChip,
2491
+ ProgramsFilterDisplay,
2492
+ FilterWrapper,
2493
+ Footer,
2494
+ LabelText,
2495
+ RenderIf,
2496
+ SectionBox,
2497
+ SimpleTabs,
2498
+ SubmitButton,
2499
+ withDataModal,
2500
+ Config,
2501
+ dateTimePatterns,
2502
+ useApiClient,
2503
+ useFormErrorHandler,
2504
+ useDeleteHandler,
2505
+ CacheUtility,
2506
+ useCacheUtility,
2507
+ useWatchForm,
2508
+ useWatchField,
2509
+ useWatchFields,
2510
+ useWatchTransform,
2511
+ useWatchDefault,
2512
+ useWatchBoolean,
2513
+ useWatchBatch,
2514
+ useWatchConditional,
2515
+ useWatchDebounced,
2516
+ useWatchSelector,
2517
+ typedWatch,
2518
+ calculateFilterCount,
2519
+ formatPatterns,
2520
+ today,
2521
+ fDateTime,
2522
+ fDate,
2523
+ fTime,
2524
+ fTimestamp,
2525
+ fToNow,
2526
+ fIsBetween,
2527
+ fIsAfter,
2528
+ fIsSame,
2529
+ fDateRangeShortLabel,
2530
+ fAdd,
2531
+ fSub,
2532
+ getEmptyObject,
2533
+ useStableRowCount
2534
+ };
2535
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vc3JjL2NvcmUvYXBpL0NvcnJlbGF0aW9uSWRHZW5lcmF0b3IudHMiLCAiLi4vc3JjL2NvcmUvYXBpL0Vycm9ycy9FcnJvck5vcm1hbGl6ZXIudHMiLCAiLi4vc3JjL2NvcmUvYXBpL0ludGVyY2VwdG9ycy9JbnRlcmNlcHRvck1hbmFnZXIudHMiLCAiLi4vc3JjL2NvcmUvYXBpL1JlcXVlc3RNYW5hZ2VyLnRzIiwgIi4uL3NyYy9jb3JlL2FwaS9SZXRyeS9SZXRyeUhhbmRsZXIudHMiLCAiLi4vc3JjL2NvcmUvYXBpL1NpZ25hbHMvU2lnbmFsTWFuYWdlci50cyIsICIuLi9zcmMvY29yZS9hcGkvVXRpbHMvUmVzcG9uc2VQYXJzZXIudHMiLCAiLi4vc3JjL2NvcmUvYXBpL1V0aWxzL1VybEJ1aWxkZXIudHMiLCAiLi4vc3JjL2NvcmUvYXBpL0FwaUNsaWVudC50cyIsICIuLi9zcmMvY29yZS9hcGkvY3JlYXRlQXBpQ2xpZW50LnRzIiwgIi4uL3NyYy9jb3JlL2FwaS90eXBlcy9DYW5jZWxUb2tlbi50cyIsICIuLi9zcmMvY29yZS9hcGkvdXNlVmFsaWRhdGlvbkVycm9ycy50cyIsICIuLi9zcmMvY29yZS9jb21wb25lbnRzL0F1dGhvcml6ZWRWaWV3L0F1dGhvcml6ZWRWaWV3LnRzeCIsICIuLi9zcmMvY29yZS9jb21wb25lbnRzL0NhbmNlbEJ1dHRvbi9DYW5jZWxCdXR0b24udHN4IiwgIi4uL3NyYy9jb3JlL2NvbXBvbmVudHMvQ2xlYXJCdXR0b24vQ2xlYXJCdXR0b24udHN4IiwgIi4uL3NyYy9jb3JlL2NvbXBvbmVudHMvQ29udGFpbmVycy9TaW1wbGVDb250YWluZXIudHN4IiwgIi4uL3NyYy9jb3JlL2NvbXBvbmVudHMvRmlsdGVyQnV0dG9uL0ZpbHRlckJ1dHRvbi50c3giLCAiLi4vc3JjL2NvcmUvY29tcG9uZW50cy9GaWx0ZXJEaXNwbGF5L0ZpbHRlckNoaXAudHN4IiwgIi4uL3NyYy9jb3JlL2NvbXBvbmVudHMvRmlsdGVyRGlzcGxheS9GaWx0ZXJEaXNwbGF5LnRzeCIsICIuLi9zcmMvY29yZS9jb21wb25lbnRzL0ZpbHRlcldyYXBwZXIvRmlsdGVyV3JhcHBlci50c3giLCAiLi4vc3JjL2NvcmUvY29tcG9uZW50cy9Gb290ZXIvRm9vdGVyLnRzeCIsICIuLi9zcmMvY29yZS9jb21wb25lbnRzL0xhYmVsVGV4dC9MYWJlbFRleHQudHN4IiwgIi4uL3NyYy9jb3JlL2NvbXBvbmVudHMvUmVuZGVySWYvUmVuZGVySWYudHN4IiwgIi4uL3NyYy9jb3JlL2NvbXBvbmVudHMvU2VjdGlvbkJveC9TZWN0aW9uQm94LnRzeCIsICIuLi9zcmMvY29yZS9jb21wb25lbnRzL1NpbXBsZVRhYnMvU2ltcGxlVGFicy50c3giLCAiLi4vc3JjL2NvcmUvY29tcG9uZW50cy9TdWJtaXRCdXR0b24vU3VibWl0QnV0dG9uLnRzeCIsICIuLi9zcmMvY29yZS9jb21wb25lbnRzL1dpdGhSZWYvV2l0aFJlZi50c3giLCAiLi4vc3JjL2NvcmUvY29uZmlnLnRzIiwgIi4uL3NyYy9jb3JlL2hvb2tzL3VzZUFwaUNsaWVudC50cyIsICIuLi9zcmMvY29yZS9ob29rcy91c2VGb3JtRXJyb3JIYW5kbGVyLnRzIiwgIi4uL3NyYy9jb3JlL3V0aWxzL0NhY2hlVXRpbGl0eS9pbmRleC50cyIsICIuLi9zcmMvY29yZS91dGlscy93YXRjaC9jb3JlLnRzIiwgIi4uL3NyYy9jb3JlL3V0aWxzL3dhdGNoL3V0aWxpdGllcy50cyIsICIuLi9zcmMvY29yZS91dGlscy93YXRjaC9pbmRleC50cyIsICIuLi9zcmMvY29yZS91dGlscy9jYWxjdWxhdGVGaWx0ZXJDb3VudC50cyIsICIuLi9zcmMvY29yZS91dGlscy9mb3JtYXQtdGltZS50cyIsICIuLi9zcmMvY29yZS91dGlscy9nZXRFbXB0eU9iamVjdC50cyIsICIuLi9zcmMvY29yZS91dGlscy91c2VTdGFibGVSb3dDb3VudC50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyoqXG4gKiBAZmlsZW92ZXJ2aWV3IFV0aWxpdGllcyBmb3IgZ2VuZXJhdGluZyBjb3JyZWxhdGlvbiBJRHMgZm9yIGRpc3RyaWJ1dGVkIHJlcXVlc3QgdHJhY2luZy5cbiAqXG4gKiBDb3JyZWxhdGlvbiBJRHMgYXJlIHVuaXF1ZSBpZGVudGlmaWVycyBhdHRhY2hlZCB0byBlYWNoIEFQSSByZXF1ZXN0IHRvIGVuYWJsZVxuICogdHJhY2tpbmcgYW5kIGRlYnVnZ2luZyBhY3Jvc3MgZGlzdHJpYnV0ZWQgc3lzdGVtcywgbWljcm9zZXJ2aWNlcywgYW5kIGxvZyBhZ2dyZWdhdGlvbi5cbiAqXG4gKiBAbW9kdWxlIENvcnJlbGF0aW9uSWRHZW5lcmF0b3JcbiAqL1xuXG4vKipcbiAqIEdlbmVyYXRlcyBhIGNyeXB0b2dyYXBoaWNhbGx5IHJhbmRvbSBVVUlEIHY0IHN0cmluZy5cbiAqXG4gKiBQcmVmZXJzIG5hdGl2ZSBgY3J5cHRvLnJhbmRvbVVVSUQoKWAgd2hlbiBhdmFpbGFibGUgKG1vZGVybiBicm93c2VycyBhbmQgTm9kZS5qcyAxOSspLFxuICogZmFsbGluZyBiYWNrIHRvIGEgY29tcGxpYW50IFVVSUQgdjQgaW1wbGVtZW50YXRpb24gdXNpbmcgYE1hdGgucmFuZG9tKClgLlxuICpcbiAqIEBpbnRlcm5hbCBUaGlzIGZ1bmN0aW9uIGlzIG5vdCBleHBvcnRlZCBmcm9tIHRoZSBwdWJsaWMgQVBJXG4gKiBAcmV0dXJucyBBIFVVSUQgdjQgc3RyaW5nIGluIHRoZSBmb3JtYXQgYHh4eHh4eHh4LXh4eHgtNHh4eC15eHh4LXh4eHh4eHh4eHh4eGBcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogY29uc3QgdXVpZCA9IGdlbmVyYXRlVVVJRCgpO1xuICogLy8gPT4gXCJhMWIyYzNkNC1lNWY2LTQ3ODktYTAxMi1iM2M0ZDVlNmY3YThcIlxuICogYGBgXG4gKi9cbmZ1bmN0aW9uIGdlbmVyYXRlVVVJRCgpOiBzdHJpbmcge1xuICBpZiAodHlwZW9mIGNyeXB0byAhPT0gJ3VuZGVmaW5lZCcgJiYgY3J5cHRvLnJhbmRvbVVVSUQpIHtcbiAgICByZXR1cm4gY3J5cHRvLnJhbmRvbVVVSUQoKTtcbiAgfVxuXG4gIC8vIEZhbGxiYWNrIFVVSUQgdjQgZ2VuZXJhdG9yXG4gIHJldHVybiAneHh4eHh4eHgteHh4eC00eHh4LXl4eHgteHh4eHh4eHh4eHh4Jy5yZXBsYWNlKC9beHldL2csIGMgPT4ge1xuICAgIGNvbnN0IHIgPSAoTWF0aC5yYW5kb20oKSAqIDE2KSB8IDA7XG4gICAgY29uc3QgdiA9IGMgPT09ICd4JyA/IHIgOiAociAmIDB4MykgfCAweDg7XG5cbiAgICByZXR1cm4gdi50b1N0cmluZygxNik7XG4gIH0pO1xufVxuXG4vKipcbiAqIEdlbmVyYXRlcyBhIHVuaXF1ZSBjb3JyZWxhdGlvbiBJRCBmb3IgQVBJIHJlcXVlc3QgdHJhY2tpbmcgYW5kIGRpc3RyaWJ1dGVkIHRyYWNpbmcuXG4gKlxuICogQ29ycmVsYXRpb24gSURzIGhlbHAgdHJhY2sgcmVxdWVzdHMgYWNyb3NzIG11bHRpcGxlIHNlcnZpY2VzIGFuZCBzeXN0ZW1zLiBUaGV5IGFyZVxuICogYXV0b21hdGljYWxseSBhdHRhY2hlZCB0byBvdXRnb2luZyByZXF1ZXN0cyB2aWEgSFRUUCBoZWFkZXJzIChgWC1Db3JyZWxhdGlvbi1JZGAgYW5kXG4gKiBgWC1SZXF1ZXN0LUlkYCkgYW5kIGNhbiBiZSB1c2VkIHRvOlxuICogLSBDb3JyZWxhdGUgbG9ncyBhY3Jvc3MgZGlzdHJpYnV0ZWQgc3lzdGVtc1xuICogLSBUcmFjayByZXF1ZXN0IGZsb3dzIGluIG1pY3Jvc2VydmljZXMgYXJjaGl0ZWN0dXJlc1xuICogLSBEZWJ1ZyBpc3N1ZXMgYnkgdHJhY2luZyBhIHNwZWNpZmljIHJlcXVlc3QncyBqb3VybmV5XG4gKiAtIE1vbml0b3IgcGVyZm9ybWFuY2UgYWNyb3NzIHNlcnZpY2UgYm91bmRhcmllc1xuICpcbiAqIFRoZSBnZW5lcmF0ZWQgSUQgZm9sbG93cyB0aGUgZm9ybWF0OiBge3ByZWZpeH0te3V1aWR9YCBvciBqdXN0IGB7dXVpZH1gIGlmIG5vIHByZWZpeCBpcyBwcm92aWRlZC5cbiAqXG4gKiBAcGFyYW0gcHJlZml4IC0gT3B0aW9uYWwgcHJlZml4IHRvIG5hbWVzcGFjZSBjb3JyZWxhdGlvbiBJRHMgYnkgY29udGV4dCAoZS5nLiwgJ2FwaScsICd3ZWInLCAnbW9iaWxlJylcbiAqIEByZXR1cm5zIEEgY29ycmVsYXRpb24gSUQgc3RyaW5nLCBvcHRpb25hbGx5IHByZWZpeGVkXG4gKiBAcHVibGljXG4gKlxuICogQGV4YW1wbGVcbiAqIEJhc2ljIHVzYWdlOlxuICogYGBgdHlwZXNjcmlwdFxuICogY29uc3QgaWQgPSBnZW5lcmF0ZUNvcnJlbGF0aW9uSWQoKTtcbiAqIC8vID0+IFwiYTFiMmMzZDQtZTVmNi00Nzg5LWEwMTItYjNjNGQ1ZTZmN2E4XCJcbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBXaXRoIHByZWZpeCBmb3IgY29udGV4dCBpZGVudGlmaWNhdGlvbjpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGNvbnN0IHdlYklkID0gZ2VuZXJhdGVDb3JyZWxhdGlvbklkKCd3ZWInKTtcbiAqIC8vID0+IFwid2ViLWExYjJjM2Q0LWU1ZjYtNDc4OS1hMDEyLWIzYzRkNWU2ZjdhOFwiXG4gKlxuICogY29uc3QgYXBpSWQgPSBnZW5lcmF0ZUNvcnJlbGF0aW9uSWQoJ2FwaScpO1xuICogLy8gPT4gXCJhcGktZjFlMmQzYzQtYjVhNi00OTg3LWEwMTItYjNjNGQ1ZTZmN2E4XCJcbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBDb25maWd1cmluZyBpbiBBcGlDbGllbnQ6XG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBjbGllbnQgPSBuZXcgQXBpQ2xpZW50KCdodHRwczovL2FwaS5leGFtcGxlLmNvbScpO1xuICogY2xpZW50LnNldENvcnJlbGF0aW9uSWRQcmVmaXgoJ21vYmlsZScpO1xuICogLy8gQWxsIHJlcXVlc3RzIHdpbGwgaGF2ZSBjb3JyZWxhdGlvbiBJRHMgbGlrZSBcIm1vYmlsZS17dXVpZH1cIlxuICogYGBgXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZW5lcmF0ZUNvcnJlbGF0aW9uSWQocHJlZml4Pzogc3RyaW5nKTogc3RyaW5nIHtcbiAgY29uc3QgdXVpZCA9IGdlbmVyYXRlVVVJRCgpO1xuXG4gIHJldHVybiBwcmVmaXggPyBgJHtwcmVmaXh9LSR7dXVpZH1gIDogdXVpZDtcbn1cbiIsICJpbXBvcnQgdHlwZSB7IEFwaUVycm9yLCBBcGlFcnJvclR5cGUgfSBmcm9tICcuLi90eXBlcy9BcGlFcnJvcic7XG5pbXBvcnQgdHlwZSB7IFJlcXVlc3RDb25maWcgfSBmcm9tICcuLi90eXBlcy9SZXF1ZXN0Q29uZmlnJztcblxuLyoqXG4gKiBOb3JtYWxpemVzIHZhcmlvdXMgZXJyb3IgdHlwZXMgaW50byBhIGNvbnNpc3RlbnQsIHN0cnVjdHVyZWQgQXBpRXJyb3IgZm9ybWF0LlxuICpcbiAqIFRoaXMgaW50ZXJuYWwgY2xhc3MgaGFuZGxlcyB0aGUgY29tcGxleCB0YXNrIG9mIGNvbnZlcnRpbmcgYW55IHR5cGUgb2YgZXJyb3JcbiAqIChuZXR3b3JrIGVycm9ycywgSFRUUCBlcnJvcnMsIGFib3J0IGVycm9ycywgdGltZW91dHMsIGV0Yy4pIGludG8gYSBzdGFuZGFyZGl6ZWRcbiAqIHtAbGluayBBcGlFcnJvcn0gc3RydWN0dXJlIHdpdGggcHJvcGVyIGNhdGVnb3JpemF0aW9uIGFuZCBtZXRhZGF0YS5cbiAqXG4gKiBLZXkgRmVhdHVyZXM6XG4gKiAtICoqRXJyb3IgQ2F0ZWdvcml6YXRpb24qKjogQ2xhc3NpZmllcyBlcnJvcnMgYnkgdHlwZSAodmFsaWRhdGlvbiwgY2xpZW50LCBzZXJ2ZXIsIG5ldHdvcmssIGV0Yy4pXG4gKiAtICoqSFRUUCBTdGF0dXMgTWFwcGluZyoqOiBDb252ZXJ0cyBzdGF0dXMgY29kZXMgdG8gaHVtYW4tcmVhZGFibGUgdGl0bGVzXG4gKiAtICoqQ29ycmVsYXRpb24gSUQgVHJhY2tpbmcqKjogUHJlc2VydmVzIG9yIGluamVjdHMgY29ycmVsYXRpb24gSURzIGZvciB0cmFjaW5nXG4gKiAtICoqQWJvcnQgRGV0ZWN0aW9uKio6IElkZW50aWZpZXMgYW5kIHByb3Blcmx5IGhhbmRsZXMgcmVxdWVzdCBjYW5jZWxsYXRpb25zXG4gKiAtICoqQ29uc2lzdGVudCBTdHJ1Y3R1cmUqKjogRW5zdXJlcyBhbGwgZXJyb3JzIGNvbmZvcm0gdG8gdGhlIEFwaUVycm9yIGludGVyZmFjZVxuICpcbiAqIEBpbnRlcm5hbCBUaGlzIGNsYXNzIGlzIG5vdCBleHBvcnRlZCBmcm9tIHRoZSBwdWJsaWMgQVBJXG4gKi9cbmV4cG9ydCBjbGFzcyBFcnJvck5vcm1hbGl6ZXIge1xuICAvKipcbiAgICogTWFwcyBhbiBIVFRQIHN0YXR1cyBjb2RlIHRvIGEgc3RhbmRhcmRpemVkIGVycm9yIHR5cGUgY2F0ZWdvcnkuXG4gICAqXG4gICAqIFRoaXMgY2F0ZWdvcml6YXRpb24gaGVscHMgY29uc3VtZXJzIGhhbmRsZSBkaWZmZXJlbnQgZXJyb3IgY2xhc3NlcyBhcHByb3ByaWF0ZWx5OlxuICAgKiAtIGB2YWxpZGF0aW9uX2Vycm9yYCAoNDAwKTogQ2xpZW50IHNlbnQgaW52YWxpZCBkYXRhXG4gICAqIC0gYGNsaWVudF9lcnJvcmAgKDQwMS00OTkpOiBDbGllbnQtc2lkZSBpc3N1ZXMgKGF1dGgsIHBlcm1pc3Npb25zLCBub3QgZm91bmQsIGV0Yy4pXG4gICAqIC0gYHNlcnZlcl9lcnJvcmAgKDUwMC01OTkpOiBTZXJ2ZXItc2lkZSBmYWlsdXJlc1xuICAgKiAtIGB1bmtub3duX2Vycm9yYDogVW5yZWNvZ25pemVkIHN0YXR1cyBjb2Rlc1xuICAgKlxuICAgKiBAcGFyYW0gc3RhdHVzIC0gSFRUUCBzdGF0dXMgY29kZSBmcm9tIHRoZSByZXNwb25zZVxuICAgKiBAcmV0dXJucyBUaGUgZXJyb3IgdHlwZSBjYXRlZ29yeSBhcyBhIHN0cmluZ1xuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIG5vcm1hbGl6ZXIuZ2V0RXJyb3JUeXBlKDQwMCk7IC8vID0+ICd2YWxpZGF0aW9uX2Vycm9yJ1xuICAgKiBub3JtYWxpemVyLmdldEVycm9yVHlwZSg0MDQpOyAvLyA9PiAnY2xpZW50X2Vycm9yJ1xuICAgKiBub3JtYWxpemVyLmdldEVycm9yVHlwZSg1MDApOyAvLyA9PiAnc2VydmVyX2Vycm9yJ1xuICAgKiBub3JtYWxpemVyLmdldEVycm9yVHlwZSgwKTsgICAvLyA9PiAndW5rbm93bl9lcnJvcidcbiAgICogYGBgXG4gICAqL1xuICBnZXRFcnJvclR5cGUoc3RhdHVzOiBudW1iZXIpOiBBcGlFcnJvclR5cGUge1xuICAgIGlmIChzdGF0dXMgPj0gNDAwICYmIHN0YXR1cyA8IDUwMCkge1xuICAgICAgcmV0dXJuIHN0YXR1cyA9PT0gNDAwID8gJ3ZhbGlkYXRpb25fZXJyb3InIDogJ2NsaWVudF9lcnJvcic7XG4gICAgfSBlbHNlIGlmIChzdGF0dXMgPj0gNTAwKSB7XG4gICAgICByZXR1cm4gJ3NlcnZlcl9lcnJvcic7XG4gICAgfVxuXG4gICAgcmV0dXJuICd1bmtub3duX2Vycm9yJztcbiAgfVxuXG4gIC8qKlxuICAgKiBNYXBzIGFuIEhUVFAgc3RhdHVzIGNvZGUgdG8gYSBodW1hbi1yZWFkYWJsZSBlcnJvciB0aXRsZS5cbiAgICpcbiAgICogUHJvdmlkZXMgdXNlci1mcmllbmRseSBlcnJvciBtZXNzYWdlcyBmb3IgY29tbW9uIEhUVFAgc3RhdHVzIGNvZGVzLlxuICAgKiBGYWxscyBiYWNrIHRvIGEgZ2VuZXJpYyBcIkhUVFAgRXJyb3Ige3N0YXR1c31cIiBmb3JtYXQgZm9yIHVubWFwcGVkIGNvZGVzLlxuICAgKlxuICAgKiBAcGFyYW0gc3RhdHVzIC0gSFRUUCBzdGF0dXMgY29kZSBmcm9tIHRoZSByZXNwb25zZVxuICAgKiBAcmV0dXJucyBBIGh1bWFuLXJlYWRhYmxlIGVycm9yIHRpdGxlXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogbm9ybWFsaXplci5nZXRFcnJvclRpdGxlKDQwNCk7IC8vID0+ICdOb3QgRm91bmQnXG4gICAqIG5vcm1hbGl6ZXIuZ2V0RXJyb3JUaXRsZSg1MDApOyAvLyA9PiAnSW50ZXJuYWwgU2VydmVyIEVycm9yJ1xuICAgKiBub3JtYWxpemVyLmdldEVycm9yVGl0bGUoOTk5KTsgLy8gPT4gJ0hUVFAgRXJyb3IgOTk5J1xuICAgKiBgYGBcbiAgICovXG4gIGdldEVycm9yVGl0bGUoc3RhdHVzOiBudW1iZXIpOiBzdHJpbmcge1xuICAgIGNvbnN0IHRpdGxlczogUmVjb3JkPG51bWJlciwgc3RyaW5nPiA9IHtcbiAgICAgIDQwMDogJ0JhZCBSZXF1ZXN0JyxcbiAgICAgIDQwMTogJ1VuYXV0aG9yaXplZCcsXG4gICAgICA0MDM6ICdGb3JiaWRkZW4nLFxuICAgICAgNDA0OiAnTm90IEZvdW5kJyxcbiAgICAgIDQwNTogJ01ldGhvZCBOb3QgQWxsb3dlZCcsXG4gICAgICA0MDg6ICdSZXF1ZXN0IFRpbWVvdXQnLFxuICAgICAgNDA5OiAnQ29uZmxpY3QnLFxuICAgICAgNDIyOiAnVW5wcm9jZXNzYWJsZSBFbnRpdHknLFxuICAgICAgNDI5OiAnVG9vIE1hbnkgUmVxdWVzdHMnLFxuICAgICAgNTAwOiAnSW50ZXJuYWwgU2VydmVyIEVycm9yJyxcbiAgICAgIDUwMjogJ0JhZCBHYXRld2F5JyxcbiAgICAgIDUwMzogJ1NlcnZpY2UgVW5hdmFpbGFibGUnLFxuICAgICAgNTA0OiAnR2F0ZXdheSBUaW1lb3V0JyxcbiAgICB9O1xuXG4gICAgcmV0dXJuIHRpdGxlc1tzdGF0dXNdIHx8IGBIVFRQIEVycm9yICR7c3RhdHVzfWA7XG4gIH1cblxuICAvKipcbiAgICogTm9ybWFsaXplcyBhbnkgZXJyb3IgaW50byBhIGNvbnNpc3RlbnQsIHN0cnVjdHVyZWQgQXBpRXJyb3IgZm9ybWF0LlxuICAgKlxuICAgKiBUaGlzIG1ldGhvZCBoYW5kbGVzIHZhcmlvdXMgZXJyb3Igc2NlbmFyaW9zIGFuZCBlbnN1cmVzIHRoZXkgYWxsIGNvbmZvcm0gdG9cbiAgICogdGhlIHtAbGluayBBcGlFcnJvcn0gaW50ZXJmYWNlIHdpdGggYXBwcm9wcmlhdGUgY2F0ZWdvcml6YXRpb24gYW5kIG1ldGFkYXRhOlxuICAgKlxuICAgKiAtICoqRXhpc3RpbmcgQXBpRXJyb3JzKio6IEVuaGFuY2VzIHdpdGggbWlzc2luZyBmaWVsZHMgKHRyYWNlSWQsIGNvbmZpZylcbiAgICogLSAqKkFib3J0RXJyb3JzKio6IE1hcmtzIGFzIGByZXF1ZXN0X2NhbmNlbGxlZGAgd2l0aCBpc0Fib3J0ZWQgZmxhZ1xuICAgKiAtICoqVGltZW91dCBFcnJvcnMqKjogQ2F0ZWdvcml6ZXMgYXMgYHRpbWVvdXRfZXJyb3JgIHdpdGggNDA4IHN0YXR1c1xuICAgKiAtICoqTmV0d29yayBFcnJvcnMqKjogQ2F0ZWdvcml6ZXMgYXMgYG5ldHdvcmtfZXJyb3JgIHdpdGggMCBzdGF0dXNcbiAgICogLSAqKlVua25vd24gRXJyb3JzKio6IEZhbGxiYWNrIGNhdGVnb3J5IGZvciB1bmV4cGVjdGVkIGVycm9yIHR5cGVzXG4gICAqXG4gICAqIEFsbCBub3JtYWxpemVkIGVycm9ycyBpbmNsdWRlOlxuICAgKiAtIGB0eXBlYDogRXJyb3IgY2F0ZWdvcnkgZm9yIHByb2dyYW1tYXRpYyBoYW5kbGluZ1xuICAgKiAtIGB0aXRsZWA6IEh1bWFuLXJlYWRhYmxlIGVycm9yIHRpdGxlXG4gICAqIC0gYHN0YXR1c2A6IEhUVFAgc3RhdHVzIGNvZGUgKG9yIDAgZm9yIG5vbi1IVFRQIGVycm9ycylcbiAgICogLSBgdHJhY2VJZGA6IENvcnJlbGF0aW9uIElEIGZvciBkaXN0cmlidXRlZCB0cmFjaW5nXG4gICAqIC0gYGlzQWJvcnRlZGA6IEJvb2xlYW4gZmxhZyBpbmRpY2F0aW5nIGlmIHJlcXVlc3Qgd2FzIGNhbmNlbGxlZFxuICAgKiAtIGBjb25maWdgOiBPcmlnaW5hbCByZXF1ZXN0IGNvbmZpZ3VyYXRpb24gZm9yIGRlYnVnZ2luZ1xuICAgKlxuICAgKiBAcGFyYW0gZXJyb3IgLSBUaGUgZXJyb3IgdG8gbm9ybWFsaXplIChjYW4gYmUgYW55IHR5cGUpXG4gICAqIEBwYXJhbSBjb25maWcgLSBUaGUgcmVxdWVzdCBjb25maWd1cmF0aW9uIHRoYXQgbGVkIHRvIHRoaXMgZXJyb3JcbiAgICogQHBhcmFtIGNvcnJlbGF0aW9uSWQgLSBPcHRpb25hbCBjb3JyZWxhdGlvbiBJRCBmb3IgdHJhY2luZ1xuICAgKiBAcmV0dXJucyBBIGZ1bGx5IHN0cnVjdHVyZWQgQXBpRXJyb3IgaW5zdGFuY2VcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogTm9ybWFsaXppbmcgYSBmZXRjaCBBYm9ydEVycm9yOlxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIHRyeSB7XG4gICAqICAgYXdhaXQgZmV0Y2godXJsLCB7IHNpZ25hbCB9KTtcbiAgICogfSBjYXRjaCAoZXJyb3IpIHtcbiAgICogICBjb25zdCBhcGlFcnJvciA9IG5vcm1hbGl6ZXIubm9ybWFsaXplRXJyb3IoZXJyb3IsIGNvbmZpZywgJ3JlcS0xMjMnKTtcbiAgICogICAvLyBhcGlFcnJvci50eXBlID09PSAncmVxdWVzdF9jYW5jZWxsZWQnXG4gICAqICAgLy8gYXBpRXJyb3IuaXNBYm9ydGVkID09PSB0cnVlXG4gICAqIH1cbiAgICogYGBgXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIE5vcm1hbGl6aW5nIGEgdGltZW91dDpcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiBjb25zdCB0aW1lb3V0RXJyb3IgPSBuZXcgRXJyb3IoJ1JlcXVlc3QgdGltZW91dCBhZnRlciAzMDAwMG1zJyk7XG4gICAqIGNvbnN0IGFwaUVycm9yID0gbm9ybWFsaXplci5ub3JtYWxpemVFcnJvcih0aW1lb3V0RXJyb3IsIGNvbmZpZyk7XG4gICAqIC8vIGFwaUVycm9yLnR5cGUgPT09ICd0aW1lb3V0X2Vycm9yJ1xuICAgKiAvLyBhcGlFcnJvci5zdGF0dXMgPT09IDQwOFxuICAgKiBgYGBcbiAgICovXG4gIG5vcm1hbGl6ZUVycm9yKFxuICAgIGVycm9yOiB1bmtub3duLFxuICAgIGNvbmZpZzogUmVxdWVzdENvbmZpZyxcbiAgICBjb3JyZWxhdGlvbklkPzogc3RyaW5nXG4gICk6IEFwaUVycm9yIHtcbiAgICAvLyBIYW5kbGUgcHJpbWl0aXZlIGVycm9ycyAoc3RyaW5ncywgbnVtYmVycywgZXRjLilcbiAgICBpZiAoZXJyb3IgPT09IG51bGwgfHwgZXJyb3IgPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIE9iamVjdC5hc3NpZ24obmV3IEVycm9yKCdBbiB1bmtub3duIGVycm9yIG9jY3VycmVkJyksIHtcbiAgICAgICAgdHlwZTogJ3Vua25vd25fZXJyb3InLFxuICAgICAgICB0aXRsZTogJ1Vua25vd24gRXJyb3InLFxuICAgICAgICBzdGF0dXM6IDAsXG4gICAgICAgIHRyYWNlSWQ6IGNvcnJlbGF0aW9uSWQsXG4gICAgICAgIGlzQWJvcnRlZDogZmFsc2UsXG4gICAgICAgIGNvbmZpZyxcbiAgICAgIH0gYXMgQXBpRXJyb3IpO1xuICAgIH1cblxuICAgIC8vIEhhbmRsZSBzdHJpbmcgZXJyb3JzXG4gICAgaWYgKHR5cGVvZiBlcnJvciA9PT0gJ3N0cmluZycpIHtcbiAgICAgIHJldHVybiBPYmplY3QuYXNzaWduKG5ldyBFcnJvcihlcnJvciksIHtcbiAgICAgICAgdHlwZTogJ3Vua25vd25fZXJyb3InLFxuICAgICAgICB0aXRsZTogJ1Vua25vd24gRXJyb3InLFxuICAgICAgICBzdGF0dXM6IDAsXG4gICAgICAgIHRyYWNlSWQ6IGNvcnJlbGF0aW9uSWQsXG4gICAgICAgIGlzQWJvcnRlZDogZmFsc2UsXG4gICAgICAgIGNvbmZpZyxcbiAgICAgIH0gYXMgQXBpRXJyb3IpO1xuICAgIH1cblxuICAgIGNvbnN0IGVyciA9IGVycm9yIGFzIFBhcnRpYWw8QXBpRXJyb3I+ICYgeyBtZXNzYWdlPzogc3RyaW5nOyBuYW1lPzogc3RyaW5nIH07XG5cbiAgICAvLyBJZiBlcnJvciBhbHJlYWR5IGhhcyBBcGlFcnJvciBzdHJ1Y3R1cmUsIGVuaGFuY2UgaXRcbiAgICBpZiAoZXJyLnR5cGUgfHwgZXJyLnRpdGxlIHx8IGVyci5lcnJvcnMpIHtcbiAgICAgIHJldHVybiBPYmplY3QuYXNzaWduKFxuICAgICAgICBlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IgOiBuZXcgRXJyb3IoZXJyLm1lc3NhZ2UgPz8gJ1Vua25vd24gZXJyb3InKSxcbiAgICAgICAge1xuICAgICAgICAgIHR5cGU6IGVyci50eXBlLFxuICAgICAgICAgIHRpdGxlOiBlcnIudGl0bGUsXG4gICAgICAgICAgc3RhdHVzOiBlcnIuc3RhdHVzLFxuICAgICAgICAgIHRyYWNlSWQ6IGVyci50cmFjZUlkIHx8IGNvcnJlbGF0aW9uSWQsXG4gICAgICAgICAgZXJyb3JzOiBlcnIuZXJyb3JzLFxuICAgICAgICAgIGlzQWJvcnRlZDogZXJyLmlzQWJvcnRlZCB8fCBmYWxzZSxcbiAgICAgICAgICBjb25maWcsXG4gICAgICAgIH1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKGVyci5uYW1lID09PSAnQWJvcnRFcnJvcicgfHwgZXJyLmlzQWJvcnRlZCkge1xuICAgICAgcmV0dXJuIE9iamVjdC5hc3NpZ24obmV3IEVycm9yKGVyci5tZXNzYWdlID8/ICdSZXF1ZXN0IHdhcyBhYm9ydGVkJyksIHtcbiAgICAgICAgdHlwZTogJ3JlcXVlc3RfY2FuY2VsbGVkJyxcbiAgICAgICAgdGl0bGU6ICdSZXF1ZXN0IHdhcyBjYW5jZWxsZWQnLFxuICAgICAgICBzdGF0dXM6IDAsXG4gICAgICAgIHRyYWNlSWQ6IGNvcnJlbGF0aW9uSWQsXG4gICAgICAgIGlzQWJvcnRlZDogdHJ1ZSxcbiAgICAgICAgY29uZmlnLFxuICAgICAgfSBhcyBBcGlFcnJvcik7XG4gICAgfVxuXG4gICAgaWYgKGVyci5tZXNzYWdlPy5pbmNsdWRlcygndGltZW91dCcpKSB7XG4gICAgICByZXR1cm4gT2JqZWN0LmFzc2lnbihuZXcgRXJyb3IoZXJyLm1lc3NhZ2UpLCB7XG4gICAgICAgIHR5cGU6ICd0aW1lb3V0X2Vycm9yJyxcbiAgICAgICAgdGl0bGU6ICdSZXF1ZXN0IFRpbWVvdXQnLFxuICAgICAgICBzdGF0dXM6IDQwOCxcbiAgICAgICAgdHJhY2VJZDogY29ycmVsYXRpb25JZCxcbiAgICAgICAgaXNBYm9ydGVkOiB0cnVlLFxuICAgICAgICBjb25maWcsXG4gICAgICB9IGFzIEFwaUVycm9yKTtcbiAgICB9XG5cbiAgICBpZiAoZXJyLm1lc3NhZ2U/LmluY2x1ZGVzKCduZXR3b3JrJykpIHtcbiAgICAgIHJldHVybiBPYmplY3QuYXNzaWduKG5ldyBFcnJvcihlcnIubWVzc2FnZSA/PyAnTmV0d29yayByZXF1ZXN0IGZhaWxlZCcpLCB7XG4gICAgICAgIHR5cGU6ICduZXR3b3JrX2Vycm9yJyxcbiAgICAgICAgdGl0bGU6ICdOZXR3b3JrIEVycm9yJyxcbiAgICAgICAgc3RhdHVzOiAwLFxuICAgICAgICB0cmFjZUlkOiBjb3JyZWxhdGlvbklkLFxuICAgICAgICBpc0Fib3J0ZWQ6IGZhbHNlLFxuICAgICAgICBjb25maWcsXG4gICAgICB9IGFzIEFwaUVycm9yKTtcbiAgICB9XG5cbiAgICByZXR1cm4gT2JqZWN0LmFzc2lnbihcbiAgICAgIG5ldyBFcnJvcihlcnIubWVzc2FnZSA/PyAnQW4gdW5rbm93biBlcnJvciBvY2N1cnJlZCcpLFxuICAgICAge1xuICAgICAgICB0eXBlOiAndW5rbm93bl9lcnJvcicsXG4gICAgICAgIHRpdGxlOiAnVW5rbm93biBFcnJvcicsXG4gICAgICAgIHN0YXR1czogMCxcbiAgICAgICAgdHJhY2VJZDogY29ycmVsYXRpb25JZCxcbiAgICAgICAgaXNBYm9ydGVkOiBmYWxzZSxcbiAgICAgICAgY29uZmlnLFxuICAgICAgfSBhcyBBcGlFcnJvclxuICAgICk7XG4gIH1cbn1cbiIsICJpbXBvcnQgdHlwZSB7IEFwaUVycm9yIH0gZnJvbSAnLi4vdHlwZXMvQXBpRXJyb3InO1xuaW1wb3J0IHR5cGUgeyBFcnJvckludGVyY2VwdG9yIH0gZnJvbSAnLi4vdHlwZXMvRXJyb3JJbnRlcmNlcHRvcic7XG5pbXBvcnQgdHlwZSB7IFJlcXVlc3RDb25maWcgfSBmcm9tICcuLi90eXBlcy9SZXF1ZXN0Q29uZmlnJztcbmltcG9ydCB0eXBlIHsgUmVxdWVzdEludGVyY2VwdG9yIH0gZnJvbSAnLi4vdHlwZXMvUmVxdWVzdEludGVyY2VwdG9yJztcbmltcG9ydCB0eXBlIHsgUmVzcG9uc2VJbnRlcmNlcHRvciB9IGZyb20gJy4uL3R5cGVzL1Jlc3BvbnNlSW50ZXJjZXB0b3InO1xuXG5pbXBvcnQgdHlwZSB7IEFwaVJlc3BvbnNlIH0gZnJvbSAnQC90eXBlcyc7XG5cbi8qKlxuICogTWFuYWdlcyB0aGUgcmVnaXN0cmF0aW9uIGFuZCBleGVjdXRpb24gb2YgcmVxdWVzdC9yZXNwb25zZS9lcnJvciBpbnRlcmNlcHRvcnMuXG4gKlxuICogVGhpcyBpbnRlcm5hbCBjbGFzcyBpbXBsZW1lbnRzIHRoZSBpbnRlcmNlcHRvciBwYXR0ZXJuLCBhbGxvd2luZyBtaWRkbGV3YXJlLXN0eWxlXG4gKiBmdW5jdGlvbnMgdG8gYmUgcmVnaXN0ZXJlZCBhbmQgZXhlY3V0ZWQgYXQgZGlmZmVyZW50IHBvaW50cyBpbiB0aGUgcmVxdWVzdCBsaWZlY3ljbGU6XG4gKiAtICoqUmVxdWVzdCBJbnRlcmNlcHRvcnMqKjogTW9kaWZ5IHJlcXVlc3RzIGJlZm9yZSB0aGV5IGFyZSBzZW50XG4gKiAtICoqUmVzcG9uc2UgSW50ZXJjZXB0b3JzKio6IFRyYW5zZm9ybSByZXNwb25zZXMgYmVmb3JlIHRoZXkgYXJlIHJldHVybmVkXG4gKiAtICoqRXJyb3IgSW50ZXJjZXB0b3JzKio6IEhhbmRsZSBvciB0cmFuc2Zvcm0gZXJyb3JzIGJlZm9yZSB0aGV5IGFyZSB0aHJvd25cbiAqXG4gKiBJbnRlcmNlcHRvcnMgYXJlIGV4ZWN1dGVkIHNlcXVlbnRpYWxseSBpbiB0aGUgb3JkZXIgdGhleSB3ZXJlIHJlZ2lzdGVyZWQsXG4gKiBmb3JtaW5nIGEgcHJvY2Vzc2luZyBwaXBlbGluZS4gRWFjaCBpbnRlcmNlcHRvciBjYW46XG4gKiAtIEluc3BlY3QgdGhlIGRhdGEgcGFzc2luZyB0aHJvdWdoXG4gKiAtIFRyYW5zZm9ybSBvciBlbmhhbmNlIHRoZSBkYXRhXG4gKiAtIFNob3J0LWNpcmN1aXQgdGhlIHBpcGVsaW5lIChmb3IgZXJyb3JzKVxuICogLSBQZXJmb3JtIHNpZGUgZWZmZWN0cyAobG9nZ2luZywgbW9uaXRvcmluZywgZXRjLilcbiAqXG4gKiBLZXkgRmVhdHVyZXM6XG4gKiAtICoqU2VxdWVudGlhbCBFeGVjdXRpb24qKjogSW50ZXJjZXB0b3JzIHJ1biBpbiByZWdpc3RyYXRpb24gb3JkZXJcbiAqIC0gKipVbnJlZ2lzdHJhdGlvbiBTdXBwb3J0Kio6IFJldHVybnMgY2xlYW51cCBmdW5jdGlvbiBmb3IgZWFjaCBpbnRlcmNlcHRvclxuICogLSAqKlR5cGUgU2FmZXR5Kio6IEZ1bGx5IHR5cGVkIGludGVyY2VwdG9yIGZ1bmN0aW9ucyB3aXRoIGdlbmVyaWNzXG4gKiAtICoqRXJyb3IgSGFuZGxpbmcqKjogR3JhY2VmdWwgaGFuZGxpbmcgb2YgaW50ZXJjZXB0b3IgZmFpbHVyZXNcbiAqXG4gKiBAaW50ZXJuYWwgVGhpcyBjbGFzcyBpcyBub3QgZXhwb3J0ZWQgZnJvbSB0aGUgcHVibGljIEFQSVxuICpcbiAqIEBleGFtcGxlXG4gKiBDb21tb24gaW50ZXJjZXB0b3IgcGF0dGVybnM6XG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBtYW5hZ2VyID0gbmV3IEludGVyY2VwdG9yTWFuYWdlcigpO1xuICpcbiAqIC8vIFJlcXVlc3QgbG9nZ2luZ1xuICogbWFuYWdlci5hZGRSZXF1ZXN0SW50ZXJjZXB0b3IoYXN5bmMgKGNvbmZpZykgPT4ge1xuICogICBjb25zb2xlLmxvZygnUmVxdWVzdDonLCBjb25maWcubWV0aG9kLCBjb25maWcudXJsKTtcbiAqICAgcmV0dXJuIGNvbmZpZztcbiAqIH0pO1xuICpcbiAqIC8vIFJlc3BvbnNlIHRyYW5zZm9ybWF0aW9uXG4gKiBtYW5hZ2VyLmFkZFJlc3BvbnNlSW50ZXJjZXB0b3IoYXN5bmMgKHJlc3BvbnNlKSA9PiB7XG4gKiAgIGlmIChyZXNwb25zZS5hcGlEYXRhKSB7XG4gKiAgICAgcmVzcG9uc2UuYXBpRGF0YSA9IGNhbWVsQ2FzZUtleXMocmVzcG9uc2UuYXBpRGF0YSk7XG4gKiAgIH1cbiAqICAgcmV0dXJuIHJlc3BvbnNlO1xuICogfSk7XG4gKlxuICogLy8gRXJyb3IgbW9uaXRvcmluZ1xuICogbWFuYWdlci5hZGRFcnJvckludGVyY2VwdG9yKGFzeW5jIChlcnJvcikgPT4ge1xuICogICBpZiAoZXJyb3Iuc3RhdHVzID49IDUwMCkge1xuICogICAgIGF3YWl0IGxvZ1RvTW9uaXRvcmluZyhlcnJvcik7XG4gKiAgIH1cbiAqICAgdGhyb3cgZXJyb3I7IC8vIFJlLXRocm93IHRvIG1haW50YWluIGVycm9yIGZsb3dcbiAqIH0pO1xuICogYGBgXG4gKi9cbmV4cG9ydCBjbGFzcyBJbnRlcmNlcHRvck1hbmFnZXIge1xuICAvKipcbiAgICogQXJyYXkgb2YgcmVnaXN0ZXJlZCByZXF1ZXN0IGludGVyY2VwdG9yc1xuICAgKiBAcHJpdmF0ZVxuICAgKi9cbiAgcHJpdmF0ZSByZWFkb25seSByZXF1ZXN0SW50ZXJjZXB0b3JzOiBSZXF1ZXN0SW50ZXJjZXB0b3JbXSA9IFtdO1xuXG4gIC8qKlxuICAgKiBBcnJheSBvZiByZWdpc3RlcmVkIHJlc3BvbnNlIGludGVyY2VwdG9yc1xuICAgKiBAcHJpdmF0ZVxuICAgKi9cbiAgcHJpdmF0ZSByZWFkb25seSByZXNwb25zZUludGVyY2VwdG9yczogUmVzcG9uc2VJbnRlcmNlcHRvcltdID0gW107XG5cbiAgLyoqXG4gICAqIEFycmF5IG9mIHJlZ2lzdGVyZWQgZXJyb3IgaW50ZXJjZXB0b3JzXG4gICAqIEBwcml2YXRlXG4gICAqL1xuICBwcml2YXRlIHJlYWRvbmx5IGVycm9ySW50ZXJjZXB0b3JzOiBFcnJvckludGVyY2VwdG9yW10gPSBbXTtcblxuICAvKipcbiAgICogUmVnaXN0ZXJzIGEgcmVxdWVzdCBpbnRlcmNlcHRvciB0byBtb2RpZnkgcmVxdWVzdHMgYmVmb3JlIHRoZXkgYXJlIHNlbnQuXG4gICAqXG4gICAqIFJlcXVlc3QgaW50ZXJjZXB0b3JzIGNhbjpcbiAgICogLSBBZGQgb3IgbW9kaWZ5IGhlYWRlcnNcbiAgICogLSBUcmFuc2Zvcm0gcmVxdWVzdCBib2RpZXNcbiAgICogLSBBZGQgcXVlcnkgcGFyYW1ldGVyc1xuICAgKiAtIEltcGxlbWVudCByZXF1ZXN0IHNpZ25pbmdcbiAgICogLSBMb2cgb3V0Z29pbmcgcmVxdWVzdHNcbiAgICpcbiAgICogQHBhcmFtIGludGVyY2VwdG9yIC0gQXN5bmMgZnVuY3Rpb24gdGhhdCByZWNlaXZlcyBhbmQgcmV0dXJucyBSZXF1ZXN0Q29uZmlnXG4gICAqIEByZXR1cm5zIENsZWFudXAgZnVuY3Rpb24gdG8gdW5yZWdpc3RlciB0aGlzIGludGVyY2VwdG9yXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogLy8gQWRkIGF1dGhlbnRpY2F0aW9uIGhlYWRlclxuICAgKiBjb25zdCB1bnJlZ2lzdGVyID0gbWFuYWdlci5hZGRSZXF1ZXN0SW50ZXJjZXB0b3IoYXN5bmMgKGNvbmZpZykgPT4ge1xuICAgKiAgIGNvbnN0IHRva2VuID0gYXdhaXQgZ2V0QXV0aFRva2VuKCk7XG4gICAqICAgY29uZmlnLmhlYWRlcnMgPSBjb25maWcuaGVhZGVycyB8fCBuZXcgSGVhZGVycygpO1xuICAgKiAgIGNvbmZpZy5oZWFkZXJzLnNldCgnQXV0aG9yaXphdGlvbicsIGBCZWFyZXIgJHt0b2tlbn1gKTtcbiAgICogICByZXR1cm4gY29uZmlnO1xuICAgKiB9KTtcbiAgICpcbiAgICogLy8gTGF0ZXIsIHJlbW92ZSB0aGUgaW50ZXJjZXB0b3JcbiAgICogdW5yZWdpc3RlcigpO1xuICAgKiBgYGBcbiAgICovXG4gIGFkZFJlcXVlc3RJbnRlcmNlcHRvcihpbnRlcmNlcHRvcjogUmVxdWVzdEludGVyY2VwdG9yKTogKCkgPT4gdm9pZCB7XG4gICAgdGhpcy5yZXF1ZXN0SW50ZXJjZXB0b3JzLnB1c2goaW50ZXJjZXB0b3IpO1xuXG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgIGNvbnN0IGluZGV4ID0gdGhpcy5yZXF1ZXN0SW50ZXJjZXB0b3JzLmluZGV4T2YoaW50ZXJjZXB0b3IpO1xuXG4gICAgICBpZiAoaW5kZXggPiAtMSkgdGhpcy5yZXF1ZXN0SW50ZXJjZXB0b3JzLnNwbGljZShpbmRleCwgMSk7XG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZWdpc3RlcnMgYSByZXNwb25zZSBpbnRlcmNlcHRvciB0byB0cmFuc2Zvcm0gcmVzcG9uc2VzIGJlZm9yZSB0aGV5IGFyZSByZXR1cm5lZC5cbiAgICpcbiAgICogUmVzcG9uc2UgaW50ZXJjZXB0b3JzIGNhbjpcbiAgICogLSBUcmFuc2Zvcm0gcmVzcG9uc2UgZGF0YSBmb3JtYXRcbiAgICogLSBFeHRyYWN0IG5lc3RlZCBkYXRhIHN0cnVjdHVyZXNcbiAgICogLSBBZGQgY29tcHV0ZWQgcHJvcGVydGllc1xuICAgKiAtIENhY2hlIHJlc3BvbnNlc1xuICAgKiAtIExvZyBzdWNjZXNzZnVsIHJlc3BvbnNlc1xuICAgKlxuICAgKiBAcGFyYW0gaW50ZXJjZXB0b3IgLSBBc3luYyBmdW5jdGlvbiB0aGF0IHJlY2VpdmVzIGFuZCByZXR1cm5zIEFwaVJlc3BvbnNlXG4gICAqIEByZXR1cm5zIENsZWFudXAgZnVuY3Rpb24gdG8gdW5yZWdpc3RlciB0aGlzIGludGVyY2VwdG9yXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogLy8gRXh0cmFjdCBkYXRhIGZyb20gZW52ZWxvcGVcbiAgICogbWFuYWdlci5hZGRSZXNwb25zZUludGVyY2VwdG9yKGFzeW5jIChyZXNwb25zZSkgPT4ge1xuICAgKiAgIGlmIChyZXNwb25zZS5hcGlEYXRhPy5yZXN1bHQpIHtcbiAgICogICAgIHJlc3BvbnNlLmFwaURhdGEgPSByZXNwb25zZS5hcGlEYXRhLnJlc3VsdDtcbiAgICogICB9XG4gICAqICAgcmV0dXJuIHJlc3BvbnNlO1xuICAgKiB9KTtcbiAgICpcbiAgICogLy8gQWRkIHRpbWVzdGFtcHNcbiAgICogbWFuYWdlci5hZGRSZXNwb25zZUludGVyY2VwdG9yKGFzeW5jIChyZXNwb25zZSkgPT4ge1xuICAgKiAgIHJldHVybiB7XG4gICAqICAgICAuLi5yZXNwb25zZSxcbiAgICogICAgIHJlY2VpdmVkQXQ6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKVxuICAgKiAgIH07XG4gICAqIH0pO1xuICAgKiBgYGBcbiAgICovXG4gIGFkZFJlc3BvbnNlSW50ZXJjZXB0b3IoaW50ZXJjZXB0b3I6IFJlc3BvbnNlSW50ZXJjZXB0b3IpOiAoKSA9PiB2b2lkIHtcbiAgICB0aGlzLnJlc3BvbnNlSW50ZXJjZXB0b3JzLnB1c2goaW50ZXJjZXB0b3IpO1xuXG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgIGNvbnN0IGluZGV4ID0gdGhpcy5yZXNwb25zZUludGVyY2VwdG9ycy5pbmRleE9mKGludGVyY2VwdG9yKTtcblxuICAgICAgaWYgKGluZGV4ID4gLTEpIHRoaXMucmVzcG9uc2VJbnRlcmNlcHRvcnMuc3BsaWNlKGluZGV4LCAxKTtcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIFJlZ2lzdGVycyBhbiBlcnJvciBpbnRlcmNlcHRvciB0byBoYW5kbGUgb3IgdHJhbnNmb3JtIGVycm9ycyBiZWZvcmUgdGhleSBhcmUgdGhyb3duLlxuICAgKlxuICAgKiBFcnJvciBpbnRlcmNlcHRvcnMgY2FuOlxuICAgKiAtIExvZyBlcnJvcnMgdG8gbW9uaXRvcmluZyBzZXJ2aWNlc1xuICAgKiAtIFRyYW5zZm9ybSBlcnJvciBmb3JtYXRzXG4gICAqIC0gSW1wbGVtZW50IHJldHJ5IGxvZ2ljXG4gICAqIC0gU2hvdyB1c2VyIG5vdGlmaWNhdGlvbnNcbiAgICogLSBFeHRyYWN0IHZhbGlkYXRpb24gZXJyb3JzXG4gICAqXG4gICAqICoqTm90ZToqKiBFcnJvciBpbnRlcmNlcHRvcnMgc2hvdWxkIHJlLXRocm93IHRoZSBlcnJvciAob3IgYSB0cmFuc2Zvcm1lZCB2ZXJzaW9uKVxuICAgKiB0byBtYWludGFpbiB0aGUgZXJyb3IgZmxvdy4gVGhlIGZpbmFsIGVycm9yIGlzIGFsd2F5cyB0aHJvd24uXG4gICAqXG4gICAqIEBwYXJhbSBpbnRlcmNlcHRvciAtIEFzeW5jIGZ1bmN0aW9uIHRoYXQgcmVjZWl2ZXMgYW5kIHJldHVybnMgKG9yIHRocm93cykgQXBpRXJyb3JcbiAgICogQHJldHVybnMgQ2xlYW51cCBmdW5jdGlvbiB0byB1bnJlZ2lzdGVyIHRoaXMgaW50ZXJjZXB0b3JcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiAvLyBMb2cgdG8gbW9uaXRvcmluZyBzZXJ2aWNlXG4gICAqIG1hbmFnZXIuYWRkRXJyb3JJbnRlcmNlcHRvcihhc3luYyAoZXJyb3IpID0+IHtcbiAgICogICBpZiAoZXJyb3Iuc3RhdHVzID49IDUwMCkge1xuICAgKiAgICAgYXdhaXQgU2VudHJ5LmNhcHR1cmVFeGNlcHRpb24oZXJyb3IsIHtcbiAgICogICAgICAgZXh0cmE6IHsgdHJhY2VJZDogZXJyb3IudHJhY2VJZCB9XG4gICAqICAgICB9KTtcbiAgICogICB9XG4gICAqICAgdGhyb3cgZXJyb3I7IC8vIFJlLXRocm93IHRvIGNvbnRpbnVlIGVycm9yIGZsb3dcbiAgICogfSk7XG4gICAqXG4gICAqIC8vIFRyYW5zZm9ybSBlcnJvciBtZXNzYWdlc1xuICAgKiBtYW5hZ2VyLmFkZEVycm9ySW50ZXJjZXB0b3IoYXN5bmMgKGVycm9yKSA9PiB7XG4gICAqICAgaWYgKGVycm9yLnN0YXR1cyA9PT0gNDA0KSB7XG4gICAqICAgICBlcnJvci50aXRsZSA9ICdSZXNvdXJjZSBub3QgZm91bmQnO1xuICAgKiAgIH1cbiAgICogICB0aHJvdyBlcnJvcjtcbiAgICogfSk7XG4gICAqIGBgYFxuICAgKi9cbiAgYWRkRXJyb3JJbnRlcmNlcHRvcihpbnRlcmNlcHRvcjogRXJyb3JJbnRlcmNlcHRvcik6ICgpID0+IHZvaWQge1xuICAgIHRoaXMuZXJyb3JJbnRlcmNlcHRvcnMucHVzaChpbnRlcmNlcHRvcik7XG5cbiAgICByZXR1cm4gKCkgPT4ge1xuICAgICAgY29uc3QgaW5kZXggPSB0aGlzLmVycm9ySW50ZXJjZXB0b3JzLmluZGV4T2YoaW50ZXJjZXB0b3IpO1xuXG4gICAgICBpZiAoaW5kZXggPiAtMSkgdGhpcy5lcnJvckludGVyY2VwdG9ycy5zcGxpY2UoaW5kZXgsIDEpO1xuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQXBwbGllcyBhbGwgcmVnaXN0ZXJlZCByZXF1ZXN0IGludGVyY2VwdG9ycyBpbiBzZXF1ZW50aWFsIG9yZGVyLlxuICAgKlxuICAgKiBFYWNoIGludGVyY2VwdG9yIHJlY2VpdmVzIHRoZSBjb25maWcgbW9kaWZpZWQgYnkgdGhlIHByZXZpb3VzIGludGVyY2VwdG9yLFxuICAgKiBmb3JtaW5nIGEgcHJvY2Vzc2luZyBwaXBlbGluZS4gSWYgYW55IGludGVyY2VwdG9yIHRocm93cyBhbiBlcnJvcixcbiAgICogdGhlIHBpcGVsaW5lIHN0b3BzIGFuZCB0aGUgZXJyb3IgcHJvcGFnYXRlcy5cbiAgICpcbiAgICogQHBhcmFtIGNvbmZpZyAtIFRoZSBpbml0aWFsIHJlcXVlc3QgY29uZmlndXJhdGlvblxuICAgKiBAcmV0dXJucyBUaGUgbW9kaWZpZWQgcmVxdWVzdCBjb25maWd1cmF0aW9uIGFmdGVyIGFsbCBpbnRlcmNlcHRvcnNcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiBjb25zdCBjb25maWcgPSB7IG1ldGhvZDogJ0dFVCcsIHVybDogJy91c2VycycgfTtcbiAgICogY29uc3QgZmluYWxDb25maWcgPSBhd2FpdCBtYW5hZ2VyLmFwcGx5UmVxdWVzdEludGVyY2VwdG9ycyhjb25maWcpO1xuICAgKiAvLyBmaW5hbENvbmZpZyBoYXMgYmVlbiBwcm9jZXNzZWQgYnkgYWxsIHJlZ2lzdGVyZWQgaW50ZXJjZXB0b3JzXG4gICAqIGBgYFxuICAgKi9cbiAgYXN5bmMgYXBwbHlSZXF1ZXN0SW50ZXJjZXB0b3JzKFxuICAgIGNvbmZpZzogUmVxdWVzdENvbmZpZ1xuICApOiBQcm9taXNlPFJlcXVlc3RDb25maWc+IHtcbiAgICBsZXQgbW9kaWZpZWRDb25maWcgPSB7IC4uLmNvbmZpZyB9O1xuXG4gICAgZm9yIChjb25zdCBpbnRlcmNlcHRvciBvZiB0aGlzLnJlcXVlc3RJbnRlcmNlcHRvcnMpIHtcbiAgICAgIG1vZGlmaWVkQ29uZmlnID0gYXdhaXQgaW50ZXJjZXB0b3IobW9kaWZpZWRDb25maWcpO1xuICAgIH1cblxuICAgIHJldHVybiBtb2RpZmllZENvbmZpZztcbiAgfVxuXG4gIC8qKlxuICAgKiBBcHBsaWVzIGFsbCByZWdpc3RlcmVkIHJlc3BvbnNlIGludGVyY2VwdG9ycyBpbiBzZXF1ZW50aWFsIG9yZGVyLlxuICAgKlxuICAgKiBFYWNoIGludGVyY2VwdG9yIHJlY2VpdmVzIHRoZSByZXNwb25zZSBtb2RpZmllZCBieSB0aGUgcHJldmlvdXMgaW50ZXJjZXB0b3IsXG4gICAqIGZvcm1pbmcgYSBwcm9jZXNzaW5nIHBpcGVsaW5lLiBJZiBhbnkgaW50ZXJjZXB0b3IgdGhyb3dzIGFuIGVycm9yLFxuICAgKiB0aGUgcGlwZWxpbmUgc3RvcHMgYW5kIHRoZSBlcnJvciBwcm9wYWdhdGVzLlxuICAgKlxuICAgKiBAdGVtcGxhdGUgVCAtIFRoZSB0eXBlIG9mIHRoZSByZXNwb25zZSBkYXRhXG4gICAqIEBwYXJhbSByZXNwb25zZSAtIFRoZSBpbml0aWFsIEFQSSByZXNwb25zZVxuICAgKiBAcmV0dXJucyBUaGUgbW9kaWZpZWQgcmVzcG9uc2UgYWZ0ZXIgYWxsIGludGVyY2VwdG9yc1xuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGNvbnN0IHJlc3BvbnNlID0geyBkYXRhOiB7IGlkOiAxLCBuYW1lOiAnSm9obicgfSB9O1xuICAgKiBjb25zdCBmaW5hbFJlc3BvbnNlID0gYXdhaXQgbWFuYWdlci5hcHBseVJlc3BvbnNlSW50ZXJjZXB0b3JzKHJlc3BvbnNlKTtcbiAgICogLy8gZmluYWxSZXNwb25zZSBoYXMgYmVlbiBwcm9jZXNzZWQgYnkgYWxsIHJlZ2lzdGVyZWQgaW50ZXJjZXB0b3JzXG4gICAqIGBgYFxuICAgKi9cbiAgYXN5bmMgYXBwbHlSZXNwb25zZUludGVyY2VwdG9yczxUPihcbiAgICByZXNwb25zZTogQXBpUmVzcG9uc2U8VD5cbiAgKTogUHJvbWlzZTxBcGlSZXNwb25zZTxUPj4ge1xuICAgIGxldCBtb2RpZmllZFJlc3BvbnNlID0gcmVzcG9uc2U7XG5cbiAgICBmb3IgKGNvbnN0IGludGVyY2VwdG9yIG9mIHRoaXMucmVzcG9uc2VJbnRlcmNlcHRvcnMpIHtcbiAgICAgIG1vZGlmaWVkUmVzcG9uc2UgPSBhd2FpdCBpbnRlcmNlcHRvcihtb2RpZmllZFJlc3BvbnNlKTtcbiAgICB9XG5cbiAgICByZXR1cm4gbW9kaWZpZWRSZXNwb25zZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBBcHBsaWVzIGFsbCByZWdpc3RlcmVkIGVycm9yIGludGVyY2VwdG9ycyBpbiBzZXF1ZW50aWFsIG9yZGVyIGFuZCByZS10aHJvd3MuXG4gICAqXG4gICAqIEVhY2ggaW50ZXJjZXB0b3IgcmVjZWl2ZXMgdGhlIGVycm9yIChwb3RlbnRpYWxseSBtb2RpZmllZCBieSBwcmV2aW91cyBpbnRlcmNlcHRvcnMpLlxuICAgKiBJbnRlcmNlcHRvcnMgY2FuIHRyYW5zZm9ybSB0aGUgZXJyb3IgYmVmb3JlIHJlLXRocm93aW5nIGl0LiBUaGUgZmluYWwgZXJyb3JcbiAgICogaXMgYWx3YXlzIHRocm93biB0byBtYWludGFpbiBlcnJvciBmbG93LlxuICAgKlxuICAgKiBJZiBhbiBpbnRlcmNlcHRvciBpdHNlbGYgdGhyb3dzIGFuIGVycm9yLCB0aGF0IGJlY29tZXMgdGhlIG5ldyBlcnJvciB0byBwcm9jZXNzXG4gICAqIGJ5IHN1YnNlcXVlbnQgaW50ZXJjZXB0b3JzLlxuICAgKlxuICAgKiBAcGFyYW0gZXJyb3IgLSBUaGUgaW5pdGlhbCBBUEkgZXJyb3JcbiAgICogQHJldHVybnMgTmV2ZXIgcmV0dXJucyAoYWx3YXlzIHRocm93cylcbiAgICogQHRocm93cyBUaGUgZmluYWwgZXJyb3IgYWZ0ZXIgYWxsIGludGVyY2VwdG9ycyBoYXZlIHByb2Nlc3NlZCBpdFxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIHRyeSB7XG4gICAqICAgYXdhaXQgbWFuYWdlci5hcHBseUVycm9ySW50ZXJjZXB0b3JzKGVycm9yKTtcbiAgICogfSBjYXRjaCAoZmluYWxFcnJvcikge1xuICAgKiAgIC8vIGZpbmFsRXJyb3IgaGFzIGJlZW4gcHJvY2Vzc2VkIGJ5IGFsbCByZWdpc3RlcmVkIGVycm9yIGludGVyY2VwdG9yc1xuICAgKiB9XG4gICAqIGBgYFxuICAgKi9cbiAgYXN5bmMgYXBwbHlFcnJvckludGVyY2VwdG9ycyhlcnJvcjogQXBpRXJyb3IpOiBQcm9taXNlPG5ldmVyPiB7XG4gICAgbGV0IG1vZGlmaWVkRXJyb3IgPSBlcnJvcjtcblxuICAgIGZvciAoY29uc3QgaW50ZXJjZXB0b3Igb2YgdGhpcy5lcnJvckludGVyY2VwdG9ycykge1xuICAgICAgdHJ5IHtcbiAgICAgICAgbW9kaWZpZWRFcnJvciA9IGF3YWl0IGludGVyY2VwdG9yKG1vZGlmaWVkRXJyb3IpO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICBtb2RpZmllZEVycm9yID0gZSBhcyBBcGlFcnJvcjtcbiAgICAgIH1cbiAgICB9XG5cbiAgICB0aHJvdyBtb2RpZmllZEVycm9yO1xuICB9XG59XG4iLCAiLyoqXG4gKiBNYW5hZ2VzIGFjdGl2ZSBIVFRQIHJlcXVlc3RzIHdpdGggc3VwcG9ydCBmb3IgY2FuY2VsbGF0aW9uIGFuZCBjb3JyZWxhdGlvbiB0cmFja2luZy5cbiAqXG4gKiBUaGlzIGludGVybmFsIGNsYXNzIG1haW50YWlucyBhIHJlZ2lzdHJ5IG9mIGluLWZsaWdodCByZXF1ZXN0cywgYWxsb3dpbmcgdGhlbSB0byBiZVxuICogY2FuY2VsbGVkIGluZGl2aWR1YWxseSBvciBpbiBidWxrLiBFYWNoIHJlcXVlc3QgaXMgdHJhY2tlZCBieSBhIHVuaXF1ZSBrZXkgYW5kXG4gKiBhc3NvY2lhdGVkIHdpdGggYSBjb3JyZWxhdGlvbiBJRCBmb3IgZGlzdHJpYnV0ZWQgdHJhY2luZy5cbiAqXG4gKiBLZXkgRmVhdHVyZXM6XG4gKiAtICoqUmVxdWVzdCBEZWR1cGxpY2F0aW9uKio6IEF1dG9tYXRpY2FsbHkgY2FuY2VscyBkdXBsaWNhdGUgcmVxdWVzdHMgd2l0aCB0aGUgc2FtZSBrZXlcbiAqIC0gKipDb3JyZWxhdGlvbiBUcmFja2luZyoqOiBNYXBzIHJlcXVlc3Qga2V5cyB0byBjb3JyZWxhdGlvbiBJRHMgZm9yIHRyYWNpbmdcbiAqIC0gKipCdWxrIENhbmNlbGxhdGlvbioqOiBDYW5jZWwgYWxsIHBlbmRpbmcgcmVxdWVzdHMgYXQgb25jZSAodXNlZnVsIGZvciBjbGVhbnVwKVxuICogLSAqKk1lbW9yeSBNYW5hZ2VtZW50Kio6IEF1dG9tYXRpY2FsbHkgY2xlYW5zIHVwIHRyYWNraW5nIGRhdGEgd2hlbiByZXF1ZXN0cyBjb21wbGV0ZVxuICpcbiAqIEBpbnRlcm5hbCBUaGlzIGNsYXNzIGlzIG5vdCBleHBvcnRlZCBmcm9tIHRoZSBwdWJsaWMgQVBJXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGNvbnN0IG1hbmFnZXIgPSBuZXcgUmVxdWVzdE1hbmFnZXIoKTtcbiAqIGNvbnN0IGNvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG4gKlxuICogLy8gVHJhY2sgYSByZXF1ZXN0XG4gKiBtYW5hZ2VyLmFkZCgnR0VUXy91c2Vycy8xMjMnLCBjb250cm9sbGVyLCAnYXBpLWFiYzEyMycpO1xuICpcbiAqIC8vIENhbmNlbCBzcGVjaWZpYyByZXF1ZXN0XG4gKiBtYW5hZ2VyLmNhbmNlbCgnR0VUXy91c2Vycy8xMjMnKTtcbiAqXG4gKiAvLyBPciBjYW5jZWwgYWxsIHJlcXVlc3RzXG4gKiBtYW5hZ2VyLmNhbmNlbEFsbCgpO1xuICogYGBgXG4gKi9cbmV4cG9ydCBjbGFzcyBSZXF1ZXN0TWFuYWdlciB7XG4gIC8qKlxuICAgKiBNYXAgb2YgYWN0aXZlIHJlcXVlc3Qga2V5cyB0byB0aGVpciBhYm9ydCBjb250cm9sbGVyc1xuICAgKiBAcHJpdmF0ZVxuICAgKi9cbiAgcHJpdmF0ZSBhY3RpdmVSZXF1ZXN0czogTWFwPHN0cmluZywgQWJvcnRDb250cm9sbGVyPiA9IG5ldyBNYXAoKTtcblxuICAvKipcbiAgICogTWFwIG9mIHJlcXVlc3Qga2V5cyB0byB0aGVpciBjb3JyZWxhdGlvbiBJRHMgZm9yIHRyYWNpbmdcbiAgICogQHByaXZhdGVcbiAgICovXG4gIHByaXZhdGUgY29ycmVsYXRpb25NYXA6IE1hcDxzdHJpbmcsIHN0cmluZz4gPSBuZXcgTWFwKCk7XG5cbiAgLyoqXG4gICAqIFJlZ2lzdGVycyBhIG5ldyByZXF1ZXN0IGZvciB0cmFja2luZyBhbmQgY2FuY2VsbGF0aW9uIG1hbmFnZW1lbnQuXG4gICAqXG4gICAqIElmIGEgcmVxdWVzdCB3aXRoIHRoZSBzYW1lIGtleSBhbHJlYWR5IGV4aXN0cywgaXQgd2lsbCBiZSBhdXRvbWF0aWNhbGx5XG4gICAqIGNhbmNlbGxlZCBiZWZvcmUgdGhlIG5ldyBvbmUgaXMgcmVnaXN0ZXJlZCAocmVxdWVzdCBkZWR1cGxpY2F0aW9uKS5cbiAgICpcbiAgICogQHBhcmFtIGtleSAtIFVuaXF1ZSBpZGVudGlmaWVyIGZvciB0aGUgcmVxdWVzdCAodHlwaWNhbGx5IG1ldGhvZCArIFVSTCArIHRpbWVzdGFtcClcbiAgICogQHBhcmFtIGNvbnRyb2xsZXIgLSBBYm9ydENvbnRyb2xsZXIgZm9yIGNhbmNlbGxpbmcgdGhlIHJlcXVlc3RcbiAgICogQHBhcmFtIGNvcnJlbGF0aW9uSWQgLSBDb3JyZWxhdGlvbiBJRCBmb3IgZGlzdHJpYnV0ZWQgdHJhY2luZ1xuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGNvbnN0IGNvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG4gICAqIG1hbmFnZXIuYWRkKCdHRVRfL2FwaS91c2Vyc18xNjk5OTk5OTk5JywgY29udHJvbGxlciwgJ2FwaS1hYmMxMjMnKTtcbiAgICogYGBgXG4gICAqL1xuICBhZGQoa2V5OiBzdHJpbmcsIGNvbnRyb2xsZXI6IEFib3J0Q29udHJvbGxlciwgY29ycmVsYXRpb25JZDogc3RyaW5nKTogdm9pZCB7XG4gICAgLy8gQ2FuY2VsIGV4aXN0aW5nIHJlcXVlc3Qgd2l0aCBzYW1lIGtleSBpZiBleGlzdHNcbiAgICB0aGlzLmNhbmNlbChrZXkpO1xuICAgIHRoaXMuYWN0aXZlUmVxdWVzdHMuc2V0KGtleSwgY29udHJvbGxlcik7XG4gICAgdGhpcy5jb3JyZWxhdGlvbk1hcC5zZXQoa2V5LCBjb3JyZWxhdGlvbklkKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZW1vdmVzIGEgcmVxdWVzdCBmcm9tIHRyYWNraW5nIHdpdGhvdXQgY2FuY2VsbGluZyBpdC5cbiAgICpcbiAgICogVGhpcyBpcyB0eXBpY2FsbHkgY2FsbGVkIHdoZW4gYSByZXF1ZXN0IGNvbXBsZXRlcyBzdWNjZXNzZnVsbHkgb3IgZmFpbHMuXG4gICAqIFVzZSB7QGxpbmsgY2FuY2VsfSBpbnN0ZWFkIGlmIHlvdSBuZWVkIHRvIGFib3J0IHRoZSByZXF1ZXN0LlxuICAgKlxuICAgKiBAcGFyYW0ga2V5IC0gVW5pcXVlIGlkZW50aWZpZXIgZm9yIHRoZSByZXF1ZXN0IHRvIHJlbW92ZVxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIC8vIENhbGxlZCBhdXRvbWF0aWNhbGx5IGFmdGVyIHJlcXVlc3QgY29tcGxldGVzXG4gICAqIG1hbmFnZXIucmVtb3ZlKCdHRVRfL2FwaS91c2Vyc18xNjk5OTk5OTk5Jyk7XG4gICAqIGBgYFxuICAgKi9cbiAgcmVtb3ZlKGtleTogc3RyaW5nKTogdm9pZCB7XG4gICAgdGhpcy5hY3RpdmVSZXF1ZXN0cy5kZWxldGUoa2V5KTtcbiAgICB0aGlzLmNvcnJlbGF0aW9uTWFwLmRlbGV0ZShrZXkpO1xuICB9XG5cbiAgLyoqXG4gICAqIENhbmNlbHMgYSBzcGVjaWZpYyByZXF1ZXN0IGFuZCByZW1vdmVzIGl0IGZyb20gdHJhY2tpbmcuXG4gICAqXG4gICAqIElmIHRoZSByZXF1ZXN0IGRvZXNuJ3QgZXhpc3Qgb3Igd2FzIGFscmVhZHkgY2FuY2VsbGVkLCB0aGlzIG9wZXJhdGlvbiBpcyBhIG5vLW9wLlxuICAgKiBUaGUgYXNzb2NpYXRlZCBBYm9ydENvbnRyb2xsZXIncyBzaWduYWwgd2lsbCBiZSB0cmlnZ2VyZWQsIGNhdXNpbmcgYW55IGFjdGl2ZVxuICAgKiBmZXRjaCBvcGVyYXRpb25zIHRvIGFib3J0LlxuICAgKlxuICAgKiBAcGFyYW0ga2V5IC0gVW5pcXVlIGlkZW50aWZpZXIgZm9yIHRoZSByZXF1ZXN0IHRvIGNhbmNlbFxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIC8vIFVzZXIgbmF2aWdhdGVzIGF3YXksIGNhbmNlbCB0aGUgcGVuZGluZyByZXF1ZXN0XG4gICAqIG1hbmFnZXIuY2FuY2VsKCdHRVRfL2FwaS91c2Vyc18xNjk5OTk5OTk5Jyk7XG4gICAqIGBgYFxuICAgKi9cbiAgY2FuY2VsKGtleTogc3RyaW5nKTogdm9pZCB7XG4gICAgY29uc3QgY29udHJvbGxlciA9IHRoaXMuYWN0aXZlUmVxdWVzdHMuZ2V0KGtleSk7XG5cbiAgICBpZiAoY29udHJvbGxlcikge1xuICAgICAgY29udHJvbGxlci5hYm9ydCgpO1xuICAgICAgdGhpcy5hY3RpdmVSZXF1ZXN0cy5kZWxldGUoa2V5KTtcbiAgICAgIHRoaXMuY29ycmVsYXRpb25NYXAuZGVsZXRlKGtleSk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIENhbmNlbHMgYWxsIGFjdGl2ZSByZXF1ZXN0cyBhbmQgY2xlYXJzIGFsbCB0cmFja2luZyBkYXRhLlxuICAgKlxuICAgKiBUaGlzIGlzIHVzZWZ1bCBmb3IgY2xlYW51cCBzY2VuYXJpb3Mgc3VjaCBhczpcbiAgICogLSBVc2VyIGxvZ291dFxuICAgKiAtIENvbXBvbmVudCB1bm1vdW50XG4gICAqIC0gTmF2aWdhdGlvbiB0byBhIGRpZmZlcmVudCBwYXJ0IG9mIHRoZSBhcHBsaWNhdGlvblxuICAgKiAtIEVycm9yIHJlY292ZXJ5IHRoYXQgcmVxdWlyZXMgYSBjbGVhbiBzbGF0ZVxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIC8vIENhbmNlbCBhbGwgcGVuZGluZyByZXF1ZXN0cyBvbiBsb2dvdXRcbiAgICogZnVuY3Rpb24gaGFuZGxlTG9nb3V0KCkge1xuICAgKiAgIGFwaUNsaWVudC5jYW5jZWxBbGxSZXF1ZXN0cygpO1xuICAgKiAgIC8vIC4uLiByZXN0IG9mIGxvZ291dCBsb2dpY1xuICAgKiB9XG4gICAqIGBgYFxuICAgKi9cbiAgY2FuY2VsQWxsKCk6IHZvaWQge1xuICAgIHRoaXMuYWN0aXZlUmVxdWVzdHMuZm9yRWFjaChjb250cm9sbGVyID0+IGNvbnRyb2xsZXIuYWJvcnQoKSk7XG4gICAgdGhpcy5hY3RpdmVSZXF1ZXN0cy5jbGVhcigpO1xuICAgIHRoaXMuY29ycmVsYXRpb25NYXAuY2xlYXIoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDaGVja3MgaWYgYSByZXF1ZXN0IHdpdGggdGhlIGdpdmVuIGtleSBpcyBjdXJyZW50bHkgYmVpbmcgdHJhY2tlZC5cbiAgICpcbiAgICogQHBhcmFtIGtleSAtIFVuaXF1ZSBpZGVudGlmaWVyIGZvciB0aGUgcmVxdWVzdFxuICAgKiBAcmV0dXJucyBgdHJ1ZWAgaWYgdGhlIHJlcXVlc3QgaXMgYWN0aXZlLCBgZmFsc2VgIG90aGVyd2lzZVxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGlmIChtYW5hZ2VyLmhhcygnR0VUXy9hcGkvdXNlcnNfMTY5OTk5OTk5OScpKSB7XG4gICAqICAgY29uc29sZS5sb2coJ1JlcXVlc3QgaXMgc3RpbGwgcGVuZGluZycpO1xuICAgKiB9XG4gICAqIGBgYFxuICAgKi9cbiAgaGFzKGtleTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuYWN0aXZlUmVxdWVzdHMuaGFzKGtleSk7XG4gIH1cblxuICAvKipcbiAgICogUmV0cmlldmVzIHRoZSBjb3JyZWxhdGlvbiBJRCBmb3IgYSBnaXZlbiByZXF1ZXN0IGtleS5cbiAgICpcbiAgICogQ29ycmVsYXRpb24gSURzIGFyZSB1c2VkIGZvciBkaXN0cmlidXRlZCB0cmFjaW5nIGFuZCByZXF1ZXN0IHRyYWNraW5nXG4gICAqIGFjcm9zcyBzZXJ2aWNlcyBhbmQgbG9ncy5cbiAgICpcbiAgICogQHBhcmFtIGtleSAtIFVuaXF1ZSBpZGVudGlmaWVyIGZvciB0aGUgcmVxdWVzdFxuICAgKiBAcmV0dXJucyBUaGUgY29ycmVsYXRpb24gSUQgaWYgZm91bmQsIGB1bmRlZmluZWRgIG90aGVyd2lzZVxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGNvbnN0IGNvcnJlbGF0aW9uSWQgPSBtYW5hZ2VyLmdldENvcnJlbGF0aW9uSWQoJ0dFVF8vYXBpL3VzZXJzXzE2OTk5OTk5OTknKTtcbiAgICogaWYgKGNvcnJlbGF0aW9uSWQpIHtcbiAgICogICBjb25zb2xlLmxvZygnVHJhY2UgcmVxdWVzdCB3aXRoIElEOicsIGNvcnJlbGF0aW9uSWQpO1xuICAgKiB9XG4gICAqIGBgYFxuICAgKi9cbiAgZ2V0Q29ycmVsYXRpb25JZChrZXk6IHN0cmluZyk6IHN0cmluZyB8IHVuZGVmaW5lZCB7XG4gICAgcmV0dXJuIHRoaXMuY29ycmVsYXRpb25NYXAuZ2V0KGtleSk7XG4gIH1cbn1cbiIsICJpbXBvcnQgdHlwZSB7IEFwaUVycm9yIH0gZnJvbSAnLi4vdHlwZXMvQXBpRXJyb3InO1xuXG4vKipcbiAqIEltcGxlbWVudHMgYXV0b21hdGljIHJldHJ5IGxvZ2ljIHdpdGggZXhwb25lbnRpYWwgYmFja29mZiBmb3IgZmFpbGVkIEhUVFAgcmVxdWVzdHMuXG4gKlxuICogVGhpcyBpbnRlcm5hbCBjbGFzcyBoYW5kbGVzIHRyYW5zaWVudCBmYWlsdXJlcyBieSBhdXRvbWF0aWNhbGx5IHJldHJ5aW5nIHJlcXVlc3RzXG4gKiB3aXRoIHByb2dyZXNzaXZlbHkgaW5jcmVhc2luZyBkZWxheXMgYmV0d2VlbiBhdHRlbXB0cy4gVGhlIGV4cG9uZW50aWFsIGJhY2tvZmZcbiAqIHN0cmF0ZWd5IGhlbHBzIHByZXZlbnQgb3ZlcndoZWxtaW5nIGEgc3RydWdnbGluZyBzZXJ2ZXIgd2hpbGUgZ2l2aW5nIHRlbXBvcmFyeVxuICogaXNzdWVzIHRpbWUgdG8gcmVzb2x2ZS5cbiAqXG4gKiBLZXkgRmVhdHVyZXM6XG4gKiAtICoqRXhwb25lbnRpYWwgQmFja29mZioqOiBEb3VibGVzIGRlbGF5IGJldHdlZW4gcmV0cmllcyAoZS5nLiwgMXMsIDJzLCA0cywgOHMpXG4gKiAtICoqU21hcnQgRXJyb3IgRGV0ZWN0aW9uKio6IFNraXBzIHJldHJ5IGZvciBub24tcmV0cnlhYmxlIGVycm9ycyAodmFsaWRhdGlvbiwgYWJvcnRzKVxuICogLSAqKkFib3J0IFN1cHBvcnQqKjogUmVzcGVjdHMgY2FuY2VsbGF0aW9uIHNpZ25hbHMgZHVyaW5nIHJldHJ5IGRlbGF5c1xuICogLSAqKlR5cGUgU2FmZXR5Kio6IEZ1bGx5IGdlbmVyaWMgaW1wbGVtZW50YXRpb24gcHJlc2VydmluZyByZXR1cm4gdHlwZXNcbiAqXG4gKiBSZXRyeSBTdHJhdGVneTpcbiAqIC0gKipSZXRyeWFibGUqKjogTmV0d29yayBlcnJvcnMsIDV4eCBzZXJ2ZXIgZXJyb3JzLCB0aW1lb3V0c1xuICogLSAqKk5vbi1SZXRyeWFibGUqKjogVmFsaWRhdGlvbiBlcnJvcnMgKDQwMCksIEFib3J0RXJyb3JzLCBjYW5jZWxsZWQgcmVxdWVzdHNcbiAqXG4gKiBAaW50ZXJuYWwgVGhpcyBjbGFzcyBpcyBub3QgZXhwb3J0ZWQgZnJvbSB0aGUgcHVibGljIEFQSVxuICpcbiAqIEBleGFtcGxlXG4gKiBSZXRyeSBjb25maWd1cmF0aW9uIGluIEFwaUNsaWVudDpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGNvbnN0IGNsaWVudCA9IG5ldyBBcGlDbGllbnQoJ2h0dHBzOi8vYXBpLmV4YW1wbGUuY29tJyk7XG4gKlxuICogLy8gUmV0cnkgdXAgdG8gMyB0aW1lcyB3aXRoIDFzIGluaXRpYWwgZGVsYXlcbiAqIGNvbnN0IHsgYXBpRGF0YSB9ID0gYXdhaXQgY2xpZW50LmdldCgnL3VzZXJzJywge1xuICogICByZXRyaWVzOiAzLFxuICogICByZXRyeURlbGF5OiAxMDAwICAvLyAxcywgMnMsIDRzXG4gKiB9KTtcbiAqIGBgYFxuICovXG5leHBvcnQgY2xhc3MgUmV0cnlIYW5kbGVyIHtcbiAgLyoqXG4gICAqIFJldHJpZXMgYSBmYWlsZWQgcmVxdWVzdCB3aXRoIGV4cG9uZW50aWFsIGJhY2tvZmYgc3RyYXRlZ3kuXG4gICAqXG4gICAqIFRoZSByZXRyeSBsb2dpYyB3b3JrcyBhcyBmb2xsb3dzOlxuICAgKiAxLiBBdHRlbXB0cyB0aGUgcmVxdWVzdCBpbW1lZGlhdGVseVxuICAgKiAyLiBPbiBmYWlsdXJlLCBjaGVja3MgaWYgdGhlIGVycm9yIGlzIHJldHJ5YWJsZVxuICAgKiAzLiBJZiByZXRyeWFibGUgYW5kIHJldHJpZXMgcmVtYWluLCB3YWl0cyBmb3IgdGhlIGN1cnJlbnQgZGVsYXlcbiAgICogNC4gRG91YmxlcyB0aGUgZGVsYXkgZm9yIHRoZSBuZXh0IGF0dGVtcHRcbiAgICogNS4gUmVwZWF0cyB1bnRpbCBzdWNjZXNzIG9yIHJldHJpZXMgZXhoYXVzdGVkXG4gICAqXG4gICAqICoqTm9uLVJldHJ5YWJsZSBFcnJvcnM6KipcbiAgICogLSBWYWxpZGF0aW9uIGVycm9ycyAoNDAwKSAtIENsaWVudCBzZW50IGJhZCBkYXRhXG4gICAqIC0gQWJvcnRFcnJvcnMgLSBSZXF1ZXN0IHdhcyBleHBsaWNpdGx5IGNhbmNlbGxlZFxuICAgKiAtIFJlcXVlc3RzIHdpdGggYWJvcnRlZCBzaWduYWxzXG4gICAqXG4gICAqICoqQWJvcnQgSGFuZGxpbmc6KipcbiAgICogSWYgdGhlIHNpZ25hbCBpcyBhYm9ydGVkIGR1cmluZyBhIHJldHJ5IGRlbGF5LCB0aGUgcmV0cnkgaXMgaW1tZWRpYXRlbHlcbiAgICogY2FuY2VsbGVkIGFuZCBhbiBBYm9ydEVycm9yIGlzIHRocm93bi5cbiAgICpcbiAgICogQHRlbXBsYXRlIFQgLSBUaGUgcmV0dXJuIHR5cGUgb2YgdGhlIGZ1bmN0aW9uIGJlaW5nIHJldHJpZWRcbiAgICogQHBhcmFtIGZuIC0gQXN5bmMgZnVuY3Rpb24gdG8gcmV0cnkgb24gZmFpbHVyZVxuICAgKiBAcGFyYW0gcmV0cmllcyAtIE51bWJlciBvZiByZXRyeSBhdHRlbXB0cyByZW1haW5pbmcgKGRlY3JlbWVudHMgZWFjaCByZXRyeSlcbiAgICogQHBhcmFtIGRlbGF5IC0gQ3VycmVudCBkZWxheSBpbiBtaWxsaXNlY29uZHMgYmVmb3JlIG5leHQgcmV0cnlcbiAgICogQHBhcmFtIHNpZ25hbCAtIE9wdGlvbmFsIEFib3J0U2lnbmFsIHRvIGNhbmNlbCByZXRyaWVzXG4gICAqIEByZXR1cm5zIFByb21pc2UgcmVzb2x2aW5nIHRvIHRoZSBmdW5jdGlvbidzIHJlc3VsdCBvbiBzdWNjZXNzXG4gICAqIEB0aHJvd3MgVGhlIGxhc3QgZXJyb3IgZW5jb3VudGVyZWQgaWYgYWxsIHJldHJpZXMgYXJlIGV4aGF1c3RlZFxuICAgKiBAdGhyb3dzIEFib3J0RXJyb3IgaWYgdGhlIHNpZ25hbCBpcyBhYm9ydGVkIGR1cmluZyBleGVjdXRpb24gb3IgZGVsYXlcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogQmFzaWMgcmV0cnkgdXNhZ2U6XG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgaGFuZGxlciA9IG5ldyBSZXRyeUhhbmRsZXIoKTtcbiAgICogY29uc3QgZmV0Y2hVc2VyID0gKCkgPT4gZmV0Y2goJy9hcGkvdXNlcnMvMTIzJykudGhlbihyID0+IHIuanNvbigpKTtcbiAgICpcbiAgICogdHJ5IHtcbiAgICogICBjb25zdCB1c2VyID0gYXdhaXQgaGFuZGxlci5yZXRyeVJlcXVlc3QoXG4gICAqICAgICBmZXRjaFVzZXIsXG4gICAqICAgICAzLCAgICAgIC8vIDMgcmV0cmllc1xuICAgKiAgICAgMTAwMCAgICAvLyBTdGFydCB3aXRoIDFzIGRlbGF5XG4gICAqICAgKTtcbiAgICogICBjb25zb2xlLmxvZygnVXNlcjonLCB1c2VyKTtcbiAgICogfSBjYXRjaCAoZXJyb3IpIHtcbiAgICogICBjb25zb2xlLmVycm9yKCdGYWlsZWQgYWZ0ZXIgYWxsIHJldHJpZXM6JywgZXJyb3IpO1xuICAgKiB9XG4gICAqIGBgYFxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBXaXRoIGNhbmNlbGxhdGlvbiBzdXBwb3J0OlxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGNvbnN0IGNvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG4gICAqIGNvbnN0IHNpZ25hbCA9IGNvbnRyb2xsZXIuc2lnbmFsO1xuICAgKlxuICAgKiAvLyBDYW5jZWwgYWZ0ZXIgNSBzZWNvbmRzXG4gICAqIHNldFRpbWVvdXQoKCkgPT4gY29udHJvbGxlci5hYm9ydCgpLCA1MDAwKTtcbiAgICpcbiAgICogdHJ5IHtcbiAgICogICBhd2FpdCBoYW5kbGVyLnJldHJ5UmVxdWVzdChmZXRjaFVzZXIsIDUsIDEwMDAsIHNpZ25hbCk7XG4gICAqIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAqICAgaWYgKGVycm9yLm5hbWUgPT09ICdBYm9ydEVycm9yJykge1xuICAgKiAgICAgY29uc29sZS5sb2coJ1JldHJ5IGNhbmNlbGxlZCcpO1xuICAgKiAgIH1cbiAgICogfVxuICAgKiBgYGBcbiAgICovXG4gIGFzeW5jIHJldHJ5UmVxdWVzdDxUPihcbiAgICBmbjogKCkgPT4gUHJvbWlzZTxUPixcbiAgICByZXRyaWVzOiBudW1iZXIsXG4gICAgZGVsYXk6IG51bWJlcixcbiAgICBzaWduYWw/OiBBYm9ydFNpZ25hbFxuICApOiBQcm9taXNlPFQ+IHtcbiAgICB0cnkge1xuICAgICAgLy8gQ2hlY2sgaWYgYWxyZWFkeSBhYm9ydGVkXG4gICAgICBpZiAoc2lnbmFsPy5hYm9ydGVkKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihzaWduYWwucmVhc29uIHx8ICdSZXF1ZXN0IGFib3J0ZWQnKTtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIGF3YWl0IGZuKCk7XG4gICAgfSBjYXRjaCAoZXJyb3I6IHVua25vd24pIHtcbiAgICAgIGNvbnN0IGVyciA9IGVycm9yIGFzIEFwaUVycm9yO1xuXG4gICAgICAvLyBEb24ndCByZXRyeSBpZiBhYm9ydGVkXG4gICAgICBpZiAoZXJyLm5hbWUgPT09ICdBYm9ydEVycm9yJyB8fCBzaWduYWw/LmFib3J0ZWQpIHtcbiAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICB9XG5cbiAgICAgIC8vIERvbid0IHJldHJ5IHZhbGlkYXRpb24gZXJyb3JzXG4gICAgICBpZiAoZXJyLnR5cGUgPT09ICd2YWxpZGF0aW9uX2Vycm9yJyB8fCBlcnIuc3RhdHVzID09PSA0MDApIHtcbiAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICB9XG5cbiAgICAgIGlmIChyZXRyaWVzID09PSAwKSB0aHJvdyBlcnJvcjtcblxuICAgICAgLy8gV2FpdCB3aXRoIGFib3J0IHN1cHBvcnRcbiAgICAgIGF3YWl0IG5ldyBQcm9taXNlPHZvaWQ+KChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgICAgY29uc3QgdGltZW91dElkID0gc2V0VGltZW91dChyZXNvbHZlLCBkZWxheSk7XG5cbiAgICAgICAgaWYgKHNpZ25hbCkge1xuICAgICAgICAgIHNpZ25hbC5hZGRFdmVudExpc3RlbmVyKFxuICAgICAgICAgICAgJ2Fib3J0JyxcbiAgICAgICAgICAgICgpID0+IHtcbiAgICAgICAgICAgICAgY2xlYXJUaW1lb3V0KHRpbWVvdXRJZCk7XG4gICAgICAgICAgICAgIHJlamVjdChuZXcgRXJyb3Ioc2lnbmFsLnJlYXNvbiB8fCAnUmVxdWVzdCBhYm9ydGVkJykpO1xuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIHsgb25jZTogdHJ1ZSB9XG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfSk7XG5cbiAgICAgIHJldHVybiB0aGlzLnJldHJ5UmVxdWVzdChmbiwgcmV0cmllcyAtIDEsIGRlbGF5ICogMiwgc2lnbmFsKTtcbiAgICB9XG4gIH1cbn1cbiIsICIvKipcbiAqIE1hbmFnZXMgQWJvcnRTaWduYWwgY3JlYXRpb24gYW5kIGNvbXBvc2l0aW9uIGZvciByZXF1ZXN0IGNhbmNlbGxhdGlvbiBhbmQgdGltZW91dHMuXG4gKlxuICogVGhpcyBpbnRlcm5hbCBjbGFzcyBwcm92aWRlcyB1dGlsaXRpZXMgZm9yIHdvcmtpbmcgd2l0aCB0aGUgQWJvcnRTaWduYWwgQVBJLFxuICogZW5hYmxpbmcgc29waGlzdGljYXRlZCBjYW5jZWxsYXRpb24gcGF0dGVybnMgYnkgY29tYmluaW5nIG11bHRpcGxlIGNhbmNlbGxhdGlvblxuICogc291cmNlcyBhbmQgY3JlYXRpbmcgdGltZW91dC1iYXNlZCBzaWduYWxzLlxuICpcbiAqIEtleSBGZWF0dXJlczpcbiAqIC0gKipTaWduYWwgQ29tcG9zaXRpb24qKjogQ29tYmluZSBtdWx0aXBsZSBBYm9ydFNpZ25hbHMgaW50byBvbmVcbiAqIC0gKipUaW1lb3V0IFNpZ25hbHMqKjogQ3JlYXRlIHNpZ25hbHMgdGhhdCBhdXRvLWFib3J0IGFmdGVyIGEgZGVsYXlcbiAqIC0gKipFYXJseSBBYm9ydCBEZXRlY3Rpb24qKjogSW1tZWRpYXRlbHkgYWJvcnQgaWYgYW55IHNvdXJjZSBpcyBhbHJlYWR5IGFib3J0ZWRcbiAqIC0gKipSZXNvdXJjZSBDbGVhbnVwKio6IEF1dG9tYXRpYyB0aW1lb3V0IGNsZWFudXAgb24gYWJvcnRcbiAqXG4gKiBVc2UgQ2FzZXM6XG4gKiAtIENvbWJpbmUgdXNlciBjYW5jZWxsYXRpb24gKyB0aW1lb3V0ICsgY29tcG9uZW50IHVubW91bnQgc2lnbmFsc1xuICogLSBDcmVhdGUgcmVxdWVzdCBkZWFkbGluZXNcbiAqIC0gSW1wbGVtZW50IGNhc2NhZGluZyBjYW5jZWxsYXRpb25cbiAqXG4gKiBAaW50ZXJuYWwgVGhpcyBjbGFzcyBpcyBub3QgZXhwb3J0ZWQgZnJvbSB0aGUgcHVibGljIEFQSVxuICpcbiAqIEBleGFtcGxlXG4gKiBDb21iaW5pbmcgbXVsdGlwbGUgY2FuY2VsbGF0aW9uIHNvdXJjZXM6XG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBtYW5hZ2VyID0gbmV3IFNpZ25hbE1hbmFnZXIoKTtcbiAqIGNvbnN0IHVzZXJDYW5jZWwgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG4gKiBjb25zdCB1bm1vdW50U2lnbmFsID0gY29tcG9uZW50VW5tb3VudFNpZ25hbDtcbiAqXG4gKiAvLyBSZXF1ZXN0IHdpbGwgYmUgY2FuY2VsbGVkIGlmIGVpdGhlciB1c2VyIGNsaWNrcyBjYW5jZWwgT1IgY29tcG9uZW50IHVubW91bnRzXG4gKiBjb25zdCBjb21iaW5lZCA9IG1hbmFnZXIuY3JlYXRlQ29tYmluZWRTaWduYWwoW1xuICogICB1c2VyQ2FuY2VsLnNpZ25hbCxcbiAqICAgdW5tb3VudFNpZ25hbFxuICogXSk7XG4gKlxuICogZmV0Y2goJy9hcGkvZGF0YScsIHsgc2lnbmFsOiBjb21iaW5lZC5zaWduYWwgfSk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGNsYXNzIFNpZ25hbE1hbmFnZXIge1xuICAvKipcbiAgICogQ3JlYXRlcyBhIGNvbWJpbmVkIEFib3J0Q29udHJvbGxlciB0aGF0IGFib3J0cyB3aGVuIGFueSBzb3VyY2Ugc2lnbmFsIGFib3J0cy5cbiAgICpcbiAgICogVGhpcyBtZXRob2QgaW1wbGVtZW50cyB0aGUgXCJhbnlcIiBwYXR0ZXJuIGZvciBjYW5jZWxsYXRpb246IHRoZSBjb21iaW5lZCBzaWduYWxcbiAgICogd2lsbCBhYm9ydCBhcyBzb29uIGFzIEFOWSBvZiB0aGUgc291cmNlIHNpZ25hbHMgYWJvcnQuIFRoaXMgaXMgdXNlZnVsIGZvclxuICAgKiBjb29yZGluYXRpbmcgbXVsdGlwbGUgY2FuY2VsbGF0aW9uIGNvbmRpdGlvbnM6XG4gICAqIC0gVXNlciBjbGlja3MgY2FuY2VsIGJ1dHRvblxuICAgKiAtIFJlcXVlc3QgdGltZW91dCBleHBpcmVzXG4gICAqIC0gQ29tcG9uZW50IHVubW91bnRzXG4gICAqIC0gUGFyZW50IHJlcXVlc3QgaXMgY2FuY2VsbGVkXG4gICAqXG4gICAqICoqRWFybHkgQWJvcnQgT3B0aW1pemF0aW9uOioqXG4gICAqIElmIGFueSBzb3VyY2Ugc2lnbmFsIGlzIGFscmVhZHkgYWJvcnRlZCB3aGVuIHRoaXMgbWV0aG9kIGlzIGNhbGxlZCxcbiAgICogdGhlIHJldHVybmVkIGNvbnRyb2xsZXIgaXMgaW1tZWRpYXRlbHkgYWJvcnRlZCB3aXRob3V0IHNldHRpbmcgdXAgbGlzdGVuZXJzLlxuICAgKlxuICAgKiAqKk1lbW9yeSBNYW5hZ2VtZW50OioqXG4gICAqIEV2ZW50IGxpc3RlbmVycyBhcmUgcmVnaXN0ZXJlZCB3aXRoIGB7IG9uY2U6IHRydWUgfWAgdG8gcHJldmVudCBtZW1vcnkgbGVha3MsXG4gICAqIGFzIHRoZXkgYXV0b21hdGljYWxseSBjbGVhbiB1cCBhZnRlciBmaXJpbmcuXG4gICAqXG4gICAqIEBwYXJhbSBzaWduYWxzIC0gQXJyYXkgb2YgQWJvcnRTaWduYWxzIHRvIGNvbWJpbmUgKHVuZGVmaW5lZCB2YWx1ZXMgYXJlIGlnbm9yZWQpXG4gICAqIEByZXR1cm5zIEEgbmV3IEFib3J0Q29udHJvbGxlciB0aGF0IGFib3J0cyB3aGVuIGFueSBzb3VyY2Ugc2lnbmFsIGFib3J0c1xuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBVc2VyIGNhbmNlbGxhdGlvbiArIHRpbWVvdXQ6XG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgdXNlckNvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG4gICAqIGNvbnN0IHRpbWVvdXQgPSBtYW5hZ2VyLmNyZWF0ZVRpbWVvdXRTaWduYWwoMzAwMDApO1xuICAgKlxuICAgKiBjb25zdCBjb21iaW5lZCA9IG1hbmFnZXIuY3JlYXRlQ29tYmluZWRTaWduYWwoW1xuICAgKiAgIHVzZXJDb250cm9sbGVyLnNpZ25hbCxcbiAgICogICB0aW1lb3V0LnNpZ25hbFxuICAgKiBdKTtcbiAgICpcbiAgICogLy8gUmVxdWVzdCB3aWxsIGJlIGNhbmNlbGxlZCBhZnRlciAzMHMgT1Igd2hlbiB1c2VyIGNsaWNrcyBjYW5jZWxcbiAgICogZmV0Y2goJy9hcGkvZGF0YScsIHsgc2lnbmFsOiBjb21iaW5lZC5zaWduYWwgfSk7XG4gICAqIGBgYFxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBSZWFjdCBjb21wb25lbnQgd2l0aCBjbGVhbnVwOlxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAqICAgY29uc3QgY29udHJvbGxlciA9IG5ldyBBYm9ydENvbnRyb2xsZXIoKTtcbiAgICpcbiAgICogICBjb25zdCBjb21iaW5lZCA9IG1hbmFnZXIuY3JlYXRlQ29tYmluZWRTaWduYWwoW1xuICAgKiAgICAgY29udHJvbGxlci5zaWduYWwsXG4gICAqICAgICB1bm1vdW50U2lnbmFsICAvLyBGcm9tIGNvbXBvbmVudCBsaWZlY3ljbGVcbiAgICogICBdKTtcbiAgICpcbiAgICogICBmZXRjaERhdGEoY29tYmluZWQuc2lnbmFsKTtcbiAgICpcbiAgICogICByZXR1cm4gKCkgPT4gY29udHJvbGxlci5hYm9ydCgpOyAvLyBDbGVhbnVwXG4gICAqIH0sIFtdKTtcbiAgICogYGBgXG4gICAqL1xuICBjcmVhdGVDb21iaW5lZFNpZ25hbChcbiAgICBzaWduYWxzOiBBcnJheTxBYm9ydFNpZ25hbCB8IHVuZGVmaW5lZD5cbiAgKTogQWJvcnRDb250cm9sbGVyIHtcbiAgICBjb25zdCBjb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpO1xuXG4gICAgZm9yIChjb25zdCBzaWduYWwgb2Ygc2lnbmFscykge1xuICAgICAgaWYgKHNpZ25hbCkge1xuICAgICAgICBpZiAoc2lnbmFsLmFib3J0ZWQpIHtcbiAgICAgICAgICBjb250cm9sbGVyLmFib3J0KHNpZ25hbC5yZWFzb24pO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICB9XG5cbiAgICAgICAgc2lnbmFsLmFkZEV2ZW50TGlzdGVuZXIoXG4gICAgICAgICAgJ2Fib3J0JyxcbiAgICAgICAgICAoKSA9PiB7XG4gICAgICAgICAgICBjb250cm9sbGVyLmFib3J0KHNpZ25hbC5yZWFzb24pO1xuICAgICAgICAgIH0sXG4gICAgICAgICAgeyBvbmNlOiB0cnVlIH1cbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICByZXR1cm4gY29udHJvbGxlcjtcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIGFuIEFib3J0Q29udHJvbGxlciB0aGF0IGF1dG9tYXRpY2FsbHkgYWJvcnRzIGFmdGVyIGEgc3BlY2lmaWVkIHRpbWVvdXQuXG4gICAqXG4gICAqIFRoaXMgbWV0aG9kIGNyZWF0ZXMgYSB0aW1lLWJhc2VkIGNhbmNlbGxhdGlvbiBtZWNoYW5pc20gdXNlZnVsIGZvciBpbXBsZW1lbnRpbmdcbiAgICogcmVxdWVzdCB0aW1lb3V0cyBhbmQgZGVhZGxpbmVzLiBUaGUgc2lnbmFsIHdpbGwgYXV0b21hdGljYWxseSBhYm9ydCBhZnRlciB0aGVcbiAgICogc3BlY2lmaWVkIGR1cmF0aW9uLCBwcm92aWRpbmcgYSBjb25zaXN0ZW50IHRpbWVvdXQgZXhwZXJpZW5jZS5cbiAgICpcbiAgICogKipBdXRvbWF0aWMgQ2xlYW51cDoqKlxuICAgKiBJZiB0aGUgc2lnbmFsIGlzIGFib3J0ZWQgYnkgb3RoZXIgbWVhbnMgYmVmb3JlIHRoZSB0aW1lb3V0IGV4cGlyZXMsIHRoZSBpbnRlcm5hbFxuICAgKiBzZXRUaW1lb3V0IGlzIGF1dG9tYXRpY2FsbHkgY2xlYXJlZCB0byBwcmV2ZW50IG1lbW9yeSBsZWFrcy5cbiAgICpcbiAgICogKipBYm9ydCBSZWFzb246KipcbiAgICogVGhlIGFib3J0IHJlYXNvbiBpbmNsdWRlcyB0aGUgdGltZW91dCBkdXJhdGlvbiBmb3IgZGVidWdnaW5nIHB1cnBvc2VzOlxuICAgKiBgXCJSZXF1ZXN0IHRpbWVvdXQgYWZ0ZXIge3RpbWVvdXR9bXNcImBcbiAgICpcbiAgICogQHBhcmFtIHRpbWVvdXQgLSBUaW1lb3V0IGR1cmF0aW9uIGluIG1pbGxpc2Vjb25kc1xuICAgKiBAcmV0dXJucyBBbiBBYm9ydENvbnRyb2xsZXIgdGhhdCB3aWxsIGFib3J0IGFmdGVyIHRoZSB0aW1lb3V0XG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIFNpbXBsZSByZXF1ZXN0IHRpbWVvdXQ6XG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgbWFuYWdlciA9IG5ldyBTaWduYWxNYW5hZ2VyKCk7XG4gICAqIGNvbnN0IHRpbWVvdXQgPSBtYW5hZ2VyLmNyZWF0ZVRpbWVvdXRTaWduYWwoNTAwMCk7IC8vIDUgc2Vjb25kc1xuICAgKlxuICAgKiB0cnkge1xuICAgKiAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2goJy9hcGkvc2xvdy1lbmRwb2ludCcsIHtcbiAgICogICAgIHNpZ25hbDogdGltZW91dC5zaWduYWxcbiAgICogICB9KTtcbiAgICogICBjb25zdCBkYXRhID0gYXdhaXQgcmVzcG9uc2UuanNvbigpO1xuICAgKiB9IGNhdGNoIChlcnJvcikge1xuICAgKiAgIGlmIChlcnJvci5uYW1lID09PSAnQWJvcnRFcnJvcicpIHtcbiAgICogICAgIGNvbnNvbGUuZXJyb3IoJ1JlcXVlc3QgdGltZWQgb3V0IGFmdGVyIDUgc2Vjb25kcycpO1xuICAgKiAgIH1cbiAgICogfVxuICAgKiBgYGBcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogRGlmZmVyZW50IHRpbWVvdXRzIGZvciBkaWZmZXJlbnQgb3BlcmF0aW9uczpcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiAvLyBTaG9ydCB0aW1lb3V0IGZvciBxdWljayBvcGVyYXRpb25zXG4gICAqIGNvbnN0IHF1aWNrVGltZW91dCA9IG1hbmFnZXIuY3JlYXRlVGltZW91dFNpZ25hbCgyMDAwKTtcbiAgICogYXdhaXQgZmV0Y2goJy9hcGkvaGVhbHRoJywgeyBzaWduYWw6IHF1aWNrVGltZW91dC5zaWduYWwgfSk7XG4gICAqXG4gICAqIC8vIExvbmcgdGltZW91dCBmb3IgaGVhdnkgb3BlcmF0aW9uc1xuICAgKiBjb25zdCBsb25nVGltZW91dCA9IG1hbmFnZXIuY3JlYXRlVGltZW91dFNpZ25hbCg2MDAwMCk7XG4gICAqIGF3YWl0IGZldGNoKCcvYXBpL2V4cG9ydCcsIHsgc2lnbmFsOiBsb25nVGltZW91dC5zaWduYWwgfSk7XG4gICAqIGBgYFxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBNYW51YWwgY2FuY2VsbGF0aW9uIGJlZm9yZSB0aW1lb3V0OlxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGNvbnN0IHRpbWVvdXQgPSBtYW5hZ2VyLmNyZWF0ZVRpbWVvdXRTaWduYWwoMzAwMDApO1xuICAgKlxuICAgKiAvLyBJZiB1c2VyIGNhbmNlbHMsIHRpbWVvdXQgaXMgYXV0b21hdGljYWxseSBjbGVhbmVkIHVwXG4gICAqIHRpbWVvdXQuYWJvcnQoJ1VzZXIgY2FuY2VsbGVkJyk7XG4gICAqIC8vIEludGVybmFsIHNldFRpbWVvdXQgaXMgY2xlYXJlZCwgbm8gbWVtb3J5IGxlYWtcbiAgICogYGBgXG4gICAqL1xuICBjcmVhdGVUaW1lb3V0U2lnbmFsKHRpbWVvdXQ6IG51bWJlcik6IEFib3J0Q29udHJvbGxlciB7XG4gICAgY29uc3QgY29udHJvbGxlciA9IG5ldyBBYm9ydENvbnRyb2xsZXIoKTtcblxuICAgIGNvbnN0IHRpbWVvdXRJZCA9IHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgY29udHJvbGxlci5hYm9ydChgUmVxdWVzdCB0aW1lb3V0IGFmdGVyICR7dGltZW91dH1tc2ApO1xuICAgIH0sIHRpbWVvdXQpO1xuXG4gICAgLy8gQ2xlYW4gdXAgdGltZW91dCB3aGVuIHNpZ25hbCBpcyBhYm9ydGVkXG4gICAgY29udHJvbGxlci5zaWduYWwuYWRkRXZlbnRMaXN0ZW5lcihcbiAgICAgICdhYm9ydCcsXG4gICAgICAoKSA9PiB7XG4gICAgICAgIGNsZWFyVGltZW91dCh0aW1lb3V0SWQpO1xuICAgICAgfSxcbiAgICAgIHsgb25jZTogdHJ1ZSB9XG4gICAgKTtcblxuICAgIHJldHVybiBjb250cm9sbGVyO1xuICB9XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBSZXNwb25zZURhdGEgfSBmcm9tICcuLi9Nb2RlbHMnO1xuXG4vKipcbiAqIFBhcnNlcyBIVFRQIHJlc3BvbnNlIGJvZGllcyBpbnRvIGFwcHJvcHJpYXRlIEphdmFTY3JpcHQgdHlwZXMgYmFzZWQgb24gQ29udGVudC1UeXBlIGhlYWRlcnMuXG4gKlxuICogVGhpcyBpbnRlcm5hbCBjbGFzcyBoYW5kbGVzIHRoZSBjb21wbGV4aXR5IG9mIGNvbnZlcnRpbmcgdmFyaW91cyBIVFRQIHJlc3BvbnNlIGZvcm1hdHNcbiAqIGludG8gdXNhYmxlIEphdmFTY3JpcHQgZGF0YSBzdHJ1Y3R1cmVzLiBJdCBpbnRlbGxpZ2VudGx5IGRldGVjdHMgdGhlIHJlc3BvbnNlIHR5cGVcbiAqIGFuZCBhcHBsaWVzIHRoZSBhcHByb3ByaWF0ZSBwYXJzaW5nIHN0cmF0ZWd5LlxuICpcbiAqIFN1cHBvcnRlZCBDb250ZW50IFR5cGVzOlxuICogLSAqKmFwcGxpY2F0aW9uL2pzb24qKjogUGFyc2VkIGFzIEpTT04gb2JqZWN0cy9hcnJheXNcbiAqIC0gKip0ZXh0LyoqKiAodGV4dC9wbGFpbiwgdGV4dC9odG1sLCBldGMuKTogUmV0dXJuZWQgYXMgc3RyaW5nXG4gKiAtICoqYXBwbGljYXRpb24vb2N0ZXQtc3RyZWFtKio6IFJldHVybmVkIGFzIEJsb2IgZm9yIGJpbmFyeSBkYXRhXG4gKiAtICoqVW5rbm93bi9NaXNzaW5nKio6IEF0dGVtcHRzIEpTT04gcGFyc2UsIGZhbGxzIGJhY2sgdG8gdGV4dFxuICpcbiAqIEtleSBGZWF0dXJlczpcbiAqIC0gKipBdXRvbWF0aWMgRGV0ZWN0aW9uKio6IFVzZXMgQ29udGVudC1UeXBlIGhlYWRlciB0byBkZXRlcm1pbmUgcGFyc2luZyBzdHJhdGVneVxuICogLSAqKkdyYWNlZnVsIEZhbGxiYWNrKio6IEF0dGVtcHRzIEpTT04gcGFyc2luZyBmb3IgdW5rbm93biB0eXBlcywgZmFsbHMgYmFjayB0byB0ZXh0XG4gKiAtICoqQmluYXJ5IFN1cHBvcnQqKjogSGFuZGxlcyBiaW5hcnkgZGF0YSB2aWEgQmxvYlxuICogLSAqKlR5cGUgU2FmZXR5Kio6IFJldHVybnMgdHlwZWQgUmVzcG9uc2VEYXRhIHVuaW9uXG4gKlxuICogQGludGVybmFsIFRoaXMgY2xhc3MgaXMgbm90IGV4cG9ydGVkIGZyb20gdGhlIHB1YmxpYyBBUElcbiAqXG4gKiBAZXhhbXBsZVxuICogUGFyc2luZyBkaWZmZXJlbnQgcmVzcG9uc2UgdHlwZXM6XG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBwYXJzZXIgPSBuZXcgUmVzcG9uc2VQYXJzZXIoKTtcbiAqXG4gKiAvLyBKU09OIHJlc3BvbnNlXG4gKiBjb25zdCBqc29uUmVzcG9uc2UgPSBuZXcgUmVzcG9uc2UoJ3tcImlkXCI6IDF9Jywge1xuICogICBoZWFkZXJzOiB7ICdDb250ZW50LVR5cGUnOiAnYXBwbGljYXRpb24vanNvbicgfVxuICogfSk7XG4gKiBjb25zdCBkYXRhID0gYXdhaXQgcGFyc2VyLnBhcnNlUmVzcG9uc2UoanNvblJlc3BvbnNlKTtcbiAqIC8vID0+IHsgaWQ6IDEgfVxuICpcbiAqIC8vIFRleHQgcmVzcG9uc2VcbiAqIGNvbnN0IHRleHRSZXNwb25zZSA9IG5ldyBSZXNwb25zZSgnSGVsbG8gV29ybGQnLCB7XG4gKiAgIGhlYWRlcnM6IHsgJ0NvbnRlbnQtVHlwZSc6ICd0ZXh0L3BsYWluJyB9XG4gKiB9KTtcbiAqIGNvbnN0IHRleHQgPSBhd2FpdCBwYXJzZXIucGFyc2VSZXNwb25zZSh0ZXh0UmVzcG9uc2UpO1xuICogLy8gPT4gXCJIZWxsbyBXb3JsZFwiXG4gKlxuICogLy8gQmluYXJ5IHJlc3BvbnNlXG4gKiBjb25zdCBibG9iUmVzcG9uc2UgPSBuZXcgUmVzcG9uc2UoYmluYXJ5RGF0YSwge1xuICogICBoZWFkZXJzOiB7ICdDb250ZW50LVR5cGUnOiAnYXBwbGljYXRpb24vb2N0ZXQtc3RyZWFtJyB9XG4gKiB9KTtcbiAqIGNvbnN0IGJsb2IgPSBhd2FpdCBwYXJzZXIucGFyc2VSZXNwb25zZShibG9iUmVzcG9uc2UpO1xuICogLy8gPT4gQmxvYiB7IC4uLiB9XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGNsYXNzIFJlc3BvbnNlUGFyc2VyIHtcbiAgLyoqXG4gICAqIFBhcnNlcyB0aGUgSFRUUCByZXNwb25zZSBib2R5IGludG8gYW4gYXBwcm9wcmlhdGUgSmF2YVNjcmlwdCB0eXBlLlxuICAgKlxuICAgKiBUaGUgcGFyc2luZyBzdHJhdGVneSBpcyBkZXRlcm1pbmVkIGJ5IHRoZSBDb250ZW50LVR5cGUgaGVhZGVyOlxuICAgKiAxLiAqKkpTT04qKiAoYXBwbGljYXRpb24vanNvbik6IENhbGxzIGByZXNwb25zZS5qc29uKClgXG4gICAqIDIuICoqVGV4dCoqICh0ZXh0LyopOiBDYWxscyBgcmVzcG9uc2UudGV4dCgpYFxuICAgKiAzLiAqKkJpbmFyeSoqIChhcHBsaWNhdGlvbi9vY3RldC1zdHJlYW0pOiBDYWxscyBgcmVzcG9uc2UuYmxvYigpYFxuICAgKiA0LiAqKlVua25vd24qKjogUmVhZHMgYXMgdGV4dCwgYXR0ZW1wdHMgSlNPTiBwYXJzZSwgZmFsbHMgYmFjayB0byByYXcgdGV4dFxuICAgKlxuICAgKiAqKkZhbGxiYWNrIEJlaGF2aW9yOioqXG4gICAqIEZvciByZXNwb25zZXMgd2l0aG91dCBhIENvbnRlbnQtVHlwZSBoZWFkZXIgb3Igd2l0aCB1bmtub3duIHR5cGVzLCB0aGUgcGFyc2VyXG4gICAqIGF0dGVtcHRzIHRvIHBhcnNlIGFzIEpTT04gZmlyc3QgKGNvbW1vbiBmb3IgQVBJcyB0aGF0IGRvbid0IHNldCBwcm9wZXIgaGVhZGVycykuXG4gICAqIElmIEpTT04gcGFyc2luZyBmYWlscywgaXQgcmV0dXJucyB0aGUgcmF3IHRleHQuXG4gICAqXG4gICAqIEBwYXJhbSByZXNwb25zZSAtIFRoZSBGZXRjaCBBUEkgUmVzcG9uc2Ugb2JqZWN0IHRvIHBhcnNlXG4gICAqIEByZXR1cm5zIFByb21pc2UgcmVzb2x2aW5nIHRvIHRoZSBwYXJzZWQgcmVzcG9uc2UgZGF0YVxuICAgKiBAcmV0dXJucyBDYW4gYmU6IEpTT04gb2JqZWN0L2FycmF5LCBzdHJpbmcsIG9yIEJsb2IgZGVwZW5kaW5nIG9uIENvbnRlbnQtVHlwZVxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBBUEkgcmVzcG9uc2UgcGFyc2luZzpcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiBjb25zdCByZXNwb25zZSA9IGF3YWl0IGZldGNoKCcvYXBpL3VzZXJzJyk7XG4gICAqIGNvbnN0IGRhdGEgPSBhd2FpdCBwYXJzZXIucGFyc2VSZXNwb25zZShyZXNwb25zZSk7XG4gICAqXG4gICAqIGlmICh0eXBlb2YgZGF0YSA9PT0gJ3N0cmluZycpIHtcbiAgICogICBjb25zb2xlLmxvZygnVGV4dCByZXNwb25zZTonLCBkYXRhKTtcbiAgICogfSBlbHNlIGlmIChkYXRhIGluc3RhbmNlb2YgQmxvYikge1xuICAgKiAgIGNvbnNvbGUubG9nKCdCaW5hcnkgcmVzcG9uc2U6JywgZGF0YS5zaXplLCAnYnl0ZXMnKTtcbiAgICogfSBlbHNlIHtcbiAgICogICBjb25zb2xlLmxvZygnSlNPTiByZXNwb25zZTonLCBkYXRhKTtcbiAgICogfVxuICAgKiBgYGBcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogSGFuZGxpbmcgZGlmZmVyZW50IGNvbnRlbnQgdHlwZXM6XG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogLy8gQ1NWIGZpbGUgZG93bmxvYWRcbiAgICogY29uc3QgY3N2UmVzcG9uc2UgPSBhd2FpdCBmZXRjaCgnL2FwaS9leHBvcnQuY3N2Jyk7XG4gICAqIGNvbnN0IGJsb2IgPSBhd2FpdCBwYXJzZXIucGFyc2VSZXNwb25zZShjc3ZSZXNwb25zZSk7XG4gICAqIC8vIFJldHVybnMgQmxvYiBmb3IgZG93bmxvYWRcbiAgICpcbiAgICogLy8gSlNPTiBBUElcbiAgICogY29uc3QganNvblJlc3BvbnNlID0gYXdhaXQgZmV0Y2goJy9hcGkvdXNlcnMnKTtcbiAgICogY29uc3QgdXNlcnMgPSBhd2FpdCBwYXJzZXIucGFyc2VSZXNwb25zZShqc29uUmVzcG9uc2UpO1xuICAgKiAvLyBSZXR1cm5zIHBhcnNlZCBKU09OIGFycmF5XG4gICAqXG4gICAqIC8vIFBsYWluIHRleHQgbG9nc1xuICAgKiBjb25zdCBsb2dSZXNwb25zZSA9IGF3YWl0IGZldGNoKCcvYXBpL2xvZ3MnKTtcbiAgICogY29uc3QgbG9ncyA9IGF3YWl0IHBhcnNlci5wYXJzZVJlc3BvbnNlKGxvZ1Jlc3BvbnNlKTtcbiAgICogLy8gUmV0dXJucyBzdHJpbmdcbiAgICogYGBgXG4gICAqL1xuICBhc3luYyBwYXJzZVJlc3BvbnNlKHJlc3BvbnNlOiBSZXNwb25zZSk6IFByb21pc2U8UmVzcG9uc2VEYXRhPiB7XG4gICAgY29uc3QgY29udGVudFR5cGUgPSByZXNwb25zZS5oZWFkZXJzLmdldCgnY29udGVudC10eXBlJyk7XG5cbiAgICBpZiAoY29udGVudFR5cGU/LmluY2x1ZGVzKCdhcHBsaWNhdGlvbi9qc29uJykpIHtcbiAgICAgIHJldHVybiByZXNwb25zZS5qc29uKCk7XG4gICAgfSBlbHNlIGlmIChjb250ZW50VHlwZT8uaW5jbHVkZXMoJ3RleHQvJykpIHtcbiAgICAgIHJldHVybiByZXNwb25zZS50ZXh0KCk7XG4gICAgfSBlbHNlIGlmIChjb250ZW50VHlwZT8uaW5jbHVkZXMoJ2FwcGxpY2F0aW9uL29jdGV0LXN0cmVhbScpKSB7XG4gICAgICByZXR1cm4gcmVzcG9uc2UuYmxvYigpO1xuICAgIH0gZWxzZSB7XG4gICAgICAvLyBUcnkgSlNPTiBmaXJzdCwgZmFsbGJhY2sgdG8gdGV4dFxuICAgICAgY29uc3QgdGV4dCA9IGF3YWl0IHJlc3BvbnNlLnRleHQoKTtcblxuICAgICAgdHJ5IHtcbiAgICAgICAgcmV0dXJuIEpTT04ucGFyc2UodGV4dCkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gICAgICB9IGNhdGNoIHtcbiAgICAgICAgcmV0dXJuIHRleHQ7XG4gICAgICB9XG4gICAgfVxuICB9XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBRdWVyeVBhcmFtcyB9IGZyb20gJy4uL01vZGVscyc7XG5cbi8qKlxuICogQ29uc3RydWN0cyBmdWxseS1xdWFsaWZpZWQgVVJMcyB3aXRoIHByb3BlciBxdWVyeSBwYXJhbWV0ZXIgZW5jb2RpbmcuXG4gKlxuICogVGhpcyBpbnRlcm5hbCBjbGFzcyBoYW5kbGVzIHRoZSBjb21wbGV4aXR5IG9mIFVSTCBjb25zdHJ1Y3Rpb24sIGNvbWJpbmluZ1xuICogYmFzZSBVUkxzLCBlbmRwb2ludHMsIGFuZCBxdWVyeSBwYXJhbWV0ZXJzIHdpdGggcHJvcGVyIGVuY29kaW5nIGFuZCBhcnJheVxuICogaGFuZGxpbmcuIEl0IHVzZXMgdGhlIG5hdGl2ZSBVUkwgQVBJIGZvciByZWxpYWJsZSBVUkwgY29tcG9zaXRpb24uXG4gKlxuICogS2V5IEZlYXR1cmVzOlxuICogLSAqKlVSTCBDb21wb3NpdGlvbioqOiBTYWZlbHkgY29tYmluZXMgYmFzZSBVUkwgYW5kIGVuZHBvaW50IHBhdGhzXG4gKiAtICoqUXVlcnkgUGFyYW1ldGVyIEVuY29kaW5nKio6IEF1dG9tYXRpYyBlbmNvZGluZyBvZiBzcGVjaWFsIGNoYXJhY3RlcnNcbiAqIC0gKipBcnJheSBTdXBwb3J0Kio6IEhhbmRsZXMgYXJyYXkgcGFyYW1ldGVycyB3aXRoIG11bHRpcGxlIHZhbHVlc1xuICogLSAqKk51bGwvVW5kZWZpbmVkIEZpbHRlcmluZyoqOiBBdXRvbWF0aWNhbGx5IHNraXBzIG51bGwvdW5kZWZpbmVkIHZhbHVlc1xuICogLSAqKlR5cGUgU2FmZXR5Kio6IFN0cm9uZ2x5IHR5cGVkIHF1ZXJ5IHBhcmFtZXRlciB2YWx1ZXNcbiAqXG4gKiBBcnJheSBIYW5kbGluZzpcbiAqIEFycmF5IHBhcmFtZXRlcnMgYXJlIHNlcmlhbGl6ZWQgYnkgYXBwZW5kaW5nIHRoZSBzYW1lIGtleSBtdWx0aXBsZSB0aW1lczpcbiAqIGA/dGFncz1qYXZhc2NyaXB0JnRhZ3M9dHlwZXNjcmlwdCZ0YWdzPXJlYWN0YFxuICpcbiAqIEBpbnRlcm5hbCBUaGlzIGNsYXNzIGlzIG5vdCBleHBvcnRlZCBmcm9tIHRoZSBwdWJsaWMgQVBJXG4gKlxuICogQGV4YW1wbGVcbiAqIEJ1aWxkaW5nIFVSTHMgd2l0aCBxdWVyeSBwYXJhbWV0ZXJzOlxuICogYGBgdHlwZXNjcmlwdFxuICogY29uc3QgYnVpbGRlciA9IG5ldyBVcmxCdWlsZGVyKCk7XG4gKlxuICogY29uc3QgdXJsID0gYnVpbGRlci5idWlsZFVSTChcbiAqICAgJ2h0dHBzOi8vYXBpLmV4YW1wbGUuY29tJyxcbiAqICAgJy91c2VycycsXG4gKiAgIHsgYWN0aXZlOiB0cnVlLCByb2xlOiAnYWRtaW4nLCBwYWdlOiAxIH1cbiAqICk7XG4gKiAvLyA9PiBcImh0dHBzOi8vYXBpLmV4YW1wbGUuY29tL3VzZXJzP2FjdGl2ZT10cnVlJnJvbGU9YWRtaW4mcGFnZT0xXCJcbiAqIGBgYFxuICovXG5leHBvcnQgY2xhc3MgVXJsQnVpbGRlciB7XG4gIC8qKlxuICAgKiBCdWlsZHMgYSBjb21wbGV0ZSBVUkwgYnkgY29tYmluaW5nIGJhc2UgVVJMLCBlbmRwb2ludCwgYW5kIHF1ZXJ5IHBhcmFtZXRlcnMuXG4gICAqXG4gICAqIFRoZSBVUkwgY29uc3RydWN0aW9uIHByb2Nlc3M6XG4gICAqIDEuIENvbWJpbmVzIGBiYXNlVVJMYCBhbmQgYGVuZHBvaW50YCB1c2luZyBVUkwgQVBJXG4gICAqIDIuIEl0ZXJhdGVzIHRocm91Z2ggcXVlcnkgcGFyYW1ldGVyc1xuICAgKiAzLiBTa2lwcyBudWxsL3VuZGVmaW5lZCB2YWx1ZXNcbiAgICogNC4gSGFuZGxlcyBhcnJheXMgYnkgYXBwZW5kaW5nIG11bHRpcGxlIHZhbHVlcyB3aXRoIHNhbWUga2V5XG4gICAqIDUuIENvbnZlcnRzIGFsbCB2YWx1ZXMgdG8gc3RyaW5nc1xuICAgKiA2LiBSZXR1cm5zIGZ1bGx5LXF1YWxpZmllZCBVUkwgc3RyaW5nXG4gICAqXG4gICAqICoqUGF0aCBIYW5kbGluZzoqKlxuICAgKiBUaGUgZW5kcG9pbnQgY2FuIGJlIGVpdGhlciByZWxhdGl2ZSBvciBhYnNvbHV0ZTpcbiAgICogLSBSZWxhdGl2ZTogYC91c2Vyc2AgXHUyMTkyIENvbWJpbmVkIHdpdGggYmFzZVVSTFxuICAgKiAtIEFic29sdXRlOiBgaHR0cHM6Ly9vdGhlci1hcGkuY29tL3VzZXJzYCBcdTIxOTIgVXNlcyBhYnNvbHV0ZSBVUkxcbiAgICpcbiAgICogKipFbmNvZGluZzoqKlxuICAgKiBBbGwgcGFyYW1ldGVyIHZhbHVlcyBhcmUgYXV0b21hdGljYWxseSBVUkwtZW5jb2RlZCBieSB0aGUgVVJMIEFQSSxcbiAgICogc28gc3BlY2lhbCBjaGFyYWN0ZXJzIChzcGFjZXMsICYsID0sIGV0Yy4pIGFyZSBzYWZlbHkgaGFuZGxlZC5cbiAgICpcbiAgICogQHBhcmFtIGJhc2VVUkwgLSBCYXNlIFVSTCBmb3IgdGhlIEFQSSAoZS5nLiwgJ2h0dHBzOi8vYXBpLmV4YW1wbGUuY29tJylcbiAgICogQHBhcmFtIGVuZHBvaW50IC0gQVBJIGVuZHBvaW50IHBhdGggcmVsYXRpdmUgdG8gYmFzZVVSTCAoZS5nLiwgJy91c2Vycy8xMjMnKVxuICAgKiBAcGFyYW0gcGFyYW1zIC0gT3B0aW9uYWwgcXVlcnkgcGFyYW1ldGVycyBhcyBrZXktdmFsdWUgcGFpcnNcbiAgICogQHJldHVybnMgVGhlIGZ1bGx5LXF1YWxpZmllZCBVUkwgc3RyaW5nIHdpdGggZW5jb2RlZCBxdWVyeSBwYXJhbWV0ZXJzXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIEJhc2ljIFVSTCBjb25zdHJ1Y3Rpb246XG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgdXJsID0gYnVpbGRlci5idWlsZFVSTChcbiAgICogICAnaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20nLFxuICAgKiAgICcvc2VhcmNoJyxcbiAgICogICB7IHE6ICdoZWxsbyB3b3JsZCcsIGxpbWl0OiAxMCB9XG4gICAqICk7XG4gICAqIC8vID0+IFwiaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20vc2VhcmNoP3E9aGVsbG8rd29ybGQmbGltaXQ9MTBcIlxuICAgKiBgYGBcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogQXJyYXkgcGFyYW1ldGVyczpcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiBjb25zdCB1cmwgPSBidWlsZGVyLmJ1aWxkVVJMKFxuICAgKiAgICdodHRwczovL2FwaS5leGFtcGxlLmNvbScsXG4gICAqICAgJy9wb3N0cycsXG4gICAqICAgeyB0YWdzOiBbJ2phdmFzY3JpcHQnLCAndHlwZXNjcmlwdCcsICdyZWFjdCddIH1cbiAgICogKTtcbiAgICogLy8gPT4gXCJodHRwczovL2FwaS5leGFtcGxlLmNvbS9wb3N0cz90YWdzPWphdmFzY3JpcHQmdGFncz10eXBlc2NyaXB0JnRhZ3M9cmVhY3RcIlxuICAgKiBgYGBcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogTnVsbC91bmRlZmluZWQgaGFuZGxpbmc6XG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgdXJsID0gYnVpbGRlci5idWlsZFVSTChcbiAgICogICAnaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20nLFxuICAgKiAgICcvdXNlcnMnLFxuICAgKiAgIHtcbiAgICogICAgIG5hbWU6ICdKb2huJyxcbiAgICogICAgIGFnZTogbnVsbCwgICAgICAgIC8vIFNraXBwZWRcbiAgICogICAgIGVtYWlsOiB1bmRlZmluZWQgIC8vIFNraXBwZWRcbiAgICogICB9XG4gICAqICk7XG4gICAqIC8vID0+IFwiaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20vdXNlcnM/bmFtZT1Kb2huXCJcbiAgICogYGBgXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIFNwZWNpYWwgY2hhcmFjdGVycyBlbmNvZGluZzpcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiBjb25zdCB1cmwgPSBidWlsZGVyLmJ1aWxkVVJMKFxuICAgKiAgICdodHRwczovL2FwaS5leGFtcGxlLmNvbScsXG4gICAqICAgJy9zZWFyY2gnLFxuICAgKiAgIHsgcTogJ2ZvbyAmIGJhcicsIGNhdGVnb3J5OiAnY29kZS9leGFtcGxlcycgfVxuICAgKiApO1xuICAgKiAvLyA9PiBcImh0dHBzOi8vYXBpLmV4YW1wbGUuY29tL3NlYXJjaD9xPWZvbyslMjYrYmFyJmNhdGVnb3J5PWNvZGUlMkZleGFtcGxlc1wiXG4gICAqIGBgYFxuICAgKi9cbiAgYnVpbGRVUkwoYmFzZVVSTDogc3RyaW5nLCBlbmRwb2ludDogc3RyaW5nLCBwYXJhbXM/OiBRdWVyeVBhcmFtcyk6IHN0cmluZyB7XG4gICAgLy8gRW5zdXJlIGVuZHBvaW50IGhhcyBsZWFkaW5nIHNsYXNoIGZvciBwcm9wZXIgVVJMIHJlc29sdXRpb25cbiAgICBjb25zdCBub3JtYWxpemVkRW5kcG9pbnQgPSBlbmRwb2ludC5zdGFydHNXaXRoKCcvJykgPyBlbmRwb2ludCA6IGAvJHtlbmRwb2ludH1gO1xuICAgIGNvbnN0IHVybCA9IG5ldyBVUkwobm9ybWFsaXplZEVuZHBvaW50LCBiYXNlVVJMKTtcblxuICAgIGlmIChwYXJhbXMpIHtcbiAgICAgIE9iamVjdC5rZXlzKHBhcmFtcykuZm9yRWFjaChrZXkgPT4ge1xuICAgICAgICBjb25zdCB2YWx1ZSA9IHBhcmFtc1trZXldO1xuXG4gICAgICAgIGlmICh2YWx1ZSAhPT0gdW5kZWZpbmVkICYmIHZhbHVlICE9PSBudWxsKSB7XG4gICAgICAgICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSB7XG4gICAgICAgICAgICB2YWx1ZS5mb3JFYWNoKHYgPT4gdXJsLnNlYXJjaFBhcmFtcy5hcHBlbmQoa2V5LCBTdHJpbmcodikpKTtcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgdXJsLnNlYXJjaFBhcmFtcy5hcHBlbmQoa2V5LCBTdHJpbmcodmFsdWUpKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH1cblxuICAgIHJldHVybiB1cmwudG9TdHJpbmcoKTtcbiAgfVxufVxuIiwgImltcG9ydCB7IGdlbmVyYXRlQ29ycmVsYXRpb25JZCB9IGZyb20gJy4vQ29ycmVsYXRpb25JZEdlbmVyYXRvcic7XG5pbXBvcnQgeyBFcnJvck5vcm1hbGl6ZXIgfSBmcm9tICcuL0Vycm9ycy9FcnJvck5vcm1hbGl6ZXInO1xuaW1wb3J0IHsgSW50ZXJjZXB0b3JNYW5hZ2VyIH0gZnJvbSAnLi9JbnRlcmNlcHRvcnMvSW50ZXJjZXB0b3JNYW5hZ2VyJztcbmltcG9ydCB0eXBlIHtcbiAgUGFnaW5hdGVkUmVzcG9uc2UsXG4gIFBvc3RNb2RlbCxcbiAgUXVlcnlQYXJhbXMsXG4gIFJlc3BvbnNlRGF0YSxcbn0gZnJvbSAnLi9Nb2RlbHMnO1xuaW1wb3J0IHsgUmVxdWVzdE1hbmFnZXIgfSBmcm9tICcuL1JlcXVlc3RNYW5hZ2VyJztcbmltcG9ydCB7IFJldHJ5SGFuZGxlciB9IGZyb20gJy4vUmV0cnkvUmV0cnlIYW5kbGVyJztcbmltcG9ydCB7IFNpZ25hbE1hbmFnZXIgfSBmcm9tICcuL1NpZ25hbHMvU2lnbmFsTWFuYWdlcic7XG5pbXBvcnQgdHlwZSB7IEFwaUVycm9yIH0gZnJvbSAnLi90eXBlcy9BcGlFcnJvcic7XG5pbXBvcnQgdHlwZSB7IEVycm9ySW50ZXJjZXB0b3IgfSBmcm9tICcuL3R5cGVzL0Vycm9ySW50ZXJjZXB0b3InO1xuaW1wb3J0IHR5cGUgeyBFcnJvclJlc3BvbnNlRGF0YSB9IGZyb20gJy4vdHlwZXMvRXJyb3JSZXNwb25zZURhdGEnO1xuaW1wb3J0IHR5cGUgeyBSZXF1ZXN0Q29uZmlnIH0gZnJvbSAnLi90eXBlcy9SZXF1ZXN0Q29uZmlnJztcbmltcG9ydCB0eXBlIHsgUmVxdWVzdEludGVyY2VwdG9yIH0gZnJvbSAnLi90eXBlcy9SZXF1ZXN0SW50ZXJjZXB0b3InO1xuaW1wb3J0IHR5cGUgeyBSZXNwb25zZUludGVyY2VwdG9yIH0gZnJvbSAnLi90eXBlcy9SZXNwb25zZUludGVyY2VwdG9yJztcbmltcG9ydCB7IFJlc3BvbnNlUGFyc2VyIH0gZnJvbSAnLi9VdGlscy9SZXNwb25zZVBhcnNlcic7XG5pbXBvcnQgeyBVcmxCdWlsZGVyIH0gZnJvbSAnLi9VdGlscy9VcmxCdWlsZGVyJztcblxuaW1wb3J0IHR5cGUgeyBBcGlSZXNwb25zZSB9IGZyb20gJ0AvdHlwZXMnO1xuXG4vKipcbiAqIEVudGVycHJpc2UtZ3JhZGUgQVBJIGNsaWVudCB3aXRoIGFkdmFuY2VkIGZlYXR1cmVzIGZvciBwcm9kdWN0aW9uIGFwcGxpY2F0aW9ucy5cbiAqXG4gKiBUaGlzIGNsaWVudCBwcm92aWRlcyBhIHJvYnVzdCwgdHlwZS1zYWZlIGFic3RyYWN0aW9uIG92ZXIgdGhlIEZldGNoIEFQSSB3aXRoOlxuICogLSBDb21wcmVoZW5zaXZlIHJlcXVlc3QvcmVzcG9uc2UvZXJyb3IgaW50ZXJjZXB0b3Igc3lzdGVtXG4gKiAtIEF1dG9tYXRpYyByZXRyeSBsb2dpYyB3aXRoIGV4cG9uZW50aWFsIGJhY2tvZmZcbiAqIC0gUmVxdWVzdCBjb3JyZWxhdGlvbiBJRCB0cmFja2luZyBmb3IgZGlzdHJpYnV0ZWQgdHJhY2luZ1xuICogLSBUb2tlbi1iYXNlZCBhdXRoZW50aWNhdGlvbiB3aXRoIGF1dG9tYXRpYyBoZWFkZXIgaW5qZWN0aW9uXG4gKiAtIFJlcXVlc3QgY2FuY2VsbGF0aW9uIGFuZCB0aW1lb3V0IG1hbmFnZW1lbnRcbiAqIC0gVHlwZS1zYWZlIGVycm9yIGhhbmRsaW5nIHdpdGggc3RydWN0dXJlZCBlcnJvciByZXNwb25zZXNcbiAqIC0gUmVhY3QgUXVlcnkgaW50ZWdyYXRpb24gc3VwcG9ydFxuICpcbiAqIEFyY2hpdGVjdHVyZSBQcmluY2lwbGVzOlxuICogLSAqKlR5cGUgU2FmZXR5Kio6IEZ1bGwgVHlwZVNjcmlwdCBpbmZlcmVuY2Ugd2l0aCBubyBgYW55YCB0eXBlc1xuICogLSAqKkV4dGVuc2liaWxpdHkqKjogSW50ZXJjZXB0b3IgcGF0dGVybiBhbGxvd3MgZm9yIGZsZXhpYmxlIG1pZGRsZXdhcmVcbiAqIC0gKipPYnNlcnZhYmlsaXR5Kio6IEJ1aWx0LWluIGNvcnJlbGF0aW9uIElEcyBmb3IgcmVxdWVzdCB0cmFjaW5nXG4gKiAtICoqUmVzaWxpZW5jZSoqOiBBdXRvbWF0aWMgcmV0cmllcywgdGltZW91dHMsIGFuZCBncmFjZWZ1bCBlcnJvciBoYW5kbGluZ1xuICogLSAqKlBlcmZvcm1hbmNlKio6IFJlcXVlc3QgZGVkdXBsaWNhdGlvbiBhbmQgY2FuY2VsbGF0aW9uIHN1cHBvcnRcbiAqIC0gKipNb2R1bGFyaXR5Kio6IE9yZ2FuaXplZCBpbnRvIGZvY3VzZWQsIHNpbmdsZS1yZXNwb25zaWJpbGl0eSBtb2R1bGVzXG4gKlxuICogQGV4YW1wbGVcbiAqIEJhc2ljIHVzYWdlOlxuICogYGBgdHlwZXNjcmlwdFxuICogY29uc3QgY2xpZW50ID0gbmV3IEFwaUNsaWVudCgnaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20nLCAzMDAwMCk7XG4gKiBjbGllbnQuc2V0QXV0aFRva2VuKCd5b3VyLWp3dC10b2tlbicpO1xuICpcbiAqIC8vIFR5cGUtc2FmZSBHRVQgcmVxdWVzdFxuICogY29uc3QgeyBhcGlEYXRhLCBlcnJvciB9ID0gYXdhaXQgY2xpZW50LmdldDxVc2VyPignL3VzZXJzLzEyMycpO1xuICogaWYgKGVycm9yKSB7XG4gKiAgIGNvbnNvbGUuZXJyb3IoJ0ZhaWxlZCB0byBmZXRjaCB1c2VyOicsIGVycm9yKTtcbiAqIH0gZWxzZSB7XG4gKiAgIGNvbnNvbGUubG9nKCdVc2VyIGRhdGE6JywgZGF0YSk7XG4gKiB9XG4gKiBgYGBcbiAqXG4gKiBAZXhhbXBsZVxuICogV2l0aCBSZWFjdCBRdWVyeTpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGNvbnN0IHF1ZXJ5Q2xpZW50ID0gbmV3IFF1ZXJ5Q2xpZW50KCk7XG4gKiBjb25zdCBhcGlDbGllbnQgPSBuZXcgQXBpQ2xpZW50KHByb2Nlc3MuZW52LkFQSV9CQVNFX1VSTCk7XG4gKlxuICogLy8gQWRkIGxvZ2dpbmcgaW50ZXJjZXB0b3JcbiAqIGFwaUNsaWVudC5hZGRSZXNwb25zZUludGVyY2VwdG9yKGFzeW5jIChyZXNwb25zZSkgPT4ge1xuICogICBjb25zb2xlLmxvZygnUmVzcG9uc2UgcmVjZWl2ZWQ6JywgcmVzcG9uc2UpO1xuICogICByZXR1cm4gcmVzcG9uc2U7XG4gKiB9KTtcbiAqXG4gKiAvLyBVc2UgaW4gcXVlcmllc1xuICogY29uc3QgeyBhcGlEYXRhIH0gPSB1c2VRdWVyeSh7XG4gKiAgIHF1ZXJ5S2V5OiBbJ3VzZXJzJywgdXNlcklkXSxcbiAqICAgcXVlcnlGbjogKCkgPT4gYXBpQ2xpZW50LmdldDxVc2VyPihgL3VzZXJzLyR7dXNlcklkfWApXG4gKiB9KTtcbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBBZHZhbmNlZCBlcnJvciBoYW5kbGluZzpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIE9wdGlvbiAxOiBFcnJvcnMgdGhyb3duIChkZWZhdWx0KVxuICogdHJ5IHtcbiAqICAgY29uc3QgeyBhcGlEYXRhIH0gPSBhd2FpdCBjbGllbnQuZ2V0PFVzZXI+KCcvdXNlcnMvMTIzJyk7XG4gKiAgIGNvbnNvbGUubG9nKGRhdGEpO1xuICogfSBjYXRjaCAoZXJyb3IpIHtcbiAqICAgaWYgKGVycm9yLnN0YXR1cyA9PT0gNDA0KSB7XG4gKiAgICAgY29uc29sZS5sb2coJ1VzZXIgbm90IGZvdW5kJyk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiAvLyBPcHRpb24gMjogRXJyb3JzIHJldHVybmVkIGluIHJlc3BvbnNlXG4gKiBjb25zdCB7IGFwaURhdGEsIGVycm9yIH0gPSBhd2FpdCBjbGllbnQuZ2V0PFVzZXI+KCcvdXNlcnMvMTIzJywge1xuICogICB0aHJvd0Vycm9yczogZmFsc2VcbiAqIH0pO1xuICogaWYgKGVycm9yKSB7XG4gKiAgIGNvbnNvbGUuZXJyb3IoJ1JlcXVlc3QgZmFpbGVkOicsIGVycm9yKTtcbiAqIH1cbiAqIGBgYFxuICpcbiAqIEBwdWJsaWNcbiAqL1xuZXhwb3J0IGNsYXNzIEFwaUNsaWVudCB7XG4gIHByaXZhdGUgcmVhZG9ubHkgYmFzZVVSTDogc3RyaW5nO1xuICBwcml2YXRlIHJlYWRvbmx5IGRlZmF1bHRUaW1lb3V0OiBudW1iZXI7XG4gIHByaXZhdGUgcmVhZG9ubHkgaW50ZXJjZXB0b3JNYW5hZ2VyOiBJbnRlcmNlcHRvck1hbmFnZXIgPVxuICAgIG5ldyBJbnRlcmNlcHRvck1hbmFnZXIoKTtcbiAgcHJpdmF0ZSByZWFkb25seSBzaWduYWxNYW5hZ2VyOiBTaWduYWxNYW5hZ2VyID0gbmV3IFNpZ25hbE1hbmFnZXIoKTtcbiAgcHJpdmF0ZSByZWFkb25seSBlcnJvck5vcm1hbGl6ZXI6IEVycm9yTm9ybWFsaXplciA9IG5ldyBFcnJvck5vcm1hbGl6ZXIoKTtcbiAgcHJpdmF0ZSByZWFkb25seSByZXNwb25zZVBhcnNlcjogUmVzcG9uc2VQYXJzZXIgPSBuZXcgUmVzcG9uc2VQYXJzZXIoKTtcbiAgcHJpdmF0ZSByZWFkb25seSB1cmxCdWlsZGVyOiBVcmxCdWlsZGVyID0gbmV3IFVybEJ1aWxkZXIoKTtcbiAgcHJpdmF0ZSByZWFkb25seSByZXRyeUhhbmRsZXI6IFJldHJ5SGFuZGxlciA9IG5ldyBSZXRyeUhhbmRsZXIoKTtcbiAgcHJpdmF0ZSByZWFkb25seSByZXF1ZXN0TWFuYWdlcjogUmVxdWVzdE1hbmFnZXIgPSBuZXcgUmVxdWVzdE1hbmFnZXIoKTtcbiAgcHJpdmF0ZSBhdXRoVG9rZW46IHN0cmluZyB8IG51bGwgPSBudWxsO1xuICBwcml2YXRlIGNvcnJlbGF0aW9uSWRQcmVmaXg6IHN0cmluZyA9ICdhcGknO1xuICBwcml2YXRlIGluY2x1ZGVDb3JyZWxhdGlvbklkOiBib29sZWFuID0gdHJ1ZTtcblxuICAvKipcbiAgICogQ3JlYXRlcyBhIG5ldyBBUEkgY2xpZW50IGluc3RhbmNlXG4gICAqIEBwYXJhbSBiYXNlVVJMIC0gQmFzZSBVUkwgZm9yIGFsbCBBUEkgcmVxdWVzdHMgKGRlZmF1bHQ6IGVtcHR5IHN0cmluZyBmb3IgcmVsYXRpdmUgVVJMcylcbiAgICogQHBhcmFtIGRlZmF1bHRUaW1lb3V0IC0gRGVmYXVsdCByZXF1ZXN0IHRpbWVvdXQgaW4gbWlsbGlzZWNvbmRzIChkZWZhdWx0OiAzMDAwMClcbiAgICovXG4gIGNvbnN0cnVjdG9yKGJhc2VVUkw6IHN0cmluZyA9ICcnLCBkZWZhdWx0VGltZW91dDogbnVtYmVyID0gMzAwMDApIHtcbiAgICB0aGlzLmJhc2VVUkwgPSBiYXNlVVJMO1xuICAgIHRoaXMuZGVmYXVsdFRpbWVvdXQgPSBkZWZhdWx0VGltZW91dDtcbiAgfVxuXG4gIC8qKlxuICAgKiBTZXRzIHRoZSBwcmVmaXggZm9yIGF1dG8tZ2VuZXJhdGVkIGNvcnJlbGF0aW9uIElEc1xuICAgKiBAcGFyYW0gcHJlZml4IC0gVGhlIHByZWZpeCB0byB1c2UgZm9yIGNvcnJlbGF0aW9uIElEcyAoZS5nLiwgJ2FwaScsICd3ZWInLCAnbW9iaWxlJylcbiAgICovXG4gIHNldENvcnJlbGF0aW9uSWRQcmVmaXgocHJlZml4OiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0aGlzLmNvcnJlbGF0aW9uSWRQcmVmaXggPSBwcmVmaXg7XG4gIH1cblxuICAvKipcbiAgICogRW5hYmxlcyBvciBkaXNhYmxlcyBhdXRvbWF0aWMgY29ycmVsYXRpb24gSUQgZ2VuZXJhdGlvblxuICAgKiBAcGFyYW0gaW5jbHVkZSAtIFdoZXRoZXIgdG8gaW5jbHVkZSBjb3JyZWxhdGlvbiBJRHMgaW4gcmVxdWVzdHNcbiAgICovXG4gIHNldEluY2x1ZGVDb3JyZWxhdGlvbklkKGluY2x1ZGU6IGJvb2xlYW4pOiB2b2lkIHtcbiAgICB0aGlzLmluY2x1ZGVDb3JyZWxhdGlvbklkID0gaW5jbHVkZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZWdpc3RlcnMgYSByZXF1ZXN0IGludGVyY2VwdG9yIHRvIG1vZGlmeSByZXF1ZXN0cyBiZWZvcmUgdGhleSdyZSBzZW50XG4gICAqIEBwYXJhbSBpbnRlcmNlcHRvciAtIEZ1bmN0aW9uIHRvIGludGVyY2VwdCBhbmQgcG90ZW50aWFsbHkgbW9kaWZ5IHJlcXVlc3QgY29uZmlnXG4gICAqIEByZXR1cm5zIEZ1bmN0aW9uIHRvIHVucmVnaXN0ZXIgdGhpcyBpbnRlcmNlcHRvclxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGNvbnN0IHVucmVnaXN0ZXIgPSBjbGllbnQuYWRkUmVxdWVzdEludGVyY2VwdG9yKGFzeW5jIChjb25maWcpID0+IHtcbiAgICogICBjb25maWcuaGVhZGVycyA9IGNvbmZpZy5oZWFkZXJzIHx8IG5ldyBIZWFkZXJzKCk7XG4gICAqICAgY29uZmlnLmhlYWRlcnMuc2V0KCdYLUNsaWVudC1WZXJzaW9uJywgJzEuMC4wJyk7XG4gICAqICAgcmV0dXJuIGNvbmZpZztcbiAgICogfSk7XG4gICAqXG4gICAqIC8vIExhdGVyLCB0byByZW1vdmUgdGhlIGludGVyY2VwdG9yOlxuICAgKiB1bnJlZ2lzdGVyKCk7XG4gICAqIGBgYFxuICAgKi9cbiAgYWRkUmVxdWVzdEludGVyY2VwdG9yKGludGVyY2VwdG9yOiBSZXF1ZXN0SW50ZXJjZXB0b3IpOiAoKSA9PiB2b2lkIHtcbiAgICByZXR1cm4gdGhpcy5pbnRlcmNlcHRvck1hbmFnZXIuYWRkUmVxdWVzdEludGVyY2VwdG9yKGludGVyY2VwdG9yKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZWdpc3RlcnMgYSByZXNwb25zZSBpbnRlcmNlcHRvciB0byBtb2RpZnkgcmVzcG9uc2VzIGJlZm9yZSB0aGV5J3JlIHJldHVybmVkXG4gICAqIEBwYXJhbSBpbnRlcmNlcHRvciAtIEZ1bmN0aW9uIHRvIGludGVyY2VwdCBhbmQgcG90ZW50aWFsbHkgbW9kaWZ5IHJlc3BvbnNlc1xuICAgKiBAcmV0dXJucyBGdW5jdGlvbiB0byB1bnJlZ2lzdGVyIHRoaXMgaW50ZXJjZXB0b3JcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiBjbGllbnQuYWRkUmVzcG9uc2VJbnRlcmNlcHRvcihhc3luYyAocmVzcG9uc2UpID0+IHtcbiAgICogICAvLyBUcmFuc2Zvcm0gZGF0YSBmb3JtYXRcbiAgICogICBpZiAocmVzcG9uc2UuYXBpRGF0YSkge1xuICAgKiAgICAgcmVzcG9uc2UuYXBpRGF0YSA9IGNhbWVsQ2FzZUtleXMocmVzcG9uc2UuYXBpRGF0YSk7XG4gICAqICAgfVxuICAgKiAgIHJldHVybiByZXNwb25zZTtcbiAgICogfSk7XG4gICAqIGBgYFxuICAgKi9cbiAgYWRkUmVzcG9uc2VJbnRlcmNlcHRvcihpbnRlcmNlcHRvcjogUmVzcG9uc2VJbnRlcmNlcHRvcik6ICgpID0+IHZvaWQge1xuICAgIHJldHVybiB0aGlzLmludGVyY2VwdG9yTWFuYWdlci5hZGRSZXNwb25zZUludGVyY2VwdG9yKGludGVyY2VwdG9yKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZWdpc3RlcnMgYW4gZXJyb3IgaW50ZXJjZXB0b3IgdG8gaGFuZGxlIG9yIHRyYW5zZm9ybSBlcnJvcnNcbiAgICogQHBhcmFtIGludGVyY2VwdG9yIC0gRnVuY3Rpb24gdG8gaW50ZXJjZXB0IGFuZCBwb3RlbnRpYWxseSBtb2RpZnkgZXJyb3JzXG4gICAqIEByZXR1cm5zIEZ1bmN0aW9uIHRvIHVucmVnaXN0ZXIgdGhpcyBpbnRlcmNlcHRvclxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGNsaWVudC5hZGRFcnJvckludGVyY2VwdG9yKGFzeW5jIChlcnJvcikgPT4ge1xuICAgKiAgIC8vIExvZyBlcnJvcnMgdG8gbW9uaXRvcmluZyBzZXJ2aWNlXG4gICAqICAgaWYgKGVycm9yLnN0YXR1cyA+PSA1MDApIHtcbiAgICogICAgIGF3YWl0IG1vbml0b3JpbmdTZXJ2aWNlLmxvZ0Vycm9yKGVycm9yKTtcbiAgICogICB9XG4gICAqICAgcmV0dXJuIGVycm9yOyAvLyBSZS10aHJvdyB0aGUgZXJyb3JcbiAgICogfSk7XG4gICAqIGBgYFxuICAgKi9cbiAgYWRkRXJyb3JJbnRlcmNlcHRvcihpbnRlcmNlcHRvcjogRXJyb3JJbnRlcmNlcHRvcik6ICgpID0+IHZvaWQge1xuICAgIHJldHVybiB0aGlzLmludGVyY2VwdG9yTWFuYWdlci5hZGRFcnJvckludGVyY2VwdG9yKGludGVyY2VwdG9yKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTZXRzIHRoZSBhdXRoZW50aWNhdGlvbiB0b2tlbiBmb3Igc3Vic2VxdWVudCByZXF1ZXN0c1xuICAgKiBAcGFyYW0gdG9rZW4gLSBKV1QgdG9rZW4gb3IgbnVsbCB0byBjbGVhciBhdXRoZW50aWNhdGlvblxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIC8vIFNldCB0b2tlbiBhZnRlciBsb2dpblxuICAgKiBjbGllbnQuc2V0QXV0aFRva2VuKGxvZ2luUmVzcG9uc2UuYWNjZXNzVG9rZW4pO1xuICAgKlxuICAgKiAvLyBDbGVhciB0b2tlbiBvbiBsb2dvdXRcbiAgICogY2xpZW50LnNldEF1dGhUb2tlbihudWxsKTtcbiAgICogYGBgXG4gICAqL1xuICBzZXRBdXRoVG9rZW4odG9rZW46IHN0cmluZyB8IG51bGwpOiB2b2lkIHtcbiAgICB0aGlzLmF1dGhUb2tlbiA9IHRva2VuO1xuICB9XG5cbiAgLyoqXG4gICAqIFJldHJpZXZlcyB0aGUgY3VycmVudCBhdXRoZW50aWNhdGlvbiB0b2tlblxuICAgKiBAcmV0dXJucyBUaGUgY3VycmVudCBhdXRoIHRva2VuIG9yIG51bGwgaWYgbm90IHNldFxuICAgKi9cbiAgZ2V0QXV0aFRva2VuKCk6IHN0cmluZyB8IG51bGwge1xuICAgIHJldHVybiB0aGlzLmF1dGhUb2tlbjtcbiAgfVxuXG4gIC8qKlxuICAgKiBDYW5jZWxzIGEgc3BlY2lmaWMgcmVxdWVzdCBieSBpdHMga2V5XG4gICAqIEBwYXJhbSBrZXkgLSBUaGUgdW5pcXVlIGtleSBpZGVudGlmeWluZyB0aGUgcmVxdWVzdCB0byBjYW5jZWxcbiAgICovXG4gIGNhbmNlbFJlcXVlc3Qoa2V5OiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0aGlzLnJlcXVlc3RNYW5hZ2VyLmNhbmNlbChrZXkpO1xuICB9XG5cbiAgLyoqXG4gICAqIENhbmNlbHMgYWxsIHBlbmRpbmcgcmVxdWVzdHNcbiAgICogVXNlZnVsIGZvciBjbGVhbnVwIG9uIG5hdmlnYXRpb24gb3IgY29tcG9uZW50IHVubW91bnRcbiAgICovXG4gIGNhbmNlbEFsbFJlcXVlc3RzKCk6IHZvaWQge1xuICAgIHRoaXMucmVxdWVzdE1hbmFnZXIuY2FuY2VsQWxsKCk7XG4gIH1cblxuICAvKipcbiAgICogQ29yZSByZXF1ZXN0IG1ldGhvZCB0aGF0IGhhbmRsZXMgYWxsIEhUVFAgb3BlcmF0aW9uc1xuICAgKiBAdGVtcGxhdGUgVCAtIFRoZSBleHBlY3RlZCByZXNwb25zZSBkYXRhIHR5cGVcbiAgICogQHBhcmFtIGVuZHBvaW50IC0gQVBJIGVuZHBvaW50IHJlbGF0aXZlIHRvIGJhc2VVUkxcbiAgICogQHBhcmFtIGNvbmZpZyAtIFJlcXVlc3QgY29uZmlndXJhdGlvbiBvcHRpb25zXG4gICAqIEByZXR1cm5zIFByb21pc2UgcmVzb2x2aW5nIHRvIEFwaVJlc3BvbnNlIHdpdGggZGF0YSBvciBlcnJvclxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBgYGB0eXBlc2NyaXB0XG4gICAqIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgY2xpZW50LnJlcXVlc3Q8VXNlcj4oJy91c2Vycy8xMjMnLCB7XG4gICAqICAgbWV0aG9kOiAnR0VUJyxcbiAgICogICB0aW1lb3V0OiA1MDAwLFxuICAgKiAgIHRocm93RXJyb3JzOiBmYWxzZVxuICAgKiB9KTtcbiAgICogYGBgXG4gICAqL1xuICBhc3luYyByZXF1ZXN0PFQgPSBSZXNwb25zZURhdGE+KFxuICAgIGVuZHBvaW50OiBzdHJpbmcsXG4gICAgY29uZmlnOiBSZXF1ZXN0Q29uZmlnID0ge31cbiAgKTogUHJvbWlzZTxBcGlSZXNwb25zZTxULCBBcGlFcnJvcj4+IHtcbiAgICAvLyBHZW5lcmF0ZSBjb3JyZWxhdGlvbiBJRFxuICAgIGNvbnN0IGNvcnJlbGF0aW9uSWQgPVxuICAgICAgY29uZmlnLmNvcnJlbGF0aW9uSWQgfHxcbiAgICAgICghY29uZmlnLnNraXBDb3JyZWxhdGlvbklkICYmIHRoaXMuaW5jbHVkZUNvcnJlbGF0aW9uSWRcbiAgICAgICAgPyBnZW5lcmF0ZUNvcnJlbGF0aW9uSWQodGhpcy5jb3JyZWxhdGlvbklkUHJlZml4KVxuICAgICAgICA6IHVuZGVmaW5lZCk7XG4gICAgLy8gR2VuZXJhdGUgcmVxdWVzdCBrZXkgZm9yIHRyYWNraW5nXG4gICAgY29uc3QgcmVxdWVzdEtleSA9IGAke2NvbmZpZy5tZXRob2QgfHwgJ0dFVCd9XyR7ZW5kcG9pbnR9XyR7RGF0ZS5ub3coKX1gO1xuXG4gICAgLy8gQ3JlYXRlIG1hc3RlciBjb250cm9sbGVyIGZvciB0aGlzIHJlcXVlc3RcbiAgICBjb25zdCBtYXN0ZXJDb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpO1xuXG4gICAgdHJ5IHtcbiAgICAgIC8vIENvbWJpbmUgYWxsIGFib3J0IHNpZ25hbHNcbiAgICAgIGNvbnN0IHNpZ25hbHM6IEFycmF5PEFib3J0U2lnbmFsIHwgdW5kZWZpbmVkPiA9IFtcbiAgICAgICAgY29uZmlnLnNpZ25hbCxcbiAgICAgICAgY29uZmlnLmNhbmNlbFRva2VuPy5zaWduYWwsXG4gICAgICAgIG1hc3RlckNvbnRyb2xsZXIuc2lnbmFsLFxuICAgICAgXTtcblxuICAgICAgLy8gQWRkIHRpbWVvdXQgc2lnbmFsIGlmIGNvbmZpZ3VyZWRcbiAgICAgIGNvbnN0IHRpbWVvdXQgPSBjb25maWcudGltZW91dCB8fCB0aGlzLmRlZmF1bHRUaW1lb3V0O1xuICAgICAgY29uc3QgdGltZW91dENvbnRyb2xsZXIgPSB0aGlzLnNpZ25hbE1hbmFnZXIuY3JlYXRlVGltZW91dFNpZ25hbCh0aW1lb3V0KTtcblxuICAgICAgc2lnbmFscy5wdXNoKHRpbWVvdXRDb250cm9sbGVyLnNpZ25hbCk7XG5cbiAgICAgIC8vIENyZWF0ZSBjb21iaW5lZCBzaWduYWxcbiAgICAgIGNvbnN0IGNvbWJpbmVkQ29udHJvbGxlciA9XG4gICAgICAgIHRoaXMuc2lnbmFsTWFuYWdlci5jcmVhdGVDb21iaW5lZFNpZ25hbChzaWduYWxzKTtcblxuICAgICAgLy8gVHJhY2sgdGhpcyByZXF1ZXN0XG4gICAgICBpZiAoY29ycmVsYXRpb25JZCkge1xuICAgICAgICB0aGlzLnJlcXVlc3RNYW5hZ2VyLmFkZChyZXF1ZXN0S2V5LCBtYXN0ZXJDb250cm9sbGVyLCBjb3JyZWxhdGlvbklkKTtcbiAgICAgIH1cblxuICAgICAgLy8gQXBwbHkgcmVxdWVzdCBpbnRlcmNlcHRvcnNcbiAgICAgIGNvbnN0IGZpbmFsQ29uZmlnID1cbiAgICAgICAgYXdhaXQgdGhpcy5pbnRlcmNlcHRvck1hbmFnZXIuYXBwbHlSZXF1ZXN0SW50ZXJjZXB0b3JzKHtcbiAgICAgICAgICAuLi5jb25maWcsXG4gICAgICAgICAgc2lnbmFsOiBjb21iaW5lZENvbnRyb2xsZXIuc2lnbmFsLFxuICAgICAgICAgIGNvcnJlbGF0aW9uSWQsXG4gICAgICAgIH0pO1xuXG4gICAgICAvLyBCdWlsZCBmdWxsIFVSTFxuICAgICAgY29uc3QgdXJsID0gdGhpcy51cmxCdWlsZGVyLmJ1aWxkVVJMKFxuICAgICAgICB0aGlzLmJhc2VVUkwsXG4gICAgICAgIGVuZHBvaW50LFxuICAgICAgICBmaW5hbENvbmZpZy5wYXJhbXMgYXMgUXVlcnlQYXJhbXNcbiAgICAgICk7XG5cbiAgICAgIC8vIEFkZCBkZWZhdWx0IGhlYWRlcnNcbiAgICAgIGNvbnN0IGhlYWRlcnMgPSBuZXcgSGVhZGVycyhmaW5hbENvbmZpZy5oZWFkZXJzKTtcblxuICAgICAgLy8gQWRkIGNvcnJlbGF0aW9uIElEIGhlYWRlclxuICAgICAgaWYgKGNvcnJlbGF0aW9uSWQpIHtcbiAgICAgICAgaGVhZGVycy5zZXQoJ1gtQ29ycmVsYXRpb24tSWQnLCBjb3JyZWxhdGlvbklkKTtcbiAgICAgICAgaGVhZGVycy5zZXQoJ1gtUmVxdWVzdC1JZCcsIGNvcnJlbGF0aW9uSWQpO1xuICAgICAgfVxuXG4gICAgICAvLyBBZGQgYXV0aCBoZWFkZXIgaWYgdG9rZW4gZXhpc3RzXG4gICAgICBpZiAodGhpcy5hdXRoVG9rZW4gJiYgIWZpbmFsQ29uZmlnLnNraXBBdXRoUmVmcmVzaCkge1xuICAgICAgICBoZWFkZXJzLnNldCgnQXV0aG9yaXphdGlvbicsIGBCZWFyZXIgJHt0aGlzLmF1dGhUb2tlbn1gKTtcbiAgICAgIH1cblxuICAgICAgLy8gU2V0IGNvbnRlbnQtdHlwZSBmb3IgSlNPTiBwYXlsb2Fkc1xuXG4gICAgICBsZXQgZmV0Y2hCb2R5OiBCb2R5SW5pdCB8IG51bGwgfCB1bmRlZmluZWQgPSBmaW5hbENvbmZpZy5ib2R5IGFzXG4gICAgICAgIHwgQm9keUluaXRcbiAgICAgICAgfCBudWxsXG4gICAgICAgIHwgdW5kZWZpbmVkO1xuXG4gICAgICBpZiAoXG4gICAgICAgIGZpbmFsQ29uZmlnLmJvZHkgJiZcbiAgICAgICAgdHlwZW9mIGZpbmFsQ29uZmlnLmJvZHkgPT09ICdvYmplY3QnICYmXG4gICAgICAgICEoZmluYWxDb25maWcuYm9keSBpbnN0YW5jZW9mIEZvcm1EYXRhKSAmJlxuICAgICAgICAhKGZpbmFsQ29uZmlnLmJvZHkgaW5zdGFuY2VvZiBCbG9iKSAmJlxuICAgICAgICAhKGZpbmFsQ29uZmlnLmJvZHkgaW5zdGFuY2VvZiBBcnJheUJ1ZmZlcikgJiZcbiAgICAgICAgIShmaW5hbENvbmZpZy5ib2R5IGluc3RhbmNlb2YgVVJMU2VhcmNoUGFyYW1zKSAmJlxuICAgICAgICAhKGZpbmFsQ29uZmlnLmJvZHkgaW5zdGFuY2VvZiBSZWFkYWJsZVN0cmVhbSlcbiAgICAgICkge1xuICAgICAgICBoZWFkZXJzLnNldCgnQ29udGVudC1UeXBlJywgJ2FwcGxpY2F0aW9uL2pzb24nKTtcbiAgICAgICAgZmV0Y2hCb2R5ID0gSlNPTi5zdHJpbmdpZnkoZmluYWxDb25maWcuYm9keSk7XG4gICAgICB9IGVsc2UgaWYgKGZpbmFsQ29uZmlnLmJvZHkgaW5zdGFuY2VvZiBGb3JtRGF0YSkge1xuICAgICAgICAvLyBSZW1vdmUgQ29udGVudC1UeXBlIGhlYWRlciBmb3IgRm9ybURhdGEgdG8gbGV0IGJyb3dzZXIgc2V0IGl0IHdpdGggYm91bmRhcnlcbiAgICAgICAgaGVhZGVycy5kZWxldGUoJ0NvbnRlbnQtVHlwZScpO1xuICAgICAgfVxuXG4gICAgICBmaW5hbENvbmZpZy5oZWFkZXJzID0gaGVhZGVycztcblxuICAgICAgLy8gQ3JlYXRlIGZldGNoIHByb21pc2VcbiAgICAgIGNvbnN0IGZldGNoUHJvbWlzZSA9IGFzeW5jICgpOiBQcm9taXNlPEFwaVJlc3BvbnNlPFQsIEFwaUVycm9yPj4gPT4ge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2godXJsLCB7XG4gICAgICAgICAgICAuLi5maW5hbENvbmZpZyxcbiAgICAgICAgICAgIGJvZHk6IGZldGNoQm9keSxcbiAgICAgICAgICAgIHNpZ25hbDogY29tYmluZWRDb250cm9sbGVyLnNpZ25hbCxcbiAgICAgICAgICB9IGFzIFJlcXVlc3RJbml0KTtcblxuICAgICAgICAgIC8vIFBhcnNlIHJlc3BvbnNlIGRhdGFcbiAgICAgICAgICBjb25zdCByZXNwb25zZURhdGEgPVxuICAgICAgICAgICAgYXdhaXQgdGhpcy5yZXNwb25zZVBhcnNlci5wYXJzZVJlc3BvbnNlKHJlc3BvbnNlKTtcblxuICAgICAgICAgIC8vIEhhbmRsZSBlcnJvciByZXNwb25zZXNcbiAgICAgICAgICBpZiAoIXJlc3BvbnNlLm9rKSB7XG4gICAgICAgICAgICAvLyBDYXN0IHRvIEVycm9yUmVzcG9uc2VEYXRhIGZvciBzdHJ1Y3R1cmVkIGVycm9yIHJlc3BvbnNlcyBmcm9tIGJhY2tlbmRcbiAgICAgICAgICAgIGNvbnN0IGVycm9yRGF0YSA9IHJlc3BvbnNlRGF0YSBhcyBFcnJvclJlc3BvbnNlRGF0YTtcblxuICAgICAgICAgICAgY29uc3QgZXJyb3I6IEFwaUVycm9yID0gT2JqZWN0LmFzc2lnbihcbiAgICAgICAgICAgICAgbmV3IEVycm9yKFxuICAgICAgICAgICAgICAgIGVycm9yRGF0YS50aXRsZSB8fFxuICAgICAgICAgICAgICAgICAgYEhUVFAgJHtyZXNwb25zZS5zdGF0dXN9OiAke3Jlc3BvbnNlLnN0YXR1c1RleHR9YFxuICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgdHlwZTpcbiAgICAgICAgICAgICAgICAgIGVycm9yRGF0YS50eXBlIHx8XG4gICAgICAgICAgICAgICAgICB0aGlzLmVycm9yTm9ybWFsaXplci5nZXRFcnJvclR5cGUocmVzcG9uc2Uuc3RhdHVzKSxcbiAgICAgICAgICAgICAgICB0aXRsZTpcbiAgICAgICAgICAgICAgICAgIGVycm9yRGF0YS50aXRsZSB8fFxuICAgICAgICAgICAgICAgICAgdGhpcy5lcnJvck5vcm1hbGl6ZXIuZ2V0RXJyb3JUaXRsZShyZXNwb25zZS5zdGF0dXMpLFxuICAgICAgICAgICAgICAgIHN0YXR1czogcmVzcG9uc2Uuc3RhdHVzLFxuICAgICAgICAgICAgICAgIHRyYWNlSWQ6IGVycm9yRGF0YS50cmFjZUlkIHx8IGNvcnJlbGF0aW9uSWQsXG4gICAgICAgICAgICAgICAgZXJyb3JzOiBlcnJvckRhdGEuZXJyb3JzLFxuICAgICAgICAgICAgICAgIGlzQWJvcnRlZDogZmFsc2UsXG4gICAgICAgICAgICAgICAgY29uZmlnOiBmaW5hbENvbmZpZyxcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgKTtcblxuICAgICAgICAgICAgLy8gQ2hlY2sgaWYgd2Ugc2hvdWxkIHRocm93IG9yIHJldHVybiBlcnJvciBpbiByZXNwb25zZVxuICAgICAgICAgICAgaWYgKGZpbmFsQ29uZmlnLnRocm93RXJyb3JzICE9PSBmYWxzZSkge1xuICAgICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIC8vIFJldHVybiBlcnJvciBpbiBBcGlSZXNwb25zZS5lcnJvciBmaWVsZFxuICAgICAgICAgICAgICByZXR1cm4gYXdhaXQgdGhpcy5pbnRlcmNlcHRvck1hbmFnZXIuYXBwbHlSZXNwb25zZUludGVyY2VwdG9ycyh7XG4gICAgICAgICAgICAgICAgZXJyb3IsXG4gICAgICAgICAgICAgIH0gYXMgQXBpUmVzcG9uc2U8VCwgQXBpRXJyb3I+KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBGb3JtYXQgc3VjY2Vzc2Z1bCByZXNwb25zZVxuICAgICAgICAgIGNvbnN0IGFwaVJlc3BvbnNlOiBBcGlSZXNwb25zZTxUPiA9IHtcbiAgICAgICAgICAgIGRhdGE6IHJlc3BvbnNlRGF0YSBhcyBULFxuICAgICAgICAgIH07XG5cbiAgICAgICAgICAvLyBBcHBseSByZXNwb25zZSBpbnRlcmNlcHRvcnNcbiAgICAgICAgICByZXR1cm4gYXdhaXQgdGhpcy5pbnRlcmNlcHRvck1hbmFnZXIuYXBwbHlSZXNwb25zZUludGVyY2VwdG9ycyhcbiAgICAgICAgICAgIGFwaVJlc3BvbnNlXG4gICAgICAgICAgKTtcbiAgICAgICAgfSBjYXRjaCAoZXJyb3I6IHVua25vd24pIHtcbiAgICAgICAgICAvLyBIYW5kbGUgYWJvcnQgZXJyb3JzXG4gICAgICAgICAgaWYgKChlcnJvciBhcyBFcnJvcikubmFtZSA9PT0gJ0Fib3J0RXJyb3InKSB7XG4gICAgICAgICAgICBjb25zdCBhYm9ydEVycm9yID0gT2JqZWN0LmFzc2lnbihcbiAgICAgICAgICAgICAgbmV3IEVycm9yKChlcnJvciBhcyBFcnJvcikubWVzc2FnZSB8fCAnUmVxdWVzdCBhYm9ydGVkJyksXG4gICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICB0eXBlOiAncmVxdWVzdF9jYW5jZWxsZWQnLFxuICAgICAgICAgICAgICAgIHRpdGxlOiAnUmVxdWVzdCB3YXMgY2FuY2VsbGVkJyxcbiAgICAgICAgICAgICAgICBzdGF0dXM6IDAsXG4gICAgICAgICAgICAgICAgdHJhY2VJZDogY29ycmVsYXRpb25JZCxcbiAgICAgICAgICAgICAgICBpc0Fib3J0ZWQ6IHRydWUsXG4gICAgICAgICAgICAgICAgY29uZmlnOiBmaW5hbENvbmZpZyxcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgKTtcblxuICAgICAgICAgICAgLy8gQ2hlY2sgaWYgd2Ugc2hvdWxkIHRocm93IG9yIHJldHVybiBlcnJvciBpbiByZXNwb25zZVxuICAgICAgICAgICAgaWYgKGZpbmFsQ29uZmlnLnRocm93RXJyb3JzICE9PSBmYWxzZSkge1xuICAgICAgICAgICAgICB0aHJvdyBhYm9ydEVycm9yO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgLy8gUmV0dXJuIGVycm9yIGluIEFwaVJlc3BvbnNlLmVycm9yIGZpZWxkXG4gICAgICAgICAgICAgIHJldHVybiBhd2FpdCB0aGlzLmludGVyY2VwdG9yTWFuYWdlci5hcHBseVJlc3BvbnNlSW50ZXJjZXB0b3JzKHtcbiAgICAgICAgICAgICAgICBlcnJvcjogYWJvcnRFcnJvcixcbiAgICAgICAgICAgICAgfSBhcyBBcGlSZXNwb25zZTxULCBBcGlFcnJvcj4pO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIHRocm93IGVycm9yO1xuICAgICAgICB9XG4gICAgICB9O1xuXG4gICAgICAvLyBIYW5kbGUgcmV0cmllcyBpZiBjb25maWd1cmVkXG4gICAgICBpZiAoZmluYWxDb25maWcucmV0cmllcyAmJiBmaW5hbENvbmZpZy5yZXRyaWVzID4gMCkge1xuICAgICAgICByZXR1cm4gYXdhaXQgdGhpcy5yZXRyeUhhbmRsZXIucmV0cnlSZXF1ZXN0KFxuICAgICAgICAgIGZldGNoUHJvbWlzZSxcbiAgICAgICAgICBmaW5hbENvbmZpZy5yZXRyaWVzLFxuICAgICAgICAgIGZpbmFsQ29uZmlnLnJldHJ5RGVsYXkgfHwgMTAwMCxcbiAgICAgICAgICBjb21iaW5lZENvbnRyb2xsZXIuc2lnbmFsXG4gICAgICAgICk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiBhd2FpdCBmZXRjaFByb21pc2UoKTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgLy8gSGFuZGxlIGVycm9yc1xuICAgICAgY29uc3QgYXBpRXJyb3I6IEFwaUVycm9yID0gdGhpcy5lcnJvck5vcm1hbGl6ZXIubm9ybWFsaXplRXJyb3IoXG4gICAgICAgIGVycm9yLFxuICAgICAgICBjb25maWcsXG4gICAgICAgIGNvcnJlbGF0aW9uSWRcbiAgICAgICk7XG5cbiAgICAgIC8vIENoZWNrIGlmIHdlIHNob3VsZCB0aHJvdyBvciByZXR1cm4gZXJyb3IgaW4gcmVzcG9uc2VcbiAgICAgIGlmIChjb25maWcudGhyb3dFcnJvcnMgIT09IGZhbHNlKSB7XG4gICAgICAgIGF3YWl0IHRoaXMuaW50ZXJjZXB0b3JNYW5hZ2VyLmFwcGx5RXJyb3JJbnRlcmNlcHRvcnMoYXBpRXJyb3IpO1xuICAgICAgICAvLyBUaGlzIGxpbmUgd2lsbCBuZXZlciBiZSByZWFjaGVkIGFzIGFwcGx5RXJyb3JJbnRlcmNlcHRvcnMgYWx3YXlzIHRocm93cyxcbiAgICAgICAgLy8gYnV0IFR5cGVTY3JpcHQgcmVxdWlyZXMgYSByZXR1cm4gc3RhdGVtZW50XG4gICAgICAgIHRocm93IGFwaUVycm9yO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8gUmV0dXJuIGVycm9yIGluIEFwaVJlc3BvbnNlLmVycm9yIGZpZWxkXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgZXJyb3I6IGFwaUVycm9yLFxuICAgICAgICB9IGFzIEFwaVJlc3BvbnNlPFQsIEFwaUVycm9yPjtcbiAgICAgIH1cbiAgICB9IGZpbmFsbHkge1xuICAgICAgLy8gQ2xlYW4gdXAgcmVxdWVzdCB0cmFja2luZ1xuICAgICAgdGhpcy5yZXF1ZXN0TWFuYWdlci5yZW1vdmUocmVxdWVzdEtleSk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFBlcmZvcm1zIGEgR0VUIHJlcXVlc3RcbiAgICogQHRlbXBsYXRlIFQgLSBUaGUgZXhwZWN0ZWQgcmVzcG9uc2UgZGF0YSB0eXBlXG4gICAqIEBwYXJhbSBlbmRwb2ludCAtIEFQSSBlbmRwb2ludFxuICAgKiBAcGFyYW0gY29uZmlnIC0gT3B0aW9uYWwgcmVxdWVzdCBjb25maWd1cmF0aW9uXG4gICAqIEByZXR1cm5zIFByb21pc2UgcmVzb2x2aW5nIHRvIEFwaVJlc3BvbnNlXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgeyBhcGlEYXRhLCBlcnJvciB9ID0gYXdhaXQgY2xpZW50LmdldDxVc2VyW10+KCcvdXNlcnMnLCB7XG4gICAqICAgcGFyYW1zOiB7IGFjdGl2ZTogdHJ1ZSB9LFxuICAgKiAgIHRpbWVvdXQ6IDUwMDBcbiAgICogfSk7XG4gICAqIGBgYFxuICAgKi9cbiAgZ2V0PFQgPSBSZXNwb25zZURhdGE+KFxuICAgIGVuZHBvaW50OiBzdHJpbmcsXG4gICAgY29uZmlnPzogUmVxdWVzdENvbmZpZ1xuICApOiBQcm9taXNlPEFwaVJlc3BvbnNlPFQsIEFwaUVycm9yPj4ge1xuICAgIHJldHVybiB0aGlzLnJlcXVlc3Q8VD4oZW5kcG9pbnQsIHsgLi4uY29uZmlnLCBtZXRob2Q6ICdHRVQnIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIFBlcmZvcm1zIGEgUE9TVCByZXF1ZXN0XG4gICAqIEB0ZW1wbGF0ZSBUIC0gVGhlIGV4cGVjdGVkIHJlc3BvbnNlIGRhdGEgdHlwZVxuICAgKiBAdGVtcGxhdGUgVERhdGEgLSBUaGUgcmVxdWVzdCBib2R5IGRhdGEgdHlwZVxuICAgKiBAcGFyYW0gZW5kcG9pbnQgLSBBUEkgZW5kcG9pbnRcbiAgICogQHBhcmFtIGRhdGEgLSBSZXF1ZXN0IGJvZHkgZGF0YVxuICAgKiBAcGFyYW0gY29uZmlnIC0gT3B0aW9uYWwgcmVxdWVzdCBjb25maWd1cmF0aW9uXG4gICAqIEByZXR1cm5zIFByb21pc2UgcmVzb2x2aW5nIHRvIEFwaVJlc3BvbnNlXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgeyBhcGlEYXRhLCBlcnJvciB9ID0gYXdhaXQgY2xpZW50LnBvc3Q8VXNlciwgQ3JlYXRlVXNlckR0bz4oJy91c2VycycsIHtcbiAgICogICBuYW1lOiAnSm9obiBEb2UnLFxuICAgKiAgIGVtYWlsOiAnam9obkBleGFtcGxlLmNvbSdcbiAgICogfSk7XG4gICAqIGBgYFxuICAgKi9cbiAgcG9zdDxcbiAgICBUID0gUmVzcG9uc2VEYXRhLFxuICAgIFREYXRhIGV4dGVuZHMgQm9keUluaXQgfCBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiB8IG51bGwgPSBSZWNvcmQ8XG4gICAgICBzdHJpbmcsXG4gICAgICB1bmtub3duXG4gICAgPixcbiAgPihcbiAgICBlbmRwb2ludDogc3RyaW5nLFxuICAgIGRhdGE/OiBURGF0YSxcbiAgICBjb25maWc/OiBSZXF1ZXN0Q29uZmlnXG4gICk6IFByb21pc2U8QXBpUmVzcG9uc2U8VCwgQXBpRXJyb3I+PiB7XG4gICAgcmV0dXJuIHRoaXMucmVxdWVzdDxUPihlbmRwb2ludCwgeyAuLi5jb25maWcsIG1ldGhvZDogJ1BPU1QnLCBib2R5OiBkYXRhIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIFBlcmZvcm1zIGEgUFVUIHJlcXVlc3RcbiAgICogQHRlbXBsYXRlIFQgLSBUaGUgZXhwZWN0ZWQgcmVzcG9uc2UgZGF0YSB0eXBlXG4gICAqIEB0ZW1wbGF0ZSBURGF0YSAtIFRoZSByZXF1ZXN0IGJvZHkgZGF0YSB0eXBlXG4gICAqIEBwYXJhbSBlbmRwb2ludCAtIEFQSSBlbmRwb2ludFxuICAgKiBAcGFyYW0gZGF0YSAtIFJlcXVlc3QgYm9keSBkYXRhXG4gICAqIEBwYXJhbSBjb25maWcgLSBPcHRpb25hbCByZXF1ZXN0IGNvbmZpZ3VyYXRpb25cbiAgICogQHJldHVybnMgUHJvbWlzZSByZXNvbHZpbmcgdG8gQXBpUmVzcG9uc2VcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiBjb25zdCB7IGFwaURhdGEsIGVycm9yIH0gPSBhd2FpdCBjbGllbnQucHV0PFVzZXIsIFVwZGF0ZVVzZXJEdG8+KFxuICAgKiAgICcvdXNlcnMvMTIzJyxcbiAgICogICB7IG5hbWU6ICdKYW5lIERvZScgfVxuICAgKiApO1xuICAgKiBgYGBcbiAgICovXG4gIHB1dDxcbiAgICBUID0gUmVzcG9uc2VEYXRhLFxuICAgIFREYXRhIGV4dGVuZHMgQm9keUluaXQgfCBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiB8IG51bGwgPSBSZWNvcmQ8XG4gICAgICBzdHJpbmcsXG4gICAgICB1bmtub3duXG4gICAgPixcbiAgPihcbiAgICBlbmRwb2ludDogc3RyaW5nLFxuICAgIGRhdGE/OiBURGF0YSxcbiAgICBjb25maWc/OiBSZXF1ZXN0Q29uZmlnXG4gICk6IFByb21pc2U8QXBpUmVzcG9uc2U8VCwgQXBpRXJyb3I+PiB7XG4gICAgcmV0dXJuIHRoaXMucmVxdWVzdDxUPihlbmRwb2ludCwgeyAuLi5jb25maWcsIG1ldGhvZDogJ1BVVCcsIGJvZHk6IGRhdGEgfSk7XG4gIH1cblxuICAvKipcbiAgICogUGVyZm9ybXMgYSBQQVRDSCByZXF1ZXN0XG4gICAqIEB0ZW1wbGF0ZSBUIC0gVGhlIGV4cGVjdGVkIHJlc3BvbnNlIGRhdGEgdHlwZVxuICAgKiBAdGVtcGxhdGUgVERhdGEgLSBUaGUgcmVxdWVzdCBib2R5IGRhdGEgdHlwZVxuICAgKiBAcGFyYW0gZW5kcG9pbnQgLSBBUEkgZW5kcG9pbnRcbiAgICogQHBhcmFtIGRhdGEgLSBSZXF1ZXN0IGJvZHkgZGF0YVxuICAgKiBAcGFyYW0gY29uZmlnIC0gT3B0aW9uYWwgcmVxdWVzdCBjb25maWd1cmF0aW9uXG4gICAqIEByZXR1cm5zIFByb21pc2UgcmVzb2x2aW5nIHRvIEFwaVJlc3BvbnNlXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgeyBhcGlEYXRhLCBlcnJvciB9ID0gYXdhaXQgY2xpZW50LnBhdGNoPFVzZXI+KFxuICAgKiAgICcvdXNlcnMvMTIzJyxcbiAgICogICB7IHN0YXR1czogJ2FjdGl2ZScgfVxuICAgKiApO1xuICAgKiBgYGBcbiAgICovXG4gIHBhdGNoPFxuICAgIFQgPSBSZXNwb25zZURhdGEsXG4gICAgVERhdGEgZXh0ZW5kcyBCb2R5SW5pdCB8IFJlY29yZDxzdHJpbmcsIHVua25vd24+IHwgbnVsbCA9IFJlY29yZDxcbiAgICAgIHN0cmluZyxcbiAgICAgIHVua25vd25cbiAgICA+LFxuICA+KFxuICAgIGVuZHBvaW50OiBzdHJpbmcsXG4gICAgZGF0YT86IFREYXRhLFxuICAgIGNvbmZpZz86IFJlcXVlc3RDb25maWdcbiAgKTogUHJvbWlzZTxBcGlSZXNwb25zZTxULCBBcGlFcnJvcj4+IHtcbiAgICByZXR1cm4gdGhpcy5yZXF1ZXN0PFQ+KGVuZHBvaW50LCB7XG4gICAgICAuLi5jb25maWcsXG4gICAgICBtZXRob2Q6ICdQQVRDSCcsXG4gICAgICBib2R5OiBkYXRhLFxuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIFBlcmZvcm1zIGEgREVMRVRFIHJlcXVlc3RcbiAgICogQHRlbXBsYXRlIFQgLSBUaGUgZXhwZWN0ZWQgcmVzcG9uc2UgZGF0YSB0eXBlXG4gICAqIEBwYXJhbSBlbmRwb2ludCAtIEFQSSBlbmRwb2ludFxuICAgKiBAcGFyYW0gY29uZmlnIC0gT3B0aW9uYWwgcmVxdWVzdCBjb25maWd1cmF0aW9uXG4gICAqIEByZXR1cm5zIFByb21pc2UgcmVzb2x2aW5nIHRvIEFwaVJlc3BvbnNlXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGBgYHR5cGVzY3JpcHRcbiAgICogY29uc3QgeyBlcnJvciB9ID0gYXdhaXQgY2xpZW50LmRlbGV0ZSgnL3VzZXJzLzEyMycpO1xuICAgKiBpZiAoIWVycm9yKSB7XG4gICAqICAgY29uc29sZS5sb2coJ1VzZXIgZGVsZXRlZCBzdWNjZXNzZnVsbHknKTtcbiAgICogfVxuICAgKiBgYGBcbiAgICovXG4gIGRlbGV0ZTxUID0gUmVzcG9uc2VEYXRhPihcbiAgICBlbmRwb2ludDogc3RyaW5nLFxuICAgIGNvbmZpZz86IFJlcXVlc3RDb25maWdcbiAgKTogUHJvbWlzZTxBcGlSZXNwb25zZTxULCBBcGlFcnJvcj4+IHtcbiAgICByZXR1cm4gdGhpcy5yZXF1ZXN0PFQ+KGVuZHBvaW50LCB7IC4uLmNvbmZpZywgbWV0aG9kOiAnREVMRVRFJyB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBQZXJmb3JtcyBhIGZpbHRlcmVkIGxpc3QgcmVxdWVzdCB3aXRoIHBhZ2luYXRpb24gYW5kIHNvcnRpbmdcbiAgICogQHRlbXBsYXRlIFRMaXN0TW9kZWwgLSBUaGUgdHlwZSBvZiBpbmRpdmlkdWFsIGxpc3QgaXRlbXNcbiAgICogQHRlbXBsYXRlIFRGaWx0ZXIgLSBUaGUgZmlsdGVyIGNyaXRlcmlhIHR5cGVcbiAgICogQHBhcmFtIHVybCAtIEFQSSBlbmRwb2ludFxuICAgKiBAcGFyYW0gZGF0YSAtIFBhZ2luYXRpb24gYW5kIGZpbHRlciBkYXRhXG4gICAqIEBwYXJhbSBjb25maWcgLSBPcHRpb25hbCByZXF1ZXN0IGNvbmZpZ3VyYXRpb25cbiAgICogQHJldHVybnMgUHJvbWlzZSByZXNvbHZpbmcgdG8gcGFnaW5hdGVkIGxpc3QgcmVzcG9uc2VcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogYGBgdHlwZXNjcmlwdFxuICAgKiBjb25zdCB7IGFwaURhdGEsIGVycm9yIH0gPSBhd2FpdCBjbGllbnQuZmlsdGVyPFVzZXIsIFVzZXJGaWx0ZXI+KFxuICAgKiAgICcvdXNlcnMvZmlsdGVyJyxcbiAgICogICB7XG4gICAqICAgICBwYWdlT2Zmc2V0OiAwLFxuICAgKiAgICAgcGFnZVNpemU6IDIwLFxuICAgKiAgICAgc29ydEZpZWxkOiAnY3JlYXRlZEF0JyxcbiAgICogICAgIHNvcnRPcmRlcjogJ2Rlc2MnLFxuICAgKiAgICAgZmlsdGVyTW9kZWw6IHsgc3RhdHVzOiAnYWN0aXZlJyB9XG4gICAqICAgfVxuICAgKiApO1xuICAgKlxuICAgKiBpZiAoYXBpRGF0YSkge1xuICAgKiAgIGNvbnNvbGUubG9nKGBGb3VuZCAke2FwaURhdGEuVG90YWx9IHVzZXJzYCk7XG4gICAqICAgY29uc29sZS5sb2coJ1VzZXJzOicsIGFwaURhdGEuRGF0YSk7XG4gICAqIH1cbiAgICogYGBgXG4gICAqL1xuICBmaWx0ZXI8VExpc3RNb2RlbCwgVEZpbHRlciA9IFJlY29yZDxzdHJpbmcsIHVua25vd24+PihcbiAgICB1cmw6IHN0cmluZyxcbiAgICBkYXRhOiBQb3N0TW9kZWw8VEZpbHRlcj4sXG4gICAgY29uZmlnPzogUmVxdWVzdENvbmZpZ1xuICApOiBQcm9taXNlPEFwaVJlc3BvbnNlPFBhZ2luYXRlZFJlc3BvbnNlPFRMaXN0TW9kZWw+LCBBcGlFcnJvcj4+IHtcbiAgICAvLyBNZXJnZSBib2R5OiB7IC4uLnBvc3RNb2RlbCwgLi4ucG9zdE1vZGVsLmZpbHRlck1vZGVsIH1cbiAgICBjb25zdCBtZXJnZWREYXRhID0geyAuLi5kYXRhLCAuLi5kYXRhLmZpbHRlck1vZGVsIH07XG5cbiAgICByZXR1cm4gdGhpcy5yZXF1ZXN0PFBhZ2luYXRlZFJlc3BvbnNlPFRMaXN0TW9kZWw+Pih1cmwsIHtcbiAgICAgIC4uLmNvbmZpZyxcbiAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgYm9keTogbWVyZ2VkRGF0YSxcbiAgICB9KTtcbiAgfVxufVxuIiwgIi8qKlxuICogQGZpbGVvdmVydmlldyBGYWN0b3J5IGZ1bmN0aW9ucyBmb3IgY3JlYXRpbmcgYW5kIG1hbmFnaW5nIEFwaUNsaWVudCBpbnN0YW5jZXMuXG4gKlxuICogUHJvdmlkZXMgY29udmVuaWVudCBmYWN0b3J5IG1ldGhvZHMgZm9yIGNyZWF0aW5nIHByZS1jb25maWd1cmVkIEFwaUNsaWVudCBpbnN0YW5jZXNcbiAqIHdpdGggc2Vuc2libGUgZGVmYXVsdHMgYW5kIGF1dG9tYXRpYyB0b2tlbiBtYW5hZ2VtZW50IGZyb20gbG9jYWxTdG9yYWdlLlxuICpcbiAqIEBtb2R1bGUgY3JlYXRlQXBpQ2xpZW50XG4gKi9cblxuaW1wb3J0IHsgQXBpQ2xpZW50IH0gZnJvbSAnLi9BcGlDbGllbnQnO1xuaW1wb3J0IHR5cGUgeyBFcnJvckludGVyY2VwdG9yIH0gZnJvbSAnLi90eXBlcy9FcnJvckludGVyY2VwdG9yJztcbmltcG9ydCB0eXBlIHsgUmVxdWVzdEludGVyY2VwdG9yIH0gZnJvbSAnLi90eXBlcy9SZXF1ZXN0SW50ZXJjZXB0b3InO1xuaW1wb3J0IHR5cGUgeyBSZXNwb25zZUludGVyY2VwdG9yIH0gZnJvbSAnLi90eXBlcy9SZXNwb25zZUludGVyY2VwdG9yJztcblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIG9wdGlvbnMgZm9yIGNyZWF0aW5nIGFuIEFwaUNsaWVudCBpbnN0YW5jZS5cbiAqXG4gKiBAcHVibGljXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQXBpQ2xpZW50Q29uZmlnIHtcbiAgLyoqXG4gICAqIEJhc2UgVVJMIGZvciBhbGwgQVBJIHJlcXVlc3RzLlxuICAgKiBAZXhhbXBsZSAnaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20nXG4gICAqL1xuICBiYXNlVVJMOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIERlZmF1bHQgdGltZW91dCBmb3IgcmVxdWVzdHMgaW4gbWlsbGlzZWNvbmRzLlxuICAgKiBAZGVmYXVsdCAzMDAwMFxuICAgKi9cbiAgdGltZW91dD86IG51bWJlcjtcblxuICAvKipcbiAgICogUHJlZml4IGZvciBhdXRvLWdlbmVyYXRlZCBjb3JyZWxhdGlvbiBJRHMgKGUuZy4sICdhcGknLCAnd2ViJywgJ21vYmlsZScpLlxuICAgKiBAZGVmYXVsdCAnYXBpJ1xuICAgKi9cbiAgY29ycmVsYXRpb25JZFByZWZpeDogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRvIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGUgYW5kIGluY2x1ZGUgY29ycmVsYXRpb24gSURzIGluIHJlcXVlc3RzLlxuICAgKiBAZGVmYXVsdCB0cnVlXG4gICAqL1xuICBpbmNsdWRlQ29ycmVsYXRpb25JZD86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIGxvY2FsU3RvcmFnZSBrZXkgdXNlZCB0byByZXRyaWV2ZSB0aGUgYXV0aCB0b2tlbi5cbiAgICogQGV4YW1wbGUgJ3NlcnZpY2VUb2tlbidcbiAgICovXG4gIHRva2VuU3RvcmFnZUtleTogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBJbml0aWFsIGF1dGhlbnRpY2F0aW9uIHRva2VuIHRvIHNldCBvbiB0aGUgY2xpZW50LlxuICAgKiBAZGVmYXVsdCB1bmRlZmluZWRcbiAgICovXG4gIGF1dGhUb2tlbj86IHN0cmluZyB8IG51bGw7XG5cbiAgLyoqXG4gICAqIEFycmF5IG9mIHJlcXVlc3QgaW50ZXJjZXB0b3JzIHRvIHJlZ2lzdGVyIGR1cmluZyBjbGllbnQgY3JlYXRpb24uXG4gICAqIEBkZWZhdWx0IFtdXG4gICAqL1xuICByZXF1ZXN0SW50ZXJjZXB0b3JzPzogUmVxdWVzdEludGVyY2VwdG9yW107XG5cbiAgLyoqXG4gICAqIEFycmF5IG9mIHJlc3BvbnNlIGludGVyY2VwdG9ycyB0byByZWdpc3RlciBkdXJpbmcgY2xpZW50IGNyZWF0aW9uLlxuICAgKiBAZGVmYXVsdCBbXVxuICAgKi9cbiAgcmVzcG9uc2VJbnRlcmNlcHRvcnM/OiBSZXNwb25zZUludGVyY2VwdG9yW107XG5cbiAgLyoqXG4gICAqIEFycmF5IG9mIGVycm9yIGludGVyY2VwdG9ycyB0byByZWdpc3RlciBkdXJpbmcgY2xpZW50IGNyZWF0aW9uLlxuICAgKiBAZGVmYXVsdCBbXVxuICAgKi9cbiAgZXJyb3JJbnRlcmNlcHRvcnM/OiBFcnJvckludGVyY2VwdG9yW107XG59XG5cbi8qKlxuICogR2xvYmFsIHNpbmdsZXRvbiBBcGlDbGllbnQgaW5zdGFuY2UuXG4gKiBAaW50ZXJuYWxcbiAqL1xubGV0IGdsb2JhbEFwaUNsaWVudDogQXBpQ2xpZW50IHwgbnVsbCA9IG51bGw7XG5cbi8qKlxuICogQ3JlYXRlcyBhIG5ldyBBcGlDbGllbnQgaW5zdGFuY2Ugd2l0aCBhdXRvbWF0aWMgdG9rZW4gbWFuYWdlbWVudC5cbiAqXG4gKiBUaGlzIGZhY3RvcnkgZnVuY3Rpb246XG4gKiAtIFJlcXVpcmVzIGJhc2UgVVJMLCBjb3JyZWxhdGlvbiBJRCBwcmVmaXgsIGFuZCB0b2tlbiBzdG9yYWdlIGtleSB0byBiZSBleHBsaWNpdGx5IHByb3ZpZGVkXG4gKiAtIFNldHMgdXAgYXV0b21hdGljIGF1dGhlbnRpY2F0aW9uIHRva2VuIGluamVjdGlvbiBmcm9tIGxvY2FsU3RvcmFnZVxuICogLSBDb25maWd1cmVzIGNvcnJlbGF0aW9uIElEIGdlbmVyYXRpb25cbiAqIC0gUmVnaXN0ZXJzIHByb3ZpZGVkIGludGVyY2VwdG9yc1xuICpcbiAqICoqTm90ZToqKiBFYWNoIGNhbGwgY3JlYXRlcyBhIE5FVyBpbnN0YW5jZS4gVXNlIHtAbGluayBnZXRHbG9iYWxBcGlDbGllbnR9IGZvciBzaW5nbGV0b24gYmVoYXZpb3IuXG4gKlxuICogQHBhcmFtIGNvbmZpZyAtIENvbmZpZ3VyYXRpb24gb3B0aW9ucyBmb3IgdGhlIEFQSSBjbGllbnQgKGJhc2VVUkwsIGNvcnJlbGF0aW9uSWRQcmVmaXgsIHRva2VuU3RvcmFnZUtleSBhcmUgcmVxdWlyZWQpXG4gKiBAcmV0dXJucyBBIGZ1bGx5IGNvbmZpZ3VyZWQgQXBpQ2xpZW50IGluc3RhbmNlXG4gKiBAcHVibGljXG4gKlxuICogQGV4YW1wbGVcbiAqIEJhc2ljIHVzYWdlOlxuICogYGBgdHlwZXNjcmlwdFxuICogY29uc3QgY2xpZW50ID0gY3JlYXRlQXBpQ2xpZW50KHtcbiAqICAgYmFzZVVSTDogJ2h0dHBzOi8vYXBpLmV4YW1wbGUuY29tJyxcbiAqICAgY29ycmVsYXRpb25JZFByZWZpeDogJ3dlYicsXG4gKiAgIHRva2VuU3RvcmFnZUtleTogJ3NlcnZpY2VUb2tlbidcbiAqIH0pO1xuICogYGBgXG4gKlxuICogQGV4YW1wbGVcbiAqIEN1c3RvbSBjb25maWd1cmF0aW9uIHdpdGggaW50ZXJjZXB0b3JzOlxuICogYGBgdHlwZXNjcmlwdFxuICogY29uc3QgY2xpZW50ID0gY3JlYXRlQXBpQ2xpZW50KHtcbiAqICAgYmFzZVVSTDogJ2h0dHBzOi8vYXBpLmV4YW1wbGUuY29tJyxcbiAqICAgY29ycmVsYXRpb25JZFByZWZpeDogJ3dlYicsXG4gKiAgIHRva2VuU3RvcmFnZUtleTogJ2F1dGhUb2tlbicsXG4gKiAgIHRpbWVvdXQ6IDYwMDAwLFxuICogICByZXF1ZXN0SW50ZXJjZXB0b3JzOiBbXG4gKiAgICAgYXN5bmMgKGNvbmZpZykgPT4ge1xuICogICAgICAgY29uc29sZS5sb2coJ01ha2luZyByZXF1ZXN0OicsIGNvbmZpZy51cmwpO1xuICogICAgICAgcmV0dXJuIGNvbmZpZztcbiAqICAgICB9XG4gKiAgIF1cbiAqIH0pO1xuICogYGBgXG4gKlxuICogQGV4YW1wbGVcbiAqIFdpdGggUmVhY3QgUXVlcnk6XG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBhcGlDbGllbnQgPSBjcmVhdGVBcGlDbGllbnQoe1xuICogICBiYXNlVVJMOiBwcm9jZXNzLmVudi5SRUFDVF9BUFBfQVBJX1VSTCxcbiAqICAgY29ycmVsYXRpb25JZFByZWZpeDogJ3dlYicsXG4gKiAgIHRva2VuU3RvcmFnZUtleTogJ3NlcnZpY2VUb2tlbidcbiAqIH0pO1xuICpcbiAqIGNvbnN0IHF1ZXJ5Q2xpZW50ID0gbmV3IFF1ZXJ5Q2xpZW50KCk7XG4gKlxuICogZnVuY3Rpb24gVXNlclByb2ZpbGUoeyB1c2VySWQgfSkge1xuICogICBjb25zdCB7IGRhdGE6IHJlc3BvbnNlIH0gPSB1c2VRdWVyeSh7XG4gKiAgICAgcXVlcnlLZXk6IFsndXNlcicsIHVzZXJJZF0sXG4gKiAgICAgcXVlcnlGbjogKCkgPT4gYXBpQ2xpZW50LmdldChgL3VzZXJzLyR7dXNlcklkfWApXG4gKiAgIH0pO1xuICogICBjb25zdCB1c2VyID0gcmVzcG9uc2U/LmFwaURhdGE7IC8vIEFjY2VzcyB0aGUgYWN0dWFsIGRhdGFcbiAqICAgLy8gLi4uXG4gKiB9XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUFwaUNsaWVudChjb25maWc6IEFwaUNsaWVudENvbmZpZyk6IEFwaUNsaWVudCB7XG4gIGNvbnN0IHtcbiAgICBiYXNlVVJMLFxuICAgIHRpbWVvdXQgPSAzMDAwMCxcbiAgICBjb3JyZWxhdGlvbklkUHJlZml4LFxuICAgIGluY2x1ZGVDb3JyZWxhdGlvbklkID0gdHJ1ZSxcbiAgICB0b2tlblN0b3JhZ2VLZXksXG4gICAgcmVxdWVzdEludGVyY2VwdG9ycyA9IFtdLFxuICAgIHJlc3BvbnNlSW50ZXJjZXB0b3JzID0gW10sXG4gICAgZXJyb3JJbnRlcmNlcHRvcnMgPSBbXSxcbiAgfSA9IGNvbmZpZztcblxuICBjb25zdCBjbGllbnQgPSBuZXcgQXBpQ2xpZW50KGJhc2VVUkwsIHRpbWVvdXQpO1xuXG4gIGNsaWVudC5hZGRSZXF1ZXN0SW50ZXJjZXB0b3IoY29uZmlnID0+IHtcbiAgICBjb25zdCB0b2tlbiA9IGxvY2FsU3RvcmFnZS5nZXRJdGVtKHRva2VuU3RvcmFnZUtleSk7XG5cbiAgICBpZiAodG9rZW4gJiYgIWNvbmZpZy5za2lwQXV0aFJlZnJlc2gpIHtcbiAgICAgIGNvbmZpZy5oZWFkZXJzID0ge1xuICAgICAgICAuLi5jb25maWcuaGVhZGVycyxcbiAgICAgICAgQXV0aG9yaXphdGlvbjogYEJlYXJlciAke3Rva2VufWAsXG4gICAgICB9O1xuICAgIH1cblxuICAgIHJldHVybiBjb25maWc7XG4gIH0pO1xuXG4gIC8vIENvbmZpZ3VyZSBjb3JyZWxhdGlvbiBJRFxuICBjbGllbnQuc2V0Q29ycmVsYXRpb25JZFByZWZpeChjb3JyZWxhdGlvbklkUHJlZml4IHx8ICd3ZWInKTtcbiAgY2xpZW50LnNldEluY2x1ZGVDb3JyZWxhdGlvbklkKGluY2x1ZGVDb3JyZWxhdGlvbklkIHx8IHRydWUpO1xuXG4gIC8vIC8vIFNldCBhdXRoIHRva2VuIGlmIHByb3ZpZGVkXG4gIC8vIGlmIChhdXRoVG9rZW4gIT09IHVuZGVmaW5lZCkge1xuICAvLyAgIGNsaWVudC5zZXRBdXRoVG9rZW4oYXV0aFRva2VuKTtcbiAgLy8gfVxuXG4gIC8vIEFkZCBpbnRlcmNlcHRvcnNcbiAgcmVxdWVzdEludGVyY2VwdG9ycy5mb3JFYWNoKGludGVyY2VwdG9yID0+IHtcbiAgICBjbGllbnQuYWRkUmVxdWVzdEludGVyY2VwdG9yKGludGVyY2VwdG9yKTtcbiAgfSk7XG5cbiAgcmVzcG9uc2VJbnRlcmNlcHRvcnMuZm9yRWFjaChpbnRlcmNlcHRvciA9PiB7XG4gICAgY2xpZW50LmFkZFJlc3BvbnNlSW50ZXJjZXB0b3IoaW50ZXJjZXB0b3IpO1xuICB9KTtcblxuICBlcnJvckludGVyY2VwdG9ycy5mb3JFYWNoKGludGVyY2VwdG9yID0+IHtcbiAgICBjbGllbnQuYWRkRXJyb3JJbnRlcmNlcHRvcihpbnRlcmNlcHRvcik7XG4gIH0pO1xuXG4gIHJldHVybiBjbGllbnQ7XG59XG5cbi8qKlxuICogR2V0cyB0aGUgZ2xvYmFsIHNpbmdsZXRvbiBBcGlDbGllbnQgaW5zdGFuY2UuXG4gKlxuICogVGhpcyBmdW5jdGlvbiByZXR1cm5zIHRoZSBleGlzdGluZyBnbG9iYWwgQXBpQ2xpZW50LiBUaGUgY2xpZW50IG11c3QgZmlyc3QgYmVcbiAqIGluaXRpYWxpemVkIHVzaW5nIHtAbGluayBpbml0aWFsaXplR2xvYmFsQXBpQ2xpZW50fS5cbiAqXG4gKiBAcmV0dXJucyBUaGUgZ2xvYmFsIEFwaUNsaWVudCBzaW5nbGV0b24gaW5zdGFuY2VcbiAqIEB0aHJvd3MgRXJyb3IgaWYgdGhlIGdsb2JhbCBjbGllbnQgaGFzIG5vdCBiZWVuIGluaXRpYWxpemVkXG4gKiBAcHVibGljXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIHNyYy9mZWF0dXJlcy91c2Vycy9hcGkudHNcbiAqIGltcG9ydCB7IGdldEdsb2JhbEFwaUNsaWVudCB9IGZyb20gJ0BnbndlYnNvZnQvdWknO1xuICpcbiAqIGNvbnN0IGNsaWVudCA9IGdldEdsb2JhbEFwaUNsaWVudCgpO1xuICpcbiAqIGV4cG9ydCBhc3luYyBmdW5jdGlvbiBmZXRjaFVzZXJzKCkge1xuICogICBjb25zdCB7IGFwaURhdGEgfSA9IGF3YWl0IGNsaWVudC5nZXQoJy91c2VycycpO1xuICogICByZXR1cm4gYXBpRGF0YTtcbiAqIH1cbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gZ2V0R2xvYmFsQXBpQ2xpZW50KCk6IEFwaUNsaWVudCB7XG4gIGlmICghZ2xvYmFsQXBpQ2xpZW50KSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgJ2dldEdsb2JhbEFwaUNsaWVudDogTm8gZ2xvYmFsIGNsaWVudCBleGlzdHMuIENhbGwgaW5pdGlhbGl6ZUdsb2JhbEFwaUNsaWVudCgpIGZpcnN0IHRvIGNvbmZpZ3VyZSB0aGUgY2xpZW50LidcbiAgICApO1xuICB9XG5cbiAgcmV0dXJuIGdsb2JhbEFwaUNsaWVudDtcbn1cblxuLyoqXG4gKiBJbml0aWFsaXplcyB0aGUgZ2xvYmFsIHNpbmdsZXRvbiBBcGlDbGllbnQgd2l0aCB0aGUgcHJvdmlkZWQgY29uZmlndXJhdGlvbi5cbiAqXG4gKiBUaGlzIHNob3VsZCBiZSBjYWxsZWQgb25jZSBkdXJpbmcgYXBwIGluaXRpYWxpemF0aW9uIChlLmcuLCBpbiB5b3VyIG1haW4gZW50cnkgcG9pbnQpLlxuICogQWZ0ZXIgaW5pdGlhbGl6YXRpb24sIHVzZSB7QGxpbmsgZ2V0R2xvYmFsQXBpQ2xpZW50fSB0byByZXRyaWV2ZSB0aGUgY2xpZW50IGFueXdoZXJlIGluIHlvdXIgYXBwLlxuICpcbiAqIEBwYXJhbSBjb25maWcgLSBDb25maWd1cmF0aW9uIG9wdGlvbnMgKGJhc2VVUkwsIGNvcnJlbGF0aW9uSWRQcmVmaXgsIHRva2VuU3RvcmFnZUtleSBhcmUgcmVxdWlyZWQpXG4gKiBAcmV0dXJucyBUaGUgaW5pdGlhbGl6ZWQgQXBpQ2xpZW50IGluc3RhbmNlXG4gKiBAdGhyb3dzIEVycm9yIGlmIHRoZSBnbG9iYWwgY2xpZW50IGhhcyBhbHJlYWR5IGJlZW4gaW5pdGlhbGl6ZWRcbiAqIEBwdWJsaWNcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gc3JjL21haW4udHN4IG9yIHNyYy9pbmRleC50c3hcbiAqIGltcG9ydCB7IGluaXRpYWxpemVHbG9iYWxBcGlDbGllbnQgfSBmcm9tICdAZ253ZWJzb2Z0L3VpJztcbiAqXG4gKiBpbml0aWFsaXplR2xvYmFsQXBpQ2xpZW50KHtcbiAqICAgYmFzZVVSTDogaW1wb3J0Lm1ldGEuZW52LlZJVEVfQVBJX1VSTCxcbiAqICAgY29ycmVsYXRpb25JZFByZWZpeDogJ3dlYicsXG4gKiAgIHRva2VuU3RvcmFnZUtleTogJ3NlcnZpY2VUb2tlbidcbiAqIH0pO1xuICogYGBgXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpbml0aWFsaXplR2xvYmFsQXBpQ2xpZW50KGNvbmZpZzogQXBpQ2xpZW50Q29uZmlnKTogQXBpQ2xpZW50IHtcbiAgaWYgKGdsb2JhbEFwaUNsaWVudCkge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICdpbml0aWFsaXplR2xvYmFsQXBpQ2xpZW50OiBHbG9iYWwgY2xpZW50IGFscmVhZHkgaW5pdGlhbGl6ZWQuIFVzZSByZXNldEdsb2JhbEFwaUNsaWVudCgpIGZpcnN0IGlmIHlvdSBuZWVkIHRvIHJlaW5pdGlhbGl6ZS4nXG4gICAgKTtcbiAgfVxuXG4gIGdsb2JhbEFwaUNsaWVudCA9IGNyZWF0ZUFwaUNsaWVudChjb25maWcpO1xuXG4gIHJldHVybiBnbG9iYWxBcGlDbGllbnQ7XG59XG5cbi8qKlxuICogUmVwbGFjZXMgdGhlIGdsb2JhbCBzaW5nbGV0b24gQXBpQ2xpZW50IHdpdGggYSBjdXN0b20gaW5zdGFuY2UuXG4gKlxuICogVXNlIHRoaXMgd2hlbiB5b3UgbmVlZCBmaW5lLWdyYWluZWQgY29udHJvbCBvdmVyIHRoZSBnbG9iYWwgY2xpZW50IGluc3RhbmNlLFxuICogc3VjaCBhcyBpbiB0ZXN0aW5nIHNjZW5hcmlvcyBvciB3aGVuIHlvdSd2ZSBtYW51YWxseSBjcmVhdGVkIGEgY2xpZW50IHdpdGhcbiAqIHNwZWNpZmljIGNvbmZpZ3VyYXRpb24uXG4gKlxuICogQHBhcmFtIGNsaWVudCAtIFRoZSBBcGlDbGllbnQgaW5zdGFuY2UgdG8gdXNlIGFzIHRoZSBnbG9iYWwgc2luZ2xldG9uXG4gKiBAcHVibGljXG4gKlxuICogQGV4YW1wbGVcbiAqIFRlc3Rpbmcgd2l0aCBhIG1vY2sgY2xpZW50OlxuICogYGBgdHlwZXNjcmlwdFxuICogY29uc3QgbW9ja0NsaWVudCA9IG5ldyBBcGlDbGllbnQoJ2h0dHA6Ly90ZXN0LmxvY2FsJyk7XG4gKiBtb2NrQ2xpZW50LmFkZFJlc3BvbnNlSW50ZXJjZXB0b3IoYXN5bmMgKHJlc3BvbnNlKSA9PiB7XG4gKiAgIC8vIE1vY2sgcmVzcG9uc2UgZGF0YVxuICogICByZXR1cm4geyBkYXRhOiBtb2NrRGF0YSB9O1xuICogfSk7XG4gKlxuICogc2V0R2xvYmFsQXBpQ2xpZW50KG1vY2tDbGllbnQpO1xuICogYGBgXG4gKlxuICogQGV4YW1wbGVcbiAqIEFkdmFuY2VkIGNvbmZpZ3VyYXRpb246XG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBjbGllbnQgPSBuZXcgQXBpQ2xpZW50KCdodHRwczovL2FwaS5leGFtcGxlLmNvbScsIDYwMDAwKTtcbiAqIGNsaWVudC5zZXRDb3JyZWxhdGlvbklkUHJlZml4KCdjdXN0b20nKTtcbiAqIGNsaWVudC5hZGRSZXF1ZXN0SW50ZXJjZXB0b3IoY3VzdG9tSW50ZXJjZXB0b3IpO1xuICpcbiAqIHNldEdsb2JhbEFwaUNsaWVudChjbGllbnQpO1xuICogYGBgXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBzZXRHbG9iYWxBcGlDbGllbnQoY2xpZW50OiBBcGlDbGllbnQpOiB2b2lkIHtcbiAgZ2xvYmFsQXBpQ2xpZW50ID0gY2xpZW50O1xufVxuXG4vKipcbiAqIFJlc2V0cyB0aGUgZ2xvYmFsIEFwaUNsaWVudCBzaW5nbGV0b24gdG8gbnVsbC5cbiAqXG4gKiBBZnRlciBjYWxsaW5nIHRoaXMsIHRoZSBuZXh0IGNhbGwgdG8ge0BsaW5rIGdldEdsb2JhbEFwaUNsaWVudH0gd2lsbCBjcmVhdGVcbiAqIGEgZnJlc2ggaW5zdGFuY2UuIFRoaXMgaXMgcGFydGljdWxhcmx5IHVzZWZ1bCBpbjpcbiAqIC0gVGVzdCBjbGVhbnVwIHRvIGVuc3VyZSB0ZXN0IGlzb2xhdGlvblxuICogLSBBcHBsaWNhdGlvbiBsb2dvdXQgdG8gY2xlYXIgYW55IGNhY2hlZCBzdGF0ZVxuICogLSBIb3QgbW9kdWxlIHJlbG9hZGluZyBkdXJpbmcgZGV2ZWxvcG1lbnRcbiAqXG4gKiBAcHVibGljXG4gKlxuICogQGV4YW1wbGVcbiAqIFRlc3QgY2xlYW51cDpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGFmdGVyRWFjaCgoKSA9PiB7XG4gKiAgIHJlc2V0R2xvYmFsQXBpQ2xpZW50KCk7XG4gKiB9KTtcbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBMb2dvdXQgZmxvdzpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGZ1bmN0aW9uIGhhbmRsZUxvZ291dCgpIHtcbiAqICAgLy8gQ2xlYXIgYXV0aGVudGljYXRpb25cbiAqICAgbG9jYWxTdG9yYWdlLnJlbW92ZUl0ZW0oJ3NlcnZpY2VUb2tlbicpO1xuICpcbiAqICAgLy8gQ2FuY2VsIGFsbCBwZW5kaW5nIHJlcXVlc3RzIGFuZCByZXNldCBjbGllbnRcbiAqICAgY29uc3QgY2xpZW50ID0gZ2V0R2xvYmFsQXBpQ2xpZW50KCk7XG4gKiAgIGNsaWVudC5jYW5jZWxBbGxSZXF1ZXN0cygpO1xuICogICByZXNldEdsb2JhbEFwaUNsaWVudCgpO1xuICpcbiAqICAgLy8gUmVkaXJlY3QgdG8gbG9naW5cbiAqICAgbmF2aWdhdGUoJy9sb2dpbicpO1xuICogfVxuICogYGBgXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiByZXNldEdsb2JhbEFwaUNsaWVudCgpOiB2b2lkIHtcbiAgZ2xvYmFsQXBpQ2xpZW50ID0gbnVsbDtcbn1cbiIsICIvLyBDYW5jZWxUb2tlbiBpbXBsZW1lbnRhdGlvblxuZXhwb3J0IGNsYXNzIENhbmNlbFRva2VuIHtcbiAgcHJpdmF0ZSBhYm9ydENvbnRyb2xsZXI6IEFib3J0Q29udHJvbGxlcjtcbiAgcHJpdmF0ZSBjYW5jZWxQcm9taXNlOiBQcm9taXNlPHZvaWQ+O1xuICBwcml2YXRlIGNhbmNlbFJlc29sdmU/OiAoKSA9PiB2b2lkO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHRoaXMuYWJvcnRDb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpO1xuICAgIHRoaXMuY2FuY2VsUHJvbWlzZSA9IG5ldyBQcm9taXNlKHJlc29sdmUgPT4ge1xuICAgICAgdGhpcy5jYW5jZWxSZXNvbHZlID0gcmVzb2x2ZTtcbiAgICB9KTtcbiAgfVxuXG4gIGdldCBzaWduYWwoKTogQWJvcnRTaWduYWwge1xuICAgIHJldHVybiB0aGlzLmFib3J0Q29udHJvbGxlci5zaWduYWw7XG4gIH1cblxuICBjYW5jZWwocmVhc29uPzogc3RyaW5nKTogdm9pZCB7XG4gICAgdGhpcy5hYm9ydENvbnRyb2xsZXIuYWJvcnQocmVhc29uKTtcbiAgICB0aGlzLmNhbmNlbFJlc29sdmU/LigpO1xuICB9XG5cbiAgZ2V0IGlzQ2FuY2VsbGVkKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLmFib3J0Q29udHJvbGxlci5zaWduYWwuYWJvcnRlZDtcbiAgfVxuXG4gIHRocm93SWZDYW5jZWxsZWQoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuaXNDYW5jZWxsZWQpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignUmVxdWVzdCBjYW5jZWxsZWQnKTtcbiAgICB9XG4gIH1cblxuICBzdGF0aWMgc291cmNlKCk6IHsgdG9rZW46IENhbmNlbFRva2VuOyBjYW5jZWw6IChyZWFzb24/OiBzdHJpbmcpID0+IHZvaWQgfSB7XG4gICAgY29uc3QgdG9rZW4gPSBuZXcgQ2FuY2VsVG9rZW4oKTtcblxuICAgIHJldHVybiB7XG4gICAgICB0b2tlbixcbiAgICAgIGNhbmNlbDogKHJlYXNvbj86IHN0cmluZykgPT4gdG9rZW4uY2FuY2VsKHJlYXNvbiksXG4gICAgfTtcbiAgfVxufVxuIiwgIi8vIFJlYWN0IEhvb2tzIHdpdGggQXBpUmVzcG9uc2UgZm9ybWF0XG5pbXBvcnQgeyB1c2VDYWxsYmFjayB9IGZyb20gJ3JlYWN0JztcblxuaW1wb3J0IHR5cGUgeyBBcGlFcnJvciB9IGZyb20gJy4vdHlwZXMnO1xuXG4vLyBFeHBvcnQgZmFjdG9yeSBtZXRob2RzIGFuZCB0eXBlc1xuZXhwb3J0IHtcbiAgY3JlYXRlQXBpQ2xpZW50LFxuICBnZXRHbG9iYWxBcGlDbGllbnQsXG4gIHNldEdsb2JhbEFwaUNsaWVudCxcbiAgcmVzZXRHbG9iYWxBcGlDbGllbnQsXG59IGZyb20gJy4vY3JlYXRlQXBpQ2xpZW50JztcblxuZXhwb3J0IHR5cGUgeyBBcGlDbGllbnRDb25maWcgfSBmcm9tICcuL2NyZWF0ZUFwaUNsaWVudCc7XG5cbmV4cG9ydCB7IEFwaUNsaWVudCB9IGZyb20gJy4vQXBpQ2xpZW50JztcblxuLy8gSGVscGVyIGhvb2sgZm9yIGZvcm0gdmFsaWRhdGlvbiBlcnJvcnNcbmV4cG9ydCBmdW5jdGlvbiB1c2VWYWxpZGF0aW9uRXJyb3JzKGVycm9yOiBBcGlFcnJvciB8IG51bGwpIHtcbiAgY29uc3QgZ2V0RmllbGRFcnJvciA9IHVzZUNhbGxiYWNrKFxuICAgIChmaWVsZDogc3RyaW5nKTogc3RyaW5nIHwgbnVsbCA9PiB7XG4gICAgICBpZiAoIWVycm9yPy5lcnJvcnMgfHwgIWVycm9yLmVycm9yc1tmaWVsZF0pIHJldHVybiBudWxsO1xuXG4gICAgICBjb25zdCBmaWVsZEVycm9yID0gZXJyb3IuZXJyb3JzW2ZpZWxkXTtcblxuICAgICAgaWYgKHR5cGVvZiBmaWVsZEVycm9yID09PSAnc3RyaW5nJykgcmV0dXJuIGZpZWxkRXJyb3I7XG4gICAgICBpZiAoQXJyYXkuaXNBcnJheShmaWVsZEVycm9yKSkgcmV0dXJuIGZpZWxkRXJyb3JbMF07XG4gICAgICBpZiAodHlwZW9mIGZpZWxkRXJyb3IgPT09ICdvYmplY3QnICYmICdtZXNzYWdlJyBpbiBmaWVsZEVycm9yKSB7XG4gICAgICAgIHJldHVybiBmaWVsZEVycm9yLm1lc3NhZ2U7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiBudWxsO1xuICAgIH0sXG4gICAgW2Vycm9yXVxuICApO1xuXG4gIGNvbnN0IGhhc0ZpZWxkRXJyb3IgPSB1c2VDYWxsYmFjayhcbiAgICAoZmllbGQ6IHN0cmluZyk6IGJvb2xlYW4gPT4ge1xuICAgICAgcmV0dXJuICEhZ2V0RmllbGRFcnJvcihmaWVsZCk7XG4gICAgfSxcbiAgICBbZ2V0RmllbGRFcnJvcl1cbiAgKTtcblxuICBjb25zdCBnZXRBbGxFcnJvcnMgPSB1c2VDYWxsYmFjaygoKTogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9PiB7XG4gICAgaWYgKCFlcnJvcj8uZXJyb3JzKSByZXR1cm4ge307XG5cbiAgICBjb25zdCByZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7fTtcblxuICAgIE9iamVjdC5lbnRyaWVzKGVycm9yLmVycm9ycykuZm9yRWFjaCgoW2tleSwgdmFsdWVdKSA9PiB7XG4gICAgICBpZiAodHlwZW9mIHZhbHVlID09PSAnc3RyaW5nJykge1xuICAgICAgICByZXN1bHRba2V5XSA9IHZhbHVlO1xuICAgICAgfSBlbHNlIGlmIChBcnJheS5pc0FycmF5KHZhbHVlKSkge1xuICAgICAgICByZXN1bHRba2V5XSA9IHZhbHVlLmpvaW4oJywgJyk7XG4gICAgICB9IGVsc2UgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ29iamVjdCcgJiYgdmFsdWUgJiYgJ21lc3NhZ2UnIGluIHZhbHVlKSB7XG4gICAgICAgIHJlc3VsdFtrZXldID0gdmFsdWUubWVzc2FnZTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIHJldHVybiByZXN1bHQ7XG4gIH0sIFtlcnJvcl0pO1xuXG4gIHJldHVybiB7XG4gICAgZ2V0RmllbGRFcnJvcixcbiAgICBoYXNGaWVsZEVycm9yLFxuICAgIGdldEFsbEVycm9ycyxcbiAgICBoYXNFcnJvcnM6IGVycm9yPy5lcnJvcnMsXG4gIH07XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBQcm9wc1dpdGhDaGlsZHJlbiB9IGZyb20gJ3JlYWN0JztcblxudHlwZSBBdXRob3JpemVkVmlld1Byb3BzID0gUHJvcHNXaXRoQ2hpbGRyZW4gJiB7XG4gIHNob3c6IGJvb2xlYW47XG59O1xuXG5leHBvcnQgY29uc3QgQXV0aG9yaXplZFZpZXcgPSAoeyBjaGlsZHJlbiwgc2hvdyB9OiBBdXRob3JpemVkVmlld1Byb3BzKSA9PiB7XG4gIGlmICghc2hvdykgcmV0dXJuIDw+PC8+O1xuXG4gIHJldHVybiA8PntjaGlsZHJlbn08Lz47XG59O1xuIiwgImltcG9ydCB0eXBlIHsgQnV0dG9uUHJvcHMgfSBmcm9tICdAbXVpL21hdGVyaWFsJztcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJ0BtdWkvbWF0ZXJpYWwnO1xuaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0JztcblxuZXhwb3J0IGNvbnN0IENhbmNlbEJ1dHRvbjogUmVhY3QuRkM8QnV0dG9uUHJvcHM+ID0gKHtcbiAgY2hpbGRyZW4gPSAnQ2FuY2VsJyxcbiAgdmFyaWFudCA9ICdvdXRsaW5lZCcsXG4gIHN4LFxuICAuLi5yZXN0XG59KSA9PiAoXG4gIDxCdXR0b24gdmFyaWFudD17dmFyaWFudH0gc3g9e3sgd2lkdGg6ICc2cmVtJywgLi4uc3ggfX0gey4uLnJlc3R9PlxuICAgIHtjaGlsZHJlbn1cbiAgPC9CdXR0b24+XG4pO1xuIiwgImltcG9ydCB0eXBlIHsgU3hQcm9wcyB9IGZyb20gJ0BtdWkvbWF0ZXJpYWwnO1xuaW1wb3J0IHsgQnV0dG9uIH0gZnJvbSAnQG11aS9tYXRlcmlhbCc7XG5cbi8qKlxuICogUHJvcHMgZm9yIHRoZSBDbGVhckJ1dHRvbiBjb21wb25lbnQuXG4gKlxuICogQHB1YmxpY1xuICovXG5pbnRlcmZhY2UgQ2xlYXJCdXR0b25Qcm9wcyB7XG4gIC8qKlxuICAgKiBJbmRpY2F0ZXMgaWYgYSBmb3JtIG9yIG9wZXJhdGlvbiBpcyBjdXJyZW50bHkgYmVpbmcgc3VibWl0dGVkLlxuICAgKiBXaGVuIHRydWUsIHRoZSBidXR0b24gaXMgZGlzYWJsZWQgdG8gcHJldmVudCBtdWx0aXBsZSBjbGVhciBhY3Rpb25zLlxuICAgKi9cbiAgaXNTdWJtaXR0aW5nOiBib29sZWFuO1xuICAvKipcbiAgICogQ2FsbGJhY2sgZnVuY3Rpb24gZXhlY3V0ZWQgd2hlbiB0aGUgY2xlYXIgYnV0dG9uIGlzIGNsaWNrZWQuXG4gICAqIFNob3VsZCBoYW5kbGUgdGhlIGNsZWFyaW5nIGxvZ2ljIChlLmcuLCBmb3JtIHJlc2V0LCBkYXRhIGNsZWFyaW5nKS5cbiAgICovXG4gIGhhbmRsZUNsZWFyOiAoKSA9PiB2b2lkO1xuICAvKipcbiAgICogT3B0aW9uYWwgTVVJIHN4IHByb3AgZm9yIGN1c3RvbSBzdHlsaW5nLlxuICAgKiBAZXhhbXBsZSB7IG10OiAyLCBjb2xvcjogJ3dhcm5pbmcubWFpbicgfVxuICAgKi9cbiAgc3g/OiBTeFByb3BzO1xuICAvKipcbiAgICogT3B0aW9uYWwgbG9jYWxTdG9yYWdlIGtleSB0byByZW1vdmUgd2hlbiBjbGVhcmluZy5cbiAgICogSWYgcHJvdmlkZWQsIHRoZSBjb3JyZXNwb25kaW5nIGxvY2FsU3RvcmFnZSBpdGVtIHdpbGwgYmUgcmVtb3ZlZCBvbiBjbGljay5cbiAgICogQGV4YW1wbGUgXCJ1c2VyLXByZWZlcmVuY2VzXCIgb3IgXCJmb3JtLWRhdGFcIlxuICAgKi9cbiAgc3RvcmVLZXk/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogU3RhbmRhcmRpemVkIGNsZWFyIGJ1dHRvbiBjb21wb25lbnQgd2l0aCBsb2NhbFN0b3JhZ2UgaW50ZWdyYXRpb24uXG4gKlxuICogVGhpcyBjb21wb25lbnQgcHJvdmlkZXMgYSBjb25zaXN0ZW50IGNsZWFyIGJ1dHRvbiBpbXBsZW1lbnRhdGlvbiB0aGF0IGhhbmRsZXNcbiAqIGJvdGggY2FsbGJhY2sgZXhlY3V0aW9uIGFuZCBvcHRpb25hbCBsb2NhbFN0b3JhZ2UgY2xlYW51cC4gSXQgYXV0b21hdGljYWxseVxuICogZGlzYWJsZXMgZHVyaW5nIGZvcm0gc3VibWlzc2lvbnMgYW5kIGNhbiBjbGVhciBzdG9yZWQgZGF0YSB3aGVuIG5lZWRlZC5cbiAqXG4gKiBAZXhhbXBsZVxuICogQmFzaWMgdXNhZ2U6XG4gKiBgYGB0c3hcbiAqIDxDbGVhckJ1dHRvblxuICogICBpc1N1Ym1pdHRpbmc9e2lzTG9hZGluZ31cbiAqICAgaGFuZGxlQ2xlYXI9eygpID0+IGZvcm0ucmVzZXQoKX1cbiAqIC8+XG4gKiBgYGBcbiAqXG4gKiBAZXhhbXBsZVxuICogV2l0aCBsb2NhbFN0b3JhZ2UgY2xlYW51cDpcbiAqIGBgYHRzeFxuICogPENsZWFyQnV0dG9uXG4gKiAgIGlzU3VibWl0dGluZz17Zm9ybS5mb3JtU3RhdGUuaXNTdWJtaXR0aW5nfVxuICogICBoYW5kbGVDbGVhcj17KCkgPT4gc2V0RmlsdGVycyh7fSl9XG4gKiAgIHN0b3JlS2V5PVwidXNlci1maWx0ZXJzXCJcbiAqIC8+XG4gKiBgYGBcbiAqXG4gKiBAcGFyYW0gcHJvcHMgLSBDb21wb25lbnQgcHJvcHMgZm9yIGNsZWFyIGJ1dHRvbiBjb25maWd1cmF0aW9uXG4gKiBAcmV0dXJucyBNVUkgQnV0dG9uIGNvbXBvbmVudCBjb25maWd1cmVkIGFzIGEgY2xlYXIgYnV0dG9uXG4gKlxuICogQHB1YmxpY1xuICovXG5leHBvcnQgY29uc3QgQ2xlYXJCdXR0b24gPSAoe1xuICBpc1N1Ym1pdHRpbmcsXG4gIGhhbmRsZUNsZWFyLFxuICBzeCxcbiAgc3RvcmVLZXksXG59OiBDbGVhckJ1dHRvblByb3BzKSA9PiB7XG4gIGNvbnN0IG9uQ2xpY2sgPSAoKSA9PiB7XG4gICAgaGFuZGxlQ2xlYXIoKTtcbiAgICBpZiAoc3RvcmVLZXkgIT0gbnVsbCkge1xuICAgICAgbG9jYWxTdG9yYWdlLnJlbW92ZUl0ZW0oc3RvcmVLZXkpO1xuICAgIH1cbiAgfTtcblxuICByZXR1cm4gKFxuICAgIDxCdXR0b25cbiAgICAgIHZhcmlhbnQ9XCJvdXRsaW5lZFwiXG4gICAgICBvbkNsaWNrPXtvbkNsaWNrfVxuICAgICAgZGlzYWJsZWQ9e2lzU3VibWl0dGluZ31cbiAgICAgIHN4PXtzeH1cbiAgICA+XG4gICAgICBDbGVhclxuICAgIDwvQnV0dG9uPlxuICApO1xufTtcbiIsICJpbXBvcnQgeyBDb250YWluZXIsIHR5cGUgU3hQcm9wcyB9IGZyb20gJ0BtdWkvbWF0ZXJpYWwnO1xuaW1wb3J0IHR5cGUgeyBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCc7XG5cbmludGVyZmFjZSBTaW1wbGVDb250YWluZXJQcm9wcyB7XG4gIGNoaWxkcmVuOiBSZWFjdE5vZGU7XG4gIGNsYXNzTmFtZT86IHN0cmluZztcbiAgc3g/OiBTeFByb3BzO1xufVxuXG5leHBvcnQgY29uc3QgU2ltcGxlQ29udGFpbmVyID0gKHtcbiAgY2hpbGRyZW4sXG4gIGNsYXNzTmFtZSxcbiAgc3gsXG59OiBTaW1wbGVDb250YWluZXJQcm9wcykgPT4gKFxuICA8Q29udGFpbmVyIGNsYXNzTmFtZT17Y2xhc3NOYW1lfSBzeD17eyAuLi5zeCB9fT5cbiAgICB7Y2hpbGRyZW59XG4gIDwvQ29udGFpbmVyPlxuKTtcbiIsICJpbXBvcnQgRmlsdGVyQWx0SWNvbiBmcm9tICdAbXVpL2ljb25zLW1hdGVyaWFsL0ZpbHRlckFsdCc7XG5pbXBvcnQgeyBMb2FkaW5nQnV0dG9uIH0gZnJvbSAnQG11aS9sYWInO1xuaW1wb3J0IHR5cGUgeyBTeFByb3BzIH0gZnJvbSAnQG11aS9tYXRlcmlhbCc7XG5pbXBvcnQgeyBCYWRnZSB9IGZyb20gJ0BtdWkvbWF0ZXJpYWwnO1xuXG4vKipcbiAqIFByb3BzIGZvciB0aGUgRmlsdGVyQnV0dG9uIGNvbXBvbmVudC5cbiAqXG4gKiBAcHVibGljXG4gKi9cbmludGVyZmFjZSBGaWx0ZXJCdXR0b25Qcm9wcyB7XG4gIC8qKlxuICAgKiBJbmRpY2F0ZXMgaWYgYSBmaWx0ZXIgb3BlcmF0aW9uIGlzIGN1cnJlbnRseSBiZWluZyBwcm9jZXNzZWQuXG4gICAqIFdoZW4gdHJ1ZSwgc2hvd3MgbG9hZGluZyBzcGlubmVyIGFuZCBkaXNhYmxlcyB0aGUgYnV0dG9uLlxuICAgKi9cbiAgaXNTdWJtaXR0aW5nOiBib29sZWFuO1xuICAvKipcbiAgICogQ29udHJvbHMgYnV0dG9uIHZpc2liaWxpdHkgYW5kIGVuYWJsZWQgc3RhdGUuXG4gICAqIFdoZW4gZmFsc2UsIHRoZSBidXR0b24gaXMgZGlzYWJsZWQuXG4gICAqIEBkZWZhdWx0VmFsdWUgdHJ1ZVxuICAgKi9cbiAgc2hvdz86IGJvb2xlYW47XG4gIC8qKlxuICAgKiBDdXN0b20gdGV4dCB0byBkaXNwbGF5IG9uIHRoZSBidXR0b24uXG4gICAqIEBkZWZhdWx0VmFsdWUgXCJGaWx0ZXJcIlxuICAgKiBAZXhhbXBsZSBcIkFwcGx5IEZpbHRlcnNcIiBvciBcIlNlYXJjaFwiXG4gICAqL1xuICB0aXRsZT86IHN0cmluZztcbiAgLyoqXG4gICAqIEN1c3RvbSBpY29uIHRvIGRpc3BsYXkgaW5zdGVhZCBvZiB0aGUgZGVmYXVsdCBmaWx0ZXIgaWNvbi5cbiAgICogQGV4YW1wbGUgPFNlYXJjaEljb24gLz5cbiAgICovXG4gIGljb24/OiBSZWFjdC5SZWFjdE5vZGU7XG4gIC8qKlxuICAgKiBPcHRpb25hbCBNVUkgc3ggcHJvcCBmb3IgY3VzdG9tIGJ1dHRvbiBzdHlsaW5nLlxuICAgKiBAZXhhbXBsZSB7IG10OiAyLCBtaW5XaWR0aDogMTIwIH1cbiAgICovXG4gIHN4PzogU3hQcm9wcztcbiAgLyoqXG4gICAqIE9wdGlvbmFsIE1VSSBzeCBwcm9wIGZvciBjdXN0b20gaWNvbiBzdHlsaW5nLlxuICAgKiBAZXhhbXBsZSB7IGNvbG9yOiAncHJpbWFyeS5tYWluJywgZm9udFNpemU6IDE4IH1cbiAgICovXG4gIGljb25TeD86IFN4UHJvcHM7XG59XG5cbi8qKlxuICogRmlsdGVyIGJ1dHRvbiBjb21wb25lbnQgd2l0aCBsb2FkaW5nIHN0YXRlcyBhbmQgY3VzdG9taXphYmxlIGFwcGVhcmFuY2UuXG4gKlxuICogVGhpcyBjb21wb25lbnQgcHJvdmlkZXMgYSBzdGFuZGFyZGl6ZWQgZmlsdGVyL3N1Ym1pdCBidXR0b24gd2l0aCBpbnRlZ3JhdGVkIGxvYWRpbmdcbiAqIHN0YXRlcywgaWNvbiBzdXBwb3J0LCBhbmQgYmFkZ2Ugc3R5bGluZy4gSXQncyBkZXNpZ25lZCBmb3IgdXNlIGluIGZpbHRlciBmb3Jtc1xuICogYW5kIHNlYXJjaCBpbnRlcmZhY2VzIHdoZXJlIHVzZXJzIG5lZWQgdG8gYXBwbHkgZmlsdGVyaW5nIGNyaXRlcmlhLlxuICpcbiAqIEBleGFtcGxlXG4gKiBCYXNpYyB1c2FnZTpcbiAqIGBgYHRzeFxuICogPEZpbHRlckJ1dHRvblxuICogICBpc1N1Ym1pdHRpbmc9e2lzTG9hZGluZ31cbiAqICAgc2hvdz17aGFzRmlsdGVyc31cbiAqIC8+XG4gKiBgYGBcbiAqXG4gKiBAZXhhbXBsZVxuICogQ3VzdG9tIHRpdGxlIGFuZCBpY29uOlxuICogYGBgdHN4XG4gKiA8RmlsdGVyQnV0dG9uXG4gKiAgIGlzU3VibWl0dGluZz17Zm9ybS5mb3JtU3RhdGUuaXNTdWJtaXR0aW5nfVxuICogICB0aXRsZT1cIkFwcGx5IFNlYXJjaFwiXG4gKiAgIGljb249ezxTZWFyY2hJY29uIC8+fVxuICogICBzaG93PXt0cnVlfVxuICogLz5cbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBXaXRoIGN1c3RvbSBzdHlsaW5nOlxuICogYGBgdHN4XG4gKiA8RmlsdGVyQnV0dG9uXG4gKiAgIGlzU3VibWl0dGluZz17aXNQcm9jZXNzaW5nfVxuICogICB0aXRsZT1cIkZpbHRlciBSZXN1bHRzXCJcbiAqICAgc3g9e3sgbWluV2lkdGg6IDE1MCwgbXQ6IDIgfX1cbiAqICAgaWNvblN4PXt7IGZvbnRTaXplOiAyMCB9fVxuICogLz5cbiAqIGBgYFxuICpcbiAqIEBwYXJhbSBwcm9wcyAtIENvbXBvbmVudCBwcm9wcyBmb3IgZmlsdGVyIGJ1dHRvbiBjb25maWd1cmF0aW9uXG4gKiBAcmV0dXJucyBNVUkgTG9hZGluZ0J1dHRvbiBjb21wb25lbnQgY29uZmlndXJlZCBhcyBhIGZpbHRlciBidXR0b25cbiAqXG4gKiBAcHVibGljXG4gKi9cbmV4cG9ydCBjb25zdCBGaWx0ZXJCdXR0b24gPSAoe1xuICBpc1N1Ym1pdHRpbmcsXG4gIHNob3csXG4gIHRpdGxlLFxuICBpY29uLFxuICBzeCxcbiAgaWNvblN4LFxufTogRmlsdGVyQnV0dG9uUHJvcHMpID0+IHtcbiAgcmV0dXJuIChcbiAgICA8TG9hZGluZ0J1dHRvblxuICAgICAgdHlwZT1cInN1Ym1pdFwiXG4gICAgICB2YXJpYW50PVwiY29udGFpbmVkXCJcbiAgICAgIGxvYWRpbmc9e2lzU3VibWl0dGluZ31cbiAgICAgIGRpc2FibGVkPXshc2hvd31cbiAgICAgIGRpc2FibGVSaXBwbGVcbiAgICAgIGNvbG9yPVwicHJpbWFyeVwiXG4gICAgICBzeD17e1xuICAgICAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgICAgIGFsaWduSXRlbXM6ICdjZW50ZXInLFxuICAgICAgICAuLi5zeCxcbiAgICAgIH19XG4gICAgICBzdGFydEljb249e1xuICAgICAgICA8QmFkZ2UgY29sb3I9XCJlcnJvclwiIHZhcmlhbnQ9XCJzdGFuZGFyZFwiPlxuICAgICAgICAgIHtpY29uID8gaWNvbiA6IDxGaWx0ZXJBbHRJY29uIHdpZHRoPVwiMjBcIiBoZWlnaHQ9XCIyMFwiIHN4PXtpY29uU3h9IC8+fVxuICAgICAgICA8L0JhZGdlPlxuICAgICAgfVxuICAgID5cbiAgICAgIHt0aXRsZT8udHJpbSgpID09PSAnJyB8fCAhdGl0bGUgPyAnRmlsdGVyJyA6IHRpdGxlfVxuICAgIDwvTG9hZGluZ0J1dHRvbj5cbiAgKTtcbn07XG4iLCAiaW1wb3J0IENoaXAgZnJvbSAnQG11aS9tYXRlcmlhbC9DaGlwJztcbmltcG9ydCB7IG1lbW8gfSBmcm9tICdyZWFjdCc7XG5cbi8vIEluZGl2aWR1YWwgY2hpcCBjb21wb25lbnQgdG8gcHJldmVudCB1bm5lY2Vzc2FyeSByZXJlbmRlcnMgb2Ygc2libGluZyBjaGlwc1xuZXhwb3J0IGNvbnN0IEZpbHRlckNoaXAgPSBtZW1vKFxuICAoe1xuICAgIGZpZWxkS2V5LFxuICAgIGZpbHRlcixcbiAgICBvbkRlbGV0ZSxcbiAgfToge1xuICAgIGZpZWxkS2V5OiBzdHJpbmc7XG4gICAgZmlsdGVyOiB7IExhYmVsOiBzdHJpbmc7IFZhbHVlOiB1bmtub3duIH07XG4gICAgb25EZWxldGU/OiAoKSA9PiB2b2lkO1xuICB9KSA9PiB7XG4gICAgY29uc3QgaGFzVmFsdWUgPVxuICAgICAgZmlsdGVyLlZhbHVlICE9PSBudWxsICYmXG4gICAgICBmaWx0ZXIuVmFsdWUgIT09IHVuZGVmaW5lZCAmJlxuICAgICAgZmlsdGVyLlZhbHVlICE9PSAnJztcbiAgICBjb25zdCBsYWJlbCA9IGAke2ZpZWxkS2V5LnJlcGxhY2UoJ1BLJywgJycpfTogJHtmaWx0ZXIuTGFiZWx9YDtcblxuICAgIHJldHVybiAoXG4gICAgICA8Q2hpcFxuICAgICAgICBrZXk9e2ZpZWxkS2V5fVxuICAgICAgICBsYWJlbD17bGFiZWx9XG4gICAgICAgIHZhcmlhbnQ9e2hhc1ZhbHVlID8gJ2ZpbGxlZCcgOiAnb3V0bGluZWQnfVxuICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICBvbkRlbGV0ZT17aGFzVmFsdWUgPyBvbkRlbGV0ZSA6IHVuZGVmaW5lZH1cbiAgICAgIC8+XG4gICAgKTtcbiAgfVxuKTtcblxuRmlsdGVyQ2hpcC5kaXNwbGF5TmFtZSA9ICdGaWx0ZXJDaGlwJztcbiIsICJpbXBvcnQgeyBDYXJkLCBDYXJkQ29udGVudCwgVHlwb2dyYXBoeSwgQm94IH0gZnJvbSAnQG11aS9tYXRlcmlhbCc7XG5pbXBvcnQgeyBtZW1vLCB1c2VNZW1vIH0gZnJvbSAncmVhY3QnO1xuXG5pbXBvcnQgeyBGaWx0ZXJDaGlwIH0gZnJvbSAnLi9GaWx0ZXJDaGlwJztcblxuaW50ZXJmYWNlIEZpbHRlckRpc3BsYXlQcm9wczxURmlsdGVyTW9kZWw+IHtcbiAgZnJpZW5kbHlGaWx0ZXI6IFJlY29yZDxzdHJpbmcsIHsgTGFiZWw6IHN0cmluZzsgVmFsdWU6IHVua25vd24gfT47XG4gIG9uRnJpZW5kbHlGaWx0ZXJDaGFuZ2U/OiAoZmllbGRLZXk6IGtleW9mIFRGaWx0ZXJNb2RlbCkgPT4gdm9pZDtcbn1cblxuZXhwb3J0IGNvbnN0IFByb2dyYW1zRmlsdGVyRGlzcGxheSA9IG1lbW8oXG4gIDxURmlsdGVyTW9kZWwgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4oXG4gICAgcHJvcHM6IEZpbHRlckRpc3BsYXlQcm9wczxURmlsdGVyTW9kZWw+XG4gICkgPT4ge1xuICAgIGNvbnN0IHsgZnJpZW5kbHlGaWx0ZXIsIG9uRnJpZW5kbHlGaWx0ZXJDaGFuZ2UgfSA9IHByb3BzO1xuXG4gICAgLy8gTWVtb2l6ZSBkZWxldGUgaGFuZGxlcnMgdG8gcHJldmVudCByZWNyZWF0aW9uXG4gICAgY29uc3QgZGVsZXRlSGFuZGxlcnMgPSB1c2VNZW1vKCgpID0+IHtcbiAgICAgIGlmICghb25GcmllbmRseUZpbHRlckNoYW5nZSkgcmV0dXJuIHt9O1xuXG4gICAgICBjb25zdCBoYW5kbGVyczogUmVjb3JkPHN0cmluZywgKCkgPT4gdm9pZD4gPSB7fTtcblxuICAgICAgZm9yIChjb25zdCBrZXkgb2YgT2JqZWN0LmtleXMoZnJpZW5kbHlGaWx0ZXIpKSB7XG4gICAgICAgIGhhbmRsZXJzW2tleV0gPSAoKSA9PiBvbkZyaWVuZGx5RmlsdGVyQ2hhbmdlKGtleSBhcyBrZXlvZiBURmlsdGVyTW9kZWwpO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gaGFuZGxlcnM7XG4gICAgfSwgW29uRnJpZW5kbHlGaWx0ZXJDaGFuZ2UsIGZyaWVuZGx5RmlsdGVyXSk7XG5cbiAgICAvLyBNZW1vaXplIGNoaXAgbGlzdCB0byBwcmV2ZW50IHVubmVjZXNzYXJ5IHJlY2FsY3VsYXRpb25zXG4gICAgY29uc3QgY2hpcExpc3QgPSB1c2VNZW1vKCgpID0+IHtcbiAgICAgIHJldHVybiBPYmplY3QuZW50cmllcyhmcmllbmRseUZpbHRlcikubWFwKChba2V5LCBmaWx0ZXJdKSA9PiAoXG4gICAgICAgIDxGaWx0ZXJDaGlwXG4gICAgICAgICAga2V5PXtrZXl9XG4gICAgICAgICAgZmllbGRLZXk9e2tleX1cbiAgICAgICAgICBmaWx0ZXI9e2ZpbHRlciBhcyB7IExhYmVsOiBzdHJpbmc7IFZhbHVlOiB1bmtub3duIH19XG4gICAgICAgICAgb25EZWxldGU9e2RlbGV0ZUhhbmRsZXJzW2tleV19XG4gICAgICAgIC8+XG4gICAgICApKTtcbiAgICB9LCBbZnJpZW5kbHlGaWx0ZXIsIGRlbGV0ZUhhbmRsZXJzXSk7XG5cbiAgICByZXR1cm4gKFxuICAgICAgPENhcmQgc3g9e3sgbWI6IDIgfX0+XG4gICAgICAgIDxDYXJkQ29udGVudD5cbiAgICAgICAgICA8VHlwb2dyYXBoeSB2YXJpYW50PVwiaDZcIiBndXR0ZXJCb3R0b20+XG4gICAgICAgICAgICBBY3RpdmUgRmlsdGVyc1xuICAgICAgICAgIDwvVHlwb2dyYXBoeT5cbiAgICAgICAgICA8Qm94IGRpc3BsYXk9XCJmbGV4XCIgZ2FwPXsxfSBmbGV4V3JhcD1cIndyYXBcIj5cbiAgICAgICAgICAgIHtjaGlwTGlzdH1cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgPC9DYXJkQ29udGVudD5cbiAgICAgIDwvQ2FyZD5cbiAgICApO1xuICB9XG4pO1xuXG5Qcm9ncmFtc0ZpbHRlckRpc3BsYXkuZGlzcGxheU5hbWUgPSAnRmlsdGVyRGlzcGxheSc7XG5cbmV4cG9ydCB0eXBlIHsgRmlsdGVyRGlzcGxheVByb3BzIH07XG4iLCAiaW1wb3J0IE1hbmFnZVNlYXJjaEljb24gZnJvbSAnQG11aS9pY29ucy1tYXRlcmlhbC9NYW5hZ2VTZWFyY2gnO1xuaW1wb3J0IHR5cGUgeyBTeFByb3BzIH0gZnJvbSAnQG11aS9tYXRlcmlhbCc7XG5pbXBvcnQge1xuICBCb3gsXG4gIENhcmQsXG4gIENhcmRDb250ZW50LFxuICBDYXJkSGVhZGVyLFxuICBEaXZpZGVyLFxuICBHcmlkLFxuICBUeXBvZ3JhcGh5LFxuICB1c2VUaGVtZSxcbn0gZnJvbSAnQG11aS9tYXRlcmlhbCc7XG5pbXBvcnQgdHlwZSB7IFByb3BzV2l0aENoaWxkcmVuLCBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCc7XG5cbi8qKlxuICogUHJvcHMgZm9yIHRoZSBGaWx0ZXJXcmFwcGVyIGNvbXBvbmVudC5cbiAqXG4gKiBAcHVibGljXG4gKi9cbnR5cGUgRmlsdGVyV3JhcHBlclByb3BzID0gUHJvcHNXaXRoQ2hpbGRyZW48e1xuICAvKipcbiAgICogVGl0bGUgdGV4dCBkaXNwbGF5ZWQgaW4gdGhlIGNhcmQgaGVhZGVyLlxuICAgKiBAZGVmYXVsdFZhbHVlIFwiRmlsdGVyXCJcbiAgICogQGV4YW1wbGUgXCJTZWFyY2ggQ3JpdGVyaWFcIiBvciBcIkFkdmFuY2VkIEZpbHRlcnNcIlxuICAgKi9cbiAgdGl0bGU/OiBzdHJpbmc7XG4gIC8qKlxuICAgKiBOdW1iZXIgb2YgYWN0aXZlIGZpbHRlcnMgdG8gZGlzcGxheSBpbiB0aGUgaGVhZGVyLlxuICAgKiBPbmx5IHNob3duIHdoZW4gc2hvd0NvdW50IGlzIHRydWUuXG4gICAqIEBleGFtcGxlIDMgZm9yIFwiRmlsdGVyICgzKVwiXG4gICAqL1xuICBmaWx0ZXJDb3VudD86IG51bWJlcjtcbiAgLyoqXG4gICAqIE9wdGlvbmFsIE1VSSBzeCBwcm9wIGZvciBjdXN0b20gY2FyZCBzdHlsaW5nLlxuICAgKiBAZXhhbXBsZSB7IG10OiAyLCBib3hTaGFkb3c6IDIgfVxuICAgKi9cbiAgY2FyZFN4PzogU3hQcm9wcztcbiAgLyoqXG4gICAqIE9wdGlvbmFsIE1VSSBzeCBwcm9wIGZvciBjdXN0b20gdGl0bGUgdGV4dCBzdHlsaW5nLlxuICAgKiBAZXhhbXBsZSB7IGZvbnRTaXplOiAxOCwgZm9udFdlaWdodDogJ2JvbGQnIH1cbiAgICovXG4gIHRleHRTeD86IFN4UHJvcHM7XG4gIC8qKlxuICAgKiBDdXN0b20gaWNvbiB0byBkaXNwbGF5IGluc3RlYWQgb2YgdGhlIGRlZmF1bHQgc2VhcmNoIGljb24uXG4gICAqIEBleGFtcGxlIDxGaWx0ZXJMaXN0SWNvbiAvPlxuICAgKi9cbiAgaWNvbj86IFJlYWN0Tm9kZTtcbiAgLyoqXG4gICAqIE9wdGlvbmFsIE1VSSBzeCBwcm9wIGZvciBjdXN0b20gaWNvbiBzdHlsaW5nLlxuICAgKiBAZXhhbXBsZSB7IGNvbG9yOiAnc2Vjb25kYXJ5Lm1haW4nLCBmb250U2l6ZTogMjQgfVxuICAgKi9cbiAgaWNvblN4PzogU3hQcm9wcztcbiAgLyoqXG4gICAqIFdoZXRoZXIgdG8gZGlzcGxheSB0aGUgZmlsdGVyIGNvdW50IGluIHRoZSBoZWFkZXIuXG4gICAqIEBkZWZhdWx0VmFsdWUgZmFsc2VcbiAgICovXG4gIHNob3dDb3VudD86IGJvb2xlYW47XG59PjtcblxuLyoqXG4gKiBDYXJkLWJhc2VkIHdyYXBwZXIgY29tcG9uZW50IGZvciBvcmdhbml6aW5nIGZpbHRlciBjb250cm9scyBhbmQgZm9ybSBlbGVtZW50cy5cbiAqXG4gKiBUaGlzIGNvbXBvbmVudCBwcm92aWRlcyBhIGNvbnNpc3RlbnQgbGF5b3V0IGZvciBmaWx0ZXIgaW50ZXJmYWNlcyB3aXRoIGEgaGVhZGVyLFxuICogb3B0aW9uYWwgZmlsdGVyIGNvdW50IGRpc3BsYXksIGFuZCBhIGdyaWQtYmFzZWQgY29udGVudCBhcmVhLiBJdCdzIGRlc2lnbmVkIHRvXG4gKiBjb250YWluIGZvcm0gY29udHJvbHMgYW5kIGZpbHRlciBlbGVtZW50cyBpbiBhIHZpc3VhbGx5IG9yZ2FuaXplZCBtYW5uZXIuXG4gKlxuICogQGV4YW1wbGVcbiAqIEJhc2ljIHVzYWdlOlxuICogYGBgdHN4XG4gKiA8RmlsdGVyV3JhcHBlcj5cbiAqICAgPEdyaWQgaXRlbSB4cz17MTJ9IG1kPXs2fT5cbiAqICAgICA8VGV4dEZpZWxkIGxhYmVsPVwiU2VhcmNoXCIgLz5cbiAqICAgPC9HcmlkPlxuICogICA8R3JpZCBpdGVtIHhzPXsxMn0gbWQ9ezZ9PlxuICogICAgIDxTZWxlY3QgbGFiZWw9XCJDYXRlZ29yeVwiIC8+XG4gKiAgIDwvR3JpZD5cbiAqIDwvRmlsdGVyV3JhcHBlcj5cbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBXaXRoIGN1c3RvbSB0aXRsZSBhbmQgY291bnQ6XG4gKiBgYGB0c3hcbiAqIDxGaWx0ZXJXcmFwcGVyXG4gKiAgIHRpdGxlPVwiQWR2YW5jZWQgU2VhcmNoXCJcbiAqICAgZmlsdGVyQ291bnQ9e2FjdGl2ZUZpbHRlcnMubGVuZ3RofVxuICogICBzaG93Q291bnQ9e3RydWV9XG4gKiA+XG4gKiAgIHtmaWx0ZXJDb250cm9sc31cbiAqIDwvRmlsdGVyV3JhcHBlcj5cbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBXaXRoIGN1c3RvbSBzdHlsaW5nOlxuICogYGBgdHN4XG4gKiA8RmlsdGVyV3JhcHBlclxuICogICB0aXRsZT1cIlByb2R1Y3QgRmlsdGVyc1wiXG4gKiAgIGljb249ezxGaWx0ZXJMaXN0SWNvbiAvPn1cbiAqICAgY2FyZFN4PXt7IG10OiAzLCBib3JkZXJSYWRpdXM6IDIgfX1cbiAqICAgdGV4dFN4PXt7IGNvbG9yOiAnc2Vjb25kYXJ5Lm1haW4nIH19XG4gKiA+XG4gKiAgIHtjaGlsZHJlbn1cbiAqIDwvRmlsdGVyV3JhcHBlcj5cbiAqIGBgYFxuICpcbiAqIEBwYXJhbSBwcm9wcyAtIENvbXBvbmVudCBwcm9wcyBpbmNsdWRpbmcgY2hpbGRyZW4gYW5kIHN0eWxpbmcgb3B0aW9uc1xuICogQHJldHVybnMgTVVJIENhcmQgY29tcG9uZW50IHdpdGggc3RydWN0dXJlZCBmaWx0ZXIgbGF5b3V0XG4gKlxuICogQHB1YmxpY1xuICovXG5leHBvcnQgY29uc3QgRmlsdGVyV3JhcHBlciA9ICh7XG4gIGNoaWxkcmVuLFxuICB0aXRsZSxcbiAgZmlsdGVyQ291bnQsXG4gIGNhcmRTeCxcbiAgdGV4dFN4LFxuICBpY29uLFxuICBpY29uU3gsXG4gIHNob3dDb3VudCxcbn06IEZpbHRlcldyYXBwZXJQcm9wcykgPT4ge1xuICBjb25zdCB0aGVtZSA9IHVzZVRoZW1lKCk7XG5cbiAgcmV0dXJuIChcbiAgICA8Q2FyZFxuICAgICAgc3g9e3tcbiAgICAgICAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gICAgICAgIGJvcmRlclJhZGl1czogJzBweCcsXG4gICAgICAgIG1iOiAyLFxuICAgICAgICAuLi5jYXJkU3gsXG4gICAgICB9fVxuICAgID5cbiAgICAgIDxDYXJkSGVhZGVyXG4gICAgICAgIHN4PXt7XG4gICAgICAgICAgZGlzcGxheTogJ2ZsZXgnLFxuICAgICAgICAgIGZsZXhXcmFwOiAnd3JhcCcsXG4gICAgICAgICAgcDogJzFyZW0nLFxuICAgICAgICAgICcuTXVpQ2FyZEhlYWRlci1hY3Rpb24nOiB7XG4gICAgICAgICAgICBtYXJnaW46IDAsXG4gICAgICAgICAgICBhbGlnblNlbGY6ICdjZW50ZXInLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gICAgICAgIH19XG4gICAgICAgIHRpdGxlPXtcbiAgICAgICAgICA8Qm94IHN4PXt7IGRpc3BsYXk6ICdmbGV4JywgYWxpZ25JdGVtczogJ2NlbnRlcicsIGdhcDogMC41IH19PlxuICAgICAgICAgICAge2ljb24gPyAoXG4gICAgICAgICAgICAgIGljb25cbiAgICAgICAgICAgICkgOiAoXG4gICAgICAgICAgICAgIDxNYW5hZ2VTZWFyY2hJY29uXG4gICAgICAgICAgICAgICAgc3g9e3tcbiAgICAgICAgICAgICAgICAgIGhlaWdodDogJzIuNXJlbScsXG4gICAgICAgICAgICAgICAgICBjb2xvcjogdGhlbWUucGFsZXR0ZS5wcmltYXJ5Lm1haW4sXG4gICAgICAgICAgICAgICAgICAuLi5pY29uU3gsXG4gICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICl9XG4gICAgICAgICAgICA8VHlwb2dyYXBoeVxuICAgICAgICAgICAgICB2YXJpYW50PVwiaDVcIlxuICAgICAgICAgICAgICBzeD17e1xuICAgICAgICAgICAgICAgIGZvbnRXZWlnaHQ6ICdib2xkJyxcbiAgICAgICAgICAgICAgICBjb2xvcjogdGhlbWUucGFsZXR0ZS5wcmltYXJ5Lm1haW4sXG4gICAgICAgICAgICAgICAgLi4udGV4dFN4LFxuICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgPlxuICAgICAgICAgICAgICB7dGl0bGUgPyB0aXRsZSA6ICdGaWx0ZXInfXsnICd9XG4gICAgICAgICAgICAgIHtzaG93Q291bnQgPyBgKCR7ZmlsdGVyQ291bnQgPyBmaWx0ZXJDb3VudCA6IDB9KWAgOiA8PjwvPn1cbiAgICAgICAgICAgIDwvVHlwb2dyYXBoeT5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgfVxuICAgICAgPjwvQ2FyZEhlYWRlcj5cbiAgICAgIDxEaXZpZGVyIC8+XG4gICAgICA8Q2FyZENvbnRlbnQgc3g9e3sgcHk6IDIgfX0+XG4gICAgICAgIDxHcmlkIGNvbnRhaW5lciBzcGFjaW5nPXsyfT5cbiAgICAgICAgICB7Y2hpbGRyZW59XG4gICAgICAgIDwvR3JpZD5cbiAgICAgIDwvQ2FyZENvbnRlbnQ+XG4gICAgPC9DYXJkPlxuICApO1xufTtcbiIsICIvLyBjb3JlL2NvbXBvbmVudHMvRm9vdGVyL2luZGV4LnRzeFxuaW1wb3J0IHsgQm94LCBUeXBvZ3JhcGh5IH0gZnJvbSAnQG11aS9tYXRlcmlhbCc7XG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuXG5leHBvcnQgY29uc3QgRm9vdGVyOiBSZWFjdC5GQyA9ICgpID0+IHtcbiAgY29uc3QgY3VycmVudFllYXIgPSBuZXcgRGF0ZSgpLmdldEZ1bGxZZWFyKCk7XG5cbiAgcmV0dXJuIChcbiAgICA8Qm94XG4gICAgICBjb21wb25lbnQ9XCJmb290ZXJcIlxuICAgICAgc3g9e3tcbiAgICAgICAgcHk6IDIsXG4gICAgICAgIHB4OiA0LFxuICAgICAgICBtdDogJ2F1dG8nLFxuICAgICAgICBiYWNrZ3JvdW5kQ29sb3I6IHRoZW1lID0+XG4gICAgICAgICAgdGhlbWUucGFsZXR0ZS5tb2RlID09PSAnbGlnaHQnXG4gICAgICAgICAgICA/IHRoZW1lLnBhbGV0dGUuZ3JleVsyMDBdXG4gICAgICAgICAgICA6IHRoZW1lLnBhbGV0dGUuZ3JleVs4MDBdLFxuICAgICAgfX1cbiAgICA+XG4gICAgICA8VHlwb2dyYXBoeSB2YXJpYW50PVwiYm9keTJcIiBjb2xvcj1cInRleHQuc2Vjb25kYXJ5XCIgYWxpZ249XCJjZW50ZXJcIj5cbiAgICAgICAge2BcdTAwQTkgQ29weXJpZ2h0ICR7Y3VycmVudFllYXJ9IEdOLiBBbGwgcmlnaHRzIHJlc2VydmVkIGJ5IFBhcnVsIFVuaXZlcnNpdHkuYH1cbiAgICAgIDwvVHlwb2dyYXBoeT5cbiAgICA8L0JveD5cbiAgKTtcbn07XG4iLCAiaW1wb3J0IHR5cGUgeyBTeFByb3BzIH0gZnJvbSAnQG11aS9tYXRlcmlhbCc7XG5pbXBvcnQgeyBHcmlkLCBUb29sdGlwLCBUeXBvZ3JhcGh5LCB1c2VUaGVtZSB9IGZyb20gJ0BtdWkvbWF0ZXJpYWwnO1xuXG4vKipcbiAqIFByb3BzIGZvciB0aGUgTGFiZWxUZXh0IGNvbXBvbmVudC5cbiAqXG4gKiBAcHVibGljXG4gKi9cbmludGVyZmFjZSBMYWJlbFRleHRQcm9wcyB7XG4gIC8qKlxuICAgKiBMYWJlbCB0ZXh0IHRvIGRpc3BsYXkgb24gdGhlIGxlZnQgc2lkZS5cbiAgICogQGV4YW1wbGUgXCJOYW1lXCIgb3IgXCJFbWFpbCBBZGRyZXNzXCJcbiAgICovXG4gIGxhYmVsOiBzdHJpbmc7XG4gIC8qKlxuICAgKiBWYWx1ZSBjb250ZW50IHRvIGRpc3BsYXkgb24gdGhlIHJpZ2h0IHNpZGUuXG4gICAqIENhbiBiZSB0ZXh0LCBudW1iZXJzLCBvciBSZWFjdCBlbGVtZW50cy5cbiAgICogQGV4YW1wbGUgXCJKb2huIERvZVwiIG9yIDxMaW5rPlZpZXcgRGV0YWlsczwvTGluaz5cbiAgICovXG4gIHZhbHVlOiBSZWFjdC5SZWFjdE5vZGU7XG4gIC8qKlxuICAgKiBDdXN0b20gZ3JpZCBzaXppbmcgZm9yIGxhYmVsIGFuZCB2YWx1ZSBzZWN0aW9ucy5cbiAgICogQGRlZmF1bHRWYWx1ZSB7IGxhYmVsU2l6ZTogeyB4czogNiwgc206IDYsIG1kOiA2IH0sIHZhbHVlU2l6ZTogeyB4czogMTIsIHNtOiA2LCBtZDogNiB9IH1cbiAgICovXG4gIGdyaWRTaXplPzoge1xuICAgIC8qKiBHcmlkIHNpemUgY29uZmlndXJhdGlvbiBmb3IgdGhlIGxhYmVsIHNlY3Rpb24gKi9cbiAgICBsYWJlbFNpemU6IHsgeHM6IG51bWJlcjsgc206IG51bWJlcjsgbWQ6IG51bWJlciB9O1xuICAgIC8qKiBHcmlkIHNpemUgY29uZmlndXJhdGlvbiBmb3IgdGhlIHZhbHVlIHNlY3Rpb24gKi9cbiAgICB2YWx1ZVNpemU6IHsgeHM6IG51bWJlcjsgc206IG51bWJlcjsgbWQ6IG51bWJlciB9O1xuICB9O1xuICAvKipcbiAgICogR3JpZCBzaXplIGNvbmZpZ3VyYXRpb24gZm9yIHRoZSBlbnRpcmUgY29udGFpbmVyLlxuICAgKiBAZGVmYXVsdFZhbHVlIHsgeHM6IDEyLCBzbTogNiwgbWQ6IDYgfVxuICAgKi9cbiAgY29udGFpbmVyU2l6ZT86IHsgeHM6IG51bWJlcjsgc206IG51bWJlcjsgbWQ6IG51bWJlciB9O1xuICAvKipcbiAgICogT3B0aW9uYWwgTVVJIHN4IHByb3AgZm9yIGN1c3RvbSBsYWJlbCBzdHlsaW5nLlxuICAgKiBAZXhhbXBsZSB7IGZvbnRXZWlnaHQ6ICdib2xkJywgY29sb3I6ICdwcmltYXJ5Lm1haW4nIH1cbiAgICovXG4gIGxhYmVsU3g/OiBTeFByb3BzO1xuICAvKipcbiAgICogT3B0aW9uYWwgTVVJIHN4IHByb3AgZm9yIGN1c3RvbSB2YWx1ZSBzdHlsaW5nLlxuICAgKiBAZXhhbXBsZSB7IGNvbG9yOiAndGV4dC5zZWNvbmRhcnknLCBmb250U3R5bGU6ICdpdGFsaWMnIH1cbiAgICovXG4gIHZhbHVlU3g/OiBTeFByb3BzO1xufVxuXG4vKipcbiAqIFJlc3BvbnNpdmUgbGFiZWwtdmFsdWUgZGlzcGxheSBjb21wb25lbnQgd2l0aCBob3ZlciBlZmZlY3RzIGFuZCB0ZXh0IHRydW5jYXRpb24uXG4gKlxuICogVGhpcyBjb21wb25lbnQgY3JlYXRlcyBhIGNvbnNpc3RlbnQgbGFiZWwtdmFsdWUgcGFpciBsYXlvdXQgdGhhdCBhZGFwdHMgdG8gZGlmZmVyZW50XG4gKiBzY3JlZW4gc2l6ZXMuIEl0IGluY2x1ZGVzIGhvdmVyIGVmZmVjdHMsIHRleHQgdHJ1bmNhdGlvbiB3aXRoIHRvb2x0aXBzLCBhbmRcbiAqIGN1c3RvbWl6YWJsZSBncmlkIHNpemluZyBmb3IgZmxleGlibGUgbGF5b3V0cy5cbiAqXG4gKiBAZXhhbXBsZVxuICogQmFzaWMgdXNhZ2U6XG4gKiBgYGB0c3hcbiAqIDxMYWJlbFRleHRcbiAqICAgbGFiZWw9XCJGdWxsIE5hbWVcIlxuICogICB2YWx1ZT1cIkpvaG4gRG9lXCJcbiAqIC8+XG4gKiBgYGBcbiAqXG4gKiBAZXhhbXBsZVxuICogV2l0aCBSZWFjdCBlbGVtZW50IHZhbHVlOlxuICogYGBgdHN4XG4gKiA8TGFiZWxUZXh0XG4gKiAgIGxhYmVsPVwiUHJvZmlsZVwiXG4gKiAgIHZhbHVlPXs8TGluayBocmVmPVwiL3Byb2ZpbGVcIj5WaWV3IFByb2ZpbGU8L0xpbms+fVxuICogLz5cbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBDdXN0b20gZ3JpZCBzaXppbmc6XG4gKiBgYGB0c3hcbiAqIDxMYWJlbFRleHRcbiAqICAgbGFiZWw9XCJEZXNjcmlwdGlvblwiXG4gKiAgIHZhbHVlPXtsb25nRGVzY3JpcHRpb259XG4gKiAgIGdyaWRTaXplPXt7XG4gKiAgICAgbGFiZWxTaXplOiB7IHhzOiAxMiwgc206IDMsIG1kOiAyIH0sXG4gKiAgICAgdmFsdWVTaXplOiB7IHhzOiAxMiwgc206IDksIG1kOiAxMCB9XG4gKiAgIH19XG4gKiAgIGNvbnRhaW5lclNpemU9e3sgeHM6IDEyLCBzbTogMTIsIG1kOiAxMiB9fVxuICogLz5cbiAqIGBgYFxuICpcbiAqIEBwYXJhbSBwcm9wcyAtIENvbXBvbmVudCBwcm9wcyBmb3IgbGFiZWwtdmFsdWUgY29uZmlndXJhdGlvblxuICogQHJldHVybnMgR3JpZC1iYXNlZCBsYXlvdXQgd2l0aCBsYWJlbCBhbmQgdmFsdWUgc2VjdGlvbnNcbiAqXG4gKiBAcHVibGljXG4gKi9cbmV4cG9ydCBjb25zdCBMYWJlbFRleHQgPSAoe1xuICBsYWJlbCxcbiAgdmFsdWUsXG4gIGdyaWRTaXplLFxuICBjb250YWluZXJTaXplLFxuICBsYWJlbFN4LFxuICB2YWx1ZVN4LFxufTogTGFiZWxUZXh0UHJvcHMpID0+IHtcbiAgY29uc3QgdGhlbWUgPSB1c2VUaGVtZSgpO1xuICBjb25zdCBkZWZhdWx0R3JpZFNpemUgPSB7XG4gICAgbGFiZWxTaXplOiB7IHhzOiA2LCBzbTogNiwgbWQ6IDYgfSxcbiAgICB2YWx1ZVNpemU6IHsgeHM6IDEyLCBzbTogNiwgbWQ6IDYgfSxcbiAgfTtcbiAgY29uc3QgZGVmYXVsdENvbnRhaW5lclNpemUgPSB7IHhzOiAxMiwgc206IDYsIG1kOiA2IH07XG4gIGNvbnN0IHNpemUgPSBncmlkU2l6ZSB8fCBkZWZhdWx0R3JpZFNpemU7XG4gIGNvbnN0IGNvbnRhaW5lciA9IGNvbnRhaW5lclNpemUgfHwgZGVmYXVsdENvbnRhaW5lclNpemU7XG5cbiAgcmV0dXJuIChcbiAgICA8R3JpZFxuICAgICAgc2l6ZT17Y29udGFpbmVyfVxuICAgICAgc3g9e3tcbiAgICAgICAgZGlzcGxheTogJ2ZsZXgnLFxuICAgICAgICBmbGV4RGlyZWN0aW9uOiB7IHhzOiAnY29sdW1uJywgc206ICdyb3cnLCBtZDogJ3JvdycgfSxcbiAgICAgICAgJyY6aG92ZXInOiB7XG4gICAgICAgICAgJyY6aG92ZXInOiB7XG4gICAgICAgICAgICBiYWNrZ3JvdW5kQ29sb3I6IHRoZW1lLnZhcnM/LnBhbGV0dGUuYWN0aW9uLmhvdmVyLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgb3ZlcmZsb3c6ICdoaWRkZW4nLFxuICAgICAgICB9LFxuICAgICAgfX1cbiAgICA+XG4gICAgICA8R3JpZFxuICAgICAgICBzaXplPXtzaXplLmxhYmVsU2l6ZX1cbiAgICAgICAgc3g9e3tcbiAgICAgICAgICBwYWRkaW5nOiAnNXB4JyxcbiAgICAgICAgICBmb250U2l6ZTogJzE0cHgnLFxuICAgICAgICAgIHRleHRBbGlnbjogeyB4czogJ2xlZnQnLCBzbTogJ3JpZ2h0JywgbWQ6ICdyaWdodCcgfSxcbiAgICAgICAgICAuLi5sYWJlbFN4LFxuICAgICAgICB9fVxuICAgICAgPlxuICAgICAgICB7bGFiZWx9IDpcbiAgICAgIDwvR3JpZD5cbiAgICAgIDxHcmlkXG4gICAgICAgIHNpemU9e3NpemUudmFsdWVTaXplfVxuICAgICAgICBzeD17eyBwYWRkaW5nOiAnNXB4JywgZGlzcGxheTogJ2ZsZXgnLCBmbGV4V3JhcDogJ3dyYXAnIH19XG4gICAgICA+XG4gICAgICAgIDxUb29sdGlwIHRpdGxlPXt2YWx1ZX0gYXJyb3c+XG4gICAgICAgICAgPFR5cG9ncmFwaHlcbiAgICAgICAgICAgIHN4PXt7XG4gICAgICAgICAgICAgIGZvbnRTaXplOiAnMTRweCcsXG4gICAgICAgICAgICAgIHdvcmRCcmVhazogJ2JyZWFrLXdvcmQnLFxuICAgICAgICAgICAgICBvdmVyZmxvdzogJ2hpZGRlbicsXG4gICAgICAgICAgICAgIGRpc3BsYXk6ICctd2Via2l0LWJveCcsXG4gICAgICAgICAgICAgIHRleHRPdmVyZmxvdzogJ2VsbGlwc2lzJyxcbiAgICAgICAgICAgICAgV2Via2l0TGluZUNsYW1wOiAyLFxuICAgICAgICAgICAgICBXZWJraXRCb3hPcmllbnQ6ICd2ZXJ0aWNhbCcsXG4gICAgICAgICAgICAgIC4uLnZhbHVlU3gsXG4gICAgICAgICAgICAgIGNvbG9yOiAnIzA3OGRlZScsXG4gICAgICAgICAgICB9fVxuICAgICAgICAgID5cbiAgICAgICAgICAgIHt2YWx1ZSA/IHZhbHVlIDogJy0nfVxuICAgICAgICAgIDwvVHlwb2dyYXBoeT5cbiAgICAgICAgPC9Ub29sdGlwPlxuICAgICAgPC9HcmlkPlxuICAgIDwvR3JpZD5cbiAgKTtcbn07XG4iLCAiaW1wb3J0IHR5cGUgeyBQcm9wc1dpdGhDaGlsZHJlbiB9IGZyb20gJ3JlYWN0JztcblxuaW50ZXJmYWNlIFJlbmRlcklmUHJvcHMge1xuICBzaG93OiBib29sZWFuO1xufVxuXG5leHBvcnQgY29uc3QgUmVuZGVySWYgPSAoe1xuICBzaG93LFxuICBjaGlsZHJlbixcbn06IFByb3BzV2l0aENoaWxkcmVuPFJlbmRlcklmUHJvcHM+KSA9PiB7XG4gIHJldHVybiBzaG93ID8gPD57Y2hpbGRyZW59PC8+IDogbnVsbDtcbn07XG4iLCAiaW1wb3J0IHR5cGUgeyBTeFByb3BzLCBUaGVtZSB9IGZyb20gJ0BtdWkvbWF0ZXJpYWwnO1xuaW1wb3J0IHsgQm94LCBEaXZpZGVyLCBHcmlkLCBTdGFjaywgVHlwb2dyYXBoeSB9IGZyb20gJ0BtdWkvbWF0ZXJpYWwnO1xuaW1wb3J0IHR5cGUgeyBQcm9wc1dpdGhDaGlsZHJlbiwgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IHsgbWVtbywgdXNlTWVtbyB9IGZyb20gJ3JlYWN0JztcblxuLy8gU2VjdGlvbiBib3ggY29uZmlndXJhdGlvblxuZXhwb3J0IGludGVyZmFjZSBTZWN0aW9uQm94UHJvcHMgZXh0ZW5kcyBQcm9wc1dpdGhDaGlsZHJlbiB7XG4gIHRpdGxlOiBzdHJpbmc7XG4gIHNwYWNpbmc/OiBudW1iZXI7XG4gIGNvbnRhaW5lclN4PzogU3hQcm9wczxUaGVtZT47XG4gIHRpdGxlU3g/OiBTeFByb3BzPFRoZW1lPjtcbiAgdmFyaWFudD86ICdkZWZhdWx0JyB8ICdmb3JtJyB8ICdpbmZvJyB8ICd3YXJuaW5nJyB8ICdlcnJvcic7XG4gIGljb24/OiBSZWFjdE5vZGU7XG4gIGFjdGlvbnM/OiBSZWFjdE5vZGU7XG4gIGNvbGxhcHNpYmxlPzogYm9vbGVhbjtcbiAgZGVmYXVsdEV4cGFuZGVkPzogYm9vbGVhbjtcbn1cblxuLy8gVGhlbWUgY29uZmlndXJhdGlvbiBmb3Igc2VjdGlvbiB2YXJpYW50c1xuY29uc3QgZ2V0U2VjdGlvblRoZW1lID0gKHZhcmlhbnQ6IFNlY3Rpb25Cb3hQcm9wc1sndmFyaWFudCddID0gJ2RlZmF1bHQnKSA9PiB7XG4gIGNvbnN0IHRoZW1lcyA9IHtcbiAgICBkZWZhdWx0OiB7XG4gICAgICBiZ2NvbG9yOiAnI2ZhZWJkNycsXG4gICAgICBjb2xvcjogJyM5MjVkMjEnLFxuICAgIH0sXG4gICAgZm9ybToge1xuICAgICAgYmdjb2xvcjogJyNjZGNlZDEnLFxuICAgICAgY29sb3I6ICdibGFjaycsXG4gICAgfSxcbiAgICBpbmZvOiB7XG4gICAgICBiZ2NvbG9yOiAnI2UzZjJmZCcsXG4gICAgICBjb2xvcjogJyMxOTc2ZDInLFxuICAgIH0sXG4gICAgd2FybmluZzoge1xuICAgICAgYmdjb2xvcjogJyNmZmYzZTAnLFxuICAgICAgY29sb3I6ICcjZjU3YzAwJyxcbiAgICB9LFxuICAgIGVycm9yOiB7XG4gICAgICBiZ2NvbG9yOiAnI2ZmZWJlZScsXG4gICAgICBjb2xvcjogJyNkMzJmMmYnLFxuICAgIH0sXG4gIH07XG5cbiAgcmV0dXJuIHRoZW1lc1t2YXJpYW50XTtcbn07XG5cbi8vIE1lbW9pemVkIFNlY3Rpb25Cb3ggY29tcG9uZW50IGZvciBwZXJmb3JtYW5jZVxuZXhwb3J0IGNvbnN0IFNlY3Rpb25Cb3ggPSBtZW1vPFNlY3Rpb25Cb3hQcm9wcz4oXG4gICh7XG4gICAgdGl0bGUsXG4gICAgY2hpbGRyZW4sXG4gICAgc3BhY2luZyA9IDAsXG4gICAgY29udGFpbmVyU3gsXG4gICAgdGl0bGVTeCxcbiAgICB2YXJpYW50ID0gJ2RlZmF1bHQnLFxuICAgIGljb24sXG4gICAgYWN0aW9ucyxcbiAgfSkgPT4ge1xuICAgIGNvbnN0IHRoZW1lQ29sb3JzID0gdXNlTWVtbygoKSA9PiBnZXRTZWN0aW9uVGhlbWUodmFyaWFudCksIFt2YXJpYW50XSk7XG5cbiAgICBjb25zdCBoZWFkZXJTeCA9IHVzZU1lbW8oXG4gICAgICAoKSA9PiAoe1xuICAgICAgICBweDogMS41LFxuICAgICAgICBweTogMC4xLFxuICAgICAgICB3aWR0aDogJ2ZpdC1jb250ZW50JyxcbiAgICAgICAgLi4udGhlbWVDb2xvcnMsXG4gICAgICAgIC4uLnRpdGxlU3gsXG4gICAgICB9KSxcbiAgICAgIFt0aGVtZUNvbG9ycywgdGl0bGVTeF1cbiAgICApO1xuXG4gICAgY29uc3QgY29udGVudFN4ID0gdXNlTWVtbyhcbiAgICAgICgpID0+ICh7XG4gICAgICAgIHBhZGRpbmc6ICcxNnB4JyxcbiAgICAgICAgLi4uY29udGFpbmVyU3gsXG4gICAgICB9KSxcbiAgICAgIFtjb250YWluZXJTeF1cbiAgICApO1xuXG4gICAgcmV0dXJuIChcbiAgICAgIDw+XG4gICAgICAgIDxCb3ggc3g9e3sgZGlzcGxheTogJ2ZsZXgnLCBmbGV4RGlyZWN0aW9uOiAnY29sdW1uJywgd2lkdGg6ICcxMDAlJyB9fT5cbiAgICAgICAgICA8U3RhY2tcbiAgICAgICAgICAgIGRpcmVjdGlvbj1cInJvd1wiXG4gICAgICAgICAgICBqdXN0aWZ5Q29udGVudD1cInNwYWNlLWJldHdlZW5cIlxuICAgICAgICAgICAgYWxpZ25JdGVtcz1cImNlbnRlclwiXG4gICAgICAgICAgICBzeD17aGVhZGVyU3h9XG4gICAgICAgICAgPlxuICAgICAgICAgICAgPFN0YWNrIGRpcmVjdGlvbj1cInJvd1wiIGFsaWduSXRlbXM9XCJjZW50ZXJcIiBzcGFjaW5nPXsxfT5cbiAgICAgICAgICAgICAge2ljb259XG4gICAgICAgICAgICAgIDxUeXBvZ3JhcGh5IHN4PXt7IGZvbnRTaXplOiAnMTVweCcsIGZvbnRXZWlnaHQ6IDQwMCB9fT5cbiAgICAgICAgICAgICAgICB7dGl0bGV9XG4gICAgICAgICAgICAgIDwvVHlwb2dyYXBoeT5cbiAgICAgICAgICAgIDwvU3RhY2s+XG4gICAgICAgICAgICB7YWN0aW9uc31cbiAgICAgICAgICA8L1N0YWNrPlxuICAgICAgICAgIDxEaXZpZGVyIC8+XG4gICAgICAgIDwvQm94PlxuICAgICAgICA8R3JpZCBjb250YWluZXIgc3BhY2luZz17c3BhY2luZ30gc3g9e2NvbnRlbnRTeH0+XG4gICAgICAgICAge2NoaWxkcmVufVxuICAgICAgICA8L0dyaWQ+XG4gICAgICA8Lz5cbiAgICApO1xuICB9XG4pO1xuIiwgImltcG9ydCB7IFRhYkNvbnRleHQgfSBmcm9tICdAbXVpL2xhYic7XG5pbXBvcnQgdHlwZSB7IFN4UHJvcHMgfSBmcm9tICdAbXVpL21hdGVyaWFsJztcbmltcG9ydCB7IEJveCwgVGFiLCBUYWJzIH0gZnJvbSAnQG11aS9tYXRlcmlhbCc7XG5pbXBvcnQgdHlwZSB7IFByb3BzV2l0aENoaWxkcmVuIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IFJlYWN0LCB7IHVzZVN0YXRlIH0gZnJvbSAncmVhY3QnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFRhYkl0ZW0ge1xuICBsYWJlbDogc3RyaW5nO1xuICB2YWx1ZTogbnVtYmVyO1xuICBwZXJtaXNzaW9uPzogYm9vbGVhbjtcbn1cblxuaW50ZXJmYWNlIEN1c3RvbVRhYnNQcm9wcyB7XG4gIHRhYnM6IFRhYkl0ZW1bXTtcbiAgZGVmYXVsdFZhbHVlPzogbnVtYmVyO1xuICBvblRhYkNoYW5nZT86IChuZXdWYWx1ZTogbnVtYmVyKSA9PiB2b2lkO1xuICB0YWJTeD86IFN4UHJvcHM7XG4gIHRhYnNTeD86IFN4UHJvcHM7XG59XG5cbmV4cG9ydCBjb25zdCBTaW1wbGVUYWJzID0gKHtcbiAgdGFicyxcbiAgZGVmYXVsdFZhbHVlID0gMSxcbiAgb25UYWJDaGFuZ2UsXG4gIGNoaWxkcmVuLFxuICB0YWJTeCxcbiAgdGFic1N4LFxufTogQ3VzdG9tVGFic1Byb3BzICYgUHJvcHNXaXRoQ2hpbGRyZW4pID0+IHtcbiAgY29uc3QgW3ZhbHVlLCBzZXRWYWx1ZV0gPSB1c2VTdGF0ZTxudW1iZXI+KGRlZmF1bHRWYWx1ZSk7XG5cbiAgY29uc3QgaGFuZGxlQ2hhbmdlID0gKGV2ZW50OiBSZWFjdC5TeW50aGV0aWNFdmVudCwgbmV3VmFsdWU6IG51bWJlcikgPT4ge1xuICAgIHNldFZhbHVlKG5ld1ZhbHVlKTtcbiAgICBpZiAob25UYWJDaGFuZ2UpIG9uVGFiQ2hhbmdlKG5ld1ZhbHVlKTtcbiAgfTtcblxuICByZXR1cm4gKFxuICAgIDxUYWJDb250ZXh0IHZhbHVlPXt2YWx1ZX0+XG4gICAgICA8Qm94IHN4PXt7IGJvcmRlckJvdHRvbTogMSwgYm9yZGVyQ29sb3I6ICdkaXZpZGVyJywgd2lkdGg6ICcxMDAlJyB9fT5cbiAgICAgICAgPFRhYnNcbiAgICAgICAgICB2YWx1ZT17dmFsdWV9XG4gICAgICAgICAgb25DaGFuZ2U9e2hhbmRsZUNoYW5nZX1cbiAgICAgICAgICBzeD17eyBweDogMiwgcHk6IDAsIC4uLnRhYnNTeCB9fVxuICAgICAgICA+XG4gICAgICAgICAge3RhYnMubWFwKHRhYiA9PiAoXG4gICAgICAgICAgICA8VGFiXG4gICAgICAgICAgICAgIGtleT17dGFiLnZhbHVlfVxuICAgICAgICAgICAgICBsYWJlbD17dGFiLmxhYmVsfVxuICAgICAgICAgICAgICB2YWx1ZT17dGFiLnZhbHVlfVxuICAgICAgICAgICAgICBkaXNhYmxlZD17dGFiLnBlcm1pc3Npb24gPT09IGZhbHNlfVxuICAgICAgICAgICAgICBzeD17eyBmb250U2l6ZTogJzFyZW0nLCAuLi50YWJTeCB9fVxuICAgICAgICAgICAgLz5cbiAgICAgICAgICApKX1cbiAgICAgICAgPC9UYWJzPlxuICAgICAgPC9Cb3g+XG5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L1RhYkNvbnRleHQ+XG4gICk7XG59O1xuIiwgImltcG9ydCB0eXBlIHsgTG9hZGluZ0J1dHRvblByb3BzIH0gZnJvbSAnQG11aS9sYWInO1xuaW1wb3J0IHsgTG9hZGluZ0J1dHRvbiB9IGZyb20gJ0BtdWkvbGFiJztcbmltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbnR5cGUgU3VibWl0QnV0dG9uUHJvcHMgPSBPbWl0PFxuICBMb2FkaW5nQnV0dG9uUHJvcHMsXG4gICdjaGlsZHJlbicgfCAndmFyaWFudCcgfCAnY29sb3InIHwgJ3R5cGUnXG4+O1xuXG5leHBvcnQgY29uc3QgU3VibWl0QnV0dG9uOiBSZWFjdC5GQzxTdWJtaXRCdXR0b25Qcm9wcz4gPSAoe1xuICBsb2FkaW5nID0gZmFsc2UsXG4gIC4uLnJlc3Rcbn0pID0+IChcbiAgPExvYWRpbmdCdXR0b25cbiAgICBsb2FkaW5nPXtsb2FkaW5nfVxuICAgIHZhcmlhbnQ9XCJjb250YWluZWRcIlxuICAgIGNvbG9yPVwicHJpbWFyeVwiXG4gICAgdHlwZT1cInN1Ym1pdFwiXG4gICAgey4uLnJlc3R9XG4gICAgc3g9e3sgZm9udFdlaWdodDogNDAwIH19XG4gID5cbiAgICBTdWJtaXRcbiAgPC9Mb2FkaW5nQnV0dG9uPlxuKTtcbiIsICJpbXBvcnQgdHlwZSBSZWFjdCBmcm9tICdyZWFjdCc7XG5pbXBvcnQgeyBmb3J3YXJkUmVmIH0gZnJvbSAncmVhY3QnO1xuXG4vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4vLyBTaGFyZWQgdHlwZXNcbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbmV4cG9ydCBpbnRlcmZhY2UgRGF0YU1vZGFsQnV0dG9ucyB7XG4gIG9uU3VibWl0PzogKCkgPT4gdm9pZDtcbiAgb25DYW5jZWw/OiAoKSA9PiB2b2lkO1xuICBpc1BlbmRpbmc/OiBib29sZWFuO1xuICBpc1N1Y2Nlc3M/OiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERhdGFNb2RhbENvbXBvbmVudFByb3BzPFREYXRhPiB7XG4gIGRhdGE/OiBURGF0YTtcbiAgaXNFZGl0aW5nPzogYm9vbGVhbjtcbiAgcmVmPzogUmVhY3QuUmVmPERhdGFNb2RhbEJ1dHRvbnM+OyAvLyBcdUQ4M0RcdURDNDggcHV0IHJlZiBpbnRvIHByb3BzXG59XG5cbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbi8vIFByb2plY3Qtc3BlY2lmaWMgSE9DXG4vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5leHBvcnQgZnVuY3Rpb24gd2l0aERhdGFNb2RhbDxURGF0YT4oXG4gIGNvbXBvbmVudDogKFxuICAgIHByb3BzOiBEYXRhTW9kYWxDb21wb25lbnRQcm9wczxURGF0YT5cbiAgKSA9PiBSZWFjdC5SZWFjdEVsZW1lbnQgfCBudWxsXG4pIHtcbiAgcmV0dXJuIGZvcndhcmRSZWY8RGF0YU1vZGFsQnV0dG9ucywgRGF0YU1vZGFsQ29tcG9uZW50UHJvcHM8VERhdGE+PihcbiAgICAocHJvcHMsIHJlZikgPT4gY29tcG9uZW50KHsgLi4ucHJvcHMsIHJlZiB9KVxuICApO1xufVxuIiwgImV4cG9ydCBpbnRlcmZhY2UgQ29uZmlnVmFsdWUge1xuICBhcGlCYXNlVXJsOiBzdHJpbmc7XG4gIGRlZmF1bHRQYWdlU2l6ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgY29uc3QgQ29uZmlnOiBDb25maWdWYWx1ZSA9IHtcbiAgZGVmYXVsdFBhZ2VTaXplOiAyMCxcbiAgYXBpQmFzZVVybDogJ2h0dHA6Ly9sb2NhbGhvc3Q6NTE0MycsXG4gIC8vIGFwaUJhc2VVcmw6ICdodHRwOi8vMTkyLjE2OC4xLjI0Njo1MTQzJyxcbn07XG5cbmV4cG9ydCBjb25zdCBkYXRlVGltZVBhdHRlcm5zID0ge1xuICBkYXRlVGltZTogJ0REIE1NTSBZWVlZIGg6bW0gQScsIC8vIDE3IEFwciAyMDIyIDEyOjAwIGFtXG4gIGRhdGU6ICdERCBNTU0gWVlZWScsIC8vIDE3IEFwciAyMDIyXG4gIG1vbnRoX3llYXJfc2hvcnRfZm9ybWF0OiAnTU1NIFlZWVknLFxuICBtb250aF95ZWFyX2Z1bGxfZm9ybWF0OiAnTU1NTSBZWVlZJyxcbiAgeWVhcjogJ1lZWVknLFxuICB0aW1lOiAnaDptbSBhJywgLy8gMTI6MDAgYW1cbiAgc3BsaXQ6IHtcbiAgICBkYXRlVGltZTogJ0REL01NL1lZWVkgaDptbSBBJywgLy8gMTcvMDQvMjAyMiAxMjowMCBhbVxuICAgIGRhdGU6ICdERC9NTS9ZWVlZJywgLy8gMTcvMDQvMjAyMlxuICB9LFxuICBwYXJhbUNhc2U6IHtcbiAgICBkYXRlVGltZTogJ0RELU1NLVlZWVkgaDptbSBBJywgLy8gMTctMDQtMjAyMiAxMjowMCBhbVxuICAgIGRhdGU6ICdERC1NTS1ZWVlZJywgLy8gMTctMDQtMjAyMlxuICAgIGRhdGVSZXZlcnNlOiAnWVlZWS1NTS1ERCcsIC8vIDIwMjItMDQtMTcgZm9yIGNvbXBhcmUgZGF0ZVxuICAgIE1vbnRoWWVhcjogJ01NTS1ZWVlZJyxcbiAgfSxcbn07XG4iLCAiaW1wb3J0IHsgdXNlTWVtbyB9IGZyb20gJ3JlYWN0JztcblxuaW1wb3J0IHsgY3JlYXRlQXBpQ2xpZW50IH0gZnJvbSAnLi4vYXBpL2NyZWF0ZUFwaUNsaWVudCc7XG5pbXBvcnQgdHlwZSB7IEFwaUNsaWVudENvbmZpZyB9IGZyb20gJy4uL2FwaS9jcmVhdGVBcGlDbGllbnQnO1xuXG5leHBvcnQgZnVuY3Rpb24gdXNlQXBpQ2xpZW50KGNvbmZpZzogQXBpQ2xpZW50Q29uZmlnKSB7XG4gIHJldHVybiB1c2VNZW1vKFxuICAgICgpID0+IGNyZWF0ZUFwaUNsaWVudChjb25maWcpLFxuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSByZWFjdC1ob29rcy9leGhhdXN0aXZlLWRlcHNcbiAgICBbXG4gICAgICBjb25maWcuYmFzZVVSTCxcbiAgICAgIGNvbmZpZy50aW1lb3V0LFxuICAgICAgY29uZmlnLmNvcnJlbGF0aW9uSWRQcmVmaXgsXG4gICAgICBjb25maWcuaW5jbHVkZUNvcnJlbGF0aW9uSWQsXG4gICAgICBjb25maWcudG9rZW5TdG9yYWdlS2V5LFxuICAgICAgY29uZmlnLmF1dGhUb2tlbixcbiAgICAgIGNvbmZpZy5yZXF1ZXN0SW50ZXJjZXB0b3JzLFxuICAgICAgY29uZmlnLnJlc3BvbnNlSW50ZXJjZXB0b3JzLFxuICAgICAgY29uZmlnLmVycm9ySW50ZXJjZXB0b3JzLFxuICAgIF1cbiAgKTtcbn1cbiIsICJpbXBvcnQgdHlwZSB7IFZhbGlkYXRpb25FcnJvcnMgfSBmcm9tICdAZ253ZWJzb2Z0L3VpJztcbmltcG9ydCB7IHVzZUNhbGxiYWNrIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IHR5cGUgeyBVc2VGb3JtU2V0RXJyb3IsIEZpZWxkVmFsdWVzLCBQYXRoIH0gZnJvbSAncmVhY3QtaG9vay1mb3JtJztcbmltcG9ydCB7IHRvYXN0IH0gZnJvbSAnc29ubmVyJztcblxuaW1wb3J0IHR5cGUgeyBBcGlFcnJvciB9IGZyb20gJy4uL2FwaS90eXBlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3VjY2Vzc01lc3NhZ2Uge1xuICBjcmVhdGU6IHN0cmluZztcbiAgdXBkYXRlOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRXJyb3JNZXNzYWdlIHtcbiAgbm9DaGFuZ2VzOiBzdHJpbmc7XG4gIGdlbmVyYWw6IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBVc2VGb3JtRXJyb3JIYW5kbGVyT3B0aW9uczxURmllbGRWYWx1ZXMgZXh0ZW5kcyBGaWVsZFZhbHVlcz4ge1xuICBzZXRFcnJvcj86IFVzZUZvcm1TZXRFcnJvcjxURmllbGRWYWx1ZXM+O1xuICBzdWNjZXNzTWVzc2FnZT86IFN1Y2Nlc3NNZXNzYWdlO1xuICBlcnJvck1lc3NhZ2U/OiBFcnJvck1lc3NhZ2U7XG59XG5cbmV4cG9ydCB0eXBlIFN1Y2Nlc3NIYW5kbGVyID0gKFxuICBpc0VkaXRpbmc6IGJvb2xlYW4sXG4gIHJvd3NBZmZlY3RlZD86IG51bWJlclxuKSA9PiBib29sZWFuO1xuXG5leHBvcnQgdHlwZSBFcnJvckhhbmRsZXIgPSAocHJvY2Vzc2VkRXJyb3I6IEFwaUVycm9yKSA9PiB2b2lkO1xuXG5leHBvcnQgaW50ZXJmYWNlIFVzZUZvcm1FcnJvckhhbmRsZXJSZXR1cm4ge1xuICBoYW5kbGVTdWNjZXNzOiBTdWNjZXNzSGFuZGxlcjtcbiAgaGFuZGxlRXJyb3I6IEVycm9ySGFuZGxlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBVc2VEZWxldGVIYW5kbGVyT3B0aW9ucyB7XG4gIHN1Y2Nlc3NNZXNzYWdlPzogc3RyaW5nO1xuICBlcnJvck1lc3NhZ2U/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogSG9vayB0byBoYW5kbGUgQVBJIGVycm9ycyBpbiBmb3JtcyB3aXRoIHN0YW5kYXJkaXplZCBlcnJvciBoYW5kbGluZyBhbmQgdG9hc3QgbWVzc2FnZXNcbiAqL1xuZXhwb3J0IGNvbnN0IHVzZUZvcm1FcnJvckhhbmRsZXIgPSA8VEZpZWxkVmFsdWVzIGV4dGVuZHMgRmllbGRWYWx1ZXM+KHtcbiAgc2V0RXJyb3IsXG4gIHN1Y2Nlc3NNZXNzYWdlID0ge1xuICAgIGNyZWF0ZTogJ0NyZWF0ZWQgc3VjY2Vzc2Z1bGx5JyxcbiAgICB1cGRhdGU6ICdVcGRhdGVkIHN1Y2Nlc3NmdWxseScsXG4gIH0sXG4gIGVycm9yTWVzc2FnZSA9IHtcbiAgICBub0NoYW5nZXM6ICdObyBjaGFuZ2VzIHdlcmUgbWFkZScsXG4gICAgZ2VuZXJhbDogJ0ZhaWxlZCB0byBzYXZlLiBQbGVhc2UgdHJ5IGFnYWluLicsXG4gIH0sXG59OiBVc2VGb3JtRXJyb3JIYW5kbGVyT3B0aW9uczxURmllbGRWYWx1ZXM+KTogVXNlRm9ybUVycm9ySGFuZGxlclJldHVybiA9PiB7XG4gIGNvbnN0IGdldEZpZWxkRXJyb3IgPSB1c2VDYWxsYmFjayhcbiAgICAoXG4gICAgICBmaWVsZHM6IFZhbGlkYXRpb25FcnJvcnMgfCB1bmRlZmluZWQsXG4gICAgICBmaWVsZE5hbWU6IHN0cmluZ1xuICAgICk6IHN0cmluZyB8IHVuZGVmaW5lZCA9PiB7XG4gICAgICBpZiAoIWZpZWxkcyB8fCAhZmllbGRzW2ZpZWxkTmFtZV0pIHJldHVybiB1bmRlZmluZWQ7XG5cbiAgICAgIGNvbnN0IGZpZWxkRXJyb3IgPSBmaWVsZHNbZmllbGROYW1lXTtcblxuICAgICAgaWYgKHR5cGVvZiBmaWVsZEVycm9yID09PSAnc3RyaW5nJykge1xuICAgICAgICByZXR1cm4gZmllbGRFcnJvcjtcbiAgICAgIH1cblxuICAgICAgaWYgKEFycmF5LmlzQXJyYXkoZmllbGRFcnJvcikpIHtcbiAgICAgICAgcmV0dXJuIGZpZWxkRXJyb3Iuam9pbignLCAnKTtcbiAgICAgIH1cblxuICAgICAgaWYgKHR5cGVvZiBmaWVsZEVycm9yID09PSAnb2JqZWN0JyAmJiAnbWVzc2FnZScgaW4gZmllbGRFcnJvcikge1xuICAgICAgICByZXR1cm4gZmllbGRFcnJvci5tZXNzYWdlO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gdW5kZWZpbmVkO1xuICAgIH0sXG4gICAgW11cbiAgKTtcblxuICBjb25zdCBoYW5kbGVTdWNjZXNzID0gdXNlQ2FsbGJhY2soXG4gICAgKGlzRWRpdGluZzogYm9vbGVhbiwgcm93c0FmZmVjdGVkPzogbnVtYmVyKSA9PiB7XG4gICAgICBpZiAocm93c0FmZmVjdGVkICE9PSB1bmRlZmluZWQgJiYgcm93c0FmZmVjdGVkID4gMCkge1xuICAgICAgICB0b2FzdC5zdWNjZXNzKFxuICAgICAgICAgIGlzRWRpdGluZyA/IHN1Y2Nlc3NNZXNzYWdlLnVwZGF0ZSA6IHN1Y2Nlc3NNZXNzYWdlLmNyZWF0ZVxuICAgICAgICApO1xuXG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgfSBlbHNlIGlmIChyb3dzQWZmZWN0ZWQgPT09IDApIHtcbiAgICAgICAgdG9hc3QuZXJyb3IoZXJyb3JNZXNzYWdlLm5vQ2hhbmdlcyk7XG5cbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgfVxuXG4gICAgICAvLyBJZiByb3dzQWZmZWN0ZWQgaXMgdW5kZWZpbmVkLCBhc3N1bWUgc3VjY2Vzc1xuICAgICAgdG9hc3Quc3VjY2Vzcyhpc0VkaXRpbmcgPyBzdWNjZXNzTWVzc2FnZS51cGRhdGUgOiBzdWNjZXNzTWVzc2FnZS5jcmVhdGUpO1xuXG4gICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9LFxuICAgIFtzdWNjZXNzTWVzc2FnZSwgZXJyb3JNZXNzYWdlXVxuICApO1xuXG4gIGNvbnN0IGhhbmRsZUVycm9yID0gdXNlQ2FsbGJhY2soXG4gICAgKHByb2Nlc3NlZEVycm9yOiBBcGlFcnJvcikgPT4ge1xuICAgICAgaWYgKFxuICAgICAgICBwcm9jZXNzZWRFcnJvci50eXBlID09PSAndmFsaWRhdGlvbl9lcnJvcicgJiZcbiAgICAgICAgcHJvY2Vzc2VkRXJyb3IuZXJyb3JzICYmXG4gICAgICAgIHNldEVycm9yXG4gICAgICApIHtcbiAgICAgICAgLy8gU2V0IGZpZWxkLXNwZWNpZmljIGVycm9ycyB1c2luZyByZWFjdC1ob29rLWZvcm0ncyBzZXRFcnJvciAob25seSBpZiBzZXRFcnJvciBpcyBwcm92aWRlZClcbiAgICAgICAgT2JqZWN0LmtleXMocHJvY2Vzc2VkRXJyb3IuZXJyb3JzKS5mb3JFYWNoKGZpZWxkTmFtZSA9PiB7XG4gICAgICAgICAgY29uc3QgZmllbGRFcnJvciA9IGdldEZpZWxkRXJyb3IocHJvY2Vzc2VkRXJyb3IuZXJyb3JzLCBmaWVsZE5hbWUpO1xuXG4gICAgICAgICAgaWYgKGZpZWxkRXJyb3IpIHtcbiAgICAgICAgICAgIHNldEVycm9yKGZpZWxkTmFtZSBhcyBQYXRoPFRGaWVsZFZhbHVlcz4sIHtcbiAgICAgICAgICAgICAgdHlwZTogJ3NlcnZlcicsXG4gICAgICAgICAgICAgIG1lc3NhZ2U6IGZpZWxkRXJyb3IsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgIH0pO1xuXG4gICAgICAgIC8vIFNob3cgZ2VuZXJhbCB2YWxpZGF0aW9uIGVycm9yIHRvYXN0XG4gICAgICAgIHRvYXN0LmVycm9yKFxuICAgICAgICAgIHByb2Nlc3NlZEVycm9yLnRpdGxlIHx8ICdQbGVhc2UgY2hlY2sgdGhlIGZvcm0gZm9yIHZhbGlkYXRpb24gZXJyb3JzJ1xuICAgICAgICApO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8gU2hvdyBnZW5lcmFsIGVycm9yIHRvYXN0IGZvciBub24tdmFsaWRhdGlvbiBlcnJvcnMgb3Igd2hlbiBzZXRFcnJvciBpcyBub3QgYXZhaWxhYmxlXG4gICAgICAgIHRvYXN0LmVycm9yKHByb2Nlc3NlZEVycm9yLnRpdGxlIHx8IGVycm9yTWVzc2FnZS5nZW5lcmFsKTtcbiAgICAgIH1cbiAgICB9LFxuICAgIFtlcnJvck1lc3NhZ2UuZ2VuZXJhbCwgZ2V0RmllbGRFcnJvciwgc2V0RXJyb3JdXG4gICk7XG5cbiAgcmV0dXJuIHtcbiAgICBoYW5kbGVTdWNjZXNzLFxuICAgIGhhbmRsZUVycm9yLFxuICB9O1xufTtcblxuLyoqXG4gKiBDb252ZW5pZW5jZSBob29rIGZvciBkZWxldGUgb3BlcmF0aW9ucyB0aGF0IGRvbid0IG5lZWQgZm9ybSBmaWVsZCB2YWxpZGF0aW9uXG4gKiBVc2VzIHVzZUZvcm1FcnJvckhhbmRsZXIgaW50ZXJuYWxseSBidXQgd2l0aCBzaW1wbGlmaWVkIG9wdGlvbnNcbiAqL1xuZXhwb3J0IGNvbnN0IHVzZURlbGV0ZUhhbmRsZXIgPSAoe1xuICBzdWNjZXNzTWVzc2FnZSA9ICdEZWxldGVkIHN1Y2Nlc3NmdWxseScsXG4gIGVycm9yTWVzc2FnZSA9ICdGYWlsZWQgdG8gZGVsZXRlLiBQbGVhc2UgdHJ5IGFnYWluLicsXG59OiBVc2VEZWxldGVIYW5kbGVyT3B0aW9ucyA9IHt9KTogVXNlRm9ybUVycm9ySGFuZGxlclJldHVybiA9PiB7XG4gIHJldHVybiB1c2VGb3JtRXJyb3JIYW5kbGVyKHtcbiAgICBzdWNjZXNzTWVzc2FnZToge1xuICAgICAgY3JlYXRlOiBzdWNjZXNzTWVzc2FnZSwgLy8gTm90IHVzZWQgZm9yIGRlbGV0ZSwgYnV0IHJlcXVpcmVkIGZvciB0eXBlXG4gICAgICB1cGRhdGU6IHN1Y2Nlc3NNZXNzYWdlLFxuICAgIH0sXG4gICAgZXJyb3JNZXNzYWdlOiB7XG4gICAgICBub0NoYW5nZXM6ICdObyBjaGFuZ2VzIHdlcmUgbWFkZScsIC8vIE5vdCB0eXBpY2FsbHkgdXNlZCBmb3IgZGVsZXRlXG4gICAgICBnZW5lcmFsOiBlcnJvck1lc3NhZ2UsXG4gICAgfSxcbiAgICAvLyBzZXRFcnJvciBpcyBvbWl0dGVkICh1bmRlZmluZWQpIGZvciBkZWxldGUgb3BlcmF0aW9uc1xuICB9KTtcbn07XG4iLCAiaW1wb3J0IHR5cGUgeyBRdWVyeUNsaWVudCwgUXVlcnlLZXkgfSBmcm9tICdAdGFuc3RhY2svcmVhY3QtcXVlcnknO1xuaW1wb3J0IHsgdXNlUXVlcnlDbGllbnQgfSBmcm9tICdAdGFuc3RhY2svcmVhY3QtcXVlcnknO1xuaW1wb3J0IHsgdXNlTWVtbyB9IGZyb20gJ3JlYWN0JztcblxuLy8gTWluaW1hbCB0eXBlLXNhZmUgY2FjaGUgdXRpbGl0eSBmb3IgcXVlcnkga2V5IGZhY3Rvcmllc1xuZXhwb3J0IGNsYXNzIENhY2hlVXRpbGl0eSB7XG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgcXVlcnlDbGllbnQ6IFF1ZXJ5Q2xpZW50KSB7fVxuXG4gIC8qKlxuICAgKiBHZXQgY2FjaGVkIGRhdGEgdXNpbmcgb25seSB0aGUgcXVlcnlLZXkgZnJvbSBxdWVyeSBmYWN0b3J5XG4gICAqL1xuICBnZXRDYWNoZWREYXRhPFQ+KHF1ZXJ5S2V5OiBRdWVyeUtleSk6IFQgfCB1bmRlZmluZWQge1xuICAgIHJldHVybiB0aGlzLnF1ZXJ5Q2xpZW50LmdldFF1ZXJ5RGF0YTxUPihxdWVyeUtleSk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IGNhY2hlZCBkYXRhIHdpdGggdHJhbnNmb3JtYXRpb24gdXNpbmcgc2VsZWN0IGZ1bmN0aW9uXG4gICAqL1xuICBnZXRDYWNoZWREYXRhV2l0aFNlbGVjdDxULCBSPihcbiAgICBxdWVyeUtleTogUXVlcnlLZXksXG4gICAgc2VsZWN0OiAoZGF0YTogVCkgPT4gUlxuICApOiBSIHwgdW5kZWZpbmVkIHtcbiAgICBjb25zdCBjYWNoZWREYXRhID0gdGhpcy5xdWVyeUNsaWVudC5nZXRRdWVyeURhdGE8VD4ocXVlcnlLZXkpO1xuXG4gICAgaWYgKGNhY2hlZERhdGEgPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgICB9XG5cbiAgICByZXR1cm4gc2VsZWN0KGNhY2hlZERhdGEpO1xuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiB1c2VDYWNoZVV0aWxpdHkoKTogQ2FjaGVVdGlsaXR5IHtcbiAgY29uc3QgcXVlcnlDbGllbnQgPSB1c2VRdWVyeUNsaWVudCgpO1xuXG4gIHJldHVybiB1c2VNZW1vKCgpID0+IG5ldyBDYWNoZVV0aWxpdHkocXVlcnlDbGllbnQpLCBbcXVlcnlDbGllbnRdKTtcbn1cbiIsICJpbXBvcnQgdHlwZSB7XG4gIENvbnRyb2wsXG4gIERlZXBQYXJ0aWFsU2tpcEFycmF5S2V5LFxuICBGaWVsZFZhbHVlcyxcbiAgUGF0aCxcbiAgUGF0aFZhbHVlLFxufSBmcm9tICdyZWFjdC1ob29rLWZvcm0nO1xuaW1wb3J0IHsgdXNlV2F0Y2ggfSBmcm9tICdyZWFjdC1ob29rLWZvcm0nO1xuXG4vKipcbiAqIENvcmUgd2F0Y2ggZnVuY3Rpb25zIGZvciBSZWFjdCBIb29rIEZvcm1cbiAqIFRoZXNlIGFyZSB0aGUgcHJpbWFyeSBidWlsZGluZyBibG9ja3MgZm9yIGZvcm0gd2F0Y2hpbmdcbiAqL1xuXG4vKipcbiAqIFV0aWxpdHkgdHlwZSB0byBlbnN1cmUgYXJyYXkgZWxlbWVudHMgYXJlIGFsbCBQYXRoPFQ+XG4gKi9cbmV4cG9ydCB0eXBlIFBhdGhBcnJheTxUIGV4dGVuZHMgRmllbGRWYWx1ZXM+ID0gUmVhZG9ubHlBcnJheTxQYXRoPFQ+PjtcblxuLyoqXG4gKiBIb29rIHRvIHdhdGNoIGVudGlyZSBmb3JtIC0gcmV0dXJucyBhbGwgZm9ybSB2YWx1ZXNcbiAqL1xuZXhwb3J0IGNvbnN0IHVzZVdhdGNoRm9ybSA9IDxURmllbGRWYWx1ZXMgZXh0ZW5kcyBGaWVsZFZhbHVlcz4oXG4gIGNvbnRyb2w6IENvbnRyb2w8VEZpZWxkVmFsdWVzPlxuKTogRGVlcFBhcnRpYWxTa2lwQXJyYXlLZXk8VEZpZWxkVmFsdWVzPiA9PiB1c2VXYXRjaCh7IGNvbnRyb2wgfSk7XG5cbi8qKlxuICogSG9vayB0byB3YXRjaCBzaW5nbGUgZmllbGQgYnkgcGF0aCAtIHN1cHBvcnRzIGFueSBuZXN0ZWQgcGF0aFxuICovXG5leHBvcnQgY29uc3QgdXNlV2F0Y2hGaWVsZCA9IDxcbiAgVEZpZWxkVmFsdWVzIGV4dGVuZHMgRmllbGRWYWx1ZXMsXG4gIFROYW1lIGV4dGVuZHMgUGF0aDxURmllbGRWYWx1ZXM+LFxuPihcbiAgY29udHJvbDogQ29udHJvbDxURmllbGRWYWx1ZXM+LFxuICBuYW1lOiBUTmFtZVxuKTogUGF0aFZhbHVlPFRGaWVsZFZhbHVlcywgVE5hbWU+ID0+IHVzZVdhdGNoKHsgY29udHJvbCwgbmFtZSB9KTtcblxuLyoqXG4gKiBIb29rIHRvIHdhdGNoIG11bHRpcGxlIGZpZWxkcyBieSBwYXRocyAtIHJldHVybnMgYXJyYXkgb2YgdmFsdWVzXG4gKi9cbmV4cG9ydCBjb25zdCB1c2VXYXRjaEZpZWxkcyA9IDxcbiAgVEZpZWxkVmFsdWVzIGV4dGVuZHMgRmllbGRWYWx1ZXMsXG4gIFROYW1lcyBleHRlbmRzIFJlYWRvbmx5QXJyYXk8UGF0aDxURmllbGRWYWx1ZXM+Pixcbj4oXG4gIGNvbnRyb2w6IENvbnRyb2w8VEZpZWxkVmFsdWVzPixcbiAgbmFtZXM6IFROYW1lc1xuKTogQXJyYXk8UGF0aFZhbHVlPFRGaWVsZFZhbHVlcywgVE5hbWVzW251bWJlcl0+PiA9PlxuICB1c2VXYXRjaCh7IGNvbnRyb2wsIG5hbWU6IG5hbWVzIH0pIGFzIEFycmF5PFxuICAgIFBhdGhWYWx1ZTxURmllbGRWYWx1ZXMsIFROYW1lc1tudW1iZXJdPlxuICA+O1xuIiwgImltcG9ydCB7IHVzZUVmZmVjdCwgdXNlTWVtbywgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCc7XG5pbXBvcnQgdHlwZSB7IENvbnRyb2wsIEZpZWxkVmFsdWVzLCBQYXRoLCBQYXRoVmFsdWUgfSBmcm9tICdyZWFjdC1ob29rLWZvcm0nO1xuaW1wb3J0IHsgdXNlV2F0Y2ggfSBmcm9tICdyZWFjdC1ob29rLWZvcm0nO1xuXG4vKipcbiAqIFV0aWxpdHkgd2F0Y2ggZnVuY3Rpb25zIGZvciBSZWFjdCBIb29rIEZvcm1cbiAqIEVuaGFuY2VkIGZ1bmN0aW9uYWxpdHkgZm9yIHNwZWNpZmljIHVzZSBjYXNlc1xuICovXG5cbi8qKlxuICogV2F0Y2ggZmllbGQgd2l0aCB0cmFuc2Zvcm1hdGlvbi9zZWxlY3RvclxuICovXG5leHBvcnQgY29uc3QgdXNlV2F0Y2hUcmFuc2Zvcm0gPSA8XG4gIFRGaWVsZFZhbHVlcyBleHRlbmRzIEZpZWxkVmFsdWVzLFxuICBUTmFtZSBleHRlbmRzIFBhdGg8VEZpZWxkVmFsdWVzPixcbiAgVE91dHB1dCxcbj4oXG4gIGNvbnRyb2w6IENvbnRyb2w8VEZpZWxkVmFsdWVzPixcbiAgbmFtZTogVE5hbWUsXG4gIHRyYW5zZm9ybTogKHZhbHVlOiBQYXRoVmFsdWU8VEZpZWxkVmFsdWVzLCBUTmFtZT4pID0+IFRPdXRwdXRcbik6IFRPdXRwdXQgPT4ge1xuICBjb25zdCB2YWx1ZSA9IHVzZVdhdGNoKHsgY29udHJvbCwgbmFtZSB9KTtcblxuICByZXR1cm4gdXNlTWVtbygoKSA9PiB0cmFuc2Zvcm0odmFsdWUpLCBbdmFsdWUsIHRyYW5zZm9ybV0pO1xufTtcblxuLyoqXG4gKiBXYXRjaCBmaWVsZCB3aXRoIGRlZmF1bHQgZmFsbGJhY2sgdmFsdWVcbiAqL1xuZXhwb3J0IGNvbnN0IHVzZVdhdGNoRGVmYXVsdCA9IDxcbiAgVEZpZWxkVmFsdWVzIGV4dGVuZHMgRmllbGRWYWx1ZXMsXG4gIFROYW1lIGV4dGVuZHMgUGF0aDxURmllbGRWYWx1ZXM+LFxuPihcbiAgY29udHJvbDogQ29udHJvbDxURmllbGRWYWx1ZXM+LFxuICBuYW1lOiBUTmFtZSxcbiAgZGVmYXVsdFZhbHVlOiBQYXRoVmFsdWU8VEZpZWxkVmFsdWVzLCBUTmFtZT5cbik6IFBhdGhWYWx1ZTxURmllbGRWYWx1ZXMsIFROYW1lPiA9PiB7XG4gIGNvbnN0IHZhbHVlID0gdXNlV2F0Y2goeyBjb250cm9sLCBuYW1lIH0pO1xuXG4gIHJldHVybiB2YWx1ZSA/PyBkZWZhdWx0VmFsdWU7XG59O1xuXG4vKipcbiAqIFdhdGNoIGZpZWxkIGFzIGJvb2xlYW4gd2l0aCBndWFyYW50ZWVkIGJvb2xlYW4gcmV0dXJuXG4gKi9cbmV4cG9ydCBjb25zdCB1c2VXYXRjaEJvb2xlYW4gPSA8XG4gIFRGaWVsZFZhbHVlcyBleHRlbmRzIEZpZWxkVmFsdWVzLFxuICBUTmFtZSBleHRlbmRzIFBhdGg8VEZpZWxkVmFsdWVzPixcbj4oXG4gIGNvbnRyb2w6IENvbnRyb2w8VEZpZWxkVmFsdWVzPixcbiAgbmFtZTogVE5hbWUsXG4gIGRlZmF1bHRWYWx1ZSA9IGZhbHNlXG4pOiBib29sZWFuID0+IHtcbiAgY29uc3QgdmFsdWUgPSB1c2VXYXRjaCh7IGNvbnRyb2wsIG5hbWUgfSk7XG5cbiAgcmV0dXJuIEJvb2xlYW4odmFsdWUgPz8gZGVmYXVsdFZhbHVlKTtcbn07XG5cbi8qKlxuICogV2F0Y2ggbXVsdGlwbGUgZmllbGRzIGFuZCByZXR1cm4gYW4gb2JqZWN0IHdpdGggZmllbGQgcGF0aHMgYXMga2V5c1xuICovXG5leHBvcnQgY29uc3QgdXNlV2F0Y2hCYXRjaCA9IDxcbiAgVEZpZWxkVmFsdWVzIGV4dGVuZHMgRmllbGRWYWx1ZXMsXG4gIFRGaWVsZHMgZXh0ZW5kcyBSZWFkb25seUFycmF5PFBhdGg8VEZpZWxkVmFsdWVzPj4sXG4+KFxuICBjb250cm9sOiBDb250cm9sPFRGaWVsZFZhbHVlcz4sXG4gIGZpZWxkczogVEZpZWxkc1xuKTogeyBbSyBpbiBURmllbGRzW251bWJlcl1dOiBQYXRoVmFsdWU8VEZpZWxkVmFsdWVzLCBLPiB9ID0+IHtcbiAgY29uc3QgdmFsdWVzID0gdXNlV2F0Y2goeyBjb250cm9sLCBuYW1lOiBmaWVsZHMgfSk7XG5cbiAgcmV0dXJuIHVzZU1lbW8oKCkgPT4ge1xuICAgIGNvbnN0IHJlc3VsdCA9IHt9IGFzIHsgW0sgaW4gVEZpZWxkc1tudW1iZXJdXTogUGF0aFZhbHVlPFRGaWVsZFZhbHVlcywgSz4gfTtcblxuICAgIGZpZWxkcy5mb3JFYWNoKChmaWVsZCwgaW5kZXgpID0+IHtcbiAgICAgIHJlc3VsdFtmaWVsZCBhcyBURmllbGRzW251bWJlcl1dID0gdmFsdWVzW2luZGV4XTtcbiAgICB9KTtcblxuICAgIHJldHVybiByZXN1bHQ7XG4gIH0sIFt2YWx1ZXMsIGZpZWxkc10pO1xufTtcblxuLyoqXG4gKiBXYXRjaCBmaWVsZCBjb25kaXRpb25hbGx5IGJhc2VkIG9uIGJvb2xlYW4gZmxhZ1xuICovXG5leHBvcnQgY29uc3QgdXNlV2F0Y2hDb25kaXRpb25hbCA9IDxcbiAgVEZpZWxkVmFsdWVzIGV4dGVuZHMgRmllbGRWYWx1ZXMsXG4gIFROYW1lIGV4dGVuZHMgUGF0aDxURmllbGRWYWx1ZXM+LFxuPihcbiAgY29udHJvbDogQ29udHJvbDxURmllbGRWYWx1ZXM+LFxuICBuYW1lOiBUTmFtZSxcbiAgc2hvdWxkV2F0Y2g6IGJvb2xlYW4sXG4gIGZhbGxiYWNrPzogUGF0aFZhbHVlPFRGaWVsZFZhbHVlcywgVE5hbWU+XG4pOiBQYXRoVmFsdWU8VEZpZWxkVmFsdWVzLCBUTmFtZT4gfCB1bmRlZmluZWQgPT4ge1xuICBjb25zdCBhY3RpdmVWYWx1ZSA9IHVzZVdhdGNoKHtcbiAgICBjb250cm9sLFxuICAgIG5hbWUsXG4gICAgZGlzYWJsZWQ6ICFzaG91bGRXYXRjaCxcbiAgfSk7XG5cbiAgcmV0dXJuIHNob3VsZFdhdGNoID8gYWN0aXZlVmFsdWUgOiBmYWxsYmFjaztcbn07XG5cbi8qKlxuICogV2F0Y2ggZmllbGQgd2l0aCBkZWJvdW5jZWQgdXBkYXRlc1xuICovXG5leHBvcnQgY29uc3QgdXNlV2F0Y2hEZWJvdW5jZWQgPSA8XG4gIFRGaWVsZFZhbHVlcyBleHRlbmRzIEZpZWxkVmFsdWVzLFxuICBUTmFtZSBleHRlbmRzIFBhdGg8VEZpZWxkVmFsdWVzPixcbj4oXG4gIGNvbnRyb2w6IENvbnRyb2w8VEZpZWxkVmFsdWVzPixcbiAgbmFtZTogVE5hbWUsXG4gIGRlbGF5ID0gMzAwXG4pOiBQYXRoVmFsdWU8VEZpZWxkVmFsdWVzLCBUTmFtZT4gPT4ge1xuICBjb25zdCB2YWx1ZSA9IHVzZVdhdGNoKHsgY29udHJvbCwgbmFtZSB9KTtcbiAgY29uc3QgW2RlYm91bmNlZFZhbHVlLCBzZXREZWJvdW5jZWRWYWx1ZV0gPVxuICAgIHVzZVN0YXRlPFBhdGhWYWx1ZTxURmllbGRWYWx1ZXMsIFROYW1lPj4odmFsdWUpO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3QgdGltZXIgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIHNldERlYm91bmNlZFZhbHVlKHZhbHVlKTtcbiAgICB9LCBkZWxheSk7XG5cbiAgICByZXR1cm4gKCkgPT4gY2xlYXJUaW1lb3V0KHRpbWVyKTtcbiAgfSwgW3ZhbHVlLCBkZWxheV0pO1xuXG4gIHJldHVybiBkZWJvdW5jZWRWYWx1ZTtcbn07XG5cbi8qKlxuICogV2F0Y2ggZmllbGQgd2l0aCBtZW1vaXplZCBzZWxlY3RvciBmdW5jdGlvblxuICovXG5leHBvcnQgY29uc3QgdXNlV2F0Y2hTZWxlY3RvciA9IDxcbiAgVEZpZWxkVmFsdWVzIGV4dGVuZHMgRmllbGRWYWx1ZXMsXG4gIFROYW1lIGV4dGVuZHMgUGF0aDxURmllbGRWYWx1ZXM+LFxuICBUT3V0cHV0LFxuPihcbiAgY29udHJvbDogQ29udHJvbDxURmllbGRWYWx1ZXM+LFxuICBuYW1lOiBUTmFtZSxcbiAgc2VsZWN0b3I6ICh2YWx1ZTogUGF0aFZhbHVlPFRGaWVsZFZhbHVlcywgVE5hbWU+KSA9PiBUT3V0cHV0LFxuICBkZXBzOiBSZWFjdC5EZXBlbmRlbmN5TGlzdCA9IFtdXG4pOiBUT3V0cHV0ID0+IHtcbiAgY29uc3QgdmFsdWUgPSB1c2VXYXRjaCh7IGNvbnRyb2wsIG5hbWUgfSk7XG5cbiAgcmV0dXJuIHVzZU1lbW8oXG4gICAgKCkgPT4gc2VsZWN0b3IodmFsdWUpLFxuICAgIFt2YWx1ZSwgc2VsZWN0b3IsIC4uLmRlcHNdIC8vIGVzbGludC1kaXNhYmxlLWxpbmUgcmVhY3QtaG9va3MvZXhoYXVzdGl2ZS1kZXBzXG4gICk7XG59O1xuIiwgIi8qKlxuICogRW5oYW5jZWQgVHlwZVNjcmlwdCB1dGlsaXRpZXMgZm9yIFJlYWN0IEhvb2sgRm9ybSdzIHVzZVdhdGNoXG4gKlxuICogVGhpcyBtb2R1bGUgcHJvdmlkZXMgYSBjb21wcmVoZW5zaXZlIHNldCBvZiB0eXBlLXNhZmUgd2F0Y2ggZnVuY3Rpb25zXG4gKiB3aXRoIGJldHRlciBlcmdvbm9taWNzIGFuZCBhZGRpdGlvbmFsIGZ1bmN0aW9uYWxpdHkuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IHVzZVdhdGNoRmllbGQsIHVzZVdhdGNoQm9vbGVhbiwgdHlwZWRXYXRjaCB9IGZyb20gJ3NyYy91dGlscy93YXRjaCc7XG4gKlxuICogLy8gRGlyZWN0IHVzYWdlIChpbnNpZGUgUmVhY3QgY29tcG9uZW50cylcbiAqIGNvbnN0IGVtYWlsID0gdXNlV2F0Y2hGaWVsZChjb250cm9sLCAndXNlci5lbWFpbCcpO1xuICogY29uc3QgaXNBZG1pbiA9IHVzZVdhdGNoQm9vbGVhbihjb250cm9sLCAndXNlci5pc0FkbWluJyk7XG4gKlxuICogLy8gT2JqZWN0LWJhc2VkIHVzYWdlIChpbnNpZGUgUmVhY3QgY29tcG9uZW50cylcbiAqIGNvbnN0IGVtYWlsID0gdHlwZWRXYXRjaC5maWVsZChjb250cm9sLCAndXNlci5lbWFpbCcpO1xuICogY29uc3QgaXNBZG1pbiA9IHR5cGVkV2F0Y2guYm9vbGVhbihjb250cm9sLCAndXNlci5pc0FkbWluJyk7XG4gKiBgYGBcbiAqL1xuXG4vLyBDb3JlIGZ1bmN0aW9uc1xuZXhwb3J0IHsgdXNlV2F0Y2hGaWVsZCwgdXNlV2F0Y2hGaWVsZHMsIHVzZVdhdGNoRm9ybSB9IGZyb20gJy4vY29yZSc7XG5cbi8vIFV0aWxpdHkgZnVuY3Rpb25zXG5leHBvcnQge1xuICB1c2VXYXRjaEJhdGNoLFxuICB1c2VXYXRjaEJvb2xlYW4sXG4gIHVzZVdhdGNoQ29uZGl0aW9uYWwsXG4gIHVzZVdhdGNoRGVib3VuY2VkLFxuICB1c2VXYXRjaERlZmF1bHQsXG4gIHVzZVdhdGNoU2VsZWN0b3IsXG4gIHVzZVdhdGNoVHJhbnNmb3JtLFxufSBmcm9tICcuL3V0aWxpdGllcyc7XG5cbmV4cG9ydCB0eXBlIHsgUGF0aEFycmF5IH0gZnJvbSAnLi9jb3JlJztcblxuLy8gSW1wb3J0IGFsbCBmdW5jdGlvbnMgZm9yIGRlZmF1bHQgZXhwb3J0XG5pbXBvcnQgeyB1c2VXYXRjaEZpZWxkLCB1c2VXYXRjaEZpZWxkcywgdXNlV2F0Y2hGb3JtIH0gZnJvbSAnLi9jb3JlJztcbmltcG9ydCB7XG4gIHVzZVdhdGNoQmF0Y2gsXG4gIHVzZVdhdGNoQm9vbGVhbixcbiAgdXNlV2F0Y2hDb25kaXRpb25hbCxcbiAgdXNlV2F0Y2hEZWJvdW5jZWQsXG4gIHVzZVdhdGNoRGVmYXVsdCxcbiAgdXNlV2F0Y2hTZWxlY3RvcixcbiAgdXNlV2F0Y2hUcmFuc2Zvcm0sXG59IGZyb20gJy4vdXRpbGl0aWVzJztcblxuLyoqXG4gKiBPcmdhbml6ZWQgdXRpbGl0aWVzIGJ5IHVzZSBjYXNlXG4gKiBQcm92aWRlcyBhIGNvbnZlbmllbnQgb2JqZWN0LWJhc2VkIEFQSSBmb3IgYWxsIHdhdGNoIGZ1bmN0aW9uc1xuICovXG5leHBvcnQgY29uc3QgdHlwZWRXYXRjaCA9IHtcbiAgLy8gPT09IENPUkUgRlVOQ1RJT05TID09PVxuICAvKiogV2F0Y2ggZW50aXJlIGZvcm0gKi9cbiAgZm9ybTogdXNlV2F0Y2hGb3JtLFxuICAvKiogV2F0Y2ggc2luZ2xlIGZpZWxkICovXG4gIGZpZWxkOiB1c2VXYXRjaEZpZWxkLFxuICAvKiogV2F0Y2ggbXVsdGlwbGUgZmllbGRzICovXG4gIGZpZWxkczogdXNlV2F0Y2hGaWVsZHMsXG5cbiAgLy8gPT09IFVUSUxJVFkgRlVOQ1RJT05TID09PVxuICAvKiogV2F0Y2ggd2l0aCB0cmFuc2Zvcm1hdGlvbiAqL1xuICB0cmFuc2Zvcm06IHVzZVdhdGNoVHJhbnNmb3JtLFxuICAvKiogV2F0Y2ggd2l0aCBkZWZhdWx0IHZhbHVlICovXG4gIHdpdGhEZWZhdWx0OiB1c2VXYXRjaERlZmF1bHQsXG4gIC8qKiBXYXRjaCBhcyBib29sZWFuICovXG4gIGJvb2xlYW46IHVzZVdhdGNoQm9vbGVhbixcbiAgLyoqIFdhdGNoIG11bHRpcGxlIHdpdGggY3VzdG9tIGtleXMgKi9cbiAgYmF0Y2g6IHVzZVdhdGNoQmF0Y2gsXG4gIC8qKiBXYXRjaCBjb25kaXRpb25hbGx5ICovXG4gIGNvbmRpdGlvbmFsOiB1c2VXYXRjaENvbmRpdGlvbmFsLFxuICAvKiogV2F0Y2ggd2l0aCBkZWJvdW5jaW5nICovXG4gIGRlYm91bmNlZDogdXNlV2F0Y2hEZWJvdW5jZWQsXG4gIC8qKiBXYXRjaCB3aXRoIHNlbGVjdG9yICovXG4gIHNlbGVjdG9yOiB1c2VXYXRjaFNlbGVjdG9yLFxufSBhcyBjb25zdDtcbiIsICJleHBvcnQgY29uc3QgY2FsY3VsYXRlRmlsdGVyQ291bnQgPSAobW9kZWw6IG9iamVjdCk6IG51bWJlciA9PlxuICBPYmplY3QudmFsdWVzKG1vZGVsKS5maWx0ZXIoXG4gICAgdiA9PiB2ICE9PSBudWxsICYmIHYgIT09IHVuZGVmaW5lZCAmJiBTdHJpbmcodikudHJpbSgpICE9PSAnJ1xuICApLmxlbmd0aDtcbiIsICJpbXBvcnQgdHlwZSB7IERheWpzLCBPcFVuaXRUeXBlIH0gZnJvbSAnZGF5anMnO1xuaW1wb3J0IGRheWpzIGZyb20gJ2RheWpzJztcbmltcG9ydCBkdXJhdGlvbiBmcm9tICdkYXlqcy9wbHVnaW4vZHVyYXRpb24nO1xuaW1wb3J0IHJlbGF0aXZlVGltZSBmcm9tICdkYXlqcy9wbHVnaW4vcmVsYXRpdmVUaW1lJztcblxuLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG4vKipcbiAqIEBEb2NzXG4gKiBodHRwczovL2RheS5qcy5vcmcvZG9jcy9lbi9kaXNwbGF5L2Zvcm1hdFxuICovXG5cbi8qKlxuICogRGVmYXVsdCB0aW1lem9uZXNcbiAqIGh0dHBzOi8vZGF5LmpzLm9yZy9kb2NzL2VuL3RpbWV6b25lL3NldC1kZWZhdWx0LXRpbWV6b25lI2RvY3NOYXZcbiAqXG4gKi9cblxuLyoqXG4gKiBVVENcbiAqIGh0dHBzOi8vZGF5LmpzLm9yZy9kb2NzL2VuL3BsdWdpbi91dGNcbiAqIEBpbnN0YWxsXG4gKiBpbXBvcnQgdXRjIGZyb20gJ2RheWpzL3BsdWdpbi91dGMnO1xuICogZGF5anMuZXh0ZW5kKHV0Yyk7XG4gKiBAdXNhZ2VcbiAqIGRheWpzKCkudXRjKCkuZm9ybWF0KClcbiAqXG4gKi9cblxuZGF5anMuZXh0ZW5kKGR1cmF0aW9uKTtcbmRheWpzLmV4dGVuZChyZWxhdGl2ZVRpbWUpO1xuXG4vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cbmV4cG9ydCB0eXBlIERhdGVQaWNrZXJGb3JtYXQgPVxuICB8IERheWpzXG4gIHwgRGF0ZVxuICB8IHN0cmluZ1xuICB8IG51bWJlclxuICB8IG51bGxcbiAgfCB1bmRlZmluZWQ7XG5cbmV4cG9ydCBjb25zdCBmb3JtYXRQYXR0ZXJucyA9IHtcbiAgZGF0ZVRpbWU6ICdERCBNTU0gWVlZWSBoOm1tIEEnLCAvLyAxNyBBcHIgMjAyMiAxMjowMCBhbVxuICBkYXRlOiAnREQgTU1NIFlZWVknLCAvLyAxNyBBcHIgMjAyMlxuICBtb250aF95ZWFyX3Nob3J0X2Zvcm1hdDogJ01NTSBZWVlZJyxcbiAgbW9udGhfeWVhcl9mdWxsX2Zvcm1hdDogJ01NTU0gWVlZWScsXG4gIHllYXI6ICdZWVlZJyxcbiAgdGltZTogJ2g6bW0gYScsIC8vIDEyOjAwIGFtXG4gIHNwbGl0OiB7XG4gICAgZGF0ZVRpbWU6ICdERC9NTS9ZWVlZIGg6bW0gQScsIC8vIDE3LzA0LzIwMjIgMTI6MDAgYW1cbiAgICBkYXRlOiAnREQvTU0vWVlZWScsIC8vIDE3LzA0LzIwMjJcbiAgfSxcbiAgcGFyYW1DYXNlOiB7XG4gICAgZGF0ZVRpbWU6ICdERC1NTS1ZWVlZIGg6bW0gQScsIC8vIDE3LTA0LTIwMjIgMTI6MDAgYW1cbiAgICBkYXRlOiAnREQtTU0tWVlZWScsIC8vIDE3LTA0LTIwMjJcbiAgICBkYXRlUmV2ZXJzZTogJ1lZWVktTU0tREQnLCAvLyAyMDIyLTA0LTE3IGZvciBjb21wYXJlIGRhdGVcbiAgICBNb250aFllYXI6ICdNTU0tWVlZWScsXG4gIH0sXG59O1xuXG5jb25zdCBpc1ZhbGlkRGF0ZSA9IChkYXRlOiBEYXRlUGlja2VyRm9ybWF0KSA9PlxuICBkYXRlICE9PSBudWxsICYmIGRhdGUgIT09IHVuZGVmaW5lZCAmJiBkYXlqcyhkYXRlKS5pc1ZhbGlkKCk7XG5cbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuZXhwb3J0IGZ1bmN0aW9uIHRvZGF5KHRlbXBsYXRlPzogc3RyaW5nKTogc3RyaW5nIHtcbiAgcmV0dXJuIGRheWpzKG5ldyBEYXRlKCkpLnN0YXJ0T2YoJ2RheScpLmZvcm1hdCh0ZW1wbGF0ZSk7XG59XG5cbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuLyoqXG4gKiBAb3V0cHV0IDE3IEFwciAyMDIyIDEyOjAwIGFtXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBmRGF0ZVRpbWUoZGF0ZTogRGF0ZVBpY2tlckZvcm1hdCwgdGVtcGxhdGU/OiBzdHJpbmcpOiBzdHJpbmcge1xuICBpZiAoIWlzVmFsaWREYXRlKGRhdGUpKSB7XG4gICAgcmV0dXJuICdJbnZhbGlkIGRhdGUnO1xuICB9XG5cbiAgcmV0dXJuIGRheWpzKGRhdGUpLmZvcm1hdCh0ZW1wbGF0ZSA/PyBmb3JtYXRQYXR0ZXJucy5kYXRlVGltZSk7XG59XG5cbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuLyoqXG4gKiBAb3V0cHV0IDE3IEFwciAyMDIyXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBmRGF0ZShkYXRlOiBEYXRlUGlja2VyRm9ybWF0LCB0ZW1wbGF0ZT86IHN0cmluZyk6IHN0cmluZyB7XG4gIGlmICghaXNWYWxpZERhdGUoZGF0ZSkpIHtcbiAgICByZXR1cm4gJ0ludmFsaWQgZGF0ZSc7XG4gIH1cblxuICByZXR1cm4gZGF5anMoZGF0ZSkuZm9ybWF0KHRlbXBsYXRlID8/IGZvcm1hdFBhdHRlcm5zLmRhdGUpO1xufVxuXG4vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cbi8qKlxuICogQG91dHB1dCAxMjowMCBhbVxuICovXG5leHBvcnQgZnVuY3Rpb24gZlRpbWUoZGF0ZTogRGF0ZVBpY2tlckZvcm1hdCwgdGVtcGxhdGU/OiBzdHJpbmcpOiBzdHJpbmcge1xuICBpZiAoIWlzVmFsaWREYXRlKGRhdGUpKSB7XG4gICAgcmV0dXJuICdJbnZhbGlkIGRhdGUnO1xuICB9XG5cbiAgcmV0dXJuIGRheWpzKGRhdGUpLmZvcm1hdCh0ZW1wbGF0ZSA/PyBmb3JtYXRQYXR0ZXJucy50aW1lKTtcbn1cblxuLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG4vKipcbiAqIEBvdXRwdXQgMTcxMzI1MDEwMFxuICovXG5leHBvcnQgZnVuY3Rpb24gZlRpbWVzdGFtcChkYXRlOiBEYXRlUGlja2VyRm9ybWF0KTogbnVtYmVyIHwgJ0ludmFsaWQgZGF0ZScge1xuICBpZiAoIWlzVmFsaWREYXRlKGRhdGUpKSB7XG4gICAgcmV0dXJuICdJbnZhbGlkIGRhdGUnO1xuICB9XG5cbiAgcmV0dXJuIGRheWpzKGRhdGUpLnZhbHVlT2YoKTtcbn1cblxuLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG4vKipcbiAqIEBvdXRwdXQgYSBmZXcgc2Vjb25kcywgMiB5ZWFyc1xuICovXG5leHBvcnQgZnVuY3Rpb24gZlRvTm93KGRhdGU6IERhdGVQaWNrZXJGb3JtYXQpOiBzdHJpbmcge1xuICBpZiAoIWlzVmFsaWREYXRlKGRhdGUpKSB7XG4gICAgcmV0dXJuICdJbnZhbGlkIGRhdGUnO1xuICB9XG5cbiAgcmV0dXJuIGRheWpzKGRhdGUpLnRvTm93KHRydWUpO1xufVxuXG4vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cbi8qKlxuICogQG91dHB1dCBib29sZWFuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBmSXNCZXR3ZWVuKFxuICBpbnB1dERhdGU6IERhdGVQaWNrZXJGb3JtYXQsXG4gIHN0YXJ0RGF0ZTogRGF0ZVBpY2tlckZvcm1hdCxcbiAgZW5kRGF0ZTogRGF0ZVBpY2tlckZvcm1hdFxuKTogYm9vbGVhbiB7XG4gIGlmIChcbiAgICAhaXNWYWxpZERhdGUoaW5wdXREYXRlKSB8fFxuICAgICFpc1ZhbGlkRGF0ZShzdGFydERhdGUpIHx8XG4gICAgIWlzVmFsaWREYXRlKGVuZERhdGUpXG4gICkge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdHRlZElucHV0RGF0ZSA9IGZUaW1lc3RhbXAoaW5wdXREYXRlKTtcbiAgY29uc3QgZm9ybWF0dGVkU3RhcnREYXRlID0gZlRpbWVzdGFtcChzdGFydERhdGUpO1xuICBjb25zdCBmb3JtYXR0ZWRFbmREYXRlID0gZlRpbWVzdGFtcChlbmREYXRlKTtcblxuICBpZiAoXG4gICAgZm9ybWF0dGVkSW5wdXREYXRlID09PSAnSW52YWxpZCBkYXRlJyB8fFxuICAgIGZvcm1hdHRlZFN0YXJ0RGF0ZSA9PT0gJ0ludmFsaWQgZGF0ZScgfHxcbiAgICBmb3JtYXR0ZWRFbmREYXRlID09PSAnSW52YWxpZCBkYXRlJ1xuICApIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICByZXR1cm4gKFxuICAgIGZvcm1hdHRlZElucHV0RGF0ZSA+PSBmb3JtYXR0ZWRTdGFydERhdGUgJiZcbiAgICBmb3JtYXR0ZWRJbnB1dERhdGUgPD0gZm9ybWF0dGVkRW5kRGF0ZVxuICApO1xufVxuXG4vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cbi8qKlxuICogQG91dHB1dCBib29sZWFuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBmSXNBZnRlcihcbiAgc3RhcnREYXRlOiBEYXRlUGlja2VyRm9ybWF0LFxuICBlbmREYXRlOiBEYXRlUGlja2VyRm9ybWF0XG4pOiBib29sZWFuIHtcbiAgaWYgKCFpc1ZhbGlkRGF0ZShzdGFydERhdGUpIHx8ICFpc1ZhbGlkRGF0ZShlbmREYXRlKSkge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIHJldHVybiBkYXlqcyhzdGFydERhdGUpLmlzQWZ0ZXIoZW5kRGF0ZSk7XG59XG5cbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuLyoqXG4gKiBAb3V0cHV0IGJvb2xlYW5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZJc1NhbWUoXG4gIHN0YXJ0RGF0ZTogRGF0ZVBpY2tlckZvcm1hdCxcbiAgZW5kRGF0ZTogRGF0ZVBpY2tlckZvcm1hdCxcbiAgdW5pdFRvQ29tcGFyZT86IE9wVW5pdFR5cGVcbik6IGJvb2xlYW4ge1xuICBpZiAoIWlzVmFsaWREYXRlKHN0YXJ0RGF0ZSkgfHwgIWlzVmFsaWREYXRlKGVuZERhdGUpKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgcmV0dXJuIGRheWpzKHN0YXJ0RGF0ZSkuaXNTYW1lKGVuZERhdGUsIHVuaXRUb0NvbXBhcmUgPz8gJ3llYXInKTtcbn1cblxuLyoqXG4gKiBAb3V0cHV0XG4gKiBTYW1lIGRheTogMjYgQXByIDIwMjRcbiAqIFNhbWUgbW9udGg6IDI1IC0gMjYgQXByIDIwMjRcbiAqIFNhbWUgbW9udGg6IDI1IC0gMjYgQXByIDIwMjRcbiAqIFNhbWUgeWVhcjogMjUgQXByIC0gMjYgTWF5IDIwMjRcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZEYXRlUmFuZ2VTaG9ydExhYmVsKFxuICBzdGFydERhdGU6IERhdGVQaWNrZXJGb3JtYXQsXG4gIGVuZERhdGU6IERhdGVQaWNrZXJGb3JtYXQsXG4gIGluaXRpYWw/OiBib29sZWFuXG4pOiBzdHJpbmcge1xuICBpZiAoXG4gICAgIWlzVmFsaWREYXRlKHN0YXJ0RGF0ZSkgfHxcbiAgICAhaXNWYWxpZERhdGUoZW5kRGF0ZSkgfHxcbiAgICBmSXNBZnRlcihzdGFydERhdGUsIGVuZERhdGUpXG4gICkge1xuICAgIHJldHVybiAnSW52YWxpZCBkYXRlJztcbiAgfVxuXG4gIGxldCBsYWJlbCA9IGAke2ZEYXRlKHN0YXJ0RGF0ZSl9IC0gJHtmRGF0ZShlbmREYXRlKX1gO1xuXG4gIGlmIChpbml0aWFsKSB7XG4gICAgcmV0dXJuIGxhYmVsO1xuICB9XG5cbiAgY29uc3QgaXNTYW1lWWVhciA9IGZJc1NhbWUoc3RhcnREYXRlLCBlbmREYXRlLCAneWVhcicpO1xuICBjb25zdCBpc1NhbWVNb250aCA9IGZJc1NhbWUoc3RhcnREYXRlLCBlbmREYXRlLCAnbW9udGgnKTtcbiAgY29uc3QgaXNTYW1lRGF5ID0gZklzU2FtZShzdGFydERhdGUsIGVuZERhdGUsICdkYXknKTtcblxuICBpZiAoaXNTYW1lWWVhciAmJiAhaXNTYW1lTW9udGgpIHtcbiAgICBsYWJlbCA9IGAke2ZEYXRlKHN0YXJ0RGF0ZSwgJ0REIE1NTScpfSAtICR7ZkRhdGUoZW5kRGF0ZSl9YDtcbiAgfSBlbHNlIGlmIChpc1NhbWVZZWFyICYmIGlzU2FtZU1vbnRoICYmICFpc1NhbWVEYXkpIHtcbiAgICBsYWJlbCA9IGAke2ZEYXRlKHN0YXJ0RGF0ZSwgJ0REJyl9IC0gJHtmRGF0ZShlbmREYXRlKX1gO1xuICB9IGVsc2UgaWYgKGlzU2FtZVllYXIgJiYgaXNTYW1lTW9udGggJiYgaXNTYW1lRGF5KSB7XG4gICAgbGFiZWwgPSBgJHtmRGF0ZShlbmREYXRlKX1gO1xuICB9XG5cbiAgcmV0dXJuIGxhYmVsO1xufVxuXG4vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cbi8qKlxuICogQG91dHB1dCAyMDI0LTA1LTI4VDA1OjU1OjMxKzAwOjAwXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRHVyYXRpb25Qcm9wcyB7XG4gIHllYXJzPzogbnVtYmVyO1xuICBtb250aHM/OiBudW1iZXI7XG4gIGRheXM/OiBudW1iZXI7XG4gIGhvdXJzPzogbnVtYmVyO1xuICBtaW51dGVzPzogbnVtYmVyO1xuICBzZWNvbmRzPzogbnVtYmVyO1xuICBtaWxsaXNlY29uZHM/OiBudW1iZXI7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBmQWRkKHtcbiAgeWVhcnMgPSAwLFxuICBtb250aHMgPSAwLFxuICBkYXlzID0gMCxcbiAgaG91cnMgPSAwLFxuICBtaW51dGVzID0gMCxcbiAgc2Vjb25kcyA9IDAsXG4gIG1pbGxpc2Vjb25kcyA9IDAsXG59OiBEdXJhdGlvblByb3BzKSB7XG4gIGNvbnN0IHJlc3VsdCA9IGRheWpzKClcbiAgICAuYWRkKFxuICAgICAgZGF5anMuZHVyYXRpb24oe1xuICAgICAgICB5ZWFycyxcbiAgICAgICAgbW9udGhzLFxuICAgICAgICBkYXlzLFxuICAgICAgICBob3VycyxcbiAgICAgICAgbWludXRlcyxcbiAgICAgICAgc2Vjb25kcyxcbiAgICAgICAgbWlsbGlzZWNvbmRzLFxuICAgICAgfSlcbiAgICApXG4gICAgLmZvcm1hdCgpO1xuXG4gIHJldHVybiByZXN1bHQ7XG59XG5cbi8qKlxuICogQG91dHB1dCAyMDI0LTA1LTI4VDA1OjU1OjMxKzAwOjAwXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBmU3ViKHtcbiAgeWVhcnMgPSAwLFxuICBtb250aHMgPSAwLFxuICBkYXlzID0gMCxcbiAgaG91cnMgPSAwLFxuICBtaW51dGVzID0gMCxcbiAgc2Vjb25kcyA9IDAsXG4gIG1pbGxpc2Vjb25kcyA9IDAsXG59OiBEdXJhdGlvblByb3BzKSB7XG4gIGNvbnN0IHJlc3VsdCA9IGRheWpzKClcbiAgICAuc3VidHJhY3QoXG4gICAgICBkYXlqcy5kdXJhdGlvbih7XG4gICAgICAgIHllYXJzLFxuICAgICAgICBtb250aHMsXG4gICAgICAgIGRheXMsXG4gICAgICAgIGhvdXJzLFxuICAgICAgICBtaW51dGVzLFxuICAgICAgICBzZWNvbmRzLFxuICAgICAgICBtaWxsaXNlY29uZHMsXG4gICAgICB9KVxuICAgIClcbiAgICAuZm9ybWF0KCk7XG5cbiAgcmV0dXJuIHJlc3VsdDtcbn1cbiIsICJ0eXBlIEVtcHR5VmFsdWU8VD4gPSBUIGV4dGVuZHMgbnVtYmVyXG4gID8gMFxuICA6IFQgZXh0ZW5kcyBzdHJpbmdcbiAgICA/IG51bGxcbiAgICA6IFQgZXh0ZW5kcyBib29sZWFuXG4gICAgICA/IG51bGxcbiAgICAgIDogVCBleHRlbmRzIERhdGVcbiAgICAgICAgPyBudWxsXG4gICAgICAgIDogVCBleHRlbmRzIG9iamVjdFxuICAgICAgICAgID8gbnVsbFxuICAgICAgICAgIDogbnVsbDtcblxudHlwZSBFbXB0eU9iamVjdDxUIGV4dGVuZHMgb2JqZWN0PiA9IHtcbiAgW0sgaW4ga2V5b2YgVF06IEVtcHR5VmFsdWU8VFtLXT47XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0RW1wdHlPYmplY3Q8VCBleHRlbmRzIG9iamVjdD4oXG4gIGRhdGE6IFQsXG4gIGRlZmF1bHRWYWx1ZXM6IFBhcnRpYWw8VD4gPSB7fVxuKTogRW1wdHlPYmplY3Q8VD4gJiBQYXJ0aWFsPFQ+IHtcbiAgY29uc3Qgb2JqID0ge30gYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG5cbiAgZm9yIChjb25zdCBrZXkgb2YgT2JqZWN0LmtleXMoZGF0YSkgYXMgQXJyYXk8a2V5b2YgVD4pIHtcbiAgICBjb25zdCB2YWx1ZSA9IGRhdGFba2V5XTtcbiAgICBjb25zdCB0eXBlID0gdHlwZW9mIHZhbHVlO1xuXG4gICAgaWYgKHR5cGUgPT09ICdudW1iZXInKSB7XG4gICAgICBvYmpba2V5IGFzIHN0cmluZ10gPSAwO1xuICAgIH0gZWxzZSBpZiAodHlwZSA9PT0gJ3N0cmluZycgfHwgdHlwZSA9PT0gJ2Jvb2xlYW4nKSB7XG4gICAgICBvYmpba2V5IGFzIHN0cmluZ10gPSBudWxsO1xuICAgIH0gZWxzZSBpZiAodmFsdWUgaW5zdGFuY2VvZiBEYXRlKSB7XG4gICAgICBvYmpba2V5IGFzIHN0cmluZ10gPSBudWxsO1xuICAgIH0gZWxzZSB7XG4gICAgICBvYmpba2V5IGFzIHN0cmluZ10gPSBudWxsO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiB7IC4uLm9iaiwgLi4uZGVmYXVsdFZhbHVlcyB9IGFzIEVtcHR5T2JqZWN0PFQ+ICYgUGFydGlhbDxUPjtcbn1cbiIsICJpbXBvcnQgeyB1c2VSZWYsIHVzZU1lbW8gfSBmcm9tICdyZWFjdCc7XG5cbi8qKlxuICogSG9vayB0byBtYWludGFpbiBzdGFibGUgcm93IGNvdW50IGZvciBkYXRhIGdyaWRzIGR1cmluZyBsb2FkaW5nIHN0YXRlcy5cbiAqIFByZXZlbnRzIHBhZ2luYXRpb24ganVtcGluZyBieSBwcmVzZXJ2aW5nIHRoZSBsYXN0IGtub3duIHRvdGFsIGNvdW50LlxuICpcbiAqIEBwYXJhbSBjdXJyZW50VG90YWwgLSBDdXJyZW50IHRvdGFsIGZyb20gQVBJIHJlc3BvbnNlXG4gKiBAcmV0dXJucyBTdGFibGUgcm93IGNvdW50IHRoYXQgcGVyc2lzdHMgZHVyaW5nIGxvYWRpbmdcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHVzZVN0YWJsZVJvd0NvdW50KGN1cnJlbnRUb3RhbDogbnVtYmVyIHwgdW5kZWZpbmVkKTogbnVtYmVyIHtcbiAgY29uc3Qgcm93Q291bnRSZWYgPSB1c2VSZWYoY3VycmVudFRvdGFsIHx8IDApO1xuXG4gIGNvbnN0IHN0YWJsZVJvd0NvdW50ID0gdXNlTWVtbygoKSA9PiB7XG4gICAgaWYgKGN1cnJlbnRUb3RhbCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICByb3dDb3VudFJlZi5jdXJyZW50ID0gY3VycmVudFRvdGFsO1xuICAgIH1cblxuICAgIHJldHVybiByb3dDb3VudFJlZi5jdXJyZW50O1xuICB9LCBbY3VycmVudFRvdGFsXSk7XG5cbiAgcmV0dXJuIHN0YWJsZVJvd0NvdW50O1xufVxuIl0sCiAgIm1hcHBpbmdzIjogIjtBQXdCQSxTQUFTLGVBQXVCO0FBQzlCLE1BQUksT0FBTyxXQUFXLGVBQWUsT0FBTyxZQUFZO0FBQ3RELFdBQU8sT0FBTyxXQUFXO0FBQUEsRUFDM0I7QUFHQSxTQUFPLHVDQUF1QyxRQUFRLFNBQVMsT0FBSztBQUNsRSxVQUFNLElBQUssS0FBSyxPQUFPLElBQUksS0FBTTtBQUNqQyxVQUFNLElBQUksTUFBTSxNQUFNLElBQUssSUFBSSxJQUFPO0FBRXRDLFdBQU8sRUFBRSxTQUFTLEVBQUU7QUFBQSxFQUN0QixDQUFDO0FBQ0g7QUE0Q08sU0FBUyxzQkFBc0IsUUFBeUI7QUFDN0QsUUFBTSxPQUFPLGFBQWE7QUFFMUIsU0FBTyxTQUFTLEdBQUcsTUFBTSxJQUFJLElBQUksS0FBSztBQUN4Qzs7O0FDakVPLElBQU0sa0JBQU4sTUFBc0I7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFxQjNCLGFBQWEsUUFBOEI7QUFDekMsUUFBSSxVQUFVLE9BQU8sU0FBUyxLQUFLO0FBQ2pDLGFBQU8sV0FB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S0FBSyxHQUFHLGNBQWM7QUFDcEM7OztBQ3RDQSxTQUFTLFFBQVEsV0FBQUcsZ0JBQWU7QUFTekIsU0FBUyxrQkFBa0IsY0FBMEM7QUFDMUUsUUFBTSxjQUFjLE9BQU8sZ0JBQWdCLENBQUM7QUFFNUMsUUFBTSxpQkFBaUJBLFNBQVEsTUFBTTtBQUNuQyxRQUFJLGlCQUFpQixRQUFXO0FBQzlCLGtCQUFZLFVBQVU7QUFBQSxJQUN4QjtBQUVBLFdBQU8sWUFBWTtBQUFBLEVBQ3JCLEdBQUcsQ0FBQyxZQUFZLENBQUM7QUFFakIsU0FBTztBQUNUOyIsCiAgIm5hbWVzIjogWyJjb25maWciLCAianN4IiwgIkJ1dHRvbiIsICJqc3giLCAianN4IiwgImpzeCIsICJqc3giLCAibWVtbyIsICJqc3giLCAibWVtbyIsICJCb3giLCAiQ2FyZCIsICJDYXJkQ29udGVudCIsICJUeXBvZ3JhcGh5IiwgIkZyYWdtZW50IiwgImpzeCIsICJqc3hzIiwgIkJveCIsICJUeXBvZ3JhcGh5IiwgImpzeCIsICJHcmlkIiwgIlR5cG9ncmFwaHkiLCAidXNlVGhlbWUiLCAianN4IiwgImpzeHMiLCAiRnJhZ21lbnQiLCAianN4IiwgIkJveCIsICJEaXZpZGVyIiwgIkdyaWQiLCAiVHlwb2dyYXBoeSIsICJtZW1vIiwgInVzZU1lbW8iLCAiRnJhZ21lbnQiLCAianN4IiwgImpzeHMiLCAiQm94IiwgImpzeCIsICJqc3hzIiwgIkxvYWRpbmdCdXR0b24iLCAianN4IiwgInVzZU1lbW8iLCAidXNlTWVtbyIsICJ1c2VDYWxsYmFjayIsICJ1c2VNZW1vIiwgInVzZU1lbW8iLCAidXNlU3RhdGUiLCAidXNlV2F0Y2giLCAidXNlTWVtbyJdCn0K