@libs-ui/utils 0.1.1-1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -0
- package/base64.d.ts +5 -0
- package/cache.d.ts +42 -0
- package/color.d.ts +11 -0
- package/communicate-micro.d.ts +16 -0
- package/constants.d.ts +10 -0
- package/crypto-3rd.d.ts +7 -0
- package/crypto.d.ts +8 -0
- package/dangerous-object.d.ts +79 -0
- package/date.d.ts +44 -0
- package/dom.d.ts +52 -0
- package/download.d.ts +3 -0
- package/esm2022/base64.mjs +43 -0
- package/esm2022/cache.mjs +388 -0
- package/esm2022/color.mjs +133 -0
- package/esm2022/communicate-micro.mjs +149 -0
- package/esm2022/constants.mjs +11 -0
- package/esm2022/crypto-3rd.mjs +38 -0
- package/esm2022/crypto.mjs +41 -0
- package/esm2022/dangerous-object.mjs +149 -0
- package/esm2022/date.mjs +191 -0
- package/esm2022/dom.mjs +256 -0
- package/esm2022/download.mjs +41 -0
- package/esm2022/file.mjs +90 -0
- package/esm2022/format-number.mjs +66 -0
- package/esm2022/format-text.mjs +149 -0
- package/esm2022/function-check-embed-frame.mjs +10 -0
- package/esm2022/get-smart-axis-scale.mjs +174 -0
- package/esm2022/helpers.mjs +651 -0
- package/esm2022/http-params.mjs +80 -0
- package/esm2022/index.mjs +30 -0
- package/esm2022/inject-token.mjs +5 -0
- package/esm2022/key-cache.mjs +31 -0
- package/esm2022/key-code.mjs +123 -0
- package/esm2022/language.mjs +70 -0
- package/esm2022/libs-ui-utils.mjs +5 -0
- package/esm2022/pattern.mjs +62 -0
- package/esm2022/random.mjs +42 -0
- package/esm2022/two-way-signal-object.mjs +131 -0
- package/esm2022/uri.mjs +25 -0
- package/esm2022/url-search-params.mjs +99 -0
- package/esm2022/uuid.mjs +18 -0
- package/esm2022/xss-filter.mjs +10 -0
- package/fesm2022/libs-ui-utils.mjs +3234 -0
- package/fesm2022/libs-ui-utils.mjs.map +1 -0
- package/file.d.ts +18 -0
- package/format-number.d.ts +2 -0
- package/format-text.d.ts +11 -0
- package/function-check-embed-frame.d.ts +2 -0
- package/get-smart-axis-scale.d.ts +34 -0
- package/helpers.d.ts +270 -0
- package/http-params.d.ts +37 -0
- package/index.d.ts +29 -0
- package/inject-token.d.ts +4 -0
- package/key-cache.d.ts +1 -0
- package/key-code.d.ts +122 -0
- package/language.d.ts +37 -0
- package/package.json +29 -0
- package/pattern.d.ts +20 -0
- package/random.d.ts +3 -0
- package/two-way-signal-object.d.ts +15 -0
- package/uri.d.ts +5 -0
- package/url-search-params.d.ts +25 -0
- package/uuid.d.ts +1 -0
- package/xss-filter.d.ts +3 -0
|
@@ -0,0 +1,651 @@
|
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
2
|
+
import { HttpParams } from '@angular/common/http';
|
|
3
|
+
import { isSignal, signal } from '@angular/core';
|
|
4
|
+
import dayjs from 'dayjs';
|
|
5
|
+
import { isArray, isDangerousObject, isDayjsObject, isMap, isRegExp, isReturnAsIsObject, isSet } from './dangerous-object';
|
|
6
|
+
import { getDayjs } from './date';
|
|
7
|
+
import { UtilsHttpParamsRequest } from './http-params';
|
|
8
|
+
import { unwrapSignal } from './two-way-signal-object';
|
|
9
|
+
/**Các hàm tương tự thư viện lodash */
|
|
10
|
+
/**
|
|
11
|
+
* Kiểm tra xem một giá trị có phải là null hoặc undefined hay không
|
|
12
|
+
* @param value Giá trị cần kiểm tra
|
|
13
|
+
* @returns true nếu giá trị là null hoặc undefined, false nếu không
|
|
14
|
+
* @example
|
|
15
|
+
* isNil(null); // true
|
|
16
|
+
* isNil(undefined); // true
|
|
17
|
+
* isNil(0); // false
|
|
18
|
+
* isNil('hello'); // false
|
|
19
|
+
*/
|
|
20
|
+
export const isNil = (value, options) => {
|
|
21
|
+
if (!options?.ignoreUnWrapSignal) {
|
|
22
|
+
value = unwrapSignal(value);
|
|
23
|
+
}
|
|
24
|
+
return value === null || value === undefined;
|
|
25
|
+
};
|
|
26
|
+
/**
|
|
27
|
+
* Kiểm tra xem một giá trị có phải là rỗng hay không
|
|
28
|
+
* @param value Giá trị cần kiểm tra
|
|
29
|
+
* @returns true nếu giá trị là null, rỗng hoặc undefined, false nếu không
|
|
30
|
+
* @example
|
|
31
|
+
* isEmpty(null); // true
|
|
32
|
+
* isEmpty(''); // true
|
|
33
|
+
* isEmpty(undefined); // true
|
|
34
|
+
* isEmpty({}); // true
|
|
35
|
+
* isEmpty([]); // true
|
|
36
|
+
* isEmpty([1, 2, 3]); // false
|
|
37
|
+
* isEmpty({ a: 1 }); // false
|
|
38
|
+
*/
|
|
39
|
+
export const isEmpty = (value, options) => {
|
|
40
|
+
if (!options?.ignoreUnWrapSignal) {
|
|
41
|
+
value = unwrapSignal(value);
|
|
42
|
+
}
|
|
43
|
+
if (options?.ignoreCheckString && value === '') {
|
|
44
|
+
return false;
|
|
45
|
+
}
|
|
46
|
+
if (options?.ignoreCheckTypePrimitive) {
|
|
47
|
+
return typeof value === 'object' && isEmptyTypeObject(value);
|
|
48
|
+
}
|
|
49
|
+
return isEmptyTypeObject(value);
|
|
50
|
+
};
|
|
51
|
+
/**
|
|
52
|
+
* Kiểm tra xem một giá trị có phải là object rỗng hay không
|
|
53
|
+
* @param value Giá trị cần kiểm tra
|
|
54
|
+
* @returns true nếu giá trị là object rỗng, false nếu không
|
|
55
|
+
* @example
|
|
56
|
+
* isEmptyTypeObject({}); // true
|
|
57
|
+
* isEmptyTypeObject({ a: 1 }); // false
|
|
58
|
+
*/
|
|
59
|
+
const isEmptyTypeObject = (value) => {
|
|
60
|
+
try {
|
|
61
|
+
if (isNil(value, { ignoreUnWrapSignal: true }) || value === '') {
|
|
62
|
+
return true;
|
|
63
|
+
}
|
|
64
|
+
if (typeof value !== 'object') {
|
|
65
|
+
return false;
|
|
66
|
+
}
|
|
67
|
+
if (Array.isArray(value)) {
|
|
68
|
+
return value.length === 0;
|
|
69
|
+
}
|
|
70
|
+
return Object.keys(value).length === 0;
|
|
71
|
+
}
|
|
72
|
+
catch (error) {
|
|
73
|
+
console.error(error);
|
|
74
|
+
return false;
|
|
75
|
+
}
|
|
76
|
+
};
|
|
77
|
+
/**
|
|
78
|
+
* Kiểm tra xem một giá trị có phải là null, rỗng, undefined hoặc 0 hay không
|
|
79
|
+
* @param value Giá trị cần kiểm tra
|
|
80
|
+
* @param options Cấu hình tùy chọn
|
|
81
|
+
* @param options.ignoreZero Nếu true, sẽ không kiểm tra giá trị 0
|
|
82
|
+
* @returns true nếu giá trị là null, rỗng, undefined hoặc 0, false nếu không
|
|
83
|
+
* @example
|
|
84
|
+
* isTruthy(null); // false
|
|
85
|
+
* isTruthy(''); // false
|
|
86
|
+
* isTruthy(undefined); // false
|
|
87
|
+
* isTruthy(0); // false
|
|
88
|
+
* isTruthy({}); // true
|
|
89
|
+
* isTruthy(0, { ignoreZero: true }); // true
|
|
90
|
+
*/
|
|
91
|
+
export const isTruthy = (value, options) => {
|
|
92
|
+
return !isFalsy(value, options);
|
|
93
|
+
};
|
|
94
|
+
/**
|
|
95
|
+
* Kiểm tra xem một giá trị có phải là null, rỗng, undefined hoặc 0 hay không
|
|
96
|
+
* @param value Giá trị cần kiểm tra
|
|
97
|
+
* @param options Cấu hình tùy chọn
|
|
98
|
+
* @param options.ignoreZero Nếu true, sẽ không kiểm tra giá trị 0
|
|
99
|
+
* @returns true nếu giá trị là null, rỗng, undefined hoặc 0, false nếu không
|
|
100
|
+
* @example
|
|
101
|
+
* isFalsy(null); // true
|
|
102
|
+
* isFalsy(''); // true
|
|
103
|
+
* isFalsy(undefined); // true
|
|
104
|
+
* isFalsy(0); // true
|
|
105
|
+
* isFalsy({}); // false
|
|
106
|
+
* isFalsy(0, { ignoreZero: true }); // false
|
|
107
|
+
*/
|
|
108
|
+
export const isFalsy = (value, options) => {
|
|
109
|
+
if (!options?.ignoreUnWrapSignal) {
|
|
110
|
+
value = unwrapSignal(value);
|
|
111
|
+
}
|
|
112
|
+
if (options?.ignoreZero && value === 0) {
|
|
113
|
+
return false;
|
|
114
|
+
}
|
|
115
|
+
if (options?.ignoreCheckString && value === '') {
|
|
116
|
+
return false;
|
|
117
|
+
}
|
|
118
|
+
return !value;
|
|
119
|
+
};
|
|
120
|
+
/**
|
|
121
|
+
* Loại bỏ các thuộc tính của đối tượng dựa trên một hàm điều kiện
|
|
122
|
+
* @param objData Đối tượng cần xử lý
|
|
123
|
+
* @param predicate Hàm điều kiện để kiểm tra giá trị của thuộc tính. Nếu hàm trả về true thì thuộc tính đó sẽ bị loại bỏ
|
|
124
|
+
* @returns Đối tượng mới sau khi đã loại bỏ các thuộc tính thỏa mãn điều kiện
|
|
125
|
+
* @example
|
|
126
|
+
* const obj = { a: 1, b: null, c: 3, d: undefined };
|
|
127
|
+
* omitBy(obj, isNil); // { a: 1, c: 3 }
|
|
128
|
+
*/
|
|
129
|
+
export const omitBy = (objData, predicate) => {
|
|
130
|
+
if (!objData || typeof objData !== 'object' || Array.isArray(objData)) {
|
|
131
|
+
return objData;
|
|
132
|
+
}
|
|
133
|
+
const newObj = {};
|
|
134
|
+
Object.keys(objData).forEach((key) => {
|
|
135
|
+
const valueOfKey = get(objData, key);
|
|
136
|
+
if (!predicate(valueOfKey)) {
|
|
137
|
+
set(newObj, key, valueOfKey);
|
|
138
|
+
}
|
|
139
|
+
});
|
|
140
|
+
return newObj;
|
|
141
|
+
};
|
|
142
|
+
/**
|
|
143
|
+
* Lấy giá trị từ đối tượng theo đường dẫn chỉ định
|
|
144
|
+
*
|
|
145
|
+
* Hàm này giúp bạn truy cập vào các thuộc tính sâu bên trong một đối tượng một cách an toàn,
|
|
146
|
+
* tránh lỗi khi thuộc tính không tồn tại.
|
|
147
|
+
*
|
|
148
|
+
* @param obj Đối tượng nguồn cần lấy giá trị
|
|
149
|
+
* @param path Đường dẫn đến thuộc tính cần lấy. Có thể là:
|
|
150
|
+
* - Chuỗi: 'user.profile.name' hoặc 'items[0].title'
|
|
151
|
+
* - Mảng: ['user', 'profile', 'name'] hoặc ['items', '0', 'title']
|
|
152
|
+
* - Chuỗi rỗng '': trả về chính đối tượng gốc
|
|
153
|
+
* @param defaultValue Giá trị mặc định trả về khi không tìm thấy thuộc tính (mặc định: undefined)
|
|
154
|
+
* @param keepLastValueIfSignal Có giữ nguyên signal cuối cùng hay không (mặc định: false - sẽ gọi signal())
|
|
155
|
+
* @returns Giá trị tìm được hoặc giá trị mặc định
|
|
156
|
+
*
|
|
157
|
+
* @example
|
|
158
|
+
* // Ví dụ cơ bản
|
|
159
|
+
* const user = { name: 'John', age: 30 };
|
|
160
|
+
* get(user, 'name'); // 'John'
|
|
161
|
+
* get(user, 'email'); // undefined
|
|
162
|
+
* get(user, 'email', 'no-email'); // 'no-email'
|
|
163
|
+
*
|
|
164
|
+
* @example
|
|
165
|
+
* // Truyền path rỗng - trả về chính đối tượng gốc
|
|
166
|
+
* const data = { name: 'Alice', age: 25 };
|
|
167
|
+
* get(data, ''); // { name: 'Alice', age: 25 } (chính đối tượng data)
|
|
168
|
+
* get(data, '', 'default'); // { name: 'Alice', age: 25 } (bỏ qua defaultValue)
|
|
169
|
+
*
|
|
170
|
+
* @example
|
|
171
|
+
* // Truy cập thuộc tính sâu
|
|
172
|
+
* const data = {
|
|
173
|
+
* user: {
|
|
174
|
+
* profile: {
|
|
175
|
+
* name: 'Alice',
|
|
176
|
+
* settings: { theme: 'dark' }
|
|
177
|
+
* }
|
|
178
|
+
* }
|
|
179
|
+
* };
|
|
180
|
+
* get(data, 'user.profile.name'); // 'Alice'
|
|
181
|
+
* get(data, 'user.profile.settings.theme'); // 'dark'
|
|
182
|
+
* get(data, 'user.profile.avatar', 'default.jpg'); // 'default.jpg'
|
|
183
|
+
*
|
|
184
|
+
* @example
|
|
185
|
+
* // Truy cập mảng
|
|
186
|
+
* const items = [
|
|
187
|
+
* { name: 'Item 1', price: 100 },
|
|
188
|
+
* { name: 'Item 2', price: 200 }
|
|
189
|
+
* ];
|
|
190
|
+
* get(items, '[0].name'); // 'Item 1'
|
|
191
|
+
* get(items, '[1].name'); // 'Item 2'
|
|
192
|
+
* get(items, '[2].name', 'Not found'); // 'Not found'
|
|
193
|
+
*
|
|
194
|
+
* @example
|
|
195
|
+
* // Sử dụng với mảng path
|
|
196
|
+
* const nested = { a: { b: { c: 'deep value' } } };
|
|
197
|
+
* get(nested, ['a', 'b', 'c']); // 'deep value'
|
|
198
|
+
* get(nested, ['a', 'b', 'd'], 'default'); // 'default'
|
|
199
|
+
*
|
|
200
|
+
* @example
|
|
201
|
+
* // Trường hợp đặc biệt
|
|
202
|
+
* get(null, 'any.path'); // undefined
|
|
203
|
+
* get(undefined, 'any.path', 'fallback'); // 'fallback'
|
|
204
|
+
*/
|
|
205
|
+
export const get = (obj, path, defaultValue, keepLastValueIfSignal) => {
|
|
206
|
+
// helper cast để tránh lặp lại kiểu điều kiện ở các return
|
|
207
|
+
const out = (v) => v;
|
|
208
|
+
if (isNil(obj)) {
|
|
209
|
+
return out(defaultValue);
|
|
210
|
+
}
|
|
211
|
+
obj = unwrapSignal(obj);
|
|
212
|
+
if (path === '') {
|
|
213
|
+
return out(obj);
|
|
214
|
+
}
|
|
215
|
+
if (obj instanceof HttpParams) {
|
|
216
|
+
return out(obj.get(`${path}`) ?? defaultValue);
|
|
217
|
+
}
|
|
218
|
+
if (obj instanceof DOMRect) {
|
|
219
|
+
return out(obj[path]);
|
|
220
|
+
}
|
|
221
|
+
const paths = Array.isArray(path)
|
|
222
|
+
? path
|
|
223
|
+
: path
|
|
224
|
+
.replace(/\[(\d+)]/g, '.$1')
|
|
225
|
+
.split('.')
|
|
226
|
+
.filter((key) => key);
|
|
227
|
+
for (const index in paths) {
|
|
228
|
+
const key = paths[index];
|
|
229
|
+
if (obj instanceof CSSStyleDeclaration) {
|
|
230
|
+
obj = obj[key];
|
|
231
|
+
continue;
|
|
232
|
+
}
|
|
233
|
+
if (obj instanceof HTMLElement) {
|
|
234
|
+
obj = obj[key];
|
|
235
|
+
continue;
|
|
236
|
+
}
|
|
237
|
+
if (isNil(obj) || !Object.prototype.hasOwnProperty.call(obj, key)) {
|
|
238
|
+
return out(defaultValue);
|
|
239
|
+
}
|
|
240
|
+
const val = isSignal(obj[key]) && !keepLastValueIfSignal ? obj[key]() : obj[key];
|
|
241
|
+
obj = val;
|
|
242
|
+
}
|
|
243
|
+
return out(obj ?? defaultValue);
|
|
244
|
+
};
|
|
245
|
+
/**
|
|
246
|
+
* Thiết lập giá trị cho một thuộc tính trong đối tượng theo đường dẫn chỉ định
|
|
247
|
+
* @param obj Đối tượng cần thiết lập giá trị
|
|
248
|
+
* @param path Đường dẫn đến thuộc tính, có thể là chuỗi (vd: 'a.b.c') hoặc mảng (vd: ['a', 'b', 'c'])
|
|
249
|
+
* @param value Giá trị cần thiết lập
|
|
250
|
+
* @returns Đối tượng sau khi đã thiết lập giá trị
|
|
251
|
+
* @throws Error nếu tham số đầu tiên không phải là đối tượng
|
|
252
|
+
* @example
|
|
253
|
+
* const obj = { a: { b: 1 } };
|
|
254
|
+
* set(obj, 'a.b', 2); // { a: { b: 2 } }
|
|
255
|
+
*/
|
|
256
|
+
export const set = (obj, path, value, options) => {
|
|
257
|
+
if (!obj || (typeof obj !== 'object' && !isSignal(obj)) || (isSignal(obj) && typeof obj() !== 'object')) {
|
|
258
|
+
throw new Error('The first argument must be an object');
|
|
259
|
+
}
|
|
260
|
+
if (obj instanceof HttpParams) {
|
|
261
|
+
return obj.set(`${path}`, value);
|
|
262
|
+
}
|
|
263
|
+
const pathArray = Array.isArray(path)
|
|
264
|
+
? path
|
|
265
|
+
: path
|
|
266
|
+
.replace(/\[(\d+)]/g, '.[$1]')
|
|
267
|
+
.split('.')
|
|
268
|
+
.filter((key) => key);
|
|
269
|
+
let currentObjectByKey = isSignal(obj) ? obj() : obj;
|
|
270
|
+
let preObjectByKey = obj;
|
|
271
|
+
pathArray.forEach((key, index) => {
|
|
272
|
+
if (index < pathArray.length - 1) {
|
|
273
|
+
if (!(key in currentObjectByKey) || (typeof currentObjectByKey[key] !== 'object' && !isSignal(currentObjectByKey[key]))) {
|
|
274
|
+
const nextKey = pathArray[index + 1];
|
|
275
|
+
key = key.replace(/\[(\d+)]/g, '$1');
|
|
276
|
+
if (isNil(currentObjectByKey[key])) {
|
|
277
|
+
currentObjectByKey[key] = /\[(\d+)]/g.test(nextKey) ? (options?.valueDefaultPathArrayUndefined ?? []) : (options?.valueDefaultPathObjectUndefined ?? {});
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
currentObjectByKey = key ? currentObjectByKey[key] : currentObjectByKey;
|
|
281
|
+
preObjectByKey = currentObjectByKey;
|
|
282
|
+
currentObjectByKey = isSignal(currentObjectByKey) ? currentObjectByKey() : currentObjectByKey;
|
|
283
|
+
return;
|
|
284
|
+
}
|
|
285
|
+
if (typeof currentObjectByKey !== 'object') {
|
|
286
|
+
return;
|
|
287
|
+
}
|
|
288
|
+
// Gán giá trị ở cuối đường dẫn
|
|
289
|
+
key = key.replace(/\[(\d+)]/g, '$1');
|
|
290
|
+
const valueOfKey = currentObjectByKey[key];
|
|
291
|
+
const valueOfKeyIsSignal = isSignal(valueOfKey);
|
|
292
|
+
if (valueOfKeyIsSignal && (typeof valueOfKey() !== 'object' || valueOfKey() === null)) {
|
|
293
|
+
valueOfKey.set(isSignal(value) ? value() : value);
|
|
294
|
+
return;
|
|
295
|
+
}
|
|
296
|
+
if (isSignal(preObjectByKey)) {
|
|
297
|
+
preObjectByKey.update((data) => {
|
|
298
|
+
const dataOfKey = data[key];
|
|
299
|
+
if (isNil(dataOfKey) || !valueOfKeyIsSignal) {
|
|
300
|
+
data[key] = value;
|
|
301
|
+
}
|
|
302
|
+
if (valueOfKeyIsSignal) {
|
|
303
|
+
valueOfKey.set(isSignal(value) ? value() : value);
|
|
304
|
+
}
|
|
305
|
+
if (Array.isArray(data)) {
|
|
306
|
+
return [...data];
|
|
307
|
+
}
|
|
308
|
+
return { ...data };
|
|
309
|
+
});
|
|
310
|
+
return;
|
|
311
|
+
}
|
|
312
|
+
if (valueOfKeyIsSignal) {
|
|
313
|
+
valueOfKey.set(isSignal(value) ? value() : value);
|
|
314
|
+
return;
|
|
315
|
+
}
|
|
316
|
+
currentObjectByKey[key] = value;
|
|
317
|
+
});
|
|
318
|
+
return obj;
|
|
319
|
+
};
|
|
320
|
+
/**
|
|
321
|
+
* Tạo một bản sao sâu của một đối tượng hoặc giá trị bất kỳ
|
|
322
|
+
* @param data Dữ liệu cần sao chép
|
|
323
|
+
* @param options Tùy chọn cấu hình
|
|
324
|
+
* @param options.ignoreSignal Nếu true, sẽ không sao chép các signal mà trả về signal gốc
|
|
325
|
+
* @param seen WeakMap để theo dõi các tham chiếu đã được sao chép, tránh lặp vô hạn với các tham chiếu vòng
|
|
326
|
+
* @returns Bản sao sâu của dữ liệu đầu vào
|
|
327
|
+
* @example
|
|
328
|
+
* const obj = {
|
|
329
|
+
* a: 1,
|
|
330
|
+
* b: { c: 2 },
|
|
331
|
+
* d: [1, 2, 3]
|
|
332
|
+
* };
|
|
333
|
+
* const clone = cloneDeep(obj);
|
|
334
|
+
* // clone là một bản sao hoàn toàn độc lập của obj
|
|
335
|
+
*/
|
|
336
|
+
export const cloneDeep = (data, options = { ignoreSignal: false }, seen = new WeakMap()) => {
|
|
337
|
+
if (data === null || (typeof data !== 'object' && !isSignal(data)) || isDangerousObject(data)) {
|
|
338
|
+
return data;
|
|
339
|
+
}
|
|
340
|
+
if (seen.has(data)) {
|
|
341
|
+
return seen.get(data);
|
|
342
|
+
}
|
|
343
|
+
if (data instanceof HttpParams) {
|
|
344
|
+
return new UtilsHttpParamsRequest(undefined, data);
|
|
345
|
+
}
|
|
346
|
+
if (data instanceof Date) {
|
|
347
|
+
return new Date(data.getTime());
|
|
348
|
+
}
|
|
349
|
+
if (isDayjsObject(data)) {
|
|
350
|
+
return getDayjs({ date: data.valueOf() });
|
|
351
|
+
}
|
|
352
|
+
if (isRegExp(data)) {
|
|
353
|
+
return new RegExp(data.source, data.flags);
|
|
354
|
+
}
|
|
355
|
+
if (isMap(data)) {
|
|
356
|
+
const mapCopy = new Map();
|
|
357
|
+
seen.set(data, mapCopy);
|
|
358
|
+
data.forEach((val, key) => {
|
|
359
|
+
mapCopy.set(cloneDeep(key, options, seen), cloneDeep(val, options, seen));
|
|
360
|
+
});
|
|
361
|
+
return mapCopy;
|
|
362
|
+
}
|
|
363
|
+
if (isSet(data)) {
|
|
364
|
+
const setCopy = new Set();
|
|
365
|
+
seen.set(data, setCopy);
|
|
366
|
+
data.forEach((val) => {
|
|
367
|
+
setCopy.add(cloneDeep(val, options, seen));
|
|
368
|
+
});
|
|
369
|
+
return setCopy;
|
|
370
|
+
}
|
|
371
|
+
if (isArray(data)) {
|
|
372
|
+
seen.set(data, data.map((item) => cloneDeep(item, options, seen)));
|
|
373
|
+
return seen.get(data);
|
|
374
|
+
}
|
|
375
|
+
if (isReturnAsIsObject(data)) {
|
|
376
|
+
return data;
|
|
377
|
+
}
|
|
378
|
+
if (isSignal(data)) {
|
|
379
|
+
if (options?.ignoreSignal) {
|
|
380
|
+
return data;
|
|
381
|
+
}
|
|
382
|
+
seen.set(data, signal(cloneDeep(data(), options, seen)));
|
|
383
|
+
return seen.get(data);
|
|
384
|
+
}
|
|
385
|
+
const result = {};
|
|
386
|
+
seen.set(data, result);
|
|
387
|
+
for (const key in data) {
|
|
388
|
+
const value = data[key];
|
|
389
|
+
if (value instanceof HttpParams) {
|
|
390
|
+
result[key] = new UtilsHttpParamsRequest(undefined, value);
|
|
391
|
+
continue;
|
|
392
|
+
}
|
|
393
|
+
if (value instanceof Date) {
|
|
394
|
+
result[key] = new Date(value.getTime());
|
|
395
|
+
continue;
|
|
396
|
+
}
|
|
397
|
+
if (dayjs.isDayjs(value)) {
|
|
398
|
+
result[key] = getDayjs({ date: value.valueOf() });
|
|
399
|
+
continue;
|
|
400
|
+
}
|
|
401
|
+
if (isRegExp(data)) {
|
|
402
|
+
result[key] = new RegExp(value.source, value.flags);
|
|
403
|
+
continue;
|
|
404
|
+
}
|
|
405
|
+
if (isReturnAsIsObject(data)) {
|
|
406
|
+
result[key] = value;
|
|
407
|
+
continue;
|
|
408
|
+
}
|
|
409
|
+
if (Object.prototype.hasOwnProperty.call(data, key)) {
|
|
410
|
+
result[key] = cloneDeep(value, options, seen);
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
return result;
|
|
414
|
+
};
|
|
415
|
+
/**
|
|
416
|
+
* Chuyển đổi một mảng các đối tượng thành một đối tượng với khóa được chỉ định
|
|
417
|
+
* @param data Mảng các đối tượng cần chuyển đổi
|
|
418
|
+
* @param key Tên thuộc tính được sử dụng làm khóa trong đối tượng kết quả
|
|
419
|
+
* @returns Đối tượng với các giá trị từ mảng được đánh key theo thuộc tính đã chỉ định
|
|
420
|
+
* @example
|
|
421
|
+
* const data = [
|
|
422
|
+
* { id: 1, name: 'John' },
|
|
423
|
+
* { id: 2, name: 'Jane' }
|
|
424
|
+
* ];
|
|
425
|
+
* keyBy(data, 'id');
|
|
426
|
+
* // Kết quả: {
|
|
427
|
+
* // '1': { id: 1, name: 'John' },
|
|
428
|
+
* // '2': { id: 2, name: 'Jane' }
|
|
429
|
+
* // }
|
|
430
|
+
*/
|
|
431
|
+
export const keyBy = (data, key) => {
|
|
432
|
+
if (!data || !data.length || !key) {
|
|
433
|
+
return {};
|
|
434
|
+
}
|
|
435
|
+
return data.reduce((dir, nextItem) => {
|
|
436
|
+
const valueOfKey = get(nextItem, key);
|
|
437
|
+
if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number' && typeof valueOfKey !== 'boolean') {
|
|
438
|
+
return dir;
|
|
439
|
+
}
|
|
440
|
+
if (!Object.keys(dir).includes(`${valueOfKey}`)) {
|
|
441
|
+
dir[`${valueOfKey}`] = nextItem;
|
|
442
|
+
}
|
|
443
|
+
return dir;
|
|
444
|
+
}, {});
|
|
445
|
+
};
|
|
446
|
+
/**
|
|
447
|
+
* Nhóm các đối tượng trong một mảng thành các nhóm dựa trên một thuộc tính cụ thể
|
|
448
|
+
* @param data Mảng các đối tượng cần nhóm
|
|
449
|
+
* @param key Tên thuộc tính được sử dụng làm khóa nhóm
|
|
450
|
+
* @returns Đối tượng với các giá trị từ mảng được nhóm theo thuộc tính đã chỉ định
|
|
451
|
+
* @example
|
|
452
|
+
* const data = [
|
|
453
|
+
* { id: 1, name: 'John' },
|
|
454
|
+
* { id: 2, name: 'Jane' },
|
|
455
|
+
* { id: 1, name: 'John' }
|
|
456
|
+
* ];
|
|
457
|
+
* groupBy(data, 'id');
|
|
458
|
+
* // Kết quả: {
|
|
459
|
+
* // '1': [
|
|
460
|
+
* // { id: 1, name: 'John' },
|
|
461
|
+
* // { id: 1, name: 'John' }
|
|
462
|
+
* // ],
|
|
463
|
+
* // '2': [
|
|
464
|
+
* // { id: 2, name: 'Jane' }
|
|
465
|
+
* // }
|
|
466
|
+
*/
|
|
467
|
+
export const groupBy = (data, key) => {
|
|
468
|
+
if (!data || !data.length || !Object.keys(get(data, '[0]')).length || !key) {
|
|
469
|
+
return {};
|
|
470
|
+
}
|
|
471
|
+
return data.reduce((dir, nextItem) => {
|
|
472
|
+
const valueOfKey = get(nextItem, key);
|
|
473
|
+
if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number' && typeof valueOfKey !== 'boolean') {
|
|
474
|
+
return dir;
|
|
475
|
+
}
|
|
476
|
+
if (!Object.keys(dir).includes(`${valueOfKey}`)) {
|
|
477
|
+
dir[`${valueOfKey}`] = [];
|
|
478
|
+
}
|
|
479
|
+
dir[`${valueOfKey}`].push(nextItem);
|
|
480
|
+
return dir;
|
|
481
|
+
}, {});
|
|
482
|
+
};
|
|
483
|
+
/**
|
|
484
|
+
* Tạo một mảng các số từ giá trị bắt đầu đến giá trị kết thúc với bước nhảy tùy chọn
|
|
485
|
+
* @param start Giá trị bắt đầu của dãy số. Nếu chỉ có một tham số, đây sẽ là giá trị kết thúc và giá trị bắt đầu sẽ là 0
|
|
486
|
+
* @param end Giá trị kết thúc của dãy số (tùy chọn)
|
|
487
|
+
* @param step Bước nhảy giữa các số trong dãy (tùy chọn). Mặc định là 1 nếu end > start, -1 nếu end < start
|
|
488
|
+
* @returns Mảng các số từ start đến end với bước nhảy step
|
|
489
|
+
* @example
|
|
490
|
+
* range(4); // [0, 1, 2, 3]
|
|
491
|
+
* range(1, 5); // [1, 2, 3, 4]
|
|
492
|
+
* range(0, 20, 5); // [0, 5, 10, 15]
|
|
493
|
+
* range(5, 2); // [5, 4, 3]
|
|
494
|
+
*/
|
|
495
|
+
export const range = (start, end, step) => {
|
|
496
|
+
if (end === undefined || end === null) {
|
|
497
|
+
end = start;
|
|
498
|
+
start = 0;
|
|
499
|
+
}
|
|
500
|
+
if (!step) {
|
|
501
|
+
step = end < 0 ? -1 : 1;
|
|
502
|
+
}
|
|
503
|
+
if (end < start && step > 0) {
|
|
504
|
+
step *= -1;
|
|
505
|
+
}
|
|
506
|
+
const valueStartByStep = step + start;
|
|
507
|
+
const direction = start < end ? 'asc' : 'desc';
|
|
508
|
+
if ((direction === 'asc' && (valueStartByStep < start || valueStartByStep > end)) || (direction === 'desc' && (valueStartByStep > start || valueStartByStep < end))) {
|
|
509
|
+
return [start];
|
|
510
|
+
}
|
|
511
|
+
const arr = new Array();
|
|
512
|
+
for (let index = 0; index < Math.abs(end - start); index++) {
|
|
513
|
+
let value = start + index * step;
|
|
514
|
+
if (index === 0) {
|
|
515
|
+
value = start;
|
|
516
|
+
}
|
|
517
|
+
if ((direction === 'asc' && (value < start || value > end)) || (direction === 'desc' && (value > start || value < end))) {
|
|
518
|
+
return arr;
|
|
519
|
+
}
|
|
520
|
+
arr.push(value);
|
|
521
|
+
}
|
|
522
|
+
return arr;
|
|
523
|
+
};
|
|
524
|
+
/**
|
|
525
|
+
* So sánh hai giá trị bất kỳ có bằng nhau hay không
|
|
526
|
+
* @param value1 Giá trị thứ nhất cần so sánh
|
|
527
|
+
* @param value2 Giá trị thứ hai cần so sánh
|
|
528
|
+
* @param exactlyPosition Có so sánh chính xác vị trí các phần tử trong mảng hay không
|
|
529
|
+
* @returns true nếu hai giá trị bằng nhau, false nếu không bằng nhau
|
|
530
|
+
* @example
|
|
531
|
+
* isEqual([1,2,3], [1,2,3]); // true
|
|
532
|
+
* isEqual([1,2,3], [3,2,1]); // true khi exactlyPosition = false
|
|
533
|
+
* isEqual([1,2,3], [3,2,1]); // false khi exactlyPosition = true
|
|
534
|
+
* isEqual({a:1}, {a:1}); // true
|
|
535
|
+
*/
|
|
536
|
+
export const isEqual = (value1, value2, options) => {
|
|
537
|
+
// Handle signals
|
|
538
|
+
if (!options?.ignoreUnWrapSignal) {
|
|
539
|
+
value1 = unwrapSignal(value1);
|
|
540
|
+
value2 = unwrapSignal(value2);
|
|
541
|
+
}
|
|
542
|
+
const { exactlyPosition = false, ignoreExactlyDataType = false } = options || {};
|
|
543
|
+
if (value1 === value2 || (value1 === null && value2 === null) || (value1 === undefined && value2 === undefined)) {
|
|
544
|
+
return true;
|
|
545
|
+
}
|
|
546
|
+
if (ignoreExactlyDataType) {
|
|
547
|
+
return isEqual(isNil(value1) ? undefined : `${value1}`, isNil(value2) ? undefined : `${value2}`);
|
|
548
|
+
}
|
|
549
|
+
if (typeof value1 !== 'object' || typeof value2 !== 'object' || (Array.isArray(value1) && !Array.isArray(value2)) || (!Array.isArray(value1) && Array.isArray(value2))) {
|
|
550
|
+
return false;
|
|
551
|
+
}
|
|
552
|
+
if (Array.isArray(value1) || Array.isArray(value2)) {
|
|
553
|
+
if (value1.length !== value2.length) {
|
|
554
|
+
return false;
|
|
555
|
+
}
|
|
556
|
+
if (!exactlyPosition) {
|
|
557
|
+
return !value1.some((item) => !value2.includes(item));
|
|
558
|
+
}
|
|
559
|
+
return !value1.some((item, index) => !isEqual(item, value2[index], options));
|
|
560
|
+
}
|
|
561
|
+
if (Object.keys(value1).length !== Object.keys(value2).length) {
|
|
562
|
+
return false;
|
|
563
|
+
}
|
|
564
|
+
return !Object.keys(value1).some((key) => !isEqual(value1[key], value2[key], options));
|
|
565
|
+
};
|
|
566
|
+
/**
|
|
567
|
+
* Loại bỏ các phần tử trùng lặp trong mảng dựa trên một thuộc tính chỉ định
|
|
568
|
+
* @param data Mảng dữ liệu cần xử lý
|
|
569
|
+
* @param key Tên thuộc tính dùng để so sánh trùng lặp. Nếu không có key thì so sánh trực tiếp giá trị
|
|
570
|
+
* @returns Mảng mới chứa các phần tử không trùng lặp
|
|
571
|
+
* @example
|
|
572
|
+
* const arr = [
|
|
573
|
+
* { id: 1, name: 'A' },
|
|
574
|
+
* { id: 2, name: 'B' },
|
|
575
|
+
* { id: 1, name: 'C' }
|
|
576
|
+
* ];
|
|
577
|
+
* uniqBy(arr, 'id'); // [{ id: 1, name: 'A' }, { id: 2, name: 'B' }]
|
|
578
|
+
*
|
|
579
|
+
* const numbers = [1, 2, 2, 3, 3];
|
|
580
|
+
* uniqBy(numbers); // [1, 2, 3]
|
|
581
|
+
*
|
|
582
|
+
* const numbersSignal = [signal(1), signal(2), signal(3), signal(2), signal(5), signal(4), signal(1), signal(6), signal(7), signal(6)];
|
|
583
|
+
* uniqBy(numbersSignal); // [signal(1), signal(2), signal(3), signal(5), signal(4), signal(6), signal(7)]
|
|
584
|
+
*/
|
|
585
|
+
export const uniqBy = (data, key) => {
|
|
586
|
+
if (!key || !data?.length || typeof get(data, '[0]') !== 'object') {
|
|
587
|
+
// Xử lý mảng chứa signal values
|
|
588
|
+
if (data[0] && isSignal(data[0])) {
|
|
589
|
+
const seen = new Set();
|
|
590
|
+
return data.filter((item) => {
|
|
591
|
+
const value = `${get(item, '')}`;
|
|
592
|
+
if (seen.has(value)) {
|
|
593
|
+
return false;
|
|
594
|
+
}
|
|
595
|
+
seen.add(value);
|
|
596
|
+
return true;
|
|
597
|
+
});
|
|
598
|
+
}
|
|
599
|
+
// Xử lý mảng primitive values
|
|
600
|
+
return Array.from(new Set(data));
|
|
601
|
+
}
|
|
602
|
+
const dataUnique = keyBy(data, key);
|
|
603
|
+
return Object.keys(dataUnique).map((key) => dataUnique[key]);
|
|
604
|
+
};
|
|
605
|
+
export const generateInterface = (obj, interfaceName) => {
|
|
606
|
+
const generateType = (value) => {
|
|
607
|
+
if (value === null) {
|
|
608
|
+
return 'null';
|
|
609
|
+
}
|
|
610
|
+
const type = typeof value;
|
|
611
|
+
if (type === 'string') {
|
|
612
|
+
return 'string';
|
|
613
|
+
}
|
|
614
|
+
if (type === 'number') {
|
|
615
|
+
return 'number';
|
|
616
|
+
}
|
|
617
|
+
if (type === 'boolean') {
|
|
618
|
+
return 'boolean';
|
|
619
|
+
}
|
|
620
|
+
if (type === 'undefined') {
|
|
621
|
+
return 'any';
|
|
622
|
+
}
|
|
623
|
+
if (value instanceof Date) {
|
|
624
|
+
return 'Date';
|
|
625
|
+
}
|
|
626
|
+
if (value instanceof RegExp) {
|
|
627
|
+
return 'RegExp';
|
|
628
|
+
}
|
|
629
|
+
if (Array.isArray(value)) {
|
|
630
|
+
if (value.length === 0) {
|
|
631
|
+
return 'Array<any>';
|
|
632
|
+
}
|
|
633
|
+
return `Array<${generateType(value[0])}>`;
|
|
634
|
+
}
|
|
635
|
+
if (type === 'object') {
|
|
636
|
+
let interfaceStr = '{\n';
|
|
637
|
+
for (const key in value) {
|
|
638
|
+
if (Object.prototype.hasOwnProperty.call(value, key)) {
|
|
639
|
+
const valueType = generateType(value[key]);
|
|
640
|
+
interfaceStr += ` ${key}: ${valueType};\n`;
|
|
641
|
+
}
|
|
642
|
+
}
|
|
643
|
+
interfaceStr += '}';
|
|
644
|
+
return interfaceStr;
|
|
645
|
+
}
|
|
646
|
+
return 'any';
|
|
647
|
+
};
|
|
648
|
+
const interfaceStr = `interface ${interfaceName} ${generateType(obj)}`;
|
|
649
|
+
return interfaceStr;
|
|
650
|
+
};
|
|
651
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGVscGVycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2xpYnMtdWkvdXRpbHMvc3JjL2hlbHBlcnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsdURBQXVEO0FBQ3ZELE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQztBQUNsRCxPQUFPLEVBQTBCLFFBQVEsRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFekUsT0FBTyxLQUFLLE1BQU0sT0FBTyxDQUFDO0FBQzFCLE9BQU8sRUFBRSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsYUFBYSxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsa0JBQWtCLEVBQUUsS0FBSyxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFDM0gsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLFFBQVEsQ0FBQztBQUNsQyxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDdkQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBRXZELHNDQUFzQztBQUN0Qzs7Ozs7Ozs7O0dBU0c7QUFDSCxNQUFNLENBQUMsTUFBTSxLQUFLLEdBQUcsQ0FBQyxLQUFjLEVBQUUsT0FBMEMsRUFBNkIsRUFBRTtJQUM3RyxJQUFJLENBQUMsT0FBTyxFQUFFLGtCQUFrQixFQUFFLENBQUM7UUFDakMsS0FBSyxHQUFHLFlBQVksQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUM5QixDQUFDO0lBQ0QsT0FBTyxLQUFLLEtBQUssSUFBSSxJQUFJLEtBQUssS0FBSyxTQUFTLENBQUM7QUFDL0MsQ0FBQyxDQUFDO0FBRUY7Ozs7Ozs7Ozs7OztHQVlHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLENBQUMsS0FBVSxFQUFFLE9BQTJHLEVBQXNDLEVBQUU7SUFDckwsSUFBSSxDQUFDLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxDQUFDO1FBQ2pDLEtBQUssR0FBRyxZQUFZ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QWEsYUFBYSxJQUFJLFlBQVksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDO0lBQ3ZFLE9BQU8sWUFBWSxDQUFDO0FBQ3RCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby1leHBsaWNpdC1hbnkgKi9cbmltcG9ydCB7IEh0dHBQYXJhbXMgfSBmcm9tICdAYW5ndWxhci9jb21tb24vaHR0cCc7XG5pbXBvcnQgeyBTaWduYWwsIFdyaXRhYmxlU2lnbmFsLCBpc1NpZ25hbCwgc2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBHZXRSZXR1cm5UeXBlLCBHZXRWYWx1ZUF0UGF0aCwgUGF0aE9mLCBUWVBFX09CSkVDVCB9IGZyb20gJ0BsaWJzLXVpL2ludGVyZmFjZXMtdHlwZXMnO1xuaW1wb3J0IGRheWpzIGZyb20gJ2RheWpzJztcbmltcG9ydCB7IGlzQXJyYXksIGlzRGFuZ2Vyb3VzT2JqZWN0LCBpc0RheWpzT2JqZWN0LCBpc01hcCwgaXNSZWdFeHAsIGlzUmV0dXJuQXNJc09iamVjdCwgaXNTZXQgfSBmcm9tICcuL2Rhbmdlcm91cy1vYmplY3QnO1xuaW1wb3J0IHsgZ2V0RGF5anMgfSBmcm9tICcuL2RhdGUnO1xuaW1wb3J0IHsgVXRpbHNIdHRwUGFyYW1zUmVxdWVzdCB9IGZyb20gJy4vaHR0cC1wYXJhbXMnO1xuaW1wb3J0IHsgdW53cmFwU2lnbmFsIH0gZnJvbSAnLi90d28td2F5LXNpZ25hbC1vYmplY3QnO1xuXG4vKipDw6FjIGjDoG0gdMawxqFuZyB04buxIHRoxrAgdmnhu4duIGxvZGFzaCAqL1xuLyoqXG4gKiBLaeG7g20gdHJhIHhlbSBt4buZdCBnacOhIHRy4buLIGPDsyBwaOG6o2kgbMOgIG51bGwgaG/hurdjIHVuZGVmaW5lZCBoYXkga2jDtG5nXG4gKiBAcGFyYW0gdmFsdWUgR2nDoSB0cuG7iyBj4bqnbiBraeG7g20gdHJhXG4gKiBAcmV0dXJucyB0cnVlIG7hur91IGdpw6EgdHLhu4sgbMOgIG51bGwgaG/hurdjIHVuZGVmaW5lZCwgZmFsc2UgbuG6v3Uga2jDtG5nXG4gKiBAZXhhbXBsZVxuICogaXNOaWwobnVsbCk7IC8vIHRydWVcbiAqIGlzTmlsKHVuZGVmaW5lZCk7IC8vIHRydWVcbiAqIGlzTmlsKDApOyAvLyBmYWxzZVxuICogaXNOaWwoJ2hlbGxvJyk7IC8vIGZhbHNlXG4gKi9cbmV4cG9ydCBjb25zdCBpc05pbCA9ICh2YWx1ZTogdW5rbm93biwgb3B0aW9ucz86IHsgaWdub3JlVW5XcmFwU2lnbmFsPzogYm9vbGVhbiB9KTogdmFsdWUgaXMgbnVsbCB8IHVuZGVmaW5lZCA9PiB7XG4gIGlmICghb3B0aW9ucz8uaWdub3JlVW5XcmFwU2lnbmFsKSB7XG4gICAgdmFsdWUgPSB1bndyYXBTaWduYWwodmFsdWUpO1xuICB9XG4gIHJldHVybiB2YWx1ZSA9PT0gbnVsbCB8fCB2YWx1ZSA9PT0gdW5kZWZpbmVkO1xufTtcblxuLyoqXG4gKiBLaeG7g20gdHJhIHhlbSBt4buZdCBnacOhIHRy4buLIGPDsyBwaOG6o2kgbMOgIHLhu5duZyBoYXkga2jDtG5nXG4gKiBAcGFyYW0gdmFsdWUgR2nDoSB0cuG7iyBj4bqnbiBraeG7g20gdHJhXG4gKiBAcmV0dXJucyB0cnVlIG7hur91IGdpw6EgdHLhu4sgbMOgIG51bGwsIHLhu5duZyBob+G6t2MgdW5kZWZpbmVkLCBmYWxzZSBu4bq/dSBraMO0bmdcbiAqIEBleGFtcGxlXG4gKiBpc0VtcHR5KG51bGwpOyAvLyB0cnVlXG4gKiBpc0VtcHR5KCcnKTsgLy8gdHJ1ZVxuICogaXNFbXB0eSh1bmRlZmluZWQpOyAvLyB0cnVlXG4gKiBpc0VtcHR5KHt9KTsgLy8gdHJ1ZVxuICogaXNFbXB0eShbXSk7IC8vIHRydWVcbiAqIGlzRW1wdHkoWzEsIDIsIDNdKTsgLy8gZmFsc2VcbiAqIGlzRW1wdHkoeyBhOiAxIH0pOyAvLyBmYWxzZVxuICovXG5leHBvcnQgY29uc3QgaXNFbXB0eSA9ICh2YWx1ZTogYW55LCBvcHRpb25zPzogeyBpZ25vcmVDaGVja1R5cGVQcmltaXRpdmU/OiBib29sZWFuOyBpZ25vcmVVbldyYXBTaWduYWw/OiBib29sZWFuOyBpZ25vcmVDaGVja1N0cmluZz86IGJvb2xlYW4gfSk6IHZhbHVlIGlzIG51bGwgfCB1bmRlZmluZWQgfCAnJyB8IDAgPT4ge1xuICBpZiAoIW9wdGlvbnM/Lmlnbm9yZVVuV3JhcFNpZ25hbCkge1xuICAgIHZhbHVlID0gdW53cmFwU2lnbmFsKHZhbHVlKTtcbiAgfVxuICBpZiAob3B0aW9ucz8uaWdub3JlQ2hlY2tTdHJpbmcgJiYgdmFsdWUgPT09ICcnKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG4gIGlmIChvcHRpb25zPy5pZ25vcmVDaGVja1R5cGVQcmltaXRpdmUpIHtcbiAgICByZXR1cm4gdHlwZW9mIHZhbHVlID09PSAnb2JqZWN0JyAmJiBpc0VtcHR5VHlwZU9iamVjdCh2YWx1ZSk7XG4gIH1cblxuICByZXR1cm4gaXNFbXB0eVR5cGVPYmplY3QodmFsdWUpO1xufTtcblxuLyoqXG4gKiBLaeG7g20gdHJhIHhlbSBt4buZdCBnacOhIHRy4buLIGPDsyBwaOG6o2kgbMOgIG9iamVjdCBy4buXbmcgaGF5IGtow7RuZ1xuICogQHBhcmFtIHZhbHVlIEdpw6EgdHLhu4sgY+G6p24ga2nhu4NtIHRyYVxuICogQHJldHVybnMgdHJ1ZSBu4bq/dSBnacOhIHRy4buLIGzDoCBvYmplY3QgcuG7l25nLCBmYWxzZSBu4bq/dSBraMO0bmdcbiAqIEBleGFtcGxlXG4gKiBpc0VtcHR5VHlwZU9iamVjdCh7fSk7IC8vIHRydWVcbiAqIGlzRW1wdHlUeXBlT2JqZWN0KHsgYTogMSB9KTsgLy8gZmFsc2VcbiAqL1xuY29uc3QgaXNFbXB0eVR5cGVPYmplY3QgPSA8VCA9IG51bGw+KHZhbHVlOiBhbnkpOiB2YWx1ZSBpcyBUID0+IHtcbiAgdHJ5IHtcbiAgICBpZiAoaXNOaWwodmFsdWUsIHsgaWdub3JlVW5XcmFwU2lnbmFsOiB0cnVlIH0pIHx8IHZhbHVlID09PSAnJykge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgdmFsdWUgIT09ICdvYmplY3QnKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSB7XG4gICAgICByZXR1cm4gdmFsdWUubGVuZ3RoID09PSAwO1xuICAgIH1cblxuICAgIHJldHVybiBPYmplY3Qua2V5cyh2YWx1ZSkubGVuZ3RoID09PSAwO1xuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxufTtcblxuLyoqXG4gKiBLaeG7g20gdHJhIHhlbSBt4buZdCBnacOhIHRy4buLIGPDsyBwaOG6o2kgbMOgIG51bGwsIHLhu5duZywgdW5kZWZpbmVkIGhv4bq3YyAwIGhheSBraMO0bmdcbiAqIEBwYXJhbSB2YWx1ZSBHacOhIHRy4buLIGPhuqduIGtp4buDbSB0cmFcbiAqIEBwYXJhbSBvcHRpb25zIEPhuqV1IGjDrG5oIHTDuXkgY2jhu41uXG4gKiBAcGFyYW0gb3B0aW9ucy5pZ25vcmVaZXJvIE7hur91IHRydWUsIHPhur0ga2jDtG5nIGtp4buDbSB0cmEgZ2nDoSB0cuG7iyAwXG4gKiBAcmV0dXJucyB0cnVlIG7hur91IGdpw6EgdHLhu4sgbMOgIG51bGwsIHLhu5duZywgdW5kZWZpbmVkIGhv4bq3YyAwLCBmYWxzZSBu4bq/dSBraMO0bmdcbiAqIEBleGFtcGxlXG4gKiBpc1RydXRoeShudWxsKTsgLy8gZmFsc2VcbiAqIGlzVHJ1dGh5KCcnKTsgLy8gZmFsc2VcbiAqIGlzVHJ1dGh5KHVuZGVmaW5lZCk7IC8vIGZhbHNlXG4gKiBpc1RydXRoeSgwKTsgLy8gZmFsc2VcbiAqIGlzVHJ1dGh5KHt9KTsgLy8gdHJ1ZVxuICogaXNUcnV0aHkoMCwgeyBpZ25vcmVaZXJvOiB0cnVlIH0pOyAvLyB0cnVlXG4gKi9cblxuZXhwb3J0IGNvbnN0IGlzVHJ1dGh5ID0gPFQ+KHZhbHVlOiBULCBvcHRpb25zPzogeyBpZ25vcmVaZXJvPzogYm9vbGVhbjsgaWdub3JlQ2hlY2tTdHJpbmc/OiBib29sZWFuOyBpZ25vcmVVbldyYXBTaWduYWw/OiBib29sZWFuIH0pOiB2YWx1ZSBpcyBOb25OdWxsYWJsZTxUPiA9PiB7XG4gIHJldHVybiAhaXNGYWxzeSh2YWx1ZSwgb3B0aW9ucyk7XG59O1xuXG4vKipcbiAqIEtp4buDbSB0cmEgeGVtIG3hu5l0IGdpw6EgdHLhu4sgY8OzIHBo4bqjaSBsw6AgbnVsbCwgcuG7l25nLCB1bmRlZmluZWQgaG/hurdjIDAgaGF5IGtow7RuZ1xuICogQHBhcmFtIHZhbHVlIEdpw6EgdHLhu4sgY+G6p24ga2nhu4NtIHRyYVxuICogQHBhcmFtIG9wdGlvbnMgQ+G6pXUgaMOsbmggdMO5eSBjaOG7jW5cbiAqIEBwYXJhbSBvcHRpb25zLmlnbm9yZVplcm8gTuG6v3UgdHJ1ZSwgc+G6vSBraMO0bmcga2nhu4NtIHRyYSBnacOhIHRy4buLIDBcbiAqIEByZXR1cm5zIHRydWUgbuG6v3UgZ2nDoSB0cuG7iyBsw6AgbnVsbCwgcuG7l25nLCB1bmRlZmluZWQgaG/hurdjIDAsIGZhbHNlIG7hur91IGtow7RuZ1xuICogQGV4YW1wbGVcbiAqIGlzRmFsc3kobnVsbCk7IC8vIHRydWVcbiAqIGlzRmFsc3koJycpOyAvLyB0cnVlXG4gKiBpc0ZhbHN5KHVuZGVmaW5lZCk7IC8vIHRydWVcbiAqIGlzRmFsc3koMCk7IC8vIHRydWVcbiAqIGlzRmFsc3koe30pOyAvLyBmYWxzZVxuICogaXNGYWxzeSgwLCB7IGlnbm9yZVplcm86IHRydWUgfSk7IC8vIGZhbHNlXG4gKi9cbmV4cG9ydCBjb25zdCBpc0ZhbHN5ID0gKHZhbHVlOiB1bmtub3duLCBvcHRpb25zPzogeyBpZ25vcmVaZXJvPzogYm9vbGVhbjsgaWdub3JlQ2hlY2tTdHJpbmc/OiBib29sZWFuOyBpZ25vcmVVbldyYXBTaWduYWw/OiBib29sZWFuIH0pOiB2YWx1ZSBpcyBudWxsIHwgdW5kZWZpbmVkIHwgJycgfCAwID0+IHtcbiAgaWYgKCFvcHRpb25zPy5pZ25vcmVVbldyYXBTaWduYWwpIHtcbiAgICB2YWx1ZSA9IHVud3JhcFNpZ25hbCh2YWx1ZSk7XG4gIH1cblxuICBpZiAob3B0aW9ucz8uaWdub3JlWmVybyAmJiB2YWx1ZSA9PT0gMCkge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIGlmIChvcHRpb25zPy5pZ25vcmVDaGVja1N0cmluZyAmJiB2YWx1ZSA9PT0gJycpIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICByZXR1cm4gIXZhbHVlO1xufTtcblxuLyoqXG4gKiBMb+G6oWkgYuG7jyBjw6FjIHRodeG7mWMgdMOtbmggY+G7p2EgxJHhu5FpIHTGsOG7o25nIGThu7FhIHRyw6puIG3hu5l0IGjDoG0gxJFp4buBdSBraeG7h25cbiAqIEBwYXJhbSBvYmpEYXRhIMSQ4buRaSB0xrDhu6NuZyBj4bqnbiB44butIGzDvVxuICogQHBhcmFtIHByZWRpY2F0ZSBIw6BtIMSRaeG7gXUga2nhu4duIMSR4buDIGtp4buDbSB0cmEgZ2nDoSB0cuG7iyBj4bunYSB0aHXhu5ljIHTDrW5oLiBO4bq/dSBow6BtIHRy4bqjIHbhu4EgdHJ1ZSB0aMOsIHRodeG7mWMgdMOtbmggxJHDsyBz4bq9IGLhu4sgbG/huqFpIGLhu49cbiAqIEByZXR1cm5zIMSQ4buRaSB0xrDhu6NuZyBt4bubaSBzYXUga2hpIMSRw6MgbG/huqFpIGLhu48gY8OhYyB0aHXhu5ljIHTDrW5oIHRo4buPYSBtw6NuIMSRaeG7gXUga2nhu4duXG4gKiBAZXhhbXBsZVxuICogY29uc3Qgb2JqID0geyBhOiAxLCBiOiBudWxsLCBjOiAzLCBkOiB1bmRlZmluZWQgfTtcbiAqIG9taXRCeShvYmosIGlzTmlsKTsgLy8geyBhOiAxLCBjOiAzIH1cbiAqL1xuZXhwb3J0IGNvbnN0IG9taXRCeSA9IDxUPihvYmpEYXRhOiBSZWNvcmQ8c3RyaW5nLCBhbnk+LCBwcmVkaWNhdGU6ICh2YWw6IGFueSkgPT4gYm9vbGVhbik6IFQgPT4ge1xuICBpZiAoIW9iakRhdGEgfHwgdHlwZW9mIG9iakRhdGEgIT09ICdvYmplY3QnIHx8IEFycmF5LmlzQXJyYXkob2JqRGF0YSkpIHtcbiAgICByZXR1cm4gb2JqRGF0YSBhcyBUO1xuICB9XG4gIGNvbnN0IG5ld09iaiA9IHt9IGFzIFQ7XG5cbiAgT2JqZWN0LmtleXMob2JqRGF0YSkuZm9yRWFjaCgoa2V5KSA9PiB7XG4gICAgY29uc3QgdmFsdWVPZktleSA9IGdldChvYmpEYXRhLCBrZXkpO1xuICAgIGlmICghcHJlZGljYXRlKHZhbHVlT2ZLZXkpKSB7XG4gICAgICBzZXQobmV3T2JqLCBrZXkgYXMgUGF0aE9mPFQ+LCB2YWx1ZU9mS2V5KTtcbiAgICB9XG4gIH0pO1xuICByZXR1cm4gbmV3T2JqO1xufTtcblxuLyoqXG4gKiBM4bqleSBnacOhIHRy4buLIHThu6sgxJHhu5FpIHTGsOG7o25nIHRoZW8gxJHGsOG7nW5nIGThuqtuIGNo4buJIMSR4buLbmhcbiAqXG4gKiBIw6BtIG7DoHkgZ2nDunAgYuG6oW4gdHJ1eSBj4bqtcCB2w6BvIGPDoWMgdGh14buZYyB0w61uaCBzw6J1IGLDqm4gdHJvbmcgbeG7mXQgxJHhu5FpIHTGsOG7o25nIG3hu5l0IGPDoWNoIGFuIHRvw6BuLFxuICogdHLDoW5oIGzhu5dpIGtoaSB0aHXhu5ljIHTDrW5oIGtow7RuZyB04buTbiB04bqhaS5cbiAqXG4gKiBAcGFyYW0gb2JqIMSQ4buRaSB0xrDhu6NuZyBuZ3Xhu5NuIGPhuqduIGzhuqV5IGdpw6EgdHLhu4tcbiAqIEBwYXJhbSBwYXRoIMSQxrDhu51uZyBk4bqrbiDEkeG6v24gdGh14buZYyB0w61uaCBj4bqnbiBs4bqleS4gQ8OzIHRo4buDIGzDoDpcbiAqICAgLSBDaHXhu5dpOiAndXNlci5wcm9maWxlLm5hbWUnIGhv4bq3YyAnaXRlbXNbMF0udGl0bGUnXG4gKiAgIC0gTeG6o25nOiBbJ3VzZXInLCAncHJvZmlsZScsICduYW1lJ10gaG/hurdjIFsnaXRlbXMnLCAnMCcsICd0aXRsZSddXG4gKiAgIC0gQ2h14buXaSBy4buXbmcgJyc6IHRy4bqjIHbhu4EgY2jDrW5oIMSR4buRaSB0xrDhu6NuZyBn4buRY1xuICogQHBhcmFtIGRlZmF1bHRWYWx1ZSBHacOhIHRy4buLIG3hurdjIMSR4buLbmggdHLhuqMgduG7gSBraGkga2jDtG5nIHTDrG0gdGjhuqV5IHRodeG7mWMgdMOtbmggKG3hurdjIMSR4buLbmg6IHVuZGVmaW5lZClcbiAqIEBwYXJhbSBrZWVwTGFzdFZhbHVlSWZTaWduYWwgQ8OzIGdp4buvIG5ndXnDqm4gc2lnbmFsIGN14buRaSBjw7luZyBoYXkga2jDtG5nICht4bq3YyDEkeG7i25oOiBmYWxzZSAtIHPhur0gZ+G7jWkgc2lnbmFsKCkpXG4gKiBAcmV0dXJucyBHacOhIHRy4buLIHTDrG0gxJHGsOG7o2MgaG/hurdjIGdpw6EgdHLhu4sgbeG6t2MgxJHhu4tuaFxuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBWw60gZOG7pSBjxqEgYuG6o25cbiAqIGNvbnN0IHVzZXIgPSB7IG5hbWU6ICdKb2huJywgYWdlOiAzMCB9O1xuICogZ2V0KHVzZXIsICduYW1lJyk7IC8vICdKb2huJ1xuICogZ2V0KHVzZXIsICdlbWFpbCcpOyAvLyB1bmRlZmluZWRcbiAqIGdldCh1c2VyLCAnZW1haWwnLCAnbm8tZW1haWwnKTsgLy8gJ25vLWVtYWlsJ1xuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBUcnV54buBbiBwYXRoIHLhu5duZyAtIHRy4bqjIHbhu4EgY2jDrW5oIMSR4buRaSB0xrDhu6NuZyBn4buRY1xuICogY29uc3QgZGF0YSA9IHsgbmFtZTogJ0FsaWNlJywgYWdlOiAyNSB9O1xuICogZ2V0KGRhdGEsICcnKTsgLy8geyBuYW1lOiAnQWxpY2UnLCBhZ2U6IDI1IH0gKGNow61uaCDEkeG7kWkgdMaw4bujbmcgZGF0YSlcbiAqIGdldChkYXRhLCAnJywgJ2RlZmF1bHQnKTsgLy8geyBuYW1lOiAnQWxpY2UnLCBhZ2U6IDI1IH0gKGLhu48gcXVhIGRlZmF1bHRWYWx1ZSlcbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gVHJ1eSBj4bqtcCB0aHXhu5ljIHTDrW5oIHPDonVcbiAqIGNvbnN0IGRhdGEgPSB7XG4gKiAgIHVzZXI6IHtcbiAqICAgICBwcm9maWxlOiB7XG4gKiAgICAgICBuYW1lOiAnQWxpY2UnLFxuICogICAgICAgc2V0dGluZ3M6IHsgdGhlbWU6ICdkYXJrJyB9XG4gKiAgICAgfVxuICogICB9XG4gKiB9O1xuICogZ2V0KGRhdGEsICd1c2VyLnByb2ZpbGUubmFtZScpOyAvLyAnQWxpY2UnXG4gKiBnZXQoZGF0YSwgJ3VzZXIucHJvZmlsZS5zZXR0aW5ncy50aGVtZScpOyAvLyAnZGFyaydcbiAqIGdldChkYXRhLCAndXNlci5wcm9maWxlLmF2YXRhcicsICdkZWZhdWx0LmpwZycpOyAvLyAnZGVmYXVsdC5qcGcnXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIFRydXkgY+G6rXAgbeG6o25nXG4gKiBjb25zdCBpdGVtcyA9IFtcbiAqICAgeyBuYW1lOiAnSXRlbSAxJywgcHJpY2U6IDEwMCB9LFxuICogICB7IG5hbWU6ICdJdGVtIDInLCBwcmljZTogMjAwIH1cbiAqIF07XG4gKiBnZXQoaXRlbXMsICdbMF0ubmFtZScpOyAvLyAnSXRlbSAxJ1xuICogZ2V0KGl0ZW1zLCAnWzFdLm5hbWUnKTsgLy8gJ0l0ZW0gMidcbiAqIGdldChpdGVtcywgJ1syXS5uYW1lJywgJ05vdCBmb3VuZCcpOyAvLyAnTm90IGZvdW5kJ1xuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBT4butIGThu6VuZyB24bubaSBt4bqjbmcgcGF0aFxuICogY29uc3QgbmVzdGVkID0geyBhOiB7IGI6IHsgYzogJ2RlZXAgdmFsdWUnIH0gfSB9O1xuICogZ2V0KG5lc3RlZCwgWydhJywgJ2InLCAnYyddKTsgLy8gJ2RlZXAgdmFsdWUnXG4gKiBnZXQobmVzdGVkLCBbJ2EnLCAnYicsICdkJ10sICdkZWZhdWx0Jyk7IC8vICdkZWZhdWx0J1xuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBUcsaw4budbmcgaOG7o3AgxJHhurdjIGJp4buHdFxuICogZ2V0KG51bGwsICdhbnkucGF0aCcpOyAvLyB1bmRlZmluZWRcbiAqIGdldCh1bmRlZmluZWQsICdhbnkucGF0aCcsICdmYWxsYmFjaycpOyAvLyAnZmFsbGJhY2snXG4gKi9cbmV4cG9ydCBjb25zdCBnZXQgPSA8TywgUCBleHRlbmRzIFBhdGhPZjxPPiA9IFBhdGhPZjxPPiwgS1MgZXh0ZW5kcyBib29sZWFuID0gZmFsc2UsIEQgZXh0ZW5kcyBHZXRWYWx1ZUF0UGF0aDxPLCBQLCBLUz4gfCB1bmRlZmluZWQgPSB1bmRlZmluZWQ+KFxuICBvYmo6IFNpZ25hbDxPPiB8IE8sXG4gIHBhdGg6IFAsXG4gIGRlZmF1bHRWYWx1ZT86IEQsXG4gIGtlZXBMYXN0VmFsdWVJZlNpZ25hbD86IEtTXG4pOiBHZXRSZXR1cm5UeXBlPE8sIFAsIEtTLCBEPiA9PiB7XG4gIC8vIGhlbHBlciBjYXN0IMSR4buDIHRyw6FuaCBs4bq3cCBs4bqhaSBraeG7g3UgxJFp4buBdSBraeG7h24g4bufIGPDoWMgcmV0dXJuXG4gIGNvbnN0IG91dCA9ICh2OiB1bmtub3duKSA9PiB2IGFzIEdldFJldHVyblR5cGU8TywgUCwgS1MsIEQ+O1xuXG4gIGlmIChpc05pbChvYmopKSB7XG4gICAgcmV0dXJuIG91dChkZWZhdWx0VmFsdWUpO1xuICB9XG4gIG9iaiA9IHVud3JhcFNpZ25hbChvYmopO1xuICBpZiAocGF0aCA9PT0gJycpIHtcbiAgICByZXR1cm4gb3V0KG9iaik7XG4gIH1cbiAgaWYgKG9iaiBpbnN0YW5jZW9mIEh0dHBQYXJhbXMpIHtcbiAgICByZXR1cm4gb3V0KG9iai5nZXQoYCR7cGF0aH1gKSA/PyBkZWZhdWx0VmFsdWUpO1xuICB9XG4gIGlmIChvYmogaW5zdGFuY2VvZiBET01SZWN0KSB7XG4gICAgcmV0dXJuIG91dChvYmpbcGF0aCBhcyBrZXlvZiBET01SZWN0XSk7XG4gIH1cblxuICBjb25zdCBwYXRocyA9IEFycmF5LmlzQXJyYXkocGF0aClcbiAgICA/IHBhdGhcbiAgICA6IChwYXRoIGFzIHN0cmluZylcbiAgICAgICAgLnJlcGxhY2UoL1xcWyhcXGQrKV0vZywgJy4kMScpXG4gICAgICAgIC5zcGxpdCgnLicpXG4gICAgICAgIC5maWx0ZXIoKGtleSkgPT4ga2V5KTtcbiAgZm9yIChjb25zdCBpbmRleCBpbiBwYXRocykge1xuICAgIGNvbnN0IGtleSA9IHBhdGhzW2luZGV4XTtcbiAgICBpZiAob2JqIGluc3RhbmNlb2YgQ1NTU3R5bGVEZWNsYXJhdGlvbikge1xuICAgICAgb2JqID0gb2JqW2tleSBhcyBrZXlvZiBDU1NTdHlsZURlY2xhcmF0aW9uXSBhcyBPO1xuICAgICAgY29udGludWU7XG4gICAgfVxuXG4gICAgaWYgKG9iaiBpbnN0YW5jZW9mIEhUTUxFbGVtZW50KSB7XG4gICAgICBvYmogPSBvYmpba2V5IGFzIGtleW9mIEhUTUxFbGVtZW50XSBhcyBPO1xuICAgICAgY29udGludWU7XG4gICAgfVxuICAgIGlmIChpc05pbChvYmopIHx8ICFPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwob2JqLCBrZXkpKSB7XG4gICAgICByZXR1cm4gb3V0KGRlZmF1bHRWYWx1ZSk7XG4gICAgfVxuXG4gICAgY29uc3QgdmFsID0gaXNTaWduYWwoKG9iaiBhcyBhbnkpW2tleV0pICYmICFrZWVwTGFzdFZhbHVlSWZTaWduYWwgPyAob2JqIGFzIGFueSlba2V5XSgpIDogKG9iaiBhcyBhbnkpW2tleV07XG5cbiAgICBvYmogPSB2YWwgYXMgTztcbiAgfVxuXG4gIHJldHVybiBvdXQob2JqID8/IGRlZmF1bHRWYWx1ZSk7XG59O1xuXG4vKipcbiAqIFRoaeG6v3QgbOG6rXAgZ2nDoSB0cuG7iyBjaG8gbeG7mXQgdGh14buZYyB0w61uaCB0cm9uZyDEkeG7kWkgdMaw4bujbmcgdGhlbyDEkcaw4budbmcgZOG6q24gY2jhu4kgxJHhu4tuaFxuICogQHBhcmFtIG9iaiDEkOG7kWkgdMaw4bujbmcgY+G6p24gdGhp4bq/dCBs4bqtcCBnacOhIHRy4buLXG4gKiBAcGFyYW0gcGF0aCDEkMaw4budbmcgZOG6q24gxJHhur9uIHRodeG7mWMgdMOtbmgsIGPDsyB0aOG7gyBsw6AgY2h14buXaSAodmQ6ICdhLmIuYycpIGhv4bq3YyBt4bqjbmcgKHZkOiBbJ2EnLCAnYicsICdjJ10pXG4gKiBAcGFyYW0gdmFsdWUgR2nDoSB0cuG7iyBj4bqnbiB0aGnhur90IGzhuq1wXG4gKiBAcmV0dXJucyDEkOG7kWkgdMaw4bujbmcgc2F1IGtoaSDEkcOjIHRoaeG6v3QgbOG6rXAgZ2nDoSB0cuG7i1xuICogQHRocm93cyBFcnJvciBu4bq/dSB0aGFtIHPhu5EgxJHhuqd1IHRpw6puIGtow7RuZyBwaOG6o2kgbMOgIMSR4buRaSB0xrDhu6NuZ1xuICogQGV4YW1wbGVcbiAqIGNvbnN0IG9iaiA9IHsgYTogeyBiOiAxIH0gfTtcbiAqIHNldChvYmosICdhLmInLCAyKTsgLy8geyBhOiB7IGI6IDIgfSB9XG4gKi9cbmV4cG9ydCBjb25zdCBzZXQgPSA8TywgUCBleHRlbmRzIFBhdGhPZjxPPiA9IFBhdGhPZjxPPiwgVCBleHRlbmRzIEdldFZhbHVlQXRQYXRoPE8sIFAsIHRydWU+ID0gR2V0VmFsdWVBdFBhdGg8TywgUCwgdHJ1ZT4+KFxuICBvYmo6IFNpZ25hbDxPPiB8IE8sXG4gIHBhdGg6IFAsXG4gIHZhbHVlOiBULFxuICBvcHRpb25zPzogeyB2YWx1ZURlZmF1bHRQYXRoT2JqZWN0VW5kZWZpbmVkPzogYW55OyB2YWx1ZURlZmF1bHRQYXRoQXJyYXlVbmRlZmluZWQ/OiBhbnkgfVxuKTogTyA9PiB7XG4gIGlmICghb2JqIHx8ICh0eXBlb2Ygb2JqICE9PSAnb2JqZWN0JyAmJiAhaXNTaWduYWwob2JqKSkgfHwgKGlzU2lnbmFsKG9iaikgJiYgdHlwZW9mIG9iaigpICE9PSAnb2JqZWN0JykpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ1RoZSBmaXJzdCBhcmd1bWVudCBtdXN0IGJlIGFuIG9iamVjdCcpO1xuICB9XG5cbiAgaWYgKG9iaiBpbnN0YW5jZW9mIEh0dHBQYXJhbXMpIHtcbiAgICByZXR1cm4gb2JqLnNldChgJHtwYXRofWAsIHZhbHVlIGFzIHN0cmluZykgYXMgTztcbiAgfVxuXG4gIGNvbnN0IHBhdGhBcnJheSA9IEFycmF5LmlzQXJyYXkocGF0aClcbiAgICA/IHBhdGhcbiAgICA6IChwYXRoIGFzIHN0cmluZylcbiAgICAgICAgLnJlcGxhY2UoL1xcWyhcXGQrKV0vZywgJy5bJDFdJylcbiAgICAgICAgLnNwbGl0KCcuJylcbiAgICAgICAgLmZpbHRlcigoa2V5KSA9PiBrZXkpO1xuXG4gIGxldCBjdXJyZW50T2JqZWN0QnlLZXk6IGFueSA9IGlzU2lnbmFsKG9iaikgPyBvYmooKSA6IG9iajtcbiAgbGV0IHByZU9iamVjdEJ5S2V5ID0gb2JqO1xuXG4gIHBhdGhBcnJheS5mb3JFYWNoKChrZXksIGluZGV4KSA9PiB7XG4gICAgaWYgKGluZGV4IDwgcGF0aEFycmF5Lmxlbmd0aCAtIDEpIHtcbiAgICAgIGlmICghKGtleSBpbiBjdXJyZW50T2JqZWN0QnlLZXkpIHx8ICh0eXBlb2YgY3VycmVudE9iamVjdEJ5S2V5W2tleV0gIT09ICdvYmplY3QnICYmICFpc1NpZ25hbChjdXJyZW50T2JqZWN0QnlLZXlba2V5XSkpKSB7XG4gICAgICAgIGNvbnN0IG5leHRLZXkgPSBwYXRoQXJyYXlbaW5kZXggKyAxXTtcbiAgICAgICAga2V5ID0ga2V5LnJlcGxhY2UoL1xcWyhcXGQrKV0vZywgJyQxJyk7XG4gICAgICAgIGlmIChpc05pbChjdXJyZW50T2JqZWN0QnlLZXlba2V5XSkpIHtcbiAgICAgICAgICBjdXJyZW50T2JqZWN0QnlLZXlba2V5XSA9IC9cXFsoXFxkKyldL2cudGVzdChuZXh0S2V5KSA/IChvcHRpb25zPy52YWx1ZURlZmF1bHRQYXRoQXJyYXlVbmRlZmluZWQgPz8gW10pIDogKG9wdGlvbnM/LnZhbHVlRGVmYXVsdFBhdGhPYmplY3RVbmRlZmluZWQgPz8ge30pO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBjdXJyZW50T2JqZWN0QnlLZXkgPSBrZXkgPyBjdXJyZW50T2JqZWN0QnlLZXlba2V5XSA6IGN1cnJlbnRPYmplY3RCeUtleTtcbiAgICAgIHByZU9iamVjdEJ5S2V5ID0gY3VycmVudE9iamVjdEJ5S2V5O1xuICAgICAgY3VycmVudE9iamVjdEJ5S2V5ID0gaXNTaWduYWwoY3VycmVudE9iamVjdEJ5S2V5KSA/IGN1cnJlbnRPYmplY3RCeUtleSgpIDogY3VycmVudE9iamVjdEJ5S2V5O1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmICh0eXBlb2YgY3VycmVudE9iamVjdEJ5S2V5ICE9PSAnb2JqZWN0Jykge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIC8vIEfDoW4gZ2nDoSB0cuG7iyDhu58gY3Xhu5FpIMSRxrDhu51uZyBk4bqrblxuICAgIGtleSA9IGtleS5yZXBsYWNlKC9cXFsoXFxkKyldL2csICckMScpO1xuICAgIGNvbnN0IHZhbHVlT2ZLZXkgPSBjdXJyZW50T2JqZWN0QnlLZXlba2V5XTtcbiAgICBjb25zdCB2YWx1ZU9mS2V5SXNTaWduYWwgPSBpc1NpZ25hbCh2YWx1ZU9mS2V5KTtcblxuICAgIGlmICh2YWx1ZU9mS2V5SXNTaWduYWwgJiYgKHR5cGVvZiB2YWx1ZU9mS2V5KCkgIT09ICdvYmplY3QnIHx8IHZhbHVlT2ZLZXkoKSA9PT0gbnVsbCkpIHtcbiAgICAgICh2YWx1ZU9mS2V5IGFzIFdyaXRhYmxlU2lnbmFsPGFueT4pLnNldChpc1NpZ25hbCh2YWx1ZSkgPyB2YWx1ZSgpIDogdmFsdWUpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBpZiAoaXNTaWduYWwocHJlT2JqZWN0QnlLZXkpKSB7XG4gICAgICAocHJlT2JqZWN0QnlLZXkgYXMgV3JpdGFibGVTaWduYWw8YW55PikudXBkYXRlKChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgY29uc3QgZGF0YU9mS2V5ID0gZGF0YVtrZXldO1xuICAgICAgICBpZiAoaXNOaWwoZGF0YU9mS2V5KSB8fCAhdmFsdWVPZktleUlzU2lnbmFsKSB7XG4gICAgICAgICAgZGF0YVtrZXldID0gdmFsdWU7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAodmFsdWVPZktleUlzU2lnbmFsKSB7XG4gICAgICAgICAgKHZhbHVlT2ZLZXkgYXMgV3JpdGFibGVTaWduYWw8YW55Pikuc2V0KGlzU2lnbmFsKHZhbHVlKSA/IHZhbHVlKCkgOiB2YWx1ZSk7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAoQXJyYXkuaXNBcnJheShkYXRhKSkge1xuICAgICAgICAgIHJldHVybiBbLi4uZGF0YV07XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHsgLi4uZGF0YSB9O1xuICAgICAgfSk7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGlmICh2YWx1ZU9mS2V5SXNTaWduYWwpIHtcbiAgICAgICh2YWx1ZU9mS2V5IGFzIFdyaXRhYmxlU2lnbmFsPGFueT4pLnNldChpc1NpZ25hbCh2YWx1ZSkgPyB2YWx1ZSgpIDogdmFsdWUpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBjdXJyZW50T2JqZWN0QnlLZXlba2V5XSA9IHZhbHVlO1xuICB9KTtcblxuICByZXR1cm4gb2JqIGFzIE87XG59O1xuXG4vKipcbiAqIFThuqFvIG3hu5l0IGLhuqNuIHNhbyBzw6J1IGPhu6dhIG3hu5l0IMSR4buRaSB0xrDhu6NuZyBob+G6t2MgZ2nDoSB0cuG7iyBi4bqldCBr4buzXG4gKiBAcGFyYW0gZGF0YSBE4buvIGxp4buHdSBj4bqnbiBzYW8gY2jDqXBcbiAqIEBwYXJhbSBvcHRpb25zIFTDuXkgY2jhu41uIGPhuqV1IGjDrG5oXG4gKiBAcGFyYW0gb3B0aW9ucy5pZ25vcmVTaWduYWwgTuG6v3UgdHJ1ZSwgc+G6vSBraMO0bmcgc2FvIGNow6lwIGPDoWMgc2lnbmFsIG3DoCB0cuG6oyB24buBIHNpZ25hbCBn4buRY1xuICogQHBhcmFtIHNlZW4gV2Vha01hcCDEkeG7gyB0aGVvIGTDtWkgY8OhYyB0aGFtIGNoaeG6v3UgxJHDoyDEkcaw4bujYyBzYW8gY2jDqXAsIHRyw6FuaCBs4bq3cCB2w7QgaOG6oW4gduG7m2kgY8OhYyB0aGFtIGNoaeG6v3UgdsOybmdcbiAqIEByZXR1cm5zIELhuqNuIHNhbyBzw6J1IGPhu6dhIGThu68gbGnhu4d1IMSR4bqndSB2w6BvXG4gKiBAZXhhbXBsZVxuICogY29uc3Qgb2JqID0ge1xuICogICBhOiAxLFxuICogICBiOiB7IGM6IDIgfSxcbiAqICAgZDogWzEsIDIsIDNdXG4gKiB9O1xuICogY29uc3QgY2xvbmUgPSBjbG9uZURlZXAob2JqKTtcbiAqIC8vIGNsb25lIGzDoCBt4buZdCBi4bqjbiBzYW8gaG/DoG4gdG/DoG4gxJHhu5ljIGzhuq1wIGPhu6dhIG9ialxuICovXG5leHBvcnQgY29uc3QgY2xvbmVEZWVwID0gPFQgPSBhbnk+KGRhdGE6IGFueSwgb3B0aW9ucyA9IHsgaWdub3JlU2lnbmFsOiBmYWxzZSB9LCBzZWVuID0gbmV3IFdlYWtNYXAoKSk6IFQgPT4ge1xuICBpZiAoZGF0YSA9PT0gbnVsbCB8fCAodHlwZW9mIGRhdGEgIT09ICdvYmplY3QnICYmICFpc1NpZ25hbChkYXRhKSkgfHwgaXNEYW5nZXJvdXNPYmplY3QoZGF0YSkpIHtcbiAgICByZXR1cm4gZGF0YTtcbiAgfVxuXG4gIGlmIChzZWVuLmhhcyhkYXRhKSkge1xuICAgIHJldHVybiBzZWVuLmdldChkYXRhKTtcbiAgfVxuXG4gIGlmIChkYXRhIGluc3RhbmNlb2YgSHR0cFBhcmFtcykge1xuICAgIHJldHVybiBuZXcgVXRpbHNIdHRwUGFyYW1zUmVxdWVzdCh1bmRlZmluZWQsIGRhdGEpIGFzIFQ7XG4gIH1cblxuICBpZiAoZGF0YSBpbnN0YW5jZW9mIERhdGUpIHtcbiAgICByZXR1cm4gbmV3IERhdGUoZGF0YS5nZXRUaW1lKCkpIGFzIFQ7XG4gIH1cblxuICBpZiAoaXNEYXlqc09iamVjdChkYXRhKSkge1xuICAgIHJldHVybiBnZXREYXlqcyh7IGRhdGU6IGRhdGEudmFsdWVPZigpIH0pIGFzIFQ7XG4gIH1cblxuICBpZiAoaXNSZWdFeHAoZGF0YSkpIHtcbiAgICByZXR1cm4gbmV3IFJlZ0V4cChkYXRhLnNvdXJjZSwgZGF0YS5mbGFncykgYXMgVDtcbiAgfVxuXG4gIGlmIChpc01hcChkYXRhKSkge1xuICAgIGNvbnN0IG1hcENvcHkgPSBuZXcgTWFwKCk7XG4gICAgc2Vlbi5zZXQoZGF0YSwgbWFwQ29weSk7XG4gICAgZGF0YS5mb3JFYWNoKCh2YWwsIGtleSkgPT4ge1xuICAgICAgbWFwQ29weS5zZXQoY2xvbmVEZWVwKGtleSwgb3B0aW9ucywgc2VlbiksIGNsb25lRGVlcCh2YWwsIG9wdGlvbnMsIHNlZW4pKTtcbiAgICB9KTtcbiAgICByZXR1cm4gbWFwQ29weSBhcyBUO1xuICB9XG5cbiAgaWYgKGlzU2V0KGRhdGEpKSB7XG4gICAgY29uc3Qgc2V0Q29weSA9IG5ldyBTZXQoKTtcbiAgICBzZWVuLnNldChkYXRhLCBzZXRDb3B5KTtcbiAgICBkYXRhLmZvckVhY2goKHZhbCkgPT4ge1xuICAgICAgc2V0Q29weS5hZGQoY2xvbmVEZWVwKHZhbCwgb3B0aW9ucywgc2VlbikpO1xuICAgIH0pO1xuICAgIHJldHVybiBzZXRDb3B5IGFzIFQ7XG4gIH1cblxuICBpZiAoaXNBcnJheShkYXRhKSkge1xuICAgIHNlZW4uc2V0KGRhdGEsIGRhdGEubWFwKChpdGVtKSA9PiBjbG9uZURlZXAoaXRlbSwgb3B0aW9ucywgc2VlbikpIGFzIFQpO1xuICAgIHJldHVybiBzZWVuLmdldChkYXRhKSBhcyBUO1xuICB9XG5cbiAgaWYgKGlzUmV0dXJuQXNJc09iamVjdChkYXRhKSkge1xuICAgIHJldHVybiBkYXRhIGFzIFQ7XG4gIH1cblxuICBpZiAoaXNTaWduYWwoZGF0YSkpIHtcbiAgICBpZiAob3B0aW9ucz8uaWdub3JlU2lnbmFsKSB7XG4gICAgICByZXR1cm4gZGF0YSBhcyBUO1xuICAgIH1cbiAgICBzZWVuLnNldChkYXRhLCBzaWduYWwoY2xvbmVEZWVwKGRhdGEoKSwgb3B0aW9ucywgc2VlbikpKTtcbiAgICByZXR1cm4gc2Vlbi5nZXQoZGF0YSkgYXMgVDtcbiAgfVxuXG4gIGNvbnN0IHJlc3VsdDogUmVjb3JkPHN0cmluZywgYW55PiA9IHt9O1xuICBzZWVuLnNldChkYXRhLCByZXN1bHQpO1xuICBmb3IgKGNvbnN0IGtleSBpbiBkYXRhKSB7XG4gICAgY29uc3QgdmFsdWUgPSBkYXRhW2tleV07XG4gICAgaWYgKHZhbHVlIGluc3RhbmNlb2YgSHR0cFBhcmFtcykge1xuICAgICAgcmVzdWx0W2tleV0gPSBuZXcgVXRpbHNIdHRwUGFyYW1zUmVxdWVzdCh1bmRlZmluZWQsIHZhbHVlKSBhcyBUO1xuICAgICAgY29udGludWU7XG4gICAgfVxuXG4gICAgaWYgKHZhbHVlIGluc3RhbmNlb2YgRGF0ZSkge1xuICAgICAgcmVzdWx0W2tleV0gPSBuZXcgRGF0ZSh2YWx1ZS5nZXRUaW1lKCkpIGFzIFQ7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG5cbiAgICBpZiAoZGF5anMuaXNEYXlqcyh2YWx1ZSkpIHtcbiAgICAgIHJlc3VsdFtrZXldID0gZ2V0RGF5anMoeyBkYXRlOiB2YWx1ZS52YWx1ZU9mKCkgfSkgYXMgVDtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cblxuICAgIGlmIChpc1JlZ0V4cChkYXRhKSkge1xuICAgICAgcmVzdWx0W2tleV0gPSBuZXcgUmVnRXhwKHZhbHVlLnNvdXJjZSwgdmFsdWUuZmxhZ3MpIGFzIFQ7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG4gICAgaWYgKGlzUmV0dXJuQXNJc09iamVjdChkYXRhKSkge1xuICAgICAgcmVzdWx0W2tleV0gPSB2YWx1ZTtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cblxuICAgIGlmIChPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwoZGF0YSwga2V5KSkge1xuICAgICAgcmVzdWx0W2tleV0gPSBjbG9uZURlZXAodmFsdWUsIG9wdGlvbnMsIHNlZW4pO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiByZXN1bHQgYXMgVDtcbn07XG5cbi8qKlxuICogQ2h1eeG7g24gxJHhu5VpIG3hu5l0IG3huqNuZyBjw6FjIMSR4buRaSB0xrDhu6NuZyB0aMOgbmggbeG7mXQgxJHhu5FpIHTGsOG7o25nIHbhu5tpIGtow7NhIMSRxrDhu6NjIGNo4buJIMSR4buLbmhcbiAqIEBwYXJhbSBkYXRhIE3huqNuZyBjw6FjIMSR4buRaSB0xrDhu6NuZyBj4bqnbiBjaHV54buDbiDEkeG7lWlcbiAqIEBwYXJhbSBrZXkgVMOqbiB0aHXhu5ljIHTDrW5oIMSRxrDhu6NjIHPhu60gZOG7pW5nIGzDoG0ga2jDs2EgdHJvbmcgxJHhu5FpIHTGsOG7o25nIGvhur90IHF14bqjXG4gKiBAcmV0dXJucyDEkOG7kWkgdMaw4bujbmcgduG7m2kgY8OhYyBnacOhIHRy4buLIHThu6sgbeG6o25nIMSRxrDhu6NjIMSRw6FuaCBrZXkgdGhlbyB0aHXhu5ljIHTDrW5oIMSRw6MgY2jhu4kgxJHhu4tuaFxuICogQGV4YW1wbGVcbiAqIGNvbnN0IGRhdGEgPSBbXG4gKiAgIHsgaWQ6IDEsIG5hbWU6ICdKb2huJyB9LFxuICogICB7IGlkOiAyLCBuYW1lOiAnSmFuZScgfVxuICogXTtcbiAqIGtleUJ5KGRhdGEsICdpZCcpO1xuICogLy8gS+G6v3QgcXXhuqM6IHtcbiAqIC8vICAgJzEnOiB7IGlkOiAxLCBuYW1lOiAnSm9obicgfSxcbiAqIC8vICAgJzInOiB7IGlkOiAyLCBuYW1lOiAnSmFuZScgfVxuICogLy8gfVxuICovXG5leHBvcnQgY29uc3Qga2V5QnkgPSAoZGF0YTogQXJyYXk8VFlQRV9PQkpFQ1Q+LCBrZXk6IHN0cmluZyk6IFRZUEVfT0JKRUNUID0+IHtcbiAgaWYgKCFkYXRhIHx8ICFkYXRhLmxlbmd0aCB8fCAha2V5KSB7XG4gICAgcmV0dXJuIHt9O1xuICB9XG4gIHJldHVybiBkYXRhLnJlZHVjZSgoZGlyLCBuZXh0SXRlbSkgPT4ge1xuICAgIGNvbnN0IHZhbHVlT2ZLZXkgPSBnZXQobmV4dEl0ZW0sIGtleSBhcyBQYXRoT2Y8VFlQRV9PQkpFQ1Q+KTtcbiAgICBpZiAodHlwZW9mIHZhbHVlT2ZLZXkgIT09ICdzdHJpbmcnICYmIHR5cGVvZiB2YWx1ZU9mS2V5ICE9PSAnbnVtYmVyJyAmJiB0eXBlb2YgdmFsdWVPZktleSAhPT0gJ2Jvb2xlYW4nKSB7XG4gICAgICByZXR1cm4gZGlyO1xuICAgIH1cbiAgICBpZiAoIU9iamVjdC5rZXlzKGRpcikuaW5jbHVkZXMoYCR7dmFsdWVPZktleX1gKSkge1xuICAgICAgZGlyW2Ake3ZhbHVlT2ZLZXl9YF0gPSBuZXh0SXRlbTtcbiAgICB9XG4gICAgcmV0dXJuIGRpcjtcbiAgfSwge30pO1xufTtcblxuLyoqXG4gKiBOaMOzbSBjw6FjIMSR4buRaSB0xrDhu6NuZyB0cm9uZyBt4buZdCBt4bqjbmcgdGjDoG5oIGPDoWMgbmjDs20gZOG7sWEgdHLDqm4gbeG7mXQgdGh14buZYyB0w61uaCBj4bulIHRo4buDXG4gKiBAcGFyYW0gZGF0YSBN4bqjbmcgY8OhYyDEkeG7kWkgdMaw4bujbmcgY+G6p24gbmjDs21cbiAqIEBwYXJhbSBrZXkgVMOqbiB0aHXhu5ljIHTDrW5oIMSRxrDhu6NjIHPhu60gZOG7pW5nIGzDoG0ga2jDs2EgbmjDs21cbiAqIEByZXR1cm5zIMSQ4buRaSB0xrDhu6NuZyB24bubaSBjw6FjIGdpw6EgdHLhu4sgdOG7qyBt4bqjbmcgxJHGsOG7o2MgbmjDs20gdGhlbyB0aHXhu5ljIHTDrW5oIMSRw6MgY2jhu4kgxJHhu4tuaFxuICogQGV4YW1wbGVcbiAqIGNvbnN0IGRhdGEgPSBbXG4gKiAgIHsgaWQ6IDEsIG5hbWU6ICdKb2huJyB9LFxuICogICB7IGlkOiAyLCBuYW1lOiAnSmFuZScgfSxcbiAqICAgeyBpZDogMSwgbmFtZTogJ0pvaG4nIH1cbiAqIF07XG4gKiBncm91cEJ5KGRhdGEsICdpZCcpO1xuICogLy8gS+G6v3QgcXXhuqM6IHtcbiAqIC8vICAgJzEnOiBbXG4gKiAvLyAgICAgeyBpZDogMSwgbmFtZTogJ0pvaG4nIH0sXG4gKiAvLyAgICAgeyBpZDogMSwgbmFtZTogJ0pvaG4nIH1cbiAqIC8vICAgXSxcbiAqIC8vICAgJzInOiBbXG4gKiAvLyAgICAgeyBpZDogMiwgbmFtZTogJ0phbmUnIH1cbiAqIC8vIH1cbiAqL1xuZXhwb3J0IGNvbnN0IGdyb3VwQnkgPSAoZGF0YTogQXJyYXk8VFlQRV9PQkpFQ1Q+LCBrZXk6IHN0cmluZyk6IFRZUEVfT0JKRUNUID0+IHtcbiAgaWYgKCFkYXRhIHx8ICFkYXRhLmxlbmd0aCB8fCAhT2JqZWN0LmtleXMoZ2V0KGRhdGEsICdbMF0nKSkubGVuZ3RoIHx8ICFrZXkpIHtcbiAgICByZXR1cm4ge307XG4gIH1cbiAgcmV0dXJuIGRhdGEucmVkdWNlKChkaXIsIG5leHRJdGVtKSA9PiB7XG4gICAgY29uc3QgdmFsdWVPZktleSA9IGdldChuZXh0SXRlbSwga2V5IGFzIFBhdGhPZjxUWVBFX09CSkVDVD4pO1xuICAgIGlmICh0eXBlb2YgdmFsdWVPZktleSAhPT0gJ3N0cmluZycgJiYgdHlwZW9mIHZhbHVlT2ZLZXkgIT09ICdudW1iZXInICYmIHR5cGVvZiB2YWx1ZU9mS2V5ICE9PSAnYm9vbGVhbicpIHtcbiAgICAgIHJldHVybiBkaXI7XG4gICAgfVxuICAgIGlmICghT2JqZWN0LmtleXMoZGlyKS5pbmNsdWRlcyhgJHt2YWx1ZU9mS2V5fWApKSB7XG4gICAgICBkaXJbYCR7dmFsdWVPZktleX1gXSA9IFtdO1xuICAgIH1cbiAgICBkaXJbYCR7dmFsdWVPZktleX1gXS5wdXNoKG5leHRJdGVtKTtcbiAgICByZXR1cm4gZGlyO1xuICB9LCB7fSk7XG59O1xuXG4vKipcbiAqIFThuqFvIG3hu5l0IG3huqNuZyBjw6FjIHPhu5EgdOG7qyBnacOhIHRy4buLIGLhuq90IMSR4bqndSDEkeG6v24gZ2nDoSB0cuG7iyBr4bq/dCB0aMO6YyB24bubaSBixrDhu5tjIG5o4bqjeSB0w7l5IGNo4buNblxuICogQHBhcmFtIHN0YXJ0IEdpw6EgdHLhu4sgYuG6r3QgxJHhuqd1IGPhu6dhIGTDo3kgc+G7kS4gTuG6v3UgY2jhu4kgY8OzIG3hu5l0IHRoYW0gc+G7kSwgxJHDonkgc+G6vSBsw6AgZ2nDoSB0cuG7iyBr4bq/dCB0aMO6YyB2w6AgZ2nDoSB0cuG7iyBi4bqvdCDEkeG6p3Ugc+G6vSBsw6AgMFxuICogQHBhcmFtIGVuZCBHacOhIHRy4buLIGvhur90IHRow7pjIGPhu6dhIGTDo3kgc+G7kSAodMO5eSBjaOG7jW4pXG4gKiBAcGFyYW0gc3RlcCBCxrDhu5tjIG5o4bqjeSBnaeG7r2EgY8OhYyBz4buRIHRyb25nIGTDo3kgKHTDuXkgY2jhu41uKS4gTeG6t2MgxJHhu4tuaCBsw6AgMSBu4bq/dSBlbmQgPiBzdGFydCwgLTEgbuG6v3UgZW5kIDwgc3RhcnRcbiAqIEByZXR1cm5zIE3huqNuZyBjw6FjIHPhu5EgdOG7qyBzdGFydCDEkeG6v24gZW5kIHbhu5tpIGLGsOG7m2MgbmjhuqN5IHN0ZXBcbiAqIEBleGFtcGxlXG4gKiByYW5nZSg0KTsgICAgICAvLyBbMCwgMSwgMiwgM11cbiAqIHJhbmdlKDEsIDUpOyAgIC8vIFsxLCAyLCAzLCA0XVxuICogcmFuZ2UoMCwgMjAsIDUpOyAvLyBbMCwgNSwgMTAsIDE1XVxuICogcmFuZ2UoNSwgMik7ICAgLy8gWzUsIDQsIDNdXG4gKi9cbmV4cG9ydCBjb25zdCByYW5nZSA9IChzdGFydDogbnVtYmVyLCBlbmQ/OiBudW1iZXIsIHN0ZXA/OiBudW1iZXIpOiBBcnJheTxudW1iZXI+ID0+IHtcbiAgaWYgKGVuZCA9PT0gdW5kZWZpbmVkIHx8IGVuZCA9PT0gbnVsbCkge1xuICAgIGVuZCA9IHN0YXJ0O1xuICAgIHN0YXJ0ID0gMDtcbiAgfVxuXG4gIGlmICghc3RlcCkge1xuICAgIHN0ZXAgPSBlbmQgPCAwID8gLTEgOiAxO1xuICB9XG5cbiAgaWYgKGVuZCA8IHN0YXJ0ICYmIHN0ZXAgPiAwKSB7XG4gICAgc3RlcCAqPSAtMTtcbiAgfVxuXG4gIGNvbnN0IHZhbHVlU3RhcnRCeVN0ZXAgPSBzdGVwICsgc3RhcnQ7XG4gIGNvbnN0IGRpcmVjdGlvbiA9IHN0YXJ0IDwgZW5kID8gJ2FzYycgOiAnZGVzYyc7XG4gIGlmICgoZGlyZWN0aW9uID09PSAnYXNjJyAmJiAodmFsdWVTdGFydEJ5U3RlcCA8IHN0YXJ0IHx8IHZhbHVlU3RhcnRCeVN0ZXAgPiBlbmQpKSB8fCAoZGlyZWN0aW9uID09PSAnZGVzYycgJiYgKHZhbHVlU3RhcnRCeVN0ZXAgPiBzdGFydCB8fCB2YWx1ZVN0YXJ0QnlTdGVwIDwgZW5kKSkpIHtcbiAgICByZXR1cm4gW3N0YXJ0XTtcbiAgfVxuXG4gIGNvbnN0IGFyciA9IG5ldyBBcnJheTxudW1iZXI+KCk7XG5cbiAgZm9yIChsZXQgaW5kZXggPSAwOyBpbmRleCA8IE1hdGguYWJzKGVuZCAtIHN0YXJ0KTsgaW5kZXgrKykge1xuICAgIGxldCB2YWx1ZSA9IHN0YXJ0ICsgaW5kZXggKiBzdGVwO1xuICAgIGlmIChpbmRleCA9PT0gMCkge1xuICAgICAgdmFsdWUgPSBzdGFydDtcbiAgICB9XG5cbiAgICBpZiAoKGRpcmVjdGlvbiA9PT0gJ2FzYycgJiYgKHZhbHVlIDwgc3RhcnQgfHwgdmFsdWUgPiBlbmQpKSB8fCAoZGlyZWN0aW9uID09PSAnZGVzYycgJiYgKHZhbHVlID4gc3RhcnQgfHwgdmFsdWUgPCBlbmQpKSkge1xuICAgICAgcmV0dXJuIGFycjtcbiAgICB9XG5cbiAgICBhcnIucHVzaCh2YWx1ZSk7XG4gIH1cblxuICByZXR1cm4gYXJyO1xufTtcblxuLyoqXG4gKiBTbyBzw6FuaCBoYWkgZ2nDoSB0cuG7iyBi4bqldCBr4buzIGPDsyBi4bqxbmcgbmhhdSBoYXkga2jDtG5nXG4gKiBAcGFyYW0gdmFsdWUxIEdpw6EgdHLhu4sgdGjhu6kgbmjhuqV0IGPhuqduIHNvIHPDoW5oXG4gKiBAcGFyYW0gdmFsdWUyIEdpw6EgdHLhu4sgdGjhu6kgaGFpIGPhuqduIHNvIHPDoW5oXG4gKiBAcGFyYW0gZXhhY3RseVBvc2l0aW9uIEPDsyBzbyBzw6FuaCBjaMOtbmggeMOhYyB24buLIHRyw60gY8OhYyBwaOG6p24gdOG7rSB0cm9uZyBt4bqjbmcgaGF5IGtow7RuZ1xuICogQHJldHVybnMgdHJ1ZSBu4bq/dSBoYWkgZ2nDoSB0cuG7iyBi4bqxbmcgbmhhdSwgZmFsc2UgbuG6v3Uga2jDtG5nIGLhurFuZyBuaGF1XG4gKiBAZXhhbXBsZVxuICogaXNFcXVhbChbMSwyLDNdLCBbMSwyLDNdKTsgLy8gdHJ1ZVxuICogaXNFcXVhbChbMSwyLDNdLCBbMywyLDFdKTsgLy8gdHJ1ZSBraGkgZXhhY3RseVBvc2l0aW9uID0gZmFsc2VcbiAqIGlzRXF1YWwoWzEsMiwzXSwgWzMsMiwxXSk7IC8vIGZhbHNlIGtoaSBleGFjdGx5UG9zaXRpb24gPSB0cnVlXG4gKiBpc0VxdWFsKHthOjF9LCB7YToxfSk7IC8vIHRydWVcbiAqL1xuZXhwb3J0IGNvbnN0IGlzRXF1YWwgPSAoXG4gIHZhbHVlMTogYW55LFxuICB2YWx1ZTI6IGFueSxcbiAgb3B0aW9ucz86IHtcbiAgICBleGFjdGx5UG9zaXRpb24/OiBib29sZWFuO1xuICAgIGlnbm9yZUV4YWN0bHlEYXRhVHlwZT86IGJvb2xlYW47XG4gICAgaWdub3JlVW5XcmFwU2lnbmFsPzogYm9vbGVhbjtcbiAgfVxuKTogYm9vbGVhbiA9PiB7XG4gIC8vIEhhbmRsZSBzaWduYWxzXG4gIGlmICghb3B0aW9ucz8uaWdub3JlVW5XcmFwU2lnbmFsKSB7XG4gICAgdmFsdWUxID0gdW53cmFwU2lnbmFsKHZhbHVlMSk7XG4gICAgdmFsdWUyID0gdW53cmFwU2lnbmFsKHZhbHVlMik7XG4gIH1cbiAgY29uc3QgeyBleGFjdGx5UG9zaXRpb24gPSBmYWxzZSwgaWdub3JlRXhhY3RseURhdGFUeXBlID0gZmFsc2UgfSA9IG9wdGlvbnMgfHwge307XG5cbiAgaWYgKHZhbHVlMSA9PT0gdmFsdWUyIHx8ICh2YWx1ZTEgPT09IG51bGwgJiYgdmFsdWUyID09PSBudWxsKSB8fCAodmFsdWUxID09PSB1bmRlZmluZWQgJiYgdmFsdWUyID09PSB1bmRlZmluZWQpKSB7XG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICBpZiAoaWdub3JlRXhhY3RseURhdGFUeXBlKSB7XG4gICAgcmV0dXJuIGlzRXF1YWwoaXNOaWwodmFsdWUxKSA/IHVuZGVmaW5lZCA6IGAke3ZhbHVlMX1gLCBpc05pbCh2YWx1ZTIpID8gdW5kZWZpbmVkIDogYCR7dmFsdWUyfWApO1xuICB9XG5cbiAgaWYgKHR5cGVvZiB2YWx1ZTEgIT09ICdvYmplY3QnIHx8IHR5cGVvZiB2YWx1ZTIgIT09ICdvYmplY3QnIHx8IChBcnJheS5pc0FycmF5KHZhbHVlMSkgJiYgIUFycmF5LmlzQXJyYXkodmFsdWUyKSkgfHwgKCFBcnJheS5pc0FycmF5KHZhbHVlMSkgJiYgQXJyYXkuaXNBcnJheSh2YWx1ZTIpKSkge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIGlmIChBcnJheS5pc0FycmF5KHZhbHVlMSkgfHwgQXJyYXkuaXNBcnJheSh2YWx1ZTIpKSB7XG4gICAgaWYgKHZhbHVlMS5sZW5ndGggIT09IHZhbHVlMi5sZW5ndGgpIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgaWYgKCFleGFjdGx5UG9zaXRpb24pIHtcbiAgICAgIHJldHVybiAhdmFsdWUxLnNvbWUoKGl0ZW06IGFueSkgPT4gIXZhbHVlMi5pbmNsdWRlcyhpdGVtKSk7XG4gICAgfVxuXG4gICAgcmV0dXJuICF2YWx1ZTEuc29tZSgoaXRlbTogYW55LCBpbmRleDogbnVtYmVyKSA9PiAhaXNFcXVhbChpdGVtLCB2YWx1ZTJbaW5kZXhdLCBvcHRpb25zKSk7XG4gIH1cblxuICBpZiAoT2JqZWN0LmtleXModmFsdWUxKS5sZW5ndGggIT09IE9iamVjdC5rZXlzKHZhbHVlMikubGVuZ3RoKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG4gIHJldHVybiAhT2JqZWN0LmtleXModmFsdWUxKS5zb21lKChrZXkpID0+ICFpc0VxdWFsKHZhbHVlMVtrZXldLCB2YWx1ZTJba2V5XSwgb3B0aW9ucykpO1xufTtcblxuLyoqXG4gKiBMb+G6oWkgYuG7jyBjw6FjIHBo4bqnbiB04butIHRyw7luZyBs4bq3cCB0cm9uZyBt4bqjbmcgZOG7sWEgdHLDqm4gbeG7mXQgdGh14buZYyB0w61uaCBjaOG7iSDEkeG7i25oXG4gKiBAcGFyYW0gZGF0YSBN4bqjbmcgZOG7ryBsaeG7h3UgY+G6p24geOG7rSBsw71cbiAqIEBwYXJhbSBrZXkgVMOqbiB0aHXhu5ljIHTDrW5oIGTDuW5nIMSR4buDIHNvIHPDoW5oIHRyw7luZyBs4bq3cC4gTuG6v3Uga2jDtG5nIGPDsyBrZXkgdGjDrCBzbyBzw6FuaCB0cuG7sWMgdGnhur9wIGdpw6EgdHLhu4tcbiAqIEByZXR1cm5zIE3huqNuZyBt4bubaSBjaOG7qWEgY8OhYyBwaOG6p24gdOG7rSBraMO0bmcgdHLDuW5nIGzhurdwXG4gKiBAZXhhbXBsZVxuICogY29uc3QgYXJyID0gW1xuICogICB7IGlkOiAxLCBuYW1lOiAnQScgfSxcbiAqICAgeyBpZDogMiwgbmFtZTogJ0InIH0sXG4gKiAgIHsgaWQ6IDEsIG5hbWU6ICdDJyB9XG4gKiBdO1xuICogdW5pcUJ5KGFyciwgJ2lkJyk7IC8vIFt7IGlkOiAxLCBuYW1lOiAnQScgfSwgeyBpZDogMiwgbmFtZTogJ0InIH1dXG4gKlxuICogY29uc3QgbnVtYmVycyA9IFsxLCAyLCAyLCAzLCAzXTtcbiAqIHVuaXFCeShudW1iZXJzKTsgLy8gWzEsIDIsIDNdXG4gKlxuICogY29uc3QgbnVtYmVyc1NpZ25hbCA9IFtzaWduYWwoMSksIHNpZ25hbCgyKSwgc2lnbmFsKDMpLCBzaWduYWwoMiksIHNpZ25hbCg1KSwgc2lnbmFsKDQpLCBzaWduYWwoMSksIHNpZ25hbCg2KSwgc2lnbmFsKDcpLCBzaWduYWwoNildO1xuICogdW5pcUJ5KG51bWJlcnNTaWduYWwpOyAvLyBbc2lnbmFsKDEpLCBzaWduYWwoMiksIHNpZ25hbCgzKSwgc2lnbmFsKDUpLCBzaWduYWwoNCksIHNpZ25hbCg2KSwgc2lnbmFsKDcpXVxuICovXG5leHBvcnQgY29uc3QgdW5pcUJ5ID0gPFQ+KGRhdGE6IEFycmF5PFQ+LCBrZXk/OiBzdHJpbmcpOiBBcnJheTxUPiA9PiB7XG4gIGlmICgha2V5IHx8ICFkYXRhPy5sZW5ndGggfHwgdHlwZW9mIGdldChkYXRhLCAnWzBdJykgIT09ICdvYmplY3QnKSB7XG4gICAgLy8gWOG7rSBsw70gbeG6o25nIGNo4bupYSBzaWduYWwgdmFsdWVzXG4gICAgaWYgKGRhdGFbMF0gJiYgaXNTaWduYWwoZGF0YVswXSkpIHtcbiAgICAgIGNvbnN0IHNlZW4gPSBuZXcgU2V0PHN0cmluZz4oKTtcblxuICAgICAgcmV0dXJuIGRhdGEuZmlsdGVyKChpdGVtKSA9PiB7XG4gICAgICAgIGNvbnN0IHZhbHVlID0gYCR7Z2V0PGFueT4oaXRlbSwgJycpfWA7XG4gICAgICAgIGlmIChzZWVuLmhhcyh2YWx1ZSkpIHtcbiAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgIH1cbiAgICAgICAgc2Vlbi5hZGQodmFsdWUpO1xuICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgIH0pO1xuICAgIH1cbiAgICAvLyBY4butIGzDvSBt4bqjbmcgcHJpbWl0aXZlIHZhbHVlc1xuICAgIHJldHVybiBBcnJheS5mcm9tKG5ldyBTZXQoZGF0YSkpO1xuICB9XG5cbiAgY29uc3QgZGF0YVVuaXF1ZSA9IGtleUJ5KGRhdGEgYXMgQXJyYXk8UmVjb3JkPHN0cmluZywgYW55Pj4sIGtleSk7XG5cbiAgcmV0dXJuIE9iamVjdC5rZXlzKGRhdGFVbmlxdWUpLm1hcCgoa2V5KSA9PiBkYXRhVW5pcXVlW2tleV0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdlbmVyYXRlSW50ZXJmYWNlID0gKG9iajogYW55LCBpbnRlcmZhY2VOYW1lOiBzdHJpbmcpOiBzdHJpbmcgPT4ge1xuICBjb25zdCBnZW5lcmF0ZVR5cGUgPSAodmFsdWU6IGFueSk6IHN0cmluZyA9PiB7XG4gICAgaWYgKHZhbHVlID09PSBudWxsKSB7XG4gICAgICByZXR1cm4gJ251bGwnO1xuICAgIH1cbiAgICBjb25zdCB0eXBlID0gdHlwZW9mIHZhbHVlO1xuXG4gICAgaWYgKHR5cGUgPT09ICdzdHJpbmcnKSB7XG4gICAgICByZXR1cm4gJ3N0cmluZyc7XG4gICAgfVxuICAgIGlmICh0eXBlID09PSAnbnVtYmVyJykge1xuICAgICAgcmV0dXJuICdudW1iZXInO1xuICAgIH1cbiAgICBpZiAodHlwZSA9PT0gJ2Jvb2xlYW4nKSB7XG4gICAgICByZXR1cm4gJ2Jvb2xlYW4nO1xuICAgIH1cbiAgICBpZiAodHlwZSA9PT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgIHJldHVybiAnYW55JztcbiAgICB9XG5cbiAgICBpZiAodmFsdWUgaW5zdGFuY2VvZiBEYXRlKSB7XG4gICAgICByZXR1cm4gJ0RhdGUnO1xuICAgIH1cblxuICAgIGlmICh2YWx1ZSBpbnN0YW5jZW9mIFJlZ0V4cCkge1xuICAgICAgcmV0dXJuICdSZWdFeHAnO1xuICAgIH1cblxuICAgIGlmIChBcnJheS5pc0FycmF5KHZhbHVlKSkge1xuICAgICAgaWYgKHZhbHVlLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICByZXR1cm4gJ0FycmF5PGFueT4nO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGBBcnJheTwke2dlbmVyYXRlVHlwZSh2YWx1ZVswXSl9PmA7XG4gICAgfVxuXG4gICAgaWYgKHR5cGUgPT09ICdvYmplY3QnKSB7XG4gICAgICBsZXQgaW50ZXJmYWNlU3RyID0gJ3tcXG4nO1xuICAgICAgZm9yIChjb25zdCBrZXkgaW4gdmFsdWUpIHtcbiAgICAgICAgaWYgKE9iamVjdC5wcm90b3R5cGUuaGFzT3duUHJvcGVydHkuY2FsbCh2YWx1ZSwga2V5KSkge1xuICAgICAgICAgIGNvbnN0IHZhbHVlVHlwZSA9IGdlbmVyYXRlVHlwZSh2YWx1ZVtrZXldKTtcbiAgICAgICAgICBpbnRlcmZhY2VTdHIgKz0gYCAgJHtrZXl9OiAke3ZhbHVlVHlwZX07XFxuYDtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgaW50ZXJmYWNlU3RyICs9ICd9JztcbiAgICAgIHJldHVybiBpbnRlcmZhY2VTdHI7XG4gICAgfVxuXG4gICAgcmV0dXJuICdhbnknO1xuICB9O1xuXG4gIGNvbnN0IGludGVyZmFjZVN0ciA9IGBpbnRlcmZhY2UgJHtpbnRlcmZhY2VOYW1lfSAke2dlbmVyYXRlVHlwZShvYmopfWA7XG4gIHJldHVybiBpbnRlcmZhY2VTdHI7XG59O1xuIl19
|