Cluster your Discord Bot to save a lot of resources. The ultimate Package, which combines the Sharding Manager & Internal Sharding.
One first package which combines sharding manager & internal sharding to save a lot of resources, which allows clustering!
In other words: “Mixing both: if you need x
shards for n
process!”
NEW: Clustering Support for all JS Libraries
If you are interested in auto-scaling & cross-hosting on other machines, check out this package npmjs.com/discord-cross-hosting
Private Community for Verified Bot Developers.
Meet big bot and small bot developers and have a nice exchange…
The sharding manager is very heavy and uses more than 300MB per shard during light usage, while internal sharding uses just 20% of it. Internal sharding reaches its’ limits at 14000 guilds and becomes slow when your bot gets bigger.
Your only solution becomes converting to the sharding manager. That’s why this new package will solve all your problems (tested by many bots with 20-170k guilds), because it spawns shards, which has internal shards. You can save up to 60% on resources!
ClusterClient
s.request()
, .reply()
, .send()
manager.queue.next(), .stop(), .resume()
Strings
& Functions with Context
support on .broadcastEval()
.broadcastEval()
to prevent memory leaksShard/Cluster
managing and cross-host communication (.broadcastEval()
, IPC
)There are clusters/master shards, which behave like regular shards in the sharding manager and spawns clusters in addition to internal shards. You do not have to spawn as many regular Shards (master shards), which can be replaced with internal shards.
“for process n
, n
internal shards”
Example: A Discord bot with 4000 guilds
Normally we would spawn 4 shards with the Sharding Manager (~4 x 300MB memory
), but in this case we start with 2 clusters/master shards, which spawns 2 internal shards ==> We just saved 2 shards in comparison to the regular Sharding Manager (~2 x 300MB memory
).
If you need help, feel free to join our Discord server. ☺
npm i discord-hybrid-sharding
------ or ---------------------
yarn add discord-hybrid-sharding
Strings
and Functions
with context
are supported in .broadcastEval()
.spawn({ amount: 20, timeout: -1 })
First, add the module into your project (into your shard/cluster file).
Filename: Cluster.js
const Cluster = require('discord-hybrid-sharding');
const manager = new Cluster.Manager(`${__dirname}/bot.js`, {
totalShards: 7, // or 'auto'
/// Check below for more options
shardsPerClusters: 2,
// totalClusters: 7,
mode: 'process', // you can also choose "worker"
token: 'YOUR_TOKEN',
});
manager.on('clusterCreate', cluster => console.log(`Launched Cluster ${cluster.id}`));
manager.spawn({ timeout: -1 });
After that, insert the code below into your bot.js
file
const Cluster = require('discord-hybrid-sharding');
const Discord = require('discord.js');
const client = new Discord.Client({
// @ts-ignore | For Typescript use Cluster.Client.getInfo() instead of Cluster.data
shards: Cluster.data.SHARD_LIST, // An array of shards that will get spawned
shardCount: Cluster.data.TOTAL_SHARDS, // Total number of shards
});
client.cluster = new Cluster.Client(client); // initialize the Client, so we access the .broadcastEval()
client.login('YOUR_TOKEN');
Following examples assume that your Discord.Client
is called client
.
client.cluster
.broadcastEval(`this.guilds.cache.size`)
.then(results => console.log(`${results.reduce((prev, val) => prev + val, 0)} total guilds`));
// or with a callback function
client.cluster
.broadcastEval(c => c.guilds.cache.size)
.then(results => console.log(`${results.reduce((prev, val) => prev + val, 0)} total guilds`));
Option | Type | Default | Description |
---|---|---|---|
totalShards | number or string | “auto” | Amount of internal shards which will be spawned |
totalClusters | number or string | “auto” | Amount of processes/clusters which will be spawned |
shardsPerClusters | number or string | - | Amount of shards which will be in one process/cluster |
shardList | Array[number] | - | OPTIONAL - On cross-hosting or spawning specific shards you can provide a shardList of internal Shard IDs, which will get spawned |
mode | “worker” or “process” | “worker” | Cluster.Manager mode for the processes |
token | string | - | OPTIONAL -Bot token is only required totalShards are set to “auto” |
The Manager.spawn options are the same as for Sharding Manager
Event | Description |
---|---|
clusterCreate | Triggered when a Cluster gets spawned |
All properties like for .broadcastEval()
are available, just replace the client.shard
with client.cluster
Other properties:
Property | Description |
---|---|
client.cluster.count | Returns the amount of all clusters |
client.cluster.id | Returns the current cluster ID |
client.cluster.ids | Returns all internal shards of the cluster |
Feel free to contribute/suggest or contact me on my Discord server or in DM: Meister#9667
Options are now labelled as cluster
instead of shard
:
- client.shard...
+ client.cluster...
- .broadcastEval((c, context) => c.guilds.cache.get(context.guildId), { context: { guildId: '1234' }, shard: 0 })
+ .broadcastEval((c, context) => c.guilds.cache.get(context.guildId), { context: { guildId: '1234' }, cluster: 0 })
Small changes in naming conventions:
- client.shard.respawnAll({ shardDelay = 5000, respawnDelay = 500, timeout = 30000 })
+ client.cluster.respawnAll({ clusterDelay = 5000, respawnDelay = 5500, timeout = 30000 })
- manager.shard.respawnAll({ shardDelay = 5000, respawnDelay = 500, timeout = 30000 })
+ manager.respawnAll({ clusterDelay = 5000, respawnDelay = 5500, timeout = 30000 })
Get current cluster ID:
- client.shard.id
+ client.cluster.id
Get current shard ID:
- client.shard.id
+ message.guild.shardId
Get total shards count:
- client.shard.count
+ client.cluster.info.TOTAL_SHARDS
Get all ShardID’s in the current cluster:
- client.shard.id
+ [...client.cluster.ids.keys()]
Zero Downtime Reclustering
:Zero Downtime Reclustering is a Plugin, which is used to reshard/recluster or even restart your bot with having a theoretical outage of some seconds.
There are two options for the restartMode
:
gracefulSwitch
: Spawns all new Clusters with the provided Info in maintenance mode, once all clusters have been spawned and the DiscordClient is ready, the clusters will exit maintenance mode, where as it will fire the client.cluster.on('ready')
event. In order to load the Database and listen to events. Moreover all Clusters will be gracefully killed, once all clusters exited maintenance mode.rolling
: Spawns the Clusters with the provided Info in maintenance mode, once the DiscordClient is ready of the Cluster, the Cluster will exit maintenance mode, where as it will fire the client.cluster.on('ready')
event. In order to load the Database and listen to events. Moreover the OldCluster will be killed, since the Cluster has exited maintenance mode. Not recommended, when shardData has not been updated.Cluster.js
const manager = new Cluster.Manager(`${__dirname}/bot.js`, {...});
manager.extend(
new Cluster.ReClusterManager()
)
... ///SOME CODE
// Start reclustering
const optional = {totalShards, totalClusters....}
manager.recluster.start({restartMode: 'gracefulSwitch', ...optional})
Bot.js
const client = new Discord.Client({});
client.cluster = new Cluster.Client(client);
if (client.cluster.maintenance) console.log(`Bot on maintenance mode with ${client.cluster.maintenance}`);
client.cluster.on('ready', () => {
// Load Events
// Handle Database stuff, to not process outdated data
});
client.login(token);
HeartbeatSystem
const manager = new Cluster.Manager(`${__dirname}/bot.js`, {...});
manager.extend(
new Cluster.HeartbeatManager({
interval: 2000, // Interval to send a heartbeat
maxMissedHeartbeats: 5, // Maximum amount of missed Heartbeats until Cluster will get respawned
})
)
Control Restarts
const manager = new Cluster.Manager(`${__dirname}/bot.js`, {
...YourOptions,
restarts: {
max: 5, // Maximum amount of restarts per cluster
interval: 60000 * 60, // Interval to reset restarts
},
});
IPC System
ClusterManager | cluster.js
const Cluster = require('discord-hybrid-sharding');
const manager = new Cluster.Manager(`${__dirname}/testbot.js`, {
totalShards: 1,
totalClusters: 1,
});
manager.on('clusterCreate', cluster => {
cluster.on('message', message => {
console.log(message);
if (!message._sRequest) return; // Check if the message needs a reply
message.reply({ content: 'hello world' });
});
setInterval(() => {
cluster.send({ content: 'I am alive' }); // Send a message to the client
cluster.request({ content: 'Are you alive?', alive: true }).then(e => console.log(e)); // Send a message to the client
}, 5000);
});
manager.spawn({ timeout: -1 });
ClusterClient | client.js
const Cluster = require('discord-hybrid-sharding');
const Discord = require('discord.js');
const client = new Discord.Client({
shards: Cluster.data.SHARD_LIST, // An array of shards that will get spawned
shardCount: Cluster.data.TOTAL_SHARDS, // Total number of shards
});
client.cluster = new Cluster.Client(client);
client.cluster.on('message', message => {
console.log(message);
if(!message._sRequest) return; // Check if the message needs a reply
if(message.alive) message.reply({ content: 'Yes I am!' }):
});
setInterval(() => {
client.cluster.send({ content: 'I am alive as well!' });
}, 5000);
client.login('YOUR_TOKEN');
With a complex code-base, you probably need a fine-grained control over the cluster spawn queue in order to respect rate limits.
The queue system can be controlled from the cluster manager.
const manager = new Cluster.Manager(`${__dirname}/bot.js`, {
totalShards: 8,
shardsPerClusters: 2,
queue: {
auto: false,
},
});
The auto
property is set with true
by default, which automatically queues the clusters, when running manager.spawn()
When the auto mode has been disabled, then you have to manually manage the queue.
Cluster.js
manager.spawn();
manager.queue.next();
The manager.queue.next()
function will spawn the next cluster in the queue.
Now you can call the function client.cluster.spawnNextCluster()
from the client to spawn the next cluster.
Property | Description |
---|---|
manager.queue.start() | Starts the queue and resolves, when the queue is empty |
manager.queue.stop() | Stops the queue and blocks all .next requests |
manager.queue.resume() | Resumes the queue and allows .next requests again |
manager.queue.next() | Spawns the next cluster in the queue |
client.cluster.spawnNextCluster() | Triggers the spawn of the next cluster in the queue |
Evals a Script on the ClusterManager:
client.cluster.evalOnManager('process.memoryUsage().rss / 1024 ** 2');
Listen to debug messages and internal stuff:
manager.on('debug', console.log);
Optional Timeout on broadcastEval (Promise will get rejected after given time):
client.cluster.broadcastEval('new Promise((resolve, reject) => {})', { timeout: 10000 });
Open a PR/Issue when you need other Functions 😃
Using the package with other libraries requires some minor changes:
const Cluster = require('discord-hybrid-sharding');
///Create your Discord Client:
/* Use the Data below for telling the Client, which shards to spawn */
const lastShard = Cluster.data.LAST_SHARD_ID;
const firstShard = Cluster.data.FIRST_SHARD_ID;
const totalShards = Cluster.data.TOTAL_SHARDS;
const shardList = Cluster.data.SHARD_LIST;
client.cluster = new Cluster.Client(client);
///When the Client is ready, You can listen to the client's ready event:
// Just add, when the client.on('ready') does not exist
client.cluster.triggerReady();
The upper code is a pseudo code and shows how you can use this package with other libraries
With some minor changes, you can even use this Package for clustering normal processes.
If you encounter any problems feel free to open an issue in our GitHub repository or join the Discord server.
Credits goes to the discord.js library for the Base Code (See changes.md
) and to this helpful server