@sochdb/sochdb 0.5.0 → 0.5.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.
@@ -18,6 +18,51 @@
18
18
  Object.defineProperty(exports, "__esModule", { value: true });
19
19
  exports.Namespace = exports.Collection = exports.DistanceMetric = exports.CollectionExistsError = exports.CollectionNotFoundError = exports.NamespaceExistsError = exports.NamespaceNotFoundError = void 0;
20
20
  const errors_1 = require("./errors");
21
+ let NativeHnsw = null;
22
+ try {
23
+ const koffi = require('koffi');
24
+ const { findLibrary } = require('./embedded/ffi/library-finder');
25
+ const libraryPath = findLibrary();
26
+ const lib = koffi.load(libraryPath);
27
+ // Define opaque pointer
28
+ const HnswIndexPtr = koffi.pointer('HnswIndexPtr3', koffi.opaque());
29
+ // Define CSearchResult struct for native search
30
+ const CSearchResultStruct = koffi.struct('CSearchResult3', {
31
+ id_lo: 'uint64',
32
+ id_hi: 'uint64',
33
+ distance: 'float'
34
+ });
35
+ NativeHnsw = {
36
+ lib,
37
+ koffi,
38
+ HnswIndexPtr,
39
+ CSearchResultStruct,
40
+ hnsw_new: lib.func('hnsw_new', HnswIndexPtr, ['size_t', 'size_t', 'size_t']),
41
+ hnsw_free: lib.func('hnsw_free', 'void', [HnswIndexPtr]),
42
+ hnsw_insert_batch: lib.func('hnsw_insert_batch', 'int', [
43
+ HnswIndexPtr,
44
+ 'uint64*', // ids
45
+ 'float*', // vectors (flat)
46
+ 'size_t', // num_vectors
47
+ 'size_t' // dimension
48
+ ]),
49
+ hnsw_len: lib.func('hnsw_len', 'size_t', [HnswIndexPtr]),
50
+ hnsw_set_ef_search: lib.func('hnsw_set_ef_search', 'void', [HnswIndexPtr, 'size_t']),
51
+ hnsw_search: lib.func('hnsw_search', 'int', [
52
+ HnswIndexPtr,
53
+ 'float*', // query vector
54
+ 'size_t', // query_len
55
+ 'size_t', // k
56
+ koffi.pointer(CSearchResultStruct), // results_out pointer
57
+ koffi.pointer('size_t') // num_results_out pointer
58
+ ]),
59
+ };
60
+ console.log('[SochDB] Native HNSW bindings loaded (batch insert + search)');
61
+ }
62
+ catch (e) {
63
+ // Native bindings not available - will use JS fallback
64
+ NativeHnsw = null;
65
+ }
21
66
  class NamespaceNotFoundError extends errors_1.SochDBError {
22
67
  constructor(namespace) {
23
68
  super(`Namespace not found: ${namespace}`);
@@ -156,8 +201,39 @@ class Collection {
156
201
  this.namespace = namespace;
157
202
  this.name = name;
158
203
  this.config = config;
204
+ this.nativeIndexPtr = null;
159
205
  this._indexReady = false;
206
+ this.nativeIdCounter = 0; // Counter for native HNSW numeric IDs
207
+ this.nativeIdToStringId = new Map(); // Map numeric ID -> string ID
160
208
  this.vectorIndex = new VectorIndex(config.dimension || 384, config.metric || DistanceMetric.Cosine);
209
+ // Try to create native HNSW index for high-performance operations
210
+ if (NativeHnsw) {
211
+ try {
212
+ const dimension = config.dimension || 384;
213
+ const maxConnections = config.hnswM || 16;
214
+ const efConstruction = config.hnswEfConstruction || 100;
215
+ this.nativeIndexPtr = NativeHnsw.hnsw_new(dimension, maxConnections, efConstruction);
216
+ if (this.nativeIndexPtr) {
217
+ // Set high ef_search for good recall (can be tuned via setEfSearch)
218
+ NativeHnsw.hnsw_set_ef_search(this.nativeIndexPtr, 500);
219
+ console.log(`[SochDB] Native HNSW index created: dim=${dimension}, M=${maxConnections}, efC=${efConstruction}, efS=500`);
220
+ }
221
+ }
222
+ catch (e) {
223
+ console.warn('[SochDB] Native HNSW creation failed:', e.message);
224
+ this.nativeIndexPtr = null;
225
+ }
226
+ }
227
+ }
228
+ /**
229
+ * Set ef_search parameter for HNSW search (controls recall vs speed tradeoff)
230
+ * Higher values = better recall but slower search
231
+ * @param efSearch - Typically 100-1000, default is 500
232
+ */
233
+ setEfSearch(efSearch) {
234
+ if (NativeHnsw && this.nativeIndexPtr) {
235
+ NativeHnsw.hnsw_set_ef_search(this.nativeIndexPtr, efSearch);
236
+ }
161
237
  }
162
238
  /**
163
239
  * Insert a vector with optional metadata
@@ -178,19 +254,75 @@ class Collection {
178
254
  await this.db.put(Buffer.from(key), Buffer.from(JSON.stringify(data)));
179
255
  // SYNCHRONOUSLY add to in-memory index
180
256
  this.vectorIndex.add(vectorId, vector, metadata);
257
+ // Also add to native HNSW index if available (for single inserts)
258
+ if (NativeHnsw && this.nativeIndexPtr) {
259
+ try {
260
+ const dimension = this.config.dimension || vector.length;
261
+ // Use auto-incrementing counter for native HNSW (it needs numeric IDs)
262
+ const numericId = this.nativeIdCounter++;
263
+ this.nativeIdToStringId.set(numericId, vectorId); // Store mapping
264
+ const idArray = new BigUint64Array([BigInt(numericId)]);
265
+ const vectorArray = new Float32Array(vector);
266
+ // Args: ptr, ids, vectors, num_vectors, dimension
267
+ NativeHnsw.hnsw_insert_batch(this.nativeIndexPtr, idArray, vectorArray, 1, dimension);
268
+ }
269
+ catch (e) {
270
+ // Fallback to JS index only (already added above)
271
+ }
272
+ }
181
273
  return vectorId;
182
274
  }
183
275
  /**
184
- * Insert multiple vectors
185
- * All vectors are indexed synchronously after insertion
276
+ * Insert multiple vectors using NATIVE HNSW batch insert when available
277
+ * This is the OPTIMIZED path - uses FFI batch insert for ~100x speedup
186
278
  */
187
279
  async insertMany(vectors, metadatas, ids) {
188
- const resultIds = [];
280
+ if (vectors.length === 0) {
281
+ return [];
282
+ }
283
+ const resultIds = ids || vectors.map((_, i) => i.toString());
284
+ const dimension = this.config.dimension || vectors[0].length;
285
+ // If native index is available, use batch insert
286
+ if (NativeHnsw && this.nativeIndexPtr) {
287
+ try {
288
+ console.log(`[SochDB] Using NATIVE batch insert for ${vectors.length} vectors...`);
289
+ const startTime = performance.now();
290
+ // Convert IDs to numeric (u64) using counter and store mapping
291
+ const numericIds = new BigUint64Array(resultIds.length);
292
+ for (let i = 0; i < resultIds.length; i++) {
293
+ const numericId = this.nativeIdCounter++;
294
+ numericIds[i] = BigInt(numericId);
295
+ this.nativeIdToStringId.set(numericId, resultIds[i]);
296
+ }
297
+ // Flatten vectors into contiguous Float32Array
298
+ const flatVectors = new Float32Array(vectors.length * dimension);
299
+ for (let i = 0; i < vectors.length; i++) {
300
+ flatVectors.set(vectors[i], i * dimension);
301
+ }
302
+ // Single FFI call to native batch insert
303
+ const result = NativeHnsw.hnsw_insert_batch(this.nativeIndexPtr, numericIds, flatVectors, vectors.length, dimension);
304
+ const elapsed = (performance.now() - startTime) / 1000;
305
+ const indexSize = NativeHnsw.hnsw_len(this.nativeIndexPtr);
306
+ console.log(`[SochDB] Native batch insert: result=${result}, index_size=${indexSize}, time=${elapsed.toFixed(3)}s (${(vectors.length / elapsed).toFixed(0)} vec/sec)`);
307
+ if (result < 0) {
308
+ throw new Error(`Native batch insert failed with error code ${result}`);
309
+ }
310
+ // Also add to JS index for metadata lookups
311
+ for (let i = 0; i < resultIds.length; i++) {
312
+ this.vectorIndex.add(resultIds[i], vectors[i], metadatas ? metadatas[i] : undefined);
313
+ }
314
+ return resultIds;
315
+ }
316
+ catch (e) {
317
+ console.warn('[SochDB] Native batch insert failed, falling back to sequential:', e.message);
318
+ }
319
+ }
320
+ // Fallback: sequential insert (slow path)
321
+ console.log(`[SochDB] Using SEQUENTIAL insert for ${vectors.length} vectors (slow path)...`);
189
322
  for (let i = 0; i < vectors.length; i++) {
190
- const id = ids ? ids[i] : undefined;
323
+ const id = resultIds[i];
191
324
  const metadata = metadatas ? metadatas[i] : undefined;
192
- const resultId = await this.insert(vectors[i], metadata, id);
193
- resultIds.push(resultId);
325
+ await this.insert(vectors[i], metadata, id);
194
326
  }
195
327
  return resultIds;
196
328
  }
@@ -223,9 +355,53 @@ class Collection {
223
355
  }
224
356
  /**
225
357
  * Search for similar vectors
226
- * Uses synchronous in-memory index - no delay
358
+ * Uses NATIVE HNSW search (O(log N)) when available, falls back to JS brute-force
227
359
  */
228
360
  async search(request) {
361
+ const k = request.k;
362
+ const queryVector = request.queryVector;
363
+ // Try native HNSW search first (much faster for large datasets)
364
+ if (NativeHnsw && this.nativeIndexPtr && NativeHnsw.hnsw_search) {
365
+ try {
366
+ const dimension = queryVector.length;
367
+ // Prepare query as Float32Array
368
+ const queryArray = new Float32Array(queryVector);
369
+ // Allocate output buffers
370
+ // CSearchResult: { id_lo: uint64, id_hi: uint64, distance: float }
371
+ // Size: 8 + 8 + 4 = 20 bytes per result, aligned to 24 bytes
372
+ const resultSize = 24;
373
+ const resultsBuffer = Buffer.alloc(resultSize * k);
374
+ const numResultsBuffer = Buffer.alloc(8); // size_t
375
+ // Call native search
376
+ const result = NativeHnsw.hnsw_search(this.nativeIndexPtr, queryArray, dimension, k, resultsBuffer, numResultsBuffer);
377
+ if (result === 0) {
378
+ const numResults = numResultsBuffer.readBigUInt64LE(0);
379
+ const nativeResults = [];
380
+ // Read results from buffer
381
+ for (let i = 0; i < Math.min(Number(numResults), k); i++) {
382
+ const offset = i * resultSize;
383
+ const id_lo = resultsBuffer.readBigUInt64LE(offset);
384
+ const distance = resultsBuffer.readFloatLE(offset + 16);
385
+ // Map numeric ID back to string ID
386
+ const numericId = Number(id_lo);
387
+ const stringId = this.nativeIdToStringId.get(numericId) || id_lo.toString();
388
+ const data = this.vectorIndex.vectors?.get(stringId);
389
+ nativeResults.push({
390
+ id: stringId,
391
+ score: 1 - distance, // Convert distance to similarity
392
+ vector: request.includeMetadata && data ? data.vector : undefined,
393
+ metadata: request.includeMetadata && data ? data.metadata : undefined,
394
+ });
395
+ }
396
+ return nativeResults;
397
+ }
398
+ }
399
+ catch (e) {
400
+ // Fall back to JS search on error
401
+ console.warn('[SochDB] Native search failed:', e.message);
402
+ }
403
+ }
404
+ // Fallback: JS brute-force search
229
405
  // Auto-rebuild index if empty but there might be data
230
406
  if (!this.isIndexReady && this.vectorIndex.size() === 0) {
231
407
  await this.rebuildIndex();
@@ -360,17 +536,44 @@ class Namespace {
360
536
  */
361
537
  async deleteCollection(name) {
362
538
  const metadataKey = `_collection/${this.name}/${name}/metadata`;
363
- const prefix = `_collection/${this.name}/${name}/`;
364
- // TODO: Delete all keys with prefix
365
- await this.db.delete(Buffer.from(metadataKey));
539
+ const prefix = Buffer.from(`_collection/${this.name}/${name}/`);
540
+ // Delete all keys with this collection prefix (vectors, metadata, etc.)
541
+ try {
542
+ const toDelete = [];
543
+ for await (const [keyBuf] of this.db.scanPrefix(prefix)) {
544
+ toDelete.push(keyBuf);
545
+ }
546
+ for (const key of toDelete) {
547
+ await this.db.delete(key);
548
+ }
549
+ }
550
+ catch {
551
+ // If scanPrefix fails, fall back to just deleting the metadata key
552
+ await this.db.delete(Buffer.from(metadataKey));
553
+ }
366
554
  return true;
367
555
  }
368
556
  /**
369
557
  * List all collections in this namespace
370
558
  */
371
559
  async listCollections() {
372
- // TODO: Implement efficient listing with range queries
373
- return [];
560
+ const prefix = Buffer.from(`_collection/${this.name}/`);
561
+ const collections = new Set();
562
+ try {
563
+ for await (const [keyBuf] of this.db.scanPrefix(prefix)) {
564
+ const key = keyBuf.toString();
565
+ // Key format: _collection/{namespace}/{collectionName}/...
566
+ const afterPrefix = key.substring(`_collection/${this.name}/`.length);
567
+ const collectionName = afterPrefix.split('/')[0];
568
+ if (collectionName) {
569
+ collections.add(collectionName);
570
+ }
571
+ }
572
+ }
573
+ catch {
574
+ // Scan not available
575
+ }
576
+ return Array.from(collections);
374
577
  }
375
578
  getName() {
376
579
  return this.name;
@@ -380,4 +583,4 @@ class Namespace {
380
583
  }
381
584
  }
382
585
  exports.Namespace = Namespace;
383
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmFtZXNwYWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL25hbWVzcGFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7Ozs7Ozs7Ozs7OztHQWVHOzs7QUFFSCxxQ0FBc0Q7QUFhdEQsTUFBYSxzQkFBdUIsU0FBUSxvQkFBVztJQUNyRCxZQUFZLFNBQWlCO1FBQzNCLEtBQUssQ0FBQyx3QkFBd0IsU0FBUyxFQUFFLENBQUMsQ0FBQztRQUMzQyxJQUFJLENBQUMsSUFBSSxHQUFHLHdCQUF3QixDQUFDO0lBQ3ZDLENBQUM7Q0FDRjtBQUxELHdEQUtDO0FBRUQsTUFBYSxvQkFBcUIsU0FBUSxvQkFBVztJQUNuRCxZQUFZLFNBQWlCO1FBQzNCLEtBQUssQ0FBQyw2QkFBNkIsU0FBUyxFQUFFLENBQUMsQ0FBQztRQUNoRCxJQUFJLENBQUMsSUFBSSxHQUFHLHNCQUFzQixDQUFDO0lBQ3JDLENBQUM7Q0FDRjtBQUxELG9EQUtDO0FBRUQsTUFBYSx1QkFBd0IsU0FBUSxvQkFBVztJQUN0RCxZQUFZLFVBQWtCO1FBQzVCLEtBQUssQ0FBQyx5QkFBeUIsVUFBVSxFQUFFLENBQUMsQ0FBQztRQUM3QyxJQUFJLENBQUMsSUFBSSxHQUFHLHlCQUF5QixDQUFDO0lBQ3hDLENBQUM7Q0FDRjtBQUxELDBEQUtDO0FBRUQsTUFBYSxxQkFBc0IsU0FBUSxvQkFBVztJQUNwRCxZQUFZLFVBQWtCO1FBQzVCLEtBQUssQ0FBQyw4QkFBOEIsVUFBVSxFQUFFLENBQUMsQ0FBQztRQUNsRCxJQUFJLENBQUMsSUFBSSxHQUFHLHVCQUF1QixDQUFDO0lBQ3RDLENBQUM7Q0FDRjtBQUxELHNEQUtDO0FBRUQsK0VBQStFO0FBQy9FLDJCQUEyQjtBQUMzQiwrRUFBK0U7QUFFL0UsSUFBWSxjQUlYO0FBSkQsV0FBWSxjQUFjO0lBQ3hCLG1DQUFpQixDQUFBO0lBQ2pCLHlDQUF1QixDQUFBO0lBQ3ZCLG9DQUFrQixDQUFBO0FBQ3BCLENBQUMsRUFK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QUFNLEdBQUcsZUFBZSxJQUFJLENBQUMsSUFBSSxJQUFJLElBQUksR0FBRyxDQUFDO1FBRW5ELG9DQUFvQztRQUNwQyxNQUFNLElBQUksQ0FBQyxFQUFFLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQztRQUUvQyxPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7SUFFRDs7T0FFRztJQUNILEtBQUssQ0FBQyxlQUFlO1FBQ25CLHVEQUF1RDtRQUN2RCxPQUFPLEVBQUUsQ0FBQztJQUNaLENBQUM7SUFFRCxPQUFPO1FBQ0wsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDO0lBQ25CLENBQUM7SUFFRCxTQUFTO1FBQ1AsT0FBTyxFQUFFLEdBQUcsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDO0lBQzVCLENBQUM7Q0FDRjtBQTFGRCw4QkEwRkMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNvY2hEQiBOYW1lc3BhY2UgQVBJXG4gKiBcbiAqIFByb3ZpZGVzIHR5cGUtc2FmZSBuYW1lc3BhY2UgaXNvbGF0aW9uIHdpdGggZmlyc3QtY2xhc3MgbmFtZXNwYWNlIGhhbmRsZXMuXG4gKiBcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBpbXBvcnQgeyBEYXRhYmFzZSB9IGZyb20gJ0Bzb2NoZGIvc29jaGRiJztcbiAqIFxuICogY29uc3QgZGIgPSBhd2FpdCBEYXRhYmFzZS5vcGVuKCcuL215ZGInKTtcbiAqIGNvbnN0IG5zID0gYXdhaXQgZGIuY3JlYXRlTmFtZXNwYWNlKCd0ZW5hbnRfMTIzJyk7XG4gKiBjb25zdCBjb2xsZWN0aW9uID0gYXdhaXQgbnMuY3JlYXRlQ29sbGVjdGlvbignZG9jdW1lbnRzJywgeyBkaW1lbnNpb246IDM4NCB9KTtcbiAqIGF3YWl0IGNvbGxlY3Rpb24uaW5zZXJ0KFsxLjAsIDIuMCwgLi4uXSwgeyBzb3VyY2U6ICd3ZWInIH0pO1xuICogY29uc3QgcmVzdWx0cyA9IGF3YWl0IGNvbGxlY3Rpb24uc2VhcmNoKHF1ZXJ5VmVjdG9yLCAxMCk7XG4gKiBgYGBcbiAqL1xuXG5pbXBvcnQgeyBTb2NoREJFcnJvciwgRGF0YWJhc2VFcnJvciB9IGZyb20gJy4vZXJyb3JzJztcblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gTmFtZXNwYWNlIENvbmZpZ3VyYXRpb25cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0IGludGVyZmFjZSBOYW1lc3BhY2VDb25maWcge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRpc3BsYXlOYW1lPzogc3RyaW5nO1xuICBsYWJlbHM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuICByZWFkT25seT86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjbGFzcyBOYW1lc3BhY2VOb3RGb3VuZEVycm9yIGV4dGVuZHMgU29jaERCRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihuYW1lc3BhY2U6IHN0cmluZykge1xuICAgIHN1cGVyKGBOYW1lc3BhY2Ugbm90IGZvdW5kOiAke25hbWVzcGFjZX1gKTtcbiAgICB0aGlzLm5hbWUgPSAnTmFtZXNwYWNlTm90Rm91bmRFcnJvcic7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIE5hbWVzcGFjZUV4aXN0c0Vycm9yIGV4dGVuZHMgU29jaERCRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihuYW1lc3BhY2U6IHN0cmluZykge1xuICAgIHN1cGVyKGBOYW1lc3BhY2UgYWxyZWFkeSBleGlzdHM6ICR7bmFtZXNwYWNlfWApO1xuICAgIHRoaXMubmFtZSA9ICdOYW1lc3BhY2VFeGlzdHNFcnJvcic7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIENvbGxlY3Rpb25Ob3RGb3VuZEVycm9yIGV4dGVuZHMgU29jaERCRXJyb3Ige1xuICBjb25zdHJ1Y3Rvcihjb2xsZWN0aW9uOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgQ29sbGVjdGlvbiBub3QgZm91bmQ6ICR7Y29sbGVjdGlvbn1gKTtcbiAgICB0aGlzLm5hbWUgPSAnQ29sbGVjdGlvbk5vdEZvdW5kRXJyb3InO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBDb2xsZWN0aW9uRXhpc3RzRXJyb3IgZXh0ZW5kcyBTb2NoREJFcnJvciB7XG4gIGNvbnN0cnVjdG9yKGNvbGxlY3Rpb246IHN0cmluZykge1xuICAgIHN1cGVyKGBDb2xsZWN0aW9uIGFscmVhZHkgZXhpc3RzOiAke2NvbGxlY3Rpb259YCk7XG4gICAgdGhpcy5uYW1lID0gJ0NvbGxlY3Rpb25FeGlzdHNFcnJvcic7XG4gIH1cbn1cblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gQ29sbGVjdGlvbiBDb25maWd1cmF0aW9uXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCBlbnVtIERpc3RhbmNlTWV0cmljIHtcbiAgQ29zaW5lID0gJ2Nvc2luZScsXG4gIEV1Y2xpZGVhbiA9ICdldWNsaWRlYW4nLFxuICBEb3RQcm9kdWN0ID0gJ2RvdCcsXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29sbGVjdGlvbkNvbmZpZyB7XG4gIG5hbWU6IHN0cmluZztcbiAgZGltZW5zaW9uPzogbnVtYmVyO1xuICBtZXRyaWM/OiBEaXN0YW5jZU1ldHJpYztcbiAgaW5kZXhlZD86IGJvb2xlYW47XG4gIGhuc3dNPzogbnVtYmVyO1xuICBobnN3RWZDb25zdHJ1Y3Rpb24/OiBudW1iZXI7XG4gIG1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55Pjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTZWFyY2hSZXF1ZXN0IHtcbiAgcXVlcnlWZWN0b3I6IG51bWJlcltdO1xuICBrOiBudW1iZXI7XG4gIGZpbHRlcj86IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIGluY2x1ZGVNZXRhZGF0YT86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VhcmNoUmVzdWx0IHtcbiAgaWQ6IHN0cmluZztcbiAgc2NvcmU6IG51bWJlcjtcbiAgdmVjdG9yPzogbnVtYmVyW107XG4gIG1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55Pjtcbn1cblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gQ29sbGVjdGlvbiBIYW5kbGVcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBJbi1tZW1vcnkgdmVjdG9yIGluZGV4IGZvciBzeW5jaHJvbm91cyBzZWFyY2hcbiAqIFVzZXMgYSBzaW1wbGUgYnV0IGVmZmljaWVudCBhcHByb2FjaCBmb3Igc21hbGwtbWVkaXVtIGRhdGFzZXRzXG4gKi9cbmNsYXNzIFZlY3RvckluZGV4IHtcbiAgcHJpdmF0ZSB2ZWN0b3JzOiBNYXA8c3RyaW5nLCB7IHZlY3RvcjogbnVtYmVyW107IG1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55PiB9PiA9IG5ldyBNYXAoKTtcbiAgcHJpdmF0ZSBkaW1lbnNpb246IG51bWJlcjtcbiAgcHJpdmF0ZSBtZXRyaWM6IERpc3RhbmNlTWV0cmljO1xuXG4gIGNvbnN0cnVjdG9yKGRpbWVuc2lvbjogbnVtYmVyLCBtZXRyaWM6IERpc3RhbmNlTWV0cmljID0gRGlzdGFuY2VNZXRyaWMuQ29zaW5lKSB7XG4gICAgdGhpcy5kaW1lbnNpb24gPSBkaW1lbnNpb247XG4gICAgdGhpcy5tZXRyaWMgPSBtZXRyaWM7XG4gIH1cblxuICBhZGQoaWQ6IHN0cmluZywgdmVjdG9yOiBudW1iZXJbXSwgbWV0YWRhdGE/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+KTogdm9pZCB7XG4gICAgaWYgKHZlY3Rvci5sZW5ndGggIT09IHRoaXMuZGltZW5zaW9uKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYFZlY3RvciBkaW1lbnNpb24gbWlzbWF0Y2g6IGV4cGVjdGVkICR7dGhpcy5kaW1lbnNpb259LCBnb3QgJHt2ZWN0b3IubGVuZ3RofWApO1xuICAgIH1cbiAgICB0aGlzLnZlY3RvcnMuc2V0KGlkLCB7IHZlY3RvciwgbWV0YWRhdGEgfSk7XG4gIH1cblxuICByZW1vdmUoaWQ6IHN0cmluZyk6IHZvaWQge1xuICAgIHRoaXMudmVjdG9ycy5kZWxldGUoaWQpO1xuICB9XG5cbiAgc2VhcmNoKHF1ZXJ5VmVjdG9yOiBudW1iZXJbXSwgazogbnVtYmVyLCBmaWx0ZXI/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+KTogU2VhcmNoUmVzdWx0W10ge1xuICAgIGNvbnN0IHJlc3VsdHM6IFNlYXJjaFJlc3VsdFtdID0gW107XG5cbiAgICBmb3IgKGNvbnN0IFtpZCwgZGF0YV0gb2YgdGhpcy52ZWN0b3JzKSB7XG4gICAgICAvLyBBcHBseSBtZXRhZGF0YSBmaWx0ZXIgaWYgcHJvdmlkZWRcbiAgICAgIGlmIChmaWx0ZXIgJiYgIXRoaXMubWF0Y2hlc0ZpbHRlcihkYXRhLm1ldGFkYXRhLCBmaWx0ZXIpKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBzY29yZSA9IHRoaXMuY2FsY3VsYXRlU2ltaWxhcml0eShxdWVyeVZlY3RvciwgZGF0YS52ZWN0b3IpO1xuICAgICAgcmVzdWx0cy5wdXNoKHtcbiAgICAgICAgaWQsXG4gICAgICAgIHNjb3JlLFxuICAgICAgICB2ZWN0b3I6IGRhdGEudmVjdG9yLFxuICAgICAgICBtZXRhZGF0YTogZGF0YS5tZXRhZGF0YSxcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIC8vIFNvcnQgYnkgc2NvcmUgKGhpZ2hlciBpcyBiZXR0ZXIgZm9yIHNpbWlsYXJpdHkpXG4gICAgcmVzdWx0cy5zb3J0KChhLCBiKSA9PiBiLnNjb3JlIC0gYS5zY29yZSk7XG5cbiAgICByZXR1cm4gcmVzdWx0cy5zbGljZSgwLCBrKTtcbiAgfVxuXG4gIHNpemUoKTogbnVtYmVyIHtcbiAgICByZXR1cm4gdGhpcy52ZWN0b3JzLnNpemU7XG4gIH1cblxuICBwcml2YXRlIGNhbGN1bGF0ZVNpbWlsYXJpdHkoYTogbnVtYmVyW10sIGI6IG51bWJlcltdKTogbnVtYmVyIHtcbiAgICBzd2l0Y2ggKHRoaXMubWV0cmljKSB7XG4gICAgICBjYXNlIERpc3RhbmNlTWV0cmljLkNvc2luZTpcbiAgICAgICAgcmV0dXJuIHRoaXMuY29zaW5lU2ltaWxhcml0eShhLCBiKTtcbiAgICAgIGNhc2UgRGlzdGFuY2VNZXRyaWMuRXVjbGlkZWFuOlxuICAgICAgICByZXR1cm4gMSAvICgxICsgdGhpcy5ldWNsaWRlYW5EaXN0YW5jZShhLCBiKSk7XG4gICAgICBjYXNlIERpc3RhbmNlTWV0cmljLkRvdFByb2R1Y3Q6XG4gICAgICAgIHJldHVybiB0aGlzLmRvdFByb2R1Y3QoYSwgYik7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gdGhpcy5jb3NpbmVTaW1pbGFyaXR5KGEsIGIpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgY29zaW5lU2ltaWxhcml0eShhOiBudW1iZXJbXSwgYjogbnVtYmVyW10pOiBudW1iZXIge1xuICAgIGxldCBkb3RQcm9kdWN0ID0gMDtcbiAgICBsZXQgbm9ybUEgPSAwO1xuICAgIGxldCBub3JtQiA9IDA7XG5cbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGEubGVuZ3RoOyBpKyspIHtcbiAgICAgIGRvdFByb2R1Y3QgKz0gYVtpXSAqIGJbaV07XG4gICAgICBub3JtQSArPSBhW2ldICogYVtpXTtcbiAgICAgIG5vcm1CICs9IGJbaV0gKiBiW2ldO1xuICAgIH1cblxuICAgIGlmIChub3JtQSA9PT0gMCB8fCBub3JtQiA9PT0gMCkgcmV0dXJuIDA7XG4gICAgcmV0dXJuIGRvdFByb2R1Y3QgLyAoTWF0aC5zcXJ0KG5vcm1BKSAqIE1hdGguc3FydChub3JtQikpO1xuICB9XG5cbiAgcHJpdmF0ZSBldWNsaWRlYW5EaXN0YW5jZShhOiBudW1iZXJbXSwgYjogbnVtYmVyW10pOiBudW1iZXIge1xuICAgIGxldCBzdW0gPSAwO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgYS5sZW5ndGg7IGkrKykge1xuICAgICAgY29uc3QgZGlmZiA9IGFbaV0gLSBiW2ldO1xuICAgICAgc3VtICs9IGRpZmYgKiBkaWZmO1xuICAgIH1cbiAgICByZXR1cm4gTWF0aC5zcXJ0KHN1bSk7XG4gIH1cblxuICBwcml2YXRlIGRvdFByb2R1Y3QoYTogbnVtYmVyW10sIGI6IG51bWJlcltdKTogbnVtYmVyIHtcbiAgICBsZXQgc3VtID0gMDtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGEubGVuZ3RoOyBpKyspIHtcbiAgICAgIHN1bSArPSBhW2ldICogYltpXTtcbiAgICB9XG4gICAgcmV0dXJuIHN1bTtcbiAgfVxuXG4gIHByaXZhdGUgbWF0Y2hlc0ZpbHRlcihtZXRhZGF0YTogUmVjb3JkPHN0cmluZywgYW55PiB8IHVuZGVmaW5lZCwgZmlsdGVyOiBSZWNvcmQ8c3RyaW5nLCBhbnk+KTogYm9vbGVhbiB7XG4gICAgaWYgKCFtZXRhZGF0YSkgcmV0dXJuIGZhbHNlO1xuICAgIFxuICAgIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKGZpbHRlcikpIHtcbiAgICAgIGlmIChtZXRhZGF0YVtrZXldICE9PSB2YWx1ZSkge1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiB0cnVlO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBDb2xsZWN0aW9uIHtcbiAgcHJpdmF0ZSB2ZWN0b3JJbmRleDogVmVjdG9ySW5kZXg7XG4gIHByaXZhdGUgX2luZGV4UmVhZHkgPSBmYWxzZTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIGRiOiBhbnksXG4gICAgcHJpdmF0ZSBuYW1lc3BhY2U6IHN0cmluZyxcbiAgICBwcml2YXRlIG5hbWU6IHN0cmluZyxcbiAgICBwcml2YXRlIGNvbmZpZzogQ29sbGVjdGlvbkNvbmZpZ1xuICApIHtcbiAgICB0aGlzLnZlY3RvckluZGV4ID0gbmV3IFZlY3RvckluZGV4KFxuICAgICAgY29uZmlnLmRpbWVuc2lvbiB8fCAzODQsXG4gICAgICBjb25maWcubWV0cmljIHx8IERpc3RhbmNlTWV0cmljLkNvc2luZVxuICAgICk7XG4gIH1cblxuICAvKipcbiAgICogSW5zZXJ0IGEgdmVjdG9yIHdpdGggb3B0aW9uYWwgbWV0YWRhdGFcbiAgICogVmVjdG9yIGlzIGltbWVkaWF0ZWx5IGluZGV4ZWQgKHN5bmNocm9ub3VzKVxuICAgKi9cbiAgYXN5bmMgaW5zZXJ0KFxuICAgIHZlY3RvcjogbnVtYmVyW10sXG4gICAgbWV0YWRhdGE/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+LFxuICAgIGlkPzogc3RyaW5nXG4gICk6IFByb21pc2U8c3RyaW5nPiB7XG4gICAgaWYgKHRoaXMuY29uZmlnLmRpbWVuc2lvbiAmJiB2ZWN0b3IubGVuZ3RoICE9PSB0aGlzLmNvbmZpZy5kaW1lbnNpb24pIHtcbiAgICAgIHRocm93IG5ldyBEYXRhYmFzZUVycm9yKFxuICAgICAgICBgVmVjdG9yIGRpbWVuc2lvbiBtaXNtYXRjaDogZXhwZWN0ZWQgJHt0aGlzLmNvbmZpZy5kaW1lbnNpb259LCBnb3QgJHt2ZWN0b3IubGVuZ3RofWBcbiAgICAgICk7XG4gICAgfVxuXG4gICAgY29uc3QgdmVjdG9ySWQgPSBpZCB8fCB0aGlzLmdlbmVyYXRlSWQoKTtcbiAgICBjb25zdCBrZXkgPSB0aGlzLnZlY3RvcktleSh2ZWN0b3JJZCk7XG4gICAgXG4gICAgY29uc3QgZGF0YSA9IHtcbiAgICAgIHZlY3RvcixcbiAgICAgIG1ldGFkYXRhOiBtZXRhZGF0YSB8fCB7fSxcbiAgICAgIHRpbWVzdGFtcDogRGF0ZS5ub3coKSxcbiAgICB9O1xuXG4gICAgLy8gU3RvcmUgdG8gZGF0YWJhc2VcbiAgICBhd2FpdCB0aGlzLmRiLnB1dChCdWZmZXIuZnJvbShrZXkpLCBCdWZmZXIuZnJvbShKU09OLnN0cmluZ2lmeShkYXRhKSkpO1xuICAgIFxuICAgIC8vIFNZTkNIUk9OT1VTTFkgYWRkIHRvIGluLW1lbW9yeSBpbmRleFxuICAgIHRoaXMudmVjdG9ySW5kZXguYWRkKHZlY3RvcklkLCB2ZWN0b3IsIG1ldGFkYXRhKTtcblxuICAgIHJldHVybiB2ZWN0b3JJZDtcbiAgfVxuXG4gIC8qKlxuICAgKiBJbnNlcnQgbXVsdGlwbGUgdmVjdG9yc1xuICAgKiBBbGwgdmVjdG9ycyBhcmUgaW5kZXhlZCBzeW5jaHJvbm91c2x5IGFmdGVyIGluc2VydGlvblxuICAgKi9cbiAgYXN5bmMgaW5zZXJ0TWFueShcbiAgICB2ZWN0b3JzOiBudW1iZXJbXVtdLFxuICAgIG1ldGFkYXRhcz86IFJlY29yZDxzdHJpbmcsIGFueT5bXSxcbiAgICBpZHM/OiBzdHJpbmdbXVxuICApOiBQcm9taXNlPHN0cmluZ1tdPiB7XG4gICAgY29uc3QgcmVzdWx0SWRzOiBzdHJpbmdbXSA9IFtdO1xuICAgIFxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgdmVjdG9ycy5sZW5ndGg7IGkrKykge1xuICAgICAgY29uc3QgaWQgPSBpZHMgPyBpZHNbaV0gOiB1bmRlZmluZWQ7XG4gICAgICBjb25zdCBtZXRhZGF0YSA9IG1ldGFkYXRhcyA/IG1ldGFkYXRhc1tpXSA6IHVuZGVmaW5lZDtcbiAgICAgIGNvbnN0IHJlc3VsdElkID0gYXdhaXQgdGhpcy5pbnNlcnQodmVjdG9yc1tpXSwgbWV0YWRhdGEsIGlkKTtcbiAgICAgIHJlc3VsdElkcy5wdXNoKHJlc3VsdElkKTtcbiAgICB9XG4gICAgXG4gICAgcmV0dXJuIHJlc3VsdElkcztcbiAgfVxuXG4gIC8qKlxuICAgKiBSZWJ1aWxkIGluZGV4IGZyb20gZGF0YWJhc2UgKGZvciByZWNvdmVyeS9zdGFydHVwKVxuICAgKi9cbiAgYXN5bmMgcmVidWlsZEluZGV4KCk6IFByb21pc2U8bnVtYmVyPiB7XG4gICAgY29uc3QgcHJlZml4ID0gdGhpcy52ZWN0b3JLZXlQcmVmaXgoKTtcbiAgICBsZXQgY291bnQgPSAwO1xuXG4gICAgdHJ5IHtcbiAgICAgIGZvciBhd2FpdCAoY29uc3QgW2tleUJ1ZmZlciwgdmFsdWVCdWZmZXJdIG9mIHRoaXMuZGIuc2NhblByZWZpeChCdWZmZXIuZnJvbShwcmVmaXgpKSkge1xuICAgICAgICBjb25zdCBrZXkgPSBrZXlCdWZmZXIudG9TdHJpbmcoKTtcbiAgICAgICAgY29uc3QgaWQgPSBrZXkucmVwbGFjZShwcmVmaXgsICcnKTtcbiAgICAgICAgY29uc3QgZGF0YSA9IEpTT04ucGFyc2UodmFsdWVCdWZmZXIudG9TdHJpbmcoKSk7XG4gICAgICAgIFxuICAgICAgICB0aGlzLnZlY3RvckluZGV4LmFkZChpZCwgZGF0YS52ZWN0b3IsIGRhdGEubWV0YWRhdGEpO1xuICAgICAgICBjb3VudCsrO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBjb25zb2xlLndhcm4oJ1tTb2NoREJdIEVycm9yIHJlYnVpbGRpbmcgdmVjdG9yIGluZGV4OicsIGVycm9yKTtcbiAgICB9XG5cbiAgICB0aGlzLl9pbmRleFJlYWR5ID0gdHJ1ZTtcbiAgICByZXR1cm4gY291bnQ7XG4gIH1cblxuICAvKipcbiAgICogQ2hlY2sgaWYgaW5kZXggaXMgcmVhZHkgKGxvYWRlZCBmcm9tIGRpc2spXG4gICAqL1xuICBnZXQgaXNJbmRleFJlYWR5KCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLl9pbmRleFJlYWR5IHx8IHRoaXMudmVjdG9ySW5kZXguc2l6ZSgpID4gMDtcbiAgfVxuXG4gIC8qKlxuICAgKiBTZWFyY2ggZm9yIHNpbWlsYXIgdmVjdG9yc1xuICAgKiBVc2VzIHN5bmNocm9ub3VzIGluLW1lbW9yeSBpbmRleCAtIG5vIGRlbGF5XG4gICAqL1xuICBhc3luYyBzZWFyY2gocmVxdWVzdDogU2VhcmNoUmVxdWVzdCk6IFByb21pc2U8U2VhcmNoUmVzdWx0W10+IHtcbiAgICAvLyBBdXRvLXJlYnVpbGQgaW5kZXggaWYgZW1wdHkgYnV0IHRoZXJlIG1pZ2h0IGJlIGRhdGFcbiAgICBpZiAoIXRoaXMuaXNJbmRleFJlYWR5ICYmIHRoaXMudmVjdG9ySW5kZXguc2l6ZSgpID09PSAwKSB7XG4gICAgICBhd2FpdCB0aGlzLnJlYnVpbGRJbmRleCgpO1xuICAgIH1cblxuICAgIGNvbnN0IHJlc3VsdHMgPSB0aGlzLnZlY3RvckluZGV4LnNlYXJjaChcbiAgICAgIHJlcXVlc3QucXVlcnlWZWN0b3IsXG4gICAgICByZXF1ZXN0LmssXG4gICAgICByZXF1ZXN0LmZpbHRlclxuICAgICk7XG5cbiAgICAvLyBNYXAgcmVzdWx0cyBiYXNlZCBvbiBpbmNsdWRlTWV0YWRhdGEgZmxhZ1xuICAgIHJldHVybiByZXN1bHRzLm1hcChyID0+ICh7XG4gICAgICBpZDogci5pZCxcbiAgICAgIHNjb3JlOiByLnNjb3JlLFxuICAgICAgdmVjdG9yOiByZXF1ZXN0LmluY2x1ZGVNZXRhZGF0YSA/IHIudmVjdG9yIDogdW5kZWZpbmVkLFxuICAgICAgbWV0YWRhdGE6IHJlcXVlc3QuaW5jbHVkZU1ldGFkYXRhID8gci5tZXRhZGF0YSA6IHVuZGVmaW5lZCxcbiAgICB9KSk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IGEgdmVjdG9yIGJ5IElEXG4gICAqL1xuICBhc3luYyBnZXQoaWQ6IHN0cmluZyk6IFByb21pc2U8eyB2ZWN0b3I6IG51bWJlcltdOyBtZXRhZGF0YT86IFJlY29yZDxzdHJpbmcsIGFueT4gfSB8IG51bGw+IHtcbiAgICBjb25zdCBrZXkgPSB0aGlzLnZlY3RvcktleShpZCk7XG4gICAgY29uc3QgdmFsdWUgPSBhd2FpdCB0aGlzLmRiLmdldChCdWZmZXIuZnJvbShrZXkpKTtcbiAgICBcbiAgICBpZiAoIXZhbHVlKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG4gICAgXG4gICAgY29uc3QgZGF0YSA9IEpTT04ucGFyc2UodmFsdWUudG9TdHJpbmcoKSk7XG4gICAgcmV0dXJuIHtcbiAgICAgIHZlY3RvcjogZGF0YS52ZWN0b3IsXG4gICAgICBtZXRhZGF0YTogZGF0YS5tZXRhZGF0YSxcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIERlbGV0ZSBhIHZlY3RvciBieSBJRFxuICAgKi9cbiAgYXN5bmMgZGVsZXRlKGlkOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+IHtcbiAgICBjb25zdCBrZXkgPSB0aGlzLnZlY3RvcktleShpZCk7XG4gICAgYXdhaXQgdGhpcy5kYi5kZWxldGUoQnVmZmVyLmZyb20oa2V5KSk7XG4gICAgLy8gUmVtb3ZlIGZyb20gaW4tbWVtb3J5IGluZGV4XG4gICAgdGhpcy52ZWN0b3JJbmRleC5yZW1vdmUoaWQpO1xuICAgIHJldHVybiB0cnVlO1xuICB9XG5cbiAgLyoqXG4gICAqIENvdW50IHZlY3RvcnMgaW4gY29sbGVjdGlvblxuICAgKi9cbiAgYXN5bmMgY291bnQoKTogUHJvbWlzZTxudW1iZXI+IHtcbiAgICAvLyBJZiBpbmRleCBpcyBsb2FkZWQsIHJldHVybiBmcm9tIGluZGV4XG4gICAgaWYgKHRoaXMuaXNJbmRleFJlYWR5KSB7XG4gICAgICByZXR1cm4gdGhpcy52ZWN0b3JJbmRleC5zaXplKCk7XG4gICAgfVxuICAgIC8vIE90aGVyd2lzZSByZWJ1aWxkIGFuZCBjb3VudFxuICAgIHJldHVybiBhd2FpdCB0aGlzLnJlYnVpbGRJbmRleCgpO1xuICB9XG5cbiAgLy8gSGVscGVyIG1ldGhvZHNcbiAgcHJpdmF0ZSB2ZWN0b3JLZXkoaWQ6IHN0cmluZyk6IHN0cmluZyB7XG4gICAgcmV0dXJuIGBfY29sbGVjdGlvbi8ke3RoaXMubmFtZXNwYWNlfS8ke3RoaXMubmFtZX0vdmVjdG9ycy8ke2lkfWA7XG4gIH1cblxuICBwcml2YXRlIHZlY3RvcktleVByZWZpeCgpOiBzdHJpbmcge1xuICAgIHJldHVybiBgX2NvbGxlY3Rpb24vJHt0aGlzLm5hbWVzcGFjZX0vJHt0aGlzLm5hbWV9L3ZlY3RvcnMvYDtcbiAgfVxuXG4gIHByaXZhdGUgbWV0YWRhdGFLZXkoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gYF9jb2xsZWN0aW9uLyR7dGhpcy5uYW1lc3BhY2V9LyR7dGhpcy5uYW1lfS9tZXRhZGF0YWA7XG4gIH1cblxuICBwcml2YXRlIGdlbmVyYXRlSWQoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gYCR7RGF0ZS5ub3coKX0tJHtNYXRoLnJhbmRvbSgpLnRvU3RyaW5nKDM2KS5zdWJzdHIoMiwgOSl9YDtcbiAgfVxuXG4gIC8vIENhbGN1bGF0ZSBjb3NpbmUgc2ltaWxhcml0eVxuICBwcml2YXRlIGNvc2luZVNpbWlsYXJpdHkoYTogbnVtYmVyW10sIGI6IG51bWJlcltdKTogbnVtYmVyIHtcbiAgICBsZXQgZG90UHJvZHVjdCA9IDA7XG4gICAgbGV0IG5vcm1BID0gMDtcbiAgICBsZXQgbm9ybUIgPSAwO1xuICAgIFxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgYS5sZW5ndGg7IGkrKykge1xuICAgICAgZG90UHJvZHVjdCArPSBhW2ldICogYltpXTtcbiAgICAgIG5vcm1BICs9IGFbaV0gKiBhW2ldO1xuICAgICAgbm9ybUIgKz0gYltpXSAqIGJbaV07XG4gICAgfVxuICAgIFxuICAgIHJldHVybiBkb3RQcm9kdWN0IC8gKE1hdGguc3FydChub3JtQSkgKiBNYXRoLnNxcnQobm9ybUIpKTtcbiAgfVxufVxuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBOYW1lc3BhY2UgSGFuZGxlXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCBjbGFzcyBOYW1lc3BhY2Uge1xuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIGRiOiBhbnksXG4gICAgcHJpdmF0ZSBuYW1lOiBzdHJpbmcsXG4gICAgcHJpdmF0ZSBjb25maWc6IE5hbWVzcGFjZUNvbmZpZ1xuICApIHt9XG5cbiAgLyoqXG4gICAqIENyZWF0ZSBhIG5ldyBjb2xsZWN0aW9uIGluIHRoaXMgbmFtZXNwYWNlXG4gICAqL1xuICBhc3luYyBjcmVhdGVDb2xsZWN0aW9uKGNvbmZpZzogQ29sbGVjdGlvbkNvbmZpZyk6IFByb21pc2U8Q29sbGVjdGlvbj4ge1xuICAgIGNvbnN0IG1ldGFkYXRhS2V5ID0gYF9jb2xsZWN0aW9uLyR7dGhpcy5uYW1lfS8ke2NvbmZpZy5uYW1lfS9tZXRhZGF0YWA7XG4gICAgXG4gICAgLy8gQ2hlY2sgaWYgY29sbGVjdGlvbiBhbHJlYWR5IGV4aXN0c1xuICAgIGNvbnN0IGV4aXN0aW5nID0gYXdhaXQgdGhpcy5kYi5nZXQoQnVmZmVyLmZyb20obWV0YWRhdGFLZXkpKTtcbiAgICBpZiAoZXhpc3RpbmcpIHtcbiAgICAgIHRocm93IG5ldyBDb2xsZWN0aW9uRXhpc3RzRXJyb3IoY29uZmlnLm5hbWUpO1xuICAgIH1cblxuICAgIC8vIFN0b3JlIGNvbGxlY3Rpb24gbWV0YWRhdGFcbiAgICBjb25zdCBtZXRhZGF0YSA9IHtcbiAgICAgIC4uLmNvbmZpZyxcbiAgICAgIGNyZWF0ZWRBdDogRGF0ZS5ub3coKSxcbiAgICB9O1xuICAgIFxuICAgIGF3YWl0IHRoaXMuZGIucHV0KFxuICAgICAgQnVmZmVyLmZyb20obWV0YWRhdGFLZXkpLFxuICAgICAgQnVmZmVyLmZyb20oSlNPTi5zdHJpbmdpZnkobWV0YWRhdGEpKVxuICAgICk7XG5cbiAgICByZXR1cm4gbmV3IENvbGxlY3Rpb24odGhpcy5kYiwgdGhpcy5uYW1lLCBjb25maWcubmFtZSwgY29uZmlnKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgYW4gZXhpc3RpbmcgY29sbGVjdGlvblxuICAgKi9cbiAgYXN5bmMgY29sbGVjdGlvbihuYW1lOiBzdHJpbmcpOiBQcm9taXNlPENvbGxlY3Rpb24+IHtcbiAgICBjb25zdCBtZXRhZGF0YUtleSA9IGBfY29sbGVjdGlvbi8ke3RoaXMubmFtZX0vJHtuYW1lfS9tZXRhZGF0YWA7XG4gICAgY29uc3QgbWV0YWRhdGEgPSBhd2FpdCB0aGlzLmRiLmdldChCdWZmZXIuZnJvbShtZXRhZGF0YUtleSkpO1xuICAgIFxuICAgIGlmICghbWV0YWRhdGEpIHtcbiAgICAgIHRocm93IG5ldyBDb2xsZWN0aW9uTm90Rm91bmRFcnJvcihuYW1lKTtcbiAgICB9XG5cbiAgICBjb25zdCBjb25maWcgPSBKU09OLnBhcnNlKG1ldGFkYXRhLnRvU3RyaW5nKCkpO1xuICAgIHJldHVybiBuZXcgQ29sbGVjdGlvbih0aGlzLmRiLCB0aGlzLm5hbWUsIG5hbWUsIGNvbmZpZyk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IG9yIGNyZWF0ZSBhIGNvbGxlY3Rpb25cbiAgICovXG4gIGFzeW5jIGdldE9yQ3JlYXRlQ29sbGVjdGlvbihjb25maWc6IENvbGxlY3Rpb25Db25maWcpOiBQcm9taXNlPENvbGxlY3Rpb24+IHtcbiAgICB0cnkge1xuICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuY29sbGVjdGlvbihjb25maWcubmFtZSk7XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGlmIChlcnJvciBpbnN0YW5jZW9mIENvbGxlY3Rpb25Ob3RGb3VuZEVycm9yKSB7XG4gICAgICAgIHJldHVybiBhd2FpdCB0aGlzLmNyZWF0ZUNvbGxlY3Rpb24oY29uZmlnKTtcbiAgICAgIH1cbiAgICAgIHRocm93IGVycm9yO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBEZWxldGUgYSBjb2xsZWN0aW9uXG4gICAqL1xuICBhc3luYyBkZWxldGVDb2xsZWN0aW9uKG5hbWU6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGNvbnN0IG1ldGFkYXRhS2V5ID0gYF9jb2xsZWN0aW9uLyR7dGhpcy5uYW1lfS8ke25hbWV9L21ldGFkYXRhYDtcbiAgICBjb25zdCBwcmVmaXggPSBgX2NvbGxlY3Rpb24vJHt0aGlzLm5hbWV9LyR7bmFtZX0vYDtcbiAgICBcbiAgICAvLyBUT0RPOiBEZWxldGUgYWxsIGtleXMgd2l0aCBwcmVmaXhcbiAgICBhd2FpdCB0aGlzLmRiLmRlbGV0ZShCdWZmZXIuZnJvbShtZXRhZGF0YUtleSkpO1xuICAgIFxuICAgIHJldHVybiB0cnVlO1xuICB9XG5cbiAgLyoqXG4gICAqIExpc3QgYWxsIGNvbGxlY3Rpb25zIGluIHRoaXMgbmFtZXNwYWNlXG4gICAqL1xuICBhc3luYyBsaXN0Q29sbGVjdGlvbnMoKTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICAgIC8vIFRPRE86IEltcGxlbWVudCBlZmZpY2llbnQgbGlzdGluZyB3aXRoIHJhbmdlIHF1ZXJpZXNcbiAgICByZXR1cm4gW107XG4gIH1cblxuICBnZXROYW1lKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMubmFtZTtcbiAgfVxuXG4gIGdldENvbmZpZygpOiBOYW1lc3BhY2VDb25maWcge1xuICAgIHJldHVybiB7IC4uLnRoaXMuY29uZmlnIH07XG4gIH1cbn1cbiJdfQ==
586
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmFtZXNwYWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL25hbWVzcGFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7Ozs7Ozs7Ozs7OztHQWVHOzs7QUFFSCxxQ0FBc0Q7QUFtQnRELElBQUksVUFBVSxHQUE4QixJQUFJLENBQUM7QUFDakQsSUFBSSxDQUFDO0lBQ0gsTUFBTSxLQUFLLEdBQUcsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQy9CLE1BQU0sRUFBRSxXQUFXLEVBQUUsR0FBRyxPQUFPLENBQUMsK0JBQStCLENBQUMsQ0FBQztJQUNqRSxNQUFNLFdBQVcsR0FBRyxXQUFXLEVBQUUsQ0FBQztJQUNsQyxNQUFNLEdBQUcsR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxDQUFDO0lBRXBDLHdCQUF3QjtJQUN4QixNQUFNLFlBQVksR0FBRyxLQUFLLENBQUMsT0FBTyxDQUFDLGVBQWUsRUFBRSxLQUFLLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztJQUVwRSxnREFBZ0Q7SUFDaEQsTUFBTSxtQkFBbUIsR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDLGdCQUFnQixFQUFFO1FBQ3pELEtBQUssRUFBRSxRQUFRO1FBQ2YsS0FBSyxFQUFFLFFBQVE7UUFDZixRQUFRLEVBQUUsT0FBTztLQUNsQixDQUFDLENBQUM7SUFFSCxVQUFVLEdBQUc7UUFDWCxHQUFHO1FBQ0gsS0FBSztRQUNMLFlBQVk7UUFDWixtQkFBbUI7UUFDbkIsUUFBUSxFQUFFLEdBQUcsQ0FBQyxJQUFJLENBQUMsVUFBVSxFQUFFLFlBQVksRUFBRSxDQUFDLFFBQVEsRUFBRSxRQUFRLEVBQUUsUUFBUSxDQUFDLENBQUM7UUFDNUUsU0FBUyxFQUFFLEdBQUcsQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLE1BQU0sRUFBRSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBQ3hELGlCQUFpQixFQUFFLEdBQUcsQ0FBQyxJQUFJLENBQUMs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IE5hbWVzcGFjZSBBUElcbiAqIFxuICogUHJvdmlkZXMgdHlwZS1zYWZlIG5hbWVzcGFjZSBpc29sYXRpb24gd2l0aCBmaXJzdC1jbGFzcyBuYW1lc3BhY2UgaGFuZGxlcy5cbiAqIFxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IERhdGFiYXNlIH0gZnJvbSAnQHNvY2hkYi9zb2NoZGInO1xuICogXG4gKiBjb25zdCBkYiA9IGF3YWl0IERhdGFiYXNlLm9wZW4oJy4vbXlkYicpO1xuICogY29uc3QgbnMgPSBhd2FpdCBkYi5jcmVhdGVOYW1lc3BhY2UoJ3RlbmFudF8xMjMnKTtcbiAqIGNvbnN0IGNvbGxlY3Rpb24gPSBhd2FpdCBucy5jcmVhdGVDb2xsZWN0aW9uKCdkb2N1bWVudHMnLCB7IGRpbWVuc2lvbjogMzg0IH0pO1xuICogYXdhaXQgY29sbGVjdGlvbi5pbnNlcnQoWzEuMCwgMi4wLCAuLi5dLCB7IHNvdXJjZTogJ3dlYicgfSk7XG4gKiBjb25zdCByZXN1bHRzID0gYXdhaXQgY29sbGVjdGlvbi5zZWFyY2gocXVlcnlWZWN0b3IsIDEwKTtcbiAqIGBgYFxuICovXG5cbmltcG9ydCB7IFNvY2hEQkVycm9yLCBEYXRhYmFzZUVycm9yIH0gZnJvbSAnLi9lcnJvcnMnO1xuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBOYXRpdmUgSE5TVyBGRkkgQmluZGluZ3MgKGZvciBoaWdoLXBlcmZvcm1hbmNlIGJhdGNoIGluc2VydCBhbmQgc2VhcmNoKVxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG5pbnRlcmZhY2UgTmF0aXZlSG5zd0JpbmRpbmdzIHtcbiAgbGliOiBhbnk7XG4gIGtvZmZpOiBhbnk7XG4gIEhuc3dJbmRleFB0cjogYW55O1xuICBDU2VhcmNoUmVzdWx0U3RydWN0OiBhbnk7XG4gIGhuc3dfbmV3OiAoZGltZW5zaW9uOiBudW1iZXIsIG1heENvbm5lY3Rpb25zOiBudW1iZXIsIGVmQ29uc3RydWN0aW9uOiBudW1iZXIpID0+IGFueTtcbiAgaG5zd19mcmVlOiAocHRyOiBhbnkpID0+IHZvaWQ7XG4gIGhuc3dfaW5zZXJ0X2JhdGNoOiAocHRyOiBhbnksIGlkczogQmlnVWludDY0QXJyYXksIHZlY3RvcnM6IEZsb2F0MzJBcnJheSwgbnVtVmVjdG9yczogbnVtYmVyLCBkaW1lbnNpb246IG51bWJlcikgPT4gbnVtYmVyO1xuICBobnN3X2xlbjogKHB0cjogYW55KSA9PiBudW1iZXI7XG4gIGhuc3dfc2V0X2VmX3NlYXJjaDogKHB0cjogYW55LCBlZlNlYXJjaDogbnVtYmVyKSA9PiB2b2lkO1xuICBobnN3X3NlYXJjaDogKHB0cjogYW55LCBxdWVyeTogRmxvYXQzMkFycmF5LCBxdWVyeUxlbjogbnVtYmVyLCBrOiBudW1iZXIsIHJlc3VsdHNPdXQ6IEJ1ZmZlciwgbnVtUmVzdWx0c091dDogQnVmZmVyKSA9PiBudW1iZXI7XG59XG5cbmxldCBOYXRpdmVIbnN3OiBOYXRpdmVIbnN3QmluZGluZ3MgfCBudWxsID0gbnVsbDtcbnRyeSB7XG4gIGNvbnN0IGtvZmZpID0gcmVxdWlyZSgna29mZmknKTtcbiAgY29uc3QgeyBmaW5kTGlicmFyeSB9ID0gcmVxdWlyZSgnLi9lbWJlZGRlZC9mZmkvbGlicmFyeS1maW5kZXInKTtcbiAgY29uc3QgbGlicmFyeVBhdGggPSBmaW5kTGlicmFyeSgpO1xuICBjb25zdCBsaWIgPSBrb2ZmaS5sb2FkKGxpYnJhcnlQYXRoKTtcbiAgXG4gIC8vIERlZmluZSBvcGFxdWUgcG9pbnRlclxuICBjb25zdCBIbnN3SW5kZXhQdHIgPSBrb2ZmaS5wb2ludGVyKCdIbnN3SW5kZXhQdHIzJywga29mZmkub3BhcXVlKCkpO1xuICBcbiAgLy8gRGVmaW5lIENTZWFyY2hSZXN1bHQgc3RydWN0IGZvciBuYXRpdmUgc2VhcmNoXG4gIGNvbnN0IENTZWFyY2hSZXN1bHRTdHJ1Y3QgPSBrb2ZmaS5zdHJ1Y3QoJ0NTZWFyY2hSZXN1bHQzJywge1xuICAgIGlkX2xvOiAndWludDY0JyxcbiAgICBpZF9oaTogJ3VpbnQ2NCcsIFxuICAgIGRpc3RhbmNlOiAnZmxvYXQnXG4gIH0pO1xuICBcbiAgTmF0aXZlSG5zdyA9IHtcbiAgICBsaWIsXG4gICAga29mZmksXG4gICAgSG5zd0luZGV4UHRyLFxuICAgIENTZWFyY2hSZXN1bHRTdHJ1Y3QsXG4gICAgaG5zd19uZXc6IGxpYi5mdW5jKCdobnN3X25ldycsIEhuc3dJbmRleFB0ciwgWydzaXplX3QnLCAnc2l6ZV90JywgJ3NpemVfdCddKSxcbiAgICBobnN3X2ZyZWU6IGxpYi5mdW5jKCdobnN3X2ZyZWUnLCAndm9pZCcsIFtIbnN3SW5kZXhQdHJdKSxcbiAgICBobnN3X2luc2VydF9iYXRjaDogbGliLmZ1bmMoJ2huc3dfaW5zZXJ0X2JhdGNoJywgJ2ludCcsIFtcbiAgICAgIEhuc3dJbmRleFB0cixcbiAgICAgICd1aW50NjQqJywgIC8vIGlkc1xuICAgICAgJ2Zsb2F0KicsICAgLy8gdmVjdG9ycyAoZmxhdClcbiAgICAgICdzaXplX3QnLCAgIC8vIG51bV92ZWN0b3JzXG4gICAgICAnc2l6ZV90JyAgICAvLyBkaW1lbnNpb25cbiAgICBdKSxcbiAgICBobnN3X2xlbjogbGliLmZ1bmMoJ2huc3dfbGVuJywgJ3NpemVfdCcsIFtIbnN3SW5kZXhQdHJdKSxcbiAgICBobnN3X3NldF9lZl9zZWFyY2g6IGxpYi5mdW5jKCdobnN3X3NldF9lZl9zZWFyY2gnLCAndm9pZCcsIFtIbnN3SW5kZXhQdHIsICdzaXplX3QnXSksXG4gICAgaG5zd19zZWFyY2g6IGxpYi5mdW5jKCdobnN3X3NlYXJjaCcsICdpbnQnLCBbXG4gICAgICBIbnN3SW5kZXhQdHIsXG4gICAgICAnZmxvYXQqJywgICAvLyBxdWVyeSB2ZWN0b3JcbiAgICAgICdzaXplX3QnLCAgIC8vIHF1ZXJ5X2xlblxuICAgICAgJ3NpemVfdCcsICAgLy8ga1xuICAgICAga29mZmkucG9pbnRlcihDU2VhcmNoUmVzdWx0U3RydWN0KSwgIC8vIHJlc3VsdHNfb3V0IHBvaW50ZXJcbiAgICAgIGtvZmZpLnBvaW50ZXIoJ3NpemVfdCcpICAvLyBudW1fcmVzdWx0c19vdXQgcG9pbnRlclxuICAgIF0pLFxuICB9O1xuICBjb25zb2xlLmxvZygnW1NvY2hEQl0gTmF0aXZlIEhOU1cgYmluZGluZ3MgbG9hZGVkIChiYXRjaCBpbnNlcnQgKyBzZWFyY2gpJyk7XG59IGNhdGNoIChlOiBhbnkpIHtcbiAgLy8gTmF0aXZlIGJpbmRpbmdzIG5vdCBhdmFpbGFibGUgLSB3aWxsIHVzZSBKUyBmYWxsYmFja1xuICBOYXRpdmVIbnN3ID0gbnVsbDtcbn1cblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gTmFtZXNwYWNlIENvbmZpZ3VyYXRpb25cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0IGludGVyZmFjZSBOYW1lc3BhY2VDb25maWcge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRpc3BsYXlOYW1lPzogc3RyaW5nO1xuICBsYWJlbHM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuICByZWFkT25seT86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjbGFzcyBOYW1lc3BhY2VOb3RGb3VuZEVycm9yIGV4dGVuZHMgU29jaERCRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihuYW1lc3BhY2U6IHN0cmluZykge1xuICAgIHN1cGVyKGBOYW1lc3BhY2Ugbm90IGZvdW5kOiAke25hbWVzcGFjZX1gKTtcbiAgICB0aGlzLm5hbWUgPSAnTmFtZXNwYWNlTm90Rm91bmRFcnJvcic7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIE5hbWVzcGFjZUV4aXN0c0Vycm9yIGV4dGVuZHMgU29jaERCRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihuYW1lc3BhY2U6IHN0cmluZykge1xuICAgIHN1cGVyKGBOYW1lc3BhY2UgYWxyZWFkeSBleGlzdHM6ICR7bmFtZXNwYWNlfWApO1xuICAgIHRoaXMubmFtZSA9ICdOYW1lc3BhY2VFeGlzdHNFcnJvcic7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIENvbGxlY3Rpb25Ob3RGb3VuZEVycm9yIGV4dGVuZHMgU29jaERCRXJyb3Ige1xuICBjb25zdHJ1Y3Rvcihjb2xsZWN0aW9uOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgQ29sbGVjdGlvbiBub3QgZm91bmQ6ICR7Y29sbGVjdGlvbn1gKTtcbiAgICB0aGlzLm5hbWUgPSAnQ29sbGVjdGlvbk5vdEZvdW5kRXJyb3InO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBDb2xsZWN0aW9uRXhpc3RzRXJyb3IgZXh0ZW5kcyBTb2NoREJFcnJvciB7XG4gIGNvbnN0cnVjdG9yKGNvbGxlY3Rpb246IHN0cmluZykge1xuICAgIHN1cGVyKGBDb2xsZWN0aW9uIGFscmVhZHkgZXhpc3RzOiAke2NvbGxlY3Rpb259YCk7XG4gICAgdGhpcy5uYW1lID0gJ0NvbGxlY3Rpb25FeGlzdHNFcnJvcic7XG4gIH1cbn1cblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gQ29sbGVjdGlvbiBDb25maWd1cmF0aW9uXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCBlbnVtIERpc3RhbmNlTWV0cmljIHtcbiAgQ29zaW5lID0gJ2Nvc2luZScsXG4gIEV1Y2xpZGVhbiA9ICdldWNsaWRlYW4nLFxuICBEb3RQcm9kdWN0ID0gJ2RvdCcsXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29sbGVjdGlvbkNvbmZpZyB7XG4gIG5hbWU6IHN0cmluZztcbiAgZGltZW5zaW9uPzogbnVtYmVyO1xuICBtZXRyaWM/OiBEaXN0YW5jZU1ldHJpYztcbiAgaW5kZXhlZD86IGJvb2xlYW47XG4gIGhuc3dNPzogbnVtYmVyO1xuICBobnN3RWZDb25zdHJ1Y3Rpb24/OiBudW1iZXI7XG4gIG1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55Pjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTZWFyY2hSZXF1ZXN0IHtcbiAgcXVlcnlWZWN0b3I6IG51bWJlcltdO1xuICBrOiBudW1iZXI7XG4gIGZpbHRlcj86IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIGluY2x1ZGVNZXRhZGF0YT86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VhcmNoUmVzdWx0IHtcbiAgaWQ6IHN0cmluZztcbiAgc2NvcmU6IG51bWJlcjtcbiAgdmVjdG9yPzogbnVtYmVyW107XG4gIG1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55Pjtcbn1cblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gQ29sbGVjdGlvbiBIYW5kbGVcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBJbi1tZW1vcnkgdmVjdG9yIGluZGV4IGZvciBzeW5jaHJvbm91cyBzZWFyY2hcbiAqIFVzZXMgYSBzaW1wbGUgYnV0IGVmZmljaWVudCBhcHByb2FjaCBmb3Igc21hbGwtbWVkaXVtIGRhdGFzZXRzXG4gKi9cbmNsYXNzIFZlY3RvckluZGV4IHtcbiAgcHJpdmF0ZSB2ZWN0b3JzOiBNYXA8c3RyaW5nLCB7IHZlY3RvcjogbnVtYmVyW107IG1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55PiB9PiA9IG5ldyBNYXAoKTtcbiAgcHJpdmF0ZSBkaW1lbnNpb246IG51bWJlcjtcbiAgcHJpdmF0ZSBtZXRyaWM6IERpc3RhbmNlTWV0cmljO1xuXG4gIGNvbnN0cnVjdG9yKGRpbWVuc2lvbjogbnVtYmVyLCBtZXRyaWM6IERpc3RhbmNlTWV0cmljID0gRGlzdGFuY2VNZXRyaWMuQ29zaW5lKSB7XG4gICAgdGhpcy5kaW1lbnNpb24gPSBkaW1lbnNpb247XG4gICAgdGhpcy5tZXRyaWMgPSBtZXRyaWM7XG4gIH1cblxuICBhZGQoaWQ6IHN0cmluZywgdmVjdG9yOiBudW1iZXJbXSwgbWV0YWRhdGE/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+KTogdm9pZCB7XG4gICAgaWYgKHZlY3Rvci5sZW5ndGggIT09IHRoaXMuZGltZW5zaW9uKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYFZlY3RvciBkaW1lbnNpb24gbWlzbWF0Y2g6IGV4cGVjdGVkICR7dGhpcy5kaW1lbnNpb259LCBnb3QgJHt2ZWN0b3IubGVuZ3RofWApO1xuICAgIH1cbiAgICB0aGlzLnZlY3RvcnMuc2V0KGlkLCB7IHZlY3RvciwgbWV0YWRhdGEgfSk7XG4gIH1cblxuICByZW1vdmUoaWQ6IHN0cmluZyk6IHZvaWQge1xuICAgIHRoaXMudmVjdG9ycy5kZWxldGUoaWQpO1xuICB9XG5cbiAgc2VhcmNoKHF1ZXJ5VmVjdG9yOiBudW1iZXJbXSwgazogbnVtYmVyLCBmaWx0ZXI/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+KTogU2VhcmNoUmVzdWx0W10ge1xuICAgIGNvbnN0IHJlc3VsdHM6IFNlYXJjaFJlc3VsdFtdID0gW107XG5cbiAgICBmb3IgKGNvbnN0IFtpZCwgZGF0YV0gb2YgdGhpcy52ZWN0b3JzKSB7XG4gICAgICAvLyBBcHBseSBtZXRhZGF0YSBmaWx0ZXIgaWYgcHJvdmlkZWRcbiAgICAgIGlmIChmaWx0ZXIgJiYgIXRoaXMubWF0Y2hlc0ZpbHRlcihkYXRhLm1ldGFkYXRhLCBmaWx0ZXIpKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBzY29yZSA9IHRoaXMuY2FsY3VsYXRlU2ltaWxhcml0eShxdWVyeVZlY3RvciwgZGF0YS52ZWN0b3IpO1xuICAgICAgcmVzdWx0cy5wdXNoKHtcbiAgICAgICAgaWQsXG4gICAgICAgIHNjb3JlLFxuICAgICAgICB2ZWN0b3I6IGRhdGEudmVjdG9yLFxuICAgICAgICBtZXRhZGF0YTogZGF0YS5tZXRhZGF0YSxcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIC8vIFNvcnQgYnkgc2NvcmUgKGhpZ2hlciBpcyBiZXR0ZXIgZm9yIHNpbWlsYXJpdHkpXG4gICAgcmVzdWx0cy5zb3J0KChhLCBiKSA9PiBiLnNjb3JlIC0gYS5zY29yZSk7XG5cbiAgICByZXR1cm4gcmVzdWx0cy5zbGljZSgwLCBrKTtcbiAgfVxuXG4gIHNpemUoKTogbnVtYmVyIHtcbiAgICByZXR1cm4gdGhpcy52ZWN0b3JzLnNpemU7XG4gIH1cblxuICBwcml2YXRlIGNhbGN1bGF0ZVNpbWlsYXJpdHkoYTogbnVtYmVyW10sIGI6IG51bWJlcltdKTogbnVtYmVyIHtcbiAgICBzd2l0Y2ggKHRoaXMubWV0cmljKSB7XG4gICAgICBjYXNlIERpc3RhbmNlTWV0cmljLkNvc2luZTpcbiAgICAgICAgcmV0dXJuIHRoaXMuY29zaW5lU2ltaWxhcml0eShhLCBiKTtcbiAgICAgIGNhc2UgRGlzdGFuY2VNZXRyaWMuRXVjbGlkZWFuOlxuICAgICAgICByZXR1cm4gMSAvICgxICsgdGhpcy5ldWNsaWRlYW5EaXN0YW5jZShhLCBiKSk7XG4gICAgICBjYXNlIERpc3RhbmNlTWV0cmljLkRvdFByb2R1Y3Q6XG4gICAgICAgIHJldHVybiB0aGlzLmRvdFByb2R1Y3QoYSwgYik7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gdGhpcy5jb3NpbmVTaW1pbGFyaXR5KGEsIGIpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgY29zaW5lU2ltaWxhcml0eShhOiBudW1iZXJbXSwgYjogbnVtYmVyW10pOiBudW1iZXIge1xuICAgIGxldCBkb3RQcm9kdWN0ID0gMDtcbiAgICBsZXQgbm9ybUEgPSAwO1xuICAgIGxldCBub3JtQiA9IDA7XG5cbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGEubGVuZ3RoOyBpKyspIHtcbiAgICAgIGRvdFByb2R1Y3QgKz0gYVtpXSAqIGJbaV07XG4gICAgICBub3JtQSArPSBhW2ldICogYVtpXTtcbiAgICAgIG5vcm1CICs9IGJbaV0gKiBiW2ldO1xuICAgIH1cblxuICAgIGlmIChub3JtQSA9PT0gMCB8fCBub3JtQiA9PT0gMCkgcmV0dXJuIDA7XG4gICAgcmV0dXJuIGRvdFByb2R1Y3QgLyAoTWF0aC5zcXJ0KG5vcm1BKSAqIE1hdGguc3FydChub3JtQikpO1xuICB9XG5cbiAgcHJpdmF0ZSBldWNsaWRlYW5EaXN0YW5jZShhOiBudW1iZXJbXSwgYjogbnVtYmVyW10pOiBudW1iZXIge1xuICAgIGxldCBzdW0gPSAwO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgYS5sZW5ndGg7IGkrKykge1xuICAgICAgY29uc3QgZGlmZiA9IGFbaV0gLSBiW2ldO1xuICAgICAgc3VtICs9IGRpZmYgKiBkaWZmO1xuICAgIH1cbiAgICByZXR1cm4gTWF0aC5zcXJ0KHN1bSk7XG4gIH1cblxuICBwcml2YXRlIGRvdFByb2R1Y3QoYTogbnVtYmVyW10sIGI6IG51bWJlcltdKTogbnVtYmVyIHtcbiAgICBsZXQgc3VtID0gMDtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGEubGVuZ3RoOyBpKyspIHtcbiAgICAgIHN1bSArPSBhW2ldICogYltpXTtcbiAgICB9XG4gICAgcmV0dXJuIHN1bTtcbiAgfVxuXG4gIHByaXZhdGUgbWF0Y2hlc0ZpbHRlcihtZXRhZGF0YTogUmVjb3JkPHN0cmluZywgYW55PiB8IHVuZGVmaW5lZCwgZmlsdGVyOiBSZWNvcmQ8c3RyaW5nLCBhbnk+KTogYm9vbGVhbiB7XG4gICAgaWYgKCFtZXRhZGF0YSkgcmV0dXJuIGZhbHNlO1xuICAgIFxuICAgIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKGZpbHRlcikpIHtcbiAgICAgIGlmIChtZXRhZGF0YVtrZXldICE9PSB2YWx1ZSkge1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiB0cnVlO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBDb2xsZWN0aW9uIHtcbiAgcHJpdmF0ZSB2ZWN0b3JJbmRleDogVmVjdG9ySW5kZXg7XG4gIHByaXZhdGUgbmF0aXZlSW5kZXhQdHI6IGFueSA9IG51bGw7XG4gIHByaXZhdGUgX2luZGV4UmVhZHkgPSBmYWxzZTtcbiAgcHJpdmF0ZSBuYXRpdmVJZENvdW50ZXIgPSAwOyAgLy8gQ291bnRlciBmb3IgbmF0aXZlIEhOU1cgbnVtZXJpYyBJRHNcbiAgcHJpdmF0ZSBuYXRpdmVJZFRvU3RyaW5nSWQgPSBuZXcgTWFwPG51bWJlciwgc3RyaW5nPigpOyAgLy8gTWFwIG51bWVyaWMgSUQgLT4gc3RyaW5nIElEXG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSBkYjogYW55LFxuICAgIHByaXZhdGUgbmFtZXNwYWNlOiBzdHJpbmcsXG4gICAgcHJpdmF0ZSBuYW1lOiBzdHJpbmcsXG4gICAgcHJpdmF0ZSBjb25maWc6IENvbGxlY3Rpb25Db25maWdcbiAgKSB7XG4gICAgdGhpcy52ZWN0b3JJbmRleCA9IG5ldyBWZWN0b3JJbmRleChcbiAgICAgIGNvbmZpZy5kaW1lbnNpb24gfHwgMzg0LFxuICAgICAgY29uZmlnLm1ldHJpYyB8fCBEaXN0YW5jZU1ldHJpYy5Db3NpbmVcbiAgICApO1xuICAgIFxuICAgIC8vIFRyeSB0byBjcmVhdGUgbmF0aXZlIEhOU1cgaW5kZXggZm9yIGhpZ2gtcGVyZm9ybWFuY2Ugb3BlcmF0aW9uc1xuICAgIGlmIChOYXRpdmVIbnN3KSB7XG4gICAgICB0cnkge1xuICAgICAgICBjb25zdCBkaW1lbnNpb24gPSBjb25maWcuZGltZW5zaW9uIHx8IDM4NDtcbiAgICAgICAgY29uc3QgbWF4Q29ubmVjdGlvbnMgPSBjb25maWcuaG5zd00gfHwgMTY7XG4gICAgICAgIGNvbnN0IGVmQ29uc3RydWN0aW9uID0gY29uZmlnLmhuc3dFZkNvbnN0cnVjdGlvbiB8fCAxMDA7XG4gICAgICAgIHRoaXMubmF0aXZlSW5kZXhQdHIgPSBOYXRpdmVIbnN3Lmhuc3dfbmV3KGRpbWVuc2lvbiwgbWF4Q29ubmVjdGlvbnMsIGVmQ29uc3RydWN0aW9uKTtcbiAgICAgICAgaWYgKHRoaXMubmF0aXZlSW5kZXhQdHIpIHtcbiAgICAgICAgICAvLyBTZXQgaGlnaCBlZl9zZWFyY2ggZm9yIGdvb2QgcmVjYWxsIChjYW4gYmUgdHVuZWQgdmlhIHNldEVmU2VhcmNoKVxuICAgICAgICAgIE5hdGl2ZUhuc3cuaG5zd19zZXRfZWZfc2VhcmNoKHRoaXMubmF0aXZlSW5kZXhQdHIsIDUwMCk7XG4gICAgICAgICAgY29uc29sZS5sb2coYFtTb2NoREJdIE5hdGl2ZSBITlNXIGluZGV4IGNyZWF0ZWQ6IGRpbT0ke2RpbWVuc2lvbn0sIE09JHttYXhDb25uZWN0aW9uc30sIGVmQz0ke2VmQ29uc3RydWN0aW9ufSwgZWZTPTUwMGApO1xuICAgICAgICB9XG4gICAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgICAgY29uc29sZS53YXJuKCdbU29jaERCXSBOYXRpdmUgSE5TVyBjcmVhdGlvbiBmYWlsZWQ6JywgZS5tZXNzYWdlKTtcbiAgICAgICAgdGhpcy5uYXRpdmVJbmRleFB0ciA9IG51bGw7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFNldCBlZl9zZWFyY2ggcGFyYW1ldGVyIGZvciBITlNXIHNlYXJjaCAoY29udHJvbHMgcmVjYWxsIHZzIHNwZWVkIHRyYWRlb2ZmKVxuICAgKiBIaWdoZXIgdmFsdWVzID0gYmV0dGVyIHJlY2FsbCBidXQgc2xvd2VyIHNlYXJjaFxuICAgKiBAcGFyYW0gZWZTZWFyY2ggLSBUeXBpY2FsbHkgMTAwLTEwMDAsIGRlZmF1bHQgaXMgNTAwXG4gICAqL1xuICBzZXRFZlNlYXJjaChlZlNlYXJjaDogbnVtYmVyKTogdm9pZCB7XG4gICAgaWYgKE5hdGl2ZUhuc3cgJiYgdGhpcy5uYXRpdmVJbmRleFB0cikge1xuICAgICAgTmF0aXZlSG5zdy5obnN3X3NldF9lZl9zZWFyY2godGhpcy5uYXRpdmVJbmRleFB0ciwgZWZTZWFyY2gpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBJbnNlcnQgYSB2ZWN0b3Igd2l0aCBvcHRpb25hbCBtZXRhZGF0YVxuICAgKiBWZWN0b3IgaXMgaW1tZWRpYXRlbHkgaW5kZXhlZCAoc3luY2hyb25vdXMpXG4gICAqL1xuICBhc3luYyBpbnNlcnQoXG4gICAgdmVjdG9yOiBudW1iZXJbXSxcbiAgICBtZXRhZGF0YT86IFJlY29yZDxzdHJpbmcsIGFueT4sXG4gICAgaWQ/OiBzdHJpbmdcbiAgKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgICBpZiAodGhpcy5jb25maWcuZGltZW5zaW9uICYmIHZlY3Rvci5sZW5ndGggIT09IHRoaXMuY29uZmlnLmRpbWVuc2lvbikge1xuICAgICAgdGhyb3cgbmV3IERhdGFiYXNlRXJyb3IoXG4gICAgICAgIGBWZWN0b3IgZGltZW5zaW9uIG1pc21hdGNoOiBleHBlY3RlZCAke3RoaXMuY29uZmlnLmRpbWVuc2lvbn0sIGdvdCAke3ZlY3Rvci5sZW5ndGh9YFxuICAgICAgKTtcbiAgICB9XG5cbiAgICBjb25zdCB2ZWN0b3JJZCA9IGlkIHx8IHRoaXMuZ2VuZXJhdGVJZCgpO1xuICAgIGNvbnN0IGtleSA9IHRoaXMudmVjdG9yS2V5KHZlY3RvcklkKTtcbiAgICBcbiAgICBjb25zdCBkYXRhID0ge1xuICAgICAgdmVjdG9yLFxuICAgICAgbWV0YWRhdGE6IG1ldGFkYXRhIHx8IHt9LFxuICAgICAgdGltZXN0YW1wOiBEYXRlLm5vdygpLFxuICAgIH07XG5cbiAgICAvLyBTdG9yZSB0byBkYXRhYmFzZVxuICAgIGF3YWl0IHRoaXMuZGIucHV0KEJ1ZmZlci5mcm9tKGtleSksIEJ1ZmZlci5mcm9tKEpTT04uc3RyaW5naWZ5KGRhdGEpKSk7XG4gICAgXG4gICAgLy8gU1lOQ0hST05PVVNMWSBhZGQgdG8gaW4tbWVtb3J5IGluZGV4XG4gICAgdGhpcy52ZWN0b3JJbmRleC5hZGQodmVjdG9ySWQsIHZlY3RvciwgbWV0YWRhdGEpO1xuICAgIFxuICAgIC8vIEFsc28gYWRkIHRvIG5hdGl2ZSBITlNXIGluZGV4IGlmIGF2YWlsYWJsZSAoZm9yIHNpbmdsZSBpbnNlcnRzKVxuICAgIGlmIChOYXRpdmVIbnN3ICYmIHRoaXMubmF0aXZlSW5kZXhQdHIpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IGRpbWVuc2lvbiA9IHRoaXMuY29uZmlnLmRpbWVuc2lvbiB8fCB2ZWN0b3IubGVuZ3RoO1xuICAgICAgICAvLyBVc2UgYXV0by1pbmNyZW1lbnRpbmcgY291bnRlciBmb3IgbmF0aXZlIEhOU1cgKGl0IG5lZWRzIG51bWVyaWMgSURzKVxuICAgICAgICBjb25zdCBudW1lcmljSWQgPSB0aGlzLm5hdGl2ZUlkQ291bnRlcisrO1xuICAgICAgICB0aGlzLm5hdGl2ZUlkVG9TdHJpbmdJZC5zZXQobnVtZXJpY0lkLCB2ZWN0b3JJZCk7ICAvLyBTdG9yZSBtYXBwaW5nXG4gICAgICAgIGNvbnN0IGlkQXJyYXkgPSBuZXcgQmlnVWludDY0QXJyYXkoW0JpZ0ludChudW1lcmljSWQpXSk7XG4gICAgICAgIGNvbnN0IHZlY3RvckFycmF5ID0gbmV3IEZsb2F0MzJBcnJheSh2ZWN0b3IpO1xuICAgICAgICAvLyBBcmdzOiBwdHIsIGlkcywgdmVjdG9ycywgbnVtX3ZlY3RvcnMsIGRpbWVuc2lvblxuICAgICAgICBOYXRpdmVIbnN3Lmhuc3dfaW5zZXJ0X2JhdGNoKHRoaXMubmF0aXZlSW5kZXhQdHIsIGlkQXJyYXksIHZlY3RvckFycmF5LCAxLCBkaW1lbnNpb24pO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICAvLyBGYWxsYmFjayB0byBKUyBpbmRleCBvbmx5IChhbHJlYWR5IGFkZGVkIGFib3ZlKVxuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiB2ZWN0b3JJZDtcbiAgfVxuXG4gIC8qKlxuICAgKiBJbnNlcnQgbXVsdGlwbGUgdmVjdG9ycyB1c2luZyBOQVRJVkUgSE5TVyBiYXRjaCBpbnNlcnQgd2hlbiBhdmFpbGFibGVcbiAgICogVGhpcyBpcyB0aGUgT1BUSU1JWkVEIHBhdGggLSB1c2VzIEZGSSBiYXRjaCBpbnNlcnQgZm9yIH4xMDB4IHNwZWVkdXBcbiAgICovXG4gIGFzeW5jIGluc2VydE1hbnkoXG4gICAgdmVjdG9yczogbnVtYmVyW11bXSxcbiAgICBtZXRhZGF0YXM/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+W10sXG4gICAgaWRzPzogc3RyaW5nW11cbiAgKTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICAgIGlmICh2ZWN0b3JzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIFtdO1xuICAgIH1cbiAgICBcbiAgICBjb25zdCByZXN1bHRJZHMgPSBpZHMgfHwgdmVjdG9ycy5tYXAoKF8sIGkpID0+IGkudG9TdHJpbmcoKSk7XG4gICAgY29uc3QgZGltZW5zaW9uID0gdGhpcy5jb25maWcuZGltZW5zaW9uIHx8IHZlY3RvcnNbMF0ubGVuZ3RoO1xuICAgIFxuICAgIC8vIElmIG5hdGl2ZSBpbmRleCBpcyBhdmFpbGFibGUsIHVzZSBiYXRjaCBpbnNlcnRcbiAgICBpZiAoTmF0aXZlSG5zdyAmJiB0aGlzLm5hdGl2ZUluZGV4UHRyKSB7XG4gICAgICB0cnkge1xuICAgICAgICBjb25zb2xlLmxvZyhgW1NvY2hEQl0gVXNpbmcgTkFUSVZFIGJhdGNoIGluc2VydCBmb3IgJHt2ZWN0b3JzLmxlbmd0aH0gdmVjdG9ycy4uLmApO1xuICAgICAgICBjb25zdCBzdGFydFRpbWUgPSBwZXJmb3JtYW5jZS5ub3coKTtcbiAgICAgICAgXG4gICAgICAgIC8vIENvbnZlcnQgSURzIHRvIG51bWVyaWMgKHU2NCkgdXNpbmcgY291bnRlciBhbmQgc3RvcmUgbWFwcGluZ1xuICAgICAgICBjb25zdCBudW1lcmljSWRzID0gbmV3IEJpZ1VpbnQ2NEFycmF5KHJlc3VsdElkcy5sZW5ndGgpO1xuICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IHJlc3VsdElkcy5sZW5ndGg7IGkrKykge1xuICAgICAgICAgIGNvbnN0IG51bWVyaWNJZCA9IHRoaXMubmF0aXZlSWRDb3VudGVyKys7XG4gICAgICAgICAgbnVtZXJpY0lkc1tpXSA9IEJpZ0ludChudW1lcmljSWQpO1xuICAgICAgICAgIHRoaXMubmF0aXZlSWRUb1N0cmluZ0lkLnNldChudW1lcmljSWQsIHJlc3VsdElkc1tpXSk7XG4gICAgICAgIH1cbiAgICAgICAgXG4gICAgICAgIC8vIEZsYXR0ZW4gdmVjdG9ycyBpbnRvIGNvbnRpZ3VvdXMgRmxvYXQzMkFycmF5XG4gICAgICAgIGNvbnN0IGZsYXRWZWN0b3JzID0gbmV3IEZsb2F0MzJBcnJheSh2ZWN0b3JzLmxlbmd0aCAqIGRpbWVuc2lvbik7XG4gICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgdmVjdG9ycy5sZW5ndGg7IGkrKykge1xuICAgICAgICAgIGZsYXRWZWN0b3JzLnNldCh2ZWN0b3JzW2ldLCBpICogZGltZW5zaW9uKTtcbiAgICAgICAgfVxuICAgICAgICBcbiAgICAgICAgLy8gU2luZ2xlIEZGSSBjYWxsIHRvIG5hdGl2ZSBiYXRjaCBpbnNlcnRcbiAgICAgICAgY29uc3QgcmVzdWx0ID0gTmF0aXZlSG5zdy5obnN3X2luc2VydF9iYXRjaChcbiAgICAgICAgICB0aGlzLm5hdGl2ZUluZGV4UHRyLFxuICAgICAgICAgIG51bWVyaWNJZHMsXG4gICAgICAgICAgZmxhdFZlY3RvcnMsXG4gICAgICAgICAgdmVjdG9ycy5sZW5ndGgsXG4gICAgICAgICAgZGltZW5zaW9uXG4gICAgICAgICk7XG4gICAgICAgIFxuICAgICAgICBjb25zdCBlbGFwc2VkID0gKHBlcmZvcm1hbmNlLm5vdygpIC0gc3RhcnRUaW1lKSAvIDEwMDA7XG4gICAgICAgIGNvbnN0IGluZGV4U2l6ZSA9IE5hdGl2ZUhuc3cuaG5zd19sZW4odGhpcy5uYXRpdmVJbmRleFB0cik7XG4gICAgICAgIGNvbnNvbGUubG9nKGBbU29jaERCXSBOYXRpdmUgYmF0Y2ggaW5zZXJ0OiByZXN1bHQ9JHtyZXN1bHR9LCBpbmRleF9zaXplPSR7aW5kZXhTaXplfSwgdGltZT0ke2VsYXBzZWQudG9GaXhlZCgzKX1zICgkeyh2ZWN0b3JzLmxlbmd0aC9lbGFwc2VkKS50b0ZpeGVkKDApfSB2ZWMvc2VjKWApO1xuICAgICAgICBcbiAgICAgICAgaWYgKHJlc3VsdCA8IDApIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYE5hdGl2ZSBiYXRjaCBpbnNlcnQgZmFpbGVkIHdpdGggZXJyb3IgY29kZSAke3Jlc3VsdH1gKTtcbiAgICAgICAgfVxuICAgICAgICBcbiAgICAgICAgLy8gQWxzbyBhZGQgdG8gSlMgaW5kZXggZm9yIG1ldGFkYXRhIGxvb2t1cHNcbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCByZXN1bHRJZHMubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgICB0aGlzLnZlY3RvckluZGV4LmFkZChyZXN1bHRJZHNbaV0sIHZlY3RvcnNbaV0sIG1ldGFkYXRhcyA/IG1ldGFkYXRhc1tpXSA6IHVuZGVmaW5lZCk7XG4gICAgICAgIH1cbiAgICAgICAgXG4gICAgICAgIHJldHVybiByZXN1bHRJZHM7XG4gICAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgICAgY29uc29sZS53YXJuKCdbU29jaERCXSBOYXRpdmUgYmF0Y2ggaW5zZXJ0IGZhaWxlZCwgZmFsbGluZyBiYWNrIHRvIHNlcXVlbnRpYWw6JywgZS5tZXNzYWdlKTtcbiAgICAgIH1cbiAgICB9XG4gICAgXG4gICAgLy8gRmFsbGJhY2s6IHNlcXVlbnRpYWwgaW5zZXJ0IChzbG93IHBhdGgpXG4gICAgY29uc29sZS5sb2coYFtTb2NoREJdIFVzaW5nIFNFUVVFTlRJQUwgaW5zZXJ0IGZvciAke3ZlY3RvcnMubGVuZ3RofSB2ZWN0b3JzIChzbG93IHBhdGgpLi4uYCk7XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCB2ZWN0b3JzLmxlbmd0aDsgaSsrKSB7XG4gICAgICBjb25zdCBpZCA9IHJlc3VsdElkc1tpXTtcbiAgICAgIGNvbnN0IG1ldGFkYXRhID0gbWV0YWRhdGFzID8gbWV0YWRhdGFzW2ldIDogdW5kZWZpbmVkO1xuICAgICAgYXdhaXQgdGhpcy5pbnNlcnQodmVjdG9yc1tpXSwgbWV0YWRhdGEsIGlkKTtcbiAgICB9XG4gICAgcmV0dXJuIHJlc3VsdElkcztcbiAgfVxuXG4gIC8qKlxuICAgKiBSZWJ1aWxkIGluZGV4IGZyb20gZGF0YWJhc2UgKGZvciByZWNvdmVyeS9zdGFydHVwKVxuICAgKi9cbiAgYXN5bmMgcmVidWlsZEluZGV4KCk6IFByb21pc2U8bnVtYmVyPiB7XG4gICAgY29uc3QgcHJlZml4ID0gdGhpcy52ZWN0b3JLZXlQcmVmaXgoKTtcbiAgICBsZXQgY291bnQgPSAwO1xuXG4gICAgdHJ5IHtcbiAgICAgIGZvciBhd2FpdCAoY29uc3QgW2tleUJ1ZmZlciwgdmFsdWVCdWZmZXJdIG9mIHRoaXMuZGIuc2NhblByZWZpeChCdWZmZXIuZnJvbShwcmVmaXgpKSkge1xuICAgICAgICBjb25zdCBrZXkgPSBrZXlCdWZmZXIudG9TdHJpbmcoKTtcbiAgICAgICAgY29uc3QgaWQgPSBrZXkucmVwbGFjZShwcmVmaXgsICcnKTtcbiAgICAgICAgY29uc3QgZGF0YSA9IEpTT04ucGFyc2UodmFsdWVCdWZmZXIudG9TdHJpbmcoKSk7XG4gICAgICAgIFxuICAgICAgICB0aGlzLnZlY3RvckluZGV4LmFkZChpZCwgZGF0YS52ZWN0b3IsIGRhdGEubWV0YWRhdGEpO1xuICAgICAgICBjb3VudCsrO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBjb25zb2xlLndhcm4oJ1tTb2NoREJdIEVycm9yIHJlYnVpbGRpbmcgdmVjdG9yIGluZGV4OicsIGVycm9yKTtcbiAgICB9XG5cbiAgICB0aGlzLl9pbmRleFJlYWR5ID0gdHJ1ZTtcbiAgICByZXR1cm4gY291bnQ7XG4gIH1cblxuICAvKipcbiAgICogQ2hlY2sgaWYgaW5kZXggaXMgcmVhZHkgKGxvYWRlZCBmcm9tIGRpc2spXG4gICAqL1xuICBnZXQgaXNJbmRleFJlYWR5KCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLl9pbmRleFJlYWR5IHx8IHRoaXMudmVjdG9ySW5kZXguc2l6ZSgpID4gMDtcbiAgfVxuXG4gIC8qKlxuICAgKiBTZWFyY2ggZm9yIHNpbWlsYXIgdmVjdG9yc1xuICAgKiBVc2VzIE5BVElWRSBITlNXIHNlYXJjaCAoTyhsb2cgTikpIHdoZW4gYXZhaWxhYmxlLCBmYWxscyBiYWNrIHRvIEpTIGJydXRlLWZvcmNlXG4gICAqL1xuICBhc3luYyBzZWFyY2gocmVxdWVzdDogU2VhcmNoUmVxdWVzdCk6IFByb21pc2U8U2VhcmNoUmVzdWx0W10+IHtcbiAgICBjb25zdCBrID0gcmVxdWVzdC5rO1xuICAgIGNvbnN0IHF1ZXJ5VmVjdG9yID0gcmVxdWVzdC5xdWVyeVZlY3RvcjtcbiAgICBcbiAgICAvLyBUcnkgbmF0aXZlIEhOU1cgc2VhcmNoIGZpcnN0IChtdWNoIGZhc3RlciBmb3IgbGFyZ2UgZGF0YXNldHMpXG4gICAgaWYgKE5hdGl2ZUhuc3cgJiYgdGhpcy5uYXRpdmVJbmRleFB0ciAmJiBOYXRpdmVIbnN3Lmhuc3dfc2VhcmNoKSB7XG4gICAgICB0cnkge1xuICAgICAgICBjb25zdCBkaW1lbnNpb24gPSBxdWVyeVZlY3Rvci5sZW5ndGg7XG4gICAgICAgIFxuICAgICAgICAvLyBQcmVwYXJlIHF1ZXJ5IGFzIEZsb2F0MzJBcnJheVxuICAgICAgICBjb25zdCBxdWVyeUFycmF5ID0gbmV3IEZsb2F0MzJBcnJheShxdWVyeVZlY3Rvcik7XG4gICAgICAgIFxuICAgICAgICAvLyBBbGxvY2F0ZSBvdXRwdXQgYnVmZmVyc1xuICAgICAgICAvLyBDU2VhcmNoUmVzdWx0OiB7IGlkX2xvOiB1aW50NjQsIGlkX2hpOiB1aW50NjQsIGRpc3RhbmNlOiBmbG9hdCB9XG4gICAgICAgIC8vIFNpemU6IDggKyA4ICsgNCA9IDIwIGJ5dGVzIHBlciByZXN1bHQsIGFsaWduZWQgdG8gMjQgYnl0ZXNcbiAgICAgICAgY29uc3QgcmVzdWx0U2l6ZSA9IDI0O1xuICAgICAgICBjb25zdCByZXN1bHRzQnVmZmVyID0gQnVmZmVyLmFsbG9jKHJlc3VsdFNpemUgKiBrKTtcbiAgICAgICAgY29uc3QgbnVtUmVzdWx0c0J1ZmZlciA9IEJ1ZmZlci5hbGxvYyg4KTsgIC8vIHNpemVfdFxuICAgICAgICBcbiAgICAgICAgLy8gQ2FsbCBuYXRpdmUgc2VhcmNoXG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IE5hdGl2ZUhuc3cuaG5zd19zZWFyY2goXG4gICAgICAgICAgdGhpcy5uYXRpdmVJbmRleFB0cixcbiAgICAgICAgICBxdWVyeUFycmF5LFxuICAgICAgICAgIGRpbWVuc2lvbixcbiAgICAgICAgICBrLFxuICAgICAgICAgIHJlc3VsdHNCdWZmZXIsXG4gICAgICAgICAgbnVtUmVzdWx0c0J1ZmZlclxuICAgICAgICApO1xuICAgICAgICBcbiAgICAgICAgaWYgKHJlc3VsdCA9PT0gMCkge1xuICAgICAgICAgIGNvbnN0IG51bVJlc3VsdHMgPSBudW1SZXN1bHRzQnVmZmVyLnJlYWRCaWdVSW50NjRMRSgwKTtcbiAgICAgICAgICBjb25zdCBuYXRpdmVSZXN1bHRzOiBTZWFyY2hSZXN1bHRbXSA9IFtdO1xuICAgICAgICAgIFxuICAgICAgICAgIC8vIFJlYWQgcmVzdWx0cyBmcm9tIGJ1ZmZlclxuICAgICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgTWF0aC5taW4oTnVtYmVyKG51bVJlc3VsdHMpLCBrKTsgaSsrKSB7XG4gICAgICAgICAgICBjb25zdCBvZmZzZXQgPSBpICogcmVzdWx0U2l6ZTtcbiAgICAgICAgICAgIGNvbnN0IGlkX2xvID0gcmVzdWx0c0J1ZmZlci5yZWFkQmlnVUludDY0TEUob2Zmc2V0KTtcbiAgICAgICAgICAgIGNvbnN0IGRpc3RhbmNlID0gcmVzdWx0c0J1ZmZlci5yZWFkRmxvYXRMRShvZmZzZXQgKyAxNik7XG4gICAgICAgICAgICBcbiAgICAgICAgICAgIC8vIE1hcCBudW1lcmljIElEIGJhY2sgdG8gc3RyaW5nIElEXG4gICAgICAgICAgICBjb25zdCBudW1lcmljSWQgPSBOdW1iZXIoaWRfbG8pO1xuICAgICAgICAgICAgY29uc3Qgc3RyaW5nSWQgPSB0aGlzLm5hdGl2ZUlkVG9TdHJpbmdJZC5nZXQobnVtZXJpY0lkKSB8fCBpZF9sby50b1N0cmluZygpO1xuICAgICAgICAgICAgY29uc3QgZGF0YSA9ICh0aGlzLnZlY3RvckluZGV4IGFzIGFueSkudmVjdG9ycz8uZ2V0KHN0cmluZ0lkKTtcbiAgICAgICAgICAgIG5hdGl2ZVJlc3VsdHMucHVzaCh7XG4gICAgICAgICAgICAgIGlkOiBzdHJpbmdJZCxcbiAgICAgICAgICAgICAgc2NvcmU6IDEgLSBkaXN0YW5jZSwgIC8vIENvbnZlcnQgZGlzdGFuY2UgdG8gc2ltaWxhcml0eVxuICAgICAgICAgICAgICB2ZWN0b3I6IHJlcXVlc3QuaW5jbHVkZU1ldGFkYXRhICYmIGRhdGEgPyBkYXRhLnZlY3RvciA6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgICAgbWV0YWRhdGE6IHJlcXVlc3QuaW5jbHVkZU1ldGFkYXRhICYmIGRhdGEgPyBkYXRhLm1ldGFkYXRhIDogdW5kZWZpbmVkLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICAgIFxuICAgICAgICAgIHJldHVybiBuYXRpdmVSZXN1bHRzO1xuICAgICAgICB9XG4gICAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgICAgLy8gRmFsbCBiYWNrIHRvIEpTIHNlYXJjaCBvbiBlcnJvclxuICAgICAgICBjb25zb2xlLndhcm4oJ1tTb2NoREJdIE5hdGl2ZSBzZWFyY2ggZmFpbGVkOicsIGUubWVzc2FnZSk7XG4gICAgICB9XG4gICAgfVxuICAgIFxuICAgIC8vIEZhbGxiYWNrOiBKUyBicnV0ZS1mb3JjZSBzZWFyY2hcbiAgICAvLyBBdXRvLXJlYnVpbGQgaW5kZXggaWYgZW1wdHkgYnV0IHRoZXJlIG1pZ2h0IGJlIGRhdGFcbiAgICBpZiAoIXRoaXMuaXNJbmRleFJlYWR5ICYmIHRoaXMudmVjdG9ySW5kZXguc2l6ZSgpID09PSAwKSB7XG4gICAgICBhd2FpdCB0aGlzLnJlYnVpbGRJbmRleCgpO1xuICAgIH1cblxuICAgIGNvbnN0IHJlc3VsdHMgPSB0aGlzLnZlY3RvckluZGV4LnNlYXJjaChcbiAgICAgIHJlcXVlc3QucXVlcnlWZWN0b3IsXG4gICAgICByZXF1ZXN0LmssXG4gICAgICByZXF1ZXN0LmZpbHRlclxuICAgICk7XG5cbiAgICAvLyBNYXAgcmVzdWx0cyBiYXNlZCBvbiBpbmNsdWRlTWV0YWRhdGEgZmxhZ1xuICAgIHJldHVybiByZXN1bHRzLm1hcChyID0+ICh7XG4gICAgICBpZDogci5pZCxcbiAgICAgIHNjb3JlOiByLnNjb3JlLFxuICAgICAgdmVjdG9yOiByZXF1ZXN0LmluY2x1ZGVNZXRhZGF0YSA/IHIudmVjdG9yIDogdW5kZWZpbmVkLFxuICAgICAgbWV0YWRhdGE6IHJlcXVlc3QuaW5jbHVkZU1ldGFkYXRhID8gci5tZXRhZGF0YSA6IHVuZGVmaW5lZCxcbiAgICB9KSk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IGEgdmVjdG9yIGJ5IElEXG4gICAqL1xuICBhc3luYyBnZXQoaWQ6IHN0cmluZyk6IFByb21pc2U8eyB2ZWN0b3I6IG51bWJlcltdOyBtZXRhZGF0YT86IFJlY29yZDxzdHJpbmcsIGFueT4gfSB8IG51bGw+IHtcbiAgICBjb25zdCBrZXkgPSB0aGlzLnZlY3RvcktleShpZCk7XG4gICAgY29uc3QgdmFsdWUgPSBhd2FpdCB0aGlzLmRiLmdldChCdWZmZXIuZnJvbShrZXkpKTtcbiAgICBcbiAgICBpZiAoIXZhbHVlKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG4gICAgXG4gICAgY29uc3QgZGF0YSA9IEpTT04ucGFyc2UodmFsdWUudG9TdHJpbmcoKSk7XG4gICAgcmV0dXJuIHtcbiAgICAgIHZlY3RvcjogZGF0YS52ZWN0b3IsXG4gICAgICBtZXRhZGF0YTogZGF0YS5tZXRhZGF0YSxcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIERlbGV0ZSBhIHZlY3RvciBieSBJRFxuICAgKi9cbiAgYXN5bmMgZGVsZXRlKGlkOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+IHtcbiAgICBjb25zdCBrZXkgPSB0aGlzLnZlY3RvcktleShpZCk7XG4gICAgYXdhaXQgdGhpcy5kYi5kZWxldGUoQnVmZmVyLmZyb20oa2V5KSk7XG4gICAgLy8gUmVtb3ZlIGZyb20gaW4tbWVtb3J5IGluZGV4XG4gICAgdGhpcy52ZWN0b3JJbmRleC5yZW1vdmUoaWQpO1xuICAgIHJldHVybiB0cnVlO1xuICB9XG5cbiAgLyoqXG4gICAqIENvdW50IHZlY3RvcnMgaW4gY29sbGVjdGlvblxuICAgKi9cbiAgYXN5bmMgY291bnQoKTogUHJvbWlzZTxudW1iZXI+IHtcbiAgICAvLyBJZiBpbmRleCBpcyBsb2FkZWQsIHJldHVybiBmcm9tIGluZGV4XG4gICAgaWYgKHRoaXMuaXNJbmRleFJlYWR5KSB7XG4gICAgICByZXR1cm4gdGhpcy52ZWN0b3JJbmRleC5zaXplKCk7XG4gICAgfVxuICAgIC8vIE90aGVyd2lzZSByZWJ1aWxkIGFuZCBjb3VudFxuICAgIHJldHVybiBhd2FpdCB0aGlzLnJlYnVpbGRJbmRleCgpO1xuICB9XG5cbiAgLy8gSGVscGVyIG1ldGhvZHNcbiAgcHJpdmF0ZSB2ZWN0b3JLZXkoaWQ6IHN0cmluZyk6IHN0cmluZyB7XG4gICAgcmV0dXJuIGBfY29sbGVjdGlvbi8ke3RoaXMubmFtZXNwYWNlfS8ke3RoaXMubmFtZX0vdmVjdG9ycy8ke2lkfWA7XG4gIH1cblxuICBwcml2YXRlIHZlY3RvcktleVByZWZpeCgpOiBzdHJpbmcge1xuICAgIHJldHVybiBgX2NvbGxlY3Rpb24vJHt0aGlzLm5hbWVzcGFjZX0vJHt0aGlzLm5hbWV9L3ZlY3RvcnMvYDtcbiAgfVxuXG4gIHByaXZhdGUgbWV0YWRhdGFLZXkoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gYF9jb2xsZWN0aW9uLyR7dGhpcy5uYW1lc3BhY2V9LyR7dGhpcy5uYW1lfS9tZXRhZGF0YWA7XG4gIH1cblxuICBwcml2YXRlIGdlbmVyYXRlSWQoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gYCR7RGF0ZS5ub3coKX0tJHtNYXRoLnJhbmRvbSgpLnRvU3RyaW5nKDM2KS5zdWJzdHIoMiwgOSl9YDtcbiAgfVxuXG4gIC8vIENhbGN1bGF0ZSBjb3NpbmUgc2ltaWxhcml0eVxuICBwcml2YXRlIGNvc2luZVNpbWlsYXJpdHkoYTogbnVtYmVyW10sIGI6IG51bWJlcltdKTogbnVtYmVyIHtcbiAgICBsZXQgZG90UHJvZHVjdCA9IDA7XG4gICAgbGV0IG5vcm1BID0gMDtcbiAgICBsZXQgbm9ybUIgPSAwO1xuICAgIFxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgYS5sZW5ndGg7IGkrKykge1xuICAgICAgZG90UHJvZHVjdCArPSBhW2ldICogYltpXTtcbiAgICAgIG5vcm1BICs9IGFbaV0gKiBhW2ldO1xuICAgICAgbm9ybUIgKz0gYltpXSAqIGJbaV07XG4gICAgfVxuICAgIFxuICAgIHJldHVybiBkb3RQcm9kdWN0IC8gKE1hdGguc3FydChub3JtQSkgKiBNYXRoLnNxcnQobm9ybUIpKTtcbiAgfVxufVxuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBOYW1lc3BhY2UgSGFuZGxlXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCBjbGFzcyBOYW1lc3BhY2Uge1xuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIGRiOiBhbnksXG4gICAgcHJpdmF0ZSBuYW1lOiBzdHJpbmcsXG4gICAgcHJpdmF0ZSBjb25maWc6IE5hbWVzcGFjZUNvbmZpZ1xuICApIHt9XG5cbiAgLyoqXG4gICAqIENyZWF0ZSBhIG5ldyBjb2xsZWN0aW9uIGluIHRoaXMgbmFtZXNwYWNlXG4gICAqL1xuICBhc3luYyBjcmVhdGVDb2xsZWN0aW9uKGNvbmZpZzogQ29sbGVjdGlvbkNvbmZpZyk6IFByb21pc2U8Q29sbGVjdGlvbj4ge1xuICAgIGNvbnN0IG1ldGFkYXRhS2V5ID0gYF9jb2xsZWN0aW9uLyR7dGhpcy5uYW1lfS8ke2NvbmZpZy5uYW1lfS9tZXRhZGF0YWA7XG4gICAgXG4gICAgLy8gQ2hlY2sgaWYgY29sbGVjdGlvbiBhbHJlYWR5IGV4aXN0c1xuICAgIGNvbnN0IGV4aXN0aW5nID0gYXdhaXQgdGhpcy5kYi5nZXQoQnVmZmVyLmZyb20obWV0YWRhdGFLZXkpKTtcbiAgICBpZiAoZXhpc3RpbmcpIHtcbiAgICAgIHRocm93IG5ldyBDb2xsZWN0aW9uRXhpc3RzRXJyb3IoY29uZmlnLm5hbWUpO1xuICAgIH1cblxuICAgIC8vIFN0b3JlIGNvbGxlY3Rpb24gbWV0YWRhdGFcbiAgICBjb25zdCBtZXRhZGF0YSA9IHtcbiAgICAgIC4uLmNvbmZpZyxcbiAgICAgIGNyZWF0ZWRBdDogRGF0ZS5ub3coKSxcbiAgICB9O1xuICAgIFxuICAgIGF3YWl0IHRoaXMuZGIucHV0KFxuICAgICAgQnVmZmVyLmZyb20obWV0YWRhdGFLZXkpLFxuICAgICAgQnVmZmVyLmZyb20oSlNPTi5zdHJpbmdpZnkobWV0YWRhdGEpKVxuICAgICk7XG5cbiAgICByZXR1cm4gbmV3IENvbGxlY3Rpb24odGhpcy5kYiwgdGhpcy5uYW1lLCBjb25maWcubmFtZSwgY29uZmlnKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgYW4gZXhpc3RpbmcgY29sbGVjdGlvblxuICAgKi9cbiAgYXN5bmMgY29sbGVjdGlvbihuYW1lOiBzdHJpbmcpOiBQcm9taXNlPENvbGxlY3Rpb24+IHtcbiAgICBjb25zdCBtZXRhZGF0YUtleSA9IGBfY29sbGVjdGlvbi8ke3RoaXMubmFtZX0vJHtuYW1lfS9tZXRhZGF0YWA7XG4gICAgY29uc3QgbWV0YWRhdGEgPSBhd2FpdCB0aGlzLmRiLmdldChCdWZmZXIuZnJvbShtZXRhZGF0YUtleSkpO1xuICAgIFxuICAgIGlmICghbWV0YWRhdGEpIHtcbiAgICAgIHRocm93IG5ldyBDb2xsZWN0aW9uTm90Rm91bmRFcnJvcihuYW1lKTtcbiAgICB9XG5cbiAgICBjb25zdCBjb25maWcgPSBKU09OLnBhcnNlKG1ldGFkYXRhLnRvU3RyaW5nKCkpO1xuICAgIHJldHVybiBuZXcgQ29sbGVjdGlvbih0aGlzLmRiLCB0aGlzLm5hbWUsIG5hbWUsIGNvbmZpZyk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IG9yIGNyZWF0ZSBhIGNvbGxlY3Rpb25cbiAgICovXG4gIGFzeW5jIGdldE9yQ3JlYXRlQ29sbGVjdGlvbihjb25maWc6IENvbGxlY3Rpb25Db25maWcpOiBQcm9taXNlPENvbGxlY3Rpb24+IHtcbiAgICB0cnkge1xuICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuY29sbGVjdGlvbihjb25maWcubmFtZSk7XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGlmIChlcnJvciBpbnN0YW5jZW9mIENvbGxlY3Rpb25Ob3RGb3VuZEVycm9yKSB7XG4gICAgICAgIHJldHVybiBhd2FpdCB0aGlzLmNyZWF0ZUNvbGxlY3Rpb24oY29uZmlnKTtcbiAgICAgIH1cbiAgICAgIHRocm93IGVycm9yO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBEZWxldGUgYSBjb2xsZWN0aW9uXG4gICAqL1xuICBhc3luYyBkZWxldGVDb2xsZWN0aW9uKG5hbWU6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGNvbnN0IG1ldGFkYXRhS2V5ID0gYF9jb2xsZWN0aW9uLyR7dGhpcy5uYW1lfS8ke25hbWV9L21ldGFkYXRhYDtcbiAgICBjb25zdCBwcmVmaXggPSBCdWZmZXIuZnJvbShgX2NvbGxlY3Rpb24vJHt0aGlzLm5hbWV9LyR7bmFtZX0vYCk7XG5cbiAgICAvLyBEZWxldGUgYWxsIGtleXMgd2l0aCB0aGlzIGNvbGxlY3Rpb24gcHJlZml4ICh2ZWN0b3JzLCBtZXRhZGF0YSwgZXRjLilcbiAgICB0cnkge1xuICAgICAgY29uc3QgdG9EZWxldGU6IEJ1ZmZlcltdID0gW107XG4gICAgICBmb3IgYXdhaXQgKGNvbnN0IFtrZXlCdWZdIG9mIHRoaXMuZGIuc2NhblByZWZpeChwcmVmaXgpKSB7XG4gICAgICAgIHRvRGVsZXRlLnB1c2goa2V5QnVmKTtcbiAgICAgIH1cbiAgICAgIGZvciAoY29uc3Qga2V5IG9mIHRvRGVsZXRlKSB7XG4gICAgICAgIGF3YWl0IHRoaXMuZGIuZGVsZXRlKGtleSk7XG4gICAgICB9XG4gICAgfSBjYXRjaCB7XG4gICAgICAvLyBJZiBzY2FuUHJlZml4IGZhaWxzLCBmYWxsIGJhY2sgdG8ganVzdCBkZWxldGluZyB0aGUgbWV0YWRhdGEga2V5XG4gICAgICBhd2FpdCB0aGlzLmRiLmRlbGV0ZShCdWZmZXIuZnJvbShtZXRhZGF0YUtleSkpO1xuICAgIH1cblxuICAgIHJldHVybiB0cnVlO1xuICB9XG5cbiAgLyoqXG4gICAqIExpc3QgYWxsIGNvbGxlY3Rpb25zIGluIHRoaXMgbmFtZXNwYWNlXG4gICAqL1xuICBhc3luYyBsaXN0Q29sbGVjdGlvbnMoKTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICAgIGNvbnN0IHByZWZpeCA9IEJ1ZmZlci5mcm9tKGBfY29sbGVjdGlvbi8ke3RoaXMubmFtZX0vYCk7XG4gICAgY29uc3QgY29sbGVjdGlvbnMgPSBuZXcgU2V0PHN0cmluZz4oKTtcblxuICAgIHRyeSB7XG4gICAgICBmb3IgYXdhaXQgKGNvbnN0IFtrZXlCdWZdIG9mIHRoaXMuZGIuc2NhblByZWZpeChwcmVmaXgpKSB7XG4gICAgICAgIGNvbnN0IGtleSA9IGtleUJ1Zi50b1N0cmluZygpO1xuICAgICAgICAvLyBLZXkgZm9ybWF0OiBfY29sbGVjdGlvbi97bmFtZXNwYWNlfS97Y29sbGVjdGlvbk5hbWV9Ly4uLlxuICAgICAgICBjb25zdCBhZnRlclByZWZpeCA9IGtleS5zdWJzdHJpbmcoYF9jb2xsZWN0aW9uLyR7dGhpcy5uYW1lfS9gLmxlbmd0aCk7XG4gICAgICAgIGNvbnN0IGNvbGxlY3Rpb25OYW1lID0gYWZ0ZXJQcmVmaXguc3BsaXQoJy8nKVswXTtcbiAgICAgICAgaWYgKGNvbGxlY3Rpb25OYW1lKSB7XG4gICAgICAgICAgY29sbGVjdGlvbnMuYWRkKGNvbGxlY3Rpb25OYW1lKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0gY2F0Y2gge1xuICAgICAgLy8gU2NhbiBub3QgYXZhaWxhYmxlXG4gICAgfVxuXG4gICAgcmV0dXJuIEFycmF5LmZyb20oY29sbGVjdGlvbnMpO1xuICB9XG5cbiAgZ2V0TmFtZSgpOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLm5hbWU7XG4gIH1cblxuICBnZXRDb25maWcoKTogTmFtZXNwYWNlQ29uZmlnIHtcbiAgICByZXR1cm4geyAuLi50aGlzLmNvbmZpZyB9O1xuICB9XG59XG4iXX0=