Skip to content

Latest commit

 

History

History
218 lines (158 loc) · 7.99 KB

README.md

File metadata and controls

218 lines (158 loc) · 7.99 KB

Pidroid

Pidroid is a custom discord bot for TheoTown written in Python using Rapptz's discord.py wrapper.

Production use

To use Pidroid in production, first we need to build a Pidroid docker image with this command:

docker build . --file Dockerfile --build-arg GIT_COMMIT=$(git rev-parse --short HEAD) --tag pidroid

The same command is provided in a build.sh file for convenience. If you do not have Git installed, you can omit the build-arg part. Pidroid uses it for transparency when providing version information.

After building the docker image, we need to make sure we have a environment file for configuration set up. You can read how to do so here.

Pidroid uses a Postgres database to store its information. You can read about setting it up here. By default, running with the provided docker compose will also create and run the postgres database.

After making sure our configuration is complete, we just need to run the bot in a docker container with the following command:

docker-compose up -d

It will automatically start a postgres database and then run the bot.

Development setup

To begin, you'll need to install Python. Pidroid requires Python 3.12 or above to work. You can check what version of Python you have installed by running this command:

python --version

After installing Python, you will also need to install Poetry. Pidroid uses it for dependency management.

After you installed Poetry, now we need to install Pidroid's dependencies:

poetry install

After installing all the required packages, we need to configure the bot itself. Please check out the configuration manual on how to do so. The bot uses a Postgres database. It accepts the login credentials as a DSN string. Please check configuration manual on where to input it.

After setting up the database, you will need to do the database table creation and migrations using alembic:

poetry run migrate

You might have to provide the postgres DSN as an environment variable as stated in the manual.

Lastly, all you have to do is run the bot. You can do so by running this command:

poetry run start -e config.env

The -e argument specifies which file to use for the environment variables.

Database

You will need a PostgreSQL 9.5 database or higher. You will need to type the following in the psql tool:

CREATE ROLE pidroid WITH LOGIN PASSWORD 'your_database_password';
CREATE DATABASE pidroid OWNER pidroid;
CREATE EXTENSION pg_trgm;

After creating your database, you'll need your DSN string.

postgresql+asyncpg://pidroid:your_database_password@127.0.0.1

postgresql+asyncpg is required to specify sqlalchemy to use asyncpg driver. You will only need to change the password field and the IP.

Do note that this is done for you automatically if you're using docker in production mode.

Configuration

Pidroid used to use a config.json file at its ./bot path for its configuration.

Pidroid 5, however, switches to using environment variables as defined in the config.env file in the project root. This is done to be compatible with Docker containers.

# Comma separated list of prefixes Pidroid will use by default
PREFIXES=P,TT
# Discord bot token
TOKEN=

# DEPRECATED: a postgres DSN string, usually used for connecting to services outside
# of a docker container.
# If you are running Pidroid in a docker container, DO NOT SET THIS VARIABLE
POSTGRES_DSN=

# Pidroid user postgres credentials
# This is the default configuration for pidroid database
# on a docker container. If postgres DSN is not set, DSN will be constructed from here.
DB_USER=pidroid
DB_PASSWORD=supersecretpassword
DB_NAME=pidroid
DB_HOST=db

# Optional: TheoTown API key to interact with backend TheoTown API
TT_API_KEY=
# Optional: DeepL API key used for translations in TheoTown guild
DEEPL_API_KEY=
# Optional: Unbelievaboat API key for economy integration in TheoTown guild
UNBELIEVABOAT_API_KEY=
# Optional: Tenor API key for gif command
TENOR_API_KEY=

Please note that if your credentials contain a dollar sign, you need to add another dollar sign to make it a literal.

Useful commands for setup

If you want to access a service on host device through a docker container, you'll need to obtain docker IP.

ip addr show | grep "\binet\b.*\bdocker0\b" | awk '{print $2}' | cut -d '/' -f 1

For development

Converters

The following is an example of how a command may fall back to using the message author for the member argument, if it could not be resolved.

@commands.command()
async def command(self, ctx: Context, member: Member = None):
    member = member or ctx.author
    ...

Linters and static analysis tools might suggest you to update typehinting to something like

@commands.command()
async def command(self, ctx: Context, member: Optional[Member] = None):
    member = member or ctx.author
    ...

However, Optional has a special meaning in discord.py for special discord types such as Member, User, etc. It suppresses conversion errors which is not always preferable and can lead to myriad of unexpected issues.

Henceforth, it is preferable to use Annotated type like so

@commands.command()
async def command(self, ctx: Context, member: Annotated[Optional[Member], Member] = None):
    member = member or ctx.author
    ...

The first argument of Annotated is for the linters, the second argument is used by discord.py

Decorators

discord.py evaluates all decorators in an ascending order.

For clarity sake, all commands that are supposed to be executed inside guilds, require a guild_only check decorator.

The same applies for custom Pidroid decorators that check for specific channel or guild, they require that you also use in-built guild_only check.

Fancy command argument error handling

Sometimes you might want to provide a custom error message for your command. Usually, it's in the case of a user not providing the required argument.

In the old days, you would make that argument optional with the default value as None. And then you would just handle that value with an if statement. However, that can sometimes lead to unexpected behaviour.

Moreover, with the new hybrid commands, making certain arguments optional is bad user experience.

Henceforth, the following is an example of a new way of handling missing argument errors nicely:

    @commands.command()
    async def role_info(self, ctx: Context[Pidroid], role: Role): # Role is a required argument here
        embed = Embed(description=role.mention, timestamp=role.created_at, colour=role.colour)
        if role.icon:
            embed.set_thumbnail(url=role.icon.with_size(4096).url)
        embed.add_field(name="Name", value=role.name)
        embed.add_field(name="ID", value=role.id)
        embed.add_field(name="Position", value=role.position)
        embed.add_field(name="Colour", value=str(role.colour))
        embed.add_field(name="Is mentionable", value=role.mentionable)
        embed.set_footer(text="Role created")
        await ctx.reply(embed=embed)

    # We define a new async method to handle role_info command errors
    @role_info.error
    async def on_role_info_command_error(self, ctx: Context[Pidroid], error: Exception): # These arguments are required
        # We check if our error is of MissingRequiredArgument
        if isinstance(error, MissingRequiredArgument):
            # With that, we check what argument it is for
            if error.param.name == "role":
                # And we return a custom error message
                return await notify(ctx, "Please specify the role to view the information for")

        # This is a special call to notify the generic error handler
        # that the error was not handled and it should be done on
        # its side.
        # This must be at the end of the method.
        setattr(error, 'unhandled', True)