Source code for discord_http.commands

import inspect
import itertools
import logging
import re

from typing import get_args as get_type_args
from typing import (
    Callable, TYPE_CHECKING, Union, Type, Protocol,
    Generic, TypeVar, Optional, Coroutine, Literal, Any,
    runtime_checkable
)

from . import utils
from .channel import (
    TextChannel, VoiceChannel,
    CategoryChannel, NewsThread,
    PublicThread, PrivateThread, StageChannel,
    DirectoryChannel, ForumChannel, StoreChannel,
    NewsChannel, BaseChannel, Thread
)
from .cooldowns import BucketType, Cooldown, CooldownCache
from .enums import ApplicationCommandType, CommandOptionType, ChannelType
from .errors import (
    UserMissingPermissions, BotMissingPermissions, CheckFailed,
    InvalidMember, CommandOnCooldown
)
from .flags import Permissions
from .member import Member
from .message import Attachment
from .object import PartialBase, Snowflake
from .response import BaseResponse, AutocompleteResponse
from .role import Role
from .user import User

if TYPE_CHECKING:
    from .client import Client
    from .context import Context

ChoiceT = TypeVar("ChoiceT", str, int, float)
ConverterT = TypeVar("ConverterT", covariant=True)

LocaleTypes = Literal[
    "id", "da", "de", "en-GB", "en-US", "es-ES", "fr",
    "hr", "it", "lt", "hu", "nl", "no", "pl", "pt-BR",
    "ro", "fi", "sv-SE", "vi", "tr", "cs", "el", "bg",
    "ru", "uk", "hi", "th", "zh-CN", "ja", "zh-TW", "ko"
]
ValidLocalesList = get_type_args(LocaleTypes)

channel_types = {
    BaseChannel: [g for g in ChannelType],
    TextChannel: [ChannelType.guild_text],
    VoiceChannel: [ChannelType.guild_voice],
    CategoryChannel: [ChannelType.guild_category],
    NewsChannel: [ChannelType.guild_news],
    StoreChannel: [ChannelType.guild_store],
    NewsThread: [ChannelType.guild_news_thread],
    PublicThread: [ChannelType.guild_public_thread],
    PrivateThread: [ChannelType.guild_private_thread],
    StageChannel: [ChannelType.guild_stage_voice],
    DirectoryChannel: [ChannelType.guild_directory],
    ForumChannel: [ChannelType.guild_forum],
    Thread: [
        ChannelType.guild_news_thread,
        ChannelType.guild_public_thread,
        ChannelType.guild_private_thread
    ]
}

_log = logging.getLogger(__name__)
_NoneType = type(None)
_type_table: dict[type, CommandOptionType] = {
    str: CommandOptionType.string,
    int: CommandOptionType.integer,
    float: CommandOptionType.number
}

__all__ = (
    "Choice",
    "Cog",
    "Command",
    "Converter",
    "Interaction",
    "Listener",
    "PartialCommand",
    "Range",
    "SubGroup",
)


[docs] class Cog: _cog_commands = dict() _cog_interactions = dict() _cog_listeners = dict() def __new__(cls, *args, **kwargs): commands = {} listeners = {} interactions = {} for base in reversed(cls.__mro__): for _, value in base.__dict__.items(): match value: case x if isinstance(x, SubCommand): continue # Do not overwrite commands just in case case x if isinstance(x, Command): commands[value.name] = value case x if isinstance(x, SubGroup): commands[value.name] = value case x if isinstance(x, Interaction): interactions[value.custom_id] = value case x if isinstance(x, Listener): listeners[value.name] = value cls._cog_commands: dict[str, "Command"] = commands cls._cog_interactions: dict[str, "Interaction"] = interactions cls._cog_listeners: dict[str, "Listener"] = listeners return super().__new__(cls) async def _inject(self, bot: "Client"): await self.cog_load() module_name = self.__class__.__module__ if module_name not in bot._cogs: bot._cogs[module_name] = [] bot._cogs[module_name].append(self) for cmd in self._cog_commands.values(): cmd.cog = self bot.add_command(cmd) if isinstance(cmd, SubGroup): for subcmd in cmd.subcommands.values(): subcmd.cog = self for listener in self._cog_listeners.values(): listener.cog = self bot.add_listener(listener) for interaction in self._cog_interactions.values(): interaction.cog = self bot.add_interaction(interaction) async def _eject(self, bot: "Client"): await self.cog_unload() module_name = self.__class__.__module__ if module_name in bot._cogs: bot._cogs[module_name].remove(self) for cmd in self._cog_commands.values(): bot.remove_command(cmd) for listener in self._cog_listeners.values(): bot.remove_listener(listener) for interaction in self._cog_interactions.values(): bot.remove_interaction(interaction)
[docs] async def cog_load(self) -> None: """ Called before the cog is loaded """ pass
[docs] async def cog_unload(self) -> None: """ Called before the cog is unloaded """ pass
[docs] class PartialCommand(PartialBase): def __init__(self, data: dict): super().__init__(id=int(data["id"])) self.name: str = data["name"] self.guild_id: Optional[int] = utils.get_int(data, "guild_id") def __str__(self) -> str: return self.name def __repr__(self): return f"<PartialCommand id={self.id} name={self.name}>"
class LocaleContainer: def __init__( self, key: str, name: str, description: Optional[str] = None ): self.key = key self.name = name self.description = description or "..."
[docs] @runtime_checkable class Converter(Protocol[ConverterT]): """ This is the base class of converting strings to whatever you desire. Instead of needing to implement checks inside the command, you can use this to convert the value on runtime, both in sync and async mode. """
[docs] async def convert(self, ctx: "Context", value: str) -> ConverterT: """ The function where you implement the logic of converting the value into whatever you need to be outputted in command. Parameters ---------- ctx: `Context` Context of the bot value: `str` The value returned by the argument in command Returns ------- `ConverterT` Your converted value """ raise NotImplementedError("convert not implemented")
[docs] class Command: def __init__( self, command: Callable, name: str, description: Optional[str] = None, guild_ids: Optional[list[Union[Snowflake, int]]] = None, guild_install: bool = True, user_install: bool = False, type: ApplicationCommandType = ApplicationCommandType.chat_input, parent: Optional["SubGroup"] = None ): self.id: Optional[int] = None self.command = command self.cog: Optional["Cog"] = None self.type: int = int(type) self.name = name self.description = description self.options = [] self.parent = parent self.guild_install = guild_install self.user_install = user_install self.list_autocompletes: dict[str, Callable] = {} self.guild_ids: list[Union[Snowflake, int]] = guild_ids or [] self._converters: dict[str, Type[Converter]] = {} self.__list_choices: list[str] = [] self.__user_objects: dict[str, Type[Member | User]] = {} if self.type == ApplicationCommandType.chat_input: if self.description is None: self.description = command.__doc__ or "No description provided." if self.name != self.name.lower(): raise ValueError("Command names must be lowercase.") if not 1 <= len(self.description) <= 100: raise ValueError("Command descriptions must be between 1 and 100 characters.") else: self.description = None if ( self.type is ApplicationCommandType.chat_input.value and not self.options ): sig = inspect.signature(self.command) self.options = [] slicer = 1 if sig.parameters.get("self", None): slicer = 2 for parameter in itertools.islice(sig.parameters.values(), slicer, None): origin = getattr( parameter.annotation, "__origin__", parameter.annotation ) option: dict[str, Any] = {} _channel_options: list[ChannelType] = [] # Check if there are multiple types, looking for: # - Union[Any, ...] / Optional[Any] / type | None # - type | type | ... if ( getattr(parameter.annotation, "__args__", None) or origin in [Union] ): if ( len(parameter.annotation.__args__) >= 2 and parameter.annotation.__args__[-1] is _NoneType ): # First one is the type we're looking for # Others except last usually is for typing purposes origin = parameter.annotation.__args__[0] # Recreate GenericAlias if it's something like Choice[str] if getattr(origin, "__origin__", None): parameter.annotation.__args__ = origin.__args__ origin = origin.__origin__ # If you're using Union[TextChannel, VoiceChannel, ...] # And also check if all the types are valid channel types elif all([ g in channel_types for g in parameter.annotation.__args__ ]): # And make sure origin triggers channel types origin = parameter.annotation.__args__[0] for i in parameter.annotation.__args__: _channel_options.extend(channel_types[i]) if origin is User or origin is Member: ptype = CommandOptionType.user self.__user_objects[parameter.name] = origin elif origin in channel_types: ptype = CommandOptionType.channel if _channel_options: # Union[] was used for channels option.update({ "channel_types": [int(i) for i in _channel_options] }) else: # Just a regular channel type option.update({ "channel_types": [ int(i) for i in channel_types[origin] ] }) elif origin in [Attachment]: ptype = CommandOptionType.attachment elif origin in [Role]: ptype = CommandOptionType.role elif isinstance(origin, Choice): self.__list_choices.append(parameter.name) ptype = origin.type # If literal, replicate Choice elif origin is Literal: # self.__list_choices.append(parameter.name) ptype = CommandOptionType.string if not getattr(self.command, "__choices_params__", {}): self.command.__choices_params__ = {} self.command.__choices_params__[parameter.name] = { str(g): str(g) for g in parameter.annotation.__args__ } # PyRight may not recognize 'Range' due to dynamic typing. # Assuming 'origin' is a Range object. elif isinstance(origin, Range): # type: ignore[arg-type] ptype = origin.type # type: ignore[arg-type] if origin.type == CommandOptionType.string: # type: ignore[arg-type] option.update({ "min_length": origin.min, # type: ignore[arg-type] "max_length": origin.max # type: ignore[arg-type] }) else: option.update({ "min_value": origin.min, # type: ignore[arg-type] "max_value": origin.max # type: ignore[arg-type] }) elif origin == int: ptype = CommandOptionType.integer elif origin == bool: ptype = CommandOptionType.boolean elif origin == float: ptype = CommandOptionType.number elif origin == str: ptype = CommandOptionType.string elif isinstance(origin, Converter): self._converters[parameter.name] = origin # type: ignore ptype = CommandOptionType.string else: ptype = CommandOptionType.string option.update({ "name": parameter.name, "description": "…", "type": ptype.value, "required": (parameter.default == parameter.empty), "autocomplete": False, "name_localizations": {}, "description_localizations": {}, }) self.options.append(option) def __repr__(self) -> str: return f"<Command name='{self.name}'>" @property def mention(self) -> str: """ `str`: Returns a mentionable string for the command """ if self.id: return f"</{self.name}:{self.id}>" return f"`/{self.name}`" @property def cooldown(self) -> Optional[CooldownCache]: """ `Optional[CooldownCache]`: Returns the cooldown rule of the command if available """ return getattr(self.command, "__cooldown__", None)
[docs] def mention_sub(self, suffix: str) -> str: """ Returns a mentionable string for a subcommand. Parameters ---------- suffix: `str` The subcommand name. Returns ------- `str` The mentionable string. """ if self.id: return f"</{self.name} {suffix}:{self.id}>" return f"`/{self.name} {suffix}`"
async def _make_context_and_run( self, context: "Context" ) -> BaseResponse: args, kwargs = await context._create_args() for name, values in getattr(self.command, "__choices_params__", {}).items(): if name not in kwargs: continue if name not in self.__list_choices: continue kwargs[name] = Choice( kwargs[name], values[kwargs[name]] ) for name, value in self.__user_objects.items(): if name not in kwargs: continue if ( isinstance(kwargs[name], Member) and value is User ): # Force User if command is expecting a User, but got a Member kwargs[name] = kwargs[name]._user if not isinstance(kwargs[name], value): raise InvalidMember( f"User given by the command `(parameter: {name})` " "is not a member of a guild." ) result = await self.run(context, *args, **kwargs) if not isinstance(result, BaseResponse): raise TypeError( f"Command {self.name} must return a " f"Response object, not {type(result)}." ) return result def _has_permissions(self, ctx: "Context") -> Permissions: _perms: Optional[Permissions] = getattr( self.command, "__has_permissions__", None ) if _perms is None: return Permissions(0) _resolved_perms: Permissions | None = getattr( ctx.user, "resolved_permissions", None ) if _resolved_perms is None: return Permissions(0) if Permissions.administrator in _resolved_perms: return Permissions(0) missing = Permissions(sum([ flag.value for flag in _perms if flag not in _resolved_perms ])) return missing def _bot_has_permissions(self, ctx: "Context") -> Permissions: _perms: Optional[Permissions] = getattr( self.command, "__bot_has_permissions__", None ) if _perms is None: return Permissions(0) if Permissions.administrator in ctx.app_permissions: return Permissions(0) missing = Permissions(sum([ flag.value for flag in _perms if flag not in ctx.app_permissions ])) return missing async def _command_checks(self, ctx: "Context") -> bool: _checks: list[Callable] = getattr( self.command, "__checks__", [] ) for g in _checks: if inspect.iscoroutinefunction(g): result = await g(ctx) else: result = g(ctx) if result is not True: raise CheckFailed(f"Check {g.__name__} failed.") return True def _cooldown_checker(self, ctx: "Context") -> None: if self.cooldown is None: return None current = ctx.created_at.timestamp() bucket = self.cooldown.get_bucket(ctx, current) retry_after = bucket.update_rate_limit(current) if not retry_after: return None # Not rate limited, good to go raise CommandOnCooldown(bucket, retry_after)
[docs] async def run( self, context: "Context", *args, **kwargs ) -> BaseResponse: """ Runs the command. Parameters ---------- context: `Context` The context of the command. Returns ------- `BaseResponse` The return type of the command, used by backend.py (Quart) Raises ------ `UserMissingPermissions` User that ran the command is missing permissions. `BotMissingPermissions` Bot is missing permissions. """ # Check custom checks await self._command_checks(context) # Check user permissions perms_user = self._has_permissions(context) if perms_user != Permissions(0): raise UserMissingPermissions(perms_user) # Check bot permissions perms_bot = self._bot_has_permissions(context) if perms_bot != Permissions(0): raise BotMissingPermissions(perms_bot) # Check cooldown self._cooldown_checker(context) if self.cog is not None: return await self.command(self.cog, context, *args, **kwargs) else: return await self.command(context, *args, **kwargs)
[docs] async def run_autocomplete( self, context: "Context", name: str, current: str ) -> dict: """ Runs the autocomplete Parameters ---------- context: `Context` Context object for the command name: `str` Name of the option current: `str` Current value of the option Returns ------- `dict` The return type of the command, used by backend.py (Quart) Raises ------ `TypeError` Autocomplete must return an AutocompleteResponse object """ if self.cog is not None: result = await self.list_autocompletes[name](self.cog, context, current) else: result = await self.list_autocompletes[name](context, current) if isinstance(result, AutocompleteResponse): return result.to_dict() raise TypeError("Autocomplete must return an AutocompleteResponse object.")
def _find_option(self, name: str) -> Optional[dict]: return next((g for g in self.options if g["name"] == name), None)
[docs] def to_dict(self) -> dict: """ Converts the Discord command to a dict. Returns ------- `dict` The dict of the command. """ _extra_locale = getattr(self.command, "__locales__", {}) _extra_params = getattr(self.command, "__describe_params__", {}) _extra_choices = getattr(self.command, "__choices_params__", {}) _default_permissions: Optional[Permissions] = getattr( self.command, "__default_permissions__", None ) _integration_types = [] if self.guild_install: _integration_types.append(0) if self.user_install: _integration_types.append(1) _integration_contexts = getattr(self.command, "__integration_contexts__", [0, 1, 2]) # Types _extra_locale: dict[LocaleTypes, list[LocaleContainer]] data = { "type": self.type, "name": self.name, "description": self.description, "options": self.options, "nsfw": getattr(self.command, "__nsfw__", False), "name_localizations": {}, "description_localizations": {}, "contexts": _integration_contexts } if _integration_types: data["integration_types"] = _integration_types for key, value in _extra_locale.items(): for loc in value: if loc.key == "_": data["name_localizations"][key] = loc.name data["description_localizations"][key] = loc.description continue opt = self._find_option(loc.key) if not opt: _log.warning( f"{self.name} -> {loc.key}: " "Option not found in command, skipping..." ) continue opt["name_localizations"][key] = loc.name opt["description_localizations"][key] = loc.description if _default_permissions: data["default_member_permissions"] = str(_default_permissions.value) for key, value in _extra_params.items(): opt = self._find_option(key) if not opt: continue opt["description"] = value for key, value in _extra_choices.items(): opt = self._find_option(key) if not opt: continue opt["choices"] = [ {"name": v, "value": k} for k, v in value.items() ] return data
[docs] def autocomplete(self, name: str): """ Decorator to set an option as an autocomplete. The function must at the end, return a `Response.send_autocomplete()` object. Example usage .. code-block:: python @commands.command() async def ping(ctx, options: str): await ctx.send(f"You chose {options}") @ping.autocomplete("options") async def search_autocomplete(ctx, current: str): return ctx.response.send_autocomplete({ "key": "Value shown to user", "feeling_lucky_tm": "I'm feeling lucky!" }) Parameters ---------- name: `str` Name of the option to set as an autocomplete. """ def wrapper(func): find_option = next(( option for option in self.options if option["name"] == name ), None) if not find_option: raise ValueError(f"Option {name} in command {self.name} not found.") find_option["autocomplete"] = True self.list_autocompletes[name] = func return func return wrapper
class SubCommand(Command): def __init__( self, func: Callable, *, name: str, description: Optional[str] = None, guild_install: bool = True, user_install: bool = False, guild_ids: Optional[list[Union[Snowflake, int]]] = None, parent: Optional["SubGroup"] = None ): super().__init__( func, name=name, description=description, guild_install=guild_install, user_install=user_install, guild_ids=guild_ids, parent=parent ) def __repr__(self) -> str: return f"<SubCommand name='{self.name}'>"
[docs] class SubGroup(Command): def __init__( self, *, name: str, description: Optional[str] = None, guild_ids: Optional[list[Union[Snowflake, int]]] = None, guild_install: bool = True, user_install: bool = False, parent: Optional["SubGroup"] = None ): self.name = name self.description = description or "..." # Only used to make Discord happy self.guild_ids: list[Union[Snowflake, int]] = guild_ids or [] self.type = int(ApplicationCommandType.chat_input) self.cog: Optional["Cog"] = None self.subcommands: dict[str, Union[SubCommand, SubGroup]] = {} self.guild_install = guild_install self.user_install = user_install self.parent: Optional["SubGroup"] = parent def __repr__(self) -> str: _subs = [g for g in self.subcommands.values()] return f"<SubGroup name='{self.name}', subcommands={_subs}>"
[docs] def command( self, name: Optional[str] = None, *, description: Optional[str] = None, guild_ids: Optional[list[Union[Snowflake, int]]] = None, guild_install: bool = True, user_install: bool = False, ): """ Decorator to add a subcommand to a subcommand group Parameters ---------- name: `Optional[str]` Name of the command (defaults to the function name) description: `Optional[str]` Description of the command (defaults to the function docstring) guild_ids: `Optional[list[Union[Snowflake, int]]]` List of guild IDs to register the command in user_install: `bool` Whether the command can be installed by users or not guild_install: `bool` Whether the command can be installed by guilds or not """ def decorator(func): subcommand = SubCommand( func, name=name or func.__name__, description=description, guild_ids=guild_ids, guild_install=guild_install, user_install=user_install, parent=self ) self.subcommands[subcommand.name] = subcommand return subcommand return decorator
[docs] def group( self, name: Optional[str] = None, *, description: Optional[str] = None ): """ Decorator to add a subcommand group to a subcommand group Parameters ---------- name: `Optional[str]` Name of the subcommand group (defaults to the function name) """ def decorator(func): subgroup = SubGroup( name=name or func.__name__, description=description, parent=self ) self.subcommands[subgroup.name] = subgroup return subgroup return decorator
[docs] def add_group(self, name: str) -> "SubGroup": """ Adds a subcommand group to a subcommand group Parameters ---------- name: `str` Name of the subcommand group Returns ------- `SubGroup` The subcommand group """ subgroup = SubGroup(name=name) self.subcommands[subgroup.name] = subgroup return subgroup
@property def options(self) -> list[dict]: """ `list[dict]`: Returns the options of the subcommand group """ def build_options(subcommands: dict) -> list[dict]: options = [] for cmd in subcommands.values(): data = cmd.to_dict() if isinstance(cmd, SubGroup): data["type"] = int(CommandOptionType.sub_command_group) # Recursively build options for nested subcommand groups data["options"] = build_options(cmd.subcommands) else: data["type"] = int(CommandOptionType.sub_command) options.append(data) return options return build_options(self.subcommands)
[docs] class Interaction: def __init__( self, func: Callable, custom_id: str, *, regex: bool = False ): self.func: Callable = func self.custom_id: str = custom_id self.regex: bool = regex self.cog: Optional["Cog"] = None self._pattern: Optional[re.Pattern] = ( re.compile(custom_id) if self.regex else None ) def __repr__(self) -> str: return ( f"<Interaction custom_id='{self.custom_id}' " f"regex={self.regex}>" )
[docs] def match(self, custom_id: str) -> bool: """ Matches the custom ID with the interaction. Will always return False if the interaction is not a regex. Parameters ---------- custom_id: `str` The custom ID to match. Returns ------- `bool` Whether the custom ID matched or not. """ if not self.regex: return False return bool(self._pattern.match(custom_id))
[docs] async def run(self, context: "Context") -> BaseResponse: """ Runs the interaction. Parameters ---------- context: `Context` The context of the interaction. Returns ------- `BaseResponse` The return type of the interaction, used by backend.py (Quart) Raises ------ `TypeError` Interaction must be a Response object """ if self.cog is not None: result = await self.func(self.cog, context) else: result = await self.func(context) if not isinstance(result, BaseResponse): raise TypeError("Interaction must be a Response object") return result
[docs] class Listener: def __init__( self, *, name: str, coro: Callable ): self.name = name self.coro = coro self.cog: Optional["Cog"] = None def __repr__(self) -> str: return f"<Listener name='{self.name}'>"
[docs] async def run(self, *args, **kwargs): """ Runs the listener """ if self.cog is not None: await self.coro(self.cog, *args, **kwargs) else: await self.coro(*args, **kwargs)
[docs] class Choice(Generic[ChoiceT]): """ Makes it possible to access both the name and value of a choice. Defaults to a string type Paramaters ---------- key: `str` The key of the choice from your dict. value: `Union[int, str, float]` The value of your choice (the one that is shown to public) """ def __init__(self, key: ChoiceT, value: ChoiceT): self.key: ChoiceT = key self.value: ChoiceT = value self.type: CommandOptionType = CommandOptionType.string def __str__(self) -> str: return str(self.key) def __class_getitem__(cls, obj): if isinstance(obj, tuple): raise TypeError("Choice can only take one type") match obj: case x if x is str: opt = CommandOptionType.string case x if x is int: opt = CommandOptionType.integer case x if x is float: opt = CommandOptionType.number case _: raise TypeError( "Range type must be str, int, " f"or float, not a {obj}" ) output = cls(obj, obj) output.type = opt return output
# Making it so pyright understands that the range type is a normal type if TYPE_CHECKING: from typing import Annotated as Range else:
[docs] class Range: """ Makes it possible to create a range rule for command arguments When used in a command, it will only return the value if it's within the range. Example usage: .. code-block:: python Range[str, 1, 10] # (min and max length of the string) Range[int, 1, 10] # (min and max value of the integer) Range[float, 1.0, 10.0] # (min and max value of the float) Parameters ---------- opt_type: `CommandOptionType` The type of the range min: `Union[int, float, str]` The minimum value of the range max: `Union[int, float, str]` The maximum value of the range """ def __init__( self, opt_type: CommandOptionType, min: Optional[Union[int, float, str]], max: Optional[Union[int, float, str]] ): self.type = opt_type self.min = min self.max = max def __class_getitem__(cls, obj): if not isinstance(obj, tuple): raise TypeError("Range must be a tuple") if len(obj) == 2: obj = (*obj, None) elif len(obj) != 3: raise TypeError("Range must be a tuple of length 2 or 3") obj_type, min, max = obj if min is None and max is None: raise TypeError("Range must have a minimum or maximum value") if min is not None and max is not None: if type(min) is not type(max): raise TypeError("Range minimum and maximum must be the same type") match obj_type: case x if x is str: opt = CommandOptionType.string case x if x is int: opt = CommandOptionType.integer case x if x is float: opt = CommandOptionType.number case _: raise TypeError( "Range type must be str, int, " f"or float, not a {obj_type}" ) cast = float if obj_type in (str, int): cast = int return cls( opt, cast(min) if min is not None else None, cast(max) if max is not None else None )
[docs] def command( name: Optional[str] = None, *, description: Optional[str] = None, guild_ids: Optional[list[Union[Snowflake, int]]] = None, guild_install: bool = True, user_install: bool = False, ): """ Decorator to register a command. Parameters ---------- name: `Optional[str]` Name of the command (defaults to the function name) description: `Optional[str]` Description of the command (defaults to the function docstring) guild_ids: `Optional[list[Union[Snowflake, int]]]` List of guild IDs to register the command in user_install: `bool` Whether the command can be installed by users or not guild_install: `bool` Whether the command can be installed by guilds or not """ def decorator(func): return Command( func, name=name or func.__name__, description=description, guild_ids=guild_ids, guild_install=guild_install, user_install=user_install ) return decorator
[docs] def user_command( name: Optional[str] = None, *, guild_ids: Optional[list[Union[Snowflake, int]]] = None, guild_install: bool = True, user_install: bool = False, ): """ Decorator to register a user command. Example usage .. code-block:: python @user_command() async def content(ctx, user: Union[Member, User]): await ctx.send(f"Target: {user.name}") Parameters ---------- name: `Optional[str]` Name of the command (defaults to the function name) guild_ids: `Optional[list[Union[Snowflake, int]]]` List of guild IDs to register the command in user_install: `bool` Whether the command can be installed by users or not guild_install: `bool` Whether the command can be installed by guilds or not """ def decorator(func): return Command( func, name=name or func.__name__, type=ApplicationCommandType.user, guild_ids=guild_ids, guild_install=guild_install, user_install=user_install ) return decorator
[docs] def cooldown( rate: int, per: float, *, type: Optional[BucketType] = None ): """ Decorator to set a cooldown for a command. Example usage .. code-block:: python @commands.command() @commands.cooldown(1, 5.0) async def ping(ctx): await ctx.send("Pong!") Parameters ---------- rate: `int` The number of times the command can be used within the cooldown period per: `float` The cooldown period in seconds key: `Optional[BucketType]` The bucket type to use for the cooldown If not set, it will be using default, which is a global cooldown """ if type is None: type = BucketType.default if not isinstance(type, BucketType): raise TypeError("Key must be a BucketType") def decorator(func): func.__cooldown__ = CooldownCache( Cooldown(rate, per), type ) return func return decorator
[docs] def message_command( name: Optional[str] = None, *, guild_ids: Optional[list[Union[Snowflake, int]]] = None, guild_install: bool = True, user_install: bool = False, ): """ Decorator to register a message command. Example usage .. code-block:: python @message_command() async def content(ctx, msg: Message): await ctx.send(f"Content: {msg.content}") Parameters ---------- name: `Optional[str]` Name of the command (defaults to the function name) guild_ids: `Optional[list[Union[Snowflake, int]]]` List of guild IDs to register the command in user_install: `bool` Whether the command can be installed by users or not guild_install: `bool` Whether the command can be installed by guilds or not """ def decorator(func): return Command( func, name=name or func.__name__, type=ApplicationCommandType.message, guild_ids=guild_ids, guild_install=guild_install, user_install=user_install ) return decorator
[docs] def locales( translations: dict[ LocaleTypes, dict[ str, Union[list[str], tuple[str], tuple[str, str]] ] ] ): """ Decorator to set translations for a command. _ = Reserved for the root command name and description. Example usage: .. code-block:: python @commands.command(name="ping") @commands.locales({ # Norwegian "no": { "_": ("ping", "Sender en 'pong' melding") "funny": ("morsomt", "Morsomt svar") } }) async def ping(ctx, funny: str): await ctx.send(f"pong {funny}") Parameters ---------- translations: `Dict[LocaleTypes, Dict[str, Union[tuple[str], tuple[str, str]]]]` The translations for the command name, description, and options. """ def decorator(func): name = func.__name__ container = {} for key, value in translations.items(): temp_value: list[LocaleContainer] = [] if not isinstance(key, str): _log.error(f"{name}: Translation key must be a string, not a {type(key)}") continue if key not in ValidLocalesList: _log.warning(f"{name}: Unsupported locale {key} skipped (might be a typo)") continue if not isinstance(value, dict): _log.error(f"{name} -> {key}: Translation value must be a dict, not a {type(value)}") continue for tname, tvalues in value.items(): if not isinstance(tname, str): _log.error(f"{name} -> {key}: Translation option must be a string, not a {type(tname)}") continue if not isinstance(tvalues, (list, tuple)): _log.error(f"{name} -> {key} -> {tname}: Translation values must be a list or tuple, not a {type(tvalues)}") continue if len(tvalues) < 1: _log.error(f"{name} -> {key} -> {tname}: Translation values must have a minimum of 1 value") continue temp_value.append( LocaleContainer( tname, *tvalues[:2] # Only use the first 2 values, ignore the rest ) ) if not temp_value: _log.warning(f"{name} -> {key}: Found an empty translation dict, skipping...") continue container[key] = temp_value func.__locales__ = container return func return decorator
[docs] def group( name: Optional[str] = None, *, description: Optional[str] = None, guild_ids: Optional[list[Union[Snowflake, int]]] = None, guild_install: bool = True, user_install: bool = False, ): """ Decorator to register a command group. Parameters ---------- name: `Optional[str]` Name of the command group (defaults to the function name) description: `Optional[str]` Description of the command group (defaults to the function docstring) guild_ids: `Optional[list[Union[Snowflake, int]]]` List of guild IDs to register the command group in user_install: `bool` Whether the command group can be installed by users or not guild_install: `bool` Whether the command group can be installed by guilds or not """ def decorator(func): return SubGroup( name=name or func.__name__, description=description, guild_ids=guild_ids, guild_install=guild_install, user_install=user_install ) return decorator
[docs] def describe(**kwargs): """ Decorator to set descriptions for a command. Example usage: .. code-block:: python @commands.command() @commands.describe(user="User to ping") async def ping(ctx, user: Member): await ctx.send(f"Pinged {user.mention}") """ def decorator(func): func.__describe_params__ = kwargs return func return decorator
[docs] def allow_contexts( *, guild: bool = True, bot_dm: bool = True, private_dm: bool = True ): """ Decorator to set the places you are allowed to use the command. Can only be used if the Command has user_install set to True. Parameters ---------- guild: `bool` Weather the command can be used in guilds. bot_dm: `bool` Weather the command can be used in bot DMs. private_dm: `bool` Weather the command can be used in private DMs. """ def decorator(func): func.__integration_contexts__ = [] if guild: func.__integration_contexts__.append(0) if bot_dm: func.__integration_contexts__.append(1) if private_dm: func.__integration_contexts__.append(2) return func return decorator
[docs] def choices( **kwargs: dict[ str | int | float, str | int | float ] ): """ Decorator to set choices for a command. Example usage: .. code-block:: python @commands.command() @commands.choices( options={ "opt1": "Choice 1", "opt2": "Choice 2", ... } ) async def ping(ctx, options: Choice[str]): await ctx.send(f"You chose {choice.value}") """ def decorator(func): for k, v in kwargs.items(): if not isinstance(v, dict): raise TypeError( f"Choice {k} must be a dict, not a {type(v)}" ) func.__choices_params__ = kwargs return func return decorator
[docs] def guild_only(): """ Decorator to set a command as guild only. This is a alias to two particular functions: - `commands.allow_contexts(guild=True, bot_dm=False, private_dm=False)` - `commands.check(...)` (which checks for Context.guild to be available) """ def _guild_only_check(ctx: "Context") -> bool: if not ctx.guild: raise CheckFailed("Command can only be used in servers") return True def decorator(func): _check_list = getattr(func, "__checks__", []) _check_list.append(_guild_only_check) func.__checks__ = _check_list func.__integration_contexts__ = [0] return func return decorator
[docs] def is_nsfw(): """ Decorator to set a command as NSFW. """ def decorator(func): func.__nsfw__ = True return func return decorator
[docs] def default_permissions(*args: Union[Permissions, str]): """ Decorator to set default permissions for a command. """ def decorator(func): if not args: return func if isinstance(args[0], Permissions): func.__default_permissions__ = args[0] else: if any(not isinstance(arg, str) for arg in args): raise TypeError( "All permissions must be strings " "or only 1 Permissions object" ) func.__default_permissions__ = Permissions.from_names( *args # type: ignore ) return func return decorator
[docs] def has_permissions(*args: Union[Permissions, str]): """ Decorator to set permissions for a command. Example usage: .. code-block:: python @commands.command() @commands.has_permissions("manage_messages") async def ban(ctx, user: Member): ... """ def decorator(func): if not args: return func if isinstance(args[0], Permissions): func.__has_permissions__ = args[0] else: if any(not isinstance(arg, str) for arg in args): raise TypeError( "All permissions must be strings " "or only 1 Permissions object" ) func.__has_permissions__ = Permissions.from_names( *args # type: ignore ) return func return decorator
[docs] def bot_has_permissions(*args: Union[Permissions, str]): """ Decorator to set permissions for a command. Example usage: .. code-block:: python @commands.command() @commands.bot_has_permissions("embed_links") async def cat(ctx): ... """ def decorator(func): if not args: return func if isinstance(args[0], Permissions): func.__bot_has_permissions__ = args[0] else: if any(not isinstance(arg, str) for arg in args): raise TypeError( "All permissions must be strings " "or only 1 Permissions object" ) func.__bot_has_permissions__ = Permissions.from_names( *args # type: ignore ) return func return decorator
[docs] def check(predicate: Union[Callable, Coroutine]): """ Decorator to set a check for a command. Example usage: .. code-block:: python def is_owner(ctx): return ctx.author.id == 123456789 @commands.command() @commands.check(is_owner) async def foo(ctx): ... """ def decorator(func): _check_list = getattr(func, "__checks__", []) _check_list.append(predicate) func.__checks__ = _check_list return func return decorator
[docs] def interaction( custom_id: str, *, regex: bool = False ): """ Decorator to register an interaction. This supports the usage of regex to match multiple custom IDs. Parameters ---------- custom_id: `str` The custom ID of the interaction. (can be partial, aka. regex) regex: `bool` Whether the custom_id is a regex or not """ def decorator(func): return Interaction( func, custom_id=custom_id, regex=regex ) return decorator
[docs] def listener(name: Optional[str] = None): """ Decorator to register a listener. Parameters ---------- name: `Optional[str]` Name of the listener (defaults to the function name) Raises ------ `TypeError` - If name was not a string - If the listener was not a coroutine function """ if name is not None and not isinstance(name, str): raise TypeError(f"Listener name must be a string, not {type(name)}") def decorator(func): actual = func if isinstance(actual, staticmethod): actual = actual.__func__ if not inspect.iscoroutinefunction(actual): raise TypeError("Listeners has to be coroutine functions") return Listener( name=name or actual.__name__, coro=func ) return decorator