ember-scoped-css 2.1.0 → 2.2.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.
@@ -0,0 +1,1416 @@
1
+ //#region rolldown:runtime
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __copyProps = (to, from, except, desc) => {
9
+ if (from && typeof from === "object" || typeof from === "function") for (var keys = __getOwnPropNames(from), i = 0, n = keys.length, key; i < n; i++) {
10
+ key = keys[i];
11
+ if (!__hasOwnProp.call(to, key) && key !== except) __defProp(to, key, {
12
+ get: ((k) => from[k]).bind(null, key),
13
+ enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
14
+ });
15
+ }
16
+ return to;
17
+ };
18
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", {
19
+ value: mod,
20
+ enumerable: true
21
+ }) : target, mod));
22
+
23
+ //#endregion
24
+ let node_assert = require("node:assert");
25
+ node_assert = __toESM(node_assert);
26
+ let node_fs = require("node:fs");
27
+ node_fs = __toESM(node_fs);
28
+ let node_module = require("node:module");
29
+ node_module = __toESM(node_module);
30
+ let node_path = require("node:path");
31
+ node_path = __toESM(node_path);
32
+ let path = require("path");
33
+ path = __toESM(path);
34
+ let unplugin = require("unplugin");
35
+ unplugin = __toESM(unplugin);
36
+ let postcss = require("postcss");
37
+ postcss = __toESM(postcss);
38
+ let postcss_selector_parser = require("postcss-selector-parser");
39
+ postcss_selector_parser = __toESM(postcss_selector_parser);
40
+ let fs = require("fs");
41
+ fs = __toESM(fs);
42
+ let postcss_scss = require("postcss-scss");
43
+ postcss_scss = __toESM(postcss_scss);
44
+ let node_process = require("node:process");
45
+ node_process = __toESM(node_process);
46
+ let ember_template_recast = require("ember-template-recast");
47
+ ember_template_recast = __toESM(ember_template_recast);
48
+
49
+ //#region src/lib/path/const.js
50
+ /**
51
+ * Join will convert to whatever is appropriate fro the current platform
52
+ */
53
+ const leadingSlashPath = {
54
+ embroiderDir: path.default.join("/node_modules/.embroider/"),
55
+ atEmbroider: path.default.join("/@embroider"),
56
+ componentsDir: path.default.join("/components/"),
57
+ templatesDir: path.default.join("/templates/"),
58
+ routesDir: path.default.join("/routes/"),
59
+ testem: path.default.join("/testem"),
60
+ src: path.default.join("/src/"),
61
+ app: path.default.join("/app/")
62
+ };
63
+ const barePath = { pnpmDir: path.default.join("node_modules/.pnpm") };
64
+
65
+ //#endregion
66
+ //#region src/lib/path/md5.js
67
+ /**
68
+ * Add integers, wrapping at 2^32.
69
+ * This uses 16-bit operations internally to work around bugs in interpreters.
70
+ *
71
+ * @param {number} x First integer
72
+ * @param {number} y Second integer
73
+ * @returns {number} Sum
74
+ */
75
+ function safeAdd(x, y) {
76
+ var lsw = (x & 65535) + (y & 65535);
77
+ return (x >> 16) + (y >> 16) + (lsw >> 16) << 16 | lsw & 65535;
78
+ }
79
+ /**
80
+ * Bitwise rotate a 32-bit number to the left.
81
+ *
82
+ * @param {number} num 32-bit number
83
+ * @param {number} cnt Rotation count
84
+ * @returns {number} Rotated number
85
+ */
86
+ function bitRotateLeft(num, cnt) {
87
+ return num << cnt | num >>> 32 - cnt;
88
+ }
89
+ /**
90
+ * Basic operation the algorithm uses.
91
+ *
92
+ * @param {number} q q
93
+ * @param {number} a a
94
+ * @param {number} b b
95
+ * @param {number} x x
96
+ * @param {number} s s
97
+ * @param {number} t t
98
+ * @returns {number} Result
99
+ */
100
+ function md5cmn(q, a, b, x, s, t) {
101
+ return safeAdd(bitRotateLeft(safeAdd(safeAdd(a, q), safeAdd(x, t)), s), b);
102
+ }
103
+ /**
104
+ * Basic operation the algorithm uses.
105
+ *
106
+ * @param {number} a a
107
+ * @param {number} b b
108
+ * @param {number} c c
109
+ * @param {number} d d
110
+ * @param {number} x x
111
+ * @param {number} s s
112
+ * @param {number} t t
113
+ * @returns {number} Result
114
+ */
115
+ function md5ff(a, b, c, d, x, s, t) {
116
+ return md5cmn(b & c | ~b & d, a, b, x, s, t);
117
+ }
118
+ /**
119
+ * Basic operation the algorithm uses.
120
+ *
121
+ * @param {number} a a
122
+ * @param {number} b b
123
+ * @param {number} c c
124
+ * @param {number} d d
125
+ * @param {number} x x
126
+ * @param {number} s s
127
+ * @param {number} t t
128
+ * @returns {number} Result
129
+ */
130
+ function md5gg(a, b, c, d, x, s, t) {
131
+ return md5cmn(b & d | c & ~d, a, b, x, s, t);
132
+ }
133
+ /**
134
+ * Basic operation the algorithm uses.
135
+ *
136
+ * @param {number} a a
137
+ * @param {number} b b
138
+ * @param {number} c c
139
+ * @param {number} d d
140
+ * @param {number} x x
141
+ * @param {number} s s
142
+ * @param {number} t t
143
+ * @returns {number} Result
144
+ */
145
+ function md5hh(a, b, c, d, x, s, t) {
146
+ return md5cmn(b ^ c ^ d, a, b, x, s, t);
147
+ }
148
+ /**
149
+ * Basic operation the algorithm uses.
150
+ *
151
+ * @param {number} a a
152
+ * @param {number} b b
153
+ * @param {number} c c
154
+ * @param {number} d d
155
+ * @param {number} x x
156
+ * @param {number} s s
157
+ * @param {number} t t
158
+ * @returns {number} Result
159
+ */
160
+ function md5ii(a, b, c, d, x, s, t) {
161
+ return md5cmn(c ^ (b | ~d), a, b, x, s, t);
162
+ }
163
+ /**
164
+ * Calculate the MD5 of an array of little-endian words, and a bit length.
165
+ *
166
+ * @param {Array} x Array of little-endian words
167
+ * @param {number} len Bit length
168
+ * @returns {Array<number>} MD5 Array
169
+ */
170
+ function binlMD5(x, len) {
171
+ x[len >> 5] |= 128 << len % 32;
172
+ x[(len + 64 >>> 9 << 4) + 14] = len;
173
+ var i;
174
+ var olda;
175
+ var oldb;
176
+ var oldc;
177
+ var oldd;
178
+ var a = 1732584193;
179
+ var b = -271733879;
180
+ var c = -1732584194;
181
+ var d = 271733878;
182
+ for (i = 0; i < x.length; i += 16) {
183
+ olda = a;
184
+ oldb = b;
185
+ oldc = c;
186
+ oldd = d;
187
+ a = md5ff(a, b, c, d, x[i], 7, -680876936);
188
+ d = md5ff(d, a, b, c, x[i + 1], 12, -389564586);
189
+ c = md5ff(c, d, a, b, x[i + 2], 17, 606105819);
190
+ b = md5ff(b, c, d, a, x[i + 3], 22, -1044525330);
191
+ a = md5ff(a, b, c, d, x[i + 4], 7, -176418897);
192
+ d = md5ff(d, a, b, c, x[i + 5], 12, 1200080426);
193
+ c = md5ff(c, d, a, b, x[i + 6], 17, -1473231341);
194
+ b = md5ff(b, c, d, a, x[i + 7], 22, -45705983);
195
+ a = md5ff(a, b, c, d, x[i + 8], 7, 1770035416);
196
+ d = md5ff(d, a, b, c, x[i + 9], 12, -1958414417);
197
+ c = md5ff(c, d, a, b, x[i + 10], 17, -42063);
198
+ b = md5ff(b, c, d, a, x[i + 11], 22, -1990404162);
199
+ a = md5ff(a, b, c, d, x[i + 12], 7, 1804603682);
200
+ d = md5ff(d, a, b, c, x[i + 13], 12, -40341101);
201
+ c = md5ff(c, d, a, b, x[i + 14], 17, -1502002290);
202
+ b = md5ff(b, c, d, a, x[i + 15], 22, 1236535329);
203
+ a = md5gg(a, b, c, d, x[i + 1], 5, -165796510);
204
+ d = md5gg(d, a, b, c, x[i + 6], 9, -1069501632);
205
+ c = md5gg(c, d, a, b, x[i + 11], 14, 643717713);
206
+ b = md5gg(b, c, d, a, x[i], 20, -373897302);
207
+ a = md5gg(a, b, c, d, x[i + 5], 5, -701558691);
208
+ d = md5gg(d, a, b, c, x[i + 10], 9, 38016083);
209
+ c = md5gg(c, d, a, b, x[i + 15], 14, -660478335);
210
+ b = md5gg(b, c, d, a, x[i + 4], 20, -405537848);
211
+ a = md5gg(a, b, c, d, x[i + 9], 5, 568446438);
212
+ d = md5gg(d, a, b, c, x[i + 14], 9, -1019803690);
213
+ c = md5gg(c, d, a, b, x[i + 3], 14, -187363961);
214
+ b = md5gg(b, c, d, a, x[i + 8], 20, 1163531501);
215
+ a = md5gg(a, b, c, d, x[i + 13], 5, -1444681467);
216
+ d = md5gg(d, a, b, c, x[i + 2], 9, -51403784);
217
+ c = md5gg(c, d, a, b, x[i + 7], 14, 1735328473);
218
+ b = md5gg(b, c, d, a, x[i + 12], 20, -1926607734);
219
+ a = md5hh(a, b, c, d, x[i + 5], 4, -378558);
220
+ d = md5hh(d, a, b, c, x[i + 8], 11, -2022574463);
221
+ c = md5hh(c, d, a, b, x[i + 11], 16, 1839030562);
222
+ b = md5hh(b, c, d, a, x[i + 14], 23, -35309556);
223
+ a = md5hh(a, b, c, d, x[i + 1], 4, -1530992060);
224
+ d = md5hh(d, a, b, c, x[i + 4], 11, 1272893353);
225
+ c = md5hh(c, d, a, b, x[i + 7], 16, -155497632);
226
+ b = md5hh(b, c, d, a, x[i + 10], 23, -1094730640);
227
+ a = md5hh(a, b, c, d, x[i + 13], 4, 681279174);
228
+ d = md5hh(d, a, b, c, x[i], 11, -358537222);
229
+ c = md5hh(c, d, a, b, x[i + 3], 16, -722521979);
230
+ b = md5hh(b, c, d, a, x[i + 6], 23, 76029189);
231
+ a = md5hh(a, b, c, d, x[i + 9], 4, -640364487);
232
+ d = md5hh(d, a, b, c, x[i + 12], 11, -421815835);
233
+ c = md5hh(c, d, a, b, x[i + 15], 16, 530742520);
234
+ b = md5hh(b, c, d, a, x[i + 2], 23, -995338651);
235
+ a = md5ii(a, b, c, d, x[i], 6, -198630844);
236
+ d = md5ii(d, a, b, c, x[i + 7], 10, 1126891415);
237
+ c = md5ii(c, d, a, b, x[i + 14], 15, -1416354905);
238
+ b = md5ii(b, c, d, a, x[i + 5], 21, -57434055);
239
+ a = md5ii(a, b, c, d, x[i + 12], 6, 1700485571);
240
+ d = md5ii(d, a, b, c, x[i + 3], 10, -1894986606);
241
+ c = md5ii(c, d, a, b, x[i + 10], 15, -1051523);
242
+ b = md5ii(b, c, d, a, x[i + 1], 21, -2054922799);
243
+ a = md5ii(a, b, c, d, x[i + 8], 6, 1873313359);
244
+ d = md5ii(d, a, b, c, x[i + 15], 10, -30611744);
245
+ c = md5ii(c, d, a, b, x[i + 6], 15, -1560198380);
246
+ b = md5ii(b, c, d, a, x[i + 13], 21, 1309151649);
247
+ a = md5ii(a, b, c, d, x[i + 4], 6, -145523070);
248
+ d = md5ii(d, a, b, c, x[i + 11], 10, -1120210379);
249
+ c = md5ii(c, d, a, b, x[i + 2], 15, 718787259);
250
+ b = md5ii(b, c, d, a, x[i + 9], 21, -343485551);
251
+ a = safeAdd(a, olda);
252
+ b = safeAdd(b, oldb);
253
+ c = safeAdd(c, oldc);
254
+ d = safeAdd(d, oldd);
255
+ }
256
+ return [
257
+ a,
258
+ b,
259
+ c,
260
+ d
261
+ ];
262
+ }
263
+ /**
264
+ * Convert an array of little-endian words to a string
265
+ *
266
+ * @param {Array<number>} input MD5 Array
267
+ * @returns {string} MD5 string
268
+ */
269
+ function binl2rstr(input) {
270
+ var i;
271
+ var output = "";
272
+ var length32 = input.length * 32;
273
+ for (i = 0; i < length32; i += 8) output += String.fromCharCode(input[i >> 5] >>> i % 32 & 255);
274
+ return output;
275
+ }
276
+ /**
277
+ * Convert a raw string to an array of little-endian words
278
+ * Characters >255 have their high-byte silently ignored.
279
+ *
280
+ * @param {string} input Raw input string
281
+ * @returns {Array<number>} Array of little-endian words
282
+ */
283
+ function rstr2binl(input) {
284
+ var i;
285
+ var output = [];
286
+ output[(input.length >> 2) - 1] = void 0;
287
+ for (i = 0; i < output.length; i += 1) output[i] = 0;
288
+ var length8 = input.length * 8;
289
+ for (i = 0; i < length8; i += 8) output[i >> 5] |= (input.charCodeAt(i / 8) & 255) << i % 32;
290
+ return output;
291
+ }
292
+ /**
293
+ * Calculate the MD5 of a raw string
294
+ *
295
+ * @param {string} s Input string
296
+ * @returns {string} Raw MD5 string
297
+ */
298
+ function rstrMD5(s) {
299
+ return binl2rstr(binlMD5(rstr2binl(s), s.length * 8));
300
+ }
301
+ /**
302
+ * Calculates the HMAC-MD5 of a key and some data (raw strings)
303
+ *
304
+ * @param {string} key HMAC key
305
+ * @param {string} data Raw input string
306
+ * @returns {string} Raw MD5 string
307
+ */
308
+ function rstrHMACMD5(key, data) {
309
+ var i;
310
+ var bkey = rstr2binl(key);
311
+ var ipad = [];
312
+ var opad = [];
313
+ var hash$2;
314
+ ipad[15] = opad[15] = void 0;
315
+ if (bkey.length > 16) bkey = binlMD5(bkey, key.length * 8);
316
+ for (i = 0; i < 16; i += 1) {
317
+ ipad[i] = bkey[i] ^ 909522486;
318
+ opad[i] = bkey[i] ^ 1549556828;
319
+ }
320
+ hash$2 = binlMD5(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
321
+ return binl2rstr(binlMD5(opad.concat(hash$2), 640));
322
+ }
323
+ /**
324
+ * Convert a raw string to a hex string
325
+ *
326
+ * @param {string} input Raw input string
327
+ * @returns {string} Hex encoded string
328
+ */
329
+ function rstr2hex(input) {
330
+ var hexTab = "0123456789abcdef";
331
+ var output = "";
332
+ var x;
333
+ var i;
334
+ for (i = 0; i < input.length; i += 1) {
335
+ x = input.charCodeAt(i);
336
+ output += hexTab.charAt(x >>> 4 & 15) + hexTab.charAt(x & 15);
337
+ }
338
+ return output;
339
+ }
340
+ /**
341
+ * Encode a string as UTF-8
342
+ *
343
+ * @param {string} input Input string
344
+ * @returns {string} UTF8 string
345
+ */
346
+ function str2rstrUTF8(input) {
347
+ return unescape(encodeURIComponent(input));
348
+ }
349
+ /**
350
+ * Encodes input string as raw MD5 string
351
+ *
352
+ * @param {string} s Input string
353
+ * @returns {string} Raw MD5 string
354
+ */
355
+ function rawMD5(s) {
356
+ return rstrMD5(str2rstrUTF8(s));
357
+ }
358
+ /**
359
+ * Encodes input string as Hex encoded string
360
+ *
361
+ * @param {string} s Input string
362
+ * @returns {string} Hex encoded string
363
+ */
364
+ function hexMD5(s) {
365
+ return rstr2hex(rawMD5(s));
366
+ }
367
+ /**
368
+ * Calculates the raw HMAC-MD5 for the given key and data
369
+ *
370
+ * @param {string} k HMAC key
371
+ * @param {string} d Input string
372
+ * @returns {string} Raw MD5 string
373
+ */
374
+ function rawHMACMD5(k, d) {
375
+ return rstrHMACMD5(str2rstrUTF8(k), str2rstrUTF8(d));
376
+ }
377
+ /**
378
+ * Calculates the Hex encoded HMAC-MD5 for the given key and data
379
+ *
380
+ * @param {string} k HMAC key
381
+ * @param {string} d Input string
382
+ * @returns {string} Raw MD5 string
383
+ */
384
+ function hexHMACMD5(k, d) {
385
+ return rstr2hex(rawHMACMD5(k, d));
386
+ }
387
+ /**
388
+ * Calculates MD5 value for a given string.
389
+ * If a key is provided, calculates the HMAC-MD5 value.
390
+ * Returns a Hex encoded string unless the raw argument is given.
391
+ *
392
+ * @param {string} string Input string
393
+ * @param {string} [key] HMAC key
394
+ * @param {boolean} [raw] Raw output switch
395
+ * @returns {string} MD5 output
396
+ */
397
+ function md5(string, key, raw) {
398
+ if (!key) {
399
+ if (!raw) return hexMD5(string);
400
+ return rawMD5(string);
401
+ }
402
+ if (!raw) return hexHMACMD5(key, string);
403
+ return rawHMACMD5(key, string);
404
+ }
405
+
406
+ //#endregion
407
+ //#region src/lib/path/hash-from-module-path.js
408
+ /**
409
+ * The intent of this function is to generate the suffix/postfix for the
410
+ * css classes, based on the module-scoped path name.
411
+ *
412
+ * for example,
413
+ * hash('my-app/components/foo')
414
+ * instead of
415
+ * hash('app/components/foo')
416
+ *
417
+ * (unless your app name is 'app')
418
+ *
419
+ * @param {string} modulePath
420
+ * @returns {string}
421
+ */
422
+ function hash$1(modulePath) {
423
+ return "e" + md5(modulePath).substring(0, 8);
424
+ }
425
+ const hashFromModulePath$1 = hash$1;
426
+
427
+ //#endregion
428
+ //#region src/lib/path/utils.js
429
+ /**
430
+ *
431
+ * @param {string} filePath
432
+ * @returns {string}
433
+ */
434
+ function hashFromModulePath(filePath) {
435
+ return hashFromModulePath$1(forcePosix(filePath));
436
+ }
437
+ /**
438
+ * @param {string} filePath
439
+ */
440
+ function forcePosix(filePath) {
441
+ const parsed = node_path.default.parse(filePath);
442
+ if (parsed.root === "") return filePath.replaceAll(node_path.default.win32.sep, node_path.default.posix.sep);
443
+ return filePath.replace(/* @__PURE__ */ new RegExp(`^${RegExp.escape(parsed.root)}`), node_path.default.posix.sep).replaceAll(node_path.default.win32.sep, node_path.default.posix.sep);
444
+ }
445
+ let here = require("url").pathToFileURL(__filename).href;
446
+ let ourRequire = globalThis.require ? globalThis.require : here && (0, node_module.createRequire)(here);
447
+ if (!ourRequire) ourRequire = require;
448
+ const IRRELEVANT_PATHS = [barePath.pnpmDir, "__vite-"];
449
+ const UNSUPPORTED_DIRECTORIES = new Set(["tests"]);
450
+ const CWD = process.cwd();
451
+ /**
452
+ * Based on ember's component location conventions,
453
+ * this function will provide a path for where we
454
+ * expect the CSS to live.
455
+ *
456
+ * For co-located structure:
457
+ * - components/my-component.hbs
458
+ * - components/my-component.css
459
+ *
460
+ * For nested co-located structure
461
+ * - components/my-component/foo.hbs
462
+ * - components/my-component/foo.css
463
+ *
464
+ * For Pods routes structure
465
+ * - routes/my-route/template.{hbs,js}
466
+ * - routes/my-route/styles.css
467
+ *
468
+ * Deliberately not supported:
469
+ * - components w/ pods -- this is deprecated in 5.10
470
+ *
471
+ * @param {string} fileName - the hbs, js, gjs, gts or whatever co-located path.
472
+ * @returns {string} - expected css path
473
+ */
474
+ function cssPathFor(fileName) {
475
+ let cssPath = withoutExtension(fileName) + ".css";
476
+ if (isPod(fileName)) cssPath = fileName.replace(/template\.js$/, "styles.css").replace(/template\.gjs/, "styles.css").replace(/template\.gts/, "styles.css").replace(/template\.hbs/, "styles.css");
477
+ return cssPath;
478
+ }
479
+ /**
480
+ * Note that components in the "pods" convention will
481
+ * never be supported.
482
+ *
483
+ * @param {string} filePath
484
+ */
485
+ function isPodTemplate(filePath) {
486
+ if (filePath.includes(leadingSlashPath.componentsDir)) return false;
487
+ return filePath.endsWith("template.js") || filePath.endsWith("template.hbs") || filePath.endsWith("template.gjs") || filePath.endsWith("template.gts");
488
+ }
489
+ /**
490
+ * Note that components in the "pods" convention will
491
+ * never be supported.
492
+ *
493
+ * Checks if a file ends with
494
+ * - template.js
495
+ * - template.hbs
496
+ * - styles.css
497
+ *
498
+ * @param {string} filePath
499
+ */
500
+ function isPod(filePath) {
501
+ if (filePath.includes(leadingSlashPath.componentsDir)) return false;
502
+ if (isPodTemplate(filePath)) return true;
503
+ return filePath.endsWith("styles.css");
504
+ }
505
+ /**
506
+ *
507
+ * @param {string} filePath
508
+ * @returns the same path, but without the extension
509
+ */
510
+ function withoutExtension(filePath) {
511
+ let parsed = node_path.default.parse(filePath);
512
+ return node_path.default.join(parsed.dir, parsed.name);
513
+ }
514
+ /**
515
+ * Examples for fileName
516
+ * - absolute on-disk path
517
+ * - in webpack
518
+ * - URL-absolute path, starting with /
519
+ *
520
+ * @param {string} fileName
521
+ * @param {{ additionalRoots?: string[]; cwd: string }} options
522
+ * @returns
523
+ */
524
+ function isRelevantFile(fileName, { additionalRoots, cwd }) {
525
+ if (fileName.startsWith(leadingSlashPath.testem)) return false;
526
+ if (fileName.startsWith("\0")) return false;
527
+ if (node_path.default.isAbsolute(fileName) === false) {
528
+ if (fileName.match(/^[a-zA-Z]/)) return false;
529
+ }
530
+ if (fileName.startsWith(leadingSlashPath.atEmbroider)) return false;
531
+ if (IRRELEVANT_PATHS.some((i) => fileName.includes(i))) return false;
532
+ let workspace = findWorkspacePath(fileName);
533
+ (0, node_assert.default)(cwd, `cwd was not passed to isRelevantFile`);
534
+ if (workspace !== findWorkspacePath(cwd)) return false;
535
+ let [, ...parts] = fileName.replace(workspace, "").split(node_path.default.sep).filter(Boolean);
536
+ if (UNSUPPORTED_DIRECTORIES.has(parts[0])) return false;
537
+ if (![
538
+ leadingSlashPath.componentsDir,
539
+ leadingSlashPath.templatesDir,
540
+ leadingSlashPath.routesDir,
541
+ ...additionalRoots || []
542
+ ].some((root) => fileName.includes(root))) return;
543
+ return true;
544
+ }
545
+ function packageScopedPathToModulePath(packageScopedPath) {
546
+ /**
547
+ * *By convention*, `src` is omitted from component paths.
548
+ * We can reflect the same behavior by replacing src/
549
+ * with an empty string.
550
+ *
551
+ * CSS isn't emitted as a co-located module, but
552
+ * to keep conventions consistent across languages,
553
+ * we can pretend it is.
554
+ *
555
+ * Any customization beyond removing `src` and `app` is potentially confusing.
556
+ * If we need further customizations, we'll want to match on `exports` in the
557
+ * corresponding package.json
558
+ */
559
+ let packageRelative = packageScopedPath.replace(/* @__PURE__ */ new RegExp(`^${RegExp.escape(leadingSlashPath.src)}`), node_path.default.sep);
560
+ let parsed = node_path.default.parse(packageRelative);
561
+ if (isPod(packageRelative))
562
+ /**
563
+ * For pods, we chop off the whole file, and use the dir name as the "modulePath"
564
+ */
565
+ return parsed.dir;
566
+ return node_path.default.join(parsed.dir, parsed.name);
567
+ }
568
+ /**
569
+ * returns the app-module path of the source file
570
+ *
571
+ * This assumes normal ember app conventions
572
+ *
573
+ * which is `<package.json#name>/path-to-file`
574
+ */
575
+ function appPath(sourcePath) {
576
+ let workspacePath = findWorkspacePath(sourcePath);
577
+ let name = moduleName(sourcePath);
578
+ /**
579
+ * Under embroider builds, the spec-compliant version of the app
580
+ * has all the files under a folder which represents the package name,
581
+ * rather than "app".
582
+ */
583
+ let packageRelative = sourcePath.replace(workspacePath, "");
584
+ /**
585
+ * But we also don't want 'app' -- which is present in the v1 addon pipeline
586
+ */
587
+ packageRelative = packageRelative.replace(leadingSlashPath.app, node_path.default.sep);
588
+ packageRelative = node_path.default.normalize(packageRelative);
589
+ return `${name}${packageScopedPathToModulePath(packageRelative)}`;
590
+ }
591
+ /**
592
+ * To avoid hitting the filesysetm, we'll store all found
593
+ * project paths bere, so we can, in memory,
594
+ * get the folder where a package.json exists, rather than
595
+ * hit the file system every time.
596
+ */
597
+ const SEEN = /* @__PURE__ */ new Set();
598
+ function getSeen(sourcePath) {
599
+ if (SEEN.has(sourcePath)) return sourcePath;
600
+ let parts = sourcePath.split(node_path.default.sep);
601
+ for (let i = parts.length - 1; i > 1; i--) {
602
+ let toCheck = parts.slice(0, i).join(node_path.default.sep);
603
+ if (SEEN.has(toCheck)) return toCheck;
604
+ }
605
+ }
606
+ function findWorkspacePath(sourcePath, options) {
607
+ let cwd = options?.cwd ?? CWD;
608
+ if (sourcePath.endsWith(node_path.default.sep)) sourcePath = sourcePath.replace(/* @__PURE__ */ new RegExp(`${RegExp.escape(node_path.default.sep)}$`), "");
609
+ let seen = getSeen(sourcePath);
610
+ if (seen) return seen;
611
+ let candidatePath = node_path.default.join(sourcePath, "package.json");
612
+ if (node_fs.default.existsSync(candidatePath)) return sourcePath;
613
+ const packageJsonPath = findPackageJsonUp(sourcePath, { cwd });
614
+ if (!packageJsonPath) throw new Error(`Could not determine project for ${sourcePath}`);
615
+ const workspacePath = node_path.default.dirname(packageJsonPath);
616
+ SEEN.add(workspacePath);
617
+ return workspacePath;
618
+ }
619
+ function findPackageJsonUp(startPath, options) {
620
+ let cwd = options?.cwd ?? CWD;
621
+ let parts = startPath.split(node_path.default.sep);
622
+ for (let i = parts.length - 1; i > 1; i--) {
623
+ let toCheck = parts.slice(0, i).join(node_path.default.sep);
624
+ let packageJson = node_path.default.join(toCheck, "package.json");
625
+ if (node_fs.default.existsSync(packageJson)) return packageJson;
626
+ if (toCheck === cwd) break;
627
+ }
628
+ return null;
629
+ }
630
+ const MANIFEST_CACHE = /* @__PURE__ */ new Map();
631
+ /**
632
+ * Will return the package.json#name, or config/environment#moudlePrefix (if v1 app)
633
+ *
634
+ * @param {string} sourcePath
635
+ */
636
+ function moduleName(sourcePath) {
637
+ return getManifest(findWorkspacePath(sourcePath)).name;
638
+ }
639
+ /**
640
+ * @param {string} workspace
641
+ */
642
+ function getManifest(workspace) {
643
+ let existing = MANIFEST_CACHE.get(workspace);
644
+ if (existing) return existing;
645
+ let content = node_fs.default.readFileSync(node_path.default.join(workspace, "package.json")).toString();
646
+ let json = JSON.parse(content);
647
+ MANIFEST_CACHE.set(workspace, json);
648
+ return json;
649
+ }
650
+
651
+ //#endregion
652
+ //#region src/lib/renameClass.js
653
+ /**
654
+ *
655
+ * @param {string} className
656
+ * @param {string} postfix
657
+ * @param {Set<string>} [classesInCss]
658
+ * @returns
659
+ */
660
+ function renameClass(className, postfix, classesInCss) {
661
+ const renamedClasses = className.split(/\s+/).filter((c) => c).map((c) => c.trim()).map((c) => {
662
+ if (!classesInCss || classesInCss.has(c)) {
663
+ if (c.endsWith(postfix)) return c;
664
+ return c + "_" + postfix;
665
+ }
666
+ return c;
667
+ }).join(" ");
668
+ return className.replace(className.trimStart().trimEnd(), renamedClasses);
669
+ }
670
+
671
+ //#endregion
672
+ //#region src/build/babel-plugin.js
673
+ function _isRelevantFile(state, cwd) {
674
+ let fileName = state.file.opts.filename;
675
+ let additionalRoots = state.opts?.additionalRoots;
676
+ return isRelevantFile(fileName, {
677
+ additionalRoots,
678
+ cwd
679
+ });
680
+ }
681
+ /**
682
+ * @param {any} env - babel plugin env, env.types is most commonly used (esp in TS)
683
+ * @param {object} options - the options for scoped-css -- this is also available in each visitor's state.opts
684
+ * @param {string} workingDirectory
685
+ */
686
+ const scopedCSS$1 = (config) => (env, options, workingDirectory) => {
687
+ options = {
688
+ ...config,
689
+ ...options
690
+ };
691
+ /**
692
+ * This babel plugin does two things:
693
+ * - removes the import of scopedClass, if it exists
694
+ * - if scopedClass was imported, it is removed from any component's "scope bag"
695
+ * (the scope bag being a low-level object used for passing what is "in scope" for a component)
696
+ */
697
+ return { visitor: {
698
+ Program: { enter(path$8, state) {
699
+ if (!_isRelevantFile(state, workingDirectory)) {
700
+ state.canSkip = true;
701
+ return;
702
+ }
703
+ state.postfix = hashFromModulePath(appPath(state.filename));
704
+ } },
705
+ ImportDeclaration(path$8, state) {
706
+ if (state.canSkip) return;
707
+ if (path$8.node.source.value === "ember-scoped-css") {
708
+ let specifier = path$8.node.specifiers.find((x) => x.imported.name === "scopedClass");
709
+ if (specifier) state.file.opts.importedScopedClass = specifier.local.name;
710
+ if (specifier.local.name !== "scopedClass") throw new Error(`The scopedClass import is a psuedo-helper, and may not be renamed as it is removed at build time.`);
711
+ path$8.remove();
712
+ }
713
+ },
714
+ CallExpression(path$8, state) {
715
+ if (state.canSkip) return;
716
+ if (path$8.node.callee.type === "Identifier" && path$8.node.callee.name === state.file.opts?.importedScopedClass) {
717
+ if (path$8.node.arguments.length !== 1 || path$8.node.arguments[0].type !== "StringLiteral") throw new Error(`The scopedClass helper only accepts a single, non-dynamic, string literal argument.`);
718
+ const original = path$8.node.arguments[0].value;
719
+ const renamed = renameClass(original, state.postfix, new Set([original]));
720
+ const transformedString = env.types.stringLiteral(renamed);
721
+ path$8.replaceWith(transformedString);
722
+ }
723
+ },
724
+ ObjectProperty(path$8, state) {
725
+ if (!state.file.opts?.importedScopedClass) return;
726
+ if (path$8.node.value.type === "Identifier" && path$8.node.value.name === state.file.opts?.importedScopedClass) path$8.remove();
727
+ }
728
+ } };
729
+ };
730
+
731
+ //#endregion
732
+ //#region src/lib/css/utils.js
733
+ /**
734
+ * @param {string} css
735
+ * @return {string} hashed down version of the CSS for disambiguating
736
+ */
737
+ function hash(css) {
738
+ return `css-${md5(css)}`;
739
+ }
740
+ function isInsideGlobal(node, func) {
741
+ const parent = node.parent;
742
+ if (!parent) return false;
743
+ if (parent.type === "pseudo" && parent.value === ":global") return true;
744
+ return isInsideGlobal(parent, func);
745
+ }
746
+ /**
747
+ * @param {string} cssPath path to a CSS file
748
+ */
749
+ function getCSSInfo(cssPath) {
750
+ if (!(0, fs.existsSync)(cssPath)) return null;
751
+ return getCSSContentInfo((0, fs.readFileSync)(cssPath, "utf8"));
752
+ }
753
+ /**
754
+ * We use this function to check each class used in the template
755
+ * to see if we need to leave it alone or transform it
756
+ *
757
+ * @param {string} css the CSS's contents
758
+ * @param {string} [lang] optional language hint (e.g. 'scss', 'sass', 'less')
759
+ * @return {{ classes: Set<string>, tags: Set<string>, css: string, id: string }}
760
+ */
761
+ function getCSSContentInfo(css, lang) {
762
+ const classes = /* @__PURE__ */ new Set();
763
+ const tags = /* @__PURE__ */ new Set();
764
+ const parseOptions = lang === "scss" || lang === "sass" ? { syntax: postcss_scss.default } : {};
765
+ const ast = postcss.default.parse(css, parseOptions);
766
+ const isScss = lang === "scss" || lang === "sass";
767
+ ast.walk((node) => {
768
+ if (node.type === "rule") getClassesAndTags(isScss ? resolveNestedSassSelector(node) : node.selector, classes, tags);
769
+ });
770
+ return {
771
+ classes,
772
+ tags,
773
+ css,
774
+ id: hash(css)
775
+ };
776
+ }
777
+ /**
778
+ * Resolves a nested SCSS selector by substituting `&` with the fully-resolved
779
+ * parent selector, recursively. This converts e.g. `&--modifier` (child of
780
+ * `.block`) into `.block--modifier`, and handles arbitrary nesting depth so
781
+ * that `&--modifier` inside `&--modifier` inside `.block` yields
782
+ * `.block--modifier--modifier`.
783
+ *
784
+ * @param {import('postcss').Rule} node
785
+ * @return {string}
786
+ */
787
+ function resolveNestedSassSelector(node) {
788
+ const { selector } = node;
789
+ if (!selector.includes("&")) return selector;
790
+ const parent = node.parent;
791
+ if (!parent || parent.type !== "rule") return selector;
792
+ const resolvedParent = resolveNestedSassSelector(parent);
793
+ return selector.replace(/&/g, resolvedParent);
794
+ }
795
+ function getClassesAndTags(sel, classes, tags) {
796
+ const transform = (sls) => {
797
+ sls.walk((selector) => {
798
+ if (selector.type === "class" && !isInsideGlobal(selector)) classes.add(selector.value);
799
+ else if (selector.type === "tag" && !isInsideGlobal(selector)) tags.add(selector.value);
800
+ });
801
+ };
802
+ (0, postcss_selector_parser.default)(transform).processSync(sel);
803
+ }
804
+
805
+ //#endregion
806
+ //#region src/lib/css/rewrite.js
807
+ const SEP$1 = "__";
808
+ function isRule(node) {
809
+ return node.type === "rule";
810
+ }
811
+ function isDeclaration(node) {
812
+ return node.type === "decl";
813
+ }
814
+ /**
815
+ * NOTE: "keyframes" is a singular definition, in that it's a block containing keyframes
816
+ * using `@keyframes {}` with only one thing on the inside doesn't make sense.
817
+ */
818
+ function rewriteReferenceable(node, postfix) {
819
+ let originalName = node.params;
820
+ let postfixedName = node.params + SEP$1 + postfix;
821
+ node.params = postfixedName;
822
+ return {
823
+ originalName,
824
+ postfixedName
825
+ };
826
+ }
827
+ function rewriteSelector(sel, postfix) {
828
+ const transform = (selectors) => {
829
+ selectors.walk((selector) => {
830
+ if (isInsideGlobal(selector)) return;
831
+ if (selector.type === "psuedo") return;
832
+ if (isNthOfType(selector)) return;
833
+ if (selector.type === "class") selector.value += "_" + postfix;
834
+ else if (selector.type === "tag") selector.replaceWith(postcss_selector_parser.default.tag({ value: selector.value }), postcss_selector_parser.default.className({ value: postfix }));
835
+ });
836
+ selectors.walk((selector) => {
837
+ if (selector.type === "pseudo" && selector.value === ":global") selector.replaceWith(...selector.nodes);
838
+ });
839
+ };
840
+ return (0, postcss_selector_parser.default)(transform).processSync(sel);
841
+ }
842
+ function isNthOfType(node) {
843
+ if (!node) return false;
844
+ return node.parent?.value === ":nth-of-type" || isNthOfType(node.parent);
845
+ }
846
+ function isInsideKeyframes(node) {
847
+ const parent = node.parent;
848
+ if (!parent) return false;
849
+ if (parent.type === "atrule" && parent.name === "keyframes") return true;
850
+ return isInsideKeyframes(parent);
851
+ }
852
+ function rewriteCss(css, postfix, fileName, layerName) {
853
+ const ast = postcss.default.parse(css);
854
+ /**
855
+ * kind => originalName => postfixedName
856
+ * @type {{ [kind: string]: { [originalName: string]: string }}}
857
+ */
858
+ const referenceables = {
859
+ keyframes: {},
860
+ "counter-style": {},
861
+ "position-try": {},
862
+ property: {}
863
+ };
864
+ const availableReferenceables = new Set(Object.keys(referenceables));
865
+ function isReferenceable(node) {
866
+ if (node.type !== "atrule") return;
867
+ return availableReferenceables.has(node.name);
868
+ }
869
+ function updateDirectReferences(node) {
870
+ if (!node.value) return;
871
+ for (let [, map] of Object.entries(referenceables)) if (map[node.value]) node.value = map[node.value];
872
+ }
873
+ function updateShorthandContents(node) {
874
+ if (node.prop === "animation") {
875
+ let match = node.value.split(" ").filter((x) => referenceables.keyframes[x]);
876
+ if (match.length) match.forEach((x) => {
877
+ let replacement = referenceables.keyframes[x];
878
+ if (!replacement) return;
879
+ node.value = node.value.replace(x, replacement);
880
+ });
881
+ }
882
+ for (let [lookFor, replaceWith] of Object.entries(referenceables.property)) {
883
+ let lookForVar = `var(${lookFor})`;
884
+ let replaceWithVar = `var(${replaceWith})`;
885
+ node.value = node.value.replace(lookForVar, replaceWithVar);
886
+ }
887
+ }
888
+ /**
889
+ * We have to do two passes:
890
+ * 1. postfix all the referenceable syntax
891
+ * 2. postfix as normal, but also checking values of CSS properties
892
+ * that could match postfixed referenceables from step 1
893
+ */
894
+ ast.walk((node) => {
895
+ /**
896
+ * @keyframes, @counter-style, etc
897
+ */
898
+ if (isReferenceable(node)) {
899
+ let name = node.name;
900
+ let { originalName, postfixedName } = rewriteReferenceable(node, postfix);
901
+ referenceables[name][originalName] = postfixedName;
902
+ return;
903
+ }
904
+ });
905
+ ast.walk((node) => {
906
+ if (isDeclaration(node)) {
907
+ updateDirectReferences(node);
908
+ updateShorthandContents(node);
909
+ return;
910
+ }
911
+ if (isRule(node)) {
912
+ /**
913
+ * The inner-contents of a keyframe are percentages, rather than selectors
914
+ */
915
+ if (isInsideKeyframes(node)) return;
916
+ node.selector = rewriteSelector(node.selector, postfix);
917
+ return;
918
+ }
919
+ });
920
+ const rewrittenCss = ast.toString();
921
+ return [
922
+ `/* ${fileName} */`,
923
+ layerName ? `@layer ${layerName} {` : "",
924
+ rewrittenCss.trimEnd(),
925
+ layerName ? `}` : ""
926
+ ].filter(Boolean).join("\n") + "\n";
927
+ }
928
+
929
+ //#endregion
930
+ //#region src/lib/request.js
931
+ const KEY = "ember-scoped.css";
932
+ const SEP = "___";
933
+ const request = {
934
+ is: {
935
+ inline(request$1) {
936
+ return request$1.includes(KEY);
937
+ },
938
+ colocated(request$1) {
939
+ return request$1.includes(".css?scoped=");
940
+ }
941
+ },
942
+ inline: {
943
+ create(cssHash, postfix, cssContents, lang) {
944
+ let url = `./${postfix}${SEP}${cssHash}.${KEY}?css=${encodeURIComponent(cssContents)}`;
945
+ if (lang) url += `&lang=${encodeURIComponent(lang)}`;
946
+ return url;
947
+ },
948
+ decode(request$1) {
949
+ let [left, qps] = request$1.split("?");
950
+ left = left.slice(2).replace(`.${KEY}`, "");
951
+ let [postfix, hash$2] = left.split(SEP);
952
+ let search = new URLSearchParams(qps);
953
+ return {
954
+ hash: hash$2,
955
+ postfix,
956
+ css: search.get("css"),
957
+ from: search.get("from"),
958
+ lang: search.get("lang")
959
+ };
960
+ }
961
+ },
962
+ colocated: {
963
+ create(cssHash, postfix, filePath) {
964
+ return `./${node_path.default.basename(filePath)}?scoped=${postfix}&cssHash=${cssHash}`;
965
+ },
966
+ decode(request$1) {
967
+ const [fileName, qs] = request$1.split("?");
968
+ const search = new URLSearchParams(qs);
969
+ return {
970
+ fileName,
971
+ cssHash: search.get("cssHash"),
972
+ postfix: search.get("scoped")
973
+ };
974
+ }
975
+ }
976
+ };
977
+
978
+ //#endregion
979
+ //#region src/build/unplugin-colocated.js
980
+ const META$1 = "scoped-css:colocated";
981
+ /** File extensions that Vite can preprocess via its CSS preprocessor pipeline */
982
+ const PREPROCESSED_EXTENSIONS = new Set([
983
+ ".scss",
984
+ ".sass",
985
+ ".less",
986
+ ".styl",
987
+ ".stylus"
988
+ ]);
989
+ /**
990
+ * Plugin for supporting colocated styles
991
+ *
992
+ * e.g.:
993
+ * src/components/my-component.js
994
+ * src/components/my-component.css
995
+ */
996
+ function colocated(options = {}) {
997
+ const CWD$1 = process.cwd();
998
+ /** @type {import('vite').ResolvedConfig | undefined} */
999
+ let viteConfig;
1000
+ /** @type {((code: string, filename: string, config: unknown) => Promise<{ code: string }>) | undefined} */
1001
+ let preprocessCSS;
1002
+ /**
1003
+ *
1004
+ * @param {string} id the request id / what was imported
1005
+ * @param {string} filePath path on disk
1006
+ * @returns
1007
+ */
1008
+ function buildResponse(id, filePath) {
1009
+ const parsed = request.colocated.decode(id);
1010
+ const relativeFilePath = node_path.default.relative(CWD$1, filePath);
1011
+ return {
1012
+ id: filePath,
1013
+ meta: { [META$1]: {
1014
+ postfix: parsed.postfix,
1015
+ fileName: relativeFilePath,
1016
+ fullPath: filePath
1017
+ } }
1018
+ };
1019
+ }
1020
+ return {
1021
+ name: "ember-scoped-css:colocated",
1022
+ resolveId(id, importer) {
1023
+ if (request.is.colocated(id)) {
1024
+ const parsed = request.colocated.decode(id);
1025
+ return buildResponse(id, node_path.default.resolve(node_path.default.dirname(importer), node_path.default.basename(parsed.fileName)));
1026
+ }
1027
+ },
1028
+ load(id) {
1029
+ const meta = this.getModuleInfo(id)?.meta?.[META$1];
1030
+ if (meta) {
1031
+ this.addWatchFile(meta.fullPath);
1032
+ return rewriteCss((0, node_fs.readFileSync)(meta.fullPath, "utf-8"), meta.postfix, meta.fileName, options.layerName);
1033
+ }
1034
+ },
1035
+ vite: {
1036
+ async configResolved(config) {
1037
+ viteConfig = config;
1038
+ try {
1039
+ preprocessCSS = (await import((0, node_module.createRequire)(config.root).resolve("vite"))).preprocessCSS;
1040
+ } catch {}
1041
+ },
1042
+ async load(id) {
1043
+ if (request.is.colocated(id)) {
1044
+ const parsed = request.colocated.decode(id);
1045
+ let code = (0, node_fs.readFileSync)(parsed.fileName, "utf-8");
1046
+ let relativeFilePath = node_path.default.relative(CWD$1, parsed.fileName);
1047
+ const ext = node_path.default.extname(parsed.fileName).toLowerCase();
1048
+ if (PREPROCESSED_EXTENSIONS.has(ext)) {
1049
+ if (!viteConfig || !preprocessCSS) throw new Error(`[ember-scoped-css] Colocated CSS file with extension '${ext}' requires Vite. CSS preprocessing is only supported in Vite builds.`);
1050
+ code = (await preprocessCSS(code, parsed.fileName, viteConfig)).code;
1051
+ }
1052
+ return rewriteCss(code, parsed.postfix, relativeFilePath, options.layerName);
1053
+ }
1054
+ }
1055
+ }
1056
+ };
1057
+ }
1058
+
1059
+ //#endregion
1060
+ //#region src/build/unplugin-inline.js
1061
+ const META = "scoped-css:inline";
1062
+ /**
1063
+ * Plugin for supporting the styles from
1064
+ *
1065
+ * <template>
1066
+ * <style>...</style>
1067
+ * </template>
1068
+ *
1069
+ * This plugin can't have HMR for CSS because changes to the CSS content alters the template content
1070
+ */
1071
+ function inline(options = {}) {
1072
+ const CWD$1 = process.cwd();
1073
+ /** @type {import('vite').ResolvedConfig | undefined} */
1074
+ let viteConfig;
1075
+ /** @type {((code: string, filename: string, config: unknown) => Promise<{ code: string }>) | undefined} */
1076
+ let preprocessCSS;
1077
+ /**
1078
+ * @param {string} id the request id / what was imported
1079
+ */
1080
+ function buildResponse(id, filePath) {
1081
+ const parsed = request.inline.decode(id);
1082
+ const relativeFilePath = node_path.default.relative(CWD$1, filePath);
1083
+ return {
1084
+ id: filePath.split("?")[0],
1085
+ meta: { [META]: {
1086
+ rawCss: parsed.css,
1087
+ postfix: parsed.postfix,
1088
+ fileName: relativeFilePath,
1089
+ lang: parsed.lang
1090
+ } }
1091
+ };
1092
+ }
1093
+ return {
1094
+ name: "ember-scoped-css:inline",
1095
+ resolveId(id, importer) {
1096
+ if (request.is.inline(id)) {
1097
+ const parsed = request.inline.decode(id);
1098
+ return buildResponse(id, node_path.default.resolve(node_path.default.dirname(importer), `${node_path.default.basename(importer, node_path.default.extname(importer))}-${parsed.hash}.css`));
1099
+ }
1100
+ },
1101
+ async load(id) {
1102
+ const meta = this.getModuleInfo(id)?.meta?.[META];
1103
+ if (meta) {
1104
+ let rawCss = meta.rawCss;
1105
+ if (meta.lang) {
1106
+ if (!viteConfig || !preprocessCSS) throw new Error(`[ember-scoped-css] <style scoped lang="${meta.lang}"> requires Vite. CSS preprocessing via the 'lang' attribute is only supported in Vite builds.`);
1107
+ const fakeFilename = `${meta.fileName}.${meta.lang}`;
1108
+ rawCss = (await preprocessCSS(rawCss, fakeFilename, viteConfig)).code;
1109
+ }
1110
+ return rewriteCss(rawCss, meta.postfix, `<inline>/${meta.fileName}`, options.layerName);
1111
+ }
1112
+ },
1113
+ vite: { async configResolved(config) {
1114
+ viteConfig = config;
1115
+ try {
1116
+ preprocessCSS = (await import((0, node_module.createRequire)(node_path.default.join(config.root, "package.json")).resolve("vite"))).preprocessCSS;
1117
+ } catch {}
1118
+ } }
1119
+ };
1120
+ }
1121
+
1122
+ //#endregion
1123
+ //#region src/build/scoped-css-unplugin.js
1124
+ /**
1125
+ * The plugin that handles CSS requests for `<style>` elements and transforms
1126
+ * for existing files
1127
+ *
1128
+ * vite: CSS files are resolved by vite. We use their resolver to also get
1129
+ * HMR. That is, for all non-physical CSS files, we extend vite by our
1130
+ * resolver and also can enrich metadata to it (for better debugging)
1131
+ */
1132
+ const unplugin$1 = (0, unplugin.createUnplugin)((options = {}) => {
1133
+ return [colocated(options), inline(options)];
1134
+ });
1135
+
1136
+ //#endregion
1137
+ //#region src/lib/path/template-transform-paths.js
1138
+ /**
1139
+ * template plugins do not hand us the correct file path.
1140
+ * additionally, we may not be able to rely on this data in the future,
1141
+ * so this functions acts as a means of normalizing _whatever_ we're given
1142
+ * in the future.
1143
+ *
1144
+ * @param {string} filename
1145
+ * @returns {string} the absolute path to the file
1146
+ */
1147
+ function fixFilename(filename) {
1148
+ let fileName = filename;
1149
+ let workspace = findWorkspacePath(fileName);
1150
+ /**
1151
+ * ember-source 5.8:
1152
+ * - the filename looks like an absolute path, but swapped out the 'app' part of the path
1153
+ * with the module name, so the file paths never exist on disk
1154
+ *
1155
+ * - in vite apps:
1156
+ * the 'app' part _may_ be `src`, so we also need to ensure that `src` is excluded as well
1157
+ */
1158
+ let hasAppDir = fileName.includes(node_path.default.join(workspace, "app"));
1159
+ let hasSrcDir = fileName.includes(node_path.default.join(workspace, "src"));
1160
+ if (!(hasAppDir || hasSrcDir) && !fileName.includes(leadingSlashPath.embroiderDir)) {
1161
+ let [maybeScope, ...rest] = fileName.replace(workspace, "").split(node_path.default.sep).filter(Boolean);
1162
+ let parts = rest;
1163
+ if (maybeScope.startsWith("@")) {
1164
+ let [, ...rester] = rest;
1165
+ parts = rester;
1166
+ }
1167
+ let relative = node_path.default.join(...parts);
1168
+ return node_path.default.join(workspace, "app", relative);
1169
+ }
1170
+ if (!fileName.includes(workspace)) return fileName;
1171
+ return fileName;
1172
+ }
1173
+
1174
+ //#endregion
1175
+ //#region src/lib/rewriteHbs.js
1176
+ function templatePlugin({ classes, tags, postfix }) {
1177
+ let stack = [];
1178
+ let scopedClassCandidates = ["scoped-class", "scopedClass"];
1179
+ function isScopedClass(str) {
1180
+ if (!str) return false;
1181
+ return scopedClassCandidates.some((candidate) => candidate === str);
1182
+ }
1183
+ return {
1184
+ AttrNode(node) {
1185
+ if (node.name === "class") {
1186
+ if (node.value.type === "TextNode" && node.value.chars) {
1187
+ const renamedClass = renameClass(node.value.chars, postfix, classes);
1188
+ node.value.chars = renamedClass;
1189
+ } else if (node.value.type === "ConcatStatement") {
1190
+ for (let part of node.value.parts) if (part.type === "TextNode" && part.chars) part.chars = renameClass(part.chars, postfix, classes);
1191
+ else if (part.type === "MustacheStatement") ember_template_recast.traverse(part, { StringLiteral(node$1) {
1192
+ node$1.value = renameClass(node$1.value, postfix, classes);
1193
+ } });
1194
+ }
1195
+ }
1196
+ },
1197
+ ElementNode(node) {
1198
+ if (tags.has(node.tag)) {
1199
+ const classAttr = node.attributes.find((attr) => attr.name === "class");
1200
+ if (classAttr) classAttr.value.chars += " " + postfix;
1201
+ else node.attributes.push(ember_template_recast.builders.attr("class", ember_template_recast.builders.text(postfix)));
1202
+ }
1203
+ },
1204
+ All: {
1205
+ enter(node) {
1206
+ stack.push(node);
1207
+ },
1208
+ exit() {
1209
+ stack.pop();
1210
+ }
1211
+ },
1212
+ MustacheStatement(node) {
1213
+ let cssClass;
1214
+ if (isScopedClass(getValue(node.path)) && node.params?.length === 1 && node.params[0].type === "StringLiteral") cssClass = node.params[0].value;
1215
+ if (isScopedClass(getValue(node.path?.path)) && node.path?.params?.length === 1 && node.path?.params[0].type === "StringLiteral") cssClass = node.path.params[0].value;
1216
+ if (cssClass) {
1217
+ const textNode = ember_template_recast.builders.text(renameClass(cssClass, postfix));
1218
+ const parent = stack[stack.length - 1];
1219
+ if (parent?.type === "AttrNode") parent.quoteType = "\"";
1220
+ return textNode;
1221
+ }
1222
+ },
1223
+ SubExpression(node) {
1224
+ if (isScopedClass(getValue(node.path)) && node.params?.length === 1 && node.params[0].type === "StringLiteral") {
1225
+ const cssClass = node.params[0].value;
1226
+ return ember_template_recast.builders.literal("StringLiteral", renameClass(cssClass, postfix));
1227
+ }
1228
+ }
1229
+ };
1230
+ }
1231
+ function getValue(path$8) {
1232
+ if (!path$8) return;
1233
+ if ("value" in path$8) return path$8.value;
1234
+ /**
1235
+ * Deprecated in ember 5.9+
1236
+ * (so we use the above for newer embers)
1237
+ */
1238
+ return path$8.original;
1239
+ }
1240
+
1241
+ //#endregion
1242
+ //#region src/build/template-plugin.js
1243
+ const noopPlugin = {
1244
+ name: "ember-scoped-css:noop",
1245
+ visitor: {}
1246
+ };
1247
+ /**
1248
+ * @returns {ASTPlugin}
1249
+ */
1250
+ function createPlugin(config) {
1251
+ /**
1252
+ *
1253
+ * @param {ASTPluginEnvironment} env
1254
+ */
1255
+ return function scopedCss(env) {
1256
+ let cwd = node_process.default.cwd();
1257
+ if (!isRelevantFile(env.filename, {
1258
+ additionalRoots: config.additionalRoots,
1259
+ cwd
1260
+ })) return noopPlugin;
1261
+ let absolutePath = fixFilename(env.filename);
1262
+ let postfix = hashFromModulePath(appPath(absolutePath));
1263
+ /**
1264
+ * The list of naked tag selectors found in the CSS
1265
+ *
1266
+ * @type {Set<string>}
1267
+ */
1268
+ let scopedTags = /* @__PURE__ */ new Set();
1269
+ /**
1270
+ * The list of classes found in the CSS
1271
+ *
1272
+ * @type {Set<string>}
1273
+ */
1274
+ let scopedClasses = /* @__PURE__ */ new Set();
1275
+ /**
1276
+ * @param {{ tags: Set<string>; classes: Set<string> }} info
1277
+ */
1278
+ function addInfo(info$1) {
1279
+ for (let item of info$1.tags) scopedTags.add(item);
1280
+ for (let item of info$1.classes) scopedClasses.add(item);
1281
+ }
1282
+ let cssPath = cssPathFor(absolutePath);
1283
+ let info = getCSSInfo(cssPath);
1284
+ let localCssPath = forcePosix(cssPath.replace(cwd + node_path.default.sep, ""));
1285
+ /**
1286
+ * This will be falsey if we don't have a co-located CSS file.
1287
+ * We'll still want to check for embedded <style scoped> tags though.
1288
+ */
1289
+ if (info) {
1290
+ addInfo(info);
1291
+ let cssRequest = request.colocated.create(info.id, postfix, localCssPath);
1292
+ /**
1293
+ * With this we don't need a JS plugin
1294
+ */
1295
+ env.meta.jsutils.importForSideEffect(cssRequest);
1296
+ }
1297
+ let visitors = templatePlugin({
1298
+ classes: scopedClasses,
1299
+ tags: scopedTags,
1300
+ postfix
1301
+ });
1302
+ return {
1303
+ name: "ember-scoped-css:template-plugin",
1304
+ visitor: {
1305
+ ...visitors,
1306
+ Template(node) {
1307
+ /**
1308
+ * We only allow a scoped <style> at the root
1309
+ */
1310
+ let styleTag = node.body.find((n) => n.type === "ElementNode" && n.tag === "style");
1311
+ if (hasScopedAttribute(styleTag)) {
1312
+ let css = textContent(styleTag);
1313
+ let lang = getLangAttribute(styleTag);
1314
+ let info$1 = getCSSContentInfo(css, lang);
1315
+ addInfo(info$1);
1316
+ if (hasInlineAttributeWithoutLang(styleTag))
1317
+ /**
1318
+ * This will be handled in ElementNode traversal
1319
+ */
1320
+ return;
1321
+ if (lang)
1322
+ /**
1323
+ * For <style scoped inline lang="..."> we cannot preprocess at Babel-time
1324
+ * (preprocessing is async and requires Vite's ResolvedConfig).
1325
+ * Remove the tag and inject via virtual CSS module and warn user.
1326
+ */
1327
+ console.warn(`[ember-scoped-css] <style scoped inline lang="${lang}"> is not supported (preprocessing is async and cannot run at Babel-time). Downgrading to non-inline: the style tag will be removed and injected as a virtual CSS module.`);
1328
+ let cssRequest = request.inline.create(info$1.id, postfix, css, lang);
1329
+ env.meta.jsutils.importForSideEffect(cssRequest);
1330
+ }
1331
+ },
1332
+ AttrNode(...args) {
1333
+ return visitors.AttrNode(...args);
1334
+ },
1335
+ ElementNode(node, walker) {
1336
+ visitors.ElementNode(node, walker);
1337
+ if (hasScopedAttribute(node)) {
1338
+ if (walker.parent?.node.type !== "Template") throw new Error("<style scoped> tags must be at the root of the template, they cannot be nested");
1339
+ if (hasInlineAttributeWithoutLang(node)) {
1340
+ let scopedText = rewriteCss(textContent(node), postfix, localCssPath, config.layerName);
1341
+ /**
1342
+ * Traverse this and allow interpolation
1343
+ */
1344
+ node.children = [env.syntax.builders.text(scopedText)];
1345
+ return;
1346
+ }
1347
+ return null;
1348
+ }
1349
+ if (hasInlineAttributeWithoutLang(node)) throw new Error(`<style inline> is not valid. Please add the scoped attribute: <style scoped inline>`);
1350
+ },
1351
+ MustacheStatement(...args) {
1352
+ return visitors.MustacheStatement(...args);
1353
+ },
1354
+ SubExpression(...args) {
1355
+ return visitors.SubExpression(...args);
1356
+ }
1357
+ }
1358
+ };
1359
+ };
1360
+ }
1361
+ /**
1362
+ * Thanks, CardStack and @ef4 for this code.
1363
+ */
1364
+ const SCOPED_ATTRIBUTE_NAME = "scoped";
1365
+ const INLINE_ATTRIBUTE_NAME = "inline";
1366
+ const LANG_ATTRIBUTE_NAME = "lang";
1367
+ function hasScopedAttribute(node) {
1368
+ if (!node) return;
1369
+ if (node.tag !== "style") return;
1370
+ if (node.type !== "ElementNode") return;
1371
+ return node.attributes.some((attribute) => attribute.name === SCOPED_ATTRIBUTE_NAME);
1372
+ }
1373
+ function hasInlineAttributeWithoutLang(node) {
1374
+ if (!node) return;
1375
+ if (node.tag !== "style") return;
1376
+ if (node.type !== "ElementNode") return;
1377
+ if (getLangAttribute(node)) return false;
1378
+ return node.attributes.some((attribute) => attribute.name === INLINE_ATTRIBUTE_NAME);
1379
+ }
1380
+ /**
1381
+ * Returns the value of the `lang` attribute on a `<style>` node, or null if absent.
1382
+ *
1383
+ * @param {object} node
1384
+ * @returns {string | null}
1385
+ */
1386
+ function getLangAttribute(node) {
1387
+ if (!node) return null;
1388
+ if (node.tag !== "style") return null;
1389
+ if (node.type !== "ElementNode") return null;
1390
+ const attr = node.attributes.find((attribute) => attribute.name === LANG_ATTRIBUTE_NAME);
1391
+ if (!attr) return null;
1392
+ const value = attr.value;
1393
+ if (value?.type === "TextNode") return value.chars || null;
1394
+ return null;
1395
+ }
1396
+ function textContent(node) {
1397
+ return node.children.filter((c) => c.type === "TextNode").map((c) => c.chars).join("");
1398
+ }
1399
+
1400
+ //#endregion
1401
+ //#region src/build/public-exports/all.js
1402
+ const scopedCSS = {
1403
+ vite: unplugin$1.vite,
1404
+ rollup: unplugin$1.rollup,
1405
+ babel: scopedCSS$1,
1406
+ template: createPlugin
1407
+ };
1408
+
1409
+ //#endregion
1410
+ Object.defineProperty(exports, 'scopedCSS', {
1411
+ enumerable: true,
1412
+ get: function () {
1413
+ return scopedCSS;
1414
+ }
1415
+ });
1416
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWxsLUR4SU54dGJwLmNqcyIsIm5hbWVzIjpbImhhc2giLCJoYXNoIiwiaGFzaEZyb21Nb2R1bGVQYXRoIiwiaGFzaFBvc2l4TW9kdWxlUGF0aCIsInBhdGgiLCJmc1N5bmMiLCJzY29wZWRDU1MiLCJwYXRoIiwic2Nzc1N5bnRheCIsIlNFUCIsInBhcnNlciIsInJlcXVlc3QiLCJoYXNoIiwicGF0aCIsIk1FVEEiLCJDV0QiLCJwYXRoIiwiQ1dEIiwicGF0aCIsInVucGx1Z2luIiwicGF0aCIsIm5vZGUiLCJyZWNhc3QiLCJwYXRoIiwicHJvY2VzcyIsImluZm8iLCJwYXRoIiwidW5wbHVnaW4iLCJiYWJlbC5zY29wZWRDU1MiLCJ0ZW1wbGF0ZS5jcmVhdGVQbHVnaW4iXSwic291cmNlcyI6WyIuLi8uLi9zcmMvbGliL3BhdGgvY29uc3QuanMiLCIuLi8uLi9zcmMvbGliL3BhdGgvbWQ1LmpzIiwiLi4vLi4vc3JjL2xpYi9wYXRoL2hhc2gtZnJvbS1tb2R1bGUtcGF0aC5qcyIsIi4uLy4uL3NyYy9saWIvcGF0aC91dGlscy5qcyIsIi4uLy4uL3NyYy9saWIvcmVuYW1lQ2xhc3MuanMiLCIuLi8uLi9zcmMvYnVpbGQvYmFiZWwtcGx1Z2luLmpzIiwiLi4vLi4vc3JjL2xpYi9jc3MvdXRpbHMuanMiLCIuLi8uLi9zcmMvbGliL2Nzcy9yZXdyaXRlLmpzIiwiLi4vLi4vc3JjL2xpYi9yZXF1ZXN0LmpzIiwiLi4vLi4vc3JjL2J1aWxkL3VucGx1Z2luLWNvbG9jYXRlZC5qcyIsIi4uLy4uL3NyYy9idWlsZC91bnBsdWdpbi1pbmxpbmUuanMiLCIuLi8uLi9zcmMvYnVpbGQvc2NvcGVkLWNzcy11bnBsdWdpbi5qcyIsIi4uLy4uL3NyYy9saWIvcGF0aC90ZW1wbGF0ZS10cmFuc2Zvcm0tcGF0aHMuanMiLCIuLi8uLi9zcmMvbGliL3Jld3JpdGVIYnMuanMiLCIuLi8uLi9zcmMvYnVpbGQvdGVtcGxhdGUtcGx1Z2luLmpzIiwiLi4vLi4vc3JjL2J1aWxkL3B1YmxpYy1leHBvcnRzL2FsbC5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgcGF0aCBmcm9tICdwYXRoJztcblxuLyoqXG4gKiBKb2luIHdpbGwgY29udmVydCB0byB3aGF0ZXZlciBpcyBhcHByb3ByaWF0ZSBmcm8gdGhlIGN1cnJlbnQgcGxhdGZvcm1cbiAqL1xuXG5leHBvcnQgY29uc3QgbGVhZGluZ1NsYXNoUGF0aCA9IHtcbiAgZW1icm9pZGVyRGlyOiBwYXRoLmpvaW4oJy9ub2RlX21vZHVsZXMvLmVtYnJvaWRlci8nKSxcbiAgYXRFbWJyb2lkZXI6IHBhdGguam9pbignL0BlbWJyb2lkZXInKSxcbiAgY29tcG9uZW50c0RpcjogcGF0aC5qb2luKCcvY29tcG9uZW50cy8nKSxcbiAgdGVtcGxhdGVzRGlyOiBwYXRoLmpvaW4oJy90ZW1wbGF0ZXMvJyksXG4gIHJvdXRlc0RpcjogcGF0aC5qb2luKCcvcm91dGVzLycpLFxuICB0ZXN0ZW06IHBhdGguam9pbignL3Rlc3RlbScpLFxuICBzcmM6IHBhdGguam9pbignL3NyYy8nKSxcbiAgYXBwOiBwYXRoLmpvaW4oJy9hcHAvJyksXG59O1xuXG5leHBvcnQgY29uc3QgYmFyZVBhdGggPSB7XG4gIHBucG1EaXI6IHBhdGguam9pbignbm9kZV9tb2R1bGVzLy5wbnBtJyksXG59O1xuIiwiLypcbiAqIEphdmFTY3JpcHQgTUQ1XG4gKiBodHRwczovL2dpdGh1Yi5jb20vYmx1ZWltcC9KYXZhU2NyaXB0LU1ENVxuICpcbiAqIENvcHlyaWdodCAyMDExLCBTZWJhc3RpYW4gVHNjaGFuXG4gKiBodHRwczovL2JsdWVpbXAubmV0XG4gKlxuICogTGljZW5zZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlOlxuICogaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVRcbiAqXG4gKiBCYXNlZCBvblxuICogQSBKYXZhU2NyaXB0IGltcGxlbWVudGF0aW9uIG9mIHRoZSBSU0EgRGF0YSBTZWN1cml0eSwgSW5jLiBNRDUgTWVzc2FnZVxuICogRGlnZXN0IEFsZ29yaXRobSwgYXMgZGVmaW5lZCBpbiBSRkMgMTMyMS5cbiAqIFZlcnNpb24gMi4yIENvcHlyaWdodCAoQykgUGF1bCBKb2huc3RvbiAxOTk5IC0gMjAwOVxuICogT3RoZXIgY29udHJpYnV0b3JzOiBHcmVnIEhvbHQsIEFuZHJldyBLZXBlcnQsIFlkbmFyLCBMb3N0aW5ldFxuICogRGlzdHJpYnV0ZWQgdW5kZXIgdGhlIEJTRCBMaWNlbnNlXG4gKiBTZWUgaHR0cDovL3BhamhvbWUub3JnLnVrL2NyeXB0L21kNSBmb3IgbW9yZSBpbmZvLlxuICpcbiAqIE1vZGlmaWNhdGlvbnM6XG4gKiAtIHJlbW92ZWQgSUlGRSwgZXhwb3J0ZWQgbWQ1ICh0aGlzIGZpbGUgaXMgbm93IEVTTSlcbiAqL1xuXG4vKipcbiAqIEFkZCBpbnRlZ2Vycywgd3JhcHBpbmcgYXQgMl4zMi5cbiAqIFRoaXMgdXNlcyAxNi1iaXQgb3BlcmF0aW9ucyBpbnRlcm5hbGx5IHRvIHdvcmsgYXJvdW5kIGJ1Z3MgaW4gaW50ZXJwcmV0ZXJzLlxuICpcbiAqIEBwYXJhbSB7bnVtYmVyfSB4IEZpcnN0IGludGVnZXJcbiAqIEBwYXJhbSB7bnVtYmVyfSB5IFNlY29uZCBpbnRlZ2VyXG4gKiBAcmV0dXJucyB7bnVtYmVyfSBTdW1cbiAqL1xuZnVuY3Rpb24gc2FmZUFkZCh4LCB5KSB7XG4gIHZhciBsc3cgPSAoeCAmIDB4ZmZmZikgKyAoeSAmIDB4ZmZmZik7XG4gIHZhciBtc3cgPSAoeCA+PiAxNikgKyAoeSA+PiAxNikgKyAobHN3ID4+IDE2KTtcblxuICByZXR1cm4gKG1zdyA8PCAxNikgfCAobHN3ICYgMHhmZmZmKTtcbn1cblxuLyoqXG4gKiBCaXR3aXNlIHJvdGF0ZSBhIDMyLWJpdCBudW1iZXIgdG8gdGhlIGxlZnQuXG4gKlxuICogQHBhcmFtIHtudW1iZXJ9IG51bSAzMi1iaXQgbnVtYmVyXG4gKiBAcGFyYW0ge251bWJlcn0gY250IFJvdGF0aW9uIGNvdW50XG4gKiBAcmV0dXJucyB7bnVtYmVyfSBSb3RhdGVkIG51bWJlclxuICovXG5mdW5jdGlvbiBiaXRSb3RhdGVMZWZ0KG51bSwgY250KSB7XG4gIHJldHVybiAobnVtIDw8IGNudCkgfCAobnVtID4+PiAoMzIgLSBjbnQpKTtcbn1cblxuLyoqXG4gKiBCYXNpYyBvcGVyYXRpb24gdGhlIGFsZ29yaXRobSB1c2VzLlxuICpcbiAqIEBwYXJhbSB7bnVtYmVyfSBxIHFcbiAqIEBwYXJhbSB7bnVtYmVyfSBhIGFcbiAqIEBwYXJhbSB7bnVtYmVyfSBiIGJcbiAqIEBwYXJhbSB7bnVtYmVyfSB4IHhcbiAqIEBwYXJhbSB7bnVtYmVyfSBzIHNcbiAqIEBwYXJhbSB7bnVtYmVyfSB0IHRcbiAqIEByZXR1cm5zIHtudW1iZXJ9IFJlc3VsdFxuICovXG5mdW5jdGlvbiBtZDVjbW4ocSwgYSwgYiwgeCwgcywgdCkge1xuICByZXR1cm4gc2FmZUFkZChiaXRSb3RhdGVMZWZ0KHNhZmVBZGQoc2FmZUFkZChhLCBxKSwgc2FmZUFkZCh4LCB0KSksIHMpLCBiKTtcbn1cblxuLyoqXG4gKiBCYXNpYyBvcGVyYXRpb24gdGhlIGFsZ29yaXRobSB1c2VzLlxuICpcbiAqIEBwYXJhbSB7bnVtYmVyfSBhIGFcbiAqIEBwYXJhbSB7bnVtYmVyfSBiIGJcbiAqIEBwYXJhbSB7bnVtYmVyfSBjIGNcbiAqIEBwYXJhbSB7bnVtYmVyfSBkIGRcbiAqIEBwYXJhbSB7bnVtYmVyfSB4IHhcbiAqIEBwYXJhbSB7bnVtYmVyfSBzIHNcbiAqIEBwYXJhbSB7bnVtYmVyfSB0IHRcbiAqIEByZXR1cm5zIHtudW1iZXJ9IFJlc3VsdFxuICovXG5mdW5jdGlvbiBtZDVmZihhLCBiLCBjLCBkLCB4LCBzLCB0KSB7XG4gIHJldHVybiBtZDVjbW4oKGIgJiBjKSB8ICh+YiAmIGQpLCBhLCBiLCB4LCBzLCB0KTtcbn1cblxuLyoqXG4gKiBCYXNpYyBvcGVyYXRpb24gdGhlIGFsZ29yaXRobSB1c2VzLlxuICpcbiAqIEBwYXJhbSB7bnVtYmVyfSBhIGFcbiAqIEBwYXJhbSB7bnVtYmVyfSBiIGJcbiAqIEBwYXJhbSB7bnVtYmVyfSBjIGNcbiAqIEBwYXJhbSB7bnVtYmVyfSBkIGRcbiAqIEBwYXJhbSB7bnVtYmVyfSB4IHhcbiAqIEBwYXJhbSB7bnVtYmVyfSBzIHNcbiAqIEBwYXJhbSB7bnVtYmVyfSB0IHRcbiAqIEByZXR1cm5zIHtudW1iZXJ9IFJlc3VsdFxuICovXG5mdW5jdGlvbiBtZDVnZyhhLCBiLCBjLCBkLCB4LCBzLCB0KSB7XG4gIHJldHVybiBtZDVjbW4oKGIgJiBkKSB8IChjICYgfmQpLCBhLCBiLCB4LCBzLCB0KTtcbn1cblxuLyoqXG4gKiBCYXNpYyBvcGVyYXRpb24gdGhlIGFsZ29yaXRobSB1c2VzLlxuICpcbiAqIEBwYXJhbSB7bnVtYmVyfSBhIGFcbiAqIEBwYXJhbSB7bnVtYmVyfSBiIGJcbiAqIEBwYXJhbSB7bnVtYmVyfSBjIGNcbiAqIEBwYXJhbSB7bnVtYmVyfSBkIGRcbiAqIEBwYXJhbSB7bnVtYmVyfSB4IHhcbiAqIEBwYXJhbSB7bnVtYmVyfSBzIHNcbiAqIEBwYXJhbSB7bnVtYmVyfSB0IHRcbiAqIEByZXR1cm5zIHtudW1iZXJ9IFJlc3VsdFxuICovXG5mdW5jdGlvbiBtZDVoaChhLCBiLCBjLCBkLCB4LCBzLCB0KSB7XG4gIHJldHVybiBtZDVjbW4oYiBeIGMgXiBkLCBhLCBiLCB4LCBzLCB0KTtcbn1cblxuLyoqXG4gKiBCYXNpYyBvcGVyYXRpb24gdGhlIGFsZ29yaXRobSB1c2VzLlxuICpcbiAqIEBwYXJhbSB7bnVtYmVyfSBhIGFcbiAqIEBwYXJhbSB7bnVtYmVyfSBiIGJcbiAqIEBwYXJhbSB7bnVtYmVyfSBjIGNcbiAqIEBwYXJhbSB7bnVtYmVyfSBkIGRcbiAqIEBwYXJhbSB7bnVtYmVyfSB4IHhcbiAqIEBwYXJhbSB7bnVtYmVyfSBzIHNcbiAqIEBwYXJhbSB7bnVtYmVyfSB0IHRcbiAqIEByZXR1cm5zIHtudW1iZXJ9IFJlc3VsdFxuICovXG5mdW5jdGlvbiBtZDVpaShhLCBiLCBjLCBkLCB4LCBzLCB0KSB7XG4gIHJldHVybiBtZDVjbW4oYyBeIChiIHwgfmQpLCBhLCBiLCB4LCBzLCB0KTtcbn1cblxuLyoqXG4gKiBDYWxjdWxhdGUgdGhlIE1ENSBvZiBhbiBhcnJheSBvZiBsaXR0bGUtZW5kaWFuIHdvcmRzLCBhbmQgYSBiaXQgbGVuZ3RoLlxuICpcbiAqIEBwYXJhbSB7QXJyYXl9IHggQXJyYXkgb2YgbGl0dGxlLWVuZGlhbiB3b3Jkc1xuICogQHBhcmFtIHtudW1iZXJ9IGxlbiBCaXQgbGVuZ3RoXG4gKiBAcmV0dXJucyB7QXJyYXk8bnVtYmVyPn0gTUQ1IEFycmF5XG4gKi9cbmZ1bmN0aW9uIGJpbmxNRDUoeCwgbGVuKSB7XG4gIC8qIGFwcGVuZCBwYWRkaW5nICovXG4gIHhbbGVuID4+IDVdIHw9IDB4ODAgPDwgbGVuICUgMzI7XG4gIHhbKCgobGVuICsgNjQpID4+PiA5KSA8PCA0KSArIDE0XSA9IGxlbjtcblxuICB2YXIgaTtcbiAgdmFyIG9sZGE7XG4gIHZhciBvbGRiO1xuICB2YXIgb2xkYztcbiAgdmFyIG9sZGQ7XG4gIHZhciBhID0gMTczMjU4NDE5MztcbiAgdmFyIGIgPSAtMjcxNzMzODc5O1xuICB2YXIgYyA9IC0xNzMyNTg0MTk0O1xuICB2YXIgZCA9IDI3MTczMzg3ODtcblxuICBmb3IgKGkgPSAwOyBpIDwgeC5sZW5ndGg7IGkgKz0gMTYpIHtcbiAgICBvbGRhID0gYTtcbiAgICBvbGRiID0gYjtcbiAgICBvbGRjID0gYztcbiAgICBvbGRkID0gZDtcblxuICAgIGEgPSBtZDVmZihhLCBiLCBjLCBkLCB4W2ldLCA3LCAtNjgwODc2OTM2KTtcbiAgICBkID0gbWQ1ZmYoZCwgYSwgYiwgYywgeFtpICsgMV0sIDEyLCAtMzg5NTY0NTg2KTtcbiAgICBjID0gbWQ1ZmYoYywgZCwgYSwgYiwgeFtpICsgMl0sIDE3LCA2MDYxMDU4MTkpO1xuICAgIGIgPSBtZDVmZihiLCBjLCBkLCBhLCB4W2kgKyAzXSwgMjIsIC0xMDQ0NTI1MzMwKTtcbiAgICBhID0gbWQ1ZmYoYSwgYiwgYywgZCwgeFtpICsgNF0sIDcsIC0xNzY0MTg4OTcpO1xuICAgIGQgPSBtZDVmZihkLCBhLCBiLCBjLCB4W2kgKyA1XSwgMTIsIDEyMDAwODA0MjYpO1xuICAgIGMgPSBtZDVmZihjLCBkLCBhLCBiLCB4W2kgKyA2XSwgMTcsIC0xNDczMjMxMzQxKTtcbiAgICBiID0gbWQ1ZmYoYiwgYywgZCwgYSwgeFtpICsgN10sIDIyLCAtNDU3MDU5ODMpO1xuICAgIGEgPSBtZDVmZihhLCBiLCBjLCBkLCB4W2kgKyA4XSwgNywgMTc3MDAzNTQxNik7XG4gICAgZCA9IG1kNWZmKGQsIGEsIGIsIGMsIHhbaSArIDldLCAxMiwgLTE5NTg0MTQ0MTcpO1xuICAgIGMgPSBtZDVmZihjLCBkLCBhLCBiLCB4W2kgKyAxMF0sIDE3LCAtNDIwNjMpO1xuICAgIGIgPSBtZDVmZihiLCBjLCBkLCBhLCB4W2kgKyAxMV0sIDIyLCAtMTk5MDQwNDE2Mik7XG4gICAgYSA9IG1kNWZmKGEsIGIsIGMsIGQsIHhbaSArIDEyXSwgNywgMTgwNDYwMzY4Mik7XG4gICAgZCA9IG1kNWZmKGQsIGEsIGIsIGMsIHhbaSArIDEzXSwgMTIsIC00MDM0MTEwMSk7XG4gICAgYyA9IG1kNWZmKGMsIGQsIGEsIGIsIHhbaSArIDE0XSwgMTcsIC0xNTAyMDAyMjkwKTtcbiAgICBiID0gbWQ1ZmYoYiwgYywgZCwgYSwgeFtpICsgMTVdLCAyMiwgMTIzNjUzNTMyOSk7XG5cbiAgICBhID0gbWQ1Z2coYSwgYiwgYywgZCwgeFtpICsgMV0sIDUsIC0xNjU3OTY1MTApO1xuICAgIGQgPSBtZDVnZyhkLCBhLCBiLCBjLCB4W2kgKyA2XSwgOSwgLTEwNjk1MDE2MzIpO1xuICAgIGMgPSBtZDVnZyhjLCBkLCBhLCBiLCB4W2kgKyAxMV0sIDE0LCA2NDM3MTc3MTMpO1xuICAgIGIgPSBtZDVnZyhiLCBjLCBkLCBhLCB4W2ldLCAyMCwgLTM3Mzg5NzMwMik7XG4gICAgYSA9IG1kNWdnKGEsIGIsIGMsIGQsIHhbaSArIDVdLCA1LCAtNzAxNTU4NjkxKTtcbiAgICBkID0gbWQ1Z2coZCwgYSwgYiwgYywgeFtpICsgMTBdLCA5LCAzODAxNjA4Myk7XG4gICAgYyA9IG1kNWdnKGMsIGQsIGEsIGIsIHhbaSArIDE1XSwgMTQsIC02NjA0NzgzMzUpO1xuICAgIGIgPSBtZDVnZyhiLCBjLCBkLCBhLCB4W2kgKyA0XSwgMjAsIC00MDU1Mzc4NDgpO1xuICAgIGEgPSBtZDVnZyhhLCBiLCBjLCBkLCB4W2kgKyA5XSwgNSwgNTY4NDQ2NDM4KTtcbiAgICBkID0gbWQ1Z2coZCwgYSwgYiwgYywgeFtpICsgMTRdLCA5LCAtMTAxOTgwMzY5MCk7XG4gICAgYyA9IG1kNWdnKGMsIGQsIGEsIGIsIHhbaSArIDNdLCAxNCwgLTE4NzM2Mzk2MSk7XG4gICAgYiA9IG1kNWdnKGIsIGMsIGQsIGEsIHhbaSArIDhdLCAyMCwgMTE2MzUzMTUwMSk7XG4gICAgYSA9IG1kNWdnKGEsIGIsIGMsIGQsIHhbaSArIDEzXSwgNSwgLTE0NDQ2ODE0NjcpO1xuICAgIGQgPSBtZDVnZyhkLCBhLCBiLCBjLCB4W2kgKyAyXSwgOSwgLTUxNDAzNzg0KTtcbiAgICBjID0gbWQ1Z2coYywgZCwgYSwgYiwgeFtpICsgN10sIDE0LCAxNzM1MzI4NDczKTtcbiAgICBiID0gbWQ1Z2coYiwgYywgZCwgYSwgeFtpICsgMTJdLCAyMCwgLTE5MjY2MDc3MzQpO1xuXG4gICAgYSA9IG1kNWhoKGEsIGIsIGMsIGQsIHhbaSArIDVdLCA0LCAtMzc4NTU4KTtcbiAgICBkID0gbWQ1aGgoZCwgYSwgYiwgYywgeFtpICsgOF0sIDExLCAtMjAyMjU3NDQ2Myk7XG4gICAgYyA9IG1kNWhoKGMsIGQsIGEsIGIsIHhbaSArIDExXSwgMTYsIDE4MzkwMzA1NjIpO1xuICAgIGIgPSBtZDVoaChiLCBjLCBkLCBhLCB4W2kgKyAxNF0sIDIzLCAtMzUzMDk1NTYpO1xuICAgIGEgPSBtZDVoaChhLCBiLCBjLCBkLCB4W2kgKyAxXSwgNCwgLTE1MzA5OTIwNjApO1xuICAgIGQgPSBtZDVoaChkLCBhLCBiLCBjLCB4W2kgKyA0XSwgMTEsIDEyNzI4OTMzNTMpO1xuICAgIGMgPSBtZDVoaChjLCBkLCBhLCBiLCB4W2kgKyA3XSwgMTYsIC0xNTU0OTc2MzIpO1xuICAgIGIgPSBtZDVoaChiLCBjLCBkLCBhLCB4W2kgKyAxMF0sIDIzLCAtMTA5NDczMDY0MCk7XG4gICAgYSA9IG1kNWhoKGEsIGIsIGMsIGQsIHhbaSArIDEzXSwgNCwgNjgxMjc5MTc0KTtcbiAgICBkID0gbWQ1aGgoZCwgYSwgYiwgYywgeFtpXSwgMTEsIC0zNTg1MzcyMjIpO1xuICAgIGMgPSBtZDVoaChjLCBkLCBhLCBiLCB4W2kgKyAzXSwgMTYsIC03MjI1MjE5NzkpO1xuICAgIGIgPSBtZDVoaChiLCBjLCBkLCBhLCB4W2kgKyA2XSwgMjMsIDc2MDI5MTg5KTtcbiAgICBhID0gbWQ1aGgoYSwgYiwgYywgZCwgeFtpICsgOV0sIDQsIC02NDAzNjQ0ODcpO1xuICAgIGQgPSBtZDVoaChkLCBhLCBiLCBjLCB4W2kgKyAxMl0sIDExLCAtNDIxODE1ODM1KTtcbiAgICBjID0gbWQ1aGgoYywgZCwgYSwgYiwgeFtpICsgMTVdLCAxNiwgNTMwNzQyNTIwKTtcbiAgICBiID0gbWQ1aGgoYiwgYywgZCwgYSwgeFtpICsgMl0sIDIzLCAtOTk1MzM4NjUxKTtcblxuICAgIGEgPSBtZDVpaShhLCBiLCBjLCBkLCB4W2ldLCA2LCAtMTk4NjMwODQ0KTtcbiAgICBkID0gbWQ1aWkoZCwgYSwgYiwgYywgeFtpICsgN10sIDEwLCAxMTI2ODkxNDE1KTtcbiAgICBjID0gbWQ1aWkoYywgZCwgYSwgYiwgeFtpICsgMTRdLCAxNSwgLTE0MTYzNTQ5MDUpO1xuICAgIGIgPSBtZDVpaShiLCBjLCBkLCBhLCB4W2kgKyA1XSwgMjEsIC01NzQzNDA1NSk7XG4gICAgYSA9IG1kNWlpKGEsIGIsIGMsIGQsIHhbaSArIDEyXSwgNiwgMTcwMDQ4NTU3MSk7XG4gICAgZCA9IG1kNWlpKGQsIGEsIGIsIGMsIHhbaSArIDNdLCAxMCwgLTE4OTQ5ODY2MDYpO1xuICAgIGMgPSBtZDVpaShjLCBkLCBhLCBiLCB4W2kgKyAxMF0sIDE1LCAtMTA1MTUyMyk7XG4gICAgYiA9IG1kNWlpKGIsIGMsIGQsIGEsIHhbaSArIDFdLCAyMSwgLTIwNTQ5MjI3OTkpO1xuICAgIGEgPSBtZDVpaShhLCBiLCBjLCBkLCB4W2kgKyA4XSwgNiwgMTg3MzMxMzM1OSk7XG4gICAgZCA9IG1kNWlpKGQsIGEsIGIsIGMsIHhbaSArIDE1XSwgMTAsIC0zMDYxMTc0NCk7XG4gICAgYyA9IG1kNWlpKGMsIGQsIGEsIGIsIHhbaSArIDZdLCAxNSwgLTE1NjAxOTgzODApO1xuICAgIGIgPSBtZDVpaShiLCBjLCBkLCBhLCB4W2kgKyAxM10sIDIxLCAxMzA5MTUxNjQ5KTtcbiAgICBhID0gbWQ1aWkoYSwgYiwgYywgZCwgeFtpICsgNF0sIDYsIC0xNDU1MjMwNzApO1xuICAgIGQgPSBtZDVpaShkLCBhLCBiLCBjLCB4W2kgKyAxMV0sIDEwLCAtMTEyMDIxMDM3OSk7XG4gICAgYyA9IG1kNWlpKGMsIGQsIGEsIGIsIHhbaSArIDJdLCAxNSwgNzE4Nzg3MjU5KTtcbiAgICBiID0gbWQ1aWkoYiwgYywgZCwgYSwgeFtpICsgOV0sIDIxLCAtMzQzNDg1NTUxKTtcblxuICAgIGEgPSBzYWZlQWRkKGEsIG9sZGEpO1xuICAgIGIgPSBzYWZlQWRkKGIsIG9sZGIpO1xuICAgIGMgPSBzYWZlQWRkKGMsIG9sZGMpO1xuICAgIGQgPSBzYWZlQWRkKGQsIG9sZGQpO1xuICB9XG5cbiAgcmV0dXJuIFthLCBiLCBjLCBkXTtcbn1cblxuLyoqXG4gKiBDb252ZXJ0IGFuIGFycmF5IG9mIGxpdHRsZS1lbmRpYW4gd29yZHMgdG8gYSBzdHJpbmdcbiAqXG4gKiBAcGFyYW0ge0FycmF5PG51bWJlcj59IGlucHV0IE1ENSBBcnJheVxuICogQHJldHVybnMge3N0cmluZ30gTUQ1IHN0cmluZ1xuICovXG5mdW5jdGlvbiBiaW5sMnJzdHIoaW5wdXQpIHtcbiAgdmFyIGk7XG4gIHZhciBvdXRwdXQgPSAnJztcbiAgdmFyIGxlbmd0aDMyID0gaW5wdXQubGVuZ3RoICogMzI7XG5cbiAgZm9yIChpID0gMDsgaSA8IGxlbmd0aDMyOyBpICs9IDgpIHtcbiAgICBvdXRwdXQgKz0gU3RyaW5nLmZyb21DaGFyQ29kZSgoaW5wdXRbaSA+PiA1XSA+Pj4gaSAlIDMyKSAmIDB4ZmYpO1xuICB9XG5cbiAgcmV0dXJuIG91dHB1dDtcbn1cblxuLyoqXG4gKiBDb252ZXJ0IGEgcmF3IHN0cmluZyB0byBhbiBhcnJheSBvZiBsaXR0bGUtZW5kaWFuIHdvcmRzXG4gKiBDaGFyYWN0ZXJzID4yNTUgaGF2ZSB0aGVpciBoaWdoLWJ5dGUgc2lsZW50bHkgaWdub3JlZC5cbiAqXG4gKiBAcGFyYW0ge3N0cmluZ30gaW5wdXQgUmF3IGlucHV0IHN0cmluZ1xuICogQHJldHVybnMge0FycmF5PG51bWJlcj59IEFycmF5IG9mIGxpdHRsZS1lbmRpYW4gd29yZHNcbiAqL1xuZnVuY3Rpb24gcnN0cjJiaW5sKGlucHV0KSB7XG4gIHZhciBpO1xuICB2YXIgb3V0cHV0ID0gW107XG4gIG91dHB1dFsoaW5wdXQubGVuZ3RoID4+IDIpIC0gMV0gPSB1bmRlZmluZWQ7XG5cbiAgZm9yIChpID0gMDsgaSA8IG91dHB1dC5sZW5ndGg7IGkgKz0gMSkge1xuICAgIG91dHB1dFtpXSA9IDA7XG4gIH1cblxuICB2YXIgbGVuZ3RoOCA9IGlucHV0Lmxlbmd0aCAqIDg7XG5cbiAgZm9yIChpID0gMDsgaSA8IGxlbmd0aDg7IGkgKz0gOCkge1xuICAgIG91dHB1dFtpID4+IDVdIHw9IChpbnB1dC5jaGFyQ29kZUF0KGkgLyA4KSAmIDB4ZmYpIDw8IGkgJSAzMjtcbiAgfVxuXG4gIHJldHVybiBvdXRwdXQ7XG59XG5cbi8qKlxuICogQ2FsY3VsYXRlIHRoZSBNRDUgb2YgYSByYXcgc3RyaW5nXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IHMgSW5wdXQgc3RyaW5nXG4gKiBAcmV0dXJucyB7c3RyaW5nfSBSYXcgTUQ1IHN0cmluZ1xuICovXG5mdW5jdGlvbiByc3RyTUQ1KHMpIHtcbiAgcmV0dXJuIGJpbmwycnN0cihiaW5sTUQ1KHJzdHIyYmlubChzKSwgcy5sZW5ndGggKiA4KSk7XG59XG5cbi8qKlxuICogQ2FsY3VsYXRlcyB0aGUgSE1BQy1NRDUgb2YgYSBrZXkgYW5kIHNvbWUgZGF0YSAocmF3IHN0cmluZ3MpXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IGtleSBITUFDIGtleVxuICogQHBhcmFtIHtzdHJpbmd9IGRhdGEgUmF3IGlucHV0IHN0cmluZ1xuICogQHJldHVybnMge3N0cmluZ30gUmF3IE1ENSBzdHJpbmdcbiAqL1xuZnVuY3Rpb24gcnN0ckhNQUNNRDUoa2V5LCBkYXRhKSB7XG4gIHZhciBpO1xuICB2YXIgYmtleSA9IHJzdHIyYmlubChrZXkpO1xuICB2YXIgaXBhZCA9IFtdO1xuICB2YXIgb3BhZCA9IFtdO1xuICB2YXIgaGFzaDtcbiAgaXBhZFsxNV0gPSBvcGFkWzE1XSA9IHVuZGVmaW5lZDtcblxuICBpZiAoYmtleS5sZW5ndGggPiAxNikge1xuICAgIGJrZXkgPSBiaW5sTUQ1KGJrZXksIGtleS5sZW5ndGggKiA4KTtcbiAgfVxuXG4gIGZvciAoaSA9IDA7IGkgPCAxNjsgaSArPSAxKSB7XG4gICAgaXBhZFtpXSA9IGJrZXlbaV0gXiAweDM2MzYzNjM2O1xuICAgIG9wYWRbaV0gPSBia2V5W2ldIF4gMHg1YzVjNWM1YztcbiAgfVxuXG4gIGhhc2ggPSBiaW5sTUQ1KGlwYWQuY29uY2F0KHJzdHIyYmlubChkYXRhKSksIDUxMiArIGRhdGEubGVuZ3RoICogOCk7XG5cbiAgcmV0dXJuIGJpbmwycnN0cihiaW5sTUQ1KG9wYWQuY29uY2F0KGhhc2gpLCA1MTIgKyAxMjgpKTtcbn1cblxuLyoqXG4gKiBDb252ZXJ0IGEgcmF3IHN0cmluZyB0byBhIGhleCBzdHJpbmdcbiAqXG4gKiBAcGFyYW0ge3N0cmluZ30gaW5wdXQgUmF3IGlucHV0IHN0cmluZ1xuICogQHJldHVybnMge3N0cmluZ30gSGV4IGVuY29kZWQgc3RyaW5nXG4gKi9cbmZ1bmN0aW9uIHJzdHIyaGV4KGlucHV0KSB7XG4gIHZhciBoZXhUYWIgPSAnMDEyMzQ1Njc4OWFiY2RlZic7XG4gIHZhciBvdXRwdXQgPSAnJztcbiAgdmFyIHg7XG4gIHZhciBpO1xuXG4gIGZvciAoaSA9IDA7IGkgPCBpbnB1dC5sZW5ndGg7IGkgKz0gMSkge1xuICAgIHggPSBpbnB1dC5jaGFyQ29kZUF0KGkpO1xuICAgIG91dHB1dCArPSBoZXhUYWIuY2hhckF0KCh4ID4+PiA0KSAmIDB4MGYpICsgaGV4VGFiLmNoYXJBdCh4ICYgMHgwZik7XG4gIH1cblxuICByZXR1cm4gb3V0cHV0O1xufVxuXG4vKipcbiAqIEVuY29kZSBhIHN0cmluZyBhcyBVVEYtOFxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBpbnB1dCBJbnB1dCBzdHJpbmdcbiAqIEByZXR1cm5zIHtzdHJpbmd9IFVURjggc3RyaW5nXG4gKi9cbmZ1bmN0aW9uIHN0cjJyc3RyVVRGOChpbnB1dCkge1xuICByZXR1cm4gdW5lc2NhcGUoZW5jb2RlVVJJQ29tcG9uZW50KGlucHV0KSk7XG59XG5cbi8qKlxuICogRW5jb2RlcyBpbnB1dCBzdHJpbmcgYXMgcmF3IE1ENSBzdHJpbmdcbiAqXG4gKiBAcGFyYW0ge3N0cmluZ30gcyBJbnB1dCBzdHJpbmdcbiAqIEByZXR1cm5zIHtzdHJpbmd9IFJhdyBNRDUgc3RyaW5nXG4gKi9cbmZ1bmN0aW9uIHJhd01ENShzKSB7XG4gIHJldHVybiByc3RyTUQ1KHN0cjJyc3RyVVRGOChzKSk7XG59XG5cbi8qKlxuICogRW5jb2RlcyBpbnB1dCBzdHJpbmcgYXMgSGV4IGVuY29kZWQgc3RyaW5nXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IHMgSW5wdXQgc3RyaW5nXG4gKiBAcmV0dXJucyB7c3RyaW5nfSBIZXggZW5jb2RlZCBzdHJpbmdcbiAqL1xuZnVuY3Rpb24gaGV4TUQ1KHMpIHtcbiAgcmV0dXJuIHJzdHIyaGV4KHJhd01ENShzKSk7XG59XG5cbi8qKlxuICogQ2FsY3VsYXRlcyB0aGUgcmF3IEhNQUMtTUQ1IGZvciB0aGUgZ2l2ZW4ga2V5IGFuZCBkYXRhXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IGsgSE1BQyBrZXlcbiAqIEBwYXJhbSB7c3RyaW5nfSBkIElucHV0IHN0cmluZ1xuICogQHJldHVybnMge3N0cmluZ30gUmF3IE1ENSBzdHJpbmdcbiAqL1xuZnVuY3Rpb24gcmF3SE1BQ01ENShrLCBkKSB7XG4gIHJldHVybiByc3RySE1BQ01ENShzdHIycnN0clVURjgoayksIHN0cjJyc3RyVVRGOChkKSk7XG59XG5cbi8qKlxuICogQ2FsY3VsYXRlcyB0aGUgSGV4IGVuY29kZWQgSE1BQy1NRDUgZm9yIHRoZSBnaXZlbiBrZXkgYW5kIGRhdGFcbiAqXG4gKiBAcGFyYW0ge3N0cmluZ30gayBITUFDIGtleVxuICogQHBhcmFtIHtzdHJpbmd9IGQgSW5wdXQgc3RyaW5nXG4gKiBAcmV0dXJucyB7c3RyaW5nfSBSYXcgTUQ1IHN0cmluZ1xuICovXG5mdW5jdGlvbiBoZXhITUFDTUQ1KGssIGQpIHtcbiAgcmV0dXJuIHJzdHIyaGV4KHJhd0hNQUNNRDUoaywgZCkpO1xufVxuXG4vKipcbiAqIENhbGN1bGF0ZXMgTUQ1IHZhbHVlIGZvciBhIGdpdmVuIHN0cmluZy5cbiAqIElmIGEga2V5IGlzIHByb3ZpZGVkLCBjYWxjdWxhdGVzIHRoZSBITUFDLU1ENSB2YWx1ZS5cbiAqIFJldHVybnMgYSBIZXggZW5jb2RlZCBzdHJpbmcgdW5sZXNzIHRoZSByYXcgYXJndW1lbnQgaXMgZ2l2ZW4uXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IHN0cmluZyBJbnB1dCBzdHJpbmdcbiAqIEBwYXJhbSB7c3RyaW5nfSBba2V5XSBITUFDIGtleVxuICogQHBhcmFtIHtib29sZWFufSBbcmF3XSBSYXcgb3V0cHV0IHN3aXRjaFxuICogQHJldHVybnMge3N0cmluZ30gTUQ1IG91dHB1dFxuICovXG5leHBvcnQgZnVuY3Rpb24gbWQ1KHN0cmluZywga2V5LCByYXcpIHtcbiAgaWYgKCFrZXkpIHtcbiAgICBpZiAoIXJhdykge1xuICAgICAgcmV0dXJuIGhleE1ENShzdHJpbmcpO1xuICAgIH1cblxuICAgIHJldHVybiByYXdNRDUoc3RyaW5nKTtcbiAgfVxuXG4gIGlmICghcmF3KSB7XG4gICAgcmV0dXJuIGhleEhNQUNNRDUoa2V5LCBzdHJpbmcpO1xuICB9XG5cbiAgcmV0dXJuIHJhd0hNQUNNRDUoa2V5LCBzdHJpbmcpO1xufVxuIiwiaW1wb3J0IHsgbWQ1IH0gZnJvbSAnLi9tZDUuanMnO1xuXG4vKipcbiAqIFRoZSBpbnRlbnQgb2YgdGhpcyBmdW5jdGlvbiBpcyB0byBnZW5lcmF0ZSB0aGUgc3VmZml4L3Bvc3RmaXggZm9yIHRoZVxuICogY3NzIGNsYXNzZXMsIGJhc2VkIG9uIHRoZSBtb2R1bGUtc2NvcGVkIHBhdGggbmFtZS5cbiAqXG4gKiBmb3IgZXhhbXBsZSxcbiAqICAgIGhhc2goJ215LWFwcC9jb21wb25lbnRzL2ZvbycpXG4gKiAgaW5zdGVhZCBvZlxuICogICAgaGFzaCgnYXBwL2NvbXBvbmVudHMvZm9vJylcbiAqXG4gKiAgKHVubGVzcyB5b3VyIGFwcCBuYW1lIGlzICdhcHAnKVxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBtb2R1bGVQYXRoXG4gKiBAcmV0dXJucyB7c3RyaW5nfVxuICovXG5leHBvcnQgZnVuY3Rpb24gaGFzaChtb2R1bGVQYXRoKSB7XG4gIHJldHVybiAnZScgKyBtZDUobW9kdWxlUGF0aCkuc3Vic3RyaW5nKDAsIDgpO1xufVxuXG5pZiAoaW1wb3J0Lm1ldGEudml0ZXN0KSB7XG4gIGNvbnN0IHsgaXQsIGV4cGVjdCB9ID0gaW1wb3J0Lm1ldGEudml0ZXN0O1xuXG4gIGl0KCdzaG91bGQgcmV0dXJuIGEgc3RyaW5nJywgZnVuY3Rpb24gKCkge1xuICAgIGNvbnN0IHBvc3RmaXggPSBoYXNoKCdmb28uY3NzJyk7XG5cbiAgICBleHBlY3QocG9zdGZpeCkudG8uYmUuYSgnc3RyaW5nJyk7XG4gIH0pO1xuXG4gIGl0KCdzaG91bGQgcmV0dXJuIGEgc3RyaW5nIHN0YXJ0aW5nIHdpdGggXCJlXCInLCBmdW5jdGlvbiAoKSB7XG4gICAgY29uc3QgcG9zdGZpeCA9IGhhc2goJ2Zvby5jc3MnKTtcblxuICAgIGV4cGVjdChwb3N0Zml4KS50by5tYXRjaCgvXmUvKTtcbiAgfSk7XG5cbiAgaXQoJ3Nob3VsZCByZXR1cm4gYSBzdHJpbmcgb2YgbGVuZ3RoIDknLCBmdW5jdGlvbiAoKSB7XG4gICAgY29uc3QgcG9zdGZpeCA9IGhhc2goJ2Zvby5jc3MnKTtcblxuICAgIGV4cGVjdChwb3N0Zml4KS50by5oYXZlLmxlbmd0aE9mKDkpO1xuICB9KTtcbn1cblxuZXhwb3J0IGNvbnN0IGhhc2hGcm9tTW9kdWxlUGF0aCA9IGhhc2g7XG4iLCJpbXBvcnQgYXNzZXJ0IGZyb20gJ25vZGU6YXNzZXJ0JztcbmltcG9ydCBmc1N5bmMsIHsgZXhpc3RzU3luYyB9IGZyb20gJ25vZGU6ZnMnO1xuaW1wb3J0IHsgY3JlYXRlUmVxdWlyZSB9IGZyb20gJ25vZGU6bW9kdWxlJztcbmltcG9ydCBwYXRoIGZyb20gJ25vZGU6cGF0aCc7XG5cbmltcG9ydCB7IGJhcmVQYXRoLCBsZWFkaW5nU2xhc2hQYXRoIH0gZnJvbSAnLi9jb25zdC5qcyc7XG5pbXBvcnQgeyBoYXNoRnJvbUFic29sdXRlUGF0aCB9IGZyb20gJy4vaGFzaC1mcm9tLWFic29sdXRlLXBhdGguanMnO1xuaW1wb3J0IHsgaGFzaEZyb21Nb2R1bGVQYXRoIGFzIGhhc2hQb3NpeE1vZHVsZVBhdGggfSBmcm9tICcuL2hhc2gtZnJvbS1tb2R1bGUtcGF0aC5qcyc7XG5cbmV4cG9ydCB7IGhhc2hGcm9tQWJzb2x1dGVQYXRoIH0gZnJvbSAnLi9oYXNoLWZyb20tYWJzb2x1dGUtcGF0aC5qcyc7XG5cbi8qKlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlUGF0aFxuICogQHJldHVybnMge3N0cmluZ31cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGhhc2hGcm9tTW9kdWxlUGF0aChmaWxlUGF0aCkge1xuICBsZXQgcG9zaXhQYXRoID0gZm9yY2VQb3NpeChmaWxlUGF0aCk7XG5cbiAgcmV0dXJuIGhhc2hQb3NpeE1vZHVsZVBhdGgocG9zaXhQYXRoKTtcbn1cblxuLyoqXG4gKiBAcGFyYW0ge3N0cmluZ30gZmlsZVBhdGhcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZvcmNlUG9zaXgoZmlsZVBhdGgpIHtcbiAgY29uc3QgcGFyc2VkID0gcGF0aC5wYXJzZShmaWxlUGF0aCk7XG5cbiAgaWYgKHBhcnNlZC5yb290ID09PSAnJykge1xuICAgIHJldHVybiBmaWxlUGF0aC5yZXBsYWNlQWxsKHBhdGgud2luMzIuc2VwLCBwYXRoLnBvc2l4LnNlcCk7XG4gIH1cblxuICBsZXQgcm9vdGxlc3MgPSBmaWxlUGF0aC5yZXBsYWNlKFxuICAgIG5ldyBSZWdFeHAoYF4ke1JlZ0V4cC5lc2NhcGUocGFyc2VkLnJvb3QpfWApLFxuICAgIHBhdGgucG9zaXguc2VwLFxuICApO1xuXG4gIHJldHVybiByb290bGVzcy5yZXBsYWNlQWxsKHBhdGgud2luMzIuc2VwLCBwYXRoLnBvc2l4LnNlcCk7XG59XG5cbmNvbnN0IENPTVBPTkVOVF9FWFRFTlNJT05TID0gWycuZ3RzJywgJy5nanMnLCAnLnRzJywgJy5qcycsICcuaGJzJ107XG5cbi8vIENKUyAvIEVTTT9cbmxldCBoZXJlID0gaW1wb3J0Lm1ldGEudXJsO1xubGV0IG91clJlcXVpcmUgPSBnbG9iYWxUaGlzLnJlcXVpcmVcbiAgPyBnbG9iYWxUaGlzLnJlcXVpcmVcbiAgOiBoZXJlICYmIGNyZWF0ZVJlcXVpcmUoaGVyZSk7XG5cbmlmICghb3VyUmVxdWlyZSkge1xuICBvdXJSZXF1aXJlID0gcmVxdWlyZTtcbn1cblxuY29uc3QgSVJSRUxFVkFOVF9QQVRIUyA9IFtiYXJlUGF0aC5wbnBtRGlyLCAnX192aXRlLSddO1xuY29uc3QgVU5TVVBQT1JURURfRElSRUNUT1JJRVMgPSBuZXcgU2V0KFsndGVzdHMnXSk7XG5cbmNvbnN0IENXRCA9IHByb2Nlc3MuY3dkKCk7XG5cbi8qKlxuICogUmVnYXJkbGVzcyBvZiB3aGF0IHRoZSBmaWxlUGF0aCBmb3JtYXQgaXMsXG4gKiB0aGlzIHdpbGwgdHJ5IHRvIHJldHVybiB0aGUgY29ycmVjdCBwb3N0Zml4LlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlUGF0aFxuICogQHJldHVybnNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGhhc2hGcm9tKGZpbGVQYXRoKSB7XG4gIGlmIChwYXRoLmlzQWJzb2x1dGUoZmlsZVBhdGgpKSB7XG4gICAgcmV0dXJuIGhhc2hGcm9tQWJzb2x1dGVQYXRoKGZpbGVQYXRoKTtcbiAgfVxuXG4gIHJldHVybiBoYXNoRnJvbU1vZHVsZVBhdGgoZmlsZVBhdGgpO1xufVxuXG4vKipcbiAqXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjc3NIYXNBc3NvY2lhdGVkQ29tcG9uZW50KGNzc1BhdGgpIHtcbiAgcmV0dXJuIGNzc0hhc1N0YW5kYXJkRmlsZShjc3NQYXRoKSB8fCBjc3NIYXNQb2RzRmlsZShjc3NQYXRoKTtcbn1cblxuZnVuY3Rpb24gY3NzSGFzU3RhbmRhcmRGaWxlKGlkKSB7XG4gIC8qKlxuICAgKiBOb3JtYWxseSB3ZSBkb24ndCBuZWVkIHRvIGNoZWNrIGEgSlMgcGF0aCBoZXJlLCBidXQgd2hlbiB1c2luZ1xuICAgKiBlbWJyb2lkZXJAMywgd2UgaGF2ZSBhIFwicmV3cml0dGVuIGFwcFwiLCB3aGljaCBoYXMgYWxsIG91ciBzb3VyY2VcbiAgICogcHJlcHJvY2Vzc2VkIGEgYml0IGJlZm9yZSBzY29wZWQtY3NzIHRyYW5zZm9ybWF0aW9ucy5cbiAgICpcbiAgICogKEluIFZpdGUsIHdlIG9wZXJhdGUgbW9yZSBkaXJlY3RseSB3aXRoIHRoZSBzb3VyY2UpXG4gICAqL1xuICBmb3IgKGxldCBleHQgb2YgQ09NUE9ORU5UX0VYVEVOU0lPTlMpIHtcbiAgICBsZXQgY2FuZGlkYXRlUGF0aCA9IGlkLnJlcGxhY2UoL1xcLmNzcyQvLCBleHQpO1xuXG4gICAgaWYgKGV4aXN0c1N5bmMoY2FuZGlkYXRlUGF0aCkpIHtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiBmYWxzZTtcbn1cblxuZnVuY3Rpb24gY3NzSGFzUG9kc0ZpbGUoaWQpIHtcbiAgaWYgKCFpZC5lbmRzV2l0aCgnc3R5bGVzLmNzcycpKSB7XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgLyoqXG4gICAqIE5vcm1hbGx5IHdlIGRvbid0IG5lZWQgdG8gY2hlY2sgYSBKUyBwYXRoIGhlcmUsIGJ1dCB3aGVuIHVzaW5nXG4gICAqIGVtYnJvaWRlckAzLCB3ZSBoYXZlIGEgXCJyZXdyaXR0ZW4gYXBwXCIsIHdoaWNoIGhhcyBhbGwgb3VyIHNvdXJjZVxuICAgKiBwcmVwcm9jZXNzZWQgYSBiaXQgYmVmb3JlIHNjb3BlZC1jc3MgdHJhbnNmb3JtYXRpb25zLlxuICAgKlxuICAgKiAoSW4gVml0ZSwgd2Ugb3BlcmF0ZSBtb3JlIGRpcmVjdGx5IHdpdGggdGhlIHNvdXJjZSlcbiAgICovXG4gIGZvciAobGV0IGV4dCBvZiBDT01QT05FTlRfRVhURU5TSU9OUykge1xuICAgIGxldCBjYW5kaWRhdGVQYXRoID0gaWQucmVwbGFjZSgvc3R5bGVzXFwuY3NzJC8sIGB0ZW1wbGF0ZSR7ZXh0fWApO1xuXG4gICAgaWYgKGV4aXN0c1N5bmMoY2FuZGlkYXRlUGF0aCkpIHtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiBmYWxzZTtcbn1cblxuLyoqXG4gKiBCYXNlZCBvbiBlbWJlcidzIGNvbXBvbmVudCBsb2NhdGlvbiBjb252ZW50aW9ucyxcbiAqIHRoaXMgZnVuY3Rpb24gd2lsbCBwcm92aWRlIGEgcGF0aCBmb3Igd2hlcmUgd2VcbiAqIGV4cGVjdCB0aGUgQ1NTIHRvIGxpdmUuXG4gKlxuICogRm9yIGNvLWxvY2F0ZWQgc3RydWN0dXJlOlxuICogICAtIGNvbXBvbmVudHMvbXktY29tcG9uZW50Lmhic1xuICogICAtIGNvbXBvbmVudHMvbXktY29tcG9uZW50LmNzc1xuICpcbiAqIEZvciBuZXN0ZWQgY28tbG9jYXRlZCBzdHJ1Y3R1cmVcbiAqICAgLSBjb21wb25lbnRzL215LWNvbXBvbmVudC9mb28uaGJzXG4gKiAgIC0gY29tcG9uZW50cy9teS1jb21wb25lbnQvZm9vLmNzc1xuICpcbiAqIEZvciBQb2RzIHJvdXRlcyBzdHJ1Y3R1cmVcbiAqICAgLSByb3V0ZXMvbXktcm91dGUvdGVtcGxhdGUue2hicyxqc31cbiAqICAgLSByb3V0ZXMvbXktcm91dGUvc3R5bGVzLmNzc1xuICpcbiAqIERlbGliZXJhdGVseSBub3Qgc3VwcG9ydGVkOlxuICogICAtIGNvbXBvbmVudHMgdy8gcG9kcyAtLSB0aGlzIGlzIGRlcHJlY2F0ZWQgaW4gNS4xMFxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlTmFtZSAtIHRoZSBoYnMsIGpzLCBnanMsIGd0cyBvciB3aGF0ZXZlciBjby1sb2NhdGVkIHBhdGguXG4gKiBAcmV0dXJucyB7c3RyaW5nfSAtIGV4cGVjdGVkIGNzcyBwYXRoXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjc3NQYXRoRm9yKGZpbGVOYW1lKSB7XG4gIGxldCB3aXRob3V0RXh0ID0gd2l0aG91dEV4dGVuc2lvbihmaWxlTmFtZSk7XG4gIGxldCBjc3NQYXRoID0gd2l0aG91dEV4dCArICcuY3NzJztcblxuICBpZiAoaXNQb2QoZmlsZU5hbWUpKSB7XG4gICAgY3NzUGF0aCA9IGZpbGVOYW1lXG4gICAgICAucmVwbGFjZSgvdGVtcGxhdGVcXC5qcyQvLCAnc3R5bGVzLmNzcycpXG4gICAgICAucmVwbGFjZSgvdGVtcGxhdGVcXC5nanMvLCAnc3R5bGVzLmNzcycpXG4gICAgICAucmVwbGFjZSgvdGVtcGxhdGVcXC5ndHMvLCAnc3R5bGVzLmNzcycpXG4gICAgICAucmVwbGFjZSgvdGVtcGxhdGVcXC5oYnMvLCAnc3R5bGVzLmNzcycpO1xuICB9XG5cbiAgcmV0dXJuIGNzc1BhdGg7XG59XG5cbi8qKlxuICogTm90ZSB0aGF0IGNvbXBvbmVudHMgaW4gdGhlIFwicG9kc1wiIGNvbnZlbnRpb24gd2lsbFxuICogbmV2ZXIgYmUgc3VwcG9ydGVkLlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlUGF0aFxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNQb2RUZW1wbGF0ZShmaWxlUGF0aCkge1xuICBpZiAoZmlsZVBhdGguaW5jbHVkZXMobGVhZGluZ1NsYXNoUGF0aC5jb21wb25lbnRzRGlyKSkge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgZmlsZVBhdGguZW5kc1dpdGgoJ3RlbXBsYXRlLmpzJykgfHxcbiAgICBmaWxlUGF0aC5lbmRzV2l0aCgndGVtcGxhdGUuaGJzJykgfHxcbiAgICBmaWxlUGF0aC5lbmRzV2l0aCgndGVtcGxhdGUuZ2pzJykgfHxcbiAgICBmaWxlUGF0aC5lbmRzV2l0aCgndGVtcGxhdGUuZ3RzJylcbiAgKTtcbn1cblxuLyoqXG4gKiBOb3RlIHRoYXQgY29tcG9uZW50cyBpbiB0aGUgXCJwb2RzXCIgY29udmVudGlvbiB3aWxsXG4gKiBuZXZlciBiZSBzdXBwb3J0ZWQuXG4gKlxuICogQ2hlY2tzIGlmIGEgZmlsZSBlbmRzIHdpdGhcbiAqIC0gdGVtcGxhdGUuanNcbiAqIC0gdGVtcGxhdGUuaGJzXG4gKiAtIHN0eWxlcy5jc3NcbiAqXG4gKiBAcGFyYW0ge3N0cmluZ30gZmlsZVBhdGhcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGlzUG9kKGZpbGVQYXRoKSB7XG4gIGlmIChmaWxlUGF0aC5pbmNsdWRlcyhsZWFkaW5nU2xhc2hQYXRoLmNvbXBvbmVudHNEaXIpKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgaWYgKGlzUG9kVGVtcGxhdGUoZmlsZVBhdGgpKSB7XG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICByZXR1cm4gZmlsZVBhdGguZW5kc1dpdGgoJ3N0eWxlcy5jc3MnKTtcbn1cblxuLyoqXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IGZpbGVQYXRoXG4gKiBAcmV0dXJucyB0aGUgc2FtZSBwYXRoLCBidXQgd2l0aG91dCB0aGUgZXh0ZW5zaW9uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB3aXRob3V0RXh0ZW5zaW9uKGZpbGVQYXRoKSB7XG4gIGxldCBwYXJzZWQgPSBwYXRoLnBhcnNlKGZpbGVQYXRoKTtcblxuICByZXR1cm4gcGF0aC5qb2luKHBhcnNlZC5kaXIsIHBhcnNlZC5uYW1lKTtcbn1cblxuLyoqXG4gKiBFeGFtcGxlcyBmb3IgZmlsZU5hbWVcbiAqIC0gYWJzb2x1dGUgb24tZGlzayBwYXRoXG4gKiAtIGluIHdlYnBhY2tcbiAqICAgLSBVUkwtYWJzb2x1dGUgcGF0aCwgc3RhcnRpbmcgd2l0aCAvXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IGZpbGVOYW1lXG4gKiBAcGFyYW0ge3sgYWRkaXRpb25hbFJvb3RzPzogc3RyaW5nW107IGN3ZDogc3RyaW5nIH19IG9wdGlvbnNcbiAqIEByZXR1cm5zXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpc1JlbGV2YW50RmlsZShmaWxlTmFtZSwgeyBhZGRpdGlvbmFsUm9vdHMsIGN3ZCB9KSB7XG4gIC8vIEZha2UgZmlsZSBoYW5kbGVkIGJ5IHRlc3RlbSBzZXJ2ZXIgd2hlbiBpdCBydW5zXG4gIGlmIChmaWxlTmFtZS5zdGFydHNXaXRoKGxlYWRpbmdTbGFzaFBhdGgudGVzdGVtKSkgcmV0dXJuIGZhbHNlO1xuICAvLyBQcml2YXRlIFZpcnR1YWwgTW9kdWxlc1xuICBpZiAoZmlsZU5hbWUuc3RhcnRzV2l0aCgnXFwwJykpIHJldHVybiBmYWxzZTtcblxuICAvLyBUaGVzZSBhcmUgbm90IHZhbGlkIHVzZXJsYW5kIG5hbWVzIChvciBhcmUgZnJvbSBsaWJyYXJpZXMpXG4gIGlmIChwYXRoLmlzQWJzb2x1dGUoZmlsZU5hbWUpID09PSBmYWxzZSkge1xuICAgIGlmIChmaWxlTmFtZS5tYXRjaCgvXlthLXpBLVpdLykpIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIC8vIEV4dGVybmFsIHRvIHVzXG4gIGlmIChmaWxlTmFtZS5zdGFydHNXaXRoKGxlYWRpbmdTbGFzaFBhdGguYXRFbWJyb2lkZXIpKSByZXR1cm4gZmFsc2U7XG4gIGlmIChJUlJFTEVWQU5UX1BBVEhTLnNvbWUoKGkpID0+IGZpbGVOYW1lLmluY2x1ZGVzKGkpKSkgcmV0dXJuIGZhbHNlO1xuXG4gIGxldCB3b3Jrc3BhY2UgPSBmaW5kV29ya3NwYWNlUGF0aChmaWxlTmFtZSk7XG5cbiAgYXNzZXJ0KGN3ZCwgYGN3ZCB3YXMgbm90IHBhc3NlZCB0byBpc1JlbGV2YW50RmlsZWApO1xuXG4gIGxldCBvdXJXb3Jrc3BhY2UgPSBmaW5kV29ya3NwYWNlUGF0aChjd2QpO1xuXG4gIGlmICh3b3Jrc3BhY2UgIT09IG91cldvcmtzcGFjZSkge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIGxldCBsb2NhbCA9IGZpbGVOYW1lLnJlcGxhY2Uod29ya3NwYWNlLCAnJyk7XG4gIGxldCBbLCAuLi5wYXJ0c10gPSBsb2NhbC5zcGxpdChwYXRoLnNlcCkuZmlsdGVyKEJvb2xlYW4pO1xuXG4gIGlmIChVTlNVUFBPUlRFRF9ESVJFQ1RPUklFUy5oYXMocGFydHNbMF0pKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgLyoqXG4gICAqIE1vc3RseSBwb2RzIHN1cHBvcnQuXG4gICAqIGZvbGtzIG5lZWQgdG8gb3B0IGluIHRvIHBvZHMgKHJvdXRlcyksIGJlY2F1c2UgZXZlcnkgcG9kcyBhcHAgY2FuIGJlIGNvbmZpZ3VyZWQgZGlmZmVyZW50bHlcbiAgICovXG4gIGxldCByb290cyA9IFtcbiAgICBsZWFkaW5nU2xhc2hQYXRoLmNvbXBvbmVudHNEaXIsXG4gICAgbGVhZGluZ1NsYXNoUGF0aC50ZW1wbGF0ZXNEaXIsXG4gICAgbGVhZGluZ1NsYXNoUGF0aC5yb3V0ZXNEaXIsXG4gICAgLi4uKGFkZGl0aW9uYWxSb290cyB8fCBbXSksXG4gIF07XG5cbiAgaWYgKCFyb290cy5zb21lKChyb290KSA9PiBmaWxlTmFtZS5pbmNsdWRlcyhyb290KSkpIHtcbiAgICByZXR1cm47XG4gIH1cblxuICByZXR1cm4gdHJ1ZTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHBhY2thZ2VTY29wZWRQYXRoVG9Nb2R1bGVQYXRoKHBhY2thZ2VTY29wZWRQYXRoKSB7XG4gIC8qKlxuICAgKiAqQnkgY29udmVudGlvbiosIGBzcmNgIGlzIG9taXR0ZWQgZnJvbSBjb21wb25lbnQgcGF0aHMuXG4gICAqIFdlIGNhbiByZWZsZWN0IHRoZSBzYW1lIGJlaGF2aW9yIGJ5IHJlcGxhY2luZyBzcmMvXG4gICAqIHdpdGggYW4gZW1wdHkgc3RyaW5nLlxuICAgKlxuICAgKiBDU1MgaXNuJ3QgZW1pdHRlZCBhcyBhIGNvLWxvY2F0ZWQgbW9kdWxlLCBidXRcbiAgICogdG8ga2VlcCBjb252ZW50aW9ucyBjb25zaXN0ZW50IGFjcm9zcyBsYW5ndWFnZXMsXG4gICAqIHdlIGNhbiBwcmV0ZW5kIGl0IGlzLlxuICAgKlxuICAgKiBBbnkgY3VzdG9taXphdGlvbiBiZXlvbmQgcmVtb3ZpbmcgYHNyY2AgYW5kIGBhcHBgIGlzIHBvdGVudGlhbGx5IGNvbmZ1c2luZy5cbiAgICogSWYgd2UgbmVlZCBmdXJ0aGVyIGN1c3RvbWl6YXRpb25zLCB3ZSdsbCB3YW50IHRvIG1hdGNoIG9uIGBleHBvcnRzYCBpbiB0aGVcbiAgICogY29ycmVzcG9uZGluZyBwYWNrYWdlLmpzb25cbiAgICovXG4gIGxldCBwYWNrYWdlUmVsYXRpdmUgPSBwYWNrYWdlU2NvcGVkUGF0aC5yZXBsYWNlKFxuICAgIG5ldyBSZWdFeHAoYF4ke1JlZ0V4cC5lc2NhcGUobGVhZGluZ1NsYXNoUGF0aC5zcmMpfWApLFxuICAgIHBhdGguc2VwLFxuICApO1xuXG4gIGxldCBwYXJzZWQgPSBwYXRoLnBhcnNlKHBhY2thZ2VSZWxhdGl2ZSk7XG5cbiAgaWYgKGlzUG9kKHBhY2thZ2VSZWxhdGl2ZSkpIHtcbiAgICAvKipcbiAgICAgKiBGb3IgcG9kcywgd2UgY2hvcCBvZmYgdGhlIHdob2xlIGZpbGUsIGFuZCB1c2UgdGhlIGRpciBuYW1lIGFzIHRoZSBcIm1vZHVsZVBhdGhcIlxuICAgICAqL1xuICAgIHJldHVybiBwYXJzZWQuZGlyO1xuICB9XG5cbiAgLyoqXG4gICAqIElmIGFuIGV4dGVuc2lvbiBpcyBwYXNzZWQsIHJlbW92ZSBpdC5cbiAgICogV2hlbiB1c2luZyBwYWNrYWdlcnMsIGZvbGtzIGFyZSB1c2VkIHRvIG5vdCBoYXZpbmcgdG8gc3BlY2lmeSBleHRlbnNpb25zIGZvciBmaWxlcy5cbiAgICogU2luY2Ugd2UgZG9uJ3QgZXZlbiBlbWl0IGNzcyBmaWxlcyBjby1sb2NhdGVkIHRvIGVhY2ggbW9kdWxlLFxuICAgKiB0aGlzIGhlbHBzIHVzIG5vdCBjb252ZXkgYSBsaWUgdGhhdCBhIGZpbGUgbWF5IGV4aXN0IGluIGF0IHJ1bnRpbWUuXG4gICAqXG4gICAqIEZvciBleGFtcGxlIGA8bW9kdWxlLW5hbWU+L2NvbXBvbmVudHMvYnV0dG9uYC5cbiAgICogSXQgZG9lc24ndCBtYXR0ZXIgd2hhdCB0aGUgZXh0ZW5zaW9uIGlzLCBiZWNhdXNlIHlvdSBjYW4gb25seSBoYXZlIG9uZSBjc3MgZmlsZVxuICAgKiBmb3IgdGhlIGJ1dHRvbiBtb2R1bGUgYW55d2F5LlxuICAgKi9cbiAgbGV0IGxvY2FsUGFja2FnZXJTdHlsZVBhdGggPSBwYXRoLmpvaW4ocGFyc2VkLmRpciwgcGFyc2VkLm5hbWUpO1xuXG4gIHJldHVybiBsb2NhbFBhY2thZ2VyU3R5bGVQYXRoO1xufVxuXG4vKipcbiAqIHJldHVybnMgdGhlIGFwcC1tb2R1bGUgcGF0aCBvZiB0aGUgc291cmNlIGZpbGVcbiAqXG4gKiBUaGlzIGFzc3VtZXMgbm9ybWFsIGVtYmVyIGFwcCBjb252ZW50aW9uc1xuICpcbiAqIHdoaWNoIGlzIGA8cGFja2FnZS5qc29uI25hbWU+L3BhdGgtdG8tZmlsZWBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGFwcFBhdGgoc291cmNlUGF0aCkge1xuICBsZXQgd29ya3NwYWNlUGF0aCA9IGZpbmRXb3Jrc3BhY2VQYXRoKHNvdXJjZVBhdGgpO1xuICBsZXQgbmFtZSA9IG1vZHVsZU5hbWUoc291cmNlUGF0aCk7XG5cbiAgLyoqXG4gICAqICBVbmRlciBlbWJyb2lkZXIgYnVpbGRzLCB0aGUgc3BlYy1jb21wbGlhbnQgdmVyc2lvbiBvZiB0aGUgYXBwXG4gICAqIGhhcyBhbGwgdGhlIGZpbGVzIHVuZGVyIGEgZm9sZGVyIHdoaWNoIHJlcHJlc2VudHMgdGhlIHBhY2thZ2UgbmFtZSxcbiAgICogcmF0aGVyIHRoYW4gXCJhcHBcIi5cbiAgICovXG4gIGxldCBwYWNrYWdlUmVsYXRpdmUgPSBzb3VyY2VQYXRoLnJlcGxhY2Uod29ya3NwYWNlUGF0aCwgJycpO1xuXG4gIC8qKlxuICAgKiBCdXQgd2UgYWxzbyBkb24ndCB3YW50ICdhcHAnIC0tIHdoaWNoIGlzIHByZXNlbnQgaW4gdGhlIHYxIGFkZG9uIHBpcGVsaW5lXG4gICAqL1xuICBwYWNrYWdlUmVsYXRpdmUgPSBwYWNrYWdlUmVsYXRpdmUucmVwbGFjZShsZWFkaW5nU2xhc2hQYXRoLmFwcCwgcGF0aC5zZXApO1xuXG4gIC8vIEFueSBvZiB0aGUgYWJvdmUgcmVscGFjZW1lbnRzIGNvdWxkIGFjY2lkZW50YWxseSBnaXZlIHVzIGFuIGV4dHJhIC8gKGRlcGVuZGluZyBvbiBvdXIgYnVpbGQgZW52aXJvbm1lbnQpXG4gIHBhY2thZ2VSZWxhdGl2ZSA9IHBhdGgubm9ybWFsaXplKHBhY2thZ2VSZWxhdGl2ZSk7XG5cbiAgbGV0IGxvY2FsUGFja2FnZXJTdHlsZVBhdGggPSBwYWNrYWdlU2NvcGVkUGF0aFRvTW9kdWxlUGF0aChwYWNrYWdlUmVsYXRpdmUpO1xuXG4gIHJldHVybiBgJHtuYW1lfSR7bG9jYWxQYWNrYWdlclN0eWxlUGF0aH1gO1xufVxuXG4vKipcbiAqIFRvIGF2b2lkIGhpdHRpbmcgdGhlIGZpbGVzeXNldG0sIHdlJ2xsIHN0b3JlIGFsbCBmb3VuZFxuICogcHJvamVjdCBwYXRocyBiZXJlLCBzbyB3ZSBjYW4sIGluIG1lbW9yeSxcbiAqIGdldCB0aGUgZm9sZGVyIHdoZXJlIGEgcGFja2FnZS5qc29uIGV4aXN0cywgcmF0aGVyIHRoYW5cbiAqIGhpdCB0aGUgZmlsZSBzeXN0ZW0gZXZlcnkgdGltZS5cbiAqL1xuY29uc3QgU0VFTiA9IG5ldyBTZXQoKTtcblxuZnVuY3Rpb24gZ2V0U2Vlbihzb3VyY2VQYXRoKSB7XG4gIGlmIChTRUVOLmhhcyhzb3VyY2VQYXRoKSkgcmV0dXJuIHNvdXJjZVBhdGg7XG5cbiAgbGV0IHBhcnRzID0gc291cmNlUGF0aC5zcGxpdChwYXRoLnNlcCk7XG5cbiAgZm9yIChsZXQgaSA9IHBhcnRzLmxlbmd0aCAtIDE7IGkgPiAxOyBpLS0pIHtcbiAgICBsZXQgdG9DaGVjayA9IHBhcnRzLnNsaWNlKDAsIGkpLmpvaW4ocGF0aC5zZXApO1xuXG4gICAgbGV0IHNlZW4gPSBTRUVOLmhhcyh0b0NoZWNrKTtcblxuICAgIGlmIChzZWVuKSB7XG4gICAgICByZXR1cm4gdG9DaGVjaztcbiAgICB9XG4gIH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGZpbmRXb3Jrc3BhY2VQYXRoKHNvdXJjZVBhdGgsIG9wdGlvbnMpIHtcbiAgbGV0IGN3ZCA9IG9wdGlvbnM/LmN3ZCA/PyBDV0Q7XG5cbiAgaWYgKHNvdXJjZVBhdGguZW5kc1dpdGgocGF0aC5zZXApKSB7XG4gICAgc291cmNlUGF0aCA9IHNvdXJjZVBhdGgucmVwbGFjZShcbiAgICAgIG5ldyBSZWdFeHAoYCR7UmVnRXhwLmVzY2FwZShwYXRoLnNlcCl9JGApLFxuICAgICAgJycsXG4gICAgKTtcbiAgfVxuXG4gIGxldCBzZWVuID0gZ2V0U2Vlbihzb3VyY2VQYXRoKTtcblxuICBpZiAoc2Vlbikge1xuICAgIHJldHVybiBzZWVuO1xuICB9XG5cbiAgbGV0IGNhbmRpZGF0ZVBhdGggPSBwYXRoLmpvaW4oc291cmNlUGF0aCwgJ3BhY2thZ2UuanNvbicpO1xuXG4gIGNvbnN0IGlzV29ya3NwYWNlID0gZnNTeW5jLmV4aXN0c1N5bmMoY2FuZGlkYXRlUGF0aCk7XG5cbiAgaWYgKGlzV29ya3NwYWNlKSB7XG4gICAgcmV0dXJuIHNvdXJjZVBhdGg7XG4gIH1cblxuICBjb25zdCBwYWNrYWdlSnNvblBhdGggPSBmaW5kUGFja2FnZUpzb25VcChzb3VyY2VQYXRoLCB7IGN3ZCB9KTtcblxuICBpZiAoIXBhY2thZ2VKc29uUGF0aCkge1xuICAgIHRocm93IG5ldyBFcnJvcihgQ291bGQgbm90IGRldGVybWluZSBwcm9qZWN0IGZvciAke3NvdXJjZVBhdGh9YCk7XG4gIH1cblxuICBjb25zdCB3b3Jrc3BhY2VQYXRoID0gcGF0aC5kaXJuYW1lKHBhY2thZ2VKc29uUGF0aCk7XG5cbiAgU0VFTi5hZGQod29ya3NwYWNlUGF0aCk7XG5cbiAgcmV0dXJuIHdvcmtzcGFjZVBhdGg7XG59XG5cbmZ1bmN0aW9uIGZpbmRQYWNrYWdlSnNvblVwKHN0YXJ0UGF0aCwgb3B0aW9ucykge1xuICBsZXQgY3dkID0gb3B0aW9ucz8uY3dkID8/IENXRDtcbiAgbGV0IHBhcnRzID0gc3RhcnRQYXRoLnNwbGl0KHBhdGguc2VwKTtcblxuICBmb3IgKGxldCBpID0gcGFydHMubGVuZ3RoIC0gMTsgaSA+IDE7IGktLSkge1xuICAgIGxldCB0b0NoZWNrID0gcGFydHMuc2xpY2UoMCwgaSkuam9pbihwYXRoLnNlcCk7XG5cbiAgICBsZXQgcGFja2FnZUpzb24gPSBwYXRoLmpvaW4odG9DaGVjaywgJ3BhY2thZ2UuanNvbicpO1xuICAgIGxldCBleGlzdHMgPSBmc1N5bmMuZXhpc3RzU3luYyhwYWNrYWdlSnNvbik7XG5cbiAgICBpZiAoZXhpc3RzKSB7XG4gICAgICByZXR1cm4gcGFja2FnZUpzb247XG4gICAgfVxuXG4gICAgLy8gRG9uJ3QgdHJhdmVyc2UgYWxsIHRoZSB3YXkgdG8gdGhlIHJvb3Qgb2YgdGhlIGZpbGUgc3lzdGVtLlxuICAgIGlmICh0b0NoZWNrID09PSBjd2QpIHtcbiAgICAgIGJyZWFrO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiBudWxsO1xufVxuXG5jb25zdCBNQU5JRkVTVF9DQUNIRSA9IG5ldyBNYXAoKTtcblxuLyoqXG4gKiBXaWxsIHJldHVybiB0aGUgcGFja2FnZS5qc29uI25hbWUsIG9yIGNvbmZpZy9lbnZpcm9ubWVudCNtb3VkbGVQcmVmaXggKGlmIHYxIGFwcClcbiAqXG4gKiBAcGFyYW0ge3N0cmluZ30gc291cmNlUGF0aFxuICovXG5leHBvcnQgZnVuY3Rpb24gbW9kdWxlTmFtZShzb3VyY2VQYXRoKSB7XG4gIGNvbnN0IHdvcmtzcGFjZSA9IGZpbmRXb3Jrc3BhY2VQYXRoKHNvdXJjZVBhdGgpO1xuICBjb25zdCBtYW5pZmVzdCA9IGdldE1hbmlmZXN0KHdvcmtzcGFjZSk7XG5cbiAgcmV0dXJuIG1hbmlmZXN0Lm5hbWU7XG59XG5cbi8qKlxuICogQHBhcmFtIHtzdHJpbmd9IHdvcmtzcGFjZVxuICovXG5mdW5jdGlvbiBnZXRNYW5pZmVzdCh3b3Jrc3BhY2UpIHtcbiAgbGV0IGV4aXN0aW5nID0gTUFOSUZFU1RfQ0FDSEUuZ2V0KHdvcmtzcGFjZSk7XG5cbiAgaWYgKGV4aXN0aW5nKSB7XG4gICAgcmV0dXJuIGV4aXN0aW5nO1xuICB9XG5cbiAgbGV0IGJ1ZmZlciA9IGZzU3luYy5yZWFkRmlsZVN5bmMocGF0aC5qb2luKHdvcmtzcGFjZSwgJ3BhY2thZ2UuanNvbicpKTtcbiAgbGV0IGNvbnRlbnQgPSBidWZmZXIudG9TdHJpbmcoKTtcbiAgbGV0IGpzb24gPSBKU09OLnBhcnNlKGNvbnRlbnQpO1xuXG4gIE1BTklGRVNUX0NBQ0hFLnNldCh3b3Jrc3BhY2UsIGpzb24pO1xuXG4gIHJldHVybiBqc29uO1xufVxuIiwiLyoqXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IGNsYXNzTmFtZVxuICogQHBhcmFtIHtzdHJpbmd9IHBvc3RmaXhcbiAqIEBwYXJhbSB7U2V0PHN0cmluZz59IFtjbGFzc2VzSW5Dc3NdXG4gKiBAcmV0dXJuc1xuICovXG5leHBvcnQgZnVuY3Rpb24gcmVuYW1lQ2xhc3MoY2xhc3NOYW1lLCBwb3N0Zml4LCBjbGFzc2VzSW5Dc3MpIHtcbiAgY29uc3QgY2xhc3NlcyA9IGNsYXNzTmFtZS5zcGxpdCgvXFxzKy8pO1xuICBjb25zdCByZW5hbWVkQ2xhc3NlcyA9IGNsYXNzZXNcbiAgICAuZmlsdGVyKChjKSA9PiBjKVxuICAgIC5tYXAoKGMpID0+IGMudHJpbSgpKVxuICAgIC5tYXAoKGMpID0+IHtcbiAgICAgIGlmICghY2xhc3Nlc0luQ3NzIHx8IGNsYXNzZXNJbkNzcy5oYXMoYykpIHtcbiAgICAgICAgaWYgKGMuZW5kc1dpdGgocG9zdGZpeCkpIHJldHVybiBjO1xuXG4gICAgICAgIHJldHVybiBjICsgJ18nICsgcG9zdGZpeDtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIGM7XG4gICAgfSlcbiAgICAuam9pbignICcpO1xuXG4gIGNvbnN0IHJlbmFtZWRXaXRoUHJlc2VydmVkU3BhY2VzID0gY2xhc3NOYW1lLnJlcGxhY2UoXG4gICAgY2xhc3NOYW1lLnRyaW1TdGFydCgpLnRyaW1FbmQoKSxcbiAgICByZW5hbWVkQ2xhc3NlcyxcbiAgKTtcblxuICByZXR1cm4gcmVuYW1lZFdpdGhQcmVzZXJ2ZWRTcGFjZXM7XG59XG4iLCJpbXBvcnQge1xuICBhcHBQYXRoLFxuICBoYXNoRnJvbU1vZHVsZVBhdGgsXG4gIGlzUmVsZXZhbnRGaWxlLFxufSBmcm9tICcuLi9saWIvcGF0aC91dGlscy5qcyc7XG5pbXBvcnQgeyByZW5hbWVDbGFzcyB9IGZyb20gJy4uL2xpYi9yZW5hbWVDbGFzcy5qcyc7XG5cbmZ1bmN0aW9uIF9pc1JlbGV2YW50RmlsZShzdGF0ZSwgY3dkKSB7XG4gIGxldCBmaWxlTmFtZSA9IHN0YXRlLmZpbGUub3B0cy5maWxlbmFtZTtcbiAgbGV0IGFkZGl0aW9uYWxSb290cyA9IHN0YXRlLm9wdHM/LmFkZGl0aW9uYWxSb290cztcblxuICByZXR1cm4gaXNSZWxldmFudEZpbGUoZmlsZU5hbWUsIHtcbiAgICBhZGRpdGlvbmFsUm9vdHMsXG4gICAgY3dkLFxuICB9KTtcbn1cblxuLyoqXG4gKiBAcGFyYW0ge2FueX0gZW52IC0gYmFiZWwgcGx1Z2luIGVudiwgZW52LnR5cGVzIGlzIG1vc3QgY29tbW9ubHkgdXNlZCAoZXNwIGluIFRTKVxuICogQHBhcmFtIHtvYmplY3R9IG9wdGlvbnMgLSB0aGUgb3B0aW9ucyBmb3Igc2NvcGVkLWNzcyAtLSB0aGlzIGlzIGFsc28gYXZhaWxhYmxlIGluIGVhY2ggdmlzaXRvcidzIHN0YXRlLm9wdHNcbiAqIEBwYXJhbSB7c3RyaW5nfSB3b3JraW5nRGlyZWN0b3J5XG4gKi9cbmV4cG9ydCBjb25zdCBzY29wZWRDU1MgPSAoY29uZmlnKSA9PiAoZW52LCBvcHRpb25zLCB3b3JraW5nRGlyZWN0b3J5KSA9PiB7XG4gIC8vIG5vdCByZWFsbHkgYSBkZWVwIG1lcmdlLCBidXQgdGhlIGlkZWEgaXMgdG8gdXNlIG9uZSBvciB0aGUgb3RoZXIgYW5kIG5vdCBtaXggdGhlIGNvbmZpZ3NcbiAgb3B0aW9ucyA9IHsgLi4uY29uZmlnLCAuLi5vcHRpb25zIH07XG5cbiAgLyoqXG4gICAqIFRoaXMgYmFiZWwgcGx1Z2luIGRvZXMgdHdvIHRoaW5nczpcbiAgICogLSByZW1vdmVzIHRoZSBpbXBvcnQgb2Ygc2NvcGVkQ2xhc3MsIGlmIGl0IGV4aXN0c1xuICAgKiAgIC0gaWYgc2NvcGVkQ2xhc3Mgd2FzIGltcG9ydGVkLCBpdCBpcyByZW1vdmVkIGZyb20gYW55IGNvbXBvbmVudCdzIFwic2NvcGUgYmFnXCJcbiAgICogICAgICh0aGUgc2NvcGUgYmFnIGJlaW5nIGEgbG93LWxldmVsIG9iamVjdCB1c2VkIGZvciBwYXNzaW5nIHdoYXQgaXMgXCJpbiBzY29wZVwiIGZvciBhIGNvbXBvbmVudClcbiAgICovXG4gIHJldHVybiB7XG4gICAgdmlzaXRvcjoge1xuICAgICAgUHJvZ3JhbToge1xuICAgICAgICBlbnRlcihwYXRoLCBzdGF0ZSkge1xuICAgICAgICAgIGlmICghX2lzUmVsZXZhbnRGaWxlKHN0YXRlLCB3b3JraW5nRGlyZWN0b3J5KSkge1xuICAgICAgICAgICAgc3RhdGUuY2FuU2tpcCA9IHRydWU7XG5cbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBsZXQgbW9kdWxlUGF0aCA9IGFwcFBhdGgoc3RhdGUuZmlsZW5hbWUpO1xuXG4gICAgICAgICAgc3RhdGUucG9zdGZpeCA9IGhhc2hGcm9tTW9kdWxlUGF0aChtb2R1bGVQYXRoKTtcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgICBJbXBvcnREZWNsYXJhdGlvbihwYXRoLCBzdGF0ZSkge1xuICAgICAgICBpZiAoc3RhdGUuY2FuU2tpcCkge1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChwYXRoLm5vZGUuc291cmNlLnZhbHVlID09PSAnZW1iZXItc2NvcGVkLWNzcycpIHtcbiAgICAgICAgICBsZXQgc3BlY2lmaWVyID0gcGF0aC5ub2RlLnNwZWNpZmllcnMuZmluZChcbiAgICAgICAgICAgICh4KSA9PiB4LmltcG9ydGVkLm5hbWUgPT09ICdzY29wZWRDbGFzcycsXG4gICAgICAgICAgKTtcblxuICAgICAgICAgIGlmIChzcGVjaWZpZXIpIHtcbiAgICAgICAgICAgIHN0YXRlLmZpbGUub3B0cy5pbXBvcnRlZFNjb3BlZENsYXNzID0gc3BlY2lmaWVyLmxvY2FsLm5hbWU7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgaWYgKHNwZWNpZmllci5sb2NhbC5uYW1lICE9PSAnc2NvcGVkQ2xhc3MnKSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICAgIGBUaGUgc2NvcGVkQ2xhc3MgaW1wb3J0IGlzIGEgcHN1ZWRvLWhlbHBlciwgYW5kIG1heSBub3QgYmUgcmVuYW1lZCBhcyBpdCBpcyByZW1vdmVkIGF0IGJ1aWxkIHRpbWUuYCxcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgcGF0aC5yZW1vdmUoKTtcbiAgICAgICAgfVxuICAgICAgfSxcbiAgICAgIC8qKlxuICAgICAgICogUmVuYW1lIHVzYWdlcyBpbiBKUy9UUy9HSlMvR1RTXG4gICAgICAgKi9cbiAgICAgIENhbGxFeHByZXNzaW9uKHBhdGgsIHN0YXRlKSB7XG4gICAgICAgIGlmIChzdGF0ZS5jYW5Ta2lwKSB7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKFxuICAgICAgICAgIHBhdGgubm9kZS5jYWxsZWUudHlwZSA9PT0gJ0lkZW50aWZpZXInICYmXG4gICAgICAgICAgcGF0aC5ub2RlLmNhbGxlZS5uYW1lID09PSBzdGF0ZS5maWxlLm9wdHM/LmltcG9ydGVkU2NvcGVkQ2xhc3NcbiAgICAgICAgKSB7XG4gICAgICAgICAgaWYgKFxuICAgICAgICAgICAgcGF0aC5ub2RlLmFyZ3VtZW50cy5sZW5ndGggIT09IDEgfHxcbiAgICAgICAgICAgIHBhdGgubm9kZS5hcmd1bWVudHNbMF0udHlwZSAhPT0gJ1N0cmluZ0xpdGVyYWwnXG4gICAgICAgICAgKSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICAgIGBUaGUgc2NvcGVkQ2xhc3MgaGVscGVyIG9ubHkgYWNjZXB0cyBhIHNpbmdsZSwgbm9uLWR5bmFtaWMsIHN0cmluZyBsaXRlcmFsIGFyZ3VtZW50LmAsXG4gICAgICAgICAgICApO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIGNvbnN0IG9yaWdpbmFsID0gcGF0aC5ub2RlLmFyZ3VtZW50c1swXS52YWx1ZTtcbiAgICAgICAgICBjb25zdCByZW5hbWVkID0gcmVuYW1lQ2xhc3MoXG4gICAgICAgICAgICBvcmlnaW5hbCxcbiAgICAgICAgICAgIHN0YXRlLnBvc3RmaXgsXG4gICAgICAgICAgICBuZXcgU2V0KFtvcmlnaW5hbF0pLFxuICAgICAgICAgICk7XG4gICAgICAgICAgY29uc3QgdHJhbnNmb3JtZWRTdHJpbmcgPSBlbnYudHlwZXMuc3RyaW5nTGl0ZXJhbChyZW5hbWVkKTtcblxuICAgICAgICAgIHBhdGgucmVwbGFjZVdpdGgodHJhbnNmb3JtZWRTdHJpbmcpO1xuICAgICAgICB9XG4gICAgICB9LFxuICAgICAgLyoqXG4gICAgICAgKiBPbmx5IGluIHN0cmljdCBtb2RlLCBkbyB3ZSBjYXJlIGFib3V0IHJlbW9uaW5nIHRoZSBzY29wZSBiYWcgcmVmZXJlbmNlXG4gICAgICAgKi9cbiAgICAgIE9iamVjdFByb3BlcnR5KHBhdGgsIHN0YXRlKSB7XG4gICAgICAgIGlmICghc3RhdGUuZmlsZS5vcHRzPy5pbXBvcnRlZFNjb3BlZENsYXNzKSByZXR1cm47XG5cbiAgICAgICAgaWYgKFxuICAgICAgICAgIHBhdGgubm9kZS52YWx1ZS50eXBlID09PSAnSWRlbnRpZmllcicgJiZcbiAgICAgICAgICBwYXRoLm5vZGUudmFsdWUubmFtZSA9PT0gc3RhdGUuZmlsZS5vcHRzPy5pbXBvcnRlZFNjb3BlZENsYXNzXG4gICAgICAgICkge1xuICAgICAgICAgIHBhdGgucmVtb3ZlKCk7XG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgfSxcbiAgfTtcbn07XG4iLCJpbXBvcnQgeyBleGlzdHNTeW5jLCByZWFkRmlsZVN5bmMgfSBmcm9tICdmcyc7XG5pbXBvcnQgcG9zdGNzcyBmcm9tICdwb3N0Y3NzJztcbmltcG9ydCBzY3NzU3ludGF4IGZyb20gJ3Bvc3Rjc3Mtc2Nzcyc7XG5pbXBvcnQgcGFyc2VyIGZyb20gJ3Bvc3Rjc3Mtc2VsZWN0b3ItcGFyc2VyJztcblxuaW1wb3J0IHsgbWQ1IH0gZnJvbSAnLi4vcGF0aC9tZDUuanMnO1xuXG4vKipcbiAqIEBwYXJhbSB7c3RyaW5nfSBjc3NcbiAqIEByZXR1cm4ge3N0cmluZ30gaGFzaGVkIGRvd24gdmVyc2lvbiBvZiB0aGUgQ1NTIGZvciBkaXNhbWJpZ3VhdGluZ1xuICovXG5leHBvcnQgZnVuY3Rpb24gaGFzaChjc3MpIHtcbiAgcmV0dXJuIGBjc3MtJHttZDUoY3NzKX1gO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gaXNJbnNpZGVHbG9iYWwobm9kZSwgZnVuYykge1xuICBjb25zdCBwYXJlbnQgPSBub2RlLnBhcmVudDtcblxuICBpZiAoIXBhcmVudCkgcmV0dXJuIGZhbHNlO1xuICBpZiAocGFyZW50LnR5cGUgPT09ICdwc2V1ZG8nICYmIHBhcmVudC52YWx1ZSA9PT0gJzpnbG9iYWwnKSByZXR1cm4gdHJ1ZTtcblxuICByZXR1cm4gaXNJbnNpZGVHbG9iYWwocGFyZW50LCBmdW5jKTtcbn1cblxuLyoqXG4gKiBAcGFyYW0ge3N0cmluZ30gY3NzUGF0aCBwYXRoIHRvIGEgQ1NTIGZpbGVcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldENTU0luZm8oY3NzUGF0aCkge1xuICBpZiAoIWV4aXN0c1N5bmMoY3NzUGF0aCkpIHtcbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIGxldCBjc3MgPSByZWFkRmlsZVN5bmMoY3NzUGF0aCwgJ3V0ZjgnKTtcblxuICByZXR1cm4gZ2V0Q1NTQ29udGVudEluZm8oY3NzKTtcbn1cblxuLyoqXG4gKiBXZSB1c2UgdGhpcyBmdW5jdGlvbiB0byBjaGVjayBlYWNoIGNsYXNzIHVzZWQgaW4gdGhlIHRlbXBsYXRlXG4gKiB0byBzZWUgaWYgd2UgbmVlZCB0byBsZWF2ZSBpdCBhbG9uZSBvciB0cmFuc2Zvcm0gaXRcbiAqXG4gKiBAcGFyYW0ge3N0cmluZ30gY3NzIHRoZSBDU1MncyBjb250ZW50c1xuICogQHBhcmFtIHtzdHJpbmd9IFtsYW5nXSBvcHRpb25hbCBsYW5ndWFnZSBoaW50IChlLmcuICdzY3NzJywgJ3Nhc3MnLCAnbGVzcycpXG4gKiBAcmV0dXJuIHt7IGNsYXNzZXM6IFNldDxzdHJpbmc+LCB0YWdzOiBTZXQ8c3RyaW5nPiwgY3NzOiBzdHJpbmcsIGlkOiBzdHJpbmcgfX1cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldENTU0NvbnRlbnRJbmZvKGNzcywgbGFuZykge1xuICBjb25zdCBjbGFzc2VzID0gbmV3IFNldCgpO1xuICBjb25zdCB0YWdzID0gbmV3IFNldCgpO1xuXG4gIGNvbnN0IHBhcnNlT3B0aW9ucyA9XG4gICAgbGFuZyA9PT0gJ3Njc3MnIHx8IGxhbmcgPT09ICdzYXNzJyA/IHsgc3ludGF4OiBzY3NzU3ludGF4IH0gOiB7fTtcblxuICBjb25zdCBhc3QgPSBwb3N0Y3NzLnBhcnNlKGNzcywgcGFyc2VPcHRpb25zKTtcblxuICBjb25zdCBpc1Njc3MgPSBsYW5nID09PSAnc2NzcycgfHwgbGFuZyA9PT0gJ3Nhc3MnO1xuXG4gIGFzdC53YWxrKChub2RlKSA9PiB7XG4gICAgaWYgKG5vZGUudHlwZSA9PT0gJ3J1bGUnKSB7XG4gICAgICBjb25zdCBzZWxlY3RvciA9IGlzU2NzcyA/IHJlc29sdmVOZXN0ZWRTYXNzU2VsZWN0b3Iobm9kZSkgOiBub2RlLnNlbGVjdG9yO1xuXG4gICAgICBnZXRDbGFzc2VzQW5kVGFncyhzZWxlY3RvciwgY2xhc3NlcywgdGFncyk7XG4gICAgfVxuICB9KTtcblxuICBsZXQgaWQgPSBoYXNoKGNzcyk7XG5cbiAgcmV0dXJuIHsgY2xhc3NlcywgdGFncywgY3NzLCBpZCB9O1xufVxuXG4vKipcbiAqIFJlc29sdmVzIGEgbmVzdGVkIFNDU1Mgc2VsZWN0b3IgYnkgc3Vic3RpdHV0aW5nIGAmYCB3aXRoIHRoZSBmdWxseS1yZXNvbHZlZFxuICogcGFyZW50IHNlbGVjdG9yLCByZWN1cnNpdmVseS4gVGhpcyBjb252ZXJ0cyBlLmcuIGAmLS1tb2RpZmllcmAgKGNoaWxkIG9mXG4gKiBgLmJsb2NrYCkgaW50byBgLmJsb2NrLS1tb2RpZmllcmAsIGFuZCBoYW5kbGVzIGFyYml0cmFyeSBuZXN0aW5nIGRlcHRoIHNvXG4gKiB0aGF0IGAmLS1tb2RpZmllcmAgaW5zaWRlIGAmLS1tb2RpZmllcmAgaW5zaWRlIGAuYmxvY2tgIHlpZWxkc1xuICogYC5ibG9jay0tbW9kaWZpZXItLW1vZGlmaWVyYC5cbiAqXG4gKiBAcGFyYW0ge2ltcG9ydCgncG9zdGNzcycpLlJ1bGV9IG5vZGVcbiAqIEByZXR1cm4ge3N0cmluZ31cbiAqL1xuZnVuY3Rpb24gcmVzb2x2ZU5lc3RlZFNhc3NTZWxlY3Rvcihub2RlKSB7XG4gIGNvbnN0IHsgc2VsZWN0b3IgfSA9IG5vZGU7XG5cbiAgaWYgKCFzZWxlY3Rvci5pbmNsdWRlcygnJicpKSB7XG4gICAgcmV0dXJuIHNlbGVjdG9yO1xuICB9XG5cbiAgY29uc3QgcGFyZW50ID0gbm9kZS5wYXJlbnQ7XG5cbiAgaWYgKCFwYXJlbnQgfHwgcGFyZW50LnR5cGUgIT09ICdydWxlJykge1xuICAgIC8vIE5vIHBhcmVudCBydWxlIOKAlCBgJmAgaGFzIG5vdGhpbmcgdG8gc3Vic3RpdHV0ZSwgcmV0dXJuIGFzLWlzXG4gICAgcmV0dXJuIHNlbGVjdG9yO1xuICB9XG5cbiAgLy8gUmVjdXJzaXZlbHkgcmVzb2x2ZSB0aGUgcGFyZW50IGZpcnN0LCB0aGVuIHN1YnN0aXR1dGUgaW50byB0aGlzIHNlbGVjdG9yXG4gIGNvbnN0IHJlc29sdmVkUGFyZW50ID0gcmVzb2x2ZU5lc3RlZFNhc3NTZWxlY3RvcihwYXJlbnQpO1xuXG4gIHJldHVybiBzZWxlY3Rvci5yZXBsYWNlKC8mL2csIHJlc29sdmVkUGFyZW50KTtcbn1cblxuZnVuY3Rpb24gZ2V0Q2xhc3Nlc0FuZFRhZ3Moc2VsLCBjbGFzc2VzLCB0YWdzKSB7XG4gIGNvbnN0IHRyYW5zZm9ybSA9IChzbHMpID0+IHtcbiAgICBzbHMud2Fsaygoc2VsZWN0b3IpID0+IHtcbiAgICAgIGlmIChzZWxlY3Rvci50eXBlID09PSAnY2xhc3MnICYmICFpc0luc2lkZUdsb2JhbChzZWxlY3RvcikpIHtcbiAgICAgICAgY2xhc3Nlcy5hZGQoc2VsZWN0b3IudmFsdWUpO1xuICAgICAgfSBlbHNlIGlmIChzZWxlY3Rvci50eXBlID09PSAndGFnJyAmJiAhaXNJbnNpZGVHbG9iYWwoc2VsZWN0b3IpKSB7XG4gICAgICAgIHRhZ3MuYWRkKHNlbGVjdG9yLnZhbHVlKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfTtcblxuICBwYXJzZXIodHJhbnNmb3JtKS5wcm9jZXNzU3luYyhzZWwpO1xufVxuXG5pZiAoaW1wb3J0Lm1ldGEudml0ZXN0KSB7XG4gIGNvbnN0IHsgaXQsIGV4cGVjdCB9ID0gaW1wb3J0Lm1ldGEudml0ZXN0O1xuXG4gIGl0KCdzaG91bGQgcmV0dXJuIGNsYXNzZXMgYW5kIHRhZ3MgdGhhdCBhcmUgbm90IGluIDpnbG9iYWwnLCBmdW5jdGlvbiAoKSB7XG4gICAgY29uc3QgY3NzID0gJy5iYXogOmdsb2JhbCguZm9vKSAuYmFyIGRpdiA6Z2xvYmFsKHApICB7IGNvbG9yOiByZWQ7IH0nO1xuICAgIGNvbnN0IHsgY2xhc3NlcywgdGFncyB9ID0gZ2V0Q1NTQ29udGVudEluZm8oY3NzKTtcblxuICAgIC8vIGNsYXNzZXMgc2hvdWxkIGJlIGJheiBhbmQgYmFyXG4gICAgZXhwZWN0KGNsYXNzZXMuc2l6ZSkudG8uZXF1YWwoMik7XG4gICAgZXhwZWN0KFsuLi5jbGFzc2VzXSkudG8uaGF2ZS5tZW1iZXJzKFsnYmF6JywgJ2JhciddKTtcbiAgICBleHBlY3QodGFncy5zaXplKS50by5lcXVhbCgxKTtcbiAgICBleHBlY3QoWy4uLnRhZ3NdKS50by5oYXZlLm1lbWJlcnMoWydkaXYnXSk7XG4gIH0pO1xuXG4gIGl0KCdzaG91bGQgcGFyc2UgU0NTUyBuZXN0aW5nIHN5bnRheCB3aXRob3V0IGNyYXNoaW5nIHdoZW4gbGFuZz1zY3NzJywgZnVuY3Rpb24gKCkge1xuICAgIGNvbnN0IHNjc3MgPSBgXG4gICAgICAkYmFzZS1jb2xvcjogI2M2NTM4YztcbiAgICAgICRib3JkZXItZGFyazogcmdiYSgkYmFzZS1jb2xvciwgMC44OCk7XG5cbiAgICAgIC5wYXJlbnQge1xuICAgICAgICAmOmhvdmVyIHsgY29sb3I6ICRiYXNlLWNvbG9yOyB9XG4gICAgICAgIC5jaGlsZCB7IGJvcmRlcjogMXB4IHNvbGlkICRib3JkZXItZGFyazsgfVxuICAgICAgICBjb2xvcjogcmVkO1xuICAgICAgfVxuICAgIGA7XG4gICAgY29uc3QgeyBjbGFzc2VzIH0gPSBnZXRDU1NDb250ZW50SW5mbyhzY3NzLCAnc2NzcycpO1xuXG4gICAgZXhwZWN0KFsuLi5jbGFzc2VzXSkudG9NYXRjaElubGluZVNuYXBzaG90KGBcbiAgICAgIFtcbiAgICAgICAgXCJwYXJlbnRcIixcbiAgICAgICAgXCJjaGlsZFwiLFxuICAgICAgXVxuICAgIGApO1xuICB9KTtcblxuICBpdCgnc2hvdWxkIHBhcnNlIFNDU1MgbmVzdGluZyBzeW50YXggd2l0aG91dCBjcmFzaGluZyB3aGVuIGxhbmc9c2FzcycsIGZ1bmN0aW9uICgpIHtcbiAgICBjb25zdCBzY3NzID0gYFxuICAgICAgJGJhc2UtY29sb3I6IGdyZWVuO1xuICAgICAgLmJsb2NrIHtcbiAgICAgICAgJi0tbW9kaWZpZXIgeyBjb2xvcjogJGJhc2UtY29sb3I7IH1cbiAgICAgIH1cbiAgICBgO1xuICAgIGNvbnN0IHsgY2xhc3NlcyB9ID0gZ2V0Q1NTQ29udGVudEluZm8oc2NzcywgJ3Nhc3MnKTtcblxuICAgIGV4cGVjdChbLi4uY2xhc3Nlc10pLnRvTWF0Y2hJbmxpbmVTbmFwc2hvdChgXG4gICAgICBbXG4gICAgICAgIFwiYmxvY2tcIixcbiAgICAgICAgXCJibG9jay0tbW9kaWZpZXJcIixcbiAgICAgIF1cbiAgICBgKTtcbiAgfSk7XG5cbiAgaXQoJ3Nob3VsZCBwYXJzZSBTQ1NTIGRlZXBseSBuZXN0ZWQgQkVNIHdoZW4gbGFuZz1zYXNzJywgZnVuY3Rpb24gKCkge1xuICAgIGNvbnN0IHNjc3MgPSBgXG4gICAgICAkYmFzZS1jb2xvcjogZ3JlZW47XG4gICAgICAuYmxvY2sge1xuICAgICAgICAmLS1tb2RpZmllciB7XG4gICAgICAgICAgY29sb3I6ICRiYXNlLWNvbG9yO1xuICAgICAgICAgICYtLW1vZGlmaWVyIHtcbiAgICAgICAgICAgIGNvbG9yOiAkYmFzZS1jb2xvcjtcbiAgICAgICAgICAgICYtLW1vZGlmaWVyIHtcbiAgICAgICAgICAgICAgY29sb3I6ICRiYXNlLWNvbG9yO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIGA7XG4gICAgY29uc3QgeyBjbGFzc2VzIH0gPSBnZXRDU1NDb250ZW50SW5mbyhzY3NzLCAnc2FzcycpO1xuXG4gICAgZXhwZWN0KFsuLi5jbGFzc2VzXSkudG9NYXRjaElubGluZVNuYXBzaG90KGBcbiAgICAgIFtcbiAgICAgICAgXCJibG9ja1wiLFxuICAgICAgICBcImJsb2NrLS1tb2RpZmllclwiLFxuICAgICAgICBcImJsb2NrLS1tb2RpZmllci0tbW9kaWZpZXJcIixcbiAgICAgICAgXCJibG9jay0tbW9kaWZpZXItLW1vZGlmaWVyLS1tb2RpZmllclwiLFxuICAgICAgXVxuICAgIGApO1xuICB9KTtcbn1cbiIsIi8qKlxuICogSW1wb3J0YW50IGRvY3M6XG4gKiAtIGh0dHBzOi8vZGV2ZWxvcGVyLm1vemlsbGEub3JnL2VuLVVTL2RvY3MvV2ViL0NTUy9cbiAqL1xuaW1wb3J0IHBvc3Rjc3MgZnJvbSAncG9zdGNzcyc7XG5pbXBvcnQgcGFyc2VyIGZyb20gJ3Bvc3Rjc3Mtc2VsZWN0b3ItcGFyc2VyJztcblxuaW1wb3J0IHsgaXNJbnNpZGVHbG9iYWwgfSBmcm9tICcuL3V0aWxzLmpzJztcblxuY29uc3QgU0VQID0gJ19fJztcblxuZnVuY3Rpb24gaXNSdWxlKG5vZGUpIHtcbiAgcmV0dXJuIG5vZGUudHlwZSA9PT0gJ3J1bGUnO1xufVxuXG5mdW5jdGlvbiBpc0RlY2xhcmF0aW9uKG5vZGUpIHtcbiAgcmV0dXJuIG5vZGUudHlwZSA9PT0gJ2RlY2wnO1xufVxuXG4vKipcbiAqIE5PVEU6IFwia2V5ZnJhbWVzXCIgaXMgYSBzaW5ndWxhciBkZWZpbml0aW9uLCBpbiB0aGF0IGl0J3MgYSBibG9jayBjb250YWluaW5nIGtleWZyYW1lc1xuICogICAgICAgdXNpbmcgYEBrZXlmcmFtZXMge31gIHdpdGggb25seSBvbmUgdGhpbmcgb24gdGhlIGluc2lkZSBkb2Vzbid0IG1ha2Ugc2Vuc2UuXG4gKi9cbmZ1bmN0aW9uIHJld3JpdGVSZWZlcmVuY2VhYmxlKG5vZGUsIHBvc3RmaXgpIHtcbiAgbGV0IG9yaWdpbmFsTmFtZSA9IG5vZGUucGFyYW1zO1xuICBsZXQgcG9zdGZpeGVkTmFtZSA9IG5vZGUucGFyYW1zICsgU0VQICsgcG9zdGZpeDtcblxuICBub2RlLnBhcmFtcyA9IHBvc3RmaXhlZE5hbWU7XG5cbiAgcmV0dXJuIHtcbiAgICBvcmlnaW5hbE5hbWUsXG4gICAgcG9zdGZpeGVkTmFtZSxcbiAgfTtcbn1cblxuZnVuY3Rpb24gcmV3cml0ZVNlbGVjdG9yKHNlbCwgcG9zdGZpeCkge1xuICBjb25zdCB0cmFuc2Zvcm0gPSAoc2VsZWN0b3JzKSA9PiB7XG4gICAgc2VsZWN0b3JzLndhbGsoKHNlbGVjdG9yKSA9PiB7XG4gICAgICBpZiAoaXNJbnNpZGVHbG9iYWwoc2VsZWN0b3IpKSByZXR1cm47XG5cbiAgICAgIC8vIFdlIG5ldmVyIHdhbnQgdG8gdG91Y2ggcHN1ZWRvIHNlbGVjdG9ycyBzaW5jZSB3ZSBhbmQgdGhlIHVzZXIgZG9lc24ndCBvd24gdGhlbS5cbiAgICAgIGlmIChzZWxlY3Rvci50eXBlID09PSAncHN1ZWRvJykgcmV0dXJuO1xuXG4gICAgICAvLyA6bnRoLW9mLXR5cGUgaGFzIHNwZWNpYWwgc3ludGF4IHdoZXJlIHRoZSB2YWx1ZXMgcGFzc2VkIHRvIG50aC1vZi10eXBlKClcbiAgICAgIC8vIG11c3QgZWl0aGVyIGJlIGV4YWN0bHkgXCJvZGRcIiwgXCJldmVuXCIsIG9yIGEgc2ltcGxlIGZvcm11bGFcbiAgICAgIC8vXG4gICAgICAvLyBodHRwczovL2RldmVsb3Blci5tb3ppbGxhLm9yZy9lbi1VUy9kb2NzL1dlYi9DU1MvOm50aC1vZi10eXBlXG4gICAgICBpZiAoaXNOdGhPZlR5cGUoc2VsZWN0b3IpKSByZXR1cm47XG5cbiAgICAgIGlmIChzZWxlY3Rvci50eXBlID09PSAnY2xhc3MnKSB7XG4gICAgICAgIHNlbGVjdG9yLnZhbHVlICs9ICdfJyArIHBvc3RmaXg7XG4gICAgICB9IGVsc2UgaWYgKHNlbGVjdG9yLnR5cGUgPT09ICd0YWcnKSB7XG4gICAgICAgIHNlbGVjdG9yLnJlcGxhY2VXaXRoKFxuICAgICAgICAgIHBhcnNlci50YWcoeyB2YWx1ZTogc2VsZWN0b3IudmFsdWUgfSksXG4gICAgICAgICAgcGFyc2VyLmNsYXNzTmFtZSh7IHZhbHVlOiBwb3N0Zml4IH0pLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgLy8gcmVtb3ZlIDpnbG9iYWxcbiAgICBzZWxlY3RvcnMud2Fsaygoc2VsZWN0b3IpID0+IHtcbiAgICAgIGlmIChzZWxlY3Rvci50eXBlID09PSAncHNldWRvJyAmJiBzZWxlY3Rvci52YWx1ZSA9PT0gJzpnbG9iYWwnKSB7XG4gICAgICAgIHNlbGVjdG9yLnJlcGxhY2VXaXRoKC4uLnNlbGVjdG9yLm5vZGVzKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfTtcbiAgY29uc3QgdHJhbnNmb3JtZWQgPSBwYXJzZXIodHJhbnNmb3JtKS5wcm9jZXNzU3luYyhzZWwpO1xuXG4gIHJldHVybiB0cmFuc2Zvcm1lZDtcbn1cblxuZnVuY3Rpb24gaXNOdGhPZlR5cGUobm9kZSkge1xuICBpZiAoIW5vZGUpIHJldHVybiBmYWxzZTtcblxuICByZXR1cm4gbm9kZS5wYXJlbnQ/LnZhbHVlID09PSAnOm50aC1vZi10eXBlJyB8fCBpc050aE9mVHlwZShub2RlLnBhcmVudCk7XG59XG5cbmZ1bmN0aW9uIGlzSW5zaWRlS2V5ZnJhbWVzKG5vZGUpIHtcbiAgY29uc3QgcGFyZW50ID0gbm9kZS5wYXJlbnQ7XG5cbiAgaWYgKCFwYXJlbnQpIHJldHVybiBmYWxzZTtcbiAgaWYgKHBhcmVudC50eXBlID09PSAnYXRydWxlJyAmJiBwYXJlbnQubmFtZSA9PT0gJ2tleWZyYW1lcycpIHJldHVybiB0cnVlO1xuXG4gIHJldHVybiBpc0luc2lkZUtleWZyYW1lcyhwYXJlbnQpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcmV3cml0ZUNzcyhjc3MsIHBvc3RmaXgsIGZpbGVOYW1lLCBsYXllck5hbWUpIHtcbiAgY29uc3QgYXN0ID0gcG9zdGNzcy5wYXJzZShjc3MpO1xuICAvKipcbiAgICoga2luZCA9PiBvcmlnaW5hbE5hbWUgPT4gcG9zdGZpeGVkTmFtZVxuICAgKiBAdHlwZSB7eyBba2luZDogc3RyaW5nXTogeyBbb3JpZ2luYWxOYW1lOiBzdHJpbmddOiBzdHJpbmcgfX19XG4gICAqL1xuICBjb25zdCByZWZlcmVuY2VhYmxlcyA9IHtcbiAgICBrZXlmcmFtZXM6IHt9LFxuICAgICdjb3VudGVyLXN0eWxlJzoge30sXG4gICAgJ3Bvc2l0aW9uLXRyeSc6IHt9LFxuICAgIHByb3BlcnR5OiB7fSxcbiAgfTtcblxuICBjb25zdCBhdmFpbGFibGVSZWZlcmVuY2VhYmxlcyA9IG5ldyBTZXQoT2JqZWN0LmtleXMocmVmZXJlbmNlYWJsZXMpKTtcblxuICBmdW5jdGlvbiBpc1JlZmVyZW5jZWFibGUobm9kZSkge1xuICAgIGlmIChub2RlLnR5cGUgIT09ICdhdHJ1bGUnKSByZXR1cm47XG5cbiAgICByZXR1cm4gYXZhaWxhYmxlUmVmZXJlbmNlYWJsZXMuaGFzKG5vZGUubmFtZSk7XG4gIH1cblxuICBmdW5jdGlvbiB1cGRhdGVEaXJlY3RSZWZlcmVuY2VzKG5vZGUpIHtcbiAgICBpZiAoIW5vZGUudmFsdWUpIHJldHVybjtcblxuICAgIGZvciAobGV0IFssIG1hcF0gb2YgT2JqZWN0LmVudHJpZXMocmVmZXJlbmNlYWJsZXMpKSB7XG4gICAgICBpZiAobWFwW25vZGUudmFsdWVdKSB7XG4gICAgICAgIG5vZGUudmFsdWUgPSBtYXBbbm9kZS52YWx1ZV07XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgZnVuY3Rpb24gdXBkYXRlU2hvcnRoYW5kQ29udGVudHMobm9kZSkge1xuICAgIGlmIChub2RlLnByb3AgPT09ICdhbmltYXRpb24nKSB7XG4gICAgICBsZXQgcGFydHMgPSBub2RlLnZhbHVlLnNwbGl0KCcgJyk7XG4gICAgICBsZXQgbWF0Y2ggPSBwYXJ0cy5maWx0ZXIoKHgpID0+IHJlZmVyZW5jZWFibGVzLmtleWZyYW1lc1t4XSk7XG5cbiAgICAgIGlmIChtYXRjaC5sZW5ndGgpIHtcbiAgICAgICAgbWF0Y2guZm9yRWFjaCgoeCkgPT4ge1xuICAgICAgICAgIGxldCByZXBsYWNlbWVudCA9IHJlZmVyZW5jZWFibGVzLmtleWZyYW1lc1t4XTtcblxuICAgICAgICAgIGlmICghcmVwbGFjZW1lbnQpIHJldHVybjtcblxuICAgICAgICAgIG5vZGUudmFsdWUgPSBub2RlLnZhbHVlLnJlcGxhY2UoeCwgcmVwbGFjZW1lbnQpO1xuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBmb3IgKGxldCBbbG9va0ZvciwgcmVwbGFjZVdpdGhdIG9mIE9iamVjdC5lbnRyaWVzKFxuICAgICAgcmVmZXJlbmNlYWJsZXMucHJvcGVydHksXG4gICAgKSkge1xuICAgICAgbGV0IGxvb2tGb3JWYXIgPSBgdmFyKCR7bG9va0Zvcn0pYDtcbiAgICAgIGxldCByZXBsYWNlV2l0aFZhciA9IGB2YXIoJHtyZXBsYWNlV2l0aH0pYDtcblxuICAgICAgbm9kZS52YWx1ZSA9IG5vZGUudmFsdWUucmVwbGFjZShsb29rRm9yVmFyLCByZXBsYWNlV2l0aFZhcik7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFdlIGhhdmUgdG8gZG8gdHdvIHBhc3NlczpcbiAgICogMS4gcG9zdGZpeCBhbGwgdGhlIHJlZmVyZW5jZWFibGUgc3ludGF4XG4gICAqIDIuIHBvc3RmaXggYXMgbm9ybWFsLCBidXQgYWxzbyBjaGVja2luZyB2YWx1ZXMgb2YgQ1NTIHByb3BlcnRpZXNcbiAgICogICAgdGhhdCBjb3VsZCBtYXRjaCBwb3N0Zml4ZWQgcmVmZXJlbmNlYWJsZXMgZnJvbSBzdGVwIDFcbiAgICovXG5cbiAgLy8gU3RlcCAxOiBmaW5kIHJlZmVyZW5jZWFibGVzXG4gIGFzdC53YWxrKChub2RlKSA9PiB7XG4gICAgLyoqXG4gICAgICogQGtleWZyYW1lcywgQGNvdW50ZXItc3R5bGUsIGV0Y1xuICAgICAqL1xuICAgIGlmIChpc1JlZmVyZW5jZWFibGUobm9kZSkpIHtcbiAgICAgIGxldCBuYW1lID0gbm9kZS5uYW1lO1xuICAgICAgbGV0IHsgb3JpZ2luYWxOYW1lLCBwb3N0Zml4ZWROYW1lIH0gPSByZXdyaXRlUmVmZXJlbmNlYWJsZShub2RlLCBwb3N0Zml4KTtcblxuICAgICAgcmVmZXJlbmNlYWJsZXNbbmFtZV1bb3JpZ2luYWxOYW1lXSA9IHBvc3RmaXhlZE5hbWU7XG5cbiAgICAgIHJldHVybjtcbiAgICB9XG4gIH0pO1xuXG4gIC8vIFN0ZXAgMjogcG9zdGZpeCBhbmQgdXBkYXRlIHJlZmVyZW5jZWQgcmVmZXJlbmNlYWJsZXNcbiAgYXN0LndhbGsoKG5vZGUpID0+IHtcbiAgICBpZiAoaXNEZWNsYXJhdGlvbihub2RlKSkge1xuICAgICAgdXBkYXRlRGlyZWN0UmVmZXJlbmNlcyhub2RlKTtcbiAgICAgIHVwZGF0ZVNob3J0aGFuZENvbnRlbnRzKG5vZGUpO1xuXG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKGlzUnVsZShub2RlKSkge1xuICAgICAgLyoqXG4gICAgICAgKiBUaGUgaW5uZXItY29udGVudHMgb2YgYSBrZXlmcmFtZSBhcmUgcGVyY2VudGFnZXMsIHJhdGhlciB0aGFuIHNlbGVjdG9yc1xuICAgICAgICovXG4gICAgICBpZiAoaXNJbnNpZGVLZXlmcmFtZXMobm9kZSkpIHJldHVybjtcblxuICAgICAgbm9kZS5zZWxlY3RvciA9IHJld3JpdGVTZWxlY3Rvcihub2RlLnNlbGVjdG9yLCBwb3N0Zml4KTtcblxuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgfSk7XG5cbiAgY29uc3QgcmV3cml0dGVuQ3NzID0gYXN0LnRvU3RyaW5nKCk7XG5cbiAgcmV0dXJuIChcbiAgICBbXG4gICAgICBgLyogJHtmaWxlTmFtZX0gKi9gLFxuICAgICAgbGF5ZXJOYW1lID8gYEBsYXllciAke2xheWVyTmFtZX0ge2AgOiAnJyxcbiAgICAgIHJld3JpdHRlbkNzcy50cmltRW5kKCksXG4gICAgICBsYXllck5hbWUgPyBgfWAgOiAnJyxcbiAgICBdXG4gICAgICAuZmlsdGVyKEJvb2xlYW4pXG4gICAgICAuam9pbignXFxuJykgKyAnXFxuJ1xuICApO1xufVxuIiwiaW1wb3J0IHBhdGggZnJvbSAnbm9kZTpwYXRoJztcblxuY29uc3QgS0VZID0gJ2VtYmVyLXNjb3BlZC5jc3MnO1xuY29uc3QgU0VQID0gJ19fXyc7XG5cbmV4cG9ydCBjb25zdCByZXF1ZXN0ID0ge1xuICBpczoge1xuICAgIGlubGluZShyZXF1ZXN0KSB7XG4gICAgICByZXR1cm4gcmVxdWVzdC5pbmNsdWRlcyhLRVkpO1xuICAgIH0sXG4gICAgY29sb2NhdGVkKHJlcXVlc3QpIHtcbiAgICAgIHJldHVybiByZXF1ZXN0LmluY2x1ZGVzKCcuY3NzP3Njb3BlZD0nKTtcbiAgICB9LFxuICB9LFxuICBpbmxpbmU6IHtcbiAgICAvKipcbiAgICAgKiBNYWtlcyByZXF1ZXN0IFVSTCBmb3IgZW1iZWRkaW5nIGA8c3R5bGU+YCBhcyBgPGxpbms+YCBpbnRvIHRoZSBgPGhlYWQ+YFxuICAgICAqXG4gICAgICogQHBhcmFtIHtzdHJpbmd9IGNzc0hhc2ggdGhlIGhhc2ggb2YgdGhlIENTUyBjb250ZW50c1xuICAgICAqIEBwYXJhbSB7c3RyaW5nfSBwb3N0Zml4IHRoZSBoYXNoIG9mIHRoZSBmaWxlIHRoYXQgX2luY2x1ZGVzXyB0aGUgbGlua2VkIGZpbGVcbiAgICAgKiBAcGFyYW0ge3N0cmluZ30gY3NzQ29udGVudHMgdGhlIGNvbnRlbnRzIG9mIHRoZSBDU1MgZmlsZVxuICAgICAqIEBwYXJhbSB7c3RyaW5nfSBbbGFuZ10gb3B0aW9uYWwgcHJlcHJvY2Vzc29yIGxhbmd1YWdlIChlLmcuICdzY3NzJywgJ3Nhc3MnLCAnbGVzcycpXG4gICAgICovXG4gICAgY3JlYXRlKGNzc0hhc2gsIHBvc3RmaXgsIGNzc0NvbnRlbnRzLCBsYW5nKSB7XG4gICAgICBsZXQgdXJsID0gYC4vJHtwb3N0Zml4fSR7U0VQfSR7Y3NzSGFzaH0uJHtLRVl9P2Nzcz0ke2VuY29kZVVSSUNvbXBvbmVudChjc3NDb250ZW50cyl9YDtcblxuICAgICAgaWYgKGxhbmcpIHtcbiAgICAgICAgdXJsICs9IGAmbGFuZz0ke2VuY29kZVVSSUNvbXBvbmVudChsYW5nKX1gO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gdXJsO1xuICAgIH0sXG4gICAgZGVjb2RlKHJlcXVlc3QpIHtcbiAgICAgIGxldCBbbGVmdCwgcXBzXSA9IHJlcXVlc3Quc3BsaXQoJz8nKTtcblxuICAgICAgbGVmdCA9IGxlZnQuc2xpY2UoMikucmVwbGFjZShgLiR7S0VZfWAsICcnKTtcblxuICAgICAgbGV0IFtwb3N0Zml4LCBoYXNoXSA9IGxlZnQuc3BsaXQoU0VQKTtcblxuICAgICAgbGV0IHNlYXJjaCA9IG5ldyBVUkxTZWFyY2hQYXJhbXMocXBzKTtcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgaGFzaCxcbiAgICAgICAgcG9zdGZpeCxcbiAgICAgICAgY3NzOiBzZWFyY2guZ2V0KCdjc3MnKSxcbiAgICAgICAgZnJvbTogc2VhcmNoLmdldCgnZnJvbScpLFxuICAgICAgICBsYW5nOiBzZWFyY2guZ2V0KCdsYW5nJyksXG4gICAgICB9O1xuICAgIH0sXG4gIH0sXG4gIGNvbG9jYXRlZDoge1xuICAgIC8qKlxuICAgICAqIE1ha2VzIHJlcXVlc3QgVVJMIGZvciBlbWJlZGRpbmcgc2VwYXJhdGUgQ1NTIEZpbGUgYXMgYDxsaW5rPmAgaW50byB0aGUgYDxoZWFkPmBcbiAgICAgKlxuICAgICAqIEBwYXJhbSB7c3RyaW5nfSBjc3NIYXNoIHRoZSBoYXNoIG9mIHRoZSBDU1MgY29udGVudHNcbiAgICAgKiBAcGFyYW0ge3N0cmluZ30gcG9zdGZpeCB0aGUgaGFzaCBvZiB0aGUgZmlsZSB0aGF0IF9pbmNsdWRlc18gdGhlIGxpbmtlZCBmaWxlXG4gICAgICogQHBhcmFtIHtzdHJpbmd9IGZpbGVQYXRoIHBhdGggdG8gdGhlIHNlcGFyYXRlIENTUyBGaWxlXG4gICAgICovXG4gICAgY3JlYXRlKGNzc0hhc2gsIHBvc3RmaXgsIGZpbGVQYXRoKSB7XG4gICAgICByZXR1cm4gYC4vJHtwYXRoLmJhc2VuYW1lKGZpbGVQYXRoKX0/c2NvcGVkPSR7cG9zdGZpeH0mY3NzSGFzaD0ke2Nzc0hhc2h9YDtcbiAgICB9LFxuICAgIGRlY29kZShyZXF1ZXN0KSB7XG4gICAgICBjb25zdCBbZmlsZU5hbWUsIHFzXSA9IHJlcXVlc3Quc3BsaXQoJz8nKTtcbiAgICAgIGNvbnN0IHNlYXJjaCA9IG5ldyBVUkxTZWFyY2hQYXJhbXMocXMpO1xuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBmaWxlTmFtZSxcbiAgICAgICAgY3NzSGFzaDogc2VhcmNoLmdldCgnY3NzSGFzaCcpLFxuICAgICAgICBwb3N0Zml4OiBzZWFyY2guZ2V0KCdzY29wZWQnKSxcbiAgICAgIH07XG4gICAgfSxcbiAgfSxcbn07XG4iLCJpbXBvcnQgeyByZWFkRmlsZVN5bmMgfSBmcm9tICdub2RlOmZzJztcbmltcG9ydCB7IGNyZWF0ZVJlcXVpcmUgfSBmcm9tICdub2RlOm1vZHVsZSc7XG5pbXBvcnQgcGF0aCBmcm9tICdub2RlOnBhdGgnO1xuXG5pbXBvcnQgeyByZXdyaXRlQ3NzIH0gZnJvbSAnLi4vbGliL2Nzcy9yZXdyaXRlLmpzJztcbmltcG9ydCB7IHJlcXVlc3QgfSBmcm9tICcuLi9saWIvcmVxdWVzdC5qcyc7XG5cbmNvbnN0IE1FVEEgPSAnc2NvcGVkLWNzczpjb2xvY2F0ZWQnO1xuXG4vKiogRmlsZSBleHRlbnNpb25zIHRoYXQgVml0ZSBjYW4gcHJlcHJvY2VzcyB2aWEgaXRzIENTUyBwcmVwcm9jZXNzb3IgcGlwZWxpbmUgKi9cbmNvbnN0IFBSRVBST0NFU1NFRF9FWFRFTlNJT05TID0gbmV3IFNldChbXG4gICcuc2NzcycsXG4gICcuc2FzcycsXG4gICcubGVzcycsXG4gICcuc3R5bCcsXG4gICcuc3R5bHVzJyxcbl0pO1xuXG4vKipcbiAqIFBsdWdpbiBmb3Igc3VwcG9ydGluZyBjb2xvY2F0ZWQgc3R5bGVzXG4gKlxuICogZS5nLjpcbiAqICBzcmMvY29tcG9uZW50cy9teS1jb21wb25lbnQuanNcbiAqICBzcmMvY29tcG9uZW50cy9teS1jb21wb25lbnQuY3NzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb2xvY2F0ZWQob3B0aW9ucyA9IHt9KSB7XG4gIGNvbnN0IENXRCA9IHByb2Nlc3MuY3dkKCk7XG5cbiAgLyoqIEB0eXBlIHtpbXBvcnQoJ3ZpdGUnKS5SZXNvbHZlZENvbmZpZyB8IHVuZGVmaW5lZH0gKi9cbiAgbGV0IHZpdGVDb25maWc7XG5cbiAgLyoqIEB0eXBlIHsoKGNvZGU6IHN0cmluZywgZmlsZW5hbWU6IHN0cmluZywgY29uZmlnOiB1bmtub3duKSA9PiBQcm9taXNlPHsgY29kZTogc3RyaW5nIH0+KSB8IHVuZGVmaW5lZH0gKi9cbiAgbGV0IHByZXByb2Nlc3NDU1M7XG5cbiAgLyoqXG4gICAqXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBpZCB0aGUgcmVxdWVzdCBpZCAvIHdoYXQgd2FzIGltcG9ydGVkXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlUGF0aCAgcGF0aCBvbiBkaXNrXG4gICAqIEByZXR1cm5zXG4gICAqL1xuICBmdW5jdGlvbiBidWlsZFJlc3BvbnNlKGlkLCBmaWxlUGF0aCkge1xuICAgIGNvbnN0IHBhcnNlZCA9IHJlcXVlc3QuY29sb2NhdGVkLmRlY29kZShpZCk7XG4gICAgY29uc3QgcmVsYXRpdmVGaWxlUGF0aCA9IHBhdGgucmVsYXRpdmUoQ1dELCBmaWxlUGF0aCk7XG5cbiAgICByZXR1cm4ge1xuICAgICAgaWQ6IGZpbGVQYXRoLFxuICAgICAgbWV0YToge1xuICAgICAgICBbTUVUQV06IHtcbiAgICAgICAgICBwb3N0Zml4OiBwYXJzZWQucG9zdGZpeCxcbiAgICAgICAgICBmaWxlTmFtZTogcmVsYXRpdmVGaWxlUGF0aCxcbiAgICAgICAgICBmdWxsUGF0aDogZmlsZVBhdGgsXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgIH07XG4gIH1cblxuICByZXR1cm4ge1xuICAgIG5hbWU6ICdlbWJlci1zY29wZWQtY3NzOmNvbG9jYXRlZCcsXG4gICAgcmVzb2x2ZUlkKGlkLCBpbXBvcnRlcikge1xuICAgICAgLy8gaGFuZGxlczogc29tZS1maWxlLmNzcz9zY29wZWQ9W3Bvc3RmaXhdXG4gICAgICAvLyB0aGlzIGlzIG9ubHkgcnVuIGluIHJvbGx1cCwgdml0ZSBoYW5kbGVzIGl0IGRpZmZlcmVudGx5XG4gICAgICBpZiAocmVxdWVzdC5pcy5jb2xvY2F0ZWQoaWQpKSB7XG4gICAgICAgIGNvbnN0IHBhcnNlZCA9IHJlcXVlc3QuY29sb2NhdGVkLmRlY29kZShpZCk7XG5cbiAgICAgICAgY29uc3QgZmlsZVBhdGggPSBwYXRoLnJlc29sdmUoXG4gICAgICAgICAgcGF0aC5kaXJuYW1lKGltcG9ydGVyKSxcbiAgICAgICAgICBwYXRoLmJhc2VuYW1lKHBhcnNlZC5maWxlTmFtZSksXG4gICAgICAgICk7XG5cbiAgICAgICAgcmV0dXJuIGJ1aWxkUmVzcG9uc2UoaWQsIGZpbGVQYXRoKTtcbiAgICAgIH1cbiAgICB9LFxuICAgIGxvYWQoaWQpIHtcbiAgICAgIGNvbnN0IG1ldGEgPSB0aGlzLmdldE1vZHVsZUluZm8oaWQpPy5tZXRhPy5bTUVUQV07XG5cbiAgICAgIGlmIChtZXRhKSB7XG4gICAgICAgIHRoaXMuYWRkV2F0Y2hGaWxlKG1ldGEuZnVsbFBhdGgpO1xuXG4gICAgICAgIGxldCBjb2RlID0gcmVhZEZpbGVTeW5jKG1ldGEuZnVsbFBhdGgsICd1dGYtOCcpO1xuXG4gICAgICAgIGxldCBjc3MgPSByZXdyaXRlQ3NzKFxuICAgICAgICAgIGNvZGUsXG4gICAgICAgICAgbWV0YS5wb3N0Zml4LFxuICAgICAgICAgIG1ldGEuZmlsZU5hbWUsXG4gICAgICAgICAgb3B0aW9ucy5sYXllck5hbWUsXG4gICAgICAgICk7XG5cbiAgICAgICAgcmV0dXJuIGNzcztcbiAgICAgIH1cbiAgICB9LFxuICAgIHZpdGU6IHtcbiAgICAgIGFzeW5jIGNvbmZpZ1Jlc29sdmVkKGNvbmZpZykge1xuICAgICAgICB2aXRlQ29uZmlnID0gY29uZmlnO1xuXG4gICAgICAgIC8vIFJlc29sdmUgVml0ZSdzIHByZXByb2Nlc3NDU1MgZnJvbSB0aGUgYXBwIHJvb3QgdG8gZW5zdXJlIHdlIGZpbmRcbiAgICAgICAgLy8gdGhlIGNvcnJlY3QgVml0ZSBpbnN0YWxsYXRpb24gKG5vdCBhIHN0YWxlIG9yIG1pc3Npbmcgb25lKS5cbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBjb25zdCByZXF1aXJlID0gY3JlYXRlUmVxdWlyZShjb25maWcucm9vdCk7XG4gICAgICAgICAgY29uc3Qgdml0ZVBhdGggPSByZXF1aXJlLnJlc29sdmUoJ3ZpdGUnKTtcbiAgICAgICAgICBjb25zdCB2aXRlTW9kdWxlID0gYXdhaXQgaW1wb3J0KHZpdGVQYXRoKTtcblxuICAgICAgICAgIHByZXByb2Nlc3NDU1MgPSB2aXRlTW9kdWxlLnByZXByb2Nlc3NDU1M7XG4gICAgICAgIH0gY2F0Y2gge1xuICAgICAgICAgIC8vIFZpdGUgbWF5IG5vdCBiZSByZXNvbHZhYmxlIGZyb20gdGhlIGNvbmZpZyByb290IGluIHNvbWUgc2V0dXBzO1xuICAgICAgICAgIC8vIHByZXByb2Nlc3NvciBzdXBwb3J0IGZvciBjb2xvY2F0ZWQgLnNjc3MgZmlsZXMgd2lsbCB0aHJvdyBhIGNsZWFyXG4gICAgICAgICAgLy8gZXJyb3IgYXQgbG9hZCB0aW1lIGlmIHVzZWQuXG4gICAgICAgIH1cbiAgICAgIH0sXG5cbiAgICAgIC8qKlxuICAgICAgICogVGhlcmUgbWF5IG5vdCBiZSBtZXRhIGZvciB0aGlzIHJlcXVlc3QgeWV0LlxuICAgICAgICpcbiAgICAgICAqIEBwYXJhbSB7Kn0gaWRcbiAgICAgICAqL1xuICAgICAgYXN5bmMgbG9hZChpZCkge1xuICAgICAgICBpZiAocmVxdWVzdC5pcy5jb2xvY2F0ZWQoaWQpKSB7XG4gICAgICAgICAgY29uc3QgcGFyc2VkID0gcmVxdWVzdC5jb2xvY2F0ZWQuZGVjb2RlKGlkKTtcblxuICAgICAgICAgIGxldCBjb2RlID0gcmVhZEZpbGVTeW5jKHBhcnNlZC5maWxlTmFtZSwgJ3V0Zi04Jyk7XG4gICAgICAgICAgbGV0IHJlbGF0aXZlRmlsZVBhdGggPSBwYXRoLnJlbGF0aXZlKENXRCwgcGFyc2VkLmZpbGVOYW1lKTtcblxuICAgICAgICAgIGNvbnN0IGV4dCA9IHBhdGguZXh0bmFtZShwYXJzZWQuZmlsZU5hbWUpLnRvTG93ZXJDYXNlKCk7XG5cbiAgICAgICAgICBpZiAoUFJFUFJPQ0VTU0VEX0VYVEVOU0lPTlMuaGFzKGV4dCkpIHtcbiAgICAgICAgICAgIGlmICghdml0ZUNvbmZpZyB8fCAhcHJlcHJvY2Vzc0NTUykge1xuICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICAgICAgYFtlbWJlci1zY29wZWQtY3NzXSBDb2xvY2F0ZWQgQ1NTIGZpbGUgd2l0aCBleHRlbnNpb24gJyR7ZXh0fScgcmVxdWlyZXMgVml0ZS4gYCArXG4gICAgICAgICAgICAgICAgICBgQ1NTIHByZXByb2Nlc3NpbmcgaXMgb25seSBzdXBwb3J0ZWQgaW4gVml0ZSBidWlsZHMuYCxcbiAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgcHJlcHJvY2Vzc0NTUyhcbiAgICAgICAgICAgICAgY29kZSxcbiAgICAgICAgICAgICAgcGFyc2VkLmZpbGVOYW1lLFxuICAgICAgICAgICAgICB2aXRlQ29uZmlnLFxuICAgICAgICAgICAgKTtcblxuICAgICAgICAgICAgY29kZSA9IHJlc3VsdC5jb2RlO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIGxldCBjc3MgPSByZXdyaXRlQ3NzKFxuICAgICAgICAgICAgY29kZSxcbiAgICAgICAgICAgIHBhcnNlZC5wb3N0Zml4LFxuICAgICAgICAgICAgcmVsYXRpdmVGaWxlUGF0aCxcbiAgICAgICAgICAgIG9wdGlvbnMubGF5ZXJOYW1lLFxuICAgICAgICAgICk7XG5cbiAgICAgICAgICByZXR1cm4gY3NzO1xuICAgICAgICB9XG4gICAgICB9LFxuICAgIH0sXG4gIH07XG59XG4iLCJpbXBvcnQgeyBjcmVhdGVSZXF1aXJlIH0gZnJvbSAnbm9kZTptb2R1bGUnO1xuaW1wb3J0IHBhdGggZnJvbSAnbm9kZTpwYXRoJztcblxuaW1wb3J0IHsgcmV3cml0ZUNzcyB9IGZyb20gJy4uL2xpYi9jc3MvcmV3cml0ZS5qcyc7XG5pbXBvcnQgeyByZXF1ZXN0IH0gZnJvbSAnLi4vbGliL3JlcXVlc3QuanMnO1xuXG5jb25zdCBNRVRBID0gJ3Njb3BlZC1jc3M6aW5saW5lJztcblxuLyoqXG4gKiBQbHVnaW4gZm9yIHN1cHBvcnRpbmcgdGhlIHN0eWxlcyBmcm9tXG4gKlxuICogPHRlbXBsYXRlPlxuICogICA8c3R5bGU+Li4uPC9zdHlsZT5cbiAqIDwvdGVtcGxhdGU+XG4gKlxuICogVGhpcyBwbHVnaW4gY2FuJ3QgaGF2ZSBITVIgZm9yIENTUyBiZWNhdXNlIGNoYW5nZXMgdG8gdGhlIENTUyBjb250ZW50IGFsdGVycyB0aGUgdGVtcGxhdGUgY29udGVudFxuICovXG5leHBvcnQgZnVuY3Rpb24gaW5saW5lKG9wdGlvbnMgPSB7fSkge1xuICBjb25zdCBDV0QgPSBwcm9jZXNzLmN3ZCgpO1xuXG4gIC8qKiBAdHlwZSB7aW1wb3J0KCd2aXRlJykuUmVzb2x2ZWRDb25maWcgfCB1bmRlZmluZWR9ICovXG4gIGxldCB2aXRlQ29uZmlnO1xuXG4gIC8qKiBAdHlwZSB7KChjb2RlOiBzdHJpbmcsIGZpbGVuYW1lOiBzdHJpbmcsIGNvbmZpZzogdW5rbm93bikgPT4gUHJvbWlzZTx7IGNvZGU6IHN0cmluZyB9PikgfCB1bmRlZmluZWR9ICovXG4gIGxldCBwcmVwcm9jZXNzQ1NTO1xuXG4gIC8qKlxuICAgKiBAcGFyYW0ge3N0cmluZ30gaWQgdGhlIHJlcXVlc3QgaWQgLyB3aGF0IHdhcyBpbXBvcnRlZFxuICAgKi9cbiAgZnVuY3Rpb24gYnVpbGRSZXNwb25zZShpZCwgZmlsZVBhdGgpIHtcbiAgICBjb25zdCBwYXJzZWQgPSByZXF1ZXN0LmlubGluZS5kZWNvZGUoaWQpO1xuXG4gICAgY29uc3QgcmVsYXRpdmVGaWxlUGF0aCA9IHBhdGgucmVsYXRpdmUoQ1dELCBmaWxlUGF0aCk7XG5cbiAgICByZXR1cm4ge1xuICAgICAgaWQ6IGZpbGVQYXRoLnNwbGl0KCc/JylbMF0sXG4gICAgICBtZXRhOiB7XG4gICAgICAgIFtNRVRBXToge1xuICAgICAgICAgIHJhd0NzczogcGFyc2VkLmNzcyxcbiAgICAgICAgICBwb3N0Zml4OiBwYXJzZWQucG9zdGZpeCxcbiAgICAgICAgICBmaWxlTmFtZTogcmVsYXRpdmVGaWxlUGF0aCxcbiAgICAgICAgICBsYW5nOiBwYXJzZWQubGFuZyxcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgfTtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgbmFtZTogJ2VtYmVyLXNjb3BlZC1jc3M6aW5saW5lJyxcbiAgICByZXNvbHZlSWQoaWQsIGltcG9ydGVyKSB7XG4gICAgICBpZiAocmVxdWVzdC5pcy5pbmxpbmUoaWQpKSB7XG4gICAgICAgIGNvbnN0IHBhcnNlZCA9IHJlcXVlc3QuaW5saW5lLmRlY29kZShpZCk7XG5cbiAgICAgICAgY29uc3QgZmlsZVBhdGggPSBwYXRoLnJlc29sdmUoXG4gICAgICAgICAgcGF0aC5kaXJuYW1lKGltcG9ydGVyKSxcbiAgICAgICAgICBgJHtwYXRoLmJhc2VuYW1lKGltcG9ydGVyLCBwYXRoLmV4dG5hbWUoaW1wb3J0ZXIpKX0tJHtwYXJzZWQuaGFzaH0uY3NzYCxcbiAgICAgICAgKTtcblxuICAgICAgICByZXR1cm4gYnVpbGRSZXNwb25zZShpZCwgZmlsZVBhdGgpO1xuICAgICAgfVxuICAgIH0sXG4gICAgYXN5bmMgbG9hZChpZCkge1xuICAgICAgY29uc3QgbWV0YSA9IHRoaXMuZ2V0TW9kdWxlSW5mbyhpZCk/Lm1ldGE/LltNRVRBXTtcblxuICAgICAgaWYgKG1ldGEpIHtcbiAgICAgICAgbGV0IHJhd0NzcyA9IG1ldGEucmF3Q3NzO1xuXG4gICAgICAgIGlmIChtZXRhLmxhbmcpIHtcbiAgICAgICAgICBpZiAoIXZpdGVDb25maWcgfHwgIXByZXByb2Nlc3NDU1MpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICAgICAgYFtlbWJlci1zY29wZWQtY3NzXSA8c3R5bGUgc2NvcGVkIGxhbmc9XCIke21ldGEubGFuZ31cIj4gcmVxdWlyZXMgVml0ZS4gYCArXG4gICAgICAgICAgICAgICAgYENTUyBwcmVwcm9jZXNzaW5nIHZpYSB0aGUgJ2xhbmcnIGF0dHJpYnV0ZSBpcyBvbmx5IHN1cHBvcnRlZCBpbiBWaXRlIGJ1aWxkcy5gLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCBmYWtlRmlsZW5hbWUgPSBgJHttZXRhLmZpbGVOYW1lfS4ke21ldGEubGFuZ31gO1xuICAgICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHByZXByb2Nlc3NDU1MocmF3Q3NzLCBmYWtlRmlsZW5hbWUsIHZpdGVDb25maWcpO1xuXG4gICAgICAgICAgcmF3Q3NzID0gcmVzdWx0LmNvZGU7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBjc3MgPSByZXdyaXRlQ3NzKFxuICAgICAgICAgIHJhd0NzcyxcbiAgICAgICAgICBtZXRhLnBvc3RmaXgsXG4gICAgICAgICAgYDxpbmxpbmU+LyR7bWV0YS5maWxlTmFtZX1gLFxuICAgICAgICAgIG9wdGlvbnMubGF5ZXJOYW1lLFxuICAgICAgICApO1xuXG4gICAgICAgIHJldHVybiBjc3M7XG4gICAgICB9XG4gICAgfSxcbiAgICB2aXRlOiB7XG4gICAgICBhc3luYyBjb25maWdSZXNvbHZlZChjb25maWcpIHtcbiAgICAgICAgdml0ZUNvbmZpZyA9IGNvbmZpZztcblxuICAgICAgICAvLyBSZXNvbHZlIFZpdGUncyBwcmVwcm9jZXNzQ1NTIGZyb20gdGhlIGFwcCByb290IHRvIGVuc3VyZSB3ZSBmaW5kXG4gICAgICAgIC8vIHRoZSBjb3JyZWN0IFZpdGUgaW5zdGFsbGF0aW9uIChub3QgYSBzdGFsZSBvciBtaXNzaW5nIG9uZSkuXG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgY29uc3QgcmVxdWlyZSA9IGNyZWF0ZVJlcXVpcmUocGF0aC5qb2luKGNvbmZpZy5yb290LCAncGFja2FnZS5qc29uJykpO1xuICAgICAgICAgIGNvbnN0IHZpdGVQYXRoID0gcmVxdWlyZS5yZXNvbHZlKCd2aXRlJyk7XG4gICAgICAgICAgY29uc3Qgdml0ZU1vZHVsZSA9IGF3YWl0IGltcG9ydCh2aXRlUGF0aCk7XG5cbiAgICAgICAgICBwcmVwcm9jZXNzQ1NTID0gdml0ZU1vZHVsZS5wcmVwcm9jZXNzQ1NTO1xuICAgICAgICB9IGNhdGNoIHtcbiAgICAgICAgICAvLyBWaXRlIG1heSBub3QgYmUgcmVzb2x2YWJsZSBmcm9tIHRoZSBjb25maWcgcm9vdCBpbiBzb21lIHNldHVwcztcbiAgICAgICAgICAvLyBsYW5nPSBzdXBwb3J0IHdpbGwgdGhyb3cgYSBjbGVhciBlcnJvciBhdCBsb2FkIHRpbWUgaWYgdXNlZC5cbiAgICAgICAgfVxuICAgICAgfSxcbiAgICB9LFxuICB9O1xufVxuIiwiaW1wb3J0IHsgY3JlYXRlVW5wbHVnaW4gfSBmcm9tICd1bnBsdWdpbic7XG5cbmltcG9ydCB7IGNvbG9jYXRlZCB9IGZyb20gJy4vdW5wbHVnaW4tY29sb2NhdGVkLmpzJztcbmltcG9ydCB7IGlubGluZSB9IGZyb20gJy4vdW5wbHVnaW4taW5saW5lLmpzJztcblxuLyoqXG4gKiBUaGUgcGx1Z2luIHRoYXQgaGFuZGxlcyBDU1MgcmVxdWVzdHMgZm9yIGA8c3R5bGU+YCBlbGVtZW50cyBhbmQgdHJhbnNmb3Jtc1xuICogZm9yIGV4aXN0aW5nIGZpbGVzXG4gKlxuICogdml0ZTogQ1NTIGZpbGVzIGFyZSByZXNvbHZlZCBieSB2aXRlLiBXZSB1c2UgdGhlaXIgcmVzb2x2ZXIgdG8gYWxzbyBnZXRcbiAqICAgICAgIEhNUi4gVGhhdCBpcywgZm9yIGFsbCBub24tcGh5c2ljYWwgQ1NTIGZpbGVzLCB3ZSBleHRlbmQgdml0ZSBieSBvdXJcbiAqICAgICAgIHJlc29sdmVyIGFuZCBhbHNvIGNhbiBlbnJpY2ggbWV0YWRhdGEgdG8gaXQgKGZvciBiZXR0ZXIgZGVidWdnaW5nKVxuICovXG5leHBvcnQgY29uc3QgdW5wbHVnaW4gPSBjcmVhdGVVbnBsdWdpbigob3B0aW9ucyA9IHt9KSA9PiB7XG4gIHJldHVybiBbY29sb2NhdGVkKG9wdGlvbnMpLCBpbmxpbmUob3B0aW9ucyldO1xufSk7XG4iLCJpbXBvcnQgcGF0aCBmcm9tICdub2RlOnBhdGgnO1xuXG5pbXBvcnQgeyBsZWFkaW5nU2xhc2hQYXRoIH0gZnJvbSAnLi9jb25zdC5qcyc7XG5pbXBvcnQgeyBmaW5kV29ya3NwYWNlUGF0aCB9IGZyb20gJy4vdXRpbHMuanMnO1xuXG4vKipcbiAqIHRlbXBsYXRlIHBsdWdpbnMgZG8gbm90IGhhbmQgdXMgdGhlIGNvcnJlY3QgZmlsZSBwYXRoLlxuICogYWRkaXRpb25hbGx5LCB3ZSBtYXkgbm90IGJlIGFibGUgdG8gcmVseSBvbiB0aGlzIGRhdGEgaW4gdGhlIGZ1dHVyZSxcbiAqIHNvIHRoaXMgZnVuY3Rpb25zIGFjdHMgYXMgYSBtZWFucyBvZiBub3JtYWxpemluZyBfd2hhdGV2ZXJfIHdlJ3JlIGdpdmVuXG4gKiBpbiB0aGUgZnV0dXJlLlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlbmFtZVxuICogQHJldHVybnMge3N0cmluZ30gdGhlIGFic29sdXRlIHBhdGggdG8gdGhlIGZpbGVcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZpeEZpbGVuYW1lKGZpbGVuYW1lKSB7XG4gIGxldCBmaWxlTmFtZSA9IGZpbGVuYW1lO1xuICBsZXQgd29ya3NwYWNlID0gZmluZFdvcmtzcGFjZVBhdGgoZmlsZU5hbWUpO1xuXG4gIC8qKlxuICAgKiBlbWJlci1zb3VyY2UgNS44OlxuICAgKiAtIHRoZSBmaWxlbmFtZSBsb29rcyBsaWtlIGFuIGFic29sdXRlIHBhdGgsIGJ1dCBzd2FwcGVkIG91dCB0aGUgJ2FwcCcgcGFydCBvZiB0aGUgcGF0aFxuICAgKiAgIHdpdGggdGhlIG1vZHVsZSBuYW1lLCBzbyB0aGUgZmlsZSBwYXRocyBuZXZlciBleGlzdCBvbiBkaXNrXG4gICAqXG4gICAqIC0gaW4gdml0ZSBhcHBzOlxuICAgKiAgIHRoZSAnYXBwJyBwYXJ0IF9tYXlfIGJlIGBzcmNgLCBzbyB3ZSBhbHNvIG5lZWQgdG8gZW5zdXJlIHRoYXQgYHNyY2AgaXMgZXhjbHVkZWQgYXMgd2VsbFxuICAgKi9cbiAgbGV0IGhhc0FwcERpciA9IGZpbGVOYW1lLmluY2x1ZGVzKHBhdGguam9pbih3b3Jrc3BhY2UsICdhcHAnKSk7XG4gIGxldCBoYXNTcmNEaXIgPSBmaWxlTmFtZS5pbmNsdWRlcyhwYXRoLmpvaW4od29ya3NwYWNlLCAnc3JjJykpO1xuXG4gIGlmIChcbiAgICAhKGhhc0FwcERpciB8fCBoYXNTcmNEaXIpICYmXG4gICAgIWZpbGVOYW1lLmluY2x1ZGVzKGxlYWRpbmdTbGFzaFBhdGguZW1icm9pZGVyRGlyKVxuICApIHtcbiAgICBsZXQgbWF5YmVNb2R1bGUgPSBmaWxlTmFtZS5yZXBsYWNlKHdvcmtzcGFjZSwgJycpO1xuICAgIGxldCBbbWF5YmVTY29wZSwgLi4ucmVzdF0gPSBtYXliZU1vZHVsZS5zcGxpdChwYXRoLnNlcCkuZmlsdGVyKEJvb2xlYW4pO1xuICAgIGxldCBwYXJ0cyA9IHJlc3Q7XG5cbiAgICBpZiAobWF5YmVTY29wZS5zdGFydHNXaXRoKCdAJykpIHtcbiAgICAgIGxldCBbLCAuLi5yZXN0ZXJdID0gcmVzdDtcblxuICAgICAgcGFydHMgPSByZXN0ZXI7XG4gICAgfVxuXG4gICAgbGV0IHJlbGF0aXZlID0gcGF0aC5qb2luKC4uLnBhcnRzKTtcblxuICAgIC8qKlxuICAgICAqIFdlIGRvbid0IGFjdHVhbGx5IGtub3cgaWYgdGhpcyBmaWxlIGlzIGFuIGFwcC5cbiAgICAgKiBpdCBjb3VsZCBiZSBhbiBhZGRvbiAodjEgb3IgdjIpXG4gICAgICpcbiAgICAgKiBTbyBoZXJlIHdlIGxvZyB0byBzZWUgaWYgd2UgaGF2ZSB1bmhhbmRsZWQgc2l0dWF0aW9ucy5cbiAgICAgKi9cbiAgICBsZXQgY2FuZGlkYXRlUGF0aCA9IHBhdGguam9pbih3b3Jrc3BhY2UsICdhcHAnLCByZWxhdGl2ZSk7XG5cbiAgICByZXR1cm4gY2FuZGlkYXRlUGF0aDtcbiAgfVxuXG4gIC8vIFRPRE86IHdoeSBhcmUgd2UgcGFzc2VkIGZpbGVzIHRvIG90aGVyIHByb2plY3RzP1xuICBpZiAoIWZpbGVOYW1lLmluY2x1ZGVzKHdvcmtzcGFjZSkpIHtcbiAgICByZXR1cm4gZmlsZU5hbWU7XG4gIH1cblxuICAvLyBGYWxsYmFjayB0byB3aGF0IHRoZSBwbHVnaW4gc3lzdGVtIGdpdmVzIHVzLlxuICAvLyBUaGlzIG1heSBiZSB3cm9uZywgYW5kIGlmIHdyb25nLCByZXZlYWxzXG4gIC8vIHVuaGFuZGxlZCBzY2VuYXJpb3Mgd2l0aCB0aGUgZmlsZSBuYW1lcyBpbiB0aGUgcGx1Z2luIGluZnJhXG4gIHJldHVybiBmaWxlTmFtZTtcbn1cbiIsImltcG9ydCAqIGFzIHJlY2FzdCBmcm9tICdlbWJlci10ZW1wbGF0ZS1yZWNhc3QnO1xuXG5pbXBvcnQgeyByZW5hbWVDbGFzcyB9IGZyb20gJy4vcmVuYW1lQ2xhc3MuanMnO1xuXG5leHBvcnQgZnVuY3Rpb24gdGVtcGxhdGVQbHVnaW4oeyBjbGFzc2VzLCB0YWdzLCBwb3N0Zml4IH0pIHtcbiAgbGV0IHN0YWNrID0gW107XG4gIC8vIHNjb3BlZC1jbGFzcyBpcyBhIGdsb2JhbCB3ZSBhbGxvdyBpbiBoYnNcbiAgLy8gc2NvcGVkQ2xhc3MgaXMgaW1wb3J0YWJsZSwgYW5kIHdlJ2xsIGVycm9yIGlmIHNvbWVvbmUgdHJpZXMgdG8gcmVuYW1lIGl0XG4gIGxldCBzY29wZWRDbGFzc0NhbmRpZGF0ZXMgPSBbJ3Njb3BlZC1jbGFzcycsICdzY29wZWRDbGFzcyddO1xuXG4gIGZ1bmN0aW9uIGlzU2NvcGVkQ2xhc3Moc3RyKSB7XG4gICAgaWYgKCFzdHIpIHJldHVybiBmYWxzZTtcblxuICAgIHJldHVybiBzY29wZWRDbGFzc0NhbmRpZGF0ZXMuc29tZSgoY2FuZGlkYXRlKSA9PiBjYW5kaWRhdGUgPT09IHN0cik7XG4gIH1cblxuICByZXR1cm4ge1xuICAgIEF0dHJOb2RlKG5vZGUpIHtcbiAgICAgIGlmIChub2RlLm5hbWUgPT09ICdjbGFzcycpIHtcbiAgICAgICAgaWYgKG5vZGUudmFsdWUudHlwZSA9PT0gJ1RleHROb2RlJyAmJiBub2RlLnZhbHVlLmNoYXJzKSB7XG4gICAgICAgICAgY29uc3QgcmVuYW1lZENsYXNzID0gcmVuYW1lQ2xhc3Mobm9kZS52YWx1ZS5jaGFycywgcG9zdGZpeCwgY2xhc3Nlcyk7XG5cbiAgICAgICAgICBub2RlLnZhbHVlLmNoYXJzID0gcmVuYW1lZENsYXNzO1xuICAgICAgICB9IGVsc2UgaWYgKG5vZGUudmFsdWUudHlwZSA9PT0gJ0NvbmNhdFN0YXRlbWVudCcpIHtcbiAgICAgICAgICBmb3IgKGxldCBwYXJ0IG9mIG5vZGUudmFsdWUucGFydHMpIHtcbiAgICAgICAgICAgIGlmIChwYXJ0LnR5cGUgPT09ICdUZXh0Tm9kZScgJiYgcGFydC5jaGFycykge1xuICAgICAgICAgICAgICBjb25zdCByZW5hbWVkQ2xhc3MgPSByZW5hbWVDbGFzcyhwYXJ0LmNoYXJzLCBwb3N0Zml4LCBjbGFzc2VzKTtcblxuICAgICAgICAgICAgICBwYXJ0LmNoYXJzID0gcmVuYW1lZENsYXNzO1xuICAgICAgICAgICAgfSBlbHNlIGlmIChwYXJ0LnR5cGUgPT09ICdNdXN0YWNoZVN0YXRlbWVudCcpIHtcbiAgICAgICAgICAgICAgcmVjYXN0LnRyYXZlcnNlKHBhcnQsIHtcbiAgICAgICAgICAgICAgICBTdHJpbmdMaXRlcmFsKG5vZGUpIHtcbiAgICAgICAgICAgICAgICAgIGNvbnN0IHJlbmFtZWRDbGFzcyA9IHJlbmFtZUNsYXNzKFxuICAgICAgICAgICAgICAgICAgICBub2RlLnZhbHVlLFxuICAgICAgICAgICAgICAgICAgICBwb3N0Zml4LFxuICAgICAgICAgICAgICAgICAgICBjbGFzc2VzLFxuICAgICAgICAgICAgICAgICAgKTtcblxuICAgICAgICAgICAgICAgICAgbm9kZS52YWx1ZSA9IHJlbmFtZWRDbGFzcztcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9LFxuXG4gICAgRWxlbWVudE5vZGUobm9kZSkge1xuICAgICAgaWYgKHRhZ3MuaGFzKG5vZGUudGFnKSkge1xuICAgICAgICAvLyBjaGVjayBpZiBjbGFzcyBhdHRyaWJ1dGUgYWxyZWFkeSBleGlzdHNcbiAgICAgICAgY29uc3QgY2xhc3NBdHRyID0gbm9kZS5hdHRyaWJ1dGVzLmZpbmQoKGF0dHIpID0+IGF0dHIubmFtZSA9PT0gJ2NsYXNzJyk7XG5cbiAgICAgICAgaWYgKGNsYXNzQXR0cikge1xuICAgICAgICAgIGNsYXNzQXR0ci52YWx1ZS5jaGFycyArPSAnICcgKyBwb3N0Zml4O1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIHB1c2ggY2xhc3MgYXR0cmlidXRlXG4gICAgICAgICAgbm9kZS5hdHRyaWJ1dGVzLnB1c2goXG4gICAgICAgICAgICByZWNhc3QuYnVpbGRlcnMuYXR0cignY2xhc3MnLCByZWNhc3QuYnVpbGRlcnMudGV4dChwb3N0Zml4KSksXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0sXG5cbiAgICBBbGw6IHtcbiAgICAgIGVudGVyKG5vZGUpIHtcbiAgICAgICAgc3RhY2sucHVzaChub2RlKTtcbiAgICAgIH0sXG4gICAgICBleGl0KCkge1xuICAgICAgICBzdGFjay5wb3AoKTtcbiAgICAgIH0sXG4gICAgfSxcblxuICAgIE11c3RhY2hlU3RhdGVtZW50KG5vZGUpIHtcbiAgICAgIGxldCBjc3NDbGFzcztcblxuICAgICAgaWYgKFxuICAgICAgICBpc1Njb3BlZENsYXNzKGdldFZhbHVlKG5vZGUucGF0aCkpICYmXG4gICAgICAgIG5vZGUucGFyYW1zPy5sZW5ndGggPT09IDEgJiZcbiAgICAgICAgbm9kZS5wYXJhbXNbMF0udHlwZSA9PT0gJ1N0cmluZ0xpdGVyYWwnXG4gICAgICApIHtcbiAgICAgICAgY3NzQ2xhc3MgPSBub2RlLnBhcmFtc1swXS52YWx1ZTtcbiAgICAgIH1cblxuICAgICAgaWYgKFxuICAgICAgICBpc1Njb3BlZENsYXNzKGdldFZhbHVlKG5vZGUucGF0aD8ucGF0aCkpICYmXG4gICAgICAgIG5vZGUucGF0aD8ucGFyYW1zPy5sZW5ndGggPT09IDEgJiZcbiAgICAgICAgbm9kZS5wYXRoPy5wYXJhbXNbMF0udHlwZSA9PT0gJ1N0cmluZ0xpdGVyYWwnXG4gICAgICApIHtcbiAgICAgICAgY3NzQ2xhc3MgPSBub2RlLnBhdGgucGFyYW1zWzBdLnZhbHVlO1xuICAgICAgfVxuXG4gICAgICBpZiAoY3NzQ2xhc3MpIHtcbiAgICAgICAgY29uc3QgdGV4dE5vZGUgPSByZWNhc3QuYnVpbGRlcnMudGV4dChyZW5hbWVDbGFzcyhjc3NDbGFzcywgcG9zdGZpeCkpO1xuICAgICAgICBjb25zdCBwYXJlbnQgPSBzdGFja1tzdGFjay5sZW5ndGggLSAxXTtcblxuICAgICAgICBpZiAocGFyZW50Py50eXBlID09PSAnQXR0ck5vZGUnKSB7XG4gICAgICAgICAgcGFyZW50LnF1b3RlVHlwZSA9ICdcIic7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gdGV4dE5vZGU7XG4gICAgICB9XG4gICAgfSxcblxuICAgIFN1YkV4cHJlc3Npb24obm9kZSkge1xuICAgICAgaWYgKFxuICAgICAgICBpc1Njb3BlZENsYXNzKGdldFZhbHVlKG5vZGUucGF0aCkpICYmXG4gICAgICAgIG5vZGUucGFyYW1zPy5sZW5ndGggPT09IDEgJiZcbiAgICAgICAgbm9kZS5wYXJhbXNbMF0udHlwZSA9PT0gJ1N0cmluZ0xpdGVyYWwnXG4gICAgICApIHtcbiAgICAgICAgY29uc3QgY3NzQ2xhc3MgPSBub2RlLnBhcmFtc1swXS52YWx1ZTtcbiAgICAgICAgY29uc3QgdGV4dE5vZGUgPSByZWNhc3QuYnVpbGRlcnMubGl0ZXJhbChcbiAgICAgICAgICAnU3RyaW5nTGl0ZXJhbCcsXG4gICAgICAgICAgcmVuYW1lQ2xhc3MoY3NzQ2xhc3MsIHBvc3RmaXgpLFxuICAgICAgICApO1xuXG4gICAgICAgIHJldHVybiB0ZXh0Tm9kZTtcbiAgICAgIH1cbiAgICB9LFxuICB9O1xufVxuXG5mdW5jdGlvbiBnZXRWYWx1ZShwYXRoKSB7XG4gIGlmICghcGF0aCkgcmV0dXJuO1xuXG4gIGlmICgndmFsdWUnIGluIHBhdGgpIHtcbiAgICByZXR1cm4gcGF0aC52YWx1ZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZXByZWNhdGVkIGluIGVtYmVyIDUuOStcbiAgICogKHNvIHdlIHVzZSB0aGUgYWJvdmUgZm9yIG5ld2VyIGVtYmVycylcbiAgICovXG4gIHJldHVybiBwYXRoLm9yaWdpbmFsO1xufVxuXG5leHBvcnQgZGVmYXVsdCBmdW5jdGlvbiByZXdyaXRlSGJzKGhicywgY2xhc3NlcywgdGFncywgcG9zdGZpeCkge1xuICBsZXQgYXN0ID0gcmVjYXN0LnBhcnNlKGhicyk7XG5cbiAgcmVjYXN0LnRyYXZlcnNlKGFzdCwgdGVtcGxhdGVQbHVnaW4oeyBjbGFzc2VzLCB0YWdzLCBwb3N0Zml4IH0pKTtcblxuICBsZXQgcmVzdWx0ID0gcmVjYXN0LnByaW50KGFzdCk7XG5cbiAgcmV0dXJuIHJlc3VsdDtcbn1cbiIsIi8qKlxuICogQHR5cGVkZWYge2ltcG9ydCgnQGdsaW1tZXIvc3ludGF4JykuQVNUUGx1Z2lufSBBU1RQbHVnaW5cbiAqIEB0eXBlZGVmIHtpbXBvcnQoJ0BnbGltbWVyL3N5bnRheCcpLkFTVFBsdWdpbkVudmlyb25tZW50fSBBU1RQbHVnaW5FbnZpcm9ubWVudFxuICpcbiAqL1xuXG5pbXBvcnQgcGF0aCBmcm9tICdub2RlOnBhdGgnO1xuaW1wb3J0IHByb2Nlc3MgZnJvbSAnbm9kZTpwcm9jZXNzJztcblxuaW1wb3J0IHsgcmV3cml0ZUNzcyB9IGZyb20gJy4uL2xpYi9jc3MvcmV3cml0ZS5qcyc7XG5pbXBvcnQgeyBnZXRDU1NDb250ZW50SW5mbywgZ2V0Q1NTSW5mbyB9IGZyb20gJy4uL2xpYi9jc3MvdXRpbHMuanMnO1xuaW1wb3J0IHsgZml4RmlsZW5hbWUgfSBmcm9tICcuLi9saWIvcGF0aC90ZW1wbGF0ZS10cmFuc2Zvcm0tcGF0aHMuanMnO1xuaW1wb3J0IHtcbiAgYXBwUGF0aCxcbiAgY3NzUGF0aEZvcixcbiAgZm9yY2VQb3NpeCxcbiAgaGFzaEZyb21Nb2R1bGVQYXRoLFxuICBpc1JlbGV2YW50RmlsZSxcbn0gZnJvbSAnLi4vbGliL3BhdGgvdXRpbHMuanMnO1xuaW1wb3J0IHsgcmVxdWVzdCB9IGZyb20gJy4uL2xpYi9yZXF1ZXN0LmpzJztcbmltcG9ydCB7IHRlbXBsYXRlUGx1Z2luIH0gZnJvbSAnLi4vbGliL3Jld3JpdGVIYnMuanMnO1xuXG5jb25zdCBub29wUGx1Z2luID0ge1xuICBuYW1lOiAnZW1iZXItc2NvcGVkLWNzczpub29wJyxcbiAgdmlzaXRvcjoge30sXG59O1xuXG4vKipcbiAqIEByZXR1cm5zIHtBU1RQbHVnaW59XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVQbHVnaW4oY29uZmlnKSB7XG4gIC8qKlxuICAgKlxuICAgKiBAcGFyYW0ge0FTVFBsdWdpbkVudmlyb25tZW50fSBlbnZcbiAgICovXG4gIHJldHVybiBmdW5jdGlvbiBzY29wZWRDc3MoZW52KSB7XG4gICAgbGV0IGN3ZCA9IHByb2Nlc3MuY3dkKCk7XG5cbiAgICBsZXQgaXNSZWxldmFudCA9IGlzUmVsZXZhbnRGaWxlKGVudi5maWxlbmFtZSwge1xuICAgICAgYWRkaXRpb25hbFJvb3RzOiBjb25maWcuYWRkaXRpb25hbFJvb3RzLFxuICAgICAgY3dkLFxuICAgIH0pO1xuXG4gICAgaWYgKCFpc1JlbGV2YW50KSB7XG4gICAgICByZXR1cm4gbm9vcFBsdWdpbjtcbiAgICB9XG5cbiAgICBsZXQgYWJzb2x1dGVQYXRoID0gZml4RmlsZW5hbWUoZW52LmZpbGVuYW1lKTtcbiAgICBsZXQgbW9kdWxlUGF0aCA9IGFwcFBhdGgoYWJzb2x1dGVQYXRoKTtcbiAgICBsZXQgcG9zdGZpeCA9IGhhc2hGcm9tTW9kdWxlUGF0aChtb2R1bGVQYXRoKTtcblxuICAgIC8qKlxuICAgICAqIFRoZSBsaXN0IG9mIG5ha2VkIHRhZyBzZWxlY3RvcnMgZm91bmQgaW4gdGhlIENTU1xuICAgICAqXG4gICAgICogQHR5cGUge1NldDxzdHJpbmc+fVxuICAgICAqL1xuICAgIGxldCBzY29wZWRUYWdzID0gbmV3IFNldCgpO1xuXG4gICAgLyoqXG4gICAgICogVGhlIGxpc3Qgb2YgY2xhc3NlcyBmb3VuZCBpbiB0aGUgQ1NTXG4gICAgICpcbiAgICAgKiBAdHlwZSB7U2V0PHN0cmluZz59XG4gICAgICovXG4gICAgbGV0IHNjb3BlZENsYXNzZXMgPSBuZXcgU2V0KCk7XG5cbiAgICAvKipcbiAgICAgKiBAcGFyYW0ge3sgdGFnczogU2V0PHN0cmluZz47IGNsYXNzZXM6IFNldDxzdHJpbmc+IH19IGluZm9cbiAgICAgKi9cbiAgICBmdW5jdGlvbiBhZGRJbmZvKGluZm8pIHtcbiAgICAgIGZvciAobGV0IGl0ZW0gb2YgaW5mby50YWdzKSB7XG4gICAgICAgIHNjb3BlZFRhZ3MuYWRkKGl0ZW0pO1xuICAgICAgfVxuXG4gICAgICBmb3IgKGxldCBpdGVtIG9mIGluZm8uY2xhc3Nlcykge1xuICAgICAgICBzY29wZWRDbGFzc2VzLmFkZChpdGVtKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBsZXQgY3NzUGF0aCA9IGNzc1BhdGhGb3IoYWJzb2x1dGVQYXRoKTtcbiAgICBsZXQgaW5mbyA9IGdldENTU0luZm8oY3NzUGF0aCk7XG4gICAgbGV0IGxvY2FsQ3NzUGF0aCA9IGZvcmNlUG9zaXgoY3NzUGF0aC5yZXBsYWNlKGN3ZCArIHBhdGguc2VwLCAnJykpO1xuXG4gICAgLyoqXG4gICAgICogVGhpcyB3aWxsIGJlIGZhbHNleSBpZiB3ZSBkb24ndCBoYXZlIGEgY28tbG9jYXRlZCBDU1MgZmlsZS5cbiAgICAgKiBXZSdsbCBzdGlsbCB3YW50IHRvIGNoZWNrIGZvciBlbWJlZGRlZCA8c3R5bGUgc2NvcGVkPiB0YWdzIHRob3VnaC5cbiAgICAgKi9cbiAgICBpZiAoaW5mbykge1xuICAgICAgYWRkSW5mbyhpbmZvKTtcblxuICAgICAgbGV0IGNzc1JlcXVlc3QgPSByZXF1ZXN0LmNvbG9jYXRlZC5jcmVhdGUoaW5mby5pZCwgcG9zdGZpeCwgbG9jYWxDc3NQYXRoKTtcblxuICAgICAgLyoqXG4gICAgICAgKiBXaXRoIHRoaXMgd2UgZG9uJ3QgbmVlZCBhIEpTIHBsdWdpblxuICAgICAgICovXG4gICAgICBlbnYubWV0YS5qc3V0aWxzLmltcG9ydEZvclNpZGVFZmZlY3QoY3NzUmVxdWVzdCk7XG4gICAgfVxuXG4gICAgbGV0IHZpc2l0b3JzID0gdGVtcGxhdGVQbHVnaW4oe1xuICAgICAgY2xhc3Nlczogc2NvcGVkQ2xhc3NlcyxcbiAgICAgIHRhZ3M6IHNjb3BlZFRhZ3MsXG4gICAgICBwb3N0Zml4LFxuICAgIH0pO1xuXG4gICAgcmV0dXJuIHtcbiAgICAgIG5hbWU6ICdlbWJlci1zY29wZWQtY3NzOnRlbXBsYXRlLXBsdWdpbicsXG4gICAgICB2aXNpdG9yOiB7XG4gICAgICAgIC8vIFN0YWNrIE1hbmFnZXJcbiAgICAgICAgLi4udmlzaXRvcnMsXG5cbiAgICAgICAgLyoqXG4gICAgICAgICAqIFdlIGhhdmUgdG8gZWFnZXJseSBnZXQgdGhlIDxzdHlsZSBzY29wZWQ+IGNvbnRlbnRzLCBzbyB3ZSBjYW4gcHJlLXBhcnNlXG4gICAgICAgICAqIHRoZSB0YWdzIGFuZCBjbGFzc2VzIHRvIHRoZW4gcGFzcyB0byB0aGUgb3RoZXIgdmlzaXRvcnMgc28gdGhhdCB0aGV5IGNhblxuICAgICAgICAgKiBhcHByb3ByaWF0ZWx5IGNoYW5nZSBtYXRjaGluZyBjbGFzc2VzIC8gdGFncy5cbiAgICAgICAgICovXG4gICAgICAgIFRlbXBsYXRlKG5vZGUpIHtcbiAgICAgICAgICAvKipcbiAgICAgICAgICAgKiBXZSBvbmx5IGFsbG93IGEgc2NvcGVkIDxzdHlsZT4gYXQgdGhlIHJvb3RcbiAgICAgICAgICAgKi9cbiAgICAgICAgICBsZXQgc3R5bGVUYWcgPSBub2RlLmJvZHkuZmluZChcbiAgICAgICAgICAgIChuKSA9PiBuLnR5cGUgPT09ICdFbGVtZW50Tm9kZScgJiYgbi50YWcgPT09ICdzdHlsZScsXG4gICAgICAgICAgKTtcblxuICAgICAgICAgIGlmIChoYXNTY29wZWRBdHRyaWJ1dGUoc3R5bGVUYWcpKSB7XG4gICAgICAgICAgICBsZXQgY3NzID0gdGV4dENvbnRlbnQoc3R5bGVUYWcpO1xuICAgICAgICAgICAgbGV0IGxhbmcgPSBnZXRMYW5nQXR0cmlidXRlKHN0eWxlVGFnKTtcbiAgICAgICAgICAgIGxldCBpbmZvID0gZ2V0Q1NTQ29udGVudEluZm8oY3NzLCBsYW5nKTtcblxuICAgICAgICAgICAgYWRkSW5mbyhpbmZvKTtcblxuICAgICAgICAgICAgaWYgKGhhc0lubGluZUF0dHJpYnV0ZVdpdGhvdXRMYW5nKHN0eWxlVGFnKSkge1xuICAgICAgICAgICAgICAvKipcbiAgICAgICAgICAgICAgICogVGhpcyB3aWxsIGJlIGhhbmRsZWQgaW4gRWxlbWVudE5vZGUgdHJhdmVyc2FsXG4gICAgICAgICAgICAgICAqL1xuICAgICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmIChsYW5nKSB7XG4gICAgICAgICAgICAgIC8qKlxuICAgICAgICAgICAgICAgKiBGb3IgPHN0eWxlIHNjb3BlZCBpbmxpbmUgbGFuZz1cIi4uLlwiPiB3ZSBjYW5ub3QgcHJlcHJvY2VzcyBhdCBCYWJlbC10aW1lXG4gICAgICAgICAgICAgICAqIChwcmVwcm9jZXNzaW5nIGlzIGFzeW5jIGFuZCByZXF1aXJlcyBWaXRlJ3MgUmVzb2x2ZWRDb25maWcpLlxuICAgICAgICAgICAgICAgKiBSZW1vdmUgdGhlIHRhZyBhbmQgaW5qZWN0IHZpYSB2aXJ0dWFsIENTUyBtb2R1bGUgYW5kIHdhcm4gdXNlci5cbiAgICAgICAgICAgICAgICovXG4gICAgICAgICAgICAgIGNvbnNvbGUud2FybihcbiAgICAgICAgICAgICAgICBgW2VtYmVyLXNjb3BlZC1jc3NdIDxzdHlsZSBzY29wZWQgaW5saW5lIGxhbmc9XCIke2xhbmd9XCI+IGlzIG5vdCBzdXBwb3J0ZWQgYCArXG4gICAgICAgICAgICAgICAgICBgKHByZXByb2Nlc3NpbmcgaXMgYXN5bmMgYW5kIGNhbm5vdCBydW4gYXQgQmFiZWwtdGltZSkuIGAgK1xuICAgICAgICAgICAgICAgICAgYERvd25ncmFkaW5nIHRvIG5vbi1pbmxpbmU6IHRoZSBzdHlsZSB0YWcgd2lsbCBiZSByZW1vdmVkIGFuZCBpbmplY3RlZCBhcyBhIHZpcnR1YWwgQ1NTIG1vZHVsZS5gLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBsZXQgY3NzUmVxdWVzdCA9IHJlcXVlc3QuaW5saW5lLmNyZWF0ZShpbmZvLmlkLCBwb3N0Zml4LCBjc3MsIGxhbmcpO1xuXG4gICAgICAgICAgICBlbnYubWV0YS5qc3V0aWxzLmltcG9ydEZvclNpZGVFZmZlY3QoY3NzUmVxdWVzdCk7XG4gICAgICAgICAgfVxuICAgICAgICB9LFxuXG4gICAgICAgIC8vIFZpc2l0b3JzIGJyb2tlbiBvdXQgbGlrZSB0aGlzIHNvIHdlIGNhbiBjb25kaXRpb25hbGx5XG4gICAgICAgIC8vIGRlYnVnIGJhc2VkIG9uIGZpbGUgcGF0aC5cbiAgICAgICAgQXR0ck5vZGUoLi4uYXJncykge1xuICAgICAgICAgIHJldHVybiB2aXNpdG9ycy5BdHRyTm9kZSguLi5hcmdzKTtcbiAgICAgICAgfSxcbiAgICAgICAgRWxlbWVudE5vZGUobm9kZSwgd2Fsa2VyKSB7XG4gICAgICAgICAgLy8gY2xhc3MgYXR0cmlidXRlIGhhbmRsaW5nXG4gICAgICAgICAgdmlzaXRvcnMuRWxlbWVudE5vZGUobm9kZSwgd2Fsa2VyKTtcblxuICAgICAgICAgIGlmIChoYXNTY29wZWRBdHRyaWJ1dGUobm9kZSkpIHtcbiAgICAgICAgICAgIGlmICh3YWxrZXIucGFyZW50Py5ub2RlLnR5cGUgIT09ICdUZW1wbGF0ZScpIHtcbiAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAgICAgICAgICc8c3R5bGUgc2NvcGVkPiB0YWdzIG11c3QgYmUgYXQgdGhlIHJvb3Qgb2YgdGhlIHRlbXBsYXRlLCB0aGV5IGNhbm5vdCBiZSBuZXN0ZWQnLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBpZiAoaGFzSW5saW5lQXR0cmlidXRlV2l0aG91dExhbmcobm9kZSkpIHtcbiAgICAgICAgICAgICAgbGV0IHRleHQgPSB0ZXh0Q29udGVudChub2RlKTtcbiAgICAgICAgICAgICAgbGV0IHNjb3BlZFRleHQgPSByZXdyaXRlQ3NzKFxuICAgICAgICAgICAgICAgIHRleHQsXG4gICAgICAgICAgICAgICAgcG9zdGZpeCxcbiAgICAgICAgICAgICAgICBsb2NhbENzc1BhdGgsXG4gICAgICAgICAgICAgICAgY29uZmlnLmxheWVyTmFtZSxcbiAgICAgICAgICAgICAgKTtcblxuICAgICAgICAgICAgICAvKipcbiAgICAgICAgICAgICAgICogVHJhdmVyc2UgdGhpcyBhbmQgYWxsb3cgaW50ZXJwb2xhdGlvblxuICAgICAgICAgICAgICAgKi9cbiAgICAgICAgICAgICAgbm9kZS5jaGlsZHJlbiA9IFtlbnYuc3ludGF4LmJ1aWxkZXJzLnRleHQoc2NvcGVkVGV4dCldO1xuXG4gICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gUmV0dXJuaW5nIG51bGwgcmVtb3ZlcyB0aGUgbm9kZVxuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgaWYgKGhhc0lubGluZUF0dHJpYnV0ZVdpdGhvdXRMYW5nKG5vZGUpKSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICAgIGA8c3R5bGUgaW5saW5lPiBpcyBub3QgdmFsaWQuIFBsZWFzZSBhZGQgdGhlIHNjb3BlZCBhdHRyaWJ1dGU6IDxzdHlsZSBzY29wZWQgaW5saW5lPmAsXG4gICAgICAgICAgICApO1xuICAgICAgICAgIH1cbiAgICAgICAgfSxcbiAgICAgICAgTXVzdGFjaGVTdGF0ZW1lbnQoLi4uYXJncykge1xuICAgICAgICAgIHJldHVybiB2aXNpdG9ycy5NdXN0YWNoZVN0YXRlbWVudCguLi5hcmdzKTtcbiAgICAgICAgfSxcbiAgICAgICAgU3ViRXhwcmVzc2lvbiguLi5hcmdzKSB7XG4gICAgICAgICAgcmV0dXJuIHZpc2l0b3JzLlN1YkV4cHJlc3Npb24oLi4uYXJncyk7XG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgIH07XG4gIH07XG59XG5cbi8qKlxuICogVGhhbmtzLCBDYXJkU3RhY2sgYW5kIEBlZjQgZm9yIHRoaXMgY29kZS5cbiAqL1xuY29uc3QgU0NPUEVEX0FUVFJJQlVURV9OQU1FID0gJ3Njb3BlZCc7XG5jb25zdCBJTkxJTkVfQVRUUklCVVRFX05BTUUgPSAnaW5saW5lJztcbmNvbnN0IExBTkdfQVRUUklCVVRFX05BTUUgPSAnbGFuZyc7XG5cbmZ1bmN0aW9uIGhhc1Njb3BlZEF0dHJpYnV0ZShub2RlKSB7XG4gIGlmICghbm9kZSkgcmV0dXJuO1xuICBpZiAobm9kZS50YWcgIT09ICdzdHlsZScpIHJldHVybjtcbiAgaWYgKG5vZGUudHlwZSAhPT0gJ0VsZW1lbnROb2RlJykgcmV0dXJuO1xuXG4gIHJldHVybiBub2RlLmF0dHJpYnV0ZXMuc29tZShcbiAgICAoYXR0cmlidXRlKSA9PiBhdHRyaWJ1dGUubmFtZSA9PT0gU0NPUEVEX0FUVFJJQlVURV9OQU1FLFxuICApO1xufVxuXG5mdW5jdGlvbiBoYXNJbmxpbmVBdHRyaWJ1dGVXaXRob3V0TGFuZyhub2RlKSB7XG4gIGlmICghbm9kZSkgcmV0dXJuO1xuICBpZiAobm9kZS50YWcgIT09ICdzdHlsZScpIHJldHVybjtcbiAgaWYgKG5vZGUudHlwZSAhPT0gJ0VsZW1lbnROb2RlJykgcmV0dXJuO1xuXG4gIGlmIChnZXRMYW5nQXR0cmlidXRlKG5vZGUpKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgcmV0dXJuIG5vZGUuYXR0cmlidXRlcy5zb21lKFxuICAgIChhdHRyaWJ1dGUpID0+IGF0dHJpYnV0ZS5uYW1lID09PSBJTkxJTkVfQVRUUklCVVRFX05BTUUsXG4gICk7XG59XG5cbi8qKlxuICogUmV0dXJucyB0aGUgdmFsdWUgb2YgdGhlIGBsYW5nYCBhdHRyaWJ1dGUgb24gYSBgPHN0eWxlPmAgbm9kZSwgb3IgbnVsbCBpZiBhYnNlbnQuXG4gKlxuICogQHBhcmFtIHtvYmplY3R9IG5vZGVcbiAqIEByZXR1cm5zIHtzdHJpbmcgfCBudWxsfVxuICovXG5mdW5jdGlvbiBnZXRMYW5nQXR0cmlidXRlKG5vZGUpIHtcbiAgaWYgKCFub2RlKSByZXR1cm4gbnVsbDtcbiAgaWYgKG5vZGUudGFnICE9PSAnc3R5bGUnKSByZXR1cm4gbnVsbDtcbiAgaWYgKG5vZGUudHlwZSAhPT0gJ0VsZW1lbnROb2RlJykgcmV0dXJuIG51bGw7XG5cbiAgY29uc3QgYXR0ciA9IG5vZGUuYXR0cmlidXRlcy5maW5kKFxuICAgIChhdHRyaWJ1dGUpID0+IGF0dHJpYnV0ZS5uYW1lID09PSBMQU5HX0FUVFJJQlVURV9OQU1FLFxuICApO1xuXG4gIGlmICghYXR0cikgcmV0dXJuIG51bGw7XG5cbiAgLy8gVGhlIGF0dHJpYnV0ZSB2YWx1ZSBpcyBhIFRleHROb2RlIGNoaWxkIG9mIHRoZSBBdHRyTm9kZSdzIHZhbHVlXG4gIGNvbnN0IHZhbHVlID0gYXR0ci52YWx1ZTtcblxuICBpZiAodmFsdWU/LnR5cGUgPT09ICdUZXh0Tm9kZScpIHJldHVybiB2YWx1ZS5jaGFycyB8fCBudWxsO1xuXG4gIHJldHVybiBudWxsO1xufVxuXG5mdW5jdGlvbiB0ZXh0Q29udGVudChub2RlKSB7XG4gIGxldCB0ZXh0Q2hpbGRyZW4gPSBub2RlLmNoaWxkcmVuLmZpbHRlcigoYykgPT4gYy50eXBlID09PSAnVGV4dE5vZGUnKTtcblxuICByZXR1cm4gdGV4dENoaWxkcmVuLm1hcCgoYykgPT4gYy5jaGFycykuam9pbignJyk7XG59XG4iLCJpbXBvcnQgKiBhcyBiYWJlbCBmcm9tICcuLi9iYWJlbC1wbHVnaW4uanMnO1xuaW1wb3J0IHsgdW5wbHVnaW4gfSBmcm9tICcuLi9zY29wZWQtY3NzLXVucGx1Z2luLmpzJztcbmltcG9ydCAqIGFzIHRlbXBsYXRlIGZyb20gJy4uL3RlbXBsYXRlLXBsdWdpbi5qcyc7XG5cbmV4cG9ydCBjb25zdCBzY29wZWRDU1MgPSB7XG4gIHZpdGU6IHVucGx1Z2luLnZpdGUsXG4gIHJvbGx1cDogdW5wbHVnaW4ucm9sbHVwLFxuICBiYWJlbDogYmFiZWwuc2NvcGVkQ1NTLFxuICB0ZW1wbGF0ZTogdGVtcGxhdGUuY3JlYXRlUGx1Z2luLFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQU1BLE1BQWEsbUJBQW1CO0NBQzlCLGNBQWMsYUFBSyxLQUFLLDRCQUE0QjtDQUNwRCxhQUFhLGFBQUssS0FBSyxjQUFjO0NBQ3JDLGVBQWUsYUFBSyxLQUFLLGVBQWU7Q0FDeEMsY0FBYyxhQUFLLEtBQUssY0FBYztDQUN0QyxXQUFXLGFBQUssS0FBSyxXQUFXO0NBQ2hDLFFBQVEsYUFBSyxLQUFLLFVBQVU7Q0FDNUIsS0FBSyxhQUFLLEtBQUssUUFBUTtDQUN2QixLQUFLLGFBQUssS0FBSyxRQUFRO0NBQ3hCO0FBRUQsTUFBYSxXQUFXLEVBQ3RCLFNBQVMsYUFBSyxLQUFLLHFCQUFxQixFQUN6Qzs7Ozs7Ozs7Ozs7O0FDV0QsU0FBUyxRQUFRLEdBQUcsR0FBRztDQUNyQixJQUFJLE9BQU8sSUFBSSxVQUFXLElBQUk7QUFHOUIsU0FGVyxLQUFLLE9BQU8sS0FBSyxPQUFPLE9BQU8sT0FFM0IsS0FBTyxNQUFNOzs7Ozs7Ozs7QUFVOUIsU0FBUyxjQUFjLEtBQUssS0FBSztBQUMvQixRQUFRLE9BQU8sTUFBUSxRQUFTLEtBQUs7Ozs7Ozs7Ozs7Ozs7QUFjdkMsU0FBUyxPQUFPLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHO0FBQ2hDLFFBQU8sUUFBUSxjQUFjLFFBQVEsUUFBUSxHQUFHLEVBQUUsRUFBRSxRQUFRLEdBQUcsRUFBRSxDQUFDLEVBQUUsRUFBRSxFQUFFLEVBQUU7Ozs7Ozs7Ozs7Ozs7O0FBZTVFLFNBQVMsTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHO0FBQ2xDLFFBQU8sT0FBUSxJQUFJLElBQU0sQ0FBQyxJQUFJLEdBQUksR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFOzs7Ozs7Ozs7Ozs7OztBQWVsRCxTQUFTLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRztBQUNsQyxRQUFPLE9BQVEsSUFBSSxJQUFNLElBQUksQ0FBQyxHQUFJLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRTs7Ozs7Ozs7Ozs7Ozs7QUFlbEQsU0FBUyxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUc7QUFDbEMsUUFBTyxPQUFPLElBQUksSUFBSSxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRTs7Ozs7Ozs7Ozs7Ozs7QUFlekMsU0FBUyxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUc7QUFDbEMsUUFBTyxPQUFPLEtBQUssSUFBSSxDQUFDLElBQUksR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFOzs7Ozs7Ozs7QUFVNUMsU0FBUyxRQUFRLEdBQUcsS0FBSztBQUV2QixHQUFFLE9BQU8sTUFBTSxPQUFRLE1BQU07QUFDN0IsSUFBSyxNQUFNLE9BQVEsS0FBTSxLQUFLLE1BQU07Q0FFcEMsSUFBSTtDQUNKLElBQUk7Q0FDSixJQUFJO0NBQ0osSUFBSTtDQUNKLElBQUk7Q0FDSixJQUFJLElBQUk7Q0FDUixJQUFJLElBQUk7Q0FDUixJQUFJLElBQUk7Q0FDUixJQUFJLElBQUk7QUFFUixNQUFLLElBQUksR0FBRyxJQUFJLEVBQUUsUUFBUSxLQUFLLElBQUk7QUFDakMsU0FBTztBQUNQLFNBQU87QUFDUCxTQUFPO0FBQ1AsU0FBTztBQUVQLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxHQUFHLFdBQVc7QUFDMUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksVUFBVTtBQUM5QyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFlBQVk7QUFDaEQsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxXQUFXO0FBQzlDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksV0FBVztBQUMvQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFlBQVk7QUFDaEQsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxVQUFVO0FBQzlDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLEdBQUcsV0FBVztBQUM5QyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFlBQVk7QUFDaEQsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssSUFBSSxPQUFPO0FBQzVDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksWUFBWTtBQUNqRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxHQUFHLFdBQVc7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssSUFBSSxVQUFVO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksWUFBWTtBQUNqRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFdBQVc7QUFFaEQsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxXQUFXO0FBQzlDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLEdBQUcsWUFBWTtBQUMvQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFVBQVU7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksV0FBVztBQUMzQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxHQUFHLFdBQVc7QUFDOUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssR0FBRyxTQUFTO0FBQzdDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksV0FBVztBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFdBQVc7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxVQUFVO0FBQzdDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLEdBQUcsWUFBWTtBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFdBQVc7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLEdBQUcsWUFBWTtBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxHQUFHLFVBQVU7QUFDN0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksWUFBWTtBQUVqRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxHQUFHLFFBQVE7QUFDM0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxZQUFZO0FBQ2hELE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksV0FBVztBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFVBQVU7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxZQUFZO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksV0FBVztBQUMvQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFdBQVc7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssSUFBSSxZQUFZO0FBQ2pELE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLEdBQUcsVUFBVTtBQUM5QyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxXQUFXO0FBQzNDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksV0FBVztBQUMvQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFNBQVM7QUFDN0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxXQUFXO0FBQzlDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksV0FBVztBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFVBQVU7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBRS9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxHQUFHLFdBQVc7QUFDMUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksWUFBWTtBQUNqRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFVBQVU7QUFDOUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssR0FBRyxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksWUFBWTtBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFNBQVM7QUFDOUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxZQUFZO0FBQ2hELE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLEdBQUcsV0FBVztBQUM5QyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFVBQVU7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxZQUFZO0FBQ2hELE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksV0FBVztBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUks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