Browse Source
Hybrid commands allow a regular command to also double as a slash command, assuming it meets the subset required to function.pull/7881/head
10 changed files with 919 additions and 28 deletions
@ -0,0 +1,458 @@ |
|||
""" |
|||
The MIT License (MIT) |
|||
|
|||
Copyright (c) 2015-present Rapptz |
|||
|
|||
Permission is hereby granted, free of charge, to any person obtaining a |
|||
copy of this software and associated documentation files (the "Software"), |
|||
to deal in the Software without restriction, including without limitation |
|||
the rights to use, copy, modify, merge, publish, distribute, sublicense, |
|||
and/or sell copies of the Software, and to permit persons to whom the |
|||
Software is furnished to do so, subject to the following conditions: |
|||
|
|||
The above copyright notice and this permission notice shall be included in |
|||
all copies or substantial portions of the Software. |
|||
|
|||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS |
|||
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
|||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
|||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
|||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
|||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
|||
DEALINGS IN THE SOFTWARE. |
|||
""" |
|||
|
|||
from __future__ import annotations |
|||
|
|||
from typing import ( |
|||
TYPE_CHECKING, |
|||
Any, |
|||
Callable, |
|||
ClassVar, |
|||
Dict, |
|||
List, |
|||
Type, |
|||
TypeVar, |
|||
Union, |
|||
Optional, |
|||
) |
|||
|
|||
import discord |
|||
import inspect |
|||
from discord import app_commands |
|||
from discord.utils import MISSING, maybe_coroutine, async_all |
|||
from .core import Command, Group |
|||
from .errors import CommandRegistrationError, CommandError, HybridCommandError, ConversionError |
|||
from .converter import Converter |
|||
from .cog import Cog |
|||
|
|||
if TYPE_CHECKING: |
|||
from typing_extensions import Self, ParamSpec, Concatenate |
|||
from ._types import ContextT, Coro, BotT |
|||
from .bot import Bot |
|||
from .context import Context |
|||
from .parameters import Parameter |
|||
from discord.app_commands.commands import Check as AppCommandCheck |
|||
|
|||
|
|||
__all__ = ( |
|||
'HybridCommand', |
|||
'HybridGroup', |
|||
'hybrid_command', |
|||
'hybrid_group', |
|||
) |
|||
|
|||
T = TypeVar('T') |
|||
CogT = TypeVar('CogT', bound='Cog') |
|||
CommandT = TypeVar('CommandT', bound='Command') |
|||
# CHT = TypeVar('CHT', bound='Check') |
|||
GroupT = TypeVar('GroupT', bound='Group') |
|||
|
|||
if TYPE_CHECKING: |
|||
P = ParamSpec('P') |
|||
P2 = ParamSpec('P2') |
|||
|
|||
CommandCallback = Union[ |
|||
Callable[Concatenate[CogT, ContextT, P], Coro[T]], |
|||
Callable[Concatenate[ContextT, P], Coro[T]], |
|||
] |
|||
else: |
|||
P = TypeVar('P') |
|||
P2 = TypeVar('P2') |
|||
|
|||
|
|||
def is_converter(converter: Any) -> bool: |
|||
return (inspect.isclass(converter) and issubclass(converter, Converter)) or isinstance(converter, Converter) |
|||
|
|||
|
|||
def make_converter_transformer(converter: Any) -> Type[app_commands.Transformer]: |
|||
async def transform(cls, interaction: discord.Interaction, value: str) -> Any: |
|||
try: |
|||
if inspect.isclass(converter) and issubclass(converter, Converter): |
|||
if inspect.ismethod(converter.convert): |
|||
return await converter.convert(interaction._baton, value) |
|||
else: |
|||
return await converter().convert(interaction._baton, value) # type: ignore |
|||
elif isinstance(converter, Converter): |
|||
return await converter.convert(interaction._baton, value) # type: ignore |
|||
except CommandError: |
|||
raise |
|||
except Exception as exc: |
|||
raise ConversionError(converter, exc) from exc # type: ignore |
|||
|
|||
return type('ConverterTransformer', (app_commands.Transformer,), {'transform': classmethod(transform)}) |
|||
|
|||
|
|||
def replace_parameters(parameters: Dict[str, Parameter], signature: inspect.Signature) -> List[inspect.Parameter]: |
|||
# Need to convert commands.Parameter back to inspect.Parameter so this will be a bit ugly |
|||
params = signature.parameters.copy() |
|||
for name, parameter in parameters.items(): |
|||
if is_converter(parameter.converter) and not hasattr(parameter.converter, '__discord_app_commands_transformer__'): |
|||
params[name] = params[name].replace(annotation=make_converter_transformer(parameter.converter)) |
|||
|
|||
return list(params.values()) |
|||
|
|||
|
|||
class HybridAppCommand(discord.app_commands.Command[CogT, P, T]): |
|||
def __init__(self, wrapped: HybridCommand[CogT, Any, T]) -> None: |
|||
signature = inspect.signature(wrapped.callback) |
|||
params = replace_parameters(wrapped.params, signature) |
|||
wrapped.callback.__signature__ = signature.replace(parameters=params) |
|||
|
|||
try: |
|||
super().__init__( |
|||
name=wrapped.name, |
|||
callback=wrapped.callback, # type: ignore # Signature doesn't match but we're overriding the invoke |
|||
description=wrapped.description or wrapped.short_doc or '…', |
|||
) |
|||
finally: |
|||
del wrapped.callback.__signature__ |
|||
|
|||
self.wrapped: HybridCommand[CogT, Any, T] = wrapped |
|||
self.binding = wrapped.cog |
|||
|
|||
def _copy_with(self, **kwargs) -> Self: |
|||
copy: Self = super()._copy_with(**kwargs) # type: ignore |
|||
copy.wrapped = self.wrapped |
|||
return copy |
|||
|
|||
async def _check_can_run(self, interaction: discord.Interaction) -> bool: |
|||
# Hybrid checks must run like so: |
|||
# - Bot global check once |
|||
# - Bot global check |
|||
# - Parent interaction check |
|||
# - Cog/group interaction check |
|||
# - Cog check |
|||
# - Local interaction checks |
|||
# - Local command checks |
|||
|
|||
bot: Bot = interaction.client # type: ignore |
|||
ctx: Context[Bot] = interaction._baton |
|||
|
|||
if not await bot.can_run(ctx, call_once=True): |
|||
return False |
|||
|
|||
if not await bot.can_run(ctx): |
|||
return False |
|||
|
|||
if self.parent is not None and self.parent is not self.binding: |
|||
# For commands with a parent which isn't the binding, i.e. |
|||
# <binding> |
|||
# <parent> |
|||
# <command> |
|||
# The parent check needs to be called first |
|||
if not await maybe_coroutine(self.parent.interaction_check, interaction): |
|||
return False |
|||
|
|||
if self.binding is not None: |
|||
try: |
|||
# Type checker does not like runtime attribute retrieval |
|||
check: AppCommandCheck = self.binding.interaction_check # type: ignore |
|||
except AttributeError: |
|||
pass |
|||
else: |
|||
ret = await maybe_coroutine(check, interaction) |
|||
if not ret: |
|||
return False |
|||
|
|||
local_check = Cog._get_overridden_method(self.binding.cog_check) |
|||
if local_check is not None: |
|||
ret = await maybe_coroutine(local_check, ctx) |
|||
if not ret: |
|||
return False |
|||
|
|||
if self.checks and not await async_all(f(interaction) for f in self.checks): # type: ignore |
|||
return False |
|||
|
|||
if self.wrapped.checks and not await async_all(f(ctx) for f in self.wrapped.checks): # type: ignore |
|||
return False |
|||
|
|||
return True |
|||
|
|||
async def _invoke_with_namespace(self, interaction: discord.Interaction, namespace: app_commands.Namespace) -> Any: |
|||
# Wrap the interaction into a Context |
|||
bot: Bot = interaction.client # type: ignore |
|||
|
|||
# Unfortunately, `get_context` has to be called for this to work. |
|||
# If someone doesn't inherit this to replace it with their custom class |
|||
# then this doesn't work. |
|||
interaction._baton = ctx = await bot.get_context(interaction) |
|||
|
|||
exc: CommandError |
|||
try: |
|||
await self.wrapped.prepare(ctx) |
|||
# This lies and just always passes a Context instead of an Interaction. |
|||
return await self._do_call(ctx, ctx.kwargs) # type: ignore |
|||
except app_commands.CommandSignatureMismatch: |
|||
raise |
|||
except app_commands.TransformerError as e: |
|||
if isinstance(e.__cause__, CommandError): |
|||
exc = e.__cause__ |
|||
else: |
|||
exc = HybridCommandError(e) |
|||
exc.__cause__ = e |
|||
except app_commands.AppCommandError as e: |
|||
exc = HybridCommandError(e) |
|||
exc.__cause__ = e |
|||
except CommandError as e: |
|||
exc = e |
|||
|
|||
await self.wrapped.dispatch_error(ctx, exc) |
|||
|
|||
|
|||
class HybridCommand(Command[CogT, P, T]): |
|||
r"""A class that is both an application command and a regular text command. |
|||
|
|||
This has the same parameters and attributes as a regular :class:`~discord.ext.commands.Command`. |
|||
However, it also doubles as an :class:`application command <discord.app_commands.Command>`. In order |
|||
for this to work, the callbacks must have the same subset that is supported by application |
|||
commands. |
|||
|
|||
These are not created manually, instead they are created via the |
|||
decorator or functional interface. |
|||
|
|||
.. versionadded:: 2.0 |
|||
""" |
|||
|
|||
__commands_is_hybrid__: ClassVar[bool] = True |
|||
|
|||
def __init__( |
|||
self, |
|||
func: CommandCallback[CogT, ContextT, P, T], |
|||
/, |
|||
**kwargs, |
|||
) -> None: |
|||
super().__init__(func, **kwargs) |
|||
self.app_command: HybridAppCommand[CogT, Any, T] = HybridAppCommand(self) |
|||
|
|||
@property |
|||
def cog(self) -> CogT: |
|||
return self._cog |
|||
|
|||
@cog.setter |
|||
def cog(self, value: CogT) -> None: |
|||
self._cog = value |
|||
self.app_command.binding = value |
|||
|
|||
async def can_run(self, ctx: Context[BotT], /) -> bool: |
|||
if ctx.interaction is None: |
|||
return await super().can_run(ctx) |
|||
else: |
|||
return await self.app_command._check_can_run(ctx.interaction) |
|||
|
|||
async def _parse_arguments(self, ctx: Context[BotT]) -> None: |
|||
interaction = ctx.interaction |
|||
if interaction is None: |
|||
return await super()._parse_arguments(ctx) |
|||
else: |
|||
ctx.kwargs = await self.app_command._transform_arguments(interaction, interaction.namespace) |
|||
|
|||
|
|||
class HybridGroup(Group[CogT, P, T]): |
|||
r"""A class that is both an application command group and a regular text group. |
|||
|
|||
This has the same parameters and attributes as a regular :class:`~discord.ext.commands.Group`. |
|||
However, it also doubles as an :class:`application command group <discord.app_commands.Group>`. |
|||
Note that application commands groups cannot have callbacks associated with them, so the callback |
|||
is only called if it's not invoked as an application command. |
|||
|
|||
These are not created manually, instead they are created via the |
|||
decorator or functional interface. |
|||
|
|||
.. versionadded:: 2.0 |
|||
""" |
|||
|
|||
__commands_is_hybrid__: ClassVar[bool] = True |
|||
|
|||
def __init__(self, *args: Any, **attrs: Any) -> None: |
|||
super().__init__(*args, **attrs) |
|||
parent = None |
|||
if self.parent is not None: |
|||
if isinstance(self.parent, HybridGroup): |
|||
parent = self.parent.app_command |
|||
else: |
|||
raise TypeError(f'HybridGroup parent must be HybridGroup not {self.parent.__class__}') |
|||
|
|||
guild_ids = attrs.pop('guild_ids', None) or getattr(self.callback, '__discord_app_commands_default_guilds__', None) |
|||
self.app_command: app_commands.Group = app_commands.Group( |
|||
name=self.name, |
|||
description=self.description or self.short_doc or '…', |
|||
guild_ids=guild_ids, |
|||
) |
|||
|
|||
# This prevents the group from re-adding the command at __init__ |
|||
self.app_command.parent = parent |
|||
|
|||
def add_command(self, command: Union[HybridGroup[CogT, ..., Any], HybridCommand[CogT, ..., Any]], /) -> None: |
|||
"""Adds a :class:`.HybridCommand` into the internal list of commands. |
|||
|
|||
This is usually not called, instead the :meth:`~.GroupMixin.command` or |
|||
:meth:`~.GroupMixin.group` shortcut decorators are used instead. |
|||
|
|||
Parameters |
|||
----------- |
|||
command: :class:`HybridCommand` |
|||
The command to add. |
|||
|
|||
Raises |
|||
------- |
|||
CommandRegistrationError |
|||
If the command or its alias is already registered by different command. |
|||
TypeError |
|||
If the command passed is not a subclass of :class:`.HybridCommand`. |
|||
""" |
|||
|
|||
if not isinstance(command, (HybridCommand, HybridGroup)): |
|||
raise TypeError('The command passed must be a subclass of HybridCommand or HybridGroup') |
|||
|
|||
if isinstance(command, HybridGroup) and self.parent is not None: |
|||
raise ValueError(f'{command.qualified_name!r} is too nested, groups can only be nested at most one level') |
|||
|
|||
self.app_command.add_command(command.app_command) |
|||
command.parent = self |
|||
|
|||
if command.name in self.all_commands: |
|||
raise CommandRegistrationError(command.name) |
|||
|
|||
self.all_commands[command.name] = command |
|||
for alias in command.aliases: |
|||
if alias in self.all_commands: |
|||
self.remove_command(command.name) |
|||
raise CommandRegistrationError(alias, alias_conflict=True) |
|||
self.all_commands[alias] = command |
|||
|
|||
def remove_command(self, name: str, /) -> Optional[Command[CogT, ..., Any]]: |
|||
cmd = super().remove_command(name) |
|||
self.app_command.remove_command(name) |
|||
return cmd |
|||
|
|||
def command( |
|||
self, |
|||
name: str = MISSING, |
|||
*args: Any, |
|||
**kwargs: Any, |
|||
) -> Callable[[CommandCallback[CogT, ContextT, P2, T]], HybridCommand[CogT, P2, T]]: |
|||
"""A shortcut decorator that invokes :func:`~discord.ext.commands.hybrid_command` and adds it to |
|||
the internal command list via :meth:`add_command`. |
|||
|
|||
Returns |
|||
-------- |
|||
Callable[..., :class:`Command`] |
|||
A decorator that converts the provided method into a Command, adds it to the bot, then returns it. |
|||
""" |
|||
|
|||
def decorator(func: CommandCallback[CogT, ContextT, P2, T]): |
|||
kwargs.setdefault('parent', self) |
|||
result = hybrid_command(name=name, *args, **kwargs)(func) |
|||
self.add_command(result) |
|||
return result |
|||
|
|||
return decorator |
|||
|
|||
def group( |
|||
self, |
|||
name: str = MISSING, |
|||
*args: Any, |
|||
**kwargs: Any, |
|||
) -> Callable[[CommandCallback[CogT, ContextT, P2, T]], HybridGroup[CogT, P2, T]]: |
|||
"""A shortcut decorator that invokes :func:`.group` and adds it to |
|||
the internal command list via :meth:`~.GroupMixin.add_command`. |
|||
|
|||
Returns |
|||
-------- |
|||
Callable[..., :class:`Group`] |
|||
A decorator that converts the provided method into a Group, adds it to the bot, then returns it. |
|||
""" |
|||
|
|||
def decorator(func: CommandCallback[CogT, ContextT, P2, T]): |
|||
kwargs.setdefault('parent', self) |
|||
result = hybrid_group(name=name, *args, **kwargs)(func) |
|||
self.add_command(result) |
|||
return result |
|||
|
|||
return decorator |
|||
|
|||
|
|||
def hybrid_command( |
|||
name: str = MISSING, |
|||
**attrs: Any, |
|||
) -> Callable[[CommandCallback[CogT, ContextT, P, T]], HybridCommand[CogT, P, T]]: |
|||
"""A decorator that transforms a function into a :class:`.HybridCommand`. |
|||
|
|||
A hybrid command is one that functions both as a regular :class:`.Command` |
|||
and one that is also a :class:`app_commands.Command <discord.app_commands.Command>`. |
|||
|
|||
The callback being attached to the command must be representable as an |
|||
application command callback. Converters are silently converted into a |
|||
:class:`~discord.app_commands.Transformer` with a |
|||
:attr:`discord.AppCommandOptionType.string` type. |
|||
|
|||
Checks and error handlers are dispatched and called as-if they were commands |
|||
similar to :class:`.Command`. This means that they take :class:`Context` as |
|||
a parameter rather than :class:`discord.Interaction`. |
|||
|
|||
All checks added using the :func:`.check` & co. decorators are added into |
|||
the function. There is no way to supply your own checks through this |
|||
decorator. |
|||
|
|||
.. versionadded:: 2.0 |
|||
|
|||
Parameters |
|||
----------- |
|||
name: :class:`str` |
|||
The name to create the command with. By default this uses the |
|||
function name unchanged. |
|||
attrs |
|||
Keyword arguments to pass into the construction of the |
|||
hybrid command. |
|||
|
|||
Raises |
|||
------- |
|||
TypeError |
|||
If the function is not a coroutine or is already a command. |
|||
""" |
|||
|
|||
def decorator(func: CommandCallback[CogT, ContextT, P, T]): |
|||
if isinstance(func, Command): |
|||
raise TypeError('Callback is already a command.') |
|||
return HybridCommand(func, name=name, **attrs) |
|||
|
|||
return decorator |
|||
|
|||
|
|||
def hybrid_group( |
|||
name: str = MISSING, |
|||
**attrs: Any, |
|||
) -> Callable[[CommandCallback[CogT, ContextT, P, T]], HybridGroup[CogT, P, T]]: |
|||
"""A decorator that transforms a function into a :class:`.HybridGroup`. |
|||
|
|||
This is similar to the :func:`~discord.ext.commands.group` decorator except it creates |
|||
a hybrid group instead. |
|||
""" |
|||
|
|||
def decorator(func: CommandCallback[CogT, ContextT, P, T]): |
|||
if isinstance(func, Command): |
|||
raise TypeError('Callback is already a command.') |
|||
return HybridGroup(func, name=name, **attrs) |
|||
|
|||
return decorator # type: ignore |
Loading…
Reference in new issue