mtrl-addons 0.5.6 → 0.7.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +567 -7
- package/dist/components/form/features/fields.d.ts +16 -9
- package/dist/components/form/features/index.d.ts +1 -1
- package/dist/components/index.d.ts +7 -3
- package/dist/components/index.js +9345 -0
- package/dist/components/index.mjs +9313 -0
- package/dist/components/vlist/types.d.ts +7 -0
- package/dist/core/gestures/index.js +508 -0
- package/dist/core/gestures/index.mjs +476 -0
- package/dist/core/layout/index.js +752 -0
- package/dist/core/layout/index.mjs +720 -0
- package/dist/core/viewport/index.js +3715 -0
- package/dist/core/viewport/index.mjs +3683 -0
- package/dist/index.js +890 -906
- package/dist/index.mjs +890 -906
- package/package.json +34 -3
|
@@ -0,0 +1,3715 @@
|
|
|
1
|
+
var __defProp = Object.defineProperty;
|
|
2
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
5
|
+
var __moduleCache = /* @__PURE__ */ new WeakMap;
|
|
6
|
+
var __toCommonJS = (from) => {
|
|
7
|
+
var entry = __moduleCache.get(from), desc;
|
|
8
|
+
if (entry)
|
|
9
|
+
return entry;
|
|
10
|
+
entry = __defProp({}, "__esModule", { value: true });
|
|
11
|
+
if (from && typeof from === "object" || typeof from === "function")
|
|
12
|
+
__getOwnPropNames(from).map((key) => !__hasOwnProp.call(entry, key) && __defProp(entry, key, {
|
|
13
|
+
get: () => from[key],
|
|
14
|
+
enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
|
|
15
|
+
}));
|
|
16
|
+
__moduleCache.set(from, entry);
|
|
17
|
+
return entry;
|
|
18
|
+
};
|
|
19
|
+
var __export = (target, all) => {
|
|
20
|
+
for (var name in all)
|
|
21
|
+
__defProp(target, name, {
|
|
22
|
+
get: all[name],
|
|
23
|
+
enumerable: true,
|
|
24
|
+
configurable: true,
|
|
25
|
+
set: (newValue) => all[name] = () => newValue
|
|
26
|
+
});
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
// src/core/viewport/index.ts
|
|
30
|
+
var exports_viewport = {};
|
|
31
|
+
__export(exports_viewport, {
|
|
32
|
+
withVirtual: () => withVirtual,
|
|
33
|
+
withScrolling: () => withScrolling,
|
|
34
|
+
withScrollbar: () => withScrollbar,
|
|
35
|
+
withRendering: () => withRendering,
|
|
36
|
+
withPlaceholders: () => withPlaceholders,
|
|
37
|
+
withEvents: () => withEvents,
|
|
38
|
+
withCollection: () => withCollection,
|
|
39
|
+
withBase: () => withBase,
|
|
40
|
+
createViewport: () => createViewport,
|
|
41
|
+
VIEWPORT_CONSTANTS: () => VIEWPORT_CONSTANTS
|
|
42
|
+
});
|
|
43
|
+
module.exports = __toCommonJS(exports_viewport);
|
|
44
|
+
|
|
45
|
+
// src/core/viewport/viewport.ts
|
|
46
|
+
var import_mtrl4 = require("mtrl");
|
|
47
|
+
|
|
48
|
+
// src/core/viewport/features/utils.ts
|
|
49
|
+
function wrapInitialize(component, featureInit) {
|
|
50
|
+
const originalInitialize = component.viewport.initialize;
|
|
51
|
+
component.viewport.initialize = () => {
|
|
52
|
+
const result = originalInitialize();
|
|
53
|
+
if (result !== false) {
|
|
54
|
+
featureInit();
|
|
55
|
+
}
|
|
56
|
+
return result;
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
function wrapDestroy(component, cleanup) {
|
|
60
|
+
if ("destroy" in component && typeof component.destroy === "function") {
|
|
61
|
+
const originalDestroy = component.destroy;
|
|
62
|
+
component.destroy = () => {
|
|
63
|
+
cleanup();
|
|
64
|
+
originalDestroy?.();
|
|
65
|
+
};
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
function getViewportState(component) {
|
|
69
|
+
return component.viewport.state;
|
|
70
|
+
}
|
|
71
|
+
function isPlaceholder(item) {
|
|
72
|
+
return item && typeof item === "object" && (item._placeholder === true || item["_placeholder"] === true);
|
|
73
|
+
}
|
|
74
|
+
function clamp(value, min, max) {
|
|
75
|
+
return Math.max(min, Math.min(max, value));
|
|
76
|
+
}
|
|
77
|
+
function storeFeatureFunction(component, name, fn) {
|
|
78
|
+
component[name] = fn;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// src/core/viewport/features/base.ts
|
|
82
|
+
var withBase = (config = {}) => {
|
|
83
|
+
return (component) => {
|
|
84
|
+
const { className = "mtrl-viewport", orientation = "vertical" } = config;
|
|
85
|
+
wrapInitialize(component, () => {
|
|
86
|
+
const element = component.element;
|
|
87
|
+
if (!element)
|
|
88
|
+
return;
|
|
89
|
+
let viewportElement = element.querySelector(".mtrl-viewport");
|
|
90
|
+
if (!viewportElement) {
|
|
91
|
+
viewportElement = document.createElement("div");
|
|
92
|
+
viewportElement.className = className;
|
|
93
|
+
viewportElement.style.cssText = `
|
|
94
|
+
position: relative;
|
|
95
|
+
width: 100%;
|
|
96
|
+
height: 100%;
|
|
97
|
+
overflow: hidden;
|
|
98
|
+
`;
|
|
99
|
+
viewportElement.setAttribute("data-orientation", orientation);
|
|
100
|
+
const itemsContainer = document.createElement("div");
|
|
101
|
+
itemsContainer.className = "mtrl-viewport-items";
|
|
102
|
+
itemsContainer.style.cssText = `
|
|
103
|
+
position: relative;
|
|
104
|
+
width: 100%;
|
|
105
|
+
`;
|
|
106
|
+
viewportElement.appendChild(itemsContainer);
|
|
107
|
+
element.appendChild(viewportElement);
|
|
108
|
+
const state = getViewportState(component);
|
|
109
|
+
if (state) {
|
|
110
|
+
state.viewportElement = viewportElement;
|
|
111
|
+
state.itemsContainer = itemsContainer;
|
|
112
|
+
}
|
|
113
|
+
component.viewportElement = viewportElement;
|
|
114
|
+
component.itemsContainer = itemsContainer;
|
|
115
|
+
}
|
|
116
|
+
});
|
|
117
|
+
return component;
|
|
118
|
+
};
|
|
119
|
+
};
|
|
120
|
+
|
|
121
|
+
// src/core/viewport/constants.ts
|
|
122
|
+
var VIEWPORT_CONSTANTS = {
|
|
123
|
+
VIRTUAL_SCROLL: {
|
|
124
|
+
DEFAULT_ITEM_SIZE: 48,
|
|
125
|
+
OVERSCAN_BUFFER: 2,
|
|
126
|
+
SCROLL_SENSITIVITY: 0.7,
|
|
127
|
+
MAX_VIRTUAL_SIZE: 100 * 1000 * 1000,
|
|
128
|
+
AUTO_DETECT_ITEM_SIZE: true
|
|
129
|
+
},
|
|
130
|
+
SCROLLING: {
|
|
131
|
+
OVERSCAN: 1
|
|
132
|
+
},
|
|
133
|
+
RENDERING: {
|
|
134
|
+
DEFAULT_MAX_POOL_SIZE: 100,
|
|
135
|
+
CLASSES: {
|
|
136
|
+
ITEM: "viewport-item"
|
|
137
|
+
}
|
|
138
|
+
},
|
|
139
|
+
LOADING: {
|
|
140
|
+
CANCEL_THRESHOLD: 25,
|
|
141
|
+
MAX_CONCURRENT_REQUESTS: 1,
|
|
142
|
+
DEFAULT_RANGE_SIZE: 20,
|
|
143
|
+
DEBOUNCE_LOADING: 150,
|
|
144
|
+
MIN_RANGE_SIZE: 10,
|
|
145
|
+
MAX_RANGE_SIZE: 100,
|
|
146
|
+
REQUEST_TIMEOUT: 5000,
|
|
147
|
+
RETRY_ATTEMPTS: 2,
|
|
148
|
+
RETRY_DELAY: 1000
|
|
149
|
+
},
|
|
150
|
+
REQUEST_QUEUE: {
|
|
151
|
+
ENABLED: true,
|
|
152
|
+
MAX_QUEUE_SIZE: 1,
|
|
153
|
+
MAX_ACTIVE_REQUESTS: 2
|
|
154
|
+
},
|
|
155
|
+
PLACEHOLDER: {
|
|
156
|
+
MASK_CHARACTER: "X",
|
|
157
|
+
CLASS: "viewport-item--placeholder",
|
|
158
|
+
MAX_SAMPLE_SIZE: 20,
|
|
159
|
+
PLACEHOLDER_FLAG: "_placeholder",
|
|
160
|
+
RANDOM_LENGTH_VARIANCE: true
|
|
161
|
+
},
|
|
162
|
+
SPEED_TRACKING: {
|
|
163
|
+
DECELERATION_FACTOR: 0.85
|
|
164
|
+
},
|
|
165
|
+
MOMENTUM: {
|
|
166
|
+
ENABLED: false,
|
|
167
|
+
DECELERATION_FACTOR: 0.95,
|
|
168
|
+
MIN_VELOCITY: 0.1,
|
|
169
|
+
MIN_DURATION: 750,
|
|
170
|
+
MIN_VELOCITY_THRESHOLD: 0.4,
|
|
171
|
+
FRAME_TIME: 16
|
|
172
|
+
},
|
|
173
|
+
INITIAL_LOAD: {
|
|
174
|
+
STRATEGY: "placeholders",
|
|
175
|
+
VIEWPORT_MULTIPLIER: 1.5,
|
|
176
|
+
MIN_ITEMS: 10,
|
|
177
|
+
MAX_ITEMS: 100,
|
|
178
|
+
PLACEHOLDER_COUNT: 20,
|
|
179
|
+
SHOW_LOADING_STATE: true,
|
|
180
|
+
LOADING_DELAY: 100
|
|
181
|
+
},
|
|
182
|
+
SELECTION: {
|
|
183
|
+
SELECTED_CLASS: "viewport-item--selected"
|
|
184
|
+
},
|
|
185
|
+
SCROLLBAR: {
|
|
186
|
+
CLASSES: {
|
|
187
|
+
SCROLLBAR: "viewport__scrollbar",
|
|
188
|
+
SCROLLBAR_TRACK: "viewport__scrollbar-track",
|
|
189
|
+
SCROLLBAR_THUMB: "viewport__scrollbar-thumb",
|
|
190
|
+
SCROLLBAR_VISIBLE: "viewport__scrollbar--visible",
|
|
191
|
+
SCROLLBAR_DRAGGING: "viewport__scrollbar--dragging",
|
|
192
|
+
SCROLLBAR_THUMB_DRAGGING: "viewport__scrollbar-thumb--dragging"
|
|
193
|
+
}
|
|
194
|
+
},
|
|
195
|
+
ORIENTATION: {
|
|
196
|
+
DEFAULT_ORIENTATION: "vertical",
|
|
197
|
+
DEFAULT_CROSS_AXIS_ALIGNMENT: "stretch",
|
|
198
|
+
REVERSE_DIRECTION: false
|
|
199
|
+
},
|
|
200
|
+
PAGINATION: {
|
|
201
|
+
DEFAULT_STRATEGY: "offset",
|
|
202
|
+
DEFAULT_LIMIT: 20,
|
|
203
|
+
STRATEGIES: {
|
|
204
|
+
PAGE: "page",
|
|
205
|
+
OFFSET: "offset",
|
|
206
|
+
CURSOR: "cursor"
|
|
207
|
+
},
|
|
208
|
+
CURSOR_CLEANUP_INTERVAL: 60000,
|
|
209
|
+
MAX_CURSOR_MAP_SIZE: 1000,
|
|
210
|
+
CURSOR_SCROLL_MARGIN_MULTIPLIER: 3,
|
|
211
|
+
CURSOR_MIN_VIRTUAL_SIZE_MULTIPLIER: 3
|
|
212
|
+
}
|
|
213
|
+
};
|
|
214
|
+
|
|
215
|
+
// src/core/viewport/features/virtual.ts
|
|
216
|
+
var withVirtual = (config = {}) => {
|
|
217
|
+
return (component) => {
|
|
218
|
+
const {
|
|
219
|
+
itemSize,
|
|
220
|
+
overscan = VIEWPORT_CONSTANTS.VIRTUAL_SCROLL.OVERSCAN_BUFFER,
|
|
221
|
+
orientation = "vertical",
|
|
222
|
+
autoDetectItemSize = itemSize === undefined ? VIEWPORT_CONSTANTS.VIRTUAL_SCROLL.AUTO_DETECT_ITEM_SIZE : false,
|
|
223
|
+
debug = false,
|
|
224
|
+
initialScrollIndex = 0
|
|
225
|
+
} = config;
|
|
226
|
+
const initialItemSize = itemSize || VIEWPORT_CONSTANTS.VIRTUAL_SCROLL.DEFAULT_ITEM_SIZE;
|
|
227
|
+
const MAX_VIRTUAL_SIZE = VIEWPORT_CONSTANTS.VIRTUAL_SCROLL.MAX_VIRTUAL_SIZE;
|
|
228
|
+
let viewportState;
|
|
229
|
+
let hasCalculatedItemSize = false;
|
|
230
|
+
let hasRecalculatedScrollForCompression = false;
|
|
231
|
+
let resizeObserver = null;
|
|
232
|
+
let lastScrollPosition = -1;
|
|
233
|
+
let lastVisibleRange = null;
|
|
234
|
+
let lastActualVisibleRange = null;
|
|
235
|
+
component.on?.("reload:start", () => {
|
|
236
|
+
hasCalculatedItemSize = false;
|
|
237
|
+
hasRecalculatedScrollForCompression = false;
|
|
238
|
+
lastScrollPosition = -1;
|
|
239
|
+
lastVisibleRange = null;
|
|
240
|
+
lastActualVisibleRange = null;
|
|
241
|
+
if (viewportState) {
|
|
242
|
+
viewportState.totalItems = 0;
|
|
243
|
+
viewportState.virtualTotalSize = 0;
|
|
244
|
+
viewportState.visibleRange = { start: 0, end: 0 };
|
|
245
|
+
delete viewportState.targetScrollIndex;
|
|
246
|
+
}
|
|
247
|
+
});
|
|
248
|
+
wrapInitialize(component, () => {
|
|
249
|
+
viewportState = getViewportState(component);
|
|
250
|
+
if (!viewportState)
|
|
251
|
+
return;
|
|
252
|
+
Object.assign(viewportState, {
|
|
253
|
+
itemSize: initialItemSize,
|
|
254
|
+
overscan,
|
|
255
|
+
containerSize: viewportState.viewportElement?.[orientation === "horizontal" ? "offsetWidth" : "offsetHeight"] || 600
|
|
256
|
+
});
|
|
257
|
+
updateTotalVirtualSize(viewportState.totalItems);
|
|
258
|
+
let initialScrollPosition = viewportState.scrollPosition || 0;
|
|
259
|
+
if (initialScrollIndex > 0) {
|
|
260
|
+
viewportState.targetScrollIndex = initialScrollIndex;
|
|
261
|
+
initialScrollPosition = initialScrollIndex * (viewportState.itemSize || initialItemSize);
|
|
262
|
+
viewportState.scrollPosition = initialScrollPosition;
|
|
263
|
+
setTimeout(() => {
|
|
264
|
+
component.emit?.("viewport:scroll-position-sync", {
|
|
265
|
+
position: initialScrollPosition,
|
|
266
|
+
source: "initial-scroll-index"
|
|
267
|
+
});
|
|
268
|
+
}, 0);
|
|
269
|
+
}
|
|
270
|
+
updateVisibleRange(initialScrollPosition);
|
|
271
|
+
requestAnimationFrame(() => {
|
|
272
|
+
updateContainerSize();
|
|
273
|
+
});
|
|
274
|
+
if (viewportState.viewportElement && typeof ResizeObserver !== "undefined") {
|
|
275
|
+
resizeObserver = new ResizeObserver((entries) => {
|
|
276
|
+
for (const entry of entries) {
|
|
277
|
+
const size = orientation === "horizontal" ? entry.contentRect.width : entry.contentRect.height;
|
|
278
|
+
if (size > 0 && size !== viewportState?.containerSize) {
|
|
279
|
+
updateContainerSize();
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
});
|
|
283
|
+
resizeObserver.observe(viewportState.viewportElement);
|
|
284
|
+
}
|
|
285
|
+
});
|
|
286
|
+
const getCompressionRatio = () => {
|
|
287
|
+
if (!viewportState?.virtualTotalSize)
|
|
288
|
+
return 1;
|
|
289
|
+
const actualSize = viewportState.totalItems * viewportState.itemSize;
|
|
290
|
+
return actualSize <= MAX_VIRTUAL_SIZE ? 1 : MAX_VIRTUAL_SIZE / actualSize;
|
|
291
|
+
};
|
|
292
|
+
const log = (message, data) => {
|
|
293
|
+
if (debug)
|
|
294
|
+
console.log(`[Virtual] ${message}`, data);
|
|
295
|
+
};
|
|
296
|
+
const calculateVisibleRange = (scrollPosition) => {
|
|
297
|
+
if (!viewportState) {
|
|
298
|
+
console.warn("[Virtual] No viewport state, returning empty range");
|
|
299
|
+
return { start: 0, end: 0 };
|
|
300
|
+
}
|
|
301
|
+
const { containerSize, totalItems } = viewportState;
|
|
302
|
+
if (!containerSize || containerSize <= 0 || !totalItems || totalItems <= 0) {
|
|
303
|
+
if (initialScrollIndex > 0) {
|
|
304
|
+
const visibleCount2 = Math.ceil((containerSize || 600) / (viewportState.itemSize || initialItemSize));
|
|
305
|
+
const start2 = Math.max(0, initialScrollIndex - overscan);
|
|
306
|
+
const end2 = initialScrollIndex + visibleCount2 + overscan;
|
|
307
|
+
return { start: start2, end: end2 };
|
|
308
|
+
}
|
|
309
|
+
log(`Invalid state: container=${containerSize}, items=${totalItems}`);
|
|
310
|
+
return { start: 0, end: 0 };
|
|
311
|
+
}
|
|
312
|
+
const virtualSize = viewportState.virtualTotalSize || totalItems * viewportState.itemSize;
|
|
313
|
+
const visibleCount = Math.ceil(containerSize / viewportState.itemSize);
|
|
314
|
+
const compressionRatio2 = getCompressionRatio();
|
|
315
|
+
let start, end;
|
|
316
|
+
const targetScrollIndex = viewportState.targetScrollIndex;
|
|
317
|
+
if (compressionRatio2 < 1) {
|
|
318
|
+
if (targetScrollIndex !== undefined && targetScrollIndex > 0) {
|
|
319
|
+
start = Math.max(0, targetScrollIndex - overscan);
|
|
320
|
+
end = Math.min(totalItems - 1, targetScrollIndex + visibleCount + overscan);
|
|
321
|
+
delete viewportState.targetScrollIndex;
|
|
322
|
+
} else {
|
|
323
|
+
const scrollRatio = scrollPosition / virtualSize;
|
|
324
|
+
const exactIndex = scrollRatio * totalItems;
|
|
325
|
+
start = Math.floor(exactIndex);
|
|
326
|
+
end = Math.ceil(exactIndex) + visibleCount;
|
|
327
|
+
}
|
|
328
|
+
const maxScroll = virtualSize - containerSize;
|
|
329
|
+
const distanceFromBottom = maxScroll - scrollPosition;
|
|
330
|
+
if (distanceFromBottom <= containerSize && distanceFromBottom >= -1) {
|
|
331
|
+
const itemsAtBottom = Math.floor(containerSize / viewportState.itemSize);
|
|
332
|
+
const firstVisibleAtBottom = Math.max(0, totalItems - itemsAtBottom);
|
|
333
|
+
const interpolation = Math.max(0, Math.min(1, 1 - distanceFromBottom / containerSize));
|
|
334
|
+
start = Math.floor(start + (firstVisibleAtBottom - start) * interpolation);
|
|
335
|
+
end = distanceFromBottom <= 1 ? totalItems - 1 : Math.min(totalItems - 1, start + visibleCount + overscan);
|
|
336
|
+
log("Near bottom calculation:", {
|
|
337
|
+
distanceFromBottom,
|
|
338
|
+
interpolation,
|
|
339
|
+
start,
|
|
340
|
+
end,
|
|
341
|
+
scrollPosition,
|
|
342
|
+
totalItems
|
|
343
|
+
});
|
|
344
|
+
}
|
|
345
|
+
start = Math.max(0, start - overscan);
|
|
346
|
+
end = Math.min(totalItems - 1, end + overscan);
|
|
347
|
+
} else {
|
|
348
|
+
start = Math.max(0, Math.floor(scrollPosition / viewportState.itemSize) - overscan);
|
|
349
|
+
end = Math.min(totalItems - 1, start + visibleCount + overscan * 2);
|
|
350
|
+
}
|
|
351
|
+
if (isNaN(start) || isNaN(end)) {
|
|
352
|
+
console.error("[Virtual] NaN in range calculation:", {
|
|
353
|
+
scrollPosition,
|
|
354
|
+
containerSize,
|
|
355
|
+
totalItems,
|
|
356
|
+
itemSize: viewportState.itemSize,
|
|
357
|
+
compressionRatio: compressionRatio2
|
|
358
|
+
});
|
|
359
|
+
return { start: 0, end: 0 };
|
|
360
|
+
}
|
|
361
|
+
return { start, end };
|
|
362
|
+
};
|
|
363
|
+
const calculateActualVisibleRange = (scrollPosition) => {
|
|
364
|
+
if (!viewportState)
|
|
365
|
+
return { start: 0, end: 0 };
|
|
366
|
+
const { containerSize, totalItems } = viewportState;
|
|
367
|
+
if (!containerSize || !totalItems)
|
|
368
|
+
return { start: 0, end: 0 };
|
|
369
|
+
const itemSize2 = viewportState.itemSize;
|
|
370
|
+
const visibleCount = Math.ceil(containerSize / itemSize2);
|
|
371
|
+
const actualSize = totalItems * itemSize2;
|
|
372
|
+
const virtualSize = viewportState.virtualTotalSize || actualSize;
|
|
373
|
+
const compressionRatio2 = actualSize > 0 ? virtualSize / actualSize : 1;
|
|
374
|
+
let start, end;
|
|
375
|
+
if (compressionRatio2 < 1) {
|
|
376
|
+
const scrollRatio = scrollPosition / virtualSize;
|
|
377
|
+
start = Math.floor(scrollRatio * totalItems);
|
|
378
|
+
end = Math.min(totalItems - 1, start + visibleCount - 1);
|
|
379
|
+
const maxScroll = virtualSize - containerSize;
|
|
380
|
+
const distanceFromBottom = maxScroll - scrollPosition;
|
|
381
|
+
if (distanceFromBottom <= containerSize && distanceFromBottom >= -1) {
|
|
382
|
+
const itemsAtBottom = Math.floor(containerSize / itemSize2);
|
|
383
|
+
const firstVisibleAtBottom = Math.max(0, totalItems - itemsAtBottom);
|
|
384
|
+
const interpolation = Math.max(0, Math.min(1, 1 - distanceFromBottom / containerSize));
|
|
385
|
+
start = Math.floor(start + (firstVisibleAtBottom - start) * interpolation);
|
|
386
|
+
end = distanceFromBottom <= 1 ? totalItems - 1 : Math.min(totalItems - 1, start + visibleCount - 1);
|
|
387
|
+
}
|
|
388
|
+
} else {
|
|
389
|
+
start = Math.floor(scrollPosition / itemSize2);
|
|
390
|
+
end = Math.min(totalItems - 1, start + visibleCount - 1);
|
|
391
|
+
}
|
|
392
|
+
start = Math.max(0, start);
|
|
393
|
+
end = Math.max(start, end);
|
|
394
|
+
return { start, end };
|
|
395
|
+
};
|
|
396
|
+
const updateVisibleRange = (scrollPosition) => {
|
|
397
|
+
if (!viewportState)
|
|
398
|
+
return;
|
|
399
|
+
if (lastScrollPosition >= 0 && Math.abs(scrollPosition - lastScrollPosition) < 1) {
|
|
400
|
+
return;
|
|
401
|
+
}
|
|
402
|
+
lastScrollPosition = scrollPosition;
|
|
403
|
+
viewportState.visibleRange = calculateVisibleRange(scrollPosition);
|
|
404
|
+
const actualVisibleRange = calculateActualVisibleRange(scrollPosition);
|
|
405
|
+
if (lastVisibleRange && lastActualVisibleRange && lastVisibleRange.start === viewportState.visibleRange.start && lastVisibleRange.end === viewportState.visibleRange.end && lastActualVisibleRange.start === actualVisibleRange.start && lastActualVisibleRange.end === actualVisibleRange.end) {
|
|
406
|
+
return;
|
|
407
|
+
}
|
|
408
|
+
lastVisibleRange = { ...viewportState.visibleRange };
|
|
409
|
+
lastActualVisibleRange = { ...actualVisibleRange };
|
|
410
|
+
component.emit?.("viewport:range-changed", {
|
|
411
|
+
range: viewportState.visibleRange,
|
|
412
|
+
visibleRange: actualVisibleRange,
|
|
413
|
+
scrollPosition
|
|
414
|
+
});
|
|
415
|
+
};
|
|
416
|
+
const updateTotalVirtualSize = (totalItems) => {
|
|
417
|
+
if (!viewportState)
|
|
418
|
+
return;
|
|
419
|
+
const oldSize = viewportState.virtualTotalSize;
|
|
420
|
+
viewportState.totalItems = totalItems;
|
|
421
|
+
const actualSize = totalItems * viewportState.itemSize;
|
|
422
|
+
let totalPadding = 0;
|
|
423
|
+
if (viewportState.itemsContainer) {
|
|
424
|
+
const computedStyle = window.getComputedStyle(viewportState.itemsContainer);
|
|
425
|
+
const paddingTop = parseFloat(computedStyle.paddingTop) || 0;
|
|
426
|
+
const paddingBottom = parseFloat(computedStyle.paddingBottom) || 0;
|
|
427
|
+
totalPadding = paddingTop + paddingBottom;
|
|
428
|
+
}
|
|
429
|
+
viewportState.virtualTotalSize = Math.min(actualSize + totalPadding, MAX_VIRTUAL_SIZE);
|
|
430
|
+
component.emit?.("viewport:virtual-size-changed", {
|
|
431
|
+
totalVirtualSize: viewportState.virtualTotalSize,
|
|
432
|
+
totalItems,
|
|
433
|
+
compressionRatio: getCompressionRatio()
|
|
434
|
+
});
|
|
435
|
+
};
|
|
436
|
+
const updateContainerSize = () => {
|
|
437
|
+
if (!viewportState || !viewportState.viewportElement)
|
|
438
|
+
return;
|
|
439
|
+
const size = viewportState.viewportElement[viewportState.orientation === "horizontal" ? "offsetWidth" : "offsetHeight"];
|
|
440
|
+
if (size !== viewportState.containerSize) {
|
|
441
|
+
viewportState.containerSize = size;
|
|
442
|
+
updateVisibleRange(viewportState.scrollPosition || 0);
|
|
443
|
+
updateTotalVirtualSize(viewportState.totalItems);
|
|
444
|
+
component.emit?.("viewport:container-size-changed", {
|
|
445
|
+
containerSize: size
|
|
446
|
+
});
|
|
447
|
+
}
|
|
448
|
+
};
|
|
449
|
+
component.viewport.getVisibleRange = () => viewportState?.visibleRange || calculateVisibleRange(viewportState?.scrollPosition || 0);
|
|
450
|
+
component.on?.("viewport:scroll", (data) => updateVisibleRange(data.position));
|
|
451
|
+
component.on?.("viewport:items-changed", (data) => {
|
|
452
|
+
if (data.totalItems !== undefined) {
|
|
453
|
+
updateTotalVirtualSize(data.totalItems);
|
|
454
|
+
updateVisibleRange(viewportState?.scrollPosition || 0);
|
|
455
|
+
}
|
|
456
|
+
});
|
|
457
|
+
component.on?.("viewport:total-items-changed", (data) => {
|
|
458
|
+
if (data.total === undefined)
|
|
459
|
+
return;
|
|
460
|
+
if (initialScrollIndex > 0 && !hasRecalculatedScrollForCompression && data.total > 0) {
|
|
461
|
+
const actualTotalSize = data.total * viewportState.itemSize;
|
|
462
|
+
const isCompressed = actualTotalSize > MAX_VIRTUAL_SIZE;
|
|
463
|
+
if (isCompressed) {
|
|
464
|
+
const ratio = initialScrollIndex / data.total;
|
|
465
|
+
const compressedPosition = ratio * MAX_VIRTUAL_SIZE;
|
|
466
|
+
viewportState.scrollPosition = compressedPosition;
|
|
467
|
+
component.emit?.("viewport:scroll-position-sync", {
|
|
468
|
+
position: compressedPosition,
|
|
469
|
+
source: "compression-recalculation"
|
|
470
|
+
});
|
|
471
|
+
delete viewportState.targetScrollIndex;
|
|
472
|
+
}
|
|
473
|
+
hasRecalculatedScrollForCompression = true;
|
|
474
|
+
}
|
|
475
|
+
updateTotalVirtualSize(data.total);
|
|
476
|
+
updateVisibleRange(viewportState?.scrollPosition || 0);
|
|
477
|
+
component.viewport?.renderItems?.();
|
|
478
|
+
});
|
|
479
|
+
component.on?.("viewport:container-size-changed", (data) => {
|
|
480
|
+
if (viewportState && data.containerSize !== viewportState.containerSize) {
|
|
481
|
+
viewportState.containerSize = data.containerSize;
|
|
482
|
+
updateTotalVirtualSize(viewportState.totalItems);
|
|
483
|
+
updateVisibleRange(viewportState.scrollPosition || 0);
|
|
484
|
+
}
|
|
485
|
+
});
|
|
486
|
+
component.on?.("collection:range-loaded", (data) => {
|
|
487
|
+
if (data.total !== undefined && data.total !== viewportState?.totalItems) {
|
|
488
|
+
updateTotalVirtualSize(data.total);
|
|
489
|
+
}
|
|
490
|
+
lastScrollPosition = -1;
|
|
491
|
+
updateVisibleRange(viewportState?.scrollPosition || 0);
|
|
492
|
+
component.viewport?.renderItems?.();
|
|
493
|
+
});
|
|
494
|
+
if (autoDetectItemSize) {
|
|
495
|
+
component.on?.("viewport:items-rendered", (data) => {
|
|
496
|
+
if (!hasCalculatedItemSize && data.elements?.length > 0 && viewportState) {
|
|
497
|
+
const sizes = [];
|
|
498
|
+
const sizeProperty = orientation === "horizontal" ? "offsetWidth" : "offsetHeight";
|
|
499
|
+
data.elements.forEach((element) => {
|
|
500
|
+
const size = element[sizeProperty];
|
|
501
|
+
if (size > 0) {
|
|
502
|
+
sizes.push(size);
|
|
503
|
+
}
|
|
504
|
+
});
|
|
505
|
+
if (sizes.length > 0) {
|
|
506
|
+
const avgSize = Math.round(sizes.reduce((sum, size) => sum + size, 0) / sizes.length);
|
|
507
|
+
const previousItemSize = viewportState.itemSize;
|
|
508
|
+
viewportState.itemSize = avgSize;
|
|
509
|
+
if (initialScrollIndex > 0 && avgSize !== previousItemSize) {
|
|
510
|
+
const newScrollPosition = initialScrollIndex * avgSize;
|
|
511
|
+
viewportState.scrollPosition = newScrollPosition;
|
|
512
|
+
component.emit?.("viewport:scroll-position-sync", {
|
|
513
|
+
position: newScrollPosition,
|
|
514
|
+
source: "item-size-detected"
|
|
515
|
+
});
|
|
516
|
+
}
|
|
517
|
+
updateTotalVirtualSize(viewportState.totalItems);
|
|
518
|
+
updateVisibleRange(viewportState.scrollPosition || 0);
|
|
519
|
+
component.viewport?.renderItems?.();
|
|
520
|
+
component.emit?.("viewport:item-size-detected", {
|
|
521
|
+
previousSize: initialItemSize,
|
|
522
|
+
detectedSize: avgSize
|
|
523
|
+
});
|
|
524
|
+
hasCalculatedItemSize = true;
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
});
|
|
528
|
+
}
|
|
529
|
+
wrapDestroy(component, () => {
|
|
530
|
+
if (resizeObserver) {
|
|
531
|
+
resizeObserver.disconnect();
|
|
532
|
+
resizeObserver = null;
|
|
533
|
+
}
|
|
534
|
+
});
|
|
535
|
+
const compressionRatio = getCompressionRatio();
|
|
536
|
+
component.virtual = {
|
|
537
|
+
calculateVisibleRange,
|
|
538
|
+
updateTotalVirtualSize,
|
|
539
|
+
getTotalVirtualSize: () => viewportState?.virtualTotalSize || 0,
|
|
540
|
+
getContainerSize: () => viewportState?.containerSize || 0,
|
|
541
|
+
updateContainerSize: (size) => {
|
|
542
|
+
if (viewportState) {
|
|
543
|
+
viewportState.containerSize = size;
|
|
544
|
+
updateVisibleRange(viewportState.scrollPosition || 0);
|
|
545
|
+
}
|
|
546
|
+
},
|
|
547
|
+
getItemSize: () => viewportState?.itemSize || initialItemSize,
|
|
548
|
+
calculateIndexFromPosition: (position) => Math.floor(position / ((viewportState?.itemSize || initialItemSize) * compressionRatio)),
|
|
549
|
+
calculatePositionForIndex: (index) => index * (viewportState?.itemSize || initialItemSize) * compressionRatio
|
|
550
|
+
};
|
|
551
|
+
return component;
|
|
552
|
+
};
|
|
553
|
+
};
|
|
554
|
+
|
|
555
|
+
// src/core/viewport/features/scrolling.ts
|
|
556
|
+
var import_mtrl = require("mtrl");
|
|
557
|
+
var createSpeedTracker = () => ({
|
|
558
|
+
velocity: 0,
|
|
559
|
+
lastPosition: 0,
|
|
560
|
+
lastTime: Date.now(),
|
|
561
|
+
direction: "forward",
|
|
562
|
+
samples: []
|
|
563
|
+
});
|
|
564
|
+
var updateSpeedTracker = (tracker, newPosition, previousPosition) => {
|
|
565
|
+
const now = Date.now();
|
|
566
|
+
const timeDelta = now - tracker.lastTime;
|
|
567
|
+
if (timeDelta === 0)
|
|
568
|
+
return tracker;
|
|
569
|
+
const positionDelta = newPosition - previousPosition;
|
|
570
|
+
const instantVelocity = positionDelta / timeDelta;
|
|
571
|
+
const samples = [...tracker.samples, { position: newPosition, time: now }];
|
|
572
|
+
const recentSamples = samples.filter((s) => now - s.time < 100);
|
|
573
|
+
let avgVelocity = instantVelocity;
|
|
574
|
+
if (recentSamples.length > 1) {
|
|
575
|
+
const oldestSample = recentSamples[0];
|
|
576
|
+
const totalDistance = newPosition - oldestSample.position;
|
|
577
|
+
const totalTime = now - oldestSample.time;
|
|
578
|
+
avgVelocity = totalTime > 0 ? totalDistance / totalTime : instantVelocity;
|
|
579
|
+
}
|
|
580
|
+
return {
|
|
581
|
+
velocity: avgVelocity,
|
|
582
|
+
lastPosition: newPosition,
|
|
583
|
+
lastTime: now,
|
|
584
|
+
direction: positionDelta >= 0 ? "forward" : "backward",
|
|
585
|
+
samples: recentSamples
|
|
586
|
+
};
|
|
587
|
+
};
|
|
588
|
+
var withScrolling = (config = {}) => {
|
|
589
|
+
return (component) => {
|
|
590
|
+
const {
|
|
591
|
+
orientation = "vertical",
|
|
592
|
+
sensitivity = VIEWPORT_CONSTANTS.VIRTUAL_SCROLL.SCROLL_SENSITIVITY,
|
|
593
|
+
smoothing = false,
|
|
594
|
+
idleTimeout = 100,
|
|
595
|
+
stopOnClick = true
|
|
596
|
+
} = config;
|
|
597
|
+
let scrollPosition = 0;
|
|
598
|
+
let totalVirtualSize = 0;
|
|
599
|
+
let containerSize = 0;
|
|
600
|
+
let isScrolling = false;
|
|
601
|
+
let isScrolledFromTop = false;
|
|
602
|
+
let lastScrollTime = 0;
|
|
603
|
+
const removeScrolledClass = () => {
|
|
604
|
+
const viewportEl = component.viewportElement || component._scrollingViewportElement;
|
|
605
|
+
if (viewportEl) {
|
|
606
|
+
const vlistContainer = viewportEl.closest(`.${import_mtrl.PREFIX}-vlist`);
|
|
607
|
+
if (vlistContainer) {
|
|
608
|
+
vlistContainer.classList.remove(`${import_mtrl.PREFIX}-vlist--scrolled`);
|
|
609
|
+
}
|
|
610
|
+
}
|
|
611
|
+
};
|
|
612
|
+
component.on?.("reload:start", () => {
|
|
613
|
+
scrollPosition = 0;
|
|
614
|
+
totalVirtualSize = 0;
|
|
615
|
+
isScrolling = false;
|
|
616
|
+
isScrolledFromTop = false;
|
|
617
|
+
lastScrollTime = 0;
|
|
618
|
+
speedTracker = createSpeedTracker();
|
|
619
|
+
hasEmittedIdle = false;
|
|
620
|
+
anchorPosition = null;
|
|
621
|
+
anchorTime = 0;
|
|
622
|
+
lastWheelTime = 0;
|
|
623
|
+
anchorInitialDelta = 0;
|
|
624
|
+
anchorLastDelta = 0;
|
|
625
|
+
anchorMinDelta = Infinity;
|
|
626
|
+
consecutiveIncreases = 0;
|
|
627
|
+
sustainedHighCount = 0;
|
|
628
|
+
renderScheduled = false;
|
|
629
|
+
pendingScrollData = null;
|
|
630
|
+
if (idleTimeoutId) {
|
|
631
|
+
clearTimeout(idleTimeoutId);
|
|
632
|
+
idleTimeoutId = null;
|
|
633
|
+
}
|
|
634
|
+
stopIdleDetection();
|
|
635
|
+
removeScrolledClass();
|
|
636
|
+
});
|
|
637
|
+
component.on?.("cleared", () => {
|
|
638
|
+
scrollPosition = 0;
|
|
639
|
+
totalVirtualSize = 0;
|
|
640
|
+
isScrolledFromTop = false;
|
|
641
|
+
removeScrolledClass();
|
|
642
|
+
});
|
|
643
|
+
let speedTracker = createSpeedTracker();
|
|
644
|
+
let idleTimeoutId = null;
|
|
645
|
+
let idleCheckFrame = null;
|
|
646
|
+
let lastIdleCheckPosition = 0;
|
|
647
|
+
let hasEmittedIdle = false;
|
|
648
|
+
let anchorPosition = null;
|
|
649
|
+
let anchorTime = 0;
|
|
650
|
+
let lastWheelTime = 0;
|
|
651
|
+
let anchorInitialDelta = 0;
|
|
652
|
+
let anchorLastDelta = 0;
|
|
653
|
+
let anchorMinDelta = Infinity;
|
|
654
|
+
let consecutiveIncreases = 0;
|
|
655
|
+
let sustainedHighCount = 0;
|
|
656
|
+
let renderScheduled = false;
|
|
657
|
+
let pendingScrollData = null;
|
|
658
|
+
const SCROLLED_THRESHOLD = 1;
|
|
659
|
+
const updateScrolledState = (viewportElement, position) => {
|
|
660
|
+
const shouldBeScrolled = position > SCROLLED_THRESHOLD;
|
|
661
|
+
if (shouldBeScrolled !== isScrolledFromTop) {
|
|
662
|
+
isScrolledFromTop = shouldBeScrolled;
|
|
663
|
+
const vlistContainer = viewportElement.closest(`.${import_mtrl.PREFIX}-vlist`);
|
|
664
|
+
if (vlistContainer) {
|
|
665
|
+
vlistContainer.classList.toggle(`${import_mtrl.PREFIX}-vlist--scrolled`, shouldBeScrolled);
|
|
666
|
+
}
|
|
667
|
+
}
|
|
668
|
+
};
|
|
669
|
+
let viewportState;
|
|
670
|
+
wrapInitialize(component, () => {
|
|
671
|
+
viewportState = getViewportState(component);
|
|
672
|
+
if (viewportState) {
|
|
673
|
+
totalVirtualSize = viewportState.virtualTotalSize || 0;
|
|
674
|
+
containerSize = viewportState.containerSize || 0;
|
|
675
|
+
if (viewportState.scrollPosition > 0) {
|
|
676
|
+
scrollPosition = viewportState.scrollPosition;
|
|
677
|
+
}
|
|
678
|
+
}
|
|
679
|
+
component.on?.("viewport:virtual-size-changed", (data) => {
|
|
680
|
+
updateScrollBounds(data.totalVirtualSize, containerSize);
|
|
681
|
+
});
|
|
682
|
+
component.on?.("viewport:scroll-position-sync", (data) => {
|
|
683
|
+
if (data.position !== undefined && data.position !== scrollPosition) {
|
|
684
|
+
scrollPosition = data.position;
|
|
685
|
+
totalVirtualSize = viewportState?.virtualTotalSize || totalVirtualSize;
|
|
686
|
+
containerSize = viewportState?.containerSize || containerSize;
|
|
687
|
+
}
|
|
688
|
+
});
|
|
689
|
+
component.on?.("viewport:container-size-changed", (data) => {
|
|
690
|
+
if (data.containerSize) {
|
|
691
|
+
containerSize = data.containerSize;
|
|
692
|
+
updateScrollBounds(totalVirtualSize, containerSize);
|
|
693
|
+
}
|
|
694
|
+
});
|
|
695
|
+
const viewportElement = viewportState?.viewportElement || component.viewportElement;
|
|
696
|
+
if (viewportElement) {
|
|
697
|
+
viewportElement.addEventListener("wheel", handleWheel, {
|
|
698
|
+
passive: false
|
|
699
|
+
});
|
|
700
|
+
updateScrolledState(viewportElement, scrollPosition);
|
|
701
|
+
if (stopOnClick) {
|
|
702
|
+
viewportElement.addEventListener("mousedown", handleMouseDown);
|
|
703
|
+
}
|
|
704
|
+
component._scrollingViewportElement = viewportElement;
|
|
705
|
+
component._scrollingStopOnClick = stopOnClick;
|
|
706
|
+
} else {
|
|
707
|
+
console.warn(`[Scrolling] No viewport element found for wheel events`);
|
|
708
|
+
}
|
|
709
|
+
});
|
|
710
|
+
const startIdleDetection = () => {
|
|
711
|
+
if (idleCheckFrame !== null) {
|
|
712
|
+
cancelAnimationFrame(idleCheckFrame);
|
|
713
|
+
idleCheckFrame = null;
|
|
714
|
+
}
|
|
715
|
+
hasEmittedIdle = false;
|
|
716
|
+
const checkIdle = () => {
|
|
717
|
+
if (scrollPosition === lastIdleCheckPosition) {
|
|
718
|
+
if (!hasEmittedIdle && (speedTracker.velocity > 0 || isScrolling)) {
|
|
719
|
+
hasEmittedIdle = true;
|
|
720
|
+
setVelocityToZero();
|
|
721
|
+
}
|
|
722
|
+
} else {
|
|
723
|
+
hasEmittedIdle = false;
|
|
724
|
+
}
|
|
725
|
+
lastIdleCheckPosition = scrollPosition;
|
|
726
|
+
idleCheckFrame = requestAnimationFrame(checkIdle);
|
|
727
|
+
};
|
|
728
|
+
idleCheckFrame = requestAnimationFrame(checkIdle);
|
|
729
|
+
};
|
|
730
|
+
const stopIdleDetection = () => {
|
|
731
|
+
if (idleCheckFrame) {
|
|
732
|
+
cancelAnimationFrame(idleCheckFrame);
|
|
733
|
+
idleCheckFrame = null;
|
|
734
|
+
}
|
|
735
|
+
};
|
|
736
|
+
const setVelocityToZero = () => {
|
|
737
|
+
stopIdleDetection();
|
|
738
|
+
speedTracker = createSpeedTracker();
|
|
739
|
+
isScrolling = false;
|
|
740
|
+
component.emit?.("viewport:velocity-changed", {
|
|
741
|
+
velocity: 0,
|
|
742
|
+
direction: speedTracker.direction
|
|
743
|
+
});
|
|
744
|
+
component.emit?.("viewport:idle", {
|
|
745
|
+
position: scrollPosition,
|
|
746
|
+
lastScrollTime
|
|
747
|
+
});
|
|
748
|
+
};
|
|
749
|
+
const updateContainerPosition = () => {
|
|
750
|
+
if (!viewportState || !viewportState.itemsContainer)
|
|
751
|
+
return;
|
|
752
|
+
};
|
|
753
|
+
const handleMouseDown = (_e) => {
|
|
754
|
+
anchorPosition = scrollPosition;
|
|
755
|
+
anchorTime = Date.now();
|
|
756
|
+
anchorInitialDelta = 0;
|
|
757
|
+
anchorLastDelta = 0;
|
|
758
|
+
anchorMinDelta = Infinity;
|
|
759
|
+
consecutiveIncreases = 0;
|
|
760
|
+
sustainedHighCount = 0;
|
|
761
|
+
if (isScrolling || speedTracker.velocity > 0) {
|
|
762
|
+
setVelocityToZero();
|
|
763
|
+
}
|
|
764
|
+
const momentumState = component.viewport.momentumState;
|
|
765
|
+
if (momentumState?.stopMomentum) {
|
|
766
|
+
momentumState.stopMomentum();
|
|
767
|
+
}
|
|
768
|
+
};
|
|
769
|
+
const handleWheel = (event) => {
|
|
770
|
+
event.preventDefault();
|
|
771
|
+
const delta = orientation === "vertical" ? event.deltaY : event.deltaX;
|
|
772
|
+
const scrollDelta = delta * sensitivity;
|
|
773
|
+
const now = Date.now();
|
|
774
|
+
const timeSinceLastWheel = now - lastWheelTime;
|
|
775
|
+
lastWheelTime = now;
|
|
776
|
+
if (anchorPosition !== null) {
|
|
777
|
+
const timeSinceAnchor = now - anchorTime;
|
|
778
|
+
const wheelDeltaMagnitude = Math.abs(scrollDelta);
|
|
779
|
+
if (anchorInitialDelta === 0) {
|
|
780
|
+
anchorInitialDelta = wheelDeltaMagnitude;
|
|
781
|
+
anchorLastDelta = wheelDeltaMagnitude;
|
|
782
|
+
anchorMinDelta = wheelDeltaMagnitude;
|
|
783
|
+
}
|
|
784
|
+
anchorMinDelta = Math.min(anchorMinDelta, wheelDeltaMagnitude);
|
|
785
|
+
const isBatchedEvent = wheelDeltaMagnitude > anchorLastDelta * 1.8 && wheelDeltaMagnitude < anchorLastDelta * 2.2;
|
|
786
|
+
if (!isBatchedEvent && wheelDeltaMagnitude > anchorLastDelta * 1.01) {
|
|
787
|
+
consecutiveIncreases++;
|
|
788
|
+
} else if (wheelDeltaMagnitude < anchorLastDelta * 0.99) {
|
|
789
|
+
consecutiveIncreases = 0;
|
|
790
|
+
}
|
|
791
|
+
if (!isBatchedEvent && wheelDeltaMagnitude > anchorMinDelta * 1.05) {
|
|
792
|
+
sustainedHighCount++;
|
|
793
|
+
} else {
|
|
794
|
+
sustainedHighCount = 0;
|
|
795
|
+
}
|
|
796
|
+
const sustainedIncrease = consecutiveIncreases >= 3;
|
|
797
|
+
const sustainedHigh = sustainedHighCount >= 5;
|
|
798
|
+
const significantIncrease = !isBatchedEvent && wheelDeltaMagnitude > anchorMinDelta * 1.15 && wheelDeltaMagnitude > anchorLastDelta * 1.08;
|
|
799
|
+
const deltaIncreasing = sustainedIncrease || sustainedHigh || significantIncrease;
|
|
800
|
+
const hasWheelGap = timeSinceLastWheel > 200;
|
|
801
|
+
const deltaDecayed = wheelDeltaMagnitude < anchorInitialDelta * 0.3;
|
|
802
|
+
const isLowDelta = wheelDeltaMagnitude < 30;
|
|
803
|
+
if (hasWheelGap) {
|
|
804
|
+
anchorPosition = null;
|
|
805
|
+
} else if (deltaIncreasing && timeSinceAnchor > 100) {
|
|
806
|
+
anchorPosition = null;
|
|
807
|
+
} else if (isLowDelta) {
|
|
808
|
+
anchorPosition = null;
|
|
809
|
+
} else if (deltaDecayed && timeSinceAnchor > 300) {
|
|
810
|
+
anchorPosition = null;
|
|
811
|
+
} else {
|
|
812
|
+
if (scrollPosition !== anchorPosition) {
|
|
813
|
+
scrollPosition = anchorPosition;
|
|
814
|
+
if (viewportState) {
|
|
815
|
+
viewportState.scrollPosition = scrollPosition;
|
|
816
|
+
}
|
|
817
|
+
component.viewport.renderItems();
|
|
818
|
+
}
|
|
819
|
+
anchorLastDelta = wheelDeltaMagnitude;
|
|
820
|
+
return;
|
|
821
|
+
}
|
|
822
|
+
anchorLastDelta = wheelDeltaMagnitude;
|
|
823
|
+
}
|
|
824
|
+
const previousPosition = scrollPosition;
|
|
825
|
+
const maxScroll = Math.max(0, totalVirtualSize - containerSize);
|
|
826
|
+
let newPosition = scrollPosition + scrollDelta;
|
|
827
|
+
if (smoothing) {
|
|
828
|
+
const smoothingFactor = 0.3;
|
|
829
|
+
newPosition = scrollPosition + scrollDelta * smoothingFactor;
|
|
830
|
+
}
|
|
831
|
+
newPosition = clamp(newPosition, 0, maxScroll);
|
|
832
|
+
if (newPosition !== scrollPosition) {
|
|
833
|
+
scrollPosition = newPosition;
|
|
834
|
+
const now2 = Date.now();
|
|
835
|
+
if (!isScrolling) {
|
|
836
|
+
isScrolling = true;
|
|
837
|
+
stopIdleDetection();
|
|
838
|
+
startIdleDetection();
|
|
839
|
+
}
|
|
840
|
+
lastScrollTime = now2;
|
|
841
|
+
speedTracker = updateSpeedTracker(speedTracker, scrollPosition, previousPosition);
|
|
842
|
+
if (viewportState) {
|
|
843
|
+
viewportState.scrollPosition = scrollPosition;
|
|
844
|
+
viewportState.velocity = speedTracker.velocity;
|
|
845
|
+
viewportState.scrollDirection = speedTracker.direction;
|
|
846
|
+
}
|
|
847
|
+
const viewportEl = viewportState?.viewportElement || component.viewportElement;
|
|
848
|
+
if (viewportEl) {
|
|
849
|
+
updateScrolledState(viewportEl, scrollPosition);
|
|
850
|
+
}
|
|
851
|
+
pendingScrollData = {
|
|
852
|
+
position: scrollPosition,
|
|
853
|
+
direction: speedTracker.direction,
|
|
854
|
+
previousPosition
|
|
855
|
+
};
|
|
856
|
+
if (!renderScheduled) {
|
|
857
|
+
renderScheduled = true;
|
|
858
|
+
requestAnimationFrame(() => {
|
|
859
|
+
renderScheduled = false;
|
|
860
|
+
if (pendingScrollData) {
|
|
861
|
+
component.emit?.("viewport:scroll", {
|
|
862
|
+
position: pendingScrollData.position,
|
|
863
|
+
direction: pendingScrollData.direction,
|
|
864
|
+
previousPosition: pendingScrollData.previousPosition
|
|
865
|
+
});
|
|
866
|
+
component.emit?.("viewport:velocity-changed", {
|
|
867
|
+
velocity: speedTracker.velocity,
|
|
868
|
+
direction: speedTracker.direction
|
|
869
|
+
});
|
|
870
|
+
component.viewport.renderItems();
|
|
871
|
+
pendingScrollData = null;
|
|
872
|
+
}
|
|
873
|
+
});
|
|
874
|
+
}
|
|
875
|
+
}
|
|
876
|
+
};
|
|
877
|
+
const scrollToPosition = (position, source) => {
|
|
878
|
+
const maxScroll = Math.max(0, totalVirtualSize - containerSize);
|
|
879
|
+
const clampedPosition = clamp(position, 0, maxScroll);
|
|
880
|
+
if (clampedPosition !== scrollPosition) {
|
|
881
|
+
const previousPosition = scrollPosition;
|
|
882
|
+
scrollPosition = clampedPosition;
|
|
883
|
+
speedTracker = updateSpeedTracker(speedTracker, scrollPosition, previousPosition);
|
|
884
|
+
if (viewportState) {
|
|
885
|
+
viewportState.scrollPosition = scrollPosition;
|
|
886
|
+
viewportState.velocity = speedTracker.velocity;
|
|
887
|
+
viewportState.scrollDirection = speedTracker.direction;
|
|
888
|
+
}
|
|
889
|
+
const direction = clampedPosition > previousPosition ? "forward" : "backward";
|
|
890
|
+
const viewportEl = viewportState?.viewportElement || component.viewportElement;
|
|
891
|
+
if (viewportEl) {
|
|
892
|
+
updateScrolledState(viewportEl, scrollPosition);
|
|
893
|
+
}
|
|
894
|
+
component.emit?.("viewport:scroll", {
|
|
895
|
+
position: scrollPosition,
|
|
896
|
+
direction,
|
|
897
|
+
previousPosition
|
|
898
|
+
});
|
|
899
|
+
component.emit?.("viewport:velocity-changed", {
|
|
900
|
+
velocity: speedTracker.velocity,
|
|
901
|
+
direction: speedTracker.direction
|
|
902
|
+
});
|
|
903
|
+
if (!isScrolling) {
|
|
904
|
+
isScrolling = true;
|
|
905
|
+
startIdleDetection();
|
|
906
|
+
}
|
|
907
|
+
lastScrollTime = Date.now();
|
|
908
|
+
component.viewport.renderItems();
|
|
909
|
+
} else {}
|
|
910
|
+
};
|
|
911
|
+
const scrollToIndex = (index, alignment = "start") => {
|
|
912
|
+
if (!viewportState) {
|
|
913
|
+
return;
|
|
914
|
+
}
|
|
915
|
+
const itemSize = viewportState.itemSize || 50;
|
|
916
|
+
const totalItems = viewportState.totalItems || 0;
|
|
917
|
+
const actualTotalSize = totalItems * itemSize;
|
|
918
|
+
const MAX_VIRTUAL_SIZE = VIEWPORT_CONSTANTS.VIRTUAL_SCROLL.MAX_VIRTUAL_SIZE;
|
|
919
|
+
const isCompressed = actualTotalSize > MAX_VIRTUAL_SIZE;
|
|
920
|
+
let targetPosition;
|
|
921
|
+
if (isCompressed) {
|
|
922
|
+
const ratio = index / totalItems;
|
|
923
|
+
targetPosition = ratio * Math.min(actualTotalSize, MAX_VIRTUAL_SIZE);
|
|
924
|
+
} else {
|
|
925
|
+
targetPosition = index * itemSize;
|
|
926
|
+
}
|
|
927
|
+
switch (alignment) {
|
|
928
|
+
case "center":
|
|
929
|
+
targetPosition -= containerSize / 2 - itemSize / 2;
|
|
930
|
+
break;
|
|
931
|
+
case "end":
|
|
932
|
+
targetPosition -= containerSize - itemSize;
|
|
933
|
+
break;
|
|
934
|
+
}
|
|
935
|
+
scrollToPosition(targetPosition, "scrollToIndex");
|
|
936
|
+
};
|
|
937
|
+
const scrollToPage = (page, limit = 20, alignment = "start") => {
|
|
938
|
+
if (typeof alignment !== "string" || !["start", "center", "end"].includes(alignment)) {
|
|
939
|
+
console.warn(`[Scrolling] Invalid alignment "${alignment}", using "start"`);
|
|
940
|
+
alignment = "start";
|
|
941
|
+
}
|
|
942
|
+
const viewportConfig = component.config;
|
|
943
|
+
const isCursorMode = viewportConfig?.pagination?.strategy === "cursor";
|
|
944
|
+
if (isCursorMode) {
|
|
945
|
+
const collection = component.viewport.collection;
|
|
946
|
+
if (collection) {
|
|
947
|
+
const highestLoadedPage = Math.floor(collection.getLoadedRanges().size);
|
|
948
|
+
if (page > highestLoadedPage + 1) {
|
|
949
|
+
const maxPagesToLoad = 10;
|
|
950
|
+
const targetPage = Math.min(page, highestLoadedPage + maxPagesToLoad);
|
|
951
|
+
console.warn(`[Scrolling] Cannot jump directly to page ${page} in cursor mode. ` + `Pages must be loaded sequentially. Current highest page: ${highestLoadedPage}. ` + `Will load up to page ${targetPage}`);
|
|
952
|
+
const targetOffset = (targetPage - 1) * limit;
|
|
953
|
+
const currentOffset = highestLoadedPage * limit;
|
|
954
|
+
const lastLoadedIndex = highestLoadedPage * limit;
|
|
955
|
+
scrollToIndex(lastLoadedIndex, alignment);
|
|
956
|
+
return;
|
|
957
|
+
}
|
|
958
|
+
}
|
|
959
|
+
}
|
|
960
|
+
const index = (page - 1) * limit;
|
|
961
|
+
scrollToIndex(index, alignment);
|
|
962
|
+
};
|
|
963
|
+
const updateScrollBounds = (newTotalSize, newContainerSize) => {
|
|
964
|
+
totalVirtualSize = newTotalSize;
|
|
965
|
+
containerSize = newContainerSize;
|
|
966
|
+
if (viewportState) {
|
|
967
|
+
viewportState.virtualTotalSize = newTotalSize;
|
|
968
|
+
viewportState.containerSize = newContainerSize;
|
|
969
|
+
}
|
|
970
|
+
const totalItems = viewportState?.totalItems || 0;
|
|
971
|
+
if (totalItems <= 0) {
|
|
972
|
+
return;
|
|
973
|
+
}
|
|
974
|
+
const maxScroll = Math.max(0, totalVirtualSize - containerSize);
|
|
975
|
+
if (scrollPosition > maxScroll) {
|
|
976
|
+
scrollToPosition(maxScroll);
|
|
977
|
+
}
|
|
978
|
+
};
|
|
979
|
+
const originalScrollToIndex = component.viewport.scrollToIndex;
|
|
980
|
+
component.viewport.scrollToIndex = (index, alignment) => {
|
|
981
|
+
scrollToIndex(index, alignment);
|
|
982
|
+
originalScrollToIndex?.(index, alignment);
|
|
983
|
+
};
|
|
984
|
+
component.viewport.scrollToPage = (page, limit, alignment) => {
|
|
985
|
+
scrollToPage(page, limit, alignment);
|
|
986
|
+
};
|
|
987
|
+
const originalScrollToPosition = component.viewport.scrollToPosition;
|
|
988
|
+
component.viewport.scrollToPosition = (position) => {
|
|
989
|
+
scrollToPosition(position, "api");
|
|
990
|
+
originalScrollToPosition?.(position);
|
|
991
|
+
};
|
|
992
|
+
const originalGetScrollPosition = component.viewport.getScrollPosition;
|
|
993
|
+
component.viewport.getScrollPosition = () => {
|
|
994
|
+
return scrollPosition;
|
|
995
|
+
};
|
|
996
|
+
if ("destroy" in component && typeof component.destroy === "function") {
|
|
997
|
+
const originalDestroy = component.destroy;
|
|
998
|
+
component.destroy = () => {
|
|
999
|
+
const viewportElement = component._scrollingViewportElement;
|
|
1000
|
+
if (viewportElement) {
|
|
1001
|
+
viewportElement.removeEventListener("wheel", handleWheel);
|
|
1002
|
+
if (component._scrollingStopOnClick) {
|
|
1003
|
+
viewportElement.removeEventListener("mousedown", handleMouseDown);
|
|
1004
|
+
}
|
|
1005
|
+
}
|
|
1006
|
+
if (idleTimeoutId) {
|
|
1007
|
+
clearTimeout(idleTimeoutId);
|
|
1008
|
+
}
|
|
1009
|
+
stopIdleDetection();
|
|
1010
|
+
originalDestroy?.();
|
|
1011
|
+
};
|
|
1012
|
+
}
|
|
1013
|
+
const scrollBy = (delta) => {
|
|
1014
|
+
const previousPosition = scrollPosition;
|
|
1015
|
+
const maxScroll = Math.max(0, totalVirtualSize - containerSize);
|
|
1016
|
+
const newPosition = clamp(scrollPosition + delta, 0, maxScroll);
|
|
1017
|
+
if (newPosition !== previousPosition) {
|
|
1018
|
+
scrollPosition = newPosition;
|
|
1019
|
+
speedTracker = updateSpeedTracker(speedTracker, scrollPosition, previousPosition);
|
|
1020
|
+
if (viewportState) {
|
|
1021
|
+
viewportState.scrollPosition = scrollPosition;
|
|
1022
|
+
viewportState.velocity = speedTracker.velocity;
|
|
1023
|
+
}
|
|
1024
|
+
const viewportEl = viewportState?.viewportElement || component.viewportElement;
|
|
1025
|
+
if (viewportEl) {
|
|
1026
|
+
updateScrolledState(viewportEl, scrollPosition);
|
|
1027
|
+
}
|
|
1028
|
+
component.emit?.("viewport:scroll", {
|
|
1029
|
+
position: scrollPosition,
|
|
1030
|
+
velocity: speedTracker.velocity,
|
|
1031
|
+
direction: speedTracker.direction
|
|
1032
|
+
});
|
|
1033
|
+
component.viewport.renderItems?.();
|
|
1034
|
+
if (!isScrolling) {
|
|
1035
|
+
isScrolling = true;
|
|
1036
|
+
startIdleDetection();
|
|
1037
|
+
}
|
|
1038
|
+
lastScrollTime = Date.now();
|
|
1039
|
+
}
|
|
1040
|
+
};
|
|
1041
|
+
component.viewport.scrollingState = {
|
|
1042
|
+
setVelocityToZero
|
|
1043
|
+
};
|
|
1044
|
+
component.viewport.scrollBy = scrollBy;
|
|
1045
|
+
component.viewport.getVelocity = () => speedTracker.velocity;
|
|
1046
|
+
component.scrolling = {
|
|
1047
|
+
handleWheel,
|
|
1048
|
+
scrollToPosition,
|
|
1049
|
+
scrollToIndex,
|
|
1050
|
+
scrollToPage,
|
|
1051
|
+
scrollBy,
|
|
1052
|
+
getScrollPosition: () => scrollPosition,
|
|
1053
|
+
updateScrollBounds,
|
|
1054
|
+
getVelocity: () => speedTracker.velocity,
|
|
1055
|
+
getDirection: () => speedTracker.direction,
|
|
1056
|
+
isScrolling: () => isScrolling
|
|
1057
|
+
};
|
|
1058
|
+
return component;
|
|
1059
|
+
};
|
|
1060
|
+
};
|
|
1061
|
+
|
|
1062
|
+
// src/core/viewport/features/momentum.ts
|
|
1063
|
+
var withMomentum = (config = {}) => {
|
|
1064
|
+
return (component) => {
|
|
1065
|
+
const {
|
|
1066
|
+
enabled = VIEWPORT_CONSTANTS.MOMENTUM.ENABLED,
|
|
1067
|
+
deceleration = VIEWPORT_CONSTANTS.MOMENTUM.DECELERATION_FACTOR,
|
|
1068
|
+
minVelocity = VIEWPORT_CONSTANTS.MOMENTUM.MIN_VELOCITY,
|
|
1069
|
+
minDuration = VIEWPORT_CONSTANTS.MOMENTUM.MIN_DURATION,
|
|
1070
|
+
minVelocityThreshold = VIEWPORT_CONSTANTS.MOMENTUM.MIN_VELOCITY_THRESHOLD,
|
|
1071
|
+
stopOnClick = true
|
|
1072
|
+
} = config;
|
|
1073
|
+
let momentumAnimationId = null;
|
|
1074
|
+
let isTouching = false;
|
|
1075
|
+
let isMouseDragging = false;
|
|
1076
|
+
let lastTouchPosition = 0;
|
|
1077
|
+
let lastMousePosition = 0;
|
|
1078
|
+
let touchStartTime = 0;
|
|
1079
|
+
let viewportState;
|
|
1080
|
+
let scrollingState;
|
|
1081
|
+
let isEnabled = enabled;
|
|
1082
|
+
const originalInitialize = component.viewport.initialize;
|
|
1083
|
+
component.viewport.initialize = () => {
|
|
1084
|
+
const result = originalInitialize();
|
|
1085
|
+
if (result === false) {
|
|
1086
|
+
return false;
|
|
1087
|
+
}
|
|
1088
|
+
viewportState = component.viewport.state;
|
|
1089
|
+
scrollingState = component.viewport.scrollingState;
|
|
1090
|
+
return result;
|
|
1091
|
+
};
|
|
1092
|
+
const stopMomentum = () => {
|
|
1093
|
+
if (momentumAnimationId) {
|
|
1094
|
+
cancelAnimationFrame(momentumAnimationId);
|
|
1095
|
+
momentumAnimationId = null;
|
|
1096
|
+
if (scrollingState) {
|
|
1097
|
+
scrollingState.setVelocityToZero?.();
|
|
1098
|
+
}
|
|
1099
|
+
}
|
|
1100
|
+
};
|
|
1101
|
+
const startMomentum = (initialVelocity) => {
|
|
1102
|
+
if (momentumAnimationId) {
|
|
1103
|
+
cancelAnimationFrame(momentumAnimationId);
|
|
1104
|
+
}
|
|
1105
|
+
let velocity = initialVelocity;
|
|
1106
|
+
const animate = () => {
|
|
1107
|
+
velocity *= deceleration;
|
|
1108
|
+
if (Math.abs(velocity) < minVelocity) {
|
|
1109
|
+
momentumAnimationId = null;
|
|
1110
|
+
if (scrollingState) {
|
|
1111
|
+
scrollingState.setVelocityToZero?.();
|
|
1112
|
+
}
|
|
1113
|
+
return;
|
|
1114
|
+
}
|
|
1115
|
+
const frameDelta = velocity * VIEWPORT_CONSTANTS.MOMENTUM.FRAME_TIME;
|
|
1116
|
+
if (component.viewport.scrollBy) {
|
|
1117
|
+
component.viewport.scrollBy(frameDelta);
|
|
1118
|
+
momentumAnimationId = requestAnimationFrame(animate);
|
|
1119
|
+
} else {
|
|
1120
|
+
momentumAnimationId = null;
|
|
1121
|
+
}
|
|
1122
|
+
};
|
|
1123
|
+
momentumAnimationId = requestAnimationFrame(animate);
|
|
1124
|
+
};
|
|
1125
|
+
const handleTouchStart = (e) => {
|
|
1126
|
+
isTouching = true;
|
|
1127
|
+
const touch = e.touches[0];
|
|
1128
|
+
const orientation = viewportState?.orientation || "vertical";
|
|
1129
|
+
lastTouchPosition = orientation === "vertical" ? touch.clientY : touch.clientX;
|
|
1130
|
+
touchStartTime = Date.now();
|
|
1131
|
+
if (momentumAnimationId) {
|
|
1132
|
+
cancelAnimationFrame(momentumAnimationId);
|
|
1133
|
+
momentumAnimationId = null;
|
|
1134
|
+
}
|
|
1135
|
+
};
|
|
1136
|
+
const handleTouchMove = (e) => {
|
|
1137
|
+
if (!isTouching)
|
|
1138
|
+
return;
|
|
1139
|
+
e.preventDefault();
|
|
1140
|
+
const touch = e.touches[0];
|
|
1141
|
+
const orientation = viewportState?.orientation || "vertical";
|
|
1142
|
+
const currentPosition = orientation === "vertical" ? touch.clientY : touch.clientX;
|
|
1143
|
+
const delta = lastTouchPosition - currentPosition;
|
|
1144
|
+
lastTouchPosition = currentPosition;
|
|
1145
|
+
if (component.viewport.scrollBy) {
|
|
1146
|
+
component.viewport.scrollBy(delta);
|
|
1147
|
+
}
|
|
1148
|
+
};
|
|
1149
|
+
const handleTouchEnd = () => {
|
|
1150
|
+
if (!isTouching)
|
|
1151
|
+
return;
|
|
1152
|
+
isTouching = false;
|
|
1153
|
+
const touchEndTime = Date.now();
|
|
1154
|
+
const touchDuration = touchEndTime - touchStartTime;
|
|
1155
|
+
const velocity = component.viewport.getVelocity?.() || 0;
|
|
1156
|
+
if (touchDuration < minDuration && Math.abs(velocity) > minVelocityThreshold) {
|
|
1157
|
+
startMomentum(velocity);
|
|
1158
|
+
}
|
|
1159
|
+
};
|
|
1160
|
+
const handleClick = (_e) => {
|
|
1161
|
+
stopMomentum();
|
|
1162
|
+
};
|
|
1163
|
+
const handleMouseDown = (e) => {
|
|
1164
|
+
isMouseDragging = true;
|
|
1165
|
+
const orientation = viewportState?.orientation || "vertical";
|
|
1166
|
+
lastMousePosition = orientation === "vertical" ? e.clientY : e.clientX;
|
|
1167
|
+
touchStartTime = Date.now();
|
|
1168
|
+
stopMomentum();
|
|
1169
|
+
e.preventDefault();
|
|
1170
|
+
};
|
|
1171
|
+
const handleMouseMove = (e) => {
|
|
1172
|
+
if (!isMouseDragging)
|
|
1173
|
+
return;
|
|
1174
|
+
const orientation = viewportState?.orientation || "vertical";
|
|
1175
|
+
const currentPosition = orientation === "vertical" ? e.clientY : e.clientX;
|
|
1176
|
+
const delta = lastMousePosition - currentPosition;
|
|
1177
|
+
lastMousePosition = currentPosition;
|
|
1178
|
+
if (component.viewport.scrollBy) {
|
|
1179
|
+
component.viewport.scrollBy(delta);
|
|
1180
|
+
}
|
|
1181
|
+
};
|
|
1182
|
+
const handleMouseUp = () => {
|
|
1183
|
+
if (!isMouseDragging)
|
|
1184
|
+
return;
|
|
1185
|
+
isMouseDragging = false;
|
|
1186
|
+
const mouseUpTime = Date.now();
|
|
1187
|
+
const dragDuration = mouseUpTime - touchStartTime;
|
|
1188
|
+
const velocity = component.viewport.getVelocity?.() || 0;
|
|
1189
|
+
if (dragDuration < minDuration && Math.abs(velocity) > minVelocityThreshold) {
|
|
1190
|
+
startMomentum(velocity);
|
|
1191
|
+
}
|
|
1192
|
+
};
|
|
1193
|
+
const originalInit = component.viewport.initialize;
|
|
1194
|
+
component.viewport.initialize = () => {
|
|
1195
|
+
const result = originalInit();
|
|
1196
|
+
if (result === false) {
|
|
1197
|
+
return false;
|
|
1198
|
+
}
|
|
1199
|
+
const viewportElement = component.viewportElement || component.viewport.state?.viewportElement;
|
|
1200
|
+
if (viewportElement) {
|
|
1201
|
+
viewportElement.addEventListener("touchstart", handleTouchStart, {
|
|
1202
|
+
passive: true
|
|
1203
|
+
});
|
|
1204
|
+
viewportElement.addEventListener("touchmove", handleTouchMove, {
|
|
1205
|
+
passive: false
|
|
1206
|
+
});
|
|
1207
|
+
viewportElement.addEventListener("touchend", handleTouchEnd, {
|
|
1208
|
+
passive: true
|
|
1209
|
+
});
|
|
1210
|
+
viewportElement.addEventListener("mousedown", handleMouseDown);
|
|
1211
|
+
viewportElement.addEventListener("mousemove", handleMouseMove);
|
|
1212
|
+
viewportElement.addEventListener("mouseup", handleMouseUp);
|
|
1213
|
+
viewportElement.addEventListener("mouseleave", handleMouseUp);
|
|
1214
|
+
if (stopOnClick) {
|
|
1215
|
+
viewportElement.addEventListener("click", handleClick);
|
|
1216
|
+
}
|
|
1217
|
+
component._momentumViewportElement = viewportElement;
|
|
1218
|
+
component._momentumStopOnClick = stopOnClick;
|
|
1219
|
+
}
|
|
1220
|
+
};
|
|
1221
|
+
if ("destroy" in component && typeof component.destroy === "function") {
|
|
1222
|
+
const originalDestroy = component.destroy;
|
|
1223
|
+
component.destroy = () => {
|
|
1224
|
+
const viewportElement = component._momentumViewportElement;
|
|
1225
|
+
if (viewportElement) {
|
|
1226
|
+
viewportElement.removeEventListener("touchstart", handleTouchStart);
|
|
1227
|
+
viewportElement.removeEventListener("touchmove", handleTouchMove);
|
|
1228
|
+
viewportElement.removeEventListener("touchend", handleTouchEnd);
|
|
1229
|
+
viewportElement.removeEventListener("mousedown", handleMouseDown);
|
|
1230
|
+
viewportElement.removeEventListener("mousemove", handleMouseMove);
|
|
1231
|
+
viewportElement.removeEventListener("mouseup", handleMouseUp);
|
|
1232
|
+
viewportElement.removeEventListener("mouseleave", handleMouseUp);
|
|
1233
|
+
if (component._momentumStopOnClick) {
|
|
1234
|
+
viewportElement.removeEventListener("click", handleClick);
|
|
1235
|
+
}
|
|
1236
|
+
}
|
|
1237
|
+
stopMomentum();
|
|
1238
|
+
originalDestroy();
|
|
1239
|
+
};
|
|
1240
|
+
}
|
|
1241
|
+
component.viewport.momentumState = {
|
|
1242
|
+
stopMomentum
|
|
1243
|
+
};
|
|
1244
|
+
return component;
|
|
1245
|
+
};
|
|
1246
|
+
};
|
|
1247
|
+
|
|
1248
|
+
// src/core/viewport/features/scrollbar.ts
|
|
1249
|
+
var import_mtrl2 = require("mtrl");
|
|
1250
|
+
function withScrollbar(config = {}) {
|
|
1251
|
+
return (component) => {
|
|
1252
|
+
const {
|
|
1253
|
+
enabled = true,
|
|
1254
|
+
autoHide = true,
|
|
1255
|
+
thumbMinHeight = 25,
|
|
1256
|
+
borderRadius = 4,
|
|
1257
|
+
fadeTimeout = 1000
|
|
1258
|
+
} = config;
|
|
1259
|
+
if (!enabled) {
|
|
1260
|
+
return {
|
|
1261
|
+
...component,
|
|
1262
|
+
scrollbar: {
|
|
1263
|
+
show: () => {},
|
|
1264
|
+
hide: () => {},
|
|
1265
|
+
updateBounds: () => {},
|
|
1266
|
+
updatePosition: () => {}
|
|
1267
|
+
}
|
|
1268
|
+
};
|
|
1269
|
+
}
|
|
1270
|
+
let viewportElement = null;
|
|
1271
|
+
let scrollbarTrack = null;
|
|
1272
|
+
let scrollbarThumb = null;
|
|
1273
|
+
let isInitialized = false;
|
|
1274
|
+
let isDragging = false;
|
|
1275
|
+
let dragStartY = 0;
|
|
1276
|
+
let dragStartScrollPosition = 0;
|
|
1277
|
+
let hideTimeout = null;
|
|
1278
|
+
let totalVirtualSize = 0;
|
|
1279
|
+
let containerSize = 0;
|
|
1280
|
+
let thumbHeight = 0;
|
|
1281
|
+
let animationFrameId = null;
|
|
1282
|
+
let lastRequestedScrollPosition = null;
|
|
1283
|
+
let isCursorMode = false;
|
|
1284
|
+
let loadedItemsCount = 0;
|
|
1285
|
+
const createScrollbarElements = () => {
|
|
1286
|
+
if (!viewportElement)
|
|
1287
|
+
return;
|
|
1288
|
+
scrollbarTrack = document.createElement("div");
|
|
1289
|
+
import_mtrl2.addClass(scrollbarTrack, VIEWPORT_CONSTANTS.SCROLLBAR.CLASSES.SCROLLBAR);
|
|
1290
|
+
scrollbarTrack.style.cssText = `
|
|
1291
|
+
position: absolute;
|
|
1292
|
+
top: 0;
|
|
1293
|
+
right: 0;
|
|
1294
|
+
height: 100%;
|
|
1295
|
+
z-index: 10;
|
|
1296
|
+
`;
|
|
1297
|
+
scrollbarThumb = document.createElement("div");
|
|
1298
|
+
import_mtrl2.addClass(scrollbarThumb, VIEWPORT_CONSTANTS.SCROLLBAR.CLASSES.SCROLLBAR_THUMB);
|
|
1299
|
+
scrollbarTrack.appendChild(scrollbarThumb);
|
|
1300
|
+
viewportElement.appendChild(scrollbarTrack);
|
|
1301
|
+
};
|
|
1302
|
+
const show = () => {
|
|
1303
|
+
if (!scrollbarTrack || totalVirtualSize <= containerSize)
|
|
1304
|
+
return;
|
|
1305
|
+
if (hideTimeout) {
|
|
1306
|
+
clearTimeout(hideTimeout);
|
|
1307
|
+
hideTimeout = null;
|
|
1308
|
+
}
|
|
1309
|
+
import_mtrl2.addClass(scrollbarTrack, VIEWPORT_CONSTANTS.SCROLLBAR.CLASSES.SCROLLBAR_VISIBLE);
|
|
1310
|
+
if (autoHide && !isDragging) {
|
|
1311
|
+
hideTimeout = setTimeout(hide, fadeTimeout);
|
|
1312
|
+
}
|
|
1313
|
+
};
|
|
1314
|
+
const hide = () => {
|
|
1315
|
+
if (!scrollbarTrack || isDragging)
|
|
1316
|
+
return;
|
|
1317
|
+
import_mtrl2.removeClass(scrollbarTrack, VIEWPORT_CONSTANTS.SCROLLBAR.CLASSES.SCROLLBAR_VISIBLE);
|
|
1318
|
+
};
|
|
1319
|
+
const updateBounds = (newTotalSize, newContainerSize) => {
|
|
1320
|
+
totalVirtualSize = newTotalSize;
|
|
1321
|
+
containerSize = newContainerSize;
|
|
1322
|
+
if (!scrollbarTrack || !scrollbarThumb)
|
|
1323
|
+
return;
|
|
1324
|
+
const needsScrollbar = totalVirtualSize > containerSize;
|
|
1325
|
+
scrollbarTrack.style.display = needsScrollbar ? "block" : "none";
|
|
1326
|
+
if (needsScrollbar) {
|
|
1327
|
+
let scrollRatio;
|
|
1328
|
+
if (isCursorMode && loadedItemsCount > 0) {
|
|
1329
|
+
const estimatedTotal = totalVirtualSize > 0 ? totalVirtualSize : loadedItemsCount * 2;
|
|
1330
|
+
scrollRatio = containerSize / Math.max(estimatedTotal, loadedItemsCount + containerSize);
|
|
1331
|
+
} else {
|
|
1332
|
+
scrollRatio = containerSize / totalVirtualSize;
|
|
1333
|
+
}
|
|
1334
|
+
thumbHeight = Math.max(thumbMinHeight, scrollRatio * containerSize);
|
|
1335
|
+
scrollbarThumb.style.height = `${thumbHeight}px`;
|
|
1336
|
+
updatePosition(component.viewport?.getScrollPosition() || 0);
|
|
1337
|
+
}
|
|
1338
|
+
};
|
|
1339
|
+
const updatePosition = (scrollPos) => {
|
|
1340
|
+
if (!scrollbarThumb || !scrollbarTrack)
|
|
1341
|
+
return;
|
|
1342
|
+
const scrollableDistance = totalVirtualSize - containerSize;
|
|
1343
|
+
if (scrollableDistance <= 0)
|
|
1344
|
+
return;
|
|
1345
|
+
const scrollRatio = Math.min(1, Math.max(0, scrollPos / scrollableDistance));
|
|
1346
|
+
const maxThumbPosition = scrollbarTrack.clientHeight - scrollbarThumb.clientHeight;
|
|
1347
|
+
const thumbPosition = scrollRatio * maxThumbPosition;
|
|
1348
|
+
scrollbarThumb.style.transform = `translateY(${thumbPosition}px)`;
|
|
1349
|
+
};
|
|
1350
|
+
const handleTrackClick = (e) => {
|
|
1351
|
+
if (!scrollbarTrack || !scrollbarThumb || e.target === scrollbarThumb)
|
|
1352
|
+
return;
|
|
1353
|
+
const trackRect = scrollbarTrack.getBoundingClientRect();
|
|
1354
|
+
const clickY = e.clientY - trackRect.top;
|
|
1355
|
+
const thumbCenterY = clickY - thumbHeight / 2;
|
|
1356
|
+
const maxThumbPosition = containerSize - thumbHeight;
|
|
1357
|
+
const thumbPosition = Math.max(0, Math.min(thumbCenterY, maxThumbPosition));
|
|
1358
|
+
const scrollRatio = thumbPosition / maxThumbPosition;
|
|
1359
|
+
const targetScrollPosition = scrollRatio * (totalVirtualSize - containerSize);
|
|
1360
|
+
component.viewport?.scrollToPosition(targetScrollPosition);
|
|
1361
|
+
};
|
|
1362
|
+
const handleThumbMouseDown = (e) => {
|
|
1363
|
+
e.preventDefault();
|
|
1364
|
+
e.stopPropagation();
|
|
1365
|
+
isDragging = true;
|
|
1366
|
+
dragStartY = e.clientY;
|
|
1367
|
+
dragStartScrollPosition = component.viewport?.getScrollPosition() || 0;
|
|
1368
|
+
if (scrollbarTrack) {
|
|
1369
|
+
import_mtrl2.addClass(scrollbarTrack, VIEWPORT_CONSTANTS.SCROLLBAR.CLASSES.SCROLLBAR_DRAGGING);
|
|
1370
|
+
}
|
|
1371
|
+
component.emit?.("viewport:drag-start", {
|
|
1372
|
+
source: "scrollbar",
|
|
1373
|
+
startPosition: dragStartScrollPosition
|
|
1374
|
+
});
|
|
1375
|
+
document.addEventListener("mousemove", handleMouseMove);
|
|
1376
|
+
document.addEventListener("mouseup", handleMouseUp);
|
|
1377
|
+
};
|
|
1378
|
+
const handleMouseMove = (e) => {
|
|
1379
|
+
if (!isDragging || !scrollbarTrack || !scrollbarThumb)
|
|
1380
|
+
return;
|
|
1381
|
+
const deltaY = e.clientY - dragStartY;
|
|
1382
|
+
const trackHeight = scrollbarTrack.clientHeight;
|
|
1383
|
+
const thumbHeight2 = scrollbarThumb.clientHeight;
|
|
1384
|
+
const maxThumbTravel = trackHeight - thumbHeight2;
|
|
1385
|
+
if (maxThumbTravel <= 0)
|
|
1386
|
+
return;
|
|
1387
|
+
const deltaRatio = deltaY / maxThumbTravel;
|
|
1388
|
+
const dragStartScrollRatio = dragStartScrollPosition / (totalVirtualSize - containerSize);
|
|
1389
|
+
const newScrollRatio = Math.max(0, Math.min(1, dragStartScrollRatio + deltaRatio));
|
|
1390
|
+
const thumbPosition = newScrollRatio * maxThumbTravel;
|
|
1391
|
+
scrollbarThumb.style.transform = `translateY(${thumbPosition}px)`;
|
|
1392
|
+
const newPosition = newScrollRatio * (totalVirtualSize - containerSize);
|
|
1393
|
+
lastRequestedScrollPosition = newPosition;
|
|
1394
|
+
if (animationFrameId === null && component.viewport) {
|
|
1395
|
+
animationFrameId = requestAnimationFrame(() => {
|
|
1396
|
+
if (lastRequestedScrollPosition !== null && component.viewport) {
|
|
1397
|
+
component.viewport.scrollToPosition(lastRequestedScrollPosition);
|
|
1398
|
+
}
|
|
1399
|
+
animationFrameId = null;
|
|
1400
|
+
});
|
|
1401
|
+
}
|
|
1402
|
+
};
|
|
1403
|
+
const handleMouseUp = () => {
|
|
1404
|
+
isDragging = false;
|
|
1405
|
+
if (animationFrameId !== null) {
|
|
1406
|
+
cancelAnimationFrame(animationFrameId);
|
|
1407
|
+
animationFrameId = null;
|
|
1408
|
+
}
|
|
1409
|
+
if (lastRequestedScrollPosition !== null && component.viewport) {
|
|
1410
|
+
component.viewport.scrollToPosition(lastRequestedScrollPosition);
|
|
1411
|
+
lastRequestedScrollPosition = null;
|
|
1412
|
+
}
|
|
1413
|
+
if (scrollbarTrack) {
|
|
1414
|
+
import_mtrl2.removeClass(scrollbarTrack, VIEWPORT_CONSTANTS.SCROLLBAR.CLASSES.SCROLLBAR_DRAGGING);
|
|
1415
|
+
}
|
|
1416
|
+
component.emit?.("viewport:drag-end", {
|
|
1417
|
+
source: "scrollbar",
|
|
1418
|
+
endPosition: component.viewport?.getScrollPosition() || 0
|
|
1419
|
+
});
|
|
1420
|
+
hide();
|
|
1421
|
+
document.removeEventListener("mousemove", handleMouseMove);
|
|
1422
|
+
document.removeEventListener("mouseup", handleMouseUp);
|
|
1423
|
+
};
|
|
1424
|
+
const initialize = () => {
|
|
1425
|
+
if (isInitialized)
|
|
1426
|
+
return;
|
|
1427
|
+
viewportElement = component.element?.querySelector(`.${import_mtrl2.PREFIX}-viewport`);
|
|
1428
|
+
if (!viewportElement) {
|
|
1429
|
+
console.warn("[Scrollbar] No viewport element found");
|
|
1430
|
+
return;
|
|
1431
|
+
}
|
|
1432
|
+
if (viewportElement.querySelector(`.${import_mtrl2.PREFIX}-${VIEWPORT_CONSTANTS.SCROLLBAR.CLASSES.SCROLLBAR}`))
|
|
1433
|
+
return;
|
|
1434
|
+
createScrollbarElements();
|
|
1435
|
+
isInitialized = true;
|
|
1436
|
+
if (scrollbarTrack && viewportElement) {
|
|
1437
|
+
scrollbarTrack.addEventListener("click", handleTrackClick);
|
|
1438
|
+
if (scrollbarThumb) {
|
|
1439
|
+
scrollbarThumb.addEventListener("mousedown", handleThumbMouseDown);
|
|
1440
|
+
}
|
|
1441
|
+
viewportElement.addEventListener("mouseenter", show, { passive: true });
|
|
1442
|
+
viewportElement.addEventListener("mouseleave", () => {
|
|
1443
|
+
if (!isDragging)
|
|
1444
|
+
hide();
|
|
1445
|
+
}, { passive: true });
|
|
1446
|
+
}
|
|
1447
|
+
if (component.viewport) {
|
|
1448
|
+
const info = component.viewport.getViewportInfo();
|
|
1449
|
+
totalVirtualSize = info.totalVirtualSize;
|
|
1450
|
+
containerSize = info.containerSize;
|
|
1451
|
+
const collection = component.collection;
|
|
1452
|
+
const viewportConfig = component.config;
|
|
1453
|
+
isCursorMode = viewportConfig?.pagination?.strategy === "cursor";
|
|
1454
|
+
updateBounds(totalVirtualSize, containerSize);
|
|
1455
|
+
updatePosition(component.viewport.getScrollPosition());
|
|
1456
|
+
}
|
|
1457
|
+
component.on?.("viewport:scroll", (data) => {
|
|
1458
|
+
if (!isDragging) {
|
|
1459
|
+
updatePosition(data.position);
|
|
1460
|
+
show();
|
|
1461
|
+
}
|
|
1462
|
+
});
|
|
1463
|
+
component.on?.("viewport:virtual-size-changed", (data) => {
|
|
1464
|
+
totalVirtualSize = data.totalVirtualSize;
|
|
1465
|
+
updateBounds(data.totalVirtualSize, containerSize);
|
|
1466
|
+
});
|
|
1467
|
+
component.on?.("viewport:container-size-changed", (data) => {
|
|
1468
|
+
containerSize = data.containerSize;
|
|
1469
|
+
updateBounds(totalVirtualSize, data.containerSize);
|
|
1470
|
+
});
|
|
1471
|
+
component.on?.("viewport:items-changed", (data) => {
|
|
1472
|
+
if (isCursorMode) {
|
|
1473
|
+
loadedItemsCount = data.loadedCount || 0;
|
|
1474
|
+
updateBounds(totalVirtualSize, containerSize);
|
|
1475
|
+
}
|
|
1476
|
+
});
|
|
1477
|
+
component.on?.("viewport:total-items-changed", (data) => {
|
|
1478
|
+
if (isCursorMode && data.total) {
|
|
1479
|
+
console.log(`[Scrollbar] Cursor mode: updating bounds for new total ${data.total}`);
|
|
1480
|
+
updateBounds(totalVirtualSize, containerSize);
|
|
1481
|
+
}
|
|
1482
|
+
});
|
|
1483
|
+
};
|
|
1484
|
+
storeFeatureFunction(component, "_scrollbarInitialize", initialize);
|
|
1485
|
+
wrapInitialize(component, initialize);
|
|
1486
|
+
const destroy = () => {
|
|
1487
|
+
if (scrollbarTrack) {
|
|
1488
|
+
scrollbarTrack.remove();
|
|
1489
|
+
scrollbarTrack = null;
|
|
1490
|
+
}
|
|
1491
|
+
scrollbarThumb = null;
|
|
1492
|
+
viewportElement = null;
|
|
1493
|
+
isInitialized = false;
|
|
1494
|
+
if (hideTimeout) {
|
|
1495
|
+
clearTimeout(hideTimeout);
|
|
1496
|
+
hideTimeout = null;
|
|
1497
|
+
}
|
|
1498
|
+
document.removeEventListener("mousemove", handleMouseMove);
|
|
1499
|
+
document.removeEventListener("mouseup", handleMouseUp);
|
|
1500
|
+
};
|
|
1501
|
+
storeFeatureFunction(component, "_scrollbarDestroy", destroy);
|
|
1502
|
+
wrapDestroy(component, destroy);
|
|
1503
|
+
return {
|
|
1504
|
+
...component,
|
|
1505
|
+
scrollbar: { show, hide, updateBounds, updatePosition }
|
|
1506
|
+
};
|
|
1507
|
+
};
|
|
1508
|
+
}
|
|
1509
|
+
|
|
1510
|
+
// src/core/viewport/features/collection.ts
|
|
1511
|
+
function withCollection(config = {}) {
|
|
1512
|
+
return (component) => {
|
|
1513
|
+
const {
|
|
1514
|
+
collection,
|
|
1515
|
+
rangeSize = VIEWPORT_CONSTANTS.LOADING.DEFAULT_RANGE_SIZE,
|
|
1516
|
+
strategy = "offset",
|
|
1517
|
+
transform,
|
|
1518
|
+
cancelLoadThreshold = VIEWPORT_CONSTANTS.LOADING.CANCEL_THRESHOLD,
|
|
1519
|
+
maxConcurrentRequests = VIEWPORT_CONSTANTS.LOADING.MAX_CONCURRENT_REQUESTS,
|
|
1520
|
+
enableRequestQueue = VIEWPORT_CONSTANTS.REQUEST_QUEUE.ENABLED,
|
|
1521
|
+
maxQueueSize = VIEWPORT_CONSTANTS.REQUEST_QUEUE.MAX_QUEUE_SIZE,
|
|
1522
|
+
loadOnDragEnd = true,
|
|
1523
|
+
initialScrollIndex = 0,
|
|
1524
|
+
selectId,
|
|
1525
|
+
autoLoad = true,
|
|
1526
|
+
autoSelectFirst = false,
|
|
1527
|
+
maxCachedItems = 1000,
|
|
1528
|
+
evictionBuffer = 150
|
|
1529
|
+
} = config;
|
|
1530
|
+
let hasCompletedInitialPositionLoad = false;
|
|
1531
|
+
const hasInitialScrollIndex = initialScrollIndex > 0;
|
|
1532
|
+
let hasManuallyLoaded = false;
|
|
1533
|
+
let reloadInProgress = false;
|
|
1534
|
+
let reloadTargetRange = null;
|
|
1535
|
+
let currentVelocity = 0;
|
|
1536
|
+
let activeLoadCount = 0;
|
|
1537
|
+
let completedLoads = 0;
|
|
1538
|
+
let failedLoads = 0;
|
|
1539
|
+
let cancelledLoads = 0;
|
|
1540
|
+
let isDragging = false;
|
|
1541
|
+
const MAX_CACHED_ITEMS = maxCachedItems;
|
|
1542
|
+
const EVICTION_BUFFER = evictionBuffer;
|
|
1543
|
+
let cachedItemCount = 0;
|
|
1544
|
+
const logMemoryStats = (caller) => {
|
|
1545
|
+
const loadedRangeCount = loadedRanges.size;
|
|
1546
|
+
const pendingRangeCount = pendingRanges.size;
|
|
1547
|
+
const abortControllerCount = abortControllers.size;
|
|
1548
|
+
};
|
|
1549
|
+
const evictDistantItems = (visibleStart, visibleEnd) => {
|
|
1550
|
+
if (cachedItemCount <= MAX_CACHED_ITEMS) {
|
|
1551
|
+
return;
|
|
1552
|
+
}
|
|
1553
|
+
const keepStart = Math.max(0, visibleStart - EVICTION_BUFFER);
|
|
1554
|
+
const keepEnd = visibleEnd + EVICTION_BUFFER;
|
|
1555
|
+
const keepRangeStart = Math.floor(keepStart / rangeSize);
|
|
1556
|
+
const keepRangeEnd = Math.floor(keepEnd / rangeSize);
|
|
1557
|
+
let evictedCount = 0;
|
|
1558
|
+
const rangesToRemove = new Set;
|
|
1559
|
+
loadedRanges.forEach((rangeId) => {
|
|
1560
|
+
if (rangeId < keepRangeStart || rangeId > keepRangeEnd) {
|
|
1561
|
+
rangesToRemove.add(rangeId);
|
|
1562
|
+
const rangeStart = rangeId * rangeSize;
|
|
1563
|
+
const rangeEnd = rangeStart + rangeSize;
|
|
1564
|
+
for (let i = rangeStart;i < rangeEnd; i++) {
|
|
1565
|
+
if (items[i] !== undefined) {
|
|
1566
|
+
delete items[i];
|
|
1567
|
+
evictedCount++;
|
|
1568
|
+
}
|
|
1569
|
+
}
|
|
1570
|
+
}
|
|
1571
|
+
});
|
|
1572
|
+
rangesToRemove.forEach((rangeId) => {
|
|
1573
|
+
loadedRanges.delete(rangeId);
|
|
1574
|
+
pendingRanges.delete(rangeId);
|
|
1575
|
+
activeRequests.delete(rangeId);
|
|
1576
|
+
const controller = abortControllers.get(rangeId);
|
|
1577
|
+
if (controller) {
|
|
1578
|
+
controller.abort();
|
|
1579
|
+
abortControllers.delete(rangeId);
|
|
1580
|
+
}
|
|
1581
|
+
});
|
|
1582
|
+
if (evictedCount > 0) {
|
|
1583
|
+
cachedItemCount -= evictedCount;
|
|
1584
|
+
component.emit?.("collection:items-evicted", {
|
|
1585
|
+
keepStart,
|
|
1586
|
+
keepEnd,
|
|
1587
|
+
evictedCount
|
|
1588
|
+
});
|
|
1589
|
+
}
|
|
1590
|
+
};
|
|
1591
|
+
const activeLoadRanges = new Set;
|
|
1592
|
+
let loadRequestQueue = [];
|
|
1593
|
+
const abortControllers = new Map;
|
|
1594
|
+
let items = [];
|
|
1595
|
+
let totalItems = 0;
|
|
1596
|
+
let loadedRanges = new Set;
|
|
1597
|
+
let pendingRanges = new Set;
|
|
1598
|
+
let failedRanges = new Map;
|
|
1599
|
+
let activeRequests = new Map;
|
|
1600
|
+
let currentCursor = null;
|
|
1601
|
+
let cursorMap = new Map;
|
|
1602
|
+
let pageToOffsetMap = new Map;
|
|
1603
|
+
let highestLoadedPage = 0;
|
|
1604
|
+
let discoveredTotal = null;
|
|
1605
|
+
let hasReachedEnd = false;
|
|
1606
|
+
component.items = items;
|
|
1607
|
+
const getRangeId = (offset, limit) => {
|
|
1608
|
+
return Math.floor(offset / rangeSize);
|
|
1609
|
+
};
|
|
1610
|
+
const getRangeKey = (range) => {
|
|
1611
|
+
return `${range.start}-${range.end}`;
|
|
1612
|
+
};
|
|
1613
|
+
const canLoad = () => {
|
|
1614
|
+
return currentVelocity <= cancelLoadThreshold;
|
|
1615
|
+
};
|
|
1616
|
+
const processQueue = () => {
|
|
1617
|
+
if (!enableRequestQueue || loadRequestQueue.length === 0)
|
|
1618
|
+
return;
|
|
1619
|
+
loadRequestQueue.sort((a, b) => {
|
|
1620
|
+
const priorityOrder = { high: 0, normal: 1, low: 2 };
|
|
1621
|
+
const priorityDiff = priorityOrder[a.priority] - priorityOrder[b.priority];
|
|
1622
|
+
return priorityDiff !== 0 ? priorityDiff : a.timestamp - b.timestamp;
|
|
1623
|
+
});
|
|
1624
|
+
while (loadRequestQueue.length > 0 && activeLoadCount < maxConcurrentRequests) {
|
|
1625
|
+
const request = loadRequestQueue.shift();
|
|
1626
|
+
if (request) {
|
|
1627
|
+
executeQueuedLoad(request);
|
|
1628
|
+
}
|
|
1629
|
+
}
|
|
1630
|
+
};
|
|
1631
|
+
const executeQueuedLoad = (request) => {
|
|
1632
|
+
activeLoadCount++;
|
|
1633
|
+
activeLoadRanges.add(getRangeKey(request.range));
|
|
1634
|
+
loadMissingRangesInternal(request.range, request.caller).then(() => {
|
|
1635
|
+
request.resolve();
|
|
1636
|
+
completedLoads++;
|
|
1637
|
+
}).catch((error) => {
|
|
1638
|
+
request.reject(error);
|
|
1639
|
+
failedLoads++;
|
|
1640
|
+
}).finally(() => {
|
|
1641
|
+
activeLoadCount--;
|
|
1642
|
+
activeLoadRanges.delete(getRangeKey(request.range));
|
|
1643
|
+
processQueue();
|
|
1644
|
+
});
|
|
1645
|
+
};
|
|
1646
|
+
const transformItems = (rawItems) => {
|
|
1647
|
+
if (!transform)
|
|
1648
|
+
return rawItems;
|
|
1649
|
+
return rawItems.map(transform);
|
|
1650
|
+
};
|
|
1651
|
+
const loadRange = async (offset, limit, caller) => {
|
|
1652
|
+
if (!collection) {
|
|
1653
|
+
console.warn("[Collection] No collection adapter configured");
|
|
1654
|
+
return [];
|
|
1655
|
+
}
|
|
1656
|
+
const rangeId = getRangeId(offset, limit);
|
|
1657
|
+
if (loadedRanges.has(rangeId)) {
|
|
1658
|
+
return items.slice(offset, offset + limit);
|
|
1659
|
+
}
|
|
1660
|
+
if (pendingRanges.has(rangeId)) {
|
|
1661
|
+
const existingRequest = activeRequests.get(rangeId);
|
|
1662
|
+
if (existingRequest) {
|
|
1663
|
+
return existingRequest;
|
|
1664
|
+
}
|
|
1665
|
+
}
|
|
1666
|
+
pendingRanges.add(rangeId);
|
|
1667
|
+
const abortController = new AbortController;
|
|
1668
|
+
abortControllers.set(rangeId, abortController);
|
|
1669
|
+
const requestPromise = (async () => {
|
|
1670
|
+
try {
|
|
1671
|
+
const page = Math.floor(offset / limit) + 1;
|
|
1672
|
+
let params;
|
|
1673
|
+
if (strategy === "cursor") {
|
|
1674
|
+
if (page === 1) {
|
|
1675
|
+
params = { limit };
|
|
1676
|
+
} else {
|
|
1677
|
+
const prevPageCursor = cursorMap.get(page - 1);
|
|
1678
|
+
if (!prevPageCursor) {
|
|
1679
|
+
console.warn(`[Collection] Cannot load page ${page} without cursor for page ${page - 1}`);
|
|
1680
|
+
throw new Error(`Sequential loading required - missing cursor for page ${page - 1}`);
|
|
1681
|
+
}
|
|
1682
|
+
params = { cursor: prevPageCursor, limit };
|
|
1683
|
+
}
|
|
1684
|
+
} else if (strategy === "page") {
|
|
1685
|
+
params = { page, limit };
|
|
1686
|
+
} else {
|
|
1687
|
+
params = { offset, limit };
|
|
1688
|
+
}
|
|
1689
|
+
const selfRef = component._selfRef;
|
|
1690
|
+
const self = selfRef?.current ?? {};
|
|
1691
|
+
if (typeof self._searchQuery === "function") {
|
|
1692
|
+
const searchQuery = self._searchQuery();
|
|
1693
|
+
if (searchQuery) {
|
|
1694
|
+
params.search = searchQuery;
|
|
1695
|
+
}
|
|
1696
|
+
}
|
|
1697
|
+
if (typeof self._filters === "function") {
|
|
1698
|
+
const filters = self._filters();
|
|
1699
|
+
if (filters && Object.keys(filters).length > 0) {
|
|
1700
|
+
params.filters = filters;
|
|
1701
|
+
}
|
|
1702
|
+
}
|
|
1703
|
+
const response = await collection.read({
|
|
1704
|
+
...params,
|
|
1705
|
+
signal: abortController.signal
|
|
1706
|
+
});
|
|
1707
|
+
const rawItems = response.data || response.items || response;
|
|
1708
|
+
const meta = response.meta || {};
|
|
1709
|
+
const responseCursor = meta.cursor || meta.nextCursor;
|
|
1710
|
+
if (strategy === "cursor" && responseCursor) {
|
|
1711
|
+
currentCursor = responseCursor;
|
|
1712
|
+
cursorMap.set(page, responseCursor);
|
|
1713
|
+
pageToOffsetMap.set(page, offset);
|
|
1714
|
+
highestLoadedPage = Math.max(highestLoadedPage, page);
|
|
1715
|
+
}
|
|
1716
|
+
if (strategy === "cursor" && meta.hasNext === false) {
|
|
1717
|
+
hasReachedEnd = true;
|
|
1718
|
+
}
|
|
1719
|
+
if (meta.total !== undefined) {
|
|
1720
|
+
discoveredTotal = meta.total;
|
|
1721
|
+
}
|
|
1722
|
+
const transformedItems = transformItems(rawItems);
|
|
1723
|
+
let newItemsAdded = 0;
|
|
1724
|
+
transformedItems.forEach((item, index) => {
|
|
1725
|
+
if (items[offset + index] === undefined) {
|
|
1726
|
+
newItemsAdded++;
|
|
1727
|
+
}
|
|
1728
|
+
items[offset + index] = item;
|
|
1729
|
+
});
|
|
1730
|
+
cachedItemCount += newItemsAdded;
|
|
1731
|
+
let newTotal = discoveredTotal ?? totalItems;
|
|
1732
|
+
if (offset === 0 && transformedItems.length === 0) {
|
|
1733
|
+
newTotal = 0;
|
|
1734
|
+
discoveredTotal = 0;
|
|
1735
|
+
}
|
|
1736
|
+
if (strategy === "cursor") {
|
|
1737
|
+
const loadedItemsCount = cachedItemCount;
|
|
1738
|
+
const marginItems = hasReachedEnd ? 0 : rangeSize * VIEWPORT_CONSTANTS.PAGINATION.CURSOR_SCROLL_MARGIN_MULTIPLIER;
|
|
1739
|
+
const minVirtualItems = rangeSize * VIEWPORT_CONSTANTS.PAGINATION.CURSOR_MIN_VIRTUAL_SIZE_MULTIPLIER;
|
|
1740
|
+
newTotal = Math.max(loadedItemsCount + marginItems, minVirtualItems);
|
|
1741
|
+
if (newTotal > totalItems) {
|
|
1742
|
+
totalItems = newTotal;
|
|
1743
|
+
setTotalItems(newTotal);
|
|
1744
|
+
}
|
|
1745
|
+
} else {
|
|
1746
|
+
newTotal = discoveredTotal ?? totalItems;
|
|
1747
|
+
}
|
|
1748
|
+
if (newTotal !== totalItems) {
|
|
1749
|
+
totalItems = newTotal;
|
|
1750
|
+
setTotalItems(newTotal);
|
|
1751
|
+
}
|
|
1752
|
+
component.items = items;
|
|
1753
|
+
component.emit?.("viewport:items-changed", {
|
|
1754
|
+
totalItems: newTotal,
|
|
1755
|
+
loadedCount: cachedItemCount
|
|
1756
|
+
});
|
|
1757
|
+
const viewportState = component.viewport.state;
|
|
1758
|
+
if (viewportState) {
|
|
1759
|
+
viewportState.totalItems = newTotal ?? items.length;
|
|
1760
|
+
}
|
|
1761
|
+
const itemsActuallyStored = transformedItems.length > 0 && transformedItems.every((_, idx) => items[offset + idx] !== undefined);
|
|
1762
|
+
if (itemsActuallyStored) {
|
|
1763
|
+
loadedRanges.add(rangeId);
|
|
1764
|
+
}
|
|
1765
|
+
pendingRanges.delete(rangeId);
|
|
1766
|
+
failedRanges.delete(rangeId);
|
|
1767
|
+
if (completedLoads % 5 === 0) {
|
|
1768
|
+
logMemoryStats(`load:${completedLoads}`);
|
|
1769
|
+
}
|
|
1770
|
+
component.emit?.("viewport:range-loaded", {
|
|
1771
|
+
offset,
|
|
1772
|
+
limit,
|
|
1773
|
+
items: transformedItems,
|
|
1774
|
+
total: newTotal
|
|
1775
|
+
});
|
|
1776
|
+
component.emit?.("collection:range-loaded", {
|
|
1777
|
+
offset,
|
|
1778
|
+
limit,
|
|
1779
|
+
items: transformedItems,
|
|
1780
|
+
total: newTotal,
|
|
1781
|
+
rangeId,
|
|
1782
|
+
itemsInMemory: items.filter((item) => item !== undefined).length,
|
|
1783
|
+
cursor: strategy === "cursor" ? currentCursor : undefined
|
|
1784
|
+
});
|
|
1785
|
+
component.viewport?.updateViewport?.();
|
|
1786
|
+
return transformedItems;
|
|
1787
|
+
} catch (error) {
|
|
1788
|
+
const isAbortError = error.name === "AbortError" || error.message === "Failed to fetch" && abortController.signal.aborted;
|
|
1789
|
+
if (isAbortError) {
|
|
1790
|
+
pendingRanges.delete(rangeId);
|
|
1791
|
+
cancelledLoads++;
|
|
1792
|
+
return [];
|
|
1793
|
+
}
|
|
1794
|
+
pendingRanges.delete(rangeId);
|
|
1795
|
+
failedLoads++;
|
|
1796
|
+
const attempts = (failedRanges.get(rangeId)?.attempts || 0) + 1;
|
|
1797
|
+
failedRanges.set(rangeId, {
|
|
1798
|
+
attempts,
|
|
1799
|
+
lastError: error,
|
|
1800
|
+
timestamp: Date.now()
|
|
1801
|
+
});
|
|
1802
|
+
component.emit?.("viewport:range-error", {
|
|
1803
|
+
offset,
|
|
1804
|
+
limit,
|
|
1805
|
+
error,
|
|
1806
|
+
attempts
|
|
1807
|
+
});
|
|
1808
|
+
throw error;
|
|
1809
|
+
} finally {
|
|
1810
|
+
activeRequests.delete(rangeId);
|
|
1811
|
+
abortControllers.delete(rangeId);
|
|
1812
|
+
}
|
|
1813
|
+
})();
|
|
1814
|
+
activeRequests.set(rangeId, requestPromise);
|
|
1815
|
+
return requestPromise;
|
|
1816
|
+
};
|
|
1817
|
+
const loadMissingRangesInternal = async (range, caller) => {
|
|
1818
|
+
if (!collection)
|
|
1819
|
+
return;
|
|
1820
|
+
if (strategy === "cursor") {
|
|
1821
|
+
const startPage = Math.floor(range.start / rangeSize) + 1;
|
|
1822
|
+
const endPage = Math.floor(range.end / rangeSize) + 1;
|
|
1823
|
+
const maxPagesToLoad = 10;
|
|
1824
|
+
const currentHighestPage = highestLoadedPage || 0;
|
|
1825
|
+
const limitedEndPage = Math.min(endPage, currentHighestPage + maxPagesToLoad);
|
|
1826
|
+
for (let page = startPage;page <= limitedEndPage; page++) {
|
|
1827
|
+
const rangeId = page - 1;
|
|
1828
|
+
const offset = rangeId * rangeSize;
|
|
1829
|
+
if (pendingRanges.has(rangeId)) {
|
|
1830
|
+
return;
|
|
1831
|
+
}
|
|
1832
|
+
if (!loadedRanges.has(rangeId) && !pendingRanges.has(rangeId)) {
|
|
1833
|
+
if (page > 1) {
|
|
1834
|
+
let canLoad2 = true;
|
|
1835
|
+
for (let prevPage = 1;prevPage < page; prevPage++) {
|
|
1836
|
+
const prevRangeId = prevPage - 1;
|
|
1837
|
+
if (!loadedRanges.has(prevRangeId)) {
|
|
1838
|
+
canLoad2 = false;
|
|
1839
|
+
if (!pendingRanges.has(prevRangeId)) {
|
|
1840
|
+
try {
|
|
1841
|
+
await loadRange(prevRangeId * rangeSize, rangeSize);
|
|
1842
|
+
} catch (error) {
|
|
1843
|
+
console.error(`[Collection] Failed to load prerequisite page ${prevPage}:`, error);
|
|
1844
|
+
return;
|
|
1845
|
+
}
|
|
1846
|
+
}
|
|
1847
|
+
break;
|
|
1848
|
+
}
|
|
1849
|
+
}
|
|
1850
|
+
if (!canLoad2) {
|
|
1851
|
+
continue;
|
|
1852
|
+
}
|
|
1853
|
+
}
|
|
1854
|
+
try {
|
|
1855
|
+
await loadRange(offset, rangeSize);
|
|
1856
|
+
} catch (error) {
|
|
1857
|
+
console.error(`[Collection] Failed to load page ${page}:`, error);
|
|
1858
|
+
break;
|
|
1859
|
+
}
|
|
1860
|
+
}
|
|
1861
|
+
}
|
|
1862
|
+
if (endPage > limitedEndPage) {}
|
|
1863
|
+
return;
|
|
1864
|
+
}
|
|
1865
|
+
const startRange = Math.floor(range.start / rangeSize);
|
|
1866
|
+
const endRange = Math.floor(range.end / rangeSize);
|
|
1867
|
+
const rangesToLoad = [];
|
|
1868
|
+
const visibleStart = range.start;
|
|
1869
|
+
const visibleEnd = range.end;
|
|
1870
|
+
let hasMissingData = false;
|
|
1871
|
+
if (totalItems === 0 || items.length === 0) {
|
|
1872
|
+
hasMissingData = true;
|
|
1873
|
+
} else {
|
|
1874
|
+
for (let i = visibleStart;i <= visibleEnd && i < totalItems; i++) {
|
|
1875
|
+
if (items[i] === undefined) {
|
|
1876
|
+
hasMissingData = true;
|
|
1877
|
+
break;
|
|
1878
|
+
}
|
|
1879
|
+
}
|
|
1880
|
+
}
|
|
1881
|
+
if (hasMissingData) {
|
|
1882
|
+
for (let rangeId = startRange;rangeId <= endRange; rangeId++) {
|
|
1883
|
+
if (pendingRanges.has(rangeId)) {
|
|
1884
|
+
continue;
|
|
1885
|
+
}
|
|
1886
|
+
if (totalItems === 0) {
|
|
1887
|
+
loadedRanges.delete(rangeId);
|
|
1888
|
+
rangesToLoad.push(rangeId);
|
|
1889
|
+
continue;
|
|
1890
|
+
}
|
|
1891
|
+
const rangeStart = rangeId * rangeSize;
|
|
1892
|
+
const rangeEnd = Math.min(rangeStart + rangeSize, totalItems);
|
|
1893
|
+
let rangeMissingData = false;
|
|
1894
|
+
for (let i = rangeStart;i < rangeEnd; i++) {
|
|
1895
|
+
if (i >= visibleStart && i <= visibleEnd && items[i] === undefined) {
|
|
1896
|
+
rangeMissingData = true;
|
|
1897
|
+
break;
|
|
1898
|
+
}
|
|
1899
|
+
}
|
|
1900
|
+
if (rangeMissingData) {
|
|
1901
|
+
loadedRanges.delete(rangeId);
|
|
1902
|
+
rangesToLoad.push(rangeId);
|
|
1903
|
+
}
|
|
1904
|
+
}
|
|
1905
|
+
}
|
|
1906
|
+
if (rangesToLoad.length === 0) {
|
|
1907
|
+
if (loadRequestQueue.length > 0 && activeLoadCount < maxConcurrentRequests) {
|
|
1908
|
+
processQueue();
|
|
1909
|
+
}
|
|
1910
|
+
return;
|
|
1911
|
+
}
|
|
1912
|
+
const promises = rangesToLoad.map((rangeId) => loadRange(rangeId * rangeSize, rangeSize));
|
|
1913
|
+
await Promise.allSettled(promises);
|
|
1914
|
+
};
|
|
1915
|
+
const loadMissingRanges = (range, caller) => {
|
|
1916
|
+
return new Promise((resolve, reject) => {
|
|
1917
|
+
const rangeKey = getRangeKey(range);
|
|
1918
|
+
if (activeLoadRanges.has(rangeKey)) {
|
|
1919
|
+
resolve();
|
|
1920
|
+
return;
|
|
1921
|
+
}
|
|
1922
|
+
if (isDragging && currentVelocity < 0.5 && currentVelocity > 0) {
|
|
1923
|
+
cancelledLoads++;
|
|
1924
|
+
resolve();
|
|
1925
|
+
return;
|
|
1926
|
+
}
|
|
1927
|
+
if (!canLoad()) {
|
|
1928
|
+
cancelledLoads++;
|
|
1929
|
+
resolve();
|
|
1930
|
+
return;
|
|
1931
|
+
}
|
|
1932
|
+
if (activeLoadCount < maxConcurrentRequests) {
|
|
1933
|
+
executeQueuedLoad({
|
|
1934
|
+
range,
|
|
1935
|
+
priority: "normal",
|
|
1936
|
+
timestamp: Date.now(),
|
|
1937
|
+
resolve,
|
|
1938
|
+
reject,
|
|
1939
|
+
caller
|
|
1940
|
+
});
|
|
1941
|
+
} else if (enableRequestQueue && loadRequestQueue.length < maxQueueSize) {
|
|
1942
|
+
loadRequestQueue.push({
|
|
1943
|
+
range,
|
|
1944
|
+
priority: "normal",
|
|
1945
|
+
timestamp: Date.now(),
|
|
1946
|
+
resolve,
|
|
1947
|
+
reject,
|
|
1948
|
+
caller
|
|
1949
|
+
});
|
|
1950
|
+
} else {
|
|
1951
|
+
if (caller === "viewport:idle" || caller === "viewport:range-changed") {
|
|
1952
|
+
loadRequestQueue.forEach((r) => {
|
|
1953
|
+
cancelledLoads++;
|
|
1954
|
+
r.resolve();
|
|
1955
|
+
});
|
|
1956
|
+
loadRequestQueue.length = 0;
|
|
1957
|
+
loadRequestQueue.push({
|
|
1958
|
+
range,
|
|
1959
|
+
priority: "high",
|
|
1960
|
+
timestamp: Date.now(),
|
|
1961
|
+
resolve,
|
|
1962
|
+
reject,
|
|
1963
|
+
caller
|
|
1964
|
+
});
|
|
1965
|
+
} else {
|
|
1966
|
+
if (loadRequestQueue.length >= maxQueueSize) {
|
|
1967
|
+
const removed = loadRequestQueue.splice(0, loadRequestQueue.length - maxQueueSize);
|
|
1968
|
+
removed.forEach((r) => {
|
|
1969
|
+
cancelledLoads++;
|
|
1970
|
+
r.resolve();
|
|
1971
|
+
});
|
|
1972
|
+
}
|
|
1973
|
+
resolve();
|
|
1974
|
+
}
|
|
1975
|
+
}
|
|
1976
|
+
});
|
|
1977
|
+
};
|
|
1978
|
+
const retryFailedRange = async (rangeId) => {
|
|
1979
|
+
failedRanges.delete(rangeId);
|
|
1980
|
+
const offset = rangeId * rangeSize;
|
|
1981
|
+
return loadRange(offset, rangeSize);
|
|
1982
|
+
};
|
|
1983
|
+
const setTotalItems = (total) => {
|
|
1984
|
+
totalItems = total;
|
|
1985
|
+
const viewportState = component.viewport.state;
|
|
1986
|
+
if (viewportState) {
|
|
1987
|
+
viewportState.totalItems = total;
|
|
1988
|
+
}
|
|
1989
|
+
component.emit?.("viewport:total-items-changed", { total });
|
|
1990
|
+
};
|
|
1991
|
+
const originalInitialize = component.viewport.initialize;
|
|
1992
|
+
component.viewport.initialize = () => {
|
|
1993
|
+
const result = originalInitialize();
|
|
1994
|
+
if (result === false) {
|
|
1995
|
+
return false;
|
|
1996
|
+
}
|
|
1997
|
+
if (component.totalItems) {
|
|
1998
|
+
totalItems = component.totalItems;
|
|
1999
|
+
}
|
|
2000
|
+
component.on?.("viewport:drag-start", () => {
|
|
2001
|
+
isDragging = true;
|
|
2002
|
+
});
|
|
2003
|
+
component.on?.("viewport:drag-end", () => {
|
|
2004
|
+
isDragging = false;
|
|
2005
|
+
if (loadOnDragEnd) {
|
|
2006
|
+
processQueue();
|
|
2007
|
+
}
|
|
2008
|
+
});
|
|
2009
|
+
component.on?.("viewport:range-changed", async (data) => {
|
|
2010
|
+
const range = data.range || data;
|
|
2011
|
+
if (!autoLoad && !hasManuallyLoaded) {
|
|
2012
|
+
return;
|
|
2013
|
+
}
|
|
2014
|
+
const { start, end } = range;
|
|
2015
|
+
if (typeof start !== "number" || typeof end !== "number") {
|
|
2016
|
+
console.warn("[Collection] Invalid range in viewport:range-changed event:", data);
|
|
2017
|
+
return;
|
|
2018
|
+
}
|
|
2019
|
+
if (hasInitialScrollIndex && !hasCompletedInitialPositionLoad) {
|
|
2020
|
+
const page1EndIndex = rangeSize - 1;
|
|
2021
|
+
if (start === 0 && end <= page1EndIndex) {
|
|
2022
|
+
return;
|
|
2023
|
+
}
|
|
2024
|
+
}
|
|
2025
|
+
if (reloadInProgress && reloadTargetRange) {
|
|
2026
|
+
const page1EndIndex = rangeSize - 1;
|
|
2027
|
+
if (start === 0 && end <= page1EndIndex) {
|
|
2028
|
+
if (reloadTargetRange.start > page1EndIndex) {
|
|
2029
|
+
return;
|
|
2030
|
+
}
|
|
2031
|
+
}
|
|
2032
|
+
}
|
|
2033
|
+
loadMissingRanges({ start, end }, "viewport:range-changed");
|
|
2034
|
+
evictDistantItems(start, end);
|
|
2035
|
+
if (strategy === "cursor" && !hasReachedEnd) {
|
|
2036
|
+
const loadedItemsCount = cachedItemCount;
|
|
2037
|
+
const marginItems = rangeSize * VIEWPORT_CONSTANTS.PAGINATION.CURSOR_SCROLL_MARGIN_MULTIPLIER;
|
|
2038
|
+
const minVirtualItems = rangeSize * VIEWPORT_CONSTANTS.PAGINATION.CURSOR_MIN_VIRTUAL_SIZE_MULTIPLIER;
|
|
2039
|
+
const dynamicTotal = Math.max(loadedItemsCount + marginItems, minVirtualItems);
|
|
2040
|
+
if (dynamicTotal !== totalItems) {
|
|
2041
|
+
setTotalItems(dynamicTotal);
|
|
2042
|
+
}
|
|
2043
|
+
}
|
|
2044
|
+
});
|
|
2045
|
+
component.on?.("viewport:velocity-changed", (data) => {
|
|
2046
|
+
const previousVelocity = currentVelocity;
|
|
2047
|
+
currentVelocity = Math.abs(data.velocity || 0);
|
|
2048
|
+
if (previousVelocity > cancelLoadThreshold && currentVelocity <= cancelLoadThreshold) {
|
|
2049
|
+
processQueue();
|
|
2050
|
+
}
|
|
2051
|
+
});
|
|
2052
|
+
component.on?.("viewport:idle", async (data) => {
|
|
2053
|
+
currentVelocity = 0;
|
|
2054
|
+
if (isDragging) {
|
|
2055
|
+
isDragging = false;
|
|
2056
|
+
}
|
|
2057
|
+
const viewportState = component.viewport.state;
|
|
2058
|
+
const visibleRange = viewportState?.visibleRange;
|
|
2059
|
+
if (visibleRange) {
|
|
2060
|
+
const buffer = rangeSize * 2;
|
|
2061
|
+
loadRequestQueue = loadRequestQueue.filter((request) => {
|
|
2062
|
+
const requestEnd = request.range.end;
|
|
2063
|
+
const requestStart = request.range.start;
|
|
2064
|
+
const isRelevant = requestEnd >= visibleRange.start - buffer && requestStart <= visibleRange.end + buffer;
|
|
2065
|
+
if (!isRelevant) {
|
|
2066
|
+
request.resolve();
|
|
2067
|
+
}
|
|
2068
|
+
return isRelevant;
|
|
2069
|
+
});
|
|
2070
|
+
await loadMissingRanges(visibleRange, "viewport:idle");
|
|
2071
|
+
}
|
|
2072
|
+
processQueue();
|
|
2073
|
+
});
|
|
2074
|
+
component.on?.("item:removed", (data) => {
|
|
2075
|
+
if (discoveredTotal !== null && discoveredTotal > 0) {
|
|
2076
|
+
discoveredTotal = discoveredTotal - 1;
|
|
2077
|
+
}
|
|
2078
|
+
});
|
|
2079
|
+
if (collection && autoLoad) {
|
|
2080
|
+
if (initialScrollIndex > 0 || selectId !== undefined && selectId !== null) {
|
|
2081
|
+
const visibleRange = component.viewport?.getVisibleRange?.();
|
|
2082
|
+
if (visibleRange && (visibleRange.start > 0 || visibleRange.end > 0)) {
|
|
2083
|
+
loadMissingRanges(visibleRange, "initial-position").then(() => {
|
|
2084
|
+
hasCompletedInitialPositionLoad = true;
|
|
2085
|
+
if (selectId !== undefined) {
|
|
2086
|
+
component.emit?.("collection:initial-load-complete", {
|
|
2087
|
+
selectId,
|
|
2088
|
+
initialScrollIndex
|
|
2089
|
+
});
|
|
2090
|
+
}
|
|
2091
|
+
}).catch((error) => {
|
|
2092
|
+
console.error("[Collection] Failed to load initial position data:", error);
|
|
2093
|
+
hasCompletedInitialPositionLoad = true;
|
|
2094
|
+
});
|
|
2095
|
+
} else {
|
|
2096
|
+
const overscan = 2;
|
|
2097
|
+
const estimatedVisibleCount = Math.ceil(600 / 50);
|
|
2098
|
+
const start = Math.max(0, initialScrollIndex - overscan);
|
|
2099
|
+
const end = initialScrollIndex + estimatedVisibleCount + overscan;
|
|
2100
|
+
loadMissingRanges({ start, end }, "initial-position-fallback").then(() => {
|
|
2101
|
+
hasCompletedInitialPositionLoad = true;
|
|
2102
|
+
if (selectId !== undefined) {
|
|
2103
|
+
component.emit?.("collection:initial-load-complete", {
|
|
2104
|
+
selectId,
|
|
2105
|
+
initialScrollIndex
|
|
2106
|
+
});
|
|
2107
|
+
}
|
|
2108
|
+
}).catch((error) => {
|
|
2109
|
+
console.error("[Collection] Failed to load initial position data (fallback):", error);
|
|
2110
|
+
hasCompletedInitialPositionLoad = true;
|
|
2111
|
+
});
|
|
2112
|
+
}
|
|
2113
|
+
return;
|
|
2114
|
+
}
|
|
2115
|
+
loadRange(0, rangeSize, "initialize-page1").then(() => {
|
|
2116
|
+
if (autoSelectFirst && items.length > 0) {
|
|
2117
|
+
const firstItem = items[0];
|
|
2118
|
+
const firstId = firstItem?._id || firstItem?.id;
|
|
2119
|
+
if (firstId !== undefined) {
|
|
2120
|
+
component.emit?.("collection:initial-load-complete", {
|
|
2121
|
+
selectId: firstId,
|
|
2122
|
+
initialScrollIndex: 0
|
|
2123
|
+
});
|
|
2124
|
+
}
|
|
2125
|
+
}
|
|
2126
|
+
}).catch((error) => {
|
|
2127
|
+
console.error("[Collection] Failed to load initial data:", error);
|
|
2128
|
+
});
|
|
2129
|
+
}
|
|
2130
|
+
return result;
|
|
2131
|
+
};
|
|
2132
|
+
const reset = () => {
|
|
2133
|
+
logMemoryStats("reset:before");
|
|
2134
|
+
abortControllers.forEach((controller) => {
|
|
2135
|
+
try {
|
|
2136
|
+
controller.abort();
|
|
2137
|
+
} catch (e) {}
|
|
2138
|
+
});
|
|
2139
|
+
abortControllers.clear();
|
|
2140
|
+
items.length = 0;
|
|
2141
|
+
loadRequestQueue.length = 0;
|
|
2142
|
+
loadedRanges.clear();
|
|
2143
|
+
pendingRanges.clear();
|
|
2144
|
+
failedRanges.clear();
|
|
2145
|
+
activeLoadRanges.clear();
|
|
2146
|
+
activeRequests.clear();
|
|
2147
|
+
cursorMap.clear();
|
|
2148
|
+
pageToOffsetMap.clear();
|
|
2149
|
+
totalItems = 0;
|
|
2150
|
+
currentCursor = null;
|
|
2151
|
+
highestLoadedPage = 0;
|
|
2152
|
+
discoveredTotal = null;
|
|
2153
|
+
hasReachedEnd = false;
|
|
2154
|
+
hasCompletedInitialPositionLoad = false;
|
|
2155
|
+
hasManuallyLoaded = false;
|
|
2156
|
+
cachedItemCount = 0;
|
|
2157
|
+
activeLoadCount = 0;
|
|
2158
|
+
completedLoads = 0;
|
|
2159
|
+
failedLoads = 0;
|
|
2160
|
+
cancelledLoads = 0;
|
|
2161
|
+
component.items = items;
|
|
2162
|
+
component.totalItems = 0;
|
|
2163
|
+
logMemoryStats("reset:after");
|
|
2164
|
+
component.emit?.("collection:reset");
|
|
2165
|
+
};
|
|
2166
|
+
const clearQueue = () => {
|
|
2167
|
+
loadRequestQueue.forEach((request) => {
|
|
2168
|
+
request.resolve();
|
|
2169
|
+
});
|
|
2170
|
+
loadRequestQueue.length = 0;
|
|
2171
|
+
};
|
|
2172
|
+
const setManuallyLoaded = () => {
|
|
2173
|
+
hasManuallyLoaded = true;
|
|
2174
|
+
};
|
|
2175
|
+
const setReloadInProgress = (inProgress, targetRange) => {
|
|
2176
|
+
reloadInProgress = inProgress;
|
|
2177
|
+
reloadTargetRange = targetRange || null;
|
|
2178
|
+
};
|
|
2179
|
+
component.viewport.collection = {
|
|
2180
|
+
loadRange: (offset, limit) => loadRange(offset, limit),
|
|
2181
|
+
loadMissingRanges: (range, caller) => loadMissingRanges(range, caller || "viewport.collection"),
|
|
2182
|
+
getLoadedRanges: () => loadedRanges,
|
|
2183
|
+
getPendingRanges: () => pendingRanges,
|
|
2184
|
+
clearFailedRanges: () => failedRanges.clear(),
|
|
2185
|
+
clearQueue,
|
|
2186
|
+
setManuallyLoaded,
|
|
2187
|
+
setReloadInProgress,
|
|
2188
|
+
retryFailedRange,
|
|
2189
|
+
setTotalItems,
|
|
2190
|
+
getTotalItems: () => totalItems,
|
|
2191
|
+
getCurrentCursor: () => currentCursor,
|
|
2192
|
+
getCursorForPage: (page) => cursorMap.get(page) || null,
|
|
2193
|
+
reset
|
|
2194
|
+
};
|
|
2195
|
+
component.collection = {
|
|
2196
|
+
items,
|
|
2197
|
+
getItems: () => items,
|
|
2198
|
+
getItem: (index) => items[index],
|
|
2199
|
+
loadRange,
|
|
2200
|
+
loadMissingRanges: (range, caller) => loadMissingRanges(range, caller || "component.collection"),
|
|
2201
|
+
getLoadingStats: () => ({
|
|
2202
|
+
pendingRequests: activeLoadCount,
|
|
2203
|
+
completedRequests: completedLoads,
|
|
2204
|
+
failedRequests: failedLoads,
|
|
2205
|
+
cancelledRequests: cancelledLoads,
|
|
2206
|
+
currentVelocity,
|
|
2207
|
+
canLoad: canLoad(),
|
|
2208
|
+
queuedRequests: loadRequestQueue.length
|
|
2209
|
+
}),
|
|
2210
|
+
getTotalItems: () => totalItems,
|
|
2211
|
+
setTotalItems,
|
|
2212
|
+
getCurrentCursor: () => currentCursor,
|
|
2213
|
+
getCursorForPage: (page) => cursorMap.get(page) || null,
|
|
2214
|
+
reset
|
|
2215
|
+
};
|
|
2216
|
+
component.items = items;
|
|
2217
|
+
const destroy = () => {
|
|
2218
|
+
logMemoryStats("destroy:before");
|
|
2219
|
+
abortControllers.forEach((controller) => {
|
|
2220
|
+
try {
|
|
2221
|
+
controller.abort();
|
|
2222
|
+
} catch (e) {}
|
|
2223
|
+
});
|
|
2224
|
+
abortControllers.clear();
|
|
2225
|
+
items.length = 0;
|
|
2226
|
+
loadRequestQueue.length = 0;
|
|
2227
|
+
loadedRanges.clear();
|
|
2228
|
+
pendingRanges.clear();
|
|
2229
|
+
failedRanges.clear();
|
|
2230
|
+
activeLoadRanges.clear();
|
|
2231
|
+
activeRequests.clear();
|
|
2232
|
+
cursorMap.clear();
|
|
2233
|
+
pageToOffsetMap.clear();
|
|
2234
|
+
totalItems = 0;
|
|
2235
|
+
currentCursor = null;
|
|
2236
|
+
highestLoadedPage = 0;
|
|
2237
|
+
discoveredTotal = null;
|
|
2238
|
+
hasReachedEnd = false;
|
|
2239
|
+
if (component.items === items) {
|
|
2240
|
+
component.items = [];
|
|
2241
|
+
}
|
|
2242
|
+
logMemoryStats("destroy:after");
|
|
2243
|
+
};
|
|
2244
|
+
wrapDestroy(component, destroy);
|
|
2245
|
+
return {
|
|
2246
|
+
...component,
|
|
2247
|
+
collection: {
|
|
2248
|
+
items,
|
|
2249
|
+
getItems: () => items,
|
|
2250
|
+
getItem: (index) => items[index],
|
|
2251
|
+
loadRange,
|
|
2252
|
+
loadMissingRanges: (range, caller) => loadMissingRanges(range, caller || "return.collection"),
|
|
2253
|
+
getLoadedRanges: () => loadedRanges,
|
|
2254
|
+
getPendingRanges: () => pendingRanges,
|
|
2255
|
+
clearFailedRanges: () => failedRanges.clear(),
|
|
2256
|
+
clearQueue,
|
|
2257
|
+
setManuallyLoaded,
|
|
2258
|
+
setReloadInProgress,
|
|
2259
|
+
retryFailedRange,
|
|
2260
|
+
setTotalItems,
|
|
2261
|
+
getTotalItems: () => totalItems,
|
|
2262
|
+
getCurrentCursor: () => currentCursor,
|
|
2263
|
+
getCursorForPage: (page) => cursorMap.get(page) || null,
|
|
2264
|
+
reset
|
|
2265
|
+
}
|
|
2266
|
+
};
|
|
2267
|
+
};
|
|
2268
|
+
}
|
|
2269
|
+
|
|
2270
|
+
// src/core/viewport/features/placeholders.ts
|
|
2271
|
+
function withPlaceholders(config = {}) {
|
|
2272
|
+
return (component) => {
|
|
2273
|
+
const {
|
|
2274
|
+
enabled = true,
|
|
2275
|
+
analyzeFirstLoad = true,
|
|
2276
|
+
maskCharacter = VIEWPORT_CONSTANTS.PLACEHOLDER.MASK_CHARACTER,
|
|
2277
|
+
randomLengthVariance = VIEWPORT_CONSTANTS.PLACEHOLDER.RANDOM_LENGTH_VARIANCE
|
|
2278
|
+
} = config;
|
|
2279
|
+
let fieldStructures = null;
|
|
2280
|
+
let hasAnalyzed = false;
|
|
2281
|
+
let placeholderIdCounter = 0;
|
|
2282
|
+
const analyzeDataStructure = (items) => {
|
|
2283
|
+
if (!enabled || hasAnalyzed || !items.length) {
|
|
2284
|
+
return;
|
|
2285
|
+
}
|
|
2286
|
+
const structures = new Map;
|
|
2287
|
+
const sampleSize = Math.min(items.length, VIEWPORT_CONSTANTS.PLACEHOLDER.MAX_SAMPLE_SIZE);
|
|
2288
|
+
const fieldStats = new Map;
|
|
2289
|
+
for (let i = 0;i < sampleSize; i++) {
|
|
2290
|
+
const item = items[i];
|
|
2291
|
+
if (!item || typeof item !== "object")
|
|
2292
|
+
continue;
|
|
2293
|
+
Object.keys(item).forEach((field) => {
|
|
2294
|
+
if (field.startsWith("_") || field === VIEWPORT_CONSTANTS.PLACEHOLDER.PLACEHOLDER_FLAG) {
|
|
2295
|
+
return;
|
|
2296
|
+
}
|
|
2297
|
+
const value = String(item[field] || "");
|
|
2298
|
+
const length = value.length;
|
|
2299
|
+
if (!fieldStats.has(field)) {
|
|
2300
|
+
fieldStats.set(field, []);
|
|
2301
|
+
}
|
|
2302
|
+
fieldStats.get(field).push(length);
|
|
2303
|
+
});
|
|
2304
|
+
}
|
|
2305
|
+
fieldStats.forEach((lengths, field) => {
|
|
2306
|
+
if (lengths.length === 0)
|
|
2307
|
+
return;
|
|
2308
|
+
const minLength = Math.min(...lengths);
|
|
2309
|
+
const maxLength = Math.max(...lengths);
|
|
2310
|
+
const avgLength = Math.round(lengths.reduce((sum, len) => sum + len, 0) / lengths.length);
|
|
2311
|
+
structures.set(field, {
|
|
2312
|
+
minLength,
|
|
2313
|
+
maxLength,
|
|
2314
|
+
avgLength
|
|
2315
|
+
});
|
|
2316
|
+
});
|
|
2317
|
+
fieldStructures = structures;
|
|
2318
|
+
hasAnalyzed = true;
|
|
2319
|
+
component.emit?.("viewport:placeholders-structure-analyzed", {
|
|
2320
|
+
structure: Object.fromEntries(structures)
|
|
2321
|
+
});
|
|
2322
|
+
};
|
|
2323
|
+
const generatePlaceholderItem = (index) => {
|
|
2324
|
+
const placeholder = {
|
|
2325
|
+
id: `placeholder-${placeholderIdCounter++}`,
|
|
2326
|
+
[VIEWPORT_CONSTANTS.PLACEHOLDER.PLACEHOLDER_FLAG]: true,
|
|
2327
|
+
_index: index
|
|
2328
|
+
};
|
|
2329
|
+
if (!fieldStructures || fieldStructures.size === 0) {
|
|
2330
|
+
placeholder.label = maskCharacter.repeat(10);
|
|
2331
|
+
return placeholder;
|
|
2332
|
+
}
|
|
2333
|
+
fieldStructures.forEach((structure, field) => {
|
|
2334
|
+
let length;
|
|
2335
|
+
if (randomLengthVariance && structure.minLength !== structure.maxLength) {
|
|
2336
|
+
length = Math.floor(Math.random() * (structure.maxLength - structure.minLength + 1) + structure.minLength);
|
|
2337
|
+
} else {
|
|
2338
|
+
length = structure.avgLength;
|
|
2339
|
+
}
|
|
2340
|
+
if (randomLengthVariance && Math.random() < 0.3) {
|
|
2341
|
+
length = Math.max(1, length + Math.floor(Math.random() * 3) - 1);
|
|
2342
|
+
}
|
|
2343
|
+
placeholder[field] = maskCharacter.repeat(length);
|
|
2344
|
+
});
|
|
2345
|
+
return placeholder;
|
|
2346
|
+
};
|
|
2347
|
+
const generatePlaceholderItems = (range) => {
|
|
2348
|
+
const items = [];
|
|
2349
|
+
for (let i = range.start;i <= range.end; i++) {
|
|
2350
|
+
items.push(generatePlaceholderItem(i));
|
|
2351
|
+
}
|
|
2352
|
+
return items;
|
|
2353
|
+
};
|
|
2354
|
+
const showPlaceholders = (range) => {
|
|
2355
|
+
if (!enabled)
|
|
2356
|
+
return;
|
|
2357
|
+
const placeholders = generatePlaceholderItems(range);
|
|
2358
|
+
if (component.items) {
|
|
2359
|
+
for (let i = 0;i < placeholders.length; i++) {
|
|
2360
|
+
const index = range.start + i;
|
|
2361
|
+
if (!component.items[index]) {
|
|
2362
|
+
component.items[index] = placeholders[i];
|
|
2363
|
+
}
|
|
2364
|
+
}
|
|
2365
|
+
}
|
|
2366
|
+
console.log(`\uD83D\uDD04 [PLACEHOLDERS] Showing ${placeholders.length} placeholders for range ${range.start}-${range.end}`);
|
|
2367
|
+
component.emit?.("viewport:placeholders-shown", {
|
|
2368
|
+
range,
|
|
2369
|
+
count: placeholders.length
|
|
2370
|
+
});
|
|
2371
|
+
};
|
|
2372
|
+
const isPlaceholder2 = (item) => {
|
|
2373
|
+
return item && typeof item === "object" && item[VIEWPORT_CONSTANTS.PLACEHOLDER.PLACEHOLDER_FLAG] === true;
|
|
2374
|
+
};
|
|
2375
|
+
const replacePlaceholders = (items, offset) => {
|
|
2376
|
+
if (!component.items)
|
|
2377
|
+
return;
|
|
2378
|
+
let replacedCount = 0;
|
|
2379
|
+
for (let i = 0;i < items.length; i++) {
|
|
2380
|
+
const index = offset + i;
|
|
2381
|
+
const currentItem = component.items[index];
|
|
2382
|
+
if (isPlaceholder2(currentItem)) {
|
|
2383
|
+
component.items[index] = items[i];
|
|
2384
|
+
replacedCount++;
|
|
2385
|
+
}
|
|
2386
|
+
}
|
|
2387
|
+
if (replacedCount > 0) {
|
|
2388
|
+
console.log(`✨ [PLACEHOLDERS] Replaced ${replacedCount} placeholders at offset ${offset}`);
|
|
2389
|
+
component.emit?.("viewport:placeholders-replaced", {
|
|
2390
|
+
offset,
|
|
2391
|
+
count: replacedCount
|
|
2392
|
+
});
|
|
2393
|
+
}
|
|
2394
|
+
};
|
|
2395
|
+
const clear = () => {
|
|
2396
|
+
fieldStructures = null;
|
|
2397
|
+
hasAnalyzed = false;
|
|
2398
|
+
placeholderIdCounter = 0;
|
|
2399
|
+
};
|
|
2400
|
+
const initialize = () => {
|
|
2401
|
+
if (!enabled)
|
|
2402
|
+
return;
|
|
2403
|
+
if (analyzeFirstLoad) {
|
|
2404
|
+
const handleRangeLoaded = (data) => {
|
|
2405
|
+
if (!hasAnalyzed && data.items && data.items.length > 0) {
|
|
2406
|
+
analyzeDataStructure(data.items);
|
|
2407
|
+
}
|
|
2408
|
+
replacePlaceholders(data.items, data.offset);
|
|
2409
|
+
};
|
|
2410
|
+
component.on?.("viewport:range-loaded", handleRangeLoaded);
|
|
2411
|
+
}
|
|
2412
|
+
const totalItems = component.collection?.getTotalItems() || 0;
|
|
2413
|
+
if (totalItems > 0) {
|
|
2414
|
+
const initialRange = {
|
|
2415
|
+
start: 0,
|
|
2416
|
+
end: Math.min(VIEWPORT_CONSTANTS.PLACEHOLDER.MAX_SAMPLE_SIZE - 1, totalItems - 1)
|
|
2417
|
+
};
|
|
2418
|
+
showPlaceholders(initialRange);
|
|
2419
|
+
}
|
|
2420
|
+
};
|
|
2421
|
+
const destroy = () => {
|
|
2422
|
+
clear();
|
|
2423
|
+
};
|
|
2424
|
+
component._placeholdersInitialize = initialize;
|
|
2425
|
+
component._placeholdersDestroy = destroy;
|
|
2426
|
+
return {
|
|
2427
|
+
...component,
|
|
2428
|
+
placeholders: {
|
|
2429
|
+
analyzeDataStructure,
|
|
2430
|
+
hasAnalyzedStructure: () => hasAnalyzed,
|
|
2431
|
+
generatePlaceholderItem,
|
|
2432
|
+
generatePlaceholderItems,
|
|
2433
|
+
showPlaceholders,
|
|
2434
|
+
isPlaceholder: isPlaceholder2,
|
|
2435
|
+
replacePlaceholders,
|
|
2436
|
+
clear
|
|
2437
|
+
}
|
|
2438
|
+
};
|
|
2439
|
+
};
|
|
2440
|
+
}
|
|
2441
|
+
|
|
2442
|
+
// src/core/viewport/features/rendering.ts
|
|
2443
|
+
var import_mtrl3 = require("mtrl");
|
|
2444
|
+
|
|
2445
|
+
// src/core/layout/schema.ts
|
|
2446
|
+
class FragmentPool {
|
|
2447
|
+
pool = [];
|
|
2448
|
+
maxSize = 8;
|
|
2449
|
+
get() {
|
|
2450
|
+
return this.pool.pop() || document.createDocumentFragment();
|
|
2451
|
+
}
|
|
2452
|
+
release(fragment) {
|
|
2453
|
+
if (this.pool.length < this.maxSize && fragment.childNodes.length === 0) {
|
|
2454
|
+
this.pool.push(fragment);
|
|
2455
|
+
}
|
|
2456
|
+
}
|
|
2457
|
+
clear() {
|
|
2458
|
+
this.pool.length = 0;
|
|
2459
|
+
}
|
|
2460
|
+
}
|
|
2461
|
+
var fragmentPool = new FragmentPool;
|
|
2462
|
+
var classCache = new Map;
|
|
2463
|
+
var PREFIX3 = "mtrl";
|
|
2464
|
+
var PREFIX_WITH_DASH = `${PREFIX3}-`;
|
|
2465
|
+
function getCachedClassName(type, property, value) {
|
|
2466
|
+
const key = `${type}-${property}-${value}`;
|
|
2467
|
+
if (!classCache.has(key)) {
|
|
2468
|
+
if (type === "item") {
|
|
2469
|
+
classCache.set(key, property === "" ? `layout__item--${value}` : `layout__item--${property}-${value}`);
|
|
2470
|
+
} else {
|
|
2471
|
+
if (property === "align") {
|
|
2472
|
+
classCache.set(key, `layout--${type}-${value}`);
|
|
2473
|
+
} else {
|
|
2474
|
+
classCache.set(key, `layout--${type}-${property}-${value}`);
|
|
2475
|
+
}
|
|
2476
|
+
}
|
|
2477
|
+
}
|
|
2478
|
+
return classCache.get(key);
|
|
2479
|
+
}
|
|
2480
|
+
function isComponent(value) {
|
|
2481
|
+
return value && typeof value === "object" && "element" in value;
|
|
2482
|
+
}
|
|
2483
|
+
function createFragment() {
|
|
2484
|
+
return fragmentPool.get();
|
|
2485
|
+
}
|
|
2486
|
+
function releaseFragment(fragment) {
|
|
2487
|
+
fragmentPool.release(fragment);
|
|
2488
|
+
}
|
|
2489
|
+
function processClassNames(options, skipPrefix = false) {
|
|
2490
|
+
if (!options)
|
|
2491
|
+
return options;
|
|
2492
|
+
const hasRawClass = options.rawClass;
|
|
2493
|
+
const hasRegularClass = options.class || options.className;
|
|
2494
|
+
if (!hasRawClass && !hasRegularClass)
|
|
2495
|
+
return options;
|
|
2496
|
+
if (hasRawClass && !hasRegularClass && skipPrefix) {
|
|
2497
|
+
const processed2 = { ...options };
|
|
2498
|
+
delete processed2.rawClass;
|
|
2499
|
+
if (typeof hasRawClass === "string") {
|
|
2500
|
+
processed2.class = hasRawClass;
|
|
2501
|
+
} else {
|
|
2502
|
+
processed2.class = hasRawClass.join(" ");
|
|
2503
|
+
}
|
|
2504
|
+
return processed2;
|
|
2505
|
+
}
|
|
2506
|
+
const processed = { ...options };
|
|
2507
|
+
let finalClasses = "";
|
|
2508
|
+
if (!skipPrefix && hasRegularClass) {
|
|
2509
|
+
let prefixedString = "";
|
|
2510
|
+
if (processed.class) {
|
|
2511
|
+
prefixedString += Array.isArray(processed.class) ? processed.class.join(" ") : processed.class;
|
|
2512
|
+
}
|
|
2513
|
+
if (processed.className) {
|
|
2514
|
+
prefixedString += (prefixedString ? " " : "") + processed.className;
|
|
2515
|
+
}
|
|
2516
|
+
if (prefixedString) {
|
|
2517
|
+
finalClasses = prefixedString.split(/\s+/).filter(Boolean).map((cls) => cls.startsWith(PREFIX_WITH_DASH) ? cls : PREFIX_WITH_DASH + cls).join(" ");
|
|
2518
|
+
}
|
|
2519
|
+
}
|
|
2520
|
+
if (hasRawClass) {
|
|
2521
|
+
const rawString = Array.isArray(hasRawClass) ? hasRawClass.filter(Boolean).join(" ") : hasRawClass;
|
|
2522
|
+
finalClasses += (finalClasses ? " " : "") + rawString;
|
|
2523
|
+
}
|
|
2524
|
+
if (finalClasses) {
|
|
2525
|
+
processed.class = finalClasses;
|
|
2526
|
+
}
|
|
2527
|
+
delete processed.className;
|
|
2528
|
+
delete processed.rawClass;
|
|
2529
|
+
return processed;
|
|
2530
|
+
}
|
|
2531
|
+
function extractParameters(schema, startIndex, defaultCreator) {
|
|
2532
|
+
const items = schema.slice(startIndex, startIndex + 3);
|
|
2533
|
+
let creator, name, options;
|
|
2534
|
+
let consumed = 1;
|
|
2535
|
+
const [first, second, third] = items;
|
|
2536
|
+
if (typeof first === "function") {
|
|
2537
|
+
creator = first;
|
|
2538
|
+
if (typeof second === "string") {
|
|
2539
|
+
name = second;
|
|
2540
|
+
consumed = 2;
|
|
2541
|
+
if (isObject(third)) {
|
|
2542
|
+
options = third;
|
|
2543
|
+
consumed = 3;
|
|
2544
|
+
}
|
|
2545
|
+
} else if (isObject(second)) {
|
|
2546
|
+
options = second;
|
|
2547
|
+
consumed = 2;
|
|
2548
|
+
}
|
|
2549
|
+
} else if (typeof first === "string") {
|
|
2550
|
+
creator = defaultCreator;
|
|
2551
|
+
name = first;
|
|
2552
|
+
if (isObject(second)) {
|
|
2553
|
+
options = second;
|
|
2554
|
+
consumed = 2;
|
|
2555
|
+
}
|
|
2556
|
+
} else if (isObject(first)) {
|
|
2557
|
+
creator = defaultCreator;
|
|
2558
|
+
options = first;
|
|
2559
|
+
}
|
|
2560
|
+
return {
|
|
2561
|
+
creator: creator || defaultCreator,
|
|
2562
|
+
name,
|
|
2563
|
+
options: options || {},
|
|
2564
|
+
consumed
|
|
2565
|
+
};
|
|
2566
|
+
}
|
|
2567
|
+
function isObject(value) {
|
|
2568
|
+
return value !== null && typeof value === "object" && !Array.isArray(value);
|
|
2569
|
+
}
|
|
2570
|
+
function createElement(options = {}) {
|
|
2571
|
+
const tag = options.tag || "div";
|
|
2572
|
+
const element = document.createElement(tag);
|
|
2573
|
+
if (options.class) {
|
|
2574
|
+
element.className = options.class;
|
|
2575
|
+
}
|
|
2576
|
+
if (options.style) {
|
|
2577
|
+
if (typeof options.style === "string") {
|
|
2578
|
+
element.setAttribute("style", options.style);
|
|
2579
|
+
} else if (typeof options.style === "object") {
|
|
2580
|
+
Object.assign(element.style, options.style);
|
|
2581
|
+
}
|
|
2582
|
+
}
|
|
2583
|
+
if (options.textContent) {
|
|
2584
|
+
element.textContent = options.textContent;
|
|
2585
|
+
}
|
|
2586
|
+
if (options.text) {
|
|
2587
|
+
element.textContent = options.text;
|
|
2588
|
+
}
|
|
2589
|
+
return element;
|
|
2590
|
+
}
|
|
2591
|
+
function addClass2(element, className) {
|
|
2592
|
+
if (element && className) {
|
|
2593
|
+
element.classList.add(className);
|
|
2594
|
+
}
|
|
2595
|
+
}
|
|
2596
|
+
function hasClass(element, className) {
|
|
2597
|
+
return element && element.classList.contains(className);
|
|
2598
|
+
}
|
|
2599
|
+
function createComponentInstance(Component, options = {}) {
|
|
2600
|
+
try {
|
|
2601
|
+
const {
|
|
2602
|
+
layout: layoutConfig,
|
|
2603
|
+
layoutItem: layoutItemConfig,
|
|
2604
|
+
style: styleConfig,
|
|
2605
|
+
attributes: attributesConfig,
|
|
2606
|
+
events: eventsConfig,
|
|
2607
|
+
event,
|
|
2608
|
+
...cleanOptions
|
|
2609
|
+
} = options;
|
|
2610
|
+
const finalEventsConfig = eventsConfig || event;
|
|
2611
|
+
if (styleConfig && typeof styleConfig === "string") {
|
|
2612
|
+
cleanOptions.style = styleConfig;
|
|
2613
|
+
}
|
|
2614
|
+
const isClass = typeof Component === "function" && Object.getOwnPropertyDescriptor(Component, "prototype")?.writable === false;
|
|
2615
|
+
const component = isClass ? new Component(cleanOptions) : Component(cleanOptions);
|
|
2616
|
+
if (component) {
|
|
2617
|
+
const element = component.element || (component instanceof HTMLElement ? component : null);
|
|
2618
|
+
if (element) {
|
|
2619
|
+
if (layoutConfig)
|
|
2620
|
+
applyLayoutClasses(element, layoutConfig);
|
|
2621
|
+
if (layoutItemConfig)
|
|
2622
|
+
applyLayoutItemClasses(element, layoutItemConfig);
|
|
2623
|
+
if (styleConfig && typeof styleConfig === "object") {
|
|
2624
|
+
Object.assign(element.style, styleConfig);
|
|
2625
|
+
}
|
|
2626
|
+
if (attributesConfig && typeof attributesConfig === "object") {
|
|
2627
|
+
for (const [key, value] of Object.entries(attributesConfig)) {
|
|
2628
|
+
if (value !== undefined && value !== null) {
|
|
2629
|
+
element.setAttribute(key, value.toString());
|
|
2630
|
+
}
|
|
2631
|
+
}
|
|
2632
|
+
}
|
|
2633
|
+
if (finalEventsConfig && typeof finalEventsConfig === "object") {
|
|
2634
|
+
if (Array.isArray(finalEventsConfig)) {
|
|
2635
|
+
for (const eventDef of finalEventsConfig) {
|
|
2636
|
+
if (Array.isArray(eventDef) && eventDef.length >= 2) {
|
|
2637
|
+
const [eventName, handler] = eventDef;
|
|
2638
|
+
if (typeof eventName === "string" && typeof handler === "function") {
|
|
2639
|
+
element.addEventListener(eventName, handler);
|
|
2640
|
+
}
|
|
2641
|
+
}
|
|
2642
|
+
}
|
|
2643
|
+
} else {
|
|
2644
|
+
for (const [eventName, handler] of Object.entries(finalEventsConfig)) {
|
|
2645
|
+
if (typeof handler === "function") {
|
|
2646
|
+
element.addEventListener(eventName, handler);
|
|
2647
|
+
}
|
|
2648
|
+
}
|
|
2649
|
+
}
|
|
2650
|
+
}
|
|
2651
|
+
}
|
|
2652
|
+
}
|
|
2653
|
+
return component;
|
|
2654
|
+
} catch (error) {
|
|
2655
|
+
console.error("Error creating component instance:", error);
|
|
2656
|
+
return document.createElement("div");
|
|
2657
|
+
}
|
|
2658
|
+
}
|
|
2659
|
+
function applyLayoutClasses(element, layoutConfig) {
|
|
2660
|
+
if (!element || !layoutConfig)
|
|
2661
|
+
return;
|
|
2662
|
+
if (layoutConfig.type) {
|
|
2663
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout--${layoutConfig.type}`);
|
|
2664
|
+
}
|
|
2665
|
+
const layoutType = layoutConfig.type || getLayoutType(element);
|
|
2666
|
+
if (layoutType) {
|
|
2667
|
+
if (layoutConfig.gap !== undefined) {
|
|
2668
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName(layoutType, "gap", layoutConfig.gap));
|
|
2669
|
+
}
|
|
2670
|
+
if (layoutConfig.align) {
|
|
2671
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName(layoutType, "align", layoutConfig.align));
|
|
2672
|
+
}
|
|
2673
|
+
if (layoutConfig.justify) {
|
|
2674
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName(layoutType, "justify", layoutConfig.justify));
|
|
2675
|
+
}
|
|
2676
|
+
}
|
|
2677
|
+
if (layoutConfig.type === "grid" || getLayoutType(element) === "grid") {
|
|
2678
|
+
if (typeof layoutConfig.columns === "number") {
|
|
2679
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("grid", "cols", layoutConfig.columns));
|
|
2680
|
+
} else if (layoutConfig.columns === "auto-fill") {
|
|
2681
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("grid", "fill", "auto"));
|
|
2682
|
+
} else if (layoutConfig.columns === "auto-fit") {
|
|
2683
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("grid", "cols", "auto-fit"));
|
|
2684
|
+
}
|
|
2685
|
+
if (layoutConfig.dense)
|
|
2686
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout--grid-dense`);
|
|
2687
|
+
if (layoutConfig.autoHeight)
|
|
2688
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout--grid-auto-height`);
|
|
2689
|
+
}
|
|
2690
|
+
if (layoutConfig.type === "row" || getLayoutType(element) === "row") {
|
|
2691
|
+
if (layoutConfig.wrap === false || layoutConfig.wrap === "nowrap") {
|
|
2692
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout--row-nowrap`);
|
|
2693
|
+
} else if (layoutConfig.wrap === "reverse") {
|
|
2694
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout--row-wrap-reverse`);
|
|
2695
|
+
}
|
|
2696
|
+
if (layoutConfig.mobileStack)
|
|
2697
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout--row-mobile-stack`);
|
|
2698
|
+
if (layoutConfig.mobileScroll)
|
|
2699
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout--row-mobile-scroll`);
|
|
2700
|
+
}
|
|
2701
|
+
if (layoutConfig.class) {
|
|
2702
|
+
layoutConfig.class.split(" ").filter(Boolean).forEach((cls) => element.classList.add(cls));
|
|
2703
|
+
}
|
|
2704
|
+
}
|
|
2705
|
+
function applyLayoutItemClasses(element, itemConfig) {
|
|
2706
|
+
if (!element || !itemConfig)
|
|
2707
|
+
return;
|
|
2708
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout__item`);
|
|
2709
|
+
if (itemConfig.width && itemConfig.width >= 1 && itemConfig.width <= 12) {
|
|
2710
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "", itemConfig.width));
|
|
2711
|
+
}
|
|
2712
|
+
if (itemConfig.sm)
|
|
2713
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "sm", itemConfig.sm));
|
|
2714
|
+
if (itemConfig.md)
|
|
2715
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "md", itemConfig.md));
|
|
2716
|
+
if (itemConfig.lg)
|
|
2717
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "lg", itemConfig.lg));
|
|
2718
|
+
if (itemConfig.xl)
|
|
2719
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "xl", itemConfig.xl));
|
|
2720
|
+
if (itemConfig.span)
|
|
2721
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "span", itemConfig.span));
|
|
2722
|
+
if (itemConfig.rowSpan)
|
|
2723
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "row-span", itemConfig.rowSpan));
|
|
2724
|
+
if (itemConfig.order)
|
|
2725
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "order", itemConfig.order));
|
|
2726
|
+
if (itemConfig.align)
|
|
2727
|
+
addClass2(element, PREFIX_WITH_DASH + getCachedClassName("item", "self", itemConfig.align));
|
|
2728
|
+
if (itemConfig.auto)
|
|
2729
|
+
addClass2(element, `${PREFIX_WITH_DASH}layout__item--auto`);
|
|
2730
|
+
}
|
|
2731
|
+
function getLayoutType(element) {
|
|
2732
|
+
return hasClass(element, `${PREFIX_WITH_DASH}layout--stack`) ? "stack" : hasClass(element, `${PREFIX_WITH_DASH}layout--row`) ? "row" : hasClass(element, `${PREFIX_WITH_DASH}layout--grid`) ? "grid" : "";
|
|
2733
|
+
}
|
|
2734
|
+
function processArraySchema(schema, parentElement = null, level = 0, options = {}) {
|
|
2735
|
+
level++;
|
|
2736
|
+
const layout = {};
|
|
2737
|
+
const components = [];
|
|
2738
|
+
const fragment = createFragment();
|
|
2739
|
+
let component = null;
|
|
2740
|
+
if (!Array.isArray(schema)) {
|
|
2741
|
+
return createLayoutResult(layout);
|
|
2742
|
+
}
|
|
2743
|
+
const defaultCreator = options.creator || createElement;
|
|
2744
|
+
for (let i = 0;i < schema.length; i++) {
|
|
2745
|
+
const item = schema[i];
|
|
2746
|
+
if (!item)
|
|
2747
|
+
continue;
|
|
2748
|
+
if (Array.isArray(item)) {
|
|
2749
|
+
const container = component || parentElement;
|
|
2750
|
+
const result = processArraySchema(item, container, level, options);
|
|
2751
|
+
if (Array.isArray(result.layout.components)) {
|
|
2752
|
+
components.push(...result.layout.components);
|
|
2753
|
+
delete result.layout.components;
|
|
2754
|
+
}
|
|
2755
|
+
Object.assign(layout, result.layout);
|
|
2756
|
+
continue;
|
|
2757
|
+
}
|
|
2758
|
+
const {
|
|
2759
|
+
creator,
|
|
2760
|
+
name,
|
|
2761
|
+
options: itemOptions,
|
|
2762
|
+
consumed
|
|
2763
|
+
} = extractParameters(schema, i, defaultCreator);
|
|
2764
|
+
if (!creator) {
|
|
2765
|
+
console.warn("Skipping unsupported item type:", item);
|
|
2766
|
+
continue;
|
|
2767
|
+
}
|
|
2768
|
+
if (creator === createElement && !("tag" in itemOptions)) {
|
|
2769
|
+
itemOptions.tag = "div";
|
|
2770
|
+
}
|
|
2771
|
+
i += consumed - 1;
|
|
2772
|
+
const shouldApplyPrefix = "prefix" in itemOptions ? itemOptions.prefix : options.prefix !== false;
|
|
2773
|
+
const processedOptions = shouldApplyPrefix || itemOptions.rawClass ? processClassNames(itemOptions, !shouldApplyPrefix) : itemOptions;
|
|
2774
|
+
if (name && !("name" in processedOptions) && !(creator === createElement || creator.isElement)) {
|
|
2775
|
+
processedOptions.name = name;
|
|
2776
|
+
}
|
|
2777
|
+
component = createComponentInstance(creator, processedOptions);
|
|
2778
|
+
const element = isComponent(component) ? component.element : component;
|
|
2779
|
+
if (level === 1)
|
|
2780
|
+
layout.element = element;
|
|
2781
|
+
if (name) {
|
|
2782
|
+
layout[name] = component;
|
|
2783
|
+
components.push([name, component]);
|
|
2784
|
+
}
|
|
2785
|
+
if (component) {
|
|
2786
|
+
if ("insert" in component && typeof component.insert === "function") {
|
|
2787
|
+
component.insert(fragment);
|
|
2788
|
+
} else {
|
|
2789
|
+
fragment.appendChild(element);
|
|
2790
|
+
}
|
|
2791
|
+
if (parentElement) {
|
|
2792
|
+
component._container = parentElement;
|
|
2793
|
+
if ("onInserted" in component && typeof component.onInserted === "function") {
|
|
2794
|
+
component.onInserted(parentElement);
|
|
2795
|
+
}
|
|
2796
|
+
}
|
|
2797
|
+
}
|
|
2798
|
+
}
|
|
2799
|
+
if (parentElement && fragment.hasChildNodes()) {
|
|
2800
|
+
const wrapper = isComponent(parentElement) ? parentElement.element : parentElement;
|
|
2801
|
+
wrapper.appendChild(fragment);
|
|
2802
|
+
}
|
|
2803
|
+
releaseFragment(fragment);
|
|
2804
|
+
layout.components = components;
|
|
2805
|
+
return createLayoutResult(layout);
|
|
2806
|
+
}
|
|
2807
|
+
function processObjectSchema(schema, parentElement = null, options = {}) {
|
|
2808
|
+
const layout = {};
|
|
2809
|
+
const defaultCreator = options.creator || createElement;
|
|
2810
|
+
if (schema.element && !parentElement) {
|
|
2811
|
+
const elementDef = schema.element;
|
|
2812
|
+
const createElementFn = elementDef.creator || defaultCreator;
|
|
2813
|
+
const elementOptions = elementDef.options || {};
|
|
2814
|
+
const processedOptions = options.prefix !== false ? processClassNames(elementOptions) : { ...elementOptions };
|
|
2815
|
+
const rootComponent = createComponentInstance(createElementFn, processedOptions);
|
|
2816
|
+
layout.element = rootComponent;
|
|
2817
|
+
if (elementDef.name)
|
|
2818
|
+
layout[elementDef.name] = rootComponent;
|
|
2819
|
+
if (elementDef.children) {
|
|
2820
|
+
const rootElement = isComponent(rootComponent) ? rootComponent.element : rootComponent;
|
|
2821
|
+
const childResult = processObjectSchema(elementDef.children, rootElement, options);
|
|
2822
|
+
Object.assign(layout, childResult.layout);
|
|
2823
|
+
}
|
|
2824
|
+
return createLayoutResult(layout);
|
|
2825
|
+
}
|
|
2826
|
+
const fragment = parentElement ? createFragment() : null;
|
|
2827
|
+
for (const key in schema) {
|
|
2828
|
+
const def = schema[key];
|
|
2829
|
+
if (!def)
|
|
2830
|
+
continue;
|
|
2831
|
+
const elementCreator = def.creator || defaultCreator;
|
|
2832
|
+
const elementOptions = def.options || {};
|
|
2833
|
+
const shouldApplyPrefix = "prefix" in elementOptions ? elementOptions.prefix : options.prefix !== false;
|
|
2834
|
+
const processedOptions = shouldApplyPrefix ? processClassNames(elementOptions) : { ...elementOptions };
|
|
2835
|
+
if (!def.name && key !== "element") {
|
|
2836
|
+
def.name = key;
|
|
2837
|
+
}
|
|
2838
|
+
const created = createComponentInstance(elementCreator, processedOptions);
|
|
2839
|
+
layout[key] = created;
|
|
2840
|
+
if (def.name && def.name !== key)
|
|
2841
|
+
layout[def.name] = created;
|
|
2842
|
+
const element = isComponent(created) ? created.element : created;
|
|
2843
|
+
if (fragment)
|
|
2844
|
+
fragment.appendChild(element);
|
|
2845
|
+
if (def.children) {
|
|
2846
|
+
const childResult = processObjectSchema(def.children, element, options);
|
|
2847
|
+
Object.assign(layout, childResult.layout);
|
|
2848
|
+
}
|
|
2849
|
+
}
|
|
2850
|
+
if (parentElement && fragment) {
|
|
2851
|
+
const parentDom = isComponent(parentElement) ? parentElement.element : parentElement;
|
|
2852
|
+
parentDom.appendChild(fragment);
|
|
2853
|
+
releaseFragment(fragment);
|
|
2854
|
+
}
|
|
2855
|
+
return createLayoutResult(layout);
|
|
2856
|
+
}
|
|
2857
|
+
function flattenLayout(layout) {
|
|
2858
|
+
const flattened = {};
|
|
2859
|
+
if (!layout || typeof layout !== "object")
|
|
2860
|
+
return flattened;
|
|
2861
|
+
for (const key in layout) {
|
|
2862
|
+
const value = layout[key];
|
|
2863
|
+
if (value && typeof value !== "function" && (value instanceof HTMLElement || typeof SVGElement !== "undefined" && value instanceof SVGElement || isComponent(value))) {
|
|
2864
|
+
flattened[key] = value;
|
|
2865
|
+
}
|
|
2866
|
+
}
|
|
2867
|
+
return flattened;
|
|
2868
|
+
}
|
|
2869
|
+
function createLayoutResult(layout) {
|
|
2870
|
+
const flattenedComponents = flattenLayout(layout);
|
|
2871
|
+
return {
|
|
2872
|
+
layout,
|
|
2873
|
+
element: layout.element,
|
|
2874
|
+
component: flattenedComponents,
|
|
2875
|
+
get(name) {
|
|
2876
|
+
return layout[name] ?? null;
|
|
2877
|
+
},
|
|
2878
|
+
getAll() {
|
|
2879
|
+
return flattenedComponents;
|
|
2880
|
+
},
|
|
2881
|
+
destroy() {
|
|
2882
|
+
const destroyed = new Set;
|
|
2883
|
+
const destroyComponent = (component) => {
|
|
2884
|
+
if (!component || destroyed.has(component))
|
|
2885
|
+
return;
|
|
2886
|
+
if (component instanceof HTMLElement || typeof component !== "object")
|
|
2887
|
+
return;
|
|
2888
|
+
if (typeof component.destroy === "function") {
|
|
2889
|
+
destroyed.add(component);
|
|
2890
|
+
try {
|
|
2891
|
+
component.destroy();
|
|
2892
|
+
} catch (e) {}
|
|
2893
|
+
}
|
|
2894
|
+
};
|
|
2895
|
+
if (Array.isArray(layout.components)) {
|
|
2896
|
+
for (const [name, component] of layout.components) {
|
|
2897
|
+
destroyComponent(component);
|
|
2898
|
+
}
|
|
2899
|
+
}
|
|
2900
|
+
for (const key in layout) {
|
|
2901
|
+
if (key === "element" || key === "components")
|
|
2902
|
+
continue;
|
|
2903
|
+
destroyComponent(layout[key]);
|
|
2904
|
+
}
|
|
2905
|
+
if (Array.isArray(layout.components)) {
|
|
2906
|
+
layout.components.length = 0;
|
|
2907
|
+
}
|
|
2908
|
+
if (layout.element) {
|
|
2909
|
+
const element = isComponent(layout.element) ? layout.element.element : layout.element;
|
|
2910
|
+
if (element && element.parentNode) {
|
|
2911
|
+
element.parentNode.removeChild(element);
|
|
2912
|
+
}
|
|
2913
|
+
}
|
|
2914
|
+
for (const key in layout) {
|
|
2915
|
+
delete layout[key];
|
|
2916
|
+
}
|
|
2917
|
+
}
|
|
2918
|
+
};
|
|
2919
|
+
}
|
|
2920
|
+
function createLayout(schema, parentElement = null, options = {}) {
|
|
2921
|
+
if (typeof schema === "function") {
|
|
2922
|
+
schema = schema();
|
|
2923
|
+
}
|
|
2924
|
+
if (typeof schema === "string") {
|
|
2925
|
+
const template = document.createElement("template");
|
|
2926
|
+
template.innerHTML = schema.trim();
|
|
2927
|
+
const fragment = template.content;
|
|
2928
|
+
if (parentElement && fragment.hasChildNodes()) {
|
|
2929
|
+
parentElement.appendChild(fragment);
|
|
2930
|
+
}
|
|
2931
|
+
const layout = { element: fragment.firstElementChild };
|
|
2932
|
+
return createLayoutResult(layout);
|
|
2933
|
+
}
|
|
2934
|
+
if (Array.isArray(schema) && schema.length >= 3 && typeof schema[0] === "function" && typeof schema[1] === "string" && isObject(schema[2])) {
|
|
2935
|
+
return processArraySchema(schema, parentElement, 0, options);
|
|
2936
|
+
}
|
|
2937
|
+
return Array.isArray(schema) ? processArraySchema(schema, parentElement, 0, options) : processObjectSchema(schema, parentElement, options);
|
|
2938
|
+
}
|
|
2939
|
+
// src/core/viewport/features/rendering.ts
|
|
2940
|
+
var withRendering = (config = {}) => {
|
|
2941
|
+
return (component) => {
|
|
2942
|
+
const {
|
|
2943
|
+
template,
|
|
2944
|
+
overscan = 5,
|
|
2945
|
+
measureItems = false,
|
|
2946
|
+
enableRecycling = true,
|
|
2947
|
+
maxPoolSize = VIEWPORT_CONSTANTS.RENDERING.DEFAULT_MAX_POOL_SIZE,
|
|
2948
|
+
maintainDomOrder = true
|
|
2949
|
+
} = config;
|
|
2950
|
+
const getCollectionItems = () => {
|
|
2951
|
+
return component.collection?.items || component.items || [];
|
|
2952
|
+
};
|
|
2953
|
+
const renderedElements = new Map;
|
|
2954
|
+
const elementPool = [];
|
|
2955
|
+
const poolStats = { created: 0, recycled: 0, poolSize: 0, released: 0 };
|
|
2956
|
+
const layoutResults = new WeakMap;
|
|
2957
|
+
const templateParser = document.createElement("template");
|
|
2958
|
+
let viewportState = null;
|
|
2959
|
+
let currentVisibleRange = { start: 0, end: 0 };
|
|
2960
|
+
let lastRenderTime = 0;
|
|
2961
|
+
let isRemovingItem = false;
|
|
2962
|
+
const getPooledElement = () => {
|
|
2963
|
+
if (enableRecycling && elementPool.length > 0) {
|
|
2964
|
+
poolStats.recycled++;
|
|
2965
|
+
return elementPool.pop();
|
|
2966
|
+
}
|
|
2967
|
+
const element = document.createElement("div");
|
|
2968
|
+
element.className = "mtrl-viewport-item";
|
|
2969
|
+
poolStats.created++;
|
|
2970
|
+
return element;
|
|
2971
|
+
};
|
|
2972
|
+
const releaseElement = (element) => {
|
|
2973
|
+
poolStats.released++;
|
|
2974
|
+
const layoutResult = layoutResults.get(element);
|
|
2975
|
+
if (layoutResult && typeof layoutResult.destroy === "function") {
|
|
2976
|
+
try {
|
|
2977
|
+
layoutResult.destroy();
|
|
2978
|
+
} catch (e) {}
|
|
2979
|
+
layoutResults.delete(element);
|
|
2980
|
+
}
|
|
2981
|
+
if (!enableRecycling) {
|
|
2982
|
+
element.remove();
|
|
2983
|
+
return;
|
|
2984
|
+
}
|
|
2985
|
+
element.className = "mtrl-viewport-item";
|
|
2986
|
+
element.removeAttribute("data-index");
|
|
2987
|
+
element.style.cssText = "";
|
|
2988
|
+
element.innerHTML = "";
|
|
2989
|
+
if (element.parentNode) {
|
|
2990
|
+
element.parentNode.removeChild(element);
|
|
2991
|
+
}
|
|
2992
|
+
if (elementPool.length < maxPoolSize) {
|
|
2993
|
+
elementPool.push(element);
|
|
2994
|
+
poolStats.poolSize = elementPool.length;
|
|
2995
|
+
}
|
|
2996
|
+
};
|
|
2997
|
+
wrapInitialize(component, () => {
|
|
2998
|
+
viewportState = getViewportState(component);
|
|
2999
|
+
if (viewportState?.itemsContainer && viewportState.virtualTotalSize > 0) {
|
|
3000
|
+
viewportState.itemsContainer.style.height = `${viewportState.virtualTotalSize}px`;
|
|
3001
|
+
}
|
|
3002
|
+
component.on?.("item:update-request", (data) => {
|
|
3003
|
+
const { index, item, previousItem } = data;
|
|
3004
|
+
const existingElement = renderedElements.get(index);
|
|
3005
|
+
const wasVisible = !!existingElement;
|
|
3006
|
+
const wasSelected = existingElement ? import_mtrl3.hasClass(existingElement, VIEWPORT_CONSTANTS.SELECTION.SELECTED_CLASS) || import_mtrl3.hasClass(existingElement, "mtrl-viewport-item--selected") : false;
|
|
3007
|
+
if (existingElement && existingElement.parentNode) {
|
|
3008
|
+
const newElement = renderItem(item, index);
|
|
3009
|
+
if (newElement) {
|
|
3010
|
+
Object.assign(newElement.style, {
|
|
3011
|
+
position: existingElement.style.position,
|
|
3012
|
+
transform: existingElement.style.transform,
|
|
3013
|
+
width: existingElement.style.width
|
|
3014
|
+
});
|
|
3015
|
+
if (wasSelected) {
|
|
3016
|
+
import_mtrl3.addClass(newElement, VIEWPORT_CONSTANTS.SELECTION.SELECTED_CLASS);
|
|
3017
|
+
import_mtrl3.addClass(newElement, "mtrl-viewport-item--selected");
|
|
3018
|
+
}
|
|
3019
|
+
import_mtrl3.addClass(newElement, "viewport-item--updated");
|
|
3020
|
+
const innerItem = newElement.firstElementChild;
|
|
3021
|
+
if (innerItem) {
|
|
3022
|
+
import_mtrl3.addClass(innerItem, "item--updated");
|
|
3023
|
+
}
|
|
3024
|
+
existingElement.parentNode.replaceChild(newElement, existingElement);
|
|
3025
|
+
renderedElements.set(index, newElement);
|
|
3026
|
+
releaseElement(existingElement);
|
|
3027
|
+
setTimeout(() => {
|
|
3028
|
+
import_mtrl3.removeClass(newElement, "viewport-item--updated");
|
|
3029
|
+
if (innerItem) {
|
|
3030
|
+
import_mtrl3.removeClass(innerItem, "item--updated");
|
|
3031
|
+
}
|
|
3032
|
+
}, 500);
|
|
3033
|
+
}
|
|
3034
|
+
}
|
|
3035
|
+
component.emit?.("item:updated", {
|
|
3036
|
+
item,
|
|
3037
|
+
index,
|
|
3038
|
+
previousItem,
|
|
3039
|
+
wasVisible,
|
|
3040
|
+
wasSelected
|
|
3041
|
+
});
|
|
3042
|
+
});
|
|
3043
|
+
component.on?.("item:remove-request", (data) => {
|
|
3044
|
+
const { index, item } = data;
|
|
3045
|
+
isRemovingItem = true;
|
|
3046
|
+
const existingElement = renderedElements.get(index);
|
|
3047
|
+
if (existingElement && existingElement.parentNode) {
|
|
3048
|
+
releaseElement(existingElement);
|
|
3049
|
+
}
|
|
3050
|
+
renderedElements.delete(index);
|
|
3051
|
+
const renderedKeys = Array.from(renderedElements.keys()).sort((a, b) => a - b);
|
|
3052
|
+
const newRenderedElements = new Map;
|
|
3053
|
+
for (const key of renderedKeys) {
|
|
3054
|
+
if (key > index) {
|
|
3055
|
+
const element = renderedElements.get(key);
|
|
3056
|
+
if (element) {
|
|
3057
|
+
element.dataset.index = String(key - 1);
|
|
3058
|
+
newRenderedElements.set(key - 1, element);
|
|
3059
|
+
}
|
|
3060
|
+
} else if (key < index) {
|
|
3061
|
+
const element = renderedElements.get(key);
|
|
3062
|
+
if (element) {
|
|
3063
|
+
newRenderedElements.set(key, element);
|
|
3064
|
+
}
|
|
3065
|
+
}
|
|
3066
|
+
}
|
|
3067
|
+
renderedElements.clear();
|
|
3068
|
+
for (const [key, element] of newRenderedElements) {
|
|
3069
|
+
renderedElements.set(key, element);
|
|
3070
|
+
}
|
|
3071
|
+
if (viewportState && viewportState.totalItems > 0) {
|
|
3072
|
+
viewportState.totalItems--;
|
|
3073
|
+
}
|
|
3074
|
+
currentVisibleRange = { start: -1, end: -1 };
|
|
3075
|
+
component.emit?.("item:removed", {
|
|
3076
|
+
item,
|
|
3077
|
+
index
|
|
3078
|
+
});
|
|
3079
|
+
renderItems();
|
|
3080
|
+
isRemovingItem = false;
|
|
3081
|
+
});
|
|
3082
|
+
component.on?.("items:add-request", (data) => {
|
|
3083
|
+
const { items: newItems, position, previousCount } = data;
|
|
3084
|
+
if (!newItems || newItems.length === 0)
|
|
3085
|
+
return;
|
|
3086
|
+
const itemsAdded = newItems.length;
|
|
3087
|
+
if (position === "start") {
|
|
3088
|
+
const renderedKeys = Array.from(renderedElements.keys()).sort((a, b) => b - a);
|
|
3089
|
+
const newRenderedElements = new Map;
|
|
3090
|
+
for (const key of renderedKeys) {
|
|
3091
|
+
const element = renderedElements.get(key);
|
|
3092
|
+
const newIndex = key + itemsAdded;
|
|
3093
|
+
element.setAttribute("data-index", String(newIndex));
|
|
3094
|
+
newRenderedElements.set(newIndex, element);
|
|
3095
|
+
}
|
|
3096
|
+
renderedElements.clear();
|
|
3097
|
+
newRenderedElements.forEach((el, idx) => renderedElements.set(idx, el));
|
|
3098
|
+
}
|
|
3099
|
+
if (viewportState) {
|
|
3100
|
+
viewportState.totalItems = (viewportState.totalItems || 0) + itemsAdded;
|
|
3101
|
+
}
|
|
3102
|
+
currentVisibleRange = { start: -1, end: -1 };
|
|
3103
|
+
const collection = component.viewport?.collection;
|
|
3104
|
+
if (collection?.getLoadedRanges) {
|
|
3105
|
+
const loadedRanges = collection.getLoadedRanges();
|
|
3106
|
+
loadedRanges.clear();
|
|
3107
|
+
}
|
|
3108
|
+
renderItems();
|
|
3109
|
+
component.emit?.("items:render-complete", {
|
|
3110
|
+
items: newItems,
|
|
3111
|
+
position,
|
|
3112
|
+
total: viewportState?.totalItems || 0
|
|
3113
|
+
});
|
|
3114
|
+
});
|
|
3115
|
+
component.on?.("collection:items-evicted", (data) => {
|
|
3116
|
+
const { keepStart, keepEnd } = data;
|
|
3117
|
+
renderedElements.forEach((element, index) => {
|
|
3118
|
+
if (index < keepStart || index > keepEnd) {
|
|
3119
|
+
releaseElement(element);
|
|
3120
|
+
renderedElements.delete(index);
|
|
3121
|
+
}
|
|
3122
|
+
});
|
|
3123
|
+
});
|
|
3124
|
+
component.on?.("reload:start", () => {
|
|
3125
|
+
currentVisibleRange = { start: -1, end: -1 };
|
|
3126
|
+
lastRenderTime = 0;
|
|
3127
|
+
isRemovingItem = false;
|
|
3128
|
+
elementPool.length = 0;
|
|
3129
|
+
poolStats.poolSize = 0;
|
|
3130
|
+
});
|
|
3131
|
+
component.on?.("collection:reset", () => {
|
|
3132
|
+
renderedElements.forEach((element) => {
|
|
3133
|
+
releaseElement(element);
|
|
3134
|
+
});
|
|
3135
|
+
renderedElements.clear();
|
|
3136
|
+
currentVisibleRange = { start: -1, end: -1 };
|
|
3137
|
+
});
|
|
3138
|
+
component.on?.("viewport:clear-pool", () => {
|
|
3139
|
+
elementPool.length = 0;
|
|
3140
|
+
poolStats.poolSize = 0;
|
|
3141
|
+
lastRenderTime = 0;
|
|
3142
|
+
});
|
|
3143
|
+
component.on?.("collection:range-loaded", (data) => {
|
|
3144
|
+
if (!data.items?.length)
|
|
3145
|
+
return;
|
|
3146
|
+
const placeholders = component.placeholders;
|
|
3147
|
+
if (placeholders && !placeholders.hasAnalyzedStructure()) {
|
|
3148
|
+
placeholders.analyzeDataStructure(data.items);
|
|
3149
|
+
}
|
|
3150
|
+
data.items.forEach((item, i) => {
|
|
3151
|
+
const index = data.offset + i;
|
|
3152
|
+
const element = renderedElements.get(index);
|
|
3153
|
+
const isPlaceholderElement = element?.classList.contains(VIEWPORT_CONSTANTS.PLACEHOLDER.CLASS) || element?.classList.contains(`mtrl-${VIEWPORT_CONSTANTS.PLACEHOLDER.CLASS}`);
|
|
3154
|
+
if (isPlaceholderElement) {
|
|
3155
|
+
const newElement = renderItem(item, index);
|
|
3156
|
+
if (newElement) {
|
|
3157
|
+
import_mtrl3.removeClass(newElement, VIEWPORT_CONSTANTS.PLACEHOLDER.CLASS);
|
|
3158
|
+
if (newElement.firstElementChild) {
|
|
3159
|
+
import_mtrl3.removeClass(newElement.firstElementChild, VIEWPORT_CONSTANTS.PLACEHOLDER.CLASS);
|
|
3160
|
+
}
|
|
3161
|
+
import_mtrl3.addClass(newElement, "viewport-item--replaced");
|
|
3162
|
+
if (element) {
|
|
3163
|
+
Object.assign(newElement.style, {
|
|
3164
|
+
position: element.style.position,
|
|
3165
|
+
transform: element.style.transform,
|
|
3166
|
+
width: element.style.width
|
|
3167
|
+
});
|
|
3168
|
+
element.parentNode?.replaceChild(newElement, element);
|
|
3169
|
+
}
|
|
3170
|
+
renderedElements.set(index, newElement);
|
|
3171
|
+
if (element) {
|
|
3172
|
+
releaseElement(element);
|
|
3173
|
+
}
|
|
3174
|
+
setTimeout(() => {
|
|
3175
|
+
import_mtrl3.removeClass(newElement, "viewport-item--replaced");
|
|
3176
|
+
}, 300);
|
|
3177
|
+
} else {
|
|
3178
|
+
if (element) {
|
|
3179
|
+
releaseElement(element);
|
|
3180
|
+
}
|
|
3181
|
+
renderedElements.delete(index);
|
|
3182
|
+
}
|
|
3183
|
+
}
|
|
3184
|
+
});
|
|
3185
|
+
const { visibleRange } = viewportState || {};
|
|
3186
|
+
if (visibleRange) {
|
|
3187
|
+
const renderStart = Math.max(0, visibleRange.start - overscan);
|
|
3188
|
+
const renderEnd = Math.min(viewportState?.totalItems ?? 0 - 1, visibleRange.end + overscan);
|
|
3189
|
+
const loadedStart = data.offset;
|
|
3190
|
+
const loadedEnd = data.offset + data.items.length - 1;
|
|
3191
|
+
const needsRender = loadedStart <= renderEnd && loadedEnd >= renderStart && renderedElements.size < renderEnd - renderStart + 1;
|
|
3192
|
+
if (needsRender)
|
|
3193
|
+
renderItems();
|
|
3194
|
+
}
|
|
3195
|
+
});
|
|
3196
|
+
component.on?.("viewport:range-changed", renderItems);
|
|
3197
|
+
component.on?.("viewport:scroll", updateItemPositions);
|
|
3198
|
+
component.on?.("viewport:virtual-size-changed", (data) => {
|
|
3199
|
+
if (viewportState?.itemsContainer && data.totalVirtualSize !== undefined) {
|
|
3200
|
+
viewportState.itemsContainer.style.height = `${data.totalVirtualSize}px`;
|
|
3201
|
+
}
|
|
3202
|
+
});
|
|
3203
|
+
});
|
|
3204
|
+
const getDefaultTemplate = () => (item, index) => {
|
|
3205
|
+
return [
|
|
3206
|
+
{
|
|
3207
|
+
tag: "div",
|
|
3208
|
+
class: "viewport-item",
|
|
3209
|
+
text: typeof item === "object" ? item.name || item.label || item.text || `Item ${index}` : String(item)
|
|
3210
|
+
}
|
|
3211
|
+
];
|
|
3212
|
+
};
|
|
3213
|
+
const processLayoutSchema = (schema, item, index) => {
|
|
3214
|
+
if (typeof schema === "string") {
|
|
3215
|
+
return schema.replace(/\{\{([^}]+)\}\}/g, (match, key) => {
|
|
3216
|
+
if (key === "index")
|
|
3217
|
+
return String(index);
|
|
3218
|
+
if (key === "item")
|
|
3219
|
+
return String(item);
|
|
3220
|
+
const value = key.split(".").reduce((obj, prop) => {
|
|
3221
|
+
return obj?.[prop.trim()];
|
|
3222
|
+
}, item);
|
|
3223
|
+
return value !== undefined ? String(value) : match;
|
|
3224
|
+
});
|
|
3225
|
+
}
|
|
3226
|
+
if (Array.isArray(schema)) {
|
|
3227
|
+
return schema.map((child) => processLayoutSchema(child, item, index));
|
|
3228
|
+
}
|
|
3229
|
+
if (typeof schema === "object" && schema !== null) {
|
|
3230
|
+
const processed = {};
|
|
3231
|
+
for (const [key, value] of Object.entries(schema)) {
|
|
3232
|
+
processed[key] = processLayoutSchema(value, item, index);
|
|
3233
|
+
}
|
|
3234
|
+
return processed;
|
|
3235
|
+
}
|
|
3236
|
+
return schema;
|
|
3237
|
+
};
|
|
3238
|
+
const calculateItemPosition = (index, scrollPosition, totalItems, itemSize, virtualTotalSize, containerSize, targetScrollIndex) => {
|
|
3239
|
+
const actualTotalSize = totalItems * itemSize;
|
|
3240
|
+
const isCompressed = actualTotalSize > virtualTotalSize && virtualTotalSize > 0;
|
|
3241
|
+
if (targetScrollIndex !== undefined && targetScrollIndex > 0) {
|
|
3242
|
+
return (index - targetScrollIndex) * itemSize;
|
|
3243
|
+
}
|
|
3244
|
+
if (!isCompressed || totalItems === 0) {
|
|
3245
|
+
return index * itemSize - scrollPosition;
|
|
3246
|
+
}
|
|
3247
|
+
const maxScrollPosition = virtualTotalSize - containerSize;
|
|
3248
|
+
const distanceFromBottom = maxScrollPosition - scrollPosition;
|
|
3249
|
+
const nearBottomThreshold = containerSize;
|
|
3250
|
+
if (distanceFromBottom <= nearBottomThreshold && distanceFromBottom >= -1) {
|
|
3251
|
+
const itemsAtBottom = Math.floor(containerSize / itemSize);
|
|
3252
|
+
const firstVisibleAtBottom = Math.max(0, totalItems - itemsAtBottom);
|
|
3253
|
+
const scrollRatio2 = scrollPosition / virtualTotalSize;
|
|
3254
|
+
const exactScrollIndex = scrollRatio2 * totalItems;
|
|
3255
|
+
const interpolation = Math.max(0, Math.min(1, 1 - distanceFromBottom / nearBottomThreshold));
|
|
3256
|
+
const bottomPosition = (index - firstVisibleAtBottom) * itemSize;
|
|
3257
|
+
const normalPosition = (index - exactScrollIndex) * itemSize;
|
|
3258
|
+
return normalPosition + (bottomPosition - normalPosition) * interpolation;
|
|
3259
|
+
}
|
|
3260
|
+
const scrollRatio = scrollPosition / virtualTotalSize;
|
|
3261
|
+
return (index - scrollRatio * totalItems) * itemSize;
|
|
3262
|
+
};
|
|
3263
|
+
const renderItem = (item, index) => {
|
|
3264
|
+
const itemTemplate = template || getDefaultTemplate();
|
|
3265
|
+
try {
|
|
3266
|
+
const result = itemTemplate(item, index);
|
|
3267
|
+
let element;
|
|
3268
|
+
if (Array.isArray(result) || typeof result === "object" && result !== null && !(result instanceof HTMLElement)) {
|
|
3269
|
+
const processedSchema = processLayoutSchema(result, item, index);
|
|
3270
|
+
const layoutResult = createLayout(processedSchema);
|
|
3271
|
+
element = layoutResult.element;
|
|
3272
|
+
if (element && element.nodeType === 1) {
|
|
3273
|
+
layoutResults.set(element, layoutResult);
|
|
3274
|
+
} else {
|
|
3275
|
+
element = getPooledElement();
|
|
3276
|
+
if (layoutResult.element) {
|
|
3277
|
+
element.appendChild(layoutResult.element);
|
|
3278
|
+
layoutResults.set(element, layoutResult);
|
|
3279
|
+
}
|
|
3280
|
+
}
|
|
3281
|
+
} else if (typeof result === "string") {
|
|
3282
|
+
templateParser.innerHTML = result;
|
|
3283
|
+
const content = templateParser.content;
|
|
3284
|
+
if (content.children.length === 1) {
|
|
3285
|
+
element = content.firstElementChild;
|
|
3286
|
+
content.removeChild(element);
|
|
3287
|
+
import_mtrl3.addClass(element, "mtrl-viewport-item");
|
|
3288
|
+
if (isPlaceholder(item)) {
|
|
3289
|
+
import_mtrl3.addClass(element, VIEWPORT_CONSTANTS.PLACEHOLDER.CLASS);
|
|
3290
|
+
}
|
|
3291
|
+
} else {
|
|
3292
|
+
element = getPooledElement();
|
|
3293
|
+
while (content.firstChild) {
|
|
3294
|
+
element.appendChild(content.firstChild);
|
|
3295
|
+
}
|
|
3296
|
+
}
|
|
3297
|
+
} else if (result instanceof HTMLElement) {
|
|
3298
|
+
element = getPooledElement();
|
|
3299
|
+
element.appendChild(result);
|
|
3300
|
+
} else {
|
|
3301
|
+
console.warn(`[Rendering] Invalid template result for item ${index}`);
|
|
3302
|
+
return null;
|
|
3303
|
+
}
|
|
3304
|
+
if (!import_mtrl3.hasClass(element, "viewport-item"))
|
|
3305
|
+
import_mtrl3.addClass(element, "viewport-item");
|
|
3306
|
+
if (isPlaceholder(item)) {
|
|
3307
|
+
import_mtrl3.addClass(element, VIEWPORT_CONSTANTS.PLACEHOLDER.CLASS);
|
|
3308
|
+
}
|
|
3309
|
+
element.dataset.index = String(index);
|
|
3310
|
+
const itemId = item?._id || item?.id;
|
|
3311
|
+
if (itemId !== undefined && itemId !== null) {
|
|
3312
|
+
element.dataset.id = String(itemId);
|
|
3313
|
+
}
|
|
3314
|
+
return element;
|
|
3315
|
+
} catch (error) {
|
|
3316
|
+
console.error(`[Rendering] Error rendering item ${index}:`, error);
|
|
3317
|
+
return null;
|
|
3318
|
+
}
|
|
3319
|
+
};
|
|
3320
|
+
const updateItemPositions = () => {
|
|
3321
|
+
if (!viewportState || renderedElements.size === 0)
|
|
3322
|
+
return;
|
|
3323
|
+
const {
|
|
3324
|
+
scrollPosition,
|
|
3325
|
+
itemSize,
|
|
3326
|
+
totalItems,
|
|
3327
|
+
virtualTotalSize,
|
|
3328
|
+
containerSize
|
|
3329
|
+
} = viewportState;
|
|
3330
|
+
const actualTotalSize = totalItems * itemSize;
|
|
3331
|
+
const isCompressed = actualTotalSize > virtualTotalSize && virtualTotalSize > 0;
|
|
3332
|
+
const sortedIndices = Array.from(renderedElements.keys()).sort((a, b) => a - b);
|
|
3333
|
+
if (!sortedIndices.length)
|
|
3334
|
+
return;
|
|
3335
|
+
const firstIndex = sortedIndices[0];
|
|
3336
|
+
let currentPosition = 0;
|
|
3337
|
+
if (isCompressed) {
|
|
3338
|
+
const maxScroll = virtualTotalSize - containerSize;
|
|
3339
|
+
const distanceFromBottom = maxScroll - scrollPosition;
|
|
3340
|
+
if (distanceFromBottom <= containerSize && distanceFromBottom >= -1) {
|
|
3341
|
+
const itemsAtBottom = Math.floor(containerSize / itemSize);
|
|
3342
|
+
const firstVisibleAtBottom = Math.max(0, totalItems - itemsAtBottom);
|
|
3343
|
+
const scrollRatio = scrollPosition / virtualTotalSize;
|
|
3344
|
+
const exactScrollIndex = scrollRatio * totalItems;
|
|
3345
|
+
const interpolation = Math.max(0, Math.min(1, 1 - distanceFromBottom / containerSize));
|
|
3346
|
+
const bottomPos = (firstIndex - firstVisibleAtBottom) * itemSize;
|
|
3347
|
+
const normalPos = (firstIndex - exactScrollIndex) * itemSize;
|
|
3348
|
+
currentPosition = normalPos + (bottomPos - normalPos) * interpolation;
|
|
3349
|
+
} else {
|
|
3350
|
+
const scrollRatio = scrollPosition / virtualTotalSize;
|
|
3351
|
+
currentPosition = (firstIndex - scrollRatio * totalItems) * itemSize;
|
|
3352
|
+
}
|
|
3353
|
+
} else {
|
|
3354
|
+
currentPosition = firstIndex * itemSize - scrollPosition;
|
|
3355
|
+
}
|
|
3356
|
+
sortedIndices.forEach((index) => {
|
|
3357
|
+
const element = renderedElements.get(index);
|
|
3358
|
+
if (element) {
|
|
3359
|
+
element.style.transform = `translateY(${Math.round(currentPosition)}px)`;
|
|
3360
|
+
currentPosition += itemSize;
|
|
3361
|
+
}
|
|
3362
|
+
});
|
|
3363
|
+
};
|
|
3364
|
+
const renderItems = () => {
|
|
3365
|
+
if (!viewportState?.itemsContainer) {
|
|
3366
|
+
return;
|
|
3367
|
+
}
|
|
3368
|
+
const {
|
|
3369
|
+
visibleRange,
|
|
3370
|
+
itemsContainer,
|
|
3371
|
+
totalItems,
|
|
3372
|
+
itemSize,
|
|
3373
|
+
scrollPosition,
|
|
3374
|
+
containerSize,
|
|
3375
|
+
virtualTotalSize
|
|
3376
|
+
} = viewportState;
|
|
3377
|
+
if (!visibleRange || totalItems <= 0 || visibleRange.start < 0 || visibleRange.start >= totalItems || visibleRange.end < visibleRange.start || isNaN(visibleRange.start) || isNaN(visibleRange.end)) {
|
|
3378
|
+
if (totalItems <= 0 && renderedElements.size > 0) {
|
|
3379
|
+
Array.from(renderedElements.entries()).forEach(([index, element]) => {
|
|
3380
|
+
if (element.parentNode)
|
|
3381
|
+
releaseElement(element);
|
|
3382
|
+
renderedElements.delete(index);
|
|
3383
|
+
});
|
|
3384
|
+
currentVisibleRange = { start: -1, end: -1 };
|
|
3385
|
+
}
|
|
3386
|
+
return;
|
|
3387
|
+
}
|
|
3388
|
+
if (visibleRange.start === currentVisibleRange.start && visibleRange.end === currentVisibleRange.end && renderedElements.size > 0) {
|
|
3389
|
+
updateItemPositions();
|
|
3390
|
+
return;
|
|
3391
|
+
}
|
|
3392
|
+
lastRenderTime = Date.now();
|
|
3393
|
+
const renderStart = Math.max(0, visibleRange.start - overscan);
|
|
3394
|
+
const renderEnd = Math.min(totalItems - 1, visibleRange.end + overscan);
|
|
3395
|
+
const toRemove = Array.from(renderedElements.entries()).filter(([index]) => index < renderStart || index > renderEnd);
|
|
3396
|
+
toRemove.forEach(([index, element]) => {
|
|
3397
|
+
if (element.parentNode)
|
|
3398
|
+
releaseElement(element);
|
|
3399
|
+
renderedElements.delete(index);
|
|
3400
|
+
});
|
|
3401
|
+
const items = getCollectionItems();
|
|
3402
|
+
const missingItems = [];
|
|
3403
|
+
const newElements = [];
|
|
3404
|
+
for (let i = renderStart;i <= renderEnd; i++) {
|
|
3405
|
+
if (i < 0 || i >= totalItems || renderedElements.has(i))
|
|
3406
|
+
continue;
|
|
3407
|
+
let item = items[i];
|
|
3408
|
+
if (!item) {
|
|
3409
|
+
missingItems.push(i);
|
|
3410
|
+
const placeholders = component.placeholders;
|
|
3411
|
+
item = placeholders?.generatePlaceholderItem(i) || {
|
|
3412
|
+
_placeholder: true,
|
|
3413
|
+
index: i,
|
|
3414
|
+
id: `placeholder-${i}`,
|
|
3415
|
+
name: VIEWPORT_CONSTANTS.PLACEHOLDER.MASK_CHARACTER.repeat(15),
|
|
3416
|
+
text: VIEWPORT_CONSTANTS.PLACEHOLDER.MASK_CHARACTER.repeat(25),
|
|
3417
|
+
description: VIEWPORT_CONSTANTS.PLACEHOLDER.MASK_CHARACTER.repeat(40)
|
|
3418
|
+
};
|
|
3419
|
+
}
|
|
3420
|
+
const element = renderItem(item, i);
|
|
3421
|
+
if (element) {
|
|
3422
|
+
const targetScrollIndex = viewportState?.targetScrollIndex;
|
|
3423
|
+
const position = calculateItemPosition(i, scrollPosition, totalItems, itemSize, virtualTotalSize, containerSize, targetScrollIndex);
|
|
3424
|
+
Object.assign(element.style, {
|
|
3425
|
+
position: "absolute",
|
|
3426
|
+
transform: `translateY(${position}px)`,
|
|
3427
|
+
width: "100%"
|
|
3428
|
+
});
|
|
3429
|
+
if (maintainDomOrder) {
|
|
3430
|
+
newElements.push([i, element]);
|
|
3431
|
+
} else {
|
|
3432
|
+
itemsContainer.appendChild(element);
|
|
3433
|
+
}
|
|
3434
|
+
renderedElements.set(i, element);
|
|
3435
|
+
}
|
|
3436
|
+
}
|
|
3437
|
+
if (maintainDomOrder && newElements.length > 0) {
|
|
3438
|
+
newElements.sort(([a], [b]) => a - b);
|
|
3439
|
+
const existingChildren = Array.from(itemsContainer.children);
|
|
3440
|
+
let newIdx = 0;
|
|
3441
|
+
let existingIdx = 0;
|
|
3442
|
+
while (newIdx < newElements.length) {
|
|
3443
|
+
const [newIndex, newElement] = newElements[newIdx];
|
|
3444
|
+
while (existingIdx < existingChildren.length && parseInt(existingChildren[existingIdx].dataset.index || "-1", 10) < newIndex) {
|
|
3445
|
+
existingIdx++;
|
|
3446
|
+
}
|
|
3447
|
+
if (existingIdx < existingChildren.length) {
|
|
3448
|
+
itemsContainer.insertBefore(newElement, existingChildren[existingIdx]);
|
|
3449
|
+
} else {
|
|
3450
|
+
itemsContainer.appendChild(newElement);
|
|
3451
|
+
}
|
|
3452
|
+
newIdx++;
|
|
3453
|
+
}
|
|
3454
|
+
}
|
|
3455
|
+
if (missingItems.length > 0 && component.viewport?.collection?.loadMissingRanges) {
|
|
3456
|
+
component.viewport.collection.loadMissingRanges({
|
|
3457
|
+
start: Math.min(...missingItems),
|
|
3458
|
+
end: Math.max(...missingItems)
|
|
3459
|
+
}, "rendering:missing-items");
|
|
3460
|
+
}
|
|
3461
|
+
currentVisibleRange = visibleRange;
|
|
3462
|
+
const renderedElementsArray = Array.from(renderedElements.values());
|
|
3463
|
+
component.emit?.("viewport:items-rendered", {
|
|
3464
|
+
elements: renderedElementsArray,
|
|
3465
|
+
range: visibleRange
|
|
3466
|
+
});
|
|
3467
|
+
component.emit?.("viewport:rendered", {
|
|
3468
|
+
range: visibleRange,
|
|
3469
|
+
renderedCount: renderedElements.size
|
|
3470
|
+
});
|
|
3471
|
+
updateItemPositions();
|
|
3472
|
+
if (renderedElements.size === 0 && totalItems > 0 && component.viewport?.collection) {
|
|
3473
|
+
component.viewport.collection.loadMissingRanges(visibleRange, "rendering:no-items");
|
|
3474
|
+
}
|
|
3475
|
+
};
|
|
3476
|
+
const originalRenderItems = component.viewport.renderItems;
|
|
3477
|
+
component.viewport.renderItems = () => {
|
|
3478
|
+
renderItems();
|
|
3479
|
+
originalRenderItems?.();
|
|
3480
|
+
};
|
|
3481
|
+
wrapDestroy(component, () => {
|
|
3482
|
+
renderedElements.forEach((element) => {
|
|
3483
|
+
releaseElement(element);
|
|
3484
|
+
});
|
|
3485
|
+
renderedElements.clear();
|
|
3486
|
+
elementPool.length = 0;
|
|
3487
|
+
currentVisibleRange = { start: -1, end: -1 };
|
|
3488
|
+
viewportState = null;
|
|
3489
|
+
lastRenderTime = 0;
|
|
3490
|
+
poolStats.created = 0;
|
|
3491
|
+
poolStats.recycled = 0;
|
|
3492
|
+
poolStats.poolSize = 0;
|
|
3493
|
+
});
|
|
3494
|
+
return component;
|
|
3495
|
+
};
|
|
3496
|
+
};
|
|
3497
|
+
|
|
3498
|
+
// src/core/viewport/features/events.ts
|
|
3499
|
+
var withEvents = (config = {}) => {
|
|
3500
|
+
return (component) => {
|
|
3501
|
+
const { debug = false } = config;
|
|
3502
|
+
const listeners = new Map;
|
|
3503
|
+
const emit = (event, data) => {
|
|
3504
|
+
const eventListeners = listeners.get(event);
|
|
3505
|
+
if (eventListeners) {
|
|
3506
|
+
eventListeners.forEach((listener) => {
|
|
3507
|
+
try {
|
|
3508
|
+
listener(data);
|
|
3509
|
+
} catch (error) {
|
|
3510
|
+
console.error(`[Events] Error in listener for ${event}:`, error);
|
|
3511
|
+
}
|
|
3512
|
+
});
|
|
3513
|
+
}
|
|
3514
|
+
};
|
|
3515
|
+
const on = (event, handler) => {
|
|
3516
|
+
if (!listeners.has(event)) {
|
|
3517
|
+
listeners.set(event, new Set);
|
|
3518
|
+
}
|
|
3519
|
+
listeners.get(event).add(handler);
|
|
3520
|
+
return () => {
|
|
3521
|
+
const eventListeners = listeners.get(event);
|
|
3522
|
+
if (eventListeners) {
|
|
3523
|
+
eventListeners.delete(handler);
|
|
3524
|
+
if (eventListeners.size === 0) {
|
|
3525
|
+
listeners.delete(event);
|
|
3526
|
+
}
|
|
3527
|
+
}
|
|
3528
|
+
};
|
|
3529
|
+
};
|
|
3530
|
+
const once = (event, handler) => {
|
|
3531
|
+
const wrappedHandler = (data) => {
|
|
3532
|
+
handler(data);
|
|
3533
|
+
off(event, wrappedHandler);
|
|
3534
|
+
};
|
|
3535
|
+
return on(event, wrappedHandler);
|
|
3536
|
+
};
|
|
3537
|
+
const off = (event, handler) => {
|
|
3538
|
+
const eventListeners = listeners.get(event);
|
|
3539
|
+
if (eventListeners) {
|
|
3540
|
+
eventListeners.delete(handler);
|
|
3541
|
+
if (eventListeners.size === 0) {
|
|
3542
|
+
listeners.delete(event);
|
|
3543
|
+
}
|
|
3544
|
+
}
|
|
3545
|
+
};
|
|
3546
|
+
const clear = (event) => {
|
|
3547
|
+
if (event) {
|
|
3548
|
+
listeners.delete(event);
|
|
3549
|
+
} else {
|
|
3550
|
+
listeners.clear();
|
|
3551
|
+
}
|
|
3552
|
+
};
|
|
3553
|
+
component.emit = emit;
|
|
3554
|
+
component.on = on;
|
|
3555
|
+
component.once = once;
|
|
3556
|
+
component.off = off;
|
|
3557
|
+
component.viewport.events = {
|
|
3558
|
+
emit,
|
|
3559
|
+
on,
|
|
3560
|
+
once,
|
|
3561
|
+
off,
|
|
3562
|
+
clear,
|
|
3563
|
+
getListenerCount: (event) => {
|
|
3564
|
+
if (event) {
|
|
3565
|
+
return listeners.get(event)?.size || 0;
|
|
3566
|
+
}
|
|
3567
|
+
let total = 0;
|
|
3568
|
+
listeners.forEach((set) => total += set.size);
|
|
3569
|
+
return total;
|
|
3570
|
+
}
|
|
3571
|
+
};
|
|
3572
|
+
if ("destroy" in component && typeof component.destroy === "function") {
|
|
3573
|
+
const originalDestroy = component.destroy;
|
|
3574
|
+
component.destroy = () => {
|
|
3575
|
+
clear();
|
|
3576
|
+
originalDestroy?.();
|
|
3577
|
+
};
|
|
3578
|
+
}
|
|
3579
|
+
return component;
|
|
3580
|
+
};
|
|
3581
|
+
};
|
|
3582
|
+
|
|
3583
|
+
// src/core/viewport/viewport.ts
|
|
3584
|
+
var createViewport = (config = {}) => {
|
|
3585
|
+
return (component) => {
|
|
3586
|
+
let isInitialized = false;
|
|
3587
|
+
const state = {
|
|
3588
|
+
scrollPosition: 0,
|
|
3589
|
+
totalItems: component.totalItems || 0,
|
|
3590
|
+
itemSize: config.virtual?.itemSize || 50,
|
|
3591
|
+
containerSize: 0,
|
|
3592
|
+
virtualTotalSize: 0,
|
|
3593
|
+
visibleRange: { start: 0, end: 0 },
|
|
3594
|
+
itemsContainer: null,
|
|
3595
|
+
velocity: 0,
|
|
3596
|
+
scrollDirection: "forward"
|
|
3597
|
+
};
|
|
3598
|
+
const viewportAPI = {
|
|
3599
|
+
initialize: () => {
|
|
3600
|
+
if (isInitialized) {
|
|
3601
|
+
return false;
|
|
3602
|
+
}
|
|
3603
|
+
isInitialized = true;
|
|
3604
|
+
if (component.element) {
|
|
3605
|
+
state.containerSize = config.scrolling?.orientation === "horizontal" ? component.element.offsetWidth : component.element.offsetHeight;
|
|
3606
|
+
}
|
|
3607
|
+
state.virtualTotalSize = state.totalItems * state.itemSize;
|
|
3608
|
+
state.visibleRange = calculateVisibleRange(state.scrollPosition);
|
|
3609
|
+
},
|
|
3610
|
+
destroy: () => {},
|
|
3611
|
+
updateViewport: () => {},
|
|
3612
|
+
isInitialized: () => isInitialized,
|
|
3613
|
+
resetInitialization: () => {
|
|
3614
|
+
isInitialized = false;
|
|
3615
|
+
},
|
|
3616
|
+
_shouldInit: () => !isInitialized,
|
|
3617
|
+
scrollToIndex: (index, alignment) => {},
|
|
3618
|
+
scrollToPosition: (position) => {},
|
|
3619
|
+
getScrollPosition: () => state.scrollPosition,
|
|
3620
|
+
getVisibleRange: () => state.visibleRange,
|
|
3621
|
+
getViewportInfo: () => ({
|
|
3622
|
+
containerSize: state.containerSize,
|
|
3623
|
+
totalVirtualSize: state.virtualTotalSize,
|
|
3624
|
+
visibleRange: state.visibleRange,
|
|
3625
|
+
virtualScrollPosition: state.scrollPosition
|
|
3626
|
+
}),
|
|
3627
|
+
renderItems: () => {},
|
|
3628
|
+
state
|
|
3629
|
+
};
|
|
3630
|
+
const calculateVisibleRange = (scrollPos) => {
|
|
3631
|
+
const itemSize = state.itemSize;
|
|
3632
|
+
const start = Math.floor(scrollPos / itemSize);
|
|
3633
|
+
const visibleCount = Math.ceil(state.containerSize / itemSize);
|
|
3634
|
+
const end = Math.min(start + visibleCount, state.totalItems - 1);
|
|
3635
|
+
return { start: Math.max(0, start), end: Math.max(0, end) };
|
|
3636
|
+
};
|
|
3637
|
+
const baseComponent = {
|
|
3638
|
+
...component,
|
|
3639
|
+
viewport: viewportAPI
|
|
3640
|
+
};
|
|
3641
|
+
const enhancers = [];
|
|
3642
|
+
enhancers.push(withEvents({
|
|
3643
|
+
debug: config.debug
|
|
3644
|
+
}));
|
|
3645
|
+
enhancers.push(withBase({
|
|
3646
|
+
className: config.className,
|
|
3647
|
+
orientation: config.scrolling?.orientation
|
|
3648
|
+
}));
|
|
3649
|
+
enhancers.push(withVirtual({
|
|
3650
|
+
itemSize: config.virtual?.itemSize,
|
|
3651
|
+
overscan: config.virtual?.overscan,
|
|
3652
|
+
orientation: config.scrolling?.orientation,
|
|
3653
|
+
autoDetectItemSize: config.virtual?.autoDetectItemSize,
|
|
3654
|
+
initialScrollIndex: config.initialScrollIndex
|
|
3655
|
+
}));
|
|
3656
|
+
enhancers.push(withScrolling({
|
|
3657
|
+
orientation: config.scrolling?.orientation,
|
|
3658
|
+
sensitivity: config.scrolling?.sensitivity,
|
|
3659
|
+
smoothing: config.scrolling?.animation,
|
|
3660
|
+
stopOnClick: config.scrolling?.stopOnClick
|
|
3661
|
+
}));
|
|
3662
|
+
enhancers.push(withMomentum({
|
|
3663
|
+
enabled: true,
|
|
3664
|
+
stopOnClick: config.scrolling?.stopOnClick,
|
|
3665
|
+
deceleration: config.scrolling?.momentum?.deceleration,
|
|
3666
|
+
minVelocity: config.scrolling?.momentum?.minVelocity,
|
|
3667
|
+
minDuration: config.scrolling?.momentum?.maxDuration,
|
|
3668
|
+
minVelocityThreshold: config.scrolling?.momentum?.velocityThreshold
|
|
3669
|
+
}));
|
|
3670
|
+
if (config.scrollbar?.enabled !== false) {
|
|
3671
|
+
enhancers.push(withScrollbar({
|
|
3672
|
+
enabled: true,
|
|
3673
|
+
autoHide: config.scrollbar?.autoHide
|
|
3674
|
+
}));
|
|
3675
|
+
}
|
|
3676
|
+
if (config.collection?.adapter) {
|
|
3677
|
+
enhancers.push(withCollection({
|
|
3678
|
+
collection: config.collection.adapter,
|
|
3679
|
+
rangeSize: config.pagination?.limit,
|
|
3680
|
+
strategy: config.pagination?.strategy,
|
|
3681
|
+
transform: config.collection?.transform,
|
|
3682
|
+
cancelLoadThreshold: config.performance?.cancelLoadThreshold,
|
|
3683
|
+
maxConcurrentRequests: config.performance?.maxConcurrentRequests,
|
|
3684
|
+
enableRequestQueue: config.performance?.enableRequestQueue !== false,
|
|
3685
|
+
initialScrollIndex: config.initialScrollIndex,
|
|
3686
|
+
selectId: config.selectId,
|
|
3687
|
+
autoLoad: config.autoLoad !== false,
|
|
3688
|
+
autoSelectFirst: config.autoSelectFirst,
|
|
3689
|
+
maxCachedItems: config.cache?.maxItems,
|
|
3690
|
+
evictionBuffer: config.cache?.evictionBuffer
|
|
3691
|
+
}));
|
|
3692
|
+
}
|
|
3693
|
+
if (config.collection?.adapter && config.placeholders?.enabled !== false) {
|
|
3694
|
+
enhancers.push(withPlaceholders({
|
|
3695
|
+
enabled: true,
|
|
3696
|
+
analyzeFirstLoad: config.placeholders?.analyzeFirstLoad ?? true,
|
|
3697
|
+
maskCharacter: config.placeholders?.maskCharacter
|
|
3698
|
+
}));
|
|
3699
|
+
}
|
|
3700
|
+
enhancers.push(withRendering({
|
|
3701
|
+
template: config.template,
|
|
3702
|
+
overscan: config.virtual?.overscan,
|
|
3703
|
+
maintainDomOrder: config.rendering?.maintainDomOrder
|
|
3704
|
+
}));
|
|
3705
|
+
const enhance = import_mtrl4.pipe(...enhancers);
|
|
3706
|
+
const enhanced = enhance(baseComponent);
|
|
3707
|
+
if (enhanced.element) {
|
|
3708
|
+
enhanced.viewport.initialize();
|
|
3709
|
+
}
|
|
3710
|
+
return enhanced;
|
|
3711
|
+
};
|
|
3712
|
+
};
|
|
3713
|
+
|
|
3714
|
+
//# debugId=BD481E0A015C99EC64756E2164756E21
|
|
3715
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2NvcmUvdmlld3BvcnQvdmlld3BvcnQudHMiLCAiLi4vLi4vLi4vc3JjL2NvcmUvdmlld3BvcnQvZmVhdHVyZXMvdXRpbHMudHMiLCAiLi4vLi4vLi4vc3JjL2NvcmUvdmlld3BvcnQvZmVhdHVyZXMvYmFzZS50cyIsICIuLi8uLi8uLi9zcmMvY29yZS92aWV3cG9ydC9jb25zdGFudHMudHMiLCAiLi4vLi4vLi4vc3JjL2NvcmUvdmlld3BvcnQvZmVhdHVyZXMvdmlydHVhbC50cyIsICIuLi8uLi8uLi9zcmMvY29yZS92aWV3cG9ydC9mZWF0dXJlcy9zY3JvbGxpbmcudHMiLCAiLi4vLi4vLi4vc3JjL2NvcmUvdmlld3BvcnQvZmVhdHVyZXMvbW9tZW50dW0udHMiLCAiLi4vLi4vLi4vc3JjL2NvcmUvdmlld3BvcnQvZmVhdHVyZXMvc2Nyb2xsYmFyLnRzIiwgIi4uLy4uLy4uL3NyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL2NvbGxlY3Rpb24udHMiLCAiLi4vLi4vLi4vc3JjL2NvcmUvdmlld3BvcnQvZmVhdHVyZXMvcGxhY2Vob2xkZXJzLnRzIiwgIi4uLy4uLy4uL3NyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL3JlbmRlcmluZy50cyIsICIuLi8uLi8uLi9zcmMvY29yZS9sYXlvdXQvc2NoZW1hLnRzIiwgIi4uLy4uLy4uL3NyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL2V2ZW50cy50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsKICAgICIvKipcbiAqIFZpZXdwb3J0IC0gQ29yZSB2aXJ0dWFsIHNjcm9sbGluZyBlbmdpbmVcbiAqL1xuXG5pbXBvcnQgdHlwZSB7XG4gIFZpZXdwb3J0Q29uZmlnLFxuICBWaWV3cG9ydENvbnRleHQsXG4gIFZpZXdwb3J0Q29tcG9uZW50LFxuICBJdGVtUmFuZ2UsXG4gIFZpZXdwb3J0SW5mbyxcbn0gZnJvbSBcIi4vdHlwZXNcIjtcbmltcG9ydCB7IHBpcGUgfSBmcm9tIFwibXRybFwiO1xuaW1wb3J0IHsgd2l0aEJhc2UgfSBmcm9tIFwiLi9mZWF0dXJlcy9iYXNlXCI7XG5pbXBvcnQgeyB3aXRoVmlydHVhbCB9IGZyb20gXCIuL2ZlYXR1cmVzL3ZpcnR1YWxcIjtcbmltcG9ydCB7IHdpdGhTY3JvbGxpbmcgfSBmcm9tIFwiLi9mZWF0dXJlcy9zY3JvbGxpbmdcIjtcbmltcG9ydCB7IHdpdGhNb21lbnR1bSB9IGZyb20gXCIuL2ZlYXR1cmVzL21vbWVudHVtXCI7XG5pbXBvcnQgeyB3aXRoU2Nyb2xsYmFyIH0gZnJvbSBcIi4vZmVhdHVyZXMvc2Nyb2xsYmFyXCI7XG5pbXBvcnQgeyB3aXRoQ29sbGVjdGlvbiB9IGZyb20gXCIuL2ZlYXR1cmVzL2NvbGxlY3Rpb25cIjtcbmltcG9ydCB7IHdpdGhQbGFjZWhvbGRlcnMgfSBmcm9tIFwiLi9mZWF0dXJlcy9wbGFjZWhvbGRlcnNcIjtcbmltcG9ydCB7IHdpdGhSZW5kZXJpbmcgfSBmcm9tIFwiLi9mZWF0dXJlcy9yZW5kZXJpbmdcIjtcbmltcG9ydCB7IHdpdGhFdmVudHMgfSBmcm9tIFwiLi9mZWF0dXJlcy9ldmVudHNcIjtcblxuLy8gSW50ZXJuYWwgdmlld3BvcnQgc3RhdGVcbmludGVyZmFjZSBWaWV3cG9ydFN0YXRlIHtcbiAgc2Nyb2xsUG9zaXRpb246IG51bWJlcjtcbiAgdG90YWxJdGVtczogbnVtYmVyO1xuICBpdGVtU2l6ZTogbnVtYmVyO1xuICBjb250YWluZXJTaXplOiBudW1iZXI7XG4gIHZpcnR1YWxUb3RhbFNpemU6IG51bWJlcjtcbiAgdmlzaWJsZVJhbmdlOiBJdGVtUmFuZ2U7XG4gIGl0ZW1zQ29udGFpbmVyOiBIVE1MRWxlbWVudCB8IG51bGw7XG4gIHZlbG9jaXR5OiBudW1iZXI7XG4gIHNjcm9sbERpcmVjdGlvbjogXCJmb3J3YXJkXCIgfCBcImJhY2t3YXJkXCI7XG59XG5cbi8qKlxuICogQ3JlYXRlcyBhIHZpZXdwb3J0LWVuaGFuY2VkIGNvbXBvbmVudCB1c2luZyBjb21wb3NpdGlvblxuICpcbiAqIEBwYXJhbSBjb25maWcgLSBWaWV3cG9ydCBjb25maWd1cmF0aW9uXG4gKiBAcmV0dXJucyBGdW5jdGlvbiB0aGF0IGVuaGFuY2VzIGEgY29tcG9uZW50IHdpdGggdmlld3BvcnQgY2FwYWJpbGl0aWVzXG4gKi9cbmV4cG9ydCBjb25zdCBjcmVhdGVWaWV3cG9ydCA9IChjb25maWc6IFZpZXdwb3J0Q29uZmlnID0ge30pID0+IHtcbiAgcmV0dXJuIDxUIGV4dGVuZHMgVmlld3BvcnRDb250ZXh0Pihjb21wb25lbnQ6IFQpOiBUICYgVmlld3BvcnRDb21wb25lbnQgPT4ge1xuICAgIC8vIFRyYWNrIGlmIHZpZXdwb3J0IGhhcyBiZWVuIGluaXRpYWxpemVkIHRvIHByZXZlbnQgbXVsdGlwbGUgaW5pdGlhbGl6YXRpb25zXG4gICAgbGV0IGlzSW5pdGlhbGl6ZWQgPSBmYWxzZTtcblxuICAgIC8vIE5vIG5vcm1hbGl6YXRpb24gbmVlZGVkIC0gd2UnbGwgdXNlIHRoZSBuZXN0ZWQgc3RydWN0dXJlIGRpcmVjdGx5XG5cbiAgICAvLyBDcmVhdGUgc2hhcmVkIHZpZXdwb3J0IHN0YXRlXG4gICAgY29uc3Qgc3RhdGU6IFZpZXdwb3J0U3RhdGUgPSB7XG4gICAgICBzY3JvbGxQb3NpdGlvbjogMCxcbiAgICAgIHRvdGFsSXRlbXM6IGNvbXBvbmVudC50b3RhbEl0ZW1zIHx8IDAsXG4gICAgICBpdGVtU2l6ZTogY29uZmlnLnZpcnR1YWw/Lml0ZW1TaXplIHx8IDUwLFxuICAgICAgY29udGFpbmVyU2l6ZTogMCxcbiAgICAgIHZpcnR1YWxUb3RhbFNpemU6IDAsXG4gICAgICB2aXNpYmxlUmFuZ2U6IHsgc3RhcnQ6IDAsIGVuZDogMCB9LFxuICAgICAgaXRlbXNDb250YWluZXI6IG51bGwsXG4gICAgICB2ZWxvY2l0eTogMCxcbiAgICAgIHNjcm9sbERpcmVjdGlvbjogXCJmb3J3YXJkXCIsXG4gICAgfTtcblxuICAgIC8vIENyZWF0ZSBiYXNlIHZpZXdwb3J0IEFQSVxuICAgIGNvbnN0IHZpZXdwb3J0QVBJID0ge1xuICAgICAgLy8gQ29yZSBBUElcbiAgICAgIGluaXRpYWxpemU6ICgpID0+IHtcbiAgICAgICAgLy8gUHJldmVudCBtdWx0aXBsZSBpbml0aWFsaXphdGlvbnNcbiAgICAgICAgaWYgKGlzSW5pdGlhbGl6ZWQpIHtcbiAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgIH1cbiAgICAgICAgaXNJbml0aWFsaXplZCA9IHRydWU7XG5cbiAgICAgICAgLy8gY29uc29sZS5sb2coXCJbVmlld3BvcnRdIEluaXRpYWxpemluZyB3aXRoIHN0YXRlOlwiLCB7XG4gICAgICAgIC8vICAgZWxlbWVudDogISFjb21wb25lbnQuZWxlbWVudCxcbiAgICAgICAgLy8gICB0b3RhbEl0ZW1zOiBjb21wb25lbnQudG90YWxJdGVtcyxcbiAgICAgICAgLy8gICBpdGVtU2l6ZTogY29uZmlnLnZpcnR1YWw/Lml0ZW1TaXplLFxuICAgICAgICAvLyB9KTtcblxuICAgICAgICAvLyBJbml0aWFsaXplIGNvbnRhaW5lciBzaXplXG4gICAgICAgIGlmIChjb21wb25lbnQuZWxlbWVudCkge1xuICAgICAgICAgIHN0YXRlLmNvbnRhaW5lclNpemUgPVxuICAgICAgICAgICAgY29uZmlnLnNjcm9sbGluZz8ub3JpZW50YXRpb24gPT09IFwiaG9yaXpvbnRhbFwiXG4gICAgICAgICAgICAgID8gY29tcG9uZW50LmVsZW1lbnQub2Zmc2V0V2lkdGhcbiAgICAgICAgICAgICAgOiBjb21wb25lbnQuZWxlbWVudC5vZmZzZXRIZWlnaHQ7XG5cbiAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcIltWaWV3cG9ydF0gQ29udGFpbmVyIHNpemU6XCIsIHN0YXRlLmNvbnRhaW5lclNpemUpO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gQ2FsY3VsYXRlIGluaXRpYWwgdmlydHVhbCBzaXplXG4gICAgICAgIHN0YXRlLnZpcnR1YWxUb3RhbFNpemUgPSBzdGF0ZS50b3RhbEl0ZW1zICogc3RhdGUuaXRlbVNpemU7XG5cbiAgICAgICAgLy8gQ2FsY3VsYXRlIGluaXRpYWwgdmlzaWJsZSByYW5nZVxuICAgICAgICBzdGF0ZS52aXNpYmxlUmFuZ2UgPSBjYWxjdWxhdGVWaXNpYmxlUmFuZ2Uoc3RhdGUuc2Nyb2xsUG9zaXRpb24pO1xuXG4gICAgICAgIC8vIGNvbnNvbGUubG9nKFwiW1ZpZXdwb3J0XSBJbml0aWFsIHN0YXRlOlwiLCB7XG4gICAgICAgIC8vICAgY29udGFpbmVyU2l6ZTogc3RhdGUuY29udGFpbmVyU2l6ZSxcbiAgICAgICAgLy8gICB2aXJ0dWFsVG90YWxTaXplOiBzdGF0ZS52aXJ0dWFsVG90YWxTaXplLFxuICAgICAgICAvLyAgIHZpc2libGVSYW5nZTogc3RhdGUudmlzaWJsZVJhbmdlLFxuICAgICAgICAvLyB9KTtcbiAgICAgIH0sXG5cbiAgICAgIGRlc3Ryb3k6ICgpID0+IHtcbiAgICAgICAgLy8gQ2xlYW51cCB3aWxsIGJlIGhhbmRsZWQgYnkgZmVhdHVyZXNcbiAgICAgIH0sXG5cbiAgICAgIHVwZGF0ZVZpZXdwb3J0OiAoKSA9PiB7XG4gICAgICAgIC8vIFdpbGwgYmUgaW1wbGVtZW50ZWQgYnkgcmVuZGVyaW5nIGZlYXR1cmVcbiAgICAgIH0sXG5cbiAgICAgIC8vIENoZWNrIGlmIGFscmVhZHkgaW5pdGlhbGl6ZWRcbiAgICAgIGlzSW5pdGlhbGl6ZWQ6ICgpID0+IGlzSW5pdGlhbGl6ZWQsXG5cbiAgICAgIC8vIFJlc2V0IGluaXRpYWxpemF0aW9uIHN0YXRlIChmb3IgcmVsb2FkIGZ1bmN0aW9uYWxpdHkpXG4gICAgICByZXNldEluaXRpYWxpemF0aW9uOiAoKSA9PiB7XG4gICAgICAgIGlzSW5pdGlhbGl6ZWQgPSBmYWxzZTtcbiAgICAgIH0sXG5cbiAgICAgIC8vIEFsbG93IGZlYXR1cmVzIHRvIGNoZWNrIGlmIGluaXQgc2hvdWxkIHByb2NlZWRcbiAgICAgIF9zaG91bGRJbml0OiAoKSA9PiAhaXNJbml0aWFsaXplZCxcblxuICAgICAgLy8gU2Nyb2xsaW5nIEFQSSAod2lsbCBiZSBvdmVycmlkZGVuIGJ5IHNjcm9sbGluZyBmZWF0dXJlKVxuICAgICAgc2Nyb2xsVG9JbmRleDogKFxuICAgICAgICBpbmRleDogbnVtYmVyLFxuICAgICAgICBhbGlnbm1lbnQ/OiBcInN0YXJ0XCIgfCBcImNlbnRlclwiIHwgXCJlbmRcIixcbiAgICAgICkgPT4ge1xuICAgICAgICAvLyBQbGFjZWhvbGRlciAtIHdpbGwgYmUgaW1wbGVtZW50ZWQgYnkgc2Nyb2xsaW5nIGZlYXR1cmVcbiAgICAgIH0sXG5cbiAgICAgIHNjcm9sbFRvUG9zaXRpb246IChwb3NpdGlvbjogbnVtYmVyKSA9PiB7XG4gICAgICAgIC8vIFdpbGwgYmUgaW1wbGVtZW50ZWQgYnkgc2Nyb2xsaW5nIGZlYXR1cmVcbiAgICAgIH0sXG5cbiAgICAgIGdldFNjcm9sbFBvc2l0aW9uOiAoKSA9PiBzdGF0ZS5zY3JvbGxQb3NpdGlvbixcblxuICAgICAgLy8gSW5mbyBBUElcbiAgICAgIGdldFZpc2libGVSYW5nZTogKCkgPT4gc3RhdGUudmlzaWJsZVJhbmdlLFxuXG4gICAgICBnZXRWaWV3cG9ydEluZm86ICgpOiBWaWV3cG9ydEluZm8gPT4gKHtcbiAgICAgICAgY29udGFpbmVyU2l6ZTogc3RhdGUuY29udGFpbmVyU2l6ZSxcbiAgICAgICAgdG90YWxWaXJ0dWFsU2l6ZTogc3RhdGUudmlydHVhbFRvdGFsU2l6ZSxcbiAgICAgICAgdmlzaWJsZVJhbmdlOiBzdGF0ZS52aXNpYmxlUmFuZ2UsXG4gICAgICAgIHZpcnR1YWxTY3JvbGxQb3NpdGlvbjogc3RhdGUuc2Nyb2xsUG9zaXRpb24sXG4gICAgICB9KSxcblxuICAgICAgLy8gUmVuZGVyaW5nIEFQSVxuICAgICAgcmVuZGVySXRlbXM6ICgpID0+IHtcbiAgICAgICAgLy8gV2lsbCBiZSBpbXBsZW1lbnRlZCBieSByZW5kZXJpbmcgZmVhdHVyZVxuICAgICAgfSxcblxuICAgICAgLy8gSW50ZXJuYWwgc3RhdGUgKGZvciBmZWF0dXJlcyB0byBhY2Nlc3MpXG4gICAgICBzdGF0ZSxcbiAgICB9O1xuXG4gICAgLy8gSGVscGVyIGZ1bmN0aW9uIHRvIGNhbGN1bGF0ZSB2aXNpYmxlIHJhbmdlXG4gICAgY29uc3QgY2FsY3VsYXRlVmlzaWJsZVJhbmdlID0gKHNjcm9sbFBvczogbnVtYmVyKTogSXRlbVJhbmdlID0+IHtcbiAgICAgIGNvbnN0IGl0ZW1TaXplID0gc3RhdGUuaXRlbVNpemU7XG4gICAgICBjb25zdCBzdGFydCA9IE1hdGguZmxvb3Ioc2Nyb2xsUG9zIC8gaXRlbVNpemUpO1xuICAgICAgY29uc3QgdmlzaWJsZUNvdW50ID0gTWF0aC5jZWlsKHN0YXRlLmNvbnRhaW5lclNpemUgLyBpdGVtU2l6ZSk7XG4gICAgICBjb25zdCBlbmQgPSBNYXRoLm1pbihzdGFydCArIHZpc2libGVDb3VudCwgc3RhdGUudG90YWxJdGVtcyAtIDEpO1xuXG4gICAgICByZXR1cm4geyBzdGFydDogTWF0aC5tYXgoMCwgc3RhcnQpLCBlbmQ6IE1hdGgubWF4KDAsIGVuZCkgfTtcbiAgICB9O1xuXG4gICAgLy8gQWRkIHZpZXdwb3J0IEFQSSB0byBjb21wb25lbnRcbiAgICBjb25zdCBiYXNlQ29tcG9uZW50ID0ge1xuICAgICAgLi4uY29tcG9uZW50LFxuICAgICAgdmlld3BvcnQ6IHZpZXdwb3J0QVBJLFxuICAgIH07XG5cbiAgICAvLyBCdWlsZCB0aGUgZW5oYW5jZW1lbnQgcGlwZWxpbmVcbiAgICBjb25zdCBlbmhhbmNlcnM6IEFycmF5PChjOiBhbnkpID0+IGFueT4gPSBbXTtcblxuICAgIC8vIEV2ZW50cyBzeXN0ZW0gKGFsd2F5cyBmaXJzdCAtIHByb3ZpZGVzIGNvbW11bmljYXRpb24gZm9yIG90aGVyIGZlYXR1cmVzKVxuICAgIGVuaGFuY2Vycy5wdXNoKFxuICAgICAgd2l0aEV2ZW50cyh7XG4gICAgICAgIGRlYnVnOiBjb25maWcuZGVidWcsXG4gICAgICB9KSxcbiAgICApO1xuXG4gICAgLy8gQmFzZSBzZXR1cCAoY3JlYXRlcyBET00gc3RydWN0dXJlKVxuICAgIGVuaGFuY2Vycy5wdXNoKFxuICAgICAgd2l0aEJhc2Uoe1xuICAgICAgICBjbGFzc05hbWU6IGNvbmZpZy5jbGFzc05hbWUsXG4gICAgICAgIG9yaWVudGF0aW9uOiBjb25maWcuc2Nyb2xsaW5nPy5vcmllbnRhdGlvbixcbiAgICAgIH0pLFxuICAgICk7XG5cbiAgICAvLyBWaXJ0dWFsIHNjcm9sbGluZyAocmVxdWlyZWQgZm9yIG1vc3QgZmVhdHVyZXMpXG4gICAgZW5oYW5jZXJzLnB1c2goXG4gICAgICB3aXRoVmlydHVhbCh7XG4gICAgICAgIGl0ZW1TaXplOiBjb25maWcudmlydHVhbD8uaXRlbVNpemUsXG4gICAgICAgIG92ZXJzY2FuOiBjb25maWcudmlydHVhbD8ub3ZlcnNjYW4sXG4gICAgICAgIG9yaWVudGF0aW9uOiBjb25maWcuc2Nyb2xsaW5nPy5vcmllbnRhdGlvbixcbiAgICAgICAgYXV0b0RldGVjdEl0ZW1TaXplOiBjb25maWcudmlydHVhbD8uYXV0b0RldGVjdEl0ZW1TaXplLFxuICAgICAgICBpbml0aWFsU2Nyb2xsSW5kZXg6IChjb25maWcgYXMgYW55KS5pbml0aWFsU2Nyb2xsSW5kZXgsXG4gICAgICB9KSxcbiAgICApO1xuXG4gICAgLy8gU2Nyb2xsaW5nIGJlaGF2aW9yXG4gICAgZW5oYW5jZXJzLnB1c2goXG4gICAgICB3aXRoU2Nyb2xsaW5nKHtcbiAgICAgICAgb3JpZW50YXRpb246IGNvbmZpZy5zY3JvbGxpbmc/Lm9yaWVudGF0aW9uLFxuICAgICAgICBzZW5zaXRpdml0eTogY29uZmlnLnNjcm9sbGluZz8uc2Vuc2l0aXZpdHksXG4gICAgICAgIHNtb290aGluZzogY29uZmlnLnNjcm9sbGluZz8uYW5pbWF0aW9uLFxuICAgICAgICBzdG9wT25DbGljazogY29uZmlnLnNjcm9sbGluZz8uc3RvcE9uQ2xpY2ssXG4gICAgICB9KSxcbiAgICApO1xuXG4gICAgLy8gTW9tZW50dW0gc2Nyb2xsaW5nICh0b3VjaC9kcmFnIHN1cHBvcnQgZm9yIG1vYmlsZSlcbiAgICBlbmhhbmNlcnMucHVzaChcbiAgICAgIHdpdGhNb21lbnR1bSh7XG4gICAgICAgIGVuYWJsZWQ6IHRydWUsXG4gICAgICAgIHN0b3BPbkNsaWNrOiBjb25maWcuc2Nyb2xsaW5nPy5zdG9wT25DbGljayxcbiAgICAgICAgZGVjZWxlcmF0aW9uOiBjb25maWcuc2Nyb2xsaW5nPy5tb21lbnR1bT8uZGVjZWxlcmF0aW9uLFxuICAgICAgICBtaW5WZWxvY2l0eTogY29uZmlnLnNjcm9sbGluZz8ubW9tZW50dW0/Lm1pblZlbG9jaXR5LFxuICAgICAgICBtaW5EdXJhdGlvbjogY29uZmlnLnNjcm9sbGluZz8ubW9tZW50dW0/Lm1heER1cmF0aW9uLFxuICAgICAgICBtaW5WZWxvY2l0eVRocmVzaG9sZDogY29uZmlnLnNjcm9sbGluZz8ubW9tZW50dW0/LnZlbG9jaXR5VGhyZXNob2xkLFxuICAgICAgfSksXG4gICAgKTtcblxuICAgIC8vIFNjcm9sbGJhciAob3B0aW9uYWwpXG4gICAgaWYgKGNvbmZpZy5zY3JvbGxiYXI/LmVuYWJsZWQgIT09IGZhbHNlKSB7XG4gICAgICBlbmhhbmNlcnMucHVzaChcbiAgICAgICAgd2l0aFNjcm9sbGJhcih7XG4gICAgICAgICAgZW5hYmxlZDogdHJ1ZSxcbiAgICAgICAgICBhdXRvSGlkZTogY29uZmlnLnNjcm9sbGJhcj8uYXV0b0hpZGUsXG4gICAgICAgIH0pLFxuICAgICAgKTtcbiAgICB9XG5cbiAgICAvLyBBZGQgY29sbGVjdGlvbiBpZiBjb25maWd1cmVkXG4gICAgaWYgKGNvbmZpZy5jb2xsZWN0aW9uPy5hZGFwdGVyKSB7XG4gICAgICBlbmhhbmNlcnMucHVzaChcbiAgICAgICAgd2l0aENvbGxlY3Rpb24oe1xuICAgICAgICAgIGNvbGxlY3Rpb246IGNvbmZpZy5jb2xsZWN0aW9uLmFkYXB0ZXIsXG4gICAgICAgICAgcmFuZ2VTaXplOiBjb25maWcucGFnaW5hdGlvbj8ubGltaXQsXG4gICAgICAgICAgc3RyYXRlZ3k6IGNvbmZpZy5wYWdpbmF0aW9uPy5zdHJhdGVneSBhc1xuICAgICAgICAgICAgfCBcIm9mZnNldFwiXG4gICAgICAgICAgICB8IFwicGFnZVwiXG4gICAgICAgICAgICB8IFwiY3Vyc29yXCJcbiAgICAgICAgICAgIHwgdW5kZWZpbmVkLFxuICAgICAgICAgIHRyYW5zZm9ybTogY29uZmlnLmNvbGxlY3Rpb24/LnRyYW5zZm9ybSxcbiAgICAgICAgICBjYW5jZWxMb2FkVGhyZXNob2xkOiBjb25maWcucGVyZm9ybWFuY2U/LmNhbmNlbExvYWRUaHJlc2hvbGQsXG4gICAgICAgICAgbWF4Q29uY3VycmVudFJlcXVlc3RzOiBjb25maWcucGVyZm9ybWFuY2U/Lm1heENvbmN1cnJlbnRSZXF1ZXN0cyxcbiAgICAgICAgICBlbmFibGVSZXF1ZXN0UXVldWU6IGNvbmZpZy5wZXJmb3JtYW5jZT8uZW5hYmxlUmVxdWVzdFF1ZXVlICE9PSBmYWxzZSxcbiAgICAgICAgICBpbml0aWFsU2Nyb2xsSW5kZXg6IChjb25maWcgYXMgYW55KS5pbml0aWFsU2Nyb2xsSW5kZXgsXG4gICAgICAgICAgc2VsZWN0SWQ6IChjb25maWcgYXMgYW55KS5zZWxlY3RJZCxcbiAgICAgICAgICBhdXRvTG9hZDogKGNvbmZpZyBhcyBhbnkpLmF1dG9Mb2FkICE9PSBmYWxzZSxcbiAgICAgICAgICBhdXRvU2VsZWN0Rmlyc3Q6IChjb25maWcgYXMgYW55KS5hdXRvU2VsZWN0Rmlyc3QsXG4gICAgICAgICAgbWF4Q2FjaGVkSXRlbXM6IGNvbmZpZy5jYWNoZT8ubWF4SXRlbXMsXG4gICAgICAgICAgZXZpY3Rpb25CdWZmZXI6IGNvbmZpZy5jYWNoZT8uZXZpY3Rpb25CdWZmZXIsXG4gICAgICAgIH0pLFxuICAgICAgKTtcbiAgICB9XG5cbiAgICAvLyBQbGFjZWhvbGRlcnMgKG9wdGlvbmFsLCByZXF1aXJlcyBjb2xsZWN0aW9uKVxuICAgIGlmIChjb25maWcuY29sbGVjdGlvbj8uYWRhcHRlciAmJiBjb25maWcucGxhY2Vob2xkZXJzPy5lbmFibGVkICE9PSBmYWxzZSkge1xuICAgICAgZW5oYW5jZXJzLnB1c2goXG4gICAgICAgIHdpdGhQbGFjZWhvbGRlcnMoe1xuICAgICAgICAgIGVuYWJsZWQ6IHRydWUsXG4gICAgICAgICAgYW5hbHl6ZUZpcnN0TG9hZDogY29uZmlnLnBsYWNlaG9sZGVycz8uYW5hbHl6ZUZpcnN0TG9hZCA/PyB0cnVlLFxuICAgICAgICAgIG1hc2tDaGFyYWN0ZXI6IGNvbmZpZy5wbGFjZWhvbGRlcnM/Lm1hc2tDaGFyYWN0ZXIsXG4gICAgICAgIH0pLFxuICAgICAgKTtcbiAgICB9XG5cbiAgICAvLyBSZW5kZXJpbmcgKGFsd2F5cyBsYXN0IHRvIGhhdmUgYWNjZXNzIHRvIGFsbCBmZWF0dXJlcylcbiAgICBlbmhhbmNlcnMucHVzaChcbiAgICAgIHdpdGhSZW5kZXJpbmcoe1xuICAgICAgICB0ZW1wbGF0ZTogY29uZmlnLnRlbXBsYXRlLFxuICAgICAgICBvdmVyc2NhbjogY29uZmlnLnZpcnR1YWw/Lm92ZXJzY2FuLFxuICAgICAgICBtYWludGFpbkRvbU9yZGVyOiBjb25maWcucmVuZGVyaW5nPy5tYWludGFpbkRvbU9yZGVyLFxuICAgICAgfSksXG4gICAgKTtcblxuICAgIC8vIENvbXBvc2UgYWxsIGVuaGFuY2Vyc1xuICAgIGNvbnN0IGVuaGFuY2UgPSBwaXBlKC4uLmVuaGFuY2Vycyk7XG5cbiAgICAvLyBBcHBseSBlbmhhbmNlbWVudHNcbiAgICBjb25zdCBlbmhhbmNlZCA9IGVuaGFuY2UoYmFzZUNvbXBvbmVudCk7XG5cbiAgICAvLyBBdXRvLWluaXRpYWxpemUgaWYgZWxlbWVudCBpcyBwcm92aWRlZFxuICAgIGlmIChlbmhhbmNlZC5lbGVtZW50KSB7XG4gICAgICBlbmhhbmNlZC52aWV3cG9ydC5pbml0aWFsaXplKCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGVuaGFuY2VkIGFzIFQgJiBWaWV3cG9ydENvbXBvbmVudDtcbiAgfTtcbn07XG4iLAogICAgIi8vIHNyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL3V0aWxzLnRzXG5cbi8qKlxuICogU2hhcmVkIHV0aWxpdGllcyBmb3Igdmlld3BvcnQgZmVhdHVyZXNcbiAqIEVsaW1pbmF0ZXMgY29kZSBkdXBsaWNhdGlvbiBhY3Jvc3MgZmVhdHVyZXNcbiAqL1xuXG5pbXBvcnQgdHlwZSB7IFZpZXdwb3J0Q29udGV4dCwgVmlld3BvcnRDb21wb25lbnQgfSBmcm9tIFwiLi4vdHlwZXNcIjtcblxuLyoqXG4gKiBXcmFwcyB2aWV3cG9ydCBpbml0aWFsaXphdGlvbiB3aXRoIGZlYXR1cmUtc3BlY2lmaWMgbG9naWNcbiAqIEVsaW1pbmF0ZXMgdGhlIHJlcGVhdGVkIGluaXRpYWxpemF0aW9uIGhvb2sgcGF0dGVyblxuICovXG5leHBvcnQgZnVuY3Rpb24gd3JhcEluaXRpYWxpemU8VCBleHRlbmRzIFZpZXdwb3J0Q29udGV4dCAmIFZpZXdwb3J0Q29tcG9uZW50PihcbiAgY29tcG9uZW50OiBULFxuICBmZWF0dXJlSW5pdDogKCkgPT4gdm9pZCxcbik6IHZvaWQge1xuICBjb25zdCBvcmlnaW5hbEluaXRpYWxpemUgPSBjb21wb25lbnQudmlld3BvcnQuaW5pdGlhbGl6ZTtcbiAgY29tcG9uZW50LnZpZXdwb3J0LmluaXRpYWxpemUgPSAoKSA9PiB7XG4gICAgLy8gQ2hlY2sgaWYgYWxyZWFkeSBpbml0aWFsaXplZCAocmV0dXJucyBmYWxzZSBpZiBhbHJlYWR5IGRvbmUpXG4gICAgY29uc3QgcmVzdWx0ID0gb3JpZ2luYWxJbml0aWFsaXplKCk7XG4gICAgLy8gT25seSBydW4gZmVhdHVyZSBpbml0IGlmIGJhc2UgaW5pdCBhY3R1YWxseSByYW4gKGRpZG4ndCByZXR1cm4gZmFsc2UpXG4gICAgaWYgKHJlc3VsdCAhPT0gZmFsc2UpIHtcbiAgICAgIGZlYXR1cmVJbml0KCk7XG4gICAgfVxuICAgIC8vIFByb3BhZ2F0ZSB0aGUgcmVzdWx0IHRocm91Z2ggdGhlIGNoYWluIHNvIG91dGVyIHdyYXBwZXJzIGtub3cgdG8gc2tpcFxuICAgIHJldHVybiByZXN1bHQ7XG4gIH07XG59XG5cbi8qKlxuICogV3JhcHMgY29tcG9uZW50IGRlc3Ryb3kgd2l0aCBjbGVhbnVwIGxvZ2ljXG4gKiBFbGltaW5hdGVzIHRoZSByZXBlYXRlZCBkZXN0cm95IGhvb2sgcGF0dGVyblxuICovXG5leHBvcnQgZnVuY3Rpb24gd3JhcERlc3Ryb3k8VCBleHRlbmRzIFJlY29yZDxzdHJpbmcsIGFueT4+KFxuICBjb21wb25lbnQ6IFQsXG4gIGNsZWFudXA6ICgpID0+IHZvaWQsXG4pOiB2b2lkIHtcbiAgaWYgKFwiZGVzdHJveVwiIGluIGNvbXBvbmVudCAmJiB0eXBlb2YgY29tcG9uZW50LmRlc3Ryb3kgPT09IFwiZnVuY3Rpb25cIikge1xuICAgIGNvbnN0IG9yaWdpbmFsRGVzdHJveSA9IGNvbXBvbmVudC5kZXN0cm95O1xuICAgIChjb21wb25lbnQgYXMgYW55KS5kZXN0cm95ID0gKCkgPT4ge1xuICAgICAgY2xlYW51cCgpO1xuICAgICAgb3JpZ2luYWxEZXN0cm95Py4oKTtcbiAgICB9O1xuICB9XG59XG5cbi8qKlxuICogR2V0cyB2aWV3cG9ydCBzdGF0ZSB3aXRoIHByb3BlciB0eXBpbmdcbiAqIEVsaW1pbmF0ZXMgcmVwZWF0ZWQgKGNvbXBvbmVudC52aWV3cG9ydCBhcyBhbnkpLnN0YXRlXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRWaWV3cG9ydFN0YXRlKGNvbXBvbmVudDogVmlld3BvcnRDb21wb25lbnQpOiBhbnkge1xuICByZXR1cm4gKGNvbXBvbmVudC52aWV3cG9ydCBhcyBhbnkpLnN0YXRlO1xufVxuXG4vKipcbiAqIENoZWNrcyBpZiBhbiBpdGVtIGlzIGEgcGxhY2Vob2xkZXJcbiAqIEVsaW1pbmF0ZXMgZHVwbGljYXRlZCBwbGFjZWhvbGRlciBkZXRlY3Rpb24gbG9naWNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGlzUGxhY2Vob2xkZXIoaXRlbTogYW55KTogYm9vbGVhbiB7XG4gIHJldHVybiAoXG4gICAgaXRlbSAmJlxuICAgIHR5cGVvZiBpdGVtID09PSBcIm9iamVjdFwiICYmXG4gICAgKGl0ZW0uX3BsYWNlaG9sZGVyID09PSB0cnVlIHx8IGl0ZW1bXCJfcGxhY2Vob2xkZXJcIl0gPT09IHRydWUpIC8vIFN1cHBvcnQgYm90aCBhY2Nlc3MgcGF0dGVybnNcbiAgKTtcbn1cblxuLyoqXG4gKiBDcmVhdGVzIGEgcmFuZ2Uga2V5IGZvciBkZWR1cGxpY2F0aW9uXG4gKiBVc2VkIGJ5IG11bHRpcGxlIGZlYXR1cmVzIGZvciB0cmFja2luZyByYW5nZXNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldFJhbmdlS2V5KHJhbmdlOiB7IHN0YXJ0OiBudW1iZXI7IGVuZDogbnVtYmVyIH0pOiBzdHJpbmcge1xuICByZXR1cm4gYCR7cmFuZ2Uuc3RhcnR9LSR7cmFuZ2UuZW5kfWA7XG59XG5cbi8qKlxuICogQ2xhbXBzIGEgdmFsdWUgYmV0d2VlbiBtaW4gYW5kIG1heFxuICogVXNlZCBieSBtdWx0aXBsZSBmZWF0dXJlcyBmb3IgYm91bmRhcnkgY2hlY2tpbmdcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNsYW1wKHZhbHVlOiBudW1iZXIsIG1pbjogbnVtYmVyLCBtYXg6IG51bWJlcik6IG51bWJlciB7XG4gIHJldHVybiBNYXRoLm1heChtaW4sIE1hdGgubWluKG1heCwgdmFsdWUpKTtcbn1cblxuLyoqXG4gKiBTdG9yZXMgYSBmdW5jdGlvbiByZWZlcmVuY2Ugb24gdGhlIGNvbXBvbmVudCBmb3IgbGF0ZXIgYWNjZXNzXG4gKiBFbGltaW5hdGVzIHBhdHRlcm4gbGlrZSAoY29tcG9uZW50IGFzIGFueSkuX2ZlYXR1cmVGdW5jdGlvbiA9IGZuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBzdG9yZUZlYXR1cmVGdW5jdGlvbjxUIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4oXG4gIGNvbXBvbmVudDogVCxcbiAgbmFtZTogc3RyaW5nLFxuICBmbjogRnVuY3Rpb24sXG4pOiB2b2lkIHtcbiAgKGNvbXBvbmVudCBhcyBhbnkpW25hbWVdID0gZm47XG59XG4iLAogICAgIi8vIHNyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL2Jhc2UudHNcblxuLyoqXG4gKiBCYXNlIFZpZXdwb3J0IEZlYXR1cmUgLSBTZXRzIHVwIGNvcmUgdmlld3BvcnQgc3RydWN0dXJlXG4gKiBQcm92aWRlcyB0aGUgZm91bmRhdGlvbiBmb3Igb3RoZXIgdmlld3BvcnQgZmVhdHVyZXNcbiAqL1xuXG5pbXBvcnQgdHlwZSB7IFZpZXdwb3J0Q29udGV4dCwgVmlld3BvcnRDb21wb25lbnQgfSBmcm9tIFwiLi4vdHlwZXNcIjtcbmltcG9ydCB7IHdyYXBJbml0aWFsaXplLCBnZXRWaWV3cG9ydFN0YXRlIH0gZnJvbSBcIi4vdXRpbHNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBCYXNlQ29uZmlnIHtcbiAgY2xhc3NOYW1lPzogc3RyaW5nO1xuICBvcmllbnRhdGlvbj86IFwidmVydGljYWxcIiB8IFwiaG9yaXpvbnRhbFwiO1xufVxuXG4vKipcbiAqIEJhc2UgZmVhdHVyZSBmb3Igdmlld3BvcnRcbiAqIFNldHMgdXAgdGhlIERPTSBzdHJ1Y3R1cmUgYW5kIGNvcmUgc3R5bGVzXG4gKi9cbmV4cG9ydCBjb25zdCB3aXRoQmFzZSA9IChjb25maWc6IEJhc2VDb25maWcgPSB7fSkgPT4ge1xuICByZXR1cm4gPFQgZXh0ZW5kcyBWaWV3cG9ydENvbnRleHQgJiBWaWV3cG9ydENvbXBvbmVudD4oY29tcG9uZW50OiBUKTogVCA9PiB7XG4gICAgY29uc3QgeyBjbGFzc05hbWUgPSBcIm10cmwtdmlld3BvcnRcIiwgb3JpZW50YXRpb24gPSBcInZlcnRpY2FsXCIgfSA9IGNvbmZpZztcblxuICAgIC8vIFVzZSB0aGUgc2hhcmVkIHdyYXBwZXIgdXRpbGl0eVxuICAgIHdyYXBJbml0aWFsaXplKGNvbXBvbmVudCwgKCkgPT4ge1xuICAgICAgY29uc3QgZWxlbWVudCA9IGNvbXBvbmVudC5lbGVtZW50O1xuICAgICAgaWYgKCFlbGVtZW50KSByZXR1cm47XG5cbiAgICAgIC8vIENyZWF0ZSB2aWV3cG9ydCBzdHJ1Y3R1cmVcbiAgICAgIGxldCB2aWV3cG9ydEVsZW1lbnQgPSBlbGVtZW50LnF1ZXJ5U2VsZWN0b3IoXG4gICAgICAgIFwiLm10cmwtdmlld3BvcnRcIlxuICAgICAgKSBhcyBIVE1MRWxlbWVudDtcblxuICAgICAgaWYgKCF2aWV3cG9ydEVsZW1lbnQpIHtcbiAgICAgICAgdmlld3BvcnRFbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImRpdlwiKTtcbiAgICAgICAgdmlld3BvcnRFbGVtZW50LmNsYXNzTmFtZSA9IGNsYXNzTmFtZTtcbiAgICAgICAgdmlld3BvcnRFbGVtZW50LnN0eWxlLmNzc1RleHQgPSBgXG4gICAgICAgICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgICAgICAgIHdpZHRoOiAxMDAlO1xuICAgICAgICAgIGhlaWdodDogMTAwJTtcbiAgICAgICAgICBvdmVyZmxvdzogaGlkZGVuO1xuICAgICAgICBgO1xuXG4gICAgICAgIC8vIFNldCBvcmllbnRhdGlvblxuICAgICAgICB2aWV3cG9ydEVsZW1lbnQuc2V0QXR0cmlidXRlKFwiZGF0YS1vcmllbnRhdGlvblwiLCBvcmllbnRhdGlvbik7XG5cbiAgICAgICAgLy8gQ3JlYXRlIGl0ZW1zIGNvbnRhaW5lclxuICAgICAgICBjb25zdCBpdGVtc0NvbnRhaW5lciA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XG4gICAgICAgIGl0ZW1zQ29udGFpbmVyLmNsYXNzTmFtZSA9IFwibXRybC12aWV3cG9ydC1pdGVtc1wiO1xuICAgICAgICBpdGVtc0NvbnRhaW5lci5zdHlsZS5jc3NUZXh0ID0gYFxuICAgICAgICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICAgICAgICB3aWR0aDogMTAwJTtcbiAgICAgICAgYDtcblxuICAgICAgICB2aWV3cG9ydEVsZW1lbnQuYXBwZW5kQ2hpbGQoaXRlbXNDb250YWluZXIpO1xuICAgICAgICBlbGVtZW50LmFwcGVuZENoaWxkKHZpZXdwb3J0RWxlbWVudCk7XG5cbiAgICAgICAgLy8gVXBkYXRlIHZpZXdwb3J0IHN0YXRlIHdpdGggY29udGFpbmVycyB1c2luZyB1dGlsaXR5XG4gICAgICAgIGNvbnN0IHN0YXRlID0gZ2V0Vmlld3BvcnRTdGF0ZShjb21wb25lbnQpO1xuICAgICAgICBpZiAoc3RhdGUpIHtcbiAgICAgICAgICBzdGF0ZS52aWV3cG9ydEVsZW1lbnQgPSB2aWV3cG9ydEVsZW1lbnQ7XG4gICAgICAgICAgc3RhdGUuaXRlbXNDb250YWluZXIgPSBpdGVtc0NvbnRhaW5lcjtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFN0b3JlIHJlZmVyZW5jZXMgb24gY29tcG9uZW50XG4gICAgICAgIChjb21wb25lbnQgYXMgYW55KS52aWV3cG9ydEVsZW1lbnQgPSB2aWV3cG9ydEVsZW1lbnQ7XG4gICAgICAgIChjb21wb25lbnQgYXMgYW55KS5pdGVtc0NvbnRhaW5lciA9IGl0ZW1zQ29udGFpbmVyO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgcmV0dXJuIGNvbXBvbmVudDtcbiAgfTtcbn07XG4iLAogICAgIi8vIHNyYy9jb3JlL3ZpZXdwb3J0L2NvbnN0YW50cy50c1xuXG4vKipcbiAqIFZpZXdwb3J0IENvbnN0YW50c1xuICogQ2VudHJhbGl6ZWQgY29uc3RhbnRzIGZvciBhbGwgdmlld3BvcnQgZnVuY3Rpb25hbGl0eVxuICogQ29uc29saWRhdGVkIGZyb20gdmlld3BvcnQsIHZpZXdwb3J0L2ZlYXR1cmVzLCBhbmQgbGlzdC1tYW5hZ2VyIGNvbnN0YW50c1xuICovXG5cbmV4cG9ydCBjb25zdCBWSUVXUE9SVF9DT05TVEFOVFMgPSB7XG4gIC8vIFZpcnR1YWwgc2Nyb2xsaW5nIGRlZmF1bHRzXG4gIFZJUlRVQUxfU0NST0xMOiB7XG4gICAgREVGQVVMVF9JVEVNX1NJWkU6IDQ4LFxuICAgIE9WRVJTQ0FOX0JVRkZFUjogMixcbiAgICBTQ1JPTExfU0VOU0lUSVZJVFk6IDAuNyxcbiAgICBNQVhfVklSVFVBTF9TSVpFOiAxMDAgKiAxMDAwICogMTAwMCwgLy8gMTAwTSBwaXhlbHMgLSBtb2Rlcm4gYnJvd3NlcnMgY2FuIGhhbmRsZSB0aGlzXG4gICAgQVVUT19ERVRFQ1RfSVRFTV9TSVpFOiB0cnVlLCAvLyBFbmFibGUgYXV0b21hdGljIGl0ZW0gc2l6ZSBkZXRlY3Rpb25cbiAgfSxcblxuICAvLyBTY3JvbGxpbmcgc2V0dGluZ3NcbiAgU0NST0xMSU5HOiB7XG4gICAgT1ZFUlNDQU46IDEsIC8vIEZyb20gZmVhdHVyZXMvY29uc3RhbnRzXG4gIH0sXG5cbiAgLy8gUmVuZGVyaW5nIHNldHRpbmdzXG4gIFJFTkRFUklORzoge1xuICAgIC8vIEVsZW1lbnQgcmVjeWNsaW5nXG4gICAgREVGQVVMVF9NQVhfUE9PTF9TSVpFOiAxMDAsXG4gICAgQ0xBU1NFUzoge1xuICAgICAgSVRFTTogXCJ2aWV3cG9ydC1pdGVtXCIsXG4gICAgfSxcbiAgfSxcblxuICAvLyBMb2FkaW5nIHNldHRpbmdzXG4gIExPQURJTkc6IHtcbiAgICBDQU5DRUxfVEhSRVNIT0xEOiAyNSwgLy8gcHgvbXMgLSB2ZWxvY2l0eSBhYm92ZSB3aGljaCBsb2FkcyBjYW5jZWxcbiAgICBNQVhfQ09OQ1VSUkVOVF9SRVFVRVNUUzogMSwgLy8gUGFyYWxsZWwgcmVxdWVzdHMgYWxsb3dlZFxuICAgIERFRkFVTFRfUkFOR0VfU0laRTogMjAsIC8vIEl0ZW1zIHBlciByZXF1ZXN0XG4gICAgREVCT1VOQ0VfTE9BRElORzogMTUwLCAvLyBEZWJvdW5jZSBkZWxheSAobXMpXG4gICAgTUlOX1JBTkdFX1NJWkU6IDEwLCAvLyBNaW5pbXVtIGl0ZW1zIHBlciBsb2FkXG4gICAgTUFYX1JBTkdFX1NJWkU6IDEwMCwgLy8gTWF4aW11bSBpdGVtcyBwZXIgbG9hZFxuICAgIFJFUVVFU1RfVElNRU9VVDogNTAwMCwgLy8gUmVxdWVzdCB0aW1lb3V0IChtcylcbiAgICBSRVRSWV9BVFRFTVBUUzogMiwgLy8gRmFpbGVkIHJlcXVlc3QgcmV0cmllc1xuICAgIFJFVFJZX0RFTEFZOiAxMDAwLCAvLyBEZWxheSBiZXR3ZWVuIHJldHJpZXMgKG1zKVxuICB9LFxuXG4gIC8vIFJlcXVlc3QgcXVldWUgY29uZmlndXJhdGlvbiAoZnJvbSBmZWF0dXJlcy9jb25zdGFudHMpXG4gIFJFUVVFU1RfUVVFVUU6IHtcbiAgICBFTkFCTEVEOiB0cnVlLCAvLyBFbmFibGUgcmVxdWVzdCBxdWV1aW5nXG4gICAgTUFYX1FVRVVFX1NJWkU6IDEsIC8vIE1heCBxdWV1ZWQgcmVxdWVzdHNcbiAgICBNQVhfQUNUSVZFX1JFUVVFU1RTOiAyLCAvLyBNYXggY29uY3VycmVudCBhY3RpdmUgcmVxdWVzdHNcbiAgfSxcblxuICAvLyBQbGFjZWhvbGRlciBzZXR0aW5nc1xuICBQTEFDRUhPTERFUjoge1xuICAgIE1BU0tfQ0hBUkFDVEVSOiBcIlhcIiwgLy8gVXBkYXRlZCBmcm9tIGxpc3QtbWFuYWdlclxuICAgIENMQVNTOiBcInZpZXdwb3J0LWl0ZW0tLXBsYWNlaG9sZGVyXCIsXG4gICAgTUFYX1NBTVBMRV9TSVpFOiAyMCxcbiAgICBQTEFDRUhPTERFUl9GTEFHOiBcIl9wbGFjZWhvbGRlclwiLFxuICAgIFJBTkRPTV9MRU5HVEhfVkFSSUFOQ0U6IHRydWUsXG4gIH0sXG5cbiAgLy8gU3BlZWQgdHJhY2tpbmcgKGZyb20gbGlzdC1tYW5hZ2VyKVxuICBTUEVFRF9UUkFDS0lORzoge1xuICAgIC8vIFZlbG9jaXR5IGNhbGN1bGF0aW9uXG4gICAgREVDRUxFUkFUSU9OX0ZBQ1RPUjogMC44NSwgLy8gdmVsb2NpdHkgZGVjYXkgcGVyIGZyYW1lXG4gIH0sXG5cbiAgLy8gTW9tZW50dW0gc2V0dGluZ3NcbiAgTU9NRU5UVU06IHtcbiAgICBFTkFCTEVEOiBmYWxzZSwgLy8gRW5hYmxlIG1vbWVudHVtIGJ5IGRlZmF1bHRcbiAgICBERUNFTEVSQVRJT05fRkFDVE9SOiAwLjk1LCAvLyBIb3cgcXVpY2tseSB2ZWxvY2l0eSBkZWNyZWFzZXMgcGVyIGZyYW1lIChoaWdoZXIgPSBsb25nZXIgY29hc3QpXG4gICAgTUlOX1ZFTE9DSVRZOiAwLjEsIC8vIE1pbmltdW0gdmVsb2NpdHkgYmVmb3JlIHN0b3BwaW5nIChweC9tcylcbiAgICBNSU5fRFVSQVRJT046IDc1MCwgLy8gTWF4aW11bSBnZXN0dXJlIGR1cmF0aW9uIHRvIHRyaWdnZXIgbW9tZW50dW0gKG1zKVxuICAgIE1JTl9WRUxPQ0lUWV9USFJFU0hPTEQ6IDAuNCwgLy8gTWluaW11bSB2ZWxvY2l0eSB0byB0cmlnZ2VyIG1vbWVudHVtIChweC9tcylcbiAgICBGUkFNRV9USU1FOiAxNiwgLy8gQXNzdW1lZCBmcmFtZSB0aW1lIGZvciBjYWxjdWxhdGlvbnMgKG1zKVxuICB9LFxuXG4gIC8vIEluaXRpYWwgbG9hZCBjb25maWd1cmF0aW9uIChmcm9tIGxpc3QtbWFuYWdlcilcbiAgSU5JVElBTF9MT0FEOiB7XG4gICAgU1RSQVRFR1k6IFwicGxhY2Vob2xkZXJzXCIsIC8vIFwicGxhY2Vob2xkZXJzXCIgfCBcImRpcmVjdFwiIHwgXCJwcm9ncmVzc2l2ZVwiXG4gICAgVklFV1BPUlRfTVVMVElQTElFUjogMS41LCAvLyBsb2FkIDEuNXggdmlld3BvcnQgY2FwYWNpdHlcbiAgICBNSU5fSVRFTVM6IDEwLFxuICAgIE1BWF9JVEVNUzogMTAwLFxuICAgIFBMQUNFSE9MREVSX0NPVU5UOiAyMCwgLy8gZGVmYXVsdCBwbGFjZWhvbGRlciBjb3VudFxuICAgIFNIT1dfTE9BRElOR19TVEFURTogdHJ1ZSxcbiAgICBMT0FESU5HX0RFTEFZOiAxMDAsIC8vIG1zIC0gZGVsYXkgYmVmb3JlIHNob3dpbmcgbG9hZGluZyBzdGF0ZVxuICB9LFxuXG4gIC8vIFNlbGVjdGlvbiBzZXR0aW5nc1xuICBTRUxFQ1RJT046IHtcbiAgICBTRUxFQ1RFRF9DTEFTUzogXCJ2aWV3cG9ydC1pdGVtLS1zZWxlY3RlZFwiLFxuICB9LFxuXG4gIC8vIFNjcm9sbGJhciBzZXR0aW5ncyAoZnJvbSBsaXN0LW1hbmFnZXIpXG4gIFNDUk9MTEJBUjoge1xuICAgIC8vIENTUyBjbGFzc2VzXG4gICAgQ0xBU1NFUzoge1xuICAgICAgU0NST0xMQkFSOiBcInZpZXdwb3J0X19zY3JvbGxiYXJcIixcbiAgICAgIFNDUk9MTEJBUl9UUkFDSzogXCJ2aWV3cG9ydF9fc2Nyb2xsYmFyLXRyYWNrXCIsXG4gICAgICBTQ1JPTExCQVJfVEhVTUI6IFwidmlld3BvcnRfX3Njcm9sbGJhci10aHVtYlwiLFxuICAgICAgU0NST0xMQkFSX1ZJU0lCTEU6IFwidmlld3BvcnRfX3Njcm9sbGJhci0tdmlzaWJsZVwiLFxuICAgICAgU0NST0xMQkFSX0RSQUdHSU5HOiBcInZpZXdwb3J0X19zY3JvbGxiYXItLWRyYWdnaW5nXCIsXG4gICAgICBTQ1JPTExCQVJfVEhVTUJfRFJBR0dJTkc6IFwidmlld3BvcnRfX3Njcm9sbGJhci10aHVtYi0tZHJhZ2dpbmdcIixcbiAgICB9LFxuICB9LFxuXG4gIC8vIE9yaWVudGF0aW9uIChmcm9tIGxpc3QtbWFuYWdlcilcbiAgT1JJRU5UQVRJT046IHtcbiAgICBERUZBVUxUX09SSUVOVEFUSU9OOiBcInZlcnRpY2FsXCIsXG4gICAgREVGQVVMVF9DUk9TU19BWElTX0FMSUdOTUVOVDogXCJzdHJldGNoXCIsXG4gICAgUkVWRVJTRV9ESVJFQ1RJT046IGZhbHNlLFxuICB9LFxuXG4gIFBBR0lOQVRJT046IHtcbiAgICBERUZBVUxUX1NUUkFURUdZOiBcIm9mZnNldFwiIGFzIFwib2Zmc2V0XCIgfCBcInBhZ2VcIiB8IFwiY3Vyc29yXCIsXG4gICAgREVGQVVMVF9MSU1JVDogMjAsXG4gICAgU1RSQVRFR0lFUzoge1xuICAgICAgUEFHRTogXCJwYWdlXCIgYXMgY29uc3QsXG4gICAgICBPRkZTRVQ6IFwib2Zmc2V0XCIgYXMgY29uc3QsXG4gICAgICBDVVJTT1I6IFwiY3Vyc29yXCIgYXMgY29uc3QsXG4gICAgfSxcbiAgICBDVVJTT1JfQ0xFQU5VUF9JTlRFUlZBTDogNjAwMDAsIC8vIENsZWFuIHVwIG9sZCBjdXJzb3JzIGV2ZXJ5IG1pbnV0ZVxuICAgIE1BWF9DVVJTT1JfTUFQX1NJWkU6IDEwMDAsIC8vIE1heGltdW0gbnVtYmVyIG9mIGN1cnNvcnMgdG8ga2VlcCBpbiBtZW1vcnlcbiAgICAvLyBDdXJzb3Itc3BlY2lmaWMgdmlydHVhbCBzaXppbmdcbiAgICBDVVJTT1JfU0NST0xMX01BUkdJTl9NVUxUSVBMSUVSOiAzLCAvLyBNdWx0aXBseSByYW5nZVNpemUgYnkgdGhpcyBmb3Igc2Nyb2xsIG1hcmdpblxuICAgIENVUlNPUl9NSU5fVklSVFVBTF9TSVpFX01VTFRJUExJRVI6IDMsIC8vIE1pbmltdW0gdmlydHVhbCBzaXplIGFzIG11bHRpcGxpZXIgb2YgcmFuZ2VTaXplXG4gIH0sXG59O1xuXG4vKipcbiAqIFR5cGUgZm9yIG92ZXJyaWRpbmcgY29uc3RhbnRzIGF0IHJ1bnRpbWVcbiAqL1xuZXhwb3J0IHR5cGUgVmlld3BvcnRDb25zdGFudHMgPSB0eXBlb2YgVklFV1BPUlRfQ09OU1RBTlRTO1xuXG4vKipcbiAqIEhlbHBlciBmdW5jdGlvbiB0byBtZXJnZSB1c2VyIGNvbnN0YW50cyB3aXRoIGRlZmF1bHRzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBtZXJnZUNvbnN0YW50cyhcbiAgdXNlckNvbnN0YW50czogUGFydGlhbDxWaWV3cG9ydENvbnN0YW50cz4gPSB7fSxcbik6IFZpZXdwb3J0Q29uc3RhbnRzIHtcbiAgcmV0dXJuIHtcbiAgICAuLi5WSUVXUE9SVF9DT05TVEFOVFMsXG4gICAgLi4udXNlckNvbnN0YW50cyxcbiAgfTtcbn1cbiIsCiAgICAiLyoqXG4gKiBWaXJ0dWFsIEZlYXR1cmUgLSBDb3JlIHZpcnR1YWwgc2Nyb2xsaW5nIGNhbGN1bGF0aW9uc1xuICogSGFuZGxlcyB2aXNpYmxlIHJhbmdlIGNhbGN1bGF0aW9uIGFuZCB0b3RhbCB2aXJ0dWFsIHNpemUgbWFuYWdlbWVudFxuICovXG5cbmltcG9ydCB0eXBlIHsgVmlld3BvcnRDb250ZXh0LCBWaWV3cG9ydENvbXBvbmVudCB9IGZyb20gXCIuLi90eXBlc1wiO1xuaW1wb3J0IHsgVklFV1BPUlRfQ09OU1RBTlRTIH0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgd3JhcEluaXRpYWxpemUsIHdyYXBEZXN0cm95LCBnZXRWaWV3cG9ydFN0YXRlIH0gZnJvbSBcIi4vdXRpbHNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBWaXJ0dWFsQ29uZmlnIHtcbiAgaXRlbVNpemU/OiBudW1iZXI7XG4gIG92ZXJzY2FuPzogbnVtYmVyO1xuICBvcmllbnRhdGlvbj86IFwidmVydGljYWxcIiB8IFwiaG9yaXpvbnRhbFwiO1xuICBhdXRvRGV0ZWN0SXRlbVNpemU/OiBib29sZWFuO1xuICBkZWJ1Zz86IGJvb2xlYW47XG4gIGluaXRpYWxTY3JvbGxJbmRleD86IG51bWJlcjtcbn1cblxuLyoqXG4gKiBWaXJ0dWFsIHNjcm9sbGluZyBmZWF0dXJlIGZvciB2aWV3cG9ydFxuICogSGFuZGxlcyB2aXNpYmxlIHJhbmdlIGNhbGN1bGF0aW9ucyB3aXRoIGNvbXByZXNzaW9uIGZvciBsYXJnZSBkYXRhc2V0c1xuICovXG5leHBvcnQgY29uc3Qgd2l0aFZpcnR1YWwgPSAoY29uZmlnOiBWaXJ0dWFsQ29uZmlnID0ge30pID0+IHtcbiAgcmV0dXJuIDxUIGV4dGVuZHMgVmlld3BvcnRDb250ZXh0ICYgVmlld3BvcnRDb21wb25lbnQ+KGNvbXBvbmVudDogVCk6IFQgPT4ge1xuICAgIGNvbnN0IHtcbiAgICAgIGl0ZW1TaXplLFxuICAgICAgb3ZlcnNjYW4gPSBWSUVXUE9SVF9DT05TVEFOVFMuVklSVFVBTF9TQ1JPTEwuT1ZFUlNDQU5fQlVGRkVSLFxuICAgICAgb3JpZW50YXRpb24gPSBcInZlcnRpY2FsXCIsXG4gICAgICBhdXRvRGV0ZWN0SXRlbVNpemUgPSBpdGVtU2l6ZSA9PT0gdW5kZWZpbmVkXG4gICAgICAgID8gVklFV1BPUlRfQ09OU1RBTlRTLlZJUlRVQUxfU0NST0xMLkFVVE9fREVURUNUX0lURU1fU0laRVxuICAgICAgICA6IGZhbHNlLFxuICAgICAgZGVidWcgPSBmYWxzZSxcbiAgICAgIGluaXRpYWxTY3JvbGxJbmRleCA9IDAsXG4gICAgfSA9IGNvbmZpZztcblxuICAgIC8vIFVzZSBwcm92aWRlZCBpdGVtU2l6ZSBvciBkZWZhdWx0LCBidXQgbWFyayBpZiB3ZSBzaG91bGQgYXV0by1kZXRlY3RcbiAgICBjb25zdCBpbml0aWFsSXRlbVNpemUgPVxuICAgICAgaXRlbVNpemUgfHwgVklFV1BPUlRfQ09OU1RBTlRTLlZJUlRVQUxfU0NST0xMLkRFRkFVTFRfSVRFTV9TSVpFO1xuXG4gICAgY29uc3QgTUFYX1ZJUlRVQUxfU0laRSA9IFZJRVdQT1JUX0NPTlNUQU5UUy5WSVJUVUFMX1NDUk9MTC5NQVhfVklSVFVBTF9TSVpFO1xuICAgIGxldCB2aWV3cG9ydFN0YXRlOiBhbnk7XG4gICAgbGV0IGhhc0NhbGN1bGF0ZWRJdGVtU2l6ZSA9IGZhbHNlO1xuICAgIGxldCBoYXNSZWNhbGN1bGF0ZWRTY3JvbGxGb3JDb21wcmVzc2lvbiA9IGZhbHNlOyAvLyBUcmFjayBpZiB3ZSd2ZSByZWNhbGN1bGF0ZWQgc2Nyb2xsIHBvc2l0aW9uIGZvciBjb21wcmVzc2lvblxuICAgIGxldCByZXNpemVPYnNlcnZlcjogUmVzaXplT2JzZXJ2ZXIgfCBudWxsID0gbnVsbDtcblxuICAgIC8vIENhY2hlIHZhcmlhYmxlcyBmb3IgcmFuZ2UgY2FsY3VsYXRpb25zIChkZWNsYXJlZCBoZXJlLCBpbml0aWFsaXplZCBsYXRlcilcbiAgICBsZXQgbGFzdFNjcm9sbFBvc2l0aW9uID0gLTE7XG4gICAgbGV0IGxhc3RWaXNpYmxlUmFuZ2U6IHsgc3RhcnQ6IG51bWJlcjsgZW5kOiBudW1iZXIgfSB8IG51bGwgPSBudWxsO1xuICAgIGxldCBsYXN0QWN0dWFsVmlzaWJsZVJhbmdlOiB7IHN0YXJ0OiBudW1iZXI7IGVuZDogbnVtYmVyIH0gfCBudWxsID0gbnVsbDtcblxuICAgIC8vIExpc3RlbiBmb3IgcmVsb2FkOnN0YXJ0IHRvIHJlc2V0IGZlYXR1cmUtc3BlY2lmaWMgc3RhdGVcbiAgICAvLyBOT1RFOiBXZSBkbyBOT1QgcmVzZXQgc2Nyb2xsUG9zaXRpb24gaGVyZSAtIHRoZSBjYWxsZXIgKHJlbG9hZCBvciByZWxvYWRBdClcbiAgICAvLyBpcyByZXNwb25zaWJsZSBmb3Igc2V0dGluZyB0aGUgc2Nyb2xsIHBvc2l0aW9uLiBUaGlzIHByZXZlbnRzIHJhY2UgY29uZGl0aW9uc1xuICAgIC8vIHdoZXJlIHJlbG9hZEF0IHNldHMgYSBwb3NpdGlvbiBidXQgd2UgcmVzZXQgaXQgdG8gMCBiZWZvcmUgdGhlIGxvYWQgY29tcGxldGVzLlxuICAgIGNvbXBvbmVudC5vbj8uKFwicmVsb2FkOnN0YXJ0XCIsICgpID0+IHtcbiAgICAgIGhhc0NhbGN1bGF0ZWRJdGVtU2l6ZSA9IGZhbHNlO1xuICAgICAgaGFzUmVjYWxjdWxhdGVkU2Nyb2xsRm9yQ29tcHJlc3Npb24gPSBmYWxzZTtcbiAgICAgIC8vIFJlc2V0IGNhY2hlXG4gICAgICBsYXN0U2Nyb2xsUG9zaXRpb24gPSAtMTtcbiAgICAgIGxhc3RWaXNpYmxlUmFuZ2UgPSBudWxsO1xuICAgICAgbGFzdEFjdHVhbFZpc2libGVSYW5nZSA9IG51bGw7XG4gICAgICAvLyBSZXNldCB2aWV3cG9ydFN0YXRlIGlmIGl0IGV4aXN0cywgYnV0IHByZXNlcnZlIHNjcm9sbFBvc2l0aW9uXG4gICAgICAvLyBUaGUgY2FsbGVyIChyZWxvYWQvcmVsb2FkQXQpIGlzIHJlc3BvbnNpYmxlIGZvciBzZXR0aW5nIHNjcm9sbFBvc2l0aW9uXG4gICAgICBpZiAodmlld3BvcnRTdGF0ZSkge1xuICAgICAgICAvLyBEb24ndCByZXNldCBzY3JvbGxQb3NpdGlvbiAtIGNhbGxlciBjb250cm9scyBpdFxuICAgICAgICAvLyB2aWV3cG9ydFN0YXRlLnNjcm9sbFBvc2l0aW9uID0gMDtcbiAgICAgICAgdmlld3BvcnRTdGF0ZS50b3RhbEl0ZW1zID0gMDtcbiAgICAgICAgdmlld3BvcnRTdGF0ZS52aXJ0dWFsVG90YWxTaXplID0gMDtcbiAgICAgICAgdmlld3BvcnRTdGF0ZS52aXNpYmxlUmFuZ2UgPSB7IHN0YXJ0OiAwLCBlbmQ6IDAgfTtcbiAgICAgICAgZGVsZXRlICh2aWV3cG9ydFN0YXRlIGFzIGFueSkudGFyZ2V0U2Nyb2xsSW5kZXg7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICAvLyBJbml0aWFsaXplIHVzaW5nIHNoYXJlZCB3cmFwcGVyXG4gICAgd3JhcEluaXRpYWxpemUoY29tcG9uZW50LCAoKSA9PiB7XG4gICAgICB2aWV3cG9ydFN0YXRlID0gZ2V0Vmlld3BvcnRTdGF0ZShjb21wb25lbnQpO1xuICAgICAgaWYgKCF2aWV3cG9ydFN0YXRlKSByZXR1cm47XG5cbiAgICAgIE9iamVjdC5hc3NpZ24odmlld3BvcnRTdGF0ZSwge1xuICAgICAgICBpdGVtU2l6ZTogaW5pdGlhbEl0ZW1TaXplLFxuICAgICAgICBvdmVyc2NhbixcbiAgICAgICAgY29udGFpbmVyU2l6ZTpcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLnZpZXdwb3J0RWxlbWVudD8uW1xuICAgICAgICAgICAgb3JpZW50YXRpb24gPT09IFwiaG9yaXpvbnRhbFwiID8gXCJvZmZzZXRXaWR0aFwiIDogXCJvZmZzZXRIZWlnaHRcIlxuICAgICAgICAgIF0gfHwgNjAwLFxuICAgICAgfSk7XG5cbiAgICAgIHVwZGF0ZVRvdGFsVmlydHVhbFNpemUodmlld3BvcnRTdGF0ZS50b3RhbEl0ZW1zKTtcblxuICAgICAgLy8gSWYgd2UgaGF2ZSBhbiBpbml0aWFsIHNjcm9sbCBpbmRleCwgY2FsY3VsYXRlIGluaXRpYWwgc2Nyb2xsIHBvc2l0aW9uXG4gICAgICAvLyBOb3RlOiBXZSBzdG9yZSB0aGUgaW5pdGlhbFNjcm9sbEluZGV4IHRvIHVzZSBkaXJlY3RseSBpbiByYW5nZSBjYWxjdWxhdGlvbnNcbiAgICAgIC8vIGJlY2F1c2Ugc2Nyb2xsIHBvc2l0aW9uIG1heSBiZSBjb21wcmVzc2VkIGZvciBsYXJnZSBsaXN0c1xuICAgICAgbGV0IGluaXRpYWxTY3JvbGxQb3NpdGlvbiA9IHZpZXdwb3J0U3RhdGUuc2Nyb2xsUG9zaXRpb24gfHwgMDtcblxuICAgICAgaWYgKGluaXRpYWxTY3JvbGxJbmRleCA+IDApIHtcbiAgICAgICAgLy8gU3RvcmUgdGhlIHRhcmdldCBpbmRleCBmb3IgdXNlIGluIHJhbmdlIGNhbGN1bGF0aW9uc1xuICAgICAgICAodmlld3BvcnRTdGF0ZSBhcyBhbnkpLnRhcmdldFNjcm9sbEluZGV4ID0gaW5pdGlhbFNjcm9sbEluZGV4O1xuXG4gICAgICAgIC8vIENhbGN1bGF0ZSBzY3JvbGwgcG9zaXRpb24gKG1heSBiZSBjb21wcmVzc2VkIGZvciBsYXJnZSBsaXN0cylcbiAgICAgICAgaW5pdGlhbFNjcm9sbFBvc2l0aW9uID1cbiAgICAgICAgICBpbml0aWFsU2Nyb2xsSW5kZXggKiAodmlld3BvcnRTdGF0ZS5pdGVtU2l6ZSB8fCBpbml0aWFsSXRlbVNpemUpO1xuICAgICAgICB2aWV3cG9ydFN0YXRlLnNjcm9sbFBvc2l0aW9uID0gaW5pdGlhbFNjcm9sbFBvc2l0aW9uO1xuXG4gICAgICAgIC8vIE5vdGlmeSBzY3JvbGxpbmcgZmVhdHVyZSB0byBzeW5jIGl0cyBsb2NhbCBzY3JvbGwgcG9zaXRpb25cbiAgICAgICAgLy8gVXNlIHNldFRpbWVvdXQgdG8gZW5zdXJlIHNjcm9sbGluZyBmZWF0dXJlIGhhcyBpbml0aWFsaXplZFxuICAgICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICBjb21wb25lbnQuZW1pdD8uKFwidmlld3BvcnQ6c2Nyb2xsLXBvc2l0aW9uLXN5bmNcIiwge1xuICAgICAgICAgICAgcG9zaXRpb246IGluaXRpYWxTY3JvbGxQb3NpdGlvbixcbiAgICAgICAgICAgIHNvdXJjZTogXCJpbml0aWFsLXNjcm9sbC1pbmRleFwiLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9LCAwKTtcbiAgICAgIH1cblxuICAgICAgdXBkYXRlVmlzaWJsZVJhbmdlKGluaXRpYWxTY3JvbGxQb3NpdGlvbik7XG5cbiAgICAgIC8vIEVuc3VyZSBjb250YWluZXIgc2l6ZSBpcyBtZWFzdXJlZCBhZnRlciBET00gaXMgcmVhZHlcbiAgICAgIHJlcXVlc3RBbmltYXRpb25GcmFtZSgoKSA9PiB7XG4gICAgICAgIHVwZGF0ZUNvbnRhaW5lclNpemUoKTtcbiAgICAgIH0pO1xuXG4gICAgICAvLyBTZXQgdXAgUmVzaXplT2JzZXJ2ZXIgdG8gaGFuZGxlIHZpZXdwb3J0IGNvbnRhaW5lciByZXNpemVcbiAgICAgIGlmIChcbiAgICAgICAgdmlld3BvcnRTdGF0ZS52aWV3cG9ydEVsZW1lbnQgJiZcbiAgICAgICAgdHlwZW9mIFJlc2l6ZU9ic2VydmVyICE9PSBcInVuZGVmaW5lZFwiXG4gICAgICApIHtcbiAgICAgICAgcmVzaXplT2JzZXJ2ZXIgPSBuZXcgUmVzaXplT2JzZXJ2ZXIoKGVudHJpZXMpID0+IHtcbiAgICAgICAgICBmb3IgKGNvbnN0IGVudHJ5IG9mIGVudHJpZXMpIHtcbiAgICAgICAgICAgIC8vIFVzZSBjb250ZW50Qm94U2l6ZSBpZiBhdmFpbGFibGUsIG90aGVyd2lzZSBmYWxsIGJhY2sgdG8gY29udGVudFJlY3RcbiAgICAgICAgICAgIGNvbnN0IHNpemUgPVxuICAgICAgICAgICAgICBvcmllbnRhdGlvbiA9PT0gXCJob3Jpem9udGFsXCJcbiAgICAgICAgICAgICAgICA/IGVudHJ5LmNvbnRlbnRSZWN0LndpZHRoXG4gICAgICAgICAgICAgICAgOiBlbnRyeS5jb250ZW50UmVjdC5oZWlnaHQ7XG5cbiAgICAgICAgICAgIGlmIChzaXplID4gMCAmJiBzaXplICE9PSB2aWV3cG9ydFN0YXRlPy5jb250YWluZXJTaXplKSB7XG4gICAgICAgICAgICAgIHVwZGF0ZUNvbnRhaW5lclNpemUoKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgICAgICByZXNpemVPYnNlcnZlci5vYnNlcnZlKHZpZXdwb3J0U3RhdGUudmlld3BvcnRFbGVtZW50KTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIC8vIEhlbHBlciBmdW5jdGlvbnNcbiAgICBjb25zdCBnZXRDb21wcmVzc2lvblJhdGlvID0gKCk6IG51bWJlciA9PiB7XG4gICAgICBpZiAoIXZpZXdwb3J0U3RhdGU/LnZpcnR1YWxUb3RhbFNpemUpIHJldHVybiAxO1xuICAgICAgY29uc3QgYWN0dWFsU2l6ZSA9IHZpZXdwb3J0U3RhdGUudG90YWxJdGVtcyAqIHZpZXdwb3J0U3RhdGUuaXRlbVNpemU7XG4gICAgICByZXR1cm4gYWN0dWFsU2l6ZSA8PSBNQVhfVklSVFVBTF9TSVpFID8gMSA6IE1BWF9WSVJUVUFMX1NJWkUgLyBhY3R1YWxTaXplO1xuICAgIH07XG5cbiAgICBjb25zdCBsb2cgPSAobWVzc2FnZTogc3RyaW5nLCBkYXRhPzogYW55KSA9PiB7XG4gICAgICBpZiAoZGVidWcpIGNvbnNvbGUubG9nKGBbVmlydHVhbF0gJHttZXNzYWdlfWAsIGRhdGEpO1xuICAgIH07XG5cbiAgICAvLyBDYWxjdWxhdGUgdmlzaWJsZSByYW5nZVxuICAgIGNvbnN0IGNhbGN1bGF0ZVZpc2libGVSYW5nZSA9IChcbiAgICAgIHNjcm9sbFBvc2l0aW9uOiBudW1iZXIsXG4gICAgKTogeyBzdGFydDogbnVtYmVyOyBlbmQ6IG51bWJlciB9ID0+IHtcbiAgICAgIGlmICghdmlld3BvcnRTdGF0ZSkge1xuICAgICAgICBjb25zb2xlLndhcm4oXCJbVmlydHVhbF0gTm8gdmlld3BvcnQgc3RhdGUsIHJldHVybmluZyBlbXB0eSByYW5nZVwiKTtcbiAgICAgICAgcmV0dXJuIHsgc3RhcnQ6IDAsIGVuZDogMCB9O1xuICAgICAgfVxuXG4gICAgICBjb25zdCB7IGNvbnRhaW5lclNpemUsIHRvdGFsSXRlbXMgfSA9IHZpZXdwb3J0U3RhdGU7XG5cbiAgICAgIC8vIEVhcmx5IHJldHVybnMgZm9yIGludmFsaWQgc3RhdGVzXG4gICAgICBpZiAoXG4gICAgICAgICFjb250YWluZXJTaXplIHx8XG4gICAgICAgIGNvbnRhaW5lclNpemUgPD0gMCB8fFxuICAgICAgICAhdG90YWxJdGVtcyB8fFxuICAgICAgICB0b3RhbEl0ZW1zIDw9IDBcbiAgICAgICkge1xuICAgICAgICAvLyBJZiB3ZSBoYXZlIGFuIGluaXRpYWxTY3JvbGxJbmRleCwgdXNlIGl0IHRvIGNhbGN1bGF0ZSB0aGUgcmFuZ2VcbiAgICAgICAgLy8gZXZlbiB3aGVuIHRvdGFsSXRlbXMgaXMgMCAobm90IHlldCBsb2FkZWQgZnJvbSBBUEkpXG4gICAgICAgIGlmIChpbml0aWFsU2Nyb2xsSW5kZXggPiAwKSB7XG4gICAgICAgICAgY29uc3QgdmlzaWJsZUNvdW50ID0gTWF0aC5jZWlsKFxuICAgICAgICAgICAgKGNvbnRhaW5lclNpemUgfHwgNjAwKSAvXG4gICAgICAgICAgICAgICh2aWV3cG9ydFN0YXRlLml0ZW1TaXplIHx8IGluaXRpYWxJdGVtU2l6ZSksXG4gICAgICAgICAgKTtcbiAgICAgICAgICBjb25zdCBzdGFydCA9IE1hdGgubWF4KDAsIGluaXRpYWxTY3JvbGxJbmRleCAtIG92ZXJzY2FuKTtcbiAgICAgICAgICBjb25zdCBlbmQgPSBpbml0aWFsU2Nyb2xsSW5kZXggKyB2aXNpYmxlQ291bnQgKyBvdmVyc2NhbjtcbiAgICAgICAgICByZXR1cm4geyBzdGFydCwgZW5kIH07XG4gICAgICAgIH1cbiAgICAgICAgbG9nKGBJbnZhbGlkIHN0YXRlOiBjb250YWluZXI9JHtjb250YWluZXJTaXplfSwgaXRlbXM9JHt0b3RhbEl0ZW1zfWApO1xuICAgICAgICByZXR1cm4geyBzdGFydDogMCwgZW5kOiAwIH07XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHZpcnR1YWxTaXplID1cbiAgICAgICAgdmlld3BvcnRTdGF0ZS52aXJ0dWFsVG90YWxTaXplIHx8IHRvdGFsSXRlbXMgKiB2aWV3cG9ydFN0YXRlLml0ZW1TaXplO1xuICAgICAgY29uc3QgdmlzaWJsZUNvdW50ID0gTWF0aC5jZWlsKGNvbnRhaW5lclNpemUgLyB2aWV3cG9ydFN0YXRlLml0ZW1TaXplKTtcbiAgICAgIGNvbnN0IGNvbXByZXNzaW9uUmF0aW8gPSBnZXRDb21wcmVzc2lvblJhdGlvKCk7XG5cbiAgICAgIGxldCBzdGFydDogbnVtYmVyLCBlbmQ6IG51bWJlcjtcblxuICAgICAgLy8gQ2hlY2sgaWYgd2UgaGF2ZSBhIHRhcmdldCBzY3JvbGwgaW5kZXggKGZvciBpbml0aWFsU2Nyb2xsSW5kZXggd2l0aCBjb21wcmVzc2lvbilcbiAgICAgIGNvbnN0IHRhcmdldFNjcm9sbEluZGV4ID0gKHZpZXdwb3J0U3RhdGUgYXMgYW55KS50YXJnZXRTY3JvbGxJbmRleDtcblxuICAgICAgaWYgKGNvbXByZXNzaW9uUmF0aW8gPCAxKSB7XG4gICAgICAgIC8vIENvbXByZXNzZWQgc3BhY2UgY2FsY3VsYXRpb25cbiAgICAgICAgLy8gSWYgd2UgaGF2ZSBhIHRhcmdldFNjcm9sbEluZGV4LCB1c2UgaXQgZGlyZWN0bHkgaW5zdGVhZCBvZiBjYWxjdWxhdGluZyBmcm9tIHNjcm9sbCBwb3NpdGlvblxuICAgICAgICAvLyBUaGlzIGVuc3VyZXMgd2Ugc2hvdyB0aGUgY29ycmVjdCBpdGVtcyBldmVuIHdoZW4gdmlydHVhbCBzcGFjZSBpcyBjb21wcmVzc2VkXG4gICAgICAgIGlmICh0YXJnZXRTY3JvbGxJbmRleCAhPT0gdW5kZWZpbmVkICYmIHRhcmdldFNjcm9sbEluZGV4ID4gMCkge1xuICAgICAgICAgIHN0YXJ0ID0gTWF0aC5tYXgoMCwgdGFyZ2V0U2Nyb2xsSW5kZXggLSBvdmVyc2Nhbik7XG4gICAgICAgICAgZW5kID0gTWF0aC5taW4oXG4gICAgICAgICAgICB0b3RhbEl0ZW1zIC0gMSxcbiAgICAgICAgICAgIHRhcmdldFNjcm9sbEluZGV4ICsgdmlzaWJsZUNvdW50ICsgb3ZlcnNjYW4sXG4gICAgICAgICAgKTtcbiAgICAgICAgICAvLyBDbGVhciB0YXJnZXRTY3JvbGxJbmRleCBhZnRlciBmaXJzdCB1c2Ugc28gbm9ybWFsIHNjcm9sbGluZyB3b3Jrc1xuICAgICAgICAgIGRlbGV0ZSAodmlld3BvcnRTdGF0ZSBhcyBhbnkpLnRhcmdldFNjcm9sbEluZGV4O1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGNvbnN0IHNjcm9sbFJhdGlvID0gc2Nyb2xsUG9zaXRpb24gLyB2aXJ0dWFsU2l6ZTtcbiAgICAgICAgICBjb25zdCBleGFjdEluZGV4ID0gc2Nyb2xsUmF0aW8gKiB0b3RhbEl0ZW1zO1xuICAgICAgICAgIHN0YXJ0ID0gTWF0aC5mbG9vcihleGFjdEluZGV4KTtcbiAgICAgICAgICBlbmQgPSBNYXRoLmNlaWwoZXhhY3RJbmRleCkgKyB2aXNpYmxlQ291bnQ7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBOZWFyLWJvdHRvbSBoYW5kbGluZ1xuICAgICAgICBjb25zdCBtYXhTY3JvbGwgPSB2aXJ0dWFsU2l6ZSAtIGNvbnRhaW5lclNpemU7XG4gICAgICAgIGNvbnN0IGRpc3RhbmNlRnJvbUJvdHRvbSA9IG1heFNjcm9sbCAtIHNjcm9sbFBvc2l0aW9uO1xuXG4gICAgICAgIGlmIChkaXN0YW5jZUZyb21Cb3R0b20gPD0gY29udGFpbmVyU2l6ZSAmJiBkaXN0YW5jZUZyb21Cb3R0b20gPj0gLTEpIHtcbiAgICAgICAgICBjb25zdCBpdGVtc0F0Qm90dG9tID0gTWF0aC5mbG9vcihcbiAgICAgICAgICAgIGNvbnRhaW5lclNpemUgLyB2aWV3cG9ydFN0YXRlLml0ZW1TaXplLFxuICAgICAgICAgICk7XG4gICAgICAgICAgY29uc3QgZmlyc3RWaXNpYmxlQXRCb3R0b20gPSBNYXRoLm1heCgwLCB0b3RhbEl0ZW1zIC0gaXRlbXNBdEJvdHRvbSk7XG4gICAgICAgICAgY29uc3QgaW50ZXJwb2xhdGlvbiA9IE1hdGgubWF4KFxuICAgICAgICAgICAgMCxcbiAgICAgICAgICAgIE1hdGgubWluKDEsIDEgLSBkaXN0YW5jZUZyb21Cb3R0b20gLyBjb250YWluZXJTaXplKSxcbiAgICAgICAgICApO1xuXG4gICAgICAgICAgc3RhcnQgPSBNYXRoLmZsb29yKFxuICAgICAgICAgICAgc3RhcnQgKyAoZmlyc3RWaXNpYmxlQXRCb3R0b20gLSBzdGFydCkgKiBpbnRlcnBvbGF0aW9uLFxuICAgICAgICAgICk7XG4gICAgICAgICAgZW5kID1cbiAgICAgICAgICAgIGRpc3RhbmNlRnJvbUJvdHRvbSA8PSAxXG4gICAgICAgICAgICAgID8gdG90YWxJdGVtcyAtIDFcbiAgICAgICAgICAgICAgOiBNYXRoLm1pbih0b3RhbEl0ZW1zIC0gMSwgc3RhcnQgKyB2aXNpYmxlQ291bnQgKyBvdmVyc2Nhbik7XG5cbiAgICAgICAgICBsb2coXCJOZWFyIGJvdHRvbSBjYWxjdWxhdGlvbjpcIiwge1xuICAgICAgICAgICAgZGlzdGFuY2VGcm9tQm90dG9tLFxuICAgICAgICAgICAgaW50ZXJwb2xhdGlvbixcbiAgICAgICAgICAgIHN0YXJ0LFxuICAgICAgICAgICAgZW5kLFxuICAgICAgICAgICAgc2Nyb2xsUG9zaXRpb24sXG4gICAgICAgICAgICB0b3RhbEl0ZW1zLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gQXBwbHkgb3ZlcnNjYW5cbiAgICAgICAgc3RhcnQgPSBNYXRoLm1heCgwLCBzdGFydCAtIG92ZXJzY2FuKTtcbiAgICAgICAgZW5kID0gTWF0aC5taW4odG90YWxJdGVtcyAtIDEsIGVuZCArIG92ZXJzY2FuKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIERpcmVjdCBjYWxjdWxhdGlvblxuICAgICAgICBzdGFydCA9IE1hdGgubWF4KFxuICAgICAgICAgIDAsXG4gICAgICAgICAgTWF0aC5mbG9vcihzY3JvbGxQb3NpdGlvbiAvIHZpZXdwb3J0U3RhdGUuaXRlbVNpemUpIC0gb3ZlcnNjYW4sXG4gICAgICAgICk7XG4gICAgICAgIGVuZCA9IE1hdGgubWluKHRvdGFsSXRlbXMgLSAxLCBzdGFydCArIHZpc2libGVDb3VudCArIG92ZXJzY2FuICogMik7XG4gICAgICB9XG5cbiAgICAgIC8vIFZhbGlkYXRlIG91dHB1dFxuICAgICAgaWYgKGlzTmFOKHN0YXJ0KSB8fCBpc05hTihlbmQpKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoXCJbVmlydHVhbF0gTmFOIGluIHJhbmdlIGNhbGN1bGF0aW9uOlwiLCB7XG4gICAgICAgICAgc2Nyb2xsUG9zaXRpb24sXG4gICAgICAgICAgY29udGFpbmVyU2l6ZSxcbiAgICAgICAgICB0b3RhbEl0ZW1zLFxuICAgICAgICAgIGl0ZW1TaXplOiB2aWV3cG9ydFN0YXRlLml0ZW1TaXplLFxuICAgICAgICAgIGNvbXByZXNzaW9uUmF0aW8sXG4gICAgICAgIH0pO1xuICAgICAgICByZXR1cm4geyBzdGFydDogMCwgZW5kOiAwIH07XG4gICAgICB9XG5cbiAgICAgIC8vIFN0cmF0ZWdpYyBsb2cgZm9yIGxhc3QgcmFuZ2VcbiAgICAgIC8vIGlmIChlbmQgPj0gdG90YWxJdGVtcyAtIDEwKSB7XG4gICAgICAvLyAgIGNvbnNvbGUubG9nKFxuICAgICAgLy8gICAgIGBbVmlydHVhbF0gTmVhciBlbmQgcmFuZ2U6ICR7c3RhcnR9LSR7ZW5kfSwgdG90YWxJdGVtcz0ke3RvdGFsSXRlbXN9LCBsYXN0SXRlbVBvcz0ke1xuICAgICAgLy8gICAgICAgZW5kICogdmlld3BvcnRTdGF0ZS5pdGVtU2l6ZVxuICAgICAgLy8gICAgIH1weCwgdmlydHVhbFNpemU9JHt2aWV3cG9ydFN0YXRlLnZpcnR1YWxUb3RhbFNpemV9cHhgXG4gICAgICAvLyAgICk7XG4gICAgICAvLyB9XG5cbiAgICAgIHJldHVybiB7IHN0YXJ0LCBlbmQgfTtcbiAgICB9O1xuXG4gICAgLy8gQ2FsY3VsYXRlIGFjdHVhbCB2aXNpYmxlIHJhbmdlICh3aXRob3V0IG92ZXJzY2FuIGJ1ZmZlcilcbiAgICBjb25zdCBjYWxjdWxhdGVBY3R1YWxWaXNpYmxlUmFuZ2UgPSAoc2Nyb2xsUG9zaXRpb246IG51bWJlcikgPT4ge1xuICAgICAgaWYgKCF2aWV3cG9ydFN0YXRlKSByZXR1cm4geyBzdGFydDogMCwgZW5kOiAwIH07XG5cbiAgICAgIGNvbnN0IHsgY29udGFpbmVyU2l6ZSwgdG90YWxJdGVtcyB9ID0gdmlld3BvcnRTdGF0ZTtcblxuICAgICAgaWYgKCFjb250YWluZXJTaXplIHx8ICF0b3RhbEl0ZW1zKSByZXR1cm4geyBzdGFydDogMCwgZW5kOiAwIH07XG5cbiAgICAgIGNvbnN0IGl0ZW1TaXplID0gdmlld3BvcnRTdGF0ZS5pdGVtU2l6ZTtcbiAgICAgIGNvbnN0IHZpc2libGVDb3VudCA9IE1hdGguY2VpbChjb250YWluZXJTaXplIC8gaXRlbVNpemUpO1xuICAgICAgY29uc3QgYWN0dWFsU2l6ZSA9IHRvdGFsSXRlbXMgKiBpdGVtU2l6ZTtcbiAgICAgIGNvbnN0IHZpcnR1YWxTaXplID0gdmlld3BvcnRTdGF0ZS52aXJ0dWFsVG90YWxTaXplIHx8IGFjdHVhbFNpemU7XG4gICAgICAvLyBjb21wcmVzc2lvblJhdGlvIDwgMSBtZWFucyBsaXN0IGlzIGNvbXByZXNzZWQgKHZpcnR1YWxTaXplIDwgYWN0dWFsU2l6ZSlcbiAgICAgIGNvbnN0IGNvbXByZXNzaW9uUmF0aW8gPSBhY3R1YWxTaXplID4gMCA/IHZpcnR1YWxTaXplIC8gYWN0dWFsU2l6ZSA6IDE7XG5cbiAgICAgIGxldCBzdGFydDogbnVtYmVyLCBlbmQ6IG51bWJlcjtcblxuICAgICAgaWYgKGNvbXByZXNzaW9uUmF0aW8gPCAxKSB7XG4gICAgICAgIC8vIENvbXByZXNzZWQgc3BhY2UgLSBjYWxjdWxhdGUgYmFzZWQgb24gc2Nyb2xsIHJhdGlvXG4gICAgICAgIGNvbnN0IHNjcm9sbFJhdGlvID0gc2Nyb2xsUG9zaXRpb24gLyB2aXJ0dWFsU2l6ZTtcbiAgICAgICAgc3RhcnQgPSBNYXRoLmZsb29yKHNjcm9sbFJhdGlvICogdG90YWxJdGVtcyk7XG4gICAgICAgIGVuZCA9IE1hdGgubWluKHRvdGFsSXRlbXMgLSAxLCBzdGFydCArIHZpc2libGVDb3VudCAtIDEpO1xuXG4gICAgICAgIC8vIE5lYXItYm90dG9tIGhhbmRsaW5nIC0gZW5zdXJlIHdlIHNob3cgdGhlIGFjdHVhbCBsYXN0IGl0ZW1zIGF0IGJvdHRvbVxuICAgICAgICBjb25zdCBtYXhTY3JvbGwgPSB2aXJ0dWFsU2l6ZSAtIGNvbnRhaW5lclNpemU7XG4gICAgICAgIGNvbnN0IGRpc3RhbmNlRnJvbUJvdHRvbSA9IG1heFNjcm9sbCAtIHNjcm9sbFBvc2l0aW9uO1xuXG4gICAgICAgIGlmIChkaXN0YW5jZUZyb21Cb3R0b20gPD0gY29udGFpbmVyU2l6ZSAmJiBkaXN0YW5jZUZyb21Cb3R0b20gPj0gLTEpIHtcbiAgICAgICAgICBjb25zdCBpdGVtc0F0Qm90dG9tID0gTWF0aC5mbG9vcihjb250YWluZXJTaXplIC8gaXRlbVNpemUpO1xuICAgICAgICAgIGNvbnN0IGZpcnN0VmlzaWJsZUF0Qm90dG9tID0gTWF0aC5tYXgoMCwgdG90YWxJdGVtcyAtIGl0ZW1zQXRCb3R0b20pO1xuICAgICAgICAgIGNvbnN0IGludGVycG9sYXRpb24gPSBNYXRoLm1heChcbiAgICAgICAgICAgIDAsXG4gICAgICAgICAgICBNYXRoLm1pbigxLCAxIC0gZGlzdGFuY2VGcm9tQm90dG9tIC8gY29udGFpbmVyU2l6ZSksXG4gICAgICAgICAgKTtcblxuICAgICAgICAgIHN0YXJ0ID0gTWF0aC5mbG9vcihcbiAgICAgICAgICAgIHN0YXJ0ICsgKGZpcnN0VmlzaWJsZUF0Qm90dG9tIC0gc3RhcnQpICogaW50ZXJwb2xhdGlvbixcbiAgICAgICAgICApO1xuICAgICAgICAgIGVuZCA9XG4gICAgICAgICAgICBkaXN0YW5jZUZyb21Cb3R0b20gPD0gMVxuICAgICAgICAgICAgICA/IHRvdGFsSXRlbXMgLSAxXG4gICAgICAgICAgICAgIDogTWF0aC5taW4odG90YWxJdGVtcyAtIDEsIHN0YXJ0ICsgdmlzaWJsZUNvdW50IC0gMSk7XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIERpcmVjdCBjYWxjdWxhdGlvblxuICAgICAgICBzdGFydCA9IE1hdGguZmxvb3Ioc2Nyb2xsUG9zaXRpb24gLyBpdGVtU2l6ZSk7XG4gICAgICAgIGVuZCA9IE1hdGgubWluKHRvdGFsSXRlbXMgLSAxLCBzdGFydCArIHZpc2libGVDb3VudCAtIDEpO1xuICAgICAgfVxuXG4gICAgICAvLyBFbnN1cmUgdmFsaWQgcmFuZ2VcbiAgICAgIHN0YXJ0ID0gTWF0aC5tYXgoMCwgc3RhcnQpO1xuICAgICAgZW5kID0gTWF0aC5tYXgoc3RhcnQsIGVuZCk7XG5cbiAgICAgIHJldHVybiB7IHN0YXJ0LCBlbmQgfTtcbiAgICB9O1xuXG4gICAgLy8gVXBkYXRlIGZ1bmN0aW9uc1xuICAgIGNvbnN0IHVwZGF0ZVZpc2libGVSYW5nZSA9IChzY3JvbGxQb3NpdGlvbjogbnVtYmVyKSA9PiB7XG4gICAgICBpZiAoIXZpZXdwb3J0U3RhdGUpIHJldHVybjtcblxuICAgICAgLy8gU2tpcCBpZiBzY3JvbGwgcG9zaXRpb24gaGFzbid0IGNoYW5nZWQgc2lnbmlmaWNhbnRseSAod2l0aGluIDFweClcbiAgICAgIGlmIChcbiAgICAgICAgbGFzdFNjcm9sbFBvc2l0aW9uID49IDAgJiZcbiAgICAgICAgTWF0aC5hYnMoc2Nyb2xsUG9zaXRpb24gLSBsYXN0U2Nyb2xsUG9zaXRpb24pIDwgMVxuICAgICAgKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cblxuICAgICAgbGFzdFNjcm9sbFBvc2l0aW9uID0gc2Nyb2xsUG9zaXRpb247XG4gICAgICB2aWV3cG9ydFN0YXRlLnZpc2libGVSYW5nZSA9IGNhbGN1bGF0ZVZpc2libGVSYW5nZShzY3JvbGxQb3NpdGlvbik7XG5cbiAgICAgIC8vIENhbGN1bGF0ZSBhY3R1YWwgdmlzaWJsZSByYW5nZSAod2l0aG91dCBvdmVyc2NhbikgZm9yIFVJIGRpc3BsYXlcbiAgICAgIGNvbnN0IGFjdHVhbFZpc2libGVSYW5nZSA9IGNhbGN1bGF0ZUFjdHVhbFZpc2libGVSYW5nZShzY3JvbGxQb3NpdGlvbik7XG5cbiAgICAgIC8vIFNraXAgZW1pdCBpZiByYW5nZXMgaGF2ZW4ndCBjaGFuZ2VkXG4gICAgICBpZiAoXG4gICAgICAgIGxhc3RWaXNpYmxlUmFuZ2UgJiZcbiAgICAgICAgbGFzdEFjdHVhbFZpc2libGVSYW5nZSAmJlxuICAgICAgICBsYXN0VmlzaWJsZVJhbmdlLnN0YXJ0ID09PSB2aWV3cG9ydFN0YXRlLnZpc2libGVSYW5nZS5zdGFydCAmJlxuICAgICAgICBsYXN0VmlzaWJsZVJhbmdlLmVuZCA9PT0gdmlld3BvcnRTdGF0ZS52aXNpYmxlUmFuZ2UuZW5kICYmXG4gICAgICAgIGxhc3RBY3R1YWxWaXNpYmxlUmFuZ2Uuc3RhcnQgPT09IGFjdHVhbFZpc2libGVSYW5nZS5zdGFydCAmJlxuICAgICAgICBsYXN0QWN0dWFsVmlzaWJsZVJhbmdlLmVuZCA9PT0gYWN0dWFsVmlzaWJsZVJhbmdlLmVuZFxuICAgICAgKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cblxuICAgICAgbGFzdFZpc2libGVSYW5nZSA9IHsgLi4udmlld3BvcnRTdGF0ZS52aXNpYmxlUmFuZ2UgfTtcbiAgICAgIGxhc3RBY3R1YWxWaXNpYmxlUmFuZ2UgPSB7IC4uLmFjdHVhbFZpc2libGVSYW5nZSB9O1xuXG4gICAgICBjb21wb25lbnQuZW1pdD8uKFwidmlld3BvcnQ6cmFuZ2UtY2hhbmdlZFwiLCB7XG4gICAgICAgIHJhbmdlOiB2aWV3cG9ydFN0YXRlLnZpc2libGVSYW5nZSxcbiAgICAgICAgdmlzaWJsZVJhbmdlOiBhY3R1YWxWaXNpYmxlUmFuZ2UsXG4gICAgICAgIHNjcm9sbFBvc2l0aW9uLFxuICAgICAgfSk7XG4gICAgfTtcblxuICAgIC8vIFVwZGF0ZSB0b3RhbCB2aXJ0dWFsIHNpemVcbiAgICBjb25zdCB1cGRhdGVUb3RhbFZpcnR1YWxTaXplID0gKHRvdGFsSXRlbXM6IG51bWJlcikgPT4ge1xuICAgICAgaWYgKCF2aWV3cG9ydFN0YXRlKSByZXR1cm47XG5cbiAgICAgIGNvbnN0IG9sZFNpemUgPSB2aWV3cG9ydFN0YXRlLnZpcnR1YWxUb3RhbFNpemU7XG4gICAgICB2aWV3cG9ydFN0YXRlLnRvdGFsSXRlbXMgPSB0b3RhbEl0ZW1zO1xuICAgICAgY29uc3QgYWN0dWFsU2l6ZSA9IHRvdGFsSXRlbXMgKiB2aWV3cG9ydFN0YXRlLml0ZW1TaXplO1xuXG4gICAgICAvLyBHZXQgcGFkZGluZyBmcm9tIHRoZSBpdGVtcyBjb250YWluZXIgaWYgYXZhaWxhYmxlXG4gICAgICBsZXQgdG90YWxQYWRkaW5nID0gMDtcbiAgICAgIGlmICh2aWV3cG9ydFN0YXRlLml0ZW1zQ29udGFpbmVyKSB7XG4gICAgICAgIGNvbnN0IGNvbXB1dGVkU3R5bGUgPSB3aW5kb3cuZ2V0Q29tcHV0ZWRTdHlsZShcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLml0ZW1zQ29udGFpbmVyLFxuICAgICAgICApO1xuICAgICAgICBjb25zdCBwYWRkaW5nVG9wID0gcGFyc2VGbG9hdChjb21wdXRlZFN0eWxlLnBhZGRpbmdUb3ApIHx8IDA7XG4gICAgICAgIGNvbnN0IHBhZGRpbmdCb3R0b20gPSBwYXJzZUZsb2F0KGNvbXB1dGVkU3R5bGUucGFkZGluZ0JvdHRvbSkgfHwgMDtcbiAgICAgICAgdG90YWxQYWRkaW5nID0gcGFkZGluZ1RvcCArIHBhZGRpbmdCb3R0b207XG4gICAgICB9XG5cbiAgICAgIC8vIEluY2x1ZGUgcGFkZGluZyBpbiB0aGUgdmlydHVhbCBzaXplXG4gICAgICB2aWV3cG9ydFN0YXRlLnZpcnR1YWxUb3RhbFNpemUgPSBNYXRoLm1pbihcbiAgICAgICAgYWN0dWFsU2l6ZSArIHRvdGFsUGFkZGluZyxcbiAgICAgICAgTUFYX1ZJUlRVQUxfU0laRSxcbiAgICAgICk7XG5cbiAgICAgIC8vIFN0cmF0ZWdpYyBsb2cgZm9yIGRlYnVnZ2luZyBnYXAgaXNzdWVcbiAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgLy8gICBgW1ZpcnR1YWxdIFRvdGFsIHNpemUgdXBkYXRlOiBpdGVtcz0ke3RvdGFsSXRlbXN9LCBpdGVtU2l6ZT0ke3ZpZXdwb3J0U3RhdGUuaXRlbVNpemV9cHgsIHBhZGRpbmc9JHt0b3RhbFBhZGRpbmd9cHgsIHZpcnR1YWxTaXplPSR7dmlld3BvcnRTdGF0ZS52aXJ0dWFsVG90YWxTaXplfXB4ICh3YXMgJHtvbGRTaXplfXB4KWBcbiAgICAgIC8vICk7XG5cbiAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDp2aXJ0dWFsLXNpemUtY2hhbmdlZFwiLCB7XG4gICAgICAgIHRvdGFsVmlydHVhbFNpemU6IHZpZXdwb3J0U3RhdGUudmlydHVhbFRvdGFsU2l6ZSxcbiAgICAgICAgdG90YWxJdGVtczogdG90YWxJdGVtcyxcbiAgICAgICAgY29tcHJlc3Npb25SYXRpbzogZ2V0Q29tcHJlc3Npb25SYXRpbygpLFxuICAgICAgfSk7XG4gICAgfTtcblxuICAgIC8vIFVwZGF0ZSBjb250YWluZXIgc2l6ZVxuICAgIGNvbnN0IHVwZGF0ZUNvbnRhaW5lclNpemUgPSAoKSA9PiB7XG4gICAgICBpZiAoIXZpZXdwb3J0U3RhdGUgfHwgIXZpZXdwb3J0U3RhdGUudmlld3BvcnRFbGVtZW50KSByZXR1cm47XG5cbiAgICAgIGNvbnN0IHNpemUgPVxuICAgICAgICB2aWV3cG9ydFN0YXRlLnZpZXdwb3J0RWxlbWVudFtcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLm9yaWVudGF0aW9uID09PSBcImhvcml6b250YWxcIlxuICAgICAgICAgICAgPyBcIm9mZnNldFdpZHRoXCJcbiAgICAgICAgICAgIDogXCJvZmZzZXRIZWlnaHRcIlxuICAgICAgICBdO1xuXG4gICAgICAvLyBMb2cgdGhlIGFjdHVhbCBtZWFzdXJlbWVudFxuICAgICAgLy8gY29uc29sZS5sb2coXG4gICAgICAvLyAgIGBbVmlydHVhbF0gQ29udGFpbmVyIHNpemUgbWVhc3VyZWQ6ICR7c2l6ZX1weCBmcm9tIHZpZXdwb3J0IGVsZW1lbnRgXG4gICAgICAvLyApO1xuXG4gICAgICBpZiAoc2l6ZSAhPT0gdmlld3BvcnRTdGF0ZS5jb250YWluZXJTaXplKSB7XG4gICAgICAgIHZpZXdwb3J0U3RhdGUuY29udGFpbmVyU2l6ZSA9IHNpemU7XG4gICAgICAgIHVwZGF0ZVZpc2libGVSYW5nZSh2aWV3cG9ydFN0YXRlLnNjcm9sbFBvc2l0aW9uIHx8IDApO1xuICAgICAgICB1cGRhdGVUb3RhbFZpcnR1YWxTaXplKHZpZXdwb3J0U3RhdGUudG90YWxJdGVtcyk7IC8vIEFsc28gdXBkYXRlIHZpcnR1YWwgc2l6ZVxuICAgICAgICBjb21wb25lbnQuZW1pdD8uKFwidmlld3BvcnQ6Y29udGFpbmVyLXNpemUtY2hhbmdlZFwiLCB7XG4gICAgICAgICAgY29udGFpbmVyU2l6ZTogc2l6ZSxcbiAgICAgICAgfSk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIE92ZXJyaWRlIHZpZXdwb3J0IG1ldGhvZHNcbiAgICBjb21wb25lbnQudmlld3BvcnQuZ2V0VmlzaWJsZVJhbmdlID0gKCkgPT5cbiAgICAgIHZpZXdwb3J0U3RhdGU/LnZpc2libGVSYW5nZSB8fFxuICAgICAgY2FsY3VsYXRlVmlzaWJsZVJhbmdlKHZpZXdwb3J0U3RhdGU/LnNjcm9sbFBvc2l0aW9uIHx8IDApO1xuXG4gICAgLy8gRXZlbnQgbGlzdGVuZXJzXG4gICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDpzY3JvbGxcIiwgKGRhdGE6IGFueSkgPT5cbiAgICAgIHVwZGF0ZVZpc2libGVSYW5nZShkYXRhLnBvc2l0aW9uKSxcbiAgICApO1xuXG4gICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDppdGVtcy1jaGFuZ2VkXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgIGlmIChkYXRhLnRvdGFsSXRlbXMgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICB1cGRhdGVUb3RhbFZpcnR1YWxTaXplKGRhdGEudG90YWxJdGVtcyk7XG4gICAgICAgIHVwZGF0ZVZpc2libGVSYW5nZSh2aWV3cG9ydFN0YXRlPy5zY3JvbGxQb3NpdGlvbiB8fCAwKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIC8vIExpc3RlbiBmb3IgdG90YWwgaXRlbXMgY2hhbmdlcyAoaW1wb3J0YW50IGZvciBjdXJzb3IgcGFnaW5hdGlvbilcbiAgICAvLyBOb3RlOiBzZXRUb3RhbEl0ZW1zKCkgdXBkYXRlcyB2aWV3cG9ydFN0YXRlLnRvdGFsSXRlbXMgQkVGT1JFIGVtaXR0aW5nIHRoaXMgZXZlbnQsXG4gICAgLy8gc28gd2UgY2FuJ3QgcmVseSBvbiB2aWV3cG9ydFN0YXRlLnRvdGFsSXRlbXMgdG8gZGV0ZWN0IHRoZSBmaXJzdCB0b3RhbC5cbiAgICAvLyBJbnN0ZWFkIHdlIHVzZSB0aGUgaGFzUmVjYWxjdWxhdGVkU2Nyb2xsRm9yQ29tcHJlc3Npb24gZmxhZy5cbiAgICBjb21wb25lbnQub24/LihcInZpZXdwb3J0OnRvdGFsLWl0ZW1zLWNoYW5nZWRcIiwgKGRhdGE6IGFueSkgPT4ge1xuICAgICAgaWYgKGRhdGEudG90YWwgPT09IHVuZGVmaW5lZCkgcmV0dXJuO1xuXG4gICAgICAvLyBGSVg6IFdoZW4gd2UgZmlyc3QgcmVjZWl2ZSB0b3RhbEl0ZW1zIHdpdGggaW5pdGlhbFNjcm9sbEluZGV4IGFuZCBjb21wcmVzc2lvbixcbiAgICAgIC8vIHJlY2FsY3VsYXRlIHNjcm9sbCBwb3NpdGlvbiB0byBhY2NvdW50IGZvciBjb21wcmVzc2lvbi5cbiAgICAgIC8vIFdpdGhvdXQgdGhpcywgaW5pdGlhbFNjcm9sbEluZGV4IGNhbGN1bGF0ZXMgcG9zaXRpb24gYXMgaW5kZXggKiBpdGVtU2l6ZSxcbiAgICAgIC8vIGJ1dCByZW5kZXJpbmcgdXNlcyBjb21wcmVzc2VkIHNwYWNlLCBjYXVzaW5nIGl0ZW1zIHRvIGFwcGVhciBvZmYtc2NyZWVuLlxuICAgICAgLy9cbiAgICAgIC8vIFdlIGNoZWNrIHRoaXMgQkVGT1JFIHVwZGF0ZVRvdGFsVmlydHVhbFNpemUgYmVjYXVzZSB3ZSBuZWVkIHRvIHJlY2FsY3VsYXRlXG4gICAgICAvLyBzY3JvbGwgcG9zaXRpb24gYmFzZWQgb24gdGhlIG5ldyB0b3RhbCwgYW5kIHRoZSBmbGFnIGVuc3VyZXMgd2Ugb25seSBkbyB0aGlzIG9uY2UuXG4gICAgICBpZiAoXG4gICAgICAgIGluaXRpYWxTY3JvbGxJbmRleCA+IDAgJiZcbiAgICAgICAgIWhhc1JlY2FsY3VsYXRlZFNjcm9sbEZvckNvbXByZXNzaW9uICYmXG4gICAgICAgIGRhdGEudG90YWwgPiAwXG4gICAgICApIHtcbiAgICAgICAgY29uc3QgYWN0dWFsVG90YWxTaXplID0gZGF0YS50b3RhbCAqIHZpZXdwb3J0U3RhdGUuaXRlbVNpemU7XG4gICAgICAgIGNvbnN0IGlzQ29tcHJlc3NlZCA9IGFjdHVhbFRvdGFsU2l6ZSA+IE1BWF9WSVJUVUFMX1NJWkU7XG5cbiAgICAgICAgaWYgKGlzQ29tcHJlc3NlZCkge1xuICAgICAgICAgIC8vIFJlY2FsY3VsYXRlIHNjcm9sbCBwb3NpdGlvbiB1c2luZyBjb21wcmVzc2lvbi1hd2FyZSBmb3JtdWxhXG4gICAgICAgICAgLy8gU2FtZSBmb3JtdWxhIHVzZWQgaW4gc2Nyb2xsVG9JbmRleFxuICAgICAgICAgIGNvbnN0IHJhdGlvID0gaW5pdGlhbFNjcm9sbEluZGV4IC8gZGF0YS50b3RhbDtcbiAgICAgICAgICBjb25zdCBjb21wcmVzc2VkUG9zaXRpb24gPSByYXRpbyAqIE1BWF9WSVJUVUFMX1NJWkU7XG5cbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLnNjcm9sbFBvc2l0aW9uID0gY29tcHJlc3NlZFBvc2l0aW9uO1xuXG4gICAgICAgICAgLy8gTm90aWZ5IHNjcm9sbGluZyBmZWF0dXJlIHRvIHN5bmMgaXRzIGxvY2FsIHNjcm9sbCBwb3NpdGlvblxuICAgICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpzY3JvbGwtcG9zaXRpb24tc3luY1wiLCB7XG4gICAgICAgICAgICBwb3NpdGlvbjogY29tcHJlc3NlZFBvc2l0aW9uLFxuICAgICAgICAgICAgc291cmNlOiBcImNvbXByZXNzaW9uLXJlY2FsY3VsYXRpb25cIixcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIC8vIENsZWFyIHRhcmdldFNjcm9sbEluZGV4IHNpbmNlIHdlJ3ZlIG5vdyBjb3JyZWN0bHkgY2FsY3VsYXRlZCB0aGUgc2Nyb2xsIHBvc2l0aW9uXG4gICAgICAgICAgLy8gVGhlIG5vcm1hbCBjb21wcmVzc2VkIHNjcm9sbGluZyBmb3JtdWxhIHdpbGwgbm93IHdvcmsgY29ycmVjdGx5XG4gICAgICAgICAgZGVsZXRlICh2aWV3cG9ydFN0YXRlIGFzIGFueSkudGFyZ2V0U2Nyb2xsSW5kZXg7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBNYXJrIGFzIGRvbmUgZXZlbiBpZiBub3QgY29tcHJlc3NlZCAtIHdlIG9ubHkgd2FudCB0byBkbyB0aGlzIGNoZWNrIG9uY2VcbiAgICAgICAgaGFzUmVjYWxjdWxhdGVkU2Nyb2xsRm9yQ29tcHJlc3Npb24gPSB0cnVlO1xuICAgICAgfVxuXG4gICAgICB1cGRhdGVUb3RhbFZpcnR1YWxTaXplKGRhdGEudG90YWwpO1xuICAgICAgdXBkYXRlVmlzaWJsZVJhbmdlKHZpZXdwb3J0U3RhdGU/LnNjcm9sbFBvc2l0aW9uIHx8IDApO1xuXG4gICAgICAvLyBUcmlnZ2VyIGEgcmVuZGVyIHRvIHVwZGF0ZSB0aGUgdmlld1xuICAgICAgY29tcG9uZW50LnZpZXdwb3J0Py5yZW5kZXJJdGVtcz8uKCk7XG4gICAgfSk7XG5cbiAgICAvLyBMaXN0ZW4gZm9yIGNvbnRhaW5lciBzaXplIGNoYW5nZXMgdG8gcmVjYWxjdWxhdGUgdmlydHVhbCBzaXplXG4gICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDpjb250YWluZXItc2l6ZS1jaGFuZ2VkXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgIGlmICh2aWV3cG9ydFN0YXRlICYmIGRhdGEuY29udGFpbmVyU2l6ZSAhPT0gdmlld3BvcnRTdGF0ZS5jb250YWluZXJTaXplKSB7XG4gICAgICAgIHZpZXdwb3J0U3RhdGUuY29udGFpbmVyU2l6ZSA9IGRhdGEuY29udGFpbmVyU2l6ZTtcbiAgICAgICAgLy8gUmVjYWxjdWxhdGUgdmlydHVhbCBzaXplIHdpdGggbmV3IGNvbnRhaW5lciBzaXplXG4gICAgICAgIHVwZGF0ZVRvdGFsVmlydHVhbFNpemUodmlld3BvcnRTdGF0ZS50b3RhbEl0ZW1zKTtcbiAgICAgICAgdXBkYXRlVmlzaWJsZVJhbmdlKHZpZXdwb3J0U3RhdGUuc2Nyb2xsUG9zaXRpb24gfHwgMCk7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICBjb21wb25lbnQub24/LihcImNvbGxlY3Rpb246cmFuZ2UtbG9hZGVkXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgIGlmIChcbiAgICAgICAgZGF0YS50b3RhbCAhPT0gdW5kZWZpbmVkICYmXG4gICAgICAgIGRhdGEudG90YWwgIT09IHZpZXdwb3J0U3RhdGU/LnRvdGFsSXRlbXNcbiAgICAgICkge1xuICAgICAgICB1cGRhdGVUb3RhbFZpcnR1YWxTaXplKGRhdGEudG90YWwpO1xuICAgICAgfVxuXG4gICAgICAvLyBGb3JjZSB2aXNpYmxlIHJhbmdlIHJlY2FsY3VsYXRpb24gYWZ0ZXIgZGF0YSBsb2Fkc1xuICAgICAgLy8gVGhpcyBpcyBjcml0aWNhbDogdG90YWxJdGVtcyBtYXkgaGF2ZSBiZWVuIHVwZGF0ZWQgYnkgY29sbGVjdGlvbi5zZXRUb3RhbEl0ZW1zKClcbiAgICAgIC8vIHdoaWNoIGJ5cGFzc2VzIHVwZGF0ZVRvdGFsVmlydHVhbFNpemUsIHNvIGxhc3RTY3JvbGxQb3NpdGlvbiB3YXNuJ3QgcmVzZXQuXG4gICAgICAvLyBXZSBtdXN0IHJlY2FsY3VsYXRlIHRvIGVuc3VyZSB2aXNpYmxlUmFuZ2UgcmVmbGVjdHMgdGhlIG5ldyB0b3RhbEl0ZW1zLlxuICAgICAgbGFzdFNjcm9sbFBvc2l0aW9uID0gLTE7XG5cbiAgICAgIHVwZGF0ZVZpc2libGVSYW5nZSh2aWV3cG9ydFN0YXRlPy5zY3JvbGxQb3NpdGlvbiB8fCAwKTtcbiAgICAgIGNvbXBvbmVudC52aWV3cG9ydD8ucmVuZGVySXRlbXM/LigpO1xuICAgIH0pO1xuXG4gICAgLy8gTGlzdGVuIGZvciBmaXJzdCBpdGVtcyByZW5kZXJlZCB0byBjYWxjdWxhdGUgaXRlbSBzaXplIChpZiBhdXRvLWRldGVjdGlvbiBpcyBlbmFibGVkKVxuICAgIGlmIChhdXRvRGV0ZWN0SXRlbVNpemUpIHtcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwidmlld3BvcnQ6aXRlbXMtcmVuZGVyZWRcIiwgKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgIWhhc0NhbGN1bGF0ZWRJdGVtU2l6ZSAmJlxuICAgICAgICAgIGRhdGEuZWxlbWVudHM/Lmxlbmd0aCA+IDAgJiZcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlXG4gICAgICAgICkge1xuICAgICAgICAgIC8vIENhbGN1bGF0ZSBhdmVyYWdlIGl0ZW0gc2l6ZSBmcm9tIGZpcnN0IHJlbmRlcmVkIGJhdGNoXG4gICAgICAgICAgY29uc3Qgc2l6ZXM6IG51bWJlcltdID0gW107XG4gICAgICAgICAgY29uc3Qgc2l6ZVByb3BlcnR5ID1cbiAgICAgICAgICAgIG9yaWVudGF0aW9uID09PSBcImhvcml6b250YWxcIiA/IFwib2Zmc2V0V2lkdGhcIiA6IFwib2Zmc2V0SGVpZ2h0XCI7XG5cbiAgICAgICAgICBkYXRhLmVsZW1lbnRzLmZvckVhY2goKGVsZW1lbnQ6IEhUTUxFbGVtZW50KSA9PiB7XG4gICAgICAgICAgICBjb25zdCBzaXplID0gZWxlbWVudFtzaXplUHJvcGVydHldO1xuICAgICAgICAgICAgaWYgKHNpemUgPiAwKSB7XG4gICAgICAgICAgICAgIHNpemVzLnB1c2goc2l6ZSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICBpZiAoc2l6ZXMubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgY29uc3QgYXZnU2l6ZSA9IE1hdGgucm91bmQoXG4gICAgICAgICAgICAgIHNpemVzLnJlZHVjZSgoc3VtLCBzaXplKSA9PiBzdW0gKyBzaXplLCAwKSAvIHNpemVzLmxlbmd0aCxcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgICBjb25zdCBwcmV2aW91c0l0ZW1TaXplID0gdmlld3BvcnRTdGF0ZS5pdGVtU2l6ZTtcbiAgICAgICAgICAgIHZpZXdwb3J0U3RhdGUuaXRlbVNpemUgPSBhdmdTaXplO1xuXG4gICAgICAgICAgICAvLyBJZiB3ZSBoYXZlIGFuIGluaXRpYWxTY3JvbGxJbmRleCwgcmVjYWxjdWxhdGUgc2Nyb2xsIHBvc2l0aW9uXG4gICAgICAgICAgICAvLyBiYXNlZCBvbiB0aGUgbmV3bHkgZGV0ZWN0ZWQgaXRlbSBzaXplXG4gICAgICAgICAgICBpZiAoaW5pdGlhbFNjcm9sbEluZGV4ID4gMCAmJiBhdmdTaXplICE9PSBwcmV2aW91c0l0ZW1TaXplKSB7XG4gICAgICAgICAgICAgIGNvbnN0IG5ld1Njcm9sbFBvc2l0aW9uID0gaW5pdGlhbFNjcm9sbEluZGV4ICogYXZnU2l6ZTtcbiAgICAgICAgICAgICAgdmlld3BvcnRTdGF0ZS5zY3JvbGxQb3NpdGlvbiA9IG5ld1Njcm9sbFBvc2l0aW9uO1xuXG4gICAgICAgICAgICAgIC8vIE5vdGlmeSBzY3JvbGxpbmcgZmVhdHVyZSB0byBzeW5jIGl0cyBsb2NhbCBzY3JvbGwgcG9zaXRpb25cbiAgICAgICAgICAgICAgY29tcG9uZW50LmVtaXQ/LihcInZpZXdwb3J0OnNjcm9sbC1wb3NpdGlvbi1zeW5jXCIsIHtcbiAgICAgICAgICAgICAgICBwb3NpdGlvbjogbmV3U2Nyb2xsUG9zaXRpb24sXG4gICAgICAgICAgICAgICAgc291cmNlOiBcIml0ZW0tc2l6ZS1kZXRlY3RlZFwiLFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gUmVjYWxjdWxhdGUgZXZlcnl0aGluZyB3aXRoIG5ldyBzaXplXG4gICAgICAgICAgICB1cGRhdGVUb3RhbFZpcnR1YWxTaXplKHZpZXdwb3J0U3RhdGUudG90YWxJdGVtcyk7XG4gICAgICAgICAgICB1cGRhdGVWaXNpYmxlUmFuZ2Uodmlld3BvcnRTdGF0ZS5zY3JvbGxQb3NpdGlvbiB8fCAwKTtcblxuICAgICAgICAgICAgLy8gUmUtcmVuZGVyIHRvIGFkanVzdCBwb3NpdGlvbnNcbiAgICAgICAgICAgIGNvbXBvbmVudC52aWV3cG9ydD8ucmVuZGVySXRlbXM/LigpO1xuXG4gICAgICAgICAgICAvLyBFbWl0IGV2ZW50IGZvciBzaXplIGNoYW5nZVxuICAgICAgICAgICAgY29tcG9uZW50LmVtaXQ/LihcInZpZXdwb3J0Oml0ZW0tc2l6ZS1kZXRlY3RlZFwiLCB7XG4gICAgICAgICAgICAgIHByZXZpb3VzU2l6ZTogaW5pdGlhbEl0ZW1TaXplLFxuICAgICAgICAgICAgICBkZXRlY3RlZFNpemU6IGF2Z1NpemUsXG4gICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgaGFzQ2FsY3VsYXRlZEl0ZW1TaXplID0gdHJ1ZTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH1cblxuICAgIC8vIENsZWFuIHVwIFJlc2l6ZU9ic2VydmVyIG9uIGRlc3Ryb3lcbiAgICB3cmFwRGVzdHJveShjb21wb25lbnQsICgpID0+IHtcbiAgICAgIGlmIChyZXNpemVPYnNlcnZlcikge1xuICAgICAgICByZXNpemVPYnNlcnZlci5kaXNjb25uZWN0KCk7XG4gICAgICAgIHJlc2l6ZU9ic2VydmVyID0gbnVsbDtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIC8vIEV4cG9zZSB2aXJ0dWFsIEFQSVxuICAgIGNvbnN0IGNvbXByZXNzaW9uUmF0aW8gPSBnZXRDb21wcmVzc2lvblJhdGlvKCk7XG4gICAgKGNvbXBvbmVudCBhcyBhbnkpLnZpcnR1YWwgPSB7XG4gICAgICBjYWxjdWxhdGVWaXNpYmxlUmFuZ2UsXG4gICAgICB1cGRhdGVUb3RhbFZpcnR1YWxTaXplLFxuICAgICAgZ2V0VG90YWxWaXJ0dWFsU2l6ZTogKCkgPT4gdmlld3BvcnRTdGF0ZT8udmlydHVhbFRvdGFsU2l6ZSB8fCAwLFxuICAgICAgZ2V0Q29udGFpbmVyU2l6ZTogKCkgPT4gdmlld3BvcnRTdGF0ZT8uY29udGFpbmVyU2l6ZSB8fCAwLFxuICAgICAgdXBkYXRlQ29udGFpbmVyU2l6ZTogKHNpemU6IG51bWJlcikgPT4ge1xuICAgICAgICBpZiAodmlld3BvcnRTdGF0ZSkge1xuICAgICAgICAgIHZpZXdwb3J0U3RhdGUuY29udGFpbmVyU2l6ZSA9IHNpemU7XG4gICAgICAgICAgdXBkYXRlVmlzaWJsZVJhbmdlKHZpZXdwb3J0U3RhdGUuc2Nyb2xsUG9zaXRpb24gfHwgMCk7XG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBnZXRJdGVtU2l6ZTogKCkgPT4gdmlld3BvcnRTdGF0ZT8uaXRlbVNpemUgfHwgaW5pdGlhbEl0ZW1TaXplLFxuICAgICAgY2FsY3VsYXRlSW5kZXhGcm9tUG9zaXRpb246IChwb3NpdGlvbjogbnVtYmVyKSA9PlxuICAgICAgICBNYXRoLmZsb29yKFxuICAgICAgICAgIHBvc2l0aW9uIC9cbiAgICAgICAgICAgICgodmlld3BvcnRTdGF0ZT8uaXRlbVNpemUgfHwgaW5pdGlhbEl0ZW1TaXplKSAqIGNvbXByZXNzaW9uUmF0aW8pLFxuICAgICAgICApLFxuICAgICAgY2FsY3VsYXRlUG9zaXRpb25Gb3JJbmRleDogKGluZGV4OiBudW1iZXIpID0+XG4gICAgICAgIGluZGV4ICogKHZpZXdwb3J0U3RhdGU/Lml0ZW1TaXplIHx8IGluaXRpYWxJdGVtU2l6ZSkgKiBjb21wcmVzc2lvblJhdGlvLFxuICAgIH07XG5cbiAgICByZXR1cm4gY29tcG9uZW50O1xuICB9O1xufTtcbiIsCiAgICAiLyoqXG4gKiBTY3JvbGxpbmcgRmVhdHVyZSAtIFZpcnR1YWwgc2Nyb2xsaW5nIHdpdGggaW50ZWdyYXRlZCB2ZWxvY2l0eSB0cmFja2luZ1xuICogSGFuZGxlcyB3aGVlbCBldmVudHMsIHRvdWNoL21vdXNlIGV2ZW50cywgc2Nyb2xsIHBvc2l0aW9uIG1hbmFnZW1lbnQsIHZlbG9jaXR5IG1lYXN1cmVtZW50LCBhbmQgbW9tZW50dW0gc2Nyb2xsaW5nXG4gKi9cblxuaW1wb3J0IHsgUFJFRklYIH0gZnJvbSBcIm10cmxcIjtcbmltcG9ydCB0eXBlIHsgVmlld3BvcnRDb250ZXh0LCBWaWV3cG9ydENvbXBvbmVudCB9IGZyb20gXCIuLi90eXBlc1wiO1xuaW1wb3J0IHsgVklFV1BPUlRfQ09OU1RBTlRTIH0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgd3JhcEluaXRpYWxpemUsIGdldFZpZXdwb3J0U3RhdGUsIGNsYW1wIH0gZnJvbSBcIi4vdXRpbHNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBTY3JvbGxpbmdDb25maWcge1xuICBvcmllbnRhdGlvbj86IFwidmVydGljYWxcIiB8IFwiaG9yaXpvbnRhbFwiO1xuICBzZW5zaXRpdml0eT86IG51bWJlcjtcbiAgc21vb3RoaW5nPzogYm9vbGVhbjtcbiAgaWRsZVRpbWVvdXQ/OiBudW1iZXI7XG4gIC8qKiBTdG9wIHNjcm9sbGluZyB3aGVuIGNsaWNraW5nIG9uIHRoZSB2aWV3cG9ydCAoZGVmYXVsdDogdHJ1ZSkgKi9cbiAgc3RvcE9uQ2xpY2s/OiBib29sZWFuO1xufVxuXG4vLyBTcGVlZCB0cmFja2VyIGludGVyZmFjZVxuaW50ZXJmYWNlIFNwZWVkVHJhY2tlciB7XG4gIHZlbG9jaXR5OiBudW1iZXI7XG4gIGxhc3RQb3NpdGlvbjogbnVtYmVyO1xuICBsYXN0VGltZTogbnVtYmVyO1xuICBkaXJlY3Rpb246IFwiZm9yd2FyZFwiIHwgXCJiYWNrd2FyZFwiO1xuICBzYW1wbGVzOiBBcnJheTx7IHBvc2l0aW9uOiBudW1iZXI7IHRpbWU6IG51bWJlciB9Pjtcbn1cblxuLyoqXG4gKiBDcmVhdGVzIGEgbmV3IHNwZWVkIHRyYWNrZXJcbiAqL1xuY29uc3QgY3JlYXRlU3BlZWRUcmFja2VyID0gKCk6IFNwZWVkVHJhY2tlciA9PiAoe1xuICB2ZWxvY2l0eTogMCxcbiAgbGFzdFBvc2l0aW9uOiAwLFxuICBsYXN0VGltZTogRGF0ZS5ub3coKSxcbiAgZGlyZWN0aW9uOiBcImZvcndhcmRcIixcbiAgc2FtcGxlczogW10sXG59KTtcblxuLyoqXG4gKiBVcGRhdGVzIHNwZWVkIHRyYWNrZXIgd2l0aCBuZXcgcG9zaXRpb25cbiAqL1xuY29uc3QgdXBkYXRlU3BlZWRUcmFja2VyID0gKFxuICB0cmFja2VyOiBTcGVlZFRyYWNrZXIsXG4gIG5ld1Bvc2l0aW9uOiBudW1iZXIsXG4gIHByZXZpb3VzUG9zaXRpb246IG51bWJlcixcbik6IFNwZWVkVHJhY2tlciA9PiB7XG4gIGNvbnN0IG5vdyA9IERhdGUubm93KCk7XG4gIGNvbnN0IHRpbWVEZWx0YSA9IG5vdyAtIHRyYWNrZXIubGFzdFRpbWU7XG5cbiAgaWYgKHRpbWVEZWx0YSA9PT0gMCkgcmV0dXJuIHRyYWNrZXI7XG5cbiAgY29uc3QgcG9zaXRpb25EZWx0YSA9IG5ld1Bvc2l0aW9uIC0gcHJldmlvdXNQb3NpdGlvbjtcbiAgLy8gS2VlcCB2ZWxvY2l0eSBzaWduZWQgKHBvc2l0aXZlID0gZm9yd2FyZC9kb3duLCBuZWdhdGl2ZSA9IGJhY2t3YXJkL3VwKVxuICBjb25zdCBpbnN0YW50VmVsb2NpdHkgPSBwb3NpdGlvbkRlbHRhIC8gdGltZURlbHRhO1xuXG4gIC8vIEFkZCBuZXcgc2FtcGxlXG4gIGNvbnN0IHNhbXBsZXMgPSBbLi4udHJhY2tlci5zYW1wbGVzLCB7IHBvc2l0aW9uOiBuZXdQb3NpdGlvbiwgdGltZTogbm93IH1dO1xuXG4gIC8vIEtlZXAgb25seSByZWNlbnQgc2FtcGxlcyAobGFzdCAxMDBtcylcbiAgY29uc3QgcmVjZW50U2FtcGxlcyA9IHNhbXBsZXMuZmlsdGVyKChzKSA9PiBub3cgLSBzLnRpbWUgPCAxMDApO1xuXG4gIC8vIENhbGN1bGF0ZSBhdmVyYWdlIHZlbG9jaXR5IGZyb20gcmVjZW50IHNhbXBsZXMgKHByZXNlcnZpbmcgc2lnbi9kaXJlY3Rpb24pXG4gIGxldCBhdmdWZWxvY2l0eSA9IGluc3RhbnRWZWxvY2l0eTtcbiAgaWYgKHJlY2VudFNhbXBsZXMubGVuZ3RoID4gMSkge1xuICAgIGNvbnN0IG9sZGVzdFNhbXBsZSA9IHJlY2VudFNhbXBsZXNbMF07XG4gICAgY29uc3QgdG90YWxEaXN0YW5jZSA9IG5ld1Bvc2l0aW9uIC0gb2xkZXN0U2FtcGxlLnBvc2l0aW9uOyAvLyBTaWduZWQgZGlzdGFuY2VcbiAgICBjb25zdCB0b3RhbFRpbWUgPSBub3cgLSBvbGRlc3RTYW1wbGUudGltZTtcbiAgICBhdmdWZWxvY2l0eSA9IHRvdGFsVGltZSA+IDAgPyB0b3RhbERpc3RhbmNlIC8gdG90YWxUaW1lIDogaW5zdGFudFZlbG9jaXR5O1xuICB9XG5cbiAgcmV0dXJuIHtcbiAgICB2ZWxvY2l0eTogYXZnVmVsb2NpdHksXG4gICAgbGFzdFBvc2l0aW9uOiBuZXdQb3NpdGlvbixcbiAgICBsYXN0VGltZTogbm93LFxuICAgIGRpcmVjdGlvbjogcG9zaXRpb25EZWx0YSA+PSAwID8gXCJmb3J3YXJkXCIgOiBcImJhY2t3YXJkXCIsXG4gICAgc2FtcGxlczogcmVjZW50U2FtcGxlcyxcbiAgfTtcbn07XG5cbi8qKlxuICogU2Nyb2xsaW5nIGZlYXR1cmUgZm9yIHZpZXdwb3J0XG4gKiBIYW5kbGVzIHdoZWVsIGV2ZW50cywgdmVsb2NpdHkgdHJhY2tpbmcsIGFuZCBpZGxlIGRldGVjdGlvblxuICovXG5leHBvcnQgY29uc3Qgd2l0aFNjcm9sbGluZyA9IChjb25maWc6IFNjcm9sbGluZ0NvbmZpZyA9IHt9KSA9PiB7XG4gIHJldHVybiA8VCBleHRlbmRzIFZpZXdwb3J0Q29udGV4dCAmIFZpZXdwb3J0Q29tcG9uZW50Pihjb21wb25lbnQ6IFQpOiBUID0+IHtcbiAgICBjb25zdCB7XG4gICAgICBvcmllbnRhdGlvbiA9IFwidmVydGljYWxcIixcbiAgICAgIHNlbnNpdGl2aXR5ID0gVklFV1BPUlRfQ09OU1RBTlRTLlZJUlRVQUxfU0NST0xMLlNDUk9MTF9TRU5TSVRJVklUWSxcbiAgICAgIHNtb290aGluZyA9IGZhbHNlLFxuICAgICAgaWRsZVRpbWVvdXQgPSAxMDAsIC8vIERlZmF1bHQgaWRsZSB0aW1lb3V0IGluIG1zXG4gICAgICBzdG9wT25DbGljayA9IHRydWUsIC8vIFN0b3Agc2Nyb2xsaW5nIG9uIGNsaWNrIGJ5IGRlZmF1bHRcbiAgICB9ID0gY29uZmlnO1xuXG4gICAgLy8gU3RhdGVcbiAgICBsZXQgc2Nyb2xsUG9zaXRpb24gPSAwO1xuICAgIGxldCB0b3RhbFZpcnR1YWxTaXplID0gMDtcbiAgICBsZXQgY29udGFpbmVyU2l6ZSA9IDA7XG4gICAgbGV0IGlzU2Nyb2xsaW5nID0gZmFsc2U7XG4gICAgbGV0IGlzU2Nyb2xsZWRGcm9tVG9wID0gZmFsc2U7IC8vIFRyYWNrIGlmIHNjcm9sbGVkIGF3YXkgZnJvbSB0b3BcbiAgICBsZXQgbGFzdFNjcm9sbFRpbWUgPSAwO1xuXG4gICAgLyoqXG4gICAgICogSGVscGVyIHRvIHJlbW92ZSB0aGUgLS1zY3JvbGxlZCBjbGFzcyBmcm9tIHRoZSB2bGlzdCBjb250YWluZXJcbiAgICAgKi9cbiAgICBjb25zdCByZW1vdmVTY3JvbGxlZENsYXNzID0gKCkgPT4ge1xuICAgICAgY29uc3Qgdmlld3BvcnRFbCA9XG4gICAgICAgIChjb21wb25lbnQgYXMgYW55KS52aWV3cG9ydEVsZW1lbnQgfHxcbiAgICAgICAgKGNvbXBvbmVudCBhcyBhbnkpLl9zY3JvbGxpbmdWaWV3cG9ydEVsZW1lbnQ7XG4gICAgICBpZiAodmlld3BvcnRFbCkge1xuICAgICAgICBjb25zdCB2bGlzdENvbnRhaW5lciA9IHZpZXdwb3J0RWwuY2xvc2VzdChgLiR7UFJFRklYfS12bGlzdGApO1xuICAgICAgICBpZiAodmxpc3RDb250YWluZXIpIHtcbiAgICAgICAgICB2bGlzdENvbnRhaW5lci5jbGFzc0xpc3QucmVtb3ZlKGAke1BSRUZJWH0tdmxpc3QtLXNjcm9sbGVkYCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9O1xuXG4gICAgLy8gTGlzdGVuIGZvciByZWxvYWQ6c3RhcnQgdG8gcmVzZXQgZmVhdHVyZS1zcGVjaWZpYyBzdGF0ZVxuICAgIGNvbXBvbmVudC5vbj8uKFwicmVsb2FkOnN0YXJ0XCIsICgpID0+IHtcbiAgICAgIHNjcm9sbFBvc2l0aW9uID0gMDtcbiAgICAgIHRvdGFsVmlydHVhbFNpemUgPSAwO1xuICAgICAgaXNTY3JvbGxpbmcgPSBmYWxzZTtcbiAgICAgIGlzU2Nyb2xsZWRGcm9tVG9wID0gZmFsc2U7XG4gICAgICBsYXN0U2Nyb2xsVGltZSA9IDA7XG4gICAgICBzcGVlZFRyYWNrZXIgPSBjcmVhdGVTcGVlZFRyYWNrZXIoKTtcbiAgICAgIGhhc0VtaXR0ZWRJZGxlID0gZmFsc2U7XG4gICAgICBhbmNob3JQb3NpdGlvbiA9IG51bGw7XG4gICAgICBhbmNob3JUaW1lID0gMDtcbiAgICAgIGxhc3RXaGVlbFRpbWUgPSAwO1xuICAgICAgYW5jaG9ySW5pdGlhbERlbHRhID0gMDtcbiAgICAgIGFuY2hvckxhc3REZWx0YSA9IDA7XG4gICAgICBhbmNob3JNaW5EZWx0YSA9IEluZmluaXR5O1xuICAgICAgY29uc2VjdXRpdmVJbmNyZWFzZXMgPSAwO1xuICAgICAgc3VzdGFpbmVkSGlnaENvdW50ID0gMDtcbiAgICAgIHJlbmRlclNjaGVkdWxlZCA9IGZhbHNlO1xuICAgICAgcGVuZGluZ1Njcm9sbERhdGEgPSBudWxsO1xuXG4gICAgICAvLyBDbGVhciBhbnkgcGVuZGluZyB0aW1lb3V0c1xuICAgICAgaWYgKGlkbGVUaW1lb3V0SWQpIHtcbiAgICAgICAgY2xlYXJUaW1lb3V0KGlkbGVUaW1lb3V0SWQpO1xuICAgICAgICBpZGxlVGltZW91dElkID0gbnVsbDtcbiAgICAgIH1cbiAgICAgIHN0b3BJZGxlRGV0ZWN0aW9uKCk7XG5cbiAgICAgIC8vIFJlbW92ZSAtLXNjcm9sbGVkIGNsYXNzIGZyb20gRE9NIHNpbmNlIHdlJ3JlIHJlc2V0dGluZyB0byB0b3BcbiAgICAgIHJlbW92ZVNjcm9sbGVkQ2xhc3MoKTtcbiAgICB9KTtcblxuICAgIC8vIExpc3RlbiBmb3IgY2xlYXJlZCBldmVudCAoZnJvbSB2bGlzdC5jbGVhcigpKSB0byByZXNldCBzY3JvbGxlZCBzdGF0ZVxuICAgIGNvbXBvbmVudC5vbj8uKFwiY2xlYXJlZFwiLCAoKSA9PiB7XG4gICAgICBzY3JvbGxQb3NpdGlvbiA9IDA7XG4gICAgICB0b3RhbFZpcnR1YWxTaXplID0gMDtcbiAgICAgIGlzU2Nyb2xsZWRGcm9tVG9wID0gZmFsc2U7XG4gICAgICByZW1vdmVTY3JvbGxlZENsYXNzKCk7XG4gICAgfSk7XG4gICAgbGV0IHNwZWVkVHJhY2tlciA9IGNyZWF0ZVNwZWVkVHJhY2tlcigpO1xuICAgIGxldCBpZGxlVGltZW91dElkOiBudW1iZXIgfCBudWxsID0gbnVsbDtcbiAgICBsZXQgaWRsZUNoZWNrRnJhbWU6IG51bWJlciB8IG51bGwgPSBudWxsO1xuICAgIGxldCBsYXN0SWRsZUNoZWNrUG9zaXRpb24gPSAwO1xuICAgIGxldCBoYXNFbWl0dGVkSWRsZSA9IGZhbHNlOyAvLyBUcmFjayBpZiB3ZSd2ZSBhbHJlYWR5IGVtaXR0ZWQgaWRsZVxuICAgIGxldCBhbmNob3JQb3NpdGlvbjogbnVtYmVyIHwgbnVsbCA9IG51bGw7IC8vIEFuY2hvciBwb3NpdGlvbiB0byBtYWludGFpbiBvbiBjbGljayAoZm9yIHN0b3BPbkNsaWNrKVxuICAgIGxldCBhbmNob3JUaW1lID0gMDsgLy8gVGltZXN0YW1wIHdoZW4gYW5jaG9yIHdhcyBzZXRcbiAgICBsZXQgbGFzdFdoZWVsVGltZSA9IDA7IC8vIFRpbWVzdGFtcCBvZiBsYXN0IHdoZWVsIGV2ZW50IChmb3IgZGV0ZWN0aW5nIGdhcHMpXG4gICAgbGV0IGFuY2hvckluaXRpYWxEZWx0YSA9IDA7IC8vIEluaXRpYWwgd2hlZWwgZGVsdGEgd2hlbiBhbmNob3Igd2FzIHNldCAodG8gZGV0ZWN0IGRlY2F5KVxuICAgIGxldCBhbmNob3JMYXN0RGVsdGEgPSAwOyAvLyBQcmV2aW91cyB3aGVlbCBkZWx0YSAodG8gZGV0ZWN0IGluY3JlYXNpbmcgZGVsdGEgPSBuZXcgc2Nyb2xsKVxuICAgIGxldCBhbmNob3JNaW5EZWx0YSA9IEluZmluaXR5OyAvLyBNaW5pbXVtIGRlbHRhIHNlZW4gc2luY2UgYW5jaG9yICh0byBkZXRlY3QgcmVhY2NlbGVyYXRpb24pXG4gICAgbGV0IGNvbnNlY3V0aXZlSW5jcmVhc2VzID0gMDsgLy8gQ291bnQgb2YgY29uc2VjdXRpdmUgZGVsdGEgaW5jcmVhc2VzICh0byBkZXRlY3Qgc3VzdGFpbmVkIG5ldyBzY3JvbGxpbmcpXG4gICAgbGV0IHN1c3RhaW5lZEhpZ2hDb3VudCA9IDA7IC8vIENvdW50IG9mIGV2ZW50cyB3aGVyZSBkZWx0YSBpcyBzaWduaWZpY2FudGx5IGFib3ZlIG1pbmltdW0gKHN1c3RhaW5lZCBuZXcgc2Nyb2xsaW5nKVxuXG4gICAgLy8gUkFGLWJhc2VkIHJlbmRlciB0aHJvdHRsaW5nIC0gcHJldmVudHMgd2FzdGVkIHJlbmRlcnMgZHVyaW5nIGZhc3Qgc2Nyb2xsaW5nXG4gICAgbGV0IHJlbmRlclNjaGVkdWxlZCA9IGZhbHNlO1xuICAgIGxldCBwZW5kaW5nU2Nyb2xsRGF0YToge1xuICAgICAgcG9zaXRpb246IG51bWJlcjtcbiAgICAgIGRpcmVjdGlvbjogXCJmb3J3YXJkXCIgfCBcImJhY2t3YXJkXCI7XG4gICAgICBwcmV2aW91c1Bvc2l0aW9uOiBudW1iZXI7XG4gICAgfSB8IG51bGwgPSBudWxsO1xuXG4gICAgLy8gY29uc29sZS5sb2coYFtTY3JvbGxpbmddIEluaXRpYWwgc3RhdGUgLSBwb3NpdGlvbjogJHtzY3JvbGxQb3NpdGlvbn1gKTtcblxuICAgIC8vIFRocmVzaG9sZCBmb3IgY29uc2lkZXJpbmcgdGhlIGxpc3QgXCJzY3JvbGxlZFwiIChpbiBwaXhlbHMpXG4gICAgY29uc3QgU0NST0xMRURfVEhSRVNIT0xEID0gMTtcblxuICAgIC8qKlxuICAgICAqIFVwZGF0ZXMgdGhlIHNjcm9sbGVkIHN0YXRlIGNsYXNzIG9uIHRoZSB2bGlzdCBjb250YWluZXJcbiAgICAgKiBBZGRzIC0tc2Nyb2xsZWQgbW9kaWZpZXIgd2hlbiBsaXN0IGlzIG5vdCBhdCB0aGUgdG9wXG4gICAgICovXG4gICAgY29uc3QgdXBkYXRlU2Nyb2xsZWRTdGF0ZSA9IChcbiAgICAgIHZpZXdwb3J0RWxlbWVudDogSFRNTEVsZW1lbnQsXG4gICAgICBwb3NpdGlvbjogbnVtYmVyLFxuICAgICkgPT4ge1xuICAgICAgY29uc3Qgc2hvdWxkQmVTY3JvbGxlZCA9IHBvc2l0aW9uID4gU0NST0xMRURfVEhSRVNIT0xEO1xuICAgICAgaWYgKHNob3VsZEJlU2Nyb2xsZWQgIT09IGlzU2Nyb2xsZWRGcm9tVG9wKSB7XG4gICAgICAgIGlzU2Nyb2xsZWRGcm9tVG9wID0gc2hvdWxkQmVTY3JvbGxlZDtcbiAgICAgICAgLy8gRmluZCB0aGUgdmxpc3QgY29udGFpbmVyIChwYXJlbnQgb2Ygdmlld3BvcnQpXG4gICAgICAgIGNvbnN0IHZsaXN0Q29udGFpbmVyID0gdmlld3BvcnRFbGVtZW50LmNsb3Nlc3QoYC4ke1BSRUZJWH0tdmxpc3RgKTtcbiAgICAgICAgaWYgKHZsaXN0Q29udGFpbmVyKSB7XG4gICAgICAgICAgdmxpc3RDb250YWluZXIuY2xhc3NMaXN0LnRvZ2dsZShcbiAgICAgICAgICAgIGAke1BSRUZJWH0tdmxpc3QtLXNjcm9sbGVkYCxcbiAgICAgICAgICAgIHNob3VsZEJlU2Nyb2xsZWQsXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH07XG5cbiAgICAvLyBHZXQgdmlld3BvcnQgc3RhdGVcbiAgICBsZXQgdmlld3BvcnRTdGF0ZTogYW55O1xuXG4gICAgLy8gVXNlIHNoYXJlZCBpbml0aWFsaXphdGlvbiB3cmFwcGVyXG4gICAgd3JhcEluaXRpYWxpemUoY29tcG9uZW50LCAoKSA9PiB7XG4gICAgICB2aWV3cG9ydFN0YXRlID0gZ2V0Vmlld3BvcnRTdGF0ZShjb21wb25lbnQpO1xuXG4gICAgICAvLyBJbml0aWFsaXplIHN0YXRlIHZhbHVlc1xuICAgICAgaWYgKHZpZXdwb3J0U3RhdGUpIHtcbiAgICAgICAgdG90YWxWaXJ0dWFsU2l6ZSA9IHZpZXdwb3J0U3RhdGUudmlydHVhbFRvdGFsU2l6ZSB8fCAwO1xuICAgICAgICBjb250YWluZXJTaXplID0gdmlld3BvcnRTdGF0ZS5jb250YWluZXJTaXplIHx8IDA7XG4gICAgICAgIC8vIFN5bmMgc2Nyb2xsUG9zaXRpb24gZnJvbSB2aWV3cG9ydFN0YXRlIChpbXBvcnRhbnQgZm9yIGluaXRpYWxTY3JvbGxJbmRleClcbiAgICAgICAgaWYgKHZpZXdwb3J0U3RhdGUuc2Nyb2xsUG9zaXRpb24gPiAwKSB7XG4gICAgICAgICAgc2Nyb2xsUG9zaXRpb24gPSB2aWV3cG9ydFN0YXRlLnNjcm9sbFBvc2l0aW9uO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIExpc3RlbiBmb3IgdmlydHVhbCBzaXplIGNoYW5nZXNcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwidmlld3BvcnQ6dmlydHVhbC1zaXplLWNoYW5nZWRcIiwgKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICAvLyBjb25zb2xlLmxvZyhcIltTY3JvbGxpbmddIFZpcnR1YWwgc2l6ZSBjaGFuZ2VkOlwiLCBkYXRhKTtcbiAgICAgICAgdXBkYXRlU2Nyb2xsQm91bmRzKGRhdGEudG90YWxWaXJ0dWFsU2l6ZSwgY29udGFpbmVyU2l6ZSk7XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciBzY3JvbGwgcG9zaXRpb24gY2hhbmdlcyBmcm9tIG90aGVyIGZlYXR1cmVzIChlLmcuLCB2aXJ0dWFsLnRzIGFmdGVyIGl0ZW0gc2l6ZSBkZXRlY3Rpb24pXG4gICAgICBjb21wb25lbnQub24/LihcInZpZXdwb3J0OnNjcm9sbC1wb3NpdGlvbi1zeW5jXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgaWYgKGRhdGEucG9zaXRpb24gIT09IHVuZGVmaW5lZCAmJiBkYXRhLnBvc2l0aW9uICE9PSBzY3JvbGxQb3NpdGlvbikge1xuICAgICAgICAgIHNjcm9sbFBvc2l0aW9uID0gZGF0YS5wb3NpdGlvbjtcbiAgICAgICAgICAvLyBBbHNvIHVwZGF0ZSBsb2NhbCB0cmFja2luZyB2YXJzXG4gICAgICAgICAgdG90YWxWaXJ0dWFsU2l6ZSA9XG4gICAgICAgICAgICB2aWV3cG9ydFN0YXRlPy52aXJ0dWFsVG90YWxTaXplIHx8IHRvdGFsVmlydHVhbFNpemU7XG4gICAgICAgICAgY29udGFpbmVyU2l6ZSA9IHZpZXdwb3J0U3RhdGU/LmNvbnRhaW5lclNpemUgfHwgY29udGFpbmVyU2l6ZTtcbiAgICAgICAgfVxuICAgICAgfSk7XG5cbiAgICAgIC8vIExpc3RlbiBmb3IgY29udGFpbmVyIHNpemUgY2hhbmdlc1xuICAgICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDpjb250YWluZXItc2l6ZS1jaGFuZ2VkXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgaWYgKGRhdGEuY29udGFpbmVyU2l6ZSkge1xuICAgICAgICAgIGNvbnRhaW5lclNpemUgPSBkYXRhLmNvbnRhaW5lclNpemU7XG4gICAgICAgICAgdXBkYXRlU2Nyb2xsQm91bmRzKHRvdGFsVmlydHVhbFNpemUsIGNvbnRhaW5lclNpemUpO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgLy8gQXR0YWNoIHdoZWVsIGV2ZW50IGxpc3RlbmVyIHRvIHZpZXdwb3J0IGVsZW1lbnRcbiAgICAgIGNvbnN0IHZpZXdwb3J0RWxlbWVudCA9XG4gICAgICAgIHZpZXdwb3J0U3RhdGU/LnZpZXdwb3J0RWxlbWVudCB8fCAoY29tcG9uZW50IGFzIGFueSkudmlld3BvcnRFbGVtZW50O1xuICAgICAgaWYgKHZpZXdwb3J0RWxlbWVudCkge1xuICAgICAgICAvLyBjb25zb2xlLmxvZyhgW1Njcm9sbGluZ10gQXR0YWNoaW5nIHdoZWVsIGV2ZW50IHRvIHZpZXdwb3J0IGVsZW1lbnRgKTtcbiAgICAgICAgdmlld3BvcnRFbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoXCJ3aGVlbFwiLCBoYW5kbGVXaGVlbCwge1xuICAgICAgICAgIHBhc3NpdmU6IGZhbHNlLFxuICAgICAgICB9KTtcblxuICAgICAgICAvLyBJbml0aWFsaXplIHNjcm9sbGVkIHN0YXRlIGJhc2VkIG9uIGN1cnJlbnQgc2Nyb2xsIHBvc2l0aW9uXG4gICAgICAgIHVwZGF0ZVNjcm9sbGVkU3RhdGUodmlld3BvcnRFbGVtZW50LCBzY3JvbGxQb3NpdGlvbik7XG5cbiAgICAgICAgLy8gQWRkIG1vdXNlZG93biBsaXN0ZW5lciB0byBzdG9wIHNjcm9sbGluZyBvbiBjbGlja1xuICAgICAgICBpZiAoc3RvcE9uQ2xpY2spIHtcbiAgICAgICAgICB2aWV3cG9ydEVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcIm1vdXNlZG93blwiLCBoYW5kbGVNb3VzZURvd24pO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gU3RvcmUgcmVmZXJlbmNlIGZvciBjbGVhbnVwXG4gICAgICAgIChjb21wb25lbnQgYXMgYW55KS5fc2Nyb2xsaW5nVmlld3BvcnRFbGVtZW50ID0gdmlld3BvcnRFbGVtZW50O1xuICAgICAgICAoY29tcG9uZW50IGFzIGFueSkuX3Njcm9sbGluZ1N0b3BPbkNsaWNrID0gc3RvcE9uQ2xpY2s7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zb2xlLndhcm4oYFtTY3JvbGxpbmddIE5vIHZpZXdwb3J0IGVsZW1lbnQgZm91bmQgZm9yIHdoZWVsIGV2ZW50c2ApO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgLy8gVXNlIGNsYW1wIGZyb20gdXRpbHNcblxuICAgIC8vIFN0YXJ0IGlkbGUgZGV0ZWN0aW9uXG4gICAgY29uc3Qgc3RhcnRJZGxlRGV0ZWN0aW9uID0gKCkgPT4ge1xuICAgICAgLy8gY29uc29sZS5sb2coXCJbU2Nyb2xsaW5nXSBTdGFydGluZyBpZGxlIGRldGVjdGlvblwiKTtcblxuICAgICAgLy8gU3RvcCBhbnkgZXhpc3RpbmcgaWRsZSBkZXRlY3Rpb24gZmlyc3RcbiAgICAgIGlmIChpZGxlQ2hlY2tGcmFtZSAhPT0gbnVsbCkge1xuICAgICAgICBjYW5jZWxBbmltYXRpb25GcmFtZShpZGxlQ2hlY2tGcmFtZSk7XG4gICAgICAgIGlkbGVDaGVja0ZyYW1lID0gbnVsbDtcbiAgICAgIH1cblxuICAgICAgaGFzRW1pdHRlZElkbGUgPSBmYWxzZTsgLy8gUmVzZXQgaWRsZSBlbWlzc2lvbiBmbGFnXG4gICAgICBjb25zdCBjaGVja0lkbGUgPSAoKSA9PiB7XG4gICAgICAgIGlmIChzY3JvbGxQb3NpdGlvbiA9PT0gbGFzdElkbGVDaGVja1Bvc2l0aW9uKSB7XG4gICAgICAgICAgLy8gUG9zaXRpb24gaGFzbid0IGNoYW5nZWQgLSB3ZSdyZSBpZGxlXG4gICAgICAgICAgaWYgKCFoYXNFbWl0dGVkSWRsZSAmJiAoc3BlZWRUcmFja2VyLnZlbG9jaXR5ID4gMCB8fCBpc1Njcm9sbGluZykpIHtcbiAgICAgICAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgLy8gICBcIltTY3JvbGxpbmddIElkbGUgZGV0ZWN0ZWQgLSBwb3NpdGlvbiBzdGFibGUsIHNldHRpbmcgdmVsb2NpdHkgdG8gemVyb1wiXG4gICAgICAgICAgICAvLyApO1xuICAgICAgICAgICAgaGFzRW1pdHRlZElkbGUgPSB0cnVlOyAvLyBNYXJrIHRoYXQgd2UndmUgZW1pdHRlZCBpZGxlXG4gICAgICAgICAgICBzZXRWZWxvY2l0eVRvWmVybygpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyBQb3NpdGlvbiBjaGFuZ2VkLCByZXNldCB0aGUgZmxhZ1xuICAgICAgICAgIGhhc0VtaXR0ZWRJZGxlID0gZmFsc2U7XG4gICAgICAgIH1cbiAgICAgICAgbGFzdElkbGVDaGVja1Bvc2l0aW9uID0gc2Nyb2xsUG9zaXRpb247XG4gICAgICAgIGlkbGVDaGVja0ZyYW1lID0gcmVxdWVzdEFuaW1hdGlvbkZyYW1lKGNoZWNrSWRsZSk7XG4gICAgICB9O1xuICAgICAgaWRsZUNoZWNrRnJhbWUgPSByZXF1ZXN0QW5pbWF0aW9uRnJhbWUoY2hlY2tJZGxlKTtcbiAgICB9O1xuXG4gICAgLy8gU3RvcCBpZGxlIGRldGVjdGlvblxuICAgIGNvbnN0IHN0b3BJZGxlRGV0ZWN0aW9uID0gKCkgPT4ge1xuICAgICAgaWYgKGlkbGVDaGVja0ZyYW1lKSB7XG4gICAgICAgIC8vIGNvbnNvbGUubG9nKFwiW1Njcm9sbGluZ10gU3RvcHBpbmcgaWRsZSBkZXRlY3Rpb25cIik7XG4gICAgICAgIGNhbmNlbEFuaW1hdGlvbkZyYW1lKGlkbGVDaGVja0ZyYW1lKTtcbiAgICAgICAgaWRsZUNoZWNrRnJhbWUgPSBudWxsO1xuICAgICAgfVxuICAgIH07XG5cbiAgICAvLyBTZXQgdmVsb2NpdHkgdG8gemVybyBhbmQgZW1pdCBpZGxlIGV2ZW50XG4gICAgY29uc3Qgc2V0VmVsb2NpdHlUb1plcm8gPSAoKSA9PiB7XG4gICAgICAvLyBTdG9wIGlkbGUgZGV0ZWN0aW9uIHNpbmNlIHdlJ3JlIG5vdyBpZGxlXG4gICAgICBzdG9wSWRsZURldGVjdGlvbigpO1xuXG4gICAgICBzcGVlZFRyYWNrZXIgPSBjcmVhdGVTcGVlZFRyYWNrZXIoKTtcbiAgICAgIGlzU2Nyb2xsaW5nID0gZmFsc2U7IC8vIFJlc2V0IHNjcm9sbGluZyBzdGF0ZVxuXG4gICAgICAvLyBFbWl0IHZlbG9jaXR5IGNoYW5nZVxuICAgICAgY29tcG9uZW50LmVtaXQ/LihcInZpZXdwb3J0OnZlbG9jaXR5LWNoYW5nZWRcIiwge1xuICAgICAgICB2ZWxvY2l0eTogMCxcbiAgICAgICAgZGlyZWN0aW9uOiBzcGVlZFRyYWNrZXIuZGlyZWN0aW9uLFxuICAgICAgfSk7XG5cbiAgICAgIC8vIEVtaXQgaWRsZSBzdGF0ZVxuICAgICAgY29tcG9uZW50LmVtaXQ/LihcInZpZXdwb3J0OmlkbGVcIiwge1xuICAgICAgICBwb3NpdGlvbjogc2Nyb2xsUG9zaXRpb24sXG4gICAgICAgIGxhc3RTY3JvbGxUaW1lLFxuICAgICAgfSk7XG4gICAgfTtcbiAgICAvLyBVcGRhdGUgY29udGFpbmVyIHBvc2l0aW9uXG4gICAgY29uc3QgdXBkYXRlQ29udGFpbmVyUG9zaXRpb24gPSAoKSA9PiB7XG4gICAgICBpZiAoIXZpZXdwb3J0U3RhdGUgfHwgIXZpZXdwb3J0U3RhdGUuaXRlbXNDb250YWluZXIpIHJldHVybjtcblxuICAgICAgLy8gSXRlbXMgY29udGFpbmVyIGRvZXNuJ3QgbW92ZSAtIGl0ZW1zIGluc2lkZSBpdCBhcmUgcG9zaXRpb25lZFxuICAgICAgLy8gVGhpcyBpcyBoYW5kbGVkIGJ5IHRoZSByZW5kZXJpbmcgZmVhdHVyZVxuICAgIH07XG5cbiAgICAvLyBIYW5kbGUgbW91c2Vkb3duIHRvIHN0b3Agc2Nyb2xsaW5nIG9uIGNsaWNrXG4gICAgY29uc3QgaGFuZGxlTW91c2VEb3duID0gKF9lOiBNb3VzZUV2ZW50KSA9PiB7XG4gICAgICAvLyBBbmNob3IgdGhlIGN1cnJlbnQgc2Nyb2xsIHBvc2l0aW9uIC0gYW55IHdoZWVsIGV2ZW50cyB3aWxsIGJlIGZvcmNlZCBiYWNrIHRvIHRoaXMgcG9zaXRpb25cbiAgICAgIC8vIFRoaXMgZWxlZ2FudGx5IGhhbmRsZXMgbW91c2Ugd2hlZWxzIHdpdGggcGh5c2ljYWwgaW5lcnRpYSAobGlrZSBMb2dpdGVjaCBmcmVlLXNwaW4pXG4gICAgICBhbmNob3JQb3NpdGlvbiA9IHNjcm9sbFBvc2l0aW9uO1xuICAgICAgYW5jaG9yVGltZSA9IERhdGUubm93KCk7XG4gICAgICBhbmNob3JJbml0aWFsRGVsdGEgPSAwOyAvLyBXaWxsIGJlIHNldCBieSBmaXJzdCB3aGVlbCBldmVudFxuICAgICAgYW5jaG9yTGFzdERlbHRhID0gMDtcbiAgICAgIGFuY2hvck1pbkRlbHRhID0gSW5maW5pdHk7XG4gICAgICBjb25zZWN1dGl2ZUluY3JlYXNlcyA9IDA7XG4gICAgICBzdXN0YWluZWRIaWdoQ291bnQgPSAwO1xuXG4gICAgICAvLyBPbmx5IHN0b3AgaWYgd2UncmUgY3VycmVudGx5IHNjcm9sbGluZ1xuICAgICAgaWYgKGlzU2Nyb2xsaW5nIHx8IHNwZWVkVHJhY2tlci52ZWxvY2l0eSA+IDApIHtcbiAgICAgICAgc2V0VmVsb2NpdHlUb1plcm8oKTtcbiAgICAgIH1cbiAgICAgIC8vIEFsc28gc3RvcCBtb21lbnR1bSBhbmltYXRpb24gaWYgYWN0aXZlXG4gICAgICBjb25zdCBtb21lbnR1bVN0YXRlID0gKGNvbXBvbmVudC52aWV3cG9ydCBhcyBhbnkpLm1vbWVudHVtU3RhdGU7XG4gICAgICBpZiAobW9tZW50dW1TdGF0ZT8uc3RvcE1vbWVudHVtKSB7XG4gICAgICAgIG1vbWVudHVtU3RhdGUuc3RvcE1vbWVudHVtKCk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIEhhbmRsZSB3aGVlbCBldmVudFxuICAgIGNvbnN0IGhhbmRsZVdoZWVsID0gKGV2ZW50OiBXaGVlbEV2ZW50KSA9PiB7XG4gICAgICBldmVudC5wcmV2ZW50RGVmYXVsdCgpO1xuXG4gICAgICBjb25zdCBkZWx0YSA9IG9yaWVudGF0aW9uID09PSBcInZlcnRpY2FsXCIgPyBldmVudC5kZWx0YVkgOiBldmVudC5kZWx0YVg7XG4gICAgICBjb25zdCBzY3JvbGxEZWx0YSA9IGRlbHRhICogc2Vuc2l0aXZpdHk7XG5cbiAgICAgIGNvbnN0IG5vdyA9IERhdGUubm93KCk7XG4gICAgICBjb25zdCB0aW1lU2luY2VMYXN0V2hlZWwgPSBub3cgLSBsYXN0V2hlZWxUaW1lO1xuICAgICAgbGFzdFdoZWVsVGltZSA9IG5vdztcblxuICAgICAgLy8gSWYgd2UgaGF2ZSBhbiBhbmNob3IgcG9zaXRpb24gKHVzZXIgY2xpY2tlZCB0byBzdG9wIHNjcm9sbGluZyksXG4gICAgICAvLyBjaGVjayBpZiB0aGlzIGlzIGludGVudGlvbmFsIG5ldyBzY3JvbGxpbmcgb3IgcmVzaWR1YWwgd2hlZWwgaW5lcnRpYVxuICAgICAgaWYgKGFuY2hvclBvc2l0aW9uICE9PSBudWxsKSB7XG4gICAgICAgIGNvbnN0IHRpbWVTaW5jZUFuY2hvciA9IG5vdyAtIGFuY2hvclRpbWU7XG4gICAgICAgIGNvbnN0IHdoZWVsRGVsdGFNYWduaXR1ZGUgPSBNYXRoLmFicyhzY3JvbGxEZWx0YSk7XG5cbiAgICAgICAgLy8gVHJhY2sgaW5pdGlhbCBkZWx0YSB0byBkZXRlY3Qgd2hlbiBpbmVydGlhIGhhcyBkZWNheWVkXG4gICAgICAgIGlmIChhbmNob3JJbml0aWFsRGVsdGEgPT09IDApIHtcbiAgICAgICAgICBhbmNob3JJbml0aWFsRGVsdGEgPSB3aGVlbERlbHRhTWFnbml0dWRlO1xuICAgICAgICAgIGFuY2hvckxhc3REZWx0YSA9IHdoZWVsRGVsdGFNYWduaXR1ZGU7XG4gICAgICAgICAgYW5jaG9yTWluRGVsdGEgPSB3aGVlbERlbHRhTWFnbml0dWRlO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gVHJhY2sgbWluaW11bSBkZWx0YSB0byBkZXRlY3QgcmVhY2NlbGVyYXRpb24gKHVzZXIgc3RhcnRlZCBzY3JvbGxpbmcgYWdhaW4pXG4gICAgICAgIGFuY2hvck1pbkRlbHRhID0gTWF0aC5taW4oYW5jaG9yTWluRGVsdGEsIHdoZWVsRGVsdGFNYWduaXR1ZGUpO1xuXG4gICAgICAgIC8vIERldGVjdCBpZiBkZWx0YSBpcyBpbmNyZWFzaW5nICh1c2VyIHN0YXJ0ZWQgbmV3IHNjcm9sbClcbiAgICAgICAgLy8gSW5lcnRpYSBvbmx5IGRlY2F5cywgc28gaW5jcmVhc2luZyBkZWx0YSA9IGludGVudGlvbmFsIG5ldyBzY3JvbGxpbmdcbiAgICAgICAgLy8gQlVUOiBmaWx0ZXIgb3V0IGJhdGNoZWQgZXZlbnRzIHdoZXJlIGRlbHRhIGlzIH4yeCB0aGUgbGFzdCAoYnJvd3NlciBiYXRjaGluZywgbm90IHJlYWwgaW5jcmVhc2UpXG4gICAgICAgIGNvbnN0IGlzQmF0Y2hlZEV2ZW50ID1cbiAgICAgICAgICB3aGVlbERlbHRhTWFnbml0dWRlID4gYW5jaG9yTGFzdERlbHRhICogMS44ICYmXG4gICAgICAgICAgd2hlZWxEZWx0YU1hZ25pdHVkZSA8IGFuY2hvckxhc3REZWx0YSAqIDIuMjtcblxuICAgICAgICAvLyBUcmFjayBjb25zZWN1dGl2ZSBpbmNyZWFzZXMgLSBzdXN0YWluZWQgaW5jcmVhc2VzIGluZGljYXRlIG5ldyBzY3JvbGxpbmdcbiAgICAgICAgLy8gVXNlIDElIHRocmVzaG9sZCB0byBjYXRjaCBncmFkdWFsIGFjY2VsZXJhdGlvblxuICAgICAgICBpZiAoIWlzQmF0Y2hlZEV2ZW50ICYmIHdoZWVsRGVsdGFNYWduaXR1ZGUgPiBhbmNob3JMYXN0RGVsdGEgKiAxLjAxKSB7XG4gICAgICAgICAgY29uc2VjdXRpdmVJbmNyZWFzZXMrKztcbiAgICAgICAgfSBlbHNlIGlmICh3aGVlbERlbHRhTWFnbml0dWRlIDwgYW5jaG9yTGFzdERlbHRhICogMC45OSkge1xuICAgICAgICAgIC8vIERlbHRhIGRlY3JlYXNpbmcgLSByZXNldCBjb3VudGVyXG4gICAgICAgICAgY29uc2VjdXRpdmVJbmNyZWFzZXMgPSAwO1xuICAgICAgICB9XG4gICAgICAgIC8vIElmIGRlbHRhIGlzIHJvdWdobHkgdGhlIHNhbWUgKHdpdGhpbiAxJSksIGRvbid0IGNoYW5nZSB0aGUgY291bnRlclxuXG4gICAgICAgIC8vIFRyYWNrIHN1c3RhaW5lZCBoaWdoIGRlbHRhIChkZWx0YSBzaWduaWZpY2FudGx5IGFib3ZlIG1pbmltdW0gZm9yIG11bHRpcGxlIGV2ZW50cylcbiAgICAgICAgLy8gVGhpcyBjYXRjaGVzIHdoZW4gdXNlciBzdGFydHMgc2Nyb2xsaW5nIGJ1dCBkZWx0YSBwbGF0ZWF1cyBpbnN0ZWFkIG9mIHN0cmljdGx5IGluY3JlYXNpbmdcbiAgICAgICAgaWYgKCFpc0JhdGNoZWRFdmVudCAmJiB3aGVlbERlbHRhTWFnbml0dWRlID4gYW5jaG9yTWluRGVsdGEgKiAxLjA1KSB7XG4gICAgICAgICAgc3VzdGFpbmVkSGlnaENvdW50Kys7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgc3VzdGFpbmVkSGlnaENvdW50ID0gMDtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFJlbGVhc2UgaWY6IDMrIGNvbnNlY3V0aXZlIGluY3JlYXNlcywgT1Igc3VzdGFpbmVkIGhpZ2ggZGVsdGEsIE9SIHNpZ25pZmljYW50IHNpbmdsZSBpbmNyZWFzZVxuICAgICAgICBjb25zdCBzdXN0YWluZWRJbmNyZWFzZSA9IGNvbnNlY3V0aXZlSW5jcmVhc2VzID49IDM7XG4gICAgICAgIGNvbnN0IHN1c3RhaW5lZEhpZ2ggPSBzdXN0YWluZWRIaWdoQ291bnQgPj0gNTsgLy8gNSsgZXZlbnRzIGF0IDUlKyBhYm92ZSBtaW5pbXVtXG4gICAgICAgIGNvbnN0IHNpZ25pZmljYW50SW5jcmVhc2UgPVxuICAgICAgICAgICFpc0JhdGNoZWRFdmVudCAmJlxuICAgICAgICAgIHdoZWVsRGVsdGFNYWduaXR1ZGUgPiBhbmNob3JNaW5EZWx0YSAqIDEuMTUgJiYgLy8gMTUlIGluY3JlYXNlIGZyb20gbWluaW11bVxuICAgICAgICAgIHdoZWVsRGVsdGFNYWduaXR1ZGUgPiBhbmNob3JMYXN0RGVsdGEgKiAxLjA4OyAvLyBhbmQgOCUgaW5jcmVhc2luZyBmcm9tIGxhc3RcbiAgICAgICAgY29uc3QgZGVsdGFJbmNyZWFzaW5nID1cbiAgICAgICAgICBzdXN0YWluZWRJbmNyZWFzZSB8fCBzdXN0YWluZWRIaWdoIHx8IHNpZ25pZmljYW50SW5jcmVhc2U7XG5cbiAgICAgICAgLy8gUmVsZWFzZSBhbmNob3IgaWY6XG4gICAgICAgIC8vIDEuIFRoZXJlIHdhcyBhIGdhcCBpbiB3aGVlbCBldmVudHMgKD4gMjAwbXMpIC0gdXNlciBzdG9wcGVkIGFuZCBzdGFydGVkIG5ldyBzY3JvbGxcbiAgICAgICAgLy8gMi4gRGVsdGEgaXMgaW5jcmVhc2luZyBmcm9tIG1pbmltdW0gLSB1c2VyIHN0YXJ0ZWQgc2Nyb2xsaW5nIGFnYWluIGR1cmluZyBpbmVydGlhXG4gICAgICAgIC8vIDMuIERlbHRhIGhhcyBkZWNheWVkIHNpZ25pZmljYW50bHkgKDwgMzAlIG9mIGluaXRpYWwpIC0gaW5lcnRpYSBpcyB3aW5kaW5nIGRvd25cbiAgICAgICAgLy8gNC4gTG93IGFic29sdXRlIGRlbHRhICg8IDMwKSAtIGdlbnRsZSBpbnRlbnRpb25hbCBzY3JvbGxcbiAgICAgICAgY29uc3QgaGFzV2hlZWxHYXAgPSB0aW1lU2luY2VMYXN0V2hlZWwgPiAyMDA7XG4gICAgICAgIGNvbnN0IGRlbHRhRGVjYXllZCA9IHdoZWVsRGVsdGFNYWduaXR1ZGUgPCBhbmNob3JJbml0aWFsRGVsdGEgKiAwLjM7XG4gICAgICAgIGNvbnN0IGlzTG93RGVsdGEgPSB3aGVlbERlbHRhTWFnbml0dWRlIDwgMzA7XG5cbiAgICAgICAgaWYgKGhhc1doZWVsR2FwKSB7XG4gICAgICAgICAgLy8gVGltZSBnYXAgLSB1c2VyIHN0b3BwZWQgYW5kIHN0YXJ0ZWQgbmV3IHNjcm9sbFxuICAgICAgICAgIGFuY2hvclBvc2l0aW9uID0gbnVsbDtcbiAgICAgICAgfSBlbHNlIGlmIChkZWx0YUluY3JlYXNpbmcgJiYgdGltZVNpbmNlQW5jaG9yID4gMTAwKSB7XG4gICAgICAgICAgLy8gRGVsdGEgaW5jcmVhc2luZyAtIHVzZXIgc3RhcnRlZCBzY3JvbGxpbmcgYWdhaW5cbiAgICAgICAgICBhbmNob3JQb3NpdGlvbiA9IG51bGw7XG4gICAgICAgIH0gZWxzZSBpZiAoaXNMb3dEZWx0YSkge1xuICAgICAgICAgIC8vIFZlcnkgbG93IGRlbHRhIC0gZ2VudGxlIGludGVudGlvbmFsIHNjcm9sbFxuICAgICAgICAgIGFuY2hvclBvc2l0aW9uID0gbnVsbDtcbiAgICAgICAgfSBlbHNlIGlmIChkZWx0YURlY2F5ZWQgJiYgdGltZVNpbmNlQW5jaG9yID4gMzAwKSB7XG4gICAgICAgICAgLy8gRGVsdGEgaGFzIGRlY2F5ZWQgYW5kIHNvbWUgdGltZSBwYXNzZWQgLSBpbmVydGlhIGlzIGRvbmVcbiAgICAgICAgICBhbmNob3JQb3NpdGlvbiA9IG51bGw7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgLy8gU3RpbGwgaW4gaW5lcnRpYSBwaGFzZSAtIHN0YXkgYW5jaG9yZWRcbiAgICAgICAgICBpZiAoc2Nyb2xsUG9zaXRpb24gIT09IGFuY2hvclBvc2l0aW9uKSB7XG4gICAgICAgICAgICBzY3JvbGxQb3NpdGlvbiA9IGFuY2hvclBvc2l0aW9uO1xuICAgICAgICAgICAgaWYgKHZpZXdwb3J0U3RhdGUpIHtcbiAgICAgICAgICAgICAgdmlld3BvcnRTdGF0ZS5zY3JvbGxQb3NpdGlvbiA9IHNjcm9sbFBvc2l0aW9uO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY29tcG9uZW50LnZpZXdwb3J0LnJlbmRlckl0ZW1zKCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIC8vIFVwZGF0ZSBsYXN0IGRlbHRhIGZvciBuZXh0IGNvbXBhcmlzb25cbiAgICAgICAgICBhbmNob3JMYXN0RGVsdGEgPSB3aGVlbERlbHRhTWFnbml0dWRlO1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFVwZGF0ZSBsYXN0IGRlbHRhIGZvciBuZXh0IGNvbXBhcmlzb25cbiAgICAgICAgYW5jaG9yTGFzdERlbHRhID0gd2hlZWxEZWx0YU1hZ25pdHVkZTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgcHJldmlvdXNQb3NpdGlvbiA9IHNjcm9sbFBvc2l0aW9uO1xuICAgICAgY29uc3QgbWF4U2Nyb2xsID0gTWF0aC5tYXgoMCwgdG90YWxWaXJ0dWFsU2l6ZSAtIGNvbnRhaW5lclNpemUpO1xuXG4gICAgICBsZXQgbmV3UG9zaXRpb24gPSBzY3JvbGxQb3NpdGlvbiArIHNjcm9sbERlbHRhO1xuXG4gICAgICAvLyBBcHBseSBzbW9vdGhpbmcgaWYgZW5hYmxlZFxuICAgICAgaWYgKHNtb290aGluZykge1xuICAgICAgICBjb25zdCBzbW9vdGhpbmdGYWN0b3IgPSAwLjM7XG4gICAgICAgIG5ld1Bvc2l0aW9uID0gc2Nyb2xsUG9zaXRpb24gKyBzY3JvbGxEZWx0YSAqIHNtb290aGluZ0ZhY3RvcjtcbiAgICAgIH1cblxuICAgICAgbmV3UG9zaXRpb24gPSBjbGFtcChuZXdQb3NpdGlvbiwgMCwgbWF4U2Nyb2xsKTtcblxuICAgICAgaWYgKG5ld1Bvc2l0aW9uICE9PSBzY3JvbGxQb3NpdGlvbikge1xuICAgICAgICBzY3JvbGxQb3NpdGlvbiA9IG5ld1Bvc2l0aW9uO1xuICAgICAgICBjb25zdCBub3cgPSBEYXRlLm5vdygpO1xuXG4gICAgICAgIC8vIFVwZGF0ZSBzY3JvbGwgc3RhdGVcbiAgICAgICAgaWYgKCFpc1Njcm9sbGluZykge1xuICAgICAgICAgIGlzU2Nyb2xsaW5nID0gdHJ1ZTtcbiAgICAgICAgICAvLyBTdG9wIGFueSBleGlzdGluZyBpZGxlIGRldGVjdGlvbiBiZWZvcmUgc3RhcnRpbmcgbmV3IG9uZVxuICAgICAgICAgIHN0b3BJZGxlRGV0ZWN0aW9uKCk7XG4gICAgICAgICAgc3RhcnRJZGxlRGV0ZWN0aW9uKCk7XG4gICAgICAgIH1cbiAgICAgICAgbGFzdFNjcm9sbFRpbWUgPSBub3c7XG5cbiAgICAgICAgLy8gVXBkYXRlIHNwZWVkIHRyYWNrZXJcbiAgICAgICAgc3BlZWRUcmFja2VyID0gdXBkYXRlU3BlZWRUcmFja2VyKFxuICAgICAgICAgIHNwZWVkVHJhY2tlcixcbiAgICAgICAgICBzY3JvbGxQb3NpdGlvbixcbiAgICAgICAgICBwcmV2aW91c1Bvc2l0aW9uLFxuICAgICAgICApO1xuXG4gICAgICAgIC8vIFVwZGF0ZSB2aWV3cG9ydCBzdGF0ZVxuICAgICAgICBpZiAodmlld3BvcnRTdGF0ZSkge1xuICAgICAgICAgIHZpZXdwb3J0U3RhdGUuc2Nyb2xsUG9zaXRpb24gPSBzY3JvbGxQb3NpdGlvbjtcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLnZlbG9jaXR5ID0gc3BlZWRUcmFja2VyLnZlbG9jaXR5O1xuICAgICAgICAgIHZpZXdwb3J0U3RhdGUuc2Nyb2xsRGlyZWN0aW9uID0gc3BlZWRUcmFja2VyLmRpcmVjdGlvbjtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFVwZGF0ZSBzY3JvbGxlZCBzdGF0ZSBjbGFzc1xuICAgICAgICBjb25zdCB2aWV3cG9ydEVsID1cbiAgICAgICAgICB2aWV3cG9ydFN0YXRlPy52aWV3cG9ydEVsZW1lbnQgfHwgKGNvbXBvbmVudCBhcyBhbnkpLnZpZXdwb3J0RWxlbWVudDtcbiAgICAgICAgaWYgKHZpZXdwb3J0RWwpIHtcbiAgICAgICAgICB1cGRhdGVTY3JvbGxlZFN0YXRlKHZpZXdwb3J0RWwsIHNjcm9sbFBvc2l0aW9uKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFN0b3JlIHBlbmRpbmcgc2Nyb2xsIGRhdGEgZm9yIFJBRi1iYXRjaGVkIHJlbmRlcmluZ1xuICAgICAgICBwZW5kaW5nU2Nyb2xsRGF0YSA9IHtcbiAgICAgICAgICBwb3NpdGlvbjogc2Nyb2xsUG9zaXRpb24sXG4gICAgICAgICAgZGlyZWN0aW9uOiBzcGVlZFRyYWNrZXIuZGlyZWN0aW9uLFxuICAgICAgICAgIHByZXZpb3VzUG9zaXRpb24sXG4gICAgICAgIH07XG5cbiAgICAgICAgLy8gU2NoZWR1bGUgcmVuZGVyIHZpYSBSQUYgaWYgbm90IGFscmVhZHkgc2NoZWR1bGVkXG4gICAgICAgIC8vIFRoaXMgZW5zdXJlcyB3ZSByZW5kZXIgYXQgbW9zdCBvbmNlIHBlciBmcmFtZSwgZXZlbiBkdXJpbmcgZmFzdCBzY3JvbGxpbmdcbiAgICAgICAgaWYgKCFyZW5kZXJTY2hlZHVsZWQpIHtcbiAgICAgICAgICByZW5kZXJTY2hlZHVsZWQgPSB0cnVlO1xuICAgICAgICAgIHJlcXVlc3RBbmltYXRpb25GcmFtZSgoKSA9PiB7XG4gICAgICAgICAgICByZW5kZXJTY2hlZHVsZWQgPSBmYWxzZTtcblxuICAgICAgICAgICAgLy8gVXNlIHRoZSBsYXRlc3Qgc2Nyb2xsIGRhdGEgKGNvYWxlc2NlcyBtdWx0aXBsZSB3aGVlbCBldmVudHMpXG4gICAgICAgICAgICBpZiAocGVuZGluZ1Njcm9sbERhdGEpIHtcbiAgICAgICAgICAgICAgLy8gRW1pdCBldmVudHMgd2l0aCB0aGUgbGF0ZXN0IHBvc2l0aW9uXG4gICAgICAgICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpzY3JvbGxcIiwge1xuICAgICAgICAgICAgICAgIHBvc2l0aW9uOiBwZW5kaW5nU2Nyb2xsRGF0YS5wb3NpdGlvbixcbiAgICAgICAgICAgICAgICBkaXJlY3Rpb246IHBlbmRpbmdTY3JvbGxEYXRhLmRpcmVjdGlvbixcbiAgICAgICAgICAgICAgICBwcmV2aW91c1Bvc2l0aW9uOiBwZW5kaW5nU2Nyb2xsRGF0YS5wcmV2aW91c1Bvc2l0aW9uLFxuICAgICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgICBjb21wb25lbnQuZW1pdD8uKFwidmlld3BvcnQ6dmVsb2NpdHktY2hhbmdlZFwiLCB7XG4gICAgICAgICAgICAgICAgdmVsb2NpdHk6IHNwZWVkVHJhY2tlci52ZWxvY2l0eSxcbiAgICAgICAgICAgICAgICBkaXJlY3Rpb246IHNwZWVkVHJhY2tlci5kaXJlY3Rpb24sXG4gICAgICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgICAgIC8vIFRyaWdnZXIgcmVuZGVyXG4gICAgICAgICAgICAgIGNvbXBvbmVudC52aWV3cG9ydC5yZW5kZXJJdGVtcygpO1xuXG4gICAgICAgICAgICAgIHBlbmRpbmdTY3JvbGxEYXRhID0gbnVsbDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH07XG5cbiAgICAvLyBTY3JvbGwgdG8gcG9zaXRpb25cbiAgICBjb25zdCBzY3JvbGxUb1Bvc2l0aW9uID0gKHBvc2l0aW9uOiBudW1iZXIsIHNvdXJjZT86IHN0cmluZykgPT4ge1xuICAgICAgY29uc3QgbWF4U2Nyb2xsID0gTWF0aC5tYXgoMCwgdG90YWxWaXJ0dWFsU2l6ZSAtIGNvbnRhaW5lclNpemUpO1xuICAgICAgY29uc3QgY2xhbXBlZFBvc2l0aW9uID0gY2xhbXAocG9zaXRpb24sIDAsIG1heFNjcm9sbCk7XG5cbiAgICAgIGlmIChjbGFtcGVkUG9zaXRpb24gIT09IHNjcm9sbFBvc2l0aW9uKSB7XG4gICAgICAgIGNvbnN0IHByZXZpb3VzUG9zaXRpb24gPSBzY3JvbGxQb3NpdGlvbjtcbiAgICAgICAgc2Nyb2xsUG9zaXRpb24gPSBjbGFtcGVkUG9zaXRpb247XG5cbiAgICAgICAgLy8gVXBkYXRlIHNwZWVkIHRyYWNrZXIgdG8gY2FsY3VsYXRlIHZlbG9jaXR5XG4gICAgICAgIHNwZWVkVHJhY2tlciA9IHVwZGF0ZVNwZWVkVHJhY2tlcihcbiAgICAgICAgICBzcGVlZFRyYWNrZXIsXG4gICAgICAgICAgc2Nyb2xsUG9zaXRpb24sXG4gICAgICAgICAgcHJldmlvdXNQb3NpdGlvbixcbiAgICAgICAgKTtcblxuICAgICAgICAvLyBVcGRhdGUgdmlld3BvcnQgc3RhdGVcbiAgICAgICAgaWYgKHZpZXdwb3J0U3RhdGUpIHtcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLnNjcm9sbFBvc2l0aW9uID0gc2Nyb2xsUG9zaXRpb247XG4gICAgICAgICAgdmlld3BvcnRTdGF0ZS52ZWxvY2l0eSA9IHNwZWVkVHJhY2tlci52ZWxvY2l0eTtcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLnNjcm9sbERpcmVjdGlvbiA9IHNwZWVkVHJhY2tlci5kaXJlY3Rpb247XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBkaXJlY3Rpb24gPVxuICAgICAgICAgIGNsYW1wZWRQb3NpdGlvbiA+IHByZXZpb3VzUG9zaXRpb24gPyBcImZvcndhcmRcIiA6IFwiYmFja3dhcmRcIjtcblxuICAgICAgICAvLyBVcGRhdGUgc2Nyb2xsZWQgc3RhdGUgY2xhc3NcbiAgICAgICAgY29uc3Qgdmlld3BvcnRFbCA9XG4gICAgICAgICAgdmlld3BvcnRTdGF0ZT8udmlld3BvcnRFbGVtZW50IHx8IChjb21wb25lbnQgYXMgYW55KS52aWV3cG9ydEVsZW1lbnQ7XG4gICAgICAgIGlmICh2aWV3cG9ydEVsKSB7XG4gICAgICAgICAgdXBkYXRlU2Nyb2xsZWRTdGF0ZSh2aWV3cG9ydEVsLCBzY3JvbGxQb3NpdGlvbik7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBFbWl0IHNjcm9sbCBldmVudFxuICAgICAgICBjb21wb25lbnQuZW1pdD8uKFwidmlld3BvcnQ6c2Nyb2xsXCIsIHtcbiAgICAgICAgICBwb3NpdGlvbjogc2Nyb2xsUG9zaXRpb24sXG4gICAgICAgICAgZGlyZWN0aW9uLFxuICAgICAgICAgIHByZXZpb3VzUG9zaXRpb24sXG4gICAgICAgIH0pO1xuXG4gICAgICAgIC8vIEVtaXQgdmVsb2NpdHkgY2hhbmdlIGV2ZW50IHNvIGNvbGxlY3Rpb24gY2FuIHRyYWNrIGl0XG4gICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDp2ZWxvY2l0eS1jaGFuZ2VkXCIsIHtcbiAgICAgICAgICB2ZWxvY2l0eTogc3BlZWRUcmFja2VyLnZlbG9jaXR5LFxuICAgICAgICAgIGRpcmVjdGlvbjogc3BlZWRUcmFja2VyLmRpcmVjdGlvbixcbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gVXBkYXRlIHNjcm9sbCBzdGF0ZSBhbmQgaWRsZSBkZXRlY3Rpb25cbiAgICAgICAgaWYgKCFpc1Njcm9sbGluZykge1xuICAgICAgICAgIGlzU2Nyb2xsaW5nID0gdHJ1ZTtcbiAgICAgICAgICBzdGFydElkbGVEZXRlY3Rpb24oKTtcbiAgICAgICAgfVxuICAgICAgICBsYXN0U2Nyb2xsVGltZSA9IERhdGUubm93KCk7XG5cbiAgICAgICAgLy8gVHJpZ2dlciByZW5kZXJcbiAgICAgICAgY29tcG9uZW50LnZpZXdwb3J0LnJlbmRlckl0ZW1zKCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICAvLyBjb25zb2xlLmxvZyhgW1Njcm9sbGluZ10gUG9zaXRpb24gdW5jaGFuZ2VkOiAke3Njcm9sbFBvc2l0aW9ufWApO1xuICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgLy8gICBgW1Njcm9sbGluZ10gUG9zaXRpb24gdW5jaGFuZ2VkOiAke3Njcm9sbFBvc2l0aW9ufSwgbm90IHJlc2V0dGluZyBpZGxlIHRpbWVvdXRgXG4gICAgICAgIC8vICk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIFNjcm9sbCB0byBpbmRleFxuICAgIGNvbnN0IHNjcm9sbFRvSW5kZXggPSAoXG4gICAgICBpbmRleDogbnVtYmVyLFxuICAgICAgYWxpZ25tZW50OiBcInN0YXJ0XCIgfCBcImNlbnRlclwiIHwgXCJlbmRcIiA9IFwic3RhcnRcIixcbiAgICApID0+IHtcbiAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgLy8gICBgW1Njcm9sbGluZ10gc2Nyb2xsVG9JbmRleCBjYWxsZWQ6IGluZGV4PSR7aW5kZXh9LCBhbGlnbm1lbnQ9JHthbGlnbm1lbnR9YFxuICAgICAgLy8gKTtcbiAgICAgIGlmICghdmlld3BvcnRTdGF0ZSkge1xuICAgICAgICAvL2NvbnNvbGUubG9nKGBbU2Nyb2xsaW5nXSBzY3JvbGxUb0luZGV4IGFib3J0ZWQ6IG5vIHZpZXdwb3J0IHN0YXRlYCk7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cblxuICAgICAgY29uc3QgaXRlbVNpemUgPSB2aWV3cG9ydFN0YXRlLml0ZW1TaXplIHx8IDUwO1xuICAgICAgY29uc3QgdG90YWxJdGVtcyA9IHZpZXdwb3J0U3RhdGUudG90YWxJdGVtcyB8fCAwO1xuICAgICAgY29uc3QgYWN0dWFsVG90YWxTaXplID0gdG90YWxJdGVtcyAqIGl0ZW1TaXplO1xuICAgICAgY29uc3QgTUFYX1ZJUlRVQUxfU0laRSA9XG4gICAgICAgIFZJRVdQT1JUX0NPTlNUQU5UUy5WSVJUVUFMX1NDUk9MTC5NQVhfVklSVFVBTF9TSVpFO1xuICAgICAgY29uc3QgaXNDb21wcmVzc2VkID0gYWN0dWFsVG90YWxTaXplID4gTUFYX1ZJUlRVQUxfU0laRTtcblxuICAgICAgbGV0IHRhcmdldFBvc2l0aW9uOiBudW1iZXI7XG5cbiAgICAgIGlmIChpc0NvbXByZXNzZWQpIHtcbiAgICAgICAgLy8gSW4gY29tcHJlc3NlZCBzcGFjZSwgbWFwIGluZGV4IHRvIHZpcnR1YWwgcG9zaXRpb25cbiAgICAgICAgY29uc3QgcmF0aW8gPSBpbmRleCAvIHRvdGFsSXRlbXM7XG4gICAgICAgIHRhcmdldFBvc2l0aW9uID0gcmF0aW8gKiBNYXRoLm1pbihhY3R1YWxUb3RhbFNpemUsIE1BWF9WSVJUVUFMX1NJWkUpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8gRGlyZWN0IGNhbGN1bGF0aW9uIHdoZW4gbm90IGNvbXByZXNzZWRcbiAgICAgICAgdGFyZ2V0UG9zaXRpb24gPSBpbmRleCAqIGl0ZW1TaXplO1xuICAgICAgfVxuXG4gICAgICAvLyBBZGp1c3QgcG9zaXRpb24gYmFzZWQgb24gYWxpZ25tZW50XG4gICAgICBzd2l0Y2ggKGFsaWdubWVudCkge1xuICAgICAgICBjYXNlIFwiY2VudGVyXCI6XG4gICAgICAgICAgdGFyZ2V0UG9zaXRpb24gLT0gY29udGFpbmVyU2l6ZSAvIDIgLSBpdGVtU2l6ZSAvIDI7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgXCJlbmRcIjpcbiAgICAgICAgICB0YXJnZXRQb3NpdGlvbiAtPSBjb250YWluZXJTaXplIC0gaXRlbVNpemU7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICB9XG5cbiAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgLy8gICBgW1Njcm9sbGluZ10gVGFyZ2V0IHBvc2l0aW9uOiAke3RhcmdldFBvc2l0aW9ufSwgaXNDb21wcmVzc2VkOiAke2lzQ29tcHJlc3NlZH1gXG4gICAgICAvLyApO1xuXG4gICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgIC8vICAgYFtTY3JvbGxpbmddIFNjcm9sbFRvSW5kZXg6IGluZGV4PSR7aW5kZXh9LCBwb3NpdGlvbj0ke3RhcmdldFBvc2l0aW9ufSwgYWxpZ25tZW50PSR7YWxpZ25tZW50fWBcbiAgICAgIC8vICk7XG5cbiAgICAgIHNjcm9sbFRvUG9zaXRpb24odGFyZ2V0UG9zaXRpb24sIFwic2Nyb2xsVG9JbmRleFwiKTtcbiAgICB9O1xuXG4gICAgLy8gU2Nyb2xsIHRvIGEgc3BlY2lmaWMgcGFnZVxuICAgIGNvbnN0IHNjcm9sbFRvUGFnZSA9IChcbiAgICAgIHBhZ2U6IG51bWJlcixcbiAgICAgIGxpbWl0OiBudW1iZXIgPSAyMCxcbiAgICAgIGFsaWdubWVudDogXCJzdGFydFwiIHwgXCJjZW50ZXJcIiB8IFwiZW5kXCIgPSBcInN0YXJ0XCIsXG4gICAgKSA9PiB7XG4gICAgICAvLyBWYWxpZGF0ZSBhbGlnbm1lbnQgcGFyYW1ldGVyXG4gICAgICBpZiAoXG4gICAgICAgIHR5cGVvZiBhbGlnbm1lbnQgIT09IFwic3RyaW5nXCIgfHxcbiAgICAgICAgIVtcInN0YXJ0XCIsIFwiY2VudGVyXCIsIFwiZW5kXCJdLmluY2x1ZGVzKGFsaWdubWVudClcbiAgICAgICkge1xuICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgYFtTY3JvbGxpbmddIEludmFsaWQgYWxpZ25tZW50IFwiJHthbGlnbm1lbnR9XCIsIHVzaW5nIFwic3RhcnRcImAsXG4gICAgICAgICk7XG4gICAgICAgIGFsaWdubWVudCA9IFwic3RhcnRcIjtcbiAgICAgIH1cblxuICAgICAgLy8gQ2hlY2sgaWYgd2UncmUgaW4gY3Vyc29yIG1vZGVcbiAgICAgIGNvbnN0IHZpZXdwb3J0Q29uZmlnID0gKGNvbXBvbmVudCBhcyBhbnkpLmNvbmZpZztcbiAgICAgIGNvbnN0IGlzQ3Vyc29yTW9kZSA9IHZpZXdwb3J0Q29uZmlnPy5wYWdpbmF0aW9uPy5zdHJhdGVneSA9PT0gXCJjdXJzb3JcIjtcblxuICAgICAgaWYgKGlzQ3Vyc29yTW9kZSkge1xuICAgICAgICAvLyBJbiBjdXJzb3IgbW9kZSwgY2hlY2sgaWYgd2UgY2FuIHNjcm9sbCB0byB0aGlzIHBhZ2VcbiAgICAgICAgY29uc3QgY29sbGVjdGlvbiA9IChjb21wb25lbnQudmlld3BvcnQgYXMgYW55KS5jb2xsZWN0aW9uO1xuICAgICAgICBpZiAoY29sbGVjdGlvbikge1xuICAgICAgICAgIGNvbnN0IGhpZ2hlc3RMb2FkZWRQYWdlID0gTWF0aC5mbG9vcihcbiAgICAgICAgICAgIGNvbGxlY3Rpb24uZ2V0TG9hZGVkUmFuZ2VzKCkuc2l6ZSxcbiAgICAgICAgICApO1xuXG4gICAgICAgICAgaWYgKHBhZ2UgPiBoaWdoZXN0TG9hZGVkUGFnZSArIDEpIHtcbiAgICAgICAgICAgIC8vIExpbWl0IGhvdyBtYW55IHBhZ2VzIHdlJ2xsIGxvYWQgYXQgb25jZSB0byBwcmV2ZW50IGV4Y2Vzc2l2ZSBBUEkgY2FsbHNcbiAgICAgICAgICAgIGNvbnN0IG1heFBhZ2VzVG9Mb2FkID0gMTA7IC8vIFJlYXNvbmFibGUgbGltaXRcbiAgICAgICAgICAgIGNvbnN0IHRhcmdldFBhZ2UgPSBNYXRoLm1pbihcbiAgICAgICAgICAgICAgcGFnZSxcbiAgICAgICAgICAgICAgaGlnaGVzdExvYWRlZFBhZ2UgKyBtYXhQYWdlc1RvTG9hZCxcbiAgICAgICAgICAgICk7XG5cbiAgICAgICAgICAgIGNvbnNvbGUud2FybihcbiAgICAgICAgICAgICAgYFtTY3JvbGxpbmddIENhbm5vdCBqdW1wIGRpcmVjdGx5IHRvIHBhZ2UgJHtwYWdlfSBpbiBjdXJzb3IgbW9kZS4gYCArXG4gICAgICAgICAgICAgICAgYFBhZ2VzIG11c3QgYmUgbG9hZGVkIHNlcXVlbnRpYWxseS4gQ3VycmVudCBoaWdoZXN0IHBhZ2U6ICR7aGlnaGVzdExvYWRlZFBhZ2V9LiBgICtcbiAgICAgICAgICAgICAgICBgV2lsbCBsb2FkIHVwIHRvIHBhZ2UgJHt0YXJnZXRQYWdlfWAsXG4gICAgICAgICAgICApO1xuXG4gICAgICAgICAgICAvLyBUcmlnZ2VyIHNlcXVlbnRpYWwgbG9hZGluZyB0byB0aGUgdGFyZ2V0IHBhZ2UgKGxpbWl0ZWQpXG4gICAgICAgICAgICBjb25zdCB0YXJnZXRPZmZzZXQgPSAodGFyZ2V0UGFnZSAtIDEpICogbGltaXQ7XG4gICAgICAgICAgICBjb25zdCBjdXJyZW50T2Zmc2V0ID0gaGlnaGVzdExvYWRlZFBhZ2UgKiBsaW1pdDtcblxuICAgICAgICAgICAgLy8gTG9hZCBwYWdlcyBzZXF1ZW50aWFsbHkgdXAgdG8gdGhlIGxpbWl0ZWQgdGFyZ2V0XG4gICAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgIC8vICAgYFtTY3JvbGxpbmddIEluaXRpYXRpbmcgc2VxdWVudGlhbCBsb2FkIGZyb20gcGFnZSAke1xuICAgICAgICAgICAgLy8gICAgIGhpZ2hlc3RMb2FkZWRQYWdlICsgMVxuICAgICAgICAgICAgLy8gICB9IHRvICR7dGFyZ2V0UGFnZX1gLFxuICAgICAgICAgICAgLy8gKTtcblxuICAgICAgICAgICAgLy8gU2Nyb2xsIHRvIHRoZSBsYXN0IGxvYWRlZCBwb3NpdGlvbiBmaXJzdFxuICAgICAgICAgICAgY29uc3QgbGFzdExvYWRlZEluZGV4ID0gaGlnaGVzdExvYWRlZFBhZ2UgKiBsaW1pdDtcbiAgICAgICAgICAgIHNjcm9sbFRvSW5kZXgobGFzdExvYWRlZEluZGV4LCBhbGlnbm1lbnQpO1xuXG4gICAgICAgICAgICAvLyBUaGUgY29sbGVjdGlvbiBmZWF0dXJlIHdpbGwgaGFuZGxlIHNlcXVlbnRpYWwgbG9hZGluZ1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyBDb252ZXJ0IHBhZ2UgdG8gaW5kZXggKHBhZ2UgMSA9IGluZGV4IDApXG4gICAgICBjb25zdCBpbmRleCA9IChwYWdlIC0gMSkgKiBsaW1pdDtcblxuICAgICAgLy8gY29uc29sZS5sb2coXG4gICAgICAvLyAgIGBbU2Nyb2xsaW5nXSBTY3JvbGxUb1BhZ2U6IHBhZ2U9JHtwYWdlfSwgbGltaXQ9JHtsaW1pdH0sIHRhcmdldEluZGV4PSR7aW5kZXh9LCBhbGlnbm1lbnQ9JHthbGlnbm1lbnR9YFxuICAgICAgLy8gKTtcblxuICAgICAgLy8gSnVzdCBzY3JvbGwgdG8gdGhlIGluZGV4IC0gbGV0IHRoZSBub3JtYWwgcmVuZGVyaW5nIGZsb3cgaGFuZGxlIGRhdGEgbG9hZGluZyBhbmQgcGxhY2Vob2xkZXJzXG4gICAgICBzY3JvbGxUb0luZGV4KGluZGV4LCBhbGlnbm1lbnQpO1xuICAgIH07XG5cbiAgICAvLyBVcGRhdGUgc2Nyb2xsIGJvdW5kc1xuICAgIGNvbnN0IHVwZGF0ZVNjcm9sbEJvdW5kcyA9IChcbiAgICAgIG5ld1RvdGFsU2l6ZTogbnVtYmVyLFxuICAgICAgbmV3Q29udGFpbmVyU2l6ZTogbnVtYmVyLFxuICAgICkgPT4ge1xuICAgICAgdG90YWxWaXJ0dWFsU2l6ZSA9IG5ld1RvdGFsU2l6ZTtcbiAgICAgIGNvbnRhaW5lclNpemUgPSBuZXdDb250YWluZXJTaXplO1xuXG4gICAgICBpZiAodmlld3BvcnRTdGF0ZSkge1xuICAgICAgICB2aWV3cG9ydFN0YXRlLnZpcnR1YWxUb3RhbFNpemUgPSBuZXdUb3RhbFNpemU7XG4gICAgICAgIHZpZXdwb3J0U3RhdGUuY29udGFpbmVyU2l6ZSA9IG5ld0NvbnRhaW5lclNpemU7XG4gICAgICB9XG5cbiAgICAgIC8vIERvbid0IGNsYW1wIHNjcm9sbCBwb3NpdGlvbiB1bnRpbCB3ZSBoYXZlIHJlYWwgZGF0YSBsb2FkZWRcbiAgICAgIC8vIFRoaXMgcHJldmVudHMgcmVzZXR0aW5nIGluaXRpYWxTY3JvbGxJbmRleCBwb3NpdGlvbiBiZWZvcmUgZGF0YSBhcnJpdmVzXG4gICAgICAvLyBDaGVjayB0b3RhbEl0ZW1zIGluc3RlYWQgb2YgdG90YWxWaXJ0dWFsU2l6ZSBzaW5jZSB2aXJ0dWFsU2l6ZSBjYW4gYmUgbm9uLXplcm8gZnJvbSBwYWRkaW5nXG4gICAgICBjb25zdCB0b3RhbEl0ZW1zID0gdmlld3BvcnRTdGF0ZT8udG90YWxJdGVtcyB8fCAwO1xuICAgICAgaWYgKHRvdGFsSXRlbXMgPD0gMCkge1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG5cbiAgICAgIC8vIENsYW1wIGN1cnJlbnQgcG9zaXRpb24gdG8gbmV3IGJvdW5kc1xuICAgICAgY29uc3QgbWF4U2Nyb2xsID0gTWF0aC5tYXgoMCwgdG90YWxWaXJ0dWFsU2l6ZSAtIGNvbnRhaW5lclNpemUpO1xuICAgICAgaWYgKHNjcm9sbFBvc2l0aW9uID4gbWF4U2Nyb2xsKSB7XG4gICAgICAgIHNjcm9sbFRvUG9zaXRpb24obWF4U2Nyb2xsKTtcbiAgICAgIH1cbiAgICB9O1xuXG4gICAgLy8gRXh0ZW5kIHZpZXdwb3J0IEFQSVxuICAgIGNvbnN0IG9yaWdpbmFsU2Nyb2xsVG9JbmRleCA9IGNvbXBvbmVudC52aWV3cG9ydC5zY3JvbGxUb0luZGV4O1xuICAgIGNvbXBvbmVudC52aWV3cG9ydC5zY3JvbGxUb0luZGV4ID0gKFxuICAgICAgaW5kZXg6IG51bWJlcixcbiAgICAgIGFsaWdubWVudD86IFwic3RhcnRcIiB8IFwiY2VudGVyXCIgfCBcImVuZFwiLFxuICAgICkgPT4ge1xuICAgICAgc2Nyb2xsVG9JbmRleChpbmRleCwgYWxpZ25tZW50KTtcbiAgICAgIG9yaWdpbmFsU2Nyb2xsVG9JbmRleD8uKGluZGV4LCBhbGlnbm1lbnQpO1xuICAgIH07XG5cbiAgICAvLyBBZGQgc2Nyb2xsVG9QYWdlIHRvIHZpZXdwb3J0IEFQSSAobmV3IG1ldGhvZCwgbm8gb3JpZ2luYWwgdG8gcHJlc2VydmUpXG4gICAgKGNvbXBvbmVudC52aWV3cG9ydCBhcyBhbnkpLnNjcm9sbFRvUGFnZSA9IChcbiAgICAgIHBhZ2U6IG51bWJlcixcbiAgICAgIGxpbWl0PzogbnVtYmVyLFxuICAgICAgYWxpZ25tZW50PzogXCJzdGFydFwiIHwgXCJjZW50ZXJcIiB8IFwiZW5kXCIsXG4gICAgKSA9PiB7XG4gICAgICBzY3JvbGxUb1BhZ2UocGFnZSwgbGltaXQsIGFsaWdubWVudCk7XG4gICAgfTtcblxuICAgIGNvbnN0IG9yaWdpbmFsU2Nyb2xsVG9Qb3NpdGlvbiA9IGNvbXBvbmVudC52aWV3cG9ydC5zY3JvbGxUb1Bvc2l0aW9uO1xuICAgIGNvbXBvbmVudC52aWV3cG9ydC5zY3JvbGxUb1Bvc2l0aW9uID0gKHBvc2l0aW9uOiBudW1iZXIpID0+IHtcbiAgICAgIHNjcm9sbFRvUG9zaXRpb24ocG9zaXRpb24sIFwiYXBpXCIpO1xuICAgICAgb3JpZ2luYWxTY3JvbGxUb1Bvc2l0aW9uPy4ocG9zaXRpb24pO1xuICAgIH07XG5cbiAgICBjb25zdCBvcmlnaW5hbEdldFNjcm9sbFBvc2l0aW9uID0gY29tcG9uZW50LnZpZXdwb3J0LmdldFNjcm9sbFBvc2l0aW9uO1xuICAgIGNvbXBvbmVudC52aWV3cG9ydC5nZXRTY3JvbGxQb3NpdGlvbiA9ICgpID0+IHtcbiAgICAgIHJldHVybiBzY3JvbGxQb3NpdGlvbjtcbiAgICB9O1xuXG4gICAgLy8gQWRkIHdoZWVsIGV2ZW50IGxpc3RlbmVyIG9uIGluaXRpYWxpemF0aW9uXG4gICAgLy8gVGhpcyBibG9jayBpcyByZW1vdmVkIGFzIHBlciB0aGUgZWRpdCBoaW50LlxuXG4gICAgLy8gQ2xlYW4gdXAgb24gZGVzdHJveVxuICAgIGlmIChcImRlc3Ryb3lcIiBpbiBjb21wb25lbnQgJiYgdHlwZW9mIGNvbXBvbmVudC5kZXN0cm95ID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgIGNvbnN0IG9yaWdpbmFsRGVzdHJveSA9IGNvbXBvbmVudC5kZXN0cm95O1xuICAgICAgY29tcG9uZW50LmRlc3Ryb3kgPSAoKSA9PiB7XG4gICAgICAgIC8vIFJlbW92ZSBldmVudCBsaXN0ZW5lcnNcbiAgICAgICAgY29uc3Qgdmlld3BvcnRFbGVtZW50ID0gKGNvbXBvbmVudCBhcyBhbnkpLl9zY3JvbGxpbmdWaWV3cG9ydEVsZW1lbnQ7XG4gICAgICAgIGlmICh2aWV3cG9ydEVsZW1lbnQpIHtcbiAgICAgICAgICB2aWV3cG9ydEVsZW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcIndoZWVsXCIsIGhhbmRsZVdoZWVsKTtcbiAgICAgICAgICBpZiAoKGNvbXBvbmVudCBhcyBhbnkpLl9zY3JvbGxpbmdTdG9wT25DbGljaykge1xuICAgICAgICAgICAgdmlld3BvcnRFbGVtZW50LnJlbW92ZUV2ZW50TGlzdGVuZXIoXCJtb3VzZWRvd25cIiwgaGFuZGxlTW91c2VEb3duKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyBDbGVhciB0aW1lb3V0c1xuICAgICAgICBpZiAoaWRsZVRpbWVvdXRJZCkge1xuICAgICAgICAgIGNsZWFyVGltZW91dChpZGxlVGltZW91dElkKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHN0b3BJZGxlRGV0ZWN0aW9uKCk7XG4gICAgICAgIG9yaWdpbmFsRGVzdHJveT8uKCk7XG4gICAgICB9O1xuICAgIH1cblxuICAgIC8vIEFkZCBzY3JvbGxCeSBtZXRob2RcbiAgICBjb25zdCBzY3JvbGxCeSA9IChkZWx0YTogbnVtYmVyKSA9PiB7XG4gICAgICBjb25zdCBwcmV2aW91c1Bvc2l0aW9uID0gc2Nyb2xsUG9zaXRpb247XG4gICAgICBjb25zdCBtYXhTY3JvbGwgPSBNYXRoLm1heCgwLCB0b3RhbFZpcnR1YWxTaXplIC0gY29udGFpbmVyU2l6ZSk7XG4gICAgICBjb25zdCBuZXdQb3NpdGlvbiA9IGNsYW1wKHNjcm9sbFBvc2l0aW9uICsgZGVsdGEsIDAsIG1heFNjcm9sbCk7XG5cbiAgICAgIGlmIChuZXdQb3NpdGlvbiAhPT0gcHJldmlvdXNQb3NpdGlvbikge1xuICAgICAgICBzY3JvbGxQb3NpdGlvbiA9IG5ld1Bvc2l0aW9uO1xuXG4gICAgICAgIC8vIFVwZGF0ZSBzcGVlZCB0cmFja2VyXG4gICAgICAgIHNwZWVkVHJhY2tlciA9IHVwZGF0ZVNwZWVkVHJhY2tlcihcbiAgICAgICAgICBzcGVlZFRyYWNrZXIsXG4gICAgICAgICAgc2Nyb2xsUG9zaXRpb24sXG4gICAgICAgICAgcHJldmlvdXNQb3NpdGlvbixcbiAgICAgICAgKTtcblxuICAgICAgICAvLyBVcGRhdGUgdmlld3BvcnQgc3RhdGVcbiAgICAgICAgaWYgKHZpZXdwb3J0U3RhdGUpIHtcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLnNjcm9sbFBvc2l0aW9uID0gc2Nyb2xsUG9zaXRpb247XG4gICAgICAgICAgdmlld3BvcnRTdGF0ZS52ZWxvY2l0eSA9IHNwZWVkVHJhY2tlci52ZWxvY2l0eTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFVwZGF0ZSBzY3JvbGxlZCBzdGF0ZSBjbGFzc1xuICAgICAgICBjb25zdCB2aWV3cG9ydEVsID1cbiAgICAgICAgICB2aWV3cG9ydFN0YXRlPy52aWV3cG9ydEVsZW1lbnQgfHwgKGNvbXBvbmVudCBhcyBhbnkpLnZpZXdwb3J0RWxlbWVudDtcbiAgICAgICAgaWYgKHZpZXdwb3J0RWwpIHtcbiAgICAgICAgICB1cGRhdGVTY3JvbGxlZFN0YXRlKHZpZXdwb3J0RWwsIHNjcm9sbFBvc2l0aW9uKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIEVtaXQgc2Nyb2xsIGV2ZW50XG4gICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpzY3JvbGxcIiwge1xuICAgICAgICAgIHBvc2l0aW9uOiBzY3JvbGxQb3NpdGlvbixcbiAgICAgICAgICB2ZWxvY2l0eTogc3BlZWRUcmFja2VyLnZlbG9jaXR5LFxuICAgICAgICAgIGRpcmVjdGlvbjogc3BlZWRUcmFja2VyLmRpcmVjdGlvbixcbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gVHJpZ2dlciByZW5kZXJcbiAgICAgICAgY29tcG9uZW50LnZpZXdwb3J0LnJlbmRlckl0ZW1zPy4oKTtcblxuICAgICAgICAvLyBTdGFydCBpZGxlIGRldGVjdGlvbiBpZiBub3Qgc2Nyb2xsaW5nXG4gICAgICAgIGlmICghaXNTY3JvbGxpbmcpIHtcbiAgICAgICAgICBpc1Njcm9sbGluZyA9IHRydWU7XG4gICAgICAgICAgc3RhcnRJZGxlRGV0ZWN0aW9uKCk7XG4gICAgICAgIH1cbiAgICAgICAgbGFzdFNjcm9sbFRpbWUgPSBEYXRlLm5vdygpO1xuICAgICAgfVxuICAgIH07XG5cbiAgICAvLyBFeHBvc2Ugc2Nyb2xsaW5nIHN0YXRlIGZvciBvdGhlciBmZWF0dXJlc1xuICAgIChjb21wb25lbnQudmlld3BvcnQgYXMgYW55KS5zY3JvbGxpbmdTdGF0ZSA9IHtcbiAgICAgIHNldFZlbG9jaXR5VG9aZXJvLFxuICAgIH07XG5cbiAgICAvLyBBZGQgc2Nyb2xsQnkgdG8gdmlld3BvcnQgQVBJXG4gICAgY29tcG9uZW50LnZpZXdwb3J0LnNjcm9sbEJ5ID0gc2Nyb2xsQnk7XG4gICAgY29tcG9uZW50LnZpZXdwb3J0LmdldFZlbG9jaXR5ID0gKCkgPT4gc3BlZWRUcmFja2VyLnZlbG9jaXR5O1xuXG4gICAgLy8gRXhwb3NlIHNjcm9sbGluZyBBUElcbiAgICAoY29tcG9uZW50IGFzIGFueSkuc2Nyb2xsaW5nID0ge1xuICAgICAgaGFuZGxlV2hlZWwsXG4gICAgICBzY3JvbGxUb1Bvc2l0aW9uLFxuICAgICAgc2Nyb2xsVG9JbmRleCxcbiAgICAgIHNjcm9sbFRvUGFnZSxcbiAgICAgIHNjcm9sbEJ5LFxuICAgICAgZ2V0U2Nyb2xsUG9zaXRpb246ICgpID0+IHNjcm9sbFBvc2l0aW9uLFxuICAgICAgdXBkYXRlU2Nyb2xsQm91bmRzLFxuICAgICAgZ2V0VmVsb2NpdHk6ICgpID0+IHNwZWVkVHJhY2tlci52ZWxvY2l0eSxcbiAgICAgIGdldERpcmVjdGlvbjogKCkgPT4gc3BlZWRUcmFja2VyLmRpcmVjdGlvbixcbiAgICAgIGlzU2Nyb2xsaW5nOiAoKSA9PiBpc1Njcm9sbGluZyxcbiAgICB9O1xuXG4gICAgcmV0dXJuIGNvbXBvbmVudDtcbiAgfTtcbn07XG4iLAogICAgIi8vIHNyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL21vbWVudHVtLnRzXG5cbi8qKlxuICogTW9tZW50dW0gRmVhdHVyZSAtIEFkZHMgaW5lcnRpYWwgc2Nyb2xsaW5nIHRvIHZpZXdwb3J0XG4gKiBQcm92aWRlcyBzbW9vdGggZGVjZWxlcmF0aW9uIGFmdGVyIHRvdWNoL21vdXNlIGRyYWcgZ2VzdHVyZXNcbiAqL1xuXG5pbXBvcnQgdHlwZSB7IFZpZXdwb3J0Q29tcG9uZW50LCBWaWV3cG9ydENvbnRleHQgfSBmcm9tIFwiLi4vdHlwZXNcIjtcbmltcG9ydCB7IFZJRVdQT1JUX0NPTlNUQU5UUyB9IGZyb20gXCIuLi9jb25zdGFudHNcIjtcblxuaW50ZXJmYWNlIE1vbWVudHVtQ29uZmlnIHtcbiAgZW5hYmxlZD86IGJvb2xlYW47XG4gIGRlY2VsZXJhdGlvbj86IG51bWJlcjtcbiAgbWluVmVsb2NpdHk/OiBudW1iZXI7XG4gIG1pbkR1cmF0aW9uPzogbnVtYmVyO1xuICBtaW5WZWxvY2l0eVRocmVzaG9sZD86IG51bWJlcjtcbiAgLyoqIFN0b3AgbW9tZW50dW0gd2hlbiBjbGlja2luZyBvbiB0aGUgdmlld3BvcnQgKGRlZmF1bHQ6IHRydWUpICovXG4gIHN0b3BPbkNsaWNrPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBBZGRzIG1vbWVudHVtIHNjcm9sbGluZyB0byB2aWV3cG9ydFxuICovXG5leHBvcnQgY29uc3Qgd2l0aE1vbWVudHVtID0gKGNvbmZpZzogTW9tZW50dW1Db25maWcgPSB7fSkgPT4ge1xuICByZXR1cm4gPFQgZXh0ZW5kcyBWaWV3cG9ydENvbnRleHQgJiBWaWV3cG9ydENvbXBvbmVudD4oY29tcG9uZW50OiBUKTogVCA9PiB7XG4gICAgY29uc3Qge1xuICAgICAgZW5hYmxlZCA9IFZJRVdQT1JUX0NPTlNUQU5UUy5NT01FTlRVTS5FTkFCTEVELFxuICAgICAgZGVjZWxlcmF0aW9uID0gVklFV1BPUlRfQ09OU1RBTlRTLk1PTUVOVFVNLkRFQ0VMRVJBVElPTl9GQUNUT1IsXG4gICAgICBtaW5WZWxvY2l0eSA9IFZJRVdQT1JUX0NPTlNUQU5UUy5NT01FTlRVTS5NSU5fVkVMT0NJVFksXG4gICAgICBtaW5EdXJhdGlvbiA9IFZJRVdQT1JUX0NPTlNUQU5UUy5NT01FTlRVTS5NSU5fRFVSQVRJT04sXG4gICAgICBtaW5WZWxvY2l0eVRocmVzaG9sZCA9IFZJRVdQT1JUX0NPTlNUQU5UUy5NT01FTlRVTS5NSU5fVkVMT0NJVFlfVEhSRVNIT0xELFxuICAgICAgc3RvcE9uQ2xpY2sgPSB0cnVlLCAvLyBTdG9wIG1vbWVudHVtIG9uIGNsaWNrIGJ5IGRlZmF1bHRcbiAgICB9ID0gY29uZmlnO1xuXG4gICAgLy8gQWx3YXlzIGFwcGx5IHRoZSBmZWF0dXJlLCBidXQgY2hlY2sgaXNFbmFibGVkIGF0IHJ1bnRpbWVcblxuICAgIC8vIE1vbWVudHVtIHN0YXRlXG4gICAgbGV0IG1vbWVudHVtQW5pbWF0aW9uSWQ6IG51bWJlciB8IG51bGwgPSBudWxsO1xuICAgIGxldCBpc1RvdWNoaW5nID0gZmFsc2U7XG4gICAgbGV0IGlzTW91c2VEcmFnZ2luZyA9IGZhbHNlO1xuICAgIGxldCBsYXN0VG91Y2hQb3NpdGlvbiA9IDA7XG4gICAgbGV0IGxhc3RNb3VzZVBvc2l0aW9uID0gMDtcbiAgICBsZXQgdG91Y2hTdGFydFRpbWUgPSAwO1xuICAgIGxldCB2aWV3cG9ydFN0YXRlOiBhbnk7XG4gICAgbGV0IHNjcm9sbGluZ1N0YXRlOiBhbnk7XG4gICAgbGV0IGlzRW5hYmxlZCA9IGVuYWJsZWQ7IC8vIFRyYWNrIGN1cnJlbnQgZW5hYmxlZCBzdGF0ZVxuXG4gICAgLy8gR2V0IHJlZmVyZW5jZXMgYWZ0ZXIgaW5pdGlhbGl6YXRpb25cbiAgICBjb25zdCBvcmlnaW5hbEluaXRpYWxpemUgPSBjb21wb25lbnQudmlld3BvcnQuaW5pdGlhbGl6ZTtcbiAgICBjb21wb25lbnQudmlld3BvcnQuaW5pdGlhbGl6ZSA9ICgpID0+IHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IG9yaWdpbmFsSW5pdGlhbGl6ZSgpO1xuICAgICAgLy8gU2tpcCBpZiBhbHJlYWR5IGluaXRpYWxpemVkXG4gICAgICBpZiAocmVzdWx0ID09PSBmYWxzZSkge1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICB9XG5cbiAgICAgIC8vIEdldCB2aWV3cG9ydCBhbmQgc2Nyb2xsaW5nIHN0YXRlc1xuICAgICAgdmlld3BvcnRTdGF0ZSA9IChjb21wb25lbnQudmlld3BvcnQgYXMgYW55KS5zdGF0ZTtcbiAgICAgIHNjcm9sbGluZ1N0YXRlID0gKGNvbXBvbmVudC52aWV3cG9ydCBhcyBhbnkpLnNjcm9sbGluZ1N0YXRlO1xuICAgICAgcmV0dXJuIHJlc3VsdDtcbiAgICB9O1xuXG4gICAgLy8gU3RvcCBtb21lbnR1bSBhbmltYXRpb24gKGNhbiBiZSBjYWxsZWQgZXh0ZXJuYWxseSlcbiAgICBjb25zdCBzdG9wTW9tZW50dW0gPSAoKSA9PiB7XG4gICAgICBpZiAobW9tZW50dW1BbmltYXRpb25JZCkge1xuICAgICAgICBjYW5jZWxBbmltYXRpb25GcmFtZShtb21lbnR1bUFuaW1hdGlvbklkKTtcbiAgICAgICAgbW9tZW50dW1BbmltYXRpb25JZCA9IG51bGw7XG4gICAgICAgIC8vIFJlc2V0IHZlbG9jaXR5IGluIHNjcm9sbGluZyBmZWF0dXJlXG4gICAgICAgIGlmIChzY3JvbGxpbmdTdGF0ZSkge1xuICAgICAgICAgIHNjcm9sbGluZ1N0YXRlLnNldFZlbG9jaXR5VG9aZXJvPy4oKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH07XG5cbiAgICAvLyBTdGFydCBtb21lbnR1bSBhbmltYXRpb25cbiAgICBjb25zdCBzdGFydE1vbWVudHVtID0gKGluaXRpYWxWZWxvY2l0eTogbnVtYmVyKSA9PiB7XG4gICAgICAvLyBDYW5jZWwgYW55IGV4aXN0aW5nIG1vbWVudHVtXG4gICAgICBpZiAobW9tZW50dW1BbmltYXRpb25JZCkge1xuICAgICAgICBjYW5jZWxBbmltYXRpb25GcmFtZShtb21lbnR1bUFuaW1hdGlvbklkKTtcbiAgICAgIH1cblxuICAgICAgbGV0IHZlbG9jaXR5ID0gaW5pdGlhbFZlbG9jaXR5O1xuXG4gICAgICBjb25zdCBhbmltYXRlID0gKCkgPT4ge1xuICAgICAgICAvLyBBcHBseSBkZWNlbGVyYXRpb25cbiAgICAgICAgdmVsb2NpdHkgKj0gZGVjZWxlcmF0aW9uO1xuXG4gICAgICAgIC8vIFN0b3AgaWYgdmVsb2NpdHkgaXMgdG9vIHNtYWxsXG4gICAgICAgIGlmIChNYXRoLmFicyh2ZWxvY2l0eSkgPCBtaW5WZWxvY2l0eSkge1xuICAgICAgICAgIG1vbWVudHVtQW5pbWF0aW9uSWQgPSBudWxsO1xuICAgICAgICAgIC8vIFJlc2V0IHZlbG9jaXR5IGluIHNjcm9sbGluZyBmZWF0dXJlXG4gICAgICAgICAgaWYgKHNjcm9sbGluZ1N0YXRlKSB7XG4gICAgICAgICAgICBzY3JvbGxpbmdTdGF0ZS5zZXRWZWxvY2l0eVRvWmVybz8uKCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIENhbGN1bGF0ZSBzY3JvbGwgZGVsdGEgKHZlbG9jaXR5IGlzIGluIHB4L21zLCBzbyBtdWx0aXBseSBieSBmcmFtZSB0aW1lKVxuICAgICAgICBjb25zdCBmcmFtZURlbHRhID0gdmVsb2NpdHkgKiBWSUVXUE9SVF9DT05TVEFOVFMuTU9NRU5UVU0uRlJBTUVfVElNRTtcblxuICAgICAgICAvLyBVc2Ugc2Nyb2xsaW5nIGZlYXR1cmUncyBBUEkgdG8gdXBkYXRlIHBvc2l0aW9uXG4gICAgICAgIGlmIChjb21wb25lbnQudmlld3BvcnQuc2Nyb2xsQnkpIHtcbiAgICAgICAgICBjb21wb25lbnQudmlld3BvcnQuc2Nyb2xsQnkoZnJhbWVEZWx0YSk7XG5cbiAgICAgICAgICAvLyBDb250aW51ZSBhbmltYXRpb25cbiAgICAgICAgICBtb21lbnR1bUFuaW1hdGlvbklkID0gcmVxdWVzdEFuaW1hdGlvbkZyYW1lKGFuaW1hdGUpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIE5vIHNjcm9sbEJ5IG1ldGhvZCwgc3RvcCBtb21lbnR1bVxuICAgICAgICAgIG1vbWVudHVtQW5pbWF0aW9uSWQgPSBudWxsO1xuICAgICAgICB9XG4gICAgICB9O1xuXG4gICAgICBtb21lbnR1bUFuaW1hdGlvbklkID0gcmVxdWVzdEFuaW1hdGlvbkZyYW1lKGFuaW1hdGUpO1xuICAgIH07XG5cbiAgICAvLyBUb3VjaCBldmVudCBoYW5kbGVyc1xuICAgIGNvbnN0IGhhbmRsZVRvdWNoU3RhcnQgPSAoZTogVG91Y2hFdmVudCkgPT4ge1xuICAgICAgaXNUb3VjaGluZyA9IHRydWU7XG4gICAgICBjb25zdCB0b3VjaCA9IGUudG91Y2hlc1swXTtcbiAgICAgIGNvbnN0IG9yaWVudGF0aW9uID0gdmlld3BvcnRTdGF0ZT8ub3JpZW50YXRpb24gfHwgXCJ2ZXJ0aWNhbFwiO1xuICAgICAgbGFzdFRvdWNoUG9zaXRpb24gPVxuICAgICAgICBvcmllbnRhdGlvbiA9PT0gXCJ2ZXJ0aWNhbFwiID8gdG91Y2guY2xpZW50WSA6IHRvdWNoLmNsaWVudFg7XG4gICAgICB0b3VjaFN0YXJ0VGltZSA9IERhdGUubm93KCk7XG5cbiAgICAgIC8vIENhbmNlbCBhbnkgb25nb2luZyBtb21lbnR1bVxuICAgICAgaWYgKG1vbWVudHVtQW5pbWF0aW9uSWQpIHtcbiAgICAgICAgY2FuY2VsQW5pbWF0aW9uRnJhbWUobW9tZW50dW1BbmltYXRpb25JZCk7XG4gICAgICAgIG1vbWVudHVtQW5pbWF0aW9uSWQgPSBudWxsO1xuICAgICAgfVxuICAgIH07XG5cbiAgICBjb25zdCBoYW5kbGVUb3VjaE1vdmUgPSAoZTogVG91Y2hFdmVudCkgPT4ge1xuICAgICAgaWYgKCFpc1RvdWNoaW5nKSByZXR1cm47XG5cbiAgICAgIGUucHJldmVudERlZmF1bHQoKTsgLy8gUHJldmVudCBuYXRpdmUgc2Nyb2xsaW5nXG5cbiAgICAgIGNvbnN0IHRvdWNoID0gZS50b3VjaGVzWzBdO1xuICAgICAgY29uc3Qgb3JpZW50YXRpb24gPSB2aWV3cG9ydFN0YXRlPy5vcmllbnRhdGlvbiB8fCBcInZlcnRpY2FsXCI7XG4gICAgICBjb25zdCBjdXJyZW50UG9zaXRpb24gPVxuICAgICAgICBvcmllbnRhdGlvbiA9PT0gXCJ2ZXJ0aWNhbFwiID8gdG91Y2guY2xpZW50WSA6IHRvdWNoLmNsaWVudFg7XG4gICAgICBjb25zdCBkZWx0YSA9IGxhc3RUb3VjaFBvc2l0aW9uIC0gY3VycmVudFBvc2l0aW9uOyAvLyBJbnZlcnRlZCBmb3IgbmF0dXJhbCBzY3JvbGxpbmdcbiAgICAgIGxhc3RUb3VjaFBvc2l0aW9uID0gY3VycmVudFBvc2l0aW9uO1xuXG4gICAgICAvLyBVc2Ugc2Nyb2xsaW5nIGZlYXR1cmUncyBBUEkgKDE6MSB0b3VjaC10by1zY3JvbGwgZm9yIG5hdHVyYWwgZmVlbClcbiAgICAgIGlmIChjb21wb25lbnQudmlld3BvcnQuc2Nyb2xsQnkpIHtcbiAgICAgICAgY29tcG9uZW50LnZpZXdwb3J0LnNjcm9sbEJ5KGRlbHRhKTtcbiAgICAgIH1cbiAgICB9O1xuXG4gICAgY29uc3QgaGFuZGxlVG91Y2hFbmQgPSAoKSA9PiB7XG4gICAgICBpZiAoIWlzVG91Y2hpbmcpIHJldHVybjtcbiAgICAgIGlzVG91Y2hpbmcgPSBmYWxzZTtcblxuICAgICAgY29uc3QgdG91Y2hFbmRUaW1lID0gRGF0ZS5ub3coKTtcbiAgICAgIGNvbnN0IHRvdWNoRHVyYXRpb24gPSB0b3VjaEVuZFRpbWUgLSB0b3VjaFN0YXJ0VGltZTtcblxuICAgICAgLy8gR2V0IGN1cnJlbnQgdmVsb2NpdHkgZnJvbSBzY3JvbGxpbmcgZmVhdHVyZVxuICAgICAgY29uc3QgdmVsb2NpdHkgPSBjb21wb25lbnQudmlld3BvcnQuZ2V0VmVsb2NpdHk/LigpIHx8IDA7XG5cbiAgICAgIC8vIE9ubHkgc3RhcnQgbW9tZW50dW0gaWYgdGhlIHRvdWNoIHdhcyBxdWljayBlbm91Z2ggYW5kIHdlIGhhdmUgdmVsb2NpdHlcbiAgICAgIGlmIChcbiAgICAgICAgdG91Y2hEdXJhdGlvbiA8IG1pbkR1cmF0aW9uICYmXG4gICAgICAgIE1hdGguYWJzKHZlbG9jaXR5KSA+IG1pblZlbG9jaXR5VGhyZXNob2xkXG4gICAgICApIHtcbiAgICAgICAgc3RhcnRNb21lbnR1bSh2ZWxvY2l0eSk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIEhhbmRsZSBjbGljayB0byBzdG9wIG1vbWVudHVtICh3aXRob3V0IHByZXZlbnRpbmcgc2VsZWN0aW9uKVxuICAgIGNvbnN0IGhhbmRsZUNsaWNrID0gKF9lOiBNb3VzZUV2ZW50KSA9PiB7XG4gICAgICBzdG9wTW9tZW50dW0oKTtcbiAgICB9O1xuXG4gICAgLy8gTW91c2UgZXZlbnQgaGFuZGxlcnMgZm9yIGRlc2t0b3BcbiAgICBjb25zdCBoYW5kbGVNb3VzZURvd24gPSAoZTogTW91c2VFdmVudCkgPT4ge1xuICAgICAgaXNNb3VzZURyYWdnaW5nID0gdHJ1ZTtcbiAgICAgIGNvbnN0IG9yaWVudGF0aW9uID0gdmlld3BvcnRTdGF0ZT8ub3JpZW50YXRpb24gfHwgXCJ2ZXJ0aWNhbFwiO1xuICAgICAgbGFzdE1vdXNlUG9zaXRpb24gPSBvcmllbnRhdGlvbiA9PT0gXCJ2ZXJ0aWNhbFwiID8gZS5jbGllbnRZIDogZS5jbGllbnRYO1xuICAgICAgdG91Y2hTdGFydFRpbWUgPSBEYXRlLm5vdygpO1xuXG4gICAgICAvLyBDYW5jZWwgYW55IG9uZ29pbmcgbW9tZW50dW1cbiAgICAgIHN0b3BNb21lbnR1bSgpO1xuXG4gICAgICAvLyBQcmV2ZW50IHRleHQgc2VsZWN0aW9uXG4gICAgICBlLnByZXZlbnREZWZhdWx0KCk7XG4gICAgfTtcblxuICAgIGNvbnN0IGhhbmRsZU1vdXNlTW92ZSA9IChlOiBNb3VzZUV2ZW50KSA9PiB7XG4gICAgICBpZiAoIWlzTW91c2VEcmFnZ2luZykgcmV0dXJuO1xuXG4gICAgICBjb25zdCBvcmllbnRhdGlvbiA9IHZpZXdwb3J0U3RhdGU/Lm9yaWVudGF0aW9uIHx8IFwidmVydGljYWxcIjtcbiAgICAgIGNvbnN0IGN1cnJlbnRQb3NpdGlvbiA9XG4gICAgICAgIG9yaWVudGF0aW9uID09PSBcInZlcnRpY2FsXCIgPyBlLmNsaWVudFkgOiBlLmNsaWVudFg7XG4gICAgICBjb25zdCBkZWx0YSA9IGxhc3RNb3VzZVBvc2l0aW9uIC0gY3VycmVudFBvc2l0aW9uO1xuICAgICAgbGFzdE1vdXNlUG9zaXRpb24gPSBjdXJyZW50UG9zaXRpb247XG5cbiAgICAgIC8vIFVzZSBzY3JvbGxpbmcgZmVhdHVyZSdzIEFQSSAoMToxIGRyYWctdG8tc2Nyb2xsIGZvciBuYXR1cmFsIGZlZWwpXG4gICAgICBpZiAoY29tcG9uZW50LnZpZXdwb3J0LnNjcm9sbEJ5KSB7XG4gICAgICAgIGNvbXBvbmVudC52aWV3cG9ydC5zY3JvbGxCeShkZWx0YSk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIGNvbnN0IGhhbmRsZU1vdXNlVXAgPSAoKSA9PiB7XG4gICAgICBpZiAoIWlzTW91c2VEcmFnZ2luZykgcmV0dXJuO1xuICAgICAgaXNNb3VzZURyYWdnaW5nID0gZmFsc2U7XG5cbiAgICAgIGNvbnN0IG1vdXNlVXBUaW1lID0gRGF0ZS5ub3coKTtcbiAgICAgIGNvbnN0IGRyYWdEdXJhdGlvbiA9IG1vdXNlVXBUaW1lIC0gdG91Y2hTdGFydFRpbWU7XG5cbiAgICAgIC8vIEdldCBjdXJyZW50IHZlbG9jaXR5IGZyb20gc2Nyb2xsaW5nIGZlYXR1cmVcbiAgICAgIGNvbnN0IHZlbG9jaXR5ID0gY29tcG9uZW50LnZpZXdwb3J0LmdldFZlbG9jaXR5Py4oKSB8fCAwO1xuXG4gICAgICAvLyBTdGFydCBtb21lbnR1bSBmb3IgcXVpY2sgZHJhZ3NcbiAgICAgIGlmIChcbiAgICAgICAgZHJhZ0R1cmF0aW9uIDwgbWluRHVyYXRpb24gJiZcbiAgICAgICAgTWF0aC5hYnModmVsb2NpdHkpID4gbWluVmVsb2NpdHlUaHJlc2hvbGRcbiAgICAgICkge1xuICAgICAgICBzdGFydE1vbWVudHVtKHZlbG9jaXR5KTtcbiAgICAgIH1cbiAgICB9O1xuXG4gICAgLy8gT3ZlcnJpZGUgaW5pdGlhbGl6ZSB0byBhZGQgZXZlbnQgbGlzdGVuZXJzXG4gICAgY29uc3Qgb3JpZ2luYWxJbml0ID0gY29tcG9uZW50LnZpZXdwb3J0LmluaXRpYWxpemU7XG4gICAgY29tcG9uZW50LnZpZXdwb3J0LmluaXRpYWxpemUgPSAoKSA9PiB7XG4gICAgICBjb25zdCByZXN1bHQgPSBvcmlnaW5hbEluaXQoKTtcbiAgICAgIC8vIFNraXAgaWYgYWxyZWFkeSBpbml0aWFsaXplZFxuICAgICAgaWYgKHJlc3VsdCA9PT0gZmFsc2UpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCB2aWV3cG9ydEVsZW1lbnQgPVxuICAgICAgICAoY29tcG9uZW50IGFzIGFueSkudmlld3BvcnRFbGVtZW50IHx8XG4gICAgICAgIChjb21wb25lbnQudmlld3BvcnQgYXMgYW55KS5zdGF0ZT8udmlld3BvcnRFbGVtZW50O1xuXG4gICAgICBpZiAodmlld3BvcnRFbGVtZW50KSB7XG4gICAgICAgIC8vIFRvdWNoIGV2ZW50c1xuICAgICAgICB2aWV3cG9ydEVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcInRvdWNoc3RhcnRcIiwgaGFuZGxlVG91Y2hTdGFydCwge1xuICAgICAgICAgIHBhc3NpdmU6IHRydWUsXG4gICAgICAgIH0pO1xuICAgICAgICB2aWV3cG9ydEVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcInRvdWNobW92ZVwiLCBoYW5kbGVUb3VjaE1vdmUsIHtcbiAgICAgICAgICBwYXNzaXZlOiBmYWxzZSxcbiAgICAgICAgfSk7XG4gICAgICAgIHZpZXdwb3J0RWxlbWVudC5hZGRFdmVudExpc3RlbmVyKFwidG91Y2hlbmRcIiwgaGFuZGxlVG91Y2hFbmQsIHtcbiAgICAgICAgICBwYXNzaXZlOiB0cnVlLFxuICAgICAgICB9KTtcblxuICAgICAgICAvLyBNb3VzZSBldmVudHMgZm9yIGRlc2t0b3AgZHJhZ2dpbmdcbiAgICAgICAgdmlld3BvcnRFbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoXCJtb3VzZWRvd25cIiwgaGFuZGxlTW91c2VEb3duKTtcbiAgICAgICAgdmlld3BvcnRFbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoXCJtb3VzZW1vdmVcIiwgaGFuZGxlTW91c2VNb3ZlKTtcbiAgICAgICAgdmlld3BvcnRFbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoXCJtb3VzZXVwXCIsIGhhbmRsZU1vdXNlVXApO1xuICAgICAgICB2aWV3cG9ydEVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcIm1vdXNlbGVhdmVcIiwgaGFuZGxlTW91c2VVcCk7XG5cbiAgICAgICAgLy8gQWRkIGNsaWNrIGxpc3RlbmVyIHRvIHN0b3AgbW9tZW50dW0gb24gY2xpY2sgKHdpdGhvdXQgYmxvY2tpbmcgc2VsZWN0aW9uKVxuICAgICAgICBpZiAoc3RvcE9uQ2xpY2spIHtcbiAgICAgICAgICB2aWV3cG9ydEVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcImNsaWNrXCIsIGhhbmRsZUNsaWNrKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFN0b3JlIHJlZmVyZW5jZSBmb3IgY2xlYW51cFxuICAgICAgICAoY29tcG9uZW50IGFzIGFueSkuX21vbWVudHVtVmlld3BvcnRFbGVtZW50ID0gdmlld3BvcnRFbGVtZW50O1xuICAgICAgICAoY29tcG9uZW50IGFzIGFueSkuX21vbWVudHVtU3RvcE9uQ2xpY2sgPSBzdG9wT25DbGljaztcbiAgICAgIH1cbiAgICB9O1xuXG4gICAgLy8gQ2xlYW4gdXAgb24gZGVzdHJveVxuICAgIGlmIChcImRlc3Ryb3lcIiBpbiBjb21wb25lbnQgJiYgdHlwZW9mIGNvbXBvbmVudC5kZXN0cm95ID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgIGNvbnN0IG9yaWdpbmFsRGVzdHJveSA9IGNvbXBvbmVudC5kZXN0cm95O1xuICAgICAgY29tcG9uZW50LmRlc3Ryb3kgPSAoKSA9PiB7XG4gICAgICAgIC8vIFJlbW92ZSBldmVudCBsaXN0ZW5lcnNcbiAgICAgICAgY29uc3Qgdmlld3BvcnRFbGVtZW50ID0gKGNvbXBvbmVudCBhcyBhbnkpLl9tb21lbnR1bVZpZXdwb3J0RWxlbWVudDtcbiAgICAgICAgaWYgKHZpZXdwb3J0RWxlbWVudCkge1xuICAgICAgICAgIHZpZXdwb3J0RWxlbWVudC5yZW1vdmVFdmVudExpc3RlbmVyKFwidG91Y2hzdGFydFwiLCBoYW5kbGVUb3VjaFN0YXJ0KTtcbiAgICAgICAgICB2aWV3cG9ydEVsZW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcInRvdWNobW92ZVwiLCBoYW5kbGVUb3VjaE1vdmUpO1xuICAgICAgICAgIHZpZXdwb3J0RWxlbWVudC5yZW1vdmVFdmVudExpc3RlbmVyKFwidG91Y2hlbmRcIiwgaGFuZGxlVG91Y2hFbmQpO1xuICAgICAgICAgIHZpZXdwb3J0RWxlbWVudC5yZW1vdmVFdmVudExpc3RlbmVyKFwibW91c2Vkb3duXCIsIGhhbmRsZU1vdXNlRG93bik7XG4gICAgICAgICAgdmlld3BvcnRFbGVtZW50LnJlbW92ZUV2ZW50TGlzdGVuZXIoXCJtb3VzZW1vdmVcIiwgaGFuZGxlTW91c2VNb3ZlKTtcbiAgICAgICAgICB2aWV3cG9ydEVsZW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcIm1vdXNldXBcIiwgaGFuZGxlTW91c2VVcCk7XG4gICAgICAgICAgdmlld3BvcnRFbGVtZW50LnJlbW92ZUV2ZW50TGlzdGVuZXIoXCJtb3VzZWxlYXZlXCIsIGhhbmRsZU1vdXNlVXApO1xuICAgICAgICAgIGlmICgoY29tcG9uZW50IGFzIGFueSkuX21vbWVudHVtU3RvcE9uQ2xpY2spIHtcbiAgICAgICAgICAgIHZpZXdwb3J0RWxlbWVudC5yZW1vdmVFdmVudExpc3RlbmVyKFwiY2xpY2tcIiwgaGFuZGxlQ2xpY2spO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIENhbmNlbCBtb21lbnR1bSBhbmltYXRpb25cbiAgICAgICAgc3RvcE1vbWVudHVtKCk7XG5cbiAgICAgICAgb3JpZ2luYWxEZXN0cm95KCk7XG4gICAgICB9O1xuICAgIH1cblxuICAgIC8vIEV4cG9zZSBtb21lbnR1bSBzdGF0ZSBmb3Igb3RoZXIgZmVhdHVyZXNcbiAgICAoY29tcG9uZW50LnZpZXdwb3J0IGFzIGFueSkubW9tZW50dW1TdGF0ZSA9IHtcbiAgICAgIHN0b3BNb21lbnR1bSxcbiAgICB9O1xuXG4gICAgcmV0dXJuIGNvbXBvbmVudDtcbiAgfTtcbn07XG4iLAogICAgIi8vIHNyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL3Njcm9sbGJhci50c1xuXG4vKipcbiAqIFNjcm9sbGJhciBGZWF0dXJlIC0gQ3VzdG9tIHNjcm9sbGJhciBpbXBsZW1lbnRhdGlvblxuICogUHJvdmlkZXMgdmlzdWFsIHNjcm9sbCBpbmRpY2F0aW9uIGFuZCBkcmFnLXRvLXNjcm9sbCBmdW5jdGlvbmFsaXR5XG4gKi9cblxuaW1wb3J0IHR5cGUgeyBWaWV3cG9ydENvbnRleHQsIFZpZXdwb3J0Q29tcG9uZW50IH0gZnJvbSBcIi4uL3R5cGVzXCI7XG5pbXBvcnQgeyBWSUVXUE9SVF9DT05TVEFOVFMgfSBmcm9tIFwiLi4vY29uc3RhbnRzXCI7XG5pbXBvcnQgeyB3cmFwSW5pdGlhbGl6ZSwgd3JhcERlc3Ryb3ksIHN0b3JlRmVhdHVyZUZ1bmN0aW9uIH0gZnJvbSBcIi4vdXRpbHNcIjtcbmltcG9ydCB7IFBSRUZJWCwgYWRkQ2xhc3MsIHJlbW92ZUNsYXNzIH0gZnJvbSBcIm10cmxcIjtcblxuZXhwb3J0IGludGVyZmFjZSBTY3JvbGxiYXJDb25maWcge1xuICBlbmFibGVkPzogYm9vbGVhbjtcbiAgYXV0b0hpZGU/OiBib29sZWFuO1xuICB0aHVtYk1pbkhlaWdodD86IG51bWJlcjtcbiAgdGh1bWJDb2xvcj86IHN0cmluZztcbiAgdHJhY2tDb2xvcj86IHN0cmluZztcbiAgYm9yZGVyUmFkaXVzPzogbnVtYmVyO1xuICBmYWRlVGltZW91dD86IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTY3JvbGxiYXJDb21wb25lbnQge1xuICBzY3JvbGxiYXI6IHtcbiAgICBzaG93OiAoKSA9PiB2b2lkO1xuICAgIGhpZGU6ICgpID0+IHZvaWQ7XG4gICAgdXBkYXRlQm91bmRzOiAodG90YWxWaXJ0dWFsU2l6ZTogbnVtYmVyLCBjb250YWluZXJTaXplOiBudW1iZXIpID0+IHZvaWQ7XG4gICAgdXBkYXRlUG9zaXRpb246IChzY3JvbGxQb3NpdGlvbjogbnVtYmVyKSA9PiB2b2lkO1xuICB9O1xufVxuXG4vKipcbiAqIEFkZHMgc2Nyb2xsYmFyIGZ1bmN0aW9uYWxpdHkgdG8gdmlld3BvcnQgY29tcG9uZW50XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB3aXRoU2Nyb2xsYmFyKGNvbmZpZzogU2Nyb2xsYmFyQ29uZmlnID0ge30pIHtcbiAgcmV0dXJuIDxUIGV4dGVuZHMgVmlld3BvcnRDb250ZXh0ICYgVmlld3BvcnRDb21wb25lbnQ+KFxuICAgIGNvbXBvbmVudDogVCxcbiAgKTogVCAmIFNjcm9sbGJhckNvbXBvbmVudCA9PiB7XG4gICAgY29uc3Qge1xuICAgICAgZW5hYmxlZCA9IHRydWUsXG4gICAgICBhdXRvSGlkZSA9IHRydWUsXG4gICAgICB0aHVtYk1pbkhlaWdodCA9IDI1LFxuICAgICAgYm9yZGVyUmFkaXVzID0gNCxcbiAgICAgIGZhZGVUaW1lb3V0ID0gMTAwMCxcbiAgICB9ID0gY29uZmlnO1xuXG4gICAgLy8gUmV0dXJuIG5vLW9wIGlmIGRpc2FibGVkXG4gICAgaWYgKCFlbmFibGVkKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICAuLi5jb21wb25lbnQsXG4gICAgICAgIHNjcm9sbGJhcjoge1xuICAgICAgICAgIHNob3c6ICgpID0+IHt9LFxuICAgICAgICAgIGhpZGU6ICgpID0+IHt9LFxuICAgICAgICAgIHVwZGF0ZUJvdW5kczogKCkgPT4ge30sXG4gICAgICAgICAgdXBkYXRlUG9zaXRpb246ICgpID0+IHt9LFxuICAgICAgICB9LFxuICAgICAgfTtcbiAgICB9XG5cbiAgICAvLyBTdGF0ZVxuICAgIGxldCB2aWV3cG9ydEVsZW1lbnQ6IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGw7XG4gICAgbGV0IHNjcm9sbGJhclRyYWNrOiBIVE1MRWxlbWVudCB8IG51bGwgPSBudWxsO1xuICAgIGxldCBzY3JvbGxiYXJUaHVtYjogSFRNTEVsZW1lbnQgfCBudWxsID0gbnVsbDtcbiAgICBsZXQgaXNJbml0aWFsaXplZCA9IGZhbHNlO1xuICAgIGxldCBpc0RyYWdnaW5nID0gZmFsc2U7XG4gICAgbGV0IGRyYWdTdGFydFkgPSAwO1xuICAgIGxldCBkcmFnU3RhcnRTY3JvbGxQb3NpdGlvbiA9IDA7XG4gICAgbGV0IGhpZGVUaW1lb3V0OiBSZXR1cm5UeXBlPHR5cGVvZiBzZXRUaW1lb3V0PiB8IG51bGwgPSBudWxsO1xuICAgIGxldCB0b3RhbFZpcnR1YWxTaXplID0gMDtcbiAgICBsZXQgY29udGFpbmVyU2l6ZSA9IDA7XG4gICAgbGV0IHRodW1iSGVpZ2h0ID0gMDtcbiAgICBsZXQgYW5pbWF0aW9uRnJhbWVJZDogbnVtYmVyIHwgbnVsbCA9IG51bGw7XG4gICAgbGV0IGxhc3RSZXF1ZXN0ZWRTY3JvbGxQb3NpdGlvbjogbnVtYmVyIHwgbnVsbCA9IG51bGw7XG4gICAgbGV0IGlzQ3Vyc29yTW9kZSA9IGZhbHNlO1xuICAgIGxldCBsb2FkZWRJdGVtc0NvdW50ID0gMDtcblxuICAgIC8vIENyZWF0ZSBzY3JvbGxiYXIgZWxlbWVudHNcbiAgICBjb25zdCBjcmVhdGVTY3JvbGxiYXJFbGVtZW50cyA9ICgpID0+IHtcbiAgICAgIGlmICghdmlld3BvcnRFbGVtZW50KSByZXR1cm47XG5cbiAgICAgIHNjcm9sbGJhclRyYWNrID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImRpdlwiKTtcbiAgICAgIGFkZENsYXNzKHNjcm9sbGJhclRyYWNrLCBWSUVXUE9SVF9DT05TVEFOVFMuU0NST0xMQkFSLkNMQVNTRVMuU0NST0xMQkFSKTtcbiAgICAgIHNjcm9sbGJhclRyYWNrLnN0eWxlLmNzc1RleHQgPSBgXG4gICAgICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICAgICAgdG9wOiAwO1xuICAgICAgICByaWdodDogMDtcbiAgICAgICAgaGVpZ2h0OiAxMDAlO1xuICAgICAgICB6LWluZGV4OiAxMDtcbiAgICAgIGA7XG5cbiAgICAgIHNjcm9sbGJhclRodW1iID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImRpdlwiKTtcblxuICAgICAgYWRkQ2xhc3MoXG4gICAgICAgIHNjcm9sbGJhclRodW1iLFxuICAgICAgICBWSUVXUE9SVF9DT05TVEFOVFMuU0NST0xMQkFSLkNMQVNTRVMuU0NST0xMQkFSX1RIVU1CLFxuICAgICAgKTtcblxuICAgICAgc2Nyb2xsYmFyVHJhY2suYXBwZW5kQ2hpbGQoc2Nyb2xsYmFyVGh1bWIpO1xuICAgICAgdmlld3BvcnRFbGVtZW50LmFwcGVuZENoaWxkKHNjcm9sbGJhclRyYWNrKTtcbiAgICB9O1xuXG4gICAgLy8gU2hvdy9oaWRlIGZ1bmN0aW9uc1xuICAgIGNvbnN0IHNob3cgPSAoKSA9PiB7XG4gICAgICBpZiAoIXNjcm9sbGJhclRyYWNrIHx8IHRvdGFsVmlydHVhbFNpemUgPD0gY29udGFpbmVyU2l6ZSkgcmV0dXJuO1xuXG4gICAgICBpZiAoaGlkZVRpbWVvdXQpIHtcbiAgICAgICAgY2xlYXJUaW1lb3V0KGhpZGVUaW1lb3V0KTtcbiAgICAgICAgaGlkZVRpbWVvdXQgPSBudWxsO1xuICAgICAgfVxuXG4gICAgICBhZGRDbGFzcyhcbiAgICAgICAgc2Nyb2xsYmFyVHJhY2ssXG4gICAgICAgIFZJRVdQT1JUX0NPTlNUQU5UUy5TQ1JPTExCQVIuQ0xBU1NFUy5TQ1JPTExCQVJfVklTSUJMRSxcbiAgICAgICk7XG5cbiAgICAgIGlmIChhdXRvSGlkZSAmJiAhaXNEcmFnZ2luZykge1xuICAgICAgICBoaWRlVGltZW91dCA9IHNldFRpbWVvdXQoaGlkZSwgZmFkZVRpbWVvdXQpO1xuICAgICAgfVxuICAgIH07XG5cbiAgICBjb25zdCBoaWRlID0gKCkgPT4ge1xuICAgICAgaWYgKCFzY3JvbGxiYXJUcmFjayB8fCBpc0RyYWdnaW5nKSByZXR1cm47XG4gICAgICByZW1vdmVDbGFzcyhcbiAgICAgICAgc2Nyb2xsYmFyVHJhY2ssXG4gICAgICAgIFZJRVdQT1JUX0NPTlNUQU5UUy5TQ1JPTExCQVIuQ0xBU1NFUy5TQ1JPTExCQVJfVklTSUJMRSxcbiAgICAgICk7XG4gICAgfTtcblxuICAgIC8vIFVwZGF0ZSBzY3JvbGxiYXIgYm91bmRzXG4gICAgY29uc3QgdXBkYXRlQm91bmRzID0gKG5ld1RvdGFsU2l6ZTogbnVtYmVyLCBuZXdDb250YWluZXJTaXplOiBudW1iZXIpID0+IHtcbiAgICAgIHRvdGFsVmlydHVhbFNpemUgPSBuZXdUb3RhbFNpemU7XG4gICAgICBjb250YWluZXJTaXplID0gbmV3Q29udGFpbmVyU2l6ZTtcblxuICAgICAgaWYgKCFzY3JvbGxiYXJUcmFjayB8fCAhc2Nyb2xsYmFyVGh1bWIpIHJldHVybjtcblxuICAgICAgY29uc3QgbmVlZHNTY3JvbGxiYXIgPSB0b3RhbFZpcnR1YWxTaXplID4gY29udGFpbmVyU2l6ZTtcbiAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgLy8gICBgW1Njcm9sbGJhcl0gdXBkYXRlQm91bmRzOiB0b3RhbFNpemU9JHt0b3RhbFZpcnR1YWxTaXplfSwgY29udGFpbmVyU2l6ZT0ke2NvbnRhaW5lclNpemV9LCBuZWVkc1Njcm9sbGJhcj0ke25lZWRzU2Nyb2xsYmFyfWBcbiAgICAgIC8vICk7XG4gICAgICBzY3JvbGxiYXJUcmFjay5zdHlsZS5kaXNwbGF5ID0gbmVlZHNTY3JvbGxiYXIgPyBcImJsb2NrXCIgOiBcIm5vbmVcIjtcblxuICAgICAgaWYgKG5lZWRzU2Nyb2xsYmFyKSB7XG4gICAgICAgIGxldCBzY3JvbGxSYXRpbzogbnVtYmVyO1xuXG4gICAgICAgIGlmIChpc0N1cnNvck1vZGUgJiYgbG9hZGVkSXRlbXNDb3VudCA+IDApIHtcbiAgICAgICAgICAvLyBGb3IgY3Vyc29yIG1vZGUsIHRodW1iIHNpemUgcmVwcmVzZW50cyBsb2FkZWQgY29udGVudCB2cyBlc3RpbWF0ZWQgdG90YWxcbiAgICAgICAgICAvLyBJZiB3ZSBkb24ndCBrbm93IHRvdGFsLCB1c2UgbG9hZGVkIGl0ZW1zICogMiBhcyBlc3RpbWF0ZVxuICAgICAgICAgIGNvbnN0IGVzdGltYXRlZFRvdGFsID1cbiAgICAgICAgICAgIHRvdGFsVmlydHVhbFNpemUgPiAwID8gdG90YWxWaXJ0dWFsU2l6ZSA6IGxvYWRlZEl0ZW1zQ291bnQgKiAyO1xuICAgICAgICAgIHNjcm9sbFJhdGlvID1cbiAgICAgICAgICAgIGNvbnRhaW5lclNpemUgL1xuICAgICAgICAgICAgTWF0aC5tYXgoZXN0aW1hdGVkVG90YWwsIGxvYWRlZEl0ZW1zQ291bnQgKyBjb250YWluZXJTaXplKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyBOb3JtYWwgY2FsY3VsYXRpb24gZm9yIG9mZnNldC9wYWdlIHN0cmF0ZWdpZXNcbiAgICAgICAgICBzY3JvbGxSYXRpbyA9IGNvbnRhaW5lclNpemUgLyB0b3RhbFZpcnR1YWxTaXplO1xuICAgICAgICB9XG5cbiAgICAgICAgdGh1bWJIZWlnaHQgPSBNYXRoLm1heCh0aHVtYk1pbkhlaWdodCwgc2Nyb2xsUmF0aW8gKiBjb250YWluZXJTaXplKTtcbiAgICAgICAgc2Nyb2xsYmFyVGh1bWIuc3R5bGUuaGVpZ2h0ID0gYCR7dGh1bWJIZWlnaHR9cHhgO1xuICAgICAgICB1cGRhdGVQb3NpdGlvbihjb21wb25lbnQudmlld3BvcnQ/LmdldFNjcm9sbFBvc2l0aW9uKCkgfHwgMCk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIFVwZGF0ZSB0aHVtYiBwb3NpdGlvblxuICAgIGNvbnN0IHVwZGF0ZVBvc2l0aW9uID0gKHNjcm9sbFBvczogbnVtYmVyKSA9PiB7XG4gICAgICBpZiAoIXNjcm9sbGJhclRodW1iIHx8ICFzY3JvbGxiYXJUcmFjaykgcmV0dXJuO1xuXG4gICAgICBjb25zdCBzY3JvbGxhYmxlRGlzdGFuY2UgPSB0b3RhbFZpcnR1YWxTaXplIC0gY29udGFpbmVyU2l6ZTtcbiAgICAgIGlmIChzY3JvbGxhYmxlRGlzdGFuY2UgPD0gMCkgcmV0dXJuO1xuXG4gICAgICBjb25zdCBzY3JvbGxSYXRpbyA9IE1hdGgubWluKFxuICAgICAgICAxLFxuICAgICAgICBNYXRoLm1heCgwLCBzY3JvbGxQb3MgLyBzY3JvbGxhYmxlRGlzdGFuY2UpLFxuICAgICAgKTtcbiAgICAgIGNvbnN0IG1heFRodW1iUG9zaXRpb24gPVxuICAgICAgICBzY3JvbGxiYXJUcmFjay5jbGllbnRIZWlnaHQgLSBzY3JvbGxiYXJUaHVtYi5jbGllbnRIZWlnaHQ7XG4gICAgICBjb25zdCB0aHVtYlBvc2l0aW9uID0gc2Nyb2xsUmF0aW8gKiBtYXhUaHVtYlBvc2l0aW9uO1xuXG4gICAgICBzY3JvbGxiYXJUaHVtYi5zdHlsZS50cmFuc2Zvcm0gPSBgdHJhbnNsYXRlWSgke3RodW1iUG9zaXRpb259cHgpYDtcbiAgICB9O1xuXG4gICAgLy8gSGFuZGxlIHRyYWNrIGNsaWNrXG4gICAgY29uc3QgaGFuZGxlVHJhY2tDbGljayA9IChlOiBNb3VzZUV2ZW50KSA9PiB7XG4gICAgICBpZiAoIXNjcm9sbGJhclRyYWNrIHx8ICFzY3JvbGxiYXJUaHVtYiB8fCBlLnRhcmdldCA9PT0gc2Nyb2xsYmFyVGh1bWIpXG4gICAgICAgIHJldHVybjtcblxuICAgICAgY29uc3QgdHJhY2tSZWN0ID0gc2Nyb2xsYmFyVHJhY2suZ2V0Qm91bmRpbmdDbGllbnRSZWN0KCk7XG4gICAgICBjb25zdCBjbGlja1kgPSBlLmNsaWVudFkgLSB0cmFja1JlY3QudG9wO1xuICAgICAgY29uc3QgdGh1bWJDZW50ZXJZID0gY2xpY2tZIC0gdGh1bWJIZWlnaHQgLyAyO1xuICAgICAgY29uc3QgbWF4VGh1bWJQb3NpdGlvbiA9IGNvbnRhaW5lclNpemUgLSB0aHVtYkhlaWdodDtcbiAgICAgIGNvbnN0IHRodW1iUG9zaXRpb24gPSBNYXRoLm1heChcbiAgICAgICAgMCxcbiAgICAgICAgTWF0aC5taW4odGh1bWJDZW50ZXJZLCBtYXhUaHVtYlBvc2l0aW9uKSxcbiAgICAgICk7XG4gICAgICBjb25zdCBzY3JvbGxSYXRpbyA9IHRodW1iUG9zaXRpb24gLyBtYXhUaHVtYlBvc2l0aW9uO1xuICAgICAgY29uc3QgdGFyZ2V0U2Nyb2xsUG9zaXRpb24gPVxuICAgICAgICBzY3JvbGxSYXRpbyAqICh0b3RhbFZpcnR1YWxTaXplIC0gY29udGFpbmVyU2l6ZSk7XG5cbiAgICAgIGNvbXBvbmVudC52aWV3cG9ydD8uc2Nyb2xsVG9Qb3NpdGlvbih0YXJnZXRTY3JvbGxQb3NpdGlvbik7XG4gICAgfTtcblxuICAgIC8vIE1vdXNlIGV2ZW50IGhhbmRsZXJzXG4gICAgY29uc3QgaGFuZGxlVGh1bWJNb3VzZURvd24gPSAoZTogTW91c2VFdmVudCkgPT4ge1xuICAgICAgZS5wcmV2ZW50RGVmYXVsdCgpO1xuICAgICAgZS5zdG9wUHJvcGFnYXRpb24oKTtcblxuICAgICAgaXNEcmFnZ2luZyA9IHRydWU7XG4gICAgICBkcmFnU3RhcnRZID0gZS5jbGllbnRZO1xuICAgICAgZHJhZ1N0YXJ0U2Nyb2xsUG9zaXRpb24gPSBjb21wb25lbnQudmlld3BvcnQ/LmdldFNjcm9sbFBvc2l0aW9uKCkgfHwgMDtcblxuICAgICAgaWYgKHNjcm9sbGJhclRyYWNrKSB7XG4gICAgICAgIGFkZENsYXNzKFxuICAgICAgICAgIHNjcm9sbGJhclRyYWNrLFxuICAgICAgICAgIFZJRVdQT1JUX0NPTlNUQU5UUy5TQ1JPTExCQVIuQ0xBU1NFUy5TQ1JPTExCQVJfRFJBR0dJTkcsXG4gICAgICAgICk7XG4gICAgICB9XG5cbiAgICAgIC8vIEVtaXQgZHJhZyBzdGFydCBldmVudCB0byBub3RpZnkgdmlld3BvcnRcbiAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpkcmFnLXN0YXJ0XCIsIHtcbiAgICAgICAgc291cmNlOiBcInNjcm9sbGJhclwiLFxuICAgICAgICBzdGFydFBvc2l0aW9uOiBkcmFnU3RhcnRTY3JvbGxQb3NpdGlvbixcbiAgICAgIH0pO1xuXG4gICAgICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKFwibW91c2Vtb3ZlXCIsIGhhbmRsZU1vdXNlTW92ZSk7XG4gICAgICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKFwibW91c2V1cFwiLCBoYW5kbGVNb3VzZVVwKTtcbiAgICB9O1xuXG4gICAgY29uc3QgaGFuZGxlTW91c2VNb3ZlID0gKGU6IE1vdXNlRXZlbnQpID0+IHtcbiAgICAgIGlmICghaXNEcmFnZ2luZyB8fCAhc2Nyb2xsYmFyVHJhY2sgfHwgIXNjcm9sbGJhclRodW1iKSByZXR1cm47XG5cbiAgICAgIGNvbnN0IGRlbHRhWSA9IGUuY2xpZW50WSAtIGRyYWdTdGFydFk7XG4gICAgICBjb25zdCB0cmFja0hlaWdodCA9IHNjcm9sbGJhclRyYWNrLmNsaWVudEhlaWdodDtcbiAgICAgIGNvbnN0IHRodW1iSGVpZ2h0ID0gc2Nyb2xsYmFyVGh1bWIuY2xpZW50SGVpZ2h0O1xuICAgICAgY29uc3QgbWF4VGh1bWJUcmF2ZWwgPSB0cmFja0hlaWdodCAtIHRodW1iSGVpZ2h0O1xuXG4gICAgICBpZiAobWF4VGh1bWJUcmF2ZWwgPD0gMCkgcmV0dXJuO1xuXG4gICAgICBjb25zdCBkZWx0YVJhdGlvID0gZGVsdGFZIC8gbWF4VGh1bWJUcmF2ZWw7XG4gICAgICBjb25zdCBkcmFnU3RhcnRTY3JvbGxSYXRpbyA9XG4gICAgICAgIGRyYWdTdGFydFNjcm9sbFBvc2l0aW9uIC8gKHRvdGFsVmlydHVhbFNpemUgLSBjb250YWluZXJTaXplKTtcbiAgICAgIGNvbnN0IG5ld1Njcm9sbFJhdGlvID0gTWF0aC5tYXgoXG4gICAgICAgIDAsXG4gICAgICAgIE1hdGgubWluKDEsIGRyYWdTdGFydFNjcm9sbFJhdGlvICsgZGVsdGFSYXRpbyksXG4gICAgICApO1xuXG4gICAgICAvLyBVcGRhdGUgdGh1bWIgcG9zaXRpb24gaW1tZWRpYXRlbHlcbiAgICAgIGNvbnN0IHRodW1iUG9zaXRpb24gPSBuZXdTY3JvbGxSYXRpbyAqIG1heFRodW1iVHJhdmVsO1xuICAgICAgc2Nyb2xsYmFyVGh1bWIuc3R5bGUudHJhbnNmb3JtID0gYHRyYW5zbGF0ZVkoJHt0aHVtYlBvc2l0aW9ufXB4KWA7XG5cbiAgICAgIC8vIENhbGN1bGF0ZSBuZXcgc2Nyb2xsIHBvc2l0aW9uXG4gICAgICBjb25zdCBuZXdQb3NpdGlvbiA9IG5ld1Njcm9sbFJhdGlvICogKHRvdGFsVmlydHVhbFNpemUgLSBjb250YWluZXJTaXplKTtcbiAgICAgIGxhc3RSZXF1ZXN0ZWRTY3JvbGxQb3NpdGlvbiA9IG5ld1Bvc2l0aW9uO1xuXG4gICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgIC8vICAgYFtTY3JvbGxiYXJdIE1vdXNlIG1vdmU6IG5ld1Bvcz0ke25ld1Bvc2l0aW9uLnRvRml4ZWQoXG4gICAgICAvLyAgICAgMFxuICAgICAgLy8gICApfSwgYW5pbWF0aW9uRnJhbWVJZD0ke2FuaW1hdGlvbkZyYW1lSWR9YFxuICAgICAgLy8gKTtcblxuICAgICAgLy8gVGhyb3R0bGUgdmlld3BvcnQgdXBkYXRlc1xuICAgICAgaWYgKGFuaW1hdGlvbkZyYW1lSWQgPT09IG51bGwgJiYgY29tcG9uZW50LnZpZXdwb3J0KSB7XG4gICAgICAgIGFuaW1hdGlvbkZyYW1lSWQgPSByZXF1ZXN0QW5pbWF0aW9uRnJhbWUoKCkgPT4ge1xuICAgICAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgICAgIC8vICAgYFtTY3JvbGxiYXJdIFJBRiBjYWxsYmFjazogc2Nyb2xsaW5nIHRvICR7bGFzdFJlcXVlc3RlZFNjcm9sbFBvc2l0aW9ufWBcbiAgICAgICAgICAvLyApO1xuICAgICAgICAgIGlmIChsYXN0UmVxdWVzdGVkU2Nyb2xsUG9zaXRpb24gIT09IG51bGwgJiYgY29tcG9uZW50LnZpZXdwb3J0KSB7XG4gICAgICAgICAgICBjb21wb25lbnQudmlld3BvcnQuc2Nyb2xsVG9Qb3NpdGlvbihsYXN0UmVxdWVzdGVkU2Nyb2xsUG9zaXRpb24pO1xuICAgICAgICAgIH1cbiAgICAgICAgICBhbmltYXRpb25GcmFtZUlkID0gbnVsbDtcbiAgICAgICAgfSk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIGNvbnN0IGhhbmRsZU1vdXNlVXAgPSAoKSA9PiB7XG4gICAgICBpc0RyYWdnaW5nID0gZmFsc2U7XG5cbiAgICAgIGlmIChhbmltYXRpb25GcmFtZUlkICE9PSBudWxsKSB7XG4gICAgICAgIGNhbmNlbEFuaW1hdGlvbkZyYW1lKGFuaW1hdGlvbkZyYW1lSWQpO1xuICAgICAgICBhbmltYXRpb25GcmFtZUlkID0gbnVsbDtcbiAgICAgIH1cblxuICAgICAgaWYgKGxhc3RSZXF1ZXN0ZWRTY3JvbGxQb3NpdGlvbiAhPT0gbnVsbCAmJiBjb21wb25lbnQudmlld3BvcnQpIHtcbiAgICAgICAgY29tcG9uZW50LnZpZXdwb3J0LnNjcm9sbFRvUG9zaXRpb24obGFzdFJlcXVlc3RlZFNjcm9sbFBvc2l0aW9uKTtcbiAgICAgICAgbGFzdFJlcXVlc3RlZFNjcm9sbFBvc2l0aW9uID0gbnVsbDtcbiAgICAgIH1cblxuICAgICAgaWYgKHNjcm9sbGJhclRyYWNrKSB7XG4gICAgICAgIHJlbW92ZUNsYXNzKFxuICAgICAgICAgIHNjcm9sbGJhclRyYWNrLFxuICAgICAgICAgIFZJRVdQT1JUX0NPTlNUQU5UUy5TQ1JPTExCQVIuQ0xBU1NFUy5TQ1JPTExCQVJfRFJBR0dJTkcsXG4gICAgICAgICk7XG4gICAgICB9XG5cbiAgICAgIC8vIEVtaXQgZHJhZyBlbmQgZXZlbnRcbiAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpkcmFnLWVuZFwiLCB7XG4gICAgICAgIHNvdXJjZTogXCJzY3JvbGxiYXJcIixcbiAgICAgICAgZW5kUG9zaXRpb246IGNvbXBvbmVudC52aWV3cG9ydD8uZ2V0U2Nyb2xsUG9zaXRpb24oKSB8fCAwLFxuICAgICAgfSk7XG5cbiAgICAgIGhpZGUoKTtcblxuICAgICAgZG9jdW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcIm1vdXNlbW92ZVwiLCBoYW5kbGVNb3VzZU1vdmUpO1xuICAgICAgZG9jdW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcIm1vdXNldXBcIiwgaGFuZGxlTW91c2VVcCk7XG4gICAgfTtcblxuICAgIC8vIEluaXRpYWxpemUgZnVuY3Rpb25cbiAgICBjb25zdCBpbml0aWFsaXplID0gKCkgPT4ge1xuICAgICAgaWYgKGlzSW5pdGlhbGl6ZWQpIHJldHVybjtcblxuICAgICAgdmlld3BvcnRFbGVtZW50ID0gY29tcG9uZW50LmVsZW1lbnQ/LnF1ZXJ5U2VsZWN0b3IoXG4gICAgICAgIGAuJHtQUkVGSVh9LXZpZXdwb3J0YCxcbiAgICAgICkgYXMgSFRNTEVsZW1lbnQ7XG5cbiAgICAgIGlmICghdmlld3BvcnRFbGVtZW50KSB7XG4gICAgICAgIGNvbnNvbGUud2FybihcIltTY3JvbGxiYXJdIE5vIHZpZXdwb3J0IGVsZW1lbnQgZm91bmRcIik7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cblxuICAgICAgaWYgKFxuICAgICAgICB2aWV3cG9ydEVsZW1lbnQucXVlcnlTZWxlY3RvcihcbiAgICAgICAgICBgLiR7UFJFRklYfS0ke1ZJRVdQT1JUX0NPTlNUQU5UUy5TQ1JPTExCQVIuQ0xBU1NFUy5TQ1JPTExCQVJ9YCxcbiAgICAgICAgKVxuICAgICAgKVxuICAgICAgICByZXR1cm47XG5cbiAgICAgIGNyZWF0ZVNjcm9sbGJhckVsZW1lbnRzKCk7XG4gICAgICBpc0luaXRpYWxpemVkID0gdHJ1ZTtcblxuICAgICAgaWYgKHNjcm9sbGJhclRyYWNrICYmIHZpZXdwb3J0RWxlbWVudCkge1xuICAgICAgICBzY3JvbGxiYXJUcmFjay5hZGRFdmVudExpc3RlbmVyKFwiY2xpY2tcIiwgaGFuZGxlVHJhY2tDbGljayk7XG5cbiAgICAgICAgaWYgKHNjcm9sbGJhclRodW1iKSB7XG4gICAgICAgICAgc2Nyb2xsYmFyVGh1bWIuYWRkRXZlbnRMaXN0ZW5lcihcIm1vdXNlZG93blwiLCBoYW5kbGVUaHVtYk1vdXNlRG93bik7XG4gICAgICAgIH1cblxuICAgICAgICB2aWV3cG9ydEVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcIm1vdXNlZW50ZXJcIiwgc2hvdywgeyBwYXNzaXZlOiB0cnVlIH0pO1xuICAgICAgICB2aWV3cG9ydEVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcbiAgICAgICAgICBcIm1vdXNlbGVhdmVcIixcbiAgICAgICAgICAoKSA9PiB7XG4gICAgICAgICAgICBpZiAoIWlzRHJhZ2dpbmcpIGhpZGUoKTtcbiAgICAgICAgICB9LFxuICAgICAgICAgIHsgcGFzc2l2ZTogdHJ1ZSB9LFxuICAgICAgICApO1xuICAgICAgfVxuXG4gICAgICAvLyBJbml0aWFsaXplIHdpdGggY3VycmVudCB2aWV3cG9ydCBzdGF0ZVxuICAgICAgaWYgKGNvbXBvbmVudC52aWV3cG9ydCkge1xuICAgICAgICBjb25zdCBpbmZvID0gY29tcG9uZW50LnZpZXdwb3J0LmdldFZpZXdwb3J0SW5mbygpO1xuICAgICAgICB0b3RhbFZpcnR1YWxTaXplID0gaW5mby50b3RhbFZpcnR1YWxTaXplO1xuICAgICAgICBjb250YWluZXJTaXplID0gaW5mby5jb250YWluZXJTaXplO1xuXG4gICAgICAgIC8vIENoZWNrIGlmIHdlJ3JlIGluIGN1cnNvciBtb2RlXG4gICAgICAgIGNvbnN0IGNvbGxlY3Rpb24gPSAoY29tcG9uZW50IGFzIGFueSkuY29sbGVjdGlvbjtcbiAgICAgICAgY29uc3Qgdmlld3BvcnRDb25maWcgPSAoY29tcG9uZW50IGFzIGFueSkuY29uZmlnO1xuICAgICAgICBpc0N1cnNvck1vZGUgPSB2aWV3cG9ydENvbmZpZz8ucGFnaW5hdGlvbj8uc3RyYXRlZ3kgPT09IFwiY3Vyc29yXCI7XG5cbiAgICAgICAgdXBkYXRlQm91bmRzKHRvdGFsVmlydHVhbFNpemUsIGNvbnRhaW5lclNpemUpO1xuICAgICAgICB1cGRhdGVQb3NpdGlvbihjb21wb25lbnQudmlld3BvcnQuZ2V0U2Nyb2xsUG9zaXRpb24oKSk7XG4gICAgICB9XG5cbiAgICAgIC8vIEV2ZW50IGxpc3RlbmVyc1xuICAgICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDpzY3JvbGxcIiwgKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICBpZiAoIWlzRHJhZ2dpbmcpIHtcbiAgICAgICAgICB1cGRhdGVQb3NpdGlvbihkYXRhLnBvc2l0aW9uKTtcbiAgICAgICAgICBzaG93KCk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuXG4gICAgICBjb21wb25lbnQub24/LihcInZpZXdwb3J0OnZpcnR1YWwtc2l6ZS1jaGFuZ2VkXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgdG90YWxWaXJ0dWFsU2l6ZSA9IGRhdGEudG90YWxWaXJ0dWFsU2l6ZTtcbiAgICAgICAgdXBkYXRlQm91bmRzKGRhdGEudG90YWxWaXJ0dWFsU2l6ZSwgY29udGFpbmVyU2l6ZSk7XG4gICAgICB9KTtcblxuICAgICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDpjb250YWluZXItc2l6ZS1jaGFuZ2VkXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgY29udGFpbmVyU2l6ZSA9IGRhdGEuY29udGFpbmVyU2l6ZTtcbiAgICAgICAgdXBkYXRlQm91bmRzKHRvdGFsVmlydHVhbFNpemUsIGRhdGEuY29udGFpbmVyU2l6ZSk7XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciBpdGVtcyBsb2FkZWQgdG8gdXBkYXRlIHNjcm9sbGJhciBpbiBjdXJzb3IgbW9kZVxuICAgICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDppdGVtcy1jaGFuZ2VkXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgaWYgKGlzQ3Vyc29yTW9kZSkge1xuICAgICAgICAgIGxvYWRlZEl0ZW1zQ291bnQgPSBkYXRhLmxvYWRlZENvdW50IHx8IDA7XG4gICAgICAgICAgdXBkYXRlQm91bmRzKHRvdGFsVmlydHVhbFNpemUsIGNvbnRhaW5lclNpemUpO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciB0b3RhbCBpdGVtcyBjaGFuZ2VzIChpbXBvcnRhbnQgZm9yIGN1cnNvciBtb2RlKVxuICAgICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDp0b3RhbC1pdGVtcy1jaGFuZ2VkXCIsIChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgaWYgKGlzQ3Vyc29yTW9kZSAmJiBkYXRhLnRvdGFsKSB7XG4gICAgICAgICAgLy8gSW4gY3Vyc29yIG1vZGUsIHRoZSB0b3RhbCBpcyBkeW5hbWljXG4gICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICBgW1Njcm9sbGJhcl0gQ3Vyc29yIG1vZGU6IHVwZGF0aW5nIGJvdW5kcyBmb3IgbmV3IHRvdGFsICR7ZGF0YS50b3RhbH1gLFxuICAgICAgICAgICk7XG4gICAgICAgICAgdXBkYXRlQm91bmRzKHRvdGFsVmlydHVhbFNpemUsIGNvbnRhaW5lclNpemUpO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgICB9O1xuXG4gICAgLy8gU3RvcmUgaW5pdGlhbGl6ZSBmdW5jdGlvblxuICAgIHN0b3JlRmVhdHVyZUZ1bmN0aW9uKGNvbXBvbmVudCwgXCJfc2Nyb2xsYmFySW5pdGlhbGl6ZVwiLCBpbml0aWFsaXplKTtcblxuICAgIC8vIEhvb2sgaW50byB2aWV3cG9ydCBpbml0aWFsaXphdGlvblxuICAgIHdyYXBJbml0aWFsaXplKGNvbXBvbmVudCwgaW5pdGlhbGl6ZSk7XG5cbiAgICAvLyBDbGVhbnVwXG4gICAgY29uc3QgZGVzdHJveSA9ICgpID0+IHtcbiAgICAgIGlmIChzY3JvbGxiYXJUcmFjaykge1xuICAgICAgICBzY3JvbGxiYXJUcmFjay5yZW1vdmUoKTtcbiAgICAgICAgc2Nyb2xsYmFyVHJhY2sgPSBudWxsO1xuICAgICAgfVxuICAgICAgc2Nyb2xsYmFyVGh1bWIgPSBudWxsO1xuICAgICAgdmlld3BvcnRFbGVtZW50ID0gbnVsbDtcbiAgICAgIGlzSW5pdGlhbGl6ZWQgPSBmYWxzZTtcblxuICAgICAgaWYgKGhpZGVUaW1lb3V0KSB7XG4gICAgICAgIGNsZWFyVGltZW91dChoaWRlVGltZW91dCk7XG4gICAgICAgIGhpZGVUaW1lb3V0ID0gbnVsbDtcbiAgICAgIH1cblxuICAgICAgZG9jdW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcIm1vdXNlbW92ZVwiLCBoYW5kbGVNb3VzZU1vdmUpO1xuICAgICAgZG9jdW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcIm1vdXNldXBcIiwgaGFuZGxlTW91c2VVcCk7XG4gICAgfTtcblxuICAgIHN0b3JlRmVhdHVyZUZ1bmN0aW9uKGNvbXBvbmVudCwgXCJfc2Nyb2xsYmFyRGVzdHJveVwiLCBkZXN0cm95KTtcbiAgICB3cmFwRGVzdHJveShjb21wb25lbnQsIGRlc3Ryb3kpO1xuXG4gICAgLy8gUmV0dXJuIGVuaGFuY2VkIGNvbXBvbmVudFxuICAgIHJldHVybiB7XG4gICAgICAuLi5jb21wb25lbnQsXG4gICAgICBzY3JvbGxiYXI6IHsgc2hvdywgaGlkZSwgdXBkYXRlQm91bmRzLCB1cGRhdGVQb3NpdGlvbiB9LFxuICAgIH07XG4gIH07XG59XG4iLAogICAgIi8vIHNyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL2NvbGxlY3Rpb24udHNcblxuLyoqXG4gKiBDb2xsZWN0aW9uIEZlYXR1cmUgLSBEYXRhIG1hbmFnZW1lbnQgYW5kIHJhbmdlIGxvYWRpbmdcbiAqIEhhbmRsZXMgY29sbGVjdGlvbiBpbnRlZ3JhdGlvbiwgcGFnaW5hdGlvbiwgYW5kIGRhdGEgZmV0Y2hpbmdcbiAqL1xuXG5pbXBvcnQgdHlwZSB7IFZpZXdwb3J0Q29udGV4dCwgVmlld3BvcnRDb21wb25lbnQgfSBmcm9tIFwiLi4vdHlwZXNcIjtcblxuLyoqXG4gKiBNdXRhYmxlIHJlZmVyZW5jZSBmb3IgbGF0ZSBiaW5kaW5nIGluIGZ1bmN0aW9uYWwgY29tcG9zaXRpb24uXG4gKiBVc2VkIGJ5IFZMaXN0IHRvIGFsbG93IGNvbGxlY3Rpb24gKGFwcGxpZWQgZWFybHkpIHRvIGFjY2Vzc1xuICogc2VhcmNoL2ZpbHRlciBmZWF0dXJlcyAoYXBwbGllZCBsYXRlcikuXG4gKi9cbmludGVyZmFjZSBTZWxmUmVmPFQ+IHtcbiAgY3VycmVudDogVCB8IG51bGw7XG59XG5cbi8qKlxuICogQ29tcG9uZW50IHdpdGggb3B0aW9uYWwgc2VhcmNoL2ZpbHRlciBjYXBhYmlsaXRpZXMuXG4gKiBUaGVzZSBhcmUgYWRkZWQgYnkgd2l0aFNlYXJjaCBhbmQgd2l0aEZpbHRlciBmZWF0dXJlcy5cbiAqL1xuaW50ZXJmYWNlIFNlYXJjaEZpbHRlckNhcGFibGUge1xuICBfc2VhcmNoUXVlcnk/OiAoKSA9PiBzdHJpbmc7XG4gIF9maWx0ZXJzPzogKCkgPT4gUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG59XG5pbXBvcnQgeyBWSUVXUE9SVF9DT05TVEFOVFMgfSBmcm9tIFwiLi4vY29uc3RhbnRzXCI7XG5pbXBvcnQgeyB3cmFwRGVzdHJveSB9IGZyb20gXCIuL3V0aWxzXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29sbGVjdGlvbkNvbmZpZyB7XG4gIGNvbGxlY3Rpb24/OiBhbnk7IC8vIENvbGxlY3Rpb24gYWRhcHRlclxuICByYW5nZVNpemU/OiBudW1iZXI7IC8vIERlZmF1bHQgcmFuZ2Ugc2l6ZSBmb3IgbG9hZGluZ1xuICBzdHJhdGVneT86IFwib2Zmc2V0XCIgfCBcInBhZ2VcIiB8IFwiY3Vyc29yXCI7IC8vIExvYWRpbmcgc3RyYXRlZ3lcbiAgdHJhbnNmb3JtPzogKGl0ZW06IGFueSkgPT4gYW55OyAvLyBJdGVtIHRyYW5zZm9ybWF0aW9uIGZ1bmN0aW9uXG4gIGNhbmNlbExvYWRUaHJlc2hvbGQ/OiBudW1iZXI7IC8vIFZlbG9jaXR5IHRocmVzaG9sZCBmb3IgY2FuY2VsbGluZyBsb2Fkc1xuICBtYXhDb25jdXJyZW50UmVxdWVzdHM/OiBudW1iZXI7XG4gIGVuYWJsZVJlcXVlc3RRdWV1ZT86IGJvb2xlYW47XG4gIG1heFF1ZXVlU2l6ZT86IG51bWJlcjtcbiAgbG9hZE9uRHJhZ0VuZD86IGJvb2xlYW47IC8vIEVuYWJsZSBsb2FkaW5nIHdoZW4gZHJhZyBlbmRzIChzYWZldHkgbWVhc3VyZSlcbiAgaW5pdGlhbFNjcm9sbEluZGV4PzogbnVtYmVyOyAvLyBJbml0aWFsIHNjcm9sbCBwb3NpdGlvbiAoMC1iYXNlZCBpbmRleClcbiAgc2VsZWN0SWQ/OiBzdHJpbmcgfCBudW1iZXI7IC8vIElEIG9mIGl0ZW0gdG8gc2VsZWN0IGFmdGVyIGluaXRpYWwgbG9hZCBjb21wbGV0ZXNcbiAgYXV0b0xvYWQ/OiBib29sZWFuOyAvLyBXaGV0aGVyIHRvIGF1dG9tYXRpY2FsbHkgbG9hZCBkYXRhIG9uIGluaXRpYWxpemF0aW9uIChkZWZhdWx0OiB0cnVlKVxuICBhdXRvU2VsZWN0Rmlyc3Q/OiBib29sZWFuOyAvLyBBdXRvbWF0aWNhbGx5IHNlbGVjdCBmaXJzdCBpdGVtIGFmdGVyIGluaXRpYWwgbG9hZCAoZGVmYXVsdDogZmFsc2UpXG4gIC8qKiBNYXhpbXVtIGl0ZW1zIHRvIGtlZXAgaW4gbWVtb3J5IChkZWZhdWx0OiAxMDAwKSAqL1xuICBtYXhDYWNoZWRJdGVtcz86IG51bWJlcjtcbiAgLyoqIEV4dHJhIGl0ZW1zIHRvIGtlZXAgYXJvdW5kIHZpc2libGUgcmFuZ2UgZHVyaW5nIGV2aWN0aW9uIChkZWZhdWx0OiAxNTApICovXG4gIGV2aWN0aW9uQnVmZmVyPzogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENvbGxlY3Rpb25Db21wb25lbnQge1xuICBjb2xsZWN0aW9uOiB7XG4gICAgbG9hZFJhbmdlOiAob2Zmc2V0OiBudW1iZXIsIGxpbWl0OiBudW1iZXIpID0+IFByb21pc2U8YW55W10+O1xuICAgIGxvYWRNaXNzaW5nUmFuZ2VzOiAoXG4gICAgICByYW5nZTogeyBzdGFydDogbnVtYmVyOyBlbmQ6IG51bWJlciB9LFxuICAgICAgY2FsbGVyPzogc3RyaW5nLFxuICAgICkgPT4gUHJvbWlzZTx2b2lkPjtcbiAgICBnZXRMb2FkZWRSYW5nZXM6ICgpID0+IFNldDxudW1iZXI+O1xuICAgIGdldFBlbmRpbmdSYW5nZXM6ICgpID0+IFNldDxudW1iZXI+O1xuICAgIGNsZWFyRmFpbGVkUmFuZ2VzOiAoKSA9PiB2b2lkO1xuICAgIGNsZWFyUXVldWU6ICgpID0+IHZvaWQ7XG4gICAgc2V0TWFudWFsbHlMb2FkZWQ6ICgpID0+IHZvaWQ7XG4gICAgc2V0UmVsb2FkSW5Qcm9ncmVzczogKFxuICAgICAgaW5Qcm9ncmVzczogYm9vbGVhbixcbiAgICAgIHRhcmdldFJhbmdlPzogeyBzdGFydDogbnVtYmVyOyBlbmQ6IG51bWJlciB9LFxuICAgICkgPT4gdm9pZDtcbiAgICByZXRyeUZhaWxlZFJhbmdlOiAocmFuZ2VJZDogbnVtYmVyKSA9PiBQcm9taXNlPGFueVtdPjtcbiAgICBzZXRUb3RhbEl0ZW1zOiAodG90YWw6IG51bWJlcikgPT4gdm9pZDtcbiAgICBnZXRUb3RhbEl0ZW1zOiAoKSA9PiBudW1iZXI7XG4gICAgLy8gQ3Vyc29yLXNwZWNpZmljIG1ldGhvZHNcbiAgICBnZXRDdXJyZW50Q3Vyc29yOiAoKSA9PiBzdHJpbmcgfCBudWxsO1xuICAgIGdldEN1cnNvckZvclBhZ2U6IChwYWdlOiBudW1iZXIpID0+IHN0cmluZyB8IG51bGw7XG4gICAgLy8gUmVzZXQgbWV0aG9kIGZvciByZWxvYWQgZnVuY3Rpb25hbGl0eVxuICAgIHJlc2V0OiAoKSA9PiB2b2lkO1xuICB9O1xufVxuXG4vKipcbiAqIEFkZHMgY29sbGVjdGlvbiBmdW5jdGlvbmFsaXR5IHRvIHZpZXdwb3J0IGNvbXBvbmVudFxuICovXG5leHBvcnQgZnVuY3Rpb24gd2l0aENvbGxlY3Rpb24oY29uZmlnOiBDb2xsZWN0aW9uQ29uZmlnID0ge30pIHtcbiAgcmV0dXJuIDxUIGV4dGVuZHMgVmlld3BvcnRDb250ZXh0ICYgVmlld3BvcnRDb21wb25lbnQ+KFxuICAgIGNvbXBvbmVudDogVCxcbiAgKTogVCAmIENvbGxlY3Rpb25Db21wb25lbnQgPT4ge1xuICAgIGNvbnN0IHtcbiAgICAgIGNvbGxlY3Rpb24sXG4gICAgICByYW5nZVNpemUgPSBWSUVXUE9SVF9DT05TVEFOVFMuTE9BRElORy5ERUZBVUxUX1JBTkdFX1NJWkUsXG4gICAgICBzdHJhdGVneSA9IFwib2Zmc2V0XCIsXG4gICAgICB0cmFuc2Zvcm0sXG4gICAgICBjYW5jZWxMb2FkVGhyZXNob2xkID0gVklFV1BPUlRfQ09OU1RBTlRTLkxPQURJTkcuQ0FOQ0VMX1RIUkVTSE9MRCxcbiAgICAgIG1heENvbmN1cnJlbnRSZXF1ZXN0cyA9IFZJRVdQT1JUX0NPTlNUQU5UUy5MT0FESU5HXG4gICAgICAgIC5NQVhfQ09OQ1VSUkVOVF9SRVFVRVNUUyxcbiAgICAgIGVuYWJsZVJlcXVlc3RRdWV1ZSA9IFZJRVdQT1JUX0NPTlNUQU5UUy5SRVFVRVNUX1FVRVVFLkVOQUJMRUQsXG4gICAgICBtYXhRdWV1ZVNpemUgPSBWSUVXUE9SVF9DT05TVEFOVFMuUkVRVUVTVF9RVUVVRS5NQVhfUVVFVUVfU0laRSxcbiAgICAgIGxvYWRPbkRyYWdFbmQgPSB0cnVlLCAvLyBEZWZhdWx0IHRvIHRydWUgYXMgc2FmZXR5IG1lYXN1cmVcbiAgICAgIGluaXRpYWxTY3JvbGxJbmRleCA9IDAsIC8vIFN0YXJ0IGZyb20gYmVnaW5uaW5nIGJ5IGRlZmF1bHRcbiAgICAgIHNlbGVjdElkLCAvLyBJRCBvZiBpdGVtIHRvIHNlbGVjdCBhZnRlciBpbml0aWFsIGxvYWRcbiAgICAgIGF1dG9Mb2FkID0gdHJ1ZSwgLy8gQXV0by1sb2FkIGluaXRpYWwgZGF0YSBieSBkZWZhdWx0XG4gICAgICBhdXRvU2VsZWN0Rmlyc3QgPSBmYWxzZSwgLy8gQXV0b21hdGljYWxseSBzZWxlY3QgZmlyc3QgaXRlbSBhZnRlciBpbml0aWFsIGxvYWRcbiAgICAgIG1heENhY2hlZEl0ZW1zID0gMTAwMCwgLy8gTWF4aW11bSBpdGVtcyB0byBrZWVwIGluIG1lbW9yeVxuICAgICAgZXZpY3Rpb25CdWZmZXIgPSAxNTAsIC8vIEV4dHJhIGl0ZW1zIHRvIGtlZXAgYXJvdW5kIHZpc2libGUgcmFuZ2VcbiAgICB9ID0gY29uZmlnO1xuXG4gICAgLy8gVHJhY2sgaWYgd2UndmUgY29tcGxldGVkIHRoZSBpbml0aWFsIGxvYWQgZm9yIGluaXRpYWxTY3JvbGxJbmRleFxuICAgIC8vIFRoaXMgcHJldmVudHMgdGhlIHZpZXdwb3J0OnJhbmdlLWNoYW5nZWQgbGlzdGVuZXIgZnJvbSBsb2FkaW5nIHBhZ2UgMVxuICAgIGxldCBoYXNDb21wbGV0ZWRJbml0aWFsUG9zaXRpb25Mb2FkID0gZmFsc2U7XG4gICAgY29uc3QgaGFzSW5pdGlhbFNjcm9sbEluZGV4ID0gaW5pdGlhbFNjcm9sbEluZGV4ID4gMDtcblxuICAgIC8vIFRyYWNrIGlmIGRhdGEgaGFzIGJlZW4gbWFudWFsbHkgbG9hZGVkIChlLmcuLCB2aWEgcmVsb2FkQXQpXG4gICAgLy8gVGhpcyBlbmFibGVzIHN1YnNlcXVlbnQgbG9hZHMgZnJvbSB2aWV3cG9ydDpyYW5nZS1jaGFuZ2VkIGV2ZW4gd2l0aCBhdXRvTG9hZDogZmFsc2VcbiAgICBsZXQgaGFzTWFudWFsbHlMb2FkZWQgPSBmYWxzZTtcblxuICAgIC8vIFRyYWNrIGlmIGEgcmVsb2FkIGlzIGluIHByb2dyZXNzIChyZWxvYWRBdCBzZXRzIHRoaXMgdG8gcHJldmVudCBzdGFsZSBwYWdlIDEgbG9hZHMpXG4gICAgLy8gVGhpcyBpcyBzZXQgYnkgcmVsb2FkQXQgQkVGT1JFIGl0IHN0YXJ0cyBsb2FkaW5nLCBhbmQgY2xlYXJlZCB3aGVuIGxvYWRpbmcgY29tcGxldGVzXG4gICAgbGV0IHJlbG9hZEluUHJvZ3Jlc3MgPSBmYWxzZTtcbiAgICBsZXQgcmVsb2FkVGFyZ2V0UmFuZ2U6IHsgc3RhcnQ6IG51bWJlcjsgZW5kOiBudW1iZXIgfSB8IG51bGwgPSBudWxsO1xuXG4gICAgLy8gY29uc29sZS5sb2coXCJbVmlld3BvcnQgQ29sbGVjdGlvbl0gSW5pdGlhbGl6ZWQgd2l0aCBjb25maWc6XCIsIHtcbiAgICAvLyAgIHN0cmF0ZWd5LFxuICAgIC8vICAgcmFuZ2VTaXplLFxuICAgIC8vICAgaW5pdGlhbFNjcm9sbEluZGV4LFxuICAgIC8vIH0pO1xuXG4gICAgLy8gTG9hZGluZyBtYW5hZ2VyIHN0YXRlXG4gICAgaW50ZXJmYWNlIFF1ZXVlZFJlcXVlc3Qge1xuICAgICAgcmFuZ2U6IHsgc3RhcnQ6IG51bWJlcjsgZW5kOiBudW1iZXIgfTtcbiAgICAgIHByaW9yaXR5OiBcImhpZ2hcIiB8IFwibm9ybWFsXCIgfCBcImxvd1wiO1xuICAgICAgdGltZXN0YW1wOiBudW1iZXI7XG4gICAgICByZXNvbHZlOiAoKSA9PiB2b2lkO1xuICAgICAgcmVqZWN0OiAoZXJyb3I6IGFueSkgPT4gdm9pZDtcbiAgICAgIGNhbGxlcj86IHN0cmluZztcbiAgICB9XG5cbiAgICAvLyBTdGF0ZSB0cmFja2luZ1xuICAgIGxldCBjdXJyZW50VmVsb2NpdHkgPSAwO1xuICAgIGxldCBhY3RpdmVMb2FkQ291bnQgPSAwO1xuICAgIGxldCBjb21wbGV0ZWRMb2FkcyA9IDA7XG4gICAgbGV0IGZhaWxlZExvYWRzID0gMDtcbiAgICBsZXQgY2FuY2VsbGVkTG9hZHMgPSAwO1xuICAgIGxldCBpc0RyYWdnaW5nID0gZmFsc2U7IC8vIFRyYWNrIGRyYWcgc3RhdGVcblxuICAgIC8vIENhY2hlIGV2aWN0aW9uIGNvbmZpZ3VyYXRpb24gKGZyb20gY29uZmlnKVxuICAgIGNvbnN0IE1BWF9DQUNIRURfSVRFTVMgPSBtYXhDYWNoZWRJdGVtcztcbiAgICBjb25zdCBFVklDVElPTl9CVUZGRVIgPSBldmljdGlvbkJ1ZmZlcjtcblxuICAgIC8vIFRyYWNrIGFjdHVhbCBpdGVtIGNvdW50IGVmZmljaWVudGx5IChhdm9pZCBPKG4pIHNwYXJzZSBhcnJheSBpdGVyYXRpb24pXG4gICAgbGV0IGNhY2hlZEl0ZW1Db3VudCA9IDA7XG5cbiAgICAvLyBNZW1vcnkgZGlhZ25vc3RpY3NcbiAgICBjb25zdCBsb2dNZW1vcnlTdGF0cyA9IChjYWxsZXI6IHN0cmluZykgPT4ge1xuICAgICAgY29uc3QgbG9hZGVkUmFuZ2VDb3VudCA9IGxvYWRlZFJhbmdlcy5zaXplO1xuICAgICAgY29uc3QgcGVuZGluZ1JhbmdlQ291bnQgPSBwZW5kaW5nUmFuZ2VzLnNpemU7XG4gICAgICBjb25zdCBhYm9ydENvbnRyb2xsZXJDb3VudCA9IGFib3J0Q29udHJvbGxlcnMuc2l6ZTtcbiAgICB9O1xuXG4gICAgLyoqXG4gICAgICogRXZpY3QgaXRlbXMgZmFyIGZyb20gdGhlIGN1cnJlbnQgdmlzaWJsZSByYW5nZSB0byBwcmV2ZW50IG1lbW9yeSBibG9hdFxuICAgICAqIEtlZXBzIGl0ZW1zIHdpdGhpbiBFVklDVElPTl9CVUZGRVIgb2YgdGhlIHZpc2libGUgcmFuZ2VcbiAgICAgKlxuICAgICAqIFBFUkZPUk1BTkNFOiBVc2VzIGxvYWRlZFJhbmdlcyB0byBpdGVyYXRlIG9ubHkgbG9hZGVkIGRhdGEsIG5vdCB0aGUgZW50aXJlXG4gICAgICogc3BhcnNlIGFycmF5LiBUaGlzIGlzIGNyaXRpY2FsIGZvciBsYXJnZSBsaXN0cyAoMU0rIGl0ZW1zKSB3aGVyZSB0aGVcbiAgICAgKiBpdGVtcyBhcnJheSBpcyBzcGFyc2UgYW5kIGl0ZXJhdGluZyBpdGVtcy5sZW5ndGggd291bGQgYmUgTyhuKS5cbiAgICAgKi9cbiAgICBjb25zdCBldmljdERpc3RhbnRJdGVtcyA9ICh2aXNpYmxlU3RhcnQ6IG51bWJlciwgdmlzaWJsZUVuZDogbnVtYmVyKSA9PiB7XG4gICAgICAvLyBPbmx5IGV2aWN0IGlmIHdlIGhhdmUgbW9yZSB0aGFuIE1BWF9DQUNIRURfSVRFTVNcbiAgICAgIC8vIFVzZSBjYWNoZWQgY291bnQgaW5zdGVhZCBvZiBPKG4pIGl0ZW1zLmZpbHRlcihCb29sZWFuKS5sZW5ndGhcbiAgICAgIGlmIChjYWNoZWRJdGVtQ291bnQgPD0gTUFYX0NBQ0hFRF9JVEVNUykge1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGtlZXBTdGFydCA9IE1hdGgubWF4KDAsIHZpc2libGVTdGFydCAtIEVWSUNUSU9OX0JVRkZFUik7XG4gICAgICBjb25zdCBrZWVwRW5kID0gdmlzaWJsZUVuZCArIEVWSUNUSU9OX0JVRkZFUjtcblxuICAgICAgLy8gQ2FsY3VsYXRlIHdoaWNoIHJhbmdlcyB0byBrZWVwIGJhc2VkIG9uIHZpc2libGUgYXJlYVxuICAgICAgY29uc3Qga2VlcFJhbmdlU3RhcnQgPSBNYXRoLmZsb29yKGtlZXBTdGFydCAvIHJhbmdlU2l6ZSk7XG4gICAgICBjb25zdCBrZWVwUmFuZ2VFbmQgPSBNYXRoLmZsb29yKGtlZXBFbmQgLyByYW5nZVNpemUpO1xuXG4gICAgICBsZXQgZXZpY3RlZENvdW50ID0gMDtcbiAgICAgIGNvbnN0IHJhbmdlc1RvUmVtb3ZlID0gbmV3IFNldDxudW1iZXI+KCk7XG5cbiAgICAgIC8vIEl0ZXJhdGUgb25seSBvdmVyIGxvYWRlZCByYW5nZXMsIG5vdCB0aGUgZW50aXJlIHNwYXJzZSBhcnJheVxuICAgICAgLy8gVGhpcyBpcyBPKGxvYWRlZCByYW5nZXMpIGluc3RlYWQgb2YgTyh0b3RhbCBpdGVtcylcbiAgICAgIGxvYWRlZFJhbmdlcy5mb3JFYWNoKChyYW5nZUlkKSA9PiB7XG4gICAgICAgIC8vIENoZWNrIGlmIHRoaXMgcmFuZ2UgaXMgb3V0c2lkZSB0aGUga2VlcCB6b25lXG4gICAgICAgIGlmIChyYW5nZUlkIDwga2VlcFJhbmdlU3RhcnQgfHwgcmFuZ2VJZCA+IGtlZXBSYW5nZUVuZCkge1xuICAgICAgICAgIHJhbmdlc1RvUmVtb3ZlLmFkZChyYW5nZUlkKTtcblxuICAgICAgICAgIC8vIEV2aWN0IGl0ZW1zIGluIHRoaXMgcmFuZ2VcbiAgICAgICAgICBjb25zdCByYW5nZVN0YXJ0ID0gcmFuZ2VJZCAqIHJhbmdlU2l6ZTtcbiAgICAgICAgICBjb25zdCByYW5nZUVuZCA9IHJhbmdlU3RhcnQgKyByYW5nZVNpemU7XG4gICAgICAgICAgZm9yIChsZXQgaSA9IHJhbmdlU3RhcnQ7IGkgPCByYW5nZUVuZDsgaSsrKSB7XG4gICAgICAgICAgICBpZiAoaXRlbXNbaV0gIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICBkZWxldGUgaXRlbXNbaV07XG4gICAgICAgICAgICAgIGV2aWN0ZWRDb3VudCsrO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSk7XG5cbiAgICAgIC8vIFJlbW92ZSBhbGwgcmFuZ2VzIHRoYXQgaGFkIGl0ZW1zIGV2aWN0ZWRcbiAgICAgIC8vIFRoaXMgaXMgY3JpdGljYWw6IGV2ZW4gaWYgb25seSBwYXJ0IG9mIGEgcmFuZ2Ugd2FzIGV2aWN0ZWQsXG4gICAgICAvLyB3ZSBtdXN0IHJlbW92ZSBpdCBmcm9tIGxvYWRlZFJhbmdlcyBzbyBpdCBnZXRzIHJlbG9hZGVkXG4gICAgICAvLyBBbHNvIHJlbW92ZSBmcm9tIHBlbmRpbmdSYW5nZXMgLSBpZiBhIHNsb3cgcmVxdWVzdCBjb21wbGV0ZXMgZm9yIGFuIGV2aWN0ZWQgcmFuZ2UsXG4gICAgICAvLyB3ZSBkb24ndCB3YW50IGl0IHRvIGJsb2NrIGZ1dHVyZSBsb2FkcyB3aGVuIHVzZXIgc2Nyb2xscyBiYWNrXG4gICAgICByYW5nZXNUb1JlbW92ZS5mb3JFYWNoKChyYW5nZUlkKSA9PiB7XG4gICAgICAgIGxvYWRlZFJhbmdlcy5kZWxldGUocmFuZ2VJZCk7XG4gICAgICAgIHBlbmRpbmdSYW5nZXMuZGVsZXRlKHJhbmdlSWQpO1xuICAgICAgICAvLyBDbGVhbiB1cCBhY3RpdmUgcmVxdWVzdHMgdHJhY2tpbmdcbiAgICAgICAgYWN0aXZlUmVxdWVzdHMuZGVsZXRlKHJhbmdlSWQpO1xuICAgICAgICAvLyBBbHNvIGFib3J0IGFueSBpbi1mbGlnaHQgcmVxdWVzdCBmb3IgdGhpcyByYW5nZSB0byBmcmVlIHVwIG5ldHdvcmsgcmVzb3VyY2VzXG4gICAgICAgIGNvbnN0IGNvbnRyb2xsZXIgPSBhYm9ydENvbnRyb2xsZXJzLmdldChyYW5nZUlkKTtcbiAgICAgICAgaWYgKGNvbnRyb2xsZXIpIHtcbiAgICAgICAgICBjb250cm9sbGVyLmFib3J0KCk7XG4gICAgICAgICAgYWJvcnRDb250cm9sbGVycy5kZWxldGUocmFuZ2VJZCk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuXG4gICAgICBpZiAoZXZpY3RlZENvdW50ID4gMCkge1xuICAgICAgICAvLyBVcGRhdGUgY2FjaGVkIGl0ZW0gY291bnRcbiAgICAgICAgY2FjaGVkSXRlbUNvdW50IC09IGV2aWN0ZWRDb3VudDtcblxuICAgICAgICAvLyBFbWl0IGV2ZW50IGZvciByZW5kZXJpbmcgdG8gYWxzbyBjbGVhbiB1cFxuICAgICAgICBjb21wb25lbnQuZW1pdD8uKFwiY29sbGVjdGlvbjppdGVtcy1ldmljdGVkXCIsIHtcbiAgICAgICAgICBrZWVwU3RhcnQsXG4gICAgICAgICAga2VlcEVuZCxcbiAgICAgICAgICBldmljdGVkQ291bnQsXG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIH07XG5cbiAgICBjb25zdCBhY3RpdmVMb2FkUmFuZ2VzID0gbmV3IFNldDxzdHJpbmc+KCk7XG4gICAgbGV0IGxvYWRSZXF1ZXN0UXVldWU6IFF1ZXVlZFJlcXVlc3RbXSA9IFtdO1xuXG4gICAgLy8gQWJvcnRDb250cm9sbGVyIG1hcCBmb3IgY2FuY2VsbGluZyBpbi1mbGlnaHQgcmVxdWVzdHNcbiAgICBjb25zdCBhYm9ydENvbnRyb2xsZXJzID0gbmV3IE1hcDxudW1iZXIsIEFib3J0Q29udHJvbGxlcj4oKTtcblxuICAgIC8vIFN0YXRlXG4gICAgbGV0IGl0ZW1zOiBhbnlbXSA9IFtdO1xuICAgIGxldCB0b3RhbEl0ZW1zID0gMDtcbiAgICBsZXQgbG9hZGVkUmFuZ2VzID0gbmV3IFNldDxudW1iZXI+KCk7XG4gICAgbGV0IHBlbmRpbmdSYW5nZXMgPSBuZXcgU2V0PG51bWJlcj4oKTtcbiAgICBsZXQgZmFpbGVkUmFuZ2VzID0gbmV3IE1hcDxcbiAgICAgIG51bWJlcixcbiAgICAgIHtcbiAgICAgICAgYXR0ZW1wdHM6IG51bWJlcjtcbiAgICAgICAgbGFzdEVycm9yOiBFcnJvcjtcbiAgICAgICAgdGltZXN0YW1wOiBudW1iZXI7XG4gICAgICB9XG4gICAgPigpO1xuICAgIGxldCBhY3RpdmVSZXF1ZXN0cyA9IG5ldyBNYXA8bnVtYmVyLCBQcm9taXNlPGFueVtdPj4oKTtcblxuICAgIC8vIEN1cnNvciBwYWdpbmF0aW9uIHN0YXRlXG4gICAgbGV0IGN1cnJlbnRDdXJzb3I6IHN0cmluZyB8IG51bGwgPSBudWxsO1xuICAgIGxldCBjdXJzb3JNYXAgPSBuZXcgTWFwPG51bWJlciwgc3RyaW5nPigpOyAvLyBNYXAgcGFnZSBudW1iZXIgdG8gY3Vyc29yXG4gICAgbGV0IHBhZ2VUb09mZnNldE1hcCA9IG5ldyBNYXA8bnVtYmVyLCBudW1iZXI+KCk7IC8vIE1hcCBwYWdlIHRvIGFjdHVhbCBvZmZzZXRcbiAgICBsZXQgaGlnaGVzdExvYWRlZFBhZ2UgPSAwO1xuICAgIGxldCBkaXNjb3ZlcmVkVG90YWw6IG51bWJlciB8IG51bGwgPSBudWxsOyAvLyBUcmFjayBkaXNjb3ZlcmVkIHRvdGFsIGZyb20gQVBJXG4gICAgbGV0IGhhc1JlYWNoZWRFbmQgPSBmYWxzZTsgLy8gVHJhY2sgaWYgd2UndmUgcmVhY2hlZCB0aGUgZW5kIG9mIGRhdGFcblxuICAgIC8vIFNoYXJlIGl0ZW1zIGFycmF5IHdpdGggY29tcG9uZW50XG4gICAgY29tcG9uZW50Lml0ZW1zID0gaXRlbXM7XG5cbiAgICAvKipcbiAgICAgKiBHZXQgYSB1bmlxdWUgSUQgZm9yIGEgcmFuZ2UgYmFzZWQgb24gb2Zmc2V0IGFuZCB0aGUgYmFzZSByYW5nZSBzaXplXG4gICAgICovXG4gICAgY29uc3QgZ2V0UmFuZ2VJZCA9IChvZmZzZXQ6IG51bWJlciwgbGltaXQ6IG51bWJlcik6IG51bWJlciA9PiB7XG4gICAgICAvLyBBbHdheXMgdXNlIHRoZSBiYXNlIHJhbmdlU2l6ZSBmb3IgY29uc2lzdGVudCBJRHNcbiAgICAgIC8vIFRoaXMgZW5zdXJlcyBtZXJnZWQgcmFuZ2VzIGNhbiBiZSB0cmFja2VkIHByb3Blcmx5XG4gICAgICByZXR1cm4gTWF0aC5mbG9vcihvZmZzZXQgLyByYW5nZVNpemUpO1xuICAgIH07XG5cbiAgICAvLyBMb2FkaW5nIG1hbmFnZXIgaGVscGVyc1xuICAgIGNvbnN0IGdldFJhbmdlS2V5ID0gKHJhbmdlOiB7IHN0YXJ0OiBudW1iZXI7IGVuZDogbnVtYmVyIH0pOiBzdHJpbmcgPT4ge1xuICAgICAgcmV0dXJuIGAke3JhbmdlLnN0YXJ0fS0ke3JhbmdlLmVuZH1gO1xuICAgIH07XG5cbiAgICBjb25zdCBjYW5Mb2FkID0gKCk6IGJvb2xlYW4gPT4ge1xuICAgICAgcmV0dXJuIGN1cnJlbnRWZWxvY2l0eSA8PSBjYW5jZWxMb2FkVGhyZXNob2xkO1xuICAgIH07XG5cbiAgICBjb25zdCBwcm9jZXNzUXVldWUgPSAoKSA9PiB7XG4gICAgICBpZiAoIWVuYWJsZVJlcXVlc3RRdWV1ZSB8fCBsb2FkUmVxdWVzdFF1ZXVlLmxlbmd0aCA9PT0gMCkgcmV0dXJuO1xuXG4gICAgICAvLyBTb3J0IHF1ZXVlIGJ5IHByaW9yaXR5IGFuZCB0aW1lc3RhbXBcbiAgICAgIGxvYWRSZXF1ZXN0UXVldWUuc29ydCgoYSwgYikgPT4ge1xuICAgICAgICBjb25zdCBwcmlvcml0eU9yZGVyID0geyBoaWdoOiAwLCBub3JtYWw6IDEsIGxvdzogMiB9O1xuICAgICAgICBjb25zdCBwcmlvcml0eURpZmYgPVxuICAgICAgICAgIHByaW9yaXR5T3JkZXJbYS5wcmlvcml0eV0gLSBwcmlvcml0eU9yZGVyW2IucHJpb3JpdHldO1xuICAgICAgICByZXR1cm4gcHJpb3JpdHlEaWZmICE9PSAwID8gcHJpb3JpdHlEaWZmIDogYS50aW1lc3RhbXAgLSBiLnRpbWVzdGFtcDtcbiAgICAgIH0pO1xuXG4gICAgICAvLyBQcm9jZXNzIHJlcXVlc3RzIHVwIHRvIGNhcGFjaXR5XG4gICAgICB3aGlsZSAoXG4gICAgICAgIGxvYWRSZXF1ZXN0UXVldWUubGVuZ3RoID4gMCAmJlxuICAgICAgICBhY3RpdmVMb2FkQ291bnQgPCBtYXhDb25jdXJyZW50UmVxdWVzdHNcbiAgICAgICkge1xuICAgICAgICBjb25zdCByZXF1ZXN0ID0gbG9hZFJlcXVlc3RRdWV1ZS5zaGlmdCgpO1xuICAgICAgICBpZiAocmVxdWVzdCkge1xuICAgICAgICAgIGV4ZWN1dGVRdWV1ZWRMb2FkKHJlcXVlc3QpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfTtcblxuICAgIGNvbnN0IGV4ZWN1dGVRdWV1ZWRMb2FkID0gKHJlcXVlc3Q6IFF1ZXVlZFJlcXVlc3QpID0+IHtcbiAgICAgIGFjdGl2ZUxvYWRDb3VudCsrO1xuICAgICAgYWN0aXZlTG9hZFJhbmdlcy5hZGQoZ2V0UmFuZ2VLZXkocmVxdWVzdC5yYW5nZSkpO1xuXG4gICAgICAvLyBDYWxsIHRoZSBhY3R1YWwgbG9hZE1pc3NpbmdSYW5nZXMgZnVuY3Rpb25cbiAgICAgIGxvYWRNaXNzaW5nUmFuZ2VzSW50ZXJuYWwocmVxdWVzdC5yYW5nZSwgcmVxdWVzdC5jYWxsZXIpXG4gICAgICAgIC50aGVuKCgpID0+IHtcbiAgICAgICAgICByZXF1ZXN0LnJlc29sdmUoKTtcbiAgICAgICAgICBjb21wbGV0ZWRMb2FkcysrO1xuICAgICAgICB9KVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgIHJlcXVlc3QucmVqZWN0KGVycm9yKTtcbiAgICAgICAgICBmYWlsZWRMb2FkcysrO1xuICAgICAgICB9KVxuICAgICAgICAuZmluYWxseSgoKSA9PiB7XG4gICAgICAgICAgYWN0aXZlTG9hZENvdW50LS07XG4gICAgICAgICAgYWN0aXZlTG9hZFJhbmdlcy5kZWxldGUoZ2V0UmFuZ2VLZXkocmVxdWVzdC5yYW5nZSkpO1xuICAgICAgICAgIHByb2Nlc3NRdWV1ZSgpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgLyoqXG4gICAgICogVHJhbnNmb3JtIGl0ZW1zIGlmIHRyYW5zZm9ybSBmdW5jdGlvbiBwcm92aWRlZFxuICAgICAqL1xuICAgIGNvbnN0IHRyYW5zZm9ybUl0ZW1zID0gKHJhd0l0ZW1zOiBhbnlbXSk6IGFueVtdID0+IHtcbiAgICAgIGlmICghdHJhbnNmb3JtKSByZXR1cm4gcmF3SXRlbXM7XG4gICAgICByZXR1cm4gcmF3SXRlbXMubWFwKHRyYW5zZm9ybSk7XG4gICAgfTtcblxuICAgIC8qKlxuICAgICAqIExvYWQgYSByYW5nZSBvZiBkYXRhXG4gICAgICovXG4gICAgY29uc3QgbG9hZFJhbmdlID0gYXN5bmMgKFxuICAgICAgb2Zmc2V0OiBudW1iZXIsXG4gICAgICBsaW1pdDogbnVtYmVyLFxuICAgICAgY2FsbGVyPzogc3RyaW5nLFxuICAgICk6IFByb21pc2U8YW55W10+ID0+IHtcbiAgICAgIGlmICghY29sbGVjdGlvbikge1xuICAgICAgICBjb25zb2xlLndhcm4oXCJbQ29sbGVjdGlvbl0gTm8gY29sbGVjdGlvbiBhZGFwdGVyIGNvbmZpZ3VyZWRcIik7XG4gICAgICAgIHJldHVybiBbXTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgcmFuZ2VJZCA9IGdldFJhbmdlSWQob2Zmc2V0LCBsaW1pdCk7XG4gICAgICAvLyBjb25zb2xlLmxvZyhgW0NvbGxlY3Rpb25dIFJhbmdlIElEOiAke3JhbmdlSWR9YCk7XG5cbiAgICAgIC8vIENoZWNrIGlmIGFscmVhZHkgbG9hZGVkXG4gICAgICBpZiAobG9hZGVkUmFuZ2VzLmhhcyhyYW5nZUlkKSkge1xuICAgICAgICByZXR1cm4gaXRlbXMuc2xpY2Uob2Zmc2V0LCBvZmZzZXQgKyBsaW1pdCk7XG4gICAgICB9XG5cbiAgICAgIC8vIENoZWNrIGlmIGFscmVhZHkgcGVuZGluZ1xuICAgICAgaWYgKHBlbmRpbmdSYW5nZXMuaGFzKHJhbmdlSWQpKSB7XG4gICAgICAgIGNvbnN0IGV4aXN0aW5nUmVxdWVzdCA9IGFjdGl2ZVJlcXVlc3RzLmdldChyYW5nZUlkKTtcbiAgICAgICAgaWYgKGV4aXN0aW5nUmVxdWVzdCkge1xuICAgICAgICAgIHJldHVybiBleGlzdGluZ1JlcXVlc3Q7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgLy8gTWFyayBhcyBwZW5kaW5nXG4gICAgICBwZW5kaW5nUmFuZ2VzLmFkZChyYW5nZUlkKTtcblxuICAgICAgLy8gQ3JlYXRlIEFib3J0Q29udHJvbGxlciBmb3IgdGhpcyByZXF1ZXN0XG4gICAgICBjb25zdCBhYm9ydENvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG4gICAgICBhYm9ydENvbnRyb2xsZXJzLnNldChyYW5nZUlkLCBhYm9ydENvbnRyb2xsZXIpO1xuXG4gICAgICAvLyBDcmVhdGUgcmVxdWVzdCBwcm9taXNlXG4gICAgICBjb25zdCByZXF1ZXN0UHJvbWlzZSA9IChhc3luYyAoKSA9PiB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgLy8gQ2FsbCBjb2xsZWN0aW9uIGFkYXB0ZXIgd2l0aCBhcHByb3ByaWF0ZSBwYXJhbWV0ZXJzXG4gICAgICAgICAgY29uc3QgcGFnZSA9IE1hdGguZmxvb3Iob2Zmc2V0IC8gbGltaXQpICsgMTtcbiAgICAgICAgICBsZXQgcGFyYW1zOiBhbnk7XG5cbiAgICAgICAgICBpZiAoc3RyYXRlZ3kgPT09IFwiY3Vyc29yXCIpIHtcbiAgICAgICAgICAgIC8vIEZvciBjdXJzb3IgcGFnaW5hdGlvblxuICAgICAgICAgICAgaWYgKHBhZ2UgPT09IDEpIHtcbiAgICAgICAgICAgICAgLy8gRmlyc3QgcGFnZSAtIG5vIGN1cnNvclxuICAgICAgICAgICAgICBwYXJhbXMgPSB7IGxpbWl0IH07XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAvLyBDaGVjayBpZiB3ZSBoYXZlIGN1cnNvciBmb3IgcHJldmlvdXMgcGFnZVxuICAgICAgICAgICAgICBjb25zdCBwcmV2UGFnZUN1cnNvciA9IGN1cnNvck1hcC5nZXQocGFnZSAtIDEpO1xuICAgICAgICAgICAgICBpZiAoIXByZXZQYWdlQ3Vyc29yKSB7XG4gICAgICAgICAgICAgICAgLy8gQ2FuJ3QgbG9hZCB0aGlzIHBhZ2Ugd2l0aG91dCBwcmV2aW91cyBjdXJzb3JcbiAgICAgICAgICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgICAgICAgICBgW0NvbGxlY3Rpb25dIENhbm5vdCBsb2FkIHBhZ2UgJHtwYWdlfSB3aXRob3V0IGN1cnNvciBmb3IgcGFnZSAke1xuICAgICAgICAgICAgICAgICAgICBwYWdlIC0gMVxuICAgICAgICAgICAgICAgICAgfWAsXG4gICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICAgICAgICBgU2VxdWVudGlhbCBsb2FkaW5nIHJlcXVpcmVkIC0gbWlzc2luZyBjdXJzb3IgZm9yIHBhZ2UgJHtcbiAgICAgICAgICAgICAgICAgICAgcGFnZSAtIDFcbiAgICAgICAgICAgICAgICAgIH1gLFxuICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgcGFyYW1zID0geyBjdXJzb3I6IHByZXZQYWdlQ3Vyc29yLCBsaW1pdCB9O1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gZWxzZSBpZiAoc3RyYXRlZ3kgPT09IFwicGFnZVwiKSB7XG4gICAgICAgICAgICBwYXJhbXMgPSB7IHBhZ2UsIGxpbWl0IH07XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIC8vIG9mZnNldCBzdHJhdGVneVxuICAgICAgICAgICAgcGFyYW1zID0geyBvZmZzZXQsIGxpbWl0IH07XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gSW5qZWN0IHNlYXJjaCBxdWVyeSBpZiBjb21wb25lbnQgaGFzIHdpdGhTZWFyY2ggZmVhdHVyZVxuICAgICAgICAgIC8vIFVzZSBfc2VsZlJlZi5jdXJyZW50IGZvciBsYXRlIGJpbmRpbmcgKHNlYXJjaC9maWx0ZXIgZmVhdHVyZXMgYXJlIGFkZGVkIGFmdGVyIHZpZXdwb3J0KVxuICAgICAgICAgIGNvbnN0IHNlbGZSZWYgPSAoXG4gICAgICAgICAgICBjb21wb25lbnQgYXMgdW5rbm93biBhcyB7IF9zZWxmUmVmPzogU2VsZlJlZjxTZWFyY2hGaWx0ZXJDYXBhYmxlPiB9XG4gICAgICAgICAgKS5fc2VsZlJlZjtcbiAgICAgICAgICBjb25zdCBzZWxmOiBQYXJ0aWFsPFNlYXJjaEZpbHRlckNhcGFibGU+ID0gc2VsZlJlZj8uY3VycmVudCA/PyB7fTtcbiAgICAgICAgICBpZiAodHlwZW9mIHNlbGYuX3NlYXJjaFF1ZXJ5ID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgICAgICAgIGNvbnN0IHNlYXJjaFF1ZXJ5ID0gc2VsZi5fc2VhcmNoUXVlcnkoKTtcbiAgICAgICAgICAgIGlmIChzZWFyY2hRdWVyeSkge1xuICAgICAgICAgICAgICBwYXJhbXMuc2VhcmNoID0gc2VhcmNoUXVlcnk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gSW5qZWN0IGZpbHRlcnMgaWYgY29tcG9uZW50IGhhcyB3aXRoRmlsdGVyIGZlYXR1cmVcbiAgICAgICAgICBpZiAodHlwZW9mIHNlbGYuX2ZpbHRlcnMgPT09IFwiZnVuY3Rpb25cIikge1xuICAgICAgICAgICAgY29uc3QgZmlsdGVycyA9IHNlbGYuX2ZpbHRlcnMoKTtcbiAgICAgICAgICAgIGlmIChmaWx0ZXJzICYmIE9iamVjdC5rZXlzKGZpbHRlcnMpLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgICAgcGFyYW1zLmZpbHRlcnMgPSBmaWx0ZXJzO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgICAgIC8vICAgYFtWaWV3cG9ydCBDb2xsZWN0aW9uXSBMb2FkaW5nIHJhbmdlIG9mZnNldD0ke29mZnNldH0sIGxpbWl0PSR7bGltaXR9LCBzdHJhdGVneT0ke3N0cmF0ZWd5fSwgY2FsY3VsYXRlZCBwYWdlPSR7cGFnZX0sIHBhcmFtczpgLFxuICAgICAgICAgIC8vICAgSlNPTi5zdHJpbmdpZnkocGFyYW1zKVxuICAgICAgICAgIC8vICk7XG5cbiAgICAgICAgICAvLyBQYXNzIGFib3J0IHNpZ25hbCB0byB0aGUgYWRhcHRlciBpZiBpdCBzdXBwb3J0cyBpdFxuICAgICAgICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgY29sbGVjdGlvbi5yZWFkKHtcbiAgICAgICAgICAgIC4uLnBhcmFtcyxcbiAgICAgICAgICAgIHNpZ25hbDogYWJvcnRDb250cm9sbGVyLnNpZ25hbCxcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIC8vIEV4dHJhY3QgaXRlbXMgYW5kIHRvdGFsXG4gICAgICAgICAgY29uc3QgcmF3SXRlbXMgPSByZXNwb25zZS5kYXRhIHx8IHJlc3BvbnNlLml0ZW1zIHx8IHJlc3BvbnNlO1xuICAgICAgICAgIGNvbnN0IG1ldGEgPSByZXNwb25zZS5tZXRhIHx8IHt9O1xuXG4gICAgICAgICAgLy8gRm9yIGN1cnNvciBwYWdpbmF0aW9uLCB0cmFjayB0aGUgY3Vyc29yIChjaGVjayBib3RoIGN1cnNvciBhbmQgbmV4dEN1cnNvcilcbiAgICAgICAgICBjb25zdCByZXNwb25zZUN1cnNvciA9IG1ldGEuY3Vyc29yIHx8IG1ldGEubmV4dEN1cnNvcjtcbiAgICAgICAgICBpZiAoc3RyYXRlZ3kgPT09IFwiY3Vyc29yXCIgJiYgcmVzcG9uc2VDdXJzb3IpIHtcbiAgICAgICAgICAgIGN1cnJlbnRDdXJzb3IgPSByZXNwb25zZUN1cnNvcjtcbiAgICAgICAgICAgIGN1cnNvck1hcC5zZXQocGFnZSwgcmVzcG9uc2VDdXJzb3IpO1xuICAgICAgICAgICAgcGFnZVRvT2Zmc2V0TWFwLnNldChwYWdlLCBvZmZzZXQpO1xuICAgICAgICAgICAgaGlnaGVzdExvYWRlZFBhZ2UgPSBNYXRoLm1heChoaWdoZXN0TG9hZGVkUGFnZSwgcGFnZSk7XG4gICAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgIC8vICAgYFtDb2xsZWN0aW9uXSBTdG9yZWQgY3Vyc29yIGZvciBwYWdlICR7cGFnZX06ICR7cmVzcG9uc2VDdXJzb3J9YCxcbiAgICAgICAgICAgIC8vICk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gQ2hlY2sgaWYgd2UndmUgcmVhY2hlZCB0aGUgZW5kXG4gICAgICAgICAgaWYgKHN0cmF0ZWd5ID09PSBcImN1cnNvclwiICYmIG1ldGEuaGFzTmV4dCA9PT0gZmFsc2UpIHtcbiAgICAgICAgICAgIGhhc1JlYWNoZWRFbmQgPSB0cnVlO1xuICAgICAgICAgICAgLy8gY29uc29sZS5sb2coXG4gICAgICAgICAgICAvLyAgIGBbQ29sbGVjdGlvbl0gUmVhY2hlZCBlbmQgb2YgY3Vyc29yIHBhZ2luYXRpb24gYXQgcGFnZSAke3BhZ2V9YCxcbiAgICAgICAgICAgIC8vICk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gVXBkYXRlIGRpc2NvdmVyZWQgdG90YWwgaWYgcHJvdmlkZWRcbiAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgICAvLyAgIGBbQ29sbGVjdGlvbl0gbWV0YS50b3RhbDogJHttZXRhLnRvdGFsfSwgZGlzY292ZXJlZFRvdGFsIGJlZm9yZTogJHtkaXNjb3ZlcmVkVG90YWx9YCxcbiAgICAgICAgICAvLyApO1xuICAgICAgICAgIGlmIChtZXRhLnRvdGFsICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGRpc2NvdmVyZWRUb3RhbCA9IG1ldGEudG90YWw7XG4gICAgICAgICAgfVxuICAgICAgICAgIC8vIGNvbnNvbGUubG9nKGBbQ29sbGVjdGlvbl0gZGlzY292ZXJlZFRvdGFsIGFmdGVyOiAke2Rpc2NvdmVyZWRUb3RhbH1gKTtcblxuICAgICAgICAgIC8vIFRyYW5zZm9ybSBpdGVtc1xuICAgICAgICAgIGNvbnN0IHRyYW5zZm9ybWVkSXRlbXMgPSB0cmFuc2Zvcm1JdGVtcyhyYXdJdGVtcyk7XG5cbiAgICAgICAgICAvLyBBZGQgaXRlbXMgdG8gYXJyYXkgYW5kIHRyYWNrIGNvdW50IGVmZmljaWVudGx5XG4gICAgICAgICAgbGV0IG5ld0l0ZW1zQWRkZWQgPSAwO1xuICAgICAgICAgIHRyYW5zZm9ybWVkSXRlbXMuZm9yRWFjaCgoaXRlbSwgaW5kZXgpID0+IHtcbiAgICAgICAgICAgIC8vIE9ubHkgY291bnQgYXMgbmV3IGlmIHNsb3Qgd2FzIGVtcHR5XG4gICAgICAgICAgICBpZiAoaXRlbXNbb2Zmc2V0ICsgaW5kZXhdID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgbmV3SXRlbXNBZGRlZCsrO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaXRlbXNbb2Zmc2V0ICsgaW5kZXhdID0gaXRlbTtcbiAgICAgICAgICB9KTtcbiAgICAgICAgICBjYWNoZWRJdGVtQ291bnQgKz0gbmV3SXRlbXNBZGRlZDtcblxuICAgICAgICAgIC8vIEZvciBjdXJzb3Igc3RyYXRlZ3ksIGNhbGN1bGF0ZSBkeW5hbWljIHRvdGFsIGJhc2VkIG9uIGxvYWRlZCBkYXRhXG4gICAgICAgICAgLy8gVXNlIG51bGxpc2ggY29hbGVzY2luZyAoPz8pIGluc3RlYWQgb2YgfHwgdG8gaGFuZGxlIGRpc2NvdmVyZWRUb3RhbCA9IDAgY29ycmVjdGx5XG4gICAgICAgICAgbGV0IG5ld1RvdGFsID0gZGlzY292ZXJlZFRvdGFsID8/IHRvdGFsSXRlbXM7XG5cbiAgICAgICAgICAvLyBDUklUSUNBTCBGSVg6IFdoZW4gQVBJIHJldHVybnMgMCBpdGVtcyBvbiBwYWdlIDEgKG9mZnNldCAwKSwgdGhlIGxpc3QgaXMgZW1wdHkuXG4gICAgICAgICAgLy8gV2UgbXVzdCBzZXQgdG90YWxJdGVtcyB0byAwIHJlZ2FyZGxlc3Mgb2YgbWV0YS50b3RhbCBiZWluZyB1bmRlZmluZWQuXG4gICAgICAgICAgLy8gVGhpcyBoYW5kbGVzIHRoZSBjYXNlIHdoZXJlIGNvdW50PWZhbHNlIGlzIHNlbnQgYnV0IHRoZSBsaXN0IGlzIGFjdHVhbGx5IGVtcHR5LlxuICAgICAgICAgIGlmIChvZmZzZXQgPT09IDAgJiYgdHJhbnNmb3JtZWRJdGVtcy5sZW5ndGggPT09IDApIHtcbiAgICAgICAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgLy8gICBgW0NvbGxlY3Rpb25dIEVtcHR5IHJlc3VsdCBvbiBwYWdlIDEgLSBmb3JjaW5nIHRvdGFsSXRlbXMgdG8gMGAsXG4gICAgICAgICAgICAvLyApO1xuICAgICAgICAgICAgbmV3VG90YWwgPSAwO1xuICAgICAgICAgICAgZGlzY292ZXJlZFRvdGFsID0gMDtcbiAgICAgICAgICB9XG4gICAgICAgICAgLy8gY29uc29sZS5sb2coXG4gICAgICAgICAgLy8gICBgW0NvbGxlY3Rpb25dIG5ld1RvdGFsIGluaXRpYWw6ICR7bmV3VG90YWx9LCB0b3RhbEl0ZW1zOiAke3RvdGFsSXRlbXN9LCBzdHJhdGVneTogJHtzdHJhdGVneX1gLFxuICAgICAgICAgIC8vICk7XG5cbiAgICAgICAgICBpZiAoc3RyYXRlZ3kgPT09IFwiY3Vyc29yXCIpIHtcbiAgICAgICAgICAgIC8vIENhbGN1bGF0ZSB0b3RhbCBiYXNlZCBvbiBsb2FkZWQgaXRlbXMgKyBtYXJnaW5cbiAgICAgICAgICAgIC8vIFVzZSBjYWNoZWRJdGVtQ291bnQgaW5zdGVhZCBvZiBPKG4pIGZpbHRlciBvcGVyYXRpb25cbiAgICAgICAgICAgIGNvbnN0IGxvYWRlZEl0ZW1zQ291bnQgPSBjYWNoZWRJdGVtQ291bnQ7XG4gICAgICAgICAgICBjb25zdCBtYXJnaW5JdGVtcyA9IGhhc1JlYWNoZWRFbmRcbiAgICAgICAgICAgICAgPyAwXG4gICAgICAgICAgICAgIDogcmFuZ2VTaXplICpcbiAgICAgICAgICAgICAgICBWSUVXUE9SVF9DT05TVEFOVFMuUEFHSU5BVElPTi5DVVJTT1JfU0NST0xMX01BUkdJTl9NVUxUSVBMSUVSO1xuICAgICAgICAgICAgY29uc3QgbWluVmlydHVhbEl0ZW1zID1cbiAgICAgICAgICAgICAgcmFuZ2VTaXplICpcbiAgICAgICAgICAgICAgVklFV1BPUlRfQ09OU1RBTlRTLlBBR0lOQVRJT04uQ1VSU09SX01JTl9WSVJUVUFMX1NJWkVfTVVMVElQTElFUjtcblxuICAgICAgICAgICAgLy8gRHluYW1pYyB0b3RhbDogbG9hZGVkIGl0ZW1zICsgbWFyZ2luICh1bmxlc3Mgd2UndmUgcmVhY2hlZCB0aGUgZW5kKVxuICAgICAgICAgICAgbmV3VG90YWwgPSBNYXRoLm1heChcbiAgICAgICAgICAgICAgbG9hZGVkSXRlbXNDb3VudCArIG1hcmdpbkl0ZW1zLFxuICAgICAgICAgICAgICBtaW5WaXJ0dWFsSXRlbXMsXG4gICAgICAgICAgICApO1xuXG4gICAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgIC8vICAgYFtDb2xsZWN0aW9uXSBDdXJzb3IgbW9kZSB2aXJ0dWFsIHNpemU6IGxvYWRlZD0ke2xvYWRlZEl0ZW1zQ291bnR9LCBtYXJnaW49JHttYXJnaW5JdGVtc30sIHRvdGFsPSR7bmV3VG90YWx9LCBoYXNSZWFjaGVkRW5kPSR7aGFzUmVhY2hlZEVuZH1gLFxuICAgICAgICAgICAgLy8gKTtcblxuICAgICAgICAgICAgLy8gVXBkYXRlIHRvdGFsIGlmIGl0IGhhcyBncm93blxuICAgICAgICAgICAgaWYgKG5ld1RvdGFsID4gdG90YWxJdGVtcykge1xuICAgICAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgLy8gICBgW0NvbGxlY3Rpb25dIFVwZGF0aW5nIGN1cnNvciB2aXJ0dWFsIHNpemUgZnJvbSAke3RvdGFsSXRlbXN9IHRvICR7bmV3VG90YWx9YCxcbiAgICAgICAgICAgICAgLy8gKTtcbiAgICAgICAgICAgICAgdG90YWxJdGVtcyA9IG5ld1RvdGFsO1xuICAgICAgICAgICAgICBzZXRUb3RhbEl0ZW1zKG5ld1RvdGFsKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgLy8gRm9yIG90aGVyIHN0cmF0ZWdpZXMsIHVzZSBkaXNjb3ZlcmVkIHRvdGFsIG9yIGN1cnJlbnQgdG90YWxcbiAgICAgICAgICAgIC8vIFVzZSBudWxsaXNoIGNvYWxlc2NpbmcgKD8/KSBpbnN0ZWFkIG9mIHx8IHRvIGhhbmRsZSBkaXNjb3ZlcmVkVG90YWwgPSAwIGNvcnJlY3RseVxuICAgICAgICAgICAgbmV3VG90YWwgPSBkaXNjb3ZlcmVkVG90YWwgPz8gdG90YWxJdGVtcztcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBVcGRhdGUgc3RhdGVcbiAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgICAvLyAgIGBbQ29sbGVjdGlvbl0gQmVmb3JlIHN0YXRlIHVwZGF0ZTogbmV3VG90YWw9JHtuZXdUb3RhbH0sIHRvdGFsSXRlbXM9JHt0b3RhbEl0ZW1zfSwgd2lsbCB1cGRhdGU6ICR7bmV3VG90YWwgIT09IHRvdGFsSXRlbXN9YCxcbiAgICAgICAgICAvLyApO1xuICAgICAgICAgIGlmIChuZXdUb3RhbCAhPT0gdG90YWxJdGVtcykge1xuICAgICAgICAgICAgLy8gY29uc29sZS5sb2coYFtDb2xsZWN0aW9uXSBDYWxsaW5nIHNldFRvdGFsSXRlbXMoJHtuZXdUb3RhbH0pYCk7XG4gICAgICAgICAgICB0b3RhbEl0ZW1zID0gbmV3VG90YWw7XG4gICAgICAgICAgICBzZXRUb3RhbEl0ZW1zKG5ld1RvdGFsKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBVcGRhdGUgY29tcG9uZW50IGl0ZW1zIHJlZmVyZW5jZVxuICAgICAgICAgIGNvbXBvbmVudC5pdGVtcyA9IGl0ZW1zO1xuXG4gICAgICAgICAgLy8gRW1pdCBpdGVtcyBjaGFuZ2VkIGV2ZW50XG4gICAgICAgICAgY29tcG9uZW50LmVtaXQ/LihcInZpZXdwb3J0Oml0ZW1zLWNoYW5nZWRcIiwge1xuICAgICAgICAgICAgdG90YWxJdGVtczogbmV3VG90YWwsXG4gICAgICAgICAgICBsb2FkZWRDb3VudDogY2FjaGVkSXRlbUNvdW50LFxuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgLy8gVXBkYXRlIHZpZXdwb3J0IHN0YXRlXG4gICAgICAgICAgY29uc3Qgdmlld3BvcnRTdGF0ZSA9IChjb21wb25lbnQudmlld3BvcnQgYXMgYW55KS5zdGF0ZTtcbiAgICAgICAgICBpZiAodmlld3BvcnRTdGF0ZSkge1xuICAgICAgICAgICAgLy8gVXNlIG51bGxpc2ggY29hbGVzY2luZyAoPz8pIGluc3RlYWQgb2YgfHwgdG8gaGFuZGxlIG5ld1RvdGFsID0gMCBjb3JyZWN0bHlcbiAgICAgICAgICAgIHZpZXdwb3J0U3RhdGUudG90YWxJdGVtcyA9IG5ld1RvdGFsID8/IGl0ZW1zLmxlbmd0aDtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBWZXJpZnkgaXRlbXMgYXJlIGFjdHVhbGx5IHN0aWxsIGluIHRoZSBhcnJheSBiZWZvcmUgbWFya2luZyBhcyBsb2FkZWRcbiAgICAgICAgICAvLyBUaGlzIHByZXZlbnRzIGEgcmFjZSBjb25kaXRpb24gd2hlcmUgZXZpY3Rpb24gaGFwcGVucyBkdXJpbmcgYXN5bmMgbG9hZDpcbiAgICAgICAgICAvLyAxLiBMb2FkIHN0YXJ0cyBmb3IgcmFuZ2UgQVxuICAgICAgICAgIC8vIDIuIFVzZXIgc2Nyb2xscyB0byByYW5nZSBCXG4gICAgICAgICAgLy8gMy4gRXZpY3Rpb24gcmVtb3ZlcyBpdGVtcyBmcm9tIHJhbmdlIEEgYW5kIHJlbW92ZXMgZnJvbSBsb2FkZWRSYW5nZXNcbiAgICAgICAgICAvLyA0LiBMb2FkIGZvciByYW5nZSBBIGNvbXBsZXRlcywgd291bGQgcmUtYWRkIHRvIGxvYWRlZFJhbmdlc1xuICAgICAgICAgIC8vIDUuIFVzZXIgc2Nyb2xscyBiYWNrIHRvIEEsIGxvYWRlZFJhbmdlcyBzYXlzIGxvYWRlZCBidXQgaXRlbXMgYXJlIGdvbmVcbiAgICAgICAgICBjb25zdCBpdGVtc0FjdHVhbGx5U3RvcmVkID1cbiAgICAgICAgICAgIHRyYW5zZm9ybWVkSXRlbXMubGVuZ3RoID4gMCAmJlxuICAgICAgICAgICAgdHJhbnNmb3JtZWRJdGVtcy5ldmVyeShcbiAgICAgICAgICAgICAgKF8sIGlkeCkgPT4gaXRlbXNbb2Zmc2V0ICsgaWR4XSAhPT0gdW5kZWZpbmVkLFxuICAgICAgICAgICAgKTtcblxuICAgICAgICAgIGlmIChpdGVtc0FjdHVhbGx5U3RvcmVkKSB7XG4gICAgICAgICAgICAvLyBNYXJrIGFzIGxvYWRlZCBvbmx5IGlmIGl0ZW1zIGFyZSBhY3R1YWxseSBwcmVzZW50XG4gICAgICAgICAgICBsb2FkZWRSYW5nZXMuYWRkKHJhbmdlSWQpO1xuICAgICAgICAgIH1cbiAgICAgICAgICAvLyBBbHdheXMgY2xlYW4gdXAgcGVuZGluZy9mYWlsZWQgc3RhdGVcbiAgICAgICAgICBwZW5kaW5nUmFuZ2VzLmRlbGV0ZShyYW5nZUlkKTtcbiAgICAgICAgICBmYWlsZWRSYW5nZXMuZGVsZXRlKHJhbmdlSWQpO1xuXG4gICAgICAgICAgLy8gY29uc29sZS5sb2coXG4gICAgICAgICAgLy8gICBgW0NvbGxlY3Rpb25dIFJhbmdlICR7cmFuZ2VJZH0gbG9hZGVkIHN1Y2Nlc3NmdWxseSB3aXRoICR7dHJhbnNmb3JtZWRJdGVtcy5sZW5ndGh9IGl0ZW1zYFxuICAgICAgICAgIC8vICk7XG5cbiAgICAgICAgICAvLyBMb2cgbWVtb3J5IHN0YXRzIHBlcmlvZGljYWxseSAoZXZlcnkgNSBsb2FkcylcbiAgICAgICAgICBpZiAoY29tcGxldGVkTG9hZHMgJSA1ID09PSAwKSB7XG4gICAgICAgICAgICBsb2dNZW1vcnlTdGF0cyhgbG9hZDoke2NvbXBsZXRlZExvYWRzfWApO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIEVtaXQgZXZlbnRzXG4gICAgICAgICAgY29tcG9uZW50LmVtaXQ/LihcInZpZXdwb3J0OnJhbmdlLWxvYWRlZFwiLCB7XG4gICAgICAgICAgICBvZmZzZXQsXG4gICAgICAgICAgICBsaW1pdCxcbiAgICAgICAgICAgIGl0ZW1zOiB0cmFuc2Zvcm1lZEl0ZW1zLFxuICAgICAgICAgICAgdG90YWw6IG5ld1RvdGFsLFxuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgLy8gRW1pdCBjb2xsZWN0aW9uIGxvYWRlZCBldmVudCB3aXRoIG1vcmUgZGV0YWlsc1xuICAgICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJjb2xsZWN0aW9uOnJhbmdlLWxvYWRlZFwiLCB7XG4gICAgICAgICAgICBvZmZzZXQsXG4gICAgICAgICAgICBsaW1pdCxcbiAgICAgICAgICAgIGl0ZW1zOiB0cmFuc2Zvcm1lZEl0ZW1zLFxuICAgICAgICAgICAgdG90YWw6IG5ld1RvdGFsLFxuICAgICAgICAgICAgcmFuZ2VJZCxcbiAgICAgICAgICAgIGl0ZW1zSW5NZW1vcnk6IGl0ZW1zLmZpbHRlcigoaXRlbSkgPT4gaXRlbSAhPT0gdW5kZWZpbmVkKS5sZW5ndGgsXG4gICAgICAgICAgICBjdXJzb3I6IHN0cmF0ZWd5ID09PSBcImN1cnNvclwiID8gY3VycmVudEN1cnNvciA6IHVuZGVmaW5lZCxcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIC8vIFRyaWdnZXIgdmlld3BvcnQgdXBkYXRlXG4gICAgICAgICAgY29tcG9uZW50LnZpZXdwb3J0Py51cGRhdGVWaWV3cG9ydD8uKCk7XG5cbiAgICAgICAgICByZXR1cm4gdHJhbnNmb3JtZWRJdGVtcztcbiAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICAvLyBIYW5kbGUgQWJvcnRFcnJvciBhbmQgXCJGYWlsZWQgdG8gZmV0Y2hcIiAod2hpY2ggY2FuIG9jY3VyIG9uIGFib3J0KSBncmFjZWZ1bGx5XG4gICAgICAgICAgY29uc3QgaXNBYm9ydEVycm9yID1cbiAgICAgICAgICAgIChlcnJvciBhcyBFcnJvcikubmFtZSA9PT0gXCJBYm9ydEVycm9yXCIgfHxcbiAgICAgICAgICAgICgoZXJyb3IgYXMgRXJyb3IpLm1lc3NhZ2UgPT09IFwiRmFpbGVkIHRvIGZldGNoXCIgJiZcbiAgICAgICAgICAgICAgYWJvcnRDb250cm9sbGVyLnNpZ25hbC5hYm9ydGVkKTtcblxuICAgICAgICAgIGlmIChpc0Fib3J0RXJyb3IpIHtcbiAgICAgICAgICAgIHBlbmRpbmdSYW5nZXMuZGVsZXRlKHJhbmdlSWQpO1xuICAgICAgICAgICAgY2FuY2VsbGVkTG9hZHMrKztcbiAgICAgICAgICAgIC8vIERvbid0IHRocm93LCBqdXN0IHJldHVybiBlbXB0eSBhcnJheVxuICAgICAgICAgICAgcmV0dXJuIFtdO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIEhhbmRsZSBvdGhlciBlcnJvcnNcbiAgICAgICAgICBwZW5kaW5nUmFuZ2VzLmRlbGV0ZShyYW5nZUlkKTtcbiAgICAgICAgICBmYWlsZWRMb2FkcysrO1xuXG4gICAgICAgICAgY29uc3QgYXR0ZW1wdHMgPSAoZmFpbGVkUmFuZ2VzLmdldChyYW5nZUlkKT8uYXR0ZW1wdHMgfHwgMCkgKyAxO1xuICAgICAgICAgIGZhaWxlZFJhbmdlcy5zZXQocmFuZ2VJZCwge1xuICAgICAgICAgICAgYXR0ZW1wdHMsXG4gICAgICAgICAgICBsYXN0RXJyb3I6IGVycm9yIGFzIEVycm9yLFxuICAgICAgICAgICAgdGltZXN0YW1wOiBEYXRlLm5vdygpLFxuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgLy8gRW1pdCBlcnJvciBldmVudFxuICAgICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpyYW5nZS1lcnJvclwiLCB7XG4gICAgICAgICAgICBvZmZzZXQsXG4gICAgICAgICAgICBsaW1pdCxcbiAgICAgICAgICAgIGVycm9yLFxuICAgICAgICAgICAgYXR0ZW1wdHMsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgfSBmaW5hbGx5IHtcbiAgICAgICAgICBhY3RpdmVSZXF1ZXN0cy5kZWxldGUocmFuZ2VJZCk7XG4gICAgICAgICAgYWJvcnRDb250cm9sbGVycy5kZWxldGUocmFuZ2VJZCk7XG4gICAgICAgIH1cbiAgICAgIH0pKCk7XG5cbiAgICAgIGFjdGl2ZVJlcXVlc3RzLnNldChyYW5nZUlkLCByZXF1ZXN0UHJvbWlzZSk7XG4gICAgICByZXR1cm4gcmVxdWVzdFByb21pc2U7XG4gICAgfTtcblxuICAgIC8qKlxuICAgICAqIExvYWQgbWlzc2luZyByYW5nZXMgZnJvbSBjb2xsZWN0aW9uXG4gICAgICovXG4gICAgY29uc3QgbG9hZE1pc3NpbmdSYW5nZXNJbnRlcm5hbCA9IGFzeW5jIChcbiAgICAgIC8vIE5vdGU6IGNhbGxlciBpcyBhZGRlZCBieSBsb2FkTWlzc2luZ1JhbmdlcyB3cmFwcGVyXG4gICAgICByYW5nZToge1xuICAgICAgICBzdGFydDogbnVtYmVyO1xuICAgICAgICBlbmQ6IG51bWJlcjtcbiAgICAgIH0sXG4gICAgICBjYWxsZXI/OiBzdHJpbmcsXG4gICAgKTogUHJvbWlzZTx2b2lkPiA9PiB7XG4gICAgICBpZiAoIWNvbGxlY3Rpb24pIHJldHVybjtcblxuICAgICAgLy8gRm9yIGN1cnNvciBwYWdpbmF0aW9uLCB3ZSBuZWVkIHRvIGxvYWQgc2VxdWVudGlhbGx5XG4gICAgICBpZiAoc3RyYXRlZ3kgPT09IFwiY3Vyc29yXCIpIHtcbiAgICAgICAgY29uc3Qgc3RhcnRQYWdlID0gTWF0aC5mbG9vcihyYW5nZS5zdGFydCAvIHJhbmdlU2l6ZSkgKyAxO1xuICAgICAgICBjb25zdCBlbmRQYWdlID0gTWF0aC5mbG9vcihyYW5nZS5lbmQgLyByYW5nZVNpemUpICsgMTtcblxuICAgICAgICAvLyBMaW1pdCBob3cgbWFueSBwYWdlcyB3ZSdsbCBsb2FkIGF0IG9uY2VcbiAgICAgICAgY29uc3QgbWF4UGFnZXNUb0xvYWQgPSAxMDtcbiAgICAgICAgY29uc3QgY3VycmVudEhpZ2hlc3RQYWdlID0gaGlnaGVzdExvYWRlZFBhZ2UgfHwgMDtcbiAgICAgICAgY29uc3QgbGltaXRlZEVuZFBhZ2UgPSBNYXRoLm1pbihcbiAgICAgICAgICBlbmRQYWdlLFxuICAgICAgICAgIGN1cnJlbnRIaWdoZXN0UGFnZSArIG1heFBhZ2VzVG9Mb2FkLFxuICAgICAgICApO1xuXG4gICAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgICAvLyAgIGBbQ29sbGVjdGlvbl0gQ3Vyc29yIG1vZGU6IG5lZWQgdG8gbG9hZCBwYWdlcyAke3N0YXJ0UGFnZX0gdG8gJHtlbmRQYWdlfSwgbGltaXRlZCB0byAke2xpbWl0ZWRFbmRQYWdlfWAsXG4gICAgICAgIC8vICk7XG5cbiAgICAgICAgLy8gQ2hlY2sgaWYgd2UgbmVlZCB0byBsb2FkIHBhZ2VzIHNlcXVlbnRpYWxseVxuICAgICAgICBmb3IgKGxldCBwYWdlID0gc3RhcnRQYWdlOyBwYWdlIDw9IGxpbWl0ZWRFbmRQYWdlOyBwYWdlKyspIHtcbiAgICAgICAgICBjb25zdCByYW5nZUlkID0gcGFnZSAtIDE7IC8vIENvbnZlcnQgdG8gMC1iYXNlZCByYW5nZUlkXG4gICAgICAgICAgY29uc3Qgb2Zmc2V0ID0gcmFuZ2VJZCAqIHJhbmdlU2l6ZTtcblxuICAgICAgICAgIC8vIFNraXAgaWYgYWxyZWFkeSBiZWluZyBsb2FkZWRcbiAgICAgICAgICBpZiAocGVuZGluZ1Jhbmdlcy5oYXMocmFuZ2VJZCkpIHtcbiAgICAgICAgICAgIC8vIFJhbmdlIGFscmVhZHkgYmVpbmcgbG9hZGVkXG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgaWYgKCFsb2FkZWRSYW5nZXMuaGFzKHJhbmdlSWQpICYmICFwZW5kaW5nUmFuZ2VzLmhhcyhyYW5nZUlkKSkge1xuICAgICAgICAgICAgLy8gRm9yIGN1cnNvciBwYWdpbmF0aW9uLCB3ZSBtdXN0IGxvYWQgc2VxdWVudGlhbGx5XG4gICAgICAgICAgICAvLyBDaGVjayBpZiB3ZSBoYXZlIGFsbCBwcmV2aW91cyBwYWdlcyBsb2FkZWRcbiAgICAgICAgICAgIGlmIChwYWdlID4gMSkge1xuICAgICAgICAgICAgICBsZXQgY2FuTG9hZCA9IHRydWU7XG4gICAgICAgICAgICAgIGZvciAobGV0IHByZXZQYWdlID0gMTsgcHJldlBhZ2UgPCBwYWdlOyBwcmV2UGFnZSsrKSB7XG4gICAgICAgICAgICAgICAgY29uc3QgcHJldlJhbmdlSWQgPSBwcmV2UGFnZSAtIDE7XG4gICAgICAgICAgICAgICAgaWYgKCFsb2FkZWRSYW5nZXMuaGFzKHByZXZSYW5nZUlkKSkge1xuICAgICAgICAgICAgICAgICAgLy8gY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICAvLyAgIGBbQ29sbGVjdGlvbl0gQ2Fubm90IGxvYWQgcGFnZSAke3BhZ2V9IC0gbmVlZCB0byBsb2FkIHBhZ2UgJHtwcmV2UGFnZX0gZmlyc3RgLFxuICAgICAgICAgICAgICAgICAgLy8gKTtcbiAgICAgICAgICAgICAgICAgIGNhbkxvYWQgPSBmYWxzZTtcblxuICAgICAgICAgICAgICAgICAgLy8gVHJ5IHRvIGxvYWQgdGhlIG1pc3NpbmcgcGFnZVxuICAgICAgICAgICAgICAgICAgaWYgKCFwZW5kaW5nUmFuZ2VzLmhhcyhwcmV2UmFuZ2VJZCkpIHtcbiAgICAgICAgICAgICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgICAgICAgICAgICBhd2FpdCBsb2FkUmFuZ2UocHJldlJhbmdlSWQgKiByYW5nZVNpemUsIHJhbmdlU2l6ZSk7XG4gICAgICAgICAgICAgICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgICAgICAgICAgICAgICAgY29uc29sZS5lcnJvcihcbiAgICAgICAgICAgICAgICAgICAgICAgIGBbQ29sbGVjdGlvbl0gRmFpbGVkIHRvIGxvYWQgcHJlcmVxdWlzaXRlIHBhZ2UgJHtwcmV2UGFnZX06YCxcbiAgICAgICAgICAgICAgICAgICAgICAgIGVycm9yLFxuICAgICAgICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICAgICAgICAgICAgcmV0dXJuOyAvLyBTdG9wIHRyeWluZyB0byBsb2FkIGZ1cnRoZXIgcGFnZXNcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgaWYgKCFjYW5Mb2FkKSB7XG4gICAgICAgICAgICAgICAgY29udGludWU7IC8vIFNraXAgdGhpcyBwYWdlIGZvciBub3dcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgICBhd2FpdCBsb2FkUmFuZ2Uob2Zmc2V0LCByYW5nZVNpemUpO1xuICAgICAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICAgICAgY29uc29sZS5lcnJvcihgW0NvbGxlY3Rpb25dIEZhaWxlZCB0byBsb2FkIHBhZ2UgJHtwYWdlfTpgLCBlcnJvcik7XG4gICAgICAgICAgICAgIGJyZWFrOyAvLyBTdG9wIHNlcXVlbnRpYWwgbG9hZGluZyBvbiBlcnJvclxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChlbmRQYWdlID4gbGltaXRlZEVuZFBhZ2UpIHtcbiAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgICAvLyAgIGBbQ29sbGVjdGlvbl0gU3RvcHBlZCBhdCBwYWdlICR7bGltaXRlZEVuZFBhZ2V9IHRvIHByZXZlbnQgZXhjZXNzaXZlIGxvYWRpbmcgKHJlcXVlc3RlZCB1cCB0byAke2VuZFBhZ2V9KWAsXG4gICAgICAgICAgLy8gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cblxuICAgICAgLy8gT3JpZ2luYWwgbG9naWMgZm9yIG9mZnNldC9wYWdlIHN0cmF0ZWdpZXNcbiAgICAgIC8vIENhbGN1bGF0ZSByYW5nZSBib3VuZGFyaWVzXG4gICAgICBjb25zdCBzdGFydFJhbmdlID0gTWF0aC5mbG9vcihyYW5nZS5zdGFydCAvIHJhbmdlU2l6ZSk7XG4gICAgICBjb25zdCBlbmRSYW5nZSA9IE1hdGguZmxvb3IocmFuZ2UuZW5kIC8gcmFuZ2VTaXplKTtcblxuICAgICAgLy8gQ29sbGVjdCByYW5nZXMgdGhhdCBuZWVkIGxvYWRpbmdcbiAgICAgIC8vIFdlIG5lZWQgdG8gY2hlY2sgaWYgdGhlIFZJU0lCTEUgaW5kaWNlcyBoYXZlIGRhdGEsIG5vdCBqdXN0IGFueSBkYXRhIGluIGEgcmFuZ2VcbiAgICAgIGNvbnN0IHJhbmdlc1RvTG9hZDogbnVtYmVyW10gPSBbXTtcblxuICAgICAgLy8gQ2hlY2sgZWFjaCB2aXNpYmxlIGluZGV4IHRvIHNlZSBpZiBpdCBoYXMgZGF0YVxuICAgICAgY29uc3QgdmlzaWJsZVN0YXJ0ID0gcmFuZ2Uuc3RhcnQ7XG4gICAgICBjb25zdCB2aXNpYmxlRW5kID0gcmFuZ2UuZW5kO1xuXG4gICAgICAvLyBGaW5kIHdoaWNoIGluZGljZXMgaW4gdGhlIHZpc2libGUgcmFuZ2UgYXJlIG1pc3NpbmcgZGF0YVxuICAgICAgLy8gSWYgdG90YWxJdGVtcyBpcyAwIG9yIHdlIGhhdmUgbm8gaXRlbXMsIHdlIG5lZWQgdG8gbG9hZCAoaW5pdGlhbCBsb2FkIGNhc2UpXG4gICAgICBsZXQgaGFzTWlzc2luZ0RhdGEgPSBmYWxzZTtcblxuICAgICAgaWYgKHRvdGFsSXRlbXMgPT09IDAgfHwgaXRlbXMubGVuZ3RoID09PSAwKSB7XG4gICAgICAgIC8vIEluaXRpYWwgbG9hZCAtIHdlIG5lZWQgdG8gZmV0Y2ggZGF0YVxuICAgICAgICBoYXNNaXNzaW5nRGF0YSA9IHRydWU7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICAvLyBDaGVjayBlYWNoIHZpc2libGUgaW5kZXggZm9yIG1pc3NpbmcgZGF0YVxuICAgICAgICBmb3IgKGxldCBpID0gdmlzaWJsZVN0YXJ0OyBpIDw9IHZpc2libGVFbmQgJiYgaSA8IHRvdGFsSXRlbXM7IGkrKykge1xuICAgICAgICAgIGlmIChpdGVtc1tpXSA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICBoYXNNaXNzaW5nRGF0YSA9IHRydWU7XG4gICAgICAgICAgICBicmVhaztcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgaWYgKGhhc01pc3NpbmdEYXRhKSB7XG4gICAgICAgIC8vIEZpbmQgd2hpY2ggcmFuZ2VzIG5lZWQgdG8gYmUgbG9hZGVkXG4gICAgICAgIGZvciAobGV0IHJhbmdlSWQgPSBzdGFydFJhbmdlOyByYW5nZUlkIDw9IGVuZFJhbmdlOyByYW5nZUlkKyspIHtcbiAgICAgICAgICBpZiAocGVuZGluZ1Jhbmdlcy5oYXMocmFuZ2VJZCkpIHtcbiAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIElmIHRvdGFsSXRlbXMgaXMgMCwgd2UncmUgZG9pbmcgaW5pdGlhbCBsb2FkIC0gYWx3YXlzIGxvYWQgdGhlIHJhbmdlXG4gICAgICAgICAgaWYgKHRvdGFsSXRlbXMgPT09IDApIHtcbiAgICAgICAgICAgIGxvYWRlZFJhbmdlcy5kZWxldGUocmFuZ2VJZCk7XG4gICAgICAgICAgICByYW5nZXNUb0xvYWQucHVzaChyYW5nZUlkKTtcbiAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIENoZWNrIGlmIHRoaXMgc3BlY2lmaWMgcmFuZ2UgaGFzIGFsbCBpdHMgZGF0YVxuICAgICAgICAgIGNvbnN0IHJhbmdlU3RhcnQgPSByYW5nZUlkICogcmFuZ2VTaXplO1xuICAgICAgICAgIGNvbnN0IHJhbmdlRW5kID0gTWF0aC5taW4ocmFuZ2VTdGFydCArIHJhbmdlU2l6ZSwgdG90YWxJdGVtcyk7XG4gICAgICAgICAgbGV0IHJhbmdlTWlzc2luZ0RhdGEgPSBmYWxzZTtcblxuICAgICAgICAgIGZvciAobGV0IGkgPSByYW5nZVN0YXJ0OyBpIDwgcmFuZ2VFbmQ7IGkrKykge1xuICAgICAgICAgICAgLy8gT25seSBjaGVjayBpbmRpY2VzIHRoYXQgYXJlIGluIHRoZSB2aXNpYmxlIHJhbmdlXG4gICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgIGkgPj0gdmlzaWJsZVN0YXJ0ICYmXG4gICAgICAgICAgICAgIGkgPD0gdmlzaWJsZUVuZCAmJlxuICAgICAgICAgICAgICBpdGVtc1tpXSA9PT0gdW5kZWZpbmVkXG4gICAgICAgICAgICApIHtcbiAgICAgICAgICAgICAgcmFuZ2VNaXNzaW5nRGF0YSA9IHRydWU7XG4gICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIGlmIChyYW5nZU1pc3NpbmdEYXRhKSB7XG4gICAgICAgICAgICBsb2FkZWRSYW5nZXMuZGVsZXRlKHJhbmdlSWQpOyAvLyBDbGVhciBzdGFsZSBsb2FkZWQgc3RhdGVcbiAgICAgICAgICAgIHJhbmdlc1RvTG9hZC5wdXNoKHJhbmdlSWQpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZiAocmFuZ2VzVG9Mb2FkLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICAvLyBDaGVjayBpZiB0aGVyZSBhcmUgcXVldWVkIHJlcXVlc3RzIHdlIHNob3VsZCBwcm9jZXNzXG4gICAgICAgIGlmIChcbiAgICAgICAgICBsb2FkUmVxdWVzdFF1ZXVlLmxlbmd0aCA+IDAgJiZcbiAgICAgICAgICBhY3RpdmVMb2FkQ291bnQgPCBtYXhDb25jdXJyZW50UmVxdWVzdHNcbiAgICAgICAgKSB7XG4gICAgICAgICAgcHJvY2Vzc1F1ZXVlKCk7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuXG4gICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgIC8vICAgYFtDb2xsZWN0aW9uXSBMb2FkaW5nICR7cmFuZ2VzVG9Mb2FkLmxlbmd0aH0gcmFuZ2VzOiBbJHtyYW5nZXNUb0xvYWQuam9pbihcIiwgXCIpfV1gLFxuICAgICAgLy8gKTtcblxuICAgICAgLy8gTG9hZCByYW5nZXMgaW5kaXZpZHVhbGx5IC0gbm8gbWVyZ2luZyB0byBhdm9pZCBsb2FkaW5nIG9sZCByYW5nZXNcbiAgICAgIGNvbnN0IHByb21pc2VzID0gcmFuZ2VzVG9Mb2FkLm1hcCgocmFuZ2VJZCkgPT5cbiAgICAgICAgbG9hZFJhbmdlKHJhbmdlSWQgKiByYW5nZVNpemUsIHJhbmdlU2l6ZSksXG4gICAgICApO1xuICAgICAgYXdhaXQgUHJvbWlzZS5hbGxTZXR0bGVkKHByb21pc2VzKTtcbiAgICB9O1xuXG4gICAgLyoqXG4gICAgICogVmVsb2NpdHktYXdhcmUgd3JhcHBlciBmb3IgbG9hZE1pc3NpbmdSYW5nZXNcbiAgICAgKi9cbiAgICBjb25zdCBsb2FkTWlzc2luZ1JhbmdlcyA9IChcbiAgICAgIHJhbmdlOiB7XG4gICAgICAgIHN0YXJ0OiBudW1iZXI7XG4gICAgICAgIGVuZDogbnVtYmVyO1xuICAgICAgfSxcbiAgICAgIGNhbGxlcj86IHN0cmluZyxcbiAgICApOiBQcm9taXNlPHZvaWQ+ID0+IHtcbiAgICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICAgIGNvbnN0IHJhbmdlS2V5ID0gZ2V0UmFuZ2VLZXkocmFuZ2UpO1xuXG4gICAgICAgIC8vIENoZWNrIGlmIGFscmVhZHkgbG9hZGluZ1xuICAgICAgICBpZiAoYWN0aXZlTG9hZFJhbmdlcy5oYXMocmFuZ2VLZXkpKSB7XG4gICAgICAgICAgcmVzb2x2ZSgpO1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFNraXAgaWYgZHJhZ2dpbmcgd2l0aCBsb3cgdmVsb2NpdHkgKGJ1dCBub3QgaWYgd2UncmUgaWRsZSlcbiAgICAgICAgaWYgKGlzRHJhZ2dpbmcgJiYgY3VycmVudFZlbG9jaXR5IDwgMC41ICYmIGN1cnJlbnRWZWxvY2l0eSA+IDApIHtcbiAgICAgICAgICBjYW5jZWxsZWRMb2FkcysrO1xuICAgICAgICAgIHJlc29sdmUoKTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICAvLyBDaGVjayB2ZWxvY2l0eSAtIGlmIHRvbyBoaWdoLCBjYW5jZWwgdGhlIHJlcXVlc3QgZW50aXJlbHlcbiAgICAgICAgaWYgKCFjYW5Mb2FkKCkpIHtcbiAgICAgICAgICBjYW5jZWxsZWRMb2FkcysrO1xuICAgICAgICAgIHJlc29sdmUoKTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICAvLyBDaGVjayBjYXBhY2l0eVxuICAgICAgICBpZiAoYWN0aXZlTG9hZENvdW50IDwgbWF4Q29uY3VycmVudFJlcXVlc3RzKSB7XG4gICAgICAgICAgLy8gRXhlY3V0ZSBpbW1lZGlhdGVseVxuICAgICAgICAgIGV4ZWN1dGVRdWV1ZWRMb2FkKHtcbiAgICAgICAgICAgIHJhbmdlLFxuICAgICAgICAgICAgcHJpb3JpdHk6IFwibm9ybWFsXCIsXG4gICAgICAgICAgICB0aW1lc3RhbXA6IERhdGUubm93KCksXG4gICAgICAgICAgICByZXNvbHZlLFxuICAgICAgICAgICAgcmVqZWN0LFxuICAgICAgICAgICAgY2FsbGVyLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9IGVsc2UgaWYgKFxuICAgICAgICAgIGVuYWJsZVJlcXVlc3RRdWV1ZSAmJlxuICAgICAgICAgIGxvYWRSZXF1ZXN0UXVldWUubGVuZ3RoIDwgbWF4UXVldWVTaXplXG4gICAgICAgICkge1xuICAgICAgICAgIC8vIFF1ZXVlIHRoZSByZXF1ZXN0XG4gICAgICAgICAgbG9hZFJlcXVlc3RRdWV1ZS5wdXNoKHtcbiAgICAgICAgICAgIHJhbmdlLFxuICAgICAgICAgICAgcHJpb3JpdHk6IFwibm9ybWFsXCIsXG4gICAgICAgICAgICB0aW1lc3RhbXA6IERhdGUubm93KCksXG4gICAgICAgICAgICByZXNvbHZlLFxuICAgICAgICAgICAgcmVqZWN0LFxuICAgICAgICAgICAgY2FsbGVyLFxuICAgICAgICAgIH0pO1xuICAgICAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgICAgIC8vICAgYFtMb2FkaW5nTWFuYWdlcl0gUXVldWVkIHJlcXVlc3QgKGF0IGNhcGFjaXR5KSwgcXVldWUgc2l6ZTogJHtsb2FkUmVxdWVzdFF1ZXVlLmxlbmd0aH1gXG4gICAgICAgICAgLy8gKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyBRdWV1ZSBpcyBmdWxsIC0gYnV0IGRvbid0IGRyb3AgaWRsZS9yYW5nZS1jaGFuZ2VkIHJlcXVlc3RzLCB0aGV5J3JlIGNyaXRpY2FsIVxuICAgICAgICAgIC8vIEluc3RlYWQsIGNsZWFyIG9sZCBxdWV1ZWQgcmVxdWVzdHMgYW5kIGFkZCB0aGUgbmV3IG9uZVxuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgIGNhbGxlciA9PT0gXCJ2aWV3cG9ydDppZGxlXCIgfHxcbiAgICAgICAgICAgIGNhbGxlciA9PT0gXCJ2aWV3cG9ydDpyYW5nZS1jaGFuZ2VkXCJcbiAgICAgICAgICApIHtcbiAgICAgICAgICAgIC8vIENsZWFyIG9sZCBxdWV1ZWQgcmVxdWVzdHMgLSB0aGV5J3JlIGZvciByYW5nZXMgdXNlciBzY3JvbGxlZCBwYXN0XG4gICAgICAgICAgICBsb2FkUmVxdWVzdFF1ZXVlLmZvckVhY2goKHIpID0+IHtcbiAgICAgICAgICAgICAgY2FuY2VsbGVkTG9hZHMrKztcbiAgICAgICAgICAgICAgci5yZXNvbHZlKCk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGxvYWRSZXF1ZXN0UXVldWUubGVuZ3RoID0gMDtcbiAgICAgICAgICAgIC8vIFF1ZXVlIHRoaXMgcHJpb3JpdHkgcmVxdWVzdFxuICAgICAgICAgICAgbG9hZFJlcXVlc3RRdWV1ZS5wdXNoKHtcbiAgICAgICAgICAgICAgcmFuZ2UsXG4gICAgICAgICAgICAgIHByaW9yaXR5OiBcImhpZ2hcIixcbiAgICAgICAgICAgICAgdGltZXN0YW1wOiBEYXRlLm5vdygpLFxuICAgICAgICAgICAgICByZXNvbHZlLFxuICAgICAgICAgICAgICByZWplY3QsXG4gICAgICAgICAgICAgIGNhbGxlcixcbiAgICAgICAgICAgIH0pO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyBRdWV1ZSBvdmVyZmxvdyAtIHJlc29sdmUgdG8gYXZvaWQgZXJyb3JzXG4gICAgICAgICAgICBpZiAobG9hZFJlcXVlc3RRdWV1ZS5sZW5ndGggPj0gbWF4UXVldWVTaXplKSB7XG4gICAgICAgICAgICAgIGNvbnN0IHJlbW92ZWQgPSBsb2FkUmVxdWVzdFF1ZXVlLnNwbGljZShcbiAgICAgICAgICAgICAgICAwLFxuICAgICAgICAgICAgICAgIGxvYWRSZXF1ZXN0UXVldWUubGVuZ3RoIC0gbWF4UXVldWVTaXplLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICByZW1vdmVkLmZvckVhY2goKHIpID0+IHtcbiAgICAgICAgICAgICAgICBjYW5jZWxsZWRMb2FkcysrO1xuICAgICAgICAgICAgICAgIHIucmVzb2x2ZSgpO1xuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJlc29sdmUoKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH07XG5cbiAgICAvKipcbiAgICAgKiBSZXRyeSBhIGZhaWxlZCByYW5nZVxuICAgICAqL1xuICAgIGNvbnN0IHJldHJ5RmFpbGVkUmFuZ2UgPSBhc3luYyAocmFuZ2VJZDogbnVtYmVyKTogUHJvbWlzZTxhbnlbXT4gPT4ge1xuICAgICAgZmFpbGVkUmFuZ2VzLmRlbGV0ZShyYW5nZUlkKTtcbiAgICAgIGNvbnN0IG9mZnNldCA9IHJhbmdlSWQgKiByYW5nZVNpemU7XG4gICAgICByZXR1cm4gbG9hZFJhbmdlKG9mZnNldCwgcmFuZ2VTaXplKTtcbiAgICB9O1xuXG4gICAgLyoqXG4gICAgICogU2V0IHRvdGFsIGl0ZW1zIGNvdW50XG4gICAgICovXG4gICAgY29uc3Qgc2V0VG90YWxJdGVtcyA9ICh0b3RhbDogbnVtYmVyKTogdm9pZCA9PiB7XG4gICAgICB0b3RhbEl0ZW1zID0gdG90YWw7XG5cbiAgICAgIC8vIFVwZGF0ZSB2aWV3cG9ydCBzdGF0ZSdzIHRvdGFsIGl0ZW1zXG4gICAgICBjb25zdCB2aWV3cG9ydFN0YXRlID0gKGNvbXBvbmVudC52aWV3cG9ydCBhcyBhbnkpLnN0YXRlO1xuICAgICAgaWYgKHZpZXdwb3J0U3RhdGUpIHtcbiAgICAgICAgdmlld3BvcnRTdGF0ZS50b3RhbEl0ZW1zID0gdG90YWw7XG4gICAgICB9XG5cbiAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDp0b3RhbC1pdGVtcy1jaGFuZ2VkXCIsIHsgdG90YWwgfSk7XG4gICAgfTtcblxuICAgIC8vIEhvb2sgaW50byB2aWV3cG9ydCBpbml0aWFsaXphdGlvblxuICAgIGNvbnN0IG9yaWdpbmFsSW5pdGlhbGl6ZSA9IGNvbXBvbmVudC52aWV3cG9ydC5pbml0aWFsaXplO1xuICAgIGNvbXBvbmVudC52aWV3cG9ydC5pbml0aWFsaXplID0gKCkgPT4ge1xuICAgICAgY29uc3QgcmVzdWx0ID0gb3JpZ2luYWxJbml0aWFsaXplKCk7XG4gICAgICAvLyBTa2lwIGlmIGFscmVhZHkgaW5pdGlhbGl6ZWRcbiAgICAgIGlmIChyZXN1bHQgPT09IGZhbHNlKSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgIH1cblxuICAgICAgLy8gU2V0IGluaXRpYWwgdG90YWwgaWYgcHJvdmlkZWRcbiAgICAgIGlmIChjb21wb25lbnQudG90YWxJdGVtcykge1xuICAgICAgICB0b3RhbEl0ZW1zID0gY29tcG9uZW50LnRvdGFsSXRlbXM7XG4gICAgICB9XG5cbiAgICAgIC8vIExpc3RlbiBmb3IgZHJhZyBldmVudHNcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwidmlld3BvcnQ6ZHJhZy1zdGFydFwiLCAoKSA9PiB7XG4gICAgICAgIGlzRHJhZ2dpbmcgPSB0cnVlO1xuICAgICAgfSk7XG5cbiAgICAgIGNvbXBvbmVudC5vbj8uKFwidmlld3BvcnQ6ZHJhZy1lbmRcIiwgKCkgPT4ge1xuICAgICAgICBpc0RyYWdnaW5nID0gZmFsc2U7XG4gICAgICAgIC8vIFByb2Nlc3MgYW55IHF1ZXVlZCByZXF1ZXN0cyBhZnRlciBkcmFnIGVuZHMgKHNhZmV0eSBtZWFzdXJlKVxuICAgICAgICAvLyBUaGlzIGVuc3VyZXMgcGxhY2Vob2xkZXJzIGFyZSByZXBsYWNlZCBldmVuIGlmIGlkbGUgZGV0ZWN0aW9uIGZhaWxzXG4gICAgICAgIGlmIChsb2FkT25EcmFnRW5kKSB7XG4gICAgICAgICAgcHJvY2Vzc1F1ZXVlKCk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuXG4gICAgICAvLyBMaXN0ZW4gZm9yIHJhbmdlIGNoYW5nZXNcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwidmlld3BvcnQ6cmFuZ2UtY2hhbmdlZFwiLCBhc3luYyAoZGF0YTogYW55KSA9PiB7XG4gICAgICAgIC8vIEV4dHJhY3QgcmFuZ2UgZnJvbSBldmVudCBkYXRhIC0gdmlydHVhbCBmZWF0dXJlIGVtaXRzIHsgcmFuZ2U6IHsgc3RhcnQsIGVuZCB9LCBzY3JvbGxQb3NpdGlvbiB9XG4gICAgICAgIGNvbnN0IHJhbmdlID0gZGF0YS5yYW5nZSB8fCBkYXRhO1xuXG4gICAgICAgIC8vIERvbid0IGxvYWQgZHVyaW5nIGZhc3Qgc2Nyb2xsaW5nIC0gbG9hZE1pc3NpbmdSYW5nZXMgd2lsbCBoYW5kbGUgdmVsb2NpdHkgY2hlY2tcblxuICAgICAgICAvLyBTa2lwIGlmIGF1dG9Mb2FkIGlzIGRpc2FibGVkIEFORCB3ZSBoYXZlbid0IG1hbnVhbGx5IGxvYWRlZCB5ZXRcbiAgICAgICAgLy8gVGhpcyBwcmV2ZW50cyB1bndhbnRlZCBwYWdlIDEgbG9hZHMgd2hlbiB1c2luZyByZWxvYWRBdCgpXG4gICAgICAgIC8vIEJ1dCBhbGxvd3Mgc3Vic2VxdWVudCBsb2FkcyBmcm9tIHNjcm9sbGluZyBhZnRlciByZWxvYWRBdCgpIGNvbXBsZXRlc1xuICAgICAgICBpZiAoIWF1dG9Mb2FkICYmICFoYXNNYW51YWxseUxvYWRlZCkge1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IHsgc3RhcnQsIGVuZCB9ID0gcmFuZ2U7XG5cbiAgICAgICAgLy8gVmFsaWRhdGUgcmFuZ2UgYmVmb3JlIGxvYWRpbmdcbiAgICAgICAgaWYgKHR5cGVvZiBzdGFydCAhPT0gXCJudW1iZXJcIiB8fCB0eXBlb2YgZW5kICE9PSBcIm51bWJlclwiKSB7XG4gICAgICAgICAgY29uc29sZS53YXJuKFxuICAgICAgICAgICAgXCJbQ29sbGVjdGlvbl0gSW52YWxpZCByYW5nZSBpbiB2aWV3cG9ydDpyYW5nZS1jaGFuZ2VkIGV2ZW50OlwiLFxuICAgICAgICAgICAgZGF0YSxcbiAgICAgICAgICApO1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFNraXAgbG9hZGluZyBwYWdlIDEgaWYgd2UgaGF2ZSBhbiBpbml0aWFsU2Nyb2xsSW5kZXggYW5kIGhhdmVuJ3QgY29tcGxldGVkIGluaXRpYWwgbG9hZCB5ZXRcbiAgICAgICAgLy8gVGhpcyBwcmV2ZW50cyB0aGUgcmVxdWVzdEFuaW1hdGlvbkZyYW1lIGluIHZpcnR1YWwudHMgZnJvbSB0cmlnZ2VyaW5nIGEgcGFnZSAxIGxvYWRcbiAgICAgICAgLy8gYWZ0ZXIgd2UndmUgYWxyZWFkeSBzdGFydGVkIGxvYWRpbmcgdGhlIGNvcnJlY3QgaW5pdGlhbCBwYWdlXG4gICAgICAgIGlmIChoYXNJbml0aWFsU2Nyb2xsSW5kZXggJiYgIWhhc0NvbXBsZXRlZEluaXRpYWxQb3NpdGlvbkxvYWQpIHtcbiAgICAgICAgICBjb25zdCBwYWdlMUVuZEluZGV4ID0gcmFuZ2VTaXplIC0gMTsgLy8gZS5nLiwgMjkgZm9yIHJhbmdlU2l6ZT0zMFxuICAgICAgICAgIGlmIChzdGFydCA9PT0gMCAmJiBlbmQgPD0gcGFnZTFFbmRJbmRleCkge1xuICAgICAgICAgICAgLy8gVGhpcyBpcyBhIHJlcXVlc3QgZm9yIHBhZ2UgMSwgc2tpcCBpdFxuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFNraXAgbG9hZGluZyBwYWdlIDEgaWYgYSByZWxvYWQgaXMgaW4gcHJvZ3Jlc3MgKGUuZy4sIHJlbG9hZEF0IGlzIGxvYWRpbmcgYSBzcGVjaWZpYyBwYWdlKVxuICAgICAgICAvLyBUaGlzIHByZXZlbnRzIHN0YWxlIHZpZXdwb3J0OnJhbmdlLWNoYW5nZWQgZXZlbnRzIGZyb20gbG9hZGluZyBwYWdlIDFcbiAgICAgICAgLy8gd2hpbGUgcmVsb2FkQXQgaXMgbG9hZGluZyBhIGRpZmZlcmVudCBwYWdlXG4gICAgICAgIGlmIChyZWxvYWRJblByb2dyZXNzICYmIHJlbG9hZFRhcmdldFJhbmdlKSB7XG4gICAgICAgICAgY29uc3QgcGFnZTFFbmRJbmRleCA9IHJhbmdlU2l6ZSAtIDE7XG4gICAgICAgICAgaWYgKHN0YXJ0ID09PSAwICYmIGVuZCA8PSBwYWdlMUVuZEluZGV4KSB7XG4gICAgICAgICAgICAvLyBUaGlzIGlzIGEgcmVxdWVzdCBmb3IgcGFnZSAxIGR1cmluZyBhIHJlbG9hZCB0YXJnZXRpbmcgYSBkaWZmZXJlbnQgcmFuZ2VcbiAgICAgICAgICAgIC8vIENoZWNrIGlmIHRoZSB0YXJnZXQgcmFuZ2UgaXMgTk9UIHBhZ2UgMVxuICAgICAgICAgICAgaWYgKHJlbG9hZFRhcmdldFJhbmdlLnN0YXJ0ID4gcGFnZTFFbmRJbmRleCkge1xuICAgICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gTG9hZCBtaXNzaW5nIHJhbmdlcyBpZiBuZWVkZWQgLSBkb24ndCBhd2FpdCB0byBhdm9pZCBibG9ja2luZyBzY3JvbGxcbiAgICAgICAgLy8gVGhlIGxvYWRpbmcgaGFwcGVucyBpbiB0aGUgYmFja2dyb3VuZCBhbmQgVUkgdXBkYXRlcyB3aGVuIGRhdGEgYXJyaXZlc1xuICAgICAgICBsb2FkTWlzc2luZ1Jhbmdlcyh7IHN0YXJ0LCBlbmQgfSwgXCJ2aWV3cG9ydDpyYW5nZS1jaGFuZ2VkXCIpO1xuXG4gICAgICAgIC8vIEV2aWN0IGRpc3RhbnQgaXRlbXMgdG8gcHJldmVudCBtZW1vcnkgYmxvYXRcbiAgICAgICAgZXZpY3REaXN0YW50SXRlbXMoc3RhcnQsIGVuZCk7XG5cbiAgICAgICAgLy8gRm9yIGN1cnNvciBtb2RlLCBjaGVjayBpZiB3ZSBuZWVkIHRvIHVwZGF0ZSB2aXJ0dWFsIHNpemVcbiAgICAgICAgaWYgKHN0cmF0ZWd5ID09PSBcImN1cnNvclwiICYmICFoYXNSZWFjaGVkRW5kKSB7XG4gICAgICAgICAgLy8gVXNlIGNhY2hlZEl0ZW1Db3VudCBpbnN0ZWFkIG9mIE8obikgZmlsdGVyIG9wZXJhdGlvblxuICAgICAgICAgIGNvbnN0IGxvYWRlZEl0ZW1zQ291bnQgPSBjYWNoZWRJdGVtQ291bnQ7XG4gICAgICAgICAgY29uc3QgbWFyZ2luSXRlbXMgPVxuICAgICAgICAgICAgcmFuZ2VTaXplICpcbiAgICAgICAgICAgIFZJRVdQT1JUX0NPTlNUQU5UUy5QQUdJTkFUSU9OLkNVUlNPUl9TQ1JPTExfTUFSR0lOX01VTFRJUExJRVI7XG4gICAgICAgICAgY29uc3QgbWluVmlydHVhbEl0ZW1zID1cbiAgICAgICAgICAgIHJhbmdlU2l6ZSAqXG4gICAgICAgICAgICBWSUVXUE9SVF9DT05TVEFOVFMuUEFHSU5BVElPTi5DVVJTT1JfTUlOX1ZJUlRVQUxfU0laRV9NVUxUSVBMSUVSO1xuICAgICAgICAgIGNvbnN0IGR5bmFtaWNUb3RhbCA9IE1hdGgubWF4KFxuICAgICAgICAgICAgbG9hZGVkSXRlbXNDb3VudCArIG1hcmdpbkl0ZW1zLFxuICAgICAgICAgICAgbWluVmlydHVhbEl0ZW1zLFxuICAgICAgICAgICk7XG5cbiAgICAgICAgICBpZiAoZHluYW1pY1RvdGFsICE9PSB0b3RhbEl0ZW1zKSB7XG4gICAgICAgICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgIC8vICAgYFtDb2xsZWN0aW9uXSBVcGRhdGluZyBjdXJzb3IgdmlydHVhbCBzaXplIGZyb20gJHt0b3RhbEl0ZW1zfSB0byAke2R5bmFtaWNUb3RhbH1gLFxuICAgICAgICAgICAgLy8gKTtcbiAgICAgICAgICAgIHNldFRvdGFsSXRlbXMoZHluYW1pY1RvdGFsKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pO1xuXG4gICAgICAvLyBMaXN0ZW4gZm9yIHZlbG9jaXR5IGNoYW5nZXNcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwidmlld3BvcnQ6dmVsb2NpdHktY2hhbmdlZFwiLCAoZGF0YTogYW55KSA9PiB7XG4gICAgICAgIGNvbnN0IHByZXZpb3VzVmVsb2NpdHkgPSBjdXJyZW50VmVsb2NpdHk7XG4gICAgICAgIGN1cnJlbnRWZWxvY2l0eSA9IE1hdGguYWJzKGRhdGEudmVsb2NpdHkgfHwgMCk7XG5cbiAgICAgICAgLy8gV2hlbiB2ZWxvY2l0eSBkcm9wcyBiZWxvdyB0aHJlc2hvbGQsIHByb2Nlc3MgcXVldWVcbiAgICAgICAgaWYgKFxuICAgICAgICAgIHByZXZpb3VzVmVsb2NpdHkgPiBjYW5jZWxMb2FkVGhyZXNob2xkICYmXG4gICAgICAgICAgY3VycmVudFZlbG9jaXR5IDw9IGNhbmNlbExvYWRUaHJlc2hvbGRcbiAgICAgICAgKSB7XG4gICAgICAgICAgcHJvY2Vzc1F1ZXVlKCk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuXG4gICAgICAvLyBMaXN0ZW4gZm9yIGlkbGUgc3RhdGUgdG8gcHJvY2VzcyBxdWV1ZVxuICAgICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDppZGxlXCIsIGFzeW5jIChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgY3VycmVudFZlbG9jaXR5ID0gMDtcblxuICAgICAgICAvLyBSZXNldCBkcmFnZ2luZyBzdGF0ZSBvbiBpZGxlIHNpbmNlIHVzZXIgaGFzIHN0b3BwZWQgbW92aW5nXG4gICAgICAgIGlmIChpc0RyYWdnaW5nKSB7XG4gICAgICAgICAgaXNEcmFnZ2luZyA9IGZhbHNlO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gR2V0IGN1cnJlbnQgdmlzaWJsZSByYW5nZSBmcm9tIHZpZXdwb3J0XG4gICAgICAgIGNvbnN0IHZpZXdwb3J0U3RhdGUgPSAoY29tcG9uZW50LnZpZXdwb3J0IGFzIGFueSkuc3RhdGU7XG4gICAgICAgIGNvbnN0IHZpc2libGVSYW5nZSA9IHZpZXdwb3J0U3RhdGU/LnZpc2libGVSYW5nZTtcblxuICAgICAgICBpZiAodmlzaWJsZVJhbmdlKSB7XG4gICAgICAgICAgLy8gQ2xlYXIgc3RhbGUgcmVxdWVzdHMgZnJvbSBxdWV1ZSB0aGF0IGFyZSBmYXIgZnJvbSBjdXJyZW50IHZpc2libGUgcmFuZ2VcbiAgICAgICAgICBjb25zdCBidWZmZXIgPSByYW5nZVNpemUgKiAyOyAvLyBBbGxvdyBzb21lIGJ1ZmZlclxuICAgICAgICAgIGxvYWRSZXF1ZXN0UXVldWUgPSBsb2FkUmVxdWVzdFF1ZXVlLmZpbHRlcigocmVxdWVzdCkgPT4ge1xuICAgICAgICAgICAgY29uc3QgcmVxdWVzdEVuZCA9IHJlcXVlc3QucmFuZ2UuZW5kO1xuICAgICAgICAgICAgY29uc3QgcmVxdWVzdFN0YXJ0ID0gcmVxdWVzdC5yYW5nZS5zdGFydDtcbiAgICAgICAgICAgIGNvbnN0IGlzUmVsZXZhbnQgPVxuICAgICAgICAgICAgICByZXF1ZXN0RW5kID49IHZpc2libGVSYW5nZS5zdGFydCAtIGJ1ZmZlciAmJlxuICAgICAgICAgICAgICByZXF1ZXN0U3RhcnQgPD0gdmlzaWJsZVJhbmdlLmVuZCArIGJ1ZmZlcjtcblxuICAgICAgICAgICAgaWYgKCFpc1JlbGV2YW50KSB7XG4gICAgICAgICAgICAgIHJlcXVlc3QucmVzb2x2ZSgpOyAvLyBSZXNvbHZlIHRvIGF2b2lkIGhhbmdpbmcgcHJvbWlzZXNcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJldHVybiBpc1JlbGV2YW50O1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgLy8gTG9hZCB0aGUgY3VycmVudCB2aXNpYmxlIHJhbmdlIGlmIG5lZWRlZFxuICAgICAgICAgIGF3YWl0IGxvYWRNaXNzaW5nUmFuZ2VzKHZpc2libGVSYW5nZSwgXCJ2aWV3cG9ydDppZGxlXCIpO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gQWxzbyBwcm9jZXNzIGFueSBxdWV1ZWQgcmVxdWVzdHNcbiAgICAgICAgcHJvY2Vzc1F1ZXVlKCk7XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciBpdGVtIHJlbW92YWwgLSBET04nVCBjbGVhciBsb2FkZWRSYW5nZXMgdG8gcHJldmVudCB1bm5lY2Vzc2FyeSByZWxvYWRcbiAgICAgIC8vIFRoZSBkYXRhIGlzIGFscmVhZHkgc2hpZnRlZCBsb2NhbGx5IGluIGFwaS50cyBhbmQgcmVuZGVyaW5nLnRzXG4gICAgICAvLyBSZWxvYWRpbmcgd291bGQgY2F1c2UgcmFjZSBjb25kaXRpb25zIGFuZCBvdmVyd3JpdGUgdGhlIGNvcnJlY3QgdG90YWxJdGVtc1xuICAgICAgY29tcG9uZW50Lm9uPy4oXCJpdGVtOnJlbW92ZWRcIiwgKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICAvLyBjb25zb2xlLmxvZyhgW0NvbGxlY3Rpb25dIGl0ZW06cmVtb3ZlZCBldmVudCAtIGluZGV4OiAke2RhdGEuaW5kZXh9YCk7XG4gICAgICAgIC8vIGNvbnNvbGUubG9nKGBbQ29sbGVjdGlvbl0gaXRlbXMubGVuZ3RoOiAke2l0ZW1zLmxlbmd0aH1gKTtcblxuICAgICAgICAvLyBVcGRhdGUgZGlzY292ZXJlZFRvdGFsIHRvIG1hdGNoIHRoZSBuZXcgY291bnRcbiAgICAgICAgLy8gVGhpcyBwcmV2ZW50cyBzdGFsZSBkaXNjb3ZlcmVkVG90YWwgZnJvbSBiZWluZyB1c2VkIG9uIG5leHQgbG9hZFxuICAgICAgICBpZiAoZGlzY292ZXJlZFRvdGFsICE9PSBudWxsICYmIGRpc2NvdmVyZWRUb3RhbCA+IDApIHtcbiAgICAgICAgICBkaXNjb3ZlcmVkVG90YWwgPSBkaXNjb3ZlcmVkVG90YWwgLSAxO1xuICAgICAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgICAgIC8vICAgYFtDb2xsZWN0aW9uXSBVcGRhdGVkIGRpc2NvdmVyZWRUb3RhbCB0bzogJHtkaXNjb3ZlcmVkVG90YWx9YCxcbiAgICAgICAgICAvLyApO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gTk9URTogRG8gTk9UIGRlY3JlbWVudCB0b3RhbEl0ZW1zIGhlcmUhXG4gICAgICAgIC8vIGFwaS50cyBhbHJlYWR5IGNhbGxzIHNldFRvdGFsSXRlbXMoKSBhZnRlciBlbWl0dGluZyBpdGVtOnJlbW92ZS1yZXF1ZXN0LFxuICAgICAgICAvLyB3aGljaCBwcm9wZXJseSB1cGRhdGVzIHRvdGFsSXRlbXMuIERlY3JlbWVudGluZyBoZXJlIHdvdWxkIGNhdXNlIGEgZG91YmxlLWRlY3JlbWVudC5cblxuICAgICAgICAvLyBET04nVCBjbGVhciBsb2FkZWRSYW5nZXMgLSB3ZSB3YW50IHRvIGtlZXAgdXNpbmcgdGhlIGxvY2FsIGRhdGFcbiAgICAgICAgLy8gVGhlIGRhdGEgaGFzIGJlZW4gc2hpZnRlZCBsb2NhbGx5IGFuZCBpcyBzdGlsbCB2YWxpZFxuICAgICAgICAvLyBDbGVhcmluZyB3b3VsZCB0cmlnZ2VyIGEgcmVsb2FkIHdoaWNoIGNhdXNlcyByYWNlIGNvbmRpdGlvbnNcbiAgICAgICAgLy8gY29uc29sZS5sb2coXG4gICAgICAgIC8vICAgYFtDb2xsZWN0aW9uXSBLZWVwaW5nIGxvYWRlZFJhbmdlcyBpbnRhY3Q6YCxcbiAgICAgICAgLy8gICBBcnJheS5mcm9tKGxvYWRlZFJhbmdlcyksXG4gICAgICAgIC8vICk7XG4gICAgICB9KTtcblxuICAgICAgLy8gTG9hZCBpbml0aWFsIGRhdGEgaWYgY29sbGVjdGlvbiBpcyBhdmFpbGFibGUgYW5kIGF1dG9Mb2FkIGlzIGVuYWJsZWRcbiAgICAgIGlmIChjb2xsZWN0aW9uICYmIGF1dG9Mb2FkKSB7XG4gICAgICAgIC8vIElmIHdlIGhhdmUgYW4gaW5pdGlhbCBzY3JvbGwgaW5kZXggT1IgYSBzZWxlY3RJZCwgbG9hZCBkYXRhIGZvciB0aGF0IHBvc2l0aW9uIGRpcmVjdGx5XG4gICAgICAgIC8vIERvbid0IHVzZSBzY3JvbGxUb0luZGV4KCkgYXMgaXQgdHJpZ2dlcnMgYW5pbWF0aW9uL3ZlbG9jaXR5IHRyYWNraW5nXG4gICAgICAgIC8vIHZpcnR1YWwudHMgaGFzIGFscmVhZHkgc2V0IHRoZSBzY3JvbGwgcG9zaXRpb24gYW5kIGNhbGN1bGF0ZWQgdGhlIHZpc2libGUgcmFuZ2VcbiAgICAgICAgaWYgKFxuICAgICAgICAgIGluaXRpYWxTY3JvbGxJbmRleCA+IDAgfHxcbiAgICAgICAgICAoc2VsZWN0SWQgIT09IHVuZGVmaW5lZCAmJiBzZWxlY3RJZCAhPT0gbnVsbClcbiAgICAgICAgKSB7XG4gICAgICAgICAgLy8gR2V0IHRoZSB2aXNpYmxlIHJhbmdlIHRoYXQgd2FzIGFscmVhZHkgY2FsY3VsYXRlZCBieSB2aXJ0dWFsLnRzXG4gICAgICAgICAgLy8gV2UgbWlzc2VkIHRoZSBpbml0aWFsIHZpZXdwb3J0OnJhbmdlLWNoYW5nZWQgZXZlbnQgYmVjYXVzZSBvdXIgbGlzdGVuZXIgd2Fzbid0IHJlYWR5IHlldFxuICAgICAgICAgIGNvbnN0IHZpc2libGVSYW5nZSA9IGNvbXBvbmVudC52aWV3cG9ydD8uZ2V0VmlzaWJsZVJhbmdlPy4oKTtcblxuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgIHZpc2libGVSYW5nZSAmJlxuICAgICAgICAgICAgKHZpc2libGVSYW5nZS5zdGFydCA+IDAgfHwgdmlzaWJsZVJhbmdlLmVuZCA+IDApXG4gICAgICAgICAgKSB7XG4gICAgICAgICAgICAvLyBVc2UgdGhlIHByZS1jYWxjdWxhdGVkIHZpc2libGUgcmFuZ2UgZnJvbSB2aXJ0dWFsLnRzXG4gICAgICAgICAgICBsb2FkTWlzc2luZ1Jhbmdlcyh2aXNpYmxlUmFuZ2UsIFwiaW5pdGlhbC1wb3NpdGlvblwiKVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgICAgICAgICAgaGFzQ29tcGxldGVkSW5pdGlhbFBvc2l0aW9uTG9hZCA9IHRydWU7XG4gICAgICAgICAgICAgICAgLy8gRW1pdCBldmVudCB0byBzZWxlY3QgaXRlbSBhZnRlciBpbml0aWFsIGxvYWQgaWYgc2VsZWN0SWQgaXMgcHJvdmlkZWRcbiAgICAgICAgICAgICAgICBpZiAoc2VsZWN0SWQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICAgICAgY29tcG9uZW50LmVtaXQ/LihcImNvbGxlY3Rpb246aW5pdGlhbC1sb2FkLWNvbXBsZXRlXCIsIHtcbiAgICAgICAgICAgICAgICAgICAgc2VsZWN0SWQsXG4gICAgICAgICAgICAgICAgICAgIGluaXRpYWxTY3JvbGxJbmRleCxcbiAgICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcikgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IoXG4gICAgICAgICAgICAgICAgICBcIltDb2xsZWN0aW9uXSBGYWlsZWQgdG8gbG9hZCBpbml0aWFsIHBvc2l0aW9uIGRhdGE6XCIsXG4gICAgICAgICAgICAgICAgICBlcnJvcixcbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGhhc0NvbXBsZXRlZEluaXRpYWxQb3NpdGlvbkxvYWQgPSB0cnVlOyAvLyBBbGxvdyBub3JtYWwgbG9hZGluZyBldmVuIG9uIGVycm9yXG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyBGYWxsYmFjazogY2FsY3VsYXRlIHJhbmdlIGZyb20gaW5pdGlhbFNjcm9sbEluZGV4XG4gICAgICAgICAgICAvLyBUaGlzIGhhbmRsZXMgZWRnZSBjYXNlcyB3aGVyZSB2aXNpYmxlUmFuZ2Ugd2Fzbid0IHJlYWR5XG4gICAgICAgICAgICBjb25zdCBvdmVyc2NhbiA9IDI7XG4gICAgICAgICAgICBjb25zdCBlc3RpbWF0ZWRWaXNpYmxlQ291bnQgPSBNYXRoLmNlaWwoNjAwIC8gNTApOyAvLyB+MTIgaXRlbXNcbiAgICAgICAgICAgIGNvbnN0IHN0YXJ0ID0gTWF0aC5tYXgoMCwgaW5pdGlhbFNjcm9sbEluZGV4IC0gb3ZlcnNjYW4pO1xuICAgICAgICAgICAgY29uc3QgZW5kID0gaW5pdGlhbFNjcm9sbEluZGV4ICsgZXN0aW1hdGVkVmlzaWJsZUNvdW50ICsgb3ZlcnNjYW47XG5cbiAgICAgICAgICAgIGxvYWRNaXNzaW5nUmFuZ2VzKHsgc3RhcnQsIGVuZCB9LCBcImluaXRpYWwtcG9zaXRpb24tZmFsbGJhY2tcIilcbiAgICAgICAgICAgICAgLnRoZW4oKCkgPT4ge1xuICAgICAgICAgICAgICAgIGhhc0NvbXBsZXRlZEluaXRpYWxQb3NpdGlvbkxvYWQgPSB0cnVlO1xuICAgICAgICAgICAgICAgIC8vIEVtaXQgZXZlbnQgdG8gc2VsZWN0IGl0ZW0gYWZ0ZXIgaW5pdGlhbCBsb2FkIGlmIHNlbGVjdElkIGlzIHByb3ZpZGVkXG4gICAgICAgICAgICAgICAgaWYgKHNlbGVjdElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJjb2xsZWN0aW9uOmluaXRpYWwtbG9hZC1jb21wbGV0ZVwiLCB7XG4gICAgICAgICAgICAgICAgICAgIHNlbGVjdElkLFxuICAgICAgICAgICAgICAgICAgICBpbml0aWFsU2Nyb2xsSW5kZXgsXG4gICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgIC5jYXRjaCgoZXJyb3IpID0+IHtcbiAgICAgICAgICAgICAgICBjb25zb2xlLmVycm9yKFxuICAgICAgICAgICAgICAgICAgXCJbQ29sbGVjdGlvbl0gRmFpbGVkIHRvIGxvYWQgaW5pdGlhbCBwb3NpdGlvbiBkYXRhIChmYWxsYmFjayk6XCIsXG4gICAgICAgICAgICAgICAgICBlcnJvcixcbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGhhc0NvbXBsZXRlZEluaXRpYWxQb3NpdGlvbkxvYWQgPSB0cnVlOyAvLyBBbGxvdyBub3JtYWwgbG9hZGluZyBldmVuIG9uIGVycm9yXG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICAvLyBObyBpbml0aWFsIHNjcm9sbCBpbmRleCAtIGxvYWQgZnJvbSBiZWdpbm5pbmcgYXMgbm9ybWFsXG4gICAgICAgIGxvYWRSYW5nZSgwLCByYW5nZVNpemUsIFwiaW5pdGlhbGl6ZS1wYWdlMVwiKVxuICAgICAgICAgIC50aGVuKCgpID0+IHtcbiAgICAgICAgICAgIC8vIEhhbmRsZSBhdXRvU2VsZWN0Rmlyc3QgLSBnZXQgZmlyc3QgaXRlbSBJRCBhbmQgZW1pdCBzZWxlY3Rpb24gZXZlbnRcbiAgICAgICAgICAgIGlmIChhdXRvU2VsZWN0Rmlyc3QgJiYgaXRlbXMubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgICBjb25zdCBmaXJzdEl0ZW0gPSBpdGVtc1swXTtcbiAgICAgICAgICAgICAgY29uc3QgZmlyc3RJZCA9IGZpcnN0SXRlbT8uX2lkIHx8IGZpcnN0SXRlbT8uaWQ7XG4gICAgICAgICAgICAgIGlmIChmaXJzdElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICBjb21wb25lbnQuZW1pdD8uKFwiY29sbGVjdGlvbjppbml0aWFsLWxvYWQtY29tcGxldGVcIiwge1xuICAgICAgICAgICAgICAgICAgc2VsZWN0SWQ6IGZpcnN0SWQsXG4gICAgICAgICAgICAgICAgICBpbml0aWFsU2Nyb2xsSW5kZXg6IDAsXG4gICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KVxuICAgICAgICAgIC5jYXRjaCgoZXJyb3IpID0+IHtcbiAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IoXCJbQ29sbGVjdGlvbl0gRmFpbGVkIHRvIGxvYWQgaW5pdGlhbCBkYXRhOlwiLCBlcnJvcik7XG4gICAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgfTtcblxuICAgIC8qKlxuICAgICAqIFJlc2V0IGNvbGxlY3Rpb24gc3RhdGUgd2l0aG91dCBkZXN0cm95aW5nIHRoZSBjb21wb25lbnRcbiAgICAgKiBDbGVhcnMgYWxsIGxvYWRlZCBkYXRhIGFuZCBwcmVwYXJlcyBmb3IgZnJlc2ggbG9hZFxuICAgICAqL1xuICAgIGNvbnN0IHJlc2V0ID0gKCkgPT4ge1xuICAgICAgbG9nTWVtb3J5U3RhdHMoXCJyZXNldDpiZWZvcmVcIik7XG5cbiAgICAgIC8vIEFib3J0IGFsbCBpbi1mbGlnaHQgcmVxdWVzdHNcbiAgICAgIGFib3J0Q29udHJvbGxlcnMuZm9yRWFjaCgoY29udHJvbGxlcikgPT4ge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIGNvbnRyb2xsZXIuYWJvcnQoKTtcbiAgICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICAgIC8vIElnbm9yZSBhYm9ydCBlcnJvcnNcbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgICBhYm9ydENvbnRyb2xsZXJzLmNsZWFyKCk7XG5cbiAgICAgIC8vIENsZWFyIGFsbCBkYXRhIHN0cnVjdHVyZXNcbiAgICAgIGl0ZW1zLmxlbmd0aCA9IDA7XG4gICAgICBsb2FkUmVxdWVzdFF1ZXVlLmxlbmd0aCA9IDA7XG4gICAgICBsb2FkZWRSYW5nZXMuY2xlYXIoKTtcbiAgICAgIHBlbmRpbmdSYW5nZXMuY2xlYXIoKTtcbiAgICAgIGZhaWxlZFJhbmdlcy5jbGVhcigpO1xuICAgICAgYWN0aXZlTG9hZFJhbmdlcy5jbGVhcigpO1xuICAgICAgYWN0aXZlUmVxdWVzdHMuY2xlYXIoKTtcbiAgICAgIGN1cnNvck1hcC5jbGVhcigpO1xuICAgICAgcGFnZVRvT2Zmc2V0TWFwLmNsZWFyKCk7XG5cbiAgICAgIC8vIFJlc2V0IHN0YXRlXG4gICAgICB0b3RhbEl0ZW1zID0gMDtcbiAgICAgIGN1cnJlbnRDdXJzb3IgPSBudWxsO1xuICAgICAgaGlnaGVzdExvYWRlZFBhZ2UgPSAwO1xuICAgICAgZGlzY292ZXJlZFRvdGFsID0gbnVsbDtcbiAgICAgIGhhc1JlYWNoZWRFbmQgPSBmYWxzZTtcbiAgICAgIGhhc0NvbXBsZXRlZEluaXRpYWxQb3NpdGlvbkxvYWQgPSBmYWxzZTtcbiAgICAgIGhhc01hbnVhbGx5TG9hZGVkID0gZmFsc2U7XG5cbiAgICAgIC8vIFJlc2V0IGNhY2hlZCBpdGVtIGNvdW50XG4gICAgICBjYWNoZWRJdGVtQ291bnQgPSAwO1xuXG4gICAgICAvLyBSZXNldCBjb3VudGVyc1xuICAgICAgYWN0aXZlTG9hZENvdW50ID0gMDtcbiAgICAgIGNvbXBsZXRlZExvYWRzID0gMDtcbiAgICAgIGZhaWxlZExvYWRzID0gMDtcbiAgICAgIGNhbmNlbGxlZExvYWRzID0gMDtcblxuICAgICAgLy8gVXBkYXRlIGNvbXBvbmVudCByZWZlcmVuY2VcbiAgICAgIGNvbXBvbmVudC5pdGVtcyA9IGl0ZW1zO1xuICAgICAgY29tcG9uZW50LnRvdGFsSXRlbXMgPSAwO1xuXG4gICAgICBsb2dNZW1vcnlTdGF0cyhcInJlc2V0OmFmdGVyXCIpO1xuXG4gICAgICAvLyBFbWl0IHJlc2V0IGV2ZW50XG4gICAgICBjb21wb25lbnQuZW1pdD8uKFwiY29sbGVjdGlvbjpyZXNldFwiKTtcbiAgICB9O1xuXG4gICAgLyoqXG4gICAgICogQ2xlYXIgdGhlIHJlcXVlc3QgcXVldWUgLSB1c2VkIGJ5IHJlbG9hZEF0IHRvIHByZXZlbnQgc3RhbGUgcXVldWVkIHJlcXVlc3RzXG4gICAgICovXG4gICAgY29uc3QgY2xlYXJRdWV1ZSA9ICgpID0+IHtcbiAgICAgIC8vIFJlc29sdmUgYWxsIHBlbmRpbmcgcHJvbWlzZXMgdG8gYXZvaWQgaGFuZ2luZ1xuICAgICAgbG9hZFJlcXVlc3RRdWV1ZS5mb3JFYWNoKChyZXF1ZXN0KSA9PiB7XG4gICAgICAgIHJlcXVlc3QucmVzb2x2ZSgpO1xuICAgICAgfSk7XG4gICAgICBsb2FkUmVxdWVzdFF1ZXVlLmxlbmd0aCA9IDA7XG4gICAgfTtcblxuICAgIC8qKlxuICAgICAqIE1hcmsgdGhhdCBkYXRhIGhhcyBiZWVuIG1hbnVhbGx5IGxvYWRlZCAoZS5nLiwgdmlhIHJlbG9hZEF0KVxuICAgICAqIFRoaXMgZW5hYmxlcyBzdWJzZXF1ZW50IGxvYWRzIGZyb20gdmlld3BvcnQ6cmFuZ2UtY2hhbmdlZFxuICAgICAqL1xuICAgIGNvbnN0IHNldE1hbnVhbGx5TG9hZGVkID0gKCkgPT4ge1xuICAgICAgaGFzTWFudWFsbHlMb2FkZWQgPSB0cnVlO1xuICAgIH07XG5cbiAgICAvKipcbiAgICAgKiBTZXQgcmVsb2FkIGluIHByb2dyZXNzIGZsYWcgd2l0aCB0YXJnZXQgcmFuZ2VcbiAgICAgKiBVc2VkIGJ5IHJlbG9hZEF0IHRvIHByZXZlbnQgdmlld3BvcnQ6cmFuZ2UtY2hhbmdlZCBmcm9tIGxvYWRpbmcgcGFnZSAxXG4gICAgICovXG4gICAgY29uc3Qgc2V0UmVsb2FkSW5Qcm9ncmVzcyA9IChcbiAgICAgIGluUHJvZ3Jlc3M6IGJvb2xlYW4sXG4gICAgICB0YXJnZXRSYW5nZT86IHsgc3RhcnQ6IG51bWJlcjsgZW5kOiBudW1iZXIgfSxcbiAgICApID0+IHtcbiAgICAgIHJlbG9hZEluUHJvZ3Jlc3MgPSBpblByb2dyZXNzO1xuICAgICAgcmVsb2FkVGFyZ2V0UmFuZ2UgPSB0YXJnZXRSYW5nZSB8fCBudWxsO1xuICAgIH07XG5cbiAgICAvLyBBZGQgY29sbGVjdGlvbiBBUEkgdG8gdmlld3BvcnRcbiAgICBjb21wb25lbnQudmlld3BvcnQuY29sbGVjdGlvbiA9IHtcbiAgICAgIGxvYWRSYW5nZTogKG9mZnNldDogbnVtYmVyLCBsaW1pdDogbnVtYmVyKSA9PiBsb2FkUmFuZ2Uob2Zmc2V0LCBsaW1pdCksXG4gICAgICBsb2FkTWlzc2luZ1JhbmdlczogKFxuICAgICAgICByYW5nZTogeyBzdGFydDogbnVtYmVyOyBlbmQ6IG51bWJlciB9LFxuICAgICAgICBjYWxsZXI/OiBzdHJpbmcsXG4gICAgICApID0+IGxvYWRNaXNzaW5nUmFuZ2VzKHJhbmdlLCBjYWxsZXIgfHwgXCJ2aWV3cG9ydC5jb2xsZWN0aW9uXCIpLFxuICAgICAgZ2V0TG9hZGVkUmFuZ2VzOiAoKSA9PiBsb2FkZWRSYW5nZXMsXG4gICAgICBnZXRQZW5kaW5nUmFuZ2VzOiAoKSA9PiBwZW5kaW5nUmFuZ2VzLFxuICAgICAgY2xlYXJGYWlsZWRSYW5nZXM6ICgpID0+IGZhaWxlZFJhbmdlcy5jbGVhcigpLFxuICAgICAgY2xlYXJRdWV1ZSxcbiAgICAgIHNldE1hbnVhbGx5TG9hZGVkLFxuICAgICAgc2V0UmVsb2FkSW5Qcm9ncmVzcyxcbiAgICAgIHJldHJ5RmFpbGVkUmFuZ2UsXG4gICAgICBzZXRUb3RhbEl0ZW1zLFxuICAgICAgZ2V0VG90YWxJdGVtczogKCkgPT4gdG90YWxJdGVtcyxcbiAgICAgIC8vIEN1cnNvci1zcGVjaWZpYyBtZXRob2RzXG4gICAgICBnZXRDdXJyZW50Q3Vyc29yOiAoKSA9PiBjdXJyZW50Q3Vyc29yLFxuICAgICAgZ2V0Q3Vyc29yRm9yUGFnZTogKHBhZ2U6IG51bWJlcikgPT4gY3Vyc29yTWFwLmdldChwYWdlKSB8fCBudWxsLFxuICAgICAgLy8gUmVzZXQgbWV0aG9kIGZvciByZWxvYWQgZnVuY3Rpb25hbGl0eVxuICAgICAgcmVzZXQsXG4gICAgfTtcblxuICAgIC8vIEFkZCBjb2xsZWN0aW9uIGRhdGEgYWNjZXNzIHdpdGggZGlyZWN0IGFzc2lnbm1lbnRcbiAgICAoY29tcG9uZW50IGFzIGFueSkuY29sbGVjdGlvbiA9IHtcbiAgICAgIGl0ZW1zLFxuICAgICAgZ2V0SXRlbXM6ICgpID0+IGl0ZW1zLFxuICAgICAgZ2V0SXRlbTogKGluZGV4OiBudW1iZXIpID0+IGl0ZW1zW2luZGV4XSxcbiAgICAgIGxvYWRSYW5nZSxcbiAgICAgIGxvYWRNaXNzaW5nUmFuZ2VzOiAoXG4gICAgICAgIHJhbmdlOiB7IHN0YXJ0OiBudW1iZXI7IGVuZDogbnVtYmVyIH0sXG4gICAgICAgIGNhbGxlcj86IHN0cmluZyxcbiAgICAgICkgPT4gbG9hZE1pc3NpbmdSYW5nZXMocmFuZ2UsIGNhbGxlciB8fCBcImNvbXBvbmVudC5jb2xsZWN0aW9uXCIpLFxuICAgICAgZ2V0TG9hZGluZ1N0YXRzOiAoKSA9PiAoe1xuICAgICAgICBwZW5kaW5nUmVxdWVzdHM6IGFjdGl2ZUxvYWRDb3VudCxcbiAgICAgICAgY29tcGxldGVkUmVxdWVzdHM6IGNvbXBsZXRlZExvYWRzLFxuICAgICAgICBmYWlsZWRSZXF1ZXN0czogZmFpbGVkTG9hZHMsXG4gICAgICAgIGNhbmNlbGxlZFJlcXVlc3RzOiBjYW5jZWxsZWRMb2FkcyxcbiAgICAgICAgY3VycmVudFZlbG9jaXR5LFxuICAgICAgICBjYW5Mb2FkOiBjYW5Mb2FkKCksXG4gICAgICAgIHF1ZXVlZFJlcXVlc3RzOiBsb2FkUmVxdWVzdFF1ZXVlLmxlbmd0aCxcbiAgICAgIH0pLFxuICAgICAgLy8gVG90YWwgaXRlbXMgbWFuYWdlbWVudFxuICAgICAgZ2V0VG90YWxJdGVtczogKCkgPT4gdG90YWxJdGVtcyxcbiAgICAgIHNldFRvdGFsSXRlbXMsXG4gICAgICAvLyBDdXJzb3IgbWV0aG9kc1xuICAgICAgZ2V0Q3VycmVudEN1cnNvcjogKCkgPT4gY3VycmVudEN1cnNvcixcbiAgICAgIGdldEN1cnNvckZvclBhZ2U6IChwYWdlOiBudW1iZXIpID0+IGN1cnNvck1hcC5nZXQocGFnZSkgfHwgbnVsbCxcbiAgICAgIC8vIFJlc2V0IG1ldGhvZCBmb3IgcmVsb2FkIGZ1bmN0aW9uYWxpdHlcbiAgICAgIHJlc2V0LFxuICAgIH07XG5cbiAgICAvLyBBbHNvIGVuc3VyZSBjb21wb25lbnQuaXRlbXMgaXMgdXBkYXRlZFxuICAgIGNvbXBvbmVudC5pdGVtcyA9IGl0ZW1zO1xuXG4gICAgLy8gQ2xlYW51cCBmdW5jdGlvbiAtIGNvbXByZWhlbnNpdmUgbWVtb3J5IGNsZWFudXBcbiAgICBjb25zdCBkZXN0cm95ID0gKCkgPT4ge1xuICAgICAgbG9nTWVtb3J5U3RhdHMoXCJkZXN0cm95OmJlZm9yZVwiKTtcblxuICAgICAgLy8gQWJvcnQgYWxsIGluLWZsaWdodCByZXF1ZXN0c1xuICAgICAgYWJvcnRDb250cm9sbGVycy5mb3JFYWNoKChjb250cm9sbGVyKSA9PiB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgY29udHJvbGxlci5hYm9ydCgpO1xuICAgICAgICB9IGNhdGNoIChlKSB7XG4gICAgICAgICAgLy8gSWdub3JlIGFib3J0IGVycm9yc1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgICAgIGFib3J0Q29udHJvbGxlcnMuY2xlYXIoKTtcblxuICAgICAgLy8gQ2xlYXIgYWxsIGRhdGEgc3RydWN0dXJlc1xuICAgICAgaXRlbXMubGVuZ3RoID0gMDtcbiAgICAgIGxvYWRSZXF1ZXN0UXVldWUubGVuZ3RoID0gMDtcbiAgICAgIGxvYWRlZFJhbmdlcy5jbGVhcigpO1xuICAgICAgcGVuZGluZ1Jhbmdlcy5jbGVhcigpO1xuICAgICAgZmFpbGVkUmFuZ2VzLmNsZWFyKCk7XG4gICAgICBhY3RpdmVMb2FkUmFuZ2VzLmNsZWFyKCk7XG4gICAgICBhY3RpdmVSZXF1ZXN0cy5jbGVhcigpO1xuICAgICAgY3Vyc29yTWFwLmNsZWFyKCk7XG4gICAgICBwYWdlVG9PZmZzZXRNYXAuY2xlYXIoKTtcblxuICAgICAgLy8gUmVzZXQgc3RhdGVcbiAgICAgIHRvdGFsSXRlbXMgPSAwO1xuICAgICAgY3VycmVudEN1cnNvciA9IG51bGw7XG4gICAgICBoaWdoZXN0TG9hZGVkUGFnZSA9IDA7XG4gICAgICBkaXNjb3ZlcmVkVG90YWwgPSBudWxsO1xuICAgICAgaGFzUmVhY2hlZEVuZCA9IGZhbHNlO1xuXG4gICAgICAvLyBDbGVhciBjb21wb25lbnQgcmVmZXJlbmNlXG4gICAgICBpZiAoY29tcG9uZW50Lml0ZW1zID09PSBpdGVtcykge1xuICAgICAgICBjb21wb25lbnQuaXRlbXMgPSBbXTtcbiAgICAgIH1cblxuICAgICAgbG9nTWVtb3J5U3RhdHMoXCJkZXN0cm95OmFmdGVyXCIpO1xuICAgIH07XG5cbiAgICAvLyBXaXJlIGRlc3Ryb3kgaW50byB0aGUgY29tcG9uZW50J3MgZGVzdHJveSBjaGFpblxuICAgIHdyYXBEZXN0cm95KGNvbXBvbmVudCwgZGVzdHJveSk7XG5cbiAgICAvLyBSZXR1cm4gZW5oYW5jZWQgY29tcG9uZW50XG4gICAgcmV0dXJuIHtcbiAgICAgIC4uLmNvbXBvbmVudCxcbiAgICAgIGNvbGxlY3Rpb246IHtcbiAgICAgICAgLy8gRGF0YSBhY2Nlc3MgbWV0aG9kc1xuICAgICAgICBpdGVtcyxcbiAgICAgICAgZ2V0SXRlbXM6ICgpID0+IGl0ZW1zLFxuICAgICAgICBnZXRJdGVtOiAoaW5kZXg6IG51bWJlcikgPT4gaXRlbXNbaW5kZXhdLFxuICAgICAgICAvLyBMb2FkaW5nIG1ldGhvZHNcbiAgICAgICAgbG9hZFJhbmdlLFxuICAgICAgICBsb2FkTWlzc2luZ1JhbmdlczogKFxuICAgICAgICAgIHJhbmdlOiB7IHN0YXJ0OiBudW1iZXI7IGVuZDogbnVtYmVyIH0sXG4gICAgICAgICAgY2FsbGVyPzogc3RyaW5nLFxuICAgICAgICApID0+IGxvYWRNaXNzaW5nUmFuZ2VzKHJhbmdlLCBjYWxsZXIgfHwgXCJyZXR1cm4uY29sbGVjdGlvblwiKSxcbiAgICAgICAgZ2V0TG9hZGVkUmFuZ2VzOiAoKSA9PiBsb2FkZWRSYW5nZXMsXG4gICAgICAgIGdldFBlbmRpbmdSYW5nZXM6ICgpID0+IHBlbmRpbmdSYW5nZXMsXG4gICAgICAgIGNsZWFyRmFpbGVkUmFuZ2VzOiAoKSA9PiBmYWlsZWRSYW5nZXMuY2xlYXIoKSxcbiAgICAgICAgY2xlYXJRdWV1ZSxcbiAgICAgICAgc2V0TWFudWFsbHlMb2FkZWQsXG4gICAgICAgIHNldFJlbG9hZEluUHJvZ3Jlc3MsXG4gICAgICAgIHJldHJ5RmFpbGVkUmFuZ2UsXG4gICAgICAgIHNldFRvdGFsSXRlbXMsXG4gICAgICAgIGdldFRvdGFsSXRlbXM6ICgpID0+IHRvdGFsSXRlbXMsXG4gICAgICAgIC8vIEN1cnNvci1zcGVjaWZpYyBtZXRob2RzXG4gICAgICAgIGdldEN1cnJlbnRDdXJzb3I6ICgpID0+IGN1cnJlbnRDdXJzb3IsXG4gICAgICAgIGdldEN1cnNvckZvclBhZ2U6IChwYWdlOiBudW1iZXIpID0+IGN1cnNvck1hcC5nZXQocGFnZSkgfHwgbnVsbCxcbiAgICAgICAgLy8gUmVzZXQgbWV0aG9kIGZvciByZWxvYWQgZnVuY3Rpb25hbGl0eVxuICAgICAgICByZXNldCxcbiAgICAgIH0sXG4gICAgfTtcbiAgfTtcbn1cbiIsCiAgICAiLy8gc3JjL2NvcmUvdmlld3BvcnQvZmVhdHVyZXMvcGxhY2Vob2xkZXJzLnRzXG5cbi8qKlxuICogUGxhY2Vob2xkZXIgRmVhdHVyZSAtIFNtYXJ0IHBsYWNlaG9sZGVyIGdlbmVyYXRpb25cbiAqIEFuYWx5emVzIGZpcnN0IGxvYWRlZCBkYXRhIHRvIGdlbmVyYXRlIHJlYWxpc3RpYyBtYXNrZWQgcGxhY2Vob2xkZXJzXG4gKiBTaG93cyBwbGFjZWhvbGRlcnMgd2hpbGUgZGF0YSBpcyBsb2FkaW5nXG4gKi9cblxuaW1wb3J0IHR5cGUgeyBWaWV3cG9ydENvbnRleHQgfSBmcm9tIFwiLi4vdHlwZXNcIjtcbmltcG9ydCB0eXBlIHsgQ29sbGVjdGlvbkNvbXBvbmVudCB9IGZyb20gXCIuL2NvbGxlY3Rpb25cIjtcbmltcG9ydCB7IFZJRVdQT1JUX0NPTlNUQU5UUyB9IGZyb20gXCIuLi9jb25zdGFudHNcIjtcblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIGZvciBwbGFjZWhvbGRlciBmZWF0dXJlXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUGxhY2Vob2xkZXJDb25maWcge1xuICBlbmFibGVkPzogYm9vbGVhbjtcbiAgYW5hbHl6ZUZpcnN0TG9hZD86IGJvb2xlYW47XG4gIG1hc2tDaGFyYWN0ZXI/OiBzdHJpbmc7XG4gIHJhbmRvbUxlbmd0aFZhcmlhbmNlPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBGaWVsZCBzdHJ1Y3R1cmUgZm9yIHBsYWNlaG9sZGVyIGdlbmVyYXRpb25cbiAqL1xuaW50ZXJmYWNlIEZpZWxkU3RydWN0dXJlIHtcbiAgbWluTGVuZ3RoOiBudW1iZXI7XG4gIG1heExlbmd0aDogbnVtYmVyO1xuICBhdmdMZW5ndGg6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQbGFjZWhvbGRlckNvbXBvbmVudCB7XG4gIHBsYWNlaG9sZGVyczoge1xuICAgIGFuYWx5emVEYXRhU3RydWN0dXJlOiAoaXRlbXM6IGFueVtdKSA9PiB2b2lkO1xuICAgIGhhc0FuYWx5emVkU3RydWN0dXJlOiAoKSA9PiBib29sZWFuO1xuICAgIGdlbmVyYXRlUGxhY2Vob2xkZXJJdGVtOiAoaW5kZXg6IG51bWJlcikgPT4gYW55O1xuICAgIGdlbmVyYXRlUGxhY2Vob2xkZXJJdGVtczogKHJhbmdlOiB7IHN0YXJ0OiBudW1iZXI7IGVuZDogbnVtYmVyIH0pID0+IGFueVtdO1xuICAgIHNob3dQbGFjZWhvbGRlcnM6IChyYW5nZTogeyBzdGFydDogbnVtYmVyOyBlbmQ6IG51bWJlciB9KSA9PiB2b2lkO1xuICAgIGlzUGxhY2Vob2xkZXI6IChpdGVtOiBhbnkpID0+IGJvb2xlYW47XG4gICAgcmVwbGFjZVBsYWNlaG9sZGVyczogKGl0ZW1zOiBhbnlbXSwgb2Zmc2V0OiBudW1iZXIpID0+IHZvaWQ7XG4gICAgY2xlYXI6ICgpID0+IHZvaWQ7XG4gIH07XG59XG5cbi8qKlxuICogQWRkcyBwbGFjZWhvbGRlciBmdW5jdGlvbmFsaXR5IHRvIHZpZXdwb3J0IGNvbXBvbmVudFxuICovXG5leHBvcnQgZnVuY3Rpb24gd2l0aFBsYWNlaG9sZGVycyhjb25maWc6IFBsYWNlaG9sZGVyQ29uZmlnID0ge30pIHtcbiAgcmV0dXJuIDxUIGV4dGVuZHMgVmlld3BvcnRDb250ZXh0ICYgQ29sbGVjdGlvbkNvbXBvbmVudD4oXG4gICAgY29tcG9uZW50OiBUXG4gICk6IFQgJiBQbGFjZWhvbGRlckNvbXBvbmVudCA9PiB7XG4gICAgY29uc3Qge1xuICAgICAgZW5hYmxlZCA9IHRydWUsXG4gICAgICBhbmFseXplRmlyc3RMb2FkID0gdHJ1ZSxcbiAgICAgIG1hc2tDaGFyYWN0ZXIgPSBWSUVXUE9SVF9DT05TVEFOVFMuUExBQ0VIT0xERVIuTUFTS19DSEFSQUNURVIsXG4gICAgICByYW5kb21MZW5ndGhWYXJpYW5jZSA9IFZJRVdQT1JUX0NPTlNUQU5UUy5QTEFDRUhPTERFUlxuICAgICAgICAuUkFORE9NX0xFTkdUSF9WQVJJQU5DRSxcbiAgICB9ID0gY29uZmlnO1xuXG4gICAgLy8gU3RhdGVcbiAgICBsZXQgZmllbGRTdHJ1Y3R1cmVzOiBNYXA8c3RyaW5nLCBGaWVsZFN0cnVjdHVyZT4gfCBudWxsID0gbnVsbDtcbiAgICBsZXQgaGFzQW5hbHl6ZWQgPSBmYWxzZTtcbiAgICBsZXQgcGxhY2Vob2xkZXJJZENvdW50ZXIgPSAwO1xuXG4gICAgLyoqXG4gICAgICogQW5hbHl6ZSBkYXRhIHN0cnVjdHVyZSBmcm9tIGZpcnN0IGxvYWRlZCBpdGVtc1xuICAgICAqL1xuICAgIGNvbnN0IGFuYWx5emVEYXRhU3RydWN0dXJlID0gKGl0ZW1zOiBhbnlbXSk6IHZvaWQgPT4ge1xuICAgICAgaWYgKCFlbmFibGVkIHx8IGhhc0FuYWx5emVkIHx8ICFpdGVtcy5sZW5ndGgpIHtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuXG4gICAgICAvLyBjb25zb2xlLmxvZyhcbiAgICAgIC8vICAgYPCflI0gW1BMQUNFSE9MREVSU10gQW5hbHl6aW5nIGRhdGEgc3RydWN0dXJlIGZyb20gJHtpdGVtcy5sZW5ndGh9IGl0ZW1zYFxuICAgICAgLy8gKTtcblxuICAgICAgY29uc3Qgc3RydWN0dXJlcyA9IG5ldyBNYXA8c3RyaW5nLCBGaWVsZFN0cnVjdHVyZT4oKTtcbiAgICAgIGNvbnN0IHNhbXBsZVNpemUgPSBNYXRoLm1pbihcbiAgICAgICAgaXRlbXMubGVuZ3RoLFxuICAgICAgICBWSUVXUE9SVF9DT05TVEFOVFMuUExBQ0VIT0xERVIuTUFYX1NBTVBMRV9TSVpFXG4gICAgICApO1xuXG4gICAgICAvLyBBbmFseXplIGVhY2ggZmllbGQgYWNyb3NzIGFsbCBzYW1wbGUgaXRlbXNcbiAgICAgIGNvbnN0IGZpZWxkU3RhdHMgPSBuZXcgTWFwPHN0cmluZywgbnVtYmVyW10+KCk7XG5cbiAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgc2FtcGxlU2l6ZTsgaSsrKSB7XG4gICAgICAgIGNvbnN0IGl0ZW0gPSBpdGVtc1tpXTtcbiAgICAgICAgaWYgKCFpdGVtIHx8IHR5cGVvZiBpdGVtICE9PSBcIm9iamVjdFwiKSBjb250aW51ZTtcblxuICAgICAgICBPYmplY3Qua2V5cyhpdGVtKS5mb3JFYWNoKChmaWVsZCkgPT4ge1xuICAgICAgICAgIC8vIFNraXAgaW50ZXJuYWwgZmllbGRzXG4gICAgICAgICAgaWYgKFxuICAgICAgICAgICAgZmllbGQuc3RhcnRzV2l0aChcIl9cIikgfHxcbiAgICAgICAgICAgIGZpZWxkID09PSBWSUVXUE9SVF9DT05TVEFOVFMuUExBQ0VIT0xERVIuUExBQ0VIT0xERVJfRkxBR1xuICAgICAgICAgICkge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIGNvbnN0IHZhbHVlID0gU3RyaW5nKGl0ZW1bZmllbGRdIHx8IFwiXCIpO1xuICAgICAgICAgIGNvbnN0IGxlbmd0aCA9IHZhbHVlLmxlbmd0aDtcblxuICAgICAgICAgIGlmICghZmllbGRTdGF0cy5oYXMoZmllbGQpKSB7XG4gICAgICAgICAgICBmaWVsZFN0YXRzLnNldChmaWVsZCwgW10pO1xuICAgICAgICAgIH1cbiAgICAgICAgICBmaWVsZFN0YXRzLmdldChmaWVsZCkhLnB1c2gobGVuZ3RoKTtcbiAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIC8vIENhbGN1bGF0ZSBzdGF0aXN0aWNzIGZvciBlYWNoIGZpZWxkXG4gICAgICBmaWVsZFN0YXRzLmZvckVhY2goKGxlbmd0aHMsIGZpZWxkKSA9PiB7XG4gICAgICAgIGlmIChsZW5ndGhzLmxlbmd0aCA9PT0gMCkgcmV0dXJuO1xuXG4gICAgICAgIGNvbnN0IG1pbkxlbmd0aCA9IE1hdGgubWluKC4uLmxlbmd0aHMpO1xuICAgICAgICBjb25zdCBtYXhMZW5ndGggPSBNYXRoLm1heCguLi5sZW5ndGhzKTtcbiAgICAgICAgY29uc3QgYXZnTGVuZ3RoID0gTWF0aC5yb3VuZChcbiAgICAgICAgICBsZW5ndGhzLnJlZHVjZSgoc3VtLCBsZW4pID0+IHN1bSArIGxlbiwgMCkgLyBsZW5ndGhzLmxlbmd0aFxuICAgICAgICApO1xuXG4gICAgICAgIHN0cnVjdHVyZXMuc2V0KGZpZWxkLCB7XG4gICAgICAgICAgbWluTGVuZ3RoLFxuICAgICAgICAgIG1heExlbmd0aCxcbiAgICAgICAgICBhdmdMZW5ndGgsXG4gICAgICAgIH0pO1xuICAgICAgfSk7XG5cbiAgICAgIGZpZWxkU3RydWN0dXJlcyA9IHN0cnVjdHVyZXM7XG4gICAgICBoYXNBbmFseXplZCA9IHRydWU7XG5cbiAgICAgIC8vIGNvbnNvbGUubG9nKFxuICAgICAgLy8gICBg4pyFIFtQTEFDRUhPTERFUlNdIFN0cnVjdHVyZSBhbmFseXplZDpgLFxuICAgICAgLy8gICBPYmplY3QuZnJvbUVudHJpZXMoc3RydWN0dXJlcylcbiAgICAgIC8vICk7XG5cbiAgICAgIC8vIEVtaXQgZXZlbnRcbiAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpwbGFjZWhvbGRlcnMtc3RydWN0dXJlLWFuYWx5emVkXCIsIHtcbiAgICAgICAgc3RydWN0dXJlOiBPYmplY3QuZnJvbUVudHJpZXMoc3RydWN0dXJlcyksXG4gICAgICB9KTtcbiAgICB9O1xuXG4gICAgLyoqXG4gICAgICogR2VuZXJhdGUgYSBzaW5nbGUgcGxhY2Vob2xkZXIgaXRlbVxuICAgICAqL1xuICAgIGNvbnN0IGdlbmVyYXRlUGxhY2Vob2xkZXJJdGVtID0gKGluZGV4OiBudW1iZXIpOiBhbnkgPT4ge1xuICAgICAgY29uc3QgcGxhY2Vob2xkZXI6IFJlY29yZDxzdHJpbmcsIGFueT4gPSB7XG4gICAgICAgIGlkOiBgcGxhY2Vob2xkZXItJHtwbGFjZWhvbGRlcklkQ291bnRlcisrfWAsXG4gICAgICAgIFtWSUVXUE9SVF9DT05TVEFOVFMuUExBQ0VIT0xERVIuUExBQ0VIT0xERVJfRkxBR106IHRydWUsXG4gICAgICAgIF9pbmRleDogaW5kZXgsXG4gICAgICB9O1xuXG4gICAgICBpZiAoIWZpZWxkU3RydWN0dXJlcyB8fCBmaWVsZFN0cnVjdHVyZXMuc2l6ZSA9PT0gMCkge1xuICAgICAgICAvLyBObyBzdHJ1Y3R1cmUgYW5hbHl6ZWQgeWV0IC0gcmV0dXJuIGJhc2ljIHBsYWNlaG9sZGVyXG4gICAgICAgIHBsYWNlaG9sZGVyLmxhYmVsID0gbWFza0NoYXJhY3Rlci5yZXBlYXQoMTApO1xuICAgICAgICByZXR1cm4gcGxhY2Vob2xkZXI7XG4gICAgICB9XG5cbiAgICAgIC8vIEdlbmVyYXRlIGZpZWxkcyBiYXNlZCBvbiBhbmFseXplZCBzdHJ1Y3R1cmVcbiAgICAgIGZpZWxkU3RydWN0dXJlcy5mb3JFYWNoKChzdHJ1Y3R1cmUsIGZpZWxkKSA9PiB7XG4gICAgICAgIGxldCBsZW5ndGg6IG51bWJlcjtcblxuICAgICAgICBpZiAoXG4gICAgICAgICAgcmFuZG9tTGVuZ3RoVmFyaWFuY2UgJiZcbiAgICAgICAgICBzdHJ1Y3R1cmUubWluTGVuZ3RoICE9PSBzdHJ1Y3R1cmUubWF4TGVuZ3RoXG4gICAgICAgICkge1xuICAgICAgICAgIC8vIFJhbmRvbSBsZW5ndGggd2l0aGluIHJhbmdlXG4gICAgICAgICAgbGVuZ3RoID0gTWF0aC5mbG9vcihcbiAgICAgICAgICAgIE1hdGgucmFuZG9tKCkgKiAoc3RydWN0dXJlLm1heExlbmd0aCAtIHN0cnVjdHVyZS5taW5MZW5ndGggKyAxKSArXG4gICAgICAgICAgICAgIHN0cnVjdHVyZS5taW5MZW5ndGhcbiAgICAgICAgICApO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIFVzZSBhdmVyYWdlIGxlbmd0aFxuICAgICAgICAgIGxlbmd0aCA9IHN0cnVjdHVyZS5hdmdMZW5ndGg7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBBcHBseSBzb21lIHZhcmlhdGlvbiB0byBtYWtlIGl0IGxvb2sgbW9yZSBuYXR1cmFsXG4gICAgICAgIGlmIChyYW5kb21MZW5ndGhWYXJpYW5jZSAmJiBNYXRoLnJhbmRvbSgpIDwgMC4zKSB7XG4gICAgICAgICAgbGVuZ3RoID0gTWF0aC5tYXgoMSwgbGVuZ3RoICsgTWF0aC5mbG9vcihNYXRoLnJhbmRvbSgpICogMykgLSAxKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHBsYWNlaG9sZGVyW2ZpZWxkXSA9IG1hc2tDaGFyYWN0ZXIucmVwZWF0KGxlbmd0aCk7XG4gICAgICB9KTtcblxuICAgICAgcmV0dXJuIHBsYWNlaG9sZGVyO1xuICAgIH07XG5cbiAgICAvKipcbiAgICAgKiBHZW5lcmF0ZSBtdWx0aXBsZSBwbGFjZWhvbGRlciBpdGVtc1xuICAgICAqL1xuICAgIGNvbnN0IGdlbmVyYXRlUGxhY2Vob2xkZXJJdGVtcyA9IChyYW5nZToge1xuICAgICAgc3RhcnQ6IG51bWJlcjtcbiAgICAgIGVuZDogbnVtYmVyO1xuICAgIH0pOiBhbnlbXSA9PiB7XG4gICAgICBjb25zdCBpdGVtczogYW55W10gPSBbXTtcbiAgICAgIGZvciAobGV0IGkgPSByYW5nZS5zdGFydDsgaSA8PSByYW5nZS5lbmQ7IGkrKykge1xuICAgICAgICBpdGVtcy5wdXNoKGdlbmVyYXRlUGxhY2Vob2xkZXJJdGVtKGkpKTtcbiAgICAgIH1cbiAgICAgIHJldHVybiBpdGVtcztcbiAgICB9O1xuXG4gICAgLyoqXG4gICAgICogU2hvdyBwbGFjZWhvbGRlcnMgZm9yIGEgcmFuZ2VcbiAgICAgKi9cbiAgICBjb25zdCBzaG93UGxhY2Vob2xkZXJzID0gKHJhbmdlOiB7IHN0YXJ0OiBudW1iZXI7IGVuZDogbnVtYmVyIH0pOiB2b2lkID0+IHtcbiAgICAgIGlmICghZW5hYmxlZCkgcmV0dXJuO1xuXG4gICAgICBjb25zdCBwbGFjZWhvbGRlcnMgPSBnZW5lcmF0ZVBsYWNlaG9sZGVySXRlbXMocmFuZ2UpO1xuXG4gICAgICAvLyBVcGRhdGUgaXRlbXMgYXJyYXlcbiAgICAgIGlmIChjb21wb25lbnQuaXRlbXMpIHtcbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCBwbGFjZWhvbGRlcnMubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgICBjb25zdCBpbmRleCA9IHJhbmdlLnN0YXJ0ICsgaTtcbiAgICAgICAgICBpZiAoIWNvbXBvbmVudC5pdGVtc1tpbmRleF0pIHtcbiAgICAgICAgICAgIGNvbXBvbmVudC5pdGVtc1tpbmRleF0gPSBwbGFjZWhvbGRlcnNbaV07XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICBg8J+UhCBbUExBQ0VIT0xERVJTXSBTaG93aW5nICR7cGxhY2Vob2xkZXJzLmxlbmd0aH0gcGxhY2Vob2xkZXJzIGZvciByYW5nZSAke3JhbmdlLnN0YXJ0fS0ke3JhbmdlLmVuZH1gXG4gICAgICApO1xuXG4gICAgICAvLyBFbWl0IGV2ZW50XG4gICAgICBjb21wb25lbnQuZW1pdD8uKFwidmlld3BvcnQ6cGxhY2Vob2xkZXJzLXNob3duXCIsIHtcbiAgICAgICAgcmFuZ2UsXG4gICAgICAgIGNvdW50OiBwbGFjZWhvbGRlcnMubGVuZ3RoLFxuICAgICAgfSk7XG4gICAgfTtcblxuICAgIC8qKlxuICAgICAqIENoZWNrIGlmIGFuIGl0ZW0gaXMgYSBwbGFjZWhvbGRlclxuICAgICAqL1xuICAgIGNvbnN0IGlzUGxhY2Vob2xkZXIgPSAoaXRlbTogYW55KTogYm9vbGVhbiA9PiB7XG4gICAgICByZXR1cm4gKFxuICAgICAgICBpdGVtICYmXG4gICAgICAgIHR5cGVvZiBpdGVtID09PSBcIm9iamVjdFwiICYmXG4gICAgICAgIGl0ZW1bVklFV1BPUlRfQ09OU1RBTlRTLlBMQUNFSE9MREVSLlBMQUNFSE9MREVSX0ZMQUddID09PSB0cnVlXG4gICAgICApO1xuICAgIH07XG5cbiAgICAvKipcbiAgICAgKiBSZXBsYWNlIHBsYWNlaG9sZGVycyB3aXRoIHJlYWwgZGF0YVxuICAgICAqL1xuICAgIGNvbnN0IHJlcGxhY2VQbGFjZWhvbGRlcnMgPSAoaXRlbXM6IGFueVtdLCBvZmZzZXQ6IG51bWJlcik6IHZvaWQgPT4ge1xuICAgICAgaWYgKCFjb21wb25lbnQuaXRlbXMpIHJldHVybjtcblxuICAgICAgbGV0IHJlcGxhY2VkQ291bnQgPSAwO1xuXG4gICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGl0ZW1zLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgIGNvbnN0IGluZGV4ID0gb2Zmc2V0ICsgaTtcbiAgICAgICAgY29uc3QgY3VycmVudEl0ZW0gPSBjb21wb25lbnQuaXRlbXNbaW5kZXhdO1xuXG4gICAgICAgIGlmIChpc1BsYWNlaG9sZGVyKGN1cnJlbnRJdGVtKSkge1xuICAgICAgICAgIGNvbXBvbmVudC5pdGVtc1tpbmRleF0gPSBpdGVtc1tpXTtcbiAgICAgICAgICByZXBsYWNlZENvdW50Kys7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgaWYgKHJlcGxhY2VkQ291bnQgPiAwKSB7XG4gICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgIGDinKggW1BMQUNFSE9MREVSU10gUmVwbGFjZWQgJHtyZXBsYWNlZENvdW50fSBwbGFjZWhvbGRlcnMgYXQgb2Zmc2V0ICR7b2Zmc2V0fWBcbiAgICAgICAgKTtcblxuICAgICAgICAvLyBFbWl0IGV2ZW50XG4gICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpwbGFjZWhvbGRlcnMtcmVwbGFjZWRcIiwge1xuICAgICAgICAgIG9mZnNldCxcbiAgICAgICAgICBjb3VudDogcmVwbGFjZWRDb3VudCxcbiAgICAgICAgfSk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8qKlxuICAgICAqIENsZWFyIGFsbCBwbGFjZWhvbGRlciBzdGF0ZVxuICAgICAqL1xuICAgIGNvbnN0IGNsZWFyID0gKCk6IHZvaWQgPT4ge1xuICAgICAgZmllbGRTdHJ1Y3R1cmVzID0gbnVsbDtcbiAgICAgIGhhc0FuYWx5emVkID0gZmFsc2U7XG4gICAgICBwbGFjZWhvbGRlcklkQ291bnRlciA9IDA7XG4gICAgfTtcblxuICAgIC8vIEluaXRpYWxpemUgZnVuY3Rpb25cbiAgICBjb25zdCBpbml0aWFsaXplID0gKCkgPT4ge1xuICAgICAgaWYgKCFlbmFibGVkKSByZXR1cm47XG5cbiAgICAgIC8vIExpc3RlbiBmb3IgZmlyc3QgZGF0YSBsb2FkIHRvIGFuYWx5emUgc3RydWN0dXJlXG4gICAgICBpZiAoYW5hbHl6ZUZpcnN0TG9hZCkge1xuICAgICAgICBjb25zdCBoYW5kbGVSYW5nZUxvYWRlZCA9IChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgICBpZiAoIWhhc0FuYWx5emVkICYmIGRhdGEuaXRlbXMgJiYgZGF0YS5pdGVtcy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICBhbmFseXplRGF0YVN0cnVjdHVyZShkYXRhLml0ZW1zKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBSZXBsYWNlIGFueSBwbGFjZWhvbGRlcnMgd2l0aCByZWFsIGRhdGFcbiAgICAgICAgICByZXBsYWNlUGxhY2Vob2xkZXJzKGRhdGEuaXRlbXMsIGRhdGEub2Zmc2V0KTtcbiAgICAgICAgfTtcblxuICAgICAgICBjb21wb25lbnQub24/LihcInZpZXdwb3J0OnJhbmdlLWxvYWRlZFwiLCBoYW5kbGVSYW5nZUxvYWRlZCk7XG4gICAgICB9XG5cbiAgICAgIC8vIFNob3cgaW5pdGlhbCBwbGFjZWhvbGRlcnMgaWYgY29uZmlndXJlZFxuICAgICAgY29uc3QgdG90YWxJdGVtcyA9IGNvbXBvbmVudC5jb2xsZWN0aW9uPy5nZXRUb3RhbEl0ZW1zKCkgfHwgMDtcbiAgICAgIGlmICh0b3RhbEl0ZW1zID4gMCkge1xuICAgICAgICBjb25zdCBpbml0aWFsUmFuZ2UgPSB7XG4gICAgICAgICAgc3RhcnQ6IDAsXG4gICAgICAgICAgZW5kOiBNYXRoLm1pbihcbiAgICAgICAgICAgIFZJRVdQT1JUX0NPTlNUQU5UUy5QTEFDRUhPTERFUi5NQVhfU0FNUExFX1NJWkUgLSAxLFxuICAgICAgICAgICAgdG90YWxJdGVtcyAtIDFcbiAgICAgICAgICApLFxuICAgICAgICB9O1xuICAgICAgICBzaG93UGxhY2Vob2xkZXJzKGluaXRpYWxSYW5nZSk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIENsZWFudXAgZnVuY3Rpb25cbiAgICBjb25zdCBkZXN0cm95ID0gKCkgPT4ge1xuICAgICAgY2xlYXIoKTtcbiAgICB9O1xuXG4gICAgLy8gU3RvcmUgZnVuY3Rpb25zIGZvciB2aWV3cG9ydCB0byBjYWxsXG4gICAgKGNvbXBvbmVudCBhcyBhbnkpLl9wbGFjZWhvbGRlcnNJbml0aWFsaXplID0gaW5pdGlhbGl6ZTtcbiAgICAoY29tcG9uZW50IGFzIGFueSkuX3BsYWNlaG9sZGVyc0Rlc3Ryb3kgPSBkZXN0cm95O1xuXG4gICAgLy8gUmV0dXJuIGVuaGFuY2VkIGNvbXBvbmVudFxuICAgIHJldHVybiB7XG4gICAgICAuLi5jb21wb25lbnQsXG4gICAgICBwbGFjZWhvbGRlcnM6IHtcbiAgICAgICAgYW5hbHl6ZURhdGFTdHJ1Y3R1cmUsXG4gICAgICAgIGhhc0FuYWx5emVkU3RydWN0dXJlOiAoKSA9PiBoYXNBbmFseXplZCxcbiAgICAgICAgZ2VuZXJhdGVQbGFjZWhvbGRlckl0ZW0sXG4gICAgICAgIGdlbmVyYXRlUGxhY2Vob2xkZXJJdGVtcyxcbiAgICAgICAgc2hvd1BsYWNlaG9sZGVycyxcbiAgICAgICAgaXNQbGFjZWhvbGRlcixcbiAgICAgICAgcmVwbGFjZVBsYWNlaG9sZGVycyxcbiAgICAgICAgY2xlYXIsXG4gICAgICB9LFxuICAgIH07XG4gIH07XG59XG4iLAogICAgIi8qKlxuICogUmVuZGVyaW5nIEZlYXR1cmUgLSBJdGVtIHJlbmRlcmluZyBhbmQgcG9zaXRpb25pbmcgZm9yIHZpZXdwb3J0XG4gKiBIYW5kbGVzIERPTSBlbGVtZW50IGNyZWF0aW9uLCBwb3NpdGlvbmluZywgcmVjeWNsaW5nLCBhbmQgdXBkYXRlc1xuICovXG5cbmltcG9ydCB7IGFkZENsYXNzLCByZW1vdmVDbGFzcywgaGFzQ2xhc3MgfSBmcm9tIFwibXRybFwiO1xuaW1wb3J0IHR5cGUgeyBWaWV3cG9ydENvbnRleHQsIFZpZXdwb3J0Q29tcG9uZW50IH0gZnJvbSBcIi4uL3R5cGVzXCI7XG5pbXBvcnQgeyBWSUVXUE9SVF9DT05TVEFOVFMgfSBmcm9tIFwiLi4vY29uc3RhbnRzXCI7XG5pbXBvcnQge1xuICBpc1BsYWNlaG9sZGVyLFxuICBnZXRWaWV3cG9ydFN0YXRlLFxuICB3cmFwSW5pdGlhbGl6ZSxcbiAgd3JhcERlc3Ryb3ksXG59IGZyb20gXCIuL3V0aWxzXCI7XG5pbXBvcnQgeyBjcmVhdGVMYXlvdXQgfSBmcm9tIFwiLi4vLi4vbGF5b3V0XCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVuZGVyaW5nQ29uZmlnIHtcbiAgdGVtcGxhdGU/OiAoXG4gICAgaXRlbTogYW55LFxuICAgIGluZGV4OiBudW1iZXIsXG4gICkgPT4gc3RyaW5nIHwgSFRNTEVsZW1lbnQgfCBhbnlbXSB8IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIG92ZXJzY2FuPzogbnVtYmVyO1xuICBtZWFzdXJlSXRlbXM/OiBib29sZWFuO1xuICBlbmFibGVSZWN5Y2xpbmc/OiBib29sZWFuO1xuICBtYXhQb29sU2l6ZT86IG51bWJlcjtcbiAgLyoqXG4gICAqIE1haW50YWluIERPTSBvcmRlciB0byBtYXRjaCB2aXN1YWwgaW5kZXggb3JkZXIuXG4gICAqIEVuYWJsZXMgQ1NTIHNlbGVjdG9ycyBsaWtlIDpmaXJzdC1jaGlsZCwgOm50aC1jaGlsZCgpIHRvIHdvcmsgY29ycmVjdGx5LlxuICAgKiBEZWZhdWx0OiB0cnVlXG4gICAqL1xuICBtYWludGFpbkRvbU9yZGVyPzogYm9vbGVhbjtcbn1cblxuaW50ZXJmYWNlIFZpZXdwb3J0U3RhdGUge1xuICBzY3JvbGxQb3NpdGlvbjogbnVtYmVyO1xuICB0b3RhbEl0ZW1zOiBudW1iZXI7XG4gIGl0ZW1TaXplOiBudW1iZXI7XG4gIGNvbnRhaW5lclNpemU6IG51bWJlcjtcbiAgdmlydHVhbFRvdGFsU2l6ZTogbnVtYmVyO1xuICB2aXNpYmxlUmFuZ2U6IHsgc3RhcnQ6IG51bWJlcjsgZW5kOiBudW1iZXIgfTtcbiAgaXRlbXNDb250YWluZXI6IEhUTUxFbGVtZW50IHwgbnVsbDtcbn1cblxuLyoqXG4gKiBSZW5kZXJpbmcgZmVhdHVyZSBmb3Igdmlld3BvcnRcbiAqL1xuZXhwb3J0IGNvbnN0IHdpdGhSZW5kZXJpbmcgPSAoY29uZmlnOiBSZW5kZXJpbmdDb25maWcgPSB7fSkgPT4ge1xuICByZXR1cm4gPFQgZXh0ZW5kcyBWaWV3cG9ydENvbnRleHQgJiBWaWV3cG9ydENvbXBvbmVudD4oY29tcG9uZW50OiBUKTogVCA9PiB7XG4gICAgY29uc3Qge1xuICAgICAgdGVtcGxhdGUsXG4gICAgICBvdmVyc2NhbiA9IDUsXG4gICAgICBtZWFzdXJlSXRlbXMgPSBmYWxzZSxcbiAgICAgIGVuYWJsZVJlY3ljbGluZyA9IHRydWUsXG4gICAgICBtYXhQb29sU2l6ZSA9IFZJRVdQT1JUX0NPTlNUQU5UUy5SRU5ERVJJTkcuREVGQVVMVF9NQVhfUE9PTF9TSVpFLFxuICAgICAgbWFpbnRhaW5Eb21PcmRlciA9IHRydWUsXG4gICAgfSA9IGNvbmZpZztcblxuICAgIC8vIEhlbHBlciB0byBnZXQgaXRlbXMgZnJvbSBjb2xsZWN0aW9uIChzaW5nbGUgc291cmNlIG9mIHRydXRoKVxuICAgIGNvbnN0IGdldENvbGxlY3Rpb25JdGVtcyA9ICgpOiBhbnlbXSA9PiB7XG4gICAgICByZXR1cm4gKFxuICAgICAgICAoY29tcG9uZW50IGFzIGFueSkuY29sbGVjdGlvbj8uaXRlbXMgfHwgKGNvbXBvbmVudCBhcyBhbnkpLml0ZW1zIHx8IFtdXG4gICAgICApO1xuICAgIH07XG5cbiAgICAvLyBTdGF0ZVxuICAgIGNvbnN0IHJlbmRlcmVkRWxlbWVudHMgPSBuZXcgTWFwPG51bWJlciwgSFRNTEVsZW1lbnQ+KCk7XG4gICAgY29uc3QgZWxlbWVudFBvb2w6IEhUTUxFbGVtZW50W10gPSBbXTtcbiAgICBjb25zdCBwb29sU3RhdHMgPSB7IGNyZWF0ZWQ6IDAsIHJlY3ljbGVkOiAwLCBwb29sU2l6ZTogMCwgcmVsZWFzZWQ6IDAgfTtcblxuICAgIC8vIFN0b3JlIGxheW91dCByZXN1bHRzIGZvciBwcm9wZXIgY2xlYW51cCAocHJldmVudHMgbWVtb3J5IGxlYWspXG4gICAgY29uc3QgbGF5b3V0UmVzdWx0cyA9IG5ldyBXZWFrTWFwPEhUTUxFbGVtZW50LCB7IGRlc3Ryb3k6ICgpID0+IHZvaWQgfT4oKTtcblxuICAgIC8vIFJldXNhYmxlIHRlbXBsYXRlIGVsZW1lbnQgZm9yIEhUTUwgc3RyaW5nIHBhcnNpbmcgKG1vcmUgZWZmaWNpZW50IHRoYW4gZGl2KVxuICAgIGNvbnN0IHRlbXBsYXRlUGFyc2VyID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcInRlbXBsYXRlXCIpO1xuXG4gICAgbGV0IHZpZXdwb3J0U3RhdGU6IFZpZXdwb3J0U3RhdGUgfCBudWxsID0gbnVsbDtcbiAgICBsZXQgY3VycmVudFZpc2libGVSYW5nZSA9IHsgc3RhcnQ6IDAsIGVuZDogMCB9O1xuICAgIGxldCBsYXN0UmVuZGVyVGltZSA9IDA7XG4gICAgbGV0IGlzUmVtb3ZpbmdJdGVtID0gZmFsc2U7XG5cbiAgICAvLyBFbGVtZW50IHBvb2wgbWFuYWdlbWVudFxuICAgIGNvbnN0IGdldFBvb2xlZEVsZW1lbnQgPSAoKTogSFRNTEVsZW1lbnQgPT4ge1xuICAgICAgaWYgKGVuYWJsZVJlY3ljbGluZyAmJiBlbGVtZW50UG9vbC5sZW5ndGggPiAwKSB7XG4gICAgICAgIHBvb2xTdGF0cy5yZWN5Y2xlZCsrO1xuICAgICAgICByZXR1cm4gZWxlbWVudFBvb2wucG9wKCkhO1xuICAgICAgfVxuICAgICAgY29uc3QgZWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XG4gICAgICBlbGVtZW50LmNsYXNzTmFtZSA9IFwibXRybC12aWV3cG9ydC1pdGVtXCI7XG4gICAgICBwb29sU3RhdHMuY3JlYXRlZCsrO1xuICAgICAgcmV0dXJuIGVsZW1lbnQ7XG4gICAgfTtcblxuICAgIGNvbnN0IHJlbGVhc2VFbGVtZW50ID0gKGVsZW1lbnQ6IEhUTUxFbGVtZW50KTogdm9pZCA9PiB7XG4gICAgICBwb29sU3RhdHMucmVsZWFzZWQrKztcblxuICAgICAgLy8gQ1JJVElDQUw6IENhbGwgZGVzdHJveSBvbiBsYXlvdXRSZXN1bHQgdG8gY2xlYW4gdXAgY29tcG9uZW50cyBhbmQgZXZlbnQgbGlzdGVuZXJzXG4gICAgICAvLyBUaGlzIGZpeGVzIHRoZSBtZW1vcnkgbGVhayB3aGVuIHVzaW5nIGxheW91dCB0ZW1wbGF0ZXNcbiAgICAgIGNvbnN0IGxheW91dFJlc3VsdCA9IGxheW91dFJlc3VsdHMuZ2V0KGVsZW1lbnQpO1xuICAgICAgaWYgKGxheW91dFJlc3VsdCAmJiB0eXBlb2YgbGF5b3V0UmVzdWx0LmRlc3Ryb3kgPT09IFwiZnVuY3Rpb25cIikge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIGxheW91dFJlc3VsdC5kZXN0cm95KCk7XG4gICAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgICAvLyBJZ25vcmUgZGVzdHJveSBlcnJvcnMgLSBlbGVtZW50IG1heSBhbHJlYWR5IGJlIGNsZWFuZWQgdXBcbiAgICAgICAgfVxuICAgICAgICBsYXlvdXRSZXN1bHRzLmRlbGV0ZShlbGVtZW50KTtcbiAgICAgIH1cblxuICAgICAgaWYgKCFlbmFibGVSZWN5Y2xpbmcpIHtcbiAgICAgICAgZWxlbWVudC5yZW1vdmUoKTtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuICAgICAgLy8gQ2xlYW4gZWxlbWVudCBmb3IgcmV1c2UgKG5vIGNsb25pbmcgLSByZXVzZSB0aGUgYWN0dWFsIGVsZW1lbnQpXG4gICAgICBlbGVtZW50LmNsYXNzTmFtZSA9IFwibXRybC12aWV3cG9ydC1pdGVtXCI7XG4gICAgICBlbGVtZW50LnJlbW92ZUF0dHJpYnV0ZShcImRhdGEtaW5kZXhcIik7XG4gICAgICBlbGVtZW50LnN0eWxlLmNzc1RleHQgPSBcIlwiO1xuICAgICAgZWxlbWVudC5pbm5lckhUTUwgPSBcIlwiO1xuXG4gICAgICAvLyBSZW1vdmUgZnJvbSBET00gZmlyc3RcbiAgICAgIGlmIChlbGVtZW50LnBhcmVudE5vZGUpIHtcbiAgICAgICAgZWxlbWVudC5wYXJlbnROb2RlLnJlbW92ZUNoaWxkKGVsZW1lbnQpO1xuICAgICAgfVxuXG4gICAgICAvLyBBZGQgdG8gcG9vbCBpZiBub3QgZnVsbFxuICAgICAgaWYgKGVsZW1lbnRQb29sLmxlbmd0aCA8IG1heFBvb2xTaXplKSB7XG4gICAgICAgIGVsZW1lbnRQb29sLnB1c2goZWxlbWVudCk7XG4gICAgICAgIHBvb2xTdGF0cy5wb29sU2l6ZSA9IGVsZW1lbnRQb29sLmxlbmd0aDtcbiAgICAgIH1cbiAgICAgIC8vIElmIHBvb2wgaXMgZnVsbCwgZWxlbWVudCBpcyBqdXN0IGRlcmVmZXJlbmNlZCBhbmQgR0MnZFxuICAgIH07XG5cbiAgICAvLyBJbml0aWFsaXplXG4gICAgd3JhcEluaXRpYWxpemUoY29tcG9uZW50LCAoKSA9PiB7XG4gICAgICB2aWV3cG9ydFN0YXRlID0gZ2V0Vmlld3BvcnRTdGF0ZShjb21wb25lbnQpIGFzIFZpZXdwb3J0U3RhdGU7XG5cbiAgICAgIC8vIFNldCBpbml0aWFsIGl0ZW1zIGNvbnRhaW5lciBoZWlnaHQgZnJvbSBjdXJyZW50IHZpcnR1YWwgc2l6ZVxuICAgICAgaWYgKHZpZXdwb3J0U3RhdGU/Lml0ZW1zQ29udGFpbmVyICYmIHZpZXdwb3J0U3RhdGUudmlydHVhbFRvdGFsU2l6ZSA+IDApIHtcbiAgICAgICAgdmlld3BvcnRTdGF0ZS5pdGVtc0NvbnRhaW5lci5zdHlsZS5oZWlnaHQgPSBgJHt2aWV3cG9ydFN0YXRlLnZpcnR1YWxUb3RhbFNpemV9cHhgO1xuICAgICAgfVxuXG4gICAgICAvLyBMaXN0ZW4gZm9yIGl0ZW0gdXBkYXRlIHJlcXVlc3RzIGZyb20gQVBJXG4gICAgICBjb21wb25lbnQub24/LihcIml0ZW06dXBkYXRlLXJlcXVlc3RcIiwgKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICBjb25zdCB7IGluZGV4LCBpdGVtLCBwcmV2aW91c0l0ZW0gfSA9IGRhdGE7XG5cbiAgICAgICAgLy8gSXRlbSBpcyBhbHJlYWR5IHVwZGF0ZWQgaW4gY29sbGVjdGlvbi5pdGVtcyBieSBhcGkudHNcblxuICAgICAgICAvLyBDaGVjayBpZiB0aGlzIGl0ZW0gaXMgY3VycmVudGx5IHJlbmRlcmVkXG4gICAgICAgIGNvbnN0IGV4aXN0aW5nRWxlbWVudCA9IHJlbmRlcmVkRWxlbWVudHMuZ2V0KGluZGV4KTtcbiAgICAgICAgY29uc3Qgd2FzVmlzaWJsZSA9ICEhZXhpc3RpbmdFbGVtZW50O1xuXG4gICAgICAgIC8vIENoZWNrIGlmIGl0ZW0gd2FzIHNlbGVjdGVkIGJlZm9yZSB1cGRhdGVcbiAgICAgICAgY29uc3Qgd2FzU2VsZWN0ZWQgPSBleGlzdGluZ0VsZW1lbnRcbiAgICAgICAgICA/IGhhc0NsYXNzKFxuICAgICAgICAgICAgICBleGlzdGluZ0VsZW1lbnQsXG4gICAgICAgICAgICAgIFZJRVdQT1JUX0NPTlNUQU5UUy5TRUxFQ1RJT04uU0VMRUNURURfQ0xBU1MsXG4gICAgICAgICAgICApIHx8IGhhc0NsYXNzKGV4aXN0aW5nRWxlbWVudCwgXCJtdHJsLXZpZXdwb3J0LWl0ZW0tLXNlbGVjdGVkXCIpXG4gICAgICAgICAgOiBmYWxzZTtcblxuICAgICAgICBpZiAoZXhpc3RpbmdFbGVtZW50ICYmIGV4aXN0aW5nRWxlbWVudC5wYXJlbnROb2RlKSB7XG4gICAgICAgICAgLy8gUmUtcmVuZGVyIHRoZSBpdGVtXG4gICAgICAgICAgY29uc3QgbmV3RWxlbWVudCA9IHJlbmRlckl0ZW0oaXRlbSwgaW5kZXgpO1xuXG4gICAgICAgICAgaWYgKG5ld0VsZW1lbnQpIHtcbiAgICAgICAgICAgIC8vIENvcHkgcG9zaXRpb24gc3R5bGVzIGZyb20gZXhpc3RpbmcgZWxlbWVudFxuICAgICAgICAgICAgT2JqZWN0LmFzc2lnbihuZXdFbGVtZW50LnN0eWxlLCB7XG4gICAgICAgICAgICAgIHBvc2l0aW9uOiBleGlzdGluZ0VsZW1lbnQuc3R5bGUucG9zaXRpb24sXG4gICAgICAgICAgICAgIHRyYW5zZm9ybTogZXhpc3RpbmdFbGVtZW50LnN0eWxlLnRyYW5zZm9ybSxcbiAgICAgICAgICAgICAgd2lkdGg6IGV4aXN0aW5nRWxlbWVudC5zdHlsZS53aWR0aCxcbiAgICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgICAvLyBQcmVzZXJ2ZSBzZWxlY3RlZCBzdGF0ZVxuICAgICAgICAgICAgaWYgKHdhc1NlbGVjdGVkKSB7XG4gICAgICAgICAgICAgIGFkZENsYXNzKG5ld0VsZW1lbnQsIFZJRVdQT1JUX0NPTlNUQU5UUy5TRUxFQ1RJT04uU0VMRUNURURfQ0xBU1MpO1xuICAgICAgICAgICAgICBhZGRDbGFzcyhuZXdFbGVtZW50LCBcIm10cmwtdmlld3BvcnQtaXRlbS0tc2VsZWN0ZWRcIik7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIEFkZCB1cGRhdGUgYW5pbWF0aW9uIGNsYXNzIHRvIGlubmVyIGNvbnRlbnQgZm9yIHZpc2liaWxpdHlcbiAgICAgICAgICAgIGFkZENsYXNzKG5ld0VsZW1lbnQsIFwidmlld3BvcnQtaXRlbS0tdXBkYXRlZFwiKTtcbiAgICAgICAgICAgIC8vIEFsc28gdHJ5IHRvIGFkZCB0byBmaXJzdCBjaGlsZCBpZiBpdCBleGlzdHMgKHRoZSBhY3R1YWwgaXRlbSBjb250ZW50KVxuICAgICAgICAgICAgY29uc3QgaW5uZXJJdGVtID0gbmV3RWxlbWVudC5maXJzdEVsZW1lbnRDaGlsZCBhcyBIVE1MRWxlbWVudDtcbiAgICAgICAgICAgIGlmIChpbm5lckl0ZW0pIHtcbiAgICAgICAgICAgICAgYWRkQ2xhc3MoaW5uZXJJdGVtLCBcIml0ZW0tLXVwZGF0ZWRcIik7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIFJlcGxhY2UgaW4gRE9NXG4gICAgICAgICAgICBleGlzdGluZ0VsZW1lbnQucGFyZW50Tm9kZS5yZXBsYWNlQ2hpbGQoXG4gICAgICAgICAgICAgIG5ld0VsZW1lbnQsXG4gICAgICAgICAgICAgIGV4aXN0aW5nRWxlbWVudCxcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgICByZW5kZXJlZEVsZW1lbnRzLnNldChpbmRleCwgbmV3RWxlbWVudCk7XG4gICAgICAgICAgICByZWxlYXNlRWxlbWVudChleGlzdGluZ0VsZW1lbnQpO1xuXG4gICAgICAgICAgICAvLyBSZW1vdmUgdGhlIGFuaW1hdGlvbiBjbGFzcyBhZnRlciB0cmFuc2l0aW9uXG4gICAgICAgICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICAgICAgcmVtb3ZlQ2xhc3MobmV3RWxlbWVudCwgXCJ2aWV3cG9ydC1pdGVtLS11cGRhdGVkXCIpO1xuICAgICAgICAgICAgICBpZiAoaW5uZXJJdGVtKSB7XG4gICAgICAgICAgICAgICAgcmVtb3ZlQ2xhc3MoaW5uZXJJdGVtLCBcIml0ZW0tLXVwZGF0ZWRcIik7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0sIDUwMCk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gRW1pdCBjb21wbGV0aW9uIGV2ZW50XG4gICAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJpdGVtOnVwZGF0ZWRcIiwge1xuICAgICAgICAgIGl0ZW0sXG4gICAgICAgICAgaW5kZXgsXG4gICAgICAgICAgcHJldmlvdXNJdGVtLFxuICAgICAgICAgIHdhc1Zpc2libGUsXG4gICAgICAgICAgd2FzU2VsZWN0ZWQsXG4gICAgICAgIH0pO1xuICAgICAgfSk7XG5cbiAgICAgIC8vIExpc3RlbiBmb3IgaXRlbSByZW1vdmUgcmVxdWVzdHMgZnJvbSBBUElcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwiaXRlbTpyZW1vdmUtcmVxdWVzdFwiLCAoZGF0YTogYW55KSA9PiB7XG4gICAgICAgIGNvbnN0IHsgaW5kZXgsIGl0ZW0gfSA9IGRhdGE7XG4gICAgICAgIGlzUmVtb3ZpbmdJdGVtID0gdHJ1ZTtcblxuICAgICAgICAvLyBJdGVtIGlzIGFscmVhZHkgcmVtb3ZlZCBmcm9tIGNvbGxlY3Rpb24uaXRlbXMgYnkgYXBpLnRzXG4gICAgICAgIC8vIFdlIGp1c3QgbmVlZCB0byB1cGRhdGUgcmVuZGVyZWQgZWxlbWVudHNcblxuICAgICAgICAvLyBSZW1vdmUgdGhlIHJlbmRlcmVkIGVsZW1lbnQgYXQgdGhpcyBpbmRleFxuICAgICAgICBjb25zdCBleGlzdGluZ0VsZW1lbnQgPSByZW5kZXJlZEVsZW1lbnRzLmdldChpbmRleCk7XG4gICAgICAgIGlmIChleGlzdGluZ0VsZW1lbnQgJiYgZXhpc3RpbmdFbGVtZW50LnBhcmVudE5vZGUpIHtcbiAgICAgICAgICByZWxlYXNlRWxlbWVudChleGlzdGluZ0VsZW1lbnQpO1xuICAgICAgICB9XG4gICAgICAgIHJlbmRlcmVkRWxlbWVudHMuZGVsZXRlKGluZGV4KTtcblxuICAgICAgICAvLyBTaGlmdCByZW5kZXJlZCBlbGVtZW50cyBpbmRpY2VzIGRvd25cbiAgICAgICAgY29uc3QgcmVuZGVyZWRLZXlzID0gQXJyYXkuZnJvbShyZW5kZXJlZEVsZW1lbnRzLmtleXMoKSkuc29ydChcbiAgICAgICAgICAoYSwgYikgPT4gYSAtIGIsXG4gICAgICAgICk7XG4gICAgICAgIGNvbnN0IG5ld1JlbmRlcmVkRWxlbWVudHMgPSBuZXcgTWFwPG51bWJlciwgSFRNTEVsZW1lbnQ+KCk7XG4gICAgICAgIGZvciAoY29uc3Qga2V5IG9mIHJlbmRlcmVkS2V5cykge1xuICAgICAgICAgIGlmIChrZXkgPiBpbmRleCkge1xuICAgICAgICAgICAgY29uc3QgZWxlbWVudCA9IHJlbmRlcmVkRWxlbWVudHMuZ2V0KGtleSk7XG4gICAgICAgICAgICBpZiAoZWxlbWVudCkge1xuICAgICAgICAgICAgICAvLyBVcGRhdGUgZGF0YS1pbmRleCBhdHRyaWJ1dGVcbiAgICAgICAgICAgICAgZWxlbWVudC5kYXRhc2V0LmluZGV4ID0gU3RyaW5nKGtleSAtIDEpO1xuICAgICAgICAgICAgICBuZXdSZW5kZXJlZEVsZW1lbnRzLnNldChrZXkgLSAxLCBlbGVtZW50KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9IGVsc2UgaWYgKGtleSA8IGluZGV4KSB7XG4gICAgICAgICAgICBjb25zdCBlbGVtZW50ID0gcmVuZGVyZWRFbGVtZW50cy5nZXQoa2V5KTtcbiAgICAgICAgICAgIGlmIChlbGVtZW50KSB7XG4gICAgICAgICAgICAgIG5ld1JlbmRlcmVkRWxlbWVudHMuc2V0KGtleSwgZWxlbWVudCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJlbmRlcmVkRWxlbWVudHMuY2xlYXIoKTtcbiAgICAgICAgZm9yIChjb25zdCBba2V5LCBlbGVtZW50XSBvZiBuZXdSZW5kZXJlZEVsZW1lbnRzKSB7XG4gICAgICAgICAgcmVuZGVyZWRFbGVtZW50cy5zZXQoa2V5LCBlbGVtZW50KTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIERlY3JlbWVudCB0b3RhbEl0ZW1zIE5PVyBzbyByZW5kZXJJdGVtcygpIHVzZXMgdGhlIGNvcnJlY3QgY291bnRcbiAgICAgICAgLy8gc2V0VG90YWxJdGVtcygpIHdpbGwgYmUgY2FsbGVkIGxhdGVyIGJ5IHRoZSBBUEksIHdoaWNoIHdpbGwgZW1pdFxuICAgICAgICAvLyB2aWV3cG9ydDp0b3RhbC1pdGVtcy1jaGFuZ2VkIGZvciB2aXJ0dWFsIHNpemUgcmVjYWxjdWxhdGlvblxuICAgICAgICBpZiAodmlld3BvcnRTdGF0ZSAmJiB2aWV3cG9ydFN0YXRlLnRvdGFsSXRlbXMgPiAwKSB7XG4gICAgICAgICAgdmlld3BvcnRTdGF0ZS50b3RhbEl0ZW1zLS07XG4gICAgICAgIH1cblxuICAgICAgICAvLyBSZXNldCB2aXNpYmxlIHJhbmdlIHRvIGZvcmNlIHJlLXJlbmRlclxuICAgICAgICBjdXJyZW50VmlzaWJsZVJhbmdlID0geyBzdGFydDogLTEsIGVuZDogLTEgfTtcblxuICAgICAgICAvLyBFbWl0IGNvbXBsZXRpb24gZXZlbnRcbiAgICAgICAgY29tcG9uZW50LmVtaXQ/LihcIml0ZW06cmVtb3ZlZFwiLCB7XG4gICAgICAgICAgaXRlbSxcbiAgICAgICAgICBpbmRleCxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gTk9URTogRG8gTk9UIGNsZWFyIGxvYWRlZFJhbmdlcyBvciBjYWxsIGxvYWRNaXNzaW5nUmFuZ2VzIGhlcmUhXG4gICAgICAgIC8vIFRoZSBpdGVtcyBhcnJheSBoYXMgYmVlbiBzcGxpY2VkIGluLXBsYWNlIGJ5IGFwaS50cywgc28gdGhlIGRhdGEgaXNcbiAgICAgICAgLy8gYWxyZWFkeSBpbiBtZW1vcnkuIFdoZW4gdGhlIHVzZXIgc2Nyb2xscyBhbmQgdGhlIHZpc2libGUgcmFuZ2UgY2hhbmdlcyxcbiAgICAgICAgLy8gdGhlIG5vcm1hbCB2aWV3cG9ydDpyYW5nZS1jaGFuZ2VkIGhhbmRsZXIgd2lsbCBmZXRjaCBtaXNzaW5nIGRhdGEgaWYgbmVlZGVkLlxuICAgICAgICAvLyBDbGVhcmluZyBsb2FkZWRSYW5nZXMgaGVyZSB3b3VsZCB0cmlnZ2VyIGFuIHVubmVjZXNzYXJ5IGZldGNoIG9uIGV2ZXJ5IHJlbW92YWwuXG5cbiAgICAgICAgLy8gVHJpZ2dlciByZS1yZW5kZXIgKHVzZXMgZGF0YSBhbHJlYWR5IGluIG1lbW9yeSlcbiAgICAgICAgcmVuZGVySXRlbXMoKTtcbiAgICAgICAgaXNSZW1vdmluZ0l0ZW0gPSBmYWxzZTtcbiAgICAgIH0pO1xuXG4gICAgICAvLyBMaXN0ZW4gZm9yIGl0ZW1zIGFkZCByZXF1ZXN0cyBmcm9tIEFQSVxuICAgICAgY29tcG9uZW50Lm9uPy4oXCJpdGVtczphZGQtcmVxdWVzdFwiLCAoZGF0YTogYW55KSA9PiB7XG4gICAgICAgIGNvbnN0IHsgaXRlbXM6IG5ld0l0ZW1zLCBwb3NpdGlvbiwgcHJldmlvdXNDb3VudCB9ID0gZGF0YTtcblxuICAgICAgICBpZiAoIW5ld0l0ZW1zIHx8IG5ld0l0ZW1zLmxlbmd0aCA9PT0gMCkgcmV0dXJuO1xuXG4gICAgICAgIC8vIEl0ZW1zIGFyZSBhbHJlYWR5IGFkZGVkIHRvIGNvbGxlY3Rpb24uaXRlbXMgYnkgYXBpLnRzXG4gICAgICAgIC8vIFdlIGp1c3QgbmVlZCB0byB1cGRhdGUgcmVuZGVyZWQgZWxlbWVudHMgaW5kaWNlcyBpZiBwcmVwZW5kaW5nXG5cbiAgICAgICAgY29uc3QgaXRlbXNBZGRlZCA9IG5ld0l0ZW1zLmxlbmd0aDtcblxuICAgICAgICBpZiAocG9zaXRpb24gPT09IFwic3RhcnRcIikge1xuICAgICAgICAgIC8vIEl0ZW1zIHdlcmUgcHJlcGVuZGVkIC0gc2hpZnQgcmVuZGVyZWQgZWxlbWVudCBpbmRpY2VzXG4gICAgICAgICAgY29uc3QgcmVuZGVyZWRLZXlzID0gQXJyYXkuZnJvbShyZW5kZXJlZEVsZW1lbnRzLmtleXMoKSkuc29ydChcbiAgICAgICAgICAgIChhLCBiKSA9PiBiIC0gYSxcbiAgICAgICAgICApO1xuICAgICAgICAgIGNvbnN0IG5ld1JlbmRlcmVkRWxlbWVudHMgPSBuZXcgTWFwPG51bWJlciwgSFRNTEVsZW1lbnQ+KCk7XG4gICAgICAgICAgZm9yIChjb25zdCBrZXkgb2YgcmVuZGVyZWRLZXlzKSB7XG4gICAgICAgICAgICBjb25zdCBlbGVtZW50ID0gcmVuZGVyZWRFbGVtZW50cy5nZXQoa2V5KSE7XG4gICAgICAgICAgICBjb25zdCBuZXdJbmRleCA9IGtleSArIGl0ZW1zQWRkZWQ7XG4gICAgICAgICAgICBlbGVtZW50LnNldEF0dHJpYnV0ZShcImRhdGEtaW5kZXhcIiwgU3RyaW5nKG5ld0luZGV4KSk7XG4gICAgICAgICAgICBuZXdSZW5kZXJlZEVsZW1lbnRzLnNldChuZXdJbmRleCwgZWxlbWVudCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJlbmRlcmVkRWxlbWVudHMuY2xlYXIoKTtcbiAgICAgICAgICBuZXdSZW5kZXJlZEVsZW1lbnRzLmZvckVhY2goKGVsLCBpZHgpID0+XG4gICAgICAgICAgICByZW5kZXJlZEVsZW1lbnRzLnNldChpZHgsIGVsKSxcbiAgICAgICAgICApO1xuICAgICAgICB9XG4gICAgICAgIC8vIEZvciBhcHBlbmQsIG5vIGluZGV4IHNoaWZ0aW5nIG5lZWRlZFxuXG4gICAgICAgIC8vIFVwZGF0ZSB0b3RhbEl0ZW1zIGluIHZpZXdwb3J0U3RhdGVcbiAgICAgICAgaWYgKHZpZXdwb3J0U3RhdGUpIHtcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlLnRvdGFsSXRlbXMgPVxuICAgICAgICAgICAgKHZpZXdwb3J0U3RhdGUudG90YWxJdGVtcyB8fCAwKSArIGl0ZW1zQWRkZWQ7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBSZXNldCB2aXNpYmxlIHJhbmdlIHRvIGZvcmNlIHJlLXJlbmRlclxuICAgICAgICBjdXJyZW50VmlzaWJsZVJhbmdlID0geyBzdGFydDogLTEsIGVuZDogLTEgfTtcblxuICAgICAgICAvLyBDbGVhciBsb2FkZWRSYW5nZXMgdG8gZW5zdXJlIHByb3BlciByZWxvYWQgdHJhY2tpbmdcbiAgICAgICAgY29uc3QgY29sbGVjdGlvbiA9IChjb21wb25lbnQudmlld3BvcnQgYXMgYW55KT8uY29sbGVjdGlvbjtcbiAgICAgICAgaWYgKGNvbGxlY3Rpb24/LmdldExvYWRlZFJhbmdlcykge1xuICAgICAgICAgIGNvbnN0IGxvYWRlZFJhbmdlcyA9IGNvbGxlY3Rpb24uZ2V0TG9hZGVkUmFuZ2VzKCk7XG4gICAgICAgICAgbG9hZGVkUmFuZ2VzLmNsZWFyKCk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBUcmlnZ2VyIHJlLXJlbmRlclxuICAgICAgICByZW5kZXJJdGVtcygpO1xuXG4gICAgICAgIC8vIEVtaXQgY29tcGxldGlvbiBldmVudFxuICAgICAgICBjb21wb25lbnQuZW1pdD8uKFwiaXRlbXM6cmVuZGVyLWNvbXBsZXRlXCIsIHtcbiAgICAgICAgICBpdGVtczogbmV3SXRlbXMsXG4gICAgICAgICAgcG9zaXRpb24sXG4gICAgICAgICAgdG90YWw6IHZpZXdwb3J0U3RhdGU/LnRvdGFsSXRlbXMgfHwgMCxcbiAgICAgICAgfSk7XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciBjb2xsZWN0aW9uIGl0ZW1zIGV2aWN0ZWQgLSBjbGVhbiB1cCByZW5kZXJlZCBlbGVtZW50c1xuICAgICAgY29tcG9uZW50Lm9uPy4oXCJjb2xsZWN0aW9uOml0ZW1zLWV2aWN0ZWRcIiwgKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICBjb25zdCB7IGtlZXBTdGFydCwga2VlcEVuZCB9ID0gZGF0YTtcblxuICAgICAgICAvLyBSZWxlYXNlIHJlbmRlcmVkIGVsZW1lbnRzIG91dHNpZGUgdGhlIGtlZXAgcmFuZ2VcbiAgICAgICAgcmVuZGVyZWRFbGVtZW50cy5mb3JFYWNoKChlbGVtZW50LCBpbmRleCkgPT4ge1xuICAgICAgICAgIGlmIChpbmRleCA8IGtlZXBTdGFydCB8fCBpbmRleCA+IGtlZXBFbmQpIHtcbiAgICAgICAgICAgIHJlbGVhc2VFbGVtZW50KGVsZW1lbnQpO1xuICAgICAgICAgICAgcmVuZGVyZWRFbGVtZW50cy5kZWxldGUoaW5kZXgpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSk7XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciByZWxvYWQ6c3RhcnQgdG8gcmVzZXQgZmVhdHVyZS1zcGVjaWZpYyBzdGF0ZVxuICAgICAgY29tcG9uZW50Lm9uPy4oXCJyZWxvYWQ6c3RhcnRcIiwgKCkgPT4ge1xuICAgICAgICAvLyBSZXNldCB2aXNpYmxlIHJhbmdlIHRyYWNraW5nXG4gICAgICAgIGN1cnJlbnRWaXNpYmxlUmFuZ2UgPSB7IHN0YXJ0OiAtMSwgZW5kOiAtMSB9O1xuICAgICAgICBsYXN0UmVuZGVyVGltZSA9IDA7XG4gICAgICAgIGlzUmVtb3ZpbmdJdGVtID0gZmFsc2U7XG5cbiAgICAgICAgLy8gQ2xlYXIgZWxlbWVudCBwb29sIHRvIGZyZWUgbWVtb3J5XG4gICAgICAgIGVsZW1lbnRQb29sLmxlbmd0aCA9IDA7XG4gICAgICAgIHBvb2xTdGF0cy5wb29sU2l6ZSA9IDA7XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciBjb2xsZWN0aW9uIHJlc2V0IHRvIGNsZWFyIHJlbmRlcmluZyBzdGF0ZVxuICAgICAgY29tcG9uZW50Lm9uPy4oXCJjb2xsZWN0aW9uOnJlc2V0XCIsICgpID0+IHtcbiAgICAgICAgLy8gUmVsZWFzZSBhbmQgY2xlYXIgYWxsIHJlbmRlcmVkIGVsZW1lbnRzXG4gICAgICAgIHJlbmRlcmVkRWxlbWVudHMuZm9yRWFjaCgoZWxlbWVudCkgPT4ge1xuICAgICAgICAgIHJlbGVhc2VFbGVtZW50KGVsZW1lbnQpO1xuICAgICAgICB9KTtcbiAgICAgICAgcmVuZGVyZWRFbGVtZW50cy5jbGVhcigpO1xuXG4gICAgICAgIC8vIFJlc2V0IHZpc2libGUgcmFuZ2UgdHJhY2tpbmdcbiAgICAgICAgY3VycmVudFZpc2libGVSYW5nZSA9IHsgc3RhcnQ6IC0xLCBlbmQ6IC0xIH07XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciBwb29sIGNsZWFyIHJlcXVlc3QgKG9uIHJlbG9hZClcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwidmlld3BvcnQ6Y2xlYXItcG9vbFwiLCAoKSA9PiB7XG4gICAgICAgIC8vIENsZWFyIGVsZW1lbnQgcG9vbCB0byBmcmVlIG1lbW9yeVxuICAgICAgICBlbGVtZW50UG9vbC5sZW5ndGggPSAwO1xuICAgICAgICBwb29sU3RhdHMucG9vbFNpemUgPSAwO1xuXG4gICAgICAgIC8vIFJlc2V0IGxhc3QgcmVuZGVyIHRpbWUgdG8gYXZvaWQgc3RhbGUgY2FsY3VsYXRpb25zXG4gICAgICAgIGxhc3RSZW5kZXJUaW1lID0gMDtcbiAgICAgIH0pO1xuXG4gICAgICAvLyBMaXN0ZW4gZm9yIGNvbGxlY3Rpb24gZGF0YSBsb2FkZWRcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwiY29sbGVjdGlvbjpyYW5nZS1sb2FkZWRcIiwgKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICBpZiAoIWRhdGEuaXRlbXM/Lmxlbmd0aCkgcmV0dXJuO1xuXG4gICAgICAgIC8vIEFuYWx5emUgZGF0YSBzdHJ1Y3R1cmUgb24gZmlyc3QgbG9hZFxuICAgICAgICBjb25zdCBwbGFjZWhvbGRlcnMgPSAoY29tcG9uZW50IGFzIGFueSkucGxhY2Vob2xkZXJzO1xuICAgICAgICBpZiAocGxhY2Vob2xkZXJzICYmICFwbGFjZWhvbGRlcnMuaGFzQW5hbHl6ZWRTdHJ1Y3R1cmUoKSkge1xuICAgICAgICAgIHBsYWNlaG9sZGVycy5hbmFseXplRGF0YVN0cnVjdHVyZShkYXRhLml0ZW1zKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIEl0ZW1zIGFyZSBhbHJlYWR5IGluIGNvbGxlY3Rpb24uaXRlbXMgLSByZXBsYWNlIGFueSBwbGFjZWhvbGRlciBET00gZWxlbWVudHNcbiAgICAgICAgZGF0YS5pdGVtcy5mb3JFYWNoKChpdGVtOiBhbnksIGk6IG51bWJlcikgPT4ge1xuICAgICAgICAgIGNvbnN0IGluZGV4ID0gZGF0YS5vZmZzZXQgKyBpO1xuICAgICAgICAgIGNvbnN0IGVsZW1lbnQgPSByZW5kZXJlZEVsZW1lbnRzLmdldChpbmRleCk7XG5cbiAgICAgICAgICAvLyBDaGVjayBpZiBjdXJyZW50IHJlbmRlcmVkIGVsZW1lbnQgaXMgYSBwbGFjZWhvbGRlciB0aGF0IG5lZWRzIHJlcGxhY2luZ1xuICAgICAgICAgIC8vIE5vdGU6IFRoZSBhY3R1YWwgY2xhc3MgaGFzICdtdHJsLScgcHJlZml4LCBzbyBjaGVjayBmb3IgYm90aFxuICAgICAgICAgIGNvbnN0IGlzUGxhY2Vob2xkZXJFbGVtZW50ID1cbiAgICAgICAgICAgIGVsZW1lbnQ/LmNsYXNzTGlzdC5jb250YWlucyhWSUVXUE9SVF9DT05TVEFOVFMuUExBQ0VIT0xERVIuQ0xBU1MpIHx8XG4gICAgICAgICAgICBlbGVtZW50Py5jbGFzc0xpc3QuY29udGFpbnMoXG4gICAgICAgICAgICAgIGBtdHJsLSR7VklFV1BPUlRfQ09OU1RBTlRTLlBMQUNFSE9MREVSLkNMQVNTfWAsXG4gICAgICAgICAgICApO1xuICAgICAgICAgIGlmIChpc1BsYWNlaG9sZGVyRWxlbWVudCkge1xuICAgICAgICAgICAgY29uc3QgbmV3RWxlbWVudCA9IHJlbmRlckl0ZW0oaXRlbSwgaW5kZXgpO1xuICAgICAgICAgICAgaWYgKG5ld0VsZW1lbnQpIHtcbiAgICAgICAgICAgICAgLy8gUmVtb3ZlIHBsYWNlaG9sZGVyIGNsYXNzZXMgZnJvbSB3cmFwcGVyXG4gICAgICAgICAgICAgIHJlbW92ZUNsYXNzKG5ld0VsZW1lbnQsIFZJRVdQT1JUX0NPTlNUQU5UUy5QTEFDRUhPTERFUi5DTEFTUyk7XG4gICAgICAgICAgICAgIC8vIEFsc28gcmVtb3ZlIGZyb20gaW5uZXIgZWxlbWVudCAoZm9yIHN0cmluZyB0ZW1wbGF0ZXMpXG4gICAgICAgICAgICAgIGlmIChuZXdFbGVtZW50LmZpcnN0RWxlbWVudENoaWxkKSB7XG4gICAgICAgICAgICAgICAgcmVtb3ZlQ2xhc3MoXG4gICAgICAgICAgICAgICAgICBuZXdFbGVtZW50LmZpcnN0RWxlbWVudENoaWxkIGFzIEhUTUxFbGVtZW50LFxuICAgICAgICAgICAgICAgICAgVklFV1BPUlRfQ09OU1RBTlRTLlBMQUNFSE9MREVSLkNMQVNTLFxuICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAvLyBBZGQgcmVwbGFjZWQgY2xhc3MgZm9yIGZhZGUtaW4gYW5pbWF0aW9uXG4gICAgICAgICAgICAgIGFkZENsYXNzKG5ld0VsZW1lbnQsIFwidmlld3BvcnQtaXRlbS0tcmVwbGFjZWRcIik7XG5cbiAgICAgICAgICAgICAgLy8gQ29weSBwb3NpdGlvbiBhbmQgcmVwbGFjZVxuICAgICAgICAgICAgICBpZiAoZWxlbWVudCkge1xuICAgICAgICAgICAgICAgIE9iamVjdC5hc3NpZ24obmV3RWxlbWVudC5zdHlsZSwge1xuICAgICAgICAgICAgICAgICAgcG9zaXRpb246IGVsZW1lbnQuc3R5bGUucG9zaXRpb24sXG4gICAgICAgICAgICAgICAgICB0cmFuc2Zvcm06IGVsZW1lbnQuc3R5bGUudHJhbnNmb3JtLFxuICAgICAgICAgICAgICAgICAgd2lkdGg6IGVsZW1lbnQuc3R5bGUud2lkdGgsXG4gICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgICAgZWxlbWVudC5wYXJlbnROb2RlPy5yZXBsYWNlQ2hpbGQobmV3RWxlbWVudCwgZWxlbWVudCk7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgcmVuZGVyZWRFbGVtZW50cy5zZXQoaW5kZXgsIG5ld0VsZW1lbnQpO1xuICAgICAgICAgICAgICBpZiAoZWxlbWVudCkge1xuICAgICAgICAgICAgICAgIHJlbGVhc2VFbGVtZW50KGVsZW1lbnQpO1xuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgLy8gUmVtb3ZlIHRoZSByZXBsYWNlZCBjbGFzcyBhZnRlciBhbmltYXRpb24gY29tcGxldGVzXG4gICAgICAgICAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICAgICAgICAgIHJlbW92ZUNsYXNzKG5ld0VsZW1lbnQsIFwidmlld3BvcnQtaXRlbS0tcmVwbGFjZWRcIik7XG4gICAgICAgICAgICAgIH0sIDMwMCk7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAvLyByZW5kZXJJdGVtIHJldHVybmVkIG51bGwgLSBzdGlsbCByZWxlYXNlIHRoZSBvbGQgZWxlbWVudFxuICAgICAgICAgICAgICBpZiAoZWxlbWVudCkge1xuICAgICAgICAgICAgICAgIHJlbGVhc2VFbGVtZW50KGVsZW1lbnQpO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIHJlbmRlcmVkRWxlbWVudHMuZGVsZXRlKGluZGV4KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH0pO1xuXG4gICAgICAgIC8vIENoZWNrIGlmIHdlIG5lZWQgdG8gcmVuZGVyIG5ldyBpdGVtcyAoaXRlbXMgdGhhdCB3ZXJlbid0IHJlbmRlcmVkIGF0IGFsbClcbiAgICAgICAgY29uc3QgeyB2aXNpYmxlUmFuZ2UgfSA9IHZpZXdwb3J0U3RhdGUgfHwge307XG4gICAgICAgIGlmICh2aXNpYmxlUmFuZ2UpIHtcbiAgICAgICAgICBjb25zdCByZW5kZXJTdGFydCA9IE1hdGgubWF4KDAsIHZpc2libGVSYW5nZS5zdGFydCAtIG92ZXJzY2FuKTtcbiAgICAgICAgICBjb25zdCByZW5kZXJFbmQgPSBNYXRoLm1pbihcbiAgICAgICAgICAgIHZpZXdwb3J0U3RhdGU/LnRvdGFsSXRlbXMgPz8gMCAtIDEsXG4gICAgICAgICAgICB2aXNpYmxlUmFuZ2UuZW5kICsgb3ZlcnNjYW4sXG4gICAgICAgICAgKTtcbiAgICAgICAgICBjb25zdCBsb2FkZWRTdGFydCA9IGRhdGEub2Zmc2V0O1xuICAgICAgICAgIGNvbnN0IGxvYWRlZEVuZCA9IGRhdGEub2Zmc2V0ICsgZGF0YS5pdGVtcy5sZW5ndGggLSAxO1xuXG4gICAgICAgICAgLy8gQ2hlY2sgaWYgbG9hZGVkIGRhdGEgb3ZlcmxhcHMgd2l0aCByZW5kZXIgcmFuZ2UgYW5kIHdlJ3JlIG1pc3NpbmcgcmVuZGVyZWQgZWxlbWVudHNcbiAgICAgICAgICBjb25zdCBuZWVkc1JlbmRlciA9XG4gICAgICAgICAgICBsb2FkZWRTdGFydCA8PSByZW5kZXJFbmQgJiZcbiAgICAgICAgICAgIGxvYWRlZEVuZCA+PSByZW5kZXJTdGFydCAmJlxuICAgICAgICAgICAgcmVuZGVyZWRFbGVtZW50cy5zaXplIDwgcmVuZGVyRW5kIC0gcmVuZGVyU3RhcnQgKyAxO1xuXG4gICAgICAgICAgaWYgKG5lZWRzUmVuZGVyKSByZW5kZXJJdGVtcygpO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgLy8gTGlzdGVuIGZvciBldmVudHNcbiAgICAgIGNvbXBvbmVudC5vbj8uKFwidmlld3BvcnQ6cmFuZ2UtY2hhbmdlZFwiLCByZW5kZXJJdGVtcyk7XG4gICAgICBjb21wb25lbnQub24/LihcInZpZXdwb3J0OnNjcm9sbFwiLCB1cGRhdGVJdGVtUG9zaXRpb25zKTtcblxuICAgICAgLy8gVXBkYXRlIGl0ZW1zIGNvbnRhaW5lciBoZWlnaHQgd2hlbiB2aXJ0dWFsIHNpemUgY2hhbmdlc1xuICAgICAgY29tcG9uZW50Lm9uPy4oXCJ2aWV3cG9ydDp2aXJ0dWFsLXNpemUtY2hhbmdlZFwiLCAoZGF0YTogYW55KSA9PiB7XG4gICAgICAgIGlmIChcbiAgICAgICAgICB2aWV3cG9ydFN0YXRlPy5pdGVtc0NvbnRhaW5lciAmJlxuICAgICAgICAgIGRhdGEudG90YWxWaXJ0dWFsU2l6ZSAhPT0gdW5kZWZpbmVkXG4gICAgICAgICkge1xuICAgICAgICAgIHZpZXdwb3J0U3RhdGUuaXRlbXNDb250YWluZXIuc3R5bGUuaGVpZ2h0ID0gYCR7ZGF0YS50b3RhbFZpcnR1YWxTaXplfXB4YDtcbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICAvLyBUZW1wbGF0ZSBoZWxwZXJzXG4gICAgY29uc3QgZ2V0RGVmYXVsdFRlbXBsYXRlID0gKCkgPT4gKGl0ZW06IGFueSwgaW5kZXg6IG51bWJlcikgPT4ge1xuICAgICAgLy8gVXNlIGxheW91dCBzeXN0ZW0gZm9yIGRlZmF1bHQgdGVtcGxhdGVcbiAgICAgIHJldHVybiBbXG4gICAgICAgIHtcbiAgICAgICAgICB0YWc6IFwiZGl2XCIsXG4gICAgICAgICAgY2xhc3M6IFwidmlld3BvcnQtaXRlbVwiLFxuICAgICAgICAgIHRleHQ6XG4gICAgICAgICAgICB0eXBlb2YgaXRlbSA9PT0gXCJvYmplY3RcIlxuICAgICAgICAgICAgICA/IGl0ZW0ubmFtZSB8fCBpdGVtLmxhYmVsIHx8IGl0ZW0udGV4dCB8fCBgSXRlbSAke2luZGV4fWBcbiAgICAgICAgICAgICAgOiBTdHJpbmcoaXRlbSksXG4gICAgICAgIH0sXG4gICAgICBdO1xuICAgIH07XG5cbiAgICAvLyBQcm9jZXNzIGxheW91dCBzY2hlbWEgd2l0aCBpdGVtIGRhdGEgc3Vic3RpdHV0aW9uXG4gICAgY29uc3QgcHJvY2Vzc0xheW91dFNjaGVtYSA9IChcbiAgICAgIHNjaGVtYTogYW55LFxuICAgICAgaXRlbTogYW55LFxuICAgICAgaW5kZXg6IG51bWJlcixcbiAgICApOiBhbnkgPT4ge1xuICAgICAgaWYgKHR5cGVvZiBzY2hlbWEgPT09IFwic3RyaW5nXCIpIHtcbiAgICAgICAgLy8gSGFuZGxlIHZhcmlhYmxlIHN1YnN0aXR1dGlvbiBsaWtlIHt7bmFtZX19LCB7e2luZGV4fX1cbiAgICAgICAgcmV0dXJuIHNjaGVtYS5yZXBsYWNlKC9cXHtcXHsoW159XSspXFx9XFx9L2csIChtYXRjaCwga2V5KSA9PiB7XG4gICAgICAgICAgaWYgKGtleSA9PT0gXCJpbmRleFwiKSByZXR1cm4gU3RyaW5nKGluZGV4KTtcbiAgICAgICAgICBpZiAoa2V5ID09PSBcIml0ZW1cIikgcmV0dXJuIFN0cmluZyhpdGVtKTtcblxuICAgICAgICAgIC8vIEhhbmRsZSBuZXN0ZWQgcHJvcGVydGllcyBsaWtlIHt7dXNlci5uYW1lfX1cbiAgICAgICAgICBjb25zdCB2YWx1ZSA9IGtleS5zcGxpdChcIi5cIikucmVkdWNlKChvYmo6IGFueSwgcHJvcDogc3RyaW5nKSA9PiB7XG4gICAgICAgICAgICByZXR1cm4gb2JqPy5bcHJvcC50cmltKCldO1xuICAgICAgICAgIH0sIGl0ZW0pO1xuXG4gICAgICAgICAgcmV0dXJuIHZhbHVlICE9PSB1bmRlZmluZWQgPyBTdHJpbmcodmFsdWUpIDogbWF0Y2g7XG4gICAgICAgIH0pO1xuICAgICAgfVxuXG4gICAgICBpZiAoQXJyYXkuaXNBcnJheShzY2hlbWEpKSB7XG4gICAgICAgIHJldHVybiBzY2hlbWEubWFwKChjaGlsZCkgPT4gcHJvY2Vzc0xheW91dFNjaGVtYShjaGlsZCwgaXRlbSwgaW5kZXgpKTtcbiAgICAgIH1cblxuICAgICAgaWYgKHR5cGVvZiBzY2hlbWEgPT09IFwib2JqZWN0XCIgJiYgc2NoZW1hICE9PSBudWxsKSB7XG4gICAgICAgIGNvbnN0IHByb2Nlc3NlZDogYW55ID0ge307XG4gICAgICAgIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKHNjaGVtYSkpIHtcbiAgICAgICAgICBwcm9jZXNzZWRba2V5XSA9IHByb2Nlc3NMYXlvdXRTY2hlbWEodmFsdWUsIGl0ZW0sIGluZGV4KTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gcHJvY2Vzc2VkO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gc2NoZW1hO1xuICAgIH07XG5cbiAgICAvLyBQb3NpdGlvbiBjYWxjdWxhdGlvblxuICAgIGNvbnN0IGNhbGN1bGF0ZUl0ZW1Qb3NpdGlvbiA9IChcbiAgICAgIGluZGV4OiBudW1iZXIsXG4gICAgICBzY3JvbGxQb3NpdGlvbjogbnVtYmVyLFxuICAgICAgdG90YWxJdGVtczogbnVtYmVyLFxuICAgICAgaXRlbVNpemU6IG51bWJlcixcbiAgICAgIHZpcnR1YWxUb3RhbFNpemU6IG51bWJlcixcbiAgICAgIGNvbnRhaW5lclNpemU6IG51bWJlcixcbiAgICAgIHRhcmdldFNjcm9sbEluZGV4PzogbnVtYmVyLFxuICAgICk6IG51bWJlciA9PiB7XG4gICAgICBjb25zdCBhY3R1YWxUb3RhbFNpemUgPSB0b3RhbEl0ZW1zICogaXRlbVNpemU7XG4gICAgICBjb25zdCBpc0NvbXByZXNzZWQgPVxuICAgICAgICBhY3R1YWxUb3RhbFNpemUgPiB2aXJ0dWFsVG90YWxTaXplICYmIHZpcnR1YWxUb3RhbFNpemUgPiAwO1xuXG4gICAgICAvLyBJZiB3ZSBoYXZlIGEgdGFyZ2V0U2Nyb2xsSW5kZXggKGZyb20gaW5pdGlhbFNjcm9sbEluZGV4KSwgdXNlIGl0IGRpcmVjdGx5XG4gICAgICAvLyBUaGlzIGVuc3VyZXMgaXRlbXMgYXJlIHBvc2l0aW9uZWQgY29ycmVjdGx5IGV2ZW4gd2l0aCBjb21wcmVzc2lvblxuICAgICAgaWYgKHRhcmdldFNjcm9sbEluZGV4ICE9PSB1bmRlZmluZWQgJiYgdGFyZ2V0U2Nyb2xsSW5kZXggPiAwKSB7XG4gICAgICAgIC8vIFBvc2l0aW9uIGl0ZW1zIHJlbGF0aXZlIHRvIHRoZSB0YXJnZXQgaW5kZXhcbiAgICAgICAgLy8gVGhlIHRhcmdldCBpbmRleCBzaG91bGQgYXBwZWFyIGF0IHRoZSB0b3Agb2YgdGhlIHZpZXdwb3J0XG4gICAgICAgIHJldHVybiAoaW5kZXggLSB0YXJnZXRTY3JvbGxJbmRleCkgKiBpdGVtU2l6ZTtcbiAgICAgIH1cblxuICAgICAgaWYgKCFpc0NvbXByZXNzZWQgfHwgdG90YWxJdGVtcyA9PT0gMCkge1xuICAgICAgICByZXR1cm4gaW5kZXggKiBpdGVtU2l6ZSAtIHNjcm9sbFBvc2l0aW9uO1xuICAgICAgfVxuXG4gICAgICAvLyBDb21wcmVzc2VkIHNwYWNlIGhhbmRsaW5nXG4gICAgICBjb25zdCBtYXhTY3JvbGxQb3NpdGlvbiA9IHZpcnR1YWxUb3RhbFNpemUgLSBjb250YWluZXJTaXplO1xuICAgICAgY29uc3QgZGlzdGFuY2VGcm9tQm90dG9tID0gbWF4U2Nyb2xsUG9zaXRpb24gLSBzY3JvbGxQb3NpdGlvbjtcbiAgICAgIGNvbnN0IG5lYXJCb3R0b21UaHJlc2hvbGQgPSBjb250YWluZXJTaXplO1xuXG4gICAgICBpZiAoXG4gICAgICAgIGRpc3RhbmNlRnJvbUJvdHRvbSA8PSBuZWFyQm90dG9tVGhyZXNob2xkICYmXG4gICAgICAgIGRpc3RhbmNlRnJvbUJvdHRvbSA+PSAtMVxuICAgICAgKSB7XG4gICAgICAgIC8vIE5lYXIgYm90dG9tIGludGVycG9sYXRpb25cbiAgICAgICAgY29uc3QgaXRlbXNBdEJvdHRvbSA9IE1hdGguZmxvb3IoY29udGFpbmVyU2l6ZSAvIGl0ZW1TaXplKTtcbiAgICAgICAgY29uc3QgZmlyc3RWaXNpYmxlQXRCb3R0b20gPSBNYXRoLm1heCgwLCB0b3RhbEl0ZW1zIC0gaXRlbXNBdEJvdHRvbSk7XG4gICAgICAgIGNvbnN0IHNjcm9sbFJhdGlvID0gc2Nyb2xsUG9zaXRpb24gLyB2aXJ0dWFsVG90YWxTaXplO1xuICAgICAgICBjb25zdCBleGFjdFNjcm9sbEluZGV4ID0gc2Nyb2xsUmF0aW8gKiB0b3RhbEl0ZW1zO1xuICAgICAgICBjb25zdCBpbnRlcnBvbGF0aW9uID0gTWF0aC5tYXgoXG4gICAgICAgICAgMCxcbiAgICAgICAgICBNYXRoLm1pbigxLCAxIC0gZGlzdGFuY2VGcm9tQm90dG9tIC8gbmVhckJvdHRvbVRocmVzaG9sZCksXG4gICAgICAgICk7XG5cbiAgICAgICAgY29uc3QgYm90dG9tUG9zaXRpb24gPSAoaW5kZXggLSBmaXJzdFZpc2libGVBdEJvdHRvbSkgKiBpdGVtU2l6ZTtcbiAgICAgICAgY29uc3Qgbm9ybWFsUG9zaXRpb24gPSAoaW5kZXggLSBleGFjdFNjcm9sbEluZGV4KSAqIGl0ZW1TaXplO1xuICAgICAgICByZXR1cm4gKFxuICAgICAgICAgIG5vcm1hbFBvc2l0aW9uICsgKGJvdHRvbVBvc2l0aW9uIC0gbm9ybWFsUG9zaXRpb24pICogaW50ZXJwb2xhdGlvblxuICAgICAgICApO1xuICAgICAgfVxuXG4gICAgICAvLyBOb3JtYWwgY29tcHJlc3NlZCBzY3JvbGxpbmdcbiAgICAgIGNvbnN0IHNjcm9sbFJhdGlvID0gc2Nyb2xsUG9zaXRpb24gLyB2aXJ0dWFsVG90YWxTaXplO1xuICAgICAgcmV0dXJuIChpbmRleCAtIHNjcm9sbFJhdGlvICogdG90YWxJdGVtcykgKiBpdGVtU2l6ZTtcbiAgICB9O1xuXG4gICAgLy8gUmVuZGVyIHNpbmdsZSBpdGVtXG4gICAgY29uc3QgcmVuZGVySXRlbSA9IChpdGVtOiBhbnksIGluZGV4OiBudW1iZXIpOiBIVE1MRWxlbWVudCB8IG51bGwgPT4ge1xuICAgICAgY29uc3QgaXRlbVRlbXBsYXRlID0gdGVtcGxhdGUgfHwgZ2V0RGVmYXVsdFRlbXBsYXRlKCk7XG5cbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IGl0ZW1UZW1wbGF0ZShpdGVtLCBpbmRleCk7XG4gICAgICAgIGxldCBlbGVtZW50OiBIVE1MRWxlbWVudDtcblxuICAgICAgICAvLyBDaGVjayBpZiByZXN1bHQgaXMgYSBsYXlvdXQgc2NoZW1hIChhcnJheSBvciBvYmplY3QpXG4gICAgICAgIGlmIChcbiAgICAgICAgICBBcnJheS5pc0FycmF5KHJlc3VsdCkgfHxcbiAgICAgICAgICAodHlwZW9mIHJlc3VsdCA9PT0gXCJvYmplY3RcIiAmJlxuICAgICAgICAgICAgcmVzdWx0ICE9PSBudWxsICYmXG4gICAgICAgICAgICAhKHJlc3VsdCBpbnN0YW5jZW9mIEhUTUxFbGVtZW50KSlcbiAgICAgICAgKSB7XG4gICAgICAgICAgLy8gUHJvY2VzcyBzY2hlbWEgdG8gc3Vic3RpdHV0ZSB2YXJpYWJsZXNcbiAgICAgICAgICBjb25zdCBwcm9jZXNzZWRTY2hlbWEgPSBwcm9jZXNzTGF5b3V0U2NoZW1hKHJlc3VsdCwgaXRlbSwgaW5kZXgpO1xuXG4gICAgICAgICAgLy8gVXNlIGxheW91dCBzeXN0ZW0gdG8gY3JlYXRlIGVsZW1lbnRcbiAgICAgICAgICBjb25zdCBsYXlvdXRSZXN1bHQgPSBjcmVhdGVMYXlvdXQocHJvY2Vzc2VkU2NoZW1hKTtcbiAgICAgICAgICBlbGVtZW50ID0gbGF5b3V0UmVzdWx0LmVsZW1lbnQ7XG5cbiAgICAgICAgICAvLyBJZiB0aGUgbGF5b3V0IGNyZWF0ZWQgYSB3cmFwcGVyLCB1c2UgaXQgZGlyZWN0bHlcbiAgICAgICAgICBpZiAoZWxlbWVudCAmJiBlbGVtZW50Lm5vZGVUeXBlID09PSAxKSB7XG4gICAgICAgICAgICAvLyBFbGVtZW50IGlzIGFscmVhZHkgY3JlYXRlZCBieSBsYXlvdXQgc3lzdGVtXG4gICAgICAgICAgICAvLyBTdG9yZSBsYXlvdXRSZXN1bHQgZm9yIGNsZWFudXAgd2hlbiBlbGVtZW50IGlzIHJlbGVhc2VkIChwcmV2ZW50cyBtZW1vcnkgbGVhaylcbiAgICAgICAgICAgIGxheW91dFJlc3VsdHMuc2V0KGVsZW1lbnQsIGxheW91dFJlc3VsdCk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIC8vIEZhbGxiYWNrIGlmIGxheW91dCBkaWRuJ3QgY3JlYXRlIGEgcHJvcGVyIGVsZW1lbnRcbiAgICAgICAgICAgIGVsZW1lbnQgPSBnZXRQb29sZWRFbGVtZW50KCk7XG4gICAgICAgICAgICBpZiAobGF5b3V0UmVzdWx0LmVsZW1lbnQpIHtcbiAgICAgICAgICAgICAgZWxlbWVudC5hcHBlbmRDaGlsZChsYXlvdXRSZXN1bHQuZWxlbWVudCk7XG4gICAgICAgICAgICAgIC8vIFN0b3JlIGxheW91dFJlc3VsdCBvbiB3cmFwcGVyIGVsZW1lbnQgZm9yIGNsZWFudXBcbiAgICAgICAgICAgICAgbGF5b3V0UmVzdWx0cy5zZXQoZWxlbWVudCwgbGF5b3V0UmVzdWx0KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH0gZWxzZSBpZiAodHlwZW9mIHJlc3VsdCA9PT0gXCJzdHJpbmdcIikge1xuICAgICAgICAgIC8vIFBhcnNlIEhUTUwgdXNpbmcgdGVtcGxhdGUgZWxlbWVudCAtIG9wdGltaXplZCBwYXRoXG4gICAgICAgICAgLy8gTW92ZSBub2RlcyBkaXJlY3RseSBpbnN0ZWFkIG9mIGNsb25pbmcgLSBtdWNoIGZhc3Rlciwgbm8gbWVtb3J5IG92ZXJoZWFkXG4gICAgICAgICAgdGVtcGxhdGVQYXJzZXIuaW5uZXJIVE1MID0gcmVzdWx0O1xuICAgICAgICAgIGNvbnN0IGNvbnRlbnQgPSB0ZW1wbGF0ZVBhcnNlci5jb250ZW50O1xuXG4gICAgICAgICAgaWYgKGNvbnRlbnQuY2hpbGRyZW4ubGVuZ3RoID09PSAxKSB7XG4gICAgICAgICAgICAvLyBTaW5nbGUgcm9vdCBlbGVtZW50IC0gbW92ZSBkaXJlY3RseSAoZmFzdGVyIHRoYW4gY2xvbmVOb2RlKVxuICAgICAgICAgICAgZWxlbWVudCA9IGNvbnRlbnQuZmlyc3RFbGVtZW50Q2hpbGQgYXMgSFRNTEVsZW1lbnQ7XG4gICAgICAgICAgICBjb250ZW50LnJlbW92ZUNoaWxkKGVsZW1lbnQpO1xuICAgICAgICAgICAgYWRkQ2xhc3MoZWxlbWVudCwgXCJtdHJsLXZpZXdwb3J0LWl0ZW1cIik7XG4gICAgICAgICAgICBpZiAoaXNQbGFjZWhvbGRlcihpdGVtKSkge1xuICAgICAgICAgICAgICBhZGRDbGFzcyhlbGVtZW50LCBWSUVXUE9SVF9DT05TVEFOVFMuUExBQ0VIT0xERVIuQ0xBU1MpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyBNdWx0aXBsZSBjaGlsZHJlbiAtIG1vdmUgYWxsIGludG8gcG9vbGVkIGVsZW1lbnRcbiAgICAgICAgICAgIGVsZW1lbnQgPSBnZXRQb29sZWRFbGVtZW50KCk7XG4gICAgICAgICAgICB3aGlsZSAoY29udGVudC5maXJzdENoaWxkKSB7XG4gICAgICAgICAgICAgIGVsZW1lbnQuYXBwZW5kQ2hpbGQoY29udGVudC5maXJzdENoaWxkKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH0gZWxzZSBpZiAocmVzdWx0IGluc3RhbmNlb2YgSFRNTEVsZW1lbnQpIHtcbiAgICAgICAgICBlbGVtZW50ID0gZ2V0UG9vbGVkRWxlbWVudCgpO1xuICAgICAgICAgIGVsZW1lbnQuYXBwZW5kQ2hpbGQocmVzdWx0KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zb2xlLndhcm4oYFtSZW5kZXJpbmddIEludmFsaWQgdGVtcGxhdGUgcmVzdWx0IGZvciBpdGVtICR7aW5kZXh9YCk7XG4gICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBBZGQgY2xhc3Nlc1xuICAgICAgICBpZiAoIWhhc0NsYXNzKGVsZW1lbnQsIFwidmlld3BvcnQtaXRlbVwiKSlcbiAgICAgICAgICBhZGRDbGFzcyhlbGVtZW50LCBcInZpZXdwb3J0LWl0ZW1cIik7XG4gICAgICAgIGlmIChpc1BsYWNlaG9sZGVyKGl0ZW0pKSB7XG4gICAgICAgICAgYWRkQ2xhc3MoZWxlbWVudCwgVklFV1BPUlRfQ09OU1RBTlRTLlBMQUNFSE9MREVSLkNMQVNTKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGVsZW1lbnQuZGF0YXNldC5pbmRleCA9IFN0cmluZyhpbmRleCk7XG5cbiAgICAgICAgLy8gQ29weSBkYXRhLWlkIGZyb20gaXRlbSB0byB2aWV3cG9ydCBlbGVtZW50IGZvciBzZWxlY3Rpb24gc3VwcG9ydFxuICAgICAgICBjb25zdCBpdGVtSWQgPSBpdGVtPy5faWQgfHwgaXRlbT8uaWQ7XG4gICAgICAgIGlmIChpdGVtSWQgIT09IHVuZGVmaW5lZCAmJiBpdGVtSWQgIT09IG51bGwpIHtcbiAgICAgICAgICBlbGVtZW50LmRhdGFzZXQuaWQgPSBTdHJpbmcoaXRlbUlkKTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gZWxlbWVudDtcbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoYFtSZW5kZXJpbmddIEVycm9yIHJlbmRlcmluZyBpdGVtICR7aW5kZXh9OmAsIGVycm9yKTtcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIFVwZGF0ZSBwb3NpdGlvbnNcbiAgICBjb25zdCB1cGRhdGVJdGVtUG9zaXRpb25zID0gKCk6IHZvaWQgPT4ge1xuICAgICAgaWYgKCF2aWV3cG9ydFN0YXRlIHx8IHJlbmRlcmVkRWxlbWVudHMuc2l6ZSA9PT0gMCkgcmV0dXJuO1xuXG4gICAgICBjb25zdCB7XG4gICAgICAgIHNjcm9sbFBvc2l0aW9uLFxuICAgICAgICBpdGVtU2l6ZTogaXRlbVNpemUsXG4gICAgICAgIHRvdGFsSXRlbXMsXG4gICAgICAgIHZpcnR1YWxUb3RhbFNpemUsXG4gICAgICAgIGNvbnRhaW5lclNpemUsXG4gICAgICB9ID0gdmlld3BvcnRTdGF0ZTtcbiAgICAgIGNvbnN0IGFjdHVhbFRvdGFsU2l6ZSA9IHRvdGFsSXRlbXMgKiBpdGVtU2l6ZTtcbiAgICAgIGNvbnN0IGlzQ29tcHJlc3NlZCA9XG4gICAgICAgIGFjdHVhbFRvdGFsU2l6ZSA+IHZpcnR1YWxUb3RhbFNpemUgJiYgdmlydHVhbFRvdGFsU2l6ZSA+IDA7XG5cbiAgICAgIGNvbnN0IHNvcnRlZEluZGljZXMgPSBBcnJheS5mcm9tKHJlbmRlcmVkRWxlbWVudHMua2V5cygpKS5zb3J0KFxuICAgICAgICAoYSwgYikgPT4gYSAtIGIsXG4gICAgICApO1xuICAgICAgaWYgKCFzb3J0ZWRJbmRpY2VzLmxlbmd0aCkgcmV0dXJuO1xuXG4gICAgICBjb25zdCBmaXJzdEluZGV4ID0gc29ydGVkSW5kaWNlc1swXTtcbiAgICAgIGxldCBjdXJyZW50UG9zaXRpb24gPSAwO1xuXG4gICAgICBpZiAoaXNDb21wcmVzc2VkKSB7XG4gICAgICAgIGNvbnN0IG1heFNjcm9sbCA9IHZpcnR1YWxUb3RhbFNpemUgLSBjb250YWluZXJTaXplO1xuICAgICAgICBjb25zdCBkaXN0YW5jZUZyb21Cb3R0b20gPSBtYXhTY3JvbGwgLSBzY3JvbGxQb3NpdGlvbjtcblxuICAgICAgICBpZiAoZGlzdGFuY2VGcm9tQm90dG9tIDw9IGNvbnRhaW5lclNpemUgJiYgZGlzdGFuY2VGcm9tQm90dG9tID49IC0xKSB7XG4gICAgICAgICAgLy8gTmVhciBib3R0b20gaW50ZXJwb2xhdGlvblxuICAgICAgICAgIGNvbnN0IGl0ZW1zQXRCb3R0b20gPSBNYXRoLmZsb29yKGNvbnRhaW5lclNpemUgLyBpdGVtU2l6ZSk7XG4gICAgICAgICAgY29uc3QgZmlyc3RWaXNpYmxlQXRCb3R0b20gPSBNYXRoLm1heCgwLCB0b3RhbEl0ZW1zIC0gaXRlbXNBdEJvdHRvbSk7XG4gICAgICAgICAgY29uc3Qgc2Nyb2xsUmF0aW8gPSBzY3JvbGxQb3NpdGlvbiAvIHZpcnR1YWxUb3RhbFNpemU7XG4gICAgICAgICAgY29uc3QgZXhhY3RTY3JvbGxJbmRleCA9IHNjcm9sbFJhdGlvICogdG90YWxJdGVtcztcbiAgICAgICAgICBjb25zdCBpbnRlcnBvbGF0aW9uID0gTWF0aC5tYXgoXG4gICAgICAgICAgICAwLFxuICAgICAgICAgICAgTWF0aC5taW4oMSwgMSAtIGRpc3RhbmNlRnJvbUJvdHRvbSAvIGNvbnRhaW5lclNpemUpLFxuICAgICAgICAgICk7XG5cbiAgICAgICAgICBjb25zdCBib3R0b21Qb3MgPSAoZmlyc3RJbmRleCAtIGZpcnN0VmlzaWJsZUF0Qm90dG9tKSAqIGl0ZW1TaXplO1xuICAgICAgICAgIGNvbnN0IG5vcm1hbFBvcyA9IChmaXJzdEluZGV4IC0gZXhhY3RTY3JvbGxJbmRleCkgKiBpdGVtU2l6ZTtcbiAgICAgICAgICBjdXJyZW50UG9zaXRpb24gPSBub3JtYWxQb3MgKyAoYm90dG9tUG9zIC0gbm9ybWFsUG9zKSAqIGludGVycG9sYXRpb247XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgY29uc3Qgc2Nyb2xsUmF0aW8gPSBzY3JvbGxQb3NpdGlvbiAvIHZpcnR1YWxUb3RhbFNpemU7XG4gICAgICAgICAgY3VycmVudFBvc2l0aW9uID0gKGZpcnN0SW5kZXggLSBzY3JvbGxSYXRpbyAqIHRvdGFsSXRlbXMpICogaXRlbVNpemU7XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGN1cnJlbnRQb3NpdGlvbiA9IGZpcnN0SW5kZXggKiBpdGVtU2l6ZSAtIHNjcm9sbFBvc2l0aW9uO1xuICAgICAgfVxuXG4gICAgICAvLyBQb3NpdGlvbiBlYWNoIGl0ZW1cbiAgICAgIHNvcnRlZEluZGljZXMuZm9yRWFjaCgoaW5kZXgpID0+IHtcbiAgICAgICAgY29uc3QgZWxlbWVudCA9IHJlbmRlcmVkRWxlbWVudHMuZ2V0KGluZGV4KTtcbiAgICAgICAgaWYgKGVsZW1lbnQpIHtcbiAgICAgICAgICBlbGVtZW50LnN0eWxlLnRyYW5zZm9ybSA9IGB0cmFuc2xhdGVZKCR7TWF0aC5yb3VuZChcbiAgICAgICAgICAgIGN1cnJlbnRQb3NpdGlvbixcbiAgICAgICAgICApfXB4KWA7XG5cbiAgICAgICAgICAvLyBTdHJhdGVnaWMgbG9nIGZvciBsYXN0IGl0ZW1zXG4gICAgICAgICAgLy8gaWYgKGluZGV4ID49IHRvdGFsSXRlbXMgLSA1KSB7XG4gICAgICAgICAgLy8gICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAvLyAgICAgYFtSZW5kZXJpbmddIExhc3QgaXRlbSBwb3NpdGlvbmVkOiBpbmRleD0ke2luZGV4fSwgcG9zaXRpb249JHtNYXRoLnJvdW5kKFxuICAgICAgICAgIC8vICAgICAgIGN1cnJlbnRQb3NpdGlvblxuICAgICAgICAgIC8vICAgICApfXB4LCBpdGVtU2l6ZT0ke2l0ZW1TaXplfXB4LCB0b3RhbEl0ZW1zPSR7dG90YWxJdGVtc31gXG4gICAgICAgICAgLy8gICApO1xuICAgICAgICAgIC8vIH1cblxuICAgICAgICAgIGN1cnJlbnRQb3NpdGlvbiArPSBpdGVtU2l6ZTtcbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgfTtcblxuICAgIC8vIE1haW4gcmVuZGVyIGZ1bmN0aW9uXG4gICAgY29uc3QgcmVuZGVySXRlbXMgPSAoKSA9PiB7XG4gICAgICBpZiAoIXZpZXdwb3J0U3RhdGU/Lml0ZW1zQ29udGFpbmVyKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cblxuICAgICAgY29uc3Qge1xuICAgICAgICB2aXNpYmxlUmFuZ2UsXG4gICAgICAgIGl0ZW1zQ29udGFpbmVyLFxuICAgICAgICB0b3RhbEl0ZW1zLFxuICAgICAgICBpdGVtU2l6ZSxcbiAgICAgICAgc2Nyb2xsUG9zaXRpb24sXG4gICAgICAgIGNvbnRhaW5lclNpemUsXG4gICAgICAgIHZpcnR1YWxUb3RhbFNpemUsXG4gICAgICB9ID0gdmlld3BvcnRTdGF0ZTtcblxuICAgICAgLy8gVmFsaWRhdGUgcmFuZ2UgLSBza2lwIHJlbmRlcmluZyBpZiBubyBpdGVtcyBvciByYW5nZSBpcyBpbnZhbGlkXG4gICAgICBpZiAoXG4gICAgICAgICF2aXNpYmxlUmFuZ2UgfHxcbiAgICAgICAgdG90YWxJdGVtcyA8PSAwIHx8XG4gICAgICAgIHZpc2libGVSYW5nZS5zdGFydCA8IDAgfHxcbiAgICAgICAgdmlzaWJsZVJhbmdlLnN0YXJ0ID49IHRvdGFsSXRlbXMgfHxcbiAgICAgICAgdmlzaWJsZVJhbmdlLmVuZCA8IHZpc2libGVSYW5nZS5zdGFydCB8fFxuICAgICAgICBpc05hTih2aXNpYmxlUmFuZ2Uuc3RhcnQpIHx8XG4gICAgICAgIGlzTmFOKHZpc2libGVSYW5nZS5lbmQpXG4gICAgICApIHtcbiAgICAgICAgLy8gSWYgdG90YWxJdGVtcyBpcyAwLCBjbGVhciBhbGwgcmVuZGVyZWQgZWxlbWVudHMgdG8gcmVtb3ZlIHN0YWxlIHBsYWNlaG9sZGVyc1xuICAgICAgICBpZiAodG90YWxJdGVtcyA8PSAwICYmIHJlbmRlcmVkRWxlbWVudHMuc2l6ZSA+IDApIHtcbiAgICAgICAgICBBcnJheS5mcm9tKHJlbmRlcmVkRWxlbWVudHMuZW50cmllcygpKS5mb3JFYWNoKChbaW5kZXgsIGVsZW1lbnRdKSA9PiB7XG4gICAgICAgICAgICBpZiAoZWxlbWVudC5wYXJlbnROb2RlKSByZWxlYXNlRWxlbWVudChlbGVtZW50KTtcbiAgICAgICAgICAgIHJlbmRlcmVkRWxlbWVudHMuZGVsZXRlKGluZGV4KTtcbiAgICAgICAgICB9KTtcbiAgICAgICAgICBjdXJyZW50VmlzaWJsZVJhbmdlID0geyBzdGFydDogLTEsIGVuZDogLTEgfTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm47XG4gICAgICB9XG5cbiAgICAgIC8vIENoZWNrIGlmIHJhbmdlIGNoYW5nZWRcbiAgICAgIGlmIChcbiAgICAgICAgdmlzaWJsZVJhbmdlLnN0YXJ0ID09PSBjdXJyZW50VmlzaWJsZVJhbmdlLnN0YXJ0ICYmXG4gICAgICAgIHZpc2libGVSYW5nZS5lbmQgPT09IGN1cnJlbnRWaXNpYmxlUmFuZ2UuZW5kICYmXG4gICAgICAgIHJlbmRlcmVkRWxlbWVudHMuc2l6ZSA+IDBcbiAgICAgICkge1xuICAgICAgICB1cGRhdGVJdGVtUG9zaXRpb25zKCk7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cblxuICAgICAgbGFzdFJlbmRlclRpbWUgPSBEYXRlLm5vdygpO1xuICAgICAgY29uc3QgcmVuZGVyU3RhcnQgPSBNYXRoLm1heCgwLCB2aXNpYmxlUmFuZ2Uuc3RhcnQgLSBvdmVyc2Nhbik7XG4gICAgICBjb25zdCByZW5kZXJFbmQgPSBNYXRoLm1pbih0b3RhbEl0ZW1zIC0gMSwgdmlzaWJsZVJhbmdlLmVuZCArIG92ZXJzY2FuKTtcblxuICAgICAgLy8gaWYgKGlzUmVtb3ZpbmdJdGVtKSB7XG4gICAgICAvLyAgIGNvbnNvbGUubG9nKFxuICAgICAgLy8gICAgIGBbUkVOREVSLUZJWF0gcmVuZGVySXRlbXMgY2FsYzogdmlzaWJsZVJhbmdlPSR7SlNPTi5zdHJpbmdpZnkodmlzaWJsZVJhbmdlKX0sIG92ZXJzY2FuPSR7b3ZlcnNjYW59LCB0b3RhbEl0ZW1zPSR7dG90YWxJdGVtc31gLFxuICAgICAgLy8gICApO1xuICAgICAgLy8gICBjb25zb2xlLmxvZyhcbiAgICAgIC8vICAgICBgW1JFTkRFUi1GSVhdIHJlbmRlckl0ZW1zIGNhbGM6IHJlbmRlclN0YXJ0PSR7cmVuZGVyU3RhcnR9LCByZW5kZXJFbmQ9JHtyZW5kZXJFbmR9YCxcbiAgICAgIC8vICAgKTtcbiAgICAgIC8vIH1cblxuICAgICAgLy8gUmVtb3ZlIGl0ZW1zIG91dHNpZGUgcmFuZ2VcbiAgICAgIGNvbnN0IHRvUmVtb3ZlID0gQXJyYXkuZnJvbShyZW5kZXJlZEVsZW1lbnRzLmVudHJpZXMoKSkuZmlsdGVyKFxuICAgICAgICAoW2luZGV4XSkgPT4gaW5kZXggPCByZW5kZXJTdGFydCB8fCBpbmRleCA+IHJlbmRlckVuZCxcbiAgICAgICk7XG5cbiAgICAgIHRvUmVtb3ZlLmZvckVhY2goKFtpbmRleCwgZWxlbWVudF0pID0+IHtcbiAgICAgICAgaWYgKGVsZW1lbnQucGFyZW50Tm9kZSkgcmVsZWFzZUVsZW1lbnQoZWxlbWVudCk7XG4gICAgICAgIHJlbmRlcmVkRWxlbWVudHMuZGVsZXRlKGluZGV4KTtcbiAgICAgIH0pO1xuXG4gICAgICAvLyBHZXQgaXRlbXMgZnJvbSBjb2xsZWN0aW9uIChzaW5nbGUgc291cmNlIG9mIHRydXRoKVxuICAgICAgY29uc3QgaXRlbXMgPSBnZXRDb2xsZWN0aW9uSXRlbXMoKTtcbiAgICAgIGNvbnN0IG1pc3NpbmdJdGVtczogbnVtYmVyW10gPSBbXTtcblxuICAgICAgLy8gQ29sbGVjdCBuZXcgZWxlbWVudHMgdG8gaW5zZXJ0IChpbmRleCAtPiBlbGVtZW50KVxuICAgICAgY29uc3QgbmV3RWxlbWVudHM6IEFycmF5PFtudW1iZXIsIEhUTUxFbGVtZW50XT4gPSBbXTtcblxuICAgICAgLy8gUmVuZGVyIGl0ZW1zIGluIHJhbmdlXG4gICAgICBmb3IgKGxldCBpID0gcmVuZGVyU3RhcnQ7IGkgPD0gcmVuZGVyRW5kOyBpKyspIHtcbiAgICAgICAgaWYgKGkgPCAwIHx8IGkgPj0gdG90YWxJdGVtcyB8fCByZW5kZXJlZEVsZW1lbnRzLmhhcyhpKSkgY29udGludWU7XG5cbiAgICAgICAgbGV0IGl0ZW0gPSBpdGVtc1tpXTtcbiAgICAgICAgaWYgKCFpdGVtKSB7XG4gICAgICAgICAgbWlzc2luZ0l0ZW1zLnB1c2goaSk7XG4gICAgICAgICAgLy8gR2VuZXJhdGUgcGxhY2Vob2xkZXIgKHJlbmRlcmVkIGR5bmFtaWNhbGx5LCBub3Qgc3RvcmVkKVxuICAgICAgICAgIGNvbnN0IHBsYWNlaG9sZGVycyA9IChjb21wb25lbnQgYXMgYW55KS5wbGFjZWhvbGRlcnM7XG4gICAgICAgICAgaXRlbSA9IHBsYWNlaG9sZGVycz8uZ2VuZXJhdGVQbGFjZWhvbGRlckl0ZW0oaSkgfHwge1xuICAgICAgICAgICAgX3BsYWNlaG9sZGVyOiB0cnVlLFxuICAgICAgICAgICAgaW5kZXg6IGksXG4gICAgICAgICAgICBpZDogYHBsYWNlaG9sZGVyLSR7aX1gLFxuICAgICAgICAgICAgbmFtZTogVklFV1BPUlRfQ09OU1RBTlRTLlBMQUNFSE9MREVSLk1BU0tfQ0hBUkFDVEVSLnJlcGVhdCgxNSksXG4gICAgICAgICAgICB0ZXh0OiBWSUVXUE9SVF9DT05TVEFOVFMuUExBQ0VIT0xERVIuTUFTS19DSEFSQUNURVIucmVwZWF0KDI1KSxcbiAgICAgICAgICAgIGRlc2NyaXB0aW9uOlxuICAgICAgICAgICAgICBWSUVXUE9SVF9DT05TVEFOVFMuUExBQ0VIT0xERVIuTUFTS19DSEFSQUNURVIucmVwZWF0KDQwKSxcbiAgICAgICAgICB9O1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgZWxlbWVudCA9IHJlbmRlckl0ZW0oaXRlbSwgaSk7XG4gICAgICAgIGlmIChlbGVtZW50KSB7XG4gICAgICAgICAgLy8gR2V0IHRhcmdldFNjcm9sbEluZGV4IGZyb20gdmlld3BvcnRTdGF0ZSBpZiBhdmFpbGFibGVcbiAgICAgICAgICBjb25zdCB0YXJnZXRTY3JvbGxJbmRleCA9ICh2aWV3cG9ydFN0YXRlIGFzIGFueSk/LnRhcmdldFNjcm9sbEluZGV4O1xuICAgICAgICAgIGNvbnN0IHBvc2l0aW9uID0gY2FsY3VsYXRlSXRlbVBvc2l0aW9uKFxuICAgICAgICAgICAgaSxcbiAgICAgICAgICAgIHNjcm9sbFBvc2l0aW9uLFxuICAgICAgICAgICAgdG90YWxJdGVtcyxcbiAgICAgICAgICAgIGl0ZW1TaXplLFxuICAgICAgICAgICAgdmlydHVhbFRvdGFsU2l6ZSxcbiAgICAgICAgICAgIGNvbnRhaW5lclNpemUsXG4gICAgICAgICAgICB0YXJnZXRTY3JvbGxJbmRleCxcbiAgICAgICAgICApO1xuXG4gICAgICAgICAgT2JqZWN0LmFzc2lnbihlbGVtZW50LnN0eWxlLCB7XG4gICAgICAgICAgICBwb3NpdGlvbjogXCJhYnNvbHV0ZVwiLFxuICAgICAgICAgICAgdHJhbnNmb3JtOiBgdHJhbnNsYXRlWSgke3Bvc2l0aW9ufXB4KWAsXG4gICAgICAgICAgICB3aWR0aDogXCIxMDAlXCIsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAvLyBDb2xsZWN0IGZvciBiYXRjaCBpbnNlcnRpb24gKGlmIG1haW50YWluaW5nIG9yZGVyKSBvciBqdXN0IGFwcGVuZFxuICAgICAgICAgIGlmIChtYWludGFpbkRvbU9yZGVyKSB7XG4gICAgICAgICAgICBuZXdFbGVtZW50cy5wdXNoKFtpLCBlbGVtZW50XSk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGl0ZW1zQ29udGFpbmVyLmFwcGVuZENoaWxkKGVsZW1lbnQpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZW5kZXJlZEVsZW1lbnRzLnNldChpLCBlbGVtZW50KTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyBCYXRjaCBpbnNlcnQgbmV3IGVsZW1lbnRzIGluIGNvcnJlY3QgRE9NIG9yZGVyIChvbmx5IHdoZW4gbWFpbnRhaW5Eb21PcmRlciBpcyBlbmFibGVkKVxuICAgICAgLy8gVGhpcyBlbnN1cmVzIENTUyBzZWxlY3RvcnMgbGlrZSA6Zmlyc3QtY2hpbGQsIDpudGgtY2hpbGQoKSB3b3JrIGNvcnJlY3RseVxuICAgICAgaWYgKG1haW50YWluRG9tT3JkZXIgJiYgbmV3RWxlbWVudHMubGVuZ3RoID4gMCkge1xuICAgICAgICAvLyBTb3J0IG5ldyBlbGVtZW50cyBieSBpbmRleFxuICAgICAgICBuZXdFbGVtZW50cy5zb3J0KChbYV0sIFtiXSkgPT4gYSAtIGIpO1xuXG4gICAgICAgIC8vIEdldCBleGlzdGluZyBET00gY2hpbGRyZW4gc29ydGVkIGJ5IHRoZWlyIGRhdGEtaW5kZXhcbiAgICAgICAgY29uc3QgZXhpc3RpbmdDaGlsZHJlbiA9IEFycmF5LmZyb20oXG4gICAgICAgICAgaXRlbXNDb250YWluZXIuY2hpbGRyZW4sXG4gICAgICAgICkgYXMgSFRNTEVsZW1lbnRbXTtcblxuICAgICAgICAvLyBNZXJnZSBuZXcgZWxlbWVudHMgaW50byBjb3JyZWN0IHBvc2l0aW9uc1xuICAgICAgICBsZXQgbmV3SWR4ID0gMDtcbiAgICAgICAgbGV0IGV4aXN0aW5nSWR4ID0gMDtcblxuICAgICAgICB3aGlsZSAobmV3SWR4IDwgbmV3RWxlbWVudHMubGVuZ3RoKSB7XG4gICAgICAgICAgY29uc3QgW25ld0luZGV4LCBuZXdFbGVtZW50XSA9IG5ld0VsZW1lbnRzW25ld0lkeF07XG5cbiAgICAgICAgICAvLyBGaW5kIHRoZSBmaXJzdCBleGlzdGluZyBjaGlsZCB3aXRoIGluZGV4ID4gbmV3SW5kZXhcbiAgICAgICAgICB3aGlsZSAoXG4gICAgICAgICAgICBleGlzdGluZ0lkeCA8IGV4aXN0aW5nQ2hpbGRyZW4ubGVuZ3RoICYmXG4gICAgICAgICAgICBwYXJzZUludChleGlzdGluZ0NoaWxkcmVuW2V4aXN0aW5nSWR4XS5kYXRhc2V0LmluZGV4IHx8IFwiLTFcIiwgMTApIDxcbiAgICAgICAgICAgICAgbmV3SW5kZXhcbiAgICAgICAgICApIHtcbiAgICAgICAgICAgIGV4aXN0aW5nSWR4Kys7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgaWYgKGV4aXN0aW5nSWR4IDwgZXhpc3RpbmdDaGlsZHJlbi5sZW5ndGgpIHtcbiAgICAgICAgICAgIC8vIEluc2VydCBiZWZvcmUgdGhlIGV4aXN0aW5nIGNoaWxkXG4gICAgICAgICAgICBpdGVtc0NvbnRhaW5lci5pbnNlcnRCZWZvcmUoXG4gICAgICAgICAgICAgIG5ld0VsZW1lbnQsXG4gICAgICAgICAgICAgIGV4aXN0aW5nQ2hpbGRyZW5bZXhpc3RpbmdJZHhdLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgLy8gQXBwZW5kIGF0IHRoZSBlbmRcbiAgICAgICAgICAgIGl0ZW1zQ29udGFpbmVyLmFwcGVuZENoaWxkKG5ld0VsZW1lbnQpO1xuICAgICAgICAgIH1cbiAgICAgICAgICBuZXdJZHgrKztcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyBSZXF1ZXN0IG1pc3NpbmcgaXRlbXNcbiAgICAgIGlmIChcbiAgICAgICAgbWlzc2luZ0l0ZW1zLmxlbmd0aCA+IDAgJiZcbiAgICAgICAgY29tcG9uZW50LnZpZXdwb3J0Py5jb2xsZWN0aW9uPy5sb2FkTWlzc2luZ1Jhbmdlc1xuICAgICAgKSB7XG4gICAgICAgIGNvbXBvbmVudC52aWV3cG9ydC5jb2xsZWN0aW9uLmxvYWRNaXNzaW5nUmFuZ2VzKFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIHN0YXJ0OiBNYXRoLm1pbiguLi5taXNzaW5nSXRlbXMpLFxuICAgICAgICAgICAgZW5kOiBNYXRoLm1heCguLi5taXNzaW5nSXRlbXMpLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgXCJyZW5kZXJpbmc6bWlzc2luZy1pdGVtc1wiLFxuICAgICAgICApO1xuICAgICAgfVxuXG4gICAgICBjdXJyZW50VmlzaWJsZVJhbmdlID0gdmlzaWJsZVJhbmdlO1xuXG4gICAgICAvLyBFbWl0IGl0ZW1zIHJlbmRlcmVkIGV2ZW50IHdpdGggZWxlbWVudHMgZm9yIHNpemUgY2FsY3VsYXRpb25cbiAgICAgIGNvbnN0IHJlbmRlcmVkRWxlbWVudHNBcnJheSA9IEFycmF5LmZyb20ocmVuZGVyZWRFbGVtZW50cy52YWx1ZXMoKSk7XG4gICAgICBjb21wb25lbnQuZW1pdD8uKFwidmlld3BvcnQ6aXRlbXMtcmVuZGVyZWRcIiwge1xuICAgICAgICBlbGVtZW50czogcmVuZGVyZWRFbGVtZW50c0FycmF5LFxuICAgICAgICByYW5nZTogdmlzaWJsZVJhbmdlLFxuICAgICAgfSk7XG5cbiAgICAgIGNvbXBvbmVudC5lbWl0Py4oXCJ2aWV3cG9ydDpyZW5kZXJlZFwiLCB7XG4gICAgICAgIHJhbmdlOiB2aXNpYmxlUmFuZ2UsXG4gICAgICAgIHJlbmRlcmVkQ291bnQ6IHJlbmRlcmVkRWxlbWVudHMuc2l6ZSxcbiAgICAgIH0pO1xuICAgICAgdXBkYXRlSXRlbVBvc2l0aW9ucygpO1xuXG4gICAgICAvLyBMb2FkIGRhdGEgaWYgbm8gaXRlbXMgcmVuZGVyZWRcbiAgICAgIGlmIChcbiAgICAgICAgcmVuZGVyZWRFbGVtZW50cy5zaXplID09PSAwICYmXG4gICAgICAgIHRvdGFsSXRlbXMgPiAwICYmXG4gICAgICAgIGNvbXBvbmVudC52aWV3cG9ydD8uY29sbGVjdGlvblxuICAgICAgKSB7XG4gICAgICAgIGNvbXBvbmVudC52aWV3cG9ydC5jb2xsZWN0aW9uLmxvYWRNaXNzaW5nUmFuZ2VzKFxuICAgICAgICAgIHZpc2libGVSYW5nZSxcbiAgICAgICAgICBcInJlbmRlcmluZzpuby1pdGVtc1wiLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH07XG5cbiAgICAvLyBFeHRlbmQgdmlld3BvcnQgQVBJXG4gICAgY29uc3Qgb3JpZ2luYWxSZW5kZXJJdGVtcyA9IGNvbXBvbmVudC52aWV3cG9ydC5yZW5kZXJJdGVtcztcbiAgICBjb21wb25lbnQudmlld3BvcnQucmVuZGVySXRlbXMgPSAoKSA9PiB7XG4gICAgICByZW5kZXJJdGVtcygpO1xuICAgICAgb3JpZ2luYWxSZW5kZXJJdGVtcz8uKCk7XG4gICAgfTtcblxuICAgIC8vIENsZWFudXBcbiAgICB3cmFwRGVzdHJveShjb21wb25lbnQsICgpID0+IHtcbiAgICAgIC8vIFJlbGVhc2UgYWxsIHJlbmRlcmVkIGVsZW1lbnRzIC0gdXNlIHJlbGVhc2VFbGVtZW50IHRvIHByb3Blcmx5IGRlc3Ryb3kgbGF5b3V0UmVzdWx0c1xuICAgICAgcmVuZGVyZWRFbGVtZW50cy5mb3JFYWNoKChlbGVtZW50KSA9PiB7XG4gICAgICAgIHJlbGVhc2VFbGVtZW50KGVsZW1lbnQpO1xuICAgICAgfSk7XG4gICAgICByZW5kZXJlZEVsZW1lbnRzLmNsZWFyKCk7XG5cbiAgICAgIC8vIENsZWFyIGVsZW1lbnQgcG9vbFxuICAgICAgZWxlbWVudFBvb2wubGVuZ3RoID0gMDtcblxuICAgICAgLy8gUmVzZXQgc3RhdGVcbiAgICAgIGN1cnJlbnRWaXNpYmxlUmFuZ2UgPSB7IHN0YXJ0OiAtMSwgZW5kOiAtMSB9O1xuICAgICAgdmlld3BvcnRTdGF0ZSA9IG51bGw7XG4gICAgICBsYXN0UmVuZGVyVGltZSA9IDA7XG5cbiAgICAgIC8vIFJlc2V0IHBvb2wgc3RhdHNcbiAgICAgIHBvb2xTdGF0cy5jcmVhdGVkID0gMDtcbiAgICAgIHBvb2xTdGF0cy5yZWN5Y2xlZCA9IDA7XG4gICAgICBwb29sU3RhdHMucG9vbFNpemUgPSAwO1xuICAgIH0pO1xuXG4gICAgcmV0dXJuIGNvbXBvbmVudDtcbiAgfTtcbn07XG4iLAogICAgIi8qKlxuICogQG1vZHVsZSBjb3JlL2xheW91dC9zY2hlbWFcbiAqIEBkZXNjcmlwdGlvbiBVbmlmaWVkIGxheW91dCBzY2hlbWEgcHJvY2Vzc29yIHdpdGggaW50ZWdyYXRlZCBvcHRpbWl6YXRpb25zXG4gKiBDb25zb2xpZGF0ZXMgYXJyYXksIG9iamVjdCwgSlNYLCBhbmQgdGVtcGxhdGUgcHJvY2Vzc2luZyB3aXRoIGJ1aWx0LWluIHBlcmZvcm1hbmNlIGVuaGFuY2VtZW50c1xuICovXG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIFRZUEVTIChFc3NlbnRpYWwgb25seSlcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0IGludGVyZmFjZSBMYXlvdXRDb25maWcge1xuICAvKiogQmFzZSBsYXlvdXQgdHlwZSAqL1xuICB0eXBlPzogXCJzdGFja1wiIHwgXCJyb3dcIiB8IFwiZ3JpZFwiIHwgc3RyaW5nO1xuICAvKiogU3BhY2luZyBiZXR3ZWVuIGVsZW1lbnRzICovXG4gIGdhcD86IG51bWJlciB8IHN0cmluZztcbiAgLyoqIEFkZGl0aW9uYWwgQ1NTIGNsYXNzZXMgKi9cbiAgY2xhc3M/OiBzdHJpbmc7XG4gIC8qKiBBbGlnbm1lbnQgb2YgaXRlbXMgYWxvbmcgdGhlIGNyb3NzIGF4aXMgKi9cbiAgYWxpZ24/OiBcInN0YXJ0XCIgfCBcImNlbnRlclwiIHwgXCJlbmRcIiB8IFwic3RyZXRjaFwiO1xuICAvKiogQWxpZ25tZW50IG9mIGl0ZW1zIGFsb25nIHRoZSBtYWluIGF4aXMgKi9cbiAganVzdGlmeT86IFwic3RhcnRcIiB8IFwiY2VudGVyXCIgfCBcImVuZFwiIHwgXCJiZXR3ZWVuXCIgfCBcImFyb3VuZFwiIHwgXCJldmVubHlcIjtcbiAgLyoqIFdoZXRoZXIgYW5kIGhvdyBpdGVtcyBzaG91bGQgd3JhcCAqL1xuICB3cmFwPzogYm9vbGVhbiB8IFwicmV2ZXJzZVwiIHwgXCJub3dyYXBcIjtcbiAgLyoqIFdoZXRoZXIgcm93IGl0ZW1zIHNob3VsZCBzdGFjayB2ZXJ0aWNhbGx5IG9uIG1vYmlsZSAqL1xuICBtb2JpbGVTdGFjaz86IGJvb2xlYW47XG4gIC8qKiBXaGV0aGVyIHJvdyBpdGVtcyBzaG91bGQgc2Nyb2xsIGhvcml6b250YWxseSBvbiBtb2JpbGUgKi9cbiAgbW9iaWxlU2Nyb2xsPzogYm9vbGVhbjtcbiAgLyoqIE51bWJlciBvZiBjb2x1bW5zIG9yIGF1dG9tYXRpYyBzaXppbmcgbWV0aG9kICovXG4gIGNvbHVtbnM/OiBudW1iZXIgfCBcImF1dG8tZml0XCIgfCBcImF1dG8tZmlsbFwiO1xuICAvKiogV2hldGhlciB0byB1c2UgZGVuc2UgcGFja2luZyBhbGdvcml0aG0gZm9yIGdyaWQgKi9cbiAgZGVuc2U/OiBib29sZWFuO1xuICAvKiogV2hldGhlciBncmlkIGl0ZW1zIHNob3VsZCBhZGp1c3QgaGVpZ2h0IGF1dG9tYXRpY2FsbHkgKi9cbiAgYXV0b0hlaWdodD86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTGF5b3V0SXRlbUNvbmZpZyB7XG4gIC8qKiBDb2x1bW4gd2lkdGggaW4gYSAxMi1jb2x1bW4gZ3JpZCAqL1xuICB3aWR0aD86IG51bWJlcjtcbiAgLyoqIFdpZHRoIG9uIHNtYWxsIHNjcmVlbnMgKi9cbiAgc20/OiBudW1iZXI7XG4gIC8qKiBXaWR0aCBvbiBtZWRpdW0gc2NyZWVucyAqL1xuICBtZD86IG51bWJlcjtcbiAgLyoqIFdpZHRoIG9uIGxhcmdlIHNjcmVlbnMgKi9cbiAgbGc/OiBudW1iZXI7XG4gIC8qKiBXaWR0aCBvbiBleHRyYS1sYXJnZSBzY3JlZW5zICovXG4gIHhsPzogbnVtYmVyO1xuICAvKiogTnVtYmVyIG9mIGdyaWQgY29sdW1ucyB0byBzcGFuICovXG4gIHNwYW4/OiBudW1iZXI7XG4gIC8qKiBOdW1iZXIgb2YgZ3JpZCByb3dzIHRvIHNwYW4gKi9cbiAgcm93U3Bhbj86IG51bWJlcjtcbiAgLyoqIERpc3BsYXkgb3JkZXIgKi9cbiAgb3JkZXI/OiBudW1iZXIgfCBcImZpcnN0XCIgfCBcImxhc3RcIjtcbiAgLyoqIFNlbGYtYWxpZ25tZW50IHdpdGhpbiBjb250YWluZXIgKi9cbiAgYWxpZ24/OiBcInN0YXJ0XCIgfCBcImNlbnRlclwiIHwgXCJlbmRcIiB8IFwic3RyZXRjaFwiO1xuICAvKiogV2hldGhlciBpdGVtIHNob3VsZCBhdXRvbWF0aWNhbGx5IHNpemUgKi9cbiAgYXV0bz86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTGF5b3V0T3B0aW9ucyB7XG4gIC8qKiBEZWZhdWx0IGNyZWF0b3IgZnVuY3Rpb24gdG8gdXNlIGlmIG5vdCBzcGVjaWZpZWQgaW4gc2NoZW1hICovXG4gIGNyZWF0b3I/OiBGdW5jdGlvbjtcbiAgLyoqIFdoZXRoZXIgdG8gYXBwbHkgQ1NTIGNsYXNzIHByZWZpeCBAZGVmYXVsdCB0cnVlICovXG4gIHByZWZpeD86IGJvb2xlYW47XG4gIC8qKiBBZGRpdGlvbmFsIG9wdGlvbnMgKi9cbiAgW2tleTogc3RyaW5nXTogYW55O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIExheW91dFJlc3VsdCB7XG4gIC8qKiBUaGUgcmF3IGxheW91dCBvYmplY3Qgd2l0aCBhbGwgY29tcG9uZW50cyAqL1xuICBsYXlvdXQ6IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIC8qKiBSZWZlcmVuY2UgdG8gdGhlIHJvb3QgZWxlbWVudCBmb3IgY29udmVuaWVuY2UgKi9cbiAgZWxlbWVudDogSFRNTEVsZW1lbnQgfCBhbnk7XG4gIC8qKiBGbGF0dGVuZWQgY29tcG9uZW50IG1hcCAqL1xuICBjb21wb25lbnQ6IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIC8qKiBHZXRzIGEgY29tcG9uZW50IGJ5IG5hbWUgKi9cbiAgZ2V0KG5hbWU6IHN0cmluZyk6IGFueTtcbiAgLyoqIEdldHMgYWxsIGNvbXBvbmVudHMgaW4gYSBmbGF0dGVuZWQgbWFwICovXG4gIGdldEFsbCgpOiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuICAvKiogRGVzdHJveXMgdGhlIGxheW91dCwgY2xlYW5pbmcgdXAgYWxsIGNvbXBvbmVudHMgKi9cbiAgZGVzdHJveSgpOiB2b2lkO1xufVxuXG50eXBlIENvbXBvbmVudExpa2UgPSB7IGVsZW1lbnQ6IEhUTUxFbGVtZW50OyBba2V5OiBzdHJpbmddOiBhbnkgfTtcbnR5cGUgU2NoZW1hSXRlbSA9IEZ1bmN0aW9uIHwgc3RyaW5nIHwgUmVjb3JkPHN0cmluZywgYW55PiB8IFNjaGVtYUl0ZW1bXTtcblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gQlVJTFQtSU4gT1BUSU1JWkFUSU9OU1xuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vKipcbiAqIEZyYWdtZW50IFBvb2wgZm9yIGVmZmljaWVudCBEb2N1bWVudEZyYWdtZW50IHJldXNlXG4gKiBSZWR1Y2VzIEdDIHByZXNzdXJlIGluIGhpZ2gtZnJlcXVlbmN5IGxheW91dCBjcmVhdGlvbiBzY2VuYXJpb3NcbiAqL1xuY2xhc3MgRnJhZ21lbnRQb29sIHtcbiAgcHJpdmF0ZSBwb29sOiBEb2N1bWVudEZyYWdtZW50W10gPSBbXTtcbiAgcHJpdmF0ZSBtYXhTaXplID0gODsgLy8gT3B0aW1pemVkIHNpemUgZm9yIG1lbW9yeSBlZmZpY2llbmN5XG5cbiAgZ2V0KCk6IERvY3VtZW50RnJhZ21lbnQge1xuICAgIHJldHVybiB0aGlzLnBvb2wucG9wKCkgfHwgZG9jdW1lbnQuY3JlYXRlRG9jdW1lbnRGcmFnbWVudCgpO1xuICB9XG5cbiAgcmVsZWFzZShmcmFnbWVudDogRG9jdW1lbnRGcmFnbWVudCk6IHZvaWQge1xuICAgIGlmICh0aGlzLnBvb2wubGVuZ3RoIDwgdGhpcy5tYXhTaXplICYmIGZyYWdtZW50LmNoaWxkTm9kZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICB0aGlzLnBvb2wucHVzaChmcmFnbWVudCk7XG4gICAgfVxuICB9XG5cbiAgY2xlYXIoKTogdm9pZCB7XG4gICAgdGhpcy5wb29sLmxlbmd0aCA9IDA7XG4gIH1cbn1cblxuY29uc3QgZnJhZ21lbnRQb29sID0gbmV3IEZyYWdtZW50UG9vbCgpO1xuXG4vKipcbiAqIENsYXNzIG5hbWUgY2FjaGUgZm9yIGxheW91dCBjb25maWd1cmF0aW9uc1xuICogUmVkdWNlcyBzdHJpbmcgb3BlcmF0aW9ucyBieSBjYWNoaW5nIGdlbmVyYXRlZCBjbGFzcyBuYW1lc1xuICovXG5jb25zdCBjbGFzc0NhY2hlID0gbmV3IE1hcDxzdHJpbmcsIHN0cmluZz4oKTtcblxuLy8gQ29uZmlndXJhdGlvbiBjb25zdGFudHNcbmNvbnN0IFBSRUZJWCA9IFwibXRybFwiOyAvLyBUT0RPOiBNYWtlIHRoaXMgY29uZmlndXJhYmxlXG5jb25zdCBQUkVGSVhfV0lUSF9EQVNIID0gYCR7UFJFRklYfS1gO1xuXG4vKipcbiAqIEdldHMgYSBjYWNoZWQgY2xhc3MgbmFtZSBmb3IgbGF5b3V0IGNvbmZpZ3VyYXRpb25cbiAqIE9wdGltaXplZCBmb3IgZGlmZmVyZW50IGNsYXNzIG5hbWluZyBwYXR0ZXJuc1xuICovXG5mdW5jdGlvbiBnZXRDYWNoZWRDbGFzc05hbWUoXG4gIHR5cGU6IHN0cmluZyxcbiAgcHJvcGVydHk6IHN0cmluZyxcbiAgdmFsdWU6IHN0cmluZyB8IG51bWJlcixcbik6IHN0cmluZyB7XG4gIGNvbnN0IGtleSA9IGAke3R5cGV9LSR7cHJvcGVydHl9LSR7dmFsdWV9YDtcbiAgaWYgKCFjbGFzc0NhY2hlLmhhcyhrZXkpKSB7XG4gICAgaWYgKHR5cGUgPT09IFwiaXRlbVwiKSB7XG4gICAgICBjbGFzc0NhY2hlLnNldChcbiAgICAgICAga2V5LFxuICAgICAgICBwcm9wZXJ0eSA9PT0gXCJcIlxuICAgICAgICAgID8gYGxheW91dF9faXRlbS0tJHt2YWx1ZX1gXG4gICAgICAgICAgOiBgbGF5b3V0X19pdGVtLS0ke3Byb3BlcnR5fS0ke3ZhbHVlfWAsXG4gICAgICApO1xuICAgIH0gZWxzZSB7XG4gICAgICAvLyBGb3IgbGF5b3V0IGNsYXNzZXMsIGFsaWduIHVzZXMgbGF5b3V0LS17dHlwZX0te3ZhbHVlfVxuICAgICAgLy8gYnV0IGp1c3RpZnkgYW5kIG90aGVycyB1c2UgbGF5b3V0LS17dHlwZX0te3Byb3BlcnR5fS17dmFsdWV9XG4gICAgICBpZiAocHJvcGVydHkgPT09IFwiYWxpZ25cIikge1xuICAgICAgICBjbGFzc0NhY2hlLnNldChrZXksIGBsYXlvdXQtLSR7dHlwZX0tJHt2YWx1ZX1gKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNsYXNzQ2FjaGUuc2V0KGtleSwgYGxheW91dC0tJHt0eXBlfS0ke3Byb3BlcnR5fS0ke3ZhbHVlfWApO1xuICAgICAgfVxuICAgIH1cbiAgfVxuICByZXR1cm4gY2xhc3NDYWNoZS5nZXQoa2V5KSE7XG59XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIE9QVElNSVpFRCBVVElMSVRJRVNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBDaGVja3MgaWYgYSB2YWx1ZSBpcyBhIGNvbXBvbmVudCBvYmplY3QgKGhhcyBhbiBlbGVtZW50IHByb3BlcnR5KVxuICovXG5mdW5jdGlvbiBpc0NvbXBvbmVudCh2YWx1ZTogYW55KTogdmFsdWUgaXMgQ29tcG9uZW50TGlrZSB7XG4gIHJldHVybiB2YWx1ZSAmJiB0eXBlb2YgdmFsdWUgPT09IFwib2JqZWN0XCIgJiYgXCJlbGVtZW50XCIgaW4gdmFsdWU7XG59XG5cbi8qKlxuICogQ3JlYXRlcyBhIGRvY3VtZW50IGZyYWdtZW50IHVzaW5nIHBvb2xpbmcgZm9yIGJldHRlciBwZXJmb3JtYW5jZVxuICovXG5mdW5jdGlvbiBjcmVhdGVGcmFnbWVudCgpOiBEb2N1bWVudEZyYWdtZW50IHtcbiAgcmV0dXJuIGZyYWdtZW50UG9vbC5nZXQoKTtcbn1cblxuLyoqXG4gKiBSZWxlYXNlcyBhIGZyYWdtZW50IGJhY2sgdG8gdGhlIHBvb2wgZm9yIHJldXNlXG4gKi9cbmZ1bmN0aW9uIHJlbGVhc2VGcmFnbWVudChmcmFnbWVudDogRG9jdW1lbnRGcmFnbWVudCk6IHZvaWQge1xuICBmcmFnbWVudFBvb2wucmVsZWFzZShmcmFnbWVudCk7XG59XG5cbi8qKlxuICogT3B0aW1pemVkIGNsYXNzIHByb2Nlc3Npbmcgd2l0aCBtaW5pbWFsIHN0cmluZyBvcGVyYXRpb25zXG4gKiBIYW5kbGVzIGFycmF5cywgc3RyaW5ncywgY2xhc3NOYW1lIGFsaWFzZXMsIGFuZCByYXdDbGFzcyBlZmZpY2llbnRseVxuICovXG5mdW5jdGlvbiBwcm9jZXNzQ2xhc3NOYW1lcyhcbiAgb3B0aW9uczogUmVjb3JkPHN0cmluZywgYW55PixcbiAgc2tpcFByZWZpeCA9IGZhbHNlLFxuKTogUmVjb3JkPHN0cmluZywgYW55PiB7XG4gIGlmICghb3B0aW9ucykgcmV0dXJuIG9wdGlvbnM7XG5cbiAgY29uc3QgaGFzUmF3Q2xhc3MgPSBvcHRpb25zLnJhd0NsYXNzO1xuICBjb25zdCBoYXNSZWd1bGFyQ2xhc3MgPSBvcHRpb25zLmNsYXNzIHx8IG9wdGlvbnMuY2xhc3NOYW1lO1xuXG4gIC8vIEZhc3QgcGF0aDogbm8gY2xhc3MgcHJvcGVydGllcyBhdCBhbGxcbiAgaWYgKCFoYXNSYXdDbGFzcyAmJiAhaGFzUmVndWxhckNsYXNzKSByZXR1cm4gb3B0aW9ucztcblxuICAvLyBGYXN0IHBhdGg6IG9ubHkgcmF3Q2xhc3MgYW5kIHNraXBwaW5nIHByZWZpeCAobW9zdCBjb21tb24gcmF3Q2xhc3Mgc2NlbmFyaW8pXG4gIGlmIChoYXNSYXdDbGFzcyAmJiAhaGFzUmVndWxhckNsYXNzICYmIHNraXBQcmVmaXgpIHtcbiAgICBjb25zdCBwcm9jZXNzZWQgPSB7IC4uLm9wdGlvbnMgfTtcbiAgICBkZWxldGUgcHJvY2Vzc2VkLnJhd0NsYXNzO1xuXG4gICAgLy8gRGlyZWN0IGFzc2lnbm1lbnQgZm9yIHNpbXBsZSBzdHJpbmdcbiAgICBpZiAodHlwZW9mIGhhc1Jhd0NsYXNzID09PSBcInN0cmluZ1wiKSB7XG4gICAgICBwcm9jZXNzZWQuY2xhc3MgPSBoYXNSYXdDbGFzcztcbiAgICB9IGVsc2Uge1xuICAgICAgLy8gSGFuZGxlIGFycmF5IGNhc2VcbiAgICAgIHByb2Nlc3NlZC5jbGFzcyA9IGhhc1Jhd0NsYXNzLmpvaW4oXCIgXCIpO1xuICAgIH1cbiAgICByZXR1cm4gcHJvY2Vzc2VkO1xuICB9XG5cbiAgLy8gRnVsbCBwcm9jZXNzaW5nIHBhdGggKG9ubHkgd2hlbiBuZWVkZWQpXG4gIGNvbnN0IHByb2Nlc3NlZCA9IHsgLi4ub3B0aW9ucyB9O1xuICBsZXQgZmluYWxDbGFzc2VzID0gXCJcIjtcblxuICAvLyBIYW5kbGUgcHJlZml4ZWQgY2xhc3NlcyBvbmx5IGlmIG5vdCBza2lwcGluZyBwcmVmaXhcbiAgaWYgKCFza2lwUHJlZml4ICYmIGhhc1JlZ3VsYXJDbGFzcykge1xuICAgIGxldCBwcmVmaXhlZFN0cmluZyA9IFwiXCI7XG5cbiAgICBpZiAocHJvY2Vzc2VkLmNsYXNzKSB7XG4gICAgICBwcmVmaXhlZFN0cmluZyArPSBBcnJheS5pc0FycmF5KHByb2Nlc3NlZC5jbGFzcylcbiAgICAgICAgPyBwcm9jZXNzZWQuY2xhc3Muam9pbihcIiBcIilcbiAgICAgICAgOiBwcm9jZXNzZWQuY2xhc3M7XG4gICAgfVxuICAgIGlmIChwcm9jZXNzZWQuY2xhc3NOYW1lKSB7XG4gICAgICBwcmVmaXhlZFN0cmluZyArPSAocHJlZml4ZWRTdHJpbmcgPyBcIiBcIiA6IFwiXCIpICsgcHJvY2Vzc2VkLmNsYXNzTmFtZTtcbiAgICB9XG5cbiAgICBpZiAocHJlZml4ZWRTdHJpbmcpIHtcbiAgICAgIGZpbmFsQ2xhc3NlcyA9IHByZWZpeGVkU3RyaW5nXG4gICAgICAgIC5zcGxpdCgvXFxzKy8pXG4gICAgICAgIC5maWx0ZXIoQm9vbGVhbilcbiAgICAgICAgLm1hcCgoY2xzKSA9PlxuICAgICAgICAgIGNscy5zdGFydHNXaXRoKFBSRUZJWF9XSVRIX0RBU0gpID8gY2xzIDogUFJFRklYX1dJVEhfREFTSCArIGNscyxcbiAgICAgICAgKVxuICAgICAgICAuam9pbihcIiBcIik7XG4gICAgfVxuICB9XG5cbiAgLy8gSGFuZGxlIHJhd0NsYXNzIChhbHdheXMgcHJvY2Vzc2VkIHdoZW4gcHJlc2VudClcbiAgaWYgKGhhc1Jhd0NsYXNzKSB7XG4gICAgY29uc3QgcmF3U3RyaW5nID0gQXJyYXkuaXNBcnJheShoYXNSYXdDbGFzcylcbiAgICAgID8gaGFzUmF3Q2xhc3MuZmlsdGVyKEJvb2xlYW4pLmpvaW4oXCIgXCIpXG4gICAgICA6IGhhc1Jhd0NsYXNzO1xuXG4gICAgZmluYWxDbGFzc2VzICs9IChmaW5hbENsYXNzZXMgPyBcIiBcIiA6IFwiXCIpICsgcmF3U3RyaW5nO1xuICB9XG5cbiAgaWYgKGZpbmFsQ2xhc3Nlcykge1xuICAgIHByb2Nlc3NlZC5jbGFzcyA9IGZpbmFsQ2xhc3NlcztcbiAgfVxuXG4gIC8vIENsZWFuIHVwIGluIG9uZSBvcGVyYXRpb25cbiAgZGVsZXRlIHByb2Nlc3NlZC5jbGFzc05hbWU7XG4gIGRlbGV0ZSBwcm9jZXNzZWQucmF3Q2xhc3M7XG5cbiAgcmV0dXJuIHByb2Nlc3NlZDtcbn1cblxuLyoqXG4gKiBPcHRpbWl6ZWQgcGFyYW1ldGVyIGV4dHJhY3Rpb24gZm9yIGFycmF5IHNjaGVtYXNcbiAqIFJlZHVjZXMgbXVsdGlwbGUgYXJyYXkgbG9va3VwcyB3aXRoIGJhdGNoIHByb2Nlc3NpbmdcbiAqL1xuaW50ZXJmYWNlIEV4dHJhY3RlZFBhcmFtZXRlcnMge1xuICBjcmVhdG9yOiBGdW5jdGlvbjtcbiAgbmFtZT86IHN0cmluZztcbiAgb3B0aW9uczogUmVjb3JkPHN0cmluZywgYW55PjtcbiAgY29uc3VtZWQ6IG51bWJlcjtcbn1cblxuZnVuY3Rpb24gZXh0cmFjdFBhcmFtZXRlcnMoXG4gIHNjaGVtYTogU2NoZW1hSXRlbVtdLFxuICBzdGFydEluZGV4OiBudW1iZXIsXG4gIGRlZmF1bHRDcmVhdG9yOiBGdW5jdGlvbixcbik6IEV4dHJhY3RlZFBhcmFtZXRlcnMge1xuICBjb25zdCBpdGVtcyA9IHNjaGVtYS5zbGljZShzdGFydEluZGV4LCBzdGFydEluZGV4ICsgMyk7XG4gIGxldCBjcmVhdG9yLCBuYW1lLCBvcHRpb25zO1xuICBsZXQgY29uc3VtZWQgPSAxO1xuXG4gIGNvbnN0IFtmaXJzdCwgc2Vjb25kLCB0aGlyZF0gPSBpdGVtcztcblxuICBpZiAodHlwZW9mIGZpcnN0ID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICBjcmVhdG9yID0gZmlyc3Q7XG4gICAgaWYgKHR5cGVvZiBzZWNvbmQgPT09IFwic3RyaW5nXCIpIHtcbiAgICAgIG5hbWUgPSBzZWNvbmQ7XG4gICAgICBjb25zdW1lZCA9IDI7XG4gICAgICBpZiAoaXNPYmplY3QodGhpcmQpKSB7XG4gICAgICAgIG9wdGlvbnMgPSB0aGlyZDtcbiAgICAgICAgY29uc3VtZWQgPSAzO1xuICAgICAgfVxuICAgIH0gZWxzZSBpZiAoaXNPYmplY3Qoc2Vjb25kKSkge1xuICAgICAgb3B0aW9ucyA9IHNlY29uZDtcbiAgICAgIGNvbnN1bWVkID0gMjtcbiAgICB9XG4gIH0gZWxzZSBpZiAodHlwZW9mIGZpcnN0ID09PSBcInN0cmluZ1wiKSB7XG4gICAgY3JlYXRvciA9IGRlZmF1bHRDcmVhdG9yO1xuICAgIG5hbWUgPSBmaXJzdDtcbiAgICBpZiAoaXNPYmplY3Qoc2Vjb25kKSkge1xuICAgICAgb3B0aW9ucyA9IHNlY29uZDtcbiAgICAgIGNvbnN1bWVkID0gMjtcbiAgICB9XG4gIH0gZWxzZSBpZiAoaXNPYmplY3QoZmlyc3QpKSB7XG4gICAgY3JlYXRvciA9IGRlZmF1bHRDcmVhdG9yO1xuICAgIG9wdGlvbnMgPSBmaXJzdDtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgY3JlYXRvcjogY3JlYXRvciB8fCBkZWZhdWx0Q3JlYXRvcixcbiAgICBuYW1lLFxuICAgIG9wdGlvbnM6IChvcHRpb25zIHx8IHt9KSBhcyBSZWNvcmQ8c3RyaW5nLCBhbnk+LFxuICAgIGNvbnN1bWVkLFxuICB9O1xufVxuXG4vKipcbiAqIFNpbXBsZSBvYmplY3QgdHlwZSBjaGVja1xuICovXG5mdW5jdGlvbiBpc09iamVjdCh2YWx1ZTogYW55KTogYm9vbGVhbiB7XG4gIHJldHVybiB2YWx1ZSAhPT0gbnVsbCAmJiB0eXBlb2YgdmFsdWUgPT09IFwib2JqZWN0XCIgJiYgIUFycmF5LmlzQXJyYXkodmFsdWUpO1xufVxuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBET00gVVRJTElUSUVTIChTaW1wbGlmaWVkKVxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vKipcbiAqIFNpbXBsZSBET00gZWxlbWVudCBjcmVhdGlvblxuICovXG5mdW5jdGlvbiBjcmVhdGVFbGVtZW50KG9wdGlvbnM6IFJlY29yZDxzdHJpbmcsIGFueT4gPSB7fSk6IEhUTUxFbGVtZW50IHtcbiAgY29uc3QgdGFnID0gb3B0aW9ucy50YWcgfHwgXCJkaXZcIjtcbiAgY29uc3QgZWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQodGFnKTtcblxuICBpZiAob3B0aW9ucy5jbGFzcykge1xuICAgIGVsZW1lbnQuY2xhc3NOYW1lID0gb3B0aW9ucy5jbGFzcztcbiAgfVxuXG4gIGlmIChvcHRpb25zLnN0eWxlKSB7XG4gICAgaWYgKHR5cGVvZiBvcHRpb25zLnN0eWxlID09PSBcInN0cmluZ1wiKSB7XG4gICAgICBlbGVtZW50LnNldEF0dHJpYnV0ZShcInN0eWxlXCIsIG9wdGlvbnMuc3R5bGUpO1xuICAgIH0gZWxzZSBpZiAodHlwZW9mIG9wdGlvbnMuc3R5bGUgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgIE9iamVjdC5hc3NpZ24oZWxlbWVudC5zdHlsZSwgb3B0aW9ucy5zdHlsZSk7XG4gICAgfVxuICB9XG5cbiAgaWYgKG9wdGlvbnMudGV4dENvbnRlbnQpIHtcbiAgICBlbGVtZW50LnRleHRDb250ZW50ID0gb3B0aW9ucy50ZXh0Q29udGVudDtcbiAgfVxuXG4gIGlmIChvcHRpb25zLnRleHQpIHtcbiAgICBlbGVtZW50LnRleHRDb250ZW50ID0gb3B0aW9ucy50ZXh0O1xuICB9XG5cbiAgcmV0dXJuIGVsZW1lbnQ7XG59XG5cbi8qKlxuICogQWRkcyBhIENTUyBjbGFzcyB0byBhbiBlbGVtZW50XG4gKi9cbmZ1bmN0aW9uIGFkZENsYXNzKGVsZW1lbnQ6IEhUTUxFbGVtZW50LCBjbGFzc05hbWU6IHN0cmluZyk6IHZvaWQge1xuICBpZiAoZWxlbWVudCAmJiBjbGFzc05hbWUpIHtcbiAgICBlbGVtZW50LmNsYXNzTGlzdC5hZGQoY2xhc3NOYW1lKTtcbiAgfVxufVxuXG4vKipcbiAqIENoZWNrcyBpZiBlbGVtZW50IGhhcyBhIENTUyBjbGFzc1xuICovXG5mdW5jdGlvbiBoYXNDbGFzcyhlbGVtZW50OiBIVE1MRWxlbWVudCwgY2xhc3NOYW1lOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgcmV0dXJuIGVsZW1lbnQgJiYgZWxlbWVudC5jbGFzc0xpc3QuY29udGFpbnMoY2xhc3NOYW1lKTtcbn1cblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gVU5JRklFRCBDT01QT05FTlQgQ1JFQVRJT04gKHdpdGggZGVzdHJ1Y3R1cmluZyBvcHRpbWl6YXRpb24pXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbi8qKlxuICogQ3JlYXRlcyBhIGNvbXBvbmVudCBpbnN0YW5jZSB3aXRoIG9wdGltaXplZCBvcHRpb24gcHJvY2Vzc2luZ1xuICogVXNlcyBkZXN0cnVjdHVyaW5nIGZvciBjbGVhbmVyIHNlcGFyYXRpb24gb2YgY29uY2VybnNcbiAqL1xuZnVuY3Rpb24gY3JlYXRlQ29tcG9uZW50SW5zdGFuY2UoXG4gIENvbXBvbmVudDogYW55LFxuICBvcHRpb25zOiBSZWNvcmQ8c3RyaW5nLCBhbnk+ID0ge30sXG4pOiBhbnkge1xuICB0cnkge1xuICAgIC8vIERlc3RydWN0dXJlIHNwZWNpYWwgY29uZmlncyBpbiBvbmUgb3BlcmF0aW9uXG4gICAgY29uc3Qge1xuICAgICAgbGF5b3V0OiBsYXlvdXRDb25maWcsXG4gICAgICBsYXlvdXRJdGVtOiBsYXlvdXRJdGVtQ29uZmlnLFxuICAgICAgc3R5bGU6IHN0eWxlQ29uZmlnLFxuICAgICAgYXR0cmlidXRlczogYXR0cmlidXRlc0NvbmZpZyxcbiAgICAgIGV2ZW50czogZXZlbnRzQ29uZmlnLFxuICAgICAgZXZlbnQsIC8vIExlZ2FjeSBzdXBwb3J0XG4gICAgICAuLi5jbGVhbk9wdGlvbnNcbiAgICB9ID0gb3B0aW9ucztcblxuICAgIC8vIFVzZSBldmVudHMgb3ZlciBldmVudCAoZXZlbnRzIGlzIHByZWZlcnJlZClcbiAgICBjb25zdCBmaW5hbEV2ZW50c0NvbmZpZyA9IGV2ZW50c0NvbmZpZyB8fCBldmVudDtcblxuICAgIC8vIElmIHN0eWxlIGlzIGEgc3RyaW5nLCBhbHdheXMgcGFzcyBpdCB0aHJvdWdoIHRvIHRoZSBjb21wb25lbnRcbiAgICBpZiAoc3R5bGVDb25maWcgJiYgdHlwZW9mIHN0eWxlQ29uZmlnID09PSBcInN0cmluZ1wiKSB7XG4gICAgICBjbGVhbk9wdGlvbnMuc3R5bGUgPSBzdHlsZUNvbmZpZztcbiAgICB9XG5cbiAgICAvLyBDcmVhdGUgY29tcG9uZW50XG4gICAgY29uc3QgaXNDbGFzcyA9XG4gICAgICB0eXBlb2YgQ29tcG9uZW50ID09PSBcImZ1bmN0aW9uXCIgJiZcbiAgICAgIE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IoQ29tcG9uZW50LCBcInByb3RvdHlwZVwiKT8ud3JpdGFibGUgPT09XG4gICAgICAgIGZhbHNlO1xuXG4gICAgY29uc3QgY29tcG9uZW50ID0gaXNDbGFzc1xuICAgICAgPyBuZXcgQ29tcG9uZW50KGNsZWFuT3B0aW9ucylcbiAgICAgIDogQ29tcG9uZW50KGNsZWFuT3B0aW9ucyk7XG5cbiAgICAvLyBBcHBseSBjb25maWd1cmF0aW9ucyBpZiBjb21wb25lbnQgaGFzIGVsZW1lbnRcbiAgICBpZiAoY29tcG9uZW50KSB7XG4gICAgICBjb25zdCBlbGVtZW50ID1cbiAgICAgICAgY29tcG9uZW50LmVsZW1lbnQgfHxcbiAgICAgICAgKGNvbXBvbmVudCBpbnN0YW5jZW9mIEhUTUxFbGVtZW50ID8gY29tcG9uZW50IDogbnVsbCk7XG4gICAgICBpZiAoZWxlbWVudCkge1xuICAgICAgICAvLyBBcHBseSBsYXlvdXQgY2xhc3Nlc1xuICAgICAgICBpZiAobGF5b3V0Q29uZmlnKSBhcHBseUxheW91dENsYXNzZXMoZWxlbWVudCwgbGF5b3V0Q29uZmlnKTtcblxuICAgICAgICAvLyBBcHBseSBsYXlvdXQgaXRlbSBjbGFzc2VzXG4gICAgICAgIGlmIChsYXlvdXRJdGVtQ29uZmlnKSBhcHBseUxheW91dEl0ZW1DbGFzc2VzKGVsZW1lbnQsIGxheW91dEl0ZW1Db25maWcpO1xuXG4gICAgICAgIC8vIEFwcGx5IHN0eWxlXG4gICAgICAgIGlmIChzdHlsZUNvbmZpZyAmJiB0eXBlb2Ygc3R5bGVDb25maWcgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgICAgICBPYmplY3QuYXNzaWduKGVsZW1lbnQuc3R5bGUsIHN0eWxlQ29uZmlnKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIEFwcGx5IGF0dHJpYnV0ZXNcbiAgICAgICAgaWYgKGF0dHJpYnV0ZXNDb25maWcgJiYgdHlwZW9mIGF0dHJpYnV0ZXNDb25maWcgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgICAgICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhhdHRyaWJ1dGVzQ29uZmlnKSkge1xuICAgICAgICAgICAgaWYgKHZhbHVlICE9PSB1bmRlZmluZWQgJiYgdmFsdWUgIT09IG51bGwpIHtcbiAgICAgICAgICAgICAgZWxlbWVudC5zZXRBdHRyaWJ1dGUoa2V5LCB2YWx1ZS50b1N0cmluZygpKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyBBcHBseSBldmVudHNcbiAgICAgICAgaWYgKGZpbmFsRXZlbnRzQ29uZmlnICYmIHR5cGVvZiBmaW5hbEV2ZW50c0NvbmZpZyA9PT0gXCJvYmplY3RcIikge1xuICAgICAgICAgIGlmIChBcnJheS5pc0FycmF5KGZpbmFsRXZlbnRzQ29uZmlnKSkge1xuICAgICAgICAgICAgZm9yIChjb25zdCBldmVudERlZiBvZiBmaW5hbEV2ZW50c0NvbmZpZykge1xuICAgICAgICAgICAgICBpZiAoQXJyYXkuaXNBcnJheShldmVudERlZikgJiYgZXZlbnREZWYubGVuZ3RoID49IDIpIHtcbiAgICAgICAgICAgICAgICBjb25zdCBbZXZlbnROYW1lLCBoYW5kbGVyXSA9IGV2ZW50RGVmO1xuICAgICAgICAgICAgICAgIGlmIChcbiAgICAgICAgICAgICAgICAgIHR5cGVvZiBldmVudE5hbWUgPT09IFwic3RyaW5nXCIgJiZcbiAgICAgICAgICAgICAgICAgIHR5cGVvZiBoYW5kbGVyID09PSBcImZ1bmN0aW9uXCJcbiAgICAgICAgICAgICAgICApIHtcbiAgICAgICAgICAgICAgICAgIGVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihldmVudE5hbWUsIGhhbmRsZXIpO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBmb3IgKGNvbnN0IFtldmVudE5hbWUsIGhhbmRsZXJdIG9mIE9iamVjdC5lbnRyaWVzKFxuICAgICAgICAgICAgICBmaW5hbEV2ZW50c0NvbmZpZyxcbiAgICAgICAgICAgICkpIHtcbiAgICAgICAgICAgICAgaWYgKHR5cGVvZiBoYW5kbGVyID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgICAgICAgICAgICBlbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoZXZlbnROYW1lLCBoYW5kbGVyKTtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiBjb21wb25lbnQ7XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihcIkVycm9yIGNyZWF0aW5nIGNvbXBvbmVudCBpbnN0YW5jZTpcIiwgZXJyb3IpO1xuICAgIHJldHVybiBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZGl2XCIpO1xuICB9XG59XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIElOVEVHUkFURUQgTEFZT1VUIENPTkZJR1VSQVRJT04gKHdpdGggY2FjaGluZylcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBBcHBsaWVzIGxheW91dCBjbGFzc2VzIGJhc2VkIG9uIGNvbmZpZ3VyYXRpb25cbiAqIFVzZXMgaW50ZWdyYXRlZCBjYWNoaW5nIGZvciBvcHRpbWFsIHBlcmZvcm1hbmNlXG4gKi9cbmZ1bmN0aW9uIGFwcGx5TGF5b3V0Q2xhc3NlcyhcbiAgZWxlbWVudDogSFRNTEVsZW1lbnQsXG4gIGxheW91dENvbmZpZzogTGF5b3V0Q29uZmlnLFxuKTogdm9pZCB7XG4gIGlmICghZWxlbWVudCB8fCAhbGF5b3V0Q29uZmlnKSByZXR1cm47XG5cbiAgLy8gQXBwbHkgYmFzZSBsYXlvdXQgdHlwZVxuICBpZiAobGF5b3V0Q29uZmlnLnR5cGUpIHtcbiAgICBhZGRDbGFzcyhlbGVtZW50LCBgJHtQUkVGSVhfV0lUSF9EQVNIfWxheW91dC0tJHtsYXlvdXRDb25maWcudHlwZX1gKTtcbiAgfVxuXG4gIC8vIEFwcGx5IHByb3BlcnRpZXMgd2l0aCBjYWNoaW5nXG4gIGNvbnN0IGxheW91dFR5cGUgPSBsYXlvdXRDb25maWcudHlwZSB8fCBnZXRMYXlvdXRUeXBlKGVsZW1lbnQpO1xuICBpZiAobGF5b3V0VHlwZSkge1xuICAgIGlmIChsYXlvdXRDb25maWcuZ2FwICE9PSB1bmRlZmluZWQpIHtcbiAgICAgIGFkZENsYXNzKFxuICAgICAgICBlbGVtZW50LFxuICAgICAgICBQUkVGSVhfV0lUSF9EQVNIICtcbiAgICAgICAgICBnZXRDYWNoZWRDbGFzc05hbWUobGF5b3V0VHlwZSwgXCJnYXBcIiwgbGF5b3V0Q29uZmlnLmdhcCksXG4gICAgICApO1xuICAgIH1cbiAgICBpZiAobGF5b3V0Q29uZmlnLmFsaWduKSB7XG4gICAgICBhZGRDbGFzcyhcbiAgICAgICAgZWxlbWVudCxcbiAgICAgICAgUFJFRklYX1dJVEhfREFTSCArXG4gICAgICAgICAgZ2V0Q2FjaGVkQ2xhc3NOYW1lKGxheW91dFR5cGUsIFwiYWxpZ25cIiwgbGF5b3V0Q29uZmlnLmFsaWduKSxcbiAgICAgICk7XG4gICAgfVxuICAgIGlmIChsYXlvdXRDb25maWcuanVzdGlmeSkge1xuICAgICAgYWRkQ2xhc3MoXG4gICAgICAgIGVsZW1lbnQsXG4gICAgICAgIFBSRUZJWF9XSVRIX0RBU0ggK1xuICAgICAgICAgIGdldENhY2hlZENsYXNzTmFtZShsYXlvdXRUeXBlLCBcImp1c3RpZnlcIiwgbGF5b3V0Q29uZmlnLmp1c3RpZnkpLFxuICAgICAgKTtcbiAgICB9XG4gIH1cblxuICAvLyBHcmlkLXNwZWNpZmljIHByb3BlcnRpZXNcbiAgaWYgKGxheW91dENvbmZpZy50eXBlID09PSBcImdyaWRcIiB8fCBnZXRMYXlvdXRUeXBlKGVsZW1lbnQpID09PSBcImdyaWRcIikge1xuICAgIGlmICh0eXBlb2YgbGF5b3V0Q29uZmlnLmNvbHVtbnMgPT09IFwibnVtYmVyXCIpIHtcbiAgICAgIGFkZENsYXNzKFxuICAgICAgICBlbGVtZW50LFxuICAgICAgICBQUkVGSVhfV0lUSF9EQVNIICtcbiAgICAgICAgICBnZXRDYWNoZWRDbGFzc05hbWUoXCJncmlkXCIsIFwiY29sc1wiLCBsYXlvdXRDb25maWcuY29sdW1ucyksXG4gICAgICApO1xuICAgIH0gZWxzZSBpZiAobGF5b3V0Q29uZmlnLmNvbHVtbnMgPT09IFwiYXV0by1maWxsXCIpIHtcbiAgICAgIGFkZENsYXNzKFxuICAgICAgICBlbGVtZW50LFxuICAgICAgICBQUkVGSVhfV0lUSF9EQVNIICsgZ2V0Q2FjaGVkQ2xhc3NOYW1lKFwiZ3JpZFwiLCBcImZpbGxcIiwgXCJhdXRvXCIpLFxuICAgICAgKTtcbiAgICB9IGVsc2UgaWYgKGxheW91dENvbmZpZy5jb2x1bW5zID09PSBcImF1dG8tZml0XCIpIHtcbiAgICAgIGFkZENsYXNzKFxuICAgICAgICBlbGVtZW50LFxuICAgICAgICBQUkVGSVhfV0lUSF9EQVNIICsgZ2V0Q2FjaGVkQ2xhc3NOYW1lKFwiZ3JpZFwiLCBcImNvbHNcIiwgXCJhdXRvLWZpdFwiKSxcbiAgICAgICk7XG4gICAgfVxuICAgIGlmIChsYXlvdXRDb25maWcuZGVuc2UpXG4gICAgICBhZGRDbGFzcyhlbGVtZW50LCBgJHtQUkVGSVhfV0lUSF9EQVNIfWxheW91dC0tZ3JpZC1kZW5zZWApO1xuICAgIGlmIChsYXlvdXRDb25maWcuYXV0b0hlaWdodClcbiAgICAgIGFkZENsYXNzKGVsZW1lbnQsIGAke1BSRUZJWF9XSVRIX0RBU0h9bGF5b3V0LS1ncmlkLWF1dG8taGVpZ2h0YCk7XG4gIH1cblxuICAvLyBSb3ctc3BlY2lmaWMgcHJvcGVydGllc1xuICBpZiAobGF5b3V0Q29uZmlnLnR5cGUgPT09IFwicm93XCIgfHwgZ2V0TGF5b3V0VHlwZShlbGVtZW50KSA9PT0gXCJyb3dcIikge1xuICAgIGlmIChsYXlvdXRDb25maWcud3JhcCA9PT0gZmFsc2UgfHwgbGF5b3V0Q29uZmlnLndyYXAgPT09IFwibm93cmFwXCIpIHtcbiAgICAgIGFkZENsYXNzKGVsZW1lbnQsIGAke1BSRUZJWF9XSVRIX0RBU0h9bGF5b3V0LS1yb3ctbm93cmFwYCk7XG4gICAgfSBlbHNlIGlmIChsYXlvdXRDb25maWcud3JhcCA9PT0gXCJyZXZlcnNlXCIpIHtcbiAgICAgIGFkZENsYXNzKGVsZW1lbnQsIGAke1BSRUZJWF9XSVRIX0RBU0h9bGF5b3V0LS1yb3ctd3JhcC1yZXZlcnNlYCk7XG4gICAgfVxuICAgIGlmIChsYXlvdXRDb25maWcubW9iaWxlU3RhY2spXG4gICAgICBhZGRDbGFzcyhlbGVtZW50LCBgJHtQUkVGSVhfV0lUSF9EQVNIfWxheW91dC0tcm93LW1vYmlsZS1zdGFja2ApO1xuICAgIGlmIChsYXlvdXRDb25maWcubW9iaWxlU2Nyb2xsKVxuICAgICAgYWRkQ2xhc3MoZWxlbWVudCwgYCR7UFJFRklYX1dJVEhfREFTSH1sYXlvdXQtLXJvdy1tb2JpbGUtc2Nyb2xsYCk7XG4gIH1cblxuICAvLyBDdXN0b20gY2xhc3Nlc1xuICBpZiAobGF5b3V0Q29uZmlnLmNsYXNzKSB7XG4gICAgbGF5b3V0Q29uZmlnLmNsYXNzXG4gICAgICAuc3BsaXQoXCIgXCIpXG4gICAgICAuZmlsdGVyKEJvb2xlYW4pXG4gICAgICAuZm9yRWFjaCgoY2xzKSA9PiBlbGVtZW50LmNsYXNzTGlzdC5hZGQoY2xzKSk7XG4gIH1cbn1cblxuLyoqXG4gKiBBcHBsaWVzIGxheW91dCBpdGVtIGNsYXNzZXMgYmFzZWQgb24gY29uZmlndXJhdGlvblxuICogVXNlcyBpbnRlZ3JhdGVkIGNhY2hpbmcgZm9yIG9wdGltYWwgcGVyZm9ybWFuY2VcbiAqL1xuZnVuY3Rpb24gYXBwbHlMYXlvdXRJdGVtQ2xhc3NlcyhcbiAgZWxlbWVudDogSFRNTEVsZW1lbnQsXG4gIGl0ZW1Db25maWc6IExheW91dEl0ZW1Db25maWcsXG4pOiB2b2lkIHtcbiAgaWYgKCFlbGVtZW50IHx8ICFpdGVtQ29uZmlnKSByZXR1cm47XG5cbiAgYWRkQ2xhc3MoZWxlbWVudCwgYCR7UFJFRklYX1dJVEhfREFTSH1sYXlvdXRfX2l0ZW1gKTtcblxuICAvLyBXaWR0aCBhbmQgcmVzcG9uc2l2ZSBjbGFzc2VzIHdpdGggY2FjaGluZ1xuICBpZiAoaXRlbUNvbmZpZy53aWR0aCAmJiBpdGVtQ29uZmlnLndpZHRoID49IDEgJiYgaXRlbUNvbmZpZy53aWR0aCA8PSAxMikge1xuICAgIGFkZENsYXNzKFxuICAgICAgZWxlbWVudCxcbiAgICAgIFBSRUZJWF9XSVRIX0RBU0ggKyBnZXRDYWNoZWRDbGFzc05hbWUoXCJpdGVtXCIsIFwiXCIsIGl0ZW1Db25maWcud2lkdGgpLFxuICAgICk7XG4gIH1cbiAgaWYgKGl0ZW1Db25maWcuc20pXG4gICAgYWRkQ2xhc3MoXG4gICAgICBlbGVtZW50LFxuICAgICAgUFJFRklYX1dJVEhfREFTSCArIGdldENhY2hlZENsYXNzTmFtZShcIml0ZW1cIiwgXCJzbVwiLCBpdGVtQ29uZmlnLnNtKSxcbiAgICApO1xuICBpZiAoaXRlbUNvbmZpZy5tZClcbiAgICBhZGRDbGFzcyhcbiAgICAgIGVsZW1lbnQsXG4gICAgICBQUkVGSVhfV0lUSF9EQVNIICsgZ2V0Q2FjaGVkQ2xhc3NOYW1lKFwiaXRlbVwiLCBcIm1kXCIsIGl0ZW1Db25maWcubWQpLFxuICAgICk7XG4gIGlmIChpdGVtQ29uZmlnLmxnKVxuICAgIGFkZENsYXNzKFxuICAgICAgZWxlbWVudCxcbiAgICAgIFBSRUZJWF9XSVRIX0RBU0ggKyBnZXRDYWNoZWRDbGFzc05hbWUoXCJpdGVtXCIsIFwibGdcIiwgaXRlbUNvbmZpZy5sZyksXG4gICAgKTtcbiAgaWYgKGl0ZW1Db25maWcueGwpXG4gICAgYWRkQ2xhc3MoXG4gICAgICBlbGVtZW50LFxuICAgICAgUFJFRklYX1dJVEhfREFTSCArIGdldENhY2hlZENsYXNzTmFtZShcIml0ZW1cIiwgXCJ4bFwiLCBpdGVtQ29uZmlnLnhsKSxcbiAgICApO1xuXG4gIC8vIEdyaWQgc3BhbiBjbGFzc2VzXG4gIGlmIChpdGVtQ29uZmlnLnNwYW4pXG4gICAgYWRkQ2xhc3MoXG4gICAgICBlbGVtZW50LFxuICAgICAgUFJFRklYX1dJVEhfREFTSCArIGdldENhY2hlZENsYXNzTmFtZShcIml0ZW1cIiwgXCJzcGFuXCIsIGl0ZW1Db25maWcuc3BhbiksXG4gICAgKTtcbiAgaWYgKGl0ZW1Db25maWcucm93U3BhbilcbiAgICBhZGRDbGFzcyhcbiAgICAgIGVsZW1lbnQsXG4gICAgICBQUkVGSVhfV0lUSF9EQVNIICtcbiAgICAgICAgZ2V0Q2FjaGVkQ2xhc3NOYW1lKFwiaXRlbVwiLCBcInJvdy1zcGFuXCIsIGl0ZW1Db25maWcucm93U3BhbiksXG4gICAgKTtcblxuICAvLyBPcmRlciBhbmQgYWxpZ25tZW50XG4gIGlmIChpdGVtQ29uZmlnLm9yZGVyKVxuICAgIGFkZENsYXNzKFxuICAgICAgZWxlbWVudCxcbiAgICAgIFBSRUZJWF9XSVRIX0RBU0ggKyBnZXRDYWNoZWRDbGFzc05hbWUoXCJpdGVtXCIsIFwib3JkZXJcIiwgaXRlbUNvbmZpZy5vcmRlciksXG4gICAgKTtcbiAgaWYgKGl0ZW1Db25maWcuYWxpZ24pXG4gICAgYWRkQ2xhc3MoXG4gICAgICBlbGVtZW50LFxuICAgICAgUFJFRklYX1dJVEhfREFTSCArIGdldENhY2hlZENsYXNzTmFtZShcIml0ZW1cIiwgXCJzZWxmXCIsIGl0ZW1Db25maWcuYWxpZ24pLFxuICAgICk7XG4gIGlmIChpdGVtQ29uZmlnLmF1dG8pXG4gICAgYWRkQ2xhc3MoZWxlbWVudCwgYCR7UFJFRklYX1dJVEhfREFTSH1sYXlvdXRfX2l0ZW0tLWF1dG9gKTtcbn1cblxuLyoqXG4gKiBHZXRzIHRoZSBsYXlvdXQgdHlwZSBmcm9tIGVsZW1lbnQgY2xhc3Nlc1xuICovXG5mdW5jdGlvbiBnZXRMYXlvdXRUeXBlKGVsZW1lbnQ6IEhUTUxFbGVtZW50KTogc3RyaW5nIHtcbiAgcmV0dXJuIGhhc0NsYXNzKGVsZW1lbnQsIGAke1BSRUZJWF9XSVRIX0RBU0h9bGF5b3V0LS1zdGFja2ApXG4gICAgPyBcInN0YWNrXCJcbiAgICA6IGhhc0NsYXNzKGVsZW1lbnQsIGAke1BSRUZJWF9XSVRIX0RBU0h9bGF5b3V0LS1yb3dgKVxuICAgICAgPyBcInJvd1wiXG4gICAgICA6IGhhc0NsYXNzKGVsZW1lbnQsIGAke1BSRUZJWF9XSVRIX0RBU0h9bGF5b3V0LS1ncmlkYClcbiAgICAgICAgPyBcImdyaWRcIlxuICAgICAgICA6IFwiXCI7XG59XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIFVOSUZJRUQgU0NIRU1BIFBST0NFU1NPUlNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBQcm9jZXNzZXMgYXJyYXktYmFzZWQgc2NoZW1hIGRlZmluaXRpb25zXG4gKiBPcHRpbWl6ZWQgd2l0aCBwYXJhbWV0ZXIgZXh0cmFjdGlvbiBhbmQgaW50ZWdyYXRlZCBjb25maWd1cmF0aW9uXG4gKi9cbmZ1bmN0aW9uIHByb2Nlc3NBcnJheVNjaGVtYShcbiAgc2NoZW1hOiBTY2hlbWFJdGVtW10gfCBhbnksXG4gIHBhcmVudEVsZW1lbnQ6IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGwsXG4gIGxldmVsOiBudW1iZXIgPSAwLFxuICBvcHRpb25zOiBMYXlvdXRPcHRpb25zID0ge30sXG4pOiBMYXlvdXRSZXN1bHQge1xuICBsZXZlbCsrO1xuICBjb25zdCBsYXlvdXQ6IFJlY29yZDxzdHJpbmcsIGFueT4gPSB7fTtcbiAgY29uc3QgY29tcG9uZW50czogQXJyYXk8W3N0cmluZywgYW55XT4gPSBbXTtcbiAgY29uc3QgZnJhZ21lbnQgPSBjcmVhdGVGcmFnbWVudCgpO1xuICBsZXQgY29tcG9uZW50ID0gbnVsbDtcblxuICBpZiAoIUFycmF5LmlzQXJyYXkoc2NoZW1hKSkge1xuICAgIHJldHVybiBjcmVhdGVMYXlvdXRSZXN1bHQobGF5b3V0KTtcbiAgfVxuXG4gIGNvbnN0IGRlZmF1bHRDcmVhdG9yID0gKG9wdGlvbnMgYXMgYW55KS5jcmVhdG9yIHx8IGNyZWF0ZUVsZW1lbnQ7XG5cbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBzY2hlbWEubGVuZ3RoOyBpKyspIHtcbiAgICBjb25zdCBpdGVtID0gc2NoZW1hW2ldO1xuICAgIGlmICghaXRlbSkgY29udGludWU7XG5cbiAgICAvLyBIYW5kbGUgbmVzdGVkIGFycmF5c1xuICAgIGlmIChBcnJheS5pc0FycmF5KGl0ZW0pKSB7XG4gICAgICBjb25zdCBjb250YWluZXIgPSBjb21wb25lbnQgfHwgcGFyZW50RWxlbWVudDtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IHByb2Nlc3NBcnJheVNjaGVtYShpdGVtLCBjb250YWluZXIsIGxldmVsLCBvcHRpb25zKTtcbiAgICAgIC8vIE1lcmdlIG5lc3RlZCBjb21wb25lbnRzIGFycmF5IGluc3RlYWQgb2Ygb3ZlcndyaXRpbmdcbiAgICAgIGlmIChBcnJheS5pc0FycmF5KHJlc3VsdC5sYXlvdXQuY29tcG9uZW50cykpIHtcbiAgICAgICAgY29tcG9uZW50cy5wdXNoKC4uLnJlc3VsdC5sYXlvdXQuY29tcG9uZW50cyk7XG4gICAgICAgIGRlbGV0ZSByZXN1bHQubGF5b3V0LmNvbXBvbmVudHM7XG4gICAgICB9XG4gICAgICBPYmplY3QuYXNzaWduKGxheW91dCwgcmVzdWx0LmxheW91dCk7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG5cbiAgICAvLyBVc2Ugb3B0aW1pemVkIHBhcmFtZXRlciBleHRyYWN0aW9uXG4gICAgY29uc3Qge1xuICAgICAgY3JlYXRvcixcbiAgICAgIG5hbWUsXG4gICAgICBvcHRpb25zOiBpdGVtT3B0aW9ucyxcbiAgICAgIGNvbnN1bWVkLFxuICAgIH0gPSBleHRyYWN0UGFyYW1ldGVycyhzY2hlbWEsIGksIGRlZmF1bHRDcmVhdG9yKTtcblxuICAgIGlmICghY3JlYXRvcikge1xuICAgICAgY29uc29sZS53YXJuKFwiU2tpcHBpbmcgdW5zdXBwb3J0ZWQgaXRlbSB0eXBlOlwiLCBpdGVtKTtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cblxuICAgIC8vIERlZmF1bHQgdG8gZGl2IGZvciBjcmVhdGVFbGVtZW50XG4gICAgaWYgKGNyZWF0b3IgPT09IGNyZWF0ZUVsZW1lbnQgJiYgIShcInRhZ1wiIGluIGl0ZW1PcHRpb25zKSkge1xuICAgICAgaXRlbU9wdGlvbnMudGFnID0gXCJkaXZcIjtcbiAgICB9XG5cbiAgICAvLyBBZHZhbmNlIGluZGV4IGJ5IGNvbnN1bWVkIGl0ZW1zIG1pbnVzIDEgKGxvb3AgaW5jcmVtZW50IGhhbmRsZXMgdGhlICsxKVxuICAgIGkgKz0gY29uc3VtZWQgLSAxO1xuXG4gICAgLy8gUHJvY2VzcyBvcHRpb25zIHdpdGggcHJlZml4IC0gb3B0aW1pemVkIGRlY2lzaW9uIGxvZ2ljXG4gICAgY29uc3Qgc2hvdWxkQXBwbHlQcmVmaXggPVxuICAgICAgXCJwcmVmaXhcIiBpbiBpdGVtT3B0aW9ucyA/IGl0ZW1PcHRpb25zLnByZWZpeCA6IG9wdGlvbnMucHJlZml4ICE9PSBmYWxzZTtcblxuICAgIC8vIEZhc3QgcGF0aDogcHJvY2VzcyBvbmx5IHdoZW4gbmVlZGVkXG4gICAgY29uc3QgcHJvY2Vzc2VkT3B0aW9ucyA9XG4gICAgICBzaG91bGRBcHBseVByZWZpeCB8fCBpdGVtT3B0aW9ucy5yYXdDbGFzc1xuICAgICAgICA/IHByb2Nlc3NDbGFzc05hbWVzKGl0ZW1PcHRpb25zLCAhc2hvdWxkQXBwbHlQcmVmaXgpXG4gICAgICAgIDogaXRlbU9wdGlvbnM7IC8vIE5vIGNvcHkgbmVlZGVkIGlmIG5vIHByb2Nlc3NpbmdcblxuICAgIC8vIEFkZCBuYW1lIHRvIG9wdGlvbnMgaWYgbmVlZGVkXG4gICAgaWYgKFxuICAgICAgbmFtZSAmJlxuICAgICAgIShcIm5hbWVcIiBpbiBwcm9jZXNzZWRPcHRpb25zKSAmJlxuICAgICAgIShjcmVhdG9yID09PSBjcmVhdGVFbGVtZW50IHx8IChjcmVhdG9yIGFzIGFueSkuaXNFbGVtZW50KVxuICAgICkge1xuICAgICAgcHJvY2Vzc2VkT3B0aW9ucy5uYW1lID0gbmFtZTtcbiAgICB9XG5cbiAgICAvLyBDcmVhdGUgY29tcG9uZW50XG4gICAgY29tcG9uZW50ID0gY3JlYXRlQ29tcG9uZW50SW5zdGFuY2UoY3JlYXRvciwgcHJvY2Vzc2VkT3B0aW9ucyk7XG4gICAgY29uc3QgZWxlbWVudCA9IGlzQ29tcG9uZW50KGNvbXBvbmVudCkgPyBjb21wb25lbnQuZWxlbWVudCA6IGNvbXBvbmVudDtcblxuICAgIGlmIChsZXZlbCA9PT0gMSkgbGF5b3V0LmVsZW1lbnQgPSBlbGVtZW50O1xuICAgIGlmIChuYW1lKSB7XG4gICAgICBsYXlvdXRbbmFtZV0gPSBjb21wb25lbnQ7XG4gICAgICBjb21wb25lbnRzLnB1c2goW25hbWUsIGNvbXBvbmVudF0pO1xuICAgIH1cblxuICAgIC8vIEFwcGVuZCB0byBET01cbiAgICBpZiAoY29tcG9uZW50KSB7XG4gICAgICBpZiAoXCJpbnNlcnRcIiBpbiBjb21wb25lbnQgJiYgdHlwZW9mIGNvbXBvbmVudC5pbnNlcnQgPT09IFwiZnVuY3Rpb25cIikge1xuICAgICAgICBjb21wb25lbnQuaW5zZXJ0KGZyYWdtZW50KTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGZyYWdtZW50LmFwcGVuZENoaWxkKGVsZW1lbnQpO1xuICAgICAgfVxuXG4gICAgICBpZiAocGFyZW50RWxlbWVudCkge1xuICAgICAgICBjb21wb25lbnQuX2NvbnRhaW5lciA9IHBhcmVudEVsZW1lbnQ7XG4gICAgICAgIGlmIChcbiAgICAgICAgICBcIm9uSW5zZXJ0ZWRcIiBpbiBjb21wb25lbnQgJiZcbiAgICAgICAgICB0eXBlb2YgY29tcG9uZW50Lm9uSW5zZXJ0ZWQgPT09IFwiZnVuY3Rpb25cIlxuICAgICAgICApIHtcbiAgICAgICAgICBjb21wb25lbnQub25JbnNlcnRlZChwYXJlbnRFbGVtZW50KTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8vIEFwcGVuZCBmcmFnbWVudCB0byBwYXJlbnRcbiAgaWYgKHBhcmVudEVsZW1lbnQgJiYgZnJhZ21lbnQuaGFzQ2hpbGROb2RlcygpKSB7XG4gICAgY29uc3Qgd3JhcHBlciA9IGlzQ29tcG9uZW50KHBhcmVudEVsZW1lbnQpXG4gICAgICA/IHBhcmVudEVsZW1lbnQuZWxlbWVudFxuICAgICAgOiBwYXJlbnRFbGVtZW50O1xuICAgIHdyYXBwZXIuYXBwZW5kQ2hpbGQoZnJhZ21lbnQpO1xuICB9XG5cbiAgLy8gUmVsZWFzZSBmcmFnbWVudCBiYWNrIHRvIHBvb2xcbiAgcmVsZWFzZUZyYWdtZW50KGZyYWdtZW50KTtcblxuICBsYXlvdXQuY29tcG9uZW50cyA9IGNvbXBvbmVudHM7XG4gIHJldHVybiBjcmVhdGVMYXlvdXRSZXN1bHQobGF5b3V0KTtcbn1cblxuLyoqXG4gKiBQcm9jZXNzZXMgb2JqZWN0LWJhc2VkIHNjaGVtYSBkZWZpbml0aW9uc1xuICogU2ltcGxpZmllZCBhbmQgb3B0aW1pemVkIGZvciBiZXR0ZXIgcGVyZm9ybWFuY2VcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc09iamVjdFNjaGVtYShcbiAgc2NoZW1hOiBSZWNvcmQ8c3RyaW5nLCBhbnk+IHwgc3RyaW5nLFxuICBwYXJlbnRFbGVtZW50OiBIVE1MRWxlbWVudCB8IG51bGwgPSBudWxsLFxuICBvcHRpb25zOiBMYXlvdXRPcHRpb25zID0ge30sXG4pOiBMYXlvdXRSZXN1bHQge1xuICBjb25zdCBsYXlvdXQ6IFJlY29yZDxzdHJpbmcsIGFueT4gPSB7fTtcbiAgY29uc3QgZGVmYXVsdENyZWF0b3IgPSBvcHRpb25zLmNyZWF0b3IgfHwgY3JlYXRlRWxlbWVudDtcblxuICAvLyBIYW5kbGUgcm9vdCBlbGVtZW50IGNyZWF0aW9uXG4gIGlmICgoc2NoZW1hIGFzIGFueSkuZWxlbWVudCAmJiAhcGFyZW50RWxlbWVudCkge1xuICAgIGNvbnN0IGVsZW1lbnREZWYgPSAoc2NoZW1hIGFzIGFueSkuZWxlbWVudDtcbiAgICBjb25zdCBjcmVhdGVFbGVtZW50Rm4gPSBlbGVtZW50RGVmLmNyZWF0b3IgfHwgZGVmYXVsdENyZWF0b3I7XG5cbiAgICBjb25zdCBlbGVtZW50T3B0aW9ucyA9IGVsZW1lbnREZWYub3B0aW9ucyB8fCB7fTtcbiAgICBjb25zdCBwcm9jZXNzZWRPcHRpb25zID1cbiAgICAgIG9wdGlvbnMucHJlZml4ICE9PSBmYWxzZVxuICAgICAgICA/IHByb2Nlc3NDbGFzc05hbWVzKGVsZW1lbnRPcHRpb25zKVxuICAgICAgICA6IHsgLi4uZWxlbWVudE9wdGlvbnMgfTtcblxuICAgIGNvbnN0IHJvb3RDb21wb25lbnQgPSBjcmVhdGVDb21wb25lbnRJbnN0YW5jZShcbiAgICAgIGNyZWF0ZUVsZW1lbnRGbixcbiAgICAgIHByb2Nlc3NlZE9wdGlvbnMsXG4gICAgKTtcbiAgICBsYXlvdXQuZWxlbWVudCA9IHJvb3RDb21wb25lbnQ7XG4gICAgaWYgKGVsZW1lbnREZWYubmFtZSkgbGF5b3V0W2VsZW1lbnREZWYubmFtZV0gPSByb290Q29tcG9uZW50O1xuXG4gICAgLy8gUHJvY2VzcyBjaGlsZHJlblxuICAgIGlmIChlbGVtZW50RGVmLmNoaWxkcmVuKSB7XG4gICAgICBjb25zdCByb290RWxlbWVudCA9IGlzQ29tcG9uZW50KHJvb3RDb21wb25lbnQpXG4gICAgICAgID8gcm9vdENvbXBvbmVudC5lbGVtZW50XG4gICAgICAgIDogcm9vdENvbXBvbmVudDtcbiAgICAgIGNvbnN0IGNoaWxkUmVzdWx0ID0gcHJvY2Vzc09iamVjdFNjaGVtYShcbiAgICAgICAgZWxlbWVudERlZi5jaGlsZHJlbixcbiAgICAgICAgcm9vdEVsZW1lbnQsXG4gICAgICAgIG9wdGlvbnMsXG4gICAgICApO1xuICAgICAgT2JqZWN0LmFzc2lnbihsYXlvdXQsIGNoaWxkUmVzdWx0LmxheW91dCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGNyZWF0ZUxheW91dFJlc3VsdChsYXlvdXQpO1xuICB9XG5cbiAgLy8gUHJvY2VzcyBub3JtYWwgc2NoZW1hIGVsZW1lbnRzXG4gIGNvbnN0IGZyYWdtZW50ID0gcGFyZW50RWxlbWVudCA/IGNyZWF0ZUZyYWdtZW50KCkgOiBudWxsO1xuXG4gIGZvciAoY29uc3Qga2V5IGluIHNjaGVtYSBhcyBSZWNvcmQ8c3RyaW5nLCBhbnk+KSB7XG4gICAgY29uc3QgZGVmID0gKHNjaGVtYSBhcyBSZWNvcmQ8c3RyaW5nLCBhbnk+KVtrZXldO1xuICAgIGlmICghZGVmKSBjb250aW51ZTtcblxuICAgIGNvbnN0IGVsZW1lbnRDcmVhdG9yID0gZGVmLmNyZWF0b3IgfHwgZGVmYXVsdENyZWF0b3I7XG4gICAgY29uc3QgZWxlbWVudE9wdGlvbnMgPSBkZWYub3B0aW9ucyB8fCB7fTtcbiAgICBjb25zdCBzaG91bGRBcHBseVByZWZpeCA9XG4gICAgICBcInByZWZpeFwiIGluIGVsZW1lbnRPcHRpb25zXG4gICAgICAgID8gZWxlbWVudE9wdGlvbnMucHJlZml4XG4gICAgICAgIDogb3B0aW9ucy5wcmVmaXggIT09IGZhbHNlO1xuICAgIGNvbnN0IHByb2Nlc3NlZE9wdGlvbnMgPSBzaG91bGRBcHBseVByZWZpeFxuICAgICAgPyBwcm9jZXNzQ2xhc3NOYW1lcyhlbGVtZW50T3B0aW9ucylcbiAgICAgIDogeyAuLi5lbGVtZW50T3B0aW9ucyB9O1xuXG4gICAgaWYgKCFkZWYubmFtZSAmJiBrZXkgIT09IFwiZWxlbWVudFwiKSB7XG4gICAgICBkZWYubmFtZSA9IGtleTtcbiAgICB9XG5cbiAgICBjb25zdCBjcmVhdGVkID0gY3JlYXRlQ29tcG9uZW50SW5zdGFuY2UoZWxlbWVudENyZWF0b3IsIHByb2Nlc3NlZE9wdGlvbnMpO1xuICAgIGxheW91dFtrZXldID0gY3JlYXRlZDtcbiAgICBpZiAoZGVmLm5hbWUgJiYgZGVmLm5hbWUgIT09IGtleSkgbGF5b3V0W2RlZi5uYW1lXSA9IGNyZWF0ZWQ7XG5cbiAgICBjb25zdCBlbGVtZW50ID0gaXNDb21wb25lbnQoY3JlYXRlZCkgPyBjcmVhdGVkLmVsZW1lbnQgOiBjcmVhdGVkO1xuICAgIGlmIChmcmFnbWVudCkgZnJhZ21lbnQuYXBwZW5kQ2hpbGQoZWxlbWVudCk7XG5cbiAgICAvLyBQcm9jZXNzIGNoaWxkcmVuXG4gICAgaWYgKGRlZi5jaGlsZHJlbikge1xuICAgICAgY29uc3QgY2hpbGRSZXN1bHQgPSBwcm9jZXNzT2JqZWN0U2NoZW1hKGRlZi5jaGlsZHJlbiwgZWxlbWVudCwgb3B0aW9ucyk7XG4gICAgICBPYmplY3QuYXNzaWduKGxheW91dCwgY2hpbGRSZXN1bHQubGF5b3V0KTtcbiAgICB9XG4gIH1cblxuICAvLyBBcHBlbmQgdG8gcGFyZW50XG4gIGlmIChwYXJlbnRFbGVtZW50ICYmIGZyYWdtZW50KSB7XG4gICAgY29uc3QgcGFyZW50RG9tID0gaXNDb21wb25lbnQocGFyZW50RWxlbWVudClcbiAgICAgID8gcGFyZW50RWxlbWVudC5lbGVtZW50XG4gICAgICA6IHBhcmVudEVsZW1lbnQ7XG4gICAgcGFyZW50RG9tLmFwcGVuZENoaWxkKGZyYWdtZW50KTtcbiAgICByZWxlYXNlRnJhZ21lbnQoZnJhZ21lbnQpO1xuICB9XG5cbiAgcmV0dXJuIGNyZWF0ZUxheW91dFJlc3VsdChsYXlvdXQpO1xufVxuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBMQVlPVVQgUkVTVUxUIENSRUFUSU9OXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbi8qKlxuICogRmxhdHRlbnMgYSBuZXN0ZWQgbGF5b3V0IGludG8gYSBzaW1wbGUgb2JqZWN0IHdpdGggZWxlbWVudCBhbmQgY29tcG9uZW50IHJlZmVyZW5jZXNcbiAqL1xuZnVuY3Rpb24gZmxhdHRlbkxheW91dChsYXlvdXQ6IFJlY29yZDxzdHJpbmcsIGFueT4pOiBSZWNvcmQ8c3RyaW5nLCBhbnk+IHtcbiAgY29uc3QgZmxhdHRlbmVkOiBSZWNvcmQ8c3RyaW5nLCBhbnk+ID0ge307XG5cbiAgaWYgKCFsYXlvdXQgfHwgdHlwZW9mIGxheW91dCAhPT0gXCJvYmplY3RcIikgcmV0dXJuIGZsYXR0ZW5lZDtcblxuICBmb3IgKGNvbnN0IGtleSBpbiBsYXlvdXQpIHtcbiAgICBjb25zdCB2YWx1ZSA9IGxheW91dFtrZXldO1xuICAgIGlmIChcbiAgICAgIHZhbHVlICYmXG4gICAgICB0eXBlb2YgdmFsdWUgIT09IFwiZnVuY3Rpb25cIiAmJlxuICAgICAgKHZhbHVlIGluc3RhbmNlb2YgSFRNTEVsZW1lbnQgfHxcbiAgICAgICAgKHR5cGVvZiBTVkdFbGVtZW50ICE9PSBcInVuZGVmaW5lZFwiICYmIHZhbHVlIGluc3RhbmNlb2YgU1ZHRWxlbWVudCkgfHxcbiAgICAgICAgaXNDb21wb25lbnQodmFsdWUpKVxuICAgICkge1xuICAgICAgZmxhdHRlbmVkW2tleV0gPSB2YWx1ZTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gZmxhdHRlbmVkO1xufVxuXG4vKipcbiAqIENyZWF0ZXMgYSBsYXlvdXQgcmVzdWx0IG9iamVjdCB3aXRoIHV0aWxpdHkgZnVuY3Rpb25zXG4gKi9cbmZ1bmN0aW9uIGNyZWF0ZUxheW91dFJlc3VsdChsYXlvdXQ6IFJlY29yZDxzdHJpbmcsIGFueT4pOiBMYXlvdXRSZXN1bHQge1xuICBjb25zdCBmbGF0dGVuZWRDb21wb25lbnRzID0gZmxhdHRlbkxheW91dChsYXlvdXQpO1xuXG4gIHJldHVybiB7XG4gICAgbGF5b3V0LFxuICAgIGVsZW1lbnQ6IGxheW91dC5lbGVtZW50LFxuICAgIGNvbXBvbmVudDogZmxhdHRlbmVkQ29tcG9uZW50cyxcblxuICAgIGdldChuYW1lOiBzdHJpbmcpOiBhbnkge1xuICAgICAgcmV0dXJuIGxheW91dFtuYW1lXSA/PyBudWxsO1xuICAgIH0sXG5cbiAgICBnZXRBbGwoKTogUmVjb3JkPHN0cmluZywgYW55PiB7XG4gICAgICByZXR1cm4gZmxhdHRlbmVkQ29tcG9uZW50cztcbiAgICB9LFxuXG4gICAgZGVzdHJveSgpOiB2b2lkIHtcbiAgICAgIC8vIFRyYWNrIGRlc3Ryb3llZCBjb21wb25lbnRzIHRvIGF2b2lkIGRvdWJsZS1kZXN0cm95XG4gICAgICBjb25zdCBkZXN0cm95ZWQgPSBuZXcgU2V0PGFueT4oKTtcblxuICAgICAgLy8gSGVscGVyIHRvIHNhZmVseSBkZXN0cm95IGEgY29tcG9uZW50XG4gICAgICBjb25zdCBkZXN0cm95Q29tcG9uZW50ID0gKGNvbXBvbmVudDogYW55KTogdm9pZCA9PiB7XG4gICAgICAgIGlmICghY29tcG9uZW50IHx8IGRlc3Ryb3llZC5oYXMoY29tcG9uZW50KSkgcmV0dXJuO1xuXG4gICAgICAgIC8vIFNraXAgcGxhaW4gSFRNTCBlbGVtZW50cyBhbmQgbm9uLW9iamVjdHNcbiAgICAgICAgaWYgKGNvbXBvbmVudCBpbnN0YW5jZW9mIEhUTUxFbGVtZW50IHx8IHR5cGVvZiBjb21wb25lbnQgIT09IFwib2JqZWN0XCIpXG4gICAgICAgICAgcmV0dXJuO1xuXG4gICAgICAgIC8vIENoZWNrIGlmIGl0J3MgYSBjb21wb25lbnQgd2l0aCBkZXN0cm95IG1ldGhvZFxuICAgICAgICBpZiAodHlwZW9mIGNvbXBvbmVudC5kZXN0cm95ID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgICAgICBkZXN0cm95ZWQuYWRkKGNvbXBvbmVudCk7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIGNvbXBvbmVudC5kZXN0cm95KCk7XG4gICAgICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICAgICAgLy8gSWdub3JlIGRlc3Ryb3kgZXJyb3JzIC0gY29tcG9uZW50IG1heSBhbHJlYWR5IGJlIGNsZWFuZWQgdXBcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH07XG5cbiAgICAgIC8vIEZpcnN0IGRlc3Ryb3kgY29tcG9uZW50cyBmcm9tIHRoZSBjb21wb25lbnRzIGFycmF5IChpZiBwcmVzZW50KVxuICAgICAgLy8gVGhpcyBlbnN1cmVzIGFsbCBuYW1lZCBjb21wb25lbnRzIGFyZSBkZXN0cm95ZWQgZXZlbiBpZiBuZXN0ZWRcbiAgICAgIGlmIChBcnJheS5pc0FycmF5KGxheW91dC5jb21wb25lbnRzKSkge1xuICAgICAgICBmb3IgKGNvbnN0IFtuYW1lLCBjb21wb25lbnRdIG9mIGxheW91dC5jb21wb25lbnRzKSB7XG4gICAgICAgICAgZGVzdHJveUNvbXBvbmVudChjb21wb25lbnQpO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIFRoZW4gaXRlcmF0ZSBvdmVyIGFsbCBsYXlvdXQga2V5cyBmb3IgYW55IG1pc3NlZCBjb21wb25lbnRzXG4gICAgICBmb3IgKGNvbnN0IGtleSBpbiBsYXlvdXQpIHtcbiAgICAgICAgaWYgKGtleSA9PT0gXCJlbGVtZW50XCIgfHwga2V5ID09PSBcImNvbXBvbmVudHNcIikgY29udGludWU7XG4gICAgICAgIGRlc3Ryb3lDb21wb25lbnQobGF5b3V0W2tleV0pO1xuICAgICAgfVxuXG4gICAgICAvLyBDbGVhciB0aGUgY29tcG9uZW50cyBhcnJheVxuICAgICAgaWYgKEFycmF5LmlzQXJyYXkobGF5b3V0LmNvbXBvbmVudHMpKSB7XG4gICAgICAgIGxheW91dC5jb21wb25lbnRzLmxlbmd0aCA9IDA7XG4gICAgICB9XG5cbiAgICAgIC8vIFJlbW92ZSByb290IGVsZW1lbnQgZnJvbSBET01cbiAgICAgIGlmIChsYXlvdXQuZWxlbWVudCkge1xuICAgICAgICBjb25zdCBlbGVtZW50ID0gaXNDb21wb25lbnQobGF5b3V0LmVsZW1lbnQpXG4gICAgICAgICAgPyBsYXlvdXQuZWxlbWVudC5lbGVtZW50XG4gICAgICAgICAgOiBsYXlvdXQuZWxlbWVudDtcbiAgICAgICAgaWYgKGVsZW1lbnQgJiYgZWxlbWVudC5wYXJlbnROb2RlKSB7XG4gICAgICAgICAgZWxlbWVudC5wYXJlbnROb2RlLnJlbW92ZUNoaWxkKGVsZW1lbnQpO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIENsZWFyIHJlZmVyZW5jZXMgdG8gaGVscCBHQ1xuICAgICAgZm9yIChjb25zdCBrZXkgaW4gbGF5b3V0KSB7XG4gICAgICAgIGRlbGV0ZSBsYXlvdXRba2V5XTtcbiAgICAgIH1cbiAgICB9LFxuICB9O1xufVxuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBVTklGSUVEIEVOVFJZIFBPSU5UXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbi8qKlxuICogQ3JlYXRlcyBhIGxheW91dCBmcm9tIHZhcmlvdXMgc2NoZW1hIGZvcm1hdHNcbiAqIFVuaWZpZWQgcHJvY2Vzc29yIGZvciBhcnJheXMsIG9iamVjdHMsIEpTWCwgYW5kIEhUTUwgc3RyaW5nc1xuICovXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlTGF5b3V0KFxuICBzY2hlbWE6IGFueSxcbiAgcGFyZW50RWxlbWVudDogSFRNTEVsZW1lbnQgfCBudWxsID0gbnVsbCxcbiAgb3B0aW9uczogTGF5b3V0T3B0aW9ucyA9IHt9LFxuKTogTGF5b3V0UmVzdWx0IHtcbiAgLy8gSGFuZGxlIGZ1bmN0aW9uIHNjaGVtYXNcbiAgaWYgKHR5cGVvZiBzY2hlbWEgPT09IFwiZnVuY3Rpb25cIikge1xuICAgIHNjaGVtYSA9IHNjaGVtYSgpO1xuICB9XG5cbiAgLy8gSGFuZGxlIEhUTUwgc3RyaW5nIHNjaGVtYXNcbiAgaWYgKHR5cGVvZiBzY2hlbWEgPT09IFwic3RyaW5nXCIpIHtcbiAgICBjb25zdCB0ZW1wbGF0ZSA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJ0ZW1wbGF0ZVwiKTtcbiAgICB0ZW1wbGF0ZS5pbm5lckhUTUwgPSBzY2hlbWEudHJpbSgpO1xuICAgIGNvbnN0IGZyYWdtZW50ID0gdGVtcGxhdGUuY29udGVudDtcblxuICAgIGlmIChwYXJlbnRFbGVtZW50ICYmIGZyYWdtZW50Lmhhc0NoaWxkTm9kZXMoKSkge1xuICAgICAgcGFyZW50RWxlbWVudC5hcHBlbmRDaGlsZChmcmFnbWVudCk7XG4gICAgfVxuXG4gICAgY29uc3QgbGF5b3V0ID0geyBlbGVtZW50OiBmcmFnbWVudC5maXJzdEVsZW1lbnRDaGlsZCBhcyBIVE1MRWxlbWVudCB9O1xuICAgIHJldHVybiBjcmVhdGVMYXlvdXRSZXN1bHQobGF5b3V0KTtcbiAgfVxuXG4gIC8vIEhhbmRsZSBKU1gtbGlrZSBzY2hlbWFzIChhcnJheSB3aXRoIGZ1bmN0aW9uLCBzdHJpbmcsIG9iamVjdCBwYXR0ZXJuKVxuICBpZiAoXG4gICAgQXJyYXkuaXNBcnJheShzY2hlbWEpICYmXG4gICAgc2NoZW1hLmxlbmd0aCA+PSAzICYmXG4gICAgdHlwZW9mIHNjaGVtYVswXSA9PT0gXCJmdW5jdGlvblwiICYmXG4gICAgdHlwZW9mIHNjaGVtYVsxXSA9PT0gXCJzdHJpbmdcIiAmJlxuICAgIGlzT2JqZWN0KHNjaGVtYVsyXSlcbiAgKSB7XG4gICAgcmV0dXJuIHByb2Nlc3NBcnJheVNjaGVtYShzY2hlbWEsIHBhcmVudEVsZW1lbnQsIDAsIG9wdGlvbnMpO1xuICB9XG5cbiAgLy8gUm91dGUgdG8gYXBwcm9wcmlhdGUgcHJvY2Vzc29yXG4gIHJldHVybiBBcnJheS5pc0FycmF5KHNjaGVtYSlcbiAgICA/IHByb2Nlc3NBcnJheVNjaGVtYShzY2hlbWEsIHBhcmVudEVsZW1lbnQsIDAsIG9wdGlvbnMpXG4gICAgOiBwcm9jZXNzT2JqZWN0U2NoZW1hKHNjaGVtYSwgcGFyZW50RWxlbWVudCwgb3B0aW9ucyk7XG59XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIEVYUE9SVFNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLy8gQ2xlYXIgZnVuY3Rpb25zIGZvciB0aGUgdW5pZmllZCBzeXN0ZW1cbmV4cG9ydCBmdW5jdGlvbiBjbGVhckNsYXNzQ2FjaGUoKTogdm9pZCB7XG4gIGNsYXNzQ2FjaGUuY2xlYXIoKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNsZWFyRnJhZ21lbnRQb29sKCk6IHZvaWQge1xuICBmcmFnbWVudFBvb2wuY2xlYXIoKTtcbn1cblxuZXhwb3J0IHtcbiAgcHJvY2Vzc0NsYXNzTmFtZXMsXG4gIGlzQ29tcG9uZW50LFxuICBmbGF0dGVuTGF5b3V0LFxuICBhcHBseUxheW91dENsYXNzZXMsXG4gIGFwcGx5TGF5b3V0SXRlbUNsYXNzZXMsXG4gIGNyZWF0ZUxheW91dFJlc3VsdCxcbn07XG4iLAogICAgIi8vIHNyYy9jb3JlL3ZpZXdwb3J0L2ZlYXR1cmVzL2V2ZW50cy50c1xuXG4vKipcbiAqIEV2ZW50cyBGZWF0dXJlIC0gQ2VudHJhbGl6ZWQgZXZlbnQgc3lzdGVtIGZvciB2aWV3cG9ydFxuICogUHJvdmlkZXMgZXZlbnQgZW1pc3Npb24gYW5kIHN1YnNjcmlwdGlvbiBmb3IgaW50ZXItZmVhdHVyZSBjb21tdW5pY2F0aW9uXG4gKi9cblxuaW1wb3J0IHR5cGUgeyBWaWV3cG9ydENvbnRleHQsIFZpZXdwb3J0Q29tcG9uZW50IH0gZnJvbSBcIi4uL3R5cGVzXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgRXZlbnRzQ29uZmlnIHtcbiAgZGVidWc/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIEV2ZW50cyBmZWF0dXJlIGZvciB2aWV3cG9ydFxuICogQ2VudHJhbGl6ZXMgYWxsIGV2ZW50IGhhbmRsaW5nIGZvciB2aWV3cG9ydCBmZWF0dXJlc1xuICovXG5leHBvcnQgY29uc3Qgd2l0aEV2ZW50cyA9IChjb25maWc6IEV2ZW50c0NvbmZpZyA9IHt9KSA9PiB7XG4gIHJldHVybiA8VCBleHRlbmRzIFZpZXdwb3J0Q29udGV4dCAmIFZpZXdwb3J0Q29tcG9uZW50Pihjb21wb25lbnQ6IFQpOiBUID0+IHtcbiAgICBjb25zdCB7IGRlYnVnID0gZmFsc2UgfSA9IGNvbmZpZztcblxuICAgIC8vIEV2ZW50IGxpc3RlbmVycyBtYXBcbiAgICBjb25zdCBsaXN0ZW5lcnMgPSBuZXcgTWFwPHN0cmluZywgU2V0PEZ1bmN0aW9uPj4oKTtcblxuICAgIC8vIEVtaXQgYW4gZXZlbnRcbiAgICBjb25zdCBlbWl0ID0gKGV2ZW50OiBzdHJpbmcsIGRhdGE/OiBhbnkpID0+IHtcbiAgICAgIC8vIGlmIChkZWJ1Zykge1xuICAgICAgLy8gICBjb25zb2xlLmxvZyhgW0V2ZW50c10gRW1pdDogJHtldmVudH1gLCBkYXRhKTtcbiAgICAgIC8vIH1cblxuICAgICAgY29uc3QgZXZlbnRMaXN0ZW5lcnMgPSBsaXN0ZW5lcnMuZ2V0KGV2ZW50KTtcbiAgICAgIGlmIChldmVudExpc3RlbmVycykge1xuICAgICAgICBldmVudExpc3RlbmVycy5mb3JFYWNoKChsaXN0ZW5lcikgPT4ge1xuICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICBsaXN0ZW5lcihkYXRhKTtcbiAgICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgY29uc29sZS5lcnJvcihgW0V2ZW50c10gRXJyb3IgaW4gbGlzdGVuZXIgZm9yICR7ZXZlbnR9OmAsIGVycm9yKTtcbiAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIH07XG5cbiAgICAvLyBTdWJzY3JpYmUgdG8gYW4gZXZlbnRcbiAgICBjb25zdCBvbiA9IChldmVudDogc3RyaW5nLCBoYW5kbGVyOiBGdW5jdGlvbik6ICgoKSA9PiB2b2lkKSA9PiB7XG4gICAgICBpZiAoIWxpc3RlbmVycy5oYXMoZXZlbnQpKSB7XG4gICAgICAgIGxpc3RlbmVycy5zZXQoZXZlbnQsIG5ldyBTZXQoKSk7XG4gICAgICB9XG5cbiAgICAgIGxpc3RlbmVycy5nZXQoZXZlbnQpIS5hZGQoaGFuZGxlcik7XG5cbiAgICAgIC8vIGlmIChkZWJ1Zykge1xuICAgICAgLy8gICBjb25zb2xlLmxvZyhgW0V2ZW50c10gU3Vic2NyaWJlZCB0bzogJHtldmVudH1gKTtcbiAgICAgIC8vIH1cblxuICAgICAgLy8gUmV0dXJuIHVuc3Vic2NyaWJlIGZ1bmN0aW9uXG4gICAgICByZXR1cm4gKCkgPT4ge1xuICAgICAgICBjb25zdCBldmVudExpc3RlbmVycyA9IGxpc3RlbmVycy5nZXQoZXZlbnQpO1xuICAgICAgICBpZiAoZXZlbnRMaXN0ZW5lcnMpIHtcbiAgICAgICAgICBldmVudExpc3RlbmVycy5kZWxldGUoaGFuZGxlcik7XG4gICAgICAgICAgaWYgKGV2ZW50TGlzdGVuZXJzLnNpemUgPT09IDApIHtcbiAgICAgICAgICAgIGxpc3RlbmVycy5kZWxldGUoZXZlbnQpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfTtcbiAgICB9O1xuXG4gICAgLy8gU3Vic2NyaWJlIHRvIGFuIGV2ZW50IG9uY2VcbiAgICBjb25zdCBvbmNlID0gKGV2ZW50OiBzdHJpbmcsIGhhbmRsZXI6IEZ1bmN0aW9uKTogKCgpID0+IHZvaWQpID0+IHtcbiAgICAgIGNvbnN0IHdyYXBwZWRIYW5kbGVyID0gKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICBoYW5kbGVyKGRhdGEpO1xuICAgICAgICBvZmYoZXZlbnQsIHdyYXBwZWRIYW5kbGVyKTtcbiAgICAgIH07XG4gICAgICByZXR1cm4gb24oZXZlbnQsIHdyYXBwZWRIYW5kbGVyKTtcbiAgICB9O1xuXG4gICAgLy8gVW5zdWJzY3JpYmUgZnJvbSBhbiBldmVudFxuICAgIGNvbnN0IG9mZiA9IChldmVudDogc3RyaW5nLCBoYW5kbGVyOiBGdW5jdGlvbikgPT4ge1xuICAgICAgY29uc3QgZXZlbnRMaXN0ZW5lcnMgPSBsaXN0ZW5lcnMuZ2V0KGV2ZW50KTtcbiAgICAgIGlmIChldmVudExpc3RlbmVycykge1xuICAgICAgICBldmVudExpc3RlbmVycy5kZWxldGUoaGFuZGxlcik7XG4gICAgICAgIGlmIChldmVudExpc3RlbmVycy5zaXplID09PSAwKSB7XG4gICAgICAgICAgbGlzdGVuZXJzLmRlbGV0ZShldmVudCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9O1xuXG4gICAgLy8gQ2xlYXIgYWxsIGxpc3RlbmVycyBmb3IgYW4gZXZlbnRcbiAgICBjb25zdCBjbGVhciA9IChldmVudD86IHN0cmluZykgPT4ge1xuICAgICAgaWYgKGV2ZW50KSB7XG4gICAgICAgIGxpc3RlbmVycy5kZWxldGUoZXZlbnQpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgbGlzdGVuZXJzLmNsZWFyKCk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIEFkZCBldmVudCBtZXRob2RzIHRvIGNvbXBvbmVudFxuICAgIGNvbXBvbmVudC5lbWl0ID0gZW1pdDtcbiAgICBjb21wb25lbnQub24gPSBvbjtcbiAgICBjb21wb25lbnQub25jZSA9IG9uY2U7XG4gICAgY29tcG9uZW50Lm9mZiA9IG9mZjtcblxuICAgIC8vIEFkZCBldmVudHMgQVBJIHRvIHZpZXdwb3J0XG4gICAgKGNvbXBvbmVudC52aWV3cG9ydCBhcyBhbnkpLmV2ZW50cyA9IHtcbiAgICAgIGVtaXQsXG4gICAgICBvbixcbiAgICAgIG9uY2UsXG4gICAgICBvZmYsXG4gICAgICBjbGVhcixcbiAgICAgIGdldExpc3RlbmVyQ291bnQ6IChldmVudD86IHN0cmluZykgPT4ge1xuICAgICAgICBpZiAoZXZlbnQpIHtcbiAgICAgICAgICByZXR1cm4gbGlzdGVuZXJzLmdldChldmVudCk/LnNpemUgfHwgMDtcbiAgICAgICAgfVxuICAgICAgICBsZXQgdG90YWwgPSAwO1xuICAgICAgICBsaXN0ZW5lcnMuZm9yRWFjaCgoc2V0KSA9PiAodG90YWwgKz0gc2V0LnNpemUpKTtcbiAgICAgICAgcmV0dXJuIHRvdGFsO1xuICAgICAgfSxcbiAgICB9O1xuXG4gICAgLy8gQ2xlYW4gdXAgb24gZGVzdHJveVxuICAgIGlmIChcImRlc3Ryb3lcIiBpbiBjb21wb25lbnQgJiYgdHlwZW9mIGNvbXBvbmVudC5kZXN0cm95ID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgIGNvbnN0IG9yaWdpbmFsRGVzdHJveSA9IGNvbXBvbmVudC5kZXN0cm95O1xuICAgICAgY29tcG9uZW50LmRlc3Ryb3kgPSAoKSA9PiB7XG4gICAgICAgIGNsZWFyKCk7XG4gICAgICAgIG9yaWdpbmFsRGVzdHJveT8uKCk7XG4g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
|