@discordeno/gateway 19.0.0-next.e2d86ea → 19.0.0-next.e33b44e

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,489 @@
1
+ /* eslint-disable @typescript-eslint/no-confusing-void-expression */ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ function _export(target, all) {
6
+ for(var name in all)Object.defineProperty(target, name, {
7
+ enumerable: true,
8
+ get: all[name]
9
+ });
10
+ }
11
+ _export(exports, {
12
+ DiscordenoShard: function() {
13
+ return DiscordenoShard;
14
+ },
15
+ default: function() {
16
+ return _default;
17
+ }
18
+ });
19
+ const _types = require("@discordeno/types");
20
+ const _utils = require("@discordeno/utils");
21
+ const _nodezlib = require("node:zlib");
22
+ const _ws = /*#__PURE__*/ _interop_require_default(require("ws"));
23
+ const _types1 = require("./types.cjs");
24
+ function _interop_require_default(obj) {
25
+ return obj && obj.__esModule ? obj : {
26
+ default: obj
27
+ };
28
+ }
29
+ let DiscordenoShard = class DiscordenoShard {
30
+ constructor(options){
31
+ /** The maximum of requests which can be send to discord per rate limit tick. Typically this value should not be changed. */ this.maxRequestsPerRateLimitTick = 120;
32
+ /** The previous payload sequence number. */ this.previousSequenceNumber = null;
33
+ /** In which interval (in milliseconds) the gateway resets it's rate limit. */ this.rateLimitResetInterval = 60000;
34
+ /** Current internal state of the this. */ this.state = _types1.ShardState.Offline;
35
+ /** The url provided by discord to use when resuming a connection for this this. */ this.resumeGatewayUrl = '';
36
+ /** The shard related event handlers. */ this.events = {};
37
+ /** Cache for pending gateway requests which should have been send while the gateway went offline. */ this.offlineSendQueue = [];
38
+ /** Resolve internal waiting states. Mapped by SelectedEvents => ResolveFunction */ this.resolves = new Map();
39
+ this.id = options.id;
40
+ this.connection = options.connection;
41
+ this.events = options.events;
42
+ this.logger = options.logger ?? _utils.logger;
43
+ this.heart = {
44
+ acknowledged: false,
45
+ interval: 45000
46
+ };
47
+ if (options.requestIdentify) this.requestIdentify = options.requestIdentify;
48
+ if (options.shardIsReady) this.shardIsReady = options.shardIsReady;
49
+ this.bucket = new _utils.LeakyBucket({
50
+ max: this.calculateSafeRequests(),
51
+ refillAmount: this.calculateSafeRequests(),
52
+ refillInterval: 60000,
53
+ logger: this.logger
54
+ });
55
+ }
56
+ /** The gateway configuration which is used to connect to Discord. */ get gatewayConfig() {
57
+ return this.connection;
58
+ }
59
+ /** The url to connect to. Initially this is the discord gateway url, and then is switched to resume gateway url once a READY is received. */ get connectionUrl() {
60
+ // Use || and not ?? here. ?? will cause a bug.
61
+ return this.resumeGatewayUrl || this.gatewayConfig.url;
62
+ }
63
+ /** Calculate the amount of requests which can safely be made per rate limit interval, before the gateway gets disconnected due to an exceeded rate limit. */ calculateSafeRequests() {
64
+ // * 2 adds extra safety layer for discords OP 1 requests that we need to respond to
65
+ const safeRequests = this.maxRequestsPerRateLimitTick - Math.ceil(this.rateLimitResetInterval / this.heart.interval) * 2;
66
+ return safeRequests < 0 ? 0 : safeRequests;
67
+ }
68
+ async checkOffline(highPriority) {
69
+ if (!this.isOpen()) {
70
+ await new Promise((resolve)=>{
71
+ // Higher priority requests get added at the beginning of the array.
72
+ if (highPriority) this.offlineSendQueue.unshift(resolve);
73
+ else this.offlineSendQueue.push(resolve);
74
+ });
75
+ }
76
+ }
77
+ /** Close the socket connection to discord if present. */ close(code, reason) {
78
+ if (this.socket?.readyState !== _ws.default.OPEN) return;
79
+ this.socket?.close(code, reason);
80
+ }
81
+ /** Connect the shard with the gateway and start heartbeating. This will not identify the shard to the gateway. */ async connect() {
82
+ // Only set the shard to `Connecting` state,
83
+ // if the connection request does not come from an identify or resume action.
84
+ if (![
85
+ _types1.ShardState.Identifying,
86
+ _types1.ShardState.Resuming
87
+ ].includes(this.state)) {
88
+ this.state = _types1.ShardState.Connecting;
89
+ }
90
+ this.events.connecting?.(this);
91
+ const url = new URL(this.connectionUrl);
92
+ url.searchParams.set('v', this.gatewayConfig.version.toString());
93
+ url.searchParams.set('encoding', 'json');
94
+ const socket = // @ts-expect-error Deno
95
+ globalThis.Deno !== undefined && Reflect.has(globalThis, 'Deno') ? new WebSocket(url.toString()) : new _ws.default(url.toString());
96
+ this.socket = socket;
97
+ // TODO: proper event handling
98
+ socket.onerror = (event)=>console.log({
99
+ error: event,
100
+ shardId: this.id
101
+ });
102
+ socket.onclose = async (event)=>await this.handleClose(event);
103
+ socket.onmessage = async (message)=>await this.handleMessage(message);
104
+ return await new Promise((resolve)=>{
105
+ socket.onopen = ()=>{
106
+ // Only set the shard to `Unidentified` state,
107
+ // if the connection request does not come from an identify or resume action.
108
+ if (![
109
+ _types1.ShardState.Identifying,
110
+ _types1.ShardState.Resuming
111
+ ].includes(this.state)) {
112
+ this.state = _types1.ShardState.Unidentified;
113
+ }
114
+ this.events.connected?.(this);
115
+ resolve(this);
116
+ };
117
+ });
118
+ }
119
+ /** Identify the shard to the gateway. If not connected, this will also connect the shard to the gateway. */ async identify() {
120
+ // A new identify has been requested even though there is already a connection open.
121
+ // Therefore we need to close the old connection and heartbeating before creating a new one.
122
+ if (this.isOpen()) {
123
+ this.logger.debug(`CLOSING EXISTING SHARD: #${this.id}`);
124
+ this.close(_types1.ShardSocketCloseCodes.ReIdentifying, 'Re-identifying closure of old connection.');
125
+ }
126
+ this.state = _types1.ShardState.Identifying;
127
+ this.events.identifying?.(this);
128
+ // It is possible that the shard is in Heartbeating state but not identified,
129
+ // so check whether there is already a gateway connection existing.
130
+ // If not we need to create one before we identify.
131
+ if (!this.isOpen()) {
132
+ await this.connect();
133
+ }
134
+ this.send({
135
+ op: _types.GatewayOpcodes.Identify,
136
+ d: {
137
+ token: `Bot ${this.gatewayConfig.token}`,
138
+ compress: this.gatewayConfig.compress,
139
+ properties: this.gatewayConfig.properties,
140
+ intents: this.gatewayConfig.intents,
141
+ shard: [
142
+ this.id,
143
+ this.gatewayConfig.totalShards
144
+ ],
145
+ presence: await this.makePresence?.()
146
+ }
147
+ }, true);
148
+ return await new Promise((resolve)=>{
149
+ this.resolves.set('READY', ()=>{
150
+ this.events.identified?.(this);
151
+ // Tells the manager that this shard is ready
152
+ this.shardIsReady();
153
+ resolve();
154
+ });
155
+ // When identifying too fast,
156
+ // Discord sends an invalid session payload.
157
+ // This can safely be ignored though and the shard starts a new identify action.
158
+ this.resolves.set('INVALID_SESSION', ()=>{
159
+ this.resolves.delete('READY');
160
+ resolve();
161
+ });
162
+ });
163
+ }
164
+ /** Check whether the connection to Discord is currently open. */ isOpen() {
165
+ return this.socket?.readyState === _ws.default.OPEN;
166
+ }
167
+ /** Attempt to resume the previous shards session with the gateway. */ async resume() {
168
+ this.logger.debug(`[Gateway] Resuming Shard #${this.id}`);
169
+ // It has been requested to resume the Shards session.
170
+ // It's possible that the shard is still connected with Discord's gateway therefore we need to forcefully close it.
171
+ if (this.isOpen()) {
172
+ this.logger.debug(`[Gateway] Resuming Shard #${this.id} in isOpen`);
173
+ this.close(_types1.ShardSocketCloseCodes.ResumeClosingOldConnection, 'Reconnecting the shard, closing old connection.');
174
+ }
175
+ // Shard has never identified, so we cannot resume.
176
+ if (!this.sessionId) {
177
+ this.logger.debug(`[Shard] Trying to resume a shard #${this.id} that was NOT first identified. (No session id found)`);
178
+ return await this.identify();
179
+ }
180
+ this.state = _types1.ShardState.Resuming;
181
+ this.logger.debug(`[Gateway] Resuming Shard #${this.id}, before connecting`);
182
+ // Before we can resume, we need to create a new connection with Discord's gateway.
183
+ await this.connect();
184
+ this.logger.debug(// eslint-disable-next-line @typescript-eslint/restrict-template-expressions
185
+ `[Gateway] Resuming Shard #${this.id}, after connecting. ${this.sessionId} | ${this.previousSequenceNumber}`);
186
+ this.send({
187
+ op: _types.GatewayOpcodes.Resume,
188
+ d: {
189
+ token: `Bot ${this.gatewayConfig.token}`,
190
+ session_id: this.sessionId,
191
+ seq: this.previousSequenceNumber ?? 0
192
+ }
193
+ }, true);
194
+ this.logger.debug(`[Gateway] Resuming Shard #${this.id} after send resumg`);
195
+ return await new Promise((resolve)=>{
196
+ this.resolves.set('RESUMED', ()=>resolve());
197
+ // If it is attempted to resume with an invalid session id,
198
+ // Discord sends an invalid session payload
199
+ // Not erroring here since it is easy that this happens, also it would be not catchable
200
+ this.resolves.set('INVALID_SESSION', ()=>{
201
+ this.resolves.delete('RESUMED');
202
+ resolve();
203
+ });
204
+ });
205
+ }
206
+ /** Send a message to Discord.
207
+ * @param {boolean} [highPriority=false] - Whether this message should be send asap.
208
+ */ async send(message, highPriority = false) {
209
+ // Before acquiring a token from the bucket, check whether the shard is currently offline or not.
210
+ // Else bucket and token wait time just get wasted.
211
+ await this.checkOffline(highPriority);
212
+ await this.bucket.acquire(highPriority);
213
+ // It's possible, that the shard went offline after a token has been acquired from the bucket.
214
+ await this.checkOffline(highPriority);
215
+ this.socket?.send(JSON.stringify(message));
216
+ }
217
+ /** Shutdown the this. Forcefully disconnect the shard from Discord. The shard may not attempt to reconnect with Discord. */ async shutdown() {
218
+ this.close(_types1.ShardSocketCloseCodes.Shutdown, 'Shard shutting down.');
219
+ this.state = _types1.ShardState.Offline;
220
+ }
221
+ /** Handle a gateway connection close. */ async handleClose(close) {
222
+ // gateway.debug("GW CLOSED", { shardId, payload: event });
223
+ this.stopHeartbeating();
224
+ switch(close.code){
225
+ case _types1.ShardSocketCloseCodes.TestingFinished:
226
+ {
227
+ this.state = _types1.ShardState.Offline;
228
+ this.events.disconnected?.(this);
229
+ return;
230
+ }
231
+ // On these codes a manual start will be done.
232
+ case _types1.ShardSocketCloseCodes.Shutdown:
233
+ case _types1.ShardSocketCloseCodes.ReIdentifying:
234
+ case _types1.ShardSocketCloseCodes.Resharded:
235
+ case _types1.ShardSocketCloseCodes.ResumeClosingOldConnection:
236
+ case _types1.ShardSocketCloseCodes.ZombiedConnection:
237
+ {
238
+ this.state = _types1.ShardState.Disconnected;
239
+ this.events.disconnected?.(this);
240
+ // gateway.debug("GW CLOSED_RECONNECT", { shardId, payload: event });
241
+ return;
242
+ }
243
+ // Gateway connection closes which require a new identify.
244
+ case _types.GatewayCloseEventCodes.UnknownOpcode:
245
+ case _types.GatewayCloseEventCodes.NotAuthenticated:
246
+ case _types.GatewayCloseEventCodes.InvalidSeq:
247
+ case _types.GatewayCloseEventCodes.RateLimited:
248
+ case _types.GatewayCloseEventCodes.SessionTimedOut:
249
+ {
250
+ this.logger.debug(`[Shard] Gateway connection closing requiring re-identify. Code: ${close.code}`);
251
+ this.state = _types1.ShardState.Identifying;
252
+ this.events.disconnected?.(this);
253
+ return await this.identify();
254
+ }
255
+ // When these codes are received something went really wrong.
256
+ // On those we cannot start a reconnect attempt.
257
+ case _types.GatewayCloseEventCodes.AuthenticationFailed:
258
+ case _types.GatewayCloseEventCodes.InvalidShard:
259
+ case _types.GatewayCloseEventCodes.ShardingRequired:
260
+ case _types.GatewayCloseEventCodes.InvalidApiVersion:
261
+ case _types.GatewayCloseEventCodes.InvalidIntents:
262
+ case _types.GatewayCloseEventCodes.DisallowedIntents:
263
+ {
264
+ this.state = _types1.ShardState.Offline;
265
+ this.events.disconnected?.(this);
266
+ throw new Error(close.reason || 'Discord gave no reason! GG! You broke Discord!');
267
+ }
268
+ // Gateway connection closes on which a resume is allowed.
269
+ case _types.GatewayCloseEventCodes.UnknownError:
270
+ case _types.GatewayCloseEventCodes.DecodeError:
271
+ case _types.GatewayCloseEventCodes.AlreadyAuthenticated:
272
+ default:
273
+ {
274
+ this.logger.info(`[Shard] closed shard #${this.id}. Resuming...`);
275
+ this.state = _types1.ShardState.Resuming;
276
+ this.events.disconnected?.(this);
277
+ return await this.resume();
278
+ }
279
+ }
280
+ }
281
+ /** Handles a incoming gateway packet. */ async handleDiscordPacket(packet) {
282
+ // Edge case start: https://github.com/discordeno/discordeno/issues/2311
283
+ this.heart.lastAck = Date.now();
284
+ this.heart.acknowledged = true;
285
+ // Edge case end!
286
+ switch(packet.op){
287
+ case _types.GatewayOpcodes.Heartbeat:
288
+ {
289
+ // TODO: can this actually happen
290
+ if (!this.isOpen()) return;
291
+ this.heart.lastBeat = Date.now();
292
+ // Discord randomly sends this requiring an immediate heartbeat back.
293
+ // Using a direct socket.send call here because heartbeat requests are reserved by us.
294
+ this.socket?.send(JSON.stringify({
295
+ op: _types.GatewayOpcodes.Heartbeat,
296
+ d: this.previousSequenceNumber
297
+ }));
298
+ this.events.heartbeat?.(this);
299
+ break;
300
+ }
301
+ case _types.GatewayOpcodes.Hello:
302
+ {
303
+ const interval = packet.d.heartbeat_interval;
304
+ this.logger.debug(`[Gateway] Hello on Shard #${this.id}`);
305
+ this.startHeartbeating(interval);
306
+ if (this.state !== _types1.ShardState.Resuming) {
307
+ const currentQueue = [
308
+ ...this.bucket.queue
309
+ ];
310
+ // HELLO has been send on a non resume action.
311
+ // This means that the shard starts a new session,
312
+ // therefore the rate limit interval has been reset too.
313
+ this.bucket = new _utils.LeakyBucket({
314
+ max: this.calculateSafeRequests(),
315
+ refillInterval: 60000,
316
+ refillAmount: this.calculateSafeRequests()
317
+ });
318
+ // Queue should not be lost on a re-identify.
319
+ this.bucket.queue.unshift(...currentQueue);
320
+ }
321
+ this.events.hello?.(this);
322
+ break;
323
+ }
324
+ case _types.GatewayOpcodes.HeartbeatACK:
325
+ {
326
+ // Manually calculating the round trip time for users who need it.
327
+ if (this.heart.lastBeat) {
328
+ this.heart.rtt = this.heart.lastAck - this.heart.lastBeat;
329
+ }
330
+ this.events.heartbeatAck?.(this);
331
+ break;
332
+ }
333
+ case _types.GatewayOpcodes.Reconnect:
334
+ {
335
+ // gateway.debug("GW RECONNECT", { shardId });
336
+ this.events.requestedReconnect?.(this);
337
+ await this.resume();
338
+ break;
339
+ }
340
+ case _types.GatewayOpcodes.InvalidSession:
341
+ {
342
+ const resumable = packet.d;
343
+ this.logger.debug(`[Shard] Received Invalid Session for Shard #${this.id} with resumeable as ${resumable.toString()}`);
344
+ this.events.invalidSession?.(this, resumable);
345
+ // We need to wait for a random amount of time between 1 and 5
346
+ // Reference: https://discord.com/developers/docs/topics/gateway#resuming
347
+ await (0, _utils.delay)(Math.floor((Math.random() * 4 + 1) * 1000));
348
+ this.resolves.get('INVALID_SESSION')?.(packet);
349
+ this.resolves.delete('INVALID_SESSION');
350
+ // When resumable is false we need to re-identify
351
+ if (!resumable) {
352
+ await this.requestIdentify();
353
+ break;
354
+ }
355
+ // The session is invalid but apparently it is resumable
356
+ await this.resume();
357
+ break;
358
+ }
359
+ }
360
+ switch(packet.t){
361
+ case 'RESUMED':
362
+ this.state = _types1.ShardState.Connected;
363
+ this.events.resumed?.(this);
364
+ // Continue the requests which have been queued since the shard went offline.
365
+ this.offlineSendQueue.map((resolve)=>resolve());
366
+ this.resolves.get('RESUMED')?.(packet);
367
+ this.resolves.delete('RESUMED');
368
+ break;
369
+ case 'READY':
370
+ {
371
+ // Important for future resumes.
372
+ const payload = packet.d;
373
+ this.resumeGatewayUrl = payload.resume_gateway_url;
374
+ this.sessionId = payload.session_id;
375
+ this.state = _types1.ShardState.Connected;
376
+ // Continue the requests which have been queued since the shard went offline.
377
+ // Important when this is a re-identify
378
+ this.offlineSendQueue.map((resolve)=>resolve());
379
+ this.resolves.get('READY')?.(packet);
380
+ this.resolves.delete('READY');
381
+ break;
382
+ }
383
+ }
384
+ // Update the sequence number if it is present
385
+ // `s` can be either `null` or a `number`.
386
+ // In order to prevent update misses when `s` is `0` we check against null.
387
+ if (packet.s !== null) {
388
+ this.previousSequenceNumber = packet.s;
389
+ }
390
+ this.forwardToBot(packet);
391
+ }
392
+ forwardToBot(packet) {
393
+ // The necessary handling required for the Shards connection has been finished.
394
+ // Now the event can be safely forwarded.
395
+ this.events.message?.(this, (0, _utils.camelize)(packet));
396
+ }
397
+ /** Handle an incoming gateway message. */ async handleMessage(message) {
398
+ let preProcessMessage = message.data;
399
+ // If message compression is enabled,
400
+ // Discord might send zlib compressed payloads.
401
+ if (this.gatewayConfig.compress && preProcessMessage instanceof Blob) {
402
+ preProcessMessage = (0, _nodezlib.inflateSync)(await preProcessMessage.arrayBuffer()).toString();
403
+ }
404
+ // Safeguard incase decompression failed to make a string.
405
+ if (typeof preProcessMessage !== 'string') return;
406
+ return await this.handleDiscordPacket(JSON.parse(preProcessMessage));
407
+ }
408
+ /**
409
+ * Override in order to make the shards presence.
410
+ * async in case devs create the presence based on eg. database values.
411
+ * Passing the shard's id there to make it easier for the dev to use this function.
412
+ */ async makePresence() {
413
+ // eslint-disable-next-line no-useless-return
414
+ return;
415
+ }
416
+ /** This function communicates with the management process, in order to know whether its free to identify. When this function resolves, this means that the shard is allowed to send an identify payload to discord. */ async requestIdentify() {}
417
+ /** This function communicates with the management process, in order to tell it can identify the next shard. */ async shardIsReady() {}
418
+ /** Start sending heartbeat payloads to Discord in the provided interval. */ startHeartbeating(interval) {
419
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id}`);
420
+ // If old heartbeast exist like after resume, clear the old ones.
421
+ if (this.heart.intervalId) clearInterval(this.heart.intervalId);
422
+ if (this.heart.timeoutId) clearTimeout(this.heart.timeoutId);
423
+ this.heart.interval = interval;
424
+ // Only set the shard's state to `Unidentified`
425
+ // if heartbeating has not been started due to an identify or resume action.
426
+ if ([
427
+ _types1.ShardState.Disconnected,
428
+ _types1.ShardState.Offline
429
+ ].includes(this.state)) {
430
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} a`);
431
+ this.state = _types1.ShardState.Unidentified;
432
+ }
433
+ // The first heartbeat needs to be send with a random delay between `0` and `interval`
434
+ // Using a `setTimeout(_, jitter)` here to accomplish that.
435
+ // `Math.random()` can be `0` so we use `0.5` if this happens
436
+ // Reference: https://discord.com/developers/docs/topics/gateway#heartbeating
437
+ const jitter = Math.ceil(this.heart.interval * (Math.random() || 0.5));
438
+ this.heart.timeoutId = setTimeout(()=>{
439
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} b`);
440
+ if (!this.isOpen()) return;
441
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} c ${this.previousSequenceNumber}`);
442
+ // Using a direct socket.send call here because heartbeat requests are reserved by us.
443
+ this.socket?.send(JSON.stringify({
444
+ op: _types.GatewayOpcodes.Heartbeat,
445
+ d: this.previousSequenceNumber
446
+ }));
447
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} d`);
448
+ this.heart.lastBeat = Date.now();
449
+ this.heart.acknowledged = false;
450
+ // After the random heartbeat jitter we can start a normal interval.
451
+ this.heart.intervalId = setInterval(async ()=>{
452
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} e`);
453
+ if (!this.isOpen()) return;
454
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} f`);
455
+ // gateway.debug("GW DEBUG", `Running setInterval in heartbeat file. Shard: ${shardId}`);
456
+ // gateway.debug("GW HEARTBEATING", { shardId, shard: currentShard });
457
+ // The Shard did not receive a heartbeat ACK from Discord in time,
458
+ // therefore we have to assume that the connection has failed or got "zombied".
459
+ // The Shard needs to start a re-identify action accordingly.
460
+ // Reference: https://discord.com/developers/docs/topics/gateway#heartbeating-example-gateway-heartbeat-ack
461
+ if (!this.heart.acknowledged) {
462
+ this.logger.debug(`[Shard] Heartbeat not acknowledged for shard #${this.id}.`);
463
+ this.close(_types1.ShardSocketCloseCodes.ZombiedConnection, 'Zombied connection, did not receive an heartbeat ACK in time.');
464
+ return await this.identify();
465
+ }
466
+ this.heart.acknowledged = false;
467
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} g`);
468
+ // Using a direct socket.send call here because heartbeat requests are reserved by us.
469
+ this.socket?.send(JSON.stringify({
470
+ op: _types.GatewayOpcodes.Heartbeat,
471
+ d: this.previousSequenceNumber
472
+ }));
473
+ this.logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} h`);
474
+ this.heart.lastBeat = Date.now();
475
+ this.events.heartbeat?.(this);
476
+ }, this.heart.interval);
477
+ }, jitter);
478
+ }
479
+ /** Stop the heartbeating process with discord. */ stopHeartbeating() {
480
+ // Clear the regular heartbeat interval.
481
+ clearInterval(this.heart.intervalId);
482
+ // It's possible that the Shard got closed before the first jittered heartbeat.
483
+ // To go safe we should clear the related timeout too.
484
+ clearTimeout(this.heart.timeoutId);
485
+ }
486
+ };
487
+ const _default = DiscordenoShard;
488
+
489
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TaGFyZC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tY29uZnVzaW5nLXZvaWQtZXhwcmVzc2lvbiAqL1xuaW1wb3J0IHR5cGUgeyBEaXNjb3JkR2F0ZXdheVBheWxvYWQsIERpc2NvcmRIZWxsbywgRGlzY29yZFJlYWR5IH0gZnJvbSAnQGRpc2NvcmRlbm8vdHlwZXMnXG5pbXBvcnQgeyBHYXRld2F5Q2xvc2VFdmVudENvZGVzLCBHYXRld2F5T3Bjb2RlcyB9IGZyb20gJ0BkaXNjb3JkZW5vL3R5cGVzJ1xuaW1wb3J0IHsgTGVha3lCdWNrZXQsIGNhbWVsaXplLCBkZWxheSwgbG9nZ2VyIH0gZnJvbSAnQGRpc2NvcmRlbm8vdXRpbHMnXG5pbXBvcnQgeyBpbmZsYXRlU3luYyB9IGZyb20gJ25vZGU6emxpYidcbmltcG9ydCBOb2RlV2ViU29ja2V0IGZyb20gJ3dzJ1xuaW1wb3J0IHR5cGUgeyBCb3RTdGF0dXNVcGRhdGUsIFNoYXJkRXZlbnRzLCBTaGFyZEdhdGV3YXlDb25maWcsIFNoYXJkSGVhcnQsIFNoYXJkU29ja2V0UmVxdWVzdCB9IGZyb20gJy4vdHlwZXMuanMnXG5pbXBvcnQgeyBTaGFyZFNvY2tldENsb3NlQ29kZXMsIFNoYXJkU3RhdGUgfSBmcm9tICcuL3R5cGVzLmpzJ1xuXG5kZWNsYXJlIGxldCBXZWJTb2NrZXQ6IGFueVxuXG5leHBvcnQgY2xhc3MgRGlzY29yZGVub1NoYXJkIHtcbiAgLyoqIFRoZSBpZCBvZiB0aGUgc2hhcmQgKi9cbiAgaWQ6IG51bWJlclxuICAvKiogVGhlIGNvbm5lY3Rpb24gY29uZmlnIGRldGFpbHMgdGhhdCB0aGlzIHNoYXJkIHdpbGwgdXNlZCB0byBjb25uZWN0IHRvIGRpc2NvcmQuICovXG4gIGNvbm5lY3Rpb246IFNoYXJkR2F0ZXdheUNvbmZpZ1xuICAvKiogVGhpcyBjb250YWlucyBhbGwgdGhlIGhlYXJ0YmVhdCBpbmZvcm1hdGlvbiAqL1xuICBoZWFydDogU2hhcmRIZWFydFxuICAvKiogVGhlIG1heGltdW0gb2YgcmVxdWVzdHMgd2hpY2ggY2FuIGJlIHNlbmQgdG8gZGlzY29yZCBwZXIgcmF0ZSBsaW1pdCB0aWNrLiBUeXBpY2FsbHkgdGhpcyB2YWx1ZSBzaG91bGQgbm90IGJlIGNoYW5nZWQuICovXG4gIG1heFJlcXVlc3RzUGVyUmF0ZUxpbWl0VGljazogbnVtYmVyID0gMTIwXG4gIC8qKiBUaGUgcHJldmlvdXMgcGF5bG9hZCBzZXF1ZW5jZSBudW1iZXIuICovXG4gIHByZXZpb3VzU2VxdWVuY2VOdW1iZXI6IG51bWJlciB8IG51bGwgPSBudWxsXG4gIC8qKiBJbiB3aGljaCBpbnRlcnZhbCAoaW4gbWlsbGlzZWNvbmRzKSB0aGUgZ2F0ZXdheSByZXNldHMgaXQncyByYXRlIGxpbWl0LiAqL1xuICByYXRlTGltaXRSZXNldEludGVydmFsOiBudW1iZXIgPSA2MDAwMFxuICAvKiogQ3VycmVudCBzZXNzaW9uIGlkIG9mIHRoZSBzaGFyZCBpZiBwcmVzZW50LiAqL1xuICBzZXNzaW9uSWQ/OiBzdHJpbmdcbiAgLyoqIFRoaXMgY29udGFpbnMgdGhlIFdlYlNvY2tldCBjb25uZWN0aW9uIHRvIERpc2NvcmQsIGlmIGN1cnJlbnRseSBjb25uZWN0ZWQuICovXG4gIHNvY2tldD86IE5vZGVXZWJTb2NrZXRcbiAgLyoqIEN1cnJlbnQgaW50ZXJuYWwgc3RhdGUgb2YgdGhlIHRoaXMuICovXG4gIHN0YXRlID0gU2hhcmRTdGF0ZS5PZmZsaW5lXG4gIC8qKiBUaGUgdXJsIHByb3ZpZGVkIGJ5IGRpc2NvcmQgdG8gdXNlIHdoZW4gcmVzdW1pbmcgYSBjb25uZWN0aW9uIGZvciB0aGlzIHRoaXMuICovXG4gIHJlc3VtZUdhdGV3YXlVcmw6IHN0cmluZyA9ICcnXG4gIC8qKiBUaGUgc2hhcmQgcmVsYXRlZCBldmVudCBoYW5kbGVycy4gKi9cbiAgZXZlbnRzOiBTaGFyZEV2ZW50cyA9IHt9XG4gIC8qKiBDYWNoZSBmb3IgcGVuZGluZyBnYXRld2F5IHJlcXVlc3RzIHdoaWNoIHNob3VsZCBoYXZlIGJlZW4gc2VuZCB3aGlsZSB0aGUgZ2F0ZXdheSB3ZW50IG9mZmxpbmUuICovXG4gIG9mZmxpbmVTZW5kUXVldWU6IEFycmF5PChfPzogdW5rbm93bikgPT4gdm9pZD4gPSBbXVxuICAvKiogUmVzb2x2ZSBpbnRlcm5hbCB3YWl0aW5nIHN0YXRlcy4gTWFwcGVkIGJ5IFNlbGVjdGVkRXZlbnRzID0+IFJlc29sdmVGdW5jdGlvbiAqL1xuICByZXNvbHZlcyA9IG5ldyBNYXA8J1JFQURZJyB8ICdSRVNVTUVEJyB8ICdJTlZBTElEX1NFU1NJT04nLCAocGF5bG9hZDogRGlzY29yZEdhdGV3YXlQYXlsb2FkKSA9PiB2b2lkPigpXG4gIC8qKiBTaGFyZCBidWNrZXQuIE9ubHkgYWNjZXNzIHRoaXMgaWYgeW91IGtub3cgd2hhdCB5b3UgYXJlIGRvaW5nLiBCdWNrZXQgZm9yIGhhbmRsaW5nIHNoYXJkIHJlcXVlc3QgcmF0ZSBsaW1pdHMuICovXG4gIGJ1Y2tldDogTGVha3lCdWNrZXRcbiAgLyoqIExvZ2dlciBmb3IgdGhlIGJ1Y2tldCAqL1xuICBsb2dnZXI6IFBpY2s8dHlwZW9mIGxvZ2dlciwgJ2RlYnVnJyB8ICdpbmZvJyB8ICd3YXJuJyB8ICdlcnJvcicgfCAnZmF0YWwnPlxuXG4gIGNvbnN0cnVjdG9yKG9wdGlvbnM6IFNoYXJkQ3JlYXRlT3B0aW9ucykge1xuICAgIHRoaXMuaWQgPSBvcHRpb25zLmlkXG4gICAgdGhpcy5jb25uZWN0aW9uID0gb3B0aW9ucy5jb25uZWN0aW9uXG4gICAgdGhpcy5ldmVudHMgPSBvcHRpb25zLmV2ZW50c1xuICAgIHRoaXMubG9nZ2VyID0gb3B0aW9ucy5sb2dnZXIgPz8gbG9nZ2VyXG5cbiAgICB0aGlzLmhlYXJ0ID0ge1xuICAgICAgYWNrbm93bGVkZ2VkOiBmYWxzZSxcbiAgICAgIGludGVydmFsOiA0NTAwMCxcbiAgICB9XG5cbiAgICBpZiAob3B0aW9ucy5yZXF1ZXN0SWRlbnRpZnkpIHRoaXMucmVxdWVzdElkZW50aWZ5ID0gb3B0aW9ucy5yZXF1ZXN0SWRlbnRpZnlcbiAgICBpZiAob3B0aW9ucy5zaGFyZElzUmVhZHkpIHRoaXMuc2hhcmRJc1JlYWR5ID0gb3B0aW9ucy5zaGFyZElzUmVhZHlcblxuICAgIHRoaXMuYnVja2V0ID0gbmV3IExlYWt5QnVja2V0KHtcbiAgICAgIG1heDogdGhpcy5jYWxjdWxhdGVTYWZlUmVxdWVzdHMoKSxcbiAgICAgIHJlZmlsbEFtb3VudDogdGhpcy5jYWxjdWxhdGVTYWZlUmVxdWVzdHMoKSxcbiAgICAgIHJlZmlsbEludGVydmFsOiA2MDAwMCxcbiAgICAgIGxvZ2dlcjogdGhpcy5sb2dnZXIsXG4gICAgfSlcbiAgfVxuXG4gIC8qKiBUaGUgZ2F0ZXdheSBjb25maWd1cmF0aW9uIHdoaWNoIGlzIHVzZWQgdG8gY29ubmVjdCB0byBEaXNjb3JkLiAqL1xuICBnZXQgZ2F0ZXdheUNvbmZpZygpOiBTaGFyZEdhdGV3YXlDb25maWcge1xuICAgIHJldHVybiB0aGlzLmNvbm5lY3Rpb25cbiAgfVxuXG4gIC8qKiBUaGUgdXJsIHRvIGNvbm5lY3QgdG8uIEluaXRpYWxseSB0aGlzIGlzIHRoZSBkaXNjb3JkIGdhdGV3YXkgdXJsLCBhbmQgdGhlbiBpcyBzd2l0Y2hlZCB0byByZXN1bWUgZ2F0ZXdheSB1cmwgb25jZSBhIFJFQURZIGlzIHJlY2VpdmVkLiAqL1xuICBnZXQgY29ubmVjdGlvblVybCgpOiBzdHJpbmcge1xuICAgIC8vIFVzZSB8fCBhbmQgbm90ID8/IGhlcmUuID8/IHdpbGwgY2F1c2UgYSBidWcuXG4gICAgcmV0dXJuIHRoaXMucmVzdW1lR2F0ZXdheVVybCB8fCB0aGlzLmdhdGV3YXlDb25maWcudXJsXG4gIH1cblxuICAvKiogQ2FsY3VsYXRlIHRoZSBhbW91bnQgb2YgcmVxdWVzdHMgd2hpY2ggY2FuIHNhZmVseSBiZSBtYWRlIHBlciByYXRlIGxpbWl0IGludGVydmFsLCBiZWZvcmUgdGhlIGdhdGV3YXkgZ2V0cyBkaXNjb25uZWN0ZWQgZHVlIHRvIGFuIGV4Y2VlZGVkIHJhdGUgbGltaXQuICovXG4gIGNhbGN1bGF0ZVNhZmVSZXF1ZXN0cygpOiBudW1iZXIge1xuICAgIC8vICogMiBhZGRzIGV4dHJhIHNhZmV0eSBsYXllciBmb3IgZGlzY29yZHMgT1AgMSByZXF1ZXN0cyB0aGF0IHdlIG5lZWQgdG8gcmVzcG9uZCB0b1xuICAgIGNvbnN0IHNhZmVSZXF1ZXN0cyA9IHRoaXMubWF4UmVxdWVzdHNQZXJSYXRlTGltaXRUaWNrIC0gTWF0aC5jZWlsKHRoaXMucmF0ZUxpbWl0UmVzZXRJbnRlcnZhbCAvIHRoaXMuaGVhcnQuaW50ZXJ2YWwpICogMlxuXG4gICAgcmV0dXJuIHNhZmVSZXF1ZXN0cyA8IDAgPyAwIDogc2FmZVJlcXVlc3RzXG4gIH1cblxuICBhc3luYyBjaGVja09mZmxpbmUoaGlnaFByaW9yaXR5OiBib29sZWFuKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgaWYgKCF0aGlzLmlzT3BlbigpKSB7XG4gICAgICBhd2FpdCBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4ge1xuICAgICAgICAvLyBIaWdoZXIgcHJpb3JpdHkgcmVxdWVzdHMgZ2V0IGFkZGVkIGF0IHRoZSBiZWdpbm5pbmcgb2YgdGhlIGFycmF5LlxuICAgICAgICBpZiAoaGlnaFByaW9yaXR5KSB0aGlzLm9mZmxpbmVTZW5kUXVldWUudW5zaGlmdChyZXNvbHZlKVxuICAgICAgICBlbHNlIHRoaXMub2ZmbGluZVNlbmRRdWV1ZS5wdXNoKHJlc29sdmUpXG4gICAgICB9KVxuICAgIH1cbiAgfVxuXG4gIC8qKiBDbG9zZSB0aGUgc29ja2V0IGNvbm5lY3Rpb24gdG8gZGlzY29yZCBpZiBwcmVzZW50LiAqL1xuICBjbG9zZShjb2RlOiBudW1iZXIsIHJlYXNvbjogc3RyaW5nKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuc29ja2V0Py5yZWFkeVN0YXRlICE9PSBOb2RlV2ViU29ja2V0Lk9QRU4pIHJldHVyblxuXG4gICAgdGhpcy5zb2NrZXQ/LmNsb3NlKGNvZGUsIHJlYXNvbilcbiAgfVxuXG4gIC8qKiBDb25uZWN0IHRoZSBzaGFyZCB3aXRoIHRoZSBnYXRld2F5IGFuZCBzdGFydCBoZWFydGJlYXRpbmcuIFRoaXMgd2lsbCBub3QgaWRlbnRpZnkgdGhlIHNoYXJkIHRvIHRoZSBnYXRld2F5LiAqL1xuICBhc3luYyBjb25uZWN0KCk6IFByb21pc2U8RGlzY29yZGVub1NoYXJkPiB7XG4gICAgLy8gT25seSBzZXQgdGhlIHNoYXJkIHRvIGBDb25uZWN0aW5nYCBzdGF0ZSxcbiAgICAvLyBpZiB0aGUgY29ubmVjdGlvbiByZXF1ZXN0IGRvZXMgbm90IGNvbWUgZnJvbSBhbiBpZGVudGlmeSBvciByZXN1bWUgYWN0aW9uLlxuICAgIGlmICghW1NoYXJkU3RhdGUuSWRlbnRpZnlpbmcsIFNoYXJkU3RhdGUuUmVzdW1pbmddLmluY2x1ZGVzKHRoaXMuc3RhdGUpKSB7XG4gICAgICB0aGlzLnN0YXRlID0gU2hhcmRTdGF0ZS5Db25uZWN0aW5nXG4gICAgfVxuICAgIHRoaXMuZXZlbnRzLmNvbm5lY3Rpbmc/Lih0aGlzKVxuXG4gICAgY29uc3QgdXJsID0gbmV3IFVSTCh0aGlzLmNvbm5lY3Rpb25VcmwpXG4gICAgdXJsLnNlYXJjaFBhcmFtcy5zZXQoJ3YnLCB0aGlzLmdhdGV3YXlDb25maWcudmVyc2lvbi50b1N0cmluZygpKVxuICAgIHVybC5zZWFyY2hQYXJhbXMuc2V0KCdlbmNvZGluZycsICdqc29uJylcblxuICAgIGNvbnN0IHNvY2tldDogTm9kZVdlYlNvY2tldCA9XG4gICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIERlbm9cbiAgICAgIGdsb2JhbFRoaXMuRGVubyAhPT0gdW5kZWZpbmVkICYmIFJlZmxlY3QuaGFzKGdsb2JhbFRoaXMsICdEZW5vJykgPyBuZXcgV2ViU29ja2V0KHVybC50b1N0cmluZygpKSA6IG5ldyBOb2RlV2ViU29ja2V0KHVybC50b1N0cmluZygpKVxuICAgIHRoaXMuc29ja2V0ID0gc29ja2V0XG5cbiAgICAvLyBUT0RPOiBwcm9wZXIgZXZlbnQgaGFuZGxpbmdcbiAgICBzb2NrZXQub25lcnJvciA9IChldmVudDogTm9kZVdlYlNvY2tldC5FcnJvckV2ZW50KSA9PiBjb25zb2xlLmxvZyh7IGVycm9yOiBldmVudCwgc2hhcmRJZDogdGhpcy5pZCB9KVxuICAgIHNvY2tldC5vbmNsb3NlID0gYXN5bmMgKGV2ZW50OiBOb2RlV2ViU29ja2V0LkNsb3NlRXZlbnQpID0+IGF3YWl0IHRoaXMuaGFuZGxlQ2xvc2UoZXZlbnQpXG4gICAgc29ja2V0Lm9ubWVzc2FnZSA9IGFzeW5jIChtZXNzYWdlOiBOb2RlV2ViU29ja2V0Lk1lc3NhZ2VFdmVudCkgPT4gYXdhaXQgdGhpcy5oYW5kbGVNZXNzYWdlKG1lc3NhZ2UpXG5cbiAgICByZXR1cm4gYXdhaXQgbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgICAgIHNvY2tldC5vbm9wZW4gPSAoKSA9PiB7XG4gICAgICAgIC8vIE9ubHkgc2V0IHRoZSBzaGFyZCB0byBgVW5pZGVudGlmaWVkYCBzdGF0ZSxcbiAgICAgICAgLy8gaWYgdGhlIGNvbm5lY3Rpb24gcmVxdWVzdCBkb2VzIG5vdCBjb21lIGZyb20gYW4gaWRlbnRpZnkgb3IgcmVzdW1lIGFjdGlvbi5cbiAgICAgICAgaWYgKCFbU2hhcmRTdGF0ZS5JZGVudGlmeWluZywgU2hhcmRTdGF0ZS5SZXN1bWluZ10uaW5jbHVkZXModGhpcy5zdGF0ZSkpIHtcbiAgICAgICAgICB0aGlzLnN0YXRlID0gU2hhcmRTdGF0ZS5VbmlkZW50aWZpZWRcbiAgICAgICAgfVxuICAgICAgICB0aGlzLmV2ZW50cy5jb25uZWN0ZWQ/Lih0aGlzKVxuXG4gICAgICAgIHJlc29sdmUodGhpcylcbiAgICAgIH1cbiAgICB9KVxuICB9XG5cbiAgLyoqIElkZW50aWZ5IHRoZSBzaGFyZCB0byB0aGUgZ2F0ZXdheS4gSWYgbm90IGNvbm5lY3RlZCwgdGhpcyB3aWxsIGFsc28gY29ubmVjdCB0aGUgc2hhcmQgdG8gdGhlIGdhdGV3YXkuICovXG4gIGFzeW5jIGlkZW50aWZ5KCk6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vIEEgbmV3IGlkZW50aWZ5IGhhcyBiZWVuIHJlcXVlc3RlZCBldmVuIHRob3VnaCB0aGVyZSBpcyBhbHJlYWR5IGEgY29ubmVjdGlvbiBvcGVuLlxuICAgIC8vIFRoZXJlZm9yZSB3ZSBuZWVkIHRvIGNsb3NlIHRoZSBvbGQgY29ubmVjdGlvbiBhbmQgaGVhcnRiZWF0aW5nIGJlZm9yZSBjcmVhdGluZyBhIG5ldyBvbmUuXG4gICAgaWYgKHRoaXMuaXNPcGVuKCkpIHtcbiAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBDTE9TSU5HIEVYSVNUSU5HIFNIQVJEOiAjJHt0aGlzLmlkfWApXG4gICAgICB0aGlzLmNsb3NlKFNoYXJkU29ja2V0Q2xvc2VDb2Rlcy5SZUlkZW50aWZ5aW5nLCAnUmUtaWRlbnRpZnlpbmcgY2xvc3VyZSBvZiBvbGQgY29ubmVjdGlvbi4nKVxuICAgIH1cblxuICAgIHRoaXMuc3RhdGUgPSBTaGFyZFN0YXRlLklkZW50aWZ5aW5nXG4gICAgdGhpcy5ldmVudHMuaWRlbnRpZnlpbmc/Lih0aGlzKVxuXG4gICAgLy8gSXQgaXMgcG9zc2libGUgdGhhdCB0aGUgc2hhcmQgaXMgaW4gSGVhcnRiZWF0aW5nIHN0YXRlIGJ1dCBub3QgaWRlbnRpZmllZCxcbiAgICAvLyBzbyBjaGVjayB3aGV0aGVyIHRoZXJlIGlzIGFscmVhZHkgYSBnYXRld2F5IGNvbm5lY3Rpb24gZXhpc3RpbmcuXG4gICAgLy8gSWYgbm90IHdlIG5lZWQgdG8gY3JlYXRlIG9uZSBiZWZvcmUgd2UgaWRlbnRpZnkuXG4gICAgaWYgKCF0aGlzLmlzT3BlbigpKSB7XG4gICAgICBhd2FpdCB0aGlzLmNvbm5lY3QoKVxuICAgIH1cblxuICAgIHRoaXMuc2VuZChcbiAgICAgIHtcbiAgICAgICAgb3A6IEdhdGV3YXlPcGNvZGVzLklkZW50aWZ5LFxuICAgICAgICBkOiB7XG4gICAgICAgICAgdG9rZW46IGBCb3QgJHt0aGlzLmdhdGV3YXlDb25maWcudG9rZW59YCxcbiAgICAgICAgICBjb21wcmVzczogdGhpcy5nYXRld2F5Q29uZmlnLmNvbXByZXNzLFxuICAgICAgICAgIHByb3BlcnRpZXM6IHRoaXMuZ2F0ZXdheUNvbmZpZy5wcm9wZXJ0aWVzLFxuICAgICAgICAgIGludGVudHM6IHRoaXMuZ2F0ZXdheUNvbmZpZy5pbnRlbnRzLFxuICAgICAgICAgIHNoYXJkOiBbdGhpcy5pZCwgdGhpcy5nYXRld2F5Q29uZmlnLnRvdGFsU2hhcmRzXSxcbiAgICAgICAgICBwcmVzZW5jZTogYXdhaXQgdGhpcy5tYWtlUHJlc2VuY2U/LigpLFxuICAgICAgICB9LFxuICAgICAgfSxcbiAgICAgIHRydWUsXG4gICAgKVxuXG4gICAgcmV0dXJuIGF3YWl0IG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiB7XG4gICAgICB0aGlzLnJlc29sdmVzLnNldCgnUkVBRFknLCAoKSA9PiB7XG4gICAgICAgIHRoaXMuZXZlbnRzLmlkZW50aWZpZWQ/Lih0aGlzKVxuICAgICAgICAvLyBUZWxscyB0aGUgbWFuYWdlciB0aGF0IHRoaXMgc2hhcmQgaXMgcmVhZHlcbiAgICAgICAgdGhpcy5zaGFyZElzUmVhZHkoKVxuICAgICAgICByZXNvbHZlKClcbiAgICAgIH0pXG4gICAgICAvLyBXaGVuIGlkZW50aWZ5aW5nIHRvbyBmYXN0LFxuICAgICAgLy8gRGlzY29yZCBzZW5kcyBhbiBpbnZhbGlkIHNlc3Npb24gcGF5bG9hZC5cbiAgICAgIC8vIFRoaXMgY2FuIHNhZmVseSBiZSBpZ25vcmVkIHRob3VnaCBhbmQgdGhlIHNoYXJkIHN0YXJ0cyBhIG5ldyBpZGVudGlmeSBhY3Rpb24uXG4gICAgICB0aGlzLnJlc29sdmVzLnNldCgnSU5WQUxJRF9TRVNTSU9OJywgKCkgPT4ge1xuICAgICAgICB0aGlzLnJlc29sdmVzLmRlbGV0ZSgnUkVBRFknKVxuICAgICAgICByZXNvbHZlKClcbiAgICAgIH0pXG4gICAgfSlcbiAgfVxuXG4gIC8qKiBDaGVjayB3aGV0aGVyIHRoZSBjb25uZWN0aW9uIHRvIERpc2NvcmQgaXMgY3VycmVudGx5IG9wZW4uICovXG4gIGlzT3BlbigpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5zb2NrZXQ/LnJlYWR5U3RhdGUgPT09IE5vZGVXZWJTb2NrZXQuT1BFTlxuICB9XG5cbiAgLyoqIEF0dGVtcHQgdG8gcmVzdW1lIHRoZSBwcmV2aW91cyBzaGFyZHMgc2Vzc2lvbiB3aXRoIHRoZSBnYXRld2F5LiAqL1xuICBhc3luYyByZXN1bWUoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgdGhpcy5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBSZXN1bWluZyBTaGFyZCAjJHt0aGlzLmlkfWApXG4gICAgLy8gSXQgaGFzIGJlZW4gcmVxdWVzdGVkIHRvIHJlc3VtZSB0aGUgU2hhcmRzIHNlc3Npb24uXG4gICAgLy8gSXQncyBwb3NzaWJsZSB0aGF0IHRoZSBzaGFyZCBpcyBzdGlsbCBjb25uZWN0ZWQgd2l0aCBEaXNjb3JkJ3MgZ2F0ZXdheSB0aGVyZWZvcmUgd2UgbmVlZCB0byBmb3JjZWZ1bGx5IGNsb3NlIGl0LlxuICAgIGlmICh0aGlzLmlzT3BlbigpKSB7XG4gICAgICB0aGlzLmxvZ2dlci5kZWJ1ZyhgW0dhdGV3YXldIFJlc3VtaW5nIFNoYXJkICMke3RoaXMuaWR9IGluIGlzT3BlbmApXG4gICAgICB0aGlzLmNsb3NlKFNoYXJkU29ja2V0Q2xvc2VDb2Rlcy5SZXN1bWVDbG9zaW5nT2xkQ29ubmVjdGlvbiwgJ1JlY29ubmVjdGluZyB0aGUgc2hhcmQsIGNsb3Npbmcgb2xkIGNvbm5lY3Rpb24uJylcbiAgICB9XG5cbiAgICAvLyBTaGFyZCBoYXMgbmV2ZXIgaWRlbnRpZmllZCwgc28gd2UgY2Fubm90IHJlc3VtZS5cbiAgICBpZiAoIXRoaXMuc2Vzc2lvbklkKSB7XG4gICAgICB0aGlzLmxvZ2dlci5kZWJ1ZyhgW1NoYXJkXSBUcnlpbmcgdG8gcmVzdW1lIGEgc2hhcmQgIyR7dGhpcy5pZH0gdGhhdCB3YXMgTk9UIGZpcnN0IGlkZW50aWZpZWQuIChObyBzZXNzaW9uIGlkIGZvdW5kKWApXG5cbiAgICAgIHJldHVybiBhd2FpdCB0aGlzLmlkZW50aWZ5KClcbiAgICB9XG5cbiAgICB0aGlzLnN0YXRlID0gU2hhcmRTdGF0ZS5SZXN1bWluZ1xuXG4gICAgdGhpcy5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBSZXN1bWluZyBTaGFyZCAjJHt0aGlzLmlkfSwgYmVmb3JlIGNvbm5lY3RpbmdgKVxuICAgIC8vIEJlZm9yZSB3ZSBjYW4gcmVzdW1lLCB3ZSBuZWVkIHRvIGNyZWF0ZSBhIG5ldyBjb25uZWN0aW9uIHdpdGggRGlzY29yZCdzIGdhdGV3YXkuXG4gICAgYXdhaXQgdGhpcy5jb25uZWN0KClcbiAgICB0aGlzLmxvZ2dlci5kZWJ1ZyhcbiAgICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvcmVzdHJpY3QtdGVtcGxhdGUtZXhwcmVzc2lvbnNcbiAgICAgIGBbR2F0ZXdheV0gUmVzdW1pbmcgU2hhcmQgIyR7dGhpcy5pZH0sIGFmdGVyIGNvbm5lY3RpbmcuICR7dGhpcy5zZXNzaW9uSWR9IHwgJHt0aGlzLnByZXZpb3VzU2VxdWVuY2VOdW1iZXJ9YCxcbiAgICApXG5cbiAgICB0aGlzLnNlbmQoXG4gICAgICB7XG4gICAgICAgIG9wOiBHYXRld2F5T3Bjb2Rlcy5SZXN1bWUsXG4gICAgICAgIGQ6IHtcbiAgICAgICAgICB0b2tlbjogYEJvdCAke3RoaXMuZ2F0ZXdheUNvbmZpZy50b2tlbn1gLFxuICAgICAgICAgIHNlc3Npb25faWQ6IHRoaXMuc2Vzc2lvbklkLFxuICAgICAgICAgIHNlcTogdGhpcy5wcmV2aW91c1NlcXVlbmNlTnVtYmVyID8/IDAsXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgICAgdHJ1ZSxcbiAgICApXG4gICAgdGhpcy5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBSZXN1bWluZyBTaGFyZCAjJHt0aGlzLmlkfSBhZnRlciBzZW5kIHJlc3VtZ2ApXG5cbiAgICByZXR1cm4gYXdhaXQgbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgICAgIHRoaXMucmVzb2x2ZXMuc2V0KCdSRVNVTUVEJywgKCkgPT4gcmVzb2x2ZSgpKVxuICAgICAgLy8gSWYgaXQgaXMgYXR0ZW1wdGVkIHRvIHJlc3VtZSB3aXRoIGFuIGludmFsaWQgc2Vzc2lvbiBpZCxcbiAgICAgIC8vIERpc2NvcmQgc2VuZHMgYW4gaW52YWxpZCBzZXNzaW9uIHBheWxvYWRcbiAgICAgIC8vIE5vdCBlcnJvcmluZyBoZXJlIHNpbmNlIGl0IGlzIGVhc3kgdGhhdCB0aGlzIGhhcHBlbnMsIGFsc28gaXQgd291bGQgYmUgbm90IGNhdGNoYWJsZVxuICAgICAgdGhpcy5yZXNvbHZlcy5zZXQoJ0lOVkFMSURfU0VTU0lPTicsICgpID0+IHtcbiAgICAgICAgdGhpcy5yZXNvbHZlcy5kZWxldGUoJ1JFU1VNRUQnKVxuICAgICAgICByZXNvbHZlKClcbiAgICAgIH0pXG4gICAgfSlcbiAgfVxuXG4gIC8qKiBTZW5kIGEgbWVzc2FnZSB0byBEaXNjb3JkLlxuICAgKiBAcGFyYW0ge2Jvb2xlYW59IFtoaWdoUHJpb3JpdHk9ZmFsc2VdIC0gV2hldGhlciB0aGlzIG1lc3NhZ2Ugc2hvdWxkIGJlIHNlbmQgYXNhcC5cbiAgICovXG4gIGFzeW5jIHNlbmQobWVzc2FnZTogU2hhcmRTb2NrZXRSZXF1ZXN0LCBoaWdoUHJpb3JpdHk6IGJvb2xlYW4gPSBmYWxzZSk6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vIEJlZm9yZSBhY3F1aXJpbmcgYSB0b2tlbiBmcm9tIHRoZSBidWNrZXQsIGNoZWNrIHdoZXRoZXIgdGhlIHNoYXJkIGlzIGN1cnJlbnRseSBvZmZsaW5lIG9yIG5vdC5cbiAgICAvLyBFbHNlIGJ1Y2tldCBhbmQgdG9rZW4gd2FpdCB0aW1lIGp1c3QgZ2V0IHdhc3RlZC5cbiAgICBhd2FpdCB0aGlzLmNoZWNrT2ZmbGluZShoaWdoUHJpb3JpdHkpXG5cbiAgICBhd2FpdCB0aGlzLmJ1Y2tldC5hY3F1aXJlKGhpZ2hQcmlvcml0eSlcblxuICAgIC8vIEl0J3MgcG9zc2libGUsIHRoYXQgdGhlIHNoYXJkIHdlbnQgb2ZmbGluZSBhZnRlciBhIHRva2VuIGhhcyBiZWVuIGFjcXVpcmVkIGZyb20gdGhlIGJ1Y2tldC5cbiAgICBhd2FpdCB0aGlzLmNoZWNrT2ZmbGluZShoaWdoUHJpb3JpdHkpXG5cbiAgICB0aGlzLnNvY2tldD8uc2VuZChKU09OLnN0cmluZ2lmeShtZXNzYWdlKSlcbiAgfVxuXG4gIC8qKiBTaHV0ZG93biB0aGUgdGhpcy4gRm9yY2VmdWxseSBkaXNjb25uZWN0IHRoZSBzaGFyZCBmcm9tIERpc2NvcmQuIFRoZSBzaGFyZCBtYXkgbm90IGF0dGVtcHQgdG8gcmVjb25uZWN0IHdpdGggRGlzY29yZC4gKi9cbiAgYXN5bmMgc2h1dGRvd24oKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgdGhpcy5jbG9zZShTaGFyZFNvY2tldENsb3NlQ29kZXMuU2h1dGRvd24sICdTaGFyZCBzaHV0dGluZyBkb3duLicpXG4gICAgdGhpcy5zdGF0ZSA9IFNoYXJkU3RhdGUuT2ZmbGluZVxuICB9XG5cbiAgLyoqIEhhbmRsZSBhIGdhdGV3YXkgY29ubmVjdGlvbiBjbG9zZS4gKi9cbiAgYXN5bmMgaGFuZGxlQ2xvc2UoY2xvc2U6IE5vZGVXZWJTb2NrZXQuQ2xvc2VFdmVudCk6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vICAgZ2F0ZXdheS5kZWJ1ZyhcIkdXIENMT1NFRFwiLCB7IHNoYXJkSWQsIHBheWxvYWQ6IGV2ZW50IH0pO1xuXG4gICAgdGhpcy5zdG9wSGVhcnRiZWF0aW5nKClcblxuICAgIHN3aXRjaCAoY2xvc2UuY29kZSkge1xuICAgICAgY2FzZSBTaGFyZFNvY2tldENsb3NlQ29kZXMuVGVzdGluZ0ZpbmlzaGVkOiB7XG4gICAgICAgIHRoaXMuc3RhdGUgPSBTaGFyZFN0YXRlLk9mZmxpbmVcbiAgICAgICAgdGhpcy5ldmVudHMuZGlzY29ubmVjdGVkPy4odGhpcylcblxuICAgICAgICByZXR1cm5cbiAgICAgIH1cbiAgICAgIC8vIE9uIHRoZXNlIGNvZGVzIGEgbWFudWFsIHN0YXJ0IHdpbGwgYmUgZG9uZS5cbiAgICAgIGNhc2UgU2hhcmRTb2NrZXRDbG9zZUNvZGVzLlNodXRkb3duOlxuICAgICAgY2FzZSBTaGFyZFNvY2tldENsb3NlQ29kZXMuUmVJZGVudGlmeWluZzpcbiAgICAgIGNhc2UgU2hhcmRTb2NrZXRDbG9zZUNvZGVzLlJlc2hhcmRlZDpcbiAgICAgIGNhc2UgU2hhcmRTb2NrZXRDbG9zZUNvZGVzLlJlc3VtZUNsb3NpbmdPbGRDb25uZWN0aW9uOlxuICAgICAgY2FzZSBTaGFyZFNvY2tldENsb3NlQ29kZXMuWm9tYmllZENvbm5lY3Rpb246IHtcbiAgICAgICAgdGhpcy5zdGF0ZSA9IFNoYXJkU3RhdGUuRGlzY29ubmVjdGVkXG4gICAgICAgIHRoaXMuZXZlbnRzLmRpc2Nvbm5lY3RlZD8uKHRoaXMpXG5cbiAgICAgICAgLy8gZ2F0ZXdheS5kZWJ1ZyhcIkdXIENMT1NFRF9SRUNPTk5FQ1RcIiwgeyBzaGFyZElkLCBwYXlsb2FkOiBldmVudCB9KTtcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG4gICAgICAvLyBHYXRld2F5IGNvbm5lY3Rpb24gY2xvc2VzIHdoaWNoIHJlcXVpcmUgYSBuZXcgaWRlbnRpZnkuXG4gICAgICBjYXNlIEdhdGV3YXlDbG9zZUV2ZW50Q29kZXMuVW5rbm93bk9wY29kZTpcbiAgICAgIGNhc2UgR2F0ZXdheUNsb3NlRXZlbnRDb2Rlcy5Ob3RBdXRoZW50aWNhdGVkOlxuICAgICAgY2FzZSBHYXRld2F5Q2xvc2VFdmVudENvZGVzLkludmFsaWRTZXE6XG4gICAgICBjYXNlIEdhdGV3YXlDbG9zZUV2ZW50Q29kZXMuUmF0ZUxpbWl0ZWQ6XG4gICAgICBjYXNlIEdhdGV3YXlDbG9zZUV2ZW50Q29kZXMuU2Vzc2lvblRpbWVkT3V0OiB7XG4gICAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBbU2hhcmRdIEdhdGV3YXkgY29ubmVjdGlvbiBjbG9zaW5nIHJlcXVpcmluZyByZS1pZGVudGlmeS4gQ29kZTogJHtjbG9zZS5jb2RlfWApXG4gICAgICAgIHRoaXMuc3RhdGUgPSBTaGFyZFN0YXRlLklkZW50aWZ5aW5nXG4gICAgICAgIHRoaXMuZXZlbnRzLmRpc2Nvbm5lY3RlZD8uKHRoaXMpXG5cbiAgICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuaWRlbnRpZnkoKVxuICAgICAgfVxuICAgICAgLy8gV2hlbiB0aGVzZSBjb2RlcyBhcmUgcmVjZWl2ZWQgc29tZXRoaW5nIHdlbnQgcmVhbGx5IHdyb25nLlxuICAgICAgLy8gT24gdGhvc2Ugd2UgY2Fubm90IHN0YXJ0IGEgcmVjb25uZWN0IGF0dGVtcHQuXG4gICAgICBjYXNlIEdhdGV3YXlDbG9zZUV2ZW50Q29kZXMuQXV0aGVudGljYXRpb25GYWlsZWQ6XG4gICAgICBjYXNlIEdhdGV3YXlDbG9zZUV2ZW50Q29kZXMuSW52YWxpZFNoYXJkOlxuICAgICAgY2FzZSBHYXRld2F5Q2xvc2VFdmVudENvZGVzLlNoYXJkaW5nUmVxdWlyZWQ6XG4gICAgICBjYXNlIEdhdGV3YXlDbG9zZUV2ZW50Q29kZXMuSW52YWxpZEFwaVZlcnNpb246XG4gICAgICBjYXNlIEdhdGV3YXlDbG9zZUV2ZW50Q29kZXMuSW52YWxpZEludGVudHM6XG4gICAgICBjYXNlIEdhdGV3YXlDbG9zZUV2ZW50Q29kZXMuRGlzYWxsb3dlZEludGVudHM6IHtcbiAgICAgICAgdGhpcy5zdGF0ZSA9IFNoYXJkU3RhdGUuT2ZmbGluZVxuICAgICAgICB0aGlzLmV2ZW50cy5kaXNjb25uZWN0ZWQ/Lih0aGlzKVxuXG4gICAgICAgIHRocm93IG5ldyBFcnJvcihjbG9zZS5yZWFzb24gfHwgJ0Rpc2NvcmQgZ2F2ZSBubyByZWFzb24hIEdHISBZb3UgYnJva2UgRGlzY29yZCEnKVxuICAgICAgfVxuICAgICAgLy8gR2F0ZXdheSBjb25uZWN0aW9uIGNsb3NlcyBvbiB3aGljaCBhIHJlc3VtZSBpcyBhbGxvd2VkLlxuICAgICAgY2FzZSBHYXRld2F5Q2xvc2VFdmVudENvZGVzLlVua25vd25FcnJvcjpcbiAgICAgIGNhc2UgR2F0ZXdheUNsb3NlRXZlbnRDb2Rlcy5EZWNvZGVFcnJvcjpcbiAgICAgIGNhc2UgR2F0ZXdheUNsb3NlRXZlbnRDb2Rlcy5BbHJlYWR5QXV0aGVudGljYXRlZDpcbiAgICAgIGRlZmF1bHQ6IHtcbiAgICAgICAgdGhpcy5sb2dnZXIuaW5mbyhgW1NoYXJkXSBjbG9zZWQgc2hhcmQgIyR7dGhpcy5pZH0uIFJlc3VtaW5nLi4uYClcbiAgICAgICAgdGhpcy5zdGF0ZSA9IFNoYXJkU3RhdGUuUmVzdW1pbmdcbiAgICAgICAgdGhpcy5ldmVudHMuZGlzY29ubmVjdGVkPy4odGhpcylcblxuICAgICAgICByZXR1cm4gYXdhaXQgdGhpcy5yZXN1bWUoKVxuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKiBIYW5kbGVzIGEgaW5jb21pbmcgZ2F0ZXdheSBwYWNrZXQuICovXG4gIGFzeW5jIGhhbmRsZURpc2NvcmRQYWNrZXQocGFja2V0OiBEaXNjb3JkR2F0ZXdheVBheWxvYWQpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBFZGdlIGNhc2Ugc3RhcnQ6IGh0dHBzOi8vZ2l0aHViLmNvbS9kaXNjb3JkZW5vL2Rpc2NvcmRlbm8vaXNzdWVzLzIzMTFcbiAgICB0aGlzLmhlYXJ0Lmxhc3RBY2sgPSBEYXRlLm5vdygpXG4gICAgdGhpcy5oZWFydC5hY2tub3dsZWRnZWQgPSB0cnVlXG4gICAgLy8gRWRnZSBjYXNlIGVuZCFcblxuICAgIHN3aXRjaCAocGFja2V0Lm9wKSB7XG4gICAgICBjYXNlIEdhdGV3YXlPcGNvZGVzLkhlYXJ0YmVhdDoge1xuICAgICAgICAvLyBUT0RPOiBjYW4gdGhpcyBhY3R1YWxseSBoYXBwZW5cbiAgICAgICAgaWYgKCF0aGlzLmlzT3BlbigpKSByZXR1cm5cblxuICAgICAgICB0aGlzLmhlYXJ0Lmxhc3RCZWF0ID0gRGF0ZS5ub3coKVxuICAgICAgICAvLyBEaXNjb3JkIHJhbmRvbWx5IHNlbmRzIHRoaXMgcmVxdWlyaW5nIGFuIGltbWVkaWF0ZSBoZWFydGJlYXQgYmFjay5cbiAgICAgICAgLy8gVXNpbmcgYSBkaXJlY3Qgc29ja2V0LnNlbmQgY2FsbCBoZXJlIGJlY2F1c2UgaGVhcnRiZWF0IHJlcXVlc3RzIGFyZSByZXNlcnZlZCBieSB1cy5cbiAgICAgICAgdGhpcy5zb2NrZXQ/LnNlbmQoXG4gICAgICAgICAgSlNPTi5zdHJpbmdpZnkoe1xuICAgICAgICAgICAgb3A6IEdhdGV3YXlPcGNvZGVzLkhlYXJ0YmVhdCxcbiAgICAgICAgICAgIGQ6IHRoaXMucHJldmlvdXNTZXF1ZW5jZU51bWJlcixcbiAgICAgICAgICB9KSxcbiAgICAgICAgKVxuICAgICAgICB0aGlzLmV2ZW50cy5oZWFydGJlYXQ/Lih0aGlzKVxuXG4gICAgICAgIGJyZWFrXG4gICAgICB9XG4gICAgICBjYXNlIEdhdGV3YXlPcGNvZGVzLkhlbGxvOiB7XG4gICAgICAgIGNvbnN0IGludGVydmFsID0gKHBhY2tldC5kIGFzIERpc2NvcmRIZWxsbykuaGVhcnRiZWF0X2ludGVydmFsXG4gICAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gSGVsbG8gb24gU2hhcmQgIyR7dGhpcy5pZH1gKVxuICAgICAgICB0aGlzLnN0YXJ0SGVhcnRiZWF0aW5nKGludGVydmFsKVxuXG4gICAgICAgIGlmICh0aGlzLnN0YXRlICE9PSBTaGFyZFN0YXRlLlJlc3VtaW5nKSB7XG4gICAgICAgICAgY29uc3QgY3VycmVudFF1ZXVlID0gWy4uLnRoaXMuYnVja2V0LnF1ZXVlXVxuICAgICAgICAgIC8vIEhFTExPIGhhcyBiZWVuIHNlbmQgb24gYSBub24gcmVzdW1lIGFjdGlvbi5cbiAgICAgICAgICAvLyBUaGlzIG1lYW5zIHRoYXQgdGhlIHNoYXJkIHN0YXJ0cyBhIG5ldyBzZXNzaW9uLFxuICAgICAgICAgIC8vIHRoZXJlZm9yZSB0aGUgcmF0ZSBsaW1pdCBpbnRlcnZhbCBoYXMgYmVlbiByZXNldCB0b28uXG4gICAgICAgICAgdGhpcy5idWNrZXQgPSBuZXcgTGVha3lCdWNrZXQoe1xuICAgICAgICAgICAgbWF4OiB0aGlzLmNhbGN1bGF0ZVNhZmVSZXF1ZXN0cygpLFxuICAgICAgICAgICAgcmVmaWxsSW50ZXJ2YWw6IDYwMDAwLFxuICAgICAgICAgICAgcmVmaWxsQW1vdW50OiB0aGlzLmNhbGN1bGF0ZVNhZmVSZXF1ZXN0cygpLFxuICAgICAgICAgIH0pXG5cbiAgICAgICAgICAvLyBRdWV1ZSBzaG91bGQgbm90IGJlIGxvc3Qgb24gYSByZS1pZGVudGlmeS5cbiAgICAgICAgICB0aGlzLmJ1Y2tldC5xdWV1ZS51bnNoaWZ0KC4uLmN1cnJlbnRRdWV1ZSlcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuZXZlbnRzLmhlbGxvPy4odGhpcylcblxuICAgICAgICBicmVha1xuICAgICAgfVxuICAgICAgY2FzZSBHYXRld2F5T3Bjb2Rlcy5IZWFydGJlYXRBQ0s6IHtcbiAgICAgICAgLy8gTWFudWFsbHkgY2FsY3VsYXRpbmcgdGhlIHJvdW5kIHRyaXAgdGltZSBmb3IgdXNlcnMgd2hvIG5lZWQgaXQuXG4gICAgICAgIGlmICh0aGlzLmhlYXJ0Lmxhc3RCZWF0KSB7XG4gICAgICAgICAgdGhpcy5oZWFydC5ydHQgPSB0aGlzLmhlYXJ0Lmxhc3RBY2sgLSB0aGlzLmhlYXJ0Lmxhc3RCZWF0XG4gICAgICAgIH1cblxuICAgICAgICB0aGlzLmV2ZW50cy5oZWFydGJlYXRBY2s/Lih0aGlzKVxuXG4gICAgICAgIGJyZWFrXG4gICAgICB9XG4gICAgICBjYXNlIEdhdGV3YXlPcGNvZGVzLlJlY29ubmVjdDoge1xuICAgICAgICAvLyAgIGdhdGV3YXkuZGVidWcoXCJHVyBSRUNPTk5FQ1RcIiwgeyBzaGFyZElkIH0pO1xuXG4gICAgICAgIHRoaXMuZXZlbnRzLnJlcXVlc3RlZFJlY29ubmVjdD8uKHRoaXMpXG5cbiAgICAgICAgYXdhaXQgdGhpcy5yZXN1bWUoKVxuXG4gICAgICAgIGJyZWFrXG4gICAgICB9XG4gICAgICBjYXNlIEdhdGV3YXlPcGNvZGVzLkludmFsaWRTZXNzaW9uOiB7XG4gICAgICAgIGNvbnN0IHJlc3VtYWJsZSA9IHBhY2tldC5kIGFzIGJvb2xlYW5cbiAgICAgICAgdGhpcy5sb2dnZXIuZGVidWcoYFtTaGFyZF0gUmVjZWl2ZWQgSW52YWxpZCBTZXNzaW9uIGZvciBTaGFyZCAjJHt0aGlzLmlkfSB3aXRoIHJlc3VtZWFibGUgYXMgJHtyZXN1bWFibGUudG9TdHJpbmcoKX1gKVxuXG4gICAgICAgIHRoaXMuZXZlbnRzLmludmFsaWRTZXNzaW9uPy4odGhpcywgcmVzdW1hYmxlKVxuXG4gICAgICAgIC8vIFdlIG5lZWQgdG8gd2FpdCBmb3IgYSByYW5kb20gYW1vdW50IG9mIHRpbWUgYmV0d2VlbiAxIGFuZCA1XG4gICAgICAgIC8vIFJlZmVyZW5jZTogaHR0cHM6Ly9kaXNjb3JkLmNvbS9kZXZlbG9wZXJzL2RvY3MvdG9waWNzL2dhdGV3YXkjcmVzdW1pbmdcbiAgICAgICAgYXdhaXQgZGVsYXkoTWF0aC5mbG9vcigoTWF0aC5yYW5kb20oKSAqIDQgKyAxKSAqIDEwMDApKVxuXG4gICAgICAgIHRoaXMucmVzb2x2ZXMuZ2V0KCdJTlZBTElEX1NFU1NJT04nKT8uKHBhY2tldClcbiAgICAgICAgdGhpcy5yZXNvbHZlcy5kZWxldGUoJ0lOVkFMSURfU0VTU0lPTicpXG5cbiAgICAgICAgLy8gV2hlbiByZXN1bWFibGUgaXMgZmFsc2Ugd2UgbmVlZCB0byByZS1pZGVudGlmeVxuICAgICAgICBpZiAoIXJlc3VtYWJsZSkge1xuICAgICAgICAgIGF3YWl0IHRoaXMucmVxdWVzdElkZW50aWZ5KClcblxuICAgICAgICAgIGJyZWFrXG4gICAgICAgIH1cblxuICAgICAgICAvLyBUaGUgc2Vzc2lvbiBpcyBpbnZhbGlkIGJ1dCBhcHBhcmVudGx5IGl0IGlzIHJlc3VtYWJsZVxuICAgICAgICBhd2FpdCB0aGlzLnJlc3VtZSgpXG5cbiAgICAgICAgYnJlYWtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBzd2l0Y2ggKHBhY2tldC50KSB7XG4gICAgICBjYXNlICdSRVNVTUVEJzpcbiAgICAgICAgdGhpcy5zdGF0ZSA9IFNoYXJkU3RhdGUuQ29ubmVjdGVkXG4gICAgICAgIHRoaXMuZXZlbnRzLnJlc3VtZWQ/Lih0aGlzKVxuXG4gICAgICAgIC8vIENvbnRpbnVlIHRoZSByZXF1ZXN0cyB3aGljaCBoYXZlIGJlZW4gcXVldWVkIHNpbmNlIHRoZSBzaGFyZCB3ZW50IG9mZmxpbmUuXG4gICAgICAgIHRoaXMub2ZmbGluZVNlbmRRdWV1ZS5tYXAoKHJlc29sdmUpID0+IHJlc29sdmUoKSlcblxuICAgICAgICB0aGlzLnJlc29sdmVzLmdldCgnUkVTVU1FRCcpPy4ocGFja2V0KVxuICAgICAgICB0aGlzLnJlc29sdmVzLmRlbGV0ZSgnUkVTVU1FRCcpXG4gICAgICAgIGJyZWFrXG4gICAgICBjYXNlICdSRUFEWSc6IHtcbiAgICAgICAgLy8gSW1wb3J0YW50IGZvciBmdXR1cmUgcmVzdW1lcy5cbiAgICAgICAgY29uc3QgcGF5bG9hZCA9IHBhY2tldC5kIGFzIERpc2NvcmRSZWFkeVxuXG4gICAgICAgIHRoaXMucmVzdW1lR2F0ZXdheVVybCA9IHBheWxvYWQucmVzdW1lX2dhdGV3YXlfdXJsXG5cbiAgICAgICAgdGhpcy5zZXNzaW9uSWQgPSBwYXlsb2FkLnNlc3Npb25faWRcbiAgICAgICAgdGhpcy5zdGF0ZSA9IFNoYXJkU3RhdGUuQ29ubmVjdGVkXG5cbiAgICAgICAgLy8gQ29udGludWUgdGhlIHJlcXVlc3RzIHdoaWNoIGhhdmUgYmVlbiBxdWV1ZWQgc2luY2UgdGhlIHNoYXJkIHdlbnQgb2ZmbGluZS5cbiAgICAgICAgLy8gSW1wb3J0YW50IHdoZW4gdGhpcyBpcyBhIHJlLWlkZW50aWZ5XG4gICAgICAgIHRoaXMub2ZmbGluZVNlbmRRdWV1ZS5tYXAoKHJlc29sdmUpID0+IHJlc29sdmUoKSlcblxuICAgICAgICB0aGlzLnJlc29sdmVzLmdldCgnUkVBRFknKT8uKHBhY2tldClcbiAgICAgICAgdGhpcy5yZXNvbHZlcy5kZWxldGUoJ1JFQURZJylcbiAgICAgICAgYnJlYWtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBVcGRhdGUgdGhlIHNlcXVlbmNlIG51bWJlciBpZiBpdCBpcyBwcmVzZW50XG4gICAgLy8gYHNgIGNhbiBiZSBlaXRoZXIgYG51bGxgIG9yIGEgYG51bWJlcmAuXG4gICAgLy8gSW4gb3JkZXIgdG8gcHJldmVudCB1cGRhdGUgbWlzc2VzIHdoZW4gYHNgIGlzIGAwYCB3ZSBjaGVjayBhZ2FpbnN0IG51bGwuXG4gICAgaWYgKHBhY2tldC5zICE9PSBudWxsKSB7XG4gICAgICB0aGlzLnByZXZpb3VzU2VxdWVuY2VOdW1iZXIgPSBwYWNrZXQuc1xuICAgIH1cblxuICAgIHRoaXMuZm9yd2FyZFRvQm90KHBhY2tldClcbiAgfVxuXG4gIGZvcndhcmRUb0JvdChwYWNrZXQ6IERpc2NvcmRHYXRld2F5UGF5bG9hZCk6IHZvaWQge1xuICAgIC8vIFRoZSBuZWNlc3NhcnkgaGFuZGxpbmcgcmVxdWlyZWQgZm9yIHRoZSBTaGFyZHMgY29ubmVjdGlvbiBoYXMgYmVlbiBmaW5pc2hlZC5cbiAgICAvLyBOb3cgdGhlIGV2ZW50IGNhbiBiZSBzYWZlbHkgZm9yd2FyZGVkLlxuICAgIHRoaXMuZXZlbnRzLm1lc3NhZ2U/Lih0aGlzLCBjYW1lbGl6ZShwYWNrZXQpKVxuICB9XG5cbiAgLyoqIEhhbmRsZSBhbiBpbmNvbWluZyBnYXRld2F5IG1lc3NhZ2UuICovXG4gIGFzeW5jIGhhbmRsZU1lc3NhZ2UobWVzc2FnZTogTm9kZVdlYlNvY2tldC5NZXNzYWdlRXZlbnQpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBsZXQgcHJlUHJvY2Vzc01lc3NhZ2UgPSBtZXNzYWdlLmRhdGFcblxuICAgIC8vIElmIG1lc3NhZ2UgY29tcHJlc3Npb24gaXMgZW5hYmxlZCxcbiAgICAvLyBEaXNjb3JkIG1pZ2h0IHNlbmQgemxpYiBjb21wcmVzc2VkIHBheWxvYWRzLlxuICAgIGlmICh0aGlzLmdhdGV3YXlDb25maWcuY29tcHJlc3MgJiYgcHJlUHJvY2Vzc01lc3NhZ2UgaW5zdGFuY2VvZiBCbG9iKSB7XG4gICAgICBwcmVQcm9jZXNzTWVzc2FnZSA9IGluZmxhdGVTeW5jKGF3YWl0IHByZVByb2Nlc3NNZXNzYWdlLmFycmF5QnVmZmVyKCkpLnRvU3RyaW5nKClcbiAgICB9XG5cbiAgICAvLyBTYWZlZ3VhcmQgaW5jYXNlIGRlY29tcHJlc3Npb24gZmFpbGVkIHRvIG1ha2UgYSBzdHJpbmcuXG4gICAgaWYgKHR5cGVvZiBwcmVQcm9jZXNzTWVzc2FnZSAhPT0gJ3N0cmluZycpIHJldHVyblxuXG4gICAgcmV0dXJuIGF3YWl0IHRoaXMuaGFuZGxlRGlzY29yZFBhY2tldChKU09OLnBhcnNlKHByZVByb2Nlc3NNZXNzYWdlKSBhcyBEaXNjb3JkR2F0ZXdheVBheWxvYWQpXG4gIH1cblxuICAvKipcbiAgICogT3ZlcnJpZGUgaW4gb3JkZXIgdG8gbWFrZSB0aGUgc2hhcmRzIHByZXNlbmNlLlxuICAgKiBhc3luYyBpbiBjYXNlIGRldnMgY3JlYXRlIHRoZSBwcmVzZW5jZSBiYXNlZCBvbiBlZy4gZGF0YWJhc2UgdmFsdWVzLlxuICAgKiBQYXNzaW5nIHRoZSBzaGFyZCdzIGlkIHRoZXJlIHRvIG1ha2UgaXQgZWFzaWVyIGZvciB0aGUgZGV2IHRvIHVzZSB0aGlzIGZ1bmN0aW9uLlxuICAgKi9cbiAgYXN5bmMgbWFrZVByZXNlbmNlKCk6IFByb21pc2U8Qm90U3RhdHVzVXBkYXRlIHwgdW5kZWZpbmVkPiB7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIG5vLXVzZWxlc3MtcmV0dXJuXG4gICAgcmV0dXJuXG4gIH1cblxuICAvKiogVGhpcyBmdW5jdGlvbiBjb21tdW5pY2F0ZXMgd2l0aCB0aGUgbWFuYWdlbWVudCBwcm9jZXNzLCBpbiBvcmRlciB0byBrbm93IHdoZXRoZXIgaXRzIGZyZWUgdG8gaWRlbnRpZnkuIFdoZW4gdGhpcyBmdW5jdGlvbiByZXNvbHZlcywgdGhpcyBtZWFucyB0aGF0IHRoZSBzaGFyZCBpcyBhbGxvd2VkIHRvIHNlbmQgYW4gaWRlbnRpZnkgcGF5bG9hZCB0byBkaXNjb3JkLiAqL1xuICBhc3luYyByZXF1ZXN0SWRlbnRpZnkoKTogUHJvbWlzZTx2b2lkPiB7fVxuXG4gIC8qKiBUaGlzIGZ1bmN0aW9uIGNvbW11bmljYXRlcyB3aXRoIHRoZSBtYW5hZ2VtZW50IHByb2Nlc3MsIGluIG9yZGVyIHRvIHRlbGwgaXQgY2FuIGlkZW50aWZ5IHRoZSBuZXh0IHNoYXJkLiAqL1xuICBhc3luYyBzaGFyZElzUmVhZHkoKTogUHJvbWlzZTx2b2lkPiB7fVxuXG4gIC8qKiBTdGFydCBzZW5kaW5nIGhlYXJ0YmVhdCBwYXlsb2FkcyB0byBEaXNjb3JkIGluIHRoZSBwcm92aWRlZCBpbnRlcnZhbC4gKi9cbiAgc3RhcnRIZWFydGJlYXRpbmcoaW50ZXJ2YWw6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gU3RhcnQgSGVhcnRiZWF0aW5nIFNoYXJkICMke3RoaXMuaWR9YClcbiAgICAvLyBJZiBvbGQgaGVhcnRiZWFzdCBleGlzdCBsaWtlIGFmdGVyIHJlc3VtZSwgY2xlYXIgdGhlIG9sZCBvbmVzLlxuICAgIGlmICh0aGlzLmhlYXJ0LmludGVydmFsSWQpIGNsZWFySW50ZXJ2YWwodGhpcy5oZWFydC5pbnRlcnZhbElkKVxuICAgIGlmICh0aGlzLmhlYXJ0LnRpbWVvdXRJZCkgY2xlYXJUaW1lb3V0KHRoaXMuaGVhcnQudGltZW91dElkKVxuXG4gICAgdGhpcy5oZWFydC5pbnRlcnZhbCA9IGludGVydmFsXG5cbiAgICAvLyBPbmx5IHNldCB0aGUgc2hhcmQncyBzdGF0ZSB0byBgVW5pZGVudGlmaWVkYFxuICAgIC8vIGlmIGhlYXJ0YmVhdGluZyBoYXMgbm90IGJlZW4gc3RhcnRlZCBkdWUgdG8gYW4gaWRlbnRpZnkgb3IgcmVzdW1lIGFjdGlvbi5cbiAgICBpZiAoW1NoYXJkU3RhdGUuRGlzY29ubmVjdGVkLCBTaGFyZFN0YXRlLk9mZmxpbmVdLmluY2x1ZGVzKHRoaXMuc3RhdGUpKSB7XG4gICAgICB0aGlzLmxvZ2dlci5kZWJ1ZyhgW0dhdGV3YXldIFN0YXJ0IEhlYXJ0YmVhdGluZyBTaGFyZCAjJHt0aGlzLmlkfSBhYClcbiAgICAgIHRoaXMuc3RhdGUgPSBTaGFyZFN0YXRlLlVuaWRlbnRpZmllZFxuICAgIH1cblxuICAgIC8vIFRoZSBmaXJzdCBoZWFydGJlYXQgbmVlZHMgdG8gYmUgc2VuZCB3aXRoIGEgcmFuZG9tIGRlbGF5IGJldHdlZW4gYDBgIGFuZCBgaW50ZXJ2YWxgXG4gICAgLy8gVXNpbmcgYSBgc2V0VGltZW91dChfLCBqaXR0ZXIpYCBoZXJlIHRvIGFjY29tcGxpc2ggdGhhdC5cbiAgICAvLyBgTWF0aC5yYW5kb20oKWAgY2FuIGJlIGAwYCBzbyB3ZSB1c2UgYDAuNWAgaWYgdGhpcyBoYXBwZW5zXG4gICAgLy8gUmVmZXJlbmNlOiBodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3MvZ2F0ZXdheSNoZWFydGJlYXRpbmdcbiAgICBjb25zdCBqaXR0ZXIgPSBNYXRoLmNlaWwodGhpcy5oZWFydC5pbnRlcnZhbCAqIChNYXRoLnJhbmRvbSgpIHx8IDAuNSkpXG4gICAgdGhpcy5oZWFydC50aW1lb3V0SWQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gU3RhcnQgSGVhcnRiZWF0aW5nIFNoYXJkICMke3RoaXMuaWR9IGJgKVxuICAgICAgaWYgKCF0aGlzLmlzT3BlbigpKSByZXR1cm5cbiAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBbR2F0ZXdheV0gU3RhcnQgSGVhcnRiZWF0aW5nIFNoYXJkICMke3RoaXMuaWR9IGMgJHt0aGlzLnByZXZpb3VzU2VxdWVuY2VOdW1iZXIhfWApXG5cbiAgICAgIC8vIFVzaW5nIGEgZGlyZWN0IHNvY2tldC5zZW5kIGNhbGwgaGVyZSBiZWNhdXNlIGhlYXJ0YmVhdCByZXF1ZXN0cyBhcmUgcmVzZXJ2ZWQgYnkgdXMuXG4gICAgICB0aGlzLnNvY2tldD8uc2VuZChcbiAgICAgICAgSlNPTi5zdHJpbmdpZnkoe1xuICAgICAgICAgIG9wOiBHYXRld2F5T3Bjb2Rlcy5IZWFydGJlYXQsXG4gICAgICAgICAgZDogdGhpcy5wcmV2aW91c1NlcXVlbmNlTnVtYmVyLFxuICAgICAgICB9KSxcbiAgICAgIClcblxuICAgICAgdGhpcy5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBTdGFydCBIZWFydGJlYXRpbmcgU2hhcmQgIyR7dGhpcy5pZH0gZGApXG4gICAgICB0aGlzLmhlYXJ0Lmxhc3RCZWF0ID0gRGF0ZS5ub3coKVxuICAgICAgdGhpcy5oZWFydC5hY2tub3dsZWRnZWQgPSBmYWxzZVxuXG4gICAgICAvLyBBZnRlciB0aGUgcmFuZG9tIGhlYXJ0YmVhdCBqaXR0ZXIgd2UgY2FuIHN0YXJ0IGEgbm9ybWFsIGludGVydmFsLlxuICAgICAgdGhpcy5oZWFydC5pbnRlcnZhbElkID0gc2V0SW50ZXJ2YWwoYXN5bmMgKCkgPT4ge1xuICAgICAgICB0aGlzLmxvZ2dlci5kZWJ1ZyhgW0dhdGV3YXldIFN0YXJ0IEhlYXJ0YmVhdGluZyBTaGFyZCAjJHt0aGlzLmlkfSBlYClcbiAgICAgICAgaWYgKCF0aGlzLmlzT3BlbigpKSByZXR1cm5cbiAgICAgICAgdGhpcy5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBTdGFydCBIZWFydGJlYXRpbmcgU2hhcmQgIyR7dGhpcy5pZH0gZmApXG4gICAgICAgIC8vIGdhdGV3YXkuZGVidWcoXCJHVyBERUJVR1wiLCBgUnVubmluZyBzZXRJbnRlcnZhbCBpbiBoZWFydGJlYXQgZmlsZS4gU2hhcmQ6ICR7c2hhcmRJZH1gKTtcblxuICAgICAgICAvLyBnYXRld2F5LmRlYnVnKFwiR1cgSEVBUlRCRUFUSU5HXCIsIHsgc2hhcmRJZCwgc2hhcmQ6IGN1cnJlbnRTaGFyZCB9KTtcblxuICAgICAgICAvLyBUaGUgU2hhcmQgZGlkIG5vdCByZWNlaXZlIGEgaGVhcnRiZWF0IEFDSyBmcm9tIERpc2NvcmQgaW4gdGltZSxcbiAgICAgICAgLy8gdGhlcmVmb3JlIHdlIGhhdmUgdG8gYXNzdW1lIHRoYXQgdGhlIGNvbm5lY3Rpb24gaGFzIGZhaWxlZCBvciBnb3QgXCJ6b21iaWVkXCIuXG4gICAgICAgIC8vIFRoZSBTaGFyZCBuZWVkcyB0byBzdGFydCBhIHJlLWlkZW50aWZ5IGFjdGlvbiBhY2NvcmRpbmdseS5cbiAgICAgICAgLy8gUmVmZXJlbmNlOiBodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3MvZ2F0ZXdheSNoZWFydGJlYXRpbmctZXhhbXBsZS1nYXRld2F5LWhlYXJ0YmVhdC1hY2tcbiAgICAgICAgaWYgKCF0aGlzLmhlYXJ0LmFja25vd2xlZGdlZCkge1xuICAgICAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBbU2hhcmRdIEhlYXJ0YmVhdCBub3QgYWNrbm93bGVkZ2VkIGZvciBzaGFyZCAjJHt0aGlzLmlkfS5gKVxuICAgICAgICAgIHRoaXMuY2xvc2UoU2hhcmRTb2NrZXRDbG9zZUNvZGVzLlpvbWJpZWRDb25uZWN0aW9uLCAnWm9tYmllZCBjb25uZWN0aW9uLCBkaWQgbm90IHJlY2VpdmUgYW4gaGVhcnRiZWF0IEFDSyBpbiB0aW1lLicpXG5cbiAgICAgICAgICByZXR1cm4gYXdhaXQgdGhpcy5pZGVudGlmeSgpXG4gICAgICAgIH1cblxuICAgICAgICB0aGlzLmhlYXJ0LmFja25vd2xlZGdlZCA9IGZhbHNlXG5cbiAgICAgICAgdGhpcy5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBTdGFydCBIZWFydGJlYXRpbmcgU2hhcmQgIyR7dGhpcy5pZH0gZ2ApXG4gICAgICAgIC8vIFVzaW5nIGEgZGlyZWN0IHNvY2tldC5zZW5kIGNhbGwgaGVyZSBiZWNhdXNlIGhlYXJ0YmVhdCByZXF1ZXN0cyBhcmUgcmVzZXJ2ZWQgYnkgdXMuXG4gICAgICAgIHRoaXMuc29ja2V0Py5zZW5kKFxuICAgICAgICAgIEpTT04uc3RyaW5naWZ5KHtcbiAgICAgICAgICAgIG9wOiBHYXRld2F5T3Bjb2Rlcy5IZWFydGJlYXQsXG4gICAgICAgICAgICBkOiB0aGlzLnByZXZpb3VzU2VxdWVuY2VOdW1iZXIsXG4gICAgICAgICAgfSksXG4gICAgICAgIClcbiAgICAgICAgdGhpcy5sb2dnZXIuZGVidWcoYFtHYXRld2F5XSBTdGFydCBIZWFydGJlYXRpbmcgU2hhcmQgIyR7dGhpcy5pZH0gaGApXG5cbiAgICAgICAgdGhpcy5oZWFydC5sYXN0QmVhdCA9IERhdGUubm93KClcblxuICAgICAgICB0aGlzLmV2ZW50cy5oZWFydGJlYXQ/Lih0aGlzKVxuICAgICAgfSwgdGhpcy5oZWFydC5pbnRlcnZhbClcbiAgICB9LCBqaXR0ZXIpXG4gIH1cblxuICAvKiogU3RvcCB0aGUgaGVhcnRiZWF0aW5nIHByb2Nlc3Mgd2l0aCBkaXNjb3JkLiAqL1xuICBzdG9wSGVhcnRiZWF0aW5nKCk6IHZvaWQge1xuICAgIC8vIENsZWFyIHRoZSByZWd1bGFyIGhlYXJ0YmVhdCBpbnRlcnZhbC5cbiAgICBjbGVhckludGVydmFsKHRoaXMuaGVhcnQuaW50ZXJ2YWxJZClcbiAgICAvLyBJdCdzIHBvc3NpYmxlIHRoYXQgdGhlIFNoYXJkIGdvdCBjbG9zZWQgYmVmb3JlIHRoZSBmaXJzdCBqaXR0ZXJlZCBoZWFydGJlYXQuXG4gICAgLy8gVG8gZ28gc2FmZSB3ZSBzaG91bGQgY2xlYXIgdGhlIHJlbGF0ZWQgdGltZW91dCB0b28uXG4gICAgY2xlYXJUaW1lb3V0KHRoaXMuaGVhcnQudGltZW91dElkKVxuICB9XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2hhcmRDcmVhdGVPcHRpb25zIHtcbiAgLyoqIFRoZSBzaGFyZCBpZCAqL1xuICBpZDogbnVtYmVyXG4gIC8qKiBUaGUgY29ubmVjdGlvbiBkZXRhaWxzICovXG4gIGNvbm5lY3Rpb246IFNoYXJkR2F0ZXdheUNvbmZpZ1xuICAvKiogVGhlIGV2ZW50IGhhbmRsZXJzIGZvciBldmVudHMgb24gdGhlIHNoYXJkLiAqL1xuICBldmVudHM6IFNoYXJkRXZlbnRzXG4gIC8qKiBUaGUgbG9nZ2VyIGZvciB0aGUgc2hhcmQgKi9cbiAgbG9nZ2VyPzogUGljazx0eXBlb2YgbG9nZ2VyLCAnZGVidWcnIHwgJ2luZm8nIHwgJ3dhcm4nIHwgJ2Vycm9yJyB8ICdmYXRhbCc+XG4gIC8qKiBUaGUgaGFuZGxlciB0byByZXF1ZXN0IGEgc3BhY2UgdG8gbWFrZSBhbiBpZGVudGlmeSByZXF1ZXN0LiAqL1xuICByZXF1ZXN0SWRlbnRpZnk/OiAoKSA9PiBQcm9taXNlPHZvaWQ+XG4gIC8qKiBUaGUgaGFuZGxlciB0byBhbGVydCB0aGUgZ2F0ZXdheSBtYW5hZ2VyIHRoYXQgdGhpcyBzaGFyZCBoYXMgcmVjZWl2ZWQgYSBSRUFEWSBldmVudC4gKi9cbiAgc2hhcmRJc1JlYWR5PzogKCkgPT4gUHJvbWlzZTx2b2lkPlxufVxuXG5leHBvcnQgZGVmYXVsdCBEaXNjb3JkZW5vU2hhcmRcbiJdLCJuYW1lcyI6WyJEaXNjb3JkZW5vU2hhcmQiLCJjb25zdHJ1Y3RvciIsIm9wdGlvbnMiLCJtYXhSZXF1ZXN0c1BlclJhdGVMaW1pdFRpY2siLCJwcmV2aW91c1NlcXVlbmNlTnVtYmVyIiwicmF0ZUxpbWl0UmVzZXRJbnRlcnZhbCIsInN0YXRlIiwiU2hhcmRTdGF0ZSIsIk9mZmxpbmUiLCJyZXN1bWVHYXRld2F5VXJsIiwiZXZlbnRzIiwib2ZmbGluZVNlbmRRdWV1ZSIsInJlc29sdmVzIiwiTWFwIiwiaWQiLCJjb25uZWN0aW9uIiwibG9nZ2VyIiwiaGVhcnQiLCJhY2tub3dsZWRnZWQiLCJpbnRlcnZhbCIsInJlcXVlc3RJZGVudGlmeSIsInNoYXJkSXNSZWFkeSIsImJ1Y2tldCIsIkxlYWt5QnVja2V0IiwibWF4IiwiY2FsY3VsYXRlU2FmZVJlcXVlc3RzIiwicmVmaWxsQW1vdW50IiwicmVmaWxsSW50ZXJ2YWwiLCJnYXRld2F5Q29uZmlnIiwiY29ubmVjdGlvblVybCIsInVybCIsInNhZmVSZXF1ZXN0cyIsIk1hdGgiLCJjZWlsIiwiY2hlY2tPZmZsaW5lIiwiaGlnaFByaW9yaXR5IiwiaXNPcGVuIiwiUHJvbWlzZSIsInJlc29sdmUiLCJ1bnNoaWZ0IiwicHVzaCIsImNsb3NlIiwiY29kZSIsInJlYXNvbiIsInNvY2tldCIsInJlYWR5U3RhdGUiLCJOb2RlV2ViU29ja2V0IiwiT1BFTiIsImNvbm5lY3QiLCJJZGVudGlmeWluZyIsIlJlc3VtaW5nIiwiaW5jbHVkZXMiLCJDb25uZWN0aW5nIiwiY29ubmVjdGluZyIsIlVSTCIsInNlYXJjaFBhcmFtcyIsInNldCIsInZlcnNpb24iLCJ0b1N0cmluZyIsImdsb2JhbFRoaXMiLCJEZW5vIiwidW5kZWZpbmVkIiwiUmVmbGVjdCIsImhhcyIsIldlYlNvY2tldCIsIm9uZXJyb3IiLCJldmVudCIsImNvbnNvbGUiLCJsb2ciLCJlcnJvciIsInNoYXJkSWQiLCJvbmNsb3NlIiwiaGFuZGxlQ2xvc2UiLCJvbm1lc3NhZ2UiLCJtZXNzYWdlIiwiaGFuZGxlTWVzc2FnZSIsIm9ub3BlbiIsIlVuaWRlbnRpZmllZCIsImNvbm5lY3RlZCIsImlkZW50aWZ5IiwiZGVidWciLCJTaGFyZFNvY2tldENsb3NlQ29kZXMiLCJSZUlkZW50aWZ5aW5nIiwiaWRlbnRpZnlpbmciLCJzZW5kIiwib3AiLCJHYXRld2F5T3Bjb2RlcyIsIklkZW50aWZ5IiwiZCIsInRva2VuIiwiY29tcHJlc3MiLCJwcm9wZXJ0aWVzIiwiaW50ZW50cyIsInNoYXJkIiwidG90YWxTaGFyZHMiLCJwcmVzZW5jZSIsIm1ha2VQcmVzZW5jZSIsImlkZW50aWZpZWQiLCJkZWxldGUiLCJyZXN1bWUiLCJSZXN1bWVDbG9zaW5nT2xkQ29ubmVjdGlvbiIsInNlc3Npb25JZCIsIlJlc3VtZSIsInNlc3Npb25faWQiLCJzZXEiLCJhY3F1aXJlIiwiSlNPTiIsInN0cmluZ2lmeSIsInNodXRkb3duIiwiU2h1dGRvd24iLCJzdG9wSGVhcnRiZWF0aW5nIiwiVGVzdGluZ0ZpbmlzaGVkIiwiZGlzY29ubmVjdGVkIiwiUmVzaGFyZGVkIiwiWm9tYmllZENvbm5lY3Rpb24iLCJEaXNjb25uZWN0ZWQiLCJHYXRld2F5Q2xvc2VFdmVudENvZGVzIiwiVW5rbm93bk9wY29kZSIsIk5vdEF1dGhlbnRpY2F0ZWQiLCJJbnZhbGlkU2VxIiwiUmF0ZUxpbWl0ZWQiLCJTZXNzaW9uVGltZWRPdXQiLCJBdXRoZW50aWNhdGlvbkZhaWxlZCIsIkludmFsaWRTaGFyZCIsIlNoYXJkaW5nUmVxdWlyZWQiLCJJbnZhbGlkQXBpVmVyc2lvbiIsIkludmFsaWRJbnRlbnRzIiwiRGlzYWxsb3dlZEludGVudHMiLCJFcnJvciIsIlVua25vd25FcnJvciIsIkRlY29kZUVycm9yIiwiQWxyZWFkeUF1dGhlbnRpY2F0ZWQiLCJpbmZvIiwiaGFuZGxlRGlzY29yZFBhY2tldCIsInBhY2tldCIsImxhc3RBY2siLCJEYXRlIiwibm93IiwiSGVhcnRiZWF0IiwibGFzdEJlYXQiLCJoZWFydGJlYXQiLCJIZWxsbyIsImhlYXJ0YmVhdF9pbnRlcnZhbCIsInN0YXJ0SGVhcnRiZWF0aW5nIiwiY3VycmVudFF1ZXVlIiwicXVldWUiLCJoZWxsbyIsIkhlYXJ0YmVhdEFDSyIsInJ0dCIsImhlYXJ0YmVhdEFjayIsIlJlY29ubmVjdCIsInJlcXVlc3RlZFJlY29ubmVjdCIsIkludmFsaWRTZXNzaW9uIiwicmVzdW1hYmxlIiwiaW52YWxpZFNlc3Npb24iLCJkZWxheSIsImZsb29yIiwicmFuZG9tIiwiZ2V0IiwidCIsIkNvbm5lY3RlZCIsInJlc3VtZWQiLCJtYXAiLCJwYXlsb2FkIiwicmVzdW1lX2dhdGV3YXlfdXJsIiwicyIsImZvcndhcmRUb0JvdCIsImNhbWVsaXplIiwicHJlUHJvY2Vzc01lc3NhZ2UiLCJkYXRhIiwiQmxvYiIsImluZmxhdGVTeW5jIiwiYXJyYXlCdWZmZXIiLCJwYXJzZSIsImludGVydmFsSWQiLCJjbGVhckludGVydmFsIiwidGltZW91dElkIiwiY2xlYXJUaW1lb3V0Iiwiaml0dGVyIiwic2V0VGltZW91dCIsInNldEludGVydmFsIl0sIm1hcHBpbmdzIjoiQUFBQSxrRUFBa0U7Ozs7Ozs7Ozs7O0lBV3JEQSxlQUFlO2VBQWZBOztJQW9sQmIsT0FBOEI7ZUFBOUI7Ozt1QkE3bEJ1RDt1QkFDRjswQkFDekI7MkRBQ0Y7d0JBRXdCOzs7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
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ _export_star(require("./manager.cjs"), exports);
6
+ _export_star(require("./Shard.cjs"), exports);
7
+ _export_star(require("./types.cjs"), exports);
8
+ function _export_star(from, to) {
9
+ Object.keys(from).forEach(function(k) {
10
+ if (k !== "default" && !Object.prototype.hasOwnProperty.call(to, k)) {
11
+ Object.defineProperty(to, k, {
12
+ enumerable: true,
13
+ get: function() {
14
+ return from[k];
15
+ }
16
+ });
17
+ }
18
+ });
19
+ return from;
20
+ }
21
+
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL21hbmFnZXIuanMnXG5leHBvcnQgKiBmcm9tICcuL1NoYXJkLmpzJ1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcy5qcydcbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7O3FCQUFjO3FCQUNBO3FCQUNBIn0=