How to code a Discord bot with Node.JS
Hey there, friends! I'm David, and today, we'll be building our very own Discord bot using Node.js. So, buckle up and let's get started!
Prerequisites and Setup
Before we begin, make sure you have Node.js installed on your computer. You can download it from nodejs.org. You'll also need a code editor, like Visual Studio Code or Sublime Text.
Once you have Node.js installed, we'll create a new folder for our project, navigate to it in the terminal, and run npm init
to create a new package.json file. This file will store our project's dependencies and configurations.
Creating a Discord Application
Now, we'll head to the Discord Developer Portal at https://discord.com/developers/applications
and create a new application by clicking the 'New Application' button. Give your bot a name and click on 'Create'.
Next, navigate to the 'Bot' tab on the left sidebar, and click 'Add Bot'. Confirm the action, and now you have a new bot!
Copy the bot's token secret (or refresh it if you don’t see it). The token is sensitive. It is what allows your code to act as the bot.
To protect this information, we will store it in a .env
file. Let’s create it at the root of our project, and inside write :
BOT_TOKEN=<the value of the token>
While we are at it, we also need to add the APP_ID
and SERVER_ID
values to the file. The APP_ID
is easy to find on the General Information
page of your app in the Discord Developer Portal. Let’s add it to the .env
file:
APP_ID=<your app id>
The SERVER_ID
is a bit more difficult to find.
First, you need to activate Developper mode
in Discord. For that, you need to head to the Advanced
tab in the user settings in the Discord client and change the setting called “Developer Mode”.
Then, still in the Discord client, go to the Discord server, right-click on the server name and select Copy ID
right at the bottom. Let’s add it to the .env
file:
SERVER_ID=<your server id>
Now an important step. Create a .gitignore
file, and add.env
file to it. This excludes the file from being tracked by git.
.env
This prevents our secret from being pushed to wherever our code is stored, if we want to track our changes with git
.
Adding the bot to a server
Before we go any further, we also need to invite the bot to the server. To do this, the Discord App section has a useful invitation link builder within the oAuth2
section.
You must select bot
and application.commands
and the permissions you want to give, such as “Send Messages” or “Embed Links”. The exact permissions depend on what exactly you want to do. Don’t worry about it; you can always add more later.
Installing Dependencies and Setting Up the Project
Now, in your project folder, run the following command in your terminal to install the Discord.js
library:
npm install discord.js
We also want to install the dotenv
library to ensure the environment variables stored in the .env
file are read and added to the code.
Let’s create a new file called index.js
in your project folder. In this file, we'll write the code to set up and run our Discord bot. First, let's import the necessary libraries and create a new Discord client. We’ll import Client
, Events
and GatewayIntentBits
from the discord.js
library.
// Require the necessary discord.js classes
const { Client, Events, GatewayIntentBits } = require('discord.js');
We also want to initialise our environment variables using dotenv
, and for that, we add:
require('dotenv').config();
Now we want to initialise the client instance. Here we call the Client constructor and specify the value of the intents
field in the parameter object. The intents are basically the rights you want your bot to have. For now, let’s specify that we want the Guilds
object from the GatewayIntentBit
. A Guild, in internal discord parlance, means a server.
// Create a new client instance
const client = new Client({ intents:[
GatewayIntentBits.Guilds
]});
Then we create a callback for the discord client to call when the login is done, where we basically print out the user the client is logged in as:
client.once(Events.ClientReady, c => {
console.log(`Ready! Logged in as ${c.user.tag}`);
});
Finally, we log in using the BOT_TOKEN
we’ve defined in our .env
file.
client.login('YOUR_BOT_TOKEN');
If we run the bot using node JS, the bot now logs in and tells us what it is logged in as :
node index.js
Now, let's create a first command for our bot to react to:
Creating our first command
First, we’ll create a commands
folder. This will store our different commands.
We’ll use the SlashCommandBuilder
utility from the discord.js
library. Let’s start by importing it.
const { SlashCommandBuilder } = require('discord.js');
First, we need to define the command options. We do this by creating a data object that is an instance of SlashCommandBuilder
.
const data = new SlashCommandBuilder()
Now we need to modify it to suit our needs. We’re creating a simple ping
command for now, so we add :
.setName('ping')
.setDescription('Replies with Pong!')
This is also where we define if there are other additional parameters, for example, by using setStringOption.
Now we need to define the code that the bot will be running when it gets the command, and to that end, we define an execute
asynchronous function that takes an interaction
object as a parameter:
const execute = async (interaction) => {
}
This object has a reply
function, which we will be using to answer “Pong” to the user.
const execute = async (interaction) => {
await interaction.reply(`Pong!`);
}
Now we need to package this all up into a module that exports the data
configuration and the execute
callback :
module.exports = {
data,
execute
};
Our ping
command is now ready to be added to the bot.
Adding the command to the bot
We need to tell the bot to execute the code we’ve just created. For that, let’s head to index.js and create a commands
object that imports the code we’ve just written:
const commands = {
ping: require('./commands/ping');
}
Now let’s create an event listener to react to Discord events :
const listener = async (interaction) => {
}
Inside this function, we first want to filter out any interaction that is not a command by checking the interaction is a “Chat Input Command” :
if (!interaction.isChatInputCommand()) return;
Now we know the interaction is a command, we can log its name using the commandName
property on the interaction:
console.log(`Received command : ${interaction.commandName}`);
Next, let’s retrieve the code that corresponds to the command :
const command = commands[interaction.commandName];
If the code does not exist, we leave:
if (!command) {
return;
}
And now, we execute the code :
await command.execute(interaction);
Ideally, if we were building a full application, we would wrap this in a try/catch.
Our function now looks something like this :
const listener = async (interaction) => {
if (!interaction.isChatInputCommand()) return;
console.log(`Received command : ${interaction.commandName}`)
const command = commands[interaction.commandName]
if (!command) {
return;
}
await command.execute(interaction);
};
Now let’s register the listener by adding:
client.on(Events.InteractionCreate, listener);
Start your bot by running node index.js in your terminal. Once the bot is online, type !ping
in your Discord server, and the bot should reply with Pong!
.
Congratulations! You've just built your very own Discord bot using Node.js. You can now extend its capabilities by adding more commands and functionality to suit your needs. Thanks for reading, and have fun coding!