ember-scoped-css 2.0.4 → 2.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/all-CFsaG5pM.cjs +1416 -0
- package/dist/cjs/all.cjs +1 -1
- package/dist/cjs/babel.cjs +1 -1
- package/dist/cjs/rollup.cjs +1 -1
- package/dist/cjs/vite.cjs +1 -1
- package/package.json +4 -3
- package/src/build/template-plugin.js +53 -9
- package/src/build/template-plugin.test.ts +140 -1
- package/src/build/unplugin-colocated.js +56 -6
- package/src/build/unplugin-inline.js +53 -13
- package/src/lib/css/rewrite.js +18 -16
- package/src/lib/css/rewrite.test.ts +68 -52
- package/src/lib/css/utils.js +107 -3
- package/src/lib/path/const.js +1 -0
- package/src/lib/path/utils.isRelevantFile.test.ts +7 -0
- package/src/lib/path/utils.js +1 -0
- package/src/lib/request.js +10 -2
- package/dist/cjs/all-DIcjm1uz.cjs +0 -1319
|
@@ -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)(config.root).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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWxsLUNGc2FHNXBNLmNqcyIsIm5hbWVzIjpbImhhc2giLCJoYXNoIiwiaGFzaEZyb21Nb2R1bGVQYXRoIiwiaGFzaFBvc2l4TW9kdWxlUGF0aCIsInBhdGgiLCJmc1N5bmMiLCJzY29wZWRDU1MiLCJwYXRoIiwic2Nzc1N5bnRheCIsIlNFUCIsInBhcnNlciIsInJlcXVlc3QiLCJoYXNoIiwicGF0aCIsIk1FVEEiLCJDV0QiLCJwYXRoIiwiQ1dEIiwicGF0aCIsInVucGx1Z2luIiwicGF0aCIsIm5vZGUiLCJyZWNhc3QiLCJwYXRoIiwicHJvY2VzcyIsImluZm8iLCJwYXRoIiwidW5wbHVnaW4iLCJiYWJlbC5zY29wZWRDU1MiLCJ0ZW1wbGF0ZS5jcmVhdGVQbHVnaW4iXSwic291cmNlcyI6WyIuLi8uLi9zcmMvbGliL3BhdGgvY29uc3QuanMiLCIuLi8uLi9zcmMvbGliL3BhdGgvbWQ1LmpzIiwiLi4vLi4vc3JjL2xpYi9wYXRoL2hhc2gtZnJvbS1tb2R1bGUtcGF0aC5qcyIsIi4uLy4uL3NyYy9saWIvcGF0aC91dGlscy5qcyIsIi4uLy4uL3NyYy9saWIvcmVuYW1lQ2xhc3MuanMiLCIuLi8uLi9zcmMvYnVpbGQvYmFiZWwtcGx1Z2luLmpzIiwiLi4vLi4vc3JjL2xpYi9jc3MvdXRpbHMuanMiLCIuLi8uLi9zcmMvbGliL2Nzcy9yZXdyaXRlLmpzIiwiLi4vLi4vc3JjL2xpYi9yZXF1ZXN0LmpzIiwiLi4vLi4vc3JjL2J1aWxkL3VucGx1Z2luLWNvbG9jYXRlZC5qcyIsIi4uLy4uL3NyYy9idWlsZC91bnBsdWdpbi1pbmxpbmUuanMiLCIuLi8uLi9zcmMvYnVpbGQvc2NvcGVkLWNzcy11bnBsdWdpbi5qcyIsIi4uLy4uL3NyYy9saWIvcGF0aC90ZW1wbGF0ZS10cmFuc2Zvcm0tcGF0aHMuanMiLCIuLi8uLi9zcmMvbGliL3Jld3JpdGVIYnMuanMiLCIuLi8uLi9zcmMvYnVpbGQvdGVtcGxhdGUtcGx1Z2luLmpzIiwiLi4vLi4vc3JjL2J1aWxkL3B1YmxpYy1leHBvcnRzL2FsbC5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgcGF0aCBmcm9tICdwYXRoJztcblxuLyoqXG4gKiBKb2luIHdpbGwgY29udmVydCB0byB3aGF0ZXZlciBpcyBhcHByb3ByaWF0ZSBmcm8gdGhlIGN1cnJlbnQgcGxhdGZvcm1cbiAqL1xuXG5leHBvcnQgY29uc3QgbGVhZGluZ1NsYXNoUGF0aCA9IHtcbiAgZW1icm9pZGVyRGlyOiBwYXRoLmpvaW4oJy9ub2RlX21vZHVsZXMvLmVtYnJvaWRlci8nKSxcbiAgYXRFbWJyb2lkZXI6IHBhdGguam9pbignL0BlbWJyb2lkZXInKSxcbiAgY29tcG9uZW50c0RpcjogcGF0aC5qb2luKCcvY29tcG9uZW50cy8nKSxcbiAgdGVtcGxhdGVzRGlyOiBwYXRoLmpvaW4oJy90ZW1wbGF0ZXMvJyksXG4gIHJvdXRlc0RpcjogcGF0aC5qb2luKCcvcm91dGVzLycpLFxuICB0ZXN0ZW06IHBhdGguam9pbignL3Rlc3RlbScpLFxuICBzcmM6IHBhdGguam9pbignL3NyYy8nKSxcbiAgYXBwOiBwYXRoLmpvaW4oJy9hcHAvJyksXG59O1xuXG5leHBvcnQgY29uc3QgYmFyZVBhdGggPSB7XG4gIHBucG1EaXI6IHBhdGguam9pbignbm9kZV9tb2R1bGVzLy5wbnBtJyksXG59O1xuIiwiLypcbiAqIEphdmFTY3JpcHQgTUQ1XG4gKiBodHRwczovL2dpdGh1Yi5jb20vYmx1ZWltcC9KYXZhU2NyaXB0LU1ENVxuICpcbiAqIENvcHlyaWdodCAyMDExLCBTZWJhc3RpYW4gVHNjaGFuXG4gKiBodHRwczovL2JsdWVpbXAubmV0XG4gKlxuICogTGljZW5zZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlOlxuICogaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVRcbiAqXG4gKiBCYXNlZCBvblxuICogQSBKYXZhU2NyaXB0IGltcGxlbWVudGF0aW9uIG9mIHRoZSBSU0EgRGF0YSBTZWN1cml0eSwgSW5jLiBNRDUgTWVzc2FnZVxuICogRGlnZXN0IEFsZ29yaXRobSwgYXMgZGVmaW5lZCBpbiBSRkMgMTMyMS5cbiAqIFZlcnNpb24gMi4yIENvcHlyaWdodCAoQykgUGF1bCBKb2huc3RvbiAxOTk5IC0gMjAwOVxuICogT3RoZXIgY29udHJpYnV0b3JzOiBHcmVnIEhvbHQsIEFuZHJldyBLZXBlcnQsIFlkbmFyLCBMb3N0aW5ldFxuICogRGlzdHJpYnV0ZWQgdW5kZXIgdGhlIEJTRCBMaWNlbnNlXG4gKiBTZWUgaHR0cDovL3BhamhvbWUub3JnLnVrL2NyeXB0L21kNSBmb3IgbW9yZSBpbmZvLlxuICpcbiAqIE1vZGlmaWNhdGlvbnM6XG4gKiAtIHJlbW92ZWQgSUlGRSwgZXhwb3J0ZWQgbWQ1ICh0aGlzIGZpbGUgaXMgbm93IEVTTSlcbiAqL1xuXG4vKipcbiAqIEFkZCBpbnRlZ2Vycywgd3JhcHBpbmcgYXQgMl4zMi5cbiAqIFRoaXMgdXNlcyAxNi1iaXQgb3BlcmF0aW9ucyBpbnRlcm5hbGx5IHRvIHdvcmsgYXJvdW5kIGJ1Z3MgaW4gaW50ZXJwcmV0ZXJzLlxuICpcbiAqIEBwYXJhbSB7bnVtYmVyfSB4IEZpcnN0IGludGVnZXJcbiAqIEBwYXJhbSB7bnVtYmVyfSB5IFNlY29uZCBpbnRlZ2VyXG4gKiBAcmV0dXJucyB7bnVtYmVyfSBTdW1cbiAqL1xuZnVuY3Rpb24gc2FmZUFkZCh4LCB5KSB7XG4gIHZhciBsc3cgPSAoeCAmIDB4ZmZmZikgKyAoeSAmIDB4ZmZmZik7XG4gIHZhciBtc3cgPSAoeCA+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+LyR7bWV0YS5maWxlTmFtZX1gLFxuICAgICAgICAgIG9wdGlvbnMubGF5ZXJOYW1lLFxuICAgICAgICApO1xuXG4gICAgICAgIHJldHVybiBjc3M7XG4gICAgICB9XG4gICAgfSxcbiAgICB2aXRlOiB7XG4gICAgICBhc3luYyBjb25maWdSZXNvbHZlZChjb25maWcpIHtcbiAgICAgICAgdml0ZUNvbmZpZyA9IGNvbmZpZztcblxuICAgICAgICAvLyBSZXNvbHZlIFZpdGUncyBwcmVwcm9jZXNzQ1NTIGZyb20gdGhlIGFwcCByb290IHRvIGVuc3VyZSB3ZSBmaW5kXG4gICAgICAgIC8vIHRoZSBjb3JyZWN0IFZpdGUgaW5zdGFsbGF0aW9uIChub3QgYSBzdGFsZSBvciBtaXNzaW5nIG9uZSkuXG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgY29uc3QgcmVxdWlyZSA9IGNyZWF0ZVJlcXVpcmUoY29uZmlnLnJvb3QpO1xuICAgICAgICAgIGNvbnN0IHZpdGVQYXRoID0gcmVxdWlyZS5yZXNvbHZlKCd2aXRlJyk7XG4gICAgICAgICAgY29uc3Qgdml0ZU1vZHVsZSA9IGF3YWl0IGltcG9ydCh2aXRlUGF0aCk7XG5cbiAgICAgICAgICBwcmVwcm9jZXNzQ1NTID0gdml0ZU1vZHVsZS5wcmVwcm9jZXNzQ1NTO1xuICAgICAgICB9IGNhdGNoIHtcbiAgICAgICAgICAvLyBWaXRlIG1heSBub3QgYmUgcmVzb2x2YWJsZSBmcm9tIHRoZSBjb25maWcgcm9vdCBpbiBzb21lIHNldHVwcztcbiAgICAgICAgICAvLyBsYW5nPSBzdXBwb3J0IHdpbGwgdGhyb3cgYSBjbGVhciBlcnJvciBhdCBsb2FkIHRpbWUgaWYgdXNlZC5cbiAgICAgICAgfVxuICAgICAgfSxcbiAgICB9LFxuICB9O1xufVxuIiwiaW1wb3J0IHsgY3JlYXRlVW5wbHVnaW4gfSBmcm9tICd1bnBsdWdpbic7XG5cbmltcG9ydCB7IGNvbG9jYXRlZCB9IGZyb20gJy4vdW5wbHVnaW4tY29sb2NhdGVkLmpzJztcbmltcG9ydCB7IGlubGluZSB9IGZyb20gJy4vdW5wbHVnaW4taW5saW5lLmpzJztcblxuLyoqXG4gKiBUaGUgcGx1Z2luIHRoYXQgaGFuZGxlcyBDU1MgcmVxdWVzdHMgZm9yIGA8c3R5bGU+YCBlbGVtZW50cyBhbmQgdHJhbnNmb3Jtc1xuICogZm9yIGV4aXN0aW5nIGZpbGVzXG4gKlxuICogdml0ZTogQ1NTIGZpbGVzIGFyZSByZXNvbHZlZCBieSB2aXRlLiBXZSB1c2UgdGhlaXIgcmVzb2x2ZXIgdG8gYWxzbyBnZXRcbiAqICAgICAgIEhNUi4gVGhhdCBpcywgZm9yIGFsbCBub24tcGh5c2ljYWwgQ1NTIGZpbGVzLCB3ZSBleHRlbmQgdml0ZSBieSBvdXJcbiAqICAgICAgIHJlc29sdmVyIGFuZCBhbHNvIGNhbiBlbnJpY2ggbWV0YWRhdGEgdG8gaXQgKGZvciBiZXR0ZXIgZGVidWdnaW5nKVxuICovXG5leHBvcnQgY29uc3QgdW5wbHVnaW4gPSBjcmVhdGVVbnBsdWdpbigob3B0aW9ucyA9IHt9KSA9PiB7XG4gIHJldHVybiBbY29sb2NhdGVkKG9wdGlvbnMpLCBpbmxpbmUob3B0aW9ucyldO1xufSk7XG4iLCJpbXBvcnQgcGF0aCBmcm9tICdub2RlOnBhdGgnO1xuXG5pbXBvcnQgeyBsZWFkaW5nU2xhc2hQYXRoIH0gZnJvbSAnLi9jb25zdC5qcyc7XG5pbXBvcnQgeyBmaW5kV29ya3NwYWNlUGF0aCB9IGZyb20gJy4vdXRpbHMuanMnO1xuXG4vKipcbiAqIHRlbXBsYXRlIHBsdWdpbnMgZG8gbm90IGhhbmQgdXMgdGhlIGNvcnJlY3QgZmlsZSBwYXRoLlxuICogYWRkaXRpb25hbGx5LCB3ZSBtYXkgbm90IGJlIGFibGUgdG8gcmVseSBvbiB0aGlzIGRhdGEgaW4gdGhlIGZ1dHVyZSxcbiAqIHNvIHRoaXMgZnVuY3Rpb25zIGFjdHMgYXMgYSBtZWFucyBvZiBub3JtYWxpemluZyBfd2hhdGV2ZXJfIHdlJ3JlIGdpdmVuXG4gKiBpbiB0aGUgZnV0dXJlLlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlbmFtZVxuICogQHJldHVybnMge3N0cmluZ30gdGhlIGFic29sdXRlIHBhdGggdG8gdGhlIGZpbGVcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZpeEZpbGVuYW1lKGZpbGVuYW1lKSB7XG4gIGxldCBmaWxlTmFtZSA9IGZpbGVuYW1lO1xuICBsZXQgd29ya3NwYWNlID0gZmluZFdvcmtzcGFjZVBhdGgoZmlsZU5hbWUpO1xuXG4gIC8qKlxuICAgKiBlbWJlci1zb3VyY2UgNS44OlxuICAgKiAtIHRoZSBmaWxlbmFtZSBsb29rcyBsaWtlIGFuIGFic29sdXRlIHBhdGgsIGJ1dCBzd2FwcGVkIG91dCB0aGUgJ2FwcCcgcGFydCBvZiB0aGUgcGF0aFxuICAgKiAgIHdpdGggdGhlIG1vZHVsZSBuYW1lLCBzbyB0aGUgZmlsZSBwYXRocyBuZXZlciBleGlzdCBvbiBkaXNrXG4gICAqXG4gICAqIC0gaW4gdml0ZSBhcHBzOlxuICAgKiAgIHRoZSAnYXBwJyBwYXJ0IF9tYXlfIGJlIGBzcmNgLCBzbyB3ZSBhbHNvIG5lZWQgdG8gZW5zdXJlIHRoYXQgYHNyY2AgaXMgZXhjbHVkZWQgYXMgd2VsbFxuICAgKi9cbiAgbGV0IGhhc0FwcERpciA9IGZpbGVOYW1lLmluY2x1ZGVzKHBhdGguam9pbih3b3Jrc3BhY2UsICdhcHAnKSk7XG4gIGxldCBoYXNTcmNEaXIgPSBmaWxlTmFtZS5pbmNsdWRlcyhwYXRoLmpvaW4od29ya3NwYWNlLCAnc3JjJykpO1xuXG4gIGlmIChcbiAgICAhKGhhc0FwcERpciB8fCBoYXNTcmNEaXIpICYmXG4gICAgIWZpbGVOYW1lLmluY2x1ZGVzKGxlYWRpbmdTbGFzaFBhdGguZW1icm9pZGVyRGlyKVxuICApIHtcbiAgICBsZXQgbWF5YmVNb2R1bGUgPSBmaWxlTmFtZS5yZXBsYWNlKHdvcmtzcGFjZSwgJycpO1xuICAgIGxldCBbbWF5YmVTY29wZSwgLi4ucmVzdF0gPSBtYXliZU1vZHVsZS5zcGxpdChwYXRoLnNlcCkuZmlsdGVyKEJvb2xlYW4pO1xuICAgIGxldCBwYXJ0cyA9IHJlc3Q7XG5cbiAgICBpZiAobWF5YmVTY29wZS5zdGFydHNXaXRoKCdAJykpIHtcbiAgICAgIGxldCBbLCAuLi5yZXN0ZXJdID0gcmVzdDtcblxuICAgICAgcGFydHMgPSByZXN0ZXI7XG4gICAgfVxuXG4gICAgbGV0IHJlbGF0aXZlID0gcGF0aC5qb2luKC4uLnBhcnRzKTtcblxuICAgIC8qKlxuICAgICAqIFdlIGRvbid0IGFjdHVhbGx5IGtub3cgaWYgdGhpcyBmaWxlIGlzIGFuIGFwcC5cbiAgICAgKiBpdCBjb3VsZCBiZSBhbiBhZGRvbiAodjEgb3IgdjIpXG4gICAgICpcbiAgICAgKiBTbyBoZXJlIHdlIGxvZyB0byBzZWUgaWYgd2UgaGF2ZSB1bmhhbmRsZWQgc2l0dWF0aW9ucy5cbiAgICAgKi9cbiAgICBsZXQgY2FuZGlkYXRlUGF0aCA9IHBhdGguam9pbih3b3Jrc3BhY2UsICdhcHAnLCByZWxhdGl2ZSk7XG5cbiAgICByZXR1cm4gY2FuZGlkYXRlUGF0aDtcbiAgfVxuXG4gIC8vIFRPRE86IHdoeSBhcmUgd2UgcGFzc2VkIGZpbGVzIHRvIG90aGVyIHByb2plY3RzP1xuICBpZiAoIWZpbGVOYW1lLmluY2x1ZGVzKHdvcmtzcGFjZSkpIHtcbiAgICByZXR1cm4gZmlsZU5hbWU7XG4gIH1cblxuICAvLyBGYWxsYmFjayB0byB3aGF0IHRoZSBwbHVnaW4gc3lzdGVtIGdpdmVzIHVzLlxuICAvLyBUaGlzIG1heSBiZSB3cm9uZywgYW5kIGlmIHdyb25nLCByZXZlYWxzXG4gIC8vIHVuaGFuZGxlZCBzY2VuYXJpb3Mgd2l0aCB0aGUgZmlsZSBuYW1lcyBpbiB0aGUgcGx1Z2luIGluZnJhXG4gIHJldHVybiBmaWxlTmFtZTtcbn1cbiIsImltcG9ydCAqIGFzIHJlY2FzdCBmcm9tICdlbWJlci10ZW1wbGF0ZS1yZWNhc3QnO1xuXG5pbXBvcnQgeyByZW5hbWVDbGFzcyB9IGZyb20gJy4vcmVuYW1lQ2xhc3MuanMnO1xuXG5leHBvcnQgZnVuY3Rpb24gdGVtcGxhdGVQbHVnaW4oeyBjbGFzc2VzLCB0YWdzLCBwb3N0Zml4IH0pIHtcbiAgbGV0IHN0YWNrID0gW107XG4gIC8vIHNjb3BlZC1jbGFzcyBpcyBhIGdsb2JhbCB3ZSBhbGxvdyBpbiBoYnNcbiAgLy8gc2NvcGVkQ2xhc3MgaXMgaW1wb3J0YWJsZSwgYW5kIHdlJ2xsIGVycm9yIGlmIHNvbWVvbmUgdHJpZXMgdG8gcmVuYW1lIGl0XG4gIGxldCBzY29wZWRDbGFzc0NhbmRpZGF0ZXMgPSBbJ3Njb3BlZC1jbGFzcycsICdzY29wZWRDbGFzcyddO1xuXG4gIGZ1bmN0aW9uIGlzU2NvcGVkQ2xhc3Moc3RyKSB7XG4gICAgaWYgKCFzdHIpIHJldHVybiBmYWxzZTtcblxuICAgIHJldHVybiBzY29wZWRDbGFzc0NhbmRpZGF0ZXMuc29tZSgoY2FuZGlkYXRlKSA9PiBjYW5kaWRhdGUgPT09IHN0cik7XG4gIH1cblxuICByZXR1cm4ge1xuICAgIEF0dHJOb2RlKG5vZGUpIHtcbiAgICAgIGlmIChub2RlLm5hbWUgPT09ICdjbGFzcycpIHtcbiAgICAgICAgaWYgKG5vZGUudmFsdWUudHlwZSA9PT0gJ1RleHROb2RlJyAmJiBub2RlLnZhbHVlLmNoYXJzKSB7XG4gICAgICAgICAgY29uc3QgcmVuYW1lZENsYXNzID0gcmVuYW1lQ2xhc3Mobm9kZS52YWx1ZS5jaGFycywgcG9zdGZpeCwgY2xhc3Nlcyk7XG5cbiAgICAgICAgICBub2RlLnZhbHVlLmNoYXJzID0gcmVuYW1lZENsYXNzO1xuICAgICAgICB9IGVsc2UgaWYgKG5vZGUudmFsdWUudHlwZSA9PT0gJ0NvbmNhdFN0YXRlbWVudCcpIHtcbiAgICAgICAgICBmb3IgKGxldCBwYXJ0IG9mIG5vZGUudmFsdWUucGFydHMpIHtcbiAgICAgICAgICAgIGlmIChwYXJ0LnR5cGUgPT09ICdUZXh0Tm9kZScgJiYgcGFydC5jaGFycykge1xuICAgICAgICAgICAgICBjb25zdCByZW5hbWVkQ2xhc3MgPSByZW5hbWVDbGFzcyhwYXJ0LmNoYXJzLCBwb3N0Zml4LCBjbGFzc2VzKTtcblxuICAgICAgICAgICAgICBwYXJ0LmNoYXJzID0gcmVuYW1lZENsYXNzO1xuICAgICAgICAgICAgfSBlbHNlIGlmIChwYXJ0LnR5cGUgPT09ICdNdXN0YWNoZVN0YXRlbWVudCcpIHtcbiAgICAgICAgICAgICAgcmVjYXN0LnRyYXZlcnNlKHBhcnQsIHtcbiAgICAgICAgICAgICAgICBTdHJpbmdMaXRlcmFsKG5vZGUpIHtcbiAgICAgICAgICAgICAgICAgIGNvbnN0IHJlbmFtZWRDbGFzcyA9IHJlbmFtZUNsYXNzKFxuICAgICAgICAgICAgICAgICAgICBub2RlLnZhbHVlLFxuICAgICAgICAgICAgICAgICAgICBwb3N0Zml4LFxuICAgICAgICAgICAgICAgICAgICBjbGFzc2VzLFxuICAgICAgICAgICAgICAgICAgKTtcblxuICAgICAgICAgICAgICAgICAgbm9kZS52YWx1ZSA9IHJlbmFtZWRDbGFzcztcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9LFxuXG4gICAgRWxlbWVudE5vZGUobm9kZSkge1xuICAgICAgaWYgKHRhZ3MuaGFzKG5vZGUudGFnKSkge1xuICAgICAgICAvLyBjaGVjayBpZiBjbGFzcyBhdHRyaWJ1dGUgYWxyZWFkeSBleGlzdHNcbiAgICAgICAgY29uc3QgY2xhc3NBdHRyID0gbm9kZS5hdHRyaWJ1dGVzLmZpbmQoKGF0dHIpID0+IGF0dHIubmFtZSA9PT0gJ2NsYXNzJyk7XG5cbiAgICAgICAgaWYgKGNsYXNzQXR0cikge1xuICAgICAgICAgIGNsYXNzQXR0ci52YWx1ZS5jaGFycyArPSAnICcgKyBwb3N0Zml4O1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIHB1c2ggY2xhc3MgYXR0cmlidXRlXG4gICAgICAgICAgbm9kZS5hdHRyaWJ1dGVzLnB1c2goXG4gICAgICAgICAgICByZWNhc3QuYnVpbGRlcnMuYXR0cignY2xhc3MnLCByZWNhc3QuYnVpbGRlcnMudGV4dChwb3N0Zml4KSksXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0sXG5cbiAgICBBbGw6IHtcbiAgICAgIGVudGVyKG5vZGUpIHtcbiAgICAgICAgc3RhY2sucHVzaChub2RlKTtcbiAgICAgIH0sXG4gICAgICBleGl0KCkge1xuICAgICAgICBzdGFjay5wb3AoKTtcbiAgICAgIH0sXG4gICAgfSxcblxuICAgIE11c3RhY2hlU3RhdGVtZW50KG5vZGUpIHtcbiAgICAgIGxldCBjc3NDbGFzcztcblxuICAgICAgaWYgKFxuICAgICAgICBpc1Njb3BlZENsYXNzKGdldFZhbHVlKG5vZGUucGF0aCkpICYmXG4gICAgICAgIG5vZGUucGFyYW1zPy5sZW5ndGggPT09IDEgJiZcbiAgICAgICAgbm9kZS5wYXJhbXNbMF0udHlwZSA9PT0gJ1N0cmluZ0xpdGVyYWwnXG4gICAgICApIHtcbiAgICAgICAgY3NzQ2xhc3MgPSBub2RlLnBhcmFtc1swXS52YWx1ZTtcbiAgICAgIH1cblxuICAgICAgaWYgKFxuICAgICAgICBpc1Njb3BlZENsYXNzKGdldFZhbHVlKG5vZGUucGF0aD8ucGF0aCkpICYmXG4gICAgICAgIG5vZGUucGF0aD8ucGFyYW1zPy5sZW5ndGggPT09IDEgJiZcbiAgICAgICAgbm9kZS5wYXRoPy5wYXJhbXNbMF0udHlwZSA9PT0gJ1N0cmluZ0xpdGVyYWwnXG4gICAgICApIHtcbiAgICAgICAgY3NzQ2xhc3MgPSBub2RlLnBhdGgucGFyYW1zWzBdLnZhbHVlO1xuICAgICAgfVxuXG4gICAgICBpZiAoY3NzQ2xhc3MpIHtcbiAgICAgICAgY29uc3QgdGV4dE5vZGUgPSByZWNhc3QuYnVpbGRlcnMudGV4dChyZW5hbWVDbGFzcyhjc3NDbGFzcywgcG9zdGZpeCkpO1xuICAgICAgICBjb25zdCBwYXJlbnQgPSBzdGFja1tzdGFjay5sZW5ndGggLSAxXTtcblxuICAgICAgICBpZiAocGFyZW50Py50eXBlID09PSAnQXR0ck5vZGUnKSB7XG4gICAgICAgICAgcGFyZW50LnF1b3RlVHlwZSA9ICdcIic7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gdGV4dE5vZGU7XG4gICAgICB9XG4gICAgfSxcblxuICAgIFN1YkV4cHJlc3Npb24obm9kZSkge1xuICAgICAgaWYgKFxuICAgICAgICBpc1Njb3BlZENsYXNzKGdldFZhbHVlKG5vZGUucGF0aCkpICYmXG4gICAgICAgIG5vZGUucGFyYW1zPy5sZW5ndGggPT09IDEgJiZcbiAgICAgICAgbm9kZS5wYXJhbXNbMF0udHlwZSA9PT0gJ1N0cmluZ0xpdGVyYWwnXG4gICAgICApIHtcbiAgICAgICAgY29uc3QgY3NzQ2xhc3MgPSBub2RlLnBhcmFtc1swXS52YWx1ZTtcbiAgICAgICAgY29uc3QgdGV4dE5vZGUgPSByZWNhc3QuYnVpbGRlcnMubGl0ZXJhbChcbiAgICAgICAgICAnU3RyaW5nTGl0ZXJhbCcsXG4gICAgICAgICAgcmVuYW1lQ2xhc3MoY3NzQ2xhc3MsIHBvc3RmaXgpLFxuICAgICAgICApO1xuXG4gICAgICAgIHJldHVybiB0ZXh0Tm9kZTtcbiAgICAgIH1cbiAgICB9LFxuICB9O1xufVxuXG5mdW5jdGlvbiBnZXRWYWx1ZShwYXRoKSB7XG4gIGlmICghcGF0aCkgcmV0dXJuO1xuXG4gIGlmICgndmFsdWUnIGluIHBhdGgpIHtcbiAgICByZXR1cm4gcGF0aC52YWx1ZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZXByZWNhdGVkIGluIGVtYmVyIDUuOStcbiAgICogKHNvIHdlIHVzZSB0aGUgYWJvdmUgZm9yIG5ld2VyIGVtYmVycylcbiAgICovXG4gIHJldHVybiBwYXRoLm9yaWdpbmFsO1xufVxuXG5leHBvcnQgZGVmYXVsdCBmdW5jdGlvbiByZXdyaXRlSGJzKGhicywgY2xhc3NlcywgdGFncywgcG9zdGZpeCkge1xuICBsZXQgYXN0ID0gcmVjYXN0LnBhcnNlKGhicyk7XG5cbiAgcmVjYXN0LnRyYXZlcnNlKGFzdCwgdGVtcGxhdGVQbHVnaW4oeyBjbGFzc2VzLCB0YWdzLCBwb3N0Zml4IH0pKTtcblxuICBsZXQgcmVzdWx0ID0gcmVjYXN0LnByaW50KGFzdCk7XG5cbiAgcmV0dXJuIHJlc3VsdDtcbn1cbiIsIi8qKlxuICogQHR5cGVkZWYge2ltcG9ydCgnQGdsaW1tZXIvc3ludGF4JykuQVNUUGx1Z2lufSBBU1RQbHVnaW5cbiAqIEB0eXBlZGVmIHtpbXBvcnQoJ0BnbGltbWVyL3N5bnRheCcpLkFTVFBsdWdpbkVudmlyb25tZW50fSBBU1RQbHVnaW5FbnZpcm9ubWVudFxuICpcbiAqL1xuXG5pbXBvcnQgcGF0aCBmcm9tICdub2RlOnBhdGgnO1xuaW1wb3J0IHByb2Nlc3MgZnJvbSAnbm9kZTpwcm9jZXNzJztcblxuaW1wb3J0IHsgcmV3cml0ZUNzcyB9IGZyb20gJy4uL2xpYi9jc3MvcmV3cml0ZS5qcyc7XG5pbXBvcnQgeyBnZXRDU1NDb250ZW50SW5mbywgZ2V0Q1NTSW5mbyB9IGZyb20gJy4uL2xpYi9jc3MvdXRpbHMuanMnO1xuaW1wb3J0IHsgZml4RmlsZW5hbWUgfSBmcm9tICcuLi9saWIvcGF0aC90ZW1wbGF0ZS10cmFuc2Zvcm0tcGF0aHMuanMnO1xuaW1wb3J0IHtcbiAgYXBwUGF0aCxcbiAgY3NzUGF0aEZvcixcbiAgZm9yY2VQb3NpeCxcbiAgaGFzaEZyb21Nb2R1bGVQYXRoLFxuICBpc1JlbGV2YW50RmlsZSxcbn0gZnJvbSAnLi4vbGliL3BhdGgvdXRpbHMuanMnO1xuaW1wb3J0IHsgcmVxdWVzdCB9IGZyb20gJy4uL2xpYi9yZXF1ZXN0LmpzJztcbmltcG9ydCB7IHRlbXBsYXRlUGx1Z2luIH0gZnJvbSAnLi4vbGliL3Jld3JpdGVIYnMuanMnO1xuXG5jb25zdCBub29wUGx1Z2luID0ge1xuICBuYW1lOiAnZW1iZXItc2NvcGVkLWNzczpub29wJyxcbiAgdmlzaXRvcjoge30sXG59O1xuXG4vKipcbiAqIEByZXR1cm5zIHtBU1RQbHVnaW59XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVQbHVnaW4oY29uZmlnKSB7XG4gIC8qKlxuICAgKlxuICAgKiBAcGFyYW0ge0FTVFBsdWdpbkVudmlyb25tZW50fSBlbnZcbiAgICovXG4gIHJldHVybiBmdW5jdGlvbiBzY29wZWRDc3MoZW52KSB7XG4gICAgbGV0IGN3ZCA9IHByb2Nlc3MuY3dkKCk7XG5cbiAgICBsZXQgaXNSZWxldmFudCA9IGlzUmVsZXZhbnRGaWxlKGVudi5maWxlbmFtZSwge1xuICAgICAgYWRkaXRpb25hbFJvb3RzOiBjb25maWcuYWRkaXRpb25hbFJvb3RzLFxuICAgICAgY3dkLFxuICAgIH0pO1xuXG4gICAgaWYgKCFpc1JlbGV2YW50KSB7XG4gICAgICByZXR1cm4gbm9vcFBsdWdpbjtcbiAgICB9XG5cbiAgICBsZXQgYWJzb2x1dGVQYXRoID0gZml4RmlsZW5hbWUoZW52LmZpbGVuYW1lKTtcbiAgICBsZXQgbW9kdWxlUGF0aCA9IGFwcFBhdGgoYWJzb2x1dGVQYXRoKTtcbiAgICBsZXQgcG9zdGZpeCA9IGhhc2hGcm9tTW9kdWxlUGF0aChtb2R1bGVQYXRoKTtcblxuICAgIC8qKlxuICAgICAqIFRoZSBsaXN0IG9mIG5ha2VkIHRhZyBzZWxlY3RvcnMgZm91bmQgaW4gdGhlIENTU1xuICAgICAqXG4gICAgICogQHR5cGUge1NldDxzdHJpbmc+fVxuICAgICAqL1xuICAgIGxldCBzY29wZWRUYWdzID0gbmV3IFNldCgpO1xuXG4gICAgLyoqXG4gICAgICogVGhlIGxpc3Qgb2YgY2xhc3NlcyBmb3VuZCBpbiB0aGUgQ1NTXG4gICAgICpcbiAgICAgKiBAdHlwZSB7U2V0PHN0cmluZz59XG4gICAgICovXG4gICAgbGV0IHNjb3BlZENsYXNzZXMgPSBuZXcgU2V0KCk7XG5cbiAgICAvKipcbiAgICAgKiBAcGFyYW0ge3sgdGFnczogU2V0PHN0cmluZz47IGNsYXNzZXM6IFNldDxzdHJpbmc+IH19IGluZm9cbiAgICAgKi9cbiAgICBmdW5jdGlvbiBhZGRJbmZvKGluZm8pIHtcbiAgICAgIGZvciAobGV0IGl0ZW0gb2YgaW5mby50YWdzKSB7XG4gICAgICAgIHNjb3BlZFRhZ3MuYWRkKGl0ZW0pO1xuICAgICAgfVxuXG4gICAgICBmb3IgKGxldCBpdGVtIG9mIGluZm8uY2xhc3Nlcykge1xuICAgICAgICBzY29wZWRDbGFzc2VzLmFkZChpdGVtKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBsZXQgY3NzUGF0aCA9IGNzc1BhdGhGb3IoYWJzb2x1dGVQYXRoKTtcbiAgICBsZXQgaW5mbyA9IGdldENTU0luZm8oY3NzUGF0aCk7XG4gICAgbGV0IGxvY2FsQ3NzUGF0aCA9IGZvcmNlUG9zaXgoY3NzUGF0aC5yZXBsYWNlKGN3ZCArIHBhdGguc2VwLCAnJykpO1xuXG4gICAgLyoqXG4gICAgICogVGhpcyB3aWxsIGJlIGZhbHNleSBpZiB3ZSBkb24ndCBoYXZlIGEgY28tbG9jYXRlZCBDU1MgZmlsZS5cbiAgICAgKiBXZSdsbCBzdGlsbCB3YW50IHRvIGNoZWNrIGZvciBlbWJlZGRlZCA8c3R5bGUgc2NvcGVkPiB0YWdzIHRob3VnaC5cbiAgICAgKi9cbiAgICBpZiAoaW5mbykge1xuICAgICAgYWRkSW5mbyhpbmZvKTtcblxuICAgICAgbGV0IGNzc1JlcXVlc3QgPSByZXF1ZXN0LmNvbG9jYXRlZC5jcmVhdGUoaW5mby5pZCwgcG9zdGZpeCwgbG9jYWxDc3NQYXRoKTtcblxuICAgICAgLyoqXG4gICAgICAgKiBXaXRoIHRoaXMgd2UgZG9uJ3QgbmVlZCBhIEpTIHBsdWdpblxuICAgICAgICovXG4gICAgICBlbnYubWV0YS5qc3V0aWxzLmltcG9ydEZvclNpZGVFZmZlY3QoY3NzUmVxdWVzdCk7XG4gICAgfVxuXG4gICAgbGV0IHZpc2l0b3JzID0gdGVtcGxhdGVQbHVnaW4oe1xuICAgICAgY2xhc3Nlczogc2NvcGVkQ2xhc3NlcyxcbiAgICAgIHRhZ3M6IHNjb3BlZFRhZ3MsXG4gICAgICBwb3N0Zml4LFxuICAgIH0pO1xuXG4gICAgcmV0dXJuIHtcbiAgICAgIG5hbWU6ICdlbWJlci1zY29wZWQtY3NzOnRlbXBsYXRlLXBsdWdpbicsXG4gICAgICB2aXNpdG9yOiB7XG4gICAgICAgIC8vIFN0YWNrIE1hbmFnZXJcbiAgICAgICAgLi4udmlzaXRvcnMsXG5cbiAgICAgICAgLyoqXG4gICAgICAgICAqIFdlIGhhdmUgdG8gZWFnZXJseSBnZXQgdGhlIDxzdHlsZSBzY29wZWQ+IGNvbnRlbnRzLCBzbyB3ZSBjYW4gcHJlLXBhcnNlXG4gICAgICAgICAqIHRoZSB0YWdzIGFuZCBjbGFzc2VzIHRvIHRoZW4gcGFzcyB0byB0aGUgb3RoZXIgdmlzaXRvcnMgc28gdGhhdCB0aGV5IGNhblxuICAgICAgICAgKiBhcHByb3ByaWF0ZWx5IGNoYW5nZSBtYXRjaGluZyBjbGFzc2VzIC8gdGFncy5cbiAgICAgICAgICovXG4gICAgICAgIFRlbXBsYXRlKG5vZGUpIHtcbiAgICAgICAgICAvKipcbiAgICAgICAgICAgKiBXZSBvbmx5IGFsbG93IGEgc2NvcGVkIDxzdHlsZT4gYXQgdGhlIHJvb3RcbiAgICAgICAgICAgKi9cbiAgICAgICAgICBsZXQgc3R5bGVUYWcgPSBub2RlLmJvZHkuZmluZChcbiAgICAgICAgICAgIChuKSA9PiBuLnR5cGUgPT09ICdFbGVtZW50Tm9kZScgJiYgbi50YWcgPT09ICdzdHlsZScsXG4gICAgICAgICAgKTtcblxuICAgICAgICAgIGlmIChoYXNTY29wZWRBdHRyaWJ1dGUoc3R5bGVUYWcpKSB7XG4gICAgICAgICAgICBsZXQgY3NzID0gdGV4dENvbnRlbnQoc3R5bGVUYWcpO1xuICAgICAgICAgICAgbGV0IGxhbmcgPSBnZXRMYW5nQXR0cmlidXRlKHN0eWxlVGFnKTtcbiAgICAgICAgICAgIGxldCBpbmZvID0gZ2V0Q1NTQ29udGVudEluZm8oY3NzLCBsYW5nKTtcblxuICAgICAgICAgICAgYWRkSW5mbyhpbmZvKTtcblxuICAgICAgICAgICAgaWYgKGhhc0lubGluZUF0dHJpYnV0ZVdpdGhvdXRMYW5nKHN0eWxlVGFnKSkge1xuICAgICAgICAgICAgICAvKipcbiAgICAgICAgICAgICAgICogVGhpcyB3aWxsIGJlIGhhbmRsZWQgaW4gRWxlbWVudE5vZGUgdHJhdmVyc2FsXG4gICAgICAgICAgICAgICAqL1xuICAgICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmIChsYW5nKSB7XG4gICAgICAgICAgICAgIC8qKlxuICAgICAgICAgICAgICAgKiBGb3IgPHN0eWxlIHNjb3BlZCBpbmxpbmUgbGFuZz1cIi4uLlwiPiB3ZSBjYW5ub3QgcHJlcHJvY2VzcyBhdCBCYWJlbC10aW1lXG4gICAgICAgICAgICAgICAqIChwcmVwcm9jZXNzaW5nIGlzIGFzeW5jIGFuZCByZXF1aXJlcyBWaXRlJ3MgUmVzb2x2ZWRDb25maWcpLlxuICAgICAgICAgICAgICAgKiBSZW1vdmUgdGhlIHRhZyBhbmQgaW5qZWN0IHZpYSB2aXJ0dWFsIENTUyBtb2R1bGUgYW5kIHdhcm4gdXNlci5cbiAgICAgICAgICAgICAgICovXG4gICAgICAgICAgICAgIGNvbnNvbGUud2FybihcbiAgICAgICAgICAgICAgICBgW2VtYmVyLXNjb3BlZC1jc3NdIDxzdHlsZSBzY29wZWQgaW5saW5lIGxhbmc9XCIke2xhbmd9XCI+IGlzIG5vdCBzdXBwb3J0ZWQgYCArXG4gICAgICAgICAgICAgICAgICBgKHByZXByb2Nlc3NpbmcgaXMgYXN5bmMgYW5kIGNhbm5vdCBydW4gYXQgQmFiZWwtdGltZSkuIGAgK1xuICAgICAgICAgICAgICAgICAgYERvd25ncmFkaW5nIHRvIG5vbi1pbmxpbmU6IHRoZSBzdHlsZSB0YWcgd2lsbCBiZSByZW1vdmVkIGFuZCBpbmplY3RlZCBhcyBhIHZpcnR1YWwgQ1NTIG1vZHVsZS5gLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBsZXQgY3NzUmVxdWVzdCA9IHJlcXVlc3QuaW5saW5lLmNyZWF0ZShpbmZvLmlkLCBwb3N0Zml4LCBjc3MsIGxhbmcpO1xuXG4gICAgICAgICAgICBlbnYubWV0YS5qc3V0aWxzLmltcG9ydEZvclNpZGVFZmZlY3QoY3NzUmVxdWVzdCk7XG4gICAgICAgICAgfVxuICAgICAgICB9LFxuXG4gICAgICAgIC8vIFZpc2l0b3JzIGJyb2tlbiBvdXQgbGlrZSB0aGlzIHNvIHdlIGNhbiBjb25kaXRpb25hbGx5XG4gICAgICAgIC8vIGRlYnVnIGJhc2VkIG9uIGZpbGUgcGF0aC5cbiAgICAgICAgQXR0ck5vZGUoLi4uYXJncykge1xuICAgICAgICAgIHJldHVybiB2aXNpdG9ycy5BdHRyTm9kZSguLi5hcmdzKTtcbiAgICAgICAgfSxcbiAgICAgICAgRWxlbWVudE5vZGUobm9kZSwgd2Fsa2VyKSB7XG4gICAgICAgICAgLy8gY2xhc3MgYXR0cmlidXRlIGhhbmRsaW5nXG4gICAgICAgICAgdmlzaXRvcnMuRWxlbWVudE5vZGUobm9kZSwgd2Fsa2VyKTtcblxuICAgICAgICAgIGlmIChoYXNTY29wZWRBdHRyaWJ1dGUobm9kZSkpIHtcbiAgICAgICAgICAgIGlmICh3YWxrZXIucGFyZW50Py5ub2RlLnR5cGUgIT09ICdUZW1wbGF0ZScpIHtcbiAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAgICAgICAgICc8c3R5bGUgc2NvcGVkPiB0YWdzIG11c3QgYmUgYXQgdGhlIHJvb3Qgb2YgdGhlIHRlbXBsYXRlLCB0aGV5IGNhbm5vdCBiZSBuZXN0ZWQnLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBpZiAoaGFzSW5saW5lQXR0cmlidXRlV2l0aG91dExhbmcobm9kZSkpIHtcbiAgICAgICAgICAgICAgbGV0IHRleHQgPSB0ZXh0Q29udGVudChub2RlKTtcbiAgICAgICAgICAgICAgbGV0IHNjb3BlZFRleHQgPSByZXdyaXRlQ3NzKFxuICAgICAgICAgICAgICAgIHRleHQsXG4gICAgICAgICAgICAgICAgcG9zdGZpeCxcbiAgICAgICAgICAgICAgICBsb2NhbENzc1BhdGgsXG4gICAgICAgICAgICAgICAgY29uZmlnLmxheWVyTmFtZSxcbiAgICAgICAgICAgICAgKTtcblxuICAgICAgICAgICAgICAvKipcbiAgICAgICAgICAgICAgICogVHJhdmVyc2UgdGhpcyBhbmQgYWxsb3cgaW50ZXJwb2xhdGlvblxuICAgICAgICAgICAgICAgKi9cbiAgICAgICAgICAgICAgbm9kZS5jaGlsZHJlbiA9IFtlbnYuc3ludGF4LmJ1aWxkZXJzLnRleHQoc2NvcGVkVGV4dCldO1xuXG4gICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gUmV0dXJuaW5nIG51bGwgcmVtb3ZlcyB0aGUgbm9kZVxuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgaWYgKGhhc0lubGluZUF0dHJpYnV0ZVdpdGhvdXRMYW5nKG5vZGUpKSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICAgIGA8c3R5bGUgaW5saW5lPiBpcyBub3QgdmFsaWQuIFBsZWFzZSBhZGQgdGhlIHNjb3BlZCBhdHRyaWJ1dGU6IDxzdHlsZSBzY29wZWQgaW5saW5lPmAsXG4gICAgICAgICAgICApO1xuICAgICAgICAgIH1cbiAgICAgICAgfSxcbiAgICAgICAgTXVzdGFjaGVTdGF0ZW1lbnQoLi4uYXJncykge1xuICAgICAgICAgIHJldHVybiB2aXNpdG9ycy5NdXN0YWNoZVN0YXRlbWVudCguLi5hcmdzKTtcbiAgICAgICAgfSxcbiAgICAgICAgU3ViRXhwcmVzc2lvbiguLi5hcmdzKSB7XG4gICAgICAgICAgcmV0dXJuIHZpc2l0b3JzLlN1YkV4cHJlc3Npb24oLi4uYXJncyk7XG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgIH07XG4gIH07XG59XG5cbi8qKlxuICogVGhhbmtzLCBDYXJkU3RhY2sgYW5kIEBlZjQgZm9yIHRoaXMgY29kZS5cbiAqL1xuY29uc3QgU0NPUEVEX0FUVFJJQlVURV9OQU1FID0gJ3Njb3BlZCc7XG5jb25zdCBJTkxJTkVfQVRUUklCVVRFX05BTUUgPSAnaW5saW5lJztcbmNvbnN0IExBTkdfQVRUUklCVVRFX05BTUUgPSAnbGFuZyc7XG5cbmZ1bmN0aW9uIGhhc1Njb3BlZEF0dHJpYnV0ZShub2RlKSB7XG4gIGlmICghbm9kZSkgcmV0dXJuO1xuICBpZiAobm9kZS50YWcgIT09ICdzdHlsZScpIHJldHVybjtcbiAgaWYgKG5vZGUudHlwZSAhPT0gJ0VsZW1lbnROb2RlJykgcmV0dXJuO1xuXG4gIHJldHVybiBub2RlLmF0dHJpYnV0ZXMuc29tZShcbiAgICAoYXR0cmlidXRlKSA9PiBhdHRyaWJ1dGUubmFtZSA9PT0gU0NPUEVEX0FUVFJJQlVURV9OQU1FLFxuICApO1xufVxuXG5mdW5jdGlvbiBoYXNJbmxpbmVBdHRyaWJ1dGVXaXRob3V0TGFuZyhub2RlKSB7XG4gIGlmICghbm9kZSkgcmV0dXJuO1xuICBpZiAobm9kZS50YWcgIT09ICdzdHlsZScpIHJldHVybjtcbiAgaWYgKG5vZGUudHlwZSAhPT0gJ0VsZW1lbnROb2RlJykgcmV0dXJuO1xuXG4gIGlmIChnZXRMYW5nQXR0cmlidXRlKG5vZGUpKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgcmV0dXJuIG5vZGUuYXR0cmlidXRlcy5zb21lKFxuICAgIChhdHRyaWJ1dGUpID0+IGF0dHJpYnV0ZS5uYW1lID09PSBJTkxJTkVfQVRUUklCVVRFX05BTUUsXG4gICk7XG59XG5cbi8qKlxuICogUmV0dXJucyB0aGUgdmFsdWUgb2YgdGhlIGBsYW5nYCBhdHRyaWJ1dGUgb24gYSBgPHN0eWxlPmAgbm9kZSwgb3IgbnVsbCBpZiBhYnNlbnQuXG4gKlxuICogQHBhcmFtIHtvYmplY3R9IG5vZGVcbiAqIEByZXR1cm5zIHtzdHJpbmcgfCBudWxsfVxuICovXG5mdW5jdGlvbiBnZXRMYW5nQXR0cmlidXRlKG5vZGUpIHtcbiAgaWYgKCFub2RlKSByZXR1cm4gbnVsbDtcbiAgaWYgKG5vZGUudGFnICE9PSAnc3R5bGUnKSByZXR1cm4gbnVsbDtcbiAgaWYgKG5vZGUudHlwZSAhPT0gJ0VsZW1lbnROb2RlJykgcmV0dXJuIG51bGw7XG5cbiAgY29uc3QgYXR0ciA9IG5vZGUuYXR0cmlidXRlcy5maW5kKFxuICAgIChhdHRyaWJ1dGUpID0+IGF0dHJpYnV0ZS5uYW1lID09PSBMQU5HX0FUVFJJQlVURV9OQU1FLFxuICApO1xuXG4gIGlmICghYXR0cikgcmV0dXJuIG51bGw7XG5cbiAgLy8gVGhlIGF0dHJpYnV0ZSB2YWx1ZSBpcyBhIFRleHROb2RlIGNoaWxkIG9mIHRoZSBBdHRyTm9kZSdzIHZhbHVlXG4gIGNvbnN0IHZhbHVlID0gYXR0ci52YWx1ZTtcblxuICBpZiAodmFsdWU/LnR5cGUgPT09ICdUZXh0Tm9kZScpIHJldHVybiB2YWx1ZS5jaGFycyB8fCBudWxsO1xuXG4gIHJldHVybiBudWxsO1xufVxuXG5mdW5jdGlvbiB0ZXh0Q29udGVudChub2RlKSB7XG4gIGxldCB0ZXh0Q2hpbGRyZW4gPSBub2RlLmNoaWxkcmVuLmZpbHRlcigoYykgPT4gYy50eXBlID09PSAnVGV4dE5vZGUnKTtcblxuICByZXR1cm4gdGV4dENoaWxkcmVuLm1hcCgoYykgPT4gYy5jaGFycykuam9pbignJyk7XG59XG4iLCJpbXBvcnQgKiBhcyBiYWJlbCBmcm9tICcuLi9iYWJlbC1wbHVnaW4uanMnO1xuaW1wb3J0IHsgdW5wbHVnaW4gfSBmcm9tICcuLi9zY29wZWQtY3NzLXVucGx1Z2luLmpzJztcbmltcG9ydCAqIGFzIHRlbXBsYXRlIGZyb20gJy4uL3RlbXBsYXRlLXBsdWdpbi5qcyc7XG5cbmV4cG9ydCBjb25zdCBzY29wZWRDU1MgPSB7XG4gIHZpdGU6IHVucGx1Z2luLnZpdGUsXG4gIHJvbGx1cDogdW5wbHVnaW4ucm9sbHVwLFxuICBiYWJlbDogYmFiZWwuc2NvcGVkQ1NTLFxuICB0ZW1wbGF0ZTogdGVtcGxhdGUuY3JlYXRlUGx1Z2luLFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQU1BLE1BQWEsbUJBQW1CO0NBQzlCLGNBQWMsYUFBSyxLQUFLLDRCQUE0QjtDQUNwRCxhQUFhLGFBQUssS0FBSyxjQUFjO0NBQ3JDLGVBQWUsYUFBSyxLQUFLLGVBQWU7Q0FDeEMsY0FBYyxhQUFLLEtBQUssY0FBYztDQUN0QyxXQUFXLGFBQUssS0FBSyxXQUFXO0NBQ2hDLFFBQVEsYUFBSyxLQUFLLFVBQVU7Q0FDNUIsS0FBSyxhQUFLLEtBQUssUUFBUTtDQUN2QixLQUFLLGFBQUssS0FBSyxRQUFRO0NBQ3hCO0FBRUQsTUFBYSxXQUFXLEVBQ3RCLFNBQVMsYUFBSyxLQUFLLHFCQUFxQixFQUN6Qzs7Ozs7Ozs7Ozs7O0FDV0QsU0FBUyxRQUFRLEdBQUcsR0FBRztDQUNyQixJQUFJLE9BQU8sSUFBSSxVQUFXLElBQUk7QUFHOUIsU0FGVyxLQUFLLE9BQU8sS0FBSyxPQUFPLE9BQU8sT0FFM0IsS0FBTyxNQUFNOzs7Ozs7Ozs7QUFVOUIsU0FBUyxjQUFjLEtBQUssS0FBSztBQUMvQixRQUFRLE9BQU8sTUFBUSxRQUFTLEtBQUs7Ozs7Ozs7Ozs7Ozs7QUFjdkMsU0FBUyxPQUFPLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHO0FBQ2hDLFFBQU8sUUFBUSxjQUFjLFFBQVEsUUFBUSxHQUFHLEVBQUUsRUFBRSxRQUFRLEdBQUcsRUFBRSxDQUFDLEVBQUUsRUFBRSxFQUFFLEVBQUU7Ozs7Ozs7Ozs7Ozs7O0FBZTVFLFNBQVMsTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHO0FBQ2xDLFFBQU8sT0FBUSxJQUFJLElBQU0sQ0FBQyxJQUFJLEdBQUksR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFOzs7Ozs7Ozs7Ozs7OztBQWVsRCxTQUFTLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRztBQUNsQyxRQUFPLE9BQVEsSUFBSSxJQUFNLElBQUksQ0FBQyxHQUFJLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRTs7Ozs7Ozs7Ozs7Ozs7QUFlbEQsU0FBUyxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUc7QUFDbEMsUUFBTyxPQUFPLElBQUksSUFBSSxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRTs7Ozs7Ozs7Ozs7Ozs7QUFlekMsU0FBUyxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsR0FBRyxHQUFHLEdBQUc7QUFDbEMsUUFBTyxPQUFPLEtBQUssSUFBSSxDQUFDLElBQUksR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFOzs7Ozs7Ozs7QUFVNUMsU0FBUyxRQUFRLEdBQUcsS0FBSztBQUV2QixHQUFFLE9BQU8sTUFBTSxPQUFRLE1BQU07QUFDN0IsSUFBSyxNQUFNLE9BQVEsS0FBTSxLQUFLLE1BQU07Q0FFcEMsSUFBSTtDQUNKLElBQUk7Q0FDSixJQUFJO0NBQ0osSUFBSTtDQUNKLElBQUk7Q0FDSixJQUFJLElBQUk7Q0FDUixJQUFJLElBQUk7Q0FDUixJQUFJLElBQUk7Q0FDUixJQUFJLElBQUk7QUFFUixNQUFLLElBQUksR0FBRyxJQUFJLEVBQUUsUUFBUSxLQUFLLElBQUk7QUFDakMsU0FBTztBQUNQLFNBQU87QUFDUCxTQUFPO0FBQ1AsU0FBTztBQUVQLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxHQUFHLFdBQVc7QUFDMUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksVUFBVTtBQUM5QyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFlBQVk7QUFDaEQsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxXQUFXO0FBQzlDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksV0FBVztBQUMvQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFlBQVk7QUFDaEQsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxVQUFVO0FBQzlDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLEdBQUcsV0FBVztBQUM5QyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFlBQVk7QUFDaEQsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssSUFBSSxPQUFPO0FBQzVDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksWUFBWTtBQUNqRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxHQUFHLFdBQVc7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssSUFBSSxVQUFVO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksWUFBWTtBQUNqRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFdBQVc7QUFFaEQsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxXQUFXO0FBQzlDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLEdBQUcsWUFBWTtBQUMvQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFVBQVU7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksV0FBVztBQUMzQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxHQUFHLFdBQVc7QUFDOUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssR0FBRyxTQUFTO0FBQzdDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksV0FBVztBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFdBQVc7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxVQUFVO0FBQzdDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLEdBQUcsWUFBWTtBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFdBQVc7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLEdBQUcsWUFBWTtBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxHQUFHLFVBQVU7QUFDN0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksWUFBWTtBQUVqRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxHQUFHLFFBQVE7QUFDM0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxZQUFZO0FBQ2hELE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksV0FBVztBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFVBQVU7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxZQUFZO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksV0FBVztBQUMvQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFdBQVc7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssSUFBSSxZQUFZO0FBQ2pELE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLEdBQUcsVUFBVTtBQUM5QyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxXQUFXO0FBQzNDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksV0FBVztBQUMvQyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFNBQVM7QUFDN0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksR0FBRyxXQUFXO0FBQzlDLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksV0FBVztBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFVBQVU7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBRS9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxHQUFHLFdBQVc7QUFDMUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksWUFBWTtBQUNqRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxJQUFJLFVBQVU7QUFDOUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLEtBQUssR0FBRyxXQUFXO0FBQy9DLE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLElBQUksWUFBWTtBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFNBQVM7QUFDOUMsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxZQUFZO0FBQ2hELE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxJQUFJLEdBQUcsV0FBVztBQUM5QyxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksS0FBSyxJQUFJLFVBQVU7QUFDL0MsTUFBSSxNQUFNLEdBQUcsR0FBRyxHQUFHLEdBQUcsRUFBRSxJQUFJLElBQUksSUFBSSxZQUFZO0FBQ2hELE1BQUksTUFBTSxHQUFHLEdBQUcsR0FBRyxHQUFHLEVBQUUsSUFBSSxLQUFLLElBQUksV0FBVztBQUNoRCxNQUFJLE1BQU0sR0FBRyxHQUFHLEdBQUcsR0FBRyxFQUFFLElBQUksSUFBSSxHQUFHLFdBQVc7QUFDOUMsTUFBSSxN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
|