@discordeno/gateway 19.0.0-next.fe00a6f → 19.0.0-next.ffdef6c

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.
@@ -0,0 +1,503 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ Object.defineProperty(exports, "createGatewayManager", {
6
+ enumerable: true,
7
+ get: function() {
8
+ return createGatewayManager;
9
+ }
10
+ });
11
+ const _types = require("@discordeno/types");
12
+ const _utils = require("@discordeno/utils");
13
+ const _Shard = /*#__PURE__*/ _interop_require_default(require("./Shard.cjs"));
14
+ const _types1 = require("./types.cjs");
15
+ function _interop_require_default(obj) {
16
+ return obj && obj.__esModule ? obj : {
17
+ default: obj
18
+ };
19
+ }
20
+ function createGatewayManager(options) {
21
+ const connectionOptions = options.connection ?? {
22
+ url: 'wss://gateway.discord.gg',
23
+ shards: 1,
24
+ sessionStartLimit: {
25
+ maxConcurrency: 1,
26
+ remaining: 1000,
27
+ total: 1000,
28
+ resetAfter: 1000 * 60 * 60 * 24
29
+ }
30
+ };
31
+ const gateway = {
32
+ events: options.events ?? {},
33
+ compress: options.compress ?? false,
34
+ transportCompression: options.transportCompression ?? null,
35
+ intents: options.intents ?? 0,
36
+ properties: {
37
+ os: options.properties?.os ?? process.platform,
38
+ browser: options.properties?.browser ?? 'Discordeno',
39
+ device: options.properties?.device ?? 'Discordeno'
40
+ },
41
+ token: options.token,
42
+ url: options.url ?? connectionOptions.url ?? 'wss://gateway.discord.gg',
43
+ version: options.version ?? 10,
44
+ connection: connectionOptions,
45
+ totalShards: options.totalShards ?? connectionOptions.shards ?? 1,
46
+ lastShardId: options.lastShardId ?? (options.totalShards ? options.totalShards - 1 : connectionOptions ? connectionOptions.shards - 1 : 0),
47
+ firstShardId: options.firstShardId ?? 0,
48
+ totalWorkers: options.totalWorkers ?? 4,
49
+ shardsPerWorker: options.shardsPerWorker ?? 25,
50
+ spawnShardDelay: options.spawnShardDelay ?? 5300,
51
+ preferSnakeCase: options.preferSnakeCase ?? false,
52
+ shards: new Map(),
53
+ buckets: new Map(),
54
+ cache: {
55
+ requestMembers: {
56
+ enabled: options.cache?.requestMembers?.enabled ?? false,
57
+ pending: new _utils.Collection()
58
+ }
59
+ },
60
+ logger: options.logger ?? _utils.logger,
61
+ makePresence: options.makePresence ?? (()=>Promise.resolve(undefined)),
62
+ resharding: {
63
+ enabled: options.resharding?.enabled ?? true,
64
+ shardsFullPercentage: options.resharding?.shardsFullPercentage ?? 80,
65
+ checkInterval: options.resharding?.checkInterval ?? 28800000,
66
+ shards: new _utils.Collection(),
67
+ pendingShards: new _utils.Collection(),
68
+ getSessionInfo: options.resharding?.getSessionInfo,
69
+ updateGuildsShardId: options.resharding?.updateGuildsShardId,
70
+ async checkIfReshardingIsNeeded () {
71
+ gateway.logger.debug('[Resharding] Checking if resharding is needed.');
72
+ if (!gateway.resharding.enabled) {
73
+ gateway.logger.debug('[Resharding] Resharding is disabled.');
74
+ return {
75
+ needed: false
76
+ };
77
+ }
78
+ if (!gateway.resharding.getSessionInfo) {
79
+ throw new Error("[Resharding] Resharding is enabled but no 'resharding.getSessionInfo()' is not provided.");
80
+ }
81
+ gateway.logger.debug('[Resharding] Resharding is enabled.');
82
+ const sessionInfo = await gateway.resharding.getSessionInfo();
83
+ gateway.logger.debug(`[Resharding] Session info retrieved: ${JSON.stringify(sessionInfo)}`);
84
+ // Don't have enough identify limits to try resharding
85
+ if (sessionInfo.sessionStartLimit.remaining < sessionInfo.shards) {
86
+ gateway.logger.debug('[Resharding] Not enough session start limits left to reshard.');
87
+ return {
88
+ needed: false,
89
+ info: sessionInfo
90
+ };
91
+ }
92
+ gateway.logger.debug('[Resharding] Able to reshard, checking whether necessary now.');
93
+ // 2500 is the max amount of guilds a single shard can handle
94
+ // 1000 is the amount of guilds discord uses to determine how many shards to recommend.
95
+ // This algo helps check if your bot has grown enough to reshard.
96
+ // While this is imprecise as discord changes the recommended number of shard every 1000 guilds it is good enough
97
+ // The alternative is to store the guild count for each shard and require the Guilds intent for `GUILD_CREATE` and `GUILD_DELETE` events
98
+ const percentage = sessionInfo.shards / (gateway.totalShards * 2500 / 1000) * 100;
99
+ // Less than necessary% being used so do nothing
100
+ if (percentage < gateway.resharding.shardsFullPercentage) {
101
+ gateway.logger.debug('[Resharding] Resharding not needed.');
102
+ return {
103
+ needed: false,
104
+ info: sessionInfo
105
+ };
106
+ }
107
+ gateway.logger.info('[Resharding] Resharding is needed.');
108
+ return {
109
+ needed: true,
110
+ info: sessionInfo
111
+ };
112
+ },
113
+ async reshard (info) {
114
+ gateway.logger.info(`[Resharding] Starting the reshard process. Previous total shards: ${gateway.totalShards}`);
115
+ // Set values on gateway
116
+ gateway.totalShards = info.shards;
117
+ // Handles preparing mid sized bots for LBS
118
+ gateway.totalShards = gateway.calculateTotalShards();
119
+ // Set first shard id if provided in info
120
+ if (typeof info.firstShardId === 'number') gateway.firstShardId = info.firstShardId;
121
+ // Set last shard id if provided in info
122
+ if (typeof info.lastShardId === 'number') gateway.lastShardId = info.lastShardId;
123
+ gateway.logger.info(`[Resharding] Starting the reshard process. New total shards: ${gateway.totalShards}`);
124
+ // Resetting buckets
125
+ gateway.buckets.clear();
126
+ // Refilling buckets with new values
127
+ gateway.prepareBuckets();
128
+ // SPREAD THIS OUT TO DIFFERENT WORKERS TO BEGIN STARTING UP
129
+ gateway.buckets.forEach(async (bucket, bucketId)=>{
130
+ for (const worker of bucket.workers){
131
+ for (const shardId of worker.queue){
132
+ await gateway.resharding.tellWorkerToPrepare(worker.id, shardId, bucketId);
133
+ }
134
+ }
135
+ });
136
+ },
137
+ async tellWorkerToPrepare (workerId, shardId, bucketId) {
138
+ gateway.logger.debug(`[Resharding] Telling worker to prepare. Worker: ${workerId} | Shard: ${shardId} | Bucket: ${bucketId}.`);
139
+ const shard = new _Shard.default({
140
+ id: shardId,
141
+ connection: {
142
+ compress: gateway.compress,
143
+ transportCompression: gateway.transportCompression ?? null,
144
+ intents: gateway.intents,
145
+ properties: gateway.properties,
146
+ token: gateway.token,
147
+ totalShards: gateway.totalShards,
148
+ url: gateway.url,
149
+ version: gateway.version
150
+ },
151
+ // Ignore events until we are ready
152
+ events: {
153
+ async message (_shard, payload) {
154
+ if (payload.t === 'READY') {
155
+ await gateway.resharding.updateGuildsShardId?.(payload.d.guilds.map((g)=>g.id), shardId);
156
+ }
157
+ }
158
+ },
159
+ logger: gateway.logger,
160
+ requestIdentify: async ()=>{
161
+ await gateway.identify(shardId);
162
+ },
163
+ shardIsReady: async ()=>{
164
+ gateway.logger.debug(`[Shard] Shard #${shardId} is ready`);
165
+ await (0, _utils.delay)(gateway.spawnShardDelay);
166
+ gateway.logger.debug(`[Shard] Resolving shard identify request`);
167
+ gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency).identifyRequests.shift()?.();
168
+ },
169
+ makePresence: gateway.makePresence
170
+ });
171
+ if (gateway.preferSnakeCase) {
172
+ shard.forwardToBot = async (payload)=>{
173
+ shard.events?.message?.(shard, payload);
174
+ };
175
+ }
176
+ gateway.resharding.shards.set(shardId, shard);
177
+ const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency);
178
+ if (!bucket) return;
179
+ return await new Promise((resolve)=>{
180
+ // Mark that we are making an identify request so another is not made.
181
+ bucket.identifyRequests.push(resolve);
182
+ gateway.logger.debug(`[Gateway] Identifying Shard #${shardId}.`);
183
+ // This will trigger identify and when READY is received it will resolve the above request.
184
+ shard?.identify().then(async ()=>{
185
+ // Tell the manager that this shard is online
186
+ return await gateway.resharding.shardIsPending(shard);
187
+ });
188
+ });
189
+ },
190
+ async shardIsPending (shard) {
191
+ // Save this in pending at the moment, until all shards are online
192
+ gateway.resharding.pendingShards.set(shard.id, shard);
193
+ gateway.logger.debug(`[Resharding] Shard #${shard.id} is now pending.`);
194
+ // Check if all shards are now online.
195
+ if (gateway.lastShardId - gateway.firstShardId >= gateway.resharding.pendingShards.size) return;
196
+ gateway.logger.info(`[Resharding] All shards are now online.`);
197
+ // New shards start processing events
198
+ for (const shard of gateway.resharding.shards.values()){
199
+ for(const event in options.events){
200
+ shard.events[event] = options.events[event];
201
+ }
202
+ }
203
+ // Old shards stop processing events
204
+ for (const shard of gateway.shards.values()){
205
+ const oldHandler = shard.events.message;
206
+ // Change with spread operator to not affect new shards, as changing anything on shard.events will directly change options.events, which changes new shards' events
207
+ shard.events = {
208
+ ...shard.events,
209
+ message: async function(_, message) {
210
+ // Member checks need to continue but others can stop
211
+ if (message.t === 'GUILD_MEMBERS_CHUNK') {
212
+ oldHandler?.(shard, message);
213
+ }
214
+ }
215
+ };
216
+ }
217
+ gateway.logger.info(`[Resharding] Shutting down old shards.`);
218
+ // Close old shards
219
+ await gateway.shutdown(_types1.ShardSocketCloseCodes.Resharded, 'Resharded!', false);
220
+ gateway.logger.info(`[Resharding] Completed.`);
221
+ // Replace old shards
222
+ gateway.shards = new _utils.Collection(gateway.resharding.shards);
223
+ // Clear our collections and keep only one reference to the shards, the one in gateway.shards
224
+ gateway.resharding.shards.clear();
225
+ gateway.resharding.pendingShards.clear();
226
+ }
227
+ },
228
+ calculateTotalShards () {
229
+ // Bots under 100k servers do not have access to LBS.
230
+ if (gateway.totalShards < 100) {
231
+ gateway.logger.debug(`[Gateway] Calculating total shards: ${gateway.totalShards}`);
232
+ return gateway.totalShards;
233
+ }
234
+ gateway.logger.debug(`[Gateway] Calculating total shards`, gateway.totalShards, gateway.connection.sessionStartLimit.maxConcurrency);
235
+ // Calculate a multiple of `maxConcurrency` which can be used to connect to the gateway.
236
+ return Math.ceil(gateway.totalShards / // If `maxConcurrency` is 1, we can safely use 16 to get `totalShards` to be in a multiple of 16 so that we can prepare bots with 100k servers for LBS.
237
+ (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency)) * (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency);
238
+ },
239
+ calculateWorkerId (shardId) {
240
+ const workerId = Math.min(shardId % gateway.shardsPerWorker, gateway.totalWorkers - 1);
241
+ gateway.logger.debug(`[Gateway] Calculating workerId: Shard: ${shardId} -> Worker: ${workerId} -> Per Worker: ${gateway.shardsPerWorker} -> Total: ${gateway.totalWorkers}`);
242
+ return workerId;
243
+ },
244
+ prepareBuckets () {
245
+ for(let i = 0; i < gateway.connection.sessionStartLimit.maxConcurrency; ++i){
246
+ gateway.logger.debug(`[Gateway] Preparing buckets for concurrency: ${i}`);
247
+ gateway.buckets.set(i, {
248
+ workers: [],
249
+ identifyRequests: []
250
+ });
251
+ }
252
+ // ORGANIZE ALL SHARDS INTO THEIR OWN BUCKETS
253
+ for(let shardId = gateway.firstShardId; shardId <= gateway.lastShardId; ++shardId){
254
+ gateway.logger.debug(`[Gateway] Preparing buckets for shard: ${shardId}`);
255
+ if (shardId >= gateway.totalShards) {
256
+ throw new Error(`Shard (id: ${shardId}) is bigger or equal to the used amount of used shards which is ${gateway.totalShards}`);
257
+ }
258
+ const bucketId = shardId % gateway.connection.sessionStartLimit.maxConcurrency;
259
+ const bucket = gateway.buckets.get(bucketId);
260
+ if (!bucket) {
261
+ throw new Error(`Shard (id: ${shardId}) got assigned to an illegal bucket id: ${bucketId}, expected a bucket id between 0 and ${gateway.connection.sessionStartLimit.maxConcurrency - 1}`);
262
+ }
263
+ // FIND A QUEUE IN THIS BUCKET THAT HAS SPACE
264
+ // const worker = bucket.workers.find((w) => w.queue.length < gateway.shardsPerWorker);
265
+ const workerId = gateway.calculateWorkerId(shardId);
266
+ const worker = bucket.workers.find((w)=>w.id === workerId);
267
+ if (worker) {
268
+ // IF THE QUEUE HAS SPACE JUST ADD IT TO THIS QUEUE
269
+ worker.queue.push(shardId);
270
+ } else {
271
+ bucket.workers.push({
272
+ id: workerId,
273
+ queue: [
274
+ shardId
275
+ ]
276
+ });
277
+ }
278
+ }
279
+ for (const bucket of gateway.buckets.values()){
280
+ for (const worker of bucket.workers.values()){
281
+ worker.queue = worker.queue.sort((a, b)=>a - b);
282
+ }
283
+ }
284
+ },
285
+ async spawnShards () {
286
+ // PREPARES ALL SHARDS IN SPECIFIC BUCKETS
287
+ gateway.prepareBuckets();
288
+ // Prefer concurrency of forEach instead of forof
289
+ await Promise.all([
290
+ ...gateway.buckets.entries()
291
+ ].map(async ([bucketId, bucket])=>{
292
+ for (const worker of bucket.workers){
293
+ for (const shardId of worker.queue){
294
+ await gateway.tellWorkerToIdentify(worker.id, shardId, bucketId);
295
+ }
296
+ }
297
+ }));
298
+ // Check and reshard automatically if auto resharding is enabled.
299
+ if (gateway.resharding.enabled && gateway.resharding.checkInterval !== -1) {
300
+ // It is better to ensure there is always only one
301
+ clearInterval(gateway.resharding.checkIntervalId);
302
+ if (!gateway.resharding.getSessionInfo) {
303
+ gateway.resharding.enabled = false;
304
+ gateway.logger.warn("[Resharding] Resharding is enabled but 'resharding.getSessionInfo()' was not provided. Disabling resharding.");
305
+ return;
306
+ }
307
+ gateway.resharding.checkIntervalId = setInterval(async ()=>{
308
+ const reshardingInfo = await gateway.resharding.checkIfReshardingIsNeeded();
309
+ if (reshardingInfo.needed && reshardingInfo.info) await gateway.resharding.reshard(reshardingInfo.info);
310
+ }, gateway.resharding.checkInterval);
311
+ }
312
+ },
313
+ async shutdown (code, reason, clearReshardingInterval = true) {
314
+ gateway.shards.forEach((shard)=>shard.close(code, reason));
315
+ if (clearReshardingInterval) clearInterval(gateway.resharding.checkIntervalId);
316
+ await (0, _utils.delay)(5000);
317
+ },
318
+ async sendPayload (shardId, payload) {
319
+ const shard = gateway.shards.get(shardId);
320
+ if (!shard) {
321
+ throw new Error(`Shard (id: ${shardId} not found`);
322
+ }
323
+ await shard.send(payload);
324
+ },
325
+ async tellWorkerToIdentify (workerId, shardId, bucketId) {
326
+ gateway.logger.debug(`[Gateway] Tell worker to identify (${workerId}, ${shardId}, ${bucketId})`);
327
+ await gateway.identify(shardId);
328
+ },
329
+ async identify (shardId) {
330
+ let shard = this.shards.get(shardId);
331
+ gateway.logger.debug(`[Gateway] Identifying ${shard ? 'existing' : 'new'} shard (${shardId})`);
332
+ if (!shard) {
333
+ shard = new _Shard.default({
334
+ id: shardId,
335
+ connection: {
336
+ compress: this.compress,
337
+ transportCompression: gateway.transportCompression,
338
+ intents: this.intents,
339
+ properties: this.properties,
340
+ token: this.token,
341
+ totalShards: this.totalShards,
342
+ url: this.url,
343
+ version: this.version
344
+ },
345
+ events: options.events ?? {},
346
+ logger: this.logger,
347
+ requestIdentify: async ()=>{
348
+ await gateway.identify(shardId);
349
+ },
350
+ shardIsReady: async ()=>{
351
+ gateway.logger.debug(`[Shard] Shard #${shardId} is ready`);
352
+ await (0, _utils.delay)(gateway.spawnShardDelay);
353
+ gateway.logger.debug(`[Shard] Resolving shard identify request`);
354
+ gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency).identifyRequests.shift()?.();
355
+ },
356
+ makePresence: gateway.makePresence
357
+ });
358
+ if (this.preferSnakeCase) {
359
+ shard.forwardToBot = async (payload)=>{
360
+ shard.events.message?.(shard, payload);
361
+ };
362
+ }
363
+ this.shards.set(shardId, shard);
364
+ }
365
+ const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency);
366
+ if (!bucket) return;
367
+ return await new Promise((resolve)=>{
368
+ // Mark that we are making an identify request so another is not made.
369
+ bucket.identifyRequests.push(resolve);
370
+ gateway.logger.debug(`[Gateway] Identifying Shard #${shardId}.`);
371
+ // This will trigger identify and when READY is received it will resolve the above request.
372
+ shard?.identify();
373
+ });
374
+ },
375
+ async kill (shardId) {
376
+ const shard = this.shards.get(shardId);
377
+ if (!shard) {
378
+ return gateway.logger.debug(`[Gateway] A kill for Shard #${shardId} was requested, but the shard could not be found`);
379
+ }
380
+ gateway.logger.debug(`[Gateway] Killing Shard #${shardId}`);
381
+ this.shards.delete(shardId);
382
+ await shard.shutdown();
383
+ },
384
+ async requestIdentify (_shardId) {
385
+ gateway.logger.debug(`[Gateway] Requesting identify`);
386
+ },
387
+ // Helpers methods below this
388
+ calculateShardId (guildId, totalShards) {
389
+ // If none is provided, use the total shards number from gateway object.
390
+ if (!totalShards) totalShards = gateway.totalShards;
391
+ // If it is only 1 shard, it will always be shard id 0
392
+ if (totalShards === 1) {
393
+ gateway.logger.debug(`[Gateway] calculateShardId (1 shard)`);
394
+ return 0;
395
+ }
396
+ gateway.logger.debug(`[Gateway] calculateShardId (guildId: ${guildId}, totalShards: ${totalShards})`);
397
+ return Number((BigInt(guildId) >> 22n) % BigInt(totalShards));
398
+ },
399
+ async joinVoiceChannel (guildId, channelId, options) {
400
+ const shardId = gateway.calculateShardId(guildId);
401
+ gateway.logger.debug(`[Gateway] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`);
402
+ await gateway.sendPayload(shardId, {
403
+ op: _types.GatewayOpcodes.VoiceStateUpdate,
404
+ d: {
405
+ guild_id: guildId.toString(),
406
+ channel_id: channelId.toString(),
407
+ self_mute: options?.selfMute ?? false,
408
+ self_deaf: options?.selfDeaf ?? true
409
+ }
410
+ });
411
+ },
412
+ async editBotStatus (data) {
413
+ gateway.logger.debug(`[Gateway] editBotStatus data: ${JSON.stringify(data)}`);
414
+ await Promise.all([
415
+ ...gateway.shards.values()
416
+ ].map(async (shard)=>{
417
+ gateway.editShardStatus(shard.id, data);
418
+ }));
419
+ },
420
+ async editShardStatus (shardId, data) {
421
+ gateway.logger.debug(`[Gateway] editShardStatus shardId: ${shardId} -> data: ${JSON.stringify(data)}`);
422
+ await gateway.sendPayload(shardId, {
423
+ op: _types.GatewayOpcodes.PresenceUpdate,
424
+ d: {
425
+ since: null,
426
+ afk: false,
427
+ activities: data.activities,
428
+ status: data.status
429
+ }
430
+ });
431
+ },
432
+ async requestMembers (guildId, options) {
433
+ const shardId = gateway.calculateShardId(guildId);
434
+ if (gateway.intents && (!options?.limit || options.limit > 1) && !(gateway.intents & _types.GatewayIntents.GuildMembers)) throw new Error('Cannot fetch more then 1 member without the GUILD_MEMBERS intent');
435
+ gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> data: ${JSON.stringify(options)}`);
436
+ if (options?.userIds?.length) {
437
+ gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> setting user limit based on userIds length: ${options.userIds.length}`);
438
+ options.limit = options.userIds.length;
439
+ }
440
+ const members = !gateway.cache.requestMembers.enabled || !options?.nonce ? [] : new Promise((resolve, reject)=>{
441
+ // Should never happen.
442
+ if (!gateway.cache.requestMembers.enabled || !options?.nonce) {
443
+ reject(new Error("Can't request the members without the nonce or with the feature disabled."));
444
+ return;
445
+ }
446
+ gateway.cache.requestMembers.pending.set(options.nonce, {
447
+ nonce: options.nonce,
448
+ resolve,
449
+ members: []
450
+ });
451
+ });
452
+ await gateway.sendPayload(shardId, {
453
+ op: _types.GatewayOpcodes.RequestGuildMembers,
454
+ d: {
455
+ guild_id: guildId.toString(),
456
+ // If a query is provided use it, OR if a limit is NOT provided use ""
457
+ query: options?.query ?? (options?.limit ? undefined : ''),
458
+ limit: options?.limit ?? 0,
459
+ presences: options?.presences ?? false,
460
+ user_ids: options?.userIds?.map((id)=>id.toString()),
461
+ nonce: options?.nonce
462
+ }
463
+ });
464
+ return await members;
465
+ },
466
+ async leaveVoiceChannel (guildId) {
467
+ const shardId = gateway.calculateShardId(guildId);
468
+ gateway.logger.debug(`[Gateway] leaveVoiceChannel guildId: ${guildId} Shard ${shardId}`);
469
+ await gateway.sendPayload(shardId, {
470
+ op: _types.GatewayOpcodes.VoiceStateUpdate,
471
+ d: {
472
+ guild_id: guildId.toString(),
473
+ channel_id: null,
474
+ self_mute: false,
475
+ self_deaf: false
476
+ }
477
+ });
478
+ },
479
+ async requestSoundboardSounds (guildIds) {
480
+ /**
481
+ * Discord will send the events for the guilds that are "under the shard" that sends the opcode.
482
+ * For this reason we need to group the ids with the shard the calculateShardId method gives
483
+ */ const map = new Map();
484
+ for (const guildId of guildIds){
485
+ const shardId = gateway.calculateShardId(guildId);
486
+ const ids = map.get(shardId) ?? [];
487
+ map.set(shardId, ids);
488
+ ids.push(guildId);
489
+ }
490
+ await Promise.all([
491
+ ...map.entries()
492
+ ].map(([shardId, ids])=>gateway.sendPayload(shardId, {
493
+ op: _types.GatewayOpcodes.RequestSoundboardSounds,
494
+ d: {
495
+ guild_ids: ids
496
+ }
497
+ })));
498
+ }
499
+ };
500
+ return gateway;
501
+ }
502
+
503
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9tYW5hZ2VyLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIHR5cGUgQXRMZWFzdE9uZSxcbiAgdHlwZSBCaWdTdHJpbmcsXG4gIHR5cGUgQ2FtZWxpemUsXG4gIHR5cGUgRGlzY29yZEdldEdhdGV3YXlCb3QsXG4gIHR5cGUgRGlzY29yZE1lbWJlcldpdGhVc2VyLFxuICB0eXBlIERpc2NvcmRSZWFkeSxcbiAgR2F0ZXdheUludGVudHMsXG4gIEdhdGV3YXlPcGNvZGVzLFxuICB0eXBlIFJlcXVlc3RHdWlsZE1lbWJlcnMsXG59IGZyb20gJ0BkaXNjb3JkZW5vL3R5cGVzJ1xuaW1wb3J0IHsgQ29sbGVjdGlvbiwgZGVsYXksIGxvZ2dlciB9IGZyb20gJ0BkaXNjb3JkZW5vL3V0aWxzJ1xuaW1wb3J0IFNoYXJkIGZyb20gJy4vU2hhcmQuanMnXG5pbXBvcnQge1xuICB0eXBlIEJvdFN0YXR1c1VwZGF0ZSxcbiAgdHlwZSBTaGFyZEV2ZW50cyxcbiAgU2hhcmRTb2NrZXRDbG9zZUNvZGVzLFxuICB0eXBlIFNoYXJkU29ja2V0UmVxdWVzdCxcbiAgdHlwZSBTdGF0dXNVcGRhdGUsXG4gIHR5cGUgVHJhbnNwb3J0Q29tcHJlc3Npb24sXG4gIHR5cGUgVXBkYXRlVm9pY2VTdGF0ZSxcbn0gZnJvbSAnLi90eXBlcy5qcydcblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUdhdGV3YXlNYW5hZ2VyKG9wdGlvbnM6IENyZWF0ZUdhdGV3YXlNYW5hZ2VyT3B0aW9ucyk6IEdhdGV3YXlNYW5hZ2VyIHtcbiAgY29uc3QgY29ubmVjdGlvbk9wdGlvbnMgPSBvcHRpb25zLmNvbm5lY3Rpb24gPz8ge1xuICAgIHVybDogJ3dzczovL2dhdGV3YXkuZGlzY29yZC5nZycsXG4gICAgc2hhcmRzOiAxLFxuICAgIHNlc3Npb25TdGFydExpbWl0OiB7XG4gICAgICBtYXhDb25jdXJyZW5jeTogMSxcbiAgICAgIHJlbWFpbmluZzogMTAwMCxcbiAgICAgIHRvdGFsOiAxMDAwLFxuICAgICAgcmVzZXRBZnRlcjogMTAwMCAqIDYwICogNjAgKiAyNCxcbiAgICB9LFxuICB9XG5cbiAgY29uc3QgZ2F0ZXdheTogR2F0ZXdheU1hbmFnZXIgPSB7XG4gICAgZXZlbnRzOiBvcHRpb25zLmV2ZW50cyA/PyB7fSxcbiAgICBjb21wcmVzczogb3B0aW9ucy5jb21wcmVzcyA/PyBmYWxzZSxcbiAgICB0cmFuc3BvcnRDb21wcmVzc2lvbjogb3B0aW9ucy50cmFuc3BvcnRDb21wcmVzc2lvbiA/PyBudWxsLFxuICAgIGludGVudHM6IG9wdGlvbnMuaW50ZW50cyA/PyAwLFxuICAgIHByb3BlcnRpZXM6IHtcbiAgICAgIG9zOiBvcHRpb25zLnByb3BlcnRpZXM/Lm9zID8/IHByb2Nlc3MucGxhdGZvcm0sXG4gICAgICBicm93c2VyOiBvcHRpb25zLnByb3BlcnRpZXM/LmJyb3dzZXIgPz8gJ0Rpc2NvcmRlbm8nLFxuICAgICAgZGV2aWNlOiBvcHRpb25zLnByb3BlcnRpZXM/LmRldmljZSA/PyAnRGlzY29yZGVubycsXG4gICAgfSxcbiAgICB0b2tlbjogb3B0aW9ucy50b2tlbixcbiAgICB1cmw6IG9wdGlvbnMudXJsID8/IGNvbm5lY3Rpb25PcHRpb25zLnVybCA/PyAnd3NzOi8vZ2F0ZXdheS5kaXNjb3JkLmdnJyxcbiAgICB2ZXJzaW9uOiBvcHRpb25zLnZlcnNpb24gPz8gMTAsXG4gICAgY29ubmVjdGlvbjogY29ubmVjdGlvbk9wdGlvbnMsXG4gICAgdG90YWxTaGFyZHM6IG9wdGlvbnMudG90YWxTaGFyZHMgPz8gY29ubmVjdGlvbk9wdGlvbnMuc2hhcmRzID8/IDEsXG4gICAgbGFzdFNoYXJkSWQ6IG9wdGlvbnMubGFzdFNoYXJkSWQgPz8gKG9wdGlvbnMudG90YWxTaGFyZHMgPyBvcHRpb25zLnRvdGFsU2hhcmRzIC0gMSA6IGNvbm5lY3Rpb25PcHRpb25zID8gY29ubmVjdGlvbk9wdGlvbnMuc2hhcmRzIC0gMSA6IDApLFxuICAgIGZpcnN0U2hhcmRJZDogb3B0aW9ucy5maXJzdFNoYXJkSWQgPz8gMCxcbiAgICB0b3RhbFdvcmtlcnM6IG9wdGlvbnMudG90YWxXb3JrZXJzID8/IDQsXG4gICAgc2hhcmRzUGVyV29ya2VyOiBvcHRpb25zLnNoYXJkc1BlcldvcmtlciA/PyAyNSxcbiAgICBzcGF3blNoYXJkRGVsYXk6IG9wdGlvbnMuc3Bhd25TaGFyZERlbGF5ID8/IDUzMDAsXG4gICAgcHJlZmVyU25ha2VDYXNlOiBvcHRpb25zLnByZWZlclNuYWtlQ2FzZSA/PyBmYWxzZSxcbiAgICBzaGFyZHM6IG5ldyBNYXAoKSxcbiAgICBidWNrZXRzOiBuZXcgTWFwKCksXG4gICAgY2FjaGU6IHtcbiAgICAgIHJlcXVlc3RNZW1iZXJzOiB7XG4gICAgICAgIGVuYWJsZWQ6IG9wdGlvbnMuY2FjaGU/LnJlcXVlc3RNZW1iZXJzPy5lbmFibGVkID8/IGZhbHNlLFxuICAgICAgICBwZW5kaW5nOiBuZXcgQ29sbGVjdGlvbigpLFxuICAgICAgfSxcbiAgICB9LFxuICAgIGxvZ2dlcjogb3B0aW9ucy5sb2dnZXIgPz8gbG9nZ2VyLFxuICAgIG1ha2VQcmVzZW5jZTogb3B0aW9ucy5tYWtlUHJlc2VuY2UgPz8gKCgpID0+IFByb21pc2UucmVzb2x2ZSh1bmRlZmluZWQpKSxcbiAgICByZXNoYXJkaW5nOiB7XG4gICAgICBlbmFibGVkOiBvcHRpb25zLnJlc2hhcmRpbmc/LmVuYWJsZWQgPz8gdHJ1ZSxcbiAgICAgIHNoYXJkc0Z1bGxQZXJjZW50YWdlOiBvcHRpb25zLnJlc2hhcmRpbmc/LnNoYXJkc0Z1bGxQZXJjZW50YWdlID8/IDgwLFxuICAgICAgY2hlY2tJbnRlcnZhbDogb3B0aW9ucy5yZXNoYXJkaW5nPy5jaGVja0ludGVydmFsID8/IDI4ODAwMDAwLCAvLyA4IGhvdXJzXG4gICAgICBzaGFyZHM6IG5ldyBDb2xsZWN0aW9uKCksXG4gICAgICBwZW5kaW5nU2hhcmRzOiBuZXcgQ29sbGVjdGlvbigpLFxuICAgICAgZ2V0U2Vzc2lvbkluZm86IG9wdGlvbnMucmVzaGFyZGluZz8uZ2V0U2Vzc2lvbkluZm8sXG4gICAgICB1cGRhdGVHdWlsZHNTaGFyZElkOiBvcHRpb25zLnJlc2hhcmRpbmc/LnVwZGF0ZUd1aWxkc1NoYXJkSWQsXG4gICAgICBhc3luYyBjaGVja0lmUmVzaGFyZGluZ0lzTmVlZGVkKCkge1xuICAgICAgICBnYXRld2F5LmxvZ2dlci5kZWJ1ZygnW1Jlc2hhcmRpbmddIENoZWNraW5nIGlmIHJlc2hhcmRpbmcgaXMgbmVlZGVkLicpXG5cbiAgICAgICAgaWYgKCFnYXRld2F5LnJlc2hhcmRpbmcuZW5hYmxlZCkge1xuICAgICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKCdbUmVzaGFyZGluZ10gUmVzaGFyZGluZyBpcyBkaXNhYmxlZC4nKVxuXG4gICAgICAgICAgcmV0dXJuIHsgbmVlZGVkOiBmYWxzZSB9XG4gICAgICAgIH1cblxuICAgICAgICBpZiAoIWdhdGV3YXkucmVzaGFyZGluZy5nZXRTZXNzaW9uSW5mbykge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcIltSZXNoYXJkaW5nXSBSZXNoYXJkaW5nIGlzIGVuYWJsZWQgYnV0IG5vICdyZXNoYXJkaW5nLmdldFNlc3Npb25JbmZvKCknIGlzIG5vdCBwcm92aWRlZC5cIilcbiAgICAgICAgfVxuXG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKCdbUmVzaGFyZGluZ10gUmVzaGFyZGluZyBpcyBlbmFibGVkLicpXG5cbiAgICAgICAgY29uc3Qgc2Vzc2lvbkluZm8gPSBhd2FpdCBnYXRld2F5LnJlc2hhcmRpbmcuZ2V0U2Vzc2lvbkluZm8oKVxuXG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbUmVzaGFyZGluZ10gU2Vzc2lvbiBpbmZvIHJldHJpZXZlZDogJHtKU09OLnN0cmluZ2lmeShzZXNzaW9uSW5mbyl9YClcblxuICAgICAgICAvLyBEb24ndCBoYXZlIGVub3VnaCBpZGVudGlmeSBsaW1pdHMgdG8gdHJ5IHJlc2hhcmRpbmdcbiAgICAgICAgaWYgKHNlc3Npb25JbmZvLnNlc3Npb25TdGFydExpbWl0LnJlbWFpbmluZyA8IHNlc3Npb25JbmZvLnNoYXJkcykge1xuICAgICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKCdbUmVzaGFyZGluZ10gTm90IGVub3VnaCBzZXNzaW9uIHN0YXJ0IGxpbWl0cyBsZWZ0IHRvIHJlc2hhcmQuJylcblxuICAgICAgICAgIHJldHVybiB7IG5lZWRlZDogZmFsc2UsIGluZm86IHNlc3Npb25JbmZvIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKCdbUmVzaGFyZGluZ10gQWJsZSB0byByZXNoYXJkLCBjaGVja2luZyB3aGV0aGVyIG5lY2Vzc2FyeSBub3cuJylcblxuICAgICAgICAvLyAyNTAwIGlzIHRoZSBtYXggYW1vdW50IG9mIGd1aWxkcyBhIHNpbmdsZSBzaGFyZCBjYW4gaGFuZGxlXG4gICAgICAgIC8vIDEwMDAgaXMgdGhlIGFtb3VudCBvZiBndWlsZHMgZGlzY29yZCB1c2VzIHRvIGRldGVybWluZSBob3cgbWFueSBzaGFyZHMgdG8gcmVjb21tZW5kLlxuICAgICAgICAvLyBUaGlzIGFsZ28gaGVscHMgY2hlY2sgaWYgeW91ciBib3QgaGFzIGdyb3duIGVub3VnaCB0byByZXNoYXJkLlxuICAgICAgICAvLyBXaGlsZSB0aGlzIGlzIGltcHJlY2lzZSBhcyBkaXNjb3JkIGNoYW5nZXMgdGhlIHJlY29tbWVuZGVkIG51bWJlciBvZiBzaGFyZCBldmVyeSAxMDAwIGd1aWxkcyBpdCBpcyBnb29kIGVub3VnaFxuICAgICAgICAvLyBUaGUgYWx0ZXJuYXRpdmUgaXMgdG8gc3RvcmUgdGhlIGd1aWxkIGNvdW50IGZvciBlYWNoIHNoYXJkIGFuZCByZXF1aXJlIHRoZSBHdWlsZHMgaW50ZW50IGZvciBgR1VJTERfQ1JFQVRFYCBhbmQgYEdVSUxEX0RFTEVURWAgZXZlbnRzXG4gICAgICAgIGNvbnN0IHBlcmNlbnRhZ2UgPSAoc2Vzc2lvbkluZm8uc2hhcmRzIC8gKChnYXRld2F5LnRvdGFsU2hhcmRzICogMjUwMCkgLyAxMDAwKSkgKiAxMDBcblxuICAgICAgICAvLyBMZXNzIHRoYW4gbmVjZXNzYXJ5JSBiZWluZyB1c2VkIHNvIGRvIG5vdGhpbmdcbiAgICAgICAgaWYgKHBlcmNlbnRhZ2UgPCBnYXRld2F5LnJlc2hhcmRpbmcuc2hhcmRzRnVsbFBlcmNlbnRhZ2UpIHtcbiAgICAgICAgICBnYXRld2F5LmxvZ2dlci5kZWJ1ZygnW1Jlc2hhcmRpbmddIFJlc2hhcmRpbmcgbm90IG5lZWRlZC4nKVxuXG4gICAgICAgICAgcmV0dXJuIHsgbmVlZGVkOiBmYWxzZSwgaW5mbzogc2Vzc2lvbkluZm8gfVxuICAgICAgICB9XG5cbiAgICAgICAgZ2F0ZXdheS5sb2dnZXIuaW5mbygnW1Jlc2hhcmRpbmddIFJlc2hhcmRpbmcgaXMgbmVlZGVkLicpXG5cbiAgICAgICAgcmV0dXJuIHsgbmVlZGVkOiB0cnVlLCBpbmZvOiBzZXNzaW9uSW5mbyB9XG4gICAgICB9LFxuICAgICAgYXN5bmMgcmVzaGFyZChpbmZvKSB7XG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmluZm8oYFtSZXNoYXJkaW5nXSBTdGFydGluZyB0aGUgcmVzaGFyZCBwcm9jZXNzLiBQcmV2aW91cyB0b3RhbCBzaGFyZHM6ICR7Z2F0ZXdheS50b3RhbFNoYXJkc31gKVxuICAgICAgICAvLyBTZXQgdmFsdWVzIG9uIGdhdGV3YXlcbiAgICAgICAgZ2F0ZXdheS50b3RhbFNoYXJkcyA9IGluZm8uc2hhcmRzXG4gICAgICAgIC8vIEhhbmRsZXMgcHJlcGFyaW5nIG1pZCBzaXplZCBib3RzIGZvciBMQlNcbiAgICAgICAgZ2F0ZXdheS50b3RhbFNoYXJkcyA9IGdhdGV3YXkuY2FsY3VsYXRlVG90YWxTaGFyZHMoKVxuICAgICAgICAvLyBTZXQgZmlyc3Qgc2hhcmQgaWQgaWYgcHJvdmlkZWQgaW4gaW5mb1xuICAgICAgICBpZiAodHlwZW9mIGluZm8uZmlyc3RTaGFyZElkID09PSAnbnVtYmVyJykgZ2F0ZXdheS5maXJzdFNoYXJkSWQgPSBpbmZvLmZpcnN0U2hhcmRJZFxuICAgICAgICAvLyBTZXQgbGFzdCBzaGFyZCBpZCBpZiBwcm92aWRlZCBpbiBpbmZvXG4gICAgICAgIGlmICh0eXBlb2YgaW5mby5sYXN0U2hhcmRJZCA9PT0gJ251bWJlcicpIGdhdGV3YXkubGFzdFNoYXJkSWQgPSBpbmZvLmxhc3RTaGFyZElkXG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmluZm8oYFtSZXNoYXJkaW5nXSBTdGFydGluZyB0aGUgcmVzaGFyZCBwcm9jZXNzLiBOZXcgdG90YWwgc2hhcmRzOiAke2dhdGV3YXkudG90YWxTaGFyZHN9YClcblxuICAgICAgICAvLyBSZXNldHRpbmcgYnVja2V0c1xuICAgICAgICBnYXRld2F5LmJ1Y2tldHMuY2xlYXIoKVxuICAgICAgICAvLyBSZWZpbGxpbmcgYnVja2V0cyB3aXRoIG5ldyB2YWx1ZXNcbiAgICAgICAgZ2F0ZXdheS5wcmVwYXJlQnVja2V0cygpXG5cbiAgICAgICAgLy8gU1BSRUFEIFRISVMgT1VUIFRPIERJRkZFUkVOVCBXT1JLRVJTIFRPIEJFR0lOIFNUQVJUSU5HIFVQXG4gICAgICAgIGdhdGV3YXkuYnVja2V0cy5mb3JFYWNoKGFzeW5jIChidWNrZXQsIGJ1Y2tldElkKSA9PiB7XG4gICAgICAgICAgZm9yIChjb25zdCB3b3JrZXIgb2YgYnVja2V0LndvcmtlcnMpIHtcbiAgICAgICAgICAgIGZvciAoY29uc3Qgc2hhcmRJZCBvZiB3b3JrZXIucXVldWUpIHtcbiAgICAgICAgICAgICAgYXdhaXQgZ2F0ZXdheS5yZXNoYXJkaW5nLnRlbGxXb3JrZXJUb1ByZXBhcmUod29ya2VyLmlkLCBzaGFyZElkLCBidWNrZXRJZClcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH0pXG4gICAgICB9LFxuICAgICAgYXN5bmMgdGVsbFdvcmtlclRvUHJlcGFyZSh3b3JrZXJJZCwgc2hhcmRJZCwgYnVja2V0SWQpIHtcbiAgICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtSZXNoYXJkaW5nXSBUZWxsaW5nIHdvcmtlciB0byBwcmVwYXJlLiBXb3JrZXI6ICR7d29ya2VySWR9IHwgU2hhcmQ6ICR7c2hhcmRJZH0gfCBCdWNrZXQ6ICR7YnVja2V0SWR9LmApXG4gICAgICAgIGNvbnN0IHNoYXJkID0gbmV3IFNoYXJkKHtcbiAgICAgICAgICBpZDogc2hhcmRJZCxcbiAgICAgICAgICBjb25uZWN0aW9uOiB7XG4gICAgICAgICAgICBjb21wcmVzczogZ2F0ZXdheS5jb21wcmVzcyxcbiAgICAgICAgICAgIHRyYW5zcG9ydENvbXByZXNzaW9uOiBnYXRld2F5LnRyYW5zcG9ydENvbXByZXNzaW9uID8/IG51bGwsXG4gICAgICAgICAgICBpbnRlbnRzOiBnYXRld2F5LmludGVudHMsXG4gICAgICAgICAgICBwcm9wZXJ0aWVzOiBnYXRld2F5LnByb3BlcnRpZXMsXG4gICAgICAgICAgICB0b2tlbjogZ2F0ZXdheS50b2tlbixcbiAgICAgICAgICAgIHRvdGFsU2hhcmRzOiBnYXRld2F5LnRvdGFsU2hhcmRzLFxuICAgICAgICAgICAgdXJsOiBnYXRld2F5LnVybCxcbiAgICAgICAgICAgIHZlcnNpb246IGdhdGV3YXkudmVyc2lvbixcbiAgICAgICAgICB9LFxuICAgICAgICAgIC8vIElnbm9yZSBldmVudHMgdW50aWwgd2UgYXJlIHJlYWR5XG4gICAgICAgICAgZXZlbnRzOiB7XG4gICAgICAgICAgICBhc3luYyBtZXNzYWdlKF9zaGFyZCwgcGF5bG9hZCkge1xuICAgICAgICAgICAgICBpZiAocGF5bG9hZC50ID09PSAnUkVBRFknKSB7XG4gICAgICAgICAgICAgICAgYXdhaXQgZ2F0ZXdheS5yZXNoYXJkaW5nLnVwZGF0ZUd1aWxkc1NoYXJkSWQ/LihcbiAgICAgICAgICAgICAgICAgIChwYXlsb2FkLmQgYXMgRGlzY29yZFJlYWR5KS5ndWlsZHMubWFwKChnKSA9PiBnLmlkKSxcbiAgICAgICAgICAgICAgICAgIHNoYXJkSWQsXG4gICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9LFxuICAgICAgICAgIH0sXG4gICAgICAgICAgbG9nZ2VyOiBnYXRld2F5LmxvZ2dlcixcbiAgICAgICAgICByZXF1ZXN0SWRlbnRpZnk6IGFzeW5jICgpID0+IHtcbiAgICAgICAgICAgIGF3YWl0IGdhdGV3YXkuaWRlbnRpZnkoc2hhcmRJZClcbiAgICAgICAgICB9LFxuICAgICAgICAgIHNoYXJkSXNSZWFkeTogYXN5bmMgKCkgPT4ge1xuICAgICAgICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtTaGFyZF0gU2hhcmQgIyR7c2hhcmRJZH0gaXMgcmVhZHlgKVxuICAgICAgICAgICAgYXdhaXQgZGVsYXkoZ2F0ZXdheS5zcGF3blNoYXJkRGVsYXkpXG4gICAgICAgICAgICBnYXRld2F5LmxvZ2dlci5kZWJ1ZyhgW1NoYXJkXSBSZXNvbHZpbmcgc2hhcmQgaWRlbnRpZnkgcmVxdWVzdGApXG4gICAgICAgICAgICBnYXRld2F5LmJ1Y2tldHMuZ2V0KHNoYXJkSWQgJSBnYXRld2F5LmNvbm5lY3Rpb24uc2Vzc2lvblN0YXJ0TGltaXQubWF4Q29uY3VycmVuY3kpIS5pZGVudGlmeVJlcXVlc3RzLnNoaWZ0KCk/LigpXG4gICAgICAgICAgfSxcbiAgICAgICAgICBtYWtlUHJlc2VuY2U6IGdhdGV3YXkubWFrZVByZXNlbmNlLFxuICAgICAgICB9KVxuXG4gICAgICAgIGlmIChnYXRld2F5LnByZWZlclNuYWtlQ2FzZSkge1xuICAgICAgICAgIHNoYXJkLmZvcndhcmRUb0JvdCA9IGFzeW5jIChwYXlsb2FkKSA9PiB7XG4gICAgICAgICAgICBzaGFyZC5ldmVudHM/Lm1lc3NhZ2U/LihzaGFyZCwgcGF5bG9hZClcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICBnYXRld2F5LnJlc2hhcmRpbmcuc2hhcmRzLnNldChzaGFyZElkLCBzaGFyZClcblxuICAgICAgICBjb25zdCBidWNrZXQgPSBnYXRld2F5LmJ1Y2tldHMuZ2V0KHNoYXJkSWQgJSBnYXRld2F5LmNvbm5lY3Rpb24uc2Vzc2lvblN0YXJ0TGltaXQubWF4Q29uY3VycmVuY3kpXG4gICAgICAgIGlmICghYnVja2V0KSByZXR1cm5cblxuICAgICAgICByZXR1cm4gYXdhaXQgbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgICAgICAgICAvLyBNYXJrIHRoYXQgd2UgYXJlIG1ha2luZyBhbiBpZGVudGlmeSByZXF1ZXN0IHNvIGFub3RoZXIgaXMgbm90IG1hZGUuXG4gICAgICAgICAgYnVja2V0LmlkZW50aWZ5UmVxdWVzdHMucHVzaChyZXNvbHZlKVxuICAgICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gSWRlbnRpZnlpbmcgU2hhcmQgIyR7c2hhcmRJZH0uYClcbiAgICAgICAgICAvLyBUaGlzIHdpbGwgdHJpZ2dlciBpZGVudGlmeSBhbmQgd2hlbiBSRUFEWSBpcyByZWNlaXZlZCBpdCB3aWxsIHJlc29sdmUgdGhlIGFib3ZlIHJlcXVlc3QuXG4gICAgICAgICAgc2hhcmQ/LmlkZW50aWZ5KCkudGhlbihhc3luYyAoKSA9PiB7XG4gICAgICAgICAgICAvLyBUZWxsIHRoZSBtYW5hZ2VyIHRoYXQgdGhpcyBzaGFyZCBpcyBvbmxpbmVcbiAgICAgICAgICAgIHJldHVybiBhd2FpdCBnYXRld2F5LnJlc2hhcmRpbmcuc2hhcmRJc1BlbmRpbmcoc2hhcmQpXG4gICAgICAgICAgfSlcbiAgICAgICAgfSlcbiAgICAgIH0sXG4gICAgICBhc3luYyBzaGFyZElzUGVuZGluZyhzaGFyZCkge1xuICAgICAgICAvLyBTYXZlIHRoaXMgaW4gcGVuZGluZyBhdCB0aGUgbW9tZW50LCB1bnRpbCBhbGwgc2hhcmRzIGFyZSBvbmxpbmVcbiAgICAgICAgZ2F0ZXdheS5yZXNoYXJkaW5nLnBlbmRpbmdTaGFyZHMuc2V0KHNoYXJkLmlkLCBzaGFyZClcbiAgICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtSZXNoYXJkaW5nXSBTaGFyZCAjJHtzaGFyZC5pZH0gaXMgbm93IHBlbmRpbmcuYClcblxuICAgICAgICAvLyBDaGVjayBpZiBhbGwgc2hhcmRzIGFyZSBub3cgb25saW5lLlxuICAgICAgICBpZiAoZ2F0ZXdheS5sYXN0U2hhcmRJZCAtIGdhdGV3YXkuZmlyc3RTaGFyZElkID49IGdhdGV3YXkucmVzaGFyZGluZy5wZW5kaW5nU2hhcmRzLnNpemUpIHJldHVyblxuXG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmluZm8oYFtSZXNoYXJkaW5nXSBBbGwgc2hhcmRzIGFyZSBub3cgb25saW5lLmApXG5cbiAgICAgICAgLy8gTmV3IHNoYXJkcyBzdGFydCBwcm9jZXNzaW5nIGV2ZW50c1xuICAgICAgICBmb3IgKGNvbnN0IHNoYXJkIG9mIGdhdGV3YXkucmVzaGFyZGluZy5zaGFyZHMudmFsdWVzKCkpIHtcbiAgICAgICAgICBmb3IgKGNvbnN0IGV2ZW50IGluIG9wdGlvbnMuZXZlbnRzKSB7XG4gICAgICAgICAgICBzaGFyZC5ldmVudHNbZXZlbnQgYXMga2V5b2YgU2hhcmRFdmVudHNdID0gb3B0aW9ucy5ldmVudHNbZXZlbnQgYXMga2V5b2YgU2hhcmRFdmVudHNdIGFzICguLi5hcmdzOiB1bmtub3duW10pID0+IHVua25vd25cbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyBPbGQgc2hhcmRzIHN0b3AgcHJvY2Vzc2luZyBldmVudHNcbiAgICAgICAgZm9yIChjb25zdCBzaGFyZCBvZiBnYXRld2F5LnNoYXJkcy52YWx1ZXMoKSkge1xuICAgICAgICAgIGNvbnN0IG9sZEhhbmRsZXIgPSBzaGFyZC5ldmVudHMubWVzc2FnZVxuXG4gICAgICAgICAgLy8gQ2hhbmdlIHdpdGggc3ByZWFkIG9wZXJhdG9yIHRvIG5vdCBhZmZlY3QgbmV3IHNoYXJkcywgYXMgY2hhbmdpbmcgYW55dGhpbmcgb24gc2hhcmQuZXZlbnRzIHdpbGwgZGlyZWN0bHkgY2hhbmdlIG9wdGlvbnMuZXZlbnRzLCB3aGljaCBjaGFuZ2VzIG5ldyBzaGFyZHMnIGV2ZW50c1xuICAgICAgICAgIHNoYXJkLmV2ZW50cyA9IHtcbiAgICAgICAgICAgIC4uLnNoYXJkLmV2ZW50cyxcbiAgICAgICAgICAgIG1lc3NhZ2U6IGFzeW5jIGZ1bmN0aW9uIChfLCBtZXNzYWdlKSB7XG4gICAgICAgICAgICAgIC8vIE1lbWJlciBjaGVja3MgbmVlZCB0byBjb250aW51ZSBidXQgb3RoZXJzIGNhbiBzdG9wXG4gICAgICAgICAgICAgIGlmIChtZXNzYWdlLnQgPT09ICdHVUlMRF9NRU1CRVJTX0NIVU5LJykge1xuICAgICAgICAgICAgICAgIG9sZEhhbmRsZXI/LihzaGFyZCwgbWVzc2FnZSlcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSxcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICBnYXRld2F5LmxvZ2dlci5pbmZvKGBbUmVzaGFyZGluZ10gU2h1dHRpbmcgZG93biBvbGQgc2hhcmRzLmApXG4gICAgICAgIC8vIENsb3NlIG9sZCBzaGFyZHNcbiAgICAgICAgYXdhaXQgZ2F0ZXdheS5zaHV0ZG93bihTaGFyZFNvY2tldENsb3NlQ29kZXMuUmVzaGFyZGVkLCAnUmVzaGFyZGVkIScsIGZhbHNlKVxuXG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmluZm8oYFtSZXNoYXJkaW5nXSBDb21wbGV0ZWQuYClcblxuICAgICAgICAvLyBSZXBsYWNlIG9sZCBzaGFyZHNcbiAgICAgICAgZ2F0ZXdheS5zaGFyZHMgPSBuZXcgQ29sbGVjdGlvbihnYXRld2F5LnJlc2hhcmRpbmcuc2hhcmRzKVxuXG4gICAgICAgIC8vIENsZWFyIG91ciBjb2xsZWN0aW9ucyBhbmQga2VlcCBvbmx5IG9uZSByZWZlcmVuY2UgdG8gdGhlIHNoYXJkcywgdGhlIG9uZSBpbiBnYXRld2F5LnNoYXJkc1xuICAgICAgICBnYXRld2F5LnJlc2hhcmRpbmcuc2hhcmRzLmNsZWFyKClcbiAgICAgICAgZ2F0ZXdheS5yZXNoYXJkaW5nLnBlbmRpbmdTaGFyZHMuY2xlYXIoKVxuICAgICAgfSxcbiAgICB9LFxuXG4gICAgY2FsY3VsYXRlVG90YWxTaGFyZHMoKSB7XG4gICAgICAvLyBCb3RzIHVuZGVyIDEwMGsgc2VydmVycyBkbyBub3QgaGF2ZSBhY2Nlc3MgdG8gTEJTLlxuICAgICAgaWYgKGdhdGV3YXkudG90YWxTaGFyZHMgPCAxMDApIHtcbiAgICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBDYWxjdWxhdGluZyB0b3RhbCBzaGFyZHM6ICR7Z2F0ZXdheS50b3RhbFNoYXJkc31gKVxuICAgICAgICByZXR1cm4gZ2F0ZXdheS50b3RhbFNoYXJkc1xuICAgICAgfVxuXG4gICAgICBnYXRld2F5LmxvZ2dlci5kZWJ1ZyhgW0dhdGV3YXldIENhbGN1bGF0aW5nIHRvdGFsIHNoYXJkc2AsIGdhdGV3YXkudG90YWxTaGFyZHMsIGdhdGV3YXkuY29ubmVjdGlvbi5zZXNzaW9uU3RhcnRMaW1pdC5tYXhDb25jdXJyZW5jeSlcbiAgICAgIC8vIENhbGN1bGF0ZSBhIG11bHRpcGxlIG9mIGBtYXhDb25jdXJyZW5jeWAgd2hpY2ggY2FuIGJlIHVzZWQgdG8gY29ubmVjdCB0byB0aGUgZ2F0ZXdheS5cbiAgICAgIHJldHVybiAoXG4gICAgICAgIE1hdGguY2VpbChcbiAgICAgICAgICBnYXRld2F5LnRvdGFsU2hhcmRzIC9cbiAgICAgICAgICAgIC8vIElmIGBtYXhDb25jdXJyZW5jeWAgaXMgMSwgd2UgY2FuIHNhZmVseSB1c2UgMTYgdG8gZ2V0IGB0b3RhbFNoYXJkc2AgdG8gYmUgaW4gYSBtdWx0aXBsZSBvZiAxNiBzbyB0aGF0IHdlIGNhbiBwcmVwYXJlIGJvdHMgd2l0aCAxMDBrIHNlcnZlcnMgZm9yIExCUy5cbiAgICAgICAgICAgIChnYXRld2F5LmNvbm5lY3Rpb24uc2Vzc2lvblN0YXJ0TGltaXQubWF4Q29uY3VycmVuY3kgPT09IDEgPyAxNiA6IGdhdGV3YXkuY29ubmVjdGlvbi5zZXNzaW9uU3RhcnRMaW1pdC5tYXhDb25jdXJyZW5jeSksXG4gICAgICAgICkgKiAoZ2F0ZXdheS5jb25uZWN0aW9uLnNlc3Npb25TdGFydExpbWl0Lm1heENvbmN1cnJlbmN5ID09PSAxID8gMTYgOiBnYXRld2F5LmNvbm5lY3Rpb24uc2Vzc2lvblN0YXJ0TGltaXQubWF4Q29uY3VycmVuY3kpXG4gICAgICApXG4gICAgfSxcbiAgICBjYWxjdWxhdGVXb3JrZXJJZChzaGFyZElkKSB7XG4gICAgICBjb25zdCB3b3JrZXJJZCA9IE1hdGgubWluKHNoYXJkSWQgJSBnYXRld2F5LnNoYXJkc1BlcldvcmtlciwgZ2F0ZXdheS50b3RhbFdvcmtlcnMgLSAxKVxuICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoXG4gICAgICAgIGBbR2F0ZXdheV0gQ2FsY3VsYXRpbmcgd29ya2VySWQ6IFNoYXJkOiAke3NoYXJkSWR9IC0+IFdvcmtlcjogJHt3b3JrZXJJZH0gLT4gUGVyIFdvcmtlcjogJHtnYXRld2F5LnNoYXJkc1Blcldvcmtlcn0gLT4gVG90YWw6ICR7Z2F0ZXdheS50b3RhbFdvcmtlcnN9YCxcbiAgICAgIClcbiAgICAgIHJldHVybiB3b3JrZXJJZFxuICAgIH0sXG4gICAgcHJlcGFyZUJ1Y2tldHMoKSB7XG4gICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGdhdGV3YXkuY29ubmVjdGlvbi5zZXNzaW9uU3RhcnRMaW1pdC5tYXhDb25jdXJyZW5jeTsgKytpKSB7XG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gUHJlcGFyaW5nIGJ1Y2tldHMgZm9yIGNvbmN1cnJlbmN5OiAke2l9YClcbiAgICAgICAgZ2F0ZXdheS5idWNrZXRzLnNldChpLCB7XG4gICAgICAgICAgd29ya2VyczogW10sXG4gICAgICAgICAgaWRlbnRpZnlSZXF1ZXN0czogW10sXG4gICAgICAgIH0pXG4gICAgICB9XG5cbiAgICAgIC8vIE9SR0FOSVpFIEFMTCBTSEFSRFMgSU5UTyBUSEVJUiBPV04gQlVDS0VUU1xuICAgICAgZm9yIChsZXQgc2hhcmRJZCA9IGdhdGV3YXkuZmlyc3RTaGFyZElkOyBzaGFyZElkIDw9IGdhdGV3YXkubGFzdFNoYXJkSWQ7ICsrc2hhcmRJZCkge1xuICAgICAgICBnYXRld2F5LmxvZ2dlci5kZWJ1ZyhgW0dhdGV3YXldIFByZXBhcmluZyBidWNrZXRzIGZvciBzaGFyZDogJHtzaGFyZElkfWApXG4gICAgICAgIGlmIChzaGFyZElkID49IGdhdGV3YXkudG90YWxTaGFyZHMpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFNoYXJkIChpZDogJHtzaGFyZElkfSkgaXMgYmlnZ2VyIG9yIGVxdWFsIHRvIHRoZSB1c2VkIGFtb3VudCBvZiB1c2VkIHNoYXJkcyB3aGljaCBpcyAke2dhdGV3YXkudG90YWxTaGFyZHN9YClcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGJ1Y2tldElkID0gc2hhcmRJZCAlIGdhdGV3YXkuY29ubmVjdGlvbi5zZXNzaW9uU3RhcnRMaW1pdC5tYXhDb25jdXJyZW5jeVxuICAgICAgICBjb25zdCBidWNrZXQgPSBnYXRld2F5LmJ1Y2tldHMuZ2V0KGJ1Y2tldElkKVxuICAgICAgICBpZiAoIWJ1Y2tldCkge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICAgIGBTaGFyZCAoaWQ6ICR7c2hhcmRJZH0pIGdvdCBhc3NpZ25lZCB0byBhbiBpbGxlZ2FsIGJ1Y2tldCBpZDogJHtidWNrZXRJZH0sIGV4cGVjdGVkIGEgYnVja2V0IGlkIGJldHdlZW4gMCBhbmQgJHtcbiAgICAgICAgICAgICAgZ2F0ZXdheS5jb25uZWN0aW9uLnNlc3Npb25TdGFydExpbWl0Lm1heENvbmN1cnJlbmN5IC0gMVxuICAgICAgICAgICAgfWAsXG4gICAgICAgICAgKVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gRklORCBBIFFVRVVFIElOIFRISVMgQlVDS0VUIFRIQVQgSEFTIFNQQUNFXG4gICAgICAgIC8vIGNvbnN0IHdvcmtlciA9IGJ1Y2tldC53b3JrZXJzLmZpbmQoKHcpID0+IHcucXVldWUubGVuZ3RoIDwgZ2F0ZXdheS5zaGFyZHNQZXJXb3JrZXIpO1xuICAgICAgICBjb25zdCB3b3JrZXJJZCA9IGdhdGV3YXkuY2FsY3VsYXRlV29ya2VySWQoc2hhcmRJZClcbiAgICAgICAgY29uc3Qgd29ya2VyID0gYnVja2V0LndvcmtlcnMuZmluZCgodykgPT4gdy5pZCA9PT0gd29ya2VySWQpXG4gICAgICAgIGlmICh3b3JrZXIpIHtcbiAgICAgICAgICAvLyBJRiBUSEUgUVVFVUUgSEFTIFNQQUNFIEpVU1QgQUREIElUIFRPIFRISVMgUVVFVUVcbiAgICAgICAgICB3b3JrZXIucXVldWUucHVzaChzaGFyZElkKVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGJ1Y2tldC53b3JrZXJzLnB1c2goeyBpZDogd29ya2VySWQsIHF1ZXVlOiBbc2hhcmRJZF0gfSlcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBmb3IgKGNvbnN0IGJ1Y2tldCBvZiBnYXRld2F5LmJ1Y2tldHMudmFsdWVzKCkpIHtcbiAgICAgICAgZm9yIChjb25zdCB3b3JrZXIgb2YgYnVja2V0LndvcmtlcnMudmFsdWVzKCkpIHtcbiAgICAgICAgICB3b3JrZXIucXVldWUgPSB3b3JrZXIucXVldWUuc29ydCgoYSwgYikgPT4gYSAtIGIpXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9LFxuICAgIGFzeW5jIHNwYXduU2hhcmRzKCkge1xuICAgICAgLy8gUFJFUEFSRVMgQUxMIFNIQVJEUyBJTiBTUEVDSUZJQyBCVUNLRVRTXG4gICAgICBnYXRld2F5LnByZXBhcmVCdWNrZXRzKClcblxuICAgICAgLy8gUHJlZmVyIGNvbmN1cnJlbmN5IG9mIGZvckVhY2ggaW5zdGVhZCBvZiBmb3JvZlxuICAgICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICAgIFsuLi5nYXRld2F5LmJ1Y2tldHMuZW50cmllcygpXS5tYXAoYXN5bmMgKFtidWNrZXRJZCwgYnVja2V0XSkgPT4ge1xuICAgICAgICAgIGZvciAoY29uc3Qgd29ya2VyIG9mIGJ1Y2tldC53b3JrZXJzKSB7XG4gICAgICAgICAgICBmb3IgKGNvbnN0IHNoYXJkSWQgb2Ygd29ya2VyLnF1ZXVlKSB7XG4gICAgICAgICAgICAgIGF3YWl0IGdhdGV3YXkudGVsbFdvcmtlclRvSWRlbnRpZnkod29ya2VyLmlkLCBzaGFyZElkLCBidWNrZXRJZClcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH0pLFxuICAgICAgKVxuXG4gICAgICAvLyBDaGVjayBhbmQgcmVzaGFyZCBhdXRvbWF0aWNhbGx5IGlmIGF1dG8gcmVzaGFyZGluZyBpcyBlbmFibGVkLlxuICAgICAgaWYgKGdhdGV3YXkucmVzaGFyZGluZy5lbmFibGVkICYmIGdhdGV3YXkucmVzaGFyZGluZy5jaGVja0ludGVydmFsICE9PSAtMSkge1xuICAgICAgICAvLyBJdCBpcyBiZXR0ZXIgdG8gZW5zdXJlIHRoZXJlIGlzIGFsd2F5cyBvbmx5IG9uZVxuICAgICAgICBjbGVhckludGVydmFsKGdhdGV3YXkucmVzaGFyZGluZy5jaGVja0ludGVydmFsSWQpXG5cbiAgICAgICAgaWYgKCFnYXRld2F5LnJlc2hhcmRpbmcuZ2V0U2Vzc2lvbkluZm8pIHtcbiAgICAgICAgICBnYXRld2F5LnJlc2hhcmRpbmcuZW5hYmxlZCA9IGZhbHNlXG4gICAgICAgICAgZ2F0ZXdheS5sb2dnZXIud2FybihcIltSZXNoYXJkaW5nXSBSZXNoYXJkaW5nIGlzIGVuYWJsZWQgYnV0ICdyZXNoYXJkaW5nLmdldFNlc3Npb25JbmZvKCknIHdhcyBub3QgcHJvdmlkZWQuIERpc2FibGluZyByZXNoYXJkaW5nLlwiKVxuXG4gICAgICAgICAgcmV0dXJuXG4gICAgICAgIH1cblxuICAgICAgICBnYXRld2F5LnJlc2hhcmRpbmcuY2hlY2tJbnRlcnZhbElkID0gc2V0SW50ZXJ2YWwoYXN5bmMgKCkgPT4ge1xuICAgICAgICAgIGNvbnN0IHJlc2hhcmRpbmdJbmZvID0gYXdhaXQgZ2F0ZXdheS5yZXNoYXJkaW5nLmNoZWNrSWZSZXNoYXJkaW5nSXNOZWVkZWQoKVxuXG4gICAgICAgICAgaWYgKHJlc2hhcmRpbmdJbmZvLm5lZWRlZCAmJiByZXNoYXJkaW5nSW5mby5pbmZvKSBhd2FpdCBnYXRld2F5LnJlc2hhcmRpbmcucmVzaGFyZChyZXNoYXJkaW5nSW5mby5pbmZvKVxuICAgICAgICB9LCBnYXRld2F5LnJlc2hhcmRpbmcuY2hlY2tJbnRlcnZhbClcbiAgICAgIH1cbiAgICB9LFxuICAgIGFzeW5jIHNodXRkb3duKGNvZGUsIHJlYXNvbiwgY2xlYXJSZXNoYXJkaW5nSW50ZXJ2YWwgPSB0cnVlKSB7XG4gICAgICBnYXRld2F5LnNoYXJkcy5mb3JFYWNoKChzaGFyZCkgPT4gc2hhcmQuY2xvc2UoY29kZSwgcmVhc29uKSlcblxuICAgICAgaWYgKGNsZWFyUmVzaGFyZGluZ0ludGVydmFsKSBjbGVhckludGVydmFsKGdhdGV3YXkucmVzaGFyZGluZy5jaGVja0ludGVydmFsSWQpXG5cbiAgICAgIGF3YWl0IGRlbGF5KDUwMDApXG4gICAgfSxcbiAgICBhc3luYyBzZW5kUGF5bG9hZChzaGFyZElkLCBwYXlsb2FkKSB7XG4gICAgICBjb25zdCBzaGFyZCA9IGdhdGV3YXkuc2hhcmRzLmdldChzaGFyZElkKVxuXG4gICAgICBpZiAoIXNoYXJkKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgU2hhcmQgKGlkOiAke3NoYXJkSWR9IG5vdCBmb3VuZGApXG4gICAgICB9XG5cbiAgICAgIGF3YWl0IHNoYXJkLnNlbmQocGF5bG9hZClcbiAgICB9LFxuICAgIGFzeW5jIHRlbGxXb3JrZXJUb0lkZW50aWZ5KHdvcmtlcklkLCBzaGFyZElkLCBidWNrZXRJZCkge1xuICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBUZWxsIHdvcmtlciB0byBpZGVudGlmeSAoJHt3b3JrZXJJZH0sICR7c2hhcmRJZH0sICR7YnVja2V0SWR9KWApXG4gICAgICBhd2FpdCBnYXRld2F5LmlkZW50aWZ5KHNoYXJkSWQpXG4gICAgfSxcbiAgICBhc3luYyBpZGVudGlmeShzaGFyZElkOiBudW1iZXIpIHtcbiAgICAgIGxldCBzaGFyZCA9IHRoaXMuc2hhcmRzLmdldChzaGFyZElkKVxuICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBJZGVudGlmeWluZyAke3NoYXJkID8gJ2V4aXN0aW5nJyA6ICduZXcnfSBzaGFyZCAoJHtzaGFyZElkfSlgKVxuXG4gICAgICBpZiAoIXNoYXJkKSB7XG4gICAgICAgIHNoYXJkID0gbmV3IFNoYXJkKHtcbiAgICAgICAgICBpZDogc2hhcmRJZCxcbiAgICAgICAgICBjb25uZWN0aW9uOiB7XG4gICAgICAgICAgICBjb21wcmVzczogdGhpcy5jb21wcmVzcyxcbiAgICAgICAgICAgIHRyYW5zcG9ydENvbXByZXNzaW9uOiBnYXRld2F5LnRyYW5zcG9ydENvbXByZXNzaW9uLFxuICAgICAgICAgICAgaW50ZW50czogdGhpcy5pbnRlbnRzLFxuICAgICAgICAgICAgcHJvcGVydGllczogdGhpcy5wcm9wZXJ0aWVzLFxuICAgICAgICAgICAgdG9rZW46IHRoaXMudG9rZW4sXG4gICAgICAgICAgICB0b3RhbFNoYXJkczogdGhpcy50b3RhbFNoYXJkcyxcbiAgICAgICAgICAgIHVybDogdGhpcy51cmwsXG4gICAgICAgICAgICB2ZXJzaW9uOiB0aGlzLnZlcnNpb24sXG4gICAgICAgICAgfSxcbiAgICAgICAgICBldmVudHM6IG9wdGlvbnMuZXZlbnRzID8/IHt9LFxuICAgICAgICAgIGxvZ2dlcjogdGhpcy5sb2dnZXIsXG4gICAgICAgICAgcmVxdWVzdElkZW50aWZ5OiBhc3luYyAoKSA9PiB7XG4gICAgICAgICAgICBhd2FpdCBnYXRld2F5LmlkZW50aWZ5KHNoYXJkSWQpXG4gICAgICAgICAgfSxcbiAgICAgICAgICBzaGFyZElzUmVhZHk6IGFzeW5jICgpID0+IHtcbiAgICAgICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbU2hhcmRdIFNoYXJkICMke3NoYXJkSWR9IGlzIHJlYWR5YClcbiAgICAgICAgICAgIGF3YWl0IGRlbGF5KGdhdGV3YXkuc3Bhd25TaGFyZERlbGF5KVxuICAgICAgICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtTaGFyZF0gUmVzb2x2aW5nIHNoYXJkIGlkZW50aWZ5IHJlcXVlc3RgKVxuICAgICAgICAgICAgZ2F0ZXdheS5idWNrZXRzLmdldChzaGFyZElkICUgZ2F0ZXdheS5jb25uZWN0aW9uLnNlc3Npb25TdGFydExpbWl0Lm1heENvbmN1cnJlbmN5KSEuaWRlbnRpZnlSZXF1ZXN0cy5zaGlmdCgpPy4oKVxuICAgICAgICAgIH0sXG4gICAgICAgICAgbWFrZVByZXNlbmNlOiBnYXRld2F5Lm1ha2VQcmVzZW5jZSxcbiAgICAgICAgfSlcblxuICAgICAgICBpZiAodGhpcy5wcmVmZXJTbmFrZUNhc2UpIHtcbiAgICAgICAgICBzaGFyZC5mb3J3YXJkVG9Cb3QgPSBhc3luYyAocGF5bG9hZCkgPT4ge1xuICAgICAgICAgICAgc2hhcmQhLmV2ZW50cy5tZXNzYWdlPy4oc2hhcmQhLCBwYXlsb2FkKVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuc2hhcmRzLnNldChzaGFyZElkLCBzaGFyZClcbiAgICAgIH1cblxuICAgICAgY29uc3QgYnVja2V0ID0gZ2F0ZXdheS5idWNrZXRzLmdldChzaGFyZElkICUgZ2F0ZXdheS5jb25uZWN0aW9uLnNlc3Npb25TdGFydExpbWl0Lm1heENvbmN1cnJlbmN5KVxuICAgICAgaWYgKCFidWNrZXQpIHJldHVyblxuXG4gICAgICByZXR1cm4gYXdhaXQgbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgICAgICAgLy8gTWFyayB0aGF0IHdlIGFyZSBtYWtpbmcgYW4gaWRlbnRpZnkgcmVxdWVzdCBzbyBhbm90aGVyIGlzIG5vdCBtYWRlLlxuICAgICAgICBidWNrZXQuaWRlbnRpZnlSZXF1ZXN0cy5wdXNoKHJlc29sdmUpXG4gICAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gSWRlbnRpZnlpbmcgU2hhcmQgIyR7c2hhcmRJZH0uYClcbiAgICAgICAgLy8gVGhpcyB3aWxsIHRyaWdnZXIgaWRlbnRpZnkgYW5kIHdoZW4gUkVBRFkgaXMgcmVjZWl2ZWQgaXQgd2lsbCByZXNvbHZlIHRoZSBhYm92ZSByZXF1ZXN0LlxuICAgICAgICBzaGFyZD8uaWRlbnRpZnkoKVxuICAgICAgfSlcbiAgICB9LFxuICAgIGFzeW5jIGtpbGwoc2hhcmRJZDogbnVtYmVyKSB7XG4gICAgICBjb25zdCBzaGFyZCA9IHRoaXMuc2hhcmRzLmdldChzaGFyZElkKVxuICAgICAgaWYgKCFzaGFyZCkge1xuICAgICAgICByZXR1cm4gZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBBIGtpbGwgZm9yIFNoYXJkICMke3NoYXJkSWR9IHdhcyByZXF1ZXN0ZWQsIGJ1dCB0aGUgc2hhcmQgY291bGQgbm90IGJlIGZvdW5kYClcbiAgICAgIH1cblxuICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBLaWxsaW5nIFNoYXJkICMke3NoYXJkSWR9YClcbiAgICAgIHRoaXMuc2hhcmRzLmRlbGV0ZShzaGFyZElkKVxuICAgICAgYXdhaXQgc2hhcmQuc2h1dGRvd24oKVxuICAgIH0sXG4gICAgYXN5bmMgcmVxdWVzdElkZW50aWZ5KF9zaGFyZElkOiBudW1iZXIpIHtcbiAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gUmVxdWVzdGluZyBpZGVudGlmeWApXG4gICAgfSxcblxuICAgIC8vIEhlbHBlcnMgbWV0aG9kcyBiZWxvdyB0aGlzXG5cbiAgICBjYWxjdWxhdGVTaGFyZElkKGd1aWxkSWQsIHRvdGFsU2hhcmRzKSB7XG4gICAgICAvLyBJZiBub25lIGlzIHByb3ZpZGVkLCB1c2UgdGhlIHRvdGFsIHNoYXJkcyBudW1iZXIgZnJvbSBnYXRld2F5IG9iamVjdC5cbiAgICAgIGlmICghdG90YWxTaGFyZHMpIHRvdGFsU2hhcmRzID0gZ2F0ZXdheS50b3RhbFNoYXJkc1xuICAgICAgLy8gSWYgaXQgaXMgb25seSAxIHNoYXJkLCBpdCB3aWxsIGFsd2F5cyBiZSBzaGFyZCBpZCAwXG4gICAgICBpZiAodG90YWxTaGFyZHMgPT09IDEpIHtcbiAgICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBjYWxjdWxhdGVTaGFyZElkICgxIHNoYXJkKWApXG4gICAgICAgIHJldHVybiAwXG4gICAgICB9XG5cbiAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gY2FsY3VsYXRlU2hhcmRJZCAoZ3VpbGRJZDogJHtndWlsZElkfSwgdG90YWxTaGFyZHM6ICR7dG90YWxTaGFyZHN9KWApXG4gICAgICByZXR1cm4gTnVtYmVyKChCaWdJbnQoZ3VpbGRJZCkgPj4gMjJuKSAlIEJpZ0ludCh0b3RhbFNoYXJkcykpXG4gICAgfSxcblxuICAgIGFzeW5jIGpvaW5Wb2ljZUNoYW5uZWwoZ3VpbGRJZCwgY2hhbm5lbElkLCBvcHRpb25zKSB7XG4gICAgICBjb25zdCBzaGFyZElkID0gZ2F0ZXdheS5jYWxjdWxhdGVTaGFyZElkKGd1aWxkSWQpXG5cbiAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gam9pblZvaWNlQ2hhbm5lbCBndWlsZElkOiAke2d1aWxkSWR9IGNoYW5uZWxJZDogJHtjaGFubmVsSWR9YClcblxuICAgICAgYXdhaXQgZ2F0ZXdheS5zZW5kUGF5bG9hZChzaGFyZElkLCB7XG4gICAgICAgIG9wOiBHYXRld2F5T3Bjb2Rlcy5Wb2ljZVN0YXRlVXBkYXRlLFxuICAgICAgICBkOiB7XG4gICAgICAgICAgZ3VpbGRfaWQ6IGd1aWxkSWQudG9TdHJpbmcoKSxcbiAgICAgICAgICBjaGFubmVsX2lkOiBjaGFubmVsSWQudG9TdHJpbmcoKSxcbiAgICAgICAgICBzZWxmX211dGU6IG9wdGlvbnM/LnNlbGZNdXRlID8/IGZhbHNlLFxuICAgICAgICAgIHNlbGZfZGVhZjogb3B0aW9ucz8uc2VsZkRlYWYgPz8gdHJ1ZSxcbiAgICAgICAgfSxcbiAgICAgIH0pXG4gICAgfSxcblxuICAgIGFzeW5jIGVkaXRCb3RTdGF0dXMoZGF0YSkge1xuICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBlZGl0Qm90U3RhdHVzIGRhdGE6ICR7SlNPTi5zdHJpbmdpZnkoZGF0YSl9YClcblxuICAgICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICAgIFsuLi5nYXRld2F5LnNoYXJkcy52YWx1ZXMoKV0ubWFwKGFzeW5jIChzaGFyZCkgPT4ge1xuICAgICAgICAgIGdhdGV3YXkuZWRpdFNoYXJkU3RhdHVzKHNoYXJkLmlkLCBkYXRhKVxuICAgICAgICB9KSxcbiAgICAgIClcbiAgICB9LFxuXG4gICAgYXN5bmMgZWRpdFNoYXJkU3RhdHVzKHNoYXJkSWQsIGRhdGEpIHtcbiAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gZWRpdFNoYXJkU3RhdHVzIHNoYXJkSWQ6ICR7c2hhcmRJZH0gLT4gZGF0YTogJHtKU09OLnN0cmluZ2lmeShkYXRhKX1gKVxuXG4gICAgICBhd2FpdCBnYXRld2F5LnNlbmRQYXlsb2FkKHNoYXJkSWQsIHtcbiAgICAgICAgb3A6IEdhdGV3YXlPcGNvZGVzLlByZXNlbmNlVXBkYXRlLFxuICAgICAgICBkOiB7XG4gICAgICAgICAgc2luY2U6IG51bGwsXG4gICAgICAgICAgYWZrOiBmYWxzZSxcbiAgICAgICAgICBhY3Rpdml0aWVzOiBkYXRhLmFjdGl2aXRpZXMsXG4gICAgICAgICAgc3RhdHVzOiBkYXRhLnN0YXR1cyxcbiAgICAgICAgfSxcbiAgICAgIH0pXG4gICAgfSxcblxuICAgIGFzeW5jIHJlcXVlc3RNZW1iZXJzKGd1aWxkSWQsIG9wdGlvbnMpIHtcbiAgICAgIGNvbnN0IHNoYXJkSWQgPSBnYXRld2F5LmNhbGN1bGF0ZVNoYXJkSWQoZ3VpbGRJZClcblxuICAgICAgaWYgKGdhdGV3YXkuaW50ZW50cyAmJiAoIW9wdGlvbnM/LmxpbWl0IHx8IG9wdGlvbnMubGltaXQgPiAxKSAmJiAhKGdhdGV3YXkuaW50ZW50cyAmIEdhdGV3YXlJbnRlbnRzLkd1aWxkTWVtYmVycykpXG4gICAgICAgIHRocm93IG5ldyBFcnJvcignQ2Fubm90IGZldGNoIG1vcmUgdGhlbiAxIG1lbWJlciB3aXRob3V0IHRoZSBHVUlMRF9NRU1CRVJTIGludGVudCcpXG5cbiAgICAgIGdhdGV3YXkubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gcmVxdWVzdE1lbWJlcnMgZ3VpbGRJZDogJHtndWlsZElkfSAtPiBkYXRhOiAke0pTT04uc3RyaW5naWZ5KG9wdGlvbnMpfWApXG5cbiAgICAgIGlmIChvcHRpb25zPy51c2VySWRzPy5sZW5ndGgpIHtcbiAgICAgICAgZ2F0ZXdheS5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSByZXF1ZXN0TWVtYmVycyBndWlsZElkOiAke2d1aWxkSWR9IC0+IHNldHRpbmcgdXNlciBsaW1pdCBiYXNlZCBvbiB1c2VySWRzIGxlbmd0aDogJHtvcHRpb25zLnVzZXJJZHMubGVuZ3RofWApXG5cbiAgICAgICAgb3B0aW9ucy5saW1pdCA9IG9wdGlvbnMudXNlcklkcy5sZW5ndGhcbiAgICAgIH1cblxuICAgICAgY29uc3QgbWVtYmVycyA9XG4gICAgICAgICFnYXRld2F5LmNhY2hlLnJlcXVlc3RNZW1iZXJzLmVuYWJsZWQgfHwgIW9wdGlvbnM/Lm5vbmNlXG4gICAgICAgICAgPyBbXVxuICAgICAgICAgIDogbmV3IFByb21pc2U8Q2FtZWxpemU8RGlzY29yZE1lbWJlcldpdGhVc2VyW10+PigocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICAgICAgICAgIC8vIFNob3VsZCBuZXZlciBoYXBwZW4uXG4gICAgICAgICAgICAgIGlmICghZ2F0ZXdheS5jYWNoZS5yZXF1ZXN0TWVtYmVycy5lbmFibGVkIHx8ICFvcHRpb25zPy5ub25jZSkge1xuICAgICAgICAgICAgICAgIHJlamVjdChuZXcgRXJyb3IoXCJDYW4ndCByZXF1ZXN0IHRoZSBtZW1iZXJzIHdpdGhvdXQgdGhlIG5vbmNlIG9yIHdpdGggdGhlIGZlYXR1cmUgZGlzYWJsZWQuXCIpKVxuICAgICAgICAgICAgICAgIHJldHVyblxuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgZ2F0ZXdheS5jYWNoZS5yZXF1ZXN0TWVtYmVycy5wZW5kaW5nLnNldChvcHRpb25zLm5vbmNlLCB7XG4gICAgICAgICAgICAgICAgbm9uY2U6IG9wdGlvbnMubm9uY2UsXG4gICAgICAgICAgICAgICAgcmVzb2x2ZSxcbiAgICAgICAgICAgICAgICBtZW1iZXJzOiBbXSxcbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIH0pXG5cbiAgICAgIGF3YWl0IGdhdGV3YXkuc2VuZFBheWxvYWQoc2hhcmRJZCwge1xuICAgICAgICBvcDogR2F0ZXdheU9wY29kZXMuUmVxdWVzdEd1aWxkTWVtYmVycyxcbiAgICAgICAgZDoge1xuICAgICAgICAgIGd1aWxkX2lkOiBndWlsZElkLnRvU3RyaW5nKCksXG4gICAgICAgICAgLy8gSWYgYSBxdWVyeSBpcyBwcm92aWRlZCB1c2UgaXQsIE9SIGlmIGEgbGltaXQgaXMgTk9UIHByb3ZpZGVkIHVzZSBcIlwiXG4gICAgICAgICAgcXVlcnk6IG9wdGlvbnM/LnF1ZXJ5ID8/IChvcHRpb25zPy5saW1pdCA/IHVuZGVmaW5lZCA6ICcnKSxcbiAgICAgICAgICBsaW1pdDogb3B0aW9ucz8ubGltaXQgPz8gMCxcbiAgICAgICAgICBwcmVzZW5jZXM6IG9wdGlvbnM/LnByZXNlbmNlcyA/PyBmYWxzZSxcbiAgICAgICAgICB1c2VyX2lkczogb3B0aW9ucz8udXNlcklkcz8ubWFwKChpZCkgPT4gaWQudG9TdHJpbmcoKSksXG4gICAgICAgICAgbm9uY2U6IG9wdGlvbnM/Lm5vbmNlLFxuICAgICAgICB9LFxuICAgICAgfSlcblxuICAgICAgcmV0dXJuIGF3YWl0IG1lbWJlcnNcbiAgICB9LFxuXG4gICAgYXN5bmMgbGVhdmVWb2ljZUNoYW5uZWwoZ3VpbGRJZCkge1xuICAgICAgY29uc3Qgc2hhcmRJZCA9IGdhdGV3YXkuY2FsY3VsYXRlU2hhcmRJZChndWlsZElkKVxuXG4gICAgICBnYXRld2F5LmxvZ2dlci5kZWJ1ZyhgW0dhdGV3YXldIGxlYXZlVm9pY2VDaGFubmVsIGd1aWxkSWQ6ICR7Z3VpbGRJZH0gU2hhcmQgJHtzaGFyZElkfWApXG5cbiAgICAgIGF3YWl0IGdhdGV3YXkuc2VuZFBheWxvYWQoc2hhcmRJZCwge1xuICAgICAgICBvcDogR2F0ZXdheU9wY29kZXMuVm9pY2VTdGF0ZVVwZGF0ZSxcbiAgICAgICAgZDoge1xuICAgICAgICAgIGd1aWxkX2lkOiBndWlsZElkLnRvU3RyaW5nKCksXG4gICAgICAgICAgY2hhbm5lbF9pZDogbnVsbCxcbiAgICAgICAgICBzZWxmX211dGU6IGZhbHNlLFxuICAgICAgICAgIHNlbGZfZGVhZjogZmFsc2UsXG4gICAgICAgIH0sXG4gICAgICB9KVxuICAgIH0sXG5cbiAgICBhc3luYyByZXF1ZXN0U291bmRib2FyZFNvdW5kcyhndWlsZElkcykge1xuICAgICAgLyoqXG4gICAgICAgKiBEaXNjb3JkIHdpbGwgc2VuZCB0aGUgZXZlbnRzIGZvciB0aGUgZ3VpbGRzIHRoYXQgYXJlIFwidW5kZXIgdGhlIHNoYXJkXCIgdGhhdCBzZW5kcyB0aGUgb3Bjb2RlLlxuICAgICAgICogRm9yIHRoaXMgcmVhc29uIHdlIG5lZWQgdG8gZ3JvdXAgdGhlIGlkcyB3aXRoIHRoZSBzaGFyZCB0aGUgY2FsY3VsYXRlU2hhcmRJZCBtZXRob2QgZ2l2ZXNcbiAgICAgICAqL1xuXG4gICAgICBjb25zdCBtYXAgPSBuZXcgTWFwPG51bWJlciwgQmlnU3RyaW5nW10+KClcblxuICAgICAgZm9yIChjb25zdCBndWlsZElkIG9mIGd1aWxkSWRzKSB7XG4gICAgICAgIGNvbnN0IHNoYXJkSWQgPSBnYXRld2F5LmNhbGN1bGF0ZVNoYXJkSWQoZ3VpbGRJZClcblxuICAgICAgICBjb25zdCBpZHMgPSBtYXAuZ2V0KHNoYXJkSWQpID8/IFtdXG4gICAgICAgIG1hcC5zZXQoc2hhcmRJZCwgaWRzKVxuXG4gICAgICAgIGlkcy5wdXNoKGd1aWxkSWQpXG4gICAgICB9XG5cbiAgICAgIGF3YWl0IFByb21pc2UuYWxsKFxuICAgICAgICBbLi4ubWFwLmVudHJpZXMoKV0ubWFwKChbc2hhcmRJZCwgaWRzXSkgPT5cbiAgICAgICAgICBnYXRld2F5LnNlbmRQYXlsb2FkKHNoYXJkSWQsIHtcbiAgICAgICAgICAgIG9wOiBHYXRld2F5T3Bjb2Rlcy5SZXF1ZXN0U291bmRib2FyZFNvdW5kcyxcbiAgICAgICAgICAgIGQ6IHtcbiAgICAgICAgICAgICAgZ3VpbGRfaWRzOiBpZHMsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIH0pLFxuICAgICAgICApLFxuICAgICAgKVxuICAgIH0sXG4gIH1cblxuICByZXR1cm4gZ2F0ZXdheVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIENyZWF0ZUdhdGV3YXlNYW5hZ2VyT3B0aW9ucyB7XG4gIC8qKlxuICAgKiBJZCBvZiB0aGUgZmlyc3QgU2hhcmQgd2hpY2ggc2hvdWxkIGdldCBjb250cm9sbGVkIGJ5IHRoaXMgbWFuYWdlci5cbiAgICogQGRlZmF1bHQgMFxuICAgKi9cbiAgZmlyc3RTaGFyZElkPzogbnVtYmVyXG4gIC8qKlxuICAgKiBJZCBvZiB0aGUgbGFzdCBTaGFyZCB3aGljaCBzaG91bGQgZ2V0IGNvbnRyb2xsZWQgYnkgdGhpcyBtYW5hZ2VyLlxuICAgKiBAZGVmYXVsdCAwXG4gICAqL1xuICBsYXN0U2hhcmRJZD86IG51bWJlclxuICAvKipcbiAgICogRGVsYXkgaW4gbWlsbGlzZWNvbmRzIHRvIHdhaXQgYmVmb3JlIHNwYXduaW5nIG5leHQgc2hhcmQuIE9QVElNQUwgSVMgQUJPVkUgNTEwMC4gWU9VIERPTidUIFdBTlQgVE8gSElUIFRIRSBSQVRFIExJTUlUISEhXG4gICAqIEBkZWZhdWx0IDUzMDBcbiAgICovXG4gIHNwYXduU2hhcmREZWxheT86IG51bWJlclxuICAvKipcbiAgICogV2hldGhlciB0byBzZW5kIHRoZSBkaXNjb3JkIHBhY2tldHMgaW4gc25ha2UgY2FzZSBmb3JtLlxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgcHJlZmVyU25ha2VDYXNlPzogYm9vbGVhblxuICAvKipcbiAgICogVG90YWwgYW1vdW50IG9mIHNoYXJkcyB5b3VyIGJvdCB1c2VzLiBVc2VmdWwgZm9yIHplcm8tZG93bnRpbWUgdXBkYXRlcyBvciByZXNoYXJkaW5nLlxuICAgKiBAZGVmYXVsdCAxXG4gICAqL1xuICB0b3RhbFNoYXJkcz86IG51bWJlclxuICAvKipcbiAgICogVGhlIGFtb3VudCBvZiBzaGFyZHMgdG8gbG9hZCBwZXIgd29ya2VyLlxuICAgKiBAZGVmYXVsdCAyNVxuICAgKi9cbiAgc2hhcmRzUGVyV29ya2VyPzogbnVtYmVyXG4gIC8qKlxuICAgKiBUaGUgdG90YWwgYW1vdW50IG9mIHdvcmtlcnMgdG8gdXNlIGZvciB5b3VyIGJvdC5cbiAgICogQGRlZmF1bHQgNFxuICAgKi9cbiAgdG90YWxXb3JrZXJzPzogbnVtYmVyXG4gIC8qKiBJbXBvcnRhbnQgZGF0YSB3aGljaCBpcyB1c2VkIGJ5IHRoZSBtYW5hZ2VyIHRvIGNvbm5lY3Qgc2hhcmRzIHRvIHRoZSBnYXRld2F5LiAqL1xuICBjb25uZWN0aW9uPzogQ2FtZWxpemU8RGlzY29yZEdldEdhdGV3YXlCb3Q+XG4gIC8qKiBXaGV0aGVyIGluY29taW5nIHBheWxvYWRzIGFyZSBjb21wcmVzc2VkIHVzaW5nIHpsaWIuXG4gICAqXG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqL1xuICBjb21wcmVzcz86IGJvb2xlYW5cbiAgLyoqIFdoYXQgdHJhbnNwb3J0IGNvbXByZXNzaW9uIHNob3VsZCBiZSB1c2VkICovXG4gIHRyYW5zcG9ydENvbXByZXNzaW9uPzogVHJhbnNwb3J0Q29tcHJlc3Npb24gfCBudWxsXG4gIC8qKiBUaGUgY2FsY3VsYXRlZCBpbnRlbnQgdmFsdWUgb2YgdGhlIGV2ZW50cyB3aGljaCB0aGUgc2hhcmQgc2hvdWxkIHJlY2VpdmUuXG4gICAqXG4gICAqIEBkZWZhdWx0IDBcbiAgICovXG4gIGludGVudHM/OiBudW1iZXJcbiAgLyoqIElkZW50aWZ5IHByb3BlcnRpZXMgdG8gdXNlICovXG4gIHByb3BlcnRpZXM/OiB7XG4gICAgLyoqIE9wZXJhdGluZyBzeXN0ZW0gdGhlIHNoYXJkIHJ1bnMgb24uXG4gICAgICpcbiAgICAgKiBAZGVmYXVsdCBcImRhcndpblwiIHwgXCJsaW51eFwiIHwgXCJ3aW5kb3dzXCJcbiAgICAgKi9cbiAgICBvczogc3RyaW5nXG4gICAgLyoqIFRoZSBcImJyb3dzZXJcIiB3aGVyZSB0aGlzIHNoYXJkIGlzIHJ1bm5pbmcgb24uXG4gICAgICpcbiAgICAgKiBAZGVmYXVsdCBcIkRpc2NvcmRlbm9cIlxuICAgICAqL1xuICAgIGJyb3dzZXI6IHN0cmluZ1xuICAgIC8qKiBUaGUgZGV2aWNlIG9uIHdoaWNoIHRoZSBzaGFyZCBpcyBydW5uaW5nLlxuICAgICAqXG4gICAgICogQGRlZmF1bHQgXCJEaXNjb3JkZW5vXCJcbiAgICAgKi9cbiAgICBkZXZpY2U6IHN0cmluZ1xuICB9XG4gIC8qKiBCb3QgdG9rZW4gd2hpY2ggaXMgdXNlZCB0byBjb25uZWN0IHRvIERpc2NvcmQgKi9cbiAgdG9rZW46IHN0cmluZ1xuICAvKiogVGhlIFVSTCBvZiB0aGUgZ2F0ZXdheSB3aGljaCBzaG91bGQgYmUgY29ubmVjdGVkIHRvLlxuICAgKlxuICAgKiBAZGVmYXVsdCBcIndzczovL2dhdGV3YXkuZGlzY29yZC5nZ1wiXG4gICAqL1xuICB1cmw/OiBzdHJpbmdcbiAgLyoqIFRoZSBnYXRld2F5IHZlcnNpb24gd2hpY2ggc2hvdWxkIGJlIHVzZWQuXG4gICAqXG4gICAqIEBkZWZhdWx0IDEwXG4gICAqL1xuICB2ZXJzaW9uPzogbnVtYmVyXG4gIC8qKiBUaGUgZXZlbnRzIGhhbmRsZXJzICovXG4gIGV2ZW50cz86IFNoYXJkRXZlbnRzXG4gIC8qKiBUaGlzIG1hbmFnZXJzIGNhY2hlIHJlbGF0ZWQgc2V0dGluZ3MuICovXG4gIGNhY2hlPzoge1xuICAgIHJlcXVlc3RNZW1iZXJzPzoge1xuICAgICAgLyoqXG4gICAgICAgKiBXaGV0aGVyIG9yIG5vdCByZXF1ZXN0IG1lbWJlciByZXF1ZXN0cyBzaG91bGQgYmUgY2FjaGVkLlxuICAgICAgICogQGRlZmF1bHQgZmFsc2VcbiAgICAgICAqL1xuICAgICAgZW5hYmxlZD86IGJvb2xlYW5cbiAgICB9XG4gIH1cbiAgLyoqXG4gICAqIFRoZSBsb2dnZXIgdGhhdCB0aGUgZ2F0ZXdheSBtYW5hZ2VyIHdpbGwgdXNlLlxuICAgKiBAZGVmYXVsdCBsb2dnZXIgLy8gVGhlIGxvZ2dlciBleHBvcnRlZCBieSBgQGRpc2NvcmRlbm8vdXRpbHNgXG4gICAqL1xuICBsb2dnZXI/OiBQaWNrPHR5cGVvZiBsb2dnZXIsICdkZWJ1ZycgfCAnaW5mbycgfCAnd2FybicgfCAnZXJyb3InIHwgJ2ZhdGFsJz5cbiAgLyoqXG4gICAqIE1ha2UgdGhlIHByZXNlbmNlIGZvciB3aGVuIHRoZSBib3QgY29ubmVjdHMgdG8gdGhlIGdhdGV3YXlcbiAgICpcbiAgICogQHJlbWFya3NcbiAgICogVGhpcyBmdW5jdGlvbiB3aWxsIGJlIGNhbGxlZCBlYWNoIHRpbWUgYSBTaGFyZCBpcyBnb2luZyB0byBpZGVudGlmeVxuICAgKi9cbiAgbWFrZVByZXNlbmNlPzogKCkgPT4gUHJvbWlzZTxCb3RTdGF0dXNVcGRhdGUgfCB1bmRlZmluZWQ+XG4gIC8qKiBPcHRpb25zIHJlbGF0ZWQgdG8gcmVzaGFyZGluZy4gKi9cbiAgcmVzaGFyZGluZz86IHtcbiAgICAvKipcbiAgICAgKiBXaGV0aGVyIG9yIG5vdCBhdXRvbWF0ZWQgcmVzaGFyZGluZyBzaG91bGQgYmUgZW5hYmxlZC5cbiAgICAgKiBAZGVmYXVsdCB0cnVlXG4gICAgICovXG4gICAgZW5hYmxlZDogYm9vbGVhblxuICAgIC8qKlxuICAgICAqIFRoZSAlIG9mIGhvdyBmdWxsIGEgc2hhcmQgaXMgd2hlbiByZXNoYXJkaW5nIHNob3VsZCBiZSB0cmlnZ2VyZWQuXG4gICAgICpcbiAgICAgKiBAcmVtYXJrc1xuICAgICAqIFdlIHVzZSBkaXNjb3JkIHJlY29tbWVuZGVkIHNoYXJkIHZhbHVlIHRvIGdldCBhbiAqKmFwcHJveGltYXRpb24qKiBvZiB0aGUgc2hhcmQgZnVsbCBwZXJjZW50YWdlIHRvIGNvbXBhcmUgd2l0aCB0aGlzIHZhbHVlIHNvIHRoZSBib3QgbWF5IG5vdCByZXNoYXJkIGF0IHRoZSBleGFjdCBwZXJjZW50YWdlIHByb3ZpZGVkIGJ1dCBtYXkgcmVzaGFyZCB3aGVuIGl0IGlzIGEgYml0IGhpZ2hlciB0aGFuIHRoZSBwcm92aWRlZCBwZXJjZW50YWdlLlxuICAgICAqIEZvciBhY2N1cmF0ZSBjYWxjdWxhdGlvbiwgeW91IG1heSBvdmVycmlkZSB0aGUgYGNoZWNrSWZSZXNoYXJkaW5nSXNOZWVkZWRgIGZ1bmN0aW9uXG4gICAgICpcbiAgICAgKiBAZGVmYXVsdCA4MCBhcyBpbiA4MCVcbiAgICAgKi9cbiAgICBzaGFyZHNGdWxsUGVyY2VudGFnZTogbnVtYmVyXG4gICAgLyoqXG4gICAgICogVGhlIGludGVydmFsIGluIG1pbGxpc2Vjb25kcywgb2YgaG93IG9mdGVuIHRvIGNoZWNrIHdoZXRoZXIgcmVzaGFyZGluZyBpcyBuZWVkZWQgYW5kIHJlc2hhcmQgYXV0b21hdGljYWxseS4gU2V0IHRvIC0xIHRvIGRpc2FibGUgYXV0byByZXNoYXJkaW5nLlxuICAgICAqIEBkZWZhdWx0IDI4ODAwMDAwICg4IGhvdXJzKVxuICAgICAqL1xuICAgIGNoZWNrSW50ZXJ2YWw6IG51bWJlclxuICAgIC8qKiBIYW5kbGVyIHRvIGdldCBzaGFyZCBjb3VudCBhbmQgb3RoZXIgc2Vzc2lvbiBpbmZvLiAqL1xuICAgIGdldFNlc3Npb25JbmZvPzogKCkgPT4gUHJvbWlzZTxDYW1lbGl6ZTxEaXNjb3JkR2V0R2F0ZXdheUJvdD4+XG4gICAgLyoqIEhhbmRsZXIgdG8gZWRpdCB0aGUgc2hhcmQgaWQgb24gYW55IGNhY2hlZCBndWlsZHMuICovXG4gICAgdXBkYXRlR3VpbGRzU2hhcmRJZD86IChndWlsZElkczogc3RyaW5nW10sIHNoYXJkSWQ6IG51bWJlcikgPT4gUHJvbWlzZTx2b2lkPlxuICB9XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgR2F0ZXdheU1hbmFnZXIgZXh0ZW5kcyBSZXF1aXJlZDxDcmVhdGVHYXRld2F5TWFuYWdlck9wdGlvbnM+IHtcbiAgLyoqIFRoZSBtYXggY29uY3VycmVuY3kgYnVja2V0cy4gVGhvc2Ugd2lsbCBiZSBjcmVhdGVkIHdoZW4gdGhlIGBzcGF3blNoYXJkc2AgKHdoaWNoIGNhbGxzIGBwcmVwYXJlQnVja2V0c2AgdW5kZXIgdGhlIGhvb2QpIGZ1bmN0aW9uIGdldHMgY2FsbGVkLiAqL1xuICBidWNrZXRzOiBNYXA8XG4gICAgbnVtYmVyLFxuICAgIHtcbiAgICAgIHdvcmtlcnM6IEFycmF5PHsgaWQ6IG51bWJlcjsgcXVldWU6IG51bWJlcltdIH0+XG4gICAgICAvKiogUmVxdWVzdHMgdG8gaWRlbnRpZnkgc2hhcmRzIGFyZSBtYWRlIGJhc2VkIG9uIHdoZXRoZXIgaXQgaXMgYXZhaWxhYmxlIHRvIGJlIG1hZGUuICovXG4gICAgICBpZGVudGlmeVJlcXVlc3RzOiBBcnJheTwodmFsdWU6IHZvaWQgfCBQcm9taXNlTGlrZTx2b2lkPikgPT4gdm9pZD5cbiAgICB9XG4gID5cbiAgLyoqIFRoZSBzaGFyZHMgdGhhdCBhcmUgY3JlYXRlZC4gKi9cbiAgc2hhcmRzOiBNYXA8bnVtYmVyLCBTaGFyZD5cbiAgLyoqIFRoZSBsb2dnZXIgZm9yIHRoZSBnYXRld2F5IG1hbmFnZXIuICovXG4gIGxvZ2dlcjogUGljazx0eXBlb2YgbG9nZ2VyLCAnZGVidWcnIHwgJ2luZm8nIHwgJ3dhcm4nIHwgJ2Vycm9yJyB8ICdmYXRhbCc+XG4gIC8qKiBFdmVyeXRoaW5nIHJlbGF0ZWQgdG8gcmVzaGFyZGluZy4gKi9cbiAgcmVzaGFyZGluZzogQ3JlYXRlR2F0ZXdheU1hbmFnZXJPcHRpb25zWydyZXNoYXJkaW5nJ10gJiB7XG4gICAgLyoqXG4gICAgICogVGhlIGludGVydmFsIGlkIG9mIHRoZSBjaGVjayBpbnRlcnZhbC4gVGhpcyBpcyB1c2VkIHRvIGNsZWFyIHRoZSBpbnRlcnZhbCB3aGVuIHRoZSBtYW5hZ2VyIGlzIHNodXRkb3duLlxuICAgICAqL1xuICAgIGNoZWNrSW50ZXJ2YWxJZD86IE5vZGVKUy5UaW1lb3V0IHwgdW5kZWZpbmVkXG4gICAgLyoqIEhvbGRzIHRoZSBzaGFyZHMgdGhhdCByZXNoYXJkaW5nIGhhcyBjcmVhdGVkLiBPbmNlIHJlc2hhcmRpbmcgaXMgZG9uZSwgdGhpcyByZXBsYWNlcyB0aGUgZ2F0ZXdheS5zaGFyZHMgKi9cbiAgICBzaGFyZHM6IENvbGxlY3Rpb248bnVtYmVyLCBTaGFyZD5cbiAgICAvKiogSG9sZHMgdGhlIHBlbmRpbmcgc2hhcmRzIHRoYXQgaGF2ZSBiZWVuIGNyZWF0ZWQgYW5kIGFyZSBwZW5kaW5nIGFsbCBzaGFyZHMgZmluaXNoIGxvYWRpbmcuICovXG4gICAgcGVuZGluZ1NoYXJkczogQ29sbGVjdGlvbjxudW1iZXIsIFNoYXJkPlxuICAgIC8qKiBIYW5kbGVyIHRvIGNoZWNrIGlmIHJlc2hhcmRpbmcgaXMgbmVjZXNzYXJ5LiAqL1xuICAgIGNoZWNrSWZSZXNoYXJkaW5nSXNOZWVkZWQ6ICgpID0+IFByb21pc2U8eyBuZWVkZWQ6IGJvb2xlYW47IGluZm8/OiBDYW1lbGl6ZTxEaXNjb3JkR2V0R2F0ZXdheUJvdD4gfT5cbiAgICAvKiogSGFuZGxlciB0byBiZWdpbiByZXNoYXJkaW5nLiAqL1xuICAgIHJlc2hhcmQ6IChpbmZvOiBDYW1lbGl6ZTxEaXNjb3JkR2V0R2F0ZXdheUJvdD4gJiB7IGZpcnN0U2hhcmRJZD86IG51bWJlcjsgbGFzdFNoYXJkSWQ/OiBudW1iZXIgfSkgPT4gUHJvbWlzZTx2b2lkPlxuICAgIC8qKiBIYW5kbGVyIHRvIGNvbW11bmljYXRlIHRvIGEgd29ya2VyIHRoYXQgYSBzaGFyZCBuZWVkcyB0byBiZSBjcmVhdGVkLiAqL1xuICAgIHRlbGxXb3JrZXJUb1ByZXBhcmU6ICh3b3JrZXJJZDogbnVtYmVyLCBzaGFyZElkOiBudW1iZXIsIGJ1Y2tldElkOiBudW1iZXIpID0+IFByb21pc2U8dm9pZD5cbiAgICAvKiogSGFuZGxlciB0byBhbGVydCB0aGUgZ2F0ZXdheSB0aGF0IGEgc2hhcmQocmVzaGFyZGVkKSBpcyBvbmxpbmUuIEl0IHNob3VsZCBub3cgd2FpdCBmb3IgYWxsIHNoYXJkcyB0byBiZSBwZW5kaW5nIGJlZm9yZSBzaHV0dGluZyBvZmYgb2xkIHNoYXJkcy4gKi9cbiAgICBzaGFyZElzUGVuZGluZzogKHNoYXJkOiBTaGFyZCkgPT4gUHJvbWlzZTx2b2lkPlxuICB9XG4gIC8qKiBEZXRlcm1pbmUgbWF4IG51bWJlciBvZiBzaGFyZHMgdG8gdXNlIGJhc2VkIHVwb24gdGhlIG1heCBjb25jdXJyZW5jeS4gKi9cbiAgY2FsY3VsYXRlVG90YWxTaGFyZHM6ICgpID0+IG51bWJlclxuICAvKiogRGV0ZXJtaW5lIHRoZSBpZCBvZiB0aGUgd29ya2VyIHdoaWNoIGlzIGhhbmRsaW5nIGEgc2hhcmQuICovXG4gIGNhbGN1bGF0ZVdvcmtlcklkOiAoc2hhcmRJZDogbnVtYmVyKSA9PiBudW1iZXJcbiAgLyoqIFByZXBhcmVzIGFsbCB0aGUgYnVja2V0cyB0aGF0IGFyZSBhdmFpbGFibGUgZm9yIGlkZW50aWZ5aW5nIHRoZSBzaGFyZHMuICovXG4gIHByZXBhcmVCdWNrZXRzOiAoKSA9PiB2b2lkXG4gIC8qKiBTdGFydCBpZGVudGlmeWluZyBhbGwgdGhlIHNoYXJkcy4gKi9cbiAgc3Bhd25TaGFyZHM6ICgpID0+IFByb21pc2U8dm9pZD5cbiAgLyoqIFNodXRkb3duIGFsbCBzaGFyZHMuICovXG4gIHNodXRkb3duOiAoY29kZTogbnVtYmVyLCByZWFzb246IHN0cmluZywgY2xlYXJSZXNoYXJkaW5nSW50ZXJ2YWw/OiBib29sZWFuKSA9PiBQcm9taXNlPHZvaWQ+XG4gIHNlbmRQYXlsb2FkOiAoc2hhcmRJZDogbnVtYmVyLCBwYXlsb2FkOiBTaGFyZFNvY2tldFJlcXVlc3QpID0+IFByb21pc2U8dm9pZD5cbiAgLyoqIEFsbG93cyB1c2VycyB0byBob29rIGluIGFuZCBjaGFuZ2UgdG8gY29tbXVuaWNhdGUgdG8gZGlmZmVyZW50IHdvcmtlcnMgYWNyb3NzIGRpZmZlcmVudCBzZXJ2ZXJzIG9yIGFueXRoaW5nIHRoZXkgbGlrZS4gRm9yIGV4YW1wbGUgdXNpbmcgcmVkaXMgcHVic3ViIHRvIHRhbGsgdG8gb3RoZXIgc2VydmVycy4gKi9cbiAgdGVsbFdvcmtlclRvSWRlbnRpZnk6ICh3b3JrZXJJZDogbnVtYmVyLCBzaGFyZElkOiBudW1iZXIsIGJ1Y2tldElkOiBudW1iZXIpID0+IFByb21pc2U8dm9pZD5cbiAgLyoqIFRlbGwgdGhlIG1hbmFnZXIgdG8gaWRlbnRpZnkgYSBTaGFyZC4gSWYgdGhpcyBTaGFyZCBpcyBub3QgYWxyZWFkeSBtYW5hZ2VkIHRoaXMgd2lsbCBhbHNvIGFkZCB0aGUgU2hhcmQgdG8gdGhlIG1hbmFnZXIuICovXG4gIGlkZW50aWZ5OiAoc2hhcmRJZDogbnVtYmVyKSA9PiBQcm9taXNlPHZvaWQ+XG4gIC8qKiBLaWxsIGEgc2hhcmQuIENsb3NlIGEgc2hhcmRzIGNvbm5lY3Rpb24gdG8gRGlzY29yZCdzIGdhdGV3YXkgKGlmIGFueSkgYW5kIHJlbW92ZSBpdCBmcm9tIHRoZSBtYW5hZ2VyLiAqL1xuICBraWxsOiAoc2hhcmRJZDogbnVtYmVyKSA9PiBQcm9taXNlPHZvaWQ+XG4gIC8qKiBUaGlzIGZ1bmN0aW9uIG1ha2VzIHN1cmUgdGhhdCB0aGUgYnVja2V0IGlzIGFsbG93ZWQgdG8gbWFrZSB0aGUgbmV4dCBpZGVudGlmeSByZXF1ZXN0LiAqL1xuICByZXF1ZXN0SWRlbnRpZnk6IChzaGFyZElkOiBudW1iZXIpID0+IFByb21pc2U8dm9pZD5cbiAgLyoqIENhbGN1bGF0ZXMgdGhlIG51bWJlciBvZiBzaGFyZHMgYmFzZWQgb24gdGhlIGd1aWxkIGlkIGFuZCB0b3RhbCBzaGFyZHMuICovXG4gIGNhbGN1bGF0ZVNoYXJkSWQ6IChndWlsZElkOiBCaWdTdHJpbmcsIHRvdGFsU2hhcmRzPzogbnVtYmVyKSA9PiBudW1iZXJcbiAgLyoqXG4gICAqIENvbm5lY3RzIHRoZSBib3QgdXNlciB0byBhIHZvaWNlIG9yIHN0YWdlIGNoYW5uZWwuXG4gICAqXG4gICAqIFRoaXMgZnVuY3Rpb24gc2VuZHMgdGhlIF9VcGRhdGUgVm9pY2UgU3RhdGVfIGdhdGV3YXkgY29tbWFuZCBvdmVyIHRoZSBnYXRld2F5IGJlaGluZCB0aGUgc2NlbmVzLlxuICAgKlxuICAgKiBAcGFyYW0gZ3VpbGRJZCAtIFRoZSBJRCBvZiB0aGUgZ3VpbGQgdGhlIHZvaWNlIGNoYW5uZWwgdG8gbGVhdmUgaXMgaW4uXG4gICAqIEBwYXJhbSBjaGFubmVsSWQgLSBUaGUgSUQgb2YgdGhlIGNoYW5uZWwgeW91IHdhbnQgdG8gam9pbi5cbiAgICpcbiAgICogQHJlbWFya3NcbiAgICogUmVxdWlyZXMgdGhlIGBDT05ORUNUYCBwZXJtaXNzaW9uLlxuICAgKlxuICAgKiBGaXJlcyBhIF9Wb2ljZSBTdGF0ZSBVcGRhdGVfIGdhdGV3YXkgZXZlbnQuXG4gICAqXG4gICAqIEBzZWUge0BsaW5rIGh0dHBzOi8vZGlzY29yZC5jb20vZGV2ZWxvcGVycy9kb2NzL3RvcGljcy9nYXRld2F5I3VwZGF0ZS12b2ljZS1zdGF0ZX1cbiAgICovXG4gIGpvaW5Wb2ljZUNoYW5uZWw6IChndWlsZElkOiBCaWdTdHJpbmcsIGNoYW5uZWxJZDogQmlnU3RyaW5nLCBvcHRpb25zPzogQXRMZWFzdE9uZTxPbWl0PFVwZGF0ZVZvaWNlU3RhdGUsICdndWlsZElkJyB8ICdjaGFubmVsSWQnPj4pID0+IFByb21pc2U8dm9pZD5cbiAgLyoqXG4gICAqIEVkaXRzIHRoZSBib3Qgc3RhdHVzIGluIGFsbCBzaGFyZHMgdGhhdCB0aGlzIGdhdGV3YXkgbWFuYWdlcy5cbiAgICpcbiAgICogQHBhcmFtIGRhdGEgVGhlIHN0YXR1cyBkYXRhIHRvIHNldCB0aGUgYm90cyBzdGF0dXMgdG8uXG4gICAqIEByZXR1cm5zIG5vdGhpbmdcbiAgICovXG4gIGVkaXRCb3RTdGF0dXM6IChkYXRhOiBTdGF0dXNVcGRhdGUpID0+IFByb21pc2U8dm9pZD5cbiAgLyoqXG4gICAqIEVkaXRzIHRoZSBib3QncyBzdGF0dXMgb24gb25lIHNoYXJkLlxuICAgKlxuICAgKiBAcGFyYW0gc2hhcmRJZCBUaGUgc2hhcmQgaWQgdG8gZWRpdCB0aGUgc3RhdHVzIGZvci5cbiAgICogQHBhcmFtIGRhdGEgVGhlIHN0YXR1cyBkYXRhIHRvIHNldCB0aGUgYm90cyBzdGF0dXMgdG8uXG4gICAqIEByZXR1cm5zIG5vdGhpbmdcbiAgICovXG4gIGVkaXRTaGFyZFN0YXR1czogKHNoYXJkSWQ6IG51bWJlciwgZGF0YTogU3RhdHVzVXBkYXRlKSA9PiBQcm9taXNlPHZvaWQ+XG4gIC8qKlxuICAgKiBGZXRjaGVzIHRoZSBsaXN0IG9mIG1lbWJlcnMgZm9yIGEgZ3VpbGQgb3ZlciB0aGUgZ2F0ZXdheS5cbiAgICpcbiAgICogQHBhcmFtIGd1aWxkSWQgLSBUaGUgSUQgb2YgdGhlIGd1aWxkIHRvIGdldCB0aGUgbGlzdCBvZiBtZW1iZXJzIGZvci5cbiAgICogQHBhcmFtIG9wdGlvbnMgLSBUaGUgcGFyYW1ldGVycyBmb3IgdGhlIGZldGNoaW5nIG9mIHRoZSBtZW1iZXJzLlxuICAgKlxuICAgKiBAcmVtYXJrc1xuICAgKiBJZiByZXF1ZXN0aW5nIHRoZSBlbnRpcmUgbWVtYmVyIGxpc3Q6XG4gICAqIC0gUmVxdWlyZXMgdGhlIGBHVUlMRF9NRU1CRVJTYCBpbnRlbnQuXG4gICAqXG4gICAqIElmIHJlcXVlc3RpbmcgcHJlc2VuY2VzICh7QGxpbmsgUmVxdWVzdEd1aWxkTWVtYmVycy5wcmVzZW5jZXMgfCBwcmVzZW5jZXN9IHNldCB0byBgdHJ1ZWApOlxuICAgKiAtIFJlcXVpcmVzIHRoZSBgR1VJTERfUFJFU0VOQ0VTYCBpbnRlbnQuXG4gICAqXG4gICAqIElmIHJlcXVlc3RpbmcgYSBwcmVmaXggKHtAbGluayBSZXF1ZXN0R3VpbGRNZW1iZXJzLnF1ZXJ5IHwgcXVlcnl9IG5vbi1gdW5kZWZpbmVkYCk6XG4gICAqIC0gUmV0dXJucyBhIG1heGltdW0gb2YgMTAwIG1lbWJlcnMuXG4gICAqXG4gICAqIElmIHJlcXVlc3RpbmcgYSB1c2VycyBieSBJRCAoe0BsaW5rIFJlcXVlc3RHdWlsZE1lbWJlcnMudXNlcklkcyB8IHVzZXJJZHN9IG5vbi1gdW5kZWZpbmVkYCk6XG4gICAqIC0gUmV0dXJucyBhIG1heGltdW0gb2YgMTAwIG1lbWJlcnMuXG4gICAqXG4gICAqIEZpcmVzIGEgX0d1aWxkIE1lbWJlcnMgQ2h1bmtfIGdhdGV3YXkgZXZlbnQgZm9yIGV2ZXJ5IDEwMDAgbWVtYmVycyBmZXRjaGVkLlxuICAgKlxuICAgKiBAc2VlIHtAbGluayBodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3MvZ2F0ZXdheSNyZXF1ZXN0LWd1aWxkLW1lbWJlcnN9XG4gICAqL1xuICByZXF1ZXN0TWVtYmVyczogKGd1aWxkSWQ6IEJpZ1N0cmluZywgb3B0aW9ucz86IE9taXQ8UmVxdWVzdEd1aWxkTWVtYmVycywgJ2d1aWxkSWQnPikgPT4gUHJvbWlzZTxDYW1lbGl6ZTxEaXNjb3JkTWVtYmVyV2l0aFVzZXJbXT4+XG4gIC8qKlxuICAgKiBMZWF2ZXMgdGhlIHZvaWNlIGNoYW5uZWwgdGhlIGJvdCB1c2VyIGlzIGN1cnJlbnRseSBpbi5cbiAgICpcbiAgICogVGhpcyBmdW5jdGlvbiBzZW5kcyB0aGUgX1VwZGF0ZSBWb2ljZSBTdGF0ZV8gZ2F0ZXdheSBjb21tYW5kIG92ZXIgdGhlIGdhdGV3YXkgYmVoaW5kIHRoZSBzY2VuZXMuXG4gICAqXG4gICAqIEBwYXJhbSBndWlsZElkIC0gVGhlIElEIG9mIHRoZSBndWlsZCB0aGUgdm9pY2UgY2hhbm5lbCB0byBsZWF2ZSBpcyBpbi5cbiAgICpcbiAgICogQHJlbWFya3NcbiAgICogRmlyZXMgYSBfVm9pY2UgU3RhdGUgVXBkYXRlXyBnYXRld2F5IGV2ZW50LlxuICAgKlxuICAgKiBAc2VlIHtAbGluayBodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3MvZ2F0ZXdheSN1cGRhdGUtdm9pY2Utc3RhdGV9XG4gICAqL1xuICBsZWF2ZVZvaWNlQ2hhbm5lbDogKGd1aWxkSWQ6IEJpZ1N0cmluZykgPT4gUHJvbWlzZTx2b2lkPlxuICAvKipcbiAgICogVXNlZCB0byByZXF1ZXN0IHNvdW5kYm9hcmQgc291bmRzIGZvciBhIGxpc3Qgb2YgZ3VpbGRzLlxuICAgKlxuICAgKiBUaGlzIGZ1bmN0aW9uIHNlbmRzIG11bHRpcGxlIChzZWUgcmVtYXJrcykgX1JlcXVlc3QgU291bmRib2FyZCBTb3VuZHNfIGdhdGV3YXkgY29tbWFuZCBvdmVyIHRoZSBnYXRld2F5IGJlaGluZCB0aGUgc2NlbmVzLlxuICAgKlxuICAgKiBAcGFyYW0gZ3VpbGRJZHMgLSBUaGUgZ3VpbGRzIHRvIGdldCB0aGUgc291bmRzIGZyb21cbiAgICpcbiAgICogQHJlbWFya3NcbiAgICogRmlyZXMgYSBfU291bmRib2FyZCBTb3VuZHNfIGdhdGV3YXkgZXZlbnQuXG4gICAqXG4gICAqIOKaoO+4jyBEaXNjb3JkIHdpbGwgc2VuZCB0aGUgX1NvdW5kYm9hcmQgU291bmRzXyBmb3IgZWFjaCBvZiB0aGUgZ3VpbGQgaWRzXG4gICAqIGhvd2V2ZXIgeW91IG1heSBub3QgcmVjZWl2ZSB0aGUgc2FtZSBudW1iZXIgb2YgZXZlbnRzIGFzIHRoZSBpZHMgcGFzc2VkIHRvIF9SZXF1ZXN0IFNvdW5kYm9hcmQgU291bmRzXyBmb3Igb25lIG9mIHRoZSBmb2xsb3dpbmcgcmVhc29uczpcbiAgICogLSBUaGUgYm90IGlzIG5vdCBpbiB0aGUgc2VydmVyIHByb3ZpZGVkXG4gICAqIC0gVGhlIHNoYXJkIHRoZSBtZXNzYWdlIGhhcyBiZWVuIHNlbnQgZnJvbSBkb2VzIG5vdCByZWNlaXZlIGV2ZW50cyBmb3IgdGhlIHNwZWNpZmllZCBndWlsZFxuICAgKlxuICAgKiBUbyBhdm9pZCB0aGlzIERpc2NvcmRlbm8gd2lsbCBhdXRvbWF0aWNhbGx5IHRyeSB0byBncm91cCB0aGUgaWRzIGJhc2VkIG9uIHdoYXQgc2hhcmQgdGhleSB3aWxsIG5lZWQgdG8gYmUgc2VudCwgYnV0IHRoaXMgaW52b2x2ZXMgc2VuZGluZyBtdWx0aXBsZSBtZXNzYWdlcyBpbiBtdWx0aXBsZSBzaGFyZHNcbiAgICpcbiAgICogQHNlZSB7QGxpbmsgaHR0cHM6Ly9kaXNjb3JkLmNvbS9kZXZlbG9wZXJzL2RvY3MvdG9waWNzL2dhdGV3YXktZXZlbnRzI3JlcXVlc3Qtc291bmRib2FyZC1zb3VuZHN9XG4gICAqL1xuICByZXF1ZXN0U291bmRib2FyZFNvdW5kczogKGd1aWxkSWRzOiBCaWdTdHJpbmdbXSkgPT4gUHJvbWlzZTx2b2lkPlxuICAvKiogVGhpcyBtYW5hZ2VycyBjYWNoZSByZWxhdGVkIHNldHRpbmdzLiAqL1xuICBjYWNoZToge1xuICAgIHJlcXVlc3RNZW1iZXJzOiB7XG4gICAgICAvKipcbiAgICAgICAqIFdoZXRoZXIgb3Igbm90IHJlcXVlc3QgbWVtYmVyIHJlcXVlc3RzIHNob3VsZCBiZSBjYWNoZWQuXG4gICAgICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgICAgICovXG4gICAgICBlbmFibGVkOiBib29sZWFuXG4gICAgICAvKiogVGhlIHBlbmRpbmcgcmVxdWVzdHMuICovXG4gICAgICBwZW5kaW5nOiBDb2xsZWN0aW9uPHN0cmluZywgUmVxdWVzdE1lbWJlclJlcXVlc3Q+XG4gICAgfVxuICB9XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVxdWVzdE1lbWJlclJlcXVlc3Qge1xuICAvKiogVGhlIHVuaXF1ZSBub25jZSBmb3IgdGhpcyByZXF1ZXN0LiAqL1xuICBub25jZTogc3RyaW5nXG4gIC8qKiBUaGUgcmVzb2x2ZXIgaGFuZGxlciB0byBydW4gd2hlbiBhbGwgbWVtYmVycyBhcnJpdmUuICovXG4gIHJlc29sdmU6ICh2YWx1ZTogQ2FtZWxpemU8RGlzY29yZE1lbWJlcldpdGhVc2VyW10+IHwgUHJvbWlzZUxpa2U8Q2FtZWxpemU8RGlzY29yZE1lbWJlcldpdGhVc2VyW10+PikgPT4gdm9pZFxuICAvKiogVGhlIG1lbWJlcnMgdGhhdCBoYXZlIGFscmVhZHkgYXJyaXZlZCBmb3IgdGhpcyByZXF1ZXN0LiAqL1xuICBtZW1iZXJzOiBEaXNjb3JkTWVtYmVyV2l0aFVzZXJbXVxufVxuIl0sIm5hbWVzIjpbImNyZWF0ZUdhdGV3YXlNYW5hZ2VyIiwib3B0aW9ucyIsImNvbm5lY3Rpb25PcHRpb25zIiwiY29ubmVjdGlvbiIsInVybCIsInNoYXJkcyIsInNlc3Npb25TdGFydExpbWl0IiwibWF4Q29uY3VycmVuY3kiLCJyZW1haW5pbmciLCJ0b3RhbCIsInJlc2V0QWZ0ZXIiLCJnYXRld2F5IiwiZXZlbnRzIiwiY29tcHJlc3MiLCJ0cmFuc3BvcnRDb21wcmVzc2lvbiIsImludGVudHMiLCJwcm9wZXJ0aWVzIiwib3MiLCJwcm9jZXNzIiwicGxhdGZvcm0iLCJicm93c2VyIiwiZGV2aWNlIiwidG9rZW4iLCJ2ZXJzaW9uIiwidG90YWxTaGFyZHMiLCJsYXN0U2hhcmRJZCIsImZpcnN0U2hhcmRJZCIsInRvdGFsV29ya2VycyIsInNoYXJkc1BlcldvcmtlciIsInNwYXduU2hhcmREZWxheSIsInByZWZlclNuYWtlQ2FzZSIsIk1hcCIsImJ1Y2tldHMiLCJjYWNoZSIsInJlcXVlc3RNZW1iZXJzIiwiZW5hYmxlZCIsInBlbmRpbmciLCJDb2xsZWN0aW9uIiwibG9nZ2VyIiwibWFrZVByZXNlbmNlIiwiUHJvbWlzZSIsInJlc29sdmUiLCJ1bmRlZmluZWQiLCJyZXNoYXJkaW5nIiwic2hhcmRzRnVsbFBlcmNlbnRhZ2UiLCJjaGVja0ludGVydmFsIiwicGVuZGluZ1NoYXJkcyIsImdldFNlc3Npb25JbmZvIiwidXBkYXRlR3VpbGRzU2hhcmRJZCIsImNoZWNrSWZSZXNoYXJkaW5nSXNOZWVkZWQiLCJkZWJ1ZyIsIm5lZWRlZCIsIkVycm9yIiwic2Vzc2lvbkluZm8iLCJKU09OIiwic3RyaW5naWZ5IiwiaW5mbyIsInBlcmNlbnRhZ2UiLCJyZXNoYXJkIiwiY2FsY3VsYXRlVG90YWxTaGFyZHMiLCJjbGVhciIsInByZXBhcmVCdWNrZXRzIiwiZm9yRWFjaCIsImJ1Y2tldCIsImJ1Y2tldElkIiwid29ya2VyIiwid29ya2VycyIsInNoYXJkSWQiLCJxdWV1ZSIsInRlbGxXb3JrZXJUb1ByZXBhcmUiLCJpZCIsIndvcmtlcklkIiwic2hhcmQiLCJTaGFyZCIsIm1lc3NhZ2UiLCJfc2hhcmQiLCJwYXlsb2FkIiwidCIsImQiLCJndWlsZHMiLCJtYXAiLCJnIiwicmVxdWVzdElkZW50aWZ5IiwiaWRlbnRpZnkiLCJzaGFyZElzUmVhZHkiLCJkZWxheSIsImdldCIsImlkZW50aWZ5UmVxdWVzdHMiLCJzaGlmdCIsImZvcndhcmRUb0JvdCIsInNldCIsInB1c2giLCJ0aGVuIiwic2hhcmRJc1BlbmRpbmciLCJzaXplIiwidmFsdWVzIiwiZXZlbnQiLCJvbGRIYW5kbGVyIiwiXyIsInNodXRkb3duIiwiU2hhcmRTb2NrZXRDbG9zZUNvZGVzIiwiUmVzaGFyZGVkIiwiTWF0aCIsImNlaWwiLCJjYWxjdWxhdGVXb3JrZXJJZCIsIm1pbiIsImkiLCJmaW5kIiwidyIsInNvcnQiLCJhIiwiYiIsInNwYXduU2hhcmRzIiwiYWxsIiwiZW50cmllcyIsInRlbGxXb3JrZXJUb0lkZW50aWZ5IiwiY2xlYXJJbnRlcnZhbCIsImNoZWNrSW50ZXJ2YWxJZCIsIndhcm4iLCJzZXRJbnRlcnZhbCIsInJlc2hhcmRpbmdJbmZvIiwiY29kZSIsInJlYXNvbiIsImNsZWFyUmVzaGFyZGluZ0ludGVydmFsIiwiY2xvc2UiLCJzZW5kUGF5bG9hZCIsInNlbmQiLCJraWxsIiwiZGVsZXRlIiwiX3NoYXJkSWQiLCJjYWxjdWxhdGVTaGFyZElkIiwiZ3VpbGRJZCIsIk51bWJlciIsIkJpZ0ludCIsImpvaW5Wb2ljZUNoYW5uZWwiLCJjaGFubmVsSWQiLCJvcCIsIkdhdGV3YXlPcGNvZGVzIiwiVm9pY2VTdGF0ZVVwZGF0ZSIsImd1aWxkX2lkIiwidG9TdHJpbmciLCJjaGFubmVsX2lkIiwic2VsZl9tdXRlIiwic2VsZk11dGUiLCJzZWxmX2RlYWYiLCJzZWxmRGVhZiIsImVkaXRCb3RTdGF0dXMiLCJkYXRhIiwiZWRpdFNoYXJkU3RhdHVzIiwiUHJlc2VuY2VVcGRhdGUiLCJzaW5jZSIsImFmayIsImFjdGl2aXRpZXMiLCJzdGF0dXMiLCJsaW1pdCIsIkdhdGV3YXlJbnRlbnRzIiwiR3VpbGRNZW1iZXJzIiwidXNlcklkcyIsImxlbmd0aCIsIm1lbWJlcnMiLCJub25jZSIsInJlamVjdCIsIlJlcXVlc3RHdWlsZE1lbWJlcnMiLCJxdWVyeSIsInByZXNlbmNlcyIsInVzZXJfaWRzIiwibGVhdmVWb2ljZUNoYW5uZWwiLCJyZXF1ZXN0U291bmRib2FyZFNvdW5kcyIsImd1aWxkSWRzIiwiaWRzIiwiUmVxdWVzdFNvdW5kYm9hcmRTb3VuZHMiLCJndWlsZF9pZHMiXSwibWFwcGluZ3MiOiI7Ozs7K0JBdUJnQkE7OztlQUFBQTs7O3VCQWJUO3VCQUNtQzs4REFDeEI7d0JBU1g7Ozs7OztBQUVBLFNBQVNBLHFCQUFxQkMsT0FBb0M7SUFDdkUsTUFBTUMsb0JBQW9CRCxRQUFRRSxVQUFVLElBQUk7UUFDOUNDLEtBQUs7UUFDTEMsUUFBUTtRQUNSQyxtQkFBbUI7WUFDakJDLGdCQUFnQjtZQUNoQkMsV0FBVztZQUNYQyxPQUFPO1lBQ1BDLFlBQVksT0FBTyxLQUFLLEtBQUs7UUFDL0I7SUFDRjtJQUVBLE1BQU1DLFVBQTBCO1FBQzlCQyxRQUFRWCxRQUFRVyxNQUFNLElBQUksQ0FBQztRQUMzQkMsVUFBVVosUUFBUVksUUFBUSxJQUFJO1FBQzlCQyxzQkFBc0JiLFFBQVFhLG9CQUFvQixJQUFJO1FBQ3REQyxTQUFTZCxRQUFRYyxPQUFPLElBQUk7UUFDNUJDLFlBQVk7WUFDVkMsSUFBSWhCLFFBQVFlLFVBQVUsRUFBRUMsTUFBTUMsUUFBUUMsUUFBUTtZQUM5Q0MsU0FBU25CLFFBQVFlLFVBQVUsRUFBRUksV0FBVztZQUN4Q0MsUUFBUXBCLFFBQVFlLFVBQVUsRUFBRUssVUFBVTtRQUN4QztRQUNBQyxPQUFPckIsUUFBUXFCLEtBQUs7UUFDcEJsQixLQUFLSCxRQUFRRyxHQUFHLElBQUlGLGtCQUFrQkUsR0FBRyxJQUFJO1FB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