babylonjs-addons 7.49.0 → 7.51.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.
Files changed (2) hide show
  1. package/babylonjs.addons.js +1711 -1711
  2. package/package.json +2 -2
@@ -7,1851 +7,1851 @@
7
7
  exports["babylonjs-addons"] = factory(require("babylonjs"));
8
8
  else
9
9
  root["ADDONS"] = factory(root["BABYLON"]);
10
- })((typeof self !== "undefined" ? self : typeof global !== "undefined" ? global : this), (__WEBPACK_EXTERNAL_MODULE_babylonjs_Meshes_mesh__) => {
10
+ })((typeof self !== "undefined" ? self : typeof global !== "undefined" ? global : this), (__WEBPACK_EXTERNAL_MODULE_babylonjs_Maths_math__) => {
11
11
  return /******/ (() => { // webpackBootstrap
12
12
  /******/ "use strict";
13
13
  /******/ var __webpack_modules__ = ({
14
14
 
15
- /***/ "../../../dev/addons/src/htmlMesh/fitStrategy.ts":
16
- /*!*******************************************************!*\
17
- !*** ../../../dev/addons/src/htmlMesh/fitStrategy.ts ***!
18
- \*******************************************************/
19
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
15
+ /***/ "../../../../node_modules/tslib/tslib.es6.mjs":
16
+ /*!****************************************************!*\
17
+ !*** ../../../../node_modules/tslib/tslib.es6.mjs ***!
18
+ \****************************************************/
19
+ /***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => {
20
20
 
21
21
  __webpack_require__.r(__webpack_exports__);
22
22
  /* harmony export */ __webpack_require__.d(__webpack_exports__, {
23
- /* harmony export */ FitStrategy: () => (/* binding */ FitStrategy)
23
+ /* harmony export */ __addDisposableResource: () => (/* binding */ __addDisposableResource),
24
+ /* harmony export */ __assign: () => (/* binding */ __assign),
25
+ /* harmony export */ __asyncDelegator: () => (/* binding */ __asyncDelegator),
26
+ /* harmony export */ __asyncGenerator: () => (/* binding */ __asyncGenerator),
27
+ /* harmony export */ __asyncValues: () => (/* binding */ __asyncValues),
28
+ /* harmony export */ __await: () => (/* binding */ __await),
29
+ /* harmony export */ __awaiter: () => (/* binding */ __awaiter),
30
+ /* harmony export */ __classPrivateFieldGet: () => (/* binding */ __classPrivateFieldGet),
31
+ /* harmony export */ __classPrivateFieldIn: () => (/* binding */ __classPrivateFieldIn),
32
+ /* harmony export */ __classPrivateFieldSet: () => (/* binding */ __classPrivateFieldSet),
33
+ /* harmony export */ __createBinding: () => (/* binding */ __createBinding),
34
+ /* harmony export */ __decorate: () => (/* binding */ __decorate),
35
+ /* harmony export */ __disposeResources: () => (/* binding */ __disposeResources),
36
+ /* harmony export */ __esDecorate: () => (/* binding */ __esDecorate),
37
+ /* harmony export */ __exportStar: () => (/* binding */ __exportStar),
38
+ /* harmony export */ __extends: () => (/* binding */ __extends),
39
+ /* harmony export */ __generator: () => (/* binding */ __generator),
40
+ /* harmony export */ __importDefault: () => (/* binding */ __importDefault),
41
+ /* harmony export */ __importStar: () => (/* binding */ __importStar),
42
+ /* harmony export */ __makeTemplateObject: () => (/* binding */ __makeTemplateObject),
43
+ /* harmony export */ __metadata: () => (/* binding */ __metadata),
44
+ /* harmony export */ __param: () => (/* binding */ __param),
45
+ /* harmony export */ __propKey: () => (/* binding */ __propKey),
46
+ /* harmony export */ __read: () => (/* binding */ __read),
47
+ /* harmony export */ __rest: () => (/* binding */ __rest),
48
+ /* harmony export */ __rewriteRelativeImportExtension: () => (/* binding */ __rewriteRelativeImportExtension),
49
+ /* harmony export */ __runInitializers: () => (/* binding */ __runInitializers),
50
+ /* harmony export */ __setFunctionName: () => (/* binding */ __setFunctionName),
51
+ /* harmony export */ __spread: () => (/* binding */ __spread),
52
+ /* harmony export */ __spreadArray: () => (/* binding */ __spreadArray),
53
+ /* harmony export */ __spreadArrays: () => (/* binding */ __spreadArrays),
54
+ /* harmony export */ __values: () => (/* binding */ __values),
55
+ /* harmony export */ "default": () => (__WEBPACK_DEFAULT_EXPORT__)
24
56
  /* harmony export */ });
25
- var FitStrategyContain = {
26
- wrapElement: function (element) {
27
- var sizingElement = document.createElement("div");
28
- sizingElement.style.display = "flex";
29
- sizingElement.style.justifyContent = "center";
30
- sizingElement.style.alignItems = "center";
31
- var scalingElement = document.createElement("div");
32
- scalingElement.style.visibility = "hidden";
33
- scalingElement.appendChild(element);
34
- sizingElement.appendChild(scalingElement);
35
- return sizingElement;
36
- },
37
- updateSize: function (sizingElement, width, height) {
38
- var scalingElement = sizingElement.firstElementChild;
39
- sizingElement.style.width = "".concat(width, "px");
40
- sizingElement.style.height = "".concat(height, "px");
41
- var _a = [scalingElement.offsetWidth, scalingElement.offsetHeight], childWidth = _a[0], childHeight = _a[1];
42
- var scale = Math.min(width / childWidth, height / childHeight);
43
- scalingElement.style.transform = "scale(".concat(scale, ")");
44
- scalingElement.style.visibility = "visible";
45
- },
46
- };
47
- var FitStrategyCover = {
48
- wrapElement: function (element) {
49
- var sizingElement = document.createElement("div");
50
- sizingElement.style.display = "flex";
51
- sizingElement.style.justifyContent = "center";
52
- sizingElement.style.alignItems = "center";
53
- sizingElement.style.overflow = "hidden";
54
- var scalingElement = document.createElement("div");
55
- scalingElement.style.visibility = "hidden";
56
- scalingElement.appendChild(element);
57
- sizingElement.appendChild(scalingElement);
58
- return sizingElement;
59
- },
60
- updateSize: function (sizingElement, width, height) {
61
- var scalingElement = sizingElement.firstElementChild;
62
- sizingElement.style.width = "".concat(width, "px");
63
- sizingElement.style.height = "".concat(height, "px");
64
- var _a = [scalingElement.offsetWidth, scalingElement.offsetHeight], childWidth = _a[0], childHeight = _a[1];
65
- var scale = Math.max(width / childWidth, height / childHeight);
66
- scalingElement.style.transform = "scale(".concat(scale, ")");
67
- scalingElement.style.visibility = "visible";
68
- },
69
- };
70
- var FitStrategyStretch = {
71
- wrapElement: function (element) {
72
- var sizingElement = document.createElement("div");
73
- sizingElement.style.display = "flex";
74
- sizingElement.style.justifyContent = "center";
75
- sizingElement.style.alignItems = "center";
76
- var scalingElement = document.createElement("div");
77
- scalingElement.style.visibility = "hidden";
78
- scalingElement.appendChild(element);
79
- sizingElement.appendChild(scalingElement);
80
- return sizingElement;
81
- },
82
- updateSize: function (sizingElement, width, height) {
83
- var scalingElement = sizingElement.firstElementChild;
84
- sizingElement.style.width = "".concat(width, "px");
85
- sizingElement.style.height = "".concat(height, "px");
86
- var _a = [scalingElement.offsetWidth, scalingElement.offsetHeight], childWidth = _a[0], childHeight = _a[1];
87
- scalingElement.style.transform = "scale(".concat(width / childWidth, ", ").concat(height / childHeight, ")");
88
- scalingElement.style.visibility = "visible";
89
- },
90
- };
91
- var FitStrategyNone = {
92
- wrapElement: function (element) {
93
- return element;
94
- },
95
- updateSize: function (sizingElement, width, height) {
96
- if (sizingElement) {
97
- sizingElement.style.width = "".concat(width, "px");
98
- sizingElement.style.height = "".concat(height, "px");
99
- }
100
- },
101
- };
102
- var FitStrategy = {
103
- CONTAIN: FitStrategyContain,
104
- COVER: FitStrategyCover,
105
- STRETCH: FitStrategyStretch,
106
- NONE: FitStrategyNone,
57
+ /******************************************************************************
58
+ Copyright (c) Microsoft Corporation.
59
+
60
+ Permission to use, copy, modify, and/or distribute this software for any
61
+ purpose with or without fee is hereby granted.
62
+
63
+ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
64
+ REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
65
+ AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
66
+ INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
67
+ LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
68
+ OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
69
+ PERFORMANCE OF THIS SOFTWARE.
70
+ ***************************************************************************** */
71
+ /* global Reflect, Promise, SuppressedError, Symbol, Iterator */
72
+
73
+ var extendStatics = function(d, b) {
74
+ extendStatics = Object.setPrototypeOf ||
75
+ ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
76
+ function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };
77
+ return extendStatics(d, b);
107
78
  };
108
79
 
80
+ function __extends(d, b) {
81
+ if (typeof b !== "function" && b !== null)
82
+ throw new TypeError("Class extends value " + String(b) + " is not a constructor or null");
83
+ extendStatics(d, b);
84
+ function __() { this.constructor = d; }
85
+ d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
86
+ }
109
87
 
110
- /***/ }),
88
+ var __assign = function() {
89
+ __assign = Object.assign || function __assign(t) {
90
+ for (var s, i = 1, n = arguments.length; i < n; i++) {
91
+ s = arguments[i];
92
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];
93
+ }
94
+ return t;
95
+ }
96
+ return __assign.apply(this, arguments);
97
+ }
111
98
 
112
- /***/ "../../../dev/addons/src/htmlMesh/htmlMesh.ts":
113
- /*!****************************************************!*\
114
- !*** ../../../dev/addons/src/htmlMesh/htmlMesh.ts ***!
115
- \****************************************************/
116
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
99
+ function __rest(s, e) {
100
+ var t = {};
101
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
102
+ t[p] = s[p];
103
+ if (s != null && typeof Object.getOwnPropertySymbols === "function")
104
+ for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
105
+ if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
106
+ t[p[i]] = s[p[i]];
107
+ }
108
+ return t;
109
+ }
117
110
 
118
- __webpack_require__.r(__webpack_exports__);
119
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
120
- /* harmony export */ HtmlMesh: () => (/* binding */ HtmlMesh)
121
- /* harmony export */ });
122
- /* harmony import */ var tslib__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! tslib */ "../../../../node_modules/tslib/tslib.es6.mjs");
123
- /* harmony import */ var babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! babylonjs/Misc/logger */ "babylonjs/Meshes/mesh");
124
- /* harmony import */ var babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__);
125
- /* harmony import */ var _pointerEventsCaptureBehavior__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./pointerEventsCaptureBehavior */ "../../../dev/addons/src/htmlMesh/pointerEventsCaptureBehavior.ts");
126
- /* harmony import */ var _fitStrategy__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./fitStrategy */ "../../../dev/addons/src/htmlMesh/fitStrategy.ts");
111
+ function __decorate(decorators, target, key, desc) {
112
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
113
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
114
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
115
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
116
+ }
127
117
 
118
+ function __param(paramIndex, decorator) {
119
+ return function (target, key) { decorator(target, key, paramIndex); }
120
+ }
128
121
 
122
+ function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {
123
+ function accept(f) { if (f !== void 0 && typeof f !== "function") throw new TypeError("Function expected"); return f; }
124
+ var kind = contextIn.kind, key = kind === "getter" ? "get" : kind === "setter" ? "set" : "value";
125
+ var target = !descriptorIn && ctor ? contextIn["static"] ? ctor : ctor.prototype : null;
126
+ var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});
127
+ var _, done = false;
128
+ for (var i = decorators.length - 1; i >= 0; i--) {
129
+ var context = {};
130
+ for (var p in contextIn) context[p] = p === "access" ? {} : contextIn[p];
131
+ for (var p in contextIn.access) context.access[p] = contextIn.access[p];
132
+ context.addInitializer = function (f) { if (done) throw new TypeError("Cannot add initializers after decoration has completed"); extraInitializers.push(accept(f || null)); };
133
+ var result = (0, decorators[i])(kind === "accessor" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);
134
+ if (kind === "accessor") {
135
+ if (result === void 0) continue;
136
+ if (result === null || typeof result !== "object") throw new TypeError("Object expected");
137
+ if (_ = accept(result.get)) descriptor.get = _;
138
+ if (_ = accept(result.set)) descriptor.set = _;
139
+ if (_ = accept(result.init)) initializers.unshift(_);
140
+ }
141
+ else if (_ = accept(result)) {
142
+ if (kind === "field") initializers.unshift(_);
143
+ else descriptor[key] = _;
144
+ }
145
+ }
146
+ if (target) Object.defineProperty(target, contextIn.name, descriptor);
147
+ done = true;
148
+ };
129
149
 
150
+ function __runInitializers(thisArg, initializers, value) {
151
+ var useValue = arguments.length > 2;
152
+ for (var i = 0; i < initializers.length; i++) {
153
+ value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);
154
+ }
155
+ return useValue ? value : void 0;
156
+ };
130
157
 
158
+ function __propKey(x) {
159
+ return typeof x === "symbol" ? x : "".concat(x);
160
+ };
131
161
 
162
+ function __setFunctionName(f, name, prefix) {
163
+ if (typeof name === "symbol") name = name.description ? "[".concat(name.description, "]") : "";
164
+ return Object.defineProperty(f, "name", { configurable: true, value: prefix ? "".concat(prefix, " ", name) : name });
165
+ };
132
166
 
167
+ function __metadata(metadataKey, metadataValue) {
168
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(metadataKey, metadataValue);
169
+ }
133
170
 
171
+ function __awaiter(thisArg, _arguments, P, generator) {
172
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
173
+ return new (P || (P = Promise))(function (resolve, reject) {
174
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
175
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
176
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
177
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
178
+ });
179
+ }
134
180
 
135
- /**
136
- * This class represents HTML content that we want to render as though it is part of the scene. The HTML content is actually
137
- * rendered below the canvas, but a depth mask is created by this class that writes to the depth buffer but does not
138
- * write to the color buffer, effectively punching a hole in the canvas. CSS transforms are used to scale, translate, and rotate
139
- * the HTML content so that it matches the camera and mesh orientation. The class supports interactions in editable and non-editable mode.
140
- * In non-editable mode (the default), events are passed to the HTML content when the pointer is over the mask (and not occluded by other meshes
141
- * in the scene).
142
- * #HVHYJC#5
143
- * #B17TC7#112
144
- */
145
- var HtmlMesh = /** @class */ (function (_super) {
146
- (0,tslib__WEBPACK_IMPORTED_MODULE_3__.__extends)(HtmlMesh, _super);
147
- /**
148
- * Contruct an instance of HtmlMesh
149
- * @param scene
150
- * @param id The id of the mesh. Will be used as the id of the HTML element as well.
151
- * @param options object with optional parameters
152
- */
153
- function HtmlMesh(scene, id, _a) {
154
- var _b = _a === void 0 ? {} : _a, _c = _b.captureOnPointerEnter, captureOnPointerEnter = _c === void 0 ? true : _c, _d = _b.isCanvasOverlay, isCanvasOverlay = _d === void 0 ? false : _d, _e = _b.fitStrategy, fitStrategy = _e === void 0 ? _fitStrategy__WEBPACK_IMPORTED_MODULE_2__.FitStrategy.NONE : _e;
155
- var _this = _super.call(this, id, scene) || this;
156
- // Override the super class's _isEnabled property so we can control when the mesh
157
- // is enabled. I.e., we don't want to render the mesh until there is content to show.
158
- _this._enabled = false;
159
- // The mesh is ready when content has been set and the content size has been set
160
- // The former is done by the user, the latter is done by the renderer.
161
- _this._ready = false;
162
- /**
163
- * @internal
164
- */
165
- _this._isCanvasOverlay = false;
166
- _this._requiresUpdate = true;
167
- _this._inverseScaleMatrix = null;
168
- _this._captureOnPointerEnter = true;
169
- _this._pointerEventCaptureBehavior = null;
170
- _this._sourceWidth = null;
171
- _this._sourceHeight = null;
172
- _this._fitStrategy = _fitStrategy__WEBPACK_IMPORTED_MODULE_2__.FitStrategy.NONE;
173
- // Requires a browser to work. Bail if we aren't running in a browser
174
- if (typeof document === "undefined") {
175
- babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.Logger.Warn("Creating an instance of an HtmlMesh with id ".concat(id, " outside of a browser. The mesh will not be visible."));
176
- return _this;
177
- }
178
- _this._fitStrategy = fitStrategy;
179
- _this._isCanvasOverlay = isCanvasOverlay;
180
- _this._createMask();
181
- _this._element = _this._createElement();
182
- // Set enabled by default, so this will show as soon as it's ready
183
- _this.setEnabled(true);
184
- _this._captureOnPointerEnter = captureOnPointerEnter;
185
- // Create a behavior to capture pointer events
186
- _this._pointerEventCaptureBehavior = new _pointerEventsCaptureBehavior__WEBPACK_IMPORTED_MODULE_1__.PointerEventsCaptureBehavior(_this.capturePointerEvents.bind(_this), _this.releasePointerEvents.bind(_this), {
187
- captureOnPointerEnter: _this._captureOnPointerEnter,
188
- });
189
- _this.addBehavior(_this._pointerEventCaptureBehavior);
190
- return _this;
191
- }
192
- Object.defineProperty(HtmlMesh.prototype, "isHtmlMesh", {
193
- /**
194
- * Helps identifying a html mesh from a regular mesh
195
- */
196
- get: function () {
197
- return true;
198
- },
199
- enumerable: false,
200
- configurable: true
201
- });
202
- Object.defineProperty(HtmlMesh.prototype, "sourceWidth", {
203
- /**
204
- * Return the source width of the content in pixels
205
- */
206
- get: function () {
207
- return this._sourceWidth;
208
- },
209
- enumerable: false,
210
- configurable: true
211
- });
212
- Object.defineProperty(HtmlMesh.prototype, "sourceHeight", {
213
- /**
214
- * Return the source height of the content in pixels
215
- */
216
- get: function () {
217
- return this._sourceHeight;
218
- },
219
- enumerable: false,
220
- configurable: true
221
- });
222
- Object.defineProperty(HtmlMesh.prototype, "width", {
223
- /**
224
- * The width of the content in pixels
225
- */
226
- get: function () {
227
- return this._width;
228
- },
229
- enumerable: false,
230
- configurable: true
231
- });
232
- Object.defineProperty(HtmlMesh.prototype, "height", {
233
- /**
234
- * The height of the content in pixels
235
- */
236
- get: function () {
237
- return this._height;
238
- },
239
- enumerable: false,
240
- configurable: true
241
- });
242
- Object.defineProperty(HtmlMesh.prototype, "element", {
243
- /**
244
- * The HTML element that is being rendered as a mesh
245
- */
246
- get: function () {
247
- return this._element;
248
- },
249
- enumerable: false,
250
- configurable: true
251
- });
252
- Object.defineProperty(HtmlMesh.prototype, "requiresUpdate", {
253
- /**
254
- * True if the mesh has been moved, rotated, or scaled since the last time this
255
- * property was read. This property is reset to false after reading.
256
- */
257
- get: function () {
258
- return this._requiresUpdate;
259
- },
260
- enumerable: false,
261
- configurable: true
262
- });
263
- Object.defineProperty(HtmlMesh.prototype, "captureOnPointerEnter", {
264
- /**
265
- * Enable capture for the pointer when entering the mesh area
266
- */
267
- set: function (captureOnPointerEnter) {
268
- this._captureOnPointerEnter = captureOnPointerEnter;
269
- if (this._pointerEventCaptureBehavior) {
270
- this._pointerEventCaptureBehavior.captureOnPointerEnter = captureOnPointerEnter;
271
- }
272
- },
273
- enumerable: false,
274
- configurable: true
275
- });
276
- /**
277
- * Disposes of the mesh and the HTML element
278
- */
279
- HtmlMesh.prototype.dispose = function () {
280
- var _a;
281
- _super.prototype.dispose.call(this);
282
- (_a = this._element) === null || _a === void 0 ? void 0 : _a.remove();
283
- this._element = undefined;
284
- if (this._pointerEventCaptureBehavior) {
285
- this._pointerEventCaptureBehavior.dispose();
286
- this._pointerEventCaptureBehavior = null;
287
- }
288
- };
289
- /**
290
- * @internal
291
- */
292
- HtmlMesh.prototype._markAsUpdated = function () {
293
- this._requiresUpdate = false;
294
- };
295
- /**
296
- * Sets the content of the element to the specified content adjusting the mesh scale to match and making it visible.
297
- * If the the specified content is undefined, then it will make the mesh invisible. In either case it will clear the
298
- * element content first.
299
- * @param element The element to render as a mesh
300
- * @param width The width of the mesh in Babylon units
301
- * @param height The height of the mesh in Babylon units
302
- */
303
- HtmlMesh.prototype.setContent = function (element, width, height) {
304
- // If content is changed, we are no longer ready
305
- this._setAsReady(false);
306
- // Also invalidate the source width and height
307
- this._sourceWidth = null;
308
- this._sourceHeight = null;
309
- if (!this._element) {
310
- return;
311
- }
312
- this._width = width;
313
- this._height = height;
314
- this._requiresUpdate = true;
315
- this.scaling.setAll(1);
316
- if (element) {
317
- this._element.appendChild(this._fitStrategy.wrapElement(element));
318
- this._updateScaleIfNecessary();
319
- }
320
- if (this.sourceWidth && this.sourceHeight) {
321
- this._setAsReady(true);
322
- }
323
- };
324
- // Overides BABYLON.Mesh.setEnabled
325
- HtmlMesh.prototype.setEnabled = function (enabled) {
326
- // Capture requested enabled state
327
- this._enabled = enabled;
328
- // If disabling or enabling and we are ready
329
- if (!enabled || this._ready) {
330
- this._doSetEnabled(enabled);
331
- }
332
- };
333
- /**
334
- * Sets the content size in pixels
335
- * @param width width of the source
336
- * @param height height of the source
337
- */
338
- HtmlMesh.prototype.setContentSizePx = function (width, height) {
339
- this._sourceWidth = width;
340
- this._sourceHeight = height;
341
- if (!this._element || !this._element.firstElementChild) {
342
- return;
343
- }
344
- this._fitStrategy.updateSize(this._element.firstElementChild, width, height);
345
- this._updateScaleIfNecessary();
346
- if (this.width && this.height) {
347
- this._setAsReady(true);
348
- }
349
- };
350
- HtmlMesh.prototype._setAsReady = function (ready) {
351
- this._ready = ready;
352
- if (ready) {
353
- this._doSetEnabled(this._enabled);
354
- }
355
- else {
356
- this._doSetEnabled(false);
357
- }
358
- };
359
- HtmlMesh.prototype._doSetEnabled = function (enabled) {
360
- var _this = this;
361
- var _a;
362
- if (!this._element) {
363
- return;
364
- }
365
- //if enabled, then start listening for changes to the
366
- // scaling, rotation, and position. otherwise stop listening
367
- if (enabled && !this._worldMatrixUpdateObserver) {
368
- this._worldMatrixUpdateObserver = this.onAfterWorldMatrixUpdateObservable.add(function () {
369
- _this._requiresUpdate = true;
370
- });
371
- }
372
- else if (!enabled) {
373
- (_a = this._worldMatrixUpdateObserver) === null || _a === void 0 ? void 0 : _a.remove();
374
- this._worldMatrixUpdateObserver = null;
375
- }
376
- // If enabled, then revert the content element display
377
- // otherwise hide it
378
- this._element.style.display = enabled ? "" : "none";
379
- // Capture the content z index
380
- this._setElementZIndex(this.position.z * -10000);
381
- _super.prototype.setEnabled.call(this, enabled);
382
- };
383
- HtmlMesh.prototype._updateScaleIfNecessary = function () {
384
- // If we have setContent before, the content scale is baked into the mesh. If we don't reset the vertices to
385
- // the original size, then we will multiply the scale when we bake the scale below. By applying the inverse, we back out
386
- // the scaling that has been done so we are starting from the same point.
387
- // First reset the scale to 1
388
- this.scaling.setAll(1);
389
- // Then back out the original vertices changes to match the content scale
390
- if (this._inverseScaleMatrix) {
391
- this.bakeTransformIntoVertices(this._inverseScaleMatrix);
392
- // Clear out the matrix so it doesn't get applied again unless we scale
393
- this._inverseScaleMatrix = null;
394
- }
395
- // Set scale to match content. Note we can't just scale the mesh, because that will scale the content as well
396
- // What we need to do is compute a scale matrix and then bake that into the mesh vertices. This will leave the
397
- // mesh scale at 1, so our content will stay it's original width and height until we scale the mesh.
398
- var scaleX = this._width || 1;
399
- var scaleY = this._height || 1;
400
- var scaleMatrix = babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.Matrix.Scaling(scaleX, scaleY, 1);
401
- this.bakeTransformIntoVertices(scaleMatrix);
402
- // Get an inverse of the scale matrix that we can use to back out the scale changes we have made so
403
- // we don't multiply the scale.
404
- this._inverseScaleMatrix = new babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.Matrix();
405
- scaleMatrix.invertToRef(this._inverseScaleMatrix);
406
- };
407
- HtmlMesh.prototype._createMask = function () {
408
- var vertexData = (0,babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.CreatePlaneVertexData)({ width: 1, height: 1 });
409
- vertexData.applyToMesh(this);
410
- var scene = this.getScene();
411
- this.checkCollisions = true;
412
- var depthMask = new babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.StandardMaterial("".concat(this.id, "-mat"), scene);
413
- if (!this._isCanvasOverlay) {
414
- depthMask.backFaceCulling = false;
415
- depthMask.disableColorWrite = true;
416
- depthMask.disableLighting = true;
417
- }
418
- this.material = depthMask;
419
- // Optimization - Freeze material since it never needs to change
420
- this.material.freeze();
421
- };
422
- HtmlMesh.prototype._setElementZIndex = function (zIndex) {
423
- if (this._element) {
424
- this._element.style.zIndex = "".concat(zIndex);
425
- }
426
- };
427
- /**
428
- * Callback used by the PointerEventsCaptureBehavior to capture pointer events
429
- */
430
- HtmlMesh.prototype.capturePointerEvents = function () {
431
- if (!this._element) {
432
- return;
433
- }
434
- // Enable dom content to capture pointer events
435
- this._element.style.pointerEvents = "auto";
436
- // Supress events outside of the dom content
437
- document.getElementsByTagName("body")[0].style.pointerEvents = "none";
438
- };
439
- /**
440
- * Callback used by the PointerEventsCaptureBehavior to release pointer events
441
- */
442
- HtmlMesh.prototype.releasePointerEvents = function () {
443
- if (!this._element) {
444
- return;
445
- }
446
- // Enable pointer events on canvas
447
- document.getElementsByTagName("body")[0].style.pointerEvents = "auto";
448
- // Disable pointer events on dom content
449
- this._element.style.pointerEvents = "none";
450
- };
451
- HtmlMesh.prototype._createElement = function () {
452
- // Requires a browser to work. Bail if we aren't running in a browser
453
- if (typeof document === "undefined") {
454
- return;
181
+ function __generator(thisArg, body) {
182
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
183
+ return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
184
+ function verb(n) { return function (v) { return step([n, v]); }; }
185
+ function step(op) {
186
+ if (f) throw new TypeError("Generator is already executing.");
187
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
188
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
189
+ if (y = 0, t) op = [op[0] & 2, t.value];
190
+ switch (op[0]) {
191
+ case 0: case 1: t = op; break;
192
+ case 4: _.label++; return { value: op[1], done: false };
193
+ case 5: _.label++; y = op[1]; op = [0]; continue;
194
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
195
+ default:
196
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
197
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
198
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
199
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
200
+ if (t[2]) _.ops.pop();
201
+ _.trys.pop(); continue;
202
+ }
203
+ op = body.call(thisArg, _);
204
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
205
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
206
+ }
207
+ }
208
+
209
+ var __createBinding = Object.create ? (function(o, m, k, k2) {
210
+ if (k2 === undefined) k2 = k;
211
+ var desc = Object.getOwnPropertyDescriptor(m, k);
212
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
213
+ desc = { enumerable: true, get: function() { return m[k]; } };
214
+ }
215
+ Object.defineProperty(o, k2, desc);
216
+ }) : (function(o, m, k, k2) {
217
+ if (k2 === undefined) k2 = k;
218
+ o[k2] = m[k];
219
+ });
220
+
221
+ function __exportStar(m, o) {
222
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);
223
+ }
224
+
225
+ function __values(o) {
226
+ var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0;
227
+ if (m) return m.call(o);
228
+ if (o && typeof o.length === "number") return {
229
+ next: function () {
230
+ if (o && i >= o.length) o = void 0;
231
+ return { value: o && o[i++], done: !o };
232
+ }
233
+ };
234
+ throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined.");
235
+ }
236
+
237
+ function __read(o, n) {
238
+ var m = typeof Symbol === "function" && o[Symbol.iterator];
239
+ if (!m) return o;
240
+ var i = m.call(o), r, ar = [], e;
241
+ try {
242
+ while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);
243
+ }
244
+ catch (error) { e = { error: error }; }
245
+ finally {
246
+ try {
247
+ if (r && !r.done && (m = i["return"])) m.call(i);
248
+ }
249
+ finally { if (e) throw e.error; }
250
+ }
251
+ return ar;
252
+ }
253
+
254
+ /** @deprecated */
255
+ function __spread() {
256
+ for (var ar = [], i = 0; i < arguments.length; i++)
257
+ ar = ar.concat(__read(arguments[i]));
258
+ return ar;
259
+ }
260
+
261
+ /** @deprecated */
262
+ function __spreadArrays() {
263
+ for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;
264
+ for (var r = Array(s), k = 0, i = 0; i < il; i++)
265
+ for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)
266
+ r[k] = a[j];
267
+ return r;
268
+ }
269
+
270
+ function __spreadArray(to, from, pack) {
271
+ if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
272
+ if (ar || !(i in from)) {
273
+ if (!ar) ar = Array.prototype.slice.call(from, 0, i);
274
+ ar[i] = from[i];
275
+ }
276
+ }
277
+ return to.concat(ar || Array.prototype.slice.call(from));
278
+ }
279
+
280
+ function __await(v) {
281
+ return this instanceof __await ? (this.v = v, this) : new __await(v);
282
+ }
283
+
284
+ function __asyncGenerator(thisArg, _arguments, generator) {
285
+ if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
286
+ var g = generator.apply(thisArg, _arguments || []), i, q = [];
287
+ return i = Object.create((typeof AsyncIterator === "function" ? AsyncIterator : Object).prototype), verb("next"), verb("throw"), verb("return", awaitReturn), i[Symbol.asyncIterator] = function () { return this; }, i;
288
+ function awaitReturn(f) { return function (v) { return Promise.resolve(v).then(f, reject); }; }
289
+ function verb(n, f) { if (g[n]) { i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; if (f) i[n] = f(i[n]); } }
290
+ function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }
291
+ function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }
292
+ function fulfill(value) { resume("next", value); }
293
+ function reject(value) { resume("throw", value); }
294
+ function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }
295
+ }
296
+
297
+ function __asyncDelegator(o) {
298
+ var i, p;
299
+ return i = {}, verb("next"), verb("throw", function (e) { throw e; }), verb("return"), i[Symbol.iterator] = function () { return this; }, i;
300
+ function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }
301
+ }
302
+
303
+ function __asyncValues(o) {
304
+ if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
305
+ var m = o[Symbol.asyncIterator], i;
306
+ return m ? m.call(o) : (o = typeof __values === "function" ? __values(o) : o[Symbol.iterator](), i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i);
307
+ function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }
308
+ function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }
309
+ }
310
+
311
+ function __makeTemplateObject(cooked, raw) {
312
+ if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; }
313
+ return cooked;
314
+ };
315
+
316
+ var __setModuleDefault = Object.create ? (function(o, v) {
317
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
318
+ }) : function(o, v) {
319
+ o["default"] = v;
320
+ };
321
+
322
+ var ownKeys = function(o) {
323
+ ownKeys = Object.getOwnPropertyNames || function (o) {
324
+ var ar = [];
325
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
326
+ return ar;
327
+ };
328
+ return ownKeys(o);
329
+ };
330
+
331
+ function __importStar(mod) {
332
+ if (mod && mod.__esModule) return mod;
333
+ var result = {};
334
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
335
+ __setModuleDefault(result, mod);
336
+ return result;
337
+ }
338
+
339
+ function __importDefault(mod) {
340
+ return (mod && mod.__esModule) ? mod : { default: mod };
341
+ }
342
+
343
+ function __classPrivateFieldGet(receiver, state, kind, f) {
344
+ if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
345
+ if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
346
+ return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
347
+ }
348
+
349
+ function __classPrivateFieldSet(receiver, state, value, kind, f) {
350
+ if (kind === "m") throw new TypeError("Private method is not writable");
351
+ if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
352
+ if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
353
+ return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
354
+ }
355
+
356
+ function __classPrivateFieldIn(state, receiver) {
357
+ if (receiver === null || (typeof receiver !== "object" && typeof receiver !== "function")) throw new TypeError("Cannot use 'in' operator on non-object");
358
+ return typeof state === "function" ? receiver === state : state.has(receiver);
359
+ }
360
+
361
+ function __addDisposableResource(env, value, async) {
362
+ if (value !== null && value !== void 0) {
363
+ if (typeof value !== "object" && typeof value !== "function") throw new TypeError("Object expected.");
364
+ var dispose, inner;
365
+ if (async) {
366
+ if (!Symbol.asyncDispose) throw new TypeError("Symbol.asyncDispose is not defined.");
367
+ dispose = value[Symbol.asyncDispose];
368
+ }
369
+ if (dispose === void 0) {
370
+ if (!Symbol.dispose) throw new TypeError("Symbol.dispose is not defined.");
371
+ dispose = value[Symbol.dispose];
372
+ if (async) inner = dispose;
373
+ }
374
+ if (typeof dispose !== "function") throw new TypeError("Object not disposable.");
375
+ if (inner) dispose = function() { try { inner.call(this); } catch (e) { return Promise.reject(e); } };
376
+ env.stack.push({ value: value, dispose: dispose, async: async });
377
+ }
378
+ else if (async) {
379
+ env.stack.push({ async: true });
380
+ }
381
+ return value;
382
+ }
383
+
384
+ var _SuppressedError = typeof SuppressedError === "function" ? SuppressedError : function (error, suppressed, message) {
385
+ var e = new Error(message);
386
+ return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e;
387
+ };
388
+
389
+ function __disposeResources(env) {
390
+ function fail(e) {
391
+ env.error = env.hasError ? new _SuppressedError(e, env.error, "An error was suppressed during disposal.") : e;
392
+ env.hasError = true;
393
+ }
394
+ var r, s = 0;
395
+ function next() {
396
+ while (r = env.stack.pop()) {
397
+ try {
398
+ if (!r.async && s === 1) return s = 0, env.stack.push(r), Promise.resolve().then(next);
399
+ if (r.dispose) {
400
+ var result = r.dispose.call(r.value);
401
+ if (r.async) return s |= 2, Promise.resolve(result).then(next, function(e) { fail(e); return next(); });
455
402
  }
456
- var div = document.createElement("div");
457
- div.id = this.id;
458
- div.style.backgroundColor = this._isCanvasOverlay ? "transparent" : "#000";
459
- div.style.zIndex = "1";
460
- div.style.position = "absolute";
461
- div.style.pointerEvents = "none";
462
- div.style.backfaceVisibility = "hidden";
463
- return div;
464
- };
465
- return HtmlMesh;
466
- }(babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.Mesh));
403
+ else s |= 1;
404
+ }
405
+ catch (e) {
406
+ fail(e);
407
+ }
408
+ }
409
+ if (s === 1) return env.hasError ? Promise.reject(env.error) : Promise.resolve();
410
+ if (env.hasError) throw env.error;
411
+ }
412
+ return next();
413
+ }
467
414
 
415
+ function __rewriteRelativeImportExtension(path, preserveJsx) {
416
+ if (typeof path === "string" && /^\.\.?\//.test(path)) {
417
+ return path.replace(/\.(tsx)$|((?:\.d)?)((?:\.[^./]+?)?)\.([cm]?)ts$/i, function (m, tsx, d, ext, cm) {
418
+ return tsx ? preserveJsx ? ".jsx" : ".js" : d && (!ext || !cm) ? m : (d + ext + "." + cm.toLowerCase() + "js");
419
+ });
420
+ }
421
+ return path;
422
+ }
423
+
424
+ /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = ({
425
+ __extends,
426
+ __assign,
427
+ __rest,
428
+ __decorate,
429
+ __param,
430
+ __esDecorate,
431
+ __runInitializers,
432
+ __propKey,
433
+ __setFunctionName,
434
+ __metadata,
435
+ __awaiter,
436
+ __generator,
437
+ __createBinding,
438
+ __exportStar,
439
+ __values,
440
+ __read,
441
+ __spread,
442
+ __spreadArrays,
443
+ __spreadArray,
444
+ __await,
445
+ __asyncGenerator,
446
+ __asyncDelegator,
447
+ __asyncValues,
448
+ __makeTemplateObject,
449
+ __importStar,
450
+ __importDefault,
451
+ __classPrivateFieldGet,
452
+ __classPrivateFieldSet,
453
+ __classPrivateFieldIn,
454
+ __addDisposableResource,
455
+ __disposeResources,
456
+ __rewriteRelativeImportExtension,
457
+ });
468
458
 
469
459
 
470
460
  /***/ }),
471
461
 
472
- /***/ "../../../dev/addons/src/htmlMesh/htmlMeshRenderer.ts":
473
- /*!************************************************************!*\
474
- !*** ../../../dev/addons/src/htmlMesh/htmlMeshRenderer.ts ***!
475
- \************************************************************/
462
+ /***/ "../../../dev/addons/src/htmlMesh/fitStrategy.ts":
463
+ /*!*******************************************************!*\
464
+ !*** ../../../dev/addons/src/htmlMesh/fitStrategy.ts ***!
465
+ \*******************************************************/
476
466
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
477
467
 
478
468
  __webpack_require__.r(__webpack_exports__);
479
469
  /* harmony export */ __webpack_require__.d(__webpack_exports__, {
480
- /* harmony export */ HtmlMeshRenderer: () => (/* binding */ HtmlMeshRenderer)
470
+ /* harmony export */ FitStrategy: () => (/* binding */ FitStrategy)
481
471
  /* harmony export */ });
482
- /* harmony import */ var babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! babylonjs/Misc/logger */ "babylonjs/Meshes/mesh");
483
- /* harmony import */ var babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__);
484
-
485
-
486
-
487
-
488
- var _positionUpdateFailMessage = "Failed to update html mesh renderer position due to failure to get canvas rect. HtmlMesh instances may not render correctly";
489
- var babylonUnitsToPixels = 100;
490
- // Returns a function that ensures that HtmlMeshes are rendered before all other meshes.
491
- // Note this will only be applied to group 0.
492
- // If neither mesh is an HtmlMesh, then the default render order is used
493
- // This prevents HtmlMeshes from appearing in front of other meshes when they are behind them
494
- var renderOrderFunc = function (defaultRenderOrder) {
495
- return function (subMeshA, subMeshB) {
496
- var meshA = subMeshA.getMesh();
497
- var meshB = subMeshB.getMesh();
498
- // Use property check instead of instanceof since it is less expensive and
499
- // this will be called many times per frame
500
- var meshAIsHtmlMesh = meshA["isHtmlMesh"];
501
- var meshBIsHtmlMesh = meshB["isHtmlMesh"];
502
- if (meshAIsHtmlMesh) {
503
- return meshBIsHtmlMesh ? (meshA.absolutePosition.z <= meshB.absolutePosition.z ? 1 : -1) : -1;
504
- }
505
- else {
506
- return meshBIsHtmlMesh ? 1 : defaultRenderOrder(subMeshA, subMeshB);
507
- }
508
- };
472
+ var FitStrategyContain = {
473
+ wrapElement: function (element) {
474
+ var sizingElement = document.createElement("div");
475
+ sizingElement.style.display = "flex";
476
+ sizingElement.style.justifyContent = "center";
477
+ sizingElement.style.alignItems = "center";
478
+ var scalingElement = document.createElement("div");
479
+ scalingElement.style.visibility = "hidden";
480
+ scalingElement.appendChild(element);
481
+ sizingElement.appendChild(scalingElement);
482
+ return sizingElement;
483
+ },
484
+ updateSize: function (sizingElement, width, height) {
485
+ var scalingElement = sizingElement.firstElementChild;
486
+ sizingElement.style.width = "".concat(width, "px");
487
+ sizingElement.style.height = "".concat(height, "px");
488
+ var _a = [scalingElement.offsetWidth, scalingElement.offsetHeight], childWidth = _a[0], childHeight = _a[1];
489
+ var scale = Math.min(width / childWidth, height / childHeight);
490
+ scalingElement.style.transform = "scale(".concat(scale, ")");
491
+ scalingElement.style.visibility = "visible";
492
+ },
509
493
  };
510
- /**
511
- * An instance of this is required to render HtmlMeshes in the scene.
512
- * if using HtmlMeshes, you must not set render order for group 0 using
513
- * scene.setRenderingOrder. You must instead pass the compare functions
514
- * to the HtmlMeshRenderer constructor. If you do not, then your render
515
- * order will be overwritten if the HtmlMeshRenderer is created after and
516
- * the HtmlMeshes will not render correctly (they will appear in front of
517
- * meshes that are actually in front of them) if the HtmlMeshRenderer is
518
- * created before.
519
- */
520
- var HtmlMeshRenderer = /** @class */ (function () {
521
- /**
522
- * Contruct an instance of HtmlMeshRenderer
523
- * @param scene
524
- * @param options object containing the following optional properties:
525
- * @returns
526
- */
527
- function HtmlMeshRenderer(scene, _a) {
528
- var _b = _a === void 0 ? {} : _a, _c = _b.parentContainerId, parentContainerId = _c === void 0 ? null : _c, _d = _b._containerId, _containerId = _d === void 0 ? "css-container" : _d, _e = _b.enableOverlayRender, enableOverlayRender = _e === void 0 ? true : _e, _f = _b.defaultOpaqueRenderOrder, defaultOpaqueRenderOrder = _f === void 0 ? babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.RenderingGroup.PainterSortCompare : _f, _g = _b.defaultAlphaTestRenderOrder, defaultAlphaTestRenderOrder = _g === void 0 ? babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.RenderingGroup.PainterSortCompare : _g, _h = _b.defaultTransparentRenderOrder, defaultTransparentRenderOrder = _h === void 0 ? babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.RenderingGroup.defaultTransparentSortCompare : _h;
529
- var _this = this;
530
- this._cache = {
531
- cameraData: { fov: 0, position: new babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Vector3(), style: "" },
532
- htmlMeshData: new WeakMap(),
533
- };
534
- this._width = 0;
535
- this._height = 0;
536
- this._heightHalf = 0;
537
- // Create some refs to avoid creating new objects every frame
538
- this._temp = {
539
- scaleTransform: new babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Vector3(),
540
- rotationTransform: new babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Quaternion(),
541
- positionTransform: new babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Vector3(),
542
- objectMatrix: babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.Identity(),
543
- cameraWorldMatrix: babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.Identity(),
544
- cameraRotationMatrix: babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.Identity(),
545
- cameraWorldMatrixAsArray: new Array(16),
546
- };
547
- // Keep track of DPR so we can resize if DPR changes
548
- // Otherwise the DOM content will scale, but the mesh won't
549
- this._lastDevicePixelRatio = window.devicePixelRatio;
550
- // Keep track of camera matrix changes so we only update the
551
- // DOM element styles when necessary
552
- this._cameraMatrixUpdated = true;
553
- // Keep track of position changes so we only update the DOM element
554
- // styles when necessary
555
- this._previousCanvasDocumentPosition = {
556
- top: 0,
557
- left: 0,
558
- };
559
- this._renderObserver = null;
560
- this._onCameraMatrixChanged = function (camera) {
561
- _this._cameraWorldMatrix = camera.getWorldMatrix();
562
- _this._cameraMatrixUpdated = true;
563
- };
564
- // Requires a browser to work. Only init if we are in a browser
565
- if (typeof document === "undefined") {
566
- return;
567
- }
568
- this._containerId = _containerId;
569
- this._init(scene, parentContainerId, enableOverlayRender, defaultOpaqueRenderOrder, defaultAlphaTestRenderOrder, defaultTransparentRenderOrder);
570
- }
571
- /**
572
- * Dispose of the HtmlMeshRenderer
573
- */
574
- HtmlMeshRenderer.prototype.dispose = function () {
575
- var _a, _b;
576
- if (this._renderObserver) {
577
- this._renderObserver.remove();
578
- this._renderObserver = null;
579
- }
580
- (_a = this._overlayElements) === null || _a === void 0 ? void 0 : _a.container.remove();
581
- this._overlayElements = null;
582
- (_b = this._inSceneElements) === null || _b === void 0 ? void 0 : _b.container.remove();
583
- this._inSceneElements = null;
584
- };
585
- HtmlMeshRenderer.prototype._init = function (scene, parentContainerId, enableOverlayRender, defaultOpaqueRenderOrder, defaultAlphaTestRenderOrder, defaultTransparentRenderOrder) {
586
- var _this = this;
587
- var _a;
588
- // Requires a browser to work. Only init if we are in a browser
589
- if (typeof document === "undefined") {
590
- return;
591
- }
592
- // Create the DOM containers
593
- var parentContainer = parentContainerId ? document.getElementById(parentContainerId) : document.body;
594
- if (!parentContainer) {
595
- parentContainer = document.body;
596
- }
597
- // if the container already exists, then remove it
598
- var inSceneContainerId = "".concat(this._containerId, "_in_scene");
599
- this._inSceneElements = this._createRenderLayerElements(inSceneContainerId);
600
- parentContainer.insertBefore(this._inSceneElements.container, parentContainer.firstChild);
601
- if (enableOverlayRender) {
602
- var overlayContainerId = "".concat(this._containerId, "_overlay");
603
- this._overlayElements = this._createRenderLayerElements(overlayContainerId);
604
- var zIndex = +((_a = scene.getEngine().getRenderingCanvas().style.zIndex) !== null && _a !== void 0 ? _a : "0") + 1;
605
- this._overlayElements.container.style.zIndex = "".concat(zIndex);
606
- this._overlayElements.container.style.pointerEvents = "none";
607
- parentContainer.insertBefore(this._overlayElements.container, parentContainer.firstChild);
608
- }
609
- this._engine = scene.getEngine();
610
- var clientRect = this._engine.getRenderingCanvasClientRect();
611
- if (!clientRect) {
612
- throw new Error("Failed to get client rect for rendering canvas");
613
- }
614
- // Set the size and resize behavior
615
- this._setSize(clientRect.width, clientRect.height);
616
- this._engine.onResizeObservable.add(function () {
617
- var clientRect = _this._engine.getRenderingCanvasClientRect();
618
- if (clientRect) {
619
- _this._setSize(clientRect.width, clientRect.height);
620
- }
621
- });
622
- var projectionObs;
623
- var matrixObs;
624
- var observeCamera = function () {
625
- var camera = scene.activeCamera;
626
- if (camera) {
627
- projectionObs = camera.onProjectionMatrixChangedObservable.add(function () {
628
- _this._onCameraMatrixChanged(camera);
629
- });
630
- matrixObs = camera.onViewMatrixChangedObservable.add(function () {
631
- _this._onCameraMatrixChanged(camera);
632
- });
633
- }
634
- };
635
- observeCamera();
636
- scene.onActiveCameraChanged.add(function () {
637
- var _a, _b;
638
- if (projectionObs) {
639
- (_a = scene.activeCamera) === null || _a === void 0 ? void 0 : _a.onProjectionMatrixChangedObservable.remove(projectionObs);
640
- }
641
- if (matrixObs) {
642
- (_b = scene.activeCamera) === null || _b === void 0 ? void 0 : _b.onViewMatrixChangedObservable.remove(matrixObs);
643
- }
644
- observeCamera();
645
- });
646
- // We need to make sure that HtmlMeshes are rendered before all other meshes
647
- // so that they don't appear in front of meshes that are actually in front of them
648
- // Updating the render order isn't ideal, but it is the only way to acheive this
649
- // The implication is that an app using the HtmlMeshRendered must set the scene render order
650
- // via the HtmlMeshRendered constructor
651
- var opaqueRenderOrder = renderOrderFunc(defaultOpaqueRenderOrder);
652
- var alphaTestRenderOrder = renderOrderFunc(defaultAlphaTestRenderOrder);
653
- var transparentRenderOrder = renderOrderFunc(defaultTransparentRenderOrder);
654
- scene.setRenderingOrder(0, opaqueRenderOrder, alphaTestRenderOrder, transparentRenderOrder);
655
- this._renderObserver = scene.onBeforeRenderObservable.add(function () {
656
- _this._render(scene, scene.activeCamera);
657
- });
658
- };
659
- HtmlMeshRenderer.prototype._createRenderLayerElements = function (containerId) {
660
- var existingContainer = document.getElementById(containerId);
661
- if (existingContainer) {
662
- existingContainer.remove();
494
+ var FitStrategyCover = {
495
+ wrapElement: function (element) {
496
+ var sizingElement = document.createElement("div");
497
+ sizingElement.style.display = "flex";
498
+ sizingElement.style.justifyContent = "center";
499
+ sizingElement.style.alignItems = "center";
500
+ sizingElement.style.overflow = "hidden";
501
+ var scalingElement = document.createElement("div");
502
+ scalingElement.style.visibility = "hidden";
503
+ scalingElement.appendChild(element);
504
+ sizingElement.appendChild(scalingElement);
505
+ return sizingElement;
506
+ },
507
+ updateSize: function (sizingElement, width, height) {
508
+ var scalingElement = sizingElement.firstElementChild;
509
+ sizingElement.style.width = "".concat(width, "px");
510
+ sizingElement.style.height = "".concat(height, "px");
511
+ var _a = [scalingElement.offsetWidth, scalingElement.offsetHeight], childWidth = _a[0], childHeight = _a[1];
512
+ var scale = Math.max(width / childWidth, height / childHeight);
513
+ scalingElement.style.transform = "scale(".concat(scale, ")");
514
+ scalingElement.style.visibility = "visible";
515
+ },
516
+ };
517
+ var FitStrategyStretch = {
518
+ wrapElement: function (element) {
519
+ var sizingElement = document.createElement("div");
520
+ sizingElement.style.display = "flex";
521
+ sizingElement.style.justifyContent = "center";
522
+ sizingElement.style.alignItems = "center";
523
+ var scalingElement = document.createElement("div");
524
+ scalingElement.style.visibility = "hidden";
525
+ scalingElement.appendChild(element);
526
+ sizingElement.appendChild(scalingElement);
527
+ return sizingElement;
528
+ },
529
+ updateSize: function (sizingElement, width, height) {
530
+ var scalingElement = sizingElement.firstElementChild;
531
+ sizingElement.style.width = "".concat(width, "px");
532
+ sizingElement.style.height = "".concat(height, "px");
533
+ var _a = [scalingElement.offsetWidth, scalingElement.offsetHeight], childWidth = _a[0], childHeight = _a[1];
534
+ scalingElement.style.transform = "scale(".concat(width / childWidth, ", ").concat(height / childHeight, ")");
535
+ scalingElement.style.visibility = "visible";
536
+ },
537
+ };
538
+ var FitStrategyNone = {
539
+ wrapElement: function (element) {
540
+ return element;
541
+ },
542
+ updateSize: function (sizingElement, width, height) {
543
+ if (sizingElement) {
544
+ sizingElement.style.width = "".concat(width, "px");
545
+ sizingElement.style.height = "".concat(height, "px");
663
546
  }
664
- var container = document.createElement("div");
665
- container.id = containerId;
666
- container.style.position = "absolute";
667
- container.style.width = "100%";
668
- container.style.height = "100%";
669
- container.style.zIndex = "-1";
670
- var domElement = document.createElement("div");
671
- domElement.style.overflow = "hidden";
672
- var cameraElement = document.createElement("div");
673
- cameraElement.style.webkitTransformStyle = "preserve-3d";
674
- cameraElement.style.transformStyle = "preserve-3d";
675
- cameraElement.style.pointerEvents = "none";
676
- domElement.appendChild(cameraElement);
677
- container.appendChild(domElement);
678
- return {
679
- container: container,
680
- domElement: domElement,
681
- cameraElement: cameraElement,
682
- };
683
- };
684
- HtmlMeshRenderer.prototype._getSize = function () {
685
- return {
686
- width: this._width,
687
- height: this._height,
688
- };
689
- };
690
- HtmlMeshRenderer.prototype._setSize = function (width, height) {
691
- this._width = width;
692
- this._height = height;
693
- this._heightHalf = this._height / 2;
694
- if (!this._inSceneElements || !this._overlayElements) {
695
- return;
547
+ },
548
+ };
549
+ var FitStrategy = {
550
+ CONTAIN: FitStrategyContain,
551
+ COVER: FitStrategyCover,
552
+ STRETCH: FitStrategyStretch,
553
+ NONE: FitStrategyNone,
554
+ };
555
+
556
+
557
+ /***/ }),
558
+
559
+ /***/ "../../../dev/addons/src/htmlMesh/htmlMesh.ts":
560
+ /*!****************************************************!*\
561
+ !*** ../../../dev/addons/src/htmlMesh/htmlMesh.ts ***!
562
+ \****************************************************/
563
+ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
564
+
565
+ __webpack_require__.r(__webpack_exports__);
566
+ /* harmony export */ __webpack_require__.d(__webpack_exports__, {
567
+ /* harmony export */ HtmlMesh: () => (/* binding */ HtmlMesh)
568
+ /* harmony export */ });
569
+ /* harmony import */ var tslib__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! tslib */ "../../../../node_modules/tslib/tslib.es6.mjs");
570
+ /* harmony import */ var babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! babylonjs/Misc/logger */ "babylonjs/Maths/math");
571
+ /* harmony import */ var babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__);
572
+ /* harmony import */ var _pointerEventsCaptureBehavior__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./pointerEventsCaptureBehavior */ "../../../dev/addons/src/htmlMesh/pointerEventsCaptureBehavior.ts");
573
+ /* harmony import */ var _fitStrategy__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./fitStrategy */ "../../../dev/addons/src/htmlMesh/fitStrategy.ts");
574
+
575
+
576
+
577
+
578
+
579
+
580
+
581
+
582
+ /**
583
+ * This class represents HTML content that we want to render as though it is part of the scene. The HTML content is actually
584
+ * rendered below the canvas, but a depth mask is created by this class that writes to the depth buffer but does not
585
+ * write to the color buffer, effectively punching a hole in the canvas. CSS transforms are used to scale, translate, and rotate
586
+ * the HTML content so that it matches the camera and mesh orientation. The class supports interactions in editable and non-editable mode.
587
+ * In non-editable mode (the default), events are passed to the HTML content when the pointer is over the mask (and not occluded by other meshes
588
+ * in the scene).
589
+ * #HVHYJC#5
590
+ * #B17TC7#112
591
+ */
592
+ var HtmlMesh = /** @class */ (function (_super) {
593
+ (0,tslib__WEBPACK_IMPORTED_MODULE_3__.__extends)(HtmlMesh, _super);
594
+ /**
595
+ * Contruct an instance of HtmlMesh
596
+ * @param scene
597
+ * @param id The id of the mesh. Will be used as the id of the HTML element as well.
598
+ * @param options object with optional parameters
599
+ */
600
+ function HtmlMesh(scene, id, _a) {
601
+ var _b = _a === void 0 ? {} : _a, _c = _b.captureOnPointerEnter, captureOnPointerEnter = _c === void 0 ? true : _c, _d = _b.isCanvasOverlay, isCanvasOverlay = _d === void 0 ? false : _d, _e = _b.fitStrategy, fitStrategy = _e === void 0 ? _fitStrategy__WEBPACK_IMPORTED_MODULE_2__.FitStrategy.NONE : _e;
602
+ var _this = _super.call(this, id, scene) || this;
603
+ // Override the super class's _isEnabled property so we can control when the mesh
604
+ // is enabled. I.e., we don't want to render the mesh until there is content to show.
605
+ _this._enabled = false;
606
+ // The mesh is ready when content has been set and the content size has been set
607
+ // The former is done by the user, the latter is done by the renderer.
608
+ _this._ready = false;
609
+ /**
610
+ * @internal
611
+ */
612
+ _this._isCanvasOverlay = false;
613
+ _this._requiresUpdate = true;
614
+ _this._inverseScaleMatrix = null;
615
+ _this._captureOnPointerEnter = true;
616
+ _this._pointerEventCaptureBehavior = null;
617
+ _this._sourceWidth = null;
618
+ _this._sourceHeight = null;
619
+ _this._fitStrategy = _fitStrategy__WEBPACK_IMPORTED_MODULE_2__.FitStrategy.NONE;
620
+ // Requires a browser to work. Bail if we aren't running in a browser
621
+ if (typeof document === "undefined") {
622
+ babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.Logger.Warn("Creating an instance of an HtmlMesh with id ".concat(id, " outside of a browser. The mesh will not be visible."));
623
+ return _this;
696
624
  }
697
- var domElements = [this._inSceneElements.domElement, this._overlayElements.domElement, this._inSceneElements.cameraElement, this._overlayElements.cameraElement];
698
- domElements.forEach(function (dom) {
699
- if (dom) {
700
- dom.style.width = "".concat(width, "px");
701
- dom.style.height = "".concat(height, "px");
702
- }
625
+ _this._fitStrategy = fitStrategy;
626
+ _this._isCanvasOverlay = isCanvasOverlay;
627
+ _this._createMask();
628
+ _this._element = _this._createElement();
629
+ // Set enabled by default, so this will show as soon as it's ready
630
+ _this.setEnabled(true);
631
+ _this._captureOnPointerEnter = captureOnPointerEnter;
632
+ // Create a behavior to capture pointer events
633
+ _this._pointerEventCaptureBehavior = new _pointerEventsCaptureBehavior__WEBPACK_IMPORTED_MODULE_1__.PointerEventsCaptureBehavior(_this.capturePointerEvents.bind(_this), _this.releasePointerEvents.bind(_this), {
634
+ captureOnPointerEnter: _this._captureOnPointerEnter,
703
635
  });
704
- };
705
- // prettier-ignore
706
- HtmlMeshRenderer.prototype._getCameraCSSMatrix = function (matrix) {
707
- var elements = matrix.m;
708
- return "matrix3d(".concat(this._epsilon(elements[0]), ",").concat(this._epsilon(-elements[1]), ",").concat(this._epsilon(elements[2]), ",").concat(this._epsilon(elements[3]), ",").concat(this._epsilon(elements[4]), ",").concat(this._epsilon(-elements[5]), ",").concat(this._epsilon(elements[6]), ",").concat(this._epsilon(elements[7]), ",").concat(this._epsilon(elements[8]), ",").concat(this._epsilon(-elements[9]), ",").concat(this._epsilon(elements[10]), ",").concat(this._epsilon(elements[11]), ",").concat(this._epsilon(elements[12]), ",").concat(this._epsilon(-elements[13]), ",").concat(this._epsilon(elements[14]), ",").concat(this._epsilon(elements[15]), ")");
709
- };
710
- // Convert a Babylon world matrix to a CSS matrix
711
- // This also handles conversion from BJS left handed coords
712
- // to CSS right handed coords
713
- // prettier-ignore
714
- HtmlMeshRenderer.prototype._getHtmlContentCSSMatrix = function (matrix, useRightHandedSystem) {
715
- var elements = matrix.m;
716
- // In a right handed coordinate system, the elements 11 to 14 have to change their direction
717
- var direction = useRightHandedSystem ? -1 : 1;
718
- var matrix3d = "matrix3d(".concat(this._epsilon(elements[0]), ",").concat(this._epsilon(elements[1]), ",").concat(this._epsilon(elements[2] * -direction), ",").concat(this._epsilon(elements[3]), ",").concat(this._epsilon(-elements[4]), ",").concat(this._epsilon(-elements[5]), ",").concat(this._epsilon(elements[6] * direction), ",").concat(this._epsilon(-elements[7]), ",").concat(this._epsilon(elements[8] * -direction), ",").concat(this._epsilon(elements[9] * -direction), ",").concat(this._epsilon(elements[10]), ",").concat(this._epsilon(elements[11] * direction), ",").concat(this._epsilon(elements[12] * direction), ",").concat(this._epsilon(elements[13] * direction), ",").concat(this._epsilon(elements[14] * direction), ",").concat(this._epsilon(elements[15]), ")");
719
- return matrix3d;
720
- };
721
- HtmlMeshRenderer.prototype._getTransformationMatrix = function (htmlMesh, useRightHandedSystem) {
722
- var _a;
723
- // Get the camera world matrix
724
- // Make sure the camera world matrix is up to date
725
- if (!this._cameraWorldMatrix) {
726
- this._cameraWorldMatrix = (_a = htmlMesh.getScene().activeCamera) === null || _a === void 0 ? void 0 : _a.getWorldMatrix();
727
- }
728
- if (!this._cameraWorldMatrix) {
729
- return babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.Identity();
730
- }
731
- var objectWorldMatrix = htmlMesh.getWorldMatrix();
732
- // Scale the object matrix by the base scale factor for the mesh
733
- // which is the ratio of the mesh width/height to the renderer
734
- // width/height divided by the babylon units to pixels ratio
735
- var widthScaleFactor = 1;
736
- var heightScaleFactor = 1;
737
- if (htmlMesh.sourceWidth && htmlMesh.sourceHeight) {
738
- widthScaleFactor = htmlMesh.width / (htmlMesh.sourceWidth / babylonUnitsToPixels);
739
- heightScaleFactor = htmlMesh.height / (htmlMesh.sourceHeight / babylonUnitsToPixels);
636
+ _this.addBehavior(_this._pointerEventCaptureBehavior);
637
+ return _this;
638
+ }
639
+ Object.defineProperty(HtmlMesh.prototype, "isHtmlMesh", {
640
+ /**
641
+ * Helps identifying a html mesh from a regular mesh
642
+ */
643
+ get: function () {
644
+ return true;
645
+ },
646
+ enumerable: false,
647
+ configurable: true
648
+ });
649
+ Object.defineProperty(HtmlMesh.prototype, "sourceWidth", {
650
+ /**
651
+ * Return the source width of the content in pixels
652
+ */
653
+ get: function () {
654
+ return this._sourceWidth;
655
+ },
656
+ enumerable: false,
657
+ configurable: true
658
+ });
659
+ Object.defineProperty(HtmlMesh.prototype, "sourceHeight", {
660
+ /**
661
+ * Return the source height of the content in pixels
662
+ */
663
+ get: function () {
664
+ return this._sourceHeight;
665
+ },
666
+ enumerable: false,
667
+ configurable: true
668
+ });
669
+ Object.defineProperty(HtmlMesh.prototype, "width", {
670
+ /**
671
+ * The width of the content in pixels
672
+ */
673
+ get: function () {
674
+ return this._width;
675
+ },
676
+ enumerable: false,
677
+ configurable: true
678
+ });
679
+ Object.defineProperty(HtmlMesh.prototype, "height", {
680
+ /**
681
+ * The height of the content in pixels
682
+ */
683
+ get: function () {
684
+ return this._height;
685
+ },
686
+ enumerable: false,
687
+ configurable: true
688
+ });
689
+ Object.defineProperty(HtmlMesh.prototype, "element", {
690
+ /**
691
+ * The HTML element that is being rendered as a mesh
692
+ */
693
+ get: function () {
694
+ return this._element;
695
+ },
696
+ enumerable: false,
697
+ configurable: true
698
+ });
699
+ Object.defineProperty(HtmlMesh.prototype, "requiresUpdate", {
700
+ /**
701
+ * True if the mesh has been moved, rotated, or scaled since the last time this
702
+ * property was read. This property is reset to false after reading.
703
+ */
704
+ get: function () {
705
+ return this._requiresUpdate;
706
+ },
707
+ enumerable: false,
708
+ configurable: true
709
+ });
710
+ Object.defineProperty(HtmlMesh.prototype, "captureOnPointerEnter", {
711
+ /**
712
+ * Enable capture for the pointer when entering the mesh area
713
+ */
714
+ set: function (captureOnPointerEnter) {
715
+ this._captureOnPointerEnter = captureOnPointerEnter;
716
+ if (this._pointerEventCaptureBehavior) {
717
+ this._pointerEventCaptureBehavior.captureOnPointerEnter = captureOnPointerEnter;
718
+ }
719
+ },
720
+ enumerable: false,
721
+ configurable: true
722
+ });
723
+ /**
724
+ * Disposes of the mesh and the HTML element
725
+ */
726
+ HtmlMesh.prototype.dispose = function () {
727
+ var _a;
728
+ _super.prototype.dispose.call(this);
729
+ (_a = this._element) === null || _a === void 0 ? void 0 : _a.remove();
730
+ this._element = undefined;
731
+ if (this._pointerEventCaptureBehavior) {
732
+ this._pointerEventCaptureBehavior.dispose();
733
+ this._pointerEventCaptureBehavior = null;
740
734
  }
741
- // Apply the scale to the object's world matrix. Note we aren't scaling
742
- // the object, just getting a matrix as though it were scaled, so we can
743
- // scale the content
744
- var scaleTransform = this._temp.scaleTransform;
745
- var rotationTransform = this._temp.rotationTransform;
746
- var positionTransform = this._temp.positionTransform;
747
- var scaledAndTranslatedObjectMatrix = this._temp.objectMatrix;
748
- objectWorldMatrix.decompose(scaleTransform, rotationTransform, positionTransform);
749
- scaleTransform.x *= widthScaleFactor;
750
- scaleTransform.y *= heightScaleFactor;
751
- babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.ComposeToRef(scaleTransform, rotationTransform, positionTransform, scaledAndTranslatedObjectMatrix);
752
- // Adjust direction of 12 and 13 of the transformation matrix based on the handedness of the system
753
- var direction = useRightHandedSystem ? -1 : 1;
754
- // Adjust translation values to be from camera vs world origin
755
- // Note that we are also adjusting these values to be pixels vs Babylon units
756
- var position = htmlMesh.getAbsolutePosition();
757
- scaledAndTranslatedObjectMatrix.setRowFromFloats(3, (-this._cameraWorldMatrix.m[12] + position.x) * babylonUnitsToPixels * direction, (-this._cameraWorldMatrix.m[13] + position.y) * babylonUnitsToPixels * direction, (this._cameraWorldMatrix.m[14] - position.z) * babylonUnitsToPixels, this._cameraWorldMatrix.m[15] * 0.00001 * babylonUnitsToPixels);
758
- // Adjust other values to be pixels vs Babylon units
759
- scaledAndTranslatedObjectMatrix.multiplyAtIndex(3, babylonUnitsToPixels);
760
- scaledAndTranslatedObjectMatrix.multiplyAtIndex(7, babylonUnitsToPixels);
761
- scaledAndTranslatedObjectMatrix.multiplyAtIndex(11, babylonUnitsToPixels);
762
- return scaledAndTranslatedObjectMatrix;
763
735
  };
764
- HtmlMeshRenderer.prototype._renderHtmlMesh = function (htmlMesh, useRightHandedSystem) {
765
- var _a, _b;
766
- if (!htmlMesh.element || !htmlMesh.element.firstElementChild) {
767
- // nothing to render, so bail
736
+ /**
737
+ * @internal
738
+ */
739
+ HtmlMesh.prototype._markAsUpdated = function () {
740
+ this._requiresUpdate = false;
741
+ };
742
+ /**
743
+ * Sets the content of the element to the specified content adjusting the mesh scale to match and making it visible.
744
+ * If the the specified content is undefined, then it will make the mesh invisible. In either case it will clear the
745
+ * element content first.
746
+ * @param element The element to render as a mesh
747
+ * @param width The width of the mesh in Babylon units
748
+ * @param height The height of the mesh in Babylon units
749
+ */
750
+ HtmlMesh.prototype.setContent = function (element, width, height) {
751
+ // If content is changed, we are no longer ready
752
+ this._setAsReady(false);
753
+ // Also invalidate the source width and height
754
+ this._sourceWidth = null;
755
+ this._sourceHeight = null;
756
+ if (!this._element) {
768
757
  return;
769
758
  }
770
- // We need to ensure html mesh data is initialized before
771
- // computing the base scale factor
772
- var htmlMeshData = this._cache.htmlMeshData.get(htmlMesh);
773
- if (!htmlMeshData) {
774
- htmlMeshData = { style: "" };
775
- this._cache.htmlMeshData.set(htmlMesh, htmlMeshData);
776
- }
777
- var cameraElement = htmlMesh._isCanvasOverlay ? (_a = this._overlayElements) === null || _a === void 0 ? void 0 : _a.cameraElement : (_b = this._inSceneElements) === null || _b === void 0 ? void 0 : _b.cameraElement;
778
- if (htmlMesh.element.parentNode !== cameraElement) {
779
- cameraElement.appendChild(htmlMesh.element);
780
- }
781
- // If the htmlMesh content has changed, update the base scale factor
782
- if (htmlMesh.requiresUpdate) {
783
- this._updateBaseScaleFactor(htmlMesh);
759
+ this._width = width;
760
+ this._height = height;
761
+ this._requiresUpdate = true;
762
+ this.scaling.setAll(1);
763
+ if (element) {
764
+ this._element.appendChild(this._fitStrategy.wrapElement(element));
765
+ this._updateScaleIfNecessary();
784
766
  }
785
- // Get the transformation matrix for the html mesh
786
- var scaledAndTranslatedObjectMatrix = this._getTransformationMatrix(htmlMesh, useRightHandedSystem);
787
- var style = "translate(-50%, -50%) ".concat(this._getHtmlContentCSSMatrix(scaledAndTranslatedObjectMatrix, useRightHandedSystem));
788
- // In a right handed system, screens are on the wrong side of the mesh, so we have to rotate by Math.PI which results in the matrix3d seen below
789
- style += "".concat(useRightHandedSystem ? "matrix3d(-1, 0, 0, 0, 0, 1, 0, 0, 0, 0, -1, 0, 0, 0, 0, 1)" : "");
790
- if (htmlMeshData.style !== style) {
791
- htmlMesh.element.style.webkitTransform = style;
792
- htmlMesh.element.style.transform = style;
767
+ if (this.sourceWidth && this.sourceHeight) {
768
+ this._setAsReady(true);
793
769
  }
794
- htmlMesh._markAsUpdated();
795
770
  };
796
- HtmlMeshRenderer.prototype._render = function (scene, camera) {
797
- var _this = this;
798
- var _a, _b, _c, _d, _e, _f;
799
- var needsUpdate = false;
800
- var useRightHandedSystem = scene.useRightHandedSystem;
801
- // Update the container position and size if necessary
802
- this._updateContainerPositionIfNeeded();
803
- // Check for a camera change
804
- if (this._cameraMatrixUpdated) {
805
- this._cameraMatrixUpdated = false;
806
- needsUpdate = true;
807
- }
808
- // If the camera position has changed, then we also need to update
809
- if (camera.position.x !== this._cache.cameraData.position.x ||
810
- camera.position.y !== this._cache.cameraData.position.y ||
811
- camera.position.z !== this._cache.cameraData.position.z) {
812
- this._cache.cameraData.position.copyFrom(camera.position);
813
- needsUpdate = true;
814
- }
815
- // Check for a dpr change
816
- if (window.devicePixelRatio !== this._lastDevicePixelRatio) {
817
- this._lastDevicePixelRatio = window.devicePixelRatio;
818
- babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Logger.Log("In render - dpr changed: ", this._lastDevicePixelRatio);
819
- needsUpdate = true;
771
+ // Overides BABYLON.Mesh.setEnabled
772
+ HtmlMesh.prototype.setEnabled = function (enabled) {
773
+ // Capture requested enabled state
774
+ this._enabled = enabled;
775
+ // If disabling or enabling and we are ready
776
+ if (!enabled || this._ready) {
777
+ this._doSetEnabled(enabled);
820
778
  }
821
- // Check if any meshes need to be updated
822
- var meshesNeedingUpdate = scene.meshes.filter(function (mesh) { return mesh["isHtmlMesh"] && (needsUpdate || mesh.requiresUpdate); });
823
- needsUpdate = needsUpdate || meshesNeedingUpdate.length > 0;
824
- if (!needsUpdate) {
779
+ };
780
+ /**
781
+ * Sets the content size in pixels
782
+ * @param width width of the source
783
+ * @param height height of the source
784
+ */
785
+ HtmlMesh.prototype.setContentSizePx = function (width, height) {
786
+ this._sourceWidth = width;
787
+ this._sourceHeight = height;
788
+ if (!this._element || !this._element.firstElementChild) {
825
789
  return;
826
790
  }
827
- // Get a projection matrix for the camera
828
- var projectionMatrix = camera.getProjectionMatrix();
829
- var fov = projectionMatrix.m[5] * this._heightHalf;
830
- if (this._cache.cameraData.fov !== fov) {
831
- if (camera.mode == babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Camera.PERSPECTIVE_CAMERA) {
832
- [(_a = this._overlayElements) === null || _a === void 0 ? void 0 : _a.domElement, (_b = this._inSceneElements) === null || _b === void 0 ? void 0 : _b.domElement].forEach(function (el) {
833
- if (el) {
834
- el.style.webkitPerspective = fov + "px";
835
- el.style.perspective = fov + "px";
836
- }
837
- });
838
- }
839
- else {
840
- [(_c = this._overlayElements) === null || _c === void 0 ? void 0 : _c.domElement, (_d = this._inSceneElements) === null || _d === void 0 ? void 0 : _d.domElement].forEach(function (el) {
841
- if (el) {
842
- el.style.webkitPerspective = "";
843
- el.style.perspective = "";
844
- }
845
- });
846
- }
847
- this._cache.cameraData.fov = fov;
848
- }
849
- // Get the CSS matrix for the camera (which will include any camera rotation)
850
- if (camera.parent === null) {
851
- camera.computeWorldMatrix();
852
- }
853
- var cameraMatrixWorld = this._temp.cameraWorldMatrix;
854
- cameraMatrixWorld.copyFrom(camera.getWorldMatrix());
855
- var cameraRotationMatrix = this._temp.cameraRotationMatrix;
856
- cameraMatrixWorld.getRotationMatrix().transposeToRef(cameraRotationMatrix);
857
- var cameraMatrixWorldAsArray = this._temp.cameraWorldMatrixAsArray;
858
- cameraMatrixWorld.copyToArray(cameraMatrixWorldAsArray);
859
- // For a few values, we have to adjust the direction based on the handedness of the system
860
- var direction = useRightHandedSystem ? 1 : -1;
861
- cameraMatrixWorldAsArray[1] = cameraRotationMatrix.m[1];
862
- cameraMatrixWorldAsArray[2] = cameraRotationMatrix.m[2] * direction;
863
- cameraMatrixWorldAsArray[4] = cameraRotationMatrix.m[4] * direction;
864
- cameraMatrixWorldAsArray[6] = cameraRotationMatrix.m[6] * direction;
865
- cameraMatrixWorldAsArray[8] = cameraRotationMatrix.m[8] * direction;
866
- cameraMatrixWorldAsArray[9] = cameraRotationMatrix.m[9] * direction;
867
- babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.FromArrayToRef(cameraMatrixWorldAsArray, 0, cameraMatrixWorld);
868
- var cameraCSSMatrix = this._getCameraCSSMatrix(cameraMatrixWorld);
869
- var style = cameraCSSMatrix;
870
- if (this._cache.cameraData.style !== style) {
871
- [(_e = this._inSceneElements) === null || _e === void 0 ? void 0 : _e.cameraElement, (_f = this._overlayElements) === null || _f === void 0 ? void 0 : _f.cameraElement].forEach(function (el) {
872
- if (el) {
873
- el.style.webkitTransform = style;
874
- el.style.transform = style;
875
- }
876
- });
877
- this._cache.cameraData.style = style;
791
+ this._fitStrategy.updateSize(this._element.firstElementChild, width, height);
792
+ this._updateScaleIfNecessary();
793
+ if (this.width && this.height) {
794
+ this._setAsReady(true);
878
795
  }
879
- // _Render objects if necessary
880
- meshesNeedingUpdate.forEach(function (mesh) {
881
- _this._renderHtmlMesh(mesh, useRightHandedSystem);
882
- });
883
796
  };
884
- HtmlMeshRenderer.prototype._updateBaseScaleFactor = function (htmlMesh) {
885
- // Get screen width and height
886
- var screenWidth = this._width;
887
- var screenHeight = this._height;
888
- // Calculate aspect ratios
889
- var htmlMeshAspectRatio = (htmlMesh.width || 1) / (htmlMesh.height || 1);
890
- var screenAspectRatio = screenWidth / screenHeight;
891
- // Adjust screen dimensions based on aspect ratios
892
- if (htmlMeshAspectRatio > screenAspectRatio) {
893
- // If the HTML mesh is wider relative to its height than the screen, adjust the screen width
894
- screenWidth = screenHeight * htmlMeshAspectRatio;
797
+ HtmlMesh.prototype._setAsReady = function (ready) {
798
+ this._ready = ready;
799
+ if (ready) {
800
+ this._doSetEnabled(this._enabled);
895
801
  }
896
802
  else {
897
- // If the HTML mesh is taller relative to its width than the screen, adjust the screen height
898
- screenHeight = screenWidth / htmlMeshAspectRatio;
803
+ this._doSetEnabled(false);
899
804
  }
900
- // Set content to fill screen so we get max resolution when it is shrunk to fit the mesh
901
- htmlMesh.setContentSizePx(screenWidth, screenHeight);
902
805
  };
903
- HtmlMeshRenderer.prototype._updateContainerPositionIfNeeded = function () {
904
- var _this = this;
905
- var _a, _b;
906
- // Determine if the canvas has moved on the screen
907
- var canvasRect = this._engine.getRenderingCanvasClientRect();
908
- // canvas rect may be null if layout not complete
909
- if (!canvasRect) {
910
- babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Logger.Warn(_positionUpdateFailMessage);
911
- return;
912
- }
913
- var scrollTop = window.scrollY;
914
- var scrollLeft = window.scrollX;
915
- var canvasDocumentTop = canvasRect.top + scrollTop;
916
- var canvasDocumentLeft = canvasRect.left + scrollLeft;
917
- if (this._previousCanvasDocumentPosition.top !== canvasDocumentTop || this._previousCanvasDocumentPosition.left !== canvasDocumentLeft) {
918
- this._previousCanvasDocumentPosition.top = canvasDocumentTop;
919
- this._previousCanvasDocumentPosition.left = canvasDocumentLeft;
920
- [(_a = this._inSceneElements) === null || _a === void 0 ? void 0 : _a.container, (_b = this._overlayElements) === null || _b === void 0 ? void 0 : _b.container].forEach(function (container) {
921
- if (!container) {
922
- return;
923
- }
924
- // set the top and left of the css container to match the canvas
925
- var containerParent = container.offsetParent;
926
- var parentRect = containerParent.getBoundingClientRect();
927
- var parentDocumentTop = parentRect.top + scrollTop;
928
- var parentDocumentLeft = parentRect.left + scrollLeft;
929
- var ancestorMarginsAndPadding = _this._getAncestorMarginsAndPadding(containerParent);
930
- // Add the body margin
931
- var bodyStyle = window.getComputedStyle(document.body);
932
- var bodyMarginTop = parseInt(bodyStyle.marginTop, 10);
933
- var bodyMarginLeft = parseInt(bodyStyle.marginLeft, 10);
934
- container.style.top = "".concat(canvasDocumentTop - parentDocumentTop - ancestorMarginsAndPadding.marginTop + ancestorMarginsAndPadding.paddingTop + bodyMarginTop, "px");
935
- container.style.left = "".concat(canvasDocumentLeft - parentDocumentLeft - ancestorMarginsAndPadding.marginLeft + ancestorMarginsAndPadding.paddingLeft + bodyMarginLeft, "px");
806
+ HtmlMesh.prototype._doSetEnabled = function (enabled) {
807
+ var _this = this;
808
+ var _a;
809
+ if (!this._element) {
810
+ return;
811
+ }
812
+ //if enabled, then start listening for changes to the
813
+ // scaling, rotation, and position. otherwise stop listening
814
+ if (enabled && !this._worldMatrixUpdateObserver) {
815
+ this._worldMatrixUpdateObserver = this.onAfterWorldMatrixUpdateObservable.add(function () {
816
+ _this._requiresUpdate = true;
936
817
  });
937
818
  }
819
+ else if (!enabled) {
820
+ (_a = this._worldMatrixUpdateObserver) === null || _a === void 0 ? void 0 : _a.remove();
821
+ this._worldMatrixUpdateObserver = null;
822
+ }
823
+ // If enabled, then revert the content element display
824
+ // otherwise hide it
825
+ this._element.style.display = enabled ? "" : "none";
826
+ // Capture the content z index
827
+ this._setElementZIndex(this.position.z * -10000);
828
+ _super.prototype.setEnabled.call(this, enabled);
938
829
  };
939
- HtmlMeshRenderer.prototype._epsilon = function (value) {
940
- return Math.abs(value) < 1e-10 ? 0 : value;
941
- };
942
- // Get total margins and padding for an element, excluding the body and document margins
943
- HtmlMeshRenderer.prototype._getAncestorMarginsAndPadding = function (element) {
944
- var marginTop = 0;
945
- var marginLeft = 0;
946
- var paddingTop = 0;
947
- var paddingLeft = 0;
948
- while (element && element !== document.body && element !== document.documentElement) {
949
- var style = window.getComputedStyle(element);
950
- marginTop += parseInt(style.marginTop, 10);
951
- marginLeft += parseInt(style.marginLeft, 10);
952
- paddingTop += parseInt(style.paddingTop, 10);
953
- paddingLeft += parseInt(style.paddingLeft, 10);
954
- element = element.offsetParent;
830
+ HtmlMesh.prototype._updateScaleIfNecessary = function () {
831
+ // If we have setContent before, the content scale is baked into the mesh. If we don't reset the vertices to
832
+ // the original size, then we will multiply the scale when we bake the scale below. By applying the inverse, we back out
833
+ // the scaling that has been done so we are starting from the same point.
834
+ // First reset the scale to 1
835
+ this.scaling.setAll(1);
836
+ // Then back out the original vertices changes to match the content scale
837
+ if (this._inverseScaleMatrix) {
838
+ this.bakeTransformIntoVertices(this._inverseScaleMatrix);
839
+ // Clear out the matrix so it doesn't get applied again unless we scale
840
+ this._inverseScaleMatrix = null;
955
841
  }
956
- return { marginTop: marginTop, marginLeft: marginLeft, paddingTop: paddingTop, paddingLeft: paddingLeft };
842
+ // Set scale to match content. Note we can't just scale the mesh, because that will scale the content as well
843
+ // What we need to do is compute a scale matrix and then bake that into the mesh vertices. This will leave the
844
+ // mesh scale at 1, so our content will stay it's original width and height until we scale the mesh.
845
+ var scaleX = this._width || 1;
846
+ var scaleY = this._height || 1;
847
+ var scaleMatrix = babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.Matrix.Scaling(scaleX, scaleY, 1);
848
+ this.bakeTransformIntoVertices(scaleMatrix);
849
+ // Get an inverse of the scale matrix that we can use to back out the scale changes we have made so
850
+ // we don't multiply the scale.
851
+ this._inverseScaleMatrix = new babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.Matrix();
852
+ scaleMatrix.invertToRef(this._inverseScaleMatrix);
957
853
  };
958
- return HtmlMeshRenderer;
959
- }());
960
-
961
-
962
-
963
- /***/ }),
964
-
965
- /***/ "../../../dev/addons/src/htmlMesh/index.ts":
966
- /*!*************************************************!*\
967
- !*** ../../../dev/addons/src/htmlMesh/index.ts ***!
968
- \*************************************************/
969
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
970
-
971
- __webpack_require__.r(__webpack_exports__);
972
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
973
- /* harmony export */ FitStrategy: () => (/* reexport safe */ _fitStrategy__WEBPACK_IMPORTED_MODULE_3__.FitStrategy),
974
- /* harmony export */ HtmlMesh: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_1__.HtmlMesh),
975
- /* harmony export */ HtmlMeshRenderer: () => (/* reexport safe */ _htmlMeshRenderer__WEBPACK_IMPORTED_MODULE_0__.HtmlMeshRenderer),
976
- /* harmony export */ PointerEventsCaptureBehavior: () => (/* reexport safe */ _pointerEventsCaptureBehavior__WEBPACK_IMPORTED_MODULE_2__.PointerEventsCaptureBehavior)
977
- /* harmony export */ });
978
- /* harmony import */ var _htmlMeshRenderer__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./htmlMeshRenderer */ "../../../dev/addons/src/htmlMesh/htmlMeshRenderer.ts");
979
- /* harmony import */ var _htmlMesh__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./htmlMesh */ "../../../dev/addons/src/htmlMesh/htmlMesh.ts");
980
- /* harmony import */ var _pointerEventsCaptureBehavior__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./pointerEventsCaptureBehavior */ "../../../dev/addons/src/htmlMesh/pointerEventsCaptureBehavior.ts");
981
- /* harmony import */ var _fitStrategy__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./fitStrategy */ "../../../dev/addons/src/htmlMesh/fitStrategy.ts");
982
-
983
-
984
-
985
-
986
- // Export public classes and functions
987
-
988
-
989
-
990
- /***/ }),
991
-
992
- /***/ "../../../dev/addons/src/htmlMesh/pointerEventsCapture.ts":
993
- /*!****************************************************************!*\
994
- !*** ../../../dev/addons/src/htmlMesh/pointerEventsCapture.ts ***!
995
- \****************************************************************/
996
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
997
-
998
- __webpack_require__.r(__webpack_exports__);
999
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1000
- /* harmony export */ getCapturingId: () => (/* binding */ getCapturingId),
1001
- /* harmony export */ releaseCurrent: () => (/* binding */ releaseCurrent),
1002
- /* harmony export */ requestCapture: () => (/* binding */ requestCapture),
1003
- /* harmony export */ requestRelease: () => (/* binding */ requestRelease)
1004
- /* harmony export */ });
1005
- /* harmony import */ var babylonjs_Misc_tools__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! babylonjs/Misc/tools */ "babylonjs/Meshes/mesh");
1006
- /* harmony import */ var babylonjs_Misc_tools__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(babylonjs_Misc_tools__WEBPACK_IMPORTED_MODULE_0__);
1007
-
1008
- var captureRequestQueue = [];
1009
- // Key is request id, value is object with capture and release callbacks
1010
- var pendingRequestCallbacks = new Map();
1011
- // Keep track of release requests with no matching capture request
1012
- // in case the release request arrived before the capture to avoid
1013
- // the capture request never getting released.
1014
- var unmatchedReleaseRequests = [];
1015
- var currentOwner = null; // Called on first capture or release request
1016
- /**
1017
- * Get the id of the object currently capturing pointer events
1018
- * @returns The id of the object currently capturing pointer events
1019
- * or null if no object is capturing pointer events
1020
- */
1021
- var getCapturingId = function () {
1022
- return currentOwner;
1023
- };
1024
- /**
1025
- * Request that the object with the given id capture pointer events. If there is no current
1026
- * owner, then the request is granted immediately. If there is a current owner, then the request
1027
- * is queued until the current owner releases pointer events.
1028
- * @param requestId An id to identify the request. This id will be used to match the capture
1029
- * request with the release request.
1030
- * @param captureCallback The callback to call when the request is granted and the object is capturing
1031
- * @param releaseCallback The callback to call when the object is no longer capturing pointer events
1032
- */
1033
- var requestCapture = function (requestId, captureCallback, releaseCallback) {
1034
- debugLog("In pointerEventsCapture.requestCapture - Pointer events capture requested for ".concat(requestId));
1035
- // If there is a release for this request, then ignore the request
1036
- if (removeUnmatchedRequest(requestId)) {
1037
- debugLog("In pointerEventsCapture.requestCapture - Capture request matched previous release request ".concat(requestId, ". Cancelling capture request"));
1038
- return;
1039
- }
1040
- else if (requestId !== currentOwner) {
1041
- // if the request is not already in the queue, add it to the queue
1042
- enqueueCaptureRequest(requestId, captureCallback, releaseCallback);
1043
- }
1044
- if (!currentOwner) {
1045
- // If there is no current owner, go ahead and grant the request
1046
- transferPointerEventsOwnership();
1047
- }
1048
- // If the request id is the current owner, do nothing
1049
- };
1050
- /**
1051
- * Release pointer events from the object with the given id. If the object is the current owner
1052
- * then pointer events are released immediately. If the object is not the current owner, then the
1053
- * associated capture request is removed from the queue. If there is no matching capture request
1054
- * in the queue, then the release request is added to a list of unmatched release requests and will
1055
- * negate the next capture request with the same id. This is to guard against the possibility that
1056
- * the release request arrived before the capture request.
1057
- * @param requestId The id which should match the id of the capture request
1058
- */
1059
- var requestRelease = function (requestId) {
1060
- debugLog("In pointerEventsCapture.requestRelease - Pointer events release requested for ".concat(requestId));
1061
- // if the requestId is the current capture holder release it
1062
- if (!requestId || requestId === currentOwner) {
1063
- transferPointerEventsOwnership();
1064
- }
1065
- else if (cancelRequest(requestId)) {
1066
- // if the request is in the queue, but not the current capture holder, remove it and it's callbacks
1067
- pendingRequestCallbacks.delete(requestId);
1068
- }
1069
- else {
1070
- debugLog("In pointerEventsCapture.requestRelease - Received release request ".concat(requestId, " but no matching capture request was received"));
1071
- // request was not current and not in queue, likely because we received a release
1072
- // request before the capture. Add it to the unmatched list to guard against this possibility
1073
- if (!unmatchedReleaseRequests.includes(requestId)) {
1074
- unmatchedReleaseRequests.push(requestId);
854
+ HtmlMesh.prototype._createMask = function () {
855
+ var vertexData = (0,babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.CreatePlaneVertexData)({ width: 1, height: 1 });
856
+ vertexData.applyToMesh(this);
857
+ var scene = this.getScene();
858
+ this.checkCollisions = true;
859
+ var depthMask = new babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.StandardMaterial("".concat(this.id, "-mat"), scene);
860
+ if (!this._isCanvasOverlay) {
861
+ depthMask.backFaceCulling = false;
862
+ depthMask.disableColorWrite = true;
863
+ depthMask.disableLighting = true;
1075
864
  }
1076
- }
1077
- };
1078
- /**
1079
- * Relase pointer events from the current owner
1080
- */
1081
- var releaseCurrent = function () {
1082
- requestRelease(currentOwner);
1083
- };
1084
- var enqueueCaptureRequest = function (requestId, capture, release) {
1085
- debugLog("In pointerEventsCapture.enqueueCaptureRequest - Enqueueing capture request for ".concat(requestId));
1086
- if (!captureRequestQueue.includes(requestId)) {
1087
- captureRequestQueue.push(requestId);
1088
- pendingRequestCallbacks.set(requestId, { capture: capture, release: release });
1089
- }
1090
- };
1091
- // Removes the request from the queue if it exists. Returns true
1092
- // if the request was found and removed, otherwise false
1093
- var cancelRequest = function (requestId) {
1094
- var removed = false;
1095
- captureRequestQueue = captureRequestQueue.filter(function (id) {
1096
- if (id !== requestId) {
1097
- return true;
865
+ this.material = depthMask;
866
+ // Optimization - Freeze material since it never needs to change
867
+ this.material.freeze();
868
+ };
869
+ HtmlMesh.prototype._setElementZIndex = function (zIndex) {
870
+ if (this._element) {
871
+ this._element.style.zIndex = "".concat(zIndex);
1098
872
  }
1099
- else {
1100
- removed = true;
1101
- debugLog("In pointerEventsCapture.cancelRequest - Canceling pointer events capture request ".concat(requestId));
1102
- return false;
873
+ };
874
+ /**
875
+ * Callback used by the PointerEventsCaptureBehavior to capture pointer events
876
+ */
877
+ HtmlMesh.prototype.capturePointerEvents = function () {
878
+ if (!this._element) {
879
+ return;
1103
880
  }
1104
- });
1105
- return removed;
1106
- };
1107
- var removeUnmatchedRequest = function (requestId) {
1108
- var removed = false;
1109
- unmatchedReleaseRequests = unmatchedReleaseRequests.filter(function (id) {
1110
- if (id !== requestId) {
1111
- return true;
881
+ // Enable dom content to capture pointer events
882
+ this._element.style.pointerEvents = "auto";
883
+ // Supress events outside of the dom content
884
+ document.getElementsByTagName("body")[0].style.pointerEvents = "none";
885
+ };
886
+ /**
887
+ * Callback used by the PointerEventsCaptureBehavior to release pointer events
888
+ */
889
+ HtmlMesh.prototype.releasePointerEvents = function () {
890
+ if (!this._element) {
891
+ return;
1112
892
  }
1113
- else {
1114
- removed = true;
1115
- return false;
893
+ // Enable pointer events on canvas
894
+ document.getElementsByTagName("body")[0].style.pointerEvents = "auto";
895
+ // Disable pointer events on dom content
896
+ this._element.style.pointerEvents = "none";
897
+ };
898
+ HtmlMesh.prototype._createElement = function () {
899
+ // Requires a browser to work. Bail if we aren't running in a browser
900
+ if (typeof document === "undefined") {
901
+ return;
1116
902
  }
1117
- });
1118
- return removed;
1119
- };
1120
- var transferPointerEventsOwnership = function () {
1121
- var newOwnerId = nextCaptureRequest();
1122
- debugLog("In pointerEventsCapture.transferPointerEventsOwnership - Transferrring pointer events from ".concat(currentOwner, " to ").concat(newOwnerId));
1123
- // Release the current owner
1124
- doRelease();
1125
- if (newOwnerId) {
1126
- doCapture(newOwnerId);
1127
- }
1128
- };
1129
- var doRelease = function () {
1130
- var _a;
1131
- debugLog("In pointerEventsCapture.doRelease - Releasing pointer events from ".concat(currentOwner));
1132
- if (currentOwner) {
1133
- // call the release callback
1134
- (_a = pendingRequestCallbacks.get(currentOwner)) === null || _a === void 0 ? void 0 : _a.release();
1135
- // And remove the callbacks
1136
- pendingRequestCallbacks.delete(currentOwner);
1137
- currentOwner = null;
1138
- }
1139
- };
1140
- var doCapture = function (newOwnerId) {
1141
- var _a;
1142
- if (newOwnerId) {
1143
- // call the capture callback
1144
- (_a = pendingRequestCallbacks.get(newOwnerId)) === null || _a === void 0 ? void 0 : _a.capture();
1145
- }
1146
- currentOwner = newOwnerId;
1147
- debugLog("In pointerEventsCapture.doCapture - Pointer events now captured by ".concat(newOwnerId));
1148
- };
1149
- var nextCaptureRequest = function () {
1150
- return captureRequestQueue.length > 0 ? captureRequestQueue.shift() : null;
1151
- };
1152
- var debugLog = function (message) {
1153
- // If we are runnning in a test runner (in node, so window is not defined)
1154
- // or if the debug flag is set, then log the message
1155
- if (typeof window === "undefined" || window["pointer-events-capture-debug"]) {
1156
- babylonjs_Misc_tools__WEBPACK_IMPORTED_MODULE_0__.Tools.Log("".concat(performance.now(), " - game.scene.pointerEvents - ").concat(message, "\ncurrentOwner: ").concat(currentOwner, "\nqueue: ").concat(captureRequestQueue, "\nunmatched: ").concat(unmatchedReleaseRequests));
1157
- }
1158
- };
1159
- // #endregion Debugging support
903
+ var div = document.createElement("div");
904
+ div.id = this.id;
905
+ div.style.backgroundColor = this._isCanvasOverlay ? "transparent" : "#000";
906
+ div.style.zIndex = "1";
907
+ div.style.position = "absolute";
908
+ div.style.pointerEvents = "none";
909
+ div.style.backfaceVisibility = "hidden";
910
+ return div;
911
+ };
912
+ return HtmlMesh;
913
+ }(babylonjs_Meshes_mesh__WEBPACK_IMPORTED_MODULE_0__.Mesh));
914
+
1160
915
 
1161
916
 
1162
917
  /***/ }),
1163
918
 
1164
- /***/ "../../../dev/addons/src/htmlMesh/pointerEventsCaptureBehavior.ts":
1165
- /*!************************************************************************!*\
1166
- !*** ../../../dev/addons/src/htmlMesh/pointerEventsCaptureBehavior.ts ***!
1167
- \************************************************************************/
919
+ /***/ "../../../dev/addons/src/htmlMesh/htmlMeshRenderer.ts":
920
+ /*!************************************************************!*\
921
+ !*** ../../../dev/addons/src/htmlMesh/htmlMeshRenderer.ts ***!
922
+ \************************************************************/
1168
923
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1169
924
 
1170
925
  __webpack_require__.r(__webpack_exports__);
1171
926
  /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1172
- /* harmony export */ PointerEventsCaptureBehavior: () => (/* binding */ PointerEventsCaptureBehavior)
927
+ /* harmony export */ HtmlMeshRenderer: () => (/* binding */ HtmlMeshRenderer)
1173
928
  /* harmony export */ });
1174
- /* harmony import */ var babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! babylonjs/Misc/logger */ "babylonjs/Meshes/mesh");
1175
- /* harmony import */ var babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__);
1176
- /* harmony import */ var _pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./pointerEventsCapture */ "../../../dev/addons/src/htmlMesh/pointerEventsCapture.ts");
929
+ /* harmony import */ var babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! babylonjs/Misc/logger */ "babylonjs/Maths/math");
930
+ /* harmony import */ var babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__);
1177
931
 
1178
932
 
1179
- // Module level variable for holding the current scene
1180
- var _scene = null;
1181
- // Module level variable to hold the count of behavior instances that are currently capturing pointer events
1182
- // on entry. This is used to determine if we need to start or stop observing pointer movement.
1183
- var captureOnEnterCount = 0;
1184
- // Map used to store instance of the PointerEventsCaptureBehavior for a mesh
1185
- // We do this because this gets checked on pointer move and we don't want to
1186
- // use getBehaviorByName() because that is a linear search
1187
- var meshToBehaviorMap = new WeakMap();
1188
- var startCaptureOnEnter = function (scene) {
1189
- // If we are not in a browser, do nothing
1190
- if (typeof document === "undefined") {
1191
- return;
1192
- }
1193
- if (captureOnEnterCount === 0) {
1194
- document.addEventListener("pointermove", onPointerMove);
1195
- document.addEventListener("touchstart", onPointerMove);
1196
- _scene = _scene !== null && _scene !== void 0 ? _scene : scene;
1197
- babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log("PointerEventsCaptureBehavior: Starting observation of pointer move events.");
1198
- _scene.onDisposeObservable.add(doStopCaptureOnEnter);
1199
- }
1200
- captureOnEnterCount++;
1201
- };
1202
- var doStopCaptureOnEnter = function () {
1203
- document.removeEventListener("pointermove", onPointerMove);
1204
- document.removeEventListener("touchstart", onPointerMove);
1205
- _scene = null;
1206
- babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log("PointerEventsCaptureBehavior: Stopping observation of pointer move events.");
1207
- captureOnEnterCount = 0;
1208
- };
1209
- var stopCaptureOnEnter = function () {
1210
- // If we are not in a browser, do nothing
1211
- if (typeof document === "undefined") {
1212
- return;
1213
- }
1214
- // If we are not observing pointer movement, do nothing
1215
- if (!_scene) {
1216
- return;
1217
- }
1218
- captureOnEnterCount--;
1219
- if (captureOnEnterCount <= 0) {
1220
- doStopCaptureOnEnter();
1221
- }
1222
- };
1223
- // Module level function used to determine if an entered mesh should capture pointer events
1224
- var onPointerMove = function (evt) {
1225
- if (!_scene) {
1226
- return;
1227
- }
1228
- var canvasRect = _scene.getEngine().getRenderingCanvasClientRect();
1229
- if (!canvasRect) {
1230
- return;
1231
- }
1232
- // Get the object that contains the client X and Y from either the pointer event or from the
1233
- // TouchEvent touch
1234
- var _a = "touches" in evt ? evt.touches[0] : evt, clientX = _a.clientX, clientY = _a.clientY;
1235
- // get the picked mesh, if any
1236
- var pointerScreenX = clientX - canvasRect.left;
1237
- var pointerScreenY = clientY - canvasRect.top;
1238
- var pointerCaptureBehavior;
1239
- var pickResult = _scene.pick(pointerScreenX, pointerScreenY, function (mesh) {
1240
- // If the mesh has an instance of PointerEventsCaptureBehavior attached to it,
1241
- // and capture on pointer enter is true, then we want to pick it
1242
- var pointerCaptureBehavior = meshToBehaviorMap.get(mesh);
1243
- return mesh.isEnabled() && typeof pointerCaptureBehavior !== "undefined" && pointerCaptureBehavior._captureOnPointerEnter;
1244
- });
1245
- var pickedMesh;
1246
- if (pickResult.hit) {
1247
- pickedMesh = pickResult.pickedMesh;
1248
- }
1249
- else {
1250
- pickedMesh = null;
1251
- }
1252
- var capturingIdAsInt = parseInt((0,_pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__.getCapturingId)() || "");
1253
- // if the picked mesh is the current capturing mesh, do nothing
1254
- if (pickedMesh && pickedMesh.uniqueId === capturingIdAsInt) {
1255
- return;
1256
- }
1257
- // If there is a capturing mesh and it is not the current picked mesh, or no
1258
- // mesh is picked, release the capturing mesh
1259
- if (capturingIdAsInt && (!pickedMesh || pickedMesh.uniqueId !== capturingIdAsInt)) {
1260
- (0,_pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__.releaseCurrent)();
1261
- }
1262
- // If there is a picked mesh and it is not the current capturing mesh, capture
1263
- // the pointer events. Note that the current capturing mesh has already been
1264
- // released above
1265
- if (pickedMesh) {
1266
- pointerCaptureBehavior = meshToBehaviorMap.get(pickedMesh);
1267
- pointerCaptureBehavior.capturePointerEvents();
1268
- }
933
+
934
+
935
+ var _positionUpdateFailMessage = "Failed to update html mesh renderer position due to failure to get canvas rect. HtmlMesh instances may not render correctly";
936
+ var babylonUnitsToPixels = 100;
937
+ // Returns a function that ensures that HtmlMeshes are rendered before all other meshes.
938
+ // Note this will only be applied to group 0.
939
+ // If neither mesh is an HtmlMesh, then the default render order is used
940
+ // This prevents HtmlMeshes from appearing in front of other meshes when they are behind them
941
+ var renderOrderFunc = function (defaultRenderOrder) {
942
+ return function (subMeshA, subMeshB) {
943
+ var meshA = subMeshA.getMesh();
944
+ var meshB = subMeshB.getMesh();
945
+ // Use property check instead of instanceof since it is less expensive and
946
+ // this will be called many times per frame
947
+ var meshAIsHtmlMesh = meshA["isHtmlMesh"];
948
+ var meshBIsHtmlMesh = meshB["isHtmlMesh"];
949
+ if (meshAIsHtmlMesh) {
950
+ return meshBIsHtmlMesh ? (meshA.absolutePosition.z <= meshB.absolutePosition.z ? 1 : -1) : -1;
951
+ }
952
+ else {
953
+ return meshBIsHtmlMesh ? 1 : defaultRenderOrder(subMeshA, subMeshB);
954
+ }
955
+ };
1269
956
  };
1270
957
  /**
1271
- * Behavior for any content that can capture pointer events, i.e. bypass the Babylon pointer event handling
1272
- * and receive pointer events directly. It will register the capture triggers and negotiate the capture and
1273
- * release of pointer events. Curerntly this applies only to HtmlMesh
958
+ * An instance of this is required to render HtmlMeshes in the scene.
959
+ * if using HtmlMeshes, you must not set render order for group 0 using
960
+ * scene.setRenderingOrder. You must instead pass the compare functions
961
+ * to the HtmlMeshRenderer constructor. If you do not, then your render
962
+ * order will be overwritten if the HtmlMeshRenderer is created after and
963
+ * the HtmlMeshes will not render correctly (they will appear in front of
964
+ * meshes that are actually in front of them) if the HtmlMeshRenderer is
965
+ * created before.
1274
966
  */
1275
- var PointerEventsCaptureBehavior = /** @class */ (function () {
1276
- function PointerEventsCaptureBehavior(_captureCallback, _releaseCallback, _a) {
1277
- var _b = _a === void 0 ? {} : _a, _c = _b.captureOnPointerEnter, captureOnPointerEnter = _c === void 0 ? true : _c;
1278
- this._captureCallback = _captureCallback;
1279
- this._releaseCallback = _releaseCallback;
1280
- /** gets or sets behavior's name */
1281
- this.name = "PointerEventsCaptureBehavior";
1282
- this._attachedMesh = null;
1283
- this._captureOnPointerEnter = captureOnPointerEnter;
1284
- // Warn if we are not in a browser
967
+ var HtmlMeshRenderer = /** @class */ (function () {
968
+ /**
969
+ * Contruct an instance of HtmlMeshRenderer
970
+ * @param scene
971
+ * @param options object containing the following optional properties:
972
+ * @returns
973
+ */
974
+ function HtmlMeshRenderer(scene, _a) {
975
+ var _b = _a === void 0 ? {} : _a, _c = _b.parentContainerId, parentContainerId = _c === void 0 ? null : _c, _d = _b._containerId, _containerId = _d === void 0 ? "css-container" : _d, _e = _b.enableOverlayRender, enableOverlayRender = _e === void 0 ? true : _e, _f = _b.defaultOpaqueRenderOrder, defaultOpaqueRenderOrder = _f === void 0 ? babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.RenderingGroup.PainterSortCompare : _f, _g = _b.defaultAlphaTestRenderOrder, defaultAlphaTestRenderOrder = _g === void 0 ? babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.RenderingGroup.PainterSortCompare : _g, _h = _b.defaultTransparentRenderOrder, defaultTransparentRenderOrder = _h === void 0 ? babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.RenderingGroup.defaultTransparentSortCompare : _h;
976
+ var _this = this;
977
+ this._cache = {
978
+ cameraData: { fov: 0, position: new babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Vector3(), style: "" },
979
+ htmlMeshData: new WeakMap(),
980
+ };
981
+ this._width = 0;
982
+ this._height = 0;
983
+ this._heightHalf = 0;
984
+ // Create some refs to avoid creating new objects every frame
985
+ this._temp = {
986
+ scaleTransform: new babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Vector3(),
987
+ rotationTransform: new babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Quaternion(),
988
+ positionTransform: new babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Vector3(),
989
+ objectMatrix: babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.Identity(),
990
+ cameraWorldMatrix: babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.Identity(),
991
+ cameraRotationMatrix: babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.Identity(),
992
+ cameraWorldMatrixAsArray: new Array(16),
993
+ };
994
+ // Keep track of DPR so we can resize if DPR changes
995
+ // Otherwise the DOM content will scale, but the mesh won't
996
+ this._lastDevicePixelRatio = window.devicePixelRatio;
997
+ // Keep track of camera matrix changes so we only update the
998
+ // DOM element styles when necessary
999
+ this._cameraMatrixUpdated = true;
1000
+ // Keep track of position changes so we only update the DOM element
1001
+ // styles when necessary
1002
+ this._previousCanvasDocumentPosition = {
1003
+ top: 0,
1004
+ left: 0,
1005
+ };
1006
+ this._renderObserver = null;
1007
+ this._onCameraMatrixChanged = function (camera) {
1008
+ _this._cameraWorldMatrix = camera.getWorldMatrix();
1009
+ _this._cameraMatrixUpdated = true;
1010
+ };
1011
+ // Requires a browser to work. Only init if we are in a browser
1012
+ if (typeof document === "undefined") {
1013
+ return;
1014
+ }
1015
+ this._containerId = _containerId;
1016
+ this._init(scene, parentContainerId, enableOverlayRender, defaultOpaqueRenderOrder, defaultAlphaTestRenderOrder, defaultTransparentRenderOrder);
1017
+ }
1018
+ /**
1019
+ * Dispose of the HtmlMeshRenderer
1020
+ */
1021
+ HtmlMeshRenderer.prototype.dispose = function () {
1022
+ var _a, _b;
1023
+ if (this._renderObserver) {
1024
+ this._renderObserver.remove();
1025
+ this._renderObserver = null;
1026
+ }
1027
+ (_a = this._overlayElements) === null || _a === void 0 ? void 0 : _a.container.remove();
1028
+ this._overlayElements = null;
1029
+ (_b = this._inSceneElements) === null || _b === void 0 ? void 0 : _b.container.remove();
1030
+ this._inSceneElements = null;
1031
+ };
1032
+ HtmlMeshRenderer.prototype._init = function (scene, parentContainerId, enableOverlayRender, defaultOpaqueRenderOrder, defaultAlphaTestRenderOrder, defaultTransparentRenderOrder) {
1033
+ var _this = this;
1034
+ var _a;
1035
+ // Requires a browser to work. Only init if we are in a browser
1285
1036
  if (typeof document === "undefined") {
1286
- babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Warn("Creating an instance of PointerEventsCaptureBehavior outside of a browser. The behavior will not work.");
1037
+ return;
1287
1038
  }
1288
- }
1289
- Object.defineProperty(PointerEventsCaptureBehavior.prototype, "attachedMesh", {
1290
- /**
1291
- * Gets or sets the mesh that the behavior is attached to
1292
- */
1293
- get: function () {
1294
- return this._attachedMesh;
1295
- },
1296
- set: function (value) {
1297
- this._attachedMesh = value;
1298
- },
1299
- enumerable: false,
1300
- configurable: true
1301
- });
1302
- Object.defineProperty(PointerEventsCaptureBehavior.prototype, "captureOnPointerEnter", {
1303
- /**
1304
- * Set if the behavior should capture pointer events when the pointer enters the mesh
1305
- */
1306
- set: function (captureOnPointerEnter) {
1307
- if (this._captureOnPointerEnter === captureOnPointerEnter) {
1308
- return;
1039
+ // Create the DOM containers
1040
+ var parentContainer = parentContainerId ? document.getElementById(parentContainerId) : document.body;
1041
+ if (!parentContainer) {
1042
+ parentContainer = document.body;
1043
+ }
1044
+ // if the container already exists, then remove it
1045
+ var inSceneContainerId = "".concat(this._containerId, "_in_scene");
1046
+ this._inSceneElements = this._createRenderLayerElements(inSceneContainerId);
1047
+ parentContainer.insertBefore(this._inSceneElements.container, parentContainer.firstChild);
1048
+ if (enableOverlayRender) {
1049
+ var overlayContainerId = "".concat(this._containerId, "_overlay");
1050
+ this._overlayElements = this._createRenderLayerElements(overlayContainerId);
1051
+ var zIndex = +((_a = scene.getEngine().getRenderingCanvas().style.zIndex) !== null && _a !== void 0 ? _a : "0") + 1;
1052
+ this._overlayElements.container.style.zIndex = "".concat(zIndex);
1053
+ this._overlayElements.container.style.pointerEvents = "none";
1054
+ parentContainer.insertBefore(this._overlayElements.container, parentContainer.firstChild);
1055
+ }
1056
+ this._engine = scene.getEngine();
1057
+ var clientRect = this._engine.getRenderingCanvasClientRect();
1058
+ if (!clientRect) {
1059
+ throw new Error("Failed to get client rect for rendering canvas");
1060
+ }
1061
+ // Set the size and resize behavior
1062
+ this._setSize(clientRect.width, clientRect.height);
1063
+ this._engine.onResizeObservable.add(function () {
1064
+ var clientRect = _this._engine.getRenderingCanvasClientRect();
1065
+ if (clientRect) {
1066
+ _this._setSize(clientRect.width, clientRect.height);
1309
1067
  }
1310
- this._captureOnPointerEnter = captureOnPointerEnter;
1311
- if (this._attachedMesh) {
1312
- if (this._captureOnPointerEnter) {
1313
- startCaptureOnEnter(this._attachedMesh.getScene());
1314
- }
1315
- else {
1316
- stopCaptureOnEnter();
1317
- }
1068
+ });
1069
+ var projectionObs;
1070
+ var matrixObs;
1071
+ var observeCamera = function () {
1072
+ var camera = scene.activeCamera;
1073
+ if (camera) {
1074
+ projectionObs = camera.onProjectionMatrixChangedObservable.add(function () {
1075
+ _this._onCameraMatrixChanged(camera);
1076
+ });
1077
+ matrixObs = camera.onViewMatrixChangedObservable.add(function () {
1078
+ _this._onCameraMatrixChanged(camera);
1079
+ });
1318
1080
  }
1319
- },
1320
- enumerable: false,
1321
- configurable: true
1322
- });
1323
- /**
1324
- * Function called when the behavior needs to be initialized (before attaching it to a target)
1325
- */
1326
- PointerEventsCaptureBehavior.prototype.init = function () { };
1327
- /**
1328
- * Called when the behavior is attached to a target
1329
- * @param mesh defines the target where the behavior is attached to
1330
- */
1331
- PointerEventsCaptureBehavior.prototype.attach = function (mesh) {
1332
- // Add a reference to this behavior on the mesh. We do this so we can get a
1333
- // reference to the behavior in the onPointerMove function without relying on
1334
- // getBehaviorByName(), which does a linear search of the behaviors array.
1335
- this.attachedMesh = mesh;
1336
- meshToBehaviorMap.set(mesh, this);
1337
- if (this._captureOnPointerEnter) {
1338
- startCaptureOnEnter(mesh.getScene());
1081
+ };
1082
+ observeCamera();
1083
+ scene.onActiveCameraChanged.add(function () {
1084
+ var _a, _b;
1085
+ if (projectionObs) {
1086
+ (_a = scene.activeCamera) === null || _a === void 0 ? void 0 : _a.onProjectionMatrixChangedObservable.remove(projectionObs);
1087
+ }
1088
+ if (matrixObs) {
1089
+ (_b = scene.activeCamera) === null || _b === void 0 ? void 0 : _b.onViewMatrixChangedObservable.remove(matrixObs);
1090
+ }
1091
+ observeCamera();
1092
+ });
1093
+ // We need to make sure that HtmlMeshes are rendered before all other meshes
1094
+ // so that they don't appear in front of meshes that are actually in front of them
1095
+ // Updating the render order isn't ideal, but it is the only way to acheive this
1096
+ // The implication is that an app using the HtmlMeshRendered must set the scene render order
1097
+ // via the HtmlMeshRendered constructor
1098
+ var opaqueRenderOrder = renderOrderFunc(defaultOpaqueRenderOrder);
1099
+ var alphaTestRenderOrder = renderOrderFunc(defaultAlphaTestRenderOrder);
1100
+ var transparentRenderOrder = renderOrderFunc(defaultTransparentRenderOrder);
1101
+ scene.setRenderingOrder(0, opaqueRenderOrder, alphaTestRenderOrder, transparentRenderOrder);
1102
+ this._renderObserver = scene.onBeforeRenderObservable.add(function () {
1103
+ _this._render(scene, scene.activeCamera);
1104
+ });
1105
+ };
1106
+ HtmlMeshRenderer.prototype._createRenderLayerElements = function (containerId) {
1107
+ var existingContainer = document.getElementById(containerId);
1108
+ if (existingContainer) {
1109
+ existingContainer.remove();
1110
+ }
1111
+ var container = document.createElement("div");
1112
+ container.id = containerId;
1113
+ container.style.position = "absolute";
1114
+ container.style.width = "100%";
1115
+ container.style.height = "100%";
1116
+ container.style.zIndex = "-1";
1117
+ var domElement = document.createElement("div");
1118
+ domElement.style.overflow = "hidden";
1119
+ var cameraElement = document.createElement("div");
1120
+ cameraElement.style.webkitTransformStyle = "preserve-3d";
1121
+ cameraElement.style.transformStyle = "preserve-3d";
1122
+ cameraElement.style.pointerEvents = "none";
1123
+ domElement.appendChild(cameraElement);
1124
+ container.appendChild(domElement);
1125
+ return {
1126
+ container: container,
1127
+ domElement: domElement,
1128
+ cameraElement: cameraElement,
1129
+ };
1130
+ };
1131
+ HtmlMeshRenderer.prototype._getSize = function () {
1132
+ return {
1133
+ width: this._width,
1134
+ height: this._height,
1135
+ };
1136
+ };
1137
+ HtmlMeshRenderer.prototype._setSize = function (width, height) {
1138
+ this._width = width;
1139
+ this._height = height;
1140
+ this._heightHalf = this._height / 2;
1141
+ if (!this._inSceneElements || !this._overlayElements) {
1142
+ return;
1143
+ }
1144
+ var domElements = [this._inSceneElements.domElement, this._overlayElements.domElement, this._inSceneElements.cameraElement, this._overlayElements.cameraElement];
1145
+ domElements.forEach(function (dom) {
1146
+ if (dom) {
1147
+ dom.style.width = "".concat(width, "px");
1148
+ dom.style.height = "".concat(height, "px");
1149
+ }
1150
+ });
1151
+ };
1152
+ // prettier-ignore
1153
+ HtmlMeshRenderer.prototype._getCameraCSSMatrix = function (matrix) {
1154
+ var elements = matrix.m;
1155
+ return "matrix3d(".concat(this._epsilon(elements[0]), ",").concat(this._epsilon(-elements[1]), ",").concat(this._epsilon(elements[2]), ",").concat(this._epsilon(elements[3]), ",").concat(this._epsilon(elements[4]), ",").concat(this._epsilon(-elements[5]), ",").concat(this._epsilon(elements[6]), ",").concat(this._epsilon(elements[7]), ",").concat(this._epsilon(elements[8]), ",").concat(this._epsilon(-elements[9]), ",").concat(this._epsilon(elements[10]), ",").concat(this._epsilon(elements[11]), ",").concat(this._epsilon(elements[12]), ",").concat(this._epsilon(-elements[13]), ",").concat(this._epsilon(elements[14]), ",").concat(this._epsilon(elements[15]), ")");
1156
+ };
1157
+ // Convert a Babylon world matrix to a CSS matrix
1158
+ // This also handles conversion from BJS left handed coords
1159
+ // to CSS right handed coords
1160
+ // prettier-ignore
1161
+ HtmlMeshRenderer.prototype._getHtmlContentCSSMatrix = function (matrix, useRightHandedSystem) {
1162
+ var elements = matrix.m;
1163
+ // In a right handed coordinate system, the elements 11 to 14 have to change their direction
1164
+ var direction = useRightHandedSystem ? -1 : 1;
1165
+ var matrix3d = "matrix3d(".concat(this._epsilon(elements[0]), ",").concat(this._epsilon(elements[1]), ",").concat(this._epsilon(elements[2] * -direction), ",").concat(this._epsilon(elements[3]), ",").concat(this._epsilon(-elements[4]), ",").concat(this._epsilon(-elements[5]), ",").concat(this._epsilon(elements[6] * direction), ",").concat(this._epsilon(-elements[7]), ",").concat(this._epsilon(elements[8] * -direction), ",").concat(this._epsilon(elements[9] * -direction), ",").concat(this._epsilon(elements[10]), ",").concat(this._epsilon(elements[11] * direction), ",").concat(this._epsilon(elements[12] * direction), ",").concat(this._epsilon(elements[13] * direction), ",").concat(this._epsilon(elements[14] * direction), ",").concat(this._epsilon(elements[15]), ")");
1166
+ return matrix3d;
1167
+ };
1168
+ HtmlMeshRenderer.prototype._getTransformationMatrix = function (htmlMesh, useRightHandedSystem) {
1169
+ var _a;
1170
+ // Get the camera world matrix
1171
+ // Make sure the camera world matrix is up to date
1172
+ if (!this._cameraWorldMatrix) {
1173
+ this._cameraWorldMatrix = (_a = htmlMesh.getScene().activeCamera) === null || _a === void 0 ? void 0 : _a.getWorldMatrix();
1174
+ }
1175
+ if (!this._cameraWorldMatrix) {
1176
+ return babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.Identity();
1177
+ }
1178
+ var objectWorldMatrix = htmlMesh.getWorldMatrix();
1179
+ // Scale the object matrix by the base scale factor for the mesh
1180
+ // which is the ratio of the mesh width/height to the renderer
1181
+ // width/height divided by the babylon units to pixels ratio
1182
+ var widthScaleFactor = 1;
1183
+ var heightScaleFactor = 1;
1184
+ if (htmlMesh.sourceWidth && htmlMesh.sourceHeight) {
1185
+ widthScaleFactor = htmlMesh.width / (htmlMesh.sourceWidth / babylonUnitsToPixels);
1186
+ heightScaleFactor = htmlMesh.height / (htmlMesh.sourceHeight / babylonUnitsToPixels);
1187
+ }
1188
+ // Apply the scale to the object's world matrix. Note we aren't scaling
1189
+ // the object, just getting a matrix as though it were scaled, so we can
1190
+ // scale the content
1191
+ var scaleTransform = this._temp.scaleTransform;
1192
+ var rotationTransform = this._temp.rotationTransform;
1193
+ var positionTransform = this._temp.positionTransform;
1194
+ var scaledAndTranslatedObjectMatrix = this._temp.objectMatrix;
1195
+ objectWorldMatrix.decompose(scaleTransform, rotationTransform, positionTransform);
1196
+ scaleTransform.x *= widthScaleFactor;
1197
+ scaleTransform.y *= heightScaleFactor;
1198
+ babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.ComposeToRef(scaleTransform, rotationTransform, positionTransform, scaledAndTranslatedObjectMatrix);
1199
+ // Adjust direction of 12 and 13 of the transformation matrix based on the handedness of the system
1200
+ var direction = useRightHandedSystem ? -1 : 1;
1201
+ // Adjust translation values to be from camera vs world origin
1202
+ // Note that we are also adjusting these values to be pixels vs Babylon units
1203
+ var position = htmlMesh.getAbsolutePosition();
1204
+ scaledAndTranslatedObjectMatrix.setRowFromFloats(3, (-this._cameraWorldMatrix.m[12] + position.x) * babylonUnitsToPixels * direction, (-this._cameraWorldMatrix.m[13] + position.y) * babylonUnitsToPixels * direction, (this._cameraWorldMatrix.m[14] - position.z) * babylonUnitsToPixels, this._cameraWorldMatrix.m[15] * 0.00001 * babylonUnitsToPixels);
1205
+ // Adjust other values to be pixels vs Babylon units
1206
+ scaledAndTranslatedObjectMatrix.multiplyAtIndex(3, babylonUnitsToPixels);
1207
+ scaledAndTranslatedObjectMatrix.multiplyAtIndex(7, babylonUnitsToPixels);
1208
+ scaledAndTranslatedObjectMatrix.multiplyAtIndex(11, babylonUnitsToPixels);
1209
+ return scaledAndTranslatedObjectMatrix;
1210
+ };
1211
+ HtmlMeshRenderer.prototype._renderHtmlMesh = function (htmlMesh, useRightHandedSystem) {
1212
+ var _a, _b;
1213
+ if (!htmlMesh.element || !htmlMesh.element.firstElementChild) {
1214
+ // nothing to render, so bail
1215
+ return;
1216
+ }
1217
+ // We need to ensure html mesh data is initialized before
1218
+ // computing the base scale factor
1219
+ var htmlMeshData = this._cache.htmlMeshData.get(htmlMesh);
1220
+ if (!htmlMeshData) {
1221
+ htmlMeshData = { style: "" };
1222
+ this._cache.htmlMeshData.set(htmlMesh, htmlMeshData);
1223
+ }
1224
+ var cameraElement = htmlMesh._isCanvasOverlay ? (_a = this._overlayElements) === null || _a === void 0 ? void 0 : _a.cameraElement : (_b = this._inSceneElements) === null || _b === void 0 ? void 0 : _b.cameraElement;
1225
+ if (htmlMesh.element.parentNode !== cameraElement) {
1226
+ cameraElement.appendChild(htmlMesh.element);
1227
+ }
1228
+ // If the htmlMesh content has changed, update the base scale factor
1229
+ if (htmlMesh.requiresUpdate) {
1230
+ this._updateBaseScaleFactor(htmlMesh);
1339
1231
  }
1232
+ // Get the transformation matrix for the html mesh
1233
+ var scaledAndTranslatedObjectMatrix = this._getTransformationMatrix(htmlMesh, useRightHandedSystem);
1234
+ var style = "translate(-50%, -50%) ".concat(this._getHtmlContentCSSMatrix(scaledAndTranslatedObjectMatrix, useRightHandedSystem));
1235
+ // In a right handed system, screens are on the wrong side of the mesh, so we have to rotate by Math.PI which results in the matrix3d seen below
1236
+ style += "".concat(useRightHandedSystem ? "matrix3d(-1, 0, 0, 0, 0, 1, 0, 0, 0, 0, -1, 0, 0, 0, 0, 1)" : "");
1237
+ if (htmlMeshData.style !== style) {
1238
+ htmlMesh.element.style.webkitTransform = style;
1239
+ htmlMesh.element.style.transform = style;
1240
+ }
1241
+ htmlMesh._markAsUpdated();
1340
1242
  };
1341
- /**
1342
- * Called when the behavior is detached from its target
1343
- */
1344
- PointerEventsCaptureBehavior.prototype.detach = function () {
1345
- if (!this.attachedMesh) {
1243
+ HtmlMeshRenderer.prototype._render = function (scene, camera) {
1244
+ var _this = this;
1245
+ var _a, _b, _c, _d, _e, _f;
1246
+ var needsUpdate = false;
1247
+ var useRightHandedSystem = scene.useRightHandedSystem;
1248
+ // Update the container position and size if necessary
1249
+ this._updateContainerPositionIfNeeded();
1250
+ // Check for a camera change
1251
+ if (this._cameraMatrixUpdated) {
1252
+ this._cameraMatrixUpdated = false;
1253
+ needsUpdate = true;
1254
+ }
1255
+ // If the camera position has changed, then we also need to update
1256
+ if (camera.position.x !== this._cache.cameraData.position.x ||
1257
+ camera.position.y !== this._cache.cameraData.position.y ||
1258
+ camera.position.z !== this._cache.cameraData.position.z) {
1259
+ this._cache.cameraData.position.copyFrom(camera.position);
1260
+ needsUpdate = true;
1261
+ }
1262
+ // Check for a dpr change
1263
+ if (window.devicePixelRatio !== this._lastDevicePixelRatio) {
1264
+ this._lastDevicePixelRatio = window.devicePixelRatio;
1265
+ babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Logger.Log("In render - dpr changed: ", this._lastDevicePixelRatio);
1266
+ needsUpdate = true;
1267
+ }
1268
+ // Check if any meshes need to be updated
1269
+ var meshesNeedingUpdate = scene.meshes.filter(function (mesh) { return mesh["isHtmlMesh"] && (needsUpdate || mesh.requiresUpdate); });
1270
+ needsUpdate = needsUpdate || meshesNeedingUpdate.length > 0;
1271
+ if (!needsUpdate) {
1346
1272
  return;
1347
1273
  }
1348
- // Remove the reference to this behavior from the mesh
1349
- meshToBehaviorMap.delete(this.attachedMesh);
1350
- if (this._captureOnPointerEnter) {
1351
- stopCaptureOnEnter();
1274
+ // Get a projection matrix for the camera
1275
+ var projectionMatrix = camera.getProjectionMatrix();
1276
+ var fov = projectionMatrix.m[5] * this._heightHalf;
1277
+ if (this._cache.cameraData.fov !== fov) {
1278
+ if (camera.mode == babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Camera.PERSPECTIVE_CAMERA) {
1279
+ [(_a = this._overlayElements) === null || _a === void 0 ? void 0 : _a.domElement, (_b = this._inSceneElements) === null || _b === void 0 ? void 0 : _b.domElement].forEach(function (el) {
1280
+ if (el) {
1281
+ el.style.webkitPerspective = fov + "px";
1282
+ el.style.perspective = fov + "px";
1283
+ }
1284
+ });
1285
+ }
1286
+ else {
1287
+ [(_c = this._overlayElements) === null || _c === void 0 ? void 0 : _c.domElement, (_d = this._inSceneElements) === null || _d === void 0 ? void 0 : _d.domElement].forEach(function (el) {
1288
+ if (el) {
1289
+ el.style.webkitPerspective = "";
1290
+ el.style.perspective = "";
1291
+ }
1292
+ });
1293
+ }
1294
+ this._cache.cameraData.fov = fov;
1352
1295
  }
1353
- this.attachedMesh = null;
1296
+ // Get the CSS matrix for the camera (which will include any camera rotation)
1297
+ if (camera.parent === null) {
1298
+ camera.computeWorldMatrix();
1299
+ }
1300
+ var cameraMatrixWorld = this._temp.cameraWorldMatrix;
1301
+ cameraMatrixWorld.copyFrom(camera.getWorldMatrix());
1302
+ var cameraRotationMatrix = this._temp.cameraRotationMatrix;
1303
+ cameraMatrixWorld.getRotationMatrix().transposeToRef(cameraRotationMatrix);
1304
+ var cameraMatrixWorldAsArray = this._temp.cameraWorldMatrixAsArray;
1305
+ cameraMatrixWorld.copyToArray(cameraMatrixWorldAsArray);
1306
+ // For a few values, we have to adjust the direction based on the handedness of the system
1307
+ var direction = useRightHandedSystem ? 1 : -1;
1308
+ cameraMatrixWorldAsArray[1] = cameraRotationMatrix.m[1];
1309
+ cameraMatrixWorldAsArray[2] = cameraRotationMatrix.m[2] * direction;
1310
+ cameraMatrixWorldAsArray[4] = cameraRotationMatrix.m[4] * direction;
1311
+ cameraMatrixWorldAsArray[6] = cameraRotationMatrix.m[6] * direction;
1312
+ cameraMatrixWorldAsArray[8] = cameraRotationMatrix.m[8] * direction;
1313
+ cameraMatrixWorldAsArray[9] = cameraRotationMatrix.m[9] * direction;
1314
+ babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Matrix.FromArrayToRef(cameraMatrixWorldAsArray, 0, cameraMatrixWorld);
1315
+ var cameraCSSMatrix = this._getCameraCSSMatrix(cameraMatrixWorld);
1316
+ var style = cameraCSSMatrix;
1317
+ if (this._cache.cameraData.style !== style) {
1318
+ [(_e = this._inSceneElements) === null || _e === void 0 ? void 0 : _e.cameraElement, (_f = this._overlayElements) === null || _f === void 0 ? void 0 : _f.cameraElement].forEach(function (el) {
1319
+ if (el) {
1320
+ el.style.webkitTransform = style;
1321
+ el.style.transform = style;
1322
+ }
1323
+ });
1324
+ this._cache.cameraData.style = style;
1325
+ }
1326
+ // _Render objects if necessary
1327
+ meshesNeedingUpdate.forEach(function (mesh) {
1328
+ _this._renderHtmlMesh(mesh, useRightHandedSystem);
1329
+ });
1354
1330
  };
1355
- /**
1356
- * Dispose the behavior
1357
- */
1358
- PointerEventsCaptureBehavior.prototype.dispose = function () {
1359
- this.detach();
1331
+ HtmlMeshRenderer.prototype._updateBaseScaleFactor = function (htmlMesh) {
1332
+ // Get screen width and height
1333
+ var screenWidth = this._width;
1334
+ var screenHeight = this._height;
1335
+ // Calculate aspect ratios
1336
+ var htmlMeshAspectRatio = (htmlMesh.width || 1) / (htmlMesh.height || 1);
1337
+ var screenAspectRatio = screenWidth / screenHeight;
1338
+ // Adjust screen dimensions based on aspect ratios
1339
+ if (htmlMeshAspectRatio > screenAspectRatio) {
1340
+ // If the HTML mesh is wider relative to its height than the screen, adjust the screen width
1341
+ screenWidth = screenHeight * htmlMeshAspectRatio;
1342
+ }
1343
+ else {
1344
+ // If the HTML mesh is taller relative to its width than the screen, adjust the screen height
1345
+ screenHeight = screenWidth / htmlMeshAspectRatio;
1346
+ }
1347
+ // Set content to fill screen so we get max resolution when it is shrunk to fit the mesh
1348
+ htmlMesh.setContentSizePx(screenWidth, screenHeight);
1360
1349
  };
1361
- // Release pointer events
1362
- PointerEventsCaptureBehavior.prototype.releasePointerEvents = function () {
1363
- if (!this.attachedMesh) {
1350
+ HtmlMeshRenderer.prototype._updateContainerPositionIfNeeded = function () {
1351
+ var _this = this;
1352
+ var _a, _b;
1353
+ // Determine if the canvas has moved on the screen
1354
+ var canvasRect = this._engine.getRenderingCanvasClientRect();
1355
+ // canvas rect may be null if layout not complete
1356
+ if (!canvasRect) {
1357
+ babylonjs_Maths_math__WEBPACK_IMPORTED_MODULE_0__.Logger.Warn(_positionUpdateFailMessage);
1364
1358
  return;
1365
1359
  }
1366
- (0,_pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__.requestRelease)(this.attachedMesh.uniqueId.toString());
1360
+ var scrollTop = window.scrollY;
1361
+ var scrollLeft = window.scrollX;
1362
+ var canvasDocumentTop = canvasRect.top + scrollTop;
1363
+ var canvasDocumentLeft = canvasRect.left + scrollLeft;
1364
+ if (this._previousCanvasDocumentPosition.top !== canvasDocumentTop || this._previousCanvasDocumentPosition.left !== canvasDocumentLeft) {
1365
+ this._previousCanvasDocumentPosition.top = canvasDocumentTop;
1366
+ this._previousCanvasDocumentPosition.left = canvasDocumentLeft;
1367
+ [(_a = this._inSceneElements) === null || _a === void 0 ? void 0 : _a.container, (_b = this._overlayElements) === null || _b === void 0 ? void 0 : _b.container].forEach(function (container) {
1368
+ if (!container) {
1369
+ return;
1370
+ }
1371
+ // set the top and left of the css container to match the canvas
1372
+ var containerParent = container.offsetParent;
1373
+ var parentRect = containerParent.getBoundingClientRect();
1374
+ var parentDocumentTop = parentRect.top + scrollTop;
1375
+ var parentDocumentLeft = parentRect.left + scrollLeft;
1376
+ var ancestorMarginsAndPadding = _this._getAncestorMarginsAndPadding(containerParent);
1377
+ // Add the body margin
1378
+ var bodyStyle = window.getComputedStyle(document.body);
1379
+ var bodyMarginTop = parseInt(bodyStyle.marginTop, 10);
1380
+ var bodyMarginLeft = parseInt(bodyStyle.marginLeft, 10);
1381
+ container.style.top = "".concat(canvasDocumentTop - parentDocumentTop - ancestorMarginsAndPadding.marginTop + ancestorMarginsAndPadding.paddingTop + bodyMarginTop, "px");
1382
+ container.style.left = "".concat(canvasDocumentLeft - parentDocumentLeft - ancestorMarginsAndPadding.marginLeft + ancestorMarginsAndPadding.paddingLeft + bodyMarginLeft, "px");
1383
+ });
1384
+ }
1367
1385
  };
1368
- // Capture pointer events
1369
- PointerEventsCaptureBehavior.prototype.capturePointerEvents = function () {
1370
- if (!this.attachedMesh) {
1371
- return;
1386
+ HtmlMeshRenderer.prototype._epsilon = function (value) {
1387
+ return Math.abs(value) < 1e-10 ? 0 : value;
1388
+ };
1389
+ // Get total margins and padding for an element, excluding the body and document margins
1390
+ HtmlMeshRenderer.prototype._getAncestorMarginsAndPadding = function (element) {
1391
+ var marginTop = 0;
1392
+ var marginLeft = 0;
1393
+ var paddingTop = 0;
1394
+ var paddingLeft = 0;
1395
+ while (element && element !== document.body && element !== document.documentElement) {
1396
+ var style = window.getComputedStyle(element);
1397
+ marginTop += parseInt(style.marginTop, 10);
1398
+ marginLeft += parseInt(style.marginLeft, 10);
1399
+ paddingTop += parseInt(style.paddingTop, 10);
1400
+ paddingLeft += parseInt(style.paddingLeft, 10);
1401
+ element = element.offsetParent;
1372
1402
  }
1373
- (0,_pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__.requestCapture)(this.attachedMesh.uniqueId.toString(), this._captureCallback, this._releaseCallback);
1403
+ return { marginTop: marginTop, marginLeft: marginLeft, paddingTop: paddingTop, paddingLeft: paddingLeft };
1374
1404
  };
1375
- return PointerEventsCaptureBehavior;
1405
+ return HtmlMeshRenderer;
1376
1406
  }());
1377
1407
 
1378
1408
 
1379
1409
 
1380
1410
  /***/ }),
1381
1411
 
1382
- /***/ "../../../dev/addons/src/index.ts":
1383
- /*!****************************************!*\
1384
- !*** ../../../dev/addons/src/index.ts ***!
1385
- \****************************************/
1412
+ /***/ "../../../dev/addons/src/htmlMesh/index.ts":
1413
+ /*!*************************************************!*\
1414
+ !*** ../../../dev/addons/src/htmlMesh/index.ts ***!
1415
+ \*************************************************/
1386
1416
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1387
1417
 
1388
1418
  __webpack_require__.r(__webpack_exports__);
1389
1419
  /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1390
- /* harmony export */ FitStrategy: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_0__.FitStrategy),
1391
- /* harmony export */ HtmlMesh: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_0__.HtmlMesh),
1392
- /* harmony export */ HtmlMeshRenderer: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_0__.HtmlMeshRenderer),
1393
- /* harmony export */ PointerEventsCaptureBehavior: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_0__.PointerEventsCaptureBehavior)
1394
- /* harmony export */ });
1395
- /* harmony import */ var _htmlMesh__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./htmlMesh */ "../../../dev/addons/src/htmlMesh/index.ts");
1396
-
1397
-
1398
-
1399
- /***/ }),
1400
-
1401
- /***/ "babylonjs/Meshes/mesh":
1402
- /*!****************************************************************************************************!*\
1403
- !*** external {"root":"BABYLON","commonjs":"babylonjs","commonjs2":"babylonjs","amd":"babylonjs"} ***!
1404
- \****************************************************************************************************/
1405
- /***/ ((module) => {
1406
-
1407
- module.exports = __WEBPACK_EXTERNAL_MODULE_babylonjs_Meshes_mesh__;
1408
-
1409
- /***/ }),
1410
-
1411
- /***/ "../../../../node_modules/tslib/tslib.es6.mjs":
1412
- /*!****************************************************!*\
1413
- !*** ../../../../node_modules/tslib/tslib.es6.mjs ***!
1414
- \****************************************************/
1415
- /***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => {
1416
-
1417
- __webpack_require__.r(__webpack_exports__);
1418
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1419
- /* harmony export */ __addDisposableResource: () => (/* binding */ __addDisposableResource),
1420
- /* harmony export */ __assign: () => (/* binding */ __assign),
1421
- /* harmony export */ __asyncDelegator: () => (/* binding */ __asyncDelegator),
1422
- /* harmony export */ __asyncGenerator: () => (/* binding */ __asyncGenerator),
1423
- /* harmony export */ __asyncValues: () => (/* binding */ __asyncValues),
1424
- /* harmony export */ __await: () => (/* binding */ __await),
1425
- /* harmony export */ __awaiter: () => (/* binding */ __awaiter),
1426
- /* harmony export */ __classPrivateFieldGet: () => (/* binding */ __classPrivateFieldGet),
1427
- /* harmony export */ __classPrivateFieldIn: () => (/* binding */ __classPrivateFieldIn),
1428
- /* harmony export */ __classPrivateFieldSet: () => (/* binding */ __classPrivateFieldSet),
1429
- /* harmony export */ __createBinding: () => (/* binding */ __createBinding),
1430
- /* harmony export */ __decorate: () => (/* binding */ __decorate),
1431
- /* harmony export */ __disposeResources: () => (/* binding */ __disposeResources),
1432
- /* harmony export */ __esDecorate: () => (/* binding */ __esDecorate),
1433
- /* harmony export */ __exportStar: () => (/* binding */ __exportStar),
1434
- /* harmony export */ __extends: () => (/* binding */ __extends),
1435
- /* harmony export */ __generator: () => (/* binding */ __generator),
1436
- /* harmony export */ __importDefault: () => (/* binding */ __importDefault),
1437
- /* harmony export */ __importStar: () => (/* binding */ __importStar),
1438
- /* harmony export */ __makeTemplateObject: () => (/* binding */ __makeTemplateObject),
1439
- /* harmony export */ __metadata: () => (/* binding */ __metadata),
1440
- /* harmony export */ __param: () => (/* binding */ __param),
1441
- /* harmony export */ __propKey: () => (/* binding */ __propKey),
1442
- /* harmony export */ __read: () => (/* binding */ __read),
1443
- /* harmony export */ __rest: () => (/* binding */ __rest),
1444
- /* harmony export */ __rewriteRelativeImportExtension: () => (/* binding */ __rewriteRelativeImportExtension),
1445
- /* harmony export */ __runInitializers: () => (/* binding */ __runInitializers),
1446
- /* harmony export */ __setFunctionName: () => (/* binding */ __setFunctionName),
1447
- /* harmony export */ __spread: () => (/* binding */ __spread),
1448
- /* harmony export */ __spreadArray: () => (/* binding */ __spreadArray),
1449
- /* harmony export */ __spreadArrays: () => (/* binding */ __spreadArrays),
1450
- /* harmony export */ __values: () => (/* binding */ __values),
1451
- /* harmony export */ "default": () => (__WEBPACK_DEFAULT_EXPORT__)
1420
+ /* harmony export */ FitStrategy: () => (/* reexport safe */ _fitStrategy__WEBPACK_IMPORTED_MODULE_3__.FitStrategy),
1421
+ /* harmony export */ HtmlMesh: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_1__.HtmlMesh),
1422
+ /* harmony export */ HtmlMeshRenderer: () => (/* reexport safe */ _htmlMeshRenderer__WEBPACK_IMPORTED_MODULE_0__.HtmlMeshRenderer),
1423
+ /* harmony export */ PointerEventsCaptureBehavior: () => (/* reexport safe */ _pointerEventsCaptureBehavior__WEBPACK_IMPORTED_MODULE_2__.PointerEventsCaptureBehavior)
1452
1424
  /* harmony export */ });
1453
- /******************************************************************************
1454
- Copyright (c) Microsoft Corporation.
1425
+ /* harmony import */ var _htmlMeshRenderer__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./htmlMeshRenderer */ "../../../dev/addons/src/htmlMesh/htmlMeshRenderer.ts");
1426
+ /* harmony import */ var _htmlMesh__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./htmlMesh */ "../../../dev/addons/src/htmlMesh/htmlMesh.ts");
1427
+ /* harmony import */ var _pointerEventsCaptureBehavior__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./pointerEventsCaptureBehavior */ "../../../dev/addons/src/htmlMesh/pointerEventsCaptureBehavior.ts");
1428
+ /* harmony import */ var _fitStrategy__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./fitStrategy */ "../../../dev/addons/src/htmlMesh/fitStrategy.ts");
1455
1429
 
1456
- Permission to use, copy, modify, and/or distribute this software for any
1457
- purpose with or without fee is hereby granted.
1458
1430
 
1459
- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
1460
- REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
1461
- AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
1462
- INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
1463
- LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
1464
- OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
1465
- PERFORMANCE OF THIS SOFTWARE.
1466
- ***************************************************************************** */
1467
- /* global Reflect, Promise, SuppressedError, Symbol, Iterator */
1468
1431
 
1469
- var extendStatics = function(d, b) {
1470
- extendStatics = Object.setPrototypeOf ||
1471
- ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
1472
- function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };
1473
- return extendStatics(d, b);
1474
- };
1475
1432
 
1476
- function __extends(d, b) {
1477
- if (typeof b !== "function" && b !== null)
1478
- throw new TypeError("Class extends value " + String(b) + " is not a constructor or null");
1479
- extendStatics(d, b);
1480
- function __() { this.constructor = d; }
1481
- d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
1482
- }
1433
+ // Export public classes and functions
1483
1434
 
1484
- var __assign = function() {
1485
- __assign = Object.assign || function __assign(t) {
1486
- for (var s, i = 1, n = arguments.length; i < n; i++) {
1487
- s = arguments[i];
1488
- for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];
1489
- }
1490
- return t;
1491
- }
1492
- return __assign.apply(this, arguments);
1493
- }
1494
1435
 
1495
- function __rest(s, e) {
1496
- var t = {};
1497
- for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
1498
- t[p] = s[p];
1499
- if (s != null && typeof Object.getOwnPropertySymbols === "function")
1500
- for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
1501
- if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
1502
- t[p[i]] = s[p[i]];
1503
- }
1504
- return t;
1505
- }
1506
1436
 
1507
- function __decorate(decorators, target, key, desc) {
1508
- var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
1509
- if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
1510
- else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
1511
- return c > 3 && r && Object.defineProperty(target, key, r), r;
1512
- }
1437
+ /***/ }),
1513
1438
 
1514
- function __param(paramIndex, decorator) {
1515
- return function (target, key) { decorator(target, key, paramIndex); }
1516
- }
1439
+ /***/ "../../../dev/addons/src/htmlMesh/pointerEventsCapture.ts":
1440
+ /*!****************************************************************!*\
1441
+ !*** ../../../dev/addons/src/htmlMesh/pointerEventsCapture.ts ***!
1442
+ \****************************************************************/
1443
+ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1517
1444
 
1518
- function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {
1519
- function accept(f) { if (f !== void 0 && typeof f !== "function") throw new TypeError("Function expected"); return f; }
1520
- var kind = contextIn.kind, key = kind === "getter" ? "get" : kind === "setter" ? "set" : "value";
1521
- var target = !descriptorIn && ctor ? contextIn["static"] ? ctor : ctor.prototype : null;
1522
- var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});
1523
- var _, done = false;
1524
- for (var i = decorators.length - 1; i >= 0; i--) {
1525
- var context = {};
1526
- for (var p in contextIn) context[p] = p === "access" ? {} : contextIn[p];
1527
- for (var p in contextIn.access) context.access[p] = contextIn.access[p];
1528
- context.addInitializer = function (f) { if (done) throw new TypeError("Cannot add initializers after decoration has completed"); extraInitializers.push(accept(f || null)); };
1529
- var result = (0, decorators[i])(kind === "accessor" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);
1530
- if (kind === "accessor") {
1531
- if (result === void 0) continue;
1532
- if (result === null || typeof result !== "object") throw new TypeError("Object expected");
1533
- if (_ = accept(result.get)) descriptor.get = _;
1534
- if (_ = accept(result.set)) descriptor.set = _;
1535
- if (_ = accept(result.init)) initializers.unshift(_);
1536
- }
1537
- else if (_ = accept(result)) {
1538
- if (kind === "field") initializers.unshift(_);
1539
- else descriptor[key] = _;
1540
- }
1541
- }
1542
- if (target) Object.defineProperty(target, contextIn.name, descriptor);
1543
- done = true;
1544
- };
1445
+ __webpack_require__.r(__webpack_exports__);
1446
+ /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1447
+ /* harmony export */ getCapturingId: () => (/* binding */ getCapturingId),
1448
+ /* harmony export */ releaseCurrent: () => (/* binding */ releaseCurrent),
1449
+ /* harmony export */ requestCapture: () => (/* binding */ requestCapture),
1450
+ /* harmony export */ requestRelease: () => (/* binding */ requestRelease)
1451
+ /* harmony export */ });
1452
+ /* harmony import */ var babylonjs_Misc_tools__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! babylonjs/Misc/tools */ "babylonjs/Maths/math");
1453
+ /* harmony import */ var babylonjs_Misc_tools__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(babylonjs_Misc_tools__WEBPACK_IMPORTED_MODULE_0__);
1545
1454
 
1546
- function __runInitializers(thisArg, initializers, value) {
1547
- var useValue = arguments.length > 2;
1548
- for (var i = 0; i < initializers.length; i++) {
1549
- value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);
1550
- }
1551
- return useValue ? value : void 0;
1455
+ var captureRequestQueue = [];
1456
+ // Key is request id, value is object with capture and release callbacks
1457
+ var pendingRequestCallbacks = new Map();
1458
+ // Keep track of release requests with no matching capture request
1459
+ // in case the release request arrived before the capture to avoid
1460
+ // the capture request never getting released.
1461
+ var unmatchedReleaseRequests = [];
1462
+ var currentOwner = null; // Called on first capture or release request
1463
+ /**
1464
+ * Get the id of the object currently capturing pointer events
1465
+ * @returns The id of the object currently capturing pointer events
1466
+ * or null if no object is capturing pointer events
1467
+ */
1468
+ var getCapturingId = function () {
1469
+ return currentOwner;
1552
1470
  };
1553
-
1554
- function __propKey(x) {
1555
- return typeof x === "symbol" ? x : "".concat(x);
1471
+ /**
1472
+ * Request that the object with the given id capture pointer events. If there is no current
1473
+ * owner, then the request is granted immediately. If there is a current owner, then the request
1474
+ * is queued until the current owner releases pointer events.
1475
+ * @param requestId An id to identify the request. This id will be used to match the capture
1476
+ * request with the release request.
1477
+ * @param captureCallback The callback to call when the request is granted and the object is capturing
1478
+ * @param releaseCallback The callback to call when the object is no longer capturing pointer events
1479
+ */
1480
+ var requestCapture = function (requestId, captureCallback, releaseCallback) {
1481
+ debugLog("In pointerEventsCapture.requestCapture - Pointer events capture requested for ".concat(requestId));
1482
+ // If there is a release for this request, then ignore the request
1483
+ if (removeUnmatchedRequest(requestId)) {
1484
+ debugLog("In pointerEventsCapture.requestCapture - Capture request matched previous release request ".concat(requestId, ". Cancelling capture request"));
1485
+ return;
1486
+ }
1487
+ else if (requestId !== currentOwner) {
1488
+ // if the request is not already in the queue, add it to the queue
1489
+ enqueueCaptureRequest(requestId, captureCallback, releaseCallback);
1490
+ }
1491
+ if (!currentOwner) {
1492
+ // If there is no current owner, go ahead and grant the request
1493
+ transferPointerEventsOwnership();
1494
+ }
1495
+ // If the request id is the current owner, do nothing
1496
+ };
1497
+ /**
1498
+ * Release pointer events from the object with the given id. If the object is the current owner
1499
+ * then pointer events are released immediately. If the object is not the current owner, then the
1500
+ * associated capture request is removed from the queue. If there is no matching capture request
1501
+ * in the queue, then the release request is added to a list of unmatched release requests and will
1502
+ * negate the next capture request with the same id. This is to guard against the possibility that
1503
+ * the release request arrived before the capture request.
1504
+ * @param requestId The id which should match the id of the capture request
1505
+ */
1506
+ var requestRelease = function (requestId) {
1507
+ debugLog("In pointerEventsCapture.requestRelease - Pointer events release requested for ".concat(requestId));
1508
+ // if the requestId is the current capture holder release it
1509
+ if (!requestId || requestId === currentOwner) {
1510
+ transferPointerEventsOwnership();
1511
+ }
1512
+ else if (cancelRequest(requestId)) {
1513
+ // if the request is in the queue, but not the current capture holder, remove it and it's callbacks
1514
+ pendingRequestCallbacks.delete(requestId);
1515
+ }
1516
+ else {
1517
+ debugLog("In pointerEventsCapture.requestRelease - Received release request ".concat(requestId, " but no matching capture request was received"));
1518
+ // request was not current and not in queue, likely because we received a release
1519
+ // request before the capture. Add it to the unmatched list to guard against this possibility
1520
+ if (!unmatchedReleaseRequests.includes(requestId)) {
1521
+ unmatchedReleaseRequests.push(requestId);
1522
+ }
1523
+ }
1524
+ };
1525
+ /**
1526
+ * Relase pointer events from the current owner
1527
+ */
1528
+ var releaseCurrent = function () {
1529
+ requestRelease(currentOwner);
1530
+ };
1531
+ var enqueueCaptureRequest = function (requestId, capture, release) {
1532
+ debugLog("In pointerEventsCapture.enqueueCaptureRequest - Enqueueing capture request for ".concat(requestId));
1533
+ if (!captureRequestQueue.includes(requestId)) {
1534
+ captureRequestQueue.push(requestId);
1535
+ pendingRequestCallbacks.set(requestId, { capture: capture, release: release });
1536
+ }
1537
+ };
1538
+ // Removes the request from the queue if it exists. Returns true
1539
+ // if the request was found and removed, otherwise false
1540
+ var cancelRequest = function (requestId) {
1541
+ var removed = false;
1542
+ captureRequestQueue = captureRequestQueue.filter(function (id) {
1543
+ if (id !== requestId) {
1544
+ return true;
1545
+ }
1546
+ else {
1547
+ removed = true;
1548
+ debugLog("In pointerEventsCapture.cancelRequest - Canceling pointer events capture request ".concat(requestId));
1549
+ return false;
1550
+ }
1551
+ });
1552
+ return removed;
1553
+ };
1554
+ var removeUnmatchedRequest = function (requestId) {
1555
+ var removed = false;
1556
+ unmatchedReleaseRequests = unmatchedReleaseRequests.filter(function (id) {
1557
+ if (id !== requestId) {
1558
+ return true;
1559
+ }
1560
+ else {
1561
+ removed = true;
1562
+ return false;
1563
+ }
1564
+ });
1565
+ return removed;
1566
+ };
1567
+ var transferPointerEventsOwnership = function () {
1568
+ var newOwnerId = nextCaptureRequest();
1569
+ debugLog("In pointerEventsCapture.transferPointerEventsOwnership - Transferrring pointer events from ".concat(currentOwner, " to ").concat(newOwnerId));
1570
+ // Release the current owner
1571
+ doRelease();
1572
+ if (newOwnerId) {
1573
+ doCapture(newOwnerId);
1574
+ }
1575
+ };
1576
+ var doRelease = function () {
1577
+ var _a;
1578
+ debugLog("In pointerEventsCapture.doRelease - Releasing pointer events from ".concat(currentOwner));
1579
+ if (currentOwner) {
1580
+ // call the release callback
1581
+ (_a = pendingRequestCallbacks.get(currentOwner)) === null || _a === void 0 ? void 0 : _a.release();
1582
+ // And remove the callbacks
1583
+ pendingRequestCallbacks.delete(currentOwner);
1584
+ currentOwner = null;
1585
+ }
1586
+ };
1587
+ var doCapture = function (newOwnerId) {
1588
+ var _a;
1589
+ if (newOwnerId) {
1590
+ // call the capture callback
1591
+ (_a = pendingRequestCallbacks.get(newOwnerId)) === null || _a === void 0 ? void 0 : _a.capture();
1592
+ }
1593
+ currentOwner = newOwnerId;
1594
+ debugLog("In pointerEventsCapture.doCapture - Pointer events now captured by ".concat(newOwnerId));
1595
+ };
1596
+ var nextCaptureRequest = function () {
1597
+ return captureRequestQueue.length > 0 ? captureRequestQueue.shift() : null;
1556
1598
  };
1557
-
1558
- function __setFunctionName(f, name, prefix) {
1559
- if (typeof name === "symbol") name = name.description ? "[".concat(name.description, "]") : "";
1560
- return Object.defineProperty(f, "name", { configurable: true, value: prefix ? "".concat(prefix, " ", name) : name });
1599
+ var debugLog = function (message) {
1600
+ // If we are runnning in a test runner (in node, so window is not defined)
1601
+ // or if the debug flag is set, then log the message
1602
+ if (typeof window === "undefined" || window["pointer-events-capture-debug"]) {
1603
+ babylonjs_Misc_tools__WEBPACK_IMPORTED_MODULE_0__.Tools.Log("".concat(performance.now(), " - game.scene.pointerEvents - ").concat(message, "\ncurrentOwner: ").concat(currentOwner, "\nqueue: ").concat(captureRequestQueue, "\nunmatched: ").concat(unmatchedReleaseRequests));
1604
+ }
1561
1605
  };
1606
+ // #endregion Debugging support
1562
1607
 
1563
- function __metadata(metadataKey, metadataValue) {
1564
- if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(metadataKey, metadataValue);
1565
- }
1566
-
1567
- function __awaiter(thisArg, _arguments, P, generator) {
1568
- function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
1569
- return new (P || (P = Promise))(function (resolve, reject) {
1570
- function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
1571
- function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
1572
- function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
1573
- step((generator = generator.apply(thisArg, _arguments || [])).next());
1574
- });
1575
- }
1576
-
1577
- function __generator(thisArg, body) {
1578
- var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
1579
- return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
1580
- function verb(n) { return function (v) { return step([n, v]); }; }
1581
- function step(op) {
1582
- if (f) throw new TypeError("Generator is already executing.");
1583
- while (g && (g = 0, op[0] && (_ = 0)), _) try {
1584
- if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
1585
- if (y = 0, t) op = [op[0] & 2, t.value];
1586
- switch (op[0]) {
1587
- case 0: case 1: t = op; break;
1588
- case 4: _.label++; return { value: op[1], done: false };
1589
- case 5: _.label++; y = op[1]; op = [0]; continue;
1590
- case 7: op = _.ops.pop(); _.trys.pop(); continue;
1591
- default:
1592
- if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
1593
- if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
1594
- if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
1595
- if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
1596
- if (t[2]) _.ops.pop();
1597
- _.trys.pop(); continue;
1598
- }
1599
- op = body.call(thisArg, _);
1600
- } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
1601
- if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
1602
- }
1603
- }
1604
-
1605
- var __createBinding = Object.create ? (function(o, m, k, k2) {
1606
- if (k2 === undefined) k2 = k;
1607
- var desc = Object.getOwnPropertyDescriptor(m, k);
1608
- if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
1609
- desc = { enumerable: true, get: function() { return m[k]; } };
1610
- }
1611
- Object.defineProperty(o, k2, desc);
1612
- }) : (function(o, m, k, k2) {
1613
- if (k2 === undefined) k2 = k;
1614
- o[k2] = m[k];
1615
- });
1616
-
1617
- function __exportStar(m, o) {
1618
- for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);
1619
- }
1620
-
1621
- function __values(o) {
1622
- var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0;
1623
- if (m) return m.call(o);
1624
- if (o && typeof o.length === "number") return {
1625
- next: function () {
1626
- if (o && i >= o.length) o = void 0;
1627
- return { value: o && o[i++], done: !o };
1628
- }
1629
- };
1630
- throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined.");
1631
- }
1632
-
1633
- function __read(o, n) {
1634
- var m = typeof Symbol === "function" && o[Symbol.iterator];
1635
- if (!m) return o;
1636
- var i = m.call(o), r, ar = [], e;
1637
- try {
1638
- while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);
1639
- }
1640
- catch (error) { e = { error: error }; }
1641
- finally {
1642
- try {
1643
- if (r && !r.done && (m = i["return"])) m.call(i);
1644
- }
1645
- finally { if (e) throw e.error; }
1646
- }
1647
- return ar;
1648
- }
1649
-
1650
- /** @deprecated */
1651
- function __spread() {
1652
- for (var ar = [], i = 0; i < arguments.length; i++)
1653
- ar = ar.concat(__read(arguments[i]));
1654
- return ar;
1655
- }
1656
-
1657
- /** @deprecated */
1658
- function __spreadArrays() {
1659
- for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;
1660
- for (var r = Array(s), k = 0, i = 0; i < il; i++)
1661
- for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)
1662
- r[k] = a[j];
1663
- return r;
1664
- }
1665
-
1666
- function __spreadArray(to, from, pack) {
1667
- if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
1668
- if (ar || !(i in from)) {
1669
- if (!ar) ar = Array.prototype.slice.call(from, 0, i);
1670
- ar[i] = from[i];
1671
- }
1672
- }
1673
- return to.concat(ar || Array.prototype.slice.call(from));
1674
- }
1675
1608
 
1676
- function __await(v) {
1677
- return this instanceof __await ? (this.v = v, this) : new __await(v);
1678
- }
1609
+ /***/ }),
1679
1610
 
1680
- function __asyncGenerator(thisArg, _arguments, generator) {
1681
- if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
1682
- var g = generator.apply(thisArg, _arguments || []), i, q = [];
1683
- return i = Object.create((typeof AsyncIterator === "function" ? AsyncIterator : Object).prototype), verb("next"), verb("throw"), verb("return", awaitReturn), i[Symbol.asyncIterator] = function () { return this; }, i;
1684
- function awaitReturn(f) { return function (v) { return Promise.resolve(v).then(f, reject); }; }
1685
- function verb(n, f) { if (g[n]) { i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; if (f) i[n] = f(i[n]); } }
1686
- function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }
1687
- function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }
1688
- function fulfill(value) { resume("next", value); }
1689
- function reject(value) { resume("throw", value); }
1690
- function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }
1691
- }
1611
+ /***/ "../../../dev/addons/src/htmlMesh/pointerEventsCaptureBehavior.ts":
1612
+ /*!************************************************************************!*\
1613
+ !*** ../../../dev/addons/src/htmlMesh/pointerEventsCaptureBehavior.ts ***!
1614
+ \************************************************************************/
1615
+ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1692
1616
 
1693
- function __asyncDelegator(o) {
1694
- var i, p;
1695
- return i = {}, verb("next"), verb("throw", function (e) { throw e; }), verb("return"), i[Symbol.iterator] = function () { return this; }, i;
1696
- function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }
1697
- }
1617
+ __webpack_require__.r(__webpack_exports__);
1618
+ /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1619
+ /* harmony export */ PointerEventsCaptureBehavior: () => (/* binding */ PointerEventsCaptureBehavior)
1620
+ /* harmony export */ });
1621
+ /* harmony import */ var babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! babylonjs/Misc/logger */ "babylonjs/Maths/math");
1622
+ /* harmony import */ var babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__);
1623
+ /* harmony import */ var _pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./pointerEventsCapture */ "../../../dev/addons/src/htmlMesh/pointerEventsCapture.ts");
1698
1624
 
1699
- function __asyncValues(o) {
1700
- if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
1701
- var m = o[Symbol.asyncIterator], i;
1702
- return m ? m.call(o) : (o = typeof __values === "function" ? __values(o) : o[Symbol.iterator](), i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i);
1703
- function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }
1704
- function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }
1705
- }
1706
1625
 
1707
- function __makeTemplateObject(cooked, raw) {
1708
- if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; }
1709
- return cooked;
1626
+ // Module level variable for holding the current scene
1627
+ var _scene = null;
1628
+ // Module level variable to hold the count of behavior instances that are currently capturing pointer events
1629
+ // on entry. This is used to determine if we need to start or stop observing pointer movement.
1630
+ var captureOnEnterCount = 0;
1631
+ // Map used to store instance of the PointerEventsCaptureBehavior for a mesh
1632
+ // We do this because this gets checked on pointer move and we don't want to
1633
+ // use getBehaviorByName() because that is a linear search
1634
+ var meshToBehaviorMap = new WeakMap();
1635
+ var startCaptureOnEnter = function (scene) {
1636
+ // If we are not in a browser, do nothing
1637
+ if (typeof document === "undefined") {
1638
+ return;
1639
+ }
1640
+ if (captureOnEnterCount === 0) {
1641
+ document.addEventListener("pointermove", onPointerMove);
1642
+ document.addEventListener("touchstart", onPointerMove);
1643
+ _scene = _scene !== null && _scene !== void 0 ? _scene : scene;
1644
+ babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log("PointerEventsCaptureBehavior: Starting observation of pointer move events.");
1645
+ _scene.onDisposeObservable.add(doStopCaptureOnEnter);
1646
+ }
1647
+ captureOnEnterCount++;
1710
1648
  };
1711
-
1712
- var __setModuleDefault = Object.create ? (function(o, v) {
1713
- Object.defineProperty(o, "default", { enumerable: true, value: v });
1714
- }) : function(o, v) {
1715
- o["default"] = v;
1649
+ var doStopCaptureOnEnter = function () {
1650
+ document.removeEventListener("pointermove", onPointerMove);
1651
+ document.removeEventListener("touchstart", onPointerMove);
1652
+ _scene = null;
1653
+ babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log("PointerEventsCaptureBehavior: Stopping observation of pointer move events.");
1654
+ captureOnEnterCount = 0;
1716
1655
  };
1717
-
1718
- var ownKeys = function(o) {
1719
- ownKeys = Object.getOwnPropertyNames || function (o) {
1720
- var ar = [];
1721
- for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
1722
- return ar;
1723
- };
1724
- return ownKeys(o);
1656
+ var stopCaptureOnEnter = function () {
1657
+ // If we are not in a browser, do nothing
1658
+ if (typeof document === "undefined") {
1659
+ return;
1660
+ }
1661
+ // If we are not observing pointer movement, do nothing
1662
+ if (!_scene) {
1663
+ return;
1664
+ }
1665
+ captureOnEnterCount--;
1666
+ if (captureOnEnterCount <= 0) {
1667
+ doStopCaptureOnEnter();
1668
+ }
1669
+ };
1670
+ // Module level function used to determine if an entered mesh should capture pointer events
1671
+ var onPointerMove = function (evt) {
1672
+ if (!_scene) {
1673
+ return;
1674
+ }
1675
+ var canvasRect = _scene.getEngine().getRenderingCanvasClientRect();
1676
+ if (!canvasRect) {
1677
+ return;
1678
+ }
1679
+ // Get the object that contains the client X and Y from either the pointer event or from the
1680
+ // TouchEvent touch
1681
+ var _a = "touches" in evt ? evt.touches[0] : evt, clientX = _a.clientX, clientY = _a.clientY;
1682
+ // get the picked mesh, if any
1683
+ var pointerScreenX = clientX - canvasRect.left;
1684
+ var pointerScreenY = clientY - canvasRect.top;
1685
+ var pointerCaptureBehavior;
1686
+ var pickResult = _scene.pick(pointerScreenX, pointerScreenY, function (mesh) {
1687
+ // If the mesh has an instance of PointerEventsCaptureBehavior attached to it,
1688
+ // and capture on pointer enter is true, then we want to pick it
1689
+ var pointerCaptureBehavior = meshToBehaviorMap.get(mesh);
1690
+ return mesh.isEnabled() && typeof pointerCaptureBehavior !== "undefined" && pointerCaptureBehavior._captureOnPointerEnter;
1691
+ });
1692
+ var pickedMesh;
1693
+ if (pickResult.hit) {
1694
+ pickedMesh = pickResult.pickedMesh;
1695
+ }
1696
+ else {
1697
+ pickedMesh = null;
1698
+ }
1699
+ var capturingIdAsInt = parseInt((0,_pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__.getCapturingId)() || "");
1700
+ // if the picked mesh is the current capturing mesh, do nothing
1701
+ if (pickedMesh && pickedMesh.uniqueId === capturingIdAsInt) {
1702
+ return;
1703
+ }
1704
+ // If there is a capturing mesh and it is not the current picked mesh, or no
1705
+ // mesh is picked, release the capturing mesh
1706
+ if (capturingIdAsInt && (!pickedMesh || pickedMesh.uniqueId !== capturingIdAsInt)) {
1707
+ (0,_pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__.releaseCurrent)();
1708
+ }
1709
+ // If there is a picked mesh and it is not the current capturing mesh, capture
1710
+ // the pointer events. Note that the current capturing mesh has already been
1711
+ // released above
1712
+ if (pickedMesh) {
1713
+ pointerCaptureBehavior = meshToBehaviorMap.get(pickedMesh);
1714
+ pointerCaptureBehavior.capturePointerEvents();
1715
+ }
1725
1716
  };
1717
+ /**
1718
+ * Behavior for any content that can capture pointer events, i.e. bypass the Babylon pointer event handling
1719
+ * and receive pointer events directly. It will register the capture triggers and negotiate the capture and
1720
+ * release of pointer events. Curerntly this applies only to HtmlMesh
1721
+ */
1722
+ var PointerEventsCaptureBehavior = /** @class */ (function () {
1723
+ function PointerEventsCaptureBehavior(_captureCallback, _releaseCallback, _a) {
1724
+ var _b = _a === void 0 ? {} : _a, _c = _b.captureOnPointerEnter, captureOnPointerEnter = _c === void 0 ? true : _c;
1725
+ this._captureCallback = _captureCallback;
1726
+ this._releaseCallback = _releaseCallback;
1727
+ /** gets or sets behavior's name */
1728
+ this.name = "PointerEventsCaptureBehavior";
1729
+ this._attachedMesh = null;
1730
+ this._captureOnPointerEnter = captureOnPointerEnter;
1731
+ // Warn if we are not in a browser
1732
+ if (typeof document === "undefined") {
1733
+ babylonjs_Misc_logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Warn("Creating an instance of PointerEventsCaptureBehavior outside of a browser. The behavior will not work.");
1734
+ }
1735
+ }
1736
+ Object.defineProperty(PointerEventsCaptureBehavior.prototype, "attachedMesh", {
1737
+ /**
1738
+ * Gets or sets the mesh that the behavior is attached to
1739
+ */
1740
+ get: function () {
1741
+ return this._attachedMesh;
1742
+ },
1743
+ set: function (value) {
1744
+ this._attachedMesh = value;
1745
+ },
1746
+ enumerable: false,
1747
+ configurable: true
1748
+ });
1749
+ Object.defineProperty(PointerEventsCaptureBehavior.prototype, "captureOnPointerEnter", {
1750
+ /**
1751
+ * Set if the behavior should capture pointer events when the pointer enters the mesh
1752
+ */
1753
+ set: function (captureOnPointerEnter) {
1754
+ if (this._captureOnPointerEnter === captureOnPointerEnter) {
1755
+ return;
1756
+ }
1757
+ this._captureOnPointerEnter = captureOnPointerEnter;
1758
+ if (this._attachedMesh) {
1759
+ if (this._captureOnPointerEnter) {
1760
+ startCaptureOnEnter(this._attachedMesh.getScene());
1761
+ }
1762
+ else {
1763
+ stopCaptureOnEnter();
1764
+ }
1765
+ }
1766
+ },
1767
+ enumerable: false,
1768
+ configurable: true
1769
+ });
1770
+ /**
1771
+ * Function called when the behavior needs to be initialized (before attaching it to a target)
1772
+ */
1773
+ PointerEventsCaptureBehavior.prototype.init = function () { };
1774
+ /**
1775
+ * Called when the behavior is attached to a target
1776
+ * @param mesh defines the target where the behavior is attached to
1777
+ */
1778
+ PointerEventsCaptureBehavior.prototype.attach = function (mesh) {
1779
+ // Add a reference to this behavior on the mesh. We do this so we can get a
1780
+ // reference to the behavior in the onPointerMove function without relying on
1781
+ // getBehaviorByName(), which does a linear search of the behaviors array.
1782
+ this.attachedMesh = mesh;
1783
+ meshToBehaviorMap.set(mesh, this);
1784
+ if (this._captureOnPointerEnter) {
1785
+ startCaptureOnEnter(mesh.getScene());
1786
+ }
1787
+ };
1788
+ /**
1789
+ * Called when the behavior is detached from its target
1790
+ */
1791
+ PointerEventsCaptureBehavior.prototype.detach = function () {
1792
+ if (!this.attachedMesh) {
1793
+ return;
1794
+ }
1795
+ // Remove the reference to this behavior from the mesh
1796
+ meshToBehaviorMap.delete(this.attachedMesh);
1797
+ if (this._captureOnPointerEnter) {
1798
+ stopCaptureOnEnter();
1799
+ }
1800
+ this.attachedMesh = null;
1801
+ };
1802
+ /**
1803
+ * Dispose the behavior
1804
+ */
1805
+ PointerEventsCaptureBehavior.prototype.dispose = function () {
1806
+ this.detach();
1807
+ };
1808
+ // Release pointer events
1809
+ PointerEventsCaptureBehavior.prototype.releasePointerEvents = function () {
1810
+ if (!this.attachedMesh) {
1811
+ return;
1812
+ }
1813
+ (0,_pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__.requestRelease)(this.attachedMesh.uniqueId.toString());
1814
+ };
1815
+ // Capture pointer events
1816
+ PointerEventsCaptureBehavior.prototype.capturePointerEvents = function () {
1817
+ if (!this.attachedMesh) {
1818
+ return;
1819
+ }
1820
+ (0,_pointerEventsCapture__WEBPACK_IMPORTED_MODULE_1__.requestCapture)(this.attachedMesh.uniqueId.toString(), this._captureCallback, this._releaseCallback);
1821
+ };
1822
+ return PointerEventsCaptureBehavior;
1823
+ }());
1726
1824
 
1727
- function __importStar(mod) {
1728
- if (mod && mod.__esModule) return mod;
1729
- var result = {};
1730
- if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
1731
- __setModuleDefault(result, mod);
1732
- return result;
1733
- }
1734
-
1735
- function __importDefault(mod) {
1736
- return (mod && mod.__esModule) ? mod : { default: mod };
1737
- }
1738
1825
 
1739
- function __classPrivateFieldGet(receiver, state, kind, f) {
1740
- if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
1741
- if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
1742
- return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
1743
- }
1744
1826
 
1745
- function __classPrivateFieldSet(receiver, state, value, kind, f) {
1746
- if (kind === "m") throw new TypeError("Private method is not writable");
1747
- if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
1748
- if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
1749
- return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
1750
- }
1827
+ /***/ }),
1751
1828
 
1752
- function __classPrivateFieldIn(state, receiver) {
1753
- if (receiver === null || (typeof receiver !== "object" && typeof receiver !== "function")) throw new TypeError("Cannot use 'in' operator on non-object");
1754
- return typeof state === "function" ? receiver === state : state.has(receiver);
1755
- }
1829
+ /***/ "../../../dev/addons/src/index.ts":
1830
+ /*!****************************************!*\
1831
+ !*** ../../../dev/addons/src/index.ts ***!
1832
+ \****************************************/
1833
+ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1756
1834
 
1757
- function __addDisposableResource(env, value, async) {
1758
- if (value !== null && value !== void 0) {
1759
- if (typeof value !== "object" && typeof value !== "function") throw new TypeError("Object expected.");
1760
- var dispose, inner;
1761
- if (async) {
1762
- if (!Symbol.asyncDispose) throw new TypeError("Symbol.asyncDispose is not defined.");
1763
- dispose = value[Symbol.asyncDispose];
1764
- }
1765
- if (dispose === void 0) {
1766
- if (!Symbol.dispose) throw new TypeError("Symbol.dispose is not defined.");
1767
- dispose = value[Symbol.dispose];
1768
- if (async) inner = dispose;
1769
- }
1770
- if (typeof dispose !== "function") throw new TypeError("Object not disposable.");
1771
- if (inner) dispose = function() { try { inner.call(this); } catch (e) { return Promise.reject(e); } };
1772
- env.stack.push({ value: value, dispose: dispose, async: async });
1773
- }
1774
- else if (async) {
1775
- env.stack.push({ async: true });
1776
- }
1777
- return value;
1778
- }
1835
+ __webpack_require__.r(__webpack_exports__);
1836
+ /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1837
+ /* harmony export */ FitStrategy: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_0__.FitStrategy),
1838
+ /* harmony export */ HtmlMesh: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_0__.HtmlMesh),
1839
+ /* harmony export */ HtmlMeshRenderer: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_0__.HtmlMeshRenderer),
1840
+ /* harmony export */ PointerEventsCaptureBehavior: () => (/* reexport safe */ _htmlMesh__WEBPACK_IMPORTED_MODULE_0__.PointerEventsCaptureBehavior)
1841
+ /* harmony export */ });
1842
+ /* harmony import */ var _htmlMesh__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./htmlMesh */ "../../../dev/addons/src/htmlMesh/index.ts");
1779
1843
 
1780
- var _SuppressedError = typeof SuppressedError === "function" ? SuppressedError : function (error, suppressed, message) {
1781
- var e = new Error(message);
1782
- return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e;
1783
- };
1784
1844
 
1785
- function __disposeResources(env) {
1786
- function fail(e) {
1787
- env.error = env.hasError ? new _SuppressedError(e, env.error, "An error was suppressed during disposal.") : e;
1788
- env.hasError = true;
1789
- }
1790
- var r, s = 0;
1791
- function next() {
1792
- while (r = env.stack.pop()) {
1793
- try {
1794
- if (!r.async && s === 1) return s = 0, env.stack.push(r), Promise.resolve().then(next);
1795
- if (r.dispose) {
1796
- var result = r.dispose.call(r.value);
1797
- if (r.async) return s |= 2, Promise.resolve(result).then(next, function(e) { fail(e); return next(); });
1798
- }
1799
- else s |= 1;
1800
- }
1801
- catch (e) {
1802
- fail(e);
1803
- }
1804
- }
1805
- if (s === 1) return env.hasError ? Promise.reject(env.error) : Promise.resolve();
1806
- if (env.hasError) throw env.error;
1807
- }
1808
- return next();
1809
- }
1810
1845
 
1811
- function __rewriteRelativeImportExtension(path, preserveJsx) {
1812
- if (typeof path === "string" && /^\.\.?\//.test(path)) {
1813
- return path.replace(/\.(tsx)$|((?:\.d)?)((?:\.[^./]+?)?)\.([cm]?)ts$/i, function (m, tsx, d, ext, cm) {
1814
- return tsx ? preserveJsx ? ".jsx" : ".js" : d && (!ext || !cm) ? m : (d + ext + "." + cm.toLowerCase() + "js");
1815
- });
1816
- }
1817
- return path;
1818
- }
1846
+ /***/ }),
1819
1847
 
1820
- /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = ({
1821
- __extends,
1822
- __assign,
1823
- __rest,
1824
- __decorate,
1825
- __param,
1826
- __esDecorate,
1827
- __runInitializers,
1828
- __propKey,
1829
- __setFunctionName,
1830
- __metadata,
1831
- __awaiter,
1832
- __generator,
1833
- __createBinding,
1834
- __exportStar,
1835
- __values,
1836
- __read,
1837
- __spread,
1838
- __spreadArrays,
1839
- __spreadArray,
1840
- __await,
1841
- __asyncGenerator,
1842
- __asyncDelegator,
1843
- __asyncValues,
1844
- __makeTemplateObject,
1845
- __importStar,
1846
- __importDefault,
1847
- __classPrivateFieldGet,
1848
- __classPrivateFieldSet,
1849
- __classPrivateFieldIn,
1850
- __addDisposableResource,
1851
- __disposeResources,
1852
- __rewriteRelativeImportExtension,
1853
- });
1848
+ /***/ "babylonjs/Maths/math":
1849
+ /*!****************************************************************************************************!*\
1850
+ !*** external {"root":"BABYLON","commonjs":"babylonjs","commonjs2":"babylonjs","amd":"babylonjs"} ***!
1851
+ \****************************************************************************************************/
1852
+ /***/ ((module) => {
1854
1853
 
1854
+ module.exports = __WEBPACK_EXTERNAL_MODULE_babylonjs_Maths_math__;
1855
1855
 
1856
1856
  /***/ })
1857
1857
 
@@ -1947,4 +1947,4 @@ __webpack_exports__ = __webpack_exports__["default"];
1947
1947
  /******/ })()
1948
1948
  ;
1949
1949
  });
1950
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFieWxvbmpzLmFkZG9ucy5qcyIsIm1hcHBpbmdzIjoiQUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7Ozs7Ozs7Ozs7OztBQ0xBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQzlGQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUVBOzs7Ozs7Ozs7QUFTQTtBQUNBO0FBQUE7QUFxREE7Ozs7O0FBS0E7QUFDQTtBQUFBO0FBQ0E7QUFwREE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7O0FBRUE7QUFDQTtBQUVBO0FBTUE7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQWtCQTtBQVdBO0FBQ0E7QUFDQTs7QUFFQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFDQTtBQS9FQTtBQUhBOztBQUVBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQWdDQTtBQUhBOztBQUVBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQUtBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7OztBQUFBO0FBeUNBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7OztBQUFBO0FBS0E7QUFIQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7O0FBQUE7QUFLQTtBQUhBOztBQUVBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQU1BO0FBSkE7OztBQUdBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQUtBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7OztBQUFBO0FBRUE7O0FBRUE7QUFDQTs7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTs7Ozs7OztBQU9BO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7Ozs7QUFJQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQUE7O0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFBQTs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDNVdBO0FBR0E7QUFFQTtBQUdBO0FBR0E7QUFDQTtBQWNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTs7Ozs7Ozs7O0FBU0E7QUFDQTtBQTRDQTs7Ozs7QUFLQTtBQUNBO0FBRUE7QUFGQTtBQTVDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUlBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFvaUJBO0FBQ0E7QUFDQTtBQUNBO0FBN2dCQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7O0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFBQTs7QUFRQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFFQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTs7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBaUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBaUNBO0FBQ0E7QUFFQTs7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBUUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7O0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFBQTs7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFBQTs7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFHQTtBQUNBO0FBQ0E7QUFPQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDbnFCQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDTkE7QUFlQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7Ozs7QUFJQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOzs7Ozs7OztBQVFBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOzs7Ozs7OztBQVFBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBVUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUdBO0FBQ0E7QUFDQTs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDeExBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOzs7O0FBSUE7QUFDQTtBQW1CQTtBQUdBO0FBRkE7QUFDQTtBQXBCQTtBQUNBO0FBc0JBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBcEJBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7OztBQUpBO0FBdUJBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQUVBOztBQUVBO0FBQ0E7QUFFQTs7O0FBR0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDbE9BOzs7Ozs7Ozs7OztBQ0FBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7Ozs7OztBQ2haQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Ozs7QUN2QkE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7Ozs7QUNQQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7OztBQ1BBOzs7OztBQ0FBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7Ozs7Ozs7Ozs7Ozs7O0FDTkE7QUFDQTtBQUVBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9BRERPTlMvd2VicGFjay91bml2ZXJzYWxNb2R1bGVEZWZpbml0aW9uIiwid2VicGFjazovL0FERE9OUy8uLi8uLi8uLi9kZXYvYWRkb25zL3NyYy9odG1sTWVzaC9maXRTdHJhdGVneS50cyIsIndlYnBhY2s6Ly9BRERPTlMvLi4vLi4vLi4vZGV2L2FkZG9ucy9zcmMvaHRtbE1lc2gvaHRtbE1lc2gudHMiLCJ3ZWJwYWNrOi8vQURET05TLy4uLy4uLy4uL2Rldi9hZGRvbnMvc3JjL2h0bWxNZXNoL2h0bWxNZXNoUmVuZGVyZXIudHMiLCJ3ZWJwYWNrOi8vQURET05TLy4uLy4uLy4uL2Rldi9hZGRvbnMvc3JjL2h0bWxNZXNoL2luZGV4LnRzIiwid2VicGFjazovL0FERE9OUy8uLi8uLi8uLi9kZXYvYWRkb25zL3NyYy9odG1sTWVzaC9wb2ludGVyRXZlbnRzQ2FwdHVyZS50cyIsIndlYnBhY2s6Ly9BRERPTlMvLi4vLi4vLi4vZGV2L2FkZG9ucy9zcmMvaHRtbE1lc2gvcG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvci50cyIsIndlYnBhY2s6Ly9BRERPTlMvLi4vLi4vLi4vZGV2L2FkZG9ucy9zcmMvaW5kZXgudHMiLCJ3ZWJwYWNrOi8vQURET05TL2V4dGVybmFsIHVtZCB7XCJyb290XCI6XCJCQUJZTE9OXCIsXCJjb21tb25qc1wiOlwiYmFieWxvbmpzXCIsXCJjb21tb25qczJcIjpcImJhYnlsb25qc1wiLFwiYW1kXCI6XCJiYWJ5bG9uanNcIn0iLCJ3ZWJwYWNrOi8vQURET05TLy4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy90c2xpYi90c2xpYi5lczYubWpzIiwid2VicGFjazovL0FERE9OUy93ZWJwYWNrL2Jvb3RzdHJhcCIsIndlYnBhY2s6Ly9BRERPTlMvd2VicGFjay9ydW50aW1lL2NvbXBhdCBnZXQgZGVmYXVsdCBleHBvcnQiLCJ3ZWJwYWNrOi8vQURET05TL3dlYnBhY2svcnVudGltZS9kZWZpbmUgcHJvcGVydHkgZ2V0dGVycyIsIndlYnBhY2s6Ly9BRERPTlMvd2VicGFjay9ydW50aW1lL2hhc093blByb3BlcnR5IHNob3J0aGFuZCIsIndlYnBhY2s6Ly9BRERPTlMvd2VicGFjay9ydW50aW1lL21ha2UgbmFtZXNwYWNlIG9iamVjdCIsIndlYnBhY2s6Ly9BRERPTlMvLi9zcmMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uIHdlYnBhY2tVbml2ZXJzYWxNb2R1bGVEZWZpbml0aW9uKHJvb3QsIGZhY3RvcnkpIHtcblx0aWYodHlwZW9mIGV4cG9ydHMgPT09ICdvYmplY3QnICYmIHR5cGVvZiBtb2R1bGUgPT09ICdvYmplY3QnKVxuXHRcdG1vZHVsZS5leHBvcnRzID0gZmFjdG9yeShyZXF1aXJlKFwiYmFieWxvbmpzXCIpKTtcblx0ZWxzZSBpZih0eXBlb2YgZGVmaW5lID09PSAnZnVuY3Rpb24nICYmIGRlZmluZS5hbWQpXG5cdFx0ZGVmaW5lKFwiYmFieWxvbmpzLWFkZG9uc1wiLCBbXCJiYWJ5bG9uanNcIl0sIGZhY3RvcnkpO1xuXHRlbHNlIGlmKHR5cGVvZiBleHBvcnRzID09PSAnb2JqZWN0Jylcblx0XHRleHBvcnRzW1wiYmFieWxvbmpzLWFkZG9uc1wiXSA9IGZhY3RvcnkocmVxdWlyZShcImJhYnlsb25qc1wiKSk7XG5cdGVsc2Vcblx0XHRyb290W1wiQURET05TXCJdID0gZmFjdG9yeShyb290W1wiQkFCWUxPTlwiXSk7XG59KSgodHlwZW9mIHNlbGYgIT09IFwidW5kZWZpbmVkXCIgPyBzZWxmIDogdHlwZW9mIGdsb2JhbCAhPT0gXCJ1bmRlZmluZWRcIiA/IGdsb2JhbCA6IHRoaXMpLCAoX19XRUJQQUNLX0VYVEVSTkFMX01PRFVMRV9iYWJ5bG9uanNfTWVzaGVzX21lc2hfXykgPT4ge1xucmV0dXJuICIsImV4cG9ydCB0eXBlIEZpdFN0cmF0ZWd5VHlwZSA9IHtcclxuICAgIHdyYXBFbGVtZW50KGVsZW1lbnQ6IEhUTUxFbGVtZW50KTogSFRNTEVsZW1lbnQ7XHJcbiAgICB1cGRhdGVTaXplKHNpemluZ0VsZW1lbnQ6IEhUTUxFbGVtZW50LCB3aWR0aDogbnVtYmVyLCBoZWlnaHQ6IG51bWJlcik6IHZvaWQ7XHJcbn07XHJcblxyXG5jb25zdCBGaXRTdHJhdGVneUNvbnRhaW46IEZpdFN0cmF0ZWd5VHlwZSA9IHtcclxuICAgIHdyYXBFbGVtZW50KGVsZW1lbnQ6IEhUTUxFbGVtZW50KTogSFRNTEVsZW1lbnQge1xyXG4gICAgICAgIGNvbnN0IHNpemluZ0VsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZGl2XCIpO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuZGlzcGxheSA9IFwiZmxleFwiO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuanVzdGlmeUNvbnRlbnQgPSBcImNlbnRlclwiO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuYWxpZ25JdGVtcyA9IFwiY2VudGVyXCI7XHJcbiAgICAgICAgY29uc3Qgc2NhbGluZ0VsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZGl2XCIpO1xyXG4gICAgICAgIHNjYWxpbmdFbGVtZW50LnN0eWxlLnZpc2liaWxpdHkgPSBcImhpZGRlblwiO1xyXG4gICAgICAgIHNjYWxpbmdFbGVtZW50LmFwcGVuZENoaWxkKGVsZW1lbnQpO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuYXBwZW5kQ2hpbGQoc2NhbGluZ0VsZW1lbnQpO1xyXG4gICAgICAgIHJldHVybiBzaXppbmdFbGVtZW50O1xyXG4gICAgfSxcclxuICAgIHVwZGF0ZVNpemUoc2l6aW5nRWxlbWVudDogSFRNTEVsZW1lbnQsIHdpZHRoOiBudW1iZXIsIGhlaWdodDogbnVtYmVyKSB7XHJcbiAgICAgICAgY29uc3Qgc2NhbGluZ0VsZW1lbnQgPSBzaXppbmdFbGVtZW50LmZpcnN0RWxlbWVudENoaWxkISBhcyBIVE1MRWxlbWVudDtcclxuICAgICAgICBzaXppbmdFbGVtZW50LnN0eWxlLndpZHRoID0gYCR7d2lkdGh9cHhgO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuaGVpZ2h0ID0gYCR7aGVpZ2h0fXB4YDtcclxuXHJcbiAgICAgICAgY29uc3QgW2NoaWxkV2lkdGgsIGNoaWxkSGVpZ2h0XSA9IFtzY2FsaW5nRWxlbWVudCEub2Zmc2V0V2lkdGgsIHNjYWxpbmdFbGVtZW50IS5vZmZzZXRIZWlnaHRdO1xyXG4gICAgICAgIGNvbnN0IHNjYWxlID0gTWF0aC5taW4od2lkdGggLyBjaGlsZFdpZHRoLCBoZWlnaHQgLyBjaGlsZEhlaWdodCk7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuc3R5bGUudHJhbnNmb3JtID0gYHNjYWxlKCR7c2NhbGV9KWA7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuc3R5bGUudmlzaWJpbGl0eSA9IFwidmlzaWJsZVwiO1xyXG4gICAgfSxcclxufTtcclxuXHJcbmNvbnN0IEZpdFN0cmF0ZWd5Q292ZXI6IEZpdFN0cmF0ZWd5VHlwZSA9IHtcclxuICAgIHdyYXBFbGVtZW50KGVsZW1lbnQ6IEhUTUxFbGVtZW50KTogSFRNTEVsZW1lbnQge1xyXG4gICAgICAgIGNvbnN0IHNpemluZ0VsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZGl2XCIpO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuZGlzcGxheSA9IFwiZmxleFwiO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuanVzdGlmeUNvbnRlbnQgPSBcImNlbnRlclwiO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuYWxpZ25JdGVtcyA9IFwiY2VudGVyXCI7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5vdmVyZmxvdyA9IFwiaGlkZGVuXCI7XHJcbiAgICAgICAgY29uc3Qgc2NhbGluZ0VsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZGl2XCIpO1xyXG4gICAgICAgIHNjYWxpbmdFbGVtZW50LnN0eWxlLnZpc2liaWxpdHkgPSBcImhpZGRlblwiO1xyXG4gICAgICAgIHNjYWxpbmdFbGVtZW50LmFwcGVuZENoaWxkKGVsZW1lbnQpO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuYXBwZW5kQ2hpbGQoc2NhbGluZ0VsZW1lbnQpO1xyXG4gICAgICAgIHJldHVybiBzaXppbmdFbGVtZW50O1xyXG4gICAgfSxcclxuICAgIHVwZGF0ZVNpemUoc2l6aW5nRWxlbWVudDogSFRNTEVsZW1lbnQsIHdpZHRoOiBudW1iZXIsIGhlaWdodDogbnVtYmVyKSB7XHJcbiAgICAgICAgY29uc3Qgc2NhbGluZ0VsZW1lbnQgPSBzaXppbmdFbGVtZW50LmZpcnN0RWxlbWVudENoaWxkISBhcyBIVE1MRWxlbWVudDtcclxuICAgICAgICBzaXppbmdFbGVtZW50LnN0eWxlLndpZHRoID0gYCR7d2lkdGh9cHhgO1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuaGVpZ2h0ID0gYCR7aGVpZ2h0fXB4YDtcclxuXHJcbiAgICAgICAgY29uc3QgW2NoaWxkV2lkdGgsIGNoaWxkSGVpZ2h0XSA9IFtzY2FsaW5nRWxlbWVudCEub2Zmc2V0V2lkdGgsIHNjYWxpbmdFbGVtZW50IS5vZmZzZXRIZWlnaHRdO1xyXG4gICAgICAgIGNvbnN0IHNjYWxlID0gTWF0aC5tYXgod2lkdGggLyBjaGlsZFdpZHRoLCBoZWlnaHQgLyBjaGlsZEhlaWdodCk7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuc3R5bGUudHJhbnNmb3JtID0gYHNjYWxlKCR7c2NhbGV9KWA7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuc3R5bGUudmlzaWJpbGl0eSA9IFwidmlzaWJsZVwiO1xyXG4gICAgfSxcclxufTtcclxuXHJcbmNvbnN0IEZpdFN0cmF0ZWd5U3RyZXRjaDogRml0U3RyYXRlZ3lUeXBlID0ge1xyXG4gICAgd3JhcEVsZW1lbnQoZWxlbWVudDogSFRNTEVsZW1lbnQpOiBIVE1MRWxlbWVudCB7XHJcbiAgICAgICAgY29uc3Qgc2l6aW5nRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5kaXNwbGF5ID0gXCJmbGV4XCI7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5qdXN0aWZ5Q29udGVudCA9IFwiY2VudGVyXCI7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5hbGlnbkl0ZW1zID0gXCJjZW50ZXJcIjtcclxuICAgICAgICBjb25zdCBzY2FsaW5nRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuc3R5bGUudmlzaWJpbGl0eSA9IFwiaGlkZGVuXCI7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuYXBwZW5kQ2hpbGQoZWxlbWVudCk7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5hcHBlbmRDaGlsZChzY2FsaW5nRWxlbWVudCk7XHJcbiAgICAgICAgcmV0dXJuIHNpemluZ0VsZW1lbnQ7XHJcbiAgICB9LFxyXG4gICAgdXBkYXRlU2l6ZShzaXppbmdFbGVtZW50OiBIVE1MRWxlbWVudCwgd2lkdGg6IG51bWJlciwgaGVpZ2h0OiBudW1iZXIpIHtcclxuICAgICAgICBjb25zdCBzY2FsaW5nRWxlbWVudCA9IHNpemluZ0VsZW1lbnQuZmlyc3RFbGVtZW50Q2hpbGQhIGFzIEhUTUxFbGVtZW50O1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUud2lkdGggPSBgJHt3aWR0aH1weGA7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5oZWlnaHQgPSBgJHtoZWlnaHR9cHhgO1xyXG5cclxuICAgICAgICBjb25zdCBbY2hpbGRXaWR0aCwgY2hpbGRIZWlnaHRdID0gW3NjYWxpbmdFbGVtZW50IS5vZmZzZXRXaWR0aCwgc2NhbGluZ0VsZW1lbnQhLm9mZnNldEhlaWdodF07XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuc3R5bGUudHJhbnNmb3JtID0gYHNjYWxlKCR7d2lkdGggLyBjaGlsZFdpZHRofSwgJHtoZWlnaHQgLyBjaGlsZEhlaWdodH0pYDtcclxuICAgICAgICBzY2FsaW5nRWxlbWVudC5zdHlsZS52aXNpYmlsaXR5ID0gXCJ2aXNpYmxlXCI7XHJcbiAgICB9LFxyXG59O1xyXG5cclxuY29uc3QgRml0U3RyYXRlZ3lOb25lOiBGaXRTdHJhdGVneVR5cGUgPSB7XHJcbiAgICB3cmFwRWxlbWVudChlbGVtZW50OiBIVE1MRWxlbWVudCk6IEhUTUxFbGVtZW50IHtcclxuICAgICAgICByZXR1cm4gZWxlbWVudDtcclxuICAgIH0sXHJcbiAgICB1cGRhdGVTaXplKHNpemluZ0VsZW1lbnQ6IEhUTUxFbGVtZW50LCB3aWR0aDogbnVtYmVyLCBoZWlnaHQ6IG51bWJlcikge1xyXG4gICAgICAgIGlmIChzaXppbmdFbGVtZW50KSB7XHJcbiAgICAgICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUud2lkdGggPSBgJHt3aWR0aH1weGA7XHJcbiAgICAgICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUuaGVpZ2h0ID0gYCR7aGVpZ2h0fXB4YDtcclxuICAgICAgICB9XHJcbiAgICB9LFxyXG59O1xyXG5cclxuZXhwb3J0IGNvbnN0IEZpdFN0cmF0ZWd5ID0ge1xyXG4gICAgQ09OVEFJTjogRml0U3RyYXRlZ3lDb250YWluLFxyXG4gICAgQ09WRVI6IEZpdFN0cmF0ZWd5Q292ZXIsXHJcbiAgICBTVFJFVENIOiBGaXRTdHJhdGVneVN0cmV0Y2gsXHJcbiAgICBOT05FOiBGaXRTdHJhdGVneU5vbmUsXHJcbn07XHJcbiIsImltcG9ydCB7IE1lc2ggfSBmcm9tIFwiY29yZS9NZXNoZXMvbWVzaFwiO1xyXG5pbXBvcnQgeyBDcmVhdGVQbGFuZVZlcnRleERhdGEgfSBmcm9tIFwiY29yZS9NZXNoZXMvQnVpbGRlcnMvcGxhbmVCdWlsZGVyXCI7XHJcbmltcG9ydCB7IFN0YW5kYXJkTWF0ZXJpYWwgfSBmcm9tIFwiY29yZS9NYXRlcmlhbHMvc3RhbmRhcmRNYXRlcmlhbFwiO1xyXG5pbXBvcnQgeyBNYXRyaXggfSBmcm9tIFwiY29yZS9NYXRocy9tYXRoXCI7XHJcbmltcG9ydCB7IFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IgfSBmcm9tIFwiLi9wb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yXCI7XHJcbmltcG9ydCB0eXBlIHsgU2NlbmUgfSBmcm9tIFwiY29yZS9zY2VuZVwiO1xyXG5pbXBvcnQgeyBMb2dnZXIgfSBmcm9tIFwiY29yZS9NaXNjL2xvZ2dlclwiO1xyXG5pbXBvcnQgdHlwZSB7IEZpdFN0cmF0ZWd5VHlwZSB9IGZyb20gXCIuL2ZpdFN0cmF0ZWd5XCI7XHJcbmltcG9ydCB7IEZpdFN0cmF0ZWd5IH0gZnJvbSBcIi4vZml0U3RyYXRlZ3lcIjtcclxuXHJcbi8qKlxyXG4gKiBUaGlzIGNsYXNzIHJlcHJlc2VudHMgSFRNTCBjb250ZW50IHRoYXQgd2Ugd2FudCB0byByZW5kZXIgYXMgdGhvdWdoIGl0IGlzIHBhcnQgb2YgdGhlIHNjZW5lLiAgVGhlIEhUTUwgY29udGVudCBpcyBhY3R1YWxseVxyXG4gKiByZW5kZXJlZCBiZWxvdyB0aGUgY2FudmFzLCBidXQgYSBkZXB0aCBtYXNrIGlzIGNyZWF0ZWQgYnkgdGhpcyBjbGFzcyB0aGF0IHdyaXRlcyB0byB0aGUgZGVwdGggYnVmZmVyIGJ1dCBkb2VzIG5vdFxyXG4gKiB3cml0ZSB0byB0aGUgY29sb3IgYnVmZmVyLCBlZmZlY3RpdmVseSBwdW5jaGluZyBhIGhvbGUgaW4gdGhlIGNhbnZhcy4gIENTUyB0cmFuc2Zvcm1zIGFyZSB1c2VkIHRvIHNjYWxlLCB0cmFuc2xhdGUsIGFuZCByb3RhdGVcclxuICogdGhlIEhUTUwgY29udGVudCBzbyB0aGF0IGl0IG1hdGNoZXMgdGhlIGNhbWVyYSBhbmQgbWVzaCBvcmllbnRhdGlvbi4gIFRoZSBjbGFzcyBzdXBwb3J0cyBpbnRlcmFjdGlvbnMgaW4gZWRpdGFibGUgYW5kIG5vbi1lZGl0YWJsZSBtb2RlLlxyXG4gKiBJbiBub24tZWRpdGFibGUgbW9kZSAodGhlIGRlZmF1bHQpLCBldmVudHMgYXJlIHBhc3NlZCB0byB0aGUgSFRNTCBjb250ZW50IHdoZW4gdGhlIHBvaW50ZXIgaXMgb3ZlciB0aGUgbWFzayAoYW5kIG5vdCBvY2NsdWRlZCBieSBvdGhlciBtZXNoZXNcclxuICogaW4gdGhlIHNjZW5lKS5cclxuICogI0hWSFlKQyM1XHJcbiAqICNCMTdUQzcjMTEyXHJcbiAqL1xyXG5leHBvcnQgY2xhc3MgSHRtbE1lc2ggZXh0ZW5kcyBNZXNoIHtcclxuICAgIC8qKlxyXG4gICAgICogSGVscHMgaWRlbnRpZnlpbmcgYSBodG1sIG1lc2ggZnJvbSBhIHJlZ3VsYXIgbWVzaFxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZ2V0IGlzSHRtbE1lc2goKSB7XHJcbiAgICAgICAgcmV0dXJuIHRydWU7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gT3ZlcnJpZGUgdGhlIHN1cGVyIGNsYXNzJ3MgX2lzRW5hYmxlZCBwcm9wZXJ0eSBzbyB3ZSBjYW4gY29udHJvbCB3aGVuIHRoZSBtZXNoXHJcbiAgICAvLyBpcyBlbmFibGVkLiAgSS5lLiwgd2UgZG9uJ3Qgd2FudCB0byByZW5kZXIgdGhlIG1lc2ggdW50aWwgdGhlcmUgaXMgY29udGVudCB0byBzaG93LlxyXG4gICAgcHJpdmF0ZSBfZW5hYmxlZCA9IGZhbHNlO1xyXG5cclxuICAgIC8vIFRoZSBtZXNoIGlzIHJlYWR5IHdoZW4gY29udGVudCBoYXMgYmVlbiBzZXQgYW5kIHRoZSBjb250ZW50IHNpemUgaGFzIGJlZW4gc2V0XHJcbiAgICAvLyBUaGUgZm9ybWVyIGlzIGRvbmUgYnkgdGhlIHVzZXIsIHRoZSBsYXR0ZXIgaXMgZG9uZSBieSB0aGUgcmVuZGVyZXIuXHJcbiAgICBwcml2YXRlIF9yZWFkeSA9IGZhbHNlO1xyXG5cclxuICAgIC8qKlxyXG4gICAgICogQGludGVybmFsXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBfaXNDYW52YXNPdmVybGF5ID0gZmFsc2U7XHJcblxyXG4gICAgcHJpdmF0ZSBfcmVxdWlyZXNVcGRhdGUgPSB0cnVlO1xyXG5cclxuICAgIHByaXZhdGUgX2VsZW1lbnQ/OiBIVE1MRWxlbWVudDtcclxuICAgIHByaXZhdGUgX3dpZHRoPzogbnVtYmVyO1xyXG4gICAgcHJpdmF0ZSBfaGVpZ2h0PzogbnVtYmVyO1xyXG5cclxuICAgIHByaXZhdGUgX2ludmVyc2VTY2FsZU1hdHJpeDogTWF0cml4IHwgbnVsbCA9IG51bGw7XHJcblxyXG4gICAgcHJpdmF0ZSBfY2FwdHVyZU9uUG9pbnRlckVudGVyOiBib29sZWFuID0gdHJ1ZTtcclxuICAgIHByaXZhdGUgX3BvaW50ZXJFdmVudENhcHR1cmVCZWhhdmlvcjogUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvciB8IG51bGwgPSBudWxsO1xyXG5cclxuICAgIHByaXZhdGUgX3NvdXJjZVdpZHRoOiBudW1iZXIgfCBudWxsID0gbnVsbDtcclxuICAgIHByaXZhdGUgX3NvdXJjZUhlaWdodDogbnVtYmVyIHwgbnVsbCA9IG51bGw7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBSZXR1cm4gdGhlIHNvdXJjZSB3aWR0aCBvZiB0aGUgY29udGVudCBpbiBwaXhlbHNcclxuICAgICAqL1xyXG4gICAgcHVibGljIGdldCBzb3VyY2VXaWR0aCgpIHtcclxuICAgICAgICByZXR1cm4gdGhpcy5fc291cmNlV2lkdGg7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBSZXR1cm4gdGhlIHNvdXJjZSBoZWlnaHQgb2YgdGhlIGNvbnRlbnQgaW4gcGl4ZWxzXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBnZXQgc291cmNlSGVpZ2h0KCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl9zb3VyY2VIZWlnaHQ7XHJcbiAgICB9XHJcblxyXG4gICAgcHJpdmF0ZSBfd29ybGRNYXRyaXhVcGRhdGVPYnNlcnZlcjogYW55O1xyXG5cclxuICAgIHByaXZhdGUgX2ZpdFN0cmF0ZWd5OiBGaXRTdHJhdGVneVR5cGUgPSBGaXRTdHJhdGVneS5OT05FO1xyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ29udHJ1Y3QgYW4gaW5zdGFuY2Ugb2YgSHRtbE1lc2hcclxuICAgICAqIEBwYXJhbSBzY2VuZVxyXG4gICAgICogQHBhcmFtIGlkIFRoZSBpZCBvZiB0aGUgbWVzaC4gIFdpbGwgYmUgdXNlZCBhcyB0aGUgaWQgb2YgdGhlIEhUTUwgZWxlbWVudCBhcyB3ZWxsLlxyXG4gICAgICogQHBhcmFtIG9wdGlvbnMgb2JqZWN0IHdpdGggb3B0aW9uYWwgcGFyYW1ldGVyc1xyXG4gICAgICovXHJcbiAgICBjb25zdHJ1Y3RvcihzY2VuZTogU2NlbmUsIGlkOiBzdHJpbmcsIHsgY2FwdHVyZU9uUG9pbnRlckVudGVyID0gdHJ1ZSwgaXNDYW52YXNPdmVybGF5ID0gZmFsc2UsIGZpdFN0cmF0ZWd5ID0gRml0U3RyYXRlZ3kuTk9ORSB9ID0ge30pIHtcclxuICAgICAgICBzdXBlcihpZCwgc2NlbmUpO1xyXG5cclxuICAgICAgICAvLyBSZXF1aXJlcyBhIGJyb3dzZXIgdG8gd29yay4gIEJhaWwgaWYgd2UgYXJlbid0IHJ1bm5pbmcgaW4gYSBicm93c2VyXHJcbiAgICAgICAgaWYgKHR5cGVvZiBkb2N1bWVudCA9PT0gXCJ1bmRlZmluZWRcIikge1xyXG4gICAgICAgICAgICBMb2dnZXIuV2FybihgQ3JlYXRpbmcgYW4gaW5zdGFuY2Ugb2YgYW4gSHRtbE1lc2ggd2l0aCBpZCAke2lkfSBvdXRzaWRlIG9mIGEgYnJvd3Nlci4gIFRoZSBtZXNoIHdpbGwgbm90IGJlIHZpc2libGUuYCk7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMuX2ZpdFN0cmF0ZWd5ID0gZml0U3RyYXRlZ3k7XHJcbiAgICAgICAgdGhpcy5faXNDYW52YXNPdmVybGF5ID0gaXNDYW52YXNPdmVybGF5O1xyXG4gICAgICAgIHRoaXMuX2NyZWF0ZU1hc2soKTtcclxuICAgICAgICB0aGlzLl9lbGVtZW50ID0gdGhpcy5fY3JlYXRlRWxlbWVudCgpO1xyXG5cclxuICAgICAgICAvLyBTZXQgZW5hYmxlZCBieSBkZWZhdWx0LCBzbyB0aGlzIHdpbGwgc2hvdyBhcyBzb29uIGFzIGl0J3MgcmVhZHlcclxuICAgICAgICB0aGlzLnNldEVuYWJsZWQodHJ1ZSk7XHJcblxyXG4gICAgICAgIHRoaXMuX2NhcHR1cmVPblBvaW50ZXJFbnRlciA9IGNhcHR1cmVPblBvaW50ZXJFbnRlcjtcclxuXHJcbiAgICAgICAgLy8gQ3JlYXRlIGEgYmVoYXZpb3IgdG8gY2FwdHVyZSBwb2ludGVyIGV2ZW50c1xyXG4gICAgICAgIHRoaXMuX3BvaW50ZXJFdmVudENhcHR1cmVCZWhhdmlvciA9IG5ldyBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yKHRoaXMuY2FwdHVyZVBvaW50ZXJFdmVudHMuYmluZCh0aGlzKSwgdGhpcy5yZWxlYXNlUG9pbnRlckV2ZW50cy5iaW5kKHRoaXMpLCB7XHJcbiAgICAgICAgICAgIGNhcHR1cmVPblBvaW50ZXJFbnRlcjogdGhpcy5fY2FwdHVyZU9uUG9pbnRlckVudGVyLFxyXG4gICAgICAgIH0pO1xyXG4gICAgICAgIHRoaXMuYWRkQmVoYXZpb3IodGhpcy5fcG9pbnRlckV2ZW50Q2FwdHVyZUJlaGF2aW9yKTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIFRoZSB3aWR0aCBvZiB0aGUgY29udGVudCBpbiBwaXhlbHNcclxuICAgICAqL1xyXG4gICAgcHVibGljIGdldCB3aWR0aCgpIHtcclxuICAgICAgICByZXR1cm4gdGhpcy5fd2lkdGg7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBUaGUgaGVpZ2h0IG9mIHRoZSBjb250ZW50IGluIHBpeGVsc1xyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZ2V0IGhlaWdodCgpIHtcclxuICAgICAgICByZXR1cm4gdGhpcy5faGVpZ2h0O1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogVGhlIEhUTUwgZWxlbWVudCB0aGF0IGlzIGJlaW5nIHJlbmRlcmVkIGFzIGEgbWVzaFxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZ2V0IGVsZW1lbnQoKSB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuX2VsZW1lbnQ7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBUcnVlIGlmIHRoZSBtZXNoIGhhcyBiZWVuIG1vdmVkLCByb3RhdGVkLCBvciBzY2FsZWQgc2luY2UgdGhlIGxhc3QgdGltZSB0aGlzXHJcbiAgICAgKiBwcm9wZXJ0eSB3YXMgcmVhZC4gIFRoaXMgcHJvcGVydHkgaXMgcmVzZXQgdG8gZmFsc2UgYWZ0ZXIgcmVhZGluZy5cclxuICAgICAqL1xyXG4gICAgcHVibGljIGdldCByZXF1aXJlc1VwZGF0ZSgpIHtcclxuICAgICAgICByZXR1cm4gdGhpcy5fcmVxdWlyZXNVcGRhdGU7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBFbmFibGUgY2FwdHVyZSBmb3IgdGhlIHBvaW50ZXIgd2hlbiBlbnRlcmluZyB0aGUgbWVzaCBhcmVhXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBzZXQgY2FwdHVyZU9uUG9pbnRlckVudGVyKGNhcHR1cmVPblBvaW50ZXJFbnRlcjogYm9vbGVhbikge1xyXG4gICAgICAgIHRoaXMuX2NhcHR1cmVPblBvaW50ZXJFbnRlciA9IGNhcHR1cmVPblBvaW50ZXJFbnRlcjtcclxuICAgICAgICBpZiAodGhpcy5fcG9pbnRlckV2ZW50Q2FwdHVyZUJlaGF2aW9yKSB7XHJcbiAgICAgICAgICAgIHRoaXMuX3BvaW50ZXJFdmVudENhcHR1cmVCZWhhdmlvci5jYXB0dXJlT25Qb2ludGVyRW50ZXIgPSBjYXB0dXJlT25Qb2ludGVyRW50ZXI7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogRGlzcG9zZXMgb2YgdGhlIG1lc2ggYW5kIHRoZSBIVE1MIGVsZW1lbnRcclxuICAgICAqL1xyXG4gICAgcHVibGljIG92ZXJyaWRlIGRpc3Bvc2UoKSB7XHJcbiAgICAgICAgc3VwZXIuZGlzcG9zZSgpO1xyXG4gICAgICAgIHRoaXMuX2VsZW1lbnQ/LnJlbW92ZSgpO1xyXG4gICAgICAgIHRoaXMuX2VsZW1lbnQgPSB1bmRlZmluZWQ7XHJcbiAgICAgICAgaWYgKHRoaXMuX3BvaW50ZXJFdmVudENhcHR1cmVCZWhhdmlvcikge1xyXG4gICAgICAgICAgICB0aGlzLl9wb2ludGVyRXZlbnRDYXB0dXJlQmVoYXZpb3IuZGlzcG9zZSgpO1xyXG4gICAgICAgICAgICB0aGlzLl9wb2ludGVyRXZlbnRDYXB0dXJlQmVoYXZpb3IgPSBudWxsO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIEBpbnRlcm5hbFxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgX21hcmtBc1VwZGF0ZWQoKSB7XHJcbiAgICAgICAgdGhpcy5fcmVxdWlyZXNVcGRhdGUgPSBmYWxzZTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIFNldHMgdGhlIGNvbnRlbnQgb2YgdGhlIGVsZW1lbnQgdG8gdGhlIHNwZWNpZmllZCBjb250ZW50IGFkanVzdGluZyB0aGUgbWVzaCBzY2FsZSB0byBtYXRjaCBhbmQgbWFraW5nIGl0IHZpc2libGUuXHJcbiAgICAgKiBJZiB0aGUgdGhlIHNwZWNpZmllZCBjb250ZW50IGlzIHVuZGVmaW5lZCwgdGhlbiBpdCB3aWxsIG1ha2UgdGhlIG1lc2ggaW52aXNpYmxlLiAgSW4gZWl0aGVyIGNhc2UgaXQgd2lsbCBjbGVhciB0aGVcclxuICAgICAqIGVsZW1lbnQgY29udGVudCBmaXJzdC5cclxuICAgICAqIEBwYXJhbSBlbGVtZW50IFRoZSBlbGVtZW50IHRvIHJlbmRlciBhcyBhIG1lc2hcclxuICAgICAqIEBwYXJhbSB3aWR0aCBUaGUgd2lkdGggb2YgdGhlIG1lc2ggaW4gQmFieWxvbiB1bml0c1xyXG4gICAgICogQHBhcmFtIGhlaWdodCBUaGUgaGVpZ2h0IG9mIHRoZSBtZXNoIGluIEJhYnlsb24gdW5pdHNcclxuICAgICAqL1xyXG4gICAgc2V0Q29udGVudChlbGVtZW50OiBIVE1MRWxlbWVudCwgd2lkdGg6IG51bWJlciwgaGVpZ2h0OiBudW1iZXIpIHtcclxuICAgICAgICAvLyBJZiBjb250ZW50IGlzIGNoYW5nZWQsIHdlIGFyZSBubyBsb25nZXIgcmVhZHlcclxuICAgICAgICB0aGlzLl9zZXRBc1JlYWR5KGZhbHNlKTtcclxuXHJcbiAgICAgICAgLy8gQWxzbyBpbnZhbGlkYXRlIHRoZSBzb3VyY2Ugd2lkdGggYW5kIGhlaWdodFxyXG4gICAgICAgIHRoaXMuX3NvdXJjZVdpZHRoID0gbnVsbDtcclxuICAgICAgICB0aGlzLl9zb3VyY2VIZWlnaHQgPSBudWxsO1xyXG5cclxuICAgICAgICBpZiAoIXRoaXMuX2VsZW1lbnQpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgdGhpcy5fd2lkdGggPSB3aWR0aDtcclxuICAgICAgICB0aGlzLl9oZWlnaHQgPSBoZWlnaHQ7XHJcbiAgICAgICAgdGhpcy5fcmVxdWlyZXNVcGRhdGUgPSB0cnVlO1xyXG5cclxuICAgICAgICB0aGlzLnNjYWxpbmcuc2V0QWxsKDEpO1xyXG5cclxuICAgICAgICBpZiAoZWxlbWVudCkge1xyXG4gICAgICAgICAgICB0aGlzLl9lbGVtZW50IS5hcHBlbmRDaGlsZCh0aGlzLl9maXRTdHJhdGVneS53cmFwRWxlbWVudChlbGVtZW50KSk7XHJcblxyXG4gICAgICAgICAgICB0aGlzLl91cGRhdGVTY2FsZUlmTmVjZXNzYXJ5KCk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBpZiAodGhpcy5zb3VyY2VXaWR0aCAmJiB0aGlzLnNvdXJjZUhlaWdodCkge1xyXG4gICAgICAgICAgICB0aGlzLl9zZXRBc1JlYWR5KHRydWUpO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICAvLyBPdmVyaWRlcyBCQUJZTE9OLk1lc2guc2V0RW5hYmxlZFxyXG4gICAgcHVibGljIG92ZXJyaWRlIHNldEVuYWJsZWQoZW5hYmxlZDogYm9vbGVhbikge1xyXG4gICAgICAgIC8vIENhcHR1cmUgcmVxdWVzdGVkIGVuYWJsZWQgc3RhdGVcclxuICAgICAgICB0aGlzLl9lbmFibGVkID0gZW5hYmxlZDtcclxuXHJcbiAgICAgICAgLy8gSWYgZGlzYWJsaW5nIG9yIGVuYWJsaW5nIGFuZCB3ZSBhcmUgcmVhZHlcclxuICAgICAgICBpZiAoIWVuYWJsZWQgfHwgdGhpcy5fcmVhZHkpIHtcclxuICAgICAgICAgICAgdGhpcy5fZG9TZXRFbmFibGVkKGVuYWJsZWQpO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIFNldHMgdGhlIGNvbnRlbnQgc2l6ZSBpbiBwaXhlbHNcclxuICAgICAqIEBwYXJhbSB3aWR0aCB3aWR0aCBvZiB0aGUgc291cmNlXHJcbiAgICAgKiBAcGFyYW0gaGVpZ2h0IGhlaWdodCBvZiB0aGUgc291cmNlXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBzZXRDb250ZW50U2l6ZVB4KHdpZHRoOiBudW1iZXIsIGhlaWdodDogbnVtYmVyKSB7XHJcbiAgICAgICAgdGhpcy5fc291cmNlV2lkdGggPSB3aWR0aDtcclxuICAgICAgICB0aGlzLl9zb3VyY2VIZWlnaHQgPSBoZWlnaHQ7XHJcblxyXG4gICAgICAgIGlmICghdGhpcy5fZWxlbWVudCB8fCAhdGhpcy5fZWxlbWVudC5maXJzdEVsZW1lbnRDaGlsZCkge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICB0aGlzLl9maXRTdHJhdGVneS51cGRhdGVTaXplKHRoaXMuX2VsZW1lbnQuZmlyc3RFbGVtZW50Q2hpbGQhIGFzIEhUTUxFbGVtZW50LCB3aWR0aCwgaGVpZ2h0KTtcclxuXHJcbiAgICAgICAgdGhpcy5fdXBkYXRlU2NhbGVJZk5lY2Vzc2FyeSgpO1xyXG5cclxuICAgICAgICBpZiAodGhpcy53aWR0aCAmJiB0aGlzLmhlaWdodCkge1xyXG4gICAgICAgICAgICB0aGlzLl9zZXRBc1JlYWR5KHRydWUpO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX3NldEFzUmVhZHkocmVhZHk6IGJvb2xlYW4pIHtcclxuICAgICAgICB0aGlzLl9yZWFkeSA9IHJlYWR5O1xyXG4gICAgICAgIGlmIChyZWFkeSkge1xyXG4gICAgICAgICAgICB0aGlzLl9kb1NldEVuYWJsZWQodGhpcy5fZW5hYmxlZCk7XHJcbiAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgdGhpcy5fZG9TZXRFbmFibGVkKGZhbHNlKTtcclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF9kb1NldEVuYWJsZWQoZW5hYmxlZDogYm9vbGVhbikge1xyXG4gICAgICAgIGlmICghdGhpcy5fZWxlbWVudCkge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvL2lmIGVuYWJsZWQsIHRoZW4gc3RhcnQgbGlzdGVuaW5nIGZvciBjaGFuZ2VzIHRvIHRoZVxyXG4gICAgICAgIC8vIHNjYWxpbmcsIHJvdGF0aW9uLCBhbmQgcG9zaXRpb24uICBvdGhlcndpc2Ugc3RvcCBsaXN0ZW5pbmdcclxuICAgICAgICBpZiAoZW5hYmxlZCAmJiAhdGhpcy5fd29ybGRNYXRyaXhVcGRhdGVPYnNlcnZlcikge1xyXG4gICAgICAgICAgICB0aGlzLl93b3JsZE1hdHJpeFVwZGF0ZU9ic2VydmVyID0gdGhpcy5vbkFmdGVyV29ybGRNYXRyaXhVcGRhdGVPYnNlcnZhYmxlLmFkZCgoKSA9PiB7XHJcbiAgICAgICAgICAgICAgICB0aGlzLl9yZXF1aXJlc1VwZGF0ZSA9IHRydWU7XHJcbiAgICAgICAgICAgIH0pO1xyXG4gICAgICAgIH0gZWxzZSBpZiAoIWVuYWJsZWQpIHtcclxuICAgICAgICAgICAgdGhpcy5fd29ybGRNYXRyaXhVcGRhdGVPYnNlcnZlcj8ucmVtb3ZlKCk7XHJcbiAgICAgICAgICAgIHRoaXMuX3dvcmxkTWF0cml4VXBkYXRlT2JzZXJ2ZXIgPSBudWxsO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gSWYgZW5hYmxlZCwgdGhlbiByZXZlcnQgdGhlIGNvbnRlbnQgZWxlbWVudCBkaXNwbGF5XHJcbiAgICAgICAgLy8gb3RoZXJ3aXNlIGhpZGUgaXRcclxuICAgICAgICB0aGlzLl9lbGVtZW50IS5zdHlsZS5kaXNwbGF5ID0gZW5hYmxlZCA/IFwiXCIgOiBcIm5vbmVcIjtcclxuICAgICAgICAvLyBDYXB0dXJlIHRoZSBjb250ZW50IHogaW5kZXhcclxuICAgICAgICB0aGlzLl9zZXRFbGVtZW50WkluZGV4KHRoaXMucG9zaXRpb24ueiAqIC0xMDAwMCk7XHJcbiAgICAgICAgc3VwZXIuc2V0RW5hYmxlZChlbmFibGVkKTtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX3VwZGF0ZVNjYWxlSWZOZWNlc3NhcnkoKSB7XHJcbiAgICAgICAgLy8gSWYgd2UgaGF2ZSBzZXRDb250ZW50IGJlZm9yZSwgdGhlIGNvbnRlbnQgc2NhbGUgaXMgYmFrZWQgaW50byB0aGUgbWVzaC4gIElmIHdlIGRvbid0IHJlc2V0IHRoZSB2ZXJ0aWNlcyB0b1xyXG4gICAgICAgIC8vIHRoZSBvcmlnaW5hbCBzaXplLCB0aGVuIHdlIHdpbGwgbXVsdGlwbHkgdGhlIHNjYWxlIHdoZW4gd2UgYmFrZSB0aGUgc2NhbGUgYmVsb3cuICBCeSBhcHBseWluZyB0aGUgaW52ZXJzZSwgd2UgYmFjayBvdXRcclxuICAgICAgICAvLyB0aGUgc2NhbGluZyB0aGF0IGhhcyBiZWVuIGRvbmUgc28gd2UgYXJlIHN0YXJ0aW5nIGZyb20gdGhlIHNhbWUgcG9pbnQuXHJcbiAgICAgICAgLy8gRmlyc3QgcmVzZXQgdGhlIHNjYWxlIHRvIDFcclxuICAgICAgICB0aGlzLnNjYWxpbmcuc2V0QWxsKDEpO1xyXG4gICAgICAgIC8vIFRoZW4gYmFjayBvdXQgdGhlIG9yaWdpbmFsIHZlcnRpY2VzIGNoYW5nZXMgdG8gbWF0Y2ggdGhlIGNvbnRlbnQgc2NhbGVcclxuICAgICAgICBpZiAodGhpcy5faW52ZXJzZVNjYWxlTWF0cml4KSB7XHJcbiAgICAgICAgICAgIHRoaXMuYmFrZVRyYW5zZm9ybUludG9WZXJ0aWNlcyh0aGlzLl9pbnZlcnNlU2NhbGVNYXRyaXgpO1xyXG4gICAgICAgICAgICAvLyBDbGVhciBvdXQgdGhlIG1hdHJpeCBzbyBpdCBkb2Vzbid0IGdldCBhcHBsaWVkIGFnYWluIHVubGVzcyB3ZSBzY2FsZVxyXG4gICAgICAgICAgICB0aGlzLl9pbnZlcnNlU2NhbGVNYXRyaXggPSBudWxsO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gU2V0IHNjYWxlIHRvIG1hdGNoIGNvbnRlbnQuICBOb3RlIHdlIGNhbid0IGp1c3Qgc2NhbGUgdGhlIG1lc2gsIGJlY2F1c2UgdGhhdCB3aWxsIHNjYWxlIHRoZSBjb250ZW50IGFzIHdlbGxcclxuICAgICAgICAvLyBXaGF0IHdlIG5lZWQgdG8gZG8gaXMgY29tcHV0ZSBhIHNjYWxlIG1hdHJpeCBhbmQgdGhlbiBiYWtlIHRoYXQgaW50byB0aGUgbWVzaCB2ZXJ0aWNlcy4gIFRoaXMgd2lsbCBsZWF2ZSB0aGVcclxuICAgICAgICAvLyBtZXNoIHNjYWxlIGF0IDEsIHNvIG91ciBjb250ZW50IHdpbGwgc3RheSBpdCdzIG9yaWdpbmFsIHdpZHRoIGFuZCBoZWlnaHQgdW50aWwgd2Ugc2NhbGUgdGhlIG1lc2guXHJcbiAgICAgICAgY29uc3Qgc2NhbGVYID0gdGhpcy5fd2lkdGggfHwgMTtcclxuICAgICAgICBjb25zdCBzY2FsZVkgPSB0aGlzLl9oZWlnaHQgfHwgMTtcclxuICAgICAgICBjb25zdCBzY2FsZU1hdHJpeCA9IE1hdHJpeC5TY2FsaW5nKHNjYWxlWCwgc2NhbGVZLCAxKTtcclxuICAgICAgICB0aGlzLmJha2VUcmFuc2Zvcm1JbnRvVmVydGljZXMoc2NhbGVNYXRyaXgpO1xyXG5cclxuICAgICAgICAvLyBHZXQgYW4gaW52ZXJzZSBvZiB0aGUgc2NhbGUgbWF0cml4IHRoYXQgd2UgY2FuIHVzZSB0byBiYWNrIG91dCB0aGUgc2NhbGUgY2hhbmdlcyB3ZSBoYXZlIG1hZGUgc29cclxuICAgICAgICAvLyB3ZSBkb24ndCBtdWx0aXBseSB0aGUgc2NhbGUuXHJcbiAgICAgICAgdGhpcy5faW52ZXJzZVNjYWxlTWF0cml4ID0gbmV3IE1hdHJpeCgpO1xyXG4gICAgICAgIHNjYWxlTWF0cml4LmludmVydFRvUmVmKHRoaXMuX2ludmVyc2VTY2FsZU1hdHJpeCk7XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF9jcmVhdGVNYXNrKCkge1xyXG4gICAgICAgIGNvbnN0IHZlcnRleERhdGEgPSBDcmVhdGVQbGFuZVZlcnRleERhdGEoeyB3aWR0aDogMSwgaGVpZ2h0OiAxIH0pO1xyXG4gICAgICAgIHZlcnRleERhdGEuYXBwbHlUb01lc2godGhpcyk7XHJcblxyXG4gICAgICAgIGNvbnN0IHNjZW5lID0gdGhpcy5nZXRTY2VuZSgpO1xyXG4gICAgICAgIHRoaXMuY2hlY2tDb2xsaXNpb25zID0gdHJ1ZTtcclxuXHJcbiAgICAgICAgY29uc3QgZGVwdGhNYXNrID0gbmV3IFN0YW5kYXJkTWF0ZXJpYWwoYCR7dGhpcy5pZH0tbWF0YCwgc2NlbmUpO1xyXG4gICAgICAgIGlmICghdGhpcy5faXNDYW52YXNPdmVybGF5KSB7XHJcbiAgICAgICAgICAgIGRlcHRoTWFzay5iYWNrRmFjZUN1bGxpbmcgPSBmYWxzZTtcclxuICAgICAgICAgICAgZGVwdGhNYXNrLmRpc2FibGVDb2xvcldyaXRlID0gdHJ1ZTtcclxuICAgICAgICAgICAgZGVwdGhNYXNrLmRpc2FibGVMaWdodGluZyA9IHRydWU7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICB0aGlzLm1hdGVyaWFsID0gZGVwdGhNYXNrO1xyXG5cclxuICAgICAgICAvLyBPcHRpbWl6YXRpb24gLSBGcmVlemUgbWF0ZXJpYWwgc2luY2UgaXQgbmV2ZXIgbmVlZHMgdG8gY2hhbmdlXHJcbiAgICAgICAgdGhpcy5tYXRlcmlhbC5mcmVlemUoKTtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX3NldEVsZW1lbnRaSW5kZXgoekluZGV4OiBudW1iZXIpIHtcclxuICAgICAgICBpZiAodGhpcy5fZWxlbWVudCkge1xyXG4gICAgICAgICAgICB0aGlzLl9lbGVtZW50IS5zdHlsZS56SW5kZXggPSBgJHt6SW5kZXh9YDtcclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDYWxsYmFjayB1c2VkIGJ5IHRoZSBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yIHRvIGNhcHR1cmUgcG9pbnRlciBldmVudHNcclxuICAgICAqL1xyXG4gICAgY2FwdHVyZVBvaW50ZXJFdmVudHMoKSB7XHJcbiAgICAgICAgaWYgKCF0aGlzLl9lbGVtZW50KSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIEVuYWJsZSBkb20gY29udGVudCB0byBjYXB0dXJlIHBvaW50ZXIgZXZlbnRzXHJcbiAgICAgICAgdGhpcy5fZWxlbWVudC5zdHlsZS5wb2ludGVyRXZlbnRzID0gXCJhdXRvXCI7XHJcblxyXG4gICAgICAgIC8vIFN1cHJlc3MgZXZlbnRzIG91dHNpZGUgb2YgdGhlIGRvbSBjb250ZW50XHJcbiAgICAgICAgZG9jdW1lbnQuZ2V0RWxlbWVudHNCeVRhZ05hbWUoXCJib2R5XCIpWzBdLnN0eWxlLnBvaW50ZXJFdmVudHMgPSBcIm5vbmVcIjtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENhbGxiYWNrIHVzZWQgYnkgdGhlIFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IgdG8gcmVsZWFzZSBwb2ludGVyIGV2ZW50c1xyXG4gICAgICovXHJcbiAgICByZWxlYXNlUG9pbnRlckV2ZW50cygpIHtcclxuICAgICAgICBpZiAoIXRoaXMuX2VsZW1lbnQpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gRW5hYmxlIHBvaW50ZXIgZXZlbnRzIG9uIGNhbnZhc1xyXG4gICAgICAgIGRvY3VtZW50LmdldEVsZW1lbnRzQnlUYWdOYW1lKFwiYm9keVwiKVswXS5zdHlsZS5wb2ludGVyRXZlbnRzID0gXCJhdXRvXCI7XHJcblxyXG4gICAgICAgIC8vIERpc2FibGUgcG9pbnRlciBldmVudHMgb24gZG9tIGNvbnRlbnRcclxuICAgICAgICB0aGlzLl9lbGVtZW50LnN0eWxlLnBvaW50ZXJFdmVudHMgPSBcIm5vbmVcIjtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX2NyZWF0ZUVsZW1lbnQoKSB7XHJcbiAgICAgICAgLy8gUmVxdWlyZXMgYSBicm93c2VyIHRvIHdvcmsuICBCYWlsIGlmIHdlIGFyZW4ndCBydW5uaW5nIGluIGEgYnJvd3NlclxyXG4gICAgICAgIGlmICh0eXBlb2YgZG9jdW1lbnQgPT09IFwidW5kZWZpbmVkXCIpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuICAgICAgICBjb25zdCBkaXYgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZGl2XCIpO1xyXG4gICAgICAgIGRpdi5pZCA9IHRoaXMuaWQ7XHJcbiAgICAgICAgZGl2LnN0eWxlLmJhY2tncm91bmRDb2xvciA9IHRoaXMuX2lzQ2FudmFzT3ZlcmxheSA/IFwidHJhbnNwYXJlbnRcIiA6IFwiIzAwMFwiO1xyXG4gICAgICAgIGRpdi5zdHlsZS56SW5kZXggPSBcIjFcIjtcclxuICAgICAgICBkaXYuc3R5bGUucG9zaXRpb24gPSBcImFic29sdXRlXCI7XHJcbiAgICAgICAgZGl2LnN0eWxlLnBvaW50ZXJFdmVudHMgPSBcIm5vbmVcIjtcclxuICAgICAgICBkaXYuc3R5bGUuYmFja2ZhY2VWaXNpYmlsaXR5ID0gXCJoaWRkZW5cIjtcclxuXHJcbiAgICAgICAgcmV0dXJuIGRpdjtcclxuICAgIH1cclxufVxyXG4iLCJpbXBvcnQgdHlwZSB7IFNjZW5lIH0gZnJvbSBcImNvcmUvc2NlbmVcIjtcclxuaW1wb3J0IHsgTWF0cml4LCBRdWF0ZXJuaW9uLCBWZWN0b3IzIH0gZnJvbSBcImNvcmUvTWF0aHMvbWF0aFwiO1xyXG5cclxuaW1wb3J0IHR5cGUgeyBIdG1sTWVzaCB9IGZyb20gXCIuL2h0bWxNZXNoXCI7XHJcbmltcG9ydCB7IENhbWVyYSB9IGZyb20gXCJjb3JlL0NhbWVyYXMvY2FtZXJhXCI7XHJcbmltcG9ydCB0eXBlIHsgU3ViTWVzaCB9IGZyb20gXCJjb3JlL01lc2hlcy9zdWJNZXNoXCI7XHJcbmltcG9ydCB7IFJlbmRlcmluZ0dyb3VwIH0gZnJvbSBcImNvcmUvUmVuZGVyaW5nL3JlbmRlcmluZ0dyb3VwXCI7XHJcblxyXG5pbXBvcnQgdHlwZSB7IE9ic2VydmVyIH0gZnJvbSBcImNvcmUvTWlzYy9vYnNlcnZhYmxlXCI7XHJcbmltcG9ydCB7IExvZ2dlciB9IGZyb20gXCJjb3JlL01pc2MvbG9nZ2VyXCI7XHJcbmltcG9ydCB0eXBlIHsgQWJzdHJhY3RFbmdpbmUgfSBmcm9tIFwiY29yZS9FbmdpbmVzXCI7XHJcblxyXG5jb25zdCBfcG9zaXRpb25VcGRhdGVGYWlsTWVzc2FnZSA9IFwiRmFpbGVkIHRvIHVwZGF0ZSBodG1sIG1lc2ggcmVuZGVyZXIgcG9zaXRpb24gZHVlIHRvIGZhaWx1cmUgdG8gZ2V0IGNhbnZhcyByZWN0LiAgSHRtbE1lc2ggaW5zdGFuY2VzIG1heSBub3QgcmVuZGVyIGNvcnJlY3RseVwiO1xyXG5jb25zdCBiYWJ5bG9uVW5pdHNUb1BpeGVscyA9IDEwMDtcclxuXHJcbi8qKlxyXG4gKiBBIGZ1bmN0aW9uIHRoYXQgY29tcGFyZXMgdHdvIHN1Ym1lc2hlcyBhbmQgcmV0dXJucyBhIG51bWJlciBpbmRpY2F0aW5nIHdoaWNoXHJcbiAqIHNob3VsZCBiZSByZW5kZXJlZCBmaXJzdC5cclxuICovXHJcbnR5cGUgUmVuZGVyT3JkZXJGdW5jdGlvbiA9IChzdWJNZXNoQTogU3ViTWVzaCwgc3ViTWVzaEI6IFN1Yk1lc2gpID0+IG51bWJlcjtcclxuXHJcbnR5cGUgUmVuZGVyTGF5ZXJFbGVtZW50cyA9IHtcclxuICAgIGNvbnRhaW5lcjogSFRNTEVsZW1lbnQ7XHJcbiAgICBkb21FbGVtZW50OiBIVE1MRWxlbWVudDtcclxuICAgIGNhbWVyYUVsZW1lbnQ6IEhUTUxFbGVtZW50O1xyXG59O1xyXG5cclxuLy8gUmV0dXJucyBhIGZ1bmN0aW9uIHRoYXQgZW5zdXJlcyB0aGF0IEh0bWxNZXNoZXMgYXJlIHJlbmRlcmVkIGJlZm9yZSBhbGwgb3RoZXIgbWVzaGVzLlxyXG4vLyBOb3RlIHRoaXMgd2lsbCBvbmx5IGJlIGFwcGxpZWQgdG8gZ3JvdXAgMC5cclxuLy8gSWYgbmVpdGhlciBtZXNoIGlzIGFuIEh0bWxNZXNoLCB0aGVuIHRoZSBkZWZhdWx0IHJlbmRlciBvcmRlciBpcyB1c2VkXHJcbi8vIFRoaXMgcHJldmVudHMgSHRtbE1lc2hlcyBmcm9tIGFwcGVhcmluZyBpbiBmcm9udCBvZiBvdGhlciBtZXNoZXMgd2hlbiB0aGV5IGFyZSBiZWhpbmQgdGhlbVxyXG5jb25zdCByZW5kZXJPcmRlckZ1bmMgPSAoZGVmYXVsdFJlbmRlck9yZGVyOiBSZW5kZXJPcmRlckZ1bmN0aW9uKTogUmVuZGVyT3JkZXJGdW5jdGlvbiA9PiB7XHJcbiAgICByZXR1cm4gKHN1Yk1lc2hBOiBTdWJNZXNoLCBzdWJNZXNoQjogU3ViTWVzaCkgPT4ge1xyXG4gICAgICAgIGNvbnN0IG1lc2hBID0gc3ViTWVzaEEuZ2V0TWVzaCgpO1xyXG4gICAgICAgIGNvbnN0IG1lc2hCID0gc3ViTWVzaEIuZ2V0TWVzaCgpO1xyXG5cclxuICAgICAgICAvLyBVc2UgcHJvcGVydHkgY2hlY2sgaW5zdGVhZCBvZiBpbnN0YW5jZW9mIHNpbmNlIGl0IGlzIGxlc3MgZXhwZW5zaXZlIGFuZFxyXG4gICAgICAgIC8vIHRoaXMgd2lsbCBiZSBjYWxsZWQgbWFueSB0aW1lcyBwZXIgZnJhbWVcclxuICAgICAgICBjb25zdCBtZXNoQUlzSHRtbE1lc2ggPSAobWVzaEEgYXMgYW55KVtcImlzSHRtbE1lc2hcIl07XHJcbiAgICAgICAgY29uc3QgbWVzaEJJc0h0bWxNZXNoID0gKG1lc2hCIGFzIGFueSlbXCJpc0h0bWxNZXNoXCJdO1xyXG4gICAgICAgIGlmIChtZXNoQUlzSHRtbE1lc2gpIHtcclxuICAgICAgICAgICAgcmV0dXJuIG1lc2hCSXNIdG1sTWVzaCA/IChtZXNoQS5hYnNvbHV0ZVBvc2l0aW9uLnogPD0gbWVzaEIuYWJzb2x1dGVQb3NpdGlvbi56ID8gMSA6IC0xKSA6IC0xO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIHJldHVybiBtZXNoQklzSHRtbE1lc2ggPyAxIDogZGVmYXVsdFJlbmRlck9yZGVyKHN1Yk1lc2hBLCBzdWJNZXNoQik7XHJcbiAgICAgICAgfVxyXG4gICAgfTtcclxufTtcclxuXHJcbi8qKlxyXG4gKiBBbiBpbnN0YW5jZSBvZiB0aGlzIGlzIHJlcXVpcmVkIHRvIHJlbmRlciBIdG1sTWVzaGVzIGluIHRoZSBzY2VuZS5cclxuICogaWYgdXNpbmcgSHRtbE1lc2hlcywgeW91IG11c3Qgbm90IHNldCByZW5kZXIgb3JkZXIgZm9yIGdyb3VwIDAgdXNpbmdcclxuICogc2NlbmUuc2V0UmVuZGVyaW5nT3JkZXIuICBZb3UgbXVzdCBpbnN0ZWFkIHBhc3MgdGhlIGNvbXBhcmUgZnVuY3Rpb25zXHJcbiAqIHRvIHRoZSBIdG1sTWVzaFJlbmRlcmVyIGNvbnN0cnVjdG9yLiAgSWYgeW91IGRvIG5vdCwgdGhlbiB5b3VyIHJlbmRlclxyXG4gKiBvcmRlciB3aWxsIGJlIG92ZXJ3cml0dGVuIGlmIHRoZSBIdG1sTWVzaFJlbmRlcmVyIGlzIGNyZWF0ZWQgYWZ0ZXIgYW5kXHJcbiAqIHRoZSBIdG1sTWVzaGVzIHdpbGwgbm90IHJlbmRlciBjb3JyZWN0bHkgKHRoZXkgd2lsbCBhcHBlYXIgaW4gZnJvbnQgb2ZcclxuICogbWVzaGVzIHRoYXQgYXJlIGFjdHVhbGx5IGluIGZyb250IG9mIHRoZW0pIGlmIHRoZSBIdG1sTWVzaFJlbmRlcmVyIGlzXHJcbiAqIGNyZWF0ZWQgYmVmb3JlLlxyXG4gKi9cclxuZXhwb3J0IGNsYXNzIEh0bWxNZXNoUmVuZGVyZXIge1xyXG4gICAgcHJpdmF0ZSBfY29udGFpbmVySWQ/OiBzdHJpbmc7XHJcbiAgICBwcml2YXRlIF9pblNjZW5lRWxlbWVudHM/OiBSZW5kZXJMYXllckVsZW1lbnRzIHwgbnVsbDtcclxuICAgIHByaXZhdGUgX292ZXJsYXlFbGVtZW50cz86IFJlbmRlckxheWVyRWxlbWVudHMgfCBudWxsO1xyXG4gICAgcHJpdmF0ZSBfZW5naW5lOiBBYnN0cmFjdEVuZ2luZTtcclxuXHJcbiAgICBwcml2YXRlIF9jYWNoZSA9IHtcclxuICAgICAgICBjYW1lcmFEYXRhOiB7IGZvdjogMCwgcG9zaXRpb246IG5ldyBWZWN0b3IzKCksIHN0eWxlOiBcIlwiIH0sXHJcbiAgICAgICAgaHRtbE1lc2hEYXRhOiBuZXcgV2Vha01hcDxvYmplY3QsIHsgc3R5bGU6IHN0cmluZyB9PigpLFxyXG4gICAgfTtcclxuICAgIHByaXZhdGUgX3dpZHRoID0gMDtcclxuICAgIHByaXZhdGUgX2hlaWdodCA9IDA7XHJcbiAgICBwcml2YXRlIF9oZWlnaHRIYWxmID0gMDtcclxuXHJcbiAgICBwcml2YXRlIF9jYW1lcmFXb3JsZE1hdHJpeD86IE1hdHJpeDtcclxuXHJcbiAgICAvLyBDcmVhdGUgc29tZSByZWZzIHRvIGF2b2lkIGNyZWF0aW5nIG5ldyBvYmplY3RzIGV2ZXJ5IGZyYW1lXHJcbiAgICBwcml2YXRlIF90ZW1wID0ge1xyXG4gICAgICAgIHNjYWxlVHJhbnNmb3JtOiBuZXcgVmVjdG9yMygpLFxyXG4gICAgICAgIHJvdGF0aW9uVHJhbnNmb3JtOiBuZXcgUXVhdGVybmlvbigpLFxyXG4gICAgICAgIHBvc2l0aW9uVHJhbnNmb3JtOiBuZXcgVmVjdG9yMygpLFxyXG4gICAgICAgIG9iamVjdE1hdHJpeDogTWF0cml4LklkZW50aXR5KCksXHJcbiAgICAgICAgY2FtZXJhV29ybGRNYXRyaXg6IE1hdHJpeC5JZGVudGl0eSgpLFxyXG4gICAgICAgIGNhbWVyYVJvdGF0aW9uTWF0cml4OiBNYXRyaXguSWRlbnRpdHkoKSxcclxuICAgICAgICBjYW1lcmFXb3JsZE1hdHJpeEFzQXJyYXk6IG5ldyBBcnJheSgxNiksXHJcbiAgICB9O1xyXG5cclxuICAgIC8vIEtlZXAgdHJhY2sgb2YgRFBSIHNvIHdlIGNhbiByZXNpemUgaWYgRFBSIGNoYW5nZXNcclxuICAgIC8vIE90aGVyd2lzZSB0aGUgRE9NIGNvbnRlbnQgd2lsbCBzY2FsZSwgYnV0IHRoZSBtZXNoIHdvbid0XHJcbiAgICBwcml2YXRlIF9sYXN0RGV2aWNlUGl4ZWxSYXRpbyA9IHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvO1xyXG5cclxuICAgIC8vIEtlZXAgdHJhY2sgb2YgY2FtZXJhIG1hdHJpeCBjaGFuZ2VzIHNvIHdlIG9ubHkgdXBkYXRlIHRoZVxyXG4gICAgLy8gRE9NIGVsZW1lbnQgc3R5bGVzIHdoZW4gbmVjZXNzYXJ5XHJcbiAgICBwcml2YXRlIF9jYW1lcmFNYXRyaXhVcGRhdGVkID0gdHJ1ZTtcclxuXHJcbiAgICAvLyBLZWVwIHRyYWNrIG9mIHBvc2l0aW9uIGNoYW5nZXMgc28gd2Ugb25seSB1cGRhdGUgdGhlIERPTSBlbGVtZW50XHJcbiAgICAvLyBzdHlsZXMgd2hlbiBuZWNlc3NhcnlcclxuICAgIHByaXZhdGUgX3ByZXZpb3VzQ2FudmFzRG9jdW1lbnRQb3NpdGlvbiA9IHtcclxuICAgICAgICB0b3A6IDAsXHJcbiAgICAgICAgbGVmdDogMCxcclxuICAgIH07XHJcblxyXG4gICAgcHJpdmF0ZSBfcmVuZGVyT2JzZXJ2ZXI6IE9ic2VydmVyPFNjZW5lPiB8IG51bGwgPSBudWxsO1xyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ29udHJ1Y3QgYW4gaW5zdGFuY2Ugb2YgSHRtbE1lc2hSZW5kZXJlclxyXG4gICAgICogQHBhcmFtIHNjZW5lXHJcbiAgICAgKiBAcGFyYW0gb3B0aW9ucyBvYmplY3QgY29udGFpbmluZyB0aGUgZm9sbG93aW5nIG9wdGlvbmFsIHByb3BlcnRpZXM6XHJcbiAgICAgKiBAcmV0dXJuc1xyXG4gICAgICovXHJcbiAgICBjb25zdHJ1Y3RvcihcclxuICAgICAgICBzY2VuZTogU2NlbmUsXHJcbiAgICAgICAge1xyXG4gICAgICAgICAgICBwYXJlbnRDb250YWluZXJJZCA9IG51bGwsXHJcbiAgICAgICAgICAgIF9jb250YWluZXJJZCA9IFwiY3NzLWNvbnRhaW5lclwiLFxyXG4gICAgICAgICAgICBlbmFibGVPdmVybGF5UmVuZGVyID0gdHJ1ZSxcclxuICAgICAgICAgICAgZGVmYXVsdE9wYXF1ZVJlbmRlck9yZGVyID0gUmVuZGVyaW5nR3JvdXAuUGFpbnRlclNvcnRDb21wYXJlLFxyXG4gICAgICAgICAgICBkZWZhdWx0QWxwaGFUZXN0UmVuZGVyT3JkZXIgPSBSZW5kZXJpbmdHcm91cC5QYWludGVyU29ydENvbXBhcmUsXHJcbiAgICAgICAgICAgIGRlZmF1bHRUcmFuc3BhcmVudFJlbmRlck9yZGVyID0gUmVuZGVyaW5nR3JvdXAuZGVmYXVsdFRyYW5zcGFyZW50U29ydENvbXBhcmUsXHJcbiAgICAgICAgfToge1xyXG4gICAgICAgICAgICBwYXJlbnRDb250YWluZXJJZD86IHN0cmluZyB8IG51bGw7XHJcbiAgICAgICAgICAgIF9jb250YWluZXJJZD86IHN0cmluZztcclxuICAgICAgICAgICAgZGVmYXVsdE9wYXF1ZVJlbmRlck9yZGVyPzogUmVuZGVyT3JkZXJGdW5jdGlvbjtcclxuICAgICAgICAgICAgZGVmYXVsdEFscGhhVGVzdFJlbmRlck9yZGVyPzogUmVuZGVyT3JkZXJGdW5jdGlvbjtcclxuICAgICAgICAgICAgZGVmYXVsdFRyYW5zcGFyZW50UmVuZGVyT3JkZXI/OiBSZW5kZXJPcmRlckZ1bmN0aW9uO1xyXG4gICAgICAgICAgICBlbmFibGVPdmVybGF5UmVuZGVyPzogYm9vbGVhbjtcclxuICAgICAgICB9ID0ge31cclxuICAgICkge1xyXG4gICAgICAgIC8vIFJlcXVpcmVzIGEgYnJvd3NlciB0byB3b3JrLiAgT25seSBpbml0IGlmIHdlIGFyZSBpbiBhIGJyb3dzZXJcclxuICAgICAgICBpZiAodHlwZW9mIGRvY3VtZW50ID09PSBcInVuZGVmaW5lZFwiKSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcbiAgICAgICAgdGhpcy5fY29udGFpbmVySWQgPSBfY29udGFpbmVySWQ7XHJcbiAgICAgICAgdGhpcy5faW5pdChzY2VuZSwgcGFyZW50Q29udGFpbmVySWQsIGVuYWJsZU92ZXJsYXlSZW5kZXIsIGRlZmF1bHRPcGFxdWVSZW5kZXJPcmRlciwgZGVmYXVsdEFscGhhVGVzdFJlbmRlck9yZGVyLCBkZWZhdWx0VHJhbnNwYXJlbnRSZW5kZXJPcmRlcik7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBEaXNwb3NlIG9mIHRoZSBIdG1sTWVzaFJlbmRlcmVyXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBkaXNwb3NlKCkge1xyXG4gICAgICAgIGlmICh0aGlzLl9yZW5kZXJPYnNlcnZlcikge1xyXG4gICAgICAgICAgICB0aGlzLl9yZW5kZXJPYnNlcnZlci5yZW1vdmUoKTtcclxuICAgICAgICAgICAgdGhpcy5fcmVuZGVyT2JzZXJ2ZXIgPSBudWxsO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgdGhpcy5fb3ZlcmxheUVsZW1lbnRzPy5jb250YWluZXIucmVtb3ZlKCk7XHJcbiAgICAgICAgdGhpcy5fb3ZlcmxheUVsZW1lbnRzID0gbnVsbDtcclxuXHJcbiAgICAgICAgdGhpcy5faW5TY2VuZUVsZW1lbnRzPy5jb250YWluZXIucmVtb3ZlKCk7XHJcbiAgICAgICAgdGhpcy5faW5TY2VuZUVsZW1lbnRzID0gbnVsbDtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX2luaXQoXHJcbiAgICAgICAgc2NlbmU6IFNjZW5lLFxyXG4gICAgICAgIHBhcmVudENvbnRhaW5lcklkOiBzdHJpbmcgfCBudWxsLFxyXG4gICAgICAgIGVuYWJsZU92ZXJsYXlSZW5kZXI6IGJvb2xlYW4sXHJcbiAgICAgICAgZGVmYXVsdE9wYXF1ZVJlbmRlck9yZGVyOiBSZW5kZXJPcmRlckZ1bmN0aW9uLFxyXG4gICAgICAgIGRlZmF1bHRBbHBoYVRlc3RSZW5kZXJPcmRlcjogUmVuZGVyT3JkZXJGdW5jdGlvbixcclxuICAgICAgICBkZWZhdWx0VHJhbnNwYXJlbnRSZW5kZXJPcmRlcjogUmVuZGVyT3JkZXJGdW5jdGlvblxyXG4gICAgKTogdm9pZCB7XHJcbiAgICAgICAgLy8gUmVxdWlyZXMgYSBicm93c2VyIHRvIHdvcmsuICBPbmx5IGluaXQgaWYgd2UgYXJlIGluIGEgYnJvd3NlclxyXG4gICAgICAgIGlmICh0eXBlb2YgZG9jdW1lbnQgPT09IFwidW5kZWZpbmVkXCIpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gQ3JlYXRlIHRoZSBET00gY29udGFpbmVyc1xyXG4gICAgICAgIGxldCBwYXJlbnRDb250YWluZXIgPSBwYXJlbnRDb250YWluZXJJZCA/IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKHBhcmVudENvbnRhaW5lcklkKSA6IGRvY3VtZW50LmJvZHk7XHJcblxyXG4gICAgICAgIGlmICghcGFyZW50Q29udGFpbmVyKSB7XHJcbiAgICAgICAgICAgIHBhcmVudENvbnRhaW5lciA9IGRvY3VtZW50LmJvZHk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBpZiB0aGUgY29udGFpbmVyIGFscmVhZHkgZXhpc3RzLCB0aGVuIHJlbW92ZSBpdFxyXG4gICAgICAgIGNvbnN0IGluU2NlbmVDb250YWluZXJJZCA9IGAke3RoaXMuX2NvbnRhaW5lcklkfV9pbl9zY2VuZWA7XHJcbiAgICAgICAgdGhpcy5faW5TY2VuZUVsZW1lbnRzID0gdGhpcy5fY3JlYXRlUmVuZGVyTGF5ZXJFbGVtZW50cyhpblNjZW5lQ29udGFpbmVySWQpO1xyXG5cclxuICAgICAgICBwYXJlbnRDb250YWluZXIuaW5zZXJ0QmVmb3JlKHRoaXMuX2luU2NlbmVFbGVtZW50cy5jb250YWluZXIsIHBhcmVudENvbnRhaW5lci5maXJzdENoaWxkKTtcclxuXHJcbiAgICAgICAgaWYgKGVuYWJsZU92ZXJsYXlSZW5kZXIpIHtcclxuICAgICAgICAgICAgY29uc3Qgb3ZlcmxheUNvbnRhaW5lcklkID0gYCR7dGhpcy5fY29udGFpbmVySWR9X292ZXJsYXlgO1xyXG4gICAgICAgICAgICB0aGlzLl9vdmVybGF5RWxlbWVudHMgPSB0aGlzLl9jcmVhdGVSZW5kZXJMYXllckVsZW1lbnRzKG92ZXJsYXlDb250YWluZXJJZCk7XHJcbiAgICAgICAgICAgIGNvbnN0IHpJbmRleCA9ICsoc2NlbmUuZ2V0RW5naW5lKCkuZ2V0UmVuZGVyaW5nQ2FudmFzKCkhLnN0eWxlLnpJbmRleCA/PyBcIjBcIikgKyAxO1xyXG4gICAgICAgICAgICB0aGlzLl9vdmVybGF5RWxlbWVudHMuY29udGFpbmVyLnN0eWxlLnpJbmRleCA9IGAke3pJbmRleH1gO1xyXG4gICAgICAgICAgICB0aGlzLl9vdmVybGF5RWxlbWVudHMuY29udGFpbmVyLnN0eWxlLnBvaW50ZXJFdmVudHMgPSBcIm5vbmVcIjtcclxuICAgICAgICAgICAgcGFyZW50Q29udGFpbmVyLmluc2VydEJlZm9yZSh0aGlzLl9vdmVybGF5RWxlbWVudHMuY29udGFpbmVyLCBwYXJlbnRDb250YWluZXIuZmlyc3RDaGlsZCk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHRoaXMuX2VuZ2luZSA9IHNjZW5lLmdldEVuZ2luZSgpO1xyXG4gICAgICAgIGNvbnN0IGNsaWVudFJlY3QgPSB0aGlzLl9lbmdpbmUuZ2V0UmVuZGVyaW5nQ2FudmFzQ2xpZW50UmVjdCgpO1xyXG4gICAgICAgIGlmICghY2xpZW50UmVjdCkge1xyXG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGYWlsZWQgdG8gZ2V0IGNsaWVudCByZWN0IGZvciByZW5kZXJpbmcgY2FudmFzXCIpO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gU2V0IHRoZSBzaXplIGFuZCByZXNpemUgYmVoYXZpb3JcclxuICAgICAgICB0aGlzLl9zZXRTaXplKGNsaWVudFJlY3Qud2lkdGgsIGNsaWVudFJlY3QuaGVpZ2h0KTtcclxuXHJcbiAgICAgICAgdGhpcy5fZW5naW5lLm9uUmVzaXplT2JzZXJ2YWJsZS5hZGQoKCkgPT4ge1xyXG4gICAgICAgICAgICBjb25zdCBjbGllbnRSZWN0ID0gdGhpcy5fZW5naW5lLmdldFJlbmRlcmluZ0NhbnZhc0NsaWVudFJlY3QoKTtcclxuICAgICAgICAgICAgaWYgKGNsaWVudFJlY3QpIHtcclxuICAgICAgICAgICAgICAgIHRoaXMuX3NldFNpemUoY2xpZW50UmVjdC53aWR0aCwgY2xpZW50UmVjdC5oZWlnaHQpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfSk7XHJcblxyXG4gICAgICAgIGxldCBwcm9qZWN0aW9uT2JzOiBPYnNlcnZlcjxDYW1lcmE+O1xyXG4gICAgICAgIGxldCBtYXRyaXhPYnM6IE9ic2VydmVyPENhbWVyYT47XHJcblxyXG4gICAgICAgIGNvbnN0IG9ic2VydmVDYW1lcmEgPSAoKSA9PiB7XHJcbiAgICAgICAgICAgIGNvbnN0IGNhbWVyYSA9IHNjZW5lLmFjdGl2ZUNhbWVyYTtcclxuICAgICAgICAgICAgaWYgKGNhbWVyYSkge1xyXG4gICAgICAgICAgICAgICAgcHJvamVjdGlvbk9icyA9IGNhbWVyYS5vblByb2plY3Rpb25NYXRyaXhDaGFuZ2VkT2JzZXJ2YWJsZS5hZGQoKCkgPT4ge1xyXG4gICAgICAgICAgICAgICAgICAgIHRoaXMuX29uQ2FtZXJhTWF0cml4Q2hhbmdlZChjYW1lcmEpO1xyXG4gICAgICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgICAgICAgICBtYXRyaXhPYnMgPSBjYW1lcmEub25WaWV3TWF0cml4Q2hhbmdlZE9ic2VydmFibGUuYWRkKCgpID0+IHtcclxuICAgICAgICAgICAgICAgICAgICB0aGlzLl9vbkNhbWVyYU1hdHJpeENoYW5nZWQoY2FtZXJhKTtcclxuICAgICAgICAgICAgICAgIH0pO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfTtcclxuXHJcbiAgICAgICAgb2JzZXJ2ZUNhbWVyYSgpO1xyXG5cclxuICAgICAgICBzY2VuZS5vbkFjdGl2ZUNhbWVyYUNoYW5nZWQuYWRkKCgpID0+IHtcclxuICAgICAgICAgICAgaWYgKHByb2plY3Rpb25PYnMpIHtcclxuICAgICAgICAgICAgICAgIHNjZW5lLmFjdGl2ZUNhbWVyYT8ub25Qcm9qZWN0aW9uTWF0cml4Q2hhbmdlZE9ic2VydmFibGUucmVtb3ZlKHByb2plY3Rpb25PYnMpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIGlmIChtYXRyaXhPYnMpIHtcclxuICAgICAgICAgICAgICAgIHNjZW5lLmFjdGl2ZUNhbWVyYT8ub25WaWV3TWF0cml4Q2hhbmdlZE9ic2VydmFibGUucmVtb3ZlKG1hdHJpeE9icyk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgb2JzZXJ2ZUNhbWVyYSgpO1xyXG4gICAgICAgIH0pO1xyXG5cclxuICAgICAgICAvLyBXZSBuZWVkIHRvIG1ha2Ugc3VyZSB0aGF0IEh0bWxNZXNoZXMgYXJlIHJlbmRlcmVkIGJlZm9yZSBhbGwgb3RoZXIgbWVzaGVzXHJcbiAgICAgICAgLy8gc28gdGhhdCB0aGV5IGRvbid0IGFwcGVhciBpbiBmcm9udCBvZiBtZXNoZXMgdGhhdCBhcmUgYWN0dWFsbHkgaW4gZnJvbnQgb2YgdGhlbVxyXG4gICAgICAgIC8vIFVwZGF0aW5nIHRoZSByZW5kZXIgb3JkZXIgaXNuJ3QgaWRlYWwsIGJ1dCBpdCBpcyB0aGUgb25seSB3YXkgdG8gYWNoZWl2ZSB0aGlzXHJcbiAgICAgICAgLy8gVGhlIGltcGxpY2F0aW9uIGlzIHRoYXQgYW4gYXBwIHVzaW5nIHRoZSBIdG1sTWVzaFJlbmRlcmVkIG11c3Qgc2V0IHRoZSBzY2VuZSByZW5kZXIgb3JkZXJcclxuICAgICAgICAvLyB2aWEgdGhlIEh0bWxNZXNoUmVuZGVyZWQgY29uc3RydWN0b3JcclxuICAgICAgICBjb25zdCBvcGFxdWVSZW5kZXJPcmRlciA9IHJlbmRlck9yZGVyRnVuYyhkZWZhdWx0T3BhcXVlUmVuZGVyT3JkZXIpO1xyXG4gICAgICAgIGNvbnN0IGFscGhhVGVzdFJlbmRlck9yZGVyID0gcmVuZGVyT3JkZXJGdW5jKGRlZmF1bHRBbHBoYVRlc3RSZW5kZXJPcmRlcik7XHJcbiAgICAgICAgY29uc3QgdHJhbnNwYXJlbnRSZW5kZXJPcmRlciA9IHJlbmRlck9yZGVyRnVuYyhkZWZhdWx0VHJhbnNwYXJlbnRSZW5kZXJPcmRlcik7XHJcbiAgICAgICAgc2NlbmUuc2V0UmVuZGVyaW5nT3JkZXIoMCwgb3BhcXVlUmVuZGVyT3JkZXIsIGFscGhhVGVzdFJlbmRlck9yZGVyLCB0cmFuc3BhcmVudFJlbmRlck9yZGVyKTtcclxuXHJcbiAgICAgICAgdGhpcy5fcmVuZGVyT2JzZXJ2ZXIgPSBzY2VuZS5vbkJlZm9yZVJlbmRlck9ic2VydmFibGUuYWRkKCgpID0+IHtcclxuICAgICAgICAgICAgdGhpcy5fcmVuZGVyKHNjZW5lLCBzY2VuZS5hY3RpdmVDYW1lcmEgYXMgQ2FtZXJhKTtcclxuICAgICAgICB9KTtcclxuICAgIH1cclxuXHJcbiAgICBwcml2YXRlIF9jcmVhdGVSZW5kZXJMYXllckVsZW1lbnRzKGNvbnRhaW5lcklkOiBzdHJpbmcpOiBSZW5kZXJMYXllckVsZW1lbnRzIHtcclxuICAgICAgICBjb25zdCBleGlzdGluZ0NvbnRhaW5lciA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKGNvbnRhaW5lcklkKTtcclxuICAgICAgICBpZiAoZXhpc3RpbmdDb250YWluZXIpIHtcclxuICAgICAgICAgICAgZXhpc3RpbmdDb250YWluZXIucmVtb3ZlKCk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGNvbnN0IGNvbnRhaW5lciA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcbiAgICAgICAgY29udGFpbmVyLmlkID0gY29udGFpbmVySWQ7XHJcbiAgICAgICAgY29udGFpbmVyLnN0eWxlLnBvc2l0aW9uID0gXCJhYnNvbHV0ZVwiO1xyXG4gICAgICAgIGNvbnRhaW5lci5zdHlsZS53aWR0aCA9IFwiMTAwJVwiO1xyXG4gICAgICAgIGNvbnRhaW5lci5zdHlsZS5oZWlnaHQgPSBcIjEwMCVcIjtcclxuICAgICAgICBjb250YWluZXIuc3R5bGUuekluZGV4ID0gXCItMVwiO1xyXG5cclxuICAgICAgICBjb25zdCBkb21FbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImRpdlwiKTtcclxuICAgICAgICBkb21FbGVtZW50LnN0eWxlLm92ZXJmbG93ID0gXCJoaWRkZW5cIjtcclxuXHJcbiAgICAgICAgY29uc3QgY2FtZXJhRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcblxyXG4gICAgICAgIGNhbWVyYUVsZW1lbnQuc3R5bGUud2Via2l0VHJhbnNmb3JtU3R5bGUgPSBcInByZXNlcnZlLTNkXCI7XHJcbiAgICAgICAgY2FtZXJhRWxlbWVudC5zdHlsZS50cmFuc2Zvcm1TdHlsZSA9IFwicHJlc2VydmUtM2RcIjtcclxuXHJcbiAgICAgICAgY2FtZXJhRWxlbWVudC5zdHlsZS5wb2ludGVyRXZlbnRzID0gXCJub25lXCI7XHJcblxyXG4gICAgICAgIGRvbUVsZW1lbnQuYXBwZW5kQ2hpbGQoY2FtZXJhRWxlbWVudCk7XHJcbiAgICAgICAgY29udGFpbmVyLmFwcGVuZENoaWxkKGRvbUVsZW1lbnQpO1xyXG4gICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICAgIGNvbnRhaW5lcixcclxuICAgICAgICAgICAgZG9tRWxlbWVudCxcclxuICAgICAgICAgICAgY2FtZXJhRWxlbWVudCxcclxuICAgICAgICB9O1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfZ2V0U2l6ZSgpOiB7IHdpZHRoOiBudW1iZXI7IGhlaWdodDogbnVtYmVyIH0ge1xyXG4gICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICAgIHdpZHRoOiB0aGlzLl93aWR0aCxcclxuICAgICAgICAgICAgaGVpZ2h0OiB0aGlzLl9oZWlnaHQsXHJcbiAgICAgICAgfTtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX3NldFNpemUod2lkdGg6IG51bWJlciwgaGVpZ2h0OiBudW1iZXIpOiB2b2lkIHtcclxuICAgICAgICB0aGlzLl93aWR0aCA9IHdpZHRoO1xyXG4gICAgICAgIHRoaXMuX2hlaWdodCA9IGhlaWdodDtcclxuICAgICAgICB0aGlzLl9oZWlnaHRIYWxmID0gdGhpcy5faGVpZ2h0IC8gMjtcclxuXHJcbiAgICAgICAgaWYgKCF0aGlzLl9pblNjZW5lRWxlbWVudHMgfHwgIXRoaXMuX292ZXJsYXlFbGVtZW50cykge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBjb25zdCBkb21FbGVtZW50cyA9IFt0aGlzLl9pblNjZW5lRWxlbWVudHMhLmRvbUVsZW1lbnQsIHRoaXMuX292ZXJsYXlFbGVtZW50cyEuZG9tRWxlbWVudCwgdGhpcy5faW5TY2VuZUVsZW1lbnRzIS5jYW1lcmFFbGVtZW50LCB0aGlzLl9vdmVybGF5RWxlbWVudHMhLmNhbWVyYUVsZW1lbnRdO1xyXG4gICAgICAgIGRvbUVsZW1lbnRzLmZvckVhY2goKGRvbSkgPT4ge1xyXG4gICAgICAgICAgICBpZiAoZG9tKSB7XHJcbiAgICAgICAgICAgICAgICBkb20uc3R5bGUud2lkdGggPSBgJHt3aWR0aH1weGA7XHJcbiAgICAgICAgICAgICAgICBkb20uc3R5bGUuaGVpZ2h0ID0gYCR7aGVpZ2h0fXB4YDtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIHByZXR0aWVyLWlnbm9yZVxyXG4gICAgcHJvdGVjdGVkIF9nZXRDYW1lcmFDU1NNYXRyaXgobWF0cml4OiBNYXRyaXgpOiBzdHJpbmcge1xyXG4gICAgICAgIGNvbnN0IGVsZW1lbnRzID0gbWF0cml4Lm07XHJcbiAgICAgICAgcmV0dXJuIGBtYXRyaXgzZCgke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1swXSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCAtIGVsZW1lbnRzWzFdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzJdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzNdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzRdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIC0gZWxlbWVudHNbNV0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbNl0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbN10gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbOF0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggLSBlbGVtZW50c1s5XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxMF0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMTFdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzEyXSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCAtIGVsZW1lbnRzWzEzXSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxNF0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMTVdIClcclxuICAgICAgICB9KWA7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gQ29udmVydCBhIEJhYnlsb24gd29ybGQgbWF0cml4IHRvIGEgQ1NTIG1hdHJpeFxyXG4gICAgLy8gVGhpcyBhbHNvIGhhbmRsZXMgY29udmVyc2lvbiBmcm9tIEJKUyBsZWZ0IGhhbmRlZCBjb29yZHNcclxuICAgIC8vIHRvIENTUyByaWdodCBoYW5kZWQgY29vcmRzXHJcbiAgICAvLyBwcmV0dGllci1pZ25vcmVcclxuICAgIHByb3RlY3RlZCBfZ2V0SHRtbENvbnRlbnRDU1NNYXRyaXgobWF0cml4OiBNYXRyaXgsIHVzZVJpZ2h0SGFuZGVkU3lzdGVtOiBib29sZWFuKTogc3RyaW5nIHtcclxuICAgICAgICBjb25zdCBlbGVtZW50cyA9IG1hdHJpeC5tO1xyXG4gICAgICAgIC8vIEluIGEgcmlnaHQgaGFuZGVkIGNvb3JkaW5hdGUgc3lzdGVtLCB0aGUgZWxlbWVudHMgMTEgdG8gMTQgaGF2ZSB0byBjaGFuZ2UgdGhlaXIgZGlyZWN0aW9uXHJcbiAgICAgICAgY29uc3QgZGlyZWN0aW9uID0gdXNlUmlnaHRIYW5kZWRTeXN0ZW0gPyAtMSA6IDE7XHJcbiAgICAgICAgY29uc3QgbWF0cml4M2QgPSBgbWF0cml4M2QoJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMF0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMV0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMl0gKiAtZGlyZWN0aW9uIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzNdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIC0gZWxlbWVudHNbNF0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggLSBlbGVtZW50c1s1XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1s2XSAgKiBkaXJlY3Rpb24gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggLSBlbGVtZW50c1s3XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1s4XSAqIC1kaXJlY3Rpb24gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbOV0gKiAtZGlyZWN0aW9uIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzEwXSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxMV0gKiBkaXJlY3Rpb24gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMTJdICogZGlyZWN0aW9uIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzEzXSAqIGRpcmVjdGlvbiApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxNF0gKiBkaXJlY3Rpb24gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMTVdIClcclxuICAgICAgICB9KWA7XHJcbiAgICAgICAgcmV0dXJuIG1hdHJpeDNkO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfZ2V0VHJhbnNmb3JtYXRpb25NYXRyaXgoaHRtbE1lc2g6IEh0bWxNZXNoLCB1c2VSaWdodEhhbmRlZFN5c3RlbTogYm9vbGVhbik6IE1hdHJpeCB7XHJcbiAgICAgICAgLy8gR2V0IHRoZSBjYW1lcmEgd29ybGQgbWF0cml4XHJcbiAgICAgICAgLy8gTWFrZSBzdXJlIHRoZSBjYW1lcmEgd29ybGQgbWF0cml4IGlzIHVwIHRvIGRhdGVcclxuICAgICAgICBpZiAoIXRoaXMuX2NhbWVyYVdvcmxkTWF0cml4KSB7XHJcbiAgICAgICAgICAgIHRoaXMuX2NhbWVyYVdvcmxkTWF0cml4ID0gaHRtbE1lc2guZ2V0U2NlbmUoKS5hY3RpdmVDYW1lcmE/LmdldFdvcmxkTWF0cml4KCk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGlmICghdGhpcy5fY2FtZXJhV29ybGRNYXRyaXgpIHtcclxuICAgICAgICAgICAgcmV0dXJuIE1hdHJpeC5JZGVudGl0eSgpO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgY29uc3Qgb2JqZWN0V29ybGRNYXRyaXggPSBodG1sTWVzaC5nZXRXb3JsZE1hdHJpeCgpO1xyXG5cclxuICAgICAgICAvLyBTY2FsZSB0aGUgb2JqZWN0IG1hdHJpeCBieSB0aGUgYmFzZSBzY2FsZSBmYWN0b3IgZm9yIHRoZSBtZXNoXHJcbiAgICAgICAgLy8gd2hpY2ggaXMgdGhlIHJhdGlvIG9mIHRoZSBtZXNoIHdpZHRoL2hlaWdodCB0byB0aGUgcmVuZGVyZXJcclxuICAgICAgICAvLyB3aWR0aC9oZWlnaHQgZGl2aWRlZCBieSB0aGUgYmFieWxvbiB1bml0cyB0byBwaXhlbHMgcmF0aW9cclxuICAgICAgICBsZXQgd2lkdGhTY2FsZUZhY3RvciA9IDE7XHJcbiAgICAgICAgbGV0IGhlaWdodFNjYWxlRmFjdG9yID0gMTtcclxuICAgICAgICBpZiAoaHRtbE1lc2guc291cmNlV2lkdGggJiYgaHRtbE1lc2guc291cmNlSGVpZ2h0KSB7XHJcbiAgICAgICAgICAgIHdpZHRoU2NhbGVGYWN0b3IgPSBodG1sTWVzaC53aWR0aCEgLyAoaHRtbE1lc2guc291cmNlV2lkdGggLyBiYWJ5bG9uVW5pdHNUb1BpeGVscyk7XHJcbiAgICAgICAgICAgIGhlaWdodFNjYWxlRmFjdG9yID0gaHRtbE1lc2guaGVpZ2h0ISAvIChodG1sTWVzaC5zb3VyY2VIZWlnaHQgLyBiYWJ5bG9uVW5pdHNUb1BpeGVscyk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBBcHBseSB0aGUgc2NhbGUgdG8gdGhlIG9iamVjdCdzIHdvcmxkIG1hdHJpeC4gIE5vdGUgd2UgYXJlbid0IHNjYWxpbmdcclxuICAgICAgICAvLyB0aGUgb2JqZWN0LCBqdXN0IGdldHRpbmcgYSBtYXRyaXggYXMgdGhvdWdoIGl0IHdlcmUgc2NhbGVkLCBzbyB3ZSBjYW5cclxuICAgICAgICAvLyBzY2FsZSB0aGUgY29udGVudFxyXG4gICAgICAgIGNvbnN0IHNjYWxlVHJhbnNmb3JtID0gdGhpcy5fdGVtcC5zY2FsZVRyYW5zZm9ybTtcclxuICAgICAgICBjb25zdCByb3RhdGlvblRyYW5zZm9ybSA9IHRoaXMuX3RlbXAucm90YXRpb25UcmFuc2Zvcm07XHJcbiAgICAgICAgY29uc3QgcG9zaXRpb25UcmFuc2Zvcm0gPSB0aGlzLl90ZW1wLnBvc2l0aW9uVHJhbnNmb3JtO1xyXG4gICAgICAgIGNvbnN0IHNjYWxlZEFuZFRyYW5zbGF0ZWRPYmplY3RNYXRyaXggPSB0aGlzLl90ZW1wLm9iamVjdE1hdHJpeDtcclxuXHJcbiAgICAgICAgb2JqZWN0V29ybGRNYXRyaXguZGVjb21wb3NlKHNjYWxlVHJhbnNmb3JtLCByb3RhdGlvblRyYW5zZm9ybSwgcG9zaXRpb25UcmFuc2Zvcm0pO1xyXG4gICAgICAgIHNjYWxlVHJhbnNmb3JtLnggKj0gd2lkdGhTY2FsZUZhY3RvcjtcclxuICAgICAgICBzY2FsZVRyYW5zZm9ybS55ICo9IGhlaWdodFNjYWxlRmFjdG9yO1xyXG5cclxuICAgICAgICBNYXRyaXguQ29tcG9zZVRvUmVmKHNjYWxlVHJhbnNmb3JtLCByb3RhdGlvblRyYW5zZm9ybSwgcG9zaXRpb25UcmFuc2Zvcm0sIHNjYWxlZEFuZFRyYW5zbGF0ZWRPYmplY3RNYXRyaXgpO1xyXG5cclxuICAgICAgICAvLyBBZGp1c3QgZGlyZWN0aW9uIG9mIDEyIGFuZCAxMyBvZiB0aGUgdHJhbnNmb3JtYXRpb24gbWF0cml4IGJhc2VkIG9uIHRoZSBoYW5kZWRuZXNzIG9mIHRoZSBzeXN0ZW1cclxuICAgICAgICBjb25zdCBkaXJlY3Rpb24gPSB1c2VSaWdodEhhbmRlZFN5c3RlbSA/IC0xIDogMTtcclxuICAgICAgICAvLyBBZGp1c3QgdHJhbnNsYXRpb24gdmFsdWVzIHRvIGJlIGZyb20gY2FtZXJhIHZzIHdvcmxkIG9yaWdpblxyXG4gICAgICAgIC8vIE5vdGUgdGhhdCB3ZSBhcmUgYWxzbyBhZGp1c3RpbmcgdGhlc2UgdmFsdWVzIHRvIGJlIHBpeGVscyB2cyBCYWJ5bG9uIHVuaXRzXHJcbiAgICAgICAgY29uc3QgcG9zaXRpb24gPSBodG1sTWVzaC5nZXRBYnNvbHV0ZVBvc2l0aW9uKCk7XHJcbiAgICAgICAgc2NhbGVkQW5kVHJhbnNsYXRlZE9iamVjdE1hdHJpeC5zZXRSb3dGcm9tRmxvYXRzKFxyXG4gICAgICAgICAgICAzLFxyXG4gICAgICAgICAgICAoLXRoaXMuX2NhbWVyYVdvcmxkTWF0cml4Lm1bMTJdICsgcG9zaXRpb24ueCkgKiBiYWJ5bG9uVW5pdHNUb1BpeGVscyAqIGRpcmVjdGlvbixcclxuICAgICAgICAgICAgKC10aGlzLl9jYW1lcmFXb3JsZE1hdHJpeC5tWzEzXSArIHBvc2l0aW9uLnkpICogYmFieWxvblVuaXRzVG9QaXhlbHMgKiBkaXJlY3Rpb24sXHJcbiAgICAgICAgICAgICh0aGlzLl9jYW1lcmFXb3JsZE1hdHJpeC5tWzE0XSAtIHBvc2l0aW9uLnopICogYmFieWxvblVuaXRzVG9QaXhlbHMsXHJcbiAgICAgICAgICAgIHRoaXMuX2NhbWVyYVdvcmxkTWF0cml4Lm1bMTVdICogMC4wMDAwMSAqIGJhYnlsb25Vbml0c1RvUGl4ZWxzXHJcbiAgICAgICAgKTtcclxuXHJcbiAgICAgICAgLy8gQWRqdXN0IG90aGVyIHZhbHVlcyB0byBiZSBwaXhlbHMgdnMgQmFieWxvbiB1bml0c1xyXG4gICAgICAgIHNjYWxlZEFuZFRyYW5zbGF0ZWRPYmplY3RNYXRyaXgubXVsdGlwbHlBdEluZGV4KDMsIGJhYnlsb25Vbml0c1RvUGl4ZWxzKTtcclxuICAgICAgICBzY2FsZWRBbmRUcmFuc2xhdGVkT2JqZWN0TWF0cml4Lm11bHRpcGx5QXRJbmRleCg3LCBiYWJ5bG9uVW5pdHNUb1BpeGVscyk7XHJcbiAgICAgICAgc2NhbGVkQW5kVHJhbnNsYXRlZE9iamVjdE1hdHJpeC5tdWx0aXBseUF0SW5kZXgoMTEsIGJhYnlsb25Vbml0c1RvUGl4ZWxzKTtcclxuXHJcbiAgICAgICAgcmV0dXJuIHNjYWxlZEFuZFRyYW5zbGF0ZWRPYmplY3RNYXRyaXg7XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF9yZW5kZXJIdG1sTWVzaChodG1sTWVzaDogSHRtbE1lc2gsIHVzZVJpZ2h0SGFuZGVkU3lzdGVtOiBib29sZWFuKSB7XHJcbiAgICAgICAgaWYgKCFodG1sTWVzaC5lbGVtZW50IHx8ICFodG1sTWVzaC5lbGVtZW50LmZpcnN0RWxlbWVudENoaWxkKSB7XHJcbiAgICAgICAgICAgIC8vIG5vdGhpbmcgdG8gcmVuZGVyLCBzbyBiYWlsXHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIFdlIG5lZWQgdG8gZW5zdXJlIGh0bWwgbWVzaCBkYXRhIGlzIGluaXRpYWxpemVkIGJlZm9yZVxyXG4gICAgICAgIC8vIGNvbXB1dGluZyB0aGUgYmFzZSBzY2FsZSBmYWN0b3JcclxuICAgICAgICBsZXQgaHRtbE1lc2hEYXRhID0gdGhpcy5fY2FjaGUuaHRtbE1lc2hEYXRhLmdldChodG1sTWVzaCk7XHJcbiAgICAgICAgaWYgKCFodG1sTWVzaERhdGEpIHtcclxuICAgICAgICAgICAgaHRtbE1lc2hEYXRhID0geyBzdHlsZTogXCJcIiB9O1xyXG4gICAgICAgICAgICB0aGlzLl9jYWNoZS5odG1sTWVzaERhdGEuc2V0KGh0bWxNZXNoLCBodG1sTWVzaERhdGEpO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgY29uc3QgY2FtZXJhRWxlbWVudCA9IGh0bWxNZXNoLl9pc0NhbnZhc092ZXJsYXkgPyB0aGlzLl9vdmVybGF5RWxlbWVudHM/LmNhbWVyYUVsZW1lbnQgOiB0aGlzLl9pblNjZW5lRWxlbWVudHM/LmNhbWVyYUVsZW1lbnQ7XHJcblxyXG4gICAgICAgIGlmIChodG1sTWVzaC5lbGVtZW50LnBhcmVudE5vZGUgIT09IGNhbWVyYUVsZW1lbnQpIHtcclxuICAgICAgICAgICAgY2FtZXJhRWxlbWVudCEuYXBwZW5kQ2hpbGQoaHRtbE1lc2guZWxlbWVudCk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBJZiB0aGUgaHRtbE1lc2ggY29udGVudCBoYXMgY2hhbmdlZCwgdXBkYXRlIHRoZSBiYXNlIHNjYWxlIGZhY3RvclxyXG4gICAgICAgIGlmIChodG1sTWVzaC5yZXF1aXJlc1VwZGF0ZSkge1xyXG4gICAgICAgICAgICB0aGlzLl91cGRhdGVCYXNlU2NhbGVGYWN0b3IoaHRtbE1lc2gpO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gR2V0IHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXggZm9yIHRoZSBodG1sIG1lc2hcclxuICAgICAgICBjb25zdCBzY2FsZWRBbmRUcmFuc2xhdGVkT2JqZWN0TWF0cml4ID0gdGhpcy5fZ2V0VHJhbnNmb3JtYXRpb25NYXRyaXgoaHRtbE1lc2gsIHVzZVJpZ2h0SGFuZGVkU3lzdGVtKTtcclxuXHJcbiAgICAgICAgbGV0IHN0eWxlID0gYHRyYW5zbGF0ZSgtNTAlLCAtNTAlKSAke3RoaXMuX2dldEh0bWxDb250ZW50Q1NTTWF0cml4KHNjYWxlZEFuZFRyYW5zbGF0ZWRPYmplY3RNYXRyaXgsIHVzZVJpZ2h0SGFuZGVkU3lzdGVtKX1gO1xyXG4gICAgICAgIC8vIEluIGEgcmlnaHQgaGFuZGVkIHN5c3RlbSwgc2NyZWVucyBhcmUgb24gdGhlIHdyb25nIHNpZGUgb2YgdGhlIG1lc2gsIHNvIHdlIGhhdmUgdG8gcm90YXRlIGJ5IE1hdGguUEkgd2hpY2ggcmVzdWx0cyBpbiB0aGUgbWF0cml4M2Qgc2VlbiBiZWxvd1xyXG4gICAgICAgIHN0eWxlICs9IGAke3VzZVJpZ2h0SGFuZGVkU3lzdGVtID8gXCJtYXRyaXgzZCgtMSwgMCwgMCwgMCwgMCwgMSwgMCwgMCwgMCwgMCwgLTEsIDAsIDAsIDAsIDAsIDEpXCIgOiBcIlwifWA7XHJcblxyXG4gICAgICAgIGlmIChodG1sTWVzaERhdGEuc3R5bGUgIT09IHN0eWxlKSB7XHJcbiAgICAgICAgICAgIGh0bWxNZXNoLmVsZW1lbnQuc3R5bGUud2Via2l0VHJhbnNmb3JtID0gc3R5bGU7XHJcbiAgICAgICAgICAgIGh0bWxNZXNoLmVsZW1lbnQuc3R5bGUudHJhbnNmb3JtID0gc3R5bGU7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBodG1sTWVzaC5fbWFya0FzVXBkYXRlZCgpO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfcmVuZGVyKHNjZW5lOiBTY2VuZSwgY2FtZXJhOiBDYW1lcmEpIHtcclxuICAgICAgICBsZXQgbmVlZHNVcGRhdGUgPSBmYWxzZTtcclxuXHJcbiAgICAgICAgY29uc3QgdXNlUmlnaHRIYW5kZWRTeXN0ZW0gPSBzY2VuZS51c2VSaWdodEhhbmRlZFN5c3RlbTtcclxuXHJcbiAgICAgICAgLy8gVXBkYXRlIHRoZSBjb250YWluZXIgcG9zaXRpb24gYW5kIHNpemUgaWYgbmVjZXNzYXJ5XHJcbiAgICAgICAgdGhpcy5fdXBkYXRlQ29udGFpbmVyUG9zaXRpb25JZk5lZWRlZCgpO1xyXG5cclxuICAgICAgICAvLyBDaGVjayBmb3IgYSBjYW1lcmEgY2hhbmdlXHJcbiAgICAgICAgaWYgKHRoaXMuX2NhbWVyYU1hdHJpeFVwZGF0ZWQpIHtcclxuICAgICAgICAgICAgdGhpcy5fY2FtZXJhTWF0cml4VXBkYXRlZCA9IGZhbHNlO1xyXG4gICAgICAgICAgICBuZWVkc1VwZGF0ZSA9IHRydWU7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBJZiB0aGUgY2FtZXJhIHBvc2l0aW9uIGhhcyBjaGFuZ2VkLCB0aGVuIHdlIGFsc28gbmVlZCB0byB1cGRhdGVcclxuICAgICAgICBpZiAoXHJcbiAgICAgICAgICAgIGNhbWVyYS5wb3NpdGlvbi54ICE9PSB0aGlzLl9jYWNoZS5jYW1lcmFEYXRhLnBvc2l0aW9uLnggfHxcclxuICAgICAgICAgICAgY2FtZXJhLnBvc2l0aW9uLnkgIT09IHRoaXMuX2NhY2hlLmNhbWVyYURhdGEucG9zaXRpb24ueSB8fFxyXG4gICAgICAgICAgICBjYW1lcmEucG9zaXRpb24ueiAhPT0gdGhpcy5fY2FjaGUuY2FtZXJhRGF0YS5wb3NpdGlvbi56XHJcbiAgICAgICAgKSB7XHJcbiAgICAgICAgICAgIHRoaXMuX2NhY2hlLmNhbWVyYURhdGEucG9zaXRpb24uY29weUZyb20oY2FtZXJhLnBvc2l0aW9uKTtcclxuICAgICAgICAgICAgbmVlZHNVcGRhdGUgPSB0cnVlO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gQ2hlY2sgZm9yIGEgZHByIGNoYW5nZVxyXG4gICAgICAgIGlmICh3aW5kb3cuZGV2aWNlUGl4ZWxSYXRpbyAhPT0gdGhpcy5fbGFzdERldmljZVBpeGVsUmF0aW8pIHtcclxuICAgICAgICAgICAgdGhpcy5fbGFzdERldmljZVBpeGVsUmF0aW8gPSB3aW5kb3cuZGV2aWNlUGl4ZWxSYXRpbztcclxuICAgICAgICAgICAgTG9nZ2VyLkxvZyhcIkluIHJlbmRlciAtIGRwciBjaGFuZ2VkOiBcIiwgdGhpcy5fbGFzdERldmljZVBpeGVsUmF0aW8pO1xyXG4gICAgICAgICAgICBuZWVkc1VwZGF0ZSA9IHRydWU7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBDaGVjayBpZiBhbnkgbWVzaGVzIG5lZWQgdG8gYmUgdXBkYXRlZFxyXG4gICAgICAgIGNvbnN0IG1lc2hlc05lZWRpbmdVcGRhdGUgPSBzY2VuZS5tZXNoZXMuZmlsdGVyKChtZXNoKSA9PiAobWVzaCBhcyBhbnkpW1wiaXNIdG1sTWVzaFwiXSAmJiAobmVlZHNVcGRhdGUgfHwgKG1lc2ggYXMgSHRtbE1lc2gpLnJlcXVpcmVzVXBkYXRlKSk7XHJcbiAgICAgICAgbmVlZHNVcGRhdGUgPSBuZWVkc1VwZGF0ZSB8fCBtZXNoZXNOZWVkaW5nVXBkYXRlLmxlbmd0aCA+IDA7XHJcblxyXG4gICAgICAgIGlmICghbmVlZHNVcGRhdGUpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gR2V0IGEgcHJvamVjdGlvbiBtYXRyaXggZm9yIHRoZSBjYW1lcmFcclxuICAgICAgICBjb25zdCBwcm9qZWN0aW9uTWF0cml4ID0gY2FtZXJhLmdldFByb2plY3Rpb25NYXRyaXgoKTtcclxuICAgICAgICBjb25zdCBmb3YgPSBwcm9qZWN0aW9uTWF0cml4Lm1bNV0gKiB0aGlzLl9oZWlnaHRIYWxmO1xyXG5cclxuICAgICAgICBpZiAodGhpcy5fY2FjaGUuY2FtZXJhRGF0YS5mb3YgIT09IGZvdikge1xyXG4gICAgICAgICAgICBpZiAoY2FtZXJhLm1vZGUgPT0gQ2FtZXJhLlBFUlNQRUNUSVZFX0NBTUVSQSkge1xyXG4gICAgICAgICAgICAgICAgW3RoaXMuX292ZXJsYXlFbGVtZW50cz8uZG9tRWxlbWVudCwgdGhpcy5faW5TY2VuZUVsZW1lbnRzPy5kb21FbGVtZW50XS5mb3JFYWNoKChlbCkgPT4ge1xyXG4gICAgICAgICAgICAgICAgICAgIGlmIChlbCkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBlbC5zdHlsZS53ZWJraXRQZXJzcGVjdGl2ZSA9IGZvdiArIFwicHhcIjtcclxuICAgICAgICAgICAgICAgICAgICAgICAgZWwuc3R5bGUucGVyc3BlY3RpdmUgPSBmb3YgKyBcInB4XCI7XHJcbiAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBbdGhpcy5fb3ZlcmxheUVsZW1lbnRzPy5kb21FbGVtZW50LCB0aGlzLl9pblNjZW5lRWxlbWVudHM/LmRvbUVsZW1lbnRdLmZvckVhY2goKGVsKSA9PiB7XHJcbiAgICAgICAgICAgICAgICAgICAgaWYgKGVsKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIGVsLnN0eWxlLndlYmtpdFBlcnNwZWN0aXZlID0gXCJcIjtcclxuICAgICAgICAgICAgICAgICAgICAgICAgZWwuc3R5bGUucGVyc3BlY3RpdmUgPSBcIlwiO1xyXG4gICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgIH0pO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIHRoaXMuX2NhY2hlLmNhbWVyYURhdGEuZm92ID0gZm92O1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gR2V0IHRoZSBDU1MgbWF0cml4IGZvciB0aGUgY2FtZXJhICh3aGljaCB3aWxsIGluY2x1ZGUgYW55IGNhbWVyYSByb3RhdGlvbilcclxuICAgICAgICBpZiAoY2FtZXJhLnBhcmVudCA9PT0gbnVsbCkge1xyXG4gICAgICAgICAgICBjYW1lcmEuY29tcHV0ZVdvcmxkTWF0cml4KCk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBjb25zdCBjYW1lcmFNYXRyaXhXb3JsZCA9IHRoaXMuX3RlbXAuY2FtZXJhV29ybGRNYXRyaXg7XHJcbiAgICAgICAgY2FtZXJhTWF0cml4V29ybGQuY29weUZyb20oY2FtZXJhLmdldFdvcmxkTWF0cml4KCkpO1xyXG4gICAgICAgIGNvbnN0IGNhbWVyYVJvdGF0aW9uTWF0cml4ID0gdGhpcy5fdGVtcC5jYW1lcmFSb3RhdGlvbk1hdHJpeDtcclxuICAgICAgICBjYW1lcmFNYXRyaXhXb3JsZC5nZXRSb3RhdGlvbk1hdHJpeCgpLnRyYW5zcG9zZVRvUmVmKGNhbWVyYVJvdGF0aW9uTWF0cml4KTtcclxuXHJcbiAgICAgICAgY29uc3QgY2FtZXJhTWF0cml4V29ybGRBc0FycmF5ID0gdGhpcy5fdGVtcC5jYW1lcmFXb3JsZE1hdHJpeEFzQXJyYXk7XHJcbiAgICAgICAgY2FtZXJhTWF0cml4V29ybGQuY29weVRvQXJyYXkoY2FtZXJhTWF0cml4V29ybGRBc0FycmF5KTtcclxuXHJcbiAgICAgICAgLy8gRm9yIGEgZmV3IHZhbHVlcywgd2UgaGF2ZSB0byBhZGp1c3QgdGhlIGRpcmVjdGlvbiBiYXNlZCBvbiB0aGUgaGFuZGVkbmVzcyBvZiB0aGUgc3lzdGVtXHJcbiAgICAgICAgY29uc3QgZGlyZWN0aW9uID0gdXNlUmlnaHRIYW5kZWRTeXN0ZW0gPyAxIDogLTE7XHJcblxyXG4gICAgICAgIGNhbWVyYU1hdHJpeFdvcmxkQXNBcnJheVsxXSA9IGNhbWVyYVJvdGF0aW9uTWF0cml4Lm1bMV07XHJcbiAgICAgICAgY2FtZXJhTWF0cml4V29ybGRBc0FycmF5WzJdID0gY2FtZXJhUm90YXRpb25NYXRyaXgubVsyXSAqIGRpcmVjdGlvbjtcclxuICAgICAgICBjYW1lcmFNYXRyaXhXb3JsZEFzQXJyYXlbNF0gPSBjYW1lcmFSb3RhdGlvbk1hdHJpeC5tWzRdICogZGlyZWN0aW9uO1xyXG4gICAgICAgIGNhbWVyYU1hdHJpeFdvcmxkQXNBcnJheVs2XSA9IGNhbWVyYVJvdGF0aW9uTWF0cml4Lm1bNl0gKiBkaXJlY3Rpb247XHJcbiAgICAgICAgY2FtZXJhTWF0cml4V29ybGRBc0FycmF5WzhdID0gY2FtZXJhUm90YXRpb25NYXRyaXgubVs4XSAqIGRpcmVjdGlvbjtcclxuICAgICAgICBjYW1lcmFNYXRyaXhXb3JsZEFzQXJyYXlbOV0gPSBjYW1lcmFSb3RhdGlvbk1hdHJpeC5tWzldICogZGlyZWN0aW9uO1xyXG5cclxuICAgICAgICBNYXRyaXguRnJvbUFycmF5VG9SZWYoY2FtZXJhTWF0cml4V29ybGRBc0FycmF5LCAwLCBjYW1lcmFNYXRyaXhXb3JsZCk7XHJcblxyXG4gICAgICAgIGNvbnN0IGNhbWVyYUNTU01hdHJpeCA9IHRoaXMuX2dldENhbWVyYUNTU01hdHJpeChjYW1lcmFNYXRyaXhXb3JsZCk7XHJcbiAgICAgICAgY29uc3Qgc3R5bGUgPSBjYW1lcmFDU1NNYXRyaXg7XHJcblxyXG4gICAgICAgIGlmICh0aGlzLl9jYWNoZS5jYW1lcmFEYXRhLnN0eWxlICE9PSBzdHlsZSkge1xyXG4gICAgICAgICAgICBbdGhpcy5faW5TY2VuZUVsZW1lbnRzPy5jYW1lcmFFbGVtZW50LCB0aGlzLl9vdmVybGF5RWxlbWVudHM/LmNhbWVyYUVsZW1lbnRdLmZvckVhY2goKGVsKSA9PiB7XHJcbiAgICAgICAgICAgICAgICBpZiAoZWwpIHtcclxuICAgICAgICAgICAgICAgICAgICBlbC5zdHlsZS53ZWJraXRUcmFuc2Zvcm0gPSBzdHlsZTtcclxuICAgICAgICAgICAgICAgICAgICBlbC5zdHlsZS50cmFuc2Zvcm0gPSBzdHlsZTtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgICAgIHRoaXMuX2NhY2hlLmNhbWVyYURhdGEuc3R5bGUgPSBzdHlsZTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIF9SZW5kZXIgb2JqZWN0cyBpZiBuZWNlc3NhcnlcclxuICAgICAgICBtZXNoZXNOZWVkaW5nVXBkYXRlLmZvckVhY2goKG1lc2gpID0+IHtcclxuICAgICAgICAgICAgdGhpcy5fcmVuZGVySHRtbE1lc2gobWVzaCBhcyBIdG1sTWVzaCwgdXNlUmlnaHRIYW5kZWRTeXN0ZW0pO1xyXG4gICAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfdXBkYXRlQmFzZVNjYWxlRmFjdG9yKGh0bWxNZXNoOiBIdG1sTWVzaCkge1xyXG4gICAgICAgIC8vIEdldCBzY3JlZW4gd2lkdGggYW5kIGhlaWdodFxyXG4gICAgICAgIGxldCBzY3JlZW5XaWR0aCA9IHRoaXMuX3dpZHRoO1xyXG4gICAgICAgIGxldCBzY3JlZW5IZWlnaHQgPSB0aGlzLl9oZWlnaHQ7XHJcblxyXG4gICAgICAgIC8vIENhbGN1bGF0ZSBhc3BlY3QgcmF0aW9zXHJcbiAgICAgICAgY29uc3QgaHRtbE1lc2hBc3BlY3RSYXRpbyA9IChodG1sTWVzaC53aWR0aCB8fCAxKSAvIChodG1sTWVzaC5oZWlnaHQgfHwgMSk7XHJcbiAgICAgICAgY29uc3Qgc2NyZWVuQXNwZWN0UmF0aW8gPSBzY3JlZW5XaWR0aCAvIHNjcmVlbkhlaWdodDtcclxuXHJcbiAgICAgICAgLy8gQWRqdXN0IHNjcmVlbiBkaW1lbnNpb25zIGJhc2VkIG9uIGFzcGVjdCByYXRpb3NcclxuICAgICAgICBpZiAoaHRtbE1lc2hBc3BlY3RSYXRpbyA+IHNjcmVlbkFzcGVjdFJhdGlvKSB7XHJcbiAgICAgICAgICAgIC8vIElmIHRoZSBIVE1MIG1lc2ggaXMgd2lkZXIgcmVsYXRpdmUgdG8gaXRzIGhlaWdodCB0aGFuIHRoZSBzY3JlZW4sIGFkanVzdCB0aGUgc2NyZWVuIHdpZHRoXHJcbiAgICAgICAgICAgIHNjcmVlbldpZHRoID0gc2NyZWVuSGVpZ2h0ICogaHRtbE1lc2hBc3BlY3RSYXRpbztcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAvLyBJZiB0aGUgSFRNTCBtZXNoIGlzIHRhbGxlciByZWxhdGl2ZSB0byBpdHMgd2lkdGggdGhhbiB0aGUgc2NyZWVuLCBhZGp1c3QgdGhlIHNjcmVlbiBoZWlnaHRcclxuICAgICAgICAgICAgc2NyZWVuSGVpZ2h0ID0gc2NyZWVuV2lkdGggLyBodG1sTWVzaEFzcGVjdFJhdGlvO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gU2V0IGNvbnRlbnQgdG8gZmlsbCBzY3JlZW4gc28gd2UgZ2V0IG1heCByZXNvbHV0aW9uIHdoZW4gaXQgaXMgc2hydW5rIHRvIGZpdCB0aGUgbWVzaFxyXG4gICAgICAgIGh0bWxNZXNoLnNldENvbnRlbnRTaXplUHgoc2NyZWVuV2lkdGgsIHNjcmVlbkhlaWdodCk7XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF91cGRhdGVDb250YWluZXJQb3NpdGlvbklmTmVlZGVkKCkge1xyXG4gICAgICAgIC8vIERldGVybWluZSBpZiB0aGUgY2FudmFzIGhhcyBtb3ZlZCBvbiB0aGUgc2NyZWVuXHJcbiAgICAgICAgY29uc3QgY2FudmFzUmVjdCA9IHRoaXMuX2VuZ2luZS5nZXRSZW5kZXJpbmdDYW52YXNDbGllbnRSZWN0KCk7XHJcblxyXG4gICAgICAgIC8vIGNhbnZhcyByZWN0IG1heSBiZSBudWxsIGlmIGxheW91dCBub3QgY29tcGxldGVcclxuICAgICAgICBpZiAoIWNhbnZhc1JlY3QpIHtcclxuICAgICAgICAgICAgTG9nZ2VyLldhcm4oX3Bvc2l0aW9uVXBkYXRlRmFpbE1lc3NhZ2UpO1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGNvbnN0IHNjcm9sbFRvcCA9IHdpbmRvdy5zY3JvbGxZO1xyXG4gICAgICAgIGNvbnN0IHNjcm9sbExlZnQgPSB3aW5kb3cuc2Nyb2xsWDtcclxuICAgICAgICBjb25zdCBjYW52YXNEb2N1bWVudFRvcCA9IGNhbnZhc1JlY3QudG9wICsgc2Nyb2xsVG9wO1xyXG4gICAgICAgIGNvbnN0IGNhbnZhc0RvY3VtZW50TGVmdCA9IGNhbnZhc1JlY3QubGVmdCArIHNjcm9sbExlZnQ7XHJcblxyXG4gICAgICAgIGlmICh0aGlzLl9wcmV2aW91c0NhbnZhc0RvY3VtZW50UG9zaXRpb24udG9wICE9PSBjYW52YXNEb2N1bWVudFRvcCB8fCB0aGlzLl9wcmV2aW91c0NhbnZhc0RvY3VtZW50UG9zaXRpb24ubGVmdCAhPT0gY2FudmFzRG9jdW1lbnRMZWZ0KSB7XHJcbiAgICAgICAgICAgIHRoaXMuX3ByZXZpb3VzQ2FudmFzRG9jdW1lbnRQb3NpdGlvbi50b3AgPSBjYW52YXNEb2N1bWVudFRvcDtcclxuICAgICAgICAgICAgdGhpcy5fcHJldmlvdXNDYW52YXNEb2N1bWVudFBvc2l0aW9uLmxlZnQgPSBjYW52YXNEb2N1bWVudExlZnQ7XHJcblxyXG4gICAgICAgICAgICBbdGhpcy5faW5TY2VuZUVsZW1lbnRzPy5jb250YWluZXIsIHRoaXMuX292ZXJsYXlFbGVtZW50cz8uY29udGFpbmVyXS5mb3JFYWNoKChjb250YWluZXIpID0+IHtcclxuICAgICAgICAgICAgICAgIGlmICghY29udGFpbmVyKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgLy8gc2V0IHRoZSB0b3AgYW5kIGxlZnQgb2YgdGhlIGNzcyBjb250YWluZXIgdG8gbWF0Y2ggdGhlIGNhbnZhc1xyXG4gICAgICAgICAgICAgICAgY29uc3QgY29udGFpbmVyUGFyZW50ID0gY29udGFpbmVyLm9mZnNldFBhcmVudCBhcyBIVE1MRWxlbWVudDtcclxuICAgICAgICAgICAgICAgIGNvbnN0IHBhcmVudFJlY3QgPSBjb250YWluZXJQYXJlbnQuZ2V0Qm91bmRpbmdDbGllbnRSZWN0KCk7XHJcbiAgICAgICAgICAgICAgICBjb25zdCBwYXJlbnREb2N1bWVudFRvcCA9IHBhcmVudFJlY3QudG9wICsgc2Nyb2xsVG9wO1xyXG4gICAgICAgICAgICAgICAgY29uc3QgcGFyZW50RG9jdW1lbnRMZWZ0ID0gcGFyZW50UmVjdC5sZWZ0ICsgc2Nyb2xsTGVmdDtcclxuXHJcbiAgICAgICAgICAgICAgICBjb25zdCBhbmNlc3Rvck1hcmdpbnNBbmRQYWRkaW5nID0gdGhpcy5fZ2V0QW5jZXN0b3JNYXJnaW5zQW5kUGFkZGluZyhjb250YWluZXJQYXJlbnQpO1xyXG5cclxuICAgICAgICAgICAgICAgIC8vIEFkZCB0aGUgYm9keSBtYXJnaW5cclxuICAgICAgICAgICAgICAgIGNvbnN0IGJvZHlTdHlsZSA9IHdpbmRvdy5nZXRDb21wdXRlZFN0eWxlKGRvY3VtZW50LmJvZHkpO1xyXG4gICAgICAgICAgICAgICAgY29uc3QgYm9keU1hcmdpblRvcCA9IHBhcnNlSW50KGJvZHlTdHlsZS5tYXJnaW5Ub3AsIDEwKTtcclxuICAgICAgICAgICAgICAgIGNvbnN0IGJvZHlNYXJnaW5MZWZ0ID0gcGFyc2VJbnQoYm9keVN0eWxlLm1hcmdpbkxlZnQsIDEwKTtcclxuXHJcbiAgICAgICAgICAgICAgICBjb250YWluZXIuc3R5bGUudG9wID0gYCR7Y2FudmFzRG9jdW1lbnRUb3AgLSBwYXJlbnREb2N1bWVudFRvcCAtIGFuY2VzdG9yTWFyZ2luc0FuZFBhZGRpbmcubWFyZ2luVG9wICsgYW5jZXN0b3JNYXJnaW5zQW5kUGFkZGluZy5wYWRkaW5nVG9wICsgYm9keU1hcmdpblRvcH1weGA7XHJcbiAgICAgICAgICAgICAgICBjb250YWluZXIuc3R5bGUubGVmdCA9IGAke1xyXG4gICAgICAgICAgICAgICAgICAgIGNhbnZhc0RvY3VtZW50TGVmdCAtIHBhcmVudERvY3VtZW50TGVmdCAtIGFuY2VzdG9yTWFyZ2luc0FuZFBhZGRpbmcubWFyZ2luTGVmdCArIGFuY2VzdG9yTWFyZ2luc0FuZFBhZGRpbmcucGFkZGluZ0xlZnQgKyBib2R5TWFyZ2luTGVmdFxyXG4gICAgICAgICAgICAgICAgfXB4YDtcclxuICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfb25DYW1lcmFNYXRyaXhDaGFuZ2VkID0gKGNhbWVyYTogQ2FtZXJhKSA9PiB7XHJcbiAgICAgICAgdGhpcy5fY2FtZXJhV29ybGRNYXRyaXggPSBjYW1lcmEuZ2V0V29ybGRNYXRyaXgoKTtcclxuICAgICAgICB0aGlzLl9jYW1lcmFNYXRyaXhVcGRhdGVkID0gdHJ1ZTtcclxuICAgIH07XHJcblxyXG4gICAgcHJpdmF0ZSBfZXBzaWxvbih2YWx1ZTogbnVtYmVyKSB7XHJcbiAgICAgICAgcmV0dXJuIE1hdGguYWJzKHZhbHVlKSA8IDFlLTEwID8gMCA6IHZhbHVlO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIEdldCB0b3RhbCBtYXJnaW5zIGFuZCBwYWRkaW5nIGZvciBhbiBlbGVtZW50LCBleGNsdWRpbmcgdGhlIGJvZHkgYW5kIGRvY3VtZW50IG1hcmdpbnNcclxuICAgIHByaXZhdGUgX2dldEFuY2VzdG9yTWFyZ2luc0FuZFBhZGRpbmcoZWxlbWVudDogSFRNTEVsZW1lbnQpIHtcclxuICAgICAgICBsZXQgbWFyZ2luVG9wID0gMDtcclxuICAgICAgICBsZXQgbWFyZ2luTGVmdCA9IDA7XHJcbiAgICAgICAgbGV0IHBhZGRpbmdUb3AgPSAwO1xyXG4gICAgICAgIGxldCBwYWRkaW5nTGVmdCA9IDA7XHJcblxyXG4gICAgICAgIHdoaWxlIChlbGVtZW50ICYmIGVsZW1lbnQgIT09IGRvY3VtZW50LmJvZHkgJiYgZWxlbWVudCAhPT0gZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50KSB7XHJcbiAgICAgICAgICAgIGNvbnN0IHN0eWxlID0gd2luZG93LmdldENvbXB1dGVkU3R5bGUoZWxlbWVudCk7XHJcbiAgICAgICAgICAgIG1hcmdpblRvcCArPSBwYXJzZUludChzdHlsZS5tYXJnaW5Ub3AsIDEwKTtcclxuICAgICAgICAgICAgbWFyZ2luTGVmdCArPSBwYXJzZUludChzdHlsZS5tYXJnaW5MZWZ0LCAxMCk7XHJcbiAgICAgICAgICAgIHBhZGRpbmdUb3AgKz0gcGFyc2VJbnQoc3R5bGUucGFkZGluZ1RvcCwgMTApO1xyXG4gICAgICAgICAgICBwYWRkaW5nTGVmdCArPSBwYXJzZUludChzdHlsZS5wYWRkaW5nTGVmdCwgMTApO1xyXG4gICAgICAgICAgICBlbGVtZW50ID0gZWxlbWVudC5vZmZzZXRQYXJlbnQgYXMgSFRNTEVsZW1lbnQ7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICByZXR1cm4geyBtYXJnaW5Ub3AsIG1hcmdpbkxlZnQsIHBhZGRpbmdUb3AsIHBhZGRpbmdMZWZ0IH07XHJcbiAgICB9XHJcbn1cclxuIiwiaW1wb3J0IHsgSHRtbE1lc2hSZW5kZXJlciB9IGZyb20gXCIuL2h0bWxNZXNoUmVuZGVyZXJcIjtcclxuaW1wb3J0IHsgSHRtbE1lc2ggfSBmcm9tIFwiLi9odG1sTWVzaFwiO1xyXG5pbXBvcnQgeyBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yIH0gZnJvbSBcIi4vcG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvclwiO1xyXG5pbXBvcnQgeyBGaXRTdHJhdGVneSB9IGZyb20gXCIuL2ZpdFN0cmF0ZWd5XCI7XHJcblxyXG4vLyBFeHBvcnQgcHVibGljIGNsYXNzZXMgYW5kIGZ1bmN0aW9uc1xyXG5leHBvcnQgeyBIdG1sTWVzaFJlbmRlcmVyLCBIdG1sTWVzaCwgUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvciwgRml0U3RyYXRlZ3kgfTtcclxuIiwiaW1wb3J0IHsgVG9vbHMgfSBmcm9tIFwiY29yZS9NaXNjL3Rvb2xzXCI7XHJcblxyXG4vLyBBIGNhcHR1cmUgbWFuYWdlbWVudCBzeXN0ZW0gdG8gZW5zdXJlIHRoYXQgdGhlIGNvcnJlY3Qgb2JqZWN0IGhhcyB0aGUgcG9pbnRlclxyXG4vLyBldmVudHMgYnkgZWxpbWluYXRpbmcgcmFjZSBjb25kaXRpb25zIHRoYXQgY2FuIGNhdXNlIHRoZSBwb2ludGVyIGV2ZW50cyB0byBiZVxyXG4vLyByZWxlYXNlZCBieSBhIGRpZmZlcmVudCBvYmplY3QgYWZ0ZXIgdGhleSBhcmUgY2FwdHVyZWQgbGVhdmluZyBubyBvYmplY3RcclxuLy8gYXMgdGhlIG93bmVyLiAgSXQgZG9lcyB0aGlzIGJ5IHF1ZXVlaW5nIHJlcXVlc3RzIGFuZCBvbmx5IGFsbG93aW5nXHJcbi8vIGNhcHR1cmUgd2hlbiB0aGUgY3VycmVudCBjYXB0dXJlIG93bmVyIHJlbGVhc2VzIHBvaW50ZXIgZXZlbnRzLlxyXG5cclxudHlwZSBDYXB0dXJlUmVsZWFzZUNhbGxiYWNrID0gKCkgPT4gdm9pZDtcclxuXHJcbnR5cGUgQ2FwdHVyZVJlbGVhc2VDYWxsYmFja3MgPSB7XHJcbiAgICBjYXB0dXJlOiBDYXB0dXJlUmVsZWFzZUNhbGxiYWNrO1xyXG4gICAgcmVsZWFzZTogQ2FwdHVyZVJlbGVhc2VDYWxsYmFjaztcclxufTtcclxuXHJcbmxldCBjYXB0dXJlUmVxdWVzdFF1ZXVlOiBzdHJpbmdbXSA9IFtdO1xyXG5cclxuLy8gS2V5IGlzIHJlcXVlc3QgaWQsIHZhbHVlIGlzIG9iamVjdCB3aXRoIGNhcHR1cmUgYW5kIHJlbGVhc2UgY2FsbGJhY2tzXHJcbmNvbnN0IHBlbmRpbmdSZXF1ZXN0Q2FsbGJhY2tzOiBNYXA8c3RyaW5nLCBDYXB0dXJlUmVsZWFzZUNhbGxiYWNrcz4gPSBuZXcgTWFwKCk7XHJcblxyXG4vLyBLZWVwIHRyYWNrIG9mIHJlbGVhc2UgcmVxdWVzdHMgd2l0aCBubyBtYXRjaGluZyBjYXB0dXJlIHJlcXVlc3RcclxuLy8gaW4gY2FzZSB0aGUgcmVsZWFzZSByZXF1ZXN0IGFycml2ZWQgYmVmb3JlIHRoZSBjYXB0dXJlIHRvIGF2b2lkXHJcbi8vIHRoZSBjYXB0dXJlIHJlcXVlc3QgbmV2ZXIgZ2V0dGluZyByZWxlYXNlZC5cclxubGV0IHVubWF0Y2hlZFJlbGVhc2VSZXF1ZXN0czogc3RyaW5nW10gPSBbXTtcclxuXHJcbmxldCBjdXJyZW50T3duZXI6IHN0cmluZyB8IG51bGwgPSBudWxsOyAvLyBDYWxsZWQgb24gZmlyc3QgY2FwdHVyZSBvciByZWxlYXNlIHJlcXVlc3RcclxuXHJcbi8qKlxyXG4gKiBHZXQgdGhlIGlkIG9mIHRoZSBvYmplY3QgY3VycmVudGx5IGNhcHR1cmluZyBwb2ludGVyIGV2ZW50c1xyXG4gKiBAcmV0dXJucyBUaGUgaWQgb2YgdGhlIG9iamVjdCBjdXJyZW50bHkgY2FwdHVyaW5nIHBvaW50ZXIgZXZlbnRzXHJcbiAqIG9yIG51bGwgaWYgbm8gb2JqZWN0IGlzIGNhcHR1cmluZyBwb2ludGVyIGV2ZW50c1xyXG4gKi9cclxuZXhwb3J0IGNvbnN0IGdldENhcHR1cmluZ0lkID0gKCkgPT4ge1xyXG4gICAgcmV0dXJuIGN1cnJlbnRPd25lcjtcclxufTtcclxuXHJcbi8qKlxyXG4gKiBSZXF1ZXN0IHRoYXQgdGhlIG9iamVjdCB3aXRoIHRoZSBnaXZlbiBpZCBjYXB0dXJlIHBvaW50ZXIgZXZlbnRzLiAgSWYgdGhlcmUgaXMgbm8gY3VycmVudFxyXG4gKiBvd25lciwgdGhlbiB0aGUgcmVxdWVzdCBpcyBncmFudGVkIGltbWVkaWF0ZWx5LiAgSWYgdGhlcmUgaXMgYSBjdXJyZW50IG93bmVyLCB0aGVuIHRoZSByZXF1ZXN0XHJcbiAqIGlzIHF1ZXVlZCB1bnRpbCB0aGUgY3VycmVudCBvd25lciByZWxlYXNlcyBwb2ludGVyIGV2ZW50cy5cclxuICogQHBhcmFtIHJlcXVlc3RJZCBBbiBpZCB0byBpZGVudGlmeSB0aGUgcmVxdWVzdC4gIFRoaXMgaWQgd2lsbCBiZSB1c2VkIHRvIG1hdGNoIHRoZSBjYXB0dXJlXHJcbiAqIHJlcXVlc3Qgd2l0aCB0aGUgcmVsZWFzZSByZXF1ZXN0LlxyXG4gKiBAcGFyYW0gY2FwdHVyZUNhbGxiYWNrIFRoZSBjYWxsYmFjayB0byBjYWxsIHdoZW4gdGhlIHJlcXVlc3QgaXMgZ3JhbnRlZCBhbmQgdGhlIG9iamVjdCBpcyBjYXB0dXJpbmdcclxuICogQHBhcmFtIHJlbGVhc2VDYWxsYmFjayBUaGUgY2FsbGJhY2sgdG8gY2FsbCB3aGVuIHRoZSBvYmplY3QgaXMgbm8gbG9uZ2VyIGNhcHR1cmluZyBwb2ludGVyIGV2ZW50c1xyXG4gKi9cclxuZXhwb3J0IGNvbnN0IHJlcXVlc3RDYXB0dXJlID0gKHJlcXVlc3RJZDogc3RyaW5nLCBjYXB0dXJlQ2FsbGJhY2s6IENhcHR1cmVSZWxlYXNlQ2FsbGJhY2ssIHJlbGVhc2VDYWxsYmFjazogQ2FwdHVyZVJlbGVhc2VDYWxsYmFjaykgPT4ge1xyXG4gICAgZGVidWdMb2coYEluIHBvaW50ZXJFdmVudHNDYXB0dXJlLnJlcXVlc3RDYXB0dXJlIC0gUG9pbnRlciBldmVudHMgY2FwdHVyZSByZXF1ZXN0ZWQgZm9yICR7cmVxdWVzdElkfWApO1xyXG5cclxuICAgIC8vIElmIHRoZXJlIGlzIGEgcmVsZWFzZSBmb3IgdGhpcyByZXF1ZXN0LCB0aGVuIGlnbm9yZSB0aGUgcmVxdWVzdFxyXG4gICAgaWYgKHJlbW92ZVVubWF0Y2hlZFJlcXVlc3QocmVxdWVzdElkKSkge1xyXG4gICAgICAgIGRlYnVnTG9nKGBJbiBwb2ludGVyRXZlbnRzQ2FwdHVyZS5yZXF1ZXN0Q2FwdHVyZSAtIENhcHR1cmUgcmVxdWVzdCBtYXRjaGVkIHByZXZpb3VzIHJlbGVhc2UgcmVxdWVzdCAke3JlcXVlc3RJZH0uICBDYW5jZWxsaW5nIGNhcHR1cmUgcmVxdWVzdGApO1xyXG4gICAgICAgIHJldHVybjtcclxuICAgIH0gZWxzZSBpZiAocmVxdWVzdElkICE9PSBjdXJyZW50T3duZXIpIHtcclxuICAgICAgICAvLyBpZiB0aGUgcmVxdWVzdCBpcyBub3QgYWxyZWFkeSBpbiB0aGUgcXVldWUsIGFkZCBpdCB0byB0aGUgcXVldWVcclxuICAgICAgICBlbnF1ZXVlQ2FwdHVyZVJlcXVlc3QocmVxdWVzdElkLCBjYXB0dXJlQ2FsbGJhY2ssIHJlbGVhc2VDYWxsYmFjayk7XHJcbiAgICB9XHJcblxyXG4gICAgaWYgKCFjdXJyZW50T3duZXIpIHtcclxuICAgICAgICAvLyBJZiB0aGVyZSBpcyBubyBjdXJyZW50IG93bmVyLCBnbyBhaGVhZCBhbmQgZ3JhbnQgdGhlIHJlcXVlc3RcclxuICAgICAgICB0cmFuc2ZlclBvaW50ZXJFdmVudHNPd25lcnNoaXAoKTtcclxuICAgIH1cclxuICAgIC8vIElmIHRoZSByZXF1ZXN0IGlkIGlzIHRoZSBjdXJyZW50IG93bmVyLCBkbyBub3RoaW5nXHJcbn07XHJcblxyXG4vKipcclxuICogUmVsZWFzZSBwb2ludGVyIGV2ZW50cyBmcm9tIHRoZSBvYmplY3Qgd2l0aCB0aGUgZ2l2ZW4gaWQuICBJZiB0aGUgb2JqZWN0IGlzIHRoZSBjdXJyZW50IG93bmVyXHJcbiAqIHRoZW4gcG9pbnRlciBldmVudHMgYXJlIHJlbGVhc2VkIGltbWVkaWF0ZWx5LiAgSWYgdGhlIG9iamVjdCBpcyBub3QgdGhlIGN1cnJlbnQgb3duZXIsIHRoZW4gdGhlXHJcbiAqIGFzc29jaWF0ZWQgY2FwdHVyZSByZXF1ZXN0IGlzIHJlbW92ZWQgZnJvbSB0aGUgcXVldWUuICBJZiB0aGVyZSBpcyBubyBtYXRjaGluZyBjYXB0dXJlIHJlcXVlc3RcclxuICogaW4gdGhlIHF1ZXVlLCB0aGVuIHRoZSByZWxlYXNlIHJlcXVlc3QgaXMgYWRkZWQgdG8gYSBsaXN0IG9mIHVubWF0Y2hlZCByZWxlYXNlIHJlcXVlc3RzIGFuZCB3aWxsXHJcbiAqIG5lZ2F0ZSB0aGUgbmV4dCBjYXB0dXJlIHJlcXVlc3Qgd2l0aCB0aGUgc2FtZSBpZC4gIFRoaXMgaXMgdG8gZ3VhcmQgYWdhaW5zdCB0aGUgcG9zc2liaWxpdHkgdGhhdFxyXG4gKiB0aGUgcmVsZWFzZSByZXF1ZXN0IGFycml2ZWQgYmVmb3JlIHRoZSBjYXB0dXJlIHJlcXVlc3QuXHJcbiAqIEBwYXJhbSByZXF1ZXN0SWQgVGhlIGlkIHdoaWNoIHNob3VsZCBtYXRjaCB0aGUgaWQgb2YgdGhlIGNhcHR1cmUgcmVxdWVzdFxyXG4gKi9cclxuZXhwb3J0IGNvbnN0IHJlcXVlc3RSZWxlYXNlID0gKHJlcXVlc3RJZDogc3RyaW5nIHwgbnVsbCkgPT4ge1xyXG4gICAgZGVidWdMb2coYEluIHBvaW50ZXJFdmVudHNDYXB0dXJlLnJlcXVlc3RSZWxlYXNlIC0gUG9pbnRlciBldmVudHMgcmVsZWFzZSByZXF1ZXN0ZWQgZm9yICR7cmVxdWVzdElkfWApO1xyXG5cclxuICAgIC8vIGlmIHRoZSByZXF1ZXN0SWQgaXMgdGhlIGN1cnJlbnQgY2FwdHVyZSBob2xkZXIgcmVsZWFzZSBpdFxyXG4gICAgaWYgKCFyZXF1ZXN0SWQgfHwgcmVxdWVzdElkID09PSBjdXJyZW50T3duZXIpIHtcclxuICAgICAgICB0cmFuc2ZlclBvaW50ZXJFdmVudHNPd25lcnNoaXAoKTtcclxuICAgIH0gZWxzZSBpZiAoY2FuY2VsUmVxdWVzdChyZXF1ZXN0SWQpKSB7XHJcbiAgICAgICAgLy8gaWYgdGhlIHJlcXVlc3QgaXMgaW4gdGhlIHF1ZXVlLCBidXQgbm90IHRoZSBjdXJyZW50IGNhcHR1cmUgaG9sZGVyLCByZW1vdmUgaXQgYW5kIGl0J3MgY2FsbGJhY2tzXHJcbiAgICAgICAgcGVuZGluZ1JlcXVlc3RDYWxsYmFja3MuZGVsZXRlKHJlcXVlc3RJZCk7XHJcbiAgICB9IGVsc2Uge1xyXG4gICAgICAgIGRlYnVnTG9nKGBJbiBwb2ludGVyRXZlbnRzQ2FwdHVyZS5yZXF1ZXN0UmVsZWFzZSAtIFJlY2VpdmVkIHJlbGVhc2UgcmVxdWVzdCAke3JlcXVlc3RJZH0gYnV0IG5vIG1hdGNoaW5nIGNhcHR1cmUgcmVxdWVzdCB3YXMgcmVjZWl2ZWRgKTtcclxuICAgICAgICAvLyByZXF1ZXN0IHdhcyBub3QgY3VycmVudCBhbmQgbm90IGluIHF1ZXVlLCBsaWtlbHkgYmVjYXVzZSB3ZSByZWNlaXZlZCBhIHJlbGVhc2VcclxuICAgICAgICAvLyByZXF1ZXN0IGJlZm9yZSB0aGUgY2FwdHVyZS4gIEFkZCBpdCB0byB0aGUgdW5tYXRjaGVkIGxpc3QgdG8gZ3VhcmQgYWdhaW5zdCB0aGlzIHBvc3NpYmlsaXR5XHJcbiAgICAgICAgaWYgKCF1bm1hdGNoZWRSZWxlYXNlUmVxdWVzdHMuaW5jbHVkZXMocmVxdWVzdElkKSkge1xyXG4gICAgICAgICAgICB1bm1hdGNoZWRSZWxlYXNlUmVxdWVzdHMucHVzaChyZXF1ZXN0SWQpO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxufTtcclxuXHJcbi8qKlxyXG4gKiBSZWxhc2UgcG9pbnRlciBldmVudHMgZnJvbSB0aGUgY3VycmVudCBvd25lclxyXG4gKi9cclxuZXhwb3J0IGNvbnN0IHJlbGVhc2VDdXJyZW50ID0gKCkgPT4ge1xyXG4gICAgcmVxdWVzdFJlbGVhc2UoY3VycmVudE93bmVyKTtcclxufTtcclxuXHJcbmNvbnN0IGVucXVldWVDYXB0dXJlUmVxdWVzdCA9IChyZXF1ZXN0SWQ6IHN0cmluZywgY2FwdHVyZTogQ2FwdHVyZVJlbGVhc2VDYWxsYmFjaywgcmVsZWFzZTogQ2FwdHVyZVJlbGVhc2VDYWxsYmFjaykgPT4ge1xyXG4gICAgZGVidWdMb2coYEluIHBvaW50ZXJFdmVudHNDYXB0dXJlLmVucXVldWVDYXB0dXJlUmVxdWVzdCAtIEVucXVldWVpbmcgY2FwdHVyZSByZXF1ZXN0IGZvciAgJHtyZXF1ZXN0SWR9YCk7XHJcbiAgICBpZiAoIWNhcHR1cmVSZXF1ZXN0UXVldWUuaW5jbHVkZXMocmVxdWVzdElkKSkge1xyXG4gICAgICAgIGNhcHR1cmVSZXF1ZXN0UXVldWUucHVzaChyZXF1ZXN0SWQpO1xyXG4gICAgICAgIHBlbmRpbmdSZXF1ZXN0Q2FsbGJhY2tzLnNldChyZXF1ZXN0SWQsIHsgY2FwdHVyZSwgcmVsZWFzZSB9KTtcclxuICAgIH1cclxufTtcclxuXHJcbi8vIFJlbW92ZXMgdGhlIHJlcXVlc3QgZnJvbSB0aGUgcXVldWUgaWYgaXQgZXhpc3RzLiAgUmV0dXJucyB0cnVlXHJcbi8vIGlmIHRoZSByZXF1ZXN0IHdhcyBmb3VuZCBhbmQgcmVtb3ZlZCwgb3RoZXJ3aXNlIGZhbHNlXHJcbmNvbnN0IGNhbmNlbFJlcXVlc3QgPSAocmVxdWVzdElkOiBzdHJpbmcgfCBudWxsKSA9PiB7XHJcbiAgICBsZXQgcmVtb3ZlZCA9IGZhbHNlO1xyXG4gICAgY2FwdHVyZVJlcXVlc3RRdWV1ZSA9IGNhcHR1cmVSZXF1ZXN0UXVldWUuZmlsdGVyKChpZCkgPT4ge1xyXG4gICAgICAgIGlmIChpZCAhPT0gcmVxdWVzdElkKSB7XHJcbiAgICAgICAgICAgIHJldHVybiB0cnVlO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIHJlbW92ZWQgPSB0cnVlO1xyXG4gICAgICAgICAgICBkZWJ1Z0xvZyhgSW4gcG9pbnRlckV2ZW50c0NhcHR1cmUuY2FuY2VsUmVxdWVzdCAtIENhbmNlbGluZyBwb2ludGVyIGV2ZW50cyBjYXB0dXJlIHJlcXVlc3QgJHtyZXF1ZXN0SWR9YCk7XHJcbiAgICAgICAgICAgIHJldHVybiBmYWxzZTtcclxuICAgICAgICB9XHJcbiAgICB9KTtcclxuICAgIHJldHVybiByZW1vdmVkO1xyXG59O1xyXG5cclxuY29uc3QgcmVtb3ZlVW5tYXRjaGVkUmVxdWVzdCA9IChyZXF1ZXN0SWQ6IHN0cmluZykgPT4ge1xyXG4gICAgbGV0IHJlbW92ZWQgPSBmYWxzZTtcclxuICAgIHVubWF0Y2hlZFJlbGVhc2VSZXF1ZXN0cyA9IHVubWF0Y2hlZFJlbGVhc2VSZXF1ZXN0cy5maWx0ZXIoKGlkKSA9PiB7XHJcbiAgICAgICAgaWYgKGlkICE9PSByZXF1ZXN0SWQpIHtcclxuICAgICAgICAgICAgcmV0dXJuIHRydWU7XHJcbiAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgcmVtb3ZlZCA9IHRydWU7XHJcbiAgICAgICAgICAgIHJldHVybiBmYWxzZTtcclxuICAgICAgICB9XHJcbiAgICB9KTtcclxuICAgIHJldHVybiByZW1vdmVkO1xyXG59O1xyXG5cclxuY29uc3QgdHJhbnNmZXJQb2ludGVyRXZlbnRzT3duZXJzaGlwID0gKCkgPT4ge1xyXG4gICAgY29uc3QgbmV3T3duZXJJZCA9IG5leHRDYXB0dXJlUmVxdWVzdCgpO1xyXG4gICAgZGVidWdMb2coYEluIHBvaW50ZXJFdmVudHNDYXB0dXJlLnRyYW5zZmVyUG9pbnRlckV2ZW50c093bmVyc2hpcCAtIFRyYW5zZmVycnJpbmcgcG9pbnRlciBldmVudHMgZnJvbSAke2N1cnJlbnRPd25lcn0gdG8gJHtuZXdPd25lcklkfWApO1xyXG4gICAgLy8gUmVsZWFzZSB0aGUgY3VycmVudCBvd25lclxyXG4gICAgZG9SZWxlYXNlKCk7XHJcbiAgICBpZiAobmV3T3duZXJJZCkge1xyXG4gICAgICAgIGRvQ2FwdHVyZShuZXdPd25lcklkKTtcclxuICAgIH1cclxufTtcclxuXHJcbmNvbnN0IGRvUmVsZWFzZSA9ICgpID0+IHtcclxuICAgIGRlYnVnTG9nKGBJbiBwb2ludGVyRXZlbnRzQ2FwdHVyZS5kb1JlbGVhc2UgLSBSZWxlYXNpbmcgcG9pbnRlciBldmVudHMgZnJvbSAke2N1cnJlbnRPd25lcn1gKTtcclxuICAgIGlmIChjdXJyZW50T3duZXIpIHtcclxuICAgICAgICAvLyBjYWxsIHRoZSByZWxlYXNlIGNhbGxiYWNrXHJcbiAgICAgICAgcGVuZGluZ1JlcXVlc3RDYWxsYmFja3MuZ2V0KGN1cnJlbnRPd25lcik/LnJlbGVhc2UoKTtcclxuICAgICAgICAvLyBBbmQgcmVtb3ZlIHRoZSBjYWxsYmFja3NcclxuICAgICAgICBwZW5kaW5nUmVxdWVzdENhbGxiYWNrcy5kZWxldGUoY3VycmVudE93bmVyKTtcclxuICAgICAgICBjdXJyZW50T3duZXIgPSBudWxsO1xyXG4gICAgfVxyXG59O1xyXG5cclxuY29uc3QgZG9DYXB0dXJlID0gKG5ld093bmVySWQ6IHN0cmluZykgPT4ge1xyXG4gICAgaWYgKG5ld093bmVySWQpIHtcclxuICAgICAgICAvLyBjYWxsIHRoZSBjYXB0dXJlIGNhbGxiYWNrXHJcbiAgICAgICAgcGVuZGluZ1JlcXVlc3RDYWxsYmFja3MuZ2V0KG5ld093bmVySWQpPy5jYXB0dXJlKCk7XHJcbiAgICB9XHJcbiAgICBjdXJyZW50T3duZXIgPSBuZXdPd25lcklkO1xyXG4gICAgZGVidWdMb2coYEluIHBvaW50ZXJFdmVudHNDYXB0dXJlLmRvQ2FwdHVyZSAtIFBvaW50ZXIgZXZlbnRzIG5vdyBjYXB0dXJlZCBieSAke25ld093bmVySWR9YCk7XHJcbn07XHJcblxyXG5jb25zdCBuZXh0Q2FwdHVyZVJlcXVlc3QgPSAoKSA9PiB7XHJcbiAgICByZXR1cm4gY2FwdHVyZVJlcXVlc3RRdWV1ZS5sZW5ndGggPiAwID8gY2FwdHVyZVJlcXVlc3RRdWV1ZS5zaGlmdCgpIDogbnVsbDtcclxufTtcclxuXHJcbi8vICNyZWdpb24gRGVidWdnaW5nIHN1cHBvcnRcclxuZGVjbGFyZSBnbG9iYWwge1xyXG4gICAgaW50ZXJmYWNlIFdpbmRvdyB7XHJcbiAgICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uYW1pbmctY29udmVudGlvblxyXG4gICAgICAgIFwicG9pbnRlci1ldmVudHMtY2FwdHVyZS1kZWJ1Z1wiOiBib29sZWFuIHwgbnVsbDtcclxuICAgIH1cclxufVxyXG5cclxuY29uc3QgZGVidWdMb2cgPSAobWVzc2FnZTogc3RyaW5nKSA9PiB7XHJcbiAgICAvLyBJZiB3ZSBhcmUgcnVubm5pbmcgaW4gYSB0ZXN0IHJ1bm5lciAoaW4gbm9kZSwgc28gd2luZG93IGlzIG5vdCBkZWZpbmVkKVxyXG4gICAgLy8gb3IgaWYgdGhlIGRlYnVnIGZsYWcgaXMgc2V0LCB0aGVuIGxvZyB0aGUgbWVzc2FnZVxyXG4gICAgaWYgKHR5cGVvZiB3aW5kb3cgPT09IFwidW5kZWZpbmVkXCIgfHwgd2luZG93W1wicG9pbnRlci1ldmVudHMtY2FwdHVyZS1kZWJ1Z1wiXSkge1xyXG4gICAgICAgIFRvb2xzLkxvZyhcclxuICAgICAgICAgICAgYCR7cGVyZm9ybWFuY2Uubm93KCl9IC0gZ2FtZS5zY2VuZS5wb2ludGVyRXZlbnRzIC0gJHttZXNzYWdlfVxcbmN1cnJlbnRPd25lcjogJHtjdXJyZW50T3duZXJ9XFxucXVldWU6ICR7Y2FwdHVyZVJlcXVlc3RRdWV1ZX1cXG51bm1hdGNoZWQ6ICR7dW5tYXRjaGVkUmVsZWFzZVJlcXVlc3RzfWBcclxuICAgICAgICApO1xyXG4gICAgfVxyXG59O1xyXG4vLyAjZW5kcmVnaW9uIERlYnVnZ2luZyBzdXBwb3J0XHJcbiIsImltcG9ydCB0eXBlIHsgQWJzdHJhY3RNZXNoIH0gZnJvbSBcImNvcmUvTWVzaGVzL2Fic3RyYWN0TWVzaFwiO1xyXG5pbXBvcnQgdHlwZSB7IEJlaGF2aW9yIH0gZnJvbSBcImNvcmUvQmVoYXZpb3JzL2JlaGF2aW9yXCI7XHJcbmltcG9ydCB0eXBlIHsgU2NlbmUgfSBmcm9tIFwiY29yZS9zY2VuZVwiO1xyXG5pbXBvcnQgeyBMb2dnZXIgfSBmcm9tIFwiY29yZS9NaXNjL2xvZ2dlclwiO1xyXG5pbXBvcnQgeyByZXF1ZXN0Q2FwdHVyZSwgcmVxdWVzdFJlbGVhc2UsIHJlbGVhc2VDdXJyZW50LCBnZXRDYXB0dXJpbmdJZCB9IGZyb20gXCIuL3BvaW50ZXJFdmVudHNDYXB0dXJlXCI7XHJcblxyXG4vLyBNb2R1bGUgbGV2ZWwgdmFyaWFibGUgZm9yIGhvbGRpbmcgdGhlIGN1cnJlbnQgc2NlbmVcclxubGV0IF9zY2VuZTogU2NlbmUgfCBudWxsID0gbnVsbDtcclxuXHJcbi8vIE1vZHVsZSBsZXZlbCB2YXJpYWJsZSB0byBob2xkIHRoZSBjb3VudCBvZiBiZWhhdmlvciBpbnN0YW5jZXMgdGhhdCBhcmUgY3VycmVudGx5IGNhcHR1cmluZyBwb2ludGVyIGV2ZW50c1xyXG4vLyBvbiBlbnRyeS4gIFRoaXMgaXMgdXNlZCB0byBkZXRlcm1pbmUgaWYgd2UgbmVlZCB0byBzdGFydCBvciBzdG9wIG9ic2VydmluZyBwb2ludGVyIG1vdmVtZW50LlxyXG5sZXQgY2FwdHVyZU9uRW50ZXJDb3VudCA9IDA7XHJcblxyXG4vLyBNYXAgdXNlZCB0byBzdG9yZSBpbnN0YW5jZSBvZiB0aGUgUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvciBmb3IgYSBtZXNoXHJcbi8vIFdlIGRvIHRoaXMgYmVjYXVzZSB0aGlzIGdldHMgY2hlY2tlZCBvbiBwb2ludGVyIG1vdmUgYW5kIHdlIGRvbid0IHdhbnQgdG9cclxuLy8gdXNlIGdldEJlaGF2aW9yQnlOYW1lKCkgYmVjYXVzZSB0aGF0IGlzIGEgbGluZWFyIHNlYXJjaFxyXG5jb25zdCBtZXNoVG9CZWhhdmlvck1hcCA9IG5ldyBXZWFrTWFwPEFic3RyYWN0TWVzaCwgUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvcj4oKTtcclxuXHJcbmNvbnN0IHN0YXJ0Q2FwdHVyZU9uRW50ZXIgPSAoc2NlbmU6IFNjZW5lKSA9PiB7XHJcbiAgICAvLyBJZiB3ZSBhcmUgbm90IGluIGEgYnJvd3NlciwgZG8gbm90aGluZ1xyXG4gICAgaWYgKHR5cGVvZiBkb2N1bWVudCA9PT0gXCJ1bmRlZmluZWRcIikge1xyXG4gICAgICAgIHJldHVybjtcclxuICAgIH1cclxuICAgIGlmIChjYXB0dXJlT25FbnRlckNvdW50ID09PSAwKSB7XHJcbiAgICAgICAgZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcInBvaW50ZXJtb3ZlXCIsIG9uUG9pbnRlck1vdmUpO1xyXG4gICAgICAgIGRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoXCJ0b3VjaHN0YXJ0XCIsIG9uUG9pbnRlck1vdmUpO1xyXG4gICAgICAgIF9zY2VuZSA9IF9zY2VuZSA/PyBzY2VuZTtcclxuICAgICAgICBMb2dnZXIuTG9nKFwiUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvcjogU3RhcnRpbmcgb2JzZXJ2YXRpb24gb2YgcG9pbnRlciBtb3ZlIGV2ZW50cy5cIik7XHJcbiAgICAgICAgX3NjZW5lLm9uRGlzcG9zZU9ic2VydmFibGUuYWRkKGRvU3RvcENhcHR1cmVPbkVudGVyKTtcclxuICAgIH1cclxuICAgIGNhcHR1cmVPbkVudGVyQ291bnQrKztcclxufTtcclxuXHJcbmNvbnN0IGRvU3RvcENhcHR1cmVPbkVudGVyID0gKCkgPT4ge1xyXG4gICAgZG9jdW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcInBvaW50ZXJtb3ZlXCIsIG9uUG9pbnRlck1vdmUpO1xyXG4gICAgZG9jdW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcihcInRvdWNoc3RhcnRcIiwgb25Qb2ludGVyTW92ZSk7XHJcbiAgICBfc2NlbmUgPSBudWxsO1xyXG4gICAgTG9nZ2VyLkxvZyhcIlBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3I6IFN0b3BwaW5nIG9ic2VydmF0aW9uIG9mIHBvaW50ZXIgbW92ZSBldmVudHMuXCIpO1xyXG4gICAgY2FwdHVyZU9uRW50ZXJDb3VudCA9IDA7XHJcbn07XHJcblxyXG5jb25zdCBzdG9wQ2FwdHVyZU9uRW50ZXIgPSAoKSA9PiB7XHJcbiAgICAvLyBJZiB3ZSBhcmUgbm90IGluIGEgYnJvd3NlciwgZG8gbm90aGluZ1xyXG4gICAgaWYgKHR5cGVvZiBkb2N1bWVudCA9PT0gXCJ1bmRlZmluZWRcIikge1xyXG4gICAgICAgIHJldHVybjtcclxuICAgIH1cclxuXHJcbiAgICAvLyBJZiB3ZSBhcmUgbm90IG9ic2VydmluZyBwb2ludGVyIG1vdmVtZW50LCBkbyBub3RoaW5nXHJcbiAgICBpZiAoIV9zY2VuZSkge1xyXG4gICAgICAgIHJldHVybjtcclxuICAgIH1cclxuXHJcbiAgICBjYXB0dXJlT25FbnRlckNvdW50LS07XHJcbiAgICBpZiAoY2FwdHVyZU9uRW50ZXJDb3VudCA8PSAwKSB7XHJcbiAgICAgICAgZG9TdG9wQ2FwdHVyZU9uRW50ZXIoKTtcclxuICAgIH1cclxufTtcclxuXHJcbi8vIE1vZHVsZSBsZXZlbCBmdW5jdGlvbiB1c2VkIHRvIGRldGVybWluZSBpZiBhbiBlbnRlcmVkIG1lc2ggc2hvdWxkIGNhcHR1cmUgcG9pbnRlciBldmVudHNcclxuY29uc3Qgb25Qb2ludGVyTW92ZSA9IChldnQ6IFBvaW50ZXJFdmVudCB8IFRvdWNoRXZlbnQpID0+IHtcclxuICAgIGlmICghX3NjZW5lKSB7XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIGNvbnN0IGNhbnZhc1JlY3QgPSBfc2NlbmUuZ2V0RW5naW5lKCkuZ2V0UmVuZGVyaW5nQ2FudmFzQ2xpZW50UmVjdCgpO1xyXG4gICAgaWYgKCFjYW52YXNSZWN0KSB7XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIEdldCB0aGUgb2JqZWN0IHRoYXQgY29udGFpbnMgdGhlIGNsaWVudCBYIGFuZCBZIGZyb20gZWl0aGVyIHRoZSBwb2ludGVyIGV2ZW50IG9yIGZyb20gdGhlXHJcbiAgICAvLyBUb3VjaEV2ZW50IHRvdWNoXHJcbiAgICBjb25zdCB7IGNsaWVudFgsIGNsaWVudFkgfSA9IFwidG91Y2hlc1wiIGluIGV2dCA/IGV2dC50b3VjaGVzWzBdIDogZXZ0O1xyXG5cclxuICAgIC8vIGdldCB0aGUgcGlja2VkIG1lc2gsIGlmIGFueVxyXG4gICAgY29uc3QgcG9pbnRlclNjcmVlblggPSBjbGllbnRYIC0gY2FudmFzUmVjdC5sZWZ0O1xyXG4gICAgY29uc3QgcG9pbnRlclNjcmVlblkgPSBjbGllbnRZIC0gY2FudmFzUmVjdC50b3A7XHJcblxyXG4gICAgbGV0IHBvaW50ZXJDYXB0dXJlQmVoYXZpb3I6IFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IgfCB1bmRlZmluZWQ7XHJcbiAgICBjb25zdCBwaWNrUmVzdWx0ID0gX3NjZW5lLnBpY2socG9pbnRlclNjcmVlblgsIHBvaW50ZXJTY3JlZW5ZLCAobWVzaCkgPT4ge1xyXG4gICAgICAgIC8vIElmIHRoZSBtZXNoIGhhcyBhbiBpbnN0YW5jZSBvZiBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yIGF0dGFjaGVkIHRvIGl0LFxyXG4gICAgICAgIC8vIGFuZCBjYXB0dXJlIG9uIHBvaW50ZXIgZW50ZXIgaXMgdHJ1ZSwgdGhlbiB3ZSB3YW50IHRvIHBpY2sgaXRcclxuICAgICAgICBjb25zdCBwb2ludGVyQ2FwdHVyZUJlaGF2aW9yID0gbWVzaFRvQmVoYXZpb3JNYXAuZ2V0KG1lc2gpO1xyXG4gICAgICAgIHJldHVybiBtZXNoLmlzRW5hYmxlZCgpICYmIHR5cGVvZiBwb2ludGVyQ2FwdHVyZUJlaGF2aW9yICE9PSBcInVuZGVmaW5lZFwiICYmIHBvaW50ZXJDYXB0dXJlQmVoYXZpb3IuX2NhcHR1cmVPblBvaW50ZXJFbnRlcjtcclxuICAgIH0pO1xyXG5cclxuICAgIGxldCBwaWNrZWRNZXNoOiBBYnN0cmFjdE1lc2ggfCBudWxsO1xyXG4gICAgaWYgKHBpY2tSZXN1bHQuaGl0KSB7XHJcbiAgICAgICAgcGlja2VkTWVzaCA9IHBpY2tSZXN1bHQucGlja2VkTWVzaDtcclxuICAgIH0gZWxzZSB7XHJcbiAgICAgICAgcGlja2VkTWVzaCA9IG51bGw7XHJcbiAgICB9XHJcblxyXG4gICAgY29uc3QgY2FwdHVyaW5nSWRBc0ludCA9IHBhcnNlSW50KGdldENhcHR1cmluZ0lkKCkgfHwgXCJcIik7XHJcblxyXG4gICAgLy8gaWYgdGhlIHBpY2tlZCBtZXNoIGlzIHRoZSBjdXJyZW50IGNhcHR1cmluZyBtZXNoLCBkbyBub3RoaW5nXHJcbiAgICBpZiAocGlja2VkTWVzaCAmJiBwaWNrZWRNZXNoLnVuaXF1ZUlkID09PSBjYXB0dXJpbmdJZEFzSW50KSB7XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIElmIHRoZXJlIGlzIGEgY2FwdHVyaW5nIG1lc2ggYW5kIGl0IGlzIG5vdCB0aGUgY3VycmVudCBwaWNrZWQgbWVzaCwgb3Igbm9cclxuICAgIC8vIG1lc2ggaXMgcGlja2VkLCByZWxlYXNlIHRoZSBjYXB0dXJpbmcgbWVzaFxyXG4gICAgaWYgKGNhcHR1cmluZ0lkQXNJbnQgJiYgKCFwaWNrZWRNZXNoIHx8IHBpY2tlZE1lc2gudW5pcXVlSWQgIT09IGNhcHR1cmluZ0lkQXNJbnQpKSB7XHJcbiAgICAgICAgcmVsZWFzZUN1cnJlbnQoKTtcclxuICAgIH1cclxuXHJcbiAgICAvLyBJZiB0aGVyZSBpcyBhIHBpY2tlZCBtZXNoIGFuZCBpdCBpcyBub3QgdGhlIGN1cnJlbnQgY2FwdHVyaW5nIG1lc2gsIGNhcHR1cmVcclxuICAgIC8vIHRoZSBwb2ludGVyIGV2ZW50cy4gIE5vdGUgdGhhdCB0aGUgY3VycmVudCBjYXB0dXJpbmcgbWVzaCBoYXMgYWxyZWFkeSBiZWVuXHJcbiAgICAvLyByZWxlYXNlZCBhYm92ZVxyXG4gICAgaWYgKHBpY2tlZE1lc2gpIHtcclxuICAgICAgICBwb2ludGVyQ2FwdHVyZUJlaGF2aW9yID0gbWVzaFRvQmVoYXZpb3JNYXAuZ2V0KHBpY2tlZE1lc2gpO1xyXG4gICAgICAgIHBvaW50ZXJDYXB0dXJlQmVoYXZpb3IhLmNhcHR1cmVQb2ludGVyRXZlbnRzKCk7XHJcbiAgICB9XHJcbn07XHJcblxyXG4vKipcclxuICogQmVoYXZpb3IgZm9yIGFueSBjb250ZW50IHRoYXQgY2FuIGNhcHR1cmUgcG9pbnRlciBldmVudHMsIGkuZS4gYnlwYXNzIHRoZSBCYWJ5bG9uIHBvaW50ZXIgZXZlbnQgaGFuZGxpbmdcclxuICogYW5kIHJlY2VpdmUgcG9pbnRlciBldmVudHMgZGlyZWN0bHkuICBJdCB3aWxsIHJlZ2lzdGVyIHRoZSBjYXB0dXJlIHRyaWdnZXJzIGFuZCBuZWdvdGlhdGUgdGhlIGNhcHR1cmUgYW5kXHJcbiAqIHJlbGVhc2Ugb2YgcG9pbnRlciBldmVudHMuICBDdXJlcm50bHkgdGhpcyBhcHBsaWVzIG9ubHkgdG8gSHRtbE1lc2hcclxuICovXHJcbmV4cG9ydCBjbGFzcyBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yIGltcGxlbWVudHMgQmVoYXZpb3I8QWJzdHJhY3RNZXNoPiB7XHJcbiAgICAvKiogZ2V0cyBvciBzZXRzIGJlaGF2aW9yJ3MgbmFtZSAqL1xyXG4gICAgcHVibGljIG5hbWUgPSBcIlBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3JcIjtcclxuXHJcbiAgICBwcml2YXRlIF9hdHRhY2hlZE1lc2g6IEFic3RyYWN0TWVzaCB8IG51bGw7XHJcbiAgICAvKiogQGludGVybmFsICovXHJcbiAgICBwdWJsaWMgX2NhcHR1cmVPblBvaW50ZXJFbnRlcjogYm9vbGVhbjtcclxuXHJcbiAgICAvKipcclxuICAgICAqIEdldHMgb3Igc2V0cyB0aGUgbWVzaCB0aGF0IHRoZSBiZWhhdmlvciBpcyBhdHRhY2hlZCB0b1xyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZ2V0IGF0dGFjaGVkTWVzaCgpIHtcclxuICAgICAgICByZXR1cm4gdGhpcy5fYXR0YWNoZWRNZXNoO1xyXG4gICAgfVxyXG5cclxuICAgIHB1YmxpYyBzZXQgYXR0YWNoZWRNZXNoKHZhbHVlOiBBYnN0cmFjdE1lc2ggfCBudWxsKSB7XHJcbiAgICAgICAgdGhpcy5fYXR0YWNoZWRNZXNoID0gdmFsdWU7XHJcbiAgICB9XHJcblxyXG4gICAgY29uc3RydWN0b3IoXHJcbiAgICAgICAgcHJpdmF0ZSBfY2FwdHVyZUNhbGxiYWNrOiAoKSA9PiB2b2lkLFxyXG4gICAgICAgIHByaXZhdGUgX3JlbGVhc2VDYWxsYmFjazogKCkgPT4gdm9pZCxcclxuICAgICAgICB7IGNhcHR1cmVPblBvaW50ZXJFbnRlciA9IHRydWUgfSA9IHt9XHJcbiAgICApIHtcclxuICAgICAgICB0aGlzLl9hdHRhY2hlZE1lc2ggPSBudWxsO1xyXG4gICAgICAgIHRoaXMuX2NhcHR1cmVPblBvaW50ZXJFbnRlciA9IGNhcHR1cmVPblBvaW50ZXJFbnRlcjtcclxuXHJcbiAgICAgICAgLy8gV2FybiBpZiB3ZSBhcmUgbm90IGluIGEgYnJvd3NlclxyXG4gICAgICAgIGlmICh0eXBlb2YgZG9jdW1lbnQgPT09IFwidW5kZWZpbmVkXCIpIHtcclxuICAgICAgICAgICAgTG9nZ2VyLldhcm4oYENyZWF0aW5nIGFuIGluc3RhbmNlIG9mIFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3Igb3V0c2lkZSBvZiBhIGJyb3dzZXIuICBUaGUgYmVoYXZpb3Igd2lsbCBub3Qgd29yay5gKTtcclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBTZXQgaWYgdGhlIGJlaGF2aW9yIHNob3VsZCBjYXB0dXJlIHBvaW50ZXIgZXZlbnRzIHdoZW4gdGhlIHBvaW50ZXIgZW50ZXJzIHRoZSBtZXNoXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBzZXQgY2FwdHVyZU9uUG9pbnRlckVudGVyKGNhcHR1cmVPblBvaW50ZXJFbnRlcjogYm9vbGVhbikge1xyXG4gICAgICAgIGlmICh0aGlzLl9jYXB0dXJlT25Qb2ludGVyRW50ZXIgPT09IGNhcHR1cmVPblBvaW50ZXJFbnRlcikge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHRoaXMuX2NhcHR1cmVPblBvaW50ZXJFbnRlciA9IGNhcHR1cmVPblBvaW50ZXJFbnRlcjtcclxuICAgICAgICBpZiAodGhpcy5fYXR0YWNoZWRNZXNoKSB7XHJcbiAgICAgICAgICAgIGlmICh0aGlzLl9jYXB0dXJlT25Qb2ludGVyRW50ZXIpIHtcclxuICAgICAgICAgICAgICAgIHN0YXJ0Q2FwdHVyZU9uRW50ZXIodGhpcy5fYXR0YWNoZWRNZXNoLmdldFNjZW5lKCkhKTtcclxuICAgICAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgICAgIHN0b3BDYXB0dXJlT25FbnRlcigpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogRnVuY3Rpb24gY2FsbGVkIHdoZW4gdGhlIGJlaGF2aW9yIG5lZWRzIHRvIGJlIGluaXRpYWxpemVkIChiZWZvcmUgYXR0YWNoaW5nIGl0IHRvIGEgdGFyZ2V0KVxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgaW5pdCgpIHt9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDYWxsZWQgd2hlbiB0aGUgYmVoYXZpb3IgaXMgYXR0YWNoZWQgdG8gYSB0YXJnZXRcclxuICAgICAqIEBwYXJhbSBtZXNoIGRlZmluZXMgdGhlIHRhcmdldCB3aGVyZSB0aGUgYmVoYXZpb3IgaXMgYXR0YWNoZWQgdG9cclxuICAgICAqL1xyXG4gICAgcHVibGljIGF0dGFjaChtZXNoOiBBYnN0cmFjdE1lc2gpIHtcclxuICAgICAgICAvLyBBZGQgYSByZWZlcmVuY2UgdG8gdGhpcyBiZWhhdmlvciBvbiB0aGUgbWVzaC4gIFdlIGRvIHRoaXMgc28gd2UgY2FuIGdldCBhXHJcbiAgICAgICAgLy8gcmVmZXJlbmNlIHRvIHRoZSBiZWhhdmlvciBpbiB0aGUgb25Qb2ludGVyTW92ZSBmdW5jdGlvbiB3aXRob3V0IHJlbHlpbmcgb25cclxuICAgICAgICAvLyBnZXRCZWhhdmlvckJ5TmFtZSgpLCB3aGljaCBkb2VzIGEgbGluZWFyIHNlYXJjaCBvZiB0aGUgYmVoYXZpb3JzIGFycmF5LlxyXG4gICAgICAgIHRoaXMuYXR0YWNoZWRNZXNoID0gbWVzaDtcclxuICAgICAgICBtZXNoVG9CZWhhdmlvck1hcC5zZXQobWVzaCwgdGhpcyk7XHJcbiAgICAgICAgaWYgKHRoaXMuX2NhcHR1cmVPblBvaW50ZXJFbnRlcikge1xyXG4gICAgICAgICAgICBzdGFydENhcHR1cmVPbkVudGVyKG1lc2guZ2V0U2NlbmUoKSEpO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENhbGxlZCB3aGVuIHRoZSBiZWhhdmlvciBpcyBkZXRhY2hlZCBmcm9tIGl0cyB0YXJnZXRcclxuICAgICAqL1xyXG4gICAgcHVibGljIGRldGFjaCgpIHtcclxuICAgICAgICBpZiAoIXRoaXMuYXR0YWNoZWRNZXNoKSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcbiAgICAgICAgLy8gUmVtb3ZlIHRoZSByZWZlcmVuY2UgdG8gdGhpcyBiZWhhdmlvciBmcm9tIHRoZSBtZXNoXHJcbiAgICAgICAgbWVzaFRvQmVoYXZpb3JNYXAuZGVsZXRlKHRoaXMuYXR0YWNoZWRNZXNoKTtcclxuICAgICAgICBpZiAodGhpcy5fY2FwdHVyZU9uUG9pbnRlckVudGVyKSB7XHJcbiAgICAgICAgICAgIHN0b3BDYXB0dXJlT25FbnRlcigpO1xyXG4gICAgICAgIH1cclxuICAgICAgICB0aGlzLmF0dGFjaGVkTWVzaCA9IG51bGw7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBEaXNwb3NlIHRoZSBiZWhhdmlvclxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZGlzcG9zZSgpIHtcclxuICAgICAgICB0aGlzLmRldGFjaCgpO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIFJlbGVhc2UgcG9pbnRlciBldmVudHNcclxuICAgIHB1YmxpYyByZWxlYXNlUG9pbnRlckV2ZW50cygpIHtcclxuICAgICAgICBpZiAoIXRoaXMuYXR0YWNoZWRNZXNoKSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcbiAgICAgICAgcmVxdWVzdFJlbGVhc2UodGhpcy5hdHRhY2hlZE1lc2gudW5pcXVlSWQudG9TdHJpbmcoKSk7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gQ2FwdHVyZSBwb2ludGVyIGV2ZW50c1xyXG4gICAgcHVibGljIGNhcHR1cmVQb2ludGVyRXZlbnRzKCkge1xyXG4gICAgICAgIGlmICghdGhpcy5hdHRhY2hlZE1lc2gpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuICAgICAgICByZXF1ZXN0Q2FwdHVyZSh0aGlzLmF0dGFjaGVkTWVzaC51bmlxdWVJZC50b1N0cmluZygpLCB0aGlzLl9jYXB0dXJlQ2FsbGJhY2ssIHRoaXMuX3JlbGVhc2VDYWxsYmFjayk7XHJcbiAgICB9XHJcbn1cclxuIiwiZXhwb3J0ICogZnJvbSBcIi4vaHRtbE1lc2hcIjtcbiIsIm1vZHVsZS5leHBvcnRzID0gX19XRUJQQUNLX0VYVEVSTkFMX01PRFVMRV9iYWJ5bG9uanNfTWVzaGVzX21lc2hfXzsiLCIvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG5Db3B5cmlnaHQgKGMpIE1pY3Jvc29mdCBDb3Jwb3JhdGlvbi5cblxuUGVybWlzc2lvbiB0byB1c2UsIGNvcHksIG1vZGlmeSwgYW5kL29yIGRpc3RyaWJ1dGUgdGhpcyBzb2Z0d2FyZSBmb3IgYW55XG5wdXJwb3NlIHdpdGggb3Igd2l0aG91dCBmZWUgaXMgaGVyZWJ5IGdyYW50ZWQuXG5cblRIRSBTT0ZUV0FSRSBJUyBQUk9WSURFRCBcIkFTIElTXCIgQU5EIFRIRSBBVVRIT1IgRElTQ0xBSU1TIEFMTCBXQVJSQU5USUVTIFdJVEhcblJFR0FSRCBUTyBUSElTIFNPRlRXQVJFIElOQ0xVRElORyBBTEwgSU1QTElFRCBXQVJSQU5USUVTIE9GIE1FUkNIQU5UQUJJTElUWVxuQU5EIEZJVE5FU1MuIElOIE5PIEVWRU5UIFNIQUxMIFRIRSBBVVRIT1IgQkUgTElBQkxFIEZPUiBBTlkgU1BFQ0lBTCwgRElSRUNULFxuSU5ESVJFQ1QsIE9SIENPTlNFUVVFTlRJQUwgREFNQUdFUyBPUiBBTlkgREFNQUdFUyBXSEFUU09FVkVSIFJFU1VMVElORyBGUk9NXG5MT1NTIE9GIFVTRSwgREFUQSBPUiBQUk9GSVRTLCBXSEVUSEVSIElOIEFOIEFDVElPTiBPRiBDT05UUkFDVCwgTkVHTElHRU5DRSBPUlxuT1RIRVIgVE9SVElPVVMgQUNUSU9OLCBBUklTSU5HIE9VVCBPRiBPUiBJTiBDT05ORUNUSU9OIFdJVEggVEhFIFVTRSBPUlxuUEVSRk9STUFOQ0UgT0YgVEhJUyBTT0ZUV0FSRS5cbioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqICovXG4vKiBnbG9iYWwgUmVmbGVjdCwgUHJvbWlzZSwgU3VwcHJlc3NlZEVycm9yLCBTeW1ib2wsIEl0ZXJhdG9yICovXG5cbnZhciBleHRlbmRTdGF0aWNzID0gZnVuY3Rpb24oZCwgYikge1xuICBleHRlbmRTdGF0aWNzID0gT2JqZWN0LnNldFByb3RvdHlwZU9mIHx8XG4gICAgICAoeyBfX3Byb3RvX186IFtdIH0gaW5zdGFuY2VvZiBBcnJheSAmJiBmdW5jdGlvbiAoZCwgYikgeyBkLl9fcHJvdG9fXyA9IGI7IH0pIHx8XG4gICAgICBmdW5jdGlvbiAoZCwgYikgeyBmb3IgKHZhciBwIGluIGIpIGlmIChPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwoYiwgcCkpIGRbcF0gPSBiW3BdOyB9O1xuICByZXR1cm4gZXh0ZW5kU3RhdGljcyhkLCBiKTtcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2V4dGVuZHMoZCwgYikge1xuICBpZiAodHlwZW9mIGIgIT09IFwiZnVuY3Rpb25cIiAmJiBiICE9PSBudWxsKVxuICAgICAgdGhyb3cgbmV3IFR5cGVFcnJvcihcIkNsYXNzIGV4dGVuZHMgdmFsdWUgXCIgKyBTdHJpbmcoYikgKyBcIiBpcyBub3QgYSBjb25zdHJ1Y3RvciBvciBudWxsXCIpO1xuICBleHRlbmRTdGF0aWNzKGQsIGIpO1xuICBmdW5jdGlvbiBfXygpIHsgdGhpcy5jb25zdHJ1Y3RvciA9IGQ7IH1cbiAgZC5wcm90b3R5cGUgPSBiID09PSBudWxsID8gT2JqZWN0LmNyZWF0ZShiKSA6IChfXy5wcm90b3R5cGUgPSBiLnByb3RvdHlwZSwgbmV3IF9fKCkpO1xufVxuXG5leHBvcnQgdmFyIF9fYXNzaWduID0gZnVuY3Rpb24oKSB7XG4gIF9fYXNzaWduID0gT2JqZWN0LmFzc2lnbiB8fCBmdW5jdGlvbiBfX2Fzc2lnbih0KSB7XG4gICAgICBmb3IgKHZhciBzLCBpID0gMSwgbiA9IGFyZ3VtZW50cy5sZW5ndGg7IGkgPCBuOyBpKyspIHtcbiAgICAgICAgICBzID0gYXJndW1lbnRzW2ldO1xuICAgICAgICAgIGZvciAodmFyIHAgaW4gcykgaWYgKE9iamVjdC5wcm90b3R5cGUuaGFzT3duUHJvcGVydHkuY2FsbChzLCBwKSkgdFtwXSA9IHNbcF07XG4gICAgICB9XG4gICAgICByZXR1cm4gdDtcbiAgfVxuICByZXR1cm4gX19hc3NpZ24uYXBwbHkodGhpcywgYXJndW1lbnRzKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fcmVzdChzLCBlKSB7XG4gIHZhciB0ID0ge307XG4gIGZvciAodmFyIHAgaW4gcykgaWYgKE9iamVjdC5wcm90b3R5cGUuaGFzT3duUHJvcGVydHkuY2FsbChzLCBwKSAmJiBlLmluZGV4T2YocCkgPCAwKVxuICAgICAgdFtwXSA9IHNbcF07XG4gIGlmIChzICE9IG51bGwgJiYgdHlwZW9mIE9iamVjdC5nZXRPd25Qcm9wZXJ0eVN5bWJvbHMgPT09IFwiZnVuY3Rpb25cIilcbiAgICAgIGZvciAodmFyIGkgPSAwLCBwID0gT2JqZWN0LmdldE93blByb3BlcnR5U3ltYm9scyhzKTsgaSA8IHAubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgICBpZiAoZS5pbmRleE9mKHBbaV0pIDwgMCAmJiBPYmplY3QucHJvdG90eXBlLnByb3BlcnR5SXNFbnVtZXJhYmxlLmNhbGwocywgcFtpXSkpXG4gICAgICAgICAgICAgIHRbcFtpXV0gPSBzW3BbaV1dO1xuICAgICAgfVxuICByZXR1cm4gdDtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fZGVjb3JhdGUoZGVjb3JhdG9ycywgdGFyZ2V0LCBrZXksIGRlc2MpIHtcbiAgdmFyIGMgPSBhcmd1bWVudHMubGVuZ3RoLCByID0gYyA8IDMgPyB0YXJnZXQgOiBkZXNjID09PSBudWxsID8gZGVzYyA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IodGFyZ2V0LCBrZXkpIDogZGVzYywgZDtcbiAgaWYgKHR5cGVvZiBSZWZsZWN0ID09PSBcIm9iamVjdFwiICYmIHR5cGVvZiBSZWZsZWN0LmRlY29yYXRlID09PSBcImZ1bmN0aW9uXCIpIHIgPSBSZWZsZWN0LmRlY29yYXRlKGRlY29yYXRvcnMsIHRhcmdldCwga2V5LCBkZXNjKTtcbiAgZWxzZSBmb3IgKHZhciBpID0gZGVjb3JhdG9ycy5sZW5ndGggLSAxOyBpID49IDA7IGktLSkgaWYgKGQgPSBkZWNvcmF0b3JzW2ldKSByID0gKGMgPCAzID8gZChyKSA6IGMgPiAzID8gZCh0YXJnZXQsIGtleSwgcikgOiBkKHRhcmdldCwga2V5KSkgfHwgcjtcbiAgcmV0dXJuIGMgPiAzICYmIHIgJiYgT2JqZWN0LmRlZmluZVByb3BlcnR5KHRhcmdldCwga2V5LCByKSwgcjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fcGFyYW0ocGFyYW1JbmRleCwgZGVjb3JhdG9yKSB7XG4gIHJldHVybiBmdW5jdGlvbiAodGFyZ2V0LCBrZXkpIHsgZGVjb3JhdG9yKHRhcmdldCwga2V5LCBwYXJhbUluZGV4KTsgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gX19lc0RlY29yYXRlKGN0b3IsIGRlc2NyaXB0b3JJbiwgZGVjb3JhdG9ycywgY29udGV4dEluLCBpbml0aWFsaXplcnMsIGV4dHJhSW5pdGlhbGl6ZXJzKSB7XG4gIGZ1bmN0aW9uIGFjY2VwdChmKSB7IGlmIChmICE9PSB2b2lkIDAgJiYgdHlwZW9mIGYgIT09IFwiZnVuY3Rpb25cIikgdGhyb3cgbmV3IFR5cGVFcnJvcihcIkZ1bmN0aW9uIGV4cGVjdGVkXCIpOyByZXR1cm4gZjsgfVxuICB2YXIga2luZCA9IGNvbnRleHRJbi5raW5kLCBrZXkgPSBraW5kID09PSBcImdldHRlclwiID8gXCJnZXRcIiA6IGtpbmQgPT09IFwic2V0dGVyXCIgPyBcInNldFwiIDogXCJ2YWx1ZVwiO1xuICB2YXIgdGFyZ2V0ID0gIWRlc2NyaXB0b3JJbiAmJiBjdG9yID8gY29udGV4dEluW1wic3RhdGljXCJdID8gY3RvciA6IGN0b3IucHJvdG90eXBlIDogbnVsbDtcbiAgdmFyIGRlc2NyaXB0b3IgPSBkZXNjcmlwdG9ySW4gfHwgKHRhcmdldCA/IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IodGFyZ2V0LCBjb250ZXh0SW4ubmFtZSkgOiB7fSk7XG4gIHZhciBfLCBkb25lID0gZmFsc2U7XG4gIGZvciAodmFyIGkgPSBkZWNvcmF0b3JzLmxlbmd0aCAtIDE7IGkgPj0gMDsgaS0tKSB7XG4gICAgICB2YXIgY29udGV4dCA9IHt9O1xuICAgICAgZm9yICh2YXIgcCBpbiBjb250ZXh0SW4pIGNvbnRleHRbcF0gPSBwID09PSBcImFjY2Vzc1wiID8ge30gOiBjb250ZXh0SW5bcF07XG4gICAgICBmb3IgKHZhciBwIGluIGNvbnRleHRJbi5hY2Nlc3MpIGNvbnRleHQuYWNjZXNzW3BdID0gY29udGV4dEluLmFjY2Vzc1twXTtcbiAgICAgIGNvbnRleHQuYWRkSW5pdGlhbGl6ZXIgPSBmdW5jdGlvbiAoZikgeyBpZiAoZG9uZSkgdGhyb3cgbmV3IFR5cGVFcnJvcihcIkNhbm5vdCBhZGQgaW5pdGlhbGl6ZXJzIGFmdGVyIGRlY29yYXRpb24gaGFzIGNvbXBsZXRlZFwiKTsgZXh0cmFJbml0aWFsaXplcnMucHVzaChhY2NlcHQoZiB8fCBudWxsKSk7IH07XG4gICAgICB2YXIgcmVzdWx0ID0gKDAsIGRlY29yYXRvcnNbaV0pKGtpbmQgPT09IFwiYWNjZXNzb3JcIiA/IHsgZ2V0OiBkZXNjcmlwdG9yLmdldCwgc2V0OiBkZXNjcmlwdG9yLnNldCB9IDogZGVzY3JpcHRvcltrZXldLCBjb250ZXh0KTtcbiAgICAgIGlmIChraW5kID09PSBcImFjY2Vzc29yXCIpIHtcbiAgICAgICAgICBpZiAocmVzdWx0ID09PSB2b2lkIDApIGNvbnRpbnVlO1xuICAgICAgICAgIGlmIChyZXN1bHQgPT09IG51bGwgfHwgdHlwZW9mIHJlc3VsdCAhPT0gXCJvYmplY3RcIikgdGhyb3cgbmV3IFR5cGVFcnJvcihcIk9iamVjdCBleHBlY3RlZFwiKTtcbiAgICAgICAgICBpZiAoXyA9IGFjY2VwdChyZXN1bHQuZ2V0KSkgZGVzY3JpcHRvci5nZXQgPSBfO1xuICAgICAgICAgIGlmIChfID0gYWNjZXB0KHJlc3VsdC5zZXQpKSBkZXNjcmlwdG9yLnNldCA9IF87XG4gICAgICAgICAgaWYgKF8gPSBhY2NlcHQocmVzdWx0LmluaXQpKSBpbml0aWFsaXplcnMudW5zaGlmdChfKTtcbiAgICAgIH1cbiAgICAgIGVsc2UgaWYgKF8gPSBhY2NlcHQocmVzdWx0KSkge1xuICAgICAgICAgIGlmIChraW5kID09PSBcImZpZWxkXCIpIGluaXRpYWxpemVycy51bnNoaWZ0KF8pO1xuICAgICAgICAgIGVsc2UgZGVzY3JpcHRvcltrZXldID0gXztcbiAgICAgIH1cbiAgfVxuICBpZiAodGFyZ2V0KSBPYmplY3QuZGVmaW5lUHJvcGVydHkodGFyZ2V0LCBjb250ZXh0SW4ubmFtZSwgZGVzY3JpcHRvcik7XG4gIGRvbmUgPSB0cnVlO1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIF9fcnVuSW5pdGlhbGl6ZXJzKHRoaXNBcmcsIGluaXRpYWxpemVycywgdmFsdWUpIHtcbiAgdmFyIHVzZVZhbHVlID0gYXJndW1lbnRzLmxlbmd0aCA+IDI7XG4gIGZvciAodmFyIGkgPSAwOyBpIDwgaW5pdGlhbGl6ZXJzLmxlbmd0aDsgaSsrKSB7XG4gICAgICB2YWx1ZSA9IHVzZVZhbHVlID8gaW5pdGlhbGl6ZXJzW2ldLmNhbGwodGhpc0FyZywgdmFsdWUpIDogaW5pdGlhbGl6ZXJzW2ldLmNhbGwodGhpc0FyZyk7XG4gIH1cbiAgcmV0dXJuIHVzZVZhbHVlID8gdmFsdWUgOiB2b2lkIDA7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gX19wcm9wS2V5KHgpIHtcbiAgcmV0dXJuIHR5cGVvZiB4ID09PSBcInN5bWJvbFwiID8geCA6IFwiXCIuY29uY2F0KHgpO1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIF9fc2V0RnVuY3Rpb25OYW1lKGYsIG5hbWUsIHByZWZpeCkge1xuICBpZiAodHlwZW9mIG5hbWUgPT09IFwic3ltYm9sXCIpIG5hbWUgPSBuYW1lLmRlc2NyaXB0aW9uID8gXCJbXCIuY29uY2F0KG5hbWUuZGVzY3JpcHRpb24sIFwiXVwiKSA6IFwiXCI7XG4gIHJldHVybiBPYmplY3QuZGVmaW5lUHJvcGVydHkoZiwgXCJuYW1lXCIsIHsgY29uZmlndXJhYmxlOiB0cnVlLCB2YWx1ZTogcHJlZml4ID8gXCJcIi5jb25jYXQocHJlZml4LCBcIiBcIiwgbmFtZSkgOiBuYW1lIH0pO1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIF9fbWV0YWRhdGEobWV0YWRhdGFLZXksIG1ldGFkYXRhVmFsdWUpIHtcbiAgaWYgKHR5cGVvZiBSZWZsZWN0ID09PSBcIm9iamVjdFwiICYmIHR5cGVvZiBSZWZsZWN0Lm1ldGFkYXRhID09PSBcImZ1bmN0aW9uXCIpIHJldHVybiBSZWZsZWN0Lm1ldGFkYXRhKG1ldGFkYXRhS2V5LCBtZXRhZGF0YVZhbHVlKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fYXdhaXRlcih0aGlzQXJnLCBfYXJndW1lbnRzLCBQLCBnZW5lcmF0b3IpIHtcbiAgZnVuY3Rpb24gYWRvcHQodmFsdWUpIHsgcmV0dXJuIHZhbHVlIGluc3RhbmNlb2YgUCA/IHZhbHVlIDogbmV3IFAoZnVuY3Rpb24gKHJlc29sdmUpIHsgcmVzb2x2ZSh2YWx1ZSk7IH0pOyB9XG4gIHJldHVybiBuZXcgKFAgfHwgKFAgPSBQcm9taXNlKSkoZnVuY3Rpb24gKHJlc29sdmUsIHJlamVjdCkge1xuICAgICAgZnVuY3Rpb24gZnVsZmlsbGVkKHZhbHVlKSB7IHRyeSB7IHN0ZXAoZ2VuZXJhdG9yLm5leHQodmFsdWUpKTsgfSBjYXRjaCAoZSkgeyByZWplY3QoZSk7IH0gfVxuICAgICAgZnVuY3Rpb24gcmVqZWN0ZWQodmFsdWUpIHsgdHJ5IHsgc3RlcChnZW5lcmF0b3JbXCJ0aHJvd1wiXSh2YWx1ZSkpOyB9IGNhdGNoIChlKSB7IHJlamVjdChlKTsgfSB9XG4gICAgICBmdW5jdGlvbiBzdGVwKHJlc3VsdCkgeyByZXN1bHQuZG9uZSA/IHJlc29sdmUocmVzdWx0LnZhbHVlKSA6IGFkb3B0KHJlc3VsdC52YWx1ZSkudGhlbihmdWxmaWxsZWQsIHJlamVjdGVkKTsgfVxuICAgICAgc3RlcCgoZ2VuZXJhdG9yID0gZ2VuZXJhdG9yLmFwcGx5KHRoaXNBcmcsIF9hcmd1bWVudHMgfHwgW10pKS5uZXh0KCkpO1xuICB9KTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fZ2VuZXJhdG9yKHRoaXNBcmcsIGJvZHkpIHtcbiAgdmFyIF8gPSB7IGxhYmVsOiAwLCBzZW50OiBmdW5jdGlvbigpIHsgaWYgKHRbMF0gJiAxKSB0aHJvdyB0WzFdOyByZXR1cm4gdFsxXTsgfSwgdHJ5czogW10sIG9wczogW10gfSwgZiwgeSwgdCwgZyA9IE9iamVjdC5jcmVhdGUoKHR5cGVvZiBJdGVyYXRvciA9PT0gXCJmdW5jdGlvblwiID8gSXRlcmF0b3IgOiBPYmplY3QpLnByb3RvdHlwZSk7XG4gIHJldHVybiBnLm5leHQgPSB2ZXJiKDApLCBnW1widGhyb3dcIl0gPSB2ZXJiKDEpLCBnW1wicmV0dXJuXCJdID0gdmVyYigyKSwgdHlwZW9mIFN5bWJvbCA9PT0gXCJmdW5jdGlvblwiICYmIChnW1N5bWJvbC5pdGVyYXRvcl0gPSBmdW5jdGlvbigpIHsgcmV0dXJuIHRoaXM7IH0pLCBnO1xuICBmdW5jdGlvbiB2ZXJiKG4pIHsgcmV0dXJuIGZ1bmN0aW9uICh2KSB7IHJldHVybiBzdGVwKFtuLCB2XSk7IH07IH1cbiAgZnVuY3Rpb24gc3RlcChvcCkge1xuICAgICAgaWYgKGYpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJHZW5lcmF0b3IgaXMgYWxyZWFkeSBleGVjdXRpbmcuXCIpO1xuICAgICAgd2hpbGUgKGcgJiYgKGcgPSAwLCBvcFswXSAmJiAoXyA9IDApKSwgXykgdHJ5IHtcbiAgICAgICAgICBpZiAoZiA9IDEsIHkgJiYgKHQgPSBvcFswXSAmIDIgPyB5W1wicmV0dXJuXCJdIDogb3BbMF0gPyB5W1widGhyb3dcIl0gfHwgKCh0ID0geVtcInJldHVyblwiXSkgJiYgdC5jYWxsKHkpLCAwKSA6IHkubmV4dCkgJiYgISh0ID0gdC5jYWxsKHksIG9wWzFdKSkuZG9uZSkgcmV0dXJuIHQ7XG4gICAgICAgICAgaWYgKHkgPSAwLCB0KSBvcCA9IFtvcFswXSAmIDIsIHQudmFsdWVdO1xuICAgICAgICAgIHN3aXRjaCAob3BbMF0pIHtcbiAgICAgICAgICAgICAgY2FzZSAwOiBjYXNlIDE6IHQgPSBvcDsgYnJlYWs7XG4gICAgICAgICAgICAgIGNhc2UgNDogXy5sYWJlbCsrOyByZXR1cm4geyB2YWx1ZTogb3BbMV0sIGRvbmU6IGZhbHNlIH07XG4gICAgICAgICAgICAgIGNhc2UgNTogXy5sYWJlbCsrOyB5ID0gb3BbMV07IG9wID0gWzBdOyBjb250aW51ZTtcbiAgICAgICAgICAgICAgY2FzZSA3OiBvcCA9IF8ub3BzLnBvcCgpOyBfLnRyeXMucG9wKCk7IGNvbnRpbnVlO1xuICAgICAgICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgICAgICAgICAgaWYgKCEodCA9IF8udHJ5cywgdCA9IHQubGVuZ3RoID4gMCAmJiB0W3QubGVuZ3RoIC0gMV0pICYmIChvcFswXSA9PT0gNiB8fCBvcFswXSA9PT0gMikpIHsgXyA9IDA7IGNvbnRpbnVlOyB9XG4gICAgICAgICAgICAgICAgICBpZiAob3BbMF0gPT09IDMgJiYgKCF0IHx8IChvcFsxXSA+IHRbMF0gJiYgb3BbMV0gPCB0WzNdKSkpIHsgXy5sYWJlbCA9IG9wWzFdOyBicmVhazsgfVxuICAgICAgICAgICAgICAgICAgaWYgKG9wWzBdID09PSA2ICYmIF8ubGFiZWwgPCB0WzFdKSB7IF8ubGFiZWwgPSB0WzFdOyB0ID0gb3A7IGJyZWFrOyB9XG4gICAgICAgICAgICAgICAgICBpZiAodCAmJiBfLmxhYmVsIDwgdFsyXSkgeyBfLmxhYmVsID0gdFsyXTsgXy5vcHMucHVzaChvcCk7IGJyZWFrOyB9XG4gICAgICAgICAgICAgICAgICBpZiAodFsyXSkgXy5vcHMucG9wKCk7XG4gICAgICAgICAgICAgICAgICBfLnRyeXMucG9wKCk7IGNvbnRpbnVlO1xuICAgICAgICAgIH1cbiAgICAgICAgICBvcCA9IGJvZHkuY2FsbCh0aGlzQXJnLCBfKTtcbiAgICAgIH0gY2F0Y2ggKGUpIHsgb3AgPSBbNiwgZV07IHkgPSAwOyB9IGZpbmFsbHkgeyBmID0gdCA9IDA7IH1cbiAgICAgIGlmIChvcFswXSAmIDUpIHRocm93IG9wWzFdOyByZXR1cm4geyB2YWx1ZTogb3BbMF0gPyBvcFsxXSA6IHZvaWQgMCwgZG9uZTogdHJ1ZSB9O1xuICB9XG59XG5cbmV4cG9ydCB2YXIgX19jcmVhdGVCaW5kaW5nID0gT2JqZWN0LmNyZWF0ZSA/IChmdW5jdGlvbihvLCBtLCBrLCBrMikge1xuICBpZiAoazIgPT09IHVuZGVmaW5lZCkgazIgPSBrO1xuICB2YXIgZGVzYyA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IobSwgayk7XG4gIGlmICghZGVzYyB8fCAoXCJnZXRcIiBpbiBkZXNjID8gIW0uX19lc01vZHVsZSA6IGRlc2Mud3JpdGFibGUgfHwgZGVzYy5jb25maWd1cmFibGUpKSB7XG4gICAgICBkZXNjID0geyBlbnVtZXJhYmxlOiB0cnVlLCBnZXQ6IGZ1bmN0aW9uKCkgeyByZXR1cm4gbVtrXTsgfSB9O1xuICB9XG4gIE9iamVjdC5kZWZpbmVQcm9wZXJ0eShvLCBrMiwgZGVzYyk7XG59KSA6IChmdW5jdGlvbihvLCBtLCBrLCBrMikge1xuICBpZiAoazIgPT09IHVuZGVmaW5lZCkgazIgPSBrO1xuICBvW2syXSA9IG1ba107XG59KTtcblxuZXhwb3J0IGZ1bmN0aW9uIF9fZXhwb3J0U3RhcihtLCBvKSB7XG4gIGZvciAodmFyIHAgaW4gbSkgaWYgKHAgIT09IFwiZGVmYXVsdFwiICYmICFPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwobywgcCkpIF9fY3JlYXRlQmluZGluZyhvLCBtLCBwKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fdmFsdWVzKG8pIHtcbiAgdmFyIHMgPSB0eXBlb2YgU3ltYm9sID09PSBcImZ1bmN0aW9uXCIgJiYgU3ltYm9sLml0ZXJhdG9yLCBtID0gcyAmJiBvW3NdLCBpID0gMDtcbiAgaWYgKG0pIHJldHVybiBtLmNhbGwobyk7XG4gIGlmIChvICYmIHR5cGVvZiBvLmxlbmd0aCA9PT0gXCJudW1iZXJcIikgcmV0dXJuIHtcbiAgICAgIG5leHQ6IGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICBpZiAobyAmJiBpID49IG8ubGVuZ3RoKSBvID0gdm9pZCAwO1xuICAgICAgICAgIHJldHVybiB7IHZhbHVlOiBvICYmIG9baSsrXSwgZG9uZTogIW8gfTtcbiAgICAgIH1cbiAgfTtcbiAgdGhyb3cgbmV3IFR5cGVFcnJvcihzID8gXCJPYmplY3QgaXMgbm90IGl0ZXJhYmxlLlwiIDogXCJTeW1ib2wuaXRlcmF0b3IgaXMgbm90IGRlZmluZWQuXCIpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gX19yZWFkKG8sIG4pIHtcbiAgdmFyIG0gPSB0eXBlb2YgU3ltYm9sID09PSBcImZ1bmN0aW9uXCIgJiYgb1tTeW1ib2wuaXRlcmF0b3JdO1xuICBpZiAoIW0pIHJldHVybiBvO1xuICB2YXIgaSA9IG0uY2FsbChvKSwgciwgYXIgPSBbXSwgZTtcbiAgdHJ5IHtcbiAgICAgIHdoaWxlICgobiA9PT0gdm9pZCAwIHx8IG4tLSA+IDApICYmICEociA9IGkubmV4dCgpKS5kb25lKSBhci5wdXNoKHIudmFsdWUpO1xuICB9XG4gIGNhdGNoIChlcnJvcikgeyBlID0geyBlcnJvcjogZXJyb3IgfTsgfVxuICBmaW5hbGx5IHtcbiAgICAgIHRyeSB7XG4gICAgICAgICAgaWYgKHIgJiYgIXIuZG9uZSAmJiAobSA9IGlbXCJyZXR1cm5cIl0pKSBtLmNhbGwoaSk7XG4gICAgICB9XG4gICAgICBmaW5hbGx5IHsgaWYgKGUpIHRocm93IGUuZXJyb3I7IH1cbiAgfVxuICByZXR1cm4gYXI7XG59XG5cbi8qKiBAZGVwcmVjYXRlZCAqL1xuZXhwb3J0IGZ1bmN0aW9uIF9fc3ByZWFkKCkge1xuICBmb3IgKHZhciBhciA9IFtdLCBpID0gMDsgaSA8IGFyZ3VtZW50cy5sZW5ndGg7IGkrKylcbiAgICAgIGFyID0gYXIuY29uY2F0KF9fcmVhZChhcmd1bWVudHNbaV0pKTtcbiAgcmV0dXJuIGFyO1xufVxuXG4vKiogQGRlcHJlY2F0ZWQgKi9cbmV4cG9ydCBmdW5jdGlvbiBfX3NwcmVhZEFycmF5cygpIHtcbiAgZm9yICh2YXIgcyA9IDAsIGkgPSAwLCBpbCA9IGFyZ3VtZW50cy5sZW5ndGg7IGkgPCBpbDsgaSsrKSBzICs9IGFyZ3VtZW50c1tpXS5sZW5ndGg7XG4gIGZvciAodmFyIHIgPSBBcnJheShzKSwgayA9IDAsIGkgPSAwOyBpIDwgaWw7IGkrKylcbiAgICAgIGZvciAodmFyIGEgPSBhcmd1bWVudHNbaV0sIGogPSAwLCBqbCA9IGEubGVuZ3RoOyBqIDwgamw7IGorKywgaysrKVxuICAgICAgICAgIHJba10gPSBhW2pdO1xuICByZXR1cm4gcjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fc3ByZWFkQXJyYXkodG8sIGZyb20sIHBhY2spIHtcbiAgaWYgKHBhY2sgfHwgYXJndW1lbnRzLmxlbmd0aCA9PT0gMikgZm9yICh2YXIgaSA9IDAsIGwgPSBmcm9tLmxlbmd0aCwgYXI7IGkgPCBsOyBpKyspIHtcbiAgICAgIGlmIChhciB8fCAhKGkgaW4gZnJvbSkpIHtcbiAgICAgICAgICBpZiAoIWFyKSBhciA9IEFycmF5LnByb3RvdHlwZS5zbGljZS5jYWxsKGZyb20sIDAsIGkpO1xuICAgICAgICAgIGFyW2ldID0gZnJvbVtpXTtcbiAgICAgIH1cbiAgfVxuICByZXR1cm4gdG8uY29uY2F0KGFyIHx8IEFycmF5LnByb3RvdHlwZS5zbGljZS5jYWxsKGZyb20pKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fYXdhaXQodikge1xuICByZXR1cm4gdGhpcyBpbnN0YW5jZW9mIF9fYXdhaXQgPyAodGhpcy52ID0gdiwgdGhpcykgOiBuZXcgX19hd2FpdCh2KTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fYXN5bmNHZW5lcmF0b3IodGhpc0FyZywgX2FyZ3VtZW50cywgZ2VuZXJhdG9yKSB7XG4gIGlmICghU3ltYm9sLmFzeW5jSXRlcmF0b3IpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJTeW1ib2wuYXN5bmNJdGVyYXRvciBpcyBub3QgZGVmaW5lZC5cIik7XG4gIHZhciBnID0gZ2VuZXJhdG9yLmFwcGx5KHRoaXNBcmcsIF9hcmd1bWVudHMgfHwgW10pLCBpLCBxID0gW107XG4gIHJldHVybiBpID0gT2JqZWN0LmNyZWF0ZSgodHlwZW9mIEFzeW5jSXRlcmF0b3IgPT09IFwiZnVuY3Rpb25cIiA/IEFzeW5jSXRlcmF0b3IgOiBPYmplY3QpLnByb3RvdHlwZSksIHZlcmIoXCJuZXh0XCIpLCB2ZXJiKFwidGhyb3dcIiksIHZlcmIoXCJyZXR1cm5cIiwgYXdhaXRSZXR1cm4pLCBpW1N5bWJvbC5hc3luY0l0ZXJhdG9yXSA9IGZ1bmN0aW9uICgpIHsgcmV0dXJuIHRoaXM7IH0sIGk7XG4gIGZ1bmN0aW9uIGF3YWl0UmV0dXJuKGYpIHsgcmV0dXJuIGZ1bmN0aW9uICh2KSB7IHJldHVybiBQcm9taXNlLnJlc29sdmUodikudGhlbihmLCByZWplY3QpOyB9OyB9XG4gIGZ1bmN0aW9uIHZlcmIobiwgZikgeyBpZiAoZ1tuXSkgeyBpW25dID0gZnVuY3Rpb24gKHYpIHsgcmV0dXJuIG5ldyBQcm9taXNlKGZ1bmN0aW9uIChhLCBiKSB7IHEucHVzaChbbiwgdiwgYSwgYl0pID4gMSB8fCByZXN1bWUobiwgdik7IH0pOyB9OyBpZiAoZikgaVtuXSA9IGYoaVtuXSk7IH0gfVxuICBmdW5jdGlvbiByZXN1bWUobiwgdikgeyB0cnkgeyBzdGVwKGdbbl0odikpOyB9IGNhdGNoIChlKSB7IHNldHRsZShxWzBdWzNdLCBlKTsgfSB9XG4gIGZ1bmN0aW9uIHN0ZXAocikgeyByLnZhbHVlIGluc3RhbmNlb2YgX19hd2FpdCA/IFByb21pc2UucmVzb2x2ZShyLnZhbHVlLnYpLnRoZW4oZnVsZmlsbCwgcmVqZWN0KSA6IHNldHRsZShxWzBdWzJdLCByKTsgfVxuICBmdW5jdGlvbiBmdWxmaWxsKHZhbHVlKSB7IHJlc3VtZShcIm5leHRcIiwgdmFsdWUpOyB9XG4gIGZ1bmN0aW9uIHJlamVjdCh2YWx1ZSkgeyByZXN1bWUoXCJ0aHJvd1wiLCB2YWx1ZSk7IH1cbiAgZnVuY3Rpb24gc2V0dGxlKGYsIHYpIHsgaWYgKGYodiksIHEuc2hpZnQoKSwgcS5sZW5ndGgpIHJlc3VtZShxWzBdWzBdLCBxWzBdWzFdKTsgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gX19hc3luY0RlbGVnYXRvcihvKSB7XG4gIHZhciBpLCBwO1xuICByZXR1cm4gaSA9IHt9LCB2ZXJiKFwibmV4dFwiKSwgdmVyYihcInRocm93XCIsIGZ1bmN0aW9uIChlKSB7IHRocm93IGU7IH0pLCB2ZXJiKFwicmV0dXJuXCIpLCBpW1N5bWJvbC5pdGVyYXRvcl0gPSBmdW5jdGlvbiAoKSB7IHJldHVybiB0aGlzOyB9LCBpO1xuICBmdW5jdGlvbiB2ZXJiKG4sIGYpIHsgaVtuXSA9IG9bbl0gPyBmdW5jdGlvbiAodikgeyByZXR1cm4gKHAgPSAhcCkgPyB7IHZhbHVlOiBfX2F3YWl0KG9bbl0odikpLCBkb25lOiBmYWxzZSB9IDogZiA/IGYodikgOiB2OyB9IDogZjsgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gX19hc3luY1ZhbHVlcyhvKSB7XG4gIGlmICghU3ltYm9sLmFzeW5jSXRlcmF0b3IpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJTeW1ib2wuYXN5bmNJdGVyYXRvciBpcyBub3QgZGVmaW5lZC5cIik7XG4gIHZhciBtID0gb1tTeW1ib2wuYXN5bmNJdGVyYXRvcl0sIGk7XG4gIHJldHVybiBtID8gbS5jYWxsKG8pIDogKG8gPSB0eXBlb2YgX192YWx1ZXMgPT09IFwiZnVuY3Rpb25cIiA/IF9fdmFsdWVzKG8pIDogb1tTeW1ib2wuaXRlcmF0b3JdKCksIGkgPSB7fSwgdmVyYihcIm5leHRcIiksIHZlcmIoXCJ0aHJvd1wiKSwgdmVyYihcInJldHVyblwiKSwgaVtTeW1ib2wuYXN5bmNJdGVyYXRvcl0gPSBmdW5jdGlvbiAoKSB7IHJldHVybiB0aGlzOyB9LCBpKTtcbiAgZnVuY3Rpb24gdmVyYihuKSB7IGlbbl0gPSBvW25dICYmIGZ1bmN0aW9uICh2KSB7IHJldHVybiBuZXcgUHJvbWlzZShmdW5jdGlvbiAocmVzb2x2ZSwgcmVqZWN0KSB7IHYgPSBvW25dKHYpLCBzZXR0bGUocmVzb2x2ZSwgcmVqZWN0LCB2LmRvbmUsIHYudmFsdWUpOyB9KTsgfTsgfVxuICBmdW5jdGlvbiBzZXR0bGUocmVzb2x2ZSwgcmVqZWN0LCBkLCB2KSB7IFByb21pc2UucmVzb2x2ZSh2KS50aGVuKGZ1bmN0aW9uKHYpIHsgcmVzb2x2ZSh7IHZhbHVlOiB2LCBkb25lOiBkIH0pOyB9LCByZWplY3QpOyB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX21ha2VUZW1wbGF0ZU9iamVjdChjb29rZWQsIHJhdykge1xuICBpZiAoT2JqZWN0LmRlZmluZVByb3BlcnR5KSB7IE9iamVjdC5kZWZpbmVQcm9wZXJ0eShjb29rZWQsIFwicmF3XCIsIHsgdmFsdWU6IHJhdyB9KTsgfSBlbHNlIHsgY29va2VkLnJhdyA9IHJhdzsgfVxuICByZXR1cm4gY29va2VkO1xufTtcblxudmFyIF9fc2V0TW9kdWxlRGVmYXVsdCA9IE9iamVjdC5jcmVhdGUgPyAoZnVuY3Rpb24obywgdikge1xuICBPYmplY3QuZGVmaW5lUHJvcGVydHkobywgXCJkZWZhdWx0XCIsIHsgZW51bWVyYWJsZTogdHJ1ZSwgdmFsdWU6IHYgfSk7XG59KSA6IGZ1bmN0aW9uKG8sIHYpIHtcbiAgb1tcImRlZmF1bHRcIl0gPSB2O1xufTtcblxudmFyIG93bktleXMgPSBmdW5jdGlvbihvKSB7XG4gIG93bktleXMgPSBPYmplY3QuZ2V0T3duUHJvcGVydHlOYW1lcyB8fCBmdW5jdGlvbiAobykge1xuICAgIHZhciBhciA9IFtdO1xuICAgIGZvciAodmFyIGsgaW4gbykgaWYgKE9iamVjdC5wcm90b3R5cGUuaGFzT3duUHJvcGVydHkuY2FsbChvLCBrKSkgYXJbYXIubGVuZ3RoXSA9IGs7XG4gICAgcmV0dXJuIGFyO1xuICB9O1xuICByZXR1cm4gb3duS2V5cyhvKTtcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2ltcG9ydFN0YXIobW9kKSB7XG4gIGlmIChtb2QgJiYgbW9kLl9fZXNNb2R1bGUpIHJldHVybiBtb2Q7XG4gIHZhciByZXN1bHQgPSB7fTtcbiAgaWYgKG1vZCAhPSBudWxsKSBmb3IgKHZhciBrID0gb3duS2V5cyhtb2QpLCBpID0gMDsgaSA8IGsubGVuZ3RoOyBpKyspIGlmIChrW2ldICE9PSBcImRlZmF1bHRcIikgX19jcmVhdGVCaW5kaW5nKHJlc3VsdCwgbW9kLCBrW2ldKTtcbiAgX19zZXRNb2R1bGVEZWZhdWx0KHJlc3VsdCwgbW9kKTtcbiAgcmV0dXJuIHJlc3VsdDtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9faW1wb3J0RGVmYXVsdChtb2QpIHtcbiAgcmV0dXJuIChtb2QgJiYgbW9kLl9fZXNNb2R1bGUpID8gbW9kIDogeyBkZWZhdWx0OiBtb2QgfTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fY2xhc3NQcml2YXRlRmllbGRHZXQocmVjZWl2ZXIsIHN0YXRlLCBraW5kLCBmKSB7XG4gIGlmIChraW5kID09PSBcImFcIiAmJiAhZikgdGhyb3cgbmV3IFR5cGVFcnJvcihcIlByaXZhdGUgYWNjZXNzb3Igd2FzIGRlZmluZWQgd2l0aG91dCBhIGdldHRlclwiKTtcbiAgaWYgKHR5cGVvZiBzdGF0ZSA9PT0gXCJmdW5jdGlvblwiID8gcmVjZWl2ZXIgIT09IHN0YXRlIHx8ICFmIDogIXN0YXRlLmhhcyhyZWNlaXZlcikpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJDYW5ub3QgcmVhZCBwcml2YXRlIG1lbWJlciBmcm9tIGFuIG9iamVjdCB3aG9zZSBjbGFzcyBkaWQgbm90IGRlY2xhcmUgaXRcIik7XG4gIHJldHVybiBraW5kID09PSBcIm1cIiA/IGYgOiBraW5kID09PSBcImFcIiA/IGYuY2FsbChyZWNlaXZlcikgOiBmID8gZi52YWx1ZSA6IHN0YXRlLmdldChyZWNlaXZlcik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2NsYXNzUHJpdmF0ZUZpZWxkU2V0KHJlY2VpdmVyLCBzdGF0ZSwgdmFsdWUsIGtpbmQsIGYpIHtcbiAgaWYgKGtpbmQgPT09IFwibVwiKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiUHJpdmF0ZSBtZXRob2QgaXMgbm90IHdyaXRhYmxlXCIpO1xuICBpZiAoa2luZCA9PT0gXCJhXCIgJiYgIWYpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJQcml2YXRlIGFjY2Vzc29yIHdhcyBkZWZpbmVkIHdpdGhvdXQgYSBzZXR0ZXJcIik7XG4gIGlmICh0eXBlb2Ygc3RhdGUgPT09IFwiZnVuY3Rpb25cIiA/IHJlY2VpdmVyICE9PSBzdGF0ZSB8fCAhZiA6ICFzdGF0ZS5oYXMocmVjZWl2ZXIpKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiQ2Fubm90IHdyaXRlIHByaXZhdGUgbWVtYmVyIHRvIGFuIG9iamVjdCB3aG9zZSBjbGFzcyBkaWQgbm90IGRlY2xhcmUgaXRcIik7XG4gIHJldHVybiAoa2luZCA9PT0gXCJhXCIgPyBmLmNhbGwocmVjZWl2ZXIsIHZhbHVlKSA6IGYgPyBmLnZhbHVlID0gdmFsdWUgOiBzdGF0ZS5zZXQocmVjZWl2ZXIsIHZhbHVlKSksIHZhbHVlO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gX19jbGFzc1ByaXZhdGVGaWVsZEluKHN0YXRlLCByZWNlaXZlcikge1xuICBpZiAocmVjZWl2ZXIgPT09IG51bGwgfHwgKHR5cGVvZiByZWNlaXZlciAhPT0gXCJvYmplY3RcIiAmJiB0eXBlb2YgcmVjZWl2ZXIgIT09IFwiZnVuY3Rpb25cIikpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJDYW5ub3QgdXNlICdpbicgb3BlcmF0b3Igb24gbm9uLW9iamVjdFwiKTtcbiAgcmV0dXJuIHR5cGVvZiBzdGF0ZSA9PT0gXCJmdW5jdGlvblwiID8gcmVjZWl2ZXIgPT09IHN0YXRlIDogc3RhdGUuaGFzKHJlY2VpdmVyKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fYWRkRGlzcG9zYWJsZVJlc291cmNlKGVudiwgdmFsdWUsIGFzeW5jKSB7XG4gIGlmICh2YWx1ZSAhPT0gbnVsbCAmJiB2YWx1ZSAhPT0gdm9pZCAwKSB7XG4gICAgaWYgKHR5cGVvZiB2YWx1ZSAhPT0gXCJvYmplY3RcIiAmJiB0eXBlb2YgdmFsdWUgIT09IFwiZnVuY3Rpb25cIikgdGhyb3cgbmV3IFR5cGVFcnJvcihcIk9iamVjdCBleHBlY3RlZC5cIik7XG4gICAgdmFyIGRpc3Bvc2UsIGlubmVyO1xuICAgIGlmIChhc3luYykge1xuICAgICAgaWYgKCFTeW1ib2wuYXN5bmNEaXNwb3NlKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiU3ltYm9sLmFzeW5jRGlzcG9zZSBpcyBub3QgZGVmaW5lZC5cIik7XG4gICAgICBkaXNwb3NlID0gdmFsdWVbU3ltYm9sLmFzeW5jRGlzcG9zZV07XG4gICAgfVxuICAgIGlmIChkaXNwb3NlID09PSB2b2lkIDApIHtcbiAgICAgIGlmICghU3ltYm9sLmRpc3Bvc2UpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJTeW1ib2wuZGlzcG9zZSBpcyBub3QgZGVmaW5lZC5cIik7XG4gICAgICBkaXNwb3NlID0gdmFsdWVbU3ltYm9sLmRpc3Bvc2VdO1xuICAgICAgaWYgKGFzeW5jKSBpbm5lciA9IGRpc3Bvc2U7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgZGlzcG9zZSAhPT0gXCJmdW5jdGlvblwiKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiT2JqZWN0IG5vdCBkaXNwb3NhYmxlLlwiKTtcbiAgICBpZiAoaW5uZXIpIGRpc3Bvc2UgPSBmdW5jdGlvbigpIHsgdHJ5IHsgaW5uZXIuY2FsbCh0aGlzKTsgfSBjYXRjaCAoZSkgeyByZXR1cm4gUHJvbWlzZS5yZWplY3QoZSk7IH0gfTtcbiAgICBlbnYuc3RhY2sucHVzaCh7IHZhbHVlOiB2YWx1ZSwgZGlzcG9zZTogZGlzcG9zZSwgYXN5bmM6IGFzeW5jIH0pO1xuICB9XG4gIGVsc2UgaWYgKGFzeW5jKSB7XG4gICAgZW52LnN0YWNrLnB1c2goeyBhc3luYzogdHJ1ZSB9KTtcbiAgfVxuICByZXR1cm4gdmFsdWU7XG59XG5cbnZhciBfU3VwcHJlc3NlZEVycm9yID0gdHlwZW9mIFN1cHByZXNzZWRFcnJvciA9PT0gXCJmdW5jdGlvblwiID8gU3VwcHJlc3NlZEVycm9yIDogZnVuY3Rpb24gKGVycm9yLCBzdXBwcmVzc2VkLCBtZXNzYWdlKSB7XG4gIHZhciBlID0gbmV3IEVycm9yKG1lc3NhZ2UpO1xuICByZXR1cm4gZS5uYW1lID0gXCJTdXBwcmVzc2VkRXJyb3JcIiwgZS5lcnJvciA9IGVycm9yLCBlLnN1cHByZXNzZWQgPSBzdXBwcmVzc2VkLCBlO1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIF9fZGlzcG9zZVJlc291cmNlcyhlbnYpIHtcbiAgZnVuY3Rpb24gZmFpbChlKSB7XG4gICAgZW52LmVycm9yID0gZW52Lmhhc0Vycm9yID8gbmV3IF9TdXBwcmVzc2VkRXJyb3IoZSwgZW52LmVycm9yLCBcIkFuIGVycm9yIHdhcyBzdXBwcmVzc2VkIGR1cmluZyBkaXNwb3NhbC5cIikgOiBlO1xuICAgIGVudi5oYXNFcnJvciA9IHRydWU7XG4gIH1cbiAgdmFyIHIsIHMgPSAwO1xuICBmdW5jdGlvbiBuZXh0KCkge1xuICAgIHdoaWxlIChyID0gZW52LnN0YWNrLnBvcCgpKSB7XG4gICAgICB0cnkge1xuICAgICAgICBpZiAoIXIuYXN5bmMgJiYgcyA9PT0gMSkgcmV0dXJuIHMgPSAwLCBlbnYuc3RhY2sucHVzaChyKSwgUHJvbWlzZS5yZXNvbHZlKCkudGhlbihuZXh0KTtcbiAgICAgICAgaWYgKHIuZGlzcG9zZSkge1xuICAgICAgICAgIHZhciByZXN1bHQgPSByLmRpc3Bvc2UuY2FsbChyLnZhbHVlKTtcbiAgICAgICAgICBpZiAoci5hc3luYykgcmV0dXJuIHMgfD0gMiwgUHJvbWlzZS5yZXNvbHZlKHJlc3VsdCkudGhlbihuZXh0LCBmdW5jdGlvbihlKSB7IGZhaWwoZSk7IHJldHVybiBuZXh0KCk7IH0pO1xuICAgICAgICB9XG4gICAgICAgIGVsc2UgcyB8PSAxO1xuICAgICAgfVxuICAgICAgY2F0Y2ggKGUpIHtcbiAgICAgICAgZmFpbChlKTtcbiAgICAgIH1cbiAgICB9XG4gICAgaWYgKHMgPT09IDEpIHJldHVybiBlbnYuaGFzRXJyb3IgPyBQcm9taXNlLnJlamVjdChlbnYuZXJyb3IpIDogUHJvbWlzZS5yZXNvbHZlKCk7XG4gICAgaWYgKGVudi5oYXNFcnJvcikgdGhyb3cgZW52LmVycm9yO1xuICB9XG4gIHJldHVybiBuZXh0KCk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX3Jld3JpdGVSZWxhdGl2ZUltcG9ydEV4dGVuc2lvbihwYXRoLCBwcmVzZXJ2ZUpzeCkge1xuICBpZiAodHlwZW9mIHBhdGggPT09IFwic3RyaW5nXCIgJiYgL15cXC5cXC4/XFwvLy50ZXN0KHBhdGgpKSB7XG4gICAgICByZXR1cm4gcGF0aC5yZXBsYWNlKC9cXC4odHN4KSR8KCg/OlxcLmQpPykoKD86XFwuW14uL10rPyk/KVxcLihbY21dPyl0cyQvaSwgZnVuY3Rpb24gKG0sIHRzeCwgZCwgZXh0LCBjbSkge1xuICAgICAgICAgIHJldHVybiB0c3ggPyBwcmVzZXJ2ZUpzeCA/IFwiLmpzeFwiIDogXCIuanNcIiA6IGQgJiYgKCFleHQgfHwgIWNtKSA/IG0gOiAoZCArIGV4dCArIFwiLlwiICsgY20udG9Mb3dlckNhc2UoKSArIFwianNcIik7XG4gICAgICB9KTtcbiAgfVxuICByZXR1cm4gcGF0aDtcbn1cblxuZXhwb3J0IGRlZmF1bHQge1xuICBfX2V4dGVuZHMsXG4gIF9fYXNzaWduLFxuICBfX3Jlc3QsXG4gIF9fZGVjb3JhdGUsXG4gIF9fcGFyYW0sXG4gIF9fZXNEZWNvcmF0ZSxcbiAgX19ydW5Jbml0aWFsaXplcnMsXG4gIF9fcHJvcEtleSxcbiAgX19zZXRGdW5jdGlvbk5hbWUsXG4gIF9fbWV0YWRhdGEsXG4gIF9fYXdhaXRlcixcbiAgX19nZW5lcmF0b3IsXG4gIF9fY3JlYXRlQmluZGluZyxcbiAgX19leHBvcnRTdGFyLFxuICBfX3ZhbHVlcyxcbiAgX19yZWFkLFxuICBfX3NwcmVhZCxcbiAgX19zcHJlYWRBcnJheXMsXG4gIF9fc3ByZWFkQXJyYXksXG4gIF9fYXdhaXQsXG4gIF9fYXN5bmNHZW5lcmF0b3IsXG4gIF9fYXN5bmNEZWxlZ2F0b3IsXG4gIF9fYXN5bmNWYWx1ZXMsXG4gIF9fbWFrZVRlbXBsYXRlT2JqZWN0LFxuICBfX2ltcG9ydFN0YXIsXG4gIF9faW1wb3J0RGVmYXVsdCxcbiAgX19jbGFzc1ByaXZhdGVGaWVsZEdldCxcbiAgX19jbGFzc1ByaXZhdGVGaWVsZFNldCxcbiAgX19jbGFzc1ByaXZhdGVGaWVsZEluLFxuICBfX2FkZERpc3Bvc2FibGVSZXNvdXJjZSxcbiAgX19kaXNwb3NlUmVzb3VyY2VzLFxuICBfX3Jld3JpdGVSZWxhdGl2ZUltcG9ydEV4dGVuc2lvbixcbn07XG4iLCIvLyBUaGUgbW9kdWxlIGNhY2hlXG52YXIgX193ZWJwYWNrX21vZHVsZV9jYWNoZV9fID0ge307XG5cbi8vIFRoZSByZXF1aXJlIGZ1bmN0aW9uXG5mdW5jdGlvbiBfX3dlYnBhY2tfcmVxdWlyZV9fKG1vZHVsZUlkKSB7XG5cdC8vIENoZWNrIGlmIG1vZHVsZSBpcyBpbiBjYWNoZVxuXHR2YXIgY2FjaGVkTW9kdWxlID0gX193ZWJwYWNrX21vZHVsZV9jYWNoZV9fW21vZHVsZUlkXTtcblx0aWYgKGNhY2hlZE1vZHVsZSAhPT0gdW5kZWZpbmVkKSB7XG5cdFx0cmV0dXJuIGNhY2hlZE1vZHVsZS5leHBvcnRzO1xuXHR9XG5cdC8vIENyZWF0ZSBhIG5ldyBtb2R1bGUgKGFuZCBwdXQgaXQgaW50byB0aGUgY2FjaGUpXG5cdHZhciBtb2R1bGUgPSBfX3dlYnBhY2tfbW9kdWxlX2NhY2hlX19bbW9kdWxlSWRdID0ge1xuXHRcdC8vIG5vIG1vZHVsZS5pZCBuZWVkZWRcblx0XHQvLyBubyBtb2R1bGUubG9hZGVkIG5lZWRlZFxuXHRcdGV4cG9ydHM6IHt9XG5cdH07XG5cblx0Ly8gRXhlY3V0ZSB0aGUgbW9kdWxlIGZ1bmN0aW9uXG5cdF9fd2VicGFja19tb2R1bGVzX19bbW9kdWxlSWRdKG1vZHVsZSwgbW9kdWxlLmV4cG9ydHMsIF9fd2VicGFja19yZXF1aXJlX18pO1xuXG5cdC8vIFJldHVybiB0aGUgZXhwb3J0cyBvZiB0aGUgbW9kdWxlXG5cdHJldHVybiBtb2R1bGUuZXhwb3J0cztcbn1cblxuIiwiLy8gZ2V0RGVmYXVsdEV4cG9ydCBmdW5jdGlvbiBmb3IgY29tcGF0aWJpbGl0eSB3aXRoIG5vbi1oYXJtb255IG1vZHVsZXNcbl9fd2VicGFja19yZXF1aXJlX18ubiA9IChtb2R1bGUpID0+IHtcblx0dmFyIGdldHRlciA9IG1vZHVsZSAmJiBtb2R1bGUuX19lc01vZHVsZSA/XG5cdFx0KCkgPT4gKG1vZHVsZVsnZGVmYXVsdCddKSA6XG5cdFx0KCkgPT4gKG1vZHVsZSk7XG5cdF9fd2VicGFja19yZXF1aXJlX18uZChnZXR0ZXIsIHsgYTogZ2V0dGVyIH0pO1xuXHRyZXR1cm4gZ2V0dGVyO1xufTsiLCIvLyBkZWZpbmUgZ2V0dGVyIGZ1bmN0aW9ucyBmb3IgaGFybW9ueSBleHBvcnRzXG5fX3dlYnBhY2tfcmVxdWlyZV9fLmQgPSAoZXhwb3J0cywgZGVmaW5pdGlvbikgPT4ge1xuXHRmb3IodmFyIGtleSBpbiBkZWZpbml0aW9uKSB7XG5cdFx0aWYoX193ZWJwYWNrX3JlcXVpcmVfXy5vKGRlZmluaXRpb24sIGtleSkgJiYgIV9fd2VicGFja19yZXF1aXJlX18ubyhleHBvcnRzLCBrZXkpKSB7XG5cdFx0XHRPYmplY3QuZGVmaW5lUHJvcGVydHkoZXhwb3J0cywga2V5LCB7IGVudW1lcmFibGU6IHRydWUsIGdldDogZGVmaW5pdGlvbltrZXldIH0pO1xuXHRcdH1cblx0fVxufTsiLCJfX3dlYnBhY2tfcmVxdWlyZV9fLm8gPSAob2JqLCBwcm9wKSA9PiAoT2JqZWN0LnByb3RvdHlwZS5oYXNPd25Qcm9wZXJ0eS5jYWxsKG9iaiwgcHJvcCkpIiwiLy8gZGVmaW5lIF9fZXNNb2R1bGUgb24gZXhwb3J0c1xuX193ZWJwYWNrX3JlcXVpcmVfXy5yID0gKGV4cG9ydHMpID0+IHtcblx0aWYodHlwZW9mIFN5bWJvbCAhPT0gJ3VuZGVmaW5lZCcgJiYgU3ltYm9sLnRvU3RyaW5nVGFnKSB7XG5cdFx0T2JqZWN0LmRlZmluZVByb3BlcnR5KGV4cG9ydHMsIFN5bWJvbC50b1N0cmluZ1RhZywgeyB2YWx1ZTogJ01vZHVsZScgfSk7XG5cdH1cblx0T2JqZWN0LmRlZmluZVByb3BlcnR5KGV4cG9ydHMsICdfX2VzTW9kdWxlJywgeyB2YWx1ZTogdHJ1ZSB9KTtcbn07IiwiLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGltcG9ydC9uby1pbnRlcm5hbC1tb2R1bGVzXHJcbmltcG9ydCAqIGFzIGFkZG9ucyBmcm9tIFwiYWRkb25zL2luZGV4XCI7XHJcblxyXG5leHBvcnQgeyBhZGRvbnMgfTtcclxuZXhwb3J0IGRlZmF1bHQgYWRkb25zO1xyXG4iXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=
1950
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFieWxvbmpzLmFkZG9ucy5qcyIsIm1hcHBpbmdzIjoiQUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDVkE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7Ozs7Ozs7Ozs7Ozs7O0FDM1lBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQzlGQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUVBOzs7Ozs7Ozs7QUFTQTtBQUNBO0FBQUE7QUFxREE7Ozs7O0FBS0E7QUFDQTtBQUFBO0FBQ0E7QUFwREE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7O0FBRUE7QUFDQTtBQUVBO0FBTUE7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQWtCQTtBQVdBO0FBQ0E7QUFDQTs7QUFFQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFDQTtBQS9FQTtBQUhBOztBQUVBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQWdDQTtBQUhBOztBQUVBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQUtBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7OztBQUFBO0FBeUNBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7OztBQUFBO0FBS0E7QUFIQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7O0FBQUE7QUFLQTtBQUhBOztBQUVBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQU1BO0FBSkE7OztBQUdBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQUtBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7OztBQUFBO0FBRUE7O0FBRUE7QUFDQTs7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTs7Ozs7OztBQU9BO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7Ozs7QUFJQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQUE7O0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFBQTs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDNVdBO0FBR0E7QUFFQTtBQUdBO0FBR0E7QUFDQTtBQWNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTs7Ozs7Ozs7O0FBU0E7QUFDQTtBQTRDQTs7Ozs7QUFLQTtBQUNBO0FBRUE7QUFGQTtBQTVDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUlBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFvaUJBO0FBQ0E7QUFDQTtBQUNBO0FBN2dCQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7O0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFBQTs7QUFRQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFFQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTs7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBaUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBaUNBO0FBQ0E7QUFFQTs7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBUUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7O0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFBQTs7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFBQTs7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFHQTtBQUNBO0FBQ0E7QUFPQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDbnFCQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDTkE7QUFlQTtBQUVBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7Ozs7QUFJQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOzs7Ozs7OztBQVFBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOzs7Ozs7OztBQVFBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBVUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUdBO0FBQ0E7QUFDQTs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDeExBO0FBQ0E7QUFFQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUVBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOzs7O0FBSUE7QUFDQTtBQW1CQTtBQUdBO0FBRkE7QUFDQTtBQXBCQTtBQUNBO0FBc0JBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBcEJBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7OztBQUpBO0FBdUJBO0FBSEE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7QUFBQTtBQUVBOztBQUVBO0FBQ0E7QUFFQTs7O0FBR0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDbE9BOzs7Ozs7Ozs7OztBQ0FBOzs7Ozs7QUNBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Ozs7QUN2QkE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7Ozs7QUNQQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7OztBQ1BBOzs7OztBQ0FBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7Ozs7Ozs7Ozs7Ozs7O0FDTkE7QUFDQTtBQUVBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9BRERPTlMvd2VicGFjay91bml2ZXJzYWxNb2R1bGVEZWZpbml0aW9uIiwid2VicGFjazovL0FERE9OUy8uLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvdHNsaWIvdHNsaWIuZXM2Lm1qcyIsIndlYnBhY2s6Ly9BRERPTlMvLi4vLi4vLi4vZGV2L2FkZG9ucy9zcmMvaHRtbE1lc2gvZml0U3RyYXRlZ3kudHMiLCJ3ZWJwYWNrOi8vQURET05TLy4uLy4uLy4uL2Rldi9hZGRvbnMvc3JjL2h0bWxNZXNoL2h0bWxNZXNoLnRzIiwid2VicGFjazovL0FERE9OUy8uLi8uLi8uLi9kZXYvYWRkb25zL3NyYy9odG1sTWVzaC9odG1sTWVzaFJlbmRlcmVyLnRzIiwid2VicGFjazovL0FERE9OUy8uLi8uLi8uLi9kZXYvYWRkb25zL3NyYy9odG1sTWVzaC9pbmRleC50cyIsIndlYnBhY2s6Ly9BRERPTlMvLi4vLi4vLi4vZGV2L2FkZG9ucy9zcmMvaHRtbE1lc2gvcG9pbnRlckV2ZW50c0NhcHR1cmUudHMiLCJ3ZWJwYWNrOi8vQURET05TLy4uLy4uLy4uL2Rldi9hZGRvbnMvc3JjL2h0bWxNZXNoL3BvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IudHMiLCJ3ZWJwYWNrOi8vQURET05TLy4uLy4uLy4uL2Rldi9hZGRvbnMvc3JjL2luZGV4LnRzIiwid2VicGFjazovL0FERE9OUy9leHRlcm5hbCB1bWQge1wicm9vdFwiOlwiQkFCWUxPTlwiLFwiY29tbW9uanNcIjpcImJhYnlsb25qc1wiLFwiY29tbW9uanMyXCI6XCJiYWJ5bG9uanNcIixcImFtZFwiOlwiYmFieWxvbmpzXCJ9Iiwid2VicGFjazovL0FERE9OUy93ZWJwYWNrL2Jvb3RzdHJhcCIsIndlYnBhY2s6Ly9BRERPTlMvd2VicGFjay9ydW50aW1lL2NvbXBhdCBnZXQgZGVmYXVsdCBleHBvcnQiLCJ3ZWJwYWNrOi8vQURET05TL3dlYnBhY2svcnVudGltZS9kZWZpbmUgcHJvcGVydHkgZ2V0dGVycyIsIndlYnBhY2s6Ly9BRERPTlMvd2VicGFjay9ydW50aW1lL2hhc093blByb3BlcnR5IHNob3J0aGFuZCIsIndlYnBhY2s6Ly9BRERPTlMvd2VicGFjay9ydW50aW1lL21ha2UgbmFtZXNwYWNlIG9iamVjdCIsIndlYnBhY2s6Ly9BRERPTlMvLi9zcmMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uIHdlYnBhY2tVbml2ZXJzYWxNb2R1bGVEZWZpbml0aW9uKHJvb3QsIGZhY3RvcnkpIHtcblx0aWYodHlwZW9mIGV4cG9ydHMgPT09ICdvYmplY3QnICYmIHR5cGVvZiBtb2R1bGUgPT09ICdvYmplY3QnKVxuXHRcdG1vZHVsZS5leHBvcnRzID0gZmFjdG9yeShyZXF1aXJlKFwiYmFieWxvbmpzXCIpKTtcblx0ZWxzZSBpZih0eXBlb2YgZGVmaW5lID09PSAnZnVuY3Rpb24nICYmIGRlZmluZS5hbWQpXG5cdFx0ZGVmaW5lKFwiYmFieWxvbmpzLWFkZG9uc1wiLCBbXCJiYWJ5bG9uanNcIl0sIGZhY3RvcnkpO1xuXHRlbHNlIGlmKHR5cGVvZiBleHBvcnRzID09PSAnb2JqZWN0Jylcblx0XHRleHBvcnRzW1wiYmFieWxvbmpzLWFkZG9uc1wiXSA9IGZhY3RvcnkocmVxdWlyZShcImJhYnlsb25qc1wiKSk7XG5cdGVsc2Vcblx0XHRyb290W1wiQURET05TXCJdID0gZmFjdG9yeShyb290W1wiQkFCWUxPTlwiXSk7XG59KSgodHlwZW9mIHNlbGYgIT09IFwidW5kZWZpbmVkXCIgPyBzZWxmIDogdHlwZW9mIGdsb2JhbCAhPT0gXCJ1bmRlZmluZWRcIiA/IGdsb2JhbCA6IHRoaXMpLCAoX19XRUJQQUNLX0VYVEVSTkFMX01PRFVMRV9iYWJ5bG9uanNfTWF0aHNfbWF0aF9fKSA9PiB7XG5yZXR1cm4gIiwiLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuQ29weXJpZ2h0IChjKSBNaWNyb3NvZnQgQ29ycG9yYXRpb24uXG5cblBlcm1pc3Npb24gdG8gdXNlLCBjb3B5LCBtb2RpZnksIGFuZC9vciBkaXN0cmlidXRlIHRoaXMgc29mdHdhcmUgZm9yIGFueVxucHVycG9zZSB3aXRoIG9yIHdpdGhvdXQgZmVlIGlzIGhlcmVieSBncmFudGVkLlxuXG5USEUgU09GVFdBUkUgSVMgUFJPVklERUQgXCJBUyBJU1wiIEFORCBUSEUgQVVUSE9SIERJU0NMQUlNUyBBTEwgV0FSUkFOVElFUyBXSVRIXG5SRUdBUkQgVE8gVEhJUyBTT0ZUV0FSRSBJTkNMVURJTkcgQUxMIElNUExJRUQgV0FSUkFOVElFUyBPRiBNRVJDSEFOVEFCSUxJVFlcbkFORCBGSVRORVNTLiBJTiBOTyBFVkVOVCBTSEFMTCBUSEUgQVVUSE9SIEJFIExJQUJMRSBGT1IgQU5ZIFNQRUNJQUwsIERJUkVDVCxcbklORElSRUNULCBPUiBDT05TRVFVRU5USUFMIERBTUFHRVMgT1IgQU5ZIERBTUFHRVMgV0hBVFNPRVZFUiBSRVNVTFRJTkcgRlJPTVxuTE9TUyBPRiBVU0UsIERBVEEgT1IgUFJPRklUUywgV0hFVEhFUiBJTiBBTiBBQ1RJT04gT0YgQ09OVFJBQ1QsIE5FR0xJR0VOQ0UgT1Jcbk9USEVSIFRPUlRJT1VTIEFDVElPTiwgQVJJU0lORyBPVVQgT0YgT1IgSU4gQ09OTkVDVElPTiBXSVRIIFRIRSBVU0UgT1JcblBFUkZPUk1BTkNFIE9GIFRISVMgU09GVFdBUkUuXG4qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiAqL1xuLyogZ2xvYmFsIFJlZmxlY3QsIFByb21pc2UsIFN1cHByZXNzZWRFcnJvciwgU3ltYm9sLCBJdGVyYXRvciAqL1xuXG52YXIgZXh0ZW5kU3RhdGljcyA9IGZ1bmN0aW9uKGQsIGIpIHtcbiAgZXh0ZW5kU3RhdGljcyA9IE9iamVjdC5zZXRQcm90b3R5cGVPZiB8fFxuICAgICAgKHsgX19wcm90b19fOiBbXSB9IGluc3RhbmNlb2YgQXJyYXkgJiYgZnVuY3Rpb24gKGQsIGIpIHsgZC5fX3Byb3RvX18gPSBiOyB9KSB8fFxuICAgICAgZnVuY3Rpb24gKGQsIGIpIHsgZm9yICh2YXIgcCBpbiBiKSBpZiAoT2JqZWN0LnByb3RvdHlwZS5oYXNPd25Qcm9wZXJ0eS5jYWxsKGIsIHApKSBkW3BdID0gYltwXTsgfTtcbiAgcmV0dXJuIGV4dGVuZFN0YXRpY3MoZCwgYik7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gX19leHRlbmRzKGQsIGIpIHtcbiAgaWYgKHR5cGVvZiBiICE9PSBcImZ1bmN0aW9uXCIgJiYgYiAhPT0gbnVsbClcbiAgICAgIHRocm93IG5ldyBUeXBlRXJyb3IoXCJDbGFzcyBleHRlbmRzIHZhbHVlIFwiICsgU3RyaW5nKGIpICsgXCIgaXMgbm90IGEgY29uc3RydWN0b3Igb3IgbnVsbFwiKTtcbiAgZXh0ZW5kU3RhdGljcyhkLCBiKTtcbiAgZnVuY3Rpb24gX18oKSB7IHRoaXMuY29uc3RydWN0b3IgPSBkOyB9XG4gIGQucHJvdG90eXBlID0gYiA9PT0gbnVsbCA/IE9iamVjdC5jcmVhdGUoYikgOiAoX18ucHJvdG90eXBlID0gYi5wcm90b3R5cGUsIG5ldyBfXygpKTtcbn1cblxuZXhwb3J0IHZhciBfX2Fzc2lnbiA9IGZ1bmN0aW9uKCkge1xuICBfX2Fzc2lnbiA9IE9iamVjdC5hc3NpZ24gfHwgZnVuY3Rpb24gX19hc3NpZ24odCkge1xuICAgICAgZm9yICh2YXIgcywgaSA9IDEsIG4gPSBhcmd1bWVudHMubGVuZ3RoOyBpIDwgbjsgaSsrKSB7XG4gICAgICAgICAgcyA9IGFyZ3VtZW50c1tpXTtcbiAgICAgICAgICBmb3IgKHZhciBwIGluIHMpIGlmIChPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwocywgcCkpIHRbcF0gPSBzW3BdO1xuICAgICAgfVxuICAgICAgcmV0dXJuIHQ7XG4gIH1cbiAgcmV0dXJuIF9fYXNzaWduLmFwcGx5KHRoaXMsIGFyZ3VtZW50cyk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX3Jlc3QocywgZSkge1xuICB2YXIgdCA9IHt9O1xuICBmb3IgKHZhciBwIGluIHMpIGlmIChPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwocywgcCkgJiYgZS5pbmRleE9mKHApIDwgMClcbiAgICAgIHRbcF0gPSBzW3BdO1xuICBpZiAocyAhPSBudWxsICYmIHR5cGVvZiBPYmplY3QuZ2V0T3duUHJvcGVydHlTeW1ib2xzID09PSBcImZ1bmN0aW9uXCIpXG4gICAgICBmb3IgKHZhciBpID0gMCwgcCA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eVN5bWJvbHMocyk7IGkgPCBwLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgaWYgKGUuaW5kZXhPZihwW2ldKSA8IDAgJiYgT2JqZWN0LnByb3RvdHlwZS5wcm9wZXJ0eUlzRW51bWVyYWJsZS5jYWxsKHMsIHBbaV0pKVxuICAgICAgICAgICAgICB0W3BbaV1dID0gc1twW2ldXTtcbiAgICAgIH1cbiAgcmV0dXJuIHQ7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2RlY29yYXRlKGRlY29yYXRvcnMsIHRhcmdldCwga2V5LCBkZXNjKSB7XG4gIHZhciBjID0gYXJndW1lbnRzLmxlbmd0aCwgciA9IGMgPCAzID8gdGFyZ2V0IDogZGVzYyA9PT0gbnVsbCA/IGRlc2MgPSBPYmplY3QuZ2V0T3duUHJvcGVydHlEZXNjcmlwdG9yKHRhcmdldCwga2V5KSA6IGRlc2MsIGQ7XG4gIGlmICh0eXBlb2YgUmVmbGVjdCA9PT0gXCJvYmplY3RcIiAmJiB0eXBlb2YgUmVmbGVjdC5kZWNvcmF0ZSA9PT0gXCJmdW5jdGlvblwiKSByID0gUmVmbGVjdC5kZWNvcmF0ZShkZWNvcmF0b3JzLCB0YXJnZXQsIGtleSwgZGVzYyk7XG4gIGVsc2UgZm9yICh2YXIgaSA9IGRlY29yYXRvcnMubGVuZ3RoIC0gMTsgaSA+PSAwOyBpLS0pIGlmIChkID0gZGVjb3JhdG9yc1tpXSkgciA9IChjIDwgMyA/IGQocikgOiBjID4gMyA/IGQodGFyZ2V0LCBrZXksIHIpIDogZCh0YXJnZXQsIGtleSkpIHx8IHI7XG4gIHJldHVybiBjID4gMyAmJiByICYmIE9iamVjdC5kZWZpbmVQcm9wZXJ0eSh0YXJnZXQsIGtleSwgciksIHI7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX3BhcmFtKHBhcmFtSW5kZXgsIGRlY29yYXRvcikge1xuICByZXR1cm4gZnVuY3Rpb24gKHRhcmdldCwga2V5KSB7IGRlY29yYXRvcih0YXJnZXQsIGtleSwgcGFyYW1JbmRleCk7IH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fZXNEZWNvcmF0ZShjdG9yLCBkZXNjcmlwdG9ySW4sIGRlY29yYXRvcnMsIGNvbnRleHRJbiwgaW5pdGlhbGl6ZXJzLCBleHRyYUluaXRpYWxpemVycykge1xuICBmdW5jdGlvbiBhY2NlcHQoZikgeyBpZiAoZiAhPT0gdm9pZCAwICYmIHR5cGVvZiBmICE9PSBcImZ1bmN0aW9uXCIpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJGdW5jdGlvbiBleHBlY3RlZFwiKTsgcmV0dXJuIGY7IH1cbiAgdmFyIGtpbmQgPSBjb250ZXh0SW4ua2luZCwga2V5ID0ga2luZCA9PT0gXCJnZXR0ZXJcIiA/IFwiZ2V0XCIgOiBraW5kID09PSBcInNldHRlclwiID8gXCJzZXRcIiA6IFwidmFsdWVcIjtcbiAgdmFyIHRhcmdldCA9ICFkZXNjcmlwdG9ySW4gJiYgY3RvciA/IGNvbnRleHRJbltcInN0YXRpY1wiXSA/IGN0b3IgOiBjdG9yLnByb3RvdHlwZSA6IG51bGw7XG4gIHZhciBkZXNjcmlwdG9yID0gZGVzY3JpcHRvckluIHx8ICh0YXJnZXQgPyBPYmplY3QuZ2V0T3duUHJvcGVydHlEZXNjcmlwdG9yKHRhcmdldCwgY29udGV4dEluLm5hbWUpIDoge30pO1xuICB2YXIgXywgZG9uZSA9IGZhbHNlO1xuICBmb3IgKHZhciBpID0gZGVjb3JhdG9ycy5sZW5ndGggLSAxOyBpID49IDA7IGktLSkge1xuICAgICAgdmFyIGNvbnRleHQgPSB7fTtcbiAgICAgIGZvciAodmFyIHAgaW4gY29udGV4dEluKSBjb250ZXh0W3BdID0gcCA9PT0gXCJhY2Nlc3NcIiA/IHt9IDogY29udGV4dEluW3BdO1xuICAgICAgZm9yICh2YXIgcCBpbiBjb250ZXh0SW4uYWNjZXNzKSBjb250ZXh0LmFjY2Vzc1twXSA9IGNvbnRleHRJbi5hY2Nlc3NbcF07XG4gICAgICBjb250ZXh0LmFkZEluaXRpYWxpemVyID0gZnVuY3Rpb24gKGYpIHsgaWYgKGRvbmUpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJDYW5ub3QgYWRkIGluaXRpYWxpemVycyBhZnRlciBkZWNvcmF0aW9uIGhhcyBjb21wbGV0ZWRcIik7IGV4dHJhSW5pdGlhbGl6ZXJzLnB1c2goYWNjZXB0KGYgfHwgbnVsbCkpOyB9O1xuICAgICAgdmFyIHJlc3VsdCA9ICgwLCBkZWNvcmF0b3JzW2ldKShraW5kID09PSBcImFjY2Vzc29yXCIgPyB7IGdldDogZGVzY3JpcHRvci5nZXQsIHNldDogZGVzY3JpcHRvci5zZXQgfSA6IGRlc2NyaXB0b3Jba2V5XSwgY29udGV4dCk7XG4gICAgICBpZiAoa2luZCA9PT0gXCJhY2Nlc3NvclwiKSB7XG4gICAgICAgICAgaWYgKHJlc3VsdCA9PT0gdm9pZCAwKSBjb250aW51ZTtcbiAgICAgICAgICBpZiAocmVzdWx0ID09PSBudWxsIHx8IHR5cGVvZiByZXN1bHQgIT09IFwib2JqZWN0XCIpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJPYmplY3QgZXhwZWN0ZWRcIik7XG4gICAgICAgICAgaWYgKF8gPSBhY2NlcHQocmVzdWx0LmdldCkpIGRlc2NyaXB0b3IuZ2V0ID0gXztcbiAgICAgICAgICBpZiAoXyA9IGFjY2VwdChyZXN1bHQuc2V0KSkgZGVzY3JpcHRvci5zZXQgPSBfO1xuICAgICAgICAgIGlmIChfID0gYWNjZXB0KHJlc3VsdC5pbml0KSkgaW5pdGlhbGl6ZXJzLnVuc2hpZnQoXyk7XG4gICAgICB9XG4gICAgICBlbHNlIGlmIChfID0gYWNjZXB0KHJlc3VsdCkpIHtcbiAgICAgICAgICBpZiAoa2luZCA9PT0gXCJmaWVsZFwiKSBpbml0aWFsaXplcnMudW5zaGlmdChfKTtcbiAgICAgICAgICBlbHNlIGRlc2NyaXB0b3Jba2V5XSA9IF87XG4gICAgICB9XG4gIH1cbiAgaWYgKHRhcmdldCkgT2JqZWN0LmRlZmluZVByb3BlcnR5KHRhcmdldCwgY29udGV4dEluLm5hbWUsIGRlc2NyaXB0b3IpO1xuICBkb25lID0gdHJ1ZTtcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBfX3J1bkluaXRpYWxpemVycyh0aGlzQXJnLCBpbml0aWFsaXplcnMsIHZhbHVlKSB7XG4gIHZhciB1c2VWYWx1ZSA9IGFyZ3VtZW50cy5sZW5ndGggPiAyO1xuICBmb3IgKHZhciBpID0gMDsgaSA8IGluaXRpYWxpemVycy5sZW5ndGg7IGkrKykge1xuICAgICAgdmFsdWUgPSB1c2VWYWx1ZSA/IGluaXRpYWxpemVyc1tpXS5jYWxsKHRoaXNBcmcsIHZhbHVlKSA6IGluaXRpYWxpemVyc1tpXS5jYWxsKHRoaXNBcmcpO1xuICB9XG4gIHJldHVybiB1c2VWYWx1ZSA/IHZhbHVlIDogdm9pZCAwO1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIF9fcHJvcEtleSh4KSB7XG4gIHJldHVybiB0eXBlb2YgeCA9PT0gXCJzeW1ib2xcIiA/IHggOiBcIlwiLmNvbmNhdCh4KTtcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBfX3NldEZ1bmN0aW9uTmFtZShmLCBuYW1lLCBwcmVmaXgpIHtcbiAgaWYgKHR5cGVvZiBuYW1lID09PSBcInN5bWJvbFwiKSBuYW1lID0gbmFtZS5kZXNjcmlwdGlvbiA/IFwiW1wiLmNvbmNhdChuYW1lLmRlc2NyaXB0aW9uLCBcIl1cIikgOiBcIlwiO1xuICByZXR1cm4gT2JqZWN0LmRlZmluZVByb3BlcnR5KGYsIFwibmFtZVwiLCB7IGNvbmZpZ3VyYWJsZTogdHJ1ZSwgdmFsdWU6IHByZWZpeCA/IFwiXCIuY29uY2F0KHByZWZpeCwgXCIgXCIsIG5hbWUpIDogbmFtZSB9KTtcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBfX21ldGFkYXRhKG1ldGFkYXRhS2V5LCBtZXRhZGF0YVZhbHVlKSB7XG4gIGlmICh0eXBlb2YgUmVmbGVjdCA9PT0gXCJvYmplY3RcIiAmJiB0eXBlb2YgUmVmbGVjdC5tZXRhZGF0YSA9PT0gXCJmdW5jdGlvblwiKSByZXR1cm4gUmVmbGVjdC5tZXRhZGF0YShtZXRhZGF0YUtleSwgbWV0YWRhdGFWYWx1ZSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2F3YWl0ZXIodGhpc0FyZywgX2FyZ3VtZW50cywgUCwgZ2VuZXJhdG9yKSB7XG4gIGZ1bmN0aW9uIGFkb3B0KHZhbHVlKSB7IHJldHVybiB2YWx1ZSBpbnN0YW5jZW9mIFAgPyB2YWx1ZSA6IG5ldyBQKGZ1bmN0aW9uIChyZXNvbHZlKSB7IHJlc29sdmUodmFsdWUpOyB9KTsgfVxuICByZXR1cm4gbmV3IChQIHx8IChQID0gUHJvbWlzZSkpKGZ1bmN0aW9uIChyZXNvbHZlLCByZWplY3QpIHtcbiAgICAgIGZ1bmN0aW9uIGZ1bGZpbGxlZCh2YWx1ZSkgeyB0cnkgeyBzdGVwKGdlbmVyYXRvci5uZXh0KHZhbHVlKSk7IH0gY2F0Y2ggKGUpIHsgcmVqZWN0KGUpOyB9IH1cbiAgICAgIGZ1bmN0aW9uIHJlamVjdGVkKHZhbHVlKSB7IHRyeSB7IHN0ZXAoZ2VuZXJhdG9yW1widGhyb3dcIl0odmFsdWUpKTsgfSBjYXRjaCAoZSkgeyByZWplY3QoZSk7IH0gfVxuICAgICAgZnVuY3Rpb24gc3RlcChyZXN1bHQpIHsgcmVzdWx0LmRvbmUgPyByZXNvbHZlKHJlc3VsdC52YWx1ZSkgOiBhZG9wdChyZXN1bHQudmFsdWUpLnRoZW4oZnVsZmlsbGVkLCByZWplY3RlZCk7IH1cbiAgICAgIHN0ZXAoKGdlbmVyYXRvciA9IGdlbmVyYXRvci5hcHBseSh0aGlzQXJnLCBfYXJndW1lbnRzIHx8IFtdKSkubmV4dCgpKTtcbiAgfSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2dlbmVyYXRvcih0aGlzQXJnLCBib2R5KSB7XG4gIHZhciBfID0geyBsYWJlbDogMCwgc2VudDogZnVuY3Rpb24oKSB7IGlmICh0WzBdICYgMSkgdGhyb3cgdFsxXTsgcmV0dXJuIHRbMV07IH0sIHRyeXM6IFtdLCBvcHM6IFtdIH0sIGYsIHksIHQsIGcgPSBPYmplY3QuY3JlYXRlKCh0eXBlb2YgSXRlcmF0b3IgPT09IFwiZnVuY3Rpb25cIiA/IEl0ZXJhdG9yIDogT2JqZWN0KS5wcm90b3R5cGUpO1xuICByZXR1cm4gZy5uZXh0ID0gdmVyYigwKSwgZ1tcInRocm93XCJdID0gdmVyYigxKSwgZ1tcInJldHVyblwiXSA9IHZlcmIoMiksIHR5cGVvZiBTeW1ib2wgPT09IFwiZnVuY3Rpb25cIiAmJiAoZ1tTeW1ib2wuaXRlcmF0b3JdID0gZnVuY3Rpb24oKSB7IHJldHVybiB0aGlzOyB9KSwgZztcbiAgZnVuY3Rpb24gdmVyYihuKSB7IHJldHVybiBmdW5jdGlvbiAodikgeyByZXR1cm4gc3RlcChbbiwgdl0pOyB9OyB9XG4gIGZ1bmN0aW9uIHN0ZXAob3ApIHtcbiAgICAgIGlmIChmKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiR2VuZXJhdG9yIGlzIGFscmVhZHkgZXhlY3V0aW5nLlwiKTtcbiAgICAgIHdoaWxlIChnICYmIChnID0gMCwgb3BbMF0gJiYgKF8gPSAwKSksIF8pIHRyeSB7XG4gICAgICAgICAgaWYgKGYgPSAxLCB5ICYmICh0ID0gb3BbMF0gJiAyID8geVtcInJldHVyblwiXSA6IG9wWzBdID8geVtcInRocm93XCJdIHx8ICgodCA9IHlbXCJyZXR1cm5cIl0pICYmIHQuY2FsbCh5KSwgMCkgOiB5Lm5leHQpICYmICEodCA9IHQuY2FsbCh5LCBvcFsxXSkpLmRvbmUpIHJldHVybiB0O1xuICAgICAgICAgIGlmICh5ID0gMCwgdCkgb3AgPSBbb3BbMF0gJiAyLCB0LnZhbHVlXTtcbiAgICAgICAgICBzd2l0Y2ggKG9wWzBdKSB7XG4gICAgICAgICAgICAgIGNhc2UgMDogY2FzZSAxOiB0ID0gb3A7IGJyZWFrO1xuICAgICAgICAgICAgICBjYXNlIDQ6IF8ubGFiZWwrKzsgcmV0dXJuIHsgdmFsdWU6IG9wWzFdLCBkb25lOiBmYWxzZSB9O1xuICAgICAgICAgICAgICBjYXNlIDU6IF8ubGFiZWwrKzsgeSA9IG9wWzFdOyBvcCA9IFswXTsgY29udGludWU7XG4gICAgICAgICAgICAgIGNhc2UgNzogb3AgPSBfLm9wcy5wb3AoKTsgXy50cnlzLnBvcCgpOyBjb250aW51ZTtcbiAgICAgICAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICAgICAgICAgIGlmICghKHQgPSBfLnRyeXMsIHQgPSB0Lmxlbmd0aCA+IDAgJiYgdFt0Lmxlbmd0aCAtIDFdKSAmJiAob3BbMF0gPT09IDYgfHwgb3BbMF0gPT09IDIpKSB7IF8gPSAwOyBjb250aW51ZTsgfVxuICAgICAgICAgICAgICAgICAgaWYgKG9wWzBdID09PSAzICYmICghdCB8fCAob3BbMV0gPiB0WzBdICYmIG9wWzFdIDwgdFszXSkpKSB7IF8ubGFiZWwgPSBvcFsxXTsgYnJlYWs7IH1cbiAgICAgICAgICAgICAgICAgIGlmIChvcFswXSA9PT0gNiAmJiBfLmxhYmVsIDwgdFsxXSkgeyBfLmxhYmVsID0gdFsxXTsgdCA9IG9wOyBicmVhazsgfVxuICAgICAgICAgICAgICAgICAgaWYgKHQgJiYgXy5sYWJlbCA8IHRbMl0pIHsgXy5sYWJlbCA9IHRbMl07IF8ub3BzLnB1c2gob3ApOyBicmVhazsgfVxuICAgICAgICAgICAgICAgICAgaWYgKHRbMl0pIF8ub3BzLnBvcCgpO1xuICAgICAgICAgICAgICAgICAgXy50cnlzLnBvcCgpOyBjb250aW51ZTtcbiAgICAgICAgICB9XG4gICAgICAgICAgb3AgPSBib2R5LmNhbGwodGhpc0FyZywgXyk7XG4gICAgICB9IGNhdGNoIChlKSB7IG9wID0gWzYsIGVdOyB5ID0gMDsgfSBmaW5hbGx5IHsgZiA9IHQgPSAwOyB9XG4gICAgICBpZiAob3BbMF0gJiA1KSB0aHJvdyBvcFsxXTsgcmV0dXJuIHsgdmFsdWU6IG9wWzBdID8gb3BbMV0gOiB2b2lkIDAsIGRvbmU6IHRydWUgfTtcbiAgfVxufVxuXG5leHBvcnQgdmFyIF9fY3JlYXRlQmluZGluZyA9IE9iamVjdC5jcmVhdGUgPyAoZnVuY3Rpb24obywgbSwgaywgazIpIHtcbiAgaWYgKGsyID09PSB1bmRlZmluZWQpIGsyID0gaztcbiAgdmFyIGRlc2MgPSBPYmplY3QuZ2V0T3duUHJvcGVydHlEZXNjcmlwdG9yKG0sIGspO1xuICBpZiAoIWRlc2MgfHwgKFwiZ2V0XCIgaW4gZGVzYyA/ICFtLl9fZXNNb2R1bGUgOiBkZXNjLndyaXRhYmxlIHx8IGRlc2MuY29uZmlndXJhYmxlKSkge1xuICAgICAgZGVzYyA9IHsgZW51bWVyYWJsZTogdHJ1ZSwgZ2V0OiBmdW5jdGlvbigpIHsgcmV0dXJuIG1ba107IH0gfTtcbiAgfVxuICBPYmplY3QuZGVmaW5lUHJvcGVydHkobywgazIsIGRlc2MpO1xufSkgOiAoZnVuY3Rpb24obywgbSwgaywgazIpIHtcbiAgaWYgKGsyID09PSB1bmRlZmluZWQpIGsyID0gaztcbiAgb1trMl0gPSBtW2tdO1xufSk7XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2V4cG9ydFN0YXIobSwgbykge1xuICBmb3IgKHZhciBwIGluIG0pIGlmIChwICE9PSBcImRlZmF1bHRcIiAmJiAhT2JqZWN0LnByb3RvdHlwZS5oYXNPd25Qcm9wZXJ0eS5jYWxsKG8sIHApKSBfX2NyZWF0ZUJpbmRpbmcobywgbSwgcCk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX3ZhbHVlcyhvKSB7XG4gIHZhciBzID0gdHlwZW9mIFN5bWJvbCA9PT0gXCJmdW5jdGlvblwiICYmIFN5bWJvbC5pdGVyYXRvciwgbSA9IHMgJiYgb1tzXSwgaSA9IDA7XG4gIGlmIChtKSByZXR1cm4gbS5jYWxsKG8pO1xuICBpZiAobyAmJiB0eXBlb2Ygby5sZW5ndGggPT09IFwibnVtYmVyXCIpIHJldHVybiB7XG4gICAgICBuZXh0OiBmdW5jdGlvbiAoKSB7XG4gICAgICAgICAgaWYgKG8gJiYgaSA+PSBvLmxlbmd0aCkgbyA9IHZvaWQgMDtcbiAgICAgICAgICByZXR1cm4geyB2YWx1ZTogbyAmJiBvW2krK10sIGRvbmU6ICFvIH07XG4gICAgICB9XG4gIH07XG4gIHRocm93IG5ldyBUeXBlRXJyb3IocyA/IFwiT2JqZWN0IGlzIG5vdCBpdGVyYWJsZS5cIiA6IFwiU3ltYm9sLml0ZXJhdG9yIGlzIG5vdCBkZWZpbmVkLlwiKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fcmVhZChvLCBuKSB7XG4gIHZhciBtID0gdHlwZW9mIFN5bWJvbCA9PT0gXCJmdW5jdGlvblwiICYmIG9bU3ltYm9sLml0ZXJhdG9yXTtcbiAgaWYgKCFtKSByZXR1cm4gbztcbiAgdmFyIGkgPSBtLmNhbGwobyksIHIsIGFyID0gW10sIGU7XG4gIHRyeSB7XG4gICAgICB3aGlsZSAoKG4gPT09IHZvaWQgMCB8fCBuLS0gPiAwKSAmJiAhKHIgPSBpLm5leHQoKSkuZG9uZSkgYXIucHVzaChyLnZhbHVlKTtcbiAgfVxuICBjYXRjaCAoZXJyb3IpIHsgZSA9IHsgZXJyb3I6IGVycm9yIH07IH1cbiAgZmluYWxseSB7XG4gICAgICB0cnkge1xuICAgICAgICAgIGlmIChyICYmICFyLmRvbmUgJiYgKG0gPSBpW1wicmV0dXJuXCJdKSkgbS5jYWxsKGkpO1xuICAgICAgfVxuICAgICAgZmluYWxseSB7IGlmIChlKSB0aHJvdyBlLmVycm9yOyB9XG4gIH1cbiAgcmV0dXJuIGFyO1xufVxuXG4vKiogQGRlcHJlY2F0ZWQgKi9cbmV4cG9ydCBmdW5jdGlvbiBfX3NwcmVhZCgpIHtcbiAgZm9yICh2YXIgYXIgPSBbXSwgaSA9IDA7IGkgPCBhcmd1bWVudHMubGVuZ3RoOyBpKyspXG4gICAgICBhciA9IGFyLmNvbmNhdChfX3JlYWQoYXJndW1lbnRzW2ldKSk7XG4gIHJldHVybiBhcjtcbn1cblxuLyoqIEBkZXByZWNhdGVkICovXG5leHBvcnQgZnVuY3Rpb24gX19zcHJlYWRBcnJheXMoKSB7XG4gIGZvciAodmFyIHMgPSAwLCBpID0gMCwgaWwgPSBhcmd1bWVudHMubGVuZ3RoOyBpIDwgaWw7IGkrKykgcyArPSBhcmd1bWVudHNbaV0ubGVuZ3RoO1xuICBmb3IgKHZhciByID0gQXJyYXkocyksIGsgPSAwLCBpID0gMDsgaSA8IGlsOyBpKyspXG4gICAgICBmb3IgKHZhciBhID0gYXJndW1lbnRzW2ldLCBqID0gMCwgamwgPSBhLmxlbmd0aDsgaiA8IGpsOyBqKyssIGsrKylcbiAgICAgICAgICByW2tdID0gYVtqXTtcbiAgcmV0dXJuIHI7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX3NwcmVhZEFycmF5KHRvLCBmcm9tLCBwYWNrKSB7XG4gIGlmIChwYWNrIHx8IGFyZ3VtZW50cy5sZW5ndGggPT09IDIpIGZvciAodmFyIGkgPSAwLCBsID0gZnJvbS5sZW5ndGgsIGFyOyBpIDwgbDsgaSsrKSB7XG4gICAgICBpZiAoYXIgfHwgIShpIGluIGZyb20pKSB7XG4gICAgICAgICAgaWYgKCFhcikgYXIgPSBBcnJheS5wcm90b3R5cGUuc2xpY2UuY2FsbChmcm9tLCAwLCBpKTtcbiAgICAgICAgICBhcltpXSA9IGZyb21baV07XG4gICAgICB9XG4gIH1cbiAgcmV0dXJuIHRvLmNvbmNhdChhciB8fCBBcnJheS5wcm90b3R5cGUuc2xpY2UuY2FsbChmcm9tKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2F3YWl0KHYpIHtcbiAgcmV0dXJuIHRoaXMgaW5zdGFuY2VvZiBfX2F3YWl0ID8gKHRoaXMudiA9IHYsIHRoaXMpIDogbmV3IF9fYXdhaXQodik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2FzeW5jR2VuZXJhdG9yKHRoaXNBcmcsIF9hcmd1bWVudHMsIGdlbmVyYXRvcikge1xuICBpZiAoIVN5bWJvbC5hc3luY0l0ZXJhdG9yKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiU3ltYm9sLmFzeW5jSXRlcmF0b3IgaXMgbm90IGRlZmluZWQuXCIpO1xuICB2YXIgZyA9IGdlbmVyYXRvci5hcHBseSh0aGlzQXJnLCBfYXJndW1lbnRzIHx8IFtdKSwgaSwgcSA9IFtdO1xuICByZXR1cm4gaSA9IE9iamVjdC5jcmVhdGUoKHR5cGVvZiBBc3luY0l0ZXJhdG9yID09PSBcImZ1bmN0aW9uXCIgPyBBc3luY0l0ZXJhdG9yIDogT2JqZWN0KS5wcm90b3R5cGUpLCB2ZXJiKFwibmV4dFwiKSwgdmVyYihcInRocm93XCIpLCB2ZXJiKFwicmV0dXJuXCIsIGF3YWl0UmV0dXJuKSwgaVtTeW1ib2wuYXN5bmNJdGVyYXRvcl0gPSBmdW5jdGlvbiAoKSB7IHJldHVybiB0aGlzOyB9LCBpO1xuICBmdW5jdGlvbiBhd2FpdFJldHVybihmKSB7IHJldHVybiBmdW5jdGlvbiAodikgeyByZXR1cm4gUHJvbWlzZS5yZXNvbHZlKHYpLnRoZW4oZiwgcmVqZWN0KTsgfTsgfVxuICBmdW5jdGlvbiB2ZXJiKG4sIGYpIHsgaWYgKGdbbl0pIHsgaVtuXSA9IGZ1bmN0aW9uICh2KSB7IHJldHVybiBuZXcgUHJvbWlzZShmdW5jdGlvbiAoYSwgYikgeyBxLnB1c2goW24sIHYsIGEsIGJdKSA+IDEgfHwgcmVzdW1lKG4sIHYpOyB9KTsgfTsgaWYgKGYpIGlbbl0gPSBmKGlbbl0pOyB9IH1cbiAgZnVuY3Rpb24gcmVzdW1lKG4sIHYpIHsgdHJ5IHsgc3RlcChnW25dKHYpKTsgfSBjYXRjaCAoZSkgeyBzZXR0bGUocVswXVszXSwgZSk7IH0gfVxuICBmdW5jdGlvbiBzdGVwKHIpIHsgci52YWx1ZSBpbnN0YW5jZW9mIF9fYXdhaXQgPyBQcm9taXNlLnJlc29sdmUoci52YWx1ZS52KS50aGVuKGZ1bGZpbGwsIHJlamVjdCkgOiBzZXR0bGUocVswXVsyXSwgcik7IH1cbiAgZnVuY3Rpb24gZnVsZmlsbCh2YWx1ZSkgeyByZXN1bWUoXCJuZXh0XCIsIHZhbHVlKTsgfVxuICBmdW5jdGlvbiByZWplY3QodmFsdWUpIHsgcmVzdW1lKFwidGhyb3dcIiwgdmFsdWUpOyB9XG4gIGZ1bmN0aW9uIHNldHRsZShmLCB2KSB7IGlmIChmKHYpLCBxLnNoaWZ0KCksIHEubGVuZ3RoKSByZXN1bWUocVswXVswXSwgcVswXVsxXSk7IH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fYXN5bmNEZWxlZ2F0b3Iobykge1xuICB2YXIgaSwgcDtcbiAgcmV0dXJuIGkgPSB7fSwgdmVyYihcIm5leHRcIiksIHZlcmIoXCJ0aHJvd1wiLCBmdW5jdGlvbiAoZSkgeyB0aHJvdyBlOyB9KSwgdmVyYihcInJldHVyblwiKSwgaVtTeW1ib2wuaXRlcmF0b3JdID0gZnVuY3Rpb24gKCkgeyByZXR1cm4gdGhpczsgfSwgaTtcbiAgZnVuY3Rpb24gdmVyYihuLCBmKSB7IGlbbl0gPSBvW25dID8gZnVuY3Rpb24gKHYpIHsgcmV0dXJuIChwID0gIXApID8geyB2YWx1ZTogX19hd2FpdChvW25dKHYpKSwgZG9uZTogZmFsc2UgfSA6IGYgPyBmKHYpIDogdjsgfSA6IGY7IH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fYXN5bmNWYWx1ZXMobykge1xuICBpZiAoIVN5bWJvbC5hc3luY0l0ZXJhdG9yKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiU3ltYm9sLmFzeW5jSXRlcmF0b3IgaXMgbm90IGRlZmluZWQuXCIpO1xuICB2YXIgbSA9IG9bU3ltYm9sLmFzeW5jSXRlcmF0b3JdLCBpO1xuICByZXR1cm4gbSA/IG0uY2FsbChvKSA6IChvID0gdHlwZW9mIF9fdmFsdWVzID09PSBcImZ1bmN0aW9uXCIgPyBfX3ZhbHVlcyhvKSA6IG9bU3ltYm9sLml0ZXJhdG9yXSgpLCBpID0ge30sIHZlcmIoXCJuZXh0XCIpLCB2ZXJiKFwidGhyb3dcIiksIHZlcmIoXCJyZXR1cm5cIiksIGlbU3ltYm9sLmFzeW5jSXRlcmF0b3JdID0gZnVuY3Rpb24gKCkgeyByZXR1cm4gdGhpczsgfSwgaSk7XG4gIGZ1bmN0aW9uIHZlcmIobikgeyBpW25dID0gb1tuXSAmJiBmdW5jdGlvbiAodikgeyByZXR1cm4gbmV3IFByb21pc2UoZnVuY3Rpb24gKHJlc29sdmUsIHJlamVjdCkgeyB2ID0gb1tuXSh2KSwgc2V0dGxlKHJlc29sdmUsIHJlamVjdCwgdi5kb25lLCB2LnZhbHVlKTsgfSk7IH07IH1cbiAgZnVuY3Rpb24gc2V0dGxlKHJlc29sdmUsIHJlamVjdCwgZCwgdikgeyBQcm9taXNlLnJlc29sdmUodikudGhlbihmdW5jdGlvbih2KSB7IHJlc29sdmUoeyB2YWx1ZTogdiwgZG9uZTogZCB9KTsgfSwgcmVqZWN0KTsgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gX19tYWtlVGVtcGxhdGVPYmplY3QoY29va2VkLCByYXcpIHtcbiAgaWYgKE9iamVjdC5kZWZpbmVQcm9wZXJ0eSkgeyBPYmplY3QuZGVmaW5lUHJvcGVydHkoY29va2VkLCBcInJhd1wiLCB7IHZhbHVlOiByYXcgfSk7IH0gZWxzZSB7IGNvb2tlZC5yYXcgPSByYXc7IH1cbiAgcmV0dXJuIGNvb2tlZDtcbn07XG5cbnZhciBfX3NldE1vZHVsZURlZmF1bHQgPSBPYmplY3QuY3JlYXRlID8gKGZ1bmN0aW9uKG8sIHYpIHtcbiAgT2JqZWN0LmRlZmluZVByb3BlcnR5KG8sIFwiZGVmYXVsdFwiLCB7IGVudW1lcmFibGU6IHRydWUsIHZhbHVlOiB2IH0pO1xufSkgOiBmdW5jdGlvbihvLCB2KSB7XG4gIG9bXCJkZWZhdWx0XCJdID0gdjtcbn07XG5cbnZhciBvd25LZXlzID0gZnVuY3Rpb24obykge1xuICBvd25LZXlzID0gT2JqZWN0LmdldE93blByb3BlcnR5TmFtZXMgfHwgZnVuY3Rpb24gKG8pIHtcbiAgICB2YXIgYXIgPSBbXTtcbiAgICBmb3IgKHZhciBrIGluIG8pIGlmIChPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwobywgaykpIGFyW2FyLmxlbmd0aF0gPSBrO1xuICAgIHJldHVybiBhcjtcbiAgfTtcbiAgcmV0dXJuIG93bktleXMobyk7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gX19pbXBvcnRTdGFyKG1vZCkge1xuICBpZiAobW9kICYmIG1vZC5fX2VzTW9kdWxlKSByZXR1cm4gbW9kO1xuICB2YXIgcmVzdWx0ID0ge307XG4gIGlmIChtb2QgIT0gbnVsbCkgZm9yICh2YXIgayA9IG93bktleXMobW9kKSwgaSA9IDA7IGkgPCBrLmxlbmd0aDsgaSsrKSBpZiAoa1tpXSAhPT0gXCJkZWZhdWx0XCIpIF9fY3JlYXRlQmluZGluZyhyZXN1bHQsIG1vZCwga1tpXSk7XG4gIF9fc2V0TW9kdWxlRGVmYXVsdChyZXN1bHQsIG1vZCk7XG4gIHJldHVybiByZXN1bHQ7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2ltcG9ydERlZmF1bHQobW9kKSB7XG4gIHJldHVybiAobW9kICYmIG1vZC5fX2VzTW9kdWxlKSA/IG1vZCA6IHsgZGVmYXVsdDogbW9kIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2NsYXNzUHJpdmF0ZUZpZWxkR2V0KHJlY2VpdmVyLCBzdGF0ZSwga2luZCwgZikge1xuICBpZiAoa2luZCA9PT0gXCJhXCIgJiYgIWYpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJQcml2YXRlIGFjY2Vzc29yIHdhcyBkZWZpbmVkIHdpdGhvdXQgYSBnZXR0ZXJcIik7XG4gIGlmICh0eXBlb2Ygc3RhdGUgPT09IFwiZnVuY3Rpb25cIiA/IHJlY2VpdmVyICE9PSBzdGF0ZSB8fCAhZiA6ICFzdGF0ZS5oYXMocmVjZWl2ZXIpKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiQ2Fubm90IHJlYWQgcHJpdmF0ZSBtZW1iZXIgZnJvbSBhbiBvYmplY3Qgd2hvc2UgY2xhc3MgZGlkIG5vdCBkZWNsYXJlIGl0XCIpO1xuICByZXR1cm4ga2luZCA9PT0gXCJtXCIgPyBmIDoga2luZCA9PT0gXCJhXCIgPyBmLmNhbGwocmVjZWl2ZXIpIDogZiA/IGYudmFsdWUgOiBzdGF0ZS5nZXQocmVjZWl2ZXIpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gX19jbGFzc1ByaXZhdGVGaWVsZFNldChyZWNlaXZlciwgc3RhdGUsIHZhbHVlLCBraW5kLCBmKSB7XG4gIGlmIChraW5kID09PSBcIm1cIikgdGhyb3cgbmV3IFR5cGVFcnJvcihcIlByaXZhdGUgbWV0aG9kIGlzIG5vdCB3cml0YWJsZVwiKTtcbiAgaWYgKGtpbmQgPT09IFwiYVwiICYmICFmKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiUHJpdmF0ZSBhY2Nlc3NvciB3YXMgZGVmaW5lZCB3aXRob3V0IGEgc2V0dGVyXCIpO1xuICBpZiAodHlwZW9mIHN0YXRlID09PSBcImZ1bmN0aW9uXCIgPyByZWNlaXZlciAhPT0gc3RhdGUgfHwgIWYgOiAhc3RhdGUuaGFzKHJlY2VpdmVyKSkgdGhyb3cgbmV3IFR5cGVFcnJvcihcIkNhbm5vdCB3cml0ZSBwcml2YXRlIG1lbWJlciB0byBhbiBvYmplY3Qgd2hvc2UgY2xhc3MgZGlkIG5vdCBkZWNsYXJlIGl0XCIpO1xuICByZXR1cm4gKGtpbmQgPT09IFwiYVwiID8gZi5jYWxsKHJlY2VpdmVyLCB2YWx1ZSkgOiBmID8gZi52YWx1ZSA9IHZhbHVlIDogc3RhdGUuc2V0KHJlY2VpdmVyLCB2YWx1ZSkpLCB2YWx1ZTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIF9fY2xhc3NQcml2YXRlRmllbGRJbihzdGF0ZSwgcmVjZWl2ZXIpIHtcbiAgaWYgKHJlY2VpdmVyID09PSBudWxsIHx8ICh0eXBlb2YgcmVjZWl2ZXIgIT09IFwib2JqZWN0XCIgJiYgdHlwZW9mIHJlY2VpdmVyICE9PSBcImZ1bmN0aW9uXCIpKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiQ2Fubm90IHVzZSAnaW4nIG9wZXJhdG9yIG9uIG5vbi1vYmplY3RcIik7XG4gIHJldHVybiB0eXBlb2Ygc3RhdGUgPT09IFwiZnVuY3Rpb25cIiA/IHJlY2VpdmVyID09PSBzdGF0ZSA6IHN0YXRlLmhhcyhyZWNlaXZlcik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2FkZERpc3Bvc2FibGVSZXNvdXJjZShlbnYsIHZhbHVlLCBhc3luYykge1xuICBpZiAodmFsdWUgIT09IG51bGwgJiYgdmFsdWUgIT09IHZvaWQgMCkge1xuICAgIGlmICh0eXBlb2YgdmFsdWUgIT09IFwib2JqZWN0XCIgJiYgdHlwZW9mIHZhbHVlICE9PSBcImZ1bmN0aW9uXCIpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJPYmplY3QgZXhwZWN0ZWQuXCIpO1xuICAgIHZhciBkaXNwb3NlLCBpbm5lcjtcbiAgICBpZiAoYXN5bmMpIHtcbiAgICAgIGlmICghU3ltYm9sLmFzeW5jRGlzcG9zZSkgdGhyb3cgbmV3IFR5cGVFcnJvcihcIlN5bWJvbC5hc3luY0Rpc3Bvc2UgaXMgbm90IGRlZmluZWQuXCIpO1xuICAgICAgZGlzcG9zZSA9IHZhbHVlW1N5bWJvbC5hc3luY0Rpc3Bvc2VdO1xuICAgIH1cbiAgICBpZiAoZGlzcG9zZSA9PT0gdm9pZCAwKSB7XG4gICAgICBpZiAoIVN5bWJvbC5kaXNwb3NlKSB0aHJvdyBuZXcgVHlwZUVycm9yKFwiU3ltYm9sLmRpc3Bvc2UgaXMgbm90IGRlZmluZWQuXCIpO1xuICAgICAgZGlzcG9zZSA9IHZhbHVlW1N5bWJvbC5kaXNwb3NlXTtcbiAgICAgIGlmIChhc3luYykgaW5uZXIgPSBkaXNwb3NlO1xuICAgIH1cbiAgICBpZiAodHlwZW9mIGRpc3Bvc2UgIT09IFwiZnVuY3Rpb25cIikgdGhyb3cgbmV3IFR5cGVFcnJvcihcIk9iamVjdCBub3QgZGlzcG9zYWJsZS5cIik7XG4gICAgaWYgKGlubmVyKSBkaXNwb3NlID0gZnVuY3Rpb24oKSB7IHRyeSB7IGlubmVyLmNhbGwodGhpcyk7IH0gY2F0Y2ggKGUpIHsgcmV0dXJuIFByb21pc2UucmVqZWN0KGUpOyB9IH07XG4gICAgZW52LnN0YWNrLnB1c2goeyB2YWx1ZTogdmFsdWUsIGRpc3Bvc2U6IGRpc3Bvc2UsIGFzeW5jOiBhc3luYyB9KTtcbiAgfVxuICBlbHNlIGlmIChhc3luYykge1xuICAgIGVudi5zdGFjay5wdXNoKHsgYXN5bmM6IHRydWUgfSk7XG4gIH1cbiAgcmV0dXJuIHZhbHVlO1xufVxuXG52YXIgX1N1cHByZXNzZWRFcnJvciA9IHR5cGVvZiBTdXBwcmVzc2VkRXJyb3IgPT09IFwiZnVuY3Rpb25cIiA/IFN1cHByZXNzZWRFcnJvciA6IGZ1bmN0aW9uIChlcnJvciwgc3VwcHJlc3NlZCwgbWVzc2FnZSkge1xuICB2YXIgZSA9IG5ldyBFcnJvcihtZXNzYWdlKTtcbiAgcmV0dXJuIGUubmFtZSA9IFwiU3VwcHJlc3NlZEVycm9yXCIsIGUuZXJyb3IgPSBlcnJvciwgZS5zdXBwcmVzc2VkID0gc3VwcHJlc3NlZCwgZTtcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBfX2Rpc3Bvc2VSZXNvdXJjZXMoZW52KSB7XG4gIGZ1bmN0aW9uIGZhaWwoZSkge1xuICAgIGVudi5lcnJvciA9IGVudi5oYXNFcnJvciA/IG5ldyBfU3VwcHJlc3NlZEVycm9yKGUsIGVudi5lcnJvciwgXCJBbiBlcnJvciB3YXMgc3VwcHJlc3NlZCBkdXJpbmcgZGlzcG9zYWwuXCIpIDogZTtcbiAgICBlbnYuaGFzRXJyb3IgPSB0cnVlO1xuICB9XG4gIHZhciByLCBzID0gMDtcbiAgZnVuY3Rpb24gbmV4dCgpIHtcbiAgICB3aGlsZSAociA9IGVudi5zdGFjay5wb3AoKSkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgaWYgKCFyLmFzeW5jICYmIHMgPT09IDEpIHJldHVybiBzID0gMCwgZW52LnN0YWNrLnB1c2gociksIFByb21pc2UucmVzb2x2ZSgpLnRoZW4obmV4dCk7XG4gICAgICAgIGlmIChyLmRpc3Bvc2UpIHtcbiAgICAgICAgICB2YXIgcmVzdWx0ID0gci5kaXNwb3NlLmNhbGwoci52YWx1ZSk7XG4gICAgICAgICAgaWYgKHIuYXN5bmMpIHJldHVybiBzIHw9IDIsIFByb21pc2UucmVzb2x2ZShyZXN1bHQpLnRoZW4obmV4dCwgZnVuY3Rpb24oZSkgeyBmYWlsKGUpOyByZXR1cm4gbmV4dCgpOyB9KTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHMgfD0gMTtcbiAgICAgIH1cbiAgICAgIGNhdGNoIChlKSB7XG4gICAgICAgIGZhaWwoZSk7XG4gICAgICB9XG4gICAgfVxuICAgIGlmIChzID09PSAxKSByZXR1cm4gZW52Lmhhc0Vycm9yID8gUHJvbWlzZS5yZWplY3QoZW52LmVycm9yKSA6IFByb21pc2UucmVzb2x2ZSgpO1xuICAgIGlmIChlbnYuaGFzRXJyb3IpIHRocm93IGVudi5lcnJvcjtcbiAgfVxuICByZXR1cm4gbmV4dCgpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gX19yZXdyaXRlUmVsYXRpdmVJbXBvcnRFeHRlbnNpb24ocGF0aCwgcHJlc2VydmVKc3gpIHtcbiAgaWYgKHR5cGVvZiBwYXRoID09PSBcInN0cmluZ1wiICYmIC9eXFwuXFwuP1xcLy8udGVzdChwYXRoKSkge1xuICAgICAgcmV0dXJuIHBhdGgucmVwbGFjZSgvXFwuKHRzeCkkfCgoPzpcXC5kKT8pKCg/OlxcLlteLi9dKz8pPylcXC4oW2NtXT8pdHMkL2ksIGZ1bmN0aW9uIChtLCB0c3gsIGQsIGV4dCwgY20pIHtcbiAgICAgICAgICByZXR1cm4gdHN4ID8gcHJlc2VydmVKc3ggPyBcIi5qc3hcIiA6IFwiLmpzXCIgOiBkICYmICghZXh0IHx8ICFjbSkgPyBtIDogKGQgKyBleHQgKyBcIi5cIiArIGNtLnRvTG93ZXJDYXNlKCkgKyBcImpzXCIpO1xuICAgICAgfSk7XG4gIH1cbiAgcmV0dXJuIHBhdGg7XG59XG5cbmV4cG9ydCBkZWZhdWx0IHtcbiAgX19leHRlbmRzLFxuICBfX2Fzc2lnbixcbiAgX19yZXN0LFxuICBfX2RlY29yYXRlLFxuICBfX3BhcmFtLFxuICBfX2VzRGVjb3JhdGUsXG4gIF9fcnVuSW5pdGlhbGl6ZXJzLFxuICBfX3Byb3BLZXksXG4gIF9fc2V0RnVuY3Rpb25OYW1lLFxuICBfX21ldGFkYXRhLFxuICBfX2F3YWl0ZXIsXG4gIF9fZ2VuZXJhdG9yLFxuICBfX2NyZWF0ZUJpbmRpbmcsXG4gIF9fZXhwb3J0U3RhcixcbiAgX192YWx1ZXMsXG4gIF9fcmVhZCxcbiAgX19zcHJlYWQsXG4gIF9fc3ByZWFkQXJyYXlzLFxuICBfX3NwcmVhZEFycmF5LFxuICBfX2F3YWl0LFxuICBfX2FzeW5jR2VuZXJhdG9yLFxuICBfX2FzeW5jRGVsZWdhdG9yLFxuICBfX2FzeW5jVmFsdWVzLFxuICBfX21ha2VUZW1wbGF0ZU9iamVjdCxcbiAgX19pbXBvcnRTdGFyLFxuICBfX2ltcG9ydERlZmF1bHQsXG4gIF9fY2xhc3NQcml2YXRlRmllbGRHZXQsXG4gIF9fY2xhc3NQcml2YXRlRmllbGRTZXQsXG4gIF9fY2xhc3NQcml2YXRlRmllbGRJbixcbiAgX19hZGREaXNwb3NhYmxlUmVzb3VyY2UsXG4gIF9fZGlzcG9zZVJlc291cmNlcyxcbiAgX19yZXdyaXRlUmVsYXRpdmVJbXBvcnRFeHRlbnNpb24sXG59O1xuIiwiZXhwb3J0IHR5cGUgRml0U3RyYXRlZ3lUeXBlID0ge1xyXG4gICAgd3JhcEVsZW1lbnQoZWxlbWVudDogSFRNTEVsZW1lbnQpOiBIVE1MRWxlbWVudDtcclxuICAgIHVwZGF0ZVNpemUoc2l6aW5nRWxlbWVudDogSFRNTEVsZW1lbnQsIHdpZHRoOiBudW1iZXIsIGhlaWdodDogbnVtYmVyKTogdm9pZDtcclxufTtcclxuXHJcbmNvbnN0IEZpdFN0cmF0ZWd5Q29udGFpbjogRml0U3RyYXRlZ3lUeXBlID0ge1xyXG4gICAgd3JhcEVsZW1lbnQoZWxlbWVudDogSFRNTEVsZW1lbnQpOiBIVE1MRWxlbWVudCB7XHJcbiAgICAgICAgY29uc3Qgc2l6aW5nRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5kaXNwbGF5ID0gXCJmbGV4XCI7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5qdXN0aWZ5Q29udGVudCA9IFwiY2VudGVyXCI7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5hbGlnbkl0ZW1zID0gXCJjZW50ZXJcIjtcclxuICAgICAgICBjb25zdCBzY2FsaW5nRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuc3R5bGUudmlzaWJpbGl0eSA9IFwiaGlkZGVuXCI7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuYXBwZW5kQ2hpbGQoZWxlbWVudCk7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5hcHBlbmRDaGlsZChzY2FsaW5nRWxlbWVudCk7XHJcbiAgICAgICAgcmV0dXJuIHNpemluZ0VsZW1lbnQ7XHJcbiAgICB9LFxyXG4gICAgdXBkYXRlU2l6ZShzaXppbmdFbGVtZW50OiBIVE1MRWxlbWVudCwgd2lkdGg6IG51bWJlciwgaGVpZ2h0OiBudW1iZXIpIHtcclxuICAgICAgICBjb25zdCBzY2FsaW5nRWxlbWVudCA9IHNpemluZ0VsZW1lbnQuZmlyc3RFbGVtZW50Q2hpbGQhIGFzIEhUTUxFbGVtZW50O1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUud2lkdGggPSBgJHt3aWR0aH1weGA7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5oZWlnaHQgPSBgJHtoZWlnaHR9cHhgO1xyXG5cclxuICAgICAgICBjb25zdCBbY2hpbGRXaWR0aCwgY2hpbGRIZWlnaHRdID0gW3NjYWxpbmdFbGVtZW50IS5vZmZzZXRXaWR0aCwgc2NhbGluZ0VsZW1lbnQhLm9mZnNldEhlaWdodF07XHJcbiAgICAgICAgY29uc3Qgc2NhbGUgPSBNYXRoLm1pbih3aWR0aCAvIGNoaWxkV2lkdGgsIGhlaWdodCAvIGNoaWxkSGVpZ2h0KTtcclxuICAgICAgICBzY2FsaW5nRWxlbWVudC5zdHlsZS50cmFuc2Zvcm0gPSBgc2NhbGUoJHtzY2FsZX0pYDtcclxuICAgICAgICBzY2FsaW5nRWxlbWVudC5zdHlsZS52aXNpYmlsaXR5ID0gXCJ2aXNpYmxlXCI7XHJcbiAgICB9LFxyXG59O1xyXG5cclxuY29uc3QgRml0U3RyYXRlZ3lDb3ZlcjogRml0U3RyYXRlZ3lUeXBlID0ge1xyXG4gICAgd3JhcEVsZW1lbnQoZWxlbWVudDogSFRNTEVsZW1lbnQpOiBIVE1MRWxlbWVudCB7XHJcbiAgICAgICAgY29uc3Qgc2l6aW5nRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5kaXNwbGF5ID0gXCJmbGV4XCI7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5qdXN0aWZ5Q29udGVudCA9IFwiY2VudGVyXCI7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5hbGlnbkl0ZW1zID0gXCJjZW50ZXJcIjtcclxuICAgICAgICBzaXppbmdFbGVtZW50LnN0eWxlLm92ZXJmbG93ID0gXCJoaWRkZW5cIjtcclxuICAgICAgICBjb25zdCBzY2FsaW5nRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuc3R5bGUudmlzaWJpbGl0eSA9IFwiaGlkZGVuXCI7XHJcbiAgICAgICAgc2NhbGluZ0VsZW1lbnQuYXBwZW5kQ2hpbGQoZWxlbWVudCk7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5hcHBlbmRDaGlsZChzY2FsaW5nRWxlbWVudCk7XHJcbiAgICAgICAgcmV0dXJuIHNpemluZ0VsZW1lbnQ7XHJcbiAgICB9LFxyXG4gICAgdXBkYXRlU2l6ZShzaXppbmdFbGVtZW50OiBIVE1MRWxlbWVudCwgd2lkdGg6IG51bWJlciwgaGVpZ2h0OiBudW1iZXIpIHtcclxuICAgICAgICBjb25zdCBzY2FsaW5nRWxlbWVudCA9IHNpemluZ0VsZW1lbnQuZmlyc3RFbGVtZW50Q2hpbGQhIGFzIEhUTUxFbGVtZW50O1xyXG4gICAgICAgIHNpemluZ0VsZW1lbnQuc3R5bGUud2lkdGggPSBgJHt3aWR0aH1weGA7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5oZWlnaHQgPSBgJHtoZWlnaHR9cHhgO1xyXG5cclxuICAgICAgICBjb25zdCBbY2hpbGRXaWR0aCwgY2hpbGRIZWlnaHRdID0gW3NjYWxpbmdFbGVtZW50IS5vZmZzZXRXaWR0aCwgc2NhbGluZ0VsZW1lbnQhLm9mZnNldEhlaWdodF07XHJcbiAgICAgICAgY29uc3Qgc2NhbGUgPSBNYXRoLm1heCh3aWR0aCAvIGNoaWxkV2lkdGgsIGhlaWdodCAvIGNoaWxkSGVpZ2h0KTtcclxuICAgICAgICBzY2FsaW5nRWxlbWVudC5zdHlsZS50cmFuc2Zvcm0gPSBgc2NhbGUoJHtzY2FsZX0pYDtcclxuICAgICAgICBzY2FsaW5nRWxlbWVudC5zdHlsZS52aXNpYmlsaXR5ID0gXCJ2aXNpYmxlXCI7XHJcbiAgICB9LFxyXG59O1xyXG5cclxuY29uc3QgRml0U3RyYXRlZ3lTdHJldGNoOiBGaXRTdHJhdGVneVR5cGUgPSB7XHJcbiAgICB3cmFwRWxlbWVudChlbGVtZW50OiBIVE1MRWxlbWVudCk6IEhUTUxFbGVtZW50IHtcclxuICAgICAgICBjb25zdCBzaXppbmdFbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImRpdlwiKTtcclxuICAgICAgICBzaXppbmdFbGVtZW50LnN0eWxlLmRpc3BsYXkgPSBcImZsZXhcIjtcclxuICAgICAgICBzaXppbmdFbGVtZW50LnN0eWxlLmp1c3RpZnlDb250ZW50ID0gXCJjZW50ZXJcIjtcclxuICAgICAgICBzaXppbmdFbGVtZW50LnN0eWxlLmFsaWduSXRlbXMgPSBcImNlbnRlclwiO1xyXG4gICAgICAgIGNvbnN0IHNjYWxpbmdFbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImRpdlwiKTtcclxuICAgICAgICBzY2FsaW5nRWxlbWVudC5zdHlsZS52aXNpYmlsaXR5ID0gXCJoaWRkZW5cIjtcclxuICAgICAgICBzY2FsaW5nRWxlbWVudC5hcHBlbmRDaGlsZChlbGVtZW50KTtcclxuICAgICAgICBzaXppbmdFbGVtZW50LmFwcGVuZENoaWxkKHNjYWxpbmdFbGVtZW50KTtcclxuICAgICAgICByZXR1cm4gc2l6aW5nRWxlbWVudDtcclxuICAgIH0sXHJcbiAgICB1cGRhdGVTaXplKHNpemluZ0VsZW1lbnQ6IEhUTUxFbGVtZW50LCB3aWR0aDogbnVtYmVyLCBoZWlnaHQ6IG51bWJlcikge1xyXG4gICAgICAgIGNvbnN0IHNjYWxpbmdFbGVtZW50ID0gc2l6aW5nRWxlbWVudC5maXJzdEVsZW1lbnRDaGlsZCEgYXMgSFRNTEVsZW1lbnQ7XHJcbiAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS53aWR0aCA9IGAke3dpZHRofXB4YDtcclxuICAgICAgICBzaXppbmdFbGVtZW50LnN0eWxlLmhlaWdodCA9IGAke2hlaWdodH1weGA7XHJcblxyXG4gICAgICAgIGNvbnN0IFtjaGlsZFdpZHRoLCBjaGlsZEhlaWdodF0gPSBbc2NhbGluZ0VsZW1lbnQhLm9mZnNldFdpZHRoLCBzY2FsaW5nRWxlbWVudCEub2Zmc2V0SGVpZ2h0XTtcclxuICAgICAgICBzY2FsaW5nRWxlbWVudC5zdHlsZS50cmFuc2Zvcm0gPSBgc2NhbGUoJHt3aWR0aCAvIGNoaWxkV2lkdGh9LCAke2hlaWdodCAvIGNoaWxkSGVpZ2h0fSlgO1xyXG4gICAgICAgIHNjYWxpbmdFbGVtZW50LnN0eWxlLnZpc2liaWxpdHkgPSBcInZpc2libGVcIjtcclxuICAgIH0sXHJcbn07XHJcblxyXG5jb25zdCBGaXRTdHJhdGVneU5vbmU6IEZpdFN0cmF0ZWd5VHlwZSA9IHtcclxuICAgIHdyYXBFbGVtZW50KGVsZW1lbnQ6IEhUTUxFbGVtZW50KTogSFRNTEVsZW1lbnQge1xyXG4gICAgICAgIHJldHVybiBlbGVtZW50O1xyXG4gICAgfSxcclxuICAgIHVwZGF0ZVNpemUoc2l6aW5nRWxlbWVudDogSFRNTEVsZW1lbnQsIHdpZHRoOiBudW1iZXIsIGhlaWdodDogbnVtYmVyKSB7XHJcbiAgICAgICAgaWYgKHNpemluZ0VsZW1lbnQpIHtcclxuICAgICAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS53aWR0aCA9IGAke3dpZHRofXB4YDtcclxuICAgICAgICAgICAgc2l6aW5nRWxlbWVudC5zdHlsZS5oZWlnaHQgPSBgJHtoZWlnaHR9cHhgO1xyXG4gICAgICAgIH1cclxuICAgIH0sXHJcbn07XHJcblxyXG5leHBvcnQgY29uc3QgRml0U3RyYXRlZ3kgPSB7XHJcbiAgICBDT05UQUlOOiBGaXRTdHJhdGVneUNvbnRhaW4sXHJcbiAgICBDT1ZFUjogRml0U3RyYXRlZ3lDb3ZlcixcclxuICAgIFNUUkVUQ0g6IEZpdFN0cmF0ZWd5U3RyZXRjaCxcclxuICAgIE5PTkU6IEZpdFN0cmF0ZWd5Tm9uZSxcclxufTtcclxuIiwiaW1wb3J0IHsgTWVzaCB9IGZyb20gXCJjb3JlL01lc2hlcy9tZXNoXCI7XHJcbmltcG9ydCB7IENyZWF0ZVBsYW5lVmVydGV4RGF0YSB9IGZyb20gXCJjb3JlL01lc2hlcy9CdWlsZGVycy9wbGFuZUJ1aWxkZXJcIjtcclxuaW1wb3J0IHsgU3RhbmRhcmRNYXRlcmlhbCB9IGZyb20gXCJjb3JlL01hdGVyaWFscy9zdGFuZGFyZE1hdGVyaWFsXCI7XHJcbmltcG9ydCB7IE1hdHJpeCB9IGZyb20gXCJjb3JlL01hdGhzL21hdGhcIjtcclxuaW1wb3J0IHsgUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvciB9IGZyb20gXCIuL3BvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3JcIjtcclxuaW1wb3J0IHR5cGUgeyBTY2VuZSB9IGZyb20gXCJjb3JlL3NjZW5lXCI7XHJcbmltcG9ydCB7IExvZ2dlciB9IGZyb20gXCJjb3JlL01pc2MvbG9nZ2VyXCI7XHJcbmltcG9ydCB0eXBlIHsgRml0U3RyYXRlZ3lUeXBlIH0gZnJvbSBcIi4vZml0U3RyYXRlZ3lcIjtcclxuaW1wb3J0IHsgRml0U3RyYXRlZ3kgfSBmcm9tIFwiLi9maXRTdHJhdGVneVwiO1xyXG5cclxuLyoqXHJcbiAqIFRoaXMgY2xhc3MgcmVwcmVzZW50cyBIVE1MIGNvbnRlbnQgdGhhdCB3ZSB3YW50IHRvIHJlbmRlciBhcyB0aG91Z2ggaXQgaXMgcGFydCBvZiB0aGUgc2NlbmUuICBUaGUgSFRNTCBjb250ZW50IGlzIGFjdHVhbGx5XHJcbiAqIHJlbmRlcmVkIGJlbG93IHRoZSBjYW52YXMsIGJ1dCBhIGRlcHRoIG1hc2sgaXMgY3JlYXRlZCBieSB0aGlzIGNsYXNzIHRoYXQgd3JpdGVzIHRvIHRoZSBkZXB0aCBidWZmZXIgYnV0IGRvZXMgbm90XHJcbiAqIHdyaXRlIHRvIHRoZSBjb2xvciBidWZmZXIsIGVmZmVjdGl2ZWx5IHB1bmNoaW5nIGEgaG9sZSBpbiB0aGUgY2FudmFzLiAgQ1NTIHRyYW5zZm9ybXMgYXJlIHVzZWQgdG8gc2NhbGUsIHRyYW5zbGF0ZSwgYW5kIHJvdGF0ZVxyXG4gKiB0aGUgSFRNTCBjb250ZW50IHNvIHRoYXQgaXQgbWF0Y2hlcyB0aGUgY2FtZXJhIGFuZCBtZXNoIG9yaWVudGF0aW9uLiAgVGhlIGNsYXNzIHN1cHBvcnRzIGludGVyYWN0aW9ucyBpbiBlZGl0YWJsZSBhbmQgbm9uLWVkaXRhYmxlIG1vZGUuXHJcbiAqIEluIG5vbi1lZGl0YWJsZSBtb2RlICh0aGUgZGVmYXVsdCksIGV2ZW50cyBhcmUgcGFzc2VkIHRvIHRoZSBIVE1MIGNvbnRlbnQgd2hlbiB0aGUgcG9pbnRlciBpcyBvdmVyIHRoZSBtYXNrIChhbmQgbm90IG9jY2x1ZGVkIGJ5IG90aGVyIG1lc2hlc1xyXG4gKiBpbiB0aGUgc2NlbmUpLlxyXG4gKiAjSFZIWUpDIzVcclxuICogI0IxN1RDNyMxMTJcclxuICovXHJcbmV4cG9ydCBjbGFzcyBIdG1sTWVzaCBleHRlbmRzIE1lc2gge1xyXG4gICAgLyoqXHJcbiAgICAgKiBIZWxwcyBpZGVudGlmeWluZyBhIGh0bWwgbWVzaCBmcm9tIGEgcmVndWxhciBtZXNoXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBnZXQgaXNIdG1sTWVzaCgpIHtcclxuICAgICAgICByZXR1cm4gdHJ1ZTtcclxuICAgIH1cclxuXHJcbiAgICAvLyBPdmVycmlkZSB0aGUgc3VwZXIgY2xhc3MncyBfaXNFbmFibGVkIHByb3BlcnR5IHNvIHdlIGNhbiBjb250cm9sIHdoZW4gdGhlIG1lc2hcclxuICAgIC8vIGlzIGVuYWJsZWQuICBJLmUuLCB3ZSBkb24ndCB3YW50IHRvIHJlbmRlciB0aGUgbWVzaCB1bnRpbCB0aGVyZSBpcyBjb250ZW50IHRvIHNob3cuXHJcbiAgICBwcml2YXRlIF9lbmFibGVkID0gZmFsc2U7XHJcblxyXG4gICAgLy8gVGhlIG1lc2ggaXMgcmVhZHkgd2hlbiBjb250ZW50IGhhcyBiZWVuIHNldCBhbmQgdGhlIGNvbnRlbnQgc2l6ZSBoYXMgYmVlbiBzZXRcclxuICAgIC8vIFRoZSBmb3JtZXIgaXMgZG9uZSBieSB0aGUgdXNlciwgdGhlIGxhdHRlciBpcyBkb25lIGJ5IHRoZSByZW5kZXJlci5cclxuICAgIHByaXZhdGUgX3JlYWR5ID0gZmFsc2U7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBAaW50ZXJuYWxcclxuICAgICAqL1xyXG4gICAgcHVibGljIF9pc0NhbnZhc092ZXJsYXkgPSBmYWxzZTtcclxuXHJcbiAgICBwcml2YXRlIF9yZXF1aXJlc1VwZGF0ZSA9IHRydWU7XHJcblxyXG4gICAgcHJpdmF0ZSBfZWxlbWVudD86IEhUTUxFbGVtZW50O1xyXG4gICAgcHJpdmF0ZSBfd2lkdGg/OiBudW1iZXI7XHJcbiAgICBwcml2YXRlIF9oZWlnaHQ/OiBudW1iZXI7XHJcblxyXG4gICAgcHJpdmF0ZSBfaW52ZXJzZVNjYWxlTWF0cml4OiBNYXRyaXggfCBudWxsID0gbnVsbDtcclxuXHJcbiAgICBwcml2YXRlIF9jYXB0dXJlT25Qb2ludGVyRW50ZXI6IGJvb2xlYW4gPSB0cnVlO1xyXG4gICAgcHJpdmF0ZSBfcG9pbnRlckV2ZW50Q2FwdHVyZUJlaGF2aW9yOiBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yIHwgbnVsbCA9IG51bGw7XHJcblxyXG4gICAgcHJpdmF0ZSBfc291cmNlV2lkdGg6IG51bWJlciB8IG51bGwgPSBudWxsO1xyXG4gICAgcHJpdmF0ZSBfc291cmNlSGVpZ2h0OiBudW1iZXIgfCBudWxsID0gbnVsbDtcclxuXHJcbiAgICAvKipcclxuICAgICAqIFJldHVybiB0aGUgc291cmNlIHdpZHRoIG9mIHRoZSBjb250ZW50IGluIHBpeGVsc1xyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZ2V0IHNvdXJjZVdpZHRoKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl9zb3VyY2VXaWR0aDtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIFJldHVybiB0aGUgc291cmNlIGhlaWdodCBvZiB0aGUgY29udGVudCBpbiBwaXhlbHNcclxuICAgICAqL1xyXG4gICAgcHVibGljIGdldCBzb3VyY2VIZWlnaHQoKSB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuX3NvdXJjZUhlaWdodDtcclxuICAgIH1cclxuXHJcbiAgICBwcml2YXRlIF93b3JsZE1hdHJpeFVwZGF0ZU9ic2VydmVyOiBhbnk7XHJcblxyXG4gICAgcHJpdmF0ZSBfZml0U3RyYXRlZ3k6IEZpdFN0cmF0ZWd5VHlwZSA9IEZpdFN0cmF0ZWd5Lk5PTkU7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDb250cnVjdCBhbiBpbnN0YW5jZSBvZiBIdG1sTWVzaFxyXG4gICAgICogQHBhcmFtIHNjZW5lXHJcbiAgICAgKiBAcGFyYW0gaWQgVGhlIGlkIG9mIHRoZSBtZXNoLiAgV2lsbCBiZSB1c2VkIGFzIHRoZSBpZCBvZiB0aGUgSFRNTCBlbGVtZW50IGFzIHdlbGwuXHJcbiAgICAgKiBAcGFyYW0gb3B0aW9ucyBvYmplY3Qgd2l0aCBvcHRpb25hbCBwYXJhbWV0ZXJzXHJcbiAgICAgKi9cclxuICAgIGNvbnN0cnVjdG9yKHNjZW5lOiBTY2VuZSwgaWQ6IHN0cmluZywgeyBjYXB0dXJlT25Qb2ludGVyRW50ZXIgPSB0cnVlLCBpc0NhbnZhc092ZXJsYXkgPSBmYWxzZSwgZml0U3RyYXRlZ3kgPSBGaXRTdHJhdGVneS5OT05FIH0gPSB7fSkge1xyXG4gICAgICAgIHN1cGVyKGlkLCBzY2VuZSk7XHJcblxyXG4gICAgICAgIC8vIFJlcXVpcmVzIGEgYnJvd3NlciB0byB3b3JrLiAgQmFpbCBpZiB3ZSBhcmVuJ3QgcnVubmluZyBpbiBhIGJyb3dzZXJcclxuICAgICAgICBpZiAodHlwZW9mIGRvY3VtZW50ID09PSBcInVuZGVmaW5lZFwiKSB7XHJcbiAgICAgICAgICAgIExvZ2dlci5XYXJuKGBDcmVhdGluZyBhbiBpbnN0YW5jZSBvZiBhbiBIdG1sTWVzaCB3aXRoIGlkICR7aWR9IG91dHNpZGUgb2YgYSBicm93c2VyLiAgVGhlIG1lc2ggd2lsbCBub3QgYmUgdmlzaWJsZS5gKTtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgdGhpcy5fZml0U3RyYXRlZ3kgPSBmaXRTdHJhdGVneTtcclxuICAgICAgICB0aGlzLl9pc0NhbnZhc092ZXJsYXkgPSBpc0NhbnZhc092ZXJsYXk7XHJcbiAgICAgICAgdGhpcy5fY3JlYXRlTWFzaygpO1xyXG4gICAgICAgIHRoaXMuX2VsZW1lbnQgPSB0aGlzLl9jcmVhdGVFbGVtZW50KCk7XHJcblxyXG4gICAgICAgIC8vIFNldCBlbmFibGVkIGJ5IGRlZmF1bHQsIHNvIHRoaXMgd2lsbCBzaG93IGFzIHNvb24gYXMgaXQncyByZWFkeVxyXG4gICAgICAgIHRoaXMuc2V0RW5hYmxlZCh0cnVlKTtcclxuXHJcbiAgICAgICAgdGhpcy5fY2FwdHVyZU9uUG9pbnRlckVudGVyID0gY2FwdHVyZU9uUG9pbnRlckVudGVyO1xyXG5cclxuICAgICAgICAvLyBDcmVhdGUgYSBiZWhhdmlvciB0byBjYXB0dXJlIHBvaW50ZXIgZXZlbnRzXHJcbiAgICAgICAgdGhpcy5fcG9pbnRlckV2ZW50Q2FwdHVyZUJlaGF2aW9yID0gbmV3IFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IodGhpcy5jYXB0dXJlUG9pbnRlckV2ZW50cy5iaW5kKHRoaXMpLCB0aGlzLnJlbGVhc2VQb2ludGVyRXZlbnRzLmJpbmQodGhpcyksIHtcclxuICAgICAgICAgICAgY2FwdHVyZU9uUG9pbnRlckVudGVyOiB0aGlzLl9jYXB0dXJlT25Qb2ludGVyRW50ZXIsXHJcbiAgICAgICAgfSk7XHJcbiAgICAgICAgdGhpcy5hZGRCZWhhdmlvcih0aGlzLl9wb2ludGVyRXZlbnRDYXB0dXJlQmVoYXZpb3IpO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogVGhlIHdpZHRoIG9mIHRoZSBjb250ZW50IGluIHBpeGVsc1xyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZ2V0IHdpZHRoKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl93aWR0aDtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIFRoZSBoZWlnaHQgb2YgdGhlIGNvbnRlbnQgaW4gcGl4ZWxzXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBnZXQgaGVpZ2h0KCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl9oZWlnaHQ7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBUaGUgSFRNTCBlbGVtZW50IHRoYXQgaXMgYmVpbmcgcmVuZGVyZWQgYXMgYSBtZXNoXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBnZXQgZWxlbWVudCgpIHtcclxuICAgICAgICByZXR1cm4gdGhpcy5fZWxlbWVudDtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIFRydWUgaWYgdGhlIG1lc2ggaGFzIGJlZW4gbW92ZWQsIHJvdGF0ZWQsIG9yIHNjYWxlZCBzaW5jZSB0aGUgbGFzdCB0aW1lIHRoaXNcclxuICAgICAqIHByb3BlcnR5IHdhcyByZWFkLiAgVGhpcyBwcm9wZXJ0eSBpcyByZXNldCB0byBmYWxzZSBhZnRlciByZWFkaW5nLlxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZ2V0IHJlcXVpcmVzVXBkYXRlKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl9yZXF1aXJlc1VwZGF0ZTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIEVuYWJsZSBjYXB0dXJlIGZvciB0aGUgcG9pbnRlciB3aGVuIGVudGVyaW5nIHRoZSBtZXNoIGFyZWFcclxuICAgICAqL1xyXG4gICAgcHVibGljIHNldCBjYXB0dXJlT25Qb2ludGVyRW50ZXIoY2FwdHVyZU9uUG9pbnRlckVudGVyOiBib29sZWFuKSB7XHJcbiAgICAgICAgdGhpcy5fY2FwdHVyZU9uUG9pbnRlckVudGVyID0gY2FwdHVyZU9uUG9pbnRlckVudGVyO1xyXG4gICAgICAgIGlmICh0aGlzLl9wb2ludGVyRXZlbnRDYXB0dXJlQmVoYXZpb3IpIHtcclxuICAgICAgICAgICAgdGhpcy5fcG9pbnRlckV2ZW50Q2FwdHVyZUJlaGF2aW9yLmNhcHR1cmVPblBvaW50ZXJFbnRlciA9IGNhcHR1cmVPblBvaW50ZXJFbnRlcjtcclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBEaXNwb3NlcyBvZiB0aGUgbWVzaCBhbmQgdGhlIEhUTUwgZWxlbWVudFxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgb3ZlcnJpZGUgZGlzcG9zZSgpIHtcclxuICAgICAgICBzdXBlci5kaXNwb3NlKCk7XHJcbiAgICAgICAgdGhpcy5fZWxlbWVudD8ucmVtb3ZlKCk7XHJcbiAgICAgICAgdGhpcy5fZWxlbWVudCA9IHVuZGVmaW5lZDtcclxuICAgICAgICBpZiAodGhpcy5fcG9pbnRlckV2ZW50Q2FwdHVyZUJlaGF2aW9yKSB7XHJcbiAgICAgICAgICAgIHRoaXMuX3BvaW50ZXJFdmVudENhcHR1cmVCZWhhdmlvci5kaXNwb3NlKCk7XHJcbiAgICAgICAgICAgIHRoaXMuX3BvaW50ZXJFdmVudENhcHR1cmVCZWhhdmlvciA9IG51bGw7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQGludGVybmFsXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBfbWFya0FzVXBkYXRlZCgpIHtcclxuICAgICAgICB0aGlzLl9yZXF1aXJlc1VwZGF0ZSA9IGZhbHNlO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogU2V0cyB0aGUgY29udGVudCBvZiB0aGUgZWxlbWVudCB0byB0aGUgc3BlY2lmaWVkIGNvbnRlbnQgYWRqdXN0aW5nIHRoZSBtZXNoIHNjYWxlIHRvIG1hdGNoIGFuZCBtYWtpbmcgaXQgdmlzaWJsZS5cclxuICAgICAqIElmIHRoZSB0aGUgc3BlY2lmaWVkIGNvbnRlbnQgaXMgdW5kZWZpbmVkLCB0aGVuIGl0IHdpbGwgbWFrZSB0aGUgbWVzaCBpbnZpc2libGUuICBJbiBlaXRoZXIgY2FzZSBpdCB3aWxsIGNsZWFyIHRoZVxyXG4gICAgICogZWxlbWVudCBjb250ZW50IGZpcnN0LlxyXG4gICAgICogQHBhcmFtIGVsZW1lbnQgVGhlIGVsZW1lbnQgdG8gcmVuZGVyIGFzIGEgbWVzaFxyXG4gICAgICogQHBhcmFtIHdpZHRoIFRoZSB3aWR0aCBvZiB0aGUgbWVzaCBpbiBCYWJ5bG9uIHVuaXRzXHJcbiAgICAgKiBAcGFyYW0gaGVpZ2h0IFRoZSBoZWlnaHQgb2YgdGhlIG1lc2ggaW4gQmFieWxvbiB1bml0c1xyXG4gICAgICovXHJcbiAgICBzZXRDb250ZW50KGVsZW1lbnQ6IEhUTUxFbGVtZW50LCB3aWR0aDogbnVtYmVyLCBoZWlnaHQ6IG51bWJlcikge1xyXG4gICAgICAgIC8vIElmIGNvbnRlbnQgaXMgY2hhbmdlZCwgd2UgYXJlIG5vIGxvbmdlciByZWFkeVxyXG4gICAgICAgIHRoaXMuX3NldEFzUmVhZHkoZmFsc2UpO1xyXG5cclxuICAgICAgICAvLyBBbHNvIGludmFsaWRhdGUgdGhlIHNvdXJjZSB3aWR0aCBhbmQgaGVpZ2h0XHJcbiAgICAgICAgdGhpcy5fc291cmNlV2lkdGggPSBudWxsO1xyXG4gICAgICAgIHRoaXMuX3NvdXJjZUhlaWdodCA9IG51bGw7XHJcblxyXG4gICAgICAgIGlmICghdGhpcy5fZWxlbWVudCkge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICB0aGlzLl93aWR0aCA9IHdpZHRoO1xyXG4gICAgICAgIHRoaXMuX2hlaWdodCA9IGhlaWdodDtcclxuICAgICAgICB0aGlzLl9yZXF1aXJlc1VwZGF0ZSA9IHRydWU7XHJcblxyXG4gICAgICAgIHRoaXMuc2NhbGluZy5zZXRBbGwoMSk7XHJcblxyXG4gICAgICAgIGlmIChlbGVtZW50KSB7XHJcbiAgICAgICAgICAgIHRoaXMuX2VsZW1lbnQhLmFwcGVuZENoaWxkKHRoaXMuX2ZpdFN0cmF0ZWd5LndyYXBFbGVtZW50KGVsZW1lbnQpKTtcclxuXHJcbiAgICAgICAgICAgIHRoaXMuX3VwZGF0ZVNjYWxlSWZOZWNlc3NhcnkoKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGlmICh0aGlzLnNvdXJjZVdpZHRoICYmIHRoaXMuc291cmNlSGVpZ2h0KSB7XHJcbiAgICAgICAgICAgIHRoaXMuX3NldEFzUmVhZHkodHJ1ZSk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8vIE92ZXJpZGVzIEJBQllMT04uTWVzaC5zZXRFbmFibGVkXHJcbiAgICBwdWJsaWMgb3ZlcnJpZGUgc2V0RW5hYmxlZChlbmFibGVkOiBib29sZWFuKSB7XHJcbiAgICAgICAgLy8gQ2FwdHVyZSByZXF1ZXN0ZWQgZW5hYmxlZCBzdGF0ZVxyXG4gICAgICAgIHRoaXMuX2VuYWJsZWQgPSBlbmFibGVkO1xyXG5cclxuICAgICAgICAvLyBJZiBkaXNhYmxpbmcgb3IgZW5hYmxpbmcgYW5kIHdlIGFyZSByZWFkeVxyXG4gICAgICAgIGlmICghZW5hYmxlZCB8fCB0aGlzLl9yZWFkeSkge1xyXG4gICAgICAgICAgICB0aGlzLl9kb1NldEVuYWJsZWQoZW5hYmxlZCk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogU2V0cyB0aGUgY29udGVudCBzaXplIGluIHBpeGVsc1xyXG4gICAgICogQHBhcmFtIHdpZHRoIHdpZHRoIG9mIHRoZSBzb3VyY2VcclxuICAgICAqIEBwYXJhbSBoZWlnaHQgaGVpZ2h0IG9mIHRoZSBzb3VyY2VcclxuICAgICAqL1xyXG4gICAgcHVibGljIHNldENvbnRlbnRTaXplUHgod2lkdGg6IG51bWJlciwgaGVpZ2h0OiBudW1iZXIpIHtcclxuICAgICAgICB0aGlzLl9zb3VyY2VXaWR0aCA9IHdpZHRoO1xyXG4gICAgICAgIHRoaXMuX3NvdXJjZUhlaWdodCA9IGhlaWdodDtcclxuXHJcbiAgICAgICAgaWYgKCF0aGlzLl9lbGVtZW50IHx8ICF0aGlzLl9lbGVtZW50LmZpcnN0RWxlbWVudENoaWxkKSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMuX2ZpdFN0cmF0ZWd5LnVwZGF0ZVNpemUodGhpcy5fZWxlbWVudC5maXJzdEVsZW1lbnRDaGlsZCEgYXMgSFRNTEVsZW1lbnQsIHdpZHRoLCBoZWlnaHQpO1xyXG5cclxuICAgICAgICB0aGlzLl91cGRhdGVTY2FsZUlmTmVjZXNzYXJ5KCk7XHJcblxyXG4gICAgICAgIGlmICh0aGlzLndpZHRoICYmIHRoaXMuaGVpZ2h0KSB7XHJcbiAgICAgICAgICAgIHRoaXMuX3NldEFzUmVhZHkodHJ1ZSk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfc2V0QXNSZWFkeShyZWFkeTogYm9vbGVhbikge1xyXG4gICAgICAgIHRoaXMuX3JlYWR5ID0gcmVhZHk7XHJcbiAgICAgICAgaWYgKHJlYWR5KSB7XHJcbiAgICAgICAgICAgIHRoaXMuX2RvU2V0RW5hYmxlZCh0aGlzLl9lbmFibGVkKTtcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICB0aGlzLl9kb1NldEVuYWJsZWQoZmFsc2UpO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX2RvU2V0RW5hYmxlZChlbmFibGVkOiBib29sZWFuKSB7XHJcbiAgICAgICAgaWYgKCF0aGlzLl9lbGVtZW50KSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vaWYgZW5hYmxlZCwgdGhlbiBzdGFydCBsaXN0ZW5pbmcgZm9yIGNoYW5nZXMgdG8gdGhlXHJcbiAgICAgICAgLy8gc2NhbGluZywgcm90YXRpb24sIGFuZCBwb3NpdGlvbi4gIG90aGVyd2lzZSBzdG9wIGxpc3RlbmluZ1xyXG4gICAgICAgIGlmIChlbmFibGVkICYmICF0aGlzLl93b3JsZE1hdHJpeFVwZGF0ZU9ic2VydmVyKSB7XHJcbiAgICAgICAgICAgIHRoaXMuX3dvcmxkTWF0cml4VXBkYXRlT2JzZXJ2ZXIgPSB0aGlzLm9uQWZ0ZXJXb3JsZE1hdHJpeFVwZGF0ZU9ic2VydmFibGUuYWRkKCgpID0+IHtcclxuICAgICAgICAgICAgICAgIHRoaXMuX3JlcXVpcmVzVXBkYXRlID0gdHJ1ZTtcclxuICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgfSBlbHNlIGlmICghZW5hYmxlZCkge1xyXG4gICAgICAgICAgICB0aGlzLl93b3JsZE1hdHJpeFVwZGF0ZU9ic2VydmVyPy5yZW1vdmUoKTtcclxuICAgICAgICAgICAgdGhpcy5fd29ybGRNYXRyaXhVcGRhdGVPYnNlcnZlciA9IG51bGw7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBJZiBlbmFibGVkLCB0aGVuIHJldmVydCB0aGUgY29udGVudCBlbGVtZW50IGRpc3BsYXlcclxuICAgICAgICAvLyBvdGhlcndpc2UgaGlkZSBpdFxyXG4gICAgICAgIHRoaXMuX2VsZW1lbnQhLnN0eWxlLmRpc3BsYXkgPSBlbmFibGVkID8gXCJcIiA6IFwibm9uZVwiO1xyXG4gICAgICAgIC8vIENhcHR1cmUgdGhlIGNvbnRlbnQgeiBpbmRleFxyXG4gICAgICAgIHRoaXMuX3NldEVsZW1lbnRaSW5kZXgodGhpcy5wb3NpdGlvbi56ICogLTEwMDAwKTtcclxuICAgICAgICBzdXBlci5zZXRFbmFibGVkKGVuYWJsZWQpO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfdXBkYXRlU2NhbGVJZk5lY2Vzc2FyeSgpIHtcclxuICAgICAgICAvLyBJZiB3ZSBoYXZlIHNldENvbnRlbnQgYmVmb3JlLCB0aGUgY29udGVudCBzY2FsZSBpcyBiYWtlZCBpbnRvIHRoZSBtZXNoLiAgSWYgd2UgZG9uJ3QgcmVzZXQgdGhlIHZlcnRpY2VzIHRvXHJcbiAgICAgICAgLy8gdGhlIG9yaWdpbmFsIHNpemUsIHRoZW4gd2Ugd2lsbCBtdWx0aXBseSB0aGUgc2NhbGUgd2hlbiB3ZSBiYWtlIHRoZSBzY2FsZSBiZWxvdy4gIEJ5IGFwcGx5aW5nIHRoZSBpbnZlcnNlLCB3ZSBiYWNrIG91dFxyXG4gICAgICAgIC8vIHRoZSBzY2FsaW5nIHRoYXQgaGFzIGJlZW4gZG9uZSBzbyB3ZSBhcmUgc3RhcnRpbmcgZnJvbSB0aGUgc2FtZSBwb2ludC5cclxuICAgICAgICAvLyBGaXJzdCByZXNldCB0aGUgc2NhbGUgdG8gMVxyXG4gICAgICAgIHRoaXMuc2NhbGluZy5zZXRBbGwoMSk7XHJcbiAgICAgICAgLy8gVGhlbiBiYWNrIG91dCB0aGUgb3JpZ2luYWwgdmVydGljZXMgY2hhbmdlcyB0byBtYXRjaCB0aGUgY29udGVudCBzY2FsZVxyXG4gICAgICAgIGlmICh0aGlzLl9pbnZlcnNlU2NhbGVNYXRyaXgpIHtcclxuICAgICAgICAgICAgdGhpcy5iYWtlVHJhbnNmb3JtSW50b1ZlcnRpY2VzKHRoaXMuX2ludmVyc2VTY2FsZU1hdHJpeCk7XHJcbiAgICAgICAgICAgIC8vIENsZWFyIG91dCB0aGUgbWF0cml4IHNvIGl0IGRvZXNuJ3QgZ2V0IGFwcGxpZWQgYWdhaW4gdW5sZXNzIHdlIHNjYWxlXHJcbiAgICAgICAgICAgIHRoaXMuX2ludmVyc2VTY2FsZU1hdHJpeCA9IG51bGw7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBTZXQgc2NhbGUgdG8gbWF0Y2ggY29udGVudC4gIE5vdGUgd2UgY2FuJ3QganVzdCBzY2FsZSB0aGUgbWVzaCwgYmVjYXVzZSB0aGF0IHdpbGwgc2NhbGUgdGhlIGNvbnRlbnQgYXMgd2VsbFxyXG4gICAgICAgIC8vIFdoYXQgd2UgbmVlZCB0byBkbyBpcyBjb21wdXRlIGEgc2NhbGUgbWF0cml4IGFuZCB0aGVuIGJha2UgdGhhdCBpbnRvIHRoZSBtZXNoIHZlcnRpY2VzLiAgVGhpcyB3aWxsIGxlYXZlIHRoZVxyXG4gICAgICAgIC8vIG1lc2ggc2NhbGUgYXQgMSwgc28gb3VyIGNvbnRlbnQgd2lsbCBzdGF5IGl0J3Mgb3JpZ2luYWwgd2lkdGggYW5kIGhlaWdodCB1bnRpbCB3ZSBzY2FsZSB0aGUgbWVzaC5cclxuICAgICAgICBjb25zdCBzY2FsZVggPSB0aGlzLl93aWR0aCB8fCAxO1xyXG4gICAgICAgIGNvbnN0IHNjYWxlWSA9IHRoaXMuX2hlaWdodCB8fCAxO1xyXG4gICAgICAgIGNvbnN0IHNjYWxlTWF0cml4ID0gTWF0cml4LlNjYWxpbmcoc2NhbGVYLCBzY2FsZVksIDEpO1xyXG4gICAgICAgIHRoaXMuYmFrZVRyYW5zZm9ybUludG9WZXJ0aWNlcyhzY2FsZU1hdHJpeCk7XHJcblxyXG4gICAgICAgIC8vIEdldCBhbiBpbnZlcnNlIG9mIHRoZSBzY2FsZSBtYXRyaXggdGhhdCB3ZSBjYW4gdXNlIHRvIGJhY2sgb3V0IHRoZSBzY2FsZSBjaGFuZ2VzIHdlIGhhdmUgbWFkZSBzb1xyXG4gICAgICAgIC8vIHdlIGRvbid0IG11bHRpcGx5IHRoZSBzY2FsZS5cclxuICAgICAgICB0aGlzLl9pbnZlcnNlU2NhbGVNYXRyaXggPSBuZXcgTWF0cml4KCk7XHJcbiAgICAgICAgc2NhbGVNYXRyaXguaW52ZXJ0VG9SZWYodGhpcy5faW52ZXJzZVNjYWxlTWF0cml4KTtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX2NyZWF0ZU1hc2soKSB7XHJcbiAgICAgICAgY29uc3QgdmVydGV4RGF0YSA9IENyZWF0ZVBsYW5lVmVydGV4RGF0YSh7IHdpZHRoOiAxLCBoZWlnaHQ6IDEgfSk7XHJcbiAgICAgICAgdmVydGV4RGF0YS5hcHBseVRvTWVzaCh0aGlzKTtcclxuXHJcbiAgICAgICAgY29uc3Qgc2NlbmUgPSB0aGlzLmdldFNjZW5lKCk7XHJcbiAgICAgICAgdGhpcy5jaGVja0NvbGxpc2lvbnMgPSB0cnVlO1xyXG5cclxuICAgICAgICBjb25zdCBkZXB0aE1hc2sgPSBuZXcgU3RhbmRhcmRNYXRlcmlhbChgJHt0aGlzLmlkfS1tYXRgLCBzY2VuZSk7XHJcbiAgICAgICAgaWYgKCF0aGlzLl9pc0NhbnZhc092ZXJsYXkpIHtcclxuICAgICAgICAgICAgZGVwdGhNYXNrLmJhY2tGYWNlQ3VsbGluZyA9IGZhbHNlO1xyXG4gICAgICAgICAgICBkZXB0aE1hc2suZGlzYWJsZUNvbG9yV3JpdGUgPSB0cnVlO1xyXG4gICAgICAgICAgICBkZXB0aE1hc2suZGlzYWJsZUxpZ2h0aW5nID0gdHJ1ZTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMubWF0ZXJpYWwgPSBkZXB0aE1hc2s7XHJcblxyXG4gICAgICAgIC8vIE9wdGltaXphdGlvbiAtIEZyZWV6ZSBtYXRlcmlhbCBzaW5jZSBpdCBuZXZlciBuZWVkcyB0byBjaGFuZ2VcclxuICAgICAgICB0aGlzLm1hdGVyaWFsLmZyZWV6ZSgpO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfc2V0RWxlbWVudFpJbmRleCh6SW5kZXg6IG51bWJlcikge1xyXG4gICAgICAgIGlmICh0aGlzLl9lbGVtZW50KSB7XHJcbiAgICAgICAgICAgIHRoaXMuX2VsZW1lbnQhLnN0eWxlLnpJbmRleCA9IGAke3pJbmRleH1gO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENhbGxiYWNrIHVzZWQgYnkgdGhlIFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IgdG8gY2FwdHVyZSBwb2ludGVyIGV2ZW50c1xyXG4gICAgICovXHJcbiAgICBjYXB0dXJlUG9pbnRlckV2ZW50cygpIHtcclxuICAgICAgICBpZiAoIXRoaXMuX2VsZW1lbnQpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gRW5hYmxlIGRvbSBjb250ZW50IHRvIGNhcHR1cmUgcG9pbnRlciBldmVudHNcclxuICAgICAgICB0aGlzLl9lbGVtZW50LnN0eWxlLnBvaW50ZXJFdmVudHMgPSBcImF1dG9cIjtcclxuXHJcbiAgICAgICAgLy8gU3VwcmVzcyBldmVudHMgb3V0c2lkZSBvZiB0aGUgZG9tIGNvbnRlbnRcclxuICAgICAgICBkb2N1bWVudC5nZXRFbGVtZW50c0J5VGFnTmFtZShcImJvZHlcIilbMF0uc3R5bGUucG9pbnRlckV2ZW50cyA9IFwibm9uZVwiO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ2FsbGJhY2sgdXNlZCBieSB0aGUgUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvciB0byByZWxlYXNlIHBvaW50ZXIgZXZlbnRzXHJcbiAgICAgKi9cclxuICAgIHJlbGVhc2VQb2ludGVyRXZlbnRzKCkge1xyXG4gICAgICAgIGlmICghdGhpcy5fZWxlbWVudCkge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBFbmFibGUgcG9pbnRlciBldmVudHMgb24gY2FudmFzXHJcbiAgICAgICAgZG9jdW1lbnQuZ2V0RWxlbWVudHNCeVRhZ05hbWUoXCJib2R5XCIpWzBdLnN0eWxlLnBvaW50ZXJFdmVudHMgPSBcImF1dG9cIjtcclxuXHJcbiAgICAgICAgLy8gRGlzYWJsZSBwb2ludGVyIGV2ZW50cyBvbiBkb20gY29udGVudFxyXG4gICAgICAgIHRoaXMuX2VsZW1lbnQuc3R5bGUucG9pbnRlckV2ZW50cyA9IFwibm9uZVwiO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfY3JlYXRlRWxlbWVudCgpIHtcclxuICAgICAgICAvLyBSZXF1aXJlcyBhIGJyb3dzZXIgdG8gd29yay4gIEJhaWwgaWYgd2UgYXJlbid0IHJ1bm5pbmcgaW4gYSBicm93c2VyXHJcbiAgICAgICAgaWYgKHR5cGVvZiBkb2N1bWVudCA9PT0gXCJ1bmRlZmluZWRcIikge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGNvbnN0IGRpdiA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJkaXZcIik7XHJcbiAgICAgICAgZGl2LmlkID0gdGhpcy5pZDtcclxuICAgICAgICBkaXYuc3R5bGUuYmFja2dyb3VuZENvbG9yID0gdGhpcy5faXNDYW52YXNPdmVybGF5ID8gXCJ0cmFuc3BhcmVudFwiIDogXCIjMDAwXCI7XHJcbiAgICAgICAgZGl2LnN0eWxlLnpJbmRleCA9IFwiMVwiO1xyXG4gICAgICAgIGRpdi5zdHlsZS5wb3NpdGlvbiA9IFwiYWJzb2x1dGVcIjtcclxuICAgICAgICBkaXYuc3R5bGUucG9pbnRlckV2ZW50cyA9IFwibm9uZVwiO1xyXG4gICAgICAgIGRpdi5zdHlsZS5iYWNrZmFjZVZpc2liaWxpdHkgPSBcImhpZGRlblwiO1xyXG5cclxuICAgICAgICByZXR1cm4gZGl2O1xyXG4gICAgfVxyXG59XHJcbiIsImltcG9ydCB0eXBlIHsgU2NlbmUgfSBmcm9tIFwiY29yZS9zY2VuZVwiO1xyXG5pbXBvcnQgeyBNYXRyaXgsIFF1YXRlcm5pb24sIFZlY3RvcjMgfSBmcm9tIFwiY29yZS9NYXRocy9tYXRoXCI7XHJcblxyXG5pbXBvcnQgdHlwZSB7IEh0bWxNZXNoIH0gZnJvbSBcIi4vaHRtbE1lc2hcIjtcclxuaW1wb3J0IHsgQ2FtZXJhIH0gZnJvbSBcImNvcmUvQ2FtZXJhcy9jYW1lcmFcIjtcclxuaW1wb3J0IHR5cGUgeyBTdWJNZXNoIH0gZnJvbSBcImNvcmUvTWVzaGVzL3N1Yk1lc2hcIjtcclxuaW1wb3J0IHsgUmVuZGVyaW5nR3JvdXAgfSBmcm9tIFwiY29yZS9SZW5kZXJpbmcvcmVuZGVyaW5nR3JvdXBcIjtcclxuXHJcbmltcG9ydCB0eXBlIHsgT2JzZXJ2ZXIgfSBmcm9tIFwiY29yZS9NaXNjL29ic2VydmFibGVcIjtcclxuaW1wb3J0IHsgTG9nZ2VyIH0gZnJvbSBcImNvcmUvTWlzYy9sb2dnZXJcIjtcclxuaW1wb3J0IHR5cGUgeyBBYnN0cmFjdEVuZ2luZSB9IGZyb20gXCJjb3JlL0VuZ2luZXNcIjtcclxuXHJcbmNvbnN0IF9wb3NpdGlvblVwZGF0ZUZhaWxNZXNzYWdlID0gXCJGYWlsZWQgdG8gdXBkYXRlIGh0bWwgbWVzaCByZW5kZXJlciBwb3NpdGlvbiBkdWUgdG8gZmFpbHVyZSB0byBnZXQgY2FudmFzIHJlY3QuICBIdG1sTWVzaCBpbnN0YW5jZXMgbWF5IG5vdCByZW5kZXIgY29ycmVjdGx5XCI7XHJcbmNvbnN0IGJhYnlsb25Vbml0c1RvUGl4ZWxzID0gMTAwO1xyXG5cclxuLyoqXHJcbiAqIEEgZnVuY3Rpb24gdGhhdCBjb21wYXJlcyB0d28gc3VibWVzaGVzIGFuZCByZXR1cm5zIGEgbnVtYmVyIGluZGljYXRpbmcgd2hpY2hcclxuICogc2hvdWxkIGJlIHJlbmRlcmVkIGZpcnN0LlxyXG4gKi9cclxudHlwZSBSZW5kZXJPcmRlckZ1bmN0aW9uID0gKHN1Yk1lc2hBOiBTdWJNZXNoLCBzdWJNZXNoQjogU3ViTWVzaCkgPT4gbnVtYmVyO1xyXG5cclxudHlwZSBSZW5kZXJMYXllckVsZW1lbnRzID0ge1xyXG4gICAgY29udGFpbmVyOiBIVE1MRWxlbWVudDtcclxuICAgIGRvbUVsZW1lbnQ6IEhUTUxFbGVtZW50O1xyXG4gICAgY2FtZXJhRWxlbWVudDogSFRNTEVsZW1lbnQ7XHJcbn07XHJcblxyXG4vLyBSZXR1cm5zIGEgZnVuY3Rpb24gdGhhdCBlbnN1cmVzIHRoYXQgSHRtbE1lc2hlcyBhcmUgcmVuZGVyZWQgYmVmb3JlIGFsbCBvdGhlciBtZXNoZXMuXHJcbi8vIE5vdGUgdGhpcyB3aWxsIG9ubHkgYmUgYXBwbGllZCB0byBncm91cCAwLlxyXG4vLyBJZiBuZWl0aGVyIG1lc2ggaXMgYW4gSHRtbE1lc2gsIHRoZW4gdGhlIGRlZmF1bHQgcmVuZGVyIG9yZGVyIGlzIHVzZWRcclxuLy8gVGhpcyBwcmV2ZW50cyBIdG1sTWVzaGVzIGZyb20gYXBwZWFyaW5nIGluIGZyb250IG9mIG90aGVyIG1lc2hlcyB3aGVuIHRoZXkgYXJlIGJlaGluZCB0aGVtXHJcbmNvbnN0IHJlbmRlck9yZGVyRnVuYyA9IChkZWZhdWx0UmVuZGVyT3JkZXI6IFJlbmRlck9yZGVyRnVuY3Rpb24pOiBSZW5kZXJPcmRlckZ1bmN0aW9uID0+IHtcclxuICAgIHJldHVybiAoc3ViTWVzaEE6IFN1Yk1lc2gsIHN1Yk1lc2hCOiBTdWJNZXNoKSA9PiB7XHJcbiAgICAgICAgY29uc3QgbWVzaEEgPSBzdWJNZXNoQS5nZXRNZXNoKCk7XHJcbiAgICAgICAgY29uc3QgbWVzaEIgPSBzdWJNZXNoQi5nZXRNZXNoKCk7XHJcblxyXG4gICAgICAgIC8vIFVzZSBwcm9wZXJ0eSBjaGVjayBpbnN0ZWFkIG9mIGluc3RhbmNlb2Ygc2luY2UgaXQgaXMgbGVzcyBleHBlbnNpdmUgYW5kXHJcbiAgICAgICAgLy8gdGhpcyB3aWxsIGJlIGNhbGxlZCBtYW55IHRpbWVzIHBlciBmcmFtZVxyXG4gICAgICAgIGNvbnN0IG1lc2hBSXNIdG1sTWVzaCA9IChtZXNoQSBhcyBhbnkpW1wiaXNIdG1sTWVzaFwiXTtcclxuICAgICAgICBjb25zdCBtZXNoQklzSHRtbE1lc2ggPSAobWVzaEIgYXMgYW55KVtcImlzSHRtbE1lc2hcIl07XHJcbiAgICAgICAgaWYgKG1lc2hBSXNIdG1sTWVzaCkge1xyXG4gICAgICAgICAgICByZXR1cm4gbWVzaEJJc0h0bWxNZXNoID8gKG1lc2hBLmFic29sdXRlUG9zaXRpb24ueiA8PSBtZXNoQi5hYnNvbHV0ZVBvc2l0aW9uLnogPyAxIDogLTEpIDogLTE7XHJcbiAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgcmV0dXJuIG1lc2hCSXNIdG1sTWVzaCA/IDEgOiBkZWZhdWx0UmVuZGVyT3JkZXIoc3ViTWVzaEEsIHN1Yk1lc2hCKTtcclxuICAgICAgICB9XHJcbiAgICB9O1xyXG59O1xyXG5cclxuLyoqXHJcbiAqIEFuIGluc3RhbmNlIG9mIHRoaXMgaXMgcmVxdWlyZWQgdG8gcmVuZGVyIEh0bWxNZXNoZXMgaW4gdGhlIHNjZW5lLlxyXG4gKiBpZiB1c2luZyBIdG1sTWVzaGVzLCB5b3UgbXVzdCBub3Qgc2V0IHJlbmRlciBvcmRlciBmb3IgZ3JvdXAgMCB1c2luZ1xyXG4gKiBzY2VuZS5zZXRSZW5kZXJpbmdPcmRlci4gIFlvdSBtdXN0IGluc3RlYWQgcGFzcyB0aGUgY29tcGFyZSBmdW5jdGlvbnNcclxuICogdG8gdGhlIEh0bWxNZXNoUmVuZGVyZXIgY29uc3RydWN0b3IuICBJZiB5b3UgZG8gbm90LCB0aGVuIHlvdXIgcmVuZGVyXHJcbiAqIG9yZGVyIHdpbGwgYmUgb3ZlcndyaXR0ZW4gaWYgdGhlIEh0bWxNZXNoUmVuZGVyZXIgaXMgY3JlYXRlZCBhZnRlciBhbmRcclxuICogdGhlIEh0bWxNZXNoZXMgd2lsbCBub3QgcmVuZGVyIGNvcnJlY3RseSAodGhleSB3aWxsIGFwcGVhciBpbiBmcm9udCBvZlxyXG4gKiBtZXNoZXMgdGhhdCBhcmUgYWN0dWFsbHkgaW4gZnJvbnQgb2YgdGhlbSkgaWYgdGhlIEh0bWxNZXNoUmVuZGVyZXIgaXNcclxuICogY3JlYXRlZCBiZWZvcmUuXHJcbiAqL1xyXG5leHBvcnQgY2xhc3MgSHRtbE1lc2hSZW5kZXJlciB7XHJcbiAgICBwcml2YXRlIF9jb250YWluZXJJZD86IHN0cmluZztcclxuICAgIHByaXZhdGUgX2luU2NlbmVFbGVtZW50cz86IFJlbmRlckxheWVyRWxlbWVudHMgfCBudWxsO1xyXG4gICAgcHJpdmF0ZSBfb3ZlcmxheUVsZW1lbnRzPzogUmVuZGVyTGF5ZXJFbGVtZW50cyB8IG51bGw7XHJcbiAgICBwcml2YXRlIF9lbmdpbmU6IEFic3RyYWN0RW5naW5lO1xyXG5cclxuICAgIHByaXZhdGUgX2NhY2hlID0ge1xyXG4gICAgICAgIGNhbWVyYURhdGE6IHsgZm92OiAwLCBwb3NpdGlvbjogbmV3IFZlY3RvcjMoKSwgc3R5bGU6IFwiXCIgfSxcclxuICAgICAgICBodG1sTWVzaERhdGE6IG5ldyBXZWFrTWFwPG9iamVjdCwgeyBzdHlsZTogc3RyaW5nIH0+KCksXHJcbiAgICB9O1xyXG4gICAgcHJpdmF0ZSBfd2lkdGggPSAwO1xyXG4gICAgcHJpdmF0ZSBfaGVpZ2h0ID0gMDtcclxuICAgIHByaXZhdGUgX2hlaWdodEhhbGYgPSAwO1xyXG5cclxuICAgIHByaXZhdGUgX2NhbWVyYVdvcmxkTWF0cml4PzogTWF0cml4O1xyXG5cclxuICAgIC8vIENyZWF0ZSBzb21lIHJlZnMgdG8gYXZvaWQgY3JlYXRpbmcgbmV3IG9iamVjdHMgZXZlcnkgZnJhbWVcclxuICAgIHByaXZhdGUgX3RlbXAgPSB7XHJcbiAgICAgICAgc2NhbGVUcmFuc2Zvcm06IG5ldyBWZWN0b3IzKCksXHJcbiAgICAgICAgcm90YXRpb25UcmFuc2Zvcm06IG5ldyBRdWF0ZXJuaW9uKCksXHJcbiAgICAgICAgcG9zaXRpb25UcmFuc2Zvcm06IG5ldyBWZWN0b3IzKCksXHJcbiAgICAgICAgb2JqZWN0TWF0cml4OiBNYXRyaXguSWRlbnRpdHkoKSxcclxuICAgICAgICBjYW1lcmFXb3JsZE1hdHJpeDogTWF0cml4LklkZW50aXR5KCksXHJcbiAgICAgICAgY2FtZXJhUm90YXRpb25NYXRyaXg6IE1hdHJpeC5JZGVudGl0eSgpLFxyXG4gICAgICAgIGNhbWVyYVdvcmxkTWF0cml4QXNBcnJheTogbmV3IEFycmF5KDE2KSxcclxuICAgIH07XHJcblxyXG4gICAgLy8gS2VlcCB0cmFjayBvZiBEUFIgc28gd2UgY2FuIHJlc2l6ZSBpZiBEUFIgY2hhbmdlc1xyXG4gICAgLy8gT3RoZXJ3aXNlIHRoZSBET00gY29udGVudCB3aWxsIHNjYWxlLCBidXQgdGhlIG1lc2ggd29uJ3RcclxuICAgIHByaXZhdGUgX2xhc3REZXZpY2VQaXhlbFJhdGlvID0gd2luZG93LmRldmljZVBpeGVsUmF0aW87XHJcblxyXG4gICAgLy8gS2VlcCB0cmFjayBvZiBjYW1lcmEgbWF0cml4IGNoYW5nZXMgc28gd2Ugb25seSB1cGRhdGUgdGhlXHJcbiAgICAvLyBET00gZWxlbWVudCBzdHlsZXMgd2hlbiBuZWNlc3NhcnlcclxuICAgIHByaXZhdGUgX2NhbWVyYU1hdHJpeFVwZGF0ZWQgPSB0cnVlO1xyXG5cclxuICAgIC8vIEtlZXAgdHJhY2sgb2YgcG9zaXRpb24gY2hhbmdlcyBzbyB3ZSBvbmx5IHVwZGF0ZSB0aGUgRE9NIGVsZW1lbnRcclxuICAgIC8vIHN0eWxlcyB3aGVuIG5lY2Vzc2FyeVxyXG4gICAgcHJpdmF0ZSBfcHJldmlvdXNDYW52YXNEb2N1bWVudFBvc2l0aW9uID0ge1xyXG4gICAgICAgIHRvcDogMCxcclxuICAgICAgICBsZWZ0OiAwLFxyXG4gICAgfTtcclxuXHJcbiAgICBwcml2YXRlIF9yZW5kZXJPYnNlcnZlcjogT2JzZXJ2ZXI8U2NlbmU+IHwgbnVsbCA9IG51bGw7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDb250cnVjdCBhbiBpbnN0YW5jZSBvZiBIdG1sTWVzaFJlbmRlcmVyXHJcbiAgICAgKiBAcGFyYW0gc2NlbmVcclxuICAgICAqIEBwYXJhbSBvcHRpb25zIG9iamVjdCBjb250YWluaW5nIHRoZSBmb2xsb3dpbmcgb3B0aW9uYWwgcHJvcGVydGllczpcclxuICAgICAqIEByZXR1cm5zXHJcbiAgICAgKi9cclxuICAgIGNvbnN0cnVjdG9yKFxyXG4gICAgICAgIHNjZW5lOiBTY2VuZSxcclxuICAgICAgICB7XHJcbiAgICAgICAgICAgIHBhcmVudENvbnRhaW5lcklkID0gbnVsbCxcclxuICAgICAgICAgICAgX2NvbnRhaW5lcklkID0gXCJjc3MtY29udGFpbmVyXCIsXHJcbiAgICAgICAgICAgIGVuYWJsZU92ZXJsYXlSZW5kZXIgPSB0cnVlLFxyXG4gICAgICAgICAgICBkZWZhdWx0T3BhcXVlUmVuZGVyT3JkZXIgPSBSZW5kZXJpbmdHcm91cC5QYWludGVyU29ydENvbXBhcmUsXHJcbiAgICAgICAgICAgIGRlZmF1bHRBbHBoYVRlc3RSZW5kZXJPcmRlciA9IFJlbmRlcmluZ0dyb3VwLlBhaW50ZXJTb3J0Q29tcGFyZSxcclxuICAgICAgICAgICAgZGVmYXVsdFRyYW5zcGFyZW50UmVuZGVyT3JkZXIgPSBSZW5kZXJpbmdHcm91cC5kZWZhdWx0VHJhbnNwYXJlbnRTb3J0Q29tcGFyZSxcclxuICAgICAgICB9OiB7XHJcbiAgICAgICAgICAgIHBhcmVudENvbnRhaW5lcklkPzogc3RyaW5nIHwgbnVsbDtcclxuICAgICAgICAgICAgX2NvbnRhaW5lcklkPzogc3RyaW5nO1xyXG4gICAgICAgICAgICBkZWZhdWx0T3BhcXVlUmVuZGVyT3JkZXI/OiBSZW5kZXJPcmRlckZ1bmN0aW9uO1xyXG4gICAgICAgICAgICBkZWZhdWx0QWxwaGFUZXN0UmVuZGVyT3JkZXI/OiBSZW5kZXJPcmRlckZ1bmN0aW9uO1xyXG4gICAgICAgICAgICBkZWZhdWx0VHJhbnNwYXJlbnRSZW5kZXJPcmRlcj86IFJlbmRlck9yZGVyRnVuY3Rpb247XHJcbiAgICAgICAgICAgIGVuYWJsZU92ZXJsYXlSZW5kZXI/OiBib29sZWFuO1xyXG4gICAgICAgIH0gPSB7fVxyXG4gICAgKSB7XHJcbiAgICAgICAgLy8gUmVxdWlyZXMgYSBicm93c2VyIHRvIHdvcmsuICBPbmx5IGluaXQgaWYgd2UgYXJlIGluIGEgYnJvd3NlclxyXG4gICAgICAgIGlmICh0eXBlb2YgZG9jdW1lbnQgPT09IFwidW5kZWZpbmVkXCIpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuICAgICAgICB0aGlzLl9jb250YWluZXJJZCA9IF9jb250YWluZXJJZDtcclxuICAgICAgICB0aGlzLl9pbml0KHNjZW5lLCBwYXJlbnRDb250YWluZXJJZCwgZW5hYmxlT3ZlcmxheVJlbmRlciwgZGVmYXVsdE9wYXF1ZVJlbmRlck9yZGVyLCBkZWZhdWx0QWxwaGFUZXN0UmVuZGVyT3JkZXIsIGRlZmF1bHRUcmFuc3BhcmVudFJlbmRlck9yZGVyKTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIERpc3Bvc2Ugb2YgdGhlIEh0bWxNZXNoUmVuZGVyZXJcclxuICAgICAqL1xyXG4gICAgcHVibGljIGRpc3Bvc2UoKSB7XHJcbiAgICAgICAgaWYgKHRoaXMuX3JlbmRlck9ic2VydmVyKSB7XHJcbiAgICAgICAgICAgIHRoaXMuX3JlbmRlck9ic2VydmVyLnJlbW92ZSgpO1xyXG4gICAgICAgICAgICB0aGlzLl9yZW5kZXJPYnNlcnZlciA9IG51bGw7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICB0aGlzLl9vdmVybGF5RWxlbWVudHM/LmNvbnRhaW5lci5yZW1vdmUoKTtcclxuICAgICAgICB0aGlzLl9vdmVybGF5RWxlbWVudHMgPSBudWxsO1xyXG5cclxuICAgICAgICB0aGlzLl9pblNjZW5lRWxlbWVudHM/LmNvbnRhaW5lci5yZW1vdmUoKTtcclxuICAgICAgICB0aGlzLl9pblNjZW5lRWxlbWVudHMgPSBudWxsO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfaW5pdChcclxuICAgICAgICBzY2VuZTogU2NlbmUsXHJcbiAgICAgICAgcGFyZW50Q29udGFpbmVySWQ6IHN0cmluZyB8IG51bGwsXHJcbiAgICAgICAgZW5hYmxlT3ZlcmxheVJlbmRlcjogYm9vbGVhbixcclxuICAgICAgICBkZWZhdWx0T3BhcXVlUmVuZGVyT3JkZXI6IFJlbmRlck9yZGVyRnVuY3Rpb24sXHJcbiAgICAgICAgZGVmYXVsdEFscGhhVGVzdFJlbmRlck9yZGVyOiBSZW5kZXJPcmRlckZ1bmN0aW9uLFxyXG4gICAgICAgIGRlZmF1bHRUcmFuc3BhcmVudFJlbmRlck9yZGVyOiBSZW5kZXJPcmRlckZ1bmN0aW9uXHJcbiAgICApOiB2b2lkIHtcclxuICAgICAgICAvLyBSZXF1aXJlcyBhIGJyb3dzZXIgdG8gd29yay4gIE9ubHkgaW5pdCBpZiB3ZSBhcmUgaW4gYSBicm93c2VyXHJcbiAgICAgICAgaWYgKHR5cGVvZiBkb2N1bWVudCA9PT0gXCJ1bmRlZmluZWRcIikge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBDcmVhdGUgdGhlIERPTSBjb250YWluZXJzXHJcbiAgICAgICAgbGV0IHBhcmVudENvbnRhaW5lciA9IHBhcmVudENvbnRhaW5lcklkID8gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQocGFyZW50Q29udGFpbmVySWQpIDogZG9jdW1lbnQuYm9keTtcclxuXHJcbiAgICAgICAgaWYgKCFwYXJlbnRDb250YWluZXIpIHtcclxuICAgICAgICAgICAgcGFyZW50Q29udGFpbmVyID0gZG9jdW1lbnQuYm9keTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIGlmIHRoZSBjb250YWluZXIgYWxyZWFkeSBleGlzdHMsIHRoZW4gcmVtb3ZlIGl0XHJcbiAgICAgICAgY29uc3QgaW5TY2VuZUNvbnRhaW5lcklkID0gYCR7dGhpcy5fY29udGFpbmVySWR9X2luX3NjZW5lYDtcclxuICAgICAgICB0aGlzLl9pblNjZW5lRWxlbWVudHMgPSB0aGlzLl9jcmVhdGVSZW5kZXJMYXllckVsZW1lbnRzKGluU2NlbmVDb250YWluZXJJZCk7XHJcblxyXG4gICAgICAgIHBhcmVudENvbnRhaW5lci5pbnNlcnRCZWZvcmUodGhpcy5faW5TY2VuZUVsZW1lbnRzLmNvbnRhaW5lciwgcGFyZW50Q29udGFpbmVyLmZpcnN0Q2hpbGQpO1xyXG5cclxuICAgICAgICBpZiAoZW5hYmxlT3ZlcmxheVJlbmRlcikge1xyXG4gICAgICAgICAgICBjb25zdCBvdmVybGF5Q29udGFpbmVySWQgPSBgJHt0aGlzLl9jb250YWluZXJJZH1fb3ZlcmxheWA7XHJcbiAgICAgICAgICAgIHRoaXMuX292ZXJsYXlFbGVtZW50cyA9IHRoaXMuX2NyZWF0ZVJlbmRlckxheWVyRWxlbWVudHMob3ZlcmxheUNvbnRhaW5lcklkKTtcclxuICAgICAgICAgICAgY29uc3QgekluZGV4ID0gKyhzY2VuZS5nZXRFbmdpbmUoKS5nZXRSZW5kZXJpbmdDYW52YXMoKSEuc3R5bGUuekluZGV4ID8/IFwiMFwiKSArIDE7XHJcbiAgICAgICAgICAgIHRoaXMuX292ZXJsYXlFbGVtZW50cy5jb250YWluZXIuc3R5bGUuekluZGV4ID0gYCR7ekluZGV4fWA7XHJcbiAgICAgICAgICAgIHRoaXMuX292ZXJsYXlFbGVtZW50cy5jb250YWluZXIuc3R5bGUucG9pbnRlckV2ZW50cyA9IFwibm9uZVwiO1xyXG4gICAgICAgICAgICBwYXJlbnRDb250YWluZXIuaW5zZXJ0QmVmb3JlKHRoaXMuX292ZXJsYXlFbGVtZW50cy5jb250YWluZXIsIHBhcmVudENvbnRhaW5lci5maXJzdENoaWxkKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgdGhpcy5fZW5naW5lID0gc2NlbmUuZ2V0RW5naW5lKCk7XHJcbiAgICAgICAgY29uc3QgY2xpZW50UmVjdCA9IHRoaXMuX2VuZ2luZS5nZXRSZW5kZXJpbmdDYW52YXNDbGllbnRSZWN0KCk7XHJcbiAgICAgICAgaWYgKCFjbGllbnRSZWN0KSB7XHJcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcIkZhaWxlZCB0byBnZXQgY2xpZW50IHJlY3QgZm9yIHJlbmRlcmluZyBjYW52YXNcIik7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBTZXQgdGhlIHNpemUgYW5kIHJlc2l6ZSBiZWhhdmlvclxyXG4gICAgICAgIHRoaXMuX3NldFNpemUoY2xpZW50UmVjdC53aWR0aCwgY2xpZW50UmVjdC5oZWlnaHQpO1xyXG5cclxuICAgICAgICB0aGlzLl9lbmdpbmUub25SZXNpemVPYnNlcnZhYmxlLmFkZCgoKSA9PiB7XHJcbiAgICAgICAgICAgIGNvbnN0IGNsaWVudFJlY3QgPSB0aGlzLl9lbmdpbmUuZ2V0UmVuZGVyaW5nQ2FudmFzQ2xpZW50UmVjdCgpO1xyXG4gICAgICAgICAgICBpZiAoY2xpZW50UmVjdCkge1xyXG4gICAgICAgICAgICAgICAgdGhpcy5fc2V0U2l6ZShjbGllbnRSZWN0LndpZHRoLCBjbGllbnRSZWN0LmhlaWdodCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9KTtcclxuXHJcbiAgICAgICAgbGV0IHByb2plY3Rpb25PYnM6IE9ic2VydmVyPENhbWVyYT47XHJcbiAgICAgICAgbGV0IG1hdHJpeE9iczogT2JzZXJ2ZXI8Q2FtZXJhPjtcclxuXHJcbiAgICAgICAgY29uc3Qgb2JzZXJ2ZUNhbWVyYSA9ICgpID0+IHtcclxuICAgICAgICAgICAgY29uc3QgY2FtZXJhID0gc2NlbmUuYWN0aXZlQ2FtZXJhO1xyXG4gICAgICAgICAgICBpZiAoY2FtZXJhKSB7XHJcbiAgICAgICAgICAgICAgICBwcm9qZWN0aW9uT2JzID0gY2FtZXJhLm9uUHJvamVjdGlvbk1hdHJpeENoYW5nZWRPYnNlcnZhYmxlLmFkZCgoKSA9PiB7XHJcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5fb25DYW1lcmFNYXRyaXhDaGFuZ2VkKGNhbWVyYSk7XHJcbiAgICAgICAgICAgICAgICB9KTtcclxuICAgICAgICAgICAgICAgIG1hdHJpeE9icyA9IGNhbWVyYS5vblZpZXdNYXRyaXhDaGFuZ2VkT2JzZXJ2YWJsZS5hZGQoKCkgPT4ge1xyXG4gICAgICAgICAgICAgICAgICAgIHRoaXMuX29uQ2FtZXJhTWF0cml4Q2hhbmdlZChjYW1lcmEpO1xyXG4gICAgICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9O1xyXG5cclxuICAgICAgICBvYnNlcnZlQ2FtZXJhKCk7XHJcblxyXG4gICAgICAgIHNjZW5lLm9uQWN0aXZlQ2FtZXJhQ2hhbmdlZC5hZGQoKCkgPT4ge1xyXG4gICAgICAgICAgICBpZiAocHJvamVjdGlvbk9icykge1xyXG4gICAgICAgICAgICAgICAgc2NlbmUuYWN0aXZlQ2FtZXJhPy5vblByb2plY3Rpb25NYXRyaXhDaGFuZ2VkT2JzZXJ2YWJsZS5yZW1vdmUocHJvamVjdGlvbk9icyk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgaWYgKG1hdHJpeE9icykge1xyXG4gICAgICAgICAgICAgICAgc2NlbmUuYWN0aXZlQ2FtZXJhPy5vblZpZXdNYXRyaXhDaGFuZ2VkT2JzZXJ2YWJsZS5yZW1vdmUobWF0cml4T2JzKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBvYnNlcnZlQ2FtZXJhKCk7XHJcbiAgICAgICAgfSk7XHJcblxyXG4gICAgICAgIC8vIFdlIG5lZWQgdG8gbWFrZSBzdXJlIHRoYXQgSHRtbE1lc2hlcyBhcmUgcmVuZGVyZWQgYmVmb3JlIGFsbCBvdGhlciBtZXNoZXNcclxuICAgICAgICAvLyBzbyB0aGF0IHRoZXkgZG9uJ3QgYXBwZWFyIGluIGZyb250IG9mIG1lc2hlcyB0aGF0IGFyZSBhY3R1YWxseSBpbiBmcm9udCBvZiB0aGVtXHJcbiAgICAgICAgLy8gVXBkYXRpbmcgdGhlIHJlbmRlciBvcmRlciBpc24ndCBpZGVhbCwgYnV0IGl0IGlzIHRoZSBvbmx5IHdheSB0byBhY2hlaXZlIHRoaXNcclxuICAgICAgICAvLyBUaGUgaW1wbGljYXRpb24gaXMgdGhhdCBhbiBhcHAgdXNpbmcgdGhlIEh0bWxNZXNoUmVuZGVyZWQgbXVzdCBzZXQgdGhlIHNjZW5lIHJlbmRlciBvcmRlclxyXG4gICAgICAgIC8vIHZpYSB0aGUgSHRtbE1lc2hSZW5kZXJlZCBjb25zdHJ1Y3RvclxyXG4gICAgICAgIGNvbnN0IG9wYXF1ZVJlbmRlck9yZGVyID0gcmVuZGVyT3JkZXJGdW5jKGRlZmF1bHRPcGFxdWVSZW5kZXJPcmRlcik7XHJcbiAgICAgICAgY29uc3QgYWxwaGFUZXN0UmVuZGVyT3JkZXIgPSByZW5kZXJPcmRlckZ1bmMoZGVmYXVsdEFscGhhVGVzdFJlbmRlck9yZGVyKTtcclxuICAgICAgICBjb25zdCB0cmFuc3BhcmVudFJlbmRlck9yZGVyID0gcmVuZGVyT3JkZXJGdW5jKGRlZmF1bHRUcmFuc3BhcmVudFJlbmRlck9yZGVyKTtcclxuICAgICAgICBzY2VuZS5zZXRSZW5kZXJpbmdPcmRlcigwLCBvcGFxdWVSZW5kZXJPcmRlciwgYWxwaGFUZXN0UmVuZGVyT3JkZXIsIHRyYW5zcGFyZW50UmVuZGVyT3JkZXIpO1xyXG5cclxuICAgICAgICB0aGlzLl9yZW5kZXJPYnNlcnZlciA9IHNjZW5lLm9uQmVmb3JlUmVuZGVyT2JzZXJ2YWJsZS5hZGQoKCkgPT4ge1xyXG4gICAgICAgICAgICB0aGlzLl9yZW5kZXIoc2NlbmUsIHNjZW5lLmFjdGl2ZUNhbWVyYSBhcyBDYW1lcmEpO1xyXG4gICAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIHByaXZhdGUgX2NyZWF0ZVJlbmRlckxheWVyRWxlbWVudHMoY29udGFpbmVySWQ6IHN0cmluZyk6IFJlbmRlckxheWVyRWxlbWVudHMge1xyXG4gICAgICAgIGNvbnN0IGV4aXN0aW5nQ29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoY29udGFpbmVySWQpO1xyXG4gICAgICAgIGlmIChleGlzdGluZ0NvbnRhaW5lcikge1xyXG4gICAgICAgICAgICBleGlzdGluZ0NvbnRhaW5lci5yZW1vdmUoKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgY29uc3QgY29udGFpbmVyID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImRpdlwiKTtcclxuICAgICAgICBjb250YWluZXIuaWQgPSBjb250YWluZXJJZDtcclxuICAgICAgICBjb250YWluZXIuc3R5bGUucG9zaXRpb24gPSBcImFic29sdXRlXCI7XHJcbiAgICAgICAgY29udGFpbmVyLnN0eWxlLndpZHRoID0gXCIxMDAlXCI7XHJcbiAgICAgICAgY29udGFpbmVyLnN0eWxlLmhlaWdodCA9IFwiMTAwJVwiO1xyXG4gICAgICAgIGNvbnRhaW5lci5zdHlsZS56SW5kZXggPSBcIi0xXCI7XHJcblxyXG4gICAgICAgIGNvbnN0IGRvbUVsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZGl2XCIpO1xyXG4gICAgICAgIGRvbUVsZW1lbnQuc3R5bGUub3ZlcmZsb3cgPSBcImhpZGRlblwiO1xyXG5cclxuICAgICAgICBjb25zdCBjYW1lcmFFbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImRpdlwiKTtcclxuXHJcbiAgICAgICAgY2FtZXJhRWxlbWVudC5zdHlsZS53ZWJraXRUcmFuc2Zvcm1TdHlsZSA9IFwicHJlc2VydmUtM2RcIjtcclxuICAgICAgICBjYW1lcmFFbGVtZW50LnN0eWxlLnRyYW5zZm9ybVN0eWxlID0gXCJwcmVzZXJ2ZS0zZFwiO1xyXG5cclxuICAgICAgICBjYW1lcmFFbGVtZW50LnN0eWxlLnBvaW50ZXJFdmVudHMgPSBcIm5vbmVcIjtcclxuXHJcbiAgICAgICAgZG9tRWxlbWVudC5hcHBlbmRDaGlsZChjYW1lcmFFbGVtZW50KTtcclxuICAgICAgICBjb250YWluZXIuYXBwZW5kQ2hpbGQoZG9tRWxlbWVudCk7XHJcbiAgICAgICAgcmV0dXJuIHtcclxuICAgICAgICAgICAgY29udGFpbmVyLFxyXG4gICAgICAgICAgICBkb21FbGVtZW50LFxyXG4gICAgICAgICAgICBjYW1lcmFFbGVtZW50LFxyXG4gICAgICAgIH07XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF9nZXRTaXplKCk6IHsgd2lkdGg6IG51bWJlcjsgaGVpZ2h0OiBudW1iZXIgfSB7XHJcbiAgICAgICAgcmV0dXJuIHtcclxuICAgICAgICAgICAgd2lkdGg6IHRoaXMuX3dpZHRoLFxyXG4gICAgICAgICAgICBoZWlnaHQ6IHRoaXMuX2hlaWdodCxcclxuICAgICAgICB9O1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBfc2V0U2l6ZSh3aWR0aDogbnVtYmVyLCBoZWlnaHQ6IG51bWJlcik6IHZvaWQge1xyXG4gICAgICAgIHRoaXMuX3dpZHRoID0gd2lkdGg7XHJcbiAgICAgICAgdGhpcy5faGVpZ2h0ID0gaGVpZ2h0O1xyXG4gICAgICAgIHRoaXMuX2hlaWdodEhhbGYgPSB0aGlzLl9oZWlnaHQgLyAyO1xyXG5cclxuICAgICAgICBpZiAoIXRoaXMuX2luU2NlbmVFbGVtZW50cyB8fCAhdGhpcy5fb3ZlcmxheUVsZW1lbnRzKSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGNvbnN0IGRvbUVsZW1lbnRzID0gW3RoaXMuX2luU2NlbmVFbGVtZW50cyEuZG9tRWxlbWVudCwgdGhpcy5fb3ZlcmxheUVsZW1lbnRzIS5kb21FbGVtZW50LCB0aGlzLl9pblNjZW5lRWxlbWVudHMhLmNhbWVyYUVsZW1lbnQsIHRoaXMuX292ZXJsYXlFbGVtZW50cyEuY2FtZXJhRWxlbWVudF07XHJcbiAgICAgICAgZG9tRWxlbWVudHMuZm9yRWFjaCgoZG9tKSA9PiB7XHJcbiAgICAgICAgICAgIGlmIChkb20pIHtcclxuICAgICAgICAgICAgICAgIGRvbS5zdHlsZS53aWR0aCA9IGAke3dpZHRofXB4YDtcclxuICAgICAgICAgICAgICAgIGRvbS5zdHlsZS5oZWlnaHQgPSBgJHtoZWlnaHR9cHhgO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfSk7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gcHJldHRpZXItaWdub3JlXHJcbiAgICBwcm90ZWN0ZWQgX2dldENhbWVyYUNTU01hdHJpeChtYXRyaXg6IE1hdHJpeCk6IHN0cmluZyB7XHJcbiAgICAgICAgY29uc3QgZWxlbWVudHMgPSBtYXRyaXgubTtcclxuICAgICAgICByZXR1cm4gYG1hdHJpeDNkKCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzBdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIC0gZWxlbWVudHNbMV0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMl0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbM10gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbNF0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggLSBlbGVtZW50c1s1XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1s2XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1s3XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1s4XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCAtIGVsZW1lbnRzWzldIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzEwXSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxMV0gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMTJdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIC0gZWxlbWVudHNbMTNdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzE0XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxNV0gKVxyXG4gICAgICAgIH0pYDtcclxuICAgIH1cclxuXHJcbiAgICAvLyBDb252ZXJ0IGEgQmFieWxvbiB3b3JsZCBtYXRyaXggdG8gYSBDU1MgbWF0cml4XHJcbiAgICAvLyBUaGlzIGFsc28gaGFuZGxlcyBjb252ZXJzaW9uIGZyb20gQkpTIGxlZnQgaGFuZGVkIGNvb3Jkc1xyXG4gICAgLy8gdG8gQ1NTIHJpZ2h0IGhhbmRlZCBjb29yZHNcclxuICAgIC8vIHByZXR0aWVyLWlnbm9yZVxyXG4gICAgcHJvdGVjdGVkIF9nZXRIdG1sQ29udGVudENTU01hdHJpeChtYXRyaXg6IE1hdHJpeCwgdXNlUmlnaHRIYW5kZWRTeXN0ZW06IGJvb2xlYW4pOiBzdHJpbmcge1xyXG4gICAgICAgIGNvbnN0IGVsZW1lbnRzID0gbWF0cml4Lm07XHJcbiAgICAgICAgLy8gSW4gYSByaWdodCBoYW5kZWQgY29vcmRpbmF0ZSBzeXN0ZW0sIHRoZSBlbGVtZW50cyAxMSB0byAxNCBoYXZlIHRvIGNoYW5nZSB0aGVpciBkaXJlY3Rpb25cclxuICAgICAgICBjb25zdCBkaXJlY3Rpb24gPSB1c2VSaWdodEhhbmRlZFN5c3RlbSA/IC0xIDogMTtcclxuICAgICAgICBjb25zdCBtYXRyaXgzZCA9IGBtYXRyaXgzZCgke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1swXSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxXSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1syXSAqIC1kaXJlY3Rpb24gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbM10gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggLSBlbGVtZW50c1s0XSApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCAtIGVsZW1lbnRzWzVdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzZdICAqIGRpcmVjdGlvbiApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCAtIGVsZW1lbnRzWzddIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzhdICogLWRpcmVjdGlvbiApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1s5XSAqIC1kaXJlY3Rpb24gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMTBdIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzExXSAqIGRpcmVjdGlvbiApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxMl0gKiBkaXJlY3Rpb24gKVxyXG4gICAgICAgIH0sJHtcclxuICAgICAgICAgICAgdGhpcy5fZXBzaWxvbiggZWxlbWVudHNbMTNdICogZGlyZWN0aW9uIClcclxuICAgICAgICB9LCR7XHJcbiAgICAgICAgICAgIHRoaXMuX2Vwc2lsb24oIGVsZW1lbnRzWzE0XSAqIGRpcmVjdGlvbiApXHJcbiAgICAgICAgfSwke1xyXG4gICAgICAgICAgICB0aGlzLl9lcHNpbG9uKCBlbGVtZW50c1sxNV0gKVxyXG4gICAgICAgIH0pYDtcclxuICAgICAgICByZXR1cm4gbWF0cml4M2Q7XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF9nZXRUcmFuc2Zvcm1hdGlvbk1hdHJpeChodG1sTWVzaDogSHRtbE1lc2gsIHVzZVJpZ2h0SGFuZGVkU3lzdGVtOiBib29sZWFuKTogTWF0cml4IHtcclxuICAgICAgICAvLyBHZXQgdGhlIGNhbWVyYSB3b3JsZCBtYXRyaXhcclxuICAgICAgICAvLyBNYWtlIHN1cmUgdGhlIGNhbWVyYSB3b3JsZCBtYXRyaXggaXMgdXAgdG8gZGF0ZVxyXG4gICAgICAgIGlmICghdGhpcy5fY2FtZXJhV29ybGRNYXRyaXgpIHtcclxuICAgICAgICAgICAgdGhpcy5fY2FtZXJhV29ybGRNYXRyaXggPSBodG1sTWVzaC5nZXRTY2VuZSgpLmFjdGl2ZUNhbWVyYT8uZ2V0V29ybGRNYXRyaXgoKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgaWYgKCF0aGlzLl9jYW1lcmFXb3JsZE1hdHJpeCkge1xyXG4gICAgICAgICAgICByZXR1cm4gTWF0cml4LklkZW50aXR5KCk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBjb25zdCBvYmplY3RXb3JsZE1hdHJpeCA9IGh0bWxNZXNoLmdldFdvcmxkTWF0cml4KCk7XHJcblxyXG4gICAgICAgIC8vIFNjYWxlIHRoZSBvYmplY3QgbWF0cml4IGJ5IHRoZSBiYXNlIHNjYWxlIGZhY3RvciBmb3IgdGhlIG1lc2hcclxuICAgICAgICAvLyB3aGljaCBpcyB0aGUgcmF0aW8gb2YgdGhlIG1lc2ggd2lkdGgvaGVpZ2h0IHRvIHRoZSByZW5kZXJlclxyXG4gICAgICAgIC8vIHdpZHRoL2hlaWdodCBkaXZpZGVkIGJ5IHRoZSBiYWJ5bG9uIHVuaXRzIHRvIHBpeGVscyByYXRpb1xyXG4gICAgICAgIGxldCB3aWR0aFNjYWxlRmFjdG9yID0gMTtcclxuICAgICAgICBsZXQgaGVpZ2h0U2NhbGVGYWN0b3IgPSAxO1xyXG4gICAgICAgIGlmIChodG1sTWVzaC5zb3VyY2VXaWR0aCAmJiBodG1sTWVzaC5zb3VyY2VIZWlnaHQpIHtcclxuICAgICAgICAgICAgd2lkdGhTY2FsZUZhY3RvciA9IGh0bWxNZXNoLndpZHRoISAvIChodG1sTWVzaC5zb3VyY2VXaWR0aCAvIGJhYnlsb25Vbml0c1RvUGl4ZWxzKTtcclxuICAgICAgICAgICAgaGVpZ2h0U2NhbGVGYWN0b3IgPSBodG1sTWVzaC5oZWlnaHQhIC8gKGh0bWxNZXNoLnNvdXJjZUhlaWdodCAvIGJhYnlsb25Vbml0c1RvUGl4ZWxzKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIEFwcGx5IHRoZSBzY2FsZSB0byB0aGUgb2JqZWN0J3Mgd29ybGQgbWF0cml4LiAgTm90ZSB3ZSBhcmVuJ3Qgc2NhbGluZ1xyXG4gICAgICAgIC8vIHRoZSBvYmplY3QsIGp1c3QgZ2V0dGluZyBhIG1hdHJpeCBhcyB0aG91Z2ggaXQgd2VyZSBzY2FsZWQsIHNvIHdlIGNhblxyXG4gICAgICAgIC8vIHNjYWxlIHRoZSBjb250ZW50XHJcbiAgICAgICAgY29uc3Qgc2NhbGVUcmFuc2Zvcm0gPSB0aGlzLl90ZW1wLnNjYWxlVHJhbnNmb3JtO1xyXG4gICAgICAgIGNvbnN0IHJvdGF0aW9uVHJhbnNmb3JtID0gdGhpcy5fdGVtcC5yb3RhdGlvblRyYW5zZm9ybTtcclxuICAgICAgICBjb25zdCBwb3NpdGlvblRyYW5zZm9ybSA9IHRoaXMuX3RlbXAucG9zaXRpb25UcmFuc2Zvcm07XHJcbiAgICAgICAgY29uc3Qgc2NhbGVkQW5kVHJhbnNsYXRlZE9iamVjdE1hdHJpeCA9IHRoaXMuX3RlbXAub2JqZWN0TWF0cml4O1xyXG5cclxuICAgICAgICBvYmplY3RXb3JsZE1hdHJpeC5kZWNvbXBvc2Uoc2NhbGVUcmFuc2Zvcm0sIHJvdGF0aW9uVHJhbnNmb3JtLCBwb3NpdGlvblRyYW5zZm9ybSk7XHJcbiAgICAgICAgc2NhbGVUcmFuc2Zvcm0ueCAqPSB3aWR0aFNjYWxlRmFjdG9yO1xyXG4gICAgICAgIHNjYWxlVHJhbnNmb3JtLnkgKj0gaGVpZ2h0U2NhbGVGYWN0b3I7XHJcblxyXG4gICAgICAgIE1hdHJpeC5Db21wb3NlVG9SZWYoc2NhbGVUcmFuc2Zvcm0sIHJvdGF0aW9uVHJhbnNmb3JtLCBwb3NpdGlvblRyYW5zZm9ybSwgc2NhbGVkQW5kVHJhbnNsYXRlZE9iamVjdE1hdHJpeCk7XHJcblxyXG4gICAgICAgIC8vIEFkanVzdCBkaXJlY3Rpb24gb2YgMTIgYW5kIDEzIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXggYmFzZWQgb24gdGhlIGhhbmRlZG5lc3Mgb2YgdGhlIHN5c3RlbVxyXG4gICAgICAgIGNvbnN0IGRpcmVjdGlvbiA9IHVzZVJpZ2h0SGFuZGVkU3lzdGVtID8gLTEgOiAxO1xyXG4gICAgICAgIC8vIEFkanVzdCB0cmFuc2xhdGlvbiB2YWx1ZXMgdG8gYmUgZnJvbSBjYW1lcmEgdnMgd29ybGQgb3JpZ2luXHJcbiAgICAgICAgLy8gTm90ZSB0aGF0IHdlIGFyZSBhbHNvIGFkanVzdGluZyB0aGVzZSB2YWx1ZXMgdG8gYmUgcGl4ZWxzIHZzIEJhYnlsb24gdW5pdHNcclxuICAgICAgICBjb25zdCBwb3NpdGlvbiA9IGh0bWxNZXNoLmdldEFic29sdXRlUG9zaXRpb24oKTtcclxuICAgICAgICBzY2FsZWRBbmRUcmFuc2xhdGVkT2JqZWN0TWF0cml4LnNldFJvd0Zyb21GbG9hdHMoXHJcbiAgICAgICAgICAgIDMsXHJcbiAgICAgICAgICAgICgtdGhpcy5fY2FtZXJhV29ybGRNYXRyaXgubVsxMl0gKyBwb3NpdGlvbi54KSAqIGJhYnlsb25Vbml0c1RvUGl4ZWxzICogZGlyZWN0aW9uLFxyXG4gICAgICAgICAgICAoLXRoaXMuX2NhbWVyYVdvcmxkTWF0cml4Lm1bMTNdICsgcG9zaXRpb24ueSkgKiBiYWJ5bG9uVW5pdHNUb1BpeGVscyAqIGRpcmVjdGlvbixcclxuICAgICAgICAgICAgKHRoaXMuX2NhbWVyYVdvcmxkTWF0cml4Lm1bMTRdIC0gcG9zaXRpb24ueikgKiBiYWJ5bG9uVW5pdHNUb1BpeGVscyxcclxuICAgICAgICAgICAgdGhpcy5fY2FtZXJhV29ybGRNYXRyaXgubVsxNV0gKiAwLjAwMDAxICogYmFieWxvblVuaXRzVG9QaXhlbHNcclxuICAgICAgICApO1xyXG5cclxuICAgICAgICAvLyBBZGp1c3Qgb3RoZXIgdmFsdWVzIHRvIGJlIHBpeGVscyB2cyBCYWJ5bG9uIHVuaXRzXHJcbiAgICAgICAgc2NhbGVkQW5kVHJhbnNsYXRlZE9iamVjdE1hdHJpeC5tdWx0aXBseUF0SW5kZXgoMywgYmFieWxvblVuaXRzVG9QaXhlbHMpO1xyXG4gICAgICAgIHNjYWxlZEFuZFRyYW5zbGF0ZWRPYmplY3RNYXRyaXgubXVsdGlwbHlBdEluZGV4KDcsIGJhYnlsb25Vbml0c1RvUGl4ZWxzKTtcclxuICAgICAgICBzY2FsZWRBbmRUcmFuc2xhdGVkT2JqZWN0TWF0cml4Lm11bHRpcGx5QXRJbmRleCgxMSwgYmFieWxvblVuaXRzVG9QaXhlbHMpO1xyXG5cclxuICAgICAgICByZXR1cm4gc2NhbGVkQW5kVHJhbnNsYXRlZE9iamVjdE1hdHJpeDtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX3JlbmRlckh0bWxNZXNoKGh0bWxNZXNoOiBIdG1sTWVzaCwgdXNlUmlnaHRIYW5kZWRTeXN0ZW06IGJvb2xlYW4pIHtcclxuICAgICAgICBpZiAoIWh0bWxNZXNoLmVsZW1lbnQgfHwgIWh0bWxNZXNoLmVsZW1lbnQuZmlyc3RFbGVtZW50Q2hpbGQpIHtcclxuICAgICAgICAgICAgLy8gbm90aGluZyB0byByZW5kZXIsIHNvIGJhaWxcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gV2UgbmVlZCB0byBlbnN1cmUgaHRtbCBtZXNoIGRhdGEgaXMgaW5pdGlhbGl6ZWQgYmVmb3JlXHJcbiAgICAgICAgLy8gY29tcHV0aW5nIHRoZSBiYXNlIHNjYWxlIGZhY3RvclxyXG4gICAgICAgIGxldCBodG1sTWVzaERhdGEgPSB0aGlzLl9jYWNoZS5odG1sTWVzaERhdGEuZ2V0KGh0bWxNZXNoKTtcclxuICAgICAgICBpZiAoIWh0bWxNZXNoRGF0YSkge1xyXG4gICAgICAgICAgICBodG1sTWVzaERhdGEgPSB7IHN0eWxlOiBcIlwiIH07XHJcbiAgICAgICAgICAgIHRoaXMuX2NhY2hlLmh0bWxNZXNoRGF0YS5zZXQoaHRtbE1lc2gsIGh0bWxNZXNoRGF0YSk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBjb25zdCBjYW1lcmFFbGVtZW50ID0gaHRtbE1lc2guX2lzQ2FudmFzT3ZlcmxheSA/IHRoaXMuX292ZXJsYXlFbGVtZW50cz8uY2FtZXJhRWxlbWVudCA6IHRoaXMuX2luU2NlbmVFbGVtZW50cz8uY2FtZXJhRWxlbWVudDtcclxuXHJcbiAgICAgICAgaWYgKGh0bWxNZXNoLmVsZW1lbnQucGFyZW50Tm9kZSAhPT0gY2FtZXJhRWxlbWVudCkge1xyXG4gICAgICAgICAgICBjYW1lcmFFbGVtZW50IS5hcHBlbmRDaGlsZChodG1sTWVzaC5lbGVtZW50KTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIElmIHRoZSBodG1sTWVzaCBjb250ZW50IGhhcyBjaGFuZ2VkLCB1cGRhdGUgdGhlIGJhc2Ugc2NhbGUgZmFjdG9yXHJcbiAgICAgICAgaWYgKGh0bWxNZXNoLnJlcXVpcmVzVXBkYXRlKSB7XHJcbiAgICAgICAgICAgIHRoaXMuX3VwZGF0ZUJhc2VTY2FsZUZhY3RvcihodG1sTWVzaCk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBHZXQgdGhlIHRyYW5zZm9ybWF0aW9uIG1hdHJpeCBmb3IgdGhlIGh0bWwgbWVzaFxyXG4gICAgICAgIGNvbnN0IHNjYWxlZEFuZFRyYW5zbGF0ZWRPYmplY3RNYXRyaXggPSB0aGlzLl9nZXRUcmFuc2Zvcm1hdGlvbk1hdHJpeChodG1sTWVzaCwgdXNlUmlnaHRIYW5kZWRTeXN0ZW0pO1xyXG5cclxuICAgICAgICBsZXQgc3R5bGUgPSBgdHJhbnNsYXRlKC01MCUsIC01MCUpICR7dGhpcy5fZ2V0SHRtbENvbnRlbnRDU1NNYXRyaXgoc2NhbGVkQW5kVHJhbnNsYXRlZE9iamVjdE1hdHJpeCwgdXNlUmlnaHRIYW5kZWRTeXN0ZW0pfWA7XHJcbiAgICAgICAgLy8gSW4gYSByaWdodCBoYW5kZWQgc3lzdGVtLCBzY3JlZW5zIGFyZSBvbiB0aGUgd3Jvbmcgc2lkZSBvZiB0aGUgbWVzaCwgc28gd2UgaGF2ZSB0byByb3RhdGUgYnkgTWF0aC5QSSB3aGljaCByZXN1bHRzIGluIHRoZSBtYXRyaXgzZCBzZWVuIGJlbG93XHJcbiAgICAgICAgc3R5bGUgKz0gYCR7dXNlUmlnaHRIYW5kZWRTeXN0ZW0gPyBcIm1hdHJpeDNkKC0xLCAwLCAwLCAwLCAwLCAxLCAwLCAwLCAwLCAwLCAtMSwgMCwgMCwgMCwgMCwgMSlcIiA6IFwiXCJ9YDtcclxuXHJcbiAgICAgICAgaWYgKGh0bWxNZXNoRGF0YS5zdHlsZSAhPT0gc3R5bGUpIHtcclxuICAgICAgICAgICAgaHRtbE1lc2guZWxlbWVudC5zdHlsZS53ZWJraXRUcmFuc2Zvcm0gPSBzdHlsZTtcclxuICAgICAgICAgICAgaHRtbE1lc2guZWxlbWVudC5zdHlsZS50cmFuc2Zvcm0gPSBzdHlsZTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGh0bWxNZXNoLl9tYXJrQXNVcGRhdGVkKCk7XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF9yZW5kZXIoc2NlbmU6IFNjZW5lLCBjYW1lcmE6IENhbWVyYSkge1xyXG4gICAgICAgIGxldCBuZWVkc1VwZGF0ZSA9IGZhbHNlO1xyXG5cclxuICAgICAgICBjb25zdCB1c2VSaWdodEhhbmRlZFN5c3RlbSA9IHNjZW5lLnVzZVJpZ2h0SGFuZGVkU3lzdGVtO1xyXG5cclxuICAgICAgICAvLyBVcGRhdGUgdGhlIGNvbnRhaW5lciBwb3NpdGlvbiBhbmQgc2l6ZSBpZiBuZWNlc3NhcnlcclxuICAgICAgICB0aGlzLl91cGRhdGVDb250YWluZXJQb3NpdGlvbklmTmVlZGVkKCk7XHJcblxyXG4gICAgICAgIC8vIENoZWNrIGZvciBhIGNhbWVyYSBjaGFuZ2VcclxuICAgICAgICBpZiAodGhpcy5fY2FtZXJhTWF0cml4VXBkYXRlZCkge1xyXG4gICAgICAgICAgICB0aGlzLl9jYW1lcmFNYXRyaXhVcGRhdGVkID0gZmFsc2U7XHJcbiAgICAgICAgICAgIG5lZWRzVXBkYXRlID0gdHJ1ZTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIElmIHRoZSBjYW1lcmEgcG9zaXRpb24gaGFzIGNoYW5nZWQsIHRoZW4gd2UgYWxzbyBuZWVkIHRvIHVwZGF0ZVxyXG4gICAgICAgIGlmIChcclxuICAgICAgICAgICAgY2FtZXJhLnBvc2l0aW9uLnggIT09IHRoaXMuX2NhY2hlLmNhbWVyYURhdGEucG9zaXRpb24ueCB8fFxyXG4gICAgICAgICAgICBjYW1lcmEucG9zaXRpb24ueSAhPT0gdGhpcy5fY2FjaGUuY2FtZXJhRGF0YS5wb3NpdGlvbi55IHx8XHJcbiAgICAgICAgICAgIGNhbWVyYS5wb3NpdGlvbi56ICE9PSB0aGlzLl9jYWNoZS5jYW1lcmFEYXRhLnBvc2l0aW9uLnpcclxuICAgICAgICApIHtcclxuICAgICAgICAgICAgdGhpcy5fY2FjaGUuY2FtZXJhRGF0YS5wb3NpdGlvbi5jb3B5RnJvbShjYW1lcmEucG9zaXRpb24pO1xyXG4gICAgICAgICAgICBuZWVkc1VwZGF0ZSA9IHRydWU7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBDaGVjayBmb3IgYSBkcHIgY2hhbmdlXHJcbiAgICAgICAgaWYgKHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvICE9PSB0aGlzLl9sYXN0RGV2aWNlUGl4ZWxSYXRpbykge1xyXG4gICAgICAgICAgICB0aGlzLl9sYXN0RGV2aWNlUGl4ZWxSYXRpbyA9IHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvO1xyXG4gICAgICAgICAgICBMb2dnZXIuTG9nKFwiSW4gcmVuZGVyIC0gZHByIGNoYW5nZWQ6IFwiLCB0aGlzLl9sYXN0RGV2aWNlUGl4ZWxSYXRpbyk7XHJcbiAgICAgICAgICAgIG5lZWRzVXBkYXRlID0gdHJ1ZTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIENoZWNrIGlmIGFueSBtZXNoZXMgbmVlZCB0byBiZSB1cGRhdGVkXHJcbiAgICAgICAgY29uc3QgbWVzaGVzTmVlZGluZ1VwZGF0ZSA9IHNjZW5lLm1lc2hlcy5maWx0ZXIoKG1lc2gpID0+IChtZXNoIGFzIGFueSlbXCJpc0h0bWxNZXNoXCJdICYmIChuZWVkc1VwZGF0ZSB8fCAobWVzaCBhcyBIdG1sTWVzaCkucmVxdWlyZXNVcGRhdGUpKTtcclxuICAgICAgICBuZWVkc1VwZGF0ZSA9IG5lZWRzVXBkYXRlIHx8IG1lc2hlc05lZWRpbmdVcGRhdGUubGVuZ3RoID4gMDtcclxuXHJcbiAgICAgICAgaWYgKCFuZWVkc1VwZGF0ZSkge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBHZXQgYSBwcm9qZWN0aW9uIG1hdHJpeCBmb3IgdGhlIGNhbWVyYVxyXG4gICAgICAgIGNvbnN0IHByb2plY3Rpb25NYXRyaXggPSBjYW1lcmEuZ2V0UHJvamVjdGlvbk1hdHJpeCgpO1xyXG4gICAgICAgIGNvbnN0IGZvdiA9IHByb2plY3Rpb25NYXRyaXgubVs1XSAqIHRoaXMuX2hlaWdodEhhbGY7XHJcblxyXG4gICAgICAgIGlmICh0aGlzLl9jYWNoZS5jYW1lcmFEYXRhLmZvdiAhPT0gZm92KSB7XHJcbiAgICAgICAgICAgIGlmIChjYW1lcmEubW9kZSA9PSBDYW1lcmEuUEVSU1BFQ1RJVkVfQ0FNRVJBKSB7XHJcbiAgICAgICAgICAgICAgICBbdGhpcy5fb3ZlcmxheUVsZW1lbnRzPy5kb21FbGVtZW50LCB0aGlzLl9pblNjZW5lRWxlbWVudHM/LmRvbUVsZW1lbnRdLmZvckVhY2goKGVsKSA9PiB7XHJcbiAgICAgICAgICAgICAgICAgICAgaWYgKGVsKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIGVsLnN0eWxlLndlYmtpdFBlcnNwZWN0aXZlID0gZm92ICsgXCJweFwiO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBlbC5zdHlsZS5wZXJzcGVjdGl2ZSA9IGZvdiArIFwicHhcIjtcclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICB9KTtcclxuICAgICAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgICAgIFt0aGlzLl9vdmVybGF5RWxlbWVudHM/LmRvbUVsZW1lbnQsIHRoaXMuX2luU2NlbmVFbGVtZW50cz8uZG9tRWxlbWVudF0uZm9yRWFjaCgoZWwpID0+IHtcclxuICAgICAgICAgICAgICAgICAgICBpZiAoZWwpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgZWwuc3R5bGUud2Via2l0UGVyc3BlY3RpdmUgPSBcIlwiO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBlbC5zdHlsZS5wZXJzcGVjdGl2ZSA9IFwiXCI7XHJcbiAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgdGhpcy5fY2FjaGUuY2FtZXJhRGF0YS5mb3YgPSBmb3Y7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBHZXQgdGhlIENTUyBtYXRyaXggZm9yIHRoZSBjYW1lcmEgKHdoaWNoIHdpbGwgaW5jbHVkZSBhbnkgY2FtZXJhIHJvdGF0aW9uKVxyXG4gICAgICAgIGlmIChjYW1lcmEucGFyZW50ID09PSBudWxsKSB7XHJcbiAgICAgICAgICAgIGNhbWVyYS5jb21wdXRlV29ybGRNYXRyaXgoKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGNvbnN0IGNhbWVyYU1hdHJpeFdvcmxkID0gdGhpcy5fdGVtcC5jYW1lcmFXb3JsZE1hdHJpeDtcclxuICAgICAgICBjYW1lcmFNYXRyaXhXb3JsZC5jb3B5RnJvbShjYW1lcmEuZ2V0V29ybGRNYXRyaXgoKSk7XHJcbiAgICAgICAgY29uc3QgY2FtZXJhUm90YXRpb25NYXRyaXggPSB0aGlzLl90ZW1wLmNhbWVyYVJvdGF0aW9uTWF0cml4O1xyXG4gICAgICAgIGNhbWVyYU1hdHJpeFdvcmxkLmdldFJvdGF0aW9uTWF0cml4KCkudHJhbnNwb3NlVG9SZWYoY2FtZXJhUm90YXRpb25NYXRyaXgpO1xyXG5cclxuICAgICAgICBjb25zdCBjYW1lcmFNYXRyaXhXb3JsZEFzQXJyYXkgPSB0aGlzLl90ZW1wLmNhbWVyYVdvcmxkTWF0cml4QXNBcnJheTtcclxuICAgICAgICBjYW1lcmFNYXRyaXhXb3JsZC5jb3B5VG9BcnJheShjYW1lcmFNYXRyaXhXb3JsZEFzQXJyYXkpO1xyXG5cclxuICAgICAgICAvLyBGb3IgYSBmZXcgdmFsdWVzLCB3ZSBoYXZlIHRvIGFkanVzdCB0aGUgZGlyZWN0aW9uIGJhc2VkIG9uIHRoZSBoYW5kZWRuZXNzIG9mIHRoZSBzeXN0ZW1cclxuICAgICAgICBjb25zdCBkaXJlY3Rpb24gPSB1c2VSaWdodEhhbmRlZFN5c3RlbSA/IDEgOiAtMTtcclxuXHJcbiAgICAgICAgY2FtZXJhTWF0cml4V29ybGRBc0FycmF5WzFdID0gY2FtZXJhUm90YXRpb25NYXRyaXgubVsxXTtcclxuICAgICAgICBjYW1lcmFNYXRyaXhXb3JsZEFzQXJyYXlbMl0gPSBjYW1lcmFSb3RhdGlvbk1hdHJpeC5tWzJdICogZGlyZWN0aW9uO1xyXG4gICAgICAgIGNhbWVyYU1hdHJpeFdvcmxkQXNBcnJheVs0XSA9IGNhbWVyYVJvdGF0aW9uTWF0cml4Lm1bNF0gKiBkaXJlY3Rpb247XHJcbiAgICAgICAgY2FtZXJhTWF0cml4V29ybGRBc0FycmF5WzZdID0gY2FtZXJhUm90YXRpb25NYXRyaXgubVs2XSAqIGRpcmVjdGlvbjtcclxuICAgICAgICBjYW1lcmFNYXRyaXhXb3JsZEFzQXJyYXlbOF0gPSBjYW1lcmFSb3RhdGlvbk1hdHJpeC5tWzhdICogZGlyZWN0aW9uO1xyXG4gICAgICAgIGNhbWVyYU1hdHJpeFdvcmxkQXNBcnJheVs5XSA9IGNhbWVyYVJvdGF0aW9uTWF0cml4Lm1bOV0gKiBkaXJlY3Rpb247XHJcblxyXG4gICAgICAgIE1hdHJpeC5Gcm9tQXJyYXlUb1JlZihjYW1lcmFNYXRyaXhXb3JsZEFzQXJyYXksIDAsIGNhbWVyYU1hdHJpeFdvcmxkKTtcclxuXHJcbiAgICAgICAgY29uc3QgY2FtZXJhQ1NTTWF0cml4ID0gdGhpcy5fZ2V0Q2FtZXJhQ1NTTWF0cml4KGNhbWVyYU1hdHJpeFdvcmxkKTtcclxuICAgICAgICBjb25zdCBzdHlsZSA9IGNhbWVyYUNTU01hdHJpeDtcclxuXHJcbiAgICAgICAgaWYgKHRoaXMuX2NhY2hlLmNhbWVyYURhdGEuc3R5bGUgIT09IHN0eWxlKSB7XHJcbiAgICAgICAgICAgIFt0aGlzLl9pblNjZW5lRWxlbWVudHM/LmNhbWVyYUVsZW1lbnQsIHRoaXMuX292ZXJsYXlFbGVtZW50cz8uY2FtZXJhRWxlbWVudF0uZm9yRWFjaCgoZWwpID0+IHtcclxuICAgICAgICAgICAgICAgIGlmIChlbCkge1xyXG4gICAgICAgICAgICAgICAgICAgIGVsLnN0eWxlLndlYmtpdFRyYW5zZm9ybSA9IHN0eWxlO1xyXG4gICAgICAgICAgICAgICAgICAgIGVsLnN0eWxlLnRyYW5zZm9ybSA9IHN0eWxlO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICB9KTtcclxuICAgICAgICAgICAgdGhpcy5fY2FjaGUuY2FtZXJhRGF0YS5zdHlsZSA9IHN0eWxlO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gX1JlbmRlciBvYmplY3RzIGlmIG5lY2Vzc2FyeVxyXG4gICAgICAgIG1lc2hlc05lZWRpbmdVcGRhdGUuZm9yRWFjaCgobWVzaCkgPT4ge1xyXG4gICAgICAgICAgICB0aGlzLl9yZW5kZXJIdG1sTWVzaChtZXNoIGFzIEh0bWxNZXNoLCB1c2VSaWdodEhhbmRlZFN5c3RlbSk7XHJcbiAgICAgICAgfSk7XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF91cGRhdGVCYXNlU2NhbGVGYWN0b3IoaHRtbE1lc2g6IEh0bWxNZXNoKSB7XHJcbiAgICAgICAgLy8gR2V0IHNjcmVlbiB3aWR0aCBhbmQgaGVpZ2h0XHJcbiAgICAgICAgbGV0IHNjcmVlbldpZHRoID0gdGhpcy5fd2lkdGg7XHJcbiAgICAgICAgbGV0IHNjcmVlbkhlaWdodCA9IHRoaXMuX2hlaWdodDtcclxuXHJcbiAgICAgICAgLy8gQ2FsY3VsYXRlIGFzcGVjdCByYXRpb3NcclxuICAgICAgICBjb25zdCBodG1sTWVzaEFzcGVjdFJhdGlvID0gKGh0bWxNZXNoLndpZHRoIHx8IDEpIC8gKGh0bWxNZXNoLmhlaWdodCB8fCAxKTtcclxuICAgICAgICBjb25zdCBzY3JlZW5Bc3BlY3RSYXRpbyA9IHNjcmVlbldpZHRoIC8gc2NyZWVuSGVpZ2h0O1xyXG5cclxuICAgICAgICAvLyBBZGp1c3Qgc2NyZWVuIGRpbWVuc2lvbnMgYmFzZWQgb24gYXNwZWN0IHJhdGlvc1xyXG4gICAgICAgIGlmIChodG1sTWVzaEFzcGVjdFJhdGlvID4gc2NyZWVuQXNwZWN0UmF0aW8pIHtcclxuICAgICAgICAgICAgLy8gSWYgdGhlIEhUTUwgbWVzaCBpcyB3aWRlciByZWxhdGl2ZSB0byBpdHMgaGVpZ2h0IHRoYW4gdGhlIHNjcmVlbiwgYWRqdXN0IHRoZSBzY3JlZW4gd2lkdGhcclxuICAgICAgICAgICAgc2NyZWVuV2lkdGggPSBzY3JlZW5IZWlnaHQgKiBodG1sTWVzaEFzcGVjdFJhdGlvO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIC8vIElmIHRoZSBIVE1MIG1lc2ggaXMgdGFsbGVyIHJlbGF0aXZlIHRvIGl0cyB3aWR0aCB0aGFuIHRoZSBzY3JlZW4sIGFkanVzdCB0aGUgc2NyZWVuIGhlaWdodFxyXG4gICAgICAgICAgICBzY3JlZW5IZWlnaHQgPSBzY3JlZW5XaWR0aCAvIGh0bWxNZXNoQXNwZWN0UmF0aW87XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyBTZXQgY29udGVudCB0byBmaWxsIHNjcmVlbiBzbyB3ZSBnZXQgbWF4IHJlc29sdXRpb24gd2hlbiBpdCBpcyBzaHJ1bmsgdG8gZml0IHRoZSBtZXNoXHJcbiAgICAgICAgaHRtbE1lc2guc2V0Q29udGVudFNpemVQeChzY3JlZW5XaWR0aCwgc2NyZWVuSGVpZ2h0KTtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgX3VwZGF0ZUNvbnRhaW5lclBvc2l0aW9uSWZOZWVkZWQoKSB7XHJcbiAgICAgICAgLy8gRGV0ZXJtaW5lIGlmIHRoZSBjYW52YXMgaGFzIG1vdmVkIG9uIHRoZSBzY3JlZW5cclxuICAgICAgICBjb25zdCBjYW52YXNSZWN0ID0gdGhpcy5fZW5naW5lLmdldFJlbmRlcmluZ0NhbnZhc0NsaWVudFJlY3QoKTtcclxuXHJcbiAgICAgICAgLy8gY2FudmFzIHJlY3QgbWF5IGJlIG51bGwgaWYgbGF5b3V0IG5vdCBjb21wbGV0ZVxyXG4gICAgICAgIGlmICghY2FudmFzUmVjdCkge1xyXG4gICAgICAgICAgICBMb2dnZXIuV2FybihfcG9zaXRpb25VcGRhdGVGYWlsTWVzc2FnZSk7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcbiAgICAgICAgY29uc3Qgc2Nyb2xsVG9wID0gd2luZG93LnNjcm9sbFk7XHJcbiAgICAgICAgY29uc3Qgc2Nyb2xsTGVmdCA9IHdpbmRvdy5zY3JvbGxYO1xyXG4gICAgICAgIGNvbnN0IGNhbnZhc0RvY3VtZW50VG9wID0gY2FudmFzUmVjdC50b3AgKyBzY3JvbGxUb3A7XHJcbiAgICAgICAgY29uc3QgY2FudmFzRG9jdW1lbnRMZWZ0ID0gY2FudmFzUmVjdC5sZWZ0ICsgc2Nyb2xsTGVmdDtcclxuXHJcbiAgICAgICAgaWYgKHRoaXMuX3ByZXZpb3VzQ2FudmFzRG9jdW1lbnRQb3NpdGlvbi50b3AgIT09IGNhbnZhc0RvY3VtZW50VG9wIHx8IHRoaXMuX3ByZXZpb3VzQ2FudmFzRG9jdW1lbnRQb3NpdGlvbi5sZWZ0ICE9PSBjYW52YXNEb2N1bWVudExlZnQpIHtcclxuICAgICAgICAgICAgdGhpcy5fcHJldmlvdXNDYW52YXNEb2N1bWVudFBvc2l0aW9uLnRvcCA9IGNhbnZhc0RvY3VtZW50VG9wO1xyXG4gICAgICAgICAgICB0aGlzLl9wcmV2aW91c0NhbnZhc0RvY3VtZW50UG9zaXRpb24ubGVmdCA9IGNhbnZhc0RvY3VtZW50TGVmdDtcclxuXHJcbiAgICAgICAgICAgIFt0aGlzLl9pblNjZW5lRWxlbWVudHM/LmNvbnRhaW5lciwgdGhpcy5fb3ZlcmxheUVsZW1lbnRzPy5jb250YWluZXJdLmZvckVhY2goKGNvbnRhaW5lcikgPT4ge1xyXG4gICAgICAgICAgICAgICAgaWYgKCFjb250YWluZXIpIHtcclxuICAgICAgICAgICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAvLyBzZXQgdGhlIHRvcCBhbmQgbGVmdCBvZiB0aGUgY3NzIGNvbnRhaW5lciB0byBtYXRjaCB0aGUgY2FudmFzXHJcbiAgICAgICAgICAgICAgICBjb25zdCBjb250YWluZXJQYXJlbnQgPSBjb250YWluZXIub2Zmc2V0UGFyZW50IGFzIEhUTUxFbGVtZW50O1xyXG4gICAgICAgICAgICAgICAgY29uc3QgcGFyZW50UmVjdCA9IGNvbnRhaW5lclBhcmVudC5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKTtcclxuICAgICAgICAgICAgICAgIGNvbnN0IHBhcmVudERvY3VtZW50VG9wID0gcGFyZW50UmVjdC50b3AgKyBzY3JvbGxUb3A7XHJcbiAgICAgICAgICAgICAgICBjb25zdCBwYXJlbnREb2N1bWVudExlZnQgPSBwYXJlbnRSZWN0LmxlZnQgKyBzY3JvbGxMZWZ0O1xyXG5cclxuICAgICAgICAgICAgICAgIGNvbnN0IGFuY2VzdG9yTWFyZ2luc0FuZFBhZGRpbmcgPSB0aGlzLl9nZXRBbmNlc3Rvck1hcmdpbnNBbmRQYWRkaW5nKGNvbnRhaW5lclBhcmVudCk7XHJcblxyXG4gICAgICAgICAgICAgICAgLy8gQWRkIHRoZSBib2R5IG1hcmdpblxyXG4gICAgICAgICAgICAgICAgY29uc3QgYm9keVN0eWxlID0gd2luZG93LmdldENvbXB1dGVkU3R5bGUoZG9jdW1lbnQuYm9keSk7XHJcbiAgICAgICAgICAgICAgICBjb25zdCBib2R5TWFyZ2luVG9wID0gcGFyc2VJbnQoYm9keVN0eWxlLm1hcmdpblRvcCwgMTApO1xyXG4gICAgICAgICAgICAgICAgY29uc3QgYm9keU1hcmdpbkxlZnQgPSBwYXJzZUludChib2R5U3R5bGUubWFyZ2luTGVmdCwgMTApO1xyXG5cclxuICAgICAgICAgICAgICAgIGNvbnRhaW5lci5zdHlsZS50b3AgPSBgJHtjYW52YXNEb2N1bWVudFRvcCAtIHBhcmVudERvY3VtZW50VG9wIC0gYW5jZXN0b3JNYXJnaW5zQW5kUGFkZGluZy5tYXJnaW5Ub3AgKyBhbmNlc3Rvck1hcmdpbnNBbmRQYWRkaW5nLnBhZGRpbmdUb3AgKyBib2R5TWFyZ2luVG9wfXB4YDtcclxuICAgICAgICAgICAgICAgIGNvbnRhaW5lci5zdHlsZS5sZWZ0ID0gYCR7XHJcbiAgICAgICAgICAgICAgICAgICAgY2FudmFzRG9jdW1lbnRMZWZ0IC0gcGFyZW50RG9jdW1lbnRMZWZ0IC0gYW5jZXN0b3JNYXJnaW5zQW5kUGFkZGluZy5tYXJnaW5MZWZ0ICsgYW5jZXN0b3JNYXJnaW5zQW5kUGFkZGluZy5wYWRkaW5nTGVmdCArIGJvZHlNYXJnaW5MZWZ0XHJcbiAgICAgICAgICAgICAgICB9cHhgO1xyXG4gICAgICAgICAgICB9KTtcclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIF9vbkNhbWVyYU1hdHJpeENoYW5nZWQgPSAoY2FtZXJhOiBDYW1lcmEpID0+IHtcclxuICAgICAgICB0aGlzLl9jYW1lcmFXb3JsZE1hdHJpeCA9IGNhbWVyYS5nZXRXb3JsZE1hdHJpeCgpO1xyXG4gICAgICAgIHRoaXMuX2NhbWVyYU1hdHJpeFVwZGF0ZWQgPSB0cnVlO1xyXG4gICAgfTtcclxuXHJcbiAgICBwcml2YXRlIF9lcHNpbG9uKHZhbHVlOiBudW1iZXIpIHtcclxuICAgICAgICByZXR1cm4gTWF0aC5hYnModmFsdWUpIDwgMWUtMTAgPyAwIDogdmFsdWU7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gR2V0IHRvdGFsIG1hcmdpbnMgYW5kIHBhZGRpbmcgZm9yIGFuIGVsZW1lbnQsIGV4Y2x1ZGluZyB0aGUgYm9keSBhbmQgZG9jdW1lbnQgbWFyZ2luc1xyXG4gICAgcHJpdmF0ZSBfZ2V0QW5jZXN0b3JNYXJnaW5zQW5kUGFkZGluZyhlbGVtZW50OiBIVE1MRWxlbWVudCkge1xyXG4gICAgICAgIGxldCBtYXJnaW5Ub3AgPSAwO1xyXG4gICAgICAgIGxldCBtYXJnaW5MZWZ0ID0gMDtcclxuICAgICAgICBsZXQgcGFkZGluZ1RvcCA9IDA7XHJcbiAgICAgICAgbGV0IHBhZGRpbmdMZWZ0ID0gMDtcclxuXHJcbiAgICAgICAgd2hpbGUgKGVsZW1lbnQgJiYgZWxlbWVudCAhPT0gZG9jdW1lbnQuYm9keSAmJiBlbGVtZW50ICE9PSBkb2N1bWVudC5kb2N1bWVudEVsZW1lbnQpIHtcclxuICAgICAgICAgICAgY29uc3Qgc3R5bGUgPSB3aW5kb3cuZ2V0Q29tcHV0ZWRTdHlsZShlbGVtZW50KTtcclxuICAgICAgICAgICAgbWFyZ2luVG9wICs9IHBhcnNlSW50KHN0eWxlLm1hcmdpblRvcCwgMTApO1xyXG4gICAgICAgICAgICBtYXJnaW5MZWZ0ICs9IHBhcnNlSW50KHN0eWxlLm1hcmdpbkxlZnQsIDEwKTtcclxuICAgICAgICAgICAgcGFkZGluZ1RvcCArPSBwYXJzZUludChzdHlsZS5wYWRkaW5nVG9wLCAxMCk7XHJcbiAgICAgICAgICAgIHBhZGRpbmdMZWZ0ICs9IHBhcnNlSW50KHN0eWxlLnBhZGRpbmdMZWZ0LCAxMCk7XHJcbiAgICAgICAgICAgIGVsZW1lbnQgPSBlbGVtZW50Lm9mZnNldFBhcmVudCBhcyBIVE1MRWxlbWVudDtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHJldHVybiB7IG1hcmdpblRvcCwgbWFyZ2luTGVmdCwgcGFkZGluZ1RvcCwgcGFkZGluZ0xlZnQgfTtcclxuICAgIH1cclxufVxyXG4iLCJpbXBvcnQgeyBIdG1sTWVzaFJlbmRlcmVyIH0gZnJvbSBcIi4vaHRtbE1lc2hSZW5kZXJlclwiO1xyXG5pbXBvcnQgeyBIdG1sTWVzaCB9IGZyb20gXCIuL2h0bWxNZXNoXCI7XHJcbmltcG9ydCB7IFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IgfSBmcm9tIFwiLi9wb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yXCI7XHJcbmltcG9ydCB7IEZpdFN0cmF0ZWd5IH0gZnJvbSBcIi4vZml0U3RyYXRlZ3lcIjtcclxuXHJcbi8vIEV4cG9ydCBwdWJsaWMgY2xhc3NlcyBhbmQgZnVuY3Rpb25zXHJcbmV4cG9ydCB7IEh0bWxNZXNoUmVuZGVyZXIsIEh0bWxNZXNoLCBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yLCBGaXRTdHJhdGVneSB9O1xyXG4iLCJpbXBvcnQgeyBUb29scyB9IGZyb20gXCJjb3JlL01pc2MvdG9vbHNcIjtcclxuXHJcbi8vIEEgY2FwdHVyZSBtYW5hZ2VtZW50IHN5c3RlbSB0byBlbnN1cmUgdGhhdCB0aGUgY29ycmVjdCBvYmplY3QgaGFzIHRoZSBwb2ludGVyXHJcbi8vIGV2ZW50cyBieSBlbGltaW5hdGluZyByYWNlIGNvbmRpdGlvbnMgdGhhdCBjYW4gY2F1c2UgdGhlIHBvaW50ZXIgZXZlbnRzIHRvIGJlXHJcbi8vIHJlbGVhc2VkIGJ5IGEgZGlmZmVyZW50IG9iamVjdCBhZnRlciB0aGV5IGFyZSBjYXB0dXJlZCBsZWF2aW5nIG5vIG9iamVjdFxyXG4vLyBhcyB0aGUgb3duZXIuICBJdCBkb2VzIHRoaXMgYnkgcXVldWVpbmcgcmVxdWVzdHMgYW5kIG9ubHkgYWxsb3dpbmdcclxuLy8gY2FwdHVyZSB3aGVuIHRoZSBjdXJyZW50IGNhcHR1cmUgb3duZXIgcmVsZWFzZXMgcG9pbnRlciBldmVudHMuXHJcblxyXG50eXBlIENhcHR1cmVSZWxlYXNlQ2FsbGJhY2sgPSAoKSA9PiB2b2lkO1xyXG5cclxudHlwZSBDYXB0dXJlUmVsZWFzZUNhbGxiYWNrcyA9IHtcclxuICAgIGNhcHR1cmU6IENhcHR1cmVSZWxlYXNlQ2FsbGJhY2s7XHJcbiAgICByZWxlYXNlOiBDYXB0dXJlUmVsZWFzZUNhbGxiYWNrO1xyXG59O1xyXG5cclxubGV0IGNhcHR1cmVSZXF1ZXN0UXVldWU6IHN0cmluZ1tdID0gW107XHJcblxyXG4vLyBLZXkgaXMgcmVxdWVzdCBpZCwgdmFsdWUgaXMgb2JqZWN0IHdpdGggY2FwdHVyZSBhbmQgcmVsZWFzZSBjYWxsYmFja3NcclxuY29uc3QgcGVuZGluZ1JlcXVlc3RDYWxsYmFja3M6IE1hcDxzdHJpbmcsIENhcHR1cmVSZWxlYXNlQ2FsbGJhY2tzPiA9IG5ldyBNYXAoKTtcclxuXHJcbi8vIEtlZXAgdHJhY2sgb2YgcmVsZWFzZSByZXF1ZXN0cyB3aXRoIG5vIG1hdGNoaW5nIGNhcHR1cmUgcmVxdWVzdFxyXG4vLyBpbiBjYXNlIHRoZSByZWxlYXNlIHJlcXVlc3QgYXJyaXZlZCBiZWZvcmUgdGhlIGNhcHR1cmUgdG8gYXZvaWRcclxuLy8gdGhlIGNhcHR1cmUgcmVxdWVzdCBuZXZlciBnZXR0aW5nIHJlbGVhc2VkLlxyXG5sZXQgdW5tYXRjaGVkUmVsZWFzZVJlcXVlc3RzOiBzdHJpbmdbXSA9IFtdO1xyXG5cclxubGV0IGN1cnJlbnRPd25lcjogc3RyaW5nIHwgbnVsbCA9IG51bGw7IC8vIENhbGxlZCBvbiBmaXJzdCBjYXB0dXJlIG9yIHJlbGVhc2UgcmVxdWVzdFxyXG5cclxuLyoqXHJcbiAqIEdldCB0aGUgaWQgb2YgdGhlIG9iamVjdCBjdXJyZW50bHkgY2FwdHVyaW5nIHBvaW50ZXIgZXZlbnRzXHJcbiAqIEByZXR1cm5zIFRoZSBpZCBvZiB0aGUgb2JqZWN0IGN1cnJlbnRseSBjYXB0dXJpbmcgcG9pbnRlciBldmVudHNcclxuICogb3IgbnVsbCBpZiBubyBvYmplY3QgaXMgY2FwdHVyaW5nIHBvaW50ZXIgZXZlbnRzXHJcbiAqL1xyXG5leHBvcnQgY29uc3QgZ2V0Q2FwdHVyaW5nSWQgPSAoKSA9PiB7XHJcbiAgICByZXR1cm4gY3VycmVudE93bmVyO1xyXG59O1xyXG5cclxuLyoqXHJcbiAqIFJlcXVlc3QgdGhhdCB0aGUgb2JqZWN0IHdpdGggdGhlIGdpdmVuIGlkIGNhcHR1cmUgcG9pbnRlciBldmVudHMuICBJZiB0aGVyZSBpcyBubyBjdXJyZW50XHJcbiAqIG93bmVyLCB0aGVuIHRoZSByZXF1ZXN0IGlzIGdyYW50ZWQgaW1tZWRpYXRlbHkuICBJZiB0aGVyZSBpcyBhIGN1cnJlbnQgb3duZXIsIHRoZW4gdGhlIHJlcXVlc3RcclxuICogaXMgcXVldWVkIHVudGlsIHRoZSBjdXJyZW50IG93bmVyIHJlbGVhc2VzIHBvaW50ZXIgZXZlbnRzLlxyXG4gKiBAcGFyYW0gcmVxdWVzdElkIEFuIGlkIHRvIGlkZW50aWZ5IHRoZSByZXF1ZXN0LiAgVGhpcyBpZCB3aWxsIGJlIHVzZWQgdG8gbWF0Y2ggdGhlIGNhcHR1cmVcclxuICogcmVxdWVzdCB3aXRoIHRoZSByZWxlYXNlIHJlcXVlc3QuXHJcbiAqIEBwYXJhbSBjYXB0dXJlQ2FsbGJhY2sgVGhlIGNhbGxiYWNrIHRvIGNhbGwgd2hlbiB0aGUgcmVxdWVzdCBpcyBncmFudGVkIGFuZCB0aGUgb2JqZWN0IGlzIGNhcHR1cmluZ1xyXG4gKiBAcGFyYW0gcmVsZWFzZUNhbGxiYWNrIFRoZSBjYWxsYmFjayB0byBjYWxsIHdoZW4gdGhlIG9iamVjdCBpcyBubyBsb25nZXIgY2FwdHVyaW5nIHBvaW50ZXIgZXZlbnRzXHJcbiAqL1xyXG5leHBvcnQgY29uc3QgcmVxdWVzdENhcHR1cmUgPSAocmVxdWVzdElkOiBzdHJpbmcsIGNhcHR1cmVDYWxsYmFjazogQ2FwdHVyZVJlbGVhc2VDYWxsYmFjaywgcmVsZWFzZUNhbGxiYWNrOiBDYXB0dXJlUmVsZWFzZUNhbGxiYWNrKSA9PiB7XHJcbiAgICBkZWJ1Z0xvZyhgSW4gcG9pbnRlckV2ZW50c0NhcHR1cmUucmVxdWVzdENhcHR1cmUgLSBQb2ludGVyIGV2ZW50cyBjYXB0dXJlIHJlcXVlc3RlZCBmb3IgJHtyZXF1ZXN0SWR9YCk7XHJcblxyXG4gICAgLy8gSWYgdGhlcmUgaXMgYSByZWxlYXNlIGZvciB0aGlzIHJlcXVlc3QsIHRoZW4gaWdub3JlIHRoZSByZXF1ZXN0XHJcbiAgICBpZiAocmVtb3ZlVW5tYXRjaGVkUmVxdWVzdChyZXF1ZXN0SWQpKSB7XHJcbiAgICAgICAgZGVidWdMb2coYEluIHBvaW50ZXJFdmVudHNDYXB0dXJlLnJlcXVlc3RDYXB0dXJlIC0gQ2FwdHVyZSByZXF1ZXN0IG1hdGNoZWQgcHJldmlvdXMgcmVsZWFzZSByZXF1ZXN0ICR7cmVxdWVzdElkfS4gIENhbmNlbGxpbmcgY2FwdHVyZSByZXF1ZXN0YCk7XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgfSBlbHNlIGlmIChyZXF1ZXN0SWQgIT09IGN1cnJlbnRPd25lcikge1xyXG4gICAgICAgIC8vIGlmIHRoZSByZXF1ZXN0IGlzIG5vdCBhbHJlYWR5IGluIHRoZSBxdWV1ZSwgYWRkIGl0IHRvIHRoZSBxdWV1ZVxyXG4gICAgICAgIGVucXVldWVDYXB0dXJlUmVxdWVzdChyZXF1ZXN0SWQsIGNhcHR1cmVDYWxsYmFjaywgcmVsZWFzZUNhbGxiYWNrKTtcclxuICAgIH1cclxuXHJcbiAgICBpZiAoIWN1cnJlbnRPd25lcikge1xyXG4gICAgICAgIC8vIElmIHRoZXJlIGlzIG5vIGN1cnJlbnQgb3duZXIsIGdvIGFoZWFkIGFuZCBncmFudCB0aGUgcmVxdWVzdFxyXG4gICAgICAgIHRyYW5zZmVyUG9pbnRlckV2ZW50c093bmVyc2hpcCgpO1xyXG4gICAgfVxyXG4gICAgLy8gSWYgdGhlIHJlcXVlc3QgaWQgaXMgdGhlIGN1cnJlbnQgb3duZXIsIGRvIG5vdGhpbmdcclxufTtcclxuXHJcbi8qKlxyXG4gKiBSZWxlYXNlIHBvaW50ZXIgZXZlbnRzIGZyb20gdGhlIG9iamVjdCB3aXRoIHRoZSBnaXZlbiBpZC4gIElmIHRoZSBvYmplY3QgaXMgdGhlIGN1cnJlbnQgb3duZXJcclxuICogdGhlbiBwb2ludGVyIGV2ZW50cyBhcmUgcmVsZWFzZWQgaW1tZWRpYXRlbHkuICBJZiB0aGUgb2JqZWN0IGlzIG5vdCB0aGUgY3VycmVudCBvd25lciwgdGhlbiB0aGVcclxuICogYXNzb2NpYXRlZCBjYXB0dXJlIHJlcXVlc3QgaXMgcmVtb3ZlZCBmcm9tIHRoZSBxdWV1ZS4gIElmIHRoZXJlIGlzIG5vIG1hdGNoaW5nIGNhcHR1cmUgcmVxdWVzdFxyXG4gKiBpbiB0aGUgcXVldWUsIHRoZW4gdGhlIHJlbGVhc2UgcmVxdWVzdCBpcyBhZGRlZCB0byBhIGxpc3Qgb2YgdW5tYXRjaGVkIHJlbGVhc2UgcmVxdWVzdHMgYW5kIHdpbGxcclxuICogbmVnYXRlIHRoZSBuZXh0IGNhcHR1cmUgcmVxdWVzdCB3aXRoIHRoZSBzYW1lIGlkLiAgVGhpcyBpcyB0byBndWFyZCBhZ2FpbnN0IHRoZSBwb3NzaWJpbGl0eSB0aGF0XHJcbiAqIHRoZSByZWxlYXNlIHJlcXVlc3QgYXJyaXZlZCBiZWZvcmUgdGhlIGNhcHR1cmUgcmVxdWVzdC5cclxuICogQHBhcmFtIHJlcXVlc3RJZCBUaGUgaWQgd2hpY2ggc2hvdWxkIG1hdGNoIHRoZSBpZCBvZiB0aGUgY2FwdHVyZSByZXF1ZXN0XHJcbiAqL1xyXG5leHBvcnQgY29uc3QgcmVxdWVzdFJlbGVhc2UgPSAocmVxdWVzdElkOiBzdHJpbmcgfCBudWxsKSA9PiB7XHJcbiAgICBkZWJ1Z0xvZyhgSW4gcG9pbnRlckV2ZW50c0NhcHR1cmUucmVxdWVzdFJlbGVhc2UgLSBQb2ludGVyIGV2ZW50cyByZWxlYXNlIHJlcXVlc3RlZCBmb3IgJHtyZXF1ZXN0SWR9YCk7XHJcblxyXG4gICAgLy8gaWYgdGhlIHJlcXVlc3RJZCBpcyB0aGUgY3VycmVudCBjYXB0dXJlIGhvbGRlciByZWxlYXNlIGl0XHJcbiAgICBpZiAoIXJlcXVlc3RJZCB8fCByZXF1ZXN0SWQgPT09IGN1cnJlbnRPd25lcikge1xyXG4gICAgICAgIHRyYW5zZmVyUG9pbnRlckV2ZW50c093bmVyc2hpcCgpO1xyXG4gICAgfSBlbHNlIGlmIChjYW5jZWxSZXF1ZXN0KHJlcXVlc3RJZCkpIHtcclxuICAgICAgICAvLyBpZiB0aGUgcmVxdWVzdCBpcyBpbiB0aGUgcXVldWUsIGJ1dCBub3QgdGhlIGN1cnJlbnQgY2FwdHVyZSBob2xkZXIsIHJlbW92ZSBpdCBhbmQgaXQncyBjYWxsYmFja3NcclxuICAgICAgICBwZW5kaW5nUmVxdWVzdENhbGxiYWNrcy5kZWxldGUocmVxdWVzdElkKTtcclxuICAgIH0gZWxzZSB7XHJcbiAgICAgICAgZGVidWdMb2coYEluIHBvaW50ZXJFdmVudHNDYXB0dXJlLnJlcXVlc3RSZWxlYXNlIC0gUmVjZWl2ZWQgcmVsZWFzZSByZXF1ZXN0ICR7cmVxdWVzdElkfSBidXQgbm8gbWF0Y2hpbmcgY2FwdHVyZSByZXF1ZXN0IHdhcyByZWNlaXZlZGApO1xyXG4gICAgICAgIC8vIHJlcXVlc3Qgd2FzIG5vdCBjdXJyZW50IGFuZCBub3QgaW4gcXVldWUsIGxpa2VseSBiZWNhdXNlIHdlIHJlY2VpdmVkIGEgcmVsZWFzZVxyXG4gICAgICAgIC8vIHJlcXVlc3QgYmVmb3JlIHRoZSBjYXB0dXJlLiAgQWRkIGl0IHRvIHRoZSB1bm1hdGNoZWQgbGlzdCB0byBndWFyZCBhZ2FpbnN0IHRoaXMgcG9zc2liaWxpdHlcclxuICAgICAgICBpZiAoIXVubWF0Y2hlZFJlbGVhc2VSZXF1ZXN0cy5pbmNsdWRlcyhyZXF1ZXN0SWQpKSB7XHJcbiAgICAgICAgICAgIHVubWF0Y2hlZFJlbGVhc2VSZXF1ZXN0cy5wdXNoKHJlcXVlc3RJZCk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG59O1xyXG5cclxuLyoqXHJcbiAqIFJlbGFzZSBwb2ludGVyIGV2ZW50cyBmcm9tIHRoZSBjdXJyZW50IG93bmVyXHJcbiAqL1xyXG5leHBvcnQgY29uc3QgcmVsZWFzZUN1cnJlbnQgPSAoKSA9PiB7XHJcbiAgICByZXF1ZXN0UmVsZWFzZShjdXJyZW50T3duZXIpO1xyXG59O1xyXG5cclxuY29uc3QgZW5xdWV1ZUNhcHR1cmVSZXF1ZXN0ID0gKHJlcXVlc3RJZDogc3RyaW5nLCBjYXB0dXJlOiBDYXB0dXJlUmVsZWFzZUNhbGxiYWNrLCByZWxlYXNlOiBDYXB0dXJlUmVsZWFzZUNhbGxiYWNrKSA9PiB7XHJcbiAgICBkZWJ1Z0xvZyhgSW4gcG9pbnRlckV2ZW50c0NhcHR1cmUuZW5xdWV1ZUNhcHR1cmVSZXF1ZXN0IC0gRW5xdWV1ZWluZyBjYXB0dXJlIHJlcXVlc3QgZm9yICAke3JlcXVlc3RJZH1gKTtcclxuICAgIGlmICghY2FwdHVyZVJlcXVlc3RRdWV1ZS5pbmNsdWRlcyhyZXF1ZXN0SWQpKSB7XHJcbiAgICAgICAgY2FwdHVyZVJlcXVlc3RRdWV1ZS5wdXNoKHJlcXVlc3RJZCk7XHJcbiAgICAgICAgcGVuZGluZ1JlcXVlc3RDYWxsYmFja3Muc2V0KHJlcXVlc3RJZCwgeyBjYXB0dXJlLCByZWxlYXNlIH0pO1xyXG4gICAgfVxyXG59O1xyXG5cclxuLy8gUmVtb3ZlcyB0aGUgcmVxdWVzdCBmcm9tIHRoZSBxdWV1ZSBpZiBpdCBleGlzdHMuICBSZXR1cm5zIHRydWVcclxuLy8gaWYgdGhlIHJlcXVlc3Qgd2FzIGZvdW5kIGFuZCByZW1vdmVkLCBvdGhlcndpc2UgZmFsc2VcclxuY29uc3QgY2FuY2VsUmVxdWVzdCA9IChyZXF1ZXN0SWQ6IHN0cmluZyB8IG51bGwpID0+IHtcclxuICAgIGxldCByZW1vdmVkID0gZmFsc2U7XHJcbiAgICBjYXB0dXJlUmVxdWVzdFF1ZXVlID0gY2FwdHVyZVJlcXVlc3RRdWV1ZS5maWx0ZXIoKGlkKSA9PiB7XHJcbiAgICAgICAgaWYgKGlkICE9PSByZXF1ZXN0SWQpIHtcclxuICAgICAgICAgICAgcmV0dXJuIHRydWU7XHJcbiAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgcmVtb3ZlZCA9IHRydWU7XHJcbiAgICAgICAgICAgIGRlYnVnTG9nKGBJbiBwb2ludGVyRXZlbnRzQ2FwdHVyZS5jYW5jZWxSZXF1ZXN0IC0gQ2FuY2VsaW5nIHBvaW50ZXIgZXZlbnRzIGNhcHR1cmUgcmVxdWVzdCAke3JlcXVlc3RJZH1gKTtcclxuICAgICAgICAgICAgcmV0dXJuIGZhbHNlO1xyXG4gICAgICAgIH1cclxuICAgIH0pO1xyXG4gICAgcmV0dXJuIHJlbW92ZWQ7XHJcbn07XHJcblxyXG5jb25zdCByZW1vdmVVbm1hdGNoZWRSZXF1ZXN0ID0gKHJlcXVlc3RJZDogc3RyaW5nKSA9PiB7XHJcbiAgICBsZXQgcmVtb3ZlZCA9IGZhbHNlO1xyXG4gICAgdW5tYXRjaGVkUmVsZWFzZVJlcXVlc3RzID0gdW5tYXRjaGVkUmVsZWFzZVJlcXVlc3RzLmZpbHRlcigoaWQpID0+IHtcclxuICAgICAgICBpZiAoaWQgIT09IHJlcXVlc3RJZCkge1xyXG4gICAgICAgICAgICByZXR1cm4gdHJ1ZTtcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICByZW1vdmVkID0gdHJ1ZTtcclxuICAgICAgICAgICAgcmV0dXJuIGZhbHNlO1xyXG4gICAgICAgIH1cclxuICAgIH0pO1xyXG4gICAgcmV0dXJuIHJlbW92ZWQ7XHJcbn07XHJcblxyXG5jb25zdCB0cmFuc2ZlclBvaW50ZXJFdmVudHNPd25lcnNoaXAgPSAoKSA9PiB7XHJcbiAgICBjb25zdCBuZXdPd25lcklkID0gbmV4dENhcHR1cmVSZXF1ZXN0KCk7XHJcbiAgICBkZWJ1Z0xvZyhgSW4gcG9pbnRlckV2ZW50c0NhcHR1cmUudHJhbnNmZXJQb2ludGVyRXZlbnRzT3duZXJzaGlwIC0gVHJhbnNmZXJycmluZyBwb2ludGVyIGV2ZW50cyBmcm9tICR7Y3VycmVudE93bmVyfSB0byAke25ld093bmVySWR9YCk7XHJcbiAgICAvLyBSZWxlYXNlIHRoZSBjdXJyZW50IG93bmVyXHJcbiAgICBkb1JlbGVhc2UoKTtcclxuICAgIGlmIChuZXdPd25lcklkKSB7XHJcbiAgICAgICAgZG9DYXB0dXJlKG5ld093bmVySWQpO1xyXG4gICAgfVxyXG59O1xyXG5cclxuY29uc3QgZG9SZWxlYXNlID0gKCkgPT4ge1xyXG4gICAgZGVidWdMb2coYEluIHBvaW50ZXJFdmVudHNDYXB0dXJlLmRvUmVsZWFzZSAtIFJlbGVhc2luZyBwb2ludGVyIGV2ZW50cyBmcm9tICR7Y3VycmVudE93bmVyfWApO1xyXG4gICAgaWYgKGN1cnJlbnRPd25lcikge1xyXG4gICAgICAgIC8vIGNhbGwgdGhlIHJlbGVhc2UgY2FsbGJhY2tcclxuICAgICAgICBwZW5kaW5nUmVxdWVzdENhbGxiYWNrcy5nZXQoY3VycmVudE93bmVyKT8ucmVsZWFzZSgpO1xyXG4gICAgICAgIC8vIEFuZCByZW1vdmUgdGhlIGNhbGxiYWNrc1xyXG4gICAgICAgIHBlbmRpbmdSZXF1ZXN0Q2FsbGJhY2tzLmRlbGV0ZShjdXJyZW50T3duZXIpO1xyXG4gICAgICAgIGN1cnJlbnRPd25lciA9IG51bGw7XHJcbiAgICB9XHJcbn07XHJcblxyXG5jb25zdCBkb0NhcHR1cmUgPSAobmV3T3duZXJJZDogc3RyaW5nKSA9PiB7XHJcbiAgICBpZiAobmV3T3duZXJJZCkge1xyXG4gICAgICAgIC8vIGNhbGwgdGhlIGNhcHR1cmUgY2FsbGJhY2tcclxuICAgICAgICBwZW5kaW5nUmVxdWVzdENhbGxiYWNrcy5nZXQobmV3T3duZXJJZCk/LmNhcHR1cmUoKTtcclxuICAgIH1cclxuICAgIGN1cnJlbnRPd25lciA9IG5ld093bmVySWQ7XHJcbiAgICBkZWJ1Z0xvZyhgSW4gcG9pbnRlckV2ZW50c0NhcHR1cmUuZG9DYXB0dXJlIC0gUG9pbnRlciBldmVudHMgbm93IGNhcHR1cmVkIGJ5ICR7bmV3T3duZXJJZH1gKTtcclxufTtcclxuXHJcbmNvbnN0IG5leHRDYXB0dXJlUmVxdWVzdCA9ICgpID0+IHtcclxuICAgIHJldHVybiBjYXB0dXJlUmVxdWVzdFF1ZXVlLmxlbmd0aCA+IDAgPyBjYXB0dXJlUmVxdWVzdFF1ZXVlLnNoaWZ0KCkgOiBudWxsO1xyXG59O1xyXG5cclxuLy8gI3JlZ2lvbiBEZWJ1Z2dpbmcgc3VwcG9ydFxyXG5kZWNsYXJlIGdsb2JhbCB7XHJcbiAgICBpbnRlcmZhY2UgV2luZG93IHtcclxuICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25hbWluZy1jb252ZW50aW9uXHJcbiAgICAgICAgXCJwb2ludGVyLWV2ZW50cy1jYXB0dXJlLWRlYnVnXCI6IGJvb2xlYW4gfCBudWxsO1xyXG4gICAgfVxyXG59XHJcblxyXG5jb25zdCBkZWJ1Z0xvZyA9IChtZXNzYWdlOiBzdHJpbmcpID0+IHtcclxuICAgIC8vIElmIHdlIGFyZSBydW5ubmluZyBpbiBhIHRlc3QgcnVubmVyIChpbiBub2RlLCBzbyB3aW5kb3cgaXMgbm90IGRlZmluZWQpXHJcbiAgICAvLyBvciBpZiB0aGUgZGVidWcgZmxhZyBpcyBzZXQsIHRoZW4gbG9nIHRoZSBtZXNzYWdlXHJcbiAgICBpZiAodHlwZW9mIHdpbmRvdyA9PT0gXCJ1bmRlZmluZWRcIiB8fCB3aW5kb3dbXCJwb2ludGVyLWV2ZW50cy1jYXB0dXJlLWRlYnVnXCJdKSB7XHJcbiAgICAgICAgVG9vbHMuTG9nKFxyXG4gICAgICAgICAgICBgJHtwZXJmb3JtYW5jZS5ub3coKX0gLSBnYW1lLnNjZW5lLnBvaW50ZXJFdmVudHMgLSAke21lc3NhZ2V9XFxuY3VycmVudE93bmVyOiAke2N1cnJlbnRPd25lcn1cXG5xdWV1ZTogJHtjYXB0dXJlUmVxdWVzdFF1ZXVlfVxcbnVubWF0Y2hlZDogJHt1bm1hdGNoZWRSZWxlYXNlUmVxdWVzdHN9YFxyXG4gICAgICAgICk7XHJcbiAgICB9XHJcbn07XHJcbi8vICNlbmRyZWdpb24gRGVidWdnaW5nIHN1cHBvcnRcclxuIiwiaW1wb3J0IHR5cGUgeyBBYnN0cmFjdE1lc2ggfSBmcm9tIFwiY29yZS9NZXNoZXMvYWJzdHJhY3RNZXNoXCI7XHJcbmltcG9ydCB0eXBlIHsgQmVoYXZpb3IgfSBmcm9tIFwiY29yZS9CZWhhdmlvcnMvYmVoYXZpb3JcIjtcclxuaW1wb3J0IHR5cGUgeyBTY2VuZSB9IGZyb20gXCJjb3JlL3NjZW5lXCI7XHJcbmltcG9ydCB7IExvZ2dlciB9IGZyb20gXCJjb3JlL01pc2MvbG9nZ2VyXCI7XHJcbmltcG9ydCB7IHJlcXVlc3RDYXB0dXJlLCByZXF1ZXN0UmVsZWFzZSwgcmVsZWFzZUN1cnJlbnQsIGdldENhcHR1cmluZ0lkIH0gZnJvbSBcIi4vcG9pbnRlckV2ZW50c0NhcHR1cmVcIjtcclxuXHJcbi8vIE1vZHVsZSBsZXZlbCB2YXJpYWJsZSBmb3IgaG9sZGluZyB0aGUgY3VycmVudCBzY2VuZVxyXG5sZXQgX3NjZW5lOiBTY2VuZSB8IG51bGwgPSBudWxsO1xyXG5cclxuLy8gTW9kdWxlIGxldmVsIHZhcmlhYmxlIHRvIGhvbGQgdGhlIGNvdW50IG9mIGJlaGF2aW9yIGluc3RhbmNlcyB0aGF0IGFyZSBjdXJyZW50bHkgY2FwdHVyaW5nIHBvaW50ZXIgZXZlbnRzXHJcbi8vIG9uIGVudHJ5LiAgVGhpcyBpcyB1c2VkIHRvIGRldGVybWluZSBpZiB3ZSBuZWVkIHRvIHN0YXJ0IG9yIHN0b3Agb2JzZXJ2aW5nIHBvaW50ZXIgbW92ZW1lbnQuXHJcbmxldCBjYXB0dXJlT25FbnRlckNvdW50ID0gMDtcclxuXHJcbi8vIE1hcCB1c2VkIHRvIHN0b3JlIGluc3RhbmNlIG9mIHRoZSBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yIGZvciBhIG1lc2hcclxuLy8gV2UgZG8gdGhpcyBiZWNhdXNlIHRoaXMgZ2V0cyBjaGVja2VkIG9uIHBvaW50ZXIgbW92ZSBhbmQgd2UgZG9uJ3Qgd2FudCB0b1xyXG4vLyB1c2UgZ2V0QmVoYXZpb3JCeU5hbWUoKSBiZWNhdXNlIHRoYXQgaXMgYSBsaW5lYXIgc2VhcmNoXHJcbmNvbnN0IG1lc2hUb0JlaGF2aW9yTWFwID0gbmV3IFdlYWtNYXA8QWJzdHJhY3RNZXNoLCBQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yPigpO1xyXG5cclxuY29uc3Qgc3RhcnRDYXB0dXJlT25FbnRlciA9IChzY2VuZTogU2NlbmUpID0+IHtcclxuICAgIC8vIElmIHdlIGFyZSBub3QgaW4gYSBicm93c2VyLCBkbyBub3RoaW5nXHJcbiAgICBpZiAodHlwZW9mIGRvY3VtZW50ID09PSBcInVuZGVmaW5lZFwiKSB7XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG4gICAgaWYgKGNhcHR1cmVPbkVudGVyQ291bnQgPT09IDApIHtcclxuICAgICAgICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKFwicG9pbnRlcm1vdmVcIiwgb25Qb2ludGVyTW92ZSk7XHJcbiAgICAgICAgZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcInRvdWNoc3RhcnRcIiwgb25Qb2ludGVyTW92ZSk7XHJcbiAgICAgICAgX3NjZW5lID0gX3NjZW5lID8/IHNjZW5lO1xyXG4gICAgICAgIExvZ2dlci5Mb2coXCJQb2ludGVyRXZlbnRzQ2FwdHVyZUJlaGF2aW9yOiBTdGFydGluZyBvYnNlcnZhdGlvbiBvZiBwb2ludGVyIG1vdmUgZXZlbnRzLlwiKTtcclxuICAgICAgICBfc2NlbmUub25EaXNwb3NlT2JzZXJ2YWJsZS5hZGQoZG9TdG9wQ2FwdHVyZU9uRW50ZXIpO1xyXG4gICAgfVxyXG4gICAgY2FwdHVyZU9uRW50ZXJDb3VudCsrO1xyXG59O1xyXG5cclxuY29uc3QgZG9TdG9wQ2FwdHVyZU9uRW50ZXIgPSAoKSA9PiB7XHJcbiAgICBkb2N1bWVudC5yZW1vdmVFdmVudExpc3RlbmVyKFwicG9pbnRlcm1vdmVcIiwgb25Qb2ludGVyTW92ZSk7XHJcbiAgICBkb2N1bWVudC5yZW1vdmVFdmVudExpc3RlbmVyKFwidG91Y2hzdGFydFwiLCBvblBvaW50ZXJNb3ZlKTtcclxuICAgIF9zY2VuZSA9IG51bGw7XHJcbiAgICBMb2dnZXIuTG9nKFwiUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvcjogU3RvcHBpbmcgb2JzZXJ2YXRpb24gb2YgcG9pbnRlciBtb3ZlIGV2ZW50cy5cIik7XHJcbiAgICBjYXB0dXJlT25FbnRlckNvdW50ID0gMDtcclxufTtcclxuXHJcbmNvbnN0IHN0b3BDYXB0dXJlT25FbnRlciA9ICgpID0+IHtcclxuICAgIC8vIElmIHdlIGFyZSBub3QgaW4gYSBicm93c2VyLCBkbyBub3RoaW5nXHJcbiAgICBpZiAodHlwZW9mIGRvY3VtZW50ID09PSBcInVuZGVmaW5lZFwiKSB7XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIElmIHdlIGFyZSBub3Qgb2JzZXJ2aW5nIHBvaW50ZXIgbW92ZW1lbnQsIGRvIG5vdGhpbmdcclxuICAgIGlmICghX3NjZW5lKSB7XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIGNhcHR1cmVPbkVudGVyQ291bnQtLTtcclxuICAgIGlmIChjYXB0dXJlT25FbnRlckNvdW50IDw9IDApIHtcclxuICAgICAgICBkb1N0b3BDYXB0dXJlT25FbnRlcigpO1xyXG4gICAgfVxyXG59O1xyXG5cclxuLy8gTW9kdWxlIGxldmVsIGZ1bmN0aW9uIHVzZWQgdG8gZGV0ZXJtaW5lIGlmIGFuIGVudGVyZWQgbWVzaCBzaG91bGQgY2FwdHVyZSBwb2ludGVyIGV2ZW50c1xyXG5jb25zdCBvblBvaW50ZXJNb3ZlID0gKGV2dDogUG9pbnRlckV2ZW50IHwgVG91Y2hFdmVudCkgPT4ge1xyXG4gICAgaWYgKCFfc2NlbmUpIHtcclxuICAgICAgICByZXR1cm47XHJcbiAgICB9XHJcblxyXG4gICAgY29uc3QgY2FudmFzUmVjdCA9IF9zY2VuZS5nZXRFbmdpbmUoKS5nZXRSZW5kZXJpbmdDYW52YXNDbGllbnRSZWN0KCk7XHJcbiAgICBpZiAoIWNhbnZhc1JlY3QpIHtcclxuICAgICAgICByZXR1cm47XHJcbiAgICB9XHJcblxyXG4gICAgLy8gR2V0IHRoZSBvYmplY3QgdGhhdCBjb250YWlucyB0aGUgY2xpZW50IFggYW5kIFkgZnJvbSBlaXRoZXIgdGhlIHBvaW50ZXIgZXZlbnQgb3IgZnJvbSB0aGVcclxuICAgIC8vIFRvdWNoRXZlbnQgdG91Y2hcclxuICAgIGNvbnN0IHsgY2xpZW50WCwgY2xpZW50WSB9ID0gXCJ0b3VjaGVzXCIgaW4gZXZ0ID8gZXZ0LnRvdWNoZXNbMF0gOiBldnQ7XHJcblxyXG4gICAgLy8gZ2V0IHRoZSBwaWNrZWQgbWVzaCwgaWYgYW55XHJcbiAgICBjb25zdCBwb2ludGVyU2NyZWVuWCA9IGNsaWVudFggLSBjYW52YXNSZWN0LmxlZnQ7XHJcbiAgICBjb25zdCBwb2ludGVyU2NyZWVuWSA9IGNsaWVudFkgLSBjYW52YXNSZWN0LnRvcDtcclxuXHJcbiAgICBsZXQgcG9pbnRlckNhcHR1cmVCZWhhdmlvcjogUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvciB8IHVuZGVmaW5lZDtcclxuICAgIGNvbnN0IHBpY2tSZXN1bHQgPSBfc2NlbmUucGljayhwb2ludGVyU2NyZWVuWCwgcG9pbnRlclNjcmVlblksIChtZXNoKSA9PiB7XHJcbiAgICAgICAgLy8gSWYgdGhlIG1lc2ggaGFzIGFuIGluc3RhbmNlIG9mIFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IgYXR0YWNoZWQgdG8gaXQsXHJcbiAgICAgICAgLy8gYW5kIGNhcHR1cmUgb24gcG9pbnRlciBlbnRlciBpcyB0cnVlLCB0aGVuIHdlIHdhbnQgdG8gcGljayBpdFxyXG4gICAgICAgIGNvbnN0IHBvaW50ZXJDYXB0dXJlQmVoYXZpb3IgPSBtZXNoVG9CZWhhdmlvck1hcC5nZXQobWVzaCk7XHJcbiAgICAgICAgcmV0dXJuIG1lc2guaXNFbmFibGVkKCkgJiYgdHlwZW9mIHBvaW50ZXJDYXB0dXJlQmVoYXZpb3IgIT09IFwidW5kZWZpbmVkXCIgJiYgcG9pbnRlckNhcHR1cmVCZWhhdmlvci5fY2FwdHVyZU9uUG9pbnRlckVudGVyO1xyXG4gICAgfSk7XHJcblxyXG4gICAgbGV0IHBpY2tlZE1lc2g6IEFic3RyYWN0TWVzaCB8IG51bGw7XHJcbiAgICBpZiAocGlja1Jlc3VsdC5oaXQpIHtcclxuICAgICAgICBwaWNrZWRNZXNoID0gcGlja1Jlc3VsdC5waWNrZWRNZXNoO1xyXG4gICAgfSBlbHNlIHtcclxuICAgICAgICBwaWNrZWRNZXNoID0gbnVsbDtcclxuICAgIH1cclxuXHJcbiAgICBjb25zdCBjYXB0dXJpbmdJZEFzSW50ID0gcGFyc2VJbnQoZ2V0Q2FwdHVyaW5nSWQoKSB8fCBcIlwiKTtcclxuXHJcbiAgICAvLyBpZiB0aGUgcGlja2VkIG1lc2ggaXMgdGhlIGN1cnJlbnQgY2FwdHVyaW5nIG1lc2gsIGRvIG5vdGhpbmdcclxuICAgIGlmIChwaWNrZWRNZXNoICYmIHBpY2tlZE1lc2gudW5pcXVlSWQgPT09IGNhcHR1cmluZ0lkQXNJbnQpIHtcclxuICAgICAgICByZXR1cm47XHJcbiAgICB9XHJcblxyXG4gICAgLy8gSWYgdGhlcmUgaXMgYSBjYXB0dXJpbmcgbWVzaCBhbmQgaXQgaXMgbm90IHRoZSBjdXJyZW50IHBpY2tlZCBtZXNoLCBvciBub1xyXG4gICAgLy8gbWVzaCBpcyBwaWNrZWQsIHJlbGVhc2UgdGhlIGNhcHR1cmluZyBtZXNoXHJcbiAgICBpZiAoY2FwdHVyaW5nSWRBc0ludCAmJiAoIXBpY2tlZE1lc2ggfHwgcGlja2VkTWVzaC51bmlxdWVJZCAhPT0gY2FwdHVyaW5nSWRBc0ludCkpIHtcclxuICAgICAgICByZWxlYXNlQ3VycmVudCgpO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIElmIHRoZXJlIGlzIGEgcGlja2VkIG1lc2ggYW5kIGl0IGlzIG5vdCB0aGUgY3VycmVudCBjYXB0dXJpbmcgbWVzaCwgY2FwdHVyZVxyXG4gICAgLy8gdGhlIHBvaW50ZXIgZXZlbnRzLiAgTm90ZSB0aGF0IHRoZSBjdXJyZW50IGNhcHR1cmluZyBtZXNoIGhhcyBhbHJlYWR5IGJlZW5cclxuICAgIC8vIHJlbGVhc2VkIGFib3ZlXHJcbiAgICBpZiAocGlja2VkTWVzaCkge1xyXG4gICAgICAgIHBvaW50ZXJDYXB0dXJlQmVoYXZpb3IgPSBtZXNoVG9CZWhhdmlvck1hcC5nZXQocGlja2VkTWVzaCk7XHJcbiAgICAgICAgcG9pbnRlckNhcHR1cmVCZWhhdmlvciEuY2FwdHVyZVBvaW50ZXJFdmVudHMoKTtcclxuICAgIH1cclxufTtcclxuXHJcbi8qKlxyXG4gKiBCZWhhdmlvciBmb3IgYW55IGNvbnRlbnQgdGhhdCBjYW4gY2FwdHVyZSBwb2ludGVyIGV2ZW50cywgaS5lLiBieXBhc3MgdGhlIEJhYnlsb24gcG9pbnRlciBldmVudCBoYW5kbGluZ1xyXG4gKiBhbmQgcmVjZWl2ZSBwb2ludGVyIGV2ZW50cyBkaXJlY3RseS4gIEl0IHdpbGwgcmVnaXN0ZXIgdGhlIGNhcHR1cmUgdHJpZ2dlcnMgYW5kIG5lZ290aWF0ZSB0aGUgY2FwdHVyZSBhbmRcclxuICogcmVsZWFzZSBvZiBwb2ludGVyIGV2ZW50cy4gIEN1cmVybnRseSB0aGlzIGFwcGxpZXMgb25seSB0byBIdG1sTWVzaFxyXG4gKi9cclxuZXhwb3J0IGNsYXNzIFBvaW50ZXJFdmVudHNDYXB0dXJlQmVoYXZpb3IgaW1wbGVtZW50cyBCZWhhdmlvcjxBYnN0cmFjdE1lc2g+IHtcclxuICAgIC8qKiBnZXRzIG9yIHNldHMgYmVoYXZpb3IncyBuYW1lICovXHJcbiAgICBwdWJsaWMgbmFtZSA9IFwiUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvclwiO1xyXG5cclxuICAgIHByaXZhdGUgX2F0dGFjaGVkTWVzaDogQWJzdHJhY3RNZXNoIHwgbnVsbDtcclxuICAgIC8qKiBAaW50ZXJuYWwgKi9cclxuICAgIHB1YmxpYyBfY2FwdHVyZU9uUG9pbnRlckVudGVyOiBib29sZWFuO1xyXG5cclxuICAgIC8qKlxyXG4gICAgICogR2V0cyBvciBzZXRzIHRoZSBtZXNoIHRoYXQgdGhlIGJlaGF2aW9yIGlzIGF0dGFjaGVkIHRvXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBnZXQgYXR0YWNoZWRNZXNoKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl9hdHRhY2hlZE1lc2g7XHJcbiAgICB9XHJcblxyXG4gICAgcHVibGljIHNldCBhdHRhY2hlZE1lc2godmFsdWU6IEFic3RyYWN0TWVzaCB8IG51bGwpIHtcclxuICAgICAgICB0aGlzLl9hdHRhY2hlZE1lc2ggPSB2YWx1ZTtcclxuICAgIH1cclxuXHJcbiAgICBjb25zdHJ1Y3RvcihcclxuICAgICAgICBwcml2YXRlIF9jYXB0dXJlQ2FsbGJhY2s6ICgpID0+IHZvaWQsXHJcbiAgICAgICAgcHJpdmF0ZSBfcmVsZWFzZUNhbGxiYWNrOiAoKSA9PiB2b2lkLFxyXG4gICAgICAgIHsgY2FwdHVyZU9uUG9pbnRlckVudGVyID0gdHJ1ZSB9ID0ge31cclxuICAgICkge1xyXG4gICAgICAgIHRoaXMuX2F0dGFjaGVkTWVzaCA9IG51bGw7XHJcbiAgICAgICAgdGhpcy5fY2FwdHVyZU9uUG9pbnRlckVudGVyID0gY2FwdHVyZU9uUG9pbnRlckVudGVyO1xyXG5cclxuICAgICAgICAvLyBXYXJuIGlmIHdlIGFyZSBub3QgaW4gYSBicm93c2VyXHJcbiAgICAgICAgaWYgKHR5cGVvZiBkb2N1bWVudCA9PT0gXCJ1bmRlZmluZWRcIikge1xyXG4gICAgICAgICAgICBMb2dnZXIuV2FybihgQ3JlYXRpbmcgYW4gaW5zdGFuY2Ugb2YgUG9pbnRlckV2ZW50c0NhcHR1cmVCZWhhdmlvciBvdXRzaWRlIG9mIGEgYnJvd3Nlci4gIFRoZSBiZWhhdmlvciB3aWxsIG5vdCB3b3JrLmApO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIFNldCBpZiB0aGUgYmVoYXZpb3Igc2hvdWxkIGNhcHR1cmUgcG9pbnRlciBldmVudHMgd2hlbiB0aGUgcG9pbnRlciBlbnRlcnMgdGhlIG1lc2hcclxuICAgICAqL1xyXG4gICAgcHVibGljIHNldCBjYXB0dXJlT25Qb2ludGVyRW50ZXIoY2FwdHVyZU9uUG9pbnRlckVudGVyOiBib29sZWFuKSB7XHJcbiAgICAgICAgaWYgKHRoaXMuX2NhcHR1cmVPblBvaW50ZXJFbnRlciA9PT0gY2FwdHVyZU9uUG9pbnRlckVudGVyKSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcbiAgICAgICAgdGhpcy5fY2FwdHVyZU9uUG9pbnRlckVudGVyID0gY2FwdHVyZU9uUG9pbnRlckVudGVyO1xyXG4gICAgICAgIGlmICh0aGlzLl9hdHRhY2hlZE1lc2gpIHtcclxuICAgICAgICAgICAgaWYgKHRoaXMuX2NhcHR1cmVPblBvaW50ZXJFbnRlcikge1xyXG4gICAgICAgICAgICAgICAgc3RhcnRDYXB0dXJlT25FbnRlcih0aGlzLl9hdHRhY2hlZE1lc2guZ2V0U2NlbmUoKSEpO1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgc3RvcENhcHR1cmVPbkVudGVyKCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBGdW5jdGlvbiBjYWxsZWQgd2hlbiB0aGUgYmVoYXZpb3IgbmVlZHMgdG8gYmUgaW5pdGlhbGl6ZWQgKGJlZm9yZSBhdHRhY2hpbmcgaXQgdG8gYSB0YXJnZXQpXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBpbml0KCkge31cclxuXHJcbiAgICAvKipcclxuICAgICAqIENhbGxlZCB3aGVuIHRoZSBiZWhhdmlvciBpcyBhdHRhY2hlZCB0byBhIHRhcmdldFxyXG4gICAgICogQHBhcmFtIG1lc2ggZGVmaW5lcyB0aGUgdGFyZ2V0IHdoZXJlIHRoZSBiZWhhdmlvciBpcyBhdHRhY2hlZCB0b1xyXG4gICAgICovXHJcbiAgICBwdWJsaWMgYXR0YWNoKG1lc2g6IEFic3RyYWN0TWVzaCkge1xyXG4gICAgICAgIC8vIEFkZCBhIHJlZmVyZW5jZSB0byB0aGlzIGJlaGF2aW9yIG9uIHRoZSBtZXNoLiAgV2UgZG8gdGhpcyBzbyB3ZSBjYW4gZ2V0IGFcclxuICAgICAgICAvLyByZWZlcmVuY2UgdG8gdGhlIGJlaGF2aW9yIGluIHRoZSBvblBvaW50ZXJNb3ZlIGZ1bmN0aW9uIHdpdGhvdXQgcmVseWluZyBvblxyXG4gICAgICAgIC8vIGdldEJlaGF2aW9yQnlOYW1lKCksIHdoaWNoIGRvZXMgYSBsaW5lYXIgc2VhcmNoIG9mIHRoZSBiZWhhdmlvcnMgYXJyYXkuXHJcbiAgICAgICAgdGhpcy5hdHRhY2hlZE1lc2ggPSBtZXNoO1xyXG4gICAgICAgIG1lc2hUb0JlaGF2aW9yTWFwLnNldChtZXNoLCB0aGlzKTtcclxuICAgICAgICBpZiAodGhpcy5fY2FwdHVyZU9uUG9pbnRlckVudGVyKSB7XHJcbiAgICAgICAgICAgIHN0YXJ0Q2FwdHVyZU9uRW50ZXIobWVzaC5nZXRTY2VuZSgpISk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ2FsbGVkIHdoZW4gdGhlIGJlaGF2aW9yIGlzIGRldGFjaGVkIGZyb20gaXRzIHRhcmdldFxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgZGV0YWNoKCkge1xyXG4gICAgICAgIGlmICghdGhpcy5hdHRhY2hlZE1lc2gpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuICAgICAgICAvLyBSZW1vdmUgdGhlIHJlZmVyZW5jZSB0byB0aGlzIGJlaGF2aW9yIGZyb20gdGhlIG1lc2hcclxuICAgICAgICBtZXNoVG9CZWhhdmlvck1hcC5kZWxldGUodGhpcy5hdHRhY2hlZE1lc2gpO1xyXG4gICAgICAgIGlmICh0aGlzLl9jYXB0dXJlT25Qb2ludGVyRW50ZXIpIHtcclxuICAgICAgICAgICAgc3RvcENhcHR1cmVPbkVudGVyKCk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHRoaXMuYXR0YWNoZWRNZXNoID0gbnVsbDtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIERpc3Bvc2UgdGhlIGJlaGF2aW9yXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBkaXNwb3NlKCkge1xyXG4gICAgICAgIHRoaXMuZGV0YWNoKCk7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gUmVsZWFzZSBwb2ludGVyIGV2ZW50c1xyXG4gICAgcHVibGljIHJlbGVhc2VQb2ludGVyRXZlbnRzKCkge1xyXG4gICAgICAgIGlmICghdGhpcy5hdHRhY2hlZE1lc2gpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuICAgICAgICByZXF1ZXN0UmVsZWFzZSh0aGlzLmF0dGFjaGVkTWVzaC51bmlxdWVJZC50b1N0cmluZygpKTtcclxuICAgIH1cclxuXHJcbiAgICAvLyBDYXB0dXJlIHBvaW50ZXIgZXZlbnRzXHJcbiAgICBwdWJsaWMgY2FwdHVyZVBvaW50ZXJFdmVudHMoKSB7XHJcbiAgICAgICAgaWYgKCF0aGlzLmF0dGFjaGVkTWVzaCkge1xyXG4gICAgICAgICAgICByZXR1cm47XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHJlcXVlc3RDYXB0dXJlKHRoaXMuYXR0YWNoZWRNZXNoLnVuaXF1ZUlkLnRvU3RyaW5nKCksIHRoaXMuX2NhcHR1cmVDYWxsYmFjaywgdGhpcy5fcmVsZWFzZUNhbGxiYWNrKTtcclxuICAgIH1cclxufVxyXG4iLCJleHBvcnQgKiBmcm9tIFwiLi9odG1sTWVzaFwiO1xuIiwibW9kdWxlLmV4cG9ydHMgPSBfX1dFQlBBQ0tfRVhURVJOQUxfTU9EVUxFX2JhYnlsb25qc19NYXRoc19tYXRoX187IiwiLy8gVGhlIG1vZHVsZSBjYWNoZVxudmFyIF9fd2VicGFja19tb2R1bGVfY2FjaGVfXyA9IHt9O1xuXG4vLyBUaGUgcmVxdWlyZSBmdW5jdGlvblxuZnVuY3Rpb24gX193ZWJwYWNrX3JlcXVpcmVfXyhtb2R1bGVJZCkge1xuXHQvLyBDaGVjayBpZiBtb2R1bGUgaXMgaW4gY2FjaGVcblx0dmFyIGNhY2hlZE1vZHVsZSA9IF9fd2VicGFja19tb2R1bGVfY2FjaGVfX1ttb2R1bGVJZF07XG5cdGlmIChjYWNoZWRNb2R1bGUgIT09IHVuZGVmaW5lZCkge1xuXHRcdHJldHVybiBjYWNoZWRNb2R1bGUuZXhwb3J0cztcblx0fVxuXHQvLyBDcmVhdGUgYSBuZXcgbW9kdWxlIChhbmQgcHV0IGl0IGludG8gdGhlIGNhY2hlKVxuXHR2YXIgbW9kdWxlID0gX193ZWJwYWNrX21vZHVsZV9jYWNoZV9fW21vZHVsZUlkXSA9IHtcblx0XHQvLyBubyBtb2R1bGUuaWQgbmVlZGVkXG5cdFx0Ly8gbm8gbW9kdWxlLmxvYWRlZCBuZWVkZWRcblx0XHRleHBvcnRzOiB7fVxuXHR9O1xuXG5cdC8vIEV4ZWN1dGUgdGhlIG1vZHVsZSBmdW5jdGlvblxuXHRfX3dlYnBhY2tfbW9kdWxlc19fW21vZHVsZUlkXShtb2R1bGUsIG1vZHVsZS5leHBvcnRzLCBfX3dlYnBhY2tfcmVxdWlyZV9fKTtcblxuXHQvLyBSZXR1cm4gdGhlIGV4cG9ydHMgb2YgdGhlIG1vZHVsZVxuXHRyZXR1cm4gbW9kdWxlLmV4cG9ydHM7XG59XG5cbiIsIi8vIGdldERlZmF1bHRFeHBvcnQgZnVuY3Rpb24gZm9yIGNvbXBhdGliaWxpdHkgd2l0aCBub24taGFybW9ueSBtb2R1bGVzXG5fX3dlYnBhY2tfcmVxdWlyZV9fLm4gPSAobW9kdWxlKSA9PiB7XG5cdHZhciBnZXR0ZXIgPSBtb2R1bGUgJiYgbW9kdWxlLl9fZXNNb2R1bGUgP1xuXHRcdCgpID0+IChtb2R1bGVbJ2RlZmF1bHQnXSkgOlxuXHRcdCgpID0+IChtb2R1bGUpO1xuXHRfX3dlYnBhY2tfcmVxdWlyZV9fLmQoZ2V0dGVyLCB7IGE6IGdldHRlciB9KTtcblx0cmV0dXJuIGdldHRlcjtcbn07IiwiLy8gZGVmaW5lIGdldHRlciBmdW5jdGlvbnMgZm9yIGhhcm1vbnkgZXhwb3J0c1xuX193ZWJwYWNrX3JlcXVpcmVfXy5kID0gKGV4cG9ydHMsIGRlZmluaXRpb24pID0+IHtcblx0Zm9yKHZhciBrZXkgaW4gZGVmaW5pdGlvbikge1xuXHRcdGlmKF9fd2VicGFja19yZXF1aXJlX18ubyhkZWZpbml0aW9uLCBrZXkpICYmICFfX3dlYnBhY2tfcmVxdWlyZV9fLm8oZXhwb3J0cywga2V5KSkge1xuXHRcdFx0T2JqZWN0LmRlZmluZVByb3BlcnR5KGV4cG9ydHMsIGtleSwgeyBlbnVtZXJhYmxlOiB0cnVlLCBnZXQ6IGRlZmluaXRpb25ba2V5XSB9KTtcblx0XHR9XG5cdH1cbn07IiwiX193ZWJwYWNrX3JlcXVpcmVfXy5vID0gKG9iaiwgcHJvcCkgPT4gKE9iamVjdC5wcm90b3R5cGUuaGFzT3duUHJvcGVydHkuY2FsbChvYmosIHByb3ApKSIsIi8vIGRlZmluZSBfX2VzTW9kdWxlIG9uIGV4cG9ydHNcbl9fd2VicGFja19yZXF1aXJlX18uciA9IChleHBvcnRzKSA9PiB7XG5cdGlmKHR5cGVvZiBTeW1ib2wgIT09ICd1bmRlZmluZWQnICYmIFN5bWJvbC50b1N0cmluZ1RhZykge1xuXHRcdE9iamVjdC5kZWZpbmVQcm9wZXJ0eShleHBvcnRzLCBTeW1ib2wudG9TdHJpbmdUYWcsIHsgdmFsdWU6ICdNb2R1bGUnIH0pO1xuXHR9XG5cdE9iamVjdC5kZWZpbmVQcm9wZXJ0eShleHBvcnRzLCAnX19lc01vZHVsZScsIHsgdmFsdWU6IHRydWUgfSk7XG59OyIsIi8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBpbXBvcnQvbm8taW50ZXJuYWwtbW9kdWxlc1xyXG5pbXBvcnQgKiBhcyBhZGRvbnMgZnJvbSBcImFkZG9ucy9pbmRleFwiO1xyXG5cclxuZXhwb3J0IHsgYWRkb25zIH07XHJcbmV4cG9ydCBkZWZhdWx0IGFkZG9ucztcclxuIl0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9