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