@wavemaker/app-ng-runtime 11.14.1-1.6289 → 11.14.1-10.6348

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 (67) hide show
  1. package/components/base/bundles/index.umd.js +86 -17
  2. package/components/base/esm2022/pipes/custom-pipes.mjs +10 -10
  3. package/components/base/esm2022/utils/widget-utils.mjs +3 -3
  4. package/components/base/esm2022/widgets/common/base/base.component.mjs +67 -7
  5. package/components/base/esm2022/widgets/common/lazy-load/lazy-load.directive.mjs +7 -3
  6. package/components/base/esm2022/widgets/framework/property-change-handler.mjs +7 -2
  7. package/components/base/fesm2022/index.mjs +87 -18
  8. package/components/base/fesm2022/index.mjs.map +1 -1
  9. package/components/base/pipes/custom-pipes.d.ts +5 -5
  10. package/components/basic/label/bundles/index.umd.js +9 -1
  11. package/components/basic/label/esm2022/label.directive.mjs +10 -2
  12. package/components/basic/label/fesm2022/index.mjs +9 -1
  13. package/components/basic/label/fesm2022/index.mjs.map +1 -1
  14. package/components/data/pagination/bundles/index.umd.js +4 -0
  15. package/components/data/pagination/esm2022/pagination.component.mjs +5 -1
  16. package/components/data/pagination/fesm2022/index.mjs +4 -0
  17. package/components/data/pagination/fesm2022/index.mjs.map +1 -1
  18. package/components/data/table/bundles/index.umd.js +371 -15
  19. package/components/data/table/esm2022/table-action/table-action.directive.mjs +8 -1
  20. package/components/data/table/esm2022/table-column/table-column.directive.mjs +107 -3
  21. package/components/data/table/esm2022/table-column-group/table-column-group.directive.mjs +9 -1
  22. package/components/data/table/esm2022/table-cud.directive.mjs +8 -2
  23. package/components/data/table/esm2022/table-filter.directive.mjs +12 -2
  24. package/components/data/table/esm2022/table-row/table-row.directive.mjs +8 -1
  25. package/components/data/table/esm2022/table-row-action/table-row-action.directive.mjs +8 -1
  26. package/components/data/table/esm2022/table.component.mjs +219 -12
  27. package/components/data/table/fesm2022/index.mjs +371 -15
  28. package/components/data/table/fesm2022/index.mjs.map +1 -1
  29. package/components/data/table/table-action/table-action.directive.d.ts +3 -2
  30. package/components/data/table/table-column/table-column.directive.d.ts +3 -2
  31. package/components/data/table/table-column-group/table-column-group.directive.d.ts +3 -2
  32. package/components/data/table/table-cud.directive.d.ts +3 -1
  33. package/components/data/table/table-filter.directive.d.ts +3 -1
  34. package/components/data/table/table-row/table-row.directive.d.ts +3 -2
  35. package/components/data/table/table-row-action/table-row-action.directive.d.ts +3 -2
  36. package/components/data/table/table.component.d.ts +6 -2
  37. package/components/navigation/menu/bundles/index.umd.js +5 -0
  38. package/components/navigation/menu/esm2022/menu.component.mjs +6 -1
  39. package/components/navigation/menu/fesm2022/index.mjs +5 -0
  40. package/components/navigation/menu/fesm2022/index.mjs.map +1 -1
  41. package/components/navigation/popover/bundles/index.umd.js +6 -6
  42. package/components/navigation/popover/esm2022/popover.component.mjs +4 -4
  43. package/components/navigation/popover/fesm2022/index.mjs +3 -3
  44. package/components/navigation/popover/fesm2022/index.mjs.map +1 -1
  45. package/components/navigation/popover/popover.component.d.ts +6 -0
  46. package/core/bundles/index.umd.js +411 -82
  47. package/core/esm2022/public_api.mjs +3 -3
  48. package/core/esm2022/utils/utils.mjs +6 -2
  49. package/core/esm2022/utils/watcher.mjs +402 -81
  50. package/core/fesm2022/index.mjs +410 -84
  51. package/core/fesm2022/index.mjs.map +1 -1
  52. package/core/public_api.d.ts +2 -2
  53. package/core/utils/utils.d.ts +1 -0
  54. package/core/utils/watcher.d.ts +28 -5
  55. package/npm-shrinkwrap.json +2 -2
  56. package/package-lock.json +2 -2
  57. package/package.json +1 -1
  58. package/runtime/base/bundles/index.umd.js +22 -2
  59. package/runtime/base/esm2022/components/app-component/app.component.mjs +4 -2
  60. package/runtime/base/esm2022/components/base-page.component.mjs +6 -2
  61. package/runtime/base/esm2022/components/base-partial.component.mjs +7 -2
  62. package/runtime/base/esm2022/components/base-prefab.component.mjs +7 -2
  63. package/runtime/base/esm2022/components/base-spa-page.component.mjs +6 -2
  64. package/runtime/base/esm2022/services/pipe-provider.service.mjs +4 -4
  65. package/runtime/base/fesm2022/index.mjs +23 -3
  66. package/runtime/base/fesm2022/index.mjs.map +1 -1
  67. package/scripts/datatable/datatable.js +101 -15
@@ -2,14 +2,99 @@ import { IDGenerator } from './id-generator';
2
2
  import { $parseExpr } from './expression-parser';
3
3
  import { findValueOf } from './utils';
4
4
  import { clone, flatten, isArray, isEqual, isObject } from "lodash-es";
5
+ // Constants
6
+ const WIDGET_ID_REGEX = /^(widget-[^_]+)/;
7
+ const WIDGET_PROPERTY_REGEX = /^widget-[^_]+_(.+)$/;
8
+ const ARRAY_INDEX_PLACEHOLDER = '[$i]';
9
+ const ARRAY_INDEX_ZERO = '[0]';
10
+ const DEBOUNCE_WAIT = 100;
11
+ const MAX_WATCH_CYCLES = 5;
5
12
  const registry = new Map();
6
13
  const watchIdGenerator = new IDGenerator('watch-id-');
7
- export const FIRST_TIME_WATCH = {};
8
- Object.freeze(FIRST_TIME_WATCH);
9
- export const isFirstTimeChange = v => v === FIRST_TIME_WATCH;
14
+ export const FIRST_TIME_WATCH = Object.freeze({});
15
+ // State
10
16
  let muted = false;
17
+ let changedByWatch = false;
18
+ let skipWatchers = false;
19
+ let ngZone;
11
20
  let appRef;
12
- export const debounce = (fn, wait = 50) => {
21
+ /********************************************************************
22
+ * CLEANUP SCHEDULER WITH:
23
+ * - Significant watcher delta trigger (+/-)
24
+ * - UI stabilization delay (1–2 seconds)
25
+ * - Cooldown to prevent repeated scheduling
26
+ ********************************************************************/
27
+ const CLEANUP_TRIGGER_DELTA = 300; // watcher change threshold
28
+ const CLEANUP_DELAY = 1500; // delay before running cleanup
29
+ const CLEANUP_COOLDOWN = 4000; // prevent re-scheduling for 4s
30
+ let lastWatcherCount = 0;
31
+ let scheduledCleanup = null;
32
+ let lastScheduledTime = 0;
33
+ function getWatcherCount() {
34
+ let count = 0;
35
+ registry.forEach((value) => {
36
+ // Check if it's a grouped structure (widget groups) or a direct watchInfo
37
+ if (value && typeof value === 'object' && !value.fn) {
38
+ // It's a widget group - count the properties (excluding metadata)
39
+ count += Object.keys(value).filter(k => k !== 'scopeType' && k !== 'scopeName').length;
40
+ }
41
+ else {
42
+ // It's a direct watchInfo
43
+ count += 1;
44
+ }
45
+ });
46
+ return count;
47
+ }
48
+ export const cleanupStaleWatchers = () => {
49
+ // console.log(".........Cleaning up stale watchers...registry.size....", registry.size);
50
+ let removed = 0;
51
+ registry.forEach((bucket, widgetId) => {
52
+ if (!document.querySelector(`[widget-id="${widgetId}"]`)) {
53
+ for (const key in bucket) {
54
+ if (bucket.hasOwnProperty(key) && key !== "scopeType" && key !== "scopeName" && typeof bucket[key] !== "function") {
55
+ let watchInfo = bucket[key];
56
+ if (watchInfo && watchInfo.destroyFn) {
57
+ watchInfo.destroyFn();
58
+ }
59
+ }
60
+ }
61
+ removed++;
62
+ registry.delete(widgetId);
63
+ return;
64
+ }
65
+ });
66
+ return removed;
67
+ };
68
+ export const scheduleThresholdCleanup = () => {
69
+ const now = performance.now();
70
+ let lastTriggerPeriod = now - lastScheduledTime;
71
+ // If a cleanup was scheduled recently, skip scheduling
72
+ if (lastTriggerPeriod < CLEANUP_COOLDOWN) {
73
+ return;
74
+ }
75
+ const current = getWatcherCount();
76
+ const delta = Math.abs(current - lastWatcherCount); // significant + or -
77
+ // If change not large enough, skip scheduling
78
+ if (delta <= CLEANUP_TRIGGER_DELTA) {
79
+ lastWatcherCount = current;
80
+ return;
81
+ }
82
+ // Prevent re-scheduling: set timestamp early
83
+ lastScheduledTime = now;
84
+ // Clear previous scheduled cleanup (if any)
85
+ if (scheduledCleanup) {
86
+ clearTimeout(scheduledCleanup);
87
+ }
88
+ // Schedule cleanup after UI stabilizes (delay prevents aggressive cleanup)
89
+ scheduledCleanup = setTimeout(() => {
90
+ cleanupStaleWatchers();
91
+ scheduledCleanup = null;
92
+ }, CLEANUP_DELAY);
93
+ lastWatcherCount = current;
94
+ };
95
+ // Utility functions
96
+ export const isFirstTimeChange = (v) => v === FIRST_TIME_WATCH;
97
+ export const debounce = (fn, wait = DEBOUNCE_WAIT) => {
13
98
  let timeout;
14
99
  return (...args) => {
15
100
  window['__zone_symbol__clearTimeout'](timeout);
@@ -23,115 +108,350 @@ export const unMuteWatchers = () => {
23
108
  muted = false;
24
109
  triggerWatchers();
25
110
  };
111
+ /**
112
+ * Extracts widget ID from identifier (e.g., "widget-id23_eventsource" -> "widget-id23")
113
+ */
114
+ const getWidgetId = (identifier) => {
115
+ if (!identifier || typeof identifier !== 'string') {
116
+ return null;
117
+ }
118
+ const match = identifier.match(WIDGET_ID_REGEX);
119
+ return match ? match[1] : null;
120
+ };
121
+ /**
122
+ * Extracts property name from identifier (e.g., "widget-id23_eventsource" -> "eventsource")
123
+ */
124
+ const getPropertyName = (identifier) => {
125
+ if (!identifier || typeof identifier !== 'string') {
126
+ return identifier;
127
+ }
128
+ const match = identifier.match(WIDGET_PROPERTY_REGEX);
129
+ return match ? match[1] : identifier;
130
+ };
131
+ /**
132
+ * Array consumer wrapper for array-based expressions
133
+ */
26
134
  const arrayConsumer = (listenerFn, restExpr, newVal, oldVal) => {
27
- let data = newVal, formattedData;
28
- if (isArray(data)) {
29
- formattedData = data.map(function (datum) {
30
- return findValueOf(datum, restExpr);
31
- });
32
- // If resulting structure is an array of array, flatten it
33
- if (isArray(formattedData[0])) {
34
- formattedData = flatten(formattedData);
35
- }
36
- listenerFn(formattedData, oldVal);
135
+ if (!isArray(newVal)) {
136
+ return;
37
137
  }
138
+ let formattedData = newVal.map(datum => findValueOf(datum, restExpr));
139
+ // Flatten if result is array of arrays
140
+ if (isArray(formattedData[0])) {
141
+ formattedData = flatten(formattedData);
142
+ }
143
+ listenerFn(formattedData, oldVal);
38
144
  };
39
- const getUpdatedWatcInfo = (expr, acceptsArray, listener) => {
40
- // listener doesn't accept array
41
- // replace all `[$i]` with `[0]` and return the expression
42
- let regex = /\[\$i\]/g, $I = '[$i]', $0 = '[0]';
145
+ /**
146
+ * Updates watch info for array expressions
147
+ */
148
+ const getUpdatedWatchInfo = (expr, acceptsArray, listener) => {
149
+ const regex = /\[\$i\]/g;
43
150
  if (!acceptsArray) {
44
151
  return {
45
- 'expr': expr.replace(regex, $0),
46
- 'listener': listener
152
+ expr: expr.replace(regex, ARRAY_INDEX_ZERO),
153
+ listener
47
154
  };
48
155
  }
49
- // listener accepts array
50
- // replace all except the last `[$i]` with `[0]` and return the expression.
51
- var index = expr.lastIndexOf($I), _expr = expr.substr(0, index).replace($I, $0), restExpr = expr.substr(index + 5), arrayConsumerFn = listener;
52
- if (restExpr) {
53
- arrayConsumerFn = arrayConsumer.bind(undefined, listener, restExpr);
54
- }
156
+ const lastIndex = expr.lastIndexOf(ARRAY_INDEX_PLACEHOLDER);
157
+ const baseExpr = expr.substring(0, lastIndex).replace(ARRAY_INDEX_PLACEHOLDER, ARRAY_INDEX_ZERO);
158
+ const restExpr = expr.substring(lastIndex + 5);
159
+ const arrayConsumerFn = restExpr
160
+ ? arrayConsumer.bind(undefined, listener, restExpr)
161
+ : listener;
55
162
  return {
56
- 'expr': _expr,
57
- 'listener': arrayConsumerFn
163
+ expr: baseExpr,
164
+ listener: arrayConsumerFn
58
165
  };
59
166
  };
167
+ /**
168
+ * Determines if an expression is static (doesn't need to be watched)
169
+ */
170
+ const STATIC_EXPRESSION_NAMES = [
171
+ "row.getProperty('investment')",
172
+ "row.getProperty('factsheetLink')",
173
+ "row.getProperty('isRebalanceEligible')"
174
+ ];
175
+ const isStaticExpression = (expr) => {
176
+ if (typeof expr !== 'string') {
177
+ return false;
178
+ }
179
+ const trimmedExpr = expr.trim();
180
+ // Expressions that always evaluate to localization strings
181
+ // if (trimmedExpr.includes('appLocale')) {
182
+ // return true;
183
+ // }
184
+ // Hard-coded static expression names
185
+ if (STATIC_EXPRESSION_NAMES.includes(trimmedExpr)) {
186
+ return true;
187
+ }
188
+ return false;
189
+ };
190
+ /**
191
+ * Gets the scope type from the scope object
192
+ */
193
+ const getScopeType = ($scope) => {
194
+ if (!$scope) {
195
+ return null;
196
+ }
197
+ if ($scope.pageName)
198
+ return 'Page';
199
+ if ($scope.prefabName)
200
+ return 'Prefab';
201
+ if ($scope.partialName)
202
+ return 'Partial';
203
+ // Check for App scope
204
+ if ($scope.Variables !== undefined &&
205
+ $scope.Actions !== undefined &&
206
+ !$scope.pageName &&
207
+ !$scope.prefabName &&
208
+ !$scope.partialName) {
209
+ return 'App';
210
+ }
211
+ if ($scope.constructor?.name === 'AppRef') {
212
+ return 'App';
213
+ }
214
+ return null;
215
+ };
216
+ /**
217
+ * Gets scope name based on scope type
218
+ */
219
+ const getScopeName = ($scope, scopeType) => {
220
+ if (!scopeType || !$scope) {
221
+ return null;
222
+ }
223
+ switch (scopeType) {
224
+ case 'Prefab': return $scope.prefabName || null;
225
+ case 'Partial': return $scope.partialName || null;
226
+ case 'Page': return $scope.pageName || null;
227
+ default: return null;
228
+ }
229
+ };
230
+ /**
231
+ * Main watch function
232
+ */
60
233
  export const $watch = (expr, $scope, $locals, listener, identifier = watchIdGenerator.nextUid(), doNotClone = false, config = {}, isMuted) => {
61
- if (expr.indexOf('[$i]') !== -1) {
62
- let watchInfo = getUpdatedWatcInfo(expr, config && (config.arrayType || config.isList), listener);
234
+ // Handle array expressions
235
+ if (expr.includes(ARRAY_INDEX_PLACEHOLDER)) {
236
+ const watchInfo = getUpdatedWatchInfo(expr, config.arrayType || config.isList || false, listener);
63
237
  expr = watchInfo.expr;
64
238
  listener = watchInfo.listener;
65
239
  }
240
+ // Handle static expressions
241
+ if (isStaticExpression(expr)) {
242
+ try {
243
+ const fn = $parseExpr(expr);
244
+ const staticValue = fn($scope, $locals);
245
+ listener(staticValue, FIRST_TIME_WATCH);
246
+ }
247
+ catch (e) {
248
+ console.warn(`Error evaluating static expression '${expr}':`, e);
249
+ listener(undefined, FIRST_TIME_WATCH);
250
+ }
251
+ return () => { }; // No-op unsubscribe
252
+ }
66
253
  const fn = $parseExpr(expr);
67
- registry.set(identifier, {
68
- fn: fn.bind(expr, $scope, $locals),
254
+ const scopeType = getScopeType($scope);
255
+ const scopeName = getScopeName($scope, scopeType);
256
+ const destroyFn = () => $unwatch(identifier);
257
+ const watchInfo = {
258
+ fn: fn.bind(null, $scope, $locals),
69
259
  listener,
70
260
  expr,
71
261
  last: FIRST_TIME_WATCH,
72
262
  doNotClone,
73
- isMuted: isMuted
263
+ isMuted,
264
+ scopeType,
265
+ scopeName,
266
+ destroyFn
267
+ };
268
+ // Store in registry
269
+ const widgetId = getWidgetId(identifier);
270
+ if (widgetId) {
271
+ const propertyName = getPropertyName(identifier);
272
+ if (!registry.has(widgetId)) {
273
+ registry.set(widgetId, {});
274
+ }
275
+ const widgetGroup = registry.get(widgetId);
276
+ widgetGroup[propertyName] = watchInfo;
277
+ }
278
+ else {
279
+ registry.set(identifier, watchInfo);
280
+ }
281
+ return destroyFn;
282
+ };
283
+ /**
284
+ * Unwatches a single identifier
285
+ */
286
+ export const $unwatch = (identifier) => {
287
+ const widgetId = getWidgetId(identifier);
288
+ if (widgetId) {
289
+ const propertyName = getPropertyName(identifier);
290
+ const widgetGroup = registry.get(widgetId);
291
+ //@ts-ignore
292
+ if (widgetGroup && typeof widgetGroup === 'object' && !widgetGroup.fn) {
293
+ const watchInfo = widgetGroup[propertyName];
294
+ if (watchInfo) {
295
+ delete widgetGroup[propertyName];
296
+ // Clean up empty widget groups
297
+ if (Object.keys(widgetGroup).length === 0) {
298
+ registry.delete(widgetId);
299
+ }
300
+ return true;
301
+ }
302
+ }
303
+ }
304
+ // Fallback to direct lookup
305
+ if (registry.has(identifier)) {
306
+ registry.delete(identifier);
307
+ return true;
308
+ }
309
+ return false;
310
+ };
311
+ /**
312
+ * Unwatches all watchers for a specific widget ID
313
+ */
314
+ export const $unwatchAll = (widgetId) => {
315
+ if (!widgetId || typeof widgetId !== 'string') {
316
+ return 0;
317
+ }
318
+ const widgetGroup = registry.get(widgetId);
319
+ if (widgetGroup && typeof widgetGroup === 'object' && !widgetGroup.fn) {
320
+ const count = Object.keys(widgetGroup).length;
321
+ registry.delete(widgetId);
322
+ return count;
323
+ }
324
+ // Fallback: find all identifiers starting with this widget ID
325
+ let removedCount = 0;
326
+ const identifiersToRemove = [];
327
+ registry.forEach((_, key) => {
328
+ if (key.startsWith(widgetId + '_')) {
329
+ identifiersToRemove.push(key);
330
+ }
331
+ });
332
+ identifiersToRemove.forEach(identifier => {
333
+ if ($unwatch(identifier)) {
334
+ removedCount++;
335
+ }
74
336
  });
75
- return () => $unwatch(identifier);
337
+ return removedCount;
76
338
  };
77
- export const $unwatch = identifier => registry.delete(identifier);
78
- let changedByWatch = false;
79
- const $RAF = window.requestAnimationFrame;
80
- let ngZone;
81
- const triggerWatchers = (ignoreMuted) => {
339
+ /**
340
+ * Unwatches all watchers for a specific scope (Page, Prefab, Partial, or App)
341
+ * Now works directly with the main registry instead of separate scoped registries
342
+ */
343
+ export const $unwatchAllByScope = (scopeType, scopeName) => {
344
+ if (!scopeType) {
345
+ return 0;
346
+ }
347
+ let removedCount = 0;
348
+ const identifiersToRemove = [];
349
+ registry.forEach((value, key) => {
350
+ // Handle grouped structure (widget groups)
351
+ if (value && typeof value === 'object' && !value.fn) {
352
+ Object.entries(value).forEach(([propertyName, watchInfo]) => {
353
+ if (watchInfo?.scopeType === scopeType &&
354
+ (!scopeName || watchInfo.scopeName === scopeName)) {
355
+ identifiersToRemove.push(`${key}_${propertyName}`);
356
+ }
357
+ });
358
+ }
359
+ else {
360
+ // Direct watchInfo
361
+ const watchInfo = value;
362
+ if (watchInfo?.scopeType === scopeType &&
363
+ (!scopeName || watchInfo.scopeName === scopeName)) {
364
+ identifiersToRemove.push(key);
365
+ }
366
+ }
367
+ });
368
+ // Unwatch all collected identifiers
369
+ identifiersToRemove.forEach(identifier => {
370
+ if ($unwatch(identifier)) {
371
+ removedCount++;
372
+ }
373
+ });
374
+ return removedCount;
375
+ };
376
+ /**
377
+ * Processes a single watch info during trigger cycle
378
+ */
379
+ const processWatchInfo = (watchInfo) => {
380
+ if (!watchInfo?.fn || (watchInfo.isMuted?.() ?? false)) {
381
+ return false;
382
+ }
383
+ let newValue;
384
+ try {
385
+ newValue = watchInfo.fn();
386
+ }
387
+ catch (e) {
388
+ console.warn(`Error executing expression: '${watchInfo.expr}'`, e);
389
+ return false;
390
+ }
391
+ if (isEqual(newValue, watchInfo.last)) {
392
+ return false;
393
+ }
394
+ // Change detected
395
+ changedByWatch = true;
396
+ watchInfo.last = isObject(newValue) && !watchInfo.doNotClone
397
+ ? clone(newValue)
398
+ : newValue;
399
+ watchInfo.listener(newValue, watchInfo.last === newValue ? FIRST_TIME_WATCH : watchInfo.last);
400
+ changedByWatch = false;
401
+ return true;
402
+ };
403
+ /**
404
+ * Triggers all watchers
405
+ */
406
+ const triggerWatchers = (ignoreMuted = false) => {
82
407
  if (muted && !ignoreMuted) {
83
408
  return;
84
409
  }
85
- const limit = 5;
86
410
  let pass = 1;
87
411
  let changeDetected;
88
412
  do {
89
413
  changeDetected = false;
90
- registry.forEach(watchInfo => {
91
- if (watchInfo.isMuted && watchInfo.isMuted()) {
92
- return;
93
- }
94
- const fn = watchInfo.fn;
95
- const listener = watchInfo.listener;
96
- const ov = watchInfo.last;
97
- let nv;
98
- try {
99
- nv = fn();
100
- }
101
- catch (e) {
102
- console.warn(`error in executing expression: '${watchInfo.expr}'`);
414
+ registry.forEach((value) => {
415
+ // Handle grouped structure
416
+ if (value && typeof value === 'object' && !value.fn) {
417
+ Object.values(value).forEach((watchInfo) => {
418
+ if (processWatchInfo(watchInfo)) {
419
+ changeDetected = true;
420
+ }
421
+ });
103
422
  }
104
- if (!isEqual(nv, ov)) {
105
- changeDetected = true;
106
- changedByWatch = true;
107
- watchInfo.last = nv;
108
- // @ts-ignore
109
- if (isObject(nv) && !watchInfo.doNotClone && nv.__cloneable__ !== false) {
110
- watchInfo.last = clone(nv);
423
+ else {
424
+ // Direct watchInfo
425
+ if (processWatchInfo(value)) {
426
+ changeDetected = true;
111
427
  }
112
- listener(nv, ov);
113
- resetChangeFromWatch();
114
428
  }
115
429
  });
116
430
  pass++;
117
- } while (changeDetected && pass < limit);
118
- if (changeDetected && pass === limit) {
119
- console.warn(`Number of watch cycles gone above set limit of: ${limit} `);
431
+ } while (changeDetected && pass < MAX_WATCH_CYCLES);
432
+ // Schedule cleanup after watchers are triggered
433
+ scheduleThresholdCleanup();
434
+ if (changeDetected && pass === MAX_WATCH_CYCLES) {
435
+ console.warn(`Watch cycles exceeded limit of ${MAX_WATCH_CYCLES}`);
120
436
  }
121
437
  };
122
- export const setNgZone = zone => ngZone = zone;
123
- export const setAppRef = ref => {
438
+ // Angular zone integration
439
+ export const setNgZone = (zone) => {
440
+ ngZone = zone;
441
+ };
442
+ export const setAppRef = (ref) => {
124
443
  appRef = ref;
125
444
  };
126
445
  export const isChangeFromWatch = () => changedByWatch;
127
- export const resetChangeFromWatch = () => changedByWatch = false;
128
- window.watchRegistry = registry;
129
- let skipWatchers;
446
+ export const resetChangeFromWatch = () => {
447
+ changedByWatch = false;
448
+ };
449
+ // Debounced trigger
130
450
  const debouncedTriggerWatchers = debounce(() => {
131
451
  skipWatchers = true;
132
452
  ngZone.run(() => triggerWatchers());
133
- }, 100);
134
- export const $invokeWatchers = (force, ignoreMuted) => {
453
+ }, DEBOUNCE_WAIT);
454
+ export const $invokeWatchers = (force = false, ignoreMuted = false) => {
135
455
  if (force) {
136
456
  triggerWatchers(ignoreMuted);
137
457
  }
@@ -145,7 +465,8 @@ export const $invokeWatchers = (force, ignoreMuted) => {
145
465
  };
146
466
  export const $appDigest = (() => {
147
467
  let queued = false;
148
- return (force) => {
468
+ const $RAF = window.requestAnimationFrame;
469
+ return (force = false) => {
149
470
  if (!appRef) {
150
471
  return;
151
472
  }
@@ -157,14 +478,14 @@ export const $appDigest = (() => {
157
478
  if (queued) {
158
479
  return;
159
480
  }
160
- else {
161
- queued = true;
162
- $RAF(() => {
163
- ngZone.run(() => appRef.tick());
164
- queued = false;
165
- });
166
- }
481
+ queued = true;
482
+ $RAF(() => {
483
+ ngZone.run(() => appRef.tick());
484
+ queued = false;
485
+ });
167
486
  }
168
487
  };
169
488
  })();
170
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2F0Y2hlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NvcmUvc3JjL3V0aWxzL3dhdGNoZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBRTdDLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUNqRCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBQ3RDLE9BQU8sRUFBQyxLQUFLLEVBQUUsT0FBTyxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsUUFBUSxFQUFDLE1BQU0sV0FBVyxDQUFDO0FBRXJFLE1BQU0sUUFBUSxHQUFHLElBQUksR0FBRyxFQUFlLENBQUM7QUFFeEMsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLFdBQVcsQ0FBQyxXQUFXLENBQUMsQ0FBQztBQUV0RCxNQUFNLENBQUMsTUFBTSxnQkFBZ0IsR0FBRyxFQUFFLENBQUM7QUFFbkMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO0FBRWhDLE1BQU0sQ0FBQyxNQUFNLGlCQUFpQixHQUFHLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxLQUFLLGdCQUFnQixDQUFDO0FBRTdELElBQUksS0FBSyxHQUFHLEtBQUssQ0FBQztBQUNsQixJQUFJLE1BQU0sQ0FBQztBQUVYLE1BQU0sQ0FBQyxNQUFNLFFBQVEsR0FBRyxDQUFDLEVBQVksRUFBRSxPQUFlLEVBQUUsRUFBRSxFQUFFO0lBQ3hELElBQUksT0FBTyxDQUFDO0lBQ1osT0FBTyxDQUFDLEdBQUcsSUFBSSxFQUFFLEVBQUU7UUFDZixNQUFNLENBQUMsNkJBQTZCLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUMvQyxPQUFPLEdBQUcsTUFBTSxDQUFDLDJCQUEyQixDQUFDLENBQUMsR0FBRyxFQUFFLENBQUMsRUFBRSxDQUFDLEdBQUcsSUFBSSxDQUFDLEVB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QUNELElBQUksS0FBSyxFQUFFLENBQUM7WUFDUixNQUFNLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO1lBQ2hDLE1BQU0sR0FBRyxLQUFLLENBQUM7UUFDbkIsQ0FBQzthQUFNLENBQUM7WUFDSixJQUFJLE1BQU0sRUFBRSxDQUFDO2dCQUNULE9BQU87WUFDWCxDQUFDO2lCQUFNLENBQUM7Z0JBQ0osTUFBTSxHQUFHLElBQUksQ0FBQztnQkFDZCxJQUFJLENBQUMsR0FBRyxFQUFFO29CQUNOLE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7b0JBQ2hDLE1BQU0sR0FBRyxLQUFLLENBQUM7Z0JBQ25CLENBQUMsQ0FBQyxDQUFDO1lBQ1AsQ0FBQztRQUNMLENBQUM7SUFDTCxDQUFDLENBQUM7QUFDTixDQUFDLENBQUMsRUFBRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSURHZW5lcmF0b3IgfSBmcm9tICcuL2lkLWdlbmVyYXRvcic7XG5cbmltcG9ydCB7ICRwYXJzZUV4cHIgfSBmcm9tICcuL2V4cHJlc3Npb24tcGFyc2VyJztcbmltcG9ydCB7IGZpbmRWYWx1ZU9mIH0gZnJvbSAnLi91dGlscyc7XG5pbXBvcnQge2Nsb25lLCBmbGF0dGVuLCBpc0FycmF5LCBpc0VxdWFsLCBpc09iamVjdH0gZnJvbSBcImxvZGFzaC1lc1wiO1xuXG5jb25zdCByZWdpc3RyeSA9IG5ldyBNYXA8c3RyaW5nLCBhbnk+KCk7XG5cbmNvbnN0IHdhdGNoSWRHZW5lcmF0b3IgPSBuZXcgSURHZW5lcmF0b3IoJ3dhdGNoLWlkLScpO1xuXG5leHBvcnQgY29uc3QgRklSU1RfVElNRV9XQVRDSCA9IHt9O1xuXG5PYmplY3QuZnJlZXplKEZJUlNUX1RJTUVfV0FUQ0gpO1xuXG5leHBvcnQgY29uc3QgaXNGaXJzdFRpbWVDaGFuZ2UgPSB2ID0+IHYgPT09IEZJUlNUX1RJTUVfV0FUQ0g7XG5cbmxldCBtdXRlZCA9IGZhbHNlO1xubGV0IGFwcFJlZjtcblxuZXhwb3J0IGNvbnN0IGRlYm91bmNlID0gKGZuOiBGdW5jdGlvbiwgd2FpdDogbnVtYmVyID0gNTApID0+IHtcbiAgICBsZXQgdGltZW91dDtcbiAgICByZXR1cm4gKC4uLmFyZ3MpID0+IHtcbiAgICAgICAgd2luZG93WydfX3pvbmVfc3ltYm9sX19jbGVhclRpbWVvdXQnXSh0aW1lb3V0KTtcbiAgICAgICAgdGltZW91dCA9IHdpbmRvd1snX196b25lX3N5bWJvbF9fc2V0VGltZW91dCddKCgpID0+IGZuKC4uLmFyZ3MpLCB3YWl0KTtcbiAgICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IG11dGVXYXRjaGVycyA9ICgpID0+IHtcbiAgICBtdXRlZCA9IHRydWU7XG59O1xuXG5leHBvcnQgY29uc3QgdW5NdXRlV2F0Y2hlcnMgPSAoKSA9PiB7XG4gICAgbXV0ZWQgPSBmYWxzZTtcbiAgICB0cmlnZ2VyV2F0Y2hlcnMoKTtcbn07XG5cbmNvbnN0IGFycmF5Q29uc3VtZXIgPSAobGlzdGVuZXJGbiwgcmVzdEV4cHIsIG5ld1ZhbCwgb2xkVmFsKSA9PiB7XG4gICAgbGV0IGRhdGEgPSBuZXdWYWwsXG4gICAgICAgIGZvcm1hdHRlZERhdGE7XG5cbiAgICBpZiAoaXNBcnJheShkYXRhKSkge1xuICAgICAgICBmb3JtYXR0ZWREYXRhID0gZGF0YS5tYXAoZnVuY3Rpb24gKGRhdHVtKSB7XG4gICAgICAgICAgICByZXR1cm4gZmluZFZhbHVlT2YoZGF0dW0sIHJlc3RFeHByKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gSWYgcmVzdWx0aW5nIHN0cnVjdHVyZSBpcyBhbiBhcnJheSBvZiBhcnJheSwgZmxhdHRlbiBpdFxuICAgICAgICBpZiAoaXNBcnJheShmb3JtYXR0ZWREYXRhWzBdKSkge1xuICAgICAgICAgICAgZm9ybWF0dGVkRGF0YSA9IGZsYXR0ZW4oZm9ybWF0dGVkRGF0YSk7XG4gICAgICAgIH1cblxuICAgICAgICBsaXN0ZW5lckZuKGZvcm1hdHRlZERhdGEsIG9sZFZhbCk7XG4gICAgfVxufTtcblxuY29uc3QgZ2V0VXBkYXRlZFdhdGNJbmZvID0gKGV4cHIsIGFjY2VwdHNBcnJheSwgbGlzdGVuZXIpID0+IHtcbiAgICAvLyBsaXN0ZW5lciBkb2Vzbid0IGFjY2VwdCBhcnJheVxuICAgIC8vIHJlcGxhY2UgYWxsIGBbJGldYCB3aXRoIGBbMF1gIGFuZCByZXR1cm4gdGhlIGV4cHJlc3Npb25cbiAgICBsZXQgcmVnZXggPSAvXFxbXFwkaVxcXS9nLFxuICAgICAgICAkSSAgICAgICAgICAgICAgICA9ICdbJGldJyxcbiAgICAgICAgJDAgICAgICAgICAgICAgICAgPSAnWzBdJztcbiAgICBpZiAoIWFjY2VwdHNBcnJheSkge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgJ2V4cHInICAgICA6IGV4cHIucmVwbGFjZShyZWdleCwgJDApLFxuICAgICAgICAgICAgJ2xpc3RlbmVyJyA6IGxpc3RlbmVyXG4gICAgICAgIH07XG4gICAgfVxuXG4gICAgLy8gbGlzdGVuZXIgYWNjZXB0cyBhcnJheVxuICAgIC8vIHJlcGxhY2UgYWxsIGV4Y2VwdCB0aGUgbGFzdCBgWyRpXWAgd2l0aCBgWzBdYCBhbmQgcmV0dXJuIHRoZSBleHByZXNzaW9uLlxuICAgIHZhciBpbmRleCAgICAgICAgICAgPSBleHByLmxhc3RJbmRleE9mKCRJKSxcbiAgICAgICAgX2V4cHIgICAgICAgICAgID0gZXhwci5zdWJzdHIoMCwgaW5kZXgpLnJlcGxhY2UoJEksICQwKSxcbiAgICAgICAgcmVzdEV4cHIgICAgICAgID0gZXhwci5zdWJzdHIoaW5kZXggKyA1KSxcbiAgICAgICAgYXJyYXlDb25zdW1lckZuID0gbGlzdGVuZXI7XG5cbiAgICBpZiAocmVzdEV4cHIpIHtcbiAgICAgICAgYXJyYXlDb25zdW1lckZuID0gYXJyYXlDb25zdW1lci5iaW5kKHVuZGVmaW5lZCwgbGlzdGVuZXIsIHJlc3RFeHByKTtcbiAgICB9XG5cbiAgICByZXR1cm4ge1xuICAgICAgICAnZXhwcicgICAgIDogX2V4cHIsXG4gICAgICAgICdsaXN0ZW5lcicgOiBhcnJheUNvbnN1bWVyRm5cbiAgICB9O1xufTtcblxuZXhwb3J0IGNvbnN0ICR3YXRjaCA9IChleHByLCAkc2NvcGUsICRsb2NhbHMsIGxpc3RlbmVyLCBpZGVudGlmaWVyID0gd2F0Y2hJZEdlbmVyYXRvci5uZXh0VWlkKCksIGRvTm90Q2xvbmUgPSBmYWxzZSwgY29uZmlnOmFueT17fSwgaXNNdXRlZD86ICgpID0+IGJvb2xlYW4pID0+IHtcbiAgICBpZiAoZXhwci5pbmRleE9mKCdbJGldJykgIT09IC0xKSB7XG4gICAgICAgIGxldCB3YXRjaEluZm8gPSBnZXRVcGRhdGVkV2F0Y0luZm8oZXhwciwgY29uZmlnICYmIChjb25maWcuYXJyYXlUeXBlIHx8IGNvbmZpZy5pc0xpc3QpLCBsaXN0ZW5lcik7XG4gICAgICAgIGV4cHIgPSB3YXRjaEluZm8uZXhwcjtcbiAgICAgICAgbGlzdGVuZXIgPSB3YXRjaEluZm8ubGlzdGVuZXI7XG4gICAgfVxuICAgIGNvbnN0IGZuID0gJHBhcnNlRXhwcihleHByKTtcblxuICAgIHJlZ2lzdHJ5LnNldChpZGVudGlmaWVyLCB7XG4gICAgICAgIGZuOiBmbi5iaW5kKGV4cHIsICRzY29wZSwgJGxvY2FscyksXG4gICAgICAgIGxpc3RlbmVyLFxuICAgICAgICBleHByLFxuICAgICAgICBsYXN0OiBGSVJTVF9USU1FX1dBVENILFxuICAgICAgICBkb05vdENsb25lLFxuICAgICAgICBpc011dGVkOiBpc011dGVkXG4gICAgfSk7XG5cbiAgICByZXR1cm4gKCkgPT4gJHVud2F0Y2goaWRlbnRpZmllcik7XG59O1xuXG5leHBvcnQgY29uc3QgJHVud2F0Y2ggPSBpZGVudGlmaWVyID0+IHJlZ2lzdHJ5LmRlbGV0ZShpZGVudGlmaWVyKTtcblxubGV0IGNoYW5nZWRCeVdhdGNoID0gZmFsc2U7XG5cbmNvbnN0ICRSQUYgPSB3aW5kb3cucmVxdWVzdEFuaW1hdGlvbkZyYW1lO1xuXG5sZXQgbmdab25lO1xuXG5jb25zdCB0cmlnZ2VyV2F0Y2hlcnMgPSAoaWdub3JlTXV0ZWQ/OiBib29sZWFuKSA9PiB7XG5cbiAgICBpZiAobXV0ZWQgJiYgIWlnbm9yZU11dGVkKSB7XG4gICAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCBsaW1pdCA9IDU7XG4gICAgbGV0IHBhc3MgPSAxO1xuICAgIGxldCBjaGFuZ2VEZXRlY3RlZDtcblxuICAgIGRvIHtcbiAgICAgICAgY2hhbmdlRGV0ZWN0ZWQgPSBmYWxzZTtcbiAgICAgICAgcmVnaXN0cnkuZm9yRWFjaCh3YXRjaEluZm8gPT4ge1xuICAgICAgICAgICAgaWYod2F0Y2hJbmZvLmlzTXV0ZWQgJiYgd2F0Y2hJbmZvLmlzTXV0ZWQoKSkge1xuICAgICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGNvbnN0IGZuID0gd2F0Y2hJbmZvLmZuO1xuICAgICAgICAgICAgY29uc3QgbGlzdGVuZXIgPSB3YXRjaEluZm8ubGlzdGVuZXI7XG4gICAgICAgICAgICBjb25zdCBvdiA9IHdhdGNoSW5mby5sYXN0O1xuICAgICAgICAgICAgbGV0IG52O1xuXG4gICAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgICAgIG52ID0gZm4oKTtcbiAgICAgICAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgICAgICAgICBjb25zb2xlLndhcm4oYGVycm9yIGluIGV4ZWN1dGluZyBleHByZXNzaW9uOiAnJHt3YXRjaEluZm8uZXhwcn0nYCk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmICghaXNFcXVhbChudiwgb3YpKSB7XG4gICAgICAgICAgICAgICAgY2hhbmdlRGV0ZWN0ZWQgPSB0cnVlO1xuICAgICAgICAgICAgICAgIGNoYW5nZWRCeVdhdGNoID0gdHJ1ZTtcbiAgICAgICAgICAgICAgICB3YXRjaEluZm8ubGFzdCA9IG52O1xuXG4gICAgICAgICAgICAgICAgLy8gQHRzLWlnbm9yZVxuICAgICAgICAgICAgICAgIGlmIChpc09iamVjdChudikgJiYgIXdhdGNoSW5mby5kb05vdENsb25lICYmIG52Ll9fY2xvbmVhYmxlX18gIT09IGZhbHNlKSB7XG4gICAgICAgICAgICAgICAgICAgIHdhdGNoSW5mby5sYXN0ID0gY2xvbmUobnYpO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBsaXN0ZW5lcihudiwgb3YpO1xuICAgICAgICAgICAgICAgIHJlc2V0Q2hhbmdlRnJvbVdhdGNoKCk7XG4gICAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgICAgICBwYXNzKys7XG5cbiAgICB9IHdoaWxlIChjaGFuZ2VEZXRlY3RlZCAmJiBwYXNzIDwgbGltaXQpO1xuXG4gICAgaWYgKGNoYW5nZURldGVjdGVkICYmIHBhc3MgPT09IGxpbWl0KSB7XG4gICAgICAgIGNvbnNvbGUud2FybihgTnVtYmVyIG9mIHdhdGNoIGN5Y2xlcyBnb25lIGFib3ZlIHNldCBsaW1pdCBvZjogJHtsaW1pdH0gYCk7XG4gICAgfVxufTtcblxuZXhwb3J0IGNvbnN0IHNldE5nWm9uZSA9IHpvbmUgPT4gbmdab25lID0gem9uZTtcblxuZXhwb3J0IGNvbnN0IHNldEFwcFJlZiA9IHJlZiA9PiB7XG4gICAgYXBwUmVmID0gcmVmXG59O1xuXG5leHBvcnQgY29uc3QgaXNDaGFuZ2VGcm9tV2F0Y2ggPSAoKSA9PiBjaGFuZ2VkQnlXYXRjaDtcbmV4cG9ydCBjb25zdCByZXNldENoYW5nZUZyb21XYXRjaCA9ICgpID0+IGNoYW5nZWRCeVdhdGNoID0gZmFsc2U7XG5cbig8YW55PndpbmRvdykud2F0Y2hSZWdpc3RyeSA9IHJlZ2lzdHJ5O1xuXG5sZXQgc2tpcFdhdGNoZXJzO1xuXG5jb25zdCBkZWJvdW5jZWRUcmlnZ2VyV2F0Y2hlcnMgPSBkZWJvdW5jZSgoKSA9PiB7XG4gICAgc2tpcFdhdGNoZXJzID0gdHJ1ZTtcbiAgICBuZ1pvbmUucnVuKCgpID0+IHRyaWdnZXJXYXRjaGVycygpKTtcbn0sIDEwMCk7XG5cbmV4cG9ydCBjb25zdCAkaW52b2tlV2F0Y2hlcnMgPSAoZm9yY2U/OiBib29sZWFuLCBpZ25vcmVNdXRlZD86IGJvb2xlYW4pID0+IHtcbiAgICBpZiAoZm9yY2UpIHtcbiAgICAgICAgdHJpZ2dlcldhdGNoZXJzKGlnbm9yZU11dGVkKTtcbiAgICB9IGVsc2Uge1xuXG4gICAgICAgIGlmIChza2lwV2F0Y2hlcnMpIHtcbiAgICAgICAgICAgIHNraXBXYXRjaGVycyA9IGZhbHNlO1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG4gICAgICAgIGRlYm91bmNlZFRyaWdnZXJXYXRjaGVycygpO1xuICAgIH1cbn07XG5cbmV4cG9ydCBjb25zdCAkYXBwRGlnZXN0ID0gKCgpID0+IHtcbiAgICBsZXQgcXVldWVkID0gZmFsc2U7XG4gICAgcmV0dXJuIChmb3JjZT86IGJvb2xlYW4pID0+IHtcbiAgICAgICAgaWYgKCFhcHBSZWYpIHtcbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuICAgICAgICBpZiAoZm9yY2UpIHtcbiAgICAgICAgICAgIG5nWm9uZS5ydW4oKCkgPT4gYXBwUmVmLnRpY2soKSk7XG4gICAgICAgICAgICBxdWV1ZWQgPSBmYWxzZTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGlmIChxdWV1ZWQpIHtcbiAgICAgICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIHF1ZXVlZCA9IHRydWU7XG4gICAgICAgICAgICAgICAgJFJBRigoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgIG5nWm9uZS5ydW4oKCkgPT4gYXBwUmVmLnRpY2soKSk7XG4gICAgICAgICAgICAgICAgICAgIHF1ZXVlZCA9IGZhbHNlO1xuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgfTtcbn0pKCk7XG4iXX0=
489
+ // Export registry for debugging
490
+ window.watchRegistry = registry;
491
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2F0Y2hlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NvcmUvc3JjL3V0aWxzL3dhdGNoZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzdDLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUNqRCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBQ3RDLE9BQU8sRUFBRSxLQUFLLEVBQUUsT0FBTyxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBRXZFLFlBQVk7QUFDWixNQUFNLGVBQWUsR0FBRyxpQkFBaUIsQ0FBQztBQUMxQyxNQUFNLHFCQUFxQixHQUFHLHFCQUFxQixDQUFDO0FBQ3BELE1BQU0sdUJBQXVCLEdBQUcsTUFBTSxDQUFDO0FBQ3ZDLE1BQU0sZ0JBQWdCLEdBQUcsS0FBSyxDQUFDO0FBQy9CLE1BQU0sYUFBYSxHQUFHLEdBQUcsQ0FBQztBQUMxQixNQUFNLGdCQUFnQixHQUFHLENBQUMsQ0FBQztBQUUzQixNQUFNLFFBQVEsR0FBRyxJQUFJLEdBQUcsRUFBZSxDQUFDO0FBQ3hDLE1BQU0sZ0JBQWdCLEdBQUcsSUFBSSxXQUFXLENBQUMsV0FBVyxDQUFDLENBQUM7QUFFdEQsTUFBTSxDQUFDLE1BQU0sZ0JBQWdCLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUMsQ0FBQztBQXNCbEQsUUFBUTtBQUNSLElBQUksS0FBSyxHQUFHLEtBQUssQ0FBQztBQUNsQixJQUFJLGNBQWMsR0FBRyxLQUFLLENBQUM7QUFDM0IsSUFBSSxZQUFZLEdBQUcsS0FBSyxDQUFDO0FBQ3pCLElBQUksTUFBVyxDQUFDO0FBQ2hCLElBQUksTUFBVyxDQUFDO0FBRWhCOzs7OztzRUFLc0U7QUFDdEUsTUFBTSxxQkFBcUIsR0FBRyxHQUFH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Q3JCLE9BQU87UUFDWCxDQUFDO1FBQ0Qsd0JBQXdCLEVBQUUsQ0FBQztJQUMvQixDQUFDO0FBQ0wsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sVUFBVSxHQUFHLENBQUMsR0FBRyxFQUFFO0lBQzVCLElBQUksTUFBTSxHQUFHLEtBQUssQ0FBQztJQUNuQixNQUFNLElBQUksR0FBRyxNQUFNLENBQUMscUJBQXFCLENBQUM7SUFFMUMsT0FBTyxDQUFDLFFBQWlCLEtBQUssRUFBUSxFQUFFO1FBQ3BDLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNWLE9BQU87UUFDWCxDQUFDO1FBRUQsSUFBSSxLQUFLLEVBQUUsQ0FBQztZQUNSLE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7WUFDaEMsTUFBTSxHQUFHLEtBQUssQ0FBQztRQUNuQixDQUFDO2FBQU0sQ0FBQztZQUNKLElBQUksTUFBTSxFQUFFLENBQUM7Z0JBQ1QsT0FBTztZQUNYLENBQUM7WUFDRCxNQUFNLEdBQUcsSUFBSSxDQUFDO1lBQ2QsSUFBSSxDQUFDLEdBQUcsRUFBRTtnQkFDTixNQUFNLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO2dCQUNoQyxNQUFNLEdBQUcsS0FBSyxDQUFDO1lBQ25CLENBQUMsQ0FBQyxDQUFDO1FBQ1AsQ0FBQztJQUNMLENBQUMsQ0FBQztBQUNOLENBQUMsQ0FBQyxFQUFFLENBQUM7QUFFTCxnQ0FBZ0M7QUFDL0IsTUFBYyxDQUFDLGFBQWEsR0FBRyxRQUFRLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJREdlbmVyYXRvciB9IGZyb20gJy4vaWQtZ2VuZXJhdG9yJztcbmltcG9ydCB7ICRwYXJzZUV4cHIgfSBmcm9tICcuL2V4cHJlc3Npb24tcGFyc2VyJztcbmltcG9ydCB7IGZpbmRWYWx1ZU9mIH0gZnJvbSAnLi91dGlscyc7XG5pbXBvcnQgeyBjbG9uZSwgZmxhdHRlbiwgaXNBcnJheSwgaXNFcXVhbCwgaXNPYmplY3QgfSBmcm9tIFwibG9kYXNoLWVzXCI7XG5cbi8vIENvbnN0YW50c1xuY29uc3QgV0lER0VUX0lEX1JFR0VYID0gL14od2lkZ2V0LVteX10rKS87XG5jb25zdCBXSURHRVRfUFJPUEVSVFlfUkVHRVggPSAvXndpZGdldC1bXl9dK18oLispJC87XG5jb25zdCBBUlJBWV9JTkRFWF9QTEFDRUhPTERFUiA9ICdbJGldJztcbmNvbnN0IEFSUkFZX0lOREVYX1pFUk8gPSAnWzBdJztcbmNvbnN0IERFQk9VTkNFX1dBSVQgPSAxMDA7XG5jb25zdCBNQVhfV0FUQ0hfQ1lDTEVTID0gNTtcblxuY29uc3QgcmVnaXN0cnkgPSBuZXcgTWFwPHN0cmluZywgYW55PigpO1xuY29uc3Qgd2F0Y2hJZEdlbmVyYXRvciA9IG5ldyBJREdlbmVyYXRvcignd2F0Y2gtaWQtJyk7XG5cbmV4cG9ydCBjb25zdCBGSVJTVF9USU1FX1dBVENIID0gT2JqZWN0LmZyZWV6ZSh7fSk7XG5cbi8vIFR5cGUgZGVmaW5pdGlvbnNcbmludGVyZmFjZSBXYXRjaEluZm8ge1xuICAgIGZuOiAoKSA9PiBhbnk7XG4gICAgbGlzdGVuZXI6IChuZXdWYWw6IGFueSwgb2xkVmFsOiBhbnkpID0+IHZvaWQ7XG4gICAgZXhwcjogc3RyaW5nO1xuICAgIGxhc3Q6IGFueTtcbiAgICBkb05vdENsb25lOiBib29sZWFuO1xuICAgIGlzTXV0ZWQ/OiAoKSA9PiBib29sZWFuO1xuICAgIHNjb3BlVHlwZTogU2NvcGVUeXBlIHwgbnVsbDtcbiAgICBzY29wZU5hbWU6IHN0cmluZyB8IG51bGw7XG4gICAgZGVzdHJveUZuPzogKCkgPT4gdm9pZDtcbn1cblxudHlwZSBTY29wZVR5cGUgPSAnUGFnZScgfCAnUHJlZmFiJyB8ICdQYXJ0aWFsJyB8ICdBcHAnO1xuXG5pbnRlcmZhY2UgV2F0Y2hDb25maWcge1xuICAgIGFycmF5VHlwZT86IGJvb2xlYW47XG4gICAgaXNMaXN0PzogYm9vbGVhbjtcbn1cblxuLy8gU3RhdGVcbmxldCBtdXRlZCA9IGZhbHNlO1xubGV0IGNoYW5nZWRCeVdhdGNoID0gZmFsc2U7XG5sZXQgc2tpcFdhdGNoZXJzID0gZmFsc2U7XG5sZXQgbmdab25lOiBhbnk7XG5sZXQgYXBwUmVmOiBhbnk7XG5cbi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICogQ0xFQU5VUCBTQ0hFRFVMRVIgV0lUSDpcbiAqICAtIFNpZ25pZmljYW50IHdhdGNoZXIgZGVsdGEgdHJpZ2dlciAoKy8tKVxuICogIC0gVUkgc3RhYmlsaXphdGlvbiBkZWxheSAoMeKAkzIgc2Vjb25kcylcbiAqICAtIENvb2xkb3duIHRvIHByZXZlbnQgcmVwZWF0ZWQgc2NoZWR1bGluZ1xuICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuY29uc3QgQ0xFQU5VUF9UUklHR0VSX0RFTFRBID0gMzAwOyAgIC8vIHdhdGNoZXIgY2hhbmdlIHRocmVzaG9sZFxuY29uc3QgQ0xFQU5VUF9ERUxBWSA9IDE1MDA7ICAgICAgICAgIC8vIGRlbGF5IGJlZm9yZSBydW5uaW5nIGNsZWFudXBcbmNvbnN0IENMRUFOVVBfQ09PTERPV04gPSA0MDAwOyAgICAgICAvLyBwcmV2ZW50IHJlLXNjaGVkdWxpbmcgZm9yIDRzXG5cbmxldCBsYXN0V2F0Y2hlckNvdW50ID0gMDtcbmxldCBzY2hlZHVsZWRDbGVhbnVwOiBhbnkgPSBudWxsO1xubGV0IGxhc3RTY2hlZHVsZWRUaW1lID0gMDtcblxuZnVuY3Rpb24gZ2V0V2F0Y2hlckNvdW50KCk6IG51bWJlciB7XG4gICAgbGV0IGNvdW50ID0gMDtcbiAgICByZWdpc3RyeS5mb3JFYWNoKCh2YWx1ZSkgPT4ge1xuICAgICAgICAvLyBDaGVjayBpZiBpdCdzIGEgZ3JvdXBlZCBzdHJ1Y3R1cmUgKHdpZGdldCBncm91cHMpIG9yIGEgZGlyZWN0IHdhdGNoSW5mb1xuICAgICAgICBpZiAodmFsdWUgJiYgdHlwZW9mIHZhbHVlID09PSAnb2JqZWN0JyAmJiAhdmFsdWUuZm4pIHtcbiAgICAgICAgICAgIC8vIEl0J3MgYSB3aWRnZXQgZ3JvdXAgLSBjb3VudCB0aGUgcHJvcGVydGllcyAoZXhjbHVkaW5nIG1ldGFkYXRhKVxuICAgICAgICAgICAgY291bnQgKz0gT2JqZWN0LmtleXModmFsdWUpLmZpbHRlcihrID0+IGsgIT09ICdzY29wZVR5cGUnICYmIGsgIT09ICdzY29wZU5hbWUnKS5sZW5ndGg7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyBJdCdzIGEgZGlyZWN0IHdhdGNoSW5mb1xuICAgICAgICAgICAgY291bnQgKz0gMTtcbiAgICAgICAgfVxuICAgIH0pO1xuICAgIHJldHVybiBjb3VudDtcbn1cblxuZXhwb3J0IGNvbnN0IGNsZWFudXBTdGFsZVdhdGNoZXJzID0gKCk6IG51bWJlciA9PiB7XG4gICAgLy8gY29uc29sZS5sb2coXCIuLi4uLi4uLi5DbGVhbmluZyB1cCBzdGFsZSB3YXRjaGVycy4uLnJlZ2lzdHJ5LnNpemUuLi4uXCIsIHJlZ2lzdHJ5LnNpemUpO1xuICAgIGxldCByZW1vdmVkID0gMDtcblxuICAgIHJlZ2lzdHJ5LmZvckVhY2goKGJ1Y2tldCwgd2lkZ2V0SWQpID0+IHtcbiAgICAgICAgaWYgKCFkb2N1bWVudC5xdWVyeVNlbGVjdG9yKGBbd2lkZ2V0LWlkPVwiJHt3aWRnZXRJZH1cIl1gKSkge1xuICAgICAgICAgICAgZm9yIChjb25zdCBrZXkgaW4gYnVja2V0KSB7XG4gICAgICAgICAgICAgICAgaWYgKGJ1Y2tldC5oYXNPd25Qcm9wZXJ0eShrZXkpICYmIGtleSAhPT0gXCJzY29wZVR5cGVcIiAmJiBrZXkgIT09IFwic2NvcGVOYW1lXCIgJiYgdHlwZW9mIGJ1Y2tldFtrZXldICE9PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgICAgICAgICAgICAgICAgbGV0IHdhdGNoSW5mbyA9IGJ1Y2tldFtrZXldO1xuICAgICAgICAgICAgICAgICAgICBpZiAod2F0Y2hJbmZvICYmIHdhdGNoSW5mby5kZXN0cm95Rm4pIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIHdhdGNoSW5mby5kZXN0cm95Rm4oKTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJlbW92ZWQrKztcbiAgICAgICAgICAgIHJlZ2lzdHJ5LmRlbGV0ZSh3aWRnZXRJZCk7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cbiAgICB9KTtcbiAgICByZXR1cm4gcmVtb3ZlZDtcbn07XG5cbmV4cG9ydCBjb25zdCBzY2hlZHVsZVRocmVzaG9sZENsZWFudXAgPSAoKTogdm9pZCA9PiB7XG4gICAgY29uc3Qgbm93ID0gcGVyZm9ybWFuY2Uubm93KCk7XG4gICAgbGV0IGxhc3RUcmlnZ2VyUGVyaW9kID0gbm93IC0gbGFzdFNjaGVkdWxlZFRpbWU7XG4gICAgLy8gSWYgYSBjbGVhbnVwIHdhcyBzY2hlZHVsZWQgcmVjZW50bHksIHNraXAgc2NoZWR1bGluZ1xuICAgIGlmIChsYXN0VHJpZ2dlclBlcmlvZCA8IENMRUFOVVBfQ09PTERPV04pIHtcbiAgICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IGN1cnJlbnQgPSBnZXRXYXRjaGVyQ291bnQoKTtcbiAgICBjb25zdCBkZWx0YSA9IE1hdGguYWJzKGN1cnJlbnQgLSBsYXN0V2F0Y2hlckNvdW50KTsgIC8vIHNpZ25pZmljYW50ICsgb3IgLVxuXG4gICAgLy8gSWYgY2hhbmdlIG5vdCBsYXJnZSBlbm91Z2gsIHNraXAgc2NoZWR1bGluZ1xuICAgIGlmIChkZWx0YSA8PSBDTEVBTlVQX1RSSUdHRVJfREVMVEEpIHtcbiAgICAgICAgbGFzdFdhdGNoZXJDb3VudCA9IGN1cnJlbnQ7XG4gICAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICAvLyBQcmV2ZW50IHJlLXNjaGVkdWxpbmc6IHNldCB0aW1lc3RhbXAgZWFybHlcbiAgICBsYXN0U2NoZWR1bGVkVGltZSA9IG5vdztcblxuICAgIC8vIENsZWFyIHByZXZpb3VzIHNjaGVkdWxlZCBjbGVhbnVwIChpZiBhbnkpXG4gICAgaWYgKHNjaGVkdWxlZENsZWFudXApIHtcbiAgICAgICAgY2xlYXJUaW1lb3V0KHNjaGVkdWxlZENsZWFudXApO1xuICAgIH1cblxuICAgIC8vIFNjaGVkdWxlIGNsZWFudXAgYWZ0ZXIgVUkgc3RhYmlsaXplcyAoZGVsYXkgcHJldmVudHMgYWdncmVzc2l2ZSBjbGVhbnVwKVxuICAgIHNjaGVkdWxlZENsZWFudXAgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgY2xlYW51cFN0YWxlV2F0Y2hlcnMoKTtcbiAgICAgICAgc2NoZWR1bGVkQ2xlYW51cCA9IG51bGw7XG4gICAgfSwgQ0xFQU5VUF9ERUxBWSk7XG5cbiAgICBsYXN0V2F0Y2hlckNvdW50ID0gY3VycmVudDtcbn07XG5cbi8vIFV0aWxpdHkgZnVuY3Rpb25zXG5leHBvcnQgY29uc3QgaXNGaXJzdFRpbWVDaGFuZ2UgPSAodjogYW55KTogYm9vbGVhbiA9PiB2ID09PSBGSVJTVF9USU1FX1dBVENIO1xuXG5leHBvcnQgY29uc3QgZGVib3VuY2UgPSAoZm46IEZ1bmN0aW9uLCB3YWl0OiBudW1iZXIgPSBERUJPVU5DRV9XQUlUKSA9PiB7XG4gICAgbGV0IHRpbWVvdXQ6IGFueTtcbiAgICByZXR1cm4gKC4uLmFyZ3M6IGFueVtdKSA9PiB7XG4gICAgICAgIHdpbmRvd1snX196b25lX3N5bWJvbF9fY2xlYXJUaW1lb3V0J10odGltZW91dCk7XG4gICAgICAgIHRpbWVvdXQgPSB3aW5kb3dbJ19fem9uZV9zeW1ib2xfX3NldFRpbWVvdXQnXSgoKSA9PiBmbiguLi5hcmdzKSwgd2FpdCk7XG4gICAgfTtcbn07XG5cbmV4cG9ydCBjb25zdCBtdXRlV2F0Y2hlcnMgPSAoKTogdm9pZCA9PiB7XG4gICAgbXV0ZWQgPSB0cnVlO1xufTtcblxuZXhwb3J0IGNvbnN0IHVuTXV0ZVdhdGNoZXJzID0gKCk6IHZvaWQgPT4ge1xuICAgIG11dGVkID0gZmFsc2U7XG4gICAgdHJpZ2dlcldhdGNoZXJzKCk7XG59O1xuXG4vKipcbiAqIEV4dHJhY3RzIHdpZGdldCBJRCBmcm9tIGlkZW50aWZpZXIgKGUuZy4sIFwid2lkZ2V0LWlkMjNfZXZlbnRzb3VyY2VcIiAtPiBcIndpZGdldC1pZDIzXCIpXG4gKi9cbmNvbnN0IGdldFdpZGdldElkID0gKGlkZW50aWZpZXI6IHN0cmluZyk6IHN0cmluZyB8IG51bGwgPT4ge1xuICAgIGlmICghaWRlbnRpZmllciB8fCB0eXBlb2YgaWRlbnRpZmllciAhPT0gJ3N0cmluZycpIHtcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuICAgIGNvbnN0IG1hdGNoID0gaWRlbnRpZmllci5tYXRjaChXSURHRVRfSURfUkVHRVgpO1xuICAgIHJldHVybiBtYXRjaCA/IG1hdGNoWzFdIDogbnVsbDtcbn07XG5cbi8qKlxuICogRXh0cmFjdHMgcHJvcGVydHkgbmFtZSBmcm9tIGlkZW50aWZpZXIgKGUuZy4sIFwid2lkZ2V0LWlkMjNfZXZlbnRzb3VyY2VcIiAtPiBcImV2ZW50c291cmNlXCIpXG4gKi9cbmNvbnN0IGdldFByb3BlcnR5TmFtZSA9IChpZGVudGlmaWVyOiBzdHJpbmcpOiBzdHJpbmcgPT4ge1xuICAgIGlmICghaWRlbnRpZmllciB8fCB0eXBlb2YgaWRlbnRpZmllciAhPT0gJ3N0cmluZycpIHtcbiAgICAgICAgcmV0dXJuIGlkZW50aWZpZXI7XG4gICAgfVxuICAgIGNvbnN0IG1hdGNoID0gaWRlbnRpZmllci5tYXRjaChXSURHRVRfUFJPUEVSVFlfUkVHRVgpO1xuICAgIHJldHVybiBtYXRjaCA/IG1hdGNoWzFdIDogaWRlbnRpZmllcjtcbn07XG5cbi8qKlxuICogQXJyYXkgY29uc3VtZXIgd3JhcHBlciBmb3IgYXJyYXktYmFzZWQgZXhwcmVzc2lvbnNcbiAqL1xuY29uc3QgYXJyYXlDb25zdW1lciA9IChsaXN0ZW5lckZuOiBhbnksIHJlc3RFeHByOiBzdHJpbmcsIG5ld1ZhbDogYW55LCBvbGRWYWw6IGFueSk6IHZvaWQgPT4ge1xuICAgIGlmICghaXNBcnJheShuZXdWYWwpKSB7XG4gICAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBsZXQgZm9ybWF0dGVkRGF0YSA9IG5ld1ZhbC5tYXAoZGF0dW0gPT4gZmluZFZhbHVlT2YoZGF0dW0sIHJlc3RFeHByKSk7XG5cbiAgICAvLyBGbGF0dGVuIGlmIHJlc3VsdCBpcyBhcnJheSBvZiBhcnJheXNcbiAgICBpZiAoaXNBcnJheShmb3JtYXR0ZWREYXRhWzBdKSkge1xuICAgICAgICBmb3JtYXR0ZWREYXRhID0gZmxhdHRlbihmb3JtYXR0ZWREYXRhKTtcbiAgICB9XG5cbiAgICBsaXN0ZW5lckZuKGZvcm1hdHRlZERhdGEsIG9sZFZhbCk7XG59O1xuXG4vKipcbiAqIFVwZGF0ZXMgd2F0Y2ggaW5mbyBmb3IgYXJyYXkgZXhwcmVzc2lvbnNcbiAqL1xuY29uc3QgZ2V0VXBkYXRlZFdhdGNoSW5mbyA9IChleHByOiBzdHJpbmcsIGFjY2VwdHNBcnJheTogYm9vbGVhbiwgbGlzdGVuZXI6IGFueSkgPT4ge1xuICAgIGNvbnN0IHJlZ2V4ID0gL1xcW1xcJGlcXF0vZztcblxuICAgIGlmICghYWNjZXB0c0FycmF5KSB7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBleHByOiBleHByLnJlcGxhY2UocmVnZXgsIEFSUkFZX0lOREVYX1pFUk8pLFxuICAgICAgICAgICAgbGlzdGVuZXJcbiAgICAgICAgfTtcbiAgICB9XG5cbiAgICBjb25zdCBsYXN0SW5kZXggPSBleHByLmxhc3RJbmRleE9mKEFSUkFZX0lOREVYX1BMQUNFSE9MREVSKTtcbiAgICBjb25zdCBiYXNlRXhwciA9IGV4cHIuc3Vic3RyaW5nKDAsIGxhc3RJbmRleCkucmVwbGFjZShBUlJBWV9JTkRFWF9QTEFDRUhPTERFUiwgQVJSQVlfSU5ERVhfWkVSTyk7XG4gICAgY29uc3QgcmVzdEV4cHIgPSBleHByLnN1YnN0cmluZyhsYXN0SW5kZXggKyA1KTtcblxuICAgIGNvbnN0IGFycmF5Q29uc3VtZXJGbiA9IHJlc3RFeHByXG4gICAgICAgID8gYXJyYXlDb25zdW1lci5iaW5kKHVuZGVmaW5lZCwgbGlzdGVuZXIsIHJlc3RFeHByKVxuICAgICAgICA6IGxpc3RlbmVyO1xuXG4gICAgcmV0dXJuIHtcbiAgICAgICAgZXhwcjogYmFzZUV4cHIsXG4gICAgICAgIGxpc3RlbmVyOiBhcnJheUNvbnN1bWVyRm5cbiAgICB9O1xufTtcblxuLyoqXG4gKiBEZXRlcm1pbmVzIGlmIGFuIGV4cHJlc3Npb24gaXMgc3RhdGljIChkb2Vzbid0IG5lZWQgdG8gYmUgd2F0Y2hlZClcbiAqL1xuY29uc3QgU1RBVElDX0VYUFJFU1NJT05fTkFNRVMgPSBbXG4gICAgXCJyb3cuZ2V0UHJvcGVydHkoJ2ludmVzdG1lbnQnKVwiLFxuICAgIFwicm93LmdldFByb3BlcnR5KCdmYWN0c2hlZXRMaW5rJylcIixcbiAgICBcInJvdy5nZXRQcm9wZXJ0eSgnaXNSZWJhbGFuY2VFbGlnaWJsZScpXCJcbl07XG5cbmNvbnN0IGlzU3RhdGljRXhwcmVzc2lvbiA9IChleHByOiBhbnkpOiBib29sZWFuID0+IHtcbiAgICBpZiAodHlwZW9mIGV4cHIgIT09ICdzdHJpbmcnKSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG5cbiAgICBjb25zdCB0cmltbWVkRXhwciA9IGV4cHIudHJpbSgpO1xuXG4gICAgLy8gRXhwcmVzc2lvbnMgdGhhdCBhbHdheXMgZXZhbHVhdGUgdG8gbG9jYWxpemF0aW9uIHN0cmluZ3NcbiAgICAvLyBpZiAodHJpbW1lZEV4cHIuaW5jbHVkZXMoJ2FwcExvY2FsZScpKSB7XG4gICAgLy8gICAgIHJldHVybiB0cnVlO1xuICAgIC8vIH1cblxuICAgIC8vIEhhcmQtY29kZWQgc3RhdGljIGV4cHJlc3Npb24gbmFtZXNcbiAgICBpZiAoU1RBVElDX0VYUFJFU1NJT05fTkFNRVMuaW5jbHVkZXModHJpbW1lZEV4cHIpKSB7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cblxuICAgIHJldHVybiBmYWxzZTtcbn07XG5cbi8qKlxuICogR2V0cyB0aGUgc2NvcGUgdHlwZSBmcm9tIHRoZSBzY29wZSBvYmplY3RcbiAqL1xuY29uc3QgZ2V0U2NvcGVUeXBlID0gKCRzY29wZTogYW55KTogU2NvcGVUeXBlIHwgbnVsbCA9PiB7XG4gICAgaWYgKCEkc2NvcGUpIHtcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgaWYgKCRzY29wZS5wYWdlTmFtZSkgcmV0dXJuICdQYWdlJztcbiAgICBpZiAoJHNjb3BlLnByZWZhYk5hbWUpIHJldHVybiAnUHJlZmFiJztcbiAgICBpZiAoJHNjb3BlLnBhcnRpYWxOYW1lKSByZXR1cm4gJ1BhcnRpYWwnO1xuXG4gICAgLy8gQ2hlY2sgZm9yIEFwcCBzY29wZVxuICAgIGlmICgkc2NvcGUuVmFyaWFibGVzICE9PSB1bmRlZmluZWQgJiZcbiAgICAgICAgJHNjb3BlLkFjdGlvbnMgIT09IHVuZGVmaW5lZCAmJlxuICAgICAgICAhJHNjb3BlLnBhZ2VOYW1lICYmXG4gICAgICAgICEkc2NvcGUucHJlZmFiTmFtZSAmJlxuICAgICAgICAhJHNjb3BlLnBhcnRpYWxOYW1lKSB7XG4gICAgICAgIHJldHVybiAnQXBwJztcbiAgICB9XG5cbiAgICBpZiAoJHNjb3BlLmNvbnN0cnVjdG9yPy5uYW1lID09PSAnQXBwUmVmJykge1xuICAgICAgICByZXR1cm4gJ0FwcCc7XG4gICAgfVxuXG4gICAgcmV0dXJuIG51bGw7XG59O1xuXG4vKipcbiAqIEdldHMgc2NvcGUgbmFtZSBiYXNlZCBvbiBzY29wZSB0eXBlXG4gKi9cbmNvbnN0IGdldFNjb3BlTmFtZSA9ICgkc2NvcGU6IGFueSwgc2NvcGVUeXBlOiBTY29wZVR5cGUgfCBudWxsKTogc3RyaW5nIHwgbnVsbCA9PiB7XG4gICAgaWYgKCFzY29wZVR5cGUgfHwgISRzY29wZSkge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBzd2l0Y2ggKHNjb3BlVHlwZSkge1xuICAgICAgICBjYXNlICdQcmVmYWInOiByZXR1cm4gJHNjb3BlLnByZWZhYk5hbWUgfHwgbnVsbDtcbiAgICAgICAgY2FzZSAnUGFydGlhbCc6IHJldHVybiAkc2NvcGUucGFydGlhbE5hbWUgfHwgbnVsbDtcbiAgICAgICAgY2FzZSAnUGFnZSc6IHJldHVybiAkc2NvcGUucGFnZU5hbWUgfHwgbnVsbDtcbiAgICAgICAgZGVmYXVsdDogcmV0dXJuIG51bGw7XG4gICAgfVxufTtcblxuLyoqXG4gKiBNYWluIHdhdGNoIGZ1bmN0aW9uXG4gKi9cbmV4cG9ydCBjb25zdCAkd2F0Y2ggPSAoXG4gICAgZXhwcjogc3RyaW5nLFxuICAgICRzY29wZTogYW55LFxuICAgICRsb2NhbHM6IGFueSxcbiAgICBsaXN0ZW5lcjogYW55LFxuICAgIGlkZW50aWZpZXI6IHN0cmluZyA9IHdhdGNoSWRHZW5lcmF0b3IubmV4dFVpZCgpLFxuICAgIGRvTm90Q2xvbmU6IGJvb2xlYW4gPSBmYWxzZSxcbiAgICBjb25maWc6IFdhdGNoQ29uZmlnID0ge30sXG4gICAgaXNNdXRlZD86ICgpID0+IGJvb2xlYW5cbikgPT4ge1xuICAgIC8vIEhhbmRsZSBhcnJheSBleHByZXNzaW9uc1xuICAgIGlmIChleHByLmluY2x1ZGVzKEFSUkFZX0lOREVYX1BMQUNFSE9MREVSKSkge1xuICAgICAgICBjb25zdCB3YXRjaEluZm8gPSBnZXRVcGRhdGVkV2F0Y2hJbmZvKGV4cHIsIGNvbmZpZy5hcnJheVR5cGUgfHwgY29uZmlnLmlzTGlzdCB8fCBmYWxzZSwgbGlzdGVuZXIpO1xuICAgICAgICBleHByID0gd2F0Y2hJbmZvLmV4cHI7XG4gICAgICAgIGxpc3RlbmVyID0gd2F0Y2hJbmZvLmxpc3RlbmVyO1xuICAgIH1cblxuICAgIC8vIEhhbmRsZSBzdGF0aWMgZXhwcmVzc2lvbnNcbiAgICBpZiAoaXNTdGF0aWNFeHByZXNzaW9uKGV4cHIpKSB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgICBjb25zdCBmbiA9ICRwYXJzZUV4cHIoZXhwcik7XG4gICAgICAgICAgICBjb25zdCBzdGF0aWNWYWx1ZSA9IGZuKCRzY29wZSwgJGxvY2Fscyk7XG4gICAgICAgICAgICBsaXN0ZW5lcihzdGF0aWNWYWx1ZSwgRklSU1RfVElNRV9XQVRDSCk7XG4gICAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgICAgIGNvbnNvbGUud2FybihgRXJyb3IgZXZhbHVhdGluZyBzdGF0aWMgZXhwcmVzc2lvbiAnJHtleHByfSc6YCwgZSk7XG4gICAgICAgICAgICBsaXN0ZW5lcih1bmRlZmluZWQsIEZJUlNUX1RJTUVfV0FUQ0gpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiAoKSA9PiB7IH07IC8vIE5vLW9wIHVuc3Vic2NyaWJlXG4gICAgfVxuXG4gICAgY29uc3QgZm4gPSAkcGFyc2VFeHByKGV4cHIpO1xuICAgIGNvbnN0IHNjb3BlVHlwZSA9IGdldFNjb3BlVHlwZSgkc2NvcGUpO1xuICAgIGNvbnN0IHNjb3BlTmFtZSA9IGdldFNjb3BlTmFtZSgkc2NvcGUsIHNjb3BlVHlwZSk7XG5cbiAgICBjb25zdCBkZXN0cm95Rm4gPSAoKSA9PiAkdW53YXRjaChpZGVudGlmaWVyKTtcbiAgICBjb25zdCB3YXRjaEluZm86IFdhdGNoSW5mbyA9IHtcbiAgICAgICAgZm46IGZuLmJpbmQobnVsbCwgJHNjb3BlLCAkbG9jYWxzKSxcbiAgICAgICAgbGlzdGVuZXIsXG4gICAgICAgIGV4cHIsXG4gICAgICAgIGxhc3Q6IEZJUlNUX1RJTUVfV0FUQ0gsXG4gICAgICAgIGRvTm90Q2xvbmUsXG4gICAgICAgIGlzTXV0ZWQsXG4gICAgICAgIHNjb3BlVHlwZSxcbiAgICAgICAgc2NvcGVOYW1lLFxuICAgICAgICBkZXN0cm95Rm5cbiAgICB9O1xuXG4gICAgLy8gU3RvcmUgaW4gcmVnaXN0cnlcbiAgICBjb25zdCB3aWRnZXRJZCA9IGdldFdpZGdldElkKGlkZW50aWZpZXIpO1xuXG4gICAgaWYgKHdpZGdldElkKSB7XG4gICAgICAgIGNvbnN0IHByb3BlcnR5TmFtZSA9IGdldFByb3BlcnR5TmFtZShpZGVudGlmaWVyKTtcbiAgICAgICAgaWYgKCFyZWdpc3RyeS5oYXMod2lkZ2V0SWQpKSB7XG4gICAgICAgICAgICByZWdpc3RyeS5zZXQod2lkZ2V0SWQsIHt9KTtcbiAgICAgICAgfVxuICAgICAgICBjb25zdCB3aWRnZXRHcm91cCA9IHJlZ2lzdHJ5LmdldCh3aWRnZXRJZCk7XG4gICAgICAgIHdpZGdldEdyb3VwW3Byb3BlcnR5TmFtZV0gPSB3YXRjaEluZm87XG4gICAgfSBlbHNlIHtcbiAgICAgICAgcmVnaXN0cnkuc2V0KGlkZW50aWZpZXIsIHdhdGNoSW5mbyk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGRlc3Ryb3lGbjtcbn07XG5cbi8qKlxuICogVW53YXRjaGVzIGEgc2luZ2xlIGlkZW50aWZpZXJcbiAqL1xuZXhwb3J0IGNvbnN0ICR1bndhdGNoID0gKGlkZW50aWZpZXI6IHN0cmluZyk6IGJvb2xlYW4gPT4ge1xuICAgIGNvbnN0IHdpZGdldElkID0gZ2V0V2lkZ2V0SWQoaWRlbnRpZmllcik7XG5cbiAgICBpZiAod2lkZ2V0SWQpIHtcbiAgICAgICAgY29uc3QgcHJvcGVydHlOYW1lID0gZ2V0UHJvcGVydHlOYW1lKGlkZW50aWZpZXIpO1xuICAgICAgICBjb25zdCB3aWRnZXRHcm91cCA9IHJlZ2lzdHJ5LmdldCh3aWRnZXRJZCk7XG5cbiAgICAgICAgLy9AdHMtaWdub3JlXG4gICAgICAgIGlmICh3aWRnZXRHcm91cCAmJiB0eXBlb2Ygd2lkZ2V0R3JvdXAgPT09ICdvYmplY3QnICYmICF3aWRnZXRHcm91cC5mbikge1xuICAgICAgICAgICAgY29uc3Qgd2F0Y2hJbmZvID0gd2lkZ2V0R3JvdXBbcHJvcGVydHlOYW1lXTtcbiAgICAgICAgICAgIGlmICh3YXRjaEluZm8pIHtcbiAgICAgICAgICAgICAgICBkZWxldGUgd2lkZ2V0R3JvdXBbcHJvcGVydHlOYW1lXTtcbiAgICAgICAgICAgICAgICAvLyBDbGVhbiB1cCBlbXB0eSB3aWRnZXQgZ3JvdXBzXG4gICAgICAgICAgICAgICAgaWYgKE9iamVjdC5rZXlzKHdpZGdldEdyb3VwKS5sZW5ndGggPT09IDApIHtcbiAgICAgICAgICAgICAgICAgICAgcmVnaXN0cnkuZGVsZXRlKHdpZGdldElkKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBGYWxsYmFjayB0byBkaXJlY3QgbG9va3VwXG4gICAgaWYgKHJlZ2lzdHJ5LmhhcyhpZGVudGlmaWVyKSkge1xuICAgICAgICByZWdpc3RyeS5kZWxldGUoaWRlbnRpZmllcik7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cblxuICAgIHJldHVybiBmYWxzZTtcbn07XG5cbi8qKlxuICogVW53YXRjaGVzIGFsbCB3YXRjaGVycyBmb3IgYSBzcGVjaWZpYyB3aWRnZXQgSURcbiAqL1xuZXhwb3J0IGNvbnN0ICR1bndhdGNoQWxsID0gKHdpZGdldElkOiBzdHJpbmcpOiBudW1iZXIgPT4ge1xuICAgIGlmICghd2lkZ2V0SWQgfHwgdHlwZW9mIHdpZGdldElkICE9PSAnc3RyaW5nJykge1xuICAgICAgICByZXR1cm4gMDtcbiAgICB9XG5cbiAgICBjb25zdCB3aWRnZXRHcm91cCA9IHJlZ2lzdHJ5LmdldCh3aWRnZXRJZCk7XG5cbiAgICBpZiAod2lkZ2V0R3JvdXAgJiYgdHlwZW9mIHdpZGdldEdyb3VwID09PSAnb2JqZWN0JyAmJiAhd2lkZ2V0R3JvdXAuZm4pIHtcbiAgICAgICAgY29uc3QgY291bnQgPSBPYmplY3Qua2V5cyh3aWRnZXRHcm91cCkubGVuZ3RoO1xuICAgICAgICByZWdpc3RyeS5kZWxldGUod2lkZ2V0SWQpO1xuICAgICAgICByZXR1cm4gY291bnQ7XG4gICAgfVxuXG4gICAgLy8gRmFsbGJhY2s6IGZpbmQgYWxsIGlkZW50aWZpZXJzIHN0YXJ0aW5nIHdpdGggdGhpcyB3aWRnZXQgSURcbiAgICBsZXQgcmVtb3ZlZENvdW50ID0gMDtcbiAgICBjb25zdCBpZGVudGlmaWVyc1RvUmVtb3ZlOiBzdHJpbmdbXSA9IFtdO1xuXG4gICAgcmVnaXN0cnkuZm9yRWFjaCgoXywga2V5KSA9PiB7XG4gICAgICAgIGlmIChrZXkuc3RhcnRzV2l0aCh3aWRnZXRJZCArICdfJykpIHtcbiAgICAgICAgICAgIGlkZW50aWZpZXJzVG9SZW1vdmUucHVzaChrZXkpO1xuICAgICAgICB9XG4gICAgfSk7XG5cbiAgICBpZGVudGlmaWVyc1RvUmVtb3ZlLmZvckVhY2goaWRlbnRpZmllciA9PiB7XG4gICAgICAgIGlmICgkdW53YXRjaChpZGVudGlmaWVyKSkge1xuICAgICAgICAgICAgcmVtb3ZlZENvdW50Kys7XG4gICAgICAgIH1cbiAgICB9KTtcblxuICAgIHJldHVybiByZW1vdmVkQ291bnQ7XG59O1xuXG4vKipcbiAqIFVud2F0Y2hlcyBhbGwgd2F0Y2hlcnMgZm9yIGEgc3BlY2lmaWMgc2NvcGUgKFBhZ2UsIFByZWZhYiwgUGFydGlhbCwgb3IgQXBwKVxuICogTm93IHdvcmtzIGRpcmVjdGx5IHdpdGggdGhlIG1haW4gcmVnaXN0cnkgaW5zdGVhZCBvZiBzZXBhcmF0ZSBzY29wZWQgcmVnaXN0cmllc1xuICovXG5leHBvcnQgY29uc3QgJHVud2F0Y2hBbGxCeVNjb3BlID0gKHNjb3BlVHlwZTogU2NvcGVUeXBlLCBzY29wZU5hbWU/OiBzdHJpbmcpOiBudW1iZXIgPT4ge1xuICAgIGlmICghc2NvcGVUeXBlKSB7XG4gICAgICAgIHJldHVybiAwO1xuICAgIH1cblxuICAgIGxldCByZW1vdmVkQ291bnQgPSAwO1xuICAgIGNvbnN0IGlkZW50aWZpZXJzVG9SZW1vdmU6IHN0cmluZ1tdID0gW107XG5cbiAgICByZWdpc3RyeS5mb3JFYWNoKCh2YWx1ZSwga2V5KSA9PiB7XG4gICAgICAgIC8vIEhhbmRsZSBncm91cGVkIHN0cnVjdHVyZSAod2lkZ2V0IGdyb3VwcylcbiAgICAgICAgaWYgKHZhbHVlICYmIHR5cGVvZiB2YWx1ZSA9PT0gJ29iamVjdCcgJiYgIXZhbHVlLmZuKSB7XG4gICAgICAgICAgICBPYmplY3QuZW50cmllcyh2YWx1ZSkuZm9yRWFjaCgoW3Byb3BlcnR5TmFtZSwgd2F0Y2hJbmZvXTogW3N0cmluZywgYW55XSkgPT4ge1xuICAgICAgICAgICAgICAgIGlmICh3YXRjaEluZm8/LnNjb3BlVHlwZSA9PT0gc2NvcGVUeXBlICYmXG4gICAgICAgICAgICAgICAgICAgICghc2NvcGVOYW1lIHx8IHdhdGNoSW5mby5zY29wZU5hbWUgPT09IHNjb3BlTmFtZSkpIHtcbiAgICAgICAgICAgICAgICAgICAgaWRlbnRpZmllcnNUb1JlbW92ZS5wdXNoKGAke2tleX1fJHtwcm9wZXJ0eU5hbWV9YCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyBEaXJlY3Qgd2F0Y2hJbmZvXG4gICAgICAgICAgICBjb25zdCB3YXRjaEluZm8gPSB2YWx1ZSBhcyBXYXRjaEluZm87XG4gICAgICAgICAgICBpZiAod2F0Y2hJbmZvPy5zY29wZVR5cGUgPT09IHNjb3BlVHlwZSAmJlxuICAgICAgICAgICAgICAgICghc2NvcGVOYW1lIHx8IHdhdGNoSW5mby5zY29wZU5hbWUgPT09IHNjb3BlTmFtZSkpIHtcbiAgICAgICAgICAgICAgICBpZGVudGlmaWVyc1RvUmVtb3ZlLnB1c2goa2V5KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgIH0pO1xuXG4gICAgLy8gVW53YXRjaCBhbGwgY29sbGVjdGVkIGlkZW50aWZpZXJzXG4gICAgaWRlbnRpZmllcnNUb1JlbW92ZS5mb3JFYWNoKGlkZW50aWZpZXIgPT4ge1xuICAgICAgICBpZiAoJHVud2F0Y2goaWRlbnRpZmllcikpIHtcbiAgICAgICAgICAgIHJlbW92ZWRDb3VudCsrO1xuICAgICAgICB9XG4gICAgfSk7XG5cbiAgICByZXR1cm4gcmVtb3ZlZENvdW50O1xufTtcblxuLyoqXG4gKiBQcm9jZXNzZXMgYSBzaW5nbGUgd2F0Y2ggaW5mbyBkdXJpbmcgdHJpZ2dlciBjeWNsZVxuICovXG5jb25zdCBwcm9jZXNzV2F0Y2hJbmZvID0gKHdhdGNoSW5mbzogV2F0Y2hJbmZvKTogYm9vbGVhbiA9PiB7XG4gICAgaWYgKCF3YXRjaEluZm8/LmZuIHx8ICh3YXRjaEluZm8uaXNNdXRlZD8uKCkgPz8gZmFsc2UpKSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG5cbiAgICBsZXQgbmV3VmFsdWU6IGFueTtcbiAgICB0cnkge1xuICAgICAgICBuZXdWYWx1ZSA9IHdhdGNoSW5mby5mbigpO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgY29uc29sZS53YXJuKGBFcnJvciBleGVjdXRpbmcgZXhwcmVzc2lvbjogJyR7d2F0Y2hJbmZvLmV4cHJ9J2AsIGUpO1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgaWYgKGlzRXF1YWwobmV3VmFsdWUsIHdhdGNoSW5mby5sYXN0KSkge1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgLy8gQ2hhbmdlIGRldGVjdGVkXG4gICAgY2hhbmdlZEJ5V2F0Y2ggPSB0cnVlO1xuICAgIHdhdGNoSW5mby5sYXN0ID0gaXNPYmplY3QobmV3VmFsdWUpICYmICF3YXRjaEluZm8uZG9Ob3RDbG9uZVxuICAgICAgICA/IGNsb25lKG5ld1ZhbHVlKVxuICAgICAgICA6IG5ld1ZhbHVlO1xuXG4gICAgd2F0Y2hJbmZvLmxpc3RlbmVyKG5ld1ZhbHVlLCB3YXRjaEluZm8ubGFzdCA9PT0gbmV3VmFsdWUgPyBGSVJTVF9USU1FX1dBVENIIDogd2F0Y2hJbmZvLmxhc3QpO1xuICAgIGNoYW5nZWRCeVdhdGNoID0gZmFsc2U7XG5cbiAgICByZXR1cm4gdHJ1ZTtcbn07XG5cbi8qKlxuICogVHJpZ2dlcnMgYWxsIHdhdGNoZXJzXG4gKi9cbmNvbnN0IHRyaWdnZXJXYXRjaGVycyA9IChpZ25vcmVNdXRlZDogYm9vbGVhbiA9IGZhbHNlKTogdm9pZCA9PiB7XG4gICAgaWYgKG11dGVkICYmICFpZ25vcmVNdXRlZCkge1xuICAgICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgbGV0IHBhc3MgPSAxO1xuICAgIGxldCBjaGFuZ2VEZXRlY3RlZDogYm9vbGVhbjtcblxuICAgIGRvIHtcbiAgICAgICAgY2hhbmdlRGV0ZWN0ZWQgPSBmYWxzZTtcblxuICAgICAgICByZWdpc3RyeS5mb3JFYWNoKCh2YWx1ZSkgPT4ge1xuICAgICAgICAgICAgLy8gSGFuZGxlIGdyb3VwZWQgc3RydWN0dXJlXG4gICAgICAgICAgICBpZiAodmFsdWUgJiYgdHlwZW9mIHZhbHVlID09PSAnb2JqZWN0JyAmJiAhdmFsdWUuZm4pIHtcbiAgICAgICAgICAgICAgICBPYmplY3QudmFsdWVzKHZhbHVlKS5mb3JFYWNoKCh3YXRjaEluZm86IGFueSkgPT4ge1xuICAgICAgICAgICAgICAgICAgICBpZiAocHJvY2Vzc1dhdGNoSW5mbyh3YXRjaEluZm8pKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjaGFuZ2VEZXRlY3RlZCA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgLy8gRGlyZWN0IHdhdGNoSW5mb1xuICAgICAgICAgICAgICAgIGlmIChwcm9jZXNzV2F0Y2hJbmZvKHZhbHVlIGFzIFdhdGNoSW5mbykpIHtcbiAgICAgICAgICAgICAgICAgICAgY2hhbmdlRGV0ZWN0ZWQgPSB0cnVlO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cbiAgICAgICAgcGFzcysrO1xuICAgIH0gd2hpbGUgKGNoYW5nZURldGVjdGVkICYmIHBhc3MgPCBNQVhfV0FUQ0hfQ1lDTEVTKTtcblxuICAgIC8vIFNjaGVkdWxlIGNsZWFudXAgYWZ0ZXIgd2F0Y2hlcnMgYXJlIHRyaWdnZXJlZFxuICAgIHNjaGVkdWxlVGhyZXNob2xkQ2xlYW51cCgpO1xuXG4gICAgaWYgKGNoYW5nZURldGVjdGVkICYmIHBhc3MgPT09IE1BWF9XQVRDSF9DWUNMRVMpIHtcbiAgICAgICAgY29uc29sZS53YXJuKGBXYXRjaCBjeWNsZXMgZXhjZWVkZWQgbGltaXQgb2YgJHtNQVhfV0FUQ0hfQ1lDTEVTfWApO1xuICAgIH1cbn07XG5cbi8vIEFuZ3VsYXIgem9uZSBpbnRlZ3JhdGlvblxuZXhwb3J0IGNvbnN0IHNldE5nWm9uZSA9ICh6b25lOiBhbnkpOiB2b2lkID0+IHtcbiAgICBuZ1pvbmUgPSB6b25lO1xufTtcblxuZXhwb3J0IGNvbnN0IHNldEFwcFJlZiA9IChyZWY6IGFueSk6IHZvaWQgPT4ge1xuICAgIGFwcFJlZiA9IHJlZjtcbn07XG5cbmV4cG9ydCBjb25zdCBpc0NoYW5nZUZyb21XYXRjaCA9ICgpOiBib29sZWFuID0+IGNoYW5nZWRCeVdhdGNoO1xuZXhwb3J0IGNvbnN0IHJlc2V0Q2hhbmdlRnJvbVdhdGNoID0gKCk6IHZvaWQgPT4ge1xuICAgIGNoYW5nZWRCeVdhdGNoID0gZmFsc2U7XG59O1xuXG4vLyBEZWJvdW5jZWQgdHJpZ2dlclxuY29uc3QgZGVib3VuY2VkVHJpZ2dlcldhdGNoZXJzID0gZGVib3VuY2UoKCkgPT4ge1xuICAgIHNraXBXYXRjaGVycyA9IHRydWU7XG4gICAgbmdab25lLnJ1bigoKSA9PiB0cmlnZ2VyV2F0Y2hlcnMoKSk7XG59LCBERUJPVU5DRV9XQUlUKTtcblxuZXhwb3J0IGNvbnN0ICRpbnZva2VXYXRjaGVycyA9IChmb3JjZTogYm9vbGVhbiA9IGZhbHNlLCBpZ25vcmVNdXRlZDogYm9vbGVhbiA9IGZhbHNlKTogdm9pZCA9PiB7XG4gICAgaWYgKGZvcmNlKSB7XG4gICAgICAgIHRyaWdnZXJXYXRjaGVycyhpZ25vcmVNdXRlZCk7XG4gICAgfSBlbHNlIHtcbiAgICAgICAgaWYgKHNraXBXYXRjaGVycykge1xuICAgICAgICAgICAgc2tpcFdhdGNoZXJzID0gZmFsc2U7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cbiAgICAgICAgZGVib3VuY2VkVHJpZ2dlcldhdGNoZXJzKCk7XG4gICAgfVxufTtcblxuZXhwb3J0IGNvbnN0ICRhcHBEaWdlc3QgPSAoKCkgPT4ge1xuICAgIGxldCBxdWV1ZWQgPSBmYWxzZTtcbiAgICBjb25zdCAkUkFGID0gd2luZG93LnJlcXVlc3RBbmltYXRpb25GcmFtZTtcblxuICAgIHJldHVybiAoZm9yY2U6IGJvb2xlYW4gPSBmYWxzZSk6IHZvaWQgPT4ge1xuICAgICAgICBpZiAoIWFwcFJlZikge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKGZvcmNlKSB7XG4gICAgICAgICAgICBuZ1pvbmUucnVuKCgpID0+IGFwcFJlZi50aWNrKCkpO1xuICAgICAgICAgICAgcXVldWVkID0gZmFsc2U7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBpZiAocXVldWVkKSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgcXVldWVkID0gdHJ1ZTtcbiAgICAgICAgICAgICRSQUYoKCkgPT4ge1xuICAgICAgICAgICAgICAgIG5nWm9uZS5ydW4oKCkgPT4gYXBwUmVmLnRpY2soKSk7XG4gICAgICAgICAgICAgICAgcXVldWVkID0gZmFsc2U7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgIH07XG59KSgpO1xuXG4vLyBFeHBvcnQgcmVnaXN0cnkgZm9yIGRlYnVnZ2luZ1xuKHdpbmRvdyBhcyBhbnkpLndhdGNoUmVnaXN0cnkgPSByZWdpc3RyeTtcbiJdfQ==