A Modular Discord Bot Framework based on pycord
Project description
Modular Discord Bot Framework
A template for Discord bots based on pycord with built-in handling of config files (YAML or TOML), easy Docker packaging, and modular components powered by Cogs.
Building a bot with MDBF
MDBF handles most of the setup for you! All you need is to write some cogs and instantiate an MDBFBot like so:
from discord import Intents
from mdbf.bot import MDBFBot
from mdbf.utils import locate_config
from cogs.example_one import ExampleCogOne
from cogs.example_two import ExampleCogTwo
intents = Intents.default()
bot = MDBFBot(
name="MyBot",
intents=intents,
config_path=locate_config(),
cogs=[ExampleCogOne, ExampleCogTwo],
cog_configs={"ExampleCogOne": "ex_one", "ExampleCogTwo": "ex_two"},
chunk_guilds_at_startup=False,
)
bot.serve()
You can pass any other arguments to an MDBFBot that you can to a normal pycord bot.
Cogs
MDBF is built on top of Cogs. A Cog can be thought of as an isolated set of functionality that pertains to a specific job performed by the bot. For example, you might have a ReactionCog that reacts to user messages based on their content, or a ModCog that handles moderation actions. Ideally, only the functionality that a specific Cog actually needs to perform should be included in that Cog. A ModCog shouldn't also be handling reaction roles, for example. Each Cog gets its own section in the config, if you define one, and holds its own state. Cogs can be reloaded individually without interrupting each other. To make a new Cog for your bot, define a subclass of the BaseCog in mdbf.cogs. Here's a simple Cog that reacts to messages containing the bot's name:
import re
import discord
from mdbf.cogs import BaseCog
class SimpleCog(BaseCog):
def update(self, config: dict) -> None:
self.emoji = config["emoji"]
@BaseCog.listener()
async def on_message(self, message: discord.Message) -> None:
if message.author.bot:
return # Ignore messages from bots!
content = message.content.lower() # We don't care about case
if (
message.guild.get_member(self.bot.user.id).nick # If this bot has a nickname...
and message.guild.get_member(self.bot.user.id).nick.lower() in content # And it's in the message...
) or (self.bot.user.name.lower() in content): # Or its global name is in the message...
await message.add_reaction(self.emoji) # React with the configured emoji
The SimpleCog's config section, if named "simple", would look like this:
simple:
emoji: ✨
Which would configure it to react with sparkles to its name being mentioned. More complex behavior can be modelled using all of the tools available to normal pycord Cogs.
Config
MDBF Handles config reloads via an application command: /reload. It can only be run by configured admins, and rather than restarting the whole bot, it triggers a reload of each Cog individually. This is the main advantage of MDBF over just using pycord: Configs are handled fully by MDBF, and exposed as normal Python dictionaries to Cogs. If the config hasn't changed, no reload is performed, and if it has, the Cog can reload itself in real time without interrupting other cogs.
Each Cog needs to implement its own update function, which should re-assign any values read from config, and perform any other config dependent initialization logic, such as compiling regexes, caching images from URLs, connecting to databases, etc. This method is automatically called by MDBF when a config change is detected during a reload, and at Cog initialization.
Configs can be provided at the following paths: config.yaml or config.yml (YAML format) or config.toml (TOML format). Only one config file should be provided! There are also two "config" values that must be passed as environment variables: BOT_TOKEN and BOT_GUILD_ID. The only config option present in MDBF itself is admins, which is a list of user IDs for users who should be considered "admins" of the bot. They will be able to reload the config, and you can use the check_admin function provided by your MDBFBot instance in your Cogs to alter behavior (for example, an if statement in an application command to send an error instead of executing a command when run by a non-admin). Any other configuration is determined by your implementation.
Packaging your bot with Docker
This repo includes a Dockerfile.example file that can be used to build your bot, assuming you use UV for project management (which you should, because it's great!) and put your Cogs in the ./cogs/ directory and your bot initialization in ./main.py, both relative to your project root. If that's the case, you can just build the image with your favorite CICD tool and run the image wherever you want to host your bot.
Hosting your bot with Docker Compose
An instance of a bot built on MDBF can only exist in a single guild!. This is intentional, to avoid the need to manage state across multiple servers. I'm open to advice on how to implement multi-guild functionality, but it isn't planned since all of my own bots are specific to certain servers. You can also just host multiple instances of the same bot in different guilds, if you want. The following mini-guide explains how to set up an individual instance using Docker Compose.
- Build the docker image. I currently use OneDev for CICD, but you can do it manually, with GitHub Actions, with Drone... In the next steps I'll use
bot:latestto refer to the image, but you should replace that with wherever your image is located. - Make a bot in the Discord Developer Console. Copy its token. I'll use
<token>where you should paste it. - Copy the guild ID of the server your bot will run in. Again, a single instance currently cannot function in multiple servers! Paste it where you see
<guild_id>. - Make a
docker-compose.ymllike this:
services:
bot:
image: bot:latest
restart: unless-stopped
environment:
BOT_TOKEN: <token>
BOT_GUILD_ID: <guild_id>
volumes:
- ./config/:/app/config/
- Make a folder named
configand put a new file in it calledconfig.yml,config.yamlorconfig.toml. If you bind the file directly, hot reloads (via/reload) will not work, since docker will not update the contents of the file when they change on the host. Add any config needed for your bot's Cogs. - Add the bot to your server via the Discord Developer Console.
- Launch the stack with
docker compose up -d. If you want to tail the logs, usedocker compose logs -f. Don't launch the bot in production without-d! If you do, it will go down if you close your terminal, control-c the process, etc! - Assuming you've done everything right up to this point, your bot is now ready to use! The sky is the limit.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file mdbf-0.3.0.tar.gz.
File metadata
- Download URL: mdbf-0.3.0.tar.gz
- Upload date:
- Size: 21.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
154473cc904585260deda4b4efecd0af1b4fd610438a8ab832811f686442a350
|
|
| MD5 |
b979d84a0d36f6736e728be20772e49f
|
|
| BLAKE2b-256 |
b3052a74eb602ed5a28d12bbe370e7fa035d1737b9f3bbf1154470a29c3d7954
|
File details
Details for the file mdbf-0.3.0-py3-none-any.whl.
File metadata
- Download URL: mdbf-0.3.0-py3-none-any.whl
- Upload date:
- Size: 20.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8e5e4d1b196efd4fff6a996ab32f4b59948d5186947e95446884bd390ad99b6e
|
|
| MD5 |
83c3cbaf45281cb5c715019f775b4eb6
|
|
| BLAKE2b-256 |
62b5683847c80ca97073b047c6970fef2918382179f6c6d2d80d427c37069b46
|