@discordeno/gateway 21.0.1-next.b6bfaa3 → 21.0.1-next.d9831f6

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.
@@ -65,8 +65,7 @@ function createGatewayManager(options) {
65
65
  enabled: options.resharding?.enabled ?? true,
66
66
  shardsFullPercentage: options.resharding?.shardsFullPercentage ?? 80,
67
67
  checkInterval: options.resharding?.checkInterval ?? 28800000,
68
- shards: new _utils.Collection(),
69
- pendingShards: new _utils.Collection(),
68
+ shards: new Map(),
70
69
  getSessionInfo: options.resharding?.getSessionInfo,
71
70
  updateGuildsShardId: options.resharding?.updateGuildsShardId,
72
71
  async checkIfReshardingIsNeeded () {
@@ -122,19 +121,23 @@ function createGatewayManager(options) {
122
121
  if (typeof info.firstShardId === 'number') gateway.firstShardId = info.firstShardId;
123
122
  // Set last shard id if provided in info
124
123
  if (typeof info.lastShardId === 'number') gateway.lastShardId = info.lastShardId;
124
+ else gateway.lastShardId = gateway.totalShards - 1;
125
125
  gateway.logger.info(`[Resharding] Starting the reshard process. New total shards: ${gateway.totalShards}`);
126
126
  // Resetting buckets
127
127
  gateway.buckets.clear();
128
128
  // Refilling buckets with new values
129
129
  gateway.prepareBuckets();
130
- // SPREAD THIS OUT TO DIFFERENT WORKERS TO BEGIN STARTING UP
131
- gateway.buckets.forEach(async (bucket, bucketId)=>{
130
+ // Call all the buckets and tell their workers & shards to identify
131
+ const promises = Array.from(gateway.buckets.entries()).map(async ([bucketId, bucket])=>{
132
132
  for (const worker of bucket.workers){
133
133
  for (const shardId of worker.queue){
134
134
  await gateway.resharding.tellWorkerToPrepare(worker.id, shardId, bucketId);
135
135
  }
136
136
  }
137
137
  });
138
+ await Promise.all(promises);
139
+ gateway.logger.info(`[Resharding] All shards are now online.`);
140
+ await gateway.resharding.onReshardingSwitch();
138
141
  },
139
142
  async tellWorkerToPrepare (workerId, shardId, bucketId) {
140
143
  gateway.logger.debug(`[Resharding] Telling worker to prepare. Worker: ${workerId} | Shard: ${shardId} | Bucket: ${bucketId}.`);
@@ -150,9 +153,9 @@ function createGatewayManager(options) {
150
153
  url: gateway.url,
151
154
  version: gateway.version
152
155
  },
153
- // Ignore events until we are ready
154
156
  events: {
155
157
  async message (_shard, payload) {
158
+ // Ignore all events until we swich from the old shards to the new ones.
156
159
  if (payload.t === 'READY') {
157
160
  await gateway.resharding.updateGuildsShardId?.(payload.d.guilds.map((g)=>g.id), shardId);
158
161
  }
@@ -168,22 +171,14 @@ function createGatewayManager(options) {
168
171
  };
169
172
  }
170
173
  gateway.resharding.shards.set(shardId, shard);
171
- const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency);
172
- if (!bucket) return;
173
- await shard.identify().then(()=>gateway.resharding.shardIsPending(shard));
174
+ await shard.identify();
175
+ gateway.logger.debug(`[Resharding] Shard #${shardId} identified.`);
174
176
  },
175
- async shardIsPending (shard) {
176
- // Save this in pending at the moment, until all shards are online
177
- gateway.resharding.pendingShards.set(shard.id, shard);
178
- gateway.logger.debug(`[Resharding] Shard #${shard.id} is now pending.`);
179
- // Check if all shards are now online.
180
- if (gateway.lastShardId - gateway.firstShardId >= gateway.resharding.pendingShards.size) return;
181
- gateway.logger.info(`[Resharding] All shards are now online.`);
182
- // New shards start processing events
177
+ async onReshardingSwitch () {
178
+ gateway.logger.debug(`[Resharding] Making the switch from the old shards to the new ones.`);
179
+ // Move the events from the old shards to the new ones
183
180
  for (const shard of gateway.resharding.shards.values()){
184
- for(const event in options.events){
185
- shard.events[event] = options.events[event];
186
- }
181
+ shard.events = options.events ?? {};
187
182
  }
188
183
  // Old shards stop processing events
189
184
  for (const shard of gateway.shards.values()){
@@ -200,14 +195,10 @@ function createGatewayManager(options) {
200
195
  };
201
196
  }
202
197
  gateway.logger.info(`[Resharding] Shutting down old shards.`);
203
- // Close old shards
204
198
  await gateway.shutdown(_types1.ShardSocketCloseCodes.Resharded, 'Resharded!', false);
205
199
  gateway.logger.info(`[Resharding] Completed.`);
206
- // Replace old shards
207
- gateway.shards = new _utils.Collection(gateway.resharding.shards);
208
- // Clear our collections and keep only one reference to the shards, the one in gateway.shards
200
+ gateway.shards = new Map(gateway.resharding.shards);
209
201
  gateway.resharding.shards.clear();
210
- gateway.resharding.pendingShards.clear();
211
202
  }
212
203
  },
213
204
  calculateTotalShards () {
@@ -484,4 +475,4 @@ function createGatewayManager(options) {
484
475
  return gateway;
485
476
  }
486
477
 
487
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/manager.ts"],"sourcesContent":["import { randomBytes } from 'node:crypto'\nimport {\n  type AtLeastOne,\n  type BigString,\n  type Camelize,\n  type DiscordGetGatewayBot,\n  type DiscordMemberWithUser,\n  type DiscordReady,\n  type DiscordUpdatePresence,\n  GatewayIntents,\n  GatewayOpcodes,\n  type RequestGuildMembers,\n} from '@discordeno/types'\nimport { Collection, LeakyBucket, jsonSafeReplacer, logger } from '@discordeno/utils'\nimport Shard from './Shard.js'\nimport { type ShardEvents, ShardSocketCloseCodes, type ShardSocketRequest, type TransportCompression, type 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    transportCompression: options.transportCompression ?? null,\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    spreadShardsInRoundRobin: options.spreadShardsInRoundRobin ?? false,\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    logger: options.logger ?? logger,\n    makePresence: options.makePresence ?? (() => Promise.resolve(undefined)),\n    resharding: {\n      enabled: options.resharding?.enabled ?? true,\n      shardsFullPercentage: options.resharding?.shardsFullPercentage ?? 80,\n      checkInterval: options.resharding?.checkInterval ?? 28800000, // 8 hours\n      shards: new Collection(),\n      pendingShards: new Collection(),\n      getSessionInfo: options.resharding?.getSessionInfo,\n      updateGuildsShardId: options.resharding?.updateGuildsShardId,\n      async checkIfReshardingIsNeeded() {\n        gateway.logger.debug('[Resharding] Checking if resharding is needed.')\n\n        if (!gateway.resharding.enabled) {\n          gateway.logger.debug('[Resharding] Resharding is disabled.')\n\n          return { needed: false }\n        }\n\n        if (!gateway.resharding.getSessionInfo) {\n          throw new Error(\"[Resharding] Resharding is enabled but no 'resharding.getSessionInfo()' is not provided.\")\n        }\n\n        gateway.logger.debug('[Resharding] Resharding is enabled.')\n\n        const sessionInfo = await gateway.resharding.getSessionInfo()\n\n        gateway.logger.debug(`[Resharding] Session info retrieved: ${JSON.stringify(sessionInfo)}`)\n\n        // Don't have enough identify limits to try resharding\n        if (sessionInfo.sessionStartLimit.remaining < sessionInfo.shards) {\n          gateway.logger.debug('[Resharding] Not enough session start limits left to reshard.')\n\n          return { needed: false, info: sessionInfo }\n        }\n\n        gateway.logger.debug('[Resharding] Able to reshard, checking whether necessary now.')\n\n        // 2500 is the max amount of guilds a single shard can handle\n        // 1000 is the amount of guilds discord uses to determine how many shards to recommend.\n        // This algo helps check if your bot has grown enough to reshard.\n        // While this is imprecise as discord changes the recommended number of shard every 1000 guilds it is good enough\n        // The alternative is to store the guild count for each shard and require the Guilds intent for `GUILD_CREATE` and `GUILD_DELETE` events\n        const percentage = (sessionInfo.shards / ((gateway.totalShards * 2500) / 1000)) * 100\n\n        // Less than necessary% being used so do nothing\n        if (percentage < gateway.resharding.shardsFullPercentage) {\n          gateway.logger.debug('[Resharding] Resharding not needed.')\n\n          return { needed: false, info: sessionInfo }\n        }\n\n        gateway.logger.info('[Resharding] Resharding is needed.')\n\n        return { needed: true, info: sessionInfo }\n      },\n      async reshard(info) {\n        gateway.logger.info(`[Resharding] Starting the reshard process. Previous total shards: ${gateway.totalShards}`)\n        // Set values on gateway\n        gateway.totalShards = info.shards\n        // Handles preparing mid sized bots for LBS\n        gateway.totalShards = gateway.calculateTotalShards()\n        // Set first shard id if provided in info\n        if (typeof info.firstShardId === 'number') gateway.firstShardId = info.firstShardId\n        // Set last shard id if provided in info\n        if (typeof info.lastShardId === 'number') gateway.lastShardId = info.lastShardId\n        gateway.logger.info(`[Resharding] Starting the reshard process. New total shards: ${gateway.totalShards}`)\n\n        // Resetting buckets\n        gateway.buckets.clear()\n        // Refilling buckets with new values\n        gateway.prepareBuckets()\n\n        // SPREAD THIS OUT TO DIFFERENT WORKERS TO BEGIN STARTING UP\n        gateway.buckets.forEach(async (bucket, bucketId) => {\n          for (const worker of bucket.workers) {\n            for (const shardId of worker.queue) {\n              await gateway.resharding.tellWorkerToPrepare(worker.id, shardId, bucketId)\n            }\n          }\n        })\n      },\n      async tellWorkerToPrepare(workerId, shardId, bucketId) {\n        gateway.logger.debug(`[Resharding] Telling worker to prepare. Worker: ${workerId} | Shard: ${shardId} | Bucket: ${bucketId}.`)\n        const shard = new Shard({\n          id: shardId,\n          connection: {\n            compress: gateway.compress,\n            transportCompression: gateway.transportCompression ?? null,\n            intents: gateway.intents,\n            properties: gateway.properties,\n            token: gateway.token,\n            totalShards: gateway.totalShards,\n            url: gateway.url,\n            version: gateway.version,\n          },\n          // Ignore events until we are ready\n          events: {\n            async message(_shard, payload) {\n              if (payload.t === 'READY') {\n                await gateway.resharding.updateGuildsShardId?.(\n                  (payload.d as DiscordReady).guilds.map((g) => g.id),\n                  shardId,\n                )\n              }\n            },\n          },\n          logger: gateway.logger,\n          requestIdentify: async () => await gateway.requestIdentify(shardId),\n          makePresence: gateway.makePresence,\n        })\n\n        if (gateway.preferSnakeCase) {\n          shard.forwardToBot = async (payload) => {\n            shard.events?.message?.(shard, payload)\n          }\n        }\n\n        gateway.resharding.shards.set(shardId, shard)\n\n        const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)\n        if (!bucket) return\n\n        await shard.identify().then(() => gateway.resharding.shardIsPending(shard))\n      },\n      async shardIsPending(shard) {\n        // Save this in pending at the moment, until all shards are online\n        gateway.resharding.pendingShards.set(shard.id, shard)\n        gateway.logger.debug(`[Resharding] Shard #${shard.id} is now pending.`)\n\n        // Check if all shards are now online.\n        if (gateway.lastShardId - gateway.firstShardId >= gateway.resharding.pendingShards.size) return\n\n        gateway.logger.info(`[Resharding] All shards are now online.`)\n\n        // New shards start processing events\n        for (const shard of gateway.resharding.shards.values()) {\n          for (const event in options.events) {\n            shard.events[event as keyof ShardEvents] = options.events[event as keyof ShardEvents] as (...args: unknown[]) => unknown\n          }\n        }\n\n        // Old shards stop processing events\n        for (const shard of gateway.shards.values()) {\n          const oldHandler = shard.events.message\n\n          // Change with spread operator to not affect new shards, as changing anything on shard.events will directly change options.events, which changes new shards' events\n          shard.events = {\n            ...shard.events,\n            message: async function (_, message) {\n              // Member checks need to continue but others can stop\n              if (message.t === 'GUILD_MEMBERS_CHUNK') {\n                oldHandler?.(shard, message)\n              }\n            },\n          }\n        }\n\n        gateway.logger.info(`[Resharding] Shutting down old shards.`)\n        // Close old shards\n        await gateway.shutdown(ShardSocketCloseCodes.Resharded, 'Resharded!', false)\n\n        gateway.logger.info(`[Resharding] Completed.`)\n\n        // Replace old shards\n        gateway.shards = new Collection(gateway.resharding.shards)\n\n        // Clear our collections and keep only one reference to the shards, the one in gateway.shards\n        gateway.resharding.shards.clear()\n        gateway.resharding.pendingShards.clear()\n      },\n    },\n\n    calculateTotalShards() {\n      // Bots under 100k servers do not have access to LBS.\n      if (gateway.totalShards < 100) {\n        gateway.logger.debug(`[Gateway] Calculating total shards: ${gateway.totalShards}`)\n        return gateway.totalShards\n      }\n\n      gateway.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 to get `totalShards` to be in a multiple of 16 so that we can prepare bots with 100k servers for LBS.\n            (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency),\n        ) * (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency)\n      )\n    },\n    calculateWorkerId(shardId) {\n      const workerId = options.spreadShardsInRoundRobin\n        ? shardId % gateway.totalWorkers\n        : Math.min(Math.floor(shardId / gateway.shardsPerWorker), gateway.totalWorkers - 1)\n      gateway.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        gateway.logger.debug(`[Gateway] Preparing buckets for concurrency: ${i}`)\n        gateway.buckets.set(i, {\n          workers: [],\n          leakyBucket: new LeakyBucket({\n            max: 1,\n            refillAmount: 1,\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        gateway.logger.debug(`[Gateway] Preparing buckets for shard: ${shardId}`)\n\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\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        // Get the worker id for this shard\n        const workerId = gateway.calculateWorkerId(shardId)\n        const worker = bucket.workers.find((w) => w.id === workerId)\n\n        // If this worker already exists, add the shard to its queue\n        if (worker) {\n          worker.queue.push(shardId)\n        } else {\n          bucket.workers.push({ id: workerId, queue: [shardId] })\n        }\n      }\n    },\n    async spawnShards() {\n      // Prepare the concurrency buckets\n      gateway.prepareBuckets()\n\n      const promises = [...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      // We use Promise.all so we can start all buckets at the same time\n      await Promise.all(promises)\n\n      // Check and reshard automatically if auto resharding is enabled.\n      if (gateway.resharding.enabled && gateway.resharding.checkInterval !== -1) {\n        // It is better to ensure there is always only one\n        clearInterval(gateway.resharding.checkIntervalId)\n\n        if (!gateway.resharding.getSessionInfo) {\n          gateway.resharding.enabled = false\n          gateway.logger.warn(\"[Resharding] Resharding is enabled but 'resharding.getSessionInfo()' was not provided. Disabling resharding.\")\n\n          return\n        }\n\n        gateway.resharding.checkIntervalId = setInterval(async () => {\n          const reshardingInfo = await gateway.resharding.checkIfReshardingIsNeeded()\n\n          if (reshardingInfo.needed && reshardingInfo.info) await gateway.resharding.reshard(reshardingInfo.info)\n        }, gateway.resharding.checkInterval)\n      }\n    },\n    async shutdown(code, reason, clearReshardingInterval = true) {\n      if (clearReshardingInterval) clearInterval(gateway.resharding.checkIntervalId)\n\n      await Promise.all(Array.from(gateway.shards.values()).map((shard) => shard.close(code, reason)))\n    },\n    async sendPayload(shardId, payload) {\n      const shard = gateway.shards.get(shardId)\n\n      if (!shard) {\n        throw new Error(`Shard (id: ${shardId} not found`)\n      }\n\n      await shard.send(payload)\n    },\n    async tellWorkerToIdentify(workerId, shardId, bucketId) {\n      gateway.logger.debug(`[Gateway] Tell worker #${workerId} to identify shard #${shardId} from bucket ${bucketId}`)\n      await gateway.identify(shardId)\n    },\n    async identify(shardId: number) {\n      let shard = this.shards.get(shardId)\n      gateway.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            transportCompression: gateway.transportCompression,\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          logger: this.logger,\n          requestIdentify: async () => await gateway.requestIdentify(shardId),\n          makePresence: gateway.makePresence,\n        })\n\n        if (this.preferSnakeCase) {\n          shard.forwardToBot = async (payload) => {\n            shard!.events.message?.(shard!, payload)\n          }\n        }\n\n        this.shards.set(shardId, shard)\n      }\n\n      await shard.identify()\n    },\n\n    async requestIdentify(shardId) {\n      gateway.logger.debug(`[Gateway] Shard #${shardId} requested an identify.`)\n\n      const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)\n\n      if (!bucket) {\n        throw new Error(\"Can't request identify for a shard that is not assigned to any bucket.\")\n      }\n\n      await bucket.leakyBucket.acquire()\n\n      gateway.logger.debug(`[Gateway] Approved identify request for Shard #${shardId}.`)\n    },\n\n    async kill(shardId: number) {\n      const shard = this.shards.get(shardId)\n      if (!shard) {\n        gateway.logger.debug(`[Gateway] Shard #${shardId} was requested to be killed, but the shard could not be found.`)\n        return\n      }\n\n      gateway.logger.debug(`[Gateway] Killing Shard #${shardId}`)\n      this.shards.delete(shardId)\n      await shard.shutdown()\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        gateway.logger.debug(`[Gateway] calculateShardId (1 shard)`)\n        return 0\n      }\n\n      gateway.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\n      gateway.logger.debug(`[Gateway] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`)\n\n      await gateway.sendPayload(shardId, {\n        op: GatewayOpcodes.VoiceStateUpdate,\n        d: {\n          guild_id: guildId.toString(),\n          channel_id: channelId.toString(),\n          self_mute: options?.selfMute ?? false,\n          self_deaf: options?.selfDeaf ?? true,\n        },\n      })\n    },\n\n    async editBotStatus(data) {\n      gateway.logger.debug(`[Gateway] editBotStatus data: ${JSON.stringify(data, jsonSafeReplacer)}`)\n\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      gateway.logger.debug(`[Gateway] editShardStatus shardId: ${shardId} -> data: ${JSON.stringify(data)}`)\n\n      await gateway.sendPayload(shardId, {\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    async requestMembers(guildId, options) {\n      const shardId = gateway.calculateShardId(guildId)\n\n      if (gateway.intents && (!options?.limit || options.limit > 1) && !(gateway.intents & GatewayIntents.GuildMembers))\n        throw new Error('Cannot fetch more then 1 member without the GUILD_MEMBERS intent')\n\n      gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> data: ${JSON.stringify(options)}`)\n\n      if (options?.userIds?.length) {\n        gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> setting user limit based on userIds length: ${options.userIds.length}`)\n\n        options.limit = options.userIds.length\n      }\n\n      if (!options?.nonce) {\n        let nonce = ''\n\n        while (!nonce || gateway.cache.requestMembers.pending.has(nonce)) {\n          nonce = randomBytes(16).toString('hex')\n        }\n\n        options ??= { limit: 0 }\n        options.nonce = nonce\n      }\n\n      const members = !gateway.cache.requestMembers.enabled\n        ? []\n        : new Promise<Camelize<DiscordMemberWithUser[]>>((resolve, reject) => {\n            // Should never happen.\n            if (!gateway.cache.requestMembers.enabled || !options?.nonce) {\n              reject(new Error(\"Can't request the members without the nonce or with the feature disabled.\"))\n              return\n            }\n\n            gateway.cache.requestMembers.pending.set(options.nonce, {\n              nonce: options.nonce,\n              resolve,\n              members: [],\n            })\n          })\n\n      await gateway.sendPayload(shardId, {\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      return await members\n    },\n\n    async leaveVoiceChannel(guildId) {\n      const shardId = gateway.calculateShardId(guildId)\n\n      gateway.logger.debug(`[Gateway] leaveVoiceChannel guildId: ${guildId} Shard ${shardId}`)\n\n      await gateway.sendPayload(shardId, {\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    async requestSoundboardSounds(guildIds) {\n      /**\n       * Discord will send the events for the guilds that are \"under the shard\" that sends the opcode.\n       * For this reason we need to group the ids with the shard the calculateShardId method gives\n       */\n\n      const map = new Map<number, BigString[]>()\n\n      for (const guildId of guildIds) {\n        const shardId = gateway.calculateShardId(guildId)\n\n        const ids = map.get(shardId) ?? []\n        map.set(shardId, ids)\n\n        ids.push(guildId)\n      }\n\n      await Promise.all(\n        [...map.entries()].map(([shardId, ids]) =>\n          gateway.sendPayload(shardId, {\n            op: GatewayOpcodes.RequestSoundboardSounds,\n            d: {\n              guild_ids: ids,\n            },\n          }),\n        ),\n      )\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  /**\n   * Whether to spread shards across workers in a round-robin manner.\n   *\n   * @remarks\n   * By default, shards are assigned to workers in contiguous blocks based on shardsPerWorker. If any shard is left over, it will be assigned to the last worker.\n   * This means that if you have 3 workers and 40 shards while shardsPerWorker is 10, the first worker will get shards 0-9, the second worker will get shards 10-19, and so on, the last worker will get shards 20-39.\n   *\n   * If this option is set to true, the shards will be assigned in a round-robin manner to spread more evenly across workers.\n   * For example, with 3 workers and 40 shards, the first shard will go to worker 0, the second shard to worker 1, the third shard to worker 2, the fourth shard to worker 0, and so on.\n   *\n   * @default false\n   */\n  spreadShardsInRoundRobin?: boolean\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  /** What transport compression should be used */\n  transportCompression?: TransportCompression | null\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    }\n  }\n  /**\n   * The logger that the gateway manager will use.\n   * @default logger // The logger exported by `@discordeno/utils`\n   */\n  logger?: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>\n  /**\n   * Make the presence for when the bot connects to the gateway\n   *\n   * @remarks\n   * This function will be called each time a Shard is going to identify\n   */\n  makePresence?: () => Promise<DiscordUpdatePresence | undefined>\n  /** Options related to resharding. */\n  resharding?: {\n    /**\n     * Whether or not automated resharding should be enabled.\n     * @default true\n     */\n    enabled: boolean\n    /**\n     * The % of how full a shard is when resharding should be triggered.\n     *\n     * @remarks\n     * We use discord recommended shard value to get an **approximation** of the shard full percentage to compare with this value so the bot may not reshard at the exact percentage provided but may reshard when it is a bit higher than the provided percentage.\n     * For accurate calculation, you may override the `checkIfReshardingIsNeeded` function\n     *\n     * @default 80 as in 80%\n     */\n    shardsFullPercentage: number\n    /**\n     * The interval in milliseconds, of how often to check whether resharding is needed and reshard automatically. Set to -1 to disable auto resharding.\n     * @default 28800000 (8 hours)\n     */\n    checkInterval: number\n    /** Handler to get shard count and other session info. */\n    getSessionInfo?: () => Promise<Camelize<DiscordGetGatewayBot>>\n    /** Handler to edit the shard id on any cached guilds. */\n    updateGuildsShardId?: (guildIds: string[], shardId: number) => Promise<void>\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      /** The bucket to queue the identifies. */\n      leakyBucket: LeakyBucket\n    }\n  >\n  /** The shards that are created. */\n  shards: Map<number, Shard>\n  /** The logger for the gateway manager. */\n  logger: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>\n  /** Everything related to resharding. */\n  resharding: CreateGatewayManagerOptions['resharding'] & {\n    /**\n     * The interval id of the check interval. This is used to clear the interval when the manager is shutdown.\n     */\n    checkIntervalId?: NodeJS.Timeout | undefined\n    /** Holds the shards that resharding has created. Once resharding is done, this replaces the gateway.shards */\n    shards: Collection<number, Shard>\n    /** Holds the pending shards that have been created and are pending all shards finish loading. */\n    pendingShards: Collection<number, Shard>\n    /** Handler to check if resharding is necessary. */\n    checkIfReshardingIsNeeded: () => Promise<{ needed: boolean; info?: Camelize<DiscordGetGatewayBot> }>\n    /** Handler to begin resharding. */\n    reshard: (info: Camelize<DiscordGetGatewayBot> & { firstShardId?: number; lastShardId?: number }) => Promise<void>\n    /** Handler to communicate to a worker that a shard needs to be created. */\n    tellWorkerToPrepare: (workerId: number, shardId: number, bucketId: number) => Promise<void>\n    /** Handler to alert the gateway that a shard(resharded) is online. It should now wait for all shards to be pending before shutting off old shards. */\n    shardIsPending: (shard: Shard) => Promise<void>\n  }\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, clearReshardingInterval?: boolean) => Promise<void>\n  sendPayload: (shardId: number, payload: ShardSocketRequest) => Promise<void>\n  /**\n   * Allows users to hook in and change to communicate to different workers across different servers or anything they like.\n   * For example using redis pubsub to talk to other servers.\n   *\n   * @remarks\n   * This should wait for the worker to have identified the shard before resolving the returned promise\n   */\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 nothing\n   */\n  editBotStatus: (data: DiscordUpdatePresence) => 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 nothing\n   */\n  editShardStatus: (shardId: number, data: DiscordUpdatePresence) => Promise<void>\n  /**\n   * Fetches the list of members for a guild over the gateway. If `gateway.cache.requestMembers.enabled` is not set, this function will return an empty array and you'll have to handle the `GUILD_MEMBERS_CHUNK` events yourself.\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<DiscordMemberWithUser[]>>\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   * Used to request soundboard sounds for a list of guilds.\n   *\n   * This function sends multiple (see remarks) _Request Soundboard Sounds_ gateway command over the gateway behind the scenes.\n   *\n   * @param guildIds - The guilds to get the sounds from\n   *\n   * @remarks\n   * Fires a _Soundboard Sounds_ gateway event.\n   *\n   * ⚠️ Discord will send the _Soundboard Sounds_ for each of the guild ids\n   * however you may not receive the same number of events as the ids passed to _Request Soundboard Sounds_ for one of the following reasons:\n   * - The bot is not in the server provided\n   * - The shard the message has been sent from does not receive events for the specified guild\n   *\n   * To avoid this Discordeno will automatically try to group the ids based on what shard they will need to be sent, but this involves sending multiple messages in multiple shards\n   *\n   * @see {@link https://discord.com/developers/docs/topics/gateway-events#request-soundboard-sounds}\n   */\n  requestSoundboardSounds: (guildIds: BigString[]) => Promise<void>\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 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<DiscordMemberWithUser[]> | PromiseLike<Camelize<DiscordMemberWithUser[]>>) => void\n  /** The members that have already arrived for this request. */\n  members: DiscordMemberWithUser[]\n}\n"],"names":["createGatewayManager","options","connectionOptions","connection","url","shards","sessionStartLimit","maxConcurrency","remaining","total","resetAfter","gateway","events","compress","transportCompression","intents","properties","os","process","platform","browser","device","token","version","totalShards","lastShardId","firstShardId","totalWorkers","shardsPerWorker","spawnShardDelay","spreadShardsInRoundRobin","preferSnakeCase","Map","buckets","cache","requestMembers","enabled","pending","Collection","logger","makePresence","Promise","resolve","undefined","resharding","shardsFullPercentage","checkInterval","pendingShards","getSessionInfo","updateGuildsShardId","checkIfReshardingIsNeeded","debug","needed","Error","sessionInfo","JSON","stringify","info","percentage","reshard","calculateTotalShards","clear","prepareBuckets","forEach","bucket","bucketId","worker","workers","shardId","queue","tellWorkerToPrepare","id","workerId","shard","Shard","message","_shard","payload","t","d","guilds","map","g","requestIdentify","forwardToBot","set","get","identify","then","shardIsPending","size","values","event","oldHandler","_","shutdown","ShardSocketCloseCodes","Resharded","Math","ceil","calculateWorkerId","min","floor","i","leakyBucket","LeakyBucket","max","refillAmount","refillInterval","find","w","push","spawnShards","promises","entries","tellWorkerToIdentify","all","clearInterval","checkIntervalId","warn","setInterval","reshardingInfo","code","reason","clearReshardingInterval","Array","from","close","sendPayload","send","acquire","kill","delete","calculateShardId","guildId","Number","BigInt","joinVoiceChannel","channelId","op","GatewayOpcodes","VoiceStateUpdate","guild_id","toString","channel_id","self_mute","selfMute","self_deaf","selfDeaf","editBotStatus","data","jsonSafeReplacer","editShardStatus","PresenceUpdate","since","afk","activities","status","limit","GatewayIntents","GuildMembers","userIds","length","nonce","has","randomBytes","members","reject","RequestGuildMembers","query","presences","user_ids","leaveVoiceChannel","requestSoundboardSounds","guildIds","ids","RequestSoundboardSounds","guild_ids"],"mappings":"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"}
478
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/manager.ts"],"sourcesContent":["import { randomBytes } from 'node:crypto'\nimport {\n  type AtLeastOne,\n  type BigString,\n  type Camelize,\n  type DiscordGetGatewayBot,\n  type DiscordMemberWithUser,\n  type DiscordReady,\n  type DiscordUpdatePresence,\n  GatewayIntents,\n  GatewayOpcodes,\n  type RequestGuildMembers,\n} from '@discordeno/types'\nimport { Collection, LeakyBucket, jsonSafeReplacer, logger } from '@discordeno/utils'\nimport Shard from './Shard.js'\nimport { type ShardEvents, ShardSocketCloseCodes, type ShardSocketRequest, type TransportCompression, type 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    transportCompression: options.transportCompression ?? null,\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    spreadShardsInRoundRobin: options.spreadShardsInRoundRobin ?? false,\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    logger: options.logger ?? logger,\n    makePresence: options.makePresence ?? (() => Promise.resolve(undefined)),\n    resharding: {\n      enabled: options.resharding?.enabled ?? true,\n      shardsFullPercentage: options.resharding?.shardsFullPercentage ?? 80,\n      checkInterval: options.resharding?.checkInterval ?? 28800000, // 8 hours\n      shards: new Map(),\n      getSessionInfo: options.resharding?.getSessionInfo,\n      updateGuildsShardId: options.resharding?.updateGuildsShardId,\n      async checkIfReshardingIsNeeded() {\n        gateway.logger.debug('[Resharding] Checking if resharding is needed.')\n\n        if (!gateway.resharding.enabled) {\n          gateway.logger.debug('[Resharding] Resharding is disabled.')\n\n          return { needed: false }\n        }\n\n        if (!gateway.resharding.getSessionInfo) {\n          throw new Error(\"[Resharding] Resharding is enabled but no 'resharding.getSessionInfo()' is not provided.\")\n        }\n\n        gateway.logger.debug('[Resharding] Resharding is enabled.')\n\n        const sessionInfo = await gateway.resharding.getSessionInfo()\n\n        gateway.logger.debug(`[Resharding] Session info retrieved: ${JSON.stringify(sessionInfo)}`)\n\n        // Don't have enough identify limits to try resharding\n        if (sessionInfo.sessionStartLimit.remaining < sessionInfo.shards) {\n          gateway.logger.debug('[Resharding] Not enough session start limits left to reshard.')\n\n          return { needed: false, info: sessionInfo }\n        }\n\n        gateway.logger.debug('[Resharding] Able to reshard, checking whether necessary now.')\n\n        // 2500 is the max amount of guilds a single shard can handle\n        // 1000 is the amount of guilds discord uses to determine how many shards to recommend.\n        // This algo helps check if your bot has grown enough to reshard.\n        // While this is imprecise as discord changes the recommended number of shard every 1000 guilds it is good enough\n        // The alternative is to store the guild count for each shard and require the Guilds intent for `GUILD_CREATE` and `GUILD_DELETE` events\n        const percentage = (sessionInfo.shards / ((gateway.totalShards * 2500) / 1000)) * 100\n\n        // Less than necessary% being used so do nothing\n        if (percentage < gateway.resharding.shardsFullPercentage) {\n          gateway.logger.debug('[Resharding] Resharding not needed.')\n\n          return { needed: false, info: sessionInfo }\n        }\n\n        gateway.logger.info('[Resharding] Resharding is needed.')\n\n        return { needed: true, info: sessionInfo }\n      },\n      async reshard(info) {\n        gateway.logger.info(`[Resharding] Starting the reshard process. Previous total shards: ${gateway.totalShards}`)\n        // Set values on gateway\n        gateway.totalShards = info.shards\n        // Handles preparing mid sized bots for LBS\n        gateway.totalShards = gateway.calculateTotalShards()\n        // Set first shard id if provided in info\n        if (typeof info.firstShardId === 'number') gateway.firstShardId = info.firstShardId\n        // Set last shard id if provided in info\n        if (typeof info.lastShardId === 'number') gateway.lastShardId = info.lastShardId\n        // If we didn't get any lastShardId, we assume all the shards are to be used\n        else gateway.lastShardId = gateway.totalShards - 1\n        gateway.logger.info(`[Resharding] Starting the reshard process. New total shards: ${gateway.totalShards}`)\n\n        // Resetting buckets\n        gateway.buckets.clear()\n        // Refilling buckets with new values\n        gateway.prepareBuckets()\n\n        // Call all the buckets and tell their workers & shards to identify\n        const promises = Array.from(gateway.buckets.entries()).map(async ([bucketId, bucket]) => {\n          for (const worker of bucket.workers) {\n            for (const shardId of worker.queue) {\n              await gateway.resharding.tellWorkerToPrepare(worker.id, shardId, bucketId)\n            }\n          }\n        })\n\n        await Promise.all(promises)\n\n        gateway.logger.info(`[Resharding] All shards are now online.`)\n\n        await gateway.resharding.onReshardingSwitch()\n      },\n      async tellWorkerToPrepare(workerId, shardId, bucketId) {\n        gateway.logger.debug(`[Resharding] Telling worker to prepare. Worker: ${workerId} | Shard: ${shardId} | Bucket: ${bucketId}.`)\n        const shard = new Shard({\n          id: shardId,\n          connection: {\n            compress: gateway.compress,\n            transportCompression: gateway.transportCompression ?? null,\n            intents: gateway.intents,\n            properties: gateway.properties,\n            token: gateway.token,\n            totalShards: gateway.totalShards,\n            url: gateway.url,\n            version: gateway.version,\n          },\n          events: {\n            async message(_shard, payload) {\n              // Ignore all events until we swich from the old shards to the new ones.\n              if (payload.t === 'READY') {\n                await gateway.resharding.updateGuildsShardId?.(\n                  (payload.d as DiscordReady).guilds.map((g) => g.id),\n                  shardId,\n                )\n              }\n            },\n          },\n          logger: gateway.logger,\n          requestIdentify: async () => await gateway.requestIdentify(shardId),\n          makePresence: gateway.makePresence,\n        })\n\n        if (gateway.preferSnakeCase) {\n          shard.forwardToBot = async (payload) => {\n            shard.events?.message?.(shard, payload)\n          }\n        }\n\n        gateway.resharding.shards.set(shardId, shard)\n\n        await shard.identify()\n\n        gateway.logger.debug(`[Resharding] Shard #${shardId} identified.`)\n      },\n      async onReshardingSwitch() {\n        gateway.logger.debug(`[Resharding] Making the switch from the old shards to the new ones.`)\n\n        // Move the events from the old shards to the new ones\n        for (const shard of gateway.resharding.shards.values()) {\n          shard.events = options.events ?? {}\n        }\n\n        // Old shards stop processing events\n        for (const shard of gateway.shards.values()) {\n          const oldHandler = shard.events.message\n\n          // Change with spread operator to not affect new shards, as changing anything on shard.events will directly change options.events, which changes new shards' events\n          shard.events = {\n            ...shard.events,\n            message: async function (_, message) {\n              // Member checks need to continue but others can stop\n              if (message.t === 'GUILD_MEMBERS_CHUNK') {\n                oldHandler?.(shard, message)\n              }\n            },\n          }\n        }\n\n        gateway.logger.info(`[Resharding] Shutting down old shards.`)\n        await gateway.shutdown(ShardSocketCloseCodes.Resharded, 'Resharded!', false)\n\n        gateway.logger.info(`[Resharding] Completed.`)\n        gateway.shards = new Map(gateway.resharding.shards)\n        gateway.resharding.shards.clear()\n      },\n    },\n\n    calculateTotalShards() {\n      // Bots under 100k servers do not have access to LBS.\n      if (gateway.totalShards < 100) {\n        gateway.logger.debug(`[Gateway] Calculating total shards: ${gateway.totalShards}`)\n        return gateway.totalShards\n      }\n\n      gateway.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 to get `totalShards` to be in a multiple of 16 so that we can prepare bots with 100k servers for LBS.\n            (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency),\n        ) * (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency)\n      )\n    },\n    calculateWorkerId(shardId) {\n      const workerId = options.spreadShardsInRoundRobin\n        ? shardId % gateway.totalWorkers\n        : Math.min(Math.floor(shardId / gateway.shardsPerWorker), gateway.totalWorkers - 1)\n      gateway.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        gateway.logger.debug(`[Gateway] Preparing buckets for concurrency: ${i}`)\n        gateway.buckets.set(i, {\n          workers: [],\n          leakyBucket: new LeakyBucket({\n            max: 1,\n            refillAmount: 1,\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        gateway.logger.debug(`[Gateway] Preparing buckets for shard: ${shardId}`)\n\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\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        // Get the worker id for this shard\n        const workerId = gateway.calculateWorkerId(shardId)\n        const worker = bucket.workers.find((w) => w.id === workerId)\n\n        // If this worker already exists, add the shard to its queue\n        if (worker) {\n          worker.queue.push(shardId)\n        } else {\n          bucket.workers.push({ id: workerId, queue: [shardId] })\n        }\n      }\n    },\n    async spawnShards() {\n      // Prepare the concurrency buckets\n      gateway.prepareBuckets()\n\n      const promises = [...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      // We use Promise.all so we can start all buckets at the same time\n      await Promise.all(promises)\n\n      // Check and reshard automatically if auto resharding is enabled.\n      if (gateway.resharding.enabled && gateway.resharding.checkInterval !== -1) {\n        // It is better to ensure there is always only one\n        clearInterval(gateway.resharding.checkIntervalId)\n\n        if (!gateway.resharding.getSessionInfo) {\n          gateway.resharding.enabled = false\n          gateway.logger.warn(\"[Resharding] Resharding is enabled but 'resharding.getSessionInfo()' was not provided. Disabling resharding.\")\n\n          return\n        }\n\n        gateway.resharding.checkIntervalId = setInterval(async () => {\n          const reshardingInfo = await gateway.resharding.checkIfReshardingIsNeeded()\n\n          if (reshardingInfo.needed && reshardingInfo.info) await gateway.resharding.reshard(reshardingInfo.info)\n        }, gateway.resharding.checkInterval)\n      }\n    },\n    async shutdown(code, reason, clearReshardingInterval = true) {\n      if (clearReshardingInterval) clearInterval(gateway.resharding.checkIntervalId)\n\n      await Promise.all(Array.from(gateway.shards.values()).map((shard) => shard.close(code, reason)))\n    },\n    async sendPayload(shardId, payload) {\n      const shard = gateway.shards.get(shardId)\n\n      if (!shard) {\n        throw new Error(`Shard (id: ${shardId} not found`)\n      }\n\n      await shard.send(payload)\n    },\n    async tellWorkerToIdentify(workerId, shardId, bucketId) {\n      gateway.logger.debug(`[Gateway] Tell worker #${workerId} to identify shard #${shardId} from bucket ${bucketId}`)\n      await gateway.identify(shardId)\n    },\n    async identify(shardId: number) {\n      let shard = this.shards.get(shardId)\n      gateway.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            transportCompression: gateway.transportCompression,\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          logger: this.logger,\n          requestIdentify: async () => await gateway.requestIdentify(shardId),\n          makePresence: gateway.makePresence,\n        })\n\n        if (this.preferSnakeCase) {\n          shard.forwardToBot = async (payload) => {\n            shard!.events.message?.(shard!, payload)\n          }\n        }\n\n        this.shards.set(shardId, shard)\n      }\n\n      await shard.identify()\n    },\n\n    async requestIdentify(shardId) {\n      gateway.logger.debug(`[Gateway] Shard #${shardId} requested an identify.`)\n\n      const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)\n\n      if (!bucket) {\n        throw new Error(\"Can't request identify for a shard that is not assigned to any bucket.\")\n      }\n\n      await bucket.leakyBucket.acquire()\n\n      gateway.logger.debug(`[Gateway] Approved identify request for Shard #${shardId}.`)\n    },\n\n    async kill(shardId: number) {\n      const shard = this.shards.get(shardId)\n      if (!shard) {\n        gateway.logger.debug(`[Gateway] Shard #${shardId} was requested to be killed, but the shard could not be found.`)\n        return\n      }\n\n      gateway.logger.debug(`[Gateway] Killing Shard #${shardId}`)\n      this.shards.delete(shardId)\n      await shard.shutdown()\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        gateway.logger.debug(`[Gateway] calculateShardId (1 shard)`)\n        return 0\n      }\n\n      gateway.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\n      gateway.logger.debug(`[Gateway] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`)\n\n      await gateway.sendPayload(shardId, {\n        op: GatewayOpcodes.VoiceStateUpdate,\n        d: {\n          guild_id: guildId.toString(),\n          channel_id: channelId.toString(),\n          self_mute: options?.selfMute ?? false,\n          self_deaf: options?.selfDeaf ?? true,\n        },\n      })\n    },\n\n    async editBotStatus(data) {\n      gateway.logger.debug(`[Gateway] editBotStatus data: ${JSON.stringify(data, jsonSafeReplacer)}`)\n\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      gateway.logger.debug(`[Gateway] editShardStatus shardId: ${shardId} -> data: ${JSON.stringify(data)}`)\n\n      await gateway.sendPayload(shardId, {\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    async requestMembers(guildId, options) {\n      const shardId = gateway.calculateShardId(guildId)\n\n      if (gateway.intents && (!options?.limit || options.limit > 1) && !(gateway.intents & GatewayIntents.GuildMembers))\n        throw new Error('Cannot fetch more then 1 member without the GUILD_MEMBERS intent')\n\n      gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> data: ${JSON.stringify(options)}`)\n\n      if (options?.userIds?.length) {\n        gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> setting user limit based on userIds length: ${options.userIds.length}`)\n\n        options.limit = options.userIds.length\n      }\n\n      if (!options?.nonce) {\n        let nonce = ''\n\n        while (!nonce || gateway.cache.requestMembers.pending.has(nonce)) {\n          nonce = randomBytes(16).toString('hex')\n        }\n\n        options ??= { limit: 0 }\n        options.nonce = nonce\n      }\n\n      const members = !gateway.cache.requestMembers.enabled\n        ? []\n        : new Promise<Camelize<DiscordMemberWithUser[]>>((resolve, reject) => {\n            // Should never happen.\n            if (!gateway.cache.requestMembers.enabled || !options?.nonce) {\n              reject(new Error(\"Can't request the members without the nonce or with the feature disabled.\"))\n              return\n            }\n\n            gateway.cache.requestMembers.pending.set(options.nonce, {\n              nonce: options.nonce,\n              resolve,\n              members: [],\n            })\n          })\n\n      await gateway.sendPayload(shardId, {\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      return await members\n    },\n\n    async leaveVoiceChannel(guildId) {\n      const shardId = gateway.calculateShardId(guildId)\n\n      gateway.logger.debug(`[Gateway] leaveVoiceChannel guildId: ${guildId} Shard ${shardId}`)\n\n      await gateway.sendPayload(shardId, {\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    async requestSoundboardSounds(guildIds) {\n      /**\n       * Discord will send the events for the guilds that are \"under the shard\" that sends the opcode.\n       * For this reason we need to group the ids with the shard the calculateShardId method gives\n       */\n\n      const map = new Map<number, BigString[]>()\n\n      for (const guildId of guildIds) {\n        const shardId = gateway.calculateShardId(guildId)\n\n        const ids = map.get(shardId) ?? []\n        map.set(shardId, ids)\n\n        ids.push(guildId)\n      }\n\n      await Promise.all(\n        [...map.entries()].map(([shardId, ids]) =>\n          gateway.sendPayload(shardId, {\n            op: GatewayOpcodes.RequestSoundboardSounds,\n            d: {\n              guild_ids: ids,\n            },\n          }),\n        ),\n      )\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  /**\n   * Whether to spread shards across workers in a round-robin manner.\n   *\n   * @remarks\n   * By default, shards are assigned to workers in contiguous blocks based on shardsPerWorker. If any shard is left over, it will be assigned to the last worker.\n   * This means that if you have 3 workers and 40 shards while shardsPerWorker is 10, the first worker will get shards 0-9, the second worker will get shards 10-19, and so on, the last worker will get shards 20-39.\n   *\n   * If this option is set to true, the shards will be assigned in a round-robin manner to spread more evenly across workers.\n   * For example, with 3 workers and 40 shards, the first shard will go to worker 0, the second shard to worker 1, the third shard to worker 2, the fourth shard to worker 0, and so on.\n   *\n   * @default false\n   */\n  spreadShardsInRoundRobin?: boolean\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  /** What transport compression should be used */\n  transportCompression?: TransportCompression | null\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    }\n  }\n  /**\n   * The logger that the gateway manager will use.\n   * @default logger // The logger exported by `@discordeno/utils`\n   */\n  logger?: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>\n  /**\n   * Make the presence for when the bot connects to the gateway\n   *\n   * @remarks\n   * This function will be called each time a Shard is going to identify\n   */\n  makePresence?: () => Promise<DiscordUpdatePresence | undefined>\n  /** Options related to resharding. */\n  resharding?: {\n    /**\n     * Whether or not automated resharding should be enabled.\n     * @default true\n     */\n    enabled: boolean\n    /**\n     * The % of how full a shard is when resharding should be triggered.\n     *\n     * @remarks\n     * We use discord recommended shard value to get an **approximation** of the shard full percentage to compare with this value so the bot may not reshard at the exact percentage provided but may reshard when it is a bit higher than the provided percentage.\n     * For accurate calculation, you may override the `checkIfReshardingIsNeeded` function\n     *\n     * @default 80 as in 80%\n     */\n    shardsFullPercentage: number\n    /**\n     * The interval in milliseconds, of how often to check whether resharding is needed and reshard automatically. Set to -1 to disable auto resharding.\n     * @default 28800000 (8 hours)\n     */\n    checkInterval: number\n    /** Handler to get shard count and other session info. */\n    getSessionInfo?: () => Promise<Camelize<DiscordGetGatewayBot>>\n    /** Handler to edit the shard id on any cached guilds. */\n    updateGuildsShardId?: (guildIds: string[], shardId: number) => Promise<void>\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      /** The bucket to queue the identifies. */\n      leakyBucket: LeakyBucket\n    }\n  >\n  /** The shards that are created. */\n  shards: Map<number, Shard>\n  /** The logger for the gateway manager. */\n  logger: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>\n  /** Everything related to resharding. */\n  resharding: CreateGatewayManagerOptions['resharding'] & {\n    /** The interval id of the check interval. This is used to clear the interval when the manager is shutdown. */\n    checkIntervalId?: NodeJS.Timeout | undefined\n    /** Holds the shards that resharding has created. Once resharding is done, this replaces the gateway.shards */\n    shards: Map<number, Shard>\n    /** Handler to check if resharding is necessary. */\n    checkIfReshardingIsNeeded: () => Promise<{ needed: boolean; info?: Camelize<DiscordGetGatewayBot> }>\n    /**\n     * Handler to begin resharding.\n     *\n     * @remarks\n     * This function will resolve once the resharding is done.\n     * So when all the calls to {@link tellWorkerToPrepare} and {@link onReshardingSwitch} are done.\n     */\n    reshard: (info: Camelize<DiscordGetGatewayBot> & { firstShardId?: number; lastShardId?: number }) => Promise<void>\n    /**\n     * Handler to communicate to a worker that it needs to spawn a new shard and identify it for the resharding.\n     *\n     * @remarks\n     * This handler works in the same way as the {@link tellWorkerToIdentify} handler.\n     * So you should wait for the worker to have identified the shard before resolving the promise\n     */\n    tellWorkerToPrepare: (workerId: number, shardId: number, bucketId: number) => Promise<void>\n    /**\n     * Handle called when all the workers have finished preparing for the resharding.\n     *\n     * This should make the new resharded shards become the active ones and shutdown the old ones\n     */\n    onReshardingSwitch: () => Promise<void>\n  }\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, clearReshardingInterval?: boolean) => Promise<void>\n  sendPayload: (shardId: number, payload: ShardSocketRequest) => Promise<void>\n  /**\n   * Allows users to hook in and change to communicate to different workers across different servers or anything they like.\n   * For example using redis pubsub to talk to other servers.\n   *\n   * @remarks\n   * This should wait for the worker to have identified the shard before resolving the returned promise\n   */\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 nothing\n   */\n  editBotStatus: (data: DiscordUpdatePresence) => 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 nothing\n   */\n  editShardStatus: (shardId: number, data: DiscordUpdatePresence) => Promise<void>\n  /**\n   * Fetches the list of members for a guild over the gateway. If `gateway.cache.requestMembers.enabled` is not set, this function will return an empty array and you'll have to handle the `GUILD_MEMBERS_CHUNK` events yourself.\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<DiscordMemberWithUser[]>>\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   * Used to request soundboard sounds for a list of guilds.\n   *\n   * This function sends multiple (see remarks) _Request Soundboard Sounds_ gateway command over the gateway behind the scenes.\n   *\n   * @param guildIds - The guilds to get the sounds from\n   *\n   * @remarks\n   * Fires a _Soundboard Sounds_ gateway event.\n   *\n   * ⚠️ Discord will send the _Soundboard Sounds_ for each of the guild ids\n   * however you may not receive the same number of events as the ids passed to _Request Soundboard Sounds_ for one of the following reasons:\n   * - The bot is not in the server provided\n   * - The shard the message has been sent from does not receive events for the specified guild\n   *\n   * To avoid this Discordeno will automatically try to group the ids based on what shard they will need to be sent, but this involves sending multiple messages in multiple shards\n   *\n   * @see {@link https://discord.com/developers/docs/topics/gateway-events#request-soundboard-sounds}\n   */\n  requestSoundboardSounds: (guildIds: BigString[]) => Promise<void>\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 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<DiscordMemberWithUser[]> | PromiseLike<Camelize<DiscordMemberWithUser[]>>) => void\n  /** The members that have already arrived for this request. */\n  members: DiscordMemberWithUser[]\n}\n"],"names":["createGatewayManager","options","connectionOptions","connection","url","shards","sessionStartLimit","maxConcurrency","remaining","total","resetAfter","gateway","events","compress","transportCompression","intents","properties","os","process","platform","browser","device","token","version","totalShards","lastShardId","firstShardId","totalWorkers","shardsPerWorker","spawnShardDelay","spreadShardsInRoundRobin","preferSnakeCase","Map","buckets","cache","requestMembers","enabled","pending","Collection","logger","makePresence","Promise","resolve","undefined","resharding","shardsFullPercentage","checkInterval","getSessionInfo","updateGuildsShardId","checkIfReshardingIsNeeded","debug","needed","Error","sessionInfo","JSON","stringify","info","percentage","reshard","calculateTotalShards","clear","prepareBuckets","promises","Array","from","entries","map","bucketId","bucket","worker","workers","shardId","queue","tellWorkerToPrepare","id","all","onReshardingSwitch","workerId","shard","Shard","message","_shard","payload","t","d","guilds","g","requestIdentify","forwardToBot","set","identify","values","oldHandler","_","shutdown","ShardSocketCloseCodes","Resharded","Math","ceil","calculateWorkerId","min","floor","i","leakyBucket","LeakyBucket","max","refillAmount","refillInterval","get","find","w","push","spawnShards","tellWorkerToIdentify","clearInterval","checkIntervalId","warn","setInterval","reshardingInfo","code","reason","clearReshardingInterval","close","sendPayload","send","acquire","kill","delete","calculateShardId","guildId","Number","BigInt","joinVoiceChannel","channelId","op","GatewayOpcodes","VoiceStateUpdate","guild_id","toString","channel_id","self_mute","selfMute","self_deaf","selfDeaf","editBotStatus","data","jsonSafeReplacer","editShardStatus","PresenceUpdate","since","afk","activities","status","limit","GatewayIntents","GuildMembers","userIds","length","nonce","has","randomBytes","members","reject","RequestGuildMembers","query","presences","user_ids","leaveVoiceChannel","requestSoundboardSounds","guildIds","ids","RequestSoundboardSounds","guild_ids"],"mappings":";;;;+BAiBgBA;;;eAAAA;;;4BAjBY;uBAYrB;uBAC2D;8DAChD;wBACiH;;;;;;AAE5H,SAASA,qBAAqBC,OAAoC;IACvE,MAAMC,oBAAoBD,QAAQE,UAAU,IAAI;QAC9CC,KAAK;QACLC,QAAQ;QACRC,mBAAmB;YACjBC,gBAAgB;YAChBC,WAAW;YACXC,OAAO;YACPC,YAAY,OAAO,KAAK,KAAK;QAC/B;IACF;IAEA,MAAMC,UAA0B;QAC9BC,QAAQX,QAAQW,MAAM,IAAI,CAAC;QAC3BC,UAAUZ,QAAQY,QAAQ,IAAI;QAC9BC,sBAAsBb,QAAQa,oBAAoB,IAAI;QACtDC,SAASd,QAAQc,OAAO,IAAI;QAC5BC,YAAY;YACVC,IAAIhB,QAAQe,UAAU,EAAEC,MAAMC,QAAQC,QAAQ;YAC9CC,SAASnB,QAAQe,UAAU,EAAEI,WAAW;YACxCC,QAAQpB,QAAQe,UAAU,EAAEK,UAAU;QACxC;QACAC,OAAOrB,QAAQqB,KAAK;QACpBlB,KAAKH,QAAQG,GAAG,IAAIF,kBAAkBE,GAAG,IAAI;QAC7CmB,SAAStB,QAAQsB,OAAO,IAAI;QAC5BpB,YAAYD;QACZsB,aAAavB,QAAQuB,WAAW,IAAItB,kBAAkBG,MAAM,IAAI;QAChEoB,aAAaxB,QAAQwB,WAAW,IAAKxB,CAAAA,QAAQuB,WAAW,GAAGvB,QAAQuB,WAAW,GAAG,IAAItB,oBAAoBA,kBAAkBG,MAAM,GAAG,IAAI,CAAA;QACxIqB,cAAczB,QAAQyB,YAAY,IAAI;QACtCC,cAAc1B,QAAQ0B,YAAY,IAAI;QACtCC,iBAAiB3B,QAAQ2B,eAAe,IAAI;QAC5CC,iBAAiB5B,QAAQ4B,eAAe,IAAI;QAC5CC,0BAA0B7B,QAAQ6B,wBAAwB,IAAI;QAC9DC,iBAAiB9B,QAAQ8B,eAAe,IAAI;QAC5C1B,QAAQ,IAAI2B;QACZC,SAAS,IAAID;QACbE,OAAO;YACLC,gBAAgB;gBACdC,SAASnC,QAAQiC,KAAK,EAAEC,gBAAgBC,WAAW;gBACnDC,SAAS,IAAIC,iBAAU;YACzB;QACF;QACAC,QAAQtC,QAAQsC,MAAM,IAAIA,aAAM;QAChCC,cAAcvC,QAAQuC,YAAY,IAAK,CAAA,IAAMC,QAAQC,OAAO,CAACC,UAAS;QACtEC,YAAY;YACVR,SAASnC,QAAQ2C,UAAU,EAAER,WAAW;YACxCS,sBAAsB5C,QAAQ2C,UAAU,EAAEC,wBAAwB;YAClEC,eAAe7C,QAAQ2C,UAAU,EAAEE,iBAAiB;YACpDzC,QAAQ,IAAI2B;YACZe,gBAAgB9C,QAAQ2C,UAAU,EAAEG;YACpCC,qBAAqB/C,QAAQ2C,UAAU,EAAEI;YACzC,MAAMC;gBACJtC,QAAQ4B,MAAM,CAACW,KAAK,CAAC;gBAErB,IAAI,CAACvC,QAAQiC,UAAU,CAACR,OAAO,EAAE;oBAC/BzB,QAAQ4B,MAAM,CAACW,KAAK,CAAC;oBAErB,OAAO;wBAAEC,QAAQ;oBAAM;gBACzB;gBAEA,IAAI,CAACxC,QAAQiC,UAAU,CAACG,cAAc,EAAE;oBACtC,MAAM,IAAIK,MAAM;gBAClB;gBAEAzC,QAAQ4B,MAAM,CAACW,KAAK,CAAC;gBAErB,MAAMG,cAAc,MAAM1C,QAAQiC,UAAU,CAACG,cAAc;gBAE3DpC,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,qCAAqC,EAAEI,KAAKC,SAAS,CAACF,cAAc;gBAE1F,sDAAsD;gBACtD,IAAIA,YAAY/C,iBAAiB,CAACE,SAAS,GAAG6C,YAAYhD,MAAM,EAAE;oBAChEM,QAAQ4B,MAAM,CAACW,KAAK,CAAC;oBAErB,OAAO;wBAAEC,QAAQ;wBAAOK,MAAMH;oBAAY;gBAC5C;gBAEA1C,QAAQ4B,MAAM,CAACW,KAAK,CAAC;gBAErB,6DAA6D;gBAC7D,uFAAuF;gBACvF,iEAAiE;gBACjE,iHAAiH;gBACjH,wIAAwI;gBACxI,MAAMO,aAAa,AAACJ,YAAYhD,MAAM,GAAI,CAAA,AAACM,QAAQa,WAAW,GAAG,OAAQ,IAAG,IAAM;gBAElF,gDAAgD;gBAChD,IAAIiC,aAAa9C,QAAQiC,UAAU,CAACC,oBAAoB,EAAE;oBACxDlC,QAAQ4B,MAAM,CAACW,KAAK,CAAC;oBAErB,OAAO;wBAAEC,QAAQ;wBAAOK,MAAMH;oBAAY;gBAC5C;gBAEA1C,QAAQ4B,MAAM,CAACiB,IAAI,CAAC;gBAEpB,OAAO;oBAAEL,QAAQ;oBAAMK,MAAMH;gBAAY;YAC3C;YACA,MAAMK,SAAQF,IAAI;gBAChB7C,QAAQ4B,MAAM,CAACiB,IAAI,CAAC,CAAC,kEAAkE,EAAE7C,QAAQa,WAAW,EAAE;gBAC9G,wBAAwB;gBACxBb,QAAQa,WAAW,GAAGgC,KAAKnD,MAAM;gBACjC,2CAA2C;gBAC3CM,QAAQa,WAAW,GAAGb,QAAQgD,oBAAoB;gBAClD,yCAAyC;gBACzC,IAAI,OAAOH,KAAK9B,YAAY,KAAK,UAAUf,QAAQe,YAAY,GAAG8B,KAAK9B,YAAY;gBACnF,wCAAwC;gBACxC,IAAI,OAAO8B,KAAK/B,WAAW,KAAK,UAAUd,QAAQc,WAAW,GAAG+B,KAAK/B,WAAW;qBAE3Ed,QAAQc,WAAW,GAAGd,QAAQa,WAAW,GAAG;gBACjDb,QAAQ4B,MAAM,CAACiB,IAAI,CAAC,CAAC,6DAA6D,EAAE7C,QAAQa,WAAW,EAAE;gBAEzG,oBAAoB;gBACpBb,QAAQsB,OAAO,CAAC2B,KAAK;gBACrB,oCAAoC;gBACpCjD,QAAQkD,cAAc;gBAEtB,mEAAmE;gBACnE,MAAMC,WAAWC,MAAMC,IAAI,CAACrD,QAAQsB,OAAO,CAACgC,OAAO,IAAIC,GAAG,CAAC,OAAO,CAACC,UAAUC,OAAO;oBAClF,KAAK,MAAMC,UAAUD,OAAOE,OAAO,CAAE;wBACnC,KAAK,MAAMC,WAAWF,OAAOG,KAAK,CAAE;4BAClC,MAAM7D,QAAQiC,UAAU,CAAC6B,mBAAmB,CAACJ,OAAOK,EAAE,EAAEH,SAASJ;wBACnE;oBACF;gBACF;gBAEA,MAAM1B,QAAQkC,GAAG,CAACb;gBAElBnD,QAAQ4B,MAAM,CAACiB,IAAI,CAAC,CAAC,uCAAuC,CAAC;gBAE7D,MAAM7C,QAAQiC,UAAU,CAACgC,kBAAkB;YAC7C;YACA,MAAMH,qBAAoBI,QAAQ,EAAEN,OAAO,EAAEJ,QAAQ;gBACnDxD,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,gDAAgD,EAAE2B,SAAS,UAAU,EAAEN,QAAQ,WAAW,EAAEJ,SAAS,CAAC,CAAC;gBAC7H,MAAMW,QAAQ,IAAIC,cAAK,CAAC;oBACtBL,IAAIH;oBACJpE,YAAY;wBACVU,UAAUF,QAAQE,QAAQ;wBAC1BC,sBAAsBH,QAAQG,oBAAoB,IAAI;wBACtDC,SAASJ,QAAQI,OAAO;wBACxBC,YAAYL,QAAQK,UAAU;wBAC9BM,OAAOX,QAAQW,KAAK;wBACpBE,aAAab,QAAQa,WAAW;wBAChCpB,KAAKO,QAAQP,GAAG;wBAChBmB,SAASZ,QAAQY,OAAO;oBAC1B;oBACAX,QAAQ;wBACN,MAAMoE,SAAQC,MAAM,EAAEC,OAAO;4BAC3B,wEAAwE;4BACxE,IAAIA,QAAQC,CAAC,KAAK,SAAS;gCACzB,MAAMxE,QAAQiC,UAAU,CAACI,mBAAmB,GAC1C,AAACkC,QAAQE,CAAC,CAAkBC,MAAM,CAACnB,GAAG,CAAC,CAACoB,IAAMA,EAAEZ,EAAE,GAClDH;4BAEJ;wBACF;oBACF;oBACAhC,QAAQ5B,QAAQ4B,MAAM;oBACtBgD,iBAAiB,UAAY,MAAM5E,QAAQ4E,eAAe,CAAChB;oBAC3D/B,cAAc7B,QAAQ6B,YAAY;gBACpC;gBAEA,IAAI7B,QAAQoB,eAAe,EAAE;oBAC3B+C,MAAMU,YAAY,GAAG,OAAON;wBAC1BJ,MAAMlE,MAAM,EAAEoE,UAAUF,OAAOI;oBACjC;gBACF;gBAEAvE,QAAQiC,UAAU,CAACvC,MAAM,CAACoF,GAAG,CAAClB,SAASO;gBAEvC,MAAMA,MAAMY,QAAQ;gBAEpB/E,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,oBAAoB,EAAEqB,QAAQ,YAAY,CAAC;YACnE;YACA,MAAMK;gBACJjE,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,mEAAmE,CAAC;gBAE1F,sDAAsD;gBACtD,KAAK,MAAM4B,SAASnE,QAAQiC,UAAU,CAACvC,MAAM,CAACsF,MAAM,GAAI;oBACtDb,MAAMlE,MAAM,GAAGX,QAAQW,MAAM,IAAI,CAAC;gBACpC;gBAEA,oCAAoC;gBACpC,KAAK,MAAMkE,SAASnE,QAAQN,MAAM,CAACsF,MAAM,GAAI;oBAC3C,MAAMC,aAAad,MAAMlE,MAAM,CAACoE,OAAO;oBAEvC,mKAAmK;oBACnKF,MAAMlE,MAAM,GAAG;wBACb,GAAGkE,MAAMlE,MAAM;wBACfoE,SAAS,eAAgBa,CAAC,EAAEb,OAAO;4BACjC,qDAAqD;4BACrD,IAAIA,QAAQG,CAAC,KAAK,uBAAuB;gCACvCS,aAAad,OAAOE;4BACtB;wBACF;oBACF;gBACF;gBAEArE,QAAQ4B,MAAM,CAACiB,IAAI,CAAC,CAAC,sCAAsC,CAAC;gBAC5D,MAAM7C,QAAQmF,QAAQ,CAACC,6BAAqB,CAACC,SAAS,EAAE,cAAc;gBAEtErF,QAAQ4B,MAAM,CAACiB,IAAI,CAAC,CAAC,uBAAuB,CAAC;gBAC7C7C,QAAQN,MAAM,GAAG,IAAI2B,IAAIrB,QAAQiC,UAAU,CAACvC,MAAM;gBAClDM,QAAQiC,UAAU,CAACvC,MAAM,CAACuD,KAAK;YACjC;QACF;QAEAD;YACE,qDAAqD;YACrD,IAAIhD,QAAQa,WAAW,GAAG,KAAK;gBAC7Bb,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,oCAAoC,EAAEvC,QAAQa,WAAW,EAAE;gBACjF,OAAOb,QAAQa,WAAW;YAC5B;YAEAb,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,kCAAkC,CAAC,EAAEvC,QAAQa,WAAW,EAAEb,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc;YACnI,wFAAwF;YACxF,OACE0F,KAAKC,IAAI,CACPvF,QAAQa,WAAW,GACjB,uJAAuJ;YACtJb,CAAAA,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc,KAAK,IAAI,KAAKI,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc,AAAD,KACnHI,CAAAA,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc,KAAK,IAAI,KAAKI,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc,AAAD;QAE5H;QACA4F,mBAAkB5B,OAAO;YACvB,MAAMM,WAAW5E,QAAQ6B,wBAAwB,GAC7CyC,UAAU5D,QAAQgB,YAAY,GAC9BsE,KAAKG,GAAG,CAACH,KAAKI,KAAK,CAAC9B,UAAU5D,QAAQiB,eAAe,GAAGjB,QAAQgB,YAAY,GAAG;YACnFhB,QAAQ4B,MAAM,CAACW,KAAK,CAClB,CAAC,uCAAuC,EAAEqB,QAAQ,YAAY,EAAEM,SAAS,gBAAgB,EAAElE,QAAQiB,eAAe,CAAC,WAAW,EAAEjB,QAAQgB,YAAY,EAAE;YAExJ,OAAOkD;QACT;QACAhB;YACE,IAAK,IAAIyC,IAAI,GAAGA,IAAI3F,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc,EAAE,EAAE+F,EAAG;gBAC5E3F,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,6CAA6C,EAAEoD,GAAG;gBACxE3F,QAAQsB,OAAO,CAACwD,GAAG,CAACa,GAAG;oBACrBhC,SAAS,EAAE;oBACXiC,aAAa,IAAIC,kBAAW,CAAC;wBAC3BC,KAAK;wBACLC,cAAc;wBACdC,gBAAgBhG,QAAQkB,eAAe;oBACzC;gBACF;YACF;YAEA,6CAA6C;YAC7C,IAAK,IAAI0C,UAAU5D,QAAQe,YAAY,EAAE6C,WAAW5D,QAAQc,WAAW,EAAE,EAAE8C,QAAS;gBAClF5D,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,uCAAuC,EAAEqB,SAAS;gBAExE,IAAIA,WAAW5D,QAAQa,WAAW,EAAE;oBAClC,MAAM,IAAI4B,MAAM,CAAC,WAAW,EAAEmB,QAAQ,gEAAgE,EAAE5D,QAAQa,WAAW,EAAE;gBAC/H;gBAEA,MAAM2C,WAAWI,UAAU5D,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc;gBAC9E,MAAM6D,SAASzD,QAAQsB,OAAO,CAAC2E,GAAG,CAACzC;gBAEnC,IAAI,CAACC,QAAQ;oBACX,MAAM,IAAIhB,MACR,CAAC,WAAW,EAAEmB,QAAQ,wCAAwC,EAAEJ,SAAS,qCAAqC,EAC5GxD,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc,GAAG,GACtD;gBAEN;gBAEA,mCAAmC;gBACnC,MAAMsE,WAAWlE,QAAQwF,iBAAiB,CAAC5B;gBAC3C,MAAMF,SAASD,OAAOE,OAAO,CAACuC,IAAI,CAAC,CAACC,IAAMA,EAAEpC,EAAE,KAAKG;gBAEnD,4DAA4D;gBAC5D,IAAIR,QAAQ;oBACVA,OAAOG,KAAK,CAACuC,IAAI,CAACxC;gBACpB,OAAO;oBACLH,OAAOE,OAAO,CAACyC,IAAI,CAAC;wBAAErC,IAAIG;wBAAUL,OAAO;4BAACD;yBAAQ;oBAAC;gBACvD;YACF;QACF;QACA,MAAMyC;YACJ,kCAAkC;YAClCrG,QAAQkD,cAAc;YAEtB,MAAMC,WAAW;mBAAInD,QAAQsB,OAAO,CAACgC,OAAO;aAAG,CAACC,GAAG,CAAC,OAAO,CAACC,UAAUC,OAAO;gBAC3E,KAAK,MAAMC,UAAUD,OAAOE,OAAO,CAAE;oBACnC,KAAK,MAAMC,WAAWF,OAAOG,KAAK,CAAE;wBAClC,MAAM7D,QAAQsG,oBAAoB,CAAC5C,OAAOK,EAAE,EAAEH,SAASJ;oBACzD;gBACF;YACF;YAEA,kEAAkE;YAClE,MAAM1B,QAAQkC,GAAG,CAACb;YAElB,iEAAiE;YACjE,IAAInD,QAAQiC,UAAU,CAACR,OAAO,IAAIzB,QAAQiC,UAAU,CAACE,aAAa,KAAK,CAAC,GAAG;gBACzE,kDAAkD;gBAClDoE,cAAcvG,QAAQiC,UAAU,CAACuE,eAAe;gBAEhD,IAAI,CAACxG,QAAQiC,UAAU,CAACG,cAAc,EAAE;oBACtCpC,QAAQiC,UAAU,CAACR,OAAO,GAAG;oBAC7BzB,QAAQ4B,MAAM,CAAC6E,IAAI,CAAC;oBAEpB;gBACF;gBAEAzG,QAAQiC,UAAU,CAACuE,eAAe,GAAGE,YAAY;oBAC/C,MAAMC,iBAAiB,MAAM3G,QAAQiC,UAAU,CAACK,yBAAyB;oBAEzE,IAAIqE,eAAenE,MAAM,IAAImE,eAAe9D,IAAI,EAAE,MAAM7C,QAAQiC,UAAU,CAACc,OAAO,CAAC4D,eAAe9D,IAAI;gBACxG,GAAG7C,QAAQiC,UAAU,CAACE,aAAa;YACrC;QACF;QACA,MAAMgD,UAASyB,IAAI,EAAEC,MAAM,EAAEC,0BAA0B,IAAI;YACzD,IAAIA,yBAAyBP,cAAcvG,QAAQiC,UAAU,CAACuE,eAAe;YAE7E,MAAM1E,QAAQkC,GAAG,CAACZ,MAAMC,IAAI,CAACrD,QAAQN,MAAM,CAACsF,MAAM,IAAIzB,GAAG,CAAC,CAACY,QAAUA,MAAM4C,KAAK,CAACH,MAAMC;QACzF;QACA,MAAMG,aAAYpD,OAAO,EAAEW,OAAO;YAChC,MAAMJ,QAAQnE,QAAQN,MAAM,CAACuG,GAAG,CAACrC;YAEjC,IAAI,CAACO,OAAO;gBACV,MAAM,IAAI1B,MAAM,CAAC,WAAW,EAAEmB,QAAQ,UAAU,CAAC;YACnD;YAEA,MAAMO,MAAM8C,IAAI,CAAC1C;QACnB;QACA,MAAM+B,sBAAqBpC,QAAQ,EAAEN,OAAO,EAAEJ,QAAQ;YACpDxD,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,uBAAuB,EAAE2B,SAAS,oBAAoB,EAAEN,QAAQ,aAAa,EAAEJ,UAAU;YAC/G,MAAMxD,QAAQ+E,QAAQ,CAACnB;QACzB;QACA,MAAMmB,UAASnB,OAAe;YAC5B,IAAIO,QAAQ,IAAI,CAACzE,MAAM,CAACuG,GAAG,CAACrC;YAC5B5D,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,sBAAsB,EAAE4B,QAAQ,aAAa,MAAM,QAAQ,EAAEP,QAAQ,CAAC,CAAC;YAE7F,IAAI,CAACO,OAAO;gBACVA,QAAQ,IAAIC,cAAK,CAAC;oBAChBL,IAAIH;oBACJpE,YAAY;wBACVU,UAAU,IAAI,CAACA,QAAQ;wBACvBC,sBAAsBH,QAAQG,oBAAoB;wBAClDC,SAAS,IAAI,CAACA,OAAO;wBACrBC,YAAY,IAAI,CAACA,UAAU;wBAC3BM,OAAO,IAAI,CAACA,KAAK;wBACjBE,aAAa,IAAI,CAACA,WAAW;wBAC7BpB,KAAK,IAAI,CAACA,GAAG;wBACbmB,SAAS,IAAI,CAACA,OAAO;oBACvB;oBACAX,QAAQX,QAAQW,MAAM,IAAI,CAAC;oBAC3B2B,QAAQ,IAAI,CAACA,MAAM;oBACnBgD,iBAAiB,UAAY,MAAM5E,QAAQ4E,eAAe,CAAChB;oBAC3D/B,cAAc7B,QAAQ6B,YAAY;gBACpC;gBAEA,IAAI,IAAI,CAACT,eAAe,EAAE;oBACxB+C,MAAMU,YAAY,GAAG,OAAON;wBAC1BJ,MAAOlE,MAAM,CAACoE,OAAO,GAAGF,OAAQI;oBAClC;gBACF;gBAEA,IAAI,CAAC7E,MAAM,CAACoF,GAAG,CAAClB,SAASO;YAC3B;YAEA,MAAMA,MAAMY,QAAQ;QACtB;QAEA,MAAMH,iBAAgBhB,OAAO;YAC3B5D,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,iBAAiB,EAAEqB,QAAQ,uBAAuB,CAAC;YAEzE,MAAMH,SAASzD,QAAQsB,OAAO,CAAC2E,GAAG,CAACrC,UAAU5D,QAAQR,UAAU,CAACG,iBAAiB,CAACC,cAAc;YAEhG,IAAI,CAAC6D,QAAQ;gBACX,MAAM,IAAIhB,MAAM;YAClB;YAEA,MAAMgB,OAAOmC,WAAW,CAACsB,OAAO;YAEhClH,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,+CAA+C,EAAEqB,QAAQ,CAAC,CAAC;QACnF;QAEA,MAAMuD,MAAKvD,OAAe;YACxB,MAAMO,QAAQ,IAAI,CAACzE,MAAM,CAACuG,GAAG,CAACrC;YAC9B,IAAI,CAACO,OAAO;gBACVnE,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,iBAAiB,EAAEqB,QAAQ,8DAA8D,CAAC;gBAChH;YACF;YAEA5D,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,yBAAyB,EAAEqB,SAAS;YAC1D,IAAI,CAAClE,MAAM,CAAC0H,MAAM,CAACxD;YACnB,MAAMO,MAAMgB,QAAQ;QACtB;QAEA,6BAA6B;QAE7BkC,kBAAiBC,OAAO,EAAEzG,WAAW;YACnC,wEAAwE;YACxE,IAAI,CAACA,aAAaA,cAAcb,QAAQa,WAAW;YACnD,sDAAsD;YACtD,IAAIA,gBAAgB,GAAG;gBACrBb,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,oCAAoC,CAAC;gBAC3D,OAAO;YACT;YAEAvC,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,qCAAqC,EAAE+E,QAAQ,eAAe,EAAEzG,YAAY,CAAC,CAAC;YACpG,OAAO0G,OAAO,AAACC,CAAAA,OAAOF,YAAY,GAAG,AAAD,IAAKE,OAAO3G;QAClD;QAEA,MAAM4G,kBAAiBH,OAAO,EAAEI,SAAS,EAAEpI,OAAO;YAChD,MAAMsE,UAAU5D,QAAQqH,gBAAgB,CAACC;YAEzCtH,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,oCAAoC,EAAE+E,QAAQ,YAAY,EAAEI,WAAW;YAE7F,MAAM1H,QAAQgH,WAAW,CAACpD,SAAS;gBACjC+D,IAAIC,qBAAc,CAACC,gBAAgB;gBACnCpD,GAAG;oBACDqD,UAAUR,QAAQS,QAAQ;oBAC1BC,YAAYN,UAAUK,QAAQ;oBAC9BE,WAAW3I,SAAS4I,YAAY;oBAChCC,WAAW7I,SAAS8I,YAAY;gBAClC;YACF;QACF;QAEA,MAAMC,eAAcC,IAAI;YACtBtI,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,8BAA8B,EAAEI,KAAKC,SAAS,CAAC0F,MAAMC,uBAAgB,GAAG;YAE9F,MAAMzG,QAAQkC,GAAG,CACf;mBAAIhE,QAAQN,MAAM,CAACsF,MAAM;aAAG,CAACzB,GAAG,CAAC,OAAOY;gBACtCnE,QAAQwI,eAAe,CAACrE,MAAMJ,EAAE,EAAEuE;YACpC;QAEJ;QAEA,MAAME,iBAAgB5E,OAAO,EAAE0E,IAAI;YACjCtI,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,mCAAmC,EAAEqB,QAAQ,UAAU,EAAEjB,KAAKC,SAAS,CAAC0F,OAAO;YAErG,MAAMtI,QAAQgH,WAAW,CAACpD,SAAS;gBACjC+D,IAAIC,qBAAc,CAACa,cAAc;gBACjChE,GAAG;oBACDiE,OAAO;oBACPC,KAAK;oBACLC,YAAYN,KAAKM,UAAU;oBAC3BC,QAAQP,KAAKO,MAAM;gBACrB;YACF;QACF;QAEA,MAAMrH,gBAAe8F,OAAO,EAAEhI,OAAO;YACnC,MAAMsE,UAAU5D,QAAQqH,gBAAgB,CAACC;YAEzC,IAAItH,QAAQI,OAAO,IAAK,CAAA,CAACd,SAASwJ,SAASxJ,QAAQwJ,KAAK,GAAG,CAAA,KAAM,CAAE9I,CAAAA,QAAQI,OAAO,GAAG2I,qBAAc,CAACC,YAAY,AAAD,GAC7G,MAAM,IAAIvG,MAAM;YAElBzC,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,kCAAkC,EAAE+E,QAAQ,UAAU,EAAE3E,KAAKC,SAAS,CAACtD,UAAU;YAEvG,IAAIA,SAAS2J,SAASC,QAAQ;gBAC5BlJ,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,kCAAkC,EAAE+E,QAAQ,gDAAgD,EAAEhI,QAAQ2J,OAAO,CAACC,MAAM,EAAE;gBAE5I5J,QAAQwJ,KAAK,GAAGxJ,QAAQ2J,OAAO,CAACC,MAAM;YACxC;YAEA,IAAI,CAAC5J,SAAS6J,OAAO;gBACnB,IAAIA,QAAQ;gBAEZ,MAAO,CAACA,SAASnJ,QAAQuB,KAAK,CAACC,cAAc,CAACE,OAAO,CAAC0H,GAAG,CAACD,OAAQ;oBAChEA,QAAQE,IAAAA,uBAAW,EAAC,IAAItB,QAAQ,CAAC;gBACnC;gBAEAzI,YAAY;oBAAEwJ,OAAO;gBAAE;gBACvBxJ,QAAQ6J,KAAK,GAAGA;YAClB;YAEA,MAAMG,UAAU,CAACtJ,QAAQuB,KAAK,CAACC,cAAc,CAACC,OAAO,GACjD,EAAE,GACF,IAAIK,QAA2C,CAACC,SAASwH;gBACvD,uBAAuB;gBACvB,IAAI,CAACvJ,QAAQuB,KAAK,CAACC,cAAc,CAACC,OAAO,IAAI,CAACnC,SAAS6J,OAAO;oBAC5DI,OAAO,IAAI9G,MAAM;oBACjB;gBACF;gBAEAzC,QAAQuB,KAAK,CAACC,cAAc,CAACE,OAAO,CAACoD,GAAG,CAACxF,QAAQ6J,KAAK,EAAE;oBACtDA,OAAO7J,QAAQ6J,KAAK;oBACpBpH;oBACAuH,SAAS,EAAE;gBACb;YACF;YAEJ,MAAMtJ,QAAQgH,WAAW,CAACpD,SAAS;gBACjC+D,IAAIC,qBAAc,CAAC4B,mBAAmB;gBACtC/E,GAAG;oBACDqD,UAAUR,QAAQS,QAAQ;oBAC1B,sEAAsE;oBACtE0B,OAAOnK,SAASmK,SAAUnK,CAAAA,SAASwJ,QAAQ9G,YAAY,EAAC;oBACxD8G,OAAOxJ,SAASwJ,SAAS;oBACzBY,WAAWpK,SAASoK,aAAa;oBACjCC,UAAUrK,SAAS2J,SAAS1F,IAAI,CAACQ,KAAOA,GAAGgE,QAAQ;oBACnDoB,OAAO7J,SAAS6J;gBAClB;YACF;YAEA,OAAO,MAAMG;QACf;QAEA,MAAMM,mBAAkBtC,OAAO;YAC7B,MAAM1D,UAAU5D,QAAQqH,gBAAgB,CAACC;YAEzCtH,QAAQ4B,MAAM,CAACW,KAAK,CAAC,CAAC,qCAAqC,EAAE+E,QAAQ,OAAO,EAAE1D,SAAS;YAEvF,MAAM5D,QAAQgH,WAAW,CAACpD,SAAS;gBACjC+D,IAAIC,qBAAc,CAACC,gBAAgB;gBACnCpD,GAAG;oBACDqD,UAAUR,QAAQS,QAAQ;oBAC1BC,YAAY;oBACZC,WAAW;oBACXE,WAAW;gBACb;YACF;QACF;QAEA,MAAM0B,yBAAwBC,QAAQ;YACpC;;;OAGC,GAED,MAAMvG,MAAM,IAAIlC;YAEhB,KAAK,MAAMiG,WAAWwC,SAAU;gBAC9B,MAAMlG,UAAU5D,QAAQqH,gBAAgB,CAACC;gBAEzC,MAAMyC,MAAMxG,IAAI0C,GAAG,CAACrC,YAAY,EAAE;gBAClCL,IAAIuB,GAAG,CAAClB,SAASmG;gBAEjBA,IAAI3D,IAAI,CAACkB;YACX;YAEA,MAAMxF,QAAQkC,GAAG,CACf;mBAAIT,IAAID,OAAO;aAAG,CAACC,GAAG,CAAC,CAAC,CAACK,SAASmG,IAAI,GACpC/J,QAAQgH,WAAW,CAACpD,SAAS;oBAC3B+D,IAAIC,qBAAc,CAACoC,uBAAuB;oBAC1CvF,GAAG;wBACDwF,WAAWF;oBACb;gBACF;QAGN;IACF;IAEA,OAAO/J;AACT"}
@@ -50,8 +50,7 @@ export function createGatewayManager(options) {
50
50
  enabled: options.resharding?.enabled ?? true,
51
51
  shardsFullPercentage: options.resharding?.shardsFullPercentage ?? 80,
52
52
  checkInterval: options.resharding?.checkInterval ?? 28800000,
53
- shards: new Collection(),
54
- pendingShards: new Collection(),
53
+ shards: new Map(),
55
54
  getSessionInfo: options.resharding?.getSessionInfo,
56
55
  updateGuildsShardId: options.resharding?.updateGuildsShardId,
57
56
  async checkIfReshardingIsNeeded () {
@@ -107,19 +106,23 @@ export function createGatewayManager(options) {
107
106
  if (typeof info.firstShardId === 'number') gateway.firstShardId = info.firstShardId;
108
107
  // Set last shard id if provided in info
109
108
  if (typeof info.lastShardId === 'number') gateway.lastShardId = info.lastShardId;
109
+ else gateway.lastShardId = gateway.totalShards - 1;
110
110
  gateway.logger.info(`[Resharding] Starting the reshard process. New total shards: ${gateway.totalShards}`);
111
111
  // Resetting buckets
112
112
  gateway.buckets.clear();
113
113
  // Refilling buckets with new values
114
114
  gateway.prepareBuckets();
115
- // SPREAD THIS OUT TO DIFFERENT WORKERS TO BEGIN STARTING UP
116
- gateway.buckets.forEach(async (bucket, bucketId)=>{
115
+ // Call all the buckets and tell their workers & shards to identify
116
+ const promises = Array.from(gateway.buckets.entries()).map(async ([bucketId, bucket])=>{
117
117
  for (const worker of bucket.workers){
118
118
  for (const shardId of worker.queue){
119
119
  await gateway.resharding.tellWorkerToPrepare(worker.id, shardId, bucketId);
120
120
  }
121
121
  }
122
122
  });
123
+ await Promise.all(promises);
124
+ gateway.logger.info(`[Resharding] All shards are now online.`);
125
+ await gateway.resharding.onReshardingSwitch();
123
126
  },
124
127
  async tellWorkerToPrepare (workerId, shardId, bucketId) {
125
128
  gateway.logger.debug(`[Resharding] Telling worker to prepare. Worker: ${workerId} | Shard: ${shardId} | Bucket: ${bucketId}.`);
@@ -135,9 +138,9 @@ export function createGatewayManager(options) {
135
138
  url: gateway.url,
136
139
  version: gateway.version
137
140
  },
138
- // Ignore events until we are ready
139
141
  events: {
140
142
  async message (_shard, payload) {
143
+ // Ignore all events until we swich from the old shards to the new ones.
141
144
  if (payload.t === 'READY') {
142
145
  await gateway.resharding.updateGuildsShardId?.(payload.d.guilds.map((g)=>g.id), shardId);
143
146
  }
@@ -153,22 +156,14 @@ export function createGatewayManager(options) {
153
156
  };
154
157
  }
155
158
  gateway.resharding.shards.set(shardId, shard);
156
- const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency);
157
- if (!bucket) return;
158
- await shard.identify().then(()=>gateway.resharding.shardIsPending(shard));
159
+ await shard.identify();
160
+ gateway.logger.debug(`[Resharding] Shard #${shardId} identified.`);
159
161
  },
160
- async shardIsPending (shard) {
161
- // Save this in pending at the moment, until all shards are online
162
- gateway.resharding.pendingShards.set(shard.id, shard);
163
- gateway.logger.debug(`[Resharding] Shard #${shard.id} is now pending.`);
164
- // Check if all shards are now online.
165
- if (gateway.lastShardId - gateway.firstShardId >= gateway.resharding.pendingShards.size) return;
166
- gateway.logger.info(`[Resharding] All shards are now online.`);
167
- // New shards start processing events
162
+ async onReshardingSwitch () {
163
+ gateway.logger.debug(`[Resharding] Making the switch from the old shards to the new ones.`);
164
+ // Move the events from the old shards to the new ones
168
165
  for (const shard of gateway.resharding.shards.values()){
169
- for(const event in options.events){
170
- shard.events[event] = options.events[event];
171
- }
166
+ shard.events = options.events ?? {};
172
167
  }
173
168
  // Old shards stop processing events
174
169
  for (const shard of gateway.shards.values()){
@@ -185,14 +180,10 @@ export function createGatewayManager(options) {
185
180
  };
186
181
  }
187
182
  gateway.logger.info(`[Resharding] Shutting down old shards.`);
188
- // Close old shards
189
183
  await gateway.shutdown(ShardSocketCloseCodes.Resharded, 'Resharded!', false);
190
184
  gateway.logger.info(`[Resharding] Completed.`);
191
- // Replace old shards
192
- gateway.shards = new Collection(gateway.resharding.shards);
193
- // Clear our collections and keep only one reference to the shards, the one in gateway.shards
185
+ gateway.shards = new Map(gateway.resharding.shards);
194
186
  gateway.resharding.shards.clear();
195
- gateway.resharding.pendingShards.clear();
196
187
  }
197
188
  },
198
189
  calculateTotalShards () {
@@ -469,4 +460,4 @@ export function createGatewayManager(options) {
469
460
  return gateway;
470
461
  }
471
462
 
472
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/manager.ts"],"sourcesContent":["import { randomBytes } from 'node:crypto'\nimport {\n  type AtLeastOne,\n  type BigString,\n  type Camelize,\n  type DiscordGetGatewayBot,\n  type DiscordMemberWithUser,\n  type DiscordReady,\n  type DiscordUpdatePresence,\n  GatewayIntents,\n  GatewayOpcodes,\n  type RequestGuildMembers,\n} from '@discordeno/types'\nimport { Collection, LeakyBucket, jsonSafeReplacer, logger } from '@discordeno/utils'\nimport Shard from './Shard.js'\nimport { type ShardEvents, ShardSocketCloseCodes, type ShardSocketRequest, type TransportCompression, type 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    transportCompression: options.transportCompression ?? null,\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    spreadShardsInRoundRobin: options.spreadShardsInRoundRobin ?? false,\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    logger: options.logger ?? logger,\n    makePresence: options.makePresence ?? (() => Promise.resolve(undefined)),\n    resharding: {\n      enabled: options.resharding?.enabled ?? true,\n      shardsFullPercentage: options.resharding?.shardsFullPercentage ?? 80,\n      checkInterval: options.resharding?.checkInterval ?? 28800000, // 8 hours\n      shards: new Collection(),\n      pendingShards: new Collection(),\n      getSessionInfo: options.resharding?.getSessionInfo,\n      updateGuildsShardId: options.resharding?.updateGuildsShardId,\n      async checkIfReshardingIsNeeded() {\n        gateway.logger.debug('[Resharding] Checking if resharding is needed.')\n\n        if (!gateway.resharding.enabled) {\n          gateway.logger.debug('[Resharding] Resharding is disabled.')\n\n          return { needed: false }\n        }\n\n        if (!gateway.resharding.getSessionInfo) {\n          throw new Error(\"[Resharding] Resharding is enabled but no 'resharding.getSessionInfo()' is not provided.\")\n        }\n\n        gateway.logger.debug('[Resharding] Resharding is enabled.')\n\n        const sessionInfo = await gateway.resharding.getSessionInfo()\n\n        gateway.logger.debug(`[Resharding] Session info retrieved: ${JSON.stringify(sessionInfo)}`)\n\n        // Don't have enough identify limits to try resharding\n        if (sessionInfo.sessionStartLimit.remaining < sessionInfo.shards) {\n          gateway.logger.debug('[Resharding] Not enough session start limits left to reshard.')\n\n          return { needed: false, info: sessionInfo }\n        }\n\n        gateway.logger.debug('[Resharding] Able to reshard, checking whether necessary now.')\n\n        // 2500 is the max amount of guilds a single shard can handle\n        // 1000 is the amount of guilds discord uses to determine how many shards to recommend.\n        // This algo helps check if your bot has grown enough to reshard.\n        // While this is imprecise as discord changes the recommended number of shard every 1000 guilds it is good enough\n        // The alternative is to store the guild count for each shard and require the Guilds intent for `GUILD_CREATE` and `GUILD_DELETE` events\n        const percentage = (sessionInfo.shards / ((gateway.totalShards * 2500) / 1000)) * 100\n\n        // Less than necessary% being used so do nothing\n        if (percentage < gateway.resharding.shardsFullPercentage) {\n          gateway.logger.debug('[Resharding] Resharding not needed.')\n\n          return { needed: false, info: sessionInfo }\n        }\n\n        gateway.logger.info('[Resharding] Resharding is needed.')\n\n        return { needed: true, info: sessionInfo }\n      },\n      async reshard(info) {\n        gateway.logger.info(`[Resharding] Starting the reshard process. Previous total shards: ${gateway.totalShards}`)\n        // Set values on gateway\n        gateway.totalShards = info.shards\n        // Handles preparing mid sized bots for LBS\n        gateway.totalShards = gateway.calculateTotalShards()\n        // Set first shard id if provided in info\n        if (typeof info.firstShardId === 'number') gateway.firstShardId = info.firstShardId\n        // Set last shard id if provided in info\n        if (typeof info.lastShardId === 'number') gateway.lastShardId = info.lastShardId\n        gateway.logger.info(`[Resharding] Starting the reshard process. New total shards: ${gateway.totalShards}`)\n\n        // Resetting buckets\n        gateway.buckets.clear()\n        // Refilling buckets with new values\n        gateway.prepareBuckets()\n\n        // SPREAD THIS OUT TO DIFFERENT WORKERS TO BEGIN STARTING UP\n        gateway.buckets.forEach(async (bucket, bucketId) => {\n          for (const worker of bucket.workers) {\n            for (const shardId of worker.queue) {\n              await gateway.resharding.tellWorkerToPrepare(worker.id, shardId, bucketId)\n            }\n          }\n        })\n      },\n      async tellWorkerToPrepare(workerId, shardId, bucketId) {\n        gateway.logger.debug(`[Resharding] Telling worker to prepare. Worker: ${workerId} | Shard: ${shardId} | Bucket: ${bucketId}.`)\n        const shard = new Shard({\n          id: shardId,\n          connection: {\n            compress: gateway.compress,\n            transportCompression: gateway.transportCompression ?? null,\n            intents: gateway.intents,\n            properties: gateway.properties,\n            token: gateway.token,\n            totalShards: gateway.totalShards,\n            url: gateway.url,\n            version: gateway.version,\n          },\n          // Ignore events until we are ready\n          events: {\n            async message(_shard, payload) {\n              if (payload.t === 'READY') {\n                await gateway.resharding.updateGuildsShardId?.(\n                  (payload.d as DiscordReady).guilds.map((g) => g.id),\n                  shardId,\n                )\n              }\n            },\n          },\n          logger: gateway.logger,\n          requestIdentify: async () => await gateway.requestIdentify(shardId),\n          makePresence: gateway.makePresence,\n        })\n\n        if (gateway.preferSnakeCase) {\n          shard.forwardToBot = async (payload) => {\n            shard.events?.message?.(shard, payload)\n          }\n        }\n\n        gateway.resharding.shards.set(shardId, shard)\n\n        const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)\n        if (!bucket) return\n\n        await shard.identify().then(() => gateway.resharding.shardIsPending(shard))\n      },\n      async shardIsPending(shard) {\n        // Save this in pending at the moment, until all shards are online\n        gateway.resharding.pendingShards.set(shard.id, shard)\n        gateway.logger.debug(`[Resharding] Shard #${shard.id} is now pending.`)\n\n        // Check if all shards are now online.\n        if (gateway.lastShardId - gateway.firstShardId >= gateway.resharding.pendingShards.size) return\n\n        gateway.logger.info(`[Resharding] All shards are now online.`)\n\n        // New shards start processing events\n        for (const shard of gateway.resharding.shards.values()) {\n          for (const event in options.events) {\n            shard.events[event as keyof ShardEvents] = options.events[event as keyof ShardEvents] as (...args: unknown[]) => unknown\n          }\n        }\n\n        // Old shards stop processing events\n        for (const shard of gateway.shards.values()) {\n          const oldHandler = shard.events.message\n\n          // Change with spread operator to not affect new shards, as changing anything on shard.events will directly change options.events, which changes new shards' events\n          shard.events = {\n            ...shard.events,\n            message: async function (_, message) {\n              // Member checks need to continue but others can stop\n              if (message.t === 'GUILD_MEMBERS_CHUNK') {\n                oldHandler?.(shard, message)\n              }\n            },\n          }\n        }\n\n        gateway.logger.info(`[Resharding] Shutting down old shards.`)\n        // Close old shards\n        await gateway.shutdown(ShardSocketCloseCodes.Resharded, 'Resharded!', false)\n\n        gateway.logger.info(`[Resharding] Completed.`)\n\n        // Replace old shards\n        gateway.shards = new Collection(gateway.resharding.shards)\n\n        // Clear our collections and keep only one reference to the shards, the one in gateway.shards\n        gateway.resharding.shards.clear()\n        gateway.resharding.pendingShards.clear()\n      },\n    },\n\n    calculateTotalShards() {\n      // Bots under 100k servers do not have access to LBS.\n      if (gateway.totalShards < 100) {\n        gateway.logger.debug(`[Gateway] Calculating total shards: ${gateway.totalShards}`)\n        return gateway.totalShards\n      }\n\n      gateway.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 to get `totalShards` to be in a multiple of 16 so that we can prepare bots with 100k servers for LBS.\n            (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency),\n        ) * (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency)\n      )\n    },\n    calculateWorkerId(shardId) {\n      const workerId = options.spreadShardsInRoundRobin\n        ? shardId % gateway.totalWorkers\n        : Math.min(Math.floor(shardId / gateway.shardsPerWorker), gateway.totalWorkers - 1)\n      gateway.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        gateway.logger.debug(`[Gateway] Preparing buckets for concurrency: ${i}`)\n        gateway.buckets.set(i, {\n          workers: [],\n          leakyBucket: new LeakyBucket({\n            max: 1,\n            refillAmount: 1,\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        gateway.logger.debug(`[Gateway] Preparing buckets for shard: ${shardId}`)\n\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\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        // Get the worker id for this shard\n        const workerId = gateway.calculateWorkerId(shardId)\n        const worker = bucket.workers.find((w) => w.id === workerId)\n\n        // If this worker already exists, add the shard to its queue\n        if (worker) {\n          worker.queue.push(shardId)\n        } else {\n          bucket.workers.push({ id: workerId, queue: [shardId] })\n        }\n      }\n    },\n    async spawnShards() {\n      // Prepare the concurrency buckets\n      gateway.prepareBuckets()\n\n      const promises = [...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      // We use Promise.all so we can start all buckets at the same time\n      await Promise.all(promises)\n\n      // Check and reshard automatically if auto resharding is enabled.\n      if (gateway.resharding.enabled && gateway.resharding.checkInterval !== -1) {\n        // It is better to ensure there is always only one\n        clearInterval(gateway.resharding.checkIntervalId)\n\n        if (!gateway.resharding.getSessionInfo) {\n          gateway.resharding.enabled = false\n          gateway.logger.warn(\"[Resharding] Resharding is enabled but 'resharding.getSessionInfo()' was not provided. Disabling resharding.\")\n\n          return\n        }\n\n        gateway.resharding.checkIntervalId = setInterval(async () => {\n          const reshardingInfo = await gateway.resharding.checkIfReshardingIsNeeded()\n\n          if (reshardingInfo.needed && reshardingInfo.info) await gateway.resharding.reshard(reshardingInfo.info)\n        }, gateway.resharding.checkInterval)\n      }\n    },\n    async shutdown(code, reason, clearReshardingInterval = true) {\n      if (clearReshardingInterval) clearInterval(gateway.resharding.checkIntervalId)\n\n      await Promise.all(Array.from(gateway.shards.values()).map((shard) => shard.close(code, reason)))\n    },\n    async sendPayload(shardId, payload) {\n      const shard = gateway.shards.get(shardId)\n\n      if (!shard) {\n        throw new Error(`Shard (id: ${shardId} not found`)\n      }\n\n      await shard.send(payload)\n    },\n    async tellWorkerToIdentify(workerId, shardId, bucketId) {\n      gateway.logger.debug(`[Gateway] Tell worker #${workerId} to identify shard #${shardId} from bucket ${bucketId}`)\n      await gateway.identify(shardId)\n    },\n    async identify(shardId: number) {\n      let shard = this.shards.get(shardId)\n      gateway.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            transportCompression: gateway.transportCompression,\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          logger: this.logger,\n          requestIdentify: async () => await gateway.requestIdentify(shardId),\n          makePresence: gateway.makePresence,\n        })\n\n        if (this.preferSnakeCase) {\n          shard.forwardToBot = async (payload) => {\n            shard!.events.message?.(shard!, payload)\n          }\n        }\n\n        this.shards.set(shardId, shard)\n      }\n\n      await shard.identify()\n    },\n\n    async requestIdentify(shardId) {\n      gateway.logger.debug(`[Gateway] Shard #${shardId} requested an identify.`)\n\n      const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)\n\n      if (!bucket) {\n        throw new Error(\"Can't request identify for a shard that is not assigned to any bucket.\")\n      }\n\n      await bucket.leakyBucket.acquire()\n\n      gateway.logger.debug(`[Gateway] Approved identify request for Shard #${shardId}.`)\n    },\n\n    async kill(shardId: number) {\n      const shard = this.shards.get(shardId)\n      if (!shard) {\n        gateway.logger.debug(`[Gateway] Shard #${shardId} was requested to be killed, but the shard could not be found.`)\n        return\n      }\n\n      gateway.logger.debug(`[Gateway] Killing Shard #${shardId}`)\n      this.shards.delete(shardId)\n      await shard.shutdown()\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        gateway.logger.debug(`[Gateway] calculateShardId (1 shard)`)\n        return 0\n      }\n\n      gateway.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\n      gateway.logger.debug(`[Gateway] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`)\n\n      await gateway.sendPayload(shardId, {\n        op: GatewayOpcodes.VoiceStateUpdate,\n        d: {\n          guild_id: guildId.toString(),\n          channel_id: channelId.toString(),\n          self_mute: options?.selfMute ?? false,\n          self_deaf: options?.selfDeaf ?? true,\n        },\n      })\n    },\n\n    async editBotStatus(data) {\n      gateway.logger.debug(`[Gateway] editBotStatus data: ${JSON.stringify(data, jsonSafeReplacer)}`)\n\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      gateway.logger.debug(`[Gateway] editShardStatus shardId: ${shardId} -> data: ${JSON.stringify(data)}`)\n\n      await gateway.sendPayload(shardId, {\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    async requestMembers(guildId, options) {\n      const shardId = gateway.calculateShardId(guildId)\n\n      if (gateway.intents && (!options?.limit || options.limit > 1) && !(gateway.intents & GatewayIntents.GuildMembers))\n        throw new Error('Cannot fetch more then 1 member without the GUILD_MEMBERS intent')\n\n      gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> data: ${JSON.stringify(options)}`)\n\n      if (options?.userIds?.length) {\n        gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> setting user limit based on userIds length: ${options.userIds.length}`)\n\n        options.limit = options.userIds.length\n      }\n\n      if (!options?.nonce) {\n        let nonce = ''\n\n        while (!nonce || gateway.cache.requestMembers.pending.has(nonce)) {\n          nonce = randomBytes(16).toString('hex')\n        }\n\n        options ??= { limit: 0 }\n        options.nonce = nonce\n      }\n\n      const members = !gateway.cache.requestMembers.enabled\n        ? []\n        : new Promise<Camelize<DiscordMemberWithUser[]>>((resolve, reject) => {\n            // Should never happen.\n            if (!gateway.cache.requestMembers.enabled || !options?.nonce) {\n              reject(new Error(\"Can't request the members without the nonce or with the feature disabled.\"))\n              return\n            }\n\n            gateway.cache.requestMembers.pending.set(options.nonce, {\n              nonce: options.nonce,\n              resolve,\n              members: [],\n            })\n          })\n\n      await gateway.sendPayload(shardId, {\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      return await members\n    },\n\n    async leaveVoiceChannel(guildId) {\n      const shardId = gateway.calculateShardId(guildId)\n\n      gateway.logger.debug(`[Gateway] leaveVoiceChannel guildId: ${guildId} Shard ${shardId}`)\n\n      await gateway.sendPayload(shardId, {\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    async requestSoundboardSounds(guildIds) {\n      /**\n       * Discord will send the events for the guilds that are \"under the shard\" that sends the opcode.\n       * For this reason we need to group the ids with the shard the calculateShardId method gives\n       */\n\n      const map = new Map<number, BigString[]>()\n\n      for (const guildId of guildIds) {\n        const shardId = gateway.calculateShardId(guildId)\n\n        const ids = map.get(shardId) ?? []\n        map.set(shardId, ids)\n\n        ids.push(guildId)\n      }\n\n      await Promise.all(\n        [...map.entries()].map(([shardId, ids]) =>\n          gateway.sendPayload(shardId, {\n            op: GatewayOpcodes.RequestSoundboardSounds,\n            d: {\n              guild_ids: ids,\n            },\n          }),\n        ),\n      )\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  /**\n   * Whether to spread shards across workers in a round-robin manner.\n   *\n   * @remarks\n   * By default, shards are assigned to workers in contiguous blocks based on shardsPerWorker. If any shard is left over, it will be assigned to the last worker.\n   * This means that if you have 3 workers and 40 shards while shardsPerWorker is 10, the first worker will get shards 0-9, the second worker will get shards 10-19, and so on, the last worker will get shards 20-39.\n   *\n   * If this option is set to true, the shards will be assigned in a round-robin manner to spread more evenly across workers.\n   * For example, with 3 workers and 40 shards, the first shard will go to worker 0, the second shard to worker 1, the third shard to worker 2, the fourth shard to worker 0, and so on.\n   *\n   * @default false\n   */\n  spreadShardsInRoundRobin?: boolean\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  /** What transport compression should be used */\n  transportCompression?: TransportCompression | null\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    }\n  }\n  /**\n   * The logger that the gateway manager will use.\n   * @default logger // The logger exported by `@discordeno/utils`\n   */\n  logger?: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>\n  /**\n   * Make the presence for when the bot connects to the gateway\n   *\n   * @remarks\n   * This function will be called each time a Shard is going to identify\n   */\n  makePresence?: () => Promise<DiscordUpdatePresence | undefined>\n  /** Options related to resharding. */\n  resharding?: {\n    /**\n     * Whether or not automated resharding should be enabled.\n     * @default true\n     */\n    enabled: boolean\n    /**\n     * The % of how full a shard is when resharding should be triggered.\n     *\n     * @remarks\n     * We use discord recommended shard value to get an **approximation** of the shard full percentage to compare with this value so the bot may not reshard at the exact percentage provided but may reshard when it is a bit higher than the provided percentage.\n     * For accurate calculation, you may override the `checkIfReshardingIsNeeded` function\n     *\n     * @default 80 as in 80%\n     */\n    shardsFullPercentage: number\n    /**\n     * The interval in milliseconds, of how often to check whether resharding is needed and reshard automatically. Set to -1 to disable auto resharding.\n     * @default 28800000 (8 hours)\n     */\n    checkInterval: number\n    /** Handler to get shard count and other session info. */\n    getSessionInfo?: () => Promise<Camelize<DiscordGetGatewayBot>>\n    /** Handler to edit the shard id on any cached guilds. */\n    updateGuildsShardId?: (guildIds: string[], shardId: number) => Promise<void>\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      /** The bucket to queue the identifies. */\n      leakyBucket: LeakyBucket\n    }\n  >\n  /** The shards that are created. */\n  shards: Map<number, Shard>\n  /** The logger for the gateway manager. */\n  logger: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>\n  /** Everything related to resharding. */\n  resharding: CreateGatewayManagerOptions['resharding'] & {\n    /**\n     * The interval id of the check interval. This is used to clear the interval when the manager is shutdown.\n     */\n    checkIntervalId?: NodeJS.Timeout | undefined\n    /** Holds the shards that resharding has created. Once resharding is done, this replaces the gateway.shards */\n    shards: Collection<number, Shard>\n    /** Holds the pending shards that have been created and are pending all shards finish loading. */\n    pendingShards: Collection<number, Shard>\n    /** Handler to check if resharding is necessary. */\n    checkIfReshardingIsNeeded: () => Promise<{ needed: boolean; info?: Camelize<DiscordGetGatewayBot> }>\n    /** Handler to begin resharding. */\n    reshard: (info: Camelize<DiscordGetGatewayBot> & { firstShardId?: number; lastShardId?: number }) => Promise<void>\n    /** Handler to communicate to a worker that a shard needs to be created. */\n    tellWorkerToPrepare: (workerId: number, shardId: number, bucketId: number) => Promise<void>\n    /** Handler to alert the gateway that a shard(resharded) is online. It should now wait for all shards to be pending before shutting off old shards. */\n    shardIsPending: (shard: Shard) => Promise<void>\n  }\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, clearReshardingInterval?: boolean) => Promise<void>\n  sendPayload: (shardId: number, payload: ShardSocketRequest) => Promise<void>\n  /**\n   * Allows users to hook in and change to communicate to different workers across different servers or anything they like.\n   * For example using redis pubsub to talk to other servers.\n   *\n   * @remarks\n   * This should wait for the worker to have identified the shard before resolving the returned promise\n   */\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 nothing\n   */\n  editBotStatus: (data: DiscordUpdatePresence) => 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 nothing\n   */\n  editShardStatus: (shardId: number, data: DiscordUpdatePresence) => Promise<void>\n  /**\n   * Fetches the list of members for a guild over the gateway. If `gateway.cache.requestMembers.enabled` is not set, this function will return an empty array and you'll have to handle the `GUILD_MEMBERS_CHUNK` events yourself.\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<DiscordMemberWithUser[]>>\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   * Used to request soundboard sounds for a list of guilds.\n   *\n   * This function sends multiple (see remarks) _Request Soundboard Sounds_ gateway command over the gateway behind the scenes.\n   *\n   * @param guildIds - The guilds to get the sounds from\n   *\n   * @remarks\n   * Fires a _Soundboard Sounds_ gateway event.\n   *\n   * ⚠️ Discord will send the _Soundboard Sounds_ for each of the guild ids\n   * however you may not receive the same number of events as the ids passed to _Request Soundboard Sounds_ for one of the following reasons:\n   * - The bot is not in the server provided\n   * - The shard the message has been sent from does not receive events for the specified guild\n   *\n   * To avoid this Discordeno will automatically try to group the ids based on what shard they will need to be sent, but this involves sending multiple messages in multiple shards\n   *\n   * @see {@link https://discord.com/developers/docs/topics/gateway-events#request-soundboard-sounds}\n   */\n  requestSoundboardSounds: (guildIds: BigString[]) => Promise<void>\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 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<DiscordMemberWithUser[]> | PromiseLike<Camelize<DiscordMemberWithUser[]>>) => void\n  /** The members that have already arrived for this request. */\n  members: DiscordMemberWithUser[]\n}\n"],"names":["randomBytes","GatewayIntents","GatewayOpcodes","Collection","LeakyBucket","jsonSafeReplacer","logger","Shard","ShardSocketCloseCodes","createGatewayManager","options","connectionOptions","connection","url","shards","sessionStartLimit","maxConcurrency","remaining","total","resetAfter","gateway","events","compress","transportCompression","intents","properties","os","process","platform","browser","device","token","version","totalShards","lastShardId","firstShardId","totalWorkers","shardsPerWorker","spawnShardDelay","spreadShardsInRoundRobin","preferSnakeCase","Map","buckets","cache","requestMembers","enabled","pending","makePresence","Promise","resolve","undefined","resharding","shardsFullPercentage","checkInterval","pendingShards","getSessionInfo","updateGuildsShardId","checkIfReshardingIsNeeded","debug","needed","Error","sessionInfo","JSON","stringify","info","percentage","reshard","calculateTotalShards","clear","prepareBuckets","forEach","bucket","bucketId","worker","workers","shardId","queue","tellWorkerToPrepare","id","workerId","shard","message","_shard","payload","t","d","guilds","map","g","requestIdentify","forwardToBot","set","get","identify","then","shardIsPending","size","values","event","oldHandler","_","shutdown","Resharded","Math","ceil","calculateWorkerId","min","floor","i","leakyBucket","max","refillAmount","refillInterval","find","w","push","spawnShards","promises","entries","tellWorkerToIdentify","all","clearInterval","checkIntervalId","warn","setInterval","reshardingInfo","code","reason","clearReshardingInterval","Array","from","close","sendPayload","send","acquire","kill","delete","calculateShardId","guildId","Number","BigInt","joinVoiceChannel","channelId","op","VoiceStateUpdate","guild_id","toString","channel_id","self_mute","selfMute","self_deaf","selfDeaf","editBotStatus","data","editShardStatus","PresenceUpdate","since","afk","activities","status","limit","GuildMembers","userIds","length","nonce","has","members","reject","RequestGuildMembers","query","presences","user_ids","leaveVoiceChannel","requestSoundboardSounds","guildIds","ids","RequestSoundboardSounds","guild_ids"],"mappings":"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"}
463
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/manager.ts"],"sourcesContent":["import { randomBytes } from 'node:crypto'\nimport {\n  type AtLeastOne,\n  type BigString,\n  type Camelize,\n  type DiscordGetGatewayBot,\n  type DiscordMemberWithUser,\n  type DiscordReady,\n  type DiscordUpdatePresence,\n  GatewayIntents,\n  GatewayOpcodes,\n  type RequestGuildMembers,\n} from '@discordeno/types'\nimport { Collection, LeakyBucket, jsonSafeReplacer, logger } from '@discordeno/utils'\nimport Shard from './Shard.js'\nimport { type ShardEvents, ShardSocketCloseCodes, type ShardSocketRequest, type TransportCompression, type 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    transportCompression: options.transportCompression ?? null,\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    spreadShardsInRoundRobin: options.spreadShardsInRoundRobin ?? false,\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    logger: options.logger ?? logger,\n    makePresence: options.makePresence ?? (() => Promise.resolve(undefined)),\n    resharding: {\n      enabled: options.resharding?.enabled ?? true,\n      shardsFullPercentage: options.resharding?.shardsFullPercentage ?? 80,\n      checkInterval: options.resharding?.checkInterval ?? 28800000, // 8 hours\n      shards: new Map(),\n      getSessionInfo: options.resharding?.getSessionInfo,\n      updateGuildsShardId: options.resharding?.updateGuildsShardId,\n      async checkIfReshardingIsNeeded() {\n        gateway.logger.debug('[Resharding] Checking if resharding is needed.')\n\n        if (!gateway.resharding.enabled) {\n          gateway.logger.debug('[Resharding] Resharding is disabled.')\n\n          return { needed: false }\n        }\n\n        if (!gateway.resharding.getSessionInfo) {\n          throw new Error(\"[Resharding] Resharding is enabled but no 'resharding.getSessionInfo()' is not provided.\")\n        }\n\n        gateway.logger.debug('[Resharding] Resharding is enabled.')\n\n        const sessionInfo = await gateway.resharding.getSessionInfo()\n\n        gateway.logger.debug(`[Resharding] Session info retrieved: ${JSON.stringify(sessionInfo)}`)\n\n        // Don't have enough identify limits to try resharding\n        if (sessionInfo.sessionStartLimit.remaining < sessionInfo.shards) {\n          gateway.logger.debug('[Resharding] Not enough session start limits left to reshard.')\n\n          return { needed: false, info: sessionInfo }\n        }\n\n        gateway.logger.debug('[Resharding] Able to reshard, checking whether necessary now.')\n\n        // 2500 is the max amount of guilds a single shard can handle\n        // 1000 is the amount of guilds discord uses to determine how many shards to recommend.\n        // This algo helps check if your bot has grown enough to reshard.\n        // While this is imprecise as discord changes the recommended number of shard every 1000 guilds it is good enough\n        // The alternative is to store the guild count for each shard and require the Guilds intent for `GUILD_CREATE` and `GUILD_DELETE` events\n        const percentage = (sessionInfo.shards / ((gateway.totalShards * 2500) / 1000)) * 100\n\n        // Less than necessary% being used so do nothing\n        if (percentage < gateway.resharding.shardsFullPercentage) {\n          gateway.logger.debug('[Resharding] Resharding not needed.')\n\n          return { needed: false, info: sessionInfo }\n        }\n\n        gateway.logger.info('[Resharding] Resharding is needed.')\n\n        return { needed: true, info: sessionInfo }\n      },\n      async reshard(info) {\n        gateway.logger.info(`[Resharding] Starting the reshard process. Previous total shards: ${gateway.totalShards}`)\n        // Set values on gateway\n        gateway.totalShards = info.shards\n        // Handles preparing mid sized bots for LBS\n        gateway.totalShards = gateway.calculateTotalShards()\n        // Set first shard id if provided in info\n        if (typeof info.firstShardId === 'number') gateway.firstShardId = info.firstShardId\n        // Set last shard id if provided in info\n        if (typeof info.lastShardId === 'number') gateway.lastShardId = info.lastShardId\n        // If we didn't get any lastShardId, we assume all the shards are to be used\n        else gateway.lastShardId = gateway.totalShards - 1\n        gateway.logger.info(`[Resharding] Starting the reshard process. New total shards: ${gateway.totalShards}`)\n\n        // Resetting buckets\n        gateway.buckets.clear()\n        // Refilling buckets with new values\n        gateway.prepareBuckets()\n\n        // Call all the buckets and tell their workers & shards to identify\n        const promises = Array.from(gateway.buckets.entries()).map(async ([bucketId, bucket]) => {\n          for (const worker of bucket.workers) {\n            for (const shardId of worker.queue) {\n              await gateway.resharding.tellWorkerToPrepare(worker.id, shardId, bucketId)\n            }\n          }\n        })\n\n        await Promise.all(promises)\n\n        gateway.logger.info(`[Resharding] All shards are now online.`)\n\n        await gateway.resharding.onReshardingSwitch()\n      },\n      async tellWorkerToPrepare(workerId, shardId, bucketId) {\n        gateway.logger.debug(`[Resharding] Telling worker to prepare. Worker: ${workerId} | Shard: ${shardId} | Bucket: ${bucketId}.`)\n        const shard = new Shard({\n          id: shardId,\n          connection: {\n            compress: gateway.compress,\n            transportCompression: gateway.transportCompression ?? null,\n            intents: gateway.intents,\n            properties: gateway.properties,\n            token: gateway.token,\n            totalShards: gateway.totalShards,\n            url: gateway.url,\n            version: gateway.version,\n          },\n          events: {\n            async message(_shard, payload) {\n              // Ignore all events until we swich from the old shards to the new ones.\n              if (payload.t === 'READY') {\n                await gateway.resharding.updateGuildsShardId?.(\n                  (payload.d as DiscordReady).guilds.map((g) => g.id),\n                  shardId,\n                )\n              }\n            },\n          },\n          logger: gateway.logger,\n          requestIdentify: async () => await gateway.requestIdentify(shardId),\n          makePresence: gateway.makePresence,\n        })\n\n        if (gateway.preferSnakeCase) {\n          shard.forwardToBot = async (payload) => {\n            shard.events?.message?.(shard, payload)\n          }\n        }\n\n        gateway.resharding.shards.set(shardId, shard)\n\n        await shard.identify()\n\n        gateway.logger.debug(`[Resharding] Shard #${shardId} identified.`)\n      },\n      async onReshardingSwitch() {\n        gateway.logger.debug(`[Resharding] Making the switch from the old shards to the new ones.`)\n\n        // Move the events from the old shards to the new ones\n        for (const shard of gateway.resharding.shards.values()) {\n          shard.events = options.events ?? {}\n        }\n\n        // Old shards stop processing events\n        for (const shard of gateway.shards.values()) {\n          const oldHandler = shard.events.message\n\n          // Change with spread operator to not affect new shards, as changing anything on shard.events will directly change options.events, which changes new shards' events\n          shard.events = {\n            ...shard.events,\n            message: async function (_, message) {\n              // Member checks need to continue but others can stop\n              if (message.t === 'GUILD_MEMBERS_CHUNK') {\n                oldHandler?.(shard, message)\n              }\n            },\n          }\n        }\n\n        gateway.logger.info(`[Resharding] Shutting down old shards.`)\n        await gateway.shutdown(ShardSocketCloseCodes.Resharded, 'Resharded!', false)\n\n        gateway.logger.info(`[Resharding] Completed.`)\n        gateway.shards = new Map(gateway.resharding.shards)\n        gateway.resharding.shards.clear()\n      },\n    },\n\n    calculateTotalShards() {\n      // Bots under 100k servers do not have access to LBS.\n      if (gateway.totalShards < 100) {\n        gateway.logger.debug(`[Gateway] Calculating total shards: ${gateway.totalShards}`)\n        return gateway.totalShards\n      }\n\n      gateway.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 to get `totalShards` to be in a multiple of 16 so that we can prepare bots with 100k servers for LBS.\n            (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency),\n        ) * (gateway.connection.sessionStartLimit.maxConcurrency === 1 ? 16 : gateway.connection.sessionStartLimit.maxConcurrency)\n      )\n    },\n    calculateWorkerId(shardId) {\n      const workerId = options.spreadShardsInRoundRobin\n        ? shardId % gateway.totalWorkers\n        : Math.min(Math.floor(shardId / gateway.shardsPerWorker), gateway.totalWorkers - 1)\n      gateway.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        gateway.logger.debug(`[Gateway] Preparing buckets for concurrency: ${i}`)\n        gateway.buckets.set(i, {\n          workers: [],\n          leakyBucket: new LeakyBucket({\n            max: 1,\n            refillAmount: 1,\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        gateway.logger.debug(`[Gateway] Preparing buckets for shard: ${shardId}`)\n\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\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        // Get the worker id for this shard\n        const workerId = gateway.calculateWorkerId(shardId)\n        const worker = bucket.workers.find((w) => w.id === workerId)\n\n        // If this worker already exists, add the shard to its queue\n        if (worker) {\n          worker.queue.push(shardId)\n        } else {\n          bucket.workers.push({ id: workerId, queue: [shardId] })\n        }\n      }\n    },\n    async spawnShards() {\n      // Prepare the concurrency buckets\n      gateway.prepareBuckets()\n\n      const promises = [...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      // We use Promise.all so we can start all buckets at the same time\n      await Promise.all(promises)\n\n      // Check and reshard automatically if auto resharding is enabled.\n      if (gateway.resharding.enabled && gateway.resharding.checkInterval !== -1) {\n        // It is better to ensure there is always only one\n        clearInterval(gateway.resharding.checkIntervalId)\n\n        if (!gateway.resharding.getSessionInfo) {\n          gateway.resharding.enabled = false\n          gateway.logger.warn(\"[Resharding] Resharding is enabled but 'resharding.getSessionInfo()' was not provided. Disabling resharding.\")\n\n          return\n        }\n\n        gateway.resharding.checkIntervalId = setInterval(async () => {\n          const reshardingInfo = await gateway.resharding.checkIfReshardingIsNeeded()\n\n          if (reshardingInfo.needed && reshardingInfo.info) await gateway.resharding.reshard(reshardingInfo.info)\n        }, gateway.resharding.checkInterval)\n      }\n    },\n    async shutdown(code, reason, clearReshardingInterval = true) {\n      if (clearReshardingInterval) clearInterval(gateway.resharding.checkIntervalId)\n\n      await Promise.all(Array.from(gateway.shards.values()).map((shard) => shard.close(code, reason)))\n    },\n    async sendPayload(shardId, payload) {\n      const shard = gateway.shards.get(shardId)\n\n      if (!shard) {\n        throw new Error(`Shard (id: ${shardId} not found`)\n      }\n\n      await shard.send(payload)\n    },\n    async tellWorkerToIdentify(workerId, shardId, bucketId) {\n      gateway.logger.debug(`[Gateway] Tell worker #${workerId} to identify shard #${shardId} from bucket ${bucketId}`)\n      await gateway.identify(shardId)\n    },\n    async identify(shardId: number) {\n      let shard = this.shards.get(shardId)\n      gateway.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            transportCompression: gateway.transportCompression,\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          logger: this.logger,\n          requestIdentify: async () => await gateway.requestIdentify(shardId),\n          makePresence: gateway.makePresence,\n        })\n\n        if (this.preferSnakeCase) {\n          shard.forwardToBot = async (payload) => {\n            shard!.events.message?.(shard!, payload)\n          }\n        }\n\n        this.shards.set(shardId, shard)\n      }\n\n      await shard.identify()\n    },\n\n    async requestIdentify(shardId) {\n      gateway.logger.debug(`[Gateway] Shard #${shardId} requested an identify.`)\n\n      const bucket = gateway.buckets.get(shardId % gateway.connection.sessionStartLimit.maxConcurrency)\n\n      if (!bucket) {\n        throw new Error(\"Can't request identify for a shard that is not assigned to any bucket.\")\n      }\n\n      await bucket.leakyBucket.acquire()\n\n      gateway.logger.debug(`[Gateway] Approved identify request for Shard #${shardId}.`)\n    },\n\n    async kill(shardId: number) {\n      const shard = this.shards.get(shardId)\n      if (!shard) {\n        gateway.logger.debug(`[Gateway] Shard #${shardId} was requested to be killed, but the shard could not be found.`)\n        return\n      }\n\n      gateway.logger.debug(`[Gateway] Killing Shard #${shardId}`)\n      this.shards.delete(shardId)\n      await shard.shutdown()\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        gateway.logger.debug(`[Gateway] calculateShardId (1 shard)`)\n        return 0\n      }\n\n      gateway.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\n      gateway.logger.debug(`[Gateway] joinVoiceChannel guildId: ${guildId} channelId: ${channelId}`)\n\n      await gateway.sendPayload(shardId, {\n        op: GatewayOpcodes.VoiceStateUpdate,\n        d: {\n          guild_id: guildId.toString(),\n          channel_id: channelId.toString(),\n          self_mute: options?.selfMute ?? false,\n          self_deaf: options?.selfDeaf ?? true,\n        },\n      })\n    },\n\n    async editBotStatus(data) {\n      gateway.logger.debug(`[Gateway] editBotStatus data: ${JSON.stringify(data, jsonSafeReplacer)}`)\n\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      gateway.logger.debug(`[Gateway] editShardStatus shardId: ${shardId} -> data: ${JSON.stringify(data)}`)\n\n      await gateway.sendPayload(shardId, {\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    async requestMembers(guildId, options) {\n      const shardId = gateway.calculateShardId(guildId)\n\n      if (gateway.intents && (!options?.limit || options.limit > 1) && !(gateway.intents & GatewayIntents.GuildMembers))\n        throw new Error('Cannot fetch more then 1 member without the GUILD_MEMBERS intent')\n\n      gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> data: ${JSON.stringify(options)}`)\n\n      if (options?.userIds?.length) {\n        gateway.logger.debug(`[Gateway] requestMembers guildId: ${guildId} -> setting user limit based on userIds length: ${options.userIds.length}`)\n\n        options.limit = options.userIds.length\n      }\n\n      if (!options?.nonce) {\n        let nonce = ''\n\n        while (!nonce || gateway.cache.requestMembers.pending.has(nonce)) {\n          nonce = randomBytes(16).toString('hex')\n        }\n\n        options ??= { limit: 0 }\n        options.nonce = nonce\n      }\n\n      const members = !gateway.cache.requestMembers.enabled\n        ? []\n        : new Promise<Camelize<DiscordMemberWithUser[]>>((resolve, reject) => {\n            // Should never happen.\n            if (!gateway.cache.requestMembers.enabled || !options?.nonce) {\n              reject(new Error(\"Can't request the members without the nonce or with the feature disabled.\"))\n              return\n            }\n\n            gateway.cache.requestMembers.pending.set(options.nonce, {\n              nonce: options.nonce,\n              resolve,\n              members: [],\n            })\n          })\n\n      await gateway.sendPayload(shardId, {\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      return await members\n    },\n\n    async leaveVoiceChannel(guildId) {\n      const shardId = gateway.calculateShardId(guildId)\n\n      gateway.logger.debug(`[Gateway] leaveVoiceChannel guildId: ${guildId} Shard ${shardId}`)\n\n      await gateway.sendPayload(shardId, {\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    async requestSoundboardSounds(guildIds) {\n      /**\n       * Discord will send the events for the guilds that are \"under the shard\" that sends the opcode.\n       * For this reason we need to group the ids with the shard the calculateShardId method gives\n       */\n\n      const map = new Map<number, BigString[]>()\n\n      for (const guildId of guildIds) {\n        const shardId = gateway.calculateShardId(guildId)\n\n        const ids = map.get(shardId) ?? []\n        map.set(shardId, ids)\n\n        ids.push(guildId)\n      }\n\n      await Promise.all(\n        [...map.entries()].map(([shardId, ids]) =>\n          gateway.sendPayload(shardId, {\n            op: GatewayOpcodes.RequestSoundboardSounds,\n            d: {\n              guild_ids: ids,\n            },\n          }),\n        ),\n      )\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  /**\n   * Whether to spread shards across workers in a round-robin manner.\n   *\n   * @remarks\n   * By default, shards are assigned to workers in contiguous blocks based on shardsPerWorker. If any shard is left over, it will be assigned to the last worker.\n   * This means that if you have 3 workers and 40 shards while shardsPerWorker is 10, the first worker will get shards 0-9, the second worker will get shards 10-19, and so on, the last worker will get shards 20-39.\n   *\n   * If this option is set to true, the shards will be assigned in a round-robin manner to spread more evenly across workers.\n   * For example, with 3 workers and 40 shards, the first shard will go to worker 0, the second shard to worker 1, the third shard to worker 2, the fourth shard to worker 0, and so on.\n   *\n   * @default false\n   */\n  spreadShardsInRoundRobin?: boolean\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  /** What transport compression should be used */\n  transportCompression?: TransportCompression | null\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    }\n  }\n  /**\n   * The logger that the gateway manager will use.\n   * @default logger // The logger exported by `@discordeno/utils`\n   */\n  logger?: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>\n  /**\n   * Make the presence for when the bot connects to the gateway\n   *\n   * @remarks\n   * This function will be called each time a Shard is going to identify\n   */\n  makePresence?: () => Promise<DiscordUpdatePresence | undefined>\n  /** Options related to resharding. */\n  resharding?: {\n    /**\n     * Whether or not automated resharding should be enabled.\n     * @default true\n     */\n    enabled: boolean\n    /**\n     * The % of how full a shard is when resharding should be triggered.\n     *\n     * @remarks\n     * We use discord recommended shard value to get an **approximation** of the shard full percentage to compare with this value so the bot may not reshard at the exact percentage provided but may reshard when it is a bit higher than the provided percentage.\n     * For accurate calculation, you may override the `checkIfReshardingIsNeeded` function\n     *\n     * @default 80 as in 80%\n     */\n    shardsFullPercentage: number\n    /**\n     * The interval in milliseconds, of how often to check whether resharding is needed and reshard automatically. Set to -1 to disable auto resharding.\n     * @default 28800000 (8 hours)\n     */\n    checkInterval: number\n    /** Handler to get shard count and other session info. */\n    getSessionInfo?: () => Promise<Camelize<DiscordGetGatewayBot>>\n    /** Handler to edit the shard id on any cached guilds. */\n    updateGuildsShardId?: (guildIds: string[], shardId: number) => Promise<void>\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      /** The bucket to queue the identifies. */\n      leakyBucket: LeakyBucket\n    }\n  >\n  /** The shards that are created. */\n  shards: Map<number, Shard>\n  /** The logger for the gateway manager. */\n  logger: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>\n  /** Everything related to resharding. */\n  resharding: CreateGatewayManagerOptions['resharding'] & {\n    /** The interval id of the check interval. This is used to clear the interval when the manager is shutdown. */\n    checkIntervalId?: NodeJS.Timeout | undefined\n    /** Holds the shards that resharding has created. Once resharding is done, this replaces the gateway.shards */\n    shards: Map<number, Shard>\n    /** Handler to check if resharding is necessary. */\n    checkIfReshardingIsNeeded: () => Promise<{ needed: boolean; info?: Camelize<DiscordGetGatewayBot> }>\n    /**\n     * Handler to begin resharding.\n     *\n     * @remarks\n     * This function will resolve once the resharding is done.\n     * So when all the calls to {@link tellWorkerToPrepare} and {@link onReshardingSwitch} are done.\n     */\n    reshard: (info: Camelize<DiscordGetGatewayBot> & { firstShardId?: number; lastShardId?: number }) => Promise<void>\n    /**\n     * Handler to communicate to a worker that it needs to spawn a new shard and identify it for the resharding.\n     *\n     * @remarks\n     * This handler works in the same way as the {@link tellWorkerToIdentify} handler.\n     * So you should wait for the worker to have identified the shard before resolving the promise\n     */\n    tellWorkerToPrepare: (workerId: number, shardId: number, bucketId: number) => Promise<void>\n    /**\n     * Handle called when all the workers have finished preparing for the resharding.\n     *\n     * This should make the new resharded shards become the active ones and shutdown the old ones\n     */\n    onReshardingSwitch: () => Promise<void>\n  }\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, clearReshardingInterval?: boolean) => Promise<void>\n  sendPayload: (shardId: number, payload: ShardSocketRequest) => Promise<void>\n  /**\n   * Allows users to hook in and change to communicate to different workers across different servers or anything they like.\n   * For example using redis pubsub to talk to other servers.\n   *\n   * @remarks\n   * This should wait for the worker to have identified the shard before resolving the returned promise\n   */\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 nothing\n   */\n  editBotStatus: (data: DiscordUpdatePresence) => 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 nothing\n   */\n  editShardStatus: (shardId: number, data: DiscordUpdatePresence) => Promise<void>\n  /**\n   * Fetches the list of members for a guild over the gateway. If `gateway.cache.requestMembers.enabled` is not set, this function will return an empty array and you'll have to handle the `GUILD_MEMBERS_CHUNK` events yourself.\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<DiscordMemberWithUser[]>>\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   * Used to request soundboard sounds for a list of guilds.\n   *\n   * This function sends multiple (see remarks) _Request Soundboard Sounds_ gateway command over the gateway behind the scenes.\n   *\n   * @param guildIds - The guilds to get the sounds from\n   *\n   * @remarks\n   * Fires a _Soundboard Sounds_ gateway event.\n   *\n   * ⚠️ Discord will send the _Soundboard Sounds_ for each of the guild ids\n   * however you may not receive the same number of events as the ids passed to _Request Soundboard Sounds_ for one of the following reasons:\n   * - The bot is not in the server provided\n   * - The shard the message has been sent from does not receive events for the specified guild\n   *\n   * To avoid this Discordeno will automatically try to group the ids based on what shard they will need to be sent, but this involves sending multiple messages in multiple shards\n   *\n   * @see {@link https://discord.com/developers/docs/topics/gateway-events#request-soundboard-sounds}\n   */\n  requestSoundboardSounds: (guildIds: BigString[]) => Promise<void>\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 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<DiscordMemberWithUser[]> | PromiseLike<Camelize<DiscordMemberWithUser[]>>) => void\n  /** The members that have already arrived for this request. */\n  members: DiscordMemberWithUser[]\n}\n"],"names":["randomBytes","GatewayIntents","GatewayOpcodes","Collection","LeakyBucket","jsonSafeReplacer","logger","Shard","ShardSocketCloseCodes","createGatewayManager","options","connectionOptions","connection","url","shards","sessionStartLimit","maxConcurrency","remaining","total","resetAfter","gateway","events","compress","transportCompression","intents","properties","os","process","platform","browser","device","token","version","totalShards","lastShardId","firstShardId","totalWorkers","shardsPerWorker","spawnShardDelay","spreadShardsInRoundRobin","preferSnakeCase","Map","buckets","cache","requestMembers","enabled","pending","makePresence","Promise","resolve","undefined","resharding","shardsFullPercentage","checkInterval","getSessionInfo","updateGuildsShardId","checkIfReshardingIsNeeded","debug","needed","Error","sessionInfo","JSON","stringify","info","percentage","reshard","calculateTotalShards","clear","prepareBuckets","promises","Array","from","entries","map","bucketId","bucket","worker","workers","shardId","queue","tellWorkerToPrepare","id","all","onReshardingSwitch","workerId","shard","message","_shard","payload","t","d","guilds","g","requestIdentify","forwardToBot","set","identify","values","oldHandler","_","shutdown","Resharded","Math","ceil","calculateWorkerId","min","floor","i","leakyBucket","max","refillAmount","refillInterval","get","find","w","push","spawnShards","tellWorkerToIdentify","clearInterval","checkIntervalId","warn","setInterval","reshardingInfo","code","reason","clearReshardingInterval","close","sendPayload","send","acquire","kill","delete","calculateShardId","guildId","Number","BigInt","joinVoiceChannel","channelId","op","VoiceStateUpdate","guild_id","toString","channel_id","self_mute","selfMute","self_deaf","selfDeaf","editBotStatus","data","editShardStatus","PresenceUpdate","since","afk","activities","status","limit","GuildMembers","userIds","length","nonce","has","members","reject","RequestGuildMembers","query","presences","user_ids","leaveVoiceChannel","requestSoundboardSounds","guildIds","ids","RequestSoundboardSounds","guild_ids"],"mappings":"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"}
@@ -164,28 +164,40 @@ export interface GatewayManager extends Required<CreateGatewayManagerOptions> {
164
164
  logger: Pick<typeof logger, 'debug' | 'info' | 'warn' | 'error' | 'fatal'>;
165
165
  /** Everything related to resharding. */
166
166
  resharding: CreateGatewayManagerOptions['resharding'] & {
167
- /**
168
- * The interval id of the check interval. This is used to clear the interval when the manager is shutdown.
169
- */
167
+ /** The interval id of the check interval. This is used to clear the interval when the manager is shutdown. */
170
168
  checkIntervalId?: NodeJS.Timeout | undefined;
171
169
  /** Holds the shards that resharding has created. Once resharding is done, this replaces the gateway.shards */
172
- shards: Collection<number, Shard>;
173
- /** Holds the pending shards that have been created and are pending all shards finish loading. */
174
- pendingShards: Collection<number, Shard>;
170
+ shards: Map<number, Shard>;
175
171
  /** Handler to check if resharding is necessary. */
176
172
  checkIfReshardingIsNeeded: () => Promise<{
177
173
  needed: boolean;
178
174
  info?: Camelize<DiscordGetGatewayBot>;
179
175
  }>;
180
- /** Handler to begin resharding. */
176
+ /**
177
+ * Handler to begin resharding.
178
+ *
179
+ * @remarks
180
+ * This function will resolve once the resharding is done.
181
+ * So when all the calls to {@link tellWorkerToPrepare} and {@link onReshardingSwitch} are done.
182
+ */
181
183
  reshard: (info: Camelize<DiscordGetGatewayBot> & {
182
184
  firstShardId?: number;
183
185
  lastShardId?: number;
184
186
  }) => Promise<void>;
185
- /** Handler to communicate to a worker that a shard needs to be created. */
187
+ /**
188
+ * Handler to communicate to a worker that it needs to spawn a new shard and identify it for the resharding.
189
+ *
190
+ * @remarks
191
+ * This handler works in the same way as the {@link tellWorkerToIdentify} handler.
192
+ * So you should wait for the worker to have identified the shard before resolving the promise
193
+ */
186
194
  tellWorkerToPrepare: (workerId: number, shardId: number, bucketId: number) => Promise<void>;
187
- /** Handler to alert the gateway that a shard(resharded) is online. It should now wait for all shards to be pending before shutting off old shards. */
188
- shardIsPending: (shard: Shard) => Promise<void>;
195
+ /**
196
+ * Handle called when all the workers have finished preparing for the resharding.
197
+ *
198
+ * This should make the new resharded shards become the active ones and shutdown the old ones
199
+ */
200
+ onReshardingSwitch: () => Promise<void>;
189
201
  };
190
202
  /** Determine max number of shards to use based upon the max concurrency. */
191
203
  calculateTotalShards: () => number;
@@ -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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@discordeno/gateway",
3
- "version": "21.0.1-next.b6bfaa3",
3
+ "version": "21.0.1-next.d9831f6",
4
4
  "main": "./dist/esm/index.js",
5
5
  "types": "./dist/types/index.d.ts",
6
6
  "exports": {
@@ -34,8 +34,8 @@
34
34
  "test:test-type": "tsc --project tests/tsconfig.json"
35
35
  },
36
36
  "dependencies": {
37
- "@discordeno/types": "21.0.1-next.b6bfaa3",
38
- "@discordeno/utils": "21.0.1-next.b6bfaa3",
37
+ "@discordeno/types": "21.0.1-next.d9831f6",
38
+ "@discordeno/utils": "21.0.1-next.d9831f6",
39
39
  "ws": "^8.18.1"
40
40
  },
41
41
  "optionalDependencies": {
@@ -44,14 +44,14 @@
44
44
  "devDependencies": {
45
45
  "@biomejs/biome": "^1.9.4",
46
46
  "@swc/cli": "^0.5.2",
47
- "@swc/core": "^1.12.9",
47
+ "@swc/core": "^1.13.3",
48
48
  "@types/chai": "^5.2.2",
49
49
  "@types/mocha": "^10.0.10",
50
- "@types/node": "^24.0.1",
50
+ "@types/node": "^24.1.0",
51
51
  "@types/sinon": "^17.0.3",
52
52
  "@types/ws": "^8.5.14",
53
53
  "c8": "^10.1.3",
54
- "chai": "^5.2.0",
54
+ "chai": "^5.2.1",
55
55
  "mocha": "patch:mocha@npm%3A11.0.2#~/.yarn/patches/mocha-npm-11.0.2-b5d6b95284.patch",
56
56
  "sinon": "^21.0.0",
57
57
  "ts-node": "^10.9.2",