@vonage/ml-transformers 1.0.1

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 (33) hide show
  1. package/README.md +9 -0
  2. package/dist/docs/.nojekyll +1 -0
  3. package/dist/docs/assets/highlight.css +85 -0
  4. package/dist/docs/assets/icons.css +1043 -0
  5. package/dist/docs/assets/icons.png +0 -0
  6. package/dist/docs/assets/icons@2x.png +0 -0
  7. package/dist/docs/assets/main.js +52 -0
  8. package/dist/docs/assets/search.js +1 -0
  9. package/dist/docs/assets/style.css +1413 -0
  10. package/dist/docs/assets/widgets.png +0 -0
  11. package/dist/docs/assets/widgets@2x.png +0 -0
  12. package/dist/docs/enums/BlurRadius.html +7 -0
  13. package/dist/docs/index.html +15 -0
  14. package/dist/docs/modules.html +30 -0
  15. package/dist/ml-transformers.es.js +1254 -0
  16. package/dist/ml-transformers.umd.js +2 -0
  17. package/dist/models/selfie_segmentation_landscape.tflite +0 -0
  18. package/dist/tflite-simd.wasm +0 -0
  19. package/dist/types/main.d.ts +95 -0
  20. package/dist/types/src/transformers/BackgroundTransformer.d.ts +34 -0
  21. package/dist/types/src/transformers/BlurPostProcess.d.ts +10 -0
  22. package/dist/types/src/transformers/PostProcessInterface.d.ts +5 -0
  23. package/dist/types/src/transformers/Queue.d.ts +13 -0
  24. package/dist/types/src/transformers/SilueteBlurPostProcess.d.ts +10 -0
  25. package/dist/types/src/transformers/VideoPostProcess.d.ts +11 -0
  26. package/dist/types/src/transformers/VirtualPostProcess.d.ts +11 -0
  27. package/dist/types/src/transformers/VonageTFLite.d.ts +28 -0
  28. package/dist/types/src/transformers/VonageTransformerConfig.d.ts +1 -0
  29. package/dist/types/src/types.d.ts +65 -0
  30. package/dist/types/src/utils/utils.d.ts +6 -0
  31. package/dist/types/src/vonage/VonageMediaProcessorThread.d.ts +1 -0
  32. package/dist/types/src/vonage/VonageMediaProcessorWorker.d.ts +10 -0
  33. package/package.json +65 -0
@@ -0,0 +1,1254 @@
1
+ function isSupported$1() {
2
+ return new Promise((resolve, reject) => {
3
+ if (typeof MediaStreamTrackProcessor === "undefined" || typeof MediaStreamTrackGenerator === "undefined") {
4
+ reject("Your browser does not support the MediaStreamTrack API for Insertable Streams of Media.");
5
+ } else {
6
+ resolve();
7
+ }
8
+ });
9
+ }
10
+ var axios$2 = { exports: {} };
11
+ var bind$2 = function bind(fn, thisArg) {
12
+ return function wrap() {
13
+ var args = new Array(arguments.length);
14
+ for (var i = 0; i < args.length; i++) {
15
+ args[i] = arguments[i];
16
+ }
17
+ return fn.apply(thisArg, args);
18
+ };
19
+ };
20
+ var bind$1 = bind$2;
21
+ var toString = Object.prototype.toString;
22
+ function isArray(val) {
23
+ return Array.isArray(val);
24
+ }
25
+ function isUndefined(val) {
26
+ return typeof val === "undefined";
27
+ }
28
+ function isBuffer(val) {
29
+ return val !== null && !isUndefined(val) && val.constructor !== null && !isUndefined(val.constructor) && typeof val.constructor.isBuffer === "function" && val.constructor.isBuffer(val);
30
+ }
31
+ function isArrayBuffer(val) {
32
+ return toString.call(val) === "[object ArrayBuffer]";
33
+ }
34
+ function isFormData(val) {
35
+ return toString.call(val) === "[object FormData]";
36
+ }
37
+ function isArrayBufferView(val) {
38
+ var result;
39
+ if (typeof ArrayBuffer !== "undefined" && ArrayBuffer.isView) {
40
+ result = ArrayBuffer.isView(val);
41
+ } else {
42
+ result = val && val.buffer && isArrayBuffer(val.buffer);
43
+ }
44
+ return result;
45
+ }
46
+ function isString(val) {
47
+ return typeof val === "string";
48
+ }
49
+ function isNumber(val) {
50
+ return typeof val === "number";
51
+ }
52
+ function isObject(val) {
53
+ return val !== null && typeof val === "object";
54
+ }
55
+ function isPlainObject(val) {
56
+ if (toString.call(val) !== "[object Object]") {
57
+ return false;
58
+ }
59
+ var prototype = Object.getPrototypeOf(val);
60
+ return prototype === null || prototype === Object.prototype;
61
+ }
62
+ function isDate(val) {
63
+ return toString.call(val) === "[object Date]";
64
+ }
65
+ function isFile(val) {
66
+ return toString.call(val) === "[object File]";
67
+ }
68
+ function isBlob(val) {
69
+ return toString.call(val) === "[object Blob]";
70
+ }
71
+ function isFunction(val) {
72
+ return toString.call(val) === "[object Function]";
73
+ }
74
+ function isStream(val) {
75
+ return isObject(val) && isFunction(val.pipe);
76
+ }
77
+ function isURLSearchParams(val) {
78
+ return toString.call(val) === "[object URLSearchParams]";
79
+ }
80
+ function trim(str) {
81
+ return str.trim ? str.trim() : str.replace(/^\s+|\s+$/g, "");
82
+ }
83
+ function isStandardBrowserEnv() {
84
+ if (typeof navigator !== "undefined" && (navigator.product === "ReactNative" || navigator.product === "NativeScript" || navigator.product === "NS")) {
85
+ return false;
86
+ }
87
+ return typeof window !== "undefined" && typeof document !== "undefined";
88
+ }
89
+ function forEach(obj, fn) {
90
+ if (obj === null || typeof obj === "undefined") {
91
+ return;
92
+ }
93
+ if (typeof obj !== "object") {
94
+ obj = [obj];
95
+ }
96
+ if (isArray(obj)) {
97
+ for (var i = 0, l = obj.length; i < l; i++) {
98
+ fn.call(null, obj[i], i, obj);
99
+ }
100
+ } else {
101
+ for (var key in obj) {
102
+ if (Object.prototype.hasOwnProperty.call(obj, key)) {
103
+ fn.call(null, obj[key], key, obj);
104
+ }
105
+ }
106
+ }
107
+ }
108
+ function merge() {
109
+ var result = {};
110
+ function assignValue(val, key) {
111
+ if (isPlainObject(result[key]) && isPlainObject(val)) {
112
+ result[key] = merge(result[key], val);
113
+ } else if (isPlainObject(val)) {
114
+ result[key] = merge({}, val);
115
+ } else if (isArray(val)) {
116
+ result[key] = val.slice();
117
+ } else {
118
+ result[key] = val;
119
+ }
120
+ }
121
+ for (var i = 0, l = arguments.length; i < l; i++) {
122
+ forEach(arguments[i], assignValue);
123
+ }
124
+ return result;
125
+ }
126
+ function extend(a, b, thisArg) {
127
+ forEach(b, function assignValue(val, key) {
128
+ if (thisArg && typeof val === "function") {
129
+ a[key] = bind$1(val, thisArg);
130
+ } else {
131
+ a[key] = val;
132
+ }
133
+ });
134
+ return a;
135
+ }
136
+ function stripBOM(content) {
137
+ if (content.charCodeAt(0) === 65279) {
138
+ content = content.slice(1);
139
+ }
140
+ return content;
141
+ }
142
+ var utils$e = {
143
+ isArray,
144
+ isArrayBuffer,
145
+ isBuffer,
146
+ isFormData,
147
+ isArrayBufferView,
148
+ isString,
149
+ isNumber,
150
+ isObject,
151
+ isPlainObject,
152
+ isUndefined,
153
+ isDate,
154
+ isFile,
155
+ isBlob,
156
+ isFunction,
157
+ isStream,
158
+ isURLSearchParams,
159
+ isStandardBrowserEnv,
160
+ forEach,
161
+ merge,
162
+ extend,
163
+ trim,
164
+ stripBOM
165
+ };
166
+ var utils$d = utils$e;
167
+ function encode(val) {
168
+ return encodeURIComponent(val).replace(/%3A/gi, ":").replace(/%24/g, "$").replace(/%2C/gi, ",").replace(/%20/g, "+").replace(/%5B/gi, "[").replace(/%5D/gi, "]");
169
+ }
170
+ var buildURL$2 = function buildURL(url, params, paramsSerializer) {
171
+ if (!params) {
172
+ return url;
173
+ }
174
+ var serializedParams;
175
+ if (paramsSerializer) {
176
+ serializedParams = paramsSerializer(params);
177
+ } else if (utils$d.isURLSearchParams(params)) {
178
+ serializedParams = params.toString();
179
+ } else {
180
+ var parts = [];
181
+ utils$d.forEach(params, function serialize(val, key) {
182
+ if (val === null || typeof val === "undefined") {
183
+ return;
184
+ }
185
+ if (utils$d.isArray(val)) {
186
+ key = key + "[]";
187
+ } else {
188
+ val = [val];
189
+ }
190
+ utils$d.forEach(val, function parseValue(v) {
191
+ if (utils$d.isDate(v)) {
192
+ v = v.toISOString();
193
+ } else if (utils$d.isObject(v)) {
194
+ v = JSON.stringify(v);
195
+ }
196
+ parts.push(encode(key) + "=" + encode(v));
197
+ });
198
+ });
199
+ serializedParams = parts.join("&");
200
+ }
201
+ if (serializedParams) {
202
+ var hashmarkIndex = url.indexOf("#");
203
+ if (hashmarkIndex !== -1) {
204
+ url = url.slice(0, hashmarkIndex);
205
+ }
206
+ url += (url.indexOf("?") === -1 ? "?" : "&") + serializedParams;
207
+ }
208
+ return url;
209
+ };
210
+ var utils$c = utils$e;
211
+ function InterceptorManager$1() {
212
+ this.handlers = [];
213
+ }
214
+ InterceptorManager$1.prototype.use = function use(fulfilled, rejected, options) {
215
+ this.handlers.push({
216
+ fulfilled,
217
+ rejected,
218
+ synchronous: options ? options.synchronous : false,
219
+ runWhen: options ? options.runWhen : null
220
+ });
221
+ return this.handlers.length - 1;
222
+ };
223
+ InterceptorManager$1.prototype.eject = function eject(id) {
224
+ if (this.handlers[id]) {
225
+ this.handlers[id] = null;
226
+ }
227
+ };
228
+ InterceptorManager$1.prototype.forEach = function forEach2(fn) {
229
+ utils$c.forEach(this.handlers, function forEachHandler(h) {
230
+ if (h !== null) {
231
+ fn(h);
232
+ }
233
+ });
234
+ };
235
+ var InterceptorManager_1 = InterceptorManager$1;
236
+ var utils$b = utils$e;
237
+ var normalizeHeaderName$1 = function normalizeHeaderName(headers, normalizedName) {
238
+ utils$b.forEach(headers, function processHeader(value, name) {
239
+ if (name !== normalizedName && name.toUpperCase() === normalizedName.toUpperCase()) {
240
+ headers[normalizedName] = value;
241
+ delete headers[name];
242
+ }
243
+ });
244
+ };
245
+ var enhanceError$2 = function enhanceError(error, config, code, request2, response) {
246
+ error.config = config;
247
+ if (code) {
248
+ error.code = code;
249
+ }
250
+ error.request = request2;
251
+ error.response = response;
252
+ error.isAxiosError = true;
253
+ error.toJSON = function toJSON() {
254
+ return {
255
+ message: this.message,
256
+ name: this.name,
257
+ description: this.description,
258
+ number: this.number,
259
+ fileName: this.fileName,
260
+ lineNumber: this.lineNumber,
261
+ columnNumber: this.columnNumber,
262
+ stack: this.stack,
263
+ config: this.config,
264
+ code: this.code,
265
+ status: this.response && this.response.status ? this.response.status : null
266
+ };
267
+ };
268
+ return error;
269
+ };
270
+ var enhanceError$1 = enhanceError$2;
271
+ var createError$2 = function createError(message, config, code, request2, response) {
272
+ var error = new Error(message);
273
+ return enhanceError$1(error, config, code, request2, response);
274
+ };
275
+ var createError$1 = createError$2;
276
+ var settle$1 = function settle(resolve, reject, response) {
277
+ var validateStatus2 = response.config.validateStatus;
278
+ if (!response.status || !validateStatus2 || validateStatus2(response.status)) {
279
+ resolve(response);
280
+ } else {
281
+ reject(createError$1("Request failed with status code " + response.status, response.config, null, response.request, response));
282
+ }
283
+ };
284
+ var utils$a = utils$e;
285
+ var cookies$1 = utils$a.isStandardBrowserEnv() ? function standardBrowserEnv() {
286
+ return {
287
+ write: function write(name, value, expires, path, domain, secure) {
288
+ var cookie = [];
289
+ cookie.push(name + "=" + encodeURIComponent(value));
290
+ if (utils$a.isNumber(expires)) {
291
+ cookie.push("expires=" + new Date(expires).toGMTString());
292
+ }
293
+ if (utils$a.isString(path)) {
294
+ cookie.push("path=" + path);
295
+ }
296
+ if (utils$a.isString(domain)) {
297
+ cookie.push("domain=" + domain);
298
+ }
299
+ if (secure === true) {
300
+ cookie.push("secure");
301
+ }
302
+ document.cookie = cookie.join("; ");
303
+ },
304
+ read: function read(name) {
305
+ var match = document.cookie.match(new RegExp("(^|;\\s*)(" + name + ")=([^;]*)"));
306
+ return match ? decodeURIComponent(match[3]) : null;
307
+ },
308
+ remove: function remove(name) {
309
+ this.write(name, "", Date.now() - 864e5);
310
+ }
311
+ };
312
+ }() : function nonStandardBrowserEnv() {
313
+ return {
314
+ write: function write() {
315
+ },
316
+ read: function read() {
317
+ return null;
318
+ },
319
+ remove: function remove() {
320
+ }
321
+ };
322
+ }();
323
+ var isAbsoluteURL$1 = function isAbsoluteURL(url) {
324
+ return /^([a-z][a-z\d+\-.]*:)?\/\//i.test(url);
325
+ };
326
+ var combineURLs$1 = function combineURLs(baseURL, relativeURL) {
327
+ return relativeURL ? baseURL.replace(/\/+$/, "") + "/" + relativeURL.replace(/^\/+/, "") : baseURL;
328
+ };
329
+ var isAbsoluteURL2 = isAbsoluteURL$1;
330
+ var combineURLs2 = combineURLs$1;
331
+ var buildFullPath$1 = function buildFullPath(baseURL, requestedURL) {
332
+ if (baseURL && !isAbsoluteURL2(requestedURL)) {
333
+ return combineURLs2(baseURL, requestedURL);
334
+ }
335
+ return requestedURL;
336
+ };
337
+ var utils$9 = utils$e;
338
+ var ignoreDuplicateOf = [
339
+ "age",
340
+ "authorization",
341
+ "content-length",
342
+ "content-type",
343
+ "etag",
344
+ "expires",
345
+ "from",
346
+ "host",
347
+ "if-modified-since",
348
+ "if-unmodified-since",
349
+ "last-modified",
350
+ "location",
351
+ "max-forwards",
352
+ "proxy-authorization",
353
+ "referer",
354
+ "retry-after",
355
+ "user-agent"
356
+ ];
357
+ var parseHeaders$1 = function parseHeaders(headers) {
358
+ var parsed = {};
359
+ var key;
360
+ var val;
361
+ var i;
362
+ if (!headers) {
363
+ return parsed;
364
+ }
365
+ utils$9.forEach(headers.split("\n"), function parser(line) {
366
+ i = line.indexOf(":");
367
+ key = utils$9.trim(line.substr(0, i)).toLowerCase();
368
+ val = utils$9.trim(line.substr(i + 1));
369
+ if (key) {
370
+ if (parsed[key] && ignoreDuplicateOf.indexOf(key) >= 0) {
371
+ return;
372
+ }
373
+ if (key === "set-cookie") {
374
+ parsed[key] = (parsed[key] ? parsed[key] : []).concat([val]);
375
+ } else {
376
+ parsed[key] = parsed[key] ? parsed[key] + ", " + val : val;
377
+ }
378
+ }
379
+ });
380
+ return parsed;
381
+ };
382
+ var utils$8 = utils$e;
383
+ var isURLSameOrigin$1 = utils$8.isStandardBrowserEnv() ? function standardBrowserEnv2() {
384
+ var msie = /(msie|trident)/i.test(navigator.userAgent);
385
+ var urlParsingNode = document.createElement("a");
386
+ var originURL;
387
+ function resolveURL(url) {
388
+ var href = url;
389
+ if (msie) {
390
+ urlParsingNode.setAttribute("href", href);
391
+ href = urlParsingNode.href;
392
+ }
393
+ urlParsingNode.setAttribute("href", href);
394
+ return {
395
+ href: urlParsingNode.href,
396
+ protocol: urlParsingNode.protocol ? urlParsingNode.protocol.replace(/:$/, "") : "",
397
+ host: urlParsingNode.host,
398
+ search: urlParsingNode.search ? urlParsingNode.search.replace(/^\?/, "") : "",
399
+ hash: urlParsingNode.hash ? urlParsingNode.hash.replace(/^#/, "") : "",
400
+ hostname: urlParsingNode.hostname,
401
+ port: urlParsingNode.port,
402
+ pathname: urlParsingNode.pathname.charAt(0) === "/" ? urlParsingNode.pathname : "/" + urlParsingNode.pathname
403
+ };
404
+ }
405
+ originURL = resolveURL(window.location.href);
406
+ return function isURLSameOrigin2(requestURL) {
407
+ var parsed = utils$8.isString(requestURL) ? resolveURL(requestURL) : requestURL;
408
+ return parsed.protocol === originURL.protocol && parsed.host === originURL.host;
409
+ };
410
+ }() : function nonStandardBrowserEnv2() {
411
+ return function isURLSameOrigin2() {
412
+ return true;
413
+ };
414
+ }();
415
+ function Cancel$3(message) {
416
+ this.message = message;
417
+ }
418
+ Cancel$3.prototype.toString = function toString2() {
419
+ return "Cancel" + (this.message ? ": " + this.message : "");
420
+ };
421
+ Cancel$3.prototype.__CANCEL__ = true;
422
+ var Cancel_1 = Cancel$3;
423
+ var utils$7 = utils$e;
424
+ var settle2 = settle$1;
425
+ var cookies = cookies$1;
426
+ var buildURL$1 = buildURL$2;
427
+ var buildFullPath2 = buildFullPath$1;
428
+ var parseHeaders2 = parseHeaders$1;
429
+ var isURLSameOrigin = isURLSameOrigin$1;
430
+ var createError2 = createError$2;
431
+ var defaults$4 = defaults_1;
432
+ var Cancel$2 = Cancel_1;
433
+ var xhr = function xhrAdapter(config) {
434
+ return new Promise(function dispatchXhrRequest(resolve, reject) {
435
+ var requestData = config.data;
436
+ var requestHeaders = config.headers;
437
+ var responseType = config.responseType;
438
+ var onCanceled;
439
+ function done() {
440
+ if (config.cancelToken) {
441
+ config.cancelToken.unsubscribe(onCanceled);
442
+ }
443
+ if (config.signal) {
444
+ config.signal.removeEventListener("abort", onCanceled);
445
+ }
446
+ }
447
+ if (utils$7.isFormData(requestData)) {
448
+ delete requestHeaders["Content-Type"];
449
+ }
450
+ var request2 = new XMLHttpRequest();
451
+ if (config.auth) {
452
+ var username = config.auth.username || "";
453
+ var password = config.auth.password ? unescape(encodeURIComponent(config.auth.password)) : "";
454
+ requestHeaders.Authorization = "Basic " + btoa(username + ":" + password);
455
+ }
456
+ var fullPath = buildFullPath2(config.baseURL, config.url);
457
+ request2.open(config.method.toUpperCase(), buildURL$1(fullPath, config.params, config.paramsSerializer), true);
458
+ request2.timeout = config.timeout;
459
+ function onloadend() {
460
+ if (!request2) {
461
+ return;
462
+ }
463
+ var responseHeaders = "getAllResponseHeaders" in request2 ? parseHeaders2(request2.getAllResponseHeaders()) : null;
464
+ var responseData = !responseType || responseType === "text" || responseType === "json" ? request2.responseText : request2.response;
465
+ var response = {
466
+ data: responseData,
467
+ status: request2.status,
468
+ statusText: request2.statusText,
469
+ headers: responseHeaders,
470
+ config,
471
+ request: request2
472
+ };
473
+ settle2(function _resolve(value) {
474
+ resolve(value);
475
+ done();
476
+ }, function _reject(err) {
477
+ reject(err);
478
+ done();
479
+ }, response);
480
+ request2 = null;
481
+ }
482
+ if ("onloadend" in request2) {
483
+ request2.onloadend = onloadend;
484
+ } else {
485
+ request2.onreadystatechange = function handleLoad() {
486
+ if (!request2 || request2.readyState !== 4) {
487
+ return;
488
+ }
489
+ if (request2.status === 0 && !(request2.responseURL && request2.responseURL.indexOf("file:") === 0)) {
490
+ return;
491
+ }
492
+ setTimeout(onloadend);
493
+ };
494
+ }
495
+ request2.onabort = function handleAbort() {
496
+ if (!request2) {
497
+ return;
498
+ }
499
+ reject(createError2("Request aborted", config, "ECONNABORTED", request2));
500
+ request2 = null;
501
+ };
502
+ request2.onerror = function handleError() {
503
+ reject(createError2("Network Error", config, null, request2));
504
+ request2 = null;
505
+ };
506
+ request2.ontimeout = function handleTimeout() {
507
+ var timeoutErrorMessage = config.timeout ? "timeout of " + config.timeout + "ms exceeded" : "timeout exceeded";
508
+ var transitional2 = config.transitional || defaults$4.transitional;
509
+ if (config.timeoutErrorMessage) {
510
+ timeoutErrorMessage = config.timeoutErrorMessage;
511
+ }
512
+ reject(createError2(timeoutErrorMessage, config, transitional2.clarifyTimeoutError ? "ETIMEDOUT" : "ECONNABORTED", request2));
513
+ request2 = null;
514
+ };
515
+ if (utils$7.isStandardBrowserEnv()) {
516
+ var xsrfValue = (config.withCredentials || isURLSameOrigin(fullPath)) && config.xsrfCookieName ? cookies.read(config.xsrfCookieName) : void 0;
517
+ if (xsrfValue) {
518
+ requestHeaders[config.xsrfHeaderName] = xsrfValue;
519
+ }
520
+ }
521
+ if ("setRequestHeader" in request2) {
522
+ utils$7.forEach(requestHeaders, function setRequestHeader(val, key) {
523
+ if (typeof requestData === "undefined" && key.toLowerCase() === "content-type") {
524
+ delete requestHeaders[key];
525
+ } else {
526
+ request2.setRequestHeader(key, val);
527
+ }
528
+ });
529
+ }
530
+ if (!utils$7.isUndefined(config.withCredentials)) {
531
+ request2.withCredentials = !!config.withCredentials;
532
+ }
533
+ if (responseType && responseType !== "json") {
534
+ request2.responseType = config.responseType;
535
+ }
536
+ if (typeof config.onDownloadProgress === "function") {
537
+ request2.addEventListener("progress", config.onDownloadProgress);
538
+ }
539
+ if (typeof config.onUploadProgress === "function" && request2.upload) {
540
+ request2.upload.addEventListener("progress", config.onUploadProgress);
541
+ }
542
+ if (config.cancelToken || config.signal) {
543
+ onCanceled = function(cancel) {
544
+ if (!request2) {
545
+ return;
546
+ }
547
+ reject(!cancel || cancel && cancel.type ? new Cancel$2("canceled") : cancel);
548
+ request2.abort();
549
+ request2 = null;
550
+ };
551
+ config.cancelToken && config.cancelToken.subscribe(onCanceled);
552
+ if (config.signal) {
553
+ config.signal.aborted ? onCanceled() : config.signal.addEventListener("abort", onCanceled);
554
+ }
555
+ }
556
+ if (!requestData) {
557
+ requestData = null;
558
+ }
559
+ request2.send(requestData);
560
+ });
561
+ };
562
+ var utils$6 = utils$e;
563
+ var normalizeHeaderName2 = normalizeHeaderName$1;
564
+ var enhanceError2 = enhanceError$2;
565
+ var DEFAULT_CONTENT_TYPE = {
566
+ "Content-Type": "application/x-www-form-urlencoded"
567
+ };
568
+ function setContentTypeIfUnset(headers, value) {
569
+ if (!utils$6.isUndefined(headers) && utils$6.isUndefined(headers["Content-Type"])) {
570
+ headers["Content-Type"] = value;
571
+ }
572
+ }
573
+ function getDefaultAdapter() {
574
+ var adapter;
575
+ if (typeof XMLHttpRequest !== "undefined") {
576
+ adapter = xhr;
577
+ } else if (typeof process !== "undefined" && Object.prototype.toString.call(process) === "[object process]") {
578
+ adapter = xhr;
579
+ }
580
+ return adapter;
581
+ }
582
+ function stringifySafely(rawValue, parser, encoder) {
583
+ if (utils$6.isString(rawValue)) {
584
+ try {
585
+ (parser || JSON.parse)(rawValue);
586
+ return utils$6.trim(rawValue);
587
+ } catch (e) {
588
+ if (e.name !== "SyntaxError") {
589
+ throw e;
590
+ }
591
+ }
592
+ }
593
+ return (encoder || JSON.stringify)(rawValue);
594
+ }
595
+ var defaults$3 = {
596
+ transitional: {
597
+ silentJSONParsing: true,
598
+ forcedJSONParsing: true,
599
+ clarifyTimeoutError: false
600
+ },
601
+ adapter: getDefaultAdapter(),
602
+ transformRequest: [function transformRequest(data2, headers) {
603
+ normalizeHeaderName2(headers, "Accept");
604
+ normalizeHeaderName2(headers, "Content-Type");
605
+ if (utils$6.isFormData(data2) || utils$6.isArrayBuffer(data2) || utils$6.isBuffer(data2) || utils$6.isStream(data2) || utils$6.isFile(data2) || utils$6.isBlob(data2)) {
606
+ return data2;
607
+ }
608
+ if (utils$6.isArrayBufferView(data2)) {
609
+ return data2.buffer;
610
+ }
611
+ if (utils$6.isURLSearchParams(data2)) {
612
+ setContentTypeIfUnset(headers, "application/x-www-form-urlencoded;charset=utf-8");
613
+ return data2.toString();
614
+ }
615
+ if (utils$6.isObject(data2) || headers && headers["Content-Type"] === "application/json") {
616
+ setContentTypeIfUnset(headers, "application/json");
617
+ return stringifySafely(data2);
618
+ }
619
+ return data2;
620
+ }],
621
+ transformResponse: [function transformResponse(data2) {
622
+ var transitional2 = this.transitional || defaults$3.transitional;
623
+ var silentJSONParsing = transitional2 && transitional2.silentJSONParsing;
624
+ var forcedJSONParsing = transitional2 && transitional2.forcedJSONParsing;
625
+ var strictJSONParsing = !silentJSONParsing && this.responseType === "json";
626
+ if (strictJSONParsing || forcedJSONParsing && utils$6.isString(data2) && data2.length) {
627
+ try {
628
+ return JSON.parse(data2);
629
+ } catch (e) {
630
+ if (strictJSONParsing) {
631
+ if (e.name === "SyntaxError") {
632
+ throw enhanceError2(e, this, "E_JSON_PARSE");
633
+ }
634
+ throw e;
635
+ }
636
+ }
637
+ }
638
+ return data2;
639
+ }],
640
+ timeout: 0,
641
+ xsrfCookieName: "XSRF-TOKEN",
642
+ xsrfHeaderName: "X-XSRF-TOKEN",
643
+ maxContentLength: -1,
644
+ maxBodyLength: -1,
645
+ validateStatus: function validateStatus(status) {
646
+ return status >= 200 && status < 300;
647
+ },
648
+ headers: {
649
+ common: {
650
+ "Accept": "application/json, text/plain, */*"
651
+ }
652
+ }
653
+ };
654
+ utils$6.forEach(["delete", "get", "head"], function forEachMethodNoData(method) {
655
+ defaults$3.headers[method] = {};
656
+ });
657
+ utils$6.forEach(["post", "put", "patch"], function forEachMethodWithData(method) {
658
+ defaults$3.headers[method] = utils$6.merge(DEFAULT_CONTENT_TYPE);
659
+ });
660
+ var defaults_1 = defaults$3;
661
+ var utils$5 = utils$e;
662
+ var defaults$2 = defaults_1;
663
+ var transformData$1 = function transformData(data2, headers, fns) {
664
+ var context = this || defaults$2;
665
+ utils$5.forEach(fns, function transform(fn) {
666
+ data2 = fn.call(context, data2, headers);
667
+ });
668
+ return data2;
669
+ };
670
+ var isCancel$1 = function isCancel(value) {
671
+ return !!(value && value.__CANCEL__);
672
+ };
673
+ var utils$4 = utils$e;
674
+ var transformData2 = transformData$1;
675
+ var isCancel2 = isCancel$1;
676
+ var defaults$1 = defaults_1;
677
+ var Cancel$1 = Cancel_1;
678
+ function throwIfCancellationRequested(config) {
679
+ if (config.cancelToken) {
680
+ config.cancelToken.throwIfRequested();
681
+ }
682
+ if (config.signal && config.signal.aborted) {
683
+ throw new Cancel$1("canceled");
684
+ }
685
+ }
686
+ var dispatchRequest$1 = function dispatchRequest(config) {
687
+ throwIfCancellationRequested(config);
688
+ config.headers = config.headers || {};
689
+ config.data = transformData2.call(config, config.data, config.headers, config.transformRequest);
690
+ config.headers = utils$4.merge(config.headers.common || {}, config.headers[config.method] || {}, config.headers);
691
+ utils$4.forEach(["delete", "get", "head", "post", "put", "patch", "common"], function cleanHeaderConfig(method) {
692
+ delete config.headers[method];
693
+ });
694
+ var adapter = config.adapter || defaults$1.adapter;
695
+ return adapter(config).then(function onAdapterResolution(response) {
696
+ throwIfCancellationRequested(config);
697
+ response.data = transformData2.call(config, response.data, response.headers, config.transformResponse);
698
+ return response;
699
+ }, function onAdapterRejection(reason) {
700
+ if (!isCancel2(reason)) {
701
+ throwIfCancellationRequested(config);
702
+ if (reason && reason.response) {
703
+ reason.response.data = transformData2.call(config, reason.response.data, reason.response.headers, config.transformResponse);
704
+ }
705
+ }
706
+ return Promise.reject(reason);
707
+ });
708
+ };
709
+ var utils$3 = utils$e;
710
+ var mergeConfig$2 = function mergeConfig(config1, config2) {
711
+ config2 = config2 || {};
712
+ var config = {};
713
+ function getMergedValue(target, source2) {
714
+ if (utils$3.isPlainObject(target) && utils$3.isPlainObject(source2)) {
715
+ return utils$3.merge(target, source2);
716
+ } else if (utils$3.isPlainObject(source2)) {
717
+ return utils$3.merge({}, source2);
718
+ } else if (utils$3.isArray(source2)) {
719
+ return source2.slice();
720
+ }
721
+ return source2;
722
+ }
723
+ function mergeDeepProperties(prop) {
724
+ if (!utils$3.isUndefined(config2[prop])) {
725
+ return getMergedValue(config1[prop], config2[prop]);
726
+ } else if (!utils$3.isUndefined(config1[prop])) {
727
+ return getMergedValue(void 0, config1[prop]);
728
+ }
729
+ }
730
+ function valueFromConfig2(prop) {
731
+ if (!utils$3.isUndefined(config2[prop])) {
732
+ return getMergedValue(void 0, config2[prop]);
733
+ }
734
+ }
735
+ function defaultToConfig2(prop) {
736
+ if (!utils$3.isUndefined(config2[prop])) {
737
+ return getMergedValue(void 0, config2[prop]);
738
+ } else if (!utils$3.isUndefined(config1[prop])) {
739
+ return getMergedValue(void 0, config1[prop]);
740
+ }
741
+ }
742
+ function mergeDirectKeys(prop) {
743
+ if (prop in config2) {
744
+ return getMergedValue(config1[prop], config2[prop]);
745
+ } else if (prop in config1) {
746
+ return getMergedValue(void 0, config1[prop]);
747
+ }
748
+ }
749
+ var mergeMap = {
750
+ "url": valueFromConfig2,
751
+ "method": valueFromConfig2,
752
+ "data": valueFromConfig2,
753
+ "baseURL": defaultToConfig2,
754
+ "transformRequest": defaultToConfig2,
755
+ "transformResponse": defaultToConfig2,
756
+ "paramsSerializer": defaultToConfig2,
757
+ "timeout": defaultToConfig2,
758
+ "timeoutMessage": defaultToConfig2,
759
+ "withCredentials": defaultToConfig2,
760
+ "adapter": defaultToConfig2,
761
+ "responseType": defaultToConfig2,
762
+ "xsrfCookieName": defaultToConfig2,
763
+ "xsrfHeaderName": defaultToConfig2,
764
+ "onUploadProgress": defaultToConfig2,
765
+ "onDownloadProgress": defaultToConfig2,
766
+ "decompress": defaultToConfig2,
767
+ "maxContentLength": defaultToConfig2,
768
+ "maxBodyLength": defaultToConfig2,
769
+ "transport": defaultToConfig2,
770
+ "httpAgent": defaultToConfig2,
771
+ "httpsAgent": defaultToConfig2,
772
+ "cancelToken": defaultToConfig2,
773
+ "socketPath": defaultToConfig2,
774
+ "responseEncoding": defaultToConfig2,
775
+ "validateStatus": mergeDirectKeys
776
+ };
777
+ utils$3.forEach(Object.keys(config1).concat(Object.keys(config2)), function computeConfigValue(prop) {
778
+ var merge2 = mergeMap[prop] || mergeDeepProperties;
779
+ var configValue = merge2(prop);
780
+ utils$3.isUndefined(configValue) && merge2 !== mergeDirectKeys || (config[prop] = configValue);
781
+ });
782
+ return config;
783
+ };
784
+ var data = {
785
+ "version": "0.25.0"
786
+ };
787
+ var VERSION = data.version;
788
+ var validators$1 = {};
789
+ ["object", "boolean", "number", "function", "string", "symbol"].forEach(function(type, i) {
790
+ validators$1[type] = function validator2(thing) {
791
+ return typeof thing === type || "a" + (i < 1 ? "n " : " ") + type;
792
+ };
793
+ });
794
+ var deprecatedWarnings = {};
795
+ validators$1.transitional = function transitional(validator2, version2, message) {
796
+ function formatMessage(opt, desc) {
797
+ return "[Axios v" + VERSION + "] Transitional option '" + opt + "'" + desc + (message ? ". " + message : "");
798
+ }
799
+ return function(value, opt, opts) {
800
+ if (validator2 === false) {
801
+ throw new Error(formatMessage(opt, " has been removed" + (version2 ? " in " + version2 : "")));
802
+ }
803
+ if (version2 && !deprecatedWarnings[opt]) {
804
+ deprecatedWarnings[opt] = true;
805
+ console.warn(formatMessage(opt, " has been deprecated since v" + version2 + " and will be removed in the near future"));
806
+ }
807
+ return validator2 ? validator2(value, opt, opts) : true;
808
+ };
809
+ };
810
+ function assertOptions(options, schema, allowUnknown) {
811
+ if (typeof options !== "object") {
812
+ throw new TypeError("options must be an object");
813
+ }
814
+ var keys = Object.keys(options);
815
+ var i = keys.length;
816
+ while (i-- > 0) {
817
+ var opt = keys[i];
818
+ var validator2 = schema[opt];
819
+ if (validator2) {
820
+ var value = options[opt];
821
+ var result = value === void 0 || validator2(value, opt, options);
822
+ if (result !== true) {
823
+ throw new TypeError("option " + opt + " must be " + result);
824
+ }
825
+ continue;
826
+ }
827
+ if (allowUnknown !== true) {
828
+ throw Error("Unknown option " + opt);
829
+ }
830
+ }
831
+ }
832
+ var validator$1 = {
833
+ assertOptions,
834
+ validators: validators$1
835
+ };
836
+ var utils$2 = utils$e;
837
+ var buildURL2 = buildURL$2;
838
+ var InterceptorManager = InterceptorManager_1;
839
+ var dispatchRequest2 = dispatchRequest$1;
840
+ var mergeConfig$1 = mergeConfig$2;
841
+ var validator = validator$1;
842
+ var validators = validator.validators;
843
+ function Axios$1(instanceConfig) {
844
+ this.defaults = instanceConfig;
845
+ this.interceptors = {
846
+ request: new InterceptorManager(),
847
+ response: new InterceptorManager()
848
+ };
849
+ }
850
+ Axios$1.prototype.request = function request(configOrUrl, config) {
851
+ if (typeof configOrUrl === "string") {
852
+ config = config || {};
853
+ config.url = configOrUrl;
854
+ } else {
855
+ config = configOrUrl || {};
856
+ }
857
+ if (!config.url) {
858
+ throw new Error("Provided config url is not valid");
859
+ }
860
+ config = mergeConfig$1(this.defaults, config);
861
+ if (config.method) {
862
+ config.method = config.method.toLowerCase();
863
+ } else if (this.defaults.method) {
864
+ config.method = this.defaults.method.toLowerCase();
865
+ } else {
866
+ config.method = "get";
867
+ }
868
+ var transitional2 = config.transitional;
869
+ if (transitional2 !== void 0) {
870
+ validator.assertOptions(transitional2, {
871
+ silentJSONParsing: validators.transitional(validators.boolean),
872
+ forcedJSONParsing: validators.transitional(validators.boolean),
873
+ clarifyTimeoutError: validators.transitional(validators.boolean)
874
+ }, false);
875
+ }
876
+ var requestInterceptorChain = [];
877
+ var synchronousRequestInterceptors = true;
878
+ this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {
879
+ if (typeof interceptor.runWhen === "function" && interceptor.runWhen(config) === false) {
880
+ return;
881
+ }
882
+ synchronousRequestInterceptors = synchronousRequestInterceptors && interceptor.synchronous;
883
+ requestInterceptorChain.unshift(interceptor.fulfilled, interceptor.rejected);
884
+ });
885
+ var responseInterceptorChain = [];
886
+ this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {
887
+ responseInterceptorChain.push(interceptor.fulfilled, interceptor.rejected);
888
+ });
889
+ var promise;
890
+ if (!synchronousRequestInterceptors) {
891
+ var chain = [dispatchRequest2, void 0];
892
+ Array.prototype.unshift.apply(chain, requestInterceptorChain);
893
+ chain = chain.concat(responseInterceptorChain);
894
+ promise = Promise.resolve(config);
895
+ while (chain.length) {
896
+ promise = promise.then(chain.shift(), chain.shift());
897
+ }
898
+ return promise;
899
+ }
900
+ var newConfig = config;
901
+ while (requestInterceptorChain.length) {
902
+ var onFulfilled = requestInterceptorChain.shift();
903
+ var onRejected = requestInterceptorChain.shift();
904
+ try {
905
+ newConfig = onFulfilled(newConfig);
906
+ } catch (error) {
907
+ onRejected(error);
908
+ break;
909
+ }
910
+ }
911
+ try {
912
+ promise = dispatchRequest2(newConfig);
913
+ } catch (error) {
914
+ return Promise.reject(error);
915
+ }
916
+ while (responseInterceptorChain.length) {
917
+ promise = promise.then(responseInterceptorChain.shift(), responseInterceptorChain.shift());
918
+ }
919
+ return promise;
920
+ };
921
+ Axios$1.prototype.getUri = function getUri(config) {
922
+ if (!config.url) {
923
+ throw new Error("Provided config url is not valid");
924
+ }
925
+ config = mergeConfig$1(this.defaults, config);
926
+ return buildURL2(config.url, config.params, config.paramsSerializer).replace(/^\?/, "");
927
+ };
928
+ utils$2.forEach(["delete", "get", "head", "options"], function forEachMethodNoData2(method) {
929
+ Axios$1.prototype[method] = function(url, config) {
930
+ return this.request(mergeConfig$1(config || {}, {
931
+ method,
932
+ url,
933
+ data: (config || {}).data
934
+ }));
935
+ };
936
+ });
937
+ utils$2.forEach(["post", "put", "patch"], function forEachMethodWithData2(method) {
938
+ Axios$1.prototype[method] = function(url, data2, config) {
939
+ return this.request(mergeConfig$1(config || {}, {
940
+ method,
941
+ url,
942
+ data: data2
943
+ }));
944
+ };
945
+ });
946
+ var Axios_1 = Axios$1;
947
+ var Cancel = Cancel_1;
948
+ function CancelToken(executor) {
949
+ if (typeof executor !== "function") {
950
+ throw new TypeError("executor must be a function.");
951
+ }
952
+ var resolvePromise;
953
+ this.promise = new Promise(function promiseExecutor(resolve) {
954
+ resolvePromise = resolve;
955
+ });
956
+ var token = this;
957
+ this.promise.then(function(cancel) {
958
+ if (!token._listeners)
959
+ return;
960
+ var i;
961
+ var l = token._listeners.length;
962
+ for (i = 0; i < l; i++) {
963
+ token._listeners[i](cancel);
964
+ }
965
+ token._listeners = null;
966
+ });
967
+ this.promise.then = function(onfulfilled) {
968
+ var _resolve;
969
+ var promise = new Promise(function(resolve) {
970
+ token.subscribe(resolve);
971
+ _resolve = resolve;
972
+ }).then(onfulfilled);
973
+ promise.cancel = function reject() {
974
+ token.unsubscribe(_resolve);
975
+ };
976
+ return promise;
977
+ };
978
+ executor(function cancel(message) {
979
+ if (token.reason) {
980
+ return;
981
+ }
982
+ token.reason = new Cancel(message);
983
+ resolvePromise(token.reason);
984
+ });
985
+ }
986
+ CancelToken.prototype.throwIfRequested = function throwIfRequested() {
987
+ if (this.reason) {
988
+ throw this.reason;
989
+ }
990
+ };
991
+ CancelToken.prototype.subscribe = function subscribe(listener) {
992
+ if (this.reason) {
993
+ listener(this.reason);
994
+ return;
995
+ }
996
+ if (this._listeners) {
997
+ this._listeners.push(listener);
998
+ } else {
999
+ this._listeners = [listener];
1000
+ }
1001
+ };
1002
+ CancelToken.prototype.unsubscribe = function unsubscribe(listener) {
1003
+ if (!this._listeners) {
1004
+ return;
1005
+ }
1006
+ var index = this._listeners.indexOf(listener);
1007
+ if (index !== -1) {
1008
+ this._listeners.splice(index, 1);
1009
+ }
1010
+ };
1011
+ CancelToken.source = function source() {
1012
+ var cancel;
1013
+ var token = new CancelToken(function executor(c) {
1014
+ cancel = c;
1015
+ });
1016
+ return {
1017
+ token,
1018
+ cancel
1019
+ };
1020
+ };
1021
+ var CancelToken_1 = CancelToken;
1022
+ var spread = function spread2(callback) {
1023
+ return function wrap(arr) {
1024
+ return callback.apply(null, arr);
1025
+ };
1026
+ };
1027
+ var utils$1 = utils$e;
1028
+ var isAxiosError = function isAxiosError2(payload) {
1029
+ return utils$1.isObject(payload) && payload.isAxiosError === true;
1030
+ };
1031
+ var utils = utils$e;
1032
+ var bind2 = bind$2;
1033
+ var Axios = Axios_1;
1034
+ var mergeConfig2 = mergeConfig$2;
1035
+ var defaults = defaults_1;
1036
+ function createInstance(defaultConfig) {
1037
+ var context = new Axios(defaultConfig);
1038
+ var instance = bind2(Axios.prototype.request, context);
1039
+ utils.extend(instance, Axios.prototype, context);
1040
+ utils.extend(instance, context);
1041
+ instance.create = function create(instanceConfig) {
1042
+ return createInstance(mergeConfig2(defaultConfig, instanceConfig));
1043
+ };
1044
+ return instance;
1045
+ }
1046
+ var axios$1 = createInstance(defaults);
1047
+ axios$1.Axios = Axios;
1048
+ axios$1.Cancel = Cancel_1;
1049
+ axios$1.CancelToken = CancelToken_1;
1050
+ axios$1.isCancel = isCancel$1;
1051
+ axios$1.VERSION = data.version;
1052
+ axios$1.all = function all(promises) {
1053
+ return Promise.all(promises);
1054
+ };
1055
+ axios$1.spread = spread;
1056
+ axios$1.isAxiosError = isAxiosError;
1057
+ axios$2.exports = axios$1;
1058
+ axios$2.exports.default = axios$1;
1059
+ var byteToHex = [];
1060
+ for (var i = 0; i < 256; ++i) {
1061
+ byteToHex.push((i + 256).toString(16).substr(1));
1062
+ }
1063
+ class InsertableStreamHelper {
1064
+ constructor() {
1065
+ this.processor_ = null;
1066
+ this.generator_ = null;
1067
+ }
1068
+ init(track) {
1069
+ return new Promise((resolve, reject) => {
1070
+ try {
1071
+ this.processor_ = new MediaStreamTrackProcessor(track);
1072
+ } catch (e) {
1073
+ console.log(`[InsertableStreamHelper] MediaStreamTrackProcessor failed: ${e}`);
1074
+ reject(e);
1075
+ }
1076
+ try {
1077
+ if (track.kind === "audio") {
1078
+ this.generator_ = new MediaStreamTrackGenerator({ kind: "audio" });
1079
+ } else if (track.kind === "video") {
1080
+ this.generator_ = new MediaStreamTrackGenerator({ kind: "video" });
1081
+ } else {
1082
+ reject("kind not supported");
1083
+ }
1084
+ } catch (e) {
1085
+ console.log(`[InsertableStreamHelper] MediaStreamTrackGenerator failed: ${e}`);
1086
+ reject(e);
1087
+ }
1088
+ resolve();
1089
+ });
1090
+ }
1091
+ getReadable() {
1092
+ return this.processor_.readable;
1093
+ }
1094
+ getWriteable() {
1095
+ return this.generator_.writable;
1096
+ }
1097
+ getProccesorTrack() {
1098
+ return this.generator_;
1099
+ }
1100
+ }
1101
+ class MediaProcessorConnector {
1102
+ constructor(vonageMediaProcessor) {
1103
+ this.insertableStreamHelper_ = new InsertableStreamHelper();
1104
+ this.mediaProcessor_ = vonageMediaProcessor;
1105
+ }
1106
+ setTrack(track) {
1107
+ return new Promise((resolve, reject) => {
1108
+ this.insertableStreamHelper_.init(track).then(() => {
1109
+ this.mediaProcessor_.transform(this.insertableStreamHelper_.getReadable(), this.insertableStreamHelper_.getWriteable()).then(() => {
1110
+ resolve(this.insertableStreamHelper_.getProccesorTrack());
1111
+ }).catch((e) => {
1112
+ reject(e);
1113
+ });
1114
+ }).catch((e) => {
1115
+ reject(e);
1116
+ });
1117
+ });
1118
+ }
1119
+ destroy() {
1120
+ return new Promise((resolve, reject) => {
1121
+ if (this.mediaProcessor_) {
1122
+ this.mediaProcessor_.destroy().then(() => {
1123
+ resolve();
1124
+ }).catch((e) => {
1125
+ reject(e);
1126
+ });
1127
+ } else {
1128
+ reject("no processor");
1129
+ }
1130
+ });
1131
+ }
1132
+ }
1133
+ const encodedJs = "KGZ1bmN0aW9uKCl7InVzZSBzdHJpY3QiO2NsYXNzIER7fUQudXBkYXRlcz17dHJhbnNmb3JtZXJfbmV3OiJOZXcgdHJhbnNmb3JtZXIiLHRyYW5zZm9ybWVyX251bGw6Ik51bGwgdHJhbnNmb3JtZXIifSxELmVycm9ycz17dHJhbnNmb3JtZXJfbm9uZToiTm8gdHJhbnNmb3JtZXJzIHByb3ZpZGVkIix0cmFuc2Zvcm1lcl9zdGFydDoiQ2Fubm90IHN0YXJ0IHRyYW5zZm9ybWVyIix0cmFuc2Zvcm1lcl90cmFuc2Zvcm06IkNhbm5vdCB0cmFuc2Zvcm0gZnJhbWUiLHRyYW5zZm9ybWVyX2ZsdXNoOiJDYW5ub3QgZmx1c2ggdHJhbnNmb3JtZXIifTt2YXIgX2U9e2V4cG9ydHM6e319LHhlPWZ1bmN0aW9uKGUsdCl7cmV0dXJuIGZ1bmN0aW9uKCl7Zm9yKHZhciBzPW5ldyBBcnJheShhcmd1bWVudHMubGVuZ3RoKSxhPTA7YTxzLmxlbmd0aDthKyspc1thXT1hcmd1bWVudHNbYV07cmV0dXJuIGUuYXBwbHkodCxzKX19LG10PXhlLGs9T2JqZWN0LnByb3RvdHlwZS50b1N0cmluZztmdW5jdGlvbiBnZShyKXtyZXR1cm4gQXJyYXkuaXNBcnJheShyKX1mdW5jdGlvbiB2ZShyKXtyZXR1cm4gdHlwZW9mIHI9PSJ1bmRlZmluZWQifWZ1bmN0aW9uIF90KHIpe3JldHVybiByIT09bnVsbCYmIXZlKHIpJiZyLmNvbnN0cnVjdG9yIT09bnVsbCYmIXZlKHIuY29uc3RydWN0b3IpJiZ0eXBlb2Ygci5jb25zdHJ1Y3Rvci5pc0J1ZmZlcj09ImZ1bmN0aW9uIiYmci5jb25zdHJ1Y3Rvci5pc0J1ZmZlcihyKX1mdW5jdGlvbiBOZShyKXtyZXR1cm4gay5jYWxsKHIpPT09IltvYmplY3QgQXJyYXlCdWZmZXJdIn1mdW5jdGlvbiBndChyKXtyZXR1cm4gay5jYWxsKHIpPT09IltvYmplY3QgRm9ybURhdGFdIn1mdW5jdGlvbiB2dChyKXt2YXIgZTtyZXR1cm4gdHlwZW9mIEFycmF5QnVmZmVyIT0idW5kZWZpbmVkIiYmQXJyYXlCdWZmZXIuaXNWaWV3P2U9QXJyYXlCdWZmZXIuaXNWaWV3KHIpOmU9ciYmci5idWZmZXImJk5lKHIuYnVmZmVyKSxlfWZ1bmN0aW9uIHl0KHIpe3JldHVybiB0eXBlb2Ygcj09InN0cmluZyJ9ZnVuY3Rpb24gd3Qocil7cmV0dXJuIHR5cGVvZiByPT0ibnVtYmVyIn1mdW5jdGlvbiBMZShyKXtyZXR1cm4gciE9PW51bGwmJnR5cGVvZiByPT0ib2JqZWN0In1mdW5jdGlvbiBLKHIpe2lmKGsuY2FsbChyKSE9PSJbb2JqZWN0IE9iamVjdF0iKXJldHVybiExO3ZhciBlPU9iamVjdC5nZXRQcm90b3R5cGVPZihyKTtyZXR1cm4gZT09PW51bGx8fGU9PT1PYmplY3QucHJvdG90eXBlfWZ1bmN0aW9uIEN0KHIpe3JldHVybiBrLmNhbGwocik9PT0iW29iamVjdCBEYXRlXSJ9ZnVuY3Rpb24gYnQocil7cmV0dXJuIGsuY2FsbChyKT09PSJbb2JqZWN0IEZpbGVdIn1mdW5jdGlvbiBUdChyKXtyZXR1cm4gay5jYWxsKHIpPT09IltvYmplY3QgQmxvYl0ifWZ1bmN0aW9uIFVlKHIpe3JldHVybiBrLmNhbGwocik9PT0iW29iamVjdCBGdW5jdGlvbl0ifWZ1bmN0aW9uIEV0KHIpe3JldHVybiBMZShyKSYmVWUoci5waXBlKX1mdW5jdGlvbiBSdChyKXtyZXR1cm4gay5jYWxsKHIpPT09IltvYmplY3QgVVJMU2VhcmNoUGFyYW1zXSJ9ZnVuY3Rpb24gUHQocil7cmV0dXJuIHIudHJpbT9yLnRyaW0oKTpyLnJlcGxhY2UoL15ccyt8XHMrJC9nLCIiKX1mdW5jdGlvbiBPdCgpe3JldHVybiB0eXBlb2YgbmF2aWdhdG9yIT0idW5kZWZpbmVkIiYmKG5hdmlnYXRvci5wcm9kdWN0PT09IlJlYWN0TmF0aXZlInx8bmF2aWdhdG9yLnByb2R1Y3Q9PT0iTmF0aXZlU2NyaXB0Inx8bmF2aWdhdG9yLnByb2R1Y3Q9PT0iTlMiKT8hMTp0eXBlb2Ygd2luZG93IT0idW5kZWZpbmVkIiYmdHlwZW9mIGRvY3VtZW50IT0idW5kZWZpbmVkIn1mdW5jdGlvbiB5ZShyLGUpe2lmKCEocj09PW51bGx8fHR5cGVvZiByPT0idW5kZWZpbmVkIikpaWYodHlwZW9mIHIhPSJvYmplY3QiJiYocj1bcl0pLGdlKHIpKWZvcih2YXIgdD0wLG49ci5sZW5ndGg7dDxuO3QrKyllLmNhbGwobnVsbCxyW3RdLHQscik7ZWxzZSBmb3IodmFyIHMgaW4gcilPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwocixzKSYmZS5jYWxsKG51bGwscltzXSxzLHIpfWZ1bmN0aW9uIHdlKCl7dmFyIHI9e307ZnVuY3Rpb24gZShzLGEpe0soclthXSkmJksocyk/clthXT13ZShyW2FdLHMpOksocyk/clthXT13ZSh7fSxzKTpnZShzKT9yW2FdPXMuc2xpY2UoKTpyW2FdPXN9Zm9yKHZhciB0PTAsbj1hcmd1bWVudHMubGVuZ3RoO3Q8bjt0KyspeWUoYXJndW1lbnRzW3RdLGUpO3JldHVybiByfWZ1bmN0aW9uIFN0KHIsZSx0KXtyZXR1cm4geWUoZSxmdW5jdGlvbihzLGEpe3QmJnR5cGVvZiBzPT0iZnVuY3Rpb24iP3JbYV09bXQocyx0KTpyW2FdPXN9KSxyfWZ1bmN0aW9uIEl0KHIpe3JldHVybiByLmNoYXJDb2RlQXQoMCk9PT02NTI3OSYmKHI9ci5zbGljZSgxKSkscn12YXIgUD17aXNBcnJheTpnZSxpc0FycmF5QnVmZmVyOk5lLGlzQnVmZmVyOl90LGlzRm9ybURhdGE6Z3QsaXNBcnJheUJ1ZmZlclZpZXc6dnQsaXNTdHJpbmc6eXQsaXNOdW1iZXI6d3QsaXNPYmplY3Q6TGUsaXNQbGFpbk9iamVjdDpLLGlzVW5kZWZpbmVkOnZlLGlzRGF0ZTpDdCxpc0ZpbGU6YnQsaXNCbG9iOlR0LGlzRnVuY3Rpb246VWUsaXNTdHJlYW06RXQsaXNVUkxTZWFyY2hQYXJhbXM6UnQsaXNTdGFuZGFyZEJyb3dzZXJFbnY6T3QsZm9yRWFjaDp5ZSxtZXJnZTp3ZSxleHRlbmQ6U3QsdHJpbTpQdCxzdHJpcEJPTTpJdH0sVz1QO2Z1bmN0aW9uIEJlKHIpe3JldHVybiBlbmNvZGVVUklDb21wb25lbnQocikucmVwbGFjZSgvJTNBL2dpLCI6IikucmVwbGFjZSgvJTI0L2csIiQiKS5yZXBsYWNlKC8lMkMvZ2ksIiwiKS5yZXBsYWNlKC8lMjAvZywiKyIpLnJlcGxhY2UoLyU1Qi9naSwiWyIpLnJlcGxhY2UoLyU1RC9naSwiXSIpfXZhciBrZT1mdW5jdGlvbihlLHQsbil7aWYoIXQpcmV0dXJuIGU7dmFyIHM7aWYobilzPW4odCk7ZWxzZSBpZihXLmlzVVJMU2VhcmNoUGFyYW1zKHQpKXM9dC50b1N0cmluZygpO2Vsc2V7dmFyIGE9W107Vy5mb3JFYWNoKHQsZnVuY3Rpb24oaCxnKXtoPT09bnVsbHx8dHlwZW9mIGg9PSJ1bmRlZmluZWQifHwoVy5pc0FycmF5KGgpP2c9ZysiW10iOmg9W2hdLFcuZm9yRWFjaChoLGZ1bmN0aW9uKHApe1cuaXNEYXRlKHApP3A9cC50b0lTT1N0cmluZygpOlcuaXNPYmplY3QocCkmJihwPUpTT04uc3RyaW5naWZ5KHApKSxhLnB1c2goQmUoZykrIj0iK0JlKHApKX0pKX0pLHM9YS5qb2luKCImIil9aWYocyl7dmFyIGY9ZS5pbmRleE9mKCIjIik7ZiE9PS0xJiYoZT1lLnNsaWNlKDAsZikpLGUrPShlLmluZGV4T2YoIj8iKT09PS0xPyI/IjoiJiIpK3N9cmV0dXJuIGV9LE10PVA7ZnVuY3Rpb24gWSgpe3RoaXMuaGFuZGxlcnM9W119WS5wcm90b3R5cGUudXNlPWZ1bmN0aW9uKGUsdCxuKXtyZXR1cm4gdGhpcy5oYW5kbGVycy5wdXNoKHtmdWxmaWxsZWQ6ZSxyZWplY3RlZDp0LHN5bmNocm9ub3VzOm4/bi5zeW5jaHJvbm91czohMSxydW5XaGVuOm4/bi5ydW5XaGVuOm51bGx9KSx0aGlzLmhhbmRsZXJzLmxlbmd0aC0xfSxZLnByb3RvdHlwZS5lamVjdD1mdW5jdGlvbihlKXt0aGlzLmhhbmRsZXJzW2VdJiYodGhpcy5oYW5kbGVyc1tlXT1udWxsKX0sWS5wcm90b3R5cGUuZm9yRWFjaD1mdW5jdGlvbihlKXtNdC5mb3JFYWNoKHRoaXMuaGFuZGxlcnMsZnVuY3Rpb24obil7biE9PW51bGwmJmUobil9KX07dmFyIEF0PVksRnQ9UCx4dD1mdW5jdGlvbihlLHQpe0Z0LmZvckVhY2goZSxmdW5jdGlvbihzLGEpe2EhPT10JiZhLnRvVXBwZXJDYXNlKCk9PT10LnRvVXBwZXJDYXNlKCkmJihlW3RdPXMsZGVsZXRlIGVbYV0pfSl9LEhlPWZ1bmN0aW9uKGUsdCxuLHMsYSl7cmV0dXJuIGUuY29uZmlnPXQsbiYmKGUuY29kZT1uKSxlLnJlcXVlc3Q9cyxlLnJlc3BvbnNlPWEsZS5pc0F4aW9zRXJyb3I9ITAsZS50b0pTT049ZnVuY3Rpb24oKXtyZXR1cm57bWVzc2FnZTp0aGlzLm1lc3NhZ2UsbmFtZTp0aGlzLm5hbWUsZGVzY3JpcHRpb246dGhpcy5kZXNjcmlwdGlvbixudW1iZXI6dGhpcy5udW1iZXIsZmlsZU5hbWU6dGhpcy5maWxlTmFtZSxsaW5lTnVtYmVyOnRoaXMubGluZU51bWJlcixjb2x1bW5OdW1iZXI6dGhpcy5jb2x1bW5OdW1iZXIsc3RhY2s6dGhpcy5zdGFjayxjb25maWc6dGhpcy5jb25maWcsY29kZTp0aGlzLmNvZGUsc3RhdHVzOnRoaXMucmVzcG9uc2UmJnRoaXMucmVzcG9uc2Uuc3RhdHVzP3RoaXMucmVzcG9uc2Uuc3RhdHVzOm51bGx9fSxlfSxOdD1IZSxEZT1mdW5jdGlvbihlLHQsbixzLGEpe3ZhciBmPW5ldyBFcnJvcihlKTtyZXR1cm4gTnQoZix0LG4scyxhKX0sTHQ9RGUsVXQ9ZnVuY3Rpb24oZSx0LG4pe3ZhciBzPW4uY29uZmlnLnZhbGlkYXRlU3RhdHVzOyFuLnN0YXR1c3x8IXN8fHMobi5zdGF0dXMpP2Uobik6dChMdCgiUmVxdWVzdCBmYWlsZWQgd2l0aCBzdGF0dXMgY29kZSAiK24uc3RhdHVzLG4uY29uZmlnLG51bGwsbi5yZXF1ZXN0LG4pKX0sWj1QLEJ0PVouaXNTdGFuZGFyZEJyb3dzZXJFbnYoKT9mdW5jdGlvbigpe3JldHVybnt3cml0ZTpmdW5jdGlvbih0LG4scyxhLGYsbCl7dmFyIGg9W107aC5wdXNoKHQrIj0iK2VuY29kZVVSSUNvbXBvbmVudChuKSksWi5pc051bWJlcihzKSYmaC5wdXNoKCJleHBpcmVzPSIrbmV3IERhdGUocykudG9HTVRTdHJpbmcoKSksWi5pc1N0cmluZyhhKSYmaC5wdXNoKCJwYXRoPSIrYSksWi5pc1N0cmluZyhmKSYmaC5wdXNoKCJkb21haW49IitmKSxsPT09ITAmJmgucHVzaCgic2VjdXJlIiksZG9jdW1lbnQuY29va2llPWguam9pbigiOyAiKX0scmVhZDpmdW5jdGlvbih0KXt2YXIgbj1kb2N1bWVudC5jb29raWUubWF0Y2gobmV3IFJlZ0V4cCgiKF58O1xccyopKCIrdCsiKT0oW147XSopIikpO3JldHVybiBuP2RlY29kZVVSSUNvbXBvbmVudChuWzNdKTpudWxsfSxyZW1vdmU6ZnVuY3Rpb24odCl7dGhpcy53cml0ZSh0LCIiLERhdGUubm93KCktODY0ZTUpfX19KCk6ZnVuY3Rpb24oKXtyZXR1cm57d3JpdGU6ZnVuY3Rpb24oKXt9LHJlYWQ6ZnVuY3Rpb24oKXtyZXR1cm4gbnVsbH0scmVtb3ZlOmZ1bmN0aW9uKCl7fX19KCksa3Q9ZnVuY3Rpb24oZSl7cmV0dXJuL14oW2Etel1bYS16XGQrXC0uXSo6KT9cL1wvL2kudGVzdChlKX0sSHQ9ZnVuY3Rpb24oZSx0KXtyZXR1cm4gdD9lLnJlcGxhY2UoL1wvKyQvLCIiKSsiLyIrdC5yZXBsYWNlKC9eXC8rLywiIik6ZX0sRHQ9a3QsJHQ9SHQsV3Q9ZnVuY3Rpb24oZSx0KXtyZXR1cm4gZSYmIUR0KHQpPyR0KGUsdCk6dH0sQ2U9UCxxdD1bImFnZSIsImF1dGhvcml6YXRpb24iLCJjb250ZW50LWxlbmd0aCIsImNvbnRlbnQtdHlwZSIsImV0YWciLCJleHBpcmVzIiwiZnJvbSIsImhvc3QiLCJpZi1tb2RpZmllZC1zaW5jZSIsImlmLXVubW9kaWZpZWQtc2luY2UiLCJsYXN0LW1vZGlmaWVkIiwibG9jYXRpb24iLCJtYXgtZm9yd2FyZHMiLCJwcm94eS1hdXRob3JpemF0aW9uIiwicmVmZXJlciIsInJldHJ5LWFmdGVyIiwidXNlci1hZ2VudCJdLGp0PWZ1bmN0aW9uKGUpe3ZhciB0PXt9LG4scyxhO3JldHVybiBlJiZDZS5mb3JFYWNoKGUuc3BsaXQoYApgKSxmdW5jdGlvbihsKXtpZihhPWwuaW5kZXhPZigiOiIpLG49Q2UudHJpbShsLnN1YnN0cigwLGEpKS50b0xvd2VyQ2FzZSgpLHM9Q2UudHJpbShsLnN1YnN0cihhKzEpKSxuKXtpZih0W25dJiZxdC5pbmRleE9mKG4pPj0wKXJldHVybjtuPT09InNldC1jb29raWUiP3Rbbl09KHRbbl0/dFtuXTpbXSkuY29uY2F0KFtzXSk6dFtuXT10W25dP3Rbbl0rIiwgIitzOnN9fSksdH0sJGU9UCxWdD0kZS5pc1N0YW5kYXJkQnJvd3NlckVudigpP2Z1bmN0aW9uKCl7dmFyIGU9Lyhtc2llfHRyaWRlbnQpL2kudGVzdChuYXZpZ2F0b3IudXNlckFnZW50KSx0PWRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoImEiKSxuO2Z1bmN0aW9uIHMoYSl7dmFyIGY9YTtyZXR1cm4gZSYmKHQuc2V0QXR0cmlidXRlKCJocmVmIixmKSxmPXQuaHJlZiksdC5zZXRBdHRyaWJ1dGUoImhyZWYiLGYpLHtocmVmOnQuaHJlZixwcm90b2NvbDp0LnByb3RvY29sP3QucHJvdG9jb2wucmVwbGFjZSgvOiQvLCIiKToiIixob3N0OnQuaG9zdCxzZWFyY2g6dC5zZWFyY2g/dC5zZWFyY2gucmVwbGFjZSgvXlw/LywiIik6IiIsaGFzaDp0Lmhhc2g/dC5oYXNoLnJlcGxhY2UoL14jLywiIik6IiIsaG9zdG5hbWU6dC5ob3N0bmFtZSxwb3J0OnQucG9ydCxwYXRobmFtZTp0LnBhdGhuYW1lLmNoYXJBdCgwKT09PSIvIj90LnBhdGhuYW1lOiIvIit0LnBhdGhuYW1lfX1yZXR1cm4gbj1zKHdpbmRvdy5sb2NhdGlvbi5ocmVmKSxmdW5jdGlvbihmKXt2YXIgbD0kZS5pc1N0cmluZyhmKT9zKGYpOmY7cmV0dXJuIGwucHJvdG9jb2w9PT1uLnByb3RvY29sJiZsLmhvc3Q9PT1uLmhvc3R9fSgpOmZ1bmN0aW9uKCl7cmV0dXJuIGZ1bmN0aW9uKCl7cmV0dXJuITB9fSgpO2Z1bmN0aW9uIGJlKHIpe3RoaXMubWVzc2FnZT1yfWJlLnByb3RvdHlwZS50b1N0cmluZz1mdW5jdGlvbigpe3JldHVybiJDYW5jZWwiKyh0aGlzLm1lc3NhZ2U/IjogIit0aGlzLm1lc3NhZ2U6IiIpfSxiZS5wcm90b3R5cGUuX19DQU5DRUxfXz0hMDt2YXIgZWU9YmUsdGU9UCx6dD1VdCxHdD1CdCxKdD1rZSxRdD1XdCxYdD1qdCxLdD1WdCxUZT1EZSxZdD1uZSxadD1lZSxXZT1mdW5jdGlvbihlKXtyZXR1cm4gbmV3IFByb21pc2UoZnVuY3Rpb24obixzKXt2YXIgYT1lLmRhdGEsZj1lLmhlYWRlcnMsbD1lLnJlc3BvbnNlVHlwZSxoO2Z1bmN0aW9uIGcoKXtlLmNhbmNlbFRva2VuJiZlLmNhbmNlbFRva2VuLnVuc3Vic2NyaWJlKGgpLGUuc2lnbmFsJiZlLnNpZ25hbC5yZW1vdmVFdmVudExpc3RlbmVyKCJhYm9ydCIsaCl9dGUuaXNGb3JtRGF0YShhKSYmZGVsZXRlIGZbIkNvbnRlbnQtVHlwZSJdO3ZhciBvPW5ldyBYTUxIdHRwUmVxdWVzdDtpZihlLmF1dGgpe3ZhciBwPWUuYXV0aC51c2VybmFtZXx8IiIsUj1lLmF1dGgucGFzc3dvcmQ/dW5lc2NhcGUoZW5jb2RlVVJJQ29tcG9uZW50KGUuYXV0aC5wYXNzd29yZCkpOiIiO2YuQXV0aG9yaXphdGlvbj0iQmFzaWMgIitidG9hKHArIjoiK1IpfXZhciB5PVF0KGUuYmFzZVVSTCxlLnVybCk7by5vcGVuKGUubWV0aG9kLnRvVXBwZXJDYXNlKCksSnQoeSxlLnBhcmFtcyxlLnBhcmFtc1NlcmlhbGl6ZXIpLCEwKSxvLnRpbWVvdXQ9ZS50aW1lb3V0O2Z1bmN0aW9uIFMoKXtpZighIW8pe3ZhciBUPSJnZXRBbGxSZXNwb25zZUhlYWRlcnMiaW4gbz9YdChvLmdldEFsbFJlc3BvbnNlSGVhZGVycygpKTpudWxsLE09IWx8fGw9PT0idGV4dCJ8fGw9PT0ianNvbiI/by5yZXNwb25zZVRleHQ6by5yZXNwb25zZSxBPXtkYXRhOk0sc3RhdHVzOm8uc3RhdHVzLHN0YXR1c1RleHQ6by5zdGF0dXNUZXh0LGhlYWRlcnM6VCxjb25maWc6ZSxyZXF1ZXN0Om99O3p0KGZ1bmN0aW9uKHope24oeiksZygpfSxmdW5jdGlvbih6KXtzKHopLGcoKX0sQSksbz1udWxsfX1pZigib25sb2FkZW5kImluIG8/by5vbmxvYWRlbmQ9UzpvLm9ucmVhZHlzdGF0ZWNoYW5nZT1mdW5jdGlvbigpeyFvfHxvLnJlYWR5U3RhdGUhPT00fHxvLnN0YXR1cz09PTAmJiEoby5yZXNwb25zZVVSTCYmby5yZXNwb25zZVVSTC5pbmRleE9mKCJmaWxlOiIpPT09MCl8fHNldFRpbWVvdXQoUyl9LG8ub25hYm9ydD1mdW5jdGlvbigpeyFvfHwocyhUZSgiUmVxdWVzdCBhYm9ydGVkIixlLCJFQ09OTkFCT1JURUQiLG8pKSxvPW51bGwpfSxvLm9uZXJyb3I9ZnVuY3Rpb24oKXtzKFRlKCJOZXR3b3JrIEVycm9yIixlLG51bGwsbykpLG89bnVsbH0sby5vbnRpbWVvdXQ9ZnVuY3Rpb24oKXt2YXIgTT1lLnRpbWVvdXQ/InRpbWVvdXQgb2YgIitlLnRpbWVvdXQrIm1zIGV4Y2VlZGVkIjoidGltZW91dCBleGNlZWRlZCIsQT1lLnRyYW5zaXRpb25hbHx8WXQudHJhbnNpdGlvbmFsO2UudGltZW91dEVycm9yTWVzc2FnZSYmKE09ZS50aW1lb3V0RXJyb3JNZXNzYWdlKSxzKFRlKE0sZSxBLmNsYXJpZnlUaW1lb3V0RXJyb3I/IkVUSU1FRE9VVCI6IkVDT05OQUJPUlRFRCIsbykpLG89bnVsbH0sdGUuaXNTdGFuZGFyZEJyb3dzZXJFbnYoKSl7dmFyIEk9KGUud2l0aENyZWRlbnRpYWxzfHxLdCh5KSkmJmUueHNyZkNvb2tpZU5hbWU/R3QucmVhZChlLnhzcmZDb29raWVOYW1lKTp2b2lkIDA7SSYmKGZbZS54c3JmSGVhZGVyTmFtZV09SSl9InNldFJlcXVlc3RIZWFkZXIiaW4gbyYmdGUuZm9yRWFjaChmLGZ1bmN0aW9uKE0sQSl7dHlwZW9mIGE9PSJ1bmRlZmluZWQiJiZBLnRvTG93ZXJDYXNlKCk9PT0iY29udGVudC10eXBlIj9kZWxldGUgZltBXTpvLnNldFJlcXVlc3RIZWFkZXIoQSxNKX0pLHRlLmlzVW5kZWZpbmVkKGUud2l0aENyZWRlbnRpYWxzKXx8KG8ud2l0aENyZWRlbnRpYWxzPSEhZS53aXRoQ3JlZGVudGlhbHMpLGwmJmwhPT0ianNvbiImJihvLnJlc3BvbnNlVHlwZT1lLnJlc3BvbnNlVHlwZSksdHlwZW9mIGUub25Eb3dubG9hZFByb2dyZXNzPT0iZnVuY3Rpb24iJiZvLmFkZEV2ZW50TGlzdGVuZXIoInByb2dyZXNzIixlLm9uRG93bmxvYWRQcm9ncmVzcyksdHlwZW9mIGUub25VcGxvYWRQcm9ncmVzcz09ImZ1bmN0aW9uIiYmby51cGxvYWQmJm8udXBsb2FkLmFkZEV2ZW50TGlzdGVuZXIoInByb2dyZXNzIixlLm9uVXBsb2FkUHJvZ3Jlc3MpLChlLmNhbmNlbFRva2VufHxlLnNpZ25hbCkmJihoPWZ1bmN0aW9uKFQpeyFvfHwocyghVHx8VCYmVC50eXBlP25ldyBadCgiY2FuY2VsZWQiKTpUKSxvLmFib3J0KCksbz1udWxsKX0sZS5jYW5jZWxUb2tlbiYmZS5jYW5jZWxUb2tlbi5zdWJzY3JpYmUoaCksZS5zaWduYWwmJihlLnNpZ25hbC5hYm9ydGVkP2goKTplLnNpZ25hbC5hZGRFdmVudExpc3RlbmVyKCJhYm9ydCIsaCkpKSxhfHwoYT1udWxsKSxvLnNlbmQoYSl9KX0sdz1QLHFlPXh0LGVyPUhlLHRyPXsiQ29udGVudC1UeXBlIjoiYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkIn07ZnVuY3Rpb24gamUocixlKXshdy5pc1VuZGVmaW5lZChyKSYmdy5pc1VuZGVmaW5lZChyWyJDb250ZW50LVR5cGUiXSkmJihyWyJDb250ZW50LVR5cGUiXT1lKX1mdW5jdGlvbiBycigpe3ZhciByO3JldHVybih0eXBlb2YgWE1MSHR0cFJlcXVlc3QhPSJ1bmRlZmluZWQifHx0eXBlb2YgcHJvY2VzcyE9InVuZGVmaW5lZCImJk9iamVjdC5wcm90b3R5cGUudG9TdHJpbmcuY2FsbChwcm9jZXNzKT09PSJbb2JqZWN0IHByb2Nlc3NdIikmJihyPVdlKSxyfWZ1bmN0aW9uIG5yKHIsZSx0KXtpZih3LmlzU3RyaW5nKHIpKXRyeXtyZXR1cm4oZXx8SlNPTi5wYXJzZSkociksdy50cmltKHIpfWNhdGNoKG4pe2lmKG4ubmFtZSE9PSJTeW50YXhFcnJvciIpdGhyb3cgbn1yZXR1cm4odHx8SlNPTi5zdHJpbmdpZnkpKHIpfXZhciByZT17dHJhbnNpdGlvbmFsOntzaWxlbnRKU09OUGFyc2luZzohMCxmb3JjZWRKU09OUGFyc2luZzohMCxjbGFyaWZ5VGltZW91dEVycm9yOiExfSxhZGFwdGVyOnJyKCksdHJhbnNmb3JtUmVxdWVzdDpbZnVuY3Rpb24oZSx0KXtyZXR1cm4gcWUodCwiQWNjZXB0IikscWUodCwiQ29udGVudC1UeXBlIiksdy5pc0Zvcm1EYXRhKGUpfHx3LmlzQXJyYXlCdWZmZXIoZSl8fHcuaXNCdWZmZXIoZSl8fHcuaXNTdHJlYW0oZSl8fHcuaXNGaWxlKGUpfHx3LmlzQmxvYihlKT9lOncuaXNBcnJheUJ1ZmZlclZpZXcoZSk/ZS5idWZmZXI6dy5pc1VSTFNlYXJjaFBhcmFtcyhlKT8oamUodCwiYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkO2NoYXJzZXQ9dXRmLTgiKSxlLnRvU3RyaW5nKCkpOncuaXNPYmplY3QoZSl8fHQmJnRbIkNvbnRlbnQtVHlwZSJdPT09ImFwcGxpY2F0aW9uL2pzb24iPyhqZSh0LCJhcHBsaWNhdGlvbi9qc29uIiksbnIoZSkpOmV9XSx0cmFuc2Zvcm1SZXNwb25zZTpbZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcy50cmFuc2l0aW9uYWx8fHJlLnRyYW5zaXRpb25hbCxuPXQmJnQuc2lsZW50SlNPTlBhcnNpbmcscz10JiZ0LmZvcmNlZEpTT05QYXJzaW5nLGE9IW4mJnRoaXMucmVzcG9uc2VUeXBlPT09Impzb24iO2lmKGF8fHMmJncuaXNTdHJpbmcoZSkmJmUubGVuZ3RoKXRyeXtyZXR1cm4gSlNPTi5wYXJzZShlKX1jYXRjaChmKXtpZihhKXRocm93IGYubmFtZT09PSJTeW50YXhFcnJvciI/ZXIoZix0aGlzLCJFX0pTT05fUEFSU0UiKTpmfXJldHVybiBlfV0sdGltZW91dDowLHhzcmZDb29raWVOYW1lOiJYU1JGLVRPS0VOIix4c3JmSGVhZGVyTmFtZToiWC1YU1JGLVRPS0VOIixtYXhDb250ZW50TGVuZ3RoOi0xLG1heEJvZHlMZW5ndGg6LTEsdmFsaWRhdGVTdGF0dXM6ZnVuY3Rpb24oZSl7cmV0dXJuIGU+PTIwMCYmZTwzMDB9LGhlYWRlcnM6e2NvbW1vbjp7QWNjZXB0OiJhcHBsaWNhdGlvbi9qc29uLCB0ZXh0L3BsYWluLCAqLyoifX19O3cuZm9yRWFjaChbImRlbGV0ZSIsImdldCIsImhlYWQiXSxmdW5jdGlvbihlKXtyZS5oZWFkZXJzW2VdPXt9fSksdy5mb3JFYWNoKFsicG9zdCIsInB1dCIsInBhdGNoIl0sZnVuY3Rpb24oZSl7cmUuaGVhZGVyc1tlXT13Lm1lcmdlKHRyKX0pO3ZhciBuZT1yZSxzcj1QLGlyPW5lLG9yPWZ1bmN0aW9uKGUsdCxuKXt2YXIgcz10aGlzfHxpcjtyZXR1cm4gc3IuZm9yRWFjaChuLGZ1bmN0aW9uKGYpe2U9Zi5jYWxsKHMsZSx0KX0pLGV9LFZlPWZ1bmN0aW9uKGUpe3JldHVybiEhKGUmJmUuX19DQU5DRUxfXyl9LHplPVAsRWU9b3IsYXI9VmUsdXI9bmUsZnI9ZWU7ZnVuY3Rpb24gUmUocil7aWYoci5jYW5jZWxUb2tlbiYmci5jYW5jZWxUb2tlbi50aHJvd0lmUmVxdWVzdGVkKCksci5zaWduYWwmJnIuc2lnbmFsLmFib3J0ZWQpdGhyb3cgbmV3IGZyKCJjYW5jZWxlZCIpfXZhciBscj1mdW5jdGlvbihlKXtSZShlKSxlLmhlYWRlcnM9ZS5oZWFkZXJzfHx7fSxlLmRhdGE9RWUuY2FsbChlLGUuZGF0YSxlLmhlYWRlcnMsZS50cmFuc2Zvcm1SZXF1ZXN0KSxlLmhlYWRlcnM9emUubWVyZ2UoZS5oZWFkZXJzLmNvbW1vbnx8e30sZS5oZWFkZXJzW2UubWV0aG9kXXx8e30sZS5oZWFkZXJzKSx6ZS5mb3JFYWNoKFsiZGVsZXRlIiwiZ2V0IiwiaGVhZCIsInBvc3QiLCJwdXQiLCJwYXRjaCIsImNvbW1vbiJdLGZ1bmN0aW9uKHMpe2RlbGV0ZSBlLmhlYWRlcnNbc119KTt2YXIgdD1lLmFkYXB0ZXJ8fHVyLmFkYXB0ZXI7cmV0dXJuIHQoZSkudGhlbihmdW5jdGlvbihzKXtyZXR1cm4gUmUoZSkscy5kYXRhPUVlLmNhbGwoZSxzLmRhdGEscy5oZWFkZXJzLGUudHJhbnNmb3JtUmVzcG9uc2UpLHN9LGZ1bmN0aW9uKHMpe3JldHVybiBhcihzKXx8KFJlKGUpLHMmJnMucmVzcG9uc2UmJihzLnJlc3BvbnNlLmRhdGE9RWUuY2FsbChlLHMucmVzcG9uc2UuZGF0YSxzLnJlc3BvbnNlLmhlYWRlcnMsZS50cmFuc2Zvcm1SZXNwb25zZSkpKSxQcm9taXNlLnJlamVjdChzKX0pfSxPPVAsR2U9ZnVuY3Rpb24oZSx0KXt0PXR8fHt9O3ZhciBuPXt9O2Z1bmN0aW9uIHMobyxwKXtyZXR1cm4gTy5pc1BsYWluT2JqZWN0KG8pJiZPLmlzUGxhaW5PYmplY3QocCk/Ty5tZXJnZShvLHApOk8uaXNQbGFpbk9iamVjdChwKT9PLm1lcmdlKHt9LHApOk8uaXNBcnJheShwKT9wLnNsaWNlKCk6cH1mdW5jdGlvbiBhKG8pe2lmKE8uaXNVbmRlZmluZWQodFtvXSkpe2lmKCFPLmlzVW5kZWZpbmVkKGVbb10pKXJldHVybiBzKHZvaWQgMCxlW29dKX1lbHNlIHJldHVybiBzKGVbb10sdFtvXSl9ZnVuY3Rpb24gZihvKXtpZighTy5pc1VuZGVmaW5lZCh0W29dKSlyZXR1cm4gcyh2b2lkIDAsdFtvXSl9ZnVuY3Rpb24gbChvKXtpZihPLmlzVW5kZWZpbmVkKHRbb10pKXtpZighTy5pc1VuZGVmaW5lZChlW29dKSlyZXR1cm4gcyh2b2lkIDAsZVtvXSl9ZWxzZSByZXR1cm4gcyh2b2lkIDAsdFtvXSl9ZnVuY3Rpb24gaChvKXtpZihvIGluIHQpcmV0dXJuIHMoZVtvXSx0W29dKTtpZihvIGluIGUpcmV0dXJuIHModm9pZCAwLGVbb10pfXZhciBnPXt1cmw6ZixtZXRob2Q6ZixkYXRhOmYsYmFzZVVSTDpsLHRyYW5zZm9ybVJlcXVlc3Q6bCx0cmFuc2Zvcm1SZXNwb25zZTpsLHBhcmFtc1NlcmlhbGl6ZXI6bCx0aW1lb3V0OmwsdGltZW91dE1lc3NhZ2U6bCx3aXRoQ3JlZGVudGlhbHM6bCxhZGFwdGVyOmwscmVzcG9uc2VUeXBlOmwseHNyZkNvb2tpZU5hbWU6bCx4c3JmSGVhZGVyTmFtZTpsLG9uVXBsb2FkUHJvZ3Jlc3M6bCxvbkRvd25sb2FkUHJvZ3Jlc3M6bCxkZWNvbXByZXNzOmwsbWF4Q29udGVudExlbmd0aDpsLG1heEJvZHlMZW5ndGg6bCx0cmFuc3BvcnQ6bCxodHRwQWdlbnQ6bCxodHRwc0FnZW50OmwsY2FuY2VsVG9rZW46bCxzb2NrZXRQYXRoOmwscmVzcG9uc2VFbmNvZGluZzpsLHZhbGlkYXRlU3RhdHVzOmh9O3JldHVybiBPLmZvckVhY2goT2JqZWN0LmtleXMoZSkuY29uY2F0KE9iamVjdC5rZXlzKHQpKSxmdW5jdGlvbihwKXt2YXIgUj1nW3BdfHxhLHk9UihwKTtPLmlzVW5kZWZpbmVkKHkpJiZSIT09aHx8KG5bcF09eSl9KSxufSxKZT17dmVyc2lvbjoiMC4yNS4wIn0sY3I9SmUudmVyc2lvbixQZT17fTtbIm9iamVjdCIsImJvb2xlYW4iLCJudW1iZXIiLCJmdW5jdGlvbiIsInN0cmluZyIsInN5bWJvbCJdLmZvckVhY2goZnVuY3Rpb24ocixlKXtQZVtyXT1mdW5jdGlvbihuKXtyZXR1cm4gdHlwZW9mIG49PT1yfHwiYSIrKGU8MT8ibiAiOiIgIikrcn19KTt2YXIgUWU9e307UGUudHJhbnNpdGlvbmFsPWZ1bmN0aW9uKGUsdCxuKXtmdW5jdGlvbiBzKGEsZil7cmV0dXJuIltBeGlvcyB2IitjcisiXSBUcmFuc2l0aW9uYWwgb3B0aW9uICciK2ErIiciK2YrKG4/Ii4gIituOiIiKX1yZXR1cm4gZnVuY3Rpb24oYSxmLGwpe2lmKGU9PT0hMSl0aHJvdyBuZXcgRXJyb3IocyhmLCIgaGFzIGJlZW4gcmVtb3ZlZCIrKHQ/IiBpbiAiK3Q6IiIpKSk7cmV0dXJuIHQmJiFRZVtmXSYmKFFlW2ZdPSEwLGNvbnNvbGUud2FybihzKGYsIiBoYXMgYmVlbiBkZXByZWNhdGVkIHNpbmNlIHYiK3QrIiBhbmQgd2lsbCBiZSByZW1vdmVkIGluIHRoZSBuZWFyIGZ1dHVyZSIpKSksZT9lKGEsZixsKTohMH19O2Z1bmN0aW9uIGhyKHIsZSx0KXtpZih0eXBlb2YgciE9Im9iamVjdCIpdGhyb3cgbmV3IFR5cGVFcnJvcigib3B0aW9ucyBtdXN0IGJlIGFuIG9iamVjdCIpO2Zvcih2YXIgbj1PYmplY3Qua2V5cyhyKSxzPW4ubGVuZ3RoO3MtLSA+MDspe3ZhciBhPW5bc10sZj1lW2FdO2lmKGYpe3ZhciBsPXJbYV0saD1sPT09dm9pZCAwfHxmKGwsYSxyKTtpZihoIT09ITApdGhyb3cgbmV3IFR5cGVFcnJvcigib3B0aW9uICIrYSsiIG11c3QgYmUgIitoKTtjb250aW51ZX1pZih0IT09ITApdGhyb3cgRXJyb3IoIlVua25vd24gb3B0aW9uICIrYSl9fXZhciBkcj17YXNzZXJ0T3B0aW9uczpocix2YWxpZGF0b3JzOlBlfSxYZT1QLHByPWtlLEtlPUF0LFllPWxyLHNlPUdlLFplPWRyLHE9WmUudmFsaWRhdG9ycztmdW5jdGlvbiBHKHIpe3RoaXMuZGVmYXVsdHM9cix0aGlzLmludGVyY2VwdG9ycz17cmVxdWVzdDpuZXcgS2UscmVzcG9uc2U6bmV3IEtlfX1HLnByb3RvdHlwZS5yZXF1ZXN0PWZ1bmN0aW9uKGUsdCl7aWYodHlwZW9mIGU9PSJzdHJpbmciPyh0PXR8fHt9LHQudXJsPWUpOnQ9ZXx8e30sIXQudXJsKXRocm93IG5ldyBFcnJvcigiUHJvdmlkZWQgY29uZmlnIHVybCBpcyBub3QgdmFsaWQiKTt0PXNlKHRoaXMuZGVmYXVsdHMsdCksdC5tZXRob2Q/dC5tZXRob2Q9dC5tZXRob2QudG9Mb3dlckNhc2UoKTp0aGlzLmRlZmF1bHRzLm1ldGhvZD90Lm1ldGhvZD10aGlzLmRlZmF1bHRzLm1ldGhvZC50b0xvd2VyQ2FzZSgpOnQubWV0aG9kPSJnZXQiO3ZhciBuPXQudHJhbnNpdGlvbmFsO24hPT12b2lkIDAmJlplLmFzc2VydE9wdGlvbnMobix7c2lsZW50SlNPTlBhcnNpbmc6cS50cmFuc2l0aW9uYWwocS5ib29sZWFuKSxmb3JjZWRKU09OUGFyc2luZzpxLnRyYW5zaXRpb25hbChxLmJvb2xlYW4pLGNsYXJpZnlUaW1lb3V0RXJyb3I6cS50cmFuc2l0aW9uYWwocS5ib29sZWFuKX0sITEpO3ZhciBzPVtdLGE9ITA7dGhpcy5pbnRlcmNlcHRvcnMucmVxdWVzdC5mb3JFYWNoKGZ1bmN0aW9uKHkpe3R5cGVvZiB5LnJ1bldoZW49PSJmdW5jdGlvbiImJnkucnVuV2hlbih0KT09PSExfHwoYT1hJiZ5LnN5bmNocm9ub3VzLHMudW5zaGlmdCh5LmZ1bGZpbGxlZCx5LnJlamVjdGVkKSl9KTt2YXIgZj1bXTt0aGlzLmludGVyY2VwdG9ycy5yZXNwb25zZS5mb3JFYWNoKGZ1bmN0aW9uKHkpe2YucHVzaCh5LmZ1bGZpbGxlZCx5LnJlamVjdGVkKX0pO3ZhciBsO2lmKCFhKXt2YXIgaD1bWWUsdm9pZCAwXTtmb3IoQXJyYXkucHJvdG90eXBlLnVuc2hpZnQuYXBwbHkoaCxzKSxoPWguY29uY2F0KGYpLGw9UHJvbWlzZS5yZXNvbHZlKHQpO2gubGVuZ3RoOylsPWwudGhlbihoLnNoaWZ0KCksaC5zaGlmdCgpKTtyZXR1cm4gbH1mb3IodmFyIGc9dDtzLmxlbmd0aDspe3ZhciBvPXMuc2hpZnQoKSxwPXMuc2hpZnQoKTt0cnl7Zz1vKGcpfWNhdGNoKFIpe3AoUik7YnJlYWt9fXRyeXtsPVllKGcpfWNhdGNoKFIpe3JldHVybiBQcm9taXNlLnJlamVjdChSKX1mb3IoO2YubGVuZ3RoOylsPWwudGhlbihmLnNoaWZ0KCksZi5zaGlmdCgpKTtyZXR1cm4gbH0sRy5wcm90b3R5cGUuZ2V0VXJpPWZ1bmN0aW9uKGUpe2lmKCFlLnVybCl0aHJvdyBuZXcgRXJyb3IoIlByb3ZpZGVkIGNvbmZpZyB1cmwgaXMgbm90IHZhbGlkIik7cmV0dXJuIGU9c2UodGhpcy5kZWZhdWx0cyxlKSxwcihlLnVybCxlLnBhcmFtcyxlLnBhcmFtc1NlcmlhbGl6ZXIpLnJlcGxhY2UoL15cPy8sIiIpfSxYZS5mb3JFYWNoKFsiZGVsZXRlIiwiZ2V0IiwiaGVhZCIsIm9wdGlvbnMiXSxmdW5jdGlvbihlKXtHLnByb3RvdHlwZVtlXT1mdW5jdGlvbih0LG4pe3JldHVybiB0aGlzLnJlcXVlc3Qoc2Uobnx8e30se21ldGhvZDplLHVybDp0LGRhdGE6KG58fHt9KS5kYXRhfSkpfX0pLFhlLmZvckVhY2goWyJwb3N0IiwicHV0IiwicGF0Y2giXSxmdW5jdGlvbihlKXtHLnByb3RvdHlwZVtlXT1mdW5jdGlvbih0LG4scyl7cmV0dXJuIHRoaXMucmVxdWVzdChzZShzfHx7fSx7bWV0aG9kOmUsdXJsOnQsZGF0YTpufSkpfX0pO3ZhciBtcj1HLF9yPWVlO2Z1bmN0aW9uIGoocil7aWYodHlwZW9mIHIhPSJmdW5jdGlvbiIpdGhyb3cgbmV3IFR5cGVFcnJvcigiZXhlY3V0b3IgbXVzdCBiZSBhIGZ1bmN0aW9uLiIpO3ZhciBlO3RoaXMucHJvbWlzZT1uZXcgUHJvbWlzZShmdW5jdGlvbihzKXtlPXN9KTt2YXIgdD10aGlzO3RoaXMucHJvbWlzZS50aGVuKGZ1bmN0aW9uKG4pe2lmKCEhdC5fbGlzdGVuZXJzKXt2YXIgcyxhPXQuX2xpc3RlbmVycy5sZW5ndGg7Zm9yKHM9MDtzPGE7cysrKXQuX2xpc3RlbmVyc1tzXShuKTt0Ll9saXN0ZW5lcnM9bnVsbH19KSx0aGlzLnByb21pc2UudGhlbj1mdW5jdGlvbihuKXt2YXIgcyxhPW5ldyBQcm9taXNlKGZ1bmN0aW9uKGYpe3Quc3Vic2NyaWJlKGYpLHM9Zn0pLnRoZW4obik7cmV0dXJuIGEuY2FuY2VsPWZ1bmN0aW9uKCl7dC51bnN1YnNjcmliZShzKX0sYX0scihmdW5jdGlvbihzKXt0LnJlYXNvbnx8KHQucmVhc29uPW5ldyBfcihzKSxlKHQucmVhc29uKSl9KX1qLnByb3RvdHlwZS50aHJvd0lmUmVxdWVzdGVkPWZ1bmN0aW9uKCl7aWYodGhpcy5yZWFzb24pdGhyb3cgdGhpcy5yZWFzb259LGoucHJvdG90eXBlLnN1YnNjcmliZT1mdW5jdGlvbihlKXtpZih0aGlzLnJlYXNvbil7ZSh0aGlzLnJlYXNvbik7cmV0dXJufXRoaXMuX2xpc3RlbmVycz90aGlzLl9saXN0ZW5lcnMucHVzaChlKTp0aGlzLl9saXN0ZW5lcnM9W2VdfSxqLnByb3RvdHlwZS51bnN1YnNjcmliZT1mdW5jdGlvbihlKXtpZighIXRoaXMuX2xpc3RlbmVycyl7dmFyIHQ9dGhpcy5fbGlzdGVuZXJzLmluZGV4T2YoZSk7dCE9PS0xJiZ0aGlzLl9saXN0ZW5lcnMuc3BsaWNlKHQsMSl9fSxqLnNvdXJjZT1mdW5jdGlvbigpe3ZhciBlLHQ9bmV3IGooZnVuY3Rpb24ocyl7ZT1zfSk7cmV0dXJue3Rva2VuOnQsY2FuY2VsOmV9fTt2YXIgZ3I9aix2cj1mdW5jdGlvbihlKXtyZXR1cm4gZnVuY3Rpb24obil7cmV0dXJuIGUuYXBwbHkobnVsbCxuKX19LHlyPVAsd3I9ZnVuY3Rpb24oZSl7cmV0dXJuIHlyLmlzT2JqZWN0KGUpJiZlLmlzQXhpb3NFcnJvcj09PSEwfSxldD1QLENyPXhlLGllPW1yLGJyPUdlLFRyPW5lO2Z1bmN0aW9uIHR0KHIpe3ZhciBlPW5ldyBpZShyKSx0PUNyKGllLnByb3RvdHlwZS5yZXF1ZXN0LGUpO3JldHVybiBldC5leHRlbmQodCxpZS5wcm90b3R5cGUsZSksZXQuZXh0ZW5kKHQsZSksdC5jcmVhdGU9ZnVuY3Rpb24ocyl7cmV0dXJuIHR0KGJyKHIscykpfSx0fXZhciBOPXR0KFRyKTtOLkF4aW9zPWllLE4uQ2FuY2VsPWVlLE4uQ2FuY2VsVG9rZW49Z3IsTi5pc0NhbmNlbD1WZSxOLlZFUlNJT049SmUudmVyc2lvbixOLmFsbD1mdW5jdGlvbihlKXtyZXR1cm4gUHJvbWlzZS5hbGwoZSl9LE4uc3ByZWFkPXZyLE4uaXNBeGlvc0Vycm9yPXdyLF9lLmV4cG9ydHM9TixfZS5leHBvcnRzLmRlZmF1bHQ9Tjt2YXIgRXI9X2UuZXhwb3J0cztjbGFzcyBke2lzRW1wdHkoKXtyZXR1cm4hdGhpcy5pc1ByZXNlbnQoKX1zdGF0aWMgb2YoZSl7aWYoZSE9bnVsbClyZXR1cm4gbmV3IHJ0KGUpO3Rocm93IG5ldyBUeXBlRXJyb3IoIlRoZSBwYXNzZWQgdmFsdWUgd2FzIG51bGwgb3IgdW5kZWZpbmVkLiIpfXN0YXRpYyBvZk5vbk51bGwoZSl7cmV0dXJuIGQub2YoZSl9c3RhdGljIG9mTnVsbGFibGUoZSl7cmV0dXJuIGUhPW51bGw/bmV3IHJ0KGUpOm5ldyBudH1zdGF0aWMgZW1wdHkoKXtyZXR1cm4gbmV3IG50fXN0YXRpYyBmcm9tKGUpe3N3aXRjaChlLmtpbmQpe2Nhc2UicHJlc2VudCI6cmV0dXJuIGQub2YoZS52YWx1ZSk7Y2FzZSJlbXB0eSI6cmV0dXJuIGQuZW1wdHkoKTtkZWZhdWx0OnRocm93IG5ldyBUeXBlRXJyb3IoIlRoZSBwYXNzZWQgdmFsdWUgd2FzIG5vdCBhbiBPcHRpb24gdHlwZS4iKX19fWNsYXNzIHJ0IGV4dGVuZHMgZHtjb25zdHJ1Y3RvcihlKXtzdXBlcigpO3RoaXMucGF5bG9hZD1lfWlzUHJlc2VudCgpe3JldHVybiEwfWdldCgpe3JldHVybiB0aGlzLnBheWxvYWR9aWZQcmVzZW50KGUpe2UodGhpcy5wYXlsb2FkKX1pZlByZXNlbnRPckVsc2UoZSx0KXtlKHRoaXMucGF5bG9hZCl9ZmlsdGVyKGUpe3JldHVybiBlKHRoaXMucGF5bG9hZCk/dGhpczpkLmVtcHR5KCl9bWFwKGUpe2NvbnN0IHQ9ZSh0aGlzLnBheWxvYWQpO3JldHVybiBkLm9mTnVsbGFibGUodCl9ZmxhdE1hcChlKXtyZXR1cm4gZSh0aGlzLnBheWxvYWQpfW9yKGUpe3JldHVybiB0aGlzfW9yRWxzZShlKXtyZXR1cm4gdGhpcy5wYXlsb2FkfW9yRWxzZUdldChlKXtyZXR1cm4gdGhpcy5wYXlsb2FkfW9yRWxzZVRocm93KGUpe3JldHVybiB0aGlzLnBheWxvYWR9b3JOdWxsKCl7cmV0dXJuIHRoaXMucGF5bG9hZH1vclVuZGVmaW5lZCgpe3JldHVybiB0aGlzLnBheWxvYWR9dG9PcHRpb24oKXtyZXR1cm57a2luZDoicHJlc2VudCIsdmFsdWU6dGhpcy5wYXlsb2FkfX1tYXRjaGVzKGUpe3JldHVybiBlLnByZXNlbnQodGhpcy5wYXlsb2FkKX10b0pTT04oZSl7cmV0dXJuIHRoaXMucGF5bG9hZH19Y2xhc3MgbnQgZXh0ZW5kcyBke2lzUHJlc2VudCgpe3JldHVybiExfWNvbnN0cnVjdG9yKCl7c3VwZXIoKX1nZXQoKXt0aHJvdyBuZXcgVHlwZUVycm9yKCJUaGUgb3B0aW9uYWwgaXMgbm90IHByZXNlbnQuIil9aWZQcmVzZW50KGUpe31pZlByZXNlbnRPckVsc2UoZSx0KXt0KCl9ZmlsdGVyKGUpe3JldHVybiB0aGlzfW1hcChlKXtyZXR1cm4gZC5lbXB0eSgpfWZsYXRNYXAoZSl7cmV0dXJuIGQuZW1wdHkoKX1vcihlKXtyZXR1cm4gZSgpfW9yRWxzZShlKXtyZXR1cm4gZX1vckVsc2VHZXQoZSl7cmV0dXJuIHRoaXMub3JFbHNlKGUoKSl9b3JFbHNlVGhyb3coZSl7dGhyb3cgZSgpfW9yTnVsbCgpe3JldHVybiBudWxsfW9yVW5kZWZpbmVkKCl7fXRvT3B0aW9uKCl7cmV0dXJue2tpbmQ6ImVtcHR5In19bWF0Y2hlcyhlKXtyZXR1cm4gZS5lbXB0eSgpfXRvSlNPTihlKXtyZXR1cm4gbnVsbH19Y29uc3QgUnI9IjEuMC4xIjtjbGFzcyBMe2NvbnN0cnVjdG9yKCl7dGhpcy5fcmVwb3J0PXthY3Rpb246ZC5lbXB0eSgpLGFwcGxpY2F0aW9uSWQ6ZC5vZk51bGxhYmxlKG51bGwpLHRpbWVzdGFtcDpEYXRlLm5vdygpLGZwczpkLmVtcHR5KCksZnJhbWVzVHJhbnNmb3JtZWQ6ZC5lbXB0eSgpLGd1aWQ6ZC5lbXB0eSgpLGhpZ2hlc3RGcmFtZVRyYW5zZm9ybUNwdTpkLmVtcHR5KCksbWVzc2FnZTpkLmVtcHR5KCksc291cmNlOmQub2ZOdWxsYWJsZShudWxsKSx0cmFuc2Zvcm1lZEZwczpkLmVtcHR5KCksdHJhbnNmb3JtZXJUeXBlOmQuZW1wdHkoKSx2YXJpYXRpb246ZC5lbXB0eSgpLHZpZGVvSGVpZ2h0OmQuZW1wdHkoKSx2aWRlb1dpZHRoOmQuZW1wdHkoKSx2ZXJzaW9uOlJyfX1hY3Rpb24oZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC5hY3Rpb249ZC5vZk51bGxhYmxlKGUpLHRoaXN9ZnJhbWVzVHJhbnNmb3JtZWQoZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC5mcmFtZXNUcmFuc2Zvcm1lZD1kLm9mTnVsbGFibGUoZSksdGhpc31mcHMoZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC5mcHM9ZC5vZk51bGxhYmxlKGUpLHRoaXN9Z3VpZChlKXtyZXR1cm4gdGhpcy5fcmVwb3J0Lmd1aWQ9ZC5vZk51bGxhYmxlKGUpLHRoaXN9bWVzc2FnZShlKXtyZXR1cm4gdGhpcy5fcmVwb3J0Lm1lc3NhZ2U9ZC5vZk51bGxhYmxlKGUpLHRoaXN9dHJhbnNmb3JtZWRGcHMoZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC50cmFuc2Zvcm1lZEZwcz1kLm9mTnVsbGFibGUoZSksdGhpc310cmFuc2Zvcm1lclR5cGUoZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC50cmFuc2Zvcm1lclR5cGU9ZC5vZk51bGxhYmxlKGUpLHRoaXN9dmFyaWF0aW9uKGUpe3JldHVybiB0aGlzLl9yZXBvcnQudmFyaWF0aW9uPWQub2ZOdWxsYWJsZShlKSx0aGlzfXZpZGVvSGVpZ2h0KGUpe3JldHVybiB0aGlzLl9yZXBvcnQudmlkZW9IZWlnaHQ9ZC5vZk51bGxhYmxlKGUpLHRoaXN9dmlkZW9XaWR0aChlKXtyZXR1cm4gdGhpcy5fcmVwb3J0LnZpZGVvV2lkdGg9ZC5vZk51bGxhYmxlKGUpLHRoaXN9YnVpbGQoKXtyZXR1cm4gdGhpcy5fcmVwb3J0fX1jb25zdCBQcj1yPT5KU09OLnN0cmluZ2lmeShyLChlLHQpPT57aWYodCE9PW51bGwpcmV0dXJuIHR9KTtjbGFzcyBVe3N0YXRpYyByZXBvcnQoZSl7cmV0dXJuIG5ldyBQcm9taXNlKCh0LG4pPT57bGV0IHM9RXIuY3JlYXRlKCksYT17dGltZW91dDoxZTQsdGltZW91dEVycm9yTWVzc2FnZToiUmVxdWVzdCB0aW1lb3V0IixoZWFkZXJzOnsiQ29udGVudC1UeXBlIjoiYXBwbGljYXRpb24vanNvbiJ9fTtjb25zdCBmPSJodHRwczovL2hsZy5kZXYudG9rYm94LmNvbS9kZXYvbG9nZ2luZy92Y3Bfd2VicnRjIjtzLnBvc3QoZixQcihlKSxhKS50aGVuKGw9Pntjb25zb2xlLmxvZyhsKSx0KCJzdWNjZXNzIil9KS5jYXRjaChsPT57Y29uc29sZS5sb2cobCksbihsKX0pfSl9fXZhciBvZSxPcj1uZXcgVWludDhBcnJheSgxNik7ZnVuY3Rpb24gU3IoKXtpZighb2UmJihvZT10eXBlb2YgY3J5cHRvIT0idW5kZWZpbmVkIiYmY3J5cHRvLmdldFJhbmRvbVZhbHVlcyYmY3J5cHRvLmdldFJhbmRvbVZhbHVlcy5iaW5kKGNyeXB0byl8fHR5cGVvZiBtc0NyeXB0byE9InVuZGVmaW5lZCImJnR5cGVvZiBtc0NyeXB0by5nZXRSYW5kb21WYWx1ZXM9PSJmdW5jdGlvbiImJm1zQ3J5cHRvLmdldFJhbmRvbVZhbHVlcy5iaW5kKG1zQ3J5cHRvKSwhb2UpKXRocm93IG5ldyBFcnJvcigiY3J5cHRvLmdldFJhbmRvbVZhbHVlcygpIG5vdCBzdXBwb3J0ZWQuIFNlZSBodHRwczovL2dpdGh1Yi5jb20vdXVpZGpzL3V1aWQjZ2V0cmFuZG9tdmFsdWVzLW5vdC1zdXBwb3J0ZWQiKTtyZXR1cm4gb2UoT3IpfXZhciBJcj0vXig/OlswLTlhLWZdezh9LVswLTlhLWZdezR9LVsxLTVdWzAtOWEtZl17M30tWzg5YWJdWzAtOWEtZl17M30tWzAtOWEtZl17MTJ9fDAwMDAwMDAwLTAwMDAtMDAwMC0wMDAwLTAwMDAwMDAwMDAwMCkkL2k7ZnVuY3Rpb24gTXIocil7cmV0dXJuIHR5cGVvZiByPT0ic3RyaW5nIiYmSXIudGVzdChyKX1mb3IodmFyIEM9W10sT2U9MDtPZTwyNTY7KytPZSlDLnB1c2goKE9lKzI1NikudG9TdHJpbmcoMTYpLnN1YnN0cigxKSk7ZnVuY3Rpb24gQXIocil7dmFyIGU9YXJndW1lbnRzLmxlbmd0aD4xJiZhcmd1bWVudHNbMV0hPT12b2lkIDA/YXJndW1lbnRzWzFdOjAsdD0oQ1tyW2UrMF1dK0NbcltlKzFdXStDW3JbZSsyXV0rQ1tyW2UrM11dKyItIitDW3JbZSs0XV0rQ1tyW2UrNV1dKyItIitDW3JbZSs2XV0rQ1tyW2UrN11dKyItIitDW3JbZSs4XV0rQ1tyW2UrOV1dKyItIitDW3JbZSsxMF1dK0NbcltlKzExXV0rQ1tyW2UrMTJdXStDW3JbZSsxM11dK0NbcltlKzE0XV0rQ1tyW2UrMTVdXSkudG9Mb3dlckNhc2UoKTtpZighTXIodCkpdGhyb3cgVHlwZUVycm9yKCJTdHJpbmdpZmllZCBVVUlEIGlzIGludmFsaWQiKTtyZXR1cm4gdH1mdW5jdGlvbiBzdChyLGUsdCl7cj1yfHx7fTt2YXIgbj1yLnJhbmRvbXx8KHIucm5nfHxTcikoKTtpZihuWzZdPW5bNl0mMTV8NjQsbls4XT1uWzhdJjYzfDEyOCxlKXt0PXR8fDA7Zm9yKHZhciBzPTA7czwxNjsrK3MpZVt0K3NdPW5bc107cmV0dXJuIGV9cmV0dXJuIEFyKG4pfWNvbnN0IEZyPTUwMDtjbGFzcyB4cntjb25zdHJ1Y3RvcihlKXt0aGlzLnV1aWRfPXN0KCksdGhpcy5mcmFtZXNUcmFuc2Zvcm1lZF89MCx0aGlzLnRyYW5zZm9ybWVyXz1lLHRoaXMuc2hvdWxkU3RvcF89ITEsdGhpcy5pc0ZsYXNoZWRfPSExLHRoaXMuZnJhbWVzRnJvbVNvdXJjZV89MCx0aGlzLmZwc189MCx0aGlzLm1lZGlhVHJhbnNmb3JtZXJRb3NSZXBvcnRTdGFydFRpbWVzdGFtcF89MCx0aGlzLnZpZGVvSGVpZ2h0Xz0wLHRoaXMudmlkZW9XaWR0aF89MCx0aGlzLnRyYW5zZm9ybWVyVHlwZV89IkN1c3RvbSIsImdldFRyYW5zZm9ybWVyVHlwZSJpbiBlJiYodGhpcy50cmFuc2Zvcm1lclR5cGVfPWUuZ2V0VHJhbnNmb3JtZXJUeXBlKCkpO2NvbnN0IHQ9bmV3IEwoKS5hY3Rpb24oIk1lZGlhVHJhbnNmb3JtZXIiKS5ndWlkKHRoaXMudXVpZF8pLnRyYW5zZm9ybWVyVHlwZSh0aGlzLnRyYW5zZm9ybWVyVHlwZV8pLnZhcmlhdGlvbigiQ3JlYXRlIikuYnVpbGQoKTtVLnJlcG9ydCh0KX1hc3luYyBzdGFydChlKXtpZih0aGlzLnRyYW5zZm9ybWVyXyYmdHlwZW9mIHRoaXMudHJhbnNmb3JtZXJfLnN0YXJ0PT0iZnVuY3Rpb24iKXRyeXthd2FpdCB0aGlzLnRyYW5zZm9ybWVyXy5zdGFydChlKX1jYXRjaHtjb25zdCBuPW5ldyBMKCkuYWN0aW9uKCJNZWRpYVRyYW5zZm9ybWVyIikuZ3VpZCh0aGlzLnV1aWRfKS5tZXNzYWdlKEQuZXJyb3JzLnRyYW5zZm9ybWVyX3N0YXJ0KS50cmFuc2Zvcm1lclR5cGUodGhpcy50cmFuc2Zvcm1lclR5cGVfKS52YXJpYXRpb24oIkVycm9yIikuYnVpbGQoKTtVLnJlcG9ydChuKX19YXN5bmMgdHJhbnNmb3JtKGUsdCl7dmFyIG4scyxhLGY7aWYodGhpcy5tZWRpYVRyYW5zZm9ybWVyUW9zUmVwb3J0U3RhcnRUaW1lc3RhbXBfPT09MCYmKHRoaXMubWVkaWFUcmFuc2Zvcm1lclFvc1JlcG9ydFN0YXJ0VGltZXN0YW1wXz1EYXRlLm5vdygpKSx0aGlzLnZpZGVvSGVpZ2h0Xz0obj1lPT1udWxsP3ZvaWQgMDplLmRpc3BsYXlIZWlnaHQpIT1udWxsP246MCx0aGlzLnZpZGVvV2lkdGhfPShzPWU9PW51bGw/dm9pZCAwOmUuZGlzcGxheVdpZHRoKSE9bnVsbD9zOjAsKyt0aGlzLmZyYW1lc0Zyb21Tb3VyY2VfLHRoaXMudHJhbnNmb3JtZXJfKWlmKHRoaXMuc2hvdWxkU3RvcF8pZS5jbG9zZSgpLHRoaXMuZmx1c2godCksdC50ZXJtaW5hdGUoKTtlbHNlIHRyeXthd2FpdCgoZj0oYT10aGlzLnRyYW5zZm9ybWVyXykudHJhbnNmb3JtKT09bnVsbD92b2lkIDA6Zi5jYWxsKGEsZSx0KSksKyt0aGlzLmZyYW1lc1RyYW5zZm9ybWVkXyx0aGlzLmZyYW1lc1RyYW5zZm9ybWVkXz09PUZyJiZ0aGlzLm1lZGlhVHJhbnNmb3JtZXJRb3NSZXBvcnQoKX1jYXRjaHtjb25zdCBoPW5ldyBMKCkuYWN0aW9uKCJNZWRpYVRyYW5zZm9ybWVyIikuZ3VpZCh0aGlzLnV1aWRfKS5tZXNzYWdlKEQuZXJyb3JzLnRyYW5zZm9ybWVyX3RyYW5zZm9ybSkudHJhbnNmb3JtZXJUeXBlKHRoaXMudHJhbnNmb3JtZXJUeXBlXykudmFyaWF0aW9uKCJFcnJvciIpLmJ1aWxkKCk7VS5yZXBvcnQoaCl9fWFzeW5jIGZsdXNoKGUpe2lmKHRoaXMudHJhbnNmb3JtZXJfJiZ0eXBlb2YgdGhpcy50cmFuc2Zvcm1lcl8uZmx1c2g9PSJmdW5jdGlvbiImJiF0aGlzLmlzRmxhc2hlZF8pe3RoaXMuaXNGbGFzaGVkXz0hMDt0cnl7YXdhaXQgdGhpcy50cmFuc2Zvcm1lcl8uZmx1c2goZSl9Y2F0Y2h7Y29uc3Qgcz1uZXcgTCgpLmFjdGlvbigiTWVkaWFUcmFuc2Zvcm1lciIpLmd1aWQodGhpcy51dWlkXykubWVzc2FnZShELmVycm9ycy50cmFuc2Zvcm1lcl90cmFuc2Zvcm0pLnRyYW5zZm9ybWVyVHlwZSh0aGlzLnRyYW5zZm9ybWVyVHlwZV8pLnZhcmlhdGlvbigiRXJyb3IiKS5idWlsZCgpO1UucmVwb3J0KHMpfX10aGlzLm1lZGlhVHJhbnNmb3JtZXJRb3NSZXBvcnQoKTtjb25zdCB0PW5ldyBMKCkuYWN0aW9uKCJNZWRpYVRyYW5zZm9ybWVyIikuZ3VpZCh0aGlzLnV1aWRfKS50cmFuc2Zvcm1lclR5cGUodGhpcy50cmFuc2Zvcm1lclR5cGVfKS52YXJpYXRpb24oIkRlbGV0ZSIpLmJ1aWxkKCk7VS5yZXBvcnQodCl9c3RvcCgpe2NvbnNvbGUubG9nKCJbUGlwZWxpbmVdIFN0b3Agc3RyZWFtLiIpLHRoaXMuc2hvdWxkU3RvcF89ITB9bWVkaWFUcmFuc2Zvcm1lclFvc1JlcG9ydCgpe2xldCBlPShEYXRlLm5vdygpLXRoaXMubWVkaWFUcmFuc2Zvcm1lclFvc1JlcG9ydFN0YXJ0VGltZXN0YW1wXykvMWUzLHQ9dGhpcy5mcmFtZXNGcm9tU291cmNlXy9lLG49dGhpcy5mcmFtZXNUcmFuc2Zvcm1lZF8vZTtjb25zdCBzPW5ldyBMKCkuYWN0aW9uKCJNZWRpYVRyYW5zZm9ybWVyIikuZnBzKHQpLnRyYW5zZm9ybWVkRnBzKG4pLmZyYW1lc1RyYW5zZm9ybWVkKHRoaXMuZnJhbWVzVHJhbnNmb3JtZWRfKS5ndWlkKHRoaXMudXVpZF8pLnRyYW5zZm9ybWVyVHlwZSh0aGlzLnRyYW5zZm9ybWVyVHlwZV8pLnZpZGVvSGVpZ2h0KHRoaXMudmlkZW9IZWlnaHRfKS52aWRlb1dpZHRoKHRoaXMudmlkZW9XaWR0aF8pLnZhcmlhdGlvbigiUW9TIikuYnVpbGQoKTtVLnJlcG9ydChzKSx0aGlzLm1lZGlhVHJhbnNmb3JtZXJRb3NSZXBvcnRTdGFydFRpbWVzdGFtcF89MCx0aGlzLmZyYW1lc0Zyb21Tb3VyY2VfPTAsdGhpcy5mcmFtZXNUcmFuc2Zvcm1lZF89MH19Y2xhc3MgTnJ7Y29uc3RydWN0b3IoZSl7dGhpcy50cmFuc2Zvcm1lcnNfPVtdO2ZvcihsZXQgdCBvZiBlKXRoaXMudHJhbnNmb3JtZXJzXy5wdXNoKG5ldyB4cih0KSl9YXN5bmMgc3RhcnQoZSx0KXtpZighdGhpcy50cmFuc2Zvcm1lcnNffHx0aGlzLnRyYW5zZm9ybWVyc18ubGVuZ3RoPT09MCl7Y29uc29sZS5sb2coIltQaXBlbGluZV0gTm8gdHJhbnNmb3JtZXJzLiIpO3JldHVybn10cnl7bGV0IG49ZTtmb3IobGV0IHMgb2YgdGhpcy50cmFuc2Zvcm1lcnNfKWU9ZS5waXBlVGhyb3VnaChuZXcgVHJhbnNmb3JtU3RyZWFtKHMpKTtlLnBpcGVUbyh0KS50aGVuKGFzeW5jKCk9Pntjb25zb2xlLmxvZygiW1BpcGVsaW5lXSBTZXR1cC4iKSxhd2FpdCB0LmFib3J0KCksYXdhaXQgbi5jYW5jZWwoKX0pLmNhdGNoKGFzeW5jIHM9PntlLmNhbmNlbCgpLnRoZW4oKCk9Pntjb25zb2xlLmxvZygiW1BpcGVsaW5lXSBTaHV0dGluZyBkb3duIHN0cmVhbXMgYWZ0ZXIgYWJvcnQuIil9KS5jYXRjaChhPT57Y29uc29sZS5lcnJvcigiW1BpcGVsaW5lXSBFcnJvciBmcm9tIHN0cmVhbSB0cmFuc2Zvcm06IixhKX0pLGF3YWl0IHQuYWJvcnQocyksYXdhaXQgbi5jYW5jZWwocyl9KX1jYXRjaHt0aGlzLmRlc3Ryb3koKTtyZXR1cm59Y29uc29sZS5sb2coIltQaXBlbGluZV0gUGlwZWxpbmUgc3RhcnRlZC4iKX1hc3luYyBkZXN0cm95KCl7Y29uc29sZS5sb2coIltQaXBlbGluZV0gRGVzdHJveWluZyBQaXBlbGluZS4iKTtmb3IobGV0IGUgb2YgdGhpcy50cmFuc2Zvcm1lcnNfKWUuc3RvcCgpfX1jbGFzcyBMcntjb25zdHJ1Y3Rvcigpe3RoaXMudXVpZF89c3QoKTtjb25zdCBlPW5ldyBMKCkuYWN0aW9uKCJNZWRpYVByb2Nlc3NvciIpLmd1aWQodGhpcy51dWlkXykudmFyaWF0aW9uKCJDcmVhdGUiKS5idWlsZCgpO1UucmVwb3J0KGUpfXRyYW5zZm9ybShlLHQpe3JldHVybiB0aGlzLnJlYWRhYmxlXz1lLHRoaXMud3JpdGFibGVfPXQsdGhpcy50cmFuc2Zvcm1JbnRlcm5hbCgpfXRyYW5zZm9ybUludGVybmFsKCl7cmV0dXJuIG5ldyBQcm9taXNlKChlLHQpPT57aWYoIXRoaXMudHJhbnNmb3JtZXJzX3x8dGhpcy50cmFuc2Zvcm1lcnNfLmxlbmd0aD09PTApe2NvbnN0IG49bmV3IEwoKS5hY3Rpb24oIk1lZGlhUHJvY2Vzc29yIikuZ3VpZCh0aGlzLnV1aWRfKS5tZXNzYWdlKEQuZXJyb3JzLnRyYW5zZm9ybWVyX25vbmUpLnZhcmlhdGlvbigiRXJyb3IiKS5idWlsZCgpO1UucmVwb3J0KG4pLHQoIltNZWRpYVByb2Nlc3Nvcl0gTmVlZCB0byBzZXQgdHJhbnNmb3JtZXJzLiIpfXRoaXMucGlwZWxpbmVfJiZ0aGlzLnBpcGVsaW5lXy5kZXN0cm95KCksdGhpcy5waXBlbGluZV89bmV3IE5yKHRoaXMudHJhbnNmb3JtZXJzXyksdGhpcy5waXBlbGluZV8uc3RhcnQodGhpcy5yZWFkYWJsZV8sdGhpcy53cml0YWJsZV8pLnRoZW4oKCk9PntlKCl9KS5jYXRjaChuPT57dChuKX0pfSl9c2V0VHJhbnNmb3JtZXJzKGUpe2NvbnN0IHQ9bmV3IEwoKS5hY3Rpb24oIk1lZGlhUHJvY2Vzc29yIikuZ3VpZCh0aGlzLnV1aWRfKS5tZXNzYWdlKEQudXBkYXRlcy50cmFuc2Zvcm1lcl9uZXcpLnZhcmlhdGlvbigiVXBkYXRlIikuYnVpbGQoKTtyZXR1cm4gVS5yZXBvcnQodCksdGhpcy50cmFuc2Zvcm1lcnNfPWUsdGhpcy5yZWFkYWJsZV8mJnRoaXMud3JpdGFibGVfP3RoaXMudHJhbnNmb3JtSW50ZXJuYWwoKTpQcm9taXNlLnJlc29sdmUoKX1kZXN0cm95KCl7cmV0dXJuIG5ldyBQcm9taXNlKGU9Pnt0aGlzLnBpcGVsaW5lXyYmdGhpcy5waXBlbGluZV8uZGVzdHJveSgpO2NvbnN0IHQ9bmV3IEwoKS5hY3Rpb24oIk1lZGlhUHJvY2Vzc29yIikuZ3VpZCh0aGlzLnV1aWRfKS52YXJpYXRpb24oIkRlbGV0ZSIpLmJ1aWxkKCk7VS5yZXBvcnQodCksZSgpfSl9fXZhciBhZTsoZnVuY3Rpb24ocil7ci5Mb3c9IkxvdyIsci5IaWdoPSJIaWdoIn0pKGFlfHwoYWU9e30pKTtjbGFzcyBpdHtjb25zdHJ1Y3RvcihlKXt0aGlzLmJsdXJGaWx0ZXJfPWU9PT1hZS5IaWdoPyJibHVyKDE1cHgpIjoiYmx1cig1cHgpIn1ydW5Qb3N0UHJvY2Vzc2luZyhlLHQpe3RyeXtlLmdsb2JhbENvbXBvc2l0ZU9wZXJhdGlvbj0iZGVzdGluYXRpb24tb3ZlciIsZS5maWx0ZXI9dGhpcy5ibHVyRmlsdGVyXyxlLmRyYXdJbWFnZSh0LDAsMCx0LndpZHRoLHQuaGVpZ2h0KX1jYXRjaChuKXtjb25zb2xlLmxvZygiW0JsdXJQb3N0UHJvY2Vzc10gRmFpbGVkIHRvIGRyYXcgY2FudmFzIixuKX19fWNsYXNzIFVye2NvbnN0cnVjdG9yKGUpe2lmKHRoaXMuaW1hZ2VDYW52YXNfPW5ldyBPZmZzY3JlZW5DYW52YXMoMSwxKSx0aGlzLmltYWdlQ3R4Xz10aGlzLmltYWdlQ2FudmFzXy5nZXRDb250ZXh0KCIyZCIse2FscGhhOiExLHdpbGxSZWFkRnJlcXVlbnRseTohMCxkZXN5bmNocm9uaXplZDohMH0pLCF0aGlzLmltYWdlQ3R4Xyl0aHJvdyBuZXcgRXJyb3IoIlVuYWJsZSB0byBjcmVhdGUgT2Zmc2NyZWVuQ2FudmFzUmVuZGVyaW5nQ29udGV4dDJEIik7dGhpcy5pbWFnZUJsb2JfPXZvaWQgMCxmZXRjaChlKS50aGVuKHQ9Pnt0LmJsb2IoKS50aGVuKG49Pnt0aGlzLmltYWdlQmxvYl89bn0pLmNhdGNoKG49Pntjb25zb2xlLmVycm9yKCJibG9iIGVycm9yIixuKX0pfSkuY2F0Y2godD0+e2NvbnNvbGUuZXJyb3IoImZldGNoIGVycm9yIix0KX0pfXJ1blBvc3RQcm9jZXNzaW5nKGUsdCl7dHJ5e2lmKHR5cGVvZiB0aGlzLmltYWdlQmxvYl8hPSJ1bmRlZmluZWQiJiYodGhpcy5pbWFnZUNhbnZhc18ud2lkdGghPXQud2lkdGh8fHRoaXMuaW1hZ2VDYW52YXNfLmhlaWdodCE9dC5oZWlnaHQpKXt0aGlzLmltYWdlQ2FudmFzXy53aWR0aD10LndpZHRoLHRoaXMuaW1hZ2VDYW52YXNfLmhlaWdodD10LmhlaWdodDtjb25zdCBuPSJoaWdoIixzPXtyZXNpemVXaWR0aDp0LndpZHRoLHJlc2l6ZUhlaWdodDp0LmhlaWdodCxyZXNpemVRdWFsaXR5Om59LGE9Y3JlYXRlSW1hZ2VCaXRtYXAodGhpcy5pbWFnZUJsb2JfLHMpLnRoZW4oZj0+e3RoaXMuaW1hZ2VDdHhfLmRyYXdJbWFnZShmLDAsMCl9KS5jYXRjaChmPT57Y29uc29sZS5lcnJvcigiZXJyb3IgY3JlYXRlSW1hZ2VCaXRtYXAiLGYpfSl9ZS5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb249ImRlc3RpbmF0aW9uLW92ZXIiLGUuZHJhd0ltYWdlKHRoaXMuaW1hZ2VDYW52YXNfLDAsMCx0aGlzLmltYWdlQ2FudmFzXy53aWR0aCx0aGlzLmltYWdlQ2FudmFzXy5oZWlnaHQpfWNhdGNoKG4pe2NvbnNvbGUubG9nKCJbVmlydHVhbFBvc3RQcm9jZXNzXSBGYWlsZWQgdG8gZHJhdyBjYW52YXMiLG4pfX19Y2xhc3MgQnJ7Y29uc3RydWN0b3IoZT0xLzApe3RoaXMuY2FwYWNpdHk9ZSx0aGlzLnN0b3JhZ2U9W119ZW5xdWV1ZShlKXtpZih0aGlzLnNpemUoKT09PXRoaXMuY2FwYWNpdHkpdGhyb3cgRXJyb3IoIlF1ZXVlIGhhcyByZWFjaGVkIG1heCBjYXBhY2l0eSwgeW91IGNhbm5vdCBhZGQgbW9yZSBpdGVtcyIpO3RoaXMuc3RvcmFnZS5wdXNoKGUpfWRlcXVldWUoKXtyZXR1cm4gdGhpcy5zdG9yYWdlLnNoaWZ0KCl9c2l6ZSgpe3JldHVybiB0aGlzLnN0b3JhZ2UubGVuZ3RofX1jbGFzcyBrcntjb25zdHJ1Y3RvcihlKXtjb25zb2xlLmxvZygiW1ZpZGVvUG9zdFByb2Nlc3NdIEN0b3IiKSx0aGlzLnF1ZXVlXz1uZXcgQnIsZSYmKHRoaXMuZnJhbWVSZWFkZXJfPWUuZ2V0UmVhZGVyKCksdGhpcy5mcmFtZVJlYWRlcl8ucmVhZCgpLnRoZW4odGhpcy5wcm9jZXNzRnJhbWUuYmluZCh0aGlzKSkuY2F0Y2godD0+Y29uc29sZS5lcnJvcih0KSkpfXByb2Nlc3NGcmFtZShlKXtlLmRvbmV8fCh0aGlzLnF1ZXVlXy5lbnF1ZXVlKGUudmFsdWUpLHRoaXMuZnJhbWVSZWFkZXJfLnJlYWQoKS50aGVuKHRoaXMucHJvY2Vzc0ZyYW1lLmJpbmQodGhpcykpLmNhdGNoKHQ9PmNvbnNvbGUuZXJyb3IodCkpKX1ydW5Qb3N0UHJvY2Vzc2luZyhlLHQpe3RyeXtpZihlLmdsb2JhbENvbXBvc2l0ZU9wZXJhdGlvbj0iZGVzdGluYXRpb24tb3ZlciIsdGhpcy5xdWV1ZV8uc2l6ZSgpPjApe2NvbnN0IG49dGhpcy5xdWV1ZV8uZGVxdWV1ZSgpO2UuZHJhd0ltYWdlKG4sMCwwLHQud2lkdGgsdC5oZWlnaHQpLG4uY2xvc2UoKX1lbHNlIGUuZHJhd0ltYWdlKHQsMCwwLHQud2lkdGgsdC5oZWlnaHQpfWNhdGNoKG4pe2NvbnNvbGUubG9nKCJbVmlkZW9Qb3N0UHJvY2Vzc10gRmFpbGVkIHRvIGRyYXcgY2FudmFzIixuKX19fWZ1bmN0aW9uIEhyKCl7cmV0dXJuInNlbGZpZV9zZWdtZW50YXRpb25fbGFuZHNjYXBlIn12YXIgdWU9ZnVuY3Rpb24oKXt2YXIgcj10eXBlb2YgZG9jdW1lbnQhPSJ1bmRlZmluZWQiJiZkb2N1bWVudC5jdXJyZW50U2NyaXB0P2RvY3VtZW50LmN1cnJlbnRTY3JpcHQuc3JjOnZvaWQgMDtyZXR1cm4gZnVuY3Rpb24oZSl7ZT1lfHx7fTt2YXIgdD10eXBlb2YgZSE9InVuZGVmaW5lZCI/ZTp7fSxuLHM7dC5yZWFkeT1uZXcgUHJvbWlzZShmdW5jdGlvbihpLHUpe249aSxzPXV9KTt2YXIgYT17fSxmO2ZvcihmIGluIHQpdC5oYXNPd25Qcm9wZXJ0eShmKSYmKGFbZl09dFtmXSk7dmFyIGw9Ii4vdGhpcy5wcm9ncmFtIixoPWZ1bmN0aW9uKGksdSl7dGhyb3cgdX0sZz0hMCxvPSIiO2Z1bmN0aW9uIHAoaSl7cmV0dXJuIHQubG9jYXRlRmlsZT90LmxvY2F0ZUZpbGUoaSxvKTpvK2l9dmFyIFI7dHlwZW9mIGRvY3VtZW50IT0idW5kZWZpbmVkIiYmZG9jdW1lbnQuY3VycmVudFNjcmlwdCYmKG89ZG9jdW1lbnQuY3VycmVudFNjcmlwdC5zcmMpLHImJihvPXIpLG8uaW5kZXhPZigiYmxvYjoiKSE9PTA/bz1vLnN1YnN0cigwLG8ucmVwbGFjZSgvWz8jXS4qLywiIikubGFzdEluZGV4T2YoIi8iKSsxKTpvPSIiO3ZhciB5PXQucHJpbnR8fGNvbnNvbGUubG9nLmJpbmQoY29uc29sZSksUz10LnByaW50RXJyfHxjb25zb2xlLndhcm4uYmluZChjb25zb2xlKTtmb3IoZiBpbiBhKWEuaGFzT3duUHJvcGVydHkoZikmJih0W2ZdPWFbZl0pO2E9bnVsbCx0LmFyZ3VtZW50cyx0LnRoaXNQcm9ncmFtJiYobD10LnRoaXNQcm9ncmFtKSx0LnF1aXQmJihoPXQucXVpdCk7dmFyIEk7dC53YXNtQmluYXJ5JiYoST10Lndhc21CaW5hcnkpO3ZhciBUPXQubm9FeGl0UnVudGltZXx8ITA7dHlwZW9mIFdlYkFzc2VtYmx5IT0ib2JqZWN0IiYmSCgibm8gbmF0aXZlIHdhc20gc3VwcG9ydCBkZXRlY3RlZCIpO3ZhciBNLEE9ITEsZmU9dHlwZW9mIFRleHREZWNvZGVyIT0idW5kZWZpbmVkIj9uZXcgVGV4dERlY29kZXIoInV0ZjgiKTp2b2lkIDA7ZnVuY3Rpb24geihpLHUsYyl7Zm9yKHZhciBtPXUrYyx2PXU7aVt2XSYmISh2Pj1tKTspKyt2O2lmKHYtdT4xNiYmaS5zdWJhcnJheSYmZmUpcmV0dXJuIGZlLmRlY29kZShpLnN1YmFycmF5KHUsdikpO2Zvcih2YXIgYj0iIjt1PHY7KXt2YXIgXz1pW3UrK107aWYoIShfJjEyOCkpe2IrPVN0cmluZy5mcm9tQ2hhckNvZGUoXyk7Y29udGludWV9dmFyIEU9aVt1KytdJjYzO2lmKChfJjIyNCk9PTE5Mil7Yis9U3RyaW5nLmZyb21DaGFyQ29kZSgoXyYzMSk8PDZ8RSk7Y29udGludWV9dmFyIEI9aVt1KytdJjYzO2lmKChfJjI0MCk9PTIyND9fPShfJjE1KTw8MTJ8RTw8NnxCOl89KF8mNyk8PDE4fEU8PDEyfEI8PDZ8aVt1KytdJjYzLF88NjU1MzYpYis9U3RyaW5nLmZyb21DaGFyQ29kZShfKTtlbHNle3ZhciBwdD1fLTY1NTM2O2IrPVN0cmluZy5mcm9tQ2hhckNvZGUoNTUyOTZ8cHQ+PjEwLDU2MzIwfHB0JjEwMjMpfX1yZXR1cm4gYn1mdW5jdGlvbiBsZShpLHUpe3JldHVybiBpP3ooSixpLHUpOiIifWZ1bmN0aW9uIHFyKGksdSxjKXtmb3IodmFyIG09MDttPGkubGVuZ3RoOysrbSljZVt1Kys+PjBdPWkuY2hhckNvZGVBdChtKTtjfHwoY2VbdT4+MF09MCl9ZnVuY3Rpb24ganIoaSx1KXtyZXR1cm4gaSV1PjAmJihpKz11LWkldSksaX12YXIgb3QsY2UsSixGO2Z1bmN0aW9uIGF0KGkpe290PWksdC5IRUFQOD1jZT1uZXcgSW50OEFycmF5KGkpLHQuSEVBUDE2PW5ldyBJbnQxNkFycmF5KGkpLHQuSEVBUDMyPUY9bmV3IEludDMyQXJyYXkoaSksdC5IRUFQVTg9Sj1uZXcgVWludDhBcnJheShpKSx0LkhFQVBVMTY9bmV3IFVpbnQxNkFycmF5KGkpLHQuSEVBUFUzMj1uZXcgVWludDMyQXJyYXkoaSksdC5IRUFQRjMyPW5ldyBGbG9hdDMyQXJyYXkoaSksdC5IRUFQRjY0PW5ldyBGbG9hdDY0QXJyYXkoaSl9dC5JTklUSUFMX01FTU9SWTt2YXIgSWUsdXQ9W10sZnQ9W10sbHQ9W10sVnI9MDtmdW5jdGlvbiB6cigpe3JldHVybiBUfHxWcj4wfWZ1bmN0aW9uIEdyKCl7aWYodC5wcmVSdW4pZm9yKHR5cGVvZiB0LnByZVJ1bj09ImZ1bmN0aW9uIiYmKHQucHJlUnVuPVt0LnByZVJ1bl0pO3QucHJlUnVuLmxlbmd0aDspWHIodC5wcmVSdW4uc2hpZnQoKSk7TWUodXQpfWZ1bmN0aW9uIEpyKCl7TWUoZnQpfWZ1bmN0aW9uIFFyKCl7aWYodC5wb3N0UnVuKWZvcih0eXBlb2YgdC5wb3N0UnVuPT0iZnVuY3Rpb24iJiYodC5wb3N0UnVuPVt0LnBvc3RSdW5dKTt0LnBvc3RSdW4ubGVuZ3RoOylZcih0LnBvc3RSdW4uc2hpZnQoKSk7TWUobHQpfWZ1bmN0aW9uIFhyKGkpe3V0LnVuc2hpZnQoaSl9ZnVuY3Rpb24gS3IoaSl7ZnQudW5zaGlmdChpKX1mdW5jdGlvbiBZcihpKXtsdC51bnNoaWZ0KGkpfXZhciAkPTAsUT1udWxsO2Z1bmN0aW9uIFpyKGkpeyQrKyx0Lm1vbml0b3JSdW5EZXBlbmRlbmNpZXMmJnQubW9uaXRvclJ1bkRlcGVuZGVuY2llcygkKX1mdW5jdGlvbiBlbihpKXtpZigkLS0sdC5tb25pdG9yUnVuRGVwZW5kZW5jaWVzJiZ0Lm1vbml0b3JSdW5EZXBlbmRlbmNpZXMoJCksJD09MCYmUSl7dmFyIHU9UTtRPW51bGwsdSgpfX10LnByZWxvYWRlZEltYWdlcz17fSx0LnByZWxvYWRlZEF1ZGlvcz17fTtmdW5jdGlvbiBIKGkpe3Qub25BYm9ydCYmdC5vbkFib3J0KGkpLGk9IkFib3J0ZWQoIitpKyIpIixTKGkpLEE9ITAsaSs9Ii4gQnVpbGQgd2l0aCAtcyBBU1NFUlRJT05TPTEgZm9yIG1vcmUgaW5mby4iO3ZhciB1PW5ldyBXZWJBc3NlbWJseS5SdW50aW1lRXJyb3IoaSk7dGhyb3cgcyh1KSx1fXZhciB0bj0iZGF0YTphcHBsaWNhdGlvbi9vY3RldC1zdHJlYW07YmFzZTY0LCI7ZnVuY3Rpb24gY3QoaSl7cmV0dXJuIGkuc3RhcnRzV2l0aCh0bil9dmFyIHg7eD0idGZsaXRlLXNpbWQud2FzbSIsY3QoeCl8fCh4PXAoeCkpO2Z1bmN0aW9uIGh0KGkpe3RyeXtpZihpPT14JiZJKXJldHVybiBuZXcgVWludDhBcnJheShJKTtpZighUil0aHJvdyJib3RoIGFzeW5jIGFuZCBzeW5jIGZldGNoaW5nIG9mIHRoZSB3YXNtIGZhaWxlZCJ9Y2F0Y2godSl7SCh1KX19ZnVuY3Rpb24gcm4oKXtyZXR1cm4hSSYmZyYmdHlwZW9mIGZldGNoPT0iZnVuY3Rpb24iP2ZldGNoKHgse2NyZWRlbnRpYWxzOiJzYW1lLW9yaWdpbiJ9KS50aGVuKGZ1bmN0aW9uKGkpe2lmKCFpLm9rKXRocm93ImZhaWxlZCB0byBsb2FkIHdhc20gYmluYXJ5IGZpbGUgYXQgJyIreCsiJyI7cmV0dXJuIGkuYXJyYXlCdWZmZXIoKX0pLmNhdGNoKGZ1bmN0aW9uKCl7cmV0dXJuIGh0KHgpfSk6UHJvbWlzZS5yZXNvbHZlKCkudGhlbihmdW5jdGlvbigpe3JldHVybiBodCh4KX0pfWZ1bmN0aW9uIG5uKCl7dmFyIGk9e2E6Um59O2Z1bmN0aW9uIHUoXyxFKXt2YXIgQj1fLmV4cG9ydHM7dC5hc209QixNPXQuYXNtLnEsYXQoTS5idWZmZXIpLEllPXQuYXNtLkQsS3IodC5hc20uciksZW4oKX1acigpO2Z1bmN0aW9uIGMoXyl7dShfLmluc3RhbmNlKX1mdW5jdGlvbiBtKF8pe3JldHVybiBybigpLnRoZW4oZnVuY3Rpb24oRSl7cmV0dXJuIFdlYkFzc2VtYmx5Lmluc3RhbnRpYXRlKEUsaSl9KS50aGVuKGZ1bmN0aW9uKEUpe3JldHVybiBFfSkudGhlbihfLGZ1bmN0aW9uKEUpe1MoImZhaWxlZCB0byBhc3luY2hyb25vdXNseSBwcmVwYXJlIHdhc206ICIrRSksSChFKX0pfWZ1bmN0aW9uIHYoKXtyZXR1cm4hSSYmdHlwZW9mIFdlYkFzc2VtYmx5Lmluc3RhbnRpYXRlU3RyZWFtaW5nPT0iZnVuY3Rpb24iJiYhY3QoeCkmJnR5cGVvZiBmZXRjaD09ImZ1bmN0aW9uIj9mZXRjaCh4LHtjcmVkZW50aWFsczoic2FtZS1vcmlnaW4ifSkudGhlbihmdW5jdGlvbihfKXt2YXIgRT1XZWJBc3NlbWJseS5pbnN0YW50aWF0ZVN0cmVhbWluZyhfLGkpO3JldHVybiBFLnRoZW4oYyxmdW5jdGlvbihCKXtyZXR1cm4gUygid2FzbSBzdHJlYW1pbmcgY29tcGlsZSBmYWlsZWQ6ICIrQiksUygiZmFsbGluZyBiYWNrIHRvIEFycmF5QnVmZmVyIGluc3RhbnRpYXRpb24iKSxtKGMpfSl9KTptKGMpfWlmKHQuaW5zdGFudGlhdGVXYXNtKXRyeXt2YXIgYj10Lmluc3RhbnRpYXRlV2FzbShpLHUpO3JldHVybiBifWNhdGNoKF8pe3JldHVybiBTKCJNb2R1bGUuaW5zdGFudGlhdGVXYXNtIGNhbGxiYWNrIGZhaWxlZCB3aXRoIGVycm9yOiAiK18pLCExfXJldHVybiB2KCkuY2F0Y2gocykse319ZnVuY3Rpb24gTWUoaSl7Zm9yKDtpLmxlbmd0aD4wOyl7dmFyIHU9aS5zaGlmdCgpO2lmKHR5cGVvZiB1PT0iZnVuY3Rpb24iKXt1KHQpO2NvbnRpbnVlfXZhciBjPXUuZnVuYzt0eXBlb2YgYz09Im51bWJlciI/dS5hcmc9PT12b2lkIDA/SWUuZ2V0KGMpKCk6SWUuZ2V0KGMpKHUuYXJnKTpjKHUuYXJnPT09dm9pZCAwP251bGw6dS5hcmcpfX1mdW5jdGlvbiBzbihpLHUsYyxtKXtIKCJBc3NlcnRpb24gZmFpbGVkOiAiK2xlKGkpKyIsIGF0OiAiK1t1P2xlKHUpOiJ1bmtub3duIGZpbGVuYW1lIixjLG0/bGUobSk6InVua25vd24gZnVuY3Rpb24iXSl9ZnVuY3Rpb24gb24oaSx1KXtIKCJUbyB1c2UgZGxvcGVuLCB5b3UgbmVlZCB0byB1c2UgRW1zY3JpcHRlbidzIGxpbmtpbmcgc3VwcG9ydCwgc2VlIGh0dHBzOi8vZ2l0aHViLmNvbS9lbXNjcmlwdGVuLWNvcmUvZW1zY3JpcHRlbi93aWtpL0xpbmtpbmciKX1mdW5jdGlvbiBhbihpLHUpe0goIlRvIHVzZSBkbG9wZW4sIHlvdSBuZWVkIHRvIHVzZSBFbXNjcmlwdGVuJ3MgbGlua2luZyBzdXBwb3J0LCBzZWUgaHR0cHM6Ly9naXRodWIuY29tL2Vtc2NyaXB0ZW4tY29yZS9lbXNjcmlwdGVuL3dpa2kvTGlua2luZyIpfWZ1bmN0aW9uIHVuKCl7SCgiIil9dmFyIGhlO2hlPWZ1bmN0aW9uKCl7cmV0dXJuIHBlcmZvcm1hbmNlLm5vdygpfTt2YXIgZm49ITA7ZnVuY3Rpb24gbG4oaSl7cmV0dXJuIEZbZHQoKT4+Ml09aSxpfWZ1bmN0aW9uIGNuKGksdSl7dmFyIGM7aWYoaT09PTApYz1EYXRlLm5vdygpO2Vsc2UgaWYoKGk9PT0xfHxpPT09NCkmJmZuKWM9aGUoKTtlbHNlIHJldHVybiBsbigyOCksLTE7cmV0dXJuIEZbdT4+Ml09Yy8xZTN8MCxGW3UrND4+Ml09YyUxZTMqMWUzKjFlM3wwLDB9ZnVuY3Rpb24gaG4oKXtyZXR1cm4gMjE0NzQ4MzY0OH1mdW5jdGlvbiBkbihpLHUsYyl7Si5jb3B5V2l0aGluKGksdSx1K2MpfWZ1bmN0aW9uIHBuKGkpe3RyeXtyZXR1cm4gTS5ncm93KGktb3QuYnl0ZUxlbmd0aCs2NTUzNT4+PjE2KSxhdChNLmJ1ZmZlciksMX1jYXRjaHt9fWZ1bmN0aW9uIG1uKGkpe3ZhciB1PUoubGVuZ3RoO2k9aT4+PjA7dmFyIGM9MjE0NzQ4MzY0ODtpZihpPmMpcmV0dXJuITE7Zm9yKHZhciBtPTE7bTw9NDttKj0yKXt2YXIgdj11KigxKy4yL20pO3Y9TWF0aC5taW4odixpKzEwMDY2MzI5Nik7dmFyIGI9TWF0aC5taW4oYyxqcihNYXRoLm1heChpLHYpLDY1NTM2KSksXz1wbihiKTtpZihfKXJldHVybiEwfXJldHVybiExfWZ1bmN0aW9uIF9uKGkpe2Zvcih2YXIgdT1oZSgpO2hlKCktdTxpOyk7fXZhciBBZT17fTtmdW5jdGlvbiBnbigpe3JldHVybiBsfHwiLi90aGlzLnByb2dyYW0ifWZ1bmN0aW9uIFgoKXtpZighWC5zdHJpbmdzKXt2YXIgaT0odHlwZW9mIG5hdmlnYXRvcj09Im9iamVjdCImJm5hdmlnYXRvci5sYW5ndWFnZXMmJm5hdmlnYXRvci5sYW5ndWFnZXNbMF18fCJDIikucmVwbGFjZSgiLSIsIl8iKSsiLlVURi04Iix1PXtVU0VSOiJ3ZWJfdXNlciIsTE9HTkFNRToid2ViX3VzZXIiLFBBVEg6Ii8iLFBXRDoiLyIsSE9NRToiL2hvbWUvd2ViX3VzZXIiLExBTkc6aSxfOmduKCl9O2Zvcih2YXIgYyBpbiBBZSlBZVtjXT09PXZvaWQgMD9kZWxldGUgdVtjXTp1W2NdPUFlW2NdO3ZhciBtPVtdO2Zvcih2YXIgYyBpbiB1KW0ucHVzaChjKyI9Iit1W2NdKTtYLnN0cmluZ3M9bX1yZXR1cm4gWC5zdHJpbmdzfXZhciBkZT17bWFwcGluZ3M6e30sYnVmZmVyczpbbnVsbCxbXSxbXV0scHJpbnRDaGFyOmZ1bmN0aW9uKGksdSl7dmFyIGM9ZGUuYnVmZmVyc1tpXTt1PT09MHx8dT09PTEwPygoaT09PTE/eTpTKSh6KGMsMCkpLGMubGVuZ3RoPTApOmMucHVzaCh1KX0sdmFyYXJnczp2b2lkIDAsZ2V0OmZ1bmN0aW9uKCl7ZGUudmFyYXJncys9NDt2YXIgaT1GW2RlLnZhcmFyZ3MtND4+Ml07cmV0dXJuIGl9LGdldFN0cjpmdW5jdGlvbihpKXt2YXIgdT1sZShpKTtyZXR1cm4gdX0sZ2V0NjQ6ZnVuY3Rpb24oaSx1KXtyZXR1cm4gaX19O2Z1bmN0aW9uIHZuKGksdSl7dmFyIGM9MDtyZXR1cm4gWCgpLmZvckVhY2goZnVuY3Rpb24obSx2KXt2YXIgYj11K2M7RltpK3YqND4+Ml09YixxcihtLGIpLGMrPW0ubGVuZ3RoKzF9KSwwfWZ1bmN0aW9uIHluKGksdSl7dmFyIGM9WCgpO0ZbaT4+Ml09Yy5sZW5ndGg7dmFyIG09MDtyZXR1cm4gYy5mb3JFYWNoKGZ1bmN0aW9uKHYpe20rPXYubGVuZ3RoKzF9KSxGW3U+PjJdPW0sMH1mdW5jdGlvbiB3bihpKXtPbihpKX1mdW5jdGlvbiBDbihpKXtyZXR1cm4gMH1mdW5jdGlvbiBibihpLHUsYyxtLHYpe31mdW5jdGlvbiBUbihpLHUsYyxtKXtmb3IodmFyIHY9MCxiPTA7YjxjO2IrKyl7Zm9yKHZhciBfPUZbdStiKjg+PjJdLEU9Rlt1KyhiKjgrNCk+PjJdLEI9MDtCPEU7QisrKWRlLnByaW50Q2hhcihpLEpbXytCXSk7dis9RX1yZXR1cm4gRlttPj4yXT12LDB9ZnVuY3Rpb24gRW4oKXtpZih0eXBlb2YgY3J5cHRvPT0ib2JqZWN0IiYmdHlwZW9mIGNyeXB0by5nZXRSYW5kb21WYWx1ZXM9PSJmdW5jdGlvbiIpe3ZhciBpPW5ldyBVaW50OEFycmF5KDEpO3JldHVybiBmdW5jdGlvbigpe3JldHVybiBjcnlwdG8uZ2V0UmFuZG9tVmFsdWVzKGkpLGlbMF19fWVsc2UgcmV0dXJuIGZ1bmN0aW9uKCl7SCgicmFuZG9tRGV2aWNlIil9fWZ1bmN0aW9uIHBlKGksdSl7cGUucmFuZG9tRGV2aWNlfHwocGUucmFuZG9tRGV2aWNlPUVuKCkpO2Zvcih2YXIgYz0wO2M8dTtjKyspY2VbaStjPj4wXT1wZS5yYW5kb21EZXZpY2UoKTtyZXR1cm4gMH12YXIgUm49e2E6c24sZTpvbixkOmFuLGI6dW4sbjpjbixoOmhuLGw6ZG4sbTptbixwOl9uLGY6dm4sZzp5bixqOnduLGk6Q24sazpibixjOlRuLG86cGV9O25uKCksdC5fX193YXNtX2NhbGxfY3RvcnM9ZnVuY3Rpb24oKXtyZXR1cm4odC5fX193YXNtX2NhbGxfY3RvcnM9dC5hc20ucikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRNb2RlbEJ1ZmZlck1lbW9yeU9mZnNldD1mdW5jdGlvbigpe3JldHVybih0Ll9nZXRNb2RlbEJ1ZmZlck1lbW9yeU9mZnNldD10LmFzbS5zKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHQuX2dldElucHV0TWVtb3J5T2Zmc2V0PWZ1bmN0aW9uKCl7cmV0dXJuKHQuX2dldElucHV0TWVtb3J5T2Zmc2V0PXQuYXNtLnQpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sdC5fZ2V0SW5wdXRIZWlnaHQ9ZnVuY3Rpb24oKXtyZXR1cm4odC5fZ2V0SW5wdXRIZWlnaHQ9dC5hc20udSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRJbnB1dFdpZHRoPWZ1bmN0aW9uKCl7cmV0dXJuKHQuX2dldElucHV0V2lkdGg9dC5hc20udikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRJbnB1dENoYW5uZWxDb3VudD1mdW5jdGlvbigpe3JldHVybih0Ll9nZXRJbnB1dENoYW5uZWxDb3VudD10LmFzbS53KS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHQuX2dldE91dHB1dE1lbW9yeU9mZnNldD1mdW5jdGlvbigpe3JldHVybih0Ll9nZXRPdXRwdXRNZW1vcnlPZmZzZXQ9dC5hc20ueCkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRPdXRwdXRIZWlnaHQ9ZnVuY3Rpb24oKXtyZXR1cm4odC5fZ2V0T3V0cHV0SGVpZ2h0PXQuYXNtLnkpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sdC5fZ2V0T3V0cHV0V2lkdGg9ZnVuY3Rpb24oKXtyZXR1cm4odC5fZ2V0T3V0cHV0V2lkdGg9dC5hc20ueikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRPdXRwdXRDaGFubmVsQ291bnQ9ZnVuY3Rpb24oKXtyZXR1cm4odC5fZ2V0T3V0cHV0Q2hhbm5lbENvdW50PXQuYXNtLkEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sdC5fbG9hZE1vZGVsPWZ1bmN0aW9uKCl7cmV0dXJuKHQuX2xvYWRNb2RlbD10LmFzbS5CKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHQuX3J1bkluZmVyZW5jZT1mdW5jdGlvbigpe3JldHVybih0Ll9ydW5JbmZlcmVuY2U9dC5hc20uQykuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgZHQ9dC5fX19lcnJub19sb2NhdGlvbj1mdW5jdGlvbigpe3JldHVybihkdD10Ll9fX2Vycm5vX2xvY2F0aW9uPXQuYXNtLkUpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sbWU7ZnVuY3Rpb24gUG4oaSl7dGhpcy5uYW1lPSJFeGl0U3RhdHVzIix0aGlzLm1lc3NhZ2U9IlByb2dyYW0gdGVybWluYXRlZCB3aXRoIGV4aXQoIitpKyIpIix0aGlzLnN0YXR1cz1pfVE9ZnVuY3Rpb24gaSgpe21lfHxGZSgpLG1lfHwoUT1pKX07ZnVuY3Rpb24gRmUoaSl7aWYoJD4wfHwoR3IoKSwkPjApKXJldHVybjtmdW5jdGlvbiB1KCl7bWV8fChtZT0hMCx0LmNhbGxlZFJ1bj0hMCwhQSYmKEpyKCksbih0KSx0Lm9uUnVudGltZUluaXRpYWxpemVkJiZ0Lm9uUnVudGltZUluaXRpYWxpemVkKCksUXIoKSkpfXQuc2V0U3RhdHVzPyh0LnNldFN0YXR1cygiUnVubmluZy4uLiIpLHNldFRpbWVvdXQoZnVuY3Rpb24oKXtzZXRUaW1lb3V0KGZ1bmN0aW9uKCl7dC5zZXRTdGF0dXMoIiIpfSwxKSx1KCl9LDEpKTp1KCl9dC5ydW49RmU7ZnVuY3Rpb24gT24oaSx1KXtTbihpKX1mdW5jdGlvbiBTbihpKXt6cigpfHwodC5vbkV4aXQmJnQub25FeGl0KGkpLEE9ITApLGgoaSxuZXcgUG4oaSkpfWlmKHQucHJlSW5pdClmb3IodHlwZW9mIHQucHJlSW5pdD09ImZ1bmN0aW9uIiYmKHQucHJlSW5pdD1bdC5wcmVJbml0XSk7dC5wcmVJbml0Lmxlbmd0aD4wOyl0LnByZUluaXQucG9wKCkoKTtyZXR1cm4gRmUoKSxlLnJlYWR5fX0oKTt0eXBlb2YgZXhwb3J0cz09Im9iamVjdCImJnR5cGVvZiBtb2R1bGU9PSJvYmplY3QiP21vZHVsZS5leHBvcnRzPXVlOnR5cGVvZiBkZWZpbmU9PSJmdW5jdGlvbiImJmRlZmluZS5hbWQ/ZGVmaW5lKFtdLGZ1bmN0aW9uKCl7cmV0dXJuIHVlfSk6dHlwZW9mIGV4cG9ydHM9PSJvYmplY3QiJiYoZXhwb3J0cy5jcmVhdGVWb25hZ2VURkxpdGVTaW1kTW9kdWxlPXVlKTtjbGFzcyBEcntjb25zdHJ1Y3Rvcigpe3RoaXMuaXNTSU1EU3VwcG9ydGVkXz0hMSx0aGlzLmlzVGhyZWFkc1N1cHBvcnRlZF89ITEsdGhpcy5pbnB1dFdpZHRoXz0wLHRoaXMuaW5wdXRIZWlnaHRfPTAsdGhpcy5pbnB1dENoYW5uZWxDb3VudF89MCx0aGlzLmlucHV0TWVtb3J5T2Zmc2V0Xz0wLHRoaXMub3V0cHV0V2lkdGhfPTAsdGhpcy5vdXRwdXRIZWlnaHRfPTAsdGhpcy5vdXRwdXRDaGFubmVsQ291bnRfPTAsdGhpcy5vdXRwdXRNZW1vcnlPZmZzZXRfPTB9fWFzeW5jIGZ1bmN0aW9uICRyKHIpe2xldCBlPW5ldyBEcjt2YXIgdD1uZXcgUHJvbWlzZShmdW5jdGlvbihuLHMpe2FzeW5jIGZ1bmN0aW9uIGEoKXtyZXR1cm4gdWUoe2xvY2F0ZUZpbGU6ZnVuY3Rpb24oaCl7cmV0dXJuIHIraH19KX1hc3luYyBmdW5jdGlvbiBmKGwpe3JldHVybiBuZXcgUHJvbWlzZShhc3luYyhoLGcpPT57Y29uc3Qgbz1sO2lmKHR5cGVvZiBvPT0idW5kZWZpbmVkIilyZXR1cm4gZygiVEZMaXRlIGJhY2tlbmQgdW5hdmFpbGFibGU6IHdhc21TaW1kIik7Y29uc3QgcD1IcigpO2NvbnNvbGUubG9nKCJMb2FkaW5nIHRmbGl0ZSBtb2RlbDoiLHApO2xldCBSPXIrYG1vZGVscy8ke3B9LnRmbGl0ZWA7Y29uc29sZS5sb2coIlRGTGl0ZSBtb2RlbFVybDogIitSKTt0cnl7Y29uc3QgUz1hd2FpdChhd2FpdCBmZXRjaChSKSkuYXJyYXlCdWZmZXIoKTtjb25zb2xlLmxvZygiTW9kZWwgYnVmZmVyIHNpemU6IixTLmJ5dGVMZW5ndGgpO2NvbnN0IEk9by5fZ2V0TW9kZWxCdWZmZXJNZW1vcnlPZmZzZXQoKTtjb25zb2xlLmxvZygiTW9kZWwgYnVmZmVyIG1lbW9yeSBvZmZzZXQ6IixJKSxjb25zb2xlLmxvZygiTG9hZGluZyBtb2RlbCBidWZmZXIuLi4iKSxvLkhFQVBVOC5zZXQobmV3IFVpbnQ4QXJyYXkoUyksSSk7Y29uc3QgVD1vLl9sb2FkTW9kZWwoUy5ieXRlTGVuZ3RoKTtpZihjb25zb2xlLmxvZygiTG9hZCBtb2RlbCByZXN1bHQ6IixUKSxUIT09MCl7ZygiQ2Fubm90IGxvYWQgbW9kZWwiKTtyZXR1cm59ZS5pbnB1dFdpZHRoXz1vLl9nZXRJbnB1dFdpZHRoKCksZS5pbnB1dEhlaWdodF89by5fZ2V0SW5wdXRIZWlnaHQoKSxlLmlucHV0Q2hhbm5lbENvdW50Xz1vLl9nZXRJbnB1dENoYW5uZWxDb3VudCgpLGUuaW5wdXRNZW1vcnlPZmZzZXRfPW8uX2dldElucHV0TWVtb3J5T2Zmc2V0KCksY29uc29sZS5sb2coIklucHV0IG1lbW9yeSBvZmZzZXQ6IixlLmlucHV0TWVtb3J5T2Zmc2V0XyksY29uc29sZS5sb2coIklucHV0IGhlaWdodDoiLGUuaW5wdXRIZWlnaHRfKSxjb25zb2xlLmxvZygiSW5wdXQgd2lkdGg6IixlLmlucHV0V2lkdGhfKSxjb25zb2xlLmxvZygiSW5wdXQgY2hhbm5lbHM6IixlLmlucHV0Q2hhbm5lbENvdW50XyksZS5vdXRwdXRNZW1vcnlPZmZzZXRfPW8uX2dldE91dHB1dE1lbW9yeU9mZnNldCgpLGUub3V0cHV0V2lkdGhfPW8uX2dldE91dHB1dFdpZHRoKCksZS5vdXRwdXRIZWlnaHRfPW8uX2dldE91dHB1dEhlaWdodCgpLGUub3V0cHV0Q2hhbm5lbENvdW50Xz1vLl9nZXRPdXRwdXRDaGFubmVsQ291bnQoKSxjb25zb2xlLmxvZygiT3V0cHV0IG1lbW9yeSBvZmZzZXQ6IixlLm91dHB1dE1lbW9yeU9mZnNldF8pLGNvbnNvbGUubG9nKCJPdXRwdXQgaGVpZ2h0OiIsZS5vdXRwdXRIZWlnaHRfKSxjb25zb2xlLmxvZygiT3V0cHV0IHdpZHRoOiIsZS5vdXRwdXRXaWR0aF8pLGNvbnNvbGUubG9nKCJPdXRwdXQgY2hhbm5lbHM6IixlLm91dHB1dENoYW5uZWxDb3VudF8pLGUuc2VsZWN0ZWRURkxpdGVfPW99Y2F0Y2goeSl7Zyh5KTtyZXR1cm59aCgpfSl9YSgpLnRoZW4obD0+e2lmKGw9PT1udWxsKXRocm93IGNvbnNvbGUuZXJyb3IoImxvYWRURkxpdGUgbnVsbCIpLCJsb2FkVEZMaXRlIG51bGwiO2YobCkudGhlbihoPT57bihlKX0pLmNhdGNoKGg9Pntjb25zb2xlLmVycm9yKCJsb2FkVEZMaXRlTW9kZWwgZXJyb3IiKSxzKCJsb2FkVEZMaXRlTW9kZWwgZXJyb3IiKX0pfSkuY2F0Y2gobD0+e2NvbnNvbGUuZXJyb3IoImxvYWRURkxpdGUgZXJyb3I6IixsKSxzKCJsb2FkVEZMaXRlIGVycm9yOiIrbCl9KX0pO3JldHVybiB0fWNsYXNzIFdye2NvbnN0cnVjdG9yKGUpe3RoaXMuYmx1ckZpbHRlcl89ZT09PWFlLkhpZ2g/ImJsdXIoMTVweCkiOiJibHVyKDVweCkifXJ1blBvc3RQcm9jZXNzaW5nKGUsdCl7dHJ5e2UuZ2xvYmFsQ29tcG9zaXRlT3BlcmF0aW9uPSJzb3VyY2UtaW4iLGUuZmlsdGVyPXRoaXMuYmx1ckZpbHRlcl8sZS5kcmF3SW1hZ2UodCwwLDAsdC53aWR0aCx0LmhlaWdodCksZS5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb249ImRlc3RpbmF0aW9uLW92ZXIiLGUuZmlsdGVyPSJibHVyKDBweCkiLGUuZHJhd0ltYWdlKHQsMCwwLHQud2lkdGgsdC5oZWlnaHQpfWNhdGNoKG4pe2NvbnNvbGUubG9nKCJbQmx1clBvc3RQcm9jZXNzXSBGYWlsZWQgdG8gZHJhdyBjYW52YXMiLG4pfX19Y29uc3QgVj1jbGFzc3tjb25zdHJ1Y3RvcihyKXtpZih0aGlzLmNvbmZpZ189cix0aGlzLmZyYW1lQ291bnRlcl89MCx0aGlzLmxhc3RQcm9jZXNzZWRUaW1lc3RhbXBfPTAsdGhpcy5yZXN1bHRDYW52YXNfPW5ldyBPZmZzY3JlZW5DYW52YXMoMSwxKSx0aGlzLnJlc3VsdEN0eF89dGhpcy5yZXN1bHRDYW52YXNfLmdldENvbnRleHQoIjJkIix7YWxwaGE6ITEsZGVzeW5jaHJvbml6ZWQ6ITB9KSwhdGhpcy5yZXN1bHRDdHhfKXRocm93IG5ldyBFcnJvcigiVW5hYmxlIHRvIGNyZWF0ZSBPZmZzY3JlZW5DYW52YXNSZW5kZXJpbmdDb250ZXh0MkQiKTtpZih0aGlzLnNlZ21lbnRhdGlvbk1hc2tDYW52YXNfPW5ldyBPZmZzY3JlZW5DYW52YXMoMSwxKSx0aGlzLnNlZ21lbnRhdGlvbk1hc2tDdHhfPXRoaXMuc2VnbWVudGF0aW9uTWFza0NhbnZhc18uZ2V0Q29udGV4dCgiMmQiLHthbHBoYTohMSxkZXN5bmNocm9uaXplZDohMH0pLCF0aGlzLnNlZ21lbnRhdGlvbk1hc2tDdHhfKXRocm93IG5ldyBFcnJvcigiVW5hYmxlIHRvIGNyZWF0ZSBPZmZzY3JlZW5DYW52YXNSZW5kZXJpbmdDb250ZXh0MkQiKTtpZih0aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUNhbnZhc189bmV3IE9mZnNjcmVlbkNhbnZhcygxLDEpLHRoaXMuc2VnbWVudGF0aW9uTWFza0ZyYW1lQ3R4Xz10aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUNhbnZhc18uZ2V0Q29udGV4dCgiMmQiLHthbHBoYTohMSxkZXN5bmNocm9uaXplZDohMH0pLCF0aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUN0eF8pdGhyb3cgbmV3IEVycm9yKCJVbmFibGUgdG8gY3JlYXRlIE9mZnNjcmVlbkNhbnZhc1JlbmRlcmluZ0NvbnRleHQyRCIpO3RoaXMuc2VnbWVudGF0aW9uUGl4ZWxDb3VudF89MCx0aGlzLmlucHV0TWVtb3J5T2Zmc2V0Xz0wLHRoaXMub3V0cHV0TWVtb3J5T2Zmc2V0Xz0wLHIudHJhbnNmb3JtZXJUeXBlPT09IlZpcnR1YWxCYWNrZ3JvdW5kIj8odGhpcy50aW1lckdhcF89Vi5WSVJUVUFMX1BST0NFU1NfR0FQX01JQ1JPX1NFQ09ORCx0aGlzLnBvc3RQcm9jZXNzSW50ZXJmYWNlXz1uZXcgVXIoci5iYWNrZ3JvdW5kQXNzZXRVcmkpKTpyLnRyYW5zZm9ybWVyVHlwZT09PSJCYWNrZ3JvdW5kQmx1ciI/KHRoaXMudGltZXJHYXBfPVYuQkxVUl9QUk9DRVNTX0dBUF9NSUNST19TRUNPTkQsdGhpcy5wb3N0UHJvY2Vzc0ludGVyZmFjZV89bmV3IGl0KHIucmFkaXVzKSk6ci50cmFuc2Zvcm1lclR5cGU9PT0iVmlkZW9CYWNrZ3JvdW5kIj90aGlzLnRpbWVyR2FwXz1WLkJMVVJfUFJPQ0VTU19HQVBfTUlDUk9fU0VDT05EOnIudHJhbnNmb3JtZXJUeXBlPT09IlNpbHVldGVCbHVyIj8odGhpcy50aW1lckdhcF89Vi5CTFVSX1BST0NFU1NfR0FQX01JQ1JPX1NFQ09ORCx0aGlzLnBvc3RQcm9jZXNzSW50ZXJmYWNlXz1uZXcgV3Ioci5yYWRpdXMpKToodGhpcy50aW1lckdhcF89Vi5CTFVSX1BST0NFU1NfR0FQX01JQ1JPX1NFQ09ORCx0aGlzLnBvc3RQcm9jZXNzSW50ZXJmYWNlXz1uZXcgaXQpfWFzeW5jIHN0YXJ0KCl7YXdhaXQgJHIodGhpcy5jb25maWdfLndhc21Bc3NldFVyaVBhdGgpLnRoZW4ocj0+e3ImJih0aGlzLnZvYW5nZVRGTGl0ZUluZm9fPXIsdGhpcy5zZWdtZW50YXRpb25NYXNrQ2FudmFzXy53aWR0aD10aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0V2lkdGhfLHRoaXMuc2VnbWVudGF0aW9uTWFza0NhbnZhc18uaGVpZ2h0PXRoaXMudm9hbmdlVEZMaXRlSW5mb18uaW5wdXRIZWlnaHRfLHRoaXMuc2VnbWVudGF0aW9uUGl4ZWxDb3VudF89dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5pbnB1dFdpZHRoXyp0aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0SGVpZ2h0Xyx0aGlzLnNlZ21lbnRhdGlvbk1hc2tfPW5ldyBJbWFnZURhdGEodGhpcy52b2FuZ2VURkxpdGVJbmZvXy5pbnB1dFdpZHRoXyx0aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0SGVpZ2h0XyksdGhpcy5pbnB1dE1lbW9yeU9mZnNldF89dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5pbnB1dE1lbW9yeU9mZnNldF8vNCx0aGlzLm91dHB1dE1lbW9yeU9mZnNldF89dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5vdXRwdXRNZW1vcnlPZmZzZXRfLzQpfSkuY2F0Y2gocj0+e2NvbnNvbGUubG9nKCJlcnJvciB1c2VURkxpdGU6IixyKX0pfWFzeW5jIHRyYW5zZm9ybShyLGUpeyh0aGlzLnJlc3VsdENhbnZhc18ud2lkdGghPXIuZGlzcGxheVdpZHRofHx0aGlzLnJlc3VsdENhbnZhc18uaGVpZ2h0IT1yLmRpc3BsYXlIZWlnaHQpJiYodGhpcy5yZXN1bHRDYW52YXNfLndpZHRoPXIuZGlzcGxheVdpZHRoLHRoaXMucmVzdWx0Q2FudmFzXy5oZWlnaHQ9ci5kaXNwbGF5SGVpZ2h0KSwodGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDYW52YXNfLndpZHRoIT1yLmRpc3BsYXlXaWR0aHx8dGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDYW52YXNfLmhlaWdodCE9ci5kaXNwbGF5SGVpZ2h0KSYmKHRoaXMuc2VnbWVudGF0aW9uTWFza0ZyYW1lQ2FudmFzXy53aWR0aD1yLmRpc3BsYXlXaWR0aCx0aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUNhbnZhc18uaGVpZ2h0PXIuZGlzcGxheUhlaWdodCk7Y29uc3QgdD1yLnRpbWVzdGFtcDtjcmVhdGVJbWFnZUJpdG1hcChyKS50aGVuKG49PntyLmNsb3NlKCksdGhpcy5wcm9jZXNzRnJhbWUoZSxuLHQpfSkuY2F0Y2gobj0+e2NvbnNvbGUuZXJyb3IoImNyZWF0ZUltYWdlQml0bWFwIixuKSxlLmVucXVldWUocil9KX1wcm9jZXNzRnJhbWUocixlLHQpe3QtdGhpcy5sYXN0UHJvY2Vzc2VkVGltZXN0YW1wXz49dGhpcy50aW1lckdhcF8mJih0aGlzLmxhc3RQcm9jZXNzZWRUaW1lc3RhbXBfPXQsdGhpcy5wcm9jZXNzU291cmNlKGUpKSx0aGlzLnJlc3VsdEN0eF8uZ2xvYmFsQ29tcG9zaXRlT3BlcmF0aW9uPSJjb3B5Iix0aGlzLnJlc3VsdEN0eF8uZHJhd0ltYWdlKHRoaXMuc2VnbWVudGF0aW9uTWFza0ZyYW1lQ2FudmFzXywwLDAsdGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDYW52YXNfLndpZHRoLHRoaXMuc2VnbWVudGF0aW9uTWFza0ZyYW1lQ2FudmFzXy5oZWlnaHQpLHRoaXMucmVzdWx0Q3R4Xy5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb249InNvdXJjZS1hdG9wIix0aGlzLnJlc3VsdEN0eF8uZmlsdGVyPSJub25lIix0aGlzLnJlc3VsdEN0eF8uZHJhd0ltYWdlKGUsMCwwLGUud2lkdGgsZS5oZWlnaHQpLHRoaXMucG9zdFByb2Nlc3NJbnRlcmZhY2VfJiZ0aGlzLnBvc3RQcm9jZXNzSW50ZXJmYWNlXy5ydW5Qb3N0UHJvY2Vzc2luZyh0aGlzLnJlc3VsdEN0eF8sZSksci5lbnF1ZXVlKG5ldyBWaWRlb0ZyYW1lKHRoaXMucmVzdWx0Q2FudmFzXyx7dGltZXN0YW1wOnQsYWxwaGE6ImRpc2NhcmQifSkpLHRoaXMuZnJhbWVDb3VudGVyXysrfXByb2Nlc3NTb3VyY2Uocil7dGhpcy5zZWdtZW50YXRpb25NYXNrQ3R4Xy5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb249ImNvcHkiLHRoaXMuc2VnbWVudGF0aW9uTWFza0N0eF8uZHJhd0ltYWdlKHIsMCwwLHIud2lkdGgsci5oZWlnaHQsMCwwLHRoaXMudm9hbmdlVEZMaXRlSW5mb18uaW5wdXRXaWR0aF8sdGhpcy52b2FuZ2VURkxpdGVJbmZvXy5pbnB1dEhlaWdodF8pO2xldCBlPXRoaXMuc2VnbWVudGF0aW9uTWFza0N0eF8uZ2V0SW1hZ2VEYXRhKDAsMCx0aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0V2lkdGhfLHRoaXMudm9hbmdlVEZMaXRlSW5mb18uaW5wdXRIZWlnaHRfKTtmb3IobGV0IHQ9MDt0PHRoaXMuc2VnbWVudGF0aW9uUGl4ZWxDb3VudF87dCsrKXRoaXMudm9hbmdlVEZMaXRlSW5mb18uc2VsZWN0ZWRURkxpdGVfLkhFQVBGMzJbdGhpcy5pbnB1dE1lbW9yeU9mZnNldF8rdCozXT1lLmRhdGFbdCo0XS8yNTUsdGhpcy52b2FuZ2VURkxpdGVJbmZvXy5zZWxlY3RlZFRGTGl0ZV8uSEVBUEYzMlt0aGlzLmlucHV0TWVtb3J5T2Zmc2V0Xyt0KjMrMV09ZS5kYXRhW3QqNCsxXS8yNTUsdGhpcy52b2FuZ2VURkxpdGVJbmZvXy5zZWxlY3RlZFRGTGl0ZV8uSEVBUEYzMlt0aGlzLmlucHV0TWVtb3J5T2Zmc2V0Xyt0KjMrMl09ZS5kYXRhW3QqNCsyXS8yNTU7dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5zZWxlY3RlZFRGTGl0ZV8uX3J1bkluZmVyZW5jZSgpO2ZvcihsZXQgdD0wO3Q8dGhpcy5zZWdtZW50YXRpb25QaXhlbENvdW50Xzt0Kyspe2NvbnN0IG49dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5zZWxlY3RlZFRGTGl0ZV8uSEVBUEYzMlt0aGlzLm91dHB1dE1lbW9yeU9mZnNldF8rdF07dGhpcy5zZWdtZW50YXRpb25NYXNrXy5kYXRhW3QqNCszXT0yNTUqbn10aGlzLnNlZ21lbnRhdGlvbk1hc2tDdHhfLnB1dEltYWdlRGF0YSh0aGlzLnNlZ21lbnRhdGlvbk1hc2tfLDAsMCksdGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDdHhfLmdsb2JhbENvbXBvc2l0ZU9wZXJhdGlvbj0iY29weSIsdGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDdHhfLmRyYXdJbWFnZSh0aGlzLnNlZ21lbnRhdGlvbk1hc2tDYW52YXNfLDAsMCx0aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0V2lkdGhfLHRoaXMudm9hbmdlVEZMaXRlSW5mb18uaW5wdXRIZWlnaHRfLDAsMCx0aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUNhbnZhc18ud2lkdGgsdGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDYW52YXNfLmhlaWdodCl9Zmx1c2goKXt9Z2V0VHJhbnNmb3JtZXJUeXBlKCl7cmV0dXJuIkJhY2tncm91bmRUcmFuc2Zvcm1lciJ9c2V0VmlkZW9CR1JlYWRhYmxlKHIpe3RoaXMuY29uZmlnXy50cmFuc2Zvcm1lclR5cGU9PT0iVmlkZW9CYWNrZ3JvdW5kIiYmKHRoaXMucG9zdFByb2Nlc3NJbnRlcmZhY2VfPW5ldyBrcihyKSl9fTtsZXQgU2U9VjtpZihTZS5CTFVSX1BST0NFU1NfR0FQX01JQ1JPX1NFQ09ORD0yZTUsU2UuVklSVFVBTF9QUk9DRVNTX0dBUF9NSUNST19TRUNPTkQ9MTVlNCx0eXBlb2YgaW1wb3J0U2NyaXB0cz09ImZ1bmN0aW9uIil7bGV0IHI9bmV3IExyO29ubWVzc2FnZT1hc3luYyBlPT57Y29uc3R7b3BlcmF0aW9uOnR9PWUuZGF0YTtpZihjb25zb2xlLmxvZygib3BlcmF0aW9uPSIsdCksdD09PSJ0cmFuc2Zvcm0iKXtjb25zdHtyZWFkYWJsZTpuLHdyaXRhYmxlOnMsY29uZmlnOmEsdmlkZW9CR1JlYWRhYmxlOmZ9PWUuZGF0YTtsZXQgbD1bXTtjb25zdCBoPW5ldyBTZShKU09OLnBhcnNlKGEpKTtoLnNldFZpZGVvQkdSZWFkYWJsZShmKSxsLnB1c2goaCksci5zZXRUcmFuc2Zvcm1lcnMobCkudGhlbigoKT0+e3IudHJhbnNmb3JtKG4scykudGhlbigoKT0+e3Bvc3RNZXNzYWdlKCJzdWNjZXNzIil9KS5jYXRjaChnPT57cG9zdE1lc3NhZ2UoImVycm9yOiIrZyl9KX0pLmNhdGNoKGc9Pntwb3N0TWVzc2FnZSgiZXJyb3I6IitnKX0pfWVsc2UgdD09PSJkZXN0cm95IiYmci5kZXN0cm95KCkudGhlbigoKT0+e3Bvc3RNZXNzYWdlKCJzdWNjZXNzIil9KS5jYXRjaChuPT57cG9zdE1lc3NhZ2UoImVycm9yOiIrbil9KX19fSkoKTsK";
1134
+ const blob = typeof window !== "undefined" && window.Blob && new Blob([atob(encodedJs)], { type: "text/javascript;charset=utf-8" });
1135
+ function WorkerWrapper() {
1136
+ const objURL = blob && (window.URL || window.webkitURL).createObjectURL(blob);
1137
+ try {
1138
+ return objURL ? new Worker(objURL) : new Worker("data:application/javascript;base64," + encodedJs, { type: "module" });
1139
+ } finally {
1140
+ objURL && (window.URL || window.webkitURL).revokeObjectURL(objURL);
1141
+ }
1142
+ }
1143
+ class VonageMediaProcessorWorker {
1144
+ constructor(config) {
1145
+ this.config_ = config;
1146
+ this.worker_ = new WorkerWrapper();
1147
+ }
1148
+ transform(readable, writable) {
1149
+ return new Promise(async (resolve, reject) => {
1150
+ console.log("Created a worker thread.");
1151
+ const onMessage = new Promise((resolve2, reject2) => {
1152
+ this.worker_.addEventListener("message", function handleMsgFromWorker(msg) {
1153
+ if (msg.data === "success") {
1154
+ resolve2();
1155
+ } else {
1156
+ reject2(msg.data);
1157
+ }
1158
+ });
1159
+ });
1160
+ if (this.config_.transformerType === "VideoBackground") {
1161
+ const bgvideo = document.createElement("video");
1162
+ bgvideo.src = this.config_.backgroundAssetUri;
1163
+ bgvideo.muted = true;
1164
+ try {
1165
+ await bgvideo.play();
1166
+ } catch (error) {
1167
+ reject(error);
1168
+ return;
1169
+ }
1170
+ const bgvideotrack = bgvideo.captureStream().getVideoTracks()[0];
1171
+ const processor = new MediaStreamTrackProcessor(bgvideotrack);
1172
+ const videoBGReadable = processor.readable;
1173
+ this.worker_.postMessage({
1174
+ operation: "transform",
1175
+ readable,
1176
+ writable,
1177
+ config: JSON.stringify(this.config_),
1178
+ videoBGReadable
1179
+ }, [readable, writable, videoBGReadable]);
1180
+ } else {
1181
+ this.worker_.postMessage({
1182
+ operation: "transform",
1183
+ readable,
1184
+ writable,
1185
+ config: JSON.stringify(this.config_)
1186
+ }, [readable, writable]);
1187
+ }
1188
+ onMessage.then(() => {
1189
+ console.log("Worker inited");
1190
+ resolve();
1191
+ }).catch((e) => {
1192
+ console.log("Worker inited error");
1193
+ reject(e);
1194
+ });
1195
+ });
1196
+ }
1197
+ destroy() {
1198
+ return new Promise(async (resolve, reject) => {
1199
+ console.log("destroy a worker thread.");
1200
+ const onMessage = new Promise((resolve2, reject2) => {
1201
+ this.worker_.addEventListener("message", function handleMsgFromWorker(msg) {
1202
+ if (msg.data === "success") {
1203
+ resolve2();
1204
+ } else {
1205
+ reject2(msg.data);
1206
+ }
1207
+ });
1208
+ });
1209
+ this.worker_.postMessage({
1210
+ operation: "destroy"
1211
+ });
1212
+ onMessage.then(() => {
1213
+ console.log("Worker destroyed");
1214
+ resolve();
1215
+ }).catch((e) => {
1216
+ console.log("Worker destroy error");
1217
+ reject(e);
1218
+ });
1219
+ });
1220
+ }
1221
+ }
1222
+ var BlurRadius;
1223
+ (function(BlurRadius2) {
1224
+ BlurRadius2["Low"] = "Low";
1225
+ BlurRadius2["High"] = "High";
1226
+ })(BlurRadius || (BlurRadius = {}));
1227
+ const simd = async () => WebAssembly.validate(new Uint8Array([0, 97, 115, 109, 1, 0, 0, 0, 1, 5, 1, 96, 0, 1, 123, 3, 2, 1, 0, 10, 10, 1, 8, 0, 65, 0, 253, 15, 253, 98, 11]));
1228
+ function isSupported() {
1229
+ return new Promise(async (resolve, reject) => {
1230
+ try {
1231
+ await isSupported$1();
1232
+ } catch (e) {
1233
+ reject(e);
1234
+ return;
1235
+ }
1236
+ simd().then((isSIMDSupported) => {
1237
+ if (isSIMDSupported === false) {
1238
+ reject("Your browser does not support WebAssembly features.");
1239
+ return;
1240
+ }
1241
+ resolve();
1242
+ }).catch((e) => {
1243
+ reject(e);
1244
+ });
1245
+ });
1246
+ }
1247
+ function createVonageMediaProcessorConnector(config) {
1248
+ return new Promise(async (resolve, reject) => {
1249
+ const connector = new MediaProcessorConnector(new VonageMediaProcessorWorker(config));
1250
+ resolve(connector);
1251
+ return;
1252
+ });
1253
+ }
1254
+ export { BlurRadius, createVonageMediaProcessorConnector, isSupported };