An opinionated Discord bot framework inspired by Django and built on top of Hikari, Tanjun, Ormar, and Alembic.
Atsume is in alpha and breaking changes should be expected. If you have any feedback or advice, feel free to find me in the Hikari Discord.
- Automatic project scaffolding/file-based organization
- Configuration instead of boilerplate
- Functionality split into modular, independent components
- Automatically restart the bot on changes during development
- Configure which components run in which servers
- Database ORM with Ormar
- Automatic database migrations with Alembic
Create a new project directory and install hikari-atsume
(make sure you are using Python 3.10+. If you are using Poetry,
your Python dependency should be python = "^3.10,<3.12"
)
# Install with your preferred database backend, SQLite recommended for beginners
pip install hikari-atsume[sqlite]
pip install hikari-atsume[mysql]
pip install hikari-atsume[postgresql]
Now to start your new project, run
atsume startproject my_bot
which should generate some files for your project. In my_bot/local.py
, add your
Discord bot token in. If you want to use message commands, make sure to set your
MESSAGE_PREFIX
and add hikari.Intents.MESSAGE_CONTENT
to your INTENTS
.
# my_bot/settings.py
INTENTS = hikari.Intents.ALL_UNPRIVILEGED | hikari.Intents.MESSAGE_CONTENT
In your project directory run
python manage.py startapp basic
which should generate a new directory called basic
.
In basic/commands.py
, write your commands using Tanjun.
Commands are declared without explicitly attaching to a Component object
(Atsume takes care of that using load_from_scope
).
This is an example of a hybrid slash/message command that takes one optional positional argument, the user to say hi to.
# basic/commands.py
@tanjun.annotations.with_annotated_args(follow_wrapped=True)
@tanjun.as_message_command("hi", "hello", "hey", "howdy")
@tanjun.as_slash_command("hi", "The bot says hi.")
async def hello(
ctx: atsume.Context,
member: Annotated[Optional[Member], "The user to say hi to.", Positional()] = None,
) -> None:
member = member if member else ctx.member
if member:
await ctx.respond(f"Hi {member.display_name}!")
Now with our new component ready, register it in the bot's settings.
# my_bot/settings.py
COMPONENTS = [
"basic"
]
Let's say we want to track how many times each member of a guild has said
hi to the bot. In our models.py
file, let's create a new database model
to keep track of this. Atsume uses Ormar
for its database models and queries.
# basic/models.py
from atsume.db import Model
import ormar
class HiCounter(Model):
user: int = ormar.BigInteger(primary_key=True) # Discord User IDs need to be stored as big integers
count: int = ormar.Integer(default=0)
Now in our commands.py
, let's increment a user's count every time they say hi.
# basic/models.py
# Skipping the decorators
async def hello(
ctx: atsume.Context,
member: Annotated[Optional[Member], "The user to say hi to.", Positional()] = None,
) -> None:
member = member if member else ctx.member
if member:
count_model, _ = await HiCounter.objects.get_or_create(user=member.user.id, _defaults={"count": 0})
count_model.count = count_model.count + 1
await count_model.upsert()
await ctx.respond(f"Hi {member.display_name}! (You've said hi {count_model.count} time[s]!)")
Before we run this though, we need to add our HiCounter model to the database! To do this, we can generate a database migration that will migrate the database from it's current state (nothing) to the state we want (a database with a table for HiCounter). Atsume can generate migrations automatically in most cases so we'll use it's tool for that here.
Run this command to generate migrations.
python manage.py makemigrations
You should see something like
Migrating ComponentConfig(name"basic")...
Create model hicounter...
Once that's done, your migration is ready to go! Run it on the database with
python manage.py upgrade
Atsume's migrations are still very much a work in progress (you can track it here). As a general rule of thumb, Atsume is good at migrations that create and delete models and fields, but currently struggles with renames. You can always review a migration and make any needed changes by looking in the component's generated migration folder. Atsume uses Alembic for migrations, so you can look at the Alembic operations docs to figure out how to write migrations manually.
Finally, with the bot configured and our component's commands and models ready , it's time to run it!
python manage.py run
- The Hikari Discord for help and feedback
- FasterSpeeding for Tanjun
- Lunarmagpie for help with the CI and linting
- The Django and django-stubs projects for their amazing work and some code that I borrowed.