@decaf-ts/for-pouch 0.2.14 → 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,731 +0,0 @@
1
- import 'reflect-metadata';
2
- import { CouchDBAdapter, generateIndexes, CouchDBKeys, IndexError } from '@decaf-ts/for-couchdb';
3
- import { InternalError, ConflictError, BaseError, NotFoundError, onCreate, onCreateUpdate } from '@decaf-ts/db-decorators';
4
- import { ConnectionError, Repository, PersistenceKeys, UnsupportedError } from '@decaf-ts/core';
5
- import { Decoration, propMetadata } from '@decaf-ts/decorator-validation';
6
- import PouchDB from 'pouchdb-core';
7
- import * as PouchMapReduce from 'pouchdb-mapreduce';
8
- import * as PouchReplication from 'pouchdb-replication';
9
- import * as PouchFind from 'pouchdb-find';
10
- import { Metadata } from '@decaf-ts/decoration';
11
-
12
- /**
13
- * @description Identifier for PouchDB flavor in the decorator system
14
- * @summary A string constant that identifies the PouchDB implementation in the decorator system.
15
- * This is used to target decorators specifically for PouchDB adapters.
16
- * @const PouchFlavour
17
- * @memberOf module:for-pouch
18
- */
19
- const PouchFlavour = "pouch";
20
- /**
21
- * @description Default relative path where local PouchDB databases are stored
22
- * @summary Used when creating a local PouchDB instance without a remote host; combined with dbName to form the filesystem path.
23
- * @const DefaultLocalStoragePath
24
- * @memberOf module:for-pouch
25
- */
26
- const DefaultLocalStoragePath = "local_dbs";
27
-
28
- /**
29
- * @description Sets the creator ID on a model during creation or update operations
30
- * @summary This function is used as a decorator handler to automatically set the creator ID field on a model
31
- * when it's being created or updated. It extracts the UUID from the context and assigns it to the specified key.
32
- * @template M - The model type that extends Model
33
- * @template R - The repository type that extends PouchRepository<M>
34
- * @template V - The relations metadata type that extends RelationsMetadata
35
- * @param {R} this - The repository instance
36
- * @param {Context<PouchFlags>} context - The operation context containing flags
37
- * @param {V} data - The relations metadata
38
- * @param key - The property key to set on the model
39
- * @param {M} model - The model instance to modify
40
- * @return {Promise<void>} A promise that resolves when the operation is complete
41
- * @function createdByOnPouchCreateUpdate
42
- * @memberOf module:for-pouch
43
- */
44
- async function createdByOnPouchCreateUpdate(context, data, key, model) {
45
- try {
46
- const uuid = context.get("UUID");
47
- model[key] = uuid;
48
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
49
- }
50
- catch (e) {
51
- throw new UnsupportedError("No User found in context. Please provide a user in the context");
52
- }
53
- }
54
- /**
55
- * @description PouchDB implementation of the CouchDBAdapter
56
- * @summary Concrete adapter that bridges the generic CouchDBAdapter to a PouchDB backend. It supports CRUD (single and bulk), indexing and Mango queries, and wires flavour-specific decorations.
57
- * @template PouchFlags - The flags specific to PouchDB operations
58
- * @template Context<PouchFlags> - The context type with PouchDB flags
59
- * @param {PouchConfig} config - Adapter configuration (remote credentials or local storage path, db name, plugins)
60
- * @param {string} [alias] - Optional alias for the database
61
- * @class PouchAdapter
62
- * @example
63
- * ```typescript
64
- * import { PouchAdapter } from '@decaf-ts/for-pouch';
65
- *
66
- * // Create a PouchAdapter with config
67
- * const adapter = new PouchAdapter({
68
- * protocol: 'http',
69
- * host: 'localhost:5984',
70
- * user: 'admin',
71
- * password: 'secret',
72
- * dbName: 'my-database',
73
- * plugins: []
74
- * });
75
- *
76
- * // Or use local storage
77
- * const localAdapter = new PouchAdapter({
78
- * protocol: 'http', // ignored for local
79
- * dbName: 'local-db',
80
- * storagePath: 'local_dbs',
81
- * plugins: []
82
- * });
83
- *
84
- * // Use the adapter for database operations
85
- * const result = await adapter.read('users', 'user-123');
86
- * ```
87
- * @mermaid
88
- * sequenceDiagram
89
- * participant Client
90
- * participant PouchAdapter
91
- * participant PouchDB
92
- * participant CouchDB
93
- *
94
- * Client->>PouchAdapter: new PouchAdapter(config, alias?)
95
- * PouchAdapter->>CouchDBAdapter: super(config, PouchFlavour, alias)
96
- *
97
- * Client->>PouchAdapter: create(table, id, model)
98
- * PouchAdapter->>PouchDB: put(model)
99
- * PouchDB->>CouchDB: HTTP PUT
100
- * CouchDB-->>PouchDB: Response
101
- * PouchDB-->>PouchAdapter: Response
102
- * PouchAdapter-->>Client: Updated model
103
- *
104
- * Client->>PouchAdapter: read(table, id)
105
- * PouchAdapter->>PouchDB: get(id)
106
- * PouchDB->>CouchDB: HTTP GET
107
- * CouchDB-->>PouchDB: Document
108
- * PouchDB-->>PouchAdapter: Document
109
- * PouchAdapter-->>Client: Model
110
- */
111
- class PouchAdapter extends CouchDBAdapter {
112
- constructor(config, alias) {
113
- super(config, PouchFlavour, alias);
114
- }
115
- /**
116
- * @description Lazily initializes and returns the underlying PouchDB client
117
- * @summary Loads required PouchDB plugins once, builds the connection URL or local storage path from config, and caches the Database instance for reuse. Throws InternalError if client creation fails.
118
- * @return {Database} A PouchDB Database instance ready to perform operations
119
- * @mermaid
120
- * sequenceDiagram
121
- * participant Caller
122
- * participant PouchAdapter
123
- * participant PouchDB
124
- * Caller->>PouchAdapter: getClient()
125
- * alt client not initialized
126
- * PouchAdapter->>PouchAdapter: register plugins
127
- * PouchAdapter->>PouchDB: new PouchDB(url or path)
128
- * alt creation fails
129
- * PouchDB-->>PouchAdapter: Error
130
- * PouchAdapter-->>Caller: throws InternalError
131
- * else success
132
- * PouchDB-->>PouchAdapter: Database
133
- * PouchAdapter-->>Caller: cached client
134
- * end
135
- * else client initialized
136
- * PouchAdapter-->>Caller: cached client
137
- * end
138
- */
139
- getClient() {
140
- if (!this._client) {
141
- const plugins = [
142
- PouchMapReduce,
143
- PouchReplication,
144
- PouchFind,
145
- ...this.config.plugins,
146
- ];
147
- for (const plugin of plugins) {
148
- try {
149
- PouchDB.plugin(plugin);
150
- }
151
- catch (e) {
152
- if (e instanceof Error && e.message.includes("redefine property"))
153
- continue; //plugin has already been loaded so it's ok
154
- throw e;
155
- }
156
- }
157
- const { host, protocol, user, password, dbName, storagePath } = this.config;
158
- try {
159
- if (host && user) {
160
- this._client = new PouchDB(`${protocol}://${user}:${password}@${host}/${dbName}`);
161
- }
162
- else
163
- this._client = new PouchDB(`${storagePath || DefaultLocalStoragePath}/${dbName}`);
164
- }
165
- catch (e) {
166
- throw new InternalError(`Failed to create PouchDB client: ${e}`);
167
- }
168
- }
169
- return this._client;
170
- }
171
- /**
172
- * @description Generates operation flags for PouchDB operations
173
- * @summary Creates a set of flags for a specific operation, including a UUID for identification.
174
- * This method extracts the user ID from the database URL or generates a random UUID if not available.
175
- * @template M - The model type that extends Model
176
- * @param {OperationKeys} operation - The operation key (create, read, update, delete)
177
- * @param {Constructor<M>} model - The model constructor
178
- * @param {Partial<PouchFlags>} flags - Partial flags to be merged
179
- * @return {Promise<PouchFlags>} The complete set of flags for the operation
180
- */
181
- async flags(operation, model, flags) {
182
- if (!this.config.user)
183
- this.config.user = crypto.randomUUID();
184
- return Object.assign(await super.flags(operation, model, flags), {
185
- UUID: this.config.user,
186
- });
187
- }
188
- /**
189
- * @description Creates database indexes for the given models
190
- * @summary Generates and creates indexes in the PouchDB database based on the provided model constructors.
191
- * This method uses the generateIndexes utility to create index definitions and then creates them in the database.
192
- * @template M - The model type that extends Model
193
- * @param models - The model constructors to create indexes for
194
- * @return {Promise<void>} A promise that resolves when all indexes are created
195
- */
196
- async index(...models) {
197
- const indexes = generateIndexes(models);
198
- for (const index of indexes) {
199
- const res = await this.client.createIndex(index);
200
- const { result } = res;
201
- if (result === "existing")
202
- throw new ConflictError(`Index ${index.name} already exists`);
203
- }
204
- }
205
- /**
206
- * @description Creates a new document in the database
207
- * @summary Inserts a new document into the PouchDB database using the put operation.
208
- * This method handles error parsing and ensures the operation was successful.
209
- * @param {string} tableName - The name of the table/collection
210
- * @param {string|number} id - The document ID
211
- * @param {Record<string, any>} model - The document data to insert
212
- * @return {Promise<Record<string, any>>} A promise that resolves to the created document with metadata
213
- * @mermaid
214
- * sequenceDiagram
215
- * participant Client
216
- * participant PouchAdapter
217
- * participant PouchDB
218
- *
219
- * Client->>PouchAdapter: create(tableName, id, model)
220
- * PouchAdapter->>PouchDB: put(model)
221
- * alt Success
222
- * PouchDB-->>PouchAdapter: Response with ok=true
223
- * PouchAdapter->>PouchAdapter: assignMetadata(model, response.rev)
224
- * PouchAdapter-->>Client: Updated model with metadata
225
- * else Error
226
- * PouchDB-->>PouchAdapter: Error
227
- * PouchAdapter->>PouchAdapter: parseError(e)
228
- * PouchAdapter-->>Client: Throws error
229
- * end
230
- */
231
- async create(tableName, id, model) {
232
- let response;
233
- try {
234
- response = await this.client.put(model);
235
- }
236
- catch (e) {
237
- throw this.parseError(e);
238
- }
239
- if (!response.ok)
240
- throw new InternalError(`Failed to insert doc id: ${id} in table ${tableName}`);
241
- return this.assignMetadata(model, response.rev);
242
- }
243
- /**
244
- * @description Creates multiple documents in the database in a single operation
245
- * @summary Inserts multiple documents into the PouchDB database using the bulkDocs operation.
246
- * This method handles error parsing and ensures all operations were successful.
247
- * @param {string} tableName - The name of the table/collection
248
- * @param {string[]|number[]} ids - The document IDs
249
- * @param models - The document data to insert
250
- * @return A promise that resolves to the created documents with metadata
251
- * @mermaid
252
- * sequenceDiagram
253
- * participant Client
254
- * participant PouchAdapter
255
- * participant PouchDB
256
- *
257
- * Client->>PouchAdapter: createAll(tableName, ids, models)
258
- * PouchAdapter->>PouchDB: bulkDocs(models)
259
- * alt Success
260
- * PouchDB-->>PouchAdapter: Array of responses with ok=true
261
- * PouchAdapter->>PouchAdapter: assignMultipleMetadata(models, revs)
262
- * PouchAdapter-->>Client: Updated models with metadata
263
- * else Error
264
- * PouchDB-->>PouchAdapter: Array with errors
265
- * PouchAdapter->>PouchAdapter: Check for errors
266
- * PouchAdapter-->>Client: Throws InternalError
267
- * end
268
- */
269
- async createAll(tableName, ids, models) {
270
- let response;
271
- try {
272
- response = await this.client.bulkDocs(models);
273
- }
274
- catch (e) {
275
- throw PouchAdapter.parseError(e);
276
- }
277
- if (!response.every((r) => r.ok)) {
278
- const errors = response.reduce((accum, el, i) => {
279
- if (el.error)
280
- accum.push(`el ${i}: ${el.error}${el.reason ? ` - ${el.reason}` : ""}`);
281
- return accum;
282
- }, []);
283
- throw new InternalError(errors.join("\n"));
284
- }
285
- return this.assignMultipleMetadata(models, response.map((r) => r.rev));
286
- }
287
- /**
288
- * @description Retrieves a document from the database by ID
289
- * @summary Fetches a document from the PouchDB database using the get operation.
290
- * This method generates the document ID based on the table name and ID, then retrieves the document.
291
- * @param {string} tableName - The name of the table/collection
292
- * @param {string|number} id - The document ID
293
- * @return {Promise<Record<string, any>>} A promise that resolves to the retrieved document with metadata
294
- * @mermaid
295
- * sequenceDiagram
296
- * participant Client
297
- * participant PouchAdapter
298
- * participant PouchDB
299
- *
300
- * Client->>PouchAdapter: read(tableName, id)
301
- * PouchAdapter->>PouchAdapter: generateId(tableName, id)
302
- * PouchAdapter->>PouchDB: get(_id)
303
- * alt Success
304
- * PouchDB-->>PouchAdapter: Document
305
- * PouchAdapter->>PouchAdapter: assignMetadata(record, record._rev)
306
- * PouchAdapter-->>Client: Document with metadata
307
- * else Error
308
- * PouchDB-->>PouchAdapter: Error
309
- * PouchAdapter->>PouchAdapter: parseError(e)
310
- * PouchAdapter-->>Client: Throws error
311
- * end
312
- */
313
- async read(tableName, id) {
314
- const _id = this.generateId(tableName, id);
315
- let record;
316
- try {
317
- record = await this.client.get(_id);
318
- }
319
- catch (e) {
320
- throw PouchAdapter.parseError(e);
321
- }
322
- return this.assignMetadata(record, record._rev);
323
- }
324
- /**
325
- * @description Retrieves multiple documents from the database by their IDs
326
- * @summary Fetches multiple documents from the PouchDB database using the bulkGet operation.
327
- * This method generates document IDs based on the table name and IDs, then retrieves the documents.
328
- * @param {string} tableName - The name of the table/collection
329
- * @param {Array<string|number|bigint>} ids - The document IDs
330
- * @return A promise that resolves to the retrieved documents with metadata
331
- * @mermaid
332
- * sequenceDiagram
333
- * participant Client
334
- * participant PouchAdapter
335
- * participant PouchDB
336
- *
337
- * Client->>PouchAdapter: readAll(tableName, ids)
338
- * PouchAdapter->>PouchAdapter: Map ids to generateId(tableName, id)
339
- * PouchAdapter->>PouchDB: bulkGet({docs})
340
- * alt Success
341
- * PouchDB-->>PouchAdapter: BulkGetResponse
342
- * PouchAdapter->>PouchAdapter: Process results
343
- * PouchAdapter->>PouchAdapter: assignMetadata for each doc
344
- * PouchAdapter-->>Client: Documents with metadata
345
- * else Error
346
- * PouchAdapter->>PouchAdapter: parseError(error)
347
- * PouchAdapter-->>Client: Throws error
348
- * end
349
- */
350
- async readAll(tableName, ids) {
351
- const results = await this.client.bulkGet({
352
- docs: ids.map((id) => ({ id: this.generateId(tableName, id) })),
353
- });
354
- const res = results.results.reduce((accum, r) => {
355
- r.docs.forEach((d) => {
356
- if (d.error || !d.ok)
357
- throw PouchAdapter.parseError(d.error ||
358
- new InternalError("Missing valid response"));
359
- const result = Object.assign({}, d.ok);
360
- accum.push(this.assignMetadata(result, d.ok[CouchDBKeys.REV]));
361
- });
362
- return accum;
363
- }, []);
364
- return res;
365
- }
366
- /**
367
- * @description Updates an existing document in the database
368
- * @summary Updates a document in the PouchDB database using the put operation.
369
- * This method handles error parsing and ensures the operation was successful.
370
- * @param {string} tableName - The name of the table/collection
371
- * @param {string|number} id - The document ID
372
- * @param {Record<string, any>} model - The updated document data
373
- * @return {Promise<Record<string, any>>} A promise that resolves to the updated document with metadata
374
- * @mermaid
375
- * sequenceDiagram
376
- * participant Client
377
- * participant PouchAdapter
378
- * participant PouchDB
379
- *
380
- * Client->>PouchAdapter: update(tableName, id, model)
381
- * PouchAdapter->>PouchDB: put(model)
382
- * alt Success
383
- * PouchDB-->>PouchAdapter: Response with ok=true
384
- * PouchAdapter->>PouchAdapter: assignMetadata(model, response.rev)
385
- * PouchAdapter-->>Client: Updated model with metadata
386
- * else Error
387
- * PouchDB-->>PouchAdapter: Error
388
- * PouchAdapter->>PouchAdapter: parseError(e)
389
- * PouchAdapter-->>Client: Throws error
390
- * end
391
- */
392
- async update(tableName, id, model) {
393
- let response;
394
- try {
395
- response = await this.client.put(model);
396
- }
397
- catch (e) {
398
- throw PouchAdapter.parseError(e);
399
- }
400
- if (!response.ok)
401
- throw new InternalError(`Failed to update doc id: ${id} in table ${tableName}`);
402
- return this.assignMetadata(model, response.rev);
403
- }
404
- /**
405
- * @description Updates multiple documents in the database in a single operation
406
- * @summary Updates multiple documents in the PouchDB database using the bulkDocs operation.
407
- * This method handles error parsing and ensures all operations were successful.
408
- * @param {string} tableName - The name of the table/collection
409
- * @param {string[]|number[]} ids - The document IDs
410
- * @param models - The updated document data
411
- * @return A promise that resolves to the updated documents with metadata
412
- * @mermaid
413
- * sequenceDiagram
414
- * participant Client
415
- * participant PouchAdapter
416
- * participant PouchDB
417
- *
418
- * Client->>PouchAdapter: updateAll(tableName, ids, models)
419
- * PouchAdapter->>PouchDB: bulkDocs(models)
420
- * alt Success
421
- * PouchDB-->>PouchAdapter: Array of responses with ok=true
422
- * PouchAdapter->>PouchAdapter: assignMultipleMetadata(models, revs)
423
- * PouchAdapter-->>Client: Updated models with metadata
424
- * else Error
425
- * PouchDB-->>PouchAdapter: Array with errors
426
- * PouchAdapter->>PouchAdapter: Check for errors
427
- * PouchAdapter-->>Client: Throws InternalError
428
- * end
429
- */
430
- async updateAll(tableName, ids, models) {
431
- let response;
432
- try {
433
- response = await this.client.bulkDocs(models);
434
- }
435
- catch (e) {
436
- throw PouchAdapter.parseError(e);
437
- }
438
- if (!response.every((r) => !r.error)) {
439
- const errors = response.reduce((accum, el, i) => {
440
- if (el.error)
441
- accum.push(`el ${i}: ${el.error}${el.reason ? ` - ${el.reason}` : ""}`);
442
- return accum;
443
- }, []);
444
- throw new InternalError(errors.join("\n"));
445
- }
446
- return this.assignMultipleMetadata(models, response.map((r) => r.rev));
447
- }
448
- /**
449
- * @description Deletes a document from the database by ID
450
- * @summary Removes a document from the PouchDB database using the remove operation.
451
- * This method first retrieves the document to get its revision, then deletes it.
452
- * @param {string} tableName - The name of the table/collection
453
- * @param {string|number} id - The document ID
454
- * @return {Promise<Record<string, any>>} A promise that resolves to the deleted document with metadata
455
- * @mermaid
456
- * sequenceDiagram
457
- * participant Client
458
- * participant PouchAdapter
459
- * participant PouchDB
460
- *
461
- * Client->>PouchAdapter: delete(tableName, id)
462
- * PouchAdapter->>PouchAdapter: generateId(tableName, id)
463
- * PouchAdapter->>PouchDB: get(_id)
464
- * PouchDB-->>PouchAdapter: Document with _rev
465
- * PouchAdapter->>PouchDB: remove(_id, record._rev)
466
- * alt Success
467
- * PouchDB-->>PouchAdapter: Success response
468
- * PouchAdapter->>PouchAdapter: assignMetadata(record, record._rev)
469
- * PouchAdapter-->>Client: Deleted document with metadata
470
- * else Error
471
- * PouchDB-->>PouchAdapter: Error
472
- * PouchAdapter->>PouchAdapter: parseError(e)
473
- * PouchAdapter-->>Client: Throws error
474
- * end
475
- */
476
- async delete(tableName, id) {
477
- const _id = this.generateId(tableName, id);
478
- let record;
479
- try {
480
- record = await this.client.get(_id);
481
- await this.client.remove(_id, record._rev);
482
- }
483
- catch (e) {
484
- throw PouchAdapter.parseError(e);
485
- }
486
- return this.assignMetadata(record, record._rev);
487
- }
488
- /**
489
- * @description Deletes multiple documents from the database by their IDs
490
- * @summary Removes multiple documents from the PouchDB database in a single operation.
491
- * This method first retrieves all documents to get their revisions, then marks them as deleted.
492
- * @param {string} tableName - The name of the table/collection
493
- * @param {Array<string|number|bigint>} ids - The document IDs
494
- * @return A promise that resolves to the deleted documents with metadata
495
- * @mermaid
496
- * sequenceDiagram
497
- * participant Client
498
- * participant PouchAdapter
499
- * participant PouchDB
500
- *
501
- * Client->>PouchAdapter: deleteAll(tableName, ids)
502
- * PouchAdapter->>PouchAdapter: Map ids to generateId(tableName, id)
503
- * PouchAdapter->>PouchDB: bulkGet({docs})
504
- * PouchDB-->>PouchAdapter: BulkGetResponse with documents
505
- * PouchAdapter->>PouchAdapter: Mark documents as deleted
506
- * PouchAdapter->>PouchDB: bulkDocs(marked documents)
507
- * alt Success
508
- * PouchDB-->>PouchAdapter: Success responses
509
- * PouchAdapter->>PouchAdapter: Process results
510
- * PouchAdapter->>PouchAdapter: assignMetadata for each doc
511
- * PouchAdapter-->>Client: Deleted documents with metadata
512
- * else Error
513
- * PouchAdapter->>PouchAdapter: Check for errors
514
- * PouchAdapter-->>Client: Throws InternalError
515
- * end
516
- */
517
- async deleteAll(tableName, ids) {
518
- const results = await this.client.bulkGet({
519
- docs: ids.map((id) => ({ id: this.generateId(tableName, id) })),
520
- });
521
- const deletion = await this.client.bulkDocs(results.results.map((r) => {
522
- r[CouchDBKeys.DELETED] = true;
523
- return r;
524
- }));
525
- const errs = deletion.filter((d) => d.error);
526
- if (errs.length)
527
- throw new InternalError(errs.join("\n"));
528
- return results.results.reduce((accum, r) => {
529
- r.docs.forEach((d) => {
530
- const result = Object.assign({}, d.ok);
531
- accum.push(this.assignMetadata(result, d.ok[CouchDBKeys.REV]));
532
- });
533
- return accum;
534
- }, []);
535
- }
536
- /**
537
- * @description Executes a raw Mango query against the database
538
- * @summary Performs a direct find operation using a Mango query object.
539
- * This method allows for complex queries beyond the standard CRUD operations.
540
- * @template V - The return type
541
- * @param {MangoQuery} rawInput - The Mango query to execute
542
- * @param {boolean} [process=true] - Whether to process the response (true returns just docs, false returns full response)
543
- * @return {Promise<V>} A promise that resolves to the query results
544
- * @mermaid
545
- * sequenceDiagram
546
- * participant Client
547
- * participant PouchAdapter
548
- * participant PouchDB
549
- *
550
- * Client->>PouchAdapter: raw<V>(rawInput, process)
551
- * PouchAdapter->>PouchDB: find(rawInput)
552
- * alt Success
553
- * PouchDB-->>PouchAdapter: FindResponse
554
- * alt process=true
555
- * PouchAdapter-->>Client: response.docs as V
556
- * else process=false
557
- * PouchAdapter-->>Client: response as V
558
- * end
559
- * else Error
560
- * PouchDB-->>PouchAdapter: Error
561
- * PouchAdapter->>PouchAdapter: parseError(e)
562
- * PouchAdapter-->>Client: Throws error
563
- * end
564
- */
565
- async raw(rawInput, process = true) {
566
- try {
567
- const response = await this.client.find(rawInput);
568
- if (response.warning)
569
- console.warn(response.warning);
570
- if (process)
571
- return response.docs;
572
- return response;
573
- }
574
- catch (e) {
575
- throw PouchAdapter.parseError(e);
576
- }
577
- }
578
- /**
579
- * @description Parses and converts errors from PouchDB to application-specific errors
580
- * @summary Converts PouchDB errors to the application's error hierarchy.
581
- * This instance method delegates to the static parseError method.
582
- * @param {Error|string} err - The error object or message to parse
583
- * @param {string} [reason] - Optional reason for the error
584
- * @return {BaseError} The converted error object
585
- */
586
- parseError(err, reason) {
587
- return PouchAdapter.parseError(err, reason);
588
- }
589
- /**
590
- * @description Static method to parse and convert errors from PouchDB to application-specific errors
591
- * @summary Converts PouchDB errors to the application's error hierarchy based on error codes and messages.
592
- * This method analyzes the error type, status code, or message to determine the appropriate error class.
593
- * @param {Error|string} err - The error object or message to parse
594
- * @param {string} [reason] - Optional reason for the error
595
- * @return {BaseError} The converted error object
596
- * @mermaid
597
- * sequenceDiagram
598
- * participant Caller
599
- * participant PouchAdapter
600
- *
601
- * Caller->>PouchAdapter: parseError(err, reason)
602
- * alt err is BaseError
603
- * PouchAdapter-->>Caller: Return err as is
604
- * else err is string
605
- * alt contains "already exist" or "update conflict"
606
- * PouchAdapter-->>Caller: ConflictError
607
- * else contains "missing" or "deleted"
608
- * PouchAdapter-->>Caller: NotFoundError
609
- * end
610
- * else err has status
611
- * alt status is 401, 412, 409
612
- * PouchAdapter-->>Caller: ConflictError
613
- * else status is 404
614
- * PouchAdapter-->>Caller: NotFoundError
615
- * else status is 400
616
- * alt message contains "No index exists"
617
- * PouchAdapter-->>Caller: IndexError
618
- * else
619
- * PouchAdapter-->>Caller: InternalError
620
- * end
621
- * else message contains "ECONNREFUSED"
622
- * PouchAdapter-->>Caller: ConnectionError
623
- * else
624
- * PouchAdapter-->>Caller: InternalError
625
- * end
626
- * end
627
- */
628
- static parseError(err, reason) {
629
- // return super.parseError(err, reason);
630
- if (err instanceof BaseError)
631
- return err;
632
- let code = "";
633
- if (typeof err === "string") {
634
- code = err;
635
- if (code.match(/already exist|update conflict/g))
636
- return new ConflictError(code);
637
- if (code.match(/missing|deleted/g))
638
- return new NotFoundError(code);
639
- }
640
- else if (err.status) {
641
- code = err.status;
642
- reason = reason || err.message;
643
- }
644
- else {
645
- code = err.message;
646
- }
647
- switch (code.toString()) {
648
- case "401":
649
- case "412":
650
- case "409":
651
- return new ConflictError(reason);
652
- case "404":
653
- return new NotFoundError(reason);
654
- case "400":
655
- if (code.toString().match(/No\sindex\sexists/g))
656
- return new IndexError(err);
657
- return new InternalError(err);
658
- default:
659
- if (code.toString().match(/ECONNREFUSED/g))
660
- return new ConnectionError(err);
661
- return new InternalError(err);
662
- }
663
- }
664
- /**
665
- * @description Sets up decorations for PouchDB-specific model properties
666
- * @summary Configures decorators for createdBy and updatedBy fields in models.
667
- * This method defines how these fields should be automatically populated during create and update operations.
668
- * @mermaid
669
- * sequenceDiagram
670
- * participant Caller
671
- * participant PouchAdapter
672
- * participant Decoration
673
- *
674
- * Caller->>PouchAdapter: decoration()
675
- * PouchAdapter->>Repository: key(PersistenceKeys.CREATED_BY)
676
- * Repository-->>PouchAdapter: createdByKey
677
- * PouchAdapter->>Repository: key(PersistenceKeys.UPDATED_BY)
678
- * Repository-->>PouchAdapter: updatedByKey
679
- *
680
- * PouchAdapter->>Decoration: flavouredAs(PouchFlavour)
681
- * Decoration-->>PouchAdapter: DecoratorBuilder
682
- * PouchAdapter->>Decoration: for(createdByKey)
683
- * PouchAdapter->>Decoration: define(onCreate, propMetadata)
684
- * PouchAdapter->>Decoration: apply()
685
- *
686
- * PouchAdapter->>Decoration: flavouredAs(PouchFlavour)
687
- * Decoration-->>PouchAdapter: DecoratorBuilder
688
- * PouchAdapter->>Decoration: for(updatedByKey)
689
- * PouchAdapter->>Decoration: define(onCreate, propMetadata)
690
- * PouchAdapter->>Decoration: apply()
691
- */
692
- static decoration() {
693
- super.decoration();
694
- const createdByKey = Repository.key(PersistenceKeys.CREATED_BY);
695
- const updatedByKey = Repository.key(PersistenceKeys.UPDATED_BY);
696
- Decoration.flavouredAs(PouchFlavour)
697
- .for(createdByKey)
698
- .define(onCreate(createdByOnPouchCreateUpdate), propMetadata(createdByKey, {}))
699
- .apply();
700
- Decoration.flavouredAs(PouchFlavour)
701
- .for(updatedByKey)
702
- .define(onCreateUpdate(createdByOnPouchCreateUpdate), propMetadata(updatedByKey, {}))
703
- .apply();
704
- }
705
- }
706
- PouchAdapter.setCurrent(PouchFlavour);
707
-
708
- PouchAdapter.decoration();
709
- /**
710
- * @description A TypeScript adapter for PouchDB integration
711
- * @summary Provides a repository-pattern implementation backed by PouchDB, exposing the {@link PouchAdapter} to interface with databases, the {@link PouchRepository} for typed data access, configuration {@link module:for-pouch|constants} like {@link PouchFlavour} and {@link DefaultLocalStoragePath}, and related {@link module:for-pouch|types}. This module wires up decorators on load to support created/updated-by fields.
712
- * @module for-pouch
713
- */
714
- /**
715
- * @description Package version identifier
716
- * @summary Stores the current version of the for-pouch package
717
- * @const VERSION
718
- * @memberOf module:for-pouch
719
- */
720
- const VERSION = "0.2.13";
721
- /**
722
- * @description Package version identifier
723
- * @summary Stores the current version of the for-pouch package
724
- * @const VERSION
725
- * @memberOf module:for-pouch
726
- */
727
- const PACKAGE_NAME = "@decaf-ts/for-pouch";
728
- Metadata.registerLibrary(PACKAGE_NAME, VERSION);
729
-
730
- export { DefaultLocalStoragePath, PACKAGE_NAME, PouchAdapter, PouchFlavour, VERSION, createdByOnPouchCreateUpdate };
731
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9yLXBvdWNoLmVzbS5janMiLCJzb3VyY2VzIjpbIi4uL3NyYy9jb25zdGFudHMudHMiLCIuLi9zcmMvYWRhcHRlci50cyIsIi4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBkZXNjcmlwdGlvbiBJZGVudGlmaWVyIGZvciBQb3VjaERCIGZsYXZvciBpbiB0aGUgZGVjb3JhdG9yIHN5c3RlbVxuICogQHN1bW1hcnkgQSBzdHJpbmcgY29uc3RhbnQgdGhhdCBpZGVudGlmaWVzIHRoZSBQb3VjaERCIGltcGxlbWVudGF0aW9uIGluIHRoZSBkZWNvcmF0b3Igc3lzdGVtLlxuICogVGhpcyBpcyB1c2VkIHRvIHRhcmdldCBkZWNvcmF0b3JzIHNwZWNpZmljYWxseSBmb3IgUG91Y2hEQiBhZGFwdGVycy5cbiAqIEBjb25zdCBQb3VjaEZsYXZvdXJcbiAqIEBtZW1iZXJPZiBtb2R1bGU6Zm9yLXBvdWNoXG4gKi9cbmV4cG9ydCBjb25zdCBQb3VjaEZsYXZvdXIgPSBcInBvdWNoXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIERlZmF1bHQgcmVsYXRpdmUgcGF0aCB3aGVyZSBsb2NhbCBQb3VjaERCIGRhdGFiYXNlcyBhcmUgc3RvcmVkXG4gKiBAc3VtbWFyeSBVc2VkIHdoZW4gY3JlYXRpbmcgYSBsb2NhbCBQb3VjaERCIGluc3RhbmNlIHdpdGhvdXQgYSByZW1vdGUgaG9zdDsgY29tYmluZWQgd2l0aCBkYk5hbWUgdG8gZm9ybSB0aGUgZmlsZXN5c3RlbSBwYXRoLlxuICogQGNvbnN0IERlZmF1bHRMb2NhbFN0b3JhZ2VQYXRoXG4gKiBAbWVtYmVyT2YgbW9kdWxlOmZvci1wb3VjaFxuICovXG5leHBvcnQgY29uc3QgRGVmYXVsdExvY2FsU3RvcmFnZVBhdGggPSBcImxvY2FsX2Ric1wiO1xuIiwiaW1wb3J0IFwicmVmbGVjdC1tZXRhZGF0YVwiO1xuaW1wb3J0IHtcbiAgQ291Y2hEQkFkYXB0ZXIsXG4gIENvdWNoREJLZXlzLFxuICBDcmVhdGVJbmRleFJlcXVlc3QsXG4gIGdlbmVyYXRlSW5kZXhlcyxcbiAgSW5kZXhFcnJvcixcbiAgTWFuZ29RdWVyeSxcbn0gZnJvbSBcIkBkZWNhZi10cy9mb3ItY291Y2hkYlwiO1xuaW1wb3J0IHtcbiAgQmFzZUVycm9yLFxuICBDb25mbGljdEVycm9yLFxuICBDb250ZXh0LFxuICBJbnRlcm5hbEVycm9yLFxuICBOb3RGb3VuZEVycm9yLFxuICBvbkNyZWF0ZSxcbiAgb25DcmVhdGVVcGRhdGUsXG4gIE9wZXJhdGlvbktleXMsXG59IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHtcbiAgQ29ubmVjdGlvbkVycm9yLFxuICBQZXJzaXN0ZW5jZUtleXMsXG4gIFJlbGF0aW9uc01ldGFkYXRhLFxuICBSZXBvc2l0b3J5LFxuICBVbnN1cHBvcnRlZEVycm9yLFxufSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCBEYXRhYmFzZSA9IFBvdWNoREIuRGF0YWJhc2U7XG5pbXBvcnQgUmVzcG9uc2UgPSBQb3VjaERCLkNvcmUuUmVzcG9uc2U7XG5pbXBvcnQgRXJyID0gUG91Y2hEQi5Db3JlLkVycm9yO1xuaW1wb3J0IElkTWV0YSA9IFBvdWNoREIuQ29yZS5JZE1ldGE7XG5pbXBvcnQgR2V0TWV0YSA9IFBvdWNoREIuQ29yZS5HZXRNZXRhO1xuaW1wb3J0IENyZWF0ZUluZGV4UmVzcG9uc2UgPSBQb3VjaERCLkZpbmQuQ3JlYXRlSW5kZXhSZXNwb25zZTtcbmltcG9ydCB7XG4gIENvbnN0cnVjdG9yLFxuICBEZWNvcmF0aW9uLFxuICBNb2RlbCxcbiAgcHJvcE1ldGFkYXRhLFxufSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgQnVsa0dldFJlc3BvbnNlID0gUG91Y2hEQi5Db3JlLkJ1bGtHZXRSZXNwb25zZTtcbmltcG9ydCBGaW5kUmVzcG9uc2UgPSBQb3VjaERCLkZpbmQuRmluZFJlc3BvbnNlO1xuaW1wb3J0IHsgUG91Y2hDb25maWcsIFBvdWNoRmxhZ3MgfSBmcm9tIFwiLi90eXBlc1wiO1xuaW1wb3J0IHsgRGVmYXVsdExvY2FsU3RvcmFnZVBhdGgsIFBvdWNoRmxhdm91ciB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgUG91Y2hSZXBvc2l0b3J5IH0gZnJvbSBcIi4vUG91Y2hSZXBvc2l0b3J5XCI7XG5pbXBvcnQgUG91Y2hEQiBmcm9tIFwicG91Y2hkYi1jb3JlXCI7XG5pbXBvcnQgKiBhcyBQb3VjaE1hcFJlZHVjZSBmcm9tIFwicG91Y2hkYi1tYXByZWR1Y2VcIjtcbmltcG9ydCAqIGFzIFBvdWNoUmVwbGljYXRpb24gZnJvbSBcInBvdWNoZGItcmVwbGljYXRpb25cIjtcbmltcG9ydCAqIGFzIFBvdWNoRmluZCBmcm9tIFwicG91Y2hkYi1maW5kXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFNldHMgdGhlIGNyZWF0b3IgSUQgb24gYSBtb2RlbCBkdXJpbmcgY3JlYXRpb24gb3IgdXBkYXRlIG9wZXJhdGlvbnNcbiAqIEBzdW1tYXJ5IFRoaXMgZnVuY3Rpb24gaXMgdXNlZCBhcyBhIGRlY29yYXRvciBoYW5kbGVyIHRvIGF1dG9tYXRpY2FsbHkgc2V0IHRoZSBjcmVhdG9yIElEIGZpZWxkIG9uIGEgbW9kZWxcbiAqIHdoZW4gaXQncyBiZWluZyBjcmVhdGVkIG9yIHVwZGF0ZWQuIEl0IGV4dHJhY3RzIHRoZSBVVUlEIGZyb20gdGhlIGNvbnRleHQgYW5kIGFzc2lnbnMgaXQgdG8gdGhlIHNwZWNpZmllZCBrZXkuXG4gKiBAdGVtcGxhdGUgTSAtIFRoZSBtb2RlbCB0eXBlIHRoYXQgZXh0ZW5kcyBNb2RlbFxuICogQHRlbXBsYXRlIFIgLSBUaGUgcmVwb3NpdG9yeSB0eXBlIHRoYXQgZXh0ZW5kcyBQb3VjaFJlcG9zaXRvcnk8TT5cbiAqIEB0ZW1wbGF0ZSBWIC0gVGhlIHJlbGF0aW9ucyBtZXRhZGF0YSB0eXBlIHRoYXQgZXh0ZW5kcyBSZWxhdGlvbnNNZXRhZGF0YVxuICogQHBhcmFtIHtSfSB0aGlzIC0gVGhlIHJlcG9zaXRvcnkgaW5zdGFuY2VcbiAqIEBwYXJhbSB7Q29udGV4dDxQb3VjaEZsYWdzPn0gY29udGV4dCAtIFRoZSBvcGVyYXRpb24gY29udGV4dCBjb250YWluaW5nIGZsYWdzXG4gKiBAcGFyYW0ge1Z9IGRhdGEgLSBUaGUgcmVsYXRpb25zIG1ldGFkYXRhXG4gKiBAcGFyYW0ga2V5IC0gVGhlIHByb3BlcnR5IGtleSB0byBzZXQgb24gdGhlIG1vZGVsXG4gKiBAcGFyYW0ge019IG1vZGVsIC0gVGhlIG1vZGVsIGluc3RhbmNlIHRvIG1vZGlmeVxuICogQHJldHVybiB7UHJvbWlzZTx2b2lkPn0gQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgd2hlbiB0aGUgb3BlcmF0aW9uIGlzIGNvbXBsZXRlXG4gKiBAZnVuY3Rpb24gY3JlYXRlZEJ5T25Qb3VjaENyZWF0ZVVwZGF0ZVxuICogQG1lbWJlck9mIG1vZHVsZTpmb3ItcG91Y2hcbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNyZWF0ZWRCeU9uUG91Y2hDcmVhdGVVcGRhdGU8XG4gIE0gZXh0ZW5kcyBNb2RlbCxcbiAgUiBleHRlbmRzIFBvdWNoUmVwb3NpdG9yeTxNPixcbiAgViBleHRlbmRzIFJlbGF0aW9uc01ldGFkYXRhLFxuPihcbiAgdGhpczogUixcbiAgY29udGV4dDogQ29udGV4dDxQb3VjaEZsYWdzPixcbiAgZGF0YTogVixcbiAga2V5OiBrZXlvZiBNLFxuICBtb2RlbDogTVxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIHRyeSB7XG4gICAgY29uc3QgdXVpZDogc3RyaW5nID0gY29udGV4dC5nZXQoXCJVVUlEXCIpO1xuICAgIG1vZGVsW2tleV0gPSB1dWlkIGFzIE1ba2V5b2YgTV07XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkRXJyb3IoXG4gICAgICBcIk5vIFVzZXIgZm91bmQgaW4gY29udGV4dC4gUGxlYXNlIHByb3ZpZGUgYSB1c2VyIGluIHRoZSBjb250ZXh0XCJcbiAgICApO1xuICB9XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFBvdWNoREIgaW1wbGVtZW50YXRpb24gb2YgdGhlIENvdWNoREJBZGFwdGVyXG4gKiBAc3VtbWFyeSBDb25jcmV0ZSBhZGFwdGVyIHRoYXQgYnJpZGdlcyB0aGUgZ2VuZXJpYyBDb3VjaERCQWRhcHRlciB0byBhIFBvdWNoREIgYmFja2VuZC4gSXQgc3VwcG9ydHMgQ1JVRCAoc2luZ2xlIGFuZCBidWxrKSwgaW5kZXhpbmcgYW5kIE1hbmdvIHF1ZXJpZXMsIGFuZCB3aXJlcyBmbGF2b3VyLXNwZWNpZmljIGRlY29yYXRpb25zLlxuICogQHRlbXBsYXRlIFBvdWNoRmxhZ3MgLSBUaGUgZmxhZ3Mgc3BlY2lmaWMgdG8gUG91Y2hEQiBvcGVyYXRpb25zXG4gKiBAdGVtcGxhdGUgQ29udGV4dDxQb3VjaEZsYWdzPiAtIFRoZSBjb250ZXh0IHR5cGUgd2l0aCBQb3VjaERCIGZsYWdzXG4gKiBAcGFyYW0ge1BvdWNoQ29uZmlnfSBjb25maWcgLSBBZGFwdGVyIGNvbmZpZ3VyYXRpb24gKHJlbW90ZSBjcmVkZW50aWFscyBvciBsb2NhbCBzdG9yYWdlIHBhdGgsIGRiIG5hbWUsIHBsdWdpbnMpXG4gKiBAcGFyYW0ge3N0cmluZ30gW2FsaWFzXSAtIE9wdGlvbmFsIGFsaWFzIGZvciB0aGUgZGF0YWJhc2VcbiAqIEBjbGFzcyBQb3VjaEFkYXB0ZXJcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBpbXBvcnQgeyBQb3VjaEFkYXB0ZXIgfSBmcm9tICdAZGVjYWYtdHMvZm9yLXBvdWNoJztcbiAqXG4gKiAvLyBDcmVhdGUgYSBQb3VjaEFkYXB0ZXIgd2l0aCBjb25maWdcbiAqIGNvbnN0IGFkYXB0ZXIgPSBuZXcgUG91Y2hBZGFwdGVyKHtcbiAqICAgcHJvdG9jb2w6ICdodHRwJyxcbiAqICAgaG9zdDogJ2xvY2FsaG9zdDo1OTg0JyxcbiAqICAgdXNlcjogJ2FkbWluJyxcbiAqICAgcGFzc3dvcmQ6ICdzZWNyZXQnLFxuICogICBkYk5hbWU6ICdteS1kYXRhYmFzZScsXG4gKiAgIHBsdWdpbnM6IFtdXG4gKiB9KTtcbiAqXG4gKiAvLyBPciB1c2UgbG9jYWwgc3RvcmFnZVxuICogY29uc3QgbG9jYWxBZGFwdGVyID0gbmV3IFBvdWNoQWRhcHRlcih7XG4gKiAgIHByb3RvY29sOiAnaHR0cCcsIC8vIGlnbm9yZWQgZm9yIGxvY2FsXG4gKiAgIGRiTmFtZTogJ2xvY2FsLWRiJyxcbiAqICAgc3RvcmFnZVBhdGg6ICdsb2NhbF9kYnMnLFxuICogICBwbHVnaW5zOiBbXVxuICogfSk7XG4gKlxuICogLy8gVXNlIHRoZSBhZGFwdGVyIGZvciBkYXRhYmFzZSBvcGVyYXRpb25zXG4gKiBjb25zdCByZXN1bHQgPSBhd2FpdCBhZGFwdGVyLnJlYWQoJ3VzZXJzJywgJ3VzZXItMTIzJyk7XG4gKiBgYGBcbiAqIEBtZXJtYWlkXG4gKiBzZXF1ZW5jZURpYWdyYW1cbiAqICAgcGFydGljaXBhbnQgQ2xpZW50XG4gKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gKiAgIHBhcnRpY2lwYW50IENvdWNoREJcbiAqXG4gKiAgIENsaWVudC0+PlBvdWNoQWRhcHRlcjogbmV3IFBvdWNoQWRhcHRlcihjb25maWcsIGFsaWFzPylcbiAqICAgUG91Y2hBZGFwdGVyLT4+Q291Y2hEQkFkYXB0ZXI6IHN1cGVyKGNvbmZpZywgUG91Y2hGbGF2b3VyLCBhbGlhcylcbiAqXG4gKiAgIENsaWVudC0+PlBvdWNoQWRhcHRlcjogY3JlYXRlKHRhYmxlLCBpZCwgbW9kZWwpXG4gKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IHB1dChtb2RlbClcbiAqICAgUG91Y2hEQi0+PkNvdWNoREI6IEhUVFAgUFVUXG4gKiAgIENvdWNoREItLT4+UG91Y2hEQjogUmVzcG9uc2VcbiAqICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IFJlc3BvbnNlXG4gKiAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IFVwZGF0ZWQgbW9kZWxcbiAqXG4gKiAgIENsaWVudC0+PlBvdWNoQWRhcHRlcjogcmVhZCh0YWJsZSwgaWQpXG4gKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IGdldChpZClcbiAqICAgUG91Y2hEQi0+PkNvdWNoREI6IEhUVFAgR0VUXG4gKiAgIENvdWNoREItLT4+UG91Y2hEQjogRG9jdW1lbnRcbiAqICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IERvY3VtZW50XG4gKiAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IE1vZGVsXG4gKi9cbmV4cG9ydCBjbGFzcyBQb3VjaEFkYXB0ZXIgZXh0ZW5kcyBDb3VjaERCQWRhcHRlcjxcbiAgUG91Y2hDb25maWcsXG4gIERhdGFiYXNlLFxuICBQb3VjaEZsYWdzLFxuICBDb250ZXh0PFBvdWNoRmxhZ3M+XG4+IHtcbiAgY29uc3RydWN0b3IoY29uZmlnOiBQb3VjaENvbmZpZywgYWxpYXM/OiBzdHJpbmcpIHtcbiAgICBzdXBlcihjb25maWcsIFBvdWNoRmxhdm91ciwgYWxpYXMpO1xuICB9XG5cbi8qKlxuICAgKiBAZGVzY3JpcHRpb24gTGF6aWx5IGluaXRpYWxpemVzIGFuZCByZXR1cm5zIHRoZSB1bmRlcmx5aW5nIFBvdWNoREIgY2xpZW50XG4gICAqIEBzdW1tYXJ5IExvYWRzIHJlcXVpcmVkIFBvdWNoREIgcGx1Z2lucyBvbmNlLCBidWlsZHMgdGhlIGNvbm5lY3Rpb24gVVJMIG9yIGxvY2FsIHN0b3JhZ2UgcGF0aCBmcm9tIGNvbmZpZywgYW5kIGNhY2hlcyB0aGUgRGF0YWJhc2UgaW5zdGFuY2UgZm9yIHJldXNlLiBUaHJvd3MgSW50ZXJuYWxFcnJvciBpZiBjbGllbnQgY3JlYXRpb24gZmFpbHMuXG4gICAqIEByZXR1cm4ge0RhdGFiYXNlfSBBIFBvdWNoREIgRGF0YWJhc2UgaW5zdGFuY2UgcmVhZHkgdG8gcGVyZm9ybSBvcGVyYXRpb25zXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENhbGxlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoREJcbiAgICogICBDYWxsZXItPj5Qb3VjaEFkYXB0ZXI6IGdldENsaWVudCgpXG4gICAqICAgYWx0IGNsaWVudCBub3QgaW5pdGlhbGl6ZWRcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogcmVnaXN0ZXIgcGx1Z2luc1xuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogbmV3IFBvdWNoREIodXJsIG9yIHBhdGgpXG4gICAqICAgICBhbHQgY3JlYXRpb24gZmFpbHNcbiAgICogICAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEVycm9yXG4gICAqICAgICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IHRocm93cyBJbnRlcm5hbEVycm9yXG4gICAqICAgICBlbHNlIHN1Y2Nlc3NcbiAgICogICAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IERhdGFiYXNlXG4gICAqICAgICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IGNhY2hlZCBjbGllbnRcbiAgICogICAgIGVuZFxuICAgKiAgIGVsc2UgY2xpZW50IGluaXRpYWxpemVkXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2FsbGVyOiBjYWNoZWQgY2xpZW50XG4gICAqICAgZW5kXG4gICAqL1xuICBvdmVycmlkZSBnZXRDbGllbnQoKTogRGF0YWJhc2Uge1xuICAgIGlmICghdGhpcy5fY2xpZW50KSB7XG4gICAgICBjb25zdCBwbHVnaW5zID0gW1xuICAgICAgICBQb3VjaE1hcFJlZHVjZSxcbiAgICAgICAgUG91Y2hSZXBsaWNhdGlvbixcbiAgICAgICAgUG91Y2hGaW5kLFxuICAgICAgICAuLi50aGlzLmNvbmZpZy5wbHVnaW5zLFxuICAgICAgXTtcbiAgICAgIGZvciAoY29uc3QgcGx1Z2luIG9mIHBsdWdpbnMpIHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBQb3VjaERCLnBsdWdpbihwbHVnaW4pO1xuICAgICAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgICAgICBpZiAoZSBpbnN0YW5jZW9mIEVycm9yICYmIGUubWVzc2FnZS5pbmNsdWRlcyhcInJlZGVmaW5lIHByb3BlcnR5XCIpKVxuICAgICAgICAgICAgY29udGludWU7IC8vcGx1Z2luIGhhcyBhbHJlYWR5IGJlZW4gbG9hZGVkIHNvIGl0J3Mgb2tcbiAgICAgICAgICB0aHJvdyBlO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHsgaG9zdCwgcHJvdG9jb2wsIHVzZXIsIHBhc3N3b3JkLCBkYk5hbWUsIHN0b3JhZ2VQYXRoIH0gPVxuICAgICAgICB0aGlzLmNvbmZpZztcblxuICAgICAgdHJ5IHtcbiAgICAgICAgaWYgKGhvc3QgJiYgdXNlcikge1xuICAgICAgICAgIHRoaXMuX2NsaWVudCA9IG5ldyBQb3VjaERCKFxuICAgICAgICAgICAgYCR7cHJvdG9jb2x9Oi8vJHt1c2VyfToke3Bhc3N3b3JkfUAke2hvc3R9LyR7ZGJOYW1lfWBcbiAgICAgICAgICApO1xuICAgICAgICB9IGVsc2VcbiAgICAgICAgICB0aGlzLl9jbGllbnQgPSBuZXcgUG91Y2hEQihcbiAgICAgICAgICAgIGAke3N0b3JhZ2VQYXRoIHx8IERlZmF1bHRMb2NhbFN0b3JhZ2VQYXRofS8ke2RiTmFtZX1gXG4gICAgICAgICAgKTtcbiAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoYEZhaWxlZCB0byBjcmVhdGUgUG91Y2hEQiBjbGllbnQ6ICR7ZX1gKTtcbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuX2NsaWVudCBhcyBEYXRhYmFzZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gR2VuZXJhdGVzIG9wZXJhdGlvbiBmbGFncyBmb3IgUG91Y2hEQiBvcGVyYXRpb25zXG4gICAqIEBzdW1tYXJ5IENyZWF0ZXMgYSBzZXQgb2YgZmxhZ3MgZm9yIGEgc3BlY2lmaWMgb3BlcmF0aW9uLCBpbmNsdWRpbmcgYSBVVUlEIGZvciBpZGVudGlmaWNhdGlvbi5cbiAgICogVGhpcyBtZXRob2QgZXh0cmFjdHMgdGhlIHVzZXIgSUQgZnJvbSB0aGUgZGF0YWJhc2UgVVJMIG9yIGdlbmVyYXRlcyBhIHJhbmRvbSBVVUlEIGlmIG5vdCBhdmFpbGFibGUuXG4gICAqIEB0ZW1wbGF0ZSBNIC0gVGhlIG1vZGVsIHR5cGUgdGhhdCBleHRlbmRzIE1vZGVsXG4gICAqIEBwYXJhbSB7T3BlcmF0aW9uS2V5c30gb3BlcmF0aW9uIC0gVGhlIG9wZXJhdGlvbiBrZXkgKGNyZWF0ZSwgcmVhZCwgdXBkYXRlLCBkZWxldGUpXG4gICAqIEBwYXJhbSB7Q29uc3RydWN0b3I8TT59IG1vZGVsIC0gVGhlIG1vZGVsIGNvbnN0cnVjdG9yXG4gICAqIEBwYXJhbSB7UGFydGlhbDxQb3VjaEZsYWdzPn0gZmxhZ3MgLSBQYXJ0aWFsIGZsYWdzIHRvIGJlIG1lcmdlZFxuICAgKiBAcmV0dXJuIHtQcm9taXNlPFBvdWNoRmxhZ3M+fSBUaGUgY29tcGxldGUgc2V0IG9mIGZsYWdzIGZvciB0aGUgb3BlcmF0aW9uXG4gICAqL1xuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgYXN5bmMgZmxhZ3M8TSBleHRlbmRzIE1vZGVsPihcbiAgICBvcGVyYXRpb246IE9wZXJhdGlvbktleXMsXG4gICAgbW9kZWw6IENvbnN0cnVjdG9yPE0+LFxuICAgIGZsYWdzOiBQYXJ0aWFsPFBvdWNoRmxhZ3M+XG4gICk6IFByb21pc2U8UG91Y2hGbGFncz4ge1xuICAgIGlmICghdGhpcy5jb25maWcudXNlcikgdGhpcy5jb25maWcudXNlciA9IGNyeXB0by5yYW5kb21VVUlEKCk7XG4gICAgcmV0dXJuIE9iamVjdC5hc3NpZ24oYXdhaXQgc3VwZXIuZmxhZ3Mob3BlcmF0aW9uLCBtb2RlbCwgZmxhZ3MpLCB7XG4gICAgICBVVUlEOiB0aGlzLmNvbmZpZy51c2VyLFxuICAgIH0pIGFzIFBvdWNoRmxhZ3M7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENyZWF0ZXMgZGF0YWJhc2UgaW5kZXhlcyBmb3IgdGhlIGdpdmVuIG1vZGVsc1xuICAgKiBAc3VtbWFyeSBHZW5lcmF0ZXMgYW5kIGNyZWF0ZXMgaW5kZXhlcyBpbiB0aGUgUG91Y2hEQiBkYXRhYmFzZSBiYXNlZCBvbiB0aGUgcHJvdmlkZWQgbW9kZWwgY29uc3RydWN0b3JzLlxuICAgKiBUaGlzIG1ldGhvZCB1c2VzIHRoZSBnZW5lcmF0ZUluZGV4ZXMgdXRpbGl0eSB0byBjcmVhdGUgaW5kZXggZGVmaW5pdGlvbnMgYW5kIHRoZW4gY3JlYXRlcyB0aGVtIGluIHRoZSBkYXRhYmFzZS5cbiAgICogQHRlbXBsYXRlIE0gLSBUaGUgbW9kZWwgdHlwZSB0aGF0IGV4dGVuZHMgTW9kZWxcbiAgICogQHBhcmFtIG1vZGVscyAtIFRoZSBtb2RlbCBjb25zdHJ1Y3RvcnMgdG8gY3JlYXRlIGluZGV4ZXMgZm9yXG4gICAqIEByZXR1cm4ge1Byb21pc2U8dm9pZD59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHdoZW4gYWxsIGluZGV4ZXMgYXJlIGNyZWF0ZWRcbiAgICovXG4gIHByb3RlY3RlZCBhc3luYyBpbmRleDxNIGV4dGVuZHMgTW9kZWw+KFxuICAgIC4uLm1vZGVsczogQ29uc3RydWN0b3I8TT5bXVxuICApOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCBpbmRleGVzOiBDcmVhdGVJbmRleFJlcXVlc3RbXSA9IGdlbmVyYXRlSW5kZXhlcyhtb2RlbHMpO1xuICAgIGZvciAoY29uc3QgaW5kZXggb2YgaW5kZXhlcykge1xuICAgICAgY29uc3QgcmVzOiBDcmVhdGVJbmRleFJlc3BvbnNlPGFueT4gPSBhd2FpdCB0aGlzLmNsaWVudC5jcmVhdGVJbmRleChcbiAgICAgICAgaW5kZXggYXMgYW55XG4gICAgICApO1xuICAgICAgY29uc3QgeyByZXN1bHQgfSA9IHJlcztcbiAgICAgIGlmIChyZXN1bHQgPT09IFwiZXhpc3RpbmdcIilcbiAgICAgICAgdGhyb3cgbmV3IENvbmZsaWN0RXJyb3IoYEluZGV4ICR7aW5kZXgubmFtZX0gYWxyZWFkeSBleGlzdHNgKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENyZWF0ZXMgYSBuZXcgZG9jdW1lbnQgaW4gdGhlIGRhdGFiYXNlXG4gICAqIEBzdW1tYXJ5IEluc2VydHMgYSBuZXcgZG9jdW1lbnQgaW50byB0aGUgUG91Y2hEQiBkYXRhYmFzZSB1c2luZyB0aGUgcHV0IG9wZXJhdGlvbi5cbiAgICogVGhpcyBtZXRob2QgaGFuZGxlcyBlcnJvciBwYXJzaW5nIGFuZCBlbnN1cmVzIHRoZSBvcGVyYXRpb24gd2FzIHN1Y2Nlc3NmdWwuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSB0YWJsZU5hbWUgLSBUaGUgbmFtZSBvZiB0aGUgdGFibGUvY29sbGVjdGlvblxuICAgKiBAcGFyYW0ge3N0cmluZ3xudW1iZXJ9IGlkIC0gVGhlIGRvY3VtZW50IElEXG4gICAqIEBwYXJhbSB7UmVjb3JkPHN0cmluZywgYW55Pn0gbW9kZWwgLSBUaGUgZG9jdW1lbnQgZGF0YSB0byBpbnNlcnRcbiAgICogQHJldHVybiB7UHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCBhbnk+Pn0gQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdGhlIGNyZWF0ZWQgZG9jdW1lbnQgd2l0aCBtZXRhZGF0YVxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gICAqXG4gICAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiBjcmVhdGUodGFibGVOYW1lLCBpZCwgbW9kZWwpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogcHV0KG1vZGVsKVxuICAgKiAgIGFsdCBTdWNjZXNzXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogUmVzcG9uc2Ugd2l0aCBvaz10cnVlXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IGFzc2lnbk1ldGFkYXRhKG1vZGVsLCByZXNwb25zZS5yZXYpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBVcGRhdGVkIG1vZGVsIHdpdGggbWV0YWRhdGFcbiAgICogICBlbHNlIEVycm9yXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogRXJyb3JcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogcGFyc2VFcnJvcihlKVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVGhyb3dzIGVycm9yXG4gICAqICAgZW5kXG4gICAqL1xuICBhc3luYyBjcmVhdGUoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWQ6IHN0cmluZyB8IG51bWJlcixcbiAgICBtb2RlbDogUmVjb3JkPHN0cmluZywgYW55PlxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT4+IHtcbiAgICBsZXQgcmVzcG9uc2U6IFJlc3BvbnNlO1xuICAgIHRyeSB7XG4gICAgICByZXNwb25zZSA9IGF3YWl0IHRoaXMuY2xpZW50LnB1dChtb2RlbCk7XG4gICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgdGhyb3cgdGhpcy5wYXJzZUVycm9yKGUgYXMgRXJyb3IpO1xuICAgIH1cblxuICAgIGlmICghcmVzcG9uc2Uub2spXG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihcbiAgICAgICAgYEZhaWxlZCB0byBpbnNlcnQgZG9jIGlkOiAke2lkfSBpbiB0YWJsZSAke3RhYmxlTmFtZX1gXG4gICAgICApO1xuICAgIHJldHVybiB0aGlzLmFzc2lnbk1ldGFkYXRhKG1vZGVsLCByZXNwb25zZS5yZXYpO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBDcmVhdGVzIG11bHRpcGxlIGRvY3VtZW50cyBpbiB0aGUgZGF0YWJhc2UgaW4gYSBzaW5nbGUgb3BlcmF0aW9uXG4gICAqIEBzdW1tYXJ5IEluc2VydHMgbXVsdGlwbGUgZG9jdW1lbnRzIGludG8gdGhlIFBvdWNoREIgZGF0YWJhc2UgdXNpbmcgdGhlIGJ1bGtEb2NzIG9wZXJhdGlvbi5cbiAgICogVGhpcyBtZXRob2QgaGFuZGxlcyBlcnJvciBwYXJzaW5nIGFuZCBlbnN1cmVzIGFsbCBvcGVyYXRpb25zIHdlcmUgc3VjY2Vzc2Z1bC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IHRhYmxlTmFtZSAtIFRoZSBuYW1lIG9mIHRoZSB0YWJsZS9jb2xsZWN0aW9uXG4gICAqIEBwYXJhbSB7c3RyaW5nW118bnVtYmVyW119IGlkcyAtIFRoZSBkb2N1bWVudCBJRHNcbiAgICogQHBhcmFtICBtb2RlbHMgLSBUaGUgZG9jdW1lbnQgZGF0YSB0byBpbnNlcnRcbiAgICogQHJldHVybiBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgY3JlYXRlZCBkb2N1bWVudHMgd2l0aCBtZXRhZGF0YVxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gICAqXG4gICAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiBjcmVhdGVBbGwodGFibGVOYW1lLCBpZHMsIG1vZGVscylcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaERCOiBidWxrRG9jcyhtb2RlbHMpXG4gICAqICAgYWx0IFN1Y2Nlc3NcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBBcnJheSBvZiByZXNwb25zZXMgd2l0aCBvaz10cnVlXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IGFzc2lnbk11bHRpcGxlTWV0YWRhdGEobW9kZWxzLCByZXZzKVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVXBkYXRlZCBtb2RlbHMgd2l0aCBtZXRhZGF0YVxuICAgKiAgIGVsc2UgRXJyb3JcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBBcnJheSB3aXRoIGVycm9yc1xuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBDaGVjayBmb3IgZXJyb3JzXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBUaHJvd3MgSW50ZXJuYWxFcnJvclxuICAgKiAgIGVuZFxuICAgKi9cbiAgb3ZlcnJpZGUgYXN5bmMgY3JlYXRlQWxsKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkczogc3RyaW5nW10gfCBudW1iZXJbXSxcbiAgICBtb2RlbHM6IFJlY29yZDxzdHJpbmcsIGFueT5bXVxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT5bXT4ge1xuICAgIGxldCByZXNwb25zZTogUmVzcG9uc2VbXSB8IEVycltdO1xuICAgIHRyeSB7XG4gICAgICByZXNwb25zZSA9IGF3YWl0IHRoaXMuY2xpZW50LmJ1bGtEb2NzKG1vZGVscyk7XG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlKTtcbiAgICB9XG4gICAgaWYgKCFyZXNwb25zZS5ldmVyeSgocjogUmVzcG9uc2UgfCBFcnIpID0+IChyIGFzIFJlc3BvbnNlKS5vaykpIHtcbiAgICAgIGNvbnN0IGVycm9ycyA9IHJlc3BvbnNlLnJlZHVjZSgoYWNjdW06IHN0cmluZ1tdLCBlbCwgaSkgPT4ge1xuICAgICAgICBpZiAoZWwuZXJyb3IpXG4gICAgICAgICAgYWNjdW0ucHVzaChcbiAgICAgICAgICAgIGBlbCAke2l9OiAke2VsLmVycm9yfSR7ZWwucmVhc29uID8gYCAtICR7ZWwucmVhc29ufWAgOiBcIlwifWBcbiAgICAgICAgICApO1xuICAgICAgICByZXR1cm4gYWNjdW07XG4gICAgICB9LCBbXSk7XG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihlcnJvcnMuam9pbihcIlxcblwiKSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHRoaXMuYXNzaWduTXVsdGlwbGVNZXRhZGF0YShcbiAgICAgIG1vZGVscyxcbiAgICAgIHJlc3BvbnNlLm1hcCgocikgPT4gci5yZXYgYXMgc3RyaW5nKVxuICAgICk7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFJldHJpZXZlcyBhIGRvY3VtZW50IGZyb20gdGhlIGRhdGFiYXNlIGJ5IElEXG4gICAqIEBzdW1tYXJ5IEZldGNoZXMgYSBkb2N1bWVudCBmcm9tIHRoZSBQb3VjaERCIGRhdGFiYXNlIHVzaW5nIHRoZSBnZXQgb3BlcmF0aW9uLlxuICAgKiBUaGlzIG1ldGhvZCBnZW5lcmF0ZXMgdGhlIGRvY3VtZW50IElEIGJhc2VkIG9uIHRoZSB0YWJsZSBuYW1lIGFuZCBJRCwgdGhlbiByZXRyaWV2ZXMgdGhlIGRvY3VtZW50LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gdGFibGVOYW1lIC0gVGhlIG5hbWUgb2YgdGhlIHRhYmxlL2NvbGxlY3Rpb25cbiAgICogQHBhcmFtIHtzdHJpbmd8bnVtYmVyfSBpZCAtIFRoZSBkb2N1bWVudCBJRFxuICAgKiBAcmV0dXJuIHtQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT4+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgcmV0cmlldmVkIGRvY3VtZW50IHdpdGggbWV0YWRhdGFcbiAgICogQG1lcm1haWRcbiAgICogc2VxdWVuY2VEaWFncmFtXG4gICAqICAgcGFydGljaXBhbnQgQ2xpZW50XG4gICAqICAgcGFydGljaXBhbnQgUG91Y2hBZGFwdGVyXG4gICAqICAgcGFydGljaXBhbnQgUG91Y2hEQlxuICAgKlxuICAgKiAgIENsaWVudC0+PlBvdWNoQWRhcHRlcjogcmVhZCh0YWJsZU5hbWUsIGlkKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogZ2VuZXJhdGVJZCh0YWJsZU5hbWUsIGlkKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IGdldChfaWQpXG4gICAqICAgYWx0IFN1Y2Nlc3NcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBEb2N1bWVudFxuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBhc3NpZ25NZXRhZGF0YShyZWNvcmQsIHJlY29yZC5fcmV2KVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogRG9jdW1lbnQgd2l0aCBtZXRhZGF0YVxuICAgKiAgIGVsc2UgRXJyb3JcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBFcnJvclxuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBwYXJzZUVycm9yKGUpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBUaHJvd3MgZXJyb3JcbiAgICogICBlbmRcbiAgICovXG4gIGFzeW5jIHJlYWQoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWQ6IHN0cmluZyB8IG51bWJlclxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT4+IHtcbiAgICBjb25zdCBfaWQgPSB0aGlzLmdlbmVyYXRlSWQodGFibGVOYW1lLCBpZCk7XG4gICAgbGV0IHJlY29yZDogSWRNZXRhICYgR2V0TWV0YTtcbiAgICB0cnkge1xuICAgICAgcmVjb3JkID0gYXdhaXQgdGhpcy5jbGllbnQuZ2V0KF9pZCk7XG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuYXNzaWduTWV0YWRhdGEocmVjb3JkLCByZWNvcmQuX3Jldik7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFJldHJpZXZlcyBtdWx0aXBsZSBkb2N1bWVudHMgZnJvbSB0aGUgZGF0YWJhc2UgYnkgdGhlaXIgSURzXG4gICAqIEBzdW1tYXJ5IEZldGNoZXMgbXVsdGlwbGUgZG9jdW1lbnRzIGZyb20gdGhlIFBvdWNoREIgZGF0YWJhc2UgdXNpbmcgdGhlIGJ1bGtHZXQgb3BlcmF0aW9uLlxuICAgKiBUaGlzIG1ldGhvZCBnZW5lcmF0ZXMgZG9jdW1lbnQgSURzIGJhc2VkIG9uIHRoZSB0YWJsZSBuYW1lIGFuZCBJRHMsIHRoZW4gcmV0cmlldmVzIHRoZSBkb2N1bWVudHMuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSB0YWJsZU5hbWUgLSBUaGUgbmFtZSBvZiB0aGUgdGFibGUvY29sbGVjdGlvblxuICAgKiBAcGFyYW0ge0FycmF5PHN0cmluZ3xudW1iZXJ8YmlnaW50Pn0gaWRzIC0gVGhlIGRvY3VtZW50IElEc1xuICAgKiBAcmV0dXJuIEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSByZXRyaWV2ZWQgZG9jdW1lbnRzIHdpdGggbWV0YWRhdGFcbiAgICogQG1lcm1haWRcbiAgICogc2VxdWVuY2VEaWFncmFtXG4gICAqICAgcGFydGljaXBhbnQgQ2xpZW50XG4gICAqICAgcGFydGljaXBhbnQgUG91Y2hBZGFwdGVyXG4gICAqICAgcGFydGljaXBhbnQgUG91Y2hEQlxuICAgKlxuICAgKiAgIENsaWVudC0+PlBvdWNoQWRhcHRlcjogcmVhZEFsbCh0YWJsZU5hbWUsIGlkcylcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IE1hcCBpZHMgdG8gZ2VuZXJhdGVJZCh0YWJsZU5hbWUsIGlkKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IGJ1bGtHZXQoe2RvY3N9KVxuICAgKiAgIGFsdCBTdWNjZXNzXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogQnVsa0dldFJlc3BvbnNlXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IFByb2Nlc3MgcmVzdWx0c1xuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBhc3NpZ25NZXRhZGF0YSBmb3IgZWFjaCBkb2NcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IERvY3VtZW50cyB3aXRoIG1ldGFkYXRhXG4gICAqICAgZWxzZSBFcnJvclxuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBwYXJzZUVycm9yKGVycm9yKVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVGhyb3dzIGVycm9yXG4gICAqICAgZW5kXG4gICAqL1xuICBvdmVycmlkZSBhc3luYyByZWFkQWxsKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkczogKHN0cmluZyB8IG51bWJlciB8IGJpZ2ludClbXVxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT5bXT4ge1xuICAgIGNvbnN0IHJlc3VsdHM6IEJ1bGtHZXRSZXNwb25zZTxhbnk+ID0gYXdhaXQgdGhpcy5jbGllbnQuYnVsa0dldCh7XG4gICAgICBkb2NzOiBpZHMubWFwKChpZCkgPT4gKHsgaWQ6IHRoaXMuZ2VuZXJhdGVJZCh0YWJsZU5hbWUsIGlkIGFzIGFueSkgfSkpLFxuICAgIH0pO1xuICAgIGNvbnN0IHJlcyA9IHJlc3VsdHMucmVzdWx0cy5yZWR1Y2UoKGFjY3VtOiBhbnlbXSwgcikgPT4ge1xuICAgICAgci5kb2NzLmZvckVhY2goKGQpID0+IHtcbiAgICAgICAgaWYgKChkIGFzIGFueSkuZXJyb3IgfHwgIShkIGFzIGFueSkub2spXG4gICAgICAgICAgdGhyb3cgUG91Y2hBZGFwdGVyLnBhcnNlRXJyb3IoXG4gICAgICAgICAgICAoKGQgYXMgeyBlcnJvcjogRXJyIH0pLmVycm9yIGFzIEVycm9yKSB8fFxuICAgICAgICAgICAgICBuZXcgSW50ZXJuYWxFcnJvcihcIk1pc3NpbmcgdmFsaWQgcmVzcG9uc2VcIilcbiAgICAgICAgICApO1xuICAgICAgICBjb25zdCByZXN1bHQgPSBPYmplY3QuYXNzaWduKHt9LCAoZCBhcyB7IG9rOiBhbnkgfSkub2spO1xuICAgICAgICBhY2N1bS5wdXNoKHRoaXMuYXNzaWduTWV0YWRhdGEocmVzdWx0LCAoZCBhcyBhbnkpLm9rW0NvdWNoREJLZXlzLlJFVl0pKTtcbiAgICAgIH0pO1xuICAgICAgcmV0dXJuIGFjY3VtO1xuICAgIH0sIFtdKTtcblxuICAgIHJldHVybiByZXM7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFVwZGF0ZXMgYW4gZXhpc3RpbmcgZG9jdW1lbnQgaW4gdGhlIGRhdGFiYXNlXG4gICAqIEBzdW1tYXJ5IFVwZGF0ZXMgYSBkb2N1bWVudCBpbiB0aGUgUG91Y2hEQiBkYXRhYmFzZSB1c2luZyB0aGUgcHV0IG9wZXJhdGlvbi5cbiAgICogVGhpcyBtZXRob2QgaGFuZGxlcyBlcnJvciBwYXJzaW5nIGFuZCBlbnN1cmVzIHRoZSBvcGVyYXRpb24gd2FzIHN1Y2Nlc3NmdWwuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSB0YWJsZU5hbWUgLSBUaGUgbmFtZSBvZiB0aGUgdGFibGUvY29sbGVjdGlvblxuICAgKiBAcGFyYW0ge3N0cmluZ3xudW1iZXJ9IGlkIC0gVGhlIGRvY3VtZW50IElEXG4gICAqIEBwYXJhbSB7UmVjb3JkPHN0cmluZywgYW55Pn0gbW9kZWwgLSBUaGUgdXBkYXRlZCBkb2N1bWVudCBkYXRhXG4gICAqIEByZXR1cm4ge1Byb21pc2U8UmVjb3JkPHN0cmluZywgYW55Pj59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSB1cGRhdGVkIGRvY3VtZW50IHdpdGggbWV0YWRhdGFcbiAgICogQG1lcm1haWRcbiAgICogc2VxdWVuY2VEaWFncmFtXG4gICAqICAgcGFydGljaXBhbnQgQ2xpZW50XG4gICAqICAgcGFydGljaXBhbnQgUG91Y2hBZGFwdGVyXG4gICAqICAgcGFydGljaXBhbnQgUG91Y2hEQlxuICAgKlxuICAgKiAgIENsaWVudC0+PlBvdWNoQWRhcHRlcjogdXBkYXRlKHRhYmxlTmFtZSwgaWQsIG1vZGVsKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IHB1dChtb2RlbClcbiAgICogICBhbHQgU3VjY2Vzc1xuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IFJlc3BvbnNlIHdpdGggb2s9dHJ1ZVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBhc3NpZ25NZXRhZGF0YShtb2RlbCwgcmVzcG9uc2UucmV2KVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVXBkYXRlZCBtb2RlbCB3aXRoIG1ldGFkYXRhXG4gICAqICAgZWxzZSBFcnJvclxuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEVycm9yXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IHBhcnNlRXJyb3IoZSlcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IFRocm93cyBlcnJvclxuICAgKiAgIGVuZFxuICAgKi9cbiAgb3ZlcnJpZGUgYXN5bmMgdXBkYXRlKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkOiBzdHJpbmcgfCBudW1iZXIsXG4gICAgbW9kZWw6IFJlY29yZDxzdHJpbmcsIGFueT5cbiAgKTogUHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCBhbnk+PiB7XG4gICAgbGV0IHJlc3BvbnNlOiBSZXNwb25zZTtcbiAgICB0cnkge1xuICAgICAgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLmNsaWVudC5wdXQobW9kZWwpO1xuICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgdGhyb3cgUG91Y2hBZGFwdGVyLnBhcnNlRXJyb3IoZSk7XG4gICAgfVxuXG4gICAgaWYgKCFyZXNwb25zZS5vaylcbiAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKFxuICAgICAgICBgRmFpbGVkIHRvIHVwZGF0ZSBkb2MgaWQ6ICR7aWR9IGluIHRhYmxlICR7dGFibGVOYW1lfWBcbiAgICAgICk7XG4gICAgcmV0dXJuIHRoaXMuYXNzaWduTWV0YWRhdGEobW9kZWwsIHJlc3BvbnNlLnJldik7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFVwZGF0ZXMgbXVsdGlwbGUgZG9jdW1lbnRzIGluIHRoZSBkYXRhYmFzZSBpbiBhIHNpbmdsZSBvcGVyYXRpb25cbiAgICogQHN1bW1hcnkgVXBkYXRlcyBtdWx0aXBsZSBkb2N1bWVudHMgaW4gdGhlIFBvdWNoREIgZGF0YWJhc2UgdXNpbmcgdGhlIGJ1bGtEb2NzIG9wZXJhdGlvbi5cbiAgICogVGhpcyBtZXRob2QgaGFuZGxlcyBlcnJvciBwYXJzaW5nIGFuZCBlbnN1cmVzIGFsbCBvcGVyYXRpb25zIHdlcmUgc3VjY2Vzc2Z1bC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IHRhYmxlTmFtZSAtIFRoZSBuYW1lIG9mIHRoZSB0YWJsZS9jb2xsZWN0aW9uXG4gICAqIEBwYXJhbSB7c3RyaW5nW118bnVtYmVyW119IGlkcyAtIFRoZSBkb2N1bWVudCBJRHNcbiAgICogQHBhcmFtIG1vZGVscyAtIFRoZSB1cGRhdGVkIGRvY3VtZW50IGRhdGFcbiAgICogQHJldHVybiBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgdXBkYXRlZCBkb2N1bWVudHMgd2l0aCBtZXRhZGF0YVxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gICAqXG4gICAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiB1cGRhdGVBbGwodGFibGVOYW1lLCBpZHMsIG1vZGVscylcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaERCOiBidWxrRG9jcyhtb2RlbHMpXG4gICAqICAgYWx0IFN1Y2Nlc3NcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBBcnJheSBvZiByZXNwb25zZXMgd2l0aCBvaz10cnVlXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IGFzc2lnbk11bHRpcGxlTWV0YWRhdGEobW9kZWxzLCByZXZzKVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVXBkYXRlZCBtb2RlbHMgd2l0aCBtZXRhZGF0YVxuICAgKiAgIGVsc2UgRXJyb3JcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBBcnJheSB3aXRoIGVycm9yc1xuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBDaGVjayBmb3IgZXJyb3JzXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBUaHJvd3MgSW50ZXJuYWxFcnJvclxuICAgKiAgIGVuZFxuICAgKi9cbiAgb3ZlcnJpZGUgYXN5bmMgdXBkYXRlQWxsKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkczogc3RyaW5nW10gfCBudW1iZXJbXSxcbiAgICBtb2RlbHM6IFJlY29yZDxzdHJpbmcsIGFueT5bXVxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT5bXT4ge1xuICAgIGxldCByZXNwb25zZTogKFJlc3BvbnNlIHwgRXJyKVtdO1xuICAgIHRyeSB7XG4gICAgICByZXNwb25zZSA9IGF3YWl0IHRoaXMuY2xpZW50LmJ1bGtEb2NzKG1vZGVscyk7XG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlKTtcbiAgICB9XG4gICAgaWYgKCFyZXNwb25zZS5ldmVyeSgocikgPT4gIShyIGFzIGFueSkuZXJyb3IpKSB7XG4gICAgICBjb25zdCBlcnJvcnMgPSByZXNwb25zZS5yZWR1Y2UoKGFjY3VtOiBzdHJpbmdbXSwgZWwsIGkpID0+IHtcbiAgICAgICAgaWYgKChlbCBhcyBhbnkpLmVycm9yKVxuICAgICAgICAgIGFjY3VtLnB1c2goXG4gICAgICAgICAgICBgZWwgJHtpfTogJHsoZWwgYXMgYW55KS5lcnJvcn0keyhlbCBhcyBhbnkpLnJlYXNvbiA/IGAgLSAkeyhlbCBhcyBhbnkpLnJlYXNvbn1gIDogXCJcIn1gXG4gICAgICAgICAgKTtcbiAgICAgICAgcmV0dXJuIGFjY3VtO1xuICAgICAgfSwgW10pO1xuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoZXJyb3JzLmpvaW4oXCJcXG5cIikpO1xuICAgIH1cblxuICAgIHJldHVybiB0aGlzLmFzc2lnbk11bHRpcGxlTWV0YWRhdGEoXG4gICAgICBtb2RlbHMsXG4gICAgICByZXNwb25zZS5tYXAoKHIpID0+IHIucmV2IGFzIHN0cmluZylcbiAgICApO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBEZWxldGVzIGEgZG9jdW1lbnQgZnJvbSB0aGUgZGF0YWJhc2UgYnkgSURcbiAgICogQHN1bW1hcnkgUmVtb3ZlcyBhIGRvY3VtZW50IGZyb20gdGhlIFBvdWNoREIgZGF0YWJhc2UgdXNpbmcgdGhlIHJlbW92ZSBvcGVyYXRpb24uXG4gICAqIFRoaXMgbWV0aG9kIGZpcnN0IHJldHJpZXZlcyB0aGUgZG9jdW1lbnQgdG8gZ2V0IGl0cyByZXZpc2lvbiwgdGhlbiBkZWxldGVzIGl0LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gdGFibGVOYW1lIC0gVGhlIG5hbWUgb2YgdGhlIHRhYmxlL2NvbGxlY3Rpb25cbiAgICogQHBhcmFtIHtzdHJpbmd8bnVtYmVyfSBpZCAtIFRoZSBkb2N1bWVudCBJRFxuICAgKiBAcmV0dXJuIHtQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT4+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgZGVsZXRlZCBkb2N1bWVudCB3aXRoIG1ldGFkYXRhXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENsaWVudFxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoREJcbiAgICpcbiAgICogICBDbGllbnQtPj5Qb3VjaEFkYXB0ZXI6IGRlbGV0ZSh0YWJsZU5hbWUsIGlkKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogZ2VuZXJhdGVJZCh0YWJsZU5hbWUsIGlkKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IGdldChfaWQpXG4gICAqICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IERvY3VtZW50IHdpdGggX3JldlxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IHJlbW92ZShfaWQsIHJlY29yZC5fcmV2KVxuICAgKiAgIGFsdCBTdWNjZXNzXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogU3VjY2VzcyByZXNwb25zZVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBhc3NpZ25NZXRhZGF0YShyZWNvcmQsIHJlY29yZC5fcmV2KVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogRGVsZXRlZCBkb2N1bWVudCB3aXRoIG1ldGFkYXRhXG4gICAqICAgZWxzZSBFcnJvclxuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEVycm9yXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IHBhcnNlRXJyb3IoZSlcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IFRocm93cyBlcnJvclxuICAgKiAgIGVuZFxuICAgKi9cbiAgb3ZlcnJpZGUgYXN5bmMgZGVsZXRlKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkOiBzdHJpbmcgfCBudW1iZXJcbiAgKTogUHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCBhbnk+PiB7XG4gICAgY29uc3QgX2lkID0gdGhpcy5nZW5lcmF0ZUlkKHRhYmxlTmFtZSwgaWQpO1xuICAgIGxldCByZWNvcmQ6IElkTWV0YSAmIEdldE1ldGE7XG4gICAgdHJ5IHtcbiAgICAgIHJlY29yZCA9IGF3YWl0IHRoaXMuY2xpZW50LmdldChfaWQpO1xuICAgICAgYXdhaXQgdGhpcy5jbGllbnQucmVtb3ZlKF9pZCwgcmVjb3JkLl9yZXYpO1xuICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgdGhyb3cgUG91Y2hBZGFwdGVyLnBhcnNlRXJyb3IoZSk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLmFzc2lnbk1ldGFkYXRhKHJlY29yZCwgcmVjb3JkLl9yZXYpO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBEZWxldGVzIG11bHRpcGxlIGRvY3VtZW50cyBmcm9tIHRoZSBkYXRhYmFzZSBieSB0aGVpciBJRHNcbiAgICogQHN1bW1hcnkgUmVtb3ZlcyBtdWx0aXBsZSBkb2N1bWVudHMgZnJvbSB0aGUgUG91Y2hEQiBkYXRhYmFzZSBpbiBhIHNpbmdsZSBvcGVyYXRpb24uXG4gICAqIFRoaXMgbWV0aG9kIGZpcnN0IHJldHJpZXZlcyBhbGwgZG9jdW1lbnRzIHRvIGdldCB0aGVpciByZXZpc2lvbnMsIHRoZW4gbWFya3MgdGhlbSBhcyBkZWxldGVkLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gdGFibGVOYW1lIC0gVGhlIG5hbWUgb2YgdGhlIHRhYmxlL2NvbGxlY3Rpb25cbiAgICogQHBhcmFtIHtBcnJheTxzdHJpbmd8bnVtYmVyfGJpZ2ludD59IGlkcyAtIFRoZSBkb2N1bWVudCBJRHNcbiAgICogQHJldHVybiBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgZGVsZXRlZCBkb2N1bWVudHMgd2l0aCBtZXRhZGF0YVxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gICAqXG4gICAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiBkZWxldGVBbGwodGFibGVOYW1lLCBpZHMpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBNYXAgaWRzIHRvIGdlbmVyYXRlSWQodGFibGVOYW1lLCBpZClcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaERCOiBidWxrR2V0KHtkb2NzfSlcbiAgICogICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogQnVsa0dldFJlc3BvbnNlIHdpdGggZG9jdW1lbnRzXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBNYXJrIGRvY3VtZW50cyBhcyBkZWxldGVkXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogYnVsa0RvY3MobWFya2VkIGRvY3VtZW50cylcbiAgICogICBhbHQgU3VjY2Vzc1xuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IFN1Y2Nlc3MgcmVzcG9uc2VzXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IFByb2Nlc3MgcmVzdWx0c1xuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBhc3NpZ25NZXRhZGF0YSBmb3IgZWFjaCBkb2NcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IERlbGV0ZWQgZG9jdW1lbnRzIHdpdGggbWV0YWRhdGFcbiAgICogICBlbHNlIEVycm9yXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IENoZWNrIGZvciBlcnJvcnNcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IFRocm93cyBJbnRlcm5hbEVycm9yXG4gICAqICAgZW5kXG4gICAqL1xuICBvdmVycmlkZSBhc3luYyBkZWxldGVBbGwoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWRzOiAoc3RyaW5nIHwgbnVtYmVyIHwgYmlnaW50KVtdXG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55PltdPiB7XG4gICAgY29uc3QgcmVzdWx0czogQnVsa0dldFJlc3BvbnNlPGFueT4gPSBhd2FpdCB0aGlzLmNsaWVudC5idWxrR2V0KHtcbiAgICAgIGRvY3M6IGlkcy5tYXAoKGlkKSA9PiAoeyBpZDogdGhpcy5nZW5lcmF0ZUlkKHRhYmxlTmFtZSwgaWQgYXMgYW55KSB9KSksXG4gICAgfSk7XG5cbiAgICBjb25zdCBkZWxldGlvbjogKFJlc3BvbnNlIHwgRXJyKVtdID0gYXdhaXQgdGhpcy5jbGllbnQuYnVsa0RvY3MoXG4gICAgICByZXN1bHRzLnJlc3VsdHMubWFwKChyKSA9PiB7XG4gICAgICAgIChyIGFzIGFueSlbQ291Y2hEQktleXMuREVMRVRFRF0gPSB0cnVlO1xuICAgICAgICByZXR1cm4gcjtcbiAgICAgIH0pXG4gICAgKTtcblxuICAgIGNvbnN0IGVycnMgPSBkZWxldGlvbi5maWx0ZXIoKGQpID0+IChkIGFzIGFueSkuZXJyb3IpO1xuICAgIGlmIChlcnJzLmxlbmd0aCkgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoZXJycy5qb2luKFwiXFxuXCIpKTtcblxuICAgIHJldHVybiByZXN1bHRzLnJlc3VsdHMucmVkdWNlKChhY2N1bTogYW55W10sIHIpID0+IHtcbiAgICAgIHIuZG9jcy5mb3JFYWNoKChkKSA9PiB7XG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IE9iamVjdC5hc3NpZ24oe30sIChkIGFzIHsgb2s6IGFueSB9KS5vayk7XG4gICAgICAgIGFjY3VtLnB1c2godGhpcy5hc3NpZ25NZXRhZGF0YShyZXN1bHQsIChkIGFzIGFueSkub2tbQ291Y2hEQktleXMuUkVWXSkpO1xuICAgICAgfSk7XG4gICAgICByZXR1cm4gYWNjdW07XG4gICAgfSwgW10pO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBFeGVjdXRlcyBhIHJhdyBNYW5nbyBxdWVyeSBhZ2FpbnN0IHRoZSBkYXRhYmFzZVxuICAgKiBAc3VtbWFyeSBQZXJmb3JtcyBhIGRpcmVjdCBmaW5kIG9wZXJhdGlvbiB1c2luZyBhIE1hbmdvIHF1ZXJ5IG9iamVjdC5cbiAgICogVGhpcyBtZXRob2QgYWxsb3dzIGZvciBjb21wbGV4IHF1ZXJpZXMgYmV5b25kIHRoZSBzdGFuZGFyZCBDUlVEIG9wZXJhdGlvbnMuXG4gICAqIEB0ZW1wbGF0ZSBWIC0gVGhlIHJldHVybiB0eXBlXG4gICAqIEBwYXJhbSB7TWFuZ29RdWVyeX0gcmF3SW5wdXQgLSBUaGUgTWFuZ28gcXVlcnkgdG8gZXhlY3V0ZVxuICAgKiBAcGFyYW0ge2Jvb2xlYW59IFtwcm9jZXNzPXRydWVdIC0gV2hldGhlciB0byBwcm9jZXNzIHRoZSByZXNwb25zZSAodHJ1ZSByZXR1cm5zIGp1c3QgZG9jcywgZmFsc2UgcmV0dXJucyBmdWxsIHJlc3BvbnNlKVxuICAgKiBAcmV0dXJuIHtQcm9taXNlPFY+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgcXVlcnkgcmVzdWx0c1xuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gICAqXG4gICAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiByYXc8Vj4ocmF3SW5wdXQsIHByb2Nlc3MpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogZmluZChyYXdJbnB1dClcbiAgICogICBhbHQgU3VjY2Vzc1xuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEZpbmRSZXNwb25zZVxuICAgKiAgICAgYWx0IHByb2Nlc3M9dHJ1ZVxuICAgKiAgICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiByZXNwb25zZS5kb2NzIGFzIFZcbiAgICogICAgIGVsc2UgcHJvY2Vzcz1mYWxzZVxuICAgKiAgICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiByZXNwb25zZSBhcyBWXG4gICAqICAgICBlbmRcbiAgICogICBlbHNlIEVycm9yXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogRXJyb3JcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogcGFyc2VFcnJvcihlKVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVGhyb3dzIGVycm9yXG4gICAqICAgZW5kXG4gICAqL1xuICBhc3luYyByYXc8Vj4ocmF3SW5wdXQ6IE1hbmdvUXVlcnksIHByb2Nlc3MgPSB0cnVlKTogUHJvbWlzZTxWPiB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHJlc3BvbnNlOiBGaW5kUmVzcG9uc2U8YW55PiA9IGF3YWl0IHRoaXMuY2xpZW50LmZpbmQoXG4gICAgICAgIHJhd0lucHV0IGFzIGFueVxuICAgICAgKTtcbiAgICAgIGlmIChyZXNwb25zZS53YXJuaW5nKSBjb25zb2xlLndhcm4ocmVzcG9uc2Uud2FybmluZyk7XG4gICAgICBpZiAocHJvY2VzcykgcmV0dXJuIHJlc3BvbnNlLmRvY3MgYXMgVjtcbiAgICAgIHJldHVybiByZXNwb25zZSBhcyBWO1xuICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgdGhyb3cgUG91Y2hBZGFwdGVyLnBhcnNlRXJyb3IoZSk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBQYXJzZXMgYW5kIGNvbnZlcnRzIGVycm9ycyBmcm9tIFBvdWNoREIgdG8gYXBwbGljYXRpb24tc3BlY2lmaWMgZXJyb3JzXG4gICAqIEBzdW1tYXJ5IENvbnZlcnRzIFBvdWNoREIgZXJyb3JzIHRvIHRoZSBhcHBsaWNhdGlvbidzIGVycm9yIGhpZXJhcmNoeS5cbiAgICogVGhpcyBpbnN0YW5jZSBtZXRob2QgZGVsZWdhdGVzIHRvIHRoZSBzdGF0aWMgcGFyc2VFcnJvciBtZXRob2QuXG4gICAqIEBwYXJhbSB7RXJyb3J8c3RyaW5nfSBlcnIgLSBUaGUgZXJyb3Igb2JqZWN0IG9yIG1lc3NhZ2UgdG8gcGFyc2VcbiAgICogQHBhcmFtIHtzdHJpbmd9IFtyZWFzb25dIC0gT3B0aW9uYWwgcmVhc29uIGZvciB0aGUgZXJyb3JcbiAgICogQHJldHVybiB7QmFzZUVycm9yfSBUaGUgY29udmVydGVkIGVycm9yIG9iamVjdFxuICAgKi9cbiAgb3ZlcnJpZGUgcGFyc2VFcnJvcihlcnI6IEVycm9yIHwgc3RyaW5nLCByZWFzb24/OiBzdHJpbmcpOiBCYXNlRXJyb3Ige1xuICAgIHJldHVybiBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlcnIsIHJlYXNvbik7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFN0YXRpYyBtZXRob2QgdG8gcGFyc2UgYW5kIGNvbnZlcnQgZXJyb3JzIGZyb20gUG91Y2hEQiB0byBhcHBsaWNhdGlvbi1zcGVjaWZpYyBlcnJvcnNcbiAgICogQHN1bW1hcnkgQ29udmVydHMgUG91Y2hEQiBlcnJvcnMgdG8gdGhlIGFwcGxpY2F0aW9uJ3MgZXJyb3IgaGllcmFyY2h5IGJhc2VkIG9uIGVycm9yIGNvZGVzIGFuZCBtZXNzYWdlcy5cbiAgICogVGhpcyBtZXRob2QgYW5hbHl6ZXMgdGhlIGVycm9yIHR5cGUsIHN0YXR1cyBjb2RlLCBvciBtZXNzYWdlIHRvIGRldGVybWluZSB0aGUgYXBwcm9wcmlhdGUgZXJyb3IgY2xhc3MuXG4gICAqIEBwYXJhbSB7RXJyb3J8c3RyaW5nfSBlcnIgLSBUaGUgZXJyb3Igb2JqZWN0IG9yIG1lc3NhZ2UgdG8gcGFyc2VcbiAgICogQHBhcmFtIHtzdHJpbmd9IFtyZWFzb25dIC0gT3B0aW9uYWwgcmVhc29uIGZvciB0aGUgZXJyb3JcbiAgICogQHJldHVybiB7QmFzZUVycm9yfSBUaGUgY29udmVydGVkIGVycm9yIG9iamVjdFxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDYWxsZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICpcbiAgICogICBDYWxsZXItPj5Qb3VjaEFkYXB0ZXI6IHBhcnNlRXJyb3IoZXJyLCByZWFzb24pXG4gICAqICAgYWx0IGVyciBpcyBCYXNlRXJyb3JcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IFJldHVybiBlcnIgYXMgaXNcbiAgICogICBlbHNlIGVyciBpcyBzdHJpbmdcbiAgICogICAgIGFsdCBjb250YWlucyBcImFscmVhZHkgZXhpc3RcIiBvciBcInVwZGF0ZSBjb25mbGljdFwiXG4gICAqICAgICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IENvbmZsaWN0RXJyb3JcbiAgICogICAgIGVsc2UgY29udGFpbnMgXCJtaXNzaW5nXCIgb3IgXCJkZWxldGVkXCJcbiAgICogICAgICAgUG91Y2hBZGFwdGVyLS0+PkNhbGxlcjogTm90Rm91bmRFcnJvclxuICAgKiAgICAgZW5kXG4gICAqICAgZWxzZSBlcnIgaGFzIHN0YXR1c1xuICAgKiAgICAgYWx0IHN0YXR1cyBpcyA0MDEsIDQxMiwgNDA5XG4gICAqICAgICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IENvbmZsaWN0RXJyb3JcbiAgICogICAgIGVsc2Ugc3RhdHVzIGlzIDQwNFxuICAgKiAgICAgICBQb3VjaEFkYXB0ZXItLT4+Q2FsbGVyOiBOb3RGb3VuZEVycm9yXG4gICAqICAgICBlbHNlIHN0YXR1cyBpcyA0MDBcbiAgICogICAgICAgYWx0IG1lc3NhZ2UgY29udGFpbnMgXCJObyBpbmRleCBleGlzdHNcIlxuICAgKiAgICAgICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IEluZGV4RXJyb3JcbiAgICogICAgICAgZWxzZVxuICAgKiAgICAgICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IEludGVybmFsRXJyb3JcbiAgICogICAgICAgZW5kXG4gICAqICAgICBlbHNlIG1lc3NhZ2UgY29udGFpbnMgXCJFQ09OTlJFRlVTRURcIlxuICAgKiAgICAgICBQb3VjaEFkYXB0ZXItLT4+Q2FsbGVyOiBDb25uZWN0aW9uRXJyb3JcbiAgICogICAgIGVsc2VcbiAgICogICAgICAgUG91Y2hBZGFwdGVyLS0+PkNhbGxlcjogSW50ZXJuYWxFcnJvclxuICAgKiAgICAgZW5kXG4gICAqICAgZW5kXG4gICAqL1xuICBzdGF0aWMgb3ZlcnJpZGUgcGFyc2VFcnJvcihlcnI6IEVycm9yIHwgc3RyaW5nLCByZWFzb24/OiBzdHJpbmcpOiBCYXNlRXJyb3Ige1xuICAgIC8vIHJldHVybiBzdXBlci5wYXJzZUVycm9yKGVyciwgcmVhc29uKTtcbiAgICBpZiAoZXJyIGluc3RhbmNlb2YgQmFzZUVycm9yKSByZXR1cm4gZXJyIGFzIGFueTtcbiAgICBsZXQgY29kZTogc3RyaW5nID0gXCJcIjtcbiAgICBpZiAodHlwZW9mIGVyciA9PT0gXCJzdHJpbmdcIikge1xuICAgICAgY29kZSA9IGVycjtcbiAgICAgIGlmIChjb2RlLm1hdGNoKC9hbHJlYWR5IGV4aXN0fHVwZGF0ZSBjb25mbGljdC9nKSlcbiAgICAgICAgcmV0dXJuIG5ldyBDb25mbGljdEVycm9yKGNvZGUpO1xuICAgICAgaWYgKGNvZGUubWF0Y2goL21pc3Npbmd8ZGVsZXRlZC9nKSkgcmV0dXJuIG5ldyBOb3RGb3VuZEVycm9yKGNvZGUpO1xuICAgIH0gZWxzZSBpZiAoKGVyciBhcyBhbnkpLnN0YXR1cykge1xuICAgICAgY29kZSA9IChlcnIgYXMgYW55KS5zdGF0dXM7XG4gICAgICByZWFzb24gPSByZWFzb24gfHwgZXJyLm1lc3NhZ2U7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvZGUgPSBlcnIubWVzc2FnZTtcbiAgICB9XG5cbiAgICBzd2l0Y2ggKGNvZGUudG9TdHJpbmcoKSkge1xuICAgICAgY2FzZSBcIjQwMVwiOlxuICAgICAgY2FzZSBcIjQxMlwiOlxuICAgICAgY2FzZSBcIjQwOVwiOlxuICAgICAgICByZXR1cm4gbmV3IENvbmZsaWN0RXJyb3IocmVhc29uIGFzIHN0cmluZyk7XG4gICAgICBjYXNlIFwiNDA0XCI6XG4gICAgICAgIHJldHVybiBuZXcgTm90Rm91bmRFcnJvcihyZWFzb24gYXMgc3RyaW5nKTtcbiAgICAgIGNhc2UgXCI0MDBcIjpcbiAgICAgICAgaWYgKGNvZGUudG9TdHJpbmcoKS5tYXRjaCgvTm9cXHNpbmRleFxcc2V4aXN0cy9nKSlcbiAgICAgICAgICByZXR1cm4gbmV3IEluZGV4RXJyb3IoZXJyKTtcbiAgICAgICAgcmV0dXJuIG5ldyBJbnRlcm5hbEVycm9yKGVycik7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICBpZiAoY29kZS50b1N0cmluZygpLm1hdGNoKC9FQ09OTlJFRlVTRUQvZykpXG4gICAgICAgICAgcmV0dXJuIG5ldyBDb25uZWN0aW9uRXJyb3IoZXJyKTtcbiAgICAgICAgcmV0dXJuIG5ldyBJbnRlcm5hbEVycm9yKGVycik7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBTZXRzIHVwIGRlY29yYXRpb25zIGZvciBQb3VjaERCLXNwZWNpZmljIG1vZGVsIHByb3BlcnRpZXNcbiAgICogQHN1bW1hcnkgQ29uZmlndXJlcyBkZWNvcmF0b3JzIGZvciBjcmVhdGVkQnkgYW5kIHVwZGF0ZWRCeSBmaWVsZHMgaW4gbW9kZWxzLlxuICAgKiBUaGlzIG1ldGhvZCBkZWZpbmVzIGhvdyB0aGVzZSBmaWVsZHMgc2hvdWxkIGJlIGF1dG9tYXRpY2FsbHkgcG9wdWxhdGVkIGR1cmluZyBjcmVhdGUgYW5kIHVwZGF0ZSBvcGVyYXRpb25zLlxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDYWxsZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBEZWNvcmF0aW9uXG4gICAqXG4gICAqICAgQ2FsbGVyLT4+UG91Y2hBZGFwdGVyOiBkZWNvcmF0aW9uKClcbiAgICogICBQb3VjaEFkYXB0ZXItPj5SZXBvc2l0b3J5OiBrZXkoUGVyc2lzdGVuY2VLZXlzLkNSRUFURURfQlkpXG4gICAqICAgUmVwb3NpdG9yeS0tPj5Qb3VjaEFkYXB0ZXI6IGNyZWF0ZWRCeUtleVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlJlcG9zaXRvcnk6IGtleShQZXJzaXN0ZW5jZUtleXMuVVBEQVRFRF9CWSlcbiAgICogICBSZXBvc2l0b3J5LS0+PlBvdWNoQWRhcHRlcjogdXBkYXRlZEJ5S2V5XG4gICAqXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+RGVjb3JhdGlvbjogZmxhdm91cmVkQXMoUG91Y2hGbGF2b3VyKVxuICAgKiAgIERlY29yYXRpb24tLT4+UG91Y2hBZGFwdGVyOiBEZWNvcmF0b3JCdWlsZGVyXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+RGVjb3JhdGlvbjogZm9yKGNyZWF0ZWRCeUtleSlcbiAgICogICBQb3VjaEFkYXB0ZXItPj5EZWNvcmF0aW9uOiBkZWZpbmUob25DcmVhdGUsIHByb3BNZXRhZGF0YSlcbiAgICogICBQb3VjaEFkYXB0ZXItPj5EZWNvcmF0aW9uOiBhcHBseSgpXG4gICAqXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+RGVjb3JhdGlvbjogZmxhdm91cmVkQXMoUG91Y2hGbGF2b3VyKVxuICAgKiAgIERlY29yYXRpb24tLT4+UG91Y2hBZGFwdGVyOiBEZWNvcmF0b3JCdWlsZGVyXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+RGVjb3JhdGlvbjogZm9yKHVwZGF0ZWRCeUtleSlcbiAgICogICBQb3VjaEFkYXB0ZXItPj5EZWNvcmF0aW9uOiBkZWZpbmUob25DcmVhdGUsIHByb3BNZXRhZGF0YSlcbiAgICogICBQb3VjaEFkYXB0ZXItPj5EZWNvcmF0aW9uOiBhcHBseSgpXG4gICAqL1xuICBzdGF0aWMgb3ZlcnJpZGUgZGVjb3JhdGlvbigpIHtcbiAgICBzdXBlci5kZWNvcmF0aW9uKCk7XG4gICAgY29uc3QgY3JlYXRlZEJ5S2V5ID0gUmVwb3NpdG9yeS5rZXkoUGVyc2lzdGVuY2VLZXlzLkNSRUFURURfQlkpO1xuICAgIGNvbnN0IHVwZGF0ZWRCeUtleSA9IFJlcG9zaXRvcnkua2V5KFBlcnNpc3RlbmNlS2V5cy5VUERBVEVEX0JZKTtcbiAgICBEZWNvcmF0aW9uLmZsYXZvdXJlZEFzKFBvdWNoRmxhdm91cilcbiAgICAgIC5mb3IoY3JlYXRlZEJ5S2V5KVxuICAgICAgLmRlZmluZShcbiAgICAgICAgb25DcmVhdGUoY3JlYXRlZEJ5T25Qb3VjaENyZWF0ZVVwZGF0ZSksXG4gICAgICAgIHByb3BNZXRhZGF0YShjcmVhdGVkQnlLZXksIHt9KVxuICAgICAgKVxuICAgICAgLmFwcGx5KCk7XG4gICAgRGVjb3JhdGlvbi5mbGF2b3VyZWRBcyhQb3VjaEZsYXZvdXIpXG4gICAgICAuZm9yKHVwZGF0ZWRCeUtleSlcbiAgICAgIC5kZWZpbmUoXG4gICAgICAgIG9uQ3JlYXRlVXBkYXRlKGNyZWF0ZWRCeU9uUG91Y2hDcmVhdGVVcGRhdGUpLFxuICAgICAgICBwcm9wTWV0YWRhdGEodXBkYXRlZEJ5S2V5LCB7fSlcbiAgICAgIClcbiAgICAgIC5hcHBseSgpO1xuICB9XG59XG5cblBvdWNoQWRhcHRlci5zZXRDdXJyZW50KFBvdWNoRmxhdm91cik7XG4iLCJpbXBvcnQgeyBQb3VjaEFkYXB0ZXIgfSBmcm9tIFwiLi9hZGFwdGVyXCI7XG5pbXBvcnQgeyBNZXRhZGF0YSB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuXG5Qb3VjaEFkYXB0ZXIuZGVjb3JhdGlvbigpO1xuXG5leHBvcnQgKiBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL1BvdWNoUmVwb3NpdG9yeVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdHlwZXNcIjtcbi8vIGxlZnQgdG8gdGhlIGVuZCBvbiBwdXJwb3NlXG5leHBvcnQgKiBmcm9tIFwiLi9hZGFwdGVyXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEEgVHlwZVNjcmlwdCBhZGFwdGVyIGZvciBQb3VjaERCIGludGVncmF0aW9uXG4gKiBAc3VtbWFyeSBQcm92aWRlcyBhIHJlcG9zaXRvcnktcGF0dGVybiBpbXBsZW1lbnRhdGlvbiBiYWNrZWQgYnkgUG91Y2hEQiwgZXhwb3NpbmcgdGhlIHtAbGluayBQb3VjaEFkYXB0ZXJ9IHRvIGludGVyZmFjZSB3aXRoIGRhdGFiYXNlcywgdGhlIHtAbGluayBQb3VjaFJlcG9zaXRvcnl9IGZvciB0eXBlZCBkYXRhIGFjY2VzcywgY29uZmlndXJhdGlvbiB7QGxpbmsgbW9kdWxlOmZvci1wb3VjaHxjb25zdGFudHN9IGxpa2Uge0BsaW5rIFBvdWNoRmxhdm91cn0gYW5kIHtAbGluayBEZWZhdWx0TG9jYWxTdG9yYWdlUGF0aH0sIGFuZCByZWxhdGVkIHtAbGluayBtb2R1bGU6Zm9yLXBvdWNofHR5cGVzfS4gVGhpcyBtb2R1bGUgd2lyZXMgdXAgZGVjb3JhdG9ycyBvbiBsb2FkIHRvIHN1cHBvcnQgY3JlYXRlZC91cGRhdGVkLWJ5IGZpZWxkcy5cbiAqIEBtb2R1bGUgZm9yLXBvdWNoXG4gKi9cblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gUGFja2FnZSB2ZXJzaW9uIGlkZW50aWZpZXJcbiAqIEBzdW1tYXJ5IFN0b3JlcyB0aGUgY3VycmVudCB2ZXJzaW9uIG9mIHRoZSBmb3ItcG91Y2ggcGFja2FnZVxuICogQGNvbnN0IFZFUlNJT05cbiAqIEBtZW1iZXJPZiBtb2R1bGU6Zm9yLXBvdWNoXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gXCIjI1ZFUlNJT04jI1wiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBQYWNrYWdlIHZlcnNpb24gaWRlbnRpZmllclxuICogQHN1bW1hcnkgU3RvcmVzIHRoZSBjdXJyZW50IHZlcnNpb24gb2YgdGhlIGZvci1wb3VjaCBwYWNrYWdlXG4gKiBAY29uc3QgVkVSU0lPTlxuICogQG1lbWJlck9mIG1vZHVsZTpmb3ItcG91Y2hcbiAqL1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9IFwiIyNQQUNLQUdFIyNcIjtcbk1ldGFkYXRhLnJlZ2lzdGVyTGlicmFyeShQQUNLQUdFX05BTUUsIFZFUlNJT04pO1xuIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7Ozs7OztBQU1HO0FBQ0ksTUFBTSxZQUFZLEdBQUc7QUFFNUI7Ozs7O0FBS0c7QUFDSSxNQUFNLHVCQUF1QixHQUFHOztBQ2lDdkM7Ozs7Ozs7Ozs7Ozs7OztBQWVHO0FBQ0ksZUFBZSw0QkFBNEIsQ0FNaEQsT0FBNEIsRUFDNUIsSUFBTyxFQUNQLEdBQVksRUFDWixLQUFRLEVBQUE7QUFFUixJQUFBLElBQUk7UUFDRixNQUFN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