@decaf-ts/for-pouch 0.2.2 → 0.2.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE.md +646 -143
- package/README.md +0 -0
- package/dist/for-pouch.cjs +417 -10
- package/dist/for-pouch.esm.cjs +417 -10
- package/lib/PouchRepository.cjs +1 -1
- package/lib/PouchRepository.d.ts +8 -0
- package/lib/adapter.cjs +403 -3
- package/lib/adapter.d.ts +401 -1
- package/lib/constants.cjs +8 -1
- package/lib/constants.d.ts +7 -0
- package/lib/esm/PouchRepository.d.ts +8 -0
- package/lib/esm/PouchRepository.js +1 -1
- package/lib/esm/adapter.d.ts +401 -1
- package/lib/esm/adapter.js +404 -4
- package/lib/esm/constants.d.ts +7 -0
- package/lib/esm/constants.js +8 -1
- package/lib/esm/index.d.ts +7 -7
- package/lib/esm/index.js +13 -13
- package/lib/esm/types.d.ts +10 -0
- package/lib/esm/types.js +1 -1
- package/lib/index.cjs +8 -8
- package/lib/index.d.ts +7 -7
- package/lib/types.cjs +1 -1
- package/lib/types.d.ts +10 -0
- package/package.json +5 -2
package/dist/for-pouch.esm.cjs
CHANGED
|
@@ -4,8 +4,31 @@ import { ConflictError, InternalError, BaseError, NotFoundError, onCreate } from
|
|
|
4
4
|
import { ConnectionError, Repository, PersistenceKeys, UnsupportedError } from '@decaf-ts/core';
|
|
5
5
|
import { Decoration, propMetadata } from '@decaf-ts/decorator-validation';
|
|
6
6
|
|
|
7
|
+
/**
|
|
8
|
+
* @description Identifier for PouchDB flavor in the decorator system
|
|
9
|
+
* @summary A string constant that identifies the PouchDB implementation in the decorator system.
|
|
10
|
+
* This is used to target decorators specifically for PouchDB adapters.
|
|
11
|
+
* @const PouchFlavour
|
|
12
|
+
* @memberOf module:for-pouch
|
|
13
|
+
*/
|
|
7
14
|
const PouchFlavour = "pouch";
|
|
8
15
|
|
|
16
|
+
/**
|
|
17
|
+
* @description Sets the creator ID on a model during creation or update operations
|
|
18
|
+
* @summary This function is used as a decorator handler to automatically set the creator ID field on a model
|
|
19
|
+
* when it's being created or updated. It extracts the UUID from the context and assigns it to the specified key.
|
|
20
|
+
* @template M - The model type that extends Model
|
|
21
|
+
* @template R - The repository type that extends PouchRepository<M>
|
|
22
|
+
* @template V - The relations metadata type that extends RelationsMetadata
|
|
23
|
+
* @param {R} this - The repository instance
|
|
24
|
+
* @param {Context<PouchFlags>} context - The operation context containing flags
|
|
25
|
+
* @param {V} data - The relations metadata
|
|
26
|
+
* @param key - The property key to set on the model
|
|
27
|
+
* @param {M} model - The model instance to modify
|
|
28
|
+
* @return {Promise<void>} A promise that resolves when the operation is complete
|
|
29
|
+
* @function createdByOnPouchCreateUpdate
|
|
30
|
+
* @memberOf module:for-pouch
|
|
31
|
+
*/
|
|
9
32
|
async function createdByOnPouchCreateUpdate(context, data, key, model) {
|
|
10
33
|
try {
|
|
11
34
|
const uuid = context.get("UUID");
|
|
@@ -16,11 +39,70 @@ async function createdByOnPouchCreateUpdate(context, data, key, model) {
|
|
|
16
39
|
throw new UnsupportedError("No User found in context. Please provide a user in the context");
|
|
17
40
|
}
|
|
18
41
|
}
|
|
42
|
+
/**
|
|
43
|
+
* @description PouchDB implementation of the CouchDBAdapter
|
|
44
|
+
* @summary This class provides a concrete implementation of the CouchDBAdapter for PouchDB.
|
|
45
|
+
* It handles all database operations like create, read, update, delete (CRUD) for both
|
|
46
|
+
* single documents and bulk operations. It also provides methods for querying and indexing.
|
|
47
|
+
* @template Database - The PouchDB database type
|
|
48
|
+
* @template PouchFlags - The flags specific to PouchDB operations
|
|
49
|
+
* @template Context<PouchFlags> - The context type with PouchDB flags
|
|
50
|
+
* @param {Database} scope - The PouchDB database instance
|
|
51
|
+
* @param {string} [alias] - Optional alias for the database
|
|
52
|
+
* @class PouchAdapter
|
|
53
|
+
* @example
|
|
54
|
+
* ```typescript
|
|
55
|
+
* import PouchDB from 'pouchdb';
|
|
56
|
+
* import { PouchAdapter } from '@decaf-ts/for-pouch';
|
|
57
|
+
*
|
|
58
|
+
* // Create a new PouchDB instance
|
|
59
|
+
* const db = new PouchDB('my-database');
|
|
60
|
+
*
|
|
61
|
+
* // Create a PouchAdapter with the database
|
|
62
|
+
* const adapter = new PouchAdapter(db);
|
|
63
|
+
*
|
|
64
|
+
* // Use the adapter for database operations
|
|
65
|
+
* const result = await adapter.read('users', 'user-123');
|
|
66
|
+
* ```
|
|
67
|
+
* @mermaid
|
|
68
|
+
* sequenceDiagram
|
|
69
|
+
* participant Client
|
|
70
|
+
* participant PouchAdapter
|
|
71
|
+
* participant PouchDB
|
|
72
|
+
* participant CouchDB
|
|
73
|
+
*
|
|
74
|
+
* Client->>PouchAdapter: new PouchAdapter(db)
|
|
75
|
+
* PouchAdapter->>CouchDBAdapter: super(scope, PouchFlavour, alias)
|
|
76
|
+
*
|
|
77
|
+
* Client->>PouchAdapter: create(table, id, model)
|
|
78
|
+
* PouchAdapter->>PouchDB: put(model)
|
|
79
|
+
* PouchDB->>CouchDB: HTTP PUT
|
|
80
|
+
* CouchDB-->>PouchDB: Response
|
|
81
|
+
* PouchDB-->>PouchAdapter: Response
|
|
82
|
+
* PouchAdapter-->>Client: Updated model
|
|
83
|
+
*
|
|
84
|
+
* Client->>PouchAdapter: read(table, id)
|
|
85
|
+
* PouchAdapter->>PouchDB: get(id)
|
|
86
|
+
* PouchDB->>CouchDB: HTTP GET
|
|
87
|
+
* CouchDB-->>PouchDB: Document
|
|
88
|
+
* PouchDB-->>PouchAdapter: Document
|
|
89
|
+
* PouchAdapter-->>Client: Model
|
|
90
|
+
*/
|
|
19
91
|
class PouchAdapter extends CouchDBAdapter {
|
|
20
92
|
constructor(scope, alias) {
|
|
21
93
|
super(scope, PouchFlavour, alias);
|
|
22
94
|
}
|
|
23
|
-
|
|
95
|
+
/**
|
|
96
|
+
* @description Generates operation flags for PouchDB operations
|
|
97
|
+
* @summary Creates a set of flags for a specific operation, including a UUID for identification.
|
|
98
|
+
* This method extracts the user ID from the database URL or generates a random UUID if not available.
|
|
99
|
+
* @template M - The model type that extends Model
|
|
100
|
+
* @param {OperationKeys} operation - The operation key (create, read, update, delete)
|
|
101
|
+
* @param {Constructor<M>} model - The model constructor
|
|
102
|
+
* @param {Partial<PouchFlags>} flags - Partial flags to be merged
|
|
103
|
+
* @return {Promise<PouchFlags>} The complete set of flags for the operation
|
|
104
|
+
*/
|
|
105
|
+
async flags(operation, model, flags) {
|
|
24
106
|
let id = "";
|
|
25
107
|
const url = this.native.name;
|
|
26
108
|
if (url) {
|
|
@@ -32,10 +114,18 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
32
114
|
if (!id) {
|
|
33
115
|
id = crypto.randomUUID();
|
|
34
116
|
}
|
|
35
|
-
return Object.assign(super.flags(operation, model, flags), {
|
|
117
|
+
return Object.assign(await super.flags(operation, model, flags), {
|
|
36
118
|
UUID: id,
|
|
37
119
|
});
|
|
38
120
|
}
|
|
121
|
+
/**
|
|
122
|
+
* @description Creates database indexes for the given models
|
|
123
|
+
* @summary Generates and creates indexes in the PouchDB database based on the provided model constructors.
|
|
124
|
+
* This method uses the generateIndexes utility to create index definitions and then creates them in the database.
|
|
125
|
+
* @template M - The model type that extends Model
|
|
126
|
+
* @param models - The model constructors to create indexes for
|
|
127
|
+
* @return {Promise<void>} A promise that resolves when all indexes are created
|
|
128
|
+
*/
|
|
39
129
|
async index(...models) {
|
|
40
130
|
const indexes = generateIndexes(models);
|
|
41
131
|
for (const index of indexes) {
|
|
@@ -45,6 +135,32 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
45
135
|
throw new ConflictError(`Index ${index.name} already exists`);
|
|
46
136
|
}
|
|
47
137
|
}
|
|
138
|
+
/**
|
|
139
|
+
* @description Creates a new document in the database
|
|
140
|
+
* @summary Inserts a new document into the PouchDB database using the put operation.
|
|
141
|
+
* This method handles error parsing and ensures the operation was successful.
|
|
142
|
+
* @param {string} tableName - The name of the table/collection
|
|
143
|
+
* @param {string|number} id - The document ID
|
|
144
|
+
* @param {Record<string, any>} model - The document data to insert
|
|
145
|
+
* @return {Promise<Record<string, any>>} A promise that resolves to the created document with metadata
|
|
146
|
+
* @mermaid
|
|
147
|
+
* sequenceDiagram
|
|
148
|
+
* participant Client
|
|
149
|
+
* participant PouchAdapter
|
|
150
|
+
* participant PouchDB
|
|
151
|
+
*
|
|
152
|
+
* Client->>PouchAdapter: create(tableName, id, model)
|
|
153
|
+
* PouchAdapter->>PouchDB: put(model)
|
|
154
|
+
* alt Success
|
|
155
|
+
* PouchDB-->>PouchAdapter: Response with ok=true
|
|
156
|
+
* PouchAdapter->>PouchAdapter: assignMetadata(model, response.rev)
|
|
157
|
+
* PouchAdapter-->>Client: Updated model with metadata
|
|
158
|
+
* else Error
|
|
159
|
+
* PouchDB-->>PouchAdapter: Error
|
|
160
|
+
* PouchAdapter->>PouchAdapter: parseError(e)
|
|
161
|
+
* PouchAdapter-->>Client: Throws error
|
|
162
|
+
* end
|
|
163
|
+
*/
|
|
48
164
|
async create(tableName, id, model) {
|
|
49
165
|
let response;
|
|
50
166
|
try {
|
|
@@ -57,6 +173,32 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
57
173
|
throw new InternalError(`Failed to insert doc id: ${id} in table ${tableName}`);
|
|
58
174
|
return this.assignMetadata(model, response.rev);
|
|
59
175
|
}
|
|
176
|
+
/**
|
|
177
|
+
* @description Creates multiple documents in the database in a single operation
|
|
178
|
+
* @summary Inserts multiple documents into the PouchDB database using the bulkDocs operation.
|
|
179
|
+
* This method handles error parsing and ensures all operations were successful.
|
|
180
|
+
* @param {string} tableName - The name of the table/collection
|
|
181
|
+
* @param {string[]|number[]} ids - The document IDs
|
|
182
|
+
* @param models - The document data to insert
|
|
183
|
+
* @return A promise that resolves to the created documents with metadata
|
|
184
|
+
* @mermaid
|
|
185
|
+
* sequenceDiagram
|
|
186
|
+
* participant Client
|
|
187
|
+
* participant PouchAdapter
|
|
188
|
+
* participant PouchDB
|
|
189
|
+
*
|
|
190
|
+
* Client->>PouchAdapter: createAll(tableName, ids, models)
|
|
191
|
+
* PouchAdapter->>PouchDB: bulkDocs(models)
|
|
192
|
+
* alt Success
|
|
193
|
+
* PouchDB-->>PouchAdapter: Array of responses with ok=true
|
|
194
|
+
* PouchAdapter->>PouchAdapter: assignMultipleMetadata(models, revs)
|
|
195
|
+
* PouchAdapter-->>Client: Updated models with metadata
|
|
196
|
+
* else Error
|
|
197
|
+
* PouchDB-->>PouchAdapter: Array with errors
|
|
198
|
+
* PouchAdapter->>PouchAdapter: Check for errors
|
|
199
|
+
* PouchAdapter-->>Client: Throws InternalError
|
|
200
|
+
* end
|
|
201
|
+
*/
|
|
60
202
|
async createAll(tableName, ids, models) {
|
|
61
203
|
let response;
|
|
62
204
|
try {
|
|
@@ -75,6 +217,32 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
75
217
|
}
|
|
76
218
|
return this.assignMultipleMetadata(models, response.map((r) => r.rev));
|
|
77
219
|
}
|
|
220
|
+
/**
|
|
221
|
+
* @description Retrieves a document from the database by ID
|
|
222
|
+
* @summary Fetches a document from the PouchDB database using the get operation.
|
|
223
|
+
* This method generates the document ID based on the table name and ID, then retrieves the document.
|
|
224
|
+
* @param {string} tableName - The name of the table/collection
|
|
225
|
+
* @param {string|number} id - The document ID
|
|
226
|
+
* @return {Promise<Record<string, any>>} A promise that resolves to the retrieved document with metadata
|
|
227
|
+
* @mermaid
|
|
228
|
+
* sequenceDiagram
|
|
229
|
+
* participant Client
|
|
230
|
+
* participant PouchAdapter
|
|
231
|
+
* participant PouchDB
|
|
232
|
+
*
|
|
233
|
+
* Client->>PouchAdapter: read(tableName, id)
|
|
234
|
+
* PouchAdapter->>PouchAdapter: generateId(tableName, id)
|
|
235
|
+
* PouchAdapter->>PouchDB: get(_id)
|
|
236
|
+
* alt Success
|
|
237
|
+
* PouchDB-->>PouchAdapter: Document
|
|
238
|
+
* PouchAdapter->>PouchAdapter: assignMetadata(record, record._rev)
|
|
239
|
+
* PouchAdapter-->>Client: Document with metadata
|
|
240
|
+
* else Error
|
|
241
|
+
* PouchDB-->>PouchAdapter: Error
|
|
242
|
+
* PouchAdapter->>PouchAdapter: parseError(e)
|
|
243
|
+
* PouchAdapter-->>Client: Throws error
|
|
244
|
+
* end
|
|
245
|
+
*/
|
|
78
246
|
async read(tableName, id) {
|
|
79
247
|
const _id = this.generateId(tableName, id);
|
|
80
248
|
let record;
|
|
@@ -86,6 +254,32 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
86
254
|
}
|
|
87
255
|
return this.assignMetadata(record, record._rev);
|
|
88
256
|
}
|
|
257
|
+
/**
|
|
258
|
+
* @description Retrieves multiple documents from the database by their IDs
|
|
259
|
+
* @summary Fetches multiple documents from the PouchDB database using the bulkGet operation.
|
|
260
|
+
* This method generates document IDs based on the table name and IDs, then retrieves the documents.
|
|
261
|
+
* @param {string} tableName - The name of the table/collection
|
|
262
|
+
* @param {Array<string|number|bigint>} ids - The document IDs
|
|
263
|
+
* @return A promise that resolves to the retrieved documents with metadata
|
|
264
|
+
* @mermaid
|
|
265
|
+
* sequenceDiagram
|
|
266
|
+
* participant Client
|
|
267
|
+
* participant PouchAdapter
|
|
268
|
+
* participant PouchDB
|
|
269
|
+
*
|
|
270
|
+
* Client->>PouchAdapter: readAll(tableName, ids)
|
|
271
|
+
* PouchAdapter->>PouchAdapter: Map ids to generateId(tableName, id)
|
|
272
|
+
* PouchAdapter->>PouchDB: bulkGet({docs})
|
|
273
|
+
* alt Success
|
|
274
|
+
* PouchDB-->>PouchAdapter: BulkGetResponse
|
|
275
|
+
* PouchAdapter->>PouchAdapter: Process results
|
|
276
|
+
* PouchAdapter->>PouchAdapter: assignMetadata for each doc
|
|
277
|
+
* PouchAdapter-->>Client: Documents with metadata
|
|
278
|
+
* else Error
|
|
279
|
+
* PouchAdapter->>PouchAdapter: parseError(error)
|
|
280
|
+
* PouchAdapter-->>Client: Throws error
|
|
281
|
+
* end
|
|
282
|
+
*/
|
|
89
283
|
async readAll(tableName, ids) {
|
|
90
284
|
const results = await this.native.bulkGet({
|
|
91
285
|
docs: ids.map((id) => ({ id: this.generateId(tableName, id) })),
|
|
@@ -102,6 +296,32 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
102
296
|
}, []);
|
|
103
297
|
return res;
|
|
104
298
|
}
|
|
299
|
+
/**
|
|
300
|
+
* @description Updates an existing document in the database
|
|
301
|
+
* @summary Updates a document in the PouchDB database using the put operation.
|
|
302
|
+
* This method handles error parsing and ensures the operation was successful.
|
|
303
|
+
* @param {string} tableName - The name of the table/collection
|
|
304
|
+
* @param {string|number} id - The document ID
|
|
305
|
+
* @param {Record<string, any>} model - The updated document data
|
|
306
|
+
* @return {Promise<Record<string, any>>} A promise that resolves to the updated document with metadata
|
|
307
|
+
* @mermaid
|
|
308
|
+
* sequenceDiagram
|
|
309
|
+
* participant Client
|
|
310
|
+
* participant PouchAdapter
|
|
311
|
+
* participant PouchDB
|
|
312
|
+
*
|
|
313
|
+
* Client->>PouchAdapter: update(tableName, id, model)
|
|
314
|
+
* PouchAdapter->>PouchDB: put(model)
|
|
315
|
+
* alt Success
|
|
316
|
+
* PouchDB-->>PouchAdapter: Response with ok=true
|
|
317
|
+
* PouchAdapter->>PouchAdapter: assignMetadata(model, response.rev)
|
|
318
|
+
* PouchAdapter-->>Client: Updated model with metadata
|
|
319
|
+
* else Error
|
|
320
|
+
* PouchDB-->>PouchAdapter: Error
|
|
321
|
+
* PouchAdapter->>PouchAdapter: parseError(e)
|
|
322
|
+
* PouchAdapter-->>Client: Throws error
|
|
323
|
+
* end
|
|
324
|
+
*/
|
|
105
325
|
async update(tableName, id, model) {
|
|
106
326
|
let response;
|
|
107
327
|
try {
|
|
@@ -114,6 +334,32 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
114
334
|
throw new InternalError(`Failed to update doc id: ${id} in table ${tableName}`);
|
|
115
335
|
return this.assignMetadata(model, response.rev);
|
|
116
336
|
}
|
|
337
|
+
/**
|
|
338
|
+
* @description Updates multiple documents in the database in a single operation
|
|
339
|
+
* @summary Updates multiple documents in the PouchDB database using the bulkDocs operation.
|
|
340
|
+
* This method handles error parsing and ensures all operations were successful.
|
|
341
|
+
* @param {string} tableName - The name of the table/collection
|
|
342
|
+
* @param {string[]|number[]} ids - The document IDs
|
|
343
|
+
* @param models - The updated document data
|
|
344
|
+
* @return A promise that resolves to the updated documents with metadata
|
|
345
|
+
* @mermaid
|
|
346
|
+
* sequenceDiagram
|
|
347
|
+
* participant Client
|
|
348
|
+
* participant PouchAdapter
|
|
349
|
+
* participant PouchDB
|
|
350
|
+
*
|
|
351
|
+
* Client->>PouchAdapter: updateAll(tableName, ids, models)
|
|
352
|
+
* PouchAdapter->>PouchDB: bulkDocs(models)
|
|
353
|
+
* alt Success
|
|
354
|
+
* PouchDB-->>PouchAdapter: Array of responses with ok=true
|
|
355
|
+
* PouchAdapter->>PouchAdapter: assignMultipleMetadata(models, revs)
|
|
356
|
+
* PouchAdapter-->>Client: Updated models with metadata
|
|
357
|
+
* else Error
|
|
358
|
+
* PouchDB-->>PouchAdapter: Array with errors
|
|
359
|
+
* PouchAdapter->>PouchAdapter: Check for errors
|
|
360
|
+
* PouchAdapter-->>Client: Throws InternalError
|
|
361
|
+
* end
|
|
362
|
+
*/
|
|
117
363
|
async updateAll(tableName, ids, models) {
|
|
118
364
|
let response;
|
|
119
365
|
try {
|
|
@@ -132,6 +378,34 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
132
378
|
}
|
|
133
379
|
return this.assignMultipleMetadata(models, response.map((r) => r.rev));
|
|
134
380
|
}
|
|
381
|
+
/**
|
|
382
|
+
* @description Deletes a document from the database by ID
|
|
383
|
+
* @summary Removes a document from the PouchDB database using the remove operation.
|
|
384
|
+
* This method first retrieves the document to get its revision, then deletes it.
|
|
385
|
+
* @param {string} tableName - The name of the table/collection
|
|
386
|
+
* @param {string|number} id - The document ID
|
|
387
|
+
* @return {Promise<Record<string, any>>} A promise that resolves to the deleted document with metadata
|
|
388
|
+
* @mermaid
|
|
389
|
+
* sequenceDiagram
|
|
390
|
+
* participant Client
|
|
391
|
+
* participant PouchAdapter
|
|
392
|
+
* participant PouchDB
|
|
393
|
+
*
|
|
394
|
+
* Client->>PouchAdapter: delete(tableName, id)
|
|
395
|
+
* PouchAdapter->>PouchAdapter: generateId(tableName, id)
|
|
396
|
+
* PouchAdapter->>PouchDB: get(_id)
|
|
397
|
+
* PouchDB-->>PouchAdapter: Document with _rev
|
|
398
|
+
* PouchAdapter->>PouchDB: remove(_id, record._rev)
|
|
399
|
+
* alt Success
|
|
400
|
+
* PouchDB-->>PouchAdapter: Success response
|
|
401
|
+
* PouchAdapter->>PouchAdapter: assignMetadata(record, record._rev)
|
|
402
|
+
* PouchAdapter-->>Client: Deleted document with metadata
|
|
403
|
+
* else Error
|
|
404
|
+
* PouchDB-->>PouchAdapter: Error
|
|
405
|
+
* PouchAdapter->>PouchAdapter: parseError(e)
|
|
406
|
+
* PouchAdapter-->>Client: Throws error
|
|
407
|
+
* end
|
|
408
|
+
*/
|
|
135
409
|
async delete(tableName, id) {
|
|
136
410
|
const _id = this.generateId(tableName, id);
|
|
137
411
|
let record;
|
|
@@ -144,6 +418,35 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
144
418
|
}
|
|
145
419
|
return this.assignMetadata(record, record._rev);
|
|
146
420
|
}
|
|
421
|
+
/**
|
|
422
|
+
* @description Deletes multiple documents from the database by their IDs
|
|
423
|
+
* @summary Removes multiple documents from the PouchDB database in a single operation.
|
|
424
|
+
* This method first retrieves all documents to get their revisions, then marks them as deleted.
|
|
425
|
+
* @param {string} tableName - The name of the table/collection
|
|
426
|
+
* @param {Array<string|number|bigint>} ids - The document IDs
|
|
427
|
+
* @return A promise that resolves to the deleted documents with metadata
|
|
428
|
+
* @mermaid
|
|
429
|
+
* sequenceDiagram
|
|
430
|
+
* participant Client
|
|
431
|
+
* participant PouchAdapter
|
|
432
|
+
* participant PouchDB
|
|
433
|
+
*
|
|
434
|
+
* Client->>PouchAdapter: deleteAll(tableName, ids)
|
|
435
|
+
* PouchAdapter->>PouchAdapter: Map ids to generateId(tableName, id)
|
|
436
|
+
* PouchAdapter->>PouchDB: bulkGet({docs})
|
|
437
|
+
* PouchDB-->>PouchAdapter: BulkGetResponse with documents
|
|
438
|
+
* PouchAdapter->>PouchAdapter: Mark documents as deleted
|
|
439
|
+
* PouchAdapter->>PouchDB: bulkDocs(marked documents)
|
|
440
|
+
* alt Success
|
|
441
|
+
* PouchDB-->>PouchAdapter: Success responses
|
|
442
|
+
* PouchAdapter->>PouchAdapter: Process results
|
|
443
|
+
* PouchAdapter->>PouchAdapter: assignMetadata for each doc
|
|
444
|
+
* PouchAdapter-->>Client: Deleted documents with metadata
|
|
445
|
+
* else Error
|
|
446
|
+
* PouchAdapter->>PouchAdapter: Check for errors
|
|
447
|
+
* PouchAdapter-->>Client: Throws InternalError
|
|
448
|
+
* end
|
|
449
|
+
*/
|
|
147
450
|
async deleteAll(tableName, ids) {
|
|
148
451
|
const results = await this.native.bulkGet({
|
|
149
452
|
docs: ids.map((id) => ({ id: this.generateId(tableName, id) })),
|
|
@@ -163,6 +466,35 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
163
466
|
return accum;
|
|
164
467
|
}, []);
|
|
165
468
|
}
|
|
469
|
+
/**
|
|
470
|
+
* @description Executes a raw Mango query against the database
|
|
471
|
+
* @summary Performs a direct find operation using a Mango query object.
|
|
472
|
+
* This method allows for complex queries beyond the standard CRUD operations.
|
|
473
|
+
* @template V - The return type
|
|
474
|
+
* @param {MangoQuery} rawInput - The Mango query to execute
|
|
475
|
+
* @param {boolean} [process=true] - Whether to process the response (true returns just docs, false returns full response)
|
|
476
|
+
* @return {Promise<V>} A promise that resolves to the query results
|
|
477
|
+
* @mermaid
|
|
478
|
+
* sequenceDiagram
|
|
479
|
+
* participant Client
|
|
480
|
+
* participant PouchAdapter
|
|
481
|
+
* participant PouchDB
|
|
482
|
+
*
|
|
483
|
+
* Client->>PouchAdapter: raw<V>(rawInput, process)
|
|
484
|
+
* PouchAdapter->>PouchDB: find(rawInput)
|
|
485
|
+
* alt Success
|
|
486
|
+
* PouchDB-->>PouchAdapter: FindResponse
|
|
487
|
+
* alt process=true
|
|
488
|
+
* PouchAdapter-->>Client: response.docs as V
|
|
489
|
+
* else process=false
|
|
490
|
+
* PouchAdapter-->>Client: response as V
|
|
491
|
+
* end
|
|
492
|
+
* else Error
|
|
493
|
+
* PouchDB-->>PouchAdapter: Error
|
|
494
|
+
* PouchAdapter->>PouchAdapter: parseError(e)
|
|
495
|
+
* PouchAdapter-->>Client: Throws error
|
|
496
|
+
* end
|
|
497
|
+
*/
|
|
166
498
|
async raw(rawInput, process = true) {
|
|
167
499
|
try {
|
|
168
500
|
const response = await this.native.find(rawInput);
|
|
@@ -176,9 +508,56 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
176
508
|
throw PouchAdapter.parseError(e);
|
|
177
509
|
}
|
|
178
510
|
}
|
|
511
|
+
/**
|
|
512
|
+
* @description Parses and converts errors from PouchDB to application-specific errors
|
|
513
|
+
* @summary Converts PouchDB errors to the application's error hierarchy.
|
|
514
|
+
* This instance method delegates to the static parseError method.
|
|
515
|
+
* @param {Error|string} err - The error object or message to parse
|
|
516
|
+
* @param {string} [reason] - Optional reason for the error
|
|
517
|
+
* @return {BaseError} The converted error object
|
|
518
|
+
*/
|
|
179
519
|
parseError(err, reason) {
|
|
180
520
|
return PouchAdapter.parseError(err, reason);
|
|
181
521
|
}
|
|
522
|
+
/**
|
|
523
|
+
* @description Static method to parse and convert errors from PouchDB to application-specific errors
|
|
524
|
+
* @summary Converts PouchDB errors to the application's error hierarchy based on error codes and messages.
|
|
525
|
+
* This method analyzes the error type, status code, or message to determine the appropriate error class.
|
|
526
|
+
* @param {Error|string} err - The error object or message to parse
|
|
527
|
+
* @param {string} [reason] - Optional reason for the error
|
|
528
|
+
* @return {BaseError} The converted error object
|
|
529
|
+
* @mermaid
|
|
530
|
+
* sequenceDiagram
|
|
531
|
+
* participant Caller
|
|
532
|
+
* participant PouchAdapter
|
|
533
|
+
*
|
|
534
|
+
* Caller->>PouchAdapter: parseError(err, reason)
|
|
535
|
+
* alt err is BaseError
|
|
536
|
+
* PouchAdapter-->>Caller: Return err as is
|
|
537
|
+
* else err is string
|
|
538
|
+
* alt contains "already exist" or "update conflict"
|
|
539
|
+
* PouchAdapter-->>Caller: ConflictError
|
|
540
|
+
* else contains "missing" or "deleted"
|
|
541
|
+
* PouchAdapter-->>Caller: NotFoundError
|
|
542
|
+
* end
|
|
543
|
+
* else err has status
|
|
544
|
+
* alt status is 401, 412, 409
|
|
545
|
+
* PouchAdapter-->>Caller: ConflictError
|
|
546
|
+
* else status is 404
|
|
547
|
+
* PouchAdapter-->>Caller: NotFoundError
|
|
548
|
+
* else status is 400
|
|
549
|
+
* alt message contains "No index exists"
|
|
550
|
+
* PouchAdapter-->>Caller: IndexError
|
|
551
|
+
* else
|
|
552
|
+
* PouchAdapter-->>Caller: InternalError
|
|
553
|
+
* end
|
|
554
|
+
* else message contains "ECONNREFUSED"
|
|
555
|
+
* PouchAdapter-->>Caller: ConnectionError
|
|
556
|
+
* else
|
|
557
|
+
* PouchAdapter-->>Caller: InternalError
|
|
558
|
+
* end
|
|
559
|
+
* end
|
|
560
|
+
*/
|
|
182
561
|
static parseError(err, reason) {
|
|
183
562
|
// return super.parseError(err, reason);
|
|
184
563
|
if (err instanceof BaseError)
|
|
@@ -215,6 +594,34 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
215
594
|
return new InternalError(err);
|
|
216
595
|
}
|
|
217
596
|
}
|
|
597
|
+
/**
|
|
598
|
+
* @description Sets up decorations for PouchDB-specific model properties
|
|
599
|
+
* @summary Configures decorators for createdBy and updatedBy fields in models.
|
|
600
|
+
* This method defines how these fields should be automatically populated during create and update operations.
|
|
601
|
+
* @mermaid
|
|
602
|
+
* sequenceDiagram
|
|
603
|
+
* participant Caller
|
|
604
|
+
* participant PouchAdapter
|
|
605
|
+
* participant Decoration
|
|
606
|
+
*
|
|
607
|
+
* Caller->>PouchAdapter: decoration()
|
|
608
|
+
* PouchAdapter->>Repository: key(PersistenceKeys.CREATED_BY)
|
|
609
|
+
* Repository-->>PouchAdapter: createdByKey
|
|
610
|
+
* PouchAdapter->>Repository: key(PersistenceKeys.UPDATED_BY)
|
|
611
|
+
* Repository-->>PouchAdapter: updatedByKey
|
|
612
|
+
*
|
|
613
|
+
* PouchAdapter->>Decoration: flavouredAs(PouchFlavour)
|
|
614
|
+
* Decoration-->>PouchAdapter: DecoratorBuilder
|
|
615
|
+
* PouchAdapter->>Decoration: for(createdByKey)
|
|
616
|
+
* PouchAdapter->>Decoration: define(onCreate, propMetadata)
|
|
617
|
+
* PouchAdapter->>Decoration: apply()
|
|
618
|
+
*
|
|
619
|
+
* PouchAdapter->>Decoration: flavouredAs(PouchFlavour)
|
|
620
|
+
* Decoration-->>PouchAdapter: DecoratorBuilder
|
|
621
|
+
* PouchAdapter->>Decoration: for(updatedByKey)
|
|
622
|
+
* PouchAdapter->>Decoration: define(onCreate, propMetadata)
|
|
623
|
+
* PouchAdapter->>Decoration: apply()
|
|
624
|
+
*/
|
|
218
625
|
static decoration() {
|
|
219
626
|
const createdByKey = Repository.key(PersistenceKeys.CREATED_BY);
|
|
220
627
|
const updatedByKey = Repository.key(PersistenceKeys.UPDATED_BY);
|
|
@@ -231,17 +638,17 @@ class PouchAdapter extends CouchDBAdapter {
|
|
|
231
638
|
|
|
232
639
|
PouchAdapter.decoration();
|
|
233
640
|
/**
|
|
234
|
-
* @
|
|
235
|
-
* @
|
|
236
|
-
* @module for-
|
|
641
|
+
* @description A TypeScript adapter for PouchDB integration
|
|
642
|
+
* @summary This module provides a repository pattern implementation for PouchDB, allowing for easy database operations with TypeScript type safety. It exports constants, repository classes, types, and adapters for working with PouchDB.
|
|
643
|
+
* @module for-pouch
|
|
237
644
|
*/
|
|
238
645
|
/**
|
|
239
|
-
* @
|
|
240
|
-
* @
|
|
646
|
+
* @description Package version identifier
|
|
647
|
+
* @summary Stores the current version of the for-pouch package
|
|
241
648
|
* @const VERSION
|
|
242
|
-
* @memberOf module:
|
|
649
|
+
* @memberOf module:for-pouch
|
|
243
650
|
*/
|
|
244
|
-
const VERSION = "0.2.
|
|
651
|
+
const VERSION = "0.2.5";
|
|
245
652
|
|
|
246
653
|
export { PouchAdapter, PouchFlavour, VERSION, createdByOnPouchCreateUpdate };
|
|
247
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9yLXBvdWNoLmVzbS5janMiLCJzb3VyY2VzIjpbIi4uL3NyYy9jb25zdGFudHMudHMiLCIuLi9zcmMvYWRhcHRlci50cyIsIi4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgUG91Y2hGbGF2b3VyID0gXCJwb3VjaFwiO1xuIiwiaW1wb3J0IFwicmVmbGVjdC1tZXRhZGF0YVwiO1xuaW1wb3J0IHtcbiAgQ291Y2hEQkFkYXB0ZXIsXG4gIENvdWNoREJLZXlzLFxuICBDcmVhdGVJbmRleFJlcXVlc3QsXG4gIGdlbmVyYXRlSW5kZXhlcyxcbiAgSW5kZXhFcnJvcixcbiAgTWFuZ29RdWVyeSxcbn0gZnJvbSBcIkBkZWNhZi10cy9mb3ItY291Y2hkYlwiO1xuaW1wb3J0IHtcbiAgQmFzZUVycm9yLFxuICBDb25mbGljdEVycm9yLFxuICBDb250ZXh0LFxuICBJbnRlcm5hbEVycm9yLFxuICBOb3RGb3VuZEVycm9yLFxuICBvbkNyZWF0ZSxcbiAgT3BlcmF0aW9uS2V5cyxcbn0gZnJvbSBcIkBkZWNhZi10cy9kYi1kZWNvcmF0b3JzXCI7XG5pbXBvcnQge1xuICBDb25uZWN0aW9uRXJyb3IsXG4gIFBlcnNpc3RlbmNlS2V5cyxcbiAgUmVsYXRpb25zTWV0YWRhdGEsXG4gIFJlcG9zaXRvcnksXG4gIFVuc3VwcG9ydGVkRXJyb3IsXG59IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IERhdGFiYXNlID0gUG91Y2hEQi5EYXRhYmFzZTtcbmltcG9ydCBSZXNwb25zZSA9IFBvdWNoREIuQ29yZS5SZXNwb25zZTtcbmltcG9ydCBFcnIgPSBQb3VjaERCLkNvcmUuRXJyb3I7XG5pbXBvcnQgSWRNZXRhID0gUG91Y2hEQi5Db3JlLklkTWV0YTtcbmltcG9ydCBHZXRNZXRhID0gUG91Y2hEQi5Db3JlLkdldE1ldGE7XG5pbXBvcnQgQ3JlYXRlSW5kZXhSZXNwb25zZSA9IFBvdWNoREIuRmluZC5DcmVhdGVJbmRleFJlc3BvbnNlO1xuaW1wb3J0IHtcbiAgQ29uc3RydWN0b3IsXG4gIERlY29yYXRpb24sXG4gIE1vZGVsLFxuICBwcm9wTWV0YWRhdGEsXG59IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCBCdWxrR2V0UmVzcG9uc2UgPSBQb3VjaERCLkNvcmUuQnVsa0dldFJlc3BvbnNlO1xuaW1wb3J0IEZpbmRSZXNwb25zZSA9IFBvdWNoREIuRmluZC5GaW5kUmVzcG9uc2U7XG5pbXBvcnQgeyBQb3VjaEZsYWdzIH0gZnJvbSBcIi4vdHlwZXNcIjtcbmltcG9ydCB7IFBvdWNoRmxhdm91ciB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgUG91Y2hSZXBvc2l0b3J5IH0gZnJvbSBcIi4vUG91Y2hSZXBvc2l0b3J5XCI7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjcmVhdGVkQnlPblBvdWNoQ3JlYXRlVXBkYXRlPFxuICBNIGV4dGVuZHMgTW9kZWwsXG4gIFIgZXh0ZW5kcyBQb3VjaFJlcG9zaXRvcnk8TT4sXG4gIFYgZXh0ZW5kcyBSZWxhdGlvbnNNZXRhZGF0YSxcbj4oXG4gIHRoaXM6IFIsXG4gIGNvbnRleHQ6IENvbnRleHQ8UG91Y2hGbGFncz4sXG4gIGRhdGE6IFYsXG4gIGtleToga2V5b2YgTSxcbiAgbW9kZWw6IE1cbik6IFByb21pc2U8dm9pZD4ge1xuICB0cnkge1xuICAgIGNvbnN0IHV1aWQ6IHN0cmluZyA9IGNvbnRleHQuZ2V0KFwiVVVJRFwiKTtcbiAgICBtb2RlbFtrZXldID0gdXVpZCBhcyBNW2tleW9mIE1dO1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW51c2VkLXZhcnNcbiAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZEVycm9yKFxuICAgICAgXCJObyBVc2VyIGZvdW5kIGluIGNvbnRleHQuIFBsZWFzZSBwcm92aWRlIGEgdXNlciBpbiB0aGUgY29udGV4dFwiXG4gICAgKTtcbiAgfVxufVxuXG5leHBvcnQgY2xhc3MgUG91Y2hBZGFwdGVyIGV4dGVuZHMgQ291Y2hEQkFkYXB0ZXI8XG4gIERhdGFiYXNlLFxuICBQb3VjaEZsYWdzLFxuICBDb250ZXh0PFBvdWNoRmxhZ3M+XG4+IHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IERhdGFiYXNlLCBhbGlhcz86IHN0cmluZykge1xuICAgIHN1cGVyKHNjb3BlLCBQb3VjaEZsYXZvdXIsIGFsaWFzKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSBmbGFnczxNIGV4dGVuZHMgTW9kZWw+KFxuICAgIG9wZXJhdGlvbjogT3BlcmF0aW9uS2V5cyxcbiAgICBtb2RlbDogQ29uc3RydWN0b3I8TT4sXG4gICAgZmxhZ3M6IFBhcnRpYWw8UG91Y2hGbGFncz5cbiAgKTogUG91Y2hGbGFncyB7XG4gICAgbGV0IGlkOiBzdHJpbmcgPSBcIlwiO1xuICAgIGNvbnN0IHVybCA9ICh0aGlzLm5hdGl2ZSBhcyB1bmtub3duIGFzIHsgbmFtZTogc3RyaW5nIH0pLm5hbWU7XG4gICAgaWYgKHVybCkge1xuICAgICAgY29uc3QgcmVnZXhwID0gL2h0dHBzPzpcXC9cXC8oLis/KTouKz9AL2c7XG4gICAgICBjb25zdCBtID0gcmVnZXhwLmV4ZWModXJsKTtcbiAgICAgIGlmIChtKSBpZCA9IG1bMV07XG4gICAgfVxuICAgIGlmICghaWQpIHtcbiAgICAgIGlkID0gY3J5cHRvLnJhbmRvbVVVSUQoKTtcbiAgICB9XG5cbiAgICByZXR1cm4gT2JqZWN0LmFzc2lnbihzdXBlci5mbGFncyhvcGVyYXRpb24sIG1vZGVsLCBmbGFncyksIHtcbiAgICAgIFVVSUQ6IGlkLFxuICAgIH0pIGFzIFBvdWNoRmxhZ3M7XG4gIH1cblxuICBwcm90ZWN0ZWQgYXN5bmMgaW5kZXg8TSBleHRlbmRzIE1vZGVsPihcbiAgICAuLi5tb2RlbHM6IENvbnN0cnVjdG9yPE0+W11cbiAgKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgaW5kZXhlczogQ3JlYXRlSW5kZXhSZXF1ZXN0W10gPSBnZW5lcmF0ZUluZGV4ZXMobW9kZWxzKTtcbiAgICBmb3IgKGNvbnN0IGluZGV4IG9mIGluZGV4ZXMpIHtcbiAgICAgIGNvbnN0IHJlczogQ3JlYXRlSW5kZXhSZXNwb25zZTxhbnk+ID0gYXdhaXQgdGhpcy5uYXRpdmUuY3JlYXRlSW5kZXgoXG4gICAgICAgIGluZGV4IGFzIGFueVxuICAgICAgKTtcbiAgICAgIGNvbnN0IHsgcmVzdWx0IH0gPSByZXM7XG4gICAgICBpZiAocmVzdWx0ID09PSBcImV4aXN0aW5nXCIpXG4gICAgICAgIHRocm93IG5ldyBDb25mbGljdEVycm9yKGBJbmRleCAke2luZGV4Lm5hbWV9IGFscmVhZHkgZXhpc3RzYCk7XG4gICAgfVxuICB9XG5cbiAgYXN5bmMgY3JlYXRlKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkOiBzdHJpbmcgfCBudW1iZXIsXG4gICAgbW9kZWw6IFJlY29yZDxzdHJpbmcsIGFueT5cbiAgKTogUHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCBhbnk+PiB7XG4gICAgbGV0IHJlc3BvbnNlOiBSZXNwb25zZTtcbiAgICB0cnkge1xuICAgICAgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLm5hdGl2ZS5wdXQobW9kZWwpO1xuICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgIHRocm93IHRoaXMucGFyc2VFcnJvcihlIGFzIEVycm9yKTtcbiAgICB9XG5cbiAgICBpZiAoIXJlc3BvbnNlLm9rKVxuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoXG4gICAgICAgIGBGYWlsZWQgdG8gaW5zZXJ0IGRvYyBpZDogJHtpZH0gaW4gdGFibGUgJHt0YWJsZU5hbWV9YFxuICAgICAgKTtcbiAgICByZXR1cm4gdGhpcy5hc3NpZ25NZXRhZGF0YShtb2RlbCwgcmVzcG9uc2UucmV2KTtcbiAgfVxuXG4gIG92ZXJyaWRlIGFzeW5jIGNyZWF0ZUFsbChcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgICBpZHM6IHN0cmluZ1tdIHwgbnVtYmVyW10sXG4gICAgbW9kZWxzOiBSZWNvcmQ8c3RyaW5nLCBhbnk+W11cbiAgKTogUHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCBhbnk+W10+IHtcbiAgICBsZXQgcmVzcG9uc2U6IFJlc3BvbnNlW10gfCBFcnJbXTtcbiAgICB0cnkge1xuICAgICAgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLm5hdGl2ZS5idWxrRG9jcyhtb2RlbHMpO1xuICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgdGhyb3cgUG91Y2hBZGFwdGVyLnBhcnNlRXJyb3IoZSk7XG4gICAgfVxuICAgIGlmICghcmVzcG9uc2UuZXZlcnkoKHI6IFJlc3BvbnNlIHwgRXJyKSA9PiAociBhcyBSZXNwb25zZSkub2spKSB7XG4gICAgICBjb25zdCBlcnJvcnMgPSByZXNwb25zZS5yZWR1Y2UoKGFjY3VtOiBzdHJpbmdbXSwgZWwsIGkpID0+IHtcbiAgICAgICAgaWYgKGVsLmVycm9yKVxuICAgICAgICAgIGFjY3VtLnB1c2goXG4gICAgICAgICAgICBgZWwgJHtpfTogJHtlbC5lcnJvcn0ke2VsLnJlYXNvbiA/IGAgLSAke2VsLnJlYXNvbn1gIDogXCJcIn1gXG4gICAgICAgICAgKTtcbiAgICAgICAgcmV0dXJuIGFjY3VtO1xuICAgICAgfSwgW10pO1xuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoZXJyb3JzLmpvaW4oXCJcXG5cIikpO1xuICAgIH1cblxuICAgIHJldHVybiB0aGlzLmFzc2lnbk11bHRpcGxlTWV0YWRhdGEoXG4gICAgICBtb2RlbHMsXG4gICAgICByZXNwb25zZS5tYXAoKHIpID0+IHIucmV2IGFzIHN0cmluZylcbiAgICApO1xuICB9XG5cbiAgYXN5bmMgcmVhZChcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgICBpZDogc3RyaW5nIHwgbnVtYmVyXG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55Pj4ge1xuICAgIGNvbnN0IF9pZCA9IHRoaXMuZ2VuZXJhdGVJZCh0YWJsZU5hbWUsIGlkKTtcbiAgICBsZXQgcmVjb3JkOiBJZE1ldGEgJiBHZXRNZXRhO1xuICAgIHRyeSB7XG4gICAgICByZWNvcmQgPSBhd2FpdCB0aGlzLm5hdGl2ZS5nZXQoX2lkKTtcbiAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgIHRocm93IFBvdWNoQWRhcHRlci5wYXJzZUVycm9yKGUpO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5hc3NpZ25NZXRhZGF0YShyZWNvcmQsIHJlY29yZC5fcmV2KTtcbiAgfVxuXG4gIG92ZXJyaWRlIGFzeW5jIHJlYWRBbGwoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWRzOiAoc3RyaW5nIHwgbnVtYmVyIHwgYmlnaW50KVtdXG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55PltdPiB7XG4gICAgY29uc3QgcmVzdWx0czogQnVsa0dldFJlc3BvbnNlPGFueT4gPSBhd2FpdCB0aGlzLm5hdGl2ZS5idWxrR2V0KHtcbiAgICAgIGRvY3M6IGlkcy5tYXAoKGlkKSA9PiAoeyBpZDogdGhpcy5nZW5lcmF0ZUlkKHRhYmxlTmFtZSwgaWQgYXMgYW55KSB9KSksXG4gICAgfSk7XG4gICAgY29uc3QgcmVzID0gcmVzdWx0cy5yZXN1bHRzLnJlZHVjZSgoYWNjdW06IGFueVtdLCByKSA9PiB7XG4gICAgICByLmRvY3MuZm9yRWFjaCgoZCkgPT4ge1xuICAgICAgICBpZiAoKGQgYXMgYW55KS5lcnJvciB8fCAhKGQgYXMgYW55KS5vaylcbiAgICAgICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihcbiAgICAgICAgICAgICgoZCBhcyB7IGVycm9yOiBFcnIgfSkuZXJyb3IgYXMgRXJyb3IpIHx8XG4gICAgICAgICAgICAgIG5ldyBJbnRlcm5hbEVycm9yKFwiTWlzc2luZyB2YWxpZCByZXNwb25zZVwiKVxuICAgICAgICAgICk7XG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IE9iamVjdC5hc3NpZ24oe30sIChkIGFzIHsgb2s6IGFueSB9KS5vayk7XG4gICAgICAgIGFjY3VtLnB1c2godGhpcy5hc3NpZ25NZXRhZGF0YShyZXN1bHQsIChkIGFzIGFueSkub2tbQ291Y2hEQktleXMuUkVWXSkpO1xuICAgICAgfSk7XG4gICAgICByZXR1cm4gYWNjdW07XG4gICAgfSwgW10pO1xuXG4gICAgcmV0dXJuIHJlcztcbiAgfVxuXG4gIG92ZXJyaWRlIGFzeW5jIHVwZGF0ZShcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgICBpZDogc3RyaW5nIHwgbnVtYmVyLFxuICAgIG1vZGVsOiBSZWNvcmQ8c3RyaW5nLCBhbnk+XG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55Pj4ge1xuICAgIGxldCByZXNwb25zZTogUmVzcG9uc2U7XG4gICAgdHJ5IHtcbiAgICAgIHJlc3BvbnNlID0gYXdhaXQgdGhpcy5uYXRpdmUucHV0KG1vZGVsKTtcbiAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgIHRocm93IFBvdWNoQWRhcHRlci5wYXJzZUVycm9yKGUpO1xuICAgIH1cblxuICAgIGlmICghcmVzcG9uc2Uub2spXG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihcbiAgICAgICAgYEZhaWxlZCB0byB1cGRhdGUgZG9jIGlkOiAke2lkfSBpbiB0YWJsZSAke3RhYmxlTmFtZX1gXG4gICAgICApO1xuICAgIHJldHVybiB0aGlzLmFzc2lnbk1ldGFkYXRhKG1vZGVsLCByZXNwb25zZS5yZXYpO1xuICB9XG5cbiAgb3ZlcnJpZGUgYXN5bmMgdXBkYXRlQWxsKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkczogc3RyaW5nW10gfCBudW1iZXJbXSxcbiAgICBtb2RlbHM6IFJlY29yZDxzdHJpbmcsIGFueT5bXVxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT5bXT4ge1xuICAgIGxldCByZXNwb25zZTogKFJlc3BvbnNlIHwgRXJyKVtdO1xuICAgIHRyeSB7XG4gICAgICByZXNwb25zZSA9IGF3YWl0IHRoaXMubmF0aXZlLmJ1bGtEb2NzKG1vZGVscyk7XG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlKTtcbiAgICB9XG4gICAgaWYgKCFyZXNwb25zZS5ldmVyeSgocikgPT4gIShyIGFzIGFueSkuZXJyb3IpKSB7XG4gICAgICBjb25zdCBlcnJvcnMgPSByZXNwb25zZS5yZWR1Y2UoKGFjY3VtOiBzdHJpbmdbXSwgZWwsIGkpID0+IHtcbiAgICAgICAgaWYgKChlbCBhcyBhbnkpLmVycm9yKVxuICAgICAgICAgIGFjY3VtLnB1c2goXG4gICAgICAgICAgICBgZWwgJHtpfTogJHsoZWwgYXMgYW55KS5lcnJvcn0keyhlbCBhcyBhbnkpLnJlYXNvbiA/IGAgLSAkeyhlbCBhcyBhbnkpLnJlYXNvbn1gIDogXCJcIn1gXG4gICAgICAgICAgKTtcbiAgICAgICAgcmV0dXJuIGFjY3VtO1xuICAgICAgfSwgW10pO1xuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoZXJyb3JzLmpvaW4oXCJcXG5cIikpO1xuICAgIH1cblxuICAgIHJldHVybiB0aGlzLmFzc2lnbk11bHRpcGxlTWV0YWRhdGEoXG4gICAgICBtb2RlbHMsXG4gICAgICByZXNwb25zZS5tYXAoKHIpID0+IHIucmV2IGFzIHN0cmluZylcbiAgICApO1xuICB9XG5cbiAgb3ZlcnJpZGUgYXN5bmMgZGVsZXRlKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkOiBzdHJpbmcgfCBudW1iZXJcbiAgKTogUHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCBhbnk+PiB7XG4gICAgY29uc3QgX2lkID0gdGhpcy5nZW5lcmF0ZUlkKHRhYmxlTmFtZSwgaWQpO1xuICAgIGxldCByZWNvcmQ6IElkTWV0YSAmIEdldE1ldGE7XG4gICAgdHJ5IHtcbiAgICAgIHJlY29yZCA9IGF3YWl0IHRoaXMubmF0aXZlLmdldChfaWQpO1xuICAgICAgYXdhaXQgdGhpcy5uYXRpdmUucmVtb3ZlKF9pZCwgcmVjb3JkLl9yZXYpO1xuICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgdGhyb3cgUG91Y2hBZGFwdGVyLnBhcnNlRXJyb3IoZSk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLmFzc2lnbk1ldGFkYXRhKHJlY29yZCwgcmVjb3JkLl9yZXYpO1xuICB9XG5cbiAgb3ZlcnJpZGUgYXN5bmMgZGVsZXRlQWxsKFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICAgIGlkczogKHN0cmluZyB8IG51bWJlciB8IGJpZ2ludClbXVxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT5bXT4ge1xuICAgIGNvbnN0IHJlc3VsdHM6IEJ1bGtHZXRSZXNwb25zZTxhbnk+ID0gYXdhaXQgdGhpcy5uYXRpdmUuYnVsa0dldCh7XG4gICAgICBkb2NzOiBpZHMubWFwKChpZCkgPT4gKHsgaWQ6IHRoaXMuZ2VuZXJhdGVJZCh0YWJsZU5hbWUsIGlkIGFzIGFueSkgfSkpLFxuICAgIH0pO1xuXG4gICAgY29uc3QgZGVsZXRpb246IChSZXNwb25zZSB8IEVycilbXSA9IGF3YWl0IHRoaXMubmF0aXZlLmJ1bGtEb2NzKFxuICAgICAgcmVzdWx0cy5yZXN1bHRzLm1hcCgocikgPT4ge1xuICAgICAgICAociBhcyBhbnkpW0NvdWNoREJLZXlzLkRFTEVURURdID0gdHJ1ZTtcbiAgICAgICAgcmV0dXJuIHI7XG4gICAgICB9KVxuICAgICk7XG5cbiAgICBjb25zdCBlcnJzID0gZGVsZXRpb24uZmlsdGVyKChkKSA9PiAoZCBhcyBhbnkpLmVycm9yKTtcbiAgICBpZiAoZXJycy5sZW5ndGgpIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKGVycnMuam9pbihcIlxcblwiKSk7XG5cbiAgICByZXR1cm4gcmVzdWx0cy5yZXN1bHRzLnJlZHVjZSgoYWNjdW06IGFueVtdLCByKSA9PiB7XG4gICAgICByLmRvY3MuZm9yRWFjaCgoZCkgPT4ge1xuICAgICAgICBjb25zdCByZXN1bHQgPSBPYmplY3QuYXNzaWduKHt9LCAoZCBhcyB7IG9rOiBhbnkgfSkub2spO1xuICAgICAgICBhY2N1bS5wdXNoKHRoaXMuYXNzaWduTWV0YWRhdGEocmVzdWx0LCAoZCBhcyBhbnkpLm9rW0NvdWNoREJLZXlzLlJFVl0pKTtcbiAgICAgIH0pO1xuICAgICAgcmV0dXJuIGFjY3VtO1xuICAgIH0sIFtdKTtcbiAgfVxuXG4gIGFzeW5jIHJhdzxWPihyYXdJbnB1dDogTWFuZ29RdWVyeSwgcHJvY2VzcyA9IHRydWUpOiBQcm9taXNlPFY+IHtcbiAgICB0cnkge1xuICAgICAgY29uc3QgcmVzcG9uc2U6IEZpbmRSZXNwb25zZTxhbnk+ID0gYXdhaXQgdGhpcy5uYXRpdmUuZmluZChcbiAgICAgICAgcmF3SW5wdXQgYXMgYW55XG4gICAgICApO1xuICAgICAgaWYgKHJlc3BvbnNlLndhcm5pbmcpIGNvbnNvbGUud2FybihyZXNwb25zZS53YXJuaW5nKTtcbiAgICAgIGlmIChwcm9jZXNzKSByZXR1cm4gcmVzcG9uc2UuZG9jcyBhcyBWO1xuICAgICAgcmV0dXJuIHJlc3BvbnNlIGFzIFY7XG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlKTtcbiAgICB9XG4gIH1cblxuICBvdmVycmlkZSBwYXJzZUVycm9yKGVycjogRXJyb3IgfCBzdHJpbmcsIHJlYXNvbj86IHN0cmluZyk6IEJhc2VFcnJvciB7XG4gICAgcmV0dXJuIFBvdWNoQWRhcHRlci5wYXJzZUVycm9yKGVyciwgcmVhc29uKTtcbiAgfVxuXG4gIHN0YXRpYyBvdmVycmlkZSBwYXJzZUVycm9yKGVycjogRXJyb3IgfCBzdHJpbmcsIHJlYXNvbj86IHN0cmluZyk6IEJhc2VFcnJvciB7XG4gICAgLy8gcmV0dXJuIHN1cGVyLnBhcnNlRXJyb3IoZXJyLCByZWFzb24pO1xuICAgIGlmIChlcnIgaW5zdGFuY2VvZiBCYXNlRXJyb3IpIHJldHVybiBlcnIgYXMgYW55O1xuICAgIGxldCBjb2RlOiBzdHJpbmcgPSBcIlwiO1xuICAgIGlmICh0eXBlb2YgZXJyID09PSBcInN0cmluZ1wiKSB7XG4gICAgICBjb2RlID0gZXJyO1xuICAgICAgaWYgKGNvZGUubWF0Y2goL2FscmVhZHkgZXhpc3R8dXBkYXRlIGNvbmZsaWN0L2cpKVxuICAgICAgICByZXR1cm4gbmV3IENvbmZsaWN0RXJyb3IoY29kZSk7XG4gICAgICBpZiAoY29kZS5tYXRjaCgvbWlzc2luZ3xkZWxldGVkL2cpKSByZXR1cm4gbmV3IE5vdEZvdW5kRXJyb3IoY29kZSk7XG4gICAgfSBlbHNlIGlmICgoZXJyIGFzIGFueSkuc3RhdHVzKSB7XG4gICAgICBjb2RlID0gKGVyciBhcyBhbnkpLnN0YXR1cztcbiAgICAgIHJlYXNvbiA9IHJlYXNvbiB8fCBlcnIubWVzc2FnZTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29kZSA9IGVyci5tZXNzYWdlO1xuICAgIH1cblxuICAgIHN3aXRjaCAoY29kZS50b1N0cmluZygpKSB7XG4gICAgICBjYXNlIFwiNDAxXCI6XG4gICAgICBjYXNlIFwiNDEyXCI6XG4gICAgICBjYXNlIFwiNDA5XCI6XG4gICAgICAgIHJldHVybiBuZXcgQ29uZmxpY3RFcnJvcihyZWFzb24gYXMgc3RyaW5nKTtcbiAgICAgIGNhc2UgXCI0MDRcIjpcbiAgICAgICAgcmV0dXJuIG5ldyBOb3RGb3VuZEVycm9yKHJlYXNvbiBhcyBzdHJpbmcpO1xuICAgICAgY2FzZSBcIjQwMFwiOlxuICAgICAgICBpZiAoY29kZS50b1N0cmluZygpLm1hdGNoKC9Ob1xcc2luZGV4XFxzZXhpc3RzL2cpKVxuICAgICAgICAgIHJldHVybiBuZXcgSW5kZXhFcnJvcihlcnIpO1xuICAgICAgICByZXR1cm4gbmV3IEludGVybmFsRXJyb3IoZXJyKTtcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIGlmIChjb2RlLnRvU3RyaW5nKCkubWF0Y2goL0VDT05OUkVGVVNFRC9nKSlcbiAgICAgICAgICByZXR1cm4gbmV3IENvbm5lY3Rpb25FcnJvcihlcnIpO1xuICAgICAgICByZXR1cm4gbmV3IEludGVybmFsRXJyb3IoZXJyKTtcbiAgICB9XG4gIH1cblxuICBzdGF0aWMgZGVjb3JhdGlvbigpIHtcbiAgICBjb25zdCBjcmVhdGVkQnlLZXkgPSBSZXBvc2l0b3J5LmtleShQZXJzaXN0ZW5jZUtleXMuQ1JFQVRFRF9CWSk7XG4gICAgY29uc3QgdXBkYXRlZEJ5S2V5ID0gUmVwb3NpdG9yeS5rZXkoUGVyc2lzdGVuY2VLZXlzLlVQREFURURfQlkpO1xuICAgIERlY29yYXRpb24uZmxhdm91cmVkQXMoUG91Y2hGbGF2b3VyKVxuICAgICAgLmZvcihjcmVhdGVkQnlLZXkpXG4gICAgICAuZGVmaW5lKFxuICAgICAgICBvbkNyZWF0ZShjcmVhdGVkQnlPblBvdWNoQ3JlYXRlVXBkYXRlKSxcbiAgICAgICAgcHJvcE1ldGFkYXRhKGNyZWF0ZWRCeUtleSwge30pXG4gICAgICApXG4gICAgICAuYXBwbHkoKTtcbiAgICBEZWNvcmF0aW9uLmZsYXZvdXJlZEFzKFBvdWNoRmxhdm91cilcbiAgICAgIC5mb3IodXBkYXRlZEJ5S2V5KVxuICAgICAgLmRlZmluZShcbiAgICAgICAgb25DcmVhdGUoY3JlYXRlZEJ5T25Qb3VjaENyZWF0ZVVwZGF0ZSksXG4gICAgICAgIHByb3BNZXRhZGF0YSh1cGRhdGVkQnlLZXksIHt9KVxuICAgICAgKVxuICAgICAgLmFwcGx5KCk7XG4gIH1cbn1cbiIsImltcG9ydCB7IFBvdWNoQWRhcHRlciB9IGZyb20gXCIuL2FkYXB0ZXJcIjtcblxuUG91Y2hBZGFwdGVyLmRlY29yYXRpb24oKTtcblxuZXhwb3J0ICogZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9Qb3VjaFJlcG9zaXRvcnlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3R5cGVzXCI7XG4vLyBsZWZ0IHRvIHRoZSBlbmQgb24gcHVycG9zZVxuZXhwb3J0ICogZnJvbSBcIi4vYWRhcHRlclwiO1xuXG4vKipcbiAqIEBzdW1tYXJ5IE1vZHVsZSBzdW1tYXJ5XG4gKiBAZGVzY3JpcHRpb24gTW9kdWxlIGRlc2NyaXB0aW9uXG4gKiBAbW9kdWxlIGZvci1uYW5vXG4gKi9cblxuLyoqXG4gKiBAc3VtbWFyeSBzdG9yZXMgdGhlIGN1cnJlbnQgcGFja2FnZSB2ZXJzaW9uXG4gKiBAZGVzY3JpcHRpb24gdGhpcyBpcyBob3cgeW91IHNob3VsZCBkb2N1bWVudCBhIGNvbnN0YW50XG4gKiBAY29uc3QgVkVSU0lPTlxuICogQG1lbWJlck9mIG1vZHVsZTp0cy13b3Jrc3BhY2VcbiAqL1xuZXhwb3J0IGNvbnN0IFZFUlNJT04gPSBcIiMjVkVSU0lPTiMjXCI7XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQU8sTUFBTSxZQUFZLEdBQUc7O0FDMkNyQixlQUFlLDRCQUE0QixDQU1oRCxPQUE0QixFQUM1QixJQUFPLEVBQ1AsR0FBWSxFQUNaLEtBQVEsRUFBQTtBQUVSLElBQUEsSUFBSTtRQUNGLE1BQU0sSUFBSSxHQUFXLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDO0FBQ3hDLFFBQUEsS0FBSyxDQUFDLEdBQUcsQ0FBQyxHQUFHLElBQWtCOzs7SUFFL0IsT0FBTyxDQUFVLEVBQUU7QUFDbkIsUUFBQSxNQUFNLElBQUksZ0JBQWdCLENBQ3hCLGdFQUFnRSxDQUNqRTs7QUFFTDtBQUVNLE1BQU8sWUFBYSxTQUFRLGNBSWpDLENBQUE7SUFDQyxXQUFZLENBQUEsS0FBZSxFQUFFLEtBQWMsRUFBQTtBQUN6QyxRQUFBLEtBQUssQ0FBQyxLQUFLLEVBQUUsWUFBWSxFQUFFLEtBQUssQ0FBQzs7QUFHaEIsSUFBQSxLQUFLLENBQ3RCLFNBQXdCLEVBQ3hCLEtBQXFCLEVBQ3JCLEtBQTBCLEVBQUE7UUFFMUIsSUFBSSxFQUFFLEdBQVcsRUFBRTtBQUNuQixRQUFBLE1BQU0sR0FBRyxHQUFJLElBQUksQ0FBQyxNQUFzQyxDQUFDLElBQUk7UUFDN0QsSUFBSSxHQUFHLEVBQUU7WUFDUCxNQUFNLE1BQU0sR0FBRyx3QkFBd0I7WUFDdkMsTUFBTSxDQUFDLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLENB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
|
|
654
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9yLXBvdWNoLmVzbS5janMiLCJzb3VyY2VzIjpbIi4uL3NyYy9jb25zdGFudHMudHMiLCIuLi9zcmMvYWRhcHRlci50cyIsIi4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBkZXNjcmlwdGlvbiBJZGVudGlmaWVyIGZvciBQb3VjaERCIGZsYXZvciBpbiB0aGUgZGVjb3JhdG9yIHN5c3RlbVxuICogQHN1bW1hcnkgQSBzdHJpbmcgY29uc3RhbnQgdGhhdCBpZGVudGlmaWVzIHRoZSBQb3VjaERCIGltcGxlbWVudGF0aW9uIGluIHRoZSBkZWNvcmF0b3Igc3lzdGVtLlxuICogVGhpcyBpcyB1c2VkIHRvIHRhcmdldCBkZWNvcmF0b3JzIHNwZWNpZmljYWxseSBmb3IgUG91Y2hEQiBhZGFwdGVycy5cbiAqIEBjb25zdCBQb3VjaEZsYXZvdXJcbiAqIEBtZW1iZXJPZiBtb2R1bGU6Zm9yLXBvdWNoXG4gKi9cbmV4cG9ydCBjb25zdCBQb3VjaEZsYXZvdXIgPSBcInBvdWNoXCI7XG4iLCJpbXBvcnQgXCJyZWZsZWN0LW1ldGFkYXRhXCI7XG5pbXBvcnQge1xuICBDb3VjaERCQWRhcHRlcixcbiAgQ291Y2hEQktleXMsXG4gIENyZWF0ZUluZGV4UmVxdWVzdCxcbiAgZ2VuZXJhdGVJbmRleGVzLFxuICBJbmRleEVycm9yLFxuICBNYW5nb1F1ZXJ5LFxufSBmcm9tIFwiQGRlY2FmLXRzL2Zvci1jb3VjaGRiXCI7XG5pbXBvcnQge1xuICBCYXNlRXJyb3IsXG4gIENvbmZsaWN0RXJyb3IsXG4gIENvbnRleHQsXG4gIEludGVybmFsRXJyb3IsXG4gIE5vdEZvdW5kRXJyb3IsXG4gIG9uQ3JlYXRlLFxuICBPcGVyYXRpb25LZXlzLFxufSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7XG4gIENvbm5lY3Rpb25FcnJvcixcbiAgUGVyc2lzdGVuY2VLZXlzLFxuICBSZWxhdGlvbnNNZXRhZGF0YSxcbiAgUmVwb3NpdG9yeSxcbiAgVW5zdXBwb3J0ZWRFcnJvcixcbn0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgRGF0YWJhc2UgPSBQb3VjaERCLkRhdGFiYXNlO1xuaW1wb3J0IFJlc3BvbnNlID0gUG91Y2hEQi5Db3JlLlJlc3BvbnNlO1xuaW1wb3J0IEVyciA9IFBvdWNoREIuQ29yZS5FcnJvcjtcbmltcG9ydCBJZE1ldGEgPSBQb3VjaERCLkNvcmUuSWRNZXRhO1xuaW1wb3J0IEdldE1ldGEgPSBQb3VjaERCLkNvcmUuR2V0TWV0YTtcbmltcG9ydCBDcmVhdGVJbmRleFJlc3BvbnNlID0gUG91Y2hEQi5GaW5kLkNyZWF0ZUluZGV4UmVzcG9uc2U7XG5pbXBvcnQge1xuICBDb25zdHJ1Y3RvcixcbiAgRGVjb3JhdGlvbixcbiAgTW9kZWwsXG4gIHByb3BNZXRhZGF0YSxcbn0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IEJ1bGtHZXRSZXNwb25zZSA9IFBvdWNoREIuQ29yZS5CdWxrR2V0UmVzcG9uc2U7XG5pbXBvcnQgRmluZFJlc3BvbnNlID0gUG91Y2hEQi5GaW5kLkZpbmRSZXNwb25zZTtcbmltcG9ydCB7IFBvdWNoRmxhZ3MgfSBmcm9tIFwiLi90eXBlc1wiO1xuaW1wb3J0IHsgUG91Y2hGbGF2b3VyIH0gZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5pbXBvcnQgeyBQb3VjaFJlcG9zaXRvcnkgfSBmcm9tIFwiLi9Qb3VjaFJlcG9zaXRvcnlcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gU2V0cyB0aGUgY3JlYXRvciBJRCBvbiBhIG1vZGVsIGR1cmluZyBjcmVhdGlvbiBvciB1cGRhdGUgb3BlcmF0aW9uc1xuICogQHN1bW1hcnkgVGhpcyBmdW5jdGlvbiBpcyB1c2VkIGFzIGEgZGVjb3JhdG9yIGhhbmRsZXIgdG8gYXV0b21hdGljYWxseSBzZXQgdGhlIGNyZWF0b3IgSUQgZmllbGQgb24gYSBtb2RlbFxuICogd2hlbiBpdCdzIGJlaW5nIGNyZWF0ZWQgb3IgdXBkYXRlZC4gSXQgZXh0cmFjdHMgdGhlIFVVSUQgZnJvbSB0aGUgY29udGV4dCBhbmQgYXNzaWducyBpdCB0byB0aGUgc3BlY2lmaWVkIGtleS5cbiAqIEB0ZW1wbGF0ZSBNIC0gVGhlIG1vZGVsIHR5cGUgdGhhdCBleHRlbmRzIE1vZGVsXG4gKiBAdGVtcGxhdGUgUiAtIFRoZSByZXBvc2l0b3J5IHR5cGUgdGhhdCBleHRlbmRzIFBvdWNoUmVwb3NpdG9yeTxNPlxuICogQHRlbXBsYXRlIFYgLSBUaGUgcmVsYXRpb25zIG1ldGFkYXRhIHR5cGUgdGhhdCBleHRlbmRzIFJlbGF0aW9uc01ldGFkYXRhXG4gKiBAcGFyYW0ge1J9IHRoaXMgLSBUaGUgcmVwb3NpdG9yeSBpbnN0YW5jZVxuICogQHBhcmFtIHtDb250ZXh0PFBvdWNoRmxhZ3M+fSBjb250ZXh0IC0gVGhlIG9wZXJhdGlvbiBjb250ZXh0IGNvbnRhaW5pbmcgZmxhZ3NcbiAqIEBwYXJhbSB7Vn0gZGF0YSAtIFRoZSByZWxhdGlvbnMgbWV0YWRhdGFcbiAqIEBwYXJhbSBrZXkgLSBUaGUgcHJvcGVydHkga2V5IHRvIHNldCBvbiB0aGUgbW9kZWxcbiAqIEBwYXJhbSB7TX0gbW9kZWwgLSBUaGUgbW9kZWwgaW5zdGFuY2UgdG8gbW9kaWZ5XG4gKiBAcmV0dXJuIHtQcm9taXNlPHZvaWQ+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB3aGVuIHRoZSBvcGVyYXRpb24gaXMgY29tcGxldGVcbiAqIEBmdW5jdGlvbiBjcmVhdGVkQnlPblBvdWNoQ3JlYXRlVXBkYXRlXG4gKiBAbWVtYmVyT2YgbW9kdWxlOmZvci1wb3VjaFxuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gY3JlYXRlZEJ5T25Qb3VjaENyZWF0ZVVwZGF0ZTxcbiAgTSBleHRlbmRzIE1vZGVsLFxuICBSIGV4dGVuZHMgUG91Y2hSZXBvc2l0b3J5PE0+LFxuICBWIGV4dGVuZHMgUmVsYXRpb25zTWV0YWRhdGEsXG4+KFxuICB0aGlzOiBSLFxuICBjb250ZXh0OiBDb250ZXh0PFBvdWNoRmxhZ3M+LFxuICBkYXRhOiBWLFxuICBrZXk6IGtleW9mIE0sXG4gIG1vZGVsOiBNXG4pOiBQcm9taXNlPHZvaWQ+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCB1dWlkOiBzdHJpbmcgPSBjb250ZXh0LmdldChcIlVVSURcIik7XG4gICAgbW9kZWxba2V5XSA9IHV1aWQgYXMgTVtrZXlvZiBNXTtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRFcnJvcihcbiAgICAgIFwiTm8gVXNlciBmb3VuZCBpbiBjb250ZXh0LiBQbGVhc2UgcHJvdmlkZSBhIHVzZXIgaW4gdGhlIGNvbnRleHRcIlxuICAgICk7XG4gIH1cbn1cblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gUG91Y2hEQiBpbXBsZW1lbnRhdGlvbiBvZiB0aGUgQ291Y2hEQkFkYXB0ZXJcbiAqIEBzdW1tYXJ5IFRoaXMgY2xhc3MgcHJvdmlkZXMgYSBjb25jcmV0ZSBpbXBsZW1lbnRhdGlvbiBvZiB0aGUgQ291Y2hEQkFkYXB0ZXIgZm9yIFBvdWNoREIuXG4gKiBJdCBoYW5kbGVzIGFsbCBkYXRhYmFzZSBvcGVyYXRpb25zIGxpa2UgY3JlYXRlLCByZWFkLCB1cGRhdGUsIGRlbGV0ZSAoQ1JVRCkgZm9yIGJvdGhcbiAqIHNpbmdsZSBkb2N1bWVudHMgYW5kIGJ1bGsgb3BlcmF0aW9ucy4gSXQgYWxzbyBwcm92aWRlcyBtZXRob2RzIGZvciBxdWVyeWluZyBhbmQgaW5kZXhpbmcuXG4gKiBAdGVtcGxhdGUgRGF0YWJhc2UgLSBUaGUgUG91Y2hEQiBkYXRhYmFzZSB0eXBlXG4gKiBAdGVtcGxhdGUgUG91Y2hGbGFncyAtIFRoZSBmbGFncyBzcGVjaWZpYyB0byBQb3VjaERCIG9wZXJhdGlvbnNcbiAqIEB0ZW1wbGF0ZSBDb250ZXh0PFBvdWNoRmxhZ3M+IC0gVGhlIGNvbnRleHQgdHlwZSB3aXRoIFBvdWNoREIgZmxhZ3NcbiAqIEBwYXJhbSB7RGF0YWJhc2V9IHNjb3BlIC0gVGhlIFBvdWNoREIgZGF0YWJhc2UgaW5zdGFuY2VcbiAqIEBwYXJhbSB7c3RyaW5nfSBbYWxpYXNdIC0gT3B0aW9uYWwgYWxpYXMgZm9yIHRoZSBkYXRhYmFzZVxuICogQGNsYXNzIFBvdWNoQWRhcHRlclxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCBQb3VjaERCIGZyb20gJ3BvdWNoZGInO1xuICogaW1wb3J0IHsgUG91Y2hBZGFwdGVyIH0gZnJvbSAnQGRlY2FmLXRzL2Zvci1wb3VjaCc7XG4gKlxuICogLy8gQ3JlYXRlIGEgbmV3IFBvdWNoREIgaW5zdGFuY2VcbiAqIGNvbnN0IGRiID0gbmV3IFBvdWNoREIoJ215LWRhdGFiYXNlJyk7XG4gKlxuICogLy8gQ3JlYXRlIGEgUG91Y2hBZGFwdGVyIHdpdGggdGhlIGRhdGFiYXNlXG4gKiBjb25zdCBhZGFwdGVyID0gbmV3IFBvdWNoQWRhcHRlcihkYik7XG4gKlxuICogLy8gVXNlIHRoZSBhZGFwdGVyIGZvciBkYXRhYmFzZSBvcGVyYXRpb25zXG4gKiBjb25zdCByZXN1bHQgPSBhd2FpdCBhZGFwdGVyLnJlYWQoJ3VzZXJzJywgJ3VzZXItMTIzJyk7XG4gKiBgYGBcbiAqIEBtZXJtYWlkXG4gKiBzZXF1ZW5jZURpYWdyYW1cbiAqICAgcGFydGljaXBhbnQgQ2xpZW50XG4gKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gKiAgIHBhcnRpY2lwYW50IENvdWNoREJcbiAqXG4gKiAgIENsaWVudC0+PlBvdWNoQWRhcHRlcjogbmV3IFBvdWNoQWRhcHRlcihkYilcbiAqICAgUG91Y2hBZGFwdGVyLT4+Q291Y2hEQkFkYXB0ZXI6IHN1cGVyKHNjb3BlLCBQb3VjaEZsYXZvdXIsIGFsaWFzKVxuICpcbiAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiBjcmVhdGUodGFibGUsIGlkLCBtb2RlbClcbiAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogcHV0KG1vZGVsKVxuICogICBQb3VjaERCLT4+Q291Y2hEQjogSFRUUCBQVVRcbiAqICAgQ291Y2hEQi0tPj5Qb3VjaERCOiBSZXNwb25zZVxuICogICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogUmVzcG9uc2VcbiAqICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVXBkYXRlZCBtb2RlbFxuICpcbiAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiByZWFkKHRhYmxlLCBpZClcbiAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogZ2V0KGlkKVxuICogICBQb3VjaERCLT4+Q291Y2hEQjogSFRUUCBHRVRcbiAqICAgQ291Y2hEQi0tPj5Qb3VjaERCOiBEb2N1bWVudFxuICogICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogRG9jdW1lbnRcbiAqICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogTW9kZWxcbiAqL1xuZXhwb3J0IGNsYXNzIFBvdWNoQWRhcHRlciBleHRlbmRzIENvdWNoREJBZGFwdGVyPFxuICBEYXRhYmFzZSxcbiAgUG91Y2hGbGFncyxcbiAgQ29udGV4dDxQb3VjaEZsYWdzPlxuPiB7XG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBEYXRhYmFzZSwgYWxpYXM/OiBzdHJpbmcpIHtcbiAgICBzdXBlcihzY29wZSwgUG91Y2hGbGF2b3VyLCBhbGlhcyk7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEdlbmVyYXRlcyBvcGVyYXRpb24gZmxhZ3MgZm9yIFBvdWNoREIgb3BlcmF0aW9uc1xuICAgKiBAc3VtbWFyeSBDcmVhdGVzIGEgc2V0IG9mIGZsYWdzIGZvciBhIHNwZWNpZmljIG9wZXJhdGlvbiwgaW5jbHVkaW5nIGEgVVVJRCBmb3IgaWRlbnRpZmljYXRpb24uXG4gICAqIFRoaXMgbWV0aG9kIGV4dHJhY3RzIHRoZSB1c2VyIElEIGZyb20gdGhlIGRhdGFiYXNlIFVSTCBvciBnZW5lcmF0ZXMgYSByYW5kb20gVVVJRCBpZiBub3QgYXZhaWxhYmxlLlxuICAgKiBAdGVtcGxhdGUgTSAtIFRoZSBtb2RlbCB0eXBlIHRoYXQgZXh0ZW5kcyBNb2RlbFxuICAgKiBAcGFyYW0ge09wZXJhdGlvbktleXN9IG9wZXJhdGlvbiAtIFRoZSBvcGVyYXRpb24ga2V5IChjcmVhdGUsIHJlYWQsIHVwZGF0ZSwgZGVsZXRlKVxuICAgKiBAcGFyYW0ge0NvbnN0cnVjdG9yPE0+fSBtb2RlbCAtIFRoZSBtb2RlbCBjb25zdHJ1Y3RvclxuICAgKiBAcGFyYW0ge1BhcnRpYWw8UG91Y2hGbGFncz59IGZsYWdzIC0gUGFydGlhbCBmbGFncyB0byBiZSBtZXJnZWRcbiAgICogQHJldHVybiB7UHJvbWlzZTxQb3VjaEZsYWdzPn0gVGhlIGNvbXBsZXRlIHNldCBvZiBmbGFncyBmb3IgdGhlIG9wZXJhdGlvblxuICAgKi9cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIGFzeW5jIGZsYWdzPE0gZXh0ZW5kcyBNb2RlbD4oXG4gICAgb3BlcmF0aW9uOiBPcGVyYXRpb25LZXlzLFxuICAgIG1vZGVsOiBDb25zdHJ1Y3RvcjxNPixcbiAgICBmbGFnczogUGFydGlhbDxQb3VjaEZsYWdzPlxuICApOiBQcm9taXNlPFBvdWNoRmxhZ3M+IHtcbiAgICBsZXQgaWQ6IHN0cmluZyA9IFwiXCI7XG4gICAgY29uc3QgdXJsID0gKHRoaXMubmF0aXZlIGFzIHVua25vd24gYXMgeyBuYW1lOiBzdHJpbmcgfSkubmFtZTtcbiAgICBpZiAodXJsKSB7XG4gICAgICBjb25zdCByZWdleHAgPSAvaHR0cHM/OlxcL1xcLyguKz8pOi4rP0AvZztcbiAgICAgIGNvbnN0IG0gPSByZWdleHAuZXhlYyh1cmwpO1xuICAgICAgaWYgKG0pIGlkID0gbVsxXTtcbiAgICB9XG4gICAgaWYgKCFpZCkge1xuICAgICAgaWQgPSBjcnlwdG8ucmFuZG9tVVVJRCgpO1xuICAgIH1cblxuICAgIHJldHVybiBPYmplY3QuYXNzaWduKGF3YWl0IHN1cGVyLmZsYWdzKG9wZXJhdGlvbiwgbW9kZWwsIGZsYWdzKSwge1xuICAgICAgVVVJRDogaWQsXG4gICAgfSkgYXMgUG91Y2hGbGFncztcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ3JlYXRlcyBkYXRhYmFzZSBpbmRleGVzIGZvciB0aGUgZ2l2ZW4gbW9kZWxzXG4gICAqIEBzdW1tYXJ5IEdlbmVyYXRlcyBhbmQgY3JlYXRlcyBpbmRleGVzIGluIHRoZSBQb3VjaERCIGRhdGFiYXNlIGJhc2VkIG9uIHRoZSBwcm92aWRlZCBtb2RlbCBjb25zdHJ1Y3RvcnMuXG4gICAqIFRoaXMgbWV0aG9kIHVzZXMgdGhlIGdlbmVyYXRlSW5kZXhlcyB1dGlsaXR5IHRvIGNyZWF0ZSBpbmRleCBkZWZpbml0aW9ucyBhbmQgdGhlbiBjcmVhdGVzIHRoZW0gaW4gdGhlIGRhdGFiYXNlLlxuICAgKiBAdGVtcGxhdGUgTSAtIFRoZSBtb2RlbCB0eXBlIHRoYXQgZXh0ZW5kcyBNb2RlbFxuICAgKiBAcGFyYW0gbW9kZWxzIC0gVGhlIG1vZGVsIGNvbnN0cnVjdG9ycyB0byBjcmVhdGUgaW5kZXhlcyBmb3JcbiAgICogQHJldHVybiB7UHJvbWlzZTx2b2lkPn0gQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgd2hlbiBhbGwgaW5kZXhlcyBhcmUgY3JlYXRlZFxuICAgKi9cbiAgcHJvdGVjdGVkIGFzeW5jIGluZGV4PE0gZXh0ZW5kcyBNb2RlbD4oXG4gICAgLi4ubW9kZWxzOiBDb25zdHJ1Y3RvcjxNPltdXG4gICk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IGluZGV4ZXM6IENyZWF0ZUluZGV4UmVxdWVzdFtdID0gZ2VuZXJhdGVJbmRleGVzKG1vZGVscyk7XG4gICAgZm9yIChjb25zdCBpbmRleCBvZiBpbmRleGVzKSB7XG4gICAgICBjb25zdCByZXM6IENyZWF0ZUluZGV4UmVzcG9uc2U8YW55PiA9IGF3YWl0IHRoaXMubmF0aXZlLmNyZWF0ZUluZGV4KFxuICAgICAgICBpbmRleCBhcyBhbnlcbiAgICAgICk7XG4gICAgICBjb25zdCB7IHJlc3VsdCB9ID0gcmVzO1xuICAgICAgaWYgKHJlc3VsdCA9PT0gXCJleGlzdGluZ1wiKVxuICAgICAgICB0aHJvdyBuZXcgQ29uZmxpY3RFcnJvcihgSW5kZXggJHtpbmRleC5uYW1lfSBhbHJlYWR5IGV4aXN0c2ApO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ3JlYXRlcyBhIG5ldyBkb2N1bWVudCBpbiB0aGUgZGF0YWJhc2VcbiAgICogQHN1bW1hcnkgSW5zZXJ0cyBhIG5ldyBkb2N1bWVudCBpbnRvIHRoZSBQb3VjaERCIGRhdGFiYXNlIHVzaW5nIHRoZSBwdXQgb3BlcmF0aW9uLlxuICAgKiBUaGlzIG1ldGhvZCBoYW5kbGVzIGVycm9yIHBhcnNpbmcgYW5kIGVuc3VyZXMgdGhlIG9wZXJhdGlvbiB3YXMgc3VjY2Vzc2Z1bC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IHRhYmxlTmFtZSAtIFRoZSBuYW1lIG9mIHRoZSB0YWJsZS9jb2xsZWN0aW9uXG4gICAqIEBwYXJhbSB7c3RyaW5nfG51bWJlcn0gaWQgLSBUaGUgZG9jdW1lbnQgSURcbiAgICogQHBhcmFtIHtSZWNvcmQ8c3RyaW5nLCBhbnk+fSBtb2RlbCAtIFRoZSBkb2N1bWVudCBkYXRhIHRvIGluc2VydFxuICAgKiBAcmV0dXJuIHtQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT4+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgY3JlYXRlZCBkb2N1bWVudCB3aXRoIG1ldGFkYXRhXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENsaWVudFxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoREJcbiAgICpcbiAgICogICBDbGllbnQtPj5Qb3VjaEFkYXB0ZXI6IGNyZWF0ZSh0YWJsZU5hbWUsIGlkLCBtb2RlbClcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaERCOiBwdXQobW9kZWwpXG4gICAqICAgYWx0IFN1Y2Nlc3NcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBSZXNwb25zZSB3aXRoIG9rPXRydWVcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogYXNzaWduTWV0YWRhdGEobW9kZWwsIHJlc3BvbnNlLnJldilcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IFVwZGF0ZWQgbW9kZWwgd2l0aCBtZXRhZGF0YVxuICAgKiAgIGVsc2UgRXJyb3JcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBFcnJvclxuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBwYXJzZUVycm9yKGUpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBUaHJvd3MgZXJyb3JcbiAgICogICBlbmRcbiAgICovXG4gIGFzeW5jIGNyZWF0ZShcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgICBpZDogc3RyaW5nIHwgbnVtYmVyLFxuICAgIG1vZGVsOiBSZWNvcmQ8c3RyaW5nLCBhbnk+XG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55Pj4ge1xuICAgIGxldCByZXNwb25zZTogUmVzcG9uc2U7XG4gICAgdHJ5IHtcbiAgICAgIHJlc3BvbnNlID0gYXdhaXQgdGhpcy5uYXRpdmUucHV0KG1vZGVsKTtcbiAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICB0aHJvdyB0aGlzLnBhcnNlRXJyb3IoZSBhcyBFcnJvcik7XG4gICAgfVxuXG4gICAgaWYgKCFyZXNwb25zZS5vaylcbiAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKFxuICAgICAgICBgRmFpbGVkIHRvIGluc2VydCBkb2MgaWQ6ICR7aWR9IGluIHRhYmxlICR7dGFibGVOYW1lfWBcbiAgICAgICk7XG4gICAgcmV0dXJuIHRoaXMuYXNzaWduTWV0YWRhdGEobW9kZWwsIHJlc3BvbnNlLnJldik7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENyZWF0ZXMgbXVsdGlwbGUgZG9jdW1lbnRzIGluIHRoZSBkYXRhYmFzZSBpbiBhIHNpbmdsZSBvcGVyYXRpb25cbiAgICogQHN1bW1hcnkgSW5zZXJ0cyBtdWx0aXBsZSBkb2N1bWVudHMgaW50byB0aGUgUG91Y2hEQiBkYXRhYmFzZSB1c2luZyB0aGUgYnVsa0RvY3Mgb3BlcmF0aW9uLlxuICAgKiBUaGlzIG1ldGhvZCBoYW5kbGVzIGVycm9yIHBhcnNpbmcgYW5kIGVuc3VyZXMgYWxsIG9wZXJhdGlvbnMgd2VyZSBzdWNjZXNzZnVsLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gdGFibGVOYW1lIC0gVGhlIG5hbWUgb2YgdGhlIHRhYmxlL2NvbGxlY3Rpb25cbiAgICogQHBhcmFtIHtzdHJpbmdbXXxudW1iZXJbXX0gaWRzIC0gVGhlIGRvY3VtZW50IElEc1xuICAgKiBAcGFyYW0gIG1vZGVscyAtIFRoZSBkb2N1bWVudCBkYXRhIHRvIGluc2VydFxuICAgKiBAcmV0dXJuIEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSBjcmVhdGVkIGRvY3VtZW50cyB3aXRoIG1ldGFkYXRhXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENsaWVudFxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoREJcbiAgICpcbiAgICogICBDbGllbnQtPj5Qb3VjaEFkYXB0ZXI6IGNyZWF0ZUFsbCh0YWJsZU5hbWUsIGlkcywgbW9kZWxzKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IGJ1bGtEb2NzKG1vZGVscylcbiAgICogICBhbHQgU3VjY2Vzc1xuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEFycmF5IG9mIHJlc3BvbnNlcyB3aXRoIG9rPXRydWVcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogYXNzaWduTXVsdGlwbGVNZXRhZGF0YShtb2RlbHMsIHJldnMpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBVcGRhdGVkIG1vZGVscyB3aXRoIG1ldGFkYXRhXG4gICAqICAgZWxzZSBFcnJvclxuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEFycmF5IHdpdGggZXJyb3JzXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IENoZWNrIGZvciBlcnJvcnNcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IFRocm93cyBJbnRlcm5hbEVycm9yXG4gICAqICAgZW5kXG4gICAqL1xuICBvdmVycmlkZSBhc3luYyBjcmVhdGVBbGwoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWRzOiBzdHJpbmdbXSB8IG51bWJlcltdLFxuICAgIG1vZGVsczogUmVjb3JkPHN0cmluZywgYW55PltdXG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55PltdPiB7XG4gICAgbGV0IHJlc3BvbnNlOiBSZXNwb25zZVtdIHwgRXJyW107XG4gICAgdHJ5IHtcbiAgICAgIHJlc3BvbnNlID0gYXdhaXQgdGhpcy5uYXRpdmUuYnVsa0RvY3MobW9kZWxzKTtcbiAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgIHRocm93IFBvdWNoQWRhcHRlci5wYXJzZUVycm9yKGUpO1xuICAgIH1cbiAgICBpZiAoIXJlc3BvbnNlLmV2ZXJ5KChyOiBSZXNwb25zZSB8IEVycikgPT4gKHIgYXMgUmVzcG9uc2UpLm9rKSkge1xuICAgICAgY29uc3QgZXJyb3JzID0gcmVzcG9uc2UucmVkdWNlKChhY2N1bTogc3RyaW5nW10sIGVsLCBpKSA9PiB7XG4gICAgICAgIGlmIChlbC5lcnJvcilcbiAgICAgICAgICBhY2N1bS5wdXNoKFxuICAgICAgICAgICAgYGVsICR7aX06ICR7ZWwuZXJyb3J9JHtlbC5yZWFzb24gPyBgIC0gJHtlbC5yZWFzb259YCA6IFwiXCJ9YFxuICAgICAgICAgICk7XG4gICAgICAgIHJldHVybiBhY2N1bTtcbiAgICAgIH0sIFtdKTtcbiAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKGVycm9ycy5qb2luKFwiXFxuXCIpKTtcbiAgICB9XG5cbiAgICByZXR1cm4gdGhpcy5hc3NpZ25NdWx0aXBsZU1ldGFkYXRhKFxuICAgICAgbW9kZWxzLFxuICAgICAgcmVzcG9uc2UubWFwKChyKSA9PiByLnJldiBhcyBzdHJpbmcpXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmV0cmlldmVzIGEgZG9jdW1lbnQgZnJvbSB0aGUgZGF0YWJhc2UgYnkgSURcbiAgICogQHN1bW1hcnkgRmV0Y2hlcyBhIGRvY3VtZW50IGZyb20gdGhlIFBvdWNoREIgZGF0YWJhc2UgdXNpbmcgdGhlIGdldCBvcGVyYXRpb24uXG4gICAqIFRoaXMgbWV0aG9kIGdlbmVyYXRlcyB0aGUgZG9jdW1lbnQgSUQgYmFzZWQgb24gdGhlIHRhYmxlIG5hbWUgYW5kIElELCB0aGVuIHJldHJpZXZlcyB0aGUgZG9jdW1lbnQuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSB0YWJsZU5hbWUgLSBUaGUgbmFtZSBvZiB0aGUgdGFibGUvY29sbGVjdGlvblxuICAgKiBAcGFyYW0ge3N0cmluZ3xudW1iZXJ9IGlkIC0gVGhlIGRvY3VtZW50IElEXG4gICAqIEByZXR1cm4ge1Byb21pc2U8UmVjb3JkPHN0cmluZywgYW55Pj59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSByZXRyaWV2ZWQgZG9jdW1lbnQgd2l0aCBtZXRhZGF0YVxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gICAqXG4gICAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiByZWFkKHRhYmxlTmFtZSwgaWQpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBnZW5lcmF0ZUlkKHRhYmxlTmFtZSwgaWQpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogZ2V0KF9pZClcbiAgICogICBhbHQgU3VjY2Vzc1xuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IERvY3VtZW50XG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IGFzc2lnbk1ldGFkYXRhKHJlY29yZCwgcmVjb3JkLl9yZXYpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBEb2N1bWVudCB3aXRoIG1ldGFkYXRhXG4gICAqICAgZWxzZSBFcnJvclxuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEVycm9yXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IHBhcnNlRXJyb3IoZSlcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IFRocm93cyBlcnJvclxuICAgKiAgIGVuZFxuICAgKi9cbiAgYXN5bmMgcmVhZChcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgICBpZDogc3RyaW5nIHwgbnVtYmVyXG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55Pj4ge1xuICAgIGNvbnN0IF9pZCA9IHRoaXMuZ2VuZXJhdGVJZCh0YWJsZU5hbWUsIGlkKTtcbiAgICBsZXQgcmVjb3JkOiBJZE1ldGEgJiBHZXRNZXRhO1xuICAgIHRyeSB7XG4gICAgICByZWNvcmQgPSBhd2FpdCB0aGlzLm5hdGl2ZS5nZXQoX2lkKTtcbiAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgIHRocm93IFBvdWNoQWRhcHRlci5wYXJzZUVycm9yKGUpO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5hc3NpZ25NZXRhZGF0YShyZWNvcmQsIHJlY29yZC5fcmV2KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmV0cmlldmVzIG11bHRpcGxlIGRvY3VtZW50cyBmcm9tIHRoZSBkYXRhYmFzZSBieSB0aGVpciBJRHNcbiAgICogQHN1bW1hcnkgRmV0Y2hlcyBtdWx0aXBsZSBkb2N1bWVudHMgZnJvbSB0aGUgUG91Y2hEQiBkYXRhYmFzZSB1c2luZyB0aGUgYnVsa0dldCBvcGVyYXRpb24uXG4gICAqIFRoaXMgbWV0aG9kIGdlbmVyYXRlcyBkb2N1bWVudCBJRHMgYmFzZWQgb24gdGhlIHRhYmxlIG5hbWUgYW5kIElEcywgdGhlbiByZXRyaWV2ZXMgdGhlIGRvY3VtZW50cy5cbiAgICogQHBhcmFtIHtzdHJpbmd9IHRhYmxlTmFtZSAtIFRoZSBuYW1lIG9mIHRoZSB0YWJsZS9jb2xsZWN0aW9uXG4gICAqIEBwYXJhbSB7QXJyYXk8c3RyaW5nfG51bWJlcnxiaWdpbnQ+fSBpZHMgLSBUaGUgZG9jdW1lbnQgSURzXG4gICAqIEByZXR1cm4gQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdGhlIHJldHJpZXZlZCBkb2N1bWVudHMgd2l0aCBtZXRhZGF0YVxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gICAqXG4gICAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiByZWFkQWxsKHRhYmxlTmFtZSwgaWRzKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogTWFwIGlkcyB0byBnZW5lcmF0ZUlkKHRhYmxlTmFtZSwgaWQpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogYnVsa0dldCh7ZG9jc30pXG4gICAqICAgYWx0IFN1Y2Nlc3NcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBCdWxrR2V0UmVzcG9uc2VcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogUHJvY2VzcyByZXN1bHRzXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IGFzc2lnbk1ldGFkYXRhIGZvciBlYWNoIGRvY1xuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogRG9jdW1lbnRzIHdpdGggbWV0YWRhdGFcbiAgICogICBlbHNlIEVycm9yXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IHBhcnNlRXJyb3IoZXJyb3IpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBUaHJvd3MgZXJyb3JcbiAgICogICBlbmRcbiAgICovXG4gIG92ZXJyaWRlIGFzeW5jIHJlYWRBbGwoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWRzOiAoc3RyaW5nIHwgbnVtYmVyIHwgYmlnaW50KVtdXG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55PltdPiB7XG4gICAgY29uc3QgcmVzdWx0czogQnVsa0dldFJlc3BvbnNlPGFueT4gPSBhd2FpdCB0aGlzLm5hdGl2ZS5idWxrR2V0KHtcbiAgICAgIGRvY3M6IGlkcy5tYXAoKGlkKSA9PiAoeyBpZDogdGhpcy5nZW5lcmF0ZUlkKHRhYmxlTmFtZSwgaWQgYXMgYW55KSB9KSksXG4gICAgfSk7XG4gICAgY29uc3QgcmVzID0gcmVzdWx0cy5yZXN1bHRzLnJlZHVjZSgoYWNjdW06IGFueVtdLCByKSA9PiB7XG4gICAgICByLmRvY3MuZm9yRWFjaCgoZCkgPT4ge1xuICAgICAgICBpZiAoKGQgYXMgYW55KS5lcnJvciB8fCAhKGQgYXMgYW55KS5vaylcbiAgICAgICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihcbiAgICAgICAgICAgICgoZCBhcyB7IGVycm9yOiBFcnIgfSkuZXJyb3IgYXMgRXJyb3IpIHx8XG4gICAgICAgICAgICAgIG5ldyBJbnRlcm5hbEVycm9yKFwiTWlzc2luZyB2YWxpZCByZXNwb25zZVwiKVxuICAgICAgICAgICk7XG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IE9iamVjdC5hc3NpZ24oe30sIChkIGFzIHsgb2s6IGFueSB9KS5vayk7XG4gICAgICAgIGFjY3VtLnB1c2godGhpcy5hc3NpZ25NZXRhZGF0YShyZXN1bHQsIChkIGFzIGFueSkub2tbQ291Y2hEQktleXMuUkVWXSkpO1xuICAgICAgfSk7XG4gICAgICByZXR1cm4gYWNjdW07XG4gICAgfSwgW10pO1xuXG4gICAgcmV0dXJuIHJlcztcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVXBkYXRlcyBhbiBleGlzdGluZyBkb2N1bWVudCBpbiB0aGUgZGF0YWJhc2VcbiAgICogQHN1bW1hcnkgVXBkYXRlcyBhIGRvY3VtZW50IGluIHRoZSBQb3VjaERCIGRhdGFiYXNlIHVzaW5nIHRoZSBwdXQgb3BlcmF0aW9uLlxuICAgKiBUaGlzIG1ldGhvZCBoYW5kbGVzIGVycm9yIHBhcnNpbmcgYW5kIGVuc3VyZXMgdGhlIG9wZXJhdGlvbiB3YXMgc3VjY2Vzc2Z1bC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IHRhYmxlTmFtZSAtIFRoZSBuYW1lIG9mIHRoZSB0YWJsZS9jb2xsZWN0aW9uXG4gICAqIEBwYXJhbSB7c3RyaW5nfG51bWJlcn0gaWQgLSBUaGUgZG9jdW1lbnQgSURcbiAgICogQHBhcmFtIHtSZWNvcmQ8c3RyaW5nLCBhbnk+fSBtb2RlbCAtIFRoZSB1cGRhdGVkIGRvY3VtZW50IGRhdGFcbiAgICogQHJldHVybiB7UHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCBhbnk+Pn0gQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdGhlIHVwZGF0ZWQgZG9jdW1lbnQgd2l0aCBtZXRhZGF0YVxuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaEFkYXB0ZXJcbiAgICogICBwYXJ0aWNpcGFudCBQb3VjaERCXG4gICAqXG4gICAqICAgQ2xpZW50LT4+UG91Y2hBZGFwdGVyOiB1cGRhdGUodGFibGVOYW1lLCBpZCwgbW9kZWwpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogcHV0KG1vZGVsKVxuICAgKiAgIGFsdCBTdWNjZXNzXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogUmVzcG9uc2Ugd2l0aCBvaz10cnVlXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IGFzc2lnbk1ldGFkYXRhKG1vZGVsLCByZXNwb25zZS5yZXYpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBVcGRhdGVkIG1vZGVsIHdpdGggbWV0YWRhdGFcbiAgICogICBlbHNlIEVycm9yXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogRXJyb3JcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogcGFyc2VFcnJvcihlKVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVGhyb3dzIGVycm9yXG4gICAqICAgZW5kXG4gICAqL1xuICBvdmVycmlkZSBhc3luYyB1cGRhdGUoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWQ6IHN0cmluZyB8IG51bWJlcixcbiAgICBtb2RlbDogUmVjb3JkPHN0cmluZywgYW55PlxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT4+IHtcbiAgICBsZXQgcmVzcG9uc2U6IFJlc3BvbnNlO1xuICAgIHRyeSB7XG4gICAgICByZXNwb25zZSA9IGF3YWl0IHRoaXMubmF0aXZlLnB1dChtb2RlbCk7XG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlKTtcbiAgICB9XG5cbiAgICBpZiAoIXJlc3BvbnNlLm9rKVxuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoXG4gICAgICAgIGBGYWlsZWQgdG8gdXBkYXRlIGRvYyBpZDogJHtpZH0gaW4gdGFibGUgJHt0YWJsZU5hbWV9YFxuICAgICAgKTtcbiAgICByZXR1cm4gdGhpcy5hc3NpZ25NZXRhZGF0YShtb2RlbCwgcmVzcG9uc2UucmV2KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVXBkYXRlcyBtdWx0aXBsZSBkb2N1bWVudHMgaW4gdGhlIGRhdGFiYXNlIGluIGEgc2luZ2xlIG9wZXJhdGlvblxuICAgKiBAc3VtbWFyeSBVcGRhdGVzIG11bHRpcGxlIGRvY3VtZW50cyBpbiB0aGUgUG91Y2hEQiBkYXRhYmFzZSB1c2luZyB0aGUgYnVsa0RvY3Mgb3BlcmF0aW9uLlxuICAgKiBUaGlzIG1ldGhvZCBoYW5kbGVzIGVycm9yIHBhcnNpbmcgYW5kIGVuc3VyZXMgYWxsIG9wZXJhdGlvbnMgd2VyZSBzdWNjZXNzZnVsLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gdGFibGVOYW1lIC0gVGhlIG5hbWUgb2YgdGhlIHRhYmxlL2NvbGxlY3Rpb25cbiAgICogQHBhcmFtIHtzdHJpbmdbXXxudW1iZXJbXX0gaWRzIC0gVGhlIGRvY3VtZW50IElEc1xuICAgKiBAcGFyYW0gbW9kZWxzIC0gVGhlIHVwZGF0ZWQgZG9jdW1lbnQgZGF0YVxuICAgKiBAcmV0dXJuIEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSB1cGRhdGVkIGRvY3VtZW50cyB3aXRoIG1ldGFkYXRhXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENsaWVudFxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoREJcbiAgICpcbiAgICogICBDbGllbnQtPj5Qb3VjaEFkYXB0ZXI6IHVwZGF0ZUFsbCh0YWJsZU5hbWUsIGlkcywgbW9kZWxzKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IGJ1bGtEb2NzKG1vZGVscylcbiAgICogICBhbHQgU3VjY2Vzc1xuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEFycmF5IG9mIHJlc3BvbnNlcyB3aXRoIG9rPXRydWVcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogYXNzaWduTXVsdGlwbGVNZXRhZGF0YShtb2RlbHMsIHJldnMpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBVcGRhdGVkIG1vZGVscyB3aXRoIG1ldGFkYXRhXG4gICAqICAgZWxzZSBFcnJvclxuICAgKiAgICAgUG91Y2hEQi0tPj5Qb3VjaEFkYXB0ZXI6IEFycmF5IHdpdGggZXJyb3JzXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IENoZWNrIGZvciBlcnJvcnNcbiAgICogICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IFRocm93cyBJbnRlcm5hbEVycm9yXG4gICAqICAgZW5kXG4gICAqL1xuICBvdmVycmlkZSBhc3luYyB1cGRhdGVBbGwoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWRzOiBzdHJpbmdbXSB8IG51bWJlcltdLFxuICAgIG1vZGVsczogUmVjb3JkPHN0cmluZywgYW55PltdXG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgYW55PltdPiB7XG4gICAgbGV0IHJlc3BvbnNlOiAoUmVzcG9uc2UgfCBFcnIpW107XG4gICAgdHJ5IHtcbiAgICAgIHJlc3BvbnNlID0gYXdhaXQgdGhpcy5uYXRpdmUuYnVsa0RvY3MobW9kZWxzKTtcbiAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgIHRocm93IFBvdWNoQWRhcHRlci5wYXJzZUVycm9yKGUpO1xuICAgIH1cbiAgICBpZiAoIXJlc3BvbnNlLmV2ZXJ5KChyKSA9PiAhKHIgYXMgYW55KS5lcnJvcikpIHtcbiAgICAgIGNvbnN0IGVycm9ycyA9IHJlc3BvbnNlLnJlZHVjZSgoYWNjdW06IHN0cmluZ1tdLCBlbCwgaSkgPT4ge1xuICAgICAgICBpZiAoKGVsIGFzIGFueSkuZXJyb3IpXG4gICAgICAgICAgYWNjdW0ucHVzaChcbiAgICAgICAgICAgIGBlbCAke2l9OiAkeyhlbCBhcyBhbnkpLmVycm9yfSR7KGVsIGFzIGFueSkucmVhc29uID8gYCAtICR7KGVsIGFzIGFueSkucmVhc29ufWAgOiBcIlwifWBcbiAgICAgICAgICApO1xuICAgICAgICByZXR1cm4gYWNjdW07XG4gICAgICB9LCBbXSk7XG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihlcnJvcnMuam9pbihcIlxcblwiKSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHRoaXMuYXNzaWduTXVsdGlwbGVNZXRhZGF0YShcbiAgICAgIG1vZGVscyxcbiAgICAgIHJlc3BvbnNlLm1hcCgocikgPT4gci5yZXYgYXMgc3RyaW5nKVxuICAgICk7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIERlbGV0ZXMgYSBkb2N1bWVudCBmcm9tIHRoZSBkYXRhYmFzZSBieSBJRFxuICAgKiBAc3VtbWFyeSBSZW1vdmVzIGEgZG9jdW1lbnQgZnJvbSB0aGUgUG91Y2hEQiBkYXRhYmFzZSB1c2luZyB0aGUgcmVtb3ZlIG9wZXJhdGlvbi5cbiAgICogVGhpcyBtZXRob2QgZmlyc3QgcmV0cmlldmVzIHRoZSBkb2N1bWVudCB0byBnZXQgaXRzIHJldmlzaW9uLCB0aGVuIGRlbGV0ZXMgaXQuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSB0YWJsZU5hbWUgLSBUaGUgbmFtZSBvZiB0aGUgdGFibGUvY29sbGVjdGlvblxuICAgKiBAcGFyYW0ge3N0cmluZ3xudW1iZXJ9IGlkIC0gVGhlIGRvY3VtZW50IElEXG4gICAqIEByZXR1cm4ge1Byb21pc2U8UmVjb3JkPHN0cmluZywgYW55Pj59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSBkZWxldGVkIGRvY3VtZW50IHdpdGggbWV0YWRhdGFcbiAgICogQG1lcm1haWRcbiAgICogc2VxdWVuY2VEaWFncmFtXG4gICAqICAgcGFydGljaXBhbnQgQ2xpZW50XG4gICAqICAgcGFydGljaXBhbnQgUG91Y2hBZGFwdGVyXG4gICAqICAgcGFydGljaXBhbnQgUG91Y2hEQlxuICAgKlxuICAgKiAgIENsaWVudC0+PlBvdWNoQWRhcHRlcjogZGVsZXRlKHRhYmxlTmFtZSwgaWQpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBnZW5lcmF0ZUlkKHRhYmxlTmFtZSwgaWQpXG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogZ2V0KF9pZClcbiAgICogICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogRG9jdW1lbnQgd2l0aCBfcmV2XG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hEQjogcmVtb3ZlKF9pZCwgcmVjb3JkLl9yZXYpXG4gICAqICAgYWx0IFN1Y2Nlc3NcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBTdWNjZXNzIHJlc3BvbnNlXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IGFzc2lnbk1ldGFkYXRhKHJlY29yZCwgcmVjb3JkLl9yZXYpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBEZWxldGVkIGRvY3VtZW50IHdpdGggbWV0YWRhdGFcbiAgICogICBlbHNlIEVycm9yXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogRXJyb3JcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogcGFyc2VFcnJvcihlKVxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVGhyb3dzIGVycm9yXG4gICAqICAgZW5kXG4gICAqL1xuICBvdmVycmlkZSBhc3luYyBkZWxldGUoXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgaWQ6IHN0cmluZyB8IG51bWJlclxuICApOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIGFueT4+IHtcbiAgICBjb25zdCBfaWQgPSB0aGlzLmdlbmVyYXRlSWQodGFibGVOYW1lLCBpZCk7XG4gICAgbGV0IHJlY29yZDogSWRNZXRhICYgR2V0TWV0YTtcbiAgICB0cnkge1xuICAgICAgcmVjb3JkID0gYXdhaXQgdGhpcy5uYXRpdmUuZ2V0KF9pZCk7XG4gICAgICBhd2FpdCB0aGlzLm5hdGl2ZS5yZW1vdmUoX2lkLCByZWNvcmQuX3Jldik7XG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuYXNzaWduTWV0YWRhdGEocmVjb3JkLCByZWNvcmQuX3Jldik7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIERlbGV0ZXMgbXVsdGlwbGUgZG9jdW1lbnRzIGZyb20gdGhlIGRhdGFiYXNlIGJ5IHRoZWlyIElEc1xuICAgKiBAc3VtbWFyeSBSZW1vdmVzIG11bHRpcGxlIGRvY3VtZW50cyBmcm9tIHRoZSBQb3VjaERCIGRhdGFiYXNlIGluIGEgc2luZ2xlIG9wZXJhdGlvbi5cbiAgICogVGhpcyBtZXRob2QgZmlyc3QgcmV0cmlldmVzIGFsbCBkb2N1bWVudHMgdG8gZ2V0IHRoZWlyIHJldmlzaW9ucywgdGhlbiBtYXJrcyB0aGVtIGFzIGRlbGV0ZWQuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSB0YWJsZU5hbWUgLSBUaGUgbmFtZSBvZiB0aGUgdGFibGUvY29sbGVjdGlvblxuICAgKiBAcGFyYW0ge0FycmF5PHN0cmluZ3xudW1iZXJ8YmlnaW50Pn0gaWRzIC0gVGhlIGRvY3VtZW50IElEc1xuICAgKiBAcmV0dXJuIEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSBkZWxldGVkIGRvY3VtZW50cyB3aXRoIG1ldGFkYXRhXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENsaWVudFxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoREJcbiAgICpcbiAgICogICBDbGllbnQtPj5Qb3VjaEFkYXB0ZXI6IGRlbGV0ZUFsbCh0YWJsZU5hbWUsIGlkcylcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IE1hcCBpZHMgdG8gZ2VuZXJhdGVJZCh0YWJsZU5hbWUsIGlkKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlBvdWNoREI6IGJ1bGtHZXQoe2RvY3N9KVxuICAgKiAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBCdWxrR2V0UmVzcG9uc2Ugd2l0aCBkb2N1bWVudHNcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IE1hcmsgZG9jdW1lbnRzIGFzIGRlbGV0ZWRcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaERCOiBidWxrRG9jcyhtYXJrZWQgZG9jdW1lbnRzKVxuICAgKiAgIGFsdCBTdWNjZXNzXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogU3VjY2VzcyByZXNwb25zZXNcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogUHJvY2VzcyByZXN1bHRzXG4gICAqICAgICBQb3VjaEFkYXB0ZXItPj5Qb3VjaEFkYXB0ZXI6IGFzc2lnbk1ldGFkYXRhIGZvciBlYWNoIGRvY1xuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogRGVsZXRlZCBkb2N1bWVudHMgd2l0aCBtZXRhZGF0YVxuICAgKiAgIGVsc2UgRXJyb3JcbiAgICogICAgIFBvdWNoQWRhcHRlci0+PlBvdWNoQWRhcHRlcjogQ2hlY2sgZm9yIGVycm9yc1xuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNsaWVudDogVGhyb3dzIEludGVybmFsRXJyb3JcbiAgICogICBlbmRcbiAgICovXG4gIG92ZXJyaWRlIGFzeW5jIGRlbGV0ZUFsbChcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgICBpZHM6IChzdHJpbmcgfCBudW1iZXIgfCBiaWdpbnQpW11cbiAgKTogUHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCBhbnk+W10+IHtcbiAgICBjb25zdCByZXN1bHRzOiBCdWxrR2V0UmVzcG9uc2U8YW55PiA9IGF3YWl0IHRoaXMubmF0aXZlLmJ1bGtHZXQoe1xuICAgICAgZG9jczogaWRzLm1hcCgoaWQpID0+ICh7IGlkOiB0aGlzLmdlbmVyYXRlSWQodGFibGVOYW1lLCBpZCBhcyBhbnkpIH0pKSxcbiAgICB9KTtcblxuICAgIGNvbnN0IGRlbGV0aW9uOiAoUmVzcG9uc2UgfCBFcnIpW10gPSBhd2FpdCB0aGlzLm5hdGl2ZS5idWxrRG9jcyhcbiAgICAgIHJlc3VsdHMucmVzdWx0cy5tYXAoKHIpID0+IHtcbiAgICAgICAgKHIgYXMgYW55KVtDb3VjaERCS2V5cy5ERUxFVEVEXSA9IHRydWU7XG4gICAgICAgIHJldHVybiByO1xuICAgICAgfSlcbiAgICApO1xuXG4gICAgY29uc3QgZXJycyA9IGRlbGV0aW9uLmZpbHRlcigoZCkgPT4gKGQgYXMgYW55KS5lcnJvcik7XG4gICAgaWYgKGVycnMubGVuZ3RoKSB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihlcnJzLmpvaW4oXCJcXG5cIikpO1xuXG4gICAgcmV0dXJuIHJlc3VsdHMucmVzdWx0cy5yZWR1Y2UoKGFjY3VtOiBhbnlbXSwgcikgPT4ge1xuICAgICAgci5kb2NzLmZvckVhY2goKGQpID0+IHtcbiAgICAgICAgY29uc3QgcmVzdWx0ID0gT2JqZWN0LmFzc2lnbih7fSwgKGQgYXMgeyBvazogYW55IH0pLm9rKTtcbiAgICAgICAgYWNjdW0ucHVzaCh0aGlzLmFzc2lnbk1ldGFkYXRhKHJlc3VsdCwgKGQgYXMgYW55KS5va1tDb3VjaERCS2V5cy5SRVZdKSk7XG4gICAgICB9KTtcbiAgICAgIHJldHVybiBhY2N1bTtcbiAgICB9LCBbXSk7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEV4ZWN1dGVzIGEgcmF3IE1hbmdvIHF1ZXJ5IGFnYWluc3QgdGhlIGRhdGFiYXNlXG4gICAqIEBzdW1tYXJ5IFBlcmZvcm1zIGEgZGlyZWN0IGZpbmQgb3BlcmF0aW9uIHVzaW5nIGEgTWFuZ28gcXVlcnkgb2JqZWN0LlxuICAgKiBUaGlzIG1ldGhvZCBhbGxvd3MgZm9yIGNvbXBsZXggcXVlcmllcyBiZXlvbmQgdGhlIHN0YW5kYXJkIENSVUQgb3BlcmF0aW9ucy5cbiAgICogQHRlbXBsYXRlIFYgLSBUaGUgcmV0dXJuIHR5cGVcbiAgICogQHBhcmFtIHtNYW5nb1F1ZXJ5fSByYXdJbnB1dCAtIFRoZSBNYW5nbyBxdWVyeSB0byBleGVjdXRlXG4gICAqIEBwYXJhbSB7Ym9vbGVhbn0gW3Byb2Nlc3M9dHJ1ZV0gLSBXaGV0aGVyIHRvIHByb2Nlc3MgdGhlIHJlc3BvbnNlICh0cnVlIHJldHVybnMganVzdCBkb2NzLCBmYWxzZSByZXR1cm5zIGZ1bGwgcmVzcG9uc2UpXG4gICAqIEByZXR1cm4ge1Byb21pc2U8Vj59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSBxdWVyeSByZXN1bHRzXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENsaWVudFxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoREJcbiAgICpcbiAgICogICBDbGllbnQtPj5Qb3VjaEFkYXB0ZXI6IHJhdzxWPihyYXdJbnB1dCwgcHJvY2VzcylcbiAgICogICBQb3VjaEFkYXB0ZXItPj5Qb3VjaERCOiBmaW5kKHJhd0lucHV0KVxuICAgKiAgIGFsdCBTdWNjZXNzXG4gICAqICAgICBQb3VjaERCLS0+PlBvdWNoQWRhcHRlcjogRmluZFJlc3BvbnNlXG4gICAqICAgICBhbHQgcHJvY2Vzcz10cnVlXG4gICAqICAgICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IHJlc3BvbnNlLmRvY3MgYXMgVlxuICAgKiAgICAgZWxzZSBwcm9jZXNzPWZhbHNlXG4gICAqICAgICAgIFBvdWNoQWRhcHRlci0tPj5DbGllbnQ6IHJlc3BvbnNlIGFzIFZcbiAgICogICAgIGVuZFxuICAgKiAgIGVsc2UgRXJyb3JcbiAgICogICAgIFBvdWNoREItLT4+UG91Y2hBZGFwdGVyOiBFcnJvclxuICAgKiAgICAgUG91Y2hBZGFwdGVyLT4+UG91Y2hBZGFwdGVyOiBwYXJzZUVycm9yKGUpXG4gICAqICAgICBQb3VjaEFkYXB0ZXItLT4+Q2xpZW50OiBUaHJvd3MgZXJyb3JcbiAgICogICBlbmRcbiAgICovXG4gIGFzeW5jIHJhdzxWPihyYXdJbnB1dDogTWFuZ29RdWVyeSwgcHJvY2VzcyA9IHRydWUpOiBQcm9taXNlPFY+IHtcbiAgICB0cnkge1xuICAgICAgY29uc3QgcmVzcG9uc2U6IEZpbmRSZXNwb25zZTxhbnk+ID0gYXdhaXQgdGhpcy5uYXRpdmUuZmluZChcbiAgICAgICAgcmF3SW5wdXQgYXMgYW55XG4gICAgICApO1xuICAgICAgaWYgKHJlc3BvbnNlLndhcm5pbmcpIGNvbnNvbGUud2FybihyZXNwb25zZS53YXJuaW5nKTtcbiAgICAgIGlmIChwcm9jZXNzKSByZXR1cm4gcmVzcG9uc2UuZG9jcyBhcyBWO1xuICAgICAgcmV0dXJuIHJlc3BvbnNlIGFzIFY7XG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aHJvdyBQb3VjaEFkYXB0ZXIucGFyc2VFcnJvcihlKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFBhcnNlcyBhbmQgY29udmVydHMgZXJyb3JzIGZyb20gUG91Y2hEQiB0byBhcHBsaWNhdGlvbi1zcGVjaWZpYyBlcnJvcnNcbiAgICogQHN1bW1hcnkgQ29udmVydHMgUG91Y2hEQiBlcnJvcnMgdG8gdGhlIGFwcGxpY2F0aW9uJ3MgZXJyb3IgaGllcmFyY2h5LlxuICAgKiBUaGlzIGluc3RhbmNlIG1ldGhvZCBkZWxlZ2F0ZXMgdG8gdGhlIHN0YXRpYyBwYXJzZUVycm9yIG1ldGhvZC5cbiAgICogQHBhcmFtIHtFcnJvcnxzdHJpbmd9IGVyciAtIFRoZSBlcnJvciBvYmplY3Qgb3IgbWVzc2FnZSB0byBwYXJzZVxuICAgKiBAcGFyYW0ge3N0cmluZ30gW3JlYXNvbl0gLSBPcHRpb25hbCByZWFzb24gZm9yIHRoZSBlcnJvclxuICAgKiBAcmV0dXJuIHtCYXNlRXJyb3J9IFRoZSBjb252ZXJ0ZWQgZXJyb3Igb2JqZWN0XG4gICAqL1xuICBvdmVycmlkZSBwYXJzZUVycm9yKGVycjogRXJyb3IgfCBzdHJpbmcsIHJlYXNvbj86IHN0cmluZyk6IEJhc2VFcnJvciB7XG4gICAgcmV0dXJuIFBvdWNoQWRhcHRlci5wYXJzZUVycm9yKGVyciwgcmVhc29uKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gU3RhdGljIG1ldGhvZCB0byBwYXJzZSBhbmQgY29udmVydCBlcnJvcnMgZnJvbSBQb3VjaERCIHRvIGFwcGxpY2F0aW9uLXNwZWNpZmljIGVycm9yc1xuICAgKiBAc3VtbWFyeSBDb252ZXJ0cyBQb3VjaERCIGVycm9ycyB0byB0aGUgYXBwbGljYXRpb24ncyBlcnJvciBoaWVyYXJjaHkgYmFzZWQgb24gZXJyb3IgY29kZXMgYW5kIG1lc3NhZ2VzLlxuICAgKiBUaGlzIG1ldGhvZCBhbmFseXplcyB0aGUgZXJyb3IgdHlwZSwgc3RhdHVzIGNvZGUsIG9yIG1lc3NhZ2UgdG8gZGV0ZXJtaW5lIHRoZSBhcHByb3ByaWF0ZSBlcnJvciBjbGFzcy5cbiAgICogQHBhcmFtIHtFcnJvcnxzdHJpbmd9IGVyciAtIFRoZSBlcnJvciBvYmplY3Qgb3IgbWVzc2FnZSB0byBwYXJzZVxuICAgKiBAcGFyYW0ge3N0cmluZ30gW3JlYXNvbl0gLSBPcHRpb25hbCByZWFzb24gZm9yIHRoZSBlcnJvclxuICAgKiBAcmV0dXJuIHtCYXNlRXJyb3J9IFRoZSBjb252ZXJ0ZWQgZXJyb3Igb2JqZWN0XG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENhbGxlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKlxuICAgKiAgIENhbGxlci0+PlBvdWNoQWRhcHRlcjogcGFyc2VFcnJvcihlcnIsIHJlYXNvbilcbiAgICogICBhbHQgZXJyIGlzIEJhc2VFcnJvclxuICAgKiAgICAgUG91Y2hBZGFwdGVyLS0+PkNhbGxlcjogUmV0dXJuIGVyciBhcyBpc1xuICAgKiAgIGVsc2UgZXJyIGlzIHN0cmluZ1xuICAgKiAgICAgYWx0IGNvbnRhaW5zIFwiYWxyZWFkeSBleGlzdFwiIG9yIFwidXBkYXRlIGNvbmZsaWN0XCJcbiAgICogICAgICAgUG91Y2hBZGFwdGVyLS0+PkNhbGxlcjogQ29uZmxpY3RFcnJvclxuICAgKiAgICAgZWxzZSBjb250YWlucyBcIm1pc3NpbmdcIiBvciBcImRlbGV0ZWRcIlxuICAgKiAgICAgICBQb3VjaEFkYXB0ZXItLT4+Q2FsbGVyOiBOb3RGb3VuZEVycm9yXG4gICAqICAgICBlbmRcbiAgICogICBlbHNlIGVyciBoYXMgc3RhdHVzXG4gICAqICAgICBhbHQgc3RhdHVzIGlzIDQwMSwgNDEyLCA0MDlcbiAgICogICAgICAgUG91Y2hBZGFwdGVyLS0+PkNhbGxlcjogQ29uZmxpY3RFcnJvclxuICAgKiAgICAgZWxzZSBzdGF0dXMgaXMgNDA0XG4gICAqICAgICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IE5vdEZvdW5kRXJyb3JcbiAgICogICAgIGVsc2Ugc3RhdHVzIGlzIDQwMFxuICAgKiAgICAgICBhbHQgbWVzc2FnZSBjb250YWlucyBcIk5vIGluZGV4IGV4aXN0c1wiXG4gICAqICAgICAgICAgUG91Y2hBZGFwdGVyLS0+PkNhbGxlcjogSW5kZXhFcnJvclxuICAgKiAgICAgICBlbHNlXG4gICAqICAgICAgICAgUG91Y2hBZGFwdGVyLS0+PkNhbGxlcjogSW50ZXJuYWxFcnJvclxuICAgKiAgICAgICBlbmRcbiAgICogICAgIGVsc2UgbWVzc2FnZSBjb250YWlucyBcIkVDT05OUkVGVVNFRFwiXG4gICAqICAgICAgIFBvdWNoQWRhcHRlci0tPj5DYWxsZXI6IENvbm5lY3Rpb25FcnJvclxuICAgKiAgICAgZWxzZVxuICAgKiAgICAgICBQb3VjaEFkYXB0ZXItLT4+Q2FsbGVyOiBJbnRlcm5hbEVycm9yXG4gICAqICAgICBlbmRcbiAgICogICBlbmRcbiAgICovXG4gIHN0YXRpYyBvdmVycmlkZSBwYXJzZUVycm9yKGVycjogRXJyb3IgfCBzdHJpbmcsIHJlYXNvbj86IHN0cmluZyk6IEJhc2VFcnJvciB7XG4gICAgLy8gcmV0dXJuIHN1cGVyLnBhcnNlRXJyb3IoZXJyLCByZWFzb24pO1xuICAgIGlmIChlcnIgaW5zdGFuY2VvZiBCYXNlRXJyb3IpIHJldHVybiBlcnIgYXMgYW55O1xuICAgIGxldCBjb2RlOiBzdHJpbmcgPSBcIlwiO1xuICAgIGlmICh0eXBlb2YgZXJyID09PSBcInN0cmluZ1wiKSB7XG4gICAgICBjb2RlID0gZXJyO1xuICAgICAgaWYgKGNvZGUubWF0Y2goL2FscmVhZHkgZXhpc3R8dXBkYXRlIGNvbmZsaWN0L2cpKVxuICAgICAgICByZXR1cm4gbmV3IENvbmZsaWN0RXJyb3IoY29kZSk7XG4gICAgICBpZiAoY29kZS5tYXRjaCgvbWlzc2luZ3xkZWxldGVkL2cpKSByZXR1cm4gbmV3IE5vdEZvdW5kRXJyb3IoY29kZSk7XG4gICAgfSBlbHNlIGlmICgoZXJyIGFzIGFueSkuc3RhdHVzKSB7XG4gICAgICBjb2RlID0gKGVyciBhcyBhbnkpLnN0YXR1cztcbiAgICAgIHJlYXNvbiA9IHJlYXNvbiB8fCBlcnIubWVzc2FnZTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29kZSA9IGVyci5tZXNzYWdlO1xuICAgIH1cblxuICAgIHN3aXRjaCAoY29kZS50b1N0cmluZygpKSB7XG4gICAgICBjYXNlIFwiNDAxXCI6XG4gICAgICBjYXNlIFwiNDEyXCI6XG4gICAgICBjYXNlIFwiNDA5XCI6XG4gICAgICAgIHJldHVybiBuZXcgQ29uZmxpY3RFcnJvcihyZWFzb24gYXMgc3RyaW5nKTtcbiAgICAgIGNhc2UgXCI0MDRcIjpcbiAgICAgICAgcmV0dXJuIG5ldyBOb3RGb3VuZEVycm9yKHJlYXNvbiBhcyBzdHJpbmcpO1xuICAgICAgY2FzZSBcIjQwMFwiOlxuICAgICAgICBpZiAoY29kZS50b1N0cmluZygpLm1hdGNoKC9Ob1xcc2luZGV4XFxzZXhpc3RzL2cpKVxuICAgICAgICAgIHJldHVybiBuZXcgSW5kZXhFcnJvcihlcnIpO1xuICAgICAgICByZXR1cm4gbmV3IEludGVybmFsRXJyb3IoZXJyKTtcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIGlmIChjb2RlLnRvU3RyaW5nKCkubWF0Y2goL0VDT05OUkVGVVNFRC9nKSlcbiAgICAgICAgICByZXR1cm4gbmV3IENvbm5lY3Rpb25FcnJvcihlcnIpO1xuICAgICAgICByZXR1cm4gbmV3IEludGVybmFsRXJyb3IoZXJyKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFNldHMgdXAgZGVjb3JhdGlvbnMgZm9yIFBvdWNoREItc3BlY2lmaWMgbW9kZWwgcHJvcGVydGllc1xuICAgKiBAc3VtbWFyeSBDb25maWd1cmVzIGRlY29yYXRvcnMgZm9yIGNyZWF0ZWRCeSBhbmQgdXBkYXRlZEJ5IGZpZWxkcyBpbiBtb2RlbHMuXG4gICAqIFRoaXMgbWV0aG9kIGRlZmluZXMgaG93IHRoZXNlIGZpZWxkcyBzaG91bGQgYmUgYXV0b21hdGljYWxseSBwb3B1bGF0ZWQgZHVyaW5nIGNyZWF0ZSBhbmQgdXBkYXRlIG9wZXJhdGlvbnMuXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IENhbGxlclxuICAgKiAgIHBhcnRpY2lwYW50IFBvdWNoQWRhcHRlclxuICAgKiAgIHBhcnRpY2lwYW50IERlY29yYXRpb25cbiAgICpcbiAgICogICBDYWxsZXItPj5Qb3VjaEFkYXB0ZXI6IGRlY29yYXRpb24oKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PlJlcG9zaXRvcnk6IGtleShQZXJzaXN0ZW5jZUtleXMuQ1JFQVRFRF9CWSlcbiAgICogICBSZXBvc2l0b3J5LS0+PlBvdWNoQWRhcHRlcjogY3JlYXRlZEJ5S2V5XG4gICAqICAgUG91Y2hBZGFwdGVyLT4+UmVwb3NpdG9yeToga2V5KFBlcnNpc3RlbmNlS2V5cy5VUERBVEVEX0JZKVxuICAgKiAgIFJlcG9zaXRvcnktLT4+UG91Y2hBZGFwdGVyOiB1cGRhdGVkQnlLZXlcbiAgICpcbiAgICogICBQb3VjaEFkYXB0ZXItPj5EZWNvcmF0aW9uOiBmbGF2b3VyZWRBcyhQb3VjaEZsYXZvdXIpXG4gICAqICAgRGVjb3JhdGlvbi0tPj5Qb3VjaEFkYXB0ZXI6IERlY29yYXRvckJ1aWxkZXJcbiAgICogICBQb3VjaEFkYXB0ZXItPj5EZWNvcmF0aW9uOiBmb3IoY3JlYXRlZEJ5S2V5KVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PkRlY29yYXRpb246IGRlZmluZShvbkNyZWF0ZSwgcHJvcE1ldGFkYXRhKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PkRlY29yYXRpb246IGFwcGx5KClcbiAgICpcbiAgICogICBQb3VjaEFkYXB0ZXItPj5EZWNvcmF0aW9uOiBmbGF2b3VyZWRBcyhQb3VjaEZsYXZvdXIpXG4gICAqICAgRGVjb3JhdGlvbi0tPj5Qb3VjaEFkYXB0ZXI6IERlY29yYXRvckJ1aWxkZXJcbiAgICogICBQb3VjaEFkYXB0ZXItPj5EZWNvcmF0aW9uOiBmb3IodXBkYXRlZEJ5S2V5KVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PkRlY29yYXRpb246IGRlZmluZShvbkNyZWF0ZSwgcHJvcE1ldGFkYXRhKVxuICAgKiAgIFBvdWNoQWRhcHRlci0+PkRlY29yYXRpb246IGFwcGx5KClcbiAgICovXG4gIHN0YXRpYyBkZWNvcmF0aW9uKCkge1xuICAgIGNvbnN0IGNyZWF0ZWRCeUtleSA9IFJlcG9zaXRvcnkua2V5KFBlcnNpc3RlbmNlS2V5cy5DUkVBVEVEX0JZKTtcbiAgICBjb25zdCB1cGRhdGVkQnlLZXkgPSBSZXBvc2l0b3J5LmtleShQZXJzaXN0ZW5jZUtleXMuVVBEQVRFRF9CWSk7XG4gICAgRGVjb3JhdGlvbi5mbGF2b3VyZWRBcyhQb3VjaEZsYXZvdXIpXG4gICAgICAuZm9yKGNyZWF0ZWRCeUtleSlcbiAgICAgIC5kZWZpbmUoXG4gICAgICAgIG9uQ3JlYXRlKGNyZWF0ZWRCeU9uUG91Y2hDcmVhdGVVcGRhdGUpLFxuICAgICAgICBwcm9wTWV0YWRhdGEoY3JlYXRlZEJ5S2V5LCB7fSlcbiAgICAgIClcbiAgICAgIC5hcHBseSgpO1xuICAgIERlY29yYXRpb24uZmxhdm91cmVkQXMoUG91Y2hGbGF2b3VyKVxuICAgICAgLmZvcih1cGRhdGVkQnlLZXkpXG4gICAgICAuZGVmaW5lKFxuICAgICAgICBvbkNyZWF0ZShjcmVhdGVkQnlPblBvdWNoQ3JlYXRlVXBkYXRlKSxcbiAgICAgICAgcHJvcE1ldGFkYXRhKHVwZGF0ZWRCeUtleSwge30pXG4gICAgICApXG4gICAgICAuYXBwbHkoKTtcbiAgfVxufVxuIiwiaW1wb3J0IHsgUG91Y2hBZGFwdGVyIH0gZnJvbSBcIi4vYWRhcHRlclwiO1xuXG5Qb3VjaEFkYXB0ZXIuZGVjb3JhdGlvbigpO1xuXG5leHBvcnQgKiBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL1BvdWNoUmVwb3NpdG9yeVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdHlwZXNcIjtcbi8vIGxlZnQgdG8gdGhlIGVuZCBvbiBwdXJwb3NlXG5leHBvcnQgKiBmcm9tIFwiLi9hZGFwdGVyXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEEgVHlwZVNjcmlwdCBhZGFwdGVyIGZvciBQb3VjaERCIGludGVncmF0aW9uXG4gKiBAc3VtbWFyeSBUaGlzIG1vZHVsZSBwcm92aWRlcyBhIHJlcG9zaXRvcnkgcGF0dGVybiBpbXBsZW1lbnRhdGlvbiBmb3IgUG91Y2hEQiwgYWxsb3dpbmcgZm9yIGVhc3kgZGF0YWJhc2Ugb3BlcmF0aW9ucyB3aXRoIFR5cGVTY3JpcHQgdHlwZSBzYWZldHkuIEl0IGV4cG9ydHMgY29uc3RhbnRzLCByZXBvc2l0b3J5IGNsYXNzZXMsIHR5cGVzLCBhbmQgYWRhcHRlcnMgZm9yIHdvcmtpbmcgd2l0aCBQb3VjaERCLlxuICogQG1vZHVsZSBmb3ItcG91Y2hcbiAqL1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBQYWNrYWdlIHZlcnNpb24gaWRlbnRpZmllclxuICogQHN1bW1hcnkgU3RvcmVzIHRoZSBjdXJyZW50IHZlcnNpb24gb2YgdGhlIGZvci1wb3VjaCBwYWNrYWdlXG4gKiBAY29uc3QgVkVSU0lPTlxuICogQG1lbWJlck9mIG1vZHVsZTpmb3ItcG91Y2hcbiAqL1xuZXhwb3J0IGNvbnN0IFZFUlNJT04gPSBcIiMjVkVSU0lPTiMjXCI7XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQUE7Ozs7OztBQU1HO0FBQ0ksTUFBTSxZQUFZLEdBQUc7O0FDb0M1Qjs7Ozs7Ozs7Ozs7Ozs7O0FBZUc7QUFDSSxlQUFlLDRCQUE0QixDQU1oRCxPQUE0QixFQUM1QixJQUFPLEVBQ1AsR0FBWSxFQUNaLEtBQVEsRUFBQTtBQUVSLElBQUEsSUFBSTtRQUNGLE1BQU0sSUFBSSxHQUFXLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDO0FBQ3hDLFFBQUEsS0FBSyxDQUFDLEdBQUcsQ0FBQyxHQUFHLElBQWtCOzs7SUFFL0IsT0FBTyxDQUFVLEVBQUU7QUFDbkIsUUFBQSxNQUFNLElBQUksZ0JBQWdCLENBQ3hCLGdFQUFnRSxDQUNqRTs7QUFFTDtBQUVBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFnREc7QUFDRyxNQUFPLFlBQWEsU0FBUSxjQUlqQyxDQUFBO0lBQ0MsV0FBWSxDQUFBLEtBQWUsRUFBRSxLQUFjLEVBQUE7QUFDekMsUUFBQSxLQUFLLENBQUMsS0FBSyxFQUFFLFlBQVksRUFBRSxLQUFLLENBQUM7O0FBR25DOzs7Ozs7Ozs7QUFTRztBQUNnQixJQUFBLE1BQU0sS0FBSyxDQUM1QixTQUF3QixFQUN4QixLQUFxQixFQUNyQixLQUEwQixFQUFBO1FBRTFCLElBQUksRUFBRSxHQUFXLEVBQUU7QUFDbkIsUUFBQSxNQUFNLEdBQUcsR0FBSSxJQUFJLENBQUMsTUFBc0MsQ0FBQyxJQUFJO1FBQzdELElBQUksR0FBRyxFQUFFO1lBQ1AsTUFBTSxNQUFNLEdBQUcsd0JBQXdCO1lBQ3ZDLE1BQU0sQ0FBQyxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDO0FBQzFCLFlBQUEsSUFBSSxDQUFDO0FBQUUsZ0JBQUEsRUFBRSxHQUFH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
|