@discordeno/gateway 19.0.0-next.bb85624 → 19.0.0-next.bccbc73

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/Shard.d.ts CHANGED
@@ -1,6 +1,6 @@
1
1
  import type { AtLeastOne, BigString, Camelize, DiscordGatewayPayload, DiscordMember, RequestGuildMembers } from '@discordeno/types';
2
2
  import { Collection, LeakyBucket } from '@discordeno/utils';
3
- import WebSocket from 'ws';
3
+ import NodeWebSocket from 'ws';
4
4
  import type { RequestMemberRequest } from './manager.js';
5
5
  import type { BotStatusUpdate, ShardEvents, ShardGatewayConfig, ShardHeart, ShardSocketRequest, StatusUpdate, UpdateVoiceState } from './types.js';
6
6
  import { ShardState } from './types.js';
@@ -20,7 +20,7 @@ export declare class DiscordenoShard {
20
20
  /** Current session id of the shard if present. */
21
21
  sessionId?: string;
22
22
  /** This contains the WebSocket connection to Discord, if currently connected. */
23
- socket?: WebSocket;
23
+ socket?: NodeWebSocket;
24
24
  /** Current internal state of the this. */
25
25
  state: ShardState;
26
26
  /** The url provided by discord to use when resuming a connection for this this. */
@@ -48,6 +48,8 @@ export declare class DiscordenoShard {
48
48
  constructor(options: ShardCreateOptions);
49
49
  /** The gateway configuration which is used to connect to Discord. */
50
50
  get gatewayConfig(): ShardGatewayConfig;
51
+ /** The url to connect to. Intially this is the discord gateway url, and then is switched to resume gateway url once a READY is received. */
52
+ get connectionUrl(): string;
51
53
  /** 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. */
52
54
  calculateSafeRequests(): number;
53
55
  checkOffline(highPriority: boolean): Promise<void>;
@@ -68,11 +70,12 @@ export declare class DiscordenoShard {
68
70
  /** Shutdown the this. Forcefully disconnect the shard from Discord. The shard may not attempt to reconnect with Discord. */
69
71
  shutdown(): Promise<void>;
70
72
  /** Handle a gateway connection close. */
71
- handleClose(close: WebSocket.CloseEvent): Promise<void>;
73
+ handleClose(close: NodeWebSocket.CloseEvent): Promise<void>;
72
74
  /** Handles a incoming gateway packet. */
73
75
  handleDiscordPacket(packet: DiscordGatewayPayload): Promise<void>;
76
+ forwardToBot(packet: DiscordGatewayPayload): void;
74
77
  /** Handle an incoming gateway message. */
75
- handleMessage(message: WebSocket.MessageEvent): Promise<void>;
78
+ handleMessage(message: NodeWebSocket.MessageEvent): Promise<void>;
76
79
  /**
77
80
  * Override in order to make the shards presence.
78
81
  * async in case devs create the presence based on eg. database values.
@@ -81,6 +84,8 @@ export declare class DiscordenoShard {
81
84
  makePresence(): Promise<BotStatusUpdate | undefined>;
82
85
  /** 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. */
83
86
  requestIdentify(): Promise<void>;
87
+ /** This function communicates with the management process, in order to tell it can identify the next shard. */
88
+ shardIsReady(): Promise<void>;
84
89
  /** Start sending heartbeat payloads to Discord in the provided interval. */
85
90
  startHeartbeating(interval: number): void;
86
91
  /** Stop the heartbeating process with discord. */
@@ -163,6 +168,8 @@ export interface ShardCreateOptions {
163
168
  events: ShardEvents;
164
169
  /** The handler to request a space to make an identify request. */
165
170
  requestIdentify?: () => Promise<void>;
171
+ /** The handler to alert the gateway manager that this shard has received a READY event. */
172
+ shardIsReady?: () => Promise<void>;
166
173
  }
167
174
  export default DiscordenoShard;
168
175
  //# sourceMappingURL=Shard.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"Shard.d.ts","sourceRoot":"","sources":["../src/Shard.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"Shard.d.ts","sourceRoot":"","sources":["../src/Shard.ts"],"names":[],"mappings":"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"}
package/dist/Shard.js CHANGED
@@ -1,27 +1,27 @@
1
- import { GatewayCloseEventCodes, GatewayIntents, GatewayOpcodes } from '@discordeno/types';
2
- import { camelize, Collection, delay, LeakyBucket, logger } from '@discordeno/utils';
1
+ /* eslint-disable @typescript-eslint/no-confusing-void-expression */ import { GatewayCloseEventCodes, GatewayIntents, GatewayOpcodes } from '@discordeno/types';
2
+ import { Collection, LeakyBucket, camelize, delay, logger } from '@discordeno/utils';
3
3
  import { inflateSync } from 'node:zlib';
4
- import WebSocket from 'ws';
4
+ import NodeWebSocket from 'ws';
5
5
  import { ShardSocketCloseCodes, ShardState } from './types.js';
6
6
  export class DiscordenoShard {
7
- /** The maximum of requests which can be send to discord per rate limit tick. Typically this value should not be changed. */ maxRequestsPerRateLimitTick = 120;
8
- /** The previous payload sequence number. */ previousSequenceNumber = null;
9
- /** In which interval (in milliseconds) the gateway resets it's rate limit. */ rateLimitResetInterval = 60000;
10
- /** Current internal state of the this. */ state = ShardState.Offline;
11
- /** The url provided by discord to use when resuming a connection for this this. */ resumeGatewayUrl = '';
12
- /** The shard related event handlers. */ events = {};
13
- /** Cache for pending gateway requests which should have been send while the gateway went offline. */ offlineSendQueue = [];
14
- /** Resolve internal waiting states. Mapped by SelectedEvents => ResolveFunction */ resolves = new Map();
15
- /** This managers cache related settings. */ cache = {
16
- requestMembers: {
17
- /**
7
+ constructor(options){
8
+ /** The maximum of requests which can be send to discord per rate limit tick. Typically this value should not be changed. */ this.maxRequestsPerRateLimitTick = 120;
9
+ /** The previous payload sequence number. */ this.previousSequenceNumber = null;
10
+ /** In which interval (in milliseconds) the gateway resets it's rate limit. */ this.rateLimitResetInterval = 60000;
11
+ /** Current internal state of the this. */ this.state = ShardState.Offline;
12
+ /** The url provided by discord to use when resuming a connection for this this. */ this.resumeGatewayUrl = '';
13
+ /** The shard related event handlers. */ this.events = {};
14
+ /** Cache for pending gateway requests which should have been send while the gateway went offline. */ this.offlineSendQueue = [];
15
+ /** Resolve internal waiting states. Mapped by SelectedEvents => ResolveFunction */ this.resolves = new Map();
16
+ /** This managers cache related settings. */ this.cache = {
17
+ requestMembers: {
18
+ /**
18
19
  * Whether or not request member requests should be cached.
19
20
  * @default false
20
21
  */ enabled: false,
21
- /** The pending requests. */ pending: new Collection()
22
- }
23
- };
24
- constructor(options){
22
+ /** The pending requests. */ pending: new Collection()
23
+ }
24
+ };
25
25
  this.id = options.id;
26
26
  this.connection = options.connection;
27
27
  this.events = options.events;
@@ -30,6 +30,7 @@ export class DiscordenoShard {
30
30
  interval: 45000
31
31
  };
32
32
  if (options.requestIdentify) this.requestIdentify = options.requestIdentify;
33
+ if (options.shardIsReady) this.shardIsReady = options.shardIsReady;
33
34
  this.bucket = new LeakyBucket({
34
35
  max: this.calculateSafeRequests(),
35
36
  refillAmount: this.calculateSafeRequests(),
@@ -39,6 +40,10 @@ export class DiscordenoShard {
39
40
  /** The gateway configuration which is used to connect to Discord. */ get gatewayConfig() {
40
41
  return this.connection;
41
42
  }
43
+ /** The url to connect to. Intially this is the discord gateway url, and then is switched to resume gateway url once a READY is received. */ get connectionUrl() {
44
+ // Use || and not ?? here. ?? will cause a bug.
45
+ return this.resumeGatewayUrl || this.gatewayConfig.url;
46
+ }
42
47
  /** 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() {
43
48
  // * 2 adds extra safety layer for discords OP 1 requests that we need to respond to
44
49
  const safeRequests = this.maxRequestsPerRateLimitTick - Math.ceil(this.rateLimitResetInterval / this.heart.interval) * 2;
@@ -54,8 +59,8 @@ export class DiscordenoShard {
54
59
  }
55
60
  }
56
61
  /** Close the socket connection to discord if present. */ close(code, reason) {
57
- if (this.socket?.readyState !== WebSocket.OPEN) return;
58
- return this.socket?.close(code, reason);
62
+ if (this.socket?.readyState !== NodeWebSocket.OPEN) return;
63
+ this.socket?.close(code, reason);
59
64
  }
60
65
  /** Connect the shard with the gateway and start heartbeating. This will not identify the shard to the gateway. */ async connect() {
61
66
  // Only set the shard to `Connecting` state,
@@ -67,20 +72,16 @@ export class DiscordenoShard {
67
72
  this.state = ShardState.Connecting;
68
73
  }
69
74
  this.events.connecting?.(this);
70
- let url = new URL(this.gatewayConfig.url);
71
- // If not connecting to a proxy but directly to discord need to handle resuming
72
- if (url.origin === 'wss://gateway.discord.gg') {
73
- if (this.state === ShardState.Resuming) {
74
- url = new URL(this.resumeGatewayUrl);
75
- }
76
- url.searchParams.set('v', this.gatewayConfig.version.toString());
77
- url.searchParams.set('encoding', 'json');
78
- }
79
- const socket = new WebSocket(url.toString());
75
+ const url = new URL(this.connectionUrl);
76
+ url.searchParams.set('v', this.gatewayConfig.version.toString());
77
+ url.searchParams.set('encoding', 'json');
78
+ const socket = // @ts-expect-error Deno
79
+ globalThis.Deno !== undefined && Reflect.has(globalThis, 'Deno') ? new WebSocket(url.toString()) : new NodeWebSocket(url.toString());
80
80
  this.socket = socket;
81
81
  // TODO: proper event handling
82
82
  socket.onerror = (event)=>console.log({
83
- error: event
83
+ error: event,
84
+ shardId: this.id
84
85
  });
85
86
  socket.onclose = async (event)=>await this.handleClose(event);
86
87
  socket.onmessage = async (message)=>await this.handleMessage(message);
@@ -114,8 +115,6 @@ export class DiscordenoShard {
114
115
  if (!this.isOpen()) {
115
116
  await this.connect();
116
117
  }
117
- // Wait until an identify is free for this this.
118
- await this.requestIdentify();
119
118
  this.send({
120
119
  op: GatewayOpcodes.Identify,
121
120
  d: {
@@ -133,6 +132,8 @@ export class DiscordenoShard {
133
132
  return await new Promise((resolve)=>{
134
133
  this.resolves.set('READY', ()=>{
135
134
  this.events.identified?.(this);
135
+ // Tells the manager that this shard is ready
136
+ this.shardIsReady();
136
137
  resolve();
137
138
  });
138
139
  // When identifying too fast,
@@ -145,27 +146,27 @@ export class DiscordenoShard {
145
146
  });
146
147
  }
147
148
  /** Check whether the connection to Discord is currently open. */ isOpen() {
148
- return this.socket?.readyState === WebSocket.OPEN;
149
+ return this.socket?.readyState === NodeWebSocket.OPEN;
149
150
  }
150
151
  /** Attempt to resume the previous shards session with the gateway. */ async resume() {
151
- // gateway.debug("GW RESUMING", { shardId });
152
+ logger.debug(`[Gateway] Resuming Shard #${this.id}`);
152
153
  // It has been requested to resume the Shards session.
153
154
  // It's possible that the shard is still connected with Discord's gateway therefore we need to forcefully close it.
154
155
  if (this.isOpen()) {
156
+ logger.debug(`[Gateway] Resuming Shard #${this.id} in isOpen`);
155
157
  this.close(ShardSocketCloseCodes.ResumeClosingOldConnection, 'Reconnecting the shard, closing old connection.');
156
158
  }
157
159
  // Shard has never identified, so we cannot resume.
158
160
  if (!this.sessionId) {
159
- // gateway.debug(
160
- // "GW DEBUG",
161
- // `[Error] Trying to resume a shard (id: ${shardId}) that was not first identified.`,
162
- // );
161
+ logger.debug(`[Shard] Trying to resume a shard #${this.id} that was NOT first identified. (No session id found)`);
163
162
  return await this.identify();
164
- // throw new Error(`[SHARD] Trying to resume a shard (id: ${this.id}) which was never identified`);
165
163
  }
166
164
  this.state = ShardState.Resuming;
165
+ logger.debug(`[Gateway] Resuming Shard #${this.id}, before connecting`);
167
166
  // Before we can resume, we need to create a new connection with Discord's gateway.
168
167
  await this.connect();
168
+ logger.debug(// eslint-disable-next-line @typescript-eslint/restrict-template-expressions
169
+ `[Gateway] Resuming Shard #${this.id}, after connecting. ${this.gatewayConfig.token} | ${this.sessionId} | ${this.previousSequenceNumber}`);
169
170
  this.send({
170
171
  op: GatewayOpcodes.Resume,
171
172
  d: {
@@ -174,6 +175,7 @@ export class DiscordenoShard {
174
175
  seq: this.previousSequenceNumber ?? 0
175
176
  }
176
177
  }, true);
178
+ logger.debug(`[Gateway] Resuming Shard #${this.id} after send resumg`);
177
179
  return await new Promise((resolve)=>{
178
180
  this.resolves.set('RESUMED', ()=>resolve());
179
181
  // If it is attempted to resume with an invalid session id,
@@ -229,6 +231,7 @@ export class DiscordenoShard {
229
231
  case GatewayCloseEventCodes.RateLimited:
230
232
  case GatewayCloseEventCodes.SessionTimedOut:
231
233
  {
234
+ logger.debug(`[Shard] Gateway connection closing requiring re-identify. Code: ${close.code}`);
232
235
  this.state = ShardState.Identifying;
233
236
  this.events.disconnected?.(this);
234
237
  return await this.identify();
@@ -252,6 +255,7 @@ export class DiscordenoShard {
252
255
  case GatewayCloseEventCodes.AlreadyAuthenticated:
253
256
  default:
254
257
  {
258
+ logger.info(`[Shard] closed shard #${this.id}. Resuming...`);
255
259
  this.state = ShardState.Resuming;
256
260
  this.events.disconnected?.(this);
257
261
  return await this.resume();
@@ -285,6 +289,7 @@ export class DiscordenoShard {
285
289
  case GatewayOpcodes.Hello:
286
290
  {
287
291
  const interval = packet.d.heartbeat_interval;
292
+ logger.debug(`[Gateway] Hello on Shard #${this.id}`);
288
293
  this.startHeartbeating(interval);
289
294
  if (this.state !== ShardState.Resuming) {
290
295
  const currentQueue = [
@@ -318,8 +323,8 @@ export class DiscordenoShard {
318
323
  }
319
324
  case GatewayOpcodes.InvalidSession:
320
325
  {
321
- // gateway.debug("GW INVALID_SESSION", { shardId, payload: packet });
322
326
  const resumable = packet.d;
327
+ logger.debug(`[Shard] Received Invalid Session for Shard #${this.id} with resumeable as ${resumable.toString()}`);
323
328
  this.events.invalidSession?.(this, resumable);
324
329
  // We need to wait for a random amount of time between 1 and 5
325
330
  // Reference: https://discord.com/developers/docs/topics/gateway#resuming
@@ -328,7 +333,7 @@ export class DiscordenoShard {
328
333
  this.resolves.delete('INVALID_SESSION');
329
334
  // When resumable is false we need to re-identify
330
335
  if (!resumable) {
331
- await this.identify();
336
+ await this.requestIdentify();
332
337
  break;
333
338
  }
334
339
  // The session is invalid but apparently it is resumable
@@ -336,25 +341,49 @@ export class DiscordenoShard {
336
341
  break;
337
342
  }
338
343
  }
339
- if (packet.t === 'RESUMED') {
340
- // gateway.debug("GW RESUMED", { shardId });
341
- this.state = ShardState.Connected;
342
- this.events.resumed?.(this);
343
- // Continue the requests which have been queued since the shard went offline.
344
- this.offlineSendQueue.map((resolve)=>resolve());
345
- this.resolves.get('RESUMED')?.(packet);
346
- this.resolves.delete('RESUMED');
347
- } else if (packet.t === 'READY') {
348
- // Important for future resumes.
349
- const payload = packet.d;
350
- this.resumeGatewayUrl = payload.resume_gateway_url;
351
- this.sessionId = payload.session_id;
352
- this.state = ShardState.Connected;
353
- // Continue the requests which have been queued since the shard went offline.
354
- // Important when this is a re-identify
355
- this.offlineSendQueue.map((resolve)=>resolve());
356
- this.resolves.get('READY')?.(packet);
357
- this.resolves.delete('READY');
344
+ switch(packet.t){
345
+ case 'RESUMED':
346
+ this.state = ShardState.Connected;
347
+ this.events.resumed?.(this);
348
+ // Continue the requests which have been queued since the shard went offline.
349
+ this.offlineSendQueue.map((resolve)=>resolve());
350
+ this.resolves.get('RESUMED')?.(packet);
351
+ this.resolves.delete('RESUMED');
352
+ break;
353
+ case 'READY':
354
+ {
355
+ // Important for future resumes.
356
+ const payload = packet.d;
357
+ this.resumeGatewayUrl = payload.resume_gateway_url;
358
+ this.sessionId = payload.session_id;
359
+ this.state = ShardState.Connected;
360
+ // Continue the requests which have been queued since the shard went offline.
361
+ // Important when this is a re-identify
362
+ this.offlineSendQueue.map((resolve)=>resolve());
363
+ this.resolves.get('READY')?.(packet);
364
+ this.resolves.delete('READY');
365
+ break;
366
+ }
367
+ case 'GUILD_MEMBERS_CHUNK':
368
+ {
369
+ // If it's not enabled skip checks.
370
+ if (!this.cache.requestMembers.enabled) break;
371
+ const payload = packet.d;
372
+ // If this request has non nonce, skip checks.
373
+ if (!payload.nonce) break;
374
+ const pending = this.cache.requestMembers.pending.get(payload.nonce);
375
+ if (!pending) break;
376
+ // If this is not the final chunk, just save to cache.
377
+ if (payload.chunk_index + 1 < payload.chunk_count) {
378
+ pending.members.push(...payload.members);
379
+ break;
380
+ }
381
+ // Resolve the promise that all requests are done.
382
+ pending.resolve(camelize(pending.members));
383
+ // Delete the cache to clean up once its done.
384
+ this.cache.requestMembers.pending.delete(payload.nonce);
385
+ break;
386
+ }
358
387
  }
359
388
  // Update the sequence number if it is present
360
389
  // `s` can be either `null` or a `number`.
@@ -362,6 +391,9 @@ export class DiscordenoShard {
362
391
  if (packet.s !== null) {
363
392
  this.previousSequenceNumber = packet.s;
364
393
  }
394
+ this.forwardToBot(packet);
395
+ }
396
+ forwardToBot(packet) {
365
397
  // The necessary handling required for the Shards connection has been finished.
366
398
  // Now the event can be safely forwarded.
367
399
  this.events.message?.(this, camelize(packet));
@@ -386,8 +418,12 @@ export class DiscordenoShard {
386
418
  return;
387
419
  }
388
420
  /** 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() {}
421
+ /** This function communicates with the management process, in order to tell it can identify the next shard. */ async shardIsReady() {}
389
422
  /** Start sending heartbeat payloads to Discord in the provided interval. */ startHeartbeating(interval) {
390
- // gateway.debug("GW HEARTBEATING_STARTED", { shardId, interval });
423
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id}`);
424
+ // If old heartbeast exist like after resume, clear the old ones.
425
+ if (this.heart.intervalId) clearInterval(this.heart.intervalId);
426
+ if (this.heart.timeoutId) clearTimeout(this.heart.timeoutId);
391
427
  this.heart.interval = interval;
392
428
  // Only set the shard's state to `Unidentified`
393
429
  // if heartbeating has not been started due to an identify or resume action.
@@ -395,6 +431,7 @@ export class DiscordenoShard {
395
431
  ShardState.Disconnected,
396
432
  ShardState.Offline
397
433
  ].includes(this.state)) {
434
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} a`);
398
435
  this.state = ShardState.Unidentified;
399
436
  }
400
437
  // The first heartbeat needs to be send with a random delay between `0` and `interval`
@@ -403,17 +440,22 @@ export class DiscordenoShard {
403
440
  // Reference: https://discord.com/developers/docs/topics/gateway#heartbeating
404
441
  const jitter = Math.ceil(this.heart.interval * (Math.random() || 0.5));
405
442
  this.heart.timeoutId = setTimeout(()=>{
443
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} b`);
406
444
  if (!this.isOpen()) return;
445
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} c ${this.previousSequenceNumber}`);
407
446
  // Using a direct socket.send call here because heartbeat requests are reserved by us.
408
447
  this.socket?.send(JSON.stringify({
409
448
  op: GatewayOpcodes.Heartbeat,
410
449
  d: this.previousSequenceNumber
411
450
  }));
451
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} d`);
412
452
  this.heart.lastBeat = Date.now();
413
453
  this.heart.acknowledged = false;
414
454
  // After the random heartbeat jitter we can start a normal interval.
415
455
  this.heart.intervalId = setInterval(async ()=>{
456
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} e`);
416
457
  if (!this.isOpen()) return;
458
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} f`);
417
459
  // gateway.debug("GW DEBUG", `Running setInterval in heartbeat file. Shard: ${shardId}`);
418
460
  // gateway.debug("GW HEARTBEATING", { shardId, shard: currentShard });
419
461
  // The Shard did not receive a heartbeat ACK from Discord in time,
@@ -421,15 +463,18 @@ export class DiscordenoShard {
421
463
  // The Shard needs to start a re-identify action accordingly.
422
464
  // Reference: https://discord.com/developers/docs/topics/gateway#heartbeating-example-gateway-heartbeat-ack
423
465
  if (!this.heart.acknowledged) {
466
+ logger.debug(`[Shard] Heartbeat not acknowledged for shard #${this.id}.`);
424
467
  this.close(ShardSocketCloseCodes.ZombiedConnection, 'Zombied connection, did not receive an heartbeat ACK in time.');
425
468
  return await this.identify();
426
469
  }
427
470
  this.heart.acknowledged = false;
471
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} g`);
428
472
  // Using a direct socket.send call here because heartbeat requests are reserved by us.
429
473
  this.socket?.send(JSON.stringify({
430
474
  op: GatewayOpcodes.Heartbeat,
431
475
  d: this.previousSequenceNumber
432
476
  }));
477
+ logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} h`);
433
478
  this.heart.lastBeat = Date.now();
434
479
  this.events.heartbeat?.(this);
435
480
  }, this.heart.interval);
@@ -458,7 +503,7 @@ export class DiscordenoShard {
458
503
  * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}
459
504
  */ async joinVoiceChannel(guildId, channelId, options) {
460
505
  logger.debug(`[Shard] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`);
461
- return await this.send({
506
+ await this.send({
462
507
  op: GatewayOpcodes.VoiceStateUpdate,
463
508
  d: {
464
509
  guild_id: guildId.toString(),
@@ -475,7 +520,7 @@ export class DiscordenoShard {
475
520
  * @returns Promise<void>
476
521
  */ async editBotStatus(data) {
477
522
  logger.debug(`[Shard] editBotStatus data: ${JSON.stringify(data)}`);
478
- return await this.editShardStatus(data);
523
+ await this.editShardStatus(data);
479
524
  }
480
525
  /**
481
526
  * Edits the bot's status on one shard.
@@ -485,7 +530,7 @@ export class DiscordenoShard {
485
530
  * @returns Promise<void>
486
531
  */ async editShardStatus(data) {
487
532
  logger.debug(`[Shard] editShardStatus shardId: ${this.id} -> data: ${JSON.stringify(data)}`);
488
- return await this.send({
533
+ await this.send({
489
534
  op: GatewayOpcodes.PresenceUpdate,
490
535
  d: {
491
536
  since: null,
@@ -527,7 +572,6 @@ export class DiscordenoShard {
527
572
  logger.debug(`[Shard] requestMembers guildId: ${guildId} -> setting user limit based on userIds length: ${options.userIds.length}`);
528
573
  options.limit = options.userIds.length;
529
574
  }
530
- const nonce = `${guildId}-${Date.now()}`;
531
575
  // Gateway does not require caching these requests so directly send and return
532
576
  if (!this.cache.requestMembers?.enabled) {
533
577
  logger.debug(`[Shard] requestMembers guildId: ${guildId} -> skipping cache -> options ${JSON.stringify(options)}`);
@@ -540,14 +584,14 @@ export class DiscordenoShard {
540
584
  limit: options?.limit ?? 0,
541
585
  presences: options?.presences ?? false,
542
586
  user_ids: options?.userIds?.map((id)=>id.toString()),
543
- nonce
587
+ nonce: options?.nonce
544
588
  }
545
589
  });
546
590
  return [];
547
591
  }
548
592
  return await new Promise((resolve)=>{
549
- this.cache.requestMembers?.pending.set(nonce, {
550
- nonce,
593
+ if (options?.nonce) this.cache.requestMembers?.pending.set(options.nonce, {
594
+ nonce: options.nonce,
551
595
  resolve,
552
596
  members: []
553
597
  });
@@ -561,7 +605,7 @@ export class DiscordenoShard {
561
605
  limit: options?.limit ?? 0,
562
606
  presences: options?.presences ?? false,
563
607
  user_ids: options?.userIds?.map((id)=>id.toString()),
564
- nonce
608
+ nonce: options?.nonce
565
609
  }
566
610
  });
567
611
  });
@@ -579,7 +623,7 @@ export class DiscordenoShard {
579
623
  * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}
580
624
  */ async leaveVoiceChannel(guildId) {
581
625
  logger.debug(`[Shard] leaveVoiceChannel guildId: ${guildId} Shard ${this.id}`);
582
- return await this.send({
626
+ await this.send({
583
627
  op: GatewayOpcodes.VoiceStateUpdate,
584
628
  d: {
585
629
  guild_id: guildId.toString(),
package/dist/Shard.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/Shard.ts"],"sourcesContent":["import type {\n AtLeastOne,\n BigString,\n Camelize,\n DiscordGatewayPayload,\n DiscordHello,\n DiscordMember,\n DiscordReady,\n RequestGuildMembers\n} from '@discordeno/types'\nimport { GatewayCloseEventCodes, GatewayIntents, GatewayOpcodes } from '@discordeno/types'\nimport { camelize, Collection, delay, LeakyBucket, logger } from '@discordeno/utils'\nimport { inflateSync } from 'node:zlib'\nimport WebSocket from 'ws'\nimport type { RequestMemberRequest } from './manager.js'\nimport type { BotStatusUpdate, ShardEvents, ShardGatewayConfig, ShardHeart, ShardSocketRequest, StatusUpdate, UpdateVoiceState } from './types.js'\nimport { ShardSocketCloseCodes, ShardState } from './types.js'\n\nexport class DiscordenoShard {\n /** The id of the shard */\n id: number\n /** The connection config details that this shard will used to connect to discord. */\n connection: ShardGatewayConfig\n /** This contains all the heartbeat information */\n heart: ShardHeart\n /** The maximum of requests which can be send to discord per rate limit tick. Typically this value should not be changed. */\n maxRequestsPerRateLimitTick: number = 120\n /** The previous payload sequence number. */\n previousSequenceNumber: number | null = null\n /** In which interval (in milliseconds) the gateway resets it's rate limit. */\n rateLimitResetInterval: number = 60000\n /** Current session id of the shard if present. */\n sessionId?: string\n /** This contains the WebSocket connection to Discord, if currently connected. */\n socket?: WebSocket\n /** Current internal state of the this. */\n state = ShardState.Offline\n /** The url provided by discord to use when resuming a connection for this this. */\n resumeGatewayUrl: string = ''\n /** The shard related event handlers. */\n events: ShardEvents = {}\n /** Cache for pending gateway requests which should have been send while the gateway went offline. */\n offlineSendQueue: Array<(_?: unknown) => void> = []\n /** Resolve internal waiting states. Mapped by SelectedEvents => ResolveFunction */\n resolves = new Map<'READY' | 'RESUMED' | 'INVALID_SESSION', (payload: DiscordGatewayPayload) => void>()\n /** Shard bucket. Only access this if you know what you are doing. Bucket for handling shard request rate limits. */\n bucket: LeakyBucket\n\n /** This managers cache related settings. */\n cache = {\n requestMembers: {\n /**\n * Whether or not request member requests should be cached.\n * @default false\n */\n enabled: false,\n /** The pending requests. */\n pending: new Collection<string, RequestMemberRequest>(),\n },\n }\n\n constructor(options: ShardCreateOptions) {\n this.id = options.id\n this.connection = options.connection\n this.events = options.events\n\n this.heart = {\n acknowledged: false,\n interval: 45000,\n }\n\n if (options.requestIdentify) this.requestIdentify = options.requestIdentify\n\n this.bucket = new LeakyBucket({\n max: this.calculateSafeRequests(),\n refillAmount: this.calculateSafeRequests(),\n refillInterval: 60000,\n })\n }\n\n /** The gateway configuration which is used to connect to Discord. */\n get gatewayConfig(): ShardGatewayConfig {\n return this.connection\n }\n\n /** 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. */\n calculateSafeRequests(): number {\n // * 2 adds extra safety layer for discords OP 1 requests that we need to respond to\n const safeRequests = this.maxRequestsPerRateLimitTick - Math.ceil(this.rateLimitResetInterval / this.heart.interval) * 2\n\n return safeRequests < 0 ? 0 : safeRequests\n }\n\n async checkOffline(highPriority: boolean): Promise<void> {\n if (!this.isOpen()) {\n await new Promise((resolve) => {\n // Higher priority requests get added at the beginning of the array.\n if (highPriority) this.offlineSendQueue.unshift(resolve)\n else this.offlineSendQueue.push(resolve)\n })\n }\n }\n\n /** Close the socket connection to discord if present. */\n close(code: number, reason: string): void {\n if (this.socket?.readyState !== WebSocket.OPEN) return\n\n return this.socket?.close(code, reason)\n }\n\n /** Connect the shard with the gateway and start heartbeating. This will not identify the shard to the gateway. */\n async connect(): Promise<DiscordenoShard> {\n // Only set the shard to `Connecting` state,\n // if the connection request does not come from an identify or resume action.\n if (![ShardState.Identifying, ShardState.Resuming].includes(this.state)) {\n this.state = ShardState.Connecting\n }\n this.events.connecting?.(this)\n\n let url = new URL(this.gatewayConfig.url)\n // If not connecting to a proxy but directly to discord need to handle resuming\n if (url.origin === 'wss://gateway.discord.gg') {\n if (this.state === ShardState.Resuming) {\n url = new URL(this.resumeGatewayUrl)\n }\n url.searchParams.set('v', this.gatewayConfig.version.toString())\n url.searchParams.set('encoding', 'json')\n }\n\n const socket = new WebSocket(url.toString())\n\n this.socket = socket\n\n // TODO: proper event handling\n socket.onerror = (event) => console.log({ error: event })\n\n socket.onclose = async (event) => await this.handleClose(event)\n\n socket.onmessage = async (message) => await this.handleMessage(message)\n\n return await new Promise((resolve) => {\n socket.onopen = () => {\n // Only set the shard to `Unidentified` state,\n // if the connection request does not come from an identify or resume action.\n if (![ShardState.Identifying, ShardState.Resuming].includes(this.state)) {\n this.state = ShardState.Unidentified\n }\n this.events.connected?.(this)\n\n resolve(this)\n }\n })\n }\n\n /** Identify the shard to the gateway. If not connected, this will also connect the shard to the gateway. */\n async identify(): Promise<void> {\n // A new identify has been requested even though there is already a connection open.\n // Therefore we need to close the old connection and heartbeating before creating a new one.\n if (this.isOpen()) {\n logger.debug(`CLOSING EXISTING SHARD: #${this.id}`)\n this.close(ShardSocketCloseCodes.ReIdentifying, 'Re-identifying closure of old connection.')\n }\n\n this.state = ShardState.Identifying\n this.events.identifying?.(this)\n\n // It is possible that the shard is in Heartbeating state but not identified,\n // so check whether there is already a gateway connection existing.\n // If not we need to create one before we identify.\n if (!this.isOpen()) {\n await this.connect()\n }\n\n // Wait until an identify is free for this this.\n await this.requestIdentify()\n\n this.send(\n {\n op: GatewayOpcodes.Identify,\n d: {\n token: `Bot ${this.gatewayConfig.token}`,\n compress: this.gatewayConfig.compress,\n properties: this.gatewayConfig.properties,\n intents: this.gatewayConfig.intents,\n shard: [this.id, this.gatewayConfig.totalShards],\n presence: await this.makePresence?.(),\n },\n },\n true,\n )\n\n return await new Promise((resolve) => {\n this.resolves.set('READY', () => {\n this.events.identified?.(this)\n resolve()\n })\n // When identifying too fast,\n // Discord sends an invalid session payload.\n // This can safely be ignored though and the shard starts a new identify action.\n this.resolves.set('INVALID_SESSION', () => {\n this.resolves.delete('READY')\n resolve()\n })\n })\n }\n\n /** Check whether the connection to Discord is currently open. */\n isOpen(): boolean {\n return this.socket?.readyState === WebSocket.OPEN\n }\n\n /** Attempt to resume the previous shards session with the gateway. */\n async resume(): Promise<void> {\n // gateway.debug(\"GW RESUMING\", { shardId });\n // It has been requested to resume the Shards session.\n // It's possible that the shard is still connected with Discord's gateway therefore we need to forcefully close it.\n if (this.isOpen()) {\n this.close(ShardSocketCloseCodes.ResumeClosingOldConnection, 'Reconnecting the shard, closing old connection.')\n }\n\n // Shard has never identified, so we cannot resume.\n if (!this.sessionId) {\n // gateway.debug(\n // \"GW DEBUG\",\n // `[Error] Trying to resume a shard (id: ${shardId}) that was not first identified.`,\n // );\n\n return await this.identify()\n\n // throw new Error(`[SHARD] Trying to resume a shard (id: ${this.id}) which was never identified`);\n }\n\n this.state = ShardState.Resuming\n\n // Before we can resume, we need to create a new connection with Discord's gateway.\n await this.connect()\n\n this.send(\n {\n op: GatewayOpcodes.Resume,\n d: {\n token: `Bot ${this.gatewayConfig.token}`,\n session_id: this.sessionId,\n seq: this.previousSequenceNumber ?? 0,\n },\n },\n true,\n )\n\n return await new Promise((resolve) => {\n this.resolves.set('RESUMED', () => resolve())\n // If it is attempted to resume with an invalid session id,\n // Discord sends an invalid session payload\n // Not erroring here since it is easy that this happens, also it would be not catchable\n this.resolves.set('INVALID_SESSION', () => {\n this.resolves.delete('RESUMED')\n resolve()\n })\n })\n }\n\n /** Send a message to Discord.\n * @param {boolean} [highPriority=false] - Whether this message should be send asap.\n */\n async send(message: ShardSocketRequest, highPriority = false): Promise<void> {\n // Before acquiring a token from the bucket, check whether the shard is currently offline or not.\n // Else bucket and token wait time just get wasted.\n await this.checkOffline(highPriority)\n\n await this.bucket.acquire(highPriority)\n\n // It's possible, that the shard went offline after a token has been acquired from the bucket.\n await this.checkOffline(highPriority)\n\n this.socket?.send(JSON.stringify(message))\n }\n\n /** Shutdown the this. Forcefully disconnect the shard from Discord. The shard may not attempt to reconnect with Discord. */\n async shutdown(): Promise<void> {\n this.close(ShardSocketCloseCodes.Shutdown, 'Shard shutting down.')\n this.state = ShardState.Offline\n }\n\n /** Handle a gateway connection close. */\n async handleClose(close: WebSocket.CloseEvent): Promise<void> {\n // gateway.debug(\"GW CLOSED\", { shardId, payload: event });\n\n this.stopHeartbeating()\n\n switch (close.code) {\n case ShardSocketCloseCodes.TestingFinished: {\n this.state = ShardState.Offline\n this.events.disconnected?.(this)\n\n return\n }\n // On these codes a manual start will be done.\n case ShardSocketCloseCodes.Shutdown:\n case ShardSocketCloseCodes.ReIdentifying:\n case ShardSocketCloseCodes.Resharded:\n case ShardSocketCloseCodes.ResumeClosingOldConnection:\n case ShardSocketCloseCodes.ZombiedConnection: {\n this.state = ShardState.Disconnected\n this.events.disconnected?.(this)\n\n // gateway.debug(\"GW CLOSED_RECONNECT\", { shardId, payload: event });\n return\n }\n // Gateway connection closes which require a new identify.\n case GatewayCloseEventCodes.UnknownOpcode:\n case GatewayCloseEventCodes.NotAuthenticated:\n case GatewayCloseEventCodes.InvalidSeq:\n case GatewayCloseEventCodes.RateLimited:\n case GatewayCloseEventCodes.SessionTimedOut: {\n this.state = ShardState.Identifying\n this.events.disconnected?.(this)\n\n return await this.identify()\n }\n // When these codes are received something went really wrong.\n // On those we cannot start a reconnect attempt.\n case GatewayCloseEventCodes.AuthenticationFailed:\n case GatewayCloseEventCodes.InvalidShard:\n case GatewayCloseEventCodes.ShardingRequired:\n case GatewayCloseEventCodes.InvalidApiVersion:\n case GatewayCloseEventCodes.InvalidIntents:\n case GatewayCloseEventCodes.DisallowedIntents: {\n this.state = ShardState.Offline\n this.events.disconnected?.(this)\n\n throw new Error(close.reason || 'Discord gave no reason! GG! You broke Discord!')\n }\n // Gateway connection closes on which a resume is allowed.\n case GatewayCloseEventCodes.UnknownError:\n case GatewayCloseEventCodes.DecodeError:\n case GatewayCloseEventCodes.AlreadyAuthenticated:\n default: {\n this.state = ShardState.Resuming\n this.events.disconnected?.(this)\n\n return await this.resume()\n }\n }\n }\n\n /** Handles a incoming gateway packet. */\n async handleDiscordPacket(packet: DiscordGatewayPayload): Promise<void> {\n // Edge case start: https://github.com/discordeno/discordeno/issues/2311\n this.heart.lastAck = Date.now()\n // Manually calculating the round trip time for users who need it.\n if (this.heart.lastBeat && !this.heart.acknowledged) {\n this.heart.rtt = this.heart.lastAck - this.heart.lastBeat\n }\n this.heart.acknowledged = true\n // Edge case end!\n\n switch (packet.op) {\n case GatewayOpcodes.Heartbeat: {\n // TODO: can this actually happen\n if (!this.isOpen()) return\n\n this.heart.lastBeat = Date.now()\n // Discord randomly sends this requiring an immediate heartbeat back.\n // Using a direct socket.send call here because heartbeat requests are reserved by us.\n this.socket?.send(\n JSON.stringify({\n op: GatewayOpcodes.Heartbeat,\n d: this.previousSequenceNumber,\n }),\n )\n this.events.heartbeat?.(this)\n\n break\n }\n case GatewayOpcodes.Hello: {\n const interval = (packet.d as DiscordHello).heartbeat_interval\n\n this.startHeartbeating(interval)\n\n if (this.state !== ShardState.Resuming) {\n const currentQueue = [...this.bucket.queue];\n // HELLO has been send on a non resume action.\n // This means that the shard starts a new session,\n // therefore the rate limit interval has been reset too.\n this.bucket = new LeakyBucket({\n max: this.calculateSafeRequests(),\n refillInterval: 60000,\n refillAmount: this.calculateSafeRequests(),\n })\n\n // Queue should not be lost on a re-identify.\n this.bucket.queue.unshift(...currentQueue);\n }\n\n this.events.hello?.(this)\n\n break\n }\n case GatewayOpcodes.HeartbeatACK: {\n this.events.heartbeatAck?.(this)\n\n break\n }\n case GatewayOpcodes.Reconnect: {\n // gateway.debug(\"GW RECONNECT\", { shardId });\n\n this.events.requestedReconnect?.(this)\n\n await this.resume()\n\n break\n }\n case GatewayOpcodes.InvalidSession: {\n // gateway.debug(\"GW INVALID_SESSION\", { shardId, payload: packet });\n const resumable = packet.d as boolean\n\n this.events.invalidSession?.(this, resumable)\n\n // We need to wait for a random amount of time between 1 and 5\n // Reference: https://discord.com/developers/docs/topics/gateway#resuming\n await delay(Math.floor((Math.random() * 4 + 1) * 1000))\n\n this.resolves.get('INVALID_SESSION')?.(packet)\n this.resolves.delete('INVALID_SESSION')\n\n // When resumable is false we need to re-identify\n if (!resumable) {\n await this.identify()\n\n break\n }\n\n // The session is invalid but apparently it is resumable\n await this.resume()\n\n break\n }\n }\n\n if (packet.t === 'RESUMED') {\n // gateway.debug(\"GW RESUMED\", { shardId });\n\n this.state = ShardState.Connected\n this.events.resumed?.(this)\n\n // Continue the requests which have been queued since the shard went offline.\n this.offlineSendQueue.map((resolve) => resolve())\n\n this.resolves.get('RESUMED')?.(packet)\n this.resolves.delete('RESUMED')\n } else if (packet.t === 'READY') {\n // Important for future resumes.\n\n const payload = packet.d as DiscordReady\n\n this.resumeGatewayUrl = payload.resume_gateway_url\n\n this.sessionId = payload.session_id\n this.state = ShardState.Connected\n\n // Continue the requests which have been queued since the shard went offline.\n // Important when this is a re-identify\n this.offlineSendQueue.map((resolve) => resolve())\n\n this.resolves.get('READY')?.(packet)\n this.resolves.delete('READY')\n }\n\n // Update the sequence number if it is present\n // `s` can be either `null` or a `number`.\n // In order to prevent update misses when `s` is `0` we check against null.\n if (packet.s !== null) {\n this.previousSequenceNumber = packet.s\n }\n\n // The necessary handling required for the Shards connection has been finished.\n // Now the event can be safely forwarded.\n this.events.message?.(this, camelize(packet))\n }\n\n /** Handle an incoming gateway message. */\n async handleMessage(message: WebSocket.MessageEvent): Promise<void> {\n let preProcessMessage = message.data\n\n // If message compression is enabled,\n // Discord might send zlib compressed payloads.\n if (this.gatewayConfig.compress && preProcessMessage instanceof Blob) {\n preProcessMessage = inflateSync(await preProcessMessage.arrayBuffer()).toString()\n }\n\n // Safeguard incase decompression failed to make a string.\n if (typeof preProcessMessage !== 'string') return\n\n return await this.handleDiscordPacket(JSON.parse(preProcessMessage) as DiscordGatewayPayload)\n }\n\n /**\n * Override in order to make the shards presence.\n * async in case devs create the presence based on eg. database values.\n * Passing the shard's id there to make it easier for the dev to use this function.\n */\n async makePresence(): Promise<BotStatusUpdate | undefined> {\n // eslint-disable-next-line no-useless-return\n return\n }\n\n /** 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. */\n async requestIdentify(): Promise<void> {}\n\n /** Start sending heartbeat payloads to Discord in the provided interval. */\n startHeartbeating(interval: number): void {\n // gateway.debug(\"GW HEARTBEATING_STARTED\", { shardId, interval });\n\n this.heart.interval = interval\n\n // Only set the shard's state to `Unidentified`\n // if heartbeating has not been started due to an identify or resume action.\n if ([ShardState.Disconnected, ShardState.Offline].includes(this.state)) {\n this.state = ShardState.Unidentified\n }\n\n // The first heartbeat needs to be send with a random delay between `0` and `interval`\n // Using a `setTimeout(_, jitter)` here to accomplish that.\n // `Math.random()` can be `0` so we use `0.5` if this happens\n // Reference: https://discord.com/developers/docs/topics/gateway#heartbeating\n const jitter = Math.ceil(this.heart.interval * (Math.random() || 0.5))\n this.heart.timeoutId = setTimeout(() => {\n if (!this.isOpen()) return\n\n // Using a direct socket.send call here because heartbeat requests are reserved by us.\n this.socket?.send(\n JSON.stringify({\n op: GatewayOpcodes.Heartbeat,\n d: this.previousSequenceNumber,\n }),\n )\n\n this.heart.lastBeat = Date.now()\n this.heart.acknowledged = false\n\n // After the random heartbeat jitter we can start a normal interval.\n this.heart.intervalId = setInterval(async () => {\n if (!this.isOpen()) return\n // gateway.debug(\"GW DEBUG\", `Running setInterval in heartbeat file. Shard: ${shardId}`);\n\n // gateway.debug(\"GW HEARTBEATING\", { shardId, shard: currentShard });\n\n // The Shard did not receive a heartbeat ACK from Discord in time,\n // therefore we have to assume that the connection has failed or got \"zombied\".\n // The Shard needs to start a re-identify action accordingly.\n // Reference: https://discord.com/developers/docs/topics/gateway#heartbeating-example-gateway-heartbeat-ack\n if (!this.heart.acknowledged) {\n this.close(ShardSocketCloseCodes.ZombiedConnection, 'Zombied connection, did not receive an heartbeat ACK in time.')\n\n return await this.identify()\n }\n\n this.heart.acknowledged = false\n\n // Using a direct socket.send call here because heartbeat requests are reserved by us.\n this.socket?.send(\n JSON.stringify({\n op: GatewayOpcodes.Heartbeat,\n d: this.previousSequenceNumber,\n }),\n )\n\n this.heart.lastBeat = Date.now()\n\n this.events.heartbeat?.(this)\n }, this.heart.interval)\n }, jitter)\n }\n\n /** Stop the heartbeating process with discord. */\n stopHeartbeating(): void {\n // Clear the regular heartbeat interval.\n clearInterval(this.heart.intervalId)\n // It's possible that the Shard got closed before the first jittered heartbeat.\n // To go safe we should clear the related timeout too.\n clearTimeout(this.heart.timeoutId)\n }\n\n /**\n * Connects the bot user to a voice or stage channel.\n *\n * This function sends the _Update Voice State_ gateway command over the gateway behind the scenes.\n *\n * @param guildId - The ID of the guild the voice channel to leave is in.\n * @param channelId - The ID of the channel you want to join.\n *\n * @remarks\n * Requires the `CONNECT` permission.\n *\n * Fires a _Voice State Update_ gateway event.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}\n */\n async joinVoiceChannel(\n guildId: BigString,\n channelId: BigString,\n options?: AtLeastOne<Omit<UpdateVoiceState, 'guildId' | 'channelId'>>,\n ): Promise<void> {\n logger.debug(`[Shard] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`)\n return await this.send({\n op: GatewayOpcodes.VoiceStateUpdate,\n d: {\n guild_id: guildId.toString(),\n channel_id: channelId.toString(),\n self_mute: Boolean(options?.selfMute),\n self_deaf: options?.selfDeaf ?? true,\n },\n })\n }\n\n /**\n * Edits the bot status in all shards that this gateway manages.\n *\n * @param data The status data to set the bots status to.\n * @returns Promise<void>\n */\n async editBotStatus(data: StatusUpdate): Promise<void> {\n logger.debug(`[Shard] editBotStatus data: ${JSON.stringify(data)}`)\n return await this.editShardStatus(data)\n }\n\n /**\n * Edits the bot's status on one shard.\n *\n * @param shardId The shard id to edit the status for.\n * @param data The status data to set the bots status to.\n * @returns Promise<void>\n */\n async editShardStatus(data: StatusUpdate): Promise<void> {\n logger.debug(`[Shard] editShardStatus shardId: ${this.id} -> data: ${JSON.stringify(data)}`)\n return await this.send({\n op: GatewayOpcodes.PresenceUpdate,\n d: {\n since: null,\n afk: false,\n activities: data.activities,\n status: data.status,\n },\n })\n }\n\n /**\n * Fetches the list of members for a guild over the gateway.\n *\n * @param guildId - The ID of the guild to get the list of members for.\n * @param options - The parameters for the fetching of the members.\n *\n * @remarks\n * If requesting the entire member list:\n * - Requires the `GUILD_MEMBERS` intent.\n *\n * If requesting presences ({@link RequestGuildMembers.presences | presences} set to `true`):\n * - Requires the `GUILD_PRESENCES` intent.\n *\n * If requesting a prefix ({@link RequestGuildMembers.query | query} non-`undefined`):\n * - Returns a maximum of 100 members.\n *\n * If requesting a users by ID ({@link RequestGuildMembers.userIds | userIds} non-`undefined`):\n * - Returns a maximum of 100 members.\n *\n * Fires a _Guild Members Chunk_ gateway event for every 1000 members fetched.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#request-guild-members}\n */\n async requestMembers(guildId: BigString, options?: Omit<RequestGuildMembers, 'guildId'>): Promise<Camelize<DiscordMember[]>> {\n // You can request 1 member without the intent\n // Check if intents is not 0 as proxy ws won't set intents in other instances\n if (this.connection.intents && (!options?.limit || options.limit > 1) && !(this.connection.intents & GatewayIntents.GuildMembers)) {\n throw new Error('MISSING_INTENT_GUILD_MEMBERS')\n }\n\n if (options?.userIds?.length) {\n logger.debug(`[Shard] requestMembers guildId: ${guildId} -> setting user limit based on userIds length: ${options.userIds.length}`)\n options.limit = options.userIds.length\n }\n\n const nonce = `${guildId}-${Date.now()}`\n\n // Gateway does not require caching these requests so directly send and return\n if (!this.cache.requestMembers?.enabled) {\n logger.debug(`[Shard] requestMembers guildId: ${guildId} -> skipping cache -> options ${JSON.stringify(options)}`)\n await this.send({\n op: GatewayOpcodes.RequestGuildMembers,\n d: {\n guild_id: guildId.toString(),\n // If a query is provided use it, OR if a limit is NOT provided use \"\"\n query: options?.query ?? (options?.limit ? undefined : ''),\n limit: options?.limit ?? 0,\n presences: options?.presences ?? false,\n user_ids: options?.userIds?.map((id) => id.toString()),\n nonce,\n },\n })\n return []\n }\n\n return await new Promise((resolve) => {\n this.cache.requestMembers?.pending.set(nonce, { nonce, resolve, members: [] })\n\n logger.debug(`[Shard] requestMembers guildId: ${guildId} -> requesting members -> data: ${JSON.stringify(options)}`)\n this.send({\n op: GatewayOpcodes.RequestGuildMembers,\n d: {\n guild_id: guildId.toString(),\n // If a query is provided use it, OR if a limit is NOT provided use \"\"\n query: options?.query ?? (options?.limit ? undefined : ''),\n limit: options?.limit ?? 0,\n presences: options?.presences ?? false,\n user_ids: options?.userIds?.map((id) => id.toString()),\n nonce,\n },\n })\n })\n }\n\n /**\n * Leaves the voice channel the bot user is currently in.\n *\n * This function sends the _Update Voice State_ gateway command over the gateway behind the scenes.\n *\n * @param guildId - The ID of the guild the voice channel to leave is in.\n *\n * @remarks\n * Fires a _Voice State Update_ gateway event.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}\n */\n async leaveVoiceChannel(guildId: BigString): Promise<void> {\n logger.debug(`[Shard] leaveVoiceChannel guildId: ${guildId} Shard ${this.id}`)\n return await this.send({\n op: GatewayOpcodes.VoiceStateUpdate,\n d: {\n guild_id: guildId.toString(),\n channel_id: null,\n self_mute: false,\n self_deaf: false,\n },\n })\n }\n}\n\nexport interface ShardCreateOptions {\n /** The shard id */\n id: number\n /** The connection details */\n connection: ShardGatewayConfig\n /** The event handlers for events on the shard. */\n events: ShardEvents\n /** The handler to request a space to make an identify request. */\n requestIdentify?: () => Promise<void>\n}\n\nexport default DiscordenoShard\n"],"names":["GatewayCloseEventCodes","GatewayIntents","GatewayOpcodes","camelize","Collection","delay","LeakyBucket","logger","inflateSync","WebSocket","ShardSocketCloseCodes","ShardState","DiscordenoShard","maxRequestsPerRateLimitTick","previousSequenceNumber","rateLimitResetInterval","state","Offline","resumeGatewayUrl","events","offlineSendQueue","resolves","Map","cache","requestMembers","enabled","pending","constructor","options","id","connection","heart","acknowledged","interval","requestIdentify","bucket","max","calculateSafeRequests","refillAmount","refillInterval","gatewayConfig","safeRequests","Math","ceil","checkOffline","highPriority","isOpen","Promise","resolve","unshift","push","close","code","reason","socket","readyState","OPEN","connect","Identifying","Resuming","includes","Connecting","connecting","url","URL","origin","searchParams","set","version","toString","onerror","event","console","log","error","onclose","handleClose","onmessage","message","handleMessage","onopen","Unidentified","connected","identify","debug","ReIdentifying","identifying","send","op","Identify","d","token","compress","properties","intents","shard","totalShards","presence","makePresence","identified","delete","resume","ResumeClosingOldConnection","sessionId","Resume","session_id","seq","acquire","JSON","stringify","shutdown","Shutdown","stopHeartbeating","TestingFinished","disconnected","Resharded","ZombiedConnection","Disconnected","UnknownOpcode","NotAuthenticated","InvalidSeq","RateLimited","SessionTimedOut","AuthenticationFailed","InvalidShard","ShardingRequired","InvalidApiVersion","InvalidIntents","DisallowedIntents","Error","UnknownError","DecodeError","AlreadyAuthenticated","handleDiscordPacket","packet","lastAck","Date","now","lastBeat","rtt","Heartbeat","heartbeat","Hello","heartbeat_interval","startHeartbeating","currentQueue","queue","hello","HeartbeatACK","heartbeatAck","Reconnect","requestedReconnect","InvalidSession","resumable","invalidSession","floor","random","get","t","Connected","resumed","map","payload","resume_gateway_url","s","preProcessMessage","data","Blob","arrayBuffer","parse","jitter","timeoutId","setTimeout","intervalId","setInterval","clearInterval","clearTimeout","joinVoiceChannel","guildId","channelId","VoiceStateUpdate","guild_id","channel_id","self_mute","Boolean","selfMute","self_deaf","selfDeaf","editBotStatus","editShardStatus","PresenceUpdate","since","afk","activities","status","limit","GuildMembers","userIds","length","nonce","RequestGuildMembers","query","undefined","presences","user_ids","members","leaveVoiceChannel"],"mappings":"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"}
1
+ {"version":3,"sources":["../src/Shard.ts"],"sourcesContent":["/* eslint-disable @typescript-eslint/no-confusing-void-expression */\nimport type {\n AtLeastOne,\n BigString,\n Camelize,\n DiscordGatewayPayload,\n DiscordGuildMembersChunk,\n DiscordHello,\n DiscordMember,\n DiscordReady,\n RequestGuildMembers,\n} from '@discordeno/types'\nimport { GatewayCloseEventCodes, GatewayIntents, GatewayOpcodes } from '@discordeno/types'\nimport { Collection, LeakyBucket, camelize, delay, logger } from '@discordeno/utils'\nimport { inflateSync } from 'node:zlib'\nimport NodeWebSocket from 'ws'\nimport type { RequestMemberRequest } from './manager.js'\nimport type { BotStatusUpdate, ShardEvents, ShardGatewayConfig, ShardHeart, ShardSocketRequest, StatusUpdate, UpdateVoiceState } from './types.js'\nimport { ShardSocketCloseCodes, ShardState } from './types.js'\n\ndeclare let WebSocket: any\n\nexport class DiscordenoShard {\n /** The id of the shard */\n id: number\n /** The connection config details that this shard will used to connect to discord. */\n connection: ShardGatewayConfig\n /** This contains all the heartbeat information */\n heart: ShardHeart\n /** The maximum of requests which can be send to discord per rate limit tick. Typically this value should not be changed. */\n maxRequestsPerRateLimitTick: number = 120\n /** The previous payload sequence number. */\n previousSequenceNumber: number | null = null\n /** In which interval (in milliseconds) the gateway resets it's rate limit. */\n rateLimitResetInterval: number = 60000\n /** Current session id of the shard if present. */\n sessionId?: string\n /** This contains the WebSocket connection to Discord, if currently connected. */\n socket?: NodeWebSocket\n /** Current internal state of the this. */\n state = ShardState.Offline\n /** The url provided by discord to use when resuming a connection for this this. */\n resumeGatewayUrl: string = ''\n /** The shard related event handlers. */\n events: ShardEvents = {}\n /** Cache for pending gateway requests which should have been send while the gateway went offline. */\n offlineSendQueue: Array<(_?: unknown) => void> = []\n /** Resolve internal waiting states. Mapped by SelectedEvents => ResolveFunction */\n resolves = new Map<'READY' | 'RESUMED' | 'INVALID_SESSION', (payload: DiscordGatewayPayload) => void>()\n /** Shard bucket. Only access this if you know what you are doing. Bucket for handling shard request rate limits. */\n bucket: LeakyBucket\n\n /** This managers cache related settings. */\n cache = {\n requestMembers: {\n /**\n * Whether or not request member requests should be cached.\n * @default false\n */\n enabled: false,\n /** The pending requests. */\n pending: new Collection<string, RequestMemberRequest>(),\n },\n }\n\n constructor(options: ShardCreateOptions) {\n this.id = options.id\n this.connection = options.connection\n this.events = options.events\n\n this.heart = {\n acknowledged: false,\n interval: 45000,\n }\n\n if (options.requestIdentify) this.requestIdentify = options.requestIdentify\n if (options.shardIsReady) this.shardIsReady = options.shardIsReady\n\n this.bucket = new LeakyBucket({\n max: this.calculateSafeRequests(),\n refillAmount: this.calculateSafeRequests(),\n refillInterval: 60000,\n })\n }\n\n /** The gateway configuration which is used to connect to Discord. */\n get gatewayConfig(): ShardGatewayConfig {\n return this.connection\n }\n\n /** The url to connect to. Intially this is the discord gateway url, and then is switched to resume gateway url once a READY is received. */\n get connectionUrl(): string {\n // Use || and not ?? here. ?? will cause a bug.\n return this.resumeGatewayUrl || this.gatewayConfig.url\n }\n\n /** 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. */\n calculateSafeRequests(): number {\n // * 2 adds extra safety layer for discords OP 1 requests that we need to respond to\n const safeRequests = this.maxRequestsPerRateLimitTick - Math.ceil(this.rateLimitResetInterval / this.heart.interval) * 2\n\n return safeRequests < 0 ? 0 : safeRequests\n }\n\n async checkOffline(highPriority: boolean): Promise<void> {\n if (!this.isOpen()) {\n await new Promise((resolve) => {\n // Higher priority requests get added at the beginning of the array.\n if (highPriority) this.offlineSendQueue.unshift(resolve)\n else this.offlineSendQueue.push(resolve)\n })\n }\n }\n\n /** Close the socket connection to discord if present. */\n close(code: number, reason: string): void {\n if (this.socket?.readyState !== NodeWebSocket.OPEN) return\n\n this.socket?.close(code, reason)\n }\n\n /** Connect the shard with the gateway and start heartbeating. This will not identify the shard to the gateway. */\n async connect(): Promise<DiscordenoShard> {\n // Only set the shard to `Connecting` state,\n // if the connection request does not come from an identify or resume action.\n if (![ShardState.Identifying, ShardState.Resuming].includes(this.state)) {\n this.state = ShardState.Connecting\n }\n this.events.connecting?.(this)\n\n const url = new URL(this.connectionUrl)\n url.searchParams.set('v', this.gatewayConfig.version.toString())\n url.searchParams.set('encoding', 'json')\n\n const socket: NodeWebSocket =\n // @ts-expect-error Deno\n globalThis.Deno !== undefined && Reflect.has(globalThis, 'Deno') ? new WebSocket(url.toString()) : new NodeWebSocket(url.toString())\n this.socket = socket\n\n // TODO: proper event handling\n socket.onerror = (event: NodeWebSocket.ErrorEvent) => console.log({ error: event, shardId: this.id })\n socket.onclose = async (event: NodeWebSocket.CloseEvent) => await this.handleClose(event)\n socket.onmessage = async (message: NodeWebSocket.MessageEvent) => await this.handleMessage(message)\n\n return await new Promise((resolve) => {\n socket.onopen = () => {\n // Only set the shard to `Unidentified` state,\n // if the connection request does not come from an identify or resume action.\n if (![ShardState.Identifying, ShardState.Resuming].includes(this.state)) {\n this.state = ShardState.Unidentified\n }\n this.events.connected?.(this)\n\n resolve(this)\n }\n })\n }\n\n /** Identify the shard to the gateway. If not connected, this will also connect the shard to the gateway. */\n async identify(): Promise<void> {\n // A new identify has been requested even though there is already a connection open.\n // Therefore we need to close the old connection and heartbeating before creating a new one.\n if (this.isOpen()) {\n logger.debug(`CLOSING EXISTING SHARD: #${this.id}`)\n this.close(ShardSocketCloseCodes.ReIdentifying, 'Re-identifying closure of old connection.')\n }\n\n this.state = ShardState.Identifying\n this.events.identifying?.(this)\n\n // It is possible that the shard is in Heartbeating state but not identified,\n // so check whether there is already a gateway connection existing.\n // If not we need to create one before we identify.\n if (!this.isOpen()) {\n await this.connect()\n }\n\n this.send(\n {\n op: GatewayOpcodes.Identify,\n d: {\n token: `Bot ${this.gatewayConfig.token}`,\n compress: this.gatewayConfig.compress,\n properties: this.gatewayConfig.properties,\n intents: this.gatewayConfig.intents,\n shard: [this.id, this.gatewayConfig.totalShards],\n presence: await this.makePresence?.(),\n },\n },\n true,\n )\n\n return await new Promise((resolve) => {\n this.resolves.set('READY', () => {\n this.events.identified?.(this)\n // Tells the manager that this shard is ready\n this.shardIsReady()\n resolve()\n })\n // When identifying too fast,\n // Discord sends an invalid session payload.\n // This can safely be ignored though and the shard starts a new identify action.\n this.resolves.set('INVALID_SESSION', () => {\n this.resolves.delete('READY')\n resolve()\n })\n })\n }\n\n /** Check whether the connection to Discord is currently open. */\n isOpen(): boolean {\n return this.socket?.readyState === NodeWebSocket.OPEN\n }\n\n /** Attempt to resume the previous shards session with the gateway. */\n async resume(): Promise<void> {\n logger.debug(`[Gateway] Resuming Shard #${this.id}`)\n // It has been requested to resume the Shards session.\n // It's possible that the shard is still connected with Discord's gateway therefore we need to forcefully close it.\n if (this.isOpen()) {\n logger.debug(`[Gateway] Resuming Shard #${this.id} in isOpen`)\n this.close(ShardSocketCloseCodes.ResumeClosingOldConnection, 'Reconnecting the shard, closing old connection.')\n }\n\n // Shard has never identified, so we cannot resume.\n if (!this.sessionId) {\n logger.debug(`[Shard] Trying to resume a shard #${this.id} that was NOT first identified. (No session id found)`)\n\n return await this.identify()\n }\n\n this.state = ShardState.Resuming\n\n logger.debug(`[Gateway] Resuming Shard #${this.id}, before connecting`)\n // Before we can resume, we need to create a new connection with Discord's gateway.\n await this.connect()\n logger.debug(\n // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n `[Gateway] Resuming Shard #${this.id}, after connecting. ${this.gatewayConfig.token} | ${this.sessionId} | ${this.previousSequenceNumber}`,\n )\n\n this.send(\n {\n op: GatewayOpcodes.Resume,\n d: {\n token: `Bot ${this.gatewayConfig.token}`,\n session_id: this.sessionId,\n seq: this.previousSequenceNumber ?? 0,\n },\n },\n true,\n )\n logger.debug(`[Gateway] Resuming Shard #${this.id} after send resumg`)\n\n return await new Promise((resolve) => {\n this.resolves.set('RESUMED', () => resolve())\n // If it is attempted to resume with an invalid session id,\n // Discord sends an invalid session payload\n // Not erroring here since it is easy that this happens, also it would be not catchable\n this.resolves.set('INVALID_SESSION', () => {\n this.resolves.delete('RESUMED')\n resolve()\n })\n })\n }\n\n /** Send a message to Discord.\n * @param {boolean} [highPriority=false] - Whether this message should be send asap.\n */\n async send(message: ShardSocketRequest, highPriority = false): Promise<void> {\n // Before acquiring a token from the bucket, check whether the shard is currently offline or not.\n // Else bucket and token wait time just get wasted.\n await this.checkOffline(highPriority)\n\n await this.bucket.acquire(highPriority)\n\n // It's possible, that the shard went offline after a token has been acquired from the bucket.\n await this.checkOffline(highPriority)\n\n this.socket?.send(JSON.stringify(message))\n }\n\n /** Shutdown the this. Forcefully disconnect the shard from Discord. The shard may not attempt to reconnect with Discord. */\n async shutdown(): Promise<void> {\n this.close(ShardSocketCloseCodes.Shutdown, 'Shard shutting down.')\n this.state = ShardState.Offline\n }\n\n /** Handle a gateway connection close. */\n async handleClose(close: NodeWebSocket.CloseEvent): Promise<void> {\n // gateway.debug(\"GW CLOSED\", { shardId, payload: event });\n\n this.stopHeartbeating()\n\n switch (close.code) {\n case ShardSocketCloseCodes.TestingFinished: {\n this.state = ShardState.Offline\n this.events.disconnected?.(this)\n\n return\n }\n // On these codes a manual start will be done.\n case ShardSocketCloseCodes.Shutdown:\n case ShardSocketCloseCodes.ReIdentifying:\n case ShardSocketCloseCodes.Resharded:\n case ShardSocketCloseCodes.ResumeClosingOldConnection:\n case ShardSocketCloseCodes.ZombiedConnection: {\n this.state = ShardState.Disconnected\n this.events.disconnected?.(this)\n\n // gateway.debug(\"GW CLOSED_RECONNECT\", { shardId, payload: event });\n return\n }\n // Gateway connection closes which require a new identify.\n case GatewayCloseEventCodes.UnknownOpcode:\n case GatewayCloseEventCodes.NotAuthenticated:\n case GatewayCloseEventCodes.InvalidSeq:\n case GatewayCloseEventCodes.RateLimited:\n case GatewayCloseEventCodes.SessionTimedOut: {\n logger.debug(`[Shard] Gateway connection closing requiring re-identify. Code: ${close.code}`)\n this.state = ShardState.Identifying\n this.events.disconnected?.(this)\n\n return await this.identify()\n }\n // When these codes are received something went really wrong.\n // On those we cannot start a reconnect attempt.\n case GatewayCloseEventCodes.AuthenticationFailed:\n case GatewayCloseEventCodes.InvalidShard:\n case GatewayCloseEventCodes.ShardingRequired:\n case GatewayCloseEventCodes.InvalidApiVersion:\n case GatewayCloseEventCodes.InvalidIntents:\n case GatewayCloseEventCodes.DisallowedIntents: {\n this.state = ShardState.Offline\n this.events.disconnected?.(this)\n\n throw new Error(close.reason || 'Discord gave no reason! GG! You broke Discord!')\n }\n // Gateway connection closes on which a resume is allowed.\n case GatewayCloseEventCodes.UnknownError:\n case GatewayCloseEventCodes.DecodeError:\n case GatewayCloseEventCodes.AlreadyAuthenticated:\n default: {\n logger.info(`[Shard] closed shard #${this.id}. Resuming...`)\n this.state = ShardState.Resuming\n this.events.disconnected?.(this)\n\n return await this.resume()\n }\n }\n }\n\n /** Handles a incoming gateway packet. */\n async handleDiscordPacket(packet: DiscordGatewayPayload): Promise<void> {\n // Edge case start: https://github.com/discordeno/discordeno/issues/2311\n this.heart.lastAck = Date.now()\n // Manually calculating the round trip time for users who need it.\n if (this.heart.lastBeat && !this.heart.acknowledged) {\n this.heart.rtt = this.heart.lastAck - this.heart.lastBeat\n }\n this.heart.acknowledged = true\n // Edge case end!\n\n switch (packet.op) {\n case GatewayOpcodes.Heartbeat: {\n // TODO: can this actually happen\n if (!this.isOpen()) return\n\n this.heart.lastBeat = Date.now()\n // Discord randomly sends this requiring an immediate heartbeat back.\n // Using a direct socket.send call here because heartbeat requests are reserved by us.\n this.socket?.send(\n JSON.stringify({\n op: GatewayOpcodes.Heartbeat,\n d: this.previousSequenceNumber,\n }),\n )\n this.events.heartbeat?.(this)\n\n break\n }\n case GatewayOpcodes.Hello: {\n const interval = (packet.d as DiscordHello).heartbeat_interval\n logger.debug(`[Gateway] Hello on Shard #${this.id}`)\n this.startHeartbeating(interval)\n\n if (this.state !== ShardState.Resuming) {\n const currentQueue = [...this.bucket.queue]\n // HELLO has been send on a non resume action.\n // This means that the shard starts a new session,\n // therefore the rate limit interval has been reset too.\n this.bucket = new LeakyBucket({\n max: this.calculateSafeRequests(),\n refillInterval: 60000,\n refillAmount: this.calculateSafeRequests(),\n })\n\n // Queue should not be lost on a re-identify.\n this.bucket.queue.unshift(...currentQueue)\n }\n\n this.events.hello?.(this)\n\n break\n }\n case GatewayOpcodes.HeartbeatACK: {\n this.events.heartbeatAck?.(this)\n\n break\n }\n case GatewayOpcodes.Reconnect: {\n // gateway.debug(\"GW RECONNECT\", { shardId });\n\n this.events.requestedReconnect?.(this)\n\n await this.resume()\n\n break\n }\n case GatewayOpcodes.InvalidSession: {\n const resumable = packet.d as boolean\n logger.debug(`[Shard] Received Invalid Session for Shard #${this.id} with resumeable as ${resumable.toString()}`)\n\n this.events.invalidSession?.(this, resumable)\n\n // We need to wait for a random amount of time between 1 and 5\n // Reference: https://discord.com/developers/docs/topics/gateway#resuming\n await delay(Math.floor((Math.random() * 4 + 1) * 1000))\n\n this.resolves.get('INVALID_SESSION')?.(packet)\n this.resolves.delete('INVALID_SESSION')\n\n // When resumable is false we need to re-identify\n if (!resumable) {\n await this.requestIdentify()\n\n break\n }\n\n // The session is invalid but apparently it is resumable\n await this.resume()\n\n break\n }\n }\n\n switch (packet.t) {\n case 'RESUMED':\n this.state = ShardState.Connected\n this.events.resumed?.(this)\n\n // Continue the requests which have been queued since the shard went offline.\n this.offlineSendQueue.map((resolve) => resolve())\n\n this.resolves.get('RESUMED')?.(packet)\n this.resolves.delete('RESUMED')\n break\n case 'READY': {\n // Important for future resumes.\n const payload = packet.d as DiscordReady\n\n this.resumeGatewayUrl = payload.resume_gateway_url\n\n this.sessionId = payload.session_id\n this.state = ShardState.Connected\n\n // Continue the requests which have been queued since the shard went offline.\n // Important when this is a re-identify\n this.offlineSendQueue.map((resolve) => resolve())\n\n this.resolves.get('READY')?.(packet)\n this.resolves.delete('READY')\n break\n }\n case 'GUILD_MEMBERS_CHUNK': {\n // If it's not enabled skip checks.\n if (!this.cache.requestMembers.enabled) break\n\n const payload = packet.d as DiscordGuildMembersChunk\n // If this request has non nonce, skip checks.\n if (!payload.nonce) break\n\n const pending = this.cache.requestMembers.pending.get(payload.nonce)\n if (!pending) break\n\n // If this is not the final chunk, just save to cache.\n if (payload.chunk_index + 1 < payload.chunk_count) {\n pending.members.push(...payload.members)\n break;\n }\n\n // Resolve the promise that all requests are done.\n pending.resolve(camelize(pending.members))\n // Delete the cache to clean up once its done.\n this.cache.requestMembers.pending.delete(payload.nonce)\n break\n }\n }\n\n // Update the sequence number if it is present\n // `s` can be either `null` or a `number`.\n // In order to prevent update misses when `s` is `0` we check against null.\n if (packet.s !== null) {\n this.previousSequenceNumber = packet.s\n }\n\n this.forwardToBot(packet)\n }\n\n forwardToBot(packet: DiscordGatewayPayload): void {\n // The necessary handling required for the Shards connection has been finished.\n // Now the event can be safely forwarded.\n this.events.message?.(this, camelize(packet))\n }\n\n /** Handle an incoming gateway message. */\n async handleMessage(message: NodeWebSocket.MessageEvent): Promise<void> {\n let preProcessMessage = message.data\n\n // If message compression is enabled,\n // Discord might send zlib compressed payloads.\n if (this.gatewayConfig.compress && preProcessMessage instanceof Blob) {\n preProcessMessage = inflateSync(await preProcessMessage.arrayBuffer()).toString()\n }\n\n // Safeguard incase decompression failed to make a string.\n if (typeof preProcessMessage !== 'string') return\n\n return await this.handleDiscordPacket(JSON.parse(preProcessMessage) as DiscordGatewayPayload)\n }\n\n /**\n * Override in order to make the shards presence.\n * async in case devs create the presence based on eg. database values.\n * Passing the shard's id there to make it easier for the dev to use this function.\n */\n async makePresence(): Promise<BotStatusUpdate | undefined> {\n // eslint-disable-next-line no-useless-return\n return\n }\n\n /** 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. */\n async requestIdentify(): Promise<void> {}\n\n /** This function communicates with the management process, in order to tell it can identify the next shard. */\n async shardIsReady(): Promise<void> {}\n\n /** Start sending heartbeat payloads to Discord in the provided interval. */\n startHeartbeating(interval: number): void {\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id}`)\n // If old heartbeast exist like after resume, clear the old ones.\n if (this.heart.intervalId) clearInterval(this.heart.intervalId)\n if (this.heart.timeoutId) clearTimeout(this.heart.timeoutId)\n\n this.heart.interval = interval\n\n // Only set the shard's state to `Unidentified`\n // if heartbeating has not been started due to an identify or resume action.\n if ([ShardState.Disconnected, ShardState.Offline].includes(this.state)) {\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} a`)\n this.state = ShardState.Unidentified\n }\n\n // The first heartbeat needs to be send with a random delay between `0` and `interval`\n // Using a `setTimeout(_, jitter)` here to accomplish that.\n // `Math.random()` can be `0` so we use `0.5` if this happens\n // Reference: https://discord.com/developers/docs/topics/gateway#heartbeating\n const jitter = Math.ceil(this.heart.interval * (Math.random() || 0.5))\n this.heart.timeoutId = setTimeout(() => {\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} b`)\n if (!this.isOpen()) return\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} c ${this.previousSequenceNumber!}`)\n\n // Using a direct socket.send call here because heartbeat requests are reserved by us.\n this.socket?.send(\n JSON.stringify({\n op: GatewayOpcodes.Heartbeat,\n d: this.previousSequenceNumber,\n }),\n )\n\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} d`)\n this.heart.lastBeat = Date.now()\n this.heart.acknowledged = false\n\n // After the random heartbeat jitter we can start a normal interval.\n this.heart.intervalId = setInterval(async () => {\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} e`)\n if (!this.isOpen()) return\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} f`)\n // gateway.debug(\"GW DEBUG\", `Running setInterval in heartbeat file. Shard: ${shardId}`);\n\n // gateway.debug(\"GW HEARTBEATING\", { shardId, shard: currentShard });\n\n // The Shard did not receive a heartbeat ACK from Discord in time,\n // therefore we have to assume that the connection has failed or got \"zombied\".\n // The Shard needs to start a re-identify action accordingly.\n // Reference: https://discord.com/developers/docs/topics/gateway#heartbeating-example-gateway-heartbeat-ack\n if (!this.heart.acknowledged) {\n logger.debug(`[Shard] Heartbeat not acknowledged for shard #${this.id}.`)\n this.close(ShardSocketCloseCodes.ZombiedConnection, 'Zombied connection, did not receive an heartbeat ACK in time.')\n\n return await this.identify()\n }\n\n this.heart.acknowledged = false\n\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} g`)\n // Using a direct socket.send call here because heartbeat requests are reserved by us.\n this.socket?.send(\n JSON.stringify({\n op: GatewayOpcodes.Heartbeat,\n d: this.previousSequenceNumber,\n }),\n )\n logger.debug(`[Gateway] Start Heartbeating Shard #${this.id} h`)\n\n this.heart.lastBeat = Date.now()\n\n this.events.heartbeat?.(this)\n }, this.heart.interval)\n }, jitter)\n }\n\n /** Stop the heartbeating process with discord. */\n stopHeartbeating(): void {\n // Clear the regular heartbeat interval.\n clearInterval(this.heart.intervalId)\n // It's possible that the Shard got closed before the first jittered heartbeat.\n // To go safe we should clear the related timeout too.\n clearTimeout(this.heart.timeoutId)\n }\n\n /**\n * Connects the bot user to a voice or stage channel.\n *\n * This function sends the _Update Voice State_ gateway command over the gateway behind the scenes.\n *\n * @param guildId - The ID of the guild the voice channel to leave is in.\n * @param channelId - The ID of the channel you want to join.\n *\n * @remarks\n * Requires the `CONNECT` permission.\n *\n * Fires a _Voice State Update_ gateway event.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}\n */\n async joinVoiceChannel(\n guildId: BigString,\n channelId: BigString,\n options?: AtLeastOne<Omit<UpdateVoiceState, 'guildId' | 'channelId'>>,\n ): Promise<void> {\n logger.debug(`[Shard] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`)\n await this.send({\n op: GatewayOpcodes.VoiceStateUpdate,\n d: {\n guild_id: guildId.toString(),\n channel_id: channelId.toString(),\n self_mute: Boolean(options?.selfMute),\n self_deaf: options?.selfDeaf ?? true,\n },\n })\n }\n\n /**\n * Edits the bot status in all shards that this gateway manages.\n *\n * @param data The status data to set the bots status to.\n * @returns Promise<void>\n */\n async editBotStatus(data: StatusUpdate): Promise<void> {\n logger.debug(`[Shard] editBotStatus data: ${JSON.stringify(data)}`)\n await this.editShardStatus(data)\n }\n\n /**\n * Edits the bot's status on one shard.\n *\n * @param shardId The shard id to edit the status for.\n * @param data The status data to set the bots status to.\n * @returns Promise<void>\n */\n async editShardStatus(data: StatusUpdate): Promise<void> {\n logger.debug(`[Shard] editShardStatus shardId: ${this.id} -> data: ${JSON.stringify(data)}`)\n await this.send({\n op: GatewayOpcodes.PresenceUpdate,\n d: {\n since: null,\n afk: false,\n activities: data.activities,\n status: data.status,\n },\n })\n }\n\n /**\n * Fetches the list of members for a guild over the gateway.\n *\n * @param guildId - The ID of the guild to get the list of members for.\n * @param options - The parameters for the fetching of the members.\n *\n * @remarks\n * If requesting the entire member list:\n * - Requires the `GUILD_MEMBERS` intent.\n *\n * If requesting presences ({@link RequestGuildMembers.presences | presences} set to `true`):\n * - Requires the `GUILD_PRESENCES` intent.\n *\n * If requesting a prefix ({@link RequestGuildMembers.query | query} non-`undefined`):\n * - Returns a maximum of 100 members.\n *\n * If requesting a users by ID ({@link RequestGuildMembers.userIds | userIds} non-`undefined`):\n * - Returns a maximum of 100 members.\n *\n * Fires a _Guild Members Chunk_ gateway event for every 1000 members fetched.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#request-guild-members}\n */\n async requestMembers(guildId: BigString, options?: Omit<RequestGuildMembers, 'guildId'>): Promise<Camelize<DiscordMember[]>> {\n // You can request 1 member without the intent\n // Check if intents is not 0 as proxy ws won't set intents in other instances\n if (this.connection.intents && (!options?.limit || options.limit > 1) && !(this.connection.intents & GatewayIntents.GuildMembers)) {\n throw new Error('MISSING_INTENT_GUILD_MEMBERS')\n }\n\n if (options?.userIds?.length) {\n logger.debug(`[Shard] requestMembers guildId: ${guildId} -> setting user limit based on userIds length: ${options.userIds.length}`)\n options.limit = options.userIds.length\n }\n\n // Gateway does not require caching these requests so directly send and return\n if (!this.cache.requestMembers?.enabled) {\n logger.debug(`[Shard] requestMembers guildId: ${guildId} -> skipping cache -> options ${JSON.stringify(options)}`)\n await this.send({\n op: GatewayOpcodes.RequestGuildMembers,\n d: {\n guild_id: guildId.toString(),\n // If a query is provided use it, OR if a limit is NOT provided use \"\"\n query: options?.query ?? (options?.limit ? undefined : ''),\n limit: options?.limit ?? 0,\n presences: options?.presences ?? false,\n user_ids: options?.userIds?.map((id) => id.toString()),\n nonce: options?.nonce,\n },\n })\n return []\n }\n\n return await new Promise((resolve) => {\n if (options?.nonce) this.cache.requestMembers?.pending.set(options.nonce, { nonce: options.nonce, resolve, members: [] })\n\n logger.debug(`[Shard] requestMembers guildId: ${guildId} -> requesting members -> data: ${JSON.stringify(options)}`)\n this.send({\n op: GatewayOpcodes.RequestGuildMembers,\n d: {\n guild_id: guildId.toString(),\n // If a query is provided use it, OR if a limit is NOT provided use \"\"\n query: options?.query ?? (options?.limit ? undefined : ''),\n limit: options?.limit ?? 0,\n presences: options?.presences ?? false,\n user_ids: options?.userIds?.map((id) => id.toString()),\n nonce: options?.nonce,\n },\n })\n })\n }\n\n /**\n * Leaves the voice channel the bot user is currently in.\n *\n * This function sends the _Update Voice State_ gateway command over the gateway behind the scenes.\n *\n * @param guildId - The ID of the guild the voice channel to leave is in.\n *\n * @remarks\n * Fires a _Voice State Update_ gateway event.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}\n */\n async leaveVoiceChannel(guildId: BigString): Promise<void> {\n logger.debug(`[Shard] leaveVoiceChannel guildId: ${guildId} Shard ${this.id}`)\n await this.send({\n op: GatewayOpcodes.VoiceStateUpdate,\n d: {\n guild_id: guildId.toString(),\n channel_id: null,\n self_mute: false,\n self_deaf: false,\n },\n })\n }\n}\n\nexport interface ShardCreateOptions {\n /** The shard id */\n id: number\n /** The connection details */\n connection: ShardGatewayConfig\n /** The event handlers for events on the shard. */\n events: ShardEvents\n /** The handler to request a space to make an identify request. */\n requestIdentify?: () => Promise<void>\n /** The handler to alert the gateway manager that this shard has received a READY event. */\n shardIsReady?: () => Promise<void>\n}\n\nexport default DiscordenoShard\n"],"names":["GatewayCloseEventCodes","GatewayIntents","GatewayOpcodes","Collection","LeakyBucket","camelize","delay","logger","inflateSync","NodeWebSocket","ShardSocketCloseCodes","ShardState","DiscordenoShard","constructor","options","maxRequestsPerRateLimitTick","previousSequenceNumber","rateLimitResetInterval","state","Offline","resumeGatewayUrl","events","offlineSendQueue","resolves","Map","cache","requestMembers","enabled","pending","id","connection","heart","acknowledged","interval","requestIdentify","shardIsReady","bucket","max","calculateSafeRequests","refillAmount","refillInterval","gatewayConfig","connectionUrl","url","safeRequests","Math","ceil","checkOffline","highPriority","isOpen","Promise","resolve","unshift","push","close","code","reason","socket","readyState","OPEN","connect","Identifying","Resuming","includes","Connecting","connecting","URL","searchParams","set","version","toString","globalThis","Deno","undefined","Reflect","has","WebSocket","onerror","event","console","log","error","shardId","onclose","handleClose","onmessage","message","handleMessage","onopen","Unidentified","connected","identify","debug","ReIdentifying","identifying","send","op","Identify","d","token","compress","properties","intents","shard","totalShards","presence","makePresence","identified","delete","resume","ResumeClosingOldConnection","sessionId","Resume","session_id","seq","acquire","JSON","stringify","shutdown","Shutdown","stopHeartbeating","TestingFinished","disconnected","Resharded","ZombiedConnection","Disconnected","UnknownOpcode","NotAuthenticated","InvalidSeq","RateLimited","SessionTimedOut","AuthenticationFailed","InvalidShard","ShardingRequired","InvalidApiVersion","InvalidIntents","DisallowedIntents","Error","UnknownError","DecodeError","AlreadyAuthenticated","info","handleDiscordPacket","packet","lastAck","Date","now","lastBeat","rtt","Heartbeat","heartbeat","Hello","heartbeat_interval","startHeartbeating","currentQueue","queue","hello","HeartbeatACK","heartbeatAck","Reconnect","requestedReconnect","InvalidSession","resumable","invalidSession","floor","random","get","t","Connected","resumed","map","payload","resume_gateway_url","nonce","chunk_index","chunk_count","members","s","forwardToBot","preProcessMessage","data","Blob","arrayBuffer","parse","intervalId","clearInterval","timeoutId","clearTimeout","jitter","setTimeout","setInterval","joinVoiceChannel","guildId","channelId","VoiceStateUpdate","guild_id","channel_id","self_mute","Boolean","selfMute","self_deaf","selfDeaf","editBotStatus","editShardStatus","PresenceUpdate","since","afk","activities","status","limit","GuildMembers","userIds","length","RequestGuildMembers","query","presences","user_ids","leaveVoiceChannel"],"mappings":"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"}
package/dist/manager.d.ts CHANGED
@@ -1,5 +1,5 @@
1
- import type { AtLeastOne, BigString, Camelize, DiscordGetGatewayBot, DiscordMember, RequestGuildMembers } from '@discordeno/types';
2
- import { Collection, LeakyBucket } from '@discordeno/utils';
1
+ import type { AtLeastOne, BigString, Camelize, DiscordGetGatewayBot, DiscordMember, DiscordMemberWithUser, RequestGuildMembers } from '@discordeno/types';
2
+ import { Collection } from '@discordeno/utils';
3
3
  import Shard from './Shard.js';
4
4
  import type { ShardEvents, StatusUpdate, UpdateVoiceState } from './types.js';
5
5
  export declare function createGatewayManager(options: CreateGatewayManagerOptions): GatewayManager;
@@ -19,6 +19,11 @@ export interface CreateGatewayManagerOptions {
19
19
  * @default 5300
20
20
  */
21
21
  spawnShardDelay?: number;
22
+ /**
23
+ * Whether to send the discord packets in snake case form.
24
+ * @default false
25
+ */
26
+ preferSnakeCase?: boolean;
22
27
  /**
23
28
  * Total amount of shards your bot uses. Useful for zero-downtime updates or resharding.
24
29
  * @default 1
@@ -98,7 +103,8 @@ export interface GatewayManager extends Required<CreateGatewayManagerOptions> {
98
103
  id: number;
99
104
  queue: number[];
100
105
  }>;
101
- leak: LeakyBucket;
106
+ /** Requests to identify shards are made based on whether it is available to be made. */
107
+ identifyRequests: Array<(value: void | PromiseLike<void>) => void>;
102
108
  }>;
103
109
  /** The shards that are created. */
104
110
  shards: Map<number, Shard>;
@@ -118,7 +124,7 @@ export interface GatewayManager extends Required<CreateGatewayManagerOptions> {
118
124
  identify: (shardId: number) => Promise<void>;
119
125
  /** Kill a shard. Close a shards connection to Discord's gateway (if any) and remove it from the manager. */
120
126
  kill: (shardId: number) => Promise<void>;
121
- /** This function communicates with the parent manager, in order to know whether this manager is allowed to identify a new shard. */
127
+ /** This function makes sure that the bucket is allowed to make the next identify request. */
122
128
  requestIdentify: (shardId: number) => Promise<void>;
123
129
  /** Calculates the number of shards based on the guild id and total shards. */
124
130
  calculateShardId: (guildId: BigString, totalShards?: number) => number;
@@ -197,6 +203,6 @@ export interface RequestMemberRequest {
197
203
  /** The resolver handler to run when all members arrive. */
198
204
  resolve: (value: Camelize<DiscordMember[]> | PromiseLike<Camelize<DiscordMember[]>>) => void;
199
205
  /** The members that have already arrived for this request. */
200
- members: Camelize<DiscordMember[]>;
206
+ members: DiscordMemberWithUser[];
201
207
  }
202
208
  //# sourceMappingURL=manager.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"manager.d.ts","sourceRoot":"","sources":["../src/manager.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"manager.d.ts","sourceRoot":"","sources":["../src/manager.ts"],"names":[],"mappings":"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"}
package/dist/manager.js CHANGED
@@ -1,18 +1,16 @@
1
- import { Collection, delay, LeakyBucket, logger } from '@discordeno/utils';
1
+ /* eslint-disable @typescript-eslint/no-confusing-void-expression */ import { Collection, delay, logger } from '@discordeno/utils';
2
2
  import Shard from './Shard.js';
3
3
  export function createGatewayManager(options) {
4
- if (!options.connection) {
5
- options.connection = {
6
- url: 'wss://gateway.discord.gg',
7
- shards: 1,
8
- sessionStartLimit: {
9
- maxConcurrency: 1,
10
- remaining: 1000,
11
- total: 1000,
12
- resetAfter: 1000 * 60 * 60 * 24
13
- }
14
- };
15
- }
4
+ const connectionOptions = options.connection ?? {
5
+ url: 'wss://gateway.discord.gg',
6
+ shards: 1,
7
+ sessionStartLimit: {
8
+ maxConcurrency: 1,
9
+ remaining: 1000,
10
+ total: 1000,
11
+ resetAfter: 1000 * 60 * 60 * 24
12
+ }
13
+ };
16
14
  const gateway = {
17
15
  events: options.events,
18
16
  compress: options.compress ?? false,
@@ -23,15 +21,16 @@ export function createGatewayManager(options) {
23
21
  device: options.properties?.device ?? 'Discordeno'
24
22
  },
25
23
  token: options.token,
26
- url: options.url ?? options.connection.url ?? 'wss://gateway.discord.gg',
24
+ url: options.url ?? connectionOptions.url ?? 'wss://gateway.discord.gg',
27
25
  version: options.version ?? 10,
28
- connection: options.connection,
29
- totalShards: options.totalShards ?? options.connection.shards ?? 1,
30
- lastShardId: options.lastShardId ?? 0,
26
+ connection: connectionOptions,
27
+ totalShards: options.totalShards ?? connectionOptions.shards ?? 1,
28
+ lastShardId: options.lastShardId ?? (options.totalShards ? options.totalShards - 1 : connectionOptions ? connectionOptions.shards - 1 : 0),
31
29
  firstShardId: options.firstShardId ?? 0,
32
30
  totalWorkers: options.totalWorkers ?? 4,
33
31
  shardsPerWorker: options.shardsPerWorker ?? 25,
34
32
  spawnShardDelay: options.spawnShardDelay ?? 5300,
33
+ preferSnakeCase: options.preferSnakeCase ?? false,
35
34
  shards: new Map(),
36
35
  buckets: new Map(),
37
36
  cache: {
@@ -52,13 +51,7 @@ export function createGatewayManager(options) {
52
51
  (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency)) * gateway.connection.sessionStartLimit.maxConcurrency;
53
52
  },
54
53
  calculateWorkerId (shardId) {
55
- // Ignore decimal numbers.
56
- let workerId = Math.floor(shardId / gateway.shardsPerWorker);
57
- // If the workerId overflows the maximal allowed workers we by default just use to last worker.
58
- if (workerId >= gateway.totalWorkers) {
59
- // The Id of the last available worker is total -1
60
- workerId = gateway.totalWorkers - 1;
61
- }
54
+ const workerId = Math.min(shardId % gateway.shardsPerWorker, gateway.totalWorkers - 1);
62
55
  logger.debug(`[Gateway] Calculating workerId: Shard: ${shardId} -> Worker: ${workerId} -> Per Worker: ${gateway.shardsPerWorker} -> Total: ${gateway.totalWorkers}`);
63
56
  return workerId;
64
57
  },
@@ -67,9 +60,7 @@ export function createGatewayManager(options) {
67
60
  logger.debug(`[Gateway] Preparing buckets for concurrency: ${i}`);
68
61
  gateway.buckets.set(i, {
69
62
  workers: [],
70
- leak: new LeakyBucket({
71
- refillInterval: gateway.spawnShardDelay
72
- })
63
+ identifyRequests: []
73
64
  });
74
65
  }
75
66
  // ORGANIZE ALL SHARDS INTO THEIR OWN BUCKETS
@@ -99,6 +90,12 @@ export function createGatewayManager(options) {
99
90
  });
100
91
  }
101
92
  }
93
+ for (const bucket of gateway.buckets.values()){
94
+ for (const worker of bucket.workers.values()){
95
+ // eslint-disable-next-line @typescript-eslint/require-array-sort-compare
96
+ worker.queue = worker.queue.sort((a, b)=>a - b);
97
+ }
98
+ }
102
99
  },
103
100
  async spawnShards () {
104
101
  // PREPARES ALL SHARDS IN SPECIFIC BUCKETS
@@ -120,7 +117,7 @@ export function createGatewayManager(options) {
120
117
  },
121
118
  async tellWorkerToIdentify (workerId, shardId, bucketId) {
122
119
  logger.debug(`[Gateway] tell worker to identify (${workerId}, ${shardId}, ${bucketId})`);
123
- return await gateway.identify(shardId);
120
+ await gateway.identify(shardId);
124
121
  },
125
122
  async identify (shardId) {
126
123
  let shard = this.shards.get(shardId);
@@ -139,15 +136,31 @@ export function createGatewayManager(options) {
139
136
  },
140
137
  events: options.events,
141
138
  requestIdentify: async ()=>{
142
- await gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency).leak.acquire();
139
+ await gateway.identify(shardId);
140
+ },
141
+ shardIsReady: async ()=>{
142
+ logger.debug(`[Shard] Shard #${shardId} is ready`);
143
+ await delay(gateway.spawnShardDelay);
144
+ logger.debug(`[Shard] Resolving shard identify request`);
145
+ gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency).identifyRequests.shift()?.();
143
146
  }
144
147
  });
148
+ if (this.preferSnakeCase) {
149
+ shard.forwardToBot = async (payload)=>{
150
+ options.events.message?.(shard, payload);
151
+ };
152
+ }
145
153
  this.shards.set(shardId, shard);
146
154
  }
147
- logger.debug(`[Gateway] requesting to identify shard #(${shardId}) from bucket.`);
148
- await gateway.requestIdentify(shard.id);
149
- logger.debug(`[Gateway] Identify request successful for shard #(${shardId}).`);
150
- return await shard.identify();
155
+ const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency);
156
+ if (!bucket) return;
157
+ return await new Promise((resolve)=>{
158
+ // Mark that we are making an identify request so another is not made.
159
+ bucket.identifyRequests.push(resolve);
160
+ logger.debug(`[Gateway] identifying shard #(${shardId}).`);
161
+ // This will trigger identify and when READY is received it will resolve the above request.
162
+ shard?.identify();
163
+ });
151
164
  },
152
165
  async kill (shardId) {
153
166
  const shard = this.shards.get(shardId);
@@ -156,11 +169,15 @@ export function createGatewayManager(options) {
156
169
  }
157
170
  logger.debug(`[Gateway] kill shard (${shardId})`);
158
171
  this.shards.delete(shardId);
159
- return await shard.shutdown();
172
+ await shard.shutdown();
160
173
  },
161
174
  async requestIdentify (shardId) {
162
175
  logger.debug(`[Gateway] requesting identify`);
163
- await gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency).leak.acquire();
176
+ // const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)
177
+ // if (!bucket) return
178
+ // return await new Promise((resolve) => {
179
+ // bucket.identifyRequests.push(resolve)
180
+ // })
164
181
  },
165
182
  // Helpers methods below this
166
183
  calculateShardId (guildId, totalShards) {
@@ -197,7 +214,7 @@ export function createGatewayManager(options) {
197
214
  throw new Error(`Shard (id: ${shardId}) not found.`);
198
215
  }
199
216
  logger.debug(`[Gateway] editShardStatus shardId: ${shardId} -> data: ${JSON.stringify(data)}`);
200
- return await shard.editShardStatus(data);
217
+ await shard.editShardStatus(data);
201
218
  },
202
219
  async requestMembers (guildId, options) {
203
220
  const shardId = gateway.calculateShardId(guildId);
@@ -215,7 +232,7 @@ export function createGatewayManager(options) {
215
232
  throw new Error(`Shard (id: ${shardId} not found`);
216
233
  }
217
234
  logger.debug(`[Gateway] leaveVoiceChannel guildId: ${guildId} Shard ${shardId}`);
218
- return await shard.leaveVoiceChannel(guildId);
235
+ await shard.leaveVoiceChannel(guildId);
219
236
  }
220
237
  };
221
238
  return gateway;
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/manager.ts"],"sourcesContent":["import type { AtLeastOne, BigString, Camelize, DiscordGetGatewayBot, DiscordMember, RequestGuildMembers } from '@discordeno/types'\nimport { Collection, delay, LeakyBucket, logger } from '@discordeno/utils'\nimport Shard from './Shard.js'\nimport type { ShardEvents, StatusUpdate, UpdateVoiceState } from './types.js'\n\nexport function createGatewayManager(options: CreateGatewayManagerOptions): GatewayManager {\n if (!options.connection) {\n options.connection = {\n url: 'wss://gateway.discord.gg',\n shards: 1,\n sessionStartLimit: {\n maxConcurrency: 1,\n remaining: 1000,\n total: 1000,\n resetAfter: 1000 * 60 * 60 * 24,\n },\n }\n }\n\n const gateway: GatewayManager = {\n events: options.events,\n compress: options.compress ?? false,\n intents: options.intents ?? 0,\n properties: {\n os: options.properties?.os ?? process.platform,\n browser: options.properties?.browser ?? 'Discordeno',\n device: options.properties?.device ?? 'Discordeno',\n },\n token: options.token,\n url: options.url ?? options.connection.url ?? 'wss://gateway.discord.gg',\n version: options.version ?? 10,\n connection: options.connection,\n totalShards: options.totalShards ?? options.connection.shards ?? 1,\n lastShardId: options.lastShardId ?? 0,\n firstShardId: options.firstShardId ?? 0,\n totalWorkers: options.totalWorkers ?? 4,\n shardsPerWorker: options.shardsPerWorker ?? 25,\n spawnShardDelay: options.spawnShardDelay ?? 5300,\n shards: new Map(),\n buckets: new Map(),\n cache: {\n requestMembers: {\n enabled: options.cache?.requestMembers?.enabled ?? false,\n pending: new Collection(),\n },\n },\n\n calculateTotalShards() {\n // Bots under 100k servers do not have access to total shards.\n if (gateway.totalShards < 100) {\n logger.debug(`[Gateway] Calculating total shards: ${gateway.totalShards}`)\n return gateway.totalShards\n }\n\n logger.debug(`[Gateway] Calculating total shards`, gateway.totalShards, gateway.connection.sessionStartLimit.maxConcurrency)\n // Calculate a multiple of `maxConcurrency` which can be used to connect to the gateway.\n return (\n Math.ceil(\n gateway.totalShards /\n // If `maxConcurrency` is 1 we can safely use 16.\n (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency),\n ) * gateway.connection.sessionStartLimit.maxConcurrency\n )\n },\n calculateWorkerId(shardId) {\n // Ignore decimal numbers.\n let workerId = Math.floor(shardId / gateway.shardsPerWorker)\n // If the workerId overflows the maximal allowed workers we by default just use to last worker.\n if (workerId >= gateway.totalWorkers) {\n // The Id of the last available worker is total -1\n workerId = gateway.totalWorkers - 1\n }\n\n logger.debug(\n `[Gateway] Calculating workerId: Shard: ${shardId} -> Worker: ${workerId} -> Per Worker: ${gateway.shardsPerWorker} -> Total: ${gateway.totalWorkers}`,\n )\n\n return workerId\n },\n prepareBuckets() {\n for (let i = 0; i < gateway.connection.sessionStartLimit.maxConcurrency; ++i) {\n logger.debug(`[Gateway] Preparing buckets for concurrency: ${i}`)\n gateway.buckets.set(i, {\n workers: [],\n leak: new LeakyBucket({\n refillInterval: gateway.spawnShardDelay,\n }),\n })\n }\n\n // ORGANIZE ALL SHARDS INTO THEIR OWN BUCKETS\n for (let shardId = gateway.firstShardId; shardId <= gateway.lastShardId; ++shardId) {\n logger.debug(`[Gateway] Preparing buckets for shard: ${shardId}`)\n if (shardId >= gateway.totalShards) {\n throw new Error(`Shard (id: ${shardId}) is bigger or equal to the used amount of used shards which is ${gateway.totalShards}`)\n }\n\n const bucketId = shardId % gateway.connection.sessionStartLimit.maxConcurrency\n const bucket = gateway.buckets.get(bucketId)\n if (!bucket) {\n throw new Error(\n `Shard (id: ${shardId}) got assigned to an illegal bucket id: ${bucketId}, expected a bucket id between 0 and ${\n gateway.connection.sessionStartLimit.maxConcurrency - 1\n }`,\n )\n }\n\n // FIND A QUEUE IN THIS BUCKET THAT HAS SPACE\n // const worker = bucket.workers.find((w) => w.queue.length < gateway.shardsPerWorker);\n const workerId = gateway.calculateWorkerId(shardId)\n const worker = bucket.workers.find((w) => w.id === workerId)\n if (worker) {\n // IF THE QUEUE HAS SPACE JUST ADD IT TO THIS QUEUE\n worker.queue.push(shardId)\n } else {\n bucket.workers.push({ id: workerId, queue: [shardId] })\n }\n }\n },\n async spawnShards() {\n // PREPARES ALL SHARDS IN SPECIFIC BUCKETS\n gateway.prepareBuckets()\n\n // Prefer concurrency of forEach instead of forof\n await Promise.all(\n [...gateway.buckets.entries()].map(async ([bucketId, bucket]) => {\n for (const worker of bucket.workers) {\n for (const shardId of worker.queue) {\n await gateway.tellWorkerToIdentify(worker.id, shardId, bucketId)\n }\n }\n }),\n )\n },\n async shutdown(code, reason) {\n gateway.shards.forEach((shard) => shard.close(code, reason))\n\n await delay(5000)\n },\n async tellWorkerToIdentify(workerId, shardId, bucketId) {\n logger.debug(`[Gateway] tell worker to identify (${workerId}, ${shardId}, ${bucketId})`)\n return await gateway.identify(shardId)\n },\n async identify(shardId: number) {\n let shard = this.shards.get(shardId)\n logger.debug(`[Gateway] identifying ${shard ? 'existing' : 'new'} shard (${shardId})`)\n\n if (!shard) {\n shard = new Shard({\n id: shardId,\n connection: {\n compress: this.compress,\n intents: this.intents,\n properties: this.properties,\n token: this.token,\n totalShards: this.totalShards,\n url: this.url,\n version: this.version,\n },\n events: options.events,\n requestIdentify: async () => {\n await gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)!.leak.acquire()\n },\n })\n\n this.shards.set(shardId, shard)\n }\n\n logger.debug(`[Gateway] requesting to identify shard #(${shardId}) from bucket.`)\n await gateway.requestIdentify(shard.id)\n logger.debug(`[Gateway] Identify request successful for shard #(${shardId}).`)\n return await shard.identify()\n },\n async kill(shardId: number) {\n const shard = this.shards.get(shardId)\n if (!shard) {\n return logger.debug(`[Gateway] kill shard but not found (${shardId})`)\n }\n\n logger.debug(`[Gateway] kill shard (${shardId})`)\n this.shards.delete(shardId)\n return await shard.shutdown()\n },\n\n async requestIdentify(shardId: number) {\n logger.debug(`[Gateway] requesting identify`)\n await gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)!.leak.acquire()\n },\n\n // Helpers methods below this\n\n calculateShardId(guildId, totalShards) {\n // If none is provided, use the total shards number from gateway object.\n if (!totalShards) totalShards = gateway.totalShards\n // If it is only 1 shard, it will always be shard id 0\n if (totalShards === 1) {\n logger.debug(`[Gateway] calculateShardId (1 shard)`)\n return 0\n }\n\n logger.debug(`[Gateway] calculateShardId (guildId: ${guildId}, totalShards: ${totalShards})`)\n return Number((BigInt(guildId) >> 22n) % BigInt(totalShards))\n },\n\n async joinVoiceChannel(guildId, channelId, options) {\n const shardId = gateway.calculateShardId(guildId)\n const shard = gateway.shards.get(shardId)\n if (!shard) {\n throw new Error(`Shard (id: ${shardId} not found`)\n }\n\n logger.debug(`[Gateway] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`)\n shard.joinVoiceChannel(guildId, channelId, options)\n },\n\n async editBotStatus(data) {\n logger.debug(`[Gateway] editBotStatus data: ${JSON.stringify(data)}`)\n await Promise.all(\n [...gateway.shards.values()].map(async (shard) => {\n gateway.editShardStatus(shard.id, data)\n }),\n )\n },\n\n async editShardStatus(shardId, data) {\n const shard = gateway.shards.get(shardId)\n if (!shard) {\n throw new Error(`Shard (id: ${shardId}) not found.`)\n }\n\n logger.debug(`[Gateway] editShardStatus shardId: ${shardId} -> data: ${JSON.stringify(data)}`)\n return await shard.editShardStatus(data)\n },\n\n async requestMembers(guildId, options) {\n const shardId = gateway.calculateShardId(guildId)\n const shard = gateway.shards.get(shardId)\n if (!shard) {\n throw new Error(`Shard (id: ${shardId}) not found.`)\n }\n\n logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> options ${JSON.stringify(options)}`)\n return await shard.requestMembers(guildId, options)\n },\n\n async leaveVoiceChannel(guildId) {\n const shardId = gateway.calculateShardId(guildId)\n const shard = gateway.shards.get(shardId)\n if (!shard) {\n throw new Error(`Shard (id: ${shardId} not found`)\n }\n\n logger.debug(`[Gateway] leaveVoiceChannel guildId: ${guildId} Shard ${shardId}`)\n return await shard.leaveVoiceChannel(guildId)\n },\n }\n\n return gateway\n}\n\nexport interface CreateGatewayManagerOptions {\n /**\n * Id of the first Shard which should get controlled by this manager.\n * @default 0\n */\n firstShardId?: number\n /**\n * Id of the last Shard which should get controlled by this manager.\n * @default 0\n */\n lastShardId?: number\n /**\n * Delay in milliseconds to wait before spawning next shard. OPTIMAL IS ABOVE 5100. YOU DON'T WANT TO HIT THE RATE LIMIT!!!\n * @default 5300\n */\n spawnShardDelay?: number\n /**\n * Total amount of shards your bot uses. Useful for zero-downtime updates or resharding.\n * @default 1\n */\n totalShards?: number\n /**\n * The amount of shards to load per worker.\n * @default 25\n */\n shardsPerWorker?: number\n /**\n * The total amount of workers to use for your bot.\n * @default 4\n */\n totalWorkers?: number\n /** Important data which is used by the manager to connect shards to the gateway. */\n connection?: Camelize<DiscordGetGatewayBot>\n /** Whether incoming payloads are compressed using zlib.\n *\n * @default false\n */\n compress?: boolean\n /** The calculated intent value of the events which the shard should receive.\n *\n * @default 0\n */\n intents?: number\n /** Identify properties to use */\n properties?: {\n /** Operating system the shard runs on.\n *\n * @default \"darwin\" | \"linux\" | \"windows\"\n */\n os: string\n /** The \"browser\" where this shard is running on.\n *\n * @default \"Discordeno\"\n */\n browser: string\n /** The device on which the shard is running.\n *\n * @default \"Discordeno\"\n */\n device: string\n }\n /** Bot token which is used to connect to Discord */\n token: string\n /** The URL of the gateway which should be connected to.\n *\n * @default \"wss://gateway.discord.gg\"\n */\n url?: string\n /** The gateway version which should be used.\n *\n * @default 10\n */\n version?: number\n /** The events handlers */\n events: ShardEvents\n /** This managers cache related settings. */\n cache?: {\n requestMembers?: {\n /**\n * Whether or not request member requests should be cached.\n * @default false\n */\n enabled?: boolean\n /** The pending requests. */\n pending: Collection<string, RequestMemberRequest>\n }\n }\n}\n\nexport interface GatewayManager extends Required<CreateGatewayManagerOptions> {\n /** The max concurrency buckets. Those will be created when the `spawnShards` (which calls `prepareBuckets` under the hood) function gets called. */\n buckets: Map<\n number,\n {\n workers: Array<{ id: number; queue: number[] }>\n leak: LeakyBucket\n }\n >\n /** The shards that are created. */\n shards: Map<number, Shard>\n /** Determine max number of shards to use based upon the max concurrency. */\n calculateTotalShards: () => number\n /** Determine the id of the worker which is handling a shard. */\n calculateWorkerId: (shardId: number) => number\n /** Prepares all the buckets that are available for identifying the shards. */\n prepareBuckets: () => void\n /** Start identifying all the shards. */\n spawnShards: () => Promise<void>\n /** Shutdown all shards. */\n shutdown: (code: number, reason: string) => Promise<void>\n /** Allows users to hook in and change to communicate to different workers across different servers or anything they like. For example using redis pubsub to talk to other servers. */\n tellWorkerToIdentify: (workerId: number, shardId: number, bucketId: number) => Promise<void>\n /** Tell the manager to identify a Shard. If this Shard is not already managed this will also add the Shard to the manager. */\n identify: (shardId: number) => Promise<void>\n /** Kill a shard. Close a shards connection to Discord's gateway (if any) and remove it from the manager. */\n kill: (shardId: number) => Promise<void>\n /** This function communicates with the parent manager, in order to know whether this manager is allowed to identify a new shard. */\n requestIdentify: (shardId: number) => Promise<void>\n /** Calculates the number of shards based on the guild id and total shards. */\n calculateShardId: (guildId: BigString, totalShards?: number) => number\n /**\n * Connects the bot user to a voice or stage channel.\n *\n * This function sends the _Update Voice State_ gateway command over the gateway behind the scenes.\n *\n * @param guildId - The ID of the guild the voice channel to leave is in.\n * @param channelId - The ID of the channel you want to join.\n *\n * @remarks\n * Requires the `CONNECT` permission.\n *\n * Fires a _Voice State Update_ gateway event.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}\n */\n joinVoiceChannel: (guildId: BigString, channelId: BigString, options?: AtLeastOne<Omit<UpdateVoiceState, 'guildId' | 'channelId'>>) => Promise<void>\n /**\n * Edits the bot status in all shards that this gateway manages.\n *\n * @param data The status data to set the bots status to.\n * @returns Promise<void>\n */\n editBotStatus: (data: StatusUpdate) => Promise<void>\n /**\n * Edits the bot's status on one shard.\n *\n * @param shardId The shard id to edit the status for.\n * @param data The status data to set the bots status to.\n * @returns Promise<void>\n */\n editShardStatus: (shardId: number, data: StatusUpdate) => Promise<void>\n /**\n * Fetches the list of members for a guild over the gateway.\n *\n * @param guildId - The ID of the guild to get the list of members for.\n * @param options - The parameters for the fetching of the members.\n *\n * @remarks\n * If requesting the entire member list:\n * - Requires the `GUILD_MEMBERS` intent.\n *\n * If requesting presences ({@link RequestGuildMembers.presences | presences} set to `true`):\n * - Requires the `GUILD_PRESENCES` intent.\n *\n * If requesting a prefix ({@link RequestGuildMembers.query | query} non-`undefined`):\n * - Returns a maximum of 100 members.\n *\n * If requesting a users by ID ({@link RequestGuildMembers.userIds | userIds} non-`undefined`):\n * - Returns a maximum of 100 members.\n *\n * Fires a _Guild Members Chunk_ gateway event for every 1000 members fetched.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#request-guild-members}\n */\n requestMembers: (guildId: BigString, options?: Omit<RequestGuildMembers, 'guildId'>) => Promise<Camelize<DiscordMember[]>>\n /**\n * Leaves the voice channel the bot user is currently in.\n *\n * This function sends the _Update Voice State_ gateway command over the gateway behind the scenes.\n *\n * @param guildId - The ID of the guild the voice channel to leave is in.\n *\n * @remarks\n * Fires a _Voice State Update_ gateway event.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}\n */\n leaveVoiceChannel: (guildId: BigString) => Promise<void>\n}\n\nexport interface RequestMemberRequest {\n /** The unique nonce for this request. */\n nonce: string\n /** The resolver handler to run when all members arrive. */\n resolve: (value: Camelize<DiscordMember[]> | PromiseLike<Camelize<DiscordMember[]>>) => void\n /** The members that have already arrived for this request. */\n members: Camelize<DiscordMember[]>\n}\n"],"names":["Collection","delay","LeakyBucket","logger","Shard","createGatewayManager","options","connection","url","shards","sessionStartLimit","maxConcurrency","remaining","total","resetAfter","gateway","events","compress","intents","properties","os","process","platform","browser","device","token","version","totalShards","lastShardId","firstShardId","totalWorkers","shardsPerWorker","spawnShardDelay","Map","buckets","cache","requestMembers","enabled","pending","calculateTotalShards","debug","Math","ceil","calculateWorkerId","shardId","workerId","floor","prepareBuckets","i","set","workers","leak","refillInterval","Error","bucketId","bucket","get","worker","find","w","id","queue","push","spawnShards","Promise","all","entries","map","tellWorkerToIdentify","shutdown","code","reason","forEach","shard","close","identify","requestIdentify","acquire","kill","delete","calculateShardId","guildId","Number","BigInt","joinVoiceChannel","channelId","editBotStatus","data","JSON","stringify","values","editShardStatus","leaveVoiceChannel"],"mappings":"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"}
1
+ {"version":3,"sources":["../src/manager.ts"],"sourcesContent":["/* eslint-disable @typescript-eslint/no-confusing-void-expression */\nimport type {\n AtLeastOne,\n BigString,\n Camelize,\n DiscordGetGatewayBot,\n DiscordMember,\n DiscordMemberWithUser,\n RequestGuildMembers,\n} from '@discordeno/types'\nimport { Collection, delay, logger } from '@discordeno/utils'\nimport Shard from './Shard.js'\nimport type { ShardEvents, StatusUpdate, UpdateVoiceState } from './types.js'\n\nexport function createGatewayManager(options: CreateGatewayManagerOptions): GatewayManager {\n const connectionOptions = options.connection ?? {\n url: 'wss://gateway.discord.gg',\n shards: 1,\n sessionStartLimit: {\n maxConcurrency: 1,\n remaining: 1000,\n total: 1000,\n resetAfter: 1000 * 60 * 60 * 24,\n },\n }\n\n const gateway: GatewayManager = {\n events: options.events,\n compress: options.compress ?? false,\n intents: options.intents ?? 0,\n properties: {\n os: options.properties?.os ?? process.platform,\n browser: options.properties?.browser ?? 'Discordeno',\n device: options.properties?.device ?? 'Discordeno',\n },\n token: options.token,\n url: options.url ?? connectionOptions.url ?? 'wss://gateway.discord.gg',\n version: options.version ?? 10,\n connection: connectionOptions,\n totalShards: options.totalShards ?? connectionOptions.shards ?? 1,\n lastShardId: options.lastShardId ?? (options.totalShards ? options.totalShards - 1 : connectionOptions ? connectionOptions.shards - 1 : 0),\n firstShardId: options.firstShardId ?? 0,\n totalWorkers: options.totalWorkers ?? 4,\n shardsPerWorker: options.shardsPerWorker ?? 25,\n spawnShardDelay: options.spawnShardDelay ?? 5300,\n preferSnakeCase: options.preferSnakeCase ?? false,\n shards: new Map(),\n buckets: new Map(),\n cache: {\n requestMembers: {\n enabled: options.cache?.requestMembers?.enabled ?? false,\n pending: new Collection(),\n },\n },\n\n calculateTotalShards() {\n // Bots under 100k servers do not have access to total shards.\n if (gateway.totalShards < 100) {\n logger.debug(`[Gateway] Calculating total shards: ${gateway.totalShards}`)\n return gateway.totalShards\n }\n\n logger.debug(`[Gateway] Calculating total shards`, gateway.totalShards, gateway.connection.sessionStartLimit.maxConcurrency)\n // Calculate a multiple of `maxConcurrency` which can be used to connect to the gateway.\n return (\n Math.ceil(\n gateway.totalShards /\n // If `maxConcurrency` is 1 we can safely use 16.\n (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency),\n ) * gateway.connection.sessionStartLimit.maxConcurrency\n )\n },\n calculateWorkerId(shardId) {\n const workerId = Math.min(shardId % gateway.shardsPerWorker, gateway.totalWorkers - 1)\n logger.debug(\n `[Gateway] Calculating workerId: Shard: ${shardId} -> Worker: ${workerId} -> Per Worker: ${gateway.shardsPerWorker} -> Total: ${gateway.totalWorkers}`,\n )\n return workerId\n },\n prepareBuckets() {\n for (let i = 0; i < gateway.connection.sessionStartLimit.maxConcurrency; ++i) {\n logger.debug(`[Gateway] Preparing buckets for concurrency: ${i}`)\n gateway.buckets.set(i, {\n workers: [],\n identifyRequests: [],\n })\n }\n\n // ORGANIZE ALL SHARDS INTO THEIR OWN BUCKETS\n for (let shardId = gateway.firstShardId; shardId <= gateway.lastShardId; ++shardId) {\n logger.debug(`[Gateway] Preparing buckets for shard: ${shardId}`)\n if (shardId >= gateway.totalShards) {\n throw new Error(`Shard (id: ${shardId}) is bigger or equal to the used amount of used shards which is ${gateway.totalShards}`)\n }\n\n const bucketId = shardId % gateway.connection.sessionStartLimit.maxConcurrency\n const bucket = gateway.buckets.get(bucketId)\n if (!bucket) {\n throw new Error(\n `Shard (id: ${shardId}) got assigned to an illegal bucket id: ${bucketId}, expected a bucket id between 0 and ${\n gateway.connection.sessionStartLimit.maxConcurrency - 1\n }`,\n )\n }\n\n // FIND A QUEUE IN THIS BUCKET THAT HAS SPACE\n // const worker = bucket.workers.find((w) => w.queue.length < gateway.shardsPerWorker);\n const workerId = gateway.calculateWorkerId(shardId)\n const worker = bucket.workers.find((w) => w.id === workerId)\n if (worker) {\n // IF THE QUEUE HAS SPACE JUST ADD IT TO THIS QUEUE\n worker.queue.push(shardId)\n } else {\n bucket.workers.push({ id: workerId, queue: [shardId] })\n }\n }\n\n for (const bucket of gateway.buckets.values()) {\n for (const worker of bucket.workers.values()) {\n // eslint-disable-next-line @typescript-eslint/require-array-sort-compare\n worker.queue = worker.queue.sort((a, b) => a - b)\n }\n }\n },\n async spawnShards() {\n // PREPARES ALL SHARDS IN SPECIFIC BUCKETS\n gateway.prepareBuckets()\n\n // Prefer concurrency of forEach instead of forof\n await Promise.all(\n [...gateway.buckets.entries()].map(async ([bucketId, bucket]) => {\n for (const worker of bucket.workers) {\n for (const shardId of worker.queue) {\n await gateway.tellWorkerToIdentify(worker.id, shardId, bucketId)\n }\n }\n }),\n )\n },\n async shutdown(code, reason) {\n gateway.shards.forEach((shard) => shard.close(code, reason))\n\n await delay(5000)\n },\n async tellWorkerToIdentify(workerId, shardId, bucketId) {\n logger.debug(`[Gateway] tell worker to identify (${workerId}, ${shardId}, ${bucketId})`)\n await gateway.identify(shardId)\n },\n async identify(shardId: number) {\n let shard = this.shards.get(shardId)\n logger.debug(`[Gateway] identifying ${shard ? 'existing' : 'new'} shard (${shardId})`)\n\n if (!shard) {\n shard = new Shard({\n id: shardId,\n connection: {\n compress: this.compress,\n intents: this.intents,\n properties: this.properties,\n token: this.token,\n totalShards: this.totalShards,\n url: this.url,\n version: this.version,\n },\n events: options.events,\n requestIdentify: async () => {\n await gateway.identify(shardId)\n },\n shardIsReady: async () => {\n logger.debug(`[Shard] Shard #${shardId} is ready`)\n await delay(gateway.spawnShardDelay)\n logger.debug(`[Shard] Resolving shard identify request`)\n gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)!.identifyRequests.shift()?.()\n },\n })\n\n if (this.preferSnakeCase) {\n shard.forwardToBot = async (payload) => {\n options.events.message?.(shard!, payload)\n }\n }\n\n this.shards.set(shardId, shard)\n }\n\n const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)\n if (!bucket) return\n\n return await new Promise((resolve) => {\n // Mark that we are making an identify request so another is not made.\n bucket.identifyRequests.push(resolve)\n logger.debug(`[Gateway] identifying shard #(${shardId}).`)\n // This will trigger identify and when READY is received it will resolve the above request.\n shard?.identify()\n })\n },\n async kill(shardId: number) {\n const shard = this.shards.get(shardId)\n if (!shard) {\n return logger.debug(`[Gateway] kill shard but not found (${shardId})`)\n }\n\n logger.debug(`[Gateway] kill shard (${shardId})`)\n this.shards.delete(shardId)\n await shard.shutdown()\n },\n\n async requestIdentify(shardId: number) {\n logger.debug(`[Gateway] requesting identify`)\n // const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)\n // if (!bucket) return\n\n // return await new Promise((resolve) => {\n // bucket.identifyRequests.push(resolve)\n // })\n },\n\n // Helpers methods below this\n\n calculateShardId(guildId, totalShards) {\n // If none is provided, use the total shards number from gateway object.\n if (!totalShards) totalShards = gateway.totalShards\n // If it is only 1 shard, it will always be shard id 0\n if (totalShards === 1) {\n logger.debug(`[Gateway] calculateShardId (1 shard)`)\n return 0\n }\n\n logger.debug(`[Gateway] calculateShardId (guildId: ${guildId}, totalShards: ${totalShards})`)\n return Number((BigInt(guildId) >> 22n) % BigInt(totalShards))\n },\n\n async joinVoiceChannel(guildId, channelId, options) {\n const shardId = gateway.calculateShardId(guildId)\n const shard = gateway.shards.get(shardId)\n if (!shard) {\n throw new Error(`Shard (id: ${shardId} not found`)\n }\n\n logger.debug(`[Gateway] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`)\n shard.joinVoiceChannel(guildId, channelId, options)\n },\n\n async editBotStatus(data) {\n logger.debug(`[Gateway] editBotStatus data: ${JSON.stringify(data)}`)\n await Promise.all(\n [...gateway.shards.values()].map(async (shard) => {\n gateway.editShardStatus(shard.id, data)\n }),\n )\n },\n\n async editShardStatus(shardId, data) {\n const shard = gateway.shards.get(shardId)\n if (!shard) {\n throw new Error(`Shard (id: ${shardId}) not found.`)\n }\n\n logger.debug(`[Gateway] editShardStatus shardId: ${shardId} -> data: ${JSON.stringify(data)}`)\n await shard.editShardStatus(data)\n },\n\n async requestMembers(guildId, options) {\n const shardId = gateway.calculateShardId(guildId)\n const shard = gateway.shards.get(shardId)\n if (!shard) {\n throw new Error(`Shard (id: ${shardId}) not found.`)\n }\n\n logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> options ${JSON.stringify(options)}`)\n return await shard.requestMembers(guildId, options)\n },\n\n async leaveVoiceChannel(guildId) {\n const shardId = gateway.calculateShardId(guildId)\n const shard = gateway.shards.get(shardId)\n if (!shard) {\n throw new Error(`Shard (id: ${shardId} not found`)\n }\n\n logger.debug(`[Gateway] leaveVoiceChannel guildId: ${guildId} Shard ${shardId}`)\n await shard.leaveVoiceChannel(guildId)\n },\n }\n\n return gateway\n}\n\nexport interface CreateGatewayManagerOptions {\n /**\n * Id of the first Shard which should get controlled by this manager.\n * @default 0\n */\n firstShardId?: number\n /**\n * Id of the last Shard which should get controlled by this manager.\n * @default 0\n */\n lastShardId?: number\n /**\n * Delay in milliseconds to wait before spawning next shard. OPTIMAL IS ABOVE 5100. YOU DON'T WANT TO HIT THE RATE LIMIT!!!\n * @default 5300\n */\n spawnShardDelay?: number\n /**\n * Whether to send the discord packets in snake case form.\n * @default false\n */\n preferSnakeCase?: boolean\n /**\n * Total amount of shards your bot uses. Useful for zero-downtime updates or resharding.\n * @default 1\n */\n totalShards?: number\n /**\n * The amount of shards to load per worker.\n * @default 25\n */\n shardsPerWorker?: number\n /**\n * The total amount of workers to use for your bot.\n * @default 4\n */\n totalWorkers?: number\n /** Important data which is used by the manager to connect shards to the gateway. */\n connection?: Camelize<DiscordGetGatewayBot>\n /** Whether incoming payloads are compressed using zlib.\n *\n * @default false\n */\n compress?: boolean\n /** The calculated intent value of the events which the shard should receive.\n *\n * @default 0\n */\n intents?: number\n /** Identify properties to use */\n properties?: {\n /** Operating system the shard runs on.\n *\n * @default \"darwin\" | \"linux\" | \"windows\"\n */\n os: string\n /** The \"browser\" where this shard is running on.\n *\n * @default \"Discordeno\"\n */\n browser: string\n /** The device on which the shard is running.\n *\n * @default \"Discordeno\"\n */\n device: string\n }\n /** Bot token which is used to connect to Discord */\n token: string\n /** The URL of the gateway which should be connected to.\n *\n * @default \"wss://gateway.discord.gg\"\n */\n url?: string\n /** The gateway version which should be used.\n *\n * @default 10\n */\n version?: number\n /** The events handlers */\n events: ShardEvents\n /** This managers cache related settings. */\n cache?: {\n requestMembers?: {\n /**\n * Whether or not request member requests should be cached.\n * @default false\n */\n enabled?: boolean\n /** The pending requests. */\n pending: Collection<string, RequestMemberRequest>\n }\n }\n}\n\nexport interface GatewayManager extends Required<CreateGatewayManagerOptions> {\n /** The max concurrency buckets. Those will be created when the `spawnShards` (which calls `prepareBuckets` under the hood) function gets called. */\n buckets: Map<\n number,\n {\n workers: Array<{ id: number; queue: number[] }>\n /** Requests to identify shards are made based on whether it is available to be made. */\n identifyRequests: Array<(value: void | PromiseLike<void>) => void>\n }\n >\n /** The shards that are created. */\n shards: Map<number, Shard>\n /** Determine max number of shards to use based upon the max concurrency. */\n calculateTotalShards: () => number\n /** Determine the id of the worker which is handling a shard. */\n calculateWorkerId: (shardId: number) => number\n /** Prepares all the buckets that are available for identifying the shards. */\n prepareBuckets: () => void\n /** Start identifying all the shards. */\n spawnShards: () => Promise<void>\n /** Shutdown all shards. */\n shutdown: (code: number, reason: string) => Promise<void>\n /** Allows users to hook in and change to communicate to different workers across different servers or anything they like. For example using redis pubsub to talk to other servers. */\n tellWorkerToIdentify: (workerId: number, shardId: number, bucketId: number) => Promise<void>\n /** Tell the manager to identify a Shard. If this Shard is not already managed this will also add the Shard to the manager. */\n identify: (shardId: number) => Promise<void>\n /** Kill a shard. Close a shards connection to Discord's gateway (if any) and remove it from the manager. */\n kill: (shardId: number) => Promise<void>\n /** This function makes sure that the bucket is allowed to make the next identify request. */\n requestIdentify: (shardId: number) => Promise<void>\n /** Calculates the number of shards based on the guild id and total shards. */\n calculateShardId: (guildId: BigString, totalShards?: number) => number\n /**\n * Connects the bot user to a voice or stage channel.\n *\n * This function sends the _Update Voice State_ gateway command over the gateway behind the scenes.\n *\n * @param guildId - The ID of the guild the voice channel to leave is in.\n * @param channelId - The ID of the channel you want to join.\n *\n * @remarks\n * Requires the `CONNECT` permission.\n *\n * Fires a _Voice State Update_ gateway event.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}\n */\n joinVoiceChannel: (guildId: BigString, channelId: BigString, options?: AtLeastOne<Omit<UpdateVoiceState, 'guildId' | 'channelId'>>) => Promise<void>\n /**\n * Edits the bot status in all shards that this gateway manages.\n *\n * @param data The status data to set the bots status to.\n * @returns Promise<void>\n */\n editBotStatus: (data: StatusUpdate) => Promise<void>\n /**\n * Edits the bot's status on one shard.\n *\n * @param shardId The shard id to edit the status for.\n * @param data The status data to set the bots status to.\n * @returns Promise<void>\n */\n editShardStatus: (shardId: number, data: StatusUpdate) => Promise<void>\n /**\n * Fetches the list of members for a guild over the gateway.\n *\n * @param guildId - The ID of the guild to get the list of members for.\n * @param options - The parameters for the fetching of the members.\n *\n * @remarks\n * If requesting the entire member list:\n * - Requires the `GUILD_MEMBERS` intent.\n *\n * If requesting presences ({@link RequestGuildMembers.presences | presences} set to `true`):\n * - Requires the `GUILD_PRESENCES` intent.\n *\n * If requesting a prefix ({@link RequestGuildMembers.query | query} non-`undefined`):\n * - Returns a maximum of 100 members.\n *\n * If requesting a users by ID ({@link RequestGuildMembers.userIds | userIds} non-`undefined`):\n * - Returns a maximum of 100 members.\n *\n * Fires a _Guild Members Chunk_ gateway event for every 1000 members fetched.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#request-guild-members}\n */\n requestMembers: (guildId: BigString, options?: Omit<RequestGuildMembers, 'guildId'>) => Promise<Camelize<DiscordMember[]>>\n /**\n * Leaves the voice channel the bot user is currently in.\n *\n * This function sends the _Update Voice State_ gateway command over the gateway behind the scenes.\n *\n * @param guildId - The ID of the guild the voice channel to leave is in.\n *\n * @remarks\n * Fires a _Voice State Update_ gateway event.\n *\n * @see {@link https://discord.com/developers/docs/topics/gateway#update-voice-state}\n */\n leaveVoiceChannel: (guildId: BigString) => Promise<void>\n}\n\nexport interface RequestMemberRequest {\n /** The unique nonce for this request. */\n nonce: string\n /** The resolver handler to run when all members arrive. */\n resolve: (value: Camelize<DiscordMember[]> | PromiseLike<Camelize<DiscordMember[]>>) => void\n /** The members that have already arrived for this request. */\n members: DiscordMemberWithUser[]\n}\n"],"names":["Collection","delay","logger","Shard","createGatewayManager","options","connectionOptions","connection","url","shards","sessionStartLimit","maxConcurrency","remaining","total","resetAfter","gateway","events","compress","intents","properties","os","process","platform","browser","device","token","version","totalShards","lastShardId","firstShardId","totalWorkers","shardsPerWorker","spawnShardDelay","preferSnakeCase","Map","buckets","cache","requestMembers","enabled","pending","calculateTotalShards","debug","Math","ceil","calculateWorkerId","shardId","workerId","min","prepareBuckets","i","set","workers","identifyRequests","Error","bucketId","bucket","get","worker","find","w","id","queue","push","values","sort","a","b","spawnShards","Promise","all","entries","map","tellWorkerToIdentify","shutdown","code","reason","forEach","shard","close","identify","requestIdentify","shardIsReady","shift","forwardToBot","payload","message","resolve","kill","delete","calculateShardId","guildId","Number","BigInt","joinVoiceChannel","channelId","editBotStatus","data","JSON","stringify","editShardStatus","leaveVoiceChannel"],"mappings":"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"}
package/dist/types.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/types.ts"],"sourcesContent":["import type { ActivityTypes, Camelize, DiscordActivity, DiscordGatewayPayload, GatewayOpcodes, PresenceStatus } from '@discordeno/types'\nimport type Shard from './Shard.js'\n\nexport enum ShardState {\n /** Shard is fully connected to the gateway and receiving events from Discord. */\n Connected = 0,\n /** Shard started to connect to the gateway. This is only used if the shard is not currently trying to identify or resume. */\n Connecting = 1,\n /** Shard got disconnected and reconnection actions have been started. */\n Disconnected = 2,\n /** The shard is connected to the gateway but only heartbeating. At this state the shard has not been identified with discord. */\n Unidentified = 3,\n /** Shard is trying to identify with the gateway to create a new session. */\n Identifying = 4,\n /** Shard is trying to resume a session with the gateway. */\n Resuming = 5,\n /** Shard got shut down studied or due to a not (self) fixable error and may not attempt to reconnect on its own. */\n Offline = 6,\n}\n\nexport interface ShardGatewayConfig {\n /** Whether incoming payloads are compressed using zlib.\n *\n * @default false\n */\n compress: boolean\n /** The calculated intent value of the events which the shard should receive.\n *\n * @default 0\n */\n intents: number\n /** Identify properties to use */\n properties: {\n /** Operating system the shard runs on.\n *\n * @default \"darwin\" | \"linux\" | \"windows\"\n */\n os: string\n /** The \"browser\" where this shard is running on.\n *\n * @default \"Discordeno\"\n */\n browser: string\n /** The device on which the shard is running.\n *\n * @default \"Discordeno\"\n */\n device: string\n }\n /** Bot token which is used to connect to Discord */\n token: string\n /** The URL of the gateway which should be connected to.\n *\n * @default \"wss://gateway.discord.gg\"\n */\n url: string\n /** The gateway version which should be used.\n *\n * @default 10\n */\n version: number\n /**\n * The total number of shards to connect to across the entire bot.\n * @default 1\n */\n totalShards: number\n}\n\nexport interface ShardHeart {\n /** Whether or not the heartbeat was acknowledged by Discord in time. */\n acknowledged: boolean\n /** Interval between heartbeats requested by Discord. */\n interval: number\n /** Id of the interval, which is used for sending the heartbeats. */\n intervalId?: NodeJS.Timer\n /** Unix (in milliseconds) timestamp when the last heartbeat ACK was received from Discord. */\n lastAck?: number\n /** Unix timestamp (in milliseconds) when the last heartbeat was sent. */\n lastBeat?: number\n /** Round trip time (in milliseconds) from Shard to Discord and back.\n * Calculated using the heartbeat system.\n * Note: this value is undefined until the first heartbeat to Discord has happened.\n */\n rtt?: number\n /** Id of the timeout which is used for sending the first heartbeat to Discord since it's \"special\". */\n timeoutId?: NodeJS.Timeout\n}\n\nexport interface ShardEvents {\n /** A heartbeat has been send. */\n heartbeat?: (shard: Shard) => unknown\n /** A heartbeat ACK was received. */\n heartbeatAck?: (shard: Shard) => unknown\n /** Shard has received a Hello payload. */\n hello?: (shard: Shard) => unknown\n /** The Shards session has been invalidated. */\n invalidSession?: (shard: Shard, resumable: boolean) => unknown\n /** The shard has started a resume action. */\n resuming?: (shard: Shard) => unknown\n /** The shard has successfully resumed an old session. */\n resumed?: (shard: Shard) => unknown\n /** Discord has requested the Shard to reconnect. */\n requestedReconnect?: (shard: Shard) => unknown\n /** The shard started to connect to Discord's gateway. */\n connecting?: (shard: Shard) => unknown\n /** The shard is connected with Discord's gateway. */\n connected?: (shard: Shard) => unknown\n /** The shard has been disconnected from Discord's gateway. */\n disconnected?: (shard: Shard) => unknown\n /** The shard has started to identify itself to Discord. */\n identifying?: (shard: Shard) => unknown\n /** The shard has successfully been identified itself with Discord. */\n identified?: (shard: Shard) => unknown\n /** The shard has received a message from Discord. */\n message?: (shard: Shard, payload: Camelize<DiscordGatewayPayload>) => unknown\n}\n\nexport enum ShardSocketCloseCodes {\n /** A regular Shard shutdown. */\n Shutdown = 3000,\n /** A resume has been requested and therefore the old connection needs to be closed. */\n ResumeClosingOldConnection = 3024,\n /** Did not receive a heartbeat ACK in time.\n * Closing the shard and creating a new session.\n */\n ZombiedConnection = 3010,\n /** Discordeno's gateway tests hae been finished, therefore the Shard can be turned off. */\n TestingFinished = 3064,\n /** Special close code reserved for Discordeno's zero-downtime resharding system. */\n Resharded = 3065,\n /** Shard is re-identifying therefore the old connection needs to be closed. */\n ReIdentifying = 3066,\n}\n\nexport interface ShardSocketRequest {\n /** The OP-Code for the payload to send. */\n op: GatewayOpcodes\n /** Payload data. */\n d: unknown\n}\n\n/** https://discord.com/developers/docs/topics/gateway-events#update-presence */\nexport interface BotStatusUpdate {\n // /** Unix time (in milliseconds) of when the client went idle, or null if the client is not idle */\n since: number | null\n /** The user's activities */\n activities: BotActivity[]\n /** The user's new status */\n status: keyof typeof PresenceStatus\n}\n\n/** https://discord.com/developers/docs/topics/gateway-events#activity-object */\nexport interface BotActivity {\n name: string\n type: ActivityTypes\n url?: string\n}\n\n/** https://discord.com/developers/docs/topics/gateway#update-voice-state */\nexport interface UpdateVoiceState {\n /** id of the guild */\n guildId: string\n /** id of the voice channel client wants to join (null if disconnecting) */\n channelId: string | null\n /** Is the client muted */\n selfMute: boolean\n /** Is the client deafened */\n selfDeaf: boolean\n}\n\n/** https://discord.com/developers/docs/topics/gateway-events#update-presence */\nexport interface StatusUpdate {\n // /** Unix time (in milliseconds) of when the client went idle, or null if the client is not idle */\n // since: number | null;\n /** The user's activities */\n activities?: Camelize<Array<Omit<DiscordActivity, 'created_at'>>>\n /** The user's new status */\n status: keyof typeof PresenceStatus\n // /** Whether or not the client is afk */\n // afk: boolean;\n}\n"],"names":["ShardState","Connected","Connecting","Disconnected","Unidentified","Identifying","Resuming","Offline","ShardSocketCloseCodes","Shutdown","ResumeClosingOldConnection","ZombiedConnection","TestingFinished","Resharded","ReIdentifying"],"mappings":"AAAA,WAGO;UAAKA,UAAU;IAAVA,WAAAA,WACV,+EAA+E,GAC/EC,eAAY,KAAZA;IAFUD,WAAAA,WAGV,2HAA2H,GAC3HE,gBAAa,KAAbA;IAJUF,WAAAA,WAKV,uEAAuE,GACvEG,kBAAe,KAAfA;IANUH,WAAAA,WAOV,+HAA+H,GAC/HI,kBAAe,KAAfA;IARUJ,WAAAA,WASV,0EAA0E,GAC1EK,iBAAc,KAAdA;IAVUL,WAAAA,WAWV,0DAA0D,GAC1DM,cAAW,KAAXA;IAZUN,WAAAA,WAaV,kHAAkH,GAClHO,aAAU,KAAVA;GAdUP,eAAAA;WAkHL;UAAKQ,qBAAqB;IAArBA,sBAAAA,sBACV,8BAA8B,GAC9BC,cAAW,QAAXA;IAFUD,sBAAAA,sBAGV,qFAAqF,GACrFE,gCAA6B,QAA7BA;IAJUF,sBAAAA,sBAKV;;GAEC,GACDG,uBAAoB,QAApBA;IARUH,sBAAAA,sBASV,yFAAyF,GACzFI,qBAAkB,QAAlBA;IAVUJ,sBAAAA,sBAWV,kFAAkF,GAClFK,eAAY,QAAZA;IAZUL,sBAAAA,sBAaV,6EAA6E,GAC7EM,mBAAgB,QAAhBA;GAdUN,0BAAAA"}
1
+ {"version":3,"sources":["../src/types.ts"],"sourcesContent":["import type { ActivityTypes, Camelize, DiscordActivity, DiscordGatewayPayload, GatewayOpcodes, PresenceStatus } from '@discordeno/types'\nimport type Shard from './Shard.js'\n\nexport enum ShardState {\n /** Shard is fully connected to the gateway and receiving events from Discord. */\n Connected = 0,\n /** Shard started to connect to the gateway. This is only used if the shard is not currently trying to identify or resume. */\n Connecting = 1,\n /** Shard got disconnected and reconnection actions have been started. */\n Disconnected = 2,\n /** The shard is connected to the gateway but only heartbeating. At this state the shard has not been identified with discord. */\n Unidentified = 3,\n /** Shard is trying to identify with the gateway to create a new session. */\n Identifying = 4,\n /** Shard is trying to resume a session with the gateway. */\n Resuming = 5,\n /** Shard got shut down studied or due to a not (self) fixable error and may not attempt to reconnect on its own. */\n Offline = 6,\n}\n\nexport interface ShardGatewayConfig {\n /** Whether incoming payloads are compressed using zlib.\n *\n * @default false\n */\n compress: boolean\n /** The calculated intent value of the events which the shard should receive.\n *\n * @default 0\n */\n intents: number\n /** Identify properties to use */\n properties: {\n /** Operating system the shard runs on.\n *\n * @default \"darwin\" | \"linux\" | \"windows\"\n */\n os: string\n /** The \"browser\" where this shard is running on.\n *\n * @default \"Discordeno\"\n */\n browser: string\n /** The device on which the shard is running.\n *\n * @default \"Discordeno\"\n */\n device: string\n }\n /** Bot token which is used to connect to Discord */\n token: string\n /** The URL of the gateway which should be connected to.\n *\n * @default \"wss://gateway.discord.gg\"\n */\n url: string\n /** The gateway version which should be used.\n *\n * @default 10\n */\n version: number\n /**\n * The total number of shards to connect to across the entire bot.\n * @default 1\n */\n totalShards: number\n}\n\nexport interface ShardHeart {\n /** Whether or not the heartbeat was acknowledged by Discord in time. */\n acknowledged: boolean\n /** Interval between heartbeats requested by Discord. */\n interval: number\n /** Id of the interval, which is used for sending the heartbeats. */\n intervalId?: NodeJS.Timer\n /** Unix (in milliseconds) timestamp when the last heartbeat ACK was received from Discord. */\n lastAck?: number\n /** Unix timestamp (in milliseconds) when the last heartbeat was sent. */\n lastBeat?: number\n /** Round trip time (in milliseconds) from Shard to Discord and back.\n * Calculated using the heartbeat system.\n * Note: this value is undefined until the first heartbeat to Discord has happened.\n */\n rtt?: number\n /** Id of the timeout which is used for sending the first heartbeat to Discord since it's \"special\". */\n timeoutId?: NodeJS.Timeout\n}\n\nexport interface ShardEvents {\n /** A heartbeat has been send. */\n heartbeat?: (shard: Shard) => unknown\n /** A heartbeat ACK was received. */\n heartbeatAck?: (shard: Shard) => unknown\n /** Shard has received a Hello payload. */\n hello?: (shard: Shard) => unknown\n /** The Shards session has been invalidated. */\n invalidSession?: (shard: Shard, resumable: boolean) => unknown\n /** The shard has started a resume action. */\n resuming?: (shard: Shard) => unknown\n /** The shard has successfully resumed an old session. */\n resumed?: (shard: Shard) => unknown\n /** Discord has requested the Shard to reconnect. */\n requestedReconnect?: (shard: Shard) => unknown\n /** The shard started to connect to Discord's gateway. */\n connecting?: (shard: Shard) => unknown\n /** The shard is connected with Discord's gateway. */\n connected?: (shard: Shard) => unknown\n /** The shard has been disconnected from Discord's gateway. */\n disconnected?: (shard: Shard) => unknown\n /** The shard has started to identify itself to Discord. */\n identifying?: (shard: Shard) => unknown\n /** The shard has successfully been identified itself with Discord. */\n identified?: (shard: Shard) => unknown\n /** The shard has received a message from Discord. */\n message?: (shard: Shard, payload: Camelize<DiscordGatewayPayload>) => unknown\n}\n\nexport enum ShardSocketCloseCodes {\n /** A regular Shard shutdown. */\n Shutdown = 3000,\n /** A resume has been requested and therefore the old connection needs to be closed. */\n ResumeClosingOldConnection = 3024,\n /** Did not receive a heartbeat ACK in time.\n * Closing the shard and creating a new session.\n */\n ZombiedConnection = 3010,\n /** Discordeno's gateway tests hae been finished, therefore the Shard can be turned off. */\n TestingFinished = 3064,\n /** Special close code reserved for Discordeno's zero-downtime resharding system. */\n Resharded = 3065,\n /** Shard is re-identifying therefore the old connection needs to be closed. */\n ReIdentifying = 3066,\n}\n\nexport interface ShardSocketRequest {\n /** The OP-Code for the payload to send. */\n op: GatewayOpcodes\n /** Payload data. */\n d: unknown\n}\n\n/** https://discord.com/developers/docs/topics/gateway-events#update-presence */\nexport interface BotStatusUpdate {\n // /** Unix time (in milliseconds) of when the client went idle, or null if the client is not idle */\n since: number | null\n /** The user's activities */\n activities: BotActivity[]\n /** The user's new status */\n status: keyof typeof PresenceStatus\n}\n\n/** https://discord.com/developers/docs/topics/gateway-events#activity-object */\nexport interface BotActivity {\n name: string\n type: ActivityTypes\n url?: string\n}\n\n/** https://discord.com/developers/docs/topics/gateway#update-voice-state */\nexport interface UpdateVoiceState {\n /** id of the guild */\n guildId: string\n /** id of the voice channel client wants to join (null if disconnecting) */\n channelId: string | null\n /** Is the client muted */\n selfMute: boolean\n /** Is the client deafened */\n selfDeaf: boolean\n}\n\n/** https://discord.com/developers/docs/topics/gateway-events#update-presence */\nexport interface StatusUpdate {\n // /** Unix time (in milliseconds) of when the client went idle, or null if the client is not idle */\n // since: number | null;\n /** The user's activities */\n activities?: Camelize<Array<Omit<DiscordActivity, 'created_at'>>>\n /** The user's new status */\n status: keyof typeof PresenceStatus\n // /** Whether or not the client is afk */\n // afk: boolean;\n}\n"],"names":["ShardState","Connected","Connecting","Disconnected","Unidentified","Identifying","Resuming","Offline","ShardSocketCloseCodes","Shutdown","ResumeClosingOldConnection","ZombiedConnection","TestingFinished","Resharded","ReIdentifying"],"mappings":"WAGO;UAAKA,UAAU;IAAVA,WAAAA,WACV,+EAA+E,GAC/EC,eAAY,KAAZA;IAFUD,WAAAA,WAGV,2HAA2H,GAC3HE,gBAAa,KAAbA;IAJUF,WAAAA,WAKV,uEAAuE,GACvEG,kBAAe,KAAfA;IANUH,WAAAA,WAOV,+HAA+H,GAC/HI,kBAAe,KAAfA;IARUJ,WAAAA,WASV,0EAA0E,GAC1EK,iBAAc,KAAdA;IAVUL,WAAAA,WAWV,0DAA0D,GAC1DM,cAAW,KAAXA;IAZUN,WAAAA,WAaV,kHAAkH,GAClHO,aAAU,KAAVA;GAdUP,eAAAA;WAkHL;UAAKQ,qBAAqB;IAArBA,sBAAAA,sBACV,8BAA8B,GAC9BC,cAAW,QAAXA;IAFUD,sBAAAA,sBAGV,qFAAqF,GACrFE,gCAA6B,QAA7BA;IAJUF,sBAAAA,sBAKV;;GAEC,GACDG,uBAAoB,QAApBA;IARUH,sBAAAA,sBASV,yFAAyF,GACzFI,qBAAkB,QAAlBA;IAVUJ,sBAAAA,sBAWV,kFAAkF,GAClFK,eAAY,QAAZA;IAZUL,sBAAAA,sBAaV,6EAA6E,GAC7EM,mBAAgB,QAAhBA;GAdUN,0BAAAA"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@discordeno/gateway",
3
- "version": "19.0.0-next.bb85624",
3
+ "version": "19.0.0-next.bccbc73",
4
4
  "main": "./dist/index.js",
5
5
  "types": "./dist/index.d.ts",
6
6
  "type": "module",
@@ -24,27 +24,27 @@
24
24
  "test:test-type": "tsc --project tsconfig.test.json"
25
25
  },
26
26
  "dependencies": {
27
- "@discordeno/types": "19.0.0-next.bb85624",
28
- "@discordeno/utils": "19.0.0-next.bb85624",
29
- "ws": "^8.11.0"
27
+ "@discordeno/types": "19.0.0-next.bccbc73",
28
+ "@discordeno/utils": "19.0.0-next.bccbc73",
29
+ "ws": "^8.13.0"
30
30
  },
31
31
  "devDependencies": {
32
- "@swc/cli": "^0.1.57",
33
- "@swc/core": "^1.3.21",
34
- "@types/chai": "^4",
35
- "@types/mocha": "^10",
36
- "@types/node": "^18.11.9",
32
+ "@swc/cli": "^0.1.62",
33
+ "@swc/core": "^1.3.40",
34
+ "@types/chai": "^4.3.4",
35
+ "@types/mocha": "^10.0.1",
36
+ "@types/node": "^18.15.3",
37
37
  "@types/sinon": "^10.0.13",
38
- "@types/ws": "^8.5.3",
39
- "c8": "^7.12.0",
38
+ "@types/ws": "^8.5.4",
39
+ "c8": "^7.13.0",
40
40
  "chai": "^4.3.7",
41
- "eslint": "^8.0.1",
41
+ "eslint": "^8.36.0",
42
42
  "eslint-config-discordeno": "*",
43
- "mocha": "^10.1.0",
44
- "sinon": "^15.0.0",
43
+ "mocha": "^10.2.0",
44
+ "sinon": "^15.0.2",
45
45
  "ts-node": "^10.9.1",
46
46
  "tsconfig": "*",
47
- "typescript": "^4.9.3",
48
- "uWebSockets.js": "uNetworking/uWebSockets.js#v20.19.0"
47
+ "typescript": "^4.9.5",
48
+ "uWebSockets.js": "https://github.com/uNetworking/uWebSockets.js.git#commit=42c9c0d5d31f46ca4115dc75672b0037ec970f28"
49
49
  }
50
50
  }