ng-http-caching 15.2.4 → 16.0.2
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 +1 -0
- package/esm2022/lib/ng-http-caching-interceptor.service.mjs +72 -0
- package/esm2022/lib/ng-http-caching.module.mjs +43 -0
- package/esm2022/lib/ng-http-caching.service.mjs +331 -0
- package/{esm2020 → esm2022}/lib/storage/ng-http-caching-browser-storage.mjs +122 -122
- package/{esm2020 → esm2022}/lib/storage/ng-http-caching-local-storage.mjs +6 -6
- package/{esm2020 → esm2022}/lib/storage/ng-http-caching-memory-storage.mjs +26 -26
- package/{esm2020 → esm2022}/lib/storage/ng-http-caching-session-storage.mjs +6 -6
- package/{esm2020 → esm2022}/lib/storage/ng-http-caching-storage.interface.mjs +1 -1
- package/{esm2020 → esm2022}/ng-http-caching.mjs +4 -4
- package/{esm2020 → esm2022}/public-api.mjs +11 -11
- package/{fesm2020 → fesm2022}/ng-http-caching.mjs +579 -578
- package/fesm2022/ng-http-caching.mjs.map +1 -0
- package/index.d.ts +5 -5
- package/lib/ng-http-caching-interceptor.service.d.ts +15 -15
- package/lib/ng-http-caching.module.d.ts +9 -9
- package/lib/ng-http-caching.service.d.ts +214 -213
- package/lib/storage/ng-http-caching-browser-storage.d.ts +18 -18
- package/lib/storage/ng-http-caching-local-storage.d.ts +4 -4
- package/lib/storage/ng-http-caching-memory-storage.d.ts +12 -12
- package/lib/storage/ng-http-caching-session-storage.d.ts +4 -4
- package/lib/storage/ng-http-caching-storage.interface.d.ts +31 -31
- package/package.json +7 -13
- package/public-api.d.ts +8 -8
- package/esm2020/lib/ng-http-caching-interceptor.service.mjs +0 -71
- package/esm2020/lib/ng-http-caching.module.mjs +0 -42
- package/esm2020/lib/ng-http-caching.service.mjs +0 -329
- package/fesm2015/ng-http-caching.mjs +0 -606
- package/fesm2015/ng-http-caching.mjs.map +0 -1
- package/fesm2020/ng-http-caching.mjs.map +0 -1
|
@@ -1,329 +0,0 @@
|
|
|
1
|
-
import { Injectable, InjectionToken, Inject, Optional, VERSION, isDevMode } from '@angular/core';
|
|
2
|
-
import { NgHttpCachingMemoryStorage } from './storage/ng-http-caching-memory-storage';
|
|
3
|
-
import * as i0 from "@angular/core";
|
|
4
|
-
export const NG_HTTP_CACHING_CONFIG = new InjectionToken('ng-http-caching.config');
|
|
5
|
-
export var NgHttpCachingStrategy;
|
|
6
|
-
(function (NgHttpCachingStrategy) {
|
|
7
|
-
/**
|
|
8
|
-
* All request are cacheable if HTTP method is into `allowedMethod`
|
|
9
|
-
*/
|
|
10
|
-
NgHttpCachingStrategy["ALLOW_ALL"] = "ALLOW_ALL";
|
|
11
|
-
/**
|
|
12
|
-
* Only the request with `X-NG-HTTP-CACHING-ALLOW-CACHE` header are cacheable if HTTP method is into `allowedMethod`
|
|
13
|
-
*/
|
|
14
|
-
NgHttpCachingStrategy["DISALLOW_ALL"] = "DISALLOW_ALL";
|
|
15
|
-
})(NgHttpCachingStrategy || (NgHttpCachingStrategy = {}));
|
|
16
|
-
export var NgHttpCachingHeaders;
|
|
17
|
-
(function (NgHttpCachingHeaders) {
|
|
18
|
-
/**
|
|
19
|
-
* Request is cacheable if HTTP method is into `allowedMethod`
|
|
20
|
-
*/
|
|
21
|
-
NgHttpCachingHeaders["ALLOW_CACHE"] = "X-NG-HTTP-CACHING-ALLOW-CACHE";
|
|
22
|
-
/**
|
|
23
|
-
* Request isn't cacheable
|
|
24
|
-
*/
|
|
25
|
-
NgHttpCachingHeaders["DISALLOW_CACHE"] = "X-NG-HTTP-CACHING-DISALLOW-CACHE";
|
|
26
|
-
/**
|
|
27
|
-
* Specific cache lifetime for the request
|
|
28
|
-
*/
|
|
29
|
-
NgHttpCachingHeaders["LIFETIME"] = "X-NG-HTTP-CACHING-LIFETIME";
|
|
30
|
-
/**
|
|
31
|
-
* You can tag multiple request by adding this header with the same tag and
|
|
32
|
-
* using `NgHttpCachingService.clearCacheByTag(tag: string)` for delete all the tagged request
|
|
33
|
-
*/
|
|
34
|
-
NgHttpCachingHeaders["TAG"] = "X-NG-HTTP-CACHING-TAG";
|
|
35
|
-
})(NgHttpCachingHeaders || (NgHttpCachingHeaders = {}));
|
|
36
|
-
export const NgHttpCachingHeadersList = Object.values(NgHttpCachingHeaders);
|
|
37
|
-
export const NG_HTTP_CACHING_SECOND_IN_MS = 1000;
|
|
38
|
-
export const NG_HTTP_CACHING_MINUTE_IN_MS = NG_HTTP_CACHING_SECOND_IN_MS * 60;
|
|
39
|
-
export const NG_HTTP_CACHING_HOUR_IN_MS = NG_HTTP_CACHING_MINUTE_IN_MS * 60;
|
|
40
|
-
export const NG_HTTP_CACHING_DAY_IN_MS = NG_HTTP_CACHING_HOUR_IN_MS * 24;
|
|
41
|
-
export const NG_HTTP_CACHING_WEEK_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 7;
|
|
42
|
-
export const NG_HTTP_CACHING_MONTH_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 30;
|
|
43
|
-
export const NG_HTTP_CACHING_YEAR_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 365;
|
|
44
|
-
export const NgHttpCachingConfigDefault = {
|
|
45
|
-
store: new NgHttpCachingMemoryStorage(),
|
|
46
|
-
lifetime: NG_HTTP_CACHING_HOUR_IN_MS,
|
|
47
|
-
version: VERSION.major,
|
|
48
|
-
allowedMethod: ['GET', 'HEAD'],
|
|
49
|
-
cacheStrategy: NgHttpCachingStrategy.ALLOW_ALL,
|
|
50
|
-
};
|
|
51
|
-
export class NgHttpCachingService {
|
|
52
|
-
constructor(config) {
|
|
53
|
-
this.queue = new Map();
|
|
54
|
-
this.gcLock = false;
|
|
55
|
-
this.devMode = isDevMode();
|
|
56
|
-
if (config) {
|
|
57
|
-
this.config = { ...NgHttpCachingConfigDefault, ...config };
|
|
58
|
-
}
|
|
59
|
-
else {
|
|
60
|
-
this.config = { ...NgHttpCachingConfigDefault };
|
|
61
|
-
}
|
|
62
|
-
// start cache clean
|
|
63
|
-
this.runGc();
|
|
64
|
-
}
|
|
65
|
-
/**
|
|
66
|
-
* Return the config
|
|
67
|
-
*/
|
|
68
|
-
getConfig() {
|
|
69
|
-
return this.config;
|
|
70
|
-
}
|
|
71
|
-
/**
|
|
72
|
-
* Return the queue map
|
|
73
|
-
*/
|
|
74
|
-
getQueue() {
|
|
75
|
-
return this.queue;
|
|
76
|
-
}
|
|
77
|
-
/**
|
|
78
|
-
* Return the cache store
|
|
79
|
-
*/
|
|
80
|
-
getStore() {
|
|
81
|
-
return this.config.store;
|
|
82
|
-
}
|
|
83
|
-
/**
|
|
84
|
-
* Return response from cache
|
|
85
|
-
*/
|
|
86
|
-
getFromCache(req) {
|
|
87
|
-
const key = this.getKey(req);
|
|
88
|
-
const cached = this.config.store.get(key);
|
|
89
|
-
if (!cached) {
|
|
90
|
-
return undefined;
|
|
91
|
-
}
|
|
92
|
-
if (this.isExpired(cached)) {
|
|
93
|
-
this.clearCacheByKey(key);
|
|
94
|
-
return undefined;
|
|
95
|
-
}
|
|
96
|
-
return this.deepFreeze(cached.response);
|
|
97
|
-
}
|
|
98
|
-
/**
|
|
99
|
-
* Add response to cache
|
|
100
|
-
*/
|
|
101
|
-
addToCache(req, res) {
|
|
102
|
-
const entry = {
|
|
103
|
-
url: req.urlWithParams,
|
|
104
|
-
response: res,
|
|
105
|
-
request: req,
|
|
106
|
-
addedTime: Date.now(),
|
|
107
|
-
version: this.config.version,
|
|
108
|
-
};
|
|
109
|
-
if (this.isValid(entry)) {
|
|
110
|
-
const key = this.getKey(req);
|
|
111
|
-
this.config.store.set(key, entry);
|
|
112
|
-
return true;
|
|
113
|
-
}
|
|
114
|
-
return false;
|
|
115
|
-
}
|
|
116
|
-
/**
|
|
117
|
-
* Delete response from cache
|
|
118
|
-
*/
|
|
119
|
-
deleteFromCache(req) {
|
|
120
|
-
const key = this.getKey(req);
|
|
121
|
-
return this.clearCacheByKey(key);
|
|
122
|
-
}
|
|
123
|
-
/**
|
|
124
|
-
* Clear the cache
|
|
125
|
-
*/
|
|
126
|
-
clearCache() {
|
|
127
|
-
this.config.store.clear();
|
|
128
|
-
}
|
|
129
|
-
/**
|
|
130
|
-
* Clear the cache by key
|
|
131
|
-
*/
|
|
132
|
-
clearCacheByKey(key) {
|
|
133
|
-
return this.config.store.delete(key);
|
|
134
|
-
}
|
|
135
|
-
/**
|
|
136
|
-
* Clear the cache by regex
|
|
137
|
-
*/
|
|
138
|
-
clearCacheByRegex(regex) {
|
|
139
|
-
this.config.store.forEach((_, key) => {
|
|
140
|
-
if (regex.test(key)) {
|
|
141
|
-
this.clearCacheByKey(key);
|
|
142
|
-
}
|
|
143
|
-
});
|
|
144
|
-
}
|
|
145
|
-
/**
|
|
146
|
-
* Clear the cache by TAG
|
|
147
|
-
*/
|
|
148
|
-
clearCacheByTag(tag) {
|
|
149
|
-
this.config.store.forEach((entry, key) => {
|
|
150
|
-
const tagHeader = entry.request.headers.get(NgHttpCachingHeaders.TAG);
|
|
151
|
-
if (tagHeader && tagHeader.split(',').includes(tag)) {
|
|
152
|
-
this.clearCacheByKey(key);
|
|
153
|
-
}
|
|
154
|
-
});
|
|
155
|
-
}
|
|
156
|
-
/**
|
|
157
|
-
* Run garbage collector (delete expired cache entry)
|
|
158
|
-
*/
|
|
159
|
-
runGc() {
|
|
160
|
-
if (this.gcLock) {
|
|
161
|
-
return false;
|
|
162
|
-
}
|
|
163
|
-
this.gcLock = true;
|
|
164
|
-
this.config.store.forEach((entry, key) => {
|
|
165
|
-
if (this.isExpired(entry)) {
|
|
166
|
-
this.clearCacheByKey(key);
|
|
167
|
-
}
|
|
168
|
-
});
|
|
169
|
-
this.gcLock = false;
|
|
170
|
-
return true;
|
|
171
|
-
}
|
|
172
|
-
/**
|
|
173
|
-
* Return true if cache entry is expired
|
|
174
|
-
*/
|
|
175
|
-
isExpired(entry) {
|
|
176
|
-
// if user provide custom method, use it
|
|
177
|
-
if (typeof this.config.isExpired === 'function') {
|
|
178
|
-
const result = this.config.isExpired(entry);
|
|
179
|
-
// if result is undefined, normal behaviour is provided
|
|
180
|
-
if (result !== undefined) {
|
|
181
|
-
return result;
|
|
182
|
-
}
|
|
183
|
-
}
|
|
184
|
-
// config/default lifetime
|
|
185
|
-
let lifetime = this.config.lifetime;
|
|
186
|
-
// request has own lifetime
|
|
187
|
-
const headerLifetime = entry.request.headers.get(NgHttpCachingHeaders.LIFETIME);
|
|
188
|
-
if (headerLifetime) {
|
|
189
|
-
lifetime = +headerLifetime;
|
|
190
|
-
}
|
|
191
|
-
// never expire if 0
|
|
192
|
-
if (lifetime === 0) {
|
|
193
|
-
return false;
|
|
194
|
-
}
|
|
195
|
-
// wrong lifetime
|
|
196
|
-
if (lifetime < 0 || isNaN(lifetime)) {
|
|
197
|
-
throw new Error('lifetime must be greater than or equal 0');
|
|
198
|
-
}
|
|
199
|
-
return entry.addedTime + lifetime < Date.now();
|
|
200
|
-
}
|
|
201
|
-
/**
|
|
202
|
-
* Return true if cache entry is valid for store in the cache
|
|
203
|
-
*/
|
|
204
|
-
isValid(entry) {
|
|
205
|
-
// if user provide custom method, use it
|
|
206
|
-
if (typeof this.config.isValid === 'function') {
|
|
207
|
-
const result = this.config.isValid(entry);
|
|
208
|
-
// if result is undefined, normal behaviour is provided
|
|
209
|
-
if (result !== undefined) {
|
|
210
|
-
return result;
|
|
211
|
-
}
|
|
212
|
-
}
|
|
213
|
-
// different version
|
|
214
|
-
if (this.config.version !== entry.version) {
|
|
215
|
-
return false;
|
|
216
|
-
}
|
|
217
|
-
return true;
|
|
218
|
-
}
|
|
219
|
-
/**
|
|
220
|
-
* Return true if the request is cacheable
|
|
221
|
-
*/
|
|
222
|
-
isCacheable(req) {
|
|
223
|
-
// if user provide custom method, use it
|
|
224
|
-
if (typeof this.config.isCacheable === 'function') {
|
|
225
|
-
const result = this.config.isCacheable(req);
|
|
226
|
-
// if result is undefined, normal behaviour is provided
|
|
227
|
-
if (result !== undefined) {
|
|
228
|
-
return result;
|
|
229
|
-
}
|
|
230
|
-
}
|
|
231
|
-
// request has disallow cache header
|
|
232
|
-
if (req.headers.has(NgHttpCachingHeaders.DISALLOW_CACHE)) {
|
|
233
|
-
return false;
|
|
234
|
-
}
|
|
235
|
-
// strategy is disallow all...
|
|
236
|
-
if (this.config.cacheStrategy === NgHttpCachingStrategy.DISALLOW_ALL) {
|
|
237
|
-
// request isn't allowed if come without allow header
|
|
238
|
-
if (!req.headers.has(NgHttpCachingHeaders.ALLOW_CACHE)) {
|
|
239
|
-
return false;
|
|
240
|
-
}
|
|
241
|
-
}
|
|
242
|
-
// if allowed method is only ALL, allow all http methos
|
|
243
|
-
if (this.config.allowedMethod.length === 1) {
|
|
244
|
-
if (this.config.allowedMethod[0] === 'ALL') {
|
|
245
|
-
return true;
|
|
246
|
-
}
|
|
247
|
-
}
|
|
248
|
-
// request is allowed if method is in allowedMethod
|
|
249
|
-
return this.config.allowedMethod.indexOf(req.method) !== -1;
|
|
250
|
-
}
|
|
251
|
-
/**
|
|
252
|
-
* Return the cache key.
|
|
253
|
-
* Default key is http method plus url with query parameters, eg.:
|
|
254
|
-
* `GET@https://github.com/nigrosimone/ng-http-caching`
|
|
255
|
-
*/
|
|
256
|
-
getKey(req) {
|
|
257
|
-
// if user provide custom method, use it
|
|
258
|
-
if (typeof this.config.getKey === 'function') {
|
|
259
|
-
const result = this.config.getKey(req);
|
|
260
|
-
// if result is undefined, normal behaviour is provided
|
|
261
|
-
if (result !== undefined) {
|
|
262
|
-
return result;
|
|
263
|
-
}
|
|
264
|
-
}
|
|
265
|
-
// default key is req.method plus url with query parameters
|
|
266
|
-
return req.method + '@' + req.urlWithParams;
|
|
267
|
-
}
|
|
268
|
-
/**
|
|
269
|
-
* Return observable from cache
|
|
270
|
-
*/
|
|
271
|
-
getFromQueue(req) {
|
|
272
|
-
const key = this.getKey(req);
|
|
273
|
-
const cached = this.queue.get(key);
|
|
274
|
-
if (!cached) {
|
|
275
|
-
return undefined;
|
|
276
|
-
}
|
|
277
|
-
return cached;
|
|
278
|
-
}
|
|
279
|
-
/**
|
|
280
|
-
* Add observable to cache
|
|
281
|
-
*/
|
|
282
|
-
addToQueue(req, obs) {
|
|
283
|
-
const key = this.getKey(req);
|
|
284
|
-
this.queue.set(key, obs);
|
|
285
|
-
}
|
|
286
|
-
/**
|
|
287
|
-
* Delete observable from cache
|
|
288
|
-
*/
|
|
289
|
-
deleteFromQueue(req) {
|
|
290
|
-
const key = this.getKey(req);
|
|
291
|
-
return this.queue.delete(key);
|
|
292
|
-
}
|
|
293
|
-
/**
|
|
294
|
-
* Recursively Object.freeze simple Javascript structures consisting of plain objects, arrays, and primitives.
|
|
295
|
-
* Make the data immutable.
|
|
296
|
-
* @returns immutable object
|
|
297
|
-
*/
|
|
298
|
-
deepFreeze(object) {
|
|
299
|
-
// No freezing in production (for better performance).
|
|
300
|
-
if (!this.devMode || !object || typeof object !== 'object') {
|
|
301
|
-
return object;
|
|
302
|
-
}
|
|
303
|
-
// When already frozen, we assume its children are frozen (for better performance).
|
|
304
|
-
// This should be true if you always use `deepFreeze` to freeze objects.
|
|
305
|
-
//
|
|
306
|
-
// Note that Object.isFrozen will also return `true` for primitives (numbers,
|
|
307
|
-
// strings, booleans, undefined, null), so there is no need to check for
|
|
308
|
-
// those explicitly.
|
|
309
|
-
if (Object.isFrozen(object)) {
|
|
310
|
-
return object;
|
|
311
|
-
}
|
|
312
|
-
// At this point we know that we're dealing with either an array or plain object, so
|
|
313
|
-
// just freeze it and recurse on its values.
|
|
314
|
-
Object.freeze(object);
|
|
315
|
-
Object.keys(object).forEach(key => this.deepFreeze(object[key]));
|
|
316
|
-
return object;
|
|
317
|
-
}
|
|
318
|
-
}
|
|
319
|
-
NgHttpCachingService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.5", ngImport: i0, type: NgHttpCachingService, deps: [{ token: NG_HTTP_CACHING_CONFIG, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
320
|
-
NgHttpCachingService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.5", ngImport: i0, type: NgHttpCachingService });
|
|
321
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.5", ngImport: i0, type: NgHttpCachingService, decorators: [{
|
|
322
|
-
type: Injectable
|
|
323
|
-
}], ctorParameters: function () { return [{ type: undefined, decorators: [{
|
|
324
|
-
type: Inject,
|
|
325
|
-
args: [NG_HTTP_CACHING_CONFIG]
|
|
326
|
-
}, {
|
|
327
|
-
type: Optional
|
|
328
|
-
}] }]; } });
|
|
329
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmctaHR0cC1jYWNoaW5nLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1odHRwLWNhY2hpbmcvc3JjL2xpYi9uZy1odHRwLWNhY2hpbmcuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLGNBQWMsRUFBRSxNQUFNLEVBQUUsUUFBUSxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFJakcsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0sMENBQTBDLENBQUM7O0FBeUJ0RixNQUFNLENBQUMsTUFBTSxzQkFBc0IsR0FBRyxJQUFJLGNBQWMsQ0FDdEQsd0JBQXdCLENBQ3pCLENBQUM7QUFFRixNQUFNLENBQU4sSUFBWSxxQkFTWDtBQVRELFdBQVkscUJBQXFCO0lBQy9COztPQUVHO0lBQ0gsZ0RBQXVCLENBQUE7SUFDdkI7O09BRUc7SUFDSCxzREFBNkIsQ0FBQTtBQUMvQixDQUFDLEVBVFcscUJBQXFCLEtBQXJCLHFCQUFxQixRQVNoQztBQUVELE1BQU0sQ0FBTixJQUFZLG9CQWtCWDtBQWxCRCxXQUFZLG9CQUFvQjtJQUM5Qjs7T0FFRztJQUNILHFFQUE2QyxDQUFBO0lBQzdDOztPQUVHO0lBQ0gsMkVBQW1ELENBQUE7SUFDbkQ7O09BRUc7SUFDSCwrREFBdUMsQ0FBQTtJQUN2Qzs7O09BR0c7SUFDSCxxREFBNkIsQ0FBQTtBQUMvQixDQUFDLEVBbEJXLG9CQUFvQixLQUFwQixvQkFBb0IsUUFrQi9CO0FBRUQsTUFBTSxDQUFDLE1BQU0sd0JBQXdCLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO0FBRTVFLE1BQU0sQ0FBQyxNQUFNLDRCQUE0QixHQUFHLElBQUksQ0FBQztBQUNqRCxNQUFNLENBQUMsTUFBTSw0QkFBNEIsR0FBRyw0QkFBNEIsR0FBRyxFQUFFLENBQUM7QUFDOUUs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NUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUN0QixNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUUsTUFBYyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUUxRSxPQUFPLE1BQXFCLENBQUM7SUFDL0IsQ0FBQzs7aUhBNVNVLG9CQUFvQixrQkFXckIsc0JBQXNCO3FIQVhyQixvQkFBb0I7MkZBQXBCLG9CQUFvQjtrQkFEaEMsVUFBVTs7MEJBWU4sTUFBTTsyQkFBQyxzQkFBc0I7OzBCQUFHLFFBQVEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlLCBJbmplY3Rpb25Ub2tlbiwgSW5qZWN0LCBPcHRpb25hbCwgVkVSU0lPTiwgaXNEZXZNb2RlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IEh0dHBSZXF1ZXN0LCBIdHRwUmVzcG9uc2UsIEh0dHBFdmVudCB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgT2JzZXJ2YWJsZSB9IGZyb20gJ3J4anMvaW50ZXJuYWwvT2JzZXJ2YWJsZSc7XHJcbmltcG9ydCB7IE5nSHR0cENhY2hpbmdTdG9yYWdlSW50ZXJmYWNlIH0gZnJvbSAnLi9zdG9yYWdlL25nLWh0dHAtY2FjaGluZy1zdG9yYWdlLmludGVyZmFjZSc7XHJcbmltcG9ydCB7IE5nSHR0cENhY2hpbmdNZW1vcnlTdG9yYWdlIH0gZnJvbSAnLi9zdG9yYWdlL25nLWh0dHAtY2FjaGluZy1tZW1vcnktc3RvcmFnZSc7XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIE5nSHR0cENhY2hpbmdFbnRyeTxLID0gYW55LCBUID0gYW55PiB7XHJcbiAgLyoqXHJcbiAgICogVVJMXHJcbiAgICovXHJcbiAgdXJsOiBzdHJpbmc7XHJcbiAgLyoqXHJcbiAgICogSHR0cFJlc3BvbnNlXHJcbiAgICovXHJcbiAgcmVzcG9uc2U6IEh0dHBSZXNwb25zZTxUPjtcclxuICAvKipcclxuICAgKiBIdHRwUmVxdWVzdFxyXG4gICAqL1xyXG4gIHJlcXVlc3Q6IEh0dHBSZXF1ZXN0PEs+O1xyXG4gIC8qKlxyXG4gICAqIFRpbWVzdGFtIG9mIGFkZCB0byBjYWNoZSB0aW1lXHJcbiAgICovXHJcbiAgYWRkZWRUaW1lOiBudW1iZXI7XHJcbiAgLyoqXHJcbiAgICogQ2FjaGUgdmVyc2lvblxyXG4gICAqL1xyXG4gIHZlcnNpb246IHN0cmluZztcclxufVxyXG5cclxuZXhwb3J0IGNvbnN0IE5HX0hUVFBfQ0FDSElOR19DT05GSUcgPSBuZXcgSW5qZWN0aW9uVG9rZW48TmdIdHRwQ2FjaGluZ0NvbmZpZz4oXHJcbiAgJ25nLWh0dHAtY2FjaGluZy5jb25maWcnXHJcbik7XHJcblxyXG5leHBvcnQgZW51bSBOZ0h0dHBDYWNoaW5nU3RyYXRlZ3kge1xyXG4gIC8qKlxyXG4gICAqIEFsbCByZXF1ZXN0IGFyZSBjYWNoZWFibGUgaWYgSFRUUCBtZXRob2QgaXMgaW50byBgYWxsb3dlZE1ldGhvZGBcclxuICAgKi9cclxuICBBTExPV19BTEwgPSAnQUxMT1dfQUxMJyxcclxuICAvKipcclxuICAgKiBPbmx5IHRoZSByZXF1ZXN0IHdpdGggYFgtTkctSFRUUC1DQUNISU5HLUFMTE9XLUNBQ0hFYCBoZWFkZXIgYXJlIGNhY2hlYWJsZSBpZiBIVFRQIG1ldGhvZCBpcyBpbnRvIGBhbGxvd2VkTWV0aG9kYFxyXG4gICAqL1xyXG4gIERJU0FMTE9XX0FMTCA9ICdESVNBTExPV19BTEwnXHJcbn1cclxuXHJcbmV4cG9ydCBlbnVtIE5nSHR0cENhY2hpbmdIZWFkZXJzIHtcclxuICAvKipcclxuICAgKiBSZXF1ZXN0IGlzIGNhY2hlYWJsZSBpZiBIVFRQIG1ldGhvZCBpcyBpbnRvIGBhbGxvd2VkTWV0aG9kYFxyXG4gICAqL1xyXG4gIEFMTE9XX0NBQ0hFID0gJ1gtTkctSFRUUC1DQUNISU5HLUFMTE9XLUNBQ0hFJyxcclxuICAvKipcclxuICAgKiBSZXF1ZXN0IGlzbid0IGNhY2hlYWJsZVxyXG4gICAqL1xyXG4gIERJU0FMTE9XX0NBQ0hFID0gJ1gtTkctSFRUUC1DQUNISU5HLURJU0FMTE9XLUNBQ0hFJyxcclxuICAvKipcclxuICAgKiBTcGVjaWZpYyBjYWNoZSBsaWZldGltZSBmb3IgdGhlIHJlcXVlc3RcclxuICAgKi9cclxuICBMSUZFVElNRSA9ICdYLU5HLUhUVFAtQ0FDSElORy1MSUZFVElNRScsXHJcbiAgLyoqXHJcbiAgICogWW91IGNhbiB0YWcgbXVsdGlwbGUgcmVxdWVzdCBieSBhZGRpbmcgdGhpcyBoZWFkZXIgd2l0aCB0aGUgc2FtZSB0YWcgYW5kIFxyXG4gICAqIHVzaW5nIGBOZ0h0dHBDYWNoaW5nU2VydmljZS5jbGVhckNhY2hlQnlUYWcodGFnOiBzdHJpbmcpYCBmb3IgZGVsZXRlIGFsbCB0aGUgdGFnZ2VkIHJlcXVlc3RcclxuICAgKi9cclxuICBUQUcgPSAnWC1ORy1IVFRQLUNBQ0hJTkctVEFHJ1xyXG59XHJcblxyXG5leHBvcnQgY29uc3QgTmdIdHRwQ2FjaGluZ0hlYWRlcnNMaXN0ID0gT2JqZWN0LnZhbHVlcyhOZ0h0dHBDYWNoaW5nSGVhZGVycyk7XHJcblxyXG5leHBvcnQgY29uc3QgTkdfSFRUUF9DQUNISU5HX1NFQ09ORF9JTl9NUyA9IDEwMDA7XHJcbmV4cG9ydCBjb25zdCBOR19IVFRQX0NBQ0hJTkdfTUlOVVRFX0lOX01TID0gTkdfSFRUUF9DQUNISU5HX1NFQ09ORF9JTl9NUyAqIDYwO1xyXG5leHBvcnQgY29uc3QgTkdfSFRUUF9DQUNISU5HX0hPVVJfSU5fTVMgPSBOR19IVFRQX0NBQ0hJTkdfTUlOVVRFX0lOX01TICogNjA7XHJcbmV4cG9ydCBjb25zdCBOR19IVFRQX0NBQ0hJTkdfREFZX0lOX01TID0gTkdfSFRUUF9DQUNISU5HX0hPVVJfSU5fTVMgKiAyNDtcclxuZXhwb3J0IGNvbnN0IE5HX0hUVFBfQ0FDSElOR19XRUVLX0lOX01TID0gTkdfSFRUUF9DQUNISU5HX0RBWV9JTl9NUyAqIDc7XHJcbmV4cG9ydCBjb25zdCBOR19IVFRQX0NBQ0hJTkdfTU9OVEhfSU5fTVMgPSBOR19IVFRQX0NBQ0hJTkdfREFZX0lOX01TICogMzA7XHJcbmV4cG9ydCBjb25zdCBOR19IVFRQX0NBQ0hJTkdfWUVBUl9JTl9NUyA9IE5HX0hUVFBfQ0FDSElOR19EQVlfSU5fTVMgKiAzNjU7XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIE5nSHR0cENhY2hpbmdDb25maWcge1xyXG4gIC8qKlxyXG4gICAqIFNldCB0aGUgY2FjaGUgc3RvcmUuIFlvdSBjYW4gaW1wbGVtZW50IHlvdXIgY3VzdG9tIHN0b3JlIGJ5IGltcGxlbWVudCB0aGUgYE5nSHR0cENhY2hpbmdTdG9yYWdlSW50ZXJmYWNlYCBpbnRlcmZhY2UsIGVnLjpcclxuICAgKi9cclxuICBzdG9yZT86IE5nSHR0cENhY2hpbmdTdG9yYWdlSW50ZXJmYWNlO1xyXG4gIC8qKlxyXG4gICAqIE51bWJlciBvZiBtaWxsaXNlY29uZCB0aGF0IGEgcmVzcG9uc2UgaXMgc3RvcmVkIGluIHRoZSBjYWNoZS4gXHJcbiAgICogWW91IGNhbiBzZXQgc3BlY2lmaWMgXCJsaWZldGltZVwiIGZvciBlYWNoIHJlcXVlc3QgYnkgYWRkIHRoZSBoZWFkZXIgYFgtTkctSFRUUC1DQUNISU5HLUxJRkVUSU1FYCAoc2VlIGV4YW1wbGUgYmVsb3cpLlxyXG4gICAqL1xyXG4gIGxpZmV0aW1lPzogbnVtYmVyO1xyXG4gIC8qKlxyXG4gICAqIEFycmF5IG9mIGFsbG93ZWQgSFRUUCBtZXRob2RzIHRvIGNhY2hlLiBcclxuICAgKiBZb3UgY2FuIGFsbG93IG11bHRpcGxlIG1ldGhvZHMsIGVnLjogYFsnR0VUJywgJ1BPU1QnLCAnUFVUJywgJ0RFTEVURScsICdIRUFEJ11gIG9yIFxyXG4gICAqIGFsbG93IGFsbCBtZXRob2RzIGJ5OiBgWydBTEwnXWAuIElmIGBhbGxvd2VkTWV0aG9kYCBpcyBhbiBlbXB0eSBhcnJheSAoYFtdYCksIG5vIHJlc3BvbnNlIGFyZSBjYWNoZWQuXHJcbiAgICogKldhcm5pbmchKiBgTmdIdHRwQ2FjaGluZ2AgdXNlIHRoZSBmdWxsIHVybCAodXJsIHdpdGggcXVlcnkgcGFyYW1ldGVycykgYXMgdW5pcXVlIGtleSBmb3IgdGhlIGNhY2hlZCByZXNwb25zZSxcclxuICAgKiB0aGlzIGlzIGNvcnJlY3QgZm9yIHRoZSBgR0VUYCByZXF1ZXN0IGJ1dCBpcyBfcG90ZW50aWFsbHlfIHdyb25nIGZvciBvdGhlciB0eXBlIG9mIHJlcXVlc3QgKGVnLiBgUE9TVGAsIGBQVVRgKS4gXHJcbiAgICogWW91IGNhbiBzZXQgYSBkaWZmZXJlbnQgXCJrZXlcIiBieSBjdXN0b21pemluZyB0aGUgYGdldEtleWAgY29uZmlnIG1ldGhvZCAoc2VlIGBnZXRLZXlgIHNlY3Rpb24pLlxyXG4gICAqL1xyXG4gIGFsbG93ZWRNZXRob2Q/OiBzdHJpbmdbXTtcclxuICAvKipcclxuICAgKiBTZXQgdGhlIGNhY2hlIHN0cmF0ZWd5LCBwb3NzaWJsZSBzdHJhdGVnaWVzIGFyZTpcclxuICAgKiAtIGBOZ0h0dHBDYWNoaW5nU3RyYXRlZ3kuQUxMT1dfQUxMYDogQWxsIHJlcXVlc3QgYXJlIGNhY2hlYWJsZSBpZiBIVFRQIG1ldGhvZCBpcyBpbnRvIGBhbGxvd2VkTWV0aG9kYDtcclxuICAgKiAtIGBOZ0h0dHBDYWNoaW5nU3RyYXRlZ3kuRElTQUxMT1dfQUxMYDogT25seSB0aGUgcmVxdWVzdCB3aXRoIGBYLU5HLUhUVFAtQ0FDSElORy1BTExPVy1DQUNIRWAgaGVhZGVyIGFyZSBjYWNoZWFibGUgaWYgSFRUUCBtZXRob2QgaXMgaW50byBgYWxsb3dlZE1ldGhvZGA7XHJcbiAgICovXHJcbiAgY2FjaGVTdHJhdGVneT86IE5nSHR0cENhY2hpbmdTdHJhdGVneTtcclxuICAvKipcclxuICAgKiBDYWNoZSB2ZXJzaW9uLiBXaGVuIHlvdSBoYXZlIGEgYnJlYWtpbmcgY2hhbmdlLCBjaGFuZ2UgdGhlIHZlcnNpb24sIGFuZCBpdCdsbCBkZWxldGUgdGhlIGN1cnJlbnQgY2FjaGUgYXV0b21hdGljYWxseS5cclxuICAgKiBUaGUgZGVmYXVsdCB2YWx1ZSBpcyBBbmd1bGFyIG1ham9yIHZlcnNpb24gKGVnLiAxMyksIGluIHRoaXMgd2F5LCB0aGUgY2FjaGUgaXMgaW52YWxpdGFkZWQgb24gZXZlcnkgQW5ndWxhciB1cGdyYWRlLlxyXG4gICAqL1xyXG4gIHZlcnNpb24/OiBzdHJpbmc7XHJcbiAgLyoqXHJcbiAgICogSWYgdGhpcyBmdW5jdGlvbiByZXR1cm4gYHRydWVgIHRoZSByZXF1ZXN0IGlzIGV4cGlyZWQgYW5kIGEgbmV3IHJlcXVlc3QgaXMgc2VuZCB0byBiYWNrZW5kLCBpZiByZXR1cm4gYGZhbHNlYCBpc24ndCBleHBpcmVkLiBcclxuICAgKiBJZiB0aGUgcmVzdWx0IGlzIGB1bmRlZmluZWRgLCB0aGUgbm9ybWFsIGJlaGF2aW91ciBpcyBwcm92aWRlZC5cclxuICAgKi9cclxuICBpc0V4cGlyZWQ/OiA8SywgVD4oZW50cnk6IE5nSHR0cENhY2hpbmdFbnRyeTxLLCBUPikgPT4gYm9vbGVhbiB8IHVuZGVmaW5lZDtcclxuICAvKipcclxuICAgKiBJZiB0aGlzIGZ1bmN0aW9uIHJldHVybiBgdHJ1ZWAgdGhlIHJlcXVlc3QgaXMgY2FjaGVhYmxlLCBpZiByZXR1cm4gYGZhbHNlYCBpc24ndCBjYWNoZWFibGUuIFxyXG4gICAqIElmIHRoZSByZXN1bHQgaXMgYHVuZGVmaW5lZGAsIHRoZSBub3JtYWwgYmVoYXZpb3VyIGlzIHByb3ZpZGVkLlxyXG4gICAqL1xyXG4gIGlzQ2FjaGVhYmxlPzogPEs+KHJlcTogSHR0cFJlcXVlc3Q8Sz4pID0+IGJvb2xlYW4gfCB1bmRlZmluZWQ7XHJcbiAgLyoqXHJcbiAgICogVGhpcyBmdW5jdGlvbiByZXR1cm4gdGhlIHVuaXF1ZSBrZXkgKGBzdHJpbmdgKSBmb3Igc3RvcmUgdGhlIHJlc3BvbnNlIGludG8gdGhlIGNhY2hlLiBcclxuICAgKiBJZiB0aGUgcmVzdWx0IGlzIGB1bmRlZmluZWRgLCB0aGUgbm9ybWFsIGJlaGF2aW91ciBpcyBwcm92aWRlZC5cclxuICAgKi9cclxuICBnZXRLZXk/OiA8Sz4ocmVxOiBIdHRwUmVxdWVzdDxLPikgPT4gc3RyaW5nIHwgdW5kZWZpbmVkO1xyXG4gIC8qKlxyXG4gICAqIElmIHRoaXMgZnVuY3Rpb24gcmV0dXJuIGB0cnVlYCB0aGUgY2FjaGUgZW50cnkgaXMgdmFsaWQgYW5kIGNhbiBiZSBzdHJvcmVkLCBpZiByZXR1cm4gYGZhbHNlYCBpc24ndCB2YWxpZC4gXHJcbiAgICogSWYgdGhlIHJlc3VsdCBpcyBgdW5kZWZpbmVkYCwgdGhlIG5vcm1hbCBiZWhhdmlvdXIgaXMgcHJvdmlkZWQuXHJcbiAgICovXHJcbiAgaXNWYWxpZD86IDxLLCBUPihlbnRyeTogTmdIdHRwQ2FjaGluZ0VudHJ5PEssIFQ+KSA9PiBib29sZWFuIHwgdW5kZWZpbmVkO1xyXG59XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIE5nSHR0cENhY2hpbmdEZWZhdWx0Q29uZmlnIGV4dGVuZHMgTmdIdHRwQ2FjaGluZ0NvbmZpZyB7XHJcbiAgc3RvcmU6IE5nSHR0cENhY2hpbmdTdG9yYWdlSW50ZXJmYWNlO1xyXG4gIGxpZmV0aW1lOiBudW1iZXI7XHJcbiAgYWxsb3dlZE1ldGhvZDogc3RyaW5nW107XHJcbiAgY2FjaGVTdHJhdGVneTogTmdIdHRwQ2FjaGluZ1N0cmF0ZWd5O1xyXG4gIHZlcnNpb246IHN0cmluZztcclxufVxyXG5cclxuZXhwb3J0IGNvbnN0IE5nSHR0cENhY2hpbmdDb25maWdEZWZhdWx0OiBSZWFkb25seTxOZ0h0dHBDYWNoaW5nRGVmYXVsdENvbmZpZz4gPSB7XHJcbiAgc3RvcmU6IG5ldyBOZ0h0dHBDYWNoaW5nTWVtb3J5U3RvcmFnZSgpLFxyXG4gIGxpZmV0aW1lOiBOR19IVFRQX0NBQ0hJTkdfSE9VUl9JTl9NUyxcclxuICB2ZXJzaW9uOiBWRVJTSU9OLm1ham9yLFxyXG4gIGFsbG93ZWRNZXRob2Q6IFsnR0VUJywgJ0hFQUQnXSxcclxuICBjYWNoZVN0cmF0ZWd5OiBOZ0h0dHBDYWNoaW5nU3RyYXRlZ3kuQUxMT1dfQUxMLFxyXG59O1xyXG5cclxuQEluamVjdGFibGUoKVxyXG5leHBvcnQgY2xhc3MgTmdIdHRwQ2FjaGluZ1NlcnZpY2Uge1xyXG5cclxuICBwcml2YXRlIHJlYWRvbmx5IHF1ZXVlID0gbmV3IE1hcDxzdHJpbmcsIE9ic2VydmFibGU8SHR0cEV2ZW50PGFueT4+PigpO1xyXG5cclxuICBwcml2YXRlIHJlYWRvbmx5IGNvbmZpZzogTmdIdHRwQ2FjaGluZ0RlZmF1bHRDb25maWc7XHJcblxyXG4gIHByaXZhdGUgZ2NMb2NrID0gZmFsc2U7XHJcblxyXG4gIHByaXZhdGUgZGV2TW9kZTogYm9vbGVhbiA9IGlzRGV2TW9kZSgpO1xyXG5cclxuICBjb25zdHJ1Y3RvcihcclxuICAgIEBJbmplY3QoTkdfSFRUUF9DQUNISU5HX0NPTkZJRykgQE9wdGlvbmFsKCkgY29uZmlnOiBSZWFkb25seTxOZ0h0dHBDYWNoaW5nQ29uZmlnPlxyXG4gICkge1xyXG4gICAgaWYgKGNvbmZpZykge1xyXG4gICAgICB0aGlzLmNvbmZpZyA9IHsgLi4uTmdIdHRwQ2FjaGluZ0NvbmZpZ0RlZmF1bHQsIC4uLmNvbmZpZyB9O1xyXG4gICAgfSBlbHNlIHtcclxuICAgICAgdGhpcy5jb25maWcgPSB7IC4uLk5nSHR0cENhY2hpbmdDb25maWdEZWZhdWx0IH07XHJcbiAgICB9XHJcbiAgICAvLyBzdGFydCBjYWNoZSBjbGVhblxyXG4gICAgdGhpcy5ydW5HYygpO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogUmV0dXJuIHRoZSBjb25maWdcclxuICAgKi9cclxuICBnZXRDb25maWcoKTogUmVhZG9ubHk8TmdIdHRwQ2FjaGluZ0NvbmZpZz4ge1xyXG4gICAgcmV0dXJuIHRoaXMuY29uZmlnO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogUmV0dXJuIHRoZSBxdWV1ZSBtYXBcclxuICAgKi9cclxuICBnZXRRdWV1ZSgpOiBSZWFkb25seTxNYXA8c3RyaW5nLCBPYnNlcnZhYmxlPEh0dHBFdmVudDxhbnk+Pj4+IHtcclxuICAgIHJldHVybiB0aGlzLnF1ZXVlO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogUmV0dXJuIHRoZSBjYWNoZSBzdG9yZVxyXG4gICAqL1xyXG4gIGdldFN0b3JlKCk6IFJlYWRvbmx5PE5nSHR0cENhY2hpbmdTdG9yYWdlSW50ZXJmYWNlPiB7XHJcbiAgICByZXR1cm4gdGhpcy5jb25maWcuc3RvcmU7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBSZXR1cm4gcmVzcG9uc2UgZnJvbSBjYWNoZVxyXG4gICAqL1xyXG4gIGdldEZyb21DYWNoZTxLLCBUPihyZXE6IEh0dHBSZXF1ZXN0PEs+KTogUmVhZG9ubHk8SHR0cFJlc3BvbnNlPFQ+PiB8IHVuZGVmaW5lZCB7XHJcbiAgICBjb25zdCBrZXk6IHN0cmluZyA9IHRoaXMuZ2V0S2V5KHJlcSk7XHJcbiAgICBjb25zdCBjYWNoZWQ6IE5nSHR0cENhY2hpbmdFbnRyeTxLLCBUPiB8IHVuZGVmaW5lZCA9IHRoaXMuY29uZmlnLnN0b3JlLmdldDxLLCBUPihrZXkpO1xyXG5cclxuICAgIGlmICghY2FjaGVkKSB7XHJcbiAgICAgIHJldHVybiB1bmRlZmluZWQ7XHJcbiAgICB9XHJcblxyXG4gICAgaWYgKHRoaXMuaXNFeHBpcmVkKGNhY2hlZCkpIHtcclxuICAgICAgdGhpcy5jbGVhckNhY2hlQnlLZXkoa2V5KTtcclxuICAgICAgcmV0dXJuIHVuZGVmaW5lZDtcclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4gdGhpcy5kZWVwRnJlZXplKGNhY2hlZC5yZXNwb25zZSk7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBBZGQgcmVzcG9uc2UgdG8gY2FjaGVcclxuICAgKi9cclxuICBhZGRUb0NhY2hlPEssIFQ+KHJlcTogSHR0cFJlcXVlc3Q8Sz4sIHJlczogSHR0cFJlc3BvbnNlPFQ+KTogYm9vbGVhbiB7XHJcbiAgICBjb25zdCBlbnRyeTogTmdIdHRwQ2FjaGluZ0VudHJ5PEssIFQ+ID0ge1xyXG4gICAgICB1cmw6IHJlcS51cmxXaXRoUGFyYW1zLFxyXG4gICAgICByZXNwb25zZTogcmVzLFxyXG4gICAgICByZXF1ZXN0OiByZXEsXHJcbiAgICAgIGFkZGVkVGltZTogRGF0ZS5ub3coKSxcclxuICAgICAgdmVyc2lvbjogdGhpcy5jb25maWcudmVyc2lvbixcclxuICAgIH07XHJcbiAgICBpZiAodGhpcy5pc1ZhbGlkKGVudHJ5KSkge1xyXG4gICAgICBjb25zdCBrZXk6IHN0cmluZyA9IHRoaXMuZ2V0S2V5KHJlcSk7XHJcbiAgICAgIHRoaXMuY29uZmlnLnN0b3JlLnNldChrZXksIGVudHJ5KTtcclxuICAgICAgcmV0dXJuIHRydWU7XHJcbiAgICB9XHJcbiAgICByZXR1cm4gZmFsc2U7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBEZWxldGUgcmVzcG9uc2UgZnJvbSBjYWNoZVxyXG4gICAqL1xyXG4gIGRlbGV0ZUZyb21DYWNoZTxLPihyZXE6IEh0dHBSZXF1ZXN0PEs+KTogYm9vbGVhbiB7XHJcbiAgICBjb25zdCBrZXk6IHN0cmluZyA9IHRoaXMuZ2V0S2V5KHJlcSk7XHJcbiAgICByZXR1cm4gdGhpcy5jbGVhckNhY2hlQnlLZXkoa2V5KTtcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIENsZWFyIHRoZSBjYWNoZVxyXG4gICAqL1xyXG4gIGNsZWFyQ2FjaGUoKTogdm9pZCB7XHJcbiAgICB0aGlzLmNvbmZpZy5zdG9yZS5jbGVhcigpO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogQ2xlYXIgdGhlIGNhY2hlIGJ5IGtleVxyXG4gICAqL1xyXG4gIGNsZWFyQ2FjaGVCeUtleShrZXk6IHN0cmluZyk6IGJvb2xlYW4ge1xyXG4gICAgcmV0dXJuIHRoaXMuY29uZmlnLnN0b3JlLmRlbGV0ZShrZXkpO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogQ2xlYXIgdGhlIGNhY2hlIGJ5IHJlZ2V4XHJcbiAgICovXHJcbiAgY2xlYXJDYWNoZUJ5UmVnZXg8SywgVD4ocmVnZXg6IFJlZ0V4cCk6IHZvaWQge1xyXG4gICAgdGhpcy5jb25maWcuc3RvcmUuZm9yRWFjaDxLLCBUPigoXzogTmdIdHRwQ2FjaGluZ0VudHJ5PEssIFQ+LCBrZXk6IHN0cmluZykgPT4ge1xyXG4gICAgICBpZiAocmVnZXgudGVzdChrZXkpKSB7XHJcbiAgICAgICAgdGhpcy5jbGVhckNhY2hlQnlLZXkoa2V5KTtcclxuICAgICAgfVxyXG4gICAgfSk7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBDbGVhciB0aGUgY2FjaGUgYnkgVEFHXHJcbiAgICovXHJcbiAgY2xlYXJDYWNoZUJ5VGFnPEssIFQ+KHRhZzogc3RyaW5nKTogdm9pZCB7XHJcbiAgICB0aGlzLmNvbmZpZy5zdG9yZS5mb3JFYWNoPEssIFQ+KChlbnRyeTogTmdIdHRwQ2FjaGluZ0VudHJ5PEssIFQ+LCBrZXk6IHN0cmluZykgPT4ge1xyXG4gICAgICBjb25zdCB0YWdIZWFkZXIgPSBlbnRyeS5yZXF1ZXN0LmhlYWRlcnMuZ2V0KE5nSHR0cENhY2hpbmdIZWFkZXJzLlRBRyk7XHJcbiAgICAgIGlmICh0YWdIZWFkZXIgJiYgdGFnSGVhZGVyLnNwbGl0KCcsJykuaW5jbHVkZXModGFnKSkge1xyXG4gICAgICAgIHRoaXMuY2xlYXJDYWNoZUJ5S2V5KGtleSk7XHJcbiAgICAgIH1cclxuICAgIH0pO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogUnVuIGdhcmJhZ2UgY29sbGVjdG9yIChkZWxldGUgZXhwaXJlZCBjYWNoZSBlbnRyeSlcclxuICAgKi9cclxuICBydW5HYzxLLCBUPigpOiBib29sZWFuIHtcclxuICAgIGlmICh0aGlzLmdjTG9jaykge1xyXG4gICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICB9XHJcbiAgICB0aGlzLmdjTG9jayA9IHRydWU7XHJcbiAgICB0aGlzLmNvbmZpZy5zdG9yZS5mb3JFYWNoPEssIFQ+KChlbnRyeTogTmdIdHRwQ2FjaGluZ0VudHJ5PEssIFQ+LCBrZXk6IHN0cmluZykgPT4ge1xyXG4gICAgICBpZiAodGhpcy5pc0V4cGlyZWQoZW50cnkpKSB7XHJcbiAgICAgICAgdGhpcy5jbGVhckNhY2hlQnlLZXkoa2V5KTtcclxuICAgICAgfVxyXG4gICAgfSk7XHJcbiAgICB0aGlzLmdjTG9jayA9IGZhbHNlO1xyXG4gICAgcmV0dXJuIHRydWU7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBSZXR1cm4gdHJ1ZSBpZiBjYWNoZSBlbnRyeSBpcyBleHBpcmVkXHJcbiAgICovXHJcbiAgaXNFeHBpcmVkPEssIFQ+KGVudHJ5OiBOZ0h0dHBDYWNoaW5nRW50cnk8SywgVD4pOiBib29sZWFuIHtcclxuICAgIC8vIGlmIHVzZXIgcHJvdmlkZSBjdXN0b20gbWV0aG9kLCB1c2UgaXRcclxuICAgIGlmICh0eXBlb2YgdGhpcy5jb25maWcuaXNFeHBpcmVkID09PSAnZnVuY3Rpb24nKSB7XHJcbiAgICAgIGNvbnN0IHJlc3VsdCA9IHRoaXMuY29uZmlnLmlzRXhwaXJlZChlbnRyeSk7XHJcbiAgICAgIC8vIGlmIHJlc3VsdCBpcyB1bmRlZmluZWQsIG5vcm1hbCBiZWhhdmlvdXIgaXMgcHJvdmlkZWRcclxuICAgICAgaWYgKHJlc3VsdCAhPT0gdW5kZWZpbmVkKSB7XHJcbiAgICAgICAgcmV0dXJuIHJlc3VsdDtcclxuICAgICAgfVxyXG4gICAgfVxyXG4gICAgLy8gY29uZmlnL2RlZmF1bHQgbGlmZXRpbWVcclxuICAgIGxldCBsaWZldGltZTogbnVtYmVyID0gdGhpcy5jb25maWcubGlmZXRpbWU7XHJcbiAgICAvLyByZXF1ZXN0IGhhcyBvd24gbGlmZXRpbWVcclxuICAgIGNvbnN0IGhlYWRlckxpZmV0aW1lID0gZW50cnkucmVxdWVzdC5oZWFkZXJzLmdldChOZ0h0dHBDYWNoaW5nSGVhZGVycy5MSUZFVElNRSk7XHJcbiAgICBpZiAoaGVhZGVyTGlmZXRpbWUpIHtcclxuICAgICAgbGlmZXRpbWUgPSAraGVhZGVyTGlmZXRpbWU7XHJcbiAgICB9XHJcbiAgICAvLyBuZXZlciBleHBpcmUgaWYgMFxyXG4gICAgaWYgKGxpZmV0aW1lID09PSAwKSB7XHJcbiAgICAgIHJldHVybiBmYWxzZTtcclxuICAgIH1cclxuICAgIC8vIHdyb25nIGxpZmV0aW1lXHJcbiAgICBpZiAobGlmZXRpbWUgPCAwIHx8IGlzTmFOKGxpZmV0aW1lKSkge1xyXG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ2xpZmV0aW1lIG11c3QgYmUgZ3JlYXRlciB0aGFuIG9yIGVxdWFsIDAnKTtcclxuICAgIH1cclxuICAgIHJldHVybiBlbnRyeS5hZGRlZFRpbWUgKyBsaWZldGltZSA8IERhdGUubm93KCk7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBSZXR1cm4gdHJ1ZSBpZiBjYWNoZSBlbnRyeSBpcyB2YWxpZCBmb3Igc3RvcmUgaW4gdGhlIGNhY2hlXHJcbiAgICovXHJcbiAgaXNWYWxpZDxLLCBUPihlbnRyeTogTmdIdHRwQ2FjaGluZ0VudHJ5PEssIFQ+KTogYm9vbGVhbiB7XHJcbiAgICAvLyBpZiB1c2VyIHByb3ZpZGUgY3VzdG9tIG1ldGhvZCwgdXNlIGl0XHJcbiAgICBpZiAodHlwZW9mIHRoaXMuY29uZmlnLmlzVmFsaWQgPT09ICdmdW5jdGlvbicpIHtcclxuICAgICAgY29uc3QgcmVzdWx0ID0gdGhpcy5jb25maWcuaXNWYWxpZChlbnRyeSk7XHJcbiAgICAgIC8vIGlmIHJlc3VsdCBpcyB1bmRlZmluZWQsIG5vcm1hbCBiZWhhdmlvdXIgaXMgcHJvdmlkZWRcclxuICAgICAgaWYgKHJlc3VsdCAhPT0gdW5kZWZpbmVkKSB7XHJcbiAgICAgICAgcmV0dXJuIHJlc3VsdDtcclxuICAgICAgfVxyXG4gICAgfVxyXG4gICAgLy8gZGlmZmVyZW50IHZlcnNpb25cclxuICAgIGlmICh0aGlzLmNvbmZpZy52ZXJzaW9uICE9PSBlbnRyeS52ZXJzaW9uKSB7XHJcbiAgICAgIHJldHVybiBmYWxzZTtcclxuICAgIH1cclxuICAgIHJldHVybiB0cnVlO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogUmV0dXJuIHRydWUgaWYgdGhlIHJlcXVlc3QgaXMgY2FjaGVhYmxlXHJcbiAgICovXHJcbiAgaXNDYWNoZWFibGU8Sz4ocmVxOiBIdHRwUmVxdWVzdDxLPik6IGJvb2xlYW4ge1xyXG4gICAgLy8gaWYgdXNlciBwcm92aWRlIGN1c3RvbSBtZXRob2QsIHVzZSBpdFxyXG4gICAgaWYgKHR5cGVvZiB0aGlzLmNvbmZpZy5pc0NhY2hlYWJsZSA9PT0gJ2Z1bmN0aW9uJykge1xyXG4gICAgICBjb25zdCByZXN1bHQgPSB0aGlzLmNvbmZpZy5pc0NhY2hlYWJsZShyZXEpO1xyXG4gICAgICAvLyBpZiByZXN1bHQgaXMgdW5kZWZpbmVkLCBub3JtYWwgYmVoYXZpb3VyIGlzIHByb3ZpZGVkXHJcbiAgICAgIGlmIChyZXN1bHQgIT09IHVuZGVmaW5lZCkge1xyXG4gICAgICAgIHJldHVybiByZXN1bHQ7XHJcbiAgICAgIH1cclxuICAgIH1cclxuICAgIC8vIHJlcXVlc3QgaGFzIGRpc2FsbG93IGNhY2hlIGhlYWRlclxyXG4gICAgaWYgKHJlcS5oZWFkZXJzLmhhcyhOZ0h0dHBDYWNoaW5nSGVhZGVycy5ESVNBTExPV19DQUNIRSkpIHtcclxuICAgICAgcmV0dXJuIGZhbHNlO1xyXG4gICAgfVxyXG4gICAgLy8gc3RyYXRlZ3kgaXMgZGlzYWxsb3cgYWxsLi4uXHJcbiAgICBpZiAodGhpcy5jb25maWcuY2FjaGVTdHJhdGVneSA9PT0gTmdIdHRwQ2FjaGluZ1N0cmF0ZWd5LkRJU0FMTE9XX0FMTCkge1xyXG4gICAgICAvLyByZXF1ZXN0IGlzbid0IGFsbG93ZWQgaWYgY29tZSB3aXRob3V0IGFsbG93IGhlYWRlclxyXG4gICAgICBpZiAoIXJlcS5oZWFkZXJzLmhhcyhOZ0h0dHBDYWNoaW5nSGVhZGVycy5BTExPV19DQUNIRSkpIHtcclxuICAgICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICAgIH1cclxuICAgIH1cclxuICAgIC8vIGlmIGFsbG93ZWQgbWV0aG9kIGlzIG9ubHkgQUxMLCBhbGxvdyBhbGwgaHR0cCBtZXRob3NcclxuICAgIGlmICh0aGlzLmNvbmZpZy5hbGxvd2VkTWV0aG9kLmxlbmd0aCA9PT0gMSkge1xyXG4gICAgICBpZiAodGhpcy5jb25maWcuYWxsb3dlZE1ldGhvZFswXSA9PT0gJ0FMTCcpIHtcclxuICAgICAgICByZXR1cm4gdHJ1ZTtcclxuICAgICAgfVxyXG4gICAgfVxyXG4gICAgLy8gcmVxdWVzdCBpcyBhbGxvd2VkIGlmIG1ldGhvZCBpcyBpbiBhbGxvd2VkTWV0aG9kXHJcbiAgICByZXR1cm4gdGhpcy5jb25maWcuYWxsb3dlZE1ldGhvZC5pbmRleE9mKHJlcS5tZXRob2QpICE9PSAtMTtcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIFJldHVybiB0aGUgY2FjaGUga2V5LlxyXG4gICAqIERlZmF1bHQga2V5IGlzIGh0dHAgbWV0aG9kIHBsdXMgdXJsIHdpdGggcXVlcnkgcGFyYW1ldGVycywgZWcuOlxyXG4gICAqIGBHRVRAaHR0cHM6Ly9naXRodWIuY29tL25pZ3Jvc2ltb25lL25nLWh0dHAtY2FjaGluZ2BcclxuICAgKi9cclxuICBnZXRLZXk8Sz4ocmVxOiBIdHRwUmVxdWVzdDxLPik6IHN0cmluZyB7XHJcbiAgICAvLyBpZiB1c2VyIHByb3ZpZGUgY3VzdG9tIG1ldGhvZCwgdXNlIGl0XHJcbiAgICBpZiAodHlwZW9mIHRoaXMuY29uZmlnLmdldEtleSA9PT0gJ2Z1bmN0aW9uJykge1xyXG4gICAgICBjb25zdCByZXN1bHQgPSB0aGlzLmNvbmZpZy5nZXRLZXkocmVxKTtcclxuICAgICAgLy8gaWYgcmVzdWx0IGlzIHVuZGVmaW5lZCwgbm9ybWFsIGJlaGF2aW91ciBpcyBwcm92aWRlZFxyXG4gICAgICBpZiAocmVzdWx0ICE9PSB1bmRlZmluZWQpIHtcclxuICAgICAgICByZXR1cm4gcmVzdWx0O1xyXG4gICAgICB9XHJcbiAgICB9XHJcbiAgICAvLyBkZWZhdWx0IGtleSBpcyByZXEubWV0aG9kIHBsdXMgdXJsIHdpdGggcXVlcnkgcGFyYW1ldGVyc1xyXG4gICAgcmV0dXJuIHJlcS5tZXRob2QgKyAnQCcgKyByZXEudXJsV2l0aFBhcmFtcztcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIFJldHVybiBvYnNlcnZhYmxlIGZyb20gY2FjaGVcclxuICAgKi9cclxuICBnZXRGcm9tUXVldWU8SywgVD4ocmVxOiBIdHRwUmVxdWVzdDxLPik6IE9ic2VydmFibGU8SHR0cEV2ZW50PFQ+PiB8IHVuZGVmaW5lZCB7XHJcbiAgICBjb25zdCBrZXk6IHN0cmluZyA9IHRoaXMuZ2V0S2V5KHJlcSk7XHJcbiAgICBjb25zdCBjYWNoZWQ6IE9ic2VydmFibGU8SHR0cEV2ZW50PFQ+PiB8IHVuZGVmaW5lZCA9IHRoaXMucXVldWUuZ2V0KGtleSk7XHJcblxyXG4gICAgaWYgKCFjYWNoZWQpIHtcclxuICAgICAgcmV0dXJuIHVuZGVmaW5lZDtcclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4gY2FjaGVkO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogQWRkIG9ic2VydmFibGUgdG8gY2FjaGVcclxuICAgKi9cclxuICBhZGRUb1F1ZXVlPEssIFQ+KHJlcTogSHR0cFJlcXVlc3Q8Sz4sIG9iczogT2JzZXJ2YWJsZTxIdHRwRXZlbnQ8VD4+KTogdm9pZCB7XHJcbiAgICBjb25zdCBrZXk6IHN0cmluZyA9IHRoaXMuZ2V0S2V5KHJlcSk7XHJcbiAgICB0aGlzLnF1ZXVlLnNldChrZXksIG9icyk7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBEZWxldGUgb2JzZXJ2YWJsZSBmcm9tIGNhY2hlXHJcbiAgICovXHJcbiAgZGVsZXRlRnJvbVF1ZXVlPEs+KHJlcTogSHR0cFJlcXVlc3Q8Sz4pOiBib29sZWFuIHtcclxuICAgIGNvbnN0IGtleTogc3RyaW5nID0gdGhpcy5nZXRLZXkocmVxKTtcclxuICAgIHJldHVybiB0aGlzLnF1ZXVlLmRlbGV0ZShrZXkpO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogUmVjdXJzaXZlbHkgT2JqZWN0LmZyZWV6ZSBzaW1wbGUgSmF2YXNjcmlwdCBzdHJ1Y3R1cmVzIGNvbnNpc3Rpbmcgb2YgcGxhaW4gb2JqZWN0cywgYXJyYXlzLCBhbmQgcHJpbWl0aXZlcy5cclxuICAgKiBNYWtlIHRoZSBkYXRhIGltbXV0YWJsZS5cclxuICAgKiBAcmV0dXJucyBpbW11dGFibGUgb2JqZWN0XHJcbiAgICovXHJcbiAgcHJpdmF0ZSBkZWVwRnJlZXplPFM+KG9iamVjdDogUyk6IFJlYWRvbmx5PFM+IHtcclxuICAgIC8vIE5vIGZyZWV6aW5nIGluIHByb2R1Y3Rpb24gKGZvciBiZXR0ZXIgcGVyZm9ybWFuY2UpLlxyXG4gICAgaWYgKCF0aGlzLmRldk1vZGUgfHwgIW9iamVjdCB8fCB0eXBlb2Ygb2JqZWN0ICE9PSAnb2JqZWN0Jykge1xyXG4gICAgICByZXR1cm4gb2JqZWN0IGFzIFJlYWRvbmx5PFM+O1xyXG4gICAgfVxyXG5cclxuICAgIC8vIFdoZW4gYWxyZWFkeSBmcm96ZW4sIHdlIGFzc3VtZSBpdHMgY2hpbGRyZW4gYXJlIGZyb3plbiAoZm9yIGJldHRlciBwZXJmb3JtYW5jZSkuXHJcbiAgICAvLyBUaGlzIHNob3VsZCBiZSB0cnVlIGlmIHlvdSBhbHdheXMgdXNlIGBkZWVwRnJlZXplYCB0byBmcmVlemUgb2JqZWN0cy5cclxuICAgIC8vXHJcbiAgICAvLyBOb3RlIHRoYXQgT2JqZWN0LmlzRnJvemVuIHdpbGwgYWxzbyByZXR1cm4gYHRydWVgIGZvciBwcmltaXRpdmVzIChudW1iZXJzLFxyXG4gICAgLy8gc3RyaW5ncywgYm9vbGVhbnMsIHVuZGVmaW5lZCwgbnVsbCksIHNvIHRoZXJlIGlzIG5vIG5lZWQgdG8gY2hlY2sgZm9yXHJcbiAgICAvLyB0aG9zZSBleHBsaWNpdGx5LlxyXG4gICAgaWYgKE9iamVjdC5pc0Zyb3plbihvYmplY3QpKSB7XHJcbiAgICAgIHJldHVybiBvYmplY3QgYXMgUmVhZG9ubHk8Uz47XHJcbiAgICB9XHJcblxyXG4gICAgLy8gQXQgdGhpcyBwb2ludCB3ZSBrbm93IHRoYXQgd2UncmUgZGVhbGluZyB3aXRoIGVpdGhlciBhbiBhcnJheSBvciBwbGFpbiBvYmplY3QsIHNvXHJcbiAgICAvLyBqdXN0IGZyZWV6ZSBpdCBhbmQgcmVjdXJzZSBvbiBpdHMgdmFsdWVzLlxyXG4gICAgT2JqZWN0LmZyZWV6ZShvYmplY3QpO1xyXG4gICAgT2JqZWN0LmtleXMob2JqZWN0KS5mb3JFYWNoKGtleSA9PiB0aGlzLmRlZXBGcmVlemUoKG9iamVjdCBhcyBhbnkpW2tleV0pKTtcclxuXHJcbiAgICByZXR1cm4gb2JqZWN0IGFzIFJlYWRvbmx5PFM+O1xyXG4gIH1cclxufVxyXG4iXX0=
|