@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.
- package/README.md +9 -0
- package/dist/docs/.nojekyll +1 -0
- package/dist/docs/assets/highlight.css +85 -0
- package/dist/docs/assets/icons.css +1043 -0
- package/dist/docs/assets/icons.png +0 -0
- package/dist/docs/assets/icons@2x.png +0 -0
- package/dist/docs/assets/main.js +52 -0
- package/dist/docs/assets/search.js +1 -0
- package/dist/docs/assets/style.css +1413 -0
- package/dist/docs/assets/widgets.png +0 -0
- package/dist/docs/assets/widgets@2x.png +0 -0
- package/dist/docs/enums/BlurRadius.html +7 -0
- package/dist/docs/index.html +15 -0
- package/dist/docs/modules.html +30 -0
- package/dist/ml-transformers.es.js +1254 -0
- package/dist/ml-transformers.umd.js +2 -0
- package/dist/models/selfie_segmentation_landscape.tflite +0 -0
- package/dist/tflite-simd.wasm +0 -0
- package/dist/types/main.d.ts +95 -0
- package/dist/types/src/transformers/BackgroundTransformer.d.ts +34 -0
- package/dist/types/src/transformers/BlurPostProcess.d.ts +10 -0
- package/dist/types/src/transformers/PostProcessInterface.d.ts +5 -0
- package/dist/types/src/transformers/Queue.d.ts +13 -0
- package/dist/types/src/transformers/SilueteBlurPostProcess.d.ts +10 -0
- package/dist/types/src/transformers/VideoPostProcess.d.ts +11 -0
- package/dist/types/src/transformers/VirtualPostProcess.d.ts +11 -0
- package/dist/types/src/transformers/VonageTFLite.d.ts +28 -0
- package/dist/types/src/transformers/VonageTransformerConfig.d.ts +1 -0
- package/dist/types/src/types.d.ts +65 -0
- package/dist/types/src/utils/utils.d.ts +6 -0
- package/dist/types/src/vonage/VonageMediaProcessorThread.d.ts +1 -0
- package/dist/types/src/vonage/VonageMediaProcessorWorker.d.ts +10 -0
- 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 };
|